//===- PassTimingInfo.cpp - LLVM Pass Timing Implementation ---------------===// // // 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 LLVM Pass Timing infrastructure for both // new and legacy pass managers. // // PassTimingInfo Class - This class is used to calculate information about the // amount of time each pass takes to execute. This only happens when // -time-passes is enabled on the command line. // //===----------------------------------------------------------------------===// #include "llvm/IR/PassTimingInfo.h" #include "llvm/ADT/Statistic.h" #include "llvm/IR/PassInstrumentation.h" #include "llvm/Pass.h" #include "llvm/Support/CommandLine.h" #include "llvm/Support/Debug.h" #include "llvm/Support/FormatVariadic.h" #include "llvm/Support/ManagedStatic.h" #include "llvm/Support/Mutex.h" #include "llvm/Support/TypeName.h" #include "llvm/Support/raw_ostream.h" #include using namespace llvm; #define DEBUG_TYPE "time-passes" namespace llvm { bool TimePassesIsEnabled = false; bool TimePassesPerRun = false; static cl::opt EnableTiming( "time-passes", cl::location(TimePassesIsEnabled), cl::Hidden, cl::desc("Time each pass, printing elapsed time for each on exit")); static cl::opt EnableTimingPerRun( "time-passes-per-run", cl::location(TimePassesPerRun), cl::Hidden, cl::desc("Time each pass run, printing elapsed time for each run on exit"), cl::callback([](const bool &) { TimePassesIsEnabled = true; })); namespace { namespace legacy { //===----------------------------------------------------------------------===// // Legacy pass manager's PassTimingInfo implementation /// Provides an interface for collecting pass timing information. /// /// It was intended to be generic but now we decided to split /// interfaces completely. This is now exclusively for legacy-pass-manager use. class PassTimingInfo { public: using PassInstanceID = void *; private: StringMap PassIDCountMap; ///< Map that counts instances of passes DenseMap> TimingData; ///< timers for pass instances TimerGroup TG; public: /// Default constructor for yet-inactive timeinfo. /// Use \p init() to activate it. PassTimingInfo(); /// Print out timing information and release timers. ~PassTimingInfo(); /// Initializes the static \p TheTimeInfo member to a non-null value when /// -time-passes is enabled. Leaves it null otherwise. /// /// This method may be called multiple times. static void init(); /// Prints out timing information and then resets the timers. /// By default it uses the stream created by CreateInfoOutputFile(). void print(raw_ostream *OutStream = nullptr); /// Returns the timer for the specified pass if it exists. Timer *getPassTimer(Pass *, PassInstanceID); static PassTimingInfo *TheTimeInfo; private: Timer *newPassTimer(StringRef PassID, StringRef PassDesc); }; static ManagedStatic> TimingInfoMutex; PassTimingInfo::PassTimingInfo() : TG("pass", "Pass execution timing report") {} PassTimingInfo::~PassTimingInfo() { // Deleting the timers accumulates their info into the TG member. // Then TG member is (implicitly) deleted, actually printing the report. TimingData.clear(); } void PassTimingInfo::init() { if (!TimePassesIsEnabled || TheTimeInfo) return; // Constructed the first time this is called, iff -time-passes is enabled. // This guarantees that the object will be constructed after static globals, // thus it will be destroyed before them. static ManagedStatic TTI; TheTimeInfo = &*TTI; } /// Prints out timing information and then resets the timers. void PassTimingInfo::print(raw_ostream *OutStream) { TG.print(OutStream ? *OutStream : *CreateInfoOutputFile(), true); } Timer *PassTimingInfo::newPassTimer(StringRef PassID, StringRef PassDesc) { unsigned &num = PassIDCountMap[PassID]; num++; // Appending description with a pass-instance number for all but the first one std::string PassDescNumbered = num <= 1 ? PassDesc.str() : formatv("{0} #{1}", PassDesc, num).str(); return new Timer(PassID, PassDescNumbered, TG); } Timer *PassTimingInfo::getPassTimer(Pass *P, PassInstanceID Pass) { if (P->getAsPMDataManager()) return nullptr; init(); sys::SmartScopedLock Lock(*TimingInfoMutex); std::unique_ptr &T = TimingData[Pass]; if (!T) { StringRef PassName = P->getPassName(); StringRef PassArgument; if (const PassInfo *PI = Pass::lookupPassInfo(P->getPassID())) PassArgument = PI->getPassArgument(); T.reset(newPassTimer(PassArgument.empty() ? PassName : PassArgument, PassName)); } return T.get(); } PassTimingInfo *PassTimingInfo::TheTimeInfo; } // namespace legacy } // namespace Timer *getPassTimer(Pass *P) { legacy::PassTimingInfo::init(); if (legacy::PassTimingInfo::TheTimeInfo) return legacy::PassTimingInfo::TheTimeInfo->getPassTimer(P, P); return nullptr; } /// If timing is enabled, report the times collected up to now and then reset /// them. void reportAndResetTimings(raw_ostream *OutStream) { if (legacy::PassTimingInfo::TheTimeInfo) legacy::PassTimingInfo::TheTimeInfo->print(OutStream); } //===----------------------------------------------------------------------===// // Pass timing handling for the New Pass Manager //===----------------------------------------------------------------------===// /// Returns the timer for the specified pass invocation of \p PassID. /// Each time it creates a new timer. Timer &TimePassesHandler::getPassTimer(StringRef PassID, bool IsPass) { TimerGroup &TG = IsPass ? PassTG : AnalysisTG; if (!PerRun) { TimerVector &Timers = TimingData[PassID]; if (Timers.size() == 0) Timers.emplace_back(new Timer(PassID, PassID, TG)); return *Timers.front(); } // Take a vector of Timers created for this \p PassID and append // one more timer to it. TimerVector &Timers = TimingData[PassID]; unsigned Count = Timers.size() + 1; std::string FullDesc = formatv("{0} #{1}", PassID, Count).str(); Timer *T = new Timer(PassID, FullDesc, TG); Timers.emplace_back(T); assert(Count == Timers.size() && "Timers vector not adjusted correctly."); return *T; } TimePassesHandler::TimePassesHandler(bool Enabled, bool PerRun) : PassTG("pass", "Pass execution timing report"), AnalysisTG("analysis", "Analysis execution timing report"), Enabled(Enabled), PerRun(PerRun) {} TimePassesHandler::TimePassesHandler() : TimePassesHandler(TimePassesIsEnabled, TimePassesPerRun) {} void TimePassesHandler::setOutStream(raw_ostream &Out) { OutStream = &Out; } void TimePassesHandler::print() { if (!Enabled) return; std::unique_ptr MaybeCreated; raw_ostream *OS = OutStream; if (OutStream) { OS = OutStream; } else { MaybeCreated = CreateInfoOutputFile(); OS = &*MaybeCreated; } PassTG.print(*OS, true); AnalysisTG.print(*OS, true); } LLVM_DUMP_METHOD void TimePassesHandler::dump() const { dbgs() << "Dumping timers for " << getTypeName() << ":\n\tRunning:\n"; for (auto &I : TimingData) { StringRef PassID = I.getKey(); const TimerVector& MyTimers = I.getValue(); for (unsigned idx = 0; idx < MyTimers.size(); idx++) { const Timer* MyTimer = MyTimers[idx].get(); if (MyTimer && MyTimer->isRunning()) dbgs() << "\tTimer " << MyTimer << " for pass " << PassID << "(" << idx << ")\n"; } } dbgs() << "\tTriggered:\n"; for (auto &I : TimingData) { StringRef PassID = I.getKey(); const TimerVector& MyTimers = I.getValue(); for (unsigned idx = 0; idx < MyTimers.size(); idx++) { const Timer* MyTimer = MyTimers[idx].get(); if (MyTimer && MyTimer->hasTriggered() && !MyTimer->isRunning()) dbgs() << "\tTimer " << MyTimer << " for pass " << PassID << "(" << idx << ")\n"; } } } static bool shouldIgnorePass(StringRef PassID) { return isSpecialPass(PassID, {"PassManager", "PassAdaptor", "AnalysisManagerProxy", "ModuleInlinerWrapperPass", "DevirtSCCRepeatedPass"}); } void TimePassesHandler::startPassTimer(StringRef PassID) { if (shouldIgnorePass(PassID)) return; assert(!ActivePassTimer && "should only have one pass timer at a time"); Timer &MyTimer = getPassTimer(PassID, /*IsPass*/ true); ActivePassTimer = &MyTimer; assert(!MyTimer.isRunning()); MyTimer.startTimer(); } void TimePassesHandler::stopPassTimer(StringRef PassID) { if (shouldIgnorePass(PassID)) return; assert(ActivePassTimer); assert(ActivePassTimer->isRunning()); ActivePassTimer->stopTimer(); ActivePassTimer = nullptr; } void TimePassesHandler::startAnalysisTimer(StringRef PassID) { // Stop the previous analysis timer to prevent double counting when an // analysis requests another analysis. if (!AnalysisActiveTimerStack.empty()) { assert(AnalysisActiveTimerStack.back()->isRunning()); AnalysisActiveTimerStack.back()->stopTimer(); } Timer &MyTimer = getPassTimer(PassID, /*IsPass*/ false); AnalysisActiveTimerStack.push_back(&MyTimer); if (!MyTimer.isRunning()) MyTimer.startTimer(); } void TimePassesHandler::stopAnalysisTimer(StringRef PassID) { assert(!AnalysisActiveTimerStack.empty() && "empty stack in popTimer"); Timer *MyTimer = AnalysisActiveTimerStack.pop_back_val(); assert(MyTimer && "timer should be present"); if (MyTimer->isRunning()) MyTimer->stopTimer(); // Restart the previously stopped timer. if (!AnalysisActiveTimerStack.empty()) { assert(!AnalysisActiveTimerStack.back()->isRunning()); AnalysisActiveTimerStack.back()->startTimer(); } } void TimePassesHandler::registerCallbacks(PassInstrumentationCallbacks &PIC) { if (!Enabled) return; PIC.registerBeforeNonSkippedPassCallback( [this](StringRef P, Any) { this->startPassTimer(P); }); PIC.registerAfterPassCallback( [this](StringRef P, Any, const PreservedAnalyses &) { this->stopPassTimer(P); }); PIC.registerAfterPassInvalidatedCallback( [this](StringRef P, const PreservedAnalyses &) { this->stopPassTimer(P); }); PIC.registerBeforeAnalysisCallback( [this](StringRef P, Any) { this->startAnalysisTimer(P); }); PIC.registerAfterAnalysisCallback( [this](StringRef P, Any) { this->stopAnalysisTimer(P); }); } } // namespace llvm