fPIC作用于编译阶段,告诉编译器产生与位置无关代码(Position-Independent Code),

  则产生的代码中,没有绝对地址,所有使用相对地址。故而代码能够被载入器载入到内存的随意

  位置,都能够正确的运行。这正是共享库所要求的,共享库被载入时。在内存的位置不是固定的。

gcc -shared -fPIC -o 1.so 1.c
这里有一个-fPIC參数
PIC就是position independent code
PIC使.so文件的代码段变为真正意义上的共享
假设不加-fPIC,则载入.so文件的代码段时,代码段引用的数据对象须要重定位,重定位会改动代码段的内容,这就造成每一个使用这个.so文件代码段的进程在内核里都会生成这个.so文件代码段的copy.每一个copy都不一样,取决于这个.so文件代码段和数据段内存映射的位置.

不加fPIC编译出来的so,是要再载入时依据载入到的位置再次重定位的.(由于它里面的代码并非位置无关代码)

假设被多个应用程序共同使用,那么它们必须每一个程序维护一份so的代码副本了.(由于so被每一个程序载入的位置都不同,显然这些重定位后的代码也不同,当然不能共享)

我们总是用fPIC来生成so,也从来不用fPIC来生成a.

fPIC与动态链接能够说基本没有关系,libc.so一样能够不用fPIC编译,仅仅是这种so必需要在载入到用户程序的地址空间时重定向全部表目.



因此,不用fPIC编译so并不总是不好.

假设你满足下面4个需求/条件:

1.该库可能须要常常更新

2.该库须要非常高的效率(尤其是有非常多全局量的使用时)

3.该库并不非常大.

4.该库基本不须要被多个应用程序共享



假设用没有加这个參数的编译后的共享库,也能够使用的话,可能是两个原因:

1:gcc默认开启-fPIC选项

2:loader使你的代码位置无关



从GCC来看。shared应该是包括fPIC选项的,但似乎不是所以系统都支持,所以最好显式加上fPIC选项。參见例如以下





`-shared'

    Produce ashared object which can then be linked with other

    objects toform an executable.  Not allsystems support this

    option.  Forpredictable results, you must also specify the same

    set ofoptions that were used to generate code (`-fpic', `-fPIC',

    or modelsuboptions) when you specify this option.(1)





-fPIC 的使用,会生成 PIC代码,.so 要求为 PIC。以达到动态链接的目的。否则,无法实现动态链接。



non-PIC 与 PIC代码的差别主要在于 access global data, jump label 的不同。

比方一条 access global data的指令。

non-PIC 的形势是:ld r3,var1

PIC 的形式则是:ld r3,var1-offset@GOT,意思是从 GOT 表的 index 为 var1-offset的地方处

指示的地址处装载一个值,即var1-offset@GOT处的4个 byte 事实上就是 var1的地址。这个地址仅仅有在执行的时候才知道,是由 dynamic-loader(ld-linux.so)填进去的。



再比方 jump label指令

non-PIC 的形势是:jumpprintf 。意思是调用 printf。

PIC 的形式则是:jumpprintf-offset@GOT,

意思是跳到 GOT 表的 index 为 printf-offset 的地方处指示的地址去运行,

这个地址处的代码摆放在 .plt section,

每一个外部函数相应一段这种代码,其功能是呼叫dynamic-loader(ld-linux.so)来查找函数的地址(本例中是 printf),然后将其地址写到 GOT 表的 index 为 printf-offset的地方,

同一时候运行这个函数。这样。第2次呼叫printf 的时候,就会直接跳到 printf 的地址,而不必再查找了。



GOT是 data section, 是一个 table, 除专用的几个 entry。每一个 entry的内容能够再运行的时候改动;

PLT是 text section, 是一段一段的 code,运行中不须要改动。

每一个 target 实现 PIC的机制不同,但大同小异。

比方 MIPS 没有 .plt, 而是叫 .stub,功能和 .plt一样。



可见,动态链接运行非常复杂。比静态链接运行时间长;可是,极大的节省了size。PIC 和动态链接技术是计算机发展史上非常重要的一个里程碑。



gccmanul上面有说

-fpic      If the GOTsize for the linked executable exceeds a machine-specific maximumsize, you get an error message from the linker indicating that-fpic does not work; in that case, recompile
with -fPIC instead.(These maximums are 8k on the SPARC and 32k on the m68k andRS/6000. The 386 has no such limit.)



-fPIC      Ifsupported for the target machine, emit position-independent code,suitable for dynamic linking and avoiding any limit on the size ofthe global offset table. This option makes a difference
on them68k, PowerPC and SPARC. Position-independent code requires specialsupport, and therefore works only on certainmachines.



关键在于GOT全局偏移量表里面的跳转项大小。

intel处理器应该是统一4字节,没有问题。

powerpc上因为汇编码或者机器码的特殊要求,所以跳转项分为短、长两种。

-fpic为了节约内存,在GOT里面预留了“短”长度。

