Files
swift-mirror/lib/Sema/DerivedConformance/DerivedConformanceDistributedActor.cpp
Konrad Malawski 9c0c70be5d [Distributed] Allow disabling SerializationRequirement by using Any
It should be possible to disable SerializationRequirement by assigning
Any to it. This effectively disables the checking, since any type
conforms to that.

Previously we would crash trying to check for this conformance; or
rather, emitting diagnostics about un-necessary casts then encountering
incomplete types in CSDiagnostics.

Avoiding the cast entirely sounds like a good solution here.

resolves rdar://159285863
2025-09-05 17:02:07 +09:00

927 lines
36 KiB
C++

//===--- DerivedConformanceActor.cpp ----------------------------*- C++ -*-===//
//
// This source file is part of the Swift.org open source project
//
// Copyright (c) 2014 - 2025 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
//
//===----------------------------------------------------------------------===//
//
// This file implements implicit derivation of the Actor protocol.
//
//===----------------------------------------------------------------------===//
#include "CodeSynthesis.h"
#include "DerivedConformance.h"
#include "TypeCheckDistributed.h"
#include "TypeChecker.h"
#include "swift/AST/AvailabilityInference.h"
#include "swift/AST/ConformanceLookup.h"
#include "swift/AST/DistributedDecl.h"
#include "swift/AST/ExistentialLayout.h"
#include "swift/AST/NameLookupRequests.h"
#include "swift/AST/ParameterList.h"
#include "swift/Basic/Assertions.h"
#include "swift/Strings.h"
using namespace swift;
bool DerivedConformance::canDeriveIdentifiable(
NominalTypeDecl *nominal, DeclContext *dc) {
// we only synthesize for concrete 'distributed actor' decls (which are class)
if (!isa<ClassDecl>(nominal))
return false;
auto &C = nominal->getASTContext();
if (!C.getLoadedModule(C.Id_Distributed))
return false;
return nominal->isDistributedActor();
}
bool DerivedConformance::canDeriveDistributedActor(
NominalTypeDecl *nominal, DeclContext *dc) {
auto &C = nominal->getASTContext();
auto classDecl = dyn_cast<ClassDecl>(nominal);
return C.getLoadedModule(C.Id_Distributed) &&
classDecl && classDecl->isDistributedActor() &&
dc == nominal;
}
bool DerivedConformance::canDeriveDistributedActorSystem(
NominalTypeDecl *nominal, DeclContext *dc) {
auto &C = nominal->getASTContext();
// Make sure ad-hoc requirements that we'll use in synthesis are present, before we try to use them.
// This leads to better error reporting because we already have errors happening (missing witnesses).
if (auto handlerType = getDistributedActorSystemResultHandlerType(nominal)) {
if (!getOnReturnOnDistributedTargetInvocationResultHandler(
handlerType->getAnyNominal()))
return false;
}
return C.getLoadedModule(C.Id_Distributed);
}
/******************************************************************************/
/******************************* RESOLVE FUNCTION *****************************/
/******************************************************************************/
/// Synthesizes the
///
/// \verbatim
/// static resolve(id: ActorID,
/// using system: DistributedActorSystem) throws -> Self {
/// <filled in by SILGenDistributed>
/// }
/// \endverbatim
///
/// factory function in the AST, with an empty body. Its body is
/// expected to be filled-in during SILGen.
static FuncDecl *deriveDistributedActor_resolve(DerivedConformance &derived) {
auto decl = dyn_cast<ClassDecl>(derived.Nominal);
assert(decl->isDistributedActor());
auto &C = decl->getASTContext();
auto idType = getDistributedActorIDType(decl);
auto actorSystemType = getDistributedActorSystemType(decl);
if (!idType || !actorSystemType)
return nullptr;
// (id: Self.ID, using system: Self.ActorSystem)
auto *params = ParameterList::create(
C,
/*LParenLoc=*/SourceLoc(),
/*params=*/{
ParamDecl::createImplicit(
C, C.Id_id, C.Id_id, idType, decl),
ParamDecl::createImplicit(
C, C.Id_using, C.Id_system, actorSystemType, decl)
},
/*RParenLoc=*/SourceLoc()
);
// Func name: resolve(id:using:)
DeclName name(C, C.Id_resolve, params);
// Expected type: (Self) -> (Self.ID, Self.ActorSystem) throws -> (Self)
auto *factoryDecl =
FuncDecl::createImplicit(C, StaticSpellingKind::KeywordStatic,
name, SourceLoc(),
/*async=*/false,
/*throws=*/true,
/*ThrownType=*/Type(),
/*genericParams=*/nullptr,
params,
/*returnType*/decl->getDeclaredInterfaceType(),
decl);
factoryDecl->setDistributedActorFactory(); // TODO(distributed): should we mark this specifically as the resolve factory?
factoryDecl->copyFormalAccessFrom(decl, /*sourceIsParentContext=*/true);
derived.addMembersToConformanceContext({factoryDecl});
return factoryDecl;
}
/******************************************************************************/
/*************** INVOKE HANDLER ON-RETURN FUNCTION ****************************/
/******************************************************************************/
namespace {
struct DoInvokeOnReturnContext {
ParamDecl *handlerParam;
ParamDecl *resultBufferParam;
};
} // namespace
static std::pair<BraceStmt *, bool>
deriveBodyDistributed_doInvokeOnReturn(AbstractFunctionDecl *afd, void *arg) {
auto &C = afd->getASTContext();
auto *context = static_cast<DoInvokeOnReturnContext *>(arg);
// mock locations, we're a thunk and don't really need detailed locations
const SourceLoc sloc = SourceLoc();
const DeclNameLoc dloc = DeclNameLoc();
bool implicit = true;
auto returnTypeParam = afd->getParameters()->get(0);
SmallVector<ASTNode, 8> stmts;
VarDecl *resultVar =
new (C) VarDecl(/*isStatic=*/false, VarDecl::Introducer::Let, sloc,
C.getIdentifier("result"), afd);
{
Expr *resultLoadCall = CallExpr::createImplicit(
C,
UnresolvedDotExpr::createImplicit(
C,
/*base=*/
new (C) DeclRefExpr(ConcreteDeclRef(context->resultBufferParam),
dloc, implicit),
/*baseName=*/DeclBaseName(C.getIdentifier("load")),
/*argLabels=*/
{C.getIdentifier("fromByteOffset"), C.getIdentifier("as")}),
ArgumentList::createImplicit(
C, {Argument(sloc, C.getIdentifier("as"),
new (C) DeclRefExpr(ConcreteDeclRef(returnTypeParam),
dloc, implicit))}));
if (C.LangOpts.hasFeature(Feature::StrictMemorySafety, /*allowMigration=*/true))
resultLoadCall = new (C) UnsafeExpr(sloc, resultLoadCall, Type(), true);
auto resultPattern = NamedPattern::createImplicit(C, resultVar);
auto resultPB = PatternBindingDecl::createImplicit(
C, swift::StaticSpellingKind::None, resultPattern,
/*expr=*/resultLoadCall, afd);
stmts.push_back(resultPB);
stmts.push_back(resultVar);
}
// call the ad-hoc `handler.onReturn`
{
// Find the ad-hoc requirement ensured function on the concrete handler:
auto onReturnFunc = getOnReturnOnDistributedTargetInvocationResultHandler(
context->handlerParam->getInterfaceType()->getAnyNominal());
assert(onReturnFunc && "did not find ad-hoc requirement witness!");
Expr *callExpr = CallExpr::createImplicit(
C,
UnresolvedDotExpr::createImplicit(
C,
/*base=*/
new (C) DeclRefExpr(ConcreteDeclRef(context->handlerParam), dloc,
implicit),
/*baseName=*/onReturnFunc->getBaseName(),
/*paramList=*/onReturnFunc->getParameters()),
ArgumentList::forImplicitCallTo(
DeclNameRef(onReturnFunc->getName()),
{new (C) DeclRefExpr(ConcreteDeclRef(resultVar), dloc, implicit)},
C));
callExpr = TryExpr::createImplicit(C, sloc, callExpr);
callExpr = AwaitExpr::createImplicit(C, sloc, callExpr);
stmts.push_back(callExpr);
}
auto body = BraceStmt::create(C, sloc, {stmts}, sloc, implicit);
return {body, /*isTypeChecked=*/false};
}
// Create local function:
// func invokeOnReturn<R: Self.SerializationRequirement>(
// _ returnType: R.Type
// ) async throws {
// let value = resultBuffer.load(as: returnType)
// try await handler.onReturn(value: value)
// }
static FuncDecl* createLocalFunc_doInvokeOnReturn(
ASTContext& C, FuncDecl* parentFunc,
NominalTypeDecl* systemNominal,
ParamDecl* handlerParam,
ParamDecl* resultBufParam) {
auto DC = parentFunc;
auto doInvokeLocalFuncIdent = C.getIdentifier("doInvokeOnReturn");
// mock locations, we're a synthesized func and don't need real locations
const SourceLoc sloc = SourceLoc();
// <R: Self.SerializationRequirement>
// We create the generic param at invalid depth, which means it'll be filled
// by semantic analysis.
auto *resultGenericParamDecl = GenericTypeParamDecl::createImplicit(
parentFunc, C.getIdentifier("R"), /*depth*/ 0, /*index*/ 0,
GenericTypeParamKind::Type);
GenericParamList *doInvokeGenericParamList =
GenericParamList::create(C, sloc, {resultGenericParamDecl}, sloc);
auto returnTypeIdent = C.getIdentifier("returnType");
auto resultTyParamDecl =
ParamDecl::createImplicit(C,
/*argument=*/returnTypeIdent,
/*parameter=*/returnTypeIdent,
resultGenericParamDecl->getInterfaceType(), DC);
ParameterList *doInvokeParamsList =
ParameterList::create(C, {resultTyParamDecl});
SmallVector<Requirement, 2> requirements;
auto serializationLayout =
getDistributedActorSystemSerializationType(systemNominal)
->getExistentialLayout();
for (auto p : serializationLayout.getProtocols()) {
auto requirement =
Requirement(RequirementKind::Conformance,
resultGenericParamDecl->getDeclaredInterfaceType(),
p->getDeclaredInterfaceType());
requirements.push_back(requirement);
}
GenericSignature doInvokeGenSig =
buildGenericSignature(C, parentFunc->getGenericSignature(),
{resultGenericParamDecl->getDeclaredInterfaceType()
->castTo<GenericTypeParamType>()},
std::move(requirements),
/*allowInverses=*/true);
FuncDecl *doInvokeOnReturnFunc = FuncDecl::createImplicit(
C, swift::StaticSpellingKind::None,
DeclName(C, doInvokeLocalFuncIdent, doInvokeParamsList),
sloc,
/*async=*/true,
/*throws=*/true,
/*ThrownType=*/Type(),
doInvokeGenericParamList, doInvokeParamsList,
/*returnType=*/C.TheEmptyTupleType, parentFunc);
doInvokeOnReturnFunc->setImplicit();
doInvokeOnReturnFunc->setSynthesized();
doInvokeOnReturnFunc->setGenericSignature(doInvokeGenSig);
auto *doInvokeContext = C.Allocate<DoInvokeOnReturnContext>();
doInvokeContext->handlerParam = handlerParam;
doInvokeContext->resultBufferParam = resultBufParam;
doInvokeOnReturnFunc->setBodySynthesizer(
deriveBodyDistributed_doInvokeOnReturn, doInvokeContext);
return doInvokeOnReturnFunc;
}
static std::pair<BraceStmt *, bool>
deriveBodyDistributed_invokeHandlerOnReturn(AbstractFunctionDecl *afd,
void *context) {
auto implicit = true;
ASTContext &C = afd->getASTContext();
auto DC = afd->getDeclContext();
// mock locations, we're a thunk and don't really need detailed locations
const SourceLoc sloc = SourceLoc();
const DeclNameLoc dloc = DeclNameLoc();
NominalTypeDecl *nominal = dyn_cast<NominalTypeDecl>(DC);
assert(nominal);
auto func = dyn_cast<FuncDecl>(afd);
assert(func);
// === parameters
auto params = func->getParameters();
assert(params->size() == 3);
auto handlerParam = params->get(0);
auto resultBufParam = params->get(1);
auto metatypeParam = params->get(2);
auto serializationRequirementTypeTy =
getDistributedActorSystemSerializationType(nominal);
auto serializationRequirementMetaTypeTy =
ExistentialMetatypeType::get(serializationRequirementTypeTy);
// Statements
SmallVector<ASTNode, 8> stmts;
// --- `let m = metatype as! SerializationRequirement.Type`
VarDecl *metatypeVar =
new (C) VarDecl(/*isStatic=*/false, VarDecl::Introducer::Let, sloc,
C.getIdentifier("m"), func);
{
metatypeVar->setImplicit();
metatypeVar->setSynthesized();
// If the SerializationRequirement requires it, we need to emit a cast:
// metatype as! <<concrete SerializationRequirement.Type>>
bool serializationRequirementIsAny =
metatypeParam->getInterfaceType()->getMetatypeInstanceType()->isEqual(
C.getAnyExistentialType());
auto metatypeRef =
new (C) DeclRefExpr(ConcreteDeclRef(metatypeParam), dloc, implicit);
auto metatypePattern = NamedPattern::createImplicit(C, metatypeVar);
PatternBindingDecl *metatypePB = serializationRequirementIsAny ?
PatternBindingDecl::createImplicit(
C, swift::StaticSpellingKind::None, metatypePattern,
/*expr=*/metatypeRef, func) :
PatternBindingDecl::createImplicit(
C, swift::StaticSpellingKind::None, metatypePattern,
/*expr=*/ForcedCheckedCastExpr::createImplicit(
C, metatypeRef, serializationRequirementMetaTypeTy), func);
stmts.push_back(metatypePB);
stmts.push_back(metatypeVar);
}
// --- Declare the local function `doInvokeOnReturn`...
FuncDecl *doInvokeOnReturnFunc = createLocalFunc_doInvokeOnReturn(
C, func,
nominal, handlerParam, resultBufParam);
stmts.push_back(doInvokeOnReturnFunc);
// --- try await _openExistential(metatypeVar, do: <<doInvokeLocalFunc>>)
{
auto openExistentialBaseIdent = C.getIdentifier("_openExistential");
auto doIdent = C.getIdentifier("do");
auto openExArgs = ArgumentList::createImplicit(
C, {
Argument(sloc, Identifier(),
new (C) DeclRefExpr(ConcreteDeclRef(metatypeVar), dloc,
implicit)),
Argument(sloc, doIdent,
new (C) DeclRefExpr(ConcreteDeclRef(doInvokeOnReturnFunc),
dloc, implicit)),
});
Expr *tryAwaitDoOpenExistential =
CallExpr::createImplicit(C,
UnresolvedDeclRefExpr::createImplicit(
C, openExistentialBaseIdent),
openExArgs);
tryAwaitDoOpenExistential =
AwaitExpr::createImplicit(C, sloc, tryAwaitDoOpenExistential);
tryAwaitDoOpenExistential =
TryExpr::createImplicit(C, sloc, tryAwaitDoOpenExistential);
stmts.push_back(tryAwaitDoOpenExistential);
}
auto body = BraceStmt::create(C, sloc, {stmts}, sloc, implicit);
return {body, /*isTypeChecked=*/false};
}
/// Synthesizes the
///
/// \verbatim
/// static func invokeHandlerOnReturn(
//// handler: ResultHandler,
//// resultBuffer: UnsafeRawPointer,
//// metatype _metatype: Any.Type
//// ) async throws
/// \endverbatim
static FuncDecl *deriveDistributedActorSystem_invokeHandlerOnReturn(
DerivedConformance &derived) {
auto system = derived.Nominal;
auto &C = system->getASTContext();
auto resultHandlerType = getDistributedActorSystemResultHandlerType(system);
auto unsafeRawPointerType = C.getUnsafeRawPointerType();
auto anyTypeType = ExistentialMetatypeType::get(C.TheAnyType); // Any.Type
// params:
// - handler: Self.ResultHandler
// - resultBuffer:
// - metatype _metatype: Any.Type
auto *params = ParameterList::create(
C,
/*LParenLoc=*/SourceLoc(),
/*params=*/
{
ParamDecl::createImplicit(
C, C.Id_handler, C.Id_handler,
system->mapTypeIntoContext(resultHandlerType), system),
ParamDecl::createImplicit(
C, C.Id_resultBuffer, C.Id_resultBuffer,
unsafeRawPointerType, system),
ParamDecl::createImplicit(
C, C.Id_metatype, C.Id_metatype,
anyTypeType, system)
},
/*RParenLoc=*/SourceLoc());
// Func name: invokeHandlerOnReturn(handler:resultBuffer:metatype)
DeclName name(C, C.Id_invokeHandlerOnReturn, params);
// Expected type: (Self.ResultHandler, UnsafeRawPointer, any Any.Type) async
// throws -> ()
auto *funcDecl =
FuncDecl::createImplicit(C, StaticSpellingKind::None, name, SourceLoc(),
/*async=*/true,
/*throws=*/true,
/*ThrownType=*/Type(),
/*genericParams=*/nullptr, params,
/*returnType=*/TupleType::getEmpty(C), system);
funcDecl->setSynthesized(true);
funcDecl->copyFormalAccessFrom(system, /*sourceIsParentContext=*/true);
funcDecl->setBodySynthesizer(deriveBodyDistributed_invokeHandlerOnReturn);
derived.addMembersToConformanceContext({funcDecl});
return funcDecl;
}
/******************************************************************************/
/******************************* PROPERTIES ***********************************/
/******************************************************************************/
static ValueDecl *deriveDistributedActor_id(DerivedConformance &derived) {
assert(derived.Nominal->isDistributedActor());
auto &C = derived.Context;
// ```
// nonisolated let id: Self.ID // Self.ActorSystem.ActorID
// ```
auto propertyType = getDistributedActorIDType(derived.Nominal);
VarDecl *propDecl;
PatternBindingDecl *pbDecl;
std::tie(propDecl, pbDecl) = derived.declareDerivedProperty(
DerivedConformance::SynthesizedIntroducer::Let, C.Id_id, propertyType,
/*isStatic=*/false, /*isFinal=*/true);
// mark as nonisolated, allowing access to it from everywhere
propDecl->getAttrs().add(NonisolatedAttr::createImplicit(C));
derived.addMemberToConformanceContext(pbDecl, /*insertAtHead=*/true);
derived.addMemberToConformanceContext(propDecl, /*insertAtHead=*/true);
return propDecl;
}
static ValueDecl *deriveDistributedActor_actorSystem(
DerivedConformance &derived) {
auto &C = derived.Context;
auto classDecl = dyn_cast<ClassDecl>(derived.Nominal);
assert(classDecl && derived.Nominal->isDistributedActor());
if (!C.getLoadedModule(C.Id_Distributed))
return nullptr;
// ```
// nonisolated let actorSystem: ActorSystem
// ```
// (no need for @actorIndependent because it is an immutable let)
auto propertyType = getDistributedActorSystemType(classDecl);
VarDecl *propDecl;
PatternBindingDecl *pbDecl;
std::tie(propDecl, pbDecl) = derived.declareDerivedProperty(
DerivedConformance::SynthesizedIntroducer::Let, C.Id_actorSystem,
propertyType, /*isStatic=*/false, /*isFinal=*/true);
// mark as nonisolated, allowing access to it from everywhere
propDecl->getAttrs().add(NonisolatedAttr::createImplicit(C));
// IMPORTANT: `id` MUST be the first field of a distributed actor, and
// `actorSystem` MUST be the second field, because for a remote instance
// we don't allocate memory after those two fields, so their order is very
// important. The `hint` below makes sure the system is inserted right after.
if (auto id = derived.Nominal->getDistributedActorIDProperty()) {
derived.addMemberToConformanceContext(propDecl, /*hint=*/id);
derived.addMemberToConformanceContext(pbDecl, /*hint=*/id);
} else {
// `id` will be synthesized next, and will insert at head,
// so in order for system to be SECOND (as it must be),
// we'll insert at head right now and as id gets synthesized we'll get
// the correct order: id, actorSystem.
derived.addMemberToConformanceContext(propDecl, /*insertAtHead=*/true);
derived.addMemberToConformanceContext(pbDecl, /*insertAtHead==*/true);
}
return propDecl;
}
/******************************************************************************/
/***************************** ASSOC TYPES ************************************/
/******************************************************************************/
static Type
deriveDistributedActorType_ActorSystem(
DerivedConformance &derived) {
assert(derived.Nominal->isDistributedActor());
auto &C = derived.Context;
// Look for a type DefaultDistributedActorSystem within the parent context.
auto defaultDistributedActorSystemLookup = TypeChecker::lookupUnqualified(
derived.getConformanceContext()->getModuleScopeContext(),
DeclNameRef(C.Id_DefaultDistributedActorSystem),
derived.ConformanceDecl->getLoc());
TypeDecl *defaultDistributedActorSystemTypeDecl = nullptr;
for (const auto &found : defaultDistributedActorSystemLookup) {
if (auto foundType = dyn_cast_or_null<TypeDecl>(found.getValueDecl())) {
if (defaultDistributedActorSystemTypeDecl) {
// Note: ambiguity, for now just fail.
return nullptr;
}
defaultDistributedActorSystemTypeDecl = foundType;
continue;
}
}
// There is no default, so fail to synthesize.
if (!defaultDistributedActorSystemTypeDecl)
return nullptr;
// Return the default system type.
return defaultDistributedActorSystemTypeDecl->getDeclaredInterfaceType();
}
static Type
deriveDistributedActorType_ID(
DerivedConformance &derived) {
if (!derived.Nominal->isDistributedActor())
return nullptr;
// Look for a type DefaultDistributedActorSystem within the parent context.
auto systemTy = getDistributedActorSystemType(derived.Nominal);
// There is no known actor system type, so fail to synthesize.
if (!systemTy || systemTy->hasError())
return nullptr;
if (auto systemNominal = systemTy->getAnyNominal()) {
return getDistributedActorSystemActorIDType(systemNominal);
}
return nullptr;
}
static Type
deriveDistributedActorType_SerializationRequirement(
DerivedConformance &derived) {
if (!derived.Nominal->isDistributedActor())
return nullptr;
// Look for a type DefaultDistributedActorSystem within the parent context.
auto systemTy = getDistributedActorSystemType(derived.Nominal);
// There is no known actor system type, so fail to synthesize.
if (!systemTy || systemTy->hasError())
return nullptr;
auto DAS = derived.Context.getDistributedActorSystemDecl();
if (!DAS)
return nullptr;
if (auto systemNominal = systemTy->getAnyNominal())
return getDistributedActorSystemSerializationType(systemNominal);
return nullptr;
}
/******************************************************************************/
/// Turn a Builtin.Executor value into an UnownedSerialExecutor.
static Expr *constructDistributedUnownedSerialExecutor(ASTContext &ctx,
Expr *arg) {
auto executorDecl = ctx.getUnownedSerialExecutorDecl();
if (!executorDecl) return nullptr;
for (auto member: executorDecl->getAllMembers()) {
auto ctor = dyn_cast<ConstructorDecl>(member);
if (!ctor) continue;
auto params = ctor->getParameters();
if (params->size() != 1 ||
!params->get(0)->getInterfaceType()->is<BuiltinExecutorType>())
continue;
Type executorType = executorDecl->getDeclaredInterfaceType();
Type ctorType = ctor->getInterfaceType();
// We have the right initializer. Build a reference to it of type:
// (UnownedSerialExecutor.Type)
// -> (Builtin.Executor) -> UnownedSerialExecutor
auto initRef = new (ctx) DeclRefExpr(ctor, DeclNameLoc(), /*implicit*/true,
AccessSemantics::Ordinary,
ctorType);
// Apply the initializer to the metatype, building an expression of type:
// (Builtin.Executor) -> UnownedSerialExecutor
auto metatypeRef = TypeExpr::createImplicit(executorType, ctx);
Type ctorAppliedType = ctorType->getAs<FunctionType>()->getResult();
auto selfApply = ConstructorRefCallExpr::create(ctx, initRef, metatypeRef,
ctorAppliedType);
selfApply->setImplicit(true);
selfApply->setThrows(nullptr);
// Call the constructor, building an expression of type
// UnownedSerialExecutor.
auto *argList = ArgumentList::forImplicitUnlabeled(ctx, {arg});
auto call = CallExpr::createImplicit(ctx, selfApply, argList);
call->setType(executorType);
call->setThrows(nullptr);
return call;
}
return nullptr;
}
static std::pair<BraceStmt *, bool>
deriveBodyDistributedActor_unownedExecutor(AbstractFunctionDecl *getter, void *) {
// var unownedExecutor: UnownedSerialExecutor {
// get {
// guard __isLocalActor(self) else {
// return buildDefaultDistributedRemoteActorExecutor(self)
// }
// return Builtin.buildDefaultActorExecutorRef(self)
// }
// }
ASTContext &ctx = getter->getASTContext();
// Produce an empty brace statement on failure.
auto failure = [&]() -> std::pair<BraceStmt *, bool> {
auto body = BraceStmt::create(
ctx, SourceLoc(), { }, SourceLoc(), /*implicit=*/true);
return { body, /*isTypeChecked=*/true };
};
// Build a reference to self.
Type selfType = getter->getImplicitSelfDecl()->getTypeInContext();
Expr *selfArg = DerivedConformance::createSelfDeclRef(getter);
selfArg->setType(selfType);
// Prepare the builtin call, we'll use it after the guard, but want to take the type
// of its return type earlier, so we prepare it here.
// The builtin call gives us a Builtin.Executor.
auto builtinCall =
DerivedConformance::createBuiltinCall(ctx,
BuiltinValueKind::BuildDefaultActorExecutorRef,
{selfType}, {selfArg});
// Turn that into an UnownedSerialExecutor.
auto initCall = constructDistributedUnownedSerialExecutor(ctx, builtinCall);
if (!initCall) return failure();
// guard __isLocalActor(self) else {
// return buildDefaultDistributedRemoteActorExecutor(self)
// }
auto isLocalActorDecl = ctx.getIsLocalDistributedActor();
DeclRefExpr *isLocalActorExpr =
new (ctx) DeclRefExpr(ConcreteDeclRef(isLocalActorDecl), DeclNameLoc(), /*implicit=*/true,
AccessSemantics::Ordinary,
FunctionType::get({AnyFunctionType::Param(ctx.getAnyObjectType())},
ctx.getBoolType()));
Expr *selfForIsLocalArg = DerivedConformance::createSelfDeclRef(getter);
selfForIsLocalArg->setType(selfType);
auto conformances = collectExistentialConformances(selfType->getCanonicalType(),
ctx.getAnyObjectType());
auto *argListForIsLocal =
ArgumentList::forImplicitSingle(ctx, Identifier(),
ErasureExpr::create(ctx, selfForIsLocalArg,
ctx.getAnyObjectType(),
conformances, {}));
CallExpr *isLocalActorCall = CallExpr::createImplicit(ctx, isLocalActorExpr, argListForIsLocal);
isLocalActorCall->setType(ctx.getBoolType());
isLocalActorCall->setThrows(nullptr);
GuardStmt* guardElseRemoteReturnExec;
{
// Find the buildDefaultDistributedRemoteActorExecutor method
auto buildRemoteExecutorDecl =
ctx.getBuildDefaultDistributedRemoteActorUnownedExecutor();
assert(buildRemoteExecutorDecl && "cannot find buildDefaultDistributedRemoteActorExecutor");
auto substitutions = SubstitutionMap::get(
buildRemoteExecutorDecl->getGenericSignature(),
[&](SubstitutableType *dependentType) {
auto gp = cast<GenericTypeParamType>(dependentType);
ASSERT(gp->getDepth() == 0 && gp->getIndex() == 0);
return getter->getImplicitSelfDecl()->getTypeInContext();
},
LookUpConformanceInModule()
);
DeclRefExpr *buildRemoteExecutorExpr =
new (ctx) DeclRefExpr(
ConcreteDeclRef(buildRemoteExecutorDecl, substitutions),
DeclNameLoc(),/*implicit=*/true,
AccessSemantics::Ordinary);
buildRemoteExecutorExpr->setType(
buildRemoteExecutorDecl->getInterfaceType()->castTo<GenericFunctionType>()
->substGenericArgs(substitutions)
);
Expr *selfForBuildRemoteExecutor = DerivedConformance::createSelfDeclRef(getter);
selfForBuildRemoteExecutor->setType(selfType);
auto *argListForBuildRemoteExecutor =
ArgumentList::forImplicitCallTo(buildRemoteExecutorDecl->getParameters(),
/*argExprs=*/{selfForBuildRemoteExecutor}, ctx);
CallExpr *buildRemoteExecutorCall = CallExpr::createImplicit(ctx, buildRemoteExecutorExpr,
argListForBuildRemoteExecutor);
buildRemoteExecutorCall->setType(ctx.getUnownedSerialExecutorType());
buildRemoteExecutorCall->setThrows(nullptr);
SmallVector<ASTNode, 1> statements = {
ReturnStmt::createImplicit(ctx, buildRemoteExecutorCall)
};
SmallVector<StmtConditionElement, 1> conditions = {
isLocalActorCall
};
// Build and return the complete guard statement.
guardElseRemoteReturnExec =
new(ctx) GuardStmt(SourceLoc(), ctx.AllocateCopy(conditions),
BraceStmt::create(ctx, SourceLoc(), statements, SourceLoc()));
}
// Finalize preparing the unowned executor for returning.
// auto wrappedCall = new (ctx) InjectIntoOptionalExpr(initCall, initCall->getType());
auto *returnDefaultExec = ReturnStmt::createImplicit(ctx, initCall);
auto body = BraceStmt::create(
ctx, SourceLoc(), { guardElseRemoteReturnExec, returnDefaultExec }, SourceLoc(), /*implicit=*/true);
return { body, /*isTypeChecked=*/true };
}
/// Derive the declaration of DistributedActor's unownedExecutor property.
static ValueDecl *deriveDistributedActor_unownedExecutor(DerivedConformance &derived) {
ASTContext &ctx = derived.Context;
// Retrieve the types and declarations we'll need to form this operation.
auto executorDecl = ctx.getUnownedSerialExecutorDecl();
if (!executorDecl) {
derived.Nominal->diagnose(
diag::concurrency_lib_missing, "UnownedSerialExecutor");
return nullptr;
}
Type executorType = executorDecl->getDeclaredInterfaceType();
if (auto classDecl = dyn_cast<ClassDecl>(derived.Nominal)) {
if (auto existing = classDecl->getUnownedExecutorProperty()) {
if (existing->getInterfaceType()->isEqual(executorType)) {
return const_cast<VarDecl *>(existing);
} else {
// bad type, should be diagnosed elsewhere
return nullptr;
}
}
}
auto propertyPair = derived.declareDerivedProperty(
DerivedConformance::SynthesizedIntroducer::Var, ctx.Id_unownedExecutor,
executorType, /*static*/ false, /*final*/ false);
auto property = propertyPair.first;
property->setSynthesized(true);
property->getAttrs().add(new (ctx) SemanticsAttr(SEMANTICS_DEFAULT_ACTOR,
SourceLoc(), SourceRange(),
/*implicit*/ true));
property->getAttrs().add(NonisolatedAttr::createImplicit(ctx));
// Make the property implicitly final.
property->getAttrs().add(new (ctx) FinalAttr(/*IsImplicit=*/true));
if (property->getFormalAccess() == AccessLevel::Open)
property->overwriteAccess(AccessLevel::Public);
// Infer availability.
SmallVector<const Decl *, 2> asAvailableAs;
asAvailableAs.push_back(executorDecl);
if (auto enclosingDecl = property->getInnermostDeclWithAvailability())
asAvailableAs.push_back(enclosingDecl);
AvailabilityInference::applyInferredAvailableAttrs(property, asAvailableAs);
auto getter = derived.addGetterToReadOnlyDerivedProperty(property);
getter->setBodySynthesizer(deriveBodyDistributedActor_unownedExecutor);
// IMPORTANT: MUST BE AFTER [id, actorSystem].
if (auto id = derived.Nominal->getDistributedActorIDProperty()) {
if (auto system = derived.Nominal->getDistributedActorSystemProperty()) {
// good, we must be after the system; this is the final order
derived.addMemberToConformanceContext(propertyPair.second, /*hint=*/system);
derived.addMemberToConformanceContext(property, /*hint=*/system);
} else {
// system was not yet synthesized, it'll insert after id and we'll be okey
derived.addMemberToConformanceContext(propertyPair.second, /*hint=*/id);
derived.addMemberToConformanceContext(property, /*hint=*/id);
}
} else {
// nor id or system synthesized yet, id will insert first and system will be after it
derived.addMemberToConformanceContext(propertyPair.second, /*insertAtHead==*/true);
derived.addMemberToConformanceContext(property, /*insertAtHead==*/true);
}
return property;
}
/******************************************************************************/
/**************************** ENTRY POINTS ************************************/
/******************************************************************************/
// !!!!!!!!!!!!! IMPORTANT WHEN MAKING CHANGES TO REQUIREMENTS !!!!!!!!!!!!!!!!!
// !! Remember to update DerivedConformance::getDerivableRequirement !!
// !! any time the signatures or list of derived requirements change. !!
// !!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!
ValueDecl *DerivedConformance::deriveDistributedActor(ValueDecl *requirement) {
if (auto var = dyn_cast<VarDecl>(requirement)) {
ValueDecl *derivedValue = nullptr;
if (var->getName() == Context.Id_id) {
derivedValue = deriveDistributedActor_id(*this);
} else if (var->getName() == Context.Id_actorSystem) {
derivedValue = deriveDistributedActor_actorSystem(*this);
} else if (var->getName() == Context.Id_unownedExecutor) {
derivedValue = deriveDistributedActor_unownedExecutor(*this);
}
if (derivedValue) {
assertRequiredSynthesizedPropertyOrder(Context, Nominal);
}
return derivedValue;
}
if (auto func = dyn_cast<FuncDecl>(requirement)) {
// just a simple name check is enough here,
// if we are invoked here we know for sure it is for the "right" function
if (func->getName().getBaseName() == Context.Id_resolve) {
return deriveDistributedActor_resolve(*this);
}
}
return nullptr;
}
std::pair<Type, TypeDecl *> DerivedConformance::deriveDistributedActor(
AssociatedTypeDecl *assocType) {
if (!canDeriveDistributedActor(Nominal, cast<DeclContext>(ConformanceDecl)))
return std::make_pair(Type(), nullptr);
if (assocType->getName() == Context.Id_ActorSystem) {
return std::make_pair(deriveDistributedActorType_ActorSystem(*this),
nullptr);
}
if (assocType->getName() == Context.Id_SerializationRequirement) {
return std::make_pair(
deriveDistributedActorType_SerializationRequirement(*this), nullptr);
}
if (assocType->getName() == Context.Id_ID) {
return std::make_pair(deriveDistributedActorType_ID(*this), nullptr);
}
Context.Diags.diagnose(assocType->getLoc(),
diag::broken_distributed_actor_requirement);
return std::make_pair(Type(), nullptr);
}
ValueDecl *
DerivedConformance::deriveDistributedActorSystem(ValueDecl *requirement) {
if (auto func = dyn_cast<FuncDecl>(requirement)) {
// just a simple name check is enough here,
// if we are invoked here we know for sure it is for the "right" function
if (func->getName().getBaseName() == Context.Id_invokeHandlerOnReturn) {
return deriveDistributedActorSystem_invokeHandlerOnReturn(*this);
}
}
return nullptr;
}
/******************************************************************************/
/*************************** ERRORS & DIAGNOSTICS *****************************/
/******************************************************************************/
void DerivedConformance::tryDiagnoseFailedDistributedActorDerivation(
DeclContext *DC, NominalTypeDecl *nominal) {
// TODO(distributed): offer better diagnosis for error scenarios here
}
void DerivedConformance::tryDiagnoseFailedDistributedActorSystemDerivation(
DeclContext *DC, NominalTypeDecl *nominal) {
// TODO(distributed): offer better diagnosis for error scenarios here
}