自从开源了我们自己开发的Modbus协议栈之后,有很多朋友建议我针对性的做几个示例。所以我们就基于平时我们的应用整理了几个简单但可以说明基本的应用方法的示例,这一篇中我们将使用协议栈实现一个Modbus RTU从站应用。

1、何为RTU从站

Modbus协议是一个主从协议,那肯定就有主站和从站之分。所谓从站就是被动动响应通讯的对象,所以从站总是响应通讯的一方。

对于RTU从站来说,它是数据的数据的生产者,从站通过响应主站数据请求的方式将数据发送给主站。这一过程如下图所示:

从上图我们不难看出,首先主站要主动发起数据请求,这也是它为什么被称之为主站的缘由。它首先告诉从站我需要哪些数据。然后从站按照主站的请求返回数据。主站得到响应后解析数据,这样就完成了主从站之间的一次数据通讯。所以主站就需要主动发起每一次数据通讯的对象。

2、如何实现RTU从站

我们已经了解的从站总是响应主站的数据请求来实现数据的传送。下面我们来看看使用协议栈如何实现一个从站。

我们知道从站是数据的生产者,对于Modbus协议来说有四类数据:线圈、状态、输入寄存器和保持寄存器。所以在从站中我们要为这四种数据定义相应的地址,以便主站能够对应的访问。所以设计一个从站我们先来设计它的数据地址,在我们的例子中我们规定如下:

我们规定了每类数据类型的数量为8,对于从站来说除了生成这些数据外,还需要根据主站的数据请求来返回相应的数据响应。在我们的协议栈中实现了0x01、0x02、0x03、0x04、0x05、0x06、0x0F以及0x10等功能码。也就是说主站对象会生成面向这些功能码的从站数据请求。从站收到请求后,解析请求并根据请求生成响应的数据响应。可以表示为下图所示:

从上图我们明白协议栈中已经实现了对收到的主站数据请求进行解析以及根据解析生成对应的响应的函数。我们使用协议栈时,主要需要做两个方面的事情:解析数据请求和生成数据响应。

在协议栈中定义了一个解析函数,该函数将收到的数据请求消息解析,并根据解析的结果生成返回的数据响应。该函数的原型如下:

uint16_t ParsingMasterAccessCommand(uint8_t *receivedMessage, uint8_t *respondBytes, uint16_t rxLength, uint8_t StationAddress)

这个函数有四个参数:uint8_t *receivedMessage是收到的数据请求消息; uint8_t *respondBytes是返回的数据响应消息,也是函数需要生成的;uint16_t rxLength是接收到的数据请求消息的长度;uint8_t StationAddress本站的地址。而函数的返回值则是生成的数据响应详细的长度。

在解析的过程中,该函数判断消息的完整性,并根据不同的功能码调用不同的回调函数来实现,包括设置本地数据和获取本地数据的相关回调函数,在后续将讨论它们的实现。

3RTU从站编码

我们已经详述了使用协议栈实现RTU从站的方法,接下来我们就来利用协议栈具体开发一个RTU从站的实例。

我们调用解析函数对接收到的数据请求进行解析,具体调用方式如下所示:

respondLength=ParsingMasterAccessCommand(hgudRxBuffer,respondBytes,hgudRxLength,StationAddress);

返回值会有3种情况,返回值为0则表示接收到的数据请求消息是错误的。返回值为65535则表示返回的消息尚未接收完整。返回的是一个合适的数值则表示解析成功,返回了数据响应的长度。

当然我们需要实现8个回调函数,分别是获取线圈量、获取状态量、获取输入寄存器和获取保持寄存器,以及预置单个线圈量、预置多个线圈量、预置单个保持寄存器和预置多个保持寄存器。函数原型定义如下:

