基于STM32以及modbus——RTU的从机程序(STM32作为从机)
创始人
2025-01-07 18:39:05
0

1.前言

        因项目中需要用到485电路以及多设备通讯,采用Modbus协议通讯方式,本文写的目的就是记录笔记也提供给初学者一点参考。里面的内容可能会有错误,仅供参考。

2.485电路

上图是项目中的电路,也算一个最基本的485电路,没什么好讲的,这个博主讲的不错,可以参考他的。

终于讲透了,史上最详细的RS485自动收发电路你一定要掌握-CSDN博客icon-default.png?t=N7T8https://blog.csdn.net/qq_39400113/article/details/122387133?ops_request_misc=%257B%2522request%255Fid%2522%253A%2522171393570316800227442090%2522%252C%2522scm%2522%253A%252220140713.130102334..%2522%257D&request_id=171393570316800227442090&biz_id=0&utm_medium=distribute.pc_search_result.none-task-blog-2~blog~top_positive~default-2-122387133-null-null.nonecase&utm_term=485%E7%94%B5%E8%B7%AF&spm=1018.2226.3001.4450

3.Modbus协议

学习以前参考以下博文链接:

STM32+RS485+Modbus-RTU(主机模式+从机模式)-标准库/HAL库开发_stm32modbus主机-CSDN博客icon-default.png?t=N7T8https://blog.csdn.net/qq_37281984/article/details/122739968

3.1 Modbus协议简介

Modbus协议是一种应用层报文传输协议,协议本身并没有定义物理层,所以支持多种电气接口,直接可以理解成他是软件层面的,各种的电气接口比如RS232、RS485、TCP/IP等,他们是硬件层面。他们之间互不影响。

3.2 Modbus通讯过程

Modbus是一主多从的通信协议

Modbus通信中只有一个设备可以发送请求。其他从设备接收主机发送的数据来进行响应,从机是任何外围设备,如I/O传感器,阀门,网络驱动器,或其他测量类型的设备。从站处理信息和使用Modbus将其数据发送给主站。

也就是说,不能Modbus同步进行通信,主机在同一时间内只能向一个从机发送请求,总线上每次只有一个数据进行传输,即主机发送,从机应答,主机不发送,总线上就没有数据通信。

从机不会自己发送消息给主站,只能回复从主机发送的消息请求。

3.3  Modbus功能码

Modbus协议同时规定了二十几种功能码,但是常用的只有3种,用于对存储区的读写,如下表所示:

 我们主要就是用到03h读取从机寄存器的数据,06h对从机指定寄存器写入指定数据,10h对从机多个寄存器写入数据。

3.4 Modbus协议格式

我们主要就是学习它的协议格式,主要用到3种功能码,也就对应与3种发送数据的格式。

3.4.1 03H功能码-读取数据-协议格式

比如 :主机发送 01 03 00 01 00 01 D5 CA

主机一共发送8个字节。

0x01:表示主机要与从机地址是0x01的设备进行通讯

0x03:功能码,代表我们发送这个指令的作用是什么,03表示我们要读取从机的数据

0x00:要读取从机寄存器地址的高位

0x01:要读取从机寄存器地址的低位

0x00:要读取从机寄存器数量的高位

0x01:要读取从机寄存器数量的低位

D5:前6位数据效验的低位

CA:前6位数据效验的高位

总得来说这段代码的含义是:查询从机地址为0x01的0x0001寄存器地址的0x0001个数据。

从机收到这段协议后,应回复如下格式 01 03 02 00 03 F8 48

从机一共发送7个字节

0x01:表示主机要与从机地址是0x01的设备进行通讯

0x03:功能码,代表我们发送这个指令的作用是什么,03表示我们要读取从机的数据

0x02:返回的数据个数(要读取的寄存器个数*2)——>返回数据的字节都是寄存器的2倍

0x00:从机返回数据的高位

0x03:从机返回数据的低位

F8 48:前面几位数据的效验码

3.4.2 06H功能码-写入数据-单寄存器-协议格式

比如 :主机发送 01 06 00 01 00 17 98 04

主机一共发送8个字节。

0x01:表示主机要与从机地址是0x01的设备进行通讯

0x03:功能码,代表我们发送这个指令的作用是什么,06表示我们要向从机写入数据

0x00:要写入从机寄存器地址的高位

0x01:要写入从机寄存器地址的低位

0x00:要写入从机数据的高位

0x17:要写入从机数据的低位

98 04:前6位数据效验码

总得来说这段代码的含义是:向从机地址为0x01的0x0001地址写入数据0x0017

