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