Red Point

十年饮冰 , 热血难凉 ; 山高万仞 , 只登一步

  博客园 :: 首页 :: 博问 :: 闪存 :: :: :: :: 管理 ::

https://blog.csdn.net/it_660/article/details/147457720

1. 启动模式与地址映射
STM32F103 的启动模式由 BOOT0 和 BOOT1 引脚配置决定,不同的启动模式对应不同的存储器映射:

启动模式 映射地址范围 说明
主 Flash 0x08000000~0x0807FFFF 用户程序存储在 Flash 中,复位后从 Flash 启动(默认模式)。
系统存储器 0x1FFFF000~0x1FFFF7FF 内置 Bootloader(ISP 编程用),用于通过串口/USB 下载程序。
内置 SRAM 0x20000000~0x2000FFFF 调试或临时运行代码时使用,需手动加载程序到 SRAM。
地址重映射:Cortex-M3 内核复位后,始终从 0x00000000 读取初始堆栈指针(SP)和复位向量。STM32 通过内部机制将上述物理地址映射到 0x00000000(例如,Flash 启动时 0x08000000 被映射到 0x00000000)。

2. 启动过程关键步骤
(1) 硬件复位
复位后,内核执行以下操作:

从 0x00000000 读取 初始堆栈指针(MSP)。

从 0x00000004 读取 复位向量地址(程序入口 Reset_Handler)。

跳转到复位向量地址执行代码。

(2) 启动文件(Startup Code)
启动文件(如 startup_stm32f103xe.s)是汇编代码,负责初始化关键硬件并引导到 main() 函数。主要流程如下:

初始化堆栈指针
定义主堆栈(MSP)和可选的进程堆栈(PSP):

1 assembly
2 ; 定义堆栈段
3 Stack_Size      EQU     0x400
4                 AREA    STACK, NOINIT, READWRITE, ALIGN=3
5 Stack_Mem       SPACE   Stack_Size
6 __initial_sp

向量表(Vector Table)
中断向量表位于 Flash 起始位置,包含中断服务函数地址:

1 assembly
2 ; 中断向量表
3 Vector_Table    DCD     __initial_sp             ; 初始堆栈指针
4                 DCD     Reset_Handler            ; 复位向量
5                 DCD     NMI_Handler              ; NMI 处理函数
6                 DCD     HardFault_Handler        ; 硬件错误处理函数
7                 ...                              ; 其他中断向量

复位处理函数(Reset_Handler)
复位后执行的核心初始化代码:

 1 assembly
 2 Reset_Handler   PROC
 3                 EXPORT  Reset_Handler           [WEAK]
 4                 IMPORT  SystemInit               ; 引入 SystemInit 函数
 5                 IMPORT  __main                  ; 引入 C 库初始化函数
 6                 LDR     R0, =SystemInit          ; 加载 SystemInit 地址
 7                 BLX     R0                       ; 调用 SystemInit(初始化时钟等)
 8                 LDR     R0, =__main              ; 加载 __main 地址
 9                 BX      R0                       ; 跳转到 __main(初始化 C 库并调用 main())
10                 ENDP

(3) SystemInit 函数
SystemInit() 在 system_stm32f1xx.c 中定义,负责初始化关键系统配置:

设置时钟源(HSI/HSE/PLL)。

配置 Flash 预取、等待周期。

初始化 AHB/APB 总线时钟。

可选:配置向量表偏移寄存器(VTOR)。

(4) __main 和 main()
__main:C 库函数,负责初始化全局变量、堆栈,最终调用用户 main()。

main():用户程序入口。

关键指令说明
启动文件中的汇编指令示例:
LDR R0, =label
将 label 的地址加载到寄存器 R0。

assembly
LDR R0, =SystemInit ; 将 SystemInit 函数的地址加载到 R0
BLX R0
跳转到 R0 指向的地址(调用函数),并保存返回地址到 LR 寄存器。

assembly
BLX R0 ; 调用 SystemInit()
BX R0
跳转到 R0 指向的地址(不保存返回地址)。

assembly
BX R0 ; 跳转到 __main
PROC / ENDP
定义函数块的开始和结束。

assembly
Reset_Handler PROC

ENDP

4. 内存布局与 VTOR
向量表偏移寄存器(VTOR):
默认向量表位于 Flash(0x08000000),可通过修改 VTOR 寄存器重定位向量表(如将向量表复制到 RAM):

c
SCB->VTOR = FLASH_BASE | VECT_TAB_OFFSET; // 在 SystemInit 中配置
典型内存分布:

0x00000000: 映射到启动存储器(Flash/系统存储器/SRAM)
0x08000000: Flash 实际地址
0x20000000: SRAM 地址
0x40000000: 外设寄存器

5. 常见问题
启动失败的可能原因

堆栈溢出导致 MSP 初始化错误。

复位向量未正确指向 Reset_Handler。

时钟配置错误(如未启用外部晶振 HSE)。

调试技巧

检查向量表是否对齐到 0x200 的倍数。

使用调试器查看 SP 和 PC 的初始值。

总结
STM32F103 的启动过程由硬件复位、启动文件初始化、系统配置和用户程序执行组成。理解地址映射、向量表和关键指令对调试和优化启动流程至关重要。开发者可通过修改启动文件或调整 BOOT 引脚选择不同的启动模式。

posted on 2025-09-07 20:15  Red_Point  阅读(28)  评论(0)    收藏  举报