linux內核動態加載模塊

一、安裝內核模塊:

一般步驟:

(1) 在/usr/src/linux/下運行make menuconfig把需要編譯成模塊的項打上(M),保存並退出。

(2) 運行make modules,這一步將在/usr/src/linux/下生成*.o或*.ko文件。

(3) 運行make modeules_install來安裝,這步會把生成的.o或ko文件拷貝到/lib/modules/`uname -r`/下。

如果你只要編譯某一個或幾個模塊,就可以用下面這個快速的方法:

(1) 找到編譯內核所需要的.config文件。 
在/usr/src/linux/arch目錄下有若干編譯內核所用的配置。選擇我們想要的配置,將它複製到/usr/src/linux目錄下,改名爲.config。 
cp /usr/src/linux/arch/x86/xxconfig /usr/src/linux/.config

(2) 修改.config文件,去掉不用的模塊,加上自己想要的模塊。 
打開.config,有許多XXXX=m的項,這些都是要被編譯爲模塊 的項,因爲我們不希望編譯這些模塊,所以要把XXXX=m的項統統去掉。然後再加上我們想要的模塊,例如將# CONFIG_NTFS_FS is not set 改爲CONFIG_NTFS_FS=m 當然,可以用你熟悉各種工具來做這件事。

(3) 編譯NTFS模塊。 
在/usr/src/linux目錄下運行命令make modules來編譯我們想要的模塊。

(4) 安裝模塊。 
編譯後得到的.o文件在/usr/src/linux/目錄下,手動將它複製到正確的目錄下。 
  例如cp /usr/src/linux/fs/ntfs/ntfs.o /lib/modules/2.2.16-22/fs/
 注意:千萬不能運行命令make modules_install,否則將帶來嚴重的後果,它會刪除你係統中的所有模塊,只安裝剛剛編譯的模塊(ntfs.o)。

二:安裝完成以後,我們就可以加載模塊了:

和linux中加載模塊有關的幾個命令分別如下:
depmod, modprobe, lsmod

先來看看depmod命令:

depmod是一個 用來產生modules.dep和map文件的程序。在modules.dep文件中空白行和以'#'開頭的行將被忽略.depmod通過讀取/lib /modules/version目錄下的每一個模塊來創建一個記錄模塊相依性的列表。這個列表就是/lib/modules/version目錄下的 modules.dep。depmod也會在/lib/modules/version目錄下創建許多map文件,例如 modules.dep,modules.isapnpmap,modules.pcimap,modules.alias這些文件將會被hotplug 用到。

OPTIONS:
-a --all Probe all modules. This option is enabled by default if no
            file names are given in the command-line.
檢查所有的模塊,這個命令是默認的如果你沒有指定模塊名字的話。

-A --quick This option scans to see if any modules are newer than the
                 modules.dep file before any work is done%3

再來看看modprobe命令:

modprobe 命令是根據depmod -a的輸出/lib/modules/version/modules.dep來加載全部的所需要模塊。可以通過modprobe -l來顯示可以當前可以加載的模塊。modprobe 在掛載模塊是不用指定模塊文件的路徑,也不用帶文件的後綴.o 或.ko, 而insmod 需要的是模塊的所在目錄的絕對路徑,並且一定要帶有模塊文件名後綴的(modulefile.o 或modulesfile.ko )。 insmod比較重要的用途是用來測試模塊的正確性,加載一般都是依靠modprobe。

用法:modprobe xxx.ko        #加載某個模塊
modprobe -r xxx.ko     #卸載某個模塊

lsmod:

lsmod 顯示當前加載的所有 模塊,相當於cat /proc/modules,假設你沒有設定開機加載某個模塊,比如ntfs,那麼開機後執行lsmod,列表裏不會有ntfs這個模塊的,這時你再執行 mount -t ntfs xxx後,執行lsmod後列表裏就會有ntfs這個模塊了。
還要注意的是lsmod顯示的是模塊名,而不是別名(alias)。

