前提,一般來說內(nèi)核代碼的錯(cuò)誤可能會(huì)引起一個(gè)用戶進(jìn)程的死亡,或者整個(gè)系統(tǒng)的癱瘓,更嚴(yán)重的后果,可能導(dǎo)致磁盤損傷~因此建議最好有一臺(tái)實(shí)驗(yàn)機(jī)進(jìn)行系統(tǒng)的測(cè)試。
第一個(gè)內(nèi)核模塊(Hello World模塊)
View Code
#includelinux/init.h>
#includelinux/module.h>
MODULE_LICENSE("Dual BSD/GPL");
static __init int hello_init(void)
{
//printk函數(shù)在內(nèi)核中定義對(duì)模塊可用,內(nèi)核需要自已的打印涵數(shù)
//因?yàn)樗孔砸堰\(yùn)行,而沒有相應(yīng)的庫函數(shù)。
//模塊能夠調(diào)用printk是因?yàn)閕nsmod加載了之后,模塊被鏈接到內(nèi)核
//因些可調(diào)用內(nèi)核的公用符號(hào),KERN_ALERT是消息的優(yōu)先級(jí)
printk(KERN_ALERT"HELLO WORLD\n");
return 0;
}
static __exit void hello_exit(void)
{
printk(KERN_ALERT"GoodBye\n");
}
module_init(hello_init);
module_exit(hello_exit);
在這個(gè)模塊中定義了兩個(gè)函數(shù),一個(gè)在模塊加載到內(nèi)核時(shí)調(diào)用(hello_init),另一個(gè)在從內(nèi)核將模塊移出時(shí)調(diào)用(hello_exit);在上面的代碼中,module_init與module_exit是兩個(gè)內(nèi)核宏定義,用于告訴內(nèi)核從哪里啟動(dòng),從哪里退出,MODULE_LICENSE宏用于聲明模塊是遵守某個(gè)自由許可證的,否則內(nèi)核加載時(shí)會(huì)出現(xiàn)警告。
好了,現(xiàn)在可以對(duì)上面這個(gè)程序進(jìn)行相應(yīng)的測(cè)試,在測(cè)試之前必須要編寫相應(yīng)的Makefile文件,模塊的編譯與普通程序的編譯是不同的
Makefile文件
View Code
#makefile for hello world
# KERNELRELEASE是在內(nèi)核源碼中定義的第一個(gè)變量
ifneq ($(KERNELRELEASE),) #判斷變量是否為空(第一次執(zhí)行時(shí)沒有定義)
#沒定義時(shí)執(zhí)行else語句
obj-m := HelloWorld.o#表明有一個(gè)模塊要從目錄文件HelloWorld.o建立,建立之后將其
#命名為HelloWorld.ko
#如果有一個(gè)模塊名為module.ko,來自于兩個(gè)源文件,假設(shè)為file1.c與file2.c
#則應(yīng)該這樣 obj-m := module.o
# module-objs:=file1.o file2.o
else
KDIR:=/lib/modules/$(shell uname -r)/build
all:
#當(dāng)make的目標(biāo)為all時(shí),-C $(KDIR)跳到內(nèi)核源碼目錄下讀取Makefile
#M=$(PWD)表示返回當(dāng)前目錄繼續(xù)讀取,執(zhí)行當(dāng)前的Makefile,當(dāng)再次執(zhí)行時(shí)
#$(KERNELRELEASE)已經(jīng)定義,make將讀取else之前的內(nèi)容
make -C $(KDIR) M=$(PWD) modules
clean:
rm -rf *.ko *.o *.mod.o *.mod.c *.symvers
endif
相應(yīng)的解釋如上
開如編譯內(nèi)核:必須要是超級(jí)用戶
在當(dāng)前路徑下輸入make
編譯完成后,輸入insmd HelloWorld.ko進(jìn)行內(nèi)核的加載,使用dmesg |tail可以查看內(nèi)核的輸出信息。
移除內(nèi)核采用rmmod HelloWorld 相應(yīng)的使用dmesg|tail可以看到打印出GoodBye
Printk可能沒有輸出到屏幕上,這與KERN_ALERT的優(yōu)先級(jí)有關(guān),說明還不夠高,內(nèi)核輸出的內(nèi)容實(shí)際在/var/log/kern.log中,可以vim /var/log/kern.log查看。
內(nèi)核模塊與應(yīng)用程序的不同:
1:應(yīng)用程序運(yùn)行后就會(huì)處理相應(yīng)的任務(wù),而內(nèi)核模塊注冊(cè)后是用來服務(wù)于將來請(qǐng)求,并且初始化函數(shù)加了__init之后,調(diào)用完后,內(nèi)存空間立即釋放。
2:應(yīng)用程序終止時(shí)可以不用負(fù)責(zé)回收資源,由操作系統(tǒng)來維護(hù),但內(nèi)核模塊在除時(shí)必須釋放資源。
3:應(yīng)用程序可以調(diào)用相應(yīng)的庫函數(shù),而內(nèi)核模塊能夠調(diào)用的則只要內(nèi)核中輸入的那些函數(shù)。在內(nèi)核模塊的編程中,源文件不應(yīng)當(dāng)包括通常的頭文件,但也有例外,如stdarg.h>等少部分頭文件是僅有的例外。
4:錯(cuò)誤的處理方式不同,在應(yīng)用程序中段錯(cuò)誤,可以由相應(yīng)的調(diào)試程序進(jìn)行檢查更改,但內(nèi)核模塊中,段錯(cuò)誤,如果不終止整個(gè)系統(tǒng)的話,就會(huì)終止當(dāng)前進(jìn)程。
用戶空間與內(nèi)核空間:
應(yīng)用程序在用戶空間運(yùn)行,而內(nèi)核模塊是在內(nèi)核空間內(nèi)運(yùn)行的。每種模式都有它自已的內(nèi)存映射,它自已的地址空間。
內(nèi)核與當(dāng)前進(jìn)程的關(guān)系:
內(nèi)核模塊做的大部份動(dòng)作是代表一個(gè)特定進(jìn)程的,內(nèi)核代碼可以引用當(dāng)前進(jìn)程,通過存取全局項(xiàng)current,它在asm/cuurent.h>定義:
#define current get_current()//通過這個(gè)宏定義可以獲取指向task_struct的任務(wù)指針
內(nèi)核代碼可以通過current來使用進(jìn)程特定的信息。
內(nèi)核符號(hào)表:
內(nèi)核模塊在加載時(shí)通過查找內(nèi)核符號(hào)表來解決未定義的符號(hào),內(nèi)核符號(hào)表包涵了全局內(nèi)核項(xiàng)的地址,當(dāng)加載一個(gè)模塊時(shí),模塊中輸出的符號(hào)也將成為內(nèi)核符號(hào)表的一部分。
模塊的輸入符號(hào)通常采用以下兩種形式:
EXPORT_SYMBOL(name)
EXPORT_SYMBOL_GPL(name)
上面的宏定義中的任何一個(gè)使得給定的符號(hào)在模塊外使用,_GPL版本的宏定義只能使符號(hào)對(duì)_GPL許可的模塊可用。
版本依賴:
模塊代碼一定要為每個(gè)它要連接的內(nèi)核版本重新編譯,在模塊編譯的過程中,其中一步是到當(dāng)前的內(nèi)核對(duì)讀取Makefile文件,在編譯的過程中會(huì)采用內(nèi)核樹中的文件(vermagic.o)連接你的模塊,在這個(gè)文件里面有許多有關(guān)內(nèi)核的信息,包括版本...
模塊參數(shù):
模塊參數(shù)由insmod與modprobe在加載時(shí)指定。
對(duì)前面的HelloWorld.c進(jìn)行修改如下:
在終端上輸入
Make
Insmod HelloWorld.ko who=”test” num=10
Dmesg|tail -3
即可以看到。
聲明數(shù)組參數(shù)時(shí)采用module_param_array(name,type,num,perm)
Name是數(shù)組的名字,type是數(shù)組元素的類型,num是數(shù)組無數(shù)的個(gè)數(shù),perm是權(quán)限
附:insmod
Insmod將內(nèi)核模塊加載到內(nèi)存中,它依賴一個(gè)在kernel/module.c中定義的系統(tǒng)調(diào)用,函數(shù)sys_init_module分配內(nèi)核內(nèi)存來存放模塊,它接著copy模塊的代碼段到這塊內(nèi)存區(qū),借助內(nèi)核符號(hào)表來解決模塊中的內(nèi)核引用,并且調(diào)用模塊的初始經(jīng)函數(shù)來啟動(dòng)所有的東西。
Modprobe工具也用來加載一個(gè)內(nèi)核模塊到內(nèi)存,與insmod不同的是,它會(huì)查看要加載的模塊,看看是否引用了當(dāng)前內(nèi)核沒有定義的符號(hào)。如要有,它會(huì)在當(dāng)前搜索路徑下尋找其他模塊,看是否這個(gè)符號(hào)的定義,如果有,則將這個(gè)模塊也加載進(jìn)內(nèi)核。
Rmmod用來去除內(nèi)核模塊,如果內(nèi)核認(rèn)為模塊還在使用,或者內(nèi)核配置了不允許去除模塊,則模塊的卸載會(huì)失敗。
Lsmod例舉出當(dāng)前系統(tǒng)中加載的所有模塊列表。
內(nèi)核模塊編程中函數(shù)通常聲明為靜態(tài)的,是因?yàn)樗鼈儾粫?huì)在文件之外可見。