嵌入式开发:STM32 硬件 CRC 使用

embedded/2024/10/11 4:23:53/

测试平台:STM32G474系列
        STM32硬件的CRC不占用MCU的资源,计算速度快。由于硬件CRC需要配置一些选项,配置不对就会导致计算结果错误,导致使用上没有软件计算CRC方便。但硬件CRC更快的速度在一些有时间资源要求的场合还是非常适合,没计算时间要求的还是用软件CRC更方便通用。

1.软件CRC16代码

//*****************************************************************************
//
//! 计算所选 CRC 多项式的 16 位 CRC。
//! \fn uint16_t calculateCRC(const uint8_t dataBytes[], uint8_t numberBytes, uint16_t initialValue)
//!
//! \param dataBytes[] 指向数据字节数组中第一个元素的指针
//! \param numberBytes CRC计算中使用的字节数
//! \param initialValue 初始值,第一次使用时使用0xFFFF,循环计算时输入上一次的结果
//! \return 16-bit  CRC16 结果
//
//*****************************************************************************
uint16_t calculateCRC(const uint8_t dataBytes[], uint8_t numberBytes, uint16_t initialValue)
{int         bitIndex, byteIndex;bool        dataMSb;						bool        crcMSb;						    uint8_t     bytesPerWord = wlength_byte_values[WLENGTH];uint16_t crc = initialValue;#ifdef CRC_CCITT  //多项式公式/* CCITT CRC polynomial = x^16 + x^12 + x^5 + 1 */const uint16_t poly = 0x1021;#endif#ifdef CRC_ANSI/* ANSI CRC polynomial = x^16 + x^15 + x^2 + 1 */const uint16_t poly = 0x8005;#endiffor (byteIndex = 0; byteIndex < numberBytes; byteIndex++){bitIndex = 0x80u;while (bitIndex > 0){dataMSb = (bool) (dataBytes[byteIndex] & bitIndex);crcMSb  = (bool) (crc & 0x8000u);crc <<= 1;              if (dataMSb ^ crcMSb){crc ^= poly;        }bitIndex >>= 1;}}return crc;
}

2.STM32CubeMX CRC配置

①选择 CRC 并开启

②这里选择以CRC16来测试

③多项式,用于计算CRC16时的多项式,这个可以后面在生成的代码里面直接改。

④计算时的初始值,软件计算代码那边对应的是 initialValue 参数。

配置完成后生成代码

\Core\Src 路径下找到生成的工程里面的 crc.c文件

在以下代码中修改多项式的参数:

多项式参数,根据多项式公式算成对应的 16进制值然后赋值。

