生成可执行的共享库
来源:互联网 发布:java扫描linux文件夹 编辑:程序博客网 时间:2024/05/21 03:20
Linux系统下共享对象文件和可执行文件的格式都是ELF格式,它们并没有什么本质 上的区别。共享对象文件也是可以执行的。例如Linux下动态链接器ld-linux.so就 是共享对象文件,它也可以像可执行文件一样执行;Glibc库函数也是一样。
共享对象文件要执行有2个问题要解决:
- 共享对象文件的加载地址(loading address)是随机的(由操作系统决定恰当的地址),不像可执行文件,每次都加载到相同的固定地址(在Linux下是0x8048000)。因此,共享对象要能正确执行,必须编译成位置无关代码(Position Independent Code, PIC)。这可以通过gcc的-fpic选项得到。
- 共享对象也会用到其他的共享库函数(如C语言库函数),引用的外部符号都必须被解析和重定位,而这些必须在运行时完成。这就需要动态链接器的帮助。所以,共享对象必须编译成依赖动态链接器,也就是说共享对象文件要有.interp段。若没有这个段,操作系统在执行时会认为这个文件不需要动态链接器的支持,因而会直接执行这个文件,最后由于对外部符号的访问是错误的(对外部符号的引用没有正确定位)导致segmentation fault。通常情况下,共享对象文件编译时是不会生成.interp段的,只有可执行文件在使用了共享库的情况下才会生成这个段。下面我就来讲讲如何在共享对象文件中生成这个段。
最好用一个简单的例子来讲解这2个问题如何解决。示例代码如下:
#include // for printf()#include // for exit()void fun(){ printf("This is fun./n"); exit(0);}
该代码保存在fun.c文件中。
第一个问题容易解决。要编译成PIC代码,只需要传递gcc选项-fpic即可:
$ gcc -fpic -shared -Wl,-e,fun -o fun.so fun.c
其中-Wl,-e,fun是通知链接器生成的对象文件的入口地址是fun
。执行文件fun.so会得到如下结果:
$ ./fun.soSegmentation fault (core dumped)
之所以这样是因为在fun.c中调用了C语言库函数printf()
,但没有对这个外部符号正确地重定位(由于没有.interp段,操作系统认为这个文件不需要动态链接库重定位),因而引用了非法地址,所以产生了段错误。通过下面的命令可以查看fun.so文件没有.interp段。
$ readelf -l ./fun.so | grep interp$
结果没有输出,说明fun.so文件中确实不存在.interp段。
要生成.interp段,可以在某个源文件中加入下面一句:
const char __invoke_dynamic_linker__[] __attribute__ ((section (".interp"))) = RUNTIME_LINKER;
这正是glibc的做法。这句代码在glibc的elf/interp.c文件中。 这也是为什么glibc库函数有.interp段的原因。 RUNTIME_LINKER就是动态链接库的名字,取决于目标机器。我们也可以用这种方式来解决这个问题,代码如下:
#include <stdio.h> // for printf()#include <stdlib.h> // for exit()const char __invoke_dynamic_linker[] __attribute__ ((section (".interp"))) = "/lib/ld-linux.so.2";void fun(){ printf("This is fun./n"); exit(0);}
注意我用/lib/ld-linux.so.2
替代了RUNTIME_LINKER,因为它就是linux上的动态链接器,在/lib目录下。
用如下方式编译并执行:
$ gcc -fpic -shared -o fun.so -Wl,-e,fun fun.c$ ./fun.so
终于大功告成。
- 生成可执行的共享库
- 创建可执行的共享库
- Linux上制作可执行的共享库示例
- Linux上制作可执行的共享库示例
- Linux上制作可执行的共享库示例
- 共享库加载失败问题排查。gcc编译器生成共享库时不检查符号的依赖项。gcc编译器生成可执行程序时,会多链接一些无用的额库。
- 共享库加载失败问题排查。gcc编译器生成共享库时不检查符号的依赖项。gcc编译器生成可执行程序时,会多链接一些无用的额库
- 生成可执行jar文件的教程
- 命令行下生成可执行的jar文件
- 如何生成可执行的jar文件
- 生成可执行jar文件的教程
- 如何生成可执行的jar包文件
- 生成java可执行程序的方法
- 使用ant生成可执行的jar包
- matlab gui 生成独立可执行的程序
- java命令生成可执行的jar包
- 如何生成可执行的jar包文件
- maven-onejar生成可执行的独立jar
- mysql调优 分区
- C/C++语言中的main函数,参数argc,argv的作用
- 初学Android,手机管理器之获取网络和SIM卡信息(六十二)
- 软件工程师之路
- SQL Server 分页
- 生成可执行的共享库
- xcode中引入静态库文件方法
- 一个很好的Android应用源码下载站——WAP地带
- EasyUI—Accorditon可折叠标签
- couch base使用记录
- C# 定义了 7 种变量类别:静态变量、实例变量、数组元素、值参数、引用参数、输出参数和局部变量
- 程序员:伤不起的三十岁
- SqlDataReader to IList<T>
- 网络性能简单检测