blob: 682a73924cb1817d915ede25721a490c8e2f3516 [file] [log] [blame]
# Copyright 2024 The Dawn & Tint Authors
#
# Redistribution and use in source and binary forms, with or without
# modification, are permitted provided that the following conditions are met:
#
# 1. Redistributions of source code must retain the above copyright notice, this
# list of conditions and the following disclaimer.
#
# 2. Redistributions in binary form must reproduce the above copyright notice,
# this list of conditions and the following disclaimer in the documentation
# and/or other materials provided with the distribution.
#
# 3. Neither the name of the copyright holder nor the names of its
# contributors may be used to endorse or promote products derived from
# this software without specific prior written permission.
#
# THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS"
# AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
# IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE
# DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT HOLDER OR CONTRIBUTORS BE LIABLE
# FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
# DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR
# SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER
# CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY,
# OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
# OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
function(bundle_libraries output_target library_type)
# This function recursively finds all dependencies of a given target.
# It populates a list variable named 'all_dependencies' in the parent scope.
# Note: This recursive approach with PARENT_SCOPE works but can be a bit
# tricky with variable lifetimes. It assumes 'all_dependencies' is
# initialized in the calling scope.
function(get_dependencies input_target)
# Resolve aliases first
get_target_property(alias ${input_target} ALIASED_TARGET)
if(TARGET ${alias})
set(input_target ${alias})
endif()
# Avoid processing the same target multiple times (circular dependency check)
# Checks if the current target is already in the list being built in the parent scope.
# Note: all_dependencies is modified in parent scope below, so check works.
if(${input_target} IN_LIST all_dependencies)
return()
endif()
# Add the current target to the list of dependencies that we have seen. It will be copied
# to the parent scope before returning at the end of this function.
list(APPEND all_dependencies ${input_target})
# Get dependencies from LINK_LIBRARIES (Private and Public linkage)
get_target_property(link_libraries ${input_target} LINK_LIBRARIES)
foreach(dependency IN LISTS link_libraries)
# Only recurse on actual targets
if(TARGET ${dependency})
get_dependencies(${dependency}) # Recursive call
endif()
endforeach()
# Get dependencies from INTERFACE_LINK_LIBRARIES (Interface and Public linkage)
get_target_property(interface_link_libraries ${input_target} INTERFACE_LINK_LIBRARIES)
foreach(dependency IN LISTS interface_link_libraries)
# Only recurse on actual targets
if(TARGET ${dependency})
get_dependencies(${dependency}) # Recursive call
endif()
endforeach()
# The final 'all_dependencies' list is available in the parent scope
# after the initial call to get_dependencies completes.
set(all_dependencies ${all_dependencies} PARENT_SCOPE) # Propagate change up
endfunction()
# ARGN contains all arguments after the named parameters so it is the list of
# all the libraries to bunble.
if(NOT ARGN)
message(FATAL_ERROR "bundle_libraries: No input targets specified for '${output_target}'.")
return()
endif()
# Initialize the list that the recursive function will populate.
# This list will exist in the scope of the bundle_libraries function.
set(all_dependencies "")
foreach(input_target IN LISTS ARGN)
if(TARGET ${input_target})
get_dependencies(${input_target})
else()
message(WARNING "bundle_libraries: Input target '${input_target}' is not a valid target. Skipping.")
endif()
endforeach()
# Collect $<TARGET_OBJECTS:...> from STATIC and OBJECT library dependencies
set(all_objects "")
foreach(dependency IN LISTS all_dependencies)
get_target_property(type ${dependency} TYPE)
# We only want object files from static or object libraries.
# This correctly excludes shared libraries, modules, executables, interfaces, etc.
if(${type} STREQUAL "STATIC_LIBRARY")
list(APPEND all_objects $<TARGET_OBJECTS:${dependency}>)
elseif(${type} STREQUAL "OBJECT_LIBRARY")
list(APPEND all_objects $<TARGET_OBJECTS:${dependency}>)
endif()
endforeach()
# Check if any object files were found
if(NOT all_objects)
message(WARNING "bundle_libraries: No object files found from the dependencies of ${ARGN}. Creating empty library '${output_target}'.")
endif()
# Create the output library using the validated type and collected objects
# If all_objects is empty, add_library will still create an empty library of the specified type.
add_library(${output_target} ${library_type} ${all_objects})
# Add dependencies to ensure input targets are built before the bundled library.
# This handles the build order correctly.
add_dependencies(${output_target} ${ARGN})
endfunction()