gpt4 book ai didi

cmake - 如何在 CMake 中构建过程后复制目标的所有运行时依赖项?

转载 作者:行者123 更新时间:2023-12-05 04:47:08 26 4
gpt4 key购买 nike

我正在尝试将可执行目标的运行时依赖项自动复制到其构建的目录中。

例如,我使用 GTest 进行链接为共享库的单元测试。构建之后,如果我尝试启动单元测试,将会出现类似“gtest.dll isn't found”的错误。对于其他外部依赖项,存在同样的问题(对于目标是共享库的本地)。

我可以在链接后将每个外部库添加到安装过程并在该目录中运行测试,但这不是启动 CTest 的实用方法。

我想使用 file(GET_RUNTIME_DEPENDENCIES) 但它必须在构建之后调用,而且我没有找到 INSTALL(CODE) 模拟 POST_BUILD 阶段。

仅使用 add_custom_command 处理 DLL 也行不通,因为可能存在依赖继承。例如,A 依赖于 B,而 B 又依赖于 C。如果我尝试通过这种方法进行依赖项复制,那么只有 B 会在 APOST_BUILD 过程中被复制。

我不喜欢将所有构建的工件存储在一个目录中,因为它会造成困惑,并可能在未来造成冲突。

最佳答案

找到了解决方案,但它不可移植且只能在 Windows 中使用。

CMake 3.21 版中有一个新的 TARGET_RUNTIME_DLLS生成器表达式,它以文件路径列表的形式返回运行时目标的所有运行时依赖项。该表达式可用于 ADD_CUSTOM_COMMAND命令在 POST_BUILD 阶段。

项目中有共享库SomeLib 和测试可执行文件SomeLib_tests 的目标。 GTest 共享库的目标由 FIND_PACKAGE 找到功能。

cmake_minimum_required(VERSION 3.21)
project(ExampleProject)

add_library(SomeLib SHARED
"include/SomeLib/api.h"
"src/SomeLib/api.cpp")
target_include_directories(SomeLib PUBLIC "${CMAKE_CURRENT_SOURCE_DIR}/include")


find_module(GTest COMPONENTS gtest gtest_main REQUIRED)

add_executable(SomeLib_tests
"ut/api_tests.cpp")
target_link_libraries(SomeLib_tests
PRIVATE SomeLib
PRIVATE GTest::gtest GTest::gtest_main)

gtest_add_tests(TARGET SomeLib_tests)

在这一步,如果我在Windows下调用CTest会出现类似“gtest.dll isn't found”的错误。

需要将可执行文件的所有依赖项复制到它的目录中。可以通过调用ADD_CUSTOM_COMMAND来实现结合TARGET_RUNTIME_DLLS表达。目标目录来自TARGET_FILE_DIR生成器表达式。

add_custom_command( TARGET SomeLib_tests POST_BUILD
COMMAND ${CMAKE_COMMAND} -E copy
$<TARGET_RUNTIME_DLLS:SomeLib_tests>
$<TARGET_FILE_DIR:SomeLib_tests>
COMMAND_EXPAND_LISTS)

这样,如果我构建项目并再次调用 CTest,测试将按预期运行。

关于cmake - 如何在 CMake 中构建过程后复制目标的所有运行时依赖项?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/68579934/

26 4 0
Copyright 2021 - 2024 cfsdn All Rights Reserved 蜀ICP备2022000587号
广告合作:1813099741@qq.com 6ren.com