/*获取想要读取的Coil量的值*/
__weak void GetCoilStatus(uint16_t startAddress,uint16_t quantity,bool *statusList)
{//如果需要Modbus TCP Server/RTU Slave应用中实现具体内容
}/*获取想要读取的InputStatus量的值*/
__weak void GetInputStatus(uint16_t startAddress,uint16_t quantity,bool *statusValue)
{//如果需要Modbus TCP Server/RTU Slave应用中实现具体内容
}/*获取想要读取的保持寄存器的值*/
__weak void GetHoldingRegister(uint16_t startAddress,uint16_t quantity,uint16_t *registerValue)
{//如果需要Modbus TCP Server/RTU Slave应用中实现具体内容
}/*获取想要读取的输入寄存器的值*/
__weak void GetInputRegister(uint16_t startAddress,uint16_t quantity,uint16_t *registerValue)
{//如果需要Modbus TCP Server/RTU Slave应用中实现具体内容
}/*设置单个线圈的值*/
__weak void SetSingleCoil(uint16_t coilAddress,bool coilValue)
{//如果需要Modbus TCP Server/RTU Slave应用中实现具体内容
}/*设置单个寄存器的值*/
__weak void SetSingleRegister(uint16_t registerAddress,uint16_t registerValue)
{//如果需要Modbus TCP Server/RTU Slave应用中实现具体内容
}/*设置多个线圈的值*/
__weak void SetMultipleCoil(uint16_t startAddress,uint16_t quantity,bool *statusValue)
{//如果需要Modbus TCP Server/RTU Slave应用中实现具体内容
}/*设置多个寄存器的值*/
__weak void SetMultipleRegister(uint16_t startAddress,uint16_t quantity,uint16_t *registerValue)
{//如果需要Modbus TCP Server/RTU Slave应用中实现具体内容
}

我们需要做的工作就是根据我们具体实例中4类数据量的地址分配来实现这8个回调函数。当然,如果从站没有某一类数据量操作,回调函数则不需要编写。在我们的实例中我们将这几个函数实现如下:

/*获取想要读取的Coil量的值*/
void GetCoilStatus(uint16_t startAddress,uint16_t quantity,bool *statusList)
{uint16_t start;uint16_t count;/*先判断地址是否处于合法范围*/start=(startAddress>CoilStartAddress)?((startAddress<=CoilEndAddress)?startAddress:CoilEndAddress):CoilStartAddress;count=((start+quantity-1)<=CoilEndAddress)?quantity:(CoilEndAddress-start);for(int i=0;i<count;i++){statusList[i]=dPara.coil[start+i];}
}/*获取想要读取的保持寄存器的值*/
void GetHoldingRegister(uint16_t startAddress,uint16_t quantity,uint16_t *registerValue)
{uint16_t start;uint16_t count;/*先判断地址是否处于合法范围*/start=(startAddress>HoldingResterStartAddress)?((startAddress<=HoldingResterEndAddress)?startAddress:HoldingResterEndAddress):HoldingResterStartAddress;count=((start+quantity-1)<=HoldingResterEndAddress)?quantity:(HoldingResterEndAddress-start);for(int i=0;i<count;i++){registerValue[i]=aPara.holdingRegister[start+i];}
}/*设置单个线圈的值*/
void SetSingleCoil(uint16_t coilAddress,bool coilValue)
{/*先判断地址是否处于合法范围*/if((4<=coilAddress)&&(coilAddress<=CoilEndAddress)){dPara.coil[coilAddress]=coilValue;}PresetSlaveCoilControll(coilAddress,coilAddress);
}/*设置多个线圈的值*/
void SetMultipleCoil(uint16_t startAddress,uint16_t quantity,bool *statusValue)
{uint16_t endAddress=startAddress+quantity-1;if((4<=startAddress)&&(startAddress<=CoilEndAddress)&&(4<=endAddress)&&(endAddress<=CoilEndAddress)){for(int i=0;i<quantity;i++){dPara.coil[i+startAddress]=statusValue[i];}}PresetSlaveCoilControll(startAddress,endAddress);
}/*设置单个寄存器的值*/
void SetSingleRegister(uint16_t registerAddress,uint16_t registerValue)
{bool noError=(bool)(((41<=registerAddress)&&(registerAddress<=42))||((44<=registerAddress)&&(registerAddress<=45))||((50<=registerAddress)&&(registerAddress<=51))||((54<=registerAddress)&&(registerAddress<=55))||((58<=registerAddress)&&(registerAddress<=59)));if(noError){aPara.holdingRegister[registerAddress]=registerValue;}WriteSlaveRegisterControll(registerAddress,registerAddress);
}/*设置多个寄存器的值*/
void SetMultipleRegister(uint16_t startAddress,uint16_t quantity,uint16_t *registerValue)
{uint16_t endAddress=startAddress+quantity-1;bool noError=(bool)(((8<=startAddress)&&(startAddress<=15)&&(8<=endAddress)&&(endAddress<=15))||((41<=startAddress)&&(startAddress<=42)&&(41<=endAddress)&&(endAddress<=42))||((44<=startAddress)&&(startAddress<=47)&&(44<=endAddress)&&(endAddress<=47))||((50<=startAddress)&&(startAddress<=51)&&(50<=endAddress)&&(endAddress<=51))||((54<=startAddress)&&(startAddress<=55)&&(54<=endAddress)&&(endAddress<=55))||((58<=startAddress)&&(startAddress<=59)&&(58<=endAddress)&&(endAddress<=59))||((62<=startAddress)&&(startAddress<=67)&&(62<=endAddress)&&(endAddress<=67))||((72<=startAddress)&&(startAddress<=77)&&(72<=endAddress)&&(endAddress<=77))||((82<=startAddress)&&(startAddress<=87)&&(82<=endAddress)&&(endAddress<=87))||((92<=startAddress)&&(startAddress<=97)&&(92<=endAddress)&&(endAddress<=97))||((100<=startAddress)&&(startAddress<=115)&&(100<=endAddress)&&(endAddress<=115)));if(noError){for(int i=0;i<quantity;i++){aPara.holdingRegister[startAddress+i]=registerValue[i];}}WriteSlaveRegisterControll(startAddress,endAddress);
}

