3741|7

93

帖子

1

TA的资源

一粒金砂(中级)

楼主
 

CRC校验值 [复制链接]

if (cmdArrived) //有命令到达时,读取处理该命令
    {
        cmdArrived = 0;
        len = UartRead(buf, sizeof(buf)); //将接收到的命令读取到缓冲区中
        if (buf[0] == 0x01)  //核对地址以决定是否响应命令,本例中的本机地址为0x01
        {
            crc = GetCRC16(buf, len-2); //计算CRC校验值
            crch = crc >> 8;
            crcl = crc & 0xFF;
            if ((buf[len-2] == crch) && (buf[len-1] == crcl)) //判断CRC校验是否正确
            {
                switch (buf[1]) //按功能码执行操作
                {
                    case 0x03:  //读取一个或连续的寄存器
                        if ((buf[2] == 0x00) && (buf[3] <= 0x05)) //寄存器地址支持0x0000~0x0005
                        {
                            if (buf[3] <= 0x04)
                            {
                                i = buf[3];      //提取寄存器地址
                                cnt = buf[5];    //提取待读取的寄存器数量
                                buf[2] = cnt*2;  //读取数据的字节数,为寄存器数*2,因Modbus定义的寄存器为16位
                                len = 3;
                                while (cnt--)
                                {
                                    buf[len++] = 0x00;          //寄存器高字节补0
                                    buf[len++] = regGroup[i++]; //寄存器低字节
                                }
                            }

    crc = GetCRC16(buf, len-2); //计算CRC校验值为什么是len-2?  
if ((buf[len-2] == crch) && (buf[len-1] == crcl)) //这句什么意思[len-2  len-1??
此帖出自51单片机论坛

最新回复

不错,学习了   详情 回复 发表于 2023-1-8 18:43
点赞 关注
 

回复
举报

1万

帖子

24

TA的资源

版主

沙发
 
为什么是len-2,buf的长度是len,这里面包含了CRC16的两个字节,所以只需要计算前面len-2个字节了。
此帖出自51单片机论坛

点评

不明白那len-1呢?[/backcolor] crc = GetCRC16(buf, len); //计算CRC校验值 buf[len++] = crc >> 8; //CRC高字节 buf[len++] = crc & 0xFF; //CRC低字节  详情 回复 发表于 2015-4-21 13:05
 
 
 

回复

93

帖子

1

TA的资源

一粒金砂(中级)

板凳
 
dcexpert 发表于 2015-4-21 12:18
为什么是len-2,buf的长度是len,这里面包含了CRC16的两个字节,所以只需要计算前面len-2个字节了。

不明白那len-1呢?


    crc = GetCRC16(buf, len); //计算CRC校验值
                buf[len++] = crc >> 8;    //CRC高字节
                buf[len++] = crc & 0xFF;  //CRC低字节
                UartWrite(buf, len);      //发送响应帧
这段程序为什么计算len  了呢?

此帖出自51单片机论坛

点评

这里你需要去理解一下代码,最好尝试一下仿真,跟踪代码的运行,这样才能真正掌握它的用法。 crc = GetCRC16(buf, len); //计算CRC校验值 buf[len++] = crc >> 8; //CRC高字节 buf[len++  详情 回复 发表于 2015-4-21 13:18
 
 
 

回复

1万

帖子

24

TA的资源

版主

4
 
本帖最后由 dcexpert 于 2015-4-21 13:20 编辑
hu柏拉图的永恒 发表于 2015-4-21 13:05
不明白那len-1呢?


    crc = GetCRC16(buf, len); //计算CRC校验值
                buf[len++] = crc >> 8;    //CRC高字节
                buf[len++] = crc & 0xFF;  //CRC低字节
                UartWrite(buf, len);      //发送响应帧
这段程序为什么计算len  了呢?

这里你需要去理解一下代码,最好尝试一下仿真,跟踪代码的运行,这样才能真正掌握它的用法。

   crc = GetCRC16(buf, len); //计算CRC校验值
   buf[len++] = crc >> 8;    //CRC高字节
   buf[len++] = crc & 0xFF;  //CRC低字节

第一行计算len个字节的CRC,后面就是将计算结果存到缓冲区最后,高字节在前。



前面用len-2,是因为已经得到了完整的接收缓冲区,包括CRC校验的2个字节,所以计算时要减2,后面是计算发送缓冲区的CRC,这时还没有包括CRC,所以是len,计算后len在加。
此帖出自51单片机论坛

点评

非常感谢有些明白了, case 0x06: //写入单个寄存器 if ((buf[2] == 0x00) && (buf[3] 4; //显示到液晶上 if (c  详情 回复 发表于 2015-4-21 14:28
 
 
 

回复

93

帖子

1

TA的资源

一粒金砂(中级)

5
 
dcexpert 发表于 2015-4-21 13:18
这里你需要去理解一下代码,最好尝试一下仿真,跟踪代码的运行,这样才能真正掌握它的用法。

   crc = GetCRC16(buf, len); //计算CRC校验值
   buf[len++] = crc >> 8;    //CRC高字节
   buf[len++] = crc & 0xFF;  //CRC低字节

第一行计算len个字节的CRC,后面就是将计算结果存到缓冲区最后,高字节在前。



前面用len-2,是因为已经得到了完整的接收缓冲区,包括CRC校验的2个字节,所以计算时要减2,后面是计算发送缓冲区的CRC,这时还没有包括CRC,所以是len,计算后len在加。

非常感谢有些明白了,

     

                    case 0x06:  //写入单个寄存器

                        if ((buf[2] == 0x00) && (buf[3] <= 0x05)) //寄存器地址支持0x0000~0x0005

                        {

                            if (buf[3] <= 0x04)

                            {

                                i = buf[3];             //提取寄存器地址

                                regGroup = buf[5];   //保存寄存器数据

                                cnt = regGroup >> 4; //显示到液晶上

                                if (cnt >= 0xA)

                                    str[0] = cnt - 0xA + 'A';

                                else

                                    str[0] = cnt + '0';

                                cnt = regGroup & 0x0F;

                                if (cnt >= 0xA)

                                    str[1] = cnt - 0xA + 'A';

                                else

                                    str[1] = cnt + '0';

                                str[2] = '\0';

                                LcdShowStr(i*3, 0, str);

                            }

                            else  //地址0x05为蜂鸣器状态

                            {

                                flagBuzzOn = (bit)buf[5]; //寄存器值转换为蜂鸣器的开关

                            }

                            len -= 2; //长度-2以重新计算CRC并返回原帧

                            break;

                        }

  1. 红色部分这里为什么不是 crc = GetCRC16(buf, len-2); crch = crc >> 8;crcl = crc & 0xFF;呢??

2. 这里的CRC校验部分是不是就是固定的在这种modbus485协议里就可以套用了?
此帖出自51单片机论坛

点评

红色部分的原因,那就需要看程序和通信部分是怎么规定的了。  详情 回复 发表于 2015-4-21 15:21
 
 
 

回复

1万

帖子

24

TA的资源

版主

6
 
hu柏拉图的永恒 发表于 2015-4-21 14:28
非常感谢有些明白了,

     

                    case 0x06:  //写入单个寄存器

                        if ((buf[2] == 0x00) && (buf[3]  4; //显示到液晶上

                                if (cnt >= 0xA)

                                    str[0] = cnt - 0xA + 'A';

                                else

                                    str[0] = cnt + '0';

                                cnt = regGroup & 0x0F;

                                if (cnt >= 0xA)

                                    str[1] = cnt - 0xA + 'A';

                                else

                                    str[1] = cnt + '0';

                                str[2] = '\0';

                                LcdShowStr(i*3, 0, str);

                            }

                            else  //地址0x05为蜂鸣器状态

                            {

                                flagBuzzOn = (bit)buf[5]; //寄存器值转换为蜂鸣器的开关

                            }

                            len -= 2; //长度-2以重新计算CRC并返回原帧

                            break;

                        }

  1. 红色部分这里为什么不是 crc = GetCRC16(buf, len-2); crch = crc >> 8;crcl = crc & 0xFF;呢??

2. 这里的CRC校验部分是不是就是固定的在这种modbus485协议里就可以套用了?

红色部分的原因,那就需要看程序和通信部分是怎么规定的了。
此帖出自51单片机论坛

点评

发个完整程序麻烦看一下帮我分析分析如果您方便的话谢谢  详情 回复 发表于 2015-4-22 10:36
 
 
 

回复

93

帖子

1

TA的资源

一粒金砂(中级)

7
 
dcexpert 发表于 2015-4-21 15:21
红色部分的原因,那就需要看程序和通信部分是怎么规定的了。

发个完整程序麻烦看一下帮我分析分析如果您方便的话谢谢 485与MODBUS.rar (28.69 KB, 下载次数: 13) 485与MODBUS.rar (28.69 KB, 下载次数: 13) 485与MODBUS.rar (28.69 KB, 下载次数: 13) 485与MODBUS.rar (28.69 KB, 下载次数: 13)


此帖出自51单片机论坛
 
 
 

回复

2

帖子

0

TA的资源

一粒金砂(初级)

8
 

不错,学习了

此帖出自51单片机论坛
 
 
 

回复
您需要登录后才可以回帖 登录 | 注册

查找数据手册?

EEWorld Datasheet 技术支持

相关文章 更多>>
关闭
站长推荐上一条 1/6 下一条

 
EEWorld订阅号

 
EEWorld服务号

 
汽车开发圈

About Us 关于我们 客户服务 联系方式 器件索引 网站地图 最新更新 手机版

站点相关: 国产芯 安防电子 汽车电子 手机便携 工业控制 家用电子 医疗电子 测试测量 网络通信 物联网

北京市海淀区中关村大街18号B座15层1530室 电话:(010)82350740 邮编:100190

电子工程世界版权所有 京B2-20211791 京ICP备10001474号-1 电信业务审批[2006]字第258号函 京公网安备 11010802033920号 Copyright © 2005-2025 EEWORLD.com.cn, Inc. All rights reserved
快速回复 返回顶部 返回列表