blob: 0c8d948ce0a7bd619ce22e476b1d6bee1ec0485f [file] [log] [blame]
Douglas Gregor065f8d12010-03-18 17:52:52 +00001//===--- 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"
Douglas Gregor796d76a2010-10-20 22:00:55 +000017#include "clang/Basic/IdentifierTable.h"
18#include "llvm/Support/ErrorHandling.h"
Douglas Gregor065f8d12010-03-18 17:52:52 +000019
20using namespace clang;
21
Douglas Gregoraae92242010-03-19 21:51:54 +000022ExternalPreprocessingRecordSource::~ExternalPreprocessingRecordSource() { }
23
Douglas Gregorf09b6c92010-11-01 15:03:47 +000024
25InclusionDirective::InclusionDirective(PreprocessingRecord &PPRec,
26 InclusionKind Kind,
27 llvm::StringRef FileName,
28 bool InQuotes, const FileEntry *File,
29 SourceRange Range)
30 : PreprocessingDirective(InclusionDirectiveKind, Range),
31 InQuotes(InQuotes), Kind(Kind), File(File)
32{
33 char *Memory
34 = (char*)PPRec.Allocate(FileName.size() + 1, llvm::alignOf<char>());
35 memcpy(Memory, FileName.data(), FileName.size());
36 Memory[FileName.size()] = 0;
37 this->FileName = llvm::StringRef(Memory, FileName.size());
38}
39
Douglas Gregoraae92242010-03-19 21:51:54 +000040void PreprocessingRecord::MaybeLoadPreallocatedEntities() const {
41 if (!ExternalSource || LoadedPreallocatedEntities)
42 return;
43
44 LoadedPreallocatedEntities = true;
45 ExternalSource->ReadPreprocessedEntities();
46}
47
Douglas Gregor998caea2011-05-06 16:33:08 +000048PreprocessingRecord::PreprocessingRecord(bool IncludeNestedMacroInstantiations)
49 : IncludeNestedMacroInstantiations(IncludeNestedMacroInstantiations),
50 ExternalSource(0), NumPreallocatedEntities(0),
Douglas Gregoraae92242010-03-19 21:51:54 +000051 LoadedPreallocatedEntities(false)
52{
53}
54
55PreprocessingRecord::iterator
56PreprocessingRecord::begin(bool OnlyLocalEntities) {
57 if (OnlyLocalEntities)
58 return PreprocessedEntities.begin() + NumPreallocatedEntities;
59
60 MaybeLoadPreallocatedEntities();
61 return PreprocessedEntities.begin();
62}
63
64PreprocessingRecord::iterator PreprocessingRecord::end(bool OnlyLocalEntities) {
65 if (!OnlyLocalEntities)
66 MaybeLoadPreallocatedEntities();
67
68 return PreprocessedEntities.end();
69}
70
71PreprocessingRecord::const_iterator
72PreprocessingRecord::begin(bool OnlyLocalEntities) const {
73 if (OnlyLocalEntities)
74 return PreprocessedEntities.begin() + NumPreallocatedEntities;
75
76 MaybeLoadPreallocatedEntities();
77 return PreprocessedEntities.begin();
78}
79
80PreprocessingRecord::const_iterator
81PreprocessingRecord::end(bool OnlyLocalEntities) const {
82 if (!OnlyLocalEntities)
83 MaybeLoadPreallocatedEntities();
84
85 return PreprocessedEntities.end();
86}
87
Douglas Gregor065f8d12010-03-18 17:52:52 +000088void PreprocessingRecord::addPreprocessedEntity(PreprocessedEntity *Entity) {
89 PreprocessedEntities.push_back(Entity);
90}
91
Douglas Gregoraae92242010-03-19 21:51:54 +000092void PreprocessingRecord::SetExternalSource(
93 ExternalPreprocessingRecordSource &Source,
94 unsigned NumPreallocatedEntities) {
95 assert(!ExternalSource &&
96 "Preprocessing record already has an external source");
97 ExternalSource = &Source;
98 this->NumPreallocatedEntities = NumPreallocatedEntities;
99 PreprocessedEntities.insert(PreprocessedEntities.begin(),
100 NumPreallocatedEntities, 0);
101}
102
103void PreprocessingRecord::SetPreallocatedEntity(unsigned Index,
104 PreprocessedEntity *Entity) {
105 assert(Index < NumPreallocatedEntities &&"Out-of-bounds preallocated entity");
106 PreprocessedEntities[Index] = Entity;
107}
108
109void PreprocessingRecord::RegisterMacroDefinition(MacroInfo *Macro,
110 MacroDefinition *MD) {
111 MacroDefinitions[Macro] = MD;
112}
113
Douglas Gregor8aaca672010-03-19 21:58:23 +0000114MacroDefinition *PreprocessingRecord::findMacroDefinition(const MacroInfo *MI) {
Douglas Gregor7dc87222010-03-19 17:12:43 +0000115 llvm::DenseMap<const MacroInfo *, MacroDefinition *>::iterator Pos
116 = MacroDefinitions.find(MI);
117 if (Pos == MacroDefinitions.end())
118 return 0;
119
120 return Pos->second;
121}
122
123void PreprocessingRecord::MacroExpands(const Token &Id, const MacroInfo* MI) {
Douglas Gregor998caea2011-05-06 16:33:08 +0000124 if (!IncludeNestedMacroInstantiations && Id.getLocation().isMacroID())
125 return;
126
Douglas Gregor8aaca672010-03-19 21:58:23 +0000127 if (MacroDefinition *Def = findMacroDefinition(MI))
128 PreprocessedEntities.push_back(
Douglas Gregor7dc87222010-03-19 17:12:43 +0000129 new (*this) MacroInstantiation(Id.getIdentifierInfo(),
130 Id.getLocation(),
Douglas Gregor8aaca672010-03-19 21:58:23 +0000131 Def));
Douglas Gregor7dc87222010-03-19 17:12:43 +0000132}
133
Craig Silverstein1a9ca212010-11-19 21:33:15 +0000134void PreprocessingRecord::MacroDefined(const Token &Id,
Douglas Gregor7dc87222010-03-19 17:12:43 +0000135 const MacroInfo *MI) {
136 SourceRange R(MI->getDefinitionLoc(), MI->getDefinitionEndLoc());
137 MacroDefinition *Def
Craig Silverstein1a9ca212010-11-19 21:33:15 +0000138 = new (*this) MacroDefinition(Id.getIdentifierInfo(),
139 MI->getDefinitionLoc(),
140 R);
Douglas Gregor7dc87222010-03-19 17:12:43 +0000141 MacroDefinitions[MI] = Def;
142 PreprocessedEntities.push_back(Def);
143}
Douglas Gregoraae92242010-03-19 21:51:54 +0000144
Craig Silverstein1a9ca212010-11-19 21:33:15 +0000145void PreprocessingRecord::MacroUndefined(const Token &Id,
Douglas Gregor8aaca672010-03-19 21:58:23 +0000146 const MacroInfo *MI) {
147 llvm::DenseMap<const MacroInfo *, MacroDefinition *>::iterator Pos
148 = MacroDefinitions.find(MI);
149 if (Pos != MacroDefinitions.end())
150 MacroDefinitions.erase(Pos);
151}
152
Chandler Carruth3cc331a2011-03-16 18:34:36 +0000153void PreprocessingRecord::InclusionDirective(
154 SourceLocation HashLoc,
155 const clang::Token &IncludeTok,
156 llvm::StringRef FileName,
157 bool IsAngled,
158 const FileEntry *File,
159 clang::SourceLocation EndLoc,
Manuel Klimek0c69fd22011-04-26 21:50:03 +0000160 llvm::StringRef SearchPath,
161 llvm::StringRef RelativePath) {
Douglas Gregor796d76a2010-10-20 22:00:55 +0000162 InclusionDirective::InclusionKind Kind = InclusionDirective::Include;
163
164 switch (IncludeTok.getIdentifierInfo()->getPPKeywordID()) {
165 case tok::pp_include:
166 Kind = InclusionDirective::Include;
167 break;
168
169 case tok::pp_import:
170 Kind = InclusionDirective::Import;
171 break;
172
173 case tok::pp_include_next:
174 Kind = InclusionDirective::IncludeNext;
175 break;
176
177 case tok::pp___include_macros:
178 Kind = InclusionDirective::IncludeMacros;
179 break;
180
181 default:
182 llvm_unreachable("Unknown include directive kind");
183 return;
184 }
185
186 clang::InclusionDirective *ID
Douglas Gregorf09b6c92010-11-01 15:03:47 +0000187 = new (*this) clang::InclusionDirective(*this, Kind, FileName, !IsAngled,
188 File, SourceRange(HashLoc, EndLoc));
Douglas Gregor796d76a2010-10-20 22:00:55 +0000189 PreprocessedEntities.push_back(ID);
190}