最簡單bootloader的實現(xiàn)與分析
學習嵌入式,我是從bootloader入手的。前些日子寫了一個bootloader,趁今天有時間發(fā)出來,以記錄自己實現(xiàn)的過程,鞏固所學到的知識,并且希望給需要幫助的人帶來一些靈感,如果有不對的地方,還望大家能給予指正。
操作系統(tǒng):Ubuntu 11.04 開發(fā)板:友善之臂mini2440 (如果用其它s3c2440或s3c2410 cpu的也差不多,大同小異) 串口調試終端:minicom 編譯器:GNU工具鏈
先修知識:arm匯編,c語言,GNU匯編的一些特殊偽指令,makefile,鏈接腳本等知識。對于我的這個bootloader,這些知識除C語言外,其它的能看得懂,會一些基本的東西就足夠了。
學習一門知識最好的方法莫過于實踐,只有通過自己的親身體會,才能對知識有更加深刻的理解,才能更好的運用?,F(xiàn)在的bootloader已經(jīng)很強大了,比如最出名的u-boot,支持多種cpu架構和不同的開發(fā)板。我們聽到最多,學得最多的也是bootloader的移植,但是為什么要這樣移植,就不見得所有人都知道了。我之所以要親手實現(xiàn)這樣的一個很簡單的bootloader也就是為了能夠更好的掌握bootloader的原理。先說下我的bootloader所實現(xiàn)的功能:目前只是最基本的功能,支持串口調試、支持命令的交互,但是具體的命令由于對掌握bootloader原理沒太大幫助就沒有實現(xiàn),對于linux內核的引導過程相對復雜許多,在這個bootloader中也不作實現(xiàn)。我的想法是越簡單越好,不想做得太復雜。
代碼組織結構模仿了u-boot,如下:
- bootloader
- board 存放與開發(fā)板相關的目錄
- s3c2440 存放s3c2440 cpu 的一些與寄存器相關的定義文件
- cpu 存放不同cpu架構的目錄
- arm920t 存放依賴于arm920t的相關文件
- drivers 存放一些驅動文件
- include 存放一些用到的頭文件
程序源代碼:http://download.csdn.net/detail/tianfangk/3621598
有關開發(fā)環(huán)境的配置等一些知識網(wǎng)上一大堆,這里就不再贅述,直接從bootloader執(zhí)行過程的角度開始分析。首先,要對一個程序進行分析,必然要先看它的入口函數(shù)。對于如何找到入口函數(shù),就要看程序的鏈接腳本了。每一個鏈接過程都由鏈接腳本(linker script, 一般以lds作為文件的后綴名)控制。 鏈接腳本主要用于規(guī)定如何把輸入文件內的section放入輸出文件內,并控制輸出文件內各部分在程序地址空間內的布局。如果在程序的鏈接過程中沒有指定鏈接腳本,則會使用連接器的默認內置連接腳本。我用得是自己的鏈接腳本link.lds文件,從中可以看出程序的入口函數(shù)在cpu/arm920t/init.S文件中,所以先從這個文件開始分析。
_start:
/* Interrupt Vector Table */
b start @ 0x00
ldr pc, undefined @ 0x04
ldr pc, software_interrupt @ 0x08
ldr pc, prefetch_abort @ 0x0C
ldr pc, data_abort @ 0x10
ldr pc, not_used @ 0x14
ldr pc, irq @ 0x18
ldr pc, fiq @ 0x1C
這一段是中斷向量表,arm規(guī)定從0x00地址開始到0x1C為中斷向量表,當程序被中斷后,就會自動跳到這個地方,執(zhí)行相應的中斷處理程序。s3c2440 cpu上電后要執(zhí)行的第一條指令在0x00000000處,所以將執(zhí)行第一條指令:b start
接著pc就跳到start處:
start:
bl svc32_mod
bl off_wtdog
bl off_int
bl init_clk
bl init_cpu
bl init_sdram
bl init_gpb
#ifdef CONFIG_DEBUG
bl set_uart
#endif
bl copy_code
bl jmp_ram
這里作一些硬件的初始化工作,設置cpu的工作模式為svc32、關閉看門狗、屏蔽所有中斷、初始化時鐘、關閉mmu、初始化內存控制寄存器、初始化與led燈相關的gpio、如果要用到打印調試的話,還要初始化串口。說了這么多感覺好像有點讓人眼花繚亂,其實原則只有一點,那就是你要用到什么硬件,就把它初始化到你想要的狀態(tài)。只要把握住這一點就會覺得做這一切都十分合理,十分清晰。具體怎么初始化,就要參看cpu的芯片手冊了,上面說得很詳細。下面對部分需要說明的地方進行說明:
接下來要做的事情就是將flash上的代碼拷貝到內存中運行了。關于這一部分,有必要說明一點,這也是s3c2440這塊芯片的特別之處。通常我們是將程序燒寫在nor flash 上,因為nor flash有獨立的地址線和數(shù)據(jù)線,可以直接尋址,所以程序可以直接在nor flash上運行。但是nand flash 不同, 它沒有獨立的地址線,因此不能直接尋址。所以s3c2440為了支持nand flash啟動,在內部設有一塊4K大小的SRAM,在S3C2440上電后,Nand Flash控制器會自動的把Nand Flash上的前4K數(shù)據(jù)搬移到內部SRAM中,并把這塊SRAM映射到0x0地址處。由于我的這個bootloader總大小在4K之內,所以全部代碼都可以直接被加載到內部SRAM中,為了簡化過程,在copy_code過程中,我沒有再進行對nand flash的操作,直接從SRAM也就是0x0地址處將代碼copy到內存中,之后就執(zhí)行jmp_ram這一過程,跳轉到內存中運行。
但是這里有一個問題,也是我至今仍在困惑的問題,希望明白的朋友給解釋一下。當代碼從SRAM拷貝到內存完成的那一刻,存在了兩處完全一樣的代碼,一處在SRAM中,一處在內存中。當cpu繼續(xù)執(zhí)行的時候,它是如何知道自己要從內存中去取那一條指令而不是SRAM?另外,代碼復制到內存中,必然經(jīng)過了一個重定向的工作,那么這一工作又是在何時完成的呢?
這一問題,先不管,接下來就要跳轉到main函數(shù)中去了,這是一個C語言函數(shù),必然會用到堆棧,所以在這之前要將堆棧指針設置好。C語言函數(shù)的可讀性要強很多,就不用多說了。在main函數(shù)中,主要進行了對串口的初始化工作,最終程序將跳入到一個死循環(huán)wait_command中,反復重復一個動作:等待用戶輸入命令,然后執(zhí)行命令。
這里還要說一下GPIO的問題,最初我在對串口進行初始化的時候,沒有對相應的GPIO進行初始化(對于s3c2440,連接物理串口0的是GPH0~GPH7),導致無法將信息送到物理串口上,糾結了很長時間才查出錯誤。
總結一下:
1. 用到什么,就初始化什么;
2. 每寫一行代碼,都要保證其運行情況在你的掌控之下,千萬不要寫模棱兩可的代碼。
3. cpu的工作方式很簡單:取指令,執(zhí)行指令。不要讓它猜你的意圖。
做到這些,基本上可以保證程序不會出現(xiàn)大的錯誤。
到這里,對于這個bootloader的分析就完了,至于如何加載并啟動內核,待以后有時間再續(xù)……