STM32启动流程详解(超全,startup_stm32xx.s分析)
创始人
2025-01-07 18:37:26
0

单片机上电后执行的第一段代码

822e5e8dcad64b5e9abf94a9a16f0e58.png

        1.初始化堆栈指针 SP=_initial_sp

        2.初始化 PC 指针=Reset_Handler

        3.初始化中断向量表

        4.配置系统时钟

        5.调用 C 库函数_main 初始化用户堆栈,然后进入 main 函数。

        在正式讲解之前,我们需要了解STM32的启动模式。

STM32的启动模式

181e8d540bb34605a9f31fb2d4f23def.png

        手册可以在Keil中跳转查看

STM32的三种启动模式

        首先要讲一下STM32的启动模式,因为启动模式决定了向量表的位置,STM32有三种启动模式:

        1. 主闪存存储器(Main Flash memory)启动

        从STM32内置的Flash启动(0x0800 0000-0x0807 FFFF),一般我们使用JTAG或者SWD模式下载程序时,就是下载到这个里面,重启后也直接从这启动程序。以0x08000000 对应的内存为例,则该块内存既可以通过0x00000000 操作也可以通过0x08000000 操作,且都是操作的同一块内存。

        2. 系统存储器(System memory)启动

        从系统存储器启动(0x1FFFF000 - 0x1FFF F7FF),这种模式启动的程序功能是由厂家设置的。一般来说,我们选用这种启动模式时,是为了从串口下载程序,因为在厂家提供的ISP程序中,提供了串口下载程序的固件,可以通过这个ISP程序将用户程序下载到系统的Flash中。以0x1FFFFFF0对应的内存为例,则该块内存既可以通过0x00000000 操作也可以通过0x1FFFFFF0操作,且都是操作的同一块内存。

        3. 片上SRAM(Embedded SRAM)启动

        从内置SRAM启动(0x2000 0000-0x3FFFFFFF),既然是SRAM,自然也就没有程序存储的能力了,这个模式一般用于程序调试。SRAM 只能通过0x20000000进行操作,与上述两者不同。从SRAM 启动时,需要在应用程序初始化代码中重新设置向量表的位置。

        用户可以通过设置BOOT0和BOOT1的引脚电平状态,来选择复位后的启动模式。

        如下图所示:

5cdb65af3d244fd38cebd60304b5b272.png

20d718ad30d94b3793839f6c78702e69.png

总结 

        启动模式只决定程序烧录的位置,加载完程序之后会有一个重映射(映射到0x00000000地址位置);真正产生复位信号的时候,CPU还是从开始位置执行。

        值得注意的是STM32上电复位以后,代码区都是从0x00000000开始的,三种启动模式只是将各自存储空间的地址映射到0x00000000中。

STM32的启动文件分析

        因为单片机上电启动过程主要是由汇编完成的,因此STM32的启动的大部分内容都是在启动文件里。我用CubeMX生成的的启动文件是startup_stm32f103xb.s,不管使用标准库还是使用HAL库,启动文件都是差不多的。

1. Stack栈

        栈的作用是用于局部变量,函数调用,函数形参等的开销,栈的大小不能超过内部SRAM 的大小。当程序较大时,需要修改栈的大小,不然可能会出现的HardFault的错误。

06efc80c404844169484668fc0874736.png

第32行:表示开辟栈的大小为 0X400(1KB),EQU是伪指令,相当于C 中的 define。

第34行:开辟一段可读可写数据空间,ARER 伪指令表示下面将开始定义一个代码段或者数据段。此处是定义数据段。ARER 后面的关键字表示这个段的属性。段名为STACK,可以任意命名;NOINIT 表示不初始化;READWRITE 表示可读可写,ALIGN=3,表示按照 8 字节对齐。

第35行:SPACE 用于分配大小等于 Stack_Size连续内存空间,单位为字节。

第37行: __initial_sp表示栈顶地址。栈是由高向低生长的。

2. Heap堆

        堆主要用来动态内存的分配,像malloc()函数申请的内存就在堆中。

5c6195ffd67f4a758ab5e6a3ec0c83da.png

        开辟堆的大小为 0X200(512 字节),名字为 HEAP,NOINIT 即不初始化,可读可写,8字节对齐。__heap_base 表示对的起始地址,__heap_limit 表示堆的结束地址。

572e655a446e4003b97139f12b1ae4e8.png

3. 向量表

        向量表是一个WORD( 32 )数组,每个下标对应一种异常,该下标元素的值则是该 ESR 的入口地址。向量表在地址空间中的位置是可以设置的,通过 NVIC 中的一个重定位寄存器来指出向量表的地址。在复位后,该寄存器的值为 0。因此,在地址 0 (即 FLASH 地址 0)处必须包含一张向量表,用于初始时的异常分配。

        值得注意的是这里有个另类: 0 号类型并不是什么入口地址,而是给出了复位后 MSP 的初值,后面会具体讲解。

62c41cc496974e99a8fee3a8bebeec71.png

