Linux驅(qū)動(dòng)開(kāi)發(fā)之Hello World
掃描二維碼
隨時(shí)隨地手機(jī)看文章
貌似在學(xué)習(xí)一種新的編程語(yǔ)言的時(shí)候,通常會(huì)使用一個(gè)Hello world!作為一個(gè)入門(mén),今天我們也用一個(gè)類(lèi)似的程序,打開(kāi)進(jìn)入linux內(nèi)核空間的大門(mén)。
要在linux的內(nèi)核空間中進(jìn)行程序設(shè)計(jì),需要準(zhǔn)備哪些準(zhǔn)備工作呢?首先你要有一個(gè)linux的開(kāi)發(fā)環(huán)境:一個(gè)linux操作系統(tǒng)、一套linux內(nèi)核源代碼和編譯環(huán)境。
操作系統(tǒng)目前可選擇的太多了,各種各樣的發(fā)行版應(yīng)有盡有,你可以在distrowatch.com看到各個(gè)版本的信息。目前最為流行的是ubuntu,采用debian包管理,安裝配置軟件非常簡(jiǎn)便,建議使用官方發(fā)布的版本,開(kāi)發(fā)linux2.6內(nèi)核下的驅(qū)動(dòng)最好使用這個(gè)版本,如果是開(kāi)發(fā)linux2.4下的驅(qū)動(dòng)建議采用redhat企業(yè)版。
linux內(nèi)核源代碼沒(méi)有選擇,你的驅(qū)動(dòng)運(yùn)行于那個(gè)linux系統(tǒng)下,你就要獲得這個(gè)linux系統(tǒng)的內(nèi)核源碼。例如我們開(kāi)發(fā)的驅(qū)動(dòng)要運(yùn)行在X86 ubuntu上,你就要獲得這個(gè)ubuntu的源碼,版本也要一致。驅(qū)動(dòng)要運(yùn)行在一個(gè)嵌入式linux單板上,你就要拿到這個(gè)單板所運(yùn)行的linux的源碼。絕對(duì)不能在kernel.org上隨便下載一個(gè)內(nèi)核,如果內(nèi)核不匹配可能會(huì)出現(xiàn)意想不到的問(wèn)題,例如什么printk overflow問(wèn)題。
編譯環(huán)境問(wèn)題比較復(fù)雜,跟開(kāi)發(fā)環(huán)境有很大關(guān)系,你的驅(qū)動(dòng)如果要運(yùn)行在嵌入式平臺(tái)上,你就需要一套交叉工具鏈,所謂的交叉工具鏈其實(shí)就是能在你的X86 CPU上編譯出運(yùn)行于mips、arm等其他架構(gòu)上的程序的一套編譯器。假定驅(qū)動(dòng)運(yùn)行在X86平臺(tái)上,我們需要安裝gcc編譯器和make utility,如何安裝這里就不一一贅述了,請(qǐng)google查一下。
另外,還有一個(gè)使用什么工具編輯程序的問(wèn)題,當(dāng)然是vi或者gvim了,我在后續(xù)的文章中介紹如何使用vi。
準(zhǔn)備工作做好了,我們開(kāi)始寫(xiě)第一個(gè)內(nèi)核程序(假定開(kāi)發(fā)環(huán)境是ubuntu linux2.6,以前一直在linux2.4下開(kāi)發(fā)linux驅(qū)動(dòng),轉(zhuǎn)到2.6下才發(fā)現(xiàn)很多東東都變了)。我們Hello world內(nèi)核程序有兩個(gè)文件組成,一個(gè)Makefile,另一個(gè)是test.c。
test.c內(nèi)容如下:
1 #include
2 #include
3 #include
4
5 static int __init eth_init(void)
6 {
7 printk("init module/n");
8 return 0;
9 }
10 static void __exit eth_exit(void)
11 {
12 printk("exit modules/n");
13 }
14
15 module_init(eth_init);
16 module_exit(eth_exit);
Makefile內(nèi)容如下:
PWD = $(shell pwd)
KERNEL_SRC = /usr/src/linux-source-2.6.15/
obj-m := test.o
module-objs := test.o
all:
$(MAKE) -C $(KERNEL_SRC) M=$(PWD) modules
clean:
rm *.ko
rm *.o
注意千萬(wàn)不能從我這篇blog中通過(guò)復(fù)制粘貼Makefile內(nèi)容去創(chuàng)建Makefile文件!!!我要下載源碼test.tgz.zip,原文件名是test.tgz,請(qǐng)去掉后面的zip后解壓。
在編譯源碼之前需要修改一下Makefile中的KERNEL_SRC,使其指向你的linux內(nèi)核源碼。修改完成后,在test.c和Makefile所在的目錄下運(yùn)行make,如果看到類(lèi)似輸出
make -C /usr/src/linux-source-2.6.15/ M=/home/vmeth modules
make[1]: Entering directory `/usr/src/linux-source-2.6.15''''''''
CC [M] /home/vmeth/test.o
Building modules, stage 2.
MODPOST
CC /home/vmeth/test.mod.o
LD [M] /home/vmeth/test.ko
make[1]: Leaving directory `/usr/src/linux-source-2.6.15''''''''
OK,一個(gè)內(nèi)核程序編譯出來(lái)了,目錄下多了一個(gè)test.ko。出現(xiàn)/usr/bin/ld: crt1.o: No such file: No such file or directory.這個(gè)錯(cuò)誤怎么辦?這個(gè)問(wèn)題的原因是usr/lib/目錄下缺少crt1.o。解決方法:安裝libc6-dev。另外在編譯這個(gè)程序之前最好先把內(nèi)核編譯一下(在內(nèi)核源碼所在的目錄下執(zhí)行make menuconfig,什么都不要修改退出,退出時(shí)會(huì)提示你是否要保存配置,選擇保存,然后執(zhí)行make,整個(gè)make過(guò)程需要很長(zhǎng)時(shí)間,----我在make menuconfig時(shí),好像提示缺少ncurses庫(kù),安裝上libncurses就OK了)。
下一步就是運(yùn)行這個(gè)程序了,在root用戶下執(zhí)行insmod ./test.ko。沒(méi)有錯(cuò)誤提示的話,這個(gè)程序就被執(zhí)行了,運(yùn)行一下dmesg,最后一行是不是可以看到init module字樣。執(zhí)行rmmod test;再運(yùn)行dmesg,最后一行又出現(xiàn)了exit modules。這個(gè)內(nèi)核程序目前只是打印了兩行字,后續(xù)我將會(huì)擴(kuò)充這個(gè)程序,完成一些有用的功能。
Makefile文件分析:
obj-m := hello.o 代表了我們要構(gòu)造的模塊名為hell.ko,make 會(huì)在該目錄下自動(dòng)找到hell.c文件進(jìn)行編譯。如果 hello.o是由其他的源文件生成(比如file1.c和file2.c)的,則在下面加上(注意紅色字體的對(duì)應(yīng)關(guān)系):
hello-objs := file1.o file2.o ......
$(MAKE) -C $(KERNELDIR) M=$(PWD) modules
其中 -C $(KERNELDIR) 指定了內(nèi)核源代碼的位置,其中保存有內(nèi)核的頂層makefile文件。
M=$(PWD) 指定了模塊源代碼的位置
modules目標(biāo)指向obj-m變量中設(shè)定的模塊。