【STM32】启动流程

1、STM32的启动过程

此处的启动过程是指从STM32芯片上电复位执行的第一条指令开始,到执行用户编写的main函数。这中间的工作是通过启动文件(.s)来完成的。

1.1、启动模式

STM32的复位方式有三种:上电复位,硬件复位和软件复位。当产生复位,并离开复位状态后,CM3内核做的第一件事就是从0x0000 0000处读取堆栈指针MSP的初始值,即栈顶指针;从0x0000 0004处去读程序计数器指针PC,即复位后执行的第一条指令。
而这两处的地址可以被重映射到其它地址空间(通过启动引脚高低电平的不同),通过映射到不同的地址空间,也就有了不同的启动模式,一般是以下三种。

引脚配置启动模式启动源应用场景
BOOT0 = 0;BOOT1 = x内部flash从内部flash存储器启动,即用户应用程序量产产品正式运行环境、掉电保存
BOOT0 = 1;BOOT1 = 1内部sram从内部sram存储器启动,程序加载到sram中执行demo调试,断电不保存
BOOT0 = 1;BOOT1 = 0系统存储器从系统存储器启动,存放厂商预装的bootloaderISP产品升级

2、.s启动文件解析

STM32的启动文件(一般是.s汇编文件,如startup_stm32f407xx.s)是STM32上电后执行的第一段代码,承担着“系统初始化化引导员”的角色。
它的主要作用是设置初始化栈指针(SP)、程序计数器(PC)、定义向量表、跳转到C库中__main函数,最终调用用户main函数。

2.1、栈和堆的初始化配置

定义栈和堆的大小,并指定分配内存空间。

Stack_Size		EQU     0x400

                AREA    STACK, NOINIT, READWRITE, ALIGN=3
Stack_Mem       SPACE   Stack_Size
__initial_sp

Heap_Size      EQU     0x200

                AREA    HEAP, NOINIT, READWRITE, ALIGN=3
__heap_base
Heap_Mem        SPACE   Heap_Size
__heap_limit

2.2、向量表

定义储存中断服务程序地址的数组,即中断向量表。

__Vectors       DCD     __initial_sp               ; Top of Stack
                DCD     Reset_Handler              ; Reset Handler
                DCD     NMI_Handler                ; NMI Handler
                ...

2.3、复位处理程序

Reset_Handler是复位处理程序,当MCU复位时,程序会跳转到这里执行。其中主要包括SystemInit函数和__main函数。

Reset_Handler    PROC
                 EXPORT  Reset_Handler             [WEAK]
        IMPORT  SystemInit
        IMPORT  __main

                 LDR     R0, =SystemInit
                 BLX     R0
                 LDR     R0, =__main
                 BX      R0
                 ENDP

2.4、其它中断处理程序

这里都是默认的异常处理程序,如非屏蔽中断(NMI)、硬故障(HardFault)等。可以在用户程序中重新定义这些中断处理程序,例如,在main.c中定义void SysTick_Handler(void)会覆盖启动文件中的弱定义。

2.5、用户栈和堆初始化

如果使用 MicroLIB,则导出栈顶指针、堆基地址和堆限制地址。否则,导入 __use_two_region_memory 符号,并定义__user_initial_stackheap函数,用于初始化用户栈和堆。
这里是用于初始化栈和堆的内存布局,并根据不同的工具链和库配置提供相应的接口。

                 IF      :DEF:__MICROLIB
                
                 EXPORT  __initial_sp
                 EXPORT  __heap_base
                 EXPORT  __heap_limit
                
                 ELSE
                
                 IMPORT  __use_two_region_memory
                 EXPORT  __user_initial_stackheap
                 
