STM32启动文件

一、复位电路

在了解启动文件之前需要明白STM32的复位中断流程,STM32的复位分为上电复位和手动复位,复位的电路图如下所示:
STM32启动文件
注意: 图中的复位电路是低电平复位,有的MCU是高电平复位。

  • 上电复位:顾名思义,上电复位就是STM32通电时,硬件自动复位的过程。从复位电路中可知,当芯片刚通电时电容两端没离子存在,所以处于充电过程,此时复位引脚等同于接地,这一过程成为上电复位。

  • 手动复位:手动复位是通过按键强行将复位引脚拉低,使芯片产生复位中断。

二、启动文件分析准备

  1. STM32的启动文件后缀是".s"的文件,打开项目是可以看到项目中有一个startup_stm32f103xb.s的文件,如下图所示:
    STM32启动文件
    打开文件后可以很清晰的看到STM32的启动流程,不过这里需要一些简单的汇编知识。没学过汇编的小伙伴也不用怕,我们只需要简单的分析即可,这里只分析流程不进行汇编指令的编写。
    STM32启动文件

  2. 启动文件中常用的汇编指令

    指令 作用
    EQU 定义字符常量,相当于C语言的 define
    AREA 汇编一个新的代码段或数据段
    SPACE 分配内存空间
    PRESERVE8 当前文件堆栈需按照 8 字节对其
    EXPORT 声明全局属性,可被外部文件使用
    DCD 以字为单位分配内存,要求4字节对齐,并初始化这些内存
    PROC 定义子程序,与 ENOP 成对使用,表示子程序结束
    WEAK 弱定义,如果外部文件声明了一个标号,则优先使用外部文件定义的标号,如果外部文件没有定义,则使用当前位置的标号
    IMPORT 声明标号来自外部文件,跟 C 语言中的 EXTERN 关键字类似
    B 跳转到一个标号
    END 到达文件的末尾,文件结束
    IF,ELSE,ENDIF 汇编条件分支语句
    LDR 从存储器中加载字到一个寄存器中
    BL 跳转到由寄存器/标号给出的地址,并把跳转前的下条指令地址保存到 LR
    BLX 跳转到由寄存器给出的地址,并根据寄存器的LSE确定处理器的状态,还要把跳转前的下条指令地址保存到LR
    BX 跳转到由寄存器/标号给出的地址,不用返回

    注意: ALIGN是编译器对指令或者数据的存放地址进行对齐,一般需要跟一个立即数,缺省表示 4 字节对齐。需要注意的是:这个不是ARM的指令,是编译器的

  3. 启动流程
    (1) 硬件上电复位。
    (2) 初始化指针 SP=_initial_sp 和 PC == Reset_Handler。
    (3) 执行复位中断服务程序。

三、启动文件的作用

  1. 初始化堆栈指针SP;
  2. 初始化程序计数器指针 PC;
  3. 设置堆、栈的大小;
  4. 设置异常向量表的入口;
  5. 配置外部SRAM作为数据存储器(这个由用户配置,一般的开发板没有外部SRAM)
  6. 设置C库的分支入口__main(最终调用mian函数)
  7. 使用库函数项目时,启动文件还调用了SystemInit函数配置系统时钟。

