AskOverflow.Dev

AskOverflow.Dev Logo AskOverflow.Dev Logo

AskOverflow.Dev Navigation

  • 主页
  • 系统&网络
  • Ubuntu
  • Unix
  • DBA
  • Computer
  • Coding
  • LangChain

Mobile menu

Close
  • 主页
  • 系统&网络
    • 最新
    • 热门
    • 标签
  • Ubuntu
    • 最新
    • 热门
    • 标签
  • Unix
    • 最新
    • 标签
  • DBA
    • 最新
    • 标签
  • Computer
    • 最新
    • 标签
  • Coding
    • 最新
    • 标签
主页 / coding / 问题

问题[cmake](coding)

Martin Hope
neo742
Asked: 2025-04-09 15:31:11 +0800 CST

为什么第一个是真,而下一个是假?

  • 7
cmake_minimum_required(VERSION 3.20)

set(on "OFF")

if(on)
    message("ON") 
else()
    message("OFF")
endif()

if(${on}) 
    message("ON")
else()
    message("OFF") 
endif()
$ cmake -P test.cmake
ON
OFF

基本表达式

如果常量为 1、ON、YES、TRUE、Y 或非零数字(包括浮点数),则为 True。如果常量为 0、OFF、NO、FALSE、N、IGNORE、NOTFOUND、空字符串或以 -NOTFOUND 为后缀,则为 False。命名布尔常量不区分大小写。如果参数不是这些特定常量之一,则将其视为变量或字符串(参见下文的变量扩展),并采用以下两种形式之一。

我对这个问题的理解是

对于预定义常量 ON,它相当于 on。

对于if(on),on将被解析为预定义常量ON。

对于if(${on}),${on}将被解析为“OFF”,相当于OFF。

不知道我的理解是否正确。

什么是on?

cmake
  • 1 个回答
  • 43 Views
Martin Hope
David Razzetti
Asked: 2025-04-07 20:45:48 +0800 CST

如何在 CMake 中从“tool_requires”使用 Conan 管理的工具?

  • 5

Linux 上的 C++ 项目使用 Conan (v2) 作为依赖项。在 conanfile.py 中使用 Conan 软件包 libavrocpp/1.11.3 时:

def requirements(self):
    self.requires("libavrocpp/1.11.3")

def build_requirements(self):
    self.tool_requires("libavrocpp/1.11.3")

然后,我想使用包中的 avrogencpp 工具在 CMake 期间从我的 avro 模式生成 C++ 类。

我可以看到该工具存在于本地 Conan 缓存 (~/.conan2) 中的多个位置,我需要从那里以某种方式引用它。我可以在 CMake 文件中硬编码 avrogencpp 的路径,但这会在我下次将软件包升级到较新版本时,或者 Conan 作者修改缓存布局时(缓存布局是专有的,可能会更改)中断。所以:

使用 Conan 软件包时,从 CMake 文件引用 avrogencpp 工具的正确方法(特定于版本)是什么?是否有指向它的符号定义?

(CMake 有 FindProtobuf,它定义符号,包括一个指向特定于版本的 protoc 编译器的符号。Avro 有类似的东西吗?我到目前为止还没有找到它......)

cmake
  • 1 个回答
  • 35 Views
Martin Hope
sdbbs
Asked: 2025-02-19 23:59:02 +0800 CST

强制 CMake 在 CMakeLists.txt 中的任意点生成构建文件(Makefile)?

  • 5

CMakeLists.txt我的raspberrypi/pico-sdk项目中有此代码片段,我有条件地调用该pico_sdk_init()函数(即宏):

# ...

if(myCONDITION STREQUAL "Something")

  message("*** BEFORE pico_sdk_init() ***")
  pico_sdk_init()
  
  message("*** BEFORE return() ***")
  return() # exit for debug
  
  # other CMake conditional code commands continue here ...
  
endif() # myCONDITION STREQUAL "Something"

# ...