从机回复格式:01 06 00 01 00 17

从机回复的内容和主机发送的一样。

3.4.3 10H功能码-写入数据-多个寄存器-协议格式

比如 :主机发送 : 01 10 00 05 00 02 04 01 02 03 04 92 9F

主机一共发送11个字节。

0x01:表示主机要与从机地址是0x01的设备进行通讯

0x10:功能码,10表示我们要向从机多个寄存器写入数据

0x00:要写入从机寄存器地址的起始地址的高位

0x05:要写入从机寄存器地址的起始地址的低位

0x00:要写入寄存器个数的高位

0x02:要写入寄存器个数的低位

0x04:要写入的字节数

01 02:写入第一个寄存器的数据

03 04:写入第二个寄存器的数据

92 9F:前面数据效验码

从机回复:01 10 00 05 00 02 51 C9

从机返回的数据可以理解为就是主机发送的前6个字节加上自己数据的效验码。表面从机01从地址0x0005开始写入2个寄存器数据。

到这样发送数据协议格式介绍完毕。

4.STM32+485电路+Modbus协议代码

Modbus只是个协议,485电路时硬件层面,通讯还是串口通讯。

连线就是485的A B连接485转USB的A B脚,485芯片的连接如上面电路所示,连接STM32串口1。

4.1 STM32串口配置

4.1.1 串口初始化

void Serial_Init(void) { 	RCC_APB2PeriphClockCmd(RCC_APB2Periph_USART1, ENABLE); 	RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOA, ENABLE); 	 	GPIO_InitTypeDef GPIO_InitStructure; 	GPIO_InitStructure.GPIO_Mode = GPIO_Mode_AF_PP; 	GPIO_InitStructure.GPIO_Pin = GPIO_Pin_9; 	GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz; 	GPIO_Init(GPIOA, &GPIO_InitStructure); 	 	GPIO_InitStructure.GPIO_Mode = GPIO_Mode_IPU; 	GPIO_InitStructure.GPIO_Pin = GPIO_Pin_10; 	GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz; 	GPIO_Init(GPIOA, &GPIO_InitStructure); 	 	USART_InitTypeDef USART_InitStructure; 	USART_InitStructure.USART_BaudRate = 9600; 	USART_InitStructure.USART_HardwareFlowControl = USART_HardwareFlowControl_None; 	USART_InitStructure.USART_Mode = USART_Mode_Tx | USART_Mode_Rx; 	USART_InitStructure.USART_Parity = USART_Parity_No; 	USART_InitStructure.USART_StopBits = USART_StopBits_1; 	USART_InitStructure.USART_WordLength = USART_WordLength_8b; 	USART_Init(USART1, &USART_InitStructure); 	 	USART_ITConfig(USART1, USART_IT_RXNE, ENABLE); 	 	NVIC_PriorityGroupConfig(NVIC_PriorityGroup_2); 	 	NVIC_InitTypeDef NVIC_InitStructure; 	NVIC_InitStructure.NVIC_IRQChannel = USART1_IRQn; 	NVIC_InitStructure.NVIC_IRQChannelCmd = ENABLE; 	NVIC_InitStructure.NVIC_IRQChannelPreemptionPriority = 1; 	NVIC_InitStructure.NVIC_IRQChannelSubPriority = 1; 	NVIC_Init(&NVIC_InitStructure); 	 	USART_Cmd(USART1, ENABLE); }

4.1.2 串口发送封装代码

我们发送数据的格式主要就是以数组的方式。

void Serial_SendByte(uint8_t Byte) { 	USART_SendData(USART1, Byte); 	while (USART_GetFlagStatus(USART1, USART_FLAG_TXE) == RESET); }  void Serial_SendArray(uint8_t *Array, uint16_t Length) { 	uint16_t i; 	for (i = 0; i < Length; i ++) 	{ 		Serial_SendByte(Array[i]); 	} }

4.1.3 串口中断接收

主要是将接收到的数据依次存放到对应的数组中,当modbus.reflag==1表示还有数据正在处理中,反之则进行数据存储,当开始存储第二个数据时开启定时器计时,主要目的判断接收数据是否完毕,如果超过一段时间没有数据,则表明这一次数据接收完毕

