RPATH

RPATH是什么


简介

rpath 在linux中的作用等

一、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
如果需要设置 O R I G I N : ‘ ORIGIN:` 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语法中有其他用途

在 CMake 构建过程中,`RPATH`(运行时库查找路径)是一个非常重要的概念,它决定了生成的可执行文件或共享库在运行时如何定位其依赖的动态链接库。合理配置 RPATH 可以避免因找不到所需库而导致的问题,并提高应用的移植性和可靠性。 ### 什么是 `RPATH`? `RPATH` 是嵌入到 ELF 文件(如 Linux 上的二进制文件和共享库)中的一个属性,用于指定加载器在启动程序时应该去哪里寻找所需的共享库。通过设置 `RPATH`,可以在不需要修改系统范围内的配置(如 `/etc/ld.so.conf.d/` 或者环境变量 `LD_LIBRARY_PATH` 的情况下),使得应用程序能够找到并正确加载所需要的外部库。 ### 在 CMake 中管理 `RPATH` CMake 提供了几种机制来管理和控制构建结果中的 `RPATH` 行为: #### 自动处理 默认情况下,当您使用 CMake 来创建目标(即编译项目),如果启用了 `BUILD_WITH_INSTALL_RPATH` 属性并且设置了有效的安装前缀 (`CMAKE_INSTALL_PREFIX`) ,那么 CMake 将会自动调整输出文件的 `RPATH` 。这意味着所生成的目标将会包含指向正确的安装目录下共享库的位置信息。 ```cmake set(CMAKE_BUILD_WITH_INSTALL_RPATH TRUE) ``` 此外,默认情况下,CMake 还会在构建阶段为目标添加源码树相关的相对路径作为初始 `RPATH` (除非明确关闭了该功能)。这有助于调试及测试未完成打包的应用程序实例。 #### 手动设置 `RPATH` 有时我们希望更精细地控制最终产品里的 `RPATH` 内容。可以利用以下变量来进行定制化操作: - **`SET_TARGET_PROPERTIES()`**: 直接对某个具体的目标设定特定的 `RPATH` 值; ```cmake set_target_properties(mytarget PROPERTIES INSTALL_RPATH "/opt/myapp/lib") ``` - **`CMAKE_INSTALL_RPATH`**: 定义所有待安装项目的公共 `RPATH`; ```cmake set(CMAKE_INSTALL_RPATH "${CMAKE_INSTALL_PREFIX}/lib:${ANOTHER_DIR}") ``` - **清理不必要的 `RPATH`** : 若想去除所有的预设 `RPATH` 并完全由自己掌控,可通过下面的方式实现; ```cmake # 移除现有 RPATHs foreach(target ${PROJECT_NAME}) get_target_property(_rpaths ${target} IMPORTED_CONFIGURATIONS) if (_rpaths) unset_target_properties(${target} PROPERTIES INSTALL_RPATH "") endif() endforeach() # 配置新的 RPATH set_target_properties(your_executable_name PROPERTIES SKIP_BUILD_RPATH TRUE) set_target_properties(your_executable_name PROPERTIES BUILD_WITH_INSTALL_RPATH FALSE) set_target_properties(your_executable_name PROPERTIES INSTALL_RPATH_USE_LINK_PATH FALSE) set_target_properties(your_executable_name PROPERTIES RUNTIME_OUTPUT_DIRECTORY "${CMAKE_BINARY_DIR}/bin") set_target_properties(your_executable_name PROPERTIES LIBRARY_OUTPUT_DIRECTORY "${CMAKE_BINARY_DIR}/lib") ``` 请注意,在实际部署之前应当考虑是否真的需要保留硬编码形式存在的绝对路径型 `RPATH`, 因为它们可能会降低软件包跨平台分发的能力. --- 为了更好地理解 `RPATH` 和相关设置的影响,您可以尝试查看已有的二进制文件中包含了哪些 `RPATH` 项以及相应的优先级顺序等细节。常用的工具包括 `chrpath` 和 GNU Binutils 包含的 `readelf`.
评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值