3.4.2 CRC校验
在ModbusRTU传输模式下,通信报文(帧)包括一个基于循环冗余校验(CRC)方法的差错校验字段。
CRC的全称是循环冗余校验,其特点是检错能力极强,开销小,易于用编码器及检测电路实现。从其检错能力来看,它不能发现的错误的几率在0.0047%以下,在Modbus通信中基本可以忽略。CRC校验包括多个版本,常用的CRC校验有CRC-8、CRC-12、CRC-16、CRC-CCITT、CRC-32等。
从性能上和开销上考虑,CRC校验均远远优于奇偶校验及算术和校验等方式,因而在数据存储和数据通信领域,CRC无处不在。例如,著名的通信协议X.25的FCS(帧检错序列)采用的是CRC-CCITT,而WinRAR、NERO、ARJ、LHA等压缩工具软件采用的是CRC-32,磁盘驱动器的读写则采用了CRC-16,通用的图像存储格式GIF、TIFF等也都用CRC作为检错手段。
而Modbus协议中,则采用了CRC-16标准校验方法。在RTU模式下,CRC自身由两个字节组成,即CRC是一个16位的值。CRC字段校验整个报文的内容,无论报文中的单个字节采用何种奇偶校验方式,整个通信报文均可应用CRC-16校验算法。CRC字段作为报文的最后字段添加到整个报文末尾。
有一点需要注意,因为CRC-16由两个字节构成,所以涉及哪个字节放在前面,哪个字节放在后面传输的问题,即大小端模式的选择问题。另外,由于Modbus协议规定寄存器为16位(即两个字节)长度,因此大小端问题的存在给很多初学者造成了困扰。下一节将重点讲解大小端模式。
接收设备在接收信息时,会通过CRC算法重新计算,并把计算值与CRC字段中接收的实际值进行比较。若两者不同,则产生一个错误,并返回一个异常响应报文(帧)告知发送设备。
Modbus协议中的RTU校验码(CRC)计算,运算规则(即CRC计算方法)如下:
(1)预置一个值为0xFFFF的16位寄存器,此寄存器为CRC寄存器。
(2)把第1个8位二进制数据(即通信消息帧的第1个字节)与16位的CRC寄存器的相异或,异或的结果仍存放于该CRC寄存器中。
(3)把CRC寄存器的内容右移一位,用0填补最高位,并检测移出位是0还是1。
(4)如果移出位为零,则重复步骤(3)(再次右移一位);如果移出位为1,则CRC寄存器与0xA001进行异或。
(5)重复步骤(3)和(4),直到右移8次,这样整个8位数据全部进行了处理。
(6)重复步骤(2)~(5),进行通信消息帧下一个字节的处理。
(7)将该通信消息帧所有字节按上述步骤计算完成后,得到的16位CRC寄存器的高、低字节进行交换。即发送时首先添加低位字节,然后添加高位字节。
(8)最后得到的CRC寄存器内容即为CRC校验码。
需要强调的一点是,在CRC计算时只有串行链路上每个字符中的8个数据位参与计算,而其他比如起始位及停止位,如有奇偶校验位也包括奇偶校验位,都不参与CRC计算。
常用的CRC-16算法有查表法和计算法。
1.查表法
CRC查表法是将移位异或的计算结果做成了一个表,就是将0~256放入一个长度为16位的寄存器中的低8位,高8位填充0,然后将该寄存器与多项式0xA001按照上述步骤(3)、(4),直到8位全部移出,最后寄存器中的值就是表格中的数据,高8位、低8位分别单独一个表。
实际上,Modbus标准协议英文版提供了CRC查表算法。
函数的输入参数意义如下:
unsigned char * puchMsg; / * 要进行CRC校验的消息 * / unsigned short usDataLen; / * 消息中字节数 * /
1 / * 函数返回 unsigned short(即 2个字节)类型的CRC值 * / 2 unsigned short CRC16(unsigned char * puchMsg, unsigned short usDataLen) 3 { 4 unsigned char uchCRCHi=0xFF; / * 高CRC字节初始化 * / 5 unsigned char uchCRCLo=0xFF; / * 低CRC字节初始化 * / 6 unsigned short uIndex; / * CRC循环表中的索引 * / 7 8 while (usDataLen--) / * 循环处理传输缓冲区消息 * / 9 { 10 uIndex=uchCRCHi ^ * puchMsg++; / * 计算CRC * / 11 uchCRCHi=uchCRCLo ^ auchCRCHi[uIndex]; 12 uchCRCLo=auchCRCLo[uIndex]; 13 } 14 15 return (uchCRCHi <<8 | uchCRCLo); 16 }
其中,auchCRCHi和auchCRCLo分别定义如下:
1 static unsigned char auchCRCHi[] = 2 { 3 0x00,0xC1,0x81,0x40,0x01,0xC0,0x80,0x41,0x01,0xC0,0x80,0x41,0x00, 0xC1,0x81, 4 0x40,0x01,0xC0,0x80,0x41,0x00,0xC1,0x81,0x40,0x00,0xC1,0x81,0x40, 0x01,0xC0, 5 0x80,0x41,0x01,0xC0,0x80,0x41,0x00,0xC1,0x81,0x40,0x00,0xC1,0x81, 0x40,0x01, 6 0xC0,0x80,0x41,0x00,0xC1,0x81,0x40,0x01,0xC0,0x80,0x41,0x01,0xC0, 0x80,0x41, 7 0x00,0xC1,0x81,0x40,0x01,0xC0,0x80,0x41,0x00,0xC1,0x81,0x40,0x00, 0xC1,0x81, 8 0x40,0x01,0xC0,0x80,0x41,0x00,0xC1,0x81,0x40,0x01,0xC0,0x80,0x41, 0x01,0xC0, 9 0x80,0x41,0x00,0xC1,0x81,0x40,0x00,0xC1,0x81,0x40,0x01,0xC0,0x80, 0x41,0x01, 10 0xC0,0x80,0x41,0x00,0xC1,0x81,0x40,0x01,0xC0,0x80,0x41,0x00,0xC1, 0x81,0x40, 11 0x00,0xC1,0x81,0x40,0x01,0xC0,0x80,0x41,0x01,0xC0,0x80,0x41,0x00, 0xC1,0x81, 12 0x40,0x00,0xC1,0x81,0x40,0x01,0xC0,0x80,0x41,0x00,0xC1,0x81,0x40, 0x01,0xC0, 13 0x80,0x41,0x01,0xC0,0x80,0x41,0x00,0xC1,0x81,0x40,0x00,0xC1,0x81, 0x40,0x01, 14 0xC0,0x80,0x41,0x01,0xC0,0x80,0x41,0x00,0xC1,0x81,0x40,0x01,0xC0, 0x80,0x41, 15 0x00,0xC1,0x81,0x40,0x00,0xC1,0x81,0x40,0x01,0xC0,0x80,0x41,0x00, 0xC1,0x81, 16 0x40,0x01,0xC0,0x80,0x41,0x01,0xC0,0x80,0x41,0x00,0xC1,0x81,0x40, 0x01,0xC0, 17 0x80,0x41,0x00,0xC1,0x81,0x40,0x00,0xC1,0x81,0x40,0x01,0xC0,0x80, 0x41,0x01, 18 0xC0,0x80,0x41,0x00,0xC1,0x81,0x40,0x00,0xC1,0x81,0x40,0x01,0xC0, 0x80,0x41, 19 0x00,0xC1,0x81,0x40,0x01,0xC0,0x80,0x41,0x01,0xC0,0x80,0x41,0x00, 0xC1,0x81, 20 0x40 21 } ; 22 23 static char auchCRCLo[] = 24 { 25 0x00,0xC0,0xC1,0x01,0xC3,0x03,0x02,0xC2,0xC6,0x06,0x07,0xC7,0x05, 0xC5,0xC4, 26 0x04,0xCC,0x0C,0x0D,0xCD,0x0F,0xCF,0xCE,0x0E,0x0A,0xCA,0xCB,0x0B, 0xC9,0x09, 27 0x08,0xC8,0xD8,0x18,0x19,0xD9,0x1B,0xDB,0xDA,0x1A,0x1E,0xDE,0xDF, 0x1F,0xDD, 28 0x1D,0x1C,0xDC,0x14,0xD4,0xD5,0x15,0xD7,0x17,0x16,0xD6,0xD2,0x12, 0x13,0xD3, 29 0x11,0xD1,0xD0,0x10,0xF0,0x30,0x31,0xF1,0x33,0xF3,0xF2,0x32,0x36, 0xF6,0xF7, 30 0x37,0xF5,0x35,0x34,0xF4,0x3C,0xFC,0xFD,0x3D,0xFF,0x3F,0x3E,0xFE, 0xFA,0x3A, 31 0x3B,0xFB,0x39,0xF9,0xF8,0x38,0x28,0xE8,0xE9,0x29,0xEB,0x2B,0x2A, 0xEA,0xEE, 32 0x2E,0x2F,0xEF,0x2D,0xED,0xEC,0x2C,0xE4,0x24,0x25,0xE5,0x27,0xE7, 0xE6,0x26, 33 0x22,0xE2,0xE3,0x23,0xE1,0x21,0x20,0xE0,0xA0,0x60,0x61,0xA1,0x63, 0xA3,0xA2, 34 0x62,0x66,0xA6,0xA7,0x67,0xA5,0x65,0x64,0xA4,0x6C,0xAC,0xAD,0x6D, 0xAF,0x6F, 35 0x6E,0xAE,0xAA,0x6A,0x6B,0xAB,0x69,0xA9,0xA8,0x68,0x78,0xB8,0xB9, 0x79,0xBB, 36 0x7B,0x7A,0xBA,0xBE,0x7E,0x7F,0xBF,0x7D,0xBD,0xBC,0x7C,0xB4,0x74, 0x75,0xB5, 37 0x77,0xB7,0xB6,0x76,0x72,0xB2,0xB3,0x73,0xB1,0x71,0x70,0xB0,0x50, 0x90,0x91, 38 0x51,0x93,0x53,0x52,0x92,0x96,0x56,0x57,0x97,0x55,0x95,0x94,0x54, 0x9C,0x5C, 39 0x5D,0x9D,0x5F,0x9F,0x9E,0x5E,0x5A,0x9A,0x9B,0x5B,0x99,0x59,0x58, 0x98,0x88, 40 0x48,0x49,0x89,0x4B,0x8B,0x8A,0x4A,0x4E,0x8E,0x8F,0x4F,0x8D,0x4D, 0x4C,0x8C, 41 0x44,0x84,0x85,0x45,0x87,0x47,0x46,0x86,0x82,0x42,0x43,0x83,0x41, 0x81,0x80, 42 0x40 43 };
注意:实际编程的时候,auchCRCHi[]和auchCRCLo[]的定义应该放在函数CRC16()之前。
查表法可以进一步简化如下:
1 unsigned short CRC16(unsigned char * puchMsg, unsigned short usDataLen) 2 { 3 static const unsigned short usCRCTable[] = 4 { 5 0X0000,0XC0C1, 0XC181, 0X0140, 0XC301, 0X03C0, 0X0280, 0XC241, 6 0XC601, 0X06C0, 0X0780, 0XC741, 0X0500, 0XC5C1, 0XC481, 0X0440, 7 0XCC01, 0X0CC0, 0X0D80, 0XCD41, 0X0F00, 0XCFC1, 0XCE81, 0X0E40, 8 0X0A00, 0XCAC1, 0XCB81, 0X0B40, 0XC901, 0X09C0, 0X0880, 0XC841, 9 0XD801, 0X18C0, 0X1980, 0XD941, 0X1B00, 0XDBC1, 0XDA81, 0X1A40, 10 0X1E00, 0XDEC1, 0XDF81, 0X1F40, 0XDD01, 0X1DC0, 0X1C80, 0XDC41, 11 0X1400, 0XD4C1, 0XD581, 0X1540, 0XD701, 0X17C0, 0X1680, 0XD641, 12 0XD201, 0X12C0, 0X1380, 0XD341, 0X1100, 0XD1C1, 0XD081, 0X1040, 13 0XF001, 0X30C0, 0X3180, 0XF141, 0X3300, 0XF3C1, 0XF281, 0X3240, 14 0X3600, 0XF6C1, 0XF781, 0X3740, 0XF501, 0X35C0, 0X3480, 0XF441, 15 0X3C00, 0XFCC1, 0XFD81, 0X3D40, 0XFF01, 0X3FC0, 0X3E80, 0XFE41, 16 0XFA01, 0X3AC0, 0X3B80, 0XFB41, 0X3900, 0XF9C1, 0XF881, 0X3840, 17 0X2800, 0XE8C1, 0XE981, 0X2940, 0XEB01, 0X2BC0, 0X2A80, 0XEA41, 18 0XEE01, 0X2EC0, 0X2F80, 0XEF41, 0X2D00, 0XEDC1, 0XEC81, 0X2C40, 19 0XE401, 0X24C0, 0X2580, 0XE541, 0X2700, 0XE7C1, 0XE681, 0X2640, 20 0X2200, 0XE2C1, 0XE381, 0X2340, 0XE101, 0X21C0, 0X2080, 0XE041, 21 0XA001, 0X60C0, 0X6180, 0XA141, 0X6300, 0XA3C1, 0XA281, 0X6240, 22 0X6600, 0XA6C1, 0XA781, 0X6740, 0XA501, 0X65C0, 0X6480, 0XA441, 23 0X6C00, 0XACC1, 0XAD81, 0X6D40, 0XAF01, 0X6FC0, 0X6E80, 0XAE41, 24 0XAA01, 0X6AC0, 0X6B80, 0XAB41, 0X6900, 0XA9C1, 0XA881, 0X6840, 25 0X7800, 0XB8C1, 0XB981, 0X7940, 0XBB01, 0X7BC0, 0X7A80, 0XBA41, 26 0XBE01, 0X7EC0, 0X7F80, 0XBF41, 0X7D00, 0XBDC1, 0XBC81, 0X7C40, 27 0XB401, 0X74C0, 0X7580, 0XB541, 0X7700, 0XB7C1, 0XB681, 0X7640, 28 0X7200, 0XB2C1, 0XB381, 0X7340, 0XB101, 0X71C0, 0X7080, 0XB041, 29 0X5000, 0X90C1, 0X9181, 0X5140, 0X9301, 0X53C0, 0X5280, 0X9241, 30 0X9601, 0X56C0, 0X5780, 0X9741, 0X5500, 0X95C1, 0X9481, 0X5440, 31 0X9C01, 0X5CC0, 0X5D80, 0X9D41, 0X5F00, 0X9FC1, 0X9E81, 0X5E40, 32 0X5A00, 0X9AC1, 0X9B81, 0X5B40, 0X9901, 0X59C0, 0X5880, 0X9841, 33 0X8801, 0X48C0, 0X4980, 0X8941, 0X4B00, 0X8BC1, 0X8A81, 0X4A40, 34 0X4E00, 0X8EC1, 0X8F81, 0X4F40, 0X8D01, 0X4DC0, 0X4C80, 0X8C41, 35 0X4400, 0X84C1, 0X8581, 0X4540, 0X8701, 0X47C0, 0X4680, 0X8641, 36 0X8201, 0X42C0, 0X4380, 0X8341, 0X4100, 0X81C1, 0X8081, 0X4040 37 }; 38 39 unsigned char nTemp; 40 unsigned short usRegCRC =0xFFFF; 41 42 while (usDataLen--) 43 { 44 nTemp = * puchMsg++^ usRegCRC; 45 usRegCRC >>=8; 46 usRegCRC ^=usCRCTable[nTemp]; 47 } 48 return usRegCRC; 49 }
查表法的特点是:以字节为单位进行计算,速度快、语句少,但表格占用一定的程序空间。
2.计算法
计算法按位计算。这个方法可以适用于所有长度的数据校验,最为灵活;但由于是按位计算,其效率并不是最优,只适用于对速度不敏感的场合。基本的算法如下:
输入参数的意义:
unsigned char * puchMsg; / * 要进行CRC校验的消息 * / unsigned short usDataLen; / * 消息中的字节数 * /
1 / * 函数返回 unsigned short(即 2个字节)类型的CRC值 * / 2 unsigned short CRC16(unsigned char * puchMsg, unsigned short usDataLen) 3 { 4 int i, j; / * 循环变量 * / 5 unsigned short usRegCRC =0xFFFF; / * 用于保存CRC值 * / 6 7 for(i =0; i <usDataLen; i++) / * 循环处理传输缓冲区消息 * / 8 { 9 usRegCRC ^= * puchMsg++; / * 异或算法得到CRC值 * / 10 for(j =0; j <8; j++) / * 循环处理每个 bit位 * / 11 { 12 if (usRegCRC & 0x0001) 13 usRegCRC =usRegCRC >>1 ^ 0xA001; 14 else 15 usRegCRC >>=1; 16 } 17 } 18 19 return usRegCRC; 20 }
这里举一个简单的例子。假设从设备地址为1,要求读取输入寄存器地址30001的值,则RTU模式下具体的查询消息帧如下:
0x01,0x04,0x00,0x00,0x00,0x01,0x31,0xCA
其中,0xCA31即为CRC值。因为Modbus规定发送时CRC必须低字节在前,高字节在后,因此实际的消息帧的发送顺序为0x31,0xCA。