小麦大叔
认证:普通会员
所在专题目录 查看专题
一文带你了解霍尔传感器
一文彻底搞懂C语言的Q格式
CPU缓存L1/L2/L3快速扫盲
一文快速认识Modbus协议
一招教你单片机固件快速瘦身
SD卡的SDIO模式/SPI模式设计扫盲
作者动态 更多
凉了!嵌入式真的要卷成下一个Java了吗?
02-28 09:03
嵌入式开发必须学习Qt吗?
02-27 09:16
这么简单的环境搭建,网友:我却搭了很久,结果还是不行...
02-26 15:44
单片机如何能运行如飞?一种高效实现数学函数的方式!
2022-01-26 11:08
keil开发过程中map和elf文件有什么作用?
2021-11-01 19:05

一招教你单片机固件快速瘦身

本文介绍了如何解决STM32如何通过修改编译器参数从而减小生成的固件大小,希望本文可以带来一些帮助;如果觉得不错,欢迎关注、分享、收藏、点赞。希望能帮助到大家,如有错误敬请指出,谢谢!

目录

  • 1 前言
  • 2 ELF格式
  • 3 编译器
    • 3.1 ARMCC
    • 3.2 GCC
  • 4 IDE
    • 4.1 MDK的设置
    • 4.2 CubeIDE
  • 5 结论

1 前言

我们平时做项目的时候,随着代码量的增加,工程变得更加臃肿,但是实际上可能只使用到其中一部分函数,与此同时,还有一部分是已经定义但是没有被使用的函数,虽然我们不使用这些功能和函数,但它们往往会浪费我们的ROM和RAM的空间

或者在使用静态库的时候,这种现象更加明显。比如,我们只需要使用静态库中的几个功能,但是编译器默认会把整个静态库全部链接到可执行程序中,从而导致可执行程序的大小大大增加。

那该如何避免这种情况呢?大部分工作交给编译器,我们只要告诉编译器不要把这些程序编译到可执行文件中即可。下面会继续解释。

2 ELF格式

ELFExecutable and Linkable Format)是可执行和可链接格式。在LinuxELF包括了链接过程中的目标文件(.o),共享库(.so)和可执行文件,同时还用于可加载的内核模块,因此作为链接过程中的目标文件也是通过ELF格式的文件来表示的;

ELF的结构至少包含两个头:

  • ELF头
  • 程序头
  • 通常还会有一个节标头;

具体如下图所示;

从图中我们可以看到,这里可以分为两种情况;

  • Linking:链接是按节划分的,在链接程序时会这样使用,这些部分包含一些目标文件信息,例如:数据,指令,重定位信息,符号,调试信息等等;
  • Exection:程序执行期间使用按段划分的执行视图。

下面是ELF格式文件的详细布局图;

3 编译器

通常在做ARM开发的时候会使用ARMCCGCC,可以参考相应编译器的手册,使用相应的编译命令就可以实现对程序的优化。

3.1 ARMCC

ARMCC中,编译器通常将函数和数据放在一起,并且将每一个类别规整到同一个section,如果在链接的时候发现某个section没有被使用,那么就会将这个section删除,从而减少可执行文件的大小。

可以使用--split_sections编译器命令行选项来指示编译器为源文件中的每个函数生成一个ELF节,这样在链接的时候可以通过--remove命令让链接器删除未使用的section

3.2 GCC

GCC在编译时可以使用 -ffunction-sections-fdata-sections 将每个函数或符号创建为一个section; 链接阶段的时候,使用 -Wl–gc-sections 来告诉链接器删除不需要的section(其中-Wl, 表示后面的参数 -gc-sections 传递给链接器),这样就能减少最终的可执行程序的大小了。

4 IDE

通常我们使用IDE的过程中,它已经帮我们做好了很多工作,比如上面提到的编译器命令需要我们自己手动写到Makefile中,但是在IDE只需要勾选相应的选项即可;

4.1 MDK的设置

MDK中使用的是ARMCC编译器,以STM32为例,纯净的HAL编译之后的结果如下图所示;

在工程的Opentios下勾选One ELF Section per Function,发现在编译器命令自动追加了--splot_sections

在这里插入图片描述

最终编译的结果如下,发现最终固件变小了很多;

开启设置之后

4.2 CubeIDE

CubeIDE中使用的是arm-none-eabi-gcc,相同的代码与上面的基本相同,创建CubeIDE的工程,编译之后如下图所示;

同样在项目的属性设置中,增加--ffunction-sections选项和-fdata-sections选项之后,构建项目;

最终结果如下所示,发现固件的大小减小很多;

5 结论

本文对于如何删除编译过程中未使用的section做了简单的介绍,从ELF文件格式的角度出发,介绍了编译器ARMCCGCC相应的命令以及MDKCubeIDE中的相应配置,最终实验表明可以减少程序大小,另外编译器的优化等级-O1-O2-O3也可以优化程序的大小以及执行时间。

笔者能力和水平有限,文中难免有错误和纰漏之处,请大佬们不吝赐教。

声明:本内容为作者独立观点,不代表电子星球立场。未经允许不得转载。授权事宜与稿件投诉,请联系:editor@netbroad.com
觉得内容不错的朋友,别忘了一键三连哦!
赞 4
收藏 4
关注 132
成为作者 赚取收益
全部留言
0/200
  • 天赐时间 2020-12-17 16:03
    讲的真好!
    回复