因此,我mkdir build && cd build在项目目录中执行CMakeLists.txt,然后运行cmake​​- 在这种情况下,我得到如下输出:

$ cmake ../ -DCMAKE_BUILD_TYPE=Debug -G "MSYS Makefiles"
...
*** BEFORE pico_sdk_init() ***
Build type is Debug
...
Using PICO_EXAMPLES_PATH from environment ...
*** BEFORE return() ***
-- Configuring done
-- Generating done
-- Build files have been written to: C:/path/to/myproject/build

在此之后,如果我从子文件夹检查./build,我会在Makefile那里(以及一大堆其他位置)看到:

$ find . -name Makefile
./Makefile
./pico-sdk/docs/Makefile
./pico-sdk/Makefile
./pico-sdk/src/common/boot_picobin_headers/Makefile
...
./pico-sdk/src/rp2_common/tinyusb/Makefile
./pico-sdk/tools/Makefile

...对于子文件Makefile夹./build,我有以下目标:

$ make #press [TAB] here
all                                              default_target
bs2_default/fast                                 depend
bs2_default_bin/fast                             edit_cache/fast
bs2_default_library/fast                         help
clean/fast                                       pioasmBuild/fast
cmake_check_build_system                         preinstall/fast
cmake_force                                      rebuild_cache/fast
cyw43_driver_picow_cyw43_bus_pio_spi_pio_h/fast

make pioasmBuild因此,我现在可以运行,并且它将构建良好:

$ make pioasmBuild
Scanning dependencies of target pioasmBuild
[ 12%] Creating directories for 'pioasmBuild'
[ 25%] No download step for 'pioasmBuild'
...
Scanning dependencies of target pioasm
[100%] Built target pioasm
[ 87%] Performing install step for 'pioasmBuild'
[100%] Built target pioasm
Install the project...
-- Install configuration: "Release"
[100%] Completed 'pioasmBuild'
[100%] Built target pioasmBuild

$ find . -name pioasm.exe
./pioasm/pioasm.exe
./pioasm-install/pioasm/pioasm.exe

伟大的。

现在,假设我不想使用return()- 退出调试,但是相反,在那时,我想运行等效的make pioasmBuild:因此从那时起,pioasm.exe可用于“其他 CMake 条件代码命令”;为此,我将代码片段重写为:

# ...

if(myCONDITION STREQUAL "Something")

  message("*** BEFORE pico_sdk_init() ***")
  pico_sdk_init()
  
  
  execute_process(
    # this command, the equivalent of `make pioasmBuild`, fails with "Error: could not load cache"
    COMMAND ${CMAKE_COMMAND} --build . --target pioasmBuild
    WORKING_DIRECTORY ${CMAKE_BINARY_DIR}
  )
  
  message("*** NOT anymore BEFORE return() ***")
  #return() # exit for debug
  
  # other CMake conditional code commands continue here ...
  
endif() # myCONDITION STREQUAL "Something"

# ...

正如评论所暗示的那样,make pioasmBuild此时的等效操作失败 -在子目录rm -rf *中清理之后./build,这是我在cmake输出中得到的内容:

$ cmake ../ -DCMAKE_BUILD_TYPE=Debug -G "MSYS Makefiles"
...
*** BEFORE pico_sdk_init() ***
Build type is Debug
...
Using PICO_EXAMPLES_PATH from environment ...
Error: could not load cache
*** NOT anymore BEFORE return() ***
... other CMake conditional code commands continue here ...

所以,据我所知,Error: could not load cache发生这种情况是因为此时子文件夹Makefile尚未创建!./build

事实上,据我目前对这个过程的理解:在前面的案例中,Makefile中的那个的创建是由提前退出本身触发的!./buildreturn()

