Modbus: Unterschied zwischen den Versionen
| KKeine Bearbeitungszusammenfassung | Ansgru (Diskussion | Beiträge)  K (→Set-Commands) | ||
| Zeile 41: | Zeile 41: | ||
| == Set-Commands == | == Set-Commands == | ||
| this low level device module doesn't provide set commands for itself but implements set   | this low level device module doesn't provide set commands for itself but implements set   | ||
| for logical device modules that make use of this module as a library. See ModbusSET for example. | for logical device modules that make use of this module as a library. See [[ModbusSET]] for example. | ||
| == Get-Commands == | == Get-Commands == | ||
| Zeile 430: | Zeile 430: | ||
| :module for the set silent heat pumps from Schmidt Energie Technik | :module for the set silent heat pumps from Schmidt Energie Technik | ||
| ;[[ModbusAttr]] | ;[[ModbusAttr]] | ||
| :generic modbus device module where the data objects, addresses, display formats, function codes and other things can be configured using FHEM attributes similar to HTTPMOD | :generic modbus device module where the data objects, addresses, display formats, function codes and other things can be configured using FHEM attributes similar to HTTPMOD  [[Kategorie:Interfaces]] | ||
| [[Kategorie:Interfaces]] | |||
Version vom 25. Januar 2022, 09:28 Uhr
| Modbus | |
|---|---|
| Zweck / Funktion | |
| Library or physical device to extract information from devices with a Modbus interface or send information to such devices | |
| Allgemein | |
| Typ | Gerätemodul | 
| Details | |
| Dokumentation | EN / DE | 
| Support (Forum) | Sonstiges | 
| Modulname | 98_Modbus.pm | 
| Ersteller | StefanStrobel (Forum / Wiki) | 
| Wichtig: sofern vorhanden, gilt im Zweifel immer die (englische) Beschreibung in der commandref! | |
Modbus defines a physical modbus interface and library functions to be called from other logical modules / devices. This low level module takes care of the communication with modbus devices and provides Get, Set and cyclic polling of Readings as well as formatting and input validation functions.
The logical device modules for individual machines only need to define the supported modbus function codes and objects of the machine with the modbus interface in data structures. These data structures are then used by this low level module to implement Set, Get and automatic updating of readings in a given interval.
The Modbus module supports Modbus RTU over serial / RS485 lines as well as Modbus TCP and Modbus RTU over TCP. It defines read / write functions for Modbus holding registers, input registers, coils and discrete inputs.
See ModbusAttr if you don't want to use a library to develop your own module and if you are looking for a generic Modbus Module instead that can be configured with attributes.
Availability
The module has been checked in.
Prerequisites
This module requires the Device::SerialPort or Win32::SerialPort module if you want to communicate with modbus devices over a serial line.
Define of a modbus interface device for serial communication
define <name> Modbus <device>
A define of a physical device based on this module is only necessary if a shared physical device like a RS485 USB adapter is used. In the case of Modbus TCP this module will be used as a library for other modules that define all the data objects and no define of the base module is needed.
Example:
define ModBusLine Modbus /dev/ttyUSB1@9600
In this example the module opens the given serial interface and other logical modules like ModbusAttr or ModbusSET can access several Modbus devices connected to this bus concurrently.
Set-Commands
this low level device module doesn't provide set commands for itself but implements set for logical device modules that make use of this module as a library. See ModbusSET for example.
Get-Commands
this low level device module doesn't provide get commands for itself but implements get for logical device modules that make use of this module as a library.
Attributes
- do_not_notify
- readingFnAttributes
- queueDelay
- modify the delay used when sending requests to the device from the internal queue, defaults to 1 second
- queueMax
- max length of the send queue, defaults to 100
- clientSwitchDelay
- defines a delay that is always enforced between the last read from the bus and the next send to the bus
- for all connected devices, but only if the next send goes to a different device than the last one
- dropQueueDoubles
- prevents new request to be queued if the same request is already in the send queue
- skipGarbage
- if set to 1 this attribute will enhance the way the module treats Modbus response frames (RTU over serial lines)
- that look as if they have a wrong Modbus id as their first byte.
- If skipGarbage is set to 1 then the module will skip all bytes until a byte with the expected modbus id is seen.
- Under normal circumstances this behavior should not do any harm and lead to more robustness.
- However since it changes the original behavior of this module it has to be turned on explicitely.
- For Modbus ASCII it skips bytes until the expected starting byte ":" is seen.
- profileInterval
- if set to something non zero it is the time period in seconds for which the module will create bus usage statistics.
- Please note that this number should be at least twice as big as the interval used for requesting values in logical devices
- that use this physical device
- The bus usage statistics create the following readings:
- Profiler_Delay_sum
 - seconds used as delays to implement the defined sendDelay and commDelay
 - Profiler_Fhem_sum
 - seconds spend processing in the module
 - Profiler_Idle_sum
 - idle time
 - Profiler_Read_sum
 - seconds spent reading and validating the data read
 - Profiler_Send_sum
 - seconds spent preparing and sending data
 - Profiler_Wait_sum
 - seconds waiting for a response to a request
 - Statistics_Requests
 - number of requests sent
 - Statistics_Timeouts
 - timeouts encountered
 
