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函数;
- 如果不确定需不需要添加宏,则不要添加。