所以我的问题是 - 是否有一种规范的方法可以强制 CMake 在CMakeLists.txt文件中的这个(或任意)点完成配置、生成和写入构建文件(使用此时可用的任何数据;并且至少在这种情况下,如前所述,该点有足够的数据来生成一个正常运行的Makefile) - 而不强制进程cmake在此时退出CMakeLists.txt,而是继续执行剩余的命令(包括可能Makefile直接从 CMake 在先前/早期生成的中构建目标)?

(当然,在这种“早期生成”的情况下,当CMakeLists.txt达到其原本“自然”的结尾时,所有Makefiles 和相关的构建文件都将被覆盖,可能会出现例如更多的目标,这是由于执行文件剩余部分而产生的更多数据)

cmake
  • 1 个回答
  • 13 Views
Martin Hope
tntnkn
Asked: 2025-01-14 00:56:59 +0800 CST

为什么目标属性不是从依赖项的属性中填充的

  • 5

CMake 文档说明如下:

通过从依赖项中读取目标属性的 INTERFACE_ 变体并将值附加到操作数的非 INTERFACE_ 变体来传播使用要求。

这似乎是错误的。请看下面的最小示例:

cmake_minimum_required(VERSION 3.31.3)

project(main)

add_library(lib OBJECT lib.cpp)
set_property(TARGET lib PROPERTY INTERFACE_COMPILE_DEFINITIONS "DEFS")

add_executable(main main.cpp)

target_link_libraries(main PUBLIC lib)


function(print_property)
    set(oneValueArgs TARGET PROPERTY)
    cmake_parse_arguments(PARSE_ARGV 0 arg "" "${oneValueArgs}" "")
    set(PROP "NONE")
    get_property(PROP TARGET ${arg_TARGET} PROPERTY ${arg_PROPERTY})
    message("${PROP}")
endfunction()

print_property(TARGET lib  PROPERTY INTERFACE_COMPILE_DEFINITIONS)
print_property(TARGET lib  PROPERTY COMPILE_DEFINITIONS)
print_property(TARGET main PROPERTY INTERFACE_COMPILE_DEFINITIONS)
print_property(TARGET main PROPERTY COMPILE_DEFINITIONS)

在配置阶段,只有第一条消息会打印一些内容(字符串DEFS),其余消息则不打印任何内容。

问题:根据问题开头的引文,COMPILE_DEFINITIONS的不应该main用 填充吗?这个问题也与的有点相关。DEFSINTERFACE_COMPILE_DEFINITIONSmain

建设阶段进展顺利 — —main.o正在-DDEFS按照预期进行建设。

cmake
  • 1 个回答
  • 14 Views
Martin Hope
FeFe
Asked: 2025-01-06 22:36:19 +0800 CST

如何在文件(GLOB ...)中查找带有 $<CONFIG> 的文件?

  • 5

我尝试使用 CMake 构建来查找特定文件,file(GLOB ...)同时考虑到$<CONFIG>确保根据构建配置选择了正确的文件版本。具体来说,我需要找到文件vc*-mtd.lib,其中*表示一个不同的版本号。

file(GLOB LIB "${BINARY_DIR}/lib/$<CONFIG>/vc*-mtd.lib")
add_custom_command(
    TARGET MyTarget POST_BUILD
    COMMAND ${CMAKE_COMMAND} -E copy_if_different
        ${LIB}
        ${CMAKE_BINARY_DIR}/lib/$<CONFIG>
)

但是,这种方法不起作用,因为$<CONFIG>似乎在 中不存在file(GLOB ...)。因此,glob 无法正确解析,我找不到直接的解决方案来使其工作。
如果file(GLOB ...)无法处理$<CONFIG>,在 CMake 中动态实现此目的的推荐方法是什么?

cmake
  • 2 个回答
  • 34 Views
Martin Hope
rpatters1
Asked: 2024-12-08 07:54:02 +0800 CST

CMake FetchContent 与仅包含标头的项目

  • 6

