当前位置: 首页 > news >正文

调试`build.sh` 和用 `CMake` 编译出来的 `.elf` / `.bin` / `.hex` 文件大小或行为不同?

这是一个经典问题,在嵌入式开发中尤为重要。主要原因可以归结为以下几点:


1. 链接方式不同

build.sh 使用的是:

arm-none-eabi-gcc -c ...
arm-none-eabi-ld ... -T blue_pill.ld

CMake 默认使用的是:

arm-none-eabi-gcc ... -T blue_pill.ld

虽然都能生成 .elfgcc 调用 ld 时自动添加了一堆默认库和启动文件,如:

  • crt0.o
  • libc.a
  • libgcc.a
  • _start 入口
  • __libc_init_array() 调用

这会导致:

  • 生成文件大小变大
  • 运行逻辑不同(比如多了 libc 初始化)

2. 启动文件和链接脚本行为不同

你的 build.sh 明确用 -Ttext=$FLASH_BASE 和自定义 .ld,手动控制内存布局、入口地址。

而 CMake 如果没有强制指定

add_link_options(-Ttext=0x08000000 -T your_linker.ld)

或者:

add_link_options(-nostartfiles -nostdlib)

它很可能链接的是:

  • 错误的启动地址(默认 _start
  • 自动插入的 crt0.o
  • 新lib初始化代码
  • 更改了 stack, heap 默认符号(比如 _stack_end

3. 编译选项不同

你的 build.sh 用的是:

-O2 -fno-hosted -fno-inline -fno-align-functions -ffunction-sections -fdata-sections

但在 CMake 里你用了:

add_compile_options(-Og -g) # Debug

即便链接脚本一样,优化等级不同也会极大影响 .elf.bin 的大小,甚至行为(比如函数没被内联或链接器无法优化函数段合并)。


4. 汇编预处理路径不同

你用了 .s 文件且其中有 #include 其他 .s,这要求你确保:

add_compile_options(-Wa,-I${CMAKE_SOURCE_DIR}/src)

缺失这个会导致 descriptors_audio.s 没被编进 .elf,最终 .bin 更小,功能缺失。


5. 是否 strip/debug 信息

你的 build.sh 调用了 --strip-debug(虽然用错位置了,但有可能有效),而 CMake 没有。

.elf 大小区别 ≠ .bin 实际写入闪存的区别

strip 会影响 .elf 体积(调试符号被去除),在一些上传/烧录工具中甚至影响校验行为。


6. 默认链接库行为

如果你没有加 -nostdlib,GCC 会默认链接这些:

libc.a
libm.a
libgcc.a

它们会加入如下代码:

  • 浮点支持(可能不需要)
  • 除法函数(比如 __aeabi_uidiv
  • _exit() 等(如果你没实现可能链接失败)

🧪 ✅ 如何验证差异:

  1. 对比 .elf 的节区:
arm-none-eabi-size -A your.elf

你会看到:

  • .text, .data, .bss, .init, .fini 等差异
  • 看是否有 libgcc, crt0.o 的残留代码段
  1. nm 工具查看符号:
arm-none-eabi-nm -n your.elf | less

特别检查:

  • 是否有 _start, __libc_init_array, __aeabi_*, __libc_fini_array
  1. readelf -aobjdump 看指令节/布局。

🛠 推荐你在 CMake 中添加以下保证一致性:

# 防止插入 crt0.o / libc / libgcc
add_link_options(-nostdlib -nostartfiles)# 禁止 C++ RTTI 和异常支持(如有)
add_compile_options(-fno-exceptions -fno-rtti)# 加入链接脚本 + 明确入口地址
add_link_options(-T ${LINKER_SCRIPT} -Ttext=0x08000000)# 汇编 include 支持
add_compile_options(-Wa,-I${CMAKE_SOURCE_DIR}/src)

✅ 总结一句话:

build.sh 是你完全掌控编译流程,CMake 如果不特别配置,会加进很多 GCC 默认行为 —— 包括启动代码、链接库、调试符号、浮点支持等。这就导致生成文件大小、内容、行为都可能不同
Cmake 需要加 add_link_options(-nostartfiles -nostdlib)


http://www.xdnf.cn/news/976879.html

相关文章:

  • 重构技术奇点的路径:三智双融认知大飞跃
  • 如何设计一个用于大规模生产任务的人工智能AI系统
  • OpenSSL 无法验证 DevSidecar 的自签名证书
  • 【数据结构】图论最短路圣器:Floyd算法如何用双矩阵征服负权图?
  • Go 协程(Goroutine)入门与基础使用
  • Go 的 fs 包(1/2):现代文件系统抽象
  • 零基础玩转物联网-串口转以太网模块如何快速实现与HTTP服务器通信
  • Solidity从入门到精通-函数及数据存储和作用域
  • 用 IRify 深入探索 WebShell 中的 Source/Sink 挖掘
  • AWS CloudFormation实战:构建可复用的ECS服务部署模板
  • AWS之混合云
  • 2025年渗透测试面试题总结-长亭科技[社招]应急响应工程师(题目+回答)
  • Roboguide工作站机器人重新安装软件包
  • 顶顶通电话机器人功能列表
  • 【前端面试】八、工程化
  • 如何顺利将电话号码转移到新iPhone?
  • 如何将文件从 iPhone 传输到闪存驱动器
  • App UI 设计中色彩搭配如何激发用户的深层情感
  • 算法第13天|继续学习二叉树:平衡二叉树(递归)、二叉树所有路径(递归)、左叶子之和(递归)
  • 基于 WebWorker 的 WebAssembly 图像处理吞吐量分析
  • Vue 事件绑定机制详解
  • 通过知识整合重新审视医学图像检索|文献速递-深度学习医疗AI最新文献
  • 基于uniapp实现自定义日历页面、年份月份选择、动态日历渲染、日期标记及备忘录、无组件依赖、多端兼容
  • 微信小程序中wxs
  • 增强现实—Where am I? Cross-View Geo-localization with Natural Language Descriptions
  • 记录rust滥用lazy_static导致的一个bug
  • Android 应用被kill问题排查和处理
  • 【DeepSeek×Draw.io 轻松构建UML】智能协作,高效建模
  • UE5 学习系列(八)材质基础认知
  • WPP 媒体推出基于人工智能的工具突破基于身份识别的定向模式