xref: /freebsd/contrib/llvm-project/llvm/lib/Support/Timer.cpp (revision 13ec1e3155c7e9bf037b12af186351b7fa9b9450)
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.
10 //
11 //===----------------------------------------------------------------------===//
12 
13 #include "llvm/Support/Timer.h"
14 
15 #include "DebugOptions.h"
16 
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>
30 
31 #if HAVE_UNISTD_H
32 #include <unistd.h>
33 #endif
34 
35 #ifdef HAVE_PROC_PID_RUSAGE
36 #include <libproc.h>
37 #endif
38 
39 using namespace llvm;
40 
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;
51 }
52 
53 static ManagedStatic<sys::SmartMutex<true> > TimerLock;
54 
55 /// Allows llvm::Timer to emit signposts when supported.
56 static ManagedStatic<SignpostEmitter> Signposts;
57 
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);
65   }
66 };
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()));
74   }
75 };
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);
85   }
86 };
87 ManagedStatic<cl::opt<bool>, CreateSortTimers> SortTimers;
88 } // namespace
89 
90 void llvm::initTimerOptions() {
91   *TrackSpace;
92   *InfoOutputFilename;
93   *SortTimers;
94 }
95 
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.
102 
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;
112 
113   errs() << "Error opening info-output-file '"
114     << OutputFilename << " for appending!\n";
115   return std::make_unique<raw_fd_ostream>(2, false); // stderr.
116 }
117 
118 namespace {
119 struct CreateDefaultTimerGroup {
120   static void *call() {
121     return new TimerGroup("misc", "Miscellaneous Ungrouped Timers");
122   }
123 };
124 } // namespace
125 static ManagedStatic<TimerGroup, CreateDefaultTimerGroup> DefaultTimerGroup;
126 static TimerGroup *getDefaultTimerGroup() { return &*DefaultTimerGroup; }
127 
128 //===----------------------------------------------------------------------===//
129 // Timer Implementation
130 //===----------------------------------------------------------------------===//
131 
132 void Timer::init(StringRef TimerName, StringRef TimerDescription) {
133   init(TimerName, TimerDescription, *getDefaultTimerGroup());
134 }
135 
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);
144 }
145 
146 Timer::~Timer() {
147   if (!TG) return;  // Never initialized, or already cleared.
148   TG->removeTimer(*this);
149 }
150 
151 static inline size_t getMemUsage() {
152   if (!*TrackSpace)
153     return 0;
154   return sys::Process::GetMallocUsage();
155 }
156 
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;
163   }
164 #endif
165   return 0;
166 }
167 
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;
173 
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();
182   }
183 
184   Result.WallTime = Seconds(now.time_since_epoch()).count();
185   Result.UserTime = Seconds(user).count();
186   Result.SystemTime = Seconds(sys).count();
187   return Result;
188 }
189 
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);
195 }
196 
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);
203 }
204 
205 void Timer::clear() {
206   Running = Triggered = false;
207   Time = StartTime = TimeRecord();
208 }
209 
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);
215 }
216 
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);
225 
226   OS << "  ";
227 
228   if (Total.getMemUsed())
229     OS << format("%9" PRId64 "  ", (int64_t)getMemUsed());
230   if (Total.getInstructionsExecuted())
231     OS << format("%9" PRId64 "  ", (int64_t)getInstructionsExecuted());
232 }
233 
234 
235 //===----------------------------------------------------------------------===//
236 //   NamedRegionTimer Implementation
237 //===----------------------------------------------------------------------===//
238 
239 namespace {
240 
241 typedef StringMap<Timer> Name2TimerMap;
242 
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;
250   }
251 
252   Timer &get(StringRef Name, StringRef Description, StringRef GroupName,
253              StringRef GroupDescription) {
254     sys::SmartScopedLock<true> L(*TimerLock);
255 
256     std::pair<TimerGroup*, Name2TimerMap> &GroupEntry = Map[GroupName];
257 
258     if (!GroupEntry.first)
259       GroupEntry.first = new TimerGroup(GroupName, GroupDescription);
260 
261     Timer &T = GroupEntry.second[Name];
262     if (!T.isInitialized())
263       T.init(Name, Description, *GroupEntry.first);
264     return T;
265   }
266 };
267 
268 }
269 
270 static ManagedStatic<Name2PairMap> NamedGroupedTimers;
271 
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)) {}
278 
279 //===----------------------------------------------------------------------===//
280 //   TimerGroup Implementation
281 //===----------------------------------------------------------------------===//
282 
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;
286 
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;
297 }
298 
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");
307 }
308 
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);
314 
315   // Remove the group from the TimerGroupList.
316   sys::SmartScopedLock<true> L(*TimerLock);
317   *Prev = Next;
318   if (Next)
319     Next->Prev = Prev;
320 }
321 
322 
323 void TimerGroup::removeTimer(Timer &T) {
324   sys::SmartScopedLock<true> L(*TimerLock);
325 
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);
329 
330   T.TG = nullptr;
331 
332   // Unlink the timer from our list.
333   *T.Prev = T.Next;
334   if (T.Next)
335     T.Next->Prev = T.Prev;
336 
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;
341 
342   std::unique_ptr<raw_ostream> OutStream = CreateInfoOutputFile();
343   PrintQueuedTimers(*OutStream);
344 }
345 
346 void TimerGroup::addTimer(Timer &T) {
347   sys::SmartScopedLock<true> L(*TimerLock);
348 
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;
355 }
356 
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);
361 
362   TimeRecord Total;
363   for (const PrintRecord &Record : TimersToPrint)
364     Total += Record.Time;
365 
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";
373 
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';
381 
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";
394 
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';
400   }
401 
402   Total.print(Total, OS);
403   OS << "Total\n\n";
404   OS.flush();
405 
406   TimersToPrint.clear();
407 }
408 
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();
416 
417     TimersToPrint.emplace_back(T->Time, T->Name, T->Description);
418 
419     if (ResetTime)
420       T->clear();
421 
422     if (WasRunning)
423       T->startTimer();
424   }
425 }
426 
427 void TimerGroup::print(raw_ostream &OS, bool ResetAfterPrint) {
428   {
429     // After preparing the timers we can free the lock
430     sys::SmartScopedLock<true> L(*TimerLock);
431     prepareToPrintList(ResetAfterPrint);
432   }
433 
434   // If any timers were started, print the group.
435   if (!TimersToPrint.empty())
436     PrintQueuedTimers(OS);
437 }
438 
439 void TimerGroup::clear() {
440   sys::SmartScopedLock<true> L(*TimerLock);
441   for (Timer *T = FirstTimer; T; T = T->Next)
442     T->clear();
443 }
444 
445 void TimerGroup::printAll(raw_ostream &OS) {
446   sys::SmartScopedLock<true> L(*TimerLock);
447 
448   for (TimerGroup *TG = TimerGroupList; TG; TG = TG->Next)
449     TG->print(OS);
450 }
451 
452 void TimerGroup::clearAll() {
453   sys::SmartScopedLock<true> L(*TimerLock);
454   for (TimerGroup *TG = TimerGroupList; TG; TG = TG->Next)
455     TG->clear();
456 }
457 
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);
467 }
468 
469 const char *TimerGroup::printJSONValues(raw_ostream &OS, const char *delim) {
470   sys::SmartScopedLock<true> L(*TimerLock);
471 
472   prepareToPrintList(false);
473   for (const PrintRecord &R : TimersToPrint) {
474     OS << delim;
475     delim = ",\n";
476 
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());
486     }
487     if (T.getInstructionsExecuted()) {
488       OS << delim;
489       printJSONValue(OS, R, ".instr", T.getInstructionsExecuted());
490     }
491   }
492   TimersToPrint.clear();
493   return delim;
494 }
495 
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;
501 }
502 
503 void TimerGroup::ConstructTimerLists() {
504   (void)*NamedGroupedTimers;
505 }
506 
507 std::unique_ptr<TimerGroup> TimerGroup::aquireDefaultGroup() {
508   return std::unique_ptr<TimerGroup>(DefaultTimerGroup.claim());
509 }
510