gcc的__attribute__編譯屬性有不少子項,用於改變做用對象的特性。這裏討論section子項的做用。 shell
__attribute__的section子項使用方式爲: 函數
__attribute__((section("section_name")))其做用是將做用的函數或數據放入指定名爲"section_name"的段。
看如下程序片斷: spa
#include <unistd.h> #include <stdint.h> #include <stdio.h> typedef void (*myown_call)(void); extern myown_call _myown_start; extern myown_call _myown_end; #define _init __attribute__((unused, section(".myown"))) #define func_init(func) myown_call _fn_##func _init = func static void mspec1(void) { write(1, "aha!\n", 5); } static void mspec2(void) { write(1, "aloha!\n", 7); } static void mspec3(void) { write(1, "hello!\n", 7); } func_init(mspec1); func_init(mspec2); func_init(mspec3); /* exactly like below: static myown_call mc1 __attribute__((unused, section(".myown"))) = mspec1; static myown_call mc2 __attribute__((unused, section(".myown"))) = mspec2; static myown_call mc3 __attribute__((unused, section(".myown"))) = mspec3; */ void do_initcalls(void) { myown_call *call_ptr = &_myown_start; do { fprintf (stderr, "call_ptr: %p\n", call_ptr); (*call_ptr)(); ++call_ptr; } while (call_ptr < &_myown_end); } int main(void) { do_initcalls(); return 0; }在自定義的.myown段依次填入mspec1/mspec2/mspec3的函數指針,並在do_initcalls中依次調用,從而達到構造並調用初始化函數列表的目的。
兩個extern變量: 指針
extern myown_call _myown_start; extern myown_call _myown_end;來自ld的連接腳本,能夠使用:
ld --verbose獲取內置lds腳本,並在:
__bss_start = .;以前添加如下內容:
_myown_start = .; .myown : { *(.myown) } = 0x90000000 _myown_end = .; code_segment : { *(code_segment) }即定義了.myown段及_myown_start/_myown_end變量(0x90000000這個數值可能須要調整)。
保存修改後的連接器腳本,假設程序爲s.c,連接器腳本保存爲s.lds,使用如下命令編譯: code
gcc s.c -Wl,-Ts.lds執行結果:
[root@localhost ]# ./a.out call_ptr: 0x8049768 aha! call_ptr: 0x804976c aloha! call_ptr: 0x8049770 hello!Have Fun!