摘要:讲述了基于 ARM处理器的嵌入式系统在上电启动后应用程序或操作系统运行前,对处理器及其内部功能模块进行初始化的过程,并结合经过实际验证的代码详细的分析了S3C44B0 Bootloader的运行过程。
关键字:ARM 嵌入式系统 Bootloader
一. 引言:
对于 PC 机,其开机后的初始化处理器配置、硬件初始化等操作是由 BIOS(Basic Input /Output System)完成的,但对于嵌入式系统来说,出于经济性、价格方面的考虑一般不配置 BIOS,因此我们必须自行编写完成这些工作的程序,这就是所需要的开机程序。而在嵌入式系统中,通常并没有像 BIOS 那样的固件程序,启动时用于完成初始化操作的这段代码被称为 Bootloader 程序,因此整个系统的加载启动任务就完全由 Bootloader 来完成。简单地说,通过这段程序,可以初始化硬件设备、建立内存空间的映射图(有的 CPU 没有内存映射功能如 S3C44B0),从而将系统的软硬件环境设定在一个合适的状态,以便为最终调用操作系统内核、运行用户应用程序准备好正确的环境。Bootloader依赖于实际的硬件和应用环境,因此要为嵌入式系统建立一个通用、标准的 Bootloader 是非常困难的。Bootloader也依赖于具体的嵌入式板级设备的配置,这也就是说,对于两块不同的嵌入式主板而言,即使它们是基于同一 CPU 而构建,要想让运行在一块板子上的 Bootloader 程序也能运行在另一块板子上,通常都需要修改 Bootloader 的源程序。
二. 启动流程
系统加电复位后,几乎所有的 CPU都从由复位地址上取指令。比如,基于 ARM7TDMI内核的 CPU 在复位时通常都从地址 0x00000000 处取它的第一条指令。而以微处理器为核心的嵌入式系统通常都有某种类型的固态存储设备(比如 EEPROM、FLASH等)被映射到这个预先设置好的地址上。因此在系统加电复位后,处理器将首先执行存放在复位地址处的程序。通过集成开发环境可以将 Bootloader 定位在复位地址开始的存储空间内,因此Bootloader是系统加电后、操作系统内核或用户应用程序运行之前,首先必须运行的一段程序代码。对于嵌入式系统来说,有的使用操作系统,也有的不使用操作系统,比如功能简单仅包括应用程序的系统,但在系统启动时都必须执行 Bootloader,为系统运行准备好软硬件运行环境。
系统的启动通常有两种方式,一种是可以直接从 Flash 启动,另一种是可以将压缩的内存映像文件从 Flash(为节省 Flash 资源、提高速度)中复制、解压到 RAM,再从 RAM 启动。当电源打开时,一般的系统会去执行 ROM(应用较多的是 Flash)里面的启动代码。这些代码是用汇编语言编写的,其主要作用在于初始化 CPU 和板上的必备硬件如内存、中断控制器等。有时候用户还必须根据自己板子的硬件资源情况做适当的调整与修改。
系统启动代码完成基本软硬件环境初始化后,对于有操作系统的情况下,启动操作系统、启动内存管理、任务调度、加载驱动程序等,最后执行应用程序或等待用户命令;对于没有操作系统的系统直接执行应用程序或等待用户命令。
启动代码是用来初始化电路以及用来为高级语言写的软件做好运行前准备的一小段汇编语言,在商业实时操作系统中,启动代码部分一般被称为板级支持包,英文缩写为 BSP。它的主要功能就是:电路初始化和为高级语言编写的软件运行做准备。系统启动流程如图 1所示,主要的过程如下:
1. 启动代码的第一步是设置中断和异常向量。
2. 完成系统启动所必须的最小配置,某些处理器芯片包含一个或几个全局寄存器,这些寄存器必须在系统启动的最初进行配置。
3. 设置看门狗,用户设计的部分外围电路如果必须在系统启动时初始化,就可以放在这一步。
4. 配置系统所使用的存储器,包括 Flash,SRAM 和DRAM 等,并为他们分配地址空间。如果系统使用了 DRAM 或其它外设,就需要设置相关的寄存器,以确定其刷新频率,数据总线宽度等信息,初始化存储器系统。有些芯片可通过寄存器编程初始化存储器系统,而对于较复杂系统通常集成有 MMU 来管理内存空间。
5. 为处理器的每个工作模式设置栈指针,ARM 处理器有多种工作模式,每种工作模式都需要设置单独的栈空间。
6. 变量初始化,这里的变量指的是在软件中定义的已经赋好初值的全局变量,启动过程中需要将这部分变量从只读区域,也就是 Flash拷贝到读写区域中,因为这部分变量的值在软件运行时有可能重新赋值。还有一种变量不需要处理,就是已经赋好初值的静态全局变量,这部分变量在软件运行过程中不会改变,因此可以直接固化在只读的 Flash或 EEPROM中。
7. 数据区准备,对于软件中所有未赋初值的全局变量,启动过程中需要将这部分变量所在区域全部清零。
8. 最后一步是调用高级语言入口函数,比如 main函数等。
三. 程序分析
下面根据实际经过测试的代码详细讲述系统的启动过程。
.text
ENTRY:
b ResetHandler
b HandlerUndef
b HandlerSWI
b HandlerPabort
b HandlerDabort
b .
b HandlerIRQ
b HandlerFIQ
… ...
… ...
ResetHandler:
Ldr r0,=WTCON
ldr r1,=0x0
str r1,[r0]
ldr r0,=INTMSK
ldr r1,=0x07ffffff
str r1,[r0]
ldr r0,=LOCKTIME
ldr r1,=0xfff
str r1,[r0]
.if PLLONSTART
ldr r0,=PLLCON
ldr r1,=((M_DIV<<12)+(P_DIV<<4)+S_DIV)
str r1,[r0]
.endif
ldr r0,=CLKCON
ldr r1,=0x7ff8
str r1,[r0]
ldr r0,=BDIDES0
ldr r1,=0x40000000
str r1,[r0]
ldr r0,=BDIDES1
ldr r1,=0x40000000
str r1,[r0]
ldr r0,=SMRDATA
ldmia r0,{r1-r13}
ldr r0,=0x01c80000
stmia r0,{r1-r13}
ldr sp, =SVCStack
bl InitStacks
ldr r0,=HandleIRQ
ldr r1,=IsrIRQ
str r1,[r0]
LDR r0, =Image_RO_Limit
LDR r1, =Image_RW_Base
LDR r3, =Image_ZI_Base
CMP r0, r1
BEQ F1
F0:
CMP r1, r3
LDRCC r2, [r0], #4
STRCC r2, [r1], #4
BCC F0
F1:
LDR r1, =Image_ZI_Base
MOV r2, #0
F2:
CMP r3, r1
STRCC r2, [r3], #4
BCC F2
MRS r0, CPSR
BIC r0, r0, #NOINT
MSR CPSR_cxsf, r0
BL Main
B.
四. 总结:
启动过程中的初始化程序就是初始化 CPU 内部各个关键的寄存器、配置外围硬件电路相关寄存器、建立中断向量表等,然后跳转到一般由高级语言编写的主函数的应用程序代码去执行,这样就可以利用高级语言来编写完成系统设计所要求的各种功能。初始化的过程对大多数初学者来说,比较难理解的是中断的处理和一些少见的操作符号,这些符号多是一些宏定义或系统用于在内存空间中对各个段的定位标识符号。掌握了 S3C44B0 的启动代码之后,对系统功能程序设计会起到很大的帮助,是进行下一步程序设计的基础。
五. 参考文献:
1. 田泽.嵌入式系统开发与应用.北京.北京航空航天大学出版社.2005
2. 田泽.嵌入式系统开发与应用实验教程.北京.北京航空航天大学出版社.2004
3. 深圳英蓓特信息技术有限公司.Embest ARM 实验教学系统用户手册.Version 2.01.2003
4. SAMSUNG 公司.S3C44B0_datasheet.pdf.