5

我正在尝试使用 CMake 添加一个自定义目标,该目标为每个给定的 .cpp 文件执行一个命令。仅当源文件本身或包含的源文件之一发生更改时,才应重新执行该命令。AFAIK 要实现这一点,我需要一个所有包含文件的列表,并将它们添加到属于我的自定义目标DEPENDS的调用选项中。add_custom_command()

那么有没有一种内置的方法来获取包含文件的列表?

我知道IMPLICIT_DEPENDS该函数的选项,add_custom_command()但它仅适用于 Makefile 生成器。我想让这项工作适用于所有发电机。

感谢您的时间

编辑:

根据要求,我将发布一些 cmake 代码来展示我想要实现的目标。我想添加一个自定义目标,它在所有给定的 .cpp 文件上运行 clang-tidy。增量构建自定义目标时,只要更改 .cpp 文件或其直接或间接包含的头文件之一,就应该重新运行 clang-tidy 命令。就像处理编译器的重新运行一样。

# ----------------------------------------------------------------------------------------
# mainTargetName The name of the target that shall be analyzed
# files A list of all the main targets .cpp files
#
function( addStaticAnalysisTarget mainTargetName files )

    set(targetName runStaticAnalysis_${mainTargetName})
    set(command "clang-tidy-4.0 -checks=* -p ${CMAKE_BINARY_DIR}")

    foreach( file ${files}  )

        get_filename_component( baseName ${file} NAME_WE)
        set(stampFile ${CMAKE_CURRENT_BINARY_DIR}/analyze_${baseName}.stamp )
        set(fullFile ${CMAKE_CURRENT_SOURCE_DIR}/${file})
        set(commandWithFile "${command} ${fullFile}")
        separate_arguments_for_platform( commandList ${commandWithFile})

        add_custom_command(
            OUTPUT ${stampFile}
            DEPENDS "${fullFile}"
            IMPLICIT_DEPENDS CXX "${fullFile}"
            COMMAND ${commandList}
            COMMAND cmake -E touch "${stampFile}"       # without creating a file as a touch-stone the command will always be re-run.
            WORKING_DIRECTORY ${CPPCODEBASE_ROOT_DIR}
            COMMENT "${commandWithFile}"
            VERBATIM
        )

        list(APPEND stampFiles ${stampFile})

    endforeach()
    set_source_files_properties(${stampFiles} PROPERTIES GENERATED TRUE)   # make the stamp files known to cmake as generated files.

    add_custom_target(
        ${targetName}
        DEPENDS ${stampFiles}
    )

endfunction()

问题在于,它似乎不起作用。当我更改包含的文件时,不会为受影响的文件重新运行 clang-tidy。我在这个例子中使用了“Unix Makefile”生成器,所以它至少应该与 make 一起工作。任何提示为什么它没有?

我希望通过在 cmake 时以某种方式获取文件依赖项,然后将它们添加到 ''''DEPENDS'''' 列表中,我可以为所有生成器实现所需的行为。但是依赖扫描必须在每次运行命令时进行,所以不能在cmake时进行。这意味着扫描必须由当前不是的 cmake 实现。

有类似问题的人: https ://gitlab.kitware.com/cmake/cmake/issues/16830

编辑 2: 我认为 IMPLICIT_DEPENDS 选项不起作用的问题是因为我没有使用正确的文件名。我在代码中更改了它,但我还没有测试它是否在项目中工作。

4

1 回答 1

1

我想我的问题的答案是......

不,您不能在 cmake 代码中使用 cmakes 依赖扫描程序。

这是有道理的,因为这个问题不能在 cmake 时解决,因为 .cpp 文件的依赖关系可能会改变,而无需重新运行 cmake。该问题必须在 make 时在 cmake 本身内解决。这是在使用该IMPLICIT_DEPENDS选项时完成的。

另外,我试图解决一个我没有真正遇到的问题,因为此时我只能在 linux 上运行 clang-tidy。但是,clang-tidy 也可能在 Windows 上可用,然后我可能会再次遇到问题。

总结评论:

  • Tambre 表示 CMake 不是编译器,因此不能这样做。我认为这是错误的。根据这篇文章,CMake 可以解析 cpp 包含依赖项,因为 make 本身没有这样的依赖项搜索器。这对我来说是个新闻,但我主要生活在 Windows 上,所以我对 make 不太熟悉。与此同时,make 也有可能被扩展为进行自己的依赖搜索。这也解释了为什么该IMPLICIT_DEPENDS选项仅适用于 make。

  • Florian 指出,没有必要创建自己的自定义目标来运行 clang-tidy。相反,可以CXX_CLANG_TIDY在编译后使用 target 属性为每个文件运行 clang-tidy。然而,这意味着静态分析不能与构建分开,这可能导致构建时间不可接受。

  • cmake -E cmake_depends命令行,可用于在 cmake 时检索依赖项。但如上所述,我错误地认为我在 cmake 时需要依赖项,而在运行时需要它们。

  • 这些IMPLICIT_DEPENDS选项不起作用,因为我的 cmake 代码中有错误。

于 2017-06-28T05:41:58.727 回答