news 2026/6/18 8:52:10

CRC校验 (查表方式) 通用源码(兼容CRC4/5/6/7/8/16/32/自定义),包含 CRC表数据的计算代码

作者头像

张小明

前端开发工程师

1.2k 24
文章封面图
CRC校验 (查表方式) 通用源码(兼容CRC4/5/6/7/8/16/32/自定义),包含 CRC表数据的计算代码

下面包含了CRC校验 通用代码CRC表数据的计算代码。实际应用可以根据CRC类型裁剪代码,可以直接把打印输出的表直接存放在ROM区,减少代码量。

#include <stdio.h> #include <string.h> /**Private typedef*************************************************************/ /* USER CODE BEGIN PT */ typedef enum { CRC4_ITU = 0, CRC5_EPC, CRC5_ITU, CRC5_USB, CRC6_ITU, CRC7_MMC, CRC8, CRC8_ITU, CRC8_ROHC, CRC8_MAXIM, CRC16_IBM, CRC16_MAXIM, CRC16_USB, CRC16_MODBUS, CRC16_CCITT, CRC16_CCITT_FALSE, CRC16_X25, CRC16_XMODEM, CRC16_DNP, CRC32, CRC32_MPEG2, CRC_NUM }CrcTypeE; typedef enum { E_FALSE = 0, //假(错误) E_TRUE = !E_FALSE //真(正确) }BoolTypeE; typedef struct { const char *Name; unsigned char Width; //宽度,即CRC比特数。 unsigned int Poly; //生成多项式的简写,以16进制表示。例如:CRC-32即是0x04C11DB7,忽略了最高位的"1",即完整的生成项是0x104C11DB7。 unsigned int CrcInit; //初始值,这是算法开始时寄存器(crc)的初始化预置值,十六进制表示。 unsigned int XorOut; //计算结果与此参数异或后得到最终的CRC值。 BoolTypeE RefIn; //待测数据的每个字节是否按位反转,E_TRUE或E_FALSE。 BoolTypeE RefOut; //在计算后之后,异或输出之前,整个数据是否按位反转,E_TRUE或E_FALSE。 }CrcInfoTypeS; /* USER CODE END PT */ /**Private variables***********************************************************/ /* USER CODE BEGIN PV */ const static CrcInfoTypeS s_crcInfoTab[CRC_NUM] = { //CRC算法名称 宽度 多项式 初始值 结果异或值 输入反转 输出反转 适用场景说明 {"CRC4_ITU", 4, 0x03, 0x00, 0x00, E_TRUE, E_TRUE}, // 电信4bit短报文 {"CRC5_EPC", 5, 0x09, 0x09, 0x00, E_FALSE, E_FALSE}, // RFID EPC电子标签 {"CRC5_ITU", 5, 0x15, 0x00, 0x00, E_TRUE, E_TRUE}, // ITU-T 5bit通信校验 {"CRC5_USB", 5, 0x05, 0x1F, 0x1F, E_TRUE, E_TRUE}, // USB低速5bit帧校验 {"CRC6_ITU", 6, 0x03, 0x00, 0x00, E_TRUE, E_TRUE}, // 6bit低速通信短帧 {"CRC7_MMC", 7, 0x09, 0x00, 0x00, E_FALSE, E_FALSE}, // SD/MMC存储卡命令校验 {"CRC8", 8, 0x07, 0x00, 0x00, E_FALSE, E_FALSE}, // 通用简易8位校验 {"CRC8_ITU", 8, 0x07, 0x00, 0x55, E_FALSE, E_FALSE}, // ITU-T标准8位CRC {"CRC8_ROHC", 8, 0x07, 0xFF, 0x00, E_TRUE, E_TRUE}, // ROHC网络压缩协议 {"CRC8_MAXIM", 8, 0x31, 0x00, 0x00, E_TRUE, E_TRUE}, // DS18B20 1-Wire总线 {"CRC16_IBM", 16, 0x8005, 0x0000, 0x0000, E_TRUE, E_TRUE}, // CRC16-ARC,通用简易16位校验 {"CRC16_MAXIM", 16, 0x8005, 0x0000, 0xFFFF, E_TRUE, E_TRUE}, // Maxim配套CRC16校验 {"CRC16_USB", 16, 0x8005, 0xFFFF, 0xFFFF, E_TRUE, E_TRUE}, // USB总线16位校验 {"CRC16_MODBUS", 16, 0x8005, 0xFFFF, 0x0000, E_TRUE, E_TRUE}, // Modbus-RTU 485工控总线(最常用16位CRC) {"CRC16_CCITT", 16, 0x1021, 0x0000, 0x0000, E_TRUE, E_TRUE}, // CCITT标准,蓝牙基带 {"CRC16_CCITT_FALSE", 16, 0x1021, 0xFFFF, 0x0000, E_FALSE, E_FALSE}, // EtherCAT FoE/X.25 串口文件传输 {"CRC16_X25", 16, 0x1021, 0xFFFF, 0xFFFF, E_TRUE, E_TRUE}, // X.25、PPP拨号物联网模组 {"CRC16_XMODEM", 16, 0x1021, 0x0000, 0x0000, E_FALSE, E_FALSE}, // Xmodem/Ymodem串口文件上传 {"CRC16_DNP", 16, 0x3D65, 0x0000, 0xFFFF, E_TRUE, E_TRUE}, // DNP3电力工控通信协议 {"CRC32", 32, 0x04C11DB7, 0xFFFFFFFF, 0xFFFFFFFF, E_TRUE, E_TRUE}, // CRC32 IEEE802.3 OTA固件/以太网FCS/ZIP文件校验 {"CRC32_MPEG2", 32, 0x04C11DB7, 0xFFFFFFFF, 0x00000000, E_FALSE, E_FALSE}, // MPEG2视频码流校验,不可用于固件升级 }; static unsigned int s_crcTab[256]; /* USER CODE END PV */ /** ******************************************************************************* * @brief 位反转 函数 * @param [in] inVal - 反转前数据 * @param [in] bits - 反转位数 * @return 反转后数据 * @note 就是将高位与低位数据顺序反过来。 ******************************************************************************* */ unsigned int BitsReverse(unsigned int inVal, unsigned char bits) { unsigned int outVal = 0; unsigned char i; for(i=0; i<bits; i++) { if(inVal & (1 << i)) outVal |= 1 << (bits - 1 - i); } return outVal; } /** ******************************************************************************* * @brief 获取 CRC(循环冗余校验) 函数 * @param [in] type - CRC 类型 * @param [in] *buf - 数据块指针 * @param [in] bufLen - 数据长度 * @return 校验码 * @note ******************************************************************************* */ unsigned int GetCRC(unsigned char type, unsigned char *buf, unsigned int bufLen) { unsigned char width = s_crcInfoTab[type].Width; //宽度,即CRC比特数。 unsigned int crc = s_crcInfoTab[type].CrcInit; //初始值,这是算法开始时寄存器(crc)的初始化预置值,十六进制表示。 unsigned int xorout = s_crcInfoTab[type].XorOut; //计算结果与此参数异或后得到最终的CRC值。 unsigned char refin = s_crcInfoTab[type].RefIn; //待测数据的每个字节是否按位反转,E_TRUE或E_FALSE。 unsigned char refout = s_crcInfoTab[type].RefOut; //在计算后之后,异或输出之前,整个数据是否按位反转,E_TRUE或E_FALSE。 unsigned char high; if(refin) //逆序 LSB 输入 { crc = BitsReverse(crc, width); //init 先逆序; if(width > 8) //为了减少移位等操作,width大于8和小于8的分开处理 { while(bufLen--) { crc = (crc >> 8) ^ s_crcTab[(crc & 0xFF) ^ *buf++]; } } else { while(bufLen--) { crc = s_crcTab[crc ^ *buf++]; } } } else //正序 MSB 输入 { if(width > 8) //为了减少移位等操作,width大于8和小于8的分开处理 { while (bufLen--) { high = crc >> (width - 8); crc = (crc << 8) ^ s_crcTab[high ^ *buf++]; } } else { crc = crc << (8 - width); while (bufLen--) { crc = s_crcTab[crc ^ *buf++]; } crc >>= 8 - width; //位数小于8时,crc在高width位,要右移到原位 } } if(refout != refin) //逆序输出 { crc = BitsReverse(crc, width); } crc ^= xorout; //异或输出 return crc & ((2 << (width - 1)) - 1); } /** ******************************************************************************* * @brief 打印CRC表 函数 * @param [in] type - CRC算法类型 * @return None * @note ******************************************************************************* */ void PrintfCrcTab(unsigned char type) { unsigned int i; printf("s_crcTab = \n{//%s", s_crcInfoTab[type].Name); for(i=0; i<256; i++) { if(i%16==0) printf("\n "); if(s_crcInfoTab[type].Width < 9) printf("0x%02X, ", s_crcTab[i]); else if(s_crcInfoTab[type].Width < 17) printf("0x%04X, ", s_crcTab[i]); else printf("0x%08X, ", s_crcTab[i]); } printf("\n};\n\n"); } /** ******************************************************************************* * @brief CRC表创建 函数 * @param [in] type - CRC算法类型 * @return None * @note ******************************************************************************* */ void CrcTableCalculate(unsigned char type) { unsigned char width = s_crcInfoTab[type].Width; //宽度,即CRC比特数。 unsigned int poly = s_crcInfoTab[type].Poly; //生成多项式的简写,以16进制表示。例如:CRC-32即是0x04C11DB7,忽略了最高位的"1",即完整的生成项是0x104C11DB7。 unsigned char refIn = s_crcInfoTab[type].RefIn; //待测数据的每个字节是否按位反转,E_TRUE或E_FALSE。 unsigned int validBits = (2 << (width - 1)) - 1; unsigned int value; unsigned int bit; unsigned int i; unsigned char j; if(refIn) //逆序LSB输入 { poly = BitsReverse(poly, width); //poly先逆序, s_crcTab 的 CrcInit = 0; for(i=0; i<256; i++) { value = i; for(j=0; j<8; j++) { if(value & 1) { value = (value >> 1) ^ poly; } else { value = (value >> 1); } } s_crcTab[i] = value & validBits; } } else //正序MSB输入 { poly = (width < 8) ? (poly << (8 - width)) : poly; //如果位数小于8,poly要左移到最高位 bit = (width > 8) ? (1 << (width - 1)) : 0x80; for(i=0; i<256; i++) { value = (width > 8) ? (i << (width - 8)) : i; for(j=0; j<8; j++) { if(value & bit) { value = (value << 1) ^ poly; } else { value = (value << 1); } } s_crcTab[i] = value & ((width < 8) ? 0xFF : validBits); } } PrintfCrcTab(type); } /** ******************************************************************************* * @brief 主函数 * @param [in] None * @return None * @note 验证校验函数 ******************************************************************************* */ void main() { char *str = "Hello World!"; unsigned int i; for(i=0; i<CRC_NUM; i++) { CrcTableCalculate(i); //CRC表创建 printf("%s (\"%s\") = 0x%0X\n\n", s_crcInfoTab[i].Name, str, GetCRC(i, (unsigned char*)str, strlen(str))); } while(1); }

