标签:
gcc的__attribute__编译属性有很多子项,用于改变作用对象的特性。这里讨论section子项的作用。
__attribute__的section子项使用方式为:
|
1
|
__attribute__((section("section_name"))) |
其作用是将作用的函数或数据放入指定名为"section_name"的段。
看以下程序片段:
|
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
|
#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 = funcstatic 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变量:
|
1
2
|
extern myown_call _myown_start;extern myown_call _myown_end; |
来自ld的链接脚本,可以使用:
|
1
|
ld --verbose |
获取内置lds脚本,并在:
|
1
|
__bss_start = .; |
之前添加以下内容:
|
1
2
3
4
|
_myown_start = .; .myown : { *(.myown) } = 0x90000000 _myown_end = .; code_segment : { *(code_segment) } |
即定义了.myown段及_myown_start/_myown_end变量(0x90000000这个数值可能需要调整)。
保存修改后的链接器脚本,假设程序为s.c,链接器脚本保存为s.lds,使用以下命令编译:
|
1
|
gcc s.c -Wl,-Ts.lds |
执行结果:
|
1
2
3
4
5
6
7
|
[root@localhost ]# ./a.out call_ptr: 0x8049768aha!call_ptr: 0x804976caloha!call_ptr: 0x8049770hello! |
Have Fun!
转载:http://my.oschina.net/senmole/blog/50710
标签:
原文地址:http://www.cnblogs.com/Roche/p/4186272.html