STM32啟動文件詳解


 

 

啟動文件使用的 ARM 匯編指令匯總

1

        啟動程序源碼注釋(點此下載)

1. 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 語句放置,表示棧的結束地址,即棧頂地址,棧是由高向低生長的。

2. Heap 堆

Heap_Size EQU 0x00000200
AREA HEAP, NOINIT, READWRITE, ALIGN=3
__heap_base
Heap_Mem SPACE Heap_Size
heap_limit

       開辟堆的大小為 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 的超集。

3. 向量表

AREA RESET, DATA, READONLY
EXPORT __Vectors
EXPORT __Vectors_End
EXPORT __Vectors_Size

        定義一個數據段,名字為 RESET,可讀。並聲明 __Vectors、 __Vectors_End 和__Vectors_Size 這三個標號具有全局屬性,可供外部的文件調用。

        EXPORT: 聲明一個標號可被外部的文件使用,使標號具有全局屬性。如果是 IAR 編譯器,則使用的是 GLOBAL 這個指令。

       當內核響應了一個發生的異常后,對應的異常服務例程(ESR)就會執行。為了決定 ESR的入口地址, 內核使用了―向量表查表機制‖。這里使用一張向量表。向量表其實是一個WORD(32 位整數)數組,每個下標對應一種異常,該下標元素的值則是該 ESR 的入口地址。向量表在地址空間中的位置是可以設置的,通過 NVIC 中的一個重定位寄存器來指出向量表的地址。在復位后,該寄存器的值為 0。因此,在地址 0 (即 FLASH 地址 0) 處必須包含一張向量表,用於初始時的異常分配。要注意的是這里有個另類: 0 號類型並不是什么入口地址,而是給出了復位后 MSP 的初值。

表格 15-2 F103 向量表

1

 

__Vectors DCD __initial_sp ;棧頂地址
DCD Reset_Handler ;復位程序地址
DCD NMI_Handler
DCD HardFault_Handler
DCD MemManage_Handler
DCD BusFault_Handler
DCD UsageFault_Handler
DCD 0 ; 0 表示保留
DCD 0
DCD 0
DCD 0
DCD SVC_Handler
DCD DebugMon_Handler
DCD 0
DCD PendSV_Handler
DCD SysTick_Handler
;外部中斷開始
DCD WWDG_IRQHandler
DCD PVD_IRQHandler
DCD TAMPER_IRQHandler
;限於篇幅,中間代碼省略
DCD DMA2_Channel2_IRQHandler
DCD DMA2_Channel3_IRQHandler
DCD DMA2_Channel4_5_IRQHandler
__Vectors_End
__Vectors_Size EQU __Vectors_End - __Vectors

        __Vectors 為向量表起始地址, __Vectors_End 為向量表結束地址,兩個相減即可算出向量表大小。

        向量表從 FLASH 的 0 地址開始放置,以 4 個字節為一個單位,地址 0 存放的是棧頂地址, 0X04 存放的是復位程序的地址,以此類推。從代碼上看,向量表中存放的都是中斷服務函數的函數名,可我們知道 C 語言中的函數名就是一個地址。

        DCD:分配一個或者多個以字為單位的內存,以四字節對齊,並要求初始化這些內存。在向量表中, DCD 分配了一堆內存,並且以 ESR 的入口地址初始化它們。

 

4. 復位程序

 AREA |.text|, CODE, READONLY

          定義一個名稱為.text 的代碼段,可讀。

Reset_Handler PROC
EXPORT Reset_Handler [WEAK]
IMPORT SystemInit
IMPORT __main
LDR R0, =SystemInit
BLX R0
LDR R0, =__main
BX R0
ENDP

         復位子程序是系統上電后第一個執行的程序,調用 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》第四章-指令集里面查詢到,具體作用見下表:

1

 

5. 中斷服務程序

        在啟動文件里面已經幫我們寫好所有中斷的中斷服務函數,跟我們平時寫的中斷服務函數不一樣的就是這些函數都是空的,真正的中斷復服務程序需要我們在外部的 C 文件里面重新實現,這里只是提前占了一個位置而已。

        如果我們在使用某個外設的時候,開啟了某個中斷,但是又忘記編寫配套的中斷服務程序或者函數名寫錯,那當中斷來臨的時,程序就會跳轉到啟動文件預先寫好的空的中斷服務程序中,並且在這個空函數中無線循環,即程序就死在這里。

NMI_Handler PROC ;系統異常
EXPORT NMI_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 TAMP_STAMP_IRQHandler [WEAK]
;限於篇幅,中間代碼省略
LTDC_IRQHandler
LTDC_ER_IRQHandler
DMA2D_IRQHandler
B .

ENDP

       B:跳轉到一個標號。這里跳轉到一個‘.’,即表示無線循環

6. 用戶堆棧初始化

ALIGN

        ALIGN:對指令或者數據存放的地址進行對齊,后面會跟一個立即數。缺省表示 4 字節對齊。

;用戶棧和堆初始化,由 C 庫函數_main 來完成
IF :DEF:__MICROLIB ;這個宏在 KEIL 里面開啟
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

        首先判斷是否定義了__MICROLIB ,如果定義了這個宏則賦予標號__initial_sp(棧頂地址)、 __heap_base(堆起始地址)、 __heap_limit(堆結束地址)全局屬性,可供外部文件調用。有關這個宏我們在 KEIL 里面配置,具體見圖 15-2。然后堆棧的初始化就由 C 庫函數_main 來完成。

1

       如 果沒 有 定 義 __MICROLIB , 則 才 用 雙 段 存 儲 器 模 式 , 且 聲 明 標 號__user_initial_stackheap 具有全局屬性,讓用戶自己來初始化堆棧。

       IF,ELSE,ENDIF:匯編的條件分支語句,跟 C 語言的 if ,else 類似

       END:文件結束

 


免責聲明!

本站轉載的文章為個人學習借鑒使用,本站對版權不負任何法律責任。如果侵犯了您的隱私權益,請聯系本站郵箱yoyou2525@163.com刪除。



 
粵ICP備18138465號   © 2018-2025 CODEPRJ.COM