一、Bootloader 的引入
1.1 Bootloader 的引入
Linux 內核的啟動是需要一定的必要條件的,但在 CPU 剛上電啟動時,一般連內存控制 器都沒有配置過,根本無法在內存中運行程序,更不可能處在 Linux 內核的啟動環境中。為 了初始化 CPU 及其他外設,使得 Linux 內核可以在系統主存中跑起來,并讓系統符合 Linux 內核啟動的必備條件,必須要有一個先于內核運行的程序,他就是所謂的引導加載程序: Bootloader。
Bootloader并不是只有Linux才需要,是幾乎所有的運行操作系統的設備都必須具備的。 PC電腦的BIOS就是bootloader的一部分,對于Linux PC來說:Bootloader = BIOS + GRUB/LILO。
1.2 嵌入式 Linux 系統軟件結構與分布
一般情況下嵌入式 Linux 系統中軟件主要由以下幾個部分組成:
1.引導加載程序:其中包括內部ROM中的固化啟動代碼和bootloader兩部分。固化ROM 是廠家在芯片生產時固化,用于引導 bootloader。
2.Linux Kernel 和Drivers。
3.文件系統:包括根文件系統和建立于Flash 內存設備之上的文件系統(ext4、UBI、 CRAMFS 等)。它是提供管理系統的各種配置文件以及系統執行用于應用程序的良好運行環 境的載體。
4.應用程序:用于自定義的應用程序,存放于文件系統之中。 在Flash 存儲器中,上面四個部分的分布如圖 1 所示:
圖1:嵌入式Linux的軟件分布
但是以上只是大部分情況下的分布,也有一些根文件系統可能是 initramfs,被一起壓縮到了內核映像里,或者沒有 bootloader 參數區等。
二、Bootloader 介紹
2.1 Bootloader 的功能
Bootloader是在操作系統內核運行之前運行的一段小程序,通過它我們可以初始化硬件設備,從而將系統的軟硬件環境帶到一個合適的狀態,以便為最終調用操作系統內核準備好 正確的環境,最后從別處(Flash、以太網、UART 等)載入內核映像并跳到入口地址運行。
簡單的說,Bootloader 就是這么一小段程序,它在系統上電時開始執行,初始化硬件設備、準備好軟件環境,最后調用操作系統內核。
可以增強 Bootloader 的功能,比如增加網絡功能、從 PC 上通過串口或網絡下載文件、 燒寫文件、將 Flash 上壓縮的文件解壓后再運行等,這就是一個功能更為強大的 Bootloader, 也稱為 Monitor。實際上,在最終產品中用戶并不需要這些功能,他們只是為了方便開發。
2.2 Bootloader 的特點
由于 Bootloader 直接操作硬件,因此它嚴重依賴于硬件,且依據所引導的操作系統的不同而不同。在嵌入式世界中建立一個通用的 Bootloader 幾乎是不可能的,而有可能的是 讓一個 Bootloader 代碼支持多種不同的架構和操作系統,并讓他方便移植。Bootloader的啟動過程通常是多階段的,這樣既能夠提供復雜的功能,又具有更好的可移植性。
大多數 Bootloader 都包含兩種不同的操作模式:本地加載模式和遠程下載模式。遠程下載模式只對開發人員才有意義。
Bootloader 都映射在 CPU復位后運行的第一條指令的地址處,以保證系統上電或復位 后首先執行 Bootloader。
2.3 Bootloader 的分類
首先區分一下“Bootloader”和“Monitor”的概念:嚴格的講,“Bootloader”只是引 導設備并執行主程序的固件;而“Monitor”還提供更多的命令行接口,可進行調試、讀寫內存、燒寫 Flash、配置環境變量等。“Monitor”在嵌入式系統開發過程中可以提供更好的調試功能,開發完成后,就完全設置成一個“Bootloader”了。所以習慣上將他們統稱為 Bootloader。表 2.1 為常見的開放源碼的 Linux 引導程序。
表2.1 常見的開放源碼 Linux 引導程序
其中 U-Boot 支持大多 CPU,可以燒寫 EXT2、JFFS2 文件系統映像,支持串口下載、網絡 下載,并提供了大量的命令。
2.4 Bootloader 的啟動模式
Bootloader 的主要功能是引導操作系統,但在開發時,通常需要使用各種命令操作 Bootloader,一般通過串口來連接 PC 和開發板,可以在串口上輸入各種命令、觀察運行結果等。這也只是對開發人員才有意義,用戶使用產品時是不用接串口來控制 Bootloader 的。 從這個角度可以將 Bootloader 分為啟動加載(Boot loading)模式和下載(Down loading)模 式。
2.4.1 啟動加載(Bootloading)模式
這種模式也稱為"自主"(Autonomous)模式。也即 Boot Loader 從目標機上的某個固態存儲設備上將操作系統加載到 RAM 中運行,整個過程并沒有用戶的介入。這種模式是 Boot Loader 的正常工作模式,因此在嵌入式產品發布的時侯,Bootloader顯然必須工作在這種模 式下。
2.4.2 下載(Downloading)模式
在這種模式下,目標機上的 Boot Loader 將通過串口連接或網絡連接等通信手段從主機(Host)下載文件,比如:下載內核映像和根文件系統映像等。從主機下載的文件通常首先 被 Boot Loader 保存到目標機的 RAM 中,然后再被 Boot Loader 寫到目標機上的 FLASH 類固 態存儲設備中。Boot Loader 的這種模式通常在第一次安裝內核與根文件系統時被使用;此 外,以后的系統更新也會使用Boot Loader的這種工作模式。工作于這種模式下的Boot Loader 通常都會向它的終端用戶提供一個簡單的命令行接口。
像 Blob 或 U-Boot 等這樣功能強大的 Boot Loader 通常同時支持這兩種工作模式,而且 允許用戶在這兩種工作模式之間進行切換。比如,Blob 在啟動時處于正常的啟動加載模式, 但是它會延時 10 秒等待終端用戶按下任意鍵而將 blob 切換到下載模式。如果在 10 秒內沒 有用戶按鍵,則 blob 繼續啟動 Linux 內核。
2.4.3 下載模式之網絡啟動方式
這種方式開發板不需要配置較大的存儲介質(跟無盤工作站有點類似),但是使用這種 啟動方式之前,需要把 Bootloader 安裝到板上的 EPPROM 或者 Flash 中。Bootloader 通過以 太網接口遠程下載 Linux 內核映像或者文件系統到 RAM 中運行。這種方式對于嵌入式系統開發來說非常重要。
使用這種方式的前提條件,就是目標板有串口、以太網接口、USB 接口或者其他鏈接方式:串口一般作為控制臺,同時可以用來下載內核映像和 RAMDISK 文件系統;用網絡接口 來掛載 NFS 文件系統;也可以使用 USB 接口虛擬成以太網口來通訊。
使用網絡啟動方式,還需要在服務器上配置啟動相關網絡服務:使用 TFTP 服務為 Bootloader 客戶端提供文件下載功能;DHCP 服務動態為 Bootloader 設置 IP 地址,配置網絡 參數。網絡啟動方式如圖 2.1 所示。
圖2.1 網絡方式啟動系統
2.5 Bootloader 的啟動流程
Bootloader 的啟動過程可以分為單階段(Single Stage)、多階段(Multi-Stage)兩種。通 常多階段的 Bootloader 能提供更為復雜的功能以及更好的可移植性。從固態存儲設備上啟動 的Bootloader 大多都是兩個階段的啟動過程。第一階段使用匯編來實現,它完成一些依賴于 CPU 體系結構的初始化,并調用第二階段的代碼;第二階段則通常使用 C 語言來實現,這樣 可以實現更復雜的功能,而且代碼會有更好的可讀性和可移植性。
一般而言,這兩個階段完成的功能可以如下分類:
2.5.1 Bootloader 第一階段的功能
Bootloader 在第一階段主要完成以下功能:
硬件設備初始化;
為加載 Bootloader 的第二階段代碼準備 RAM 空間;
復制 Bootloader 的第二階段代碼到 RAM 空間中;
設置好棧;
跳轉到第二階段代碼的 C 入口點;
在第一階段進行的硬件初始化一般包括:關閉 WATCHDOG、關中斷、設置 CPU 的速度 和時鐘頻率、RAM 初始化等。這些并不都是必須的,比如 S3C2410/S3C2440 的開發板所使 用的 U-Boot 中,就將 CPU 的速度和時鐘頻率放在第二階段進行設置。
甚至,將第二階段的代碼復制到 RAM 空間也不是必須的,對于 Nor Flash 等支持 XIP 的 存儲設備,完全可以在上面直接執行代碼,只不過相比在 RAM 中執行效率大為降低。
2.5.2 Bootloader 第二階段的功能
Bootloader 在第二階段主要完成以下功能:
初始化本階段要使用到的硬件設備;
將內核映像和根文件系統映像從 Flash 上讀到 RAM 空間中;
為內核設置啟動參數;
調用內核;
為了方便開發,只要要初始化一個串口以便程序員與 Bootloader 進行交互。
所謂檢測內存映射,就是確定板上使用了多少內存、他們的地址空間是什么。由于嵌入 式開發中的 Bootloader 多是針對某類板子進行編寫,所以可以根據板子的情況直接設置,不 需要考慮可以適用于各類情況的復雜算法。
Flash 上的內核映像有可能是經過壓縮的,在讀到 RAM 之后,還需要進行解壓。當然, 對于有自解壓功能的內核,不需要Bootloader 來解壓。
將根文件系統映像復制到 RAM 中并不是必須的,這取決于是什么類型的根文件系統,以及內核訪問它的方法。
將內核放在適當的位置后,在跳入執行內核之前,需要根據內核啟動的需求,配置相應 的啟動參數。如 Linux 內核的啟動要求如表 2.2 所示。
表2.2 Linux 內核啟動條件
在 C 語言中,可以像下列示例代碼一樣來調用內核:
void (*theKernel)(int zero, int arch, u32params_addr) = (void (*)(int, int, u32))KERNEL_RAM_BASE;
theKernel(0,ARCH_NUMBER, (u32)kernel_params_start);
2.6 Bootloader 與內核的交互
U-Boot 與內核之間的交互是單向的,U-Boot 將各類參數傳遞給內核。由于他們(U-Boot 與內核)不能同時運行,傳遞的辦法只有一個:U-Boot 將參數放在放在某個約定的地方之 后,再啟動內核,內核啟動后從這個地方獲得參數。
除了約定好參數存放的地址外,還要規定參數的結構。Linux 2.4.x 以后的內核都期望以標記列表(tagged list)的形式來傳遞啟動參數。標記,就是一種數據結構;標記列表,就 是挨著存放的多個標記。標記列表以標記 ATAG_CORE 開始,以標記 ATAG_NONE 結束。
標記的數據結構為tag,它由一個tag_header結構和一個聯合體(union)組成。tag_header 結構標示標記的類型及長度,比如是表示內存還是表示命令行參數等。對于不同類型的標記使用不同的聯合體(union),比如表示內存時使用tag_mem32,表示命令時使用tag_cmdline。
數據結構 tag 和tag_header 定義在 Linux 內核源碼的include/asm/setup.h 頭文件中(在 U-Boot 的 include/asm-arm/目錄下的 setup.h 中也有定義),如下所示:
struct tag_header {
u32 size;
u32 tag;
};
struct tag {
struct tag_header hdr;
union {
struct tag_core core;
struct tag_mem32 mem;
struct tag_videotext videotext;
struct tag_ramdisk ramdisk;
struct tag_initrd initrd;
struct tag_serialnr serialnr;
struct tag_revision revision;
struct tag_videolfb videolfb;
struct tag_cmdline cmdline;
/*
*/
struct tag_acorn acorn;
/*
* DC21285 specific
*/
struct tag_memclk memclk;
} u;
};
下面以設置內存標記、命令標記為例說明參數的傳遞。
2.6.1 設置標記ATAG_CORE
標記列表以標記 ATAG_CORE 開始,其結構體定義如下:
/* The list must start with an ATAG_COREnode */
#define ATAG_CORE 0x54410001
struct tag_core {
u32 flags; /* bit 0 =read-only */
u32 pagesize;
u32 rootdev;
};
假設 Bootloader與內核約定的參數存放地址為 0x30000100,則可以以如下代碼設置標 記 ATAG_CORE:
#define tag_next(t) ((struct tag *)((u32*)(t) + (t)->hdr.size))
static struct tag *params; //定義全局變量 params,為 struct tag 結構
static void setup_start_tag (bd_t *bd)
{
params = (struct tag *) bd->bi_boot_params; //bi_boot_params =0x30000100
params->hdr.tag = ATAG_CORE;// ATAG_CORE 在前面定義,指定標記類型
params->hdr.size = tag_size (tag_core); //計算標記大小
// 設置 ATAG_CORE 標記的內容
params->u.core.flags = 0;
params->u.core.pagesize = 0;
params->u.core.rootdev = 0;
params = tag_next (params);
}
其中,tag_next定義為指向當前標記的末尾。
2.6.2 設置內存標記
內存標記 tag_mem32 的定義如下:
struct tag_mem32 {
u32 size; // 內存的大小
u32 start; /* physical start address */
};
假設開發板使用的內存起始地址為 0x30000000,大小為 0x4000000,則內存標記可以如 下設置:
static void setup_memory_tags (bd_t *bd)
{
params->hdr.tag = ATAG_MEM;
params->hdr.size = tag_size (tag_mem32);
params->u.mem.start = 0x30000000;
params->u.mem.size = 0x4000000;
params = tag_next (params);
}
2.6.3 設置命令行標記
命令行就是一個字符串,它被用來控制內核的一些行為。比如“root=/dev/mtdlock2 init=/linuxrc console=ttySAC0”表示根文件系統在 MTD2 分區上,系統啟動后執行的第一個程序為/linuxrc,控制臺為 ttySAC0。
命令行可以在 Bootloader 中通過命令設置好,然后按如下構造標記傳給內核。
char *p = ”root=/dev/mtdlock 2init=/linuxrc console=ttySAC0”;
params->hdr.tag = ATAG_CMDLINE;
params->hdr.size = (sizeof (structtag_header) + strlen (p) + 1 + 4) >> 2;
strcpy (params->u.cmdline.cmdline,p);
params = tag_next (params);
2.6.4 設置標記ATAG_NONE
標記列表以標記 ATAG_NONE 結束,如下設置:
static void setup_end_tag (bd_t *bd)
{
params->hdr.tag = ATAG_NONE;
params->hdr.size = 0;
}
-
嵌入式
+關注
關注
5090文章
19173瀏覽量
306844 -
Linux
+關注
關注
87文章
11339瀏覽量
210119 -
bootloader
+關注
關注
2文章
235瀏覽量
45683
原文標題:詳談嵌入式之Bootloader
文章出處:【微信號:gh_c472c2199c88,微信公眾號:嵌入式微處理器】歡迎添加關注!文章轉載請注明出處。
發布評論請先 登錄
相關推薦
評論