http://blog.linux.org.tw/~jserv/archives/001871.htmlJanuary 11, 2007透過 User-Mode-Linux 來學習核心設計 (1)因為收到不少網友來信指教,小弟決定整理過去的心得與筆記,分享如何透過 [User-Mode Linux] (以下簡寫 UML,注意該術語與軟體工程的 Unified Modeling Language 無任何關係) 來學習核心設計、體驗修改並驗證的新途徑。UML 顧名思義是將 Linux Kernel 移植到 user-space,如此一來,就可將這個修改的 "Kernel" 當作一般的 Linux process 來執行,這有什麼好處呢?簡單來說,至少有以下應用:
知名的 [LEAF] (Linux Embedded Appliance Firewall) 計劃的文件中,有份 [Developing and using LEAF in a virtual environment] 開發文獻很值得一看,不僅克服不必要的微調動作,還能兼具設計與測試雙軌運作的趨勢,透過建構 UML 作為基礎的開發環境,並讓 [LEAF] 能在其上開發。 [UserModeLinux for KNOPPIX] 很有創意也相當實用的途徑,透過 UML 來測試不同的 [Knoppix Linux LiveCD 配置組態,可以避免無謂的燒錄測試,而且可以透過 UML 搭配特定 debugger 與 benchmark profiling 來進行微調。 無獨有偶,[GPE] (GPE Palmtop Environment) 官方網站文件中也提及這類 cross-development 的方式,請見 [GPE: UML GTK 2 cross compiling environment],如此一來,ARM target 的 layout 就可以在設計的初期界定了,相當方便的途徑,當然,隨著 [qemu] 具備 ARM system emulation 後,使用 UML 的途徑就稍微彆扭些,但核心的想法是雷同的。 也因此,Linux Kernel 2.6 source tree 正式收錄 UML,我們也不再需要跟一堆 patch 搏鬥,不過 UML 經歷過許多重大修改,這使得不同時期的 HOW-TO 文件幾乎都有重大出入 (早於 2004 年的文件就不需要太拘泥細節了,畢竟設定方式更動過大),對一般使用者來說,徒增困擾而難以入門,這也是為何小弟試圖撰寫這系列筆記的原因,需要留意的 是,因為 Linux Kernel 總是捨棄相容性包袱,所以難確保有一致的文件,也因此,本系列文件基本上以新版修正舊版缺失的方式撰寫,對系統造成的損失或不便處,請多見諒,但歡迎 [來信指教]。以下是本系列文件著墨之處 (不依據時間順序):
![]() 而 UML 則是在宿主系統上另執行一核心,如下圖: ![]() 由此可見,UML 提供的虛擬機器,允許模擬出比實體裝置更豐富的配置方式,而且 UML 所使用的檔案系統對宿主 Linux 來說也不過只是單純的檔案,一切都好比置身於保護的 sandbox (原文的意思就是「貓沙盒」,給調皮的貓咪一個自得其樂卻不傷害家具的器具,引申為保護的自我封裝機制),經由適當配置,我們大可放心對虛擬機器作任何更 動,而不必擔憂損害到真實的硬體與系統。 相當重要的觀念是:UML 本身就是全功能的 kernel,具備專屬的虛擬環境,對硬體的支援僅仰賴於宿主 Linux 系統,從 UML 內部的觀點來看,基本上支援基本的硬體裝置,如:
$ tar jxvf linux-2.6.19.1.tar.bz2 $ cd linux-2.6.19.1 $ patch -p1 < ../uml-2_6_19_1-compilation.patch $ cp ../uml-dot-config .config $ make menuconfig ARCH=um $ make linux ARCH=um注意到包含 "make" 的那兩行都有 "ARCH=um" 的 build variable,這是建構 UML 一定要加上的,無論是 kernel source 或 external kernel module,都需在 make 時加上,至於原因,看官花點時間看 Linux Kernel 的 Makefile 就可見其端倪。不過,就算不小心忘了加上,請立即執行以下操作: $ make mrproper經過冗長的編譯過程後,會發現有個新檔案被建立,即 "linux",但別急著執行,因為我們需要合用的 rootfs,原本這是苦差事,但 Debian/Ubuntu 有個強大的工具 [debootstrap] 可輕易生成具備 Debian/Ubuntu base 的 rootfs,以下是套件簡介:
#!/bin/sh BASE_DIR=`pwd` # --- Modified as you need --- TARGET_DIR=$BASE_DIR/ext2fs ETC_SAMPLE=etc_sample.tar.bz2 ROOTFS_FILE=ubuntu-root # Create rootfs (400Mb) echo "Creating root file system..." rm -f $ROOTFS_FILE dd if=/dev/zero of=$ROOTFS_FILE bs=1024K count=400 if [ ! -f $ROOTFS_FILE ]; then echo "Error: creation of image file fails." exit fi yes y | mkfs.ext2 $ROOTFS_FILE mkdir -p $TARGET_DIR mount -o loop $ROOTFS_FILE $TARGET_DIR # Make use of Debian's debootstrap tool to construct Ubuntu Dapper (6.10) base echo "Invoking debootstrap..." debootstrap --arch i386 dapper \ $TARGET_DIR \ http://archive.ubuntulinux.org/ubuntu # Extract sample configure files if [ -f $ETC_SAMPLE ]; then cd $TARGET_DIR tar jcvf $ETC_SAMPLE cd $BASE_DIR fi # mknod for ubd0 (specific to UML) if [ -d $TARGET_DIR/dev ]; then cd $TARGET_DIR/dev mknod --mode=660 ubd0 b 98 0 chown root:disk ubd0 cd $BASE_DIR else echo "Error: debootstrap fails." exit fi # Finish sync umount ext2fs echo "Done" echo "Please assign the rootfs: " $ROOTFS_FILE沒有意外的話,以 root 權限執行以上 script 後,[debootstrap] 會幫我們建構具備 Ubuntu Dapper (6.10) base 的 rootfs,其容量為 400 Mb。注意 [debootstrap] 下載 Debian package 是透過 wget,如果在防火牆的環境中,請將相關設定準備好。輸出的 "ubuntu-root" 檔案即是 UML 所需的 rootfs,是的,就是一個檔案,損毀的話就再重跑以上流程,無論 UML 內部發生什麼事情,宿主 Linux 仍沒有大影響。 由於後續操作都會大量使用終端機,請確認安裝合用的終端機,如 [rxvt-unicode] 就是一個不錯的選擇。我們將剛剛產生的 "ubuntu-root" 檔案放在 linux-2.6.19.1 目錄下,於是可進行啟動程序: ./linux ubd0=`pwd`/ubuntu-rootubd 也就是 UML Block Device 之意,對 UML 相當重要,將透過該 device 存取到 rootfs,我們應該會見到類似以下輸出: Checking that ptrace can change system call numbers...OK Checking syscall emulation patch for ptrace...OK Checking advanced syscall emulation patch for ptrace...OK Checking for tmpfs mount on /dev/shm...OK Checking PROT_EXEC mmap in /dev/shm/...OK Checking for the skas3 patch in the host: - /proc/mm...not found - PTRACE_FAULTINFO...not found - PTRACE_LDT...not found UML running in SKAS0 mode Checking that ptrace can change system call numbers...OK Checking syscall emulation patch for ptrace...OK Checking advanced syscall emulation patch for ptrace...OK [42949372.960000] Linux version 2.6.19.1 ([EMAIL PROTECTED]) (gcc version 4.1.2 20070106 (prerelease) (Ubuntu 4.1.1-21ubuntu7)) #4 Wed Jan 10 20:53:03 CST 2007 [42949372.960000] Built 1 zonelists. Total pages: 8128 [42949372.960000] Kernel command line: ubd0=/opt/src/ubuntu-root root=98:0 [42949372.960000] PID hash table entries: 128 (order: 7, 512 bytes) [42949372.960000] Dentry cache hash table entries: 4096 (order: 2, 16384 bytes) [42949372.960000] Inode-cache hash table entries: 2048 (order: 1, 8192 bytes) [42949372.960000] Memory: 30100k available [42949373.230000] Mount-cache hash table entries: 512 ... 省略 ...預設 root 是不需要密碼,我們試著登入並結束系統: Ubuntu 6.06 LTS uml tty0 uml login: root [42949456.230000] line_ioctl: tty0: unknown ioctl: 0x5603 Last login: Wed Jan 10 16:47:59 2007 on tty0 Linux uml 2.6.19.1 #4 Wed Jan 10 20:53:03 CST 2007 i686 GNU/Linux [EMAIL PROTECTED]:~# halt Broadcast message from [EMAIL PROTECTED] (tty0) (Wed Jan 10 16:48:40 2007): The system is going down for system halt NOW! * INIT: Switching to runlevel: 0 * INIT: Sending processes the TERM signal hwclock is unable to get I/O port access: the iopl(3) call failed. * Stopping kernel log... [ ok ] * Stopping system log... [ ok ] * Terminating any remaining processes... [ ok ] * Unmounting remote filesystems... [ ok ] * Deconfiguring network interfaces... [ ok ] * Unmounting local filesystems... [ ok ] * Deactivating swap... [ ok ] * Will now halt [42949503.180000] System halted.粗體字是我們鍵入的部份,當有 ioctl 抱怨錯誤訊息,請忽略,因為 UML 並未完全實做該功能,這不影響我們的進行。這短暫的過程我們見到 UML 虛擬機器的啟動與終結,佛語常說,一切萬物,自有緣起緣滅之時,且讓我們來探究 UML 的因果與深入設計,進而體驗 Linux Kernel 的美妙。 (待續) 由 jserv 發表於 January 11, 2007 12:59 AM 迴響
兩張圖重複了,第二張圖應該換成 uml_part_2.png 由 yaocl 發表於 January 11, 2007 09:28 AMcreate-uml-rootfs 有一點小 typo。 arch/i386/lib/semaphore.S:34: Error: unknown pseudo-op: `.cfi_startproc' make kernel error . My linux is Redhat9. Why? 由 wenhsuan 發表於 January 26, 2007 03:40 PMTo wenhsuan, Please do "make menuconfig ARCH=um", and disable the option "Compile the kernel with frame unwind information" in "Kernel hacking" section. Thanks for your feedback. 由 jserv 發表於 January 27, 2007 05:33 PMthanks, I do make successful as you said. 由 wenhsuan 發表於 January 30, 2007 12:21 PMYour introduction of UML linux is very good. I will give a try. can I use UML method to treat REDHAT 8 in CentOS4.4? 由 Hugo 發表於 February 1, 2007 02:46 PM發表迴響
![]() |