LPC2114啟動(dòng)代碼分析
1.異常向量表的建立
2.MCU各種模式堆棧的初始化
3.系統基本的初始化工作
下面分別進(jìn)行介紹.
(一)ARM相關(guān)指令及偽指令
LDR PC,ResetAddr
將ResetAddr標號地址所指的內容傳送給PC寄存器
LDR PC,=ResetAddr
將ResetAddr標號地址傳送給PC寄存器
ResetAddr DCD ResetInit
為ResetAddr分配一個(gè)字的地址空間,以ResetInit初始化,即ResetAddr地址所指的內容為ResetInit標號地址
SvcStackSpace Space SVC_STACK_LENGTH*4
為SvcStackSpace分配一塊SVC_STACK_LENGTH*4大小的地址區域,并以0初始化區域內容
(二)異常向量表的建立
異常是有內部或外部源產(chǎn)生的,以引起處理器處理的一個(gè)事件,異常出現后,CPU強制從異常類(lèi)型對應的固定存儲地址開(kāi)始執行程序,如當IRQ中斷產(chǎn)生后,CPU強制跳轉到0x00000018出執行代碼,我們要做的就是在這個(gè)代碼地址出編寫(xiě)相應的指令,讓它順利執行IRQ中斷程序.通常我們會(huì )在這里放置一條轉移指令,因為0x00000018只給你一個(gè)字的編程空間.
程序如下:
AREA vectors,CODE,READONLY
ENTRY
;interrupt vectors
;中斷向量表
Reset
LDR PC, ResetAddr
LDR PC, UndefinedAddr
LDR PC, SWI_Addr
LDR PC, PrefetchAddr
LDR PC, DataAbortAddr
DCD 0xb9205f80
LDR PC, [PC, #-0xff0]
LDR PC, FIQ_Addr
ResetAddr DCD ResetInit
UndefinedAddr DCD Undefined
SWI_Addr DCD SoftwareInterrupt
PrefetchAddr DCD PrefetchAbort
DataAbortAddr DCD DataAbort
Nouse DCD 0
IRQ_Addr DCD 0
FIQ_Addr DCD FIQ_Handler
;未定義指令
Undefined
B Undefined
;軟中斷
SoftwareInterrupt
B SoftwareInterrupt
;取指令中止
PrefetchAbort
B PrefetchAbort
;取數據中止
DataAbort
B DataAbort
;快速中斷
FIQ_Handler
STMFD SP!, {R0-R3, LR}
BL FIQ_Exception
LDMFD SP!, {R0-R3, LR}
SUBS PC, LR, #4
IRQ_Addr DCD 0
FIQ_Addr DCD FIQ_Handler
**
**
ResetInit
BL InitStack ;初始化堆棧 Initialize the stack
BL TargetResetInit ;目標板基本初始化 Initialize the target board
;跳轉到c語(yǔ)言入口 Jump to the entry point of C program
B __main
我們可以看到,每種異常都有相應的處理程序,如:當系統復位后,程序跳轉到0x00000000處執行指令,那么就執行 LDR PC, ResetAddr,及執行ResetInit地址處的代碼,這里放置了BL InitStack指令,負責完成各種模式下堆棧的初始化,接著(zhù)執行BL TargetResetInit,完成目標板基本初始化,最后進(jìn)入c語(yǔ)言入口,執行main函數.
(三)MCU各種模式堆棧的初始化
由于各種異常模式下都有自身的SP堆棧指針,因此就必須先進(jìn)入各自的異常模式進(jìn)行SP的設置,各種模式的切換可以通過(guò)改變CPSR來(lái)實(shí)現,程序如下:
InitStack
MOV R0, LR
;Build the SVC stack
;設置管理模式堆棧
MSR CPSR_c, #0xd3
LDR SP, StackSvc
;Build the IRQ stack
;設置中斷模式堆棧
MSR CPSR_c, #0xd2
LDR SP, StackIrq
;Build the FIQ stack
;設置快速中斷模式堆棧
MSR CPSR_c, #0xd1
LDR SP, StackFiq
;Build the DATAABORT stack
;設置中止模式堆棧
MSR CPSR_c, #0xd7
LDR SP, StackAbt
;Build the UDF stack
;設置未定義模式堆棧
MSR CPSR_c, #0xdb
LDR SP, StackUnd
;Build the SYS stack
;設置系統模式堆棧
MSR CPSR_c, #0x5f
LDR SP, =StackUsr
MOV PC, R0
**
**
StackSvc DCD SvcStackSpace + (SVC_STACK_LEGTH - 1)* 4
StackIrq DCD IrqStackSpace + (IRQ_STACK_LEGTH - 1)* 4
StackFiq DCD FiqStackSpace + (FIQ_STACK_LEGTH - 1)* 4
StackAbt DCD AbtStackSpace + (ABT_STACK_LEGTH - 1)* 4
StackUnd DCD UndtStackSpace + (UND_STACK_LEGTH - 1)* 4
**
**
;/* 分配堆??臻g */
AREA MyStacks, DATA, NOINIT, ALIGN=2
SvcStackSpace SPACE SVC_STACK_LEGTH * 4 ;Stack spaces for Administration Mode 管理模式堆??臻g
IrqStackSpace SPACE IRQ_STACK_LEGTH * 4 ;Stack spaces for Interrupt ReQuest Mode 中斷模式堆??臻g
FiqStackSpace SPACE FIQ_STACK_LEGTH * 4 ;Stack spaces for Fast Interrupt reQuest Mode 快速中斷模式堆??臻g
AbtStackSpace SPACE ABT_STACK_LEGTH * 4 ;Stack spaces for Suspend Mode 中止義模式堆??臻g
UndtStackSpace SPACE UND_STACK_LEGTH * 4 ;Stack spaces for Undefined Mode 未定義模式堆棧
****
***
AREA Stacks, DATA, NOINIT
StackUsr
說(shuō)明:MSR CPSR_c, #XX指令完成各種異常模式的切換,如切換到IRQ模式,則執行MSR CPSR_c, #0xd2,而SP的值由SvcStackSpace的地址加上SVC_STACK_LEGTH 的所指的大小而定.
(四)系統基本的初始化工作
完成堆棧初始化后執行BL TargetResetInit,它負責系統一些基本的初始化,如地址重映射,PLL時(shí)鐘初始化,VIC中斷初始化等,具體代碼見(jiàn)Easy Arm2100開(kāi)發(fā)版的target.c文件.
最后,執行B __main,__main是ADS的一個(gè)系統函數,它負責一些初始化環(huán)境然后執行用戶(hù)的main函數.
評論