您的位置:首页 > 运维架构 > Linux

Linux中__init、__devinit等初始化宏

2013-01-28 17:09 225 查看
内核使用了大量不同的宏来标记具有不同作用的函数和数据结构。如宏__init 、__devinit 等。这些宏在include/linux/init.h 头文件中定义。编译器通过这些宏可以把代码优化放到合适的内存位置,以减少内存占用和提高内核效率。下面是一些常用的宏:· __init ,标记内核启动时使用的初始化代码,内核启动完成后不再需要。以此标记的代码位于.init.text 内存区域。它的宏定义是这样的:
·   #define _ _init _ _attribute_ _ ((_ _section_ _ (".text.init")))
· __exit ,标记退出代码,对于非模块无效。· __initdata ,标记内核启动时使用的初始化数据结构,内核启动完成后不再需要。以此标记的代码位于.init.data 内存区域。· __devinit ,标记设备初始化使用的代码。· __devinitdata ,标记初始化设备数据结构的函数。· __devexit ,标记移除设备时使用的代码。· xxx_initcall ,一系列的初始化代码,按降序优先级排列。
初始化代码的内存结构
_init_begin -------------------
 | .init.text | ---- __init
 |-------------------|
 | .init.data | ---- __initdata
_setup_start |-------------------|
 | .init.setup | ---- __setup_param
__initcall_start |-------------------|
 | .initcall1.init | ---- core_initcall
 |-------------------|
 | .initcall2.init | ---- postcore_initcall
 |-------------------|
 | .initcall3.init | ---- arch_initcall
 |-------------------|
 | .initcall4.init | ---- subsys_initcall
 |-------------------|
 | .initcall5.init | ---- fs_initcall
 |-------------------|
 | .initcall6.init | ---- device_initcall
 |-------------------|
 | .initcall7.init | ---- late_initcall
__initcall_end |-------------------|
 | |
 | ... ... ... |
 | |
__init_end -------------------
初始化代码的特点是:在系统启动运行,且一旦运行后马上退出内存,不再占用内存。对于驱动程序模块来说,这些优化标记使用的情况如下:· 通过module_init() 和module_exit() 函数调用的函数就需要使用__init 和__exit 宏来标记。· pci_driver 数据结构不需标记。· probe() 和remove() 函数应该使用__devinit 和__devexit 标记,且只能标记probe() 和remove()· 如果remove() 使用__devexit 标记,则在pci_driver 结构中要用__devexit_p(remove) 来引用remove()函数。· 如果你不确定需不需要添加优化宏则不要添加。补充:例如,如果要在驱动中使用i2c或者spi总线,那么他们的初始化常常会被赋值为subsys_initcall()等====================================================================================================分类: LINUX在内核里经常可以看到__init, __devinit这样的语句,这都是在init.h中定义的宏,gcc在编译时会将被修饰的内容放到这些宏所代表的section。其典型的定义如下:#define __init __section(.init.text) __coldnotrace#define __initdata __section(.init.data)#define __initconst __section(.init.rodata)#define __exitdata __section(.exit.data)#define __exit_call __used __section(.exitcall.exit)其典型用法如下:static int __init xxx_drv_init(void){return pci_register_driver(&xxx_driver);}根据上面的定义与用法,xxx_drv_init()函数将会被link到.init.text段。之所以加入这样的宏,原因有2:1,一部分内核初始化机制依赖与它。如kernel将初始化要执行的init函数,分为7个级别,core_initcall, postcore_initcall, arch_initcall, subsys_initcall, fs_iitcall, device_initcall, late_initcall。这7个级别优先级递减,即先执行core_initcall, 最后执行late_initcall。通过使用文中提到的宏,gcc会将初始化代码按下面的结构安排:在内核初始化时,从__initcall_start到__initcall_end之间的initcall被一次执行。2,提高系统效率初始化代码的特点是,在系统启动时运行,且一旦运行后马上推出内存,不再占用内存。================================================================================常用的宏:__init,标记内核启动时所用的初始化代码,内核启动完成后就不再使用。其所修饰的内容被放到.init.text section中。__exit,标记模块退出代码,对非模块无效__initdata,标记内核启动时所用的初始化数据结构,内核启动完成后不再使用。其所修饰的内容被放到.init.data section中。__devinit,标记设备初始化所用的代码__devinitdata,标记设备初始化所用的数据结构__devexit,标记设备移除时所用的代码xxx_initcall,7个级别的初始化函数==================================================================================driver中的使用:module_init, module_exit函数所调用的函数,需要分别用__init和__exit来标记pci_driver数据结构不需要标记probe和remove函数用__devinit和__devexit来标记如果remove使用__devexit标记,则在pci_drvier结构中要用__devexit_p(remove)来引用remove函数如果不确定需不需要添加宏,则不要添加
内容来自用户分享和网络整理,不保证内容的准确性,如有侵权内容,可联系管理员处理 点击这里给我发消息
标签: