Modbus RTU 通信工具设计

Modbus 是一个工业上常用的通讯协议、一种通讯约定。

ModBus 协议是应用层报文传输协议(OSI 模型第7层),它定义了一个与通信层无关的协议数据单元(PDU),即PDU=功能码+数据域。 ModBus 协议能够应用在不同类型的总线或网络。对应不同的总线或网络,Modbus 协议引入一些附加域映射成应用数据单元(ADU),即ADU=附加域+PDU。目前,Modbus 有下列三种通信方式: 1.    以太网,对应的通信模式是Modbus TCP。 2.    异步串行传输(各种介质如有线RS-232-/422/485/;光纤、无线等),对应的通信模式是 Modbus RTU 或 Modbus ASCII。        Modbus 的ASCII、RTU 协议规定了消息、数据的结构、命令和应答的方式,数据通讯采用Maser/Slave方式。 3.    高速令牌传递网络,对应的通信模式是Modbus PLUS。

 

Modbus 需要对数据进行校验,串行协议中除有奇偶校验外,ASCII 模式采用LRC 校验;RTU 模式采用16位CRC 校验;TCP 模式没有额外规定校验,因为TCP 是一个面向连接的可靠协议。

 

Modbus 协议的应用中,最常用的是Modbus RTU 传输模式。

 

RTU 传输模式

当设备使用RTU (Remote Terminal Unit) 模式在 Modbus  串行链路通信, 报文中每个8位字节含有两个4位十六进制字符。这种模式的主要优点是较高的数据密度,在相同的波特率下比ASCII 模式有更高的吞吐率。每个报文必须以连续的字符流传送。

 

RTU 模式每个字节 ( 11 位 ) 的格式为:

编码系统:  8位二进制。 报文中每个8位的字节含有两个4位十六进制字符(0–9, A–F)

Bits per Byte:  1 起始位

8 数据位, 首先发送最低有效位

1 位作为奇偶校验

1 停止位

偶校验是要求的,其它模式 ( 奇校验, 无校验 ) 也可以使用。为了保证与其它产品的最大兼容性,同时支持无校验模式是建议的。默认校验模式模式 必须为偶校验。注:使用无校验要求2 个停止位。 

字符的串行传送方式:

每个字符或字节均由此顺序发送(从左到右):最低有效位 (LSB) . . . 最高有效位 (MSB)

图1:RTU 模式位序列

 

设备配置为奇校验、偶校验或无校验都可以接受。如果无奇偶校验,将传送一个附加的停止位以填充字符帧:

图2:RTU 模式位序列 (无校验的特殊情况)

帧检验域:循环冗余校验 (CRC)

在RTU 模式包含一个对全部报文内容执行的,基于循环冗余校验 (CRC - Cyclical Redundancy Checking) 算法的错误检验域。

CRC 域检验整个报文的内容。不管报文有无奇偶校验,均执行此检验。

CRC 包含由两个8位字节组成的一个16位值。

CRC 域作为报文的最后的域附加在报文之后。计算后,首先附加低字节,然后是高字节。CRC 高字节为报文发送的最后一个子节。

附加在报文后面的CRC 的值由发送设备计算。接收设备在接收报文时重新计算 CRC 的值,并将计算结果于实际接收到的CRC 值相比较。如果两个值不相等,则为错误。

CRC 的计算,开始对一个16位寄存器预装全1。 然后将报文中的连续的8位子节对其进行后续的计算。只有字符中的8个数据位参与生成CRC 的运算,起始位,停止位和校验位不参与 CRC 计算。

CRC 的生成过程中, 每个 8–位字符与寄存器中的值异或。然后结果向最低有效位(LSB)方向移动(Shift) 1位,而最高有效位(MSB)位置充零。 然后提取并检查 LSB:如果LSB 为1, 则寄存器中的值与一个固定的预置值异或;如果LSB 为 0, 则不进行异或操作。

这个过程将重复直到执行完8次移位。完成最后一次(第8次)移位及相关操作后,下一个8位字节与寄存器的当前值异或,然后又同上面描述过的一样重复8次。当所有报文中子节都运算之后得到的寄存器中的最终值,就是CRC。

当CRC 附加在报文之后时,首先附加低字节,然后是高字节。

CRC 算法如下:

private bool CheckResponse(byte[] response)
{
//Perform a basic CRC check:
byte[] CRC = new byte[];
GetCRC(response, ref CRC);
if (CRC[] == response[response.Length - ] && CRC[] == response[response.Length - ])
return true;
else
return false;
} private void GetCRC(byte[] message, ref byte[] CRC)
{
//Function expects a modbus message of any length as well as a 2 byte CRC array in which to
//return the CRC values: ushort CRCFull = 0xFFFF;
byte CRCHigh = 0xFF, CRCLow = 0xFF;
char CRCLSB; for (int i = ; i < (message.Length) - ; i++)
{
CRCFull = (ushort)(CRCFull ^ message[i]); for (int j = ; j < ; j++)
{
CRCLSB = (char)(CRCFull & 0x0001);
CRCFull = (ushort)((CRCFull >> ) & 0x7FFF); if (CRCLSB == )
CRCFull = (ushort)(CRCFull ^ 0xA001);
}
}
CRC[] = CRCHigh = (byte)((CRCFull >> ) & 0xFF);
CRC[] = CRCLow = (byte)(CRCFull & 0xFF);
}

帧描述 (如下图所示) :

图3:RTU 报文帧

注意:Modbus  RTU 帧最大为256字节。

 

下面是我为公司设计的一个 Modbus RTU 通信测试小工具,界面截图如下:

图4:Modbus RTU 通信工具

 

我的通用Modbus RTU 动态库,modbus.cs 如下:

modbus.cs

