xref: /freebsd/contrib/llvm-project/llvm/lib/ProfileData/GCOV.cpp (revision 63f537551380d2dab29fa402ad1269feae17e594)
1 //===- GCOV.cpp - LLVM coverage tool --------------------------------------===//
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 // GCOV implements the interface to read and write coverage files that use
10 // 'gcov' format.
11 //
12 //===----------------------------------------------------------------------===//
13 
14 #include "llvm/ProfileData/GCOV.h"
15 #include "llvm/ADT/STLExtras.h"
16 #include "llvm/ADT/SmallSet.h"
17 #include "llvm/Config/llvm-config.h"
18 #include "llvm/Demangle/Demangle.h"
19 #include "llvm/Support/Debug.h"
20 #include "llvm/Support/FileSystem.h"
21 #include "llvm/Support/Format.h"
22 #include "llvm/Support/MD5.h"
23 #include "llvm/Support/Path.h"
24 #include "llvm/Support/raw_ostream.h"
25 #include <algorithm>
26 #include <optional>
27 #include <system_error>
28 
29 using namespace llvm;
30 
31 enum : uint32_t {
32   GCOV_ARC_ON_TREE = 1 << 0,
33   GCOV_ARC_FALLTHROUGH = 1 << 2,
34 
35   GCOV_TAG_FUNCTION = 0x01000000,
36   GCOV_TAG_BLOCKS = 0x01410000,
37   GCOV_TAG_ARCS = 0x01430000,
38   GCOV_TAG_LINES = 0x01450000,
39   GCOV_TAG_COUNTER_ARCS = 0x01a10000,
40   // GCOV_TAG_OBJECT_SUMMARY superseded GCOV_TAG_PROGRAM_SUMMARY in GCC 9.
41   GCOV_TAG_OBJECT_SUMMARY = 0xa1000000,
42   GCOV_TAG_PROGRAM_SUMMARY = 0xa3000000,
43 };
44 
45 namespace {
46 struct Summary {
47   Summary(StringRef Name) : Name(Name) {}
48 
49   StringRef Name;
50   uint64_t lines = 0;
51   uint64_t linesExec = 0;
52   uint64_t branches = 0;
53   uint64_t branchesExec = 0;
54   uint64_t branchesTaken = 0;
55 };
56 
57 struct LineInfo {
58   SmallVector<const GCOVBlock *, 1> blocks;
59   uint64_t count = 0;
60   bool exists = false;
61 };
62 
63 struct SourceInfo {
64   StringRef filename;
65   SmallString<0> displayName;
66   std::vector<std::vector<const GCOVFunction *>> startLineToFunctions;
67   std::vector<LineInfo> lines;
68   bool ignored = false;
69   SourceInfo(StringRef filename) : filename(filename) {}
70 };
71 
72 class Context {
73 public:
74   Context(const GCOV::Options &Options) : options(Options) {}
75   void print(StringRef filename, StringRef gcno, StringRef gcda,
76              GCOVFile &file);
77 
78 private:
79   std::string getCoveragePath(StringRef filename, StringRef mainFilename) const;
80   void printFunctionDetails(const GCOVFunction &f, raw_ostream &os) const;
81   void printBranchInfo(const GCOVBlock &Block, uint32_t &edgeIdx,
82                        raw_ostream &OS) const;
83   void printSummary(const Summary &summary, raw_ostream &os) const;
84 
85   void collectFunction(GCOVFunction &f, Summary &summary);
86   void collectSourceLine(SourceInfo &si, Summary *summary, LineInfo &line,
87                          size_t lineNum) const;
88   void collectSource(SourceInfo &si, Summary &summary) const;
89   void annotateSource(SourceInfo &si, const GCOVFile &file, StringRef gcno,
90                       StringRef gcda, raw_ostream &os) const;
91   void printSourceToIntermediate(const SourceInfo &si, raw_ostream &os) const;
92 
93   const GCOV::Options &options;
94   std::vector<SourceInfo> sources;
95 };
96 } // namespace
97 
98 //===----------------------------------------------------------------------===//
99 // GCOVFile implementation.
100 
101 /// readGCNO - Read GCNO buffer.
102 bool GCOVFile::readGCNO(GCOVBuffer &buf) {
103   if (!buf.readGCNOFormat())
104     return false;
105   if (!buf.readGCOVVersion(version))
106     return false;
107 
108   checksum = buf.getWord();
109   if (version >= GCOV::V900 && !buf.readString(cwd))
110     return false;
111   if (version >= GCOV::V800)
112     buf.getWord(); // hasUnexecutedBlocks
113 
114   uint32_t tag, length;
115   GCOVFunction *fn = nullptr;
116   while ((tag = buf.getWord())) {
117     if (!buf.readInt(length))
118       return false;
119     uint32_t pos = buf.cursor.tell();
120     if (tag == GCOV_TAG_FUNCTION) {
121       functions.push_back(std::make_unique<GCOVFunction>(*this));
122       fn = functions.back().get();
123       fn->ident = buf.getWord();
124       fn->linenoChecksum = buf.getWord();
125       if (version >= GCOV::V407)
126         fn->cfgChecksum = buf.getWord();
127       buf.readString(fn->Name);
128       StringRef filename;
129       if (version < GCOV::V800) {
130         if (!buf.readString(filename))
131           return false;
132         fn->startLine = buf.getWord();
133       } else {
134         fn->artificial = buf.getWord();
135         if (!buf.readString(filename))
136           return false;
137         fn->startLine = buf.getWord();
138         fn->startColumn = buf.getWord();
139         fn->endLine = buf.getWord();
140         if (version >= GCOV::V900)
141           fn->endColumn = buf.getWord();
142       }
143       auto r = filenameToIdx.try_emplace(filename, filenameToIdx.size());
144       if (r.second)
145         filenames.emplace_back(filename);
146       fn->srcIdx = r.first->second;
147       identToFunction[fn->ident] = fn;
148     } else if (tag == GCOV_TAG_BLOCKS && fn) {
149       if (version < GCOV::V800) {
150         for (uint32_t i = 0; i != length; ++i) {
151           buf.getWord(); // Ignored block flags
152           fn->blocks.push_back(std::make_unique<GCOVBlock>(i));
153         }
154       } else {
155         uint32_t num = buf.getWord();
156         for (uint32_t i = 0; i != num; ++i)
157           fn->blocks.push_back(std::make_unique<GCOVBlock>(i));
158       }
159     } else if (tag == GCOV_TAG_ARCS && fn) {
160       uint32_t srcNo = buf.getWord();
161       if (srcNo >= fn->blocks.size()) {
162         errs() << "unexpected block number: " << srcNo << " (in "
163                << fn->blocks.size() << ")\n";
164         return false;
165       }
166       GCOVBlock *src = fn->blocks[srcNo].get();
167       const uint32_t e =
168           version >= GCOV::V1200 ? (length / 4 - 1) / 2 : (length - 1) / 2;
169       for (uint32_t i = 0; i != e; ++i) {
170         uint32_t dstNo = buf.getWord(), flags = buf.getWord();
171         GCOVBlock *dst = fn->blocks[dstNo].get();
172         auto arc = std::make_unique<GCOVArc>(*src, *dst, flags);
173         src->addDstEdge(arc.get());
174         dst->addSrcEdge(arc.get());
175         if (arc->onTree())
176           fn->treeArcs.push_back(std::move(arc));
177         else
178           fn->arcs.push_back(std::move(arc));
179       }
180     } else if (tag == GCOV_TAG_LINES && fn) {
181       uint32_t srcNo = buf.getWord();
182       if (srcNo >= fn->blocks.size()) {
183         errs() << "unexpected block number: " << srcNo << " (in "
184                << fn->blocks.size() << ")\n";
185         return false;
186       }
187       GCOVBlock &Block = *fn->blocks[srcNo];
188       for (;;) {
189         uint32_t line = buf.getWord();
190         if (line)
191           Block.addLine(line);
192         else {
193           StringRef filename;
194           buf.readString(filename);
195           if (filename.empty())
196             break;
197           // TODO Unhandled
198         }
199       }
200     }
201     pos += version >= GCOV::V1200 ? length : 4 * length;
202     if (pos < buf.cursor.tell())
203       return false;
204     buf.de.skip(buf.cursor, pos - buf.cursor.tell());
205   }
206 
207   GCNOInitialized = true;
208   return true;
209 }
210 
211 /// readGCDA - Read GCDA buffer. It is required that readGCDA() can only be
212 /// called after readGCNO().
213 bool GCOVFile::readGCDA(GCOVBuffer &buf) {
214   assert(GCNOInitialized && "readGCDA() can only be called after readGCNO()");
215   if (!buf.readGCDAFormat())
216     return false;
217   GCOV::GCOVVersion GCDAVersion;
218   if (!buf.readGCOVVersion(GCDAVersion))
219     return false;
220   if (version != GCDAVersion) {
221     errs() << "GCOV versions do not match.\n";
222     return false;
223   }
224 
225   uint32_t GCDAChecksum;
226   if (!buf.readInt(GCDAChecksum))
227     return false;
228   if (checksum != GCDAChecksum) {
229     errs() << "file checksums do not match: " << checksum
230            << " != " << GCDAChecksum << "\n";
231     return false;
232   }
233   uint32_t dummy, tag, length;
234   uint32_t ident;
235   GCOVFunction *fn = nullptr;
236   while ((tag = buf.getWord())) {
237     if (!buf.readInt(length))
238       return false;
239     uint32_t pos = buf.cursor.tell();
240     if (tag == GCOV_TAG_OBJECT_SUMMARY) {
241       buf.readInt(runCount);
242       buf.readInt(dummy);
243       // clang<11 uses a fake 4.2 format which sets length to 9.
244       if (length == 9)
245         buf.readInt(runCount);
246     } else if (tag == GCOV_TAG_PROGRAM_SUMMARY) {
247       // clang<11 uses a fake 4.2 format which sets length to 0.
248       if (length > 0) {
249         buf.readInt(dummy);
250         buf.readInt(dummy);
251         buf.readInt(runCount);
252       }
253       ++programCount;
254     } else if (tag == GCOV_TAG_FUNCTION) {
255       if (length == 0) // Placeholder
256         continue;
257       // As of GCC 10, GCOV_TAG_FUNCTION_LENGTH has never been larger than 3.
258       // However, clang<11 uses a fake 4.2 format which may set length larger
259       // than 3.
260       if (length < 2 || !buf.readInt(ident))
261         return false;
262       auto It = identToFunction.find(ident);
263       uint32_t linenoChecksum, cfgChecksum = 0;
264       buf.readInt(linenoChecksum);
265       if (version >= GCOV::V407)
266         buf.readInt(cfgChecksum);
267       if (It != identToFunction.end()) {
268         fn = It->second;
269         if (linenoChecksum != fn->linenoChecksum ||
270             cfgChecksum != fn->cfgChecksum) {
271           errs() << fn->Name
272                  << format(": checksum mismatch, (%u, %u) != (%u, %u)\n",
273                            linenoChecksum, cfgChecksum, fn->linenoChecksum,
274                            fn->cfgChecksum);
275           return false;
276         }
277       }
278     } else if (tag == GCOV_TAG_COUNTER_ARCS && fn) {
279       uint32_t expected = 2 * fn->arcs.size();
280       if (version >= GCOV::V1200)
281         expected *= 4;
282       if (length != expected) {
283         errs() << fn->Name
284                << format(
285                       ": GCOV_TAG_COUNTER_ARCS mismatch, got %u, expected %u\n",
286                       length, expected);
287         return false;
288       }
289       for (std::unique_ptr<GCOVArc> &arc : fn->arcs) {
290         if (!buf.readInt64(arc->count))
291           return false;
292         arc->src.count += arc->count;
293       }
294 
295       if (fn->blocks.size() >= 2) {
296         GCOVBlock &src = *fn->blocks[0];
297         GCOVBlock &sink =
298             version < GCOV::V408 ? *fn->blocks.back() : *fn->blocks[1];
299         auto arc = std::make_unique<GCOVArc>(sink, src, GCOV_ARC_ON_TREE);
300         sink.addDstEdge(arc.get());
301         src.addSrcEdge(arc.get());
302         fn->treeArcs.push_back(std::move(arc));
303 
304         for (GCOVBlock &block : fn->blocksRange())
305           fn->propagateCounts(block, nullptr);
306         for (size_t i = fn->treeArcs.size() - 1; i; --i)
307           fn->treeArcs[i - 1]->src.count += fn->treeArcs[i - 1]->count;
308       }
309     }
310     pos += version >= GCOV::V1200 ? length : 4 * length;
311     if (pos < buf.cursor.tell())
312       return false;
313     buf.de.skip(buf.cursor, pos - buf.cursor.tell());
314   }
315 
316   return true;
317 }
318 
319 void GCOVFile::print(raw_ostream &OS) const {
320   for (const GCOVFunction &f : *this)
321     f.print(OS);
322 }
323 
324 #if !defined(NDEBUG) || defined(LLVM_ENABLE_DUMP)
325 /// dump - Dump GCOVFile content to dbgs() for debugging purposes.
326 LLVM_DUMP_METHOD void GCOVFile::dump() const { print(dbgs()); }
327 #endif
328 
329 bool GCOVArc::onTree() const { return flags & GCOV_ARC_ON_TREE; }
330 
331 //===----------------------------------------------------------------------===//
332 // GCOVFunction implementation.
333 
334 StringRef GCOVFunction::getName(bool demangle) const {
335   if (!demangle)
336     return Name;
337   if (demangled.empty()) {
338     do {
339       if (Name.startswith("_Z")) {
340         int status = 0;
341         // Name is guaranteed to be NUL-terminated.
342         char *res = itaniumDemangle(Name.data(), nullptr, nullptr, &status);
343         if (status == 0) {
344           demangled = res;
345           free(res);
346           break;
347         }
348       }
349       demangled = Name;
350     } while (false);
351   }
352   return demangled;
353 }
354 StringRef GCOVFunction::getFilename() const { return file.filenames[srcIdx]; }
355 
356 /// getEntryCount - Get the number of times the function was called by
357 /// retrieving the entry block's count.
358 uint64_t GCOVFunction::getEntryCount() const {
359   return blocks.front()->getCount();
360 }
361 
362 GCOVBlock &GCOVFunction::getExitBlock() const {
363   return file.getVersion() < GCOV::V408 ? *blocks.back() : *blocks[1];
364 }
365 
366 // For each basic block, the sum of incoming edge counts equals the sum of
367 // outgoing edge counts by Kirchoff's circuit law. If the unmeasured arcs form a
368 // spanning tree, the count for each unmeasured arc (GCOV_ARC_ON_TREE) can be
369 // uniquely identified.
370 uint64_t GCOVFunction::propagateCounts(const GCOVBlock &v, GCOVArc *pred) {
371   // If GCOV_ARC_ON_TREE edges do form a tree, visited is not needed; otherwise
372   // this prevents infinite recursion.
373   if (!visited.insert(&v).second)
374     return 0;
375 
376   uint64_t excess = 0;
377   for (GCOVArc *e : v.srcs())
378     if (e != pred)
379       excess += e->onTree() ? propagateCounts(e->src, e) : e->count;
380   for (GCOVArc *e : v.dsts())
381     if (e != pred)
382       excess -= e->onTree() ? propagateCounts(e->dst, e) : e->count;
383   if (int64_t(excess) < 0)
384     excess = -excess;
385   if (pred)
386     pred->count = excess;
387   return excess;
388 }
389 
390 void GCOVFunction::print(raw_ostream &OS) const {
391   OS << "===== " << Name << " (" << ident << ") @ " << getFilename() << ":"
392      << startLine << "\n";
393   for (const auto &Block : blocks)
394     Block->print(OS);
395 }
396 
397 #if !defined(NDEBUG) || defined(LLVM_ENABLE_DUMP)
398 /// dump - Dump GCOVFunction content to dbgs() for debugging purposes.
399 LLVM_DUMP_METHOD void GCOVFunction::dump() const { print(dbgs()); }
400 #endif
401 
402 /// collectLineCounts - Collect line counts. This must be used after
403 /// reading .gcno and .gcda files.
404 
405 //===----------------------------------------------------------------------===//
406 // GCOVBlock implementation.
407 
408 void GCOVBlock::print(raw_ostream &OS) const {
409   OS << "Block : " << number << " Counter : " << count << "\n";
410   if (!pred.empty()) {
411     OS << "\tSource Edges : ";
412     for (const GCOVArc *Edge : pred)
413       OS << Edge->src.number << " (" << Edge->count << "), ";
414     OS << "\n";
415   }
416   if (!succ.empty()) {
417     OS << "\tDestination Edges : ";
418     for (const GCOVArc *Edge : succ) {
419       if (Edge->flags & GCOV_ARC_ON_TREE)
420         OS << '*';
421       OS << Edge->dst.number << " (" << Edge->count << "), ";
422     }
423     OS << "\n";
424   }
425   if (!lines.empty()) {
426     OS << "\tLines : ";
427     for (uint32_t N : lines)
428       OS << (N) << ",";
429     OS << "\n";
430   }
431 }
432 
433 #if !defined(NDEBUG) || defined(LLVM_ENABLE_DUMP)
434 /// dump - Dump GCOVBlock content to dbgs() for debugging purposes.
435 LLVM_DUMP_METHOD void GCOVBlock::dump() const { print(dbgs()); }
436 #endif
437 
438 uint64_t
439 GCOVBlock::augmentOneCycle(GCOVBlock *src,
440                            std::vector<std::pair<GCOVBlock *, size_t>> &stack) {
441   GCOVBlock *u;
442   size_t i;
443   stack.clear();
444   stack.emplace_back(src, 0);
445   src->incoming = (GCOVArc *)1; // Mark u available for cycle detection
446   for (;;) {
447     std::tie(u, i) = stack.back();
448     if (i == u->succ.size()) {
449       u->traversable = false;
450       stack.pop_back();
451       if (stack.empty())
452         break;
453       continue;
454     }
455     ++stack.back().second;
456     GCOVArc *succ = u->succ[i];
457     // Ignore saturated arcs (cycleCount has been reduced to 0) and visited
458     // blocks. Ignore self arcs to guard against bad input (.gcno has no
459     // self arcs).
460     if (succ->cycleCount == 0 || !succ->dst.traversable || &succ->dst == u)
461       continue;
462     if (succ->dst.incoming == nullptr) {
463       succ->dst.incoming = succ;
464       stack.emplace_back(&succ->dst, 0);
465       continue;
466     }
467     uint64_t minCount = succ->cycleCount;
468     for (GCOVBlock *v = u;;) {
469       minCount = std::min(minCount, v->incoming->cycleCount);
470       v = &v->incoming->src;
471       if (v == &succ->dst)
472         break;
473     }
474     succ->cycleCount -= minCount;
475     for (GCOVBlock *v = u;;) {
476       v->incoming->cycleCount -= minCount;
477       v = &v->incoming->src;
478       if (v == &succ->dst)
479         break;
480     }
481     return minCount;
482   }
483   return 0;
484 }
485 
486 // Get the total execution count of loops among blocks on the same line.
487 // Assuming a reducible flow graph, the count is the sum of back edge counts.
488 // Identifying loops is complex, so we simply find cycles and perform cycle
489 // cancelling iteratively.
490 uint64_t GCOVBlock::getCyclesCount(const BlockVector &blocks) {
491   std::vector<std::pair<GCOVBlock *, size_t>> stack;
492   uint64_t count = 0, d;
493   for (;;) {
494     // Make blocks on the line traversable and try finding a cycle.
495     for (const auto *b : blocks) {
496       const_cast<GCOVBlock *>(b)->traversable = true;
497       const_cast<GCOVBlock *>(b)->incoming = nullptr;
498     }
499     d = 0;
500     for (const auto *block : blocks) {
501       auto *b = const_cast<GCOVBlock *>(block);
502       if (b->traversable && (d = augmentOneCycle(b, stack)) > 0)
503         break;
504     }
505     if (d == 0)
506       break;
507     count += d;
508   }
509   // If there is no more loop, all traversable bits should have been cleared.
510   // This property is needed by subsequent calls.
511   for (const auto *b : blocks) {
512     assert(!b->traversable);
513     (void)b;
514   }
515   return count;
516 }
517 
518 //===----------------------------------------------------------------------===//
519 // FileInfo implementation.
520 
521 // Format dividend/divisor as a percentage. Return 1 if the result is greater
522 // than 0% and less than 1%.
523 static uint32_t formatPercentage(uint64_t dividend, uint64_t divisor) {
524   if (!dividend || !divisor)
525     return 0;
526   dividend *= 100;
527   return dividend < divisor ? 1 : dividend / divisor;
528 }
529 
530 // This custom division function mimics gcov's branch ouputs:
531 //   - Round to closest whole number
532 //   - Only output 0% or 100% if it's exactly that value
533 static uint32_t branchDiv(uint64_t Numerator, uint64_t Divisor) {
534   if (!Numerator)
535     return 0;
536   if (Numerator == Divisor)
537     return 100;
538 
539   uint8_t Res = (Numerator * 100 + Divisor / 2) / Divisor;
540   if (Res == 0)
541     return 1;
542   if (Res == 100)
543     return 99;
544   return Res;
545 }
546 
547 namespace {
548 struct formatBranchInfo {
549   formatBranchInfo(const GCOV::Options &Options, uint64_t Count, uint64_t Total)
550       : Options(Options), Count(Count), Total(Total) {}
551 
552   void print(raw_ostream &OS) const {
553     if (!Total)
554       OS << "never executed";
555     else if (Options.BranchCount)
556       OS << "taken " << Count;
557     else
558       OS << "taken " << branchDiv(Count, Total) << "%";
559   }
560 
561   const GCOV::Options &Options;
562   uint64_t Count;
563   uint64_t Total;
564 };
565 
566 static raw_ostream &operator<<(raw_ostream &OS, const formatBranchInfo &FBI) {
567   FBI.print(OS);
568   return OS;
569 }
570 
571 class LineConsumer {
572   std::unique_ptr<MemoryBuffer> Buffer;
573   StringRef Remaining;
574 
575 public:
576   LineConsumer() = default;
577   LineConsumer(StringRef Filename) {
578     // Open source files without requiring a NUL terminator. The concurrent
579     // modification may nullify the NUL terminator condition.
580     ErrorOr<std::unique_ptr<MemoryBuffer>> BufferOrErr =
581         MemoryBuffer::getFileOrSTDIN(Filename, /*IsText=*/false,
582                                      /*RequiresNullTerminator=*/false);
583     if (std::error_code EC = BufferOrErr.getError()) {
584       errs() << Filename << ": " << EC.message() << "\n";
585       Remaining = "";
586     } else {
587       Buffer = std::move(BufferOrErr.get());
588       Remaining = Buffer->getBuffer();
589     }
590   }
591   bool empty() { return Remaining.empty(); }
592   void printNext(raw_ostream &OS, uint32_t LineNum) {
593     StringRef Line;
594     if (empty())
595       Line = "/*EOF*/";
596     else
597       std::tie(Line, Remaining) = Remaining.split("\n");
598     OS << format("%5u:", LineNum) << Line << "\n";
599   }
600 };
601 } // end anonymous namespace
602 
603 /// Convert a path to a gcov filename. If PreservePaths is true, this
604 /// translates "/" to "#", ".." to "^", and drops ".", to match gcov.
605 static std::string mangleCoveragePath(StringRef Filename, bool PreservePaths) {
606   if (!PreservePaths)
607     return sys::path::filename(Filename).str();
608 
609   // This behaviour is defined by gcov in terms of text replacements, so it's
610   // not likely to do anything useful on filesystems with different textual
611   // conventions.
612   llvm::SmallString<256> Result("");
613   StringRef::iterator I, S, E;
614   for (I = S = Filename.begin(), E = Filename.end(); I != E; ++I) {
615     if (*I != '/')
616       continue;
617 
618     if (I - S == 1 && *S == '.') {
619       // ".", the current directory, is skipped.
620     } else if (I - S == 2 && *S == '.' && *(S + 1) == '.') {
621       // "..", the parent directory, is replaced with "^".
622       Result.append("^#");
623     } else {
624       if (S < I)
625         // Leave other components intact,
626         Result.append(S, I);
627       // And separate with "#".
628       Result.push_back('#');
629     }
630     S = I + 1;
631   }
632 
633   if (S < I)
634     Result.append(S, I);
635   return std::string(Result.str());
636 }
637 
638 std::string Context::getCoveragePath(StringRef filename,
639                                      StringRef mainFilename) const {
640   if (options.NoOutput)
641     // This is probably a bug in gcov, but when -n is specified, paths aren't
642     // mangled at all, and the -l and -p options are ignored. Here, we do the
643     // same.
644     return std::string(filename);
645 
646   std::string CoveragePath;
647   if (options.LongFileNames && !filename.equals(mainFilename))
648     CoveragePath =
649         mangleCoveragePath(mainFilename, options.PreservePaths) + "##";
650   CoveragePath += mangleCoveragePath(filename, options.PreservePaths);
651   if (options.HashFilenames) {
652     MD5 Hasher;
653     MD5::MD5Result Result;
654     Hasher.update(filename.str());
655     Hasher.final(Result);
656     CoveragePath += "##" + std::string(Result.digest());
657   }
658   CoveragePath += ".gcov";
659   return CoveragePath;
660 }
661 
662 void Context::collectFunction(GCOVFunction &f, Summary &summary) {
663   SourceInfo &si = sources[f.srcIdx];
664   if (f.startLine >= si.startLineToFunctions.size())
665     si.startLineToFunctions.resize(f.startLine + 1);
666   si.startLineToFunctions[f.startLine].push_back(&f);
667   SmallSet<uint32_t, 16> lines;
668   SmallSet<uint32_t, 16> linesExec;
669   for (const GCOVBlock &b : f.blocksRange()) {
670     if (b.lines.empty())
671       continue;
672     uint32_t maxLineNum = *std::max_element(b.lines.begin(), b.lines.end());
673     if (maxLineNum >= si.lines.size())
674       si.lines.resize(maxLineNum + 1);
675     for (uint32_t lineNum : b.lines) {
676       LineInfo &line = si.lines[lineNum];
677       if (lines.insert(lineNum).second)
678         ++summary.lines;
679       if (b.count && linesExec.insert(lineNum).second)
680         ++summary.linesExec;
681       line.exists = true;
682       line.count += b.count;
683       line.blocks.push_back(&b);
684     }
685   }
686 }
687 
688 void Context::collectSourceLine(SourceInfo &si, Summary *summary,
689                                 LineInfo &line, size_t lineNum) const {
690   uint64_t count = 0;
691   for (const GCOVBlock *b : line.blocks) {
692     if (b->number == 0) {
693       // For nonstandard control flows, arcs into the exit block may be
694       // duplicately counted (fork) or not be counted (abnormal exit), and thus
695       // the (exit,entry) counter may be inaccurate. Count the entry block with
696       // the outgoing arcs.
697       for (const GCOVArc *arc : b->succ)
698         count += arc->count;
699     } else {
700       // Add counts from predecessors that are not on the same line.
701       for (const GCOVArc *arc : b->pred)
702         if (!llvm::is_contained(line.blocks, &arc->src))
703           count += arc->count;
704     }
705     for (GCOVArc *arc : b->succ)
706       arc->cycleCount = arc->count;
707   }
708 
709   count += GCOVBlock::getCyclesCount(line.blocks);
710   line.count = count;
711   if (line.exists) {
712     ++summary->lines;
713     if (line.count != 0)
714       ++summary->linesExec;
715   }
716 
717   if (options.BranchInfo)
718     for (const GCOVBlock *b : line.blocks) {
719       if (b->getLastLine() != lineNum)
720         continue;
721       int branches = 0, execBranches = 0, takenBranches = 0;
722       for (const GCOVArc *arc : b->succ) {
723         ++branches;
724         if (count != 0)
725           ++execBranches;
726         if (arc->count != 0)
727           ++takenBranches;
728       }
729       if (branches > 1) {
730         summary->branches += branches;
731         summary->branchesExec += execBranches;
732         summary->branchesTaken += takenBranches;
733       }
734     }
735 }
736 
737 void Context::collectSource(SourceInfo &si, Summary &summary) const {
738   size_t lineNum = 0;
739   for (LineInfo &line : si.lines) {
740     collectSourceLine(si, &summary, line, lineNum);
741     ++lineNum;
742   }
743 }
744 
745 void Context::annotateSource(SourceInfo &si, const GCOVFile &file,
746                              StringRef gcno, StringRef gcda,
747                              raw_ostream &os) const {
748   auto source =
749       options.Intermediate ? LineConsumer() : LineConsumer(si.filename);
750 
751   os << "        -:    0:Source:" << si.displayName << '\n';
752   os << "        -:    0:Graph:" << gcno << '\n';
753   os << "        -:    0:Data:" << gcda << '\n';
754   os << "        -:    0:Runs:" << file.runCount << '\n';
755   if (file.version < GCOV::V900)
756     os << "        -:    0:Programs:" << file.programCount << '\n';
757 
758   for (size_t lineNum = 1; !source.empty(); ++lineNum) {
759     if (lineNum >= si.lines.size()) {
760       os << "        -:";
761       source.printNext(os, lineNum);
762       continue;
763     }
764 
765     const LineInfo &line = si.lines[lineNum];
766     if (options.BranchInfo && lineNum < si.startLineToFunctions.size())
767       for (const auto *f : si.startLineToFunctions[lineNum])
768         printFunctionDetails(*f, os);
769     if (!line.exists)
770       os << "        -:";
771     else if (line.count == 0)
772       os << "    #####:";
773     else
774       os << format("%9" PRIu64 ":", line.count);
775     source.printNext(os, lineNum);
776 
777     uint32_t blockIdx = 0, edgeIdx = 0;
778     for (const GCOVBlock *b : line.blocks) {
779       if (b->getLastLine() != lineNum)
780         continue;
781       if (options.AllBlocks) {
782         if (b->getCount() == 0)
783           os << "    $$$$$:";
784         else
785           os << format("%9" PRIu64 ":", b->count);
786         os << format("%5u-block %2u\n", lineNum, blockIdx++);
787       }
788       if (options.BranchInfo) {
789         size_t NumEdges = b->succ.size();
790         if (NumEdges > 1)
791           printBranchInfo(*b, edgeIdx, os);
792         else if (options.UncondBranch && NumEdges == 1) {
793           uint64_t count = b->succ[0]->count;
794           os << format("unconditional %2u ", edgeIdx++)
795              << formatBranchInfo(options, count, count) << '\n';
796         }
797       }
798     }
799   }
800 }
801 
802 void Context::printSourceToIntermediate(const SourceInfo &si,
803                                         raw_ostream &os) const {
804   os << "file:" << si.filename << '\n';
805   for (const auto &fs : si.startLineToFunctions)
806     for (const GCOVFunction *f : fs)
807       os << "function:" << f->startLine << ',' << f->getEntryCount() << ','
808          << f->getName(options.Demangle) << '\n';
809   for (size_t lineNum = 1, size = si.lines.size(); lineNum < size; ++lineNum) {
810     const LineInfo &line = si.lines[lineNum];
811     if (line.blocks.empty())
812       continue;
813     // GCC 8 (r254259) added third third field for Ada:
814     // lcount:<line>,<count>,<has_unexecuted_blocks>
815     // We don't need the third field.
816     os << "lcount:" << lineNum << ',' << line.count << '\n';
817 
818     if (!options.BranchInfo)
819       continue;
820     for (const GCOVBlock *b : line.blocks) {
821       if (b->succ.size() < 2 || b->getLastLine() != lineNum)
822         continue;
823       for (const GCOVArc *arc : b->succ) {
824         const char *type =
825             b->getCount() ? arc->count ? "taken" : "nottaken" : "notexec";
826         os << "branch:" << lineNum << ',' << type << '\n';
827       }
828     }
829   }
830 }
831 
832 void Context::print(StringRef filename, StringRef gcno, StringRef gcda,
833                     GCOVFile &file) {
834   for (StringRef filename : file.filenames) {
835     sources.emplace_back(filename);
836     SourceInfo &si = sources.back();
837     si.displayName = si.filename;
838     if (!options.SourcePrefix.empty() &&
839         sys::path::replace_path_prefix(si.displayName, options.SourcePrefix,
840                                        "") &&
841         !si.displayName.empty()) {
842       // TODO replace_path_prefix may strip the prefix even if the remaining
843       // part does not start with a separator.
844       if (sys::path::is_separator(si.displayName[0]))
845         si.displayName.erase(si.displayName.begin());
846       else
847         si.displayName = si.filename;
848     }
849     if (options.RelativeOnly && sys::path::is_absolute(si.displayName))
850       si.ignored = true;
851   }
852 
853   raw_ostream &os = llvm::outs();
854   for (GCOVFunction &f : make_pointee_range(file.functions)) {
855     Summary summary(f.getName(options.Demangle));
856     collectFunction(f, summary);
857     if (options.FuncCoverage && !options.UseStdout) {
858       os << "Function '" << summary.Name << "'\n";
859       printSummary(summary, os);
860       os << '\n';
861     }
862   }
863 
864   for (SourceInfo &si : sources) {
865     if (si.ignored)
866       continue;
867     Summary summary(si.displayName);
868     collectSource(si, summary);
869 
870     // Print file summary unless -t is specified.
871     std::string gcovName = getCoveragePath(si.filename, filename);
872     if (!options.UseStdout) {
873       os << "File '" << summary.Name << "'\n";
874       printSummary(summary, os);
875       if (!options.NoOutput && !options.Intermediate)
876         os << "Creating '" << gcovName << "'\n";
877       os << '\n';
878     }
879 
880     if (options.NoOutput || options.Intermediate)
881       continue;
882     std::optional<raw_fd_ostream> os;
883     if (!options.UseStdout) {
884       std::error_code ec;
885       os.emplace(gcovName, ec, sys::fs::OF_TextWithCRLF);
886       if (ec) {
887         errs() << ec.message() << '\n';
888         continue;
889       }
890     }
891     annotateSource(si, file, gcno, gcda,
892                    options.UseStdout ? llvm::outs() : *os);
893   }
894 
895   if (options.Intermediate && !options.NoOutput) {
896     // gcov 7.* unexpectedly create multiple .gcov files, which was fixed in 8.0
897     // (PR GCC/82702). We create just one file.
898     std::string outputPath(sys::path::filename(filename));
899     std::error_code ec;
900     raw_fd_ostream os(outputPath + ".gcov", ec, sys::fs::OF_TextWithCRLF);
901     if (ec) {
902       errs() << ec.message() << '\n';
903       return;
904     }
905 
906     for (const SourceInfo &si : sources)
907       printSourceToIntermediate(si, os);
908   }
909 }
910 
911 void Context::printFunctionDetails(const GCOVFunction &f,
912                                    raw_ostream &os) const {
913   const uint64_t entryCount = f.getEntryCount();
914   uint32_t blocksExec = 0;
915   const GCOVBlock &exitBlock = f.getExitBlock();
916   uint64_t exitCount = 0;
917   for (const GCOVArc *arc : exitBlock.pred)
918     exitCount += arc->count;
919   for (const GCOVBlock &b : f.blocksRange())
920     if (b.number != 0 && &b != &exitBlock && b.getCount())
921       ++blocksExec;
922 
923   os << "function " << f.getName(options.Demangle) << " called " << entryCount
924      << " returned " << formatPercentage(exitCount, entryCount)
925      << "% blocks executed "
926      << formatPercentage(blocksExec, f.blocks.size() - 2) << "%\n";
927 }
928 
929 /// printBranchInfo - Print conditional branch probabilities.
930 void Context::printBranchInfo(const GCOVBlock &Block, uint32_t &edgeIdx,
931                               raw_ostream &os) const {
932   uint64_t total = 0;
933   for (const GCOVArc *arc : Block.dsts())
934     total += arc->count;
935   for (const GCOVArc *arc : Block.dsts())
936     os << format("branch %2u ", edgeIdx++)
937        << formatBranchInfo(options, arc->count, total) << '\n';
938 }
939 
940 void Context::printSummary(const Summary &summary, raw_ostream &os) const {
941   os << format("Lines executed:%.2f%% of %" PRIu64 "\n",
942                double(summary.linesExec) * 100 / summary.lines, summary.lines);
943   if (options.BranchInfo) {
944     if (summary.branches == 0) {
945       os << "No branches\n";
946     } else {
947       os << format("Branches executed:%.2f%% of %" PRIu64 "\n",
948                    double(summary.branchesExec) * 100 / summary.branches,
949                    summary.branches);
950       os << format("Taken at least once:%.2f%% of %" PRIu64 "\n",
951                    double(summary.branchesTaken) * 100 / summary.branches,
952                    summary.branches);
953     }
954     os << "No calls\n";
955   }
956 }
957 
958 void llvm::gcovOneInput(const GCOV::Options &options, StringRef filename,
959                         StringRef gcno, StringRef gcda, GCOVFile &file) {
960   Context fi(options);
961   fi.print(filename, gcno, gcda, file);
962 }
963