//===- CGSCCPassManager.cpp - Managing & running CGSCC passes -------------===// // // The LLVM Compiler Infrastructure // // This file is distributed under the University of Illinois Open Source // License. See LICENSE.TXT for details. // //===----------------------------------------------------------------------===// #include "llvm/Analysis/CGSCCPassManager.h" #include "llvm/Support/CommandLine.h" #include "llvm/Support/Debug.h" using namespace llvm; static cl::opt DebugPM("debug-cgscc-pass-manager", cl::Hidden, cl::desc("Print CGSCC pass management debugging information")); PreservedAnalyses CGSCCPassManager::run(LazyCallGraph::SCC *C, CGSCCAnalysisManager *AM) { PreservedAnalyses PA = PreservedAnalyses::all(); if (DebugPM) dbgs() << "Starting CGSCC pass manager run.\n"; for (unsigned Idx = 0, Size = Passes.size(); Idx != Size; ++Idx) { if (DebugPM) dbgs() << "Running CGSCC pass: " << Passes[Idx]->name() << "\n"; PreservedAnalyses PassPA = Passes[Idx]->run(C, AM); if (AM) AM->invalidate(C, PassPA); PA.intersect(std::move(PassPA)); } if (DebugPM) dbgs() << "Finished CGSCC pass manager run.\n"; return PA; } bool CGSCCAnalysisManager::empty() const { assert(CGSCCAnalysisResults.empty() == CGSCCAnalysisResultLists.empty() && "The storage and index of analysis results disagree on how many there " "are!"); return CGSCCAnalysisResults.empty(); } void CGSCCAnalysisManager::clear() { CGSCCAnalysisResults.clear(); CGSCCAnalysisResultLists.clear(); } CGSCCAnalysisManager::ResultConceptT & CGSCCAnalysisManager::getResultImpl(void *PassID, LazyCallGraph::SCC *C) { CGSCCAnalysisResultMapT::iterator RI; bool Inserted; std::tie(RI, Inserted) = CGSCCAnalysisResults.insert(std::make_pair( std::make_pair(PassID, C), CGSCCAnalysisResultListT::iterator())); // If we don't have a cached result for this function, look up the pass and // run it to produce a result, which we then add to the cache. if (Inserted) { CGSCCAnalysisResultListT &ResultList = CGSCCAnalysisResultLists[C]; ResultList.emplace_back(PassID, lookupPass(PassID).run(C, this)); RI->second = std::prev(ResultList.end()); } return *RI->second->second; } CGSCCAnalysisManager::ResultConceptT * CGSCCAnalysisManager::getCachedResultImpl(void *PassID, LazyCallGraph::SCC *C) const { CGSCCAnalysisResultMapT::const_iterator RI = CGSCCAnalysisResults.find(std::make_pair(PassID, C)); return RI == CGSCCAnalysisResults.end() ? nullptr : &*RI->second->second; } void CGSCCAnalysisManager::invalidateImpl(void *PassID, LazyCallGraph::SCC *C) { CGSCCAnalysisResultMapT::iterator RI = CGSCCAnalysisResults.find(std::make_pair(PassID, C)); if (RI == CGSCCAnalysisResults.end()) return; CGSCCAnalysisResultLists[C].erase(RI->second); } void CGSCCAnalysisManager::invalidateImpl(LazyCallGraph::SCC *C, const PreservedAnalyses &PA) { // Clear all the invalidated results associated specifically with this // function. SmallVector InvalidatedPassIDs; CGSCCAnalysisResultListT &ResultsList = CGSCCAnalysisResultLists[C]; for (CGSCCAnalysisResultListT::iterator I = ResultsList.begin(), E = ResultsList.end(); I != E;) if (I->second->invalidate(C, PA)) { InvalidatedPassIDs.push_back(I->first); I = ResultsList.erase(I); } else { ++I; } while (!InvalidatedPassIDs.empty()) CGSCCAnalysisResults.erase( std::make_pair(InvalidatedPassIDs.pop_back_val(), C)); CGSCCAnalysisResultLists.erase(C); } char CGSCCAnalysisManagerModuleProxy::PassID; CGSCCAnalysisManagerModuleProxy::Result CGSCCAnalysisManagerModuleProxy::run(Module *M) { assert(CGAM->empty() && "CGSCC analyses ran prior to the module proxy!"); return Result(*CGAM); } CGSCCAnalysisManagerModuleProxy::Result::~Result() { // Clear out the analysis manager if we're being destroyed -- it means we // didn't even see an invalidate call when we got invalidated. CGAM->clear(); } bool CGSCCAnalysisManagerModuleProxy::Result::invalidate( Module *M, const PreservedAnalyses &PA) { // If this proxy isn't marked as preserved, then we can't even invalidate // individual CGSCC analyses, there may be an invalid set of SCC objects in // the cache making it impossible to incrementally preserve them. // Just clear the entire manager. if (!PA.preserved(ID())) CGAM->clear(); // Return false to indicate that this result is still a valid proxy. return false; } char ModuleAnalysisManagerCGSCCProxy::PassID; char FunctionAnalysisManagerCGSCCProxy::PassID; FunctionAnalysisManagerCGSCCProxy::Result FunctionAnalysisManagerCGSCCProxy::run(LazyCallGraph::SCC *C) { assert(FAM->empty() && "Function analyses ran prior to the CGSCC proxy!"); return Result(*FAM); } FunctionAnalysisManagerCGSCCProxy::Result::~Result() { // Clear out the analysis manager if we're being destroyed -- it means we // didn't even see an invalidate call when we got invalidated. FAM->clear(); } bool FunctionAnalysisManagerCGSCCProxy::Result::invalidate( LazyCallGraph::SCC *C, const PreservedAnalyses &PA) { // If this proxy isn't marked as preserved, then we can't even invalidate // individual function analyses, there may be an invalid set of Function // objects in the cache making it impossible to incrementally preserve them. // Just clear the entire manager. if (!PA.preserved(ID())) FAM->clear(); // Return false to indicate that this result is still a valid proxy. return false; } char CGSCCAnalysisManagerFunctionProxy::PassID;