编译驱动的Makefile详解

Posted 正在起飞的蜗牛

tags:

篇首语:本文由小常识网(cha138.com)小编为大家整理,主要介绍了编译驱动的Makefile详解相关的知识,希望对你有一定的参考价值。

1、Makefile源码

#ubuntu的内核源码树,如果要编译在ubuntu中安装的模块
#KERN_VER = $(shell uname -r)
#KERN_DIR = /lib/modules/$(KERN_VER)/build	

# linux内核的源码树目录
KERN_DIR = ~/kernel

obj-m	+= module.o	#构造的模块名字是module.ko
module-objs :=file1.o file2.o	#模块依赖file1.o和file2.o两个源文件

all:
	make -C $(KERN_DIR) M=`pwd` modules 

.PHONY: clean	
clean:
	make -C $(KERN_DIR) M=`pwd` modules clean

(1)KERN_DIR:表示编译驱动所依赖的内核源码树的路径。内核源码要和加载驱动的内核一致,否则可能报错;
(2)make -C:这是Makefile的命令,表示跳转到-C后指定的目录执行该目录下的Makefile。其实就是去执行内核的顶层Makefile,通过传参给顶层 Makefile编译成驱动;
(3)M=pwd:Makefile在构造modules目标之前返回到模块源代码目录;
(4)modules:标明是要编译成驱动文件,具体要去看内核顶层Makefile中对modules传参的处理;

2、xxx.mod.c文件

2.1、xxx.mod.c文件源码

#include <linux/module.h>
#include <linux/vermagic.h>
#include <linux/compiler.h>

MODULE_INFO(vermagic, VERMAGIC_STRING);

struct module __this_module
__attribute__((section(".gnu.linkonce.this_module"))) = 
 .name = KBUILD_MODNAME,
 .init = init_module,
#ifdef CONFIG_MODULE_UNLOAD
 .exit = cleanup_module,
#endif
 .arch = MODULE_ARCH_INIT,
;

static const char __module_depends[]
__used
__attribute__((section(".modinfo"))) =
"depends=";

2.2、xxx.mod.c文件分析

(1)xxx.mod.c是编译驱动的时候自动产生的,假设我们编译module.ko,当编译时就会产生一个module.mod.c文件,文件内容如上;
(2)文件里定义了struct module类型的变量__this_module,__this_module变量初始化的信息就是用来描述驱动,这个结构体会被链接进ko文件中,
将来在加载ko驱动时,内核中会有代码将这个结构体从ko文件中解析出来,并根据这个结构体来加载驱动;
(3)init_module:这个函数是驱动代码中用module_init宏声明的驱动加载函数,init_module是驱动加载函数的统一别名;
(4)cleanup_module:这个函数是驱动代码中用module_exit宏声明的驱动卸载函数,cleanup_module是驱动卸载函数的统一别名;
补充:module_init宏和module_exit宏参考博客:

以上是关于编译驱动的Makefile详解的主要内容,如果未能解决你的问题,请参考以下文章

不修改Linux内核文件,直接用makefile编译驱动,是否要先把内核编译一遍?

RK3568平台开发系列讲解(驱动基础篇)Makefile 详解

Linux2.6 如何编写Makefile,使驱动程序能够编译链接静态库

Redis源码编译篇之Makefile文件详解

Makefile详解

如何创建用于从子目录中的其他 Makefile 编译多个驱动程序的 Makeflow