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

相关推荐
清风6666664 小时前
基于单片机与DAC0832的双路波形信号发生系统设计
单片机·嵌入式硬件·毕业设计·课程设计·期末大作业
azwsm6 小时前
电路元器件和GPIO控制器
单片机·嵌入式硬件
kebidaixu9 小时前
FreeRTOS 移植到 STM32F407VETX 记录(一)
stm32·单片机·嵌入式硬件
CSDN官方博客9 小时前
「谁说嵌入式只是调包和焊板子?」—— 2026嵌入式全栈技术征锋令
嵌入式硬件·物联网·embedding
半条-咸鱼9 小时前
【INACCESSIBLE_BOOT_DEVICE】安装 Config Tool 后 Windows 蓝屏,最终通过 VMware 虚拟机解决
windows·stm32·vmware·芯片
点灯小铭10 小时前
基于单片机的数码管定时插座设计与定时开关功能实现
单片机·嵌入式硬件·毕业设计·课程设计·期末大作业
云栖梦泽10 小时前
玩转RK3506SDK
linux·嵌入式硬件
数智工坊12 小时前
机器人四大主控板系统分层选型指南:树莓派、ESP32、STM32与Arduino的能力边界与实战定位
stm32·嵌入式硬件·机器人
某林21212 小时前
跨越底层与AI的鸿沟:ROS2+多模态大模型(Qwen-VL)机器人全链路排障实录
人工智能·stm32·机器人·人机交互·ros2·技术复盘
进击的小头12 小时前
第8篇:IGBT 从零到精通:核心原理、关键参数、选型指南与工业级应用要点
经验分享·嵌入式硬件·学习