新闻中心

EEPW首页 > 嵌入式系统 > 设计应用 > S3C2440-启动分析

S3C2440-启动分析

作者: 时间:2016-12-02 来源:网络 收藏
IsrIRQ
sub sp,sp,#4 ;首先在栈中留出一个字的位置
stmfd sp!,{r8-r9} ;由于要用到r8,r9所以先压栈,保存
ldr r9,=INTOFFSET ;把INTOFFSET寄存器的地址放入r9中
ldr r9,[r9] ;把r9的内容放入r9寄存器中
ldr r8,=HandleEINT0 ;得到中断向量表的基地址
add r8,r8,r9,lsl #2 ;用基址加变址的方式得到中断向量表的地址
ldr r8,[r8] ;把中断处理程序的入口地址放入r8
str r8,[sp,#8] ;将r8压入栈中
ldmfd sp!,{r8-r9,pc} ; 使r8,r9和入口地址出栈,并跳转到中断服务程序中
LTORG
/*******************************************************************
这个是程序的入口,主要进行了关闭看门狗,屏蔽所有中断,设置时钟,设置内存控制,检查启动方式,如果是Nandflash,进行代码的搬运。如果是Norflash,将代码复制到RO指定的位置。初始化各个模式下的堆栈。
*******************************************************************/
;=======
; ENTRY
;=======
ResetHandler
ldr r0,=WTCON
ldr r1,=0x0 ;WTCON的第0位是复位信号输出允许位
str r1,[r0]
ldr r0,=INTMSK
ldr r1,=0xffffffff ;屏蔽所有的中断
str r1,[r0]
ldr r0,=INTSUBMSK ;屏蔽所有的子中断
ldr r1,=0x7fff
str r1,[r0]
;led显示
[ {FALSE}
; rGPFDAT = (rGPFDAT & "(0xf<<4)) | (("data & 0xf)<<4);
; Led_Display
ldr r0,=GPFCON
ldr r1,=0x5500
str r1,[r0]
ldr r0,=GPFDAT
ldr r1,=0x10
str r1,[r0]
]
/**********************************************************************************
设置系统时钟,主要设置PLL锁定时间和UPLL(USB时钟)和MPLL的主频。S3C2440A的时钟控制逻辑有两个PLL,一个是UPLL,一个是MPLL。MPLL可以产生FCLK(for CPU),HCLK(for AHB bus),PCLK(for APB bus)。
**********************************************************************************/
;To reduce PLL lock time, adjust the LOCKTIME register.
ldr r0,=LOCKTIME
ldr r1,=0xffffff ;复位后 LOCKTIME 是0xffffffff
str r1,[r0]
[ PLL_ON_START
; Added for confirm clock divide. for 2440.
; Setting value Fclk:Hclkclk
ldr r0,=CLKDIVN
ldr r1,=CLKDIV_VAL ; 0=1:1:1, 1=1:1:2, 2=1:2:2, 3=1:2:4, 4=1:4:4, 5=1:4:8, 6=1:3:3, 7=1:3:6. ;设置分频值
str r1,[r0]
[ CLKDIV_VAL>1 ; means Fclk:Hclk is not 1:1.
mrc p15,0,r0,c1,c0,0
orr r0,r0,#0xc0000000;R1_nF:OR:R1_iA
mcr p15,0,r0,c1,c0,0
|
mrc p15,0,r0,c1,c0,0
bic r0,r0,#0xc0000000;R1_iA:OR:R1_nF
mcr p15,0,r0,c1,c0,0
]
;Configure UPLL
ldr r0,=UPLLCON
ldr r1,=((U_MDIV<<12)+(U_PDIV<<4)+U_SDIV)
str r1,[r0]
nop ; Caution: After UPLL setting, at least 7-clocks delay must be inserted for setting hardware be completed.
nop
nop
nop
nop
nop
nop
;Configure MPLL
ldr r0,=MPLLCON
ldr r1,=((M_MDIV<<12)+(M_PDIV<<4)+M_SDIV) ;Fin=16.9344MHz
str r1,[r0]
]
;Check if the boot is caused by the wake-up from SLEEP mode.
ldr r1,=GSTATUS2
ldr r0,[r1]
tst r0,#0x2
;In case of the wake-up from SLEEP mode, go to SLEEP_WAKEUP handler.
bne WAKEUP_SLEEP
; EXPORT StartPointAfterSleepWakeUp
;StartPointAfterSleepWakeUp
;Set memory control registers
;ldr r0,=SMRDATA
adrl r0, SMRDATA
ldr r1,=BWSCON ;BWSCON Address
add r2, r0, #52 ;End address of SMRDATA
0
ldr r3, [r0], #4
str r3, [r1], #4
cmp r2, r0
bne %B0
;delay
mov r0, #&1000
1
subs r0, r0, #1
bne %B1
;===
;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;
;;;;;;;;;;;;; When EINT0 is pressed, Clear SDRAM
;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;
; check if EIN0 button is pressed
;检查EIN0按钮是否被按下
ldr r0,=GPFCON
ldr r1,=0x0
str r1,[r0] ;GPFCON=0,F口为输入
ldr r0,=GPFUP
ldr r1,=0xff
str r1,[r0] ;GPFUP=0xff,上拉功能无效
ldr r1,=GPFDAT
ldr r0,[r1] ;读取F口数据
bic r0,r0,#(0x1e<<1) ;仅保留第1位数据,其他清0
tst r0,#0x1 ;判断第1位
bne %F1 ;不为0表示按钮没有被按下,则向前跳转,不执行清空SDRAM
; Clear SDRAM Start
;清空SDRAM
ldr r0,=GPFCON
ldr r1,=0x55aa
str r1,[r0] ;GPF7"4为输出,GPF3"0为中断
; ldr r0,=GPFUP
; ldr r1,=0xff
; str r1,[r0] ;上拉功能无效
ldr r0,=GPFDAT
ldr r1,=0x0
str r1,[r0] ;GPFDAT = 0
mov r1,#0
mov r2,#0
mov r3,#0
mov r4,#0
mov r5,#0
mov r6,#0
mov r7,#0
mov r8,#0
ldr r9,=0x4000000 ;64MB RAM
ldr r0,=0x30000000 ;RAM首地址
;清空64MB的RAM
0
stmia r0!,{r1-r8}
subs r9,r9,#32
bne %B0
;Clear SDRAM End
1
;Initialize stacks
bl InitStacks
; Setup IRQ handler//建立中断表
ldr r0,=HandleIRQ ;This routine is needed
ldr r1,=IsrIRQ ;if there isn t subs pc,lr,#4 at 0x18, 0x1c
str r1,[r0]
;===========================================================
;// 判断是从nor启动还是从nand启动
;===========================================================
;下面的代码为把ROM中的数据复制到RAM中
ldr r0, =BWSCON
ldr r0, [r0]
ands r0, r0, #6 ;读取OM[1:0]引脚状态
;为0表示从NAND Flash启动,不为0则从NOR Flash启动
bne copy_proc_beg ;跳转,不用读取NAND Flash
adr r0, ResetEntry ;OM[1:0] == 0,从NAND Flash启动
cmp r0, #0 ;if use Multi-ice,
bne copy_proc_beg ;do not read nand flash for boot
;nop
/*******************************************************************
这里是2440进行代码搬运。如果是从Nandflash中启动,首先要把启动代码的前4k拷贝到称为“SteppingStone”的4kSRAM中,然后在这4k代码中将剩下的代码从Nandflash拷贝到SDRAM中。
*******************************************************************/
nand_boot_beg
[ {TRUE}
bl RdNF2SDRAM ;复制NAND Flash到SDRAM
]
ldr pc, =copy_proc_beg
/*******************************************************************
比较ResetEntry和BaseOfROM,如果相等,在内存运行ICE,无需赋值code区中的RO段,但要复制code中的RW段。下面这个针对代码在NorFlash时的拷贝方法。功能为把从ResetEntry起,TopOfROM-BaseOfROM大小的数据拷贝到BaseOfROM。TopOfROM和BaseOfROM为|Image$$RO$$Limit|和|Image$$RO$$Base|,|Image$$RO$$Limit|和|Image$$RO$$Base|由连接器生成。为生成的代码的代码段运行时的起始和终止地址。开始时r0=RetEntry(source),r2=BaseOfROM(destination)。终止条件:复制了TopOfROM-BaseOfROM大小。
*******************************************************************/
copy_proc_beg
adr r0, ResetEntry
ldr r2, BaseOfROM
cmp r0, r2
ldreq r0, TopOfROM
beq InitRam
ldr r3, TopOfROM
0
ldmia r0!, {r4-r7}
stmia r2!, {r4-r7}
cmp r2, r3
bcc %B0
sub r2, r2, r3
sub r0, r0, r2
/*******************************************************************
比较BaseOfBSS和BaseOfZero,当代码在内存中运行时,r0(初始值)=TopOfROM,这之后的BaseOfZero-BaseOfBSS仍属于code,需要拷贝到BaseOfBSS。
*******************************************************************/*
InitRam
ldr r2, BaseOfBS
ldr r3, BaseOfZero
0
cmp r2, r3
ldrcc r1, [r0], #4
strcc r1, [r2], #4
bcc %B0
/*******************************************************************
下面的循环体是为未初始化的全局变量赋值为0
*******************************************************************/
mov r0, #0
ldr r3, EndOfBSS
1
cmp r2, r3
strcc r0, [r2], #4
bcc %B1
ldr pc, =%F2 ;goto compiler address
2
; [ CLKDIV_VAL>1 ;if FCLK:HCLK≠1:1
; bl MMU_SetAsyncBusMode ;设置时钟模式为异步模式
; |
; bl MMU_SetFastBusMode ;设置时钟模式为快速总线模式

评论


技术专区

关闭