Linux系统下的RPATH
什么是RPATH
rpath全称是run-time search path
。Linux下所有elf格式的文件都包含它,特别是可执行文件。它规定了可执行文件在寻找.so
文件时的第一优先位置。
另外,elf文件中还规定了runpath。它与rpath相同,只是优先级低一些。
搜索.so
的优先级顺序
- RPATH: 写在elf文件中
- LD_LIBRARY_PATH: 环境变量
- RUNPATH: 写在elf文件中
ldconfig
的缓存: 配置/etc/ld.conf*
可改变- 默认的
/lib
,/usr/lib
可以看到,RPATH与RUNPATH中间隔着LD_LIBRARY_PATH。为了让用户可以通过修改LD_LIBRARY_PATH来指定.so
文件,大多数编译器都将输出的RPATH留空,并用RUNPATH代替RPATH。
查看RPATH
对于任意的elf文件,可以使用$ readelf -d xxx | grep 'R*PATH'
来查看。
结果有两类,一个是RPATH,另一个是RUNPATH。前文也说了,一般情况下,RPATH为空,而RUNPATH不为空。
RPATH中有个特殊的标识符$ORIGIN
。这个标识符代表elf文件自身所在的目录。当希望使用相对位置寻找.so
文件,就需要利用$ORIGIN
设置RPATH。多个路径之间使用冒号:
隔开。
设置RPATH
1、在gcc中,设置RPATH的办法很简单,就是设置linker的rpath选项:
gcc -Wl,-rpath,/your/rpath/ test.cpp
如果需要设置$ORIGIN
:
gcc -Wl,-rpath,'$ORIGIN/lib' test.cpp。
注意,虽然选项里写着RPATH,但它设置的还是RUNPATH。原因在前文有交代。
2、在Makefile中的写法如下:
CPPFLAGS= '-Wl,-rpath,$$ORIGIN:$$ORIGIN/../lib'
3、在CMake中,事情则有些不同。由于CMake需要包揽软件安装的事宜,因此CMake使用两个变量来控制RPATH:INSTALL_RPATH
和BUILD_RPATH
。
设置的办法是:
SET_TARGET_PROPERTIES(target PROPERTIES INSTALL_RPATH "$ORIGIN")
注意,在CMake中,多个RPATH使用分号隔开,而不是冒号。这是估计是因为冒号在CMake语法中有其他用途.
4、在QMake中,也可以设置RPATH:
unix:!mac{ QMAKE_LFLAGS += "-Wl,-rpath,\'\$$ORIGIN/\'" }
修改RPATH
chrpath -r "/lib64" xxx.so chrpath -r "\$ORIGIN" xxx.so
注意:
RPATH仅在unix系列的操作系统中有用,在windows下,动态库需要放在可执行程序同目录下或者环境变量path
指定搜索动态库的目录。