c++ - CMake 为 VS 项目生成循环依赖项,但不生成 make 文件。如何避免?

标签 c++ visual-studio makefile cmake

这个示例是从我正在从事的一个实际项目中总结出来的;某种 MWE(好吧,工作会更合适)。

考虑以下项目结构:

.
├── CMakeLists.txt
├── build
├── include
│   └── hello.hpp
└── src
    └── hello.cpp

build 目录用于进行构建,并将包含构建工件和中间文件。其余文件的内容将在本问题的底部给出。

这个想法是 install() 构建的库及其 header (以及实际项目中的其他一些内容)。

add_custom_command(
    TARGET ${PRJNAME}
    POST_BUILD
        COMMAND ${CMAKE_COMMAND} --build . --target install
    WORKING_DIRECTORY "${CMAKE_BINARY_DIR}"
    VERBATIM
)

... 与 install() 一起用于完成此任务,并且适用于生成器 Unix MakefilesNMake Makefiles,但是为 Visual Studio 16 2019 创建循环依赖项(也适用于 Visual Studio 14 2015,但我没有进行任何进一步的测试)。

现在我的第一 react 是,这是生成器之间的不一致,但另一方面,除了 make 文件之外的任何项目生成 - IMO - 一直是 CMake 的弱点之一。

目前,我的解决方法只是使用 NMake Makefiles,但该方法的缺点是我必须先“检测”Visual Studio 并使用 vcvarsall.bat >、vcvars32.bat 和 friend 们。但与循环依赖相比,这只是一个小麻烦。

如何使用 Visual Studio 项目生成器之一来实现我想要的并避免循环依赖?

在 Windows 上,我使用的是 CMake 3.15.1(撰写本文时最新版本)。

注意:我意识到我可以使用一些东西来达到 ${CMAKE_COMMAND} --build 的效果。 --target install 通过创建自定义命令/目标,并撒上一些 file() 命令。但这并不能保留它DRY现在,可以吗?


CMakeLists.txt

set(CMAKE_RULE_MESSAGES OFF)
set(CMAKE_VERBOSE_MAKEFILE ON)

cmake_minimum_required(VERSION 3.12 FATAL_ERROR)
set(PRJNAME FOOBAR)

project (${PRJNAME})

set(SOURCE_DIR "src")
set(HEADER_DIR "include")
set(TARGET_DIR "${CMAKE_CURRENT_BINARY_DIR}/install-target")
set(PUBLIC_HEADER "${HEADER_DIR}/hello.hpp")
set(PROJ_SOURCES "${SOURCE_DIR}/hello.cpp")

add_library(${PRJNAME} SHARED ${PROJ_SOURCES})

list(TRANSFORM PUBLIC_HEADER PREPEND "${CMAKE_CURRENT_BINARY_DIR}/" OUTPUT_VARIABLE PUBLIC_HEADER_PP)
set_target_properties(
    ${PRJNAME}
    PROPERTIES
        CXX_STANDARD 11
        CXX_EXTENSIONS OFF
        CXX_STANDARD_REQUIRED ON
        PUBLIC_HEADER "${PUBLIC_HEADER_PP}"
        POSITION_INDEPENDENT_CODE 1
)

set(CMAKE_INSTALL_PREFIX ${TARGET_DIR})
set(CMAKE_INSTALL_LIBDIR lib)
set(CMAKE_INSTALL_INCLUDEDIR ${HEADER_DIR})

install(
    TARGETS ${PRJNAME}
    ARCHIVE DESTINATION ${CMAKE_INSTALL_LIBDIR}
    LIBRARY DESTINATION ${CMAKE_INSTALL_LIBDIR}
    RUNTIME DESTINATION ${CMAKE_INSTALL_LIBDIR}
    PUBLIC_HEADER DESTINATION ${CMAKE_INSTALL_INCLUDEDIR}
)

