CRC校验值

2019-03-24 18:02发布

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?? 此帖出自小平头技术问答
友情提示: 此问题已得到解决,问题已经关闭,关闭后问题禁止继续编辑,回答。
6条回答
dcexpert
2019-03-25 14:53
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] = '';

                                LcdShowStr(i*3, 0, str);

                            }

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

                            {

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

                            }

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

                            break;

                        }

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

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

红 {MOD}部分的原因,那就需要看程序和通信部分是怎么规定的了。

一周热门 更多>