|  | # 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() |