我正在尝试使用FetchContent包含一个依赖项,该依赖项是仅包含头文件的 C++ 项目。它不需要构建。但是,它包含一个构建其测试的 CMakeLists.txt 文件,并且它没有以标准方式执行。我想完全绕过 make 步骤,以免构建无关的二进制文件,也不会看到它发出的警告。

我的代码目前是:

FetchContent_Declare(
  TheProject
  GIT_REPOSITORY <the git repo>
  GIT_TAG <the git commit>
)
FetchContent_MakeAvailable(TheProject)

这可以正确下载项目,但它会调用 make 系统,而这并不是我想要的。我可以通过将其更改为以下内容来修复此问题:

FetchContent_Declare(
  TheProject
  GIT_REPOSITORY <the git repo>
  GIT_TAG <the git commit>
)
FetchContent_Populate(TheProject)

这有效,并且它不会调用构建系统。太棒了。但它发出一个诅咒警告,指出FetchContent_Populate单独调用已被弃用,并将从 cmake 的未来版本中删除。有没有一种合适的方法可以在不使用弃用功能的情况下完成我想要的功能?

cmake
  • 1 个回答
  • 24 Views
Martin Hope
Daniel Aviv
Asked: 2024-11-25 17:56:02 +0800 CST

buildPreset 和 testPreset 条目中的 configurePreset 的用途是什么?

  • 5

在 中,我可以在和中CMakePresets.json指定,但我不完全确定它实际上起了什么作用。configurePresetbuildPresettestPreset

例如:

{
  ...
  "configurePreset": {
    "name": "default",
    "toolchainFile": "$env{VCPKG_ROOT}/scripts/buildsystems/vcpkg.cmake",
    "binaryDir": "build",
    "warnings": {
      "dev": false
    },
    "cacheVariables": {
      "BUILD_TESTING": true
    }
  },
  "buildPreset": {
    "name": "debug",
    "configurePreset": "default",
    "configuration": "Debug",
    "jobs": 20
  }
  ...
}

但是,以下命令不起作用:

cmake --build --preset debug

这是因为我需要先配置项目。

cmake
  • 1 个回答
  • 15 Views
Martin Hope
uuu777
Asked: 2024-11-16 01:39:17 +0800 CST

通过 make 传递 cpplint 命令行属性

  • 5

我有一个小的 cmake 项目。我将 cpplint 添加到编译中,它运行良好。我正在向 cpplint 调用添加一个命令行参数。

find_program(CPPLINT_EXECUTABLE cpplint)

if(CPPLINT_EXECUTABLE)
    set(CMAKE_CXX_CPPLINT "${CPPLINT_EXECUTABLE} --quiet")
    set(CMAKE_CXX_CPPLINT_EXTENSIONS "*.cc; *.h")
 else()
    message(WARNING "cpplint not found.")
endif()

但失败了。我运行的是 3.30.2 版的 cmake

Error running '/usr/local/bin/cpplint --quiet': no such file or directory
make[2]: *** [src/CMakeFiles/libxdemo.dir/xdemo.cc.o] Error 1
make[1]: *** [src/CMakeFiles/libxdemo.dir/all] Error 2
cmake
  • 1 个回答
  • 20 Views
Martin Hope
pacmaninbw
Asked: 2024-11-14 01:06:26 +0800 CST

防止 CMake 项目中的并行构建

  • 7

我有一个包含 2 个子目录的顶级 CMakeLists.txt 文件。只要 cmake 命令行上没有 -j 或 --parallel,它就可以正确构建。CMakeLists.txt 文件记录了这个问题。除了记录问题之外,还有其他方法可以防止 CMakeLists.txt 文件中的并行构建吗?

cmake_minimum_required(VERSION 3.25)

# Do not attempt to build in parallel using either --parallel or -j
# There can be a race codition between generating the performance test
# header file and building the tests.

project(TestDictionary LANGUAGES CXX)

set(CMAKE_CXX_STANDARD 23)
set(CMAKE_CXX_STANDARD_REQUIRED ON)

