init程序需要讀取配置文件/etc/inittab,。inittab是一個不可執(zhí)行的文本文件,,它有若干行指令所組成。
理解Runlevel:
runlevel用來表示在init進程結束之后的系統狀態(tài),,在系統的硬件中沒有固定的信息來表示runlevel,,它純粹是一種軟件結構。init和inittab是runlevel影響系統狀態(tài)的唯一原因,。在上述例子中inittab文件起始階段的注釋主
用來描述runlevel:
Runlevel 0 是讓init關閉所有進程并終止系統,。
Runlevel 1 是用來將系統轉到單用戶模式,單用戶模式只能有系統管理員進入,,在該模式下處理那些在有登錄用戶的情況下不能進行更改的文件,,改runlevel的編號1也可以用S代替。
Runlevel 2 是允許系統進入多用戶的模式,,但并不支持文件共享,,這種模式很少應用。
Runlevel 3 是最常用的運行模式,,主要用來提供真正的多用戶模式,,也是多數服務器的缺省模式。
Runlevel 4 一般不被系統使用,,用戶可以設計自己的系統狀態(tài)并將其應用到runlevel 4階段,,盡管很少使用,但使用該系統可以實現一些特定的登錄請求,。
Runlevel 5 是將系統初始化為專用的X Window終端,。對功能強大的Linux系統來說,這并不是好的選擇,,但用戶如果需要這樣,,也可以通過在runlevel啟動來實現該方案。
Runlevel 6 是關閉所有運行的進程并重新啟動系統。
inittab文件內容:
###表示當前缺省運行級別為5(initdefault);
id:5:initdefault: /*啟動后進入圖形界面,設為3則進入字符終端界面*/
###啟動時自動執(zhí)行/etc/rc.d/rc.sysinit腳本(sysinit)
/*
在inittab文件中以#開頭的所有行都是注釋行,。注釋行有助于用戶理解inittab文件,,inittab文件中的值都是如下格式:
label:runlevel:action:process
label是1~4個字符的標簽,用來標示輸入的值,。一些系統只支持2個字符的標簽,。鑒于此原因,多數人都將標簽字符的個數限制在2個以內,。該標簽可以是任意字符構成的字符串,,但實際上,某些特定的標簽是常用的,,在Red Hat Linux中使用的標簽是:
代碼:
id 用來定義缺省的init運行的級別
si 是系統初始化的進程
ln 其中的n從1~6,指明該進程可以使用的runlevel的級別
ud 是升級進程
ca 指明當按下Ctrl+Alt+Del時運行的進程
pf 指當UPS表明斷電時運行的進程
pr 是在系統真正關閉之前,,UPS發(fā)出電源恢復的信號時需要運行的進程
x 是將系統轉入X終端時需要運行的進程
runlevel字段指定runlevel的級別??梢灾付ǘ鄠€runlevel級別,,也可以不為runlevel字段指定特定的值。
action字段定義了該進程應該運行在何種狀態(tài)下:
代碼:
boot 在系統啟動時運行,,忽略runlevel
bootwait 在系統啟動時運行,,init等待進程完成。忽略runlevel
ctrlaltdel 當Ctrl+Alt+Del三個鍵同時按下時運行,,把SIGINT信號發(fā)送給init,。忽略runlevel
initdefault 不要執(zhí)行這個進程,它用于設置默認runlevel
kbrequest 當init從鍵盤中收到信號時運行,。這里要求鍵盤組合符合KeyBoardSigral(參見/usr/share/doc/kbd-*關于鍵盤組合的文檔)
off 禁止進入,,因此該進程不運行
once 每一個runlevel級別運行一次
ondemand 當系統指定特定的運行級別A、B,、C時運行
powerfail 當init收到SIGPWR信號時運行
powerokwait 當收到SIGPWD信號且/etc/文件中的電源狀態(tài)包含OK時運行
powerwait 當收到SIGPWD信號,,并且init等待進程結束時運行
respawn 不管何時終止都重新啟動進程
sysinit 在運行boot或bootwait進程之前運行
wait 運行進程等待輸入運行模式
process字段包含init執(zhí)行的進程,該進程采用的格式與在命令行下運行該進程的格式一樣,,因此process字段都以該進程的名字開頭,緊跟著是 運行時,,緊跟著是運行時要傳遞給該進程的參數,。比如/sbin/shutdown -t3 -r now,該進程在按下Ctrl+Alt+Del時執(zhí)行,,在命令行下也可以直接輸入來重新啟動系統,。
特殊目的的記錄
仔細學習例子文件,學習應用其中關于inittab的語法格式,。該文件的大多數內容都可以忽略,,因為超過一半的內容都是注釋,剩余的一些文件內容主要是用來實現某些特殊的功能:
id 的值表明缺省的runlevel是3。
ud 的值可以喚醒/sbin/update進程,,該進程為保持磁盤的完整性,,將在對磁盤進行I/O操作之前清空整個I/O緩沖區(qū)。
pf,、pr和ca的值只被特定的中斷所調用,。
如果系統是專用的X終端,則只需x的輸入值,。
getty進程來提供虛擬終端設備的服務,,例如:
3:2345:respawn:/sbin/mingetty tty3
標簽字段的值是3,3是設備tty3的數字后綴,tty3與相應的進程相關聯,該getty進程可以啟動的runlevel是2,、3,、4和5, 當該進程終止時,init馬上就重新啟動它,。啟動進程的路徑名是/sbin/mingetty,,該進程是實現虛擬終端支持的最小版本的getty,為tty3提供啟動虛擬設備的進程,。
si::sysinit:/etc/rc.d/rc.sysinit
該值告訴init程序運行/etc/rc.d/rc.sysinit腳本文件來初始化系統,,該腳本文件與所有啟動的腳本類似,它只是一個包含Linux的shell命令的可執(zhí)行文件,,注意輸入的字符串必須包括該腳本的完整路徑,。不同版本的Linux存放該腳本的位置也不相同,但不用刻意去記憶這些位置,,只需查看/etc/inittab文件即可,,該文件中包含啟動腳本文件的確切位置。*/
# System initialization.
si::sysinit:/etc/rc.d/rc.sysinit
l0:0:wait:/etc/rc.d/rc 0
l1:1:wait:/etc/rc.d/rc 1
l2:2:wait:/etc/rc.d/rc 2
l3:3:wait:/etc/rc.d/rc 3
l4:4:wait:/etc/rc.d/rc 4
###當運行級別為5時,,以5為參數運行/etc/rc.d/rc腳本,,init將等待其返回(wait)
l5:5:wait:/etc/rc.d/rc 5
l6:6:wait:/etc/rc.d/rc 6
###在啟動過程中允許按CTRL-ALT-DELETE重啟系統
# Trap CTRL-ALT-DELETE
ca::ctrlaltdel:/sbin/shutdown -t3 -r now
# When our UPS tells us power has failed, assume we have a few minutes
# of power left. Schedule a shutdown for 2 minutes from now.
# This does, of course, assume you have powerd installed and your
# UPS connected and working correctly.
pf::powerfail:/sbin/shutdown -f -h +2 "Power Failure; System Shutting Down"
# If power was restored before the shutdown kicked in, cancel it.
pr:12345:powerokwait:/sbin/shutdown -c "Power Restored; Shutdown Cancelled"
###在2、3,、4,、5級別上以ttyX為參數執(zhí)行/sbin/mingetty程序,打開ttyX終端用于用戶登錄,,
###如果進程退出則再次運行mingetty程序(respawn)
# Run gettys in standard runlevels
1:2345:respawn:/sbin/mingetty tty1
2:2345:respawn:/sbin/mingetty tty2
3:2345:respawn:/sbin/mingetty tty3
4:2345:respawn:/sbin/mingetty tty4
5:2345:respawn:/sbin/mingetty tty5
6:2345:respawn:/sbin/mingetty tty6
###在5級別上運行xdm程序,,提供xdm圖形方式登錄界面,并在退出時重新執(zhí)行(respawn)
# Run xdm in runlevel 5
x:5:respawn:/etc/X11/prefdm -nodaemon
以上面的inittab文件為例,,來說明一下inittab的格式,。其中以#開始的行是注釋行,除了注釋行之外,,每一行都有以下格式:
id:runlevel:action:process
對上面各項的詳細解釋如下:
1. id
id是指入口標識符,,它是一個字符串,對于getty或mingetty等其他login程序項,要求id與tty的編號相同,,否則getty程序將不能正常工作,。
2. runlevel
runlevel是init所處于的運行級別的標識,一般使用0-6以及S或s,。0,、1、6運行級別被系統保留:其中0作為shutdown動作,,1作為重啟至單用戶模式,,6為重啟;S和s意義相同,表示單用戶模式,,且無需inittab文件,,因此也不在inittab中出現,實際上,,進入單用戶模式時,,init直接在控制臺(/dev/console)上運行/sbin/sulogin。在一般的系統實現中,,都使用了2,、3、4,、5幾個級別,,在Redhat系統中,2表示無NFS支持的多用戶模式,,3表示完全多用戶模式(也是最常用的級別),,4保留給用戶自定義,5表示XDM圖形登錄方式,。7- 9級別也是可以使用的,,傳統的Unix系統沒有定義這幾個級別。runlevel可以是并列的多個值,,以匹配多個運行級別,,對大多數action來說,僅當runlevel與當前運行級別匹配成功才會執(zhí)行,。
3. action
action是描述其后的process的運行方式的,。action可取的值包括:initdefault、sysinit,、boot,、bootwait等:
initdefault是一個特殊的action值,,用于標識缺省的啟動級別;當init由核心激活以后,,它將讀取inittab中的initdefault項,取得其中的runlevel,并作為當前的運行級別,。如果沒有inittab文件,,或者其中沒有initdefault項,init將在控制臺上請求輸入runlevel,。
sysinit,、boot、bootwait等action將在系統啟動時無條件運行,,而忽略其中的runlevel,。
其余的action(不含initdefault)都與某個runlevel相關。各個action的定義在inittab的man手冊中有詳細的描述,。
4. process
process為具體的執(zhí)行程序,。程序后面可以帶參數。
第三部分:系統初始化
在init的配置文件中有這么一行:
si::sysinit:/etc/rc.d/rc.sysinit
它調用執(zhí)行了/etc/rc.d/rc.sysinit,,而rc.sysinit是一個bash shell的腳本,,它主要
在init的配置文件中有這么一行:
si::sysinit:/etc/rc.d/rc.sysinit
它調用執(zhí)行了/etc/rc.d/rc.sysinit,而rc.sysinit是一個bash shell的腳本,,它主要是完成一些系統初始化的工作,,rc.sysinit是每一個運行級別都要首先運行的重要腳本。它主要完成的工作有:激活交換分區(qū),,檢查磁盤,,加載硬件模塊以及其它一些需要優(yōu)先執(zhí)行任務。
rc.sysinit約有850多行,,但是每個單一的功能還是比較簡單,,而且?guī)в凶⑨專ㄗh有興趣的用戶可以自行閱讀自己機器上的該文件,,以了解系統初始化所詳細情況,。由于此文件較長,所以不在本文中列出來,,也不做具體的介紹,。
當rc.sysinit程序執(zhí)行完畢后,將返回init繼續(xù)下一步,。
第四部分:啟動對應運行級別的守護進程
在rc.sysinit執(zhí)行后,,將返回init繼續(xù)其它的動作,通常接下來會執(zhí)行到/etc/rc.d/rc程序,。以運行級別5為例,,init將執(zhí)行配置文件inittab中的以下這行:
l5:5:wait:/etc/rc.d/rc 5
這一行表示以5為參數運行/etc/rc.d/rc,/etc/rc.d/rc是一個Shell腳本,,它接受5作為參數,,去執(zhí)行/etc /rc.d/rc5.d/目錄下的所有的rc啟動腳本,,/etc/rc.d/rc5.d/目錄中的這些啟動腳本實際上都是一些鏈接文件,而不是真正的rc 啟動腳本,,真正的rc啟動腳本實際上都是放在/etc/rc.d/init.d/目錄下,。而這些rc啟動腳本有著類似的用法,它們一般能接受start,、stop,、restart、status等參數,。
/etc/rc.d/rc5.d/中的rc啟動腳本通常是K或S開頭的鏈接文件,,對于以以S開頭的啟動腳本,將以start參數來運行,。而如果發(fā)現存在相應的腳本也存在K打頭的鏈接,,而且已經處于運行態(tài)了(以/var/lock/subsys/下的文件作為標志),則將首先以stop為參數停止這些已經啟動了的守護進程,,然后再重新運行,。這樣做是為了保證是當init改變運行級別時,所有相關的守護進程都將重啟,。
至于在每個運行級中將運行哪些守護進程,,用戶可以通過chkconfig或setup中的"System Services"來自行設定。常見的守護進程有:
amd:自動安裝NFS守護進程
apmd:高級電源管理守護進程
arpwatch:記錄日志并構建一個在LAN接口上看到的以太網地址和IP地址對數據庫
autofs:自動安裝管理進程automount,,與NFS相關,,依賴于NIS
crond:Linux下的計劃任務的守護進程
named:DNS服務器
netfs:安裝NFS、Samba和NetWare網絡文件系統
network:激活已配置網絡接口的腳本程序
nfs:打開NFS服務
portmap:RPC portmap管理器,,它管理基于RPC服務的連接
sendmail:郵件服務器sendmail
smb:Samba文件共享/打印服務
syslog:一個讓系統引導時起動syslog和klogd系統日志守候進程的腳本
xfs:X Window字型服務器,,為本地和遠程X服務器提供字型集
Xinetd:支持多種網絡服務的核心守護進程,可以管理wuftp,、sshd,、telnet等服務
這些守護進程也啟動完成了,rc程序也就執(zhí)行完了,,然后又將返回init繼續(xù)下一步,。
第五部分:建立終端
rc執(zhí)行完畢后,返回init,。這時基本系統環(huán)境已經設置好了,,各種守護進程也已經啟動了。init接下來會打開6個終端,,以便用戶登錄系統,。通過按Alt+Fn(n對應1-6)可以在這6個終端中切換。在inittab中的以下6行就是定義了6個終端:
1:2345:respawn:/sbin/mingetty tty1
2:2345:respawn:/sbin/mingetty tty2
3:2345:respawn:/sbin/mingetty tty3
4:2345:respawn:/sbin/mingetty tty4
5:2345:respawn:/sbin/mingetty tty5
6:2345:respawn:/sbin/mingetty tty6
從上面可以看出在2,、3,、4,、5的運行級別中都將以respawn方式運行mingetty程序,mingetty程序能打開終端,、設置模式。同時它會顯示一個文本登錄界面,,這個界面就是我們經??吹降牡卿浗缑妫谶@個登錄界面中會提示用戶輸入用戶名,,而用戶輸入的用戶將作為參數傳給login 程序來驗證用戶的身份,。
第六部分:登錄系統,啟動完成
對于運行級別為5的圖形方式用戶來說,,他們的登錄是通過一個圖形化的登錄界面,。登錄成功后可以直接進入KDE、Gnome等窗口管理器,。而本文主要講的還是文本方式登錄的情況:
當我們看到mingetty的登錄界面時,,我們就可以輸入用戶名和密碼來登錄系統了。
Linux的賬號驗證程序是login,,login會接收mingetty傳來的用戶名作為用戶名參數,。然后login會對用戶名進行分析:如果用戶名不是root,且存在/etc/nologin文件,,login將輸出nologin文件的內容,,然后退出。這通常用來系統維護時防止非root 用戶登錄,。只有/etc/securetty中登記了的終端才允許root用戶登錄,,如果不存在這個文件,則root可以在任何終端上登錄,。/etc /usertty文件用于對用戶作出附加訪問限制,,如果不存在這個文件,則沒有其他限制,。
在分析完用戶名后,,login將搜索/etc/passwd以及/etc/shadow來驗證密碼以及設置賬戶的其它信息,比如:主目錄是什么,、使用何種shell,。如果沒有指定主目錄,將默認為根目錄;如果沒有指定shell,,將默認為/bin/bash,。
login程序成功后,會向對應的終端在輸出最近一次登錄的信息(在/var/log/lastlog中有記錄),,并檢查用戶是否有新郵件(在/usr/spool/mail/的對應用戶名目錄下),。然后開始設置各種環(huán)境變量:對于bash來說,,系統首先尋找/etc/profile腳本文件,并執(zhí)行它;然后如果用戶的主目錄中存在.bash_profile文件,,就執(zhí)行它,,在這些文件中又可能調用了其它配置文件,所有的配置文件執(zhí)行后后,,各種環(huán)境變量也設好了,,這時會出現大家熟悉的命令行提示符,到此整個啟動過程就結束了,。
inittab 文檔描述在系統引導及通常的操作期間, 都啟動哪些進程(比如/etc/init.d/boot, /etc/init.d/rc, getty 等等).
Init
(8) 討論有關runlevels (運行級) 的概念, 每一個運行級都有他自己啟動進程的集合. 有效的運行級為0 -6 加上用于ondemand 條目的A , B 和C . inittab 文檔中的每一個條目有如下的格式:
id :runlevels :action :process
以`#’ 開頭的行被忽略.
id
inittab 文檔中條目的唯一標識, 限于1-4 個字符(假如是用版本號小于5.2.18 或a.out 的庫編譯生成的sysvinit 程式, 則僅限于2 個字符).
注意: 對于getty 或其他的注冊進程, id 必須是響應的終端線路的tty 后綴, 如1 響應tty1 , 否則, 注冊過程不能正常的工作.
runlevels
列出發(fā)生指定動作的運行級.
action
描述要發(fā)生的動作.
process
要執(zhí)行的進程. 假如process 域以一個`+’ 開頭, init 不會在utmp 和wtmp 文檔中為此進程記帳. 這是由于getty 自己主持utmp/wtmp 記帳的需要, 同時這也是個歷史遺留的漏洞.
runlevels 域能夠包含表示不同運行級的多個字符, 例如123 表示本進程在運行級為1, 2 和3 時都要啟動. 用于ondemand 條目的runlevels 域能夠包含A , B , 或C . 用于sysinit , boot , 和bootwait 條目的runlevels 域被忽略.
當改變運行級時, 在新運行級中沒有給出的那些正在運行的進程被殺死, 先使用SIGTERM 信號, 然后是SIGKILL.
action 域能夠使用的動作有:
respawn
該進程只要終止就立即重新啟動(如getty).
wait
只要進入指定的運行級就啟動本進程, 并且init 等待該進程的結束.
once
只要進入指定的運行級就啟動一次本進程.
boot
在系統引導期間執(zhí)行本進程. runlevels 域被忽略.
bootwait
在系統引導期間執(zhí)行本進程. 并且init 等待該進程的結束(如/etc/rc). runlevels 域被忽略.
off
什么也不做.
ondemand
在進入ondemand 運行級時才會執(zhí)行標記為ondemand 的那些進程. 無論怎樣, 實際上沒有改變運行級(ondemand 運行級就是`a’, `b’, 和`c’).
initdefault
initdefault 條目給出系統引導完成后進入的運行級, 假如不存在這樣的條目, init 就會在控制臺詢問要進入的運行級. process 域被忽略.
sysinit
系統引導期間執(zhí)行此進程. 本進程會在boot 或bootwait 條目之前得到執(zhí)行. runlevels 域被忽略.
powerwait
本進程在電源不足時執(zhí)行. 通常在有進程把UPS 和電腦相連時通知init 進程, Init 在繼續(xù)其他工作之前要等待此進程結束.
powerfail
類似powerwait , 但是init 不等待此進程完成.
powerokwait
在init 收到電源已恢復的通知后立即執(zhí)行此進程.
powerfailnow
本進程在init 被告知UPS 電源快耗盡同時外部電源失敗(無效) 時被執(zhí)行. (假設UPS 和監(jiān)控進程能夠發(fā)現這樣的情況).
ctrlaltdel
在init 收到SIGINT 信號時執(zhí)行此進程. 這意味著有人在控制臺按下了CTRL-ALT-DEL 組合鍵, 典型地, 可能是想執(zhí)行類似shutdown 然后進入單用戶模式或重新引導機器.
kbrequest
本進程在init 收到一個從控制臺鍵盤產生的特別組合按鍵信號時執(zhí)行.
對于此功能本文檔尚未完成; 能夠在kbd-x.xx 包中找到更多信息(在寫作本文檔時最新的是kbd-0.94). 當然您可能想為某些"KeyboardSignal" 行為映射組合鍵, 如為了映射(Alt-上箭頭) 能夠在鍵盤映射文檔中 使用如下的方式:
alt keycode 103 = KeyboardSignal
舉例
這是個和老的Linux inittab 文檔類似的例子文檔:
# inittab for linux
id:1:initdefault:
rc::bootwait:/etc/rc
1:1:respawn:/etc/getty 9600 tty1
2:1:respawn:/etc/getty 9600 tty2
3:1:respawn:/etc/getty 9600 tty3
4:1:respawn:/etc/getty 9600 tty4
本文檔在引導時執(zhí)行/etc/rc 并且在ty1-tty4 上啟動getty 進程.
一個更詳盡的inittab 會有不同的運行級(參考本身的注釋):
# 進入默認的運行級
id:2:initdefault:
# 在進行其他工作之前先完成系統初始化.
si::sysinit:/etc/rc.d/bcheckrc
# 運行級0 掛起系統, 6 重新引導, 1 單用戶模式.
l0:0:wait:/etc/rc.d/rc.halt
l1:1:wait:/etc/rc.d/rc.single
l2:2345:wait:/etc/rc.d/rc.multi
l6:6:wait:/etc/rc.d/rc.reboot
# "3 個鍵" 按下時要做的工作.
ca::ctrlaltdel:/sbin/shutdown -t5 -rf now
# 運行級2和3: 在控制臺生成getty 進程, 運行級為3時在modem 上生成getty.
1:23:respawn:/sbin/getty tty1 VC linux
2:23:respawn:/sbin/getty tty2 VC linux
3:23:respawn:/sbin/getty tty3 VC linux
4:23:respawn:/sbin/getty tty4 VC linux
S2:3:respawn:/sbin/uugetty ttyS2 M19200
文檔
/etc/inittab
zylonite 上是
::sysinit:/etc/init.d/rcS
ttyS0::respawn:-/bin/sh
ttyS1::respawn:-/bin/sh
ttyS2::respawn:-/bin/sh
tty1::respawn:-/bin/sh
tty2::respawn:-/bin/sh
::ctrlaltdel:/bin/umount -a -r
fedra 3 上是
#
# inittab This file describes how the INIT process should set up
# the system in a certain run-level.
#
# Author: Miquel van Smoorenburg,
# Modified for RHS Linux by Marc Ewing and Donnie Barnes
#
# Default runlevel. The runlevels used by RHS are:
# 0 - halt (Do NOT set initdefault to this)
# 1 - Single user mode
# 2 - Multiuser, without NFS (The same as 3, if you do not have networking)
# 3 - Full multiuser mode
# 4 - unused
# 5 - X11
# 6 - reboot (Do NOT set initdefault to this)
#
id:3:initdefault:
# System initialization.
si::sysinit:/etc/rc.d/rc.sysinit
l0:0:wait:/etc/rc.d/rc 0
l1:1:wait:/etc/rc.d/rc 1
l2:2:wait:/etc/rc.d/rc 2
l3:3:wait:/etc/rc.d/rc 3
l4:4:wait:/etc/rc.d/rc 4
l5:5:wait:/etc/rc.d/rc 5
l6:6:wait:/etc/rc.d/rc 6
# Trap CTRL-ALT-DELETE
ca::ctrlaltdel:/sbin/shutdown -t3 -r now
# When our UPS tells us power has failed, assume we have a few minutes
# of power left. Schedule a shutdown for 2 minutes from now.
# This does, of course, assume you have powerd installed and your
# UPS connected and working correctly.
pf::powerfail:/sbin/shutdown -f -h +2 "Power Failure; System Shutting Down"
# If power was restored before the shutdown kicked in, cancel it.
pr:12345:powerokwait:/sbin/shutdown -c "Power Restored; Shutdown Cancelled"
# Run gettys in standard runlevels
1:2345:respawn:/sbin/mingetty tty1
2:2345:respawn:/sbin/mingetty tty2
3:2345:respawn:/sbin/mingetty tty3
4:2345:respawn:/sbin/mingetty tty4
5:2345:respawn:/sbin/mingetty tty5
6:2345:respawn:/sbin/mingetty tty6
# Run xdm in runlevel 5
x:5:respawn:/etc/X11/prefdm -nodaemon