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_RPATHBUILD_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指定搜索动态库的目录。

posted @ 2022-08-05 09:17  禅元天道  阅读(1876)  评论(0编辑  收藏  举报