Writing modules for devices using this module as a library
Writing a module for a physical device with modbus interface is easy when you use the 98_Modbus.pm module as a library. 
To use this module as a library for other fhem modules you only have to define a data structure that defines the mapping between modbus data objects (holding registers, input registers, coils or discrete inputs) and fhem readings.
Additionally the module needs to contain a few package and use statements and an initialize function at the beginning, that assigns a few special variables to point to functions of the Modbus base module. 
The most easy way to start is to use ModbusAttr to define all objects and data types and then issue a set saveAsModule command which creates a new module automatically.
Example for a module that is called ModbusSET:
package main;
use strict;
use warnings;
sub ModbusSET_Initialize($)
{
    my ($modHash) = @_;
    LoadModule "Modbus";
    require "$attr{global}{modpath}/FHEM/DevIo.pm";
    $modHash->{parseInfo}  = \%SET10parseInfo;              # defines registers, inputs, coils etc. for this Modbus Device
    $modHash->{deviceInfo} = \%SET10deviceInfo;             # defines properties of the device, defaults and supported function codes
    ModbusLD_Initialize($modHash);                          # Generic function of the Modbus module does the rest
    
    $modHash->{AttrList} = $modHash->{AttrList} . " " .     # Standard Attributes like IODEv etc 
        $modHash->{ObjAttrList} . " " .                     # Attributes to add or overwrite parseInfo definitions
        $modHash->{DevAttrList};                            # Attributes to add or overwrite devInfo definitions
}
The name of the initialize-Function has to match the name of the module. In the above example this is ModbusSET_Initialize. Most of the steps needed in an initialize function are provided by the library function ModbusLD_Initialize. This function tells fhem to use the library functions for define, set, get and other typical functions in a module. See DevelopmentModuleIntro for more background information on writing fhem modules if you are curious.
Introduction to the parseInfo structure
Typically the data structure to map between data objects of the modbus device and fhem readings is named parseInfo with a part of the name of the module itself as prefix. In the example of the module 98_ModbusSET.pm which uses Modbus.pm to implement a module for SET Silent 10 heat pumps, the structure is called SET10parseInfo.
This strucure contains keys with values that directly correspond to attributes which can be used with the module ModbusAttr so it is advisable to prototype a new module with ModbusAttr and then translate the attributes to entries in the parseInfo structure. The values in the parseInfo structure can later still be overwritten / extended with the attributes documented in ModbusAttr.
As an example a very simple definition of a parseInfo structure for a heat pump could look like this:
my %XYparseInfo = (
    "h256"  =>  {   reading => "Temp_Wasser_Ein",   # name of the reading for this value
                },
    "h258"  =>  {   reading => "Temp_Wasser_Aus",
                },
    "h770"  =>  {   reading => "Temp_Soll", 
                    min     => 10,                  # input validation for set: min value
                    max     => 32,                  # input validation for set: max value
                    set     => 1,                   # this value can be set
                }
);
the corresponding attributes for ModbusAttr for prototyping or overwriting values would be obj-h256-reading, obj-h258-reading, obj-h770-reading, obj-h770-min and so on.
This parseInfo structure would be the main part of the module and map from holding register 256 to a fhem reading named Temp_Wasser_Ein, holding register 258 to Temp_Wasser_Aus and 770 to Temp_Soll. 
All readings will be read from the device in an interval that the user can specify when he issues the define command for your module.
The meaning of set => 1 is that the holding register 770 can also be written to with a set command. FHEM will check that the value written is not smaller than 10 and not bigger than 32 as specified above.
More complex example:
my %SET10parseInfo = (
    "h256"  =>  {   reading => "Temp_Wasser_Ein",   # name of the reading for this value
                    name    => "Pb1",               # internal name of this register in the hardware doc
                    expr    => '$val / 10',         # conversion of raw value to visible value 
                    len     => 1,
                },
    "h770"  =>  {   reading => "Temp_Soll", 
                    name    => "ST03",
                    expr    => '$val / 10',         # convert raw value to readable temp
                    setexpr => '$val * 10',         # expression to convert a set value to the internal value 
                    min     => 10,                  # input validation for set: min value
                    max     => 32,                  # input validation for set: max value
                    hint    => "8,10,20,25,28,29,30,30.5,31,31.5,32",
                    set     => 1,                   # this value can be set
                },
    "h771"  =>  {   reading => "Hysterese",         # Hex Adr 303
                    name    => "ST04",
                    expr    => '$val / 10',
                    setexpr => '$val * 10',
                    poll    => "x10",               # only poll every 10th iteration.
                    min     => 0.5,
                    max     => 3,
                    set     => 1,
                },
    "h777"  =>  {   reading => "Hyst_Mode",         # Hex Adr 0309
                    name    => "ST10",
                    map     => "0:mittig, 1:oberhalb, 2:unterhalb", 
                    poll    => "once",              # only poll once (or after a set)
                    set     => 1,
                },
    "i800"  =>  {   reading => "Voltage",           # Input register 
                    unpack  => "f>",                # this value is a float
                    len     => 2,                   # the float occupies two input registers, 800 and 801
                },
);
There are many more options that can be specified for each data object / reading. If these options are not specified, the base module assumes defaults that typically make sense. However if you want to modify the defaults, you can either define explicit values in the parseInfo structure or you can define another data structure typically called deviceInfo.
Introduction to the deviceInfo structure
The deviceInfo structure is completely optional. If you don't define it in your module, the base module takes default values that work in ost cases. If you only want to override a few of the defaults, you can just define them and leave other options or sections out. A simple device info structure that modifies some defaults could look like this:
my %SET10deviceInfo = (
    "timing"    => {
            timeout     =>  3,      # timeout is 3 seconds /default would be 2
            commDelay   =>  0.7,    # wait 0.7 seconds before sending after receiving
            sendDelay   =>  0.7,    # wait at least 0.7 seconds for another send
            }, 
    "c"     =>  {               
            read        =>  1,      # function code 1 to read coils (this could be omitted because it is the default anyways
            write       =>  5,      # dito
            },
    "h"     =>  {               
            read        =>  3,      
            write       =>  6,      
            defLen      =>  1,      # default legth is 1 object
            combine     =>  5,      
            defShowGet  =>  1,      
            defUnpack   =>  "s>",   # default data format is a signed 16 bit integer for holding registers 
            },
);
The deviceInfo structure contains five optional parts. Timing defines timing values and the remaining parts define settings or defaults for coils (c), discrete inputs (d), input registers (i) and holding registers (h).
for each modbus object type you can change what function code should be used to read or write to the object. This is completely optional and if nothing is specified, the base module assumes function codes 1,2,3 and 4 for reading as well as 5 and 6 for writing which works for many modbus devices. If you prefer to use function code 16 for writing to holding registers, you can specify "write => 16" in the "h" part.
usage of a module created this way
a logical module written this way will have a define command that can work in two ways. If your module would be called ModbusSET and it is using a serial line connection (Modbus RTU over RS485 oer over RS232):
define <iodevice> Modbus /dev/device@baudrate define <name> ModbusSET <Id> <Interval> </code>
In this case, a physical serial interface device is defined first using the Modbus module. Then a device based on your module (ModbusSET in the example) is defined for each physical modbus device connected to the serial line. For a RS485 bus, several devices with different Ids can be connected to the same bus.
Example:
define ModbusRS485 Modbus /dev/rs485@9600 define PWP ModbusAttr 5 60
this defines the device and it will use the readings that you coded in the parseInfo data structure.
Alternatively your module would also support Modbus TCP or Modbus RTU over TCP with the following define syntax:
define <name> ModbusAttr <Id> <Interval> <Address:Port> <RTU|TCP>
In this case no serial interface device is necessary and your module connects to the modbus device directly via TCP using either Modbus TCP or Modbus RTU over TCP.
Example:
define PWP ModbusAttr 1 30 192.168.1.115:502 TCP
General information about data objects
Modbus devices can use many different ways to encode values in their data objects. A temperature might be stored multiplied with 10 as a 16 bit integer value in one holding register so you have to read the integer and divide it by 10 to get the real temperature value back. It might also be stored as a 32 bit float data type that spans two adjacent input registers. The modbus base module implements a very generic way to handle different encodings without real programming: It lets you define the Perl unpack code to convert a raw data string to a Perl value, a Perl expression to do further computation and a length in data objects.
This way a temperature stored in a 16 bit signed integer as the value multiplied by 10 can be described with the unpack code "s>" and the expression "$val / 10". A float value spanning 2 registers would be described with an unpack code "f>" and a len of 2. No expression is needed in this case. See Perldoc on the pack function for a detailed explation of pack and unpack codes.
The idea here is that you should be able to define any mapping, encoding, transformation or formatting of data objects without programming by simpy describing them.
most important options in parseInfo
Most options here are optional and can be used if there is a need but they can also be omitted. If most readings require the same options and the option is different from the default, it is also possible to define a different default per modbus data object type in another data structure (see deviceInfo). For a list of all options please refer to the attributes documentation of the module ModbusAttr. The attributes there can be translated to parseInfo or deviceInfo keys as shown above.
- reading
- name of the reading to be used in FHEM e.g. Temp_Wasser_ein
- expr
- perl expression to convert a string after it has been read. The original value is in $val e.g. $val / 10
- map
- a map string to convert an value from the device to a more readable output string or to convert a user input to the machine representation e.g. "0:mittig, 1:oberhalb, 2:unterhalb"
- format
- a format string for sprintf to format a value read, e.g. %.1f
- len
- number of Registers this value spans, can be 2 for a 32 bit float which is stored in 2 registers
- unpack
- defines the translation between data in the module and in the communication frame see the documentation of the perl pack function for details. example: "n" for an unsigned 16 bit value or "f>" for a float that is stored in two registers or "s>" for signed 16 bit integer in big endian format
- showget
- can be set to 1 to allow a FHEM get command to read this value from the device. All defined objects can be used in a get command that is issued on the command line. This parameter only controls if fhemweb will offer a get command for the object.
- poll
- defines if this value is included in the read that the module does every defined interval this can be changed by a user with an attribute
- polldelay
- if a value should not be read in each iteration (after interval has passed), this value can be set to an explicit time in seconds. The update function will then verify if this delay has elapsed since the last read of this object. If not, the read is skipped.
- set
- can be set to 1 to allow writing this value with a FHEM set-command
- min
- min value for input validation in a set command. If the user issues e.g. set Device Temp_Soll 10, FHEM will check if the given value 10 is bigger or equal the defined min and smaller or equal the defined max.
- max
- max value for input validation in a set command
- hint
- string for fhemweb to create a selection or slider
- setexpr
- per expression to convert an input string to the machine format before writing this is typically the reverse of the above expr, e.g. $val * 10
- name
- optional internal name of the value in the modbus documentation of the physical device, e.g. pb1
most important options in deviceInfo
Keys in the timing section:
- timeout
- how long to wait for a response from the device, can be overwritten by attribute timeout in logical device. Defaults to two seconds if this is not specified
- commDelay
- minimal delay in secounds between two communications e.g. a read a the next write, can be overwritten with attribute commDelay if added to AttrList in _Initialize below defaults to 0.1 seconds if not specified
- sendDelay
- minimal delay in seconds between two sends, can be overwritten with the attribute sendDelay if added to AttrList in _Initialize function below. Defaults to 0.1 seconds if not specified
Keys per object type (c = coil, d = discrete input, i = input register, h = holding register)
- read
- function code to use for reading this object type (e.g. 3 for holding registers) defaults to function codes 1-4 depending on the object types if nothing else is specified (3 to read holding register, 1 to read coils and so on)
- write
- function code to use for writing this object type (e.g. 6 or 16 for holding registers) defaults to function codes 5 and 6 depending on the object types if nothing else is specified (6 to read holding register, 5 to write coils and so on)
- defLen
- default len for objects using this type (e.g. can be set to 2 if the device mainly provides float values that span 2 registers (2 times 16 Bit) can be overwritten in parseInfo per reading by specifying the key "len" defaults to 1 if not specified
- defFormat
- format string to do sprintf with the value can be overwritten in parseInfo per reading by specifying the key "format" if no format is specified here and none in parseInfo, the the reading is set without further formatting (which is typically fine)
- defUnpack
- default pack / unpack code to convert raw values, e.g. "n" for a 16 bit integer or "f>" for a big endian float can be overwritten in parseInfo per reading by specifying the key "unpack" if not specified here and not in parseInfo, then the raw value is interpreted as "n" which is 16 bit unsigned integer in big endian format
- defPoll
- defines that objects of this type should be polled by default unless specified otherwise in parseInfo or by attributes can be overwritten in parseInfo per reading by specifying the key defaultpoll if not specified here or in parseInfo, the object is not polled
- defShowGet
- defines that FHEMweb shows a Get option (by returning it as reslut to get ?) for objects of this type can be overwritten in parseInfo per reading by specifying the key showget defaults to 0.
- combine
- max number of registers that the device is willing to deliver in one read request. The modbus application layer protocol specification allows for more than 100 but most devices limit this to 5, 10 or some other number. This option defaults to 1 if not specified.
For an example of a full module that is based on the mechanisms described here see 98_ModbusSET.pm.
Attributes of your module
a module based on the base module / library 98_Modbus.pm can also allow the end user to modify properties of each reading if you want to allow it. All you have to do is to offer an attribute by adding its name to the variable $modHash->{AttrList} in your initialize function.
If for example you want to allow the user to modify the maximum value for the reading Temp_Soll, you can add "Temp_Soll-max " to this variable and the user can then set this attribute. The attribute takes precedence over the max potentially already defined in your parseInfo structure.
There are two ways that the base module accepts such readings. One is the reading name followed by "-" and the option to override, the alternative syntax is "obj-" followed by the first letter of an object type (c/d/h/i) and a decimal address just like the main key of an object in the parseInfo structure.
Instead of allowing the attribute Temp_Soll-max for the max value of reading Temp_Soll which corresponds to holding register 770, you can alternatively add the attribute name "obj-h770-min " to $modHash->{AttrList}.
If the user is allowd to specify such attributes solely depends on the contents of the $modHash->{AttrList} variable. All the processing is already built into the base module.
If you want to allow the user the override the formatting of readings then you can add "obj-[cdih][1-9][0-9]*-format " as a regular expression that allows format specifications for all possible data objects.
The module 98_ModbusAttr for example is also based on 98_Modbus.pm and allows all possible attributes so the user can completely define his device with attributes and without a parseInfo or deviceInfo structure.
In the same way you can allow the user to override the device specific options and defaults with attributes that start with "dev-", followed by the section of the deviceInfo and the name of the option. If you want to allow the user to modify the function code to be used for writing holding registers, you can add the attribute "obj-h-write " and the user can then set this attribute to 6 or 16 as he prefers. It is up to the module author to decide if this makes sense.
An assignment that allows most options to the user could be:
    $modHash->{AttrList} = $modHash->{AttrList} . " " .
        "obj-[cdih][1-9][0-9]*-reading " .
        "obj-[cdih][1-9][0-9]*-name " .
        "obj-[cdih][1-9][0-9]*-set " .
        "obj-[cdih][1-9][0-9]*-min " .
        "obj-[cdih][1-9][0-9]*-max " .
        "obj-[cdih][1-9][0-9]*-hint " .
        "obj-[cdih][1-9][0-9]*-expr " .
        "obj-[cdih][1-9][0-9]*-map " .
        "obj-[cdih][1-9][0-9]*-setexpr " .
        "obj-[cdih][1-9][0-9]*-format " .
        "obj-[cdih][1-9][0-9]*-len " .
        "obj-[cdih][1-9][0-9]*-unpack " .
        "obj-[cdih][1-9][0-9]*-showget " .
        
        "obj-[cdih][1-9][0-9]*-poll " .
        "obj-[cdih][1-9][0-9]*-polldelay " .
        "poll-.* " .
        "polldelay-.* " .
        
        "dev-([cdih]-)*read " .
        "dev-([cdih]-)*write " .
        "dev-([cdih]-)*combine " .
        "dev-([cdih]-)*defLen " .
        "dev-([cdih]-)*defFormat " .
        "dev-([cdih]-)*defUnpack " .
        "dev-([cdih]-)*defPoll " .
        "dev-([cdih]-)*defShowGet " .
        "dev-timing-timeout " .
        "dev-timing-sendDelay " .
        "dev-timing-commDelay ";
}
Examples for logical device modules that use this base module
- SDM220M
- SDM630M
- modules for energy meters from B+G E-Tech & EASTON written by Roger
- UMG103
- UMG604
- modules for the UMG103 and UMG604 meters from Janitza
- ModbusSET
- module for the set silent heat pumps from Schmidt Energie Technik
- ModbusAttr
- generic modbus device module where the data objects, addresses, display formats, function codes and other things can be configured using FHEM attributes similar to HTTPMOD