Favicon

NModbus4.Wrapper

Peponi2023.6 - 2023.69m

Open source

1. Instruction

1.1. About NModbus4

NModbus is a C# implementation of the Modbus protocol.
Provides connectivity to Modbus slave compatible devices and applications.
Supports serial ASCII, serial RTU, TCP, and UDP protocols.
NModbus4 it's a fork of NModbus(https://code.google.com/p/nmodbus).
NModbus4 differs from original NModbus in following:
 
1. removed USB support(FtdAdapter.dll)
2. removed log4net dependency
3. removed Unme.Common.dll dependency
4. assembly renamed to NModbus4.dll
5. target framework changed to .NET 4

1.1.1. NModbus4 License

The MIT License (MIT)
 
Copyright (c) 2006 Scott Alexander, 2015 Dmitry Turin
 
Permission is hereby granted, free of charge, to any person obtaining a copy
of this software and associated documentation files (the "Software"), to deal
in the Software without restriction, including without limitation the rights
to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
copies of the Software, and to permit persons to whom the Software is
furnished to do so, subject to the following conditions:
 
The above copyright notice and this permission notice shall be included in
all copies or substantial portions of the Software.
 
THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN
THE SOFTWARE.

1.2. About NModbus4.Wrapper

NModbus4.Wrapper is driven from NModbus4 for integrated usage.
Difference of NModbus4.Wrapper from NModbus4 is: 
 
1. integrate class (RTU, TCP, UDP...) into ModbusService
2. support C# data types (bool, int, float)
3. support threading when using Client

1.2.1. NModbus4.Wrapper License

The MIT License (MIT)
 
Copyright (c) 2023 Peponi
 
Permission is hereby granted, free of charge, to any person obtaining a copy
of this software and associated documentation files (the "Software"), to deal
in the Software without restriction, including without limitation the rights
to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
copies of the Software, and to permit persons to whom the Software is
furnished to do so, subject to the following conditions:
 
The above copyright notice and this permission notice shall be included in
all copies or substantial portions of the Software.
 
THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN
THE SOFTWARE.

1.2.2. NModbus4.Wrapper install

1.2.2.1. Package Manager
NuGet\Install-Package NModbus4.Wrapper

1.3. Code Example

1.3.1. From constructor to connect

string interfaceName = "Test";
ModbusType modbusType = ModbusType.TCP_Slave;
int slaveNo = 1;
EthernetInformation ethernetInformation = new EthernetInformation
{
    Address = "127.0.0.1",
    Port = 50001
};
Endian endian = Endian.Big;
 
ModbusInterface interfaceData = new ModbusInterface(interfaceName, modbusType, slaveNo, ethernetInformation, endian);
ModbusService modbus = new ModbusService(interfaceData, CheckConnectionStatus);
 
modbus.ModbusLog += Modbus_LogWrite;
modbus.ModbusDataReceived += Modbus_ModbusDataReceived;
modbus.ModbusCommunicationException += Modbus_ModbusCommunicationException;
modbus.ModbusGeneralException += Modbus_ModbusGeneralException;
 
modbus.Connect();
 
void CheckConnectionStatus(ModbusInterface modbusInterface, bool connectStatus)
{
    if (connectStatus)
    {
        // Do something...
    }
    else
    {
        // Reconnect, or other things to do..
    }
}

1.3.2. Disconnect

modbus.Dispose();

1.3.3. Restart

if (modbus != null) modbus.Dispose();
 
modbus = new Modbus(interfaceData, CheckConnectionStatus);
 
modbus.ModbusLog += Modbus_LogWrite;
modbus.ModbusDataReceived += Modbus_ModbusDataReceived;
modbus.ModbusCommunicationException += Modbus_ModbusCommunicationException;
modbus.ModbusGeneralException += Modbus_ModbusGeneralException;
 
modbus.Connect();

1.3.4. Read data synchronous

// 1. Single case
 
var commData = new CommunicationData(DataStorage.HoldingRegister, 0, default(double), modbus.Interface.EndianOption);
 
if (modbus.ReadData(ref commData))
{
    // Data process...
}
else
{
    // Do something for error handling...
}
 
// 2. List case
 
List<CommunicationData> commDatas = new List<CommunicationData>();
 
commDatas.Add(new CommunicationData(DataStorage.HoldingRegister, 0, default(float), modbus.Interface.EndianOption));
commDatas.Add(new CommunicationData(DataStorage.HoldingRegister, 2, default(bool), modbus.Interface.EndianOption));
commDatas.Add(new CommunicationData(DataStorage.HoldingRegister, 3, default(int), modbus.Interface.EndianOption));
 
if (modbus.ReadData(ref commDatas))
{
    // Data process...
}
else
{
    // Do something for error handling...
}

1.3.5. Write data asynchronous

// 1. Single case
 
var commData = new CommunicationData(DataStorage.HoldingRegister, 0, 10.12, modbus.Interface.EndianOption);
 
if (!await modbus.WriteDataAsync(commData))
{
    // Do something for error handling...
}
 
// 2. List case
 
List<CommunicationData> commDatas = new List<CommunicationData>();
 
