天天看點

淺談Linux核心動态子產品的編譯(轉)

Linux核心給開發者提供了靈活的子產品加載方式,開發者可根據需要适當地選擇靜态或動态的方式将子產品加入核心當中。靜态加載的時候我們隻需将驅動檔案添加到對應的檔案夾中并修改相應的Makefile和Kconfig即可,但是通過動态加載子產品的方式,如果隻是編譯一個小的驅動檔案而對核心大動幹戈,那樣編譯的時間不僅長而且修改核心配置檔案也是一件多餘的工作。那麼我們是否可以在不修改核心的前提下單獨編譯它是如何編譯核心的呢?下面我們就通過分析它的Makefile入手簡單介紹一個編譯驅動(子產品)檔案的新方法。以下是 一個簡單的hello核心子產品的Makefile。

ifeq ($(KERNELRELEASE),)

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

        PWD := $(shell pwd)

        modules:

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

        modules_install:

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

        clean:

                rm -rf *.o *~ core .depend .*.cmd *.ko *.mod.c .tmp_versions modules* Module*

        .PHONY: modules modules_install clean

        else

                obj-m := hello.o

        endif

當我們寫完一個hello子產品,隻要使用以上的makefile。然後make一下就行。假設我們把hello子產品的源代碼放在//home/linux/study/char_driver/hello/下。當我們在這個目錄運作make時,但是具體過程是如何的呢?

首先,由于make 後面沒有目标,是以make會在Makefile中的第一個不是以.開頭的目标作為預設的目标執行。于是default成為make的目标。make會執行 $(MAKE) -C $(KERNELDIR) M=$(PWD) modules ,我的ubuntu系統核心版本是2.6.35-22-generic,是以$(shell uname -r)的結果是2.6.35-22-generic (uname –r 是shell的指令),這裡實際運作的是 :

make -C /lib/modules/2.6.35-22-generic/build M=/home/linux/study/char_driver/hello/

modules/lib/modules/2.6.35-22-generic/build是一個指向核心源代碼linux-headers-2.6.35-22-generic的符号連結。可見,make執行了兩次。第一次執行時是讀hello子產品的源代碼所在目錄/home/linux/study/char_driver/hello/下的Makefile。第二次執行時是執行/usr/src/linux-headers-2.6.35-22-generic/下的Makefile時.

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

1、這個KERNELRELEASE也很令人困惑,它是什麼呢?

在/home/linux/study/char_driver/hello/Makefile中是沒有定義這個變量的,是以起作用的是else…endif這一段。不過,如果把hello子產品移動到核心源代碼中。例如放到/usr/src/linux-headers-2.6.35-22-generic/drivers/中, KERNELRELEASE就有定義了。在/usr/src/linux-headers-2.6.35-22-generic/Makefile中有KERNELRELEASE=$(VERSION).$(PATCHLEVEL).$(SUBLEVEL)$(EXTRAVERSION)$(LOCALVERSION) 這時候,hello子產品也不再是單獨用make編譯,而是在核心中用make modules進行 編譯。用這種方式,該Makefile在單獨編譯和作為核心一部分編譯時都能正常工作。

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

make -C /lib/modules/2.6.35-22-generic /build M=/home/linux/study/char_driver/hello/ modules時,make 去/usr/src/linux-headers-2.6.35-22-generic/Makefile中尋找目标modules: .PHONY: modules modules: $(vmlinux-dirs) $(if $(KBUILD_BUILTIN),vmlinux) @echo ' Building modules, stage 2.'; $(Q)$(MAKE) -rR -f $(srctree)/scripts/Makefile.modpost 。

可以看出,分兩個stage: 1.編譯出hello.o檔案。 2.生成hello.mod.o hello.ko 在這過程中,會調用 make -f scripts/Makefile.build obj=/home/linux/study/char_driver/hello/ ,而在 scripts/Makefile.build會包含很多檔案: 011 -include .config 012 013 include $(if $(wildcard $(obj)/Kbuild), $(obj)/Kbuild, $(obj)/Makefile) 其中就有/home/linux/study/char_driver/hello/Makefile 這時 KERNELRELEASE已經存在。 是以執行的是: obj-m:=hello.o。

關于make modules的更詳細的過程可以在scripts/Makefile.modpost檔案的注釋 中找到。如果想檢視make的整個執行過程,可以運作make -n。

繼續閱讀