关于STM32启动文件startup_stm32f10x_hd.s的代码

发布者:悠闲自在最新更新时间:2024-07-23 来源: elecfans关键字:STM32  启动文件  startup_stm32f10x_hd 手机看文章 扫描二维码
随时随地手机看文章

本文对STM32启动文件startup_stm32f10x_hd.s的代码进行讲解,此文件的代码在任何一个STM32F10x工程中都可以找到。


启动文件使用的ARM汇编指令汇总

2cbc5340-fd0b-11ec-ba43-dac502259ad0.png?imageView2/2/w/1000

Stack——栈


Stack_Size EQU 0x00000400

AREA STACK, NOINIT, READWRITE, ALIGN=Stack_Mem SPACE Stack_Size__initial_sp

开辟栈的大小为 0X00000400(1KB),名字为 STACK, NOINIT 即不初始化,可读可写, 8(2^3)字节对齐。


栈的作用是用于局部变量,函数调用,函数形参等的开销,栈的大小不能超过内部SRAM 的大小。如果编写的程序比较大,定义的局部变量很多,那么就需要修改栈的大小。如果某一天,你写的程序出现了莫名奇怪的错误,并进入了硬 fault 的时候,这时你就要考虑下是不是栈不够大,溢出了。


EQU:宏定义的伪指令,相当于等于,类似于C 中的 define。


AREA:告诉汇编器汇编一个新的代码段或者数据段。STACK 表示段名,这个可以任意命名;NOINIT 表示不初始化;READWRITE 表示可读可写, ALIGN=3,表示按照 2^3对齐,即 8 字节对齐。


SPACE:用于分配一定大小的内存空间,单位为字节。这里指定大小等于 Stack_Size。


标号__initial_sp 紧挨着 SPACE 语句放置,表示栈的结束地址,即栈顶地址,栈是由高向低生长的。


Heap——堆


2ccf121e-fd0b-11ec-ba43-dac502259ad0.png?imageView2/2/w/1000

开辟堆的大小为 0X00000200(512 字节),名字为 HEAP, NOINIT 即不初始化,可读可写, 8(2^3)字节对齐。__heap_base 表示对的起始地址, __heap_limit 表示堆的结束地址。堆是由低向高生长的,跟栈的生长方向相反。


堆主要用来动态内存的分配,像 malloc()函数申请的内存就在堆上面。这个在 STM32里面用的比较少。


PRESERVE8 THUMB

PRESERVE8:指定当前文件的堆栈按照 8 字节对齐。


THUMB:表示后面指令兼容 THUMB 指令。THUBM 是 ARM 以前的指令集, 16bit,现在 Cortex-M 系列的都使用 THUMB-2 指令集, THUMB-2 是 32 位的,兼容 16 位和 32 位的指令,是 THUMB 的超集。关于堆栈的文章:关于C语言堆栈的经典讲解。


向量表


AREA RESET, DATA, READONLYEXPORT __VectorsEXPORT __Vectors_EndEXPORT __Vectors_Size

定义一个数据段,名字为 RESET,可读。并声明 __Vectors、 __Vectors_End 和__Vectors_Size 这三个标号具有全局属性,可供外部的文件调用。


EXPORT:声明一个标号可被外部的文件使用,使标号具有全局属性。如果是 IAR 编译器,则使用的是 GLOBAL 这个指令。


当内核响应了一个发生的异常后,对应的异常服务例程(ESR)就会执行。为了决定 ESR的入口地址, 内核使用了―向量表查表机制‖。这里使用一张向量表。向量表其实是一个WORD(32 位整数)数组,每个下标对应一种异常,该下标元素的值则是该 ESR 的入口地址。向量表在地址空间中的位置是可以设置的,通过 NVIC 中的一个重定位寄存器来指出向量表的地址。在复位后,该寄存器的值为 0。因此,在地址 0 (即 FLASH 地址 0) 处必须包含一张向量表,用于初始时的异常分配。要注意的是这里有个另类:0 号类型并不是什么入口地址,而是给出了复位后 MSP 的初值。下图是F103的向量表。

2ce4a6e2-fd0b-11ec-ba43-dac502259ad0.png?imageView2/2/w/1000

