深入理解linux啓動過程

inux系統的啓動過程是由很多步驟組成的,但是,無論你是啓 動一個標準的x86桌面計算機,還是一個嵌入式PowerPC的目標板,大多數的流程是驚人得相似的。這篇文章,探索了linux從最初的啓動準備到用戶 空間中某個程序被開啓之間的啓動過程,跟隨這個流程,你還能學到其他許多與啓動有關的知識,例如,boot loaders,內核解壓縮,初始化內存盤,以及其他一些linux啓動的部分。
在很早的時候,啓動一個計算機意味着去喂那些包含啓動程序的紙帶,或者通過手工使用前面板那密密麻麻的地址/數據/控制開關來加載啓動程序,但是,簡化這個流程並非必須的。
讓我們首先從一個較高的高度上來對linux的啓動做一個全面的分析,然後,我們會回顧一下,在每一個獨立的過程中,發生了什麼事情。在這途中的參考源代碼將幫助你在內核樹中不至於迷失方向,並能繼續深入挖掘下去。

一、 概述 圖1將是對啓動過程的整體回顧:

Figure 1. The 20,000-foot view of the Linux boot process
High-level view of the Linux kernel boot  
當系統一開始被啓動,或者重新啓動時,處理器將在衆所周知的位置執行代碼。在個人電腦上,這個位置處於基本輸入輸出系統中,也就是我們所說的BIOS,它 是被存儲在主板上的閃存中的。在嵌入式系統中,中央處理器(CPU)則將復位區域激活,來開始執行flash或者ROM中的已知的程序。在其它情況中,基 本的過程是一樣的。由於個人電腦提供了衆多靈活的可選設備,BIOS必須確定,由哪一個設備來執行啓動過程。在接下來的文章中我們將涉及這方面的更多的內 容。
當找到啓動設備時,第一階段所用的boot loader被裝載到RAM中並被執行。這裏的boot loader在大小上小於一個扇區的大小,也就是512字節,而它的任務,就是加載第二階段的boot loader。
當負責第二階段的boot loader位於內存中並被執行時,通常會顯示一個一閃而過的屏幕,然後linux以及可選的初始化內存盤(一種臨時的根文件系統,如果想得到具體的介紹,請訪問
http://likunarmstrong.bokee.com/5502266.html ) 會被裝載到存儲器中。當系統鏡像被加載時,第二階段的boot loader將把控制權轉交給內核鏡像,與此同時,內核開始自解壓並初始化。在這個階段,第二階段的boot loader會檢查系統的硬件,枚舉那些附加的硬件設備,掛載根設備,之後加載需要的內核模塊。完成之後,第一個用戶空間程序(init)開始執行,更高 層次的系統初始化開始。
這就是從表面上看,linux的啓動過程。好了,現在,讓我們更進一步,更深入地探索linux啓動過程中的一些細節。

二、 系統的啓動
系統啓動的階段,依賴於linux在哪個硬件設備上啓動。在嵌入式系統中,當系統被打開或者重新啓動的時候,就要使用啓動加載的環境。這方面的例子包括 U-BOOT,RedBoot,和Lucent推出的MicroMonitor。嵌入式平臺通常是綁定了啓動監視器的。這些程序位於目標硬件上flash 存儲器的特定位置,提供了將linux內核鏡像下載到flash存儲器的方法,並在接下來的過程中執行它。除了擁有存儲和啓動linux鏡像的功能外,這 些啓動監視器還能進行一定程度上的系統檢測和硬件初始化。在一個嵌入式的目標板中,這些啓動監視器通常覆蓋了第一階段與第二階段boot loader的功能。

/************************************************************************************************/
小知識:如何查看你的MBR內容。如果你希望查看你MBR的具體內容,請用以下命令:
# dd if=/dev/hda of=mbr.bin bs=512 count=1
# od -xa mbr.bin
需要以root身份運行的dd命令,讀取你的第一個集成電子驅動器或者IDE驅動器的前512字節,並將他們寫入
mbr.bim文件。od命令則是以十六進制和ASCII碼形式打印出這個二進制文件
/************************************************************************************************/

在個人電腦中,linux的啓動是從0xFFFF0地址開始的。BIOS的第一步動作就是進行上電自檢(POST)。POST的工作是檢查硬件設備。BIOS的第二步動作就是枚舉本地設備並初始化。
由於BIOS功能使用上的不同,它由兩個部分組成:POST碼runtime服務。POST完成後,它將從存儲器中被清除,但是BIOS runtime服務會被保留,用於目標操作系統。
爲了啓動操作系統,BIOS的runtime服務將搜索那些激活狀態的或是可引導啓動的設備,搜索的順序則由CMOS設置決定(也就是我們平時所謂的在 BIOS中設置的啓動順序)。一個軟驅,一臺光驅,一個硬盤上的分區,網絡上的設備甚至一個usb 閃存盤都可以作爲一個啓動設備。
當然,linux通常是從硬盤啓動的。硬盤上的MBR(主啓動記錄)包含有基本的boot loader,它是一個512字節大小的扇區,位於磁盤的第一個扇區(0磁頭0磁道1扇區)。當MBR被裝載到RAM中後,BIOS就會將控制權轉交給MBR。

三、 第一階段boot loader

Figure 2. Anatomy of the MBR
Anatomy of the MBR  
位於MBR中的主boot loader是一個512字節的鏡像,其中不僅包含了程序代碼,還包含了一個小的分區表,如圖2所示。最初的446字節是主boot loader,它裏面就包含有可執行代碼以及錯誤消息文本。接下來的64字節是分區表,其中包含有四個分區的各自的記錄(一個分區佔16字節)。MBR通 過特殊數字0xAA55(譯者注:在電子界中AA55確實是具有傳奇色彩的數字,想知道爲什麼麼?將它展開成二進制形式,看看有什麼規律)作爲兩個字節的 結束標誌。0x55AA同時也是MBR有效的校驗確認。

主boot loader的工作是尋找並加載第二boot loader。它通過分析分區表,找出激活分區來完成這個任務,當它找到一個激活分區時,它將繼續掃描剩下的分區表中的分區,以便確認他們都是未激活的。 確認完畢後,激活分區的啓動記錄從設備中被讀到RAM,並被執行。

四、 第二階段boot loader
起着次作用,或者說是第二boot loader,可以更加形象得被稱爲內核加載程序。這個階段的任務就是加載linux內核,以及可選的初始化內存盤。

/*******************************************************************/
小知識:GRUB階段的boot loaders
在/boot/grub目錄中包含有stage1,stage2和stage1.5的boot loaders,同時還有不少可選的loaders(例如,CD-ROM使用的就是iso9660_stage_1_5)
/*******************************************************************/

把第一階段和第二階段的boot loaders聯合起來,就是在x86個人電腦中,我們所說的linux loader(LILO)或者GRand Unified Bootloader(GRUB)。由於GRUB修正了一些LILO中存在的缺陷,因此下面就讓我們來看看GRUB(如果你希望得到更多的關於 GRUB,LILO和與之相關話題的討論資源,請見文後的參考資料)
對於GRUB來說,一個比較好的方面就是它包含了linux文件系統的知識。與LILO使用裸扇區不同的是,GRUB能夠從ext2或者ext3文件系統 中加載linux內核。它是通過將本來兩階段的boot loader轉換成三個階段的boot loader。在第一階段(MBR)中會啓動stage1.5的boot loader來理解linux內核鏡像中的特殊的文件系統格式,例如,reiserfs_stage1-5(用於從reiserf日誌文件系統中進行加 載)或e2fs+stage1_5(用於從wxt2或ext3文件系統進行加載)。當stage1.5的boot loader被加載並運行時,stage2 的boot loader才能被加載。
當stage2被加載時,GRUB能根據請求的情況顯示一個可選內核的清單(在 /etc/grub.conf 中進行定義,同時還有幾個軟符號鏈接 /etc/grub/menu.lst 和 /etc/grub.conf)。你可以選擇一個內核,修改其附加的內核參數。同時,你可以選擇使用命令行的shell來對啓動過程進行更深層次的手工控 制。
在第二階段boot loader存在與內存中後,就可以對文件系統進行查詢了,同時,默認的內核鏡像以及初始化內存盤鏡像也被加載到內存中。一切準備完畢之後,第二階段的boot loader就會調用內核鏡像。

五、 內核
(譯者注:在翻譯本章的時候,譯者發現IBM網站上已有譯好的文章,因此從本章開始以官方網站上的內容爲主)。當內核映像被加載到內存中,並且階段 2 的引導加載程序釋放控制權之後,內核階段就開始了。內核映像並不是一個可執行的內核,而是一個壓縮過的內核映像。通常它是一個 zImage(壓縮映像,小於 512KB)或一個 bzImage(較大的壓縮映像,大於 512KB),它是提前使用 zlib 進行壓縮過的。在這個內核映像前面是一個例程,它實現少量硬件設置,並對內核映像中包含的內核進行解壓,然後將其放入高端內存中,如果有初始 RAM 磁盤映像,就會將它移動到內存中,並標明以後使用。然後該例程會調用內核,並開始啓動內核引導的過程。

當 bzImage(用於 i386 映像)被調用時,我們從 ./arch/i386/boot/head.S 的 start 彙編例程開始執行(主要流程圖請參看圖 3)。這個例程會執行一些基本的硬件設置,並調用 ./arch/i386/boot/compressed/head.S 中的 startup_32 例程。此例程會設置一個基本的環境(堆棧等),並清除 Block Started by Symbol(BSS)。然後調用一個叫做 decompress_kernel 的 C 函數(在 ./arch/i386/boot/compressed/misc.c 中)來解壓內核。當內核被解壓到內存中之後,就可以調用它了。這是另外一個 startup_32 函數,但是這個函數在 ./arch/i386/kernel/head.S 中。

在這個新的 startup_32 函數(也稱爲清除程序或進程 0)中,會對頁表進行初始化,並啓用內存分頁功能。然後會爲任何可選的浮點單元(FPU)檢測 CPU 的類型,並將其存儲起來供以後使用。然後調用 start_kernel 函數(在 init/main.c 中),它會將您帶入與體系結構無關的 Linux 內核部分。實際上,這就是 Linux 內核的 main 函數。
/*******************************************************************/
小知識:GRUB 中的手工引導
在 GRUB 命令行中,我們可以使用 initrd 映像引導一個特定的內核,方法如下:

grub> kernel /bzImage-2.6.14.2
[Linux-bzImage, setup=0x1400, size=0x29672e]

grub> initrd /initrd-2.6.14.2.img
[Linux-initrd @ 0x5f13000, 0xcc199 bytes]

grub> boot

Uncompressing Linux... Ok, booting the kernel.

如果您不知道要引導的內核的名稱,只需使用斜線(/)然後按下 Tab 鍵即可。GRUB 會顯示內核和 initrd 映像列表。
/*******************************************************************/

Figure 3. Major functions flow for the Linux kernel i386 boot
Major Functions in Linux Kernel i386 Boot Process  
通過調用 start_kernel,會調用一系列初始化函數來設置中斷,執行進一步的內存配置,並加載初始 RAM 磁盤。最後,要調用 kernel_thread(在 arch/i386/kernel/process.c 中)來啓動 init 函數,這是第一個用戶空間進程(user-space process)。最後,啓動空任務,現在調度器就可以接管控制權了(在調用 cpu_idle 之後)。通過啓用中斷,搶佔式的調度器就可以週期性地接管控制權,從而提供多任務處理能力。
在內核引導過程中,初始 RAM 磁盤(initrd)是由階段 2 引導加載程序加載到內存中的,它會被複制到 RAM 中並掛載到系統上。這個 initrd 會作爲 RAM 中的臨時根文件系統使用,並允許內核在沒有掛載任何物理磁盤的情況下完整地實現引導。由於與外圍設備進行交互所需要的模塊可能是 initrd 的一部分,因此內核可以非常小,但是仍然需要支持大量可能的硬件配置。在內核引導之後,就可以正式裝備根文件系統了(通過 pivot_root):此時會將 initrd 根文件系統卸載掉,並掛載真正的根文件系統。
initrd 函數讓我們可以創建一個小型的 Linux 內核,其中包括作爲可加載模塊編譯的驅動程序。這些可加載的模塊爲內核提供了訪問磁盤和磁盤上的文件系統的方法,併爲其他硬件提供了驅動程序。由於根文件 系統是磁盤上的一個文件系統,因此 initrd 函數會提供一種啓動方法來獲得對磁盤的訪問,並掛載真正的根文件系統。在一個沒有硬盤的嵌入式環境中,initrd 可以是最終的根文件系統,或者也可以通過網絡文件系統(NFS)來掛載最終的根文件系統。

六、 init的介紹與結束語
當內核被引導並進行初始化之後,內核就可以啓動自己的第一個用戶空間應用程序了。這是第一個調用的使用標準 C 庫編譯的程序。在此之前,還沒有執行任何標準的 C 應用程序。
在桌面 Linux 系統上,第一個啓動的程序通常是 /sbin/init。但是這不是一定的。很少有嵌入式系統會需要使用 init 所提供的豐富初始化功能(這是通過 /etc/inittab 進行配置的)。在很多情況下,我們可以調用一個簡單的 shell 腳本來啓動必需的嵌入式應用程序。
與Linux 本身非常類似,Linux 的引導過程也非常靈活,可以支持衆多的處理器和硬件平臺。最初,加載引導加載程序提供了一種簡單的方法,不用任何花架子就可以引導 Linux。LILO 引導加載程序對引導能力進行了擴充,但是它卻缺少文件系統的感知能力。最新一代的引導加載程序,例如 GRUB,允許 Linux 從一些文件系統(從 Minix 到 Reise)上進行引導。

發表評論
所有評論
還沒有人評論,想成為第一個評論的人麼? 請在上方評論欄輸入並且點擊發布.
相關文章