liunx 内核动态模块初始化 __attribute__((section(XXX)))

一 概述

传统的应用编写时,每添加一个模块,都需要在main中添加新模块的初始化。也就是说增加的一个不能算是真正的独立模块,得在main中修改代码才能集成这模块功能。有没有什么办法可以实现main跟其他模块之间隔离呢?main不再关心有什么模块,模块的删减也不需要修改main?

二 liunx内核模块初始化

如果你对liunx模块有一定了解,你应该知道liunx模块都是独立加载,加载模块,不需要修改main代码。甚至不需要重新编译代码。那么内核是如何实现的呢?

1. module_init 函数

模块初始化都会调用 module_init ,那么这函数做了哪些东西呢?我们着入点就从这个module_init 函数开始

// module_init定义在 
#define module_init(x)	__initcall(x);
// __initcall 定义在 
#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是如何将放入数据段的,模块接口都执行了一遍了呢?

2. 链接脚本处理

内核是根据不同的架构,调用内核自己写的对应的链接脚本。而这部分本人也没有完全理解,这里就也深入内核讨论这块。
ld链接命令有两个关键的选项如下

ld -T 
                    
                    

你可能感兴趣的:(嵌入式,—,Linux)