Files
swift-mirror/lib/AST/Concurrency.cpp
Michael Gottesman bbc4816861 [sema] Move getConcurrencyDiagnosticBehaviorLimit and hasExplicitSendableConformance out of Sema and into libAST.
I am going to expose this on TypeBase, so I am attempting to prevent a layering
violation in between AST and Sema.
2024-08-23 20:42:51 -04:00

83 lines
3.3 KiB
C++

//===--- Concurrency.cpp --------------------------------------------------===//
//
// This source file is part of the Swift.org open source project
//
// Copyright (c) 2014 - 2024 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/AST/Concurrency.h"
#include "swift/AST/ASTContext.h"
#include "swift/AST/ConformanceLookup.h"
#include "swift/AST/Decl.h"
#include "swift/AST/ProtocolConformance.h"
#include "swift/AST/SourceFile.h"
using namespace swift;
std::optional<DiagnosticBehavior>
swift::getConcurrencyDiagnosticBehaviorLimit(NominalTypeDecl *nominal,
const DeclContext *fromDC,
bool ignoreExplicitConformance) {
ModuleDecl *importedModule = nullptr;
if (nominal->getAttrs().hasAttribute<PreconcurrencyAttr>()) {
// If the declaration itself has the @preconcurrency attribute,
// respect it.
importedModule = nominal->getParentModule();
} else {
// Determine whether this nominal type is visible via a @preconcurrency
// import.
auto import = nominal->findImport(fromDC);
auto sourceFile = fromDC->getParentSourceFile();
if (!import || !import->options.contains(ImportFlags::Preconcurrency))
return std::nullopt;
if (sourceFile)
sourceFile->setImportUsedPreconcurrency(*import);
importedModule = import->module.importedModule;
}
// When the type is explicitly non-Sendable, @preconcurrency imports
// downgrade the diagnostic to a warning in Swift 6.
if (!ignoreExplicitConformance && hasExplicitSendableConformance(nominal))
return DiagnosticBehavior::Warning;
// When the type is implicitly non-Sendable, `@preconcurrency` suppresses
// diagnostics until the imported module enables Swift 6.
return importedModule->isConcurrencyChecked() ? DiagnosticBehavior::Warning
: DiagnosticBehavior::Ignore;
}
/// Determine whether the given nominal type has an explicit Sendable
/// conformance (regardless of its availability).
bool swift::hasExplicitSendableConformance(NominalTypeDecl *nominal,
bool applyModuleDefault) {
ASTContext &ctx = nominal->getASTContext();
auto nominalModule = nominal->getParentModule();
// In a concurrency-checked module, a missing conformance is equivalent to
// an explicitly unavailable one. If we want to apply this rule, do so now.
if (applyModuleDefault && nominalModule->isConcurrencyChecked())
return true;
// Look for any conformance to `Sendable`.
auto proto = ctx.getProtocol(KnownProtocolKind::Sendable);
if (!proto)
return false;
// Look for a conformance. If it's present and not (directly) missing,
// we're done.
auto conformance = lookupConformance(nominal->getDeclaredInterfaceType(),
proto, /*allowMissing=*/true);
return conformance &&
!(isa<BuiltinProtocolConformance>(conformance.getConcrete()) &&
cast<BuiltinProtocolConformance>(conformance.getConcrete())
->isMissing());
}