forked from OSchip/llvm-project
435 lines
17 KiB
C++
435 lines
17 KiB
C++
//===- ExternalASTMerger.cpp - Merging External AST Interface ---*- C++ -*-===//
|
|
//
|
|
// Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions.
|
|
// See https://llvm.org/LICENSE.txt for license information.
|
|
// SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception
|
|
//
|
|
//===----------------------------------------------------------------------===//
|
|
//
|
|
// This file implements the ExternalASTMerger, which vends a combination of
|
|
// ASTs from several different ASTContext/FileManager pairs
|
|
//
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
#include "clang/AST/ASTContext.h"
|
|
#include "clang/AST/Decl.h"
|
|
#include "clang/AST/DeclCXX.h"
|
|
#include "clang/AST/DeclObjC.h"
|
|
#include "clang/AST/DeclTemplate.h"
|
|
#include "clang/AST/ExternalASTMerger.h"
|
|
|
|
using namespace clang;
|
|
|
|
namespace {
|
|
|
|
template <typename T> struct Source {
|
|
T t;
|
|
Source(T t) : t(t) {}
|
|
operator T() { return t; }
|
|
template <typename U = T> U &get() { return t; }
|
|
template <typename U = T> const U &get() const { return t; }
|
|
template <typename U> operator Source<U>() { return Source<U>(t); }
|
|
};
|
|
|
|
typedef std::pair<Source<NamedDecl *>, ASTImporter *> Candidate;
|
|
|
|
/// For the given DC, return the DC that is safe to perform lookups on. This is
|
|
/// the DC we actually want to work with most of the time.
|
|
const DeclContext *CanonicalizeDC(const DeclContext *DC) {
|
|
if (isa<LinkageSpecDecl>(DC))
|
|
return DC->getRedeclContext();
|
|
return DC;
|
|
}
|
|
|
|
Source<const DeclContext *>
|
|
LookupSameContext(Source<TranslationUnitDecl *> SourceTU, const DeclContext *DC,
|
|
ASTImporter &ReverseImporter) {
|
|
DC = CanonicalizeDC(DC);
|
|
if (DC->isTranslationUnit()) {
|
|
return SourceTU;
|
|
}
|
|
Source<const DeclContext *> SourceParentDC =
|
|
LookupSameContext(SourceTU, DC->getParent(), ReverseImporter);
|
|
if (!SourceParentDC) {
|
|
// If we couldn't find the parent DC in this TranslationUnit, give up.
|
|
return nullptr;
|
|
}
|
|
auto *ND = cast<NamedDecl>(DC);
|
|
DeclarationName Name = ND->getDeclName();
|
|
Source<DeclarationName> SourceName = ReverseImporter.Import(Name);
|
|
DeclContext::lookup_result SearchResult =
|
|
SourceParentDC.get()->lookup(SourceName.get());
|
|
size_t SearchResultSize = SearchResult.size();
|
|
if (SearchResultSize == 0 || SearchResultSize > 1) {
|
|
// There are two cases here. First, we might not find the name.
|
|
// We might also find multiple copies, in which case we have no
|
|
// guarantee that the one we wanted is the one we pick. (E.g.,
|
|
// if we have two specializations of the same template it is
|
|
// very hard to determine which is the one you want.)
|
|
//
|
|
// The Origins map fixes this problem by allowing the origin to be
|
|
// explicitly recorded, so we trigger that recording by returning
|
|
// nothing (rather than a possibly-inaccurate guess) here.
|
|
return nullptr;
|
|
} else {
|
|
NamedDecl *SearchResultDecl = SearchResult[0];
|
|
if (isa<DeclContext>(SearchResultDecl) &&
|
|
SearchResultDecl->getKind() == DC->getDeclKind())
|
|
return cast<DeclContext>(SearchResultDecl)->getPrimaryContext();
|
|
return nullptr; // This type of lookup is unsupported
|
|
}
|
|
}
|
|
|
|
/// A custom implementation of ASTImporter, for ExternalASTMerger's purposes.
|
|
///
|
|
/// There are several modifications:
|
|
///
|
|
/// - It enables lazy lookup (via the HasExternalLexicalStorage flag and a few
|
|
/// others), which instructs Clang to refer to ExternalASTMerger. Also, it
|
|
/// forces MinimalImport to true, which is necessary to make this work.
|
|
/// - It maintains a reverse importer for use with names. This allows lookup of
|
|
/// arbitrary names in the source context.
|
|
/// - It updates the ExternalASTMerger's origin map as needed whenever a
|
|
/// it sees a DeclContext.
|
|
class LazyASTImporter : public ASTImporter {
|
|
private:
|
|
ExternalASTMerger &Parent;
|
|
ASTImporter Reverse;
|
|
const ExternalASTMerger::OriginMap &FromOrigins;
|
|
|
|
llvm::raw_ostream &logs() { return Parent.logs(); }
|
|
public:
|
|
LazyASTImporter(ExternalASTMerger &_Parent, ASTContext &ToContext,
|
|
FileManager &ToFileManager, ASTContext &FromContext,
|
|
FileManager &FromFileManager,
|
|
const ExternalASTMerger::OriginMap &_FromOrigins)
|
|
: ASTImporter(ToContext, ToFileManager, FromContext, FromFileManager,
|
|
/*MinimalImport=*/true),
|
|
Parent(_Parent), Reverse(FromContext, FromFileManager, ToContext,
|
|
ToFileManager, /*MinimalImport=*/true), FromOrigins(_FromOrigins) {}
|
|
|
|
/// Whenever a DeclContext is imported, ensure that ExternalASTSource's origin
|
|
/// map is kept up to date. Also set the appropriate flags.
|
|
Decl *Imported(Decl *From, Decl *To) override {
|
|
if (auto *ToDC = dyn_cast<DeclContext>(To)) {
|
|
const bool LoggingEnabled = Parent.LoggingEnabled();
|
|
if (LoggingEnabled)
|
|
logs() << "(ExternalASTMerger*)" << (void*)&Parent
|
|
<< " imported (DeclContext*)" << (void*)ToDC
|
|
<< ", (ASTContext*)" << (void*)&getToContext()
|
|
<< " from (DeclContext*)" << (void*)llvm::cast<DeclContext>(From)
|
|
<< ", (ASTContext*)" << (void*)&getFromContext()
|
|
<< "\n";
|
|
Source<DeclContext *> FromDC(
|
|
cast<DeclContext>(From)->getPrimaryContext());
|
|
if (FromOrigins.count(FromDC) &&
|
|
Parent.HasImporterForOrigin(*FromOrigins.at(FromDC).AST)) {
|
|
if (LoggingEnabled)
|
|
logs() << "(ExternalASTMerger*)" << (void*)&Parent
|
|
<< " forced origin (DeclContext*)"
|
|
<< (void*)FromOrigins.at(FromDC).DC
|
|
<< ", (ASTContext*)"
|
|
<< (void*)FromOrigins.at(FromDC).AST
|
|
<< "\n";
|
|
Parent.ForceRecordOrigin(ToDC, FromOrigins.at(FromDC));
|
|
} else {
|
|
if (LoggingEnabled)
|
|
logs() << "(ExternalASTMerger*)" << (void*)&Parent
|
|
<< " maybe recording origin (DeclContext*)" << (void*)FromDC
|
|
<< ", (ASTContext*)" << (void*)&getFromContext()
|
|
<< "\n";
|
|
Parent.MaybeRecordOrigin(ToDC, {FromDC, &getFromContext()});
|
|
}
|
|
}
|
|
if (auto *ToTag = dyn_cast<TagDecl>(To)) {
|
|
ToTag->setHasExternalLexicalStorage();
|
|
ToTag->getPrimaryContext()->setMustBuildLookupTable();
|
|
assert(Parent.CanComplete(ToTag));
|
|
} else if (auto *ToNamespace = dyn_cast<NamespaceDecl>(To)) {
|
|
ToNamespace->setHasExternalVisibleStorage();
|
|
assert(Parent.CanComplete(ToNamespace));
|
|
} else if (auto *ToContainer = dyn_cast<ObjCContainerDecl>(To)) {
|
|
ToContainer->setHasExternalLexicalStorage();
|
|
ToContainer->getPrimaryContext()->setMustBuildLookupTable();
|
|
assert(Parent.CanComplete(ToContainer));
|
|
}
|
|
return To;
|
|
}
|
|
ASTImporter &GetReverse() { return Reverse; }
|
|
};
|
|
|
|
bool HasDeclOfSameType(llvm::ArrayRef<Candidate> Decls, const Candidate &C) {
|
|
if (isa<FunctionDecl>(C.first.get()))
|
|
return false;
|
|
return llvm::any_of(Decls, [&](const Candidate &D) {
|
|
return C.first.get()->getKind() == D.first.get()->getKind();
|
|
});
|
|
}
|
|
|
|
} // end namespace
|
|
|
|
ASTImporter &ExternalASTMerger::ImporterForOrigin(ASTContext &OriginContext) {
|
|
for (const std::unique_ptr<ASTImporter> &I : Importers)
|
|
if (&I->getFromContext() == &OriginContext)
|
|
return *I;
|
|
llvm_unreachable("We should have an importer for this origin!");
|
|
}
|
|
|
|
namespace {
|
|
LazyASTImporter &LazyImporterForOrigin(ExternalASTMerger &Merger,
|
|
ASTContext &OriginContext) {
|
|
return static_cast<LazyASTImporter &>(
|
|
Merger.ImporterForOrigin(OriginContext));
|
|
}
|
|
}
|
|
|
|
bool ExternalASTMerger::HasImporterForOrigin(ASTContext &OriginContext) {
|
|
for (const std::unique_ptr<ASTImporter> &I : Importers)
|
|
if (&I->getFromContext() == &OriginContext)
|
|
return true;
|
|
return false;
|
|
}
|
|
|
|
template <typename CallbackType>
|
|
void ExternalASTMerger::ForEachMatchingDC(const DeclContext *DC,
|
|
CallbackType Callback) {
|
|
if (Origins.count(DC)) {
|
|
ExternalASTMerger::DCOrigin Origin = Origins[DC];
|
|
LazyASTImporter &Importer = LazyImporterForOrigin(*this, *Origin.AST);
|
|
Callback(Importer, Importer.GetReverse(), Origin.DC);
|
|
} else {
|
|
bool DidCallback = false;
|
|
for (const std::unique_ptr<ASTImporter> &Importer : Importers) {
|
|
Source<TranslationUnitDecl *> SourceTU =
|
|
Importer->getFromContext().getTranslationUnitDecl();
|
|
ASTImporter &Reverse =
|
|
static_cast<LazyASTImporter *>(Importer.get())->GetReverse();
|
|
if (auto SourceDC = LookupSameContext(SourceTU, DC, Reverse)) {
|
|
DidCallback = true;
|
|
if (Callback(*Importer, Reverse, SourceDC))
|
|
break;
|
|
}
|
|
}
|
|
if (!DidCallback && LoggingEnabled())
|
|
logs() << "(ExternalASTMerger*)" << (void*)this
|
|
<< " asserting for (DeclContext*)" << (const void*)DC
|
|
<< ", (ASTContext*)" << (void*)&Target.AST
|
|
<< "\n";
|
|
assert(DidCallback && "Couldn't find a source context matching our DC");
|
|
}
|
|
}
|
|
|
|
void ExternalASTMerger::CompleteType(TagDecl *Tag) {
|
|
assert(Tag->hasExternalLexicalStorage());
|
|
ForEachMatchingDC(Tag, [&](ASTImporter &Forward, ASTImporter &Reverse,
|
|
Source<const DeclContext *> SourceDC) -> bool {
|
|
auto *SourceTag = const_cast<TagDecl *>(cast<TagDecl>(SourceDC.get()));
|
|
if (SourceTag->hasExternalLexicalStorage())
|
|
SourceTag->getASTContext().getExternalSource()->CompleteType(SourceTag);
|
|
if (!SourceTag->getDefinition())
|
|
return false;
|
|
Forward.MapImported(SourceTag, Tag);
|
|
if (llvm::Error Err = Forward.ImportDefinition_New(SourceTag))
|
|
llvm::consumeError(std::move(Err));
|
|
Tag->setCompleteDefinition(SourceTag->isCompleteDefinition());
|
|
return true;
|
|
});
|
|
}
|
|
|
|
void ExternalASTMerger::CompleteType(ObjCInterfaceDecl *Interface) {
|
|
assert(Interface->hasExternalLexicalStorage());
|
|
ForEachMatchingDC(
|
|
Interface, [&](ASTImporter &Forward, ASTImporter &Reverse,
|
|
Source<const DeclContext *> SourceDC) -> bool {
|
|
auto *SourceInterface = const_cast<ObjCInterfaceDecl *>(
|
|
cast<ObjCInterfaceDecl>(SourceDC.get()));
|
|
if (SourceInterface->hasExternalLexicalStorage())
|
|
SourceInterface->getASTContext().getExternalSource()->CompleteType(
|
|
SourceInterface);
|
|
if (!SourceInterface->getDefinition())
|
|
return false;
|
|
Forward.MapImported(SourceInterface, Interface);
|
|
if (llvm::Error Err = Forward.ImportDefinition_New(SourceInterface))
|
|
llvm::consumeError(std::move(Err));
|
|
return true;
|
|
});
|
|
}
|
|
|
|
bool ExternalASTMerger::CanComplete(DeclContext *Interface) {
|
|
assert(Interface->hasExternalLexicalStorage() ||
|
|
Interface->hasExternalVisibleStorage());
|
|
bool FoundMatchingDC = false;
|
|
ForEachMatchingDC(Interface,
|
|
[&](ASTImporter &Forward, ASTImporter &Reverse,
|
|
Source<const DeclContext *> SourceDC) -> bool {
|
|
FoundMatchingDC = true;
|
|
return true;
|
|
});
|
|
return FoundMatchingDC;
|
|
}
|
|
|
|
namespace {
|
|
bool IsSameDC(const DeclContext *D1, const DeclContext *D2) {
|
|
if (isa<ObjCContainerDecl>(D1) && isa<ObjCContainerDecl>(D2))
|
|
return true; // There are many cases where Objective-C is ambiguous.
|
|
if (auto *T1 = dyn_cast<TagDecl>(D1))
|
|
if (auto *T2 = dyn_cast<TagDecl>(D2))
|
|
if (T1->getFirstDecl() == T2->getFirstDecl())
|
|
return true;
|
|
return D1 == D2 || D1 == CanonicalizeDC(D2);
|
|
}
|
|
}
|
|
|
|
void ExternalASTMerger::MaybeRecordOrigin(const DeclContext *ToDC,
|
|
DCOrigin Origin) {
|
|
LazyASTImporter &Importer = LazyImporterForOrigin(*this, *Origin.AST);
|
|
ASTImporter &Reverse = Importer.GetReverse();
|
|
Source<const DeclContext *> FoundFromDC =
|
|
LookupSameContext(Origin.AST->getTranslationUnitDecl(), ToDC, Reverse);
|
|
const bool DoRecord = !FoundFromDC || !IsSameDC(FoundFromDC.get(), Origin.DC);
|
|
if (DoRecord)
|
|
RecordOriginImpl(ToDC, Origin, Importer);
|
|
if (LoggingEnabled())
|
|
logs() << "(ExternalASTMerger*)" << (void*)this
|
|
<< (DoRecord ? " decided " : " decided NOT")
|
|
<< " to record origin (DeclContext*)" << (void*)Origin.DC
|
|
<< ", (ASTContext*)" << (void*)&Origin.AST
|
|
<< "\n";
|
|
}
|
|
|
|
void ExternalASTMerger::ForceRecordOrigin(const DeclContext *ToDC,
|
|
DCOrigin Origin) {
|
|
RecordOriginImpl(ToDC, Origin, ImporterForOrigin(*Origin.AST));
|
|
}
|
|
|
|
void ExternalASTMerger::RecordOriginImpl(const DeclContext *ToDC, DCOrigin Origin,
|
|
ASTImporter &Importer) {
|
|
Origins[ToDC] = Origin;
|
|
Importer.ASTImporter::MapImported(cast<Decl>(Origin.DC), const_cast<Decl*>(cast<Decl>(ToDC)));
|
|
}
|
|
|
|
ExternalASTMerger::ExternalASTMerger(const ImporterTarget &Target,
|
|
llvm::ArrayRef<ImporterSource> Sources) : LogStream(&llvm::nulls()), Target(Target) {
|
|
AddSources(Sources);
|
|
}
|
|
|
|
void ExternalASTMerger::AddSources(llvm::ArrayRef<ImporterSource> Sources) {
|
|
for (const ImporterSource &S : Sources) {
|
|
assert(&S.AST != &Target.AST);
|
|
Importers.push_back(llvm::make_unique<LazyASTImporter>(
|
|
*this, Target.AST, Target.FM, S.AST, S.FM, S.OM));
|
|
}
|
|
}
|
|
|
|
void ExternalASTMerger::RemoveSources(llvm::ArrayRef<ImporterSource> Sources) {
|
|
if (LoggingEnabled())
|
|
for (const ImporterSource &S : Sources)
|
|
logs() << "(ExternalASTMerger*)" << (void*)this
|
|
<< " removing source (ASTContext*)" << (void*)&S.AST
|
|
<< "\n";
|
|
Importers.erase(
|
|
std::remove_if(Importers.begin(), Importers.end(),
|
|
[&Sources](std::unique_ptr<ASTImporter> &Importer) -> bool {
|
|
for (const ImporterSource &S : Sources) {
|
|
if (&Importer->getFromContext() == &S.AST)
|
|
return true;
|
|
}
|
|
return false;
|
|
}),
|
|
Importers.end());
|
|
for (OriginMap::iterator OI = Origins.begin(), OE = Origins.end(); OI != OE; ) {
|
|
std::pair<const DeclContext *, DCOrigin> Origin = *OI;
|
|
bool Erase = false;
|
|
for (const ImporterSource &S : Sources) {
|
|
if (&S.AST == Origin.second.AST) {
|
|
Erase = true;
|
|
break;
|
|
}
|
|
}
|
|
if (Erase)
|
|
OI = Origins.erase(OI);
|
|
else
|
|
++OI;
|
|
}
|
|
}
|
|
|
|
template <typename DeclTy>
|
|
static bool importSpecializations(DeclTy *D, ASTImporter *Importer) {
|
|
for (auto *Spec : D->specializations())
|
|
if (!Importer->Import(Spec))
|
|
return true;
|
|
return false;
|
|
}
|
|
|
|
/// Imports specializations from template declarations that can be specialized.
|
|
static bool importSpecializationsIfNeeded(Decl *D, ASTImporter *Importer) {
|
|
if (!isa<TemplateDecl>(D))
|
|
return false;
|
|
if (auto *FunctionTD = dyn_cast<FunctionTemplateDecl>(D))
|
|
return importSpecializations(FunctionTD, Importer);
|
|
else if (auto *ClassTD = dyn_cast<ClassTemplateDecl>(D))
|
|
return importSpecializations(ClassTD, Importer);
|
|
else if (auto *VarTD = dyn_cast<VarTemplateDecl>(D))
|
|
return importSpecializations(VarTD, Importer);
|
|
return false;
|
|
}
|
|
|
|
bool ExternalASTMerger::FindExternalVisibleDeclsByName(const DeclContext *DC,
|
|
DeclarationName Name) {
|
|
llvm::SmallVector<NamedDecl *, 1> Decls;
|
|
llvm::SmallVector<Candidate, 4> Candidates;
|
|
|
|
auto FilterFoundDecl = [&Candidates](const Candidate &C) {
|
|
if (!HasDeclOfSameType(Candidates, C))
|
|
Candidates.push_back(C);
|
|
};
|
|
|
|
ForEachMatchingDC(DC, [&](ASTImporter &Forward, ASTImporter &Reverse,
|
|
Source<const DeclContext *> SourceDC) -> bool {
|
|
DeclarationName FromName = Reverse.Import(Name);
|
|
DeclContextLookupResult Result = SourceDC.get()->lookup(FromName);
|
|
for (NamedDecl *FromD : Result) {
|
|
FilterFoundDecl(std::make_pair(FromD, &Forward));
|
|
}
|
|
return false;
|
|
});
|
|
|
|
if (Candidates.empty())
|
|
return false;
|
|
|
|
Decls.reserve(Candidates.size());
|
|
for (const Candidate &C : Candidates) {
|
|
Decl *LookupRes = C.first.get();
|
|
ASTImporter *Importer = C.second;
|
|
NamedDecl *ND = cast_or_null<NamedDecl>(Importer->Import(LookupRes));
|
|
assert(ND);
|
|
// If we don't import specialization, they are not available via lookup
|
|
// because the lookup result is imported TemplateDecl and it does not
|
|
// reference its specializations until they are imported explicitly.
|
|
bool IsSpecImportFailed =
|
|
importSpecializationsIfNeeded(LookupRes, Importer);
|
|
assert(!IsSpecImportFailed);
|
|
(void)IsSpecImportFailed;
|
|
Decls.push_back(ND);
|
|
}
|
|
SetExternalVisibleDeclsForName(DC, Name, Decls);
|
|
return true;
|
|
}
|
|
|
|
void ExternalASTMerger::FindExternalLexicalDecls(
|
|
const DeclContext *DC, llvm::function_ref<bool(Decl::Kind)> IsKindWeWant,
|
|
SmallVectorImpl<Decl *> &Result) {
|
|
ForEachMatchingDC(DC, [&](ASTImporter &Forward, ASTImporter &Reverse,
|
|
Source<const DeclContext *> SourceDC) -> bool {
|
|
for (const Decl *SourceDecl : SourceDC.get()->decls()) {
|
|
if (IsKindWeWant(SourceDecl->getKind())) {
|
|
Decl *ImportedDecl = Forward.Import(const_cast<Decl *>(SourceDecl));
|
|
assert(!ImportedDecl || IsSameDC(ImportedDecl->getDeclContext(), DC));
|
|
(void)ImportedDecl;
|
|
}
|
|
}
|
|
return false;
|
|
});
|
|
}
|
|
|