Xiaobo Sun

          Eclipse-Unix http://umlfact.berlios.de/~s_xsun/

          Linux 啟動

          前言
          linux有自己一套完整的啟動體系,抓住了linux啟動的脈絡,linux的啟動過程將不再神秘。
          閱讀之前建議先看一下附圖。

          本文中假設inittab中設置的init tree為:
          /etc/rc.d/rc0.d
          /etc/rc.d/rc1.d
          /etc/rc.d/rc2.d
          /etc/rc.d/rc3.d
          /etc/rc.d/rc4.d
          /etc/rc.d/rc5.d
          /etc/rc.d/rc6.d
          /etc/rc.d/init.d

          目錄
          1. 關于linux的啟動
          2. 關于rc.d
          3. 啟動腳本示例
          4. 關于rc.local
          5. 關于bash啟動腳本
          6. 關于開機程序的自動啟動




          1. 關于linux的啟動
          init是所有進程之父
          init讀取/etc/inittab,執行rc.sysinit腳本
          (注意文件名是不一定的,有些unix甚至會將語句直接寫在inittab中)
          rc.sysinit腳本作了很多工作:

          init $PATH
          config network
          start swap function
          set hostname
          check root file system, repair if needed
          check root space
          ....

          rc.sysinit根據inittab執行rc?.d腳本
          linux是多用戶系統,getty是多用戶與單用戶的分水嶺
          在getty之前運行的是系統腳本

          2. 關于rc.d
          所有啟動腳本放置在 /etc/rc.d/init.d下
          rc?.d中放置的是init.d中腳本的鏈接,命名格式是:
          S{number}{name}
          K{number}{name}
          S開始的文件向腳本傳遞start參數
          K開始的文件向腳本傳遞stop參數
          number決定執行的順序

          3. 啟動腳本示例

          這是一個用來啟動httpd的 /etc/rc.d/init.d/apache 腳本:

          代碼:
          #!/bin/bash

          source /etc/sysconfig/rc
          source $rc_functions

          case "$1" in
                  start)
                          echo "Starting Apache daemon..."
                          /usr/local/apache2/bin/apachectl -k start
                          evaluate_retval
                          ;;

                  stop)
                          echo "Stopping Apache daemon..."
                          /usr/local/apache2/bin/apachectl -k stop
                          evaluate_retval
                          ;;

                  restart)
                          echo "Restarting Apache daemon..."
                          /usr/local/apache2/bin/apachectl -k restart
                          evaluate_retval
                          ;;

                  status)
                          statusproc /usr/local/apache2/bin/httpd
                          ;;
                          
                  *)
                          echo "Usage: $0 {start|stop|restart|status}"
                          exit 1
                          ;;
          esac
          可以看出他接受start,stop,restart,status參數

          然后可以這樣建立rc?.d的鏈接:

          代碼:
          cd /etc/rc.d/init.d &&
          ln -sf ../init.d/apache ../rc0.d/K28apache &&
          ln -sf ../init.d/apache ../rc1.d/K28apache &&
          ln -sf ../init.d/apache ../rc2.d/K28apache &&
          ln -sf ../init.d/apache ../rc3.d/S32apache &&
          ln -sf ../init.d/apache ../rc4.d/S32apache &&
          ln -sf ../init.d/apache ../rc5.d/S32apache &&
          ln -sf ../init.d/apache ../rc6.d/K28apache

          4. 關于rc.local

          經常使用的 rc.local 則完全是習慣問題,不是標準。
          各個發行版有不同的實現方法,可以這樣實現:

          代碼:
          touch /etc/rc.d/rc.local
          chmod +x /etc/rc.d/rc.local
          ln -sf /etc/rc.d/rc.local /etc/rc.d/rc1.d/S999rc.local &&
          ln -sf /etc/rc.d/rc.local /etc/rc.d/rc2.d/S999rc.local &&
          ln -sf /etc/rc.d/rc.local /etc/rc.d/rc3.d/S999rc.local &&
          ln -sf /etc/rc.d/rc.local /etc/rc.d/rc4.d/S999rc.local &&
          ln -sf /etc/rc.d/rc.local /etc/rc.d/rc5.d/S999rc.local &&
          ln -sf /etc/rc.d/rc.local /etc/rc.d/rc6.d/S999rc.local

          5. 關于bash啟動腳本

          /etc/profile
          /etc/bashrc
          ~/.bash_profile
          ~/.bashrc
          是bash的啟動腳本

          一般用來設置單用戶的啟動環境,也可以實現開機單用戶的程序,但要明確他們都是屬于bash范疇而不是系統范疇。

          他們的具體作用介紹如下:

          /bin/bash這個命令解釋程序(后面簡稱shell)使用了一系列啟動文件來建立一個運行環境:

          /etc/profile
          /etc/bashrc
          ~/.bash_profile
          ~/.bashrc
          ~/.bash_logout
          每一個文件都有特殊的功用并對登陸和交互環境有不同的影響。
          /etc/profile 和 ~/.bash_profile 是在啟動一個交互登陸shell的時候被調用。
          /etc/bashrc 和 ~/.bashrc 是在一個交互的非登陸shell啟動的時候被調用。
          ~/.bash_logout 在用戶注銷登陸的時候被讀取

          一個交互的登陸shell會在 /bin/login 成功登陸之后運行。一個交互的非登陸shell是通過命令行來運行的,如 [prompt]$/bin/bash。一般一個非交互的shell出現在運行shell腳本的時候。之所以叫非交互的shell,是因為它不在命令行上 等待輸入而只是執行腳本程序。
          =====================================================================================================
          本文以RedHat9.0和i386平臺為例,剖析了從用戶打開電源直到屏幕出現命令行提示符的整個Linux啟動過程。并且介紹了啟動中涉及到的各種文件。

            閱讀Linux源代碼,無疑是深入學習Linux的最好方法。在本文對Linux啟動過程的介紹中,我們也嘗試從源代碼的視角來更深入的剖析Linux的啟動過程,所以其中也簡單涉及到部分相關的Linux源代碼,Linux啟動這部分的源碼主要使用的是C語言,也涉及到了少量的匯編。而啟動過程中也執行了大量的shell(主要是bash shell)所寫腳本。為了方便讀者閱讀,筆者將整個Linux啟動過程分成以下幾個部分逐一介紹,大家可以參考下圖:

            當用戶打開PC的電源,BIOS開機自檢,按BIOS中設置的啟動設備(通常是硬盤)啟動,接著啟動設備上安裝的引導程序lilo 或grub開始引導Linux,Linux首先進行內核的引導,接下來執行init程序,init程序調用了rc.sysinit和rc等程 序,rc.sysinit和rc當完成系統初始化和運行服務的任務后,返回init;init啟動了mingetty后,打開了終端供用戶登錄系統,用戶 登錄成功后進入了Shell,這樣就完成了從開機到登錄的整個啟動過程。

          下面就將逐一介紹其中幾個關鍵的部分:

            第一部分:內核的引導(核內引導)

            Red Hat9.0可以使用lilo或grub等引導程序開 始引導Linux系統,當引導程序成功完成引導任務后,Linux從它們手中接管了CPU的控制權,然后CPU就開始執行Linux的核心映象代碼,開始 了Linux啟動過程。這里使用了幾個匯編程序來引導Linux,這一步泛及到Linux源代碼樹中的“arch/i386/boot”下的這幾個文 件:bootsect.S、setup.S、video.S等。

            其中bootsect.S是生成引導扇區的匯編源碼,它完成加載動作后直接跳轉到setup.S的程序入口。setup.S的主要功能就是將系 統參數(包括內存、磁盤等,由BIOS返回)拷貝到特別內存中,以便以后這些參數被保護模式下的代碼來讀取。此外,setup.S還將video.S中的 代碼包含進來,檢測和設置顯示器和顯示模式。最后,setup.S將系統轉換到保護模式,并跳轉到 0x100000。

            那么0x100000這個內存地址中存放的是什么代碼?而這些代碼又是從何而來的呢?

            0x100000這個內存地址存放的是解壓后的內核,因為Red Hat提供的內核包含了眾多驅動和 功能而顯得比較大,所以在內核編譯中使用了“makebzImage”方式,從而生成壓縮過的內核,在RedHat中內核常常被命名為vmlinuz,在 Linux的最初引導過程中,是通過"arch/i386/boot/compressed/"中的head.S利用misc.c中定義的 decompress_kernel()函數,將內核vmlinuz解壓到0x100000的。

            當CPU跳到0x100000時,將執行"arch/i386/kernel/head.S"中的startup_32,它也是vmlinux 的入口,然后就跳轉到start_kernel()中去了。start_kernel()是"init/main.c"中的定義的函 數,start_kernel()中調用了一系列初始化函數,以完成kernel本身的設置。start_kernel()函數中,做了大量的工作來建立 基本的Linux核心環境。如果順利執行完start_kernel(),則基本的Linux核心環境已經建立起來了。

            在start_kernel()的最后,通過調用init()函數,系統創建第一個核心線程,啟動了init過程。而核心線程init()主要 是來進行一些外設初始化的工作的,包括調用do_basic_setup()完成外設及其驅動程序的加載和初始化。并完成文件系統初始化和root文件系 統的安裝。

            當do_basic_setup()函數返回init(),init()又打開了/dev/console設備,重定向三個標準的輸入輸出文件 stdin、stdout和stderr到控制臺,最后,搜索文件系統中的init程序(或者由init=命令行參數指定的程序),并使用 execve()系統調用加載執行init程序。到此init()函數結束,內核的引導部分也到此結束了,

          第二部分:運行init

            init的進程號是1,從這一點就能看出,init進程是系統所有進程的起點,Linux在完成核內引導以后,就開始運行init程序,。init程序需要讀取配置文件/etc/inittab。inittab是一個不可執行的文本文件,它有若干行指令所組成。在Redhat系統中,inittab的內容如下所示(以“###"開始的中注釋為筆者增加的):

            #
          # inittab       This file describes how the INIT process should set up
          #               the system in a certain run-level.
          #
          # Author:       Miquel van Smoorenburg, <miquels@drinkel.nl.mugnet.org>
          #               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 havenetworking)
          #   3 - Full multiuser mode
          #   4 - unused
          #   5 - X11
          #   6 - reboot (Do NOT set initdefault to this)
          #
          ###表示當前缺省運行級別為5(initdefault);
          id:5:initdefault:

            ###啟動時自動執行/etc/rc.d/rc.sysinit腳本(sysinit)
          # 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為參數執行/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圖形方式登錄界面,并在退出時重新執行(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與當前運行級別匹配成功才會執行。

            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為具體的執行程序。程序后面可以帶參數。

            第三部分:系統初始化

            在init的配置文件中有這么一行:

            si::sysinit:/etc/rc.d/rc.sysinit

            它調用執行了/etc/rc.d/rc.sysinit,而rc.sysinit是一個bash shell的腳本,它主要是完成一些系統初始化的工作,rc.sysinit是每一個運行級別都要首先運行的重要腳本。它主要完成的工作有:激活交換分區,檢查磁盤,加載硬件模塊以及其它一些需要優先執行任務。

            rc.sysinit約有850多行,但是每個單一的功能還是比較簡單,而且帶有注釋,建議有興趣的用戶可以自行閱讀自己機器上的該文件,以了解系統初始化所詳細情況。由于此文件較長,所以不在本文中列出來,也不做具體的介紹。

            當rc.sysinit程序執行完畢后,將返回init繼續下一步。

          第四部分:啟動對應運行級別的守護進程

            在rc.sysinit執行后,將返回init繼續其它的動作,通常接下來會執行到/etc/rc.d/rc程序。以運行級別3為例,init將執行配置文件inittab中的以下這行:

            l5:5:wait:/etc/rc.d/rc 5

            這一行表示以5為參數運行/etc/rc.d/rc,/etc/rc.d/rc是一個Shell腳本,它接受5作為參數,去執行/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參數來運行。而如果 發現存在相應的腳本也存在K打頭的鏈接,而且已經處于運行態了(以/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程序也就執行完了,然后又將返回init繼續下一步。

          第五部分:建立終端

            rc執行完畢后,返回init。這時基本系統環境已經設置好了,各種守護進程也已經啟動了。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/的對應用戶名目錄下)。然后開始設置各種環境變量:對于bash來說,系統首先尋找/etc/profile腳本文件, 并執行它;然后如果用戶的主目錄中存在.bash_profile文件,就執行它,在這些文件中又可能調用了其它配置文件,所有的配置文件執行后后,各種 環境變量也設好了,這時會出現大家熟悉的命令行提示符,到此整個啟動過程就結束了。

            希望通過上面對Linux啟動過程的剖析能幫助那些想深入學習Linux用戶建立一個相關Linux啟動過程的清晰概念,進而可以進一步研究Linux接下來是如何工作的。



          posted on 2008-12-10 18:23 Xiaobo Sun 閱讀(596) 評論(1)  編輯  收藏 所屬分類: Unix

          評論

          # re: Linux 啟動 2008-12-10 20:32 Xiaobo Sun

          啟動服務

          安裝完成后,vsftpd是作為一個獨立的服務啟動,不接受xinetd的管理。系統會自動在Service列表中添加名稱vsftpd的服務。啟動服務的步驟如下:

            1、Yast→System→System Services(Run level);

            2、在服務列表中選擇vsftpd,點擊下方的Enable按鈕,啟動服務,Finish。

            注意,在啟動vsftpd時可能會不能啟動,有時是因為它所依賴的服務沒有啟動,如syslog,這時選中syslog,點擊Enable按鈕。如果還是不能啟動,可以重新啟動xinetd服務,然后再啟動vsftpd服務。  回復  更多評論   

          <2008年12月>
          30123456
          78910111213
          14151617181920
          21222324252627
          28293031123
          45678910

          導航

          統計

          常用鏈接

          留言簿(3)

          隨筆分類

          隨筆檔案

          搜索

          最新評論

          閱讀排行榜

          評論排行榜

          主站蜘蛛池模板: 偏关县| 温泉县| 玛多县| 平江县| 天峻县| 静宁县| 贵阳市| 迁西县| 泸州市| 高雄县| 阳谷县| 岢岚县| 兴城市| 海丰县| 堆龙德庆县| 通化县| 富阳市| 原平市| 邵武市| 泸溪县| 邻水| 抚宁县| 龙里县| 安仁县| 上虞市| 新宁县| 漳平市| 都安| 桂平市| 化德县| 双流县| 枞阳县| 凉山| 闵行区| 合阳县| 巴林右旗| 连城县| 黄石市| 内江市| 昌都县| 夏河县|