file(MAKE_DIRECTORY "${CMAKE_CURRENT_BINARY_DIR}/${HEADER_DIR}")
configure_file("${PUBLIC_HEADER}" "${CMAKE_CURRENT_BINARY_DIR}/${HEADER_DIR}/" COPYONLY)
include_directories("${CMAKE_CURRENT_BINARY_DIR}/${HEADER_DIR}")

add_custom_command(
    TARGET ${PRJNAME}
    POST_BUILD
        COMMAND ${CMAKE_COMMAND} --build . --target install
    WORKING_DIRECTORY "${CMAKE_BINARY_DIR}"
    VERBATIM
)

set(PKGNAME "foobar-package.zip")

add_custom_command(
    OUTPUT ${CMAKE_CURRENT_BINARY_DIR}/${PKGNAME}
        COMMAND ${CMAKE_COMMAND} -E tar cv "${CMAKE_CURRENT_BINARY_DIR}/${PKGNAME}" -- .
    WORKING_DIRECTORY "${TARGET_DIR}"
    VERBATIM
)

set_property(DIRECTORY APPEND PROPERTY ADDITIONAL_MAKE_CLEAN_FILES ${CMAKE_CURRENT_BINARY_DIR}/${PKGNAME})

add_custom_target(
    lib
    DEPENDS ${PRJNAME}
)

src/hello.cpp

#ifdef _WIN32
#define WIN32_LEAN_AND_MEAN
#include <Windows.h>

BOOL APIENTRY DllMain(HMODULE, DWORD, LPVOID) { return TRUE; }
#endif
#include "hello.hpp"

int hello() { return 42; }

include/hello.cpp

#pragma once

#ifdef _WIN32
#   ifdef FOOBAR_EXPORTS
#       define FOOBAR_API  __declspec(dllexport)
#   else
#       define FOOBAR_API  __declspec(dllimport)
#   endif
#else
#   define FOOBAR_API
#endif

FOOBAR_API int hello();

最佳答案

在我看来,您的用例超出了 CMake 的设计目的。 您正在尝试构建 INSTALL 目标,该目标又构建 FOOBAR 目标,后者又调用 INSTALL 命令。

如果我们通过分成两个不同的目标来打破这里的链条:FOOBARFOOBAR_INSTALL 像这样

add_custom_target(${PRJNAME}_INSTALL ALL
    ${CMAKE_COMMAND} --build . --target install
    WORKING_DIRECTORY "${CMAKE_BINARY_DIR}"
    VERBATIM
)

我们仍然可以通过ALL_BUILD目标获得循环引用。 因此,如果我们删除 ALL 参数,引用将被破坏,但不会从默认构建命令中调用目标,这并不比调用 make 更好。进行安装

打破依赖关系的正确位置是在INSTALL 和其他目标之间。您可以直接调用生成的 cmake_install.cmake 脚本,而不是构建 INSTALL 目标:

add_custom_command(
    TARGET ${PRJNAME}
    POST_BUILD
        COMMAND ${CMAKE_COMMAND} -D CMAKE_INSTALL_CONFIG_NAME=$<CONFIG> -P cmake_install.cmake
    WORKING_DIRECTORY "${CMAKE_BINARY_DIR}"
    VERBATIM
)

关于c++ - CMake 为 VS 项目生成循环依赖项,但不生成 make 文件。如何避免?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/57354368/

相关文章:

c++ - 使用 boost::archive::binary_iarchive 的内存泄漏

c++ - 了解如何创建 atoi;字符是如何比较的?

c# - 可以自动增加DLL版本

c# - 使用usings在asp中发送电子邮件

Makefile处理同扩展名的文件

c++ - Mixin和接口(interface)实现

visual-studio - 如何在 Java 项目中的 VS Code 中悬停时显示 Javadoc?

gcc - 如何在 gcc 和 make 中使用预编译头?

makefile - 当 grep 输出为空时,"grep THIS foo.txt > THIS.txt"在 Makefile 中给出错误,而不是在 bash 中给出错误

c++ - 析构函数无法删除已分配的连续内存块