Skip to content

Latest commit

 

History

History
 
 

ModbusTCPMaster

Folders and files

NameName
Last commit message
Last commit date

parent directory

..
 
 
 
 
 
 
 
 
 
 

ModbusTCPMaster

This library allows an imp to communicate with other devices via TCP/IP. It requires the use of Wiznet to transmit the packets between devices via Ethernet.

To use this library, add the following statements to the top of your device code:

#require "ModbusRTU.device.lib.nut:1.0.1"
#require "ModbusMaster.device.lib.nut:1.0.1"
#require "ModbusTCPMaster.device.lib.nut:1.1.0"
#require "W5500.device.nut:1.0.0"

The following instructions are applicable to Electric Imp’s impAccelerator™ Fieldbus Gateway.

  1. Connect the antenna to the Fieldbus Gateway
  2. Wire RS485 A on the Fieldbus Gateway to port A / positive(+) on the other device
  3. Wire RS485 B on the Fieldbus Gateway to port B / negative(-) on the other device
  4. Wire both devices’ ground ports together
  5. Fit jumper J2 on the Fieldbus Gateway motherboard to enable RS485
  6. Power up the Fieldbus Gateway
  7. Configure the Fieldbus Gateway for Internet access using BlinkUp™

ModbusTCPMaster Class Usage

This is the main library class. It implements most of the functions listed in the Modbus specification.

Constructor: ModbusTCPMaster(wiz[, debug])

Instantiate a new ModbusTCPMaster object. It takes one required parameter: wiz, the Wiznet W5500 object that is driving the Ethernet link, and one optional boolean parameter: debug which, if enabled, prints the outgoing and incoming ADU for debugging purposes. The default value of debug is false.

Example

// Configure SPI
local spi = hardware.spi0;
spi.configure(CLOCK_IDLE_LOW | MSB_FIRST | USE_CS_L, 1000);

local wiz = W5500(spi, hardware.pinXC, null, hardware.pinXA);
wiz.configureNetworkSettings("192.168.1.30", "255.255.255.0", "192.168.1.1");

// instantiate a modbus object
modbus <- ModbusTCPMaster(wiz);

ModbusTCPMaster Class Methods

connect(connectionSettings[, onConnectCallback][, onReconnectCallback])

This method configures the network and opens a TCP connection with the device. It will try to reconnect when the connection is not closed intentionally. It takes the following parameters:

Parameter Data Type Required Default Value Description
connectionSettings Table Yes N/A The device IP address and port. The device IP address can either be a string or an array of four bytes, for example: [192, 168, 1, 37] or "192.168.1.37". The port can either be an integer or array of two bytes (the high and low bytes of an unsigned two-byte integer value), for example: [0x10, 0x92] or 4242
onConnectCallback Function No Null The function to be fired when the connection is established. The callback takes error and conn as parameters
onReconnectCallback Function No Null The function to be fired when the connection is re-established. The callback takes error and conn as parameters

Note

  1. If an onReconnectCallback is not supplied, when the connection is re-established, the onConnectCallback will be fired.
  2. Depending on the configuration of the device, the connection will be severed after a certain amount of time and try to re-establish itself. Users of concern are advised to pass in a onReconnectCallback to handle this situation. Please refer to the device spec for more information on how to configure the default idle time.

Example

// The device address and port
local connectionSettings = {
    "destIP"     : "192.168.1.90",
    "destPort"   : 502
};

// Open the connection
modbus.connect(connectionSettings, function(error, conn){
    if (error) {
        server.log(error);
    } else {
        // do something here
    }
});

disconnect([callback])

This method closes the existing TCP connection. It takes one optional parameter: callback, a function that will be executed when the connection has been closed.

Example

modbus.disconnect();

read(targetType, startingAddress, quantity[, callback][, deviceAddress])

Function Code : 01, 02, 03, 04

This is a generic method used to read values from a single coil, register, or multiple coils and registers. It has the following parameters:

Parameter Data Type Required Default Value Description
targetType Constant Yes N/A Refer to ‘Target Type’ table, below
startingAddress Integer Yes N/A The address from which it begins reading values
quantity Integer Yes N/A The number of consecutive addresses the values are read from
callback Function No Null The function to be fired when it receives response regarding this request. It takes two parameters, error and result
deviceAddress Integer No 0x00 The unique address that identifies a device
Target Type Value Access
Coil MODBUSRTU_TARGET_TYPE.COIL Read-Write
Discrete Input MODBUSRTU_TARGET_TYPE.DISCRETE_INPUT Read Only
Input Register MODBUSRTU_TARGET_TYPE.INPUT_REGISTER Read Only
Holding Register MODBUSRTU_TARGET_TYPE.HOLDING_REGISTER Read-Write

Example

// Read from a single coil
modbus.read(MODBUSRTU_TARGET_TYPE.DISCRETE_INPUT, 0x01, 1, function(error, result) {
    if (error) {
        server.error(error);
    } else {
        server.log(result);
    }
}.bindenv(this));

// Read from multiple registers
modbus.read(MODBUSRTU_TARGET_TYPE.INPUT_REGISTER, 0x01 , 5, function(error, results) {
    if (error) {
        server.error(error);
    } else {
        foreach(key, value in results) {
          server.log(key + " : " + value);
        }
    }
}.bindenv(this));

write(targetType, startingAddress, quantity, values[, callback][, deviceAddress])

Function Code : 05, 06, 15, 16

This is a generic method used to write values into coils or registers. It has the following parameters:

Parameter Data Type Required Default Value Description
targetType Constant Yes N/A Refer to the ‘Target Type’ table, above
startingAddress Integer Yes N/A The address from which it begins writing values
quantity Integer Yes N/A The number of consecutive addresses the values are written into
values Integer, array, bool, blob Yes N/A The values written into Coils or Registers. Please view ‘Notes’, below
callback Function No Null The function to be fired when it receives response regarding this request. It takes two parameters, error and result
deviceAddress Integer No 0x00 The unique address that identifies a device

Notes:

  1. Integer, blob and array[integer] are applicable to MODBUSRTU_TARGET_TYPE.HOLDING_REGISTER. Use array[integer] only when quantity is greater than one.

  2. Integer, bool, blob, and array[integer, bool] are applicable to MODBUSRTU_TARGET_TYPE.COIL. Use array[integer, bool] only when quantity is greater than one. The integer value set to coils can be either 0x0000 or 0xFF00. Other values are ignored.

Example

// Write to a single coil
modbus.write(MODBUSRTU_TARGET_TYPE.COIL, 0x01, 1, true, function(error, result) {
    if (error){
        server.error(error);
    } else {
        server.log(result);
    }
}.bindenv(this));

// Write to multiple registers
modbus.write(MODBUSRTU_TARGET_TYPE.HOLDING_REGISTER, 0x01, 5, [false, true, false, true, true], function(error, results) {
    if (error) {
        server.error(error);
    } else {
        foreach(key, value in results) {
            server.log(key + " : " + value);
        }
    }
}.bindenv(this));

readExceptionStatus([callback][, deviceAddress])

Function Code : 07

This method reads the contents of eight Exception Status outputs in a remote device. The optional callback function will, if supplied, be fired when a response regarding this request is received. The callback takes two parameters, error and result.

Example

modbus.readExceptionStatus(function(error, result) {
    if (error) {
        server.error(error);
    } else {
        server.log(result);
    }
}.bindenv(this));

diagnostics(subFunctionCode, data[, callback][, deviceAddress])

Function Code : 08

This method provides a series of tests for checking the communication system between a client (Master) device and a server (Slave), or for checking various internal error conditions within a server. It takes the following parameters:

Parameter Data Type Required Default Value Description
subFunctionCode Constant Yes N/A Refer to the ‘Sub-function Code’ table, below
data Blob Yes N/A The data field required by Modbus request
callback Function No Null The function to be fired when it receives response regarding this request. It takes two parameters, error and result
deviceAddress Integer No 0x00 The unique address that identifies a device
Sub-function Code Value (Hex)
MODBUSRTU_SUB_FUNCTION_CODE.RETURN_QUERY_DATA 0x0000
MODBUSRTU_SUB_FUNCTION_CODE.RESTART_COMMUNICATION_OPTION 0x0001
MODBUSRTU_SUB_FUNCTION_CODE.RETURN_DIAGNOSTICS_REGISTER 0x0002
MODBUSRTU_SUB_FUNCTION_CODE.CHANGE_ASCII_INPUT_DELIMITER 0x0003
MODBUSRTU_SUB_FUNCTION_CODE.FORCE_LISTEN_ONLY_MODE 0x0004
MODBUSRTU_SUB_FUNCTION_CODE.CLEAR_COUNTERS_AND_DIAGNOSTICS_REGISTER 0x000A
MODBUSRTU_SUB_FUNCTION_CODE.RETURN_BUS_MESSAGE_COUNT 0x000B
MODBUSRTU_SUB_FUNCTION_CODE.RETURN_BUS_COMMUNICATION_ERROR_COUNT 0x000C
MODBUSRTU_SUB_FUNCTION_CODE.RETURN_BUS_EXCEPTION_ERROR_COUNT 0x000D
MODBUSRTU_SUB_FUNCTION_CODE.RETURN_SLAVE_MESSAGE_COUNT 0x000E
MODBUSRTU_SUB_FUNCTION_CODE.RETURN_SLAVE_NO_RESPONSE_COUNT 0x000F
MODBUSRTU_SUB_FUNCTION_CODE.RETURN_SLAVE_NAK_COUNT 0x0010
MODBUSRTU_SUB_FUNCTION_CODE.RETURN_SLAVE_BUSY_COUNT 0x0011
MODBUSRTU_SUB_FUNCTION_CODE.RETURN_BUS_CHARACTER_OVERRUN_COUNT 0x0012
MODBUSRTU_SUB_FUNCTION_CODE.CLEAR_OVERRUN_COUNTER_AND_FLAG 0x0014

Example

local data = blob(2);
data.writen(0xFF00, 'w');
data.swap2();

modbus.diagnostics(MODBUSRTU_SUB_FUNCTION_CODE.RESTART_COMMUNICATION_OPTION, data, function(error, result) {
    if (error) {
        server.error(error);
    } else {
        server.log(result);
    }
}.bindenv(this));

reportSlaveID([callback][, deviceAddress])

Function Code : 17

This method reads the description of the type, the current status, and other information specific to a remote device whose address is specified in the method’s last parameter. The optional callback function will, if supplied, be fired when a response regarding this request is received. The callback takes two parameters, error and result.

Example

modbus.reportSlaveID(function(error, result) {
    if (error) {
        server.error(error);
    } else {
        server.log("Run indicator : " + result.runIndicator);
        server.log(result.slaveId);
    }
}.bindenv(this));

maskWriteRegister(referenceAddress, AND_Mask, OR_Mask[, callback][, deviceAddress])

Function Code : 22

This method modifies the contents of a specified holding register using a combination of an AND mask, an OR mask and the register’s current contents. The function can be used to set or clear individual bits in the register. It takes the following parameters:

Parameter Data Type Required Default Value Description
referenceAddress Integer Yes N/A The address of the holding register the value is written into
AND_mask Integer Yes N/A The AND mask
OR_mask Integer Yes N/A The OR mask
callback Function No Null The function to be fired when it receives response regarding this request. It takes two parameters, error and result
deviceAddress Integer No 0x00 The unique address that identifies a device

Example

modbus.maskWriteRegister(0x10, 0xFFFF, 0x0000, function(error, result) {
    if (error) {
        server.error(error);
    } else {
        server.log(result);
    }
}.bindenv(this));

