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

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

1、何为RTU主站

Modbus协议是一个主从协议,那肯定就有主站和从站之分。所谓主站说的简单一点就是能够主动发起通讯的对象,所以主站就是发起通讯的一方。

对于RTU主站来说,自己并不会产生数据,而是要从从站获取数据。在Modbus RTU协议中从站不会主动向外发送数据,所以需要主站发送数据请求,从站才会向其返回请求的数据。这一过程如下图所示:

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

2、如何实现RTU主站

我们已经简单的说明了什么是RTU的主站,那么如何实现这一主站呢?其实在协议栈中,我们已经实现了主站的数据请求命令的合成以及响应数据的解析,所以我们使用协议栈时就是要控制何时将协议栈合成的主站请求命令发出以及如何解析数据响应进而得到想要的数据的过程。

在我们的协议栈中实现了0x01、0x02、0x03、0x04、0x05、0x06、0x0F以及0x10等功能码。也就是说主站对象可以生成面向这些功能码的从站数据请求。也可以解析面向这些功能码的从站数据响应。可以表示为下图所示:

从上图我们很清楚,协议栈已经实现了面向这些功能码的数据请求命令的生成以及数据响应消息的解析。我们使用协议栈时需要做的就是要告诉协议栈我要生成哪些数据请求命令以及如何解析数据响应消息。

2.1、怎么生成数据请求

对于数据请求,我们不一定需要面向全部功能码的请求,我们只需要根据我们的需求合成我们想要的请求。

在协议栈中,针对数据请求的生成我们定义了一个从站访问命令生成函数。该函数的原型如下:

uint16_t CreateAccessSlaveCommand(ObjAccessInfo objInfo,void *dataList,uint8_t *commandBytes)

该函数有3个参数,其中ObjAccessInfo objInfo为对象访问信息;void *dataList为数据列表指针,该参数主要用于写从站功能的命令生成;uint8_t *commandBytes为返回的从站访问命令。

ObjAccessInfo是一个结构体,向函数传递我们想要生成的从站访问命令的相关信息,包括站地址,功能码,起始地址和数量。该结构体的定义如下:

/*定义用于传递要访问从站(服务器)的信息*/
typedef struct{uint8_t unitID;FunctionCode functionCode;uint16_t startingAddress;uint16_t quantity;
}ObjAccessInfo;

2.2、怎么解析数据响应

对于数据响应,我们同样不需要考虑全部的操作码,我们一般需要考虑读请求的响应,因为他们的数据需要解析。而对于写请求返回数响应只是告诉主站成功或者不成功,即使不成功只需要在写一次就可以了,不存在数据更新的问题。

在协议栈中,我们实现了主站解析从站数据响应的解析函数。使用这一函数我们只需要将收到的数据响应报文传递给解析函数就可以完成解析。该函数的原型定义如下:

void ParsingSlaveRespondMessage(RTULocalMasterType *master,uint8_t *recievedMessage,uint8_t *command)

这个函数有3个参数,其中RTULocalMasterType *master为主站对象;uint8_t *recievedMessage为接收到的响应消息;uint8_t *command为发送的命令序列。将这几个参数传递给解析函数就可实现数据响应的解析。

RTULocalMasterType是一个结构体,用以生命一个主站对象,这个对象就是我们要实现各种操作的主站,这一结构体的定义如下:

/* 定义本地RTU主站对象类型 */
typedef struct LocalRTUMasterType{uint32_t flagWriteSlave[8];   //写一个站控制标志位,最多256个站,与站地址对应。uint16_t slaveNumber;         //从站列表中从站的数量uint16_t readOrder;           //当前从站在从站列表中的位置RTUAccessedSlaveType *pSlave;         //从站列表UpdateCoilStatusType pUpdateCoilStatus;       //更新线圈量函数UpdateInputStatusType pUpdateInputStatus;     //更新输入状态量函数UpdateHoldingRegisterType pUpdateHoldingRegister;     //更新保持寄存器量函数UpdateInputResgisterType pUpdateInputResgister;       //更新输入寄存器量函数
}RTULocalMasterType;

3RTU主站编码

