新聞中心

EEPW首頁 > 嵌入式系統(tǒng) > 設計應用 > Linux基礎命令之: Linux啟動過程詳解

Linux基礎命令之: Linux啟動過程詳解

作者: 時間:2013-09-13 來源:網(wǎng)絡 收藏

本文引用地址:http://butianyuan.cn/article/257157.htm

2.2啟動過程詳解

在了解了的常見命令之后,下面詳細講解的啟動過程。Linux的啟動過程包含了Linux工作原理的精髓,而且在嵌入式開發(fā)過程中非常需要這方面的知識。

2.2.1概述

用戶開機啟動Linux過程如下:

(1)當用戶打開PC(intelCPU)的電源時,CPU將自動進入實模式,并從地址0xFFFF0000開始自動執(zhí)行程序代碼,這個地址通常是ROM-BIOS中的地址。這時BIOS進行開機自檢,并按BIOS中設置的啟動設備(通常是硬盤)進行啟動,接著啟動設備上安裝的引導程序lilo或grub開始引導Linux(也就是啟動設備的第一個扇區(qū)),這時,Linux才獲得了啟動權。

(2)第二階段,Linux首先進行內(nèi)核的引導,主要完成磁盤引導、讀取機器系統(tǒng)數(shù)據(jù)、實模式和保護模式的切換、加載數(shù)據(jù)段寄存器以及重置中斷描述符表等。

(3)第三階段執(zhí)行init程序(也就是系統(tǒng)初始化工作),init程序調(diào)用了rc.sysinit和rc等程序,而rc.sysinit和rc在完成系統(tǒng)初始化和運行服務的任務后,返回init。

(4)第四階段,init啟動mingetty,打開終端供用戶登錄系統(tǒng),用戶登錄成功后進入了shell,這樣就完成了從開機到登錄的整個啟動過程。

Linux啟動總體流程如圖2.2所示,其中的4個階段分別由同步棒隔開。第一階段不涉及Linux自身的啟動過程,下面分別對第二和第三階段進行詳細講解。

圖2.2Linux啟動總體流程圖

2.2.2階段

在grub或lilo等引導程序成功完成引導Linux系統(tǒng)的任務后,Linux就從它們手中接管了CPU的控制權。用戶可以從www.kernel.org上下載最新版本的源碼進行閱讀,其目錄為:linux-2.6.*.*/arch/i386/boot。在啟動過程中主要用到該目錄下的幾個文件:.S、setup.S以及compressed子目錄下的head.S等。

Linux的內(nèi)核通常是壓縮過的,包括上述提到的那幾個重要的匯編程序,它們都是在壓縮內(nèi)核vmlinuz中的。Linux中提供的內(nèi)核包含了眾多驅動和功能,容量較大,壓縮內(nèi)核可以節(jié)省大量的空間,壓縮的內(nèi)核在啟動時可以對自身進行解包。

(1)階段

當grub讀入vmlinuz后,會根據(jù)(512字節(jié))把它自身和setup程序段讀到以不大于0x90000開始的的內(nèi)存里(注意:在以往的引導協(xié)議里是放在0x90000,但現(xiàn)在有所變化),然后grub會跳過bootsect那512字節(jié)的程序段,直接運行setup里的第一跳指令。就是說bzImage里bootsect的程序沒有再被執(zhí)行了,而bootsect.S在完成了指令搬移以后就退出了。之后執(zhí)行權就轉到了setup.S的程序中。

(2)setup階段。

setup.S的主要功能是利用ROMBIOS中斷讀取機器系統(tǒng)數(shù)據(jù),并將系統(tǒng)參數(shù)(包括內(nèi)存、磁盤等)保存到以0x90000~0x901FF開始的內(nèi)存中。

此外,setup.S還將video.S中的代碼包含進來,檢測和設置顯示器和顯示模式。

最后,它還會設置CPU的控制寄存器CR0(也稱機器狀態(tài)字),從而進入32位保護模式運行,并跳轉到絕對地址為0x100000(虛擬地址0xC0000000+0x100000)的位置。當CPU跳到0x100000時,將執(zhí)行“arch/i386/kernel/head.S”中的startup_32。

(3)head.S階段。

當運行到head.S時,系統(tǒng)已經(jīng)運行在保護模式,而head.S完成的一個重要任務就是將內(nèi)核解壓。內(nèi)核是通過壓縮的方式放在內(nèi)存中的,head.S通過調(diào)用misc.c中定義的decompress_kernel()函數(shù),將內(nèi)核vmlinuz解壓到0x100000。

接下來head.S程序完成寄存器、分頁表的初始化工作,但要注意的是,這個head.S程序與完成解壓縮工作的head.S程序是不同的,它在源代碼中的位置是arch/i386/kernel/head.S。

在完成了初始化之后,head.S就跳轉到start_kernel()函數(shù)中去了。

(4)main.c階段。

start_kernel()是“init/main.c”中定義的函數(shù),start_kernel()調(diào)用了一系列初始化函數(shù),進行內(nèi)核的初始化工作。要注意的是,在初始化之前系統(tǒng)中斷仍然是被屏蔽的,另外內(nèi)核也處于被鎖定狀態(tài),以保證只有一個CPU用于Linux系統(tǒng)的啟動。

在start_kernel()的最后,調(diào)用了init()函數(shù),也就是下面要講述的。

2.2.3

在加載了內(nèi)核之后,由內(nèi)核執(zhí)行引導的第一個進程是init進程,該進程號始終是“1”。init進程根據(jù)其配置文件“/etc/inittab”主要完成系統(tǒng)的一系列初始化的任務。由于該配置文件是init進程執(zhí)行的惟一依據(jù),因此先對它的格式進行統(tǒng)一講解。

inittab文件中除了注釋行外,每一行都有如下格式:

id:runlevels:action:process

(1)id。

id是配置記錄標識符,由1~4個字符組成,對于getty或mingetty等其他login程序項,要求id與tty的編號相同,否則getty程序將不能正常工作。

(2)runlevels。

runlevels是運行級別記錄符,一般使用0~6以及S和s。其中,0、1、6運行級別為系統(tǒng)保留:0作為shutdown動作,1作為重啟至單用戶模式,6為重啟;S和s意義相同,表示單用戶模式,且無需inittab文件,因此也不在inittab中出現(xiàn)。7~9級別也是可以使用的,傳統(tǒng)的UNIX系統(tǒng)沒有定義這幾個級別。

runlevel可以是并列的多個值,對大多數(shù)action來說,僅當runlevel與當前運行級別匹配成功才會執(zhí)行。

(3)action。

action字段用于描述系統(tǒng)執(zhí)行的特定操作,它的常見設置有:initdefault、sysinit、boot、bootwait、respawn等。

initdefault用于標識系統(tǒng)缺省的啟動級別。當init由內(nèi)核激活以后,它將讀取inittab中的initdefault項,取得其中的runlevel,并作為當前的運行級別。如果沒有inittab文件,或者其中沒有initdefault項,init將在控制臺上請求輸入runlevel。

sysinit、boot、bootwait等action將在系統(tǒng)啟動時無條件運行,忽略其中的runlevel。

respawn字段表示該類進程在結束后會重新啟動運行。

(4)process。

process字段設置啟動進程所執(zhí)行的命令。

以下結合筆者系統(tǒng)中的inittab配置文件詳細講解該配置文件完成的功能。

linux操作系統(tǒng)文章專題:linux操作系統(tǒng)詳解(linux不再難懂)

linux相關文章:linux教程



上一頁 1 2 下一頁

評論


相關推薦

技術專區(qū)

關閉