当前位置: 代码迷 >> 综合 >> Linux动态库.so文件加载搜索路径详解
  详细解决方案

Linux动态库.so文件加载搜索路径详解

热度:87   发布时间:2024-03-08 15:49:44.0

因为这一段要集成代码,除了组内的,还有组间的,还有第三方的,这里面都采用动态库的方式进行链接,但是在集成的时候经常出现so文件找不到的问题,使用ldd查看,也是某些so文件找不到。有的文件明明就在那里,但是却找不到,针对这些麻烦,我们头疼医头脚疼医脚,暂时解决了这些问题,但是并没有从根本上了解清楚,解决思路单一,都是修改ld.so.conf文件,然后执行ldconfig。今天看了帖子,就顺便把这个so文件查找顺序的问题彻底了解了,省得后面再出问题的时候,又修改ld.so.conf去解决。

首先,我们要知道so文件是怎么查找的。

 动态链接器ld-linux.so按照下面的顺序来搜索需要的动态共享库对于elf格式的可执行程序,是由ld-linux.so*来完成的,它先后搜索elf文件的DT_RPATH段(不可控) -->  环境变量LD_LIBRARY_PATH --> /etc/ld.so.cache文件列表 --> /lib/和/usr/lib 目录找到库文件后载入内存。

1.ELF可执行文件中动态段中DT_RPATH所指定的路径。这实际上是通过一种不算很常用,却比较实用的方法所设置的:编译目标代码时,可以对gcc/g++加入链接参数“-Wl,-rpath”指定动态库搜索路径; 实际上,这种方式我们基本没用过,所以这个查找方式基本不会起作用。
2.环境变量LD_LIBRARY_PATH指定的动态库搜索路径;一般需要修改用户的bashrc或者系统的profile文件 ,一般小程序也不会修改这俩文件,除非是对于/usr/local/lib这种通用的路径,有可能会放进去,但也不一定。
3./etc/ld.so.cache中所缓存的动态库路径(如果支持ld.so.cache的话,这个文件还不小有几百k,有一大堆具体的so文件)。这可以通过修改配置文件/etc/ld.so.conf中指定的动态库搜索路径来改变; 
4.默认的动态库搜索路径/lib或者是/lib64; 
5.默认的动态库搜索路径/usr/lib或者是/usr/lib64

简单例子 printf.cpp

#include <iostream>void Print()
{std::cout<<"Hello,world!"<<std::endl;
}

动态库编译

 g++ printf.cpp -shared -fPIC -o libprintf.so

 main.cpp

#include <iostream>
#include <printf.cpp>extern void Print();int main()
{Print();return 0;
}

1)指定动态库的搜索路径为当前本地路径

g++ main.cpp -L./  -lprintf -Wl,-rpath=./

查看可执行程序,内部的细节

readelf -d a.outDynamic section at offset 0xe08 contains 26 entries:Tag        Type                         Name/Value0x0000000000000001 (NEEDED)             Shared library: [libprintf.so]0x0000000000000001 (NEEDED)             Shared library: [libc.so.6]0x000000000000000f (RPATH)              Library rpath: [./]

指定了动态库链接的位置

2)增加系统变量

export LD_LIBRARY_PATH=./ g++ main.cpp -L./  -lprintf

去掉指定目录指令

unset LD_LIBRARY_PATH 
echo $LD_LIBRARY_PATH