PART1
系统初始化流程如下:禁止看门狗----》在中断控制器中屏蔽所有中断----》系统时钟设置----》初始化端口----》DMA设置----》cashe和总线设置----》存储器设置,初始化SDRAM----》初始化堆栈----》设置IRQ和FIQ的入口----》地址重映射。必须由汇编来完成的任务有:异常中断向量表的设置、IRQ向量表(向量模式或ISR 初始化(非向量模式、二级ISR地址表的定义、Flash和SDRAM的设置(否则系统无法加载代码、堆栈设置和模式切换、拷贝RW和ZI代码、设置系统时钟等。对于非向量模式,不使用IRQ中断向量表,但二级ISR地址表的设置是相同的。
PART2
理解启动代码(ADS
所谓启动代码,就是处理器在启动的时候执行的一段代码,主要任务是初始化处理器模式,设置堆栈,初始化变量等等.由于以上的操作均与处理器体系结构和系统配置密切相关,所以一般由汇编来编写.
具体到S64,启动代码分成两部分,一是与ARM7TDMI内核相关的部分,包括处理器各异常向量的配置,各处理器模式的堆栈设置,如有必要,复制向量到RAM,以便remap之后处理器正确处理异常,初始化数据(包括RW与ZI,最后跳转到Main.二是与处理器外部设备相关的部
分,这和厂商的联系比较大.虽然都采用了ARM7TDMI的内核,但是不同的厂家整合了不同的片上外设,需要不同的初始化,其中比较重要的是初始化WDT,初始化各子系统时钟,有必要的话,进行remap.这一部分与一般控制器的初始化类似,因此,本文不作重点描述.
在进行分析之前,请确认如下相关概念:
S64片上FLASH起始于0x100000,共64kB,片上RAM起始于0x200000,共
16kB.
S64复位之后,程序会从0开始执行,此时FLASH被映射到0地址,因此,S64可以取得指令并执行.显然,此时还是驻留在0x100000地址.如果使用remap命令,将会把RAM映射到0地址,同样的这时0地址的内容也只是RAM的镜像.
S64的FLASH可以保证在最差情况时以30MHz进行单周期访问,而RAM可以保证在最大速度时的单周期访问.
OK,以下开始分析启动代码.
一,处理器异常
S64将异常向量至于0地址开始的几个直接,这些是必需要处理的.由于复位向量位于0,也需要一条跳转指令.具体代码如下:
RESET
B SYSINIT ; Reset
B UDFHANDLER ; UNDEFINED
B SWIHANDLER ; SWI
B PABTHANDLER ; PREFETCH ABORT
B DABTHANDLER ; DATA ABORT
B . ; RESERVED
B VECTORED_IRQ_HANDLER
B . ; ADD FIQ CODE HERE UDFHANDLER
B . SWIHANDLER
B . PABTHANDLER
B . DABTHANDLER
B .
请注意,B指令经汇编后会替换为当前PC值加上一个修正值(+/-,所以这条指令是代码位置无关的,也就是不管这条指令是在0地址还是在0x100000执行,都能跳转到指定的位置,而LDR PC,=???将向PC直接装载一个标号的值,请注意,标号在编译
过后将被替换为一个与RO 相对应的值,也就是说,这样的指令无论在哪里执行,都只会跳转到一个指定的位置.下面举一个具体的例子来说明两者的区别: 假定有如下程序:
RESET
B INIT 或者LDR PC,=INIT
…
INIT
…
其中RESET为起始时的代码,也就是这条代码的偏移为0,设INIT的偏移量为offset.如果将这段程序按照RO=0x1000000编译, 那么B INIT可理解为ADD PC, PC, #offset,而LDR PC,=INIT可被理解为MOV PC,#(RO+offset .显然当系统复位时,程序从0开始运行,而0地址有FLASH的副本,执行B INIT将把PC指向位于0地址处
的镜像代码位置,也即INIT;如果执行LDR PC,=INIT将会将PC 直接指向位于FLASH中的原始代码.因此以上两者都能正确运行.下面将RO设置为0x200000,编
译后生成代码,还是得烧写到FLASH中,也就是还是0x100000,系统复位后从0地址执行,还是FLASH的副本,此时执行B INIT,将跳到副本中的INIT位置执行,此处有
对应的代码;但是如果执行LDR PC,=INIT,将向PC加载0x200000+offset,这将使得PC跳到RAM中,而此时由于代码没有复制,RAM中的指定位置并没有代码,程序无法运行.
二,处理器模式
ARM的处理器可工作于多种模式,不同模式有不同的堆栈,以下设置各模式及其堆栈.
预定义一些参数:
MODUSR EQU 0x10
MODSYS EQU 0x1F
MODSVC EQU 0x13
MODABT EQU 0x17
MODUDF EQU 0x1B
MODIRQ EQU 0x12
MODFIQ EQU 0x11
IRQBIT EQU 0x80
FIQBIT EQU 0x40
RAMEND EQU 0x00204000 ; S64 : 16KB RAM
VECTSIZE EQU 0x100 ;
UsrStkSz EQU 8 ; size of USR stack SysStkSz EQU 128 ; size of SYS stack SvcStkSz EQU 8 ; size of SVC stack UdfStkSz EQU 8 ; size of UDF stack AbtStkSz EQU 8 ; size of ABT stack IrqStkSz EQU 128 ; size of IRQ stack FiqStkSz EQU 16 ; size of FIQ stack
修改这些值即可修改相应模式堆栈的尺寸.
以下为各模式代码:
SYSINIT
;
MRS R0,CPSR
BIC R0,R0,#0x1F
MOV R2,#RAMEND
ORR R1,R0,#(MODSVC :OR: IRQBIT :OR: FIQBIT MSR cpsr_cxsf,R1 ; ENTER SVC MODE
MOV sp,R2
SUB R2,R2,#SvcStkSz
ORR R1,R0,#(MODFIQ :OR: IRQBIT :OR: FIQBIT MSR CPSR_cxsf,R1 ; ENTER FIQ MODE
MOV sp,R2
SUB R2,R2,#FiqStkSz
ORR R1,R0,#(MODIRQ :OR: IRQBIT :OR: FIQBIT MSR CPSR_cxsf,R1 ; ENTER IRQ MODE
MOV sp,R2
SUB R2,R2,#IrqStkSz
ORR R1,R0,#(MODUDF :OR: IRQBIT :OR: FIQBIT
MSR CPSR_cxsf,R1 ; ENTER UDF MODE
MOV sp,R2
SUB R2,R2,#UdfStkSz
ORR R1,R0,#(MODABT :OR: IRQBIT :OR: FIQBIT
MSR CPSR_cxsf,R1 ; ENTER ABT MODE
MOV sp,R2
SUB R2,R2,#AbtStkSz
;ORR R1,R0,#(MODUSR :OR: IRQBIT :OR: FIQBIT
;MSR CPSR_cxsf,R1 ; ENTER USR MODE
;MOV sp,R2
;SUB R2,R2,#UsrStkSz
ORR R1,R0,#(MODSYS :OR: IRQBIT :OR: FIQBIT
MSR CPSR_cxsf,R1 ; ENTER SYS MODE
MOV sp,R2 ;
三,初始化变量
编译完成之后,连接器会生成三个基本的段,分别是RO,RW,ZI,并会在image中顺序摆放.显然,RW,ZI在运行开始时并不位于指定的
RW位置,因此必须初始化
LDR R0,=|Image$$RO$$Limit|
LDR R1,=|Image$$RW$$Base|
LDR R2,=|Image$$ZI$$Base|
1
CMP R1,R2
LDRLO R3,[R0],#4
STRLO R3,[R1],#4
BLO %B1
MOV R3,#0
LDR R1,=|Image$$ZI$$Limit|
2
CMP R2,R1
STRLO R3,[R2],#4
BLO %B2
四,复制异常向量
由于代码于RAM运行时,有明显的速度优势,而且变量可以动态配置,因此可以通过remap将RAM映射到0,使得出现异常时ARM从RAM中取得向量.
IMPORT |Image$$RO$$Base|
IMPORT |Image$$RO$$Limit|
IMPORT |Image$$RW$$Base|
IMPORT |Image$$RW$$Limit|
IMPORT |Image$$ZI$$Base|
IMPORT |Image$$ZI$$Limit|
COPY_VECT_TO_RAM
LDR R0,=|Image$$RO$$Base|
LDR R1,=SYSINIT
LDR R2,=0x200000 ; RAM START
CMP R0,R1
LDRLO R3,[R0],#4
STRLO R3,[R2],#4
BLO %B0
这段程序将SYSINIT之前的代码,也就是异常处理函数,全部复制到RAM中, 这就意味着不能将RW设置为0x200000,这样会使得向量被冲掉.
四,在RAM中运行
如果有必要,且代码足够小,可以将代码置于RAM中运行,由于RAM中本身没有代码,就需要将代码复制到RAM中:
COPY_BEGIN
LDR R0,=0x200000
LDR R1,=RESET ; =|Image$$RO$$Base|
CMP R1,R0 ;
BLO COPY_END ;
ADR R0,RESET
ADR R2,COPY_END
SUB R0,R2,R0
ADD R1,R1,R0
LDR R3,=|Image$$RO$$Limit|
3
CMP R1,R3
LDRLO R4,[R2],#4
STRLO R4,[R1],#4
BLO %B3
LDR PC,=COPY_END
COPY_END
程序首先取得RESET的连接地址,判断程序是否时是在RAM中运行,方法是与RAM起始地址比较,如果小于,那么就跳过代码复制.
在复制代码的时候需要注意,在这段程序结束之前的代码没有必要复制,因为这些代码都已经执行过了,所以,先取得COPY_END,作为复制起始地址,然后计算其相对RESET的偏移,然后以RO的值加上这个偏移,就是复制目的地的起始地址,然后开始复制.
五,开始主程序
以上步骤完成,就可以跳转到main运行
IMPORT Main
LDR PC,=Main
B .
六,器件初始化
主程序首先要进行器件的初始化,对S64而言,应该先初始化WDT,因为默认情况下,WDT是打开的,然后是各设备的时钟分配,最后应该remap.
以上是必要的启动步骤,实际应用中可以根据具体情况添加一些代码.
PART3
ARM启动程序分析
字体: 小中大 | 打印发布: 2007-7-17 11:05 作者: 网络转载来源: 网络查看: 4次
ARM启动程序分析
TIMER1用来触发和IRQ中断代码在FLASH中运行
这个例子有一下几个文件:
1. 中断向量表(ivt.s
2. 汇编启动代码 (init.s
3. C主函数文件
中断向量表
;代码必须链接在地址0X0。这里提供了ARM内核的中断向量
; ––––––––––––––––––––––––––––––––––––––––––––––––––––––––– ; 汇编指令
; ––––––––––––––––––––––––––––––––––––––––––––––––––––––––– AREA IVT, CODE ; 新代码段
CODE32 ; ARM指令
IMPORT start ; 在这段中没有定义
Entry ; 定义程序进入点
; ––––––––––––––––––––––––––––––––––––––––––––––––––––––––– LDR PC,
=start
LDR PC, Undefined_Addr
LDR PC,_Addr
LDR PC, Prefetch_Addr
LDR PC, Abort_Addr
; 在地址0x14用户应该插入一个标记(校验和.
; 此标记让引导程序决定是否在Flash中有有效的用户代码,目前大多数FLASH 编程工具(debuggers and ISP utility
; 有这个内置的功能,因此用户不用考虑这个问题.假如开发工具没有提供这种
功能,那么这个值必须自己计算,并把计算得到的值写到地址0x14. 计算校验和的细
节可以参考LPC2104/5/6用户指南的FLASH编程章节.
DCD ... .
LDR PC, [PC, #–0xFF0]
LDR PC, FIQ_Addr
Undefined_Addr DCD Undefined_HandlerPrefetch_Addr DCD Prefetch_Handler Abort_Addr DCD Abort_Handler
FIQ_Addr DCD FIQ_Handler
; ––––––––––––––––––––––––––––––––––––––––––––––––––––––––– ; 异常处理
; ––––––––––––––––––––––––––––––––––––––––––––––––––––––––Undefined_Handler
B Undefined_HandlerB_Handler
Prefetch_Handler
B Prefetch_Handler
Abort_Handler
B Abort_Handler
FIQ_Handler
B FIQ_Handler
END
系统复位后,第一条要执行的指令是:
LDR PC,=start
这条指令将跳转到汇编的起始代码(startup,他将中断使能,为IRQ 和管理模式设置堆栈
;比较重要的中断向量是IRQ中断.
LDR PC, [PC, #–0xFF0]
;这条指令将从中断向量控制控制器(VIC中的地址寄存器 (0xFFFF F030 导入PC,并执行.
;所有剩余的向量都有中断句柄
.
启动程序汇编代码
; ––––––––––––––––––––––––––––––––––––––––––––––––––––––––– ; 汇编语言
; ––––––––––––––––––––––––––––––––––––––––––––––––––––––––– AREA asm_code, CODE ; 新代码段
CODE32 ; ARM指令
IMPORT __main ; main在这个程序段中没有定义
EXPORT start ; 全局变量
; 参考ivt.s文件
; –––––––––––––––––––––––––––––––––––––––––––––––––––––––––
start
; 中断使能
MSR cpsr_c,#0x13 ; 在管理模式时设定堆栈指针SP
LDR SP,=0x4 .. ; 设置IRQ模式.设置SP_irq,然后返回管理模式
MRS R0, CPSR
BIC R1, R0,#0x1F
ORR R1, R1,#0x12
MSR cpsr_c, R1 ;进入中断模式
LDR SP, =0x4 ..
MSR cpsr_c, R0 ;返回管理模式
; 跳转到c程序
LDR lr, =__main
MOV pc, lr
END
这段代码在与ivt.s链接. 假如设置堆栈指针失败将导致数据中止异常,因此堆栈的初始化应该在跳转到c main(函数之前。
C 代码
Timer1 的寄存器这么配置,当匹配的时候将中断内核,并且复位。Timer1运行在60 MHz. 这段代码在一个评估板上运行,其外部始终使用10 MHz晶体,并且内部PLL 设定好。时钟操作的具体细节请参考LPC2106/5/4 中的相关章节。中断服务程序为空,用户可以根据需
要添加。在IRQHandler( 函数中可以点亮一些LED’s 或触发一些引脚. _irq 关键字编译器用来定义 IRQHandler (函数作为IRQ 中断服务程序。
C main函数在init.s 文件后执行。
LDR lr, =__main
MOV pc, lr
C代码如下:
/********************************************************** 函数定义
**********************************************************/ __irq void IRQHandler(void;
void feed(void;
void Initialize(void;
/********************************************************** 头文件
**********************************************************/
#include“LPC210x.h”
/********************************************************** MAIN **********************************************************/ int main( {
/* 系统初始化 */
Initialize(;
/* 启动时钟 */
T1_TCR=0x1;
while(1
{
}
}
/********************************************************** 初始化
**********************************************************/ void Initialize(
{
/*
* 初始化 PLL 10MHz×6=60MHz
*/
/* 设置乘除数值*/
PLLCFG=0x25;
feed(;
/* 使能 PLL */
PLLCON=0x1;
feed(;
/* 等待PLL锁定设定的频率 */
while(!(PLLSTAT & PLOCK{}
/* 连接PLL作为时钟源 */
PLLCON=0x3;
feed(;
/*
* 使能存储器加速模块MAM,设定访问FLASH的时钟*/ MAMCR=0x2;
MAMTIM=0x4;
/*
* 设定外围模块工作时钟(pclk与系统频率的值(cclk
*/
VPBDIV=0x1;
/* 初始化 GPIO */
IODIR=0xFFFF;
IOSET=0xFFFF;
/* 初始化T1 */
T1_TCR=0x0;
T1_TC=0x0;
T1_PR=0x0;
T1_PC=0x0;
/* 用户设定匹配值 */
T1_MR0=0x ..;
/* 匹配时复位和中断 */
T1_MCR=0x3;
/* 初始化VIC */
VICINTSEL=0x0; /* Timer 1 设为 IRQ */
VICINTEN= 0x20; /* Timer 1 中断使能 */
VICCNTL0= 0x25;
/* 中断服务程序地址 */
VICV ADDR0=(unsigned longIRQHandler;
}
/********************************************************** Timer 1 中断服务程序
**********************************************************/ __irq void IRQHandler(
{
/*中断服务程序将在这里执行,中断清除必须在Timer1中,写地址必须在 VIC 向量地址寄存器中.
* 在这里用户能点亮LED或者触发端口
*/
T1_IR=0x1;
VICV ADDR=0xff;
} /********************************************************** 为 PLL 提供时序 **********************************************************/ void feed( { PLLFEED=0xAA; PLLFEED=0x55; } 以上代码在 SRAM 中运行需要额外修改的地方汇编起在配置中断向量时 T 应该将中断向量表(ivt.s连接在 SRAM
(0x40000000,确定相关的中断向量存在 0x40000000–0x4000003F. 其他的文件可以链接在 SRAM。中断向量必须重新映射到 SRAM.。使用 MEMAP 寄存器和配置用户 RAM 模式可以做到这点。 int main( { ... /* 初始化*/ MEMAP=0x2; ...
}