利用gcc的__attribute__編譯屬性section子項構建初始化函數表

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!
相關文章
相關標籤/搜索