From acb1a3889a46366b79ef980d7f01a31e32f06e26 Mon Sep 17 00:00:00 2001 From: Vincent Richard Date: Tue, 30 May 2023 12:24:30 +0900 Subject: [PATCH] feat: support components in ament_export_dependencies Signed-off-by: Vincent Richard --- ..._cmake_export_dependencies-extras.cmake.in | 13 +++++- .../cmake/ament_export_dependencies.cmake | 40 +++++++++++++++---- 2 files changed, 44 insertions(+), 9 deletions(-) diff --git a/ament_cmake_export_dependencies/cmake/ament_cmake_export_dependencies-extras.cmake.in b/ament_cmake_export_dependencies/cmake/ament_cmake_export_dependencies-extras.cmake.in index ab3d1e3f..c9b2e0c5 100644 --- a/ament_cmake_export_dependencies/cmake/ament_cmake_export_dependencies-extras.cmake.in +++ b/ament_cmake_export_dependencies/cmake/ament_cmake_export_dependencies-extras.cmake.in @@ -17,8 +17,17 @@ if(NOT _exported_dependencies STREQUAL "") set(@PROJECT_NAME@_RECURSIVE_DEPENDENCIES ${_exported_dependencies}) set(_libraries) foreach(_dep ${_exported_dependencies}) - if(NOT ${_dep}_FOUND) - find_package("${_dep}" QUIET REQUIRED) + # unpack components (if any) + string(REGEX REPLACE ":" ";" _unpacked "${_dep}") + list(POP_FRONT _unpacked _dep) + if(NOT "${_unpacked}" STREQUAL "") + # has components + find_package("${_dep}" QUIET REQUIRED COMPONENTS "${_unpacked}") + else() + # no components + if(NOT ${_dep}_FOUND) + find_package("${_dep}" QUIET REQUIRED) + endif() endif() # if a package provides modern CMake interface targets use them # exclusively assuming the classic CMake variables only exist for diff --git a/ament_cmake_export_dependencies/cmake/ament_export_dependencies.cmake b/ament_cmake_export_dependencies/cmake/ament_export_dependencies.cmake index 6c38d99e..6c1c2255 100644 --- a/ament_cmake_export_dependencies/cmake/ament_export_dependencies.cmake +++ b/ament_cmake_export_dependencies/cmake/ament_export_dependencies.cmake @@ -15,12 +15,18 @@ # # Export dependencies to downstream packages. # +# Behavior depends on whether ``COMPONENTS`` parameter is provided. If provided, +# ``ament_export_dependencies`` expects a single package name, followed a list +# of components to export. Otherwise, it expects a list of packages to export. # Each package name must be find_package()-able with the exact same case. # Additionally the exported variables must have a prefix with the same case # and the suffixes must be INCLUDE_DIRS and LIBRARIES. # # :param ARGN: a list of package names # :type ARGN: list of strings +# :param COMPONENTS: optional list of components. In such cas, only 1 package +# name shall be listed +# :type COMPONENTS: list of strings # # @public # @@ -32,12 +38,32 @@ macro(ament_export_dependencies) if(${ARGC} GREATER 0) _ament_cmake_export_dependencies_register_package_hook() - foreach(_arg ${ARGN}) - # only pass package name - # will be resolved by downstream packages - # must be find_package()-able - # and provide _INCLUDE_DIRS and _LIBRARIES - list(APPEND _AMENT_CMAKE_EXPORT_DEPENDENCIES "${_arg}") - endforeach() + + cmake_parse_arguments(_ARGS "" "" "COMPONENTS" ${ARGN}) + if(NOT "${_ARGS_COMPONENTS}" STREQUAL "") + # parse a single dependency with its components + list(LENGTH _ARGS_UNPARSED_ARGUMENTS _unparsed_length) + if (NOT ${_unparsed_length} EQUAL "1") + message(FATAL_ERROR + "ament_export_dependencies() called with unknown arguments: ${ARGN}") + endif() + # pack the dependency and all its components "dep:compA:compB:compC" + # TODO use AMENT_BUILD_CONFIGURATION_KEYWORD_SEPARATOR-like variable instead of hardcoded ":"? + set (_unpacked "${_ARGS_UNPARSED_ARGUMENTS}") + list(APPEND _unpacked "${_ARGS_COMPONENTS}") + string(REGEX REPLACE ";" ":" _packed "${_unpacked}") + list(APPEND _AMENT_CMAKE_EXPORT_DEPENDENCIES "${_packed}") + + else() + foreach(_arg ${ARGN}) + # only pass package name + # will be resolved by downstream packages + # must be find_package()-able + # and provide _INCLUDE_DIRS and _LIBRARIES + list(APPEND _AMENT_CMAKE_EXPORT_DEPENDENCIES "${_arg}") + endforeach() + + endif() endif() + endmacro()