LINUX.ORG.RU

Как заставить cmake не удалять файлы при make clean?

 ,


0

1

Делаю русификацию. Вот начало:

set(PO_FILE ${LCPATH}/messages.po)
set(MO_FILE ${LCPATH}/LC_MESSAGES/${PROJ}.mo)
set(RU_FILE ${LCPATH}/ru.po)

find_package(Gettext REQUIRED)
find_program(GETTEXT_XGETTEXT_EXECUTABLE xgettext)
if(NOT GETTEXT_XGETTEXT_EXECUTABLE OR NOT GETTEXT_MSGFMT_EXECUTABLE)
	message(FATAL_ERROR "xgettext not found")
endif()
file(MAKE_DIRECTORY ${LCPATH})
file(MAKE_DIRECTORY ${LCPATH}/LC_MESSAGES)

add_custom_command(
	OUTPUT ${PO_FILE}
	WORKING_DIRECTORY ${CMAKE_CURRENT_SOURCE_DIR}
	COMMAND ${GETTEXT_XGETTEXT_EXECUTABLE} --from-code=utf-8 ${SOURCES} -c -k_ -kN_ -o ${PO_FILE}
	COMMAND sed -i 's/charset=.*\\\\n/charset=koi8-r\\\\n/' ${PO_FILE}
	COMMAND enconv ${PO_FILE}
	DEPENDS ${SOURCES}
)
А вот дальше — полная фигня!

Я хочу, чтобы можно было сказав make MO_FILE перекомпилировать его после изменения файла русификации. Пишу:

add_custom_target(
	MO_FILE
	COMMAND ${GETTEXT_MSGFMT_EXECUTABLE} ${RU_FILE} -o ${MO_FILE}
	DEPENDS ${RU_FILE}
)
Однако, чтобы этот файл сгенерировался при простом make, мне нужно писать
add_custom_command(
	OUTPUT ${MO_FILE}
	COMMAND ${GETTEXT_MSGFMT_EXECUTABLE} ${RU_FILE} -o ${MO_FILE}
	DEPENDS ${RU_FILE}
)
Что приводит к удалению сего файла при make clean.

Аналогичная ситуация с RU_FILE: если я пишу

add_custom_command(
	OUTPUT ${RU_FILE}
	COMMAND [ -f ${RU_FILE} ] && ${GETTEXT_MSGMERGE_EXECUTABLE} -Uis ${RU_FILE} ${PO_FILE} || cp ${PO_FILE} ${RU_FILE}
	DEPENDS ${PO_FILE}
)
, то весь перевод нафиг удаляется после make clean, т.е. нужно писать
add_custom_target(
	RU_FILE
	COMMAND [ -f ${RU_FILE} ] && ${GETTEXT_MSGMERGE_EXECUTABLE} -Uis ${RU_FILE} ${PO_FILE} || cp ${PO_FILE} ${RU_FILE}
	DEPENDS ${PO_FILE}
)
add_custom_command(
	OUTPUT ${MO_FILE}
	COMMAND make RU_FILE && ${GETTEXT_MSGFMT_EXECUTABLE} ${RU_FILE} -o ${MO_FILE}
	DEPENDS ${RU_FILE}
)
Однако, в этом случае (если опустить custom_command для RU_FILE) получаю (как и следовало ожидать) ошибку:
make
make[2]: *** Нет правила для сборки цели `../locale/ru/ru.po', требуемой для `../locale/ru/LC_MESSAGES/fitsread.mo'.  Останов.
make[1]: *** [CMakeFiles/fitsread.dir/all] Ошибка 2
make: *** [all] Ошибка 2
, а еще файл ru.po перестает обновляться при изменении PO_FILE.

Как из этого замкнутого круга выйти и сделать так, чтобы файл русификации (ru.po) не удалялся при make clean, но в то же время при изменении ru.po файл .mo автоматом генерировался бы через make, а при изменении PO_FILE ru.po автоматом обновлялся?

☆☆☆☆☆

Однако, чтобы этот файл сгенерировался при простом make

- add_custom_target(MO_FILE ...)
+ add_custom_target(MO_FILE ALL ...)

Я так понимаю PO это исходный, а MO скомпиленый? То-есть МО есть продукт компиляции и нужен только для рантайма? Значит удаление его при очистке это правильное поведение.

ru.po должен обновляться из messages.po, верно? То-есть ru.po это исходник, и CMake здесь должен просто помочь его правильно обновить при сборке (чтобы не делать это руками) и дальше сей файл должен редактироваться человеком и уйти в систему контроля версий?

В этом случае ru.po НЕ должен фигурировать в OUTPUT. Для это есть BYPRODUCTS.

- add_custom_target(RU_FILE ...)
+ add_custom_target(RU_FILE ... BYPRODUCTS ${RU_FILE})

Но вообще обновлять исходные файлы при сборке скорее моветон. Есть смысл разорвать жёсткую зависимость между MO и RU и заставить человека явно вызывать make RU_FILE перед тем как ему захочется полокализировать.

Dendy ★★★★★
()
Ответ на: комментарий от Dendy

То-есть МО есть продукт компиляции и нужен только для рантайма? Значит удаление его при очистке это правильное поведение.

В принципе, да.

То-есть ru.po это исходник, и CMake здесь должен просто помочь его правильно обновить при сборке (чтобы не делать это руками) и дальше сей файл должен редактироваться человеком и уйти в систему контроля версий?

Да.

В этом случае ru.po НЕ должен фигурировать в OUTPUT. Для это есть BYPRODUCTS.

Не работает. Делаю так:

add_executable(${PROJ} ${SOURCES})
add_custom_command(
	OUTPUT ${MO_FILE}
	COMMAND make MO_FILE
	DEPENDS ${RU_FILE}
)
add_custom_target(
	RU_FILE BYPRODUCTS ${RU_FILE}
	COMMAND [ -f ${RU_FILE} ] && ${GETTEXT_MSGMERGE_EXECUTABLE} -Uis ${RU_FILE} ${PO_FILE} || cp ${PO_FILE} ${RU_FILE}
	DEPENDS ${PO_FILE}
)
add_custom_target(
	MO_FILE ALL
	COMMAND ${GETTEXT_MSGFMT_EXECUTABLE} ${RU_FILE} -o ${MO_FILE}
	DEPENDS ${RU_FILE}
)

Все равно ошибка:
make
Scanning dependencies of target MO_FILE
make[2]: *** Нет правила для сборки цели `../locale/ru/ru.po', требуемой для `CMakeFiles/MO_FILE'.  Останов.
make[1]: *** [CMakeFiles/MO_FILE.dir/all] Ошибка 2
make: *** [all] Ошибка 2

Есть смысл разорвать жёсткую зависимость между MO и RU и заставить человека явно вызывать make RU_FILE перед тем как ему захочется полокализировать

Я точно буду забывать это делать.

Eddy_Em ☆☆☆☆☆
() автор топика
Ответ на: комментарий от Dendy

Ну и чтобы два раза не переспрашивать вот финальный код:

add_custom_command(
  OUTPUT ${MO_FILE}
  COMMAND ${GETTEXT_MSGFMT_EXECUTABLE} ${RU_FILE} -o ${MO_FILE}
  DEPENDS ${RU_FILE}
)

add_custom_target(
  RU_FILE
  COMMAND [ -f ${RU_FILE} ] && ${GETTEXT_MSGMERGE_EXECUTABLE} -Uis ${RU_FILE} ${PO_FILE} || cp ${PO_FILE} ${RU_FILE}
  BYPRODUCTS ${RU_FILE}
  DEPENDS ${PO_FILE}
)

add_custom_target(MO_FILE ALL DEPENDS RU_FILE ${MO_FILE})
Dendy ★★★★★
()
Ответ на: комментарий от Eddy_Em

Эх, не успел, так и знал, что сейчас эта ошибка вылезет (-: Смотри последний код. А именно:

add_custom_target(MO_FILE ALL DEPENDS RU_FILE ${MO_FILE})

Dendy ★★★★★
()
Ответ на: комментарий от Eddy_Em
add_custom_command(
	OUTPUT ${MO_FILE}
	COMMAND make MO_FILE
	DEPENDS ${RU_FILE}
)

Никогда так не делай.

Dendy ★★★★★
()
Ответ на: комментарий от Dendy

Спасибо, работает!

А то я уже другой велосипед набросал (тоже работающий):

add_custom_command(
	OUTPUT ${MO_FILE}
	COMMAND make MO_FILE
	DEPENDS RU_FILE
)
add_custom_command(
	OUTPUT RU_FILE
	COMMAND make RU_FILE && touch RU_FILE
	DEPENDS ${PO_FILE}
)
add_custom_target(
	RU_FILE
	COMMAND [ -f ${RU_FILE} ] && ${GETTEXT_MSGMERGE_EXECUTABLE} -Uis ${RU_FILE} ${PO_FILE} || cp ${PO_FILE} ${RU_FILE}
	DEPENDS ${PO_FILE}
)
add_custom_target(
	MO_FILE ALL
	COMMAND ${GETTEXT_MSGFMT_EXECUTABLE} ${RU_FILE} -o ${MO_FILE}
	DEPENDS RU_FILE
)

Eddy_Em ☆☆☆☆☆
() автор топика
Ответ на: комментарий от Eddy_Em

COMMAND make

Вот чтобы таким не заниматься (вызывать из системы сборки саму себя) и придумали двупроходную сборку на основе BYPRODUCTS, советую почитать подробней в документации.

Dendy ★★★★★
()
Ответ на: комментарий от Eddy_Em

И в итоге у меня родился такой шаблон для автогенерации CMakeLists.txt:

cmake_minimum_required(VERSION 2.8)
set(PROJ <project name>)
set(MINOR_VERSION "1")
set(MID_VERSION "0")
set(MAJOR_VERSION "0")
set(VERSION "${MAJOR_VERSION}.${MID_VERSION}.${MINOR_VERSION}")

enable_language(C)

message("VER: ${VERSION}")

# default flags
set(CFLAGS -O2 -Wextra -Wall -Werror -W -std=gnu99)

set(CMAKE_COLOR_MAKEFILE ON)

# here is one of two variants: all .c in directory or .c files in list
aux_source_directory(${CMAKE_CURRENT_SOURCE_DIR} SOURCES)
#list(REMOVE_ITEM SOURCES ${CMAKE_CURRENT_SOURCE_DIR}/<file to remove>)
#set(SOURCES list_of_c_files)

# we can change file list
#if(NOT DEFINED something)
#	set(SOURCES ${SOURCES} one_more_list)
#	add_definitions(-DSOME_DEFS)
#endif()

# cmake -DDEBUG=1 -> debugging
if(DEFINED DEBUG)
	add_definitions(-DEBUG)
endif()

# directory should contain dir locale/ru for gettext translations
set(LCPATH ${CMAKE_SOURCE_DIR}/locale/ru)

if(NOT DEFINED LOCALEDIR)
	if(DEFINED DEBUG)
		set(LOCALEDIR ${CMAKE_CURRENT_SOURCE_DIR}/locale)
	else()
		set(LOCALEDIR ${CMAKE_INSTALL_PREFIX}/share/locale)
	endif()
endif()

###### pkgconfig ######
# pkg-config modules (for pkg-check-modules)
set(MODULES <modules>)
# additional modules on condition
#if(DEFINED SOMETHING)
#	set(MODULES ${MODULES} more_modules>=version)
#	add_definitions(-DSOMEDEFS)
#endif()
# find packages:
find_package(PkgConfig REQUIRED)
# non-required
#find_package(<pkg1>)
pkg_check_modules(${PROJ} REQUIRED ${MODULES})

# external modules like OpenMP:
#include(FindOpenMP)
#if(OPENMP_FOUND)
#	set(CMAKE_C_FLAGS "${CMAKE_C_FLAGS} ${OpenMP_C_FLAGS}")
#	set(CMAKE_CXX_FLAGS "${CMAKE_CXX_FLAGS} ${OpenMP_CXX_FLAGS}")
#	set(CMAKE_EXE_LINKER_FLAGS "${CMAKE_EXE_LINKER_FLAGS} ${OpenMP_EXE_LINKER_FLAGS}")
#endif()

# append data from find_package:
#list(APPEND ${PROJ}_INCLUDE_DIRS ${<pkg1>_INCLUDE_DIR} ${<pkg2>_INCLUDE_DIR})
#list(APPEND ${PROJ}_LIBRARIES ${<pkg1>_LIBRARY} ${<pkg2>_LIBRARY})
#list(APPEND ${${PROJ}_LIBRARY_DIRS}  ${<pkg1>_LIBRARY_DIRS})

###### additional flags ######
#set(CMAKE_C_FLAGS "${CMAKE_C_FLAGS} -lfftw3_threads")
# conditional check:
#if(NOT DEFINED NO_GSL)
#	pkg_check_modules(GSL gsl)
#endif()
#if(NOT DEFINED GSL_VERSION)
#	message("GSL not found, some mathematics functions wouldn't be avialable")
#else()
#	add_definitions(-DGSL_FOUND)
#endif()

project(${PROJ})
# change wrong behaviour with install prefix
if(CMAKE_INSTALL_PREFIX_INITIALIZED_TO_DEFAULT AND CMAKE_INSTALL_PREFIX MATCHES "/usr/local")
	message("Change default install path to /usr")
	set(CMAKE_INSTALL_PREFIX "/usr")
endif()
message("Install dir prefix: ${CMAKE_INSTALL_PREFIX}")

# gettext files
set(PO_FILE ${LCPATH}/messages.po)
set(MO_FILE ${LCPATH}/LC_MESSAGES/${PROJ}.mo)
set(RU_FILE ${LCPATH}/ru.po)

# exe file
add_executable(${PROJ} ${SOURCES})
# another exe, depending on some other files
#add_executable(test_client client.c usefull_macros.c parceargs.c)
# -I
include_directories(${${PROJ}_INCLUDE_DIRS})
# -L
link_directories(${${PROJ}_LIBRARY_DIRS})
# -D
add_definitions(${CFLAGS} -DLOCALEDIR=\"${LOCALEDIR}\"
		-DPACKAGE_VERSION=\"${VERSION}\" -DGETTEXT_PACKAGE=\"${PROJ}\"
		-DMINOR_VERSION=\"${MINOR_VERSION}\" -DMID_VERSION=\"${MID_VERSION}\"
		-DMAJOR_VERSION=\"${MAJOR_VESION}\")

###### pthreads ######
find_package(Threads REQUIRED)
if(THREADS_HAVE_PTHREAD_ARG)
  set_property(TARGET ${PROJ} PROPERTY COMPILE_OPTIONS "-pthread")
  set_property(TARGET ${PROJ} PROPERTY INTERFACE_COMPILE_OPTIONS "-pthread")
endif()
if(CMAKE_THREAD_LIBS_INIT)
  list(APPEND ${PROJ}_LIBRARIES "${CMAKE_THREAD_LIBS_INIT}")
endif()

# target libraries
target_link_libraries(${PROJ} ${${PROJ}_LIBRARIES})

# Installation of the program
INSTALL(FILES ${MO_FILE} DESTINATION "share/locale/ru/LC_MESSAGES")
        #PERMISSIONS OWNER_WRITE OWNER_READ GROUP_READ WORLD_READ)
INSTALL(TARGETS ${PROJ} DESTINATION "bin")
        #PERMISSIONS OWNER_WRITE OWNER_READ OWNER_EXECUTE GROUP_READ GROUP_EXECUTE WORLD_READ WORLD_EXECUTE)
# Script to be executed at installation time (kind of post-intallation script) to
# change the right accesses on the installed files
#INSTALL(SCRIPT inst.cmake)

###### gettext ######
find_package(Gettext REQUIRED)
find_program(GETTEXT_XGETTEXT_EXECUTABLE xgettext)
if(NOT GETTEXT_XGETTEXT_EXECUTABLE OR NOT GETTEXT_MSGFMT_EXECUTABLE)
	message(FATAL_ERROR "xgettext not found")
endif()
file(MAKE_DIRECTORY ${LCPATH})
file(MAKE_DIRECTORY ${LCPATH}/LC_MESSAGES)

add_custom_command(
	OUTPUT ${PO_FILE}
	WORKING_DIRECTORY ${CMAKE_CURRENT_SOURCE_DIR}
	COMMAND ${GETTEXT_XGETTEXT_EXECUTABLE} --from-code=utf-8 ${SOURCES} -c -k_ -kN_ -o ${PO_FILE}
	COMMAND sed -i 's/charset=.*\\\\n/charset=koi8-r\\\\n/' ${PO_FILE}
	COMMAND enconv ${PO_FILE}
	DEPENDS ${SOURCES}
)

# we need this to prewent ru.po from deleting by make clean
add_custom_command(
	OUTPUT ${MO_FILE}
	COMMAND ${GETTEXT_MSGFMT_EXECUTABLE} ${RU_FILE} -o ${MO_FILE}
	DEPENDS ${RU_FILE}
)

add_custom_target(
	RU_FILE
	COMMAND [ -f ${RU_FILE} ] && ${GETTEXT_MSGMERGE_EXECUTABLE} -Uis ${RU_FILE} ${PO_FILE} || cp ${PO_FILE} ${RU_FILE}
	BYPRODUCTS ${RU_FILE}
	DEPENDS ${PO_FILE}
)

add_custom_target(MO_FILE ALL DEPENDS RU_FILE ${MO_FILE})

Eddy_Em ☆☆☆☆☆
() автор топика
Ответ на: комментарий от Dendy

советую почитать подробней в документации

Дык, эту долбаную документацию пока прочтешь до конца, начало уже забудешь ☺

Eddy_Em ☆☆☆☆☆
() автор топика
Ответ на: комментарий от Eddy_Em

Ну и да, как можно наверное заметить merge будет вызываться каждый раз при вызове make, потому что у цели RU_FILE нет выходного файла. Чтобы вызывать merge только когда обновился PO_FILE нужно сделать ему выходной файл.

add_custom_command(
  OUTPUT ${MO_FILE}
  COMMAND ${GETTEXT_MSGFMT_EXECUTABLE} ${RU_FILE} -o ${MO_FILE}
  DEPENDS ${RU_FILE} ru_file_updated
)

add_custom_command(
  OUTPUT ru_file_updated
  COMMAND [ -f ${RU_FILE} ] && ${GETTEXT_MSGMERGE_EXECUTABLE} -Uis ${RU_FILE} ${PO_FILE} || cp ${PO_FILE} ${RU_FILE}
  COMMAND ${CMAKE_COMMAND} -E touch ru_file_updated
  BYPRODUCTS ${RU_FILE}
  DEPENDS ${PO_FILE}
)

add_custom_target(MO_FILE ALL DEPENDS ${MO_FILE})
Dendy ★★★★★
()
Ответ на: комментарий от Dendy

Естественно. У меня везде только КОИ8-Р.

Eddy_Em ☆☆☆☆☆
() автор топика
Вы не можете добавлять комментарии в эту тему. Тема перемещена в архив.