void MX_CRC_Init(void)
{/* USER CODE BEGIN CRC_Init 0 *//* USER CODE END CRC_Init 0 *//* USER CODE BEGIN CRC_Init 1 *//* USER CODE END CRC_Init 1 */hcrc.Instance = CRC;hcrc.Init.DefaultPolynomialUse = DEFAULT_POLYNOMIAL_DISABLE;hcrc.Init.DefaultInitValueUse = DEFAULT_INIT_VALUE_DISABLE;hcrc.Init.GeneratingPolynomial = 0x1021;                     //在这里修改多项式hcrc.Init.CRCLength = CRC_POLYLENGTH_16B;hcrc.Init.InitValue = 0xFFFF;                                //在这里修改初始值hcrc.Init.InputDataInversionMode = CRC_INPUTDATA_INVERSION_NONE;hcrc.Init.OutputDataInversionMode = CRC_OUTPUTDATA_INVERSION_DISABLE;hcrc.InputDataFormat = CRC_INPUTDATA_FORMAT_BYTES;if (HAL_CRC_Init(&hcrc) != HAL_OK){Error_Handler();}/* USER CODE BEGIN CRC_Init 2 *//* USER CODE END CRC_Init 2 */}

多项式值的计算或获取

直接获取:http://www.ip33.com/crc.html

可以通过以上网站直接计算CRC值,和获取对应多项式的16进制值。

如这次实验使用的多项式: CRC-16/CCITT   :  x16 + x12 + x5 + 1

自己计算:CRC-16/CCITT   :  x16 + x12 + x5 + 1

x16 + x12 + x5 + 1: 表示的是一个二进制数, x16表示第16位是1, x12表示第12位是1,x5表示第5位是1,其余的位置都是0。 二进制位值是从0开始计数。对应二进制值如下

x16 + x12 + x5 = 1   0001   0000  0010  0000 = 0x11020

x16 + x12 + x5 + 1 = 0x11020 +1 = 0x11021

由于是CRC16 所以取2个字节值 = 0x1021

3.CRC 库函数

参考链接:https://blog.csdn.net/usjjjsj/article/details/141832938

uint32_t HAL_CRC_Accumulate(CRC_HandleTypeDef *hcrc, uint32_t pBuffer[], uint32_t BufferLength)

 以上一次CRC校验的结果作为初始值继续进行校验 (适用于连续多次校验的第2、3、4... ...次)


*hcrc: 指向 CRC_HandleTypeDef CRC 校验总控制结构体的指针
pBuffer:待校验的数据
BufferLength:待校验的数据长度
返回值:校验结果
        该函数在第一使用时需要调用HAL_CRC_Calculate,计算出第一次数据的校验位,然后由第一位的数据位的校验位作为下一位的的初始值。计算出最后一位的数据位作为整个传递数据的校验位。

uint32_t HAL_CRC_Calculate(CRC_HandleTypeDef *hcrc, uint32_t pBuffer[], uint32_t BufferLength)

使用默认初始值进行校验计算 (适用于单次校验 或 多次校验的第一次)

该函数一次性将全部数据的校验位检测出来,且初值仍为0xFFFFFFFF,或者CRC初始化时配置的初始值。

HAL_CRC_StateTypeDef HAL_CRC_GetState(CRC_HandleTypeDef *hcrc)

获取状态的函数

HAL_CRC_StateTypeDef HAL_CRC_GetState(CRC_HandleTypeDef *hcrc
返回值:CRC校验总控制结构体内的 State 值
HAL_CRC_STATE_RESET  尚未初始化
HAL_CRC_STATE_READY 初始化并准备使用
HAL_CRC_STATE_BUSY 忙
HAL_CRC_STATE_TIMEOUT 超时
HAL_CRC_STATE_ERROR 错误

4.CRC计算

经过以上配置后,通过软件计算的CRC和硬件计算的CRC将获得相同的结果,如果有高低字节不同,可以自己调整一下。

HAL_CRC_Calculate(); //硬件计算

calculateCRC(); //软件计算,文章开始有对应的代码。

网站在线计算可作计算结果的对照:http://www.ip33.com/crc.html

感谢这两位博主的文章,在这两份文章的帮助下我成功的通过硬件CRC计算出了正确的值。

https://blog.csdn.net/usjjjsj/article/details/141832938

https://blog.csdn.net/13011803189/article/details/122366072


http://www.ppmy.cn/embedded/125700.html

相关文章

Django-rest-framework(DRF)怎么使用redis

一、redis作用 Redis&#xff0c;作为一款开源的内存数据结构存储系统&#xff0c;它可以用作数据库、缓存和消息中间件。以下是Redis的主要作用&#xff1a; 1、高速缓存 Redis能够极大地加速数据的读取速度&#xff0c;因为它将数据存储在内存中通过缓存热点数据&#xff…

运维工具之ansible

Ansible 1.什么是ansible? ​ ansible是基于ssh架构的自动化运维工具&#xff0c;由python语言实现&#xff0c;通过ansible可以远程批量部署等。 2.部署前提 ​ 控制端需要安装ansible,被控制端要开启ssh服务&#xff0c;并允许远程登录&#xff0c;被管理主机需要安装py…

喜讯!迈威通信TSN产品通过“时间敏感网络(TSN)产业链名录计划”评测,各项指标名列前茅

TSN技术&#xff0c;作为推动企业网络化与智能化转型的关键力量&#xff0c;已成为工业网络迈向下一代演进的共识方向&#xff0c;正加速重构工业网络的技术架构与产业生态。为响应这一趋势&#xff0c;工业互联网产业联盟携手中国信息通信研究院及50余家产学研用单位&#xff…

Qt QPushButton clicked信号浅析

前言 Qt 的 QPushButton clicked 信号原型&#xff1a; void clicked(bool checked false);通常&#xff0c;使用 Qt 的 QPushButton clicked 点击信号时&#xff0c;会以如下方式使用&#xff1a; connect(ui->pushButton, &QPushButton::clicked, this, [](){qDeb…

影视cms泛目录用什么程序?苹果cms二次开发泛目录插件

影视CMS泛目录一般使用的程序有很多种&#xff0c;&#xff08;maccmscn&#xff09;以下是其中几种常见的程序&#xff1a; WordPress&#xff1a;WordPress是一个非常流行的开源内容管理系统&#xff0c;可以通过安装一些插件来实现影视CMS泛目录功能。其中&#xff0c;一款常…

R包:APAlyzer从RNA-seq数据计算APA表达丰度

文章目录 介绍教程实战案例数据脚本运行 介绍 今天安利APAlyzer工具&#xff0c;它是通过RNA-seq数据获取3′UTR APA, intronic APA等表达谱的R包。 APAlyzer将bam文件比对到PolyA-DB数据库识别APA。 Most eukaryotic genes produce alternative polyadenylation (APA) isofo…

YOLOv10改进,YOLOv10添加CA注意力机制,二次创新C2f结构,助力涨点

改进前训练结果: 二次创新C2f结构训练结果: 摘要 在本文中,提出了一种新的移动网络注意力机制,将位置信息嵌入到信道注意力中称之为“协调注意力”。与渠道关注不同通过 2D 全局池将特征张量转换为单个特征向量,坐标注意力因子将通道注意力转化为两个 1D 特征编码过程…

OpenAI .NET 库稳定版发布,支持 GPT-4o 并改进 API 功能

penAI 在6月推出其官方 .NET 库的 beta 版之后&#xff0c;如今终于发布了稳定版。该库已在 NuGet 上作为包发布&#xff0c;支持最新的模型&#xff0c;如 GPT-4o 和 GPT-4o mini&#xff0c;并且提供完整的 OpenAI REST API。这次发布包括同步和异步 API&#xff0c;以及流式…