转:linux下共享库的注意点之-fpic
转: http://www.cnblogs.com/leo0000/p/5691483.html
linux下共享库的注意点之-fpic
在编译共享库必须加上-fpic。这是为什么呢?
首先看一个简单的例子:
#include <stdio.h> int fun1()
{
printf("fun1\n");
}
先不加-fpic的情况下生成库,反汇编查看fun1的机器码

0000044c <fun1>:
44c: 55 push %ebp
44d: 89 e5 mov %esp,%ebp
44f: 83 ec 18 sub $0x18,%esp
452: c7 04 24 b2 04 00 00 movl $0x4b2,(%esp)
459: e8 fc ff ff ff call 45a <fun1+0xe>
45e: c9 leave
45f: c3 ret

可以看出调用printf的位置是那个唯一的一个call,并不是跳转到plt表,有关plt表的内容可以查看我前面的博文。也就是说在该库被加载时需要修改代码段来达到重定位的效果。那么每一个加载这个共享库的程序都要有这个库的一份拷贝,这样实际上就没有达到共享库的效果。
看下运行时的机器码

0xb771d44c <+0>: 55 push %ebp
0xb771d44d <+1>: 89 e5 mov %esp,%ebp
0xb771d44f <+3>: 83 ec 18 sub $0x18,%esp
0xb771d452 <+6>: c7 04 24 b2 d4 71 b7 movl $0xb771d4b2,(%esp)
0xb771d459 <+13>: e8 42 b2 ea ff call 0xb75c86a0 <puts>
0xb771d45e <+18>: c9 leave
0xb771d45f <+19>: c3 ret

显然代码段被修改了。
再看一下再加了-fpic的情况下生成的库,反汇编看下fun1的机器码

0000045c <fun1>:
45c: 55 push %ebp
45d: 89 e5 mov %esp,%ebp
45f: 53 push %ebx
460: 83 ec 14 sub $0x14,%esp
463: e8 ef ff ff ff call 457 <__i686.get_pc_thunk.bx>
468: 81 c3 8c 1b 00 00 add $0x1b8c,%ebx
46e: 8d 83 ee e4 ff ff lea -0x1b12(%ebx),%eax
474: 89 04 24 mov %eax,(%esp)
477: e8 04 ff ff ff call 380 <puts@plt>
47c: 83 c4 14 add $0x14,%esp
47f: 5b pop %ebx
480: 5d pop %ebp
481: c3 ret
482: 90 nop
483: 90 nop
484: 90 nop
485: 90 nop
486: 90 nop
487: 90 nop
488: 90 nop

看过很多汇编代码的人知道printf有时候是puts,所以这段机器码中printf就对应第二个call,也就是跳转到plt表中去查找puts符号,那么这样就达到了共享库的效果,此时每一个需要该库的程序只是有一个plt表的拷贝,而代码段所有应用程序是共享的。
再看下运行时机器码

0xb773045c <+0>: 55 push %ebp
0xb773045d <+1>: 89 e5 mov %esp,%ebp
0xb773045f <+3>: 53 push %ebx
0xb7730460 <+4>: 83 ec 14 sub $0x14,%esp
0xb7730463 <+7>: e8 ef ff ff ff call 0xb7730457 <__i686.get_pc_thunk.bx>
0xb7730468 <+12>: 81 c3 8c 1b 00 00 add $0x1b8c,%ebx
0xb773046e <+18>: 8d 83 ee e4 ff ff lea -0x1b12(%ebx),%eax
0xb7730474 <+24>: 89 04 24 mov %eax,(%esp)
0xb7730477 <+27>: e8 04 ff ff ff call 0xb7730380 <puts@plt>
0xb773047c <+32>: 83 c4 14 add $0x14,%esp
0xb773047f <+35>: 5b pop %ebx
0xb7730480 <+36>: 5d pop %ebp
0xb7730481 <+37>: c3 ret

