Modbus RTU驱动程序开发指引



0?wx_fmt=gif

Modbus RTU
驱动开发

摘要

这篇笔记主要介绍基于飞思卡尔kv4x系列MCU的modbus RTU(Remote Terminal Unit)驱动程序开发,以搭载飞思卡尔kv46MCU的飞思卡尔官方塔式系统开发板为例,阅读这篇文章前,基于你已经了解modbus协议要求,以及数据帧格式。

UART

UART(Universal Asynchronous Receiver Transmitter)通用收发器,现在基本上所有的MCU都会含有UART模块,有的甚至不止一路UART,今天我们要讲的飞思卡尔Kv46MCU就含有两路UART模块。

飞思卡尔kv4xMCU除含有UART的基本功能外,还包含支持RS-485操作,硬件流控制,数据端可配置,支持FIFO(Fisrt In First Out),等功能,更多功能介绍可以参阅官方参考手册描述,这里不再赘述。

Kv46MCU的UART模块比51单片机的UART要复杂的多,操作的时候要按照手册上的要求,仔细配置寄存器。

0?wx_fmt=jpeg

从框图中可以看出,发送器主要包含数据寄存器,波特率产生,发送移位寄存器,发送控制,校验,中断等子模块构成。

0?wx_fmt=jpeg

从框图可以看出,接收器主要包含数据缓冲区,波特率生成,移位寄存器,接收控制,中断,接收检测等子功能块。

在了解了kv46MCU的UART构成后,开发标准Midbus RTU的驱动程序主要的工作就是如何根据Mosbud标准要求来驱动UART模块,准确的检测通信帧。最主要的就是帧间大于3.5个字符的时间间隔,字节之间的时间间隔不能超过1.5个字节间隔,否则为错误帧。

ge:ZH-CN'>操作,硬件流控制,数据端可配置,支持FIFO(Fisrt In First Out),等功能,更多功能介绍可以参阅官方参考手册描述,这里不再赘述。

Kv46MCU的UART模块比51单片机的UART要复杂的多,操作的时候要按照手册上的要求,仔细配置寄存器。

硬件平台

任何含有kv46MCU的支持UART通信的硬件板件都可以,本文以飞思卡尔官方塔式系统为例,塔式系统应用的时候要注意,它是将MCU的UART1通过USB输出的,所以要按照硬件原理图跳线到正确的硬件上,如果你的硬件平台本身就含有这部分硬件电路,则可以完全用自己的平台实现

软件实现

软件方面主要是初始化和中断的程序开发,以及用于检测t3.5和t1.5时间间隔的定时器中断开发。

初始化主要包含

使能串口时钟

管脚映射

定时器配置

校验位配置

波特率配置

使能发送接收

难点是要懂得根据不同的波特率计算t3.5和t1.5的时间间隔

以9600bps为例, modbus每一个字节的发送接收有11位,所以每一位的时间

t = 1/9600

T3.5 = 3.5*t*11= 4.0104ms,根据这个值和定时器的时钟来配置定时器的装载值。

中断实现

Modbus驱动的中断也就是UART的中断实现主要是根据UARTx_S1寄存器的检测来区分是发送中断和接收中断,以及帧错去中断,校验错误中断等。需要注意的是错误中断和正常的接收发送中断不是一个中断向量,所以要分别在不同的中断函数检测状态寄存器的位来实现中断程序

如发送接收中断

void Uart_vISR1(void)

{

if(UART1_S1 & UART_S1_RDRF_MASK) //receive data register full flag

{

if(…..)/*9位模式*/

{

}

else/*8位模式*/

{

}

}

else if(UART1_S1 & UART_S1_TDRE_MASK) // transmit data register empty flag

{

if(….)/*发送完成*/

{

}

else

{

}

}

else

{

}

}

嵌入式

程序猿

微信号:InterruptISR

嵌入式程序猿致力于打造程序猿工程师交流分享的精品移动平台,欢迎各位猿友加入和分享。微信搜索嵌入式程序猿添加关注,或者长按下方二维码,选择识别图中二维码添加关注。

