Allow skipping imports in the module visitor.

Skip imports when we know that we do not need to visit any imports
because we've already deserialized the redecls from a module.

llvm-svn: 237782
This commit is contained in:
Manuel Klimek 2015-05-20 10:29:23 +00:00
parent 38810f430b
commit 9eff8b1426
4 changed files with 124 additions and 46 deletions

View File

@ -32,6 +32,10 @@ class ModuleManager {
/// \brief The chain of AST files. The first entry is the one named by the /// \brief The chain of AST files. The first entry is the one named by the
/// user, the last one is the one that doesn't depend on anything further. /// user, the last one is the one that doesn't depend on anything further.
SmallVector<ModuleFile *, 2> Chain; SmallVector<ModuleFile *, 2> Chain;
// \brief The roots of the dependency DAG of AST files. This is used
// to implement short-circuiting logic when running DFS over the dependencies.
SmallVector<ModuleFile *, 2> Roots;
/// \brief All loaded modules, indexed by name. /// \brief All loaded modules, indexed by name.
llvm::DenseMap<const FileEntry *, ModuleFile *> Modules; llvm::DenseMap<const FileEntry *, ModuleFile *> Modules;
@ -264,25 +268,35 @@ public:
/// manager that is *not* in this set can be skipped. /// manager that is *not* in this set can be skipped.
void visit(bool (*Visitor)(ModuleFile &M, void *UserData), void *UserData, void visit(bool (*Visitor)(ModuleFile &M, void *UserData), void *UserData,
llvm::SmallPtrSetImpl<ModuleFile *> *ModuleFilesHit = nullptr); llvm::SmallPtrSetImpl<ModuleFile *> *ModuleFilesHit = nullptr);
/// \brief Control DFS behavior during preorder visitation.
enum DFSPreorderControl {
Continue, /// Continue visiting all nodes.
Abort, /// Stop the visitation immediately.
SkipImports, /// Do not visit imports of the current node.
};
/// \brief Visit each of the modules with a depth-first traversal. /// \brief Visit each of the modules with a depth-first traversal.
/// ///
/// This routine visits each of the modules known to the module /// This routine visits each of the modules known to the module
/// manager using a depth-first search, starting with the first /// manager using a depth-first search, starting with the first
/// loaded module. The traversal invokes the callback both before /// loaded module. The traversal invokes one callback before
/// traversing the children (preorder traversal) and after /// traversing the imports (preorder traversal) and one after
/// traversing the children (postorder traversal). /// traversing the imports (postorder traversal).
/// ///
/// \param Visitor A visitor function that will be invoked with each /// \param PreorderVisitor A visitor function that will be invoked with each
/// module and given a \c Preorder flag that indicates whether we're /// module before visiting its imports. The visitor can control how to
/// visiting the module before or after visiting its children. The /// continue the visitation through its return value.
/// visitor may return true at any time to abort the depth-first ///
/// visitation. /// \param PostorderVisitor A visitor function taht will be invoked with each
/// module after visiting its imports. The visitor may return true at any time
/// to abort the depth-first visitation.
/// ///
/// \param UserData User data ssociated with the visitor object, /// \param UserData User data ssociated with the visitor object,
/// which will be passed along to the user. /// which will be passed along to the user.
void visitDepthFirst(bool (*Visitor)(ModuleFile &M, bool Preorder, void visitDepthFirst(DFSPreorderControl (*PreorderVisitor)(ModuleFile &M,
void *UserData), void *UserData),
bool (*PostorderVisitor)(ModuleFile &M, void *UserData),
void *UserData); void *UserData);
/// \brief Attempt to resolve the given module file name to a file entry. /// \brief Attempt to resolve the given module file name to a file entry.

View File

@ -6157,10 +6157,7 @@ namespace {
PredefsVisited[I] = false; PredefsVisited[I] = false;
} }
static bool visit(ModuleFile &M, bool Preorder, void *UserData) { static bool visitPostorder(ModuleFile &M, void *UserData) {
if (Preorder)
return false;
FindExternalLexicalDeclsVisitor *This FindExternalLexicalDeclsVisitor *This
= static_cast<FindExternalLexicalDeclsVisitor *>(UserData); = static_cast<FindExternalLexicalDeclsVisitor *>(UserData);
@ -6202,7 +6199,8 @@ ExternalLoadResult ASTReader::FindExternalLexicalDecls(const DeclContext *DC,
// There might be lexical decls in multiple modules, for the TU at // There might be lexical decls in multiple modules, for the TU at
// least. Walk all of the modules in the order they were loaded. // least. Walk all of the modules in the order they were loaded.
FindExternalLexicalDeclsVisitor Visitor(*this, DC, isKindWeWant, Decls); FindExternalLexicalDeclsVisitor Visitor(*this, DC, isKindWeWant, Decls);
ModuleMgr.visitDepthFirst(&FindExternalLexicalDeclsVisitor::visit, &Visitor); ModuleMgr.visitDepthFirst(
nullptr, &FindExternalLexicalDeclsVisitor::visitPostorder, &Visitor);
++NumLexicalDeclContextsRead; ++NumLexicalDeclContextsRead;
return ELR_Success; return ELR_Success;
} }

View File

@ -3311,11 +3311,13 @@ namespace {
addToChain(Reader.GetDecl(CanonID)); addToChain(Reader.GetDecl(CanonID));
} }
static bool visit(ModuleFile &M, bool Preorder, void *UserData) { static ModuleManager::DFSPreorderControl
if (Preorder) visitPreorder(ModuleFile &M, void *UserData) {
return false; return static_cast<RedeclChainVisitor *>(UserData)->visitPreorder(M);
}
return static_cast<RedeclChainVisitor *>(UserData)->visit(M); static bool visitPostorder(ModuleFile &M, void *UserData) {
return static_cast<RedeclChainVisitor *>(UserData)->visitPostorder(M);
} }
void addToChain(Decl *D) { void addToChain(Decl *D) {
@ -3368,8 +3370,36 @@ namespace {
for (unsigned I = 0; I != N; ++I) for (unsigned I = 0; I != N; ++I)
addToChain(Reader.GetLocalDecl(M, M.RedeclarationChains[Offset++])); addToChain(Reader.GetLocalDecl(M, M.RedeclarationChains[Offset++]));
} }
bool visit(ModuleFile &M) { bool needsToVisitImports(ModuleFile &M, GlobalDeclID GlobalID) {
DeclID ID = Reader.mapGlobalIDToModuleFileGlobalID(M, GlobalID);
if (!ID)
return false;
const LocalRedeclarationsInfo Compare = {ID, 0};
const LocalRedeclarationsInfo *Result = std::lower_bound(
M.RedeclarationsMap,
M.RedeclarationsMap + M.LocalNumRedeclarationsInMap, Compare);
if (Result == M.RedeclarationsMap + M.LocalNumRedeclarationsInMap ||
Result->FirstID != ID) {
return true;
}
unsigned Offset = Result->Offset;
unsigned N = M.RedeclarationChains[Offset];
// We don't need to visit a module or any of its imports if we've already
// deserialized the redecls from this module.
return N != 0;
}
ModuleManager::DFSPreorderControl visitPreorder(ModuleFile &M) {
for (unsigned I = 0, N = SearchDecls.size(); I != N; ++I) {
if (needsToVisitImports(M, SearchDecls[I]))
return ModuleManager::Continue;
}
return ModuleManager::SkipImports;
}
bool visitPostorder(ModuleFile &M) {
// Visit each of the declarations. // Visit each of the declarations.
for (unsigned I = 0, N = SearchDecls.size(); I != N; ++I) for (unsigned I = 0, N = SearchDecls.size(); I != N; ++I)
searchForID(M, SearchDecls[I]); searchForID(M, SearchDecls[I]);
@ -3401,11 +3431,12 @@ void ASTReader::loadPendingDeclChain(Decl *CanonDecl) {
// Build up the list of redeclarations. // Build up the list of redeclarations.
RedeclChainVisitor Visitor(*this, SearchDecls, RedeclsDeserialized, CanonID); RedeclChainVisitor Visitor(*this, SearchDecls, RedeclsDeserialized, CanonID);
ModuleMgr.visitDepthFirst(&RedeclChainVisitor::visit, &Visitor); ModuleMgr.visitDepthFirst(&RedeclChainVisitor::visitPreorder,
&RedeclChainVisitor::visitPostorder, &Visitor);
// Retrieve the chains. // Retrieve the chains.
ArrayRef<Decl *> Chain = Visitor.getChain(); ArrayRef<Decl *> Chain = Visitor.getChain();
if (Chain.empty()) if (Chain.empty() || (Chain.size() == 1 && Chain[0] == CanonDecl))
return; return;
// Hook up the chains. // Hook up the chains.

View File

@ -94,6 +94,8 @@ ModuleManager::addModule(StringRef FileName, ModuleKind Type,
New->File = Entry; New->File = Entry;
New->ImportLoc = ImportLoc; New->ImportLoc = ImportLoc;
Chain.push_back(New); Chain.push_back(New);
if (!ImportedBy)
Roots.push_back(New);
NewModule = true; NewModule = true;
ModuleEntry = New; ModuleEntry = New;
@ -155,7 +157,12 @@ ModuleManager::addModule(StringRef FileName, ModuleKind Type,
// invalidate the file cache for Entry, and that is not safe if this // invalidate the file cache for Entry, and that is not safe if this
// module is *itself* up to date, but has an out-of-date importer. // module is *itself* up to date, but has an out-of-date importer.
Modules.erase(Entry); Modules.erase(Entry);
assert(Chain.back() == ModuleEntry);
Chain.pop_back(); Chain.pop_back();
if (Roots.back() == ModuleEntry)
Roots.pop_back();
else
assert(ImportedBy);
delete ModuleEntry; delete ModuleEntry;
} }
return OutOfDate; return OutOfDate;
@ -186,12 +193,15 @@ void ModuleManager::removeModules(
// Collect the set of module file pointers that we'll be removing. // Collect the set of module file pointers that we'll be removing.
llvm::SmallPtrSet<ModuleFile *, 4> victimSet(first, last); llvm::SmallPtrSet<ModuleFile *, 4> victimSet(first, last);
auto IsVictim = [&](ModuleFile *MF) {
return victimSet.count(MF);
};
// Remove any references to the now-destroyed modules. // Remove any references to the now-destroyed modules.
for (unsigned i = 0, n = Chain.size(); i != n; ++i) { for (unsigned i = 0, n = Chain.size(); i != n; ++i) {
Chain[i]->ImportedBy.remove_if([&](ModuleFile *MF) { Chain[i]->ImportedBy.remove_if(IsVictim);
return victimSet.count(MF);
});
} }
Roots.erase(std::remove_if(Roots.begin(), Roots.end(), IsVictim),
Roots.end());
// Delete the modules and erase them from the various structures. // Delete the modules and erase them from the various structures.
for (ModuleIterator victim = first; victim != last; ++victim) { for (ModuleIterator victim = first; victim != last; ++victim) {
@ -398,16 +408,38 @@ ModuleManager::visit(bool (*Visitor)(ModuleFile &M, void *UserData),
returnVisitState(State); returnVisitState(State);
} }
static void markVisitedDepthFirst(ModuleFile &M,
SmallVectorImpl<bool> &Visited) {
for (llvm::SetVector<ModuleFile *>::iterator IM = M.Imports.begin(),
IMEnd = M.Imports.end();
IM != IMEnd; ++IM) {
if (Visited[(*IM)->Index])
continue;
Visited[(*IM)->Index] = true;
if (!M.DirectlyImported)
markVisitedDepthFirst(**IM, Visited);
}
}
/// \brief Perform a depth-first visit of the current module. /// \brief Perform a depth-first visit of the current module.
static bool visitDepthFirst(ModuleFile &M, static bool visitDepthFirst(
bool (*Visitor)(ModuleFile &M, bool Preorder, ModuleFile &M,
void *UserData), ModuleManager::DFSPreorderControl (*PreorderVisitor)(ModuleFile &M,
void *UserData, void *UserData),
SmallVectorImpl<bool> &Visited) { bool (*PostorderVisitor)(ModuleFile &M, void *UserData), void *UserData,
// Preorder visitation SmallVectorImpl<bool> &Visited) {
if (Visitor(M, /*Preorder=*/true, UserData)) if (PreorderVisitor) {
return true; switch (PreorderVisitor(M, UserData)) {
case ModuleManager::Abort:
return true;
case ModuleManager::SkipImports:
markVisitedDepthFirst(M, Visited);
return false;
case ModuleManager::Continue:
break;
}
}
// Visit children // Visit children
for (llvm::SetVector<ModuleFile *>::iterator IM = M.Imports.begin(), for (llvm::SetVector<ModuleFile *>::iterator IM = M.Imports.begin(),
IMEnd = M.Imports.end(); IMEnd = M.Imports.end();
@ -416,24 +448,27 @@ static bool visitDepthFirst(ModuleFile &M,
continue; continue;
Visited[(*IM)->Index] = true; Visited[(*IM)->Index] = true;
if (visitDepthFirst(**IM, Visitor, UserData, Visited)) if (visitDepthFirst(**IM, PreorderVisitor, PostorderVisitor, UserData, Visited))
return true; return true;
} }
// Postorder visitation if (PostorderVisitor)
return Visitor(M, /*Preorder=*/false, UserData); return PostorderVisitor(M, UserData);
return false;
} }
void ModuleManager::visitDepthFirst(bool (*Visitor)(ModuleFile &M, bool Preorder, void ModuleManager::visitDepthFirst(
void *UserData), ModuleManager::DFSPreorderControl (*PreorderVisitor)(ModuleFile &M,
void *UserData) { void *UserData),
bool (*PostorderVisitor)(ModuleFile &M, void *UserData), void *UserData) {
SmallVector<bool, 16> Visited(size(), false); SmallVector<bool, 16> Visited(size(), false);
for (unsigned I = 0, N = Chain.size(); I != N; ++I) { for (unsigned I = 0, N = Roots.size(); I != N; ++I) {
if (Visited[Chain[I]->Index]) if (Visited[Roots[I]->Index])
continue; continue;
Visited[Chain[I]->Index] = true; Visited[Roots[I]->Index] = true;
if (::visitDepthFirst(*Chain[I], Visitor, UserData, Visited)) if (::visitDepthFirst(*Roots[I], PreorderVisitor, PostorderVisitor, UserData, Visited))
return; return;
} }
} }