STM32 启动流程

我用最容易理解 的方式,把 STM32 从上电 → 运行 main () 的全过程讲清楚,Cortex‑M 内核通用(STM32F1/F4/H7/L4 全都一样)。


一、一句话总结启动流程

上电 → 从 Flash 读取栈顶地址 → 加载复位中断函数地址 → 跳转到复位函数 → 初始化硬件 → 跳转到 main () → 死循环


二、详细 7 步启动流程(必须掌握)

第 1 步:上电 / 复位

  • 电源接通,或复位键按下
  • CPU 进入复位状态

第 2 步:从 0x08000000 读取 栈顶地址 (SP)

  • STM32 启动时,强制从 Flash 起始地址 0x0800_0000 取第一个字
  • 这个字是 栈顶指针(Stack Pointer)
  • 作用:给程序运行准备堆栈空间

第 3 步:读取 复位中断服务函数地址

  • Flash 第 2 个地址:0x0800_0004
  • 这里存放的是 复位函数 Reset_Handler 的入口地址

**如上三步都是CPU 硬件行为,和任何代码、startup.s 无关,**Cortex‑M 内核硬件强制做两件事:

  1. 0x0800 0000 读取 4 字节 → 赋值给 SP 栈指针
  2. 0x0800 0004 读取 4 字节 → 赋值给 PC 程序计数器

第 4 步:跳转到 Reset_Handler 复位函数

这是程序真正执行的第一行代码,由启动文件(startup_stm32xx.s)提供。

第 5 步:初始化数据段(.data)

  • 初始化变量 从 Flash 复制到 RAM
  • 例如:int a = 10;

第 6 步:清零未初始化数据段(.bss)

  • int b; 这类未赋值变量全部清 0
  • 防止随机值导致程序异常

第 7 步:跳转到 C 语言的 main () 函数

复制代码
main();

三、启动文件(startup_stm32xx.s)到底干了什么?

它是汇编文件,是 MCU 上电后第一个执行的代码。

主要做 4 件事:

  1. 设置栈顶 SP
  2. 设置中断向量表(复位、异常、中断入口)
  3. 初始化 RAM 数据
  4. 调用 main ()

四、最关键:中断向量表(启动核心)

Flash 开头固定放中断向量表

复制代码
0x08000000:   栈顶地址      (第一个指令)
0x08000004:   复位函数地址  (第二个指令)
0x08000008:   硬fault地址
0x0800000C:   存储器管理fault
...

复位函数地址 = 程序真正的入口


五、超简洁流程图(秒懂)

复制代码
上电
  ↓
读栈顶地址(SP)
  ↓
读复位函数地址
  ↓
执行 Reset_Handler
  ↓
初始化 .data 和 .bss
  ↓
进入 main()
  ↓
while(1) 死循环

六、最核心的 3 个知识点(面试必考)

  1. STM32 程序入口不是 main (),而是 Reset_Handler ()
  2. 中断向量表放在 Flash 起始位置
  3. 启动文件负责初始化堆栈和 RAM,再跳转到 main

七、一句话终极总结

STM32 上电先找栈,再找复位函数,初始化 RAM 后才进入 main ()。

**问题:**启动文件中的Reset_Handler函数具体做了哪些初始化操作?

Reset_Handler 一共只做 5 件核心初始化事

这 5 件事做完,才会跳去 main()


1. 初始化栈指针(硬件自动完成,但属于复位流程)

  • 0x08000000 取出栈顶地址
  • 赋值给 SP(栈指针)
  • 让程序有栈可用,不崩溃

2. 复制 .data 段(初始化全局变量)

作用:把已经赋初值的全局变量 / 静态变量Flash 搬到 RAM

例如:

复制代码
int a = 100;

这些数据存在 Flash,运行时必须放 RAM。


3. 清零 .bss 段(未初始化全局变量)

作用:把未赋值的全局变量 全部设为 0

例如:

复制代码
int b;

如果不清零,值是随机乱码,程序会飞。


4. 调用 SystemInit(系统核心初始化)

这是 ST 官方提供的 C 函数,做:

  • 打开外部时钟 / 配置 PLL
  • 设置系统主频(如 72MHz、108MHz、200MHz)
  • 关闭看门狗
  • 配置 Flash 等待周期
  • 配置中断分组基础

一句话:让系统从默认低速 → 跑到最高性能


5. 调用 C 库初始化(__libc_init_array)

作用:

  • 初始化 C++ 全局构造函数
  • 初始化标准库
  • 初始化浮点运算单元(如果有)

普通 C 项目也会执行,保证 C 环境完整。


最后一步:跳转到 main ()

复制代码
bl main

如果 main 退出,就进死循环:

复制代码
LoopForever:
  b LoopForever

终极极简总结(背这个)

Reset_Handler 只做 5 件事:

  1. 设置栈指针 SP
  2. 拷贝 .data 段(初始化全局变量)
  3. 清零 .bss 段(未初始化变量)
  4. 执行 SystemInit(时钟 / 系统配置)
  5. 初始化 C 库环境
  6. 进入 main ()

一句话面试标准答案

Reset_Handler 是上电后第一个执行的函数,负责初始化堆栈、搬运数据、配置系统时钟,最后跳转到 main 函数运行用户程序。

相关推荐
weixin_520649872 小时前
短距离硬件通信
单片机·嵌入式硬件
振南的单片机世界4 小时前
HAL慢、ODR中、BSRR快:GPIO翻转速度的三重境界
单片机·嵌入式硬件
实在太懒于是不想取名4 小时前
STM32N6的开发日记(7):摄像头+NPU联动,完成 YOLO 实时目标检测与显示
stm32·嵌入式硬件·yolo
Joseph Cooper5 小时前
STM32MP157 Linux驱动学习笔记(四):典型总线与设备模型(SPI/USB)
linux·stm32·学习
小柯博客6 小时前
STM32MP2安全启动技术深度解析
c语言·c++·stm32·嵌入式硬件·安全·开源·github
实在太懒于是不想取名6 小时前
STM32N6的开发日记(8):在N6中部署自训练的火焰检测模型
stm32·单片机·嵌入式硬件
木子单片机7 小时前
基于51单片机温度上下限报警设计 数码管显示
stm32·单片机·嵌入式硬件·51单片机·keil
森利威尔电子-8 小时前
森利威尔SL3180 替换LTC3639 150V降压恒压芯片 3A峰值电流能力转换效率高 静态电流低
单片机·嵌入式硬件·集成电路·芯片·电源芯片
qq_401700418 小时前
合宙4G模块Air724UG
嵌入式硬件