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 函数运行用户程序。

相关推荐
刘延林.13 小时前
esp32 s3+micpython快速验证ML307R 是否能正常连接4G
单片机·嵌入式硬件
不做无法实现的梦~18 小时前
86步进电机和DM860H驱动器的使用方法和记录
单片机·嵌入式硬件
Aaron158818 小时前
RFSOC+VU13P/VU9P+GPU多通道同步一体化解决方案
人工智能·嵌入式硬件·算法·matlab·fpga开发·硬件架构·基带工程
所见即所得1111119 小时前
stm32烧录过程中串口问题(串口被占用无法使用)
stm32·单片机·嵌入式硬件
Freak嵌入式19 小时前
WIZnet-EVB-Pico2开始,用MicroPython玩转以太网开发
arm开发·人工智能·python·嵌入式硬件·机器人·嵌入式·micropython
Ligocious19 小时前
stm32---1.两种开发方式点亮LED
stm32·单片机
黑白园19 小时前
STM32F103C8TC使用ST-Link下载
stm32·单片机·嵌入式硬件
时空自由民.20 小时前
嵌入式MCU的中断系统工作流程及其原理
单片机·嵌入式硬件
LCG元20 小时前
STM32实战:基于STM32F103的智慧教室环境监控系统(CO₂+光照+人数统计)
前端·stm32·嵌入式硬件
振南的单片机世界20 小时前
推挽输出:上管推、下管拉,驱动强但不“合群”
arm开发·stm32·单片机·嵌入式硬件