当前位置:首页 > 单片机 > 单片机
[导读]这几天因为工作需要,移植了modbus RTU到STM32来,之前也听说过modbus,但是没有深入了解过,还以为会像usb 那样复杂的,经过这几天的折腾,发现真的太简单了。为了防止过段时间又忘记了怎么移植,在这里把移植过程记

 

这几天因为工作需要,移植了modbus RTU到STM32来,之前也听说过modbus,但是没有深入了解过,还以为会像usb 那样复杂的,经过这几天的折腾,发现真的太简单了。为了防止过段时间又忘记了怎么移植,在这里把移植过程记录下来,也为了方便初次接触modbus的人。

废话少说,首先去下载源码,我下载的是freemodbus-v1.5.0,解压后如图所示:

在demo文件夹中有很多移植好的例子,但是没有STM32的,不要紧,我们参考已有的例子来操作就可以了。在demo文件夹下新建一个文件夹,命名为STM32,将BARE文件夹里的文件全部复制过来。
然后,我们建立一个STM32的工程,我用的是mdk4.72,关于怎么建立工程我就不啰嗦了,在工程里添加modbus 和 port两个文件夹,并在文件夹里添加需要的文件,这些modbus的是在modbus文件夹下面,port的是在刚才新建的那个stm32文件夹下,port.c是我从别的地方弄过来的,里面就是一个开中断和一个关中断的函数,可以不要,我的工程如图:

乍一看,也有十几个文件,其实这些文件内容不多,很好理解,而且需要修改的只有port文件夹下的portserial.c 和 porttimer.c 。这两个文件里面有几个空函数,我们看名字就知道这些函数的作用了。
portserial.c如下:

[C]纯文本查看复制代码

?

010203040506070809101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263#include "port.h"/* ----------------------- Modbus includes ----------------------------------*/#include "mb.h"#include "mbport.h"/* ----------------------- static functions ---------------------------------*/staticvoidprvvUARTTxReadyISR(void);staticvoidprvvUARTRxISR(void);/* ----------------------- Start implementation -----------------------------*/voidvMBPortSerialEnable(BOOLxRxEnable,BOOLxTxEnable ){/* If xRXEnable enable serial receive interrupts. If xTxENable enable* transmitter empty interrupts.*/}BOOLxMBPortSerialInit(UCHARucPORT,ULONGulBaudRate,UCHARucDataBits, eMBParity eParity ){returnFALSE;}BOOLxMBPortSerialPutByte(CHARucByte ){/* Put a byte in the UARTs transmit buffer. This function is called* by the protocol stack if pxMBFrameCBTransmitterEmpty( ) has been* called. */returnTRUE;}BOOLxMBPortSerialGetByte(CHAR* pucByte ){/* Return the byte in the UARTs receive buffer. This function is called* by the protocol stack after pxMBFrameCBByteReceived( ) has been called.*/returnTRUE;}/* Create an interrupt handler for the transmit buffer empty interrupt* (or an equivalent) for your target processor. This function should then* call pxMBFrameCBTransmitterEmpty( ) which tells the protocol stack that* a new character can be sent. The protocol stack will then call* xMBPortSerialPutByte( ) to send the character.*/staticvoidprvvUARTTxReadyISR(void){pxMBFrameCBTransmitterEmpty( );}/* Create an interrupt handler for the receive interrupt for your target* processor. This function should then call pxMBFrameCBByteReceived( ). The* protocol stack will then call xMBPortSerialGetByte( ) to retrieve the* character.*/staticvoidprvvUARTRxISR(void){pxMBFrameCBByteReceived( );}

根据注释,可以知道vMBPortSerialEnable是串口发送和接收中断的控制的,包括发送中断和接收中断,在这里,我们用的是RXNE 和 TXE中断,代码如下:

[C]纯文本查看复制代码

?

0102030405060708091011121314151617181920voidvMBPortSerialEnable(BOOLxRxEnable,BOOLxTxEnable ){if(TRUE==xRxEnable){USART_ITConfig(USART1, USART_IT_RXNE, ENABLE);}else{USART_ITConfig(USART1, USART_IT_RXNE, DISABLE);}if(TRUE==xTxEnable){USART_ITConfig(USART1, USART_IT_TXE, ENABLE);}else{USART_ITConfig(USART1, USART_IT_TXE, DISABLE);}}