三、在內核中有一個“Automatic kernel module loading"功能被編譯到了內核中。當用戶嘗試打開某類型的文件時,內核會根據需要嘗試加載相應的模塊。我們來看看驅動程序自動加載是怎麼實現的:

      每一個設備都有Verdon ID, Device ID, SubVendor ID等信息。而每一個設備驅動程序,必須說明自己能夠爲哪些Verdon ID, Deviece

  ID, SubVendor ID的設備提供服務。以PCI設備爲例,它是通過一個pci_device_id的數據結構來實現這個功能的。例如:RTL8139的pci_device_id定義爲:

static struct pci_device_id rtl8139_pci_tbl[] = {

      {0x10ec, 0x8139, PCI_ANY_ID, PCI_ANY_ID, 0, 0, RTL8139 },

      {0x10ec, 0x8138, PCI_ANY_ID, PCI_ANY_ID, 0, 0, RTL8139 },

      ......

在模塊安裝的時候,depmod會根據模塊中的rtl8139_pci_tbl的信息,生成下面的信息,保存到/lib/modules/uname-r /modules.alias文件中,其內容如下:

alias pci:v000010ECd00008138sv*sd*bc*sc*i* 8139too

alias pci:v000010ECd00008139sv*sd*bc*sc*i* 8139too

......

另外在/lib/modules/uname-r /modules.dep文件中還保存這模塊之間的依賴關係,其內容如下:

(這裏省去了路徑信息。)

8139too.ko:mii.ko

在內核啓動過程中,總線驅動程序會會總線協議進行總線枚舉(總線驅動程序總是集成在內核之中,不能夠按模塊方式加載,你可以通過make menuconfig進入Bus  options,這裏面的各種總線,你只能夠選擇Y或N,而不能選擇M.),並且爲每一個設備建立一個設備對象。每一個總線對象有一個kset對象,每一 個設備對象嵌入了一個kobject對象,kobject連接在kset對象上,這樣總線和總線之間,總線和設備設備之間就組織成一顆樹狀結構。當總線驅 動程序爲掃描到的設備建立設備對象時,會初始化kobject對象,並把它連接到設備樹中,同時會調用kobject_uevent()把這個(添加新設 備的)事件,以及相關信息(包括設備的VendorID,DeviceID等信息。)通過netlink發送到用戶態中。在用戶態的udevd檢測到這個 事件,就可以根據這些信息,打開/lib/modules/uname-r /modules.alias文件,根據  alias pci:v000010ECd00008138sv*sd*bc*sc*i* 8139too
 得知這個新掃描到的設備驅動模塊爲8139too。於是modprobe就知道要加載8139too這個模塊了,同時modprobe根據 modules.dep文件發現,8139too依賴於mii.ko,如果mii.ko沒有加載,modprobe就先加載mii.ko,接着再加載 8139too.ko。

試驗

在你的shell中,運行:

# ps aux | grep udevd

# kill -9 25063 然後跟蹤udevd,在shell中運行:

# strace -f /sbin/udevd --daemon
 這時,我們看到udevd的輸出如下:  ......
 close(8)                                = 0
 munmap(0xb7f8c000, 4096)                = 0
 select(7, [3 4 5 6], NULL, NULL, NULL
我們發現udevd在這裏被阻塞在select()函數中。  select函數原型如下:  int select(int nfds, fd_set *readfds, fd_set *writefds, fd_set *exceptfds, struct timeval *timeout);
 第一個參數:nfds表示最大的文件描述符號,這裏爲7(明明是6 ?)。
 第二個參數:readfds爲讀文件描述符集合,這裏爲3,4,5,6.
 第三個參數:writefds爲寫文件描述符集合,這裏爲NULL。
 第四個參數:exceptfds爲異常文件描述符集合,這裏爲NULL。
 第五個參數:timeout指定超時時間,這裏爲NULL。
select函數的作用是:如果readfds中的任何一個文件有數據可讀,或者witefds中的任何一個文件可以寫入,或者exceptfds中的任 何一個文件出現異常時,就返回。否則阻塞當前進程,直到上訴條件滿足,或者因阻塞時間超過了timeout指定的時間,當前進程被喚醒,select返 回。 所以,在這裏udevd等待3,4,5,6這幾個文件有數據可讀,纔會被喚醒。現在,到shell中運行:  # ps aux | grep udevd
 root     27615  ...... strace -o /tmp/udevd.debug -f /sbin/udevd --daemon
 root     27617  ...... /sbin/udevd --daemon
 udevd的進程id爲27617,現在我們來看看select等待的幾個文件:  # cd /proc/27615/fd
 # ls -l
 udevd的標準輸入,標準輸出,標準錯誤全部爲/dev/null.
 0 -> /dev/null
 1 -> /dev/null
 2 -> /dev/null
 udevd在下面這幾個文件上等待。
 3 -> /inotify
 4 -> socket:[331468]
 5 -> socket:[331469]
 6 -> pipe:[331470]
 7 -> pipe:[331470]
由於不方便在運行中插入一塊8139的網卡,因此現在我們以一個U盤來做試驗,當你插入一個U盤後,你將會看到strace的輸出,從它的輸出可以看到 udevd在select返回後,調用了modprobe加載驅動模塊,並調用了sys_mknod,在dev目錄下建立了相應的節點。 
 execve("/sbin/modprobe", ["/sbin/modprobe", "-Q", "usb:v05ACp1301d0100dc00dsc00dp00"...]
 ......
 mknod("/dev/sdb", S_IFBLK|0660, makedev(8, 16)) = 0
 ......
這裏modprobe的參數"usb:v05AC..."對應modules.alias中的某個模塊。  可以通過udevmonitor來查看內核通過netlink發送給udevd的消息,在shell中運行:  # udevmonitor --env 
 然後再插入U盤,就會看到相關的發送給udevd的消息。
 

四、內核模塊加載的配置:

有時候需要一次性加載許多模塊,需要在一個地方統一配置modprobe的選項等,有一個比較重要的文件:/etc/modprobe.conf,在opensuse中,和它有關的還有modprobe.d/文件夾下的許多文件和modprobe.conf.local文件,在 /etc/modprobe.conf裏會include其它所說的文件,一般建議在modprobe.conf.local中修改自己的配置。 /etc/modprobe.conf其實就是用於 寫入模塊的加載命令或模塊的別名的定義等。man modprobe.conf:

alias my-mod really_long_modulename爲模塊定義一個便於使用的別名

options modulename option...在加載模塊時添加選項

install modulename command...使用自己定義的命令去加載指定的模塊,如install fred /sbin/modprobe barney; /sbin/modprobe
              --ignore-install fred" 每次加載fred模塊的時候用的是
 “/sbin/modprobe barney; /sbin/modprobe
              --ignore-install fred”命令

remove modulename command...用自己定義的命令刪除指定的模塊。

include filename 引入其它文件

blacklist modulename 不再加載某個模塊


五、內核模塊開機自動掛載:

對於開機自動掛載模塊,在redhat系統裏,網上說在內核啓動的過程中,init執行/etc/rc.d/rc.sysinit後,啓動內核外掛模塊 時會讀取/etc/modprobe.conf這個文件。在2.4的內核中, 只 要直接修改/etc/modprobe.conf加入install xxx即可。2.6內核則需修改/etc/rc.d/rc.sysinit文件。具體的過程可以看:http://blog.csdn.net /ioriqqe/archive/2009/11/05/4772033.aspx

而在suse裏,可以在root權限編輯/etc/sysconfig/kernel文件,添加需要啓動的模塊。

發佈了24 篇原創文章 · 獲贊 53 · 訪問量 100萬+
發表評論
所有評論
還沒有人評論,想成為第一個評論的人麼? 請在上方評論欄輸入並且點擊發布.
相關文章