blob: 27a08696320858e4c74e7afaf46d983a5f844c79 [file] [log] [blame]
Chris Lattner283c0612009-04-27 05:27:42 +00001//===--- PCHReaderDecl.cpp - Decl Deserialization ---------------*- 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 PCHReader::ReadDeclRecord method, which is the
11// entrypoint for loading a decl.
12//
13//===----------------------------------------------------------------------===//
14
15#include "clang/Frontend/PCHReader.h"
16#include "clang/AST/ASTConsumer.h"
17#include "clang/AST/ASTContext.h"
18#include "clang/AST/DeclVisitor.h"
19#include "clang/AST/DeclGroup.h"
20#include "clang/AST/Expr.h"
21using namespace clang;
22
Chris Lattner283c0612009-04-27 05:27:42 +000023
24//===----------------------------------------------------------------------===//
25// Declaration deserialization
26//===----------------------------------------------------------------------===//
27
28namespace {
29 class PCHDeclReader : public DeclVisitor<PCHDeclReader, void> {
30 PCHReader &Reader;
31 const PCHReader::RecordData &Record;
32 unsigned &Idx;
33
34 public:
35 PCHDeclReader(PCHReader &Reader, const PCHReader::RecordData &Record,
36 unsigned &Idx)
37 : Reader(Reader), Record(Record), Idx(Idx) { }
38
39 void VisitDecl(Decl *D);
40 void VisitTranslationUnitDecl(TranslationUnitDecl *TU);
41 void VisitNamedDecl(NamedDecl *ND);
42 void VisitTypeDecl(TypeDecl *TD);
43 void VisitTypedefDecl(TypedefDecl *TD);
44 void VisitTagDecl(TagDecl *TD);
45 void VisitEnumDecl(EnumDecl *ED);
46 void VisitRecordDecl(RecordDecl *RD);
47 void VisitValueDecl(ValueDecl *VD);
48 void VisitEnumConstantDecl(EnumConstantDecl *ECD);
49 void VisitFunctionDecl(FunctionDecl *FD);
50 void VisitFieldDecl(FieldDecl *FD);
51 void VisitVarDecl(VarDecl *VD);
52 void VisitImplicitParamDecl(ImplicitParamDecl *PD);
53 void VisitParmVarDecl(ParmVarDecl *PD);
54 void VisitOriginalParmVarDecl(OriginalParmVarDecl *PD);
55 void VisitFileScopeAsmDecl(FileScopeAsmDecl *AD);
56 void VisitBlockDecl(BlockDecl *BD);
57 std::pair<uint64_t, uint64_t> VisitDeclContext(DeclContext *DC);
58 void VisitObjCMethodDecl(ObjCMethodDecl *D);
59 void VisitObjCContainerDecl(ObjCContainerDecl *D);
60 void VisitObjCInterfaceDecl(ObjCInterfaceDecl *D);
61 void VisitObjCIvarDecl(ObjCIvarDecl *D);
62 void VisitObjCProtocolDecl(ObjCProtocolDecl *D);
63 void VisitObjCAtDefsFieldDecl(ObjCAtDefsFieldDecl *D);
64 void VisitObjCClassDecl(ObjCClassDecl *D);
65 void VisitObjCForwardProtocolDecl(ObjCForwardProtocolDecl *D);
66 void VisitObjCCategoryDecl(ObjCCategoryDecl *D);
67 void VisitObjCImplDecl(ObjCImplDecl *D);
68 void VisitObjCCategoryImplDecl(ObjCCategoryImplDecl *D);
69 void VisitObjCImplementationDecl(ObjCImplementationDecl *D);
70 void VisitObjCCompatibleAliasDecl(ObjCCompatibleAliasDecl *D);
71 void VisitObjCPropertyDecl(ObjCPropertyDecl *D);
72 void VisitObjCPropertyImplDecl(ObjCPropertyImplDecl *D);
73 };
74}
75
76void PCHDeclReader::VisitDecl(Decl *D) {
77 D->setDeclContext(cast_or_null<DeclContext>(Reader.GetDecl(Record[Idx++])));
78 D->setLexicalDeclContext(
79 cast_or_null<DeclContext>(Reader.GetDecl(Record[Idx++])));
80 D->setLocation(SourceLocation::getFromRawEncoding(Record[Idx++]));
81 D->setInvalidDecl(Record[Idx++]);
82 if (Record[Idx++])
Douglas Gregor98da6ae2009-06-18 16:11:24 +000083 D->addAttr(*Reader.getContext(), Reader.ReadAttributes());
Chris Lattner283c0612009-04-27 05:27:42 +000084 D->setImplicit(Record[Idx++]);
Douglas Gregor98189262009-06-19 23:52:42 +000085 D->setUsed(Record[Idx++]);
Chris Lattner283c0612009-04-27 05:27:42 +000086 D->setAccess((AccessSpecifier)Record[Idx++]);
87}
88
89void PCHDeclReader::VisitTranslationUnitDecl(TranslationUnitDecl *TU) {
90 VisitDecl(TU);
91}
92
93void PCHDeclReader::VisitNamedDecl(NamedDecl *ND) {
94 VisitDecl(ND);
95 ND->setDeclName(Reader.ReadDeclarationName(Record, Idx));
96}
97
98void PCHDeclReader::VisitTypeDecl(TypeDecl *TD) {
99 VisitNamedDecl(TD);
100 TD->setTypeForDecl(Reader.GetType(Record[Idx++]).getTypePtr());
101}
102
103void PCHDeclReader::VisitTypedefDecl(TypedefDecl *TD) {
104 // Note that we cannot use VisitTypeDecl here, because we need to
105 // set the underlying type of the typedef *before* we try to read
106 // the type associated with the TypedefDecl.
107 VisitNamedDecl(TD);
108 TD->setUnderlyingType(Reader.GetType(Record[Idx + 1]));
109 TD->setTypeForDecl(Reader.GetType(Record[Idx]).getTypePtr());
110 Idx += 2;
111}
112
113void PCHDeclReader::VisitTagDecl(TagDecl *TD) {
114 VisitTypeDecl(TD);
115 TD->setTagKind((TagDecl::TagKind)Record[Idx++]);
116 TD->setDefinition(Record[Idx++]);
117 TD->setTypedefForAnonDecl(
118 cast_or_null<TypedefDecl>(Reader.GetDecl(Record[Idx++])));
119}
120
121void PCHDeclReader::VisitEnumDecl(EnumDecl *ED) {
122 VisitTagDecl(ED);
123 ED->setIntegerType(Reader.GetType(Record[Idx++]));
Douglas Gregor8d717432009-05-27 17:20:35 +0000124 // FIXME: C++ InstantiatedFrom
Chris Lattner283c0612009-04-27 05:27:42 +0000125}
126
127void PCHDeclReader::VisitRecordDecl(RecordDecl *RD) {
128 VisitTagDecl(RD);
129 RD->setHasFlexibleArrayMember(Record[Idx++]);
130 RD->setAnonymousStructOrUnion(Record[Idx++]);
131}
132
133void PCHDeclReader::VisitValueDecl(ValueDecl *VD) {
134 VisitNamedDecl(VD);
135 VD->setType(Reader.GetType(Record[Idx++]));
136}
137
138void PCHDeclReader::VisitEnumConstantDecl(EnumConstantDecl *ECD) {
139 VisitValueDecl(ECD);
140 if (Record[Idx++])
Chris Lattner3ef21962009-04-27 05:58:23 +0000141 ECD->setInitExpr(Reader.ReadDeclExpr());
Chris Lattner283c0612009-04-27 05:27:42 +0000142 ECD->setInitVal(Reader.ReadAPSInt(Record, Idx));
143}
144
145void PCHDeclReader::VisitFunctionDecl(FunctionDecl *FD) {
146 VisitValueDecl(FD);
147 if (Record[Idx++])
Chris Lattner3ef21962009-04-27 05:58:23 +0000148 FD->setLazyBody(Reader.getDeclsCursor().GetCurrentBitNo());
Chris Lattner283c0612009-04-27 05:27:42 +0000149 FD->setPreviousDeclaration(
150 cast_or_null<FunctionDecl>(Reader.GetDecl(Record[Idx++])));
151 FD->setStorageClass((FunctionDecl::StorageClass)Record[Idx++]);
152 FD->setInline(Record[Idx++]);
153 FD->setC99InlineDefinition(Record[Idx++]);
Anders Carlssonf9855602009-05-14 22:15:41 +0000154 FD->setVirtualAsWritten(Record[Idx++]);
Chris Lattner283c0612009-04-27 05:27:42 +0000155 FD->setPure(Record[Idx++]);
Anders Carlssond9d6d502009-05-14 21:46:00 +0000156 FD->setHasInheritedPrototype(Record[Idx++]);
157 FD->setHasWrittenPrototype(Record[Idx++]);
Chris Lattner283c0612009-04-27 05:27:42 +0000158 FD->setDeleted(Record[Idx++]);
159 FD->setTypeSpecStartLoc(SourceLocation::getFromRawEncoding(Record[Idx++]));
Douglas Gregor260059c2009-05-14 21:06:31 +0000160 // FIXME: C++ TemplateOrInstantiation
Chris Lattner283c0612009-04-27 05:27:42 +0000161 unsigned NumParams = Record[Idx++];
162 llvm::SmallVector<ParmVarDecl *, 16> Params;
163 Params.reserve(NumParams);
164 for (unsigned I = 0; I != NumParams; ++I)
165 Params.push_back(cast<ParmVarDecl>(Reader.GetDecl(Record[Idx++])));
Jay Foad9e6bef42009-05-21 09:52:38 +0000166 FD->setParams(*Reader.getContext(), Params.data(), NumParams);
Chris Lattner283c0612009-04-27 05:27:42 +0000167}
168
169void PCHDeclReader::VisitObjCMethodDecl(ObjCMethodDecl *MD) {
170 VisitNamedDecl(MD);
171 if (Record[Idx++]) {
172 // In practice, this won't be executed (since method definitions
173 // don't occur in header files).
Chris Lattner3ef21962009-04-27 05:58:23 +0000174 MD->setBody(Reader.ReadDeclStmt());
Chris Lattner283c0612009-04-27 05:27:42 +0000175 MD->setSelfDecl(cast<ImplicitParamDecl>(Reader.GetDecl(Record[Idx++])));
176 MD->setCmdDecl(cast<ImplicitParamDecl>(Reader.GetDecl(Record[Idx++])));
177 }
178 MD->setInstanceMethod(Record[Idx++]);
179 MD->setVariadic(Record[Idx++]);
180 MD->setSynthesized(Record[Idx++]);
181 MD->setDeclImplementation((ObjCMethodDecl::ImplementationControl)Record[Idx++]);
182 MD->setObjCDeclQualifier((Decl::ObjCDeclQualifier)Record[Idx++]);
183 MD->setResultType(Reader.GetType(Record[Idx++]));
184 MD->setEndLoc(SourceLocation::getFromRawEncoding(Record[Idx++]));
185 unsigned NumParams = Record[Idx++];
186 llvm::SmallVector<ParmVarDecl *, 16> Params;
187 Params.reserve(NumParams);
188 for (unsigned I = 0; I != NumParams; ++I)
189 Params.push_back(cast<ParmVarDecl>(Reader.GetDecl(Record[Idx++])));
Jay Foad9e6bef42009-05-21 09:52:38 +0000190 MD->setMethodParams(*Reader.getContext(), Params.data(), NumParams);
Chris Lattner283c0612009-04-27 05:27:42 +0000191}
192
193void PCHDeclReader::VisitObjCContainerDecl(ObjCContainerDecl *CD) {
194 VisitNamedDecl(CD);
195 CD->setAtEndLoc(SourceLocation::getFromRawEncoding(Record[Idx++]));
196}
197
198void PCHDeclReader::VisitObjCInterfaceDecl(ObjCInterfaceDecl *ID) {
199 VisitObjCContainerDecl(ID);
200 ID->setTypeForDecl(Reader.GetType(Record[Idx++]).getTypePtr());
201 ID->setSuperClass(cast_or_null<ObjCInterfaceDecl>
202 (Reader.GetDecl(Record[Idx++])));
203 unsigned NumProtocols = Record[Idx++];
204 llvm::SmallVector<ObjCProtocolDecl *, 16> Protocols;
205 Protocols.reserve(NumProtocols);
206 for (unsigned I = 0; I != NumProtocols; ++I)
207 Protocols.push_back(cast<ObjCProtocolDecl>(Reader.GetDecl(Record[Idx++])));
Jay Foad9e6bef42009-05-21 09:52:38 +0000208 ID->setProtocolList(Protocols.data(), NumProtocols, *Reader.getContext());
Chris Lattner283c0612009-04-27 05:27:42 +0000209 unsigned NumIvars = Record[Idx++];
210 llvm::SmallVector<ObjCIvarDecl *, 16> IVars;
211 IVars.reserve(NumIvars);
212 for (unsigned I = 0; I != NumIvars; ++I)
213 IVars.push_back(cast<ObjCIvarDecl>(Reader.GetDecl(Record[Idx++])));
Jay Foad9e6bef42009-05-21 09:52:38 +0000214 ID->setIVarList(IVars.data(), NumIvars, *Reader.getContext());
Chris Lattner283c0612009-04-27 05:27:42 +0000215 ID->setCategoryList(
216 cast_or_null<ObjCCategoryDecl>(Reader.GetDecl(Record[Idx++])));
217 ID->setForwardDecl(Record[Idx++]);
218 ID->setImplicitInterfaceDecl(Record[Idx++]);
219 ID->setClassLoc(SourceLocation::getFromRawEncoding(Record[Idx++]));
220 ID->setSuperClassLoc(SourceLocation::getFromRawEncoding(Record[Idx++]));
221 ID->setAtEndLoc(SourceLocation::getFromRawEncoding(Record[Idx++]));
222}
223
224void PCHDeclReader::VisitObjCIvarDecl(ObjCIvarDecl *IVD) {
225 VisitFieldDecl(IVD);
226 IVD->setAccessControl((ObjCIvarDecl::AccessControl)Record[Idx++]);
227}
228
229void PCHDeclReader::VisitObjCProtocolDecl(ObjCProtocolDecl *PD) {
230 VisitObjCContainerDecl(PD);
231 PD->setForwardDecl(Record[Idx++]);
232 PD->setLocEnd(SourceLocation::getFromRawEncoding(Record[Idx++]));
233 unsigned NumProtoRefs = Record[Idx++];
234 llvm::SmallVector<ObjCProtocolDecl *, 16> ProtoRefs;
235 ProtoRefs.reserve(NumProtoRefs);
236 for (unsigned I = 0; I != NumProtoRefs; ++I)
237 ProtoRefs.push_back(cast<ObjCProtocolDecl>(Reader.GetDecl(Record[Idx++])));
Jay Foad9e6bef42009-05-21 09:52:38 +0000238 PD->setProtocolList(ProtoRefs.data(), NumProtoRefs, *Reader.getContext());
Chris Lattner283c0612009-04-27 05:27:42 +0000239}
240
241void PCHDeclReader::VisitObjCAtDefsFieldDecl(ObjCAtDefsFieldDecl *FD) {
242 VisitFieldDecl(FD);
243}
244
245void PCHDeclReader::VisitObjCClassDecl(ObjCClassDecl *CD) {
246 VisitDecl(CD);
247 unsigned NumClassRefs = Record[Idx++];
248 llvm::SmallVector<ObjCInterfaceDecl *, 16> ClassRefs;
249 ClassRefs.reserve(NumClassRefs);
250 for (unsigned I = 0; I != NumClassRefs; ++I)
251 ClassRefs.push_back(cast<ObjCInterfaceDecl>(Reader.GetDecl(Record[Idx++])));
Douglas Gregor4e284192009-05-22 22:45:36 +0000252 CD->setClassList(*Reader.getContext(), ClassRefs.data(), NumClassRefs);
Chris Lattner283c0612009-04-27 05:27:42 +0000253}
254
255void PCHDeclReader::VisitObjCForwardProtocolDecl(ObjCForwardProtocolDecl *FPD) {
256 VisitDecl(FPD);
257 unsigned NumProtoRefs = Record[Idx++];
258 llvm::SmallVector<ObjCProtocolDecl *, 16> ProtoRefs;
259 ProtoRefs.reserve(NumProtoRefs);
260 for (unsigned I = 0; I != NumProtoRefs; ++I)
261 ProtoRefs.push_back(cast<ObjCProtocolDecl>(Reader.GetDecl(Record[Idx++])));
Douglas Gregor4e284192009-05-22 22:45:36 +0000262 FPD->setProtocolList(ProtoRefs.data(), NumProtoRefs, *Reader.getContext());
Chris Lattner283c0612009-04-27 05:27:42 +0000263}
264
265void PCHDeclReader::VisitObjCCategoryDecl(ObjCCategoryDecl *CD) {
266 VisitObjCContainerDecl(CD);
267 CD->setClassInterface(cast<ObjCInterfaceDecl>(Reader.GetDecl(Record[Idx++])));
268 unsigned NumProtoRefs = Record[Idx++];
269 llvm::SmallVector<ObjCProtocolDecl *, 16> ProtoRefs;
270 ProtoRefs.reserve(NumProtoRefs);
271 for (unsigned I = 0; I != NumProtoRefs; ++I)
272 ProtoRefs.push_back(cast<ObjCProtocolDecl>(Reader.GetDecl(Record[Idx++])));
Ted Kremenek1b38e642009-05-22 22:34:23 +0000273 CD->setProtocolList(ProtoRefs.data(), NumProtoRefs, *Reader.getContext());
Chris Lattner283c0612009-04-27 05:27:42 +0000274 CD->setNextClassCategory(cast_or_null<ObjCCategoryDecl>(Reader.GetDecl(Record[Idx++])));
275 CD->setLocEnd(SourceLocation::getFromRawEncoding(Record[Idx++]));
276}
277
278void PCHDeclReader::VisitObjCCompatibleAliasDecl(ObjCCompatibleAliasDecl *CAD) {
279 VisitNamedDecl(CAD);
280 CAD->setClassInterface(cast<ObjCInterfaceDecl>(Reader.GetDecl(Record[Idx++])));
281}
282
283void PCHDeclReader::VisitObjCPropertyDecl(ObjCPropertyDecl *D) {
284 VisitNamedDecl(D);
285 D->setType(Reader.GetType(Record[Idx++]));
286 // FIXME: stable encoding
287 D->setPropertyAttributes(
288 (ObjCPropertyDecl::PropertyAttributeKind)Record[Idx++]);
289 // FIXME: stable encoding
290 D->setPropertyImplementation(
291 (ObjCPropertyDecl::PropertyControl)Record[Idx++]);
292 D->setGetterName(Reader.ReadDeclarationName(Record, Idx).getObjCSelector());
293 D->setSetterName(Reader.ReadDeclarationName(Record, Idx).getObjCSelector());
294 D->setGetterMethodDecl(
295 cast_or_null<ObjCMethodDecl>(Reader.GetDecl(Record[Idx++])));
296 D->setSetterMethodDecl(
297 cast_or_null<ObjCMethodDecl>(Reader.GetDecl(Record[Idx++])));
298 D->setPropertyIvarDecl(
299 cast_or_null<ObjCIvarDecl>(Reader.GetDecl(Record[Idx++])));
300}
301
302void PCHDeclReader::VisitObjCImplDecl(ObjCImplDecl *D) {
303 VisitNamedDecl(D);
304 D->setClassInterface(
305 cast_or_null<ObjCInterfaceDecl>(Reader.GetDecl(Record[Idx++])));
306 D->setLocEnd(SourceLocation::getFromRawEncoding(Record[Idx++]));
307}
308
309void PCHDeclReader::VisitObjCCategoryImplDecl(ObjCCategoryImplDecl *D) {
310 VisitObjCImplDecl(D);
311 D->setIdentifier(Reader.GetIdentifierInfo(Record, Idx));
312}
313
314void PCHDeclReader::VisitObjCImplementationDecl(ObjCImplementationDecl *D) {
315 VisitObjCImplDecl(D);
316 D->setSuperClass(
317 cast_or_null<ObjCInterfaceDecl>(Reader.GetDecl(Record[Idx++])));
318}
319
320
321void PCHDeclReader::VisitObjCPropertyImplDecl(ObjCPropertyImplDecl *D) {
322 VisitDecl(D);
323 D->setAtLoc(SourceLocation::getFromRawEncoding(Record[Idx++]));
324 D->setPropertyDecl(
325 cast_or_null<ObjCPropertyDecl>(Reader.GetDecl(Record[Idx++])));
326 D->setPropertyIvarDecl(
327 cast_or_null<ObjCIvarDecl>(Reader.GetDecl(Record[Idx++])));
328}
329
330void PCHDeclReader::VisitFieldDecl(FieldDecl *FD) {
331 VisitValueDecl(FD);
332 FD->setMutable(Record[Idx++]);
333 if (Record[Idx++])
Chris Lattner3ef21962009-04-27 05:58:23 +0000334 FD->setBitWidth(Reader.ReadDeclExpr());
Chris Lattner283c0612009-04-27 05:27:42 +0000335}
336
337void PCHDeclReader::VisitVarDecl(VarDecl *VD) {
338 VisitValueDecl(VD);
339 VD->setStorageClass((VarDecl::StorageClass)Record[Idx++]);
340 VD->setThreadSpecified(Record[Idx++]);
341 VD->setCXXDirectInitializer(Record[Idx++]);
342 VD->setDeclaredInCondition(Record[Idx++]);
343 VD->setPreviousDeclaration(
344 cast_or_null<VarDecl>(Reader.GetDecl(Record[Idx++])));
345 VD->setTypeSpecStartLoc(SourceLocation::getFromRawEncoding(Record[Idx++]));
346 if (Record[Idx++])
Douglas Gregor4833ff02009-05-26 18:54:04 +0000347 VD->setInit(*Reader.getContext(), Reader.ReadDeclExpr());
Chris Lattner283c0612009-04-27 05:27:42 +0000348}
349
350void PCHDeclReader::VisitImplicitParamDecl(ImplicitParamDecl *PD) {
351 VisitVarDecl(PD);
352}
353
354void PCHDeclReader::VisitParmVarDecl(ParmVarDecl *PD) {
355 VisitVarDecl(PD);
356 PD->setObjCDeclQualifier((Decl::ObjCDeclQualifier)Record[Idx++]);
357 // FIXME: default argument (C++ only)
358}
359
360void PCHDeclReader::VisitOriginalParmVarDecl(OriginalParmVarDecl *PD) {
361 VisitParmVarDecl(PD);
362 PD->setOriginalType(Reader.GetType(Record[Idx++]));
363}
364
365void PCHDeclReader::VisitFileScopeAsmDecl(FileScopeAsmDecl *AD) {
366 VisitDecl(AD);
Chris Lattner3ef21962009-04-27 05:58:23 +0000367 AD->setAsmString(cast<StringLiteral>(Reader.ReadDeclExpr()));
Chris Lattner283c0612009-04-27 05:27:42 +0000368}
369
370void PCHDeclReader::VisitBlockDecl(BlockDecl *BD) {
371 VisitDecl(BD);
Chris Lattner3ef21962009-04-27 05:58:23 +0000372 BD->setBody(cast_or_null<CompoundStmt>(Reader.ReadDeclStmt()));
Chris Lattner283c0612009-04-27 05:27:42 +0000373 unsigned NumParams = Record[Idx++];
374 llvm::SmallVector<ParmVarDecl *, 16> Params;
375 Params.reserve(NumParams);
376 for (unsigned I = 0; I != NumParams; ++I)
377 Params.push_back(cast<ParmVarDecl>(Reader.GetDecl(Record[Idx++])));
Douglas Gregor4e284192009-05-22 22:45:36 +0000378 BD->setParams(*Reader.getContext(), Params.data(), NumParams);
Chris Lattner283c0612009-04-27 05:27:42 +0000379}
380
381std::pair<uint64_t, uint64_t>
382PCHDeclReader::VisitDeclContext(DeclContext *DC) {
383 uint64_t LexicalOffset = Record[Idx++];
384 uint64_t VisibleOffset = Record[Idx++];
385 return std::make_pair(LexicalOffset, VisibleOffset);
386}
387
388//===----------------------------------------------------------------------===//
Chris Lattner99dccc82009-04-27 06:01:06 +0000389// Attribute Reading
Chris Lattner283c0612009-04-27 05:27:42 +0000390//===----------------------------------------------------------------------===//
391
Chris Lattner99dccc82009-04-27 06:01:06 +0000392/// \brief Reads attributes from the current stream position.
393Attr *PCHReader::ReadAttributes() {
394 unsigned Code = DeclsCursor.ReadCode();
395 assert(Code == llvm::bitc::UNABBREV_RECORD &&
396 "Expected unabbreviated record"); (void)Code;
397
398 RecordData Record;
399 unsigned Idx = 0;
400 unsigned RecCode = DeclsCursor.ReadRecord(Code, Record);
401 assert(RecCode == pch::DECL_ATTR && "Expected attribute record");
402 (void)RecCode;
403
404#define SIMPLE_ATTR(Name) \
405 case Attr::Name: \
Chris Lattner270d29a2009-04-27 21:45:14 +0000406 New = ::new (*Context) Name##Attr(); \
Chris Lattner99dccc82009-04-27 06:01:06 +0000407 break
408
409#define STRING_ATTR(Name) \
410 case Attr::Name: \
Chris Lattner270d29a2009-04-27 21:45:14 +0000411 New = ::new (*Context) Name##Attr(ReadString(Record, Idx)); \
Chris Lattner99dccc82009-04-27 06:01:06 +0000412 break
413
414#define UNSIGNED_ATTR(Name) \
415 case Attr::Name: \
Chris Lattner270d29a2009-04-27 21:45:14 +0000416 New = ::new (*Context) Name##Attr(Record[Idx++]); \
Chris Lattner99dccc82009-04-27 06:01:06 +0000417 break
418
419 Attr *Attrs = 0;
420 while (Idx < Record.size()) {
421 Attr *New = 0;
422 Attr::Kind Kind = (Attr::Kind)Record[Idx++];
423 bool IsInherited = Record[Idx++];
424
425 switch (Kind) {
426 STRING_ATTR(Alias);
427 UNSIGNED_ATTR(Aligned);
428 SIMPLE_ATTR(AlwaysInline);
429 SIMPLE_ATTR(AnalyzerNoReturn);
430 STRING_ATTR(Annotate);
431 STRING_ATTR(AsmLabel);
432
433 case Attr::Blocks:
Chris Lattner270d29a2009-04-27 21:45:14 +0000434 New = ::new (*Context) BlocksAttr(
Chris Lattner99dccc82009-04-27 06:01:06 +0000435 (BlocksAttr::BlocksAttrTypes)Record[Idx++]);
436 break;
437
438 case Attr::Cleanup:
Chris Lattner270d29a2009-04-27 21:45:14 +0000439 New = ::new (*Context) CleanupAttr(
Chris Lattner99dccc82009-04-27 06:01:06 +0000440 cast<FunctionDecl>(GetDecl(Record[Idx++])));
441 break;
442
443 SIMPLE_ATTR(Const);
444 UNSIGNED_ATTR(Constructor);
445 SIMPLE_ATTR(DLLExport);
446 SIMPLE_ATTR(DLLImport);
447 SIMPLE_ATTR(Deprecated);
448 UNSIGNED_ATTR(Destructor);
449 SIMPLE_ATTR(FastCall);
450
451 case Attr::Format: {
452 std::string Type = ReadString(Record, Idx);
453 unsigned FormatIdx = Record[Idx++];
454 unsigned FirstArg = Record[Idx++];
Chris Lattner270d29a2009-04-27 21:45:14 +0000455 New = ::new (*Context) FormatAttr(Type, FormatIdx, FirstArg);
Chris Lattner99dccc82009-04-27 06:01:06 +0000456 break;
457 }
Fariborz Jahanian180f3412009-05-13 18:09:35 +0000458
Fariborz Jahanian306d7252009-05-20 17:41:43 +0000459 case Attr::FormatArg: {
460 unsigned FormatIdx = Record[Idx++];
461 New = ::new (*Context) FormatArgAttr(FormatIdx);
462 break;
463 }
464
Fariborz Jahanian180f3412009-05-13 18:09:35 +0000465 case Attr::Sentinel: {
466 int sentinel = Record[Idx++];
467 int nullPos = Record[Idx++];
468 New = ::new (*Context) SentinelAttr(sentinel, nullPos);
469 break;
470 }
Chris Lattner99dccc82009-04-27 06:01:06 +0000471
472 SIMPLE_ATTR(GNUInline);
473
474 case Attr::IBOutletKind:
Chris Lattner270d29a2009-04-27 21:45:14 +0000475 New = ::new (*Context) IBOutletAttr();
Chris Lattner99dccc82009-04-27 06:01:06 +0000476 break;
477
478 SIMPLE_ATTR(NoReturn);
479 SIMPLE_ATTR(NoThrow);
480 SIMPLE_ATTR(Nodebug);
481 SIMPLE_ATTR(Noinline);
482
483 case Attr::NonNull: {
484 unsigned Size = Record[Idx++];
485 llvm::SmallVector<unsigned, 16> ArgNums;
486 ArgNums.insert(ArgNums.end(), &Record[Idx], &Record[Idx] + Size);
487 Idx += Size;
Douglas Gregor4e284192009-05-22 22:45:36 +0000488 New = ::new (*Context) NonNullAttr(ArgNums.data(), Size);
Chris Lattner99dccc82009-04-27 06:01:06 +0000489 break;
490 }
491
492 SIMPLE_ATTR(ObjCException);
493 SIMPLE_ATTR(ObjCNSObject);
Ted Kremenek13ddd1a2009-05-09 02:44:38 +0000494 SIMPLE_ATTR(CFReturnsRetained);
495 SIMPLE_ATTR(NSReturnsRetained);
Chris Lattner99dccc82009-04-27 06:01:06 +0000496 SIMPLE_ATTR(Overloadable);
497 UNSIGNED_ATTR(Packed);
498 SIMPLE_ATTR(Pure);
499 UNSIGNED_ATTR(Regparm);
500 STRING_ATTR(Section);
501 SIMPLE_ATTR(StdCall);
502 SIMPLE_ATTR(TransparentUnion);
503 SIMPLE_ATTR(Unavailable);
504 SIMPLE_ATTR(Unused);
505 SIMPLE_ATTR(Used);
506
507 case Attr::Visibility:
Chris Lattner270d29a2009-04-27 21:45:14 +0000508 New = ::new (*Context) VisibilityAttr(
Chris Lattner99dccc82009-04-27 06:01:06 +0000509 (VisibilityAttr::VisibilityTypes)Record[Idx++]);
510 break;
511
512 SIMPLE_ATTR(WarnUnusedResult);
513 SIMPLE_ATTR(Weak);
514 SIMPLE_ATTR(WeakImport);
515 }
516
517 assert(New && "Unable to decode attribute?");
518 New->setInherited(IsInherited);
519 New->setNext(Attrs);
520 Attrs = New;
521 }
522#undef UNSIGNED_ATTR
523#undef STRING_ATTR
524#undef SIMPLE_ATTR
525
526 // The list of attributes was built backwards. Reverse the list
527 // before returning it.
528 Attr *PrevAttr = 0, *NextAttr = 0;
529 while (Attrs) {
530 NextAttr = Attrs->getNext();
531 Attrs->setNext(PrevAttr);
532 PrevAttr = Attrs;
533 Attrs = NextAttr;
534 }
535
536 return PrevAttr;
537}
538
539//===----------------------------------------------------------------------===//
540// PCHReader Implementation
541//===----------------------------------------------------------------------===//
Chris Lattner283c0612009-04-27 05:27:42 +0000542
543/// \brief Note that we have loaded the declaration with the given
544/// Index.
545///
546/// This routine notes that this declaration has already been loaded,
547/// so that future GetDecl calls will return this declaration rather
548/// than trying to load a new declaration.
549inline void PCHReader::LoadedDecl(unsigned Index, Decl *D) {
550 assert(!DeclsLoaded[Index] && "Decl loaded twice?");
551 DeclsLoaded[Index] = D;
552}
553
554
555/// \brief Determine whether the consumer will be interested in seeing
556/// this declaration (via HandleTopLevelDecl).
557///
558/// This routine should return true for anything that might affect
559/// code generation, e.g., inline function definitions, Objective-C
560/// declarations with metadata, etc.
561static bool isConsumerInterestedIn(Decl *D) {
562 if (VarDecl *Var = dyn_cast<VarDecl>(D))
563 return Var->isFileVarDecl() && Var->getInit();
564 if (FunctionDecl *Func = dyn_cast<FunctionDecl>(D))
565 return Func->isThisDeclarationADefinition();
566 return isa<ObjCProtocolDecl>(D);
567}
568
569/// \brief Read the declaration at the given offset from the PCH file.
570Decl *PCHReader::ReadDeclRecord(uint64_t Offset, unsigned Index) {
571 // Keep track of where we are in the stream, then jump back there
572 // after reading this declaration.
Chris Lattner3ef21962009-04-27 05:58:23 +0000573 SavedStreamPosition SavedPosition(DeclsCursor);
Chris Lattner283c0612009-04-27 05:27:42 +0000574
Chris Lattner3ef21962009-04-27 05:58:23 +0000575 DeclsCursor.JumpToBit(Offset);
Chris Lattner283c0612009-04-27 05:27:42 +0000576 RecordData Record;
Chris Lattner3ef21962009-04-27 05:58:23 +0000577 unsigned Code = DeclsCursor.ReadCode();
Chris Lattner283c0612009-04-27 05:27:42 +0000578 unsigned Idx = 0;
579 PCHDeclReader Reader(*this, Record, Idx);
580
Chris Lattner3ef21962009-04-27 05:58:23 +0000581 Decl *D = 0;
582 switch ((pch::DeclCode)DeclsCursor.ReadRecord(Code, Record)) {
Chris Lattner283c0612009-04-27 05:27:42 +0000583 case pch::DECL_ATTR:
584 case pch::DECL_CONTEXT_LEXICAL:
585 case pch::DECL_CONTEXT_VISIBLE:
586 assert(false && "Record cannot be de-serialized with ReadDeclRecord");
587 break;
Chris Lattner283c0612009-04-27 05:27:42 +0000588 case pch::DECL_TRANSLATION_UNIT:
589 assert(Index == 0 && "Translation unit must be at index 0");
Chris Lattner270d29a2009-04-27 21:45:14 +0000590 D = Context->getTranslationUnitDecl();
Chris Lattner283c0612009-04-27 05:27:42 +0000591 break;
Chris Lattner99dccc82009-04-27 06:01:06 +0000592 case pch::DECL_TYPEDEF:
Chris Lattner270d29a2009-04-27 21:45:14 +0000593 D = TypedefDecl::Create(*Context, 0, SourceLocation(), 0, QualType());
Chris Lattner283c0612009-04-27 05:27:42 +0000594 break;
Chris Lattner99dccc82009-04-27 06:01:06 +0000595 case pch::DECL_ENUM:
Chris Lattner270d29a2009-04-27 21:45:14 +0000596 D = EnumDecl::Create(*Context, 0, SourceLocation(), 0, 0);
Chris Lattner283c0612009-04-27 05:27:42 +0000597 break;
Chris Lattner99dccc82009-04-27 06:01:06 +0000598 case pch::DECL_RECORD:
Chris Lattner270d29a2009-04-27 21:45:14 +0000599 D = RecordDecl::Create(*Context, TagDecl::TK_struct, 0, SourceLocation(),
Chris Lattner283c0612009-04-27 05:27:42 +0000600 0, 0);
601 break;
Chris Lattner99dccc82009-04-27 06:01:06 +0000602 case pch::DECL_ENUM_CONSTANT:
Chris Lattner270d29a2009-04-27 21:45:14 +0000603 D = EnumConstantDecl::Create(*Context, 0, SourceLocation(), 0, QualType(),
Chris Lattner283c0612009-04-27 05:27:42 +0000604 0, llvm::APSInt());
605 break;
Chris Lattner99dccc82009-04-27 06:01:06 +0000606 case pch::DECL_FUNCTION:
Chris Lattner270d29a2009-04-27 21:45:14 +0000607 D = FunctionDecl::Create(*Context, 0, SourceLocation(), DeclarationName(),
Chris Lattner283c0612009-04-27 05:27:42 +0000608 QualType());
609 break;
Chris Lattner99dccc82009-04-27 06:01:06 +0000610 case pch::DECL_OBJC_METHOD:
Chris Lattner270d29a2009-04-27 21:45:14 +0000611 D = ObjCMethodDecl::Create(*Context, SourceLocation(), SourceLocation(),
Chris Lattner283c0612009-04-27 05:27:42 +0000612 Selector(), QualType(), 0);
613 break;
Chris Lattner99dccc82009-04-27 06:01:06 +0000614 case pch::DECL_OBJC_INTERFACE:
Chris Lattner270d29a2009-04-27 21:45:14 +0000615 D = ObjCInterfaceDecl::Create(*Context, 0, SourceLocation(), 0);
Chris Lattner283c0612009-04-27 05:27:42 +0000616 break;
Chris Lattner99dccc82009-04-27 06:01:06 +0000617 case pch::DECL_OBJC_IVAR:
Chris Lattner270d29a2009-04-27 21:45:14 +0000618 D = ObjCIvarDecl::Create(*Context, 0, SourceLocation(), 0, QualType(),
Chris Lattner283c0612009-04-27 05:27:42 +0000619 ObjCIvarDecl::None);
620 break;
Chris Lattner99dccc82009-04-27 06:01:06 +0000621 case pch::DECL_OBJC_PROTOCOL:
Chris Lattner270d29a2009-04-27 21:45:14 +0000622 D = ObjCProtocolDecl::Create(*Context, 0, SourceLocation(), 0);
Chris Lattner283c0612009-04-27 05:27:42 +0000623 break;
Chris Lattner99dccc82009-04-27 06:01:06 +0000624 case pch::DECL_OBJC_AT_DEFS_FIELD:
Chris Lattner270d29a2009-04-27 21:45:14 +0000625 D = ObjCAtDefsFieldDecl::Create(*Context, 0, SourceLocation(), 0,
Chris Lattner283c0612009-04-27 05:27:42 +0000626 QualType(), 0);
627 break;
Chris Lattner99dccc82009-04-27 06:01:06 +0000628 case pch::DECL_OBJC_CLASS:
Chris Lattner270d29a2009-04-27 21:45:14 +0000629 D = ObjCClassDecl::Create(*Context, 0, SourceLocation());
Chris Lattner283c0612009-04-27 05:27:42 +0000630 break;
Chris Lattner99dccc82009-04-27 06:01:06 +0000631 case pch::DECL_OBJC_FORWARD_PROTOCOL:
Chris Lattner270d29a2009-04-27 21:45:14 +0000632 D = ObjCForwardProtocolDecl::Create(*Context, 0, SourceLocation());
Chris Lattner283c0612009-04-27 05:27:42 +0000633 break;
Chris Lattner99dccc82009-04-27 06:01:06 +0000634 case pch::DECL_OBJC_CATEGORY:
Chris Lattner270d29a2009-04-27 21:45:14 +0000635 D = ObjCCategoryDecl::Create(*Context, 0, SourceLocation(), 0);
Chris Lattner283c0612009-04-27 05:27:42 +0000636 break;
Chris Lattner99dccc82009-04-27 06:01:06 +0000637 case pch::DECL_OBJC_CATEGORY_IMPL:
Chris Lattner270d29a2009-04-27 21:45:14 +0000638 D = ObjCCategoryImplDecl::Create(*Context, 0, SourceLocation(), 0, 0);
Chris Lattner283c0612009-04-27 05:27:42 +0000639 break;
Chris Lattner99dccc82009-04-27 06:01:06 +0000640 case pch::DECL_OBJC_IMPLEMENTATION:
Chris Lattner270d29a2009-04-27 21:45:14 +0000641 D = ObjCImplementationDecl::Create(*Context, 0, SourceLocation(), 0, 0);
Chris Lattner283c0612009-04-27 05:27:42 +0000642 break;
Chris Lattner99dccc82009-04-27 06:01:06 +0000643 case pch::DECL_OBJC_COMPATIBLE_ALIAS:
Chris Lattner270d29a2009-04-27 21:45:14 +0000644 D = ObjCCompatibleAliasDecl::Create(*Context, 0, SourceLocation(), 0, 0);
Chris Lattner283c0612009-04-27 05:27:42 +0000645 break;
Chris Lattner99dccc82009-04-27 06:01:06 +0000646 case pch::DECL_OBJC_PROPERTY:
Chris Lattner270d29a2009-04-27 21:45:14 +0000647 D = ObjCPropertyDecl::Create(*Context, 0, SourceLocation(), 0, QualType());
Chris Lattner283c0612009-04-27 05:27:42 +0000648 break;
Chris Lattner99dccc82009-04-27 06:01:06 +0000649 case pch::DECL_OBJC_PROPERTY_IMPL:
Chris Lattner270d29a2009-04-27 21:45:14 +0000650 D = ObjCPropertyImplDecl::Create(*Context, 0, SourceLocation(),
Chris Lattner283c0612009-04-27 05:27:42 +0000651 SourceLocation(), 0,
652 ObjCPropertyImplDecl::Dynamic, 0);
653 break;
Chris Lattner99dccc82009-04-27 06:01:06 +0000654 case pch::DECL_FIELD:
Chris Lattner270d29a2009-04-27 21:45:14 +0000655 D = FieldDecl::Create(*Context, 0, SourceLocation(), 0, QualType(), 0,
Chris Lattner283c0612009-04-27 05:27:42 +0000656 false);
657 break;
Chris Lattner283c0612009-04-27 05:27:42 +0000658 case pch::DECL_VAR:
Chris Lattner270d29a2009-04-27 21:45:14 +0000659 D = VarDecl::Create(*Context, 0, SourceLocation(), 0, QualType(),
Chris Lattner283c0612009-04-27 05:27:42 +0000660 VarDecl::None, SourceLocation());
661 break;
662
663 case pch::DECL_IMPLICIT_PARAM:
Chris Lattner270d29a2009-04-27 21:45:14 +0000664 D = ImplicitParamDecl::Create(*Context, 0, SourceLocation(), 0, QualType());
Chris Lattner283c0612009-04-27 05:27:42 +0000665 break;
666
Chris Lattner99dccc82009-04-27 06:01:06 +0000667 case pch::DECL_PARM_VAR:
Chris Lattner270d29a2009-04-27 21:45:14 +0000668 D = ParmVarDecl::Create(*Context, 0, SourceLocation(), 0, QualType(),
Chris Lattner283c0612009-04-27 05:27:42 +0000669 VarDecl::None, 0);
670 break;
Chris Lattner99dccc82009-04-27 06:01:06 +0000671 case pch::DECL_ORIGINAL_PARM_VAR:
Chris Lattner270d29a2009-04-27 21:45:14 +0000672 D = OriginalParmVarDecl::Create(*Context, 0, SourceLocation(), 0,
Chris Lattner99dccc82009-04-27 06:01:06 +0000673 QualType(), QualType(), VarDecl::None, 0);
Chris Lattner283c0612009-04-27 05:27:42 +0000674 break;
Chris Lattner99dccc82009-04-27 06:01:06 +0000675 case pch::DECL_FILE_SCOPE_ASM:
Chris Lattner270d29a2009-04-27 21:45:14 +0000676 D = FileScopeAsmDecl::Create(*Context, 0, SourceLocation(), 0);
Chris Lattner283c0612009-04-27 05:27:42 +0000677 break;
Chris Lattner99dccc82009-04-27 06:01:06 +0000678 case pch::DECL_BLOCK:
Chris Lattner270d29a2009-04-27 21:45:14 +0000679 D = BlockDecl::Create(*Context, 0, SourceLocation());
Chris Lattner283c0612009-04-27 05:27:42 +0000680 break;
681 }
Chris Lattner283c0612009-04-27 05:27:42 +0000682
683 assert(D && "Unknown declaration reading PCH file");
Chris Lattner99dccc82009-04-27 06:01:06 +0000684 LoadedDecl(Index, D);
685 Reader.Visit(D);
Chris Lattner283c0612009-04-27 05:27:42 +0000686
687 // If this declaration is also a declaration context, get the
688 // offsets for its tables of lexical and visible declarations.
689 if (DeclContext *DC = dyn_cast<DeclContext>(D)) {
690 std::pair<uint64_t, uint64_t> Offsets = Reader.VisitDeclContext(DC);
691 if (Offsets.first || Offsets.second) {
692 DC->setHasExternalLexicalStorage(Offsets.first != 0);
693 DC->setHasExternalVisibleStorage(Offsets.second != 0);
694 DeclContextOffsets[DC] = Offsets;
695 }
696 }
697 assert(Idx == Record.size());
698
699 // If we have deserialized a declaration that has a definition the
700 // AST consumer might need to know about, notify the consumer
701 // about that definition now or queue it for later.
702 if (isConsumerInterestedIn(D)) {
703 if (Consumer) {
704 DeclGroupRef DG(D);
705 Consumer->HandleTopLevelDecl(DG);
706 } else {
707 InterestingDecls.push_back(D);
708 }
709 }
710
711 return D;
712}
713