__user_initial_stackheap

                 LDR     R0, =  Heap_Mem
                 LDR     R1, =(Stack_Mem + Stack_Size)
                 LDR     R2, = (Heap_Mem +  Heap_Size)
                 LDR     R3, = Stack_Mem
                 BX      LR

                 ALIGN

                 ENDIF
  • 1、MicroLIB分支
    • 适用于Flash/RAM较小的场景(如F1系列)。
    • 减小代码体积。
    • 直接通过全局变量管理堆和栈,无需函数调用开销。
  • 2、标准库
    • 适用于需要完整的C库功能的场景(如malloc()printf())。
    • 通过__user_initial_stackheap函数动态配置堆和栈,支持更灵活的内存管理。
    • 双区域模型避免栈和堆的冲突(如栈溢出导致覆盖堆数据)

3、STM32启动流程

  • 1、复位:当 STM32 微控制器复位时,处理器从地址 0 处读取栈顶指针(SP)的值,并将其加载到 SP 寄存器中。
  • 2、跳转到复位处理程序:接着从地址 4 处读取复位处理程序的地址,并将其加载到程序计数器(PC)中,程序跳转到 Reset_Handler 执行。
  • 3、系统初始化:在 Reset_Handler 中,首先调用 SystemInit 函数进行系统时钟等硬件配置的初始化。
  • 4、进入 C 库启动函数:然后调用 __main 函数,__main 函数会完成一些初始化工作,如全局变量的初始化等,最终调用用户的 main 函数。
  • 5、用户程序执行:程序进入用户的 main 函数,开始执行用户编写的代码。
  • 6、异常处理:当发生异常或中断时,处理器会根据向量表中的地址跳转到相应的异常处理程序执行。

4、常见使用技巧

  • 1、增大栈内存,解决栈溢出问题
  • 2、重映射中断向量表(IAP升级、双程序、A/B固件备份)
  • 3、裁剪启动流程,适配应用场景。
### STM32微控制器启动过程详解 STM32 是基于 ARM Cortex-M 架构的微控制器系列,其启动过程涉及硬件初始化和软件执行两个主要阶段。以下是关于 STM32 启动过程的具体描述: #### 1. 上电复位与 Boot 模式选择 当 STM32 微控制器上电或发生复位事件时,内部硬件电路会自动进入复位状态。此时,微控制器的核心寄存器被设置为其默认值,同时系统会选择从哪个存储区域加载初始指令。 Boot 模式的选择由引脚 `BOOT0` 和 `BOOT1` 的组合决定[^1]。具体来说: - 如果 `BOOT0=0` 并且 `BOOT1=x`,则设备将从 Flash 存储器启动。 - 如果 `BOOT0=1` 并且 `BOOT1=0`,则设备将从 System Memory 启动。 - 如果 `BOOT0=1` 并且 `BOOT1=1`,则设备将从 SRAM 启动。 这种模式的选择允许开发者灵活地指定不同的启动源,例如运行预置的 bootloader 或者调试代码。 #### 2. 系统时钟配置 在完成 Boot 模式选择之后,STM32 进一步初始化系统时钟。默认情况下,微控制器会在复位后使用内部 RC 振荡器 (HSI) 作为系统时钟源。随后,可以通过修改 RCC(Reset and Clock Control)寄存器来切换至外部晶振 (HSE),并启用 PLL 来提高主频以达到最佳性能[^1]。 #### 3. 堆栈指针初始化 一旦选择了正确的时钟源,堆栈指针 (`SP`) 就会被初始化为链接器文件中定义的位置。该地址位于向量表的第一个条目处,而向量表本身通常驻留在闪存起始位置或者重新映射后的 RAM 地址范围内[^1]。 #### 4. 跳转到主函数入口点 紧接着,在向量表中的第二个条目保存着 Reset Handler 函数的地址。CPU 控制权转移至此函数,标志着正式进入了用户编写的程序逻辑部分。一般而言,Reset Handler 定义了一系列必要的低级初始化操作,比如 GPIO 设置、外设使能以及中断优先级分配等[^3]。 ```c void Reset_Handler(void) { // 初始化堆栈和其他环境变量 __initialize_hardware(); // 调用 C 库初始化例程 _SystemInit(); // 开始执行 main() 函数 main(); } ``` 以上即构成了整个 STM32 微控制器完整的启动流程概述。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值