linux核心段屬性機制
以subsys_initcall和module_init為例
subsys_initcall是一個宏,定義在linux/init.h中。經過對這個宏進行展開,發現這個宏的功能是:將其聲明的函數放到一個特定的段:.initcall4.init
subsys_initcall __define_initcall("4",fn,4)
以下檔案在/include/linux/init.h:
分析module_init宏,可以看出它將函數放到.initcall6.init段
module_init __initcall device_initcall __define_initcall("6",fn,6)
開啟編譯過的核心源碼樹中的的/arch/arm/kernel/vmlinux.lds檔案(沒編譯沒有這個檔案):
SECTIONS{ . = 0xC0000000 + 0x00008000; .init : { /* Init code and data */ _stext = .; _sinittext = .; *(.head.text) *(.init.text) *(.cpuinit.text) *(.meminit.text) ...... . = ALIGN(16); __setup_start = .; *(.init.setup) __setup_end = .; __initcall_start = .; *(.initcallearly.init) __early_initcall_end = .; *(.initcall0.init) *(.initcall0s.init) *(.initcall1.init) *(.initcall1s.init) *(.initcall2.init) ... __initcall_end = .; ......
核心在啟動過程中需要順序的做很多事,核心如何?按照先後順序去做很多初始化操作。核心的解決方案就是給核心啟動時要調用的所有函數歸類,執行核心某一個函數然後每個類就會按照一定的次序被調用執行。這些分類名就叫.initcallx.init。x的值從1到8。核心開發人員在編寫核心代碼時只要將函數設定合適的層級,這些函數就會被連結的時候放入特定的段,核心啟動時再按照段順序去依次執行各個段即可(通過某一個函數,連結指令碼只是規定了某一程式段在記憶體中的存放位置)。
核心原始碼:
以下檔案在/init/main.c
extern initcall_t __initcall_start[], __initcall_end[], __early_initcall_end[];static void __init do_initcalls(void){ initcall_t *fn; for (fn = __early_initcall_end; fn < __initcall_end; fn++) do_one_initcall(*fn); /* Make sure there is no pending stuff from the initcall sequence */ flush_scheduled_work();}
執行do_initcalls就會按照設定好的順序去執行,通過函數的內容可以猜測出其原理就是連結指令碼設定好的順序,然後do_initcalls執行就會去按照連結指令碼設定好的順序一個個遍曆