|
| 1 | +# CMake function for generating Python bindings |
| 2 | +# |
| 3 | +# This function: |
| 4 | +# 1. Creates a custom command to generate the core and enums Python modules |
| 5 | +# 2. Creates a custom command to copy all files to the output directory |
| 6 | +# 3. Creates a custom target that depends on all output files |
| 7 | +# |
| 8 | +# Usage: |
| 9 | +# generate_python_bindings( |
| 10 | +# TARGET_NAME <target_name> |
| 11 | +# DISPLAY_NAME <display_name> |
| 12 | +# GENERATOR_TARGET <generator_executable_target> |
| 13 | +# HEADER_FILE <path_to_header> |
| 14 | +# [TEMPLATE_FILE <path_to_template>] |
| 15 | +# OUTPUT_DIRECTORY <output_directory> |
| 16 | +# CORE_OUTPUT_FILE <core_output_file> |
| 17 | +# ENUMS_OUTPUT_FILE <enums_output_file> |
| 18 | +# [PYTHON_SOURCES <list_of_python_source_files>] |
| 19 | +# ) |
| 20 | +# |
| 21 | +function(generate_python_bindings) |
| 22 | + set(options) |
| 23 | + set(oneValueArgs TARGET_NAME DISPLAY_NAME GENERATOR_TARGET HEADER_FILE TEMPLATE_FILE OUTPUT_DIRECTORY CORE_OUTPUT_FILE ENUMS_OUTPUT_FILE) |
| 24 | + set(multiValueArgs PYTHON_SOURCES) |
| 25 | + |
| 26 | + cmake_parse_arguments(PARSE_ARGV 0 ARGS "${options}" "${oneValueArgs}" "${multiValueArgs}") |
| 27 | + |
| 28 | + foreach(required_arg TARGET_NAME DISPLAY_NAME GENERATOR_TARGET HEADER_FILE OUTPUT_DIRECTORY CORE_OUTPUT_FILE ENUMS_OUTPUT_FILE) |
| 29 | + if(NOT ARGS_${required_arg}) |
| 30 | + message(FATAL_ERROR "${required_arg} is required") |
| 31 | + endif() |
| 32 | + endforeach() |
| 33 | + |
| 34 | + set(CORE_SOURCE_PATH ${PROJECT_SOURCE_DIR}/${ARGS_CORE_OUTPUT_FILE}) |
| 35 | + set(ENUMS_SOURCE_PATH ${PROJECT_SOURCE_DIR}/${ARGS_ENUMS_OUTPUT_FILE}) |
| 36 | + |
| 37 | + set(GENERATOR_DEPENDS ${ARGS_HEADER_FILE} $<TARGET_FILE:${ARGS_GENERATOR_TARGET}>) |
| 38 | + list(APPEND GENERATOR_DEPENDS ${ARGS_TEMPLATE_FILE}) |
| 39 | + |
| 40 | + add_custom_command( |
| 41 | + OUTPUT ${CORE_SOURCE_PATH} ${ENUMS_SOURCE_PATH} |
| 42 | + DEPENDS ${GENERATOR_DEPENDS} |
| 43 | + COMMENT "Generating ${ARGS_DISPLAY_NAME} Python Sources" |
| 44 | + COMMAND ${CMAKE_COMMAND} -E env ASAN_OPTIONS=detect_leaks=0 $<TARGET_FILE:${ARGS_GENERATOR_TARGET}> |
| 45 | + ${ARGS_HEADER_FILE} |
| 46 | + ${CORE_SOURCE_PATH} |
| 47 | + ${ARGS_TEMPLATE_FILE} |
| 48 | + ${ENUMS_SOURCE_PATH} |
| 49 | + VERBATIM |
| 50 | + ) |
| 51 | + |
| 52 | + set(PYTHON_OUTPUT_FILES) |
| 53 | + foreach(SOURCE_FILE ${ARGS_PYTHON_SOURCES}) |
| 54 | + cmake_path(RELATIVE_PATH SOURCE_FILE BASE_DIRECTORY ${PROJECT_SOURCE_DIR} OUTPUT_VARIABLE REL_PATH) |
| 55 | + list(APPEND PYTHON_OUTPUT_FILES ${ARGS_OUTPUT_DIRECTORY}/${REL_PATH}) |
| 56 | + endforeach() |
| 57 | + |
| 58 | + list(APPEND PYTHON_OUTPUT_FILES ${ARGS_OUTPUT_DIRECTORY}/${ARGS_CORE_OUTPUT_FILE}) |
| 59 | + list(APPEND PYTHON_OUTPUT_FILES ${ARGS_OUTPUT_DIRECTORY}/${ARGS_ENUMS_OUTPUT_FILE}) |
| 60 | + |
| 61 | + set(COPY_DEPENDENCIES ${CORE_SOURCE_PATH} ${ENUMS_SOURCE_PATH}) |
| 62 | + list(APPEND COPY_DEPENDENCIES ${ARGS_PYTHON_SOURCES}) |
| 63 | + |
| 64 | + # Generate a script to copy the generated Python files, preserving their directory structure. |
| 65 | + file(GENERATE OUTPUT ${PROJECT_BINARY_DIR}/copy_python_sources.cmake |
| 66 | + CONTENT " |
| 67 | + foreach(PYTHON_SOURCE ${ARGS_PYTHON_SOURCES}) |
| 68 | + cmake_path(RELATIVE_PATH PYTHON_SOURCE BASE_DIRECTORY ${PROJECT_SOURCE_DIR} OUTPUT_VARIABLE OUTPUT_SUBPATH) |
| 69 | + cmake_path(REMOVE_FILENAME OUTPUT_SUBPATH) |
| 70 | + file(COPY $\{PYTHON_SOURCE\} DESTINATION ${ARGS_OUTPUT_DIRECTORY}/$\{OUTPUT_SUBPATH\}) |
| 71 | + endforeach() |
| 72 | + " |
| 73 | + ) |
| 74 | + |
| 75 | + add_custom_command( |
| 76 | + OUTPUT ${PYTHON_OUTPUT_FILES} |
| 77 | + DEPENDS ${COPY_DEPENDENCIES} |
| 78 | + COMMENT "Copying ${ARGS_DISPLAY_NAME} Python Sources" |
| 79 | + COMMAND ${CMAKE_COMMAND} -E make_directory ${ARGS_OUTPUT_DIRECTORY} |
| 80 | + COMMAND ${CMAKE_COMMAND} -P ${PROJECT_BINARY_DIR}/copy_python_sources.cmake |
| 81 | + COMMAND ${CMAKE_COMMAND} -E copy ${CORE_SOURCE_PATH} ${ARGS_OUTPUT_DIRECTORY} |
| 82 | + COMMAND ${CMAKE_COMMAND} -E copy ${ENUMS_SOURCE_PATH} ${ARGS_OUTPUT_DIRECTORY} |
| 83 | + VERBATIM |
| 84 | + ) |
| 85 | + |
| 86 | + # Create target that depends on all output files |
| 87 | + add_custom_target(${ARGS_TARGET_NAME} ALL DEPENDS ${PYTHON_OUTPUT_FILES}) |
| 88 | +endfunction() |
0 commit comments