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 MULTI QML COMMENTS FOLDER) set(oneValueArgs OUT_DIR TARGET GENERATED_TARGET) 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) if(NOT DEFINED QT_PROTOBUF_EXECUTABLE) set(QT_PROTOBUF_EXECUTABLE "${QT_PROTOBUF_EXECUTABLE_INSTALL}") endif() #Options handling set(GENERATION_TYPE "SIGNLE") if(qtprotobuf_generate_MULTI) set(GENERATION_TYPE "MULTI") set(qtprotobuf_generate_FOLDER TRUE) message(STATUS "Multi-file generation is set. Folder-based generation enabled automatically for ${GENERATED_TARGET_NAME}") endif() set(GENERATION_OPTIONS ${GENERATION_TYPE}) if(qtprotobuf_generate_QML) message(STATUS "Enabled QML generation for ${GENERATED_TARGET_NAME}") set(GENERATION_OPTIONS "${GENERATION_OPTIONS}:QML") endif() if(qtprotobuf_generate_COMMENTS) message(STATUS "Enabled COMMENTS generation for ${GENERATED_TARGET_NAME}") set(GENERATION_OPTIONS "${GENERATION_OPTIONS}:COMMENTS") endif() if(qtprotobuf_generate_FOLDER) message(STATUS "Enabled FOLDER generation for ${GENERATED_TARGET_NAME}") set(GENERATION_OPTIONS "${GENERATION_OPTIONS}:FOLDER") set(FOLDER_ENABLED "FOLDER") endif() if(WIN32) set(PROTOC_COMMAND set QT_PROTOBUF_OPTIONS=${GENERATION_OPTIONS}&& $) else() set(PROTOC_COMMAND "QT_PROTOBUF_OPTIONS=${GENERATION_OPTIONS}" $) endif() set(PROTO_INCLUDES ${qtprotobuf_generate_PROTO_INCLUDES}) if(DEFINED qtprotobuf_generate_GENERATED_HEADERS) set(GENERATED_HEADERS ${qtprotobuf_generate_GENERATED_HEADERS}) else() find_program(GO_EXECUTABLE "go") if (GO_EXECUTABLE STREQUAL GO_EXECUTABLE-NOTFOUND) message(FATAL_ERROR "Golang is mandatory dependency for QtProtobuf if GENERATED_HEADERS is not specified. Please install it and ensure that it's accessible by PATH environment variable") endif() 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} ${FOLDER_ENABLED} OUTPUT_VARIABLE GENERATED_HEADERS_PART ERROR_VARIABLE PARSER_ERROR) set(GENERATED_HEADERS ${GENERATED_HEADERS} ${GENERATED_HEADERS_PART}) endforeach() endif() if(qtprotobuf_generate_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() if(NOT "${PARSER_ERROR}" STREQUAL "") message(FATAL_ERROR "Go parser error ${PARSER_ERROR}") 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(GENERATED_SOURCES_FULL) unset(GENERATED_HEADERS_FULL) foreach(GENERATED_HEADER IN LISTS GENERATED_HEADERS) get_filename_component(GENERATED_DIRECTORY ${GENERATED_HEADER} DIRECTORY) get_filename_component(GENERATED_BASENAME ${GENERATED_HEADER} NAME) string(REGEX REPLACE "\\.[^.]*$" "" GENERATED_BASENAME ${GENERATED_BASENAME}) if(qtprotobuf_generate_FOLDER) list(APPEND GENERATED_SOURCES_FULL ${OUT_DIR}/${GENERATED_DIRECTORY}/${GENERATED_BASENAME}.cpp) list(APPEND GENERATED_HEADERS_FULL ${OUT_DIR}/${GENERATED_DIRECTORY}/${GENERATED_BASENAME}.h) else() list(APPEND GENERATED_SOURCES_FULL ${OUT_DIR}/${GENERATED_BASENAME}.cpp) list(APPEND GENERATED_HEADERS_FULL ${OUT_DIR}/${GENERATED_BASENAME}.h) endif() set_property(SOURCE ${OUT_DIR}/${GENERATED_DIRECTORY}/${GENERATED_BASENAME}.cpp PROPERTY SKIP_AUTOMOC ON) endforeach() set_source_files_properties(${GENERATED_SOURCES_FULL} PROPERTIES GENERATED TRUE) qt5_wrap_cpp(MOC_SOURCES ${GENERATED_HEADERS_FULL}) add_custom_command( OUTPUT ${GENERATED_SOURCES_FULL} ${GENERATED_HEADERS_FULL} COMMAND ${PROTOC_COMMAND} --plugin=protoc-gen-${GENERATOR_TARGET}=${QT_PROTOBUF_EXECUTABLE} --${GENERATOR_TARGET}_out=${OUT_DIR} ${PROTO_INCLUDES} ${qtprotobuf_generate_PROTO_FILES} WORKING_DIRECTORY ${OUT_DIR} DEPENDS ${qtprotobuf_generate_PROTO_FILES} ${QT_PROTOBUF_EXECUTABLE} COMMENT "Generating QtProtobuf ${GENERATED_TARGET_NAME} sources..." ) add_custom_target(${GEN_TARGET} DEPENDS ${GENERATED_SOURCES_FULL} ${GENERATED_HEADERS_FULL} ${qtprotobuf_generate_PROTO_FILES}) add_library(${GENERATED_TARGET_NAME} ${GENERATED_SOURCES_FULL} ${MOC_SOURCES}) add_dependencies(${GENERATED_TARGET_NAME} ${GEN_TARGET}) set_target_properties(${GENERATED_TARGET_NAME} PROPERTIES PUBLIC_HEADER "${GENERATED_HEADERS_FULL}") #Add include directories in case if projects are enabled by find_project target_include_directories(${GENERATED_TARGET_NAME} PUBLIC ${OUT_DIR} PRIVATE ${Qt5Core_INCLUDE_DIRS} $) if(TARGET ${QT_PROTOBUF_PROJECT}::QtGrpc) target_include_directories(${GENERATED_TARGET_NAME} PRIVATE $) endif() if(NOT WIN32) if(TARGET ${QT_PROTOBUF_PROJECT}::QtProtobufWellKnownTypes) target_include_directories(${GENERATED_TARGET_NAME} PRIVATE $) endif() endif() #Automatically link whole static library to specified in parameters target if(DEFINED qtprotobuf_generate_TARGET) qtprotobuf_link_archive(${qtprotobuf_generate_TARGET} ${GENERATED_TARGET_NAME}) endif() endfunction()