HEX与BIN文件格式详解及嵌入式开发应用

张开发
2026/4/10 2:47:42 15 分钟阅读

分享文章

HEX与BIN文件格式详解及嵌入式开发应用
1. 文件格式的本质差异第一次接触单片机开发时我也曾被HEX和BIN文件搞得晕头转向。直到有次烧录出错导致硬件锁死才真正明白这两种格式的区别绝非表面那么简单。HEX文件本质上是一种带地址信息的文本编码格式而BIN文件则是纯粹的二进制映像。举个例子当你用文本编辑器打开HEX文件时会看到类似:100000000C9434000C9446000C9446000C9446006A的字符串而BIN文件打开后全是乱码——这就是最直观的区别。HEX文件采用Intel HEX格式规范每条记录包含起始符:字节数如10表示16字节地址域4字符十六进制数记录类型00数据记录/01文件结束等数据域n字节十六进制数据校验和而BIN文件就像内存的照片直接按顺序存储机器码。比如对于STM32F1030x08000000地址的代码在BIN文件中就位于文件开头没有任何元数据。关键提示HEX文件中的地址可能不连续但BIN文件必须保证地址连续性。这就是为什么有些HEX转换BIN工具需要指定填充值通常0xFF。2. 开发工具链中的处理差异在Keil MDK中编译生成HEX文件时编译器会执行以下关键步骤将各个.o目标文件链接成绝对地址的ELF文件提取.text、.data等段内容生成带地址标记的HEX记录添加启动地址记录通常是03类型的扩展线性地址记录而生成BIN文件时工具链的工作流程截然不同fromelf --bin -o project.bin project.axf这个命令直接从ELF文件提取二进制内容不做任何地址编码。我曾遇到过因忘记指定--bin参数导致生成错误格式的情况烧录后单片机直接跑飞。IAR Embedded Workbench的处理更有意思——它的HEX文件默认使用Motorola S-record格式与Intel HEX的差异就像大端序和小端序的区别。有次我将IAR生成的HEX文件直接给产线烧录结果设备全部无法启动排查半天才发现格式不兼容。3. 烧录过程的底层机制使用ST-Link烧录HEX文件时编程器会解析每条HEX记录的地址和数据自动处理地址跨页如STM32的Flash每页2KB根据记录类型判断文件结束计算并验证校验和而烧录BIN文件时则必须手动指定起始地址st-flash write firmware.bin 0x08000000这个0x08000000对STM32来说就是Flash的起始地址。有次我误写成0x8000000少个0结果程序烧到了RAM区上电后代码立刻丢失。J-Link的处理更智能些它的命令行工具可以自动检测HEX文件中的起始地址JLinkExe -device STM32F407VG -CommanderScript flash.jlink其中flash.jlink内容loadfile firmware.hex r q但对BIN文件仍需明确地址loadfile firmware.bin 0x080000004. 实际项目中的选型建议在汽车ECU开发中我们强制要求使用HEX格式因为产线烧录时需要验证地址范围避免误刷Bootloader区支持多段非连续地址如单独配置参数区内置校验机制降低不良率但在IoT设备OTA升级时BIN文件更合适文件体积小比HEX小30%-50%云端不需要解析复杂格式单片机内置Bootloader可直接解析有个经典案例某智能家居项目最初用HEX格式OTA导致升级包过大用户流量费用激增。改为BIN格式后单个升级包从120KB缩减到78KB每年节省数百万服务器流量成本。5. 格式转换的实用技巧使用objcopy工具转换时这个命令堪称万能arm-none-eabi-objcopy -O ihex firmware.elf firmware.hex arm-none-eabi-objcopy -O binary firmware.elf firmware.bin但要注意两个坑BIN文件转换可能丢失调试信息建议保留ELF文件跨工具链转换时注意字节序问题我常用的HEX转BIN方法是import intelhex ih intelhex.IntelHex(firmware.hex) ih.tobinfile(firmware.bin, start0x08000000)这个Python脚本可以精确控制输出范围比某些GUI工具更可靠。6. 校验机制深度解析HEX文件每条记录都有校验和计算公式是校验和 0x100 - (字节数 地址高字节 地址低字节 记录类型 所有数据字节之和的低8位)例如记录:100000000C9434000C9446000C9446000C9446006A的校验和0x6A验证0x100 - (0x10 0x00 0x00 0x00 0x0C0x940x340x000x0C0x940x460x00 0x0C0x940x460x000x0C0x940x460x00) 0x6A而BIN文件通常需要额外生成CRC校验。STM32的硬件CRC模块计算示例uint32_t calc_crc(const uint8_t *data, size_t len) { CRC-CR | CRC_CR_RESET; for(size_t i0; ilen; i4) { uint32_t word *(uint32_t*)(datai); CRC-DR __RBIT(word); // 字节序转换 } return __RBIT(CRC-DR); }7. 调试排错实战记录曾遇到一个诡异问题HEX文件烧录正常但转成BIN后运行异常。最终发现是链接脚本里这个配置作祟MEMORY { FLASH (rx) : ORIGIN 0x08000000, LENGTH 512K RAM (xrw) : ORIGIN 0x20000000, LENGTH 128K }HEX文件会跳过未使用的Flash区域而BIN文件会包含这些空白区域填充0xFF导致文件超大。解决方案是在objcopy时指定只提取.text段arm-none-eabi-objcopy -O binary -j .text firmware.elf firmware.bin另一个典型案例某客户反映HEX文件烧录后部分配置丢失。经查是HEX文件中包含多个相同地址的记录后一条覆盖了前一条。这种问题用BIN文件就不会出现因为每个地址只能对应一个字节。

更多文章