1. 模块的编译
驱动编译分为静态编译和动态编译。静态编译即为将驱动直接编译进内核,动态编译即为将驱动编译成模块。
而动态编译又分为两种:
(1)内部编译
在内核源码目录内编译
(2)外部编译
在内核源码的目录外编译
2. 具体编译过程分析
注:本次编译是外部编译,使用的内核源码是Ubuntu 的源代码,而非开发板所用linux 3.14内核源码,运行平台为X86。
对于一个普通的linux
设备驱动模块,以下是一个经典的makefile
代码,使用下面这个makefile
可以完成大部分驱动的编译,使用时只需要修改一下要编译生成的驱动名称即可。只需修改obj-m
的值。
ifneq ($(KERNELRELEASE),)
obj-m:=hello.o
else
KDIR := /lib/modules/$(shell uname -r)/build
PWD:=$(shell pwd)
all:
make -C $(KDIR) M=$(PWD) modules
clean:
rm -f *.ko *.o *.symvers *.cmd *.cmd.o
endif
2.1 makefile 中的变量
先说明以下makefile中一些变量意义:
(1)KERNELRELEASE:在内核源码树的Makefile
中定义,在当前的Makefile
中,它的值为空
(2)$(shell uname-r):获得当系统的Linux
内核版本
(3)shell pwd:取得当前工作路径
(4)KDIR:制定当前Linux
操作系统源代码路径,即编译生成的模块是在当前系统中使用。如果想将你写的模块,用在你的开发板上运行的Linux
系统中,只需在KDIR
变量中指定你开发板Linux
系统源码树的路径。
关于linux源码的目录有两个,分别为:
/lib/modules/$(shell uname -r)/build
/usr/src/linux-header-$(shell uname -r)/
但如果编译过内核就会知道,usr
目录下那个源代码一般是我们自己下载后解压的,而lib
目录下的则是在编译时自动copy
过去的,两者的文件结构完全一样,因此有时也将内核源码目录设置成/usr/src/linux-header-$(shell uname -r)/
。关于内核源码目录可以根据自己的存放位置进行修改。
(5)make -C $(LINUX_KERNEL_PATH) M=$(CURRENT_PATH) modules
这就是编译模块了:
1)首先改变目录到-C
选项指定的位置(即内核源代码目录),其中保存有内核的顶层makefile
;
2)M= 选项让该makefile在构造modules目标之前返回到模块源代码目录;然后,modueles目标指向obj-m变量中设定的模块;在上面的例子中,我们将该变量设置成了hello.o
。
2.2 make 的的执行步骤
(1)在模块的源代码目录下执行make,此时,宏“KERNELRELEASE”【内核源码树的Makefile会定义】没有定义,因此进入else;
(2)记录内核路径KDIR和当前工作目录PWD;
由于make 后面没有目标,所以make会在Makefile中的第一个不是以.开头的目标作为默认的目标执行。默认执行all这个规则。
(3)make -C $(KDIR) M=$(PWD) modules
-C 进入到内核的目录执行Makefile ,在执行的时候KERNELRELEASE就会被赋值,M=$(PWD)表示返回当前目录,再次执行makefile,modules 编译成模块的意思
所以这里实际运行的是:
make -C /lib/modules/2.6.13-study/build M=/home/fs/code/1/module/hello/ modules
(4)再次执行该makefile
,KERNELRELEASE
就有值了,就会执行obj-m:=hello.o
obj-m
表示会将hello.o
目标编译成.ko
模块;它告诉linux
源码树顶层Makefile
是动态编译(编译成模块)而不是编译进内核(obj-y
),linux
源码树顶层Makefile
会根据hello.o
找到hello.c
文件。
可以看出make在这里一共调用了3次
1)make
2)linux内核源码树的顶层makedile调用,产生.o
文件
3)linux内核源码树makefile调用,把.o
文件链接成.ko
文件
2.3 编译多文件
若有多个源文件,则采用如下方法:
obj-m := hello.o
hello-objs := file1.o file2.o file3.o
3. 内部编译简单说明
如果把hello
模块移动到内核源代码中。例如放到/usr/src/linux/driver/
中, KERNELRELEASE
就有定义了。
在/usr/src/linux/Makefile
中有
KERNELRELEASE=$(VERSION).$(PATCHLEVEL).$(SUBLEVEL)$(EXTRAVERSION)$(LOCALVERSION)
这时候,hello
模块也不再是单独用make
编译,而是在内核中用make modules
进行编译,此时驱动模块便和内核编译在一起。