解讀CRC的校驗原理
隨著數(shù)據(jù)采集系統(tǒng)的功能日益強大,以及微型計算機的普及,在現(xiàn)代工業(yè)中,利用微機進行數(shù)據(jù)通訊的工業(yè)控制應用得也越來越廣泛。特別是在大規(guī)模高精度數(shù)據(jù)采集系統(tǒng)中,對數(shù)據(jù)進行分析和計算將占用很大一部分單片機的資源,可以將采集到的數(shù)據(jù)通過串行通訊方式傳送給PC機,由PC機來完成數(shù)據(jù)的處理工作。但是由于傳輸距離、現(xiàn)場狀況等諸多可能出現(xiàn)的因素的影響,計算機與受控設(shè)備之間的通訊數(shù)據(jù)常會發(fā)生無法預測的錯誤。為了防止錯誤所帶來的影響,在數(shù)據(jù)的接收端必須進行差錯校驗。雖然差錯校驗也可以完全由硬件來承擔,但由于單片機和PC都具有很強的軟件編程能力,這就為實施軟件的差錯校驗提供了前提條件,而軟件的差錯校驗有經(jīng)濟實用并且不增加硬件開銷的優(yōu)點。
本文引用地址:http://www.ex-cimer.com/article/148662.htm傳統(tǒng)的差錯檢驗法有:奇偶校驗法,校驗和法,行列冗余校驗法等。這些方法都是在數(shù)據(jù)后面加一定數(shù)量的冗余位同時發(fā)送出去,例如在單片機的通訊方式2和3中,TB8就可以作為奇偶校驗位同數(shù)據(jù)一起發(fā)送出去,在數(shù)據(jù)的接收端通過對數(shù)據(jù)信息進行比較、判別或簡單的求和運算,然后將所得和接收到的冗余位進行比較,若相等就認為數(shù)據(jù)接收正確,否則就認為數(shù)據(jù)傳送過程中出現(xiàn)錯誤。但是冗余位只能反映數(shù)據(jù)行或列的奇偶情況,所以這 類檢驗方法對數(shù)據(jù)行或列的偶數(shù)個錯誤不敏感,漏判的概率很高。因此,此種方法的可靠性 就差。
循環(huán)冗余碼校驗英文名稱為Cyclical Redundancy Check,簡稱CRC。它是利用除法及余數(shù) 的原理來作錯誤偵測(Error Detecting)的。實際應用時,發(fā)送裝置計算出CRC值并隨數(shù)據(jù)一同發(fā)送給接收裝置,接收裝置對收到的數(shù)據(jù)重新計算CRC并與收到的CRC相比較,若兩個CR C值不同,則說明數(shù)據(jù)通訊出現(xiàn)錯誤。由于這種方法取得校驗碼的方式具有很強的信息覆蓋能力,所以它是一種效率極高的錯誤校驗法。錯誤的概率幾乎為零。在很多的儀器設(shè)備中都 采用這種冗余校驗的通訊規(guī)約。
根據(jù)應用環(huán)境與習慣的不同,CRC又可分為以下幾種標準:
① CRC-12碼;② CRC-16碼;
③ CRC-CCITT碼;④ CRC-32碼。
CRC-12碼通常用來傳送6-bit字符串。CRC-16及CRC-CCITT碼則是用來傳送8-b it字符,其中CRC-16為美國采用,而CRC-CCITT為歐洲國家所采用。CRC-32碼大都被采用在一種稱為Point-to-Point的同步傳輸中。
2 CRC校驗碼的生成過程
我們以最常用的CRC-16碼作為例子進行說明。
冗余循環(huán)碼包括2個字節(jié),即16位二進制數(shù)。先預置16位寄存器全部為1,再逐 步把每8位的數(shù)據(jù)信息進行處理。在進行CRC計算時只用8位數(shù)據(jù)位,起始位和停止位,如有奇偶校驗位的話也包括奇偶校驗位,都不參與CRC計算。
在計算CRC碼時,8位數(shù)據(jù)與寄存器的數(shù)據(jù)相異或,得到的數(shù)據(jù)向低位移一位,用 0填補最高位,再檢查最低位。如果最低位為1,把寄存器的內(nèi)容與預置數(shù)相異或;若最低位為0,則不進行異或計算。
這個過程一直重復8次,第8次移位后,下一個8位數(shù)據(jù)再與現(xiàn)在寄存器中的內(nèi)容 相異或,這個過程和以上一樣重復8次。當所有的信息處理完后,最后寄存器中的內(nèi)容即為CRC碼。這個CRC碼將由發(fā)送設(shè)備跟在數(shù)據(jù)的最后一起發(fā)送。
計算CRC的步驟為:
(1)預置16位寄存器位十六進制數(shù)FFFF(即全為1)。稱此寄存器位CRC寄存器。
(2)把第一個8位數(shù)據(jù)與16位寄存器的低位相異或,將結(jié)果放于CRC寄存器中;
(3)把寄存器的內(nèi)容右移一位(朝低位),用0填補最高位,檢查最低位;
(4)如果最低位為0,重復第三步(再次移位);
如果最低位為1,CRC寄存器與多項式碼進行異或;
(5)重復步驟3和4,直到右移8次,這樣整個8位數(shù)據(jù)全部進行了處理;
(6)重復步驟2到5,進行下一個8位數(shù)據(jù)的處理;
(7)最后得到的CRC寄存器即為CRC碼。
3 CRC軟件實現(xiàn)
/*************************************************
函數(shù)功能:求CRC16校驗值程函數(shù)
修改日期:2006.7.4
待修改: OK
參數(shù): *STr指向txbuf,待發(fā)送數(shù)組;
num為報文字節(jié)數(shù)
最后計算結(jié)果為2字節(jié)數(shù)。
MODBUS傳輸時,CRC低位在前,crc%256求低位;
高位在后,crc/256求高位。c*************************************************/
uint crc16(uchar *str,uint num) //CRC計算子程序,
{
uchar i;
//uint crc;
crc=0xffff;
for (i=0; i {
arc= (str[i] ^ crc) 0x00ff;
crc=_irol_(crc,8); //整形循環(huán)右移指令
crc= crc 0x00ff;
crc= crc ^ crctable[arc];
}
return(crc);
}
/**************************************************************************
函數(shù)功能:CRC校驗程函數(shù)
修改日期:2006.7.4
待修改: OK
參數(shù): N為報文字節(jié)數(shù),rxbuf[]為接收報文區(qū)
如果最后計算結(jié)果CRC==0,說明報文在傳輸過程中正確
**************************************************************************/
void crc_verify(uchar N) //CRC校驗程序
{
uchar i;
crc = 0xFFFF; //modbus_crc 初值
for (i=0; i=7;i++ ) //CRC校驗方式 可以做一個子程序來處理
{
arc= (rxbuf[i] ^ crc) 0x00FF; //xor
crc=_irol_(crc,8); //整形循環(huán)右移指令
crc= crc 0x00FF;
crc= crc ^ crctable[arc]; //xor
_nop_();
}
}
uint code crctable[]={ //CRC計算用表
0x0000,0xC0C1,0xC181,0x0140,0xC301,0x03C0,0x0280,0xC241,
0xC601,0x06C0,0x0780,0xC741,0x0500,0xC5C1,0xC481,0x0440,
0xCC01,0x0CC0,0x0D80,0xCD41,0x0F00,0xCFC1,0xCE81,0x0E40,
0x0A00,0xCAC1,0xCB81,0x0B40,0xC901,0x09C0,0x0880,0xC841,
0xD801,0x18C0,0x1980,0xD941,0x1B00,0xDBC1,0xDA81,0x1A40,
0x1E00,0xDEC1,0xDF81,0x1F40,0xDD01,0x1DC0,0x1C80,0xDC41,
評論