原创 crc 个人总结

2020-8-9 11:57 2896 37 6 分类: 汽车电子 文集: 杂项
一、介绍
CRC即循环冗余校验码(Cyclic Redundancy Check):是一种根据网络数据包或计算机文件等数据产生简短固定位数校验码的一种信道编码技术,主要用来检测或校验数据传输或者保存后可能出现的错误。数据通信领域中最常用的一种差错校验码,其信息字段和校验字段长度可以任意指定,但要求通信双方定义的CRC标准一致。
二、原理
基础原理来源于模2运算,通过除多项式取余数的方法得到CRC校验码。
多项式写法  例如  crc 16 ibm 多项式为:x16 + x15 + x2 + 1  16进制形式为 0x18005 其简记形式为0x8005, 我们大部分看到的是0x8005。
因为多项式的首尾必定为1。 1 xor 1 结果必定是0.
        输入值反转的意思是在计算之前先将二项式反转,然后再用得到的新值和数据进行计算。
        输入输出 初始值需异或。即先将要计算的数据与初始值的最低字节进行异或,然后再与多项式进行计算。
三、在线计算工具
CRC(循环冗余校验)在线计算_ip33.com 
网址:http://www.ip33.com/crc.html
特点:只有 crc 校验不过有个好处可以自定义crc 格式。
        CRC校验工具-在线工具 
        网址:http://www.metools.info/code/c15.html
特点:不仅有crc 校验工具还有其他计算工具,不足是crc 只能选着标准格式,不能自定义格式。
总结:两个工具使用都很不错,支持16进制和ascii 对复制过来的文件也很友好。
四、编程实现
实现方式主要分为两种
  1. 查表法。优点速度快,缺点占用一定的内存空间。
  2. 直接计算法。 优点占用内存空间小,缺点需要一定的计算量。
      源码实现(来源于网络没有找到原作者):