__Vectors DCD __initial_sp ;栈顶地址DCD Reset_Handler ;复位程序地址DCD NMI_HandlerDCD HardFault_HandlerDCD MemManage_HandlerDCD BusFault_HandlerDCD UsageFault_HandlerDCD 0 ; 0 表示保留DCD 0DCD 0DCD 0DCD SVC_HandlerDCD DebugMon_HandlerDCD 0DCD PendSV_HandlerDCD SysTick_Handler;外部中断开始DCD WWDG_IRQHandlerDCD PVD_IRQHandlerDCD TAMPER_IRQHandler;限于篇幅,中间代码省略DCD DMA2_Channel2_IRQHandlerDCD DMA2_Channel3_IRQHandlerDCD DMA2_Channel4_5_IRQHandler__Vectors_End__Vectors_Size EQU __Vectors_End - __Vectors

__Vectors 为向量表起始地址, __Vectors_End 为向量表结束地址,两个相减即可算出向量表大小。


向量表从 FLASH 的 0 地址开始放置,以 4 个字节为一个单位,地址 0 存放的是栈顶地址, 0X04 存放的是复位程序的地址,以此类推。从代码上看,向量表中存放的都是中断服务函数的函数名,可我们知道 C 语言中的函数名就是一个地址。


DCD:分配一个或者多个以字为单位的内存,以四字节对齐,并要求初始化这些内存。在向量表中, DCD 分配了一堆内存,并且以 ESR 的入口地址初始化它们。


复位程序


AREA |.text|, CODE, READONLY

定义一个名称为.text 的代码段,可读。

2d195fc2-fd0b-11ec-ba43-dac502259ad0.png?imageView2/2/w/1000

复位子程序是系统上电后第一个执行的程序,调用 SystemInit 函数初始化系统时钟,然后调用 C 库函数_mian,最终调用 main 函数去到 C 的世界。


WEAK:表示弱定义,如果外部文件优先定义了该标号则首先引用该标号,如果外部文件没有声明也不会出错。这里表示复位子程序可以由用户在其他文件重新实现,这里并不是唯一的。


IMPORT:表示该标号来自外部文件,跟 C 语言中的 EXTERN 关键字类似。这里表示 SystemInit 和__main 这两个函数均来自外部的文件。


SystemInit()是一个标准的库函数,在 system_stm32f10x.c 这个库文件中定义。主要作用是配置系统时钟,这里调用这个函数之后,单片机的系统时钟配被配置为 72M。__main 是一个标准的 C 库函数,主要作用是初始化用户堆栈,并在函数的最后调用main 函数去到 C 的世界。这就是为什么我们写的程序都有一个 main 函数的原因。


LDR、 BLX、 BX 是 CM4 内核的指令,可在《CM3 权威指南 CnR2》第四章-指令集里面查询到,具体作用见下表:


2d2861ac-fd0b-11ec-ba43-dac502259ad0.png?imageView2/2/w/1000

中断服务程序


在启动文件里面已经帮我们写好所有中断的中断服务函数,跟我们平时写的中断服务函数不一样的就是这些函数都是空的,真正的中断服务程序需要我们在外部的 C 文件里面重新实现,这里只是提前占了一个位置而已。


如果我们在使用某个外设的时候,开启了某个中断,但是又忘记编写配套的中断服务程序或者函数名写错,那当中断来临的时,程序就会跳转到启动文件预先写好的空的中断服务程序中,并且在这个空函数中无线循环,即程序就死在这里。


NMI_Handler PROC ;系统异常EXPORT NMI_Handler [WEAK]B .ENDP;限于篇幅,中间代码省略SysTick_Handler PROCEXPORT SysTick_Handler [WEAK]B .ENDPDefault_Handler PROC ;外部中断EXPORT WWDG_IRQHandler [WEAK]EXPORT PVD_IRQHandler [WEAK]EXPORT TAMP_STAMP_IRQHandler [WEAK];限于篇幅,中间代码省略LTDC_IRQHandlerLTDC_ER_IRQHandlerDMA2D_IRQHandlerB .ENDP

B:跳转到一个标号。这里跳转到一个‘.’,即表示无线循环


用户堆栈初始化


ALIGN

ALIGN:对指令或者数据存放的地址进行对齐,后面会跟一个立即数。缺省表示 4 字节对齐。


