探讨CMake中关于RPATH的使用

 最近研究CMake,发现CMake对于RPATH的管理也非常人性化。官方说法是当动态库的编译也和执行档在同级目录下的时候,CMake会自动给执行档加入适当的RPATH.具体可以通过readelf -d executable来查看相关的RPATH路径。所以有的时候用QMake编译之后还得手动加一把LD_LIBRARY_PATH,而用CMake编译之后,直接执行执行档就可以了。

    相关RPATH的说明,同学可以自行google学习,简单来说,RPATH就是在LD_LIBRARY_PATH之前,会优先让执行档去寻找相应的动态库,当然了有的操作系统支持RUNPATH的话,会在RUNPATH设置情况下自动忽略RPATH,而会先去寻找LD_LIBRARY_PATH之后再去着RUNPATH,(注意这里面的顺序关系,RUNPATH在LD_LIBRARY_PATH之后,而其会让RPATH忽略,但RPATH却在LD_LIBRARY_PATH之前),相关顺序这里简单交代下: RPATH   --   LD_LIBRARY_PATH -- RUNPATH(出现会屏蔽RPATH) -- /etc/ld.so.conf -- builtin dircetories(/lib, /usr/lib)。

    在这里我主要介绍下CMake对于RPATH的相关管控。默认行为已经在第一段说明了,那就是CMake在默认情况下是会给你的exe加入相关RPATH的,这个时候可能会有不想要相关RPATH的需求,仔细查阅寻找之后发现,CMake里面维护了3个比较重要的RPATH变量,即CMAKE_SKIP_RPATH,CMAKE_SKIP_BUILD_RPATH,CMKAE_INSTALL_RPATH.

    对于第一个变量CMAKE_SKIP_RPATH,简单的来说,就是强制CMake不在构建期间和安装install期间给你加上它所认为的RPATH.

cmake .. -DCMAKE_SKIP_RPATH=TRUE

   第二个和第三个变量也比较简单,就是分别在构建期间和安装期间不允许CMake给你加入相关RPATH

cmake .. -DCMAKE_SKIP_BUILD_RPATH=TRUE
cmake .. -DCMAKE_SKIP_INSTALL_RPATH=TRUE

当然了,如果你之后想要追加RPATH,只需要对这三个变量设置成FALSE就可以了。

 

参考链接: www.cmake.org/Wiki/CMake_RPATH_handling#No_RPATH_at_all

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`.
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值