文档章节

利用gcc的__attribute__编译属性section子项构建初始化函数表

a
 alex_wei
发布于 2013/11/18 14:39
字数 383
阅读 9684
收藏 4

gcc的__attribute__编译属性有很多子项,用于改变作用对象的特性。这里讨论section子项的作用。

__attribute__的section子项使用方式为:

__attribute__((section("section_name")))
其作用是将作用的函数或数据放入指定名为"section_name"的段。

看以下程序片段:

#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,使用以下命令编译:

gcc s.c -Wl,-Ts.lds
执行结果:
[root@localhost ]# ./a.out 
call_ptr: 0x8049768
aha!
call_ptr: 0x804976c
aloha!
call_ptr: 0x8049770
hello!
Have Fun!

© 著作权归作者所有

共有 人打赏支持
a
粉丝 2
博文 6
码字总数 2777
作品 0
海淀
程序员
加载中

评论(2)

t
totopper
写的不错,顶一下
hardboy_du
hardboy_du
GCC的C语言扩展笔记(二)-- linux gcc的属性解析

本来打算接着摘抄《GCC参考手册》关于属性扩展(attribute)的说明,不过由于翻译太差,许多地方看得莫名其妙,就到网上找了下,下面是一篇关于属性的文章,讲得不错,转帖于下。 不敢掠美,附...

栖在云端
2012/03/23
0
0
__attribute__机制介绍

1. attribute GNU C的一大特色(却不被初学者所知)就是attribute机制。 attribute可以设置函数属性(Function Attribute)、变量属性(Variable Attribute)和类型属性(Type Attribute) attrib...

长平狐
2013/01/06
61
0
__attribute__机制介绍

attribute GNU C的一大特色(却不被初学者所知)就是attribute机制。 attribute可以设置函数属性(Function Attribute)、变量属性(Variable Attribute)和类型属性(Type Attribute) attribute前...

慎思
2012/08/14
0
0
android源码分析--从logger内核驱动开始

logger的驱动程序为文件logger.c, 位于内核driver/staging/android目录. 从最后一行()入口, 内核在启动时调用函数. deviceinitcall是在内核include/linux/init.h中定义的宏, 其临近有多个如x...

Lukes
2013/08/31
0
0
GCC __attribute__特性的妙用

GCC有一个很好的特性attribute,可以告知编译器某个声明的属性,从而在编译阶段检查引用该符号的地方调用方式是否符合其属性,从而产生警告信息,让错误尽早发现。 attribute format在print...

jaden1q84
2013/03/21
0
0

没有更多内容

加载失败,请刷新页面

加载更多

下一页

java并发备忘

不安全的“先检查后执行”,代码形式如下: if(条件满足){ //这里容易出现线程安全问题//doSomething}else{//doOther} 读取-修改-写入 原子操作:使用CAS技术,即首先从V中读取...

Funcy1122
今天
0
0
SpringBoot2.0 停机

最近新建了个SpringBoot2.0的项目,因为原来一直使用的是传统的Tomcat部署war包的形式,所以这次SpringBoot内置Tomcat部署jar包的时候遇到了很多问题。其中一个就是因为没有外置的Tomcat容器...

Canaan_
昨天
0
1
Confluence 6 外部参考

一个外部参考的意思是任何站点链接到你 Confluence 的实例。任何时候当 Confluence 的用户单击这个外部链接的时候,Confluence 可以记录这次单击为参考。 在默认的情况下,外部链接的参考链接...

honeymose
昨天
0
0
Android中的设计模式之抽象工厂模式

参考 《设计模式解析》 第十一章 Abstract Factory模式 《设计模式:可复用面向对象软件的基础 》3.1 Abstract Factory 抽象工厂 对象创建型模式 《Android源码设计模式解析与实战》第6章 创...

newtrek
昨天
0
0
Redis | 地理空间(GEO)的一个坑

Redis的地理空间(Geo)是个好东西,轻轻松松的就可以把地图描点的问题处理了, 最近却遇到一个坑...Redis采用的Msater-Slave模式, 运用GEORADIUS在salve读取对应的数据,新增了从节点但是从不返...

云迹
昨天
1
0

没有更多内容

加载失败,请刷新页面

加载更多

下一页

返回顶部
顶部