而xMBPortSerialInit函数显然是串口初始化的了,因为我在usart.c已经有一个串口初始化函数,这里直接调用该初始化函数usart_init(ulBaudRate);同时将return FALSE 改成 return TRUE; 注意这里我们只用了波特率这个参数,其他参数直接忽略,你也可以根据自己需要改一下。
然后xMBPortSerialPutByte 和xMBPortSerialGetByte 分别是发送和接收一个字节数据的函数,这里我直接调用库函数;

[C]纯文本查看复制代码

?

0102030405060708091011121314151617BOOLxMBPortSerialPutByte(CHARucByte ){USART_SendData(USART1, ucByte);while(USART_GetFlagStatus(USART1, USART_FLAG_TC) == RESET)/*????·????ê??*/{}returnTRUE;}BOOLxMBPortSerialGetByte(CHAR* pucByte ){*pucByte = USART_ReceiveData(USART1);returnTRUE;}

最后还有两个中断处理函数,把前面的static 去掉,因为我不想把我的串口中断函数放到这个文件。然后我们在stm32f10x_it.c添加串口中断函数,如下:

[C]纯文本查看复制代码

?

0102030405060708091011121314voidUSART1_IRQHandler(void){if(USART_GetITStatus(USART1, USART_IT_RXNE) == SET){prvvUARTRxISR();USART_ClearITPendingBit(USART1, USART_IT_RXNE);}if(USART_GetITStatus(USART1, USART_IT_TXE) == SET){prvvUARTTxReadyISR();// USART_ClearITPendingBit(USART1, USART_IT_TXE);}}

至此,portserial.c处理完毕。
porttimer.c的移植和portserial.c十分相似,但是要特别注意定时器中断的时间长度应该是3.5个字符时间,我这里只是简单粗暴的按照波特率是9600时候计算的。文件很短,直接上代码

[C]纯文本查看复制代码

?

01020304050607080910111213141516171819202122232425262728293031323334353637383940414243444546474849BOOLxMBPortTimersInit(USHORTusTim1Timerout50us ){TIM_TimeBaseInitTypeDef TIM_TimeBaseStructure;RCC_APB1PeriphClockCmd(RCC_APB1Periph_TIM2, ENABLE);TIM_DeInit(TIM2);#if 0TIM_TimeBaseStructure.TIM_Period = 0x7E54;//CLK==24MHz ((1000000000/9600)*11*3.5)/(1000/24) == 0x7e54TIM_TimeBaseStructure.TIM_Prescaler = 0x3;#endif// ?????????¤·?????????7200/72M = 0.0001,????100us????????1//10us x 50 = 5ms,??5ms????????TIM_TimeBaseStructure.TIM_Period = 50;TIM_TimeBaseStructure.TIM_Prescaler = (7200 - 1);TIM_TimeBaseStructure.TIM_ClockDivision = 0x0;TIM_TimeBaseStructure.TIM_CounterMode = TIM_CounterMode_Up;TIM_TimeBaseInit(TIM2, &TIM_TimeBaseStructure);// TIM_ITConfig(TIM2, TIM_IT_Update, ENABLE);TIM_Cmd(TIM2, ENABLE);returnTRUE;}voidvMBPortTimersEnable( ){TIM_ClearITPendingBit(TIM2, TIM_IT_Update);TIM_SetCounter(TIM2, 0);//TIM_Cmd(TIM2, ENABLE);TIM_ITConfig(TIM2, TIM_IT_Update, ENABLE);}voidvMBPortTimersDisable( ){TIM_SetCounter(TIM2, 0);//TIM_Cmd(TIM2, DISABLE);TIM_ITConfig(TIM2, TIM_IT_Update, DISABLE);}voidTIMERExpiredISR(void){(void)pxMBPortCBTimerExpired();}

同样,在stm32f10x_it.c添加定时器中断处理函数,

[C]纯文本查看复制代码

?

12345voidTIM2_IRQHandler(void){TIMERExpiredISR();TIM_ClearITPendingBit(TIM2, TIM_IT_Update);}


然后,我们还需要自己写四个回调函数,分别是读输入寄存器函数、读写保持寄存器函数、读写线圈函数和读离散寄存器函数,一般只用读写保持寄存器函数即可,具体怎么实现可以参考demo文件夹里面众多的demo.c文件。

 

 

本站声明: 本文章由作者或相关机构授权发布,目的在于传递更多信息,并不代表本站赞同其观点,本站亦不保证或承诺内容真实性等。需要转载请联系该专栏作者,如若文章内容侵犯您的权益,请及时联系本站删除。
换一批
延伸阅读

在嵌入式开发中,STM32的时钟系统因其灵活性和复杂性成为开发者关注的焦点。然而,看似简单的时钟配置背后,隐藏着诸多易被忽视的陷阱,轻则导致系统不稳定,重则引发硬件损坏。本文从时钟源选择、PLL配置、总线时钟分配等关键环...

关键字: STM32 时钟系统

在嵌入式系统开发中,STM32系列微控制器的内部温度传感器因其低成本、高集成度特性,广泛应用于设备自检、环境监测等场景。然而,受芯片工艺差异和电源噪声影响,其原始数据存在±1.5℃的固有误差。本文从硬件配置、校准算法、软...

关键字: STM32 温度传感器

在能源效率与智能化需求双重驱动下,AC-DC转换器的数字控制技术正经历从传统模拟方案向全数字架构的深刻变革。基于STM32微控制器的PFM(脉冲频率调制)+PWM(脉冲宽度调制)混合调制策略,结合动态电压调整(Dynam...

关键字: AC-DC STM32

当前智能家居产品需求不断增长 ,在这一背景下 ,对现有浇花装置缺陷进行了改进 ,设计出基于STM32单片机的全 自动家用浇花机器人。该设计主要由机械结构和控制系统构成 ,机械结构通过麦克纳姆轮底盘与喷洒装置的结合实现机器...

关键字: STM32 麦克纳姆轮 安全可靠 通过性强

用c++编程似乎是让你的Arduino项目起步的障碍吗?您想要一种更直观的微控制器编程方式吗?那你需要了解一下Visuino!这个图形化编程平台将复杂电子项目的创建变成了拖动和连接块的简单任务。在本文中,我们将带您完成使...

关键字: Visuino Arduino ESP32 STM32

基于STM32与LoRa技术的无线传感网络凭借其低功耗、广覆盖、抗干扰等特性,成为环境监测、工业自动化等场景的核心解决方案。然而,如何在复杂电磁环境中实现高效休眠调度与动态信道优化,成为提升网络能效与可靠性的关键挑战。本...

关键字: STM32 LoRa

在实时控制系统、高速通信协议处理及高精度数据采集等对时间敏感的应用场景中,中断响应延迟的优化直接决定了系统的可靠性与性能上限。STM32系列微控制器凭借其灵活的嵌套向量中断控制器(NVIC)、多通道直接内存访问(DMA)...

关键字: STM32 DMA

数字电源技术向高功率密度、高效率与高动态响应方向加速演进,STM32微控制器凭借其基于DSP库的算法加速能力与对LLC谐振变换器的精准控制架构,成为优化电源动态性能的核心平台。相较于传统模拟控制或通用型数字控制器,STM...

关键字: STM32 数字电源

STM32微控制器凭借其针对电机控制场景的深度优化,成为高精度、高可靠性驱动系统的核心选择。相较于通用型MCU,STM32在电机控制领域的核心优势集中体现在FOC(磁场定向控制)算法的硬件加速引擎与PWM死区时间的动态补...

关键字: STM32 电机控制

无线充电技术加速渗透消费电子与汽车电子领域,基于Qi协议的无线充电发射端开发成为智能设备能量补给的核心课题。传统模拟控制方案存在响应滞后、参数调整困难等问题,而基于STM32的数字PID控制结合FOD(Foreign O...

关键字: STM32 无线充电
关闭