0?wx_fmt=jpeg

0?wx_fmt=jpeg

  • 0
    点赞
  • 2
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
以下是一个简单的RTTHREAD的MODBUS RTU驱动程序示例,该程序使用串口通信协议进行通信: ```c #include <rtthread.h> #include <rtdevice.h> #define MODBUS_SLAVE_ADDRESS 1 #define MODBUS_FUNCTION_CODE 3 #define MODBUS_START_ADDRESS 0 #define MODBUS_NUM_REGISTERS 10 static rt_device_t modbus_dev; static int modbus_read_registers(rt_uint8_t slave_addr, rt_uint8_t func_code, rt_uint16_t start_addr, rt_uint16_t num_regs, rt_uint16_t *regs) { rt_uint8_t request[8]; rt_uint8_t response[5 + 2 * num_regs]; rt_uint16_t crc; rt_uint16_t i; rt_size_t size; /* Construct request message */ request[0] = slave_addr; request[1] = func_code; request[2] = (start_addr >> 8) & 0xff; request[3] = start_addr & 0xff; request[4] = (num_regs >> 8) & 0xff; request[5] = num_regs & 0xff; /* Calculate CRC */ crc = rt_modbus_crc(request, 6); request[6] = crc & 0xff; request[7] = (crc >> 8) & 0xff; /* Send request message */ size = rt_device_write(modbus_dev, 0, request, sizeof(request)); if (size != sizeof(request)) { rt_kprintf("MODBUS RTU: Failed to send request message!\n"); return -RT_ERROR; } /* Receive response message */ size = rt_device_read(modbus_dev, 0, response, sizeof(response)); if (size != sizeof(response)) { rt_kprintf("MODBUS RTU: Failed to receive response message!\n"); return -RT_ERROR; } /* Check response message */ if (response[0] != slave_addr || response[1] != func_code || response[2] != 2 * num_regs) { rt_kprintf("MODBUS RTU: Invalid response message!\n"); return -RT_ERROR; } /* Extract data from response message */ for (i = 0; i < num_regs; ++i) { regs[i] = (response[3 + 2 * i] << 8) | response[4 + 2 * i]; } return RT_EOK; } static void modbus_thread_entry(void *parameter) { rt_uint16_t regs[MODBUS_NUM_REGISTERS]; while (1) { /* Read registers from MODBUS slave device */ if (modbus_read_registers(MODBUS_SLAVE_ADDRESS, MODBUS_FUNCTION_CODE, MODBUS_START_ADDRESS, MODBUS_NUM_REGISTERS, regs) == RT_EOK) { /* Process the received data */ // ... } /* Wait for next polling interval */ rt_thread_mdelay(1000); } } int modbus_init(void) { /* Open MODBUS RTU device */ modbus_dev = rt_device_find("uart1"); if (modbus_dev == RT_NULL) { rt_kprintf("MODBUS RTU: Failed to open device!\n"); return -RT_ERROR; } if (rt_device_open(modbus_dev, RT_DEVICE_OFLAG_RDWR) != RT_EOK) { rt_kprintf("MODBUS RTU: Failed to open device!\n"); return -RT_ERROR; } /* Create MODBUS RTU polling thread */ rt_thread_t tid = rt_thread_create("modbus", modbus_thread_entry, RT_NULL, 1024, 25, 5); if (tid != RT_NULL) { rt_thread_startup(tid); return RT_EOK; } else { rt_kprintf("MODBUS RTU: Failed to create thread!\n"); return -RT_ERROR; } } ``` 该程序包含一个 `modbus_read_registers()` 函数,该函数可以向MODBUS从设备发送读取寄存器请求,并接收从设备返回的数据。在主线程中,我们可以通过调用该函数来获取MODBUS从设备的寄存器值,并对其进行处理。 该程序还包含一个 `modbus_init()` 函数,该函数打开MODBUS RTU设备并创建一个MODBUS RTU轮询线程。在主程序中,我们可以通过调用该函数来启动MODBUS RTU驱动程序
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值