void USART1_IRQHandler(void) { 	u8 st,Res;  	st = USART_GetITStatus(USART1, USART_IT_RXNE);  	if(st == SET)//接收中断  	{  		Res =USART_ReceiveData(USART1);	//读取接收到的数据 		  	   	 if( modbus.reflag==1)  //有数据包正在处理  	  {  		   return ;  		}		  	  modbus.rcbuf[modbus.recount++] = Res; 		//USART_SendData(USART2, Res);//接受到数据之后返回给串口1  		modbus.timout = 0;  		if(modbus.recount == 1)  //已经收到了第二个字符数据  		{  		  modbus.timrun = 1;  //开启modbus定时器计时  		}  	}	 }

4.2定时器2配置

4.2.1 定时器初始化

配置定时器1ms进入一次中断。这里主要说明下1ms怎么配置的。

定时时间 = [(PSC +1) *(ARR+1)]/72M

这里的PSC指定就是分频倍数,ARR是溢出数,72M是时钟频率。

void Timer_Init(void) { 	RCC_APB1PeriphClockCmd(RCC_APB1Periph_TIM2, ENABLE); 	 	TIM_InternalClockConfig(TIM2); 	 	TIM_TimeBaseInitTypeDef TIM_TimeBaseInitStructure; 	TIM_TimeBaseInitStructure.TIM_ClockDivision = TIM_CKD_DIV1; 	TIM_TimeBaseInitStructure.TIM_CounterMode = TIM_CounterMode_Up; 	TIM_TimeBaseInitStructure.TIM_Period = 1000 - 1; 	TIM_TimeBaseInitStructure.TIM_Prescaler = 72 - 1; //定时1ms 	TIM_TimeBaseInitStructure.TIM_RepetitionCounter = 0; 	TIM_TimeBaseInit(TIM2, &TIM_TimeBaseInitStructure); 	 	TIM_ClearFlag(TIM2, TIM_FLAG_Update); 	TIM_ITConfig(TIM2, TIM_IT_Update, ENABLE); 	 	NVIC_PriorityGroupConfig(NVIC_PriorityGroup_2); 	 	NVIC_InitTypeDef NVIC_InitStructure; 	NVIC_InitStructure.NVIC_IRQChannel = TIM2_IRQn; 	NVIC_InitStructure.NVIC_IRQChannelCmd = ENABLE; 	NVIC_InitStructure.NVIC_IRQChannelPreemptionPriority = 2; 	NVIC_InitStructure.NVIC_IRQChannelSubPriority = 1; 	NVIC_Init(&NVIC_InitStructure); 	 	TIM_Cmd(TIM2, ENABLE); }

4.2.2 定时中断

定时器设置1ms进入中断1次,运行时间不为0的情况下开始计时,超过8ms则表明这一次接收数据完毕,将数据接收结束标志位置1处理(modbus.reflag = 1),当数据接收完毕,则STM32可以对接收到的数据进行数据分析和处理执行相应的操作了

void TIM2_IRQHandler(void) { 	if (TIM_GetITStatus(TIM2, TIM_IT_Update) == SET) 	{ 		if(modbus.timrun != 0)//运行时间!=0表明  		 {  		  modbus.timout++;  		  if(modbus.timout >=8)  		  {  		   modbus.timrun = 0;  			 modbus.reflag = 1;//接收数据完毕  		  }  			  		 } 		TIM_ClearITPendingBit(TIM2, TIM_IT_Update); 	 	} }

4.3 Modbus协议处理代码

4.3.1 Modbus数据结构体

STM32作为从机时需要用到的数据。

typedef struct   {  	//作为从机时使用   u8  myadd;        //本设备从机地址  	u8  rcbuf[100];   //modbus接受缓冲区  	u8  timout;       //modbus数据持续时间  	u8  recount;      //modbus端口接收到的数据个数  	u8  timrun;       //modbus定时器是否计时标志  	u8  reflag;       //modbus一帧数据接受完成标志位  	u8  sendbuf[100]; //modbus接发送缓冲区  	  }MODBUS;

4.3.2 功能码03H处理函数

void Modbus_Func3(void)  {  	uint16_t Regadd,Reglen;  	uint8_t i,j;	  	//得到要读取寄存器的首地址  	Regadd = modbus.rcbuf[2]*256+modbus.rcbuf[3];//读取的首地址  	//得到要读取寄存器的数据长度  	Reglen = modbus.rcbuf[4]*256+modbus.rcbuf[5];//读取的寄存器个数  	//发送回应数据包  	i = 0;  	modbus.sendbuf[i++] = modbus.myadd;      //ID号:发送本机设备地址  	modbus.sendbuf[i++] = 0x03;              //发送功能码 	modbus.sendbuf[i++] = ((Reglen*2)%256);   //返回字节个数  	for(j=0;j
 4.3.2.1 实验过程1

串口助手发送: 01 03 00 03 00 01 串口助手会自动计算前面的效验值,一起发送给STM32从机。查询寄存器地址为0x0003的1个数据 。

 

从机返回的是 :2个字节,指定地址的数据为0x0004。数据返回没问题

现在查询2个数据,发送 01 03 00 03 00 02 

从机返回的是 :4个字节,指定地址的数据为0x0004和0x0005,数据返回没问题

4.3.3 功能码06H处理函数

void Modbus_Func6(void) { 	u16 Regadd;//地址16位  	u16 val;//值  	u16 i;  	i=0; 	Regadd=modbus.rcbuf[2]*256+modbus.rcbuf[3];  //得到要修改的地址   	val=modbus.rcbuf[4]*256+modbus.rcbuf[5];     //修改后的值(要写入的数据)  	Reg[Regadd]=val;  //修改本设备相应的寄存器  	  	//以下为回应主机  	modbus.sendbuf[i++]=modbus.myadd;//本设备地址 	modbus.sendbuf[i++]=0x06;        //功能码  	modbus.sendbuf[i++]=Regadd/256;//写入的地址  	modbus.sendbuf[i++]=Regadd%256;  	modbus.sendbuf[i++]=val/256;//写入的数值  	modbus.sendbuf[i++]=val%256;  	Modbus_CRC16(modbus.sendbuf,i);//获取crc校验位    //crc校验位加入包中 	modbus.sendbuf[i++] = Modbus_RCR[0]; 	modbus.sendbuf[i++] = Modbus_RCR[1];   	//数据发送包打包完毕   	Serial_SendArray(modbus.sendbuf,i);//发送从机数据 }
 4.3.3.1 实验过程2

发送前我们先查询0x0001的本来的数据为多少,再进行写入,写入后再查询是否写入成功。

串口助手首先发送:01 03 00 01 00 01 

串口助手再发送:01 06 00 01 00 06 

串口助手最后发送:01 03 00 01 00 01 

 首先回复是0x0001寄存器的数据为 0x0002

其次修改0x0001寄存器数据为0x0006

最后查询0x0001寄存器的数据就是为0x0006,表明数据写入成功。

4.3.4 功能码10H处理函数

//这是往多个寄存器器中写入数据  //功能码0x10指令即十进制16 void Modbus_Func16(void) { 	uint16_t Regadd;//地址16位 	uint16_t Reglen; 	uint16_t i; 	Regadd=	modbus.rcbuf[2]*256+modbus.rcbuf[3];  //要修改内容的起始地址  	Reglen = modbus.rcbuf[4]*256+modbus.rcbuf[5];//读取的寄存器个数 	 	for(i=0;i
 4.3.4.1 实验过程3

发送前我们先查询0x0001寄存器地址的2个数据为多少,再进行写入,写入后再查询是否写入成功。

串口助手首先发送:01 03 00 01 00 02 

串口助手再发送:01 10 00 01 00 02 04  00 08 00 90 

串口助手最后发送:01 03 00 01 00 02 

 首先回复是0x0001起始寄存器的数据为 0x0002, 0x0003

其次修改0x0001起始寄存器数据为0x0008,0x0009

最后查询0x0001起始寄存器的数据就是为0x0008,0x0009,表明数据写入成功。

4.3.5 效验函数

随便百度的一个。

unsigned int Modbus_CRC16(unsigned char *data, unsigned int len) {     unsigned int i, j, tmp, CRC16;      CRC16 = 0xFFFF;             //CRC寄存器初始值     for (i = 0; i < len; i++)     {         CRC16 ^= data[i];         for (j = 0; j < 8; j++)         {             tmp = (unsigned int)(CRC16 & 0x0001);             CRC16 >>= 1;             if (tmp == 1)             {                 CRC16 ^= 0xA001;    //异或多项式             }         }     } 	//低位在前 		Modbus_RCR[0] = (unsigned char) (CRC16 & 0x00FF); 		Modbus_RCR[1] = (unsigned char) ((CRC16 & 0xFF00)>>8);     return CRC16; }

4.3.6 事件处理函数

只有到数据接收完成才能进行数据处理

  1. 首先判断自主计算的CRC校验位和接收到数据的校验位是否一致
  2. 其次判断从机地址是不是自己的地址
  3. 数据传输正确且从机地址正确的情况下再根据不同的功能码去执行对应的函数操作
void Modbus_Event(void)  {  	//uint16_t crc,rccrc;//crc和接收到的crc  	//没有收到数据包    if(modbus.reflag == 0)  //如果接收未完成则返回空  	{ 			  	   return;  	}  	//收到数据包(接收完成)  	//通过读到的数据帧计算CRC  	//参数1是数组首地址,参数2是要计算的长度(除了CRC校验位其余全算)  	Modbus_CRC16(modbus.rcbuf,modbus.recount-2); //获取CRC校验位  	// 读取数据帧的CRC 	//printf("%x","%x",(crc & 0x00FF),((crc & 0xFF00)>>8));   	//rccrc = modbus.rcbuf[modbus.recount-2]*256+modbus.rcbuf[modbus.recount-1];//计算读取的CRC校验位 	  	//等价于下面这条语句  	//rccrc=modbus.rcbuf[modbus.recount-1]|(((u16)modbus.rcbuf[modbus.recount-2])<<8);//获取接收到的CRC  	if(Modbus_RCR[0] == modbus.rcbuf[modbus.recount-2] && Modbus_RCR[1] == modbus.rcbuf[modbus.recount-1]) //CRC检验成功 开始分析包  	{	  	   if(modbus.rcbuf[0] == modbus.myadd)  // 检查地址是否时自己的地址  		 {  		   switch(modbus.rcbuf[1])   //分析modbus功能码  			 {  			   case 0:             break;  				 case 1:             break;  				 case 2:             break;  				 case 3:      Modbus_Func3();      break;//这是读取寄存器的数据  				 case 4:             break;  				 case 5:             break; 				case 6:      Modbus_Func6();      break;//这是写入单个寄存器数据  				 case 7:             break;  				 case 8:             break;  				 case 9:             break;  				 case 16:     Modbus_Func16(); 			break;//写入多个寄存器数据  			 }  		 }  		 else if(modbus.rcbuf[0] == 0) //广播地址不予回应  		 {  		      		 }	   	}	 		modbus.recount = 0;//接收计数清零 		modbus.reflag = 0; //接收标志清零  }

 4.3.7 Modbus初始化

定义一个Reg数组,来模拟STM32作为从机寄存器的数据。

将STM32作为从机的地址为0x01

uint16_t  Reg[] = { 	0x0001, 	0x0002, 	0x0003, 	0x0004, 	0x0005, };  void Modbus_Init(void)  { 	modbus.myadd = 0x01; //从机设备地址为 	modbus.timrun = 0;    //modbus定时器停止计算  }

4.4 main函数

int main(void) { 	 	Serial_Init(); 	Timer_Init(); 	Modbus_Init(); 	 	while (1) 	{ 		Modbus_Event(); 	} }

5.总结

到此,STM32作为从机的代码写完了,也验证成功。下一部分将记录STM32作为主机给另外的一个STM32作为从机进行通讯。以上内容仅供参考。

相关内容

热门资讯

3分钟了解微扑克透明挂原来真的... 3分钟了解微扑克透明挂原来真的是有挂,太嚣张了原来真的有挂,详细教程(有挂秘诀);德扑锦标赛是一项微...
正则表达式 文章目录正则表达式Java中常用的正则RegexUtils正则表达式正则表达式(Reg...
黑科技攻略Wepoke机器人软... 黑科技攻略Wepoke机器人软件透明挂!(辅助挂)太嚣张了原来确实真的是有挂的(2022已更新)(哔...
信息共享《德州wpk透视辅助》... 信息共享《德州wpk透视辅助》外挂辅助挂脚本(2020已更新)(哔哩哔哩);1、不需要AI权限,帮助...
【C++】C语言和C++的区别 目录面向过程 vs 面向对象类和对象继承多态模板异常处理标准库内存管理命名空间运算符重载构造函数和析...
《人生苦短,我用python·... 基本异常处理 Python 提供了 try…except 语句来处理异常。其基本结构如下࿱...
基于搜索二叉树的停车收费管理系... 系统效果:录入汽车信息查看汽车信息收费信息查看查询车库车辆 代码展示: ...
玩家必看攻略!(WPK小程序)... 玩家必看攻略!(WPK小程序)透视辅助!(透视)外挂辅助工具(2025已更新)(哔哩哔哩);1、这是...
安全防御---防火墙实验1 安全防御—防火墙实验1一、实验拓扑与要求要求:1、DMZ区内的服务器,办...
揭秘真相wpk安卓版本原来是真... 揭秘真相wpk安卓版本原来是真的有挂,太坑了其实确实真有挂(有挂技术)-哔哩哔哩;致您一封信;亲爱w...