add_subdirectory(src/PerformanceTestGenerator)
add_subdirectory(src/tests)
cmake_minimum_required(VERSION 3.25)

project(createPerformanceTest LANGUAGES CXX)

set(CMAKE_CXX_STANDARD 23)
set(CMAKE_CXX_STANDARD_REQUIRED ON)

add_executable(createPerformanceTest main.cpp PerformanceTestGenerator.cpp TestParameters.cpp)

target_compile_features(createPerformanceTest PUBLIC cxx_std_23)

if (CMAKE_CXX_COMPILER_ID MATCHES "MSVC")
    # warning level 4 and all warnings as errors
    target_compile_options(createPerformanceTest PRIVATE /W3 /WX -D_CRT_SECURE_NO_WARNINGS)
elseif (CMAKE_CXX_COMPILER_ID MATCHES "GNU|Clang")
    if("${CMAKE_CXX_COMPILER_ID}" STREQUAL "GNU")
        # require at least gcc 12 to get full C++20 support
        if (CMAKE_CXX_COMPILER_VERSION VERSION_LESS 12)
            message(FATAL_ERROR "GCC version must be at least 12!")
        endif()
    endif()
    # lots of warnings and all warnings as errors
    target_compile_options(createPerformanceTest PRIVATE -Wall -Wextra -pedantic -Werror)
endif()

add_custom_command(
  OUTPUT ${CMAKE_CURRENT_SOURCE_DIR}/performanceTest.h
  COMMAND createPerformanceTest --output-file ${CMAKE_CURRENT_SOURCE_DIR}/performanceTest.h --test-count 3 --radix-16 --test-size 10 --test-size 20 --test-size ff --both-tests
  COMMENT "Generating performanceTest.h header file"
  VERBATIM
)

add_custom_target(
  GeneratedHeader ALL
  DEPENDS ${CMAKE_CURRENT_SOURCE_DIR}/performanceTest.h
)
cmake_minimum_required(VERSION 3.25)

project(testTest LANGUAGES CXX)

set(CMAKE_CXX_STANDARD 23)
set(CMAKE_CXX_STANDARD_REQUIRED ON)

add_executable(testTest testtest.cpp FunctionalityTests.cpp TestGenericDictionary.cpp)

target_include_directories(testTest
    PUBLIC ../..
    PRIVATE ./ ../PerformanceTestGenerator 
)

target_compile_features(testTest PUBLIC cxx_std_23)

if (CMAKE_CXX_COMPILER_ID MATCHES "MSVC")
    # warning level 4 and all warnings as errors
    target_compile_options(testTest PRIVATE /W4 /WX -D_CRT_SECURE_NO_WARNINGS)
elseif (CMAKE_CXX_COMPILER_ID MATCHES "GNU|Clang")
    if("${CMAKE_CXX_COMPILER_ID}" STREQUAL "GNU")
        # require at least gcc 12 to get full C++20 support
        if (CMAKE_CXX_COMPILER_VERSION VERSION_LESS 12)
            message(FATAL_ERROR "GCC version must be at least 12!")
        endif()
    endif()
    # lots of warnings and all warnings as errors
    target_compile_options(testTest PRIVATE -Wall -Wextra -pedantic -Werror)
endif()
cmake
  • 1 个回答
  • 37 Views
Martin Hope
user1785730
Asked: 2024-09-25 03:20:25 +0800 CST

FetchContent 的官方文档没有描述其所有选项

  • 6

FetchContent 的官方文档看起来更像是一个概述。但我正在寻找完整的参考。cmake --help-module FetchContent产生与上述链接相同的文本。

在某些项目中我看到了这种情况,例如:GIT_SHALLOW TRUE。在这种情况下:

FetchContent_Declare(
    box2d
    GIT_REPOSITORY https://github.com/erincatto/box2d.git
    GIT_TAG 7b273b12409252d247b89940d29a8aab0386f570
    GIT_SHALLOW TRUE
    GIT_PROGRESS TRUE
)