四、启动代码详解

  1. 开辟栈(stack)空间,用于局部变量、函数调用、函数的参数等使用。栈的大小不能超过内部SRAM大小。

    Stack_Size		EQU     0x400          AREA    STACK, NOINIT, READWRITE, ALIGN=3 Stack_Mem       SPACE   Stack_Size __initial_sp 
    • EQU:表示宏定义的伪指令,伪指令并不会生成二进制程序代码,也不会引起变量空间分配。0x400表示栈大小,注意这里是以字节为单位。
    • AREA:开辟一段可读写的数据空间,段名为stack,按照 8 字节对齐。AREA后面的关键字表示这个段的属性。
      (1) STACK:表示这个段的名字,可以任意命名。
      (2) NOINIT:表示此数据段不需要填入初始化数据。
      (3) READWRITE:表示此段可读写。
      (4) ALIGN=3:表示首地址按照 2 的 3次方对齐,也就是按照8字节对齐。
    • SPACE 这行指令告诉汇编器给STACK段分配0x400字节的连续内存空间。
    • __initial_sp紧挨SPACE放置,表示栈的结束机制,栈是从高往低使用,所以结束地址就是栈顶地址。
  2. 开辟堆(heap)空间,主要用于动态内存分配,也就是malloc、calloc、realloc等函数分配的变量空间是在堆上。

    Heap_Size      EQU     0x200          AREA    HEAP, NOINIT, READWRITE, ALIGN=3 __heap_base Heap_Mem        SPACE   Heap_Size __heap_limit 
    • __heap_base:表示堆的开始地址
    • __heap_limit:表示堆的结束地址
  3. 文件属性定义

    PRESERVE8 THUMB      ; Vector Table Mapped to Address 0 at Reset AREA    RESET, DATA, READONLY EXPORT  __Vectors EXPORT  __Vectors_End EXPORT  __Vectors_Size 
    • PRESERVE8:指定当前文件保持堆栈 8 字节对齐
    • THUMB:表示后面的指令是 THUMB 指令集,CM4 采用的是 THUMB - 2指令集
    • AREA:定义一块代码段,只读,段名是RESET。READONLY表示只读,缺省就表示代码段了
    • EXPORT:语句将3个标号声明为可被外部引用,主要提供给连接器用于连接库文件或其他文件
  4. 中断向量表

    __Vectors   DCD     __initial_sp               ; Top of Stack             DCD     Reset_Handler              ; Reset Handler             DCD     NMI_Handler                ; NMI Handler             DCD     HardFault_Handler          ; Hard Fault Handler             DCD     MemManage_Handler          ; MPU Fault Handler             DCD     BusFault_Handler           ; Bus Fault Handler             DCD     UsageFault_Handler         ; Usage Fault Handler             DCD     0                          ; Reserved             DCD     0                          ; Reserved             DCD     0                          ; Reserved             DCD     0                          ; Reserved             DCD     SVC_Handler                ; SVCall Handler             DCD     DebugMon_Handler           ; Debug Monitor Handler             DCD     0                          ; Reserved             DCD     PendSV_Handler             ; PendSV Handler             DCD     SysTick_Handler            ; SysTick Handler              ; External Interrupts             DCD     WWDG_IRQHandler            ; Window Watchdog             DCD     PVD_IRQHandler             ; PVD through EXTI Line detect             DCD     TAMPER_IRQHandler          ; Tamper             DCD     RTC_IRQHandler             ; RTC             DCD     FLASH_IRQHandler           ; Flash             DCD     RCC_IRQHandler             ; RCC             DCD     EXTI0_IRQHandler           ; EXTI Line 0             DCD     EXTI1_IRQHandler           ; EXTI Line 1             DCD     EXTI2_IRQHandler           ; EXTI Line 2             DCD     EXTI3_IRQHandler           ; EXTI Line 3             DCD     EXTI4_IRQHandler           ; EXTI Line 4             DCD     DMA1_Channel1_IRQHandler   ; DMA1 Channel 1             DCD     DMA1_Channel2_IRQHandler   ; DMA1 Channel 2             DCD     DMA1_Channel3_IRQHandler   ; DMA1 Channel 3             DCD     DMA1_Channel4_IRQHandler   ; DMA1 Channel 4             DCD     DMA1_Channel5_IRQHandler   ; DMA1 Channel 5             DCD     DMA1_Channel6_IRQHandler   ; DMA1 Channel 6             DCD     DMA1_Channel7_IRQHandler   ; DMA1 Channel 7             DCD     ADC1_2_IRQHandler          ; ADC1_2             DCD     USB_HP_CAN1_TX_IRQHandler  ; USB High Priority or CAN1 TX             DCD     USB_LP_CAN1_RX0_IRQHandler ; USB Low  Priority or CAN1 RX0             DCD     CAN1_RX1_IRQHandler        ; CAN1 RX1             DCD     CAN1_SCE_IRQHandler        ; CAN1 SCE             DCD     EXTI9_5_IRQHandler         ; EXTI Line 9..5             DCD     TIM1_BRK_IRQHandler        ; TIM1 Break             DCD     TIM1_UP_IRQHandler         ; TIM1 Update             DCD     TIM1_TRG_COM_IRQHandler    ; TIM1 Trigger and Commutation             DCD     TIM1_CC_IRQHandler         ; TIM1 Capture Compare             DCD     TIM2_IRQHandler            ; TIM2             DCD     TIM3_IRQHandler            ; TIM3             DCD     TIM4_IRQHandler            ; TIM4             DCD     I2C1_EV_IRQHandler         ; I2C1 Event             DCD     I2C1_ER_IRQHandler         ; I2C1 Error             DCD     I2C2_EV_IRQHandler         ; I2C2 Event             DCD     I2C2_ER_IRQHandler         ; I2C2 Error             DCD     SPI1_IRQHandler            ; SPI1             DCD     SPI2_IRQHandler            ; SPI2             DCD     USART1_IRQHandler          ; USART1             DCD     USART2_IRQHandler          ; USART2             DCD     USART3_IRQHandler          ; USART3             DCD     EXTI15_10_IRQHandler       ; EXTI Line 15..10             DCD     RTC_Alarm_IRQHandler        ; RTC Alarm through EXTI Line             DCD     USBWakeUp_IRQHandler       ; USB Wakeup from suspend __Vectors_End __Vectors_Size  EQU  __Vectors_End - __Vectors 
    • __Vectors:为向量表其实地址
    • __Vectors_End:为向量表结束地址
    • __Vectors_Size:为向量表的大小。
    • DCD:表示分配1个4字节的空间。每行DCD都会生成一个4字节的二进制代码。中断向量表存放的实际上是中断服务程序的入口地址,当异常(也就是中断实践)发生时,CPU的中断系统会将相应的入口地址赋值给PC程序计数器,之后就开始执行中断服务程序。
  5. 定义可读代码段

    AREA    |.text|, CODE, READONLY 
    • AREA:定义一个名为.test的可读代码段
  6. 复位程序

    ; Reset handler Reset_Handler    PROC              EXPORT  Reset_Handler             [WEAK] IMPORT  __main IMPORT  SystemInit              LDR     R0, =SystemInit              BLX     R0              LDR     R0, =__main              BX      R0              ENDP 
    • 复位子程序是系统上电后第一个执行的程序,调用SystemInit()函数初始化系统时钟,然后调用C库函数__main。
  7. 中断复位子程序

    ; Dummy Exception Handlers (infinite loops which can be modified)  NMI_Handler     PROC                 EXPORT  NMI_Handler                [WEAK]                 B       .                 ENDP HardFault_Handler                 PROC                 EXPORT  HardFault_Handler          [WEAK]                 B       .                 ENDP MemManage_Handler                 PROC                 EXPORT  MemManage_Handler          [WEAK]                 B       .                 ENDP BusFault_Handler                 PROC                 EXPORT  BusFault_Handler           [WEAK]                 B       .                 ENDP UsageFault_Handler                 PROC                 EXPORT  UsageFault_Handler         [WEAK]                 B       .                 ENDP SVC_Handler     PROC                 EXPORT  SVC_Handler                [WEAK]                 B       .                 ENDP DebugMon_Handler                 PROC                 EXPORT  DebugMon_Handler           [WEAK]                 B       .                 ENDP PendSV_Handler  PROC                 EXPORT  PendSV_Handler             [WEAK]                 B       .                 ENDP SysTick_Handler PROC                 EXPORT  SysTick_Handler            [WEAK]                 B       .                 ENDP  Default_Handler PROC                  EXPORT  WWDG_IRQHandler            [WEAK]                 EXPORT  PVD_IRQHandler             [WEAK]                 EXPORT  TAMPER_IRQHandler          [WEAK]                 EXPORT  RTC_IRQHandler             [WEAK]                 EXPORT  FLASH_IRQHandler           [WEAK]                 EXPORT  RCC_IRQHandler             [WEAK]                 EXPORT  EXTI0_IRQHandler           [WEAK]                 EXPORT  EXTI1_IRQHandler           [WEAK]                 EXPORT  EXTI2_IRQHandler           [WEAK]                 EXPORT  EXTI3_IRQHandler           [WEAK]                 EXPORT  EXTI4_IRQHandler           [WEAK]                 EXPORT  DMA1_Channel1_IRQHandler   [WEAK]                 EXPORT  DMA1_Channel2_IRQHandler   [WEAK]                 EXPORT  DMA1_Channel3_IRQHandler   [WEAK]                 EXPORT  DMA1_Channel4_IRQHandler   [WEAK]                 EXPORT  DMA1_Channel5_IRQHandler   [WEAK]                 EXPORT  DMA1_Channel6_IRQHandler   [WEAK]                 EXPORT  DMA1_Channel7_IRQHandler   [WEAK]                 EXPORT  ADC1_2_IRQHandler          [WEAK]                 EXPORT  USB_HP_CAN1_TX_IRQHandler  [WEAK]                 EXPORT  USB_LP_CAN1_RX0_IRQHandler [WEAK]                 EXPORT  CAN1_RX1_IRQHandler        [WEAK]                 EXPORT  CAN1_SCE_IRQHandler        [WEAK]                 EXPORT  EXTI9_5_IRQHandler         [WEAK]                 EXPORT  TIM1_BRK_IRQHandler        [WEAK]                 EXPORT  TIM1_UP_IRQHandler         [WEAK]                 EXPORT  TIM1_TRG_COM_IRQHandler    [WEAK]                 EXPORT  TIM1_CC_IRQHandler         [WEAK]                 EXPORT  TIM2_IRQHandler            [WEAK]                 EXPORT  TIM3_IRQHandler            [WEAK]                 EXPORT  TIM4_IRQHandler            [WEAK]                 EXPORT  I2C1_EV_IRQHandler         [WEAK]                 EXPORT  I2C1_ER_IRQHandler         [WEAK]                 EXPORT  I2C2_EV_IRQHandler         [WEAK]                 EXPORT  I2C2_ER_IRQHandler         [WEAK]                 EXPORT  SPI1_IRQHandler            [WEAK]                 EXPORT  SPI2_IRQHandler            [WEAK]                 EXPORT  USART1_IRQHandler          [WEAK]                 EXPORT  USART2_IRQHandler          [WEAK]                 EXPORT  USART3_IRQHandler          [WEAK]                 EXPORT  EXTI15_10_IRQHandler       [WEAK]                 EXPORT  RTC_Alarm_IRQHandler        [WEAK]                 EXPORT  USBWakeUp_IRQHandler       [WEAK]  WWDG_IRQHandler PVD_IRQHandler TAMPER_IRQHandler RTC_IRQHandler FLASH_IRQHandler RCC_IRQHandler EXTI0_IRQHandler EXTI1_IRQHandler EXTI2_IRQHandler EXTI3_IRQHandler EXTI4_IRQHandler DMA1_Channel1_IRQHandler DMA1_Channel2_IRQHandler DMA1_Channel3_IRQHandler DMA1_Channel4_IRQHandler DMA1_Channel5_IRQHandler DMA1_Channel6_IRQHandler DMA1_Channel7_IRQHandler ADC1_2_IRQHandler USB_HP_CAN1_TX_IRQHandler USB_LP_CAN1_RX0_IRQHandler CAN1_RX1_IRQHandler CAN1_SCE_IRQHandler EXTI9_5_IRQHandler TIM1_BRK_IRQHandler TIM1_UP_IRQHandler TIM1_TRG_COM_IRQHandler TIM1_CC_IRQHandler TIM2_IRQHandler TIM3_IRQHandler TIM4_IRQHandler I2C1_EV_IRQHandler I2C1_ER_IRQHandler I2C2_EV_IRQHandler I2C2_ER_IRQHandler SPI1_IRQHandler SPI2_IRQHandler USART1_IRQHandler USART2_IRQHandler USART3_IRQHandler EXTI15_10_IRQHandler RTC_Alarm_IRQHandler USBWakeUp_IRQHandler                  B       .                  ENDP  
    • WEAK:如果外部文件中定义了此中断函数,优先使用外部文件中的中断函数,反之使用当前中断函数。
    • ".":表示无限循环
    • 如果我们在使用某个外设的时候,开启了某个中断,但是又忘记编写配套的中断服务程序需或者函数名写错时,当相应的中断产生时,程序就会跳转到启动文件预先写好的空中断函数中,并且在这个空函数中无限循环,即程序就死在这里。
  8. 用户堆栈初始化

                   ALIGN  ;******************************************************************************* ; User Stack and Heap initialization ;*******************************************************************************                  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                   END 
    • ALIGN:对指令或者数据存放的地址进行对齐,后面会跟一个立即数。缺省表示4字节对齐。
    • __user_initial_stackheap:初始化栈和堆(使用默认C库时,由__main函数进行调用)
    • 如果使用微库时,调用__MICROLIB部分的程序,反之使用默认的C库,然后初始化用户堆栈大小。微库的使用如下图所示:
      STM32启动文件

参考文献

STM32启动文件————startup_stm32f10x_hd.s:https://wenku.baidu.com/view/3275eee00ba1284ac850ad02de80d4d8d15a0198.html

发表评论

相关文章