commDatas.Add(new CommunicationData(DataStorage.HoldingRegister, 0, 3.465F, modbus.Interface.EndianOption));
commDatas.Add(new CommunicationData(DataStorage.Coil, 0, true, modbus.Interface.EndianOption));
commDatas.Add(new CommunicationData(DataStorage.HoldingRegister, 2, 534, modbus.Interface.EndianOption));
 
if (!await modbus.WriteDataAsync(commDatas))
{
    // Do something for error handling...
}

2. API Information

2.1. namespace NModbus4.Wrapper.Define

2.1.1. enum LogLevel

NameValue
Communication1
Exception2

2.1.2. enum CommunicationException

NameValueDescription
SlaveFunctionCodeException1
SlaveUnimplementedException2
MasterTransportNullException3Remote connection lost

2.1.3. enum DataType

  • Support data type
NameValue
Bool1
Int2
Float4

2.1.4. enum ModbusType

  • Set Network type, Master / Slave mode
NameValue
RTU_Master1
TCP_Master2
UDP_Master3
RTU_Slave11
TCP_Slave12
UDP_Slave13

2.1.5. enum DataStorage

NameValue
Coil0
DiscreteInput1
InputRegister2
HoldingRegister3

2.1.6. enum Endian

  • Remote's endian matching enum
NameValue
Big1
Little2

2.1.7. struct SerialPortInformation

TypeName
stringPort
intBaudrate
System.IO.Ports.ParityParity
intDatabits
System.IO.Ports.StopBitsStopbits
System.IO.Ports.HandshakeHandshake

2.1.8. struct EthernetInformation

TypeName
stringAddress
intPort

2.1.9. class CommunicationData

  • Built in type for communication of NModbus4.Wrapper
2.1.9.1. Members, Methods
  1. Members
TypeNameDescription
DataStorageDataStorage
DataTypeDataType
EndianRemoteEndian
ushortStartAddressAddress of data. StartAddress >= 0
objectValuebool, int, float
  1. Methods
Return typeMethodDescription
voidCommunicationData(DataStorage, ushort, object, Endian)ctor
List<ushort>GetHexData()Parsing Value to hex data

2.1.10. class ModbusInterface

2.1.10.1 Members, Methods
  1. Members
TypeNameDescription
const intTransactionLimitFor packet dividing
stringName
ModbusTypeModbusType
intSlaveNumberFor master mode
SerialPortInformation?SerialPortInformation
EthernetInformation?EthernetInformation
EndianEndianOptionRemote's endian
  1. Methods
Return typeMethodDescription
voidModbusInterface(string, ModbusType, int, SerialPortInformation, Endian)ctor
voidModbusInterface(string, ModbusType, int, EthernetInformation, Endian)ctor

2.2. namespace NModbus4.Wrapper

2.2.1. class ModbusService

2.2.1.1 Members, Methods, Events
  1. Members

    TypeNameDescription
    ModbusInterfaceInterfaceCommunication information
  2. Methods

    Return typeMethodDescription
    voidModbusService(ModbusInterface, Action<ModbusInterface, bool>)ctor
    voidConnect()Connect to remote master / slave
    voidDispose()Disconnect connection / release thread
    Action<ModbusInterface, bool>ConnectCallbackConnection status changed notice
    Read methods
    boolReadData<T>(DataStorage, int, out T)Read function
    boolReadData<T>(DataStorage, int, int, out List<T>)Read function
    boolReadData(ref CommunicationData)Read function using built in type
    boolReadData(ref List<CommunicationData>)Read function using built in type
    Task<(bool, T)>ReadDataAsync<T>(DataStorage, int)Async read function
    Task<(bool, List<T>)>ReadDataAsync<T>(DataStorage, int, int)Async read function
    Task<(bool, CommunicationData)>ReadDataAsync(CommunicationData)Async read function using built in type
    Task<(bool, List<CommunicationData>)>ReadDataAsync(List<CommunicationData>)Async read function using built in type
    Write methods
    boolWriteData<T>(DataStorage, int, T)Write function
    boolWriteData<T>(DataStorage, int, List<T>)Write function
    boolWriteData(CommunicationData)Write function using built in type
    boolWriteData(List<CommunicationData>)Write function using built in type
    Task<bool>WriteDataAsync<T>(DataStorage, int, T)Async write function
    Task<bool>WriteDataAsync<T>(DataStorage, int, List<T>)Async write function
    Task<bool>WriteDataAsync(CommunicationData)Async write function using built in type
    Task<bool>WriteDataAsync(List<CommunicationData>)Async write function using built in type
    Master methods
    -
    Slave methods
    voidClearDataStore()Clear and initialize all data
  3. Events

    TypeEventDescription
    voidModbusGeneralExceptionHandler(ModbusInterface)Deal general, unknown exception
    voidModbusCommunicationExceptionHandler(ModbusInterface, CommunicationException)Usually need to recreate class for communication after called this event
    voidModbusLogHandler(ModbusInterface, LogLevel, string)Modbus log
    Master events
    -
    Slave events
    voidModbusDataReceivedHandler(ModbusInterface, DataStorage, List<int>, List<ushort>)Remote master wrote data to this slave