Files
swift-mirror/stdlib/toolchain/CompatibilityConcurrency/CompatibilityConcurrency.cpp
Doug Gregor 5b027ca456 Back-deploy creation of global-actor-qualified function type metadata.
When back-deploying, create global-actor-qualified function types via a
separate entrypoint
(`swift_getFunctionTypeMetadataGlobalActorBackDeploy`) in the
compatibility library, which checks whether it is running with a
new-enough runtime to use `swift_getFunctionTypeMetadataGlobalActor`.
Failing that, it calls into a separate copy of the implementation that
exists only in the back-deployed concurrency library.

Fixes rdar://79153988.
2021-09-13 10:35:47 -07:00

83 lines
3.4 KiB
C++

//===--- CompatibilityConcurrency.cpp -------------------------------------===//
//
// This source file is part of the Swift.org open source project
//
// Copyright (c) 2014 - 2021 Apple Inc. and the Swift project authors
// Licensed under Apache License v2.0 with Runtime Library Exception
//
// See https://swift.org/LICENSE.txt for license information
// See https://swift.org/CONTRIBUTORS.txt for the list of Swift project authors
//
//===----------------------------------------------------------------------===//
#include "swift/Runtime/Metadata.h"
#include <dispatch/dispatch.h>
#include <dlfcn.h>
// Allow this library to get force-loaded by autolinking
__attribute__((weak, visibility("hidden"))) extern "C" char
_swift_FORCE_LOAD_$_swiftCompatibilityConcurrency = 0;
using namespace swift;
namespace swift {
// Entrypoint provided by the runtime in the OS that first contains support for
// concurrency. Explicitly redeclared as "weak" because this code will run on
// systems where it is not present.
const FunctionTypeMetadata *
swift_getFunctionTypeMetadataGlobalActor(
FunctionTypeFlags flags, FunctionMetadataDifferentiabilityKind diffKind,
const Metadata *const *parameters, const uint32_t *parameterFlags,
const Metadata *result, const Metadata *globalActor
) SWIFT_RUNTIME_WEAK_IMPORT;
// Entrypoint provided only in the back-deployed concurrency library, which
// has a separate allocation area for global-actor-qualified function types.
extern "C"
const FunctionTypeMetadata *
swift_getFunctionTypeMetadataGlobalActorStandalone(
FunctionTypeFlags flags, FunctionMetadataDifferentiabilityKind diffKind,
const Metadata *const *parameters, const uint32_t *parameterFlags,
const Metadata *result, const Metadata *globalActor
) SWIFT_RUNTIME_WEAK_IMPORT;
// Entrypoint called by the compiler when back-deploying concurrency, which
// switches between the real implementation of
// swift_getFunctionTypeMetadataGlobalActor and
// swift_getFunctionTypeMetadataGlobalActorStandalone depending on what system
// it is running on.
SWIFT_RUNTIME_STDLIB_INTERNAL
const FunctionTypeMetadata *
swift_getFunctionTypeMetadataGlobalActorBackDeploy(
FunctionTypeFlags flags, FunctionMetadataDifferentiabilityKind diffKind,
const Metadata *const *parameters, const uint32_t *parameterFlags,
const Metadata *result, const Metadata *globalActor);
} // end namespace swift
const FunctionTypeMetadata *
swift::swift_getFunctionTypeMetadataGlobalActorBackDeploy(
FunctionTypeFlags flags, FunctionMetadataDifferentiabilityKind diffKind,
const Metadata *const *parameters, const uint32_t *parameterFlags,
const Metadata *result, const Metadata *globalActor) {
using BuilderFn = const FunctionTypeMetadata *(*)(
FunctionTypeFlags, FunctionMetadataDifferentiabilityKind,
const Metadata *const *, const uint32_t *,
const Metadata *, const Metadata *);
static BuilderFn builderFn;
static dispatch_once_t builderToken;
dispatch_once(&builderToken, ^{
if (swift_getFunctionTypeMetadataGlobalActor) {
builderFn = swift_getFunctionTypeMetadataGlobalActor;
return;
}
builderFn = reinterpret_cast<BuilderFn>(
dlsym(RTLD_DEFAULT,
"swift_getFunctionTypeMetadataGlobalActorStandalone"));
});
assert(builderFn && "No way to build global-actor-qualified function type");
return builderFn(
flags, diffKind, parameters, parameterFlags, result, globalActor);
}