我正在尝试为我正在处理的图书馆项目创建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.cppfoo
和src/bar.cppbar
,显然将源列表中的最终文件名放在可执行文件的前面。
这是由我在使用foreach
或调用add_executable
时不知道的原因引起的吗? message(STATUS ${testName})
调用将打印出我所期望的内容,仅显示不带路径或扩展名的文件名。我还尝试了在${testName}
中引用/不引用add_executable
以及从NAME
中删除COMMAND
和add_test
的各种组合,但是我没有认为这是其中的一部分。这是我的第一个使用CMake的项目,但为什么会这样,我有点迷茫。
是什么导致最终源文件名的开头,我该如何避免呢?
答案 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_)
我猜这会创建一个变量,其名称为空字符串,但是为什么这会影响生成的二进制文件的名称呢?
只需放置dummy
或match_output
或任何其他内容,问题就会消失:
string(REGEX MATCH "^.*/src/(.*)\\.[^.]*$" match_output ${src})