2018-06-06【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

在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_RPATHBUILD_RPATH
设置的办法是:

SET_TARGET_PROPERTIES(target
    PROPERTIES INSTALL_RPATH "$ORIGIN;/another/run/path")

注意,在CMake中,多个RPATH使用分号隔开,而不是冒号。这是估计是因为冒号在CMake语法中有其他用途。

你可能感兴趣的:(2018-06-06【RPATH是什么】)