function(qtprotobuf_link_archive TARGET GENERATED_TARGET) set_target_properties(${qtprotobuf_generate_TARGET} PROPERTIES LINKER_LANGUAGE CXX) set_target_properties(${TARGET} PROPERTIES LINK_DIRECTORIES "$") get_target_property(EXISTING_FLAGS ${TARGET} LINK_FLAGS) if(WIN32) if("${EXISTING_FLAGS}" STREQUAL "EXISTING_FLAGS-NOTFOUND") set_target_properties(${TARGET} PROPERTIES LINK_FLAGS "/WHOLEARCHIVE:${GENERATED_TARGET}") else() set_target_properties(${TARGET} PROPERTIES LINK_FLAGS "${EXISTING_FLAGS} /WHOLEARCHIVE:${GENERATED_TARGET}") endif() else() target_link_libraries(${TARGET} PRIVATE -Wl,--whole-archive $ -Wl,--no-whole-archive) endif() target_include_directories(${TARGET} PRIVATE $) add_dependencies(${TARGET} ${GENERATED_TARGET}) endfunction() function(qtprotobuf_generate) set(options) set(oneValueArgs OUT_DIR TARGET GENERATED_TARGET MULTI QML GENERATED_HEADERS_VAR) set(multiValueArgs GENERATED_HEADERS EXCLUDE_HEADERS PROTO_FILES PROTO_INCLUDES) cmake_parse_arguments(qtprotobuf_generate "${options}" "${oneValueArgs}" "${multiValueArgs}" ${ARGN}) if(DEFINED qtprotobuf_generate_GENERATED_TARGET) set(GENERATED_TARGET_NAME ${qtprotobuf_generate_GENERATED_TARGET}) elseif(DEFINED qtprotobuf_generate_TARGET) set(GENERATED_TARGET_NAME ${qtprotobuf_generate_TARGET}_qtprotobuf_gen) else() message(FATAL_ERROR "Either TARGET or GENERATED_TARGET must be specified") endif() set(GEN_TARGET ${GENERATED_TARGET_NAME}_generate) set(GENERATED_HEADERS ${qtprotobuf_generate_GENERATED_HEADERS}) if(NOT DEFINED QTPROTOBUF_EXECUTABLE) set(QTPROTOBUF_EXECUTABLE "${QTPROTOBUF_EXECUTABLE_INSTALL}") endif() set(GENERATION_TYPE "SINGLE") if("${qtprotobuf_generate_MULTI}" STREQUAL "TRUE") set(GENERATION_TYPE "MULTI") #TODO: add globalenums by default. But it's better to verify if proto file contains any global enum set(GENERATED_HEADERS ${GENERATED_HEADERS} globalenums.h) endif() set(GENERATION_OPTIONS ${GENERATION_TYPE}) if("${qtprotobuf_generate_QML}" STREQUAL "TRUE") message(STATUS "Enabled QML generation for ${GENERATED_TARGET_NAME}") set(GENERATION_OPTIONS "${GENERATION_OPTIONS}:QML") endif() find_program(GO_EXECUTABLE "go") if (GO_EXECUTABLE STREQUAL GO_EXECUTABLE-NOTFOUND) message(FATAL_ERROR "Golang is mandatory dependency for QtProtobuf. Please install it and ensure that it's accessible by PATH environment variable") endif() set(PROTO_INCLUDES ${qtprotobuf_generate_PROTO_INCLUDES}) foreach(PROTO_FILE IN LISTS qtprotobuf_generate_PROTO_FILES) get_filename_component(BASE_DIR ${PROTO_FILE} DIRECTORY) set(PROTO_INCLUDES -I"${BASE_DIR}" ${PROTO_INCLUDES}) execute_process(COMMAND ${GO_EXECUTABLE} run ${PROTO_PARSER} ${PROTO_FILE} ${GENERATION_TYPE} OUTPUT_VARIABLE GENERATED_HEADERS_PART ERROR_VARIABLE PARSER_ERROR) set(GENERATED_HEADERS ${GENERATED_HEADERS} ${GENERATED_HEADERS_PART}) endforeach() if(NOT "${PARSER_ERROR}" STREQUAL "") message(FATAL_ERROR "Go parser error ${PARSER_ERROR}") endif() if(DEFINED qtprotobuf_generate_GENERATED_HEADERS) set(GENERATED_HEADERS ${qtprotobuf_generate_GENERATED_HEADERS}) endif() foreach(EXCLUDE_HEADER IN LISTS qtprotobuf_generate_EXCLUDE_HEADERS) list(REMOVE_ITEM GENERATED_HEADERS ${EXCLUDE_HEADER}) endforeach() list(REMOVE_DUPLICATES GENERATED_HEADERS) if(NOT DEFINED qtprotobuf_generate_OUT_DIR) set(OUT_DIR ${CMAKE_CURRENT_BINARY_DIR}) else() set(OUT_DIR ${qtprotobuf_generate_OUT_DIR}) endif() file(MAKE_DIRECTORY ${OUT_DIR}) unset(QTPROTOBUF_GENERATED_SOURCES) unset(QTPROTOBUF_GENERATED_HEADERS) foreach(GENERATED_HEADER IN LISTS GENERATED_HEADERS) get_filename_component(GENERATED_BASENAME ${GENERATED_HEADER} NAME) string(REGEX REPLACE "\\.[^.]*$" "" GENERATED_BASENAME ${GENERATED_BASENAME}) list(APPEND QTPROTOBUF_GENERATED_SOURCES ${OUT_DIR}/${GENERATED_BASENAME}.cpp) list(APPEND QTPROTOBUF_GENERATED_HEADERS ${OUT_DIR}/${GENERATED_BASENAME}.h) set_property(SOURCE ${OUT_DIR}/${GENERATED_BASENAME}.cpp PROPERTY SKIP_AUTOMOC ON) endforeach() if(WIN32) set(PROTOC_COMMAND set QT_PROTOBUF_OPTIONS=${GENERATION_OPTIONS}&& $) else() set(PROTOC_COMMAND "QT_PROTOBUF_OPTIONS=${GENERATION_OPTIONS}" $) endif() add_custom_command( OUTPUT ${QTPROTOBUF_GENERATED_SOURCES} ${QTPROTOBUF_GENERATED_HEADERS} COMMAND ${PROTOC_COMMAND} --plugin=protoc-gen-${GENERATOR_TARGET}=${QTPROTOBUF_EXECUTABLE} --${GENERATOR_TARGET}_out=${OUT_DIR} ${PROTO_INCLUDES} ${qtprotobuf_generate_PROTO_FILES} WORKING_DIRECTORY ${OUT_DIR} DEPENDS ${qtprotobuf_generate_PROTO_FILES} ${QTPROTOBUF_EXECUTABLE} COMMENT "Generating QtProtobuf ${GENERATED_TARGET_NAME} sources..." ) add_custom_target(${GEN_TARGET} DEPENDS ${QTPROTOBUF_GENERATED_SOURCES} ${QTPROTOBUF_GENERATED_HEADERS} ${qtprotobuf_generate_PROTO_FILES}) if(NOT "${qtprotobuf_generate_GENERATED_HEADERS_VAR}" STREQUAL "") set(${qtprotobuf_generate_GENERATED_HEADERS_VAR} ${QTPROTOBUF_GENERATED_HEADERS} PARENT_SCOPE) endif() qt5_wrap_cpp(MOC_SOURCES ${QTPROTOBUF_GENERATED_HEADERS}) list(APPEND GENERATED_SOURCES ${MOC_SOURCES}) set_source_files_properties(${QTPROTOBUF_GENERATED_SOURCES} PROPERTIES GENERATED TRUE) add_library(${GENERATED_TARGET_NAME} ${QTPROTOBUF_GENERATED_SOURCES} ${MOC_SOURCES}) add_dependencies(${GENERATED_TARGET_NAME} ${GEN_TARGET}) target_include_directories(${GENERATED_TARGET_NAME} PUBLIC ${OUT_DIR} PRIVATE ${Qt5Core_INCLUDE_DIRS} $ $ ${OUT_DIR}) if(NOT WIN32) target_include_directories(${GENERATED_TARGET_NAME} PRIVATE $) endif() if(DEFINED qtprotobuf_generate_TARGET) qtprotobuf_link_archive(${qtprotobuf_generate_TARGET} ${GENERATED_TARGET_NAME}) endif() endfunction()