1*ee02499aSAlex Lorenz //===--- CoverageMappingGen.cpp - Coverage mapping generation ---*- C++ -*-===//
2*ee02499aSAlex Lorenz //
3*ee02499aSAlex Lorenz //                     The LLVM Compiler Infrastructure
4*ee02499aSAlex Lorenz //
5*ee02499aSAlex Lorenz // This file is distributed under the University of Illinois Open Source
6*ee02499aSAlex Lorenz // License. See LICENSE.TXT for details.
7*ee02499aSAlex Lorenz //
8*ee02499aSAlex Lorenz //===----------------------------------------------------------------------===//
9*ee02499aSAlex Lorenz //
10*ee02499aSAlex Lorenz // Instrumentation-based code coverage mapping generator
11*ee02499aSAlex Lorenz //
12*ee02499aSAlex Lorenz //===----------------------------------------------------------------------===//
13*ee02499aSAlex Lorenz 
14*ee02499aSAlex Lorenz #include "CoverageMappingGen.h"
15*ee02499aSAlex Lorenz #include "CodeGenFunction.h"
16*ee02499aSAlex Lorenz #include "clang/AST/StmtVisitor.h"
17*ee02499aSAlex Lorenz #include "clang/Lex/Lexer.h"
18*ee02499aSAlex Lorenz #include "llvm/ProfileData/InstrProfReader.h"
19*ee02499aSAlex Lorenz #include "llvm/ProfileData/CoverageMapping.h"
20*ee02499aSAlex Lorenz #include "llvm/ProfileData/CoverageMappingWriter.h"
21*ee02499aSAlex Lorenz #include "llvm/Support/FileSystem.h"
22*ee02499aSAlex Lorenz 
23*ee02499aSAlex Lorenz using namespace clang;
24*ee02499aSAlex Lorenz using namespace CodeGen;
25*ee02499aSAlex Lorenz using namespace llvm::coverage;
26*ee02499aSAlex Lorenz 
27*ee02499aSAlex Lorenz void CoverageSourceInfo::SourceRangeSkipped(SourceRange Range) {
28*ee02499aSAlex Lorenz   SkippedRanges.push_back(Range);
29*ee02499aSAlex Lorenz }
30*ee02499aSAlex Lorenz 
31*ee02499aSAlex Lorenz namespace {
32*ee02499aSAlex Lorenz 
33*ee02499aSAlex Lorenz /// \brief A region of source code that can be mapped to a counter.
34*ee02499aSAlex Lorenz struct SourceMappingRegion {
35*ee02499aSAlex Lorenz   enum RegionFlags {
36*ee02499aSAlex Lorenz     /// \brief This region won't be emitted if it wasn't extended.
37*ee02499aSAlex Lorenz     /// This is useful so that we won't emit source ranges for single tokens
38*ee02499aSAlex Lorenz     /// that we don't really care that much about, like:
39*ee02499aSAlex Lorenz     ///   the '(' token in #define MACRO (
40*ee02499aSAlex Lorenz     IgnoreIfNotExtended = 0x0001,
41*ee02499aSAlex Lorenz   };
42*ee02499aSAlex Lorenz 
43*ee02499aSAlex Lorenz   FileID File, MacroArgumentFile;
44*ee02499aSAlex Lorenz 
45*ee02499aSAlex Lorenz   Counter Count;
46*ee02499aSAlex Lorenz 
47*ee02499aSAlex Lorenz   /// \brief A statement that initiated the count of Zero.
48*ee02499aSAlex Lorenz   ///
49*ee02499aSAlex Lorenz   /// This initiator statement is useful to prevent merging of unreachable
50*ee02499aSAlex Lorenz   /// regions with different statements that caused the counter to become
51*ee02499aSAlex Lorenz   /// unreachable.
52*ee02499aSAlex Lorenz   const Stmt *UnreachableInitiator;
53*ee02499aSAlex Lorenz 
54*ee02499aSAlex Lorenz   /// \brief A statement that separates certain mapping regions into groups.
55*ee02499aSAlex Lorenz   ///
56*ee02499aSAlex Lorenz   /// The group statement is sometimes useful when we are emitting the source
57*ee02499aSAlex Lorenz   /// regions not in their correct lexical order, e.g. the regions for the
58*ee02499aSAlex Lorenz   /// incrementation expression in the 'for' construct. By marking the regions
59*ee02499aSAlex Lorenz   /// in the incrementation expression with the group statement, we avoid the
60*ee02499aSAlex Lorenz   /// merging of the regions from the incrementation expression and the loop's
61*ee02499aSAlex Lorenz   /// body.
62*ee02499aSAlex Lorenz   const Stmt *Group;
63*ee02499aSAlex Lorenz 
64*ee02499aSAlex Lorenz   /// \brief The region's starting location.
65*ee02499aSAlex Lorenz   SourceLocation LocStart;
66*ee02499aSAlex Lorenz 
67*ee02499aSAlex Lorenz   /// \brief The region's ending location.
68*ee02499aSAlex Lorenz   SourceLocation LocEnd, AlternativeLocEnd;
69*ee02499aSAlex Lorenz   unsigned Flags;
70*ee02499aSAlex Lorenz   CounterMappingRegion::RegionKind Kind;
71*ee02499aSAlex Lorenz 
72*ee02499aSAlex Lorenz   SourceMappingRegion(FileID File, FileID MacroArgumentFile, Counter Count,
73*ee02499aSAlex Lorenz                       const Stmt *UnreachableInitiator, const Stmt *Group,
74*ee02499aSAlex Lorenz                       SourceLocation LocStart, SourceLocation LocEnd,
75*ee02499aSAlex Lorenz                       unsigned Flags = 0,
76*ee02499aSAlex Lorenz                       CounterMappingRegion::RegionKind Kind =
77*ee02499aSAlex Lorenz                           CounterMappingRegion::CodeRegion)
78*ee02499aSAlex Lorenz       : File(File), MacroArgumentFile(MacroArgumentFile), Count(Count),
79*ee02499aSAlex Lorenz         UnreachableInitiator(UnreachableInitiator), Group(Group),
80*ee02499aSAlex Lorenz         LocStart(LocStart), LocEnd(LocEnd), AlternativeLocEnd(LocStart),
81*ee02499aSAlex Lorenz         Flags(Flags), Kind(Kind) {}
82*ee02499aSAlex Lorenz 
83*ee02499aSAlex Lorenz   bool hasFlag(RegionFlags Flag) const { return (Flags & Flag) != 0; }
84*ee02499aSAlex Lorenz 
85*ee02499aSAlex Lorenz   void setFlag(RegionFlags Flag) { Flags |= Flag; }
86*ee02499aSAlex Lorenz 
87*ee02499aSAlex Lorenz   void clearFlag(RegionFlags Flag) { Flags &= ~Flag; }
88*ee02499aSAlex Lorenz 
89*ee02499aSAlex Lorenz   /// \brief Return true if two regions can be merged together.
90*ee02499aSAlex Lorenz   bool isMergeable(SourceMappingRegion &R) {
91*ee02499aSAlex Lorenz     return File == R.File && MacroArgumentFile == R.MacroArgumentFile &&
92*ee02499aSAlex Lorenz            Count == R.Count && UnreachableInitiator == R.UnreachableInitiator &&
93*ee02499aSAlex Lorenz            Group == R.Group && Kind == R.Kind;
94*ee02499aSAlex Lorenz   }
95*ee02499aSAlex Lorenz 
96*ee02499aSAlex Lorenz   /// \brief Merge two regions by extending the 'this' region to cover the
97*ee02499aSAlex Lorenz   /// given region.
98*ee02499aSAlex Lorenz   void mergeByExtendingTo(SourceMappingRegion &R) {
99*ee02499aSAlex Lorenz     LocEnd = R.LocEnd;
100*ee02499aSAlex Lorenz     AlternativeLocEnd = R.LocStart;
101*ee02499aSAlex Lorenz     if (hasFlag(IgnoreIfNotExtended))
102*ee02499aSAlex Lorenz       clearFlag(IgnoreIfNotExtended);
103*ee02499aSAlex Lorenz   }
104*ee02499aSAlex Lorenz };
105*ee02499aSAlex Lorenz 
106*ee02499aSAlex Lorenz /// \brief The state of the coverage mapping builder.
107*ee02499aSAlex Lorenz struct SourceMappingState {
108*ee02499aSAlex Lorenz   Counter CurrentRegionCount;
109*ee02499aSAlex Lorenz   const Stmt *CurrentSourceGroup;
110*ee02499aSAlex Lorenz   const Stmt *CurrentUnreachableRegionInitiator;
111*ee02499aSAlex Lorenz 
112*ee02499aSAlex Lorenz   SourceMappingState(Counter CurrentRegionCount, const Stmt *CurrentSourceGroup,
113*ee02499aSAlex Lorenz                      const Stmt *CurrentUnreachableRegionInitiator)
114*ee02499aSAlex Lorenz       : CurrentRegionCount(CurrentRegionCount),
115*ee02499aSAlex Lorenz         CurrentSourceGroup(CurrentSourceGroup),
116*ee02499aSAlex Lorenz         CurrentUnreachableRegionInitiator(CurrentUnreachableRegionInitiator) {}
117*ee02499aSAlex Lorenz };
118*ee02499aSAlex Lorenz 
119*ee02499aSAlex Lorenz /// \brief Provides the common functionality for the different
120*ee02499aSAlex Lorenz /// coverage mapping region builders.
121*ee02499aSAlex Lorenz class CoverageMappingBuilder {
122*ee02499aSAlex Lorenz public:
123*ee02499aSAlex Lorenz   CoverageMappingModuleGen &CVM;
124*ee02499aSAlex Lorenz   SourceManager &SM;
125*ee02499aSAlex Lorenz   const LangOptions &LangOpts;
126*ee02499aSAlex Lorenz 
127*ee02499aSAlex Lorenz private:
128*ee02499aSAlex Lorenz   struct FileInfo {
129*ee02499aSAlex Lorenz     /// \brief The file id that will be used by the coverage mapping system.
130*ee02499aSAlex Lorenz     unsigned CovMappingFileID;
131*ee02499aSAlex Lorenz     const FileEntry *Entry;
132*ee02499aSAlex Lorenz 
133*ee02499aSAlex Lorenz     FileInfo(unsigned CovMappingFileID, const FileEntry *Entry)
134*ee02499aSAlex Lorenz         : CovMappingFileID(CovMappingFileID), Entry(Entry) {}
135*ee02499aSAlex Lorenz   };
136*ee02499aSAlex Lorenz 
137*ee02499aSAlex Lorenz   /// \brief This mapping maps clang's FileIDs to file ids used
138*ee02499aSAlex Lorenz   /// by the coverage mapping system and clang's file entries.
139*ee02499aSAlex Lorenz   llvm::SmallDenseMap<FileID, FileInfo, 8> FileIDMapping;
140*ee02499aSAlex Lorenz 
141*ee02499aSAlex Lorenz public:
142*ee02499aSAlex Lorenz   /// \brief The statement that corresponds to the current source group.
143*ee02499aSAlex Lorenz   const Stmt *CurrentSourceGroup;
144*ee02499aSAlex Lorenz 
145*ee02499aSAlex Lorenz   /// \brief The statement the initiated the current unreachable region.
146*ee02499aSAlex Lorenz   const Stmt *CurrentUnreachableRegionInitiator;
147*ee02499aSAlex Lorenz 
148*ee02499aSAlex Lorenz   /// \brief The coverage mapping regions for this function
149*ee02499aSAlex Lorenz   llvm::SmallVector<CounterMappingRegion, 32> MappingRegions;
150*ee02499aSAlex Lorenz   /// \brief The source mapping regions for this function.
151*ee02499aSAlex Lorenz   llvm::SmallVector<SourceMappingRegion, 32> SourceRegions;
152*ee02499aSAlex Lorenz 
153*ee02499aSAlex Lorenz   CoverageMappingBuilder(CoverageMappingModuleGen &CVM, SourceManager &SM,
154*ee02499aSAlex Lorenz                          const LangOptions &LangOpts)
155*ee02499aSAlex Lorenz       : CVM(CVM), SM(SM), LangOpts(LangOpts),
156*ee02499aSAlex Lorenz         CurrentSourceGroup(nullptr),
157*ee02499aSAlex Lorenz         CurrentUnreachableRegionInitiator(nullptr) {}
158*ee02499aSAlex Lorenz 
159*ee02499aSAlex Lorenz   /// \brief Return the precise end location for the given token.
160*ee02499aSAlex Lorenz   SourceLocation getPreciseTokenLocEnd(SourceLocation Loc) {
161*ee02499aSAlex Lorenz     return Lexer::getLocForEndOfToken(SM.getSpellingLoc(Loc), 0, SM, LangOpts);
162*ee02499aSAlex Lorenz   }
163*ee02499aSAlex Lorenz 
164*ee02499aSAlex Lorenz   /// \brief Create the mapping that maps from the function's file ids to
165*ee02499aSAlex Lorenz   /// the indices for the translation unit's filenames.
166*ee02499aSAlex Lorenz   void createFileIDMapping(SmallVectorImpl<unsigned> &Mapping) {
167*ee02499aSAlex Lorenz     Mapping.resize(FileIDMapping.size(), 0);
168*ee02499aSAlex Lorenz     for (const auto &I : FileIDMapping)
169*ee02499aSAlex Lorenz       Mapping[I.second.CovMappingFileID] = CVM.getFileID(I.second.Entry);
170*ee02499aSAlex Lorenz   }
171*ee02499aSAlex Lorenz 
172*ee02499aSAlex Lorenz   /// \brief Get the coverage mapping file id that corresponds to the given
173*ee02499aSAlex Lorenz   /// clang file id. If such file id doesn't exist, it gets added to the
174*ee02499aSAlex Lorenz   /// mapping that maps from clang's file ids to coverage mapping file ids.
175*ee02499aSAlex Lorenz   /// Return true if there was an error getting the coverage mapping file id.
176*ee02499aSAlex Lorenz   /// An example of an when this function fails is when the region tries
177*ee02499aSAlex Lorenz   /// to get a coverage file id for a location in a built-in macro.
178*ee02499aSAlex Lorenz   bool getCoverageFileID(SourceLocation LocStart, FileID File,
179*ee02499aSAlex Lorenz                          FileID SpellingFile, unsigned &Result) {
180*ee02499aSAlex Lorenz     auto Mapping = FileIDMapping.find(File);
181*ee02499aSAlex Lorenz     if (Mapping != FileIDMapping.end()) {
182*ee02499aSAlex Lorenz       Result = Mapping->second.CovMappingFileID;
183*ee02499aSAlex Lorenz       return false;
184*ee02499aSAlex Lorenz     }
185*ee02499aSAlex Lorenz 
186*ee02499aSAlex Lorenz     auto Entry = SM.getFileEntryForID(SpellingFile);
187*ee02499aSAlex Lorenz     if (!Entry)
188*ee02499aSAlex Lorenz       return true;
189*ee02499aSAlex Lorenz 
190*ee02499aSAlex Lorenz     Result = FileIDMapping.size();
191*ee02499aSAlex Lorenz     FileIDMapping.insert(std::make_pair(File, FileInfo(Result, Entry)));
192*ee02499aSAlex Lorenz     createFileExpansionRegion(LocStart, File);
193*ee02499aSAlex Lorenz     return false;
194*ee02499aSAlex Lorenz   }
195*ee02499aSAlex Lorenz 
196*ee02499aSAlex Lorenz   /// \brief Get the coverage mapping file id that corresponds to the given
197*ee02499aSAlex Lorenz   /// clang file id.
198*ee02499aSAlex Lorenz   /// Return true if there was an error getting the coverage mapping file id.
199*ee02499aSAlex Lorenz   bool getExistingCoverageFileID(FileID File, unsigned &Result) {
200*ee02499aSAlex Lorenz     // Make sure that the file is valid.
201*ee02499aSAlex Lorenz     if (File.isInvalid())
202*ee02499aSAlex Lorenz       return true;
203*ee02499aSAlex Lorenz     auto Mapping = FileIDMapping.find(File);
204*ee02499aSAlex Lorenz     if (Mapping != FileIDMapping.end()) {
205*ee02499aSAlex Lorenz       Result = Mapping->second.CovMappingFileID;
206*ee02499aSAlex Lorenz       return false;
207*ee02499aSAlex Lorenz     }
208*ee02499aSAlex Lorenz     return true;
209*ee02499aSAlex Lorenz   }
210*ee02499aSAlex Lorenz 
211*ee02499aSAlex Lorenz   /// \brief Return true if the given clang's file id has a corresponding
212*ee02499aSAlex Lorenz   /// coverage file id.
213*ee02499aSAlex Lorenz   bool hasExistingCoverageFileID(FileID File) const {
214*ee02499aSAlex Lorenz     return FileIDMapping.count(File);
215*ee02499aSAlex Lorenz   }
216*ee02499aSAlex Lorenz 
217*ee02499aSAlex Lorenz   /// \brief Gather all the regions that were skipped by the preprocessor
218*ee02499aSAlex Lorenz   /// using the constructs like #if.
219*ee02499aSAlex Lorenz   void gatherSkippedRegions() {
220*ee02499aSAlex Lorenz     /// An array of the minimum lineStarts and the maximum lineEnds
221*ee02499aSAlex Lorenz     /// for mapping regions from the appropriate source files.
222*ee02499aSAlex Lorenz     llvm::SmallVector<std::pair<unsigned, unsigned>, 8> FileLineRanges;
223*ee02499aSAlex Lorenz     FileLineRanges.resize(
224*ee02499aSAlex Lorenz         FileIDMapping.size(),
225*ee02499aSAlex Lorenz         std::make_pair(std::numeric_limits<unsigned>::max(), 0));
226*ee02499aSAlex Lorenz     for (const auto &R : MappingRegions) {
227*ee02499aSAlex Lorenz       FileLineRanges[R.FileID].first =
228*ee02499aSAlex Lorenz           std::min(FileLineRanges[R.FileID].first, R.LineStart);
229*ee02499aSAlex Lorenz       FileLineRanges[R.FileID].second =
230*ee02499aSAlex Lorenz           std::max(FileLineRanges[R.FileID].second, R.LineEnd);
231*ee02499aSAlex Lorenz     }
232*ee02499aSAlex Lorenz 
233*ee02499aSAlex Lorenz     auto SkippedRanges = CVM.getSourceInfo().getSkippedRanges();
234*ee02499aSAlex Lorenz     for (const auto &I : SkippedRanges) {
235*ee02499aSAlex Lorenz       auto LocStart = I.getBegin();
236*ee02499aSAlex Lorenz       auto LocEnd = I.getEnd();
237*ee02499aSAlex Lorenz       auto FileStart = SM.getFileID(LocStart);
238*ee02499aSAlex Lorenz       if (!hasExistingCoverageFileID(FileStart))
239*ee02499aSAlex Lorenz         continue;
240*ee02499aSAlex Lorenz       auto ActualFileStart = SM.getDecomposedSpellingLoc(LocStart).first;
241*ee02499aSAlex Lorenz       if (ActualFileStart != SM.getDecomposedSpellingLoc(LocEnd).first)
242*ee02499aSAlex Lorenz         // Ignore regions that span across multiple files.
243*ee02499aSAlex Lorenz         continue;
244*ee02499aSAlex Lorenz 
245*ee02499aSAlex Lorenz       unsigned CovFileID;
246*ee02499aSAlex Lorenz       if (getCoverageFileID(LocStart, FileStart, ActualFileStart, CovFileID))
247*ee02499aSAlex Lorenz         continue;
248*ee02499aSAlex Lorenz       unsigned LineStart = SM.getSpellingLineNumber(LocStart);
249*ee02499aSAlex Lorenz       unsigned ColumnStart = SM.getSpellingColumnNumber(LocStart);
250*ee02499aSAlex Lorenz       unsigned LineEnd = SM.getSpellingLineNumber(LocEnd);
251*ee02499aSAlex Lorenz       unsigned ColumnEnd = SM.getSpellingColumnNumber(LocEnd);
252*ee02499aSAlex Lorenz       CounterMappingRegion Region(Counter(), CovFileID, LineStart, ColumnStart,
253*ee02499aSAlex Lorenz                                   LineEnd, ColumnEnd, false,
254*ee02499aSAlex Lorenz                                   CounterMappingRegion::SkippedRegion);
255*ee02499aSAlex Lorenz       // Make sure that we only collect the regions that are inside
256*ee02499aSAlex Lorenz       // the souce code of this function.
257*ee02499aSAlex Lorenz       if (Region.LineStart >= FileLineRanges[CovFileID].first &&
258*ee02499aSAlex Lorenz           Region.LineEnd <= FileLineRanges[CovFileID].second)
259*ee02499aSAlex Lorenz         MappingRegions.push_back(Region);
260*ee02499aSAlex Lorenz     }
261*ee02499aSAlex Lorenz   }
262*ee02499aSAlex Lorenz 
263*ee02499aSAlex Lorenz   /// \brief Create a mapping region that correponds to an expansion of
264*ee02499aSAlex Lorenz   /// a macro or an embedded include.
265*ee02499aSAlex Lorenz   void createFileExpansionRegion(SourceLocation Loc, FileID ExpandedFile) {
266*ee02499aSAlex Lorenz     SourceLocation LocStart;
267*ee02499aSAlex Lorenz     if (Loc.isMacroID())
268*ee02499aSAlex Lorenz       LocStart = SM.getImmediateExpansionRange(Loc).first;
269*ee02499aSAlex Lorenz     else {
270*ee02499aSAlex Lorenz       LocStart = SM.getIncludeLoc(ExpandedFile);
271*ee02499aSAlex Lorenz       if (LocStart.isInvalid())
272*ee02499aSAlex Lorenz         return; // This file has no expansion region.
273*ee02499aSAlex Lorenz     }
274*ee02499aSAlex Lorenz 
275*ee02499aSAlex Lorenz     auto File = SM.getFileID(LocStart);
276*ee02499aSAlex Lorenz     auto SpellingFile = SM.getDecomposedSpellingLoc(LocStart).first;
277*ee02499aSAlex Lorenz     unsigned CovFileID, ExpandedFileID;
278*ee02499aSAlex Lorenz     if (getExistingCoverageFileID(ExpandedFile, ExpandedFileID))
279*ee02499aSAlex Lorenz       return;
280*ee02499aSAlex Lorenz     if (getCoverageFileID(LocStart, File, SpellingFile, CovFileID))
281*ee02499aSAlex Lorenz       return;
282*ee02499aSAlex Lorenz     unsigned LineStart = SM.getSpellingLineNumber(LocStart);
283*ee02499aSAlex Lorenz     unsigned ColumnStart = SM.getSpellingColumnNumber(LocStart);
284*ee02499aSAlex Lorenz     unsigned LineEnd = LineStart;
285*ee02499aSAlex Lorenz     // Compute the end column manually as Lexer::getLocForEndOfToken doesn't
286*ee02499aSAlex Lorenz     // give the correct result in all cases.
287*ee02499aSAlex Lorenz     unsigned ColumnEnd =
288*ee02499aSAlex Lorenz         ColumnStart +
289*ee02499aSAlex Lorenz         Lexer::MeasureTokenLength(SM.getSpellingLoc(LocStart), SM, LangOpts);
290*ee02499aSAlex Lorenz 
291*ee02499aSAlex Lorenz     MappingRegions.push_back(CounterMappingRegion(
292*ee02499aSAlex Lorenz         Counter(), CovFileID, LineStart, ColumnStart, LineEnd, ColumnEnd,
293*ee02499aSAlex Lorenz         false, CounterMappingRegion::ExpansionRegion));
294*ee02499aSAlex Lorenz     MappingRegions.back().ExpandedFileID = ExpandedFileID;
295*ee02499aSAlex Lorenz   }
296*ee02499aSAlex Lorenz 
297*ee02499aSAlex Lorenz   /// \brief Enter a source region group that is identified by the given
298*ee02499aSAlex Lorenz   /// statement.
299*ee02499aSAlex Lorenz   /// It's not possible to enter a group when there is already
300*ee02499aSAlex Lorenz   /// another group present.
301*ee02499aSAlex Lorenz   void beginSourceRegionGroup(const Stmt *Group) {
302*ee02499aSAlex Lorenz     assert(!CurrentSourceGroup);
303*ee02499aSAlex Lorenz     CurrentSourceGroup = Group;
304*ee02499aSAlex Lorenz   }
305*ee02499aSAlex Lorenz 
306*ee02499aSAlex Lorenz   /// \brief Exit the current source region group.
307*ee02499aSAlex Lorenz   void endSourceRegionGroup() { CurrentSourceGroup = nullptr; }
308*ee02499aSAlex Lorenz 
309*ee02499aSAlex Lorenz   /// \brief Brings a region that has the same counter and file to the back
310*ee02499aSAlex Lorenz   /// of the source regions array.
311*ee02499aSAlex Lorenz   void bringSimilarRegionBack(Counter Count, FileID File,
312*ee02499aSAlex Lorenz                               FileID MacroArgumentFile,
313*ee02499aSAlex Lorenz                               const Stmt *UnreachableInitiator,
314*ee02499aSAlex Lorenz                               const Stmt *SourceGroup) {
315*ee02499aSAlex Lorenz     for (size_t I = SourceRegions.size(); I != 0;) {
316*ee02499aSAlex Lorenz       --I;
317*ee02499aSAlex Lorenz       if (SourceRegions[I].Count == Count && SourceRegions[I].File == File &&
318*ee02499aSAlex Lorenz           SourceRegions[I].MacroArgumentFile == MacroArgumentFile &&
319*ee02499aSAlex Lorenz           SourceRegions[I].UnreachableInitiator == UnreachableInitiator &&
320*ee02499aSAlex Lorenz           SourceRegions[I].Group == SourceGroup) {
321*ee02499aSAlex Lorenz         if (I != SourceRegions.size() - 1)
322*ee02499aSAlex Lorenz           std::swap(SourceRegions[I], SourceRegions.back());
323*ee02499aSAlex Lorenz         return;
324*ee02499aSAlex Lorenz       }
325*ee02499aSAlex Lorenz     }
326*ee02499aSAlex Lorenz   }
327*ee02499aSAlex Lorenz 
328*ee02499aSAlex Lorenz   /// \brief Associate a counter with a given source code range.
329*ee02499aSAlex Lorenz   void mapSourceCodeRange(SourceLocation LocStart, SourceLocation LocEnd,
330*ee02499aSAlex Lorenz                           Counter Count, const Stmt *UnreachableInitiator,
331*ee02499aSAlex Lorenz                           const Stmt *SourceGroup, unsigned Flags = 0,
332*ee02499aSAlex Lorenz                           FileID MacroArgumentFile = FileID()) {
333*ee02499aSAlex Lorenz     if (SM.isMacroArgExpansion(LocStart)) {
334*ee02499aSAlex Lorenz       // Map the code range with the macro argument's value.
335*ee02499aSAlex Lorenz       mapSourceCodeRange(SM.getImmediateSpellingLoc(LocStart),
336*ee02499aSAlex Lorenz                          SM.getImmediateSpellingLoc(LocEnd), Count,
337*ee02499aSAlex Lorenz                          UnreachableInitiator, SourceGroup, Flags,
338*ee02499aSAlex Lorenz                          SM.getFileID(LocStart));
339*ee02499aSAlex Lorenz       // Map the code range where the macro argument is referenced.
340*ee02499aSAlex Lorenz       SourceLocation RefLocStart(SM.getImmediateExpansionRange(LocStart).first);
341*ee02499aSAlex Lorenz       SourceLocation RefLocEnd(RefLocStart);
342*ee02499aSAlex Lorenz       if (SM.isMacroArgExpansion(RefLocStart))
343*ee02499aSAlex Lorenz         mapSourceCodeRange(RefLocStart, RefLocEnd, Count, UnreachableInitiator,
344*ee02499aSAlex Lorenz                            SourceGroup, 0, SM.getFileID(RefLocStart));
345*ee02499aSAlex Lorenz       else
346*ee02499aSAlex Lorenz         mapSourceCodeRange(RefLocStart, RefLocEnd, Count, UnreachableInitiator,
347*ee02499aSAlex Lorenz                            SourceGroup);
348*ee02499aSAlex Lorenz       return;
349*ee02499aSAlex Lorenz     }
350*ee02499aSAlex Lorenz     auto File = SM.getFileID(LocStart);
351*ee02499aSAlex Lorenz     // Make sure that the file id is valid.
352*ee02499aSAlex Lorenz     if (File.isInvalid())
353*ee02499aSAlex Lorenz       return;
354*ee02499aSAlex Lorenz     bringSimilarRegionBack(Count, File, MacroArgumentFile, UnreachableInitiator,
355*ee02499aSAlex Lorenz                            SourceGroup);
356*ee02499aSAlex Lorenz     SourceMappingRegion R(File, MacroArgumentFile, Count, UnreachableInitiator,
357*ee02499aSAlex Lorenz                           SourceGroup, LocStart, LocEnd, Flags);
358*ee02499aSAlex Lorenz     if (SourceRegions.empty() || !SourceRegions.back().isMergeable(R)) {
359*ee02499aSAlex Lorenz       SourceRegions.push_back(R);
360*ee02499aSAlex Lorenz       return;
361*ee02499aSAlex Lorenz     }
362*ee02499aSAlex Lorenz     SourceRegions.back().mergeByExtendingTo(R);
363*ee02499aSAlex Lorenz   }
364*ee02499aSAlex Lorenz 
365*ee02499aSAlex Lorenz   void mapSourceCodeRange(SourceLocation LocStart, SourceLocation LocEnd,
366*ee02499aSAlex Lorenz                           Counter Count, unsigned Flags = 0) {
367*ee02499aSAlex Lorenz     mapSourceCodeRange(LocStart, LocEnd, Count,
368*ee02499aSAlex Lorenz                        CurrentUnreachableRegionInitiator, CurrentSourceGroup,
369*ee02499aSAlex Lorenz                        Flags);
370*ee02499aSAlex Lorenz   }
371*ee02499aSAlex Lorenz 
372*ee02499aSAlex Lorenz   void mapSourceCodeRange(const SourceMappingState &State,
373*ee02499aSAlex Lorenz                           SourceLocation LocStart, SourceLocation LocEnd,
374*ee02499aSAlex Lorenz                           unsigned Flags = 0) {
375*ee02499aSAlex Lorenz     mapSourceCodeRange(LocStart, LocEnd, State.CurrentRegionCount,
376*ee02499aSAlex Lorenz                        State.CurrentUnreachableRegionInitiator,
377*ee02499aSAlex Lorenz                        State.CurrentSourceGroup, Flags);
378*ee02499aSAlex Lorenz   }
379*ee02499aSAlex Lorenz 
380*ee02499aSAlex Lorenz   /// \brief Generate the coverage counter mapping regions from collected
381*ee02499aSAlex Lorenz   /// source regions.
382*ee02499aSAlex Lorenz   void emitSourceRegions() {
383*ee02499aSAlex Lorenz     for (const auto &R : SourceRegions) {
384*ee02499aSAlex Lorenz       SourceLocation LocStart = R.LocStart;
385*ee02499aSAlex Lorenz       SourceLocation LocEnd = R.LocEnd;
386*ee02499aSAlex Lorenz       if (SM.getFileID(LocEnd) != R.File)
387*ee02499aSAlex Lorenz         LocEnd = R.AlternativeLocEnd;
388*ee02499aSAlex Lorenz 
389*ee02499aSAlex Lorenz       if (R.hasFlag(SourceMappingRegion::IgnoreIfNotExtended) &&
390*ee02499aSAlex Lorenz           LocStart == LocEnd)
391*ee02499aSAlex Lorenz         continue;
392*ee02499aSAlex Lorenz 
393*ee02499aSAlex Lorenz       LocEnd = getPreciseTokenLocEnd(LocEnd);
394*ee02499aSAlex Lorenz       unsigned LineStart = SM.getSpellingLineNumber(LocStart);
395*ee02499aSAlex Lorenz       unsigned ColumnStart = SM.getSpellingColumnNumber(LocStart);
396*ee02499aSAlex Lorenz       unsigned LineEnd = SM.getSpellingLineNumber(LocEnd);
397*ee02499aSAlex Lorenz       unsigned ColumnEnd = SM.getSpellingColumnNumber(LocEnd);
398*ee02499aSAlex Lorenz 
399*ee02499aSAlex Lorenz       auto SpellingFile = SM.getDecomposedSpellingLoc(R.LocStart).first;
400*ee02499aSAlex Lorenz       unsigned CovFileID;
401*ee02499aSAlex Lorenz       if (getCoverageFileID(R.LocStart, R.File, SpellingFile, CovFileID))
402*ee02499aSAlex Lorenz         continue;
403*ee02499aSAlex Lorenz 
404*ee02499aSAlex Lorenz       assert(LineStart <= LineEnd);
405*ee02499aSAlex Lorenz       MappingRegions.push_back(CounterMappingRegion(
406*ee02499aSAlex Lorenz           R.Count, CovFileID, LineStart, ColumnStart, LineEnd, ColumnEnd,
407*ee02499aSAlex Lorenz           false, CounterMappingRegion::CodeRegion));
408*ee02499aSAlex Lorenz     }
409*ee02499aSAlex Lorenz   }
410*ee02499aSAlex Lorenz };
411*ee02499aSAlex Lorenz 
412*ee02499aSAlex Lorenz /// \brief Creates unreachable coverage regions for the functions that
413*ee02499aSAlex Lorenz /// are not emitted.
414*ee02499aSAlex Lorenz struct EmptyCoverageMappingBuilder : public CoverageMappingBuilder {
415*ee02499aSAlex Lorenz   EmptyCoverageMappingBuilder(CoverageMappingModuleGen &CVM, SourceManager &SM,
416*ee02499aSAlex Lorenz                               const LangOptions &LangOpts)
417*ee02499aSAlex Lorenz       : CoverageMappingBuilder(CVM, SM, LangOpts) {}
418*ee02499aSAlex Lorenz 
419*ee02499aSAlex Lorenz   void VisitDecl(const Decl *D) {
420*ee02499aSAlex Lorenz     if (!D->hasBody())
421*ee02499aSAlex Lorenz       return;
422*ee02499aSAlex Lorenz     auto Body = D->getBody();
423*ee02499aSAlex Lorenz     mapSourceCodeRange(Body->getLocStart(), Body->getLocEnd(), Counter());
424*ee02499aSAlex Lorenz   }
425*ee02499aSAlex Lorenz 
426*ee02499aSAlex Lorenz   /// \brief Write the mapping data to the output stream
427*ee02499aSAlex Lorenz   void write(llvm::raw_ostream &OS) {
428*ee02499aSAlex Lorenz     emitSourceRegions();
429*ee02499aSAlex Lorenz     SmallVector<unsigned, 16> FileIDMapping;
430*ee02499aSAlex Lorenz     createFileIDMapping(FileIDMapping);
431*ee02499aSAlex Lorenz 
432*ee02499aSAlex Lorenz     CoverageMappingWriter Writer(
433*ee02499aSAlex Lorenz         FileIDMapping, ArrayRef<CounterExpression>(), MappingRegions);
434*ee02499aSAlex Lorenz     Writer.write(OS);
435*ee02499aSAlex Lorenz   }
436*ee02499aSAlex Lorenz };
437*ee02499aSAlex Lorenz 
438*ee02499aSAlex Lorenz /// \brief A StmtVisitor that creates coverage mapping regions which map
439*ee02499aSAlex Lorenz /// from the source code locations to the PGO counters.
440*ee02499aSAlex Lorenz struct CounterCoverageMappingBuilder
441*ee02499aSAlex Lorenz     : public CoverageMappingBuilder,
442*ee02499aSAlex Lorenz       public ConstStmtVisitor<CounterCoverageMappingBuilder> {
443*ee02499aSAlex Lorenz   /// \brief The map of statements to count values.
444*ee02499aSAlex Lorenz   llvm::DenseMap<const Stmt *, unsigned> &CounterMap;
445*ee02499aSAlex Lorenz 
446*ee02499aSAlex Lorenz   Counter CurrentRegionCount;
447*ee02499aSAlex Lorenz 
448*ee02499aSAlex Lorenz   CounterExpressionBuilder Builder;
449*ee02499aSAlex Lorenz 
450*ee02499aSAlex Lorenz   /// \brief Return a counter that represents the
451*ee02499aSAlex Lorenz   /// expression that subracts rhs from lhs.
452*ee02499aSAlex Lorenz   Counter subtractCounters(Counter LHS, Counter RHS) {
453*ee02499aSAlex Lorenz     return Builder.subtract(LHS, RHS);
454*ee02499aSAlex Lorenz   }
455*ee02499aSAlex Lorenz 
456*ee02499aSAlex Lorenz   /// \brief Return a counter that represents the
457*ee02499aSAlex Lorenz   /// the exression that adds lhs and rhs.
458*ee02499aSAlex Lorenz   Counter addCounters(Counter LHS, Counter RHS) {
459*ee02499aSAlex Lorenz     return Builder.add(LHS, RHS);
460*ee02499aSAlex Lorenz   }
461*ee02499aSAlex Lorenz 
462*ee02499aSAlex Lorenz   /// \brief Return the region counter for the given statement.
463*ee02499aSAlex Lorenz   /// This should only be called on statements that have a dedicated counter.
464*ee02499aSAlex Lorenz   unsigned getRegionCounter(const Stmt *S) { return CounterMap[S]; }
465*ee02499aSAlex Lorenz 
466*ee02499aSAlex Lorenz   /// \brief Return the region count for the counter at the given index.
467*ee02499aSAlex Lorenz   Counter getRegionCount(unsigned CounterId) {
468*ee02499aSAlex Lorenz     return Counter::getCounter(CounterId);
469*ee02499aSAlex Lorenz   }
470*ee02499aSAlex Lorenz 
471*ee02499aSAlex Lorenz   /// \brief Return the counter value of the current region.
472*ee02499aSAlex Lorenz   Counter getCurrentRegionCount() { return CurrentRegionCount; }
473*ee02499aSAlex Lorenz 
474*ee02499aSAlex Lorenz   /// \brief Set the counter value for the current region.
475*ee02499aSAlex Lorenz   /// This is used to keep track of changes to the most recent counter
476*ee02499aSAlex Lorenz   /// from control flow and non-local exits.
477*ee02499aSAlex Lorenz   void setCurrentRegionCount(Counter Count) {
478*ee02499aSAlex Lorenz     CurrentRegionCount = Count;
479*ee02499aSAlex Lorenz     CurrentUnreachableRegionInitiator = nullptr;
480*ee02499aSAlex Lorenz   }
481*ee02499aSAlex Lorenz 
482*ee02499aSAlex Lorenz   /// \brief Indicate that the current region is never reached,
483*ee02499aSAlex Lorenz   /// and thus should have a counter value of zero.
484*ee02499aSAlex Lorenz   /// This is important so that subsequent regions can correctly track
485*ee02499aSAlex Lorenz   /// their parent counts.
486*ee02499aSAlex Lorenz   void setCurrentRegionUnreachable(const Stmt *Initiator) {
487*ee02499aSAlex Lorenz     CurrentRegionCount = Counter::getZero();
488*ee02499aSAlex Lorenz     CurrentUnreachableRegionInitiator = Initiator;
489*ee02499aSAlex Lorenz   }
490*ee02499aSAlex Lorenz 
491*ee02499aSAlex Lorenz   /// \brief A counter for a particular region.
492*ee02499aSAlex Lorenz   /// This is the primary interface through
493*ee02499aSAlex Lorenz   /// which the coverage mapping builder manages counters and their values.
494*ee02499aSAlex Lorenz   class RegionMapper {
495*ee02499aSAlex Lorenz     CounterCoverageMappingBuilder &Mapping;
496*ee02499aSAlex Lorenz     Counter Count;
497*ee02499aSAlex Lorenz     Counter ParentCount;
498*ee02499aSAlex Lorenz     Counter RegionCount;
499*ee02499aSAlex Lorenz     Counter Adjust;
500*ee02499aSAlex Lorenz 
501*ee02499aSAlex Lorenz   public:
502*ee02499aSAlex Lorenz     RegionMapper(CounterCoverageMappingBuilder *Mapper, const Stmt *S)
503*ee02499aSAlex Lorenz         : Mapping(*Mapper),
504*ee02499aSAlex Lorenz           Count(Mapper->getRegionCount(Mapper->getRegionCounter(S))),
505*ee02499aSAlex Lorenz           ParentCount(Mapper->getCurrentRegionCount()) {}
506*ee02499aSAlex Lorenz 
507*ee02499aSAlex Lorenz     /// Get the value of the counter. In most cases this is the number of times
508*ee02499aSAlex Lorenz     /// the region of the counter was entered, but for switch labels it's the
509*ee02499aSAlex Lorenz     /// number of direct jumps to that label.
510*ee02499aSAlex Lorenz     Counter getCount() const { return Count; }
511*ee02499aSAlex Lorenz 
512*ee02499aSAlex Lorenz     /// Get the value of the counter with adjustments applied. Adjustments occur
513*ee02499aSAlex Lorenz     /// when control enters or leaves the region abnormally; i.e., if there is a
514*ee02499aSAlex Lorenz     /// jump to a label within the region, or if the function can return from
515*ee02499aSAlex Lorenz     /// within the region. The adjusted count, then, is the value of the counter
516*ee02499aSAlex Lorenz     /// at the end of the region.
517*ee02499aSAlex Lorenz     Counter getAdjustedCount() const {
518*ee02499aSAlex Lorenz       return Mapping.addCounters(Count, Adjust);
519*ee02499aSAlex Lorenz     }
520*ee02499aSAlex Lorenz 
521*ee02499aSAlex Lorenz     /// Get the value of the counter in this region's parent, i.e., the region
522*ee02499aSAlex Lorenz     /// that was active when this region began. This is useful for deriving
523*ee02499aSAlex Lorenz     /// counts in implicitly counted regions, like the false case of a condition
524*ee02499aSAlex Lorenz     /// or the normal exits of a loop.
525*ee02499aSAlex Lorenz     Counter getParentCount() const { return ParentCount; }
526*ee02499aSAlex Lorenz 
527*ee02499aSAlex Lorenz     /// Activate the counter by emitting an increment and starting to track
528*ee02499aSAlex Lorenz     /// adjustments. If AddIncomingFallThrough is true, the current region count
529*ee02499aSAlex Lorenz     /// will be added to the counter for the purposes of tracking the region.
530*ee02499aSAlex Lorenz     void beginRegion(bool AddIncomingFallThrough = false) {
531*ee02499aSAlex Lorenz       RegionCount = Count;
532*ee02499aSAlex Lorenz       if (AddIncomingFallThrough)
533*ee02499aSAlex Lorenz         RegionCount =
534*ee02499aSAlex Lorenz             Mapping.addCounters(RegionCount, Mapping.getCurrentRegionCount());
535*ee02499aSAlex Lorenz       Mapping.setCurrentRegionCount(RegionCount);
536*ee02499aSAlex Lorenz     }
537*ee02499aSAlex Lorenz 
538*ee02499aSAlex Lorenz     /// For counters on boolean branches, begins tracking adjustments for the
539*ee02499aSAlex Lorenz     /// uncounted path.
540*ee02499aSAlex Lorenz     void beginElseRegion() {
541*ee02499aSAlex Lorenz       RegionCount = Mapping.subtractCounters(ParentCount, Count);
542*ee02499aSAlex Lorenz       Mapping.setCurrentRegionCount(RegionCount);
543*ee02499aSAlex Lorenz     }
544*ee02499aSAlex Lorenz 
545*ee02499aSAlex Lorenz     /// Reset the current region count.
546*ee02499aSAlex Lorenz     void setCurrentRegionCount(Counter CurrentCount) {
547*ee02499aSAlex Lorenz       RegionCount = CurrentCount;
548*ee02499aSAlex Lorenz       Mapping.setCurrentRegionCount(RegionCount);
549*ee02499aSAlex Lorenz     }
550*ee02499aSAlex Lorenz 
551*ee02499aSAlex Lorenz     /// Adjust for non-local control flow after emitting a subexpression or
552*ee02499aSAlex Lorenz     /// substatement. This must be called to account for constructs such as
553*ee02499aSAlex Lorenz     /// gotos,
554*ee02499aSAlex Lorenz     /// labels, and returns, so that we can ensure that our region's count is
555*ee02499aSAlex Lorenz     /// correct in the code that follows.
556*ee02499aSAlex Lorenz     void adjustForControlFlow() {
557*ee02499aSAlex Lorenz       Adjust = Mapping.addCounters(
558*ee02499aSAlex Lorenz           Adjust, Mapping.subtractCounters(Mapping.getCurrentRegionCount(),
559*ee02499aSAlex Lorenz                                            RegionCount));
560*ee02499aSAlex Lorenz       // Reset the region count in case this is called again later.
561*ee02499aSAlex Lorenz       RegionCount = Mapping.getCurrentRegionCount();
562*ee02499aSAlex Lorenz     }
563*ee02499aSAlex Lorenz 
564*ee02499aSAlex Lorenz     /// Commit all adjustments to the current region. If the region is a loop,
565*ee02499aSAlex Lorenz     /// the LoopAdjust value should be the count of all the breaks and continues
566*ee02499aSAlex Lorenz     /// from the loop, to compensate for those counts being deducted from the
567*ee02499aSAlex Lorenz     /// adjustments for the body of the loop.
568*ee02499aSAlex Lorenz     void applyAdjustmentsToRegion() {
569*ee02499aSAlex Lorenz       Mapping.setCurrentRegionCount(Mapping.addCounters(ParentCount, Adjust));
570*ee02499aSAlex Lorenz     }
571*ee02499aSAlex Lorenz     void applyAdjustmentsToRegion(Counter LoopAdjust) {
572*ee02499aSAlex Lorenz       Mapping.setCurrentRegionCount(Mapping.addCounters(
573*ee02499aSAlex Lorenz           Mapping.addCounters(ParentCount, Adjust), LoopAdjust));
574*ee02499aSAlex Lorenz     }
575*ee02499aSAlex Lorenz   };
576*ee02499aSAlex Lorenz 
577*ee02499aSAlex Lorenz   /// \brief Keep counts of breaks and continues inside loops.
578*ee02499aSAlex Lorenz   struct BreakContinue {
579*ee02499aSAlex Lorenz     Counter BreakCount;
580*ee02499aSAlex Lorenz     Counter ContinueCount;
581*ee02499aSAlex Lorenz   };
582*ee02499aSAlex Lorenz   SmallVector<BreakContinue, 8> BreakContinueStack;
583*ee02499aSAlex Lorenz 
584*ee02499aSAlex Lorenz   CounterCoverageMappingBuilder(
585*ee02499aSAlex Lorenz       CoverageMappingModuleGen &CVM,
586*ee02499aSAlex Lorenz       llvm::DenseMap<const Stmt *, unsigned> &CounterMap,
587*ee02499aSAlex Lorenz       unsigned NumRegionCounters, SourceManager &SM,
588*ee02499aSAlex Lorenz       const LangOptions &LangOpts)
589*ee02499aSAlex Lorenz       : CoverageMappingBuilder(CVM, SM, LangOpts), CounterMap(CounterMap),
590*ee02499aSAlex Lorenz         Builder(NumRegionCounters) {}
591*ee02499aSAlex Lorenz 
592*ee02499aSAlex Lorenz   /// \brief Write the mapping data to the output stream
593*ee02499aSAlex Lorenz   void write(llvm::raw_ostream &OS) {
594*ee02499aSAlex Lorenz     emitSourceRegions();
595*ee02499aSAlex Lorenz     llvm::SmallVector<unsigned, 8> VirtualFileMapping;
596*ee02499aSAlex Lorenz     createFileIDMapping(VirtualFileMapping);
597*ee02499aSAlex Lorenz     gatherSkippedRegions();
598*ee02499aSAlex Lorenz 
599*ee02499aSAlex Lorenz     CoverageMappingWriter Writer(
600*ee02499aSAlex Lorenz         VirtualFileMapping, Builder.getExpressions(), MappingRegions);
601*ee02499aSAlex Lorenz     Writer.write(OS);
602*ee02499aSAlex Lorenz   }
603*ee02499aSAlex Lorenz 
604*ee02499aSAlex Lorenz   /// \brief Return the current source mapping state.
605*ee02499aSAlex Lorenz   SourceMappingState getCurrentState() const {
606*ee02499aSAlex Lorenz     return SourceMappingState(CurrentRegionCount, CurrentSourceGroup,
607*ee02499aSAlex Lorenz                               CurrentUnreachableRegionInitiator);
608*ee02499aSAlex Lorenz   }
609*ee02499aSAlex Lorenz 
610*ee02499aSAlex Lorenz   /// \brief Associate the source code range with the current region count.
611*ee02499aSAlex Lorenz   void mapSourceCodeRange(SourceLocation LocStart, SourceLocation LocEnd,
612*ee02499aSAlex Lorenz                           unsigned Flags = 0) {
613*ee02499aSAlex Lorenz     CoverageMappingBuilder::mapSourceCodeRange(LocStart, LocEnd,
614*ee02499aSAlex Lorenz                                                CurrentRegionCount, Flags);
615*ee02499aSAlex Lorenz   }
616*ee02499aSAlex Lorenz 
617*ee02499aSAlex Lorenz   void mapSourceCodeRange(SourceLocation LocStart) {
618*ee02499aSAlex Lorenz     CoverageMappingBuilder::mapSourceCodeRange(LocStart, LocStart,
619*ee02499aSAlex Lorenz                                                CurrentRegionCount);
620*ee02499aSAlex Lorenz   }
621*ee02499aSAlex Lorenz 
622*ee02499aSAlex Lorenz   /// \brief Associate the source range of a token with the current region
623*ee02499aSAlex Lorenz   /// count.
624*ee02499aSAlex Lorenz   /// Ignore the source range for this token if it produces a distinct
625*ee02499aSAlex Lorenz   /// mapping region with no other source ranges.
626*ee02499aSAlex Lorenz   void mapToken(SourceLocation LocStart) {
627*ee02499aSAlex Lorenz     CoverageMappingBuilder::mapSourceCodeRange(
628*ee02499aSAlex Lorenz         LocStart, LocStart, CurrentRegionCount,
629*ee02499aSAlex Lorenz         SourceMappingRegion::IgnoreIfNotExtended);
630*ee02499aSAlex Lorenz   }
631*ee02499aSAlex Lorenz 
632*ee02499aSAlex Lorenz   void mapToken(const SourceMappingState &State, SourceLocation LocStart) {
633*ee02499aSAlex Lorenz     CoverageMappingBuilder::mapSourceCodeRange(
634*ee02499aSAlex Lorenz         State, LocStart, LocStart, SourceMappingRegion::IgnoreIfNotExtended);
635*ee02499aSAlex Lorenz   }
636*ee02499aSAlex Lorenz 
637*ee02499aSAlex Lorenz   void VisitStmt(const Stmt *S) {
638*ee02499aSAlex Lorenz     mapSourceCodeRange(S->getLocStart());
639*ee02499aSAlex Lorenz     for (Stmt::const_child_range I = S->children(); I; ++I) {
640*ee02499aSAlex Lorenz       if (*I)
641*ee02499aSAlex Lorenz         this->Visit(*I);
642*ee02499aSAlex Lorenz     }
643*ee02499aSAlex Lorenz   }
644*ee02499aSAlex Lorenz 
645*ee02499aSAlex Lorenz   /// \brief If the given statement is a compound statement,
646*ee02499aSAlex Lorenz   /// map '}' with the same count as '{'.
647*ee02499aSAlex Lorenz   void VisitSubStmtRBraceState(const Stmt *S) {
648*ee02499aSAlex Lorenz     if (!isa<CompoundStmt>(S))
649*ee02499aSAlex Lorenz       return Visit(S);
650*ee02499aSAlex Lorenz     const auto *CS = cast<CompoundStmt>(S);
651*ee02499aSAlex Lorenz     auto State = getCurrentState();
652*ee02499aSAlex Lorenz     mapSourceCodeRange(CS->getLBracLoc());
653*ee02499aSAlex Lorenz     for (Stmt::const_child_range I = S->children(); I; ++I) {
654*ee02499aSAlex Lorenz       if (*I)
655*ee02499aSAlex Lorenz         this->Visit(*I);
656*ee02499aSAlex Lorenz     }
657*ee02499aSAlex Lorenz     CoverageMappingBuilder::mapSourceCodeRange(State, CS->getRBracLoc(),
658*ee02499aSAlex Lorenz                                                CS->getRBracLoc());
659*ee02499aSAlex Lorenz   }
660*ee02499aSAlex Lorenz 
661*ee02499aSAlex Lorenz   void VisitDecl(const Decl *D) {
662*ee02499aSAlex Lorenz     if (!D->hasBody())
663*ee02499aSAlex Lorenz       return;
664*ee02499aSAlex Lorenz     // Counter tracks entry to the function body.
665*ee02499aSAlex Lorenz     auto Body = D->getBody();
666*ee02499aSAlex Lorenz     RegionMapper Cnt(this, Body);
667*ee02499aSAlex Lorenz     Cnt.beginRegion();
668*ee02499aSAlex Lorenz     VisitSubStmtRBraceState(Body);
669*ee02499aSAlex Lorenz   }
670*ee02499aSAlex Lorenz 
671*ee02499aSAlex Lorenz   void VisitDeclStmt(const DeclStmt *S) {
672*ee02499aSAlex Lorenz     mapSourceCodeRange(S->getLocStart());
673*ee02499aSAlex Lorenz     for (Stmt::const_child_range I = static_cast<const Stmt *>(S)->children();
674*ee02499aSAlex Lorenz          I; ++I) {
675*ee02499aSAlex Lorenz       if (*I)
676*ee02499aSAlex Lorenz         this->Visit(*I);
677*ee02499aSAlex Lorenz     }
678*ee02499aSAlex Lorenz   }
679*ee02499aSAlex Lorenz 
680*ee02499aSAlex Lorenz   void VisitCompoundStmt(const CompoundStmt *S) {
681*ee02499aSAlex Lorenz     mapSourceCodeRange(S->getLBracLoc());
682*ee02499aSAlex Lorenz     for (Stmt::const_child_range I = S->children(); I; ++I) {
683*ee02499aSAlex Lorenz       if (*I)
684*ee02499aSAlex Lorenz         this->Visit(*I);
685*ee02499aSAlex Lorenz     }
686*ee02499aSAlex Lorenz     mapSourceCodeRange(S->getRBracLoc(), S->getRBracLoc());
687*ee02499aSAlex Lorenz   }
688*ee02499aSAlex Lorenz 
689*ee02499aSAlex Lorenz   void VisitReturnStmt(const ReturnStmt *S) {
690*ee02499aSAlex Lorenz     mapSourceCodeRange(S->getLocStart());
691*ee02499aSAlex Lorenz     if (S->getRetValue())
692*ee02499aSAlex Lorenz       Visit(S->getRetValue());
693*ee02499aSAlex Lorenz     setCurrentRegionUnreachable(S);
694*ee02499aSAlex Lorenz   }
695*ee02499aSAlex Lorenz 
696*ee02499aSAlex Lorenz   void VisitGotoStmt(const GotoStmt *S) {
697*ee02499aSAlex Lorenz     mapSourceCodeRange(S->getLocStart());
698*ee02499aSAlex Lorenz     mapToken(S->getLabelLoc());
699*ee02499aSAlex Lorenz     setCurrentRegionUnreachable(S);
700*ee02499aSAlex Lorenz   }
701*ee02499aSAlex Lorenz 
702*ee02499aSAlex Lorenz   void VisitLabelStmt(const LabelStmt *S) {
703*ee02499aSAlex Lorenz     // Counter tracks the block following the label.
704*ee02499aSAlex Lorenz     RegionMapper Cnt(this, S);
705*ee02499aSAlex Lorenz     Cnt.beginRegion();
706*ee02499aSAlex Lorenz     mapSourceCodeRange(S->getLocStart());
707*ee02499aSAlex Lorenz     // Can't map the ':' token as its location isn't known.
708*ee02499aSAlex Lorenz     Visit(S->getSubStmt());
709*ee02499aSAlex Lorenz   }
710*ee02499aSAlex Lorenz 
711*ee02499aSAlex Lorenz   void VisitBreakStmt(const BreakStmt *S) {
712*ee02499aSAlex Lorenz     mapSourceCodeRange(S->getLocStart());
713*ee02499aSAlex Lorenz     assert(!BreakContinueStack.empty() && "break not in a loop or switch!");
714*ee02499aSAlex Lorenz     BreakContinueStack.back().BreakCount = addCounters(
715*ee02499aSAlex Lorenz         BreakContinueStack.back().BreakCount, getCurrentRegionCount());
716*ee02499aSAlex Lorenz     setCurrentRegionUnreachable(S);
717*ee02499aSAlex Lorenz   }
718*ee02499aSAlex Lorenz 
719*ee02499aSAlex Lorenz   void VisitContinueStmt(const ContinueStmt *S) {
720*ee02499aSAlex Lorenz     mapSourceCodeRange(S->getLocStart());
721*ee02499aSAlex Lorenz     assert(!BreakContinueStack.empty() && "continue stmt not in a loop!");
722*ee02499aSAlex Lorenz     BreakContinueStack.back().ContinueCount = addCounters(
723*ee02499aSAlex Lorenz         BreakContinueStack.back().ContinueCount, getCurrentRegionCount());
724*ee02499aSAlex Lorenz     setCurrentRegionUnreachable(S);
725*ee02499aSAlex Lorenz   }
726*ee02499aSAlex Lorenz 
727*ee02499aSAlex Lorenz   void VisitWhileStmt(const WhileStmt *S) {
728*ee02499aSAlex Lorenz     mapSourceCodeRange(S->getLocStart());
729*ee02499aSAlex Lorenz     // Counter tracks the body of the loop.
730*ee02499aSAlex Lorenz     RegionMapper Cnt(this, S);
731*ee02499aSAlex Lorenz     BreakContinueStack.push_back(BreakContinue());
732*ee02499aSAlex Lorenz     // Visit the body region first so the break/continue adjustments can be
733*ee02499aSAlex Lorenz     // included when visiting the condition.
734*ee02499aSAlex Lorenz     Cnt.beginRegion();
735*ee02499aSAlex Lorenz     VisitSubStmtRBraceState(S->getBody());
736*ee02499aSAlex Lorenz     Cnt.adjustForControlFlow();
737*ee02499aSAlex Lorenz 
738*ee02499aSAlex Lorenz     // ...then go back and propagate counts through the condition. The count
739*ee02499aSAlex Lorenz     // at the start of the condition is the sum of the incoming edges,
740*ee02499aSAlex Lorenz     // the backedge from the end of the loop body, and the edges from
741*ee02499aSAlex Lorenz     // continue statements.
742*ee02499aSAlex Lorenz     BreakContinue BC = BreakContinueStack.pop_back_val();
743*ee02499aSAlex Lorenz     Cnt.setCurrentRegionCount(
744*ee02499aSAlex Lorenz         addCounters(Cnt.getParentCount(),
745*ee02499aSAlex Lorenz                     addCounters(Cnt.getAdjustedCount(), BC.ContinueCount)));
746*ee02499aSAlex Lorenz     beginSourceRegionGroup(S->getCond());
747*ee02499aSAlex Lorenz     Visit(S->getCond());
748*ee02499aSAlex Lorenz     endSourceRegionGroup();
749*ee02499aSAlex Lorenz     Cnt.adjustForControlFlow();
750*ee02499aSAlex Lorenz     Cnt.applyAdjustmentsToRegion(addCounters(BC.BreakCount, BC.ContinueCount));
751*ee02499aSAlex Lorenz   }
752*ee02499aSAlex Lorenz 
753*ee02499aSAlex Lorenz   void VisitDoStmt(const DoStmt *S) {
754*ee02499aSAlex Lorenz     mapSourceCodeRange(S->getLocStart());
755*ee02499aSAlex Lorenz     // Counter tracks the body of the loop.
756*ee02499aSAlex Lorenz     RegionMapper Cnt(this, S);
757*ee02499aSAlex Lorenz     BreakContinueStack.push_back(BreakContinue());
758*ee02499aSAlex Lorenz     Cnt.beginRegion(/*AddIncomingFallThrough=*/true);
759*ee02499aSAlex Lorenz     VisitSubStmtRBraceState(S->getBody());
760*ee02499aSAlex Lorenz     Cnt.adjustForControlFlow();
761*ee02499aSAlex Lorenz 
762*ee02499aSAlex Lorenz     BreakContinue BC = BreakContinueStack.pop_back_val();
763*ee02499aSAlex Lorenz     // The count at the start of the condition is equal to the count at the
764*ee02499aSAlex Lorenz     // end of the body. The adjusted count does not include either the
765*ee02499aSAlex Lorenz     // fall-through count coming into the loop or the continue count, so add
766*ee02499aSAlex Lorenz     // both of those separately. This is coincidentally the same equation as
767*ee02499aSAlex Lorenz     // with while loops but for different reasons.
768*ee02499aSAlex Lorenz     Cnt.setCurrentRegionCount(
769*ee02499aSAlex Lorenz         addCounters(Cnt.getParentCount(),
770*ee02499aSAlex Lorenz                     addCounters(Cnt.getAdjustedCount(), BC.ContinueCount)));
771*ee02499aSAlex Lorenz     Visit(S->getCond());
772*ee02499aSAlex Lorenz     Cnt.adjustForControlFlow();
773*ee02499aSAlex Lorenz     Cnt.applyAdjustmentsToRegion(addCounters(BC.BreakCount, BC.ContinueCount));
774*ee02499aSAlex Lorenz   }
775*ee02499aSAlex Lorenz 
776*ee02499aSAlex Lorenz   void VisitForStmt(const ForStmt *S) {
777*ee02499aSAlex Lorenz     mapSourceCodeRange(S->getLocStart());
778*ee02499aSAlex Lorenz     if (S->getInit())
779*ee02499aSAlex Lorenz       Visit(S->getInit());
780*ee02499aSAlex Lorenz 
781*ee02499aSAlex Lorenz     // Counter tracks the body of the loop.
782*ee02499aSAlex Lorenz     RegionMapper Cnt(this, S);
783*ee02499aSAlex Lorenz     BreakContinueStack.push_back(BreakContinue());
784*ee02499aSAlex Lorenz     // Visit the body region first. (This is basically the same as a while
785*ee02499aSAlex Lorenz     // loop; see further comments in VisitWhileStmt.)
786*ee02499aSAlex Lorenz     Cnt.beginRegion();
787*ee02499aSAlex Lorenz     VisitSubStmtRBraceState(S->getBody());
788*ee02499aSAlex Lorenz     Cnt.adjustForControlFlow();
789*ee02499aSAlex Lorenz 
790*ee02499aSAlex Lorenz     // The increment is essentially part of the body but it needs to include
791*ee02499aSAlex Lorenz     // the count for all the continue statements.
792*ee02499aSAlex Lorenz     if (S->getInc()) {
793*ee02499aSAlex Lorenz       Cnt.setCurrentRegionCount(addCounters(
794*ee02499aSAlex Lorenz           getCurrentRegionCount(), BreakContinueStack.back().ContinueCount));
795*ee02499aSAlex Lorenz       beginSourceRegionGroup(S->getInc());
796*ee02499aSAlex Lorenz       Visit(S->getInc());
797*ee02499aSAlex Lorenz       endSourceRegionGroup();
798*ee02499aSAlex Lorenz       Cnt.adjustForControlFlow();
799*ee02499aSAlex Lorenz     }
800*ee02499aSAlex Lorenz 
801*ee02499aSAlex Lorenz     BreakContinue BC = BreakContinueStack.pop_back_val();
802*ee02499aSAlex Lorenz 
803*ee02499aSAlex Lorenz     // ...then go back and propagate counts through the condition.
804*ee02499aSAlex Lorenz     if (S->getCond()) {
805*ee02499aSAlex Lorenz       Cnt.setCurrentRegionCount(
806*ee02499aSAlex Lorenz           addCounters(addCounters(Cnt.getParentCount(), Cnt.getAdjustedCount()),
807*ee02499aSAlex Lorenz                       BC.ContinueCount));
808*ee02499aSAlex Lorenz       beginSourceRegionGroup(S->getCond());
809*ee02499aSAlex Lorenz       Visit(S->getCond());
810*ee02499aSAlex Lorenz       endSourceRegionGroup();
811*ee02499aSAlex Lorenz       Cnt.adjustForControlFlow();
812*ee02499aSAlex Lorenz     }
813*ee02499aSAlex Lorenz     Cnt.applyAdjustmentsToRegion(addCounters(BC.BreakCount, BC.ContinueCount));
814*ee02499aSAlex Lorenz   }
815*ee02499aSAlex Lorenz 
816*ee02499aSAlex Lorenz   void VisitCXXForRangeStmt(const CXXForRangeStmt *S) {
817*ee02499aSAlex Lorenz     mapSourceCodeRange(S->getLocStart());
818*ee02499aSAlex Lorenz     Visit(S->getRangeStmt());
819*ee02499aSAlex Lorenz     Visit(S->getBeginEndStmt());
820*ee02499aSAlex Lorenz     // Counter tracks the body of the loop.
821*ee02499aSAlex Lorenz     RegionMapper Cnt(this, S);
822*ee02499aSAlex Lorenz     BreakContinueStack.push_back(BreakContinue());
823*ee02499aSAlex Lorenz     // Visit the body region first. (This is basically the same as a while
824*ee02499aSAlex Lorenz     // loop; see further comments in VisitWhileStmt.)
825*ee02499aSAlex Lorenz     Cnt.beginRegion();
826*ee02499aSAlex Lorenz     VisitSubStmtRBraceState(S->getBody());
827*ee02499aSAlex Lorenz     Cnt.adjustForControlFlow();
828*ee02499aSAlex Lorenz     BreakContinue BC = BreakContinueStack.pop_back_val();
829*ee02499aSAlex Lorenz     Cnt.applyAdjustmentsToRegion(addCounters(BC.BreakCount, BC.ContinueCount));
830*ee02499aSAlex Lorenz   }
831*ee02499aSAlex Lorenz 
832*ee02499aSAlex Lorenz   void VisitObjCForCollectionStmt(const ObjCForCollectionStmt *S) {
833*ee02499aSAlex Lorenz     mapSourceCodeRange(S->getLocStart());
834*ee02499aSAlex Lorenz     Visit(S->getElement());
835*ee02499aSAlex Lorenz     // Counter tracks the body of the loop.
836*ee02499aSAlex Lorenz     RegionMapper Cnt(this, S);
837*ee02499aSAlex Lorenz     BreakContinueStack.push_back(BreakContinue());
838*ee02499aSAlex Lorenz     VisitSubStmtRBraceState(S->getBody());
839*ee02499aSAlex Lorenz     BreakContinue BC = BreakContinueStack.pop_back_val();
840*ee02499aSAlex Lorenz     Cnt.adjustForControlFlow();
841*ee02499aSAlex Lorenz     Cnt.applyAdjustmentsToRegion(addCounters(BC.BreakCount, BC.ContinueCount));
842*ee02499aSAlex Lorenz   }
843*ee02499aSAlex Lorenz 
844*ee02499aSAlex Lorenz   void VisitSwitchStmt(const SwitchStmt *S) {
845*ee02499aSAlex Lorenz     mapSourceCodeRange(S->getLocStart());
846*ee02499aSAlex Lorenz     Visit(S->getCond());
847*ee02499aSAlex Lorenz     BreakContinueStack.push_back(BreakContinue());
848*ee02499aSAlex Lorenz     // Map the '}' for the body to have the same count as the regions after
849*ee02499aSAlex Lorenz     // the switch.
850*ee02499aSAlex Lorenz     SourceLocation RBracLoc;
851*ee02499aSAlex Lorenz     if (const auto *CS = dyn_cast<CompoundStmt>(S->getBody())) {
852*ee02499aSAlex Lorenz       mapSourceCodeRange(CS->getLBracLoc());
853*ee02499aSAlex Lorenz       setCurrentRegionUnreachable(S);
854*ee02499aSAlex Lorenz       for (Stmt::const_child_range I = CS->children(); I; ++I) {
855*ee02499aSAlex Lorenz         if (*I)
856*ee02499aSAlex Lorenz           this->Visit(*I);
857*ee02499aSAlex Lorenz       }
858*ee02499aSAlex Lorenz       RBracLoc = CS->getRBracLoc();
859*ee02499aSAlex Lorenz     } else {
860*ee02499aSAlex Lorenz       setCurrentRegionUnreachable(S);
861*ee02499aSAlex Lorenz       Visit(S->getBody());
862*ee02499aSAlex Lorenz     }
863*ee02499aSAlex Lorenz     // If the switch is inside a loop, add the continue counts.
864*ee02499aSAlex Lorenz     BreakContinue BC = BreakContinueStack.pop_back_val();
865*ee02499aSAlex Lorenz     if (!BreakContinueStack.empty())
866*ee02499aSAlex Lorenz       BreakContinueStack.back().ContinueCount = addCounters(
867*ee02499aSAlex Lorenz           BreakContinueStack.back().ContinueCount, BC.ContinueCount);
868*ee02499aSAlex Lorenz     // Counter tracks the exit block of the switch.
869*ee02499aSAlex Lorenz     RegionMapper ExitCnt(this, S);
870*ee02499aSAlex Lorenz     ExitCnt.beginRegion();
871*ee02499aSAlex Lorenz     if (RBracLoc.isValid())
872*ee02499aSAlex Lorenz       mapSourceCodeRange(RBracLoc);
873*ee02499aSAlex Lorenz   }
874*ee02499aSAlex Lorenz 
875*ee02499aSAlex Lorenz   void VisitCaseStmt(const CaseStmt *S) {
876*ee02499aSAlex Lorenz     // Counter for this particular case. This counts only jumps from the
877*ee02499aSAlex Lorenz     // switch header and does not include fallthrough from the case before
878*ee02499aSAlex Lorenz     // this one.
879*ee02499aSAlex Lorenz     RegionMapper Cnt(this, S);
880*ee02499aSAlex Lorenz     Cnt.beginRegion(/*AddIncomingFallThrough=*/true);
881*ee02499aSAlex Lorenz     mapSourceCodeRange(S->getLocStart());
882*ee02499aSAlex Lorenz     mapToken(S->getColonLoc());
883*ee02499aSAlex Lorenz     Visit(S->getSubStmt());
884*ee02499aSAlex Lorenz   }
885*ee02499aSAlex Lorenz 
886*ee02499aSAlex Lorenz   void VisitDefaultStmt(const DefaultStmt *S) {
887*ee02499aSAlex Lorenz     // Counter for this default case. This does not include fallthrough from
888*ee02499aSAlex Lorenz     // the previous case.
889*ee02499aSAlex Lorenz     RegionMapper Cnt(this, S);
890*ee02499aSAlex Lorenz     Cnt.beginRegion(/*AddIncomingFallThrough=*/true);
891*ee02499aSAlex Lorenz     mapSourceCodeRange(S->getLocStart());
892*ee02499aSAlex Lorenz     mapToken(S->getColonLoc());
893*ee02499aSAlex Lorenz     Visit(S->getSubStmt());
894*ee02499aSAlex Lorenz   }
895*ee02499aSAlex Lorenz 
896*ee02499aSAlex Lorenz   void VisitIfStmt(const IfStmt *S) {
897*ee02499aSAlex Lorenz     mapSourceCodeRange(S->getLocStart());
898*ee02499aSAlex Lorenz     Visit(S->getCond());
899*ee02499aSAlex Lorenz     mapToken(S->getElseLoc());
900*ee02499aSAlex Lorenz 
901*ee02499aSAlex Lorenz     // Counter tracks the "then" part of an if statement. The count for
902*ee02499aSAlex Lorenz     // the "else" part, if it exists, will be calculated from this counter.
903*ee02499aSAlex Lorenz     RegionMapper Cnt(this, S);
904*ee02499aSAlex Lorenz     Cnt.beginRegion();
905*ee02499aSAlex Lorenz     VisitSubStmtRBraceState(S->getThen());
906*ee02499aSAlex Lorenz     Cnt.adjustForControlFlow();
907*ee02499aSAlex Lorenz 
908*ee02499aSAlex Lorenz     if (S->getElse()) {
909*ee02499aSAlex Lorenz       Cnt.beginElseRegion();
910*ee02499aSAlex Lorenz       VisitSubStmtRBraceState(S->getElse());
911*ee02499aSAlex Lorenz       Cnt.adjustForControlFlow();
912*ee02499aSAlex Lorenz     }
913*ee02499aSAlex Lorenz     Cnt.applyAdjustmentsToRegion();
914*ee02499aSAlex Lorenz   }
915*ee02499aSAlex Lorenz 
916*ee02499aSAlex Lorenz   void VisitCXXTryStmt(const CXXTryStmt *S) {
917*ee02499aSAlex Lorenz     mapSourceCodeRange(S->getLocStart());
918*ee02499aSAlex Lorenz     Visit(S->getTryBlock());
919*ee02499aSAlex Lorenz     for (unsigned I = 0, E = S->getNumHandlers(); I < E; ++I)
920*ee02499aSAlex Lorenz       Visit(S->getHandler(I));
921*ee02499aSAlex Lorenz     // Counter tracks the continuation block of the try statement.
922*ee02499aSAlex Lorenz     RegionMapper Cnt(this, S);
923*ee02499aSAlex Lorenz     Cnt.beginRegion();
924*ee02499aSAlex Lorenz   }
925*ee02499aSAlex Lorenz 
926*ee02499aSAlex Lorenz   void VisitCXXCatchStmt(const CXXCatchStmt *S) {
927*ee02499aSAlex Lorenz     mapSourceCodeRange(S->getLocStart());
928*ee02499aSAlex Lorenz     // Counter tracks the catch statement's handler block.
929*ee02499aSAlex Lorenz     RegionMapper Cnt(this, S);
930*ee02499aSAlex Lorenz     Cnt.beginRegion();
931*ee02499aSAlex Lorenz     VisitSubStmtRBraceState(S->getHandlerBlock());
932*ee02499aSAlex Lorenz   }
933*ee02499aSAlex Lorenz 
934*ee02499aSAlex Lorenz   void VisitAbstractConditionalOperator(const AbstractConditionalOperator *E) {
935*ee02499aSAlex Lorenz     Visit(E->getCond());
936*ee02499aSAlex Lorenz     mapToken(E->getQuestionLoc());
937*ee02499aSAlex Lorenz     auto State = getCurrentState();
938*ee02499aSAlex Lorenz 
939*ee02499aSAlex Lorenz     // Counter tracks the "true" part of a conditional operator. The
940*ee02499aSAlex Lorenz     // count in the "false" part will be calculated from this counter.
941*ee02499aSAlex Lorenz     RegionMapper Cnt(this, E);
942*ee02499aSAlex Lorenz     Cnt.beginRegion();
943*ee02499aSAlex Lorenz     Visit(E->getTrueExpr());
944*ee02499aSAlex Lorenz     Cnt.adjustForControlFlow();
945*ee02499aSAlex Lorenz 
946*ee02499aSAlex Lorenz     mapToken(State, E->getColonLoc());
947*ee02499aSAlex Lorenz 
948*ee02499aSAlex Lorenz     Cnt.beginElseRegion();
949*ee02499aSAlex Lorenz     Visit(E->getFalseExpr());
950*ee02499aSAlex Lorenz     Cnt.adjustForControlFlow();
951*ee02499aSAlex Lorenz 
952*ee02499aSAlex Lorenz     Cnt.applyAdjustmentsToRegion();
953*ee02499aSAlex Lorenz   }
954*ee02499aSAlex Lorenz 
955*ee02499aSAlex Lorenz   void VisitBinLAnd(const BinaryOperator *E) {
956*ee02499aSAlex Lorenz     Visit(E->getLHS());
957*ee02499aSAlex Lorenz     mapToken(E->getOperatorLoc());
958*ee02499aSAlex Lorenz     // Counter tracks the right hand side of a logical and operator.
959*ee02499aSAlex Lorenz     RegionMapper Cnt(this, E);
960*ee02499aSAlex Lorenz     Cnt.beginRegion();
961*ee02499aSAlex Lorenz     Visit(E->getRHS());
962*ee02499aSAlex Lorenz     Cnt.adjustForControlFlow();
963*ee02499aSAlex Lorenz     Cnt.applyAdjustmentsToRegion();
964*ee02499aSAlex Lorenz   }
965*ee02499aSAlex Lorenz 
966*ee02499aSAlex Lorenz   void VisitBinLOr(const BinaryOperator *E) {
967*ee02499aSAlex Lorenz     Visit(E->getLHS());
968*ee02499aSAlex Lorenz     mapToken(E->getOperatorLoc());
969*ee02499aSAlex Lorenz     // Counter tracks the right hand side of a logical or operator.
970*ee02499aSAlex Lorenz     RegionMapper Cnt(this, E);
971*ee02499aSAlex Lorenz     Cnt.beginRegion();
972*ee02499aSAlex Lorenz     Visit(E->getRHS());
973*ee02499aSAlex Lorenz     Cnt.adjustForControlFlow();
974*ee02499aSAlex Lorenz     Cnt.applyAdjustmentsToRegion();
975*ee02499aSAlex Lorenz   }
976*ee02499aSAlex Lorenz 
977*ee02499aSAlex Lorenz   void VisitParenExpr(const ParenExpr *E) {
978*ee02499aSAlex Lorenz     mapToken(E->getLParen());
979*ee02499aSAlex Lorenz     Visit(E->getSubExpr());
980*ee02499aSAlex Lorenz     mapToken(E->getRParen());
981*ee02499aSAlex Lorenz   }
982*ee02499aSAlex Lorenz 
983*ee02499aSAlex Lorenz   void VisitBinaryOperator(const BinaryOperator *E) {
984*ee02499aSAlex Lorenz     Visit(E->getLHS());
985*ee02499aSAlex Lorenz     mapToken(E->getOperatorLoc());
986*ee02499aSAlex Lorenz     Visit(E->getRHS());
987*ee02499aSAlex Lorenz   }
988*ee02499aSAlex Lorenz 
989*ee02499aSAlex Lorenz   void VisitUnaryOperator(const UnaryOperator *E) {
990*ee02499aSAlex Lorenz     bool Postfix = E->isPostfix();
991*ee02499aSAlex Lorenz     if (!Postfix)
992*ee02499aSAlex Lorenz       mapToken(E->getOperatorLoc());
993*ee02499aSAlex Lorenz     Visit(E->getSubExpr());
994*ee02499aSAlex Lorenz     if (Postfix)
995*ee02499aSAlex Lorenz       mapToken(E->getOperatorLoc());
996*ee02499aSAlex Lorenz   }
997*ee02499aSAlex Lorenz 
998*ee02499aSAlex Lorenz   void VisitMemberExpr(const MemberExpr *E) {
999*ee02499aSAlex Lorenz     Visit(E->getBase());
1000*ee02499aSAlex Lorenz     mapToken(E->getMemberLoc());
1001*ee02499aSAlex Lorenz   }
1002*ee02499aSAlex Lorenz 
1003*ee02499aSAlex Lorenz   void VisitCallExpr(const CallExpr *E) {
1004*ee02499aSAlex Lorenz     Visit(E->getCallee());
1005*ee02499aSAlex Lorenz     for (const auto &Arg : E->arguments())
1006*ee02499aSAlex Lorenz       Visit(Arg);
1007*ee02499aSAlex Lorenz     mapToken(E->getRParenLoc());
1008*ee02499aSAlex Lorenz   }
1009*ee02499aSAlex Lorenz 
1010*ee02499aSAlex Lorenz   void VisitArraySubscriptExpr(const ArraySubscriptExpr *E) {
1011*ee02499aSAlex Lorenz     Visit(E->getLHS());
1012*ee02499aSAlex Lorenz     Visit(E->getRHS());
1013*ee02499aSAlex Lorenz     mapToken(E->getRBracketLoc());
1014*ee02499aSAlex Lorenz   }
1015*ee02499aSAlex Lorenz 
1016*ee02499aSAlex Lorenz   void VisitCStyleCastExpr(const CStyleCastExpr *E) {
1017*ee02499aSAlex Lorenz     mapToken(E->getLParenLoc());
1018*ee02499aSAlex Lorenz     mapToken(E->getRParenLoc());
1019*ee02499aSAlex Lorenz     Visit(E->getSubExpr());
1020*ee02499aSAlex Lorenz   }
1021*ee02499aSAlex Lorenz 
1022*ee02499aSAlex Lorenz   // Map literals as tokens so that the macros like #define PI 3.14
1023*ee02499aSAlex Lorenz   // won't generate coverage mapping regions.
1024*ee02499aSAlex Lorenz 
1025*ee02499aSAlex Lorenz   void VisitIntegerLiteral(const IntegerLiteral *E) {
1026*ee02499aSAlex Lorenz     mapToken(E->getLocStart());
1027*ee02499aSAlex Lorenz   }
1028*ee02499aSAlex Lorenz 
1029*ee02499aSAlex Lorenz   void VisitFloatingLiteral(const FloatingLiteral *E) {
1030*ee02499aSAlex Lorenz     mapToken(E->getLocStart());
1031*ee02499aSAlex Lorenz   }
1032*ee02499aSAlex Lorenz 
1033*ee02499aSAlex Lorenz   void VisitCharacterLiteral(const CharacterLiteral *E) {
1034*ee02499aSAlex Lorenz     mapToken(E->getLocStart());
1035*ee02499aSAlex Lorenz   }
1036*ee02499aSAlex Lorenz 
1037*ee02499aSAlex Lorenz   void VisitStringLiteral(const StringLiteral *E) {
1038*ee02499aSAlex Lorenz     mapToken(E->getLocStart());
1039*ee02499aSAlex Lorenz   }
1040*ee02499aSAlex Lorenz 
1041*ee02499aSAlex Lorenz   void VisitImaginaryLiteral(const ImaginaryLiteral *E) {
1042*ee02499aSAlex Lorenz     mapToken(E->getLocStart());
1043*ee02499aSAlex Lorenz   }
1044*ee02499aSAlex Lorenz };
1045*ee02499aSAlex Lorenz }
1046*ee02499aSAlex Lorenz 
1047*ee02499aSAlex Lorenz static bool isMachO(const CodeGenModule &CGM) {
1048*ee02499aSAlex Lorenz   return CGM.getTarget().getTriple().isOSBinFormatMachO();
1049*ee02499aSAlex Lorenz }
1050*ee02499aSAlex Lorenz 
1051*ee02499aSAlex Lorenz static StringRef getCoverageSection(const CodeGenModule &CGM) {
1052*ee02499aSAlex Lorenz   return isMachO(CGM) ? "__DATA,__llvm_covmap" : "__llvm_covmap";
1053*ee02499aSAlex Lorenz }
1054*ee02499aSAlex Lorenz 
1055*ee02499aSAlex Lorenz void CoverageMappingModuleGen::addFunctionMappingRecord(
1056*ee02499aSAlex Lorenz     llvm::GlobalVariable *FunctionName, unsigned FunctionNameSize,
1057*ee02499aSAlex Lorenz     const std::string &CoverageMapping) {
1058*ee02499aSAlex Lorenz   llvm::LLVMContext &Ctx = CGM.getLLVMContext();
1059*ee02499aSAlex Lorenz   auto *Int32Ty = llvm::Type::getInt32Ty(Ctx);
1060*ee02499aSAlex Lorenz   auto *Int8PtrTy = llvm::Type::getInt8PtrTy(Ctx);
1061*ee02499aSAlex Lorenz   if (!FunctionRecordTy) {
1062*ee02499aSAlex Lorenz     llvm::Type *FunctionRecordTypes[] = {Int8PtrTy, Int32Ty, Int32Ty};
1063*ee02499aSAlex Lorenz     FunctionRecordTy =
1064*ee02499aSAlex Lorenz         llvm::StructType::get(Ctx, makeArrayRef(FunctionRecordTypes));
1065*ee02499aSAlex Lorenz   }
1066*ee02499aSAlex Lorenz 
1067*ee02499aSAlex Lorenz   llvm::Constant *FunctionRecordVals[] = {
1068*ee02499aSAlex Lorenz       llvm::ConstantExpr::getBitCast(FunctionName, Int8PtrTy),
1069*ee02499aSAlex Lorenz       llvm::ConstantInt::get(Int32Ty, FunctionNameSize),
1070*ee02499aSAlex Lorenz       llvm::ConstantInt::get(Int32Ty, CoverageMapping.size())};
1071*ee02499aSAlex Lorenz   FunctionRecords.push_back(llvm::ConstantStruct::get(
1072*ee02499aSAlex Lorenz       FunctionRecordTy, makeArrayRef(FunctionRecordVals)));
1073*ee02499aSAlex Lorenz   CoverageMappings += CoverageMapping;
1074*ee02499aSAlex Lorenz }
1075*ee02499aSAlex Lorenz 
1076*ee02499aSAlex Lorenz void CoverageMappingModuleGen::emit() {
1077*ee02499aSAlex Lorenz   if (FunctionRecords.empty())
1078*ee02499aSAlex Lorenz     return;
1079*ee02499aSAlex Lorenz   llvm::LLVMContext &Ctx = CGM.getLLVMContext();
1080*ee02499aSAlex Lorenz   auto *Int32Ty = llvm::Type::getInt32Ty(Ctx);
1081*ee02499aSAlex Lorenz 
1082*ee02499aSAlex Lorenz   // Create the filenames and merge them with coverage mappings
1083*ee02499aSAlex Lorenz   llvm::SmallVector<std::string, 16> FilenameStrs;
1084*ee02499aSAlex Lorenz   llvm::SmallVector<StringRef, 16> FilenameRefs;
1085*ee02499aSAlex Lorenz   FilenameStrs.resize(FileEntries.size());
1086*ee02499aSAlex Lorenz   FilenameRefs.resize(FileEntries.size());
1087*ee02499aSAlex Lorenz   for (const auto &Entry : FileEntries) {
1088*ee02499aSAlex Lorenz     llvm::SmallString<256> Path(Entry.first->getName());
1089*ee02499aSAlex Lorenz     llvm::sys::fs::make_absolute(Path);
1090*ee02499aSAlex Lorenz 
1091*ee02499aSAlex Lorenz     auto I = Entry.second;
1092*ee02499aSAlex Lorenz     FilenameStrs[I] = std::move(std::string(Path.begin(), Path.end()));
1093*ee02499aSAlex Lorenz     FilenameRefs[I] = FilenameStrs[I];
1094*ee02499aSAlex Lorenz   }
1095*ee02499aSAlex Lorenz 
1096*ee02499aSAlex Lorenz   std::string FilenamesAndCoverageMappings;
1097*ee02499aSAlex Lorenz   llvm::raw_string_ostream OS(FilenamesAndCoverageMappings);
1098*ee02499aSAlex Lorenz   CoverageFilenamesSectionWriter(FilenameRefs).write(OS);
1099*ee02499aSAlex Lorenz   OS << CoverageMappings;
1100*ee02499aSAlex Lorenz   size_t CoverageMappingSize = CoverageMappings.size();
1101*ee02499aSAlex Lorenz   size_t FilenamesSize = OS.str().size() - CoverageMappingSize;
1102*ee02499aSAlex Lorenz   // Append extra zeroes if necessary to ensure that the size of the filenames
1103*ee02499aSAlex Lorenz   // and coverage mappings is a multiple of 8.
1104*ee02499aSAlex Lorenz   if (size_t Rem = OS.str().size() % 8) {
1105*ee02499aSAlex Lorenz     CoverageMappingSize += 8 - Rem;
1106*ee02499aSAlex Lorenz     for (size_t I = 0, S = 8 - Rem; I < S; ++I)
1107*ee02499aSAlex Lorenz       OS << '\0';
1108*ee02499aSAlex Lorenz   }
1109*ee02499aSAlex Lorenz   auto *FilenamesAndMappingsVal =
1110*ee02499aSAlex Lorenz       llvm::ConstantDataArray::getString(Ctx, OS.str(), false);
1111*ee02499aSAlex Lorenz 
1112*ee02499aSAlex Lorenz   // Create the deferred function records array
1113*ee02499aSAlex Lorenz   auto RecordsTy =
1114*ee02499aSAlex Lorenz       llvm::ArrayType::get(FunctionRecordTy, FunctionRecords.size());
1115*ee02499aSAlex Lorenz   auto RecordsVal = llvm::ConstantArray::get(RecordsTy, FunctionRecords);
1116*ee02499aSAlex Lorenz 
1117*ee02499aSAlex Lorenz   // Create the coverage data record
1118*ee02499aSAlex Lorenz   llvm::Type *CovDataTypes[] = {Int32Ty,   Int32Ty,
1119*ee02499aSAlex Lorenz                                 Int32Ty,   Int32Ty,
1120*ee02499aSAlex Lorenz                                 RecordsTy, FilenamesAndMappingsVal->getType()};
1121*ee02499aSAlex Lorenz   auto CovDataTy = llvm::StructType::get(Ctx, makeArrayRef(CovDataTypes));
1122*ee02499aSAlex Lorenz   llvm::Constant *TUDataVals[] = {
1123*ee02499aSAlex Lorenz       llvm::ConstantInt::get(Int32Ty, FunctionRecords.size()),
1124*ee02499aSAlex Lorenz       llvm::ConstantInt::get(Int32Ty, FilenamesSize),
1125*ee02499aSAlex Lorenz       llvm::ConstantInt::get(Int32Ty, CoverageMappingSize),
1126*ee02499aSAlex Lorenz       llvm::ConstantInt::get(Int32Ty,
1127*ee02499aSAlex Lorenz                              /*Version=*/CoverageMappingVersion1),
1128*ee02499aSAlex Lorenz       RecordsVal, FilenamesAndMappingsVal};
1129*ee02499aSAlex Lorenz   auto CovDataVal =
1130*ee02499aSAlex Lorenz       llvm::ConstantStruct::get(CovDataTy, makeArrayRef(TUDataVals));
1131*ee02499aSAlex Lorenz   auto CovData = new llvm::GlobalVariable(CGM.getModule(), CovDataTy, true,
1132*ee02499aSAlex Lorenz                                           llvm::GlobalValue::InternalLinkage,
1133*ee02499aSAlex Lorenz                                           CovDataVal,
1134*ee02499aSAlex Lorenz                                           "__llvm_coverage_mapping");
1135*ee02499aSAlex Lorenz 
1136*ee02499aSAlex Lorenz   CovData->setSection(getCoverageSection(CGM));
1137*ee02499aSAlex Lorenz   CovData->setAlignment(8);
1138*ee02499aSAlex Lorenz 
1139*ee02499aSAlex Lorenz   // Make sure the data doesn't get deleted.
1140*ee02499aSAlex Lorenz   CGM.addUsedGlobal(CovData);
1141*ee02499aSAlex Lorenz }
1142*ee02499aSAlex Lorenz 
1143*ee02499aSAlex Lorenz unsigned CoverageMappingModuleGen::getFileID(const FileEntry *File) {
1144*ee02499aSAlex Lorenz   auto It = FileEntries.find(File);
1145*ee02499aSAlex Lorenz   if (It != FileEntries.end())
1146*ee02499aSAlex Lorenz     return It->second;
1147*ee02499aSAlex Lorenz   unsigned FileID = FileEntries.size();
1148*ee02499aSAlex Lorenz   FileEntries.insert(std::make_pair(File, FileID));
1149*ee02499aSAlex Lorenz   return FileID;
1150*ee02499aSAlex Lorenz }
1151*ee02499aSAlex Lorenz 
1152*ee02499aSAlex Lorenz void CoverageMappingGen::emitCounterMapping(const Decl *D,
1153*ee02499aSAlex Lorenz                                             llvm::raw_ostream &OS) {
1154*ee02499aSAlex Lorenz   assert(CounterMap);
1155*ee02499aSAlex Lorenz   CounterCoverageMappingBuilder Walker(CVM, *CounterMap, NumRegionCounters, SM,
1156*ee02499aSAlex Lorenz                                        LangOpts);
1157*ee02499aSAlex Lorenz   Walker.VisitDecl(D);
1158*ee02499aSAlex Lorenz   Walker.write(OS);
1159*ee02499aSAlex Lorenz }
1160*ee02499aSAlex Lorenz 
1161*ee02499aSAlex Lorenz void CoverageMappingGen::emitEmptyMapping(const Decl *D,
1162*ee02499aSAlex Lorenz                                           llvm::raw_ostream &OS) {
1163*ee02499aSAlex Lorenz   EmptyCoverageMappingBuilder Walker(CVM, SM, LangOpts);
1164*ee02499aSAlex Lorenz   Walker.VisitDecl(D);
1165*ee02499aSAlex Lorenz   Walker.write(OS);
1166*ee02499aSAlex Lorenz }
1167