[ORC] Add std::tuple support to SimplePackedSerialization.
[llvm-project.git] / llvm / lib / Support / Timer.cpp
blobf025ecd3d45c41992fb2453c23037672c7b0c9fd
1 //===-- Timer.cpp - Interval Timing Support -------------------------------===//
2 //
3 // Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions.
4 // See https://llvm.org/LICENSE.txt for license information.
5 // SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception
6 //
7 //===----------------------------------------------------------------------===//
8 //
9 /// \file Interval Timing implementation.
11 //===----------------------------------------------------------------------===//
13 #include "llvm/Support/Timer.h"
15 #include "DebugOptions.h"
17 #include "llvm/ADT/Statistic.h"
18 #include "llvm/ADT/StringMap.h"
19 #include "llvm/Config/config.h"
20 #include "llvm/Support/CommandLine.h"
21 #include "llvm/Support/FileSystem.h"
22 #include "llvm/Support/Format.h"
23 #include "llvm/Support/ManagedStatic.h"
24 #include "llvm/Support/Mutex.h"
25 #include "llvm/Support/Process.h"
26 #include "llvm/Support/Signposts.h"
27 #include "llvm/Support/YAMLTraits.h"
28 #include "llvm/Support/raw_ostream.h"
29 #include <limits>
31 #if HAVE_UNISTD_H
32 #include <unistd.h>
33 #endif
35 #ifdef HAVE_PROC_PID_RUSAGE
36 #include <libproc.h>
37 #endif
39 using namespace llvm;
41 // This ugly hack is brought to you courtesy of constructor/destructor ordering
42 // being unspecified by C++. Basically the problem is that a Statistic object
43 // gets destroyed, which ends up calling 'GetLibSupportInfoOutputFile()'
44 // (below), which calls this function. LibSupportInfoOutputFilename used to be
45 // a global variable, but sometimes it would get destroyed before the Statistic,
46 // causing havoc to ensue. We "fix" this by creating the string the first time
47 // it is needed and never destroying it.
48 static ManagedStatic<std::string> LibSupportInfoOutputFilename;
49 static std::string &getLibSupportInfoOutputFilename() {
50 return *LibSupportInfoOutputFilename;
53 static ManagedStatic<sys::SmartMutex<true> > TimerLock;
55 /// Allows llvm::Timer to emit signposts when supported.
56 static ManagedStatic<SignpostEmitter> Signposts;
58 namespace {
59 struct CreateTrackSpace {
60 static void *call() {
61 return new cl::opt<bool>("track-memory",
62 cl::desc("Enable -time-passes memory "
63 "tracking (this may be slow)"),
64 cl::Hidden);
67 static ManagedStatic<cl::opt<bool>, CreateTrackSpace> TrackSpace;
68 struct CreateInfoOutputFilename {
69 static void *call() {
70 return new cl::opt<std::string, true>(
71 "info-output-file", cl::value_desc("filename"),
72 cl::desc("File to append -stats and -timer output to"), cl::Hidden,
73 cl::location(getLibSupportInfoOutputFilename()));
76 static ManagedStatic<cl::opt<std::string, true>, CreateInfoOutputFilename>
77 InfoOutputFilename;
78 struct CreateSortTimers {
79 static void *call() {
80 return new cl::opt<bool>(
81 "sort-timers",
82 cl::desc("In the report, sort the timers in each group "
83 "in wall clock time order"),
84 cl::init(true), cl::Hidden);
87 ManagedStatic<cl::opt<bool>, CreateSortTimers> SortTimers;
88 } // namespace
90 void llvm::initTimerOptions() {
91 *TrackSpace;
92 *InfoOutputFilename;
93 *SortTimers;
96 std::unique_ptr<raw_fd_ostream> llvm::CreateInfoOutputFile() {
97 const std::string &OutputFilename = getLibSupportInfoOutputFilename();
98 if (OutputFilename.empty())
99 return std::make_unique<raw_fd_ostream>(2, false); // stderr.
100 if (OutputFilename == "-")
101 return std::make_unique<raw_fd_ostream>(1, false); // stdout.
103 // Append mode is used because the info output file is opened and closed
104 // each time -stats or -time-passes wants to print output to it. To
105 // compensate for this, the test-suite Makefiles have code to delete the
106 // info output file before running commands which write to it.
107 std::error_code EC;
108 auto Result = std::make_unique<raw_fd_ostream>(
109 OutputFilename, EC, sys::fs::OF_Append | sys::fs::OF_TextWithCRLF);
110 if (!EC)
111 return Result;
113 errs() << "Error opening info-output-file '"
114 << OutputFilename << " for appending!\n";
115 return std::make_unique<raw_fd_ostream>(2, false); // stderr.
118 namespace {
119 struct CreateDefaultTimerGroup {
120 static void *call() {
121 return new TimerGroup("misc", "Miscellaneous Ungrouped Timers");
124 } // namespace
125 static ManagedStatic<TimerGroup, CreateDefaultTimerGroup> DefaultTimerGroup;
126 static TimerGroup *getDefaultTimerGroup() { return &*DefaultTimerGroup; }
128 //===----------------------------------------------------------------------===//
129 // Timer Implementation
130 //===----------------------------------------------------------------------===//
132 void Timer::init(StringRef TimerName, StringRef TimerDescription) {
133 init(TimerName, TimerDescription, *getDefaultTimerGroup());
136 void Timer::init(StringRef TimerName, StringRef TimerDescription,
137 TimerGroup &tg) {
138 assert(!TG && "Timer already initialized");
139 Name.assign(TimerName.begin(), TimerName.end());
140 Description.assign(TimerDescription.begin(), TimerDescription.end());
141 Running = Triggered = false;
142 TG = &tg;
143 TG->addTimer(*this);
146 Timer::~Timer() {
147 if (!TG) return; // Never initialized, or already cleared.
148 TG->removeTimer(*this);
151 static inline size_t getMemUsage() {
152 if (!*TrackSpace)
153 return 0;
154 return sys::Process::GetMallocUsage();
157 static uint64_t getCurInstructionsExecuted() {
158 #if defined(HAVE_UNISTD_H) && defined(HAVE_PROC_PID_RUSAGE) && \
159 defined(RUSAGE_INFO_V4)
160 struct rusage_info_v4 ru;
161 if (proc_pid_rusage(getpid(), RUSAGE_INFO_V4, (rusage_info_t *)&ru) == 0) {
162 return ru.ri_instructions;
164 #endif
165 return 0;
168 TimeRecord TimeRecord::getCurrentTime(bool Start) {
169 using Seconds = std::chrono::duration<double, std::ratio<1>>;
170 TimeRecord Result;
171 sys::TimePoint<> now;
172 std::chrono::nanoseconds user, sys;
174 if (Start) {
175 Result.MemUsed = getMemUsage();
176 Result.InstructionsExecuted = getCurInstructionsExecuted();
177 sys::Process::GetTimeUsage(now, user, sys);
178 } else {
179 sys::Process::GetTimeUsage(now, user, sys);
180 Result.InstructionsExecuted = getCurInstructionsExecuted();
181 Result.MemUsed = getMemUsage();
184 Result.WallTime = Seconds(now.time_since_epoch()).count();
185 Result.UserTime = Seconds(user).count();
186 Result.SystemTime = Seconds(sys).count();
187 return Result;
190 void Timer::startTimer() {
191 assert(!Running && "Cannot start a running timer");
192 Running = Triggered = true;
193 Signposts->startInterval(this, getName());
194 StartTime = TimeRecord::getCurrentTime(true);
197 void Timer::stopTimer() {
198 assert(Running && "Cannot stop a paused timer");
199 Running = false;
200 Time += TimeRecord::getCurrentTime(false);
201 Time -= StartTime;
202 Signposts->endInterval(this);
205 void Timer::clear() {
206 Running = Triggered = false;
207 Time = StartTime = TimeRecord();
210 static void printVal(double Val, double Total, raw_ostream &OS) {
211 if (Total < 1e-7) // Avoid dividing by zero.
212 OS << " ----- ";
213 else
214 OS << format(" %7.4f (%5.1f%%)", Val, Val*100/Total);
217 void TimeRecord::print(const TimeRecord &Total, raw_ostream &OS) const {
218 if (Total.getUserTime())
219 printVal(getUserTime(), Total.getUserTime(), OS);
220 if (Total.getSystemTime())
221 printVal(getSystemTime(), Total.getSystemTime(), OS);
222 if (Total.getProcessTime())
223 printVal(getProcessTime(), Total.getProcessTime(), OS);
224 printVal(getWallTime(), Total.getWallTime(), OS);
226 OS << " ";
228 if (Total.getMemUsed())
229 OS << format("%9" PRId64 " ", (int64_t)getMemUsed());
230 if (Total.getInstructionsExecuted())
231 OS << format("%9" PRId64 " ", (int64_t)getInstructionsExecuted());
235 //===----------------------------------------------------------------------===//
236 // NamedRegionTimer Implementation
237 //===----------------------------------------------------------------------===//
239 namespace {
241 typedef StringMap<Timer> Name2TimerMap;
243 class Name2PairMap {
244 StringMap<std::pair<TimerGroup*, Name2TimerMap> > Map;
245 public:
246 ~Name2PairMap() {
247 for (StringMap<std::pair<TimerGroup*, Name2TimerMap> >::iterator
248 I = Map.begin(), E = Map.end(); I != E; ++I)
249 delete I->second.first;
252 Timer &get(StringRef Name, StringRef Description, StringRef GroupName,
253 StringRef GroupDescription) {
254 sys::SmartScopedLock<true> L(*TimerLock);
256 std::pair<TimerGroup*, Name2TimerMap> &GroupEntry = Map[GroupName];
258 if (!GroupEntry.first)
259 GroupEntry.first = new TimerGroup(GroupName, GroupDescription);
261 Timer &T = GroupEntry.second[Name];
262 if (!T.isInitialized())
263 T.init(Name, Description, *GroupEntry.first);
264 return T;
270 static ManagedStatic<Name2PairMap> NamedGroupedTimers;
272 NamedRegionTimer::NamedRegionTimer(StringRef Name, StringRef Description,
273 StringRef GroupName,
274 StringRef GroupDescription, bool Enabled)
275 : TimeRegion(!Enabled ? nullptr
276 : &NamedGroupedTimers->get(Name, Description, GroupName,
277 GroupDescription)) {}
279 //===----------------------------------------------------------------------===//
280 // TimerGroup Implementation
281 //===----------------------------------------------------------------------===//
283 /// This is the global list of TimerGroups, maintained by the TimerGroup
284 /// ctor/dtor and is protected by the TimerLock lock.
285 static TimerGroup *TimerGroupList = nullptr;
287 TimerGroup::TimerGroup(StringRef Name, StringRef Description)
288 : Name(Name.begin(), Name.end()),
289 Description(Description.begin(), Description.end()) {
290 // Add the group to TimerGroupList.
291 sys::SmartScopedLock<true> L(*TimerLock);
292 if (TimerGroupList)
293 TimerGroupList->Prev = &Next;
294 Next = TimerGroupList;
295 Prev = &TimerGroupList;
296 TimerGroupList = this;
299 TimerGroup::TimerGroup(StringRef Name, StringRef Description,
300 const StringMap<TimeRecord> &Records)
301 : TimerGroup(Name, Description) {
302 TimersToPrint.reserve(Records.size());
303 for (const auto &P : Records)
304 TimersToPrint.emplace_back(P.getValue(), std::string(P.getKey()),
305 std::string(P.getKey()));
306 assert(TimersToPrint.size() == Records.size() && "Size mismatch");
309 TimerGroup::~TimerGroup() {
310 // If the timer group is destroyed before the timers it owns, accumulate and
311 // print the timing data.
312 while (FirstTimer)
313 removeTimer(*FirstTimer);
315 // Remove the group from the TimerGroupList.
316 sys::SmartScopedLock<true> L(*TimerLock);
317 *Prev = Next;
318 if (Next)
319 Next->Prev = Prev;
323 void TimerGroup::removeTimer(Timer &T) {
324 sys::SmartScopedLock<true> L(*TimerLock);
326 // If the timer was started, move its data to TimersToPrint.
327 if (T.hasTriggered())
328 TimersToPrint.emplace_back(T.Time, T.Name, T.Description);
330 T.TG = nullptr;
332 // Unlink the timer from our list.
333 *T.Prev = T.Next;
334 if (T.Next)
335 T.Next->Prev = T.Prev;
337 // Print the report when all timers in this group are destroyed if some of
338 // them were started.
339 if (FirstTimer || TimersToPrint.empty())
340 return;
342 std::unique_ptr<raw_ostream> OutStream = CreateInfoOutputFile();
343 PrintQueuedTimers(*OutStream);
346 void TimerGroup::addTimer(Timer &T) {
347 sys::SmartScopedLock<true> L(*TimerLock);
349 // Add the timer to our list.
350 if (FirstTimer)
351 FirstTimer->Prev = &T.Next;
352 T.Next = FirstTimer;
353 T.Prev = &FirstTimer;
354 FirstTimer = &T;
357 void TimerGroup::PrintQueuedTimers(raw_ostream &OS) {
358 // Perhaps sort the timers in descending order by amount of time taken.
359 if (*SortTimers)
360 llvm::sort(TimersToPrint);
362 TimeRecord Total;
363 for (const PrintRecord &Record : TimersToPrint)
364 Total += Record.Time;
366 // Print out timing header.
367 OS << "===" << std::string(73, '-') << "===\n";
368 // Figure out how many spaces to indent TimerGroup name.
369 unsigned Padding = (80-Description.length())/2;
370 if (Padding > 80) Padding = 0; // Don't allow "negative" numbers
371 OS.indent(Padding) << Description << '\n';
372 OS << "===" << std::string(73, '-') << "===\n";
374 // If this is not an collection of ungrouped times, print the total time.
375 // Ungrouped timers don't really make sense to add up. We still print the
376 // TOTAL line to make the percentages make sense.
377 if (this != getDefaultTimerGroup())
378 OS << format(" Total Execution Time: %5.4f seconds (%5.4f wall clock)\n",
379 Total.getProcessTime(), Total.getWallTime());
380 OS << '\n';
382 if (Total.getUserTime())
383 OS << " ---User Time---";
384 if (Total.getSystemTime())
385 OS << " --System Time--";
386 if (Total.getProcessTime())
387 OS << " --User+System--";
388 OS << " ---Wall Time---";
389 if (Total.getMemUsed())
390 OS << " ---Mem---";
391 if (Total.getInstructionsExecuted())
392 OS << " ---Instr---";
393 OS << " --- Name ---\n";
395 // Loop through all of the timing data, printing it out.
396 for (const PrintRecord &Record : make_range(TimersToPrint.rbegin(),
397 TimersToPrint.rend())) {
398 Record.Time.print(Total, OS);
399 OS << Record.Description << '\n';
402 Total.print(Total, OS);
403 OS << "Total\n\n";
404 OS.flush();
406 TimersToPrint.clear();
409 void TimerGroup::prepareToPrintList(bool ResetTime) {
410 // See if any of our timers were started, if so add them to TimersToPrint.
411 for (Timer *T = FirstTimer; T; T = T->Next) {
412 if (!T->hasTriggered()) continue;
413 bool WasRunning = T->isRunning();
414 if (WasRunning)
415 T->stopTimer();
417 TimersToPrint.emplace_back(T->Time, T->Name, T->Description);
419 if (ResetTime)
420 T->clear();
422 if (WasRunning)
423 T->startTimer();
427 void TimerGroup::print(raw_ostream &OS, bool ResetAfterPrint) {
429 // After preparing the timers we can free the lock
430 sys::SmartScopedLock<true> L(*TimerLock);
431 prepareToPrintList(ResetAfterPrint);
434 // If any timers were started, print the group.
435 if (!TimersToPrint.empty())
436 PrintQueuedTimers(OS);
439 void TimerGroup::clear() {
440 sys::SmartScopedLock<true> L(*TimerLock);
441 for (Timer *T = FirstTimer; T; T = T->Next)
442 T->clear();
445 void TimerGroup::printAll(raw_ostream &OS) {
446 sys::SmartScopedLock<true> L(*TimerLock);
448 for (TimerGroup *TG = TimerGroupList; TG; TG = TG->Next)
449 TG->print(OS);
452 void TimerGroup::clearAll() {
453 sys::SmartScopedLock<true> L(*TimerLock);
454 for (TimerGroup *TG = TimerGroupList; TG; TG = TG->Next)
455 TG->clear();
458 void TimerGroup::printJSONValue(raw_ostream &OS, const PrintRecord &R,
459 const char *suffix, double Value) {
460 assert(yaml::needsQuotes(Name) == yaml::QuotingType::None &&
461 "TimerGroup name should not need quotes");
462 assert(yaml::needsQuotes(R.Name) == yaml::QuotingType::None &&
463 "Timer name should not need quotes");
464 constexpr auto max_digits10 = std::numeric_limits<double>::max_digits10;
465 OS << "\t\"time." << Name << '.' << R.Name << suffix
466 << "\": " << format("%.*e", max_digits10 - 1, Value);
469 const char *TimerGroup::printJSONValues(raw_ostream &OS, const char *delim) {
470 sys::SmartScopedLock<true> L(*TimerLock);
472 prepareToPrintList(false);
473 for (const PrintRecord &R : TimersToPrint) {
474 OS << delim;
475 delim = ",\n";
477 const TimeRecord &T = R.Time;
478 printJSONValue(OS, R, ".wall", T.getWallTime());
479 OS << delim;
480 printJSONValue(OS, R, ".user", T.getUserTime());
481 OS << delim;
482 printJSONValue(OS, R, ".sys", T.getSystemTime());
483 if (T.getMemUsed()) {
484 OS << delim;
485 printJSONValue(OS, R, ".mem", T.getMemUsed());
487 if (T.getInstructionsExecuted()) {
488 OS << delim;
489 printJSONValue(OS, R, ".instr", T.getInstructionsExecuted());
492 TimersToPrint.clear();
493 return delim;
496 const char *TimerGroup::printAllJSONValues(raw_ostream &OS, const char *delim) {
497 sys::SmartScopedLock<true> L(*TimerLock);
498 for (TimerGroup *TG = TimerGroupList; TG; TG = TG->Next)
499 delim = TG->printJSONValues(OS, delim);
500 return delim;
503 void TimerGroup::ConstructTimerLists() {
504 (void)*NamedGroupedTimers;
507 std::unique_ptr<TimerGroup> TimerGroup::aquireDefaultGroup() {
508 return std::unique_ptr<TimerGroup>(DefaultTimerGroup.claim());