最小Linux系統(tǒng)制作例程
一,什么是BabyLinux
BabyLinux不是一個完整的發(fā)行版,他是利用原有的一套完整的linux系統(tǒng)的內核原代碼和編譯工具,利用busybox內建的強大功能,在一張軟盤上做的一個很小的linux系統(tǒng).他具備一個linux系統(tǒng)的基本特征,支持linux系統(tǒng)最常用的一百多個命令,支持多種文件系統(tǒng),支持網絡等等,你可以把他當做一張linux起動盤和修復盤來用,你也可以把他當做一個靜態(tài)路由的路由器軟件,當然,你也可以把他當做一個linux玩具,向你的朋友炫耀 linux可以做的多么小.我把他叫做BabyLinux因為他很小巧,小的很可愛,像一個剛剛出生的小baby.
二,為什么要作這樣一個linux
先說說我一開始的想法,當我一開始接觸linux的時候,看到書上說,linux通常安裝只需要60M左右的空間,但是我發(fā)現(xiàn)裝在我硬盤上的Redhat 6.0確要占據(jù)好幾百M的空間.為什么我的linux這么大呢? 后來我發(fā)現(xiàn),裝在我機器上的那么多東西只有不到30%是我平時常用的,還有30%是我極少用到的,另外的40%基本上是不用的.于是,我和大多數(shù)初學者一樣,開始抱怨,為什么linux不能做的精簡一點呢?于是,我萌發(fā)了自己裁減系統(tǒng)的想法.可惜那個時候我還沒有聽說過有LFS和Debain.等到我積累了足夠的linux知識后,我開始制作這樣一個小系統(tǒng).
制作這樣一個小系統(tǒng)最大的意義在于,你可以通過制作系統(tǒng)了解linux的啟動過程,學會ramdisk的使用,讓你在短時間內學到更多的linux知識. 當然,你會得到很大的樂趣.這個項目只是做一個具有基本特征的linux系統(tǒng),如果你想自己做一個具有完整功能的linux,請閱讀Linux From Scratch (LFS)文檔.
三,什么人適合讀這篇文檔
如果你是一個linux愛好者,并且很想了解linux的啟動過程和系統(tǒng)的基本結構,而且是一個喜歡動手研究小玩意的人,那么這個文檔可以滿足你的需求. 如果你僅僅是用linux來做一些普通的日常工作,而不在乎你的linux到底怎么工作,那么這份文檔也許不太適合你.另外,如果你是linux愛好者, 但是目前還是一個剛剛入門的newbi,我建議你先把linux命令學好.不過我想我會盡可能的把這份文檔寫詳細一些,如果你有足夠的毅力,或許一個 newbi也能成功做一個babylinux.或者,你遇到一件很不巧的事情,比如你的老婆來例假了,你的這個周末就泡湯了,那么閱讀這篇文檔并做一個 linux小玩具可以打發(fā)你的時間.
四,應該具備的知識
在做一個babylinux之前,你應當已經會應用linux最常用的命令.并且至少有一次成功編譯并安裝系統(tǒng)內核的經歷,會通過編譯源代碼來安裝軟件. 如果你具備了這些條件,那么做這樣一個小系統(tǒng)會很順利,如果你還沒有掌握這些知識,你可能會遇到一些困難.但是只要有毅力,也可以成功.你不需要具備編程的知識,因為我的目標是:讓具有中等以上linux水平的愛好者可以通過閱讀文檔輕松完成這個項目.關于一張軟盤上的linux還有一個很著名的 linux叫LOAP (Linux On A
Floppy) 但是他是由比較專業(yè)的人員需要編寫很多程序完成的.而且沒有關于他制作過程的文檔.
五,linux系統(tǒng)引導過程簡介
首先,主板的BIOS會讀取硬盤的主引導記錄(MBR),MBR中存放的是一段很小的程序,他的功能是從硬盤讀取操作系統(tǒng)核心文件并運行,因為這個小程序太小了,因此通常這個小程序不具備直接引導系統(tǒng)內核的能力,他先去引導另一個稍微大一點的小程序,再由這個大一點的小程序去引導系統(tǒng)內核.在linux系統(tǒng)中這樣的小程序有LILO和GRUB.在這個項目中,我決定用LILO來做系統(tǒng)引導程序.在軟盤上啟動linux系統(tǒng)的過程和在硬盤上啟動的過程相似.
Linux系統(tǒng)內核被引導程序裝入內核并運行后,linux內核會檢測系統(tǒng)中的各種硬件.并做好各種硬件的初始化工作,使他們在系統(tǒng)正式運行后能正常工作.之后內核做的最后一個工作是運行
/sbin 下的init程序,init是英文單詞initialization(初始化)的簡稱,init程序的工作是讀取/etc/inittab文件中描述的指令,對系統(tǒng)的各種軟硬件環(huán)境做最初化設定.最后運行mingetty等待用戶輸入用戶名登錄系統(tǒng).所有的工作就這么簡單,雖然linux啟動的時候有很多內容,看上去十分高深,但是都不過是對這個過程的擴充.明白了這個道理,你可以寫一些腳本程序讓他在系統(tǒng)啟動的特定時間運行完成任務.事實上系統(tǒng)內核并不關心/sbin下的init是不是真的init,只要是放在/sbin下名叫init的可執(zhí)行程序他都可以執(zhí)行.可以做以下實驗:
編寫一個非常簡單的C程序:
main()
{
printf(“hello,world!n”);
}
保存后以init.c保存他,并用gcc編譯.
#gcc –-static -o init init.c
這里的--static 參數(shù)告訴gcc把這個程序靜態(tài)聯(lián)接,這樣這個程序不倚賴任何庫就能運行.把編譯好的init程序拷貝到/sbin下,備份好原來的那個.重新啟動系統(tǒng)最后系統(tǒng)的輸出結果是: hello,world!
然后停在那里.做這個實驗以前先確定你知道如何把系統(tǒng)恢復到原來的狀態(tài),有一個簡單的方法,在內核啟動前給他加上init=參數(shù),比如你原先的init被你改成了init.bak 只要在啟動的時候給內核加上init=/sbin/init.bak就可以用原來的init程序啟動系統(tǒng).
做完以上實驗,就明白了內核和init程序之間的關系.此外,init程序不一定是一個二進制可執(zhí)行程序,他可以是一個bash腳本,一個指向另一個程序的聯(lián)接,他的位置也并不一定要在/sbin下,只要在啟動內核時,給內核加上init參數(shù)就能被運行,比如,開始時給內核加上init= /bin/bash參數(shù),內核在最后一步就直接運行bash給出提示符,不用登錄系統(tǒng)就可以輸入命令了.其功能類似單用戶模式啟動系統(tǒng). /sbin/init 程序只是內核默認運行的第一個程序.[!--empirenews.page--]六,編譯一個linux系統(tǒng)內核
1,編譯前的規(guī)劃和準備
在編譯內核前,請先確定你的需求,把你的需求羅列成一張詳細的表格.你需要讓內核支持什么硬件,支持多少種分區(qū)類型和文件系統(tǒng),支持哪些網卡,支持哪些網絡協(xié)議.等等.請盡可能詳細的羅列這些內容,但是你也不要太貪心,因為你所有能利用的空間只有1440K,如果你編譯出一個大于1440K或很接近這個數(shù)字的內核,你的這個項目就不能完成了,你已經沒有空間再放ramdisk映象文件,除非你原意再多出一張軟盤,做一個兩張軟盤的小linux系統(tǒng).對于聲卡驅動之類,我勸你還是放棄吧,因為一個聲卡驅動也許只讓你的內核增大了十多K,但是你有了一個聲卡驅動就務必要有一個播放器吧,否則聲卡驅動就沒有意義,可一個播放器的大小可不是一張軟盤可以裝得下的.在我先前制作的babylinux內核有900多K,其中,文件系統(tǒng)部分站了大部分,因為我的目標是把他做成一個系統(tǒng)修復盤.因此我在內核中編譯7種文件系統(tǒng)的支持,每減少一個文件系統(tǒng)就可以減小幾十甚至200多K的內核大小.越是復雜,越是安全的文件系統(tǒng),其支持模塊也越大,比如在linux下FAT模塊只有32K,VFAT只有17K,但是ext3的模塊就有86K,JFS達到216K, reiserfs模塊是224K,可以想像,編譯一個支持7個文件系統(tǒng)的900多K的內核,文件系統(tǒng)部分就占了600K以上的空間,所以如果某一個文件系統(tǒng)是你根本不用的,那么還是不要編譯進內核把,這樣至少可以省下100多K的空間.對于其他的驅動,比如網卡,通常大小只有8,9K,最大的也不過10多 K,因此可以把常用的網卡芯片的驅動都編譯進去.另外如果你想讓你的babylinux支持U盤,那么scsi的驅動模塊也是不可小看的,他通常要接近 150K,因為U盤是被當做scsi設備來驅動的.另外你還需要讓你的內核支持即插即用,這些都是不小的空間開銷,我的建議是你放棄一兩個你不用的文件系統(tǒng).總之,你最后編譯出來的內核大小最好不要超過900K,否則你在busybox里只能編譯進去很少的命令.
在我編譯的busybox中,我編譯進去120多個命令,基本上把busybox支持的命令都包括進去了.加上小系統(tǒng)所必需的文件系統(tǒng)目錄,/dev下的設備文件,以及/etc下幾個必需的配置文件,做成ramdisk壓縮后的大小是440多K, 加上900K左右的內核剛好可以放入一張1440K軟盤,請注意,你應該留下至少50K的空間,因為我們要在軟盤上創(chuàng)建一個ext2文件系統(tǒng),而文件系統(tǒng)本生需要占據(jù)大概25K的磁盤空間.另外lilo的引導文件boot.b的大小是5.7K,還有裝上lilo后自動產生的map文件也要10多K的空間, map文件的具體大小由內核安裝的實際大小決定,通常不會超過30K.
綜上所述,請遵循下面的公式:
內核大小+文件系統(tǒng)壓縮印象文件+50K <= 1440K
另外一點需要說明的是:以上所羅列的文件系統(tǒng)模塊大小是察看我現(xiàn)在使用的Redhat 9 的
/lib/modules下的模塊文件得到的,實際編譯進內核大小會小一點,因為我們用make bzImage
在內核源代碼目錄樹下生成的內核是經過壓縮過的.
如果你對以上說的內容不太明白也沒有關系,我會在下面的內容中做詳細的說明.
2,必需編譯進內核的內容
首先,我們制作的這個小系統(tǒng)是基于一張軟盤的,因此,你的內核必需支持軟盤.另外對IDE硬盤和cdrom的支持也是不可少的,否則做出來的 babylinux就沒有實用價值,因為他不能訪問硬盤和光盤上的內容這樣的linux雖然可以做的更小,但是制造一個完全沒有用的東西是浪費時間.其他的包括framebuffer等,如果你需要支持在字符界面下以高分辨率顯示,以看到更多的屏幕內容,那么就必需把framebuffer支持編譯進內核,此外在高分辨率下使用的8x8字體也必需編譯進去.否則即使你給內核傳遞了vga= 參數(shù),內核會因為沒有可用的小字體而自動轉跳到低分辨率模式下,這是以前困擾我好幾天想不明白的事情,后來通過反復試驗才明白原來是缺少字體的文體.這里我先大致提一下需要注意的事情.在下一小節(jié)具體編譯時,我會繼續(xù)就某些細節(jié)問題說明.
3,關于內核的版本
我是在Redhat 9 linux系統(tǒng)下打造的babylinux小系統(tǒng).使用的是Redhat 9 自帶的2.4.20版的內核.
為什么我不用最新的2.6的內核?
一開始我也企圖用最新的內核,但是通過試驗我發(fā)現(xiàn),在用最新的2.6.9內核的情況下,我編譯一個all-no的(即所有內容都選N,不支持任何硬件,只有一個最基本的內核)最小化內核就要460K左右,如果我在這個基礎上再加入幾種文件系統(tǒng)和必要的驅動,那么內核的大小就不能裝下一張1440K
的軟盤,而我用2.4.20的內核編譯一個最小化的內核只需要217K,的大小.如果優(yōu)化了gcc參數(shù)他還能再小些.這樣我就立即省下了200多K的空間,在平時,200多K的內容微不足道,但是在babylinux里,這個數(shù)目是整個空間的 1/7,相當于一個reiserfs文件系統(tǒng)模塊的大小.當然,我也嘗試了2.2以及更老的內核,但是他們缺少我需要的東西,因此最后權衡下來用2.4的內核是比較合理的.如果你用的是2.6內核的FC系統(tǒng),那么最好還是去下載一個2.4版的內核,www.kernel.org 有各個時期的內核可以下載.
4, 內核的配置
如果你對linux內核的配置和編譯已經很熟悉了,請?zhí)^這一段,直接看busybox的編譯.
以root身份登錄系統(tǒng)
進入/usr/src/linux目錄
[root@gucuiwen root]# cd /usr/src/linux
如果你下載了一個2.4版本的內核,為了避免麻煩,請將他拷貝到/usr/src下,然后接壓縮,再做一個指向他的名為linux的鏈接.雖然這并不是必需的,但是根據(jù)我以往的經驗,如果我把linux源代碼放在其他目錄下解開并編譯,偶然會有一些莫名其妙的小問題發(fā)生.
#cp linux-2.4.20.bz2 /usr/src/
#cd /usr/src
#tar xfvj linux-2.4.20.bz2
如果是tar.gz格式,可以這樣解開
#tar xfvz linux-2.4.20.tar.gz
為了方便,做一個到目錄linux-2.4.20的連接:
#ln -s linux-2.4.20 linux
進入linux源代碼目錄:
#cd linux
清理源代碼樹:
#make mrproper
運行配置程序:
#make xconfig
code maturity level options
先選擇N,當我們配置好常規(guī)的東西,要加入framebuffer支持時再將這一項選擇Y,因為在2.4.20中,framebuffer支持尚屬于實驗性代碼.如果不在code maturity level options選擇為Y,將不能配置framebuffer.
Loadable module support
選擇N,為了簡化系統(tǒng)的制作,我在這個項目中不選擇可加載內核模塊的支持.
processor type and features
processor family 中選擇你需要的CPU類型,如果你想讓老至386,新到P4的CPU都能運行babaylinux那么請選擇386CPU,否則請按自己的實際情況選擇.
其他選項都選擇N.這些在babylinux中都是不需要的.
General setup
networking support 選擇Y
PCI support 選擇Y 除非你不用PCI設備,不過一般人都是需要的,因為現(xiàn)在網卡大部分是PCI的.
System V ipc 選擇Y
systrl support選擇Y
kernel support for ELF 選擇Y
其余內容都可以選擇N,如果有特殊需求,比如的網卡是ISA的,那么請將相應的內容選上.但是不能貪心,時刻牢記,我們能利用的空間只有 1440K ,內核的大小絕不能超過 900K,任何不必要的東西都應該從內核中去除.
memory technology devices (MTD)
Parallel port support
Plug and Play configuration
以上三個大項中的所有內容選擇N
block devices
Normal floppy disk support
Loopback device support
RAM disk support
initial RAM disk (initrd) support
Per partition statics in /proc/partitions
以上幾項選擇Y,其余全部選擇N.
這里的選項比較重要,我想重點說明一下.對于軟盤的支持,那是不必說的,那是必備的.
loopback device 即回環(huán)設備,我們平時用命令
#mount -o loop somecd.ISO /mnt/cdrom
掛裝光盤映象文件,或者其他文件系統(tǒng)映象文件時就用到了內核中的loopback 模塊,如果沒有編譯進這個模塊,你將不能用上面的命令掛裝光盤映象和文件系統(tǒng)映象.
個人認為這個功能是非常重要的,所以編譯了進去.
RAM disk support 即內存磁盤(比較貼切的說法是虛擬磁盤,即撥出一部分內存當做磁盤用).這是制作babylinux項目中的核心內容,由于一張軟盤的空間有限, babylinux的根文件系統(tǒng)是用gzip壓縮法高度壓縮的,在運行時,將解壓縮后的文件拷貝到一個RAM disk運行,所以在運行時,你在根文件系統(tǒng)上的所有操作實際上是在內存上進行的.但是在形式上和在真正的磁盤上運行一樣.只不過放在RAM disk上的所有內容會在系統(tǒng)關機后全部消失.
不僅在運行babylinux時用到ramdisk,我們在制作壓縮的根文件系統(tǒng)時也要用到ramdisk,學習ramdisk的使用是做一個babylinux的重要目的之一. 在linux中,還支持另外一種虛擬磁盤,叫做shm,
(shared memory),這種虛擬磁盤機制比ramdisk更加先進,ramdisk的大小是固定的,由編譯內核時候的default ram disk size 決定.默認為4096K(4M),也可以在內核裝載前加上ramdisk_size=參數(shù)來決定他的大小,但是系統(tǒng)一旦啟動,ramdisk的大小是不能改變的,而shm的大小卻動態(tài)的改變.默認情況下為物理內存的一半,當系統(tǒng)需要更多內存的時,他就自動縮小.系統(tǒng)內存富余時,他自動增大,這樣可以充分靈活的利用內存空間,shm通常用來作為系統(tǒng)的磁盤高速緩存,存放系統(tǒng)運行中的臨時文件等.redaht 的linux在默認情況下都有shm的支持,可以用mount和df察看他的掛裝點和大小,如下命令:
[root@gucuiwen linux]# mount
/dev/hda1 on / type ext3 (rw)
none on /proc type proc (rw)
usbdevfs on /proc/bus/usb type usbdevfs (rw)
none on /dev/pts type devpts (rw,gid=5,mode=620)
/dev/hda6 on /home type ext3 (rw)
/dev/hda5 on /oracle type ext3 (rw)
none on /dev/shm type tmpfs (rw)
/dev/hda7 on /var type ext3 (rw)
[root@gucuiwen linux]# df -h
文件系統(tǒng) 容量 已用 可用 已用% 掛載點
/dev/hda1 2.9G 2.7G 26M 100% /
/dev/hda6 3.8G 1.8G 1.8G 50% /home
/dev/hda5 5.7G 677M 4.8G 13% /oracle
none 125M 0 125M 0% /dev/shm
/dev/hda7 711M 91M 584M 14% /var
雖然shm有這么多的優(yōu)點,我還是選擇了ramdisk,因為ramdisk可以很方便地在系統(tǒng)啟動的時候加載,而shm卻沒那么容易,下面就來講一下關于內核啟動時加載ramdisk映象的相關內容.
initial RAM disk (initrd) support
即初始化ramdisk支持,這個選項讓內核有能力在內核加載階段就能裝入RAMDISK,并運行其中的內容,否則只能在系統(tǒng)運行階段用ramdisk ,我們平時在編譯了一個新內核后,如果你的根文件系統(tǒng)用的是ext3,而你沒有把ext3編譯進內核,而只作為一個模塊編譯了,那么就需要用 mkinitrd命令做一個initrd (initializtion ramdisk),這個ramdisk里放了ext3的模塊,這樣內核在加載根文件系統(tǒng)前就能正確識別ext3文件系統(tǒng).否則,內核加載的最后一步就會出現(xiàn)kernel panic cant not find init .... 的錯誤.
在babylinux項目中,這個選項是必需的,這里的作用是把解壓的根文件系統(tǒng)映象裝入ramdisk.
Per partition statics in /proc/partitions
這個選項不是必需的,但是我發(fā)現(xiàn)如果我不把這個功能編譯進內核,那么當我在掛裝文件系統(tǒng)的時候會有些小問題,比如我不能以簡寫的掛裝命令來掛裝文件系統(tǒng). 我不確定到底是不是這個選項的關系,但是把這個選項編譯進內核只增大一點點內核空間,所以為了避免麻煩,我把他編譯了進去.
Multi-device support (RAID and LVM)
Cryptography support (CryptoAPI)
這兩個大項全部選擇N,因為在個人用PC上,及少牽涉到這兩項,如果你真的有RAID設備或者LVM,那么就自己摸索著配置一下吧.
Networking options
這一大項中,只需要把下列項目編譯進內核:
Packet socket :mmapped IO
TCP/IP networking
對于IP:advanced router這項,如果你想重點把babylinux用做靜態(tài)路由軟件,那么把這項編譯進去,而對于network packet filtering (replaces ipchains)這一項,沒有必要編譯進去了,因為busybox沒有提供iptables工具來設置包過濾防火墻.同樣,unix domain sockets這項也不必選擇,只有運行X的情況下才需要選這項.
Telephony Support 選擇N
ATA/IDE/MFM/RLL support
選擇Y,然后下面的'IDE,ATA and ATAPI Block Devices'按鈕就被激活
下面幾項請選擇Y,其余都可以是N.
Enhanced IDE/MFM/RLL disk/cdrom/tape/floppy support
Include IDE/ATA-2 DISK support
Auto-Geometry Resizing support
Include IDE/ATA CDROM support
如果你的內核要運行在一臺很老的pentium或486上,請把CMD640 chipset bugfix/support編譯進去,因為那時候主板的CMD640 IDE控制芯片大多有莫名其妙的BUG,把這項編譯進去會修復這個bug.