1 //===--- PreprocessingRecord.cpp - Record of Preprocessing ------*- C++ -*-===//
2 //
3 // The LLVM Compiler Infrastructure
4 //
5 // This file is distributed under the University of Illinois Open Source
6 // License. See LICENSE.TXT for details.
7 //
8 //===----------------------------------------------------------------------===//
9 //
10 // This file implements the PreprocessingRecord class, which maintains a record
11 // of what occurred during preprocessing, and its helpers.
12 //
13 //===----------------------------------------------------------------------===//
14 #include "clang/Lex/PreprocessingRecord.h"
15 #include "clang/Lex/MacroInfo.h"
16 #include "clang/Lex/Token.h"
17 #include "llvm/Support/Capacity.h"
18 #include "llvm/Support/ErrorHandling.h"
19
20 using namespace clang;
21
~ExternalPreprocessingRecordSource()22 ExternalPreprocessingRecordSource::~ExternalPreprocessingRecordSource() { }
23
24
InclusionDirective(PreprocessingRecord & PPRec,InclusionKind Kind,StringRef FileName,bool InQuotes,bool ImportedModule,const FileEntry * File,SourceRange Range)25 InclusionDirective::InclusionDirective(PreprocessingRecord &PPRec,
26 InclusionKind Kind,
27 StringRef FileName,
28 bool InQuotes, bool ImportedModule,
29 const FileEntry *File,
30 SourceRange Range)
31 : PreprocessingDirective(InclusionDirectiveKind, Range),
32 InQuotes(InQuotes), Kind(Kind), ImportedModule(ImportedModule), File(File)
33 {
34 char *Memory
35 = (char*)PPRec.Allocate(FileName.size() + 1, llvm::alignOf<char>());
36 memcpy(Memory, FileName.data(), FileName.size());
37 Memory[FileName.size()] = 0;
38 this->FileName = StringRef(Memory, FileName.size());
39 }
40
PreprocessingRecord(SourceManager & SM)41 PreprocessingRecord::PreprocessingRecord(SourceManager &SM)
42 : SourceMgr(SM),
43 ExternalSource(nullptr) {
44 }
45
46 /// \brief Returns a pair of [Begin, End) iterators of preprocessed entities
47 /// that source range \p Range encompasses.
48 llvm::iterator_range<PreprocessingRecord::iterator>
getPreprocessedEntitiesInRange(SourceRange Range)49 PreprocessingRecord::getPreprocessedEntitiesInRange(SourceRange Range) {
50 if (Range.isInvalid())
51 return llvm::make_range(iterator(), iterator());
52
53 if (CachedRangeQuery.Range == Range) {
54 return llvm::make_range(iterator(this, CachedRangeQuery.Result.first),
55 iterator(this, CachedRangeQuery.Result.second));
56 }
57
58 std::pair<int, int> Res = getPreprocessedEntitiesInRangeSlow(Range);
59
60 CachedRangeQuery.Range = Range;
61 CachedRangeQuery.Result = Res;
62
63 return llvm::make_range(iterator(this, Res.first),
64 iterator(this, Res.second));
65 }
66
isPreprocessedEntityIfInFileID(PreprocessedEntity * PPE,FileID FID,SourceManager & SM)67 static bool isPreprocessedEntityIfInFileID(PreprocessedEntity *PPE, FileID FID,
68 SourceManager &SM) {
69 assert(FID.isValid());
70 if (!PPE)
71 return false;
72
73 SourceLocation Loc = PPE->getSourceRange().getBegin();
74 if (Loc.isInvalid())
75 return false;
76
77 return SM.isInFileID(SM.getFileLoc(Loc), FID);
78 }
79
80 /// \brief Returns true if the preprocessed entity that \arg PPEI iterator
81 /// points to is coming from the file \arg FID.
82 ///
83 /// Can be used to avoid implicit deserializations of preallocated
84 /// preprocessed entities if we only care about entities of a specific file
85 /// and not from files \#included in the range given at
86 /// \see getPreprocessedEntitiesInRange.
isEntityInFileID(iterator PPEI,FileID FID)87 bool PreprocessingRecord::isEntityInFileID(iterator PPEI, FileID FID) {
88 if (FID.isInvalid())
89 return false;
90
91 int Pos = std::distance(iterator(this, 0), PPEI);
92 if (Pos < 0) {
93 if (unsigned(-Pos-1) >= LoadedPreprocessedEntities.size()) {
94 assert(0 && "Out-of bounds loaded preprocessed entity");
95 return false;
96 }
97 assert(ExternalSource && "No external source to load from");
98 unsigned LoadedIndex = LoadedPreprocessedEntities.size()+Pos;
99 if (PreprocessedEntity *PPE = LoadedPreprocessedEntities[LoadedIndex])
100 return isPreprocessedEntityIfInFileID(PPE, FID, SourceMgr);
101
102 // See if the external source can see if the entity is in the file without
103 // deserializing it.
104 Optional<bool> IsInFile =
105 ExternalSource->isPreprocessedEntityInFileID(LoadedIndex, FID);
106 if (IsInFile.hasValue())
107 return IsInFile.getValue();
108
109 // The external source did not provide a definite answer, go and deserialize
110 // the entity to check it.
111 return isPreprocessedEntityIfInFileID(
112 getLoadedPreprocessedEntity(LoadedIndex),
113 FID, SourceMgr);
114 }
115
116 if (unsigned(Pos) >= PreprocessedEntities.size()) {
117 assert(0 && "Out-of bounds local preprocessed entity");
118 return false;
119 }
120 return isPreprocessedEntityIfInFileID(PreprocessedEntities[Pos],
121 FID, SourceMgr);
122 }
123
124 /// \brief Returns a pair of [Begin, End) iterators of preprocessed entities
125 /// that source range \arg R encompasses.
126 std::pair<int, int>
getPreprocessedEntitiesInRangeSlow(SourceRange Range)127 PreprocessingRecord::getPreprocessedEntitiesInRangeSlow(SourceRange Range) {
128 assert(Range.isValid());
129 assert(!SourceMgr.isBeforeInTranslationUnit(Range.getEnd(),Range.getBegin()));
130
131 std::pair<unsigned, unsigned>
132 Local = findLocalPreprocessedEntitiesInRange(Range);
133
134 // Check if range spans local entities.
135 if (!ExternalSource || SourceMgr.isLocalSourceLocation(Range.getBegin()))
136 return std::make_pair(Local.first, Local.second);
137
138 std::pair<unsigned, unsigned>
139 Loaded = ExternalSource->findPreprocessedEntitiesInRange(Range);
140
141 // Check if range spans local entities.
142 if (Loaded.first == Loaded.second)
143 return std::make_pair(Local.first, Local.second);
144
145 unsigned TotalLoaded = LoadedPreprocessedEntities.size();
146
147 // Check if range spans loaded entities.
148 if (Local.first == Local.second)
149 return std::make_pair(int(Loaded.first)-TotalLoaded,
150 int(Loaded.second)-TotalLoaded);
151
152 // Range spands loaded and local entities.
153 return std::make_pair(int(Loaded.first)-TotalLoaded, Local.second);
154 }
155
156 std::pair<unsigned, unsigned>
findLocalPreprocessedEntitiesInRange(SourceRange Range) const157 PreprocessingRecord::findLocalPreprocessedEntitiesInRange(
158 SourceRange Range) const {
159 if (Range.isInvalid())
160 return std::make_pair(0,0);
161 assert(!SourceMgr.isBeforeInTranslationUnit(Range.getEnd(),Range.getBegin()));
162
163 unsigned Begin = findBeginLocalPreprocessedEntity(Range.getBegin());
164 unsigned End = findEndLocalPreprocessedEntity(Range.getEnd());
165 return std::make_pair(Begin, End);
166 }
167
168 namespace {
169
170 template <SourceLocation (SourceRange::*getRangeLoc)() const>
171 struct PPEntityComp {
172 const SourceManager &SM;
173
PPEntityComp__anon975eb77c0111::PPEntityComp174 explicit PPEntityComp(const SourceManager &SM) : SM(SM) { }
175
operator ()__anon975eb77c0111::PPEntityComp176 bool operator()(PreprocessedEntity *L, PreprocessedEntity *R) const {
177 SourceLocation LHS = getLoc(L);
178 SourceLocation RHS = getLoc(R);
179 return SM.isBeforeInTranslationUnit(LHS, RHS);
180 }
181
operator ()__anon975eb77c0111::PPEntityComp182 bool operator()(PreprocessedEntity *L, SourceLocation RHS) const {
183 SourceLocation LHS = getLoc(L);
184 return SM.isBeforeInTranslationUnit(LHS, RHS);
185 }
186
operator ()__anon975eb77c0111::PPEntityComp187 bool operator()(SourceLocation LHS, PreprocessedEntity *R) const {
188 SourceLocation RHS = getLoc(R);
189 return SM.isBeforeInTranslationUnit(LHS, RHS);
190 }
191
getLoc__anon975eb77c0111::PPEntityComp192 SourceLocation getLoc(PreprocessedEntity *PPE) const {
193 SourceRange Range = PPE->getSourceRange();
194 return (Range.*getRangeLoc)();
195 }
196 };
197
198 }
199
findBeginLocalPreprocessedEntity(SourceLocation Loc) const200 unsigned PreprocessingRecord::findBeginLocalPreprocessedEntity(
201 SourceLocation Loc) const {
202 if (SourceMgr.isLoadedSourceLocation(Loc))
203 return 0;
204
205 size_t Count = PreprocessedEntities.size();
206 size_t Half;
207 std::vector<PreprocessedEntity *>::const_iterator
208 First = PreprocessedEntities.begin();
209 std::vector<PreprocessedEntity *>::const_iterator I;
210
211 // Do a binary search manually instead of using std::lower_bound because
212 // The end locations of entities may be unordered (when a macro expansion
213 // is inside another macro argument), but for this case it is not important
214 // whether we get the first macro expansion or its containing macro.
215 while (Count > 0) {
216 Half = Count/2;
217 I = First;
218 std::advance(I, Half);
219 if (SourceMgr.isBeforeInTranslationUnit((*I)->getSourceRange().getEnd(),
220 Loc)){
221 First = I;
222 ++First;
223 Count = Count - Half - 1;
224 } else
225 Count = Half;
226 }
227
228 return First - PreprocessedEntities.begin();
229 }
230
findEndLocalPreprocessedEntity(SourceLocation Loc) const231 unsigned PreprocessingRecord::findEndLocalPreprocessedEntity(
232 SourceLocation Loc) const {
233 if (SourceMgr.isLoadedSourceLocation(Loc))
234 return 0;
235
236 std::vector<PreprocessedEntity *>::const_iterator
237 I = std::upper_bound(PreprocessedEntities.begin(),
238 PreprocessedEntities.end(),
239 Loc,
240 PPEntityComp<&SourceRange::getBegin>(SourceMgr));
241 return I - PreprocessedEntities.begin();
242 }
243
244 PreprocessingRecord::PPEntityID
addPreprocessedEntity(PreprocessedEntity * Entity)245 PreprocessingRecord::addPreprocessedEntity(PreprocessedEntity *Entity) {
246 assert(Entity);
247 SourceLocation BeginLoc = Entity->getSourceRange().getBegin();
248
249 if (isa<MacroDefinitionRecord>(Entity)) {
250 assert((PreprocessedEntities.empty() ||
251 !SourceMgr.isBeforeInTranslationUnit(
252 BeginLoc,
253 PreprocessedEntities.back()->getSourceRange().getBegin())) &&
254 "a macro definition was encountered out-of-order");
255 PreprocessedEntities.push_back(Entity);
256 return getPPEntityID(PreprocessedEntities.size()-1, /*isLoaded=*/false);
257 }
258
259 // Check normal case, this entity begin location is after the previous one.
260 if (PreprocessedEntities.empty() ||
261 !SourceMgr.isBeforeInTranslationUnit(BeginLoc,
262 PreprocessedEntities.back()->getSourceRange().getBegin())) {
263 PreprocessedEntities.push_back(Entity);
264 return getPPEntityID(PreprocessedEntities.size()-1, /*isLoaded=*/false);
265 }
266
267 // The entity's location is not after the previous one; this can happen with
268 // include directives that form the filename using macros, e.g:
269 // "#include MACRO(STUFF)"
270 // or with macro expansions inside macro arguments where the arguments are
271 // not expanded in the same order as listed, e.g:
272 // \code
273 // #define M1 1
274 // #define M2 2
275 // #define FM(x,y) y x
276 // FM(M1, M2)
277 // \endcode
278
279 typedef std::vector<PreprocessedEntity *>::iterator pp_iter;
280
281 // Usually there are few macro expansions when defining the filename, do a
282 // linear search for a few entities.
283 unsigned count = 0;
284 for (pp_iter RI = PreprocessedEntities.end(),
285 Begin = PreprocessedEntities.begin();
286 RI != Begin && count < 4; --RI, ++count) {
287 pp_iter I = RI;
288 --I;
289 if (!SourceMgr.isBeforeInTranslationUnit(BeginLoc,
290 (*I)->getSourceRange().getBegin())) {
291 pp_iter insertI = PreprocessedEntities.insert(RI, Entity);
292 return getPPEntityID(insertI - PreprocessedEntities.begin(),
293 /*isLoaded=*/false);
294 }
295 }
296
297 // Linear search unsuccessful. Do a binary search.
298 pp_iter I = std::upper_bound(PreprocessedEntities.begin(),
299 PreprocessedEntities.end(),
300 BeginLoc,
301 PPEntityComp<&SourceRange::getBegin>(SourceMgr));
302 pp_iter insertI = PreprocessedEntities.insert(I, Entity);
303 return getPPEntityID(insertI - PreprocessedEntities.begin(),
304 /*isLoaded=*/false);
305 }
306
SetExternalSource(ExternalPreprocessingRecordSource & Source)307 void PreprocessingRecord::SetExternalSource(
308 ExternalPreprocessingRecordSource &Source) {
309 assert(!ExternalSource &&
310 "Preprocessing record already has an external source");
311 ExternalSource = &Source;
312 }
313
allocateLoadedEntities(unsigned NumEntities)314 unsigned PreprocessingRecord::allocateLoadedEntities(unsigned NumEntities) {
315 unsigned Result = LoadedPreprocessedEntities.size();
316 LoadedPreprocessedEntities.resize(LoadedPreprocessedEntities.size()
317 + NumEntities);
318 return Result;
319 }
320
RegisterMacroDefinition(MacroInfo * Macro,MacroDefinitionRecord * Def)321 void PreprocessingRecord::RegisterMacroDefinition(MacroInfo *Macro,
322 MacroDefinitionRecord *Def) {
323 MacroDefinitions[Macro] = Def;
324 }
325
326 /// \brief Retrieve the preprocessed entity at the given ID.
getPreprocessedEntity(PPEntityID PPID)327 PreprocessedEntity *PreprocessingRecord::getPreprocessedEntity(PPEntityID PPID){
328 if (PPID.ID < 0) {
329 unsigned Index = -PPID.ID - 1;
330 assert(Index < LoadedPreprocessedEntities.size() &&
331 "Out-of bounds loaded preprocessed entity");
332 return getLoadedPreprocessedEntity(Index);
333 }
334
335 if (PPID.ID == 0)
336 return nullptr;
337 unsigned Index = PPID.ID - 1;
338 assert(Index < PreprocessedEntities.size() &&
339 "Out-of bounds local preprocessed entity");
340 return PreprocessedEntities[Index];
341 }
342
343 /// \brief Retrieve the loaded preprocessed entity at the given index.
344 PreprocessedEntity *
getLoadedPreprocessedEntity(unsigned Index)345 PreprocessingRecord::getLoadedPreprocessedEntity(unsigned Index) {
346 assert(Index < LoadedPreprocessedEntities.size() &&
347 "Out-of bounds loaded preprocessed entity");
348 assert(ExternalSource && "No external source to load from");
349 PreprocessedEntity *&Entity = LoadedPreprocessedEntities[Index];
350 if (!Entity) {
351 Entity = ExternalSource->ReadPreprocessedEntity(Index);
352 if (!Entity) // Failed to load.
353 Entity = new (*this)
354 PreprocessedEntity(PreprocessedEntity::InvalidKind, SourceRange());
355 }
356 return Entity;
357 }
358
359 MacroDefinitionRecord *
findMacroDefinition(const MacroInfo * MI)360 PreprocessingRecord::findMacroDefinition(const MacroInfo *MI) {
361 llvm::DenseMap<const MacroInfo *, MacroDefinitionRecord *>::iterator Pos =
362 MacroDefinitions.find(MI);
363 if (Pos == MacroDefinitions.end())
364 return nullptr;
365
366 return Pos->second;
367 }
368
addMacroExpansion(const Token & Id,const MacroInfo * MI,SourceRange Range)369 void PreprocessingRecord::addMacroExpansion(const Token &Id,
370 const MacroInfo *MI,
371 SourceRange Range) {
372 // We don't record nested macro expansions.
373 if (Id.getLocation().isMacroID())
374 return;
375
376 if (MI->isBuiltinMacro())
377 addPreprocessedEntity(new (*this)
378 MacroExpansion(Id.getIdentifierInfo(), Range));
379 else if (MacroDefinitionRecord *Def = findMacroDefinition(MI))
380 addPreprocessedEntity(new (*this) MacroExpansion(Def, Range));
381 }
382
Ifdef(SourceLocation Loc,const Token & MacroNameTok,const MacroDefinition & MD)383 void PreprocessingRecord::Ifdef(SourceLocation Loc, const Token &MacroNameTok,
384 const MacroDefinition &MD) {
385 // This is not actually a macro expansion but record it as a macro reference.
386 if (MD)
387 addMacroExpansion(MacroNameTok, MD.getMacroInfo(),
388 MacroNameTok.getLocation());
389 }
390
Ifndef(SourceLocation Loc,const Token & MacroNameTok,const MacroDefinition & MD)391 void PreprocessingRecord::Ifndef(SourceLocation Loc, const Token &MacroNameTok,
392 const MacroDefinition &MD) {
393 // This is not actually a macro expansion but record it as a macro reference.
394 if (MD)
395 addMacroExpansion(MacroNameTok, MD.getMacroInfo(),
396 MacroNameTok.getLocation());
397 }
398
Defined(const Token & MacroNameTok,const MacroDefinition & MD,SourceRange Range)399 void PreprocessingRecord::Defined(const Token &MacroNameTok,
400 const MacroDefinition &MD,
401 SourceRange Range) {
402 // This is not actually a macro expansion but record it as a macro reference.
403 if (MD)
404 addMacroExpansion(MacroNameTok, MD.getMacroInfo(),
405 MacroNameTok.getLocation());
406 }
407
SourceRangeSkipped(SourceRange Range)408 void PreprocessingRecord::SourceRangeSkipped(SourceRange Range) {
409 SkippedRanges.push_back(Range);
410 }
411
MacroExpands(const Token & Id,const MacroDefinition & MD,SourceRange Range,const MacroArgs * Args)412 void PreprocessingRecord::MacroExpands(const Token &Id,
413 const MacroDefinition &MD,
414 SourceRange Range,
415 const MacroArgs *Args) {
416 addMacroExpansion(Id, MD.getMacroInfo(), Range);
417 }
418
MacroDefined(const Token & Id,const MacroDirective * MD)419 void PreprocessingRecord::MacroDefined(const Token &Id,
420 const MacroDirective *MD) {
421 const MacroInfo *MI = MD->getMacroInfo();
422 SourceRange R(MI->getDefinitionLoc(), MI->getDefinitionEndLoc());
423 MacroDefinitionRecord *Def =
424 new (*this) MacroDefinitionRecord(Id.getIdentifierInfo(), R);
425 addPreprocessedEntity(Def);
426 MacroDefinitions[MI] = Def;
427 }
428
MacroUndefined(const Token & Id,const MacroDefinition & MD)429 void PreprocessingRecord::MacroUndefined(const Token &Id,
430 const MacroDefinition &MD) {
431 MD.forAllDefinitions([&](MacroInfo *MI) { MacroDefinitions.erase(MI); });
432 }
433
InclusionDirective(SourceLocation HashLoc,const clang::Token & IncludeTok,StringRef FileName,bool IsAngled,CharSourceRange FilenameRange,const FileEntry * File,StringRef SearchPath,StringRef RelativePath,const Module * Imported)434 void PreprocessingRecord::InclusionDirective(
435 SourceLocation HashLoc,
436 const clang::Token &IncludeTok,
437 StringRef FileName,
438 bool IsAngled,
439 CharSourceRange FilenameRange,
440 const FileEntry *File,
441 StringRef SearchPath,
442 StringRef RelativePath,
443 const Module *Imported) {
444 InclusionDirective::InclusionKind Kind = InclusionDirective::Include;
445
446 switch (IncludeTok.getIdentifierInfo()->getPPKeywordID()) {
447 case tok::pp_include:
448 Kind = InclusionDirective::Include;
449 break;
450
451 case tok::pp_import:
452 Kind = InclusionDirective::Import;
453 break;
454
455 case tok::pp_include_next:
456 Kind = InclusionDirective::IncludeNext;
457 break;
458
459 case tok::pp___include_macros:
460 Kind = InclusionDirective::IncludeMacros;
461 break;
462
463 default:
464 llvm_unreachable("Unknown include directive kind");
465 }
466
467 SourceLocation EndLoc;
468 if (!IsAngled) {
469 EndLoc = FilenameRange.getBegin();
470 } else {
471 EndLoc = FilenameRange.getEnd();
472 if (FilenameRange.isCharRange())
473 EndLoc = EndLoc.getLocWithOffset(-1); // the InclusionDirective expects
474 // a token range.
475 }
476 clang::InclusionDirective *ID
477 = new (*this) clang::InclusionDirective(*this, Kind, FileName, !IsAngled,
478 (bool)Imported,
479 File, SourceRange(HashLoc, EndLoc));
480 addPreprocessedEntity(ID);
481 }
482
getTotalMemory() const483 size_t PreprocessingRecord::getTotalMemory() const {
484 return BumpAlloc.getTotalMemory()
485 + llvm::capacity_in_bytes(MacroDefinitions)
486 + llvm::capacity_in_bytes(PreprocessedEntities)
487 + llvm::capacity_in_bytes(LoadedPreprocessedEntities);
488 }
489