给stm32cubeide编译出来的bin文件追加crc32

article/2025/7/12 19:36:58

在工程目录下创建ci目录,将AddCrc32.exe丢进去,在stm32cubeide的properties----C/C++ Build----Settings----Build Steps----Post-build steps Command:添加AddCrc32.exe的路径:

source code如下:

#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#include <stdint.h>
#include <direct.h>
#include <io.h>#define MAX_PATH    256
#define POLYNOMIAL  0xEDB88320#define LOOK_UP_TABLE       1#if (LOOK_UP_TABLE == 1)
const unsigned long crc32_table[] = {0x00000000, 0x77073096, 0xEE0E612C, 0x990951BA, 0x076DC419, 0x706AF48F, 0xE963A535, 0x9E6495A3, 0x0EDB8832, 0x79DCB8A4, 0xE0D5E91E, 0x97D2D988, 0x09B64C2B, 0x7EB17CBD, 0xE7B82D07, 0x90BF1D91,0x1DB71064, 0x6AB020F2, 0xF3B97148, 0x84BE41DE, 0x1ADAD47D, 0x6DDDE4EB, 0xF4D4B551, 0x83D385C7, 0x136C9856, 0x646BA8C0, 0xFD62F97A, 0x8A65C9EC, 0x14015C4F, 0x63066CD9, 0xFA0F3D63, 0x8D080DF5,0x3B6E20C8, 0x4C69105E, 0xD56041E4, 0xA2677172, 0x3C03E4D1, 0x4B04D447, 0xD20D85FD, 0xA50AB56B, 0x35B5A8FA, 0x42B2986C, 0xDBBBC9D6, 0xACBCF940, 0x32D86CE3, 0x45DF5C75, 0xDCD60DCF, 0xABD13D59,0x26D930AC, 0x51DE003A, 0xC8D75180, 0xBFD06116, 0x21B4F4B5, 0x56B3C423, 0xCFBA9599, 0xB8BDA50F, 0x2802B89E, 0x5F058808, 0xC60CD9B2, 0xB10BE924, 0x2F6F7C87, 0x58684C11, 0xC1611DAB, 0xB6662D3D,0x76DC4190, 0x01DB7106, 0x98D220BC, 0xEFD5102A, 0x71B18589, 0x06B6B51F, 0x9FBFE4A5, 0xE8B8D433, 0x7807C9A2, 0x0F00F934, 0x9609A88E, 0xE10E9818, 0x7F6A0DBB, 0x086D3D2D, 0x91646C97, 0xE6635C01,0x6B6B51F4, 0x1C6C6162, 0x856530D8, 0xF262004E, 0x6C0695ED, 0x1B01A57B, 0x8208F4C1, 0xF50FC457, 0x65B0D9C6, 0x12B7E950, 0x8BBEB8EA, 0xFCB9887C, 0x62DD1DDF, 0x15DA2D49, 0x8CD37CF3, 0xFBD44C65,0x4DB26158, 0x3AB551CE, 0xA3BC0074, 0xD4BB30E2, 0x4ADFA541, 0x3DD895D7, 0xA4D1C46D, 0xD3D6F4FB, 0x4369E96A, 0x346ED9FC, 0xAD678846, 0xDA60B8D0, 0x44042D73, 0x33031DE5, 0xAA0A4C5F, 0xDD0D7CC9,0x5005713C, 0x270241AA, 0xBE0B1010, 0xC90C2086, 0x5768B525, 0x206F85B3, 0xB966D409, 0xCE61E49F, 0x5EDEF90E, 0x29D9C998, 0xB0D09822, 0xC7D7A8B4, 0x59B33D17, 0x2EB40D81, 0xB7BD5C3B, 0xC0BA6CAD,0xEDB88320, 0x9ABFB3B6, 0x03B6E20C, 0x74B1D29A, 0xEAD54739, 0x9DD277AF, 0x04DB2615, 0x73DC1683, 0xE3630B12, 0x94643B84, 0x0D6D6A3E, 0x7A6A5AA8, 0xE40ECF0B, 0x9309FF9D, 0x0A00AE27, 0x7D079EB1,0xF00F9344, 0x8708A3D2, 0x1E01F268, 0x6906C2FE, 0xF762575D, 0x806567CB, 0x196C3671, 0x6E6B06E7, 0xFED41B76, 0x89D32BE0, 0x10DA7A5A, 0x67DD4ACC, 0xF9B9DF6F, 0x8EBEEFF9, 0x17B7BE43, 0x60B08ED5,0xD6D6A3E8, 0xA1D1937E, 0x38D8C2C4, 0x4FDFF252, 0xD1BB67F1, 0xA6BC5767, 0x3FB506DD, 0x48B2364B, 0xD80D2BDA, 0xAF0A1B4C, 0x36034AF6, 0x41047A60, 0xDF60EFC3, 0xA867DF55, 0x316E8EEF, 0x4669BE79,0xCB61B38C, 0xBC66831A, 0x256FD2A0, 0x5268E236, 0xCC0C7795, 0xBB0B4703, 0x220216B9, 0x5505262F, 0xC5BA3BBE, 0xB2BD0B28, 0x2BB45A92, 0x5CB36A04, 0xC2D7FFA7, 0xB5D0CF31, 0x2CD99E8B, 0x5BDEAE1D,0x9B64C2B0, 0xEC63F226, 0x756AA39C, 0x026D930A, 0x9C0906A9, 0xEB0E363F, 0x72076785, 0x05005713, 0x95BF4A82, 0xE2B87A14, 0x7BB12BAE, 0x0CB61B38, 0x92D28E9B, 0xE5D5BE0D, 0x7CDCEFB7, 0x0BDBDF21,0x86D3D2D4, 0xF1D4E242, 0x68DDB3F8, 0x1FDA836E, 0x81BE16CD, 0xF6B9265B, 0x6FB077E1, 0x18B74777, 0x88085AE6, 0xFF0F6A70, 0x66063BCA, 0x11010B5C, 0x8F659EFF, 0xF862AE69, 0x616BFFD3, 0x166CCF45,0xA00AE278, 0xD70DD2EE, 0x4E048354, 0x3903B3C2, 0xA7672661, 0xD06016F7, 0x4969474D, 0x3E6E77DB, 0xAED16A4A, 0xD9D65ADC, 0x40DF0B66, 0x37D83BF0, 0xA9BCAE53, 0xDEBB9EC5, 0x47B2CF7F, 0x30B5FFE9,0xBDBDF21C, 0xCABAC28A, 0x53B39330, 0x24B4A3A6, 0xBAD03605, 0xCDD70693, 0x54DE5729, 0x23D967BF, 0xB3667A2E, 0xC4614AB8, 0x5D681B02, 0x2A6F2B94, 0xB40BBE37, 0xC30C8EA1, 0x5A05DF1B, 0x2D02EF8D,
};unsigned long calculate_crc32(const unsigned char* data, size_t length) 
{unsigned long crc = 0xFFFFFFFF;unsigned char index = 0;for (size_t i = 0; i < length; i++) {index = (crc ^ data[i]) & 0xFF;crc = (crc >> 8) ^ crc32_table[index];}return crc ^ 0xFFFFFFFF;
}#else
// CRC32 表生成函数
void generate_crc32_table(unsigned int* crc_table) {unsigned long polynomial = 0xEDB88320;unsigned long crc = 0;unsigned long i = 0;unsigned long j = 0;for (i = 0; i < 256; i++) {crc = i;for (j = 8; j > 0; j--) {if (crc & 1) {crc = (crc >> 1) ^ polynomial;} else {crc >>= 1;}}crc_table[i] = crc;}
}// 计算数据的 CRC32 值
unsigned long calculate_crc32(const unsigned char* data, size_t length)
{static unsigned long crc_table[256] = { 0 };unsigned long crc = 0xFFFFFFFF;unsigned char index = 0;size_t i = 0;if (crc_table[0] == 0) {generate_crc32_table(crc_table);}for (i = 0; i < length; i++) {index = (crc ^ data[i]) & 0xFF;crc = (crc >> 8) ^ crc_table[index];}return crc ^ 0xFFFFFFFF;
}
#endifbool rename_bin_file(char* src, char* dest)
{unsigned short i = 0;char append_string[] = "_AddCrc32.bin";bool is_append_valid = false;for (i = 0; i < MAX_PATH; i++) {if (src[i + 0] == '.' && src[i + 1] == 'b' && src[i + 2] == 'i' && src[i + 3] == 'n') {strncpy_s(dest, MAX_PATH, src, i);dest[i] = '\0';strcat_s(dest, MAX_PATH, "_AddCrc32.bin");is_append_valid = true;break;}}return is_append_valid;
}int main() {char current_path[MAX_PATH] = { 0 };char target_path[MAX_PATH] = { 0 };char dest_path[MAX_PATH] = { 0 };struct _finddata_t file_info = { 0 };intptr_t handle = 0;bool is_append_valid = false;FILE* input_file = NULL;FILE* output_file = NULL;unsigned char* buffer = NULL;size_t file_size = 0;unsigned long crc32 = 0;char dest_file_name[32] = { 0 };// 写入 CRC32 (4字节,小端格式)unsigned char crc_bytes[4] = { 0 };// 获取当前目录if (_getcwd(current_path, sizeof(current_path)) == NULL) {perror("Failed to retrieve the current directory!\n");return 1;}printf("current directory: %s\n", current_path);// 构建目标目录路径:跳到上一级,再进入Debug目录strcpy_s(target_path, MAX_PATH, current_path);char* last_slash = strrchr(target_path, '\\');if (last_slash != NULL) {*last_slash = '\0';  // 截断到上一级目录strcat_s(target_path, MAX_PATH, "\\Debug");} else {printf("Unable to determine the previous directory!\n");return 1;}printf("the previous directory: %s\n", target_path);// 检查目标目录是否存在if (_chdir(target_path) != 0) {printf("destination directory does not exist: %s\n", target_path);return 1;}// 查找.bin文件if ((handle = _findfirst("*.bin", &file_info)) == -1L) {printf("In the diretory %s, could not find .bin file\n", target_path);return 1;}// 打印找到的第一个.bin文件路径strcat_s(target_path, MAX_PATH, "\\");strcat_s(target_path, MAX_PATH, file_info.name);printf("found .bin file: %s\n", target_path);printf("file name: %s\n", file_info.name);_findclose(handle);is_append_valid = rename_bin_file(target_path, dest_path);if (is_append_valid) {printf("target file path: %s\n", dest_path);}fopen_s(&input_file, target_path, "rb");if (input_file == NULL) {perror("Unable to open input file\n");//system("pause");return 1;}fseek(input_file, 0, SEEK_END);file_size = ftell(input_file);fseek(input_file, 0, SEEK_SET);buffer = (unsigned char*)malloc(file_size);if (buffer == NULL) {perror("memory allocation failed!\n");fclose(input_file);//system("pause");return 1;}if (fread(buffer, 1, file_size, input_file) != file_size) {perror("fail to read file\n");free(buffer);fclose(input_file);//system("pause");return 1;}fclose(input_file);// 计算 CRC32crc32 = calculate_crc32(buffer, file_size);printf("Original file CRC32: 0x%08X\n", crc32);crc_bytes[0] = (unsigned char)(crc32 & 0xFF);crc_bytes[1] = (unsigned char)((crc32 >> 8) & 0xFF);crc_bytes[2] = (unsigned char)((crc32 >> 16) & 0xFF);crc_bytes[3] = (unsigned char)((crc32 >> 24) & 0xFF);// 打开输出文件fopen_s(&output_file, dest_path, "wb");if (output_file == NULL) {perror("Unable to open output file\n");free(buffer);//system("pause");return 1;}// 写入原始数据if (fwrite(buffer, 1, file_size, output_file) != file_size) {perror("fail to write to file\n");free(buffer);fclose(output_file);//system("pause");return 1;}if (fwrite(crc_bytes, 1, 4, output_file) != 4) {perror("Writing CRC32 failed!\n");free(buffer);fclose(output_file);//system("pause");return 1;}// 关闭文件并释放内存fclose(output_file);free(buffer);rename_bin_file(file_info.name, dest_file_name);printf("Processing completed, saved to %s\n", dest_file_name);//system("pause");return 0;
}

