基于hello.c的Makefile:(在我的《Linux驱动—模块化编程》)
执行模块的Makefile要进来 三次!!!!!!!!
第一次:
KERNELRELEASE是内核源码树的一个变量,我们第一次敲make的时候,其实它是没有的,为空,逗号右边也没有填东西,也为空 故而 ifneq ($(KERNELRELEASE),) 的结果“真”,执行 else 的语句: 进入else后:
KDIR是一个内核的环境变量,绝对路径 ,“:=”为赋值变量
$(shell uname -r)取字符串拼接起来
让我们来看一看 uname -r 是什么命令: 让我们进入/lib/modules/5.4.0-77-generic/build 这个目录下: 我们看到这里面非常像我们内核树目录,这里面就放我们编译生成一个内核目标文件所依赖的一个内核
PWD :=$(shell pwd)
PWD保存make执行的路径
接下来执行 (make前面一定是tab键)
make -C $(KDIR) M=$(PWD) modules
$(KDIR):进入KDIR(内核源码树)去Makefile M=$(PWD):会到这个目录来找目标文件(例如.c文件)
编译内核有两种方式: ①make … modules:将一个驱动文件独立于内核之外编译,再动态加载进去(此处是这种) ②也可以把编译进内核镜像里
我们来看最顶层的Makefile里的内容: 此处我们对应的就是这个命令
注意:不会顺序执行clean
第二次
在第一次执行完make -C $(KDIR) M=$(PWD) modules 后,因为M=$(PWD) ,所以又进来这个目录一次,这次又执行Makefile,这是第二次进入
此次进入后KERNELRELEASE就有值了(进入了一次顶层Makefile,在里面给KERNELRELEASE赋了值) 此时ifneq ($(KERNELRELEASE),) 就成立
执行obj-m:=hello.o “obj-m”:生成模块化目标文件 “:=hello.o”:依赖于hello.o 这个时候它就知道需要将hello.c编译成hello.o,编译好后就又出来了
第三次
从第二次出来以后,顶层Makefile又会去执行一个链接操作,这个链接操作又会再次进入这个目录下面,它会把我们刚刚生成的hello.o文件和其他的一些依赖文件从内核路径/lib/modules/5.4.0-77-generic/build 链接成一个hello.ko文件
我们来验证一下
最后
first:进入内核源码树、获取路径 second:第一次结束时进入了顶层Makefile,顶层Makefile再次进入路径获取生成模块的.c文件,编译为.o third:顶层Makefile将所需要的依赖链接成.ko文件 这个图好像不太生动,嗯。。。。。。放这里装杯吧
|