uint8_t crc4_itu(uint8_t *data, uint_len length);
  • uint8_t crc5_epc(uint8_t *data, uint_len length);
  • uint8_t crc5_itu(uint8_t *data, uint_len length);
  • uint8_t crc5_usb(uint8_t *data, uint_len length);
  • uint8_t crc6_itu(uint8_t *data, uint_len length);
  • uint8_t crc7_mmc(uint8_t *data, uint_len length);
  • uint8_t crc8(uint8_t *data, uint_len length);
  • uint8_t crc8_itu(uint8_t *data, uint_len length);
  • uint8_t crc8_rohc(uint8_t *data, uint_len length);
  • uint8_t crc8_maxim(uint8_t *data, uint_len length);//DS18B20
  • uint16_t crc16_ibm(uint8_t *data, uint_len length);
  • uint16_t crc16_maxim(uint8_t *data, uint_len length);
  • uint16_t crc16_usb(uint8_t *data, uint_len length);
  • uint16_t crc16_modbus(uint8_t *data, uint_len length);
  • uint16_t crc16_ccitt(uint8_t *data, uint_len length);
  • uint16_t crc16_ccitt_false(uint8_t *data, uint_len length);
  • uint16_t crc16_x25(uint8_t *data, uint_len length);
  • uint16_t crc16_xmodem(uint8_t *data, uint_len length);
  • uint16_t crc16_dnp(uint8_t *data, uint_len length);
  • uint32_t crc32(uint8_t *data, uint_len length);
  • uint32_t crc32_mpeg_2(uint8_t *data, uint_len length);
  • /******************************************************************************
  • * Name: CRC-4/ITU x4+x+1
  • * Poly: 0x03
  • * Init: 0x00
  • * Refin: True
  • * Refout: True
  • * Xorout: 0x00
  • * Note:
  • *****************************************************************************/
  • uint8_t crc4_itu(uint8_t *data, uint_len length)
  • {
  • uint8_t i;
  • uint8_t crc = 0; // Initial value
  • while(length--)
  • {
  • crc ^= *data++; // crc ^= *data; data++;
  • for (i = 0; i < 8; ++i)
  • {
  • if (crc & 1)
  • crc = (crc >> 1) ^ 0x0C;// 0x0C = (reverse 0x03)>>(8-4)
  • else
  • crc = (crc >> 1);
  • }
  • }
  • return crc;
  • }
  • /******************************************************************************
  • * Name: CRC-5/EPC x5+x3+1
  • * Poly: 0x09
  • * Init: 0x09
  • * Refin: False
  • * Refout: False
  • * Xorout: 0x00
  • * Note:
  • *****************************************************************************/
  • uint8_t crc5_epc(uint8_t *data, uint_len length)
  • {
  • uint8_t i;
  • uint8_t crc = 0x48; // Initial value: 0x48 = 0x09<<(8-5)
  • while(length--)
  • {
  • crc ^= *data++; // crc ^= *data; data++;
  • for ( i = 0; i < 8; i++ )
  • {
  • if ( crc & 0x80 )
  • crc = (crc << 1) ^ 0x48; // 0x48 = 0x09<<(8-5)
  • else
  • crc <<= 1;
  • }
  • }
  • return crc >> 3;
  • }
  • /******************************************************************************
  • * Name: CRC-5/ITU x5+x4+x2+1
  • * Poly: 0x15
  • * Init: 0x00
  • * Refin: True
  • * Refout: True
  • * Xorout: 0x00
  • * Note:
  • *****************************************************************************/
  • uint8_t crc5_itu(uint8_t *data, uint_len length)
  • {
  • uint8_t i;
  • uint8_t crc = 0; // Initial value
  • while(length--)
  • {
  • crc ^= *data++; // crc ^= *data; data++;
  • for (i = 0; i < 8; ++i)
  • {
  • if (crc & 1)
  • crc = (crc >> 1) ^ 0x15;// 0x15 = (reverse 0x15)>>(8-5)
  • else
  • crc = (crc >> 1);
  • }
  • }
  • return crc;
  • }
  • /******************************************************************************
  • * Name: CRC-5/USB x5+x2+1
  • * Poly: 0x05
  • * Init: 0x1F
  • * Refin: True
  • * Refout: True
  • * Xorout: 0x1F
  • * Note:
  • *****************************************************************************/
  • uint8_t crc5_usb(uint8_t *data, uint_len length)
  • {
  • uint8_t i;
  • uint8_t crc = 0x1F; // Initial value
  • while(length--)
  • {
  • crc ^= *data++; // crc ^= *data; data++;
  • for (i = 0; i < 8; ++i)
  • {
  • if (crc & 1)
  • crc = (crc >> 1) ^ 0x14;// 0x14 = (reverse 0x05)>>(8-5)
  • else
  • crc = (crc >> 1);
  • }
  • }
  • return crc ^ 0x1F;
  • }
  • /******************************************************************************
  • * Name: CRC-6/ITU x6+x+1
  • * Poly: 0x03
  • * Init: 0x00
  • * Refin: True
  • * Refout: True
  • * Xorout: 0x00
  • * Note:
  • *****************************************************************************/
  • uint8_t crc6_itu(uint8_t *data, uint_len length)
  • {
  • uint8_t i;
  • uint8_t crc = 0; // Initial value
  • while(length--)
  • {
  • crc ^= *data++; // crc ^= *data; data++;
  • for (i = 0; i < 8; ++i)
  • {
  • if (crc & 1)
  • crc = (crc >> 1) ^ 0x30;// 0x30 = (reverse 0x03)>>(8-6)
  • else
  • crc = (crc >> 1);
  • }
  • }
  • return crc;
  • }
  • /******************************************************************************
  • * Name: CRC-7/MMC x7+x3+1
  • * Poly: 0x09
  • * Init: 0x00
  • * Refin: False
  • * Refout: False
  • * Xorout: 0x00
  • * Use: MultiMediaCard,SD,ect.
  • *****************************************************************************/
  • uint8_t crc7_mmc(uint8_t *data, uint_len length)
  • {
  • uint8_t i;
  • uint8_t crc = 0; // Initial value
  • while(length--)
  • {
  • crc ^= *data++; // crc ^= *data; data++;
  • for ( i = 0; i < 8; i++ )
  • {
  • if ( crc & 0x80 )
  • crc = (crc << 1) ^ 0x12; // 0x12 = 0x09<<(8-7)
  • else
  • crc <<= 1;
  • }
  • }
  • return crc >> 1;
  • }
  • /******************************************************************************
  • * Name: CRC-8 x8+x2+x+1
  • * Poly: 0x07
  • * Init: 0x00
  • * Refin: False
  • * Refout: False
  • * Xorout: 0x00
  • * Note:
  • *****************************************************************************/
  • uint8_t crc8(uint8_t *data, uint_len length)
  • {
  • uint8_t i;
  • uint8_t crc = 0; // Initial value
  • while(length--)
  • {
  • crc ^= *data++; // crc ^= *data; data++;
  • for ( i = 0; i < 8; i++ )
  • {
  • if ( crc & 0x80 )
  • crc = (crc << 1) ^ 0x07;
  • else
  • crc <<= 1;
  • }
  • }
  • return crc;
  • }
  • /******************************************************************************
  • * Name: CRC-8/ITU x8+x2+x+1
  • * Poly: 0x07
  • * Init: 0x00
  • * Refin: False
  • * Refout: False
  • * Xorout: 0x55
  • * Alias: CRC-8/ATM
  • *****************************************************************************/
  • uint8_t crc8_itu(uint8_t *data, uint_len length)
  • {
  • uint8_t i;
  • uint8_t crc = 0; // Initial value
  • while(length--)
  • {
  • crc ^= *data++; // crc ^= *data; data++;
  • for ( i = 0; i < 8; i++ )
  • {
  • if ( crc & 0x80 )
  • crc = (crc << 1) ^ 0x07;
  • else
  • crc <<= 1;
  • }
  • }
  • return crc ^ 0x55;
  • }
  • /******************************************************************************
  • * Name: CRC-8/ROHC x8+x2+x+1
  • * Poly: 0x07
  • * Init: 0xFF
  • * Refin: True
  • * Refout: True
  • * Xorout: 0x00
  • * Note:
  • *****************************************************************************/
  • uint8_t crc8_rohc(uint8_t *data, uint_len length)
  • {
  • uint8_t i;
  • uint8_t crc = 0xFF; // Initial value
  • while(length--)
  • {
  • crc ^= *data++; // crc ^= *data; data++;
  • for (i = 0; i < 8; ++i)
  • {
  • if (crc & 1)
  • crc = (crc >> 1) ^ 0xE0; // 0xE0 = reverse 0x07
  • else
  • crc = (crc >> 1);
  • }
  • }
  • return crc;
  • }
  • /******************************************************************************
  • * Name: CRC-8/MAXIM x8+x5+x4+1
  • * Poly: 0x31
  • * Init: 0x00
  • * Refin: True
  • * Refout: True
  • * Xorout: 0x00
  • * Alias: DOW-CRC,CRC-8/IBUTTON
  • * Use: Maxim(Dallas)'s some devices,e.g. DS18B20
  • *****************************************************************************/
  • uint8_t crc8_maxim(uint8_t *data, uint_len length)
  • {
  • uint8_t i;
  • uint8_t crc = 0; // Initial value
  • while(length--)
  • {
  • crc ^= *data++; // crc ^= *data; data++;
  • for (i = 0; i < 8; i++)
  • {
  • if (crc & 1)
  • crc = (crc >> 1) ^ 0x8C; // 0x8C = reverse 0x31
  • else
  • crc >>= 1;
  • }
  • }
  • return crc;
  • }
  • /******************************************************************************
  • * Name: CRC-16/IBM x16+x15+x2+1
  • * Poly: 0x8005
  • * Init: 0x0000
  • * Refin: True
  • * Refout: True
  • * Xorout: 0x0000
  • * Alias: CRC-16,CRC-16/ARC,CRC-16/LHA
  • *****************************************************************************/
  • uint16_t crc16_ibm(uint8_t *data, uint_len length)
  • {
  • uint8_t i;
  • uint16_t crc = 0; // Initial value
  • while(length--)
  • {
  • crc ^= *data++; // crc ^= *data; data++;
  • for (i = 0; i < 8; ++i)
  • {
  • if (crc & 1)
  • crc = (crc >> 1) ^ 0xA001; // 0xA001 = reverse 0x8005
  • else
  • crc = (crc >> 1);
  • }
  • }
  • return crc;
  • }
  • /******************************************************************************
  • * Name: CRC-16/MAXIM x16+x15+x2+1
  • * Poly: 0x8005
  • * Init: 0x0000
  • * Refin: True
  • * Refout: True
  • * Xorout: 0xFFFF
  • * Note:
  • *****************************************************************************/
  • uint16_t crc16_maxim(uint8_t *data, uint_len length)
  • {
  • uint8_t i;
  • uint16_t crc = 0; // Initial value
  • while(length--)
  • {
  • crc ^= *data++; // crc ^= *data; data++;
  • for (i = 0; i < 8; ++i)
  • {
  • if (crc & 1)
  • crc = (crc >> 1) ^ 0xA001; // 0xA001 = reverse 0x8005
  • else
  • crc = (crc >> 1);
  • }
  • }
  • return ~crc; // crc^0xffff
  • }
  • /******************************************************************************
  • * Name: CRC-16/USB x16+x15+x2+1
  • * Poly: 0x8005
  • * Init: 0xFFFF
  • * Refin: True
  • * Refout: True
  • * Xorout: 0xFFFF
  • * Note:
  • *****************************************************************************/
  • uint16_t crc16_usb(uint8_t *data, uint_len length)
  • {
  • uint8_t i;
  • uint16_t crc = 0xffff; // Initial value
  • while(length--)
  • {
  • crc ^= *data++; // crc ^= *data; data++;
  • for (i = 0; i < 8; ++i)
  • {
  • if (crc & 1)
  • crc = (crc >> 1) ^ 0xA001; // 0xA001 = reverse 0x8005
  • else
  • crc = (crc >> 1);
  • }
  • }
  • return ~crc; // crc^0xffff
  • }
  • /******************************************************************************
  • * Name: CRC-16/MODBUS x16+x15+x2+1
  • * Poly: 0x8005
  • * Init: 0xFFFF
  • * Refin: True
  • * Refout: True
  • * Xorout: 0x0000
  • * Note:
  • *****************************************************************************/
  • uint16_t crc16_modbus(uint8_t *data, uint_len length)
  • {
  • uint8_t i;
  • uint16_t crc = 0xffff; // Initial value
  • while(length--)
  • {
  • crc ^= *data++; // crc ^= *data; data++;
  • for (i = 0; i < 8; ++i)
  • {
  • if (crc & 1)
  • crc = (crc >> 1) ^ 0xA001; // 0xA001 = reverse 0x8005
  • else
  • crc = (crc >> 1);
  • }
  • }
  • return crc;
  • }
  • /******************************************************************************
  • * Name: CRC-16/CCITT x16+x12+x5+1
  • * Poly: 0x1021
  • * Init: 0x0000
  • * Refin: True
  • * Refout: True
  • * Xorout: 0x0000
  • * Alias: CRC-CCITT,CRC-16/CCITT-TRUE,CRC-16/KERMIT
  • *****************************************************************************/
  • uint16_t crc16_ccitt(uint8_t *data, uint_len length)
  • {
  • uint8_t i;
  • uint16_t crc = 0; // Initial value
  • while(length--)
  • {
  • crc ^= *data++; // crc ^= *data; data++;
  • for (i = 0; i < 8; ++i)
  • {
  • if (crc & 1)
  • crc = (crc >> 1) ^ 0x8408; // 0x8408 = reverse 0x1021
  • else
  • crc = (crc >> 1);
  • }
  • }
  • return crc;
  • }
  • /******************************************************************************
  • * Name: CRC-16/CCITT-FALSE x16+x12+x5+1
  • * Poly: 0x1021
  • * Init: 0xFFFF
  • * Refin: False
  • * Refout: False
  • * Xorout: 0x0000
  • * Note:
  • *****************************************************************************/
  • uint16_t crc16_ccitt_false(uint8_t *data, uint_len length)
  • {
  • uint8_t i;
  • uint16_t crc = 0xffff; //Initial value
  • while(length--)
  • {
  • crc ^= (uint16_t)(*data++) << 8; // crc ^= (uint6_t)(*data)<<8; data++;
  • for (i = 0; i < 8; ++i)
  • {
  • if ( crc & 0x8000 )
  • crc = (crc << 1) ^ 0x1021;
  • else
  • crc <<= 1;
  • }
  • }
  • return crc;
  • }
  • /******************************************************************************
  • * Name: CRC-16/X25 x16+x12+x5+1
  • * Poly: 0x1021
  • * Init: 0xFFFF
  • * Refin: True
  • * Refout: True
  • * Xorout: 0XFFFF
  • * Note:
  • *****************************************************************************/
  • uint16_t crc16_x25(uint8_t *data, uint_len length)
  • {
  • uint8_t i;
  • uint16_t crc = 0xffff; // Initial value
  • while(length--)
  • {
  • crc ^= *data++; // crc ^= *data; data++;
  • for (i = 0; i < 8; ++i)
  • {
  • if (crc & 1)
  • crc = (crc >> 1) ^ 0x8408; // 0x8408 = reverse 0x1021
  • else
  • crc = (crc >> 1);
  • }
  • }
  • return ~crc; // crc^Xorout
  • }
  • /******************************************************************************
  • * Name: CRC-16/XMODEM x16+x12+x5+1
  • * Poly: 0x1021
  • * Init: 0x0000
  • * Refin: False
  • * Refout: False
  • * Xorout: 0x0000
  • * Alias: CRC-16/ZMODEM,CRC-16/ACORN
  • *****************************************************************************/
  • uint16_t crc16_xmodem(uint8_t *data, uint_len length)
  • {
  • uint8_t i;
  • uint16_t crc = 0; // Initial value
  • while(length--)
  • {
  • crc ^= (uint16_t)(*data++) << 8; // crc ^= (uint16_t)(*data)<<8; data++;
  • for (i = 0; i < 8; ++i)
  • {
  • if ( crc & 0x8000 )
  • crc = (crc << 1) ^ 0x1021;
  • else
  • crc <<= 1;
  • }
  • }
  • return crc;
  • }
  • /******************************************************************************
  • * Name: CRC-16/DNP x16+x13+x12+x11+x10+x8+x6+x5+x2+1
  • * Poly: 0x3D65
  • * Init: 0x0000
  • * Refin: True
  • * Refout: True
  • * Xorout: 0xFFFF
  • * Use: M-Bus,ect.
  • *****************************************************************************/
  • uint16_t crc16_dnp(uint8_t *data, uint_len length)
  • {
  • uint8_t i;
  • uint16_t crc = 0; // Initial value
  • while(length--)
  • {
  • crc ^= *data++; // crc ^= *data; data++;
  • for (i = 0; i < 8; ++i)
  • {
  • if (crc & 1)
  • crc = (crc >> 1) ^ 0xA6BC; // 0xA6BC = reverse 0x3D65
  • else
  • crc = (crc >> 1);
  • }
  • }
  • return ~crc; // crc^Xorout
  • }
  • /******************************************************************************
  • * Name: CRC-32 x32+x26+x23+x22+x16+x12+x11+x10+x8+x7+x5+x4+x2+x+1
  • * Poly: 0x4C11DB7
  • * Init: 0xFFFFFFF
  • * Refin: True
  • * Refout: True
  • * Xorout: 0xFFFFFFF
  • * Alias: CRC_32/ADCCP
  • * Use: WinRAR,ect.
  • *****************************************************************************/
  • uint32_t crc32(uint8_t *data, uint_len length)
  • {
  • uint8_t i;
  • uint32_t crc = 0xffffffff; // Initial value
  • while(length--)
  • {
  • crc ^= *data++; // crc ^= *data; data++;
  • for (i = 0; i < 8; ++i)
  • {
  • if (crc & 1)
  • crc = (crc >> 1) ^ 0xEDB88320;// 0xEDB88320= reverse 0x04C11DB7
  • else
  • crc = (crc >> 1);
  • }
  • }
  • return ~crc;
  • }
  • /******************************************************************************
  • * Name: CRC-32/MPEG-2 x32+x26+x23+x22+x16+x12+x11+x10+x8+x7+x5+x4+x2+x+1
  • * Poly: 0x4C11DB7
  • * Init: 0xFFFFFFF
  • * Refin: False
  • * Refout: False
  • * Xorout: 0x0000000
  • * Note:
  • *****************************************************************************/
  • uint32_t crc32_mpeg_2(uint8_t *data, uint_len length)
  • {
  • uint8_t i;
  • uint32_t crc = 0xffffffff; // Initial value
  • while(length--)
  • {
  • crc ^= (uint32_t)(*data++) << 24;// crc ^=(uint32_t)(*data)<<24; data++;
  • for (i = 0; i < 8; ++i)
  • {
  • if ( crc & 0x80000000 )
  • crc = (crc << 1) ^ 0x04C11DB7;
  • else
  • crc <<= 1;
  • }
  • }
  • return crc;
  • }
  • 复制代码

    最后附上看着原理写的不错的链接 :https://segmentfault.com/a/1190000018094567
    ﹌﹌﹌﹌﹌﹌﹌﹌﹌﹌﹌﹌﹌﹌﹌﹌﹌﹌﹌﹌﹌﹌﹌﹌﹌﹌﹌﹌﹌﹌﹌﹌﹌﹌﹌﹌﹌﹌﹌﹌﹌﹌﹌﹌﹌﹌
    免责声明:本文部分内容来源网络,版权归原作者所有,如涉及作品版权问题,请及时与我们联系,谢谢!
    crc

    作者: jinsheng, 来源:面包板社区

    链接: https://mbb.eet-china.com/blog/uid-me-425232.html

    版权声明:本文为博主原创,未经本人允许,禁止转载!

    PARTNER CONTENT

    文章评论2条评论)

    登录后参与讨论

    intel001 2020-9-2 10:46

    厉害

    Evan.i 2020-8-24 09:53

    感谢分享
    相关推荐阅读
    jinsheng 2020-08-23 17:10
    十二、freeRTOS 软件定时器
    定时器硬件定时器:CPU内部自带的定时器模块,通过初始化、配置可以实现定时,定时时间到以后就会执行相应的定时器中断处理函数。硬件定时器一般都带有其他功能,比如PWM输出、输入捕获等功能。但缺点是硬件定...
    jinsheng 2020-08-23 17:07
    十一、freeRTOS 优先级翻转与互斥信号量
    优先级翻转优先级翻转简介:就是高优先级的任务运行起来的效果好像成了低优先级,而低优先级比高优先级先运行。优先级翻转如下所示:优先级翻转过程:为什么会发生优先级翻转?因为两个任务(L和H)使用了同一个二...
    jinsheng 2020-08-23 17:05
    十、freeRTOS 计数型信号量
    计数型信号量简介:计数型信号量的创建:计数型信号量动态创建函数:释放和获取信号量(与二值信号量相同)释放信号量:获取信号量: 测试实验用按键来模拟事件,按键按下后表示有事件发生,则释放计数型...
    jinsheng 2020-08-23 16:54
    九、freeRTOS 二值信号量
    信号量1 信号量用于共享资源的访问:2 信号量用于任务同步:为什么一直说在中断服务函数中,不能够做太多的事情?在进入中断服务函数时,低优先级的中断就不能响应,同类型的中断也无法响应,所以就要求ISR一...
    jinsheng 2020-08-23 16:49
    八、freeRTOS 队列
    队列简介:注意,队列长度并不是每个队列的字节数,而是能存多少条数据。如我们创建一个队列,每条消息最大10字节,队列可以保存4条消息,那么队列的长度为4。队列的功能:1 数据拷贝为什么一定要使用数据拷贝...
    jinsheng 2020-08-23 16:45
    七、FreeRTOS时间管理
    FreeRTOS时间管理FreeRTOS的两个延时函数:vTaskDelay()          相对延时vTaskDelayUntil()&nbs...
    我要评论
    2
    37
    关闭 站长推荐上一条 /3 下一条