CMake意外地将最后一个文件名添加到foreach中的可执行文件

时间:2019-06-25 13:59:59

标签: c++ cmake

我正在尝试为我正在处理的图书馆项目创建CMakeLists.txt。这是一个相当简单的结构,通过以下设置,我已经能够重新创建我遇到的问题的简单示例:

project
├── CMakeLists.txt
├── include
│   ├── doctest.hpp
│   └── foo.hpp
└── src
    ├── foo.cpp
    ├── bar.cpp
    └── test-main.cpp

,示例代码在this gist中可用(_的要点由/代替)。 C ++源代码和标头实际上并不影响该问题,因此我不包括它们的内容,但是我的CMakeLists.txt实际上是这样的:

cmake_minimum_required(VERSION 3.13)
project(foo VERSION 0.0.1)

set(source_files
        ${CMAKE_SOURCE_DIR}/src/foo.cpp
        ${CMAKE_SOURCE_DIR}/src/bar.cpp
)

set(header_files
        ${CMAKE_SOURCE_DIR}/include/foo.hpp
)

set(DOCTEST_INCLUDE_DIR ${CMAKE_SOURCE_DIR}/include)
add_library(Doctest IMPORTED INTERFACE)
target_include_directories(Doctest INTERFACE ${DOCTEST_INCLUDE_DIR})

add_library(test-main OBJECT ${CMAKE_SOURCE_DIR}/src/test-main.cpp)
target_link_libraries(test-main Doctest)

foreach(src ${source_files})
        # Strip off leading path and extension, leaving only filename without extension to be ${testName}
        string(REGEX MATCH "^.*/src/(.*)\\.[^.]*$" "" ${src})
        set(testName ${CMAKE_MATCH_1})
        message(STATUS ${testName})

        add_executable("${testName}" "${src}" $<TARGET_OBJECTS:test-main>)
        target_include_directories(${testName} PRIVATE ${CMAKE_SOURCE_DIR}/include)
        add_test(NAME ${testName} COMMAND ${testName})
endforeach()

问题在于,尽管cmake .make一样运行正常,但输出可执行文件却不是我期望的src/[filenameWithoutExtensionHere](来自line 24 in CMakeLists)它最终以 src/bar.cppfoosrc/bar.cppbar,显然将源列表中的最终文件名放在可执行文件的前面。

这是由我在使用foreach或调用add_executable时不知道的原因引起的吗? message(STATUS ${testName})调用将打印出我所期望的内容,仅显示不带路径或扩展名的文件名。我还尝试了在${testName}中引用/不引用add_executable以及从NAME中删除COMMANDadd_test的各种组合,但是我没有认为这是其中的一部分。这是我的第一个使用CMake的项目,但为什么会这样,我有点迷茫。

是什么导致最终源文件名的开头,我该如何避免呢?

1 个答案:

答案 0 :(得分:1)

这是由行触发的

string(REGEX MATCH "^.*/src/(.*)\\.[^.]*$" "" ${src})

通过为""指定<output variable>,您似乎试图忽略它,因为您只关心捕获组...但是这似乎有一些非常奇怪和不幸的副作用。您可以通过最小的CMakeLists.txt来获得相同的效果:

cmake_minimum_required(VERSION 3.13)
add_executable(foo ${CMAKE_SOURCE_DIR}/src/foo.cpp)
set("" some_prefix_)

我猜这会创建一个变量,其名称为空字符串,但是为什么这会影响生成的二进制文件的名称呢?

只需放置dummymatch_output或任何其他内容,问题就会消失:

string(REGEX MATCH "^.*/src/(.*)\\.[^.]*$" match_output ${src})