QtProtobufGen.cmake 7.1 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156
  1. function(qtprotobuf_link_archive TARGET GENERATED_TARGET)
  2. set_target_properties(${qtprotobuf_generate_TARGET} PROPERTIES LINKER_LANGUAGE CXX)
  3. set_target_properties(${TARGET} PROPERTIES LINK_DIRECTORIES "$<TARGET_FILE_DIR:${GENERATED_TARGET}>")
  4. get_target_property(EXISTING_FLAGS ${TARGET} LINK_FLAGS)
  5. if(WIN32)
  6. if("${EXISTING_FLAGS}" STREQUAL "EXISTING_FLAGS-NOTFOUND")
  7. set_target_properties(${TARGET} PROPERTIES LINK_FLAGS "/WHOLEARCHIVE:${GENERATED_TARGET}")
  8. else()
  9. set_target_properties(${TARGET} PROPERTIES LINK_FLAGS "${EXISTING_FLAGS} /WHOLEARCHIVE:${GENERATED_TARGET}")
  10. endif()
  11. else()
  12. target_link_libraries(${TARGET} PRIVATE
  13. -Wl,--whole-archive $<TARGET_FILE:${GENERATED_TARGET}> -Wl,--no-whole-archive)
  14. endif()
  15. target_include_directories(${TARGET} PRIVATE $<TARGET_PROPERTY:${GENERATED_TARGET},INCLUDE_DIRECTORIES>)
  16. add_dependencies(${TARGET} ${GENERATED_TARGET})
  17. endfunction()
  18. function(qtprotobuf_generate)
  19. set(options MULTI QML COMMENTS)
  20. set(oneValueArgs OUT_DIR TARGET GENERATED_TARGET)
  21. set(multiValueArgs GENERATED_HEADERS EXCLUDE_HEADERS PROTO_FILES PROTO_INCLUDES)
  22. cmake_parse_arguments(qtprotobuf_generate "${options}" "${oneValueArgs}" "${multiValueArgs}" ${ARGN})
  23. if(DEFINED qtprotobuf_generate_GENERATED_TARGET)
  24. set(GENERATED_TARGET_NAME ${qtprotobuf_generate_GENERATED_TARGET})
  25. elseif(DEFINED qtprotobuf_generate_TARGET)
  26. set(GENERATED_TARGET_NAME ${qtprotobuf_generate_TARGET}_qtprotobuf_gen)
  27. else()
  28. message(FATAL_ERROR "Either TARGET or GENERATED_TARGET must be specified")
  29. endif()
  30. set(GEN_TARGET ${GENERATED_TARGET_NAME}_generate)
  31. if(NOT DEFINED QT_PROTOBUF_EXECUTABLE)
  32. set(QT_PROTOBUF_EXECUTABLE "${QT_PROTOBUF_EXECUTABLE_INSTALL}")
  33. endif()
  34. #Options handling
  35. set(GENERATION_TYPE "SIGNLE")
  36. if(qtprotobuf_generate_MULTI)
  37. set(GENERATION_TYPE "MULTI")
  38. endif()
  39. set(GENERATION_OPTIONS ${GENERATION_TYPE})
  40. if(qtprotobuf_generate_QML)
  41. message(STATUS "Enabled QML generation for ${GENERATED_TARGET_NAME}")
  42. set(GENERATION_OPTIONS "${GENERATION_OPTIONS}:QML")
  43. endif()
  44. if(qtprotobuf_generate_COMMENTS)
  45. message(STATUS "Enabled COMMENTS generation for ${GENERATED_TARGET_NAME}")
  46. set(GENERATION_OPTIONS "${GENERATION_OPTIONS}:COMMENTS")
  47. endif()
  48. if(WIN32)
  49. set(PROTOC_COMMAND set QT_PROTOBUF_OPTIONS=${GENERATION_OPTIONS}&& $<TARGET_FILE:protobuf::protoc>)
  50. else()
  51. set(PROTOC_COMMAND "QT_PROTOBUF_OPTIONS=${GENERATION_OPTIONS}" $<TARGET_FILE:protobuf::protoc>)
  52. endif()
  53. set(PROTO_INCLUDES ${qtprotobuf_generate_PROTO_INCLUDES})
  54. if(DEFINED qtprotobuf_generate_GENERATED_HEADERS)
  55. set(GENERATED_HEADERS ${qtprotobuf_generate_GENERATED_HEADERS})
  56. else()
  57. find_program(GO_EXECUTABLE "go")
  58. if (GO_EXECUTABLE STREQUAL GO_EXECUTABLE-NOTFOUND)
  59. 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")
  60. endif()
  61. foreach(PROTO_FILE IN LISTS qtprotobuf_generate_PROTO_FILES)
  62. get_filename_component(BASE_DIR ${PROTO_FILE} DIRECTORY)
  63. set(PROTO_INCLUDES -I"${BASE_DIR}" ${PROTO_INCLUDES})
  64. execute_process(COMMAND ${GO_EXECUTABLE} run ${PROTO_PARSER} ${PROTO_FILE} ${GENERATION_TYPE} OUTPUT_VARIABLE GENERATED_HEADERS_PART ERROR_VARIABLE PARSER_ERROR)
  65. set(GENERATED_HEADERS ${GENERATED_HEADERS} ${GENERATED_HEADERS_PART})
  66. endforeach()
  67. endif()
  68. if(qtprotobuf_generate_MULTI)
  69. #TODO: add globalenums by default. But it's better to verify if proto file contains any global enum
  70. set(GENERATED_HEADERS ${GENERATED_HEADERS} globalenums.h)
  71. endif()
  72. if(NOT "${PARSER_ERROR}" STREQUAL "")
  73. message(FATAL_ERROR "Go parser error ${PARSER_ERROR}")
  74. endif()
  75. foreach(EXCLUDE_HEADER IN LISTS qtprotobuf_generate_EXCLUDE_HEADERS)
  76. list(REMOVE_ITEM GENERATED_HEADERS ${EXCLUDE_HEADER})
  77. endforeach()
  78. list(REMOVE_DUPLICATES GENERATED_HEADERS)
  79. if(NOT DEFINED qtprotobuf_generate_OUT_DIR)
  80. set(OUT_DIR ${CMAKE_CURRENT_BINARY_DIR})
  81. else()
  82. set(OUT_DIR ${qtprotobuf_generate_OUT_DIR})
  83. endif()
  84. file(MAKE_DIRECTORY ${OUT_DIR})
  85. unset(GENERATED_SOURCES_FULL)
  86. unset(GENERATED_HEADERS_FULL)
  87. foreach(GENERATED_HEADER IN LISTS GENERATED_HEADERS)
  88. get_filename_component(GENERATED_DIRECTORY ${GENERATED_HEADER} DIRECTORY)
  89. get_filename_component(GENERATED_BASENAME ${GENERATED_HEADER} NAME)
  90. string(REGEX REPLACE "\\.[^.]*$" "" GENERATED_BASENAME ${GENERATED_BASENAME})
  91. list(APPEND GENERATED_SOURCES_FULL ${OUT_DIR}/${GENERATED_DIRECTORY}/${GENERATED_BASENAME}.cpp)
  92. list(APPEND GENERATED_HEADERS_FULL ${OUT_DIR}/${GENERATED_DIRECTORY}/${GENERATED_BASENAME}.h)
  93. set_property(SOURCE ${OUT_DIR}/${GENERATED_DIRECTORY}/${GENERATED_BASENAME}.cpp PROPERTY SKIP_AUTOMOC ON)
  94. endforeach()
  95. set_source_files_properties(${GENERATED_SOURCES_FULL} PROPERTIES GENERATED TRUE)
  96. qt5_wrap_cpp(MOC_SOURCES ${GENERATED_HEADERS_FULL})
  97. add_custom_command(
  98. OUTPUT ${GENERATED_SOURCES_FULL} ${GENERATED_HEADERS_FULL}
  99. COMMAND ${PROTOC_COMMAND}
  100. --plugin=protoc-gen-${GENERATOR_TARGET}=${QT_PROTOBUF_EXECUTABLE}
  101. --${GENERATOR_TARGET}_out=${OUT_DIR}
  102. ${PROTO_INCLUDES}
  103. ${qtprotobuf_generate_PROTO_FILES}
  104. WORKING_DIRECTORY ${OUT_DIR}
  105. DEPENDS ${qtprotobuf_generate_PROTO_FILES} ${QT_PROTOBUF_EXECUTABLE}
  106. COMMENT "Generating QtProtobuf ${GENERATED_TARGET_NAME} sources..."
  107. )
  108. add_custom_target(${GEN_TARGET} DEPENDS ${GENERATED_SOURCES_FULL} ${GENERATED_HEADERS_FULL} ${qtprotobuf_generate_PROTO_FILES})
  109. add_library(${GENERATED_TARGET_NAME} ${GENERATED_SOURCES_FULL} ${MOC_SOURCES})
  110. add_dependencies(${GENERATED_TARGET_NAME} ${GEN_TARGET})
  111. set_target_properties(${GENERATED_TARGET_NAME} PROPERTIES PUBLIC_HEADER "${GENERATED_HEADERS_FULL}")
  112. #Add include directories in case if projects are enabled by find_project
  113. target_include_directories(${GENERATED_TARGET_NAME} PUBLIC ${OUT_DIR} PRIVATE ${Qt5Core_INCLUDE_DIRS}
  114. $<TARGET_PROPERTY:${QT_PROTOBUF_PROJECT}::QtProtobuf,INTERFACE_INCLUDE_DIRECTORIES>)
  115. if(TARGET ${QT_PROTOBUF_PROJECT}::QtGrpc)
  116. target_include_directories(${GENERATED_TARGET_NAME} PRIVATE
  117. $<TARGET_PROPERTY:${QT_PROTOBUF_PROJECT}::QtGrpc,INTERFACE_INCLUDE_DIRECTORIES>)
  118. endif()
  119. if(NOT WIN32)
  120. if(TARGET ${QT_PROTOBUF_PROJECT}::QtProtobufWellKnownTypes)
  121. target_include_directories(${GENERATED_TARGET_NAME} PRIVATE
  122. $<TARGET_PROPERTY:${QT_PROTOBUF_PROJECT}::QtProtobufWellKnownTypes,INTERFACE_INCLUDE_DIRECTORIES>)
  123. endif()
  124. endif()
  125. #Automatically link whole static library to specified in parameters target
  126. if(DEFINED qtprotobuf_generate_TARGET)
  127. qtprotobuf_link_archive(${qtprotobuf_generate_TARGET} ${GENERATED_TARGET_NAME})
  128. endif()
  129. endfunction()