mirror of
https://github.com/apple/swift.git
synced 2025-12-14 20:36:38 +01:00
2011 lines
67 KiB
C++
2011 lines
67 KiB
C++
//===--- Module.cpp - Swift Language Module Implementation ----------------===//
|
|
//
|
|
// This source file is part of the Swift.org open source project
|
|
//
|
|
// Copyright (c) 2014 - 2017 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 the Module class and subclasses.
|
|
//
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
#include "swift/AST/Module.h"
|
|
#include "swift/AST/AccessScope.h"
|
|
#include "swift/AST/ASTContext.h"
|
|
#include "swift/AST/ASTMangler.h"
|
|
#include "swift/AST/ASTPrinter.h"
|
|
#include "swift/AST/ASTWalker.h"
|
|
#include "swift/AST/Builtins.h"
|
|
#include "swift/AST/DiagnosticsSema.h"
|
|
#include "swift/AST/ExistentialLayout.h"
|
|
#include "swift/AST/GenericEnvironment.h"
|
|
#include "swift/AST/LazyResolver.h"
|
|
#include "swift/AST/LinkLibrary.h"
|
|
#include "swift/AST/ModuleLoader.h"
|
|
#include "swift/AST/NameLookup.h"
|
|
#include "swift/AST/ReferencedNameTracker.h"
|
|
#include "swift/AST/PrettyStackTrace.h"
|
|
#include "swift/AST/PrintOptions.h"
|
|
#include "swift/AST/ProtocolConformance.h"
|
|
#include "swift/AST/TypeCheckRequests.h"
|
|
#include "swift/Basic/Compiler.h"
|
|
#include "swift/Basic/SourceManager.h"
|
|
#include "swift/Basic/Statistic.h"
|
|
#include "swift/Demangling/ManglingMacros.h"
|
|
#include "swift/Parse/Token.h"
|
|
#include "swift/Strings.h"
|
|
#include "swift/Syntax/SyntaxNodes.h"
|
|
#include "clang/Basic/Module.h"
|
|
#include "llvm/ADT/DenseMap.h"
|
|
#include "llvm/ADT/DenseSet.h"
|
|
#include "llvm/ADT/TinyPtrVector.h"
|
|
#include "llvm/ADT/SmallPtrSet.h"
|
|
#include "llvm/ADT/StringExtras.h"
|
|
#include "llvm/ADT/STLExtras.h"
|
|
#include "llvm/Support/MD5.h"
|
|
#include "llvm/Support/MemoryBuffer.h"
|
|
#include "llvm/Support/Path.h"
|
|
#include "llvm/Support/raw_ostream.h"
|
|
#include "llvm/Support/SaveAndRestore.h"
|
|
|
|
using namespace swift;
|
|
|
|
//===----------------------------------------------------------------------===//
|
|
// Builtin Module Name lookup
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
class BuiltinUnit::LookupCache {
|
|
/// The cache of identifiers we've already looked up. We use a
|
|
/// single hashtable for both types and values as a minor
|
|
/// optimization; this prevents us from having both a builtin type
|
|
/// and a builtin value with the same name, but that's okay.
|
|
llvm::DenseMap<Identifier, ValueDecl*> Cache;
|
|
|
|
public:
|
|
void lookupValue(Identifier Name, NLKind LookupKind, const BuiltinUnit &M,
|
|
SmallVectorImpl<ValueDecl*> &Result);
|
|
};
|
|
|
|
BuiltinUnit::LookupCache &BuiltinUnit::getCache() const {
|
|
// FIXME: This leaks. Sticking this into ASTContext isn't enough because then
|
|
// the DenseMap will leak.
|
|
if (!Cache)
|
|
const_cast<BuiltinUnit *>(this)->Cache = llvm::make_unique<LookupCache>();
|
|
return *Cache;
|
|
}
|
|
|
|
void BuiltinUnit::LookupCache::lookupValue(
|
|
Identifier Name, NLKind LookupKind, const BuiltinUnit &M,
|
|
SmallVectorImpl<ValueDecl*> &Result) {
|
|
// Only qualified lookup ever finds anything in the builtin module.
|
|
if (LookupKind != NLKind::QualifiedLookup) return;
|
|
|
|
ValueDecl *&Entry = Cache[Name];
|
|
ASTContext &Ctx = M.getParentModule()->getASTContext();
|
|
if (!Entry) {
|
|
if (Type Ty = getBuiltinType(Ctx, Name.str())) {
|
|
auto *TAD = new (Ctx) TypeAliasDecl(SourceLoc(), SourceLoc(),
|
|
Name, SourceLoc(),
|
|
/*genericparams*/nullptr,
|
|
const_cast<BuiltinUnit*>(&M));
|
|
TAD->setUnderlyingType(Ty);
|
|
TAD->setAccess(AccessLevel::Public);
|
|
Entry = TAD;
|
|
}
|
|
}
|
|
|
|
if (!Entry)
|
|
Entry = getBuiltinValueDecl(Ctx, Name);
|
|
|
|
if (Entry)
|
|
Result.push_back(Entry);
|
|
}
|
|
|
|
// Out-of-line because std::unique_ptr wants LookupCache to be complete.
|
|
BuiltinUnit::BuiltinUnit(ModuleDecl &M)
|
|
: FileUnit(FileUnitKind::Builtin, M) {
|
|
M.getASTContext().addDestructorCleanup(*this);
|
|
}
|
|
|
|
//===----------------------------------------------------------------------===//
|
|
// Normal Module Name Lookup
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
SourceFile::~SourceFile() = default;
|
|
|
|
/// A utility for caching global lookups into SourceFiles and modules of
|
|
/// SourceFiles. This is used for lookup of top-level declarations, as well
|
|
/// as operator lookup (which looks into types) and AnyObject dynamic lookup
|
|
/// (which looks at all class members).
|
|
class swift::SourceLookupCache {
|
|
/// A lookup map for value decls. When declarations are added they are added
|
|
/// under all variants of the name they can be found under.
|
|
class DeclMap {
|
|
llvm::DenseMap<DeclName, TinyPtrVector<ValueDecl*>> Members;
|
|
|
|
public:
|
|
void add(ValueDecl *VD) {
|
|
if (!VD->hasName()) return;
|
|
VD->getFullName().addToLookupTable(Members, VD);
|
|
}
|
|
|
|
void clear() {
|
|
Members.shrink_and_clear();
|
|
}
|
|
|
|
decltype(Members)::const_iterator begin() const { return Members.begin(); }
|
|
decltype(Members)::const_iterator end() const { return Members.end(); }
|
|
decltype(Members)::const_iterator find(DeclName Name) const {
|
|
return Members.find(Name);
|
|
}
|
|
};
|
|
|
|
DeclMap TopLevelValues;
|
|
DeclMap ClassMembers;
|
|
bool MemberCachePopulated = false;
|
|
|
|
template<typename Range>
|
|
void addToUnqualifiedLookupCache(Range decls, bool onlyOperators);
|
|
template<typename Range>
|
|
void addToMemberCache(Range decls);
|
|
public:
|
|
typedef ModuleDecl::AccessPathTy AccessPathTy;
|
|
|
|
SourceLookupCache(const SourceFile &SF);
|
|
SourceLookupCache(const ModuleDecl &Mod);
|
|
|
|
/// Throw away as much memory as possible.
|
|
void invalidate();
|
|
|
|
void lookupValue(AccessPathTy AccessPath, DeclName Name,
|
|
NLKind LookupKind, SmallVectorImpl<ValueDecl*> &Result);
|
|
|
|
void lookupVisibleDecls(AccessPathTy AccessPath,
|
|
VisibleDeclConsumer &Consumer,
|
|
NLKind LookupKind);
|
|
|
|
void populateMemberCache(const SourceFile &SF);
|
|
void populateMemberCache(const ModuleDecl &Mod);
|
|
|
|
void lookupClassMembers(AccessPathTy AccessPath,
|
|
VisibleDeclConsumer &consumer);
|
|
|
|
void lookupClassMember(AccessPathTy accessPath,
|
|
DeclName name,
|
|
SmallVectorImpl<ValueDecl*> &results);
|
|
|
|
SmallVector<ValueDecl *, 0> AllVisibleValues;
|
|
};
|
|
SourceLookupCache &SourceFile::getCache() const {
|
|
if (!Cache) {
|
|
const_cast<SourceFile *>(this)->Cache =
|
|
llvm::make_unique<SourceLookupCache>(*this);
|
|
}
|
|
return *Cache;
|
|
}
|
|
|
|
template<typename Range>
|
|
void SourceLookupCache::addToUnqualifiedLookupCache(Range decls,
|
|
bool onlyOperators) {
|
|
for (Decl *D : decls) {
|
|
if (auto *VD = dyn_cast<ValueDecl>(D)) {
|
|
if (onlyOperators ? VD->isOperator() : VD->hasName()) {
|
|
// Cache the value under both its compound name and its full name.
|
|
TopLevelValues.add(VD);
|
|
}
|
|
}
|
|
|
|
if (auto *NTD = dyn_cast<NominalTypeDecl>(D))
|
|
if (!NTD->hasUnparsedMembers() || NTD->maybeHasOperatorDeclarations())
|
|
addToUnqualifiedLookupCache(NTD->getMembers(), true);
|
|
|
|
// Avoid populating the cache with the members of invalid extension
|
|
// declarations. These members can be used to point validation inside of
|
|
// a malformed context.
|
|
if (D->isInvalid()) continue;
|
|
|
|
if (auto *ED = dyn_cast<ExtensionDecl>(D))
|
|
if (!ED->hasUnparsedMembers() || ED->maybeHasOperatorDeclarations())
|
|
addToUnqualifiedLookupCache(ED->getMembers(), true);
|
|
}
|
|
}
|
|
|
|
void SourceLookupCache::populateMemberCache(const SourceFile &SF) {
|
|
if (MemberCachePopulated)
|
|
return;
|
|
|
|
FrontendStatsTracer tracer(SF.getASTContext().Stats,
|
|
"populate-source-file-class-member-cache");
|
|
addToMemberCache(SF.Decls);
|
|
MemberCachePopulated = true;
|
|
}
|
|
|
|
void SourceLookupCache::populateMemberCache(const ModuleDecl &Mod) {
|
|
if (MemberCachePopulated)
|
|
return;
|
|
|
|
FrontendStatsTracer tracer(Mod.getASTContext().Stats,
|
|
"populate-module-class-member-cache");
|
|
|
|
for (const FileUnit *file : Mod.getFiles()) {
|
|
auto &SF = *cast<SourceFile>(file);
|
|
addToMemberCache(SF.Decls);
|
|
}
|
|
|
|
MemberCachePopulated = true;
|
|
}
|
|
|
|
template <typename Range>
|
|
void SourceLookupCache::addToMemberCache(Range decls) {
|
|
for (Decl *D : decls) {
|
|
if (auto *NTD = dyn_cast<NominalTypeDecl>(D)) {
|
|
if (!NTD->hasUnparsedMembers() ||
|
|
NTD->maybeHasNestedClassDeclarations() ||
|
|
NTD->mayContainMembersAccessedByDynamicLookup())
|
|
addToMemberCache(NTD->getMembers());
|
|
|
|
} else if (auto *ED = dyn_cast<ExtensionDecl>(D)) {
|
|
if (!ED->hasUnparsedMembers() ||
|
|
ED->maybeHasNestedClassDeclarations() ||
|
|
ED->mayContainMembersAccessedByDynamicLookup())
|
|
addToMemberCache(ED->getMembers());
|
|
|
|
} else if (auto *VD = dyn_cast<ValueDecl>(D)) {
|
|
if (VD->canBeAccessedByDynamicLookup())
|
|
ClassMembers.add(VD);
|
|
}
|
|
}
|
|
}
|
|
|
|
/// Populate our cache on the first name lookup.
|
|
SourceLookupCache::SourceLookupCache(const SourceFile &SF) {
|
|
FrontendStatsTracer tracer(SF.getASTContext().Stats,
|
|
"source-file-populate-cache");
|
|
addToUnqualifiedLookupCache(SF.Decls, false);
|
|
}
|
|
|
|
SourceLookupCache::SourceLookupCache(const ModuleDecl &M) {
|
|
FrontendStatsTracer tracer(M.getASTContext().Stats,
|
|
"module-populate-cache");
|
|
for (const FileUnit *file : M.getFiles()) {
|
|
auto &SF = *cast<SourceFile>(file);
|
|
addToUnqualifiedLookupCache(SF.Decls, false);
|
|
}
|
|
}
|
|
|
|
void SourceLookupCache::lookupValue(AccessPathTy AccessPath, DeclName Name,
|
|
NLKind LookupKind,
|
|
SmallVectorImpl<ValueDecl*> &Result) {
|
|
// If this import is specific to some named type or decl ("import Swift.int")
|
|
// then filter out any lookups that don't match.
|
|
if (!ModuleDecl::matchesAccessPath(AccessPath, Name))
|
|
return;
|
|
|
|
auto I = TopLevelValues.find(Name);
|
|
if (I == TopLevelValues.end()) return;
|
|
|
|
Result.reserve(I->second.size());
|
|
for (ValueDecl *Elt : I->second)
|
|
Result.push_back(Elt);
|
|
}
|
|
|
|
void SourceLookupCache::lookupVisibleDecls(AccessPathTy AccessPath,
|
|
VisibleDeclConsumer &Consumer,
|
|
NLKind LookupKind) {
|
|
assert(AccessPath.size() <= 1 && "can only refer to top-level decls");
|
|
|
|
if (!AccessPath.empty()) {
|
|
auto I = TopLevelValues.find(AccessPath.front().first);
|
|
if (I == TopLevelValues.end()) return;
|
|
|
|
for (auto vd : I->second)
|
|
Consumer.foundDecl(vd, DeclVisibilityKind::VisibleAtTopLevel);
|
|
return;
|
|
}
|
|
|
|
for (auto &tlv : TopLevelValues) {
|
|
for (ValueDecl *vd : tlv.second) {
|
|
// Declarations are added under their full and simple names. Skip the
|
|
// entry for the simple name so that we report each declaration once.
|
|
if (tlv.first.isSimpleName() && !vd->getFullName().isSimpleName())
|
|
continue;
|
|
Consumer.foundDecl(vd, DeclVisibilityKind::VisibleAtTopLevel);
|
|
}
|
|
}
|
|
}
|
|
|
|
void SourceLookupCache::lookupClassMembers(AccessPathTy accessPath,
|
|
VisibleDeclConsumer &consumer) {
|
|
assert(accessPath.size() <= 1 && "can only refer to top-level decls");
|
|
|
|
if (!accessPath.empty()) {
|
|
for (auto &member : ClassMembers) {
|
|
// Non-simple names are also stored under their simple name, so make
|
|
// sure to only report them once.
|
|
if (!member.first.isSimpleName())
|
|
continue;
|
|
|
|
for (ValueDecl *vd : member.second) {
|
|
auto *nominal = vd->getDeclContext()->getSelfNominalTypeDecl();
|
|
if (nominal && nominal->getName() == accessPath.front().first)
|
|
consumer.foundDecl(vd, DeclVisibilityKind::DynamicLookup,
|
|
DynamicLookupInfo::AnyObject);
|
|
}
|
|
}
|
|
return;
|
|
}
|
|
|
|
for (auto &member : ClassMembers) {
|
|
// Non-simple names are also stored under their simple name, so make sure to
|
|
// only report them once.
|
|
if (!member.first.isSimpleName())
|
|
continue;
|
|
|
|
for (ValueDecl *vd : member.second)
|
|
consumer.foundDecl(vd, DeclVisibilityKind::DynamicLookup,
|
|
DynamicLookupInfo::AnyObject);
|
|
}
|
|
}
|
|
|
|
void SourceLookupCache::lookupClassMember(AccessPathTy accessPath,
|
|
DeclName name,
|
|
SmallVectorImpl<ValueDecl*> &results) {
|
|
assert(accessPath.size() <= 1 && "can only refer to top-level decls");
|
|
|
|
auto iter = ClassMembers.find(name);
|
|
if (iter == ClassMembers.end())
|
|
return;
|
|
|
|
if (!accessPath.empty()) {
|
|
for (ValueDecl *vd : iter->second) {
|
|
auto *nominal = vd->getDeclContext()->getSelfNominalTypeDecl();
|
|
if (nominal && nominal->getName() == accessPath.front().first)
|
|
results.push_back(vd);
|
|
}
|
|
return;
|
|
}
|
|
|
|
results.append(iter->second.begin(), iter->second.end());
|
|
}
|
|
|
|
void SourceLookupCache::invalidate() {
|
|
TopLevelValues.clear();
|
|
ClassMembers.clear();
|
|
MemberCachePopulated = false;
|
|
|
|
// std::move AllVisibleValues into a temporary to destroy its contents.
|
|
using SameSizeSmallVector = decltype(AllVisibleValues);
|
|
(void)SameSizeSmallVector{std::move(AllVisibleValues)};
|
|
}
|
|
|
|
//===----------------------------------------------------------------------===//
|
|
// Module Implementation
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
ModuleDecl::ModuleDecl(Identifier name, ASTContext &ctx)
|
|
: DeclContext(DeclContextKind::Module, nullptr),
|
|
TypeDecl(DeclKind::Module, &ctx, name, SourceLoc(), { }) {
|
|
|
|
ctx.addDestructorCleanup(*this);
|
|
setImplicit();
|
|
setInterfaceType(ModuleType::get(this));
|
|
|
|
// validateDecl() should return immediately given a ModuleDecl.
|
|
setValidationToChecked();
|
|
|
|
setAccess(AccessLevel::Public);
|
|
}
|
|
|
|
bool ModuleDecl::isClangModule() const {
|
|
return findUnderlyingClangModule() != nullptr;
|
|
}
|
|
|
|
void ModuleDecl::addFile(FileUnit &newFile) {
|
|
// Require Main and REPL files to be the first file added.
|
|
assert(Files.empty() ||
|
|
!isa<SourceFile>(newFile) ||
|
|
cast<SourceFile>(newFile).Kind == SourceFileKind::Library ||
|
|
cast<SourceFile>(newFile).Kind == SourceFileKind::SIL);
|
|
Files.push_back(&newFile);
|
|
clearLookupCache();
|
|
}
|
|
|
|
void ModuleDecl::removeFile(FileUnit &existingFile) {
|
|
// Do a reverse search; usually the file to be deleted will be at the end.
|
|
std::reverse_iterator<decltype(Files)::iterator> I(Files.end()),
|
|
E(Files.begin());
|
|
I = std::find(I, E, &existingFile);
|
|
assert(I != E);
|
|
|
|
// Adjust for the std::reverse_iterator offset.
|
|
++I;
|
|
Files.erase(I.base());
|
|
clearLookupCache();
|
|
}
|
|
|
|
#define FORWARD(name, args) \
|
|
for (const FileUnit *file : getFiles()) \
|
|
file->name args;
|
|
|
|
SourceLookupCache &ModuleDecl::getSourceLookupCache() const {
|
|
if (!Cache) {
|
|
const_cast<ModuleDecl *>(this)->Cache =
|
|
llvm::make_unique<SourceLookupCache>(*this);
|
|
}
|
|
return *Cache;
|
|
}
|
|
|
|
static bool isParsedModule(const ModuleDecl *mod) {
|
|
// FIXME: If we ever get mixed modules that contain both SourceFiles and other
|
|
// kinds of file units, this will break; there all callers of this function should
|
|
// themselves assert that all file units in the module are SourceFiles when this
|
|
// function returns true.
|
|
auto files = mod->getFiles();
|
|
return (files.size() > 0 &&
|
|
isa<SourceFile>(files[0]) &&
|
|
cast<SourceFile>(files[0])->Kind != SourceFileKind::SIL);
|
|
}
|
|
|
|
void ModuleDecl::lookupValue(AccessPathTy AccessPath, DeclName Name,
|
|
NLKind LookupKind,
|
|
SmallVectorImpl<ValueDecl*> &Result) const {
|
|
auto *stats = getASTContext().Stats;
|
|
if (stats)
|
|
stats->getFrontendCounters().NumModuleLookupValue++;
|
|
|
|
FrontendStatsTracer tracer(stats, "module-lookup-value");
|
|
|
|
if (isParsedModule(this)) {
|
|
FrontendStatsTracer tracer(getASTContext().Stats, "source-file-lookup-value");
|
|
getSourceLookupCache().lookupValue(AccessPath, Name, LookupKind, Result);
|
|
return;
|
|
}
|
|
|
|
FORWARD(lookupValue, (AccessPath, Name, LookupKind, Result));
|
|
}
|
|
|
|
TypeDecl * ModuleDecl::lookupLocalType(StringRef MangledName) const {
|
|
for (auto file : getFiles()) {
|
|
auto TD = file->lookupLocalType(MangledName);
|
|
if (TD)
|
|
return TD;
|
|
}
|
|
return nullptr;
|
|
}
|
|
|
|
OpaqueTypeDecl *
|
|
ModuleDecl::lookupOpaqueResultType(StringRef MangledName,
|
|
LazyResolver *resolver) {
|
|
for (auto file : getFiles()) {
|
|
auto OTD = file->lookupOpaqueResultType(MangledName, resolver);
|
|
if (OTD)
|
|
return OTD;
|
|
}
|
|
return nullptr;
|
|
}
|
|
|
|
void ModuleDecl::lookupMember(SmallVectorImpl<ValueDecl*> &results,
|
|
DeclContext *container, DeclName name,
|
|
Identifier privateDiscriminator) const {
|
|
size_t oldSize = results.size();
|
|
bool alreadyInPrivateContext = false;
|
|
|
|
auto containerDecl = container->getAsDecl();
|
|
// If FileUnit, then use FileUnit::lookupValue instead.
|
|
assert(containerDecl != nullptr && "This context does not support lookup.");
|
|
|
|
if (auto nominal = dyn_cast<NominalTypeDecl>(containerDecl)) {
|
|
auto lookupResults = nominal->lookupDirect(name);
|
|
|
|
// Filter out declarations from other modules.
|
|
llvm::copy_if(lookupResults,
|
|
std::back_inserter(results),
|
|
[this](const ValueDecl *VD) -> bool {
|
|
return VD->getModuleContext() == this;
|
|
});
|
|
|
|
auto AS = nominal->getFormalAccessScope();
|
|
if (AS.isPrivate() || AS.isFileScope())
|
|
alreadyInPrivateContext = true;
|
|
} else if (isa<ModuleDecl>(containerDecl)) {
|
|
assert(container == this);
|
|
this->lookupValue({}, name, NLKind::QualifiedLookup, results);
|
|
} else if (!isa<GenericTypeDecl>(containerDecl)) {
|
|
// If ExtensionDecl, then use ExtensionDecl::lookupDirect instead.
|
|
llvm_unreachable("This context does not support lookup.");
|
|
}
|
|
|
|
// Filter by private-discriminator, or filter out private decls if there isn't
|
|
// one...unless we're already in a private context, in which case everything
|
|
// is private and a discriminator is unnecessary.
|
|
if (alreadyInPrivateContext) {
|
|
assert(privateDiscriminator.empty() && "unnecessary private discriminator");
|
|
// Don't remove anything; everything here is private anyway.
|
|
|
|
} else if (privateDiscriminator.empty()) {
|
|
auto newEnd = std::remove_if(results.begin()+oldSize, results.end(),
|
|
[](const ValueDecl *VD) -> bool {
|
|
return VD->getFormalAccess() <= AccessLevel::FilePrivate;
|
|
});
|
|
results.erase(newEnd, results.end());
|
|
|
|
} else {
|
|
auto newEnd = std::remove_if(results.begin()+oldSize, results.end(),
|
|
[=](const ValueDecl *VD) -> bool {
|
|
if (VD->getFormalAccess() > AccessLevel::FilePrivate)
|
|
return true;
|
|
auto enclosingFile =
|
|
cast<FileUnit>(VD->getDeclContext()->getModuleScopeContext());
|
|
auto discriminator = enclosingFile->getDiscriminatorForPrivateValue(VD);
|
|
return discriminator != privateDiscriminator;
|
|
});
|
|
results.erase(newEnd, results.end());
|
|
}
|
|
}
|
|
|
|
void ModuleDecl::lookupObjCMethods(
|
|
ObjCSelector selector,
|
|
SmallVectorImpl<AbstractFunctionDecl *> &results) const {
|
|
FORWARD(lookupObjCMethods, (selector, results));
|
|
}
|
|
|
|
void BuiltinUnit::lookupValue(ModuleDecl::AccessPathTy accessPath, DeclName name,
|
|
NLKind lookupKind,
|
|
SmallVectorImpl<ValueDecl*> &result) const {
|
|
getCache().lookupValue(name.getBaseIdentifier(), lookupKind, *this, result);
|
|
}
|
|
|
|
void BuiltinUnit::lookupObjCMethods(
|
|
ObjCSelector selector,
|
|
SmallVectorImpl<AbstractFunctionDecl *> &results) const {
|
|
// No @objc methods in the Builtin module.
|
|
}
|
|
|
|
void SourceFile::lookupValue(ModuleDecl::AccessPathTy accessPath, DeclName name,
|
|
NLKind lookupKind,
|
|
SmallVectorImpl<ValueDecl*> &result) const {
|
|
getCache().lookupValue(accessPath, name, lookupKind, result);
|
|
}
|
|
|
|
void ModuleDecl::lookupVisibleDecls(AccessPathTy AccessPath,
|
|
VisibleDeclConsumer &Consumer,
|
|
NLKind LookupKind) const {
|
|
if (isParsedModule(this))
|
|
return getSourceLookupCache().lookupVisibleDecls(
|
|
AccessPath, Consumer, LookupKind);
|
|
|
|
FORWARD(lookupVisibleDecls, (AccessPath, Consumer, LookupKind));
|
|
}
|
|
|
|
void SourceFile::lookupVisibleDecls(ModuleDecl::AccessPathTy AccessPath,
|
|
VisibleDeclConsumer &Consumer,
|
|
NLKind LookupKind) const {
|
|
getCache().lookupVisibleDecls(AccessPath, Consumer, LookupKind);
|
|
}
|
|
|
|
void ModuleDecl::lookupClassMembers(AccessPathTy accessPath,
|
|
VisibleDeclConsumer &consumer) const {
|
|
if (isParsedModule(this)) {
|
|
auto &cache = getSourceLookupCache();
|
|
cache.populateMemberCache(*this);
|
|
cache.lookupClassMembers(accessPath, consumer);
|
|
return;
|
|
}
|
|
|
|
FORWARD(lookupClassMembers, (accessPath, consumer));
|
|
}
|
|
|
|
void SourceFile::lookupClassMembers(ModuleDecl::AccessPathTy accessPath,
|
|
VisibleDeclConsumer &consumer) const {
|
|
auto &cache = getCache();
|
|
cache.populateMemberCache(*this);
|
|
cache.lookupClassMembers(accessPath, consumer);
|
|
}
|
|
|
|
void ModuleDecl::lookupClassMember(AccessPathTy accessPath,
|
|
DeclName name,
|
|
SmallVectorImpl<ValueDecl*> &results) const {
|
|
auto *stats = getASTContext().Stats;
|
|
if (stats)
|
|
stats->getFrontendCounters().NumModuleLookupClassMember++;
|
|
|
|
if (isParsedModule(this)) {
|
|
FrontendStatsTracer tracer(getASTContext().Stats, "source-file-lookup-class-member");
|
|
auto &cache = getSourceLookupCache();
|
|
cache.populateMemberCache(*this);
|
|
cache.lookupClassMember(accessPath, name, results);
|
|
return;
|
|
}
|
|
|
|
FORWARD(lookupClassMember, (accessPath, name, results));
|
|
}
|
|
|
|
void SourceFile::lookupClassMember(ModuleDecl::AccessPathTy accessPath,
|
|
DeclName name,
|
|
SmallVectorImpl<ValueDecl*> &results) const {
|
|
FrontendStatsTracer tracer(getASTContext().Stats, "source-file-lookup-class-member");
|
|
auto &cache = getCache();
|
|
cache.populateMemberCache(*this);
|
|
cache.lookupClassMember(accessPath, name, results);
|
|
}
|
|
|
|
void SourceFile::lookupObjCMethods(
|
|
ObjCSelector selector,
|
|
SmallVectorImpl<AbstractFunctionDecl *> &results) const {
|
|
// FIXME: Make sure this table is complete, somehow.
|
|
auto known = ObjCMethods.find(selector);
|
|
if (known == ObjCMethods.end()) return;
|
|
results.append(known->second.begin(), known->second.end());
|
|
}
|
|
|
|
void ModuleDecl::getLocalTypeDecls(SmallVectorImpl<TypeDecl*> &Results) const {
|
|
FORWARD(getLocalTypeDecls, (Results));
|
|
}
|
|
|
|
void ModuleDecl::getTopLevelDecls(SmallVectorImpl<Decl*> &Results) const {
|
|
FORWARD(getTopLevelDecls, (Results));
|
|
}
|
|
|
|
void SourceFile::getTopLevelDecls(SmallVectorImpl<Decl*> &Results) const {
|
|
Results.append(Decls.begin(), Decls.end());
|
|
}
|
|
|
|
void ModuleDecl::getPrecedenceGroups(
|
|
SmallVectorImpl<PrecedenceGroupDecl*> &Results) const {
|
|
FORWARD(getPrecedenceGroups, (Results));
|
|
}
|
|
|
|
void SourceFile::getPrecedenceGroups(
|
|
SmallVectorImpl<PrecedenceGroupDecl*> &Results) const {
|
|
for (auto pair : PrecedenceGroups) {
|
|
if (pair.second.getPointer() && pair.second.getInt()) {
|
|
Results.push_back(pair.second.getPointer());
|
|
}
|
|
}
|
|
}
|
|
|
|
void SourceFile::getLocalTypeDecls(SmallVectorImpl<TypeDecl*> &Results) const {
|
|
Results.append(LocalTypeDecls.begin(), LocalTypeDecls.end());
|
|
}
|
|
|
|
void
|
|
SourceFile::getOpaqueReturnTypeDecls(SmallVectorImpl<OpaqueTypeDecl*> &Results)
|
|
const {
|
|
for (auto &member : ValidatedOpaqueReturnTypes) {
|
|
Results.push_back(member.second);
|
|
}
|
|
}
|
|
|
|
TypeDecl *SourceFile::lookupLocalType(llvm::StringRef mangledName) const {
|
|
ASTContext &ctx = getASTContext();
|
|
for (auto typeDecl : LocalTypeDecls) {
|
|
auto typeMangledName = evaluateOrDefault(ctx.evaluator,
|
|
MangleLocalTypeDeclRequest { typeDecl },
|
|
std::string());
|
|
if (mangledName == typeMangledName)
|
|
return typeDecl;
|
|
}
|
|
|
|
return nullptr;
|
|
}
|
|
|
|
void ModuleDecl::getDisplayDecls(SmallVectorImpl<Decl*> &Results) const {
|
|
// FIXME: Should this do extra access control filtering?
|
|
FORWARD(getDisplayDecls, (Results));
|
|
}
|
|
|
|
Optional<ProtocolConformanceRef>
|
|
ModuleDecl::lookupExistentialConformance(Type type, ProtocolDecl *protocol) {
|
|
ASTContext &ctx = getASTContext();
|
|
|
|
assert(type->isExistentialType());
|
|
|
|
// If the existential type cannot be represented or the protocol does not
|
|
// conform to itself, there's no point in looking further.
|
|
if (!protocol->existentialConformsToSelf())
|
|
return None;
|
|
|
|
auto layout = type->getExistentialLayout();
|
|
|
|
// Due to an IRGen limitation, witness tables cannot be passed from an
|
|
// existential to an archetype parameter, so for now we restrict this to
|
|
// @objc protocols.
|
|
if (!layout.isObjC()) {
|
|
// There's a specific exception for protocols with self-conforming
|
|
// witness tables, but the existential has to be *exactly* that type.
|
|
// TODO: synthesize witness tables on-demand for protocol compositions
|
|
// that can satisfy the requirement.
|
|
if (protocol->requiresSelfConformanceWitnessTable() &&
|
|
type->is<ProtocolType>() &&
|
|
type->castTo<ProtocolType>()->getDecl() == protocol)
|
|
return ProtocolConformanceRef(ctx.getSelfConformance(protocol));
|
|
|
|
return None;
|
|
}
|
|
|
|
// If the existential is class-constrained, the class might conform
|
|
// concretely.
|
|
if (auto superclass = layout.explicitSuperclass) {
|
|
if (auto result = lookupConformance(superclass, protocol))
|
|
return result;
|
|
}
|
|
|
|
// Otherwise, the existential might conform abstractly.
|
|
for (auto proto : layout.getProtocols()) {
|
|
auto *protoDecl = proto->getDecl();
|
|
|
|
// If we found the protocol we're looking for, return an abstract
|
|
// conformance to it.
|
|
if (protoDecl == protocol)
|
|
return ProtocolConformanceRef(ctx.getSelfConformance(protocol));
|
|
|
|
// If the protocol has a superclass constraint, we might conform
|
|
// concretely.
|
|
if (auto superclass = protoDecl->getSuperclass()) {
|
|
if (auto result = lookupConformance(superclass, protocol))
|
|
return result;
|
|
}
|
|
|
|
// Now check refined protocols.
|
|
if (protoDecl->inheritsFrom(protocol))
|
|
return ProtocolConformanceRef(ctx.getSelfConformance(protocol));
|
|
}
|
|
|
|
// We didn't find our protocol in the existential's list; it doesn't
|
|
// conform.
|
|
return None;
|
|
}
|
|
|
|
Optional<ProtocolConformanceRef>
|
|
ModuleDecl::lookupConformance(Type type, ProtocolDecl *protocol) {
|
|
ASTContext &ctx = getASTContext();
|
|
|
|
// A dynamic Self type conforms to whatever its underlying type
|
|
// conforms to.
|
|
if (auto selfType = type->getAs<DynamicSelfType>())
|
|
type = selfType->getSelfType();
|
|
|
|
// An archetype conforms to a protocol if the protocol is listed in the
|
|
// archetype's list of conformances, or if the archetype has a superclass
|
|
// constraint and the superclass conforms to the protocol.
|
|
if (auto archetype = type->getAs<ArchetypeType>()) {
|
|
|
|
// The generic signature builder drops conformance requirements that are made
|
|
// redundant by a superclass requirement, so check for a concrete
|
|
// conformance first, since an abstract conformance might not be
|
|
// able to be resolved by a substitution that makes the archetype
|
|
// concrete.
|
|
if (auto super = archetype->getSuperclass()) {
|
|
if (auto inheritedConformance = lookupConformance(super, protocol)) {
|
|
return ProtocolConformanceRef(
|
|
ctx.getInheritedConformance(
|
|
type,
|
|
inheritedConformance->getConcrete()));
|
|
}
|
|
}
|
|
|
|
for (auto ap : archetype->getConformsTo()) {
|
|
if (ap == protocol || ap->inheritsFrom(protocol))
|
|
return ProtocolConformanceRef(protocol);
|
|
}
|
|
|
|
return None;
|
|
}
|
|
|
|
// An existential conforms to a protocol if the protocol is listed in the
|
|
// existential's list of conformances and the existential conforms to
|
|
// itself.
|
|
if (type->isExistentialType())
|
|
return lookupExistentialConformance(type, protocol);
|
|
|
|
// Type variables have trivial conformances.
|
|
if (type->isTypeVariableOrMember())
|
|
return ProtocolConformanceRef(protocol);
|
|
|
|
// UnresolvedType is a placeholder for an unknown type used when generating
|
|
// diagnostics. We consider it to conform to all protocols, since the
|
|
// intended type might have.
|
|
if (type->is<UnresolvedType>())
|
|
return ProtocolConformanceRef(protocol);
|
|
|
|
auto nominal = type->getAnyNominal();
|
|
|
|
// If we don't have a nominal type, there are no conformances.
|
|
if (!nominal || isa<ProtocolDecl>(nominal)) return None;
|
|
|
|
// Find the (unspecialized) conformance.
|
|
SmallVector<ProtocolConformance *, 2> conformances;
|
|
if (!nominal->lookupConformance(this, protocol, conformances))
|
|
return None;
|
|
|
|
// FIXME: Ambiguity resolution.
|
|
auto conformance = conformances.front();
|
|
|
|
// Rebuild inherited conformances based on the root normal conformance.
|
|
// FIXME: This is a hack to work around our inability to handle multiple
|
|
// levels of substitution through inherited conformances elsewhere in the
|
|
// compiler.
|
|
if (auto inherited = dyn_cast<InheritedProtocolConformance>(conformance)) {
|
|
// Dig out the conforming nominal type.
|
|
auto rootConformance = inherited->getRootConformance();
|
|
auto conformingClass
|
|
= rootConformance->getType()->getClassOrBoundGenericClass();
|
|
|
|
// Map up to our superclass's type.
|
|
auto superclassTy = type->getSuperclassForDecl(conformingClass);
|
|
|
|
// Compute the conformance for the inherited type.
|
|
auto inheritedConformance = lookupConformance(superclassTy, protocol);
|
|
assert(inheritedConformance &&
|
|
"We already found the inherited conformance");
|
|
|
|
// Create the inherited conformance entry.
|
|
conformance
|
|
= ctx.getInheritedConformance(type, inheritedConformance->getConcrete());
|
|
return ProtocolConformanceRef(conformance);
|
|
}
|
|
|
|
// If the type is specialized, find the conformance for the generic type.
|
|
if (type->isSpecialized()) {
|
|
// Figure out the type that's explicitly conforming to this protocol.
|
|
Type explicitConformanceType = conformance->getType();
|
|
DeclContext *explicitConformanceDC = conformance->getDeclContext();
|
|
|
|
// If the explicit conformance is associated with a type that is different
|
|
// from the type we're checking, retrieve generic conformance.
|
|
if (!explicitConformanceType->isEqual(type)) {
|
|
// Gather the substitutions we need to map the generic conformance to
|
|
// the specialized conformance.
|
|
auto subMap = type->getContextSubstitutionMap(this, explicitConformanceDC);
|
|
|
|
// Create the specialized conformance entry.
|
|
auto result = ctx.getSpecializedConformance(type, conformance, subMap);
|
|
return ProtocolConformanceRef(result);
|
|
}
|
|
}
|
|
|
|
// Record and return the simple conformance.
|
|
return ProtocolConformanceRef(conformance);
|
|
}
|
|
|
|
namespace {
|
|
template <typename T>
|
|
using OperatorMap = SourceFile::OperatorMap<T>;
|
|
|
|
template <typename T>
|
|
struct OperatorLookup {
|
|
// Don't fold this into the static_assert: this would trigger an MSVC bug
|
|
// that causes the assertion to fail.
|
|
static constexpr T* ptr = static_cast<T*>(nullptr);
|
|
static_assert(ptr, "Only usable with operators");
|
|
};
|
|
|
|
template <>
|
|
struct OperatorLookup<PrefixOperatorDecl> {
|
|
template <typename T>
|
|
static PrefixOperatorDecl *lookup(T &container, Identifier name) {
|
|
return cast_or_null<PrefixOperatorDecl>(
|
|
container.lookupOperator(name, DeclKind::PrefixOperator));
|
|
}
|
|
};
|
|
|
|
template <>
|
|
struct OperatorLookup<InfixOperatorDecl> {
|
|
template <typename T>
|
|
static InfixOperatorDecl *lookup(T &container, Identifier name) {
|
|
return cast_or_null<InfixOperatorDecl>(
|
|
container.lookupOperator(name, DeclKind::InfixOperator));
|
|
}
|
|
};
|
|
|
|
template <>
|
|
struct OperatorLookup<PostfixOperatorDecl> {
|
|
template <typename T>
|
|
static PostfixOperatorDecl *lookup(T &container, Identifier name) {
|
|
return cast_or_null<PostfixOperatorDecl>(
|
|
container.lookupOperator(name, DeclKind::PostfixOperator));
|
|
}
|
|
};
|
|
|
|
template <>
|
|
struct OperatorLookup<PrecedenceGroupDecl> {
|
|
template <typename T>
|
|
static PrecedenceGroupDecl *lookup(T &container, Identifier name) {
|
|
return container.lookupPrecedenceGroup(name);
|
|
}
|
|
};
|
|
} // end anonymous namespace
|
|
|
|
/// A helper class to sneak around C++ access control rules.
|
|
class SourceFile::Impl {
|
|
public:
|
|
/// Only intended for use by lookupOperatorDeclForName.
|
|
static ArrayRef<SourceFile::ImportedModuleDesc>
|
|
getImportsForSourceFile(const SourceFile &SF) {
|
|
return SF.Imports;
|
|
}
|
|
};
|
|
|
|
struct SourceFile::SourceFileSyntaxInfo {
|
|
const bool Enable;
|
|
/// The root of the syntax tree representing the source file.
|
|
Optional<syntax::SourceFileSyntax> SyntaxRoot;
|
|
SourceFileSyntaxInfo(bool Enable): Enable(Enable) {}
|
|
};
|
|
|
|
bool SourceFile::hasSyntaxRoot() const {
|
|
return SyntaxInfo->SyntaxRoot.hasValue();
|
|
}
|
|
|
|
syntax::SourceFileSyntax SourceFile::getSyntaxRoot() const {
|
|
assert(hasSyntaxRoot() && "no syntax root is set.");
|
|
return *SyntaxInfo->SyntaxRoot;
|
|
}
|
|
|
|
void SourceFile::setSyntaxRoot(syntax::SourceFileSyntax &&Root) {
|
|
SyntaxInfo->SyntaxRoot.emplace(Root);
|
|
}
|
|
|
|
template<typename OP_DECL>
|
|
static Optional<OP_DECL *>
|
|
lookupOperatorDeclForName(ModuleDecl *M, SourceLoc Loc, Identifier Name,
|
|
OperatorMap<OP_DECL *> SourceFile::*OP_MAP);
|
|
|
|
template<typename OP_DECL>
|
|
using ImportedOperatorsMap = llvm::SmallDenseMap<OP_DECL*, bool, 16>;
|
|
|
|
template<typename OP_DECL>
|
|
static typename ImportedOperatorsMap<OP_DECL>::iterator
|
|
checkOperatorConflicts(const SourceFile &SF, SourceLoc loc,
|
|
ImportedOperatorsMap<OP_DECL> &importedOperators) {
|
|
// Check for conflicts.
|
|
auto i = importedOperators.begin(), end = importedOperators.end();
|
|
auto start = i;
|
|
for (++i; i != end; ++i) {
|
|
if (i->first->conflictsWith(start->first)) {
|
|
if (loc.isValid()) {
|
|
ASTContext &C = SF.getASTContext();
|
|
C.Diags.diagnose(loc, diag::ambiguous_operator_decls);
|
|
C.Diags.diagnose(start->first->getLoc(),
|
|
diag::found_this_operator_decl);
|
|
C.Diags.diagnose(i->first->getLoc(), diag::found_this_operator_decl);
|
|
}
|
|
return end;
|
|
}
|
|
}
|
|
return start;
|
|
}
|
|
|
|
template<>
|
|
typename ImportedOperatorsMap<PrecedenceGroupDecl>::iterator
|
|
checkOperatorConflicts(const SourceFile &SF, SourceLoc loc,
|
|
ImportedOperatorsMap<PrecedenceGroupDecl> &importedGroups) {
|
|
if (importedGroups.size() == 1)
|
|
return importedGroups.begin();
|
|
|
|
// Any sort of ambiguity is an error.
|
|
if (loc.isValid()) {
|
|
ASTContext &C = SF.getASTContext();
|
|
C.Diags.diagnose(loc, diag::ambiguous_precedence_groups);
|
|
for (auto &entry : importedGroups) {
|
|
C.Diags.diagnose(entry.first->getLoc(),
|
|
diag::found_this_precedence_group);
|
|
}
|
|
}
|
|
return importedGroups.end();
|
|
}
|
|
|
|
// Returns None on error, Optional(nullptr) if no operator decl found, or
|
|
// Optional(decl) if decl was found.
|
|
template<typename OP_DECL>
|
|
static Optional<OP_DECL *>
|
|
lookupOperatorDeclForName(const FileUnit &File, SourceLoc Loc, Identifier Name,
|
|
bool includePrivate,
|
|
OperatorMap<OP_DECL *> SourceFile::*OP_MAP)
|
|
{
|
|
switch (File.getKind()) {
|
|
case FileUnitKind::Builtin:
|
|
// The Builtin module declares no operators.
|
|
return nullptr;
|
|
case FileUnitKind::Source:
|
|
break;
|
|
case FileUnitKind::SerializedAST:
|
|
case FileUnitKind::ClangModule:
|
|
case FileUnitKind::DWARFModule:
|
|
return OperatorLookup<OP_DECL>::lookup(cast<LoadedFile>(File), Name);
|
|
}
|
|
|
|
auto &SF = cast<SourceFile>(File);
|
|
assert(SF.ASTStage >= SourceFile::NameBound);
|
|
|
|
// Look for an operator declaration in the current module.
|
|
auto found = (SF.*OP_MAP).find(Name);
|
|
if (found != (SF.*OP_MAP).end() && (includePrivate || found->second.getInt()))
|
|
return found->second.getPointer();
|
|
|
|
// Look for imported operator decls.
|
|
// Record whether they come from re-exported modules.
|
|
// FIXME: We ought to prefer operators elsewhere in this module before we
|
|
// check imports.
|
|
auto ownModule = SF.getParentModule();
|
|
ImportedOperatorsMap<OP_DECL> importedOperators;
|
|
for (auto &imported : SourceFile::Impl::getImportsForSourceFile(SF)) {
|
|
// Protect against source files that contrive to import their own modules.
|
|
if (imported.module.second == ownModule)
|
|
continue;
|
|
|
|
bool isExported =
|
|
imported.importOptions.contains(SourceFile::ImportFlags::Exported);
|
|
if (!includePrivate && !isExported)
|
|
continue;
|
|
|
|
Optional<OP_DECL *> maybeOp =
|
|
lookupOperatorDeclForName(imported.module.second, Loc, Name, OP_MAP);
|
|
if (!maybeOp)
|
|
return None;
|
|
|
|
if (OP_DECL *op = *maybeOp)
|
|
importedOperators[op] |= isExported;
|
|
}
|
|
|
|
typename OperatorMap<OP_DECL *>::mapped_type result = { nullptr, true };
|
|
|
|
if (!importedOperators.empty()) {
|
|
auto start = checkOperatorConflicts(SF, Loc, importedOperators);
|
|
if (start == importedOperators.end())
|
|
return None;
|
|
result = { start->first, start->second };
|
|
}
|
|
|
|
if (includePrivate) {
|
|
// Cache the mapping so we don't need to troll imports next time.
|
|
// It's not safe to cache the non-private results because we didn't search
|
|
// private imports there, but in most non-private cases the result will
|
|
// be cached in the final lookup.
|
|
auto &mutableOpMap = const_cast<OperatorMap<OP_DECL *> &>(SF.*OP_MAP);
|
|
mutableOpMap[Name] = result;
|
|
}
|
|
|
|
if (includePrivate || result.getInt())
|
|
return result.getPointer();
|
|
return nullptr;
|
|
}
|
|
|
|
template<typename OP_DECL>
|
|
static Optional<OP_DECL *>
|
|
lookupOperatorDeclForName(ModuleDecl *M, SourceLoc Loc, Identifier Name,
|
|
OperatorMap<OP_DECL *> SourceFile::*OP_MAP)
|
|
{
|
|
OP_DECL *result = nullptr;
|
|
for (const FileUnit *File : M->getFiles()) {
|
|
auto next = lookupOperatorDeclForName(*File, Loc, Name, false, OP_MAP);
|
|
if (!next.hasValue())
|
|
return next;
|
|
|
|
// FIXME: Diagnose ambiguity.
|
|
if (*next && result)
|
|
return None;
|
|
if (*next)
|
|
result = *next;
|
|
}
|
|
return result;
|
|
}
|
|
|
|
#define LOOKUP_OPERATOR(Kind) \
|
|
Kind##Decl * \
|
|
ModuleDecl::lookup##Kind(Identifier name, SourceLoc loc) { \
|
|
auto result = lookupOperatorDeclForName(this, loc, name, \
|
|
&SourceFile::Kind##s); \
|
|
return result ? *result : nullptr; \
|
|
} \
|
|
Kind##Decl * \
|
|
SourceFile::lookup##Kind(Identifier name, bool isCascading, SourceLoc loc) { \
|
|
auto result = lookupOperatorDeclForName(*this, loc, name, true, \
|
|
&SourceFile::Kind##s); \
|
|
if (!result.hasValue()) \
|
|
return nullptr; \
|
|
if (ReferencedNames) {\
|
|
if (!result.getValue() || \
|
|
result.getValue()->getDeclContext()->getModuleScopeContext() != this) {\
|
|
ReferencedNames->addTopLevelName(name, isCascading); \
|
|
} \
|
|
} \
|
|
if (!result.getValue()) { \
|
|
result = lookupOperatorDeclForName(getParentModule(), loc, name, \
|
|
&SourceFile::Kind##s); \
|
|
} \
|
|
return result.hasValue() ? result.getValue() : nullptr; \
|
|
}
|
|
|
|
LOOKUP_OPERATOR(PrefixOperator)
|
|
LOOKUP_OPERATOR(InfixOperator)
|
|
LOOKUP_OPERATOR(PostfixOperator)
|
|
LOOKUP_OPERATOR(PrecedenceGroup)
|
|
#undef LOOKUP_OPERATOR
|
|
|
|
void ModuleDecl::getImportedModules(SmallVectorImpl<ImportedModule> &modules,
|
|
ModuleDecl::ImportFilter filter) const {
|
|
FORWARD(getImportedModules, (modules, filter));
|
|
}
|
|
|
|
void
|
|
SourceFile::getImportedModules(SmallVectorImpl<ModuleDecl::ImportedModule> &modules,
|
|
ModuleDecl::ImportFilter filter) const {
|
|
assert(ASTStage >= Parsed || Kind == SourceFileKind::SIL);
|
|
assert(filter && "no imports requested?");
|
|
for (auto desc : Imports) {
|
|
ModuleDecl::ImportFilterKind requiredKind;
|
|
if (desc.importOptions.contains(ImportFlags::Exported))
|
|
requiredKind = ModuleDecl::ImportFilterKind::Public;
|
|
else if (desc.importOptions.contains(ImportFlags::ImplementationOnly))
|
|
requiredKind = ModuleDecl::ImportFilterKind::ImplementationOnly;
|
|
else
|
|
requiredKind = ModuleDecl::ImportFilterKind::Private;
|
|
|
|
if (filter.contains(requiredKind))
|
|
modules.push_back(desc.module);
|
|
}
|
|
}
|
|
|
|
void ModuleDecl::getImportedModulesForLookup(
|
|
SmallVectorImpl<ImportedModule> &modules) const {
|
|
FORWARD(getImportedModulesForLookup, (modules));
|
|
}
|
|
|
|
bool ModuleDecl::isSameAccessPath(AccessPathTy lhs, AccessPathTy rhs) {
|
|
using AccessPathElem = std::pair<Identifier, SourceLoc>;
|
|
if (lhs.size() != rhs.size())
|
|
return false;
|
|
return std::equal(lhs.begin(), lhs.end(), rhs.begin(),
|
|
[](const AccessPathElem &lElem,
|
|
const AccessPathElem &rElem) {
|
|
return lElem.first == rElem.first;
|
|
});
|
|
}
|
|
|
|
ModuleDecl::ReverseFullNameIterator::ReverseFullNameIterator(
|
|
const ModuleDecl *M) {
|
|
assert(M);
|
|
// Note: This will look through overlays as well, but that's fine for name
|
|
// generation purposes. The point of an overlay is to
|
|
if (auto *clangModule = M->findUnderlyingClangModule())
|
|
current = clangModule;
|
|
else
|
|
current = M;
|
|
}
|
|
|
|
StringRef ModuleDecl::ReverseFullNameIterator::operator*() const {
|
|
assert(current && "all name components exhausted");
|
|
|
|
if (auto *swiftModule = current.dyn_cast<const ModuleDecl *>())
|
|
return swiftModule->getName().str();
|
|
|
|
auto *clangModule =
|
|
static_cast<const clang::Module *>(current.get<const void *>());
|
|
return clangModule->Name;
|
|
}
|
|
|
|
ModuleDecl::ReverseFullNameIterator &
|
|
ModuleDecl::ReverseFullNameIterator::operator++() {
|
|
if (!current)
|
|
return *this;
|
|
|
|
if (auto *swiftModule = current.dyn_cast<const ModuleDecl *>()) {
|
|
current = nullptr;
|
|
return *this;
|
|
}
|
|
|
|
auto *clangModule =
|
|
static_cast<const clang::Module *>(current.get<const void *>());
|
|
if (clangModule->Parent)
|
|
current = clangModule->Parent;
|
|
else
|
|
current = nullptr;
|
|
return *this;
|
|
}
|
|
|
|
void
|
|
ModuleDecl::ReverseFullNameIterator::printForward(raw_ostream &out) const {
|
|
SmallVector<StringRef, 8> elements(*this, {});
|
|
swift::interleave(swift::reversed(elements),
|
|
[&out](StringRef next) { out << next; },
|
|
[&out] { out << '.'; });
|
|
}
|
|
|
|
void
|
|
ModuleDecl::removeDuplicateImports(SmallVectorImpl<ImportedModule> &imports) {
|
|
std::sort(imports.begin(), imports.end(),
|
|
[](const ImportedModule &lhs, const ImportedModule &rhs) -> bool {
|
|
// Arbitrarily sort by name to get a deterministic order.
|
|
if (lhs.second != rhs.second) {
|
|
return std::lexicographical_compare(
|
|
lhs.second->getReverseFullModuleName(), {},
|
|
rhs.second->getReverseFullModuleName(), {});
|
|
}
|
|
using AccessPathElem = std::pair<Identifier, SourceLoc>;
|
|
return std::lexicographical_compare(lhs.first.begin(), lhs.first.end(),
|
|
rhs.first.begin(), rhs.first.end(),
|
|
[](const AccessPathElem &lElem,
|
|
const AccessPathElem &rElem) {
|
|
return lElem.first.str() < rElem.first.str();
|
|
});
|
|
});
|
|
auto last = std::unique(imports.begin(), imports.end(),
|
|
[](const ImportedModule &lhs,
|
|
const ImportedModule &rhs) -> bool {
|
|
if (lhs.second != rhs.second)
|
|
return false;
|
|
return ModuleDecl::isSameAccessPath(lhs.first, rhs.first);
|
|
});
|
|
imports.erase(last, imports.end());
|
|
}
|
|
|
|
|
|
StringRef ModuleDecl::getModuleFilename() const {
|
|
// FIXME: Audit uses of this function and figure out how to migrate them to
|
|
// per-file names. Modules can consist of more than one file.
|
|
StringRef Result;
|
|
for (auto F : getFiles()) {
|
|
Result = F->getParseableInterface();
|
|
if (!Result.empty())
|
|
return Result;
|
|
|
|
if (auto SF = dyn_cast<SourceFile>(F)) {
|
|
if (!Result.empty())
|
|
return StringRef();
|
|
Result = SF->getFilename();
|
|
continue;
|
|
}
|
|
if (auto LF = dyn_cast<LoadedFile>(F)) {
|
|
if (!Result.empty())
|
|
return StringRef();
|
|
Result = LF->getFilename();
|
|
continue;
|
|
}
|
|
return StringRef();
|
|
}
|
|
return Result;
|
|
}
|
|
|
|
bool ModuleDecl::isStdlibModule() const {
|
|
return !getParent() && getName() == getASTContext().StdlibModuleName;
|
|
}
|
|
|
|
bool ModuleDecl::isSwiftShimsModule() const {
|
|
return !getParent() && getName() == getASTContext().SwiftShimsModuleName;
|
|
}
|
|
|
|
bool ModuleDecl::isOnoneSupportModule() const {
|
|
return !getParent() && getName().str() == SWIFT_ONONE_SUPPORT;
|
|
}
|
|
|
|
bool ModuleDecl::isBuiltinModule() const {
|
|
return this == getASTContext().TheBuiltinModule;
|
|
}
|
|
|
|
bool SourceFile::registerMainClass(ClassDecl *mainClass, SourceLoc diagLoc) {
|
|
if (mainClass == MainClass)
|
|
return false;
|
|
|
|
ArtificialMainKind kind = mainClass->getArtificialMainKind();
|
|
if (getParentModule()->registerEntryPointFile(this, diagLoc, kind))
|
|
return true;
|
|
|
|
MainClass = mainClass;
|
|
MainClassDiagLoc = diagLoc;
|
|
return false;
|
|
}
|
|
|
|
bool ModuleDecl::registerEntryPointFile(FileUnit *file, SourceLoc diagLoc,
|
|
Optional<ArtificialMainKind> kind) {
|
|
if (!EntryPointInfo.hasEntryPoint()) {
|
|
EntryPointInfo.setEntryPointFile(file);
|
|
return false;
|
|
}
|
|
|
|
if (diagLoc.isInvalid())
|
|
return true;
|
|
|
|
assert(kind.hasValue() && "multiple entry points without attributes");
|
|
|
|
// %select indices for UI/NSApplication-related diagnostics.
|
|
enum : unsigned {
|
|
UIApplicationMainClass = 0,
|
|
NSApplicationMainClass = 1,
|
|
} mainClassDiagKind;
|
|
|
|
switch (kind.getValue()) {
|
|
case ArtificialMainKind::UIApplicationMain:
|
|
mainClassDiagKind = UIApplicationMainClass;
|
|
break;
|
|
case ArtificialMainKind::NSApplicationMain:
|
|
mainClassDiagKind = NSApplicationMainClass;
|
|
break;
|
|
}
|
|
|
|
FileUnit *existingFile = EntryPointInfo.getEntryPointFile();
|
|
const ClassDecl *existingClass = existingFile->getMainClass();
|
|
SourceLoc existingDiagLoc;
|
|
|
|
if (auto *sourceFile = dyn_cast<SourceFile>(existingFile)) {
|
|
if (existingClass) {
|
|
existingDiagLoc = sourceFile->getMainClassDiagLoc();
|
|
} else {
|
|
if (auto bufID = sourceFile->getBufferID())
|
|
existingDiagLoc = getASTContext().SourceMgr.getLocForBufferStart(*bufID);
|
|
}
|
|
}
|
|
|
|
if (existingClass) {
|
|
if (EntryPointInfo.markDiagnosedMultipleMainClasses()) {
|
|
// If we already have a main class, and we haven't diagnosed it,
|
|
// do so now.
|
|
if (existingDiagLoc.isValid()) {
|
|
getASTContext().Diags.diagnose(existingDiagLoc, diag::attr_ApplicationMain_multiple,
|
|
mainClassDiagKind);
|
|
} else {
|
|
getASTContext().Diags.diagnose(existingClass, diag::attr_ApplicationMain_multiple,
|
|
mainClassDiagKind);
|
|
}
|
|
}
|
|
|
|
// Always diagnose the new class.
|
|
getASTContext().Diags.diagnose(diagLoc, diag::attr_ApplicationMain_multiple,
|
|
mainClassDiagKind);
|
|
|
|
} else {
|
|
// We don't have an existing class, but we /do/ have a file in script mode.
|
|
// Diagnose that.
|
|
if (EntryPointInfo.markDiagnosedMainClassWithScript()) {
|
|
getASTContext().Diags.diagnose(diagLoc, diag::attr_ApplicationMain_with_script,
|
|
mainClassDiagKind);
|
|
|
|
if (existingDiagLoc.isValid()) {
|
|
getASTContext().Diags.diagnose(existingDiagLoc,
|
|
diag::attr_ApplicationMain_script_here);
|
|
}
|
|
}
|
|
}
|
|
|
|
return true;
|
|
}
|
|
|
|
template<bool respectVisibility>
|
|
static bool
|
|
forAllImportedModules(ModuleDecl *topLevel, ModuleDecl::AccessPathTy thisPath,
|
|
llvm::function_ref<bool(ModuleDecl::ImportedModule)> fn) {
|
|
using ImportedModule = ModuleDecl::ImportedModule;
|
|
using AccessPathTy = ModuleDecl::AccessPathTy;
|
|
|
|
llvm::SmallSet<ImportedModule, 32, ModuleDecl::OrderImportedModules> visited;
|
|
SmallVector<ImportedModule, 32> stack;
|
|
|
|
ModuleDecl::ImportFilter filter = ModuleDecl::ImportFilterKind::Public;
|
|
if (!respectVisibility)
|
|
filter |= ModuleDecl::ImportFilterKind::Private;
|
|
|
|
ModuleDecl::ImportFilter topLevelFilter = filter;
|
|
if (!respectVisibility)
|
|
topLevelFilter |= ModuleDecl::ImportFilterKind::ImplementationOnly;
|
|
topLevel->getImportedModules(stack, topLevelFilter);
|
|
|
|
// Make sure the top-level module is first; we want pre-order-ish traversal.
|
|
AccessPathTy overridingPath;
|
|
if (respectVisibility)
|
|
overridingPath = thisPath;
|
|
stack.push_back(ImportedModule(overridingPath, topLevel));
|
|
|
|
while (!stack.empty()) {
|
|
auto next = stack.pop_back_val();
|
|
|
|
// Filter any whole-module imports, and skip specific-decl imports if the
|
|
// import path doesn't match exactly.
|
|
if (next.first.empty() || !respectVisibility)
|
|
next.first = overridingPath;
|
|
else if (!overridingPath.empty() &&
|
|
!ModuleDecl::isSameAccessPath(next.first, overridingPath)) {
|
|
// If we ever allow importing non-top-level decls, it's possible the rule
|
|
// above isn't what we want.
|
|
assert(next.first.size() == 1 && "import of non-top-level decl");
|
|
continue;
|
|
}
|
|
|
|
if (!visited.insert(next).second)
|
|
continue;
|
|
|
|
if (!fn(next))
|
|
return false;
|
|
|
|
if (respectVisibility)
|
|
next.second->getImportedModulesForLookup(stack);
|
|
else
|
|
next.second->getImportedModules(stack, filter);
|
|
}
|
|
|
|
return true;
|
|
}
|
|
|
|
bool
|
|
ModuleDecl::forAllVisibleModules(AccessPathTy thisPath,
|
|
llvm::function_ref<bool(ImportedModule)> fn) {
|
|
return forAllImportedModules<true>(this, thisPath, fn);
|
|
}
|
|
|
|
bool FileUnit::forAllVisibleModules(
|
|
llvm::function_ref<bool(ModuleDecl::ImportedModule)> fn) {
|
|
if (!getParentModule()->forAllVisibleModules(ModuleDecl::AccessPathTy(), fn))
|
|
return false;
|
|
|
|
if (auto SF = dyn_cast<SourceFile>(this)) {
|
|
// Handle privately visible modules as well.
|
|
ModuleDecl::ImportFilter importFilter;
|
|
importFilter |= ModuleDecl::ImportFilterKind::Private;
|
|
importFilter |= ModuleDecl::ImportFilterKind::ImplementationOnly;
|
|
SmallVector<ModuleDecl::ImportedModule, 4> imports;
|
|
SF->getImportedModules(imports, importFilter);
|
|
for (auto importPair : imports)
|
|
if (!importPair.second->forAllVisibleModules(importPair.first, fn))
|
|
return false;
|
|
}
|
|
|
|
return true;
|
|
}
|
|
|
|
void ModuleDecl::collectLinkLibraries(LinkLibraryCallback callback) {
|
|
// FIXME: The proper way to do this depends on the decls used.
|
|
FORWARD(collectLinkLibraries, (callback));
|
|
}
|
|
|
|
void
|
|
SourceFile::collectLinkLibraries(ModuleDecl::LinkLibraryCallback callback) const {
|
|
forAllImportedModules<false>(getParentModule(), /*thisPath*/{},
|
|
[=](ModuleDecl::ImportedModule import) -> bool {
|
|
swift::ModuleDecl *next = import.second;
|
|
if (next->getName() == getParentModule()->getName())
|
|
return true;
|
|
|
|
// Hack: Assume other REPL files already have their libraries linked.
|
|
if (!next->getFiles().empty())
|
|
if (auto *nextSource = dyn_cast<SourceFile>(next->getFiles().front()))
|
|
if (nextSource->Kind == SourceFileKind::REPL)
|
|
return true;
|
|
|
|
next->collectLinkLibraries(callback);
|
|
return true;
|
|
});
|
|
}
|
|
|
|
bool ModuleDecl::walk(ASTWalker &Walker) {
|
|
llvm::SaveAndRestore<ASTWalker::ParentTy> SAR(Walker.Parent, this);
|
|
for (auto SF : getFiles())
|
|
if (SF->walk(Walker))
|
|
return true;
|
|
return false;
|
|
}
|
|
|
|
const clang::Module *ModuleDecl::findUnderlyingClangModule() const {
|
|
for (auto *FU : getFiles()) {
|
|
if (auto *Mod = FU->getUnderlyingClangModule())
|
|
return Mod;
|
|
}
|
|
return nullptr;
|
|
}
|
|
|
|
//===----------------------------------------------------------------------===//
|
|
// SourceFile Implementation
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
void SourceFile::print(raw_ostream &OS, const PrintOptions &PO) {
|
|
StreamPrinter Printer(OS);
|
|
print(Printer, PO);
|
|
}
|
|
|
|
void SourceFile::print(ASTPrinter &Printer, const PrintOptions &PO) {
|
|
std::set<DeclKind> MajorDeclKinds = {DeclKind::Class, DeclKind::Enum,
|
|
DeclKind::Extension, DeclKind::Protocol, DeclKind::Struct};
|
|
for (auto decl : Decls) {
|
|
if (!decl->shouldPrintInContext(PO))
|
|
continue;
|
|
// For a major decl, we print an empty line before it.
|
|
if (MajorDeclKinds.find(decl->getKind()) != MajorDeclKinds.end())
|
|
Printer << "\n";
|
|
if (decl->print(Printer, PO))
|
|
Printer << "\n";
|
|
}
|
|
}
|
|
|
|
void SourceFile::addImports(ArrayRef<ImportedModuleDesc> IM) {
|
|
if (IM.empty())
|
|
return;
|
|
ASTContext &ctx = getASTContext();
|
|
auto newBuf =
|
|
ctx.AllocateUninitialized<ImportedModuleDesc>(Imports.size() + IM.size());
|
|
|
|
auto iter = newBuf.begin();
|
|
iter = std::uninitialized_copy(Imports.begin(), Imports.end(), iter);
|
|
iter = std::uninitialized_copy(IM.begin(), IM.end(), iter);
|
|
assert(iter == newBuf.end());
|
|
|
|
Imports = newBuf;
|
|
|
|
// Update the HasImplementationOnlyImports flag.
|
|
if (!HasImplementationOnlyImports) {
|
|
for (auto &desc : IM) {
|
|
if (desc.importOptions.contains(ImportFlags::ImplementationOnly))
|
|
HasImplementationOnlyImports = true;
|
|
}
|
|
}
|
|
}
|
|
|
|
bool SourceFile::hasTestableOrPrivateImport(
|
|
AccessLevel accessLevel, const swift::ValueDecl *ofDecl,
|
|
SourceFile::ImportQueryKind queryKind) const {
|
|
auto *module = ofDecl->getModuleContext();
|
|
switch (accessLevel) {
|
|
case AccessLevel::Internal:
|
|
case AccessLevel::Public:
|
|
// internal/public access only needs an import marked as @_private. The
|
|
// filename does not need to match (and we don't serialize it for such
|
|
// decls).
|
|
return std::any_of(
|
|
Imports.begin(), Imports.end(),
|
|
[module, queryKind](ImportedModuleDesc desc) -> bool {
|
|
if (queryKind == ImportQueryKind::TestableAndPrivate)
|
|
return desc.module.second == module &&
|
|
(desc.importOptions.contains(ImportFlags::PrivateImport) ||
|
|
desc.importOptions.contains(ImportFlags::Testable));
|
|
else if (queryKind == ImportQueryKind::TestableOnly)
|
|
return desc.module.second == module &&
|
|
desc.importOptions.contains(ImportFlags::Testable);
|
|
else {
|
|
assert(queryKind == ImportQueryKind::PrivateOnly);
|
|
return desc.module.second == module &&
|
|
desc.importOptions.contains(ImportFlags::PrivateImport);
|
|
}
|
|
});
|
|
case AccessLevel::Open:
|
|
return true;
|
|
case AccessLevel::FilePrivate:
|
|
case AccessLevel::Private:
|
|
// Fallthrough.
|
|
break;
|
|
}
|
|
|
|
if (queryKind == ImportQueryKind::TestableOnly)
|
|
return false;
|
|
|
|
auto *DC = ofDecl->getDeclContext();
|
|
if (!DC)
|
|
return false;
|
|
auto *scope = DC->getModuleScopeContext();
|
|
if (!scope)
|
|
return false;
|
|
|
|
StringRef filename;
|
|
if (auto *file = dyn_cast<LoadedFile>(scope)) {
|
|
filename = file->getFilenameForPrivateDecl(ofDecl);
|
|
} else
|
|
return false;
|
|
|
|
if (filename.empty())
|
|
return false;
|
|
|
|
return std::any_of(Imports.begin(), Imports.end(),
|
|
[module, filename](ImportedModuleDesc desc) -> bool {
|
|
return desc.module.second == module &&
|
|
desc.importOptions.contains(
|
|
ImportFlags::PrivateImport) &&
|
|
desc.filename == filename;
|
|
});
|
|
}
|
|
|
|
bool SourceFile::isImportedImplementationOnly(const ModuleDecl *module) const {
|
|
// Implementation-only imports are (currently) always source-file-specific,
|
|
// so if we don't have any, we know the search is complete.
|
|
if (!hasImplementationOnlyImports())
|
|
return false;
|
|
|
|
auto isImportedBy = [](const ModuleDecl *dest, const ModuleDecl *src) {
|
|
// Fast path.
|
|
if (dest == src) return true;
|
|
|
|
// Walk the transitive imports, respecting visibility.
|
|
// This return true if the search *didn't* short-circuit, and it short
|
|
// circuits if we found `dest`, so we need to invert the sense before
|
|
// returning.
|
|
return !const_cast<ModuleDecl*>(src)
|
|
->forAllVisibleModules({}, [dest](ModuleDecl::ImportedModule im) {
|
|
// Continue searching as long as we haven't found `dest` yet.
|
|
return im.second != dest;
|
|
});
|
|
};
|
|
|
|
// Look at the imports of this source file.
|
|
for (auto &desc : Imports) {
|
|
// Ignore implementation-only imports.
|
|
if (desc.importOptions.contains(ImportFlags::ImplementationOnly))
|
|
continue;
|
|
|
|
// If the module is imported this way, it's not imported
|
|
// implementation-only.
|
|
if (isImportedBy(module, desc.module.second))
|
|
return false;
|
|
}
|
|
|
|
// Now check this file's enclosing module in case there are re-exports.
|
|
return !isImportedBy(module, getParentModule());
|
|
}
|
|
|
|
void ModuleDecl::clearLookupCache() {
|
|
if (!Cache)
|
|
return;
|
|
|
|
// Abandon any current cache. We'll rebuild it on demand.
|
|
Cache->invalidate();
|
|
Cache.reset();
|
|
}
|
|
|
|
void SourceFile::clearLookupCache() {
|
|
getParentModule()->clearLookupCache();
|
|
|
|
if (!Cache)
|
|
return;
|
|
|
|
// Abandon any current cache. We'll rebuild it on demand.
|
|
Cache->invalidate();
|
|
Cache.reset();
|
|
}
|
|
|
|
void
|
|
SourceFile::cacheVisibleDecls(SmallVectorImpl<ValueDecl*> &&globals) const {
|
|
SmallVectorImpl<ValueDecl*> &cached = getCache().AllVisibleValues;
|
|
cached = std::move(globals);
|
|
}
|
|
|
|
const SmallVectorImpl<ValueDecl *> &
|
|
SourceFile::getCachedVisibleDecls() const {
|
|
return getCache().AllVisibleValues;
|
|
}
|
|
|
|
static void performAutoImport(
|
|
SourceFile &SF,
|
|
SourceFile::ImplicitModuleImportKind implicitModuleImportKind) {
|
|
if (SF.Kind == SourceFileKind::SIL)
|
|
assert(implicitModuleImportKind ==
|
|
SourceFile::ImplicitModuleImportKind::None);
|
|
|
|
ASTContext &Ctx = SF.getASTContext();
|
|
ModuleDecl *M = nullptr;
|
|
|
|
switch (implicitModuleImportKind) {
|
|
case SourceFile::ImplicitModuleImportKind::None:
|
|
return;
|
|
case SourceFile::ImplicitModuleImportKind::Builtin:
|
|
M = Ctx.TheBuiltinModule;
|
|
break;
|
|
case SourceFile::ImplicitModuleImportKind::Stdlib:
|
|
M = Ctx.getStdlibModule(true);
|
|
break;
|
|
}
|
|
|
|
assert(M && "unable to auto-import module");
|
|
|
|
// FIXME: These will be the same for most source files, but we copy them
|
|
// over and over again.
|
|
auto Imports = SourceFile::ImportedModuleDesc(
|
|
ModuleDecl::ImportedModule({}, M), SourceFile::ImportOptions());
|
|
SF.addImports(Imports);
|
|
}
|
|
|
|
SourceFile::SourceFile(ModuleDecl &M, SourceFileKind K,
|
|
Optional<unsigned> bufferID,
|
|
ImplicitModuleImportKind ModImpKind,
|
|
bool KeepParsedTokens, bool BuildSyntaxTree)
|
|
: FileUnit(FileUnitKind::Source, M),
|
|
BufferID(bufferID ? *bufferID : -1),
|
|
Kind(K), SyntaxInfo(new SourceFileSyntaxInfo(BuildSyntaxTree)) {
|
|
M.getASTContext().addDestructorCleanup(*this);
|
|
performAutoImport(*this, ModImpKind);
|
|
|
|
if (isScriptMode()) {
|
|
bool problem = M.registerEntryPointFile(this, SourceLoc(), None);
|
|
assert(!problem && "multiple main files?");
|
|
(void)problem;
|
|
}
|
|
if (KeepParsedTokens) {
|
|
AllCorrectedTokens = std::vector<Token>();
|
|
}
|
|
}
|
|
|
|
std::vector<Token> &SourceFile::getTokenVector() {
|
|
assert(shouldCollectToken() && "Disabled");
|
|
return *AllCorrectedTokens;
|
|
}
|
|
|
|
ArrayRef<Token> SourceFile::getAllTokens() const {
|
|
assert(shouldCollectToken() && "Disabled");
|
|
return *AllCorrectedTokens;
|
|
}
|
|
|
|
bool SourceFile::shouldCollectToken() const {
|
|
switch (Kind) {
|
|
case SourceFileKind::Library:
|
|
case SourceFileKind::Main:
|
|
case SourceFileKind::Interface:
|
|
return (bool)AllCorrectedTokens;
|
|
case SourceFileKind::REPL:
|
|
case SourceFileKind::SIL:
|
|
return false;
|
|
}
|
|
llvm_unreachable("unhandled kind");
|
|
}
|
|
|
|
bool SourceFile::shouldBuildSyntaxTree() const {
|
|
return canBeParsedInFull() && SyntaxInfo->Enable;
|
|
}
|
|
|
|
bool SourceFile::canBeParsedInFull() const {
|
|
switch (Kind) {
|
|
case SourceFileKind::Library:
|
|
case SourceFileKind::Main:
|
|
case SourceFileKind::Interface:
|
|
return true;
|
|
case SourceFileKind::REPL:
|
|
case SourceFileKind::SIL:
|
|
return false;
|
|
}
|
|
llvm_unreachable("unhandled kind");
|
|
}
|
|
|
|
bool FileUnit::walk(ASTWalker &walker) {
|
|
SmallVector<Decl *, 64> Decls;
|
|
getTopLevelDecls(Decls);
|
|
llvm::SaveAndRestore<ASTWalker::ParentTy> SAR(walker.Parent,
|
|
getParentModule());
|
|
for (Decl *D : Decls) {
|
|
#ifndef NDEBUG
|
|
PrettyStackTraceDecl debugStack("walking into decl", D);
|
|
#endif
|
|
|
|
if (D->walk(walker))
|
|
return true;
|
|
|
|
if (walker.shouldWalkAccessorsTheOldWay()) {
|
|
// Pretend that accessors share a parent with the storage.
|
|
//
|
|
// FIXME: Update existing ASTWalkers to deal with accessors appearing as
|
|
// children of the storage instead.
|
|
if (auto *ASD = dyn_cast<AbstractStorageDecl>(D)) {
|
|
for (auto AD : ASD->getAllAccessors()) {
|
|
if (AD->walk(walker))
|
|
return true;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
return false;
|
|
}
|
|
|
|
bool SourceFile::walk(ASTWalker &walker) {
|
|
llvm::SaveAndRestore<ASTWalker::ParentTy> SAR(walker.Parent,
|
|
getParentModule());
|
|
for (Decl *D : Decls) {
|
|
#ifndef NDEBUG
|
|
PrettyStackTraceDecl debugStack("walking into decl", D);
|
|
#endif
|
|
|
|
if (D->walk(walker))
|
|
return true;
|
|
|
|
if (walker.shouldWalkAccessorsTheOldWay()) {
|
|
// Pretend that accessors share a parent with the storage.
|
|
//
|
|
// FIXME: Update existing ASTWalkers to deal with accessors appearing as
|
|
// children of the storage instead.
|
|
if (auto *ASD = dyn_cast<AbstractStorageDecl>(D)) {
|
|
for (auto AD : ASD->getAllAccessors()) {
|
|
if (AD->walk(walker))
|
|
return true;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
return false;
|
|
}
|
|
|
|
StringRef SourceFile::getFilename() const {
|
|
if (BufferID == -1)
|
|
return "";
|
|
SourceManager &SM = getASTContext().SourceMgr;
|
|
return SM.getIdentifierForBuffer(BufferID);
|
|
}
|
|
|
|
ASTScope &SourceFile::getScope() {
|
|
if (!Scope)
|
|
Scope = std::unique_ptr<ASTScope>(new (getASTContext()) ASTScope(this));
|
|
return *Scope.get();
|
|
}
|
|
|
|
|
|
Identifier
|
|
SourceFile::getDiscriminatorForPrivateValue(const ValueDecl *D) const {
|
|
assert(D->getDeclContext()->getModuleScopeContext() == this);
|
|
|
|
if (!PrivateDiscriminator.empty())
|
|
return PrivateDiscriminator;
|
|
|
|
StringRef name = getFilename();
|
|
if (name.empty()) {
|
|
assert(1 == count_if(getParentModule()->getFiles(),
|
|
[](const FileUnit *FU) -> bool {
|
|
return isa<SourceFile>(FU) &&
|
|
cast<SourceFile>(FU)->getFilename().empty();
|
|
}) &&
|
|
"can't promise uniqueness if multiple source files are nameless");
|
|
|
|
// We still need a discriminator, so keep going.
|
|
}
|
|
|
|
// Use a hash of the basename of the source file as our discriminator.
|
|
// This keeps us from leaking information about the original filename
|
|
// while still providing uniqueness. Using the basename makes the
|
|
// discriminator invariant across source checkout locations.
|
|
// FIXME: Use a faster hash here? We don't need security, just uniqueness.
|
|
llvm::MD5 hash;
|
|
hash.update(getParentModule()->getName().str());
|
|
hash.update(llvm::sys::path::filename(name));
|
|
llvm::MD5::MD5Result result;
|
|
hash.final(result);
|
|
|
|
// Use the hash as a hex string, prefixed with an underscore to make sure
|
|
// it is a valid identifier.
|
|
// FIXME: There are more compact ways to encode a 16-byte value.
|
|
SmallString<33> buffer{"_"};
|
|
SmallString<32> hashString;
|
|
llvm::MD5::stringifyResult(result, hashString);
|
|
buffer += hashString;
|
|
PrivateDiscriminator = getASTContext().getIdentifier(buffer.str().upper());
|
|
return PrivateDiscriminator;
|
|
}
|
|
|
|
TypeRefinementContext *SourceFile::getTypeRefinementContext() {
|
|
return TRC;
|
|
}
|
|
|
|
void SourceFile::setTypeRefinementContext(TypeRefinementContext *Root) {
|
|
TRC = Root;
|
|
}
|
|
|
|
void SourceFile::createReferencedNameTracker() {
|
|
assert(!ReferencedNames && "This file already has a name tracker.");
|
|
ReferencedNames.emplace(ReferencedNameTracker());
|
|
}
|
|
|
|
OpaqueTypeDecl *
|
|
SourceFile::lookupOpaqueResultType(StringRef MangledName,
|
|
LazyResolver *resolver) {
|
|
// Check already-validated decls.
|
|
auto found = ValidatedOpaqueReturnTypes.find(MangledName);
|
|
if (found != ValidatedOpaqueReturnTypes.end())
|
|
return found->second;
|
|
|
|
// If there are unvalidated decls with opaque types, go through and validate
|
|
// them now.
|
|
if (resolver && !UnvalidatedDeclsWithOpaqueReturnTypes.empty()) {
|
|
while (!UnvalidatedDeclsWithOpaqueReturnTypes.empty()) {
|
|
ValueDecl *decl = *UnvalidatedDeclsWithOpaqueReturnTypes.begin();
|
|
UnvalidatedDeclsWithOpaqueReturnTypes.erase(decl);
|
|
resolver->resolveDeclSignature(decl);
|
|
}
|
|
|
|
found = ValidatedOpaqueReturnTypes.find(MangledName);
|
|
if (found != ValidatedOpaqueReturnTypes.end())
|
|
return found->second;
|
|
}
|
|
|
|
// Otherwise, we don't have a matching opaque decl.
|
|
return nullptr;
|
|
}
|
|
|
|
void SourceFile::markDeclWithOpaqueResultTypeAsValidated(ValueDecl *vd) {
|
|
UnvalidatedDeclsWithOpaqueReturnTypes.erase(vd);
|
|
if (auto opaqueDecl = vd->getOpaqueResultTypeDecl()) {
|
|
auto inserted = ValidatedOpaqueReturnTypes.insert(
|
|
{opaqueDecl->getOpaqueReturnTypeIdentifier().str(), opaqueDecl});
|
|
if (inserted.second) {
|
|
OpaqueReturnTypes.push_back(opaqueDecl);
|
|
}
|
|
}
|
|
}
|
|
|
|
//===----------------------------------------------------------------------===//
|
|
// Miscellaneous
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
void FileUnit::anchor() {}
|
|
void *FileUnit::operator new(size_t Bytes, ASTContext &C, unsigned Alignment) {
|
|
return C.Allocate(Bytes, Alignment);
|
|
}
|
|
|
|
StringRef LoadedFile::getFilename() const {
|
|
return "";
|
|
}
|
|
|
|
static const clang::Module *
|
|
getClangModule(llvm::PointerUnion<const ModuleDecl *, const void *> Union) {
|
|
return static_cast<const clang::Module *>(Union.get<const void *>());
|
|
}
|
|
|
|
StringRef ModuleEntity::getName() const {
|
|
assert(!Mod.isNull());
|
|
if (auto SwiftMod = Mod.dyn_cast<const ModuleDecl*>())
|
|
return SwiftMod->getName().str();
|
|
return getClangModule(Mod)->Name;
|
|
}
|
|
|
|
std::string ModuleEntity::getFullName() const {
|
|
assert(!Mod.isNull());
|
|
if (auto SwiftMod = Mod.dyn_cast<const ModuleDecl*>())
|
|
return SwiftMod->getName().str();
|
|
return getClangModule(Mod)->getFullModuleName();
|
|
}
|
|
|
|
bool ModuleEntity::isSystemModule() const {
|
|
assert(!Mod.isNull());
|
|
if (auto SwiftMod = Mod.dyn_cast<const ModuleDecl*>())
|
|
return SwiftMod->isSystemModule();
|
|
return getClangModule(Mod)->IsSystem;
|
|
}
|
|
|
|
bool ModuleEntity::isBuiltinModule() const {
|
|
assert(!Mod.isNull());
|
|
if (auto SwiftMod = Mod.dyn_cast<const ModuleDecl*>())
|
|
return SwiftMod->isBuiltinModule();
|
|
return false;
|
|
}
|
|
|
|
const ModuleDecl* ModuleEntity::getAsSwiftModule() const {
|
|
assert(!Mod.isNull());
|
|
if (auto SwiftMod = Mod.dyn_cast<const ModuleDecl*>())
|
|
return SwiftMod;
|
|
return nullptr;
|
|
}
|
|
|
|
const clang::Module* ModuleEntity::getAsClangModule() const {
|
|
assert(!Mod.isNull());
|
|
if (Mod.is<const ModuleDecl*>())
|
|
return nullptr;
|
|
return getClangModule(Mod);
|
|
}
|
|
|
|
// See swift/Basic/Statistic.h for declaration: this enables tracing SourceFiles, is
|
|
// defined here to avoid too much layering violation / circular linkage
|
|
// dependency.
|
|
|
|
struct SourceFileTraceFormatter : public UnifiedStatsReporter::TraceFormatter {
|
|
void traceName(const void *Entity, raw_ostream &OS) const {
|
|
if (!Entity)
|
|
return;
|
|
const SourceFile *SF = static_cast<const SourceFile *>(Entity);
|
|
OS << llvm::sys::path::filename(SF->getFilename());
|
|
}
|
|
void traceLoc(const void *Entity, SourceManager *SM,
|
|
clang::SourceManager *CSM, raw_ostream &OS) const {
|
|
// SourceFiles don't have SourceLocs of their own; they contain them.
|
|
}
|
|
};
|
|
|
|
static SourceFileTraceFormatter TF;
|
|
|
|
template<>
|
|
const UnifiedStatsReporter::TraceFormatter*
|
|
FrontendStatsTracer::getTraceFormatter<const SourceFile *>() {
|
|
return &TF;
|
|
}
|