转自:https://my.oschina.net/u/180497/blog/177206

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(, "aha!\n", );
} static void mspec2(void)
{
write(, "aloha!\n", );
} static void mspec3(void)
{
write(, "hello!\n", );
} 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 ;
} 在自定义的.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!
© 著作权归作者所有

利用gcc的__attribute__编译属性section子项构建初始化函数表【转】的更多相关文章

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

    参考链接 :    https://my.oschina.net/u/180497/blog/177206

  2. 利用__attribute__((section()))构建初始化函数表【转】

    转自: https://mp.weixin.qq.com/s?__biz=MzAwMDUwNDgxOA==&mid=2652663356&idx=1&sn=7797629530 ...

  3. 廖威雄: 思维导图:利用__attribute__((section()))构建初始化函数表与Linux内核init的实现

    版权声明:本文为博主原创文章,未经博主同意不得转载. https://blog.csdn.net/juS3Ve/article/details/79049404 本文具体解说了利用__attribut ...

  4. GCC的__attribute__ ((constructor))和__attribute__ ((destructor))

    通过一个简单的例子介绍一下gcc的__attribute__ ((constructor))属性的作用.gcc允许为函数设置__attribute__ ((constructor))和__attrib ...

  5. [转]GCC系列: __attribute__((visibility("")))

    在 objc-api.h 里面有很多关于__attribute__ 的定义. 例如 #if !defined(OBJC_VISIBLE) # if TARGET_OS_WIN32 # if defin ...

  6. 利用GCC编译器生成动态链接库和静态链接库

    转载请标明:http://www.cnblogs.com/winifred-tang94/ 1.编译过程 gcc –fPIC –c xxx.c 其中-fPIC是通知gcc编译器产生位置独立的目标代码. ...

  7. 利用@property实现可控的属性操作

    利用@property实现可控的属性操作 Python中没有访问控制符, 不像java之类的 public class Person{ private int x public int getAge( ...

  8. gcc与makefile编译 BY 四喜三顺

    gcc编译控制过程:(假设源代码为a.c)(1)源文件到预处理文件:    gcc -E -o a.cxx a.c    a.cxx显示调用哪些头文件(2)生成汇编代码:              g ...

  9. java利用反射获取类的属性及类型

    java利用反射获取类的属性及类型. import java.lang.reflect.Field; import java.math.BigDecimal; import java.util.Map ...

随机推荐

  1. jenkins配置slave节点 构建项目并执行操作

    1.新建与配置结点 [系统管理]-> [管理结点]-> [新建结点] 2.配置slave 说明: Name: 定义slave的唯一名称标识,可以是任意字符串,通常设置为slave主机名.i ...

  2. 520的信心赛——点点玩deeeep

                                   3.点点玩 deeeep(deeeep.cpp) 描述 点点最近迷上了 deeeep(此 de 非彼 de),在研究一个特殊的最长树链问题 ...

  3. 【MVVM 原生】原生MVVM的使用

    一.前言       前些天需要完成一个任务,该任务属于公司的一些核心代码,为了避免不必要的麻烦,任务要求不能使用第三方的MVVM框架,必须用原生的. 平时习惯了Dev与MVVMLight,遇上原生的 ...

  4. 【CF813E】Army Creation(主席树)

    [CF813E]Army Creation(主席树) 题面 CF 洛谷 翻译 by ppl 见洛谷 题解 考虑最多只会有\(K\)个相同的数 那么,也就是说,如果一个数会被选 那么,和它相等的数中,在 ...

  5. 「CodePlus 2017 11 月赛」可做题

    这种题先二进制拆位,显然改的位置只有每一段确定的数的开头和结尾,只需要对于每一个可决策位置都尝试一下填1和0,然后取min即可. #include<iostream> #include&l ...

  6. 《javascript高级程序设计(第3版)》-1

    javascript有下列三个不同的部分组成: ECMAScript,由ECMA-262定义,提供核心语言功能 文档对象模型(DOM),提供访问和操作网页内容的方法和接口 浏览器对象模型(BOM),提 ...

  7. 关于OpenCV的stitching使用

    配置环境:VS2010+OpenCV2.4.9 为了使用OpenCV实现图像拼接头痛了好长时间,一直都没时间做,今天下定决心去实现基本的图像拼接. 首先,看一看使用OpenCV进行拼接的方法 基本都是 ...

  8. C++并发编程 异步任务

    C++并发编程 异步任务 异步任务 std::async (1) std::async 会返回一个 std::future 对象, 这个对象持有最终计算出来的结果. 当需要这个值时, 只需要调用对象的 ...

  9. bzoj 2081 [Poi2010]Beads hash+调和级数

    2081: [Poi2010]Beads Time Limit: 10 Sec  Memory Limit: 259 MBSubmit: 1003  Solved: 334[Submit][Statu ...

  10. select、poll和epoll多路I/O复用

    一.三者的区别 select  select最早于1983年出现在4.2BSD中,它通过一个select()系统调用来监视多个文件描述符的数组,当select()返回后,该数组中就绪的文件描述符便会被 ...