到这里对从站的开发实际已经完成。对于这些回调函数并不是全部需要编写,而是要根据我们自己定义的从站各类参数的地址分配来实现。

4RTU从站小结

我们实现了一个简单的RTU从站实例,我们可以通过一些RTU主站软件来测试它。这样的软件有很多,常见的如Modscan、Modbus Poll等。这里我们使用Modbus Poll来测试一下,如下图所示:

RTU从站的实现相对较简单,因为在同一台设备上只需实现一个从站,哪怕是通过不同的端口来访问。这一点与主站是不一样的,原因是从站的数据是自己产生,而且只需被动响应主站请求,而且理论上同一条总线只会有一个主站。

接下来我们来总结一下使用协议栈实现RTU从站的工作流程,或者说实现的步骤。首先从站要解析从主站送来的数据请求。在协议栈中已经封装了数据请求的解析函数、所以我们实现从站时首先就是调用这一函数来解析接收到的数据请求消息。

然后将解析函数返回的数据响应消息发送到主站就可以了。也就是说使用协议栈,只需要调用一下这个函数从站功能就实现了。这是因为这个函数实现了整个从站的响应过程,大致分三个步骤:第一步,解析收到的主站数据请求消息;第二步,根据解析的结果预置数据或者获取数据,预置和获取数据由8个回调函数实现;第三步,生成从站数据响应消息。说到这里我们已经清楚,RTU从站必须实现这些回调函数,其它工作则全由协议栈完成。

源码下载:https://download.csdn.net/download/foxclever/12882021

欢迎关注:

