blob: 40b3977ecbd9b92dbaea277803af952d4c591e2f [file] [log] [blame]
Fedor Sergeev9f6be222018-08-28 21:06:51 +00001//===- PassTimingInfo.cpp - LLVM Pass Timing Implementation ---------------===//
2//
3// The LLVM Compiler Infrastructure
4//
5// This file is distributed under the University of Illinois Open Source
6// License. See LICENSE.TXT for details.
7//
8//===----------------------------------------------------------------------===//
9//
10// This file implements the LLVM Pass Timing infrastructure for both
11// new and legacy pass managers.
12//
Fedor Sergeevb8bd68f2018-09-26 13:01:43 +000013// PassTimingInfo Class - This class is used to calculate information about the
Fedor Sergeev9f6be222018-08-28 21:06:51 +000014// amount of time each pass takes to execute. This only happens when
15// -time-passes is enabled on the command line.
16//
17//===----------------------------------------------------------------------===//
18
19#include "llvm/IR/PassTimingInfo.h"
20#include "llvm/ADT/DenseMap.h"
21#include "llvm/ADT/Statistic.h"
22#include "llvm/ADT/StringRef.h"
Fedor Sergeev17dce3b2018-10-05 22:32:01 +000023#include "llvm/IR/PassInstrumentation.h"
Fedor Sergeev9f6be222018-08-28 21:06:51 +000024#include "llvm/Pass.h"
25#include "llvm/Support/CommandLine.h"
26#include "llvm/Support/Debug.h"
Fedor Sergeev0f20aef2018-09-04 06:12:28 +000027#include "llvm/Support/FormatVariadic.h"
Fedor Sergeev9f6be222018-08-28 21:06:51 +000028#include "llvm/Support/ManagedStatic.h"
29#include "llvm/Support/Mutex.h"
30#include "llvm/Support/Timer.h"
31#include "llvm/Support/raw_ostream.h"
Fedor Sergeevc7144772018-10-04 12:49:57 +000032#include <memory>
Fedor Sergeev9f6be222018-08-28 21:06:51 +000033#include <string>
34
35using namespace llvm;
36
Fedor Sergeev17dce3b2018-10-05 22:32:01 +000037#define DEBUG_TYPE "time-passes"
38
Fedor Sergeev9f6be222018-08-28 21:06:51 +000039namespace llvm {
40
Fedor Sergeev9f6be222018-08-28 21:06:51 +000041bool TimePassesIsEnabled = false;
Fedor Sergeevb8bd68f2018-09-26 13:01:43 +000042
Fedor Sergeev9f6be222018-08-28 21:06:51 +000043static cl::opt<bool, true> EnableTiming(
44 "time-passes", cl::location(TimePassesIsEnabled), cl::Hidden,
45 cl::desc("Time each pass, printing elapsed time for each on exit"));
46
47namespace {
Fedor Sergeevb8bd68f2018-09-26 13:01:43 +000048namespace legacy {
Fedor Sergeev9f6be222018-08-28 21:06:51 +000049
Fedor Sergeevb8bd68f2018-09-26 13:01:43 +000050//===----------------------------------------------------------------------===//
Fedor Sergeev17dce3b2018-10-05 22:32:01 +000051// Legacy pass manager's PassTimingInfo implementation
Fedor Sergeevb8bd68f2018-09-26 13:01:43 +000052
53/// Provides an interface for collecting pass timing information.
54///
55/// It was intended to be generic but now we decided to split
56/// interfaces completely. This is now exclusively for legacy-pass-manager use.
57class PassTimingInfo {
58public:
59 using PassInstanceID = void *;
60
61private:
62 StringMap<unsigned> PassIDCountMap; ///< Map that counts instances of passes
Fedor Sergeevc7144772018-10-04 12:49:57 +000063 DenseMap<PassInstanceID, std::unique_ptr<Timer>> TimingData; ///< timers for pass instances
Fedor Sergeevb8bd68f2018-09-26 13:01:43 +000064 TimerGroup TG;
65
66public:
67 /// Default constructor for yet-inactive timeinfo.
68 /// Use \p init() to activate it.
69 PassTimingInfo();
70
71 /// Print out timing information and release timers.
72 ~PassTimingInfo();
73
74 /// Initializes the static \p TheTimeInfo member to a non-null value when
75 /// -time-passes is enabled. Leaves it null otherwise.
76 ///
77 /// This method may be called multiple times.
78 static void init();
79
80 /// Prints out timing information and then resets the timers.
81 void print();
82
83 /// Returns the timer for the specified pass if it exists.
84 Timer *getPassTimer(Pass *, PassInstanceID);
85
86 static PassTimingInfo *TheTimeInfo;
87
88private:
89 Timer *newPassTimer(StringRef PassID, StringRef PassDesc);
90};
91
92static ManagedStatic<sys::SmartMutex<true>> TimingInfoMutex;
93
94PassTimingInfo::PassTimingInfo()
Fedor Sergeev9f6be222018-08-28 21:06:51 +000095 : TG("pass", "... Pass execution timing report ...") {}
96
Fedor Sergeevb8bd68f2018-09-26 13:01:43 +000097PassTimingInfo::~PassTimingInfo() {
Fedor Sergeev9f6be222018-08-28 21:06:51 +000098 // Deleting the timers accumulates their info into the TG member.
99 // Then TG member is (implicitly) deleted, actually printing the report.
Fedor Sergeevc7144772018-10-04 12:49:57 +0000100 TimingData.clear();
Fedor Sergeev9f6be222018-08-28 21:06:51 +0000101}
102
Fedor Sergeevb8bd68f2018-09-26 13:01:43 +0000103void PassTimingInfo::init() {
Fedor Sergeev9f6be222018-08-28 21:06:51 +0000104 if (!TimePassesIsEnabled || TheTimeInfo)
105 return;
106
107 // Constructed the first time this is called, iff -time-passes is enabled.
Fedor Sergeev0f20aef2018-09-04 06:12:28 +0000108 // This guarantees that the object will be constructed after static globals,
Fedor Sergeev9f6be222018-08-28 21:06:51 +0000109 // thus it will be destroyed before them.
110 static ManagedStatic<PassTimingInfo> TTI;
111 TheTimeInfo = &*TTI;
112}
113
114/// Prints out timing information and then resets the timers.
Fedor Sergeevb8bd68f2018-09-26 13:01:43 +0000115void PassTimingInfo::print() { TG.print(*CreateInfoOutputFile()); }
Fedor Sergeev9f6be222018-08-28 21:06:51 +0000116
Fedor Sergeevb8bd68f2018-09-26 13:01:43 +0000117Timer *PassTimingInfo::newPassTimer(StringRef PassID, StringRef PassDesc) {
Fedor Sergeev0f20aef2018-09-04 06:12:28 +0000118 unsigned &num = PassIDCountMap[PassID];
119 num++;
120 // Appending description with a pass-instance number for all but the first one
121 std::string PassDescNumbered =
122 num <= 1 ? PassDesc.str() : formatv("{0} #{1}", PassDesc, num).str();
123 return new Timer(PassID, PassDescNumbered, TG);
124}
125
Fedor Sergeevb8bd68f2018-09-26 13:01:43 +0000126Timer *PassTimingInfo::getPassTimer(Pass *P, PassInstanceID Pass) {
Fedor Sergeev9f6be222018-08-28 21:06:51 +0000127 if (P->getAsPMDataManager())
128 return nullptr;
129
130 init();
131 sys::SmartScopedLock<true> Lock(*TimingInfoMutex);
Fedor Sergeevc7144772018-10-04 12:49:57 +0000132 std::unique_ptr<Timer> &T = TimingData[Pass];
Fedor Sergeev9f6be222018-08-28 21:06:51 +0000133
134 if (!T) {
Fedor Sergeev0f20aef2018-09-04 06:12:28 +0000135 StringRef PassName = P->getPassName();
Fedor Sergeev9f6be222018-08-28 21:06:51 +0000136 StringRef PassArgument;
137 if (const PassInfo *PI = Pass::lookupPassInfo(P->getPassID()))
138 PassArgument = PI->getPassArgument();
Fedor Sergeevc7144772018-10-04 12:49:57 +0000139 T.reset(newPassTimer(PassArgument.empty() ? PassName : PassArgument, PassName));
Fedor Sergeev9f6be222018-08-28 21:06:51 +0000140 }
Fedor Sergeevc7144772018-10-04 12:49:57 +0000141 return T.get();
Fedor Sergeev9f6be222018-08-28 21:06:51 +0000142}
143
Fedor Sergeevb8bd68f2018-09-26 13:01:43 +0000144PassTimingInfo *PassTimingInfo::TheTimeInfo;
145} // namespace legacy
146} // namespace
Fedor Sergeev9f6be222018-08-28 21:06:51 +0000147
148Timer *getPassTimer(Pass *P) {
Fedor Sergeevb8bd68f2018-09-26 13:01:43 +0000149 legacy::PassTimingInfo::init();
150 if (legacy::PassTimingInfo::TheTimeInfo)
151 return legacy::PassTimingInfo::TheTimeInfo->getPassTimer(P, P);
Fedor Sergeev9f6be222018-08-28 21:06:51 +0000152 return nullptr;
153}
154
155/// If timing is enabled, report the times collected up to now and then reset
156/// them.
157void reportAndResetTimings() {
Fedor Sergeevb8bd68f2018-09-26 13:01:43 +0000158 if (legacy::PassTimingInfo::TheTimeInfo)
159 legacy::PassTimingInfo::TheTimeInfo->print();
Fedor Sergeev9f6be222018-08-28 21:06:51 +0000160}
161
Fedor Sergeev17dce3b2018-10-05 22:32:01 +0000162//===----------------------------------------------------------------------===//
163// Pass timing handling for the New Pass Manager
164//===----------------------------------------------------------------------===//
165
166/// Returns the timer for the specified pass invocation of \p PassID.
167/// Each time it creates a new timer.
168Timer &TimePassesHandler::getPassTimer(StringRef PassID) {
169 // Bump counts for each request of the timer.
170 unsigned Count = nextPassID(PassID);
171
172 // Unconditionally appending description with a pass-invocation number.
173 std::string FullDesc = formatv("{0} #{1}", PassID, Count).str();
174
175 PassInvocationID UID{PassID, Count};
176 Timer *T = new Timer(PassID, FullDesc, TG);
177 auto Pair = TimingData.try_emplace(UID, T);
178 assert(Pair.second && "should always create a new timer");
179 return *(Pair.first->second.get());
180}
181
182TimePassesHandler::TimePassesHandler(bool Enabled)
183 : TG("pass", "... Pass execution timing report ..."), Enabled(Enabled) {}
184
185void TimePassesHandler::print() { TG.print(*CreateInfoOutputFile()); }
186
187LLVM_DUMP_METHOD void TimePassesHandler::dump() const {
188 dbgs() << "Dumping timers for " << getTypeName<TimePassesHandler>()
189 << ":\n\tRunning:\n";
190 for (auto &I : TimingData) {
191 const Timer *MyTimer = I.second.get();
192 if (!MyTimer || MyTimer->isRunning())
193 dbgs() << "\tTimer " << MyTimer << " for pass " << I.first.first << "("
194 << I.first.second << ")\n";
195 }
196 dbgs() << "\tTriggered:\n";
197 for (auto &I : TimingData) {
198 const Timer *MyTimer = I.second.get();
199 if (!MyTimer || (MyTimer->hasTriggered() && !MyTimer->isRunning()))
200 dbgs() << "\tTimer " << MyTimer << " for pass " << I.first.first << "("
201 << I.first.second << ")\n";
202 }
203}
204
205void TimePassesHandler::startTimer(StringRef PassID) {
206 Timer &MyTimer = getPassTimer(PassID);
207 TimerStack.push_back(&MyTimer);
208 if (!MyTimer.isRunning())
209 MyTimer.startTimer();
210}
211
212void TimePassesHandler::stopTimer(StringRef PassID) {
213 assert(TimerStack.size() > 0 && "empty stack in popTimer");
214 Timer *MyTimer = TimerStack.pop_back_val();
215 assert(MyTimer && "timer should be present");
216 if (MyTimer->isRunning())
217 MyTimer->stopTimer();
218}
219
220static bool matchPassManager(StringRef PassID) {
221 size_t prefix_pos = PassID.find('<');
222 if (prefix_pos == StringRef::npos)
223 return false;
224 StringRef Prefix = PassID.substr(0, prefix_pos);
225 return Prefix.endswith("PassManager") || Prefix.endswith("PassAdaptor") ||
226 Prefix.endswith("AnalysisManagerProxy");
227}
228
Fedor Sergeev601226c2018-12-11 19:05:35 +0000229bool TimePassesHandler::runBeforePass(StringRef PassID) {
Fedor Sergeev17dce3b2018-10-05 22:32:01 +0000230 if (matchPassManager(PassID))
231 return true;
232
233 startTimer(PassID);
234
235 LLVM_DEBUG(dbgs() << "after runBeforePass(" << PassID << ")\n");
236 LLVM_DEBUG(dump());
237
238 // we are not going to skip this pass, thus return true.
239 return true;
240}
241
Fedor Sergeev601226c2018-12-11 19:05:35 +0000242void TimePassesHandler::runAfterPass(StringRef PassID) {
Fedor Sergeev17dce3b2018-10-05 22:32:01 +0000243 if (matchPassManager(PassID))
244 return;
245
246 stopTimer(PassID);
247
248 LLVM_DEBUG(dbgs() << "after runAfterPass(" << PassID << ")\n");
249 LLVM_DEBUG(dump());
250}
251
252void TimePassesHandler::registerCallbacks(PassInstrumentationCallbacks &PIC) {
253 if (!Enabled)
254 return;
255
256 PIC.registerBeforePassCallback(
Fedor Sergeev601226c2018-12-11 19:05:35 +0000257 [this](StringRef P, Any) { return this->runBeforePass(P); });
Fedor Sergeev17dce3b2018-10-05 22:32:01 +0000258 PIC.registerAfterPassCallback(
Fedor Sergeev601226c2018-12-11 19:05:35 +0000259 [this](StringRef P, Any) { this->runAfterPass(P); });
260 PIC.registerAfterPassInvalidatedCallback(
261 [this](StringRef P) { this->runAfterPass(P); });
Fedor Sergeev17dce3b2018-10-05 22:32:01 +0000262 PIC.registerBeforeAnalysisCallback(
Fedor Sergeev601226c2018-12-11 19:05:35 +0000263 [this](StringRef P, Any) { this->runBeforePass(P); });
Fedor Sergeev17dce3b2018-10-05 22:32:01 +0000264 PIC.registerAfterAnalysisCallback(
Fedor Sergeev601226c2018-12-11 19:05:35 +0000265 [this](StringRef P, Any) { this->runAfterPass(P); });
Fedor Sergeev17dce3b2018-10-05 22:32:01 +0000266}
267
Fedor Sergeev9f6be222018-08-28 21:06:51 +0000268} // namespace llvm