运行结果:(篇幅大,截取了部分输出)

版权声明: 本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若内容造成侵权/违法违规/事实不符,请联系邮箱:809451989@qq.com进行投诉反馈,一经查实,立即删除!
网站建设 2026/6/18 8:48:58

肝火旺还是胃火旺?1分钟分清5种上火,喝对降火茶

降火第一步&#xff1a;分清你身上烧的是哪把"火"很多人一上火就喝凉茶、吃牛黄解毒片&#xff0c;结果火没降下来&#xff0c;胃先不舒服了。原因很简单&#xff1a;上火分好几种&#xff0c;降法完全不同。用错方法&#xff0c;不仅无效&#xff0c;还可能火上浇油…

作者头像 李华
网站建设 2026/6/18 8:41:50

计算机Java毕设实战-基于 Spring Boot 的高校科研纵向项目全流程管控系统的设计与实现 基于 Spring Boot 的高校纵向科研【完整源码+LW+部署说明+演示视频,全bao一条龙等】

博主介绍&#xff1a;✌️码农一枚 &#xff0c;专注于大学生项目实战开发、讲解和毕业&#x1f6a2;文撰写修改等。全栈领域优质创作者&#xff0c;博客之星、掘金/华为云/阿里云/InfoQ等平台优质作者、专注于Java、小程序技术领域和毕业项目实战 ✌️技术范围&#xff1a;&am…

作者头像 李华
网站建设 2026/6/18 8:23:08

大模型/深度学习/机器学习PPT绘图模板

适用于从事大模型、深度学习、机器学习相关领域的模型结构图绘制&#xff0c;所有模板的线条、结构、文字均支持100% 自定义修改&#xff01;无论是调整线条粗细、改变节点颜色&#xff0c;还是替换专业术语&#xff0c;都能轻松实现。 共100页PPT&#xff0c;包含了100图片模…

作者头像 李华
网站建设 2026/6/18 8:22:00

Takahashi凸性结构与非对称赋范空间理论解析

1. Takahashi凸性结构的基本概念与应用背景在数学分析领域&#xff0c;凸性结构的研究一直是几何与泛函分析的重要课题。Takahashi凸性结构作为一种广义的凸性概念&#xff0c;突破了传统线性空间的限制&#xff0c;为在更一般的度量空间中建立凸性理论提供了框架。这种结构最初…

作者头像 李华