Loader的功能和啟動過程?
作用原理
編輯
⒈ Boot Loader 所支持的 CPU 和嵌入式板
每種不同的 CPU 體系結構都有不同的Boot Loader。有些 Boot Loader 也支持多種體系結構的 CPU,比如 U-Boot 就同時支持 ARM 體系結構和MIPS 體系結構。除了依賴于 CPU的體系結構外,Boot Loader 實際上也依賴于具體的嵌入式板級設備的配置。這也就是說,對于兩塊不同的嵌入式板而言,即使它們是基于同一種 CPU 而構建的,要想讓運行在一塊板子上的 Boot Loader 程序也能運行在另一塊板子上,通常也都需要修改 Boot Loader 的源程序。
⒉ Boot Loader 的安裝媒介(Installation Medium)
系統加電或復位后,所有的CPU 通常都從某個由 CPU 制造商預先安排的地址上取指令。比如,基于 arm7TDMI core 的 CPU 在復位時通常都從地址 0x00000000 取它的第一條指令。而基于CPU 構建的嵌入式系統通常都有某種類型的固態存儲設備(比如:ROM、EEPROM 或 FLASH 等)被映射到這個預先安排的地址上。因此在系統加電后,CPU 將首先執行Boot Loader 程序。
⒊ 用來控制 Boot Loader 的設備或機制
主機和目標機之間一般通過串口建立連接,Boot Loader 軟件在執行時通常會通過串口來進行輸入、輸出,比如:輸出打印信息到串口,從串口讀取用戶控制字符等。
⒋ Boot Loader 的啟動過程
BootLoader 的啟動過程可分為單階段(Single-Stage)和多階段(Multi-Stage)兩種。通常多階段的Boot Loader 具有更復雜的功能,更好的可移植性。從固態存儲設備上啟動的Boot Loader 大多采用兩階段,即啟動過程可以分為 stage1和stage2:stage1完成初始化硬件,為stage2準備內存空間,并將stage2復制到內存中,設置堆棧,然后跳轉到stage2。
⒌ Boot Loader 的操作模式 (Operation Mode)
大多數Boot Loader都包含兩種不同的操作模式。啟動加載模式和下載模式。
(1)啟動加載(Boot loading)模式:這種模式也稱為“自主”模式,也即Boot Loader從目標機上的某個固態存儲設備上將操作系統加載到RAM中運行,整個過程并沒有用戶的介入。這種模式是Boot Loader的正常工作模式。
(2)下載(Down loading)模式:在這種模式下目標機上的Boot Loader將通過串口連接或網絡連接等通信手段從主機下載文件。從主機下載的文件通常首先被Boot Loader保存到目標機的RAM中然后再被Boot Loader寫到目標機上的固態存儲設備中。
⒍ Boot Loader 與主機之間進行文件傳輸所用的通信設備及協議
分為兩種情況。一種是目標機使用串口與主機相連。這時的傳輸協議通常是xmodem/ymodem/zmodem中的一種。第二種可以用網絡連接的方式傳輸文件,這時使用的協議多為tftp。
解析
編輯
網上關于Linux的BOOTLOADER文章不少了,但是大都是vivi,blob等比較龐大的程序,讀起來不太方便,編譯出的文件也比較大,而且更多的是面向開發用的引導代碼,做成產品時還要裁減,這一定程度影響了開發速度,對初學者學習開銷也比較大,在此分析一種簡單的BOOTLOADER,是在三星公司提供的2410 BOOTLOADER上稍微修改后的結果,編譯出來的文件大小不超過4k,希望對大家有所幫助.
重要概念
COMPRESSED KERNEL and DECOMPRESSED KERNEL
壓縮后的KERNEL,按照文檔資料,現在不提倡使用DECOMPRESSED KERNEL,而要使用COMPRESSED KERNEL,它包括了解壓器.因此要在ram分配時給壓縮和解壓的KERNEL提供足夠空間,這樣它們不會相互覆蓋.
當執行指令跳轉到COMPRESSED KERNEL后,解壓器就開始工作,如果解壓器探測到解壓的代碼會覆蓋掉COMPRESSED KERNEL,那它會直接跳到COMPRESSED KERNEL后存放數據,并且重新定位KERNEL,所以如果沒有足夠空間,就會出錯.
Jffs2 File System
可以使armlinux應用中產生的數據保存在FLASH上,我的板子還沒用到這個.
RAMDISK
使用RAMDISK可以使ROOT FILE SYSTEM在沒有其他設備的情況下啟動.一般有兩種加載方式,最常用的一種是,把COMPRESSED RAMDISK IMAGE放到指定地址,然后由BOOTLOADER把這個地址通過啟動參數的方式ATAG_INITRD2傳遞給KERNEL.具體看代碼分析.
啟動參數(摘自IBM developer)
在調用內核之前,應該作一步準備工作,即:設置 Linux 內核的啟動參數。Linux 2.4.x 以后的內核都期望以標記列表(tagged list)的形式來傳遞啟動參數。啟動參數標記列表以標記 ATAG_CORE 開始,以標記 ATAG_NONE 結束。每個標記由標識被傳遞參數的 tag_header 結構以及隨后的參數值數據結構來組成。數據結構 tag 和 tag_header 定義在 Linux 內核源碼的include/asm/setup.h 頭文件中.
在嵌入式 Linux 系統中,通常需要由 BOOTLOADER 設置的常見啟動參數有:ATAG_CORE、ATAG_MEM、ATAG_CMDLINE、ATAG_RAMDISK、ATAG_INITRD等。
(注)參數也可以用COMMANDLINE來設定,在我的BOOTLOADER里,我兩種都用了.
開發環境
CPU:S3C2410,BANK6上有64M的SDRAM(兩塊),BANK0上有32M NOR FLASH,串口當然是逃不掉的.這樣,按照數據手冊,地址分配如下:
0x4000_0000開始是4k的片內DRAM.
0x0000_0000開始是32M FLASH 16bit寬度
0x3000_0000開始是64M SDRAM 32bit寬度
注意:控制寄存器中的BANK6和BANK7部分必須相同.
0x4000_0000(片內DRAM)存放4k以內的BOOTLOADER IMAGE
0x3000_0100開始存放啟動參數
0x3120_0000 存放COMPRESSED KERNEL IMAGE
0x3200_0000 存放COMPRESSED RAMDISK
0x3000_8000 指定為DECOMPRESSED KERNEL IMAGE ADDRESS
0x3040_0000 指定為DECOMPRESSED RAMDISK IMAGE ADDRESS
開發環境:Redhat Linux,armgcc toolchain,armlinux KERNEL
如何建立armgcc的編譯環境:建議使用toolchain,而不要自己去編譯armgcc,偶試過好多次,都以失敗告終.
先下載arm-gcc 3.3.2 toolchain
將arm-linux-gcc-3.3.2.tar.bz2 解壓到 /toolchain
tar jxvf arm-linux-gcc-3.3.2.tar.bz2mv /usr/local/arm/3.3.2 /toolchain在makefile 中在把arch=arm CROSS_COMPILE設置成toolchain的路徑
還有就是INCLUDE = -I ../include -I /root/my/usr/local/arm/3.3.2/include.,否則庫函數就不能用了
啟動方式
可以放在FLASH里啟動,或者用Jtag仿真器.由于使用NOR FLASH,根據2410的手冊,片內的4K DRAM在不需要設置便可以直接使用,而其他存儲器必須先初始化,比如告訴memory controller,BANK6里有兩塊SDRAM,數據寬度是32bit,= =.否則memory control會按照復位后的默認值來處理存儲器.這樣讀寫就會產生錯誤.
所以第一步,通過仿真器把執行代碼放到0x4000_0000,(在編譯的時候,設定TEXT_BAS
E=0x40000000)
第二步,通過 AxD把linux KERNEL IMAGE放到目標地址(SDRAM)中,等待調用
第三步,執行BOOTLOADER代碼,從串口得到調試數據,引導armlinux
代碼分析
講了那么多執行的步驟,是想讓大家對啟動有個大概印象,接著就是BOOTLOADER內部的代碼分析了,BOOTLOADER文章內容網上很多,我這里精簡了下,刪除了不必要的功能.
BOOTLOADER一般分為2部分,匯編部分和c語言部分,匯編部分執行簡單的硬件初始化,C部分負責復制數據,設置啟動參數,串口通信等功能.
BOOTLOADER的生命周期:
⒈ 初始化硬件,比如設置UART(至少設置一個),檢測存儲器= =.
⒉ 設置啟動參數,這是為了告訴內核硬件的信息,比如用哪個啟動界面,波特率 = =.
⒊ 跳轉到Linux KERNEL的首地址.
⒋ 消亡
同時在linux中 GRUB(GRand Unified Bootloader)是一個系統默認自帶的多重啟動管理器。它可以在多個操作系統共存時選擇引導哪個系統。盡管引導操作系統看上去是件平凡且瑣碎的任務,但它實際上很重要。如果引導裝入器不能很好地完成工作或者不具有彈性,那么就可能鎖住系統或者無法引導計算機……