diff options
| author | Fedor Sergeev <fedor.sergeev@azul.com> | 2018-10-05 22:32:01 +0000 | 
|---|---|---|
| committer | Fedor Sergeev <fedor.sergeev@azul.com> | 2018-10-05 22:32:01 +0000 | 
| commit | 884e52a7d230cf64f4814d0a3c367938083fa855 (patch) | |
| tree | aea0f960688b046078592b3208ada883e33680fe /llvm/lib | |
| parent | 0a6c000c1687d70d849f15a6d51f0dad7680359f (diff) | |
| download | bcm5719-llvm-884e52a7d230cf64f4814d0a3c367938083fa855.tar.gz bcm5719-llvm-884e52a7d230cf64f4814d0a3c367938083fa855.zip | |
[New PM][PassTiming] implement -time-passes for the new pass manager
Enable time-passes functionality through PassInstrumentation callbacks
for passes and analyses.
TimePassesHandler class keeps all the callbacks, the timing data as it
is being collected as well as the stack of currently active timers.
Parts of the fix that might be somewhat unobvious:
  - mapping of passes into Timer (TimingData) can not be done per-instance.
    PassID name provided into the callback is common for all the pass invocations.
    Thus the only way to get a timing with reasonable granularity is to collect
    timing data per pass invocation, getting a new timer for each BeforePass.
    Hence the key for TimingData uses a pair of <StringRef/unsigned count> to
    uniquely identify a pass invocation.
  - consequently, this new-pass-manager implementation performs no aggregation
    of timing data, reporting timings for each pass invocation separately.
    In that it differs from legacy-pass-manager time-passes implementation that
    reports timing data aggregated per pass instance.
  - pass managers and adaptors are not tracked, similar to how pass managers are
    not tracked in legacy time-passes.
  - TimerStack tracks timers that are active, each BeforePass pushes the new timer
    on stack, each AfterPass pops active timer from stack and stops it.
