前言
前文中介绍了 module 的基本组成部分,现在在上一节的基础上,对 module 做一点点深入的理解。
正文
我们的 module 模块,如果动态的加载到内核中,通常在 Makefile 中,以这样的形式进行标识
obj-m +=first_drv.o
如果静态的加载到内核中,使用的是下列的方式
obj-y +=first_drv.o
module_init分析
module_init 这个宏在 include/linux/init.h 这个目录下,一下仅仅摘录和module相关的代码
typedef void (*exitcall_t)(void);
#ifndef MODULE
#define __define_initcall(level,fn,id) \
static initcall_t __initcall_##fn##id __used \
__attribute__((__section__(".initcall" level ".init"))) = fn
#define device_initcall(fn) __define_initcall("6",fn,6)
#define __initcall(fn) device_initcall(fn)
#define __exitcall(fn) \
static exitcall_t __exitcall_##fn __exit_call = fn
#define module_init(x) __initcall(x);
#define module_exit(x) __exitcall(x);
#else
#define module_init(initfn) \
static inline initcall_t __inittest(void) \
{ return initfn; } \
int init_module(void) __attribute__((alias(#initfn)));
#define module_exit(exitfn) \
static inline exitcall_t __exittest(void) \
{ return exitfn; } \
void cleanup_module(void) __attribute__((alias(#exitfn)));
首先解决第一个疑惑,这里列举了两个 module_init ,到底选择哪一个 module_init
这两个 moudle_init 使用一个 MODULE 的宏进行分隔,这里的 MOUDLE 是在 Makefile 中进行确定是否进行定义的,前边知道如果编译为模块在 makefile 中会使用 obj-m,此时使用的是第二个 module_init 的定义。
动态加载
动态加载使用的命令为 insmod 或者 rmmod ,在加载模块的时候,只会加载默认的init_module 和 cleanup_module 这两个函数;所以使用 module_init 和 module_exit 这两个宏的目的就是给我们写的入口函数和出口函数定义一个别名为 module_init 和 module_exit,这样就能实现正常的加载。
在这里使用了 _attribute_((alias(#initfn))); 进行定义别名。
## 静态加载
静态加载的时候,会将入口函数,和出口函数,分别放入对应的 section 段中。
具体可参考如下文章
module_init宏分析
总结
静态加载的模块,存放在特定的 section 段中,静态加载模块,使用module_init这个宏给起一个别名 init_module,供加载模块的是进行调用。
参考文献
#ifdef MODULE 关于MODULE的问题
module_init宏分析