1

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

具体来说,我使用的是 Coco/R,它要求Parser.frameScanner.frame文件存在于与语法文件相同的目录中或由命令行参数指定的目录中,并且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,文件就生成了,但是如果我Copyright.frame稍后创建,当然构建程序不会知道它。

4

1 回答 1

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 的脚本,以及这个调用add_custom_command( ${CMAKE_COMMAND} -Pthe_other_script.cmake DEPFILE ...).

尽管如此,如果您添加Copyright.frame文件,cmake不会在第一次运行时选择它,但如果您修改其他内容,则会重新生成 depfile 并选择依赖关系。或者您可以在每次构建时重新生成 depfile。

于 2021-07-06T19:03:56.443 回答