什么是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
在gcc中,设置RPATH的办法很简单,就是设置linker的rpath选项:$ gcc -Wl,-rpath,/your/rpath/ test.cpp
如果需要设置$ORIGIN
:$ gcc -Wl,-rpath,'$ORIGIN/lib' test.cpp
。
注意,虽然选项里写着RPATH,但它设置的还是RUNPATH。原因在前文有交代。
在CMake中,事情则有些不同。由于CMake需要包揽软件安装的事宜,因此CMake使用两个变量来控制RPATH:INSTALL_RPATH
和BUILD_RPATH
。
设置的办法是:
SET_TARGET_PROPERTIES(target
PROPERTIES INSTALL_RPATH "$ORIGIN;/another/run/path")
注意,在CMake中,多个RPATH使用分号隔开,而不是冒号。这是估计是因为冒号在CMake语法中有其他用途。