| # CAPNP_GENERATE_CPP =========================================================== |
| # |
| # Example usage: |
| # find_package(CapnProto) |
| # capnp_generate_cpp(CAPNP_SRCS CAPNP_HDRS schema.capnp) |
| # add_executable(foo main.cpp ${CAPNP_SRCS}) |
| # target_link_libraries(foo PRIVATE CapnProto::capnp-rpc) |
| # target_include_directories(foo PRIVATE ${CMAKE_CURRENT_BINARY_DIR}) |
| # |
| # If you are not using the RPC features you can use 'CapnProto::capnp' in the |
| # target_link_libraries call |
| # |
| # Configuration variables (optional): |
| # CAPNPC_OUTPUT_DIR |
| # Directory to place compiled schema sources (default: CMAKE_CURRENT_BINARY_DIR). |
| # CAPNPC_IMPORT_DIRS |
| # List of additional include directories for the schema compiler. |
| # (CAPNPC_SRC_PREFIX and CAPNP_INCLUDE_DIRECTORY are always included.) |
| # CAPNPC_SRC_PREFIX |
| # Schema file source prefix (default: CMAKE_CURRENT_SOURCE_DIR). |
| # CAPNPC_FLAGS |
| # Additional flags to pass to the schema compiler. |
| # |
| # TODO: convert to cmake_parse_arguments |
| |
| function(CAPNP_GENERATE_CPP SOURCES HEADERS) |
| if(NOT ARGN) |
| message(SEND_ERROR "CAPNP_GENERATE_CPP() called without any source files.") |
| endif() |
| set(tool_depends ${EMPTY_STRING}) |
| #Use cmake targets available |
| if(TARGET capnp_tool) |
| if(NOT CAPNP_EXECUTABLE) |
| set(CAPNP_EXECUTABLE $<TARGET_FILE:capnp_tool>) |
| endif() |
| if(NOT CAPNPC_CXX_EXECUTABLE) |
| get_target_property(CAPNPC_CXX_EXECUTABLE capnpc_cpp CAPNPC_CXX_EXECUTABLE) |
| endif() |
| if(NOT CAPNP_INCLUDE_DIRECTORY) |
| get_target_property(CAPNP_INCLUDE_DIRECTORY capnp_tool CAPNP_INCLUDE_DIRECTORY) |
| endif() |
| list(APPEND tool_depends capnp_tool capnpc_cpp) |
| endif() |
| if(NOT CAPNP_EXECUTABLE) |
| message(SEND_ERROR "Could not locate capnp executable (CAPNP_EXECUTABLE).") |
| endif() |
| if(NOT CAPNPC_CXX_EXECUTABLE) |
| message(SEND_ERROR "Could not locate capnpc-c++ executable (CAPNPC_CXX_EXECUTABLE).") |
| endif() |
| if(NOT CAPNP_INCLUDE_DIRECTORY) |
| message(SEND_ERROR "Could not locate capnp header files (CAPNP_INCLUDE_DIRECTORY).") |
| endif() |
| |
| if(DEFINED CAPNPC_OUTPUT_DIR) |
| # Prepend a ':' to get the format for the '-o' flag right |
| set(output_dir ":${CAPNPC_OUTPUT_DIR}") |
| else() |
| set(output_dir ":.") |
| endif() |
| |
| if(NOT DEFINED CAPNPC_SRC_PREFIX) |
| set(CAPNPC_SRC_PREFIX "${CMAKE_CURRENT_SOURCE_DIR}") |
| endif() |
| get_filename_component(CAPNPC_SRC_PREFIX "${CAPNPC_SRC_PREFIX}" ABSOLUTE) |
| |
| # Default compiler includes. Note that in capnp's own test usage of capnp_generate_cpp(), these |
| # two variables will end up evaluating to the same directory. However, it's difficult to |
| # deduplicate them because if CAPNP_INCLUDE_DIRECTORY came from the capnp_tool target property, |
| # then it must be a generator expression in order to handle usages in both the build tree and the |
| # install tree. This vastly overcomplicates duplication detection, so the duplication doesn't seem |
| # worth fixing. |
| set(include_path -I "${CAPNPC_SRC_PREFIX}" -I "${CAPNP_INCLUDE_DIRECTORY}") |
| |
| if(DEFINED CAPNPC_IMPORT_DIRS) |
| # Append each directory as a series of '-I' flags in ${include_path} |
| foreach(directory ${CAPNPC_IMPORT_DIRS}) |
| get_filename_component(absolute_path "${directory}" ABSOLUTE) |
| list(APPEND include_path -I "${absolute_path}") |
| endforeach() |
| endif() |
| |
| set(${SOURCES}) |
| set(${HEADERS}) |
| foreach(schema_file ${ARGN}) |
| get_filename_component(file_path "${schema_file}" ABSOLUTE) |
| get_filename_component(file_dir "${file_path}" PATH) |
| if(NOT EXISTS "${file_path}") |
| message(FATAL_ERROR "Cap'n Proto schema file '${file_path}' does not exist!") |
| endif() |
| |
| # Figure out where the output files will go |
| if (NOT DEFINED CAPNPC_OUTPUT_DIR) |
| set(CAPNPC_OUTPUT_DIR "${CMAKE_CURRENT_BINARY_DIR}/") |
| endif() |
| # Output files are placed in CAPNPC_OUTPUT_DIR, at a location as if they were |
| # relative to CAPNPC_SRC_PREFIX. |
| string(LENGTH "${CAPNPC_SRC_PREFIX}" prefix_len) |
| string(SUBSTRING "${file_path}" 0 ${prefix_len} output_prefix) |
| if(NOT "${CAPNPC_SRC_PREFIX}" STREQUAL "${output_prefix}") |
| message(SEND_ERROR "Could not determine output path for '${schema_file}' ('${file_path}') with source prefix '${CAPNPC_SRC_PREFIX}' into '${CAPNPC_OUTPUT_DIR}'.") |
| endif() |
| |
| string(SUBSTRING "${file_path}" ${prefix_len} -1 output_path) |
| set(output_base "${CAPNPC_OUTPUT_DIR}${output_path}") |
| |
| add_custom_command( |
| OUTPUT "${output_base}.c++" "${output_base}.h" |
| COMMAND "${CAPNP_EXECUTABLE}" |
| ARGS compile |
| -o ${CAPNPC_CXX_EXECUTABLE}${output_dir} |
| --src-prefix ${CAPNPC_SRC_PREFIX} |
| ${include_path} |
| ${CAPNPC_FLAGS} |
| ${file_path} |
| DEPENDS "${schema_file}" ${tool_depends} |
| COMMENT "Compiling Cap'n Proto schema ${schema_file}" |
| VERBATIM |
| ) |
| |
| list(APPEND ${SOURCES} "${output_base}.c++") |
| list(APPEND ${HEADERS} "${output_base}.h") |
| endforeach() |
| |
| set_source_files_properties(${${SOURCES}} ${${HEADERS}} PROPERTIES GENERATED TRUE) |
| set(${SOURCES} ${${SOURCES}} PARENT_SCOPE) |
| set(${HEADERS} ${${HEADERS}} PARENT_SCOPE) |
| endfunction() |