82f155ef9d124c68905ed17929629c40.png

第55行:定义一块代码段,段名字是RESET,READONLY 表示只读。

第56-58行:使用EXPORT将3个标识符申明为可被外部引用,声明 __Vectors、__Vectors_End 和__Vectors_Size 具有全局属性。

第60行:__Vectors 表示向量表起始地址,DCD 表示分配 1 个 4 字节的空间。每行 DCD 都会生成一个 4 字节的二进制代码,中断向量表 存放的实际上是中断服务程序的入口地址。当异常(也即是中断事件)发生时,CPU 的中断系统会将相应的入口地址赋值给 PC 程序计数器,之后就开始执行中断服务程序。在60行之后,依次定义了中断服务程序的入口地址。

第121行:__Vectors_End 为向量表结束地址。

第123行:__Vectors_Size则是向量表的大小,向量表的大小是通过__Vectors 和__Vectors_End 相减得到的。

4. 复位程序

        复位程序是系统上电后执行的第一个程序,复位程序也是中断程序,只是这个程序比较特殊,因此单独提出来讲解。

e4dc62936c9949e186d21f6a3aca21fc.png

第128行:定义了一个服务程序,PROC表示程序的开始。

第129行:使用EXPORT将Reset_Handler申明为可被外部引用,后面WEAK表示弱定义,如果外部文件定义了该标号则首先引用该标号,如果外部文件没有声明也不会出错。这里表示复位程序可以由用户在其他文件重新实现,这种写法在HAL库中是很常见的。

第130-131行:表示该标号来自外部文件,SystemInit()是一个库函数,在system_stm32f1xx.c中定义的,__main 是一个标准的 C 库函数,主要作用是初始化用户堆栈,这个是由编译器完成的,该函数最终会调用我们自己写的main函数,从而进入C世界中。

第132行:这是一条汇编指令,表示从存储器中加载SystemInit到一个寄存器R0的地址中。

第133行:汇编指令,表示跳转到寄存器R0的地址,并根据寄存器的 LSE 确定处理器的状态,还要把跳转前的下条指令地址保存到 LR。

第134行:和132行是一个意思,表示从存储器中加载__main到一个寄存器R0的地址中。

第135行:和133稍微不同,这里跳转到至指定寄存器的地址后,不会返回。

第136行:和PROC是对应的,表示程序的结束。

5. 中断服务程序

        我们平时要使用哪个中断,就需要编写相应的中断服务程序,只是启动文件把这些函数留出来了,但是内容都是空的,真正的中断复服务程序需要我们在外部的 C 文件里面重新实现,这里只是提前占了一个位置罢了。

        这部分没啥好说的,和服务程序类似的,只需要注意‘B .’语句,B表示跳转,这里跳转到一个‘.’,即表示无线循环。

c6f1c67f76a94e83a258fa4508c14d5e.png

b4836b2c21d3483281fe227506248674.png

 6. 堆栈初始化

        堆栈初始化是由一个IF条件来实现的,MICROLIB的定义与否决定了堆栈的初始化方式。

        这个定义是在Options->Target中设置的

3339e34492734b5b8a2561f2cd27d61f.png

        这部分也没啥讲的,需要注意的是,ALIGN表示对指令或者数据存放的地址进行对齐,缺省表示4字节对齐。

e7c11f8a3d0b40019860c8d4cca34d1e.png

        

相关内容

热门资讯

Pod亲和性、Pod反亲和性、... 一、Pod亲和性  pod亲和性的对象为Pod,目的是实现,新建Pod和...
`nmap`模块是一个用于与N... 在Python中,nmap模块是一个用于与Nmap安全扫描器交互的库。Nmap...
深度学习调参 此文整理总结github上的一个资料,结尾附上链接。对于工程应用很有现实参考ÿ...
容器安全最佳实践和工具 容器安全最佳实践和工具什么是容器安全容器安全是指保护容器化应用程序和基础设施免受潜在威胁和攻击的措施...
重大科普!(wpk安卓版本)辅... 重大科普!(wpk安卓版本)辅助透视!(透视)外挂辅助器脚本(2020已更新)(哔哩哔哩);wpk软...
Macbook pro插移动硬... 为了弥补Macbook pro硬盘容量的缺失,我们有时候会使用到外接硬盘或移动硬盘。一...
方法辅助挂Wepoke辅助挂软... 方法辅助挂Wepoke辅助挂软件透明挂!(辅助挂)太离谱了原来确实真的是有挂(2021已更新)(哔哩...
C++客户端Qt开发——常用控... 7.布局管理器之前使用Qt在界面上创建的控件,都是通过"绝对定位"的方式来设定的也就是...
SCI一区级 | Matlab... 目录效果一览基本介绍程序设计参考资料效果一览基本介绍1.【SCI一区级】Matlab实现SSA-CN...
玩家必用!(WPK技术)透视辅... 玩家必用!(WPK技术)透视辅助!(透视)外挂辅助挂模拟器(2020已更新)(哔哩哔哩);超受欢迎的...