;用户栈和堆初始化,由 C 库函数_main 来完成IF __MICROLIB ;这个宏在 KEIL 里面开启EXPORT __initial_spEXPORT __heap_baseEXPORT __heap_limitELSEIMPORT __use_two_region_memory ; 这个函数由用户自己实现EXPORT __user_initial_stackheap__user_initial_stackheapLDR R0, = Heap_MemLDR R1, =(Stack_Mem + Stack_Size)LDR R2, = (Heap_Mem + Heap_Size)LDR R3, = Stack_MemBX LRALIGNENDIFEND

首先判断是否定义了__MICROLIB ,如果定义了这个宏则赋予标号__initial_sp(栈顶地址)、 __heap_base(堆起始地址)、 __heap_limit(堆结束地址)全局属性,可供外部文件调用。有关这个宏我们在 KEIL 里面配置,具体见下图。然后堆栈的初始化就由 C 库函数_main 来完成。

2d373236-fd0b-11ec-ba43-dac502259ad0.png?imageView2/2/w/1000

如果没有定义__MICROLIB,则才用双段存储器模式,且声明标号__user_initial_stackheap 具有全局属性,让用户自己来初始化堆栈。


前文的汇编代码,需要注意:


IF,ELSE,ENDIF:汇编的条件分支语句,跟 C 语言的 if ,else 类似


关键字:STM32  启动文件  startup_stm32f10x_hd 引用地址:关于STM32启动文件startup_stm32f10x_hd.s的代码

上一篇:讲讲开发STM32的四种库
下一篇:STM32WB55_NUCLEO开发(4)----手机与STM32WB通过密钥配对

推荐阅读最新更新时间:2026-03-20 11:11

