Linux内核中的初始化宏(_init、_exit等)

/************************************************************************************

*本文为个人学习记录,如有错误,欢迎指正。

*本文参考资料: 

*        https://blog.csdn.net/ffmxnjm/article/details/71713670

*        https://blog.csdn.net/qingkongyeyue/article/details/72935439

************************************************************************************/

在Linux内核的/kernel/include/linux/init.h文件中,定义了一些初始化宏。

1. 初始化宏的作用

初始化宏的作用主要有以下两点:

                1)保证内核的初始化函数按照指定的顺序来执行;

                2)提高系统效率。

(1)保证内核的初始化函数按照指定的顺序来执行

初始化宏的实质是段声明。在内核链接的时候,被初始化宏修饰的函数将被链接至指定的段。内核初始化时,会按照相应段的优先级来顺序执行初始化函数。

kernel将初始化要执行的init函数,分为7个级别:

1)core_initcall

2)postcore_initcall

3)arch_initcall

4)subsys_initcall

5)fs_iitcall

6)device_initcall

7)late_initcall

这7个级别优先级递减,即先执行core_initcall, 最后执行late_initcall。通过使用初始化宏,gcc会将初始化代码按下面的结构安排:

在内核初始化时,从__initcall_start到__initcall_end之间的initcall被一次执行。

(2)提高系统效率

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

2. 常用的宏

初始化宏的实质是段声明。在内核链接的时候,被初始化宏修饰的函数将被链接至指定的段。

//标记内核启动时所用的初始化代码,内核启动完成后就不再使用
#define __init          __section(.init.text) __cold notrace

//标记内核启动时所用的初始化数据结构,内核启动完成后不再使用
#define __initdata      __section(.init.data)

//标记模块退出代码,对非模块无效
#define __exit          __section(.exit.text) __exitused __cold

//标记设备初始化所用的代码
#define __devinit       __section(.devinit.text) __cold

//标记设备初始化所用的数据结构
#define __devinitdata   __section(.devinit.data)

//标记设备移除时所用的代码
#define __devexit       __section(.devexit.text) __exitused __cold

对于驱动程序模块来说,这些优化标记使用的情况如下:

  • module_init, module_exit函数所调用的函数,需要分别用__init和__exit来标记;
  • driver数据结构不需要标记;
  • probe和remove函数用__devinit和__devexit来标记;
  • 如果remove使用__devexit标记,则在drvier结构中要用__devexit_p(remove)来引用remove函数;
  • 如果不确定需不需要添加宏,则不要添加。

 

posted @ 2018-08-27 17:00  LinFeng-Learning  阅读(1055)  评论(0编辑  收藏  举报