当前位置: 首页 > news >正文

野火STM32Modbus主机读取寄存器/线圈失败(二)-解决CRC校验错误

文章目录

  • 前情提要
  • 问题背景
  • CRC校验失败
    • 问题现象
    • 原始问题数据
    • 问题分析
      • 1. CRC校验算法验证
      • 2. 手动计算验证
      • 问题解决思路
  • 问题解决
    • 根本原因
    • 解决方式1
    • 解决方式2
  • 重新编译测试

前情提要

在自己的开发板上移植了野火的modbus主机程序并尝试使用。

问题背景

我使用STM32显示板作为Modbus主机连接电脑,并在电脑上运行Modbus Slave软件。测试中发现,读取保持寄存器和输入寄存器均失败,但写入操作正常。Modbus Slave可以正确接收到请求帧:

这说明主机发出的命令没有问题。然而,在我的代码中,用于存储保持寄存器的数组 usMRegHoldBuf[][]始终为0,未能更新。进一步排查发现,程序并未进入回调函数 eMBMasterRegHoldingCB(UCHAR * pucRegBuffer, USHORT usAddress, USHORT usNRegs, eMBRegisterMode eMode)

已经解决了接收中断不触发的问题。在使用FreeModbus库进行通信时,发现从机返回的数据CRC校验失败,导致数据解析异常。

CRC校验失败

问题现象

mb_m.c文件的eMBErrorCode eMBMasterPoll( void )中的分支添加解析结果的测试,结果显示parse_success_count一直为0,parse_fail_count递增,排查原因是CRC校验失败,导致解析失败。

        case EV_MASTER_FRAME_RECEIVED:eStatus = peMBMasterFrameReceiveCur( &ucRcvAddress, &ucMBFrame, &usLength );/* 测试:记录解析结果 */extern volatile uint32_t parse_success_count, parse_fail_count;if(eStatus == MB_ENOERR) {parse_success_count++;} else {parse_fail_count++;}/* Check if the frame is for us. If not ,send an error process event. */if ( ( eStatus == MB_ENOERR ) && ( ucRcvAddress == ucMBMasterGetDestAddress() ) ){( void ) xMBMasterPortEventPost( EV_MASTER_EXECUTE );}else{vMBMasterSetErrorType(EV_ERROR_RECEIVE_DATA);( void ) xMBMasterPortEventPost( EV_MASTER_ERROR_PROCESS );}break;

原始问题数据

  • 主机发送: 01 03 00 01 00 04 15 C9
  • 从机返回: 01 03 08 00 02 00 03 00 04 00 05 73 55(实际接收)
  • 期望返回: 01 03 08 00 02 00 03 00 04 00 05 73 D5

问题分析

1. CRC校验算法验证

使用Modbus CRC16算法对接收到的数据进行校验:

// Modbus CRC16计算函数
USHORT usMBCRC16(UCHAR * pucFrame, USHORT usLen)
{UCHAR ucCRCHi = 0xFF;UCHAR ucCRCLo = 0xFF;int iIndex;while(usLen--){iIndex = ucCRCLo ^ *(pucFrame++);ucCRCLo = (UCHAR)(ucCRCHi ^ aucCRCHi[iIndex]);ucCRCHi = aucCRCLo[iIndex];}return (USHORT)(ucCRCHi << 8 | ucCRCLo);
}

2. 手动计算验证

对从机返回的数据进行计算:

  • 数据部分: 01 03 08 00 02 00 03 00 04 00 05 (11字节)
  • 接收CRC: 73 55
  • 计算CRC: 73 D5

发现差异: 最后一个字节应为 D5 但实际接收为 55

问题解决思路

通过检查代码配置,发现串口校验位不匹配

  • ModbusSlave工具配置: 115200-8-E-1 (偶校验)
  • 代码中配置: UART_PARITY_NONE (无校验)
// 原错误配置
#define MB_MASTER_USART_PARITY UART_PARITY_NONE// 正确配置应为
#define MB_MASTER_USART_PARITY UART_PARITY_EVEN