STM32启动文件:startup_stm32f10x_hd.s启动文件的简单描述
在官方的库文件中,分别有如下文件: startup │ │ │ ├─arm │ │ │ │ startup_stm32f10x_cl.s │ │ │ │ startup_stm32f10x_hd.s │ │ │ │ startup_stm32f10x_hd_vl.s │ │ │ │ startup_stm32f10x_ld.s │ │ │ │ startup_stm32f10x_ld_vl.s │ │ │ │ startup_stm32f10x_md.s │ │ │ │ s
[单片机]
STM32 CM3/CM4 ------ startup.s 启动文件分析 ------ GCC RT-Thread Studio 版本
startup.s 功能 startup.s 文件定义了向量表,包含栈初始值和各个中断服务函数指针。 芯片一上电,自动设置SP,PC,然后执行复位中断:   设置栈顶寄存器的值 --- 多余,芯片一上电自动设置了   data段的数据从 flash 拷贝到 SRAM   bss段位于SRAM,初始化为0   执行 SystemInit,初始化时钟,设置SCB- VTOR的值为向量表起始地址   跳转到 entry 函数,entry() 函数添加一些我们要在 main() 函数之前执行的代码,entry() 函数的最后调用 main()。 向量表 在 startup.s 内,定义了一个向量表,由链接脚本决
[单片机]
<font color='red'>STM32</font> CM3/CM4 ------ startup.<font color='red'>s</font> <font color='red'>启动文件</font>分析 ------ GCC RT-Thread Studio 版本
stm32启动文件--startup_stm32f10x_hd.s
在学习stm32 定时器中断时,发现我写的程序进入不了中断服务程序,而且我也没找到定时器服务程序的注册入口,上网查了一下,大家都使用的都是下面这个函数:void TIM3_IRQHandler(void),那为什么其他人的中断程序可以执行,而我的却执行不了呢。 经过请教得知,我的工程中没有包含该一个叫做startup_stm32f10x_hd.s的文件,而我所谓的中断服务程序的注册入口就是包含在这里。startup_stm32f10x_hd.s是用汇编语言编写的STM32处理器的初始化程序,它规定了系统的堆栈大小等系统信息,也包含了一个向量表,这个向量表中对所有的中断入口函数都进行了分配,如TIM3的中断入口是: ……
[单片机]
STM32 程序卡死在启动文件startup_stm32fxxxx.s的SysTick_Handler中
出现的问题 硬件调试的时候发现程序一直在启动文件下面箭头这行。显然是跟SysTick_Handler有关 由于我使用了FreeRTOS,所以需要SysTick为系统时基单元,所以需要SysTick_Handler,但是我却没有定义此中断函数,所以导致上面的程序一直进入不了以SysTick_handler命名的中断函数;填上就好了; 解决问题 添加SysTick_Handler函数:
[单片机]
<font color='red'>STM32</font> 程序卡死在<font color='red'>启动文件</font>startup_stm32fxxxx.<font color='red'>s</font>的SysTick_Handler中
STM32启动文件分析——startup_stm32f10x_hd.s
一、启动文件的作用 (关于启动代码的作用,前面已经提到过了,这里再啰嗦一下) (1)初始化堆栈指针 SP; (2)初始化程序计数器指针 PC; (3)设置堆、栈的大小; (4)设置异常向量表的入口地址; (5)配置外部 SRAM作为数据存储器(这个由用户配置,一般的开发板可没有外部 SRAM); (6)设置 C库的分支入口__main(最终用来调用 main函数); (7)在 3.5版的启动文件还调用了在 system_stm32f10x.c文件中的SystemIni()函数配置系统时钟。 二、启动文件中提到的汇编指令 指令 作用 EQU 给数字常量取一个符号名,相当于 C 语言中的
[单片机]
STM32启动文件简介、详细步骤及代码讲解
01启动文件简介 startup_stm32f429_439xx.s是STM32的启动文件。 刚开始我一直认为STM32程序开始执行是从main函数开始。后来网上查查不是。原来在执行main函数之前,需要先执行一段汇编程序和完成C语言资源硬件的初始化工作。就是以下几个功能: 1--初始化栈指针MSP=_initial_sp。 2--初始化复位程序计数寄存器值=Reset_Handler。 3--初始化异常/ 中断向量表。 4--系统时钟配置。 5--C库函数_main初始化用户堆栈的调用 。 02文件启动步骤 1-在启动的时候,先对堆栈的大小定义,并在代码区的起始位置建立异常中断向量表。然后在复位中断中服务程序中跳转执行C标
[单片机]
<font color='red'>STM32</font><font color='red'>启动文件</font>简介、详细步骤及代码讲解
关于STM32启动文件的几个小问题
基于STM32芯片的工程代码里有个很重要的文件,即启动文件。该文件主要由汇编语言写成,文件名冠以.s结尾,它是芯片程序运行首先要执行的一个文件。其功能及作用简单点说就是做执行用户程序前的基本准备,比方执行复位程序初始化栈、堆,做时钟系统的默认配置、中断矢量表的定义与分配等。 网络上有些文章对该文件做了不错的整体性介绍,这里仅就该文件中的几个小问题一起交流分享下。【注:下面用到的工程是基于STM32F429的,使用IDE为ARM MDK】 怎么在编译后的MAP文件里看不到变量__heap_base? 有人发现在启动文件里明明有定义__initial_sp和__heap_base,可在MAP文件里只看到__initial_sp的地
[单片机]
STM32学习笔记(三)---启动文件
启动文件简介 启动文件由汇编编写,是系统上电复位后第一个执行的程序。 初始化堆栈指针 SP=_initial_sp 初始化PC指针=Reset_Handler 初始化中断向量表 配置系统时钟 调用C库函数_main初始化用户堆栈,最终调用main函数去到C的世界 启动文件中用到的ARM汇编指令 启动文件分析 1. 栈 Stack_Size EQU 0x00000400 AREA STACK, NOINIT, READWRITE, ALIGN=3 Stack_Mem SPACE Stack_Size __initial_sp 开辟栈空间大小为0x00000400(1kb),名字为ST
[单片机]
<font color='red'>STM32</font>学习笔记(三)---<font color='red'>启动文件</font>
小广播
最新单片机文章
何立民专栏 单片机及嵌入式宝典

北京航空航天大学教授,20余年来致力于单片机与嵌入式系统推广工作。

厂商技术中心

 
EEWorld订阅号

 
EEWorld服务号

 
汽车开发圈

 
机器人开发圈

电子工程世界版权所有 京ICP证060456号 京ICP备10001474号-1 电信业务审批[2006]字第258号函 京公网安备 11010802033920号 Copyright © 2005-2026 EEWORLD.com.cn, Inc. All rights reserved