http://www.hkcw.cn/article/WlHxHFGOEM.shtml

相关文章

算法-集合的使用

1、set常用操作 set<int> q; //以int型为例 默认按键值升序 set<int,greater<int>> p; //降序排列 int x; q.insert(x); //将x插入q中 q.erase(x); //删除q中的x元素,返回0或1,0表示set中不存在x q.clear(); //清空q q.empty(); //判断q是否为空&a…

网络地址转换

网络地址转换 网络地址转换(Network Address Translation&#xff0c;NAT)的功能是将企业内部自行定义的私有IP地址转换为Internet上可识别的合法IP地址。由于现行IP地址标准--IPv4的限制&#xff0c;Internet面临着IP地址空间短缺的问题&#xff0c;因此从ISP申请并给企业的每…

4.大语言模型预备数学知识

大语言模型预备数学知识 复习一下在大语言模型中用到的矩阵和向量的运算&#xff0c;及概率统计和神经网络中常用概念。 矩阵的运算 矩阵 矩阵加减法 条件&#xff1a;行数列数相同的矩阵才能做矩阵加减法 数值与矩阵的乘除法 矩阵乘法 条件&#xff1a;矩阵A的列数 矩阵…

leetcode hot100刷题日记——35.子集

解答&#xff1a; 方法一&#xff1a;选or不选的dfs&#xff08;输入视角&#xff09; 思路&#xff1a;[1,2,3]的全部子集可以看成是对数组的每一位数字做选择。 eg.空集就是一个数字都不选&#xff0c;[1,2]就是1&#xff0c;2选&#xff0c;3不选。 class Solution { pub…

【数据库】关系数据库标准语言-SQL(金仓)下

4、数据查询 语法&#xff1a; SELECT [ALL | DISTINCT] <目标列表达式> [,<目标列表达式>] … FROM <表名或视图名>[, <表名或视图名> ] … [ WHERE <条件表达式> ] [ GROUP BY <列名1> [ HAVING <条件表达式> ] ] [ ORDER BY <…

好用的C/C++/嵌入式 IDE: CLion的下载安装教程(保姆级教程)

CLion简介 CLion是由著名的JetBrains公司开发出的一个C/C的IDE。它原是付费软件&#xff0c;但在最近(指2025年5月)开放了非商业用途免费&#xff0c;就像WebStorm、Rider、RustRover等。 除了这些&#xff0c;JetBrains的IntelliJ IDEA(社区版)和PyCharm(社区版)也是免费的。…

SpringBoot统一功能处理

1.拦截器 拦截器是Spring框架提供的核心功能之一,主要是用来拦截用户的请求,在指定方法前后,根据业务需要执行预先设定的…

prometheus v3.4.1正式发布!解析全新特性与安装指南,打造高效云原生监控体系

一、引言 随着云原生时代的快速发展&#xff0c;监控系统成为保障业务平稳运行的核心利器。作为CNCF&#xff08;Cloud Native Computing Foundation&#xff09;旗下的开源监控项目&#xff0c;Prometheus凭借其卓越的多维数据模型、灵活强大的查询语言及自主运行的架构设计&a…

PCA(K-L变换)人脸识别(python实现)

数据集分析 ORL数据集&#xff0c; 总共40个人&#xff0c;每个人拍摄10张人脸照片 照片格式为灰度图像&#xff0c;尺寸112 * 92 特点&#xff1a; 图像质量高&#xff0c;无需灰度运算、去噪等预处理 人脸已经位于图像正中央&#xff0c;但部分图像角度倾斜&#xff08;可…

资源预加载+懒加载组合拳:从I/O拖慢到首帧渲染的全面优化方案

简介 在移动应用开发领域,首帧渲染性能已成为用户体验的关键指标之一。根据2025年最新行业数据,首屏加载时间每延迟1秒,用户跳出率可能增加32%,直接影响应用评分和留存率。当应用启动时,布局解析、图片解码等I/O操作往往成为首帧渲染的主要瓶颈,导致用户看到白屏或黑屏时…

【Doris基础】Apache Doris中的Coordinator节点作用详解

目录 1 Doris架构概述 2 Coordinator节点的核心作用 2.1 查询协调与调度 2.2 执行计划生成与优化 2.3 资源管理与负载均衡 2.4 容错与故障恢复 3 Coordinator节点的关键实现机制 3.1 两阶段执行模型 3.2 流水线执行引擎 3.3 分布式事务管理 4 Coordinator节点的高可…

【基于阿里云搭建数据仓库(离线)】IDEA导出Jar包(包括第三方依赖)

1.双击"package”即可进行打包呈jar 2.双击后就会自动打包生成jar了&#xff0c; 生成的jar在这个目录下 3.右击&#xff0c;点击“复制路径/引用”&#xff0c;即可获得“绝对路径”、“根路径”等相关信息

id()函数:窥探Python变量内存地址的奥秘

在Python程序设计中&#xff0c;变量、对象和内存是紧密相连的核心概念。理解变量的内存地址&#xff0c;是理解Python变量本质、内存管理与性能优化的关键。Python内置函数id()&#xff0c;作为变量与对象身份&#xff08;identity&#xff09;的“指纹识别器”&#xff0c;为…

MySQL中的事务

事物特性 原子性:事物时最小的执行单位&#xff0c;不允许分割。事物的原子性确保动作要么全部完成&#xff0c;要么完全不起作用&#xff0c;如果在执行过程中发生错误&#xff0c;会被回滚到事物开始前的状态&#xff0c;就像这个事务从来没有执行过一样。一致性&#xff1a…

像素转换案例实战

本案例介绍像素单位的基本知识与像素单位转换API的使用。通过像素转换案例&#xff0c;向开发者讲解了如何使用像素单位设置组件的尺寸、字体的大小以及不同像素单位之间的转换方法。主要功能包括&#xff1a; 展示了不同像素单位的使用。展示了像素单位转换相关API的使用。 …

结构型设计模式之桥接模式

文章目录 1. 桥接模式概述2. 模式结构3. 桥接模式的优缺点优点缺点 4. 桥接模式的应用场景5. C#代码示例5.1 简单示例 - 形状与颜色5.2 更复杂的示例 - 跨平台消息发送系统 6. 桥接模式与其他模式的比较7. 真实世界中的桥接模式应用7.1 数据库驱动7.2 UI框架中的渲染机制 8. 桥…

RAG系统中如何检测幻觉?

虽然我们的 RAG 系统通过将答案基于真实的医学证据来减少幻觉,但我们发现了一个关键的差距:即使有引用,系统仍然可能产生不可靠的输出。 想想看:仅仅因为一个系统可以引用来源,并不意味着它正确地使用了这些来源。 模型可能会: 从检索到的文档中提取不相关的信息不适当…

world quant教程学习

Understanding Corporate Fundamental Data &#x1f50d; 了解企业基本面数据 Lets explore fundamental data&#x1f60a; Fundamentals capture the underlying business, financial and operational health of a company, usually reported every quarter. This data is t…

详解鸿蒙仓颉开发语言中的计时器

今天又到了大家喜闻乐见的科普环节&#xff0c;也可以说是踩坑环节&#xff0c;哈哈哈。今天聊一聊仓颉开发语言中的计时器&#xff0c;这部分可老有意思了。 为什么这么说呢&#xff0c;因为关于仓颉的计时器你几乎搜不到任何的文档&#xff0c;也没有相关的代码提示&#xf…

70多套创业商业融资计划书PPT模板分享

70多套创业商业融资计划书PPT模板分享&#xff0c;商业计划书、融资计划书为主的欧美风格PPT模板。 70多套创业商业融资计划书PPT模板分享&#xff1a;创业商业融资计划书PPT模板https://pan.quark.cn/s/e09456cd487b