add_custom_command

时间:2021-07-06 15:49:22

标签: cmake cmake-language

如果我在 DEPENDSadd_custom_command 部分包含不存在的文件,它会失败。有没有办法指定依赖项可能不存在,但如果它被创建,命令应该重新运行?

具体来说,我使用的是 Coco/R,它要求 Parser.frameScanner.frame 文件与语法文件或由命令行参数指定的目录存在于同一目录中,并且 {{1 }} 可以存在与否。所以我这样做:

Copyright.frame

如果我在 foreach(FRAME_DIR IN ITEMS ${GRAMMAR_DIR} ${ARG_FRAME_DIR}) foreach(FRAME_FILE_NAME IN ITEMS Copyright Scanner Parser) set(FULL_FRAME_FILE ${FRAME_DIR}/${FRAME_FILE_NAME}.frame) if(EXISTS ${FULL_FRAME_FILE}) list(APPEND FRAME_FILES ${FULL_FRAME_FILE}) endif() endforeach() endforeach() ... set(GEN_SOURCES ${ARG_OUTPUT_DIR}/Scanner.h ${ARG_OUTPUT_DIR}/Parser.h ${ARG_OUTPUT_DIR}/Scanner.cpp ${ARG_OUTPUT_DIR}/Parser.cpp) add_custom_command(OUTPUT ${GEN_SOURCES} COMMAND ${COCOCPP} ${COCOCPP_ARGS} MAIN_DEPENDENCY ${ARG_GRAMMAR} DEPENDS ${FRAME_FILES} VERBATIM) add_library(${ARG_TARGET} ${GEN_SOURCES}) 中注释掉 if,我得到

foreach

在构建时。有了这个 ninja: error: '../verification/debug/config/Copyright.frame', needed by 'verification/gen/config/Parser.h', missing and no known rule to make it ,文件就生成了,但如果我稍后创建 if,构建程序当然不会知道它。

1 个答案:

答案 0 :(得分:2)

对于常见的生成器,您可以使用 DEPFILE 生成 Make-ish 依赖项,就像 gcc -MT -MD 对包含文件所做的一样。您必须编写一个包装器来检测文件是否存在,然后生成文件。

depfile 大致如下所示:

output: source1 source2

而且 output 必须与 CMAKE_BINARY_DIR 相关(我花了很长时间才发现),source1 是绝对的。而且您还必须转义名称中的空格、制表符、换行符和 # (!)。一般:

# CMakeLists.txt
# Example for one file
set(output ${ARG_OUTPUT_DIR}/Scanner.h)
# Relative to BINARY_DIR!
file(RELATIVE_PATH outputbinrela "${CMAKE_BINARY_DIR}" "${output}")
set(depfile "${CMAKE_CURRENT_BINARY_DIR}/Scanner.h.d" ABSOLUTE)
add_custom_command(
   OUTPUT ${someoutput}
   DEPFILE ${depfile}
   # TODO: write it in cmake, to be portable
   COMMAND sh ./thescript.sh
        "${outputbinrela}" "${depfile}" "${input}"
        ${FRAME_DIR}/Copyright.frame
   DEPENDS ${input}
   WORKING_DIRECTORY "${CMAKE_CURRENT_SOURCE_DIR}"
   VERBATIM
)

# ./thescript.sh
outputbinrela="$1"
depfile="$2"
input="$3"
copyright="$4"

# Check if the file exists - if it does, add it to depfile
if [[ -e "$copyright" ]]; then
    copyright=$(readlink -f "$copyright") # absolute!
    echo "$outputbinrela: $copyright" > "$depfile"
else
    echo "$outputbinrela:" > "$depfile"
fi

# Run the command to generate output.
yourcommand "$copyright" > "$input"

我曾经为m4预处理器编写过这样的系统,您可以查看运行m4预处理器并抓取m4包含文件并生成depfile的this script,以及调用{{1}的this script }.

不过,如果你添加 add_custom_command( ${CMAKE_COMMAND} -Pthe_other_script.cmake DEPFILE ...) 文件,cmake 不会在第一次运行时把它捡起来,但是如果你修改了别的东西,那么 depfile 将重新生成并且 depdency 将是采摘。或者,您可以例如在每次构建时重新生成 depfile。

相关问题