关于__init
、__initdata
、__exit
、__exitdata
及类似的宏
打开Linux Kernel源代码树中的文件:include/init.h,可以看到有下面的宏定议:
#define __init __attribute__ ((__section__ (".init.text"))) __cold
#define __initdata __attribute__ (( __section__ (".init.data")))
#define __exitdata __attribute__ (( __section__ (".exit.data")))
#define __exit_call __attribute_used__ __attribute__ (( __section__ (".exitcall.exit")))
#define __init_refok oninline __attribute__ ((__section__ (".text.init.refok")))
#define __initdata_refok __attribute__ ((__section__ (".data.init.refok")))
#define __exit_refok noinline __attribute__ ((__section__ (".exit.text.refok")))
.........
#ifdef MODULE
#define __exit __attribute__ (( __section__ (".exit.text"))) __cold
#else
#define __exit __attribute_used__ __attribute__ ((__section__ (".exit.text"))) __cold
#endif
__init
,__initdata
等属性标志,是要把这种属性的代码放入目标文件的.init.text节,数据放入.init.data
节──这一过程是通过编译内核时为相关目标平台提供了xxx.lds链接脚本来指导ld完成的。
对编译成module的代码和数据来说,当模块加载时,__init
属性的函数就被执行;
对静态编入内核的代码和数据来说,当内核引导时,do_basic_setup()
函数调用do_initcalls()
函数,后者负责所有.init
节函数的执行。
关于__attribute__
要了解Linux Kernel代码的分段信息,需要了解一下gcc的__attribute__
的编绎属性或定义的函数或数,__attribute__
主要用于改变所声明据的特性,它有很多子项,用于改变作用对象的特性。比如对函数,noline将禁止进行内联扩展、noreturn表示没有返回值、pure表明函数除返回值外,不会通过其它(如全局变量、指针)对函数外部产生任何影响。但这里对代码段起作用是子项section。
__attribute__
的section子项的使用格式为:
__attribute__((section("section_name")))
其作或用是将作用的函数数据放入指定名为”section_name”输入段。
这里还要注意一下两个概念:输入段和输出段
输入段和输出段是相对于要生成最终的elf或binary时的Link过程说的,Link过程的输入大都是由源代码编绎生成的目标文件.o,那么这些.o文件中包含的段相对link过程来说就是输入段,而Link的输出一般是可执行文件elf或库等,这些输出文件中也包含有段,这些输出文件中的段就叫做输出段。输入段和输出段本来没有什么必然的联系,是互相独立,只是在Link过程中,Link程序会根据一定的规则(这些规则其实来源于Link Script),将不同的输入段重新组合到不同的输出段中,即使是段的名字,输入段和输出段可以完全不同。
其用法举例如下:
int var __attribute__((se