来自:http://www.17xie.com/read-54054.html
内核使用了大量不同的宏来标记具有不同作用的函数和数据结构。如宏__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()等