C语言CRC16_CCITT_FALSE函数法和查表法实现
函数法:
// CRC16_CCITT_FALSE 参数
#define CRC16_CCITT_FALSE_POLY 0x1021
#define CRC16_CCITT_FALSE_INIT 0xFFFF
uint16_t CRC16_CCITT_FALSE(const uint8_t *data, uint16_t length) {
uint16_t crc = CRC16_CCITT_FALSE_INIT;
for (uint16_t i = 0; i < length; i++) {
crc ^= (uint16_t)data[i] << 8;
for (uint8_t bit = 0; bit < 8; bit++) {
if (crc & 0x8000) {
crc = (crc << 1) ^ CRC16_CCITT_FALSE_POLY;
} else {
crc <<= 1;
}
}
}
return crc;
}
查表法(结构体INT_08BIT_2 小端CPU ‘L’在前‘H’在后,大端CPU ‘H’在前‘L’在后)
typedef struct
{
uchar L;
uchar H;
} INT_08BIT_2;
typedef union
{
u16 Int16Bit;
INT_08BIT_2 Int08Bit;
} INT_16BIT_08BIT;
u16 const CRC_Tab[] = {
0x00000,0x01021,0x02042,0x03063,
0x04084,0x050a5,0x060c6,0x070e7,
0x08108,0x09129,0x0a14a,0x0b16b,
0x0c18c,0x0d1ad,0x0e1ce,0x0f1ef,
0x01231,0x00210,0x03273,0x02252,
0x052b5,0x04294,0x072f7,0x062d6,
0x09339,0x08318,0x0b37b,0x0a35a,
0x0d3bd,0x0c39c,0x0f3ff,0x0e3de,
0x02462,0x03443,0x00420,0x01401,
0x064e6,0x074c7,0x044a4,0x05485,
0x0a56a,0x0b54b,0x08528,0x09509,
0x0e5ee,0x0f5cf,0x0c5ac,0x0d58d,
0x03653,0x02672,0x01611,0x00630,
0x076d7,0x066f6,0x05695,0x046b4,
0x0b75b,0x0a77a,0x09719,0x08738,
0x0f7df,0x0e7fe,0x0d79d,0x0c7bc,
0x048c4,0x058e5,0x06886,0x078a7,
0x00840,0x01861,0x02802,0x03823,
0x0c9cc,0x0d9ed,0x0e98e,0x0f9af,
0x08948,0x09969,0x0a90a,0x0b92b,
0x05af5,0x04ad4,0x07ab7,0x06a96,
0x01a71,0x00a50,0x03a33,0x02a12,
0x0dbfd,0x0cbdc,0x0fbbf,0x0eb9e,
0x09b79,0x08b58,0x0bb3b,0x0ab1a,
0x06ca6,0x07c87,0x04ce4,0x05cc5,
0x02c22,0x03c03,0x00c60,0x01c41,
0x0edae,0x0fd8f,0x0cdec,0x0ddcd,
0x0ad2a,0x0bd0b,0x08d68,0x09d49,
0x07e97,0x06eb6,0x05ed5,0x04ef4,
0x03e13,0x02e32,0x01e51,0x00e70,
0x0ff9f,0x0efbe,0x0dfdd,0x0cffc,
0x0bf1b,0x0af3a,0x09f59,0x08f78,
0x09188,0x081a9,0x0b1ca,0x0a1eb,
0x0d10c,0x0c12d,0x0f14e,0x0e16f,
0x01080,0x000a1,0x030c2,0x020e3,
0x05004,0x04025,0x07046,0x06067,
0x083b9,0x09398,0x0a3fb,0x0b3da,
0x0c33d,0x0d31c,0x0e37f,0x0f35e,
0x002b1,0x01290,0x022f3,0x032d2,
0x04235,0x05214,0x06277,0x07256,
0x0b5ea,0x0a5cb,0x095a8,0x08589,
0x0f56e,0x0e54f,0x0d52c,0x0c50d,
0x034e2,0x024c3,0x014a0,0x00481,
0x07466,0x06447,0x05424,0x04405,
0x0a7db,0x0b7fa,0x08799,0x097b8,
0x0e75f,0x0f77e,0x0c71d,0x0d73c,
0x026d3,0x036f2,0x00691,0x016b0,
0x06657,0x07676,0x04615,0x05634,
0x0d94c,0x0c96d,0x0f90e,0x0e92f,
0x099c8,0x089e9,0x0b98a,0x0a9ab,
0x05844,0x04865,0x07806,0x06827,
0x018c0,0x008e1,0x03882,0x028a3,
0x0cb7d,0x0db5c,0x0eb3f,0x0fb1e,
0x08bf9,0x09bd8,0x0abbb,0x0bb9a,
0x04a75,0x05a54,0x06a37,0x07a16,
0x00af1,0x01ad0,0x02ab3,0x03a92,
0x0fd2e,0x0ed0f,0x0dd6c,0x0cd4d,
0x0bdaa,0x0ad8b,0x09de8,0x08dc9,
0x07c26,0x06c07,0x05c64,0x04c45,
0x03ca2,0x02c83,0x01ce0,0x00cc1,
0x0ef1f,0x0ff3e,0x0cf5d,0x0df7c,
0x0af9b,0x0bfba,0x08fd9,0x09ff8,
0x06e17,0x07e36,0x04e55,0x05e74,
0x02e93,0x03eb2,0x00ed1,0x01ef0
};
/*
mode==0 Normal mode
mode==1 Add the CRC result to the end of the array
*/
u16 AH_CRC_Verify( uchar *str, uint size, uchar mode )
{
uint32 i;
INT_16BIT_08BIT crc;
INT_16BIT_08BIT y;
crc.Int16Bit = 0xffff;
for( i = 0 ; i < size ; i++ )
{
y.Int16Bit = CRC_Tab[ str[ i ] ^ crc.Int08Bit.H ];
y.Int08Bit.H ^= crc.Int08Bit.L;
crc.Int16Bit = y.Int16Bit;
}
if( mode == 1 )
{
str[ i++ ] = y.Int08Bit.H ;
str[ i ] = y.Int08Bit.L ;
}
return( crc.Int16Bit );
}
串口通信数据包介绍和包结构定义实例-CSDN博客
电脑串口和手机蓝牙BLE串口数据包通信调试工具-CSDN博客