内核初始化优化宏(__init, __devinit)

 

在内核里经常可以看到__init, __devinit这样的语句,这都是在init.h中定义的宏,gcc在编译时会将被修饰的内容放到这些宏所代表的section。

 

原文地址:http://blog.chinaunix.net/uid-25871104-id-2854544.html

本文地址:http://blog.csdn.net/younger_china/article/details/13755403

 

1.      定义与用法

典型的定义如下:

#define __init        __section(.init.text) __cold notrace

#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段。即所有的__init xxx()函数都将会被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. 提高系统效率

初始化代码的特点是,在系统启动时运行,且一旦运行后马上推出内存,不再占用内存。

 

3.      常用宏

内核中使用的:

1.  __init,标记内核启动时所用的初始化代码,内核启动完成后就不再使用。其所修饰的内容被放到.init.text section中。

2.  __exit,标记模块退出代码,对非模块无效

3.  __initdata,标记内核启动时所用的初始化数据结构,内核启动完成后不再使用。其所修饰的内容被放到.init.data section中。

4.  __devinit,标记设备初始化所用的代码

5.  __devinitdata,标记设备初始化所用的数据结构

6.  __devexit,标记设备移除时所用的代码

7.  xxx_initcall,7个级别的初始化函数

 

driver中的使用:

1.  module_init, module_exit函数所调用的函数,需要分别用__init和__exit来标记

2.  pci_driver数据结构不需要标记

3.  probe和remove函数用__devinit和__devexit来标记

4.  如果remove使用__devexit标记,则在pci_drvier结构中要用__devexit_p(remove)来引用remove函数

5.  如果不确定需不需要添加宏,则不要添加

 

posted on 2013-10-30 22:33  YoungerChina  阅读(630)  评论(0编辑  收藏  举报

导航