using System;
using System.Collections.Generic;
using System.Text;
using System.IO.Ports;
using System.Threading; namespace SerialPort_Lib
{
public class modbus
{
private SerialPort sp = new SerialPort();
public string modbusStatus; #region Constructor / Deconstructor
public modbus()
{
}
~modbus()
{
}
#endregion #region Open / Close Procedures
public bool Open(string portName, int baudRate, int databits, Parity parity, StopBits stopBits)
{
//Ensure port isn't already opened:
if (!sp.IsOpen)
{
//Assign desired settings to the serial port:
sp.PortName = portName;
sp.BaudRate = baudRate;
sp.DataBits = databits;
sp.Parity = parity;
sp.StopBits = stopBits;
//These timeouts are default and cannot be editted through the class at this point:
sp.ReadTimeout = -;
sp.WriteTimeout = ; try
{
sp.Open();
}
catch (Exception err)
{
modbusStatus = "Error opening " + portName + ": " + err.Message;
return false;
}
modbusStatus = portName + " opened successfully";
return true;
}
else
{
modbusStatus = portName + " already opened";
return false;
}
}
public bool Close()
{
//Ensure port is opened before attempting to close:
if (sp.IsOpen)
{
try
{
sp.Close();
}
catch (Exception err)
{
modbusStatus = "Error closing " + sp.PortName + ": " + err.Message;
return false;
}
modbusStatus = sp.PortName + " closed successfully";
return true;
}
else
{
modbusStatus = sp.PortName + " is not open";
return false;
}
}
#endregion #region CRC Computation
private void GetCRC(byte[] message, ref byte[] CRC)
{
//Function expects a modbus message of any length as well as a 2 byte CRC array in which to
//return the CRC values: ushort CRCFull = 0xFFFF;
byte CRCHigh = 0xFF, CRCLow = 0xFF;
char CRCLSB; for (int i = ; i < (message.Length) - ; i++)
{
CRCFull = (ushort)(CRCFull ^ message[i]); for (int j = ; j < ; j++)
{
CRCLSB = (char)(CRCFull & 0x0001);
CRCFull = (ushort)((CRCFull >> ) & 0x7FFF); if (CRCLSB == )
CRCFull = (ushort)(CRCFull ^ 0xA001);
}
}
CRC[] = CRCHigh = (byte)((CRCFull >> ) & 0xFF);
CRC[] = CRCLow = (byte)(CRCFull & 0xFF);
}
#endregion #region Build Message
private void BuildMessage(byte address, byte type, ushort start, ushort registers, ref byte[] message)
{
//Array to receive CRC bytes:
byte[] CRC = new byte[]; message[] = address;
message[] = type;
message[] = (byte)(start >> );
message[] = (byte)start;
message[] = (byte)(registers >> );
message[] = (byte)registers; GetCRC(message, ref CRC);
message[message.Length - ] = CRC[];
message[message.Length - ] = CRC[];
}
#endregion #region Check Response
private bool CheckResponse(byte[] response)
{
//Perform a basic CRC check:
byte[] CRC = new byte[];
GetCRC(response, ref CRC);
if (CRC[] == response[response.Length - ] && CRC[] == response[response.Length - ])
return true;
else
return false;
}
#endregion #region Get Response
private void GetResponse(ref byte[] response)
{
//There is a bug in .Net 2.0 DataReceived Event that prevents people from using this
//event as an interrupt to handle data (it doesn't fire all of the time). Therefore
//we have to use the ReadByte command for a fixed length as it's been shown to be reliable.
for (int i = ; i < response.Length; i++)
{
response[i] = (byte)(sp.ReadByte());
}
}
#endregion #region GetModbusData 获得接收数据
public bool GetModbusData(ref byte[] values)
{
//Ensure port is open:
if (sp.IsOpen)
{
// 等待线程进入
//Monitor.Enter(sp); //Clear in/out buffers:
//sp.DiscardOutBuffer();
//sp.DiscardInBuffer(); //Message is 1 addr + 1 type + N Data + 2 CRC try
{
//GetResponse(ref readBuffer);
//string str = readBuffer.ToString(); int count = sp.BytesToRead;
if (count > )
{
byte[] readBuffer = new byte[count]; GetResponse(ref readBuffer); // readData = new byte[29];
// Array.Copy(readBuffer, readData, readData.Length); // CRC 验证
if (CheckResponse(readBuffer))
{
//显示输入数据
values = readBuffer; modbusStatus = "Write successful"; sp.DiscardInBuffer(); //values = System.Text.Encoding.ASCII.GetString(readData);
return true;
}
else
{
modbusStatus = "CRC error"; sp.DiscardInBuffer(); return false;
}
}
else return false;
}
catch (Exception err)
{
modbusStatus = "Error in write event: " + err.Message; sp.DiscardInBuffer(); return false;
} //finally
//{
// 通知其它对象
//Monitor.Pulse(sp);
// 释放对象锁
//Monitor.Exit(sp);
//}
}
else
{
modbusStatus = "Serial port not open";
return false;
}
}
#endregion #region SendModbusData 打包发送数据
public bool SendModbusData(ref byte[] values)
{
//Ensure port is open:
if (sp.IsOpen)
{
//Clear in/out buffers:
sp.DiscardOutBuffer();
sp.DiscardInBuffer(); //Function 3 response buffer:
byte[] response = new byte[values.Length + ];
Array.Copy(values, response, values.Length); //BuildMessage(address, (byte)3, start, registers, ref message); //打包带有 CRC 验证的modbus 数据包:
byte[] CRC = new byte[];
GetCRC(response, ref CRC);
response[response.Length - ] = CRC[];
response[response.Length - ] = CRC[]; values = response; //返回带有 CRC 验证的modbus 数据包 //Send modbus message to Serial Port:
try
{
sp.Write(response, , response.Length);
//GetResponse(ref response);
return true;
}
catch (Exception err)
{
modbusStatus = "Error in read event: " + err.Message;
return false;
}
//Evaluate message:
//if (CheckResponse(response))
//{
// //Return requested register values:
// for (int i = 0; i < (response.Length - 5) / 2; i++)
// {
// values[i] = response[2 * i + 3];
// values[i] <<= 8;
// values[i] += response[2 * i + 4];
// }
// modbusStatus = "Read successful";
// return true;
//}
//else
//{
// modbusStatus = "CRC error";
// return false;
//}
}
else
{
modbusStatus = "Serial port not open";
return false;
} }
#endregion }
}

调用的主要代码如下:

modbus类的winform调用代码

public partial class FormConfig : Form,IModbusData
{
//业务处理类
B_ModbusData ModbusDataBLL = new B_ModbusData(); modbus mb = new modbus();
//SerialPort sp = new SerialPort();
System.Timers.Timer timer = new System.Timers.Timer(); public FormConfig()
{
InitializeComponent(); timer.Elapsed += new ElapsedEventHandler(timer_Elapsed);
} #region Timer Elapsed 事件处理程序
bool runEnd = true;
void timer_Elapsed(object sender, ElapsedEventArgs e)
{
if (runEnd == true)
{
runEnd = false;
PollFunction();
runEnd = true;
}
} //定时器调用方法
private void PollFunction()
{
byte[] values = null;
try
{
mb.GetModbusData(ref values);
//while (!mb.SendFc3(Convert.ToByte(txtSlaveID.Text), pollStart, pollLength, ref values)) ;
}
catch (Exception err)
{
DoGUIStatus("Error in modbus read: " + err.Message);
} if (values != null)
{
//业务处理
byte[] sendData = ModbusDataProcess(values);
}
}
#endregion #region IModbusData 接口成员处理
public byte[] ModbusDataProcess(byte[] _data)
{
byte[] sendData = ModbusDataBLL.ModbusDataProcess(_data); // CRC验证,并打包发送数据。
mb.SendModbusData(ref sendData); return sendData;
}
#endregion
}

其实,三步就能成功调用:

modbus mb = new modbus();
mb.GetModbusData(ref values); // 从串口设备获得数据。
byte[] sendData = ModbusDataBLL.ModbusDataProcess(values); // 你的业务处理,并产生最终返回数据。
mb.SendModbusData(ref sendData); // CRC验证,并打包发送数据。

主要代码已全部提供,由于工作原因暂不提供完整工具源代码,见谅!

(完)

Modbus RTU 通信工具设计(转)的更多相关文章

  1. Modbus RTU 通信应用案例

    如何打开项目归档文件 例程中的TIA博途项目文件与STEP 7项目文件均为归档文件,需要按如下方式打开: TIA博途项目文件 1. 打开TIA博途软件,通过软件左下方“项目视图”按钮切换至项目视图: ...

  2. FreeModbus 移植于STM32 实现Modbus RTU通信

    http://ntn314.blog.163.com/blog/static/161743584201233084434579/ 毕业设计自己要做个基于STM32的PLC能直接跑语句表的,现在看来好像 ...

  3. C# 开发Modbus Rtu客户端 modbus测试Demo,Modbus 串口通信 , 虚拟MODBUS-RTU测试

    前言 本文将使用一个NuGet公开的组件技术来实现一个ModBus RTU的客户端,方便的对Modbus rtu的服务器进行读写,这个服务器可以是电脑端C#设计的,也可以是PLC实现的,也可以是其他任 ...

  4. C# NModbus RTU通信实现

    Modbus协议时应用于电子控制器上的一种通用语言.通过此协议,控制器相互之间.控制器经由网络/串口和其它设备之间可以进行通信.它已经成为了一种工业标准.有了这个通信协议,不同的厂商生成的控制设备就可 ...

  5. Modbus RTU新版本指令介绍

    Modbus RTU新版本指令介绍 TIA V13 SP1版本软件中提供了2个版本的Modbus RTU指令: 图1. 两个版本Modbus RTU指令 早期版本的Modbus RTU指令(图1. 中 ...

  6. Modbus RTU 介绍

    S7-1200 Modbus RTU 通信概述 Modbus具有两种串行传输模式:分别为ASCII和RTU.Modbus是一种单主站的主从通信模式,Modbus网络上只能有一个主站存在,主站在Modb ...

  7. Modbus通用数据读取工具设计及使用

    一.公共功能码定义 二.能读取的数据类型 1.bit类型,比如01功能码,读到的就是位的状态,是ON 还是OFF,也就是对应着0或1. 2.byte类型,比如03功能码. 3.short类型,比如03 ...

  8. 多平台下Modbus通信协议库的设计(一)

    1.背景 1.1.范围 MODBUS 是 OSI 模型第 7 层上的应用层报文传输协议, 它在连接至不同类型总线或网络的设备之间提供客户机/服务器通信. 自从 1979 年出现工业串行链路的事实标准以 ...

  9. 0-20ma 0-5V,0-10V ,0-15V ,0-20V,0-30V模拟量(范围可以定制)多功能采集模块,支持1路继电器输出,2路Di输入,8路Ai输入,可电脑控制,支持485 modbus rtu协议。端口参数可以配置保存,支持定制修改。

    多功能模拟量采集模块MRD-5017具有8 通道模拟量采集(支持0-20mA,0-5V,0-10V混合测量),2路DI,1路继电器输出,1路485接口(支持MODBUS RTU),能实现8路AI(12 ...

随机推荐

  1. Ansible11:变量详解【转】

    一.在Inventory中定义变量 详见<Ansible2:主机清单> 二.在Playbook中定义变量 1.通过vars关键字定义: vars: http_port: 80 server ...

  2. android:onTouch()和onTouchEvent()的区别?看完这篇文章就知道了

    Android Touch Screen 与传统Click Touch Screen不同,会有一些手势(Gesture),例如Fling,Scroll等等.这些Gesture会使用户体验大大提升. A ...

  3. 用Chrome开发者工具做JavaScript性能分析

    来源: http://blog.jobbole.com/31178/ 你的网站正常运转.现在我们来让它运转的更快.网站的性能由页面载入速度和代码执行效率决定.一些服务可以让你的网站载入更快,比如压缩J ...

  4. WPF子窗体:ChildWindow

    wpf的子窗体选择有很多种,如最常见的是项目新建窗体(Window)作为子窗体 ,或者新建wpf用户控件(UserControl).而其实利用Xceed.Wpf.Toolkit.dll 可以轻松布局如 ...

  5. Socks

    Socks time limit per test 2 seconds memory limit per test 256 megabytes input standard input output ...

  6. %3f URL --> '?'拼接引发的问题

    转载自:https://www.reddit.com/r/swift/comments/2w19kp/how_do_you_send_a_through_nsmutableurlrequest/ ho ...

  7. Dom++完美版得到元素到html的距离6/4/21

    function getTop(obj) { var pos={left:0,top:0}; while(obj) { pos.left+=obj.offsetLeft; pos.top+=obj.o ...

  8. jQuery checkbox 全选

    jQuery 1.6版本以后 if($("#id").attr("checked")) 不能返回 ture 和 false 高版本中jQuery 提供prop ...

  9. android 内存优化一

    常见内存泄露原因 Context对象泄漏 1.如果一个类持有Context对象的强引用,就需要检查其生存周期是否比Context对象更长.否则就可能发生Context泄漏. 2.View持有其创建所在 ...

  10. Linux学习 -- 启动管理

    1 CentOS 6.x启动管理 系统运行级别 运行级别命令 #runlevel   查看级别 #init 运行级别  改变级别 系统默认运行级别 配置文件 /etc/inittab id:3:ini ...