问题解决

根本原因

校验位不匹配导致数据接收错误
虽然官方给的文档中说串口配置是无校验位
在这里插入图片描述
实际程序中的串口定义也是无校验位
在这里插入图片描述)
但是主函数中,设置的是偶校验
在这里插入图片描述)
而串口的数据位配置的是8位,这就导致了数据接收错误

解决方式1

修改校验位设置为

MB_PAR_NONE, /*!< No parity. */

	/* FreeModbus主机初始化 */eMBMasterInit(MB_RTU, MB_MASTER_USARTx, MB_MASTER_USART_BAUDRATE, MB_PAR_NONE);

解决方式2

修改usart.cMX_USART2_UART_Init的工作模式配置,让它根据选择的校验模式自行判断应该配置成多少位。

/*** @brief  DEBUG_USART GPIO 配置,工作模式配置。115200 8-N-1* @param  无* @retval 无*/  
void MX_USART2_UART_Init(uint8_t ucPORT, uint32_t ulBaudRate, uint8_t eParity)
{if(ucPORT != 2) //必须设置为串口2return ;huart2.Instance = DEBUG_USART;huart2.Init.BaudRate = ulBaudRate;huart2.Init.StopBits = UART_STOPBITS_1;huart2.Init.Parity = eParity;/* 根据校验位设置数据位长度 */if(eParity == UART_PARITY_NONE) {huart2.Init.WordLength = UART_WORDLENGTH_8B;} else {huart2.Init.WordLength = UART_WORDLENGTH_9B;}huart2.Init.Mode = UART_MODE_TX_RX;huart2.Init.HwFlowCtl = UART_HWCONTROL_NONE;huart2.Init.OverSampling = UART_OVERSAMPLING_16;if (HAL_UART_Init(&huart2) != HAL_OK){while(1);}}

重新编译测试

修改后重新编译程序,测试结果:

  • ✅ 接收数据正确: 01 03 08 00 02 00 03 00 04 00 05 73 D5
  • ✅ CRC校验通过
  • ✅ 数据解析正常
http://www.dtcms.com/a/354164.html

相关文章:

  • 让ai写一个类github首页
  • Web前端之JavaScript时间体系全解析、performance、Date、now
  • Go语言循环性能终极对决:for vs range 深度剖析
  • 如何用Postman做接口测试?
  • k8s中的服务(Service),详细列举
  • JavaSE:类和对象2
  • Redis集群介绍——主从、哨兵、集群
  • 单兵图传设备如何接入指挥中心平台?国标GB/T28181协议的20位ID有何含义?如何进行配置?
  • [手写系列]Go手写db — — 第二版
  • spring-boot-test与 spring-boot-starter-test 区别
  • 前端架构设计模式与AI驱动的智能化演进
  • 嵌入式学习日志————USART串口协议
  • 【开发便利】让远程Linux服务器能够访问内网git仓库
  • 目标检测基础
  • [系统架构设计师]论文(二十三)
  • 控制系统仿真之时域分析(二)
  • 计算机组成原理(13) 第二章 - DRAM SRAM SDRAM ROM
  • 通信原理(005)——带宽、宽带、传输速率、流量
  • 农业物联网:科技赋能现代农业新篇章
  • uC/OS-III 队列相关接口
  • Linux 命令浏览文件内容
  • 机器视觉的车载触摸屏玻璃盖板贴合应用
  • 【Bluetooth】【调试工具篇】第九章 实时抓取工具 btsnoop
  • [vcpkg] Windows入门使用介绍
  • 致远OA新闻公告讨论调查信息查询SQL
  • 模拟电路中什么时候适合使用电流传递信号,什么时候合适使用电压传递信号
  • 世界的接口:数学、心智与未知的协作
  • 【前端】jsmpeg 介绍及使用
  • Libvio 访问异常排查指南:从现象到根源的深度剖析
  • 专项智能练习(关系数据库)