Y часть:
У меня есть макрос CMake, который запускает внешнюю программу и создает переменные, специфичные для проекта. Что я хотел бы сделать, чтобы CMake делал, так это запускать этот скрипт при каждой сборке, проверять, не изменились ли переменные, и если они есть (для предотвращения полной перестройки проекта, когда этого не произошло), заново генерировать заголовок CONFIGURE_FILE … I ‘ Я не совсем уверен, как это сделать, хотя. (При добавлении пользовательской цели каждый раз перестраивается файл заголовка, и он не может вызывать макросы, как и в случае добавления пользовательской команды).
X часть:
Итак, я написал следующий скрипт для извлечения информации о версии hg для использования в проекте c ++:
macro (ReadProjectRevisionStatus)
exec_program(hg ${PROJECT_SOURCE_DIR} ARGS paths OUTPUT_VARIABLE ${PROJECT_NAME}_HGPATHS)
message(STATUS "${PROJECT_NAME}_HGPATHS=${${PROJECT_NAME}_HGPATHS}}")
if (NOT(${PROJECT_NAME}_HGPATHS STREQUAL ""))
string(REPLACE "\n" ";" ${PROJECT_NAME}_HGPATHS ${${PROJECT_NAME}_HGPATHS})
foreach(HGPATH ${${PROJECT_NAME}_HGPATHS})
string(SUBSTRING "${HGPATH}" 0 10 HGPATHSTART)
if (HGPATHSTART MATCHES "default = ")
string(LENGTH "${HGPATH}" HGPATHLENGTH)
math(EXPR HGSUBLEN "${HGPATHLENGTH}-10")
string(SUBSTRING "${HGPATH}" 10 ${HGSUBLEN} ${PROJECT_NAME}_HGREMOTEDIR)
endif()
endforeach()
endif()
if (NOT ${PROJECT_NAME}_HGREMOTEDIR)
message(WARNING "No remote repository set. Will use current direcoty for build number, but this value may be inaccurate.")
set(${PROJECT_NAME}_HGREMOTEDIR ${PROJECT_SOURCE_DIR})
else()
exec_program(hg ${${PROJECT_NAME}_HGREMOTEDIR} ARGS status RETURN_VALUE HGREMOTESTATUSVALUE OUTPUT_VARIABLE NUL)
if (NOT HGREMOTESTATUSVALUE EQUAL 0)
message(WARNING "Cannot connect to remote repository at ${${PROJECT_NAME}_HGREMOTEDIR}. Will use current direcoty for build number, but this value may be inaccurate.")
set(${PROJECT_NAME}_HGREMOTEDIR ${PROJECT_SOURCE_DIR})
endif()
endif()
#Identify changeset
exec_program(hg ${PROJECT_SOURCE_DIR} ARGS "id" "-i" OUTPUT_VARIABLE OUTPUT_VARIABLE ${PROJECT_NAME}_HGHASHCODE)
if (${PROJECT_NAME}_HGHASHCODE MATCHES ".*\\+")
MESSAGE(STATUS "Node is dirty. Will generate temporary version number...")
set (${PROJECT_NAME}_HGDIRTY 1)
string(LENGTH ${${PROJECT_NAME}_HGHASHCODE} HGHASHLEN)
MATH(EXPR HGHASHLEN "${HGHASHLEN}-1")
string(SUBSTRING ${${PROJECT_NAME}_HGHASHCODE} 0 ${HGHASHLEN} ${PROJECT_NAME}_HGHASHCODE)
endif()
#check if remote repository contains changeset
exec_program(hg ${${PROJECT_NAME}_HGREMOTEDIR}
ARGS "log" "-r" "${${PROJECT_NAME}_HGHASHCODE}"RETURN_VALUE HGREMOTEHASCHANGESET
OUTPUT_VARIABLE NUL)
if (NOT HGREMOTEHASCHANGESET EQUAL 0)
message(WARNING "Remote repository ${${PROJECT_NAME}_HGREMOTEDIR} does not have changeset ${${PROJECT_NAME}_HGHASHCODE}. Will use current direcoty for build number, but this value may be inaccurate.")
set(${PROJECT_NAME}_HGREMOTEDIR ${PROJECT_SOURCE_DIR})
endif()
exec_program(hg ${${PROJECT_NAME}_HGREMOTEDIR}
ARGS "log" "-r" ${${PROJECT_NAME}_HGHASHCODE} "--template" "{latesttag}"OUTPUT_VARIABLE ${PROJECT_NAME}_HGMAJORMINORVERSION)
MESSAGE(STATUS "${PROJECT_NAME}_HGMAJORMINORVERSION=${${PROJECT_NAME}_HGMAJORMINORVERSION}")
exec_program(hg ${${PROJECT_NAME}_HGREMOTEDIR}
ARGS "log" "-r" ${${PROJECT_NAME}_HGHASHCODE} "--template" "{latesttagdistance}"OUTPUT_VARIABLE ${PROJECT_NAME}_HGBUILDNUMBER)
if (${PROJECT_NAME}_HGDIRTY)
MATH(EXPR ${PROJECT_NAME}_HGBUILDNUMBER "${${PROJECT_NAME}_HGBUILDNUMBER}+1")
set(${PROJECT_NAME}_HGHASHCODE "${${PROJECT_NAME}_HGHASHCODE}+")
endif()
MESSAGE(STATUS "Version=${${PROJECT_NAME}_HGMAJORMINORVERSION}.${${PROJECT_NAME}_HGBUILDNUMBER}.${${PROJECT_NAME}_HGHASHCODE}")
endmacro()
И я в настоящее время называю это так.
ReadProjectRevisionStatus()
CONFIGURE_FILE(${CMAKE_CURRENT_SOURCE_DIR}/main.h.in ${CMAKE_CURRENT_BINARY_DIR}/main.h)
Где main.h.in это:
#ifndef MAIN_H
#define MAIN_H
#include <string>
const std::string ${PROJECT_NAME}_HGMAJORMINORVERSION = "${${PROJECT_NAME}_HGMAJORMINORVERSION}";
const std::string ${PROJECT_NAME}_HGBUILDNUMBER = "${${PROJECT_NAME}_HGBUILDNUMBER}";
const std::string ${PROJECT_NAME}_HGHASHCODE = "${${PROJECT_NAME}_HGHASHCODE}";
const std::string ${PROJECT_NAME}_HG_SHORT_VERSION =
${PROJECT_NAME}_HGMAJORMINORVERSION+"."+
${PROJECT_NAME}_HGBUILDNUMBER;
const std::string ${PROJECT_NAME}_HG_VERSION =
${PROJECT_NAME}_HG_SHORT_VERSION + "." +
${PROJECT_NAME}_HGHASHCODE;
#endif
То, что я хотел бы сделать, это запустить этот макрос для каждой команды сборки (я могу использовать ADD_CUSTOM_TARGET, а затем установить переменные вместо чтения их из проекта), но я хочу перегенерировать main.h только если переменные изменились (на предотвратить ненужную перекомпиляцию).
РЕДАКТИРОВАТЬ: рабочий раствор
Сделал HGVersion.CMake
cmake_minimum_required (VERSION 2.8)
macro (ReadProjectRevisionStatus)
message(STATUS PROJECT_SOURCE_DIR=${PROJECT_SOURCE_DIR})
exec_program(hg ${PROJECT_SOURCE_DIR} ARGS paths OUTPUT_VARIABLE ${PROJECT_NAME}_HGPATHS)
message(STATUS "${PROJECT_NAME}_HGPATHS=${${PROJECT_NAME}_HGPATHS}}")
if (NOT(${PROJECT_NAME}_HGPATHS STREQUAL ""))
string(REPLACE "\n" ";" ${PROJECT_NAME}_HGPATHS ${${PROJECT_NAME}_HGPATHS})
foreach(HGPATH ${${PROJECT_NAME}_HGPATHS})
string(SUBSTRING "${HGPATH}" 0 10 HGPATHSTART)
if (HGPATHSTART MATCHES "default = ")
string(LENGTH "${HGPATH}" HGPATHLENGTH)
math(EXPR HGSUBLEN "${HGPATHLENGTH}-10")
string(SUBSTRING "${HGPATH}" 10 ${HGSUBLEN} ${PROJECT_NAME}_HGREMOTEDIR)
endif()
endforeach()
endif()
if (NOT ${PROJECT_NAME}_HGREMOTEDIR)
message(WARNING "No remote repository set. Will use current direcoty for build number, but this value may be inaccurate.")
set(${PROJECT_NAME}_HGREMOTEDIR ${PROJECT_SOURCE_DIR})
else()
exec_program(hg ${${PROJECT_NAME}_HGREMOTEDIR} ARGS status RETURN_VALUE HGREMOTESTATUSVALUE OUTPUT_VARIABLE NUL)
if (NOT HGREMOTESTATUSVALUE EQUAL 0)
message(WARNING "Cannot connect to remote repository at ${${PROJECT_NAME}_HGREMOTEDIR}. Will use current direcoty for build number, but this value may be inaccurate.")
set(${PROJECT_NAME}_HGREMOTEDIR ${PROJECT_SOURCE_DIR})
endif()
endif()
#Identify changeset
exec_program(hg ${PROJECT_SOURCE_DIR} ARGS "id" "-i" OUTPUT_VARIABLE OUTPUT_VARIABLE ${PROJECT_NAME}_HGHASHCODE)
if (${PROJECT_NAME}_HGHASHCODE MATCHES ".*\\+")
MESSAGE(STATUS "Node is dirty. Will generate temporary version number...")
set (${PROJECT_NAME}_HGDIRTY 1)
string(LENGTH ${${PROJECT_NAME}_HGHASHCODE} HGHASHLEN)
MATH(EXPR HGHASHLEN "${HGHASHLEN}-1")
string(SUBSTRING ${${PROJECT_NAME}_HGHASHCODE} 0 ${HGHASHLEN} ${PROJECT_NAME}_HGHASHCODE)
endif()
#check if remote repository contains changeset
exec_program(hg ${${PROJECT_NAME}_HGREMOTEDIR}
ARGS "log" "-r" "${${PROJECT_NAME}_HGHASHCODE}"RETURN_VALUE HGREMOTEHASCHANGESET
OUTPUT_VARIABLE NUL)
if (NOT HGREMOTEHASCHANGESET EQUAL 0)
message(WARNING "Remote repository ${${PROJECT_NAME}_HGREMOTEDIR} does not have changeset ${${PROJECT_NAME}_HGHASHCODE}. Will use current direcoty for build number, but this value may be inaccurate.")
set(${PROJECT_NAME}_HGREMOTEDIR ${PROJECT_SOURCE_DIR})
endif()
exec_program(hg ${${PROJECT_NAME}_HGREMOTEDIR}
ARGS "log" "-r" ${${PROJECT_NAME}_HGHASHCODE} "--template" "{latesttag}"OUTPUT_VARIABLE ${PROJECT_NAME}_HGMAJORMINORVERSION)
MESSAGE(STATUS "${PROJECT_NAME}_HGMAJORMINORVERSION=${${PROJECT_NAME}_HGMAJORMINORVERSION}")
exec_program(hg ${${PROJECT_NAME}_HGREMOTEDIR}
ARGS "log" "-r" ${${PROJECT_NAME}_HGHASHCODE} "--template" "{latesttagdistance}"OUTPUT_VARIABLE ${PROJECT_NAME}_HGBUILDNUMBER)
if (${PROJECT_NAME}_HGDIRTY)
MATH(EXPR ${PROJECT_NAME}_HGBUILDNUMBER "${${PROJECT_NAME}_HGBUILDNUMBER}+1")
set(${PROJECT_NAME}_HGHASHCODE "${${PROJECT_NAME}_HGHASHCODE}+")
endif()
MESSAGE(STATUS "Version=${${PROJECT_NAME}_HGMAJORMINORVERSION}.${${PROJECT_NAME}_HGBUILDNUMBER}.${${PROJECT_NAME}_HGHASHCODE}")
endmacro()
message(STATUS "GETTING HG VERSION")
ReadProjectRevisionStatus()
CONFIGURE_FILE(${PROJECT_SOURCE_DIR}/main.h.in ${PROJECT_BINARY_DIR}/main.h)
В основном файле cmake
add_custom_target(
${PROJECT_NAME}_hg_version_target
DEPENDS ${CMAKE_CURRENT_SOURCE_DIR}/main.h.in
COMMAND ${CMAKE_COMMAND}
ARGS -DPROJECT_SOURCE_DIR=${PROJECT_SOURCE_DIR}
-DPROJECT_NAME=${PROJECT_NAME}
-DPROJECT_BINARY_DIR=${PROJECT_BINARY_DIR}
-P "${CMAKE_CURRENT_SOURCE_DIR}/HGVersion.CMake")
а также
add_dependencies ($ {PROJECT_NAME} $ {PROJECT_NAME} _hg_version_target)
Работает, но кажется … очень грязно. Кто-нибудь есть что-то лучше?
Если что-то в процессе сборки определяет содержимое файла, который нужно сгенерировать, вы не можете сгенерировать этот файл с помощью файла configure_file, потому что файл configure_file работает во время фазы конфигурации, а не во время сборки. Это выбор дизайна от CMake.
В вашем случае вам действительно нужно использовать add_custom_command и add_custom_target для генерации этого файла в процессе сборки, как вы уже узнали.
Для предложенного вами решения может быть возможность немного упростить это. Когда вы генерируете заголовочный файл C / C ++, внутренний синтаксический анализатор CMake должен автоматически определять, какие из ваших обычных файлов исходного кода зависят от main.h. Поэтому должно быть достаточно предоставить пользовательскую команду с add_custom_command, которая объявляет в качестве своего OUTPUT файл main.h, который она сгенерирует. CMake должен вызывать эту команду автоматически, как только он пытается создать цель, где от нее зависит один из файлов C / C ++. Таким образом, вам, вероятно, не нужна пользовательская цель и объявленная вручную зависимость.
Других решений пока нет …