对一个简单的模块:
lzel@lzel-laptop:~/works/kernel/DeviceDrivers/1/modules$ cat simple.c
#include <linux/init.h>
#include <linux/module.h>
//#include <linux/kernel.h>
MODULE_LICENSE("Dual BSD/GPL");
static int hello_init(void)
{
printk("<1>Hello module init./n");
return 0;
}
static void hello_exit(void)
{
printk("<1>Goodbye module exit./n");
}
module_init(hello_init);
module_exit(hello_exit);
MODULE_AUTHOR("lizeliang");
MODULE_DESCRIPTION("a simple module");
MODULE_ALIAS("hello");
编译过程首先会到内核源码目录下,读取顶层的Makefile文件,然后再返回模块源码所在目录。
<一>比较简单的写法,有人认为这不是正真的Makefile
lzel@lzel-laptop:~/works/kernel/DeviceDrivers/1/modules$echo obj -m :=simple.o>Makefile //重定向显示内容到Makefile文件
lzel@lzel-laptop:~/works/kernel/DeviceDrivers/1/modules$chmod +x Makefile //为什么要这样,我也没弄清楚,谁知道补充一下;其实在我的机子上不加这一句也是可以的
lzel@lzel-laptop:~/works/kernel/DeviceDrivers/1/modules$ make -C /usr/src/linux-headers-2.6.24-15-generic M=$PWD modules //M=$PWD M代表要编译的模块的绝对路径 #make -C linux-headers-`uname -r`/dir 表示make将调到/dir下去执行make
<二>对陈老师《Linux操作系统原理与应用》上2.4内核下的一点改进:
lzel@lzel-laptop:~/works/kernel/DeviceDrivers/1/modules$ cat Makefile
obj-m +=simple.o
all:
make -C /usr/src/linux-headers-`uname -r` M=`pwd` modules #编译
clean:
make -C /usr/src/linux-headers-`uname -r` M=`pwd` clean #清理
lzel@lzel-laptop:~/works/kernel/DeviceDrivers/1/modules$ make
lzel@lzel-laptop:~/works/kernel/DeviceDrivers/1/modules$ make clean
<三>IBM网站上
# Makefile2.6
ifneq ($(KERNELRELEASE),) #KERNELRELEASE是在内核源码的顶层Makefile中定义的一个变量,在第一次读取执行此Makefile 时,KERNELRELEASE没有被定义,所以make将读取执行else之后的内容。
#kbuild syntax. dependency relationshsip of files and target modules are listed here.
#mymodule-objs := file1.o file2.o ... #表示mymoudule.o 由file1.o与file2.o 等连接生成。如果有多个文件,需要把这一句加上
obj-m := mymodule.o #表示编译连接后将生成mymodule.o模块。
else
PWD := $(shell pwd) #执行shell命令,把当前路径赋值给PWD
KVER ?= $(shell uname -r) #执行shell命令,将当前系统内核版本号赋值给KVER
KDIR := /lib/modules/$(KVER)/build
all:
$(MAKE) -C $(KDIR) M=$(PWD) #与"$(MAKE) -C $(KDIR) SUBDIRS =$(PWD)"的作用是等效的,SUBDIRS是较老的使用方法
clean:
rm -rf .*.cmd *.o *.mod.c *.ko .tmp_versions
endif
执行路径:如果make的目标是clean,直接执行clean操作,然后结束。当make的目标为all时,-C $(KDIR) 指明跳转到内核源码目录下读取那里的Makefile;M=$(PWD) 表明然后返回到当前目录继续读入、执行当前的Makefile。当从内核源码目录返回时,KERNELRELEASE已被被定义,kbuild也被启动去解析kbuild语法的语句,make将继续读取else之前的内容。else之前的内容为kbuild语法的语句, 指明模块源码中各文件的依赖关系,以及要生成的目标模块名。