Modbus-RTU-crc16校驗(yàn)方法C語(yǔ)言實(shí)現(xiàn)
一、CRC16校驗(yàn)碼的使用
現(xiàn)選擇最常用的CRC-16校驗(yàn),說(shuō)明它的使用方法。 根據(jù)Modbus協(xié)議,常規(guī)485通訊的信息發(fā)送形式如下: 地址 功能碼 數(shù)據(jù)信息 校驗(yàn)碼 1byte 1byte nbyte 2byte CRC校驗(yàn)是前面幾段數(shù)據(jù)內(nèi)容的校驗(yàn)值,為一個(gè)16位數(shù)據(jù),發(fā)送時(shí), ?。。?!低8位在前,高8為最后。這點(diǎn)很重要?。。?! 例如:信息字段代碼為: 1011001,校驗(yàn)字段為:1010。 發(fā)送方:發(fā)出的傳輸字段為: 1 0 1 1 0 0 1 1 0 10 信息字段 校驗(yàn)字段 接收方:使用相同的計(jì)算方法計(jì)算出信息字段的校驗(yàn)碼,對(duì)比接收到的實(shí)際校驗(yàn)碼,如果相等及信息正確,不相等則信息錯(cuò)誤;或者將接受到的所有信息除多項(xiàng)式,如果能夠除盡,則信息正確。二、CRC16校驗(yàn)碼計(jì)算方法
常用查表法和計(jì)算法。計(jì)算方法一般都是: (1)、預(yù)置1個(gè)16位的寄存器值0xFFFF,稱此寄存器為CRC寄存器; (2)、把第一個(gè)8位二進(jìn)制數(shù)據(jù)(既通訊信息幀的第一個(gè)字節(jié))與16位的CRC寄存器的低 8位相異或,把結(jié)果放于CRC寄存器,高八位數(shù)據(jù)不變; (3)、把CRC寄存器的內(nèi)容右移一位(朝高位)用0填補(bǔ)最高位,并檢查右移后的移出位; (4)、如果移出位為0:重復(fù)第3步(再次右移一位);如果移出位為1,CRC寄存器與一多 項(xiàng)式(A001)進(jìn)行異或; (5)、重復(fù)步驟3和4,直到右移8次,這樣整個(gè)8位數(shù)據(jù)全部進(jìn)行了處理; (6)、重復(fù)步驟2到步驟5,進(jìn)行通訊信息幀下一個(gè)字節(jié)的處理; (7)、將該通訊信息幀所有字節(jié)按上述步驟計(jì)算完成后,得到的16位CRC寄存器的高、低 字節(jié)進(jìn)行交換; (8)、最后得到的CRC寄存器內(nèi)容即為:CRC碼。 以上計(jì)算步驟中的多項(xiàng)式A001是8005按位顛倒后的結(jié)果。 少了一個(gè)高低字節(jié)交換的語(yǔ)句!!!------------------------------------------------- unsigned int CRC16; #define SEED 0xFFFF //initialization for CRC16 #define GP 0xA001 //generating polynomial //for standard CRC16 //(remainder of division) //to start a new CRC, set CRC16 = SEED //then for each byte call Calc_CRC(byte, &CRC16); //CRC16 will contain the result //(if you calculate all of the incoming data //INCLUDING the CRC, the result should be 0x0000 //and if you are sending the CRC be sure to //send the bytes in the correct order) void Calc_CRC(unsigned byte b, unsigned int* CRC) { BOOL carry; int i; CRC[0] ^= b & 0xFF; for (i=0; i<8; i++) { carry = CRC[0] & 0x0001; CRC[0]>>=1; if (carry) CRC[0] ^= GP; } } -------------------------------------------------
這個(gè)代碼是正確的!!!! unsigned int CRC16_2(unsigned char *buf, int len) { unsigned int crc = 0xFFFF; for (int pos = 0; pos < len; pos++) { crc ^= (unsigned int)buf[pos]; // XOR byte into least sig. byte of crc for (int i = 8; i != 0; i--) // Loop over each bit { if ((crc & 0x0001) != 0) // If the LSB is set { crc >>= 1; // Shift right and XOR 0xA001 crc ^= 0xA001; } else // Else LSB is not set { crc >>= 1; // Just shift right } } } //高低字節(jié)轉(zhuǎn)換 crc = ((crc & 0x00ff) << 8) | ((crc & 0xff00) >> 8); return crc; }?????????????????
本文參與 騰訊云自媒體分享計(jì)劃 ,歡迎正在閱讀的你也加入,一起分享。 免責(zé)聲明:本文內(nèi)容由21ic獲得授權(quán)后發(fā)布,版權(quán)歸原作者所有,本平臺(tái)僅提供信息存儲(chǔ)服務(wù)。文章僅代表作者個(gè)人觀點(diǎn),不代表本平臺(tái)立場(chǎng),如有問(wèn)題,請(qǐng)聯(lián)系我們,謝謝!