1 //===- xray-account.h - XRay Function Call Accounting ---------------------===//
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 // This file implements basic function call accounting from an XRay trace.
10 //
11 //===----------------------------------------------------------------------===//
12 
13 #include <algorithm>
14 #include <cassert>
15 #include <numeric>
16 #include <system_error>
17 #include <utility>
18 
19 #include "xray-account.h"
20 #include "xray-registry.h"
21 #include "llvm/Support/ErrorHandling.h"
22 #include "llvm/Support/FormatVariadic.h"
23 #include "llvm/XRay/InstrumentationMap.h"
24 #include "llvm/XRay/Trace.h"
25 
26 #include <cmath>
27 
28 using namespace llvm;
29 using namespace llvm::xray;
30 
31 static cl::SubCommand Account("account", "Function call accounting");
32 static cl::opt<std::string> AccountInput(cl::Positional,
33                                          cl::desc("<xray log file>"),
34                                          cl::Required, cl::sub(Account));
35 static cl::opt<bool>
36     AccountKeepGoing("keep-going", cl::desc("Keep going on errors encountered"),
37                      cl::sub(Account), cl::init(false));
38 static cl::alias AccountKeepGoing2("k", cl::aliasopt(AccountKeepGoing),
39                                    cl::desc("Alias for -keep_going"));
40 static cl::opt<bool> AccountRecursiveCallsOnly(
41     "recursive-calls-only", cl::desc("Only count the calls that are recursive"),
42     cl::sub(Account), cl::init(false));
43 static cl::opt<bool> AccountDeduceSiblingCalls(
44     "deduce-sibling-calls",
45     cl::desc("Deduce sibling calls when unrolling function call stacks"),
46     cl::sub(Account), cl::init(false));
47 static cl::alias
48     AccountDeduceSiblingCalls2("d", cl::aliasopt(AccountDeduceSiblingCalls),
49                                cl::desc("Alias for -deduce_sibling_calls"));
50 static cl::opt<std::string>
51     AccountOutput("output", cl::value_desc("output file"), cl::init("-"),
52                   cl::desc("output file; use '-' for stdout"),
53                   cl::sub(Account));
54 static cl::alias AccountOutput2("o", cl::aliasopt(AccountOutput),
55                                 cl::desc("Alias for -output"));
56 enum class AccountOutputFormats { TEXT, CSV };
57 static cl::opt<AccountOutputFormats>
58     AccountOutputFormat("format", cl::desc("output format"),
59                         cl::values(clEnumValN(AccountOutputFormats::TEXT,
60                                               "text", "report stats in text"),
61                                    clEnumValN(AccountOutputFormats::CSV, "csv",
62                                               "report stats in csv")),
63                         cl::sub(Account));
64 static cl::alias AccountOutputFormat2("f", cl::desc("Alias of -format"),
65                                       cl::aliasopt(AccountOutputFormat));
66 
67 enum class SortField {
68   FUNCID,
69   COUNT,
70   MIN,
71   MED,
72   PCT90,
73   PCT99,
74   MAX,
75   SUM,
76   FUNC,
77 };
78 
79 static cl::opt<SortField> AccountSortOutput(
80     "sort", cl::desc("sort output by this field"), cl::value_desc("field"),
81     cl::sub(Account), cl::init(SortField::FUNCID),
82     cl::values(clEnumValN(SortField::FUNCID, "funcid", "function id"),
83                clEnumValN(SortField::COUNT, "count", "function call counts"),
84                clEnumValN(SortField::MIN, "min", "minimum function durations"),
85                clEnumValN(SortField::MED, "med", "median function durations"),
86                clEnumValN(SortField::PCT90, "90p", "90th percentile durations"),
87                clEnumValN(SortField::PCT99, "99p", "99th percentile durations"),
88                clEnumValN(SortField::MAX, "max", "maximum function durations"),
89                clEnumValN(SortField::SUM, "sum", "sum of call durations"),
90                clEnumValN(SortField::FUNC, "func", "function names")));
91 static cl::alias AccountSortOutput2("s", cl::aliasopt(AccountSortOutput),
92                                     cl::desc("Alias for -sort"));
93 
94 enum class SortDirection {
95   ASCENDING,
96   DESCENDING,
97 };
98 static cl::opt<SortDirection> AccountSortOrder(
99     "sortorder", cl::desc("sort ordering"), cl::init(SortDirection::ASCENDING),
100     cl::values(clEnumValN(SortDirection::ASCENDING, "asc", "ascending"),
101                clEnumValN(SortDirection::DESCENDING, "dsc", "descending")),
102     cl::sub(Account));
103 static cl::alias AccountSortOrder2("r", cl::aliasopt(AccountSortOrder),
104                                    cl::desc("Alias for -sortorder"));
105 
106 static cl::opt<int> AccountTop("top", cl::desc("only show the top N results"),
107                                cl::value_desc("N"), cl::sub(Account),
108                                cl::init(-1));
109 static cl::alias AccountTop2("p", cl::desc("Alias for -top"),
110                              cl::aliasopt(AccountTop));
111 
112 static cl::opt<std::string>
113     AccountInstrMap("instr_map",
114                     cl::desc("binary with the instrumentation map, or "
115                              "a separate instrumentation map"),
116                     cl::value_desc("binary with xray_instr_map"),
117                     cl::sub(Account), cl::init(""));
118 static cl::alias AccountInstrMap2("m", cl::aliasopt(AccountInstrMap),
119                                   cl::desc("Alias for -instr_map"));
120 
121 namespace {
122 
123 template <class T, class U> void setMinMax(std::pair<T, T> &MM, U &&V) {
124   if (MM.first == 0 || MM.second == 0)
125     MM = std::make_pair(std::forward<U>(V), std::forward<U>(V));
126   else
127     MM = std::make_pair(std::min(MM.first, V), std::max(MM.second, V));
128 }
129 
130 template <class T> T diff(T L, T R) { return std::max(L, R) - std::min(L, R); }
131 
132 } // namespace
133 
134 using RecursionStatus = LatencyAccountant::FunctionStack::RecursionStatus;
135 RecursionStatus &RecursionStatus::operator++() {
136   auto Depth = Bitfield::get<RecursionStatus::Depth>(Storage);
137   assert(Depth >= 0 && Depth < std::numeric_limits<decltype(Depth)>::max());
138   ++Depth;
139   Bitfield::set<RecursionStatus::Depth>(Storage, Depth); // ++Storage
140   // Did this function just (maybe indirectly) call itself the first time?
141   if (!isRecursive() && Depth == 2) // Storage == 2  /  Storage s> 1
142     Bitfield::set<RecursionStatus::IsRecursive>(Storage,
143                                                 true); // Storage |= INT_MIN
144   return *this;
145 }
146 RecursionStatus &RecursionStatus::operator--() {
147   auto Depth = Bitfield::get<RecursionStatus::Depth>(Storage);
148   assert(Depth > 0);
149   --Depth;
150   Bitfield::set<RecursionStatus::Depth>(Storage, Depth); // --Storage
151   // Did we leave a function that previouly (maybe indirectly) called itself?
152   if (isRecursive() && Depth == 0) // Storage == INT_MIN
153     Bitfield::set<RecursionStatus::IsRecursive>(Storage, false); // Storage = 0
154   return *this;
155 }
156 bool RecursionStatus::isRecursive() const {
157   return Bitfield::get<RecursionStatus::IsRecursive>(Storage); // Storage s< 0
158 }
159 
160 bool LatencyAccountant::accountRecord(const XRayRecord &Record) {
161   setMinMax(PerThreadMinMaxTSC[Record.TId], Record.TSC);
162   setMinMax(PerCPUMinMaxTSC[Record.CPU], Record.TSC);
163 
164   if (CurrentMaxTSC == 0)
165     CurrentMaxTSC = Record.TSC;
166 
167   if (Record.TSC < CurrentMaxTSC)
168     return false;
169 
170   auto &ThreadStack = PerThreadFunctionStack[Record.TId];
171   if (RecursiveCallsOnly && !ThreadStack.RecursionDepth)
172     ThreadStack.RecursionDepth.emplace();
173   switch (Record.Type) {
174   case RecordTypes::CUSTOM_EVENT:
175   case RecordTypes::TYPED_EVENT:
176     // TODO: Support custom and typed event accounting in the future.
177     return true;
178   case RecordTypes::ENTER:
179   case RecordTypes::ENTER_ARG: {
180     ThreadStack.Stack.emplace_back(Record.FuncId, Record.TSC);
181     if (ThreadStack.RecursionDepth)
182       ++(*ThreadStack.RecursionDepth)[Record.FuncId];
183     break;
184   }
185   case RecordTypes::EXIT:
186   case RecordTypes::TAIL_EXIT: {
187     if (ThreadStack.Stack.empty())
188       return false;
189 
190     if (ThreadStack.Stack.back().first == Record.FuncId) {
191       const auto &Top = ThreadStack.Stack.back();
192       if (!ThreadStack.RecursionDepth ||
193           (*ThreadStack.RecursionDepth)[Top.first].isRecursive())
194         recordLatency(Top.first, diff(Top.second, Record.TSC));
195       if (ThreadStack.RecursionDepth)
196         --(*ThreadStack.RecursionDepth)[Top.first];
197       ThreadStack.Stack.pop_back();
198       break;
199     }
200 
201     if (!DeduceSiblingCalls)
202       return false;
203 
204     // Look for the parent up the stack.
205     auto Parent =
206         llvm::find_if(llvm::reverse(ThreadStack.Stack),
207                       [&](const std::pair<const int32_t, uint64_t> &E) {
208                         return E.first == Record.FuncId;
209                       });
210     if (Parent == ThreadStack.Stack.rend())
211       return false;
212 
213     // Account time for this apparently sibling call exit up the stack.
214     // Considering the following case:
215     //
216     //   f()
217     //    g()
218     //      h()
219     //
220     // We might only ever see the following entries:
221     //
222     //   -> f()
223     //   -> g()
224     //   -> h()
225     //   <- h()
226     //   <- f()
227     //
228     // Now we don't see the exit to g() because some older version of the XRay
229     // runtime wasn't instrumenting tail exits. If we don't deduce tail calls,
230     // we may potentially never account time for g() -- and this code would have
231     // already bailed out, because `<- f()` doesn't match the current "top" of
232     // stack where we're waiting for the exit to `g()` instead. This is not
233     // ideal and brittle -- so instead we provide a potentially inaccurate
234     // accounting of g() instead, computing it from the exit of f().
235     //
236     // While it might be better that we account the time between `-> g()` and
237     // `-> h()` as the proper accounting of time for g() here, this introduces
238     // complexity to do correctly (need to backtrack, etc.).
239     //
240     // FIXME: Potentially implement the more complex deduction algorithm?
241     auto R = make_range(std::next(Parent).base(), ThreadStack.Stack.end());
242     for (auto &E : R) {
243       if (!ThreadStack.RecursionDepth ||
244           (*ThreadStack.RecursionDepth)[E.first].isRecursive())
245         recordLatency(E.first, diff(E.second, Record.TSC));
246     }
247     for (auto &Top : reverse(R)) {
248       if (ThreadStack.RecursionDepth)
249         --(*ThreadStack.RecursionDepth)[Top.first];
250       ThreadStack.Stack.pop_back();
251     }
252     break;
253   }
254   }
255 
256   return true;
257 }
258 
259 namespace {
260 
261 // We consolidate the data into a struct which we can output in various forms.
262 struct ResultRow {
263   uint64_t Count;
264   double Min;
265   double Median;
266   double Pct90;
267   double Pct99;
268   double Max;
269   double Sum;
270   std::string DebugInfo;
271   std::string Function;
272 };
273 
274 ResultRow getStats(MutableArrayRef<uint64_t> Timings) {
275   assert(!Timings.empty());
276   ResultRow R;
277   R.Sum = std::accumulate(Timings.begin(), Timings.end(), 0.0);
278   auto MinMax = std::minmax_element(Timings.begin(), Timings.end());
279   R.Min = *MinMax.first;
280   R.Max = *MinMax.second;
281   R.Count = Timings.size();
282 
283   auto MedianOff = Timings.size() / 2;
284   std::nth_element(Timings.begin(), Timings.begin() + MedianOff, Timings.end());
285   R.Median = Timings[MedianOff];
286 
287   auto Pct90Off = std::floor(Timings.size() * 0.9);
288   std::nth_element(Timings.begin(), Timings.begin() + (uint64_t)Pct90Off,
289                    Timings.end());
290   R.Pct90 = Timings[Pct90Off];
291 
292   auto Pct99Off = std::floor(Timings.size() * 0.99);
293   std::nth_element(Timings.begin(), Timings.begin() + (uint64_t)Pct99Off,
294                    Timings.end());
295   R.Pct99 = Timings[Pct99Off];
296   return R;
297 }
298 
299 } // namespace
300 
301 using TupleType = std::tuple<int32_t, uint64_t, ResultRow>;
302 
303 template <typename F>
304 static void sortByKey(std::vector<TupleType> &Results, F Fn) {
305   bool ASC = AccountSortOrder == SortDirection::ASCENDING;
306   llvm::sort(Results, [=](const TupleType &L, const TupleType &R) {
307     return ASC ? Fn(L) < Fn(R) : Fn(L) > Fn(R);
308   });
309 }
310 
311 template <class F>
312 void LatencyAccountant::exportStats(const XRayFileHeader &Header, F Fn) const {
313   std::vector<TupleType> Results;
314   Results.reserve(FunctionLatencies.size());
315   for (auto FT : FunctionLatencies) {
316     const auto &FuncId = FT.first;
317     auto &Timings = FT.second;
318     Results.emplace_back(FuncId, Timings.size(), getStats(Timings));
319     auto &Row = std::get<2>(Results.back());
320     if (Header.CycleFrequency) {
321       double CycleFrequency = Header.CycleFrequency;
322       Row.Min /= CycleFrequency;
323       Row.Median /= CycleFrequency;
324       Row.Pct90 /= CycleFrequency;
325       Row.Pct99 /= CycleFrequency;
326       Row.Max /= CycleFrequency;
327       Row.Sum /= CycleFrequency;
328     }
329 
330     Row.Function = FuncIdHelper.SymbolOrNumber(FuncId);
331     Row.DebugInfo = FuncIdHelper.FileLineAndColumn(FuncId);
332   }
333 
334   // Sort the data according to user-provided flags.
335   switch (AccountSortOutput) {
336   case SortField::FUNCID:
337     sortByKey(Results, [](const TupleType &X) { return std::get<0>(X); });
338     break;
339   case SortField::COUNT:
340     sortByKey(Results, [](const TupleType &X) { return std::get<1>(X); });
341     break;
342   case SortField::MIN:
343     sortByKey(Results, [](const TupleType &X) { return std::get<2>(X).Min; });
344     break;
345   case SortField::MED:
346     sortByKey(Results, [](const TupleType &X) { return std::get<2>(X).Median; });
347     break;
348   case SortField::PCT90:
349     sortByKey(Results, [](const TupleType &X) { return std::get<2>(X).Pct90; });
350     break;
351   case SortField::PCT99:
352     sortByKey(Results, [](const TupleType &X) { return std::get<2>(X).Pct99; });
353     break;
354   case SortField::MAX:
355     sortByKey(Results, [](const TupleType &X) { return std::get<2>(X).Max; });
356     break;
357   case SortField::SUM:
358     sortByKey(Results, [](const TupleType &X) { return std::get<2>(X).Sum; });
359     break;
360   case SortField::FUNC:
361     llvm_unreachable("Not implemented");
362   }
363 
364   if (AccountTop > 0) {
365     auto MaxTop =
366         std::min(AccountTop.getValue(), static_cast<int>(Results.size()));
367     Results.erase(Results.begin() + MaxTop, Results.end());
368   }
369 
370   for (const auto &R : Results)
371     Fn(std::get<0>(R), std::get<1>(R), std::get<2>(R));
372 }
373 
374 void LatencyAccountant::exportStatsAsText(raw_ostream &OS,
375                                           const XRayFileHeader &Header) const {
376   OS << "Functions with latencies: " << FunctionLatencies.size() << "\n";
377 
378   // We spend some effort to make the text output more readable, so we do the
379   // following formatting decisions for each of the fields:
380   //
381   //   - funcid: 32-bit, but we can determine the largest number and be
382   //   between
383   //     a minimum of 5 characters, up to 9 characters, right aligned.
384   //   - count:  64-bit, but we can determine the largest number and be
385   //   between
386   //     a minimum of 5 characters, up to 9 characters, right aligned.
387   //   - min, median, 90pct, 99pct, max: double precision, but we want to keep
388   //     the values in seconds, with microsecond precision (0.000'001), so we
389   //     have at most 6 significant digits, with the whole number part to be
390   //     at
391   //     least 1 character. For readability we'll right-align, with full 9
392   //     characters each.
393   //   - debug info, function name: we format this as a concatenation of the
394   //     debug info and the function name.
395   //
396   static constexpr char StatsHeaderFormat[] =
397       "{0,+9} {1,+10} [{2,+9}, {3,+9}, {4,+9}, {5,+9}, {6,+9}] {7,+9}";
398   static constexpr char StatsFormat[] =
399       R"({0,+9} {1,+10} [{2,+9:f6}, {3,+9:f6}, {4,+9:f6}, {5,+9:f6}, {6,+9:f6}] {7,+9:f6})";
400   OS << llvm::formatv(StatsHeaderFormat, "funcid", "count", "min", "med", "90p",
401                       "99p", "max", "sum")
402      << llvm::formatv("  {0,-12}\n", "function");
403   exportStats(Header, [&](int32_t FuncId, size_t Count, const ResultRow &Row) {
404     OS << llvm::formatv(StatsFormat, FuncId, Count, Row.Min, Row.Median,
405                         Row.Pct90, Row.Pct99, Row.Max, Row.Sum)
406        << "  " << Row.DebugInfo << ": " << Row.Function << "\n";
407   });
408 }
409 
410 void LatencyAccountant::exportStatsAsCSV(raw_ostream &OS,
411                                          const XRayFileHeader &Header) const {
412   OS << "funcid,count,min,median,90%ile,99%ile,max,sum,debug,function\n";
413   exportStats(Header, [&](int32_t FuncId, size_t Count, const ResultRow &Row) {
414     OS << FuncId << ',' << Count << ',' << Row.Min << ',' << Row.Median << ','
415        << Row.Pct90 << ',' << Row.Pct99 << ',' << Row.Max << "," << Row.Sum
416        << ",\"" << Row.DebugInfo << "\",\"" << Row.Function << "\"\n";
417   });
418 }
419 
420 using namespace llvm::xray;
421 
422 namespace llvm {
423 template <> struct format_provider<llvm::xray::RecordTypes> {
424   static void format(const llvm::xray::RecordTypes &T, raw_ostream &Stream,
425                      StringRef Style) {
426     switch (T) {
427     case RecordTypes::ENTER:
428       Stream << "enter";
429       break;
430     case RecordTypes::ENTER_ARG:
431       Stream << "enter-arg";
432       break;
433     case RecordTypes::EXIT:
434       Stream << "exit";
435       break;
436     case RecordTypes::TAIL_EXIT:
437       Stream << "tail-exit";
438       break;
439     case RecordTypes::CUSTOM_EVENT:
440       Stream << "custom-event";
441       break;
442     case RecordTypes::TYPED_EVENT:
443       Stream << "typed-event";
444       break;
445     }
446   }
447 };
448 } // namespace llvm
449 
450 static CommandRegistration Unused(&Account, []() -> Error {
451   InstrumentationMap Map;
452   if (!AccountInstrMap.empty()) {
453     auto InstrumentationMapOrError = loadInstrumentationMap(AccountInstrMap);
454     if (!InstrumentationMapOrError)
455       return joinErrors(make_error<StringError>(
456                             Twine("Cannot open instrumentation map '") +
457                                 AccountInstrMap + "'",
458                             std::make_error_code(std::errc::invalid_argument)),
459                         InstrumentationMapOrError.takeError());
460     Map = std::move(*InstrumentationMapOrError);
461   }
462 
463   std::error_code EC;
464   raw_fd_ostream OS(AccountOutput, EC, sys::fs::OpenFlags::OF_TextWithCRLF);
465   if (EC)
466     return make_error<StringError>(
467         Twine("Cannot open file '") + AccountOutput + "' for writing.", EC);
468 
469   const auto &FunctionAddresses = Map.getFunctionAddresses();
470   symbolize::LLVMSymbolizer Symbolizer;
471   llvm::xray::FuncIdConversionHelper FuncIdHelper(AccountInstrMap, Symbolizer,
472                                                   FunctionAddresses);
473   xray::LatencyAccountant FCA(FuncIdHelper, AccountRecursiveCallsOnly,
474                               AccountDeduceSiblingCalls);
475   auto TraceOrErr = loadTraceFile(AccountInput);
476   if (!TraceOrErr)
477     return joinErrors(
478         make_error<StringError>(
479             Twine("Failed loading input file '") + AccountInput + "'",
480             std::make_error_code(std::errc::executable_format_error)),
481         TraceOrErr.takeError());
482 
483   auto &T = *TraceOrErr;
484   for (const auto &Record : T) {
485     if (FCA.accountRecord(Record))
486       continue;
487     errs()
488         << "Error processing record: "
489         << llvm::formatv(
490                R"({{type: {0}; cpu: {1}; record-type: {2}; function-id: {3}; tsc: {4}; thread-id: {5}; process-id: {6}}})",
491                Record.RecordType, Record.CPU, Record.Type, Record.FuncId,
492                Record.TSC, Record.TId, Record.PId)
493         << '\n';
494     for (const auto &ThreadStack : FCA.getPerThreadFunctionStack()) {
495       errs() << "Thread ID: " << ThreadStack.first << "\n";
496       if (ThreadStack.second.Stack.empty()) {
497         errs() << "  (empty stack)\n";
498         continue;
499       }
500       auto Level = ThreadStack.second.Stack.size();
501       for (const auto &Entry : llvm::reverse(ThreadStack.second.Stack))
502         errs() << "  #" << Level-- << "\t"
503                << FuncIdHelper.SymbolOrNumber(Entry.first) << '\n';
504     }
505     if (!AccountKeepGoing)
506       return make_error<StringError>(
507           Twine("Failed accounting function calls in file '") + AccountInput +
508               "'.",
509           std::make_error_code(std::errc::executable_format_error));
510   }
511   switch (AccountOutputFormat) {
512   case AccountOutputFormats::TEXT:
513     FCA.exportStatsAsText(OS, T.getFileHeader());
514     break;
515   case AccountOutputFormats::CSV:
516     FCA.exportStatsAsCSV(OS, T.getFileHeader());
517     break;
518   }
519 
520   return Error::success();
521 });
522