新聞中心

        EEPW首頁 > 嵌入式系統 > 設計應用 > kbuild系統-內核模塊的編譯

        kbuild系統-內核模塊的編譯

        作者: 時間:2012-07-30 來源:網絡 收藏

        Linux是一種單體,但是通過動態加載的方式,使它的開發非常靈活方便。那么,它是如何的呢?我們可以通過分析它的Makefile入手。以下是一個簡單的hello內核的Makefile.

        本文引用地址:http://www.104case.com/article/148696.htm

        ifneq ($(KERNELRELEASE),)

        obj-m:=hello.o

        else

        KERNELDIR:=/lib/modules/$(shell uname -r)/build

        PWD:=$(shell pwd)

        default:

        $(MAKE) -C $(KERNELDIR) M=$(PWD) modules

        clean:

        rm -rf *.o *.mod.c *.mod.o *.ko

        endif

        當我們寫完一個hello,只要使用以上的makefile。然后make一下就行。

        假設我們把hello模塊的源代碼放在/home/study/prog/mod/hello/下。

        當我們在這個目錄運行make時,make是怎么執行的呢?

        LDD3第二章第四節“和裝載”中只是簡略地說到該Makefile被執行了兩次,但是具體過程是如何的呢?

        首先,由于make 后面沒有目標,所以make會在Makefile中的第一個不是以.開頭的目標作為默認的目標執行。于是default成為make的目標。make會執行$(MAKE) -C $(KERNELDIR) M=$(PWD) modules

        shell是make內部的函數,假設當前內核版本是2.6.13-study,所以$(shell uname -r)的結果是 2.6.13-study

        這里,實際運行的是

        make -C /lib/modules/2.6.13-study/build M=/home/study/prog/mod/hello/ modules

        /lib/modules/2.6.13-study/build是一個指向內核源代碼/usr/src/linux的符號鏈接。

        可見,make執行了兩次。第一次執行時是讀hello模塊的源代碼所在目錄/home/study/prog/mod/hello/下的Makefile。第二次執行時是執行/usr/src/linux/下的Makefile時.

        但是還是有不少令人困惑的問題:

        1.這個KERNELRELEASE也很令人困惑,它是什么呢?在/home/study/prog/mod/hello/Makefile中是沒有定義這個變量的,所以起作用的是else...endif這一段。不過,如果把hello模塊移動到內核源代碼中。例如放到/usr/src/linux/driver/中,KERNELRELEASE就有定義了。

        在/usr/src/linux/Makefile中有

        162 KERNELRELEASE=$(VERSION).$(PATCHLEVEL).$(SUBLEVEL)$(EXTRAVERSION)$(LOCALVERSION)

        這時候,hello模塊也不再是單獨用make,而是在內核中用make modules進行編譯。

        用這種方式,該Makefile在單獨編譯和作為內核一部分編譯時都能正常工作。

        2.這個obj-m := hello.o什么時候會執行到呢?

        在執行:

        make -C /lib/modules/2.6.13-study/build M=/home/study/prog/mod/hello/ modules

        時,make 去/usr/src/linux/Makefile中尋找目標modules:

        862 .PHONY: modules

        863 modules: $(vmlinux-dirs) $(if $(KBUILD_BUILTIN),vmlinux)

        864 @echo ' Building modules, stage 2.';

        865 $(Q)$(MAKE) -rR -f $(srctree)/scripts/Makefile.modpost

        可以看出,分兩個stage:

        1.編譯出hello.o文件。

        2.生成hello.mod.o hello.ko

        在這過程中,會調用

        make -f scripts/Makefile.build bj=/home/study/prog/mod/hello

        而在 scripts/Makefile.build會包含很多文件:

        011 -include .config

        012

        013 include $(if $(wildcard $(obj)/Kbuild), $(obj)/Kbuild, $(obj)/Makefile)

        其中就有/home/study/prog/mod/hello/Makefile

        這時 KERNELRELEASE已經存在。

        所以執行的是:

        obj-m:=hello.o

        關于make modules的更詳細的過程可以在scripts/Makefile.modpost文件的注釋中找到。如果想查看make的整個執行過程,可以運行make -n。

        由此可見,內核的Kbuild龐大而復雜。



        評論


        相關推薦

        技術專區

        關閉
        主站蜘蛛池模板: 鄂托克前旗| 大连市| 五家渠市| 定结县| 广平县| 鹤庆县| 阿城市| 扎赉特旗| 平顶山市| 长乐市| 绵竹市| 山西省| 静海县| 屏南县| 平顶山市| 湘阴县| 邵阳市| 南乐县| 黄梅县| 宣威市| 通城县| 虎林市| 虹口区| 阿瓦提县| 玛纳斯县| 兴仁县| 太白县| 新丰县| 福清市| 祥云县| 吉安市| 隆化县| 珠海市| 二手房| 长乐市| 旺苍县| 富源县| 高台县| 长白| 微博| 丰台区|