Hello.c
#include <linux/init.h>
#include <linux/module.h>
MODULE_LICENSE("GPL");
MODULE_AUTHOR("PD");
static int hello_init(void)
{
printk(KERN_SOH"hello_init \n");
return 0;
}
static void hello_exit(void)
{
printk("hello_exit \n");
return;
}
module_init(hello_init); //insmod
module_exit(hello_exit);//rmmod
Makefile
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 *.mod.o *.symvers *.cmd *.mod.c *.order
endif
insmode时会执行hello_init这个函数。
rmmode时会执行hello_exit这个函数
Makefile会执行三遍。
当敲Make时会进来,
w@w:~/linux_kernel/hello$ sudo make
make -C /lib/modules/5.4.0-104-generic/build M=/home/w/linux_kernel/hello modules
make[1]: Entering directory '/usr/src/linux-headers-5.4.0-104-generic'
CC [M] /home/w/linux_kernel/hello/hello.o
Building modules, stage 2.
MODPOST 1 modules
CC [M] /home/w/linux_kernel/hello/hello.mod.o
LD [M] /home/w/linux_kernel/hello/hello.ko
make[1]: Leaving directory '/usr/src/linux-headers-5.4.0-104-generic'
第一句,ifneq ($(KERNELRELEASE),)。 KERNELRELEASE是内核源码树下面的环境变量,第一次敲Make时,这个变量是没有内容的,为空。 逗号右边没有填写东西,为空。
如果相同,则执行obj-m:hello.o
空对空,相同,则走到else去执行。
KDIR :=/lib/modules/$(shell uname -r)/build KDIR环境变量为内核路径。 := 为赋值语句。 $取字符串,提取(shell uname -r)执行的内容。利用shell执行uname -r这一条命令
PWD :=$(shell pwd) 将执行make的路径赋值给PWD这个变量
all: 第一个目标,会执行一根默认的目标。 执行的是make 等同于make all,不是make clean。 执行all下面的命令
make -C $(KDIR) M=$(PWD) modules make 进入内核build目录下面, modules:驱动文件独立于内核模块进行编译,动态加载,而不是编译到镜像。
执行完后,就不再执行后面的命令。
M=$(PWD) 进入PWD路径下面去寻找.c文件,编译为目标文件hello.o文件
然后,第二次进入。 第二次进来的时候,KERNELRELEASE就已经有内容了,然后走到第2行
obj-m:=hello.o 生成模块化目标文件,依赖hello.o,然后将其他目标文件链接生成hello.ko文件
一共会进入三次。第一次打make去执行; 后面2次是因为内核里面的makefile再次进来两次,第一次内核源码树的Makefile生成hello.o,第二次将hello.o编译为hello.mod.o; 链接文件LD将相关文件链接生成hello.ko
验证
在Makefile中加入$(info XXX)来打印信息
w@w:~/linux_kernel$ cd hello
w@w:~/linux_kernel/hello$ ls
hello.c hello.ko hello.mod hello.mod.c hello.mod.o hello.o Makefile modules.order Module.symvers
w@w:~/linux_kernel/hello$ sudo make clean
[sudo] password for w:
rm -f *.ko *.o *.mod.o *.symvers *.cmd *.mod.c *.order
w@w:~/linux_kernel/hello$ ls
hello.c hello.mod Makefile
w@w:~/linux_kernel/hello$ cat Makefile
ifneq ($(KERNELRELEASE),)
$(info "2nd")
obj-m:=hello.o
else
KDIR :=/lib/modules/$(shell uname -r)/build
PWD :=$(shell pwd)
all:
$(info "1st")
make -C $(KDIR) M=$(PWD) modules
clean:
rm -f *.ko *.o *.mod.o *.symvers *.cmd *.mod.c *.order
endif
w@w:~/linux_kernel/hello$ sudo make
"1st"
make -C /lib/modules/5.4.0-104-generic/build M=/home/w/linux_kernel/hello modules
make[1]: Entering directory '/usr/src/linux-headers-5.4.0-104-generic' 第一次首先执行make all
"2nd"
CC [M] /home/w/linux_kernel/hello/hello.o 第二次生成hello.o
"2nd" 第三次生成hello.ko
Building modules, stage 2.
MODPOST 1 modules
CC [M] /home/w/linux_kernel/hello/hello.mod.o
LD [M] /home/w/linux_kernel/hello/hello.ko
make[1]: Leaving directory '/usr/src/linux-headers-5.4.0-104-generic'