显然是一致的。
所以,在编译共享库时是必须加上-fpic的选项的,否则共享库剩(省?)下的仅仅是硬盘上的空间,而没有剩(省?)下内存。
转:linux下共享库的注意点之-fpic的更多相关文章
- linux下共享库的注意点之-fpic
在编译共享库必须加上-fpic.这是为什么呢? 首先看一个简单的例子: #include <stdio.h> int fun1() { printf("fun1\n") ...
- Linux下共享库嵌套依赖问题 (转载)
转自:http://my.oschina.net/moooofly/blog/506466 问题场景: 动态库 librabbitmq_r.so 内部依赖动态库 libevent_core.so 和 ...
- Linux下的库操作工具-nm、ar、ldd、ldconfig和ld.so
Linux下的库操作工具-nm.ar.ldd.ldconfig和ld.so .nm [options] file 列出file中的所有符号 [option] -c 将符号转化为用户级的名字 -s 当用 ...
- linux下静态库和动态库一些东西
http://www.cnblogs.com/changefuture/archive/2011/12/22/2297460.html Linux 动态链接库和静态库示例 文件预览 文件目录树如下, ...
- 谈谈Linux下动态库查找路径的问题 ldconfig LD_LIBRARY_PATH PKG_CONFIG_PATH
谈谈Linux下动态库查找路径的问题 ldconfig LD_LIBRARY_PATH PKG_CONFIG_PATH 转载自:http://blog.chinaunix.net/xmlrpc.ph ...
- Linux下动态库生成和使用
Linux下动态库生成和使用 一.动态库的基本概念 1.动态链接库是程序运行时加载的库,当动态链接库正确安装后,所有的程序都可以使用动态库来运行程序.动态链接库是目标文件的集合,目标文件在动态链接库中 ...
- Linux下静态库生成和使用
Linux下静态库生成和使用 一.静态库概念 1.库是预编译的目标文件(object files)的集合,它们可以被链接进程序.静态库以后缀为”.a”的特殊的存档(archive file)存储. ...
- 查看Linux下*.a库文件中文件、函数、变量
查看Linux下*.a库文件中文件.函数.变量等情况在Linux 下经常需要链接一些 *.a的库文件,那怎么查看这些*.a 中包 含哪些文件.函数.变量: 1. 查看文件:ar -t xxx.a 2. ...
- 深入理解LINUX下动态库链接器/加载器ld-linux.so.2
[ld-linux-x86-64.so.2] 最近在Linux 环境下开发,搞了好几天 Compiler 和 linker,觉得有必要来写一篇关于Linux环境下 ld.so的文章了,google上搜 ...
随机推荐
- java.sql.date 插入数据库没有时分秒
java.sql.date 插入数据库没有时分秒 把java中实体类的sql.date类型改成java.sql.Timestamp类型即可 util.date 转 Timestamp: java.sq ...
- 如何用纯 CSS 创作一个菱形 loader 动画
效果预览 在线演示 按下右侧的"点击预览"按钮可以在当前页面预览,点击链接可以全屏预览. https://codepen.io/comehope/pen/eKzjqK 可交互视频教 ...
- day20-python之装饰器
1.装饰器 #!/usr/bin/env python # -*- coding:utf-8 -*- import time def cal(l): start_time=time.time() re ...
- (转)iOS完成学习路线
转自 MJ大神博客 原文地址http://blog.csdn.net/q199109106q/article/details/8596506 晚特地花时间整理出了iOS的完整学习路线图,希望对大家有帮 ...
- PAT Basic 1028
1028 人口普查 某城镇进行人口普查,得到了全体居民的生日.现请你写个程序,找出镇上最年长和最年轻的人. 这里确保每个输入的日期都是合法的,但不一定是合理的——假设已知镇上没有超过200岁的老人,而 ...
- Cocoa-Cocoa对象
2.Cocoa对象 2.1 Objective-C是面向对象的语言 Objective-C和Java C++一样,有封装,继承,多态,重用.但是它不像C++那样有重载操作法.模版和多继承,也没有Jav ...
- php 上传文件名出现乱码
想必很多朋友在进行utf8编码的php开发上传功能的时候,都会遇到这样的一个问题,就是上传中文文件名的文件时,文件名会变成乱码,其实我们可以用iconv函数对文件名进行重新编码就解决问题了 可能会有不 ...
- x86实模式到保护模式 李忠 王晓波
x86实模式到保护模式 李忠 王晓波 第3到4章 各个进制间的转换省略 实验环境 编译器 nasm 虚拟机 virtual box 小程序 hexview 观察编译后的机器代码 fixvhd ...
- js对文字进行编码
js对文字进行编码涉及3个函数:escape,encodeURI,encodeURIComponent,相应3个解码函数:unescape,decodeURI,decodeURIComponent 1 ...
- 让Sublime Text成为静态WEB服务器:SublimeServer
如果你使用Sublime Text作为你的编辑器,那么在进行HTML和Java开发的时候有一个很有用的功能,帮你完成前端的联调测试,那就是Sublime Text的服务器插件:SublimeServe ...