而-fPIC则採用了更大的跳转项。

ppc_85xx-gcc -shared -fPIC liberr.c -o liberr.so的更多相关文章

  1. linux中[gcc -shared -fPIC]的含义

    linux在gcc编译时加上 -shared 参数时,目的是使源码编译成动态库 .so 文件: 而-fPIC的作用是 告知编译器 生成位置无关代码(编译产生的代码没有绝对位置,只有相对位置):从而可以 ...

  2. gcc编译参数-fPIC问题 `a local symbol' can not be used when making a shared object;

    gcc -shared -o hack.so hack.c/usr/bin/ld: /tmp/ccUZREwA.o: relocation R_X86_64_32 against `a local s ...

  3. gcc编译参数-fPIC的一些问题

    gcc编译参数-fPIC的一些问题 (2012-07-26 15:41:08) 转载▼ 标签: linux compiler gcc -fpic it 分类: NSN_BspDriver ppc_85 ...

  4. -shared -fPIC

    gcc -shared -fPIC -o 1.so 1.c   这里有一个-fPIC参数 PIC就是position independent code PIC使.so文件的代码段变为真正意义上的共享

  5. GCC中-fpic解惑(转载)

    参考: 1.<3.18 Options for Code Generation Conventions>2.<Options for Linking>3.<GCC -fP ...

  6. gcc shared object

    介绍一个生成动态链接库*.so的例子: 首先新建1个头文件test.h: #include <stdio.h> void first(); void second(); void thir ...

  7. Linux编译工具:gcc入门

    1. 什么是gcc gcc的全称是GNU Compiler Collection,它是一个能够编译多种语言的编译器.最开始gcc是作为C语言的编译器(GNU C Compiler),现在除了c语言,还 ...

  8. Linux下Gcc生成和使用静态库和动态库详解(转)

    一.基本概念 1.1什么是库 在windows平台和linux平台下都大量存在着库. 本质上来说库是一种可执行代码的二进制形式,可以被操作系统载入内存执行. 由于windows和linux的平台不同( ...

  9. linux下的gcc编译器

    1. 编译单个文件成可执行文件 gcc hello.c 生成默认的可执行文件a.out 2. 编译单个文件为指定名字的可执行文件,使用-o选项指定文件名字 gcc -o test main.c 生成文 ...

随机推荐

  1. Google浏览器开发者工具:CSSViewer(一个Css查看器)

    CSSViewer的简介 CSSViewer是一款可以帮助用户快速查看当前的网页元素的CSS属性的谷歌浏览器插件,在Chrome中安装了CSSViewer插件以后,用户就可以在设计网页的时候,快速地模 ...

  2. Android(java)学习笔记196:ContentProvider使用之内容观察者01

    1. 内容观察者 不属于四大组件,只是内容提供者ContentProvider对应的小功能. 如果发现数据库内容变化了,就会立刻观察到. 下面是逻辑图:       当A应用中银行内部的数据发生变化的 ...

  3. bzero - 向字符串写入零

    总览 (SYNOPSIS) #include <string.h> void bzero(void *s, size_t n); 描述 (DESCRIPTION) bzero() 函数 把 ...

  4. windows SDK创建一个窗体

    #include <windows.h> /* Declare Windows procedure */ LRESULT CALLBACK WindowProcedure (HWND, U ...

  5. zabbix监控tcp连接并发数

    第一步,想在zabbix的web监控tcp连接数,那么要看zabbix-server的版本和zabbix-agent版本是否一致,不然TCP-status图没有数据 也会报错.下图就是版本不符合报错的 ...

  6. CSS hover 改变另外一个元素状态

    Part.1 问题 我们写页面时也不少遇到这个问题,在没有使用任何预处理语言前提下,当hover 一个元素的时候怎么改变其它的元素? 这里我把它分为两种情况(除自身以外) hover时 1: 改变本身 ...

  7. 用数据集跑一个模型遇到bug如何解决

    自己在用fast rcnn和ssd跑自己数据集过程中都遇到了bug,fast rcnn中是loss下降但值较高,并且测试出来结果一直不对,ssd是loss从一开始到后面loss都一直为0. 遇到这种情 ...

  8. 在 VS2015+EF6.0中使用Mysql 遇到的坑

    1)首先是要在vs2015中安装mysql Database 默认是不存在的 1)下载mysql-connector-net-6.9.9.msi    地址:https://dev.mysql.com ...

  9. 笔试算法题(41):线索二叉树(Threaded Binary Tree)

    议题:线索二叉树(Threaded Binary Tree) 分析: 为除第一个节点外的每个节点添加一个指向其前驱节点的指针,为除最后一个节点外的每个节点添加一个指向其后续节点的指针,通过这些额外的指 ...

  10. Git的入门

    Git的基本介绍: Git:是一个版本控制工具. Github:是非常有名的在线版本管理网站(速度比较慢). Oschina:中国版本的github,(旗下的的码云地址:gitee.com,速度快) ...