【嵌入式常识篇】一个C项目工程在IDE中是怎么一步步编译成一个固件包的

embedded/2025/1/15 3:47:20/

前言:初学C语言的时候是在Linux环境下,那时候只知道需要通过GCC工具编译成可执行文件才可以在运行,后来进入到了嵌入式行业发现需要IDE将一个C项目工程编译成一个固件包,那时候经常会产生一个疑问:一个C项目工程在IDE中是怎么一步步编译成一个固件包的呢?下面就解答一下这个疑问。时光荏苒,也算是给当年刚入行的自己一个答案。


1️⃣,流程简述

将一个 C 项目工程源代码 编译成 固件包(hex、bin、elf 等) 的过程,涉及 编译工具链IDE 的工作原理。一般来说,固件编译流程分为以下关键步骤:

这个过程通常由 编译工具链(如 GCCIARKeilClang 等)完成,而 IDE(如 Keil、STM32CubeIDE、IAR 等)负责调用这些工具链,并提供一个图形化的界面来简化开发流程

源码 (C/ASM/头文件) ↓
预处理器 (Preprocessor) ↓
编译器 (Compiler) ↓
汇编器 (Assembler) ↓
链接器 (Linker) ↓
固件包 (HEX/BIN/ELF 文件)

对应的执行流程框图如下: 


2️⃣,预处理(Preprocessing)

