initcall机制
参考:initcall机制
/* include/linux/init.h: */ /* For assembly routines */ #define __HEAD .section ".head.text","ax" #define __INIT .section ".init.text","ax" #define __CPUINIT .section ".cpuinit.text", "ax" #define __init __section(.init.text) __cold notrace /* __initcall_start = .; *(.initcallearly.init) //#define early_initcall(fn) __define_initcall("early",fn,early) __early_initcall_end = .; *(.initcall0.init) //#define pure_initcall(fn) __define_initcall("0",fn,0) *(.initcall0s.init) *(.initcall1.init) //#define core_initcall(fn) __define_initcall("1",fn,1) *(.initcall1s.init) //#define core_initcall_sync(fn) __define_initcall("1s",fn,1s) *(.initcall2.init) //#define postcore_initcall(fn) __define_initcall("2",fn,2) *(.initcall2s.init) //#define postcore_initcall_sync(fn) __define_initcall("2s",fn,2s) *(.initcall3.init) //#define arch_initcall(fn) __define_initcall("3",fn,3) *(.initcall3s.init) //#define arch_initcall_sync(fn) __define_initcall("3s",fn,3s) *(.initcall4.init) //#define subsys_initcall(fn) __define_initcall("4",fn,4) *(.initcall4s.init) //#define subsys_initcall_sync(fn) __define_initcall("4s",fn,4s) *(.initcall5.init) //#define fs_initcall(fn) __define_initcall("5",fn,5) *(.initcall5s.init) //#define fs_initcall_sync(fn) __define_initcall("5s",fn,5s) *(.initcallrootfs.init) //#define rootfs_initcall(fn) __define_initcall("rootfs",fn,rootfs) *(.initcall6.init) //#define device_initcall(fn) __define_initcall("6",fn,6) *(.initcall6s.init) //#define device_initcall_sync(fn) __define_initcall("6s",fn,6s) *(.initcall7.init) //#define late_initcall(fn) __define_initcall("7",fn,7) *(.initcall7s.init) //#define late_initcall_sync(fn) __define_initcall("7s",fn,7s) __initcall_end = .; #define module_init(x) __initcall(x); #define __initcall(fn) device_initcall(fn) #define device_initcall(fn) __define_initcall("6",fn,6) */
下面以arch_initcall(customize_machine);为例分析宏的展开过程
static int __init customize_machine(void) { /* customizes platform devices, or adds new ones */ if (machine_desc->init_machine) machine_desc->init_machine(); return 0; } arch_initcall(customize_machine);
typedef int (*initcall_t)(void); //定义函数指针,无参数,返回int
#define arch_initcall(fn) __define_initcall("3",fn,3) /* initcalls are now grouped by functionality into separate * subsections. Ordering inside the subsections is determined * by link order. * For backwards compatibility, initcall() puts the call in * the device init subsection. * * The `id' arg to __define_initcall() is needed so that multiple initcalls * can point at the same handler without causing duplicate-symbol build errors. */ #define __define_initcall(level,fn,id) \ static initcall_t __initcall_##fn##id __used \ __attribute__((__section__(".initcall" level ".init"))) = fn
arch_initcall(customize_machine)展开为
static initcall_t __initcall_customize_machine3 __used __attribute__((__section__(".initcall3.init"))) = customize_machine