有了前面的说明,我们基于协议栈实现一个主站应用就很容易了。接下来我们就基于协议栈具体实现一个主站应用。

3.1、定义主站对象

首先我们要声明一个主站对象,这是我们操作的基础。在接下来的各种操作中我们都是基于这一对象来实现的。具体操作如下:

RTULocalMasterType rtuMaster;

定义了这个主站对象后,我们还需要对这一对象进行初始化。协议栈同样提供了一个主站对象的初始化函数。函数的原型定义如下:

/*初始化RTU主站对象*/
void InitializeRTUMasterObject(RTULocalMasterType *master,uint16_t slaveNumber,RTUAccessedSlaveType *pSlave,UpdateCoilStatusType pUpdateCoilStatus,UpdateInputStatusType pUpdateInputStatus,UpdateHoldingRegisterType pUpdateHoldingRegister,UpdateInputResgisterType pUpdateInputResgister)

该函数的参数除了主站对象外,还有从站的数量即从站对象列表,还有四个数据更新函数指针。这几个函数指针将应用于数据响应的解析过程中,具体在后面描述。使用这一初始化函数实现对主站对象的初始化,使其能够实现各项操作,具体如下:

/*初始化RTU主站对象*/

  InitializeRTUMasterObject(&hgraMaster,2,hgraSlave,NULL,NULL,NULL,NULL);

这里我们将几个数据处理函数指针变量传入NULL,表示初始化为默认的操作函数,当然我们也可以编写这些函数,在后续的数据解析时将会详细说明。

3.2、生成数据请求

在前面,我们已经描述了数据请求命令的生成函数,该函数有一个ObjAccessInfo参数,这个参数用于传递需要生成命令的信息。这是一个结构体,我们需要定义一个对象变量。

ObjAccessInfo hgraInfo;

然后使用这个对象来实现数据请求的生成。具体操作如下所示:

  /* 生成1号从站访问命令 */hgraInfo.unitID=hgraSlave[0].stationAddress;hgraInfo.functionCode=ReadCoilStatus;hgraInfo.startingAddress=0x0000;hgraInfo.quantity=8;CreateAccessSlaveCommand(hgraInfo,NULL,slave1ReadCommand[0]);

生成的数据请求什么时候发送给完全由主进程来实现已经与协议栈没有关系了。

3.3、解析数据响应

收到数据响应后我们需要对其进行解析。前面我们已经介绍了解析从站数据响应的函数。具体的调用形式如下:

ParsingSlaveRespondMessage(&hgraMaster,hgraRxBuffer,NULL);

我们对hgraMaster主站对象收到的从站响应hgraRxBuffer进行解析。最后传入的NULL表示我们不指定主站发送的数据请求,而是让主站从请求列表中去自己查找。

当然我们需要实现数据更新处理回调函数。这几个函数是在对象初始化的时候以函数指针的形式传递的。原型如下:

/*更新读回来的线圈状态*/
__weak void UpdateCoilStatus(uint8_t salveAddress,uint16_t startAddress,uint16_t quantity,bool *stateValue)
{//在客户端(主站)应用中实现
}/*更新读回来的输入状态值*/
__weak void UpdateInputStatus(uint8_t salveAddress,uint16_t startAddress,uint16_t quantity,bool *stateValue)
{//在客户端(主站)应用中实现
}/*更新读回来的保持寄存器*/
__weak void UpdateHoldingRegister(uint8_t salveAddress,uint16_t startAddress,uint16_t quantity,uint16_t *registerValue)
{//在客户端(主站)应用中实现
}/*更新读回来的输入寄存器*/
__weak void UpdateInputResgister(uint8_t salveAddress,uint16_t startAddress,uint16_t quantity,uint16_t *registerValue)
{//在客户端(主站)应用中实现
}

我们可根据需要重定义这些函数,当然我们没有响应的数据可以不必实现,如我们没有使用输入寄存器,那么更新输入寄存器的回调函数则可以不用重定义。如下在我们的例子中重定义为:

/*更新读回来的保持寄存器*/void UpdateHoldingRegister(uint16_t startAddress,uint16_t quantity,uint16_t *registerValue)
{uint16_t startRegister=HoldingResterEndAddress+1;switch(salveAddress){case BPQStationAddress:       //更新读取的变频器参数{startRegister=36;break;}case PUMPStationAddress:      //更新蠕动泵{
//      aPara.phyPara.pumpRotateSpeed=registerValue[1];startRegister=HoldingResterEndAddress+1;break;}case JIG1StationAddress:      //更新摆臂小电机{startRegister=48;break;}case JIG2StationAddress:      //更新摆臂小电机{startRegister=52;break;}case JIG3StationAddress:      //更新摆臂小电机{startRegister=56;break;}case HLPStationAddress:       //更新红外温度{aPara.phyPara.hlpObjectTemperature=registerValue[0]/100.0;startRegister=HoldingResterEndAddress+1;break;}case ROL1StationAddress:      //更新摆臂控制{startRegister=quantity<3?60:62;break;}case ROL2StationAddress:      //更新摆臂控制{startRegister=quantity<3?70:72;break;}case ROL3StationAddress:      //更新摆臂控制{startRegister=quantity<3?80:82;break;}case DRUMStationAddress:      //更新滚筒电机{startRegister=quantity<3?90:92;break;}default:                      //故障态{startRegister=HoldingResterEndAddress+1;break;}}if(startRegister<=HoldingResterEndAddress){for(int i=0;i<quantity;i++){aPara.holdingRegister[startRegister+i]=registerValue[i];}}
}/*更新读回来的输入寄存器*/
void UpdateInputResgister(uint16_t startAddress,uint16_t quantity,uint16_t *registerValue)
{uint16_t startRegister=HoldingResterEndAddress+1;switch(salveAddress){case BPQStationAddress:       //更新读取的变频器参数{startRegister=HoldingResterEndAddress+1;break;}case PUMPStationAddress:      //更新蠕动泵{//aPara.phyPara.pumpRotateSpeed=registerValue[1]; //第一版背板aPara.phyPara.pumpRotateSpeed=(uint16_t)((float)registerValue[1]*6.0/128.0+0.5); //第二版背板startRegister=HoldingResterEndAddress+1;break;}case JIG1StationAddress:      //更新摆臂小电机{startRegister=HoldingResterEndAddress+1;break;}case JIG2StationAddress:      //更新摆臂小电机{startRegister=HoldingResterEndAddress+1;break;}case JIG3StationAddress:      //更新摆臂小电机{startRegister=HoldingResterEndAddress+1;break;}case ROL1StationAddress:      //更新摆臂控制{startRegister=HoldingResterEndAddress+1;break;}case ROL2StationAddress:      //更新摆臂控制{startRegister=HoldingResterEndAddress+1;break;}case ROL3StationAddress:      //更新摆臂控制{startRegister=HoldingResterEndAddress+1;break;}case DRUMStationAddress:      //更新滚筒电机{startRegister=HoldingResterEndAddress+1;break;}default:                      //故障态{startRegister=HoldingResterEndAddress+1;break;}}if(startRegister<=HoldingResterEndAddress){for(int i=0;i<quantity;i++){aPara.holdingRegister[startRegister+i]=registerValue[i];}}
}

4RTU主站小结

我们实现了这个RTU主站实例,我们可以使用如Modsim这样的软件在PC上模拟Modbus RTU从站来测试这个主站应用,操作结果是没有问题的。

在使用协议栈实现RTU主站时需要注意,协议栈支持在同一设备上以不同的通讯端口实现不同的主站应用,而且每一台主站都支持多个从站。具体实现只需要根据协议栈定义就可以了。

我们来总结一下使用协议栈实现主站应用的步骤,以方便大家使用协议栈实现Modbus RTU主站应用。

第一步,使用主站对象类型声明一个主站对象。然后对这个主站对象进行初始化。初始化主站对象时。需要指定从站数量,从站列表以及更新数据的回调函数指针。

第二步,生成访问从站的数据请求列表。这个数据请求列表是按每一台从站来划分的,将列表的指针存在对应的从站对象中。然后在需要的时候发送相应的数据请求。

第三步,解析接收的从站数据响应。协议栈已经定义好了解析函数,只需传入消息就可自动解析。但是更新数据的回调函数必须根据具体的变量来编写。可以每台主站独立编写也可使用默认的函数。不过建议每台主站独立编写,这样比较清晰。

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

欢迎关注:

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/499362.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

uCOS-III应用开发笔记之一:uCOS-III在STM32的移植

uCOS-III实时操作系统在MCU平台被广泛使用&#xff0c;在这里我们将简单的记录如何将uCOS-III实时操作系统移植到目标平台上并运行。 1、必要的准备 在开始uCOS-III实时操作系统的移植前&#xff0c;我们还需要做一些必要的准备&#xff0c;如确定目标板、准备目标工程及uCOS…

外设驱动库开发笔记10:SHT2x系列温湿度传感器驱动

温湿度检测是嵌入式编程中经常应用到的一项功能。在我们的产品中亦经常使用。SHT2x系列温湿度传感器作为一种高精度低成本的集成模块&#xff0c;一直应用于我们的产品中。在这里我们讨论如何封装SHT2x系列温湿度传感器的驱动。 1、功能概述 SHT20配有一个全新设计的CMOSens芯…

Modbus协议栈应用实例之二:Modbus RTU从站应用

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

外设驱动库开发笔记11:SHT3x系列温湿度传感器驱动

在我们的产品中经常会遇到温湿度检测的需求。可以用于检测温湿度的传感器元件也有很多。我们经常使用的SHT各系列数字温湿度传感器来实现应用需求。在这里我们将设计并实现SHT3x系列温湿度传感器的驱动。 1、功能概述 SHT3x系列温湿度传感器是适用于各种应用的高品质湿度传感…

LwIP应用开发笔记之十:LwIP带操作系统基本移植

现在&#xff0c;TCP/IP协议的应用无处不在。随着物联网的火爆&#xff0c;嵌入式领域使用TCP/IP协议进行通讯也越来越广泛。在我们的相关产品中&#xff0c;也都有应用&#xff0c;所以我们结合应用实际对相关应用作相应的总结。 1、技术准备 我们采用的开发平台是STM32F407…

ThreadX应用开发笔记之一:移植ThreadX到STM32平台

现在一些小型系统中也往往有多任务处理的需求&#xff0c;这就为实时操作系统提供了用武之地。事实上国内外各种各样的RTOS有很多&#xff0c;而且基本都在走开源的路线&#xff0c;ThreadX也不例外&#xff0c;在这一篇中我们就来学习ThreadX初步应用并将其移植到STM32平台中。…

外设驱动库开发笔记12:TSEV01CL55红外温度传感器驱动

有时候我们需要检测一些无法直接接触的器件的温度。为了实现这一需求&#xff0c;我们通常会选择红外温度传感器来实现这一功能。考虑到复用的问题&#xff0c;我们一般会将操作元器件的代码抽象为驱动函数以备调用。这里我们就来设计并实现TSEV01CL55红外温度传感器的驱动。 …

FreeRTOS应用开发笔记之一:FreeRTOS在STM32的移植

FreeRTOS是如今在小型嵌入式领域应用比较广泛的一种实时操作系统。它是一种开源且免费的操作系统&#xff0c;而且移植和使用都非常的简单。在这里我们将学习并移植FreeRTOS。 1、必要的准备 工欲善其事&#xff0c;必先利其器&#xff0c;在开始学习和移植之前&#xff0c;相…

外设驱动库开发笔记13:MLX90614红外温度传感器驱动

红外温度传感器一般用于非接触式的温度检测。在我们的系统中经常会有这样的需求。所以我们将其设计为通用的驱动库以备复用。这一篇我们将讲述MLX90614红外温度传感器驱动的设计与实现。 1、功能概述 MLX90614是一种红外温度计&#xff0c;用于非接触式温度测量。红外测温是根…

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

自从开源了我们自己开发的Modbus协议栈之后&#xff0c;有很多朋友建议我针对性的做几个示例。所以我们就基于平时我们的应用整理了几个简单但可以说明基本的应用方法的示例&#xff0c;这一篇中我们将解说如何使用协议栈实现一个Modbus TCP客户端。 1、何为TCP客户端 Modbus…

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

自从开源了我们自己开发的Modbus协议栈之后&#xff0c;有很多朋友建议我针对性的做几个示例。所以我们就基于平时我们的应用整理了几个简单但可以说明基本的应用方法的示例&#xff0c;这一篇中我们来简述如何使用协议栈实现一个Modbus TCP服务器应用。 1、何为TCP服务器 Mo…

外设驱动库开发笔记14:DS18B20温度变送器驱动

在一时候我们需要相对简单的检测温度信号&#xff0c;而DS18B20就是一款功能和应用都相对简单的温度传感器&#xff0c;通过单线就可以实现检测温度信号的需求。这一篇我们就来实现操作DS18B20获取温度数据的驱动。 1、功能概述 DS18B20是常用的数字温度传感器&#xff0c;其…

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

自从开源了我们自己开发的Modbus协议栈之后&#xff0c;有很多朋友建议我针对性的做几个示例。所以我们就基于平时我们的应用整理了几个简单但可以说明基本的应用方法的示例&#xff0c;这一篇中我们来使用协议栈实现Modbus ASCII主站应用。 1、何为ASCII主站 我们知道Modbus…

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

自从开源了我们自己开发的Modbus协议栈之后&#xff0c;有很多朋友建议我针对性的做几个示例。所以我们就基于平时我们的应用整理了几个简单但可以说明基本的应用方法的示例&#xff0c;这一篇中我们来使用协议栈实现Modbus ASCII从站应用。 1、何为ASCII从站 我们知道Modbus…

外设驱动库开发笔记15:DHT11温湿度传感器驱动

与DS18B20一样DHT11也是采用单总线&#xff0c;但所不同的是DHT11可同时实现温度和湿度的检测。在我们的产品中经常使用它来检测环境的温湿度信息。这一篇我们将设计并封装DHT11的驱动程序&#xff0c;以方便重复使用。 1、功能概述 DHT11数字温湿度传感器是一款含有已校准数…

外设驱动库开发笔记16:MS5536C压力变送器驱动

压力检测也是经常会遇到的需求&#xff0c;比如环境压力或者低压气体等都会用到压力检测。这类检测压力都比较低&#xff0c;一般不会超过大气压&#xff0c;有时甚至是负压。这一篇我们要讨论的MS5536C就属于这类器件。接下来我们将设计并实现MS5536C的驱动。 1、功能概述 M…

外设驱动库开发笔记17:MS5803压力变送器驱动

很多时候我们需要检测被控对象的绝对压力&#xff0c;而且在我们的多款产品中也有这样的需求。当然检测绝对压力的传感器有很多&#xff0c;我们经常使用MS5803来实现压力检测。本篇中我们将设计并实现MS5803系列压力传感器的驱动。 1、功能概述 MS5803系列产品包含压阻传感器…

通讯接口应用笔记1:RS485通讯上下拉电阻的选择

RS485是一种常见的通讯接口方式&#xff0c;在我们的实际产品中也是多次使用。但我们平常并不会去过多考虑某一实现的细节问题&#xff0c;不过最近我们遇到了一个因如上下拉电阻的选择问题而造成的通讯故障&#xff0c;所以在这一片中我们来讨论一下RS485总线上下拉电阻的选择…

外设驱动库开发笔记18:MS5837压力变送器驱动

绝对压力的检测是常见的需求。在我们的系统中也常常会遇到。而MS5837压力传感器也是我们进场会采用的方案。在这篇里我们将讨论并实现MS5837压力传感器的驱动。 1、功能概述 MS5837压力传感器是一种可用于电路板上&#xff0c;适用于检测10-1200mbar压力范围的传感器&#xf…

外设驱动库开发笔记19:BMP280压力温度传感器驱动

压力和温度监测在嵌入式系统开发中是非常常见的需求&#xff0c;特别是对环境大气压力和温度的检测需求就更常见了。我们一般都会选择一些封装较小操作比较方便的压力传感器。BMP280就是满足这一要求的器件。在这一篇中我们将设计并实现BMP280的驱动。 1、功能概述 BMP280是一…