1 //===--- CoverageMappingGen.cpp - Coverage mapping generation ---*- C++ -*-===//
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
7 //===----------------------------------------------------------------------===//
9 // Instrumentation-based code coverage mapping generator
11 //===----------------------------------------------------------------------===//
13 #include "CoverageMappingGen.h"
14 #include "CodeGenFunction.h"
15 #include "clang/AST/StmtVisitor.h"
16 #include "clang/Basic/Diagnostic.h"
17 #include "clang/Basic/FileManager.h"
18 #include "clang/Frontend/FrontendDiagnostic.h"
19 #include "clang/Lex/Lexer.h"
20 #include "llvm/ADT/SmallSet.h"
21 #include "llvm/ADT/StringExtras.h"
22 #include "llvm/ProfileData/Coverage/CoverageMapping.h"
23 #include "llvm/ProfileData/Coverage/CoverageMappingReader.h"
24 #include "llvm/ProfileData/Coverage/CoverageMappingWriter.h"
25 #include "llvm/ProfileData/InstrProfReader.h"
26 #include "llvm/Support/FileSystem.h"
27 #include "llvm/Support/Path.h"
30 // This selects the coverage mapping format defined when `InstrProfData.inc`
31 // is textually included.
34 static llvm::cl::opt
<bool> EmptyLineCommentCoverage(
35 "emptyline-comment-coverage",
36 llvm::cl::desc("Emit emptylines and comment lines as skipped regions (only "
37 "disable it on test)"),
38 llvm::cl::init(true), llvm::cl::Hidden
);
40 static llvm::cl::opt
<bool> SystemHeadersCoverage(
41 "system-headers-coverage",
42 llvm::cl::desc("Enable collecting coverage from system headers"),
43 llvm::cl::init(false), llvm::cl::Hidden
);
45 using namespace clang
;
46 using namespace CodeGen
;
47 using namespace llvm::coverage
;
50 CoverageMappingModuleGen::setUpCoverageCallbacks(Preprocessor
&PP
) {
51 CoverageSourceInfo
*CoverageInfo
=
52 new CoverageSourceInfo(PP
.getSourceManager());
53 PP
.addPPCallbacks(std::unique_ptr
<PPCallbacks
>(CoverageInfo
));
54 if (EmptyLineCommentCoverage
) {
55 PP
.addCommentHandler(CoverageInfo
);
56 PP
.setEmptylineHandler(CoverageInfo
);
57 PP
.setPreprocessToken(true);
58 PP
.setTokenWatcher([CoverageInfo
](clang::Token Tok
) {
59 // Update previous token location.
60 CoverageInfo
->PrevTokLoc
= Tok
.getLocation();
61 if (Tok
.getKind() != clang::tok::eod
)
62 CoverageInfo
->updateNextTokLoc(Tok
.getLocation());
68 void CoverageSourceInfo::AddSkippedRange(SourceRange Range
,
69 SkippedRange::Kind RangeKind
) {
70 if (EmptyLineCommentCoverage
&& !SkippedRanges
.empty() &&
71 PrevTokLoc
== SkippedRanges
.back().PrevTokLoc
&&
72 SourceMgr
.isWrittenInSameFile(SkippedRanges
.back().Range
.getEnd(),
74 SkippedRanges
.back().Range
.setEnd(Range
.getEnd());
76 SkippedRanges
.push_back({Range
, RangeKind
, PrevTokLoc
});
79 void CoverageSourceInfo::SourceRangeSkipped(SourceRange Range
, SourceLocation
) {
80 AddSkippedRange(Range
, SkippedRange::PPIfElse
);
83 void CoverageSourceInfo::HandleEmptyline(SourceRange Range
) {
84 AddSkippedRange(Range
, SkippedRange::EmptyLine
);
87 bool CoverageSourceInfo::HandleComment(Preprocessor
&PP
, SourceRange Range
) {
88 AddSkippedRange(Range
, SkippedRange::Comment
);
92 void CoverageSourceInfo::updateNextTokLoc(SourceLocation Loc
) {
93 if (!SkippedRanges
.empty() && SkippedRanges
.back().NextTokLoc
.isInvalid())
94 SkippedRanges
.back().NextTokLoc
= Loc
;
99 /// A region of source code that can be mapped to a counter.
100 class SourceMappingRegion
{
101 /// Primary Counter that is also used for Branch Regions for "True" branches.
104 /// Secondary Counter used for Branch Regions for "False" branches.
105 std::optional
<Counter
> FalseCount
;
107 /// The region's starting location.
108 std::optional
<SourceLocation
> LocStart
;
110 /// The region's ending location.
111 std::optional
<SourceLocation
> LocEnd
;
113 /// Whether this region is a gap region. The count from a gap region is set
114 /// as the line execution count if there are no other regions on the line.
118 SourceMappingRegion(Counter Count
, std::optional
<SourceLocation
> LocStart
,
119 std::optional
<SourceLocation
> LocEnd
,
120 bool GapRegion
= false)
121 : Count(Count
), LocStart(LocStart
), LocEnd(LocEnd
), GapRegion(GapRegion
) {
124 SourceMappingRegion(Counter Count
, std::optional
<Counter
> FalseCount
,
125 std::optional
<SourceLocation
> LocStart
,
126 std::optional
<SourceLocation
> LocEnd
,
127 bool GapRegion
= false)
128 : Count(Count
), FalseCount(FalseCount
), LocStart(LocStart
),
129 LocEnd(LocEnd
), GapRegion(GapRegion
) {}
131 const Counter
&getCounter() const { return Count
; }
133 const Counter
&getFalseCounter() const {
134 assert(FalseCount
&& "Region has no alternate counter");
138 void setCounter(Counter C
) { Count
= C
; }
140 bool hasStartLoc() const { return LocStart
.has_value(); }
142 void setStartLoc(SourceLocation Loc
) { LocStart
= Loc
; }
144 SourceLocation
getBeginLoc() const {
145 assert(LocStart
&& "Region has no start location");
149 bool hasEndLoc() const { return LocEnd
.has_value(); }
151 void setEndLoc(SourceLocation Loc
) {
152 assert(Loc
.isValid() && "Setting an invalid end location");
156 SourceLocation
getEndLoc() const {
157 assert(LocEnd
&& "Region has no end location");
161 bool isGap() const { return GapRegion
; }
163 void setGap(bool Gap
) { GapRegion
= Gap
; }
165 bool isBranch() const { return FalseCount
.has_value(); }
168 /// Spelling locations for the start and end of a source region.
169 struct SpellingRegion
{
170 /// The line where the region starts.
173 /// The column where the region starts.
174 unsigned ColumnStart
;
176 /// The line where the region ends.
179 /// The column where the region ends.
182 SpellingRegion(SourceManager
&SM
, SourceLocation LocStart
,
183 SourceLocation LocEnd
) {
184 LineStart
= SM
.getSpellingLineNumber(LocStart
);
185 ColumnStart
= SM
.getSpellingColumnNumber(LocStart
);
186 LineEnd
= SM
.getSpellingLineNumber(LocEnd
);
187 ColumnEnd
= SM
.getSpellingColumnNumber(LocEnd
);
190 SpellingRegion(SourceManager
&SM
, SourceMappingRegion
&R
)
191 : SpellingRegion(SM
, R
.getBeginLoc(), R
.getEndLoc()) {}
193 /// Check if the start and end locations appear in source order, i.e
194 /// top->bottom, left->right.
195 bool isInSourceOrder() const {
196 return (LineStart
< LineEnd
) ||
197 (LineStart
== LineEnd
&& ColumnStart
<= ColumnEnd
);
201 /// Provides the common functionality for the different
202 /// coverage mapping region builders.
203 class CoverageMappingBuilder
{
205 CoverageMappingModuleGen
&CVM
;
207 const LangOptions
&LangOpts
;
210 /// Map of clang's FileIDs to IDs used for coverage mapping.
211 llvm::SmallDenseMap
<FileID
, std::pair
<unsigned, SourceLocation
>, 8>
215 /// The coverage mapping regions for this function
216 llvm::SmallVector
<CounterMappingRegion
, 32> MappingRegions
;
217 /// The source mapping regions for this function.
218 std::vector
<SourceMappingRegion
> SourceRegions
;
220 /// A set of regions which can be used as a filter.
222 /// It is produced by emitExpansionRegions() and is used in
223 /// emitSourceRegions() to suppress producing code regions if
224 /// the same area is covered by expansion regions.
225 typedef llvm::SmallSet
<std::pair
<SourceLocation
, SourceLocation
>, 8>
228 CoverageMappingBuilder(CoverageMappingModuleGen
&CVM
, SourceManager
&SM
,
229 const LangOptions
&LangOpts
)
230 : CVM(CVM
), SM(SM
), LangOpts(LangOpts
) {}
232 /// Return the precise end location for the given token.
233 SourceLocation
getPreciseTokenLocEnd(SourceLocation Loc
) {
234 // We avoid getLocForEndOfToken here, because it doesn't do what we want for
235 // macro locations, which we just treat as expanded files.
237 Lexer::MeasureTokenLength(SM
.getSpellingLoc(Loc
), SM
, LangOpts
);
238 return Loc
.getLocWithOffset(TokLen
);
241 /// Return the start location of an included file or expanded macro.
242 SourceLocation
getStartOfFileOrMacro(SourceLocation Loc
) {
244 return Loc
.getLocWithOffset(-SM
.getFileOffset(Loc
));
245 return SM
.getLocForStartOfFile(SM
.getFileID(Loc
));
248 /// Return the end location of an included file or expanded macro.
249 SourceLocation
getEndOfFileOrMacro(SourceLocation Loc
) {
251 return Loc
.getLocWithOffset(SM
.getFileIDSize(SM
.getFileID(Loc
)) -
252 SM
.getFileOffset(Loc
));
253 return SM
.getLocForEndOfFile(SM
.getFileID(Loc
));
256 /// Find out where the current file is included or macro is expanded.
257 SourceLocation
getIncludeOrExpansionLoc(SourceLocation Loc
) {
258 return Loc
.isMacroID() ? SM
.getImmediateExpansionRange(Loc
).getBegin()
259 : SM
.getIncludeLoc(SM
.getFileID(Loc
));
262 /// Return true if \c Loc is a location in a built-in macro.
263 bool isInBuiltin(SourceLocation Loc
) {
264 return SM
.getBufferName(SM
.getSpellingLoc(Loc
)) == "<built-in>";
267 /// Check whether \c Loc is included or expanded from \c Parent.
268 bool isNestedIn(SourceLocation Loc
, FileID Parent
) {
270 Loc
= getIncludeOrExpansionLoc(Loc
);
273 } while (!SM
.isInFileID(Loc
, Parent
));
277 /// Get the start of \c S ignoring macro arguments and builtin macros.
278 SourceLocation
getStart(const Stmt
*S
) {
279 SourceLocation Loc
= S
->getBeginLoc();
280 while (SM
.isMacroArgExpansion(Loc
) || isInBuiltin(Loc
))
281 Loc
= SM
.getImmediateExpansionRange(Loc
).getBegin();
285 /// Get the end of \c S ignoring macro arguments and builtin macros.
286 SourceLocation
getEnd(const Stmt
*S
) {
287 SourceLocation Loc
= S
->getEndLoc();
288 while (SM
.isMacroArgExpansion(Loc
) || isInBuiltin(Loc
))
289 Loc
= SM
.getImmediateExpansionRange(Loc
).getBegin();
290 return getPreciseTokenLocEnd(Loc
);
293 /// Find the set of files we have regions for and assign IDs
295 /// Fills \c Mapping with the virtual file mapping needed to write out
296 /// coverage and collects the necessary file information to emit source and
297 /// expansion regions.
298 void gatherFileIDs(SmallVectorImpl
<unsigned> &Mapping
) {
299 FileIDMapping
.clear();
301 llvm::SmallSet
<FileID
, 8> Visited
;
302 SmallVector
<std::pair
<SourceLocation
, unsigned>, 8> FileLocs
;
303 for (const auto &Region
: SourceRegions
) {
304 SourceLocation Loc
= Region
.getBeginLoc();
305 FileID File
= SM
.getFileID(Loc
);
306 if (!Visited
.insert(File
).second
)
309 // Do not map FileID's associated with system headers unless collecting
310 // coverage from system headers is explicitly enabled.
311 if (!SystemHeadersCoverage
&& SM
.isInSystemHeader(SM
.getSpellingLoc(Loc
)))
315 for (SourceLocation Parent
= getIncludeOrExpansionLoc(Loc
);
316 Parent
.isValid(); Parent
= getIncludeOrExpansionLoc(Parent
))
318 FileLocs
.push_back(std::make_pair(Loc
, Depth
));
320 llvm::stable_sort(FileLocs
, llvm::less_second());
322 for (const auto &FL
: FileLocs
) {
323 SourceLocation Loc
= FL
.first
;
324 FileID SpellingFile
= SM
.getDecomposedSpellingLoc(Loc
).first
;
325 auto Entry
= SM
.getFileEntryRefForID(SpellingFile
);
329 FileIDMapping
[SM
.getFileID(Loc
)] = std::make_pair(Mapping
.size(), Loc
);
330 Mapping
.push_back(CVM
.getFileID(*Entry
));
334 /// Get the coverage mapping file ID for \c Loc.
336 /// If such file id doesn't exist, return std::nullopt.
337 std::optional
<unsigned> getCoverageFileID(SourceLocation Loc
) {
338 auto Mapping
= FileIDMapping
.find(SM
.getFileID(Loc
));
339 if (Mapping
!= FileIDMapping
.end())
340 return Mapping
->second
.first
;
344 /// This shrinks the skipped range if it spans a line that contains a
345 /// non-comment token. If shrinking the skipped range would make it empty,
346 /// this returns std::nullopt.
347 /// Note this function can potentially be expensive because
348 /// getSpellingLineNumber uses getLineNumber, which is expensive.
349 std::optional
<SpellingRegion
> adjustSkippedRange(SourceManager
&SM
,
350 SourceLocation LocStart
,
351 SourceLocation LocEnd
,
352 SourceLocation PrevTokLoc
,
353 SourceLocation NextTokLoc
) {
354 SpellingRegion SR
{SM
, LocStart
, LocEnd
};
356 if (PrevTokLoc
.isValid() && SM
.isWrittenInSameFile(LocStart
, PrevTokLoc
) &&
357 SR
.LineStart
== SM
.getSpellingLineNumber(PrevTokLoc
))
359 if (NextTokLoc
.isValid() && SM
.isWrittenInSameFile(LocEnd
, NextTokLoc
) &&
360 SR
.LineEnd
== SM
.getSpellingLineNumber(NextTokLoc
)) {
364 if (SR
.isInSourceOrder())
369 /// Gather all the regions that were skipped by the preprocessor
370 /// using the constructs like #if or comments.
371 void gatherSkippedRegions() {
372 /// An array of the minimum lineStarts and the maximum lineEnds
373 /// for mapping regions from the appropriate source files.
374 llvm::SmallVector
<std::pair
<unsigned, unsigned>, 8> FileLineRanges
;
375 FileLineRanges
.resize(
376 FileIDMapping
.size(),
377 std::make_pair(std::numeric_limits
<unsigned>::max(), 0));
378 for (const auto &R
: MappingRegions
) {
379 FileLineRanges
[R
.FileID
].first
=
380 std::min(FileLineRanges
[R
.FileID
].first
, R
.LineStart
);
381 FileLineRanges
[R
.FileID
].second
=
382 std::max(FileLineRanges
[R
.FileID
].second
, R
.LineEnd
);
385 auto SkippedRanges
= CVM
.getSourceInfo().getSkippedRanges();
386 for (auto &I
: SkippedRanges
) {
387 SourceRange Range
= I
.Range
;
388 auto LocStart
= Range
.getBegin();
389 auto LocEnd
= Range
.getEnd();
390 assert(SM
.isWrittenInSameFile(LocStart
, LocEnd
) &&
391 "region spans multiple files");
393 auto CovFileID
= getCoverageFileID(LocStart
);
396 std::optional
<SpellingRegion
> SR
;
398 SR
= adjustSkippedRange(SM
, LocStart
, LocEnd
, I
.PrevTokLoc
,
400 else if (I
.isPPIfElse() || I
.isEmptyLine())
401 SR
= {SM
, LocStart
, LocEnd
};
405 auto Region
= CounterMappingRegion::makeSkipped(
406 *CovFileID
, SR
->LineStart
, SR
->ColumnStart
, SR
->LineEnd
,
408 // Make sure that we only collect the regions that are inside
409 // the source code of this function.
410 if (Region
.LineStart
>= FileLineRanges
[*CovFileID
].first
&&
411 Region
.LineEnd
<= FileLineRanges
[*CovFileID
].second
)
412 MappingRegions
.push_back(Region
);
416 /// Generate the coverage counter mapping regions from collected
418 void emitSourceRegions(const SourceRegionFilter
&Filter
) {
419 for (const auto &Region
: SourceRegions
) {
420 assert(Region
.hasEndLoc() && "incomplete region");
422 SourceLocation LocStart
= Region
.getBeginLoc();
423 assert(SM
.getFileID(LocStart
).isValid() && "region in invalid file");
425 // Ignore regions from system headers unless collecting coverage from
426 // system headers is explicitly enabled.
427 if (!SystemHeadersCoverage
&&
428 SM
.isInSystemHeader(SM
.getSpellingLoc(LocStart
)))
431 auto CovFileID
= getCoverageFileID(LocStart
);
432 // Ignore regions that don't have a file, such as builtin macros.
436 SourceLocation LocEnd
= Region
.getEndLoc();
437 assert(SM
.isWrittenInSameFile(LocStart
, LocEnd
) &&
438 "region spans multiple files");
440 // Don't add code regions for the area covered by expansion regions.
441 // This not only suppresses redundant regions, but sometimes prevents
442 // creating regions with wrong counters if, for example, a statement's
443 // body ends at the end of a nested macro.
444 if (Filter
.count(std::make_pair(LocStart
, LocEnd
)))
447 // Find the spelling locations for the mapping region.
448 SpellingRegion SR
{SM
, LocStart
, LocEnd
};
449 assert(SR
.isInSourceOrder() && "region start and end out of order");
451 if (Region
.isGap()) {
452 MappingRegions
.push_back(CounterMappingRegion::makeGapRegion(
453 Region
.getCounter(), *CovFileID
, SR
.LineStart
, SR
.ColumnStart
,
454 SR
.LineEnd
, SR
.ColumnEnd
));
455 } else if (Region
.isBranch()) {
456 MappingRegions
.push_back(CounterMappingRegion::makeBranchRegion(
457 Region
.getCounter(), Region
.getFalseCounter(), *CovFileID
,
458 SR
.LineStart
, SR
.ColumnStart
, SR
.LineEnd
, SR
.ColumnEnd
));
460 MappingRegions
.push_back(CounterMappingRegion::makeRegion(
461 Region
.getCounter(), *CovFileID
, SR
.LineStart
, SR
.ColumnStart
,
462 SR
.LineEnd
, SR
.ColumnEnd
));
467 /// Generate expansion regions for each virtual file we've seen.
468 SourceRegionFilter
emitExpansionRegions() {
469 SourceRegionFilter Filter
;
470 for (const auto &FM
: FileIDMapping
) {
471 SourceLocation ExpandedLoc
= FM
.second
.second
;
472 SourceLocation ParentLoc
= getIncludeOrExpansionLoc(ExpandedLoc
);
473 if (ParentLoc
.isInvalid())
476 auto ParentFileID
= getCoverageFileID(ParentLoc
);
479 auto ExpandedFileID
= getCoverageFileID(ExpandedLoc
);
480 assert(ExpandedFileID
&& "expansion in uncovered file");
482 SourceLocation LocEnd
= getPreciseTokenLocEnd(ParentLoc
);
483 assert(SM
.isWrittenInSameFile(ParentLoc
, LocEnd
) &&
484 "region spans multiple files");
485 Filter
.insert(std::make_pair(ParentLoc
, LocEnd
));
487 SpellingRegion SR
{SM
, ParentLoc
, LocEnd
};
488 assert(SR
.isInSourceOrder() && "region start and end out of order");
489 MappingRegions
.push_back(CounterMappingRegion::makeExpansion(
490 *ParentFileID
, *ExpandedFileID
, SR
.LineStart
, SR
.ColumnStart
,
491 SR
.LineEnd
, SR
.ColumnEnd
));
497 /// Creates unreachable coverage regions for the functions that
499 struct EmptyCoverageMappingBuilder
: public CoverageMappingBuilder
{
500 EmptyCoverageMappingBuilder(CoverageMappingModuleGen
&CVM
, SourceManager
&SM
,
501 const LangOptions
&LangOpts
)
502 : CoverageMappingBuilder(CVM
, SM
, LangOpts
) {}
504 void VisitDecl(const Decl
*D
) {
507 auto Body
= D
->getBody();
508 SourceLocation Start
= getStart(Body
);
509 SourceLocation End
= getEnd(Body
);
510 if (!SM
.isWrittenInSameFile(Start
, End
)) {
511 // Walk up to find the common ancestor.
512 // Correct the locations accordingly.
513 FileID StartFileID
= SM
.getFileID(Start
);
514 FileID EndFileID
= SM
.getFileID(End
);
515 while (StartFileID
!= EndFileID
&& !isNestedIn(End
, StartFileID
)) {
516 Start
= getIncludeOrExpansionLoc(Start
);
517 assert(Start
.isValid() &&
518 "Declaration start location not nested within a known region");
519 StartFileID
= SM
.getFileID(Start
);
521 while (StartFileID
!= EndFileID
) {
522 End
= getPreciseTokenLocEnd(getIncludeOrExpansionLoc(End
));
523 assert(End
.isValid() &&
524 "Declaration end location not nested within a known region");
525 EndFileID
= SM
.getFileID(End
);
528 SourceRegions
.emplace_back(Counter(), Start
, End
);
531 /// Write the mapping data to the output stream
532 void write(llvm::raw_ostream
&OS
) {
533 SmallVector
<unsigned, 16> FileIDMapping
;
534 gatherFileIDs(FileIDMapping
);
535 emitSourceRegions(SourceRegionFilter());
537 if (MappingRegions
.empty())
540 CoverageMappingWriter
Writer(FileIDMapping
, std::nullopt
, MappingRegions
);
545 /// A StmtVisitor that creates coverage mapping regions which map
546 /// from the source code locations to the PGO counters.
547 struct CounterCoverageMappingBuilder
548 : public CoverageMappingBuilder
,
549 public ConstStmtVisitor
<CounterCoverageMappingBuilder
> {
550 /// The map of statements to count values.
551 llvm::DenseMap
<const Stmt
*, unsigned> &CounterMap
;
553 /// A stack of currently live regions.
554 std::vector
<SourceMappingRegion
> RegionStack
;
556 CounterExpressionBuilder Builder
;
558 /// A location in the most recently visited file or macro.
560 /// This is used to adjust the active source regions appropriately when
561 /// expressions cross file or macro boundaries.
562 SourceLocation MostRecentLocation
;
564 /// Whether the visitor at a terminate statement.
565 bool HasTerminateStmt
= false;
567 /// Gap region counter after terminate statement.
568 Counter GapRegionCounter
;
570 /// Return a counter for the subtraction of \c RHS from \c LHS
571 Counter
subtractCounters(Counter LHS
, Counter RHS
, bool Simplify
= true) {
572 return Builder
.subtract(LHS
, RHS
, Simplify
);
575 /// Return a counter for the sum of \c LHS and \c RHS.
576 Counter
addCounters(Counter LHS
, Counter RHS
, bool Simplify
= true) {
577 return Builder
.add(LHS
, RHS
, Simplify
);
580 Counter
addCounters(Counter C1
, Counter C2
, Counter C3
,
581 bool Simplify
= true) {
582 return addCounters(addCounters(C1
, C2
, Simplify
), C3
, Simplify
);
585 /// Return the region counter for the given statement.
587 /// This should only be called on statements that have a dedicated counter.
588 Counter
getRegionCounter(const Stmt
*S
) {
589 return Counter::getCounter(CounterMap
[S
]);
592 /// Push a region onto the stack.
594 /// Returns the index on the stack where the region was pushed. This can be
595 /// used with popRegions to exit a "scope", ending the region that was pushed.
596 size_t pushRegion(Counter Count
,
597 std::optional
<SourceLocation
> StartLoc
= std::nullopt
,
598 std::optional
<SourceLocation
> EndLoc
= std::nullopt
,
599 std::optional
<Counter
> FalseCount
= std::nullopt
) {
601 if (StartLoc
&& !FalseCount
) {
602 MostRecentLocation
= *StartLoc
;
605 // If either of these locations is invalid, something elsewhere in the
606 // compiler has broken.
607 assert((!StartLoc
|| StartLoc
->isValid()) && "Start location is not valid");
608 assert((!EndLoc
|| EndLoc
->isValid()) && "End location is not valid");
610 // However, we can still recover without crashing.
611 // If either location is invalid, set it to std::nullopt to avoid
612 // letting users of RegionStack think that region has a valid start/end
614 if (StartLoc
&& StartLoc
->isInvalid())
615 StartLoc
= std::nullopt
;
616 if (EndLoc
&& EndLoc
->isInvalid())
617 EndLoc
= std::nullopt
;
618 RegionStack
.emplace_back(Count
, FalseCount
, StartLoc
, EndLoc
);
620 return RegionStack
.size() - 1;
623 size_t locationDepth(SourceLocation Loc
) {
625 while (Loc
.isValid()) {
626 Loc
= getIncludeOrExpansionLoc(Loc
);
632 /// Pop regions from the stack into the function's list of regions.
634 /// Adds all regions from \c ParentIndex to the top of the stack to the
635 /// function's \c SourceRegions.
636 void popRegions(size_t ParentIndex
) {
637 assert(RegionStack
.size() >= ParentIndex
&& "parent not in stack");
638 while (RegionStack
.size() > ParentIndex
) {
639 SourceMappingRegion
&Region
= RegionStack
.back();
640 if (Region
.hasStartLoc() &&
641 (Region
.hasEndLoc() || RegionStack
[ParentIndex
].hasEndLoc())) {
642 SourceLocation StartLoc
= Region
.getBeginLoc();
643 SourceLocation EndLoc
= Region
.hasEndLoc()
645 : RegionStack
[ParentIndex
].getEndLoc();
646 bool isBranch
= Region
.isBranch();
647 size_t StartDepth
= locationDepth(StartLoc
);
648 size_t EndDepth
= locationDepth(EndLoc
);
649 while (!SM
.isWrittenInSameFile(StartLoc
, EndLoc
)) {
650 bool UnnestStart
= StartDepth
>= EndDepth
;
651 bool UnnestEnd
= EndDepth
>= StartDepth
;
653 // The region ends in a nested file or macro expansion. If the
654 // region is not a branch region, create a separate region for each
655 // expansion, and for all regions, update the EndLoc. Branch
656 // regions should not be split in order to keep a straightforward
657 // correspondance between the region and its associated branch
658 // condition, even if the condition spans multiple depths.
659 SourceLocation NestedLoc
= getStartOfFileOrMacro(EndLoc
);
660 assert(SM
.isWrittenInSameFile(NestedLoc
, EndLoc
));
662 if (!isBranch
&& !isRegionAlreadyAdded(NestedLoc
, EndLoc
))
663 SourceRegions
.emplace_back(Region
.getCounter(), NestedLoc
,
666 EndLoc
= getPreciseTokenLocEnd(getIncludeOrExpansionLoc(EndLoc
));
667 if (EndLoc
.isInvalid())
668 llvm::report_fatal_error(
669 "File exit not handled before popRegions");
673 // The region ends in a nested file or macro expansion. If the
674 // region is not a branch region, create a separate region for each
675 // expansion, and for all regions, update the StartLoc. Branch
676 // regions should not be split in order to keep a straightforward
677 // correspondance between the region and its associated branch
678 // condition, even if the condition spans multiple depths.
679 SourceLocation NestedLoc
= getEndOfFileOrMacro(StartLoc
);
680 assert(SM
.isWrittenInSameFile(StartLoc
, NestedLoc
));
682 if (!isBranch
&& !isRegionAlreadyAdded(StartLoc
, NestedLoc
))
683 SourceRegions
.emplace_back(Region
.getCounter(), StartLoc
,
686 StartLoc
= getIncludeOrExpansionLoc(StartLoc
);
687 if (StartLoc
.isInvalid())
688 llvm::report_fatal_error(
689 "File exit not handled before popRegions");
693 Region
.setStartLoc(StartLoc
);
694 Region
.setEndLoc(EndLoc
);
697 MostRecentLocation
= EndLoc
;
698 // If this region happens to span an entire expansion, we need to
699 // make sure we don't overlap the parent region with it.
700 if (StartLoc
== getStartOfFileOrMacro(StartLoc
) &&
701 EndLoc
== getEndOfFileOrMacro(EndLoc
))
702 MostRecentLocation
= getIncludeOrExpansionLoc(EndLoc
);
705 assert(SM
.isWrittenInSameFile(Region
.getBeginLoc(), EndLoc
));
706 assert(SpellingRegion(SM
, Region
).isInSourceOrder());
707 SourceRegions
.push_back(Region
);
709 RegionStack
.pop_back();
713 /// Return the currently active region.
714 SourceMappingRegion
&getRegion() {
715 assert(!RegionStack
.empty() && "statement has no region");
716 return RegionStack
.back();
719 /// Propagate counts through the children of \p S if \p VisitChildren is true.
720 /// Otherwise, only emit a count for \p S itself.
721 Counter
propagateCounts(Counter TopCount
, const Stmt
*S
,
722 bool VisitChildren
= true) {
723 SourceLocation StartLoc
= getStart(S
);
724 SourceLocation EndLoc
= getEnd(S
);
725 size_t Index
= pushRegion(TopCount
, StartLoc
, EndLoc
);
728 Counter ExitCount
= getRegion().getCounter();
731 // The statement may be spanned by an expansion. Make sure we handle a file
732 // exit out of this expansion before moving to the next statement.
733 if (SM
.isBeforeInTranslationUnit(StartLoc
, S
->getBeginLoc()))
734 MostRecentLocation
= EndLoc
;
739 /// Determine whether the given condition can be constant folded.
740 bool ConditionFoldsToBool(const Expr
*Cond
) {
741 Expr::EvalResult Result
;
742 return (Cond
->EvaluateAsInt(Result
, CVM
.getCodeGenModule().getContext()));
745 /// Create a Branch Region around an instrumentable condition for coverage
746 /// and add it to the function's SourceRegions. A branch region tracks a
747 /// "True" counter and a "False" counter for boolean expressions that
748 /// result in the generation of a branch.
749 void createBranchRegion(const Expr
*C
, Counter TrueCnt
, Counter FalseCnt
) {
750 // Check for NULL conditions.
754 // Ensure we are an instrumentable condition (i.e. no "&&" or "||"). Push
755 // region onto RegionStack but immediately pop it (which adds it to the
756 // function's SourceRegions) because it doesn't apply to any other source
757 // code other than the Condition.
758 if (CodeGenFunction::isInstrumentedCondition(C
)) {
759 // If a condition can fold to true or false, the corresponding branch
760 // will be removed. Create a region with both counters hard-coded to
761 // zero. This allows us to visualize them in a special way.
762 // Alternatively, we can prevent any optimization done via
763 // constant-folding by ensuring that ConstantFoldsToSimpleInteger() in
764 // CodeGenFunction.c always returns false, but that is very heavy-handed.
765 if (ConditionFoldsToBool(C
))
766 popRegions(pushRegion(Counter::getZero(), getStart(C
), getEnd(C
),
767 Counter::getZero()));
769 // Otherwise, create a region with the True counter and False counter.
770 popRegions(pushRegion(TrueCnt
, getStart(C
), getEnd(C
), FalseCnt
));
774 /// Create a Branch Region around a SwitchCase for code coverage
775 /// and add it to the function's SourceRegions.
776 void createSwitchCaseRegion(const SwitchCase
*SC
, Counter TrueCnt
,
778 // Push region onto RegionStack but immediately pop it (which adds it to
779 // the function's SourceRegions) because it doesn't apply to any other
780 // source other than the SwitchCase.
781 popRegions(pushRegion(TrueCnt
, getStart(SC
), SC
->getColonLoc(), FalseCnt
));
784 /// Check whether a region with bounds \c StartLoc and \c EndLoc
785 /// is already added to \c SourceRegions.
786 bool isRegionAlreadyAdded(SourceLocation StartLoc
, SourceLocation EndLoc
,
787 bool isBranch
= false) {
789 llvm::reverse(SourceRegions
), [&](const SourceMappingRegion
&Region
) {
790 return Region
.getBeginLoc() == StartLoc
&&
791 Region
.getEndLoc() == EndLoc
&& Region
.isBranch() == isBranch
;
795 /// Adjust the most recently visited location to \c EndLoc.
797 /// This should be used after visiting any statements in non-source order.
798 void adjustForOutOfOrderTraversal(SourceLocation EndLoc
) {
799 MostRecentLocation
= EndLoc
;
800 // The code region for a whole macro is created in handleFileExit() when
801 // it detects exiting of the virtual file of that macro. If we visited
802 // statements in non-source order, we might already have such a region
803 // added, for example, if a body of a loop is divided among multiple
804 // macros. Avoid adding duplicate regions in such case.
805 if (getRegion().hasEndLoc() &&
806 MostRecentLocation
== getEndOfFileOrMacro(MostRecentLocation
) &&
807 isRegionAlreadyAdded(getStartOfFileOrMacro(MostRecentLocation
),
808 MostRecentLocation
, getRegion().isBranch()))
809 MostRecentLocation
= getIncludeOrExpansionLoc(MostRecentLocation
);
812 /// Adjust regions and state when \c NewLoc exits a file.
814 /// If moving from our most recently tracked location to \c NewLoc exits any
815 /// files, this adjusts our current region stack and creates the file regions
816 /// for the exited file.
817 void handleFileExit(SourceLocation NewLoc
) {
818 if (NewLoc
.isInvalid() ||
819 SM
.isWrittenInSameFile(MostRecentLocation
, NewLoc
))
822 // If NewLoc is not in a file that contains MostRecentLocation, walk up to
823 // find the common ancestor.
824 SourceLocation LCA
= NewLoc
;
825 FileID ParentFile
= SM
.getFileID(LCA
);
826 while (!isNestedIn(MostRecentLocation
, ParentFile
)) {
827 LCA
= getIncludeOrExpansionLoc(LCA
);
828 if (LCA
.isInvalid() || SM
.isWrittenInSameFile(LCA
, MostRecentLocation
)) {
829 // Since there isn't a common ancestor, no file was exited. We just need
830 // to adjust our location to the new file.
831 MostRecentLocation
= NewLoc
;
834 ParentFile
= SM
.getFileID(LCA
);
837 llvm::SmallSet
<SourceLocation
, 8> StartLocs
;
838 std::optional
<Counter
> ParentCounter
;
839 for (SourceMappingRegion
&I
: llvm::reverse(RegionStack
)) {
840 if (!I
.hasStartLoc())
842 SourceLocation Loc
= I
.getBeginLoc();
843 if (!isNestedIn(Loc
, ParentFile
)) {
844 ParentCounter
= I
.getCounter();
848 while (!SM
.isInFileID(Loc
, ParentFile
)) {
849 // The most nested region for each start location is the one with the
850 // correct count. We avoid creating redundant regions by stopping once
851 // we've seen this region.
852 if (StartLocs
.insert(Loc
).second
) {
854 SourceRegions
.emplace_back(I
.getCounter(), I
.getFalseCounter(), Loc
,
855 getEndOfFileOrMacro(Loc
), I
.isBranch());
857 SourceRegions
.emplace_back(I
.getCounter(), Loc
,
858 getEndOfFileOrMacro(Loc
));
860 Loc
= getIncludeOrExpansionLoc(Loc
);
862 I
.setStartLoc(getPreciseTokenLocEnd(Loc
));
866 // If the file is contained completely by another region and doesn't
867 // immediately start its own region, the whole file gets a region
868 // corresponding to the parent.
869 SourceLocation Loc
= MostRecentLocation
;
870 while (isNestedIn(Loc
, ParentFile
)) {
871 SourceLocation FileStart
= getStartOfFileOrMacro(Loc
);
872 if (StartLocs
.insert(FileStart
).second
) {
873 SourceRegions
.emplace_back(*ParentCounter
, FileStart
,
874 getEndOfFileOrMacro(Loc
));
875 assert(SpellingRegion(SM
, SourceRegions
.back()).isInSourceOrder());
877 Loc
= getIncludeOrExpansionLoc(Loc
);
881 MostRecentLocation
= NewLoc
;
884 /// Ensure that \c S is included in the current region.
885 void extendRegion(const Stmt
*S
) {
886 SourceMappingRegion
&Region
= getRegion();
887 SourceLocation StartLoc
= getStart(S
);
889 handleFileExit(StartLoc
);
890 if (!Region
.hasStartLoc())
891 Region
.setStartLoc(StartLoc
);
894 /// Mark \c S as a terminator, starting a zero region.
895 void terminateRegion(const Stmt
*S
) {
897 SourceMappingRegion
&Region
= getRegion();
898 SourceLocation EndLoc
= getEnd(S
);
899 if (!Region
.hasEndLoc())
900 Region
.setEndLoc(EndLoc
);
901 pushRegion(Counter::getZero());
902 HasTerminateStmt
= true;
905 /// Find a valid gap range between \p AfterLoc and \p BeforeLoc.
906 std::optional
<SourceRange
> findGapAreaBetween(SourceLocation AfterLoc
,
907 SourceLocation BeforeLoc
) {
908 // If AfterLoc is in function-like macro, use the right parenthesis
910 if (AfterLoc
.isMacroID()) {
911 FileID FID
= SM
.getFileID(AfterLoc
);
912 const SrcMgr::ExpansionInfo
*EI
= &SM
.getSLocEntry(FID
).getExpansion();
913 if (EI
->isFunctionMacroExpansion())
914 AfterLoc
= EI
->getExpansionLocEnd();
917 size_t StartDepth
= locationDepth(AfterLoc
);
918 size_t EndDepth
= locationDepth(BeforeLoc
);
919 while (!SM
.isWrittenInSameFile(AfterLoc
, BeforeLoc
)) {
920 bool UnnestStart
= StartDepth
>= EndDepth
;
921 bool UnnestEnd
= EndDepth
>= StartDepth
;
923 assert(SM
.isWrittenInSameFile(getStartOfFileOrMacro(BeforeLoc
),
926 BeforeLoc
= getIncludeOrExpansionLoc(BeforeLoc
);
927 assert(BeforeLoc
.isValid());
931 assert(SM
.isWrittenInSameFile(AfterLoc
,
932 getEndOfFileOrMacro(AfterLoc
)));
934 AfterLoc
= getIncludeOrExpansionLoc(AfterLoc
);
935 assert(AfterLoc
.isValid());
936 AfterLoc
= getPreciseTokenLocEnd(AfterLoc
);
937 assert(AfterLoc
.isValid());
941 AfterLoc
= getPreciseTokenLocEnd(AfterLoc
);
942 // If the start and end locations of the gap are both within the same macro
943 // file, the range may not be in source order.
944 if (AfterLoc
.isMacroID() || BeforeLoc
.isMacroID())
946 if (!SM
.isWrittenInSameFile(AfterLoc
, BeforeLoc
) ||
947 !SpellingRegion(SM
, AfterLoc
, BeforeLoc
).isInSourceOrder())
949 return {{AfterLoc
, BeforeLoc
}};
952 /// Emit a gap region between \p StartLoc and \p EndLoc with the given count.
953 void fillGapAreaWithCount(SourceLocation StartLoc
, SourceLocation EndLoc
,
955 if (StartLoc
== EndLoc
)
957 assert(SpellingRegion(SM
, StartLoc
, EndLoc
).isInSourceOrder());
958 handleFileExit(StartLoc
);
959 size_t Index
= pushRegion(Count
, StartLoc
, EndLoc
);
960 getRegion().setGap(true);
961 handleFileExit(EndLoc
);
965 /// Keep counts of breaks and continues inside loops.
966 struct BreakContinue
{
968 Counter ContinueCount
;
970 SmallVector
<BreakContinue
, 8> BreakContinueStack
;
972 CounterCoverageMappingBuilder(
973 CoverageMappingModuleGen
&CVM
,
974 llvm::DenseMap
<const Stmt
*, unsigned> &CounterMap
, SourceManager
&SM
,
975 const LangOptions
&LangOpts
)
976 : CoverageMappingBuilder(CVM
, SM
, LangOpts
), CounterMap(CounterMap
) {}
978 /// Write the mapping data to the output stream
979 void write(llvm::raw_ostream
&OS
) {
980 llvm::SmallVector
<unsigned, 8> VirtualFileMapping
;
981 gatherFileIDs(VirtualFileMapping
);
982 SourceRegionFilter Filter
= emitExpansionRegions();
983 emitSourceRegions(Filter
);
984 gatherSkippedRegions();
986 if (MappingRegions
.empty())
989 CoverageMappingWriter
Writer(VirtualFileMapping
, Builder
.getExpressions(),
994 void VisitStmt(const Stmt
*S
) {
995 if (S
->getBeginLoc().isValid())
997 const Stmt
*LastStmt
= nullptr;
998 bool SaveTerminateStmt
= HasTerminateStmt
;
999 HasTerminateStmt
= false;
1000 GapRegionCounter
= Counter::getZero();
1001 for (const Stmt
*Child
: S
->children())
1003 // If last statement contains terminate statements, add a gap area
1004 // between the two statements. Skipping attributed statements, because
1005 // they don't have valid start location.
1006 if (LastStmt
&& HasTerminateStmt
&& !isa
<AttributedStmt
>(Child
)) {
1007 auto Gap
= findGapAreaBetween(getEnd(LastStmt
), getStart(Child
));
1009 fillGapAreaWithCount(Gap
->getBegin(), Gap
->getEnd(),
1011 SaveTerminateStmt
= true;
1012 HasTerminateStmt
= false;
1017 if (SaveTerminateStmt
)
1018 HasTerminateStmt
= true;
1019 handleFileExit(getEnd(S
));
1022 void VisitDecl(const Decl
*D
) {
1023 Stmt
*Body
= D
->getBody();
1025 // Do not propagate region counts into system headers unless collecting
1026 // coverage from system headers is explicitly enabled.
1027 if (!SystemHeadersCoverage
&& Body
&&
1028 SM
.isInSystemHeader(SM
.getSpellingLoc(getStart(Body
))))
1031 // Do not visit the artificial children nodes of defaulted methods. The
1032 // lexer may not be able to report back precise token end locations for
1033 // these children nodes (llvm.org/PR39822), and moreover users will not be
1034 // able to see coverage for them.
1035 Counter BodyCounter
= getRegionCounter(Body
);
1036 bool Defaulted
= false;
1037 if (auto *Method
= dyn_cast
<CXXMethodDecl
>(D
))
1038 Defaulted
= Method
->isDefaulted();
1039 if (auto *Ctor
= dyn_cast
<CXXConstructorDecl
>(D
)) {
1040 for (auto *Initializer
: Ctor
->inits()) {
1041 if (Initializer
->isWritten()) {
1042 auto *Init
= Initializer
->getInit();
1043 if (getStart(Init
).isValid() && getEnd(Init
).isValid())
1044 propagateCounts(BodyCounter
, Init
);
1049 propagateCounts(BodyCounter
, Body
,
1050 /*VisitChildren=*/!Defaulted
);
1051 assert(RegionStack
.empty() && "Regions entered but never exited");
1054 void VisitReturnStmt(const ReturnStmt
*S
) {
1056 if (S
->getRetValue())
1057 Visit(S
->getRetValue());
1061 void VisitCoroutineBodyStmt(const CoroutineBodyStmt
*S
) {
1063 Visit(S
->getBody());
1066 void VisitCoreturnStmt(const CoreturnStmt
*S
) {
1068 if (S
->getOperand())
1069 Visit(S
->getOperand());
1073 void VisitCXXThrowExpr(const CXXThrowExpr
*E
) {
1075 if (E
->getSubExpr())
1076 Visit(E
->getSubExpr());
1080 void VisitGotoStmt(const GotoStmt
*S
) { terminateRegion(S
); }
1082 void VisitLabelStmt(const LabelStmt
*S
) {
1083 Counter LabelCount
= getRegionCounter(S
);
1084 SourceLocation Start
= getStart(S
);
1085 // We can't extendRegion here or we risk overlapping with our new region.
1086 handleFileExit(Start
);
1087 pushRegion(LabelCount
, Start
);
1088 Visit(S
->getSubStmt());
1091 void VisitBreakStmt(const BreakStmt
*S
) {
1092 assert(!BreakContinueStack
.empty() && "break not in a loop or switch!");
1093 BreakContinueStack
.back().BreakCount
= addCounters(
1094 BreakContinueStack
.back().BreakCount
, getRegion().getCounter());
1095 // FIXME: a break in a switch should terminate regions for all preceding
1096 // case statements, not just the most recent one.
1100 void VisitContinueStmt(const ContinueStmt
*S
) {
1101 assert(!BreakContinueStack
.empty() && "continue stmt not in a loop!");
1102 BreakContinueStack
.back().ContinueCount
= addCounters(
1103 BreakContinueStack
.back().ContinueCount
, getRegion().getCounter());
1107 void VisitCallExpr(const CallExpr
*E
) {
1110 // Terminate the region when we hit a noreturn function.
1111 // (This is helpful dealing with switch statements.)
1112 QualType CalleeType
= E
->getCallee()->getType();
1113 if (getFunctionExtInfo(*CalleeType
).getNoReturn())
1117 void VisitWhileStmt(const WhileStmt
*S
) {
1120 Counter ParentCount
= getRegion().getCounter();
1121 Counter BodyCount
= getRegionCounter(S
);
1123 // Handle the body first so that we can get the backedge count.
1124 BreakContinueStack
.push_back(BreakContinue());
1125 extendRegion(S
->getBody());
1126 Counter BackedgeCount
= propagateCounts(BodyCount
, S
->getBody());
1127 BreakContinue BC
= BreakContinueStack
.pop_back_val();
1129 bool BodyHasTerminateStmt
= HasTerminateStmt
;
1130 HasTerminateStmt
= false;
1132 // Go back to handle the condition.
1134 addCounters(ParentCount
, BackedgeCount
, BC
.ContinueCount
);
1135 propagateCounts(CondCount
, S
->getCond());
1136 adjustForOutOfOrderTraversal(getEnd(S
));
1138 // The body count applies to the area immediately after the increment.
1139 auto Gap
= findGapAreaBetween(S
->getRParenLoc(), getStart(S
->getBody()));
1141 fillGapAreaWithCount(Gap
->getBegin(), Gap
->getEnd(), BodyCount
);
1144 addCounters(BC
.BreakCount
, subtractCounters(CondCount
, BodyCount
));
1145 if (OutCount
!= ParentCount
) {
1146 pushRegion(OutCount
);
1147 GapRegionCounter
= OutCount
;
1148 if (BodyHasTerminateStmt
)
1149 HasTerminateStmt
= true;
1152 // Create Branch Region around condition.
1153 createBranchRegion(S
->getCond(), BodyCount
,
1154 subtractCounters(CondCount
, BodyCount
));
1157 void VisitDoStmt(const DoStmt
*S
) {
1160 Counter ParentCount
= getRegion().getCounter();
1161 Counter BodyCount
= getRegionCounter(S
);
1163 BreakContinueStack
.push_back(BreakContinue());
1164 extendRegion(S
->getBody());
1165 Counter BackedgeCount
=
1166 propagateCounts(addCounters(ParentCount
, BodyCount
), S
->getBody());
1167 BreakContinue BC
= BreakContinueStack
.pop_back_val();
1169 bool BodyHasTerminateStmt
= HasTerminateStmt
;
1170 HasTerminateStmt
= false;
1172 Counter CondCount
= addCounters(BackedgeCount
, BC
.ContinueCount
);
1173 propagateCounts(CondCount
, S
->getCond());
1176 addCounters(BC
.BreakCount
, subtractCounters(CondCount
, BodyCount
));
1177 if (OutCount
!= ParentCount
) {
1178 pushRegion(OutCount
);
1179 GapRegionCounter
= OutCount
;
1182 // Create Branch Region around condition.
1183 createBranchRegion(S
->getCond(), BodyCount
,
1184 subtractCounters(CondCount
, BodyCount
));
1186 if (BodyHasTerminateStmt
)
1187 HasTerminateStmt
= true;
1190 void VisitForStmt(const ForStmt
*S
) {
1193 Visit(S
->getInit());
1195 Counter ParentCount
= getRegion().getCounter();
1196 Counter BodyCount
= getRegionCounter(S
);
1198 // The loop increment may contain a break or continue.
1200 BreakContinueStack
.emplace_back();
1202 // Handle the body first so that we can get the backedge count.
1203 BreakContinueStack
.emplace_back();
1204 extendRegion(S
->getBody());
1205 Counter BackedgeCount
= propagateCounts(BodyCount
, S
->getBody());
1206 BreakContinue BodyBC
= BreakContinueStack
.pop_back_val();
1208 bool BodyHasTerminateStmt
= HasTerminateStmt
;
1209 HasTerminateStmt
= false;
1211 // The increment is essentially part of the body but it needs to include
1212 // the count for all the continue statements.
1213 BreakContinue IncrementBC
;
1214 if (const Stmt
*Inc
= S
->getInc()) {
1215 propagateCounts(addCounters(BackedgeCount
, BodyBC
.ContinueCount
), Inc
);
1216 IncrementBC
= BreakContinueStack
.pop_back_val();
1219 // Go back to handle the condition.
1220 Counter CondCount
= addCounters(
1221 addCounters(ParentCount
, BackedgeCount
, BodyBC
.ContinueCount
),
1222 IncrementBC
.ContinueCount
);
1223 if (const Expr
*Cond
= S
->getCond()) {
1224 propagateCounts(CondCount
, Cond
);
1225 adjustForOutOfOrderTraversal(getEnd(S
));
1228 // The body count applies to the area immediately after the increment.
1229 auto Gap
= findGapAreaBetween(S
->getRParenLoc(), getStart(S
->getBody()));
1231 fillGapAreaWithCount(Gap
->getBegin(), Gap
->getEnd(), BodyCount
);
1233 Counter OutCount
= addCounters(BodyBC
.BreakCount
, IncrementBC
.BreakCount
,
1234 subtractCounters(CondCount
, BodyCount
));
1235 if (OutCount
!= ParentCount
) {
1236 pushRegion(OutCount
);
1237 GapRegionCounter
= OutCount
;
1238 if (BodyHasTerminateStmt
)
1239 HasTerminateStmt
= true;
1242 // Create Branch Region around condition.
1243 createBranchRegion(S
->getCond(), BodyCount
,
1244 subtractCounters(CondCount
, BodyCount
));
1247 void VisitCXXForRangeStmt(const CXXForRangeStmt
*S
) {
1250 Visit(S
->getInit());
1251 Visit(S
->getLoopVarStmt());
1252 Visit(S
->getRangeStmt());
1254 Counter ParentCount
= getRegion().getCounter();
1255 Counter BodyCount
= getRegionCounter(S
);
1257 BreakContinueStack
.push_back(BreakContinue());
1258 extendRegion(S
->getBody());
1259 Counter BackedgeCount
= propagateCounts(BodyCount
, S
->getBody());
1260 BreakContinue BC
= BreakContinueStack
.pop_back_val();
1262 bool BodyHasTerminateStmt
= HasTerminateStmt
;
1263 HasTerminateStmt
= false;
1265 // The body count applies to the area immediately after the range.
1266 auto Gap
= findGapAreaBetween(S
->getRParenLoc(), getStart(S
->getBody()));
1268 fillGapAreaWithCount(Gap
->getBegin(), Gap
->getEnd(), BodyCount
);
1271 addCounters(ParentCount
, BackedgeCount
, BC
.ContinueCount
);
1273 addCounters(BC
.BreakCount
, subtractCounters(LoopCount
, BodyCount
));
1274 if (OutCount
!= ParentCount
) {
1275 pushRegion(OutCount
);
1276 GapRegionCounter
= OutCount
;
1277 if (BodyHasTerminateStmt
)
1278 HasTerminateStmt
= true;
1281 // Create Branch Region around condition.
1282 createBranchRegion(S
->getCond(), BodyCount
,
1283 subtractCounters(LoopCount
, BodyCount
));
1286 void VisitObjCForCollectionStmt(const ObjCForCollectionStmt
*S
) {
1288 Visit(S
->getElement());
1290 Counter ParentCount
= getRegion().getCounter();
1291 Counter BodyCount
= getRegionCounter(S
);
1293 BreakContinueStack
.push_back(BreakContinue());
1294 extendRegion(S
->getBody());
1295 Counter BackedgeCount
= propagateCounts(BodyCount
, S
->getBody());
1296 BreakContinue BC
= BreakContinueStack
.pop_back_val();
1298 // The body count applies to the area immediately after the collection.
1299 auto Gap
= findGapAreaBetween(S
->getRParenLoc(), getStart(S
->getBody()));
1301 fillGapAreaWithCount(Gap
->getBegin(), Gap
->getEnd(), BodyCount
);
1304 addCounters(ParentCount
, BackedgeCount
, BC
.ContinueCount
);
1306 addCounters(BC
.BreakCount
, subtractCounters(LoopCount
, BodyCount
));
1307 if (OutCount
!= ParentCount
) {
1308 pushRegion(OutCount
);
1309 GapRegionCounter
= OutCount
;
1313 void VisitSwitchStmt(const SwitchStmt
*S
) {
1316 Visit(S
->getInit());
1317 Visit(S
->getCond());
1319 BreakContinueStack
.push_back(BreakContinue());
1321 const Stmt
*Body
= S
->getBody();
1323 if (const auto *CS
= dyn_cast
<CompoundStmt
>(Body
)) {
1324 if (!CS
->body_empty()) {
1325 // Make a region for the body of the switch. If the body starts with
1326 // a case, that case will reuse this region; otherwise, this covers
1327 // the unreachable code at the beginning of the switch body.
1328 size_t Index
= pushRegion(Counter::getZero(), getStart(CS
));
1329 getRegion().setGap(true);
1332 // Set the end for the body of the switch, if it isn't already set.
1333 for (size_t i
= RegionStack
.size(); i
!= Index
; --i
) {
1334 if (!RegionStack
[i
- 1].hasEndLoc())
1335 RegionStack
[i
- 1].setEndLoc(getEnd(CS
->body_back()));
1341 propagateCounts(Counter::getZero(), Body
);
1342 BreakContinue BC
= BreakContinueStack
.pop_back_val();
1344 if (!BreakContinueStack
.empty())
1345 BreakContinueStack
.back().ContinueCount
= addCounters(
1346 BreakContinueStack
.back().ContinueCount
, BC
.ContinueCount
);
1348 Counter ParentCount
= getRegion().getCounter();
1349 Counter ExitCount
= getRegionCounter(S
);
1350 SourceLocation ExitLoc
= getEnd(S
);
1351 pushRegion(ExitCount
);
1352 GapRegionCounter
= ExitCount
;
1354 // Ensure that handleFileExit recognizes when the end location is located
1355 // in a different file.
1356 MostRecentLocation
= getStart(S
);
1357 handleFileExit(ExitLoc
);
1359 // Create a Branch Region around each Case. Subtract the case's
1360 // counter from the Parent counter to track the "False" branch count.
1361 Counter CaseCountSum
;
1362 bool HasDefaultCase
= false;
1363 const SwitchCase
*Case
= S
->getSwitchCaseList();
1364 for (; Case
; Case
= Case
->getNextSwitchCase()) {
1365 HasDefaultCase
= HasDefaultCase
|| isa
<DefaultStmt
>(Case
);
1367 addCounters(CaseCountSum
, getRegionCounter(Case
), /*Simplify=*/false);
1368 createSwitchCaseRegion(
1369 Case
, getRegionCounter(Case
),
1370 subtractCounters(ParentCount
, getRegionCounter(Case
)));
1372 // Simplify is skipped while building the counters above: it can get really
1373 // slow on top of switches with thousands of cases. Instead, trigger
1374 // simplification by adding zero to the last counter.
1375 CaseCountSum
= addCounters(CaseCountSum
, Counter::getZero());
1377 // If no explicit default case exists, create a branch region to represent
1378 // the hidden branch, which will be added later by the CodeGen. This region
1379 // will be associated with the switch statement's condition.
1380 if (!HasDefaultCase
) {
1381 Counter DefaultTrue
= subtractCounters(ParentCount
, CaseCountSum
);
1382 Counter DefaultFalse
= subtractCounters(ParentCount
, DefaultTrue
);
1383 createBranchRegion(S
->getCond(), DefaultTrue
, DefaultFalse
);
1387 void VisitSwitchCase(const SwitchCase
*S
) {
1390 SourceMappingRegion
&Parent
= getRegion();
1392 Counter Count
= addCounters(Parent
.getCounter(), getRegionCounter(S
));
1393 // Reuse the existing region if it starts at our label. This is typical of
1394 // the first case in a switch.
1395 if (Parent
.hasStartLoc() && Parent
.getBeginLoc() == getStart(S
))
1396 Parent
.setCounter(Count
);
1398 pushRegion(Count
, getStart(S
));
1400 GapRegionCounter
= Count
;
1402 if (const auto *CS
= dyn_cast
<CaseStmt
>(S
)) {
1403 Visit(CS
->getLHS());
1404 if (const Expr
*RHS
= CS
->getRHS())
1407 Visit(S
->getSubStmt());
1410 void VisitIfStmt(const IfStmt
*S
) {
1413 Visit(S
->getInit());
1415 // Extend into the condition before we propagate through it below - this is
1416 // needed to handle macros that generate the "if" but not the condition.
1417 if (!S
->isConsteval())
1418 extendRegion(S
->getCond());
1420 Counter ParentCount
= getRegion().getCounter();
1421 Counter ThenCount
= getRegionCounter(S
);
1423 if (!S
->isConsteval()) {
1424 // Emitting a counter for the condition makes it easier to interpret the
1425 // counter for the body when looking at the coverage.
1426 propagateCounts(ParentCount
, S
->getCond());
1428 // The 'then' count applies to the area immediately after the condition.
1429 std::optional
<SourceRange
> Gap
=
1430 findGapAreaBetween(S
->getRParenLoc(), getStart(S
->getThen()));
1432 fillGapAreaWithCount(Gap
->getBegin(), Gap
->getEnd(), ThenCount
);
1435 extendRegion(S
->getThen());
1436 Counter OutCount
= propagateCounts(ThenCount
, S
->getThen());
1438 Counter ElseCount
= subtractCounters(ParentCount
, ThenCount
);
1439 if (const Stmt
*Else
= S
->getElse()) {
1440 bool ThenHasTerminateStmt
= HasTerminateStmt
;
1441 HasTerminateStmt
= false;
1442 // The 'else' count applies to the area immediately after the 'then'.
1443 std::optional
<SourceRange
> Gap
=
1444 findGapAreaBetween(getEnd(S
->getThen()), getStart(Else
));
1446 fillGapAreaWithCount(Gap
->getBegin(), Gap
->getEnd(), ElseCount
);
1448 OutCount
= addCounters(OutCount
, propagateCounts(ElseCount
, Else
));
1450 if (ThenHasTerminateStmt
)
1451 HasTerminateStmt
= true;
1453 OutCount
= addCounters(OutCount
, ElseCount
);
1455 if (OutCount
!= ParentCount
) {
1456 pushRegion(OutCount
);
1457 GapRegionCounter
= OutCount
;
1460 if (!S
->isConsteval()) {
1461 // Create Branch Region around condition.
1462 createBranchRegion(S
->getCond(), ThenCount
,
1463 subtractCounters(ParentCount
, ThenCount
));
1467 void VisitCXXTryStmt(const CXXTryStmt
*S
) {
1469 // Handle macros that generate the "try" but not the rest.
1470 extendRegion(S
->getTryBlock());
1472 Counter ParentCount
= getRegion().getCounter();
1473 propagateCounts(ParentCount
, S
->getTryBlock());
1475 for (unsigned I
= 0, E
= S
->getNumHandlers(); I
< E
; ++I
)
1476 Visit(S
->getHandler(I
));
1478 Counter ExitCount
= getRegionCounter(S
);
1479 pushRegion(ExitCount
);
1482 void VisitCXXCatchStmt(const CXXCatchStmt
*S
) {
1483 propagateCounts(getRegionCounter(S
), S
->getHandlerBlock());
1486 void VisitAbstractConditionalOperator(const AbstractConditionalOperator
*E
) {
1489 Counter ParentCount
= getRegion().getCounter();
1490 Counter TrueCount
= getRegionCounter(E
);
1492 propagateCounts(ParentCount
, E
->getCond());
1495 if (!isa
<BinaryConditionalOperator
>(E
)) {
1496 // The 'then' count applies to the area immediately after the condition.
1498 findGapAreaBetween(E
->getQuestionLoc(), getStart(E
->getTrueExpr()));
1500 fillGapAreaWithCount(Gap
->getBegin(), Gap
->getEnd(), TrueCount
);
1502 extendRegion(E
->getTrueExpr());
1503 OutCount
= propagateCounts(TrueCount
, E
->getTrueExpr());
1506 extendRegion(E
->getFalseExpr());
1507 OutCount
= addCounters(
1508 OutCount
, propagateCounts(subtractCounters(ParentCount
, TrueCount
),
1509 E
->getFalseExpr()));
1511 if (OutCount
!= ParentCount
) {
1512 pushRegion(OutCount
);
1513 GapRegionCounter
= OutCount
;
1516 // Create Branch Region around condition.
1517 createBranchRegion(E
->getCond(), TrueCount
,
1518 subtractCounters(ParentCount
, TrueCount
));
1521 void VisitBinLAnd(const BinaryOperator
*E
) {
1522 extendRegion(E
->getLHS());
1523 propagateCounts(getRegion().getCounter(), E
->getLHS());
1524 handleFileExit(getEnd(E
->getLHS()));
1526 // Counter tracks the right hand side of a logical and operator.
1527 extendRegion(E
->getRHS());
1528 propagateCounts(getRegionCounter(E
), E
->getRHS());
1530 // Extract the RHS's Execution Counter.
1531 Counter RHSExecCnt
= getRegionCounter(E
);
1533 // Extract the RHS's "True" Instance Counter.
1534 Counter RHSTrueCnt
= getRegionCounter(E
->getRHS());
1536 // Extract the Parent Region Counter.
1537 Counter ParentCnt
= getRegion().getCounter();
1539 // Create Branch Region around LHS condition.
1540 createBranchRegion(E
->getLHS(), RHSExecCnt
,
1541 subtractCounters(ParentCnt
, RHSExecCnt
));
1543 // Create Branch Region around RHS condition.
1544 createBranchRegion(E
->getRHS(), RHSTrueCnt
,
1545 subtractCounters(RHSExecCnt
, RHSTrueCnt
));
1548 // Determine whether the right side of OR operation need to be visited.
1549 bool shouldVisitRHS(const Expr
*LHS
) {
1550 bool LHSIsTrue
= false;
1551 bool LHSIsConst
= false;
1552 if (!LHS
->isValueDependent())
1553 LHSIsConst
= LHS
->EvaluateAsBooleanCondition(
1554 LHSIsTrue
, CVM
.getCodeGenModule().getContext());
1555 return !LHSIsConst
|| (LHSIsConst
&& !LHSIsTrue
);
1558 void VisitBinLOr(const BinaryOperator
*E
) {
1559 extendRegion(E
->getLHS());
1560 Counter OutCount
= propagateCounts(getRegion().getCounter(), E
->getLHS());
1561 handleFileExit(getEnd(E
->getLHS()));
1563 // Counter tracks the right hand side of a logical or operator.
1564 extendRegion(E
->getRHS());
1565 propagateCounts(getRegionCounter(E
), E
->getRHS());
1567 // Extract the RHS's Execution Counter.
1568 Counter RHSExecCnt
= getRegionCounter(E
);
1570 // Extract the RHS's "False" Instance Counter.
1571 Counter RHSFalseCnt
= getRegionCounter(E
->getRHS());
1573 if (!shouldVisitRHS(E
->getLHS())) {
1574 GapRegionCounter
= OutCount
;
1577 // Extract the Parent Region Counter.
1578 Counter ParentCnt
= getRegion().getCounter();
1580 // Create Branch Region around LHS condition.
1581 createBranchRegion(E
->getLHS(), subtractCounters(ParentCnt
, RHSExecCnt
),
1584 // Create Branch Region around RHS condition.
1585 createBranchRegion(E
->getRHS(), subtractCounters(RHSExecCnt
, RHSFalseCnt
),
1589 void VisitLambdaExpr(const LambdaExpr
*LE
) {
1590 // Lambdas are treated as their own functions for now, so we shouldn't
1591 // propagate counts into them.
1594 void VisitPseudoObjectExpr(const PseudoObjectExpr
*POE
) {
1595 // Just visit syntatic expression as this is what users actually write.
1596 VisitStmt(POE
->getSyntacticForm());
1599 void VisitOpaqueValueExpr(const OpaqueValueExpr
* OVE
) {
1600 Visit(OVE
->getSourceExpr());
1604 } // end anonymous namespace
1606 static void dump(llvm::raw_ostream
&OS
, StringRef FunctionName
,
1607 ArrayRef
<CounterExpression
> Expressions
,
1608 ArrayRef
<CounterMappingRegion
> Regions
) {
1609 OS
<< FunctionName
<< ":\n";
1610 CounterMappingContext
Ctx(Expressions
);
1611 for (const auto &R
: Regions
) {
1614 case CounterMappingRegion::CodeRegion
:
1616 case CounterMappingRegion::ExpansionRegion
:
1619 case CounterMappingRegion::SkippedRegion
:
1622 case CounterMappingRegion::GapRegion
:
1625 case CounterMappingRegion::BranchRegion
:
1630 OS
<< "File " << R
.FileID
<< ", " << R
.LineStart
<< ":" << R
.ColumnStart
1631 << " -> " << R
.LineEnd
<< ":" << R
.ColumnEnd
<< " = ";
1632 Ctx
.dump(R
.Count
, OS
);
1634 if (R
.Kind
== CounterMappingRegion::BranchRegion
) {
1636 Ctx
.dump(R
.FalseCount
, OS
);
1639 if (R
.Kind
== CounterMappingRegion::ExpansionRegion
)
1640 OS
<< " (Expanded file = " << R
.ExpandedFileID
<< ")";
1645 CoverageMappingModuleGen::CoverageMappingModuleGen(
1646 CodeGenModule
&CGM
, CoverageSourceInfo
&SourceInfo
)
1647 : CGM(CGM
), SourceInfo(SourceInfo
) {}
1649 std::string
CoverageMappingModuleGen::getCurrentDirname() {
1650 if (!CGM
.getCodeGenOpts().CoverageCompilationDir
.empty())
1651 return CGM
.getCodeGenOpts().CoverageCompilationDir
;
1653 SmallString
<256> CWD
;
1654 llvm::sys::fs::current_path(CWD
);
1655 return CWD
.str().str();
1658 std::string
CoverageMappingModuleGen::normalizeFilename(StringRef Filename
) {
1659 llvm::SmallString
<256> Path(Filename
);
1660 llvm::sys::path::remove_dots(Path
, /*remove_dot_dot=*/true);
1662 /// Traverse coverage prefix map in reverse order because prefix replacements
1663 /// are applied in reverse order starting from the last one when multiple
1664 /// prefix replacement options are provided.
1665 for (const auto &[From
, To
] :
1666 llvm::reverse(CGM
.getCodeGenOpts().CoveragePrefixMap
)) {
1667 if (llvm::sys::path::replace_path_prefix(Path
, From
, To
))
1670 return Path
.str().str();
1673 static std::string
getInstrProfSection(const CodeGenModule
&CGM
,
1674 llvm::InstrProfSectKind SK
) {
1675 return llvm::getInstrProfSectionName(
1676 SK
, CGM
.getContext().getTargetInfo().getTriple().getObjectFormat());
1679 void CoverageMappingModuleGen::emitFunctionMappingRecord(
1680 const FunctionInfo
&Info
, uint64_t FilenamesRef
) {
1681 llvm::LLVMContext
&Ctx
= CGM
.getLLVMContext();
1683 // Assign a name to the function record. This is used to merge duplicates.
1684 std::string FuncRecordName
= "__covrec_" + llvm::utohexstr(Info
.NameHash
);
1686 // A dummy description for a function included-but-not-used in a TU can be
1687 // replaced by full description provided by a different TU. The two kinds of
1688 // descriptions play distinct roles: therefore, assign them different names
1689 // to prevent `linkonce_odr` merging.
1691 FuncRecordName
+= "u";
1693 // Create the function record type.
1694 const uint64_t NameHash
= Info
.NameHash
;
1695 const uint64_t FuncHash
= Info
.FuncHash
;
1696 const std::string
&CoverageMapping
= Info
.CoverageMapping
;
1697 #define COVMAP_FUNC_RECORD(Type, LLVMType, Name, Init) LLVMType,
1698 llvm::Type
*FunctionRecordTypes
[] = {
1699 #include "llvm/ProfileData/InstrProfData.inc"
1701 auto *FunctionRecordTy
=
1702 llvm::StructType::get(Ctx
, ArrayRef(FunctionRecordTypes
),
1705 // Create the function record constant.
1706 #define COVMAP_FUNC_RECORD(Type, LLVMType, Name, Init) Init,
1707 llvm::Constant
*FunctionRecordVals
[] = {
1708 #include "llvm/ProfileData/InstrProfData.inc"
1710 auto *FuncRecordConstant
=
1711 llvm::ConstantStruct::get(FunctionRecordTy
, ArrayRef(FunctionRecordVals
));
1713 // Create the function record global.
1714 auto *FuncRecord
= new llvm::GlobalVariable(
1715 CGM
.getModule(), FunctionRecordTy
, /*isConstant=*/true,
1716 llvm::GlobalValue::LinkOnceODRLinkage
, FuncRecordConstant
,
1718 FuncRecord
->setVisibility(llvm::GlobalValue::HiddenVisibility
);
1719 FuncRecord
->setSection(getInstrProfSection(CGM
, llvm::IPSK_covfun
));
1720 FuncRecord
->setAlignment(llvm::Align(8));
1721 if (CGM
.supportsCOMDAT())
1722 FuncRecord
->setComdat(CGM
.getModule().getOrInsertComdat(FuncRecordName
));
1724 // Make sure the data doesn't get deleted.
1725 CGM
.addUsedGlobal(FuncRecord
);
1728 void CoverageMappingModuleGen::addFunctionMappingRecord(
1729 llvm::GlobalVariable
*NamePtr
, StringRef NameValue
, uint64_t FuncHash
,
1730 const std::string
&CoverageMapping
, bool IsUsed
) {
1731 const uint64_t NameHash
= llvm::IndexedInstrProf::ComputeHash(NameValue
);
1732 FunctionRecords
.push_back({NameHash
, FuncHash
, CoverageMapping
, IsUsed
});
1735 FunctionNames
.push_back(NamePtr
);
1737 if (CGM
.getCodeGenOpts().DumpCoverageMapping
) {
1738 // Dump the coverage mapping data for this function by decoding the
1739 // encoded data. This allows us to dump the mapping regions which were
1740 // also processed by the CoverageMappingWriter which performs
1741 // additional minimization operations such as reducing the number of
1743 llvm::SmallVector
<std::string
, 16> FilenameStrs
;
1744 std::vector
<StringRef
> Filenames
;
1745 std::vector
<CounterExpression
> Expressions
;
1746 std::vector
<CounterMappingRegion
> Regions
;
1747 FilenameStrs
.resize(FileEntries
.size() + 1);
1748 FilenameStrs
[0] = normalizeFilename(getCurrentDirname());
1749 for (const auto &Entry
: FileEntries
) {
1750 auto I
= Entry
.second
;
1751 FilenameStrs
[I
] = normalizeFilename(Entry
.first
.getName());
1753 ArrayRef
<std::string
> FilenameRefs
= llvm::ArrayRef(FilenameStrs
);
1754 RawCoverageMappingReader
Reader(CoverageMapping
, FilenameRefs
, Filenames
,
1755 Expressions
, Regions
);
1758 dump(llvm::outs(), NameValue
, Expressions
, Regions
);
1762 void CoverageMappingModuleGen::emit() {
1763 if (FunctionRecords
.empty())
1765 llvm::LLVMContext
&Ctx
= CGM
.getLLVMContext();
1766 auto *Int32Ty
= llvm::Type::getInt32Ty(Ctx
);
1768 // Create the filenames and merge them with coverage mappings
1769 llvm::SmallVector
<std::string
, 16> FilenameStrs
;
1770 FilenameStrs
.resize(FileEntries
.size() + 1);
1771 // The first filename is the current working directory.
1772 FilenameStrs
[0] = normalizeFilename(getCurrentDirname());
1773 for (const auto &Entry
: FileEntries
) {
1774 auto I
= Entry
.second
;
1775 FilenameStrs
[I
] = normalizeFilename(Entry
.first
.getName());
1778 std::string Filenames
;
1780 llvm::raw_string_ostream
OS(Filenames
);
1781 CoverageFilenamesSectionWriter(FilenameStrs
).write(OS
);
1783 auto *FilenamesVal
=
1784 llvm::ConstantDataArray::getString(Ctx
, Filenames
, false);
1785 const int64_t FilenamesRef
= llvm::IndexedInstrProf::ComputeHash(Filenames
);
1787 // Emit the function records.
1788 for (const FunctionInfo
&Info
: FunctionRecords
)
1789 emitFunctionMappingRecord(Info
, FilenamesRef
);
1791 const unsigned NRecords
= 0;
1792 const size_t FilenamesSize
= Filenames
.size();
1793 const unsigned CoverageMappingSize
= 0;
1794 llvm::Type
*CovDataHeaderTypes
[] = {
1795 #define COVMAP_HEADER(Type, LLVMType, Name, Init) LLVMType,
1796 #include "llvm/ProfileData/InstrProfData.inc"
1798 auto CovDataHeaderTy
=
1799 llvm::StructType::get(Ctx
, ArrayRef(CovDataHeaderTypes
));
1800 llvm::Constant
*CovDataHeaderVals
[] = {
1801 #define COVMAP_HEADER(Type, LLVMType, Name, Init) Init,
1802 #include "llvm/ProfileData/InstrProfData.inc"
1804 auto CovDataHeaderVal
=
1805 llvm::ConstantStruct::get(CovDataHeaderTy
, ArrayRef(CovDataHeaderVals
));
1807 // Create the coverage data record
1808 llvm::Type
*CovDataTypes
[] = {CovDataHeaderTy
, FilenamesVal
->getType()};
1809 auto CovDataTy
= llvm::StructType::get(Ctx
, ArrayRef(CovDataTypes
));
1810 llvm::Constant
*TUDataVals
[] = {CovDataHeaderVal
, FilenamesVal
};
1811 auto CovDataVal
= llvm::ConstantStruct::get(CovDataTy
, ArrayRef(TUDataVals
));
1812 auto CovData
= new llvm::GlobalVariable(
1813 CGM
.getModule(), CovDataTy
, true, llvm::GlobalValue::PrivateLinkage
,
1814 CovDataVal
, llvm::getCoverageMappingVarName());
1816 CovData
->setSection(getInstrProfSection(CGM
, llvm::IPSK_covmap
));
1817 CovData
->setAlignment(llvm::Align(8));
1819 // Make sure the data doesn't get deleted.
1820 CGM
.addUsedGlobal(CovData
);
1821 // Create the deferred function records array
1822 if (!FunctionNames
.empty()) {
1823 auto NamesArrTy
= llvm::ArrayType::get(llvm::Type::getInt8PtrTy(Ctx
),
1824 FunctionNames
.size());
1825 auto NamesArrVal
= llvm::ConstantArray::get(NamesArrTy
, FunctionNames
);
1826 // This variable will *NOT* be emitted to the object file. It is used
1827 // to pass the list of names referenced to codegen.
1828 new llvm::GlobalVariable(CGM
.getModule(), NamesArrTy
, true,
1829 llvm::GlobalValue::InternalLinkage
, NamesArrVal
,
1830 llvm::getCoverageUnusedNamesVarName());
1834 unsigned CoverageMappingModuleGen::getFileID(FileEntryRef File
) {
1835 auto It
= FileEntries
.find(File
);
1836 if (It
!= FileEntries
.end())
1838 unsigned FileID
= FileEntries
.size() + 1;
1839 FileEntries
.insert(std::make_pair(File
, FileID
));
1843 void CoverageMappingGen::emitCounterMapping(const Decl
*D
,
1844 llvm::raw_ostream
&OS
) {
1846 CounterCoverageMappingBuilder
Walker(CVM
, *CounterMap
, SM
, LangOpts
);
1847 Walker
.VisitDecl(D
);
1851 void CoverageMappingGen::emitEmptyMapping(const Decl
*D
,
1852 llvm::raw_ostream
&OS
) {
1853 EmptyCoverageMappingBuilder
Walker(CVM
, SM
, LangOpts
);
1854 Walker
.VisitDecl(D
);