readWriteMultipleRegisters(readingStartAddress, readQuantity, writeStartAddress, writeQuantity, writeValue[, callback][, deviceAddress])

Function Code : 23

This method performs a combination of one read operation and one write operation in a single Modbus transaction. The write operation is performed before the read. It takes the following parameters:

Parameter Data Type Required Default Value Description
readingStartAddress Integer Yes N/A The address from which it begins reading values
readQuantity Integer Yes N/A The number of consecutive addresses values are read from
writeStartAddress Integer Yes N/A The address from which it begins writing values
writeQuantity Integer Yes N/A The number of consecutive addresses values are written into
writeValue Blob Yes N/A The value written into the holding register
callback Function No Null The function to be fired when it receives response regarding this request. It takes two parameters, error and result
deviceAddress Integer No 0x00 The unique address that identifies a device

Note The actual order of operation is determined by the implementation of user's device.

Example

modbus.readWriteMultipleRegisters(0x0A, 2, 0x0A, 2, [28, 88], function(error, result) {
    if (error) {
        server.error(error);
    } else {
        foreach (index, value in result) {
            server.log(format("Index : %d, value : %d", index, value));
        }
    }
});

readDeviceIdentification(readDeviceIdCode, objectId[, callback][, deviceAddress])

Function Code : 43/14

This method lets you read the identification and additional information relative to the physical and functional description of a remote device. It takes the following parameters:

Parameter Data Type Required Default Value Description
readDeviceIdCode Constant Yes N/A Refer to the ‘Read Device ID’ table, below
objectId Constant Yes N/A Refer to the ‘Object ID’ table, below
callback Function No Null The function to be fired when it receives response regarding this request. It takes two parameters, error and result
deviceAddress Integer No 0x00 The unique address that identifies a device
Read Device ID Code Description
MODBUSRTU_READ_DEVICE_CODE.BASIC Get the basic device identification (stream access)
MODBUSRTU_READ_DEVICE_CODE.REGULAR Get the regular device identification (stream access)
MODBUSRTU_READ_DEVICE_CODE.EXTENDED Get the extended device identification (stream access)
MODBUSRTU_READ_DEVICE_CODE.SPECIFIC Get one specific identification object (individual access)
Object ID Category
MODBUSRTU_OBJECT_ID.VENDOR_NAME Basic
MODBUSRTU_OBJECT_ID.PRODUCT_CODE Basic
MODBUSRTU_OBJECT_ID.MAJOR_MINOR_REVISION Basic
MODBUSRTU_OBJECT_ID.VENDOR_URL Regular
MODBUSRTU_OBJECT_ID.PRODUCT_NAME Regular
MODBUSRTU_OBJECT_ID.MODEL_NAME Regular
MODBUSRTU_OBJECT_ID.USER_APPLICATION_NAME Regular

Example

modbus.readDeviceIdentification(MODBUSRTU_READ_DEVICE_CODE.BASIC, MODBUSRTU_OBJECT_ID.VENDOR_NAME, function(error, objects) {
    if (error) {
        server.error("Error: " + error + ", Objects: " + objects);
    } else {
        local info = "DeviceId: ";
        foreach (id, val in objects) {
            info += format("[%d] %s, ", id, val.tostring());
        }
        server.log(info);
    }
}.bindenv(this));

Exception Codes

The table below enumerates all the exception codes that can be possibly encountered. Refer to the Modbus specification for more detailed description on Modbus-specific exceptions.

Value (Dec) Description
1 Illegal Function
2 Illegal Data Address
3 Illegal Data Value
4 Slave Device Fail
5 Acknowledge
6 Slave Device Busy
7 Negative Acknowledge
8 Memory Parity Error
80 Response Timeout
81 Invalid CRC
82 Invalid Argument Length
83 Invalid Device Address
87 Invalid Target Type
88 Invalid Values
89 Invalid Quantity

License

The ModbusTCPMaster library is licensed under the MIT License.