Reviewers: chandlerc, philip.pfaffe
Differential Revision: https://reviews.llvm.org/D51276
llvm-svn: 343898
Diffstat (limited to 'llvm/lib')
| -rw-r--r-- | llvm/lib/IR/PassTimingInfo.cpp | 109 | ||||
| -rw-r--r-- | llvm/lib/Passes/StandardInstrumentations.cpp | 1 | 
2 files changed, 109 insertions, 1 deletions
| diff --git a/llvm/lib/IR/PassTimingInfo.cpp b/llvm/lib/IR/PassTimingInfo.cpp index 895cbc1f0ca..8165704a9ec 100644 --- a/llvm/lib/IR/PassTimingInfo.cpp +++ b/llvm/lib/IR/PassTimingInfo.cpp @@ -20,6 +20,7 @@  #include "llvm/ADT/DenseMap.h"  #include "llvm/ADT/Statistic.h"  #include "llvm/ADT/StringRef.h" +#include "llvm/IR/PassInstrumentation.h"  #include "llvm/Pass.h"  #include "llvm/Support/CommandLine.h"  #include "llvm/Support/Debug.h" @@ -33,6 +34,8 @@  using namespace llvm; +#define DEBUG_TYPE "time-passes" +  namespace llvm {  bool TimePassesIsEnabled = false; @@ -45,7 +48,7 @@ namespace {  namespace legacy {  //===----------------------------------------------------------------------===// -// TimingInfo implementation +// Legacy pass manager's PassTimingInfo implementation  /// Provides an interface for collecting pass timing information.  /// @@ -156,4 +159,108 @@ void reportAndResetTimings() {      legacy::PassTimingInfo::TheTimeInfo->print();  } +//===----------------------------------------------------------------------===// +// 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) { +  // Bump counts for each request of the timer. +  unsigned Count = nextPassID(PassID); + +  // Unconditionally appending description with a pass-invocation number. +  std::string FullDesc = formatv("{0} #{1}", PassID, Count).str(); + +  PassInvocationID UID{PassID, Count}; +  Timer *T = new Timer(PassID, FullDesc, TG); +  auto Pair = TimingData.try_emplace(UID, T); +  assert(Pair.second && "should always create a new timer"); +  return *(Pair.first->second.get()); +} + +TimePassesHandler::TimePassesHandler(bool Enabled) +    : TG("pass", "... Pass execution timing report ..."), Enabled(Enabled) {} + +void TimePassesHandler::print() { TG.print(*CreateInfoOutputFile()); } + +LLVM_DUMP_METHOD void TimePassesHandler::dump() const { +  dbgs() << "Dumping timers for " << getTypeName<TimePassesHandler>() +         << ":\n\tRunning:\n"; +  for (auto &I : TimingData) { +    const Timer *MyTimer = I.second.get(); +    if (!MyTimer || MyTimer->isRunning()) +      dbgs() << "\tTimer " << MyTimer << " for pass " << I.first.first << "(" +             << I.first.second << ")\n"; +  } +  dbgs() << "\tTriggered:\n"; +  for (auto &I : TimingData) { +    const Timer *MyTimer = I.second.get(); +    if (!MyTimer || (MyTimer->hasTriggered() && !MyTimer->isRunning())) +      dbgs() << "\tTimer " << MyTimer << " for pass " << I.first.first << "(" +             << I.first.second << ")\n"; +  } +} + +void TimePassesHandler::startTimer(StringRef PassID) { +  Timer &MyTimer = getPassTimer(PassID); +  TimerStack.push_back(&MyTimer); +  if (!MyTimer.isRunning()) +    MyTimer.startTimer(); +} + +void TimePassesHandler::stopTimer(StringRef PassID) { +  assert(TimerStack.size() > 0 && "empty stack in popTimer"); +  Timer *MyTimer = TimerStack.pop_back_val(); +  assert(MyTimer && "timer should be present"); +  if (MyTimer->isRunning()) +    MyTimer->stopTimer(); +} + +static bool matchPassManager(StringRef PassID) { +  size_t prefix_pos = PassID.find('<'); +  if (prefix_pos == StringRef::npos) +    return false; +  StringRef Prefix = PassID.substr(0, prefix_pos); +  return Prefix.endswith("PassManager") || Prefix.endswith("PassAdaptor") || +         Prefix.endswith("AnalysisManagerProxy"); +} + +bool TimePassesHandler::runBeforePass(StringRef PassID, Any IR) { +  if (matchPassManager(PassID)) +    return true; + +  startTimer(PassID); + +  LLVM_DEBUG(dbgs() << "after runBeforePass(" << PassID << ")\n"); +  LLVM_DEBUG(dump()); + +  // we are not going to skip this pass, thus return true. +  return true; +} + +void TimePassesHandler::runAfterPass(StringRef PassID, Any IR) { +  if (matchPassManager(PassID)) +    return; + +  stopTimer(PassID); + +  LLVM_DEBUG(dbgs() << "after runAfterPass(" << PassID << ")\n"); +  LLVM_DEBUG(dump()); +} + +void TimePassesHandler::registerCallbacks(PassInstrumentationCallbacks &PIC) { +  if (!Enabled) +    return; + +  PIC.registerBeforePassCallback( +      [this](StringRef P, Any IR) { return this->runBeforePass(P, IR); }); +  PIC.registerAfterPassCallback( +      [this](StringRef P, Any IR) { this->runAfterPass(P, IR); }); +  PIC.registerBeforeAnalysisCallback( +      [this](StringRef P, Any IR) { this->runBeforePass(P, IR); }); +  PIC.registerAfterAnalysisCallback( +      [this](StringRef P, Any IR) { this->runAfterPass(P, IR); }); +} +  } // namespace llvm diff --git a/llvm/lib/Passes/StandardInstrumentations.cpp b/llvm/lib/Passes/StandardInstrumentations.cpp index 6abd39c61ac..aa34584fa12 100644 --- a/llvm/lib/Passes/StandardInstrumentations.cpp +++ b/llvm/lib/Passes/StandardInstrumentations.cpp @@ -112,4 +112,5 @@ void StandardInstrumentations::registerCallbacks(      PIC.registerBeforePassCallback(PrintIR::printBeforePass);    if (llvm::shouldPrintAfterPass())      PIC.registerAfterPassCallback(PrintIR::printAfterPass); +  TimePasses.registerCallbacks(PIC);  } | 