Modbus协议栈应用实例之二:Modbus RTU从站应用相关推荐

  1. Modbus协议栈应用实例之一:Modbus RTU主站应用

    自从开源了我们自己开发的Modbus协议栈之后,有很多朋友建议我针对性的做几个示例.所以我们就基于平时我们的应用整理了几个简单但可以说明基本的应用方法的示例,在这一篇中我们先来使用协议栈实现Modbu ...

  2. Modbus协议栈应用实例之六:Modbus ASCII从站应用

    自从开源了我们自己开发的Modbus协议栈之后,有很多朋友建议我针对性的做几个示例.所以我们就基于平时我们的应用整理了几个简单但可以说明基本的应用方法的示例,这一篇中我们来使用协议栈实现Modbus ...

  3. Modbus协议栈应用实例之五:Modbus ASCII主站应用

    自从开源了我们自己开发的Modbus协议栈之后,有很多朋友建议我针对性的做几个示例.所以我们就基于平时我们的应用整理了几个简单但可以说明基本的应用方法的示例,这一篇中我们来使用协议栈实现Modbus ...

  4. Modbus协议栈应用实例之三:Modbus TCP客户端应用

    自从开源了我们自己开发的Modbus协议栈之后,有很多朋友建议我针对性的做几个示例.所以我们就基于平时我们的应用整理了几个简单但可以说明基本的应用方法的示例,这一篇中我们将解说如何使用协议栈实现一个M ...

  5. Modbus协议栈综合实例设计

      自我们开源了我们的Modbus协议栈之后,就一直有朋友来信说希望提供示例.这次我们整理了几个例子以供参考. 1.应用实例规划   在这次的实例中,我们使用的目标板拥有一个以太网接口.一个RS232 ...

  6. Modbus协议栈应用实例之四:ModbusTCP服务器应用

    自从开源了我们自己开发的Modbus协议栈之后,有很多朋友建议我针对性的做几个示例.所以我们就基于平时我们的应用整理了几个简单但可以说明基本的应用方法的示例,这一篇中我们来简述如何使用协议栈实现一个M ...

  7. Modbus协议栈开发笔记之二:Modbus消息帧的生成

    前面我们已经对Modbus的基本事务作了说明,也据此设计了我们将要实现的主从站的操作流程.这其中与Modbus直接相关的就是Modbus消息帧的生成.Modbus消息帧也是实现Modbus通讯协议的根 ...

  8. modbus tcp主站和从站_【技术】Modbus协议栈应用实例之四:ModbusTCP服务器应用

    源码下载:https://download.csdn.net/download/foxclever/12838885 自从开源了我们自己开发的Modbus协议栈之后,有很多朋友建议我针对性的做几个示例 ...

  9. Modbus协议栈开发笔记之六:Modbus RTU Master开发

    这一节我们来封装最后一种应用(Modbus RTU Master应用),RTU主站的开发与TCP客户端的开发是一致的.同样的我们也不是做具体的应用,而是实现RTU主站的基本功能.我们将RTU主站的功能 ...

最新文章

  1. vs2015 x86 opencv3.3(编译)
  2. Access处理DISTINCT的Bug?
  3. 各种梯度下降 bgd sgd mbgd adam
  4. java建一个conversion_Scala中的JavaConverters和JavaConversions之间有什么区别?
  5. 老李推荐:第6章6节《MonkeyRunner源码剖析》Monkey原理分析-事件源-事件源概览-命令队列...
  6. zabbix-2.0.8日常巡检-检测项目状态
  7. 最大似然估计和最大后验估计
  8. 计算机木马不会主动传播什么疾病,研究如何制作自动运行的木马病毒以及如何传播...
  9. 设计模式:UML类图、策略模式、单例模式、工厂模式、观察者模式
  10. 数据库索引怎么实现的
  11. Linux下 lnmp一键安装
  12. Windows server 2008 安装Hyper-V
  13. 机房收费系统心得总结
  14. ROS2极简总结-新增概念
  15. 实验八 页面置换模拟程序设计
  16. 智云通CRM:采购决策有哪些关键节点?
  17. 狗狗变形记:任选4点的投影变换warpPerspective OpenCV-Python案例
  18. java饲养员喂动物_做一个饲养员给动物喂食物的例子体现JAVA中的面向对象思想,接口(抽象类)的用处...
  19. 数学建模学习——聚类(包含优秀建模论文中的应用)
  20. cmpp发送超长短信息1

热门文章

  1. [bzoj4994][Usaco2017 Feb]Why Did the Cow Cross the Road III_树状数组
  2. ACM训练计划建议(转)
  3. 老李谈HTTP1.1的长连接
  4. Thinkphp系统常量
  5. 地灾应急暨地灾危险性评估培训班学习笔记
  6. 计算机网络(二十)-广域网-PPP协议和HDLC协议
  7. 操作系统基本特性——并发、共享、虚拟、异步
  8. yaml parse python_python-yaml
  9. php为图片添加渐变背景,HTML_CSS实例:通过定义渐变边框给图片加阴影,一般我们可以使用背景图的方 - phpStudy...
  10. springboot导入项目依赖报错_最详细的 Spring Boot 多模块开发与排坑指南