预处理器会在编译前对源代码进行 文本替换和展开,主要包括:

  • 处理 #include 头文件的引用。
  • 替换宏定义(#define)。
  • 处理条件编译指令(#ifdef#ifndef#endif 等)。
  • 移除注释。

举个例子:下面的这个是源代码

#include "my_header.h"
#define LED_PIN 13void main() {int pin = LED_PIN;
}

当预处理之后:

void main() {int pin = 13;
}

对应的预处理工具与命令:(最经典的就是GCC,当然如果是其他的芯片环境平台可能就需要用到交叉编译工具了)

GCC: gcc -E main.c -o main.i


3️⃣,编译(Compilation)

预处理完成的 .i 文件(纯文本 C 代码)会被编译器转换成 汇编代码

  • 编译器的任务
    • 语法分析,检查代码的语法和语义。
    • 生成对应的汇编代码(.s 文件)

同样是上面的例子:编译后的汇编代码:

mov r0, #13
str r0, [sp, #4]

 编译工具:

  • GCC: gcc -S main.i -o main.s

4️⃣,链接(Linking)

链接器的任务是将多个 目标文件(.o 文件)库文件(.a 或 .lib 文件)启动文件(startup 文件) 合并成一个 可执行的固件文件(.elf/.bin/.hex)

  • 链接的工作内容
    • 解析和解决函数、变量的外部引用。
    • 合并不同模块的目标文件。
    • 分配内存地址(根据链接脚本 linker script)。
    • 生成可执行文件(如 .elf.bin.hex)。

链接工具:

  • GCC: gcc main.o -o main.elf -T linker_script.ld

示例:

如果有多个文件:

main.o      // 主程序
startup.o   // 启动代码
libc.a      // 标准库

链接后的 ELF 文件:

main.elf

📋 ELF 文件包含的信息

  • 可执行机器指令。
  • 符号表、调试信息。
  • 内存布局(.text、.data、.bss、堆栈等段)。

5️⃣,转换(Conversion)

编译完成后生成的 ELF 文件,可能还需要转换成 HEXBIN 格式,便于烧录到芯片中。

  • .elf 文件是包含调试信息的可执行文件,通常用于开发和调试阶段。
  • .hex 文件是Intel HEX 格式的固件包,通常用于烧录工具。
  • .bin 文件是纯二进制格式的固件包

转换工具:

  • objcopy 工具可以将 ELF 文件转换成 HEX/BIN 文件。
# 转换成 HEX 文件
arm-none-eabi-objcopy -O ihex main.elf main.hex# 转换成 BIN 文件
arm-none-eabi-objcopy -O binary main.elf main.bin

🛠️ IDE 的编译流程

常用的 IDE(如 Keil、IAR、STM32CubeIDE)会自动调用编译工具链,按以下步骤完成编译:

  1. 解析项目文件project.uvprojx.cproject 等)。

  2. 调用编译器,对每个源文件进行预处理、编译、汇编,生成 .o 文件。

  3. 调用链接器,根据链接脚本生成 .elf 文件。

  4. 调用转换工具,生成 .hex.bin 文件。

  5. 调用烧录工具,将固件烧录到芯片中(如果配置了烧录选项)。


📌 链接脚本(Linker Script)

链接脚本控制着固件的内存布局,包括:

  • 代码段(.text) 放在 Flash 中。
  • 初始化数据段(.data) 放在 RAM 中。
  • 未初始化数据段(.bss) 放在 RAM 中。
  • 堆栈和堆的分配

典型的链接脚本片段:

MEMORY
{FLASH (rx) : ORIGIN = 0x08000000, LENGTH = 512KRAM (rwx) : ORIGIN = 0x20000000, LENGTH = 128K
}SECTIONS
{.text : {*(.text)*(.rodata)} > FLASH.data : {*(.data)} > RAM AT > FLASH.bss : {*(.bss)} > RAM
}


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

相关文章

【HTML+CSS+JS+VUE】web前端教程-31-css3新特性

圆角 div{width: 100px;height: 100px;background-color: saddlebrown;border-radius: 5px;}阴影 div{width: 200px;height: 100px;background-color: saddlebrown;margin: 0 auto;box-shadow: 10px 10px 20px rgba(0, 0, 0, 0.5);}

Redis数据结构服务器

Redis数据结构服务器 什么是Redis数据结构服务器 的概念和特点 是一个开源(BSD许可),内存中的数据结构存储服务器,可用作数据库、缓存和消息中间件。它支持多种类型的数据结构,如字符串(strings&#xff09…

服务器引导异常,Grub报错: error: ../../grub-core/fs/fshelp.c:258:file xxxx.img not found.

服务器引导异常,Grub报错: error: ../../grub-core/fs/fshelp.c:258:file xxxx.img not found. 1. 故障现象2. 解决思路3. 故障分析4. 案件回溯5. 解决问题 1. 故障现象 有一台服务器业务报无法连接. 尝试用Ping命令发现无法ping通. 通过控制台查看发现有以下报错: error: ..…

贪心算法汇总

1.贪心算法 贪心的本质是选择每一阶段的局部最优,从而达到全局最优。 如何能看出局部最优是否能推出整体最优 靠自己手动模拟,如果模拟可行,就可以试一试贪心策略,如果不可行,可能需要动态规划。 如何验证可不可以…

Shader -> RadialGradient圆心渐变着色器详解

XML文件 <com.example.myapplication.MyViewxmlns:android"http://schemas.android.com/apk/res/android"android:layout_width"match_parent"android:layout_gravity"center"android:layout_height"400dp"/>自定义View代码 c…

【数学】概率论与数理统计(五)

文章目录 [toc] 二维随机向量及其分布随机向量离散型随机向量的概率分布律性质示例问题解答 连续型随机向量的概率密度函数随机向量的分布函数性质连续型随机向量均匀分布 边缘分布边缘概率分布律边缘概率密度函数二维正态分布示例问题解答 边缘分布函数 二维随机向量及其分布 …

IP 地址与蜜罐技术

基于IP的地址的蜜罐技术是一种主动防御策略&#xff0c;它能够通过在网络上布置的一些看似正常没问题的IP地址来吸引恶意者的注意&#xff0c;将恶意者引导到预先布置好的伪装的目标之中。 如何实现蜜罐技术 当恶意攻击者在网络中四处扫描&#xff0c;寻找可入侵的目标时&…

Java-数据结构-链表(LinkedList)-双向链表

一、LinkedList(无头双向链表) 在之前的学习中&#xff0c;我们已经学习过"单向链表"并通过做题加深了对"单向链表"的认知&#xff0c;而今天我们继续来学习链表&#xff0c;也就是"无头双向链表"~ 在了解"无头双向链表"之前&#x…