码迷,mamicode.com
首页 > 其他好文 > 详细

gcc attribute 初始化函数列表

时间:2014-12-26 12:57:15      阅读:312      评论:0      收藏:0      [点我收藏+]

标签:

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 = 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变量:

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: 0x8049768
aha!
call_ptr: 0x804976c
aloha!
call_ptr: 0x8049770
hello!

Have Fun!

 

转载:http://my.oschina.net/senmole/blog/50710

gcc attribute 初始化函数列表

标签:

原文地址:http://www.cnblogs.com/Roche/p/4186272.html

(0)
(0)
   
举报
评论 一句话评论(0
登录后才能评论!
© 2014 mamicode.com 版权所有  联系我们:gaon5@hotmail.com
迷上了代码!