xref: /src/contrib/llvm-project/llvm/lib/Support/TimeProfiler.cpp (revision 0fca6ea1d4eea4c934cfff25ac9ee8ad6fe95583)
1e6d15924SDimitry Andric //===-- TimeProfiler.cpp - Hierarchical Time Profiler ---------------------===//
2e6d15924SDimitry Andric //
3e6d15924SDimitry Andric // Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions.
4e6d15924SDimitry Andric // See https://llvm.org/LICENSE.txt for license information.
5e6d15924SDimitry Andric // SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception
6e6d15924SDimitry Andric //
7e6d15924SDimitry Andric //===----------------------------------------------------------------------===//
8e6d15924SDimitry Andric //
9e6d15924SDimitry Andric // This file implements hierarchical time profiler.
10e6d15924SDimitry Andric //
11e6d15924SDimitry Andric //===----------------------------------------------------------------------===//
12e6d15924SDimitry Andric 
13e6d15924SDimitry Andric #include "llvm/Support/TimeProfiler.h"
14ac9a064cSDimitry Andric #include "llvm/ADT/STLExtras.h"
156f8fc217SDimitry Andric #include "llvm/ADT/STLFunctionalExtras.h"
16e6d15924SDimitry Andric #include "llvm/ADT/StringMap.h"
17e6d15924SDimitry Andric #include "llvm/Support/JSON.h"
18706b4fc4SDimitry Andric #include "llvm/Support/Path.h"
19cfca06d7SDimitry Andric #include "llvm/Support/Process.h"
20cfca06d7SDimitry Andric #include "llvm/Support/Threading.h"
21cfca06d7SDimitry Andric #include <algorithm>
22e6d15924SDimitry Andric #include <cassert>
23e6d15924SDimitry Andric #include <chrono>
24ac9a064cSDimitry Andric #include <memory>
25cfca06d7SDimitry Andric #include <mutex>
26e6d15924SDimitry Andric #include <string>
27e6d15924SDimitry Andric #include <vector>
28e6d15924SDimitry Andric 
29cfca06d7SDimitry Andric using namespace llvm;
30e6d15924SDimitry Andric 
31e3b55780SDimitry Andric namespace {
32e3b55780SDimitry Andric 
33e3b55780SDimitry Andric using std::chrono::duration;
34e3b55780SDimitry Andric using std::chrono::duration_cast;
35e3b55780SDimitry Andric using std::chrono::microseconds;
36e3b55780SDimitry Andric using std::chrono::steady_clock;
37e3b55780SDimitry Andric using std::chrono::system_clock;
38e3b55780SDimitry Andric using std::chrono::time_point;
39e3b55780SDimitry Andric using std::chrono::time_point_cast;
40e3b55780SDimitry Andric 
41e3b55780SDimitry Andric struct TimeTraceProfilerInstances {
42e3b55780SDimitry Andric   std::mutex Lock;
43e3b55780SDimitry Andric   std::vector<TimeTraceProfiler *> List;
44e3b55780SDimitry Andric };
45e3b55780SDimitry Andric 
getTimeTraceProfilerInstances()46e3b55780SDimitry Andric TimeTraceProfilerInstances &getTimeTraceProfilerInstances() {
47e3b55780SDimitry Andric   static TimeTraceProfilerInstances Instances;
48e3b55780SDimitry Andric   return Instances;
49e3b55780SDimitry Andric }
50e3b55780SDimitry Andric 
51e3b55780SDimitry Andric } // anonymous namespace
52e3b55780SDimitry Andric 
53cfca06d7SDimitry Andric // Per Thread instance
54cfca06d7SDimitry Andric static LLVM_THREAD_LOCAL TimeTraceProfiler *TimeTraceProfilerInstance = nullptr;
55e6d15924SDimitry Andric 
getTimeTraceProfilerInstance()56cfca06d7SDimitry Andric TimeTraceProfiler *llvm::getTimeTraceProfilerInstance() {
57cfca06d7SDimitry Andric   return TimeTraceProfilerInstance;
58cfca06d7SDimitry Andric }
59e6d15924SDimitry Andric 
60cfca06d7SDimitry Andric namespace {
61e3b55780SDimitry Andric 
62e3b55780SDimitry Andric using ClockType = steady_clock;
63e3b55780SDimitry Andric using TimePointType = time_point<ClockType>;
64e3b55780SDimitry Andric using DurationType = duration<ClockType::rep, ClockType::period>;
65e3b55780SDimitry Andric using CountAndDurationType = std::pair<size_t, DurationType>;
66e3b55780SDimitry Andric using NameAndCountAndDurationType =
67e3b55780SDimitry Andric     std::pair<std::string, CountAndDurationType>;
68e3b55780SDimitry Andric 
69ac9a064cSDimitry Andric } // anonymous namespace
70ac9a064cSDimitry Andric 
71e3b55780SDimitry Andric /// Represents an open or completed time section entry to be captured.
72ac9a064cSDimitry Andric struct llvm::TimeTraceProfilerEntry {
73706b4fc4SDimitry Andric   const TimePointType Start;
741d5ae102SDimitry Andric   TimePointType End;
75706b4fc4SDimitry Andric   const std::string Name;
76ac9a064cSDimitry Andric   TimeTraceMetadata Metadata;
77ac9a064cSDimitry Andric 
78ac9a064cSDimitry Andric   const bool AsyncEvent = false;
TimeTraceProfilerEntryllvm::TimeTraceProfilerEntry79ac9a064cSDimitry Andric   TimeTraceProfilerEntry(TimePointType &&S, TimePointType &&E, std::string &&N,
80ac9a064cSDimitry Andric                          std::string &&Dt, bool Ae)
81ac9a064cSDimitry Andric       : Start(std::move(S)), End(std::move(E)), Name(std::move(N)), Metadata(),
82ac9a064cSDimitry Andric         AsyncEvent(Ae) {
83ac9a064cSDimitry Andric     Metadata.Detail = std::move(Dt);
84ac9a064cSDimitry Andric   }
85e6d15924SDimitry Andric 
TimeTraceProfilerEntryllvm::TimeTraceProfilerEntry86e3b55780SDimitry Andric   TimeTraceProfilerEntry(TimePointType &&S, TimePointType &&E, std::string &&N,
87ac9a064cSDimitry Andric                          TimeTraceMetadata &&Mt, bool Ae)
881d5ae102SDimitry Andric       : Start(std::move(S)), End(std::move(E)), Name(std::move(N)),
89ac9a064cSDimitry Andric         Metadata(std::move(Mt)), AsyncEvent(Ae) {}
901d5ae102SDimitry Andric 
911d5ae102SDimitry Andric   // Calculate timings for FlameGraph. Cast time points to microsecond precision
92e3b55780SDimitry Andric   // rather than casting duration. This avoids truncation issues causing inner
931d5ae102SDimitry Andric   // scopes overruning outer scopes.
getFlameGraphStartUsllvm::TimeTraceProfilerEntry94e3b55780SDimitry Andric   ClockType::rep getFlameGraphStartUs(TimePointType StartTime) const {
951d5ae102SDimitry Andric     return (time_point_cast<microseconds>(Start) -
961d5ae102SDimitry Andric             time_point_cast<microseconds>(StartTime))
971d5ae102SDimitry Andric         .count();
981d5ae102SDimitry Andric   }
991d5ae102SDimitry Andric 
getFlameGraphDurUsllvm::TimeTraceProfilerEntry100e3b55780SDimitry Andric   ClockType::rep getFlameGraphDurUs() const {
1011d5ae102SDimitry Andric     return (time_point_cast<microseconds>(End) -
1021d5ae102SDimitry Andric             time_point_cast<microseconds>(Start))
1031d5ae102SDimitry Andric         .count();
1041d5ae102SDimitry Andric   }
105e6d15924SDimitry Andric };
106e3b55780SDimitry Andric 
107cfca06d7SDimitry Andric struct llvm::TimeTraceProfiler {
TimeTraceProfilerllvm::TimeTraceProfiler108ac9a064cSDimitry Andric   TimeTraceProfiler(unsigned TimeTraceGranularity = 0, StringRef ProcName = "",
109ac9a064cSDimitry Andric                     bool TimeTraceVerbose = false)
110e3b55780SDimitry Andric       : BeginningOfTime(system_clock::now()), StartTime(ClockType::now()),
111cfca06d7SDimitry Andric         ProcName(ProcName), Pid(sys::Process::getProcessId()),
112ac9a064cSDimitry Andric         Tid(llvm::get_threadid()), TimeTraceGranularity(TimeTraceGranularity),
113ac9a064cSDimitry Andric         TimeTraceVerbose(TimeTraceVerbose) {
114cfca06d7SDimitry Andric     llvm::get_thread_name(ThreadName);
115cfca06d7SDimitry Andric   }
116e6d15924SDimitry Andric 
beginllvm::TimeTraceProfiler117ac9a064cSDimitry Andric   TimeTraceProfilerEntry *begin(std::string Name,
118ac9a064cSDimitry Andric                                 llvm::function_ref<std::string()> Detail,
119ac9a064cSDimitry Andric                                 bool AsyncEvent = false) {
120ac9a064cSDimitry Andric     Stack.emplace_back(std::make_unique<TimeTraceProfilerEntry>(
121ac9a064cSDimitry Andric         ClockType::now(), TimePointType(), std::move(Name), Detail(),
122ac9a064cSDimitry Andric         AsyncEvent));
123ac9a064cSDimitry Andric     return Stack.back().get();
124ac9a064cSDimitry Andric   }
125ac9a064cSDimitry Andric 
126ac9a064cSDimitry Andric   TimeTraceProfilerEntry *
beginllvm::TimeTraceProfiler127ac9a064cSDimitry Andric   begin(std::string Name, llvm::function_ref<TimeTraceMetadata()> Metadata,
128ac9a064cSDimitry Andric         bool AsyncEvent = false) {
129ac9a064cSDimitry Andric     Stack.emplace_back(std::make_unique<TimeTraceProfilerEntry>(
130ac9a064cSDimitry Andric         ClockType::now(), TimePointType(), std::move(Name), Metadata(),
131ac9a064cSDimitry Andric         AsyncEvent));
132ac9a064cSDimitry Andric     return Stack.back().get();
133e6d15924SDimitry Andric   }
134e6d15924SDimitry Andric 
endllvm::TimeTraceProfiler135e6d15924SDimitry Andric   void end() {
136e6d15924SDimitry Andric     assert(!Stack.empty() && "Must call begin() first");
137ac9a064cSDimitry Andric     end(*Stack.back());
138ac9a064cSDimitry Andric   }
139e6d15924SDimitry Andric 
endllvm::TimeTraceProfiler140ac9a064cSDimitry Andric   void end(TimeTraceProfilerEntry &E) {
141ac9a064cSDimitry Andric     assert(!Stack.empty() && "Must call begin() first");
142ac9a064cSDimitry Andric     E.End = ClockType::now();
1431d5ae102SDimitry Andric 
1441d5ae102SDimitry Andric     // Calculate duration at full precision for overall counts.
1451d5ae102SDimitry Andric     DurationType Duration = E.End - E.Start;
1461d5ae102SDimitry Andric 
1471d5ae102SDimitry Andric     // Only include sections longer or equal to TimeTraceGranularity msec.
1481d5ae102SDimitry Andric     if (duration_cast<microseconds>(Duration).count() >= TimeTraceGranularity)
149e6d15924SDimitry Andric       Entries.emplace_back(E);
150e6d15924SDimitry Andric 
151e6d15924SDimitry Andric     // Track total time taken by each "name", but only the topmost levels of
152e6d15924SDimitry Andric     // them; e.g. if there's a template instantiation that instantiates other
153e6d15924SDimitry Andric     // templates from within, we only want to add the topmost one. "topmost"
154e6d15924SDimitry Andric     // happens to be the ones that don't have any currently open entries above
155e6d15924SDimitry Andric     // itself.
156c0981da4SDimitry Andric     if (llvm::none_of(llvm::drop_begin(llvm::reverse(Stack)),
157ac9a064cSDimitry Andric                       [&](const std::unique_ptr<TimeTraceProfilerEntry> &Val) {
158ac9a064cSDimitry Andric                         return Val->Name == E.Name;
159e3b55780SDimitry Andric                       })) {
160e6d15924SDimitry Andric       auto &CountAndTotal = CountAndTotalPerName[E.Name];
161e6d15924SDimitry Andric       CountAndTotal.first++;
1621d5ae102SDimitry Andric       CountAndTotal.second += Duration;
163ac9a064cSDimitry Andric     };
164e6d15924SDimitry Andric 
165ac9a064cSDimitry Andric     llvm::erase_if(Stack,
166ac9a064cSDimitry Andric                    [&](const std::unique_ptr<TimeTraceProfilerEntry> &Val) {
167ac9a064cSDimitry Andric                      return Val.get() == &E;
168ac9a064cSDimitry Andric                    });
169e6d15924SDimitry Andric   }
170e6d15924SDimitry Andric 
171cfca06d7SDimitry Andric   // Write events from this TimeTraceProfilerInstance and
172cfca06d7SDimitry Andric   // ThreadTimeTraceProfilerInstances.
writellvm::TimeTraceProfiler173cfca06d7SDimitry Andric   void write(raw_pwrite_stream &OS) {
174cfca06d7SDimitry Andric     // Acquire Mutex as reading ThreadTimeTraceProfilerInstances.
175e3b55780SDimitry Andric     auto &Instances = getTimeTraceProfilerInstances();
176e3b55780SDimitry Andric     std::lock_guard<std::mutex> Lock(Instances.Lock);
177e6d15924SDimitry Andric     assert(Stack.empty() &&
178cfca06d7SDimitry Andric            "All profiler sections should be ended when calling write");
179e3b55780SDimitry Andric     assert(llvm::all_of(Instances.List,
180cfca06d7SDimitry Andric                         [](const auto &TTP) { return TTP->Stack.empty(); }) &&
181cfca06d7SDimitry Andric            "All profiler sections should be ended when calling write");
182cfca06d7SDimitry Andric 
183e6d15924SDimitry Andric     json::OStream J(OS);
184e6d15924SDimitry Andric     J.objectBegin();
185e6d15924SDimitry Andric     J.attributeBegin("traceEvents");
186e6d15924SDimitry Andric     J.arrayBegin();
187e6d15924SDimitry Andric 
188e6d15924SDimitry Andric     // Emit all events for the main flame graph.
189cfca06d7SDimitry Andric     auto writeEvent = [&](const auto &E, uint64_t Tid) {
1901d5ae102SDimitry Andric       auto StartUs = E.getFlameGraphStartUs(StartTime);
1911d5ae102SDimitry Andric       auto DurUs = E.getFlameGraphDurUs();
192e6d15924SDimitry Andric 
193e6d15924SDimitry Andric       J.object([&] {
194cfca06d7SDimitry Andric         J.attribute("pid", Pid);
195cfca06d7SDimitry Andric         J.attribute("tid", int64_t(Tid));
196e6d15924SDimitry Andric         J.attribute("ts", StartUs);
197ac9a064cSDimitry Andric         if (E.AsyncEvent) {
198ac9a064cSDimitry Andric           J.attribute("cat", E.Name);
199ac9a064cSDimitry Andric           J.attribute("ph", "b");
200ac9a064cSDimitry Andric           J.attribute("id", 0);
201ac9a064cSDimitry Andric         } else {
202ac9a064cSDimitry Andric           J.attribute("ph", "X");
203e6d15924SDimitry Andric           J.attribute("dur", DurUs);
204ac9a064cSDimitry Andric         }
205e6d15924SDimitry Andric         J.attribute("name", E.Name);
206ac9a064cSDimitry Andric         if (!E.Metadata.isEmpty()) {
207ac9a064cSDimitry Andric           J.attributeObject("args", [&] {
208ac9a064cSDimitry Andric             if (!E.Metadata.Detail.empty())
209ac9a064cSDimitry Andric               J.attribute("detail", E.Metadata.Detail);
210ac9a064cSDimitry Andric             if (!E.Metadata.File.empty())
211ac9a064cSDimitry Andric               J.attribute("file", E.Metadata.File);
212ac9a064cSDimitry Andric             if (E.Metadata.Line > 0)
213ac9a064cSDimitry Andric               J.attribute("line", E.Metadata.Line);
214ac9a064cSDimitry Andric           });
215706b4fc4SDimitry Andric         }
216e6d15924SDimitry Andric       });
217ac9a064cSDimitry Andric 
218ac9a064cSDimitry Andric       if (E.AsyncEvent) {
219ac9a064cSDimitry Andric         J.object([&] {
220ac9a064cSDimitry Andric           J.attribute("pid", Pid);
221ac9a064cSDimitry Andric           J.attribute("tid", int64_t(Tid));
222ac9a064cSDimitry Andric           J.attribute("ts", StartUs + DurUs);
223ac9a064cSDimitry Andric           J.attribute("cat", E.Name);
224ac9a064cSDimitry Andric           J.attribute("ph", "e");
225ac9a064cSDimitry Andric           J.attribute("id", 0);
226ac9a064cSDimitry Andric           J.attribute("name", E.Name);
227ac9a064cSDimitry Andric         });
228ac9a064cSDimitry Andric       }
229cfca06d7SDimitry Andric     };
230e3b55780SDimitry Andric     for (const TimeTraceProfilerEntry &E : Entries)
231cfca06d7SDimitry Andric       writeEvent(E, this->Tid);
232e3b55780SDimitry Andric     for (const TimeTraceProfiler *TTP : Instances.List)
233e3b55780SDimitry Andric       for (const TimeTraceProfilerEntry &E : TTP->Entries)
234cfca06d7SDimitry Andric         writeEvent(E, TTP->Tid);
235e6d15924SDimitry Andric 
236e6d15924SDimitry Andric     // Emit totals by section name as additional "thread" events, sorted from
237e6d15924SDimitry Andric     // longest one.
238cfca06d7SDimitry Andric     // Find highest used thread id.
239cfca06d7SDimitry Andric     uint64_t MaxTid = this->Tid;
240e3b55780SDimitry Andric     for (const TimeTraceProfiler *TTP : Instances.List)
241cfca06d7SDimitry Andric       MaxTid = std::max(MaxTid, TTP->Tid);
242e6d15924SDimitry Andric 
243cfca06d7SDimitry Andric     // Combine all CountAndTotalPerName from threads into one.
244cfca06d7SDimitry Andric     StringMap<CountAndDurationType> AllCountAndTotalPerName;
245cfca06d7SDimitry Andric     auto combineStat = [&](const auto &Stat) {
246cfca06d7SDimitry Andric       StringRef Key = Stat.getKey();
247cfca06d7SDimitry Andric       auto Value = Stat.getValue();
248cfca06d7SDimitry Andric       auto &CountAndTotal = AllCountAndTotalPerName[Key];
249cfca06d7SDimitry Andric       CountAndTotal.first += Value.first;
250cfca06d7SDimitry Andric       CountAndTotal.second += Value.second;
251cfca06d7SDimitry Andric     };
252cfca06d7SDimitry Andric     for (const auto &Stat : CountAndTotalPerName)
253cfca06d7SDimitry Andric       combineStat(Stat);
254e3b55780SDimitry Andric     for (const TimeTraceProfiler *TTP : Instances.List)
255cfca06d7SDimitry Andric       for (const auto &Stat : TTP->CountAndTotalPerName)
256cfca06d7SDimitry Andric         combineStat(Stat);
257cfca06d7SDimitry Andric 
258cfca06d7SDimitry Andric     std::vector<NameAndCountAndDurationType> SortedTotals;
259cfca06d7SDimitry Andric     SortedTotals.reserve(AllCountAndTotalPerName.size());
260cfca06d7SDimitry Andric     for (const auto &Total : AllCountAndTotalPerName)
261cfca06d7SDimitry Andric       SortedTotals.emplace_back(std::string(Total.getKey()), Total.getValue());
262cfca06d7SDimitry Andric 
263cfca06d7SDimitry Andric     llvm::sort(SortedTotals, [](const NameAndCountAndDurationType &A,
264e6d15924SDimitry Andric                                 const NameAndCountAndDurationType &B) {
265e6d15924SDimitry Andric       return A.second.second > B.second.second;
266e6d15924SDimitry Andric     });
267cfca06d7SDimitry Andric 
268cfca06d7SDimitry Andric     // Report totals on separate threads of tracing file.
269cfca06d7SDimitry Andric     uint64_t TotalTid = MaxTid + 1;
270cfca06d7SDimitry Andric     for (const NameAndCountAndDurationType &Total : SortedTotals) {
271cfca06d7SDimitry Andric       auto DurUs = duration_cast<microseconds>(Total.second.second).count();
272cfca06d7SDimitry Andric       auto Count = AllCountAndTotalPerName[Total.first].first;
273e6d15924SDimitry Andric 
274e6d15924SDimitry Andric       J.object([&] {
275cfca06d7SDimitry Andric         J.attribute("pid", Pid);
276cfca06d7SDimitry Andric         J.attribute("tid", int64_t(TotalTid));
277e6d15924SDimitry Andric         J.attribute("ph", "X");
278e6d15924SDimitry Andric         J.attribute("ts", 0);
279e6d15924SDimitry Andric         J.attribute("dur", DurUs);
280cfca06d7SDimitry Andric         J.attribute("name", "Total " + Total.first);
281e6d15924SDimitry Andric         J.attributeObject("args", [&] {
282e6d15924SDimitry Andric           J.attribute("count", int64_t(Count));
283e6d15924SDimitry Andric           J.attribute("avg ms", int64_t(DurUs / Count / 1000));
284e6d15924SDimitry Andric         });
285e6d15924SDimitry Andric       });
286e6d15924SDimitry Andric 
287cfca06d7SDimitry Andric       ++TotalTid;
288e6d15924SDimitry Andric     }
289e6d15924SDimitry Andric 
290cfca06d7SDimitry Andric     auto writeMetadataEvent = [&](const char *Name, uint64_t Tid,
291cfca06d7SDimitry Andric                                   StringRef arg) {
292e6d15924SDimitry Andric       J.object([&] {
293e6d15924SDimitry Andric         J.attribute("cat", "");
294cfca06d7SDimitry Andric         J.attribute("pid", Pid);
295cfca06d7SDimitry Andric         J.attribute("tid", int64_t(Tid));
296e6d15924SDimitry Andric         J.attribute("ts", 0);
297e6d15924SDimitry Andric         J.attribute("ph", "M");
298cfca06d7SDimitry Andric         J.attribute("name", Name);
299cfca06d7SDimitry Andric         J.attributeObject("args", [&] { J.attribute("name", arg); });
300e6d15924SDimitry Andric       });
301cfca06d7SDimitry Andric     };
302cfca06d7SDimitry Andric 
303cfca06d7SDimitry Andric     writeMetadataEvent("process_name", Tid, ProcName);
304cfca06d7SDimitry Andric     writeMetadataEvent("thread_name", Tid, ThreadName);
305e3b55780SDimitry Andric     for (const TimeTraceProfiler *TTP : Instances.List)
306cfca06d7SDimitry Andric       writeMetadataEvent("thread_name", TTP->Tid, TTP->ThreadName);
307e6d15924SDimitry Andric 
308e6d15924SDimitry Andric     J.arrayEnd();
309e6d15924SDimitry Andric     J.attributeEnd();
310cfca06d7SDimitry Andric 
311cfca06d7SDimitry Andric     // Emit the absolute time when this TimeProfiler started.
312cfca06d7SDimitry Andric     // This can be used to combine the profiling data from
313cfca06d7SDimitry Andric     // multiple processes and preserve actual time intervals.
314cfca06d7SDimitry Andric     J.attribute("beginningOfTime",
315cfca06d7SDimitry Andric                 time_point_cast<microseconds>(BeginningOfTime)
316cfca06d7SDimitry Andric                     .time_since_epoch()
317cfca06d7SDimitry Andric                     .count());
318cfca06d7SDimitry Andric 
319e6d15924SDimitry Andric     J.objectEnd();
320e6d15924SDimitry Andric   }
321e6d15924SDimitry Andric 
322ac9a064cSDimitry Andric   SmallVector<std::unique_ptr<TimeTraceProfilerEntry>, 16> Stack;
323e3b55780SDimitry Andric   SmallVector<TimeTraceProfilerEntry, 128> Entries;
324e6d15924SDimitry Andric   StringMap<CountAndDurationType> CountAndTotalPerName;
325e3b55780SDimitry Andric   // System clock time when the session was begun.
326cfca06d7SDimitry Andric   const time_point<system_clock> BeginningOfTime;
327e3b55780SDimitry Andric   // Profiling clock time when the session was begun.
328706b4fc4SDimitry Andric   const TimePointType StartTime;
329706b4fc4SDimitry Andric   const std::string ProcName;
330cfca06d7SDimitry Andric   const sys::Process::Pid Pid;
331cfca06d7SDimitry Andric   SmallString<0> ThreadName;
332cfca06d7SDimitry Andric   const uint64_t Tid;
3331d5ae102SDimitry Andric 
3341d5ae102SDimitry Andric   // Minimum time granularity (in microseconds)
335706b4fc4SDimitry Andric   const unsigned TimeTraceGranularity;
336ac9a064cSDimitry Andric 
337ac9a064cSDimitry Andric   // Make time trace capture verbose event details (e.g. source filenames). This
338ac9a064cSDimitry Andric   // can increase the size of the output by 2-3 times.
339ac9a064cSDimitry Andric   const bool TimeTraceVerbose;
340e6d15924SDimitry Andric };
341e6d15924SDimitry Andric 
isTimeTraceVerbose()342ac9a064cSDimitry Andric bool llvm::isTimeTraceVerbose() {
343ac9a064cSDimitry Andric   return getTimeTraceProfilerInstance() &&
344ac9a064cSDimitry Andric          getTimeTraceProfilerInstance()->TimeTraceVerbose;
345ac9a064cSDimitry Andric }
346ac9a064cSDimitry Andric 
timeTraceProfilerInitialize(unsigned TimeTraceGranularity,StringRef ProcName,bool TimeTraceVerbose)347cfca06d7SDimitry Andric void llvm::timeTraceProfilerInitialize(unsigned TimeTraceGranularity,
348ac9a064cSDimitry Andric                                        StringRef ProcName,
349ac9a064cSDimitry Andric                                        bool TimeTraceVerbose) {
350e6d15924SDimitry Andric   assert(TimeTraceProfilerInstance == nullptr &&
351e6d15924SDimitry Andric          "Profiler should not be initialized");
352706b4fc4SDimitry Andric   TimeTraceProfilerInstance = new TimeTraceProfiler(
353ac9a064cSDimitry Andric       TimeTraceGranularity, llvm::sys::path::filename(ProcName),
354ac9a064cSDimitry Andric       TimeTraceVerbose);
355e6d15924SDimitry Andric }
356e6d15924SDimitry Andric 
357cfca06d7SDimitry Andric // Removes all TimeTraceProfilerInstances.
358cfca06d7SDimitry Andric // Called from main thread.
timeTraceProfilerCleanup()359cfca06d7SDimitry Andric void llvm::timeTraceProfilerCleanup() {
360e6d15924SDimitry Andric   delete TimeTraceProfilerInstance;
361c0981da4SDimitry Andric   TimeTraceProfilerInstance = nullptr;
362e3b55780SDimitry Andric 
363e3b55780SDimitry Andric   auto &Instances = getTimeTraceProfilerInstances();
364e3b55780SDimitry Andric   std::lock_guard<std::mutex> Lock(Instances.Lock);
365e3b55780SDimitry Andric   for (auto *TTP : Instances.List)
366cfca06d7SDimitry Andric     delete TTP;
367e3b55780SDimitry Andric   Instances.List.clear();
368cfca06d7SDimitry Andric }
369cfca06d7SDimitry Andric 
370cfca06d7SDimitry Andric // Finish TimeTraceProfilerInstance on a worker thread.
371cfca06d7SDimitry Andric // This doesn't remove the instance, just moves the pointer to global vector.
timeTraceProfilerFinishThread()372cfca06d7SDimitry Andric void llvm::timeTraceProfilerFinishThread() {
373e3b55780SDimitry Andric   auto &Instances = getTimeTraceProfilerInstances();
374e3b55780SDimitry Andric   std::lock_guard<std::mutex> Lock(Instances.Lock);
375e3b55780SDimitry Andric   Instances.List.push_back(TimeTraceProfilerInstance);
376e6d15924SDimitry Andric   TimeTraceProfilerInstance = nullptr;
377e6d15924SDimitry Andric }
378e6d15924SDimitry Andric 
timeTraceProfilerWrite(raw_pwrite_stream & OS)379cfca06d7SDimitry Andric void llvm::timeTraceProfilerWrite(raw_pwrite_stream &OS) {
380e6d15924SDimitry Andric   assert(TimeTraceProfilerInstance != nullptr &&
381e6d15924SDimitry Andric          "Profiler object can't be null");
382cfca06d7SDimitry Andric   TimeTraceProfilerInstance->write(OS);
383e6d15924SDimitry Andric }
384e6d15924SDimitry Andric 
timeTraceProfilerWrite(StringRef PreferredFileName,StringRef FallbackFileName)385cfca06d7SDimitry Andric Error llvm::timeTraceProfilerWrite(StringRef PreferredFileName,
386cfca06d7SDimitry Andric                                    StringRef FallbackFileName) {
387cfca06d7SDimitry Andric   assert(TimeTraceProfilerInstance != nullptr &&
388cfca06d7SDimitry Andric          "Profiler object can't be null");
389cfca06d7SDimitry Andric 
390cfca06d7SDimitry Andric   std::string Path = PreferredFileName.str();
391cfca06d7SDimitry Andric   if (Path.empty()) {
392cfca06d7SDimitry Andric     Path = FallbackFileName == "-" ? "out" : FallbackFileName.str();
393cfca06d7SDimitry Andric     Path += ".time-trace";
394cfca06d7SDimitry Andric   }
395cfca06d7SDimitry Andric 
396cfca06d7SDimitry Andric   std::error_code EC;
397344a3780SDimitry Andric   raw_fd_ostream OS(Path, EC, sys::fs::OF_TextWithCRLF);
398cfca06d7SDimitry Andric   if (EC)
399cfca06d7SDimitry Andric     return createStringError(EC, "Could not open " + Path);
400cfca06d7SDimitry Andric 
401cfca06d7SDimitry Andric   timeTraceProfilerWrite(OS);
402cfca06d7SDimitry Andric   return Error::success();
403cfca06d7SDimitry Andric }
404cfca06d7SDimitry Andric 
timeTraceProfilerBegin(StringRef Name,StringRef Detail)405ac9a064cSDimitry Andric TimeTraceProfilerEntry *llvm::timeTraceProfilerBegin(StringRef Name,
406ac9a064cSDimitry Andric                                                      StringRef Detail) {
407e6d15924SDimitry Andric   if (TimeTraceProfilerInstance != nullptr)
408ac9a064cSDimitry Andric     return TimeTraceProfilerInstance->begin(
409ac9a064cSDimitry Andric         std::string(Name), [&]() { return std::string(Detail); }, false);
410ac9a064cSDimitry Andric   return nullptr;
411e6d15924SDimitry Andric }
412e6d15924SDimitry Andric 
413ac9a064cSDimitry Andric TimeTraceProfilerEntry *
timeTraceProfilerBegin(StringRef Name,llvm::function_ref<std::string ()> Detail)414ac9a064cSDimitry Andric llvm::timeTraceProfilerBegin(StringRef Name,
415e6d15924SDimitry Andric                              llvm::function_ref<std::string()> Detail) {
416e6d15924SDimitry Andric   if (TimeTraceProfilerInstance != nullptr)
417ac9a064cSDimitry Andric     return TimeTraceProfilerInstance->begin(std::string(Name), Detail, false);
418ac9a064cSDimitry Andric   return nullptr;
419ac9a064cSDimitry Andric }
420ac9a064cSDimitry Andric 
421ac9a064cSDimitry Andric TimeTraceProfilerEntry *
timeTraceProfilerBegin(StringRef Name,llvm::function_ref<TimeTraceMetadata ()> Metadata)422ac9a064cSDimitry Andric llvm::timeTraceProfilerBegin(StringRef Name,
423ac9a064cSDimitry Andric                              llvm::function_ref<TimeTraceMetadata()> Metadata) {
424ac9a064cSDimitry Andric   if (TimeTraceProfilerInstance != nullptr)
425ac9a064cSDimitry Andric     return TimeTraceProfilerInstance->begin(std::string(Name), Metadata, false);
426ac9a064cSDimitry Andric   return nullptr;
427ac9a064cSDimitry Andric }
428ac9a064cSDimitry Andric 
timeTraceAsyncProfilerBegin(StringRef Name,StringRef Detail)429ac9a064cSDimitry Andric TimeTraceProfilerEntry *llvm::timeTraceAsyncProfilerBegin(StringRef Name,
430ac9a064cSDimitry Andric                                                           StringRef Detail) {
431ac9a064cSDimitry Andric   if (TimeTraceProfilerInstance != nullptr)
432ac9a064cSDimitry Andric     return TimeTraceProfilerInstance->begin(
433ac9a064cSDimitry Andric         std::string(Name), [&]() { return std::string(Detail); }, true);
434ac9a064cSDimitry Andric   return nullptr;
435e6d15924SDimitry Andric }
436e6d15924SDimitry Andric 
timeTraceProfilerEnd()437cfca06d7SDimitry Andric void llvm::timeTraceProfilerEnd() {
438e6d15924SDimitry Andric   if (TimeTraceProfilerInstance != nullptr)
439e6d15924SDimitry Andric     TimeTraceProfilerInstance->end();
440e6d15924SDimitry Andric }
441ac9a064cSDimitry Andric 
timeTraceProfilerEnd(TimeTraceProfilerEntry * E)442ac9a064cSDimitry Andric void llvm::timeTraceProfilerEnd(TimeTraceProfilerEntry *E) {
443ac9a064cSDimitry Andric   if (TimeTraceProfilerInstance != nullptr)
444ac9a064cSDimitry Andric     TimeTraceProfilerInstance->end(*E);
445ac9a064cSDimitry Andric }
446