一 概述
传统的应用编写时,每添加一个模块,都需要在main中添加新模块的初始化。也就是说增加的一个不能算是真正的独立模块,得在main中修改代码才能集成这模块功能。有没有什么办法可以实现main跟其他模块之间隔离呢?main不再关心有什么模块,模块的删减也不需要修改main?
二 liunx内核模块初始化
如果你对liunx模块有一定了解,你应该知道liunx模块都是独立加载,加载模块,不需要修改main代码。甚至不需要重新编译代码。那么内核是如何实现的呢?
1. module_init 函数
模块初始化都会调用 module_init ,那么这函数做了哪些东西呢?我们着入点就从这个module_init 函数开始
// module_init定义在<include/linux/module.h>
#define module_init(x) __initcall(x);
// __initcall 定义在<include/linux/init.h>
#define __initcall(fn) device_initcall(fn)
#define device_initcall(fn) __define_initcall(fn, 6)
#define __define_initcall(fn, id) \
static initcall_t __initcall_##fn##id __used \
__attribute__((__section__(".initcall" #id ".init"))) = fn;
// 例如 初始化一个iptables 模块 module_init(ip_tables_init) 其实就等效于
// static initcall_t _initcall_ip_tables_init_6 __attribute__((unused, section(".initcall6.init"))) = ip_tables_init ;
__ attribute__ ((section(”name“)))是gcc编译器支持的一个编译特性(arm编译器也支持此特性),实现在编译时把某个函数/数据放到name的数据段中。原理如下
- 模块通过__ attribute__((section(“name”))) ,会构建初始化函数表。放到你命名的name数据段中
- 而默认链接脚本缺少自定义的数据段的声明,需要在链接脚本添加你定义的数据段的声明
- 而main在执行初始化时,只需要把name数据段中的所有初始化接口执行一遍即可.
那么这里有两个问题 :
- 如何再链接脚本中添加自己定义的数据段的声明呢?
- main是如何将放入数据段的,模块接口都执行了一遍了呢? </