mirror of
https://github.com/apple/swift.git
synced 2026-06-20 15:42:51 +02:00
72c8c21104
Add a new --enable-caching option that enables compilation caching for both C/C++ (via clang-cache as compiler launcher) and Swift code (via -cache-compile-job flags when bootstrapping=hosttools). New options: - --enable-caching: main toggle, incompatible with --sccache/--distcc - --caching-cas-path: CAS directory (default: $BUILD_ROOT/cas) - --caching-depscan-socket: depscan daemon socket path - --caching-plugin-path: CAS plugin library path - --caching-plugin-option: CAS plugin options (repeatable) - --caching-prefix-map: enable source/SDK/toolchain prefix mapping - --caching-remote-service-path: remote caching service with auto plugin inference from Xcode and implied prefix mapping The build script starts a clang-cache depscan daemon with reliable cleanup via atexit and SIGTERM handlers. Per-product build directories get .cas-config and compilation-prefix-map.json files written automatically. Caching flags are applied to all Swift host compilation targets: compiler sources, pure-swift host libraries (ASTGen, macros), swift-syntax, and the new runtime build when --build-runtime-with-host-compiler is used. When not using --caching-remote-service-path, enables CAS backend (-Xfrontend -cas-backend -Xllvm -cas-friendly-debug-info) unless SWIFT_CACHE_DISABLE_MCCAS is set. A ninja wrapper is generated at build/<subdir>/build-utils/ninja for cached incremental builds outside the build-script. rdar://155876033 Assisted-By: Claude
289 lines
9.8 KiB
CMake
289 lines
9.8 KiB
CMake
include(CMakeParseArguments)
|
|
|
|
function(precondition var)
|
|
cmake_parse_arguments(
|
|
PRECONDITION # prefix
|
|
"NEGATE" # options
|
|
"MESSAGE" # single-value args
|
|
"" # multi-value args
|
|
${ARGN})
|
|
|
|
if (PRECONDITION_NEGATE)
|
|
if (${var})
|
|
if (PRECONDITION_MESSAGE)
|
|
message(FATAL_ERROR "Error! ${PRECONDITION_MESSAGE}")
|
|
else()
|
|
message(FATAL_ERROR "Error! Variable ${var} is true or not empty. The value of ${var} is ${${var}}.")
|
|
endif()
|
|
endif()
|
|
else()
|
|
if (NOT ${var})
|
|
if (PRECONDITION_MESSAGE)
|
|
message(FATAL_ERROR "Error! ${PRECONDITION_MESSAGE}")
|
|
else()
|
|
message(FATAL_ERROR "Error! Variable ${var} is false, empty or not set.")
|
|
endif()
|
|
endif()
|
|
endif()
|
|
endfunction()
|
|
|
|
# Assert is 'NOT ${LHS} ${OP} ${RHS}' is true.
|
|
function(precondition_binary_op OP LHS RHS)
|
|
cmake_parse_arguments(
|
|
PRECONDITIONBINOP # prefix
|
|
"NEGATE" # options
|
|
"MESSAGE" # single-value args
|
|
"" # multi-value args
|
|
${ARGN})
|
|
|
|
if (PRECONDITIONBINOP_NEGATE)
|
|
if (${LHS} ${OP} ${RHS})
|
|
if (PRECONDITIONBINOP_MESSAGE)
|
|
message(FATAL_ERROR "Error! ${PRECONDITIONBINOP_MESSAGE}")
|
|
else()
|
|
message(FATAL_ERROR "Error! ${LHS} ${OP} ${RHS} is true!")
|
|
endif()
|
|
endif()
|
|
else()
|
|
if (NOT ${LHS} ${OP} ${RHS})
|
|
if (PRECONDITIONBINOP_MESSAGE)
|
|
message(FATAL_ERROR "Error! ${PRECONDITIONBINOP_MESSAGE}")
|
|
else()
|
|
message(FATAL_ERROR "Error! ${LHS} ${OP} ${RHS} is false!")
|
|
endif()
|
|
endif()
|
|
endif()
|
|
endfunction()
|
|
|
|
# Translate a yes/no variable to the presence of a given string in a
|
|
# variable.
|
|
#
|
|
# Usage:
|
|
# translate_flag(is_set flag_name var_name)
|
|
#
|
|
# If is_set is true, sets ${var_name} to ${flag_name}. Otherwise,
|
|
# unsets ${var_name}.
|
|
function(translate_flag is_set flag_name var_name)
|
|
if(${is_set})
|
|
set("${var_name}" "${flag_name}" PARENT_SCOPE)
|
|
else()
|
|
set("${var_name}" "" PARENT_SCOPE)
|
|
endif()
|
|
endfunction()
|
|
|
|
macro(translate_flags prefix options)
|
|
foreach(var ${options})
|
|
translate_flag("${${prefix}_${var}}" "${var}" "${prefix}_${var}_keyword")
|
|
endforeach()
|
|
endmacro()
|
|
|
|
# Set ${outvar} to ${${invar}}, asserting if ${invar} is not set.
|
|
function(precondition_translate_flag invar outvar)
|
|
precondition(${invar})
|
|
set(${outvar} "${${invar}}" PARENT_SCOPE)
|
|
endfunction()
|
|
|
|
function(get_bootstrapping_path path_var orig_path bootstrapping)
|
|
if("${bootstrapping}" STREQUAL "")
|
|
set(${path_var} ${orig_path} PARENT_SCOPE)
|
|
else()
|
|
file(RELATIVE_PATH relative_path ${CMAKE_BINARY_DIR} ${orig_path})
|
|
set(${path_var} "${CMAKE_BINARY_DIR}/bootstrapping${bootstrapping}/${relative_path}" PARENT_SCOPE)
|
|
endif()
|
|
endfunction()
|
|
|
|
# When building the stdlib in bootstrapping, return the swift library path
|
|
# from the previous bootstrapping stage.
|
|
function(get_bootstrapping_swift_lib_dir bs_lib_dir bootstrapping)
|
|
set(bs_lib_dir "")
|
|
if(BOOTSTRAPPING_MODE STREQUAL "BOOTSTRAPPING")
|
|
set(lib_dir
|
|
"${SWIFTLIB_DIR}/${SWIFT_SDK_${SWIFT_HOST_VARIANT_SDK}_LIB_SUBDIR}")
|
|
# If building the stdlib with bootstrapping, the compiler has to pick up
|
|
# the swift libraries of the previous bootstrapping level (because in the
|
|
# current lib-directory they are not built yet.
|
|
if ("${bootstrapping}" STREQUAL "1")
|
|
get_bootstrapping_path(bs_lib_dir ${lib_dir} "0")
|
|
elseif("${bootstrapping}" STREQUAL "")
|
|
get_bootstrapping_path(bs_lib_dir ${lib_dir} "1")
|
|
endif()
|
|
elseif(BOOTSTRAPPING_MODE STREQUAL "HOSTTOOLS")
|
|
if(SWIFT_HOST_VARIANT_SDK MATCHES "LINUX|ANDROID|OPENBSD|FREEBSD")
|
|
# Compiler's INSTALL_RPATH is set to libs in the build directory
|
|
# For building stdlib, use stdlib in the builder's resource directory
|
|
# because the runtime may not be built yet.
|
|
# FIXME: This assumes the ABI hasn't changed since the builder.
|
|
get_filename_component(swift_bin_dir ${CMAKE_Swift_COMPILER} DIRECTORY)
|
|
get_filename_component(swift_dir ${swift_bin_dir} DIRECTORY)
|
|
|
|
# Detect and handle swiftly-managed hosts.
|
|
if(swift_bin_dir MATCHES ".*/swiftly/bin")
|
|
execute_process(COMMAND swiftly use --print-location
|
|
OUTPUT_VARIABLE swiftly_dir
|
|
ERROR_VARIABLE err)
|
|
if(err)
|
|
message(SEND_ERROR "Failed to find swiftly Swift compiler")
|
|
endif()
|
|
string(STRIP "${swiftly_dir}" swiftly_dir)
|
|
set(swift_dir "${swiftly_dir}/usr")
|
|
endif()
|
|
|
|
set(bs_lib_dir "${swift_dir}/lib/swift/${SWIFT_SDK_${SWIFT_HOST_VARIANT_SDK}_LIB_SUBDIR}")
|
|
endif()
|
|
endif()
|
|
set(bs_lib_dir ${bs_lib_dir} PARENT_SCOPE)
|
|
endfunction()
|
|
|
|
function(is_build_type_optimized build_type result_var_name)
|
|
if("${build_type}" STREQUAL "Debug")
|
|
set("${result_var_name}" FALSE PARENT_SCOPE)
|
|
elseif("${build_type}" STREQUAL "RelWithDebInfo" OR
|
|
"${build_type}" STREQUAL "Release" OR
|
|
"${build_type}" STREQUAL "MinSizeRel")
|
|
set("${result_var_name}" TRUE PARENT_SCOPE)
|
|
else()
|
|
message(FATAL_ERROR "Unknown build type: ${build_type}")
|
|
endif()
|
|
endfunction()
|
|
|
|
function(is_build_type_with_debuginfo build_type result_var_name)
|
|
if("${build_type}" STREQUAL "Debug" OR
|
|
"${build_type}" STREQUAL "RelWithDebInfo")
|
|
set("${result_var_name}" TRUE PARENT_SCOPE)
|
|
elseif("${build_type}" STREQUAL "Release" OR
|
|
"${build_type}" STREQUAL "MinSizeRel")
|
|
set("${result_var_name}" FALSE PARENT_SCOPE)
|
|
else()
|
|
message(FATAL_ERROR "Unknown build type: ${build_type}")
|
|
endif()
|
|
endfunction()
|
|
|
|
# Set variable to value if value is not null or false. Otherwise set variable to
|
|
# default_value.
|
|
function(set_with_default variable value)
|
|
cmake_parse_argument(
|
|
SWD
|
|
""
|
|
"DEFAULT"
|
|
"" ${ARGN})
|
|
precondition(SWD_DEFAULT
|
|
MESSAGE "Must specify a default argument")
|
|
if (value)
|
|
set(${variable} ${value} PARENT_SCOPE)
|
|
else()
|
|
set(${variable} ${SWD_DEFAULT} PARENT_SCOPE)
|
|
endif()
|
|
endfunction()
|
|
|
|
function(swift_create_post_build_symlink target)
|
|
set(options IS_DIRECTORY)
|
|
set(oneValueArgs SOURCE DESTINATION WORKING_DIRECTORY COMMENT)
|
|
cmake_parse_arguments(CS
|
|
"${options}"
|
|
"${oneValueArgs}"
|
|
""
|
|
${ARGN})
|
|
|
|
if(CS_IS_DIRECTORY)
|
|
set(cmake_symlink_option "${SWIFT_COPY_OR_SYMLINK_DIR}")
|
|
else()
|
|
set(cmake_symlink_option "${SWIFT_COPY_OR_SYMLINK}")
|
|
endif()
|
|
|
|
set(comment_arg)
|
|
if(CS_COMMENT)
|
|
set(comment_arg COMMENT "${CS_COMMENT}")
|
|
endif()
|
|
|
|
add_custom_command(TARGET "${target}" POST_BUILD
|
|
COMMAND
|
|
"${CMAKE_COMMAND}" "-E" "${cmake_symlink_option}"
|
|
"${CS_SOURCE}"
|
|
"${CS_DESTINATION}"
|
|
WORKING_DIRECTORY "${CS_WORKING_DIRECTORY}"
|
|
${comment_arg})
|
|
endfunction()
|
|
|
|
# Once swift-frontend is built, if the standalone (early) swift-driver has been built,
|
|
# we create a `swift-driver` symlink adjacent to the `swift` and `swiftc` executables
|
|
# to ensure that `swiftc` forwards to the standalone driver when invoked.
|
|
function(swift_create_early_driver_copies target)
|
|
set(SWIFT_EARLY_SWIFT_DRIVER_BUILD "" CACHE PATH "Path to early swift-driver build")
|
|
|
|
if(NOT SWIFT_EARLY_SWIFT_DRIVER_BUILD)
|
|
return()
|
|
endif()
|
|
|
|
if(EXISTS ${SWIFT_EARLY_SWIFT_DRIVER_BUILD}/swift-driver${CMAKE_EXECUTABLE_SUFFIX})
|
|
message(STATUS "Creating early SwiftDriver symlinks")
|
|
|
|
# Use `configure_file` instead of `file(COPY ...)` to establish a
|
|
# dependency. Further changes to `swift-driver` will cause it to be copied
|
|
# over.
|
|
configure_file(${SWIFT_EARLY_SWIFT_DRIVER_BUILD}/swift-driver${CMAKE_EXECUTABLE_SUFFIX}
|
|
${SWIFT_RUNTIME_OUTPUT_INTDIR}/swift-driver${CMAKE_EXECUTABLE_SUFFIX}
|
|
COPYONLY)
|
|
configure_file(${SWIFT_EARLY_SWIFT_DRIVER_BUILD}/swift-help${CMAKE_EXECUTABLE_SUFFIX}
|
|
${SWIFT_RUNTIME_OUTPUT_INTDIR}/swift-help${CMAKE_EXECUTABLE_SUFFIX}
|
|
COPYONLY)
|
|
else()
|
|
message(STATUS "Not creating early SwiftDriver symlinks (swift-driver not found)")
|
|
endif()
|
|
endfunction()
|
|
|
|
function(dump_swift_vars)
|
|
set(SWIFT_STDLIB_GLOBAL_CMAKE_CACHE)
|
|
get_cmake_property(variableNames VARIABLES)
|
|
foreach(variableName ${variableNames})
|
|
if(variableName MATCHES "^SWIFT")
|
|
set(SWIFT_STDLIB_GLOBAL_CMAKE_CACHE "${SWIFT_STDLIB_GLOBAL_CMAKE_CACHE}set(${variableName} \"${${variableName}}\")\n")
|
|
message("set(${variableName} \"${${variableName}}\")")
|
|
endif()
|
|
endforeach()
|
|
endfunction()
|
|
|
|
function(is_sdk_requested name result_var_name)
|
|
if("${SWIFT_HOST_VARIANT_SDK}" STREQUAL "${name}")
|
|
set("${result_var_name}" "TRUE" PARENT_SCOPE)
|
|
else()
|
|
if("${name}" IN_LIST SWIFT_SDKS)
|
|
set("${result_var_name}" "TRUE" PARENT_SCOPE)
|
|
else()
|
|
set("${result_var_name}" "FALSE" PARENT_SCOPE)
|
|
endif()
|
|
endif()
|
|
endfunction()
|
|
|
|
# Append Swift compilation-caching flags (driven by the SWIFT_CACHING_BUILD_*
|
|
# cache variables) to the named list variable. Callers are responsible for
|
|
# deciding whether caching applies to their target (e.g. checking
|
|
# SWIFT_CACHING_BUILD and the relevant host/runtime guards).
|
|
function(swift_append_caching_compile_flags result_var)
|
|
list(APPEND ${result_var}
|
|
"-explicit-module-build"
|
|
"-cache-compile-job"
|
|
"-cas-path" "${SWIFT_CACHING_BUILD_CAS_PATH}")
|
|
if(SWIFT_CACHING_BUILD_PLUGIN_PATH)
|
|
list(APPEND ${result_var}
|
|
"-cas-plugin-path" "${SWIFT_CACHING_BUILD_PLUGIN_PATH}")
|
|
endif()
|
|
if(SWIFT_CACHING_BUILD_PLUGIN_OPTIONS)
|
|
string(REPLACE ":" ";" _plugin_opts "${SWIFT_CACHING_BUILD_PLUGIN_OPTIONS}")
|
|
foreach(_opt IN LISTS _plugin_opts)
|
|
list(APPEND ${result_var} "-cas-plugin-option" "${_opt}")
|
|
endforeach()
|
|
endif()
|
|
if(SWIFT_CACHING_BUILD_PREFIX_MAP)
|
|
list(APPEND ${result_var}
|
|
"-scanner-prefix-map-sdk" "/^sdk"
|
|
"-scanner-prefix-map-toolchain" "/^toolchain"
|
|
"-scanner-prefix-map" "${SWIFT_CACHING_BUILD_SOURCE_ROOT}=/^src")
|
|
endif()
|
|
if(SWIFT_CACHING_BUILD_ENABLE_MCCAS)
|
|
list(APPEND ${result_var}
|
|
"-Xfrontend" "-cas-backend"
|
|
"-Xllvm" "-cas-friendly-debug-info")
|
|
endif()
|
|
set(${result_var} "${${result_var}}" PARENT_SCOPE)
|
|
endfunction()
|