链接选项-rpath的一个问题记录
问题简述
大概是这么一个情况,有一个过去已经写好的程序,这个程序用于处理网络通信,接收一些操作指令。具体的指令操作通过运行时加载动态库的形式进行扩展。(类似于net-snmp二次开发的一种形式)
问题是这样的,用于扩展功能的动态库,其本身又链接了一系列的动态库,这些库的位置与这个扩展用的动态库存放的位置是相对的。大概如下
程序 program
|
| 运行时加载(dlopen/LoadLibrary等)
sodir/lib1.so
| 相对路径存放
sodir/lib2/lib2.so
问题在于编译lib1.so
的时候,并不知道将来lib1.so
会存放于相对程序program
的工作路径的什么位置,导致不知道lib2.so
的位置在哪里。
如果链接的时候直接使用-rpath=./lib2
的话,这就需要lib2.so
在程序工作路径下的lib2
目录下方可,因为运行时加载路径并不是相对于lib1.so
而是相对于program
的。
这里还有一个问题,那就是如果lib2.so还有依赖,且生成lib2.so的时候没有使用-rpath选项
那就比较麻烦了,因为没有相对路径,必须放置在系统默认的so搜索路径或者在/etc/ld.so.conf
中添加。
假定lib2.so
依赖lib3.so
,那么即便是将lib3.so
加入到lib1.so
的链接选项中,也不会链接进去(很早之前的时候ld程序应该是把所以的都链接进去的,不管有没有使用,但最近我测试都是没有的,应该是默认使用--as-needed
选项),所以查找的时候没有使用lib1.so
中指定的runpath
路径。这个可以使用readelf -d lib1.so
命令查看。这个可以给ld
添加--no-as-needed
选项来强制链接所有指定的库,而不是仅仅链接需要的。
解决办法
当前没有好的解决办法。
一个简单的方法是写一个libwarp
来包装一下lib1
,在libwarp
中先设置一下LD_LIBRARY_PATH
环境变量(设置自身进程的没有用,要设置父进程或全局的(加载自身进程的进程)),加上自身所在的路径(dladdr
/GetModulePath
获取),然后再使用ldopen
的方式加载lib1
。而程序则按照原来加载lib1.so
的方式加载libwarp.so
即可。
这种方式实现起来比较麻烦,需要把自己终结之后重新启动以使环境变量生效,不如直接写一个脚本去运行program
,在这之前去设置LD_LIBRARY_PATH
环境变量。
这个问题大概可以用下面的代码来表示(就不详细叙述了)
https://files.cnblogs.com/files/oloroso/sopath.tar.gz
链接选项-rpath的一个问题记录的更多相关文章
- Linux ToolChain (二) --- Linker (1)链接选项 -L -rpath -rpath-link
一.动态库的链接和链接选项-L,-rpath-link,-rpath (1). 现代连接器在处理动态库时将链接时路径(Link-time path)和运行时路径(Run-time path)分开, 用 ...
- 动态库的链接和链接选项-L,-rpath-link,-rpath
https://my.oschina.net/shelllife/blog/115958 链接动态库 如何程序在连接时使用了共享库,就必须在运行的时候能够找到共享库的位置.linux的可执行程序在执行 ...
- ld 链接选项-L,-rpath-link,-rpath
转载自:http://blog.csdn.net/q1302182594/article/details/42102961 1. 三个C文件 1. world.c #include<stdio. ...
- --with-http_realip_module选项(后台Nginx服务器记录原始客户端的IP地址 )
转自:http://blog.itpub.net/27043155/viewspace-734234/ 通过这个模块允许我们改变客户端请求头中客户端IP地址值(例如,X-Real-IP 或 X-For ...
- Linux C/C++ 链接选项之静态库--whole-archive,--no-whole-archive和--start-group, --end-group
参照这两篇博客: http://stackoverflow.com/questions/805555/ld-linker-question-the-whole-archive-option http: ...
- linux 2.6.32.220的一个crash记录
有同事分析一个crash,我参与了分析,记录如下,供遇到相同crash的兄弟参考: crash> bt PID: TASK: ffff881723ce8080 CPU: COMMAND: &qu ...
- 1 - JVM随笔分类(java虚拟机的内存区域分配(一个不断记录和推翻以及再记录的一个过程))
java虚拟机的内存区域分配 在JVM运行时,类加载器ClassLoader在加载到类的字节码后,交由jvm的执行引擎处理, 执行过程中需要空间来存储数据(类似于Cpu及主存),此时的这段空间的分 ...
- HorizontalScrollView做页卡的一个小记录
用HorizontalScrollView做页卡,实现一个如下图的效果:
- window.setTimeout和window.setInterval的区别,及用其中一个方法记录时间。
window.setTimeout(语句,时间)是在多久之后执行语句,语句只执行一次. window.setInterval(语句,时间)是每隔多久执行一次语句,语句循环执行. <!DOCTYP ...
随机推荐
- 腾讯浏览服务X5内核集成
腾讯浏览服务X5内核集成https://www.jianshu.com/p/8a7224ff371a TBS 腾讯浏览器服务接入文档https://x5.tencent.com/tbs/guide/s ...
- C#中decimal ,double,float的区别
浮点型 Name CTS Type Description Significant Figures Range (approximate) float System.Single 32-bit sin ...
- zjoi2017 仙人掌
题解: 好难的dp啊...看题解看了好久才看懂 http://blog.csdn.net/akak__ii/article/details/65935711 如果一开始的图就不是仙人掌,答案显然为0, ...
- Codeforces 442C Artem and Array (看题解)
Artem and Array 经过分析我们能发现, 如果对于一个a[ i ] <= a[ i + 1 ] && a[ i ] <= a[ i - 1 ]可以直接删掉. 最 ...
- Codeforces 1041F Ray in the tube (看题解)
Ray in the tube 感觉是套路题.. 如果确定一个差值x我们如何取确定答案呢, 我们把a[ i ] -> a[ i ] % (2 * x), 把b[ i ] -> (b[ i ...
- class path resource [spring/applicationContext.xml] cannot be opened because it does not exist
1.查看路径有没有写错 2.编辑器认为你的文件不是 source folders(原文件),需要你手动将文件改过来
- P1219 八皇后 含优化 1/5
题目描述 检查一个如下的6 x 6的跳棋棋盘,有六个棋子被放置在棋盘上,使得每行.每列有且只有一个,每条对角线(包括两条主对角线的所有平行线)上至多有一个棋子. 上面的布局可以用序列2 4 6 1 3 ...
- 今天刚学到truncate和delete的区别,做个总结吧
truncate table : 删除内容,释放空间(表中数据会被删除,但不会进入oracle回收站,直接删除),不删除定义 delete table : 删除内容,不释放空间(表中数据虽被删除,但是 ...
- DNS信息收集命令nslookup
DNS信息收集--NSLOOKUP 前面文章我介绍了dig命令,NSLOOKUP命令用法差不多 简单查询 语法:nslookup domain [dns-server] nslookup www.si ...
- AGC01 A - BBQ Easy
目录 题目链接 题解 代码 题目链接 AGC01 A - BBQ Easy 题解 贪心 排序之后从大到小,没两组取小的那个 代码 #include<cstdio> #include< ...