我想查看它的文档。其他的想法我也没找到文档。

cmake
  • 1 个回答
  • 20 Views

Sidebar

Stats

  • 问题 205573
  • 回答 270741
  • 最佳答案 135370
  • 用户 68524
  • 热门
  • 回答
  • Marko Smith

    重新格式化数字,在固定位置插入分隔符

    • 6 个回答
  • Marko Smith

    为什么 C++20 概念会导致循环约束错误,而老式的 SFINAE 不会?

    • 2 个回答
  • Marko Smith

    VScode 自动卸载扩展的问题(Material 主题)

    • 2 个回答
  • Marko Smith

    Vue 3:创建时出错“预期标识符但发现‘导入’”[重复]

    • 1 个回答
  • Marko Smith

    具有指定基础类型但没有枚举器的“枚举类”的用途是什么?

    • 1 个回答
  • Marko Smith

    如何修复未手动导入的模块的 MODULE_NOT_FOUND 错误?

    • 6 个回答
  • Marko Smith

    `(表达式,左值) = 右值` 在 C 或 C++ 中是有效的赋值吗?为什么有些编译器会接受/拒绝它?

    • 3 个回答
  • Marko Smith

    在 C++ 中,一个不执行任何操作的空程序需要 204KB 的堆,但在 C 中则不需要

    • 1 个回答
  • Marko Smith

    PowerBI 目前与 BigQuery 不兼容:Simba 驱动程序与 Windows 更新有关

    • 2 个回答
  • Marko Smith

    AdMob:MobileAds.initialize() - 对于某些设备,“java.lang.Integer 无法转换为 java.lang.String”

    • 1 个回答
  • Martin Hope
    Fantastic Mr Fox msvc std::vector 实现中仅不接受可复制类型 2025-04-23 06:40:49 +0800 CST
  • Martin Hope
    Howard Hinnant 使用 chrono 查找下一个工作日 2025-04-21 08:30:25 +0800 CST
  • Martin Hope
    Fedor 构造函数的成员初始化程序可以包含另一个成员的初始化吗? 2025-04-15 01:01:44 +0800 CST
  • Martin Hope
    Petr Filipský 为什么 C++20 概念会导致循环约束错误,而老式的 SFINAE 不会? 2025-03-23 21:39:40 +0800 CST
  • Martin Hope
    Catskul C++20 是否进行了更改,允许从已知绑定数组“type(&)[N]”转换为未知绑定数组“type(&)[]”? 2025-03-04 06:57:53 +0800 CST
  • Martin Hope
    Stefan Pochmann 为什么 {2,3,10} 和 {x,3,10} (x=2) 的顺序不同? 2025-01-13 23:24:07 +0800 CST
  • Martin Hope
    Chad Feller 在 5.2 版中,bash 条件语句中的 [[ .. ]] 中的分号现在是可选的吗? 2024-10-21 05:50:33 +0800 CST
  • Martin Hope
    Wrench 为什么双破折号 (--) 会导致此 MariaDB 子句评估为 true? 2024-05-05 13:37:20 +0800 CST
  • Martin Hope
    Waket Zheng 为什么 `dict(id=1, **{'id': 2})` 有时会引发 `KeyError: 'id'` 而不是 TypeError? 2024-05-04 14:19:19 +0800 CST
  • Martin Hope
    user924 AdMob:MobileAds.initialize() - 对于某些设备,“java.lang.Integer 无法转换为 java.lang.String” 2024-03-20 03:12:31 +0800 CST

热门标签

python javascript c++ c# java typescript sql reactjs html

Explore

  • 主页
  • 问题
    • 最新
    • 热门
  • 标签
  • 帮助

Footer

AskOverflow.Dev

关于我们

  • 关于我们
  • 联系我们

Legal Stuff

  • Privacy Policy

Language

  • Pt
  • Server
  • Unix

© 2023 AskOverflow.DEV All Rights Reserve