基于Nandflash的Bootloader的設計與實現(xiàn)
關鍵詞:Bootloader; 移植;Nandflash;S3C2410
0 引言
Bootloader通常稱為系統(tǒng)的引導加載程序,是系統(tǒng)加電或復位后執(zhí)行的第一段代碼[ 1 ]。一般它只在系統(tǒng)啟動時運行非常短的時間,但對于嵌入式系統(tǒng)來說,這是一個非常重要的系統(tǒng)組成部分。通過這段小程序,可以初始化硬件設備、建立內(nèi)存空間的映射圖,從而將系統(tǒng)的軟硬件環(huán)境帶到一個合適的狀態(tài),以便為調(diào)用操作系統(tǒng)內(nèi)核準備好正確的環(huán)境,并同時提供基本輸入、輸出系統(tǒng)監(jiān)控功能和程序調(diào)試功能。
Bootloader是嚴重地依賴于硬件而實現(xiàn)的。每種不同體系結構的處理器都有不同的Bootloader。除了依賴于處理器的體系結構以外,Bootloader實際上也依賴于具體的嵌入式板級設備的配置,也就是說,對于兩塊不同的嵌入式板而言,即使它們是基于同一種處理器而構建的,要想讓運行在一塊板子上的Bootloader程序也能運行在另一塊板子上,通常也都需要修改與目標硬件相關的代碼。因此有必要分析Bootloader,并理解和找出其中的原理和規(guī)律,就其特定的嵌入式系統(tǒng),移植或開發(fā)屬于自己的Bootloader。
1 系統(tǒng)硬件平臺簡介
本系統(tǒng)采用的是SamSung公司的S3C2410處理器[ 2 ],它是專門為移動手持設備提供的高性價比和高性能的嵌入式微處理器解決方案。其內(nèi)核是ARM920T,最高能工作在202.8MHz,為了減少系統(tǒng)總成本和減少外圍器件,它集成了如下部件:分別為16KB指令和數(shù)據(jù)Cahce、1個LCD控制器、SDRAM控制器、NANDFLASH控制器、3通道UART、4通道DMA、4個具有PWM功能的計時器和1個內(nèi)部時鐘、8通道10位ADC、觸摸屏接口、I²S總線接口,2個USB主機接口、1個USB設備接口,2個SPI接口、SD和MMC卡接口、看門狗定時器、117位通用IO口、24位外部中斷源、8通道10位AD控制器等。本文涉及的S3C2410開發(fā)板的硬件結構如圖1所示,本文主要闡述從Nandflash引導操作系統(tǒng)要完成的主要任務和實現(xiàn)方法,至于從Norflash引導操作系統(tǒng),不打算具體實現(xiàn)。
圖1 S3C2410硬件結構圖
2 存儲空間分布和映射圖
硬件平臺的Nandflash(型號是:K9F1208U0M[ 3])空間為64MB,SDRAM(型號是:HY57V561620[ 4 ],32Mx2)空間為64M(0x30000000-0x33ffffff),采用如圖2所示的存儲空間分布圖,因為Nandflash只能存儲程序,無法運行程序。為了能夠從Nandflash啟動,上電復位時,S3C2410通過硬件邏輯把Nandflash的前4KB的內(nèi)容復制到片內(nèi)SRAM中,而片內(nèi)SRAM被映射到地址0x0,這樣就可以從地址0x0處取到有效指令,開始執(zhí)行bootloader,完成把Nandflash中的內(nèi)核代碼復制到sdram中等工作。
圖2 引導代碼和操作系統(tǒng)內(nèi)核在Nandflash和存儲空間中的分布情況
3 Bootloader的設計流程
Bootloader引導程序是硬件上電復位后首先運行的代碼,由它來加載嵌入式操作系統(tǒng)。然后由操作系統(tǒng)接管整個系統(tǒng),進行進程管理、內(nèi)存管理、磁盤管理和各個外設管理等工作。 BootLoader是操作系統(tǒng)內(nèi)核運行之前的一段自舉程序,用來初始化硬件設備、改變處理器運行模式和重組中斷向量,建立內(nèi)存空間的映射圖,將系統(tǒng)的軟硬件環(huán)境帶到一個由用戶定制的特定狀態(tài),然后加載操作系統(tǒng)內(nèi)核。從操作系統(tǒng)的角度來看,Bootloader的總目標就是正確地調(diào)用內(nèi)核來執(zhí)行。Bootloader一般分為stage1和stage2兩大部分[ 5 ],對于依賴于CPU體系結構的代碼,比如設備初始化代碼等,通常都放在stage1中,而且通常都用匯編語言來實現(xiàn),以達到短小精悍的目的,也就是前面說的啟動代碼。而stage2則通常用C語言來實現(xiàn),這樣可以實現(xiàn)復雜的功能,而且代碼會具有更好的可讀性和可移植性。
3.1 Bootloader的stage1
這部分代碼必須首先完成一些基本的硬件初始化。為stage2 的執(zhí)行以及隨后的內(nèi)核的執(zhí)行準備好一些基本的硬件環(huán)境。Bootloader 的stage1 一般通用的內(nèi)容包括:
(1)設置中斷和異常向量;(2)禁止看門狗;(3) 屏蔽所有的中斷, 在Boot Loader 的執(zhí)行全過程中可以不必響應任何中斷, 中斷屏蔽可以通過寫CPU 的中斷屏蔽寄存器或狀態(tài)寄存器CPSR 寄存器來完成;(4) 設置CPU 的速度和時鐘頻率;(5) 對RAM進行初始化, 包括正確設置系統(tǒng)的內(nèi)存控制器的功能寄存器等;(6)初始化LED或UART,就是通過GPIO來驅動LED,也可以通過初始化UART向串口打印Bootloader的調(diào)試信息來表明系統(tǒng)的狀態(tài)是OK還是ERROR,以便跟蹤系統(tǒng)運行情況;(7)關閉CPU 內(nèi)部指令/數(shù)據(jù)高速緩存(cache);(8)為加載Bootloader的stage2準備RAM空間;(9)設置好堆棧;(10)跳轉到stage2的C入口點;其流程圖如圖3所示。
圖3 Bootloader的stage1的實現(xiàn)
3.2 Bootloader的stage2
為了讓程序跳入C 語言的“main”函數(shù), 我們采用直接跳轉到“main”函數(shù)的方法, 實現(xiàn)代碼如下:
b Main
進入main 函數(shù)后即可以開始本階段stage2 的初始化任務, 這包括:
(1) 如果在stage1沒有初始化UART,這時候至少初始化一個串口, 以便和終端用戶進行交互,當然也可以繼續(xù)點亮或熄滅LED來判斷程序執(zhí)行情況;
(2) 修改時鐘頻率;
(3) 使能指令Cache;
(5) 從串口中打印一些必要的交互信息,了解系統(tǒng)狀態(tài);
(6) 初始化中斷, 包括屏蔽中斷, 清除中斷懸掛標志, 初始化中斷向量表, 注冊需要的中斷處理函數(shù)等;
(8)打印版本、時間等信息,并從Nandflash復制內(nèi)核到SDRAM中;
(9)修改指針,直接跳到內(nèi)核在SDRAM中的首地址處,至此,完成了Bootloader的全部運行加載工作;
下面是main()函數(shù)和從Nandflash復制內(nèi)核到SDRAM中的ReadImageFromNandflash()函數(shù)的具體實現(xiàn),但省略了一些具體細節(jié),包括從串口打印的啟動、交互、調(diào)試信息和一些具體函數(shù)的實現(xiàn)。一些具體函數(shù)的實現(xiàn)可以參考三星評估版源代碼。
void Main(void)
{
JumpAddr=0x30200000; //拷貝內(nèi)核到sdram中的起始地址,也是內(nèi)核開始執(zhí)行的地址
ChangeClockDivider(1,1); //1:2:4
ChangeMPllValue(0x5c,0x1,0x1); // FCLK=202.8MHz
MMU_EnableICache(); //使能指令Cache
Uart_Init(); // 初始化串口
Port_Init(); //初始化I/O口
NF_Init(); //初始化Nandflash控制器
NF_ReadID(); //讀取Nandflash存儲器ID號
ReadImageFromNandflash();//把存儲在Nandflash中的內(nèi)核拷貝到SDRAM中
rINTMSK=BIT_ALLMSK; //屏蔽所有中斷
Launch(JumpAddr); //跳轉到sdram中內(nèi)核開始處,并運行內(nèi)核
}
從Nandflash(Flash是K9F1208U0M)拷貝內(nèi)核到SDRAM的函數(shù)具體實現(xiàn)如下:
void ReadImageFromNandflash(void)
{
U8 Image_Buf[512];
U32 Sram_Space=0;
U32 j,k, numberblock;
static U32 i, SECTOR_SIZE=512;
static U8 isbad;
volatile U32 IMAGE_BASE=0x30200000; //內(nèi)核在sdram中運行的開始地址
rINTMSK = BIT_ALLMSK; //屏蔽所有中斷
i=2; //從第2個block開始拷貝內(nèi)核,第0個用于存儲本文的bootloader,第1個沒用到
numberblock=2047; //拷貝多少個block到sdram中,視內(nèi)核大小設置此值
while(1)
{
nextblock:
isbad=0;
isbad=NF_IsBadBlock(i); //判斷正在拷貝的block是否是壞block
if (isbad) //是壞block,就進行相應的處理;否則就忽略此處,進行下面的拷貝
{
i=i+1; //調(diào)整,指向下一個block
isbad=0;
if(i>= numberblock) //判斷是否拷貝完了所需的block
{
Launch(JumpAddr); //拷貝完了所需的block,就跳到sdram中內(nèi)核開始處
}
goto nextblock;
}
for(k=0;k<32;k++) //1 block=32 pages
{ // FMD_ReadSector()函數(shù)實現(xiàn)從Nandflash存儲器中讀取數(shù)據(jù)到數(shù)據(jù)緩沖區(qū)中
FMD_ReadSector(i, (U8 *)&Image_Buf, k);
for (j=0;j<SECTOR_SIZE;j++) //1 page=512 bytes
{ //從數(shù)據(jù)緩沖區(qū)中拷貝到sdram中
*((U8 *)(IMAGE_BASE+Sram_Space+j))=Image_Buf[j];
}
Sram_Space=Sram_Space+SECTOR_SIZE; //調(diào)整sdram中的偏移地址
}
i=i+1; //調(diào)整,指向下一個block
if(i>= numberblock) //判斷是否拷貝完了所需的block
{
Launch(JumpAddr); //拷貝完了所需的block,就跳到sdram中內(nèi)核開始處
}
}
}
4 試驗結果
由于三星公司的S3C2410集成了Nandflash控制器,它通過硬件邏輯把Nandflash的前4KB內(nèi)容,即把Bootloader復制到片內(nèi)sram中,并被映射到地址0x0處。通過跳線設置默認從nandflash啟動,那么,系統(tǒng)每次上電或復位后,首先開始運行的就是Bootloader。使用ADS1.2集成開發(fā)環(huán)境建立Bootloader應用工程,添加必需的文件并設置好編譯環(huán)境,如Bootloader的RO_Base設置為0x0,RW_Base設置為0x33ff0000等,調(diào)試并最后生成可執(zhí)行二進制文件,通過JTAG接口把Bootloader燒寫到Nandflash的第0個block地址開始處,通過usb下載工具把操作系統(tǒng)燒寫到第2個block地址開始處,復位啟動系統(tǒng)運行后的結果如圖4所示,該程序用于基于uCOS操作系統(tǒng)的圖像采集系統(tǒng)的引導。用同樣的方法燒寫不同的操作系統(tǒng)內(nèi)核應用程序,試驗結果每一次表明:Bootloader運行良好,啟動加載內(nèi)核快,且簡單、實用、可靠。
圖4 Bootloader引導運行的系統(tǒng)
5 結論
Bootloader的設計與實現(xiàn)是一個非常復雜的過程,因此要根據(jù)具體的硬件和軟件需求分析來進行移植或設計。本文設計的Bootloader完成的主要功能包括:試驗板硬件的初始化、串口初始化、時鐘頻率修改以及從Nandflash復制操作系統(tǒng)到SDRAM中運行等,并通過PC機上的超級終端顯示了正確的啟動運行信息,且可執(zhí)行代碼只有3K左右。因此,本文所詳細描述的Bootloader啟動運行的全過程,對理解、設計和移植Bootloader具有一定的參考意義。
參考文獻:
[1] 徐宇清,黃彥平等.S3C44B0X的Bootloader技術分析[J]. 上海理工大學學報,2005,27(4):369-372.
[2]SAMSUNG公司. Samsung s3c2410a User Manual v1.0.pdf.
[3]SAMSUNG公司.K9F1208U0M-YCB0.pdf.
[4] http://www.icpdf.com/pdf/HY57V561620.htm. HY57V561620(L)T.pdf.
[5]張崳編著.32位嵌入式系統(tǒng)硬件設計與調(diào)試.北京:機械工業(yè)出版社,2005,7.
[作者簡介]
郝衛(wèi)東(1964-),男,漢族,河北定縣人,桂林電子科技大學高級工程師,碩士研究生導師,主要從事機器人技術和醫(yī)療電子方面的研究。
劉溯奇(1977-),男,漢族,廣西桂林人,廣西桂林電子科技大學機器人中心碩士研究生,主要從事機器人技術、嵌入式系統(tǒng)應用方面的研究。