QtProtobufGen.cmake 8.1 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186
  1. function(qtprotobuf_link_target TARGET GENERATED_TARGET)
  2. target_sources(${TARGET} PRIVATE $<TARGET_OBJECTS:${GENERATED_TARGET}>)
  3. target_include_directories(${TARGET} PRIVATE $<TARGET_PROPERTY:${GENERATED_TARGET},INTERFACE_INCLUDE_DIRECTORIES>)
  4. add_dependencies(${TARGET} ${GENERATED_TARGET})
  5. endfunction()
  6. function(qtprotobuf_generate)
  7. set(options MULTI QML COMMENTS FOLDER FIELDENUM)
  8. set(oneValueArgs OUT_DIR TARGET GENERATED_TARGET EXTRA_NAMESPACE)
  9. set(multiValueArgs GENERATED_HEADERS EXCLUDE_HEADERS PROTO_FILES PROTO_INCLUDES)
  10. cmake_parse_arguments(qtprotobuf_generate "${options}" "${oneValueArgs}" "${multiValueArgs}" ${ARGN})
  11. if(DEFINED qtprotobuf_generate_GENERATED_TARGET)
  12. set(GENERATED_TARGET_NAME ${qtprotobuf_generate_GENERATED_TARGET})
  13. elseif(DEFINED qtprotobuf_generate_TARGET)
  14. set(GENERATED_TARGET_NAME ${qtprotobuf_generate_TARGET}_qtprotobuf_gen)
  15. else()
  16. message(FATAL_ERROR "Either TARGET or GENERATED_TARGET must be specified")
  17. endif()
  18. set(deps_target ${GENERATED_TARGET_NAME}_deps)
  19. set(QT_PROTOBUF_EXECUTABLE $<TARGET_FILE:${QT_PROTOBUF_NAMESPACE}::qtprotobufgen>)
  20. #Options handling
  21. set(GENERATION_TYPE "SIGNLE")
  22. if(qtprotobuf_generate_MULTI)
  23. set(GENERATION_TYPE "MULTI")
  24. set(qtprotobuf_generate_FOLDER TRUE)
  25. message(STATUS "Multi-file generation is set. Folder-based generation enabled automatically for ${GENERATED_TARGET_NAME}")
  26. endif()
  27. set(GENERATION_OPTIONS ${GENERATION_TYPE})
  28. if(qtprotobuf_generate_QML)
  29. message(STATUS "Enabled QML generation for ${GENERATED_TARGET_NAME}")
  30. set(GENERATION_OPTIONS "${GENERATION_OPTIONS}:QML")
  31. endif()
  32. if(qtprotobuf_generate_COMMENTS)
  33. message(STATUS "Enabled COMMENTS generation for ${GENERATED_TARGET_NAME}")
  34. set(GENERATION_OPTIONS "${GENERATION_OPTIONS}:COMMENTS")
  35. endif()
  36. if(qtprotobuf_generate_FOLDER)
  37. message(STATUS "Enabled FOLDER generation for ${GENERATED_TARGET_NAME}")
  38. set(GENERATION_OPTIONS "${GENERATION_OPTIONS}:FOLDER")
  39. set(FOLDER_ENABLED "FOLDER")
  40. endif()
  41. if(qtprotobuf_generate_FIELDENUM)
  42. message(STATUS "Enabled FIELDENUM generation for ${GENERATED_TARGET_NAME}")
  43. set(GENERATION_OPTIONS "${GENERATION_OPTIONS}:FIELDENUM")
  44. endif()
  45. if(qtprotobuf_generate_EXTRA_NAMESPACE)
  46. set(GENERATION_OPTIONS
  47. "${GENERATION_OPTIONS}:EXTRA_NAMESPACE=\"${qtprotobuf_generate_EXTRA_NAMESPACE}\""
  48. )
  49. endif()
  50. if(WIN32)
  51. set(PROTOC_COMMAND set QT_PROTOBUF_OPTIONS=${GENERATION_OPTIONS}&& $<TARGET_FILE:protobuf::protoc>)
  52. else()
  53. set(PROTOC_COMMAND "QT_PROTOBUF_OPTIONS=${GENERATION_OPTIONS}" $<TARGET_FILE:protobuf::protoc>)
  54. endif()
  55. set(PROTO_INCLUDES ${qtprotobuf_generate_PROTO_INCLUDES})
  56. if(DEFINED qtprotobuf_generate_GENERATED_HEADERS)
  57. set(GENERATED_HEADERS ${qtprotobuf_generate_GENERATED_HEADERS})
  58. else()
  59. find_program(GO_EXECUTABLE "go")
  60. if (GO_EXECUTABLE STREQUAL GO_EXECUTABLE-NOTFOUND)
  61. 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")
  62. endif()
  63. get_target_property(PROTO_PARSER ${QT_PROTOBUF_NAMESPACE}::qtprotobufgen PROTO_PARSER)
  64. if(NOT PROTO_PARSER)
  65. message(FATAL_ERROR "Unable to locate parsemessages.go script")
  66. endif()
  67. foreach(PROTO_FILE IN LISTS qtprotobuf_generate_PROTO_FILES)
  68. get_filename_component(BASE_DIR ${PROTO_FILE} DIRECTORY)
  69. set(PROTO_INCLUDES "-I\"${BASE_DIR}\"" ${PROTO_INCLUDES})
  70. execute_process(COMMAND ${GO_EXECUTABLE} run ${PROTO_PARSER} ${PROTO_FILE} ${GENERATION_TYPE} ${FOLDER_ENABLED} OUTPUT_VARIABLE GENERATED_HEADERS_PART ERROR_VARIABLE PARSER_ERROR)
  71. list(APPEND GENERATED_HEADERS ${GENERATED_HEADERS_PART})
  72. endforeach()
  73. endif()
  74. if(qtprotobuf_generate_MULTI)
  75. #TODO: add globalenums by default. But it's better to verify if proto file contains any global enum
  76. set(GENERATED_HEADERS ${GENERATED_HEADERS} globalenums.h)
  77. endif()
  78. if(NOT "${PARSER_ERROR}" STREQUAL "")
  79. message(FATAL_ERROR "Go parser error ${PARSER_ERROR}")
  80. endif()
  81. foreach(EXCLUDE_HEADER IN LISTS qtprotobuf_generate_EXCLUDE_HEADERS)
  82. list(REMOVE_ITEM GENERATED_HEADERS ${EXCLUDE_HEADER})
  83. endforeach()
  84. list(REMOVE_DUPLICATES GENERATED_HEADERS)
  85. if(NOT DEFINED qtprotobuf_generate_OUT_DIR)
  86. set(OUT_DIR ${CMAKE_CURRENT_BINARY_DIR})
  87. else()
  88. set(OUT_DIR ${qtprotobuf_generate_OUT_DIR})
  89. endif()
  90. file(MAKE_DIRECTORY ${OUT_DIR})
  91. unset(GENERATED_SOURCES_FULL)
  92. unset(GENERATED_HEADERS_FULL)
  93. foreach(GENERATED_HEADER IN LISTS GENERATED_HEADERS)
  94. get_filename_component(GENERATED_DIRECTORY ${GENERATED_HEADER} DIRECTORY)
  95. get_filename_component(GENERATED_BASENAME ${GENERATED_HEADER} NAME)
  96. string(REGEX REPLACE "\\.[^.]*$" "" GENERATED_BASENAME ${GENERATED_BASENAME})
  97. if(qtprotobuf_generate_FOLDER)
  98. list(APPEND GENERATED_SOURCES_FULL "${OUT_DIR}/${GENERATED_DIRECTORY}/${GENERATED_BASENAME}.cpp")
  99. list(APPEND GENERATED_HEADERS_FULL "${OUT_DIR}/${GENERATED_DIRECTORY}/${GENERATED_BASENAME}.h")
  100. else()
  101. list(APPEND GENERATED_SOURCES_FULL "${OUT_DIR}/${GENERATED_BASENAME}.cpp")
  102. list(APPEND GENERATED_HEADERS_FULL "${OUT_DIR}/${GENERATED_BASENAME}.h")
  103. endif()
  104. endforeach()
  105. set_source_files_properties(${GENERATED_SOURCES_FULL};${GENERATED_HEADERS_FULL} PROPERTIES
  106. GENERATED TRUE
  107. SKIP_AUTOMOC ON
  108. SKIP_AUTOUIC ON
  109. SKIP_AUTOGEN ON
  110. )
  111. qt5_wrap_cpp(MOC_SOURCES ${GENERATED_HEADERS_FULL})
  112. add_custom_command(
  113. OUTPUT ${GENERATED_SOURCES_FULL} ${GENERATED_HEADERS_FULL}
  114. COMMAND ${PROTOC_COMMAND}
  115. --plugin=protoc-gen-qtprotobufgen=${QT_PROTOBUF_EXECUTABLE}
  116. --qtprotobufgen_out=${OUT_DIR}
  117. ${PROTO_INCLUDES}
  118. ${qtprotobuf_generate_PROTO_FILES}
  119. WORKING_DIRECTORY ${OUT_DIR}
  120. DEPENDS ${qtprotobuf_generate_PROTO_FILES} ${QT_PROTOBUF_EXECUTABLE}
  121. COMMENT "Generating QtProtobuf ${GENERATED_TARGET_NAME} sources..."
  122. COMMAND_EXPAND_LISTS
  123. )
  124. add_custom_target(${deps_target} DEPENDS ${GENERATED_SOURCES_FULL} ${GENERATED_HEADERS_FULL}
  125. ${qtprotobuf_generate_PROTO_FILES}
  126. )
  127. add_library(${GENERATED_TARGET_NAME} OBJECT ${GENERATED_SOURCES_FULL} ${MOC_SOURCES})
  128. add_dependencies(${GENERATED_TARGET_NAME} ${deps_target})
  129. if(qtprotobuf_generate_TARGET)
  130. set_property(TARGET ${qtprotobuf_generate_TARGET} APPEND PROPERTY PUBLIC_HEADER ${GENERATED_HEADERS_FULL})
  131. endif()
  132. set_target_properties(${GENERATED_TARGET_NAME} PROPERTIES PUBLIC_HEADER "${GENERATED_HEADERS_FULL}")
  133. #Add include directories in case if targets are found by find_project or in source tree
  134. target_include_directories(${GENERATED_TARGET_NAME} PUBLIC ${OUT_DIR} PRIVATE
  135. $<TARGET_PROPERTY:${QT_PROTOBUF_NAMESPACE}::Protobuf,INTERFACE_INCLUDE_DIRECTORIES>)
  136. #TODO: Do not link targets if they are not used in .proto files.
  137. if(TARGET ${QT_PROTOBUF_NAMESPACE}::Grpc)
  138. target_include_directories(${GENERATED_TARGET_NAME} PRIVATE
  139. $<TARGET_PROPERTY:${QT_PROTOBUF_NAMESPACE}::Grpc,INTERFACE_INCLUDE_DIRECTORIES>)
  140. endif()
  141. if(TARGET ${QT_PROTOBUF_NAMESPACE}::ProtobufWellKnownTypes)
  142. target_include_directories(${GENERATED_TARGET_NAME} PRIVATE
  143. $<TARGET_PROPERTY:${QT_PROTOBUF_NAMESPACE}::ProtobufWellKnownTypes,INTERFACE_INCLUDE_DIRECTORIES>)
  144. endif()
  145. if(TARGET ${QT_PROTOBUF_NAMESPACE}::ProtobufQtTypes)
  146. target_include_directories(${GENERATED_TARGET_NAME} PRIVATE
  147. $<TARGET_PROPERTY:${QT_PROTOBUF_NAMESPACE}::ProtobufQtTypes,INTERFACE_INCLUDE_DIRECTORIES>)
  148. endif()
  149. #Automatically link whole static library to specified in parameters target
  150. if(DEFINED qtprotobuf_generate_TARGET)
  151. qtprotobuf_link_target(${qtprotobuf_generate_TARGET} ${GENERATED_TARGET_NAME})
  152. endif()
  153. endfunction()