blob: c279a217139026d4e267576f044b071ed28f624d [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
23// FIXME: Temporarily cloned from PCHReader.cpp.
24namespace {
25 /// \brief Helper class that saves the current stream position and
26 /// then restores it when destroyed.
27 struct SavedStreamPosition {
28 explicit SavedStreamPosition(llvm::BitstreamCursor &Cursor)
29 : Cursor(Cursor), Offset(Cursor.GetCurrentBitNo()) { }
30
31 ~SavedStreamPosition() {
32 Cursor.JumpToBit(Offset);
33 }
34
35 private:
36 llvm::BitstreamCursor &Cursor;
37 uint64_t Offset;
38 };
39}
40
41
42//===----------------------------------------------------------------------===//
43// Declaration deserialization
44//===----------------------------------------------------------------------===//
45
46namespace {
47 class PCHDeclReader : public DeclVisitor<PCHDeclReader, void> {
48 PCHReader &Reader;
49 const PCHReader::RecordData &Record;
50 unsigned &Idx;
51
52 public:
53 PCHDeclReader(PCHReader &Reader, const PCHReader::RecordData &Record,
54 unsigned &Idx)
55 : Reader(Reader), Record(Record), Idx(Idx) { }
56
57 void VisitDecl(Decl *D);
58 void VisitTranslationUnitDecl(TranslationUnitDecl *TU);
59 void VisitNamedDecl(NamedDecl *ND);
60 void VisitTypeDecl(TypeDecl *TD);
61 void VisitTypedefDecl(TypedefDecl *TD);
62 void VisitTagDecl(TagDecl *TD);
63 void VisitEnumDecl(EnumDecl *ED);
64 void VisitRecordDecl(RecordDecl *RD);
65 void VisitValueDecl(ValueDecl *VD);
66 void VisitEnumConstantDecl(EnumConstantDecl *ECD);
67 void VisitFunctionDecl(FunctionDecl *FD);
68 void VisitFieldDecl(FieldDecl *FD);
69 void VisitVarDecl(VarDecl *VD);
70 void VisitImplicitParamDecl(ImplicitParamDecl *PD);
71 void VisitParmVarDecl(ParmVarDecl *PD);
72 void VisitOriginalParmVarDecl(OriginalParmVarDecl *PD);
73 void VisitFileScopeAsmDecl(FileScopeAsmDecl *AD);
74 void VisitBlockDecl(BlockDecl *BD);
75 std::pair<uint64_t, uint64_t> VisitDeclContext(DeclContext *DC);
76 void VisitObjCMethodDecl(ObjCMethodDecl *D);
77 void VisitObjCContainerDecl(ObjCContainerDecl *D);
78 void VisitObjCInterfaceDecl(ObjCInterfaceDecl *D);
79 void VisitObjCIvarDecl(ObjCIvarDecl *D);
80 void VisitObjCProtocolDecl(ObjCProtocolDecl *D);
81 void VisitObjCAtDefsFieldDecl(ObjCAtDefsFieldDecl *D);
82 void VisitObjCClassDecl(ObjCClassDecl *D);
83 void VisitObjCForwardProtocolDecl(ObjCForwardProtocolDecl *D);
84 void VisitObjCCategoryDecl(ObjCCategoryDecl *D);
85 void VisitObjCImplDecl(ObjCImplDecl *D);
86 void VisitObjCCategoryImplDecl(ObjCCategoryImplDecl *D);
87 void VisitObjCImplementationDecl(ObjCImplementationDecl *D);
88 void VisitObjCCompatibleAliasDecl(ObjCCompatibleAliasDecl *D);
89 void VisitObjCPropertyDecl(ObjCPropertyDecl *D);
90 void VisitObjCPropertyImplDecl(ObjCPropertyImplDecl *D);
91 };
92}
93
94void PCHDeclReader::VisitDecl(Decl *D) {
95 D->setDeclContext(cast_or_null<DeclContext>(Reader.GetDecl(Record[Idx++])));
96 D->setLexicalDeclContext(
97 cast_or_null<DeclContext>(Reader.GetDecl(Record[Idx++])));
98 D->setLocation(SourceLocation::getFromRawEncoding(Record[Idx++]));
99 D->setInvalidDecl(Record[Idx++]);
100 if (Record[Idx++])
101 D->addAttr(Reader.ReadAttributes());
102 D->setImplicit(Record[Idx++]);
103 D->setAccess((AccessSpecifier)Record[Idx++]);
104}
105
106void PCHDeclReader::VisitTranslationUnitDecl(TranslationUnitDecl *TU) {
107 VisitDecl(TU);
108}
109
110void PCHDeclReader::VisitNamedDecl(NamedDecl *ND) {
111 VisitDecl(ND);
112 ND->setDeclName(Reader.ReadDeclarationName(Record, Idx));
113}
114
115void PCHDeclReader::VisitTypeDecl(TypeDecl *TD) {
116 VisitNamedDecl(TD);
117 TD->setTypeForDecl(Reader.GetType(Record[Idx++]).getTypePtr());
118}
119
120void PCHDeclReader::VisitTypedefDecl(TypedefDecl *TD) {
121 // Note that we cannot use VisitTypeDecl here, because we need to
122 // set the underlying type of the typedef *before* we try to read
123 // the type associated with the TypedefDecl.
124 VisitNamedDecl(TD);
125 TD->setUnderlyingType(Reader.GetType(Record[Idx + 1]));
126 TD->setTypeForDecl(Reader.GetType(Record[Idx]).getTypePtr());
127 Idx += 2;
128}
129
130void PCHDeclReader::VisitTagDecl(TagDecl *TD) {
131 VisitTypeDecl(TD);
132 TD->setTagKind((TagDecl::TagKind)Record[Idx++]);
133 TD->setDefinition(Record[Idx++]);
134 TD->setTypedefForAnonDecl(
135 cast_or_null<TypedefDecl>(Reader.GetDecl(Record[Idx++])));
136}
137
138void PCHDeclReader::VisitEnumDecl(EnumDecl *ED) {
139 VisitTagDecl(ED);
140 ED->setIntegerType(Reader.GetType(Record[Idx++]));
141}
142
143void PCHDeclReader::VisitRecordDecl(RecordDecl *RD) {
144 VisitTagDecl(RD);
145 RD->setHasFlexibleArrayMember(Record[Idx++]);
146 RD->setAnonymousStructOrUnion(Record[Idx++]);
147}
148
149void PCHDeclReader::VisitValueDecl(ValueDecl *VD) {
150 VisitNamedDecl(VD);
151 VD->setType(Reader.GetType(Record[Idx++]));
152}
153
154void PCHDeclReader::VisitEnumConstantDecl(EnumConstantDecl *ECD) {
155 VisitValueDecl(ECD);
156 if (Record[Idx++])
Chris Lattner3ef21962009-04-27 05:58:23 +0000157 ECD->setInitExpr(Reader.ReadDeclExpr());
Chris Lattner283c0612009-04-27 05:27:42 +0000158 ECD->setInitVal(Reader.ReadAPSInt(Record, Idx));
159}
160
161void PCHDeclReader::VisitFunctionDecl(FunctionDecl *FD) {
162 VisitValueDecl(FD);
163 if (Record[Idx++])
Chris Lattner3ef21962009-04-27 05:58:23 +0000164 FD->setLazyBody(Reader.getDeclsCursor().GetCurrentBitNo());
Chris Lattner283c0612009-04-27 05:27:42 +0000165 FD->setPreviousDeclaration(
166 cast_or_null<FunctionDecl>(Reader.GetDecl(Record[Idx++])));
167 FD->setStorageClass((FunctionDecl::StorageClass)Record[Idx++]);
168 FD->setInline(Record[Idx++]);
169 FD->setC99InlineDefinition(Record[Idx++]);
170 FD->setVirtual(Record[Idx++]);
171 FD->setPure(Record[Idx++]);
172 FD->setInheritedPrototype(Record[Idx++]);
173 FD->setHasPrototype(Record[Idx++]);
174 FD->setDeleted(Record[Idx++]);
175 FD->setTypeSpecStartLoc(SourceLocation::getFromRawEncoding(Record[Idx++]));
176 unsigned NumParams = Record[Idx++];
177 llvm::SmallVector<ParmVarDecl *, 16> Params;
178 Params.reserve(NumParams);
179 for (unsigned I = 0; I != NumParams; ++I)
180 Params.push_back(cast<ParmVarDecl>(Reader.GetDecl(Record[Idx++])));
181 FD->setParams(Reader.getContext(), &Params[0], NumParams);
182}
183
184void PCHDeclReader::VisitObjCMethodDecl(ObjCMethodDecl *MD) {
185 VisitNamedDecl(MD);
186 if (Record[Idx++]) {
187 // In practice, this won't be executed (since method definitions
188 // don't occur in header files).
Chris Lattner3ef21962009-04-27 05:58:23 +0000189 MD->setBody(Reader.ReadDeclStmt());
Chris Lattner283c0612009-04-27 05:27:42 +0000190 MD->setSelfDecl(cast<ImplicitParamDecl>(Reader.GetDecl(Record[Idx++])));
191 MD->setCmdDecl(cast<ImplicitParamDecl>(Reader.GetDecl(Record[Idx++])));
192 }
193 MD->setInstanceMethod(Record[Idx++]);
194 MD->setVariadic(Record[Idx++]);
195 MD->setSynthesized(Record[Idx++]);
196 MD->setDeclImplementation((ObjCMethodDecl::ImplementationControl)Record[Idx++]);
197 MD->setObjCDeclQualifier((Decl::ObjCDeclQualifier)Record[Idx++]);
198 MD->setResultType(Reader.GetType(Record[Idx++]));
199 MD->setEndLoc(SourceLocation::getFromRawEncoding(Record[Idx++]));
200 unsigned NumParams = Record[Idx++];
201 llvm::SmallVector<ParmVarDecl *, 16> Params;
202 Params.reserve(NumParams);
203 for (unsigned I = 0; I != NumParams; ++I)
204 Params.push_back(cast<ParmVarDecl>(Reader.GetDecl(Record[Idx++])));
205 MD->setMethodParams(Reader.getContext(), &Params[0], NumParams);
206}
207
208void PCHDeclReader::VisitObjCContainerDecl(ObjCContainerDecl *CD) {
209 VisitNamedDecl(CD);
210 CD->setAtEndLoc(SourceLocation::getFromRawEncoding(Record[Idx++]));
211}
212
213void PCHDeclReader::VisitObjCInterfaceDecl(ObjCInterfaceDecl *ID) {
214 VisitObjCContainerDecl(ID);
215 ID->setTypeForDecl(Reader.GetType(Record[Idx++]).getTypePtr());
216 ID->setSuperClass(cast_or_null<ObjCInterfaceDecl>
217 (Reader.GetDecl(Record[Idx++])));
218 unsigned NumProtocols = Record[Idx++];
219 llvm::SmallVector<ObjCProtocolDecl *, 16> Protocols;
220 Protocols.reserve(NumProtocols);
221 for (unsigned I = 0; I != NumProtocols; ++I)
222 Protocols.push_back(cast<ObjCProtocolDecl>(Reader.GetDecl(Record[Idx++])));
223 ID->setProtocolList(&Protocols[0], NumProtocols, Reader.getContext());
224 unsigned NumIvars = Record[Idx++];
225 llvm::SmallVector<ObjCIvarDecl *, 16> IVars;
226 IVars.reserve(NumIvars);
227 for (unsigned I = 0; I != NumIvars; ++I)
228 IVars.push_back(cast<ObjCIvarDecl>(Reader.GetDecl(Record[Idx++])));
229 ID->setIVarList(&IVars[0], NumIvars, Reader.getContext());
230 ID->setCategoryList(
231 cast_or_null<ObjCCategoryDecl>(Reader.GetDecl(Record[Idx++])));
232 ID->setForwardDecl(Record[Idx++]);
233 ID->setImplicitInterfaceDecl(Record[Idx++]);
234 ID->setClassLoc(SourceLocation::getFromRawEncoding(Record[Idx++]));
235 ID->setSuperClassLoc(SourceLocation::getFromRawEncoding(Record[Idx++]));
236 ID->setAtEndLoc(SourceLocation::getFromRawEncoding(Record[Idx++]));
237}
238
239void PCHDeclReader::VisitObjCIvarDecl(ObjCIvarDecl *IVD) {
240 VisitFieldDecl(IVD);
241 IVD->setAccessControl((ObjCIvarDecl::AccessControl)Record[Idx++]);
242}
243
244void PCHDeclReader::VisitObjCProtocolDecl(ObjCProtocolDecl *PD) {
245 VisitObjCContainerDecl(PD);
246 PD->setForwardDecl(Record[Idx++]);
247 PD->setLocEnd(SourceLocation::getFromRawEncoding(Record[Idx++]));
248 unsigned NumProtoRefs = Record[Idx++];
249 llvm::SmallVector<ObjCProtocolDecl *, 16> ProtoRefs;
250 ProtoRefs.reserve(NumProtoRefs);
251 for (unsigned I = 0; I != NumProtoRefs; ++I)
252 ProtoRefs.push_back(cast<ObjCProtocolDecl>(Reader.GetDecl(Record[Idx++])));
253 PD->setProtocolList(&ProtoRefs[0], NumProtoRefs, Reader.getContext());
254}
255
256void PCHDeclReader::VisitObjCAtDefsFieldDecl(ObjCAtDefsFieldDecl *FD) {
257 VisitFieldDecl(FD);
258}
259
260void PCHDeclReader::VisitObjCClassDecl(ObjCClassDecl *CD) {
261 VisitDecl(CD);
262 unsigned NumClassRefs = Record[Idx++];
263 llvm::SmallVector<ObjCInterfaceDecl *, 16> ClassRefs;
264 ClassRefs.reserve(NumClassRefs);
265 for (unsigned I = 0; I != NumClassRefs; ++I)
266 ClassRefs.push_back(cast<ObjCInterfaceDecl>(Reader.GetDecl(Record[Idx++])));
267 CD->setClassList(Reader.getContext(), &ClassRefs[0], NumClassRefs);
268}
269
270void PCHDeclReader::VisitObjCForwardProtocolDecl(ObjCForwardProtocolDecl *FPD) {
271 VisitDecl(FPD);
272 unsigned NumProtoRefs = Record[Idx++];
273 llvm::SmallVector<ObjCProtocolDecl *, 16> ProtoRefs;
274 ProtoRefs.reserve(NumProtoRefs);
275 for (unsigned I = 0; I != NumProtoRefs; ++I)
276 ProtoRefs.push_back(cast<ObjCProtocolDecl>(Reader.GetDecl(Record[Idx++])));
277 FPD->setProtocolList(&ProtoRefs[0], NumProtoRefs, Reader.getContext());
278}
279
280void PCHDeclReader::VisitObjCCategoryDecl(ObjCCategoryDecl *CD) {
281 VisitObjCContainerDecl(CD);
282 CD->setClassInterface(cast<ObjCInterfaceDecl>(Reader.GetDecl(Record[Idx++])));
283 unsigned NumProtoRefs = Record[Idx++];
284 llvm::SmallVector<ObjCProtocolDecl *, 16> ProtoRefs;
285 ProtoRefs.reserve(NumProtoRefs);
286 for (unsigned I = 0; I != NumProtoRefs; ++I)
287 ProtoRefs.push_back(cast<ObjCProtocolDecl>(Reader.GetDecl(Record[Idx++])));
288 CD->setProtocolList(&ProtoRefs[0], NumProtoRefs, Reader.getContext());
289 CD->setNextClassCategory(cast_or_null<ObjCCategoryDecl>(Reader.GetDecl(Record[Idx++])));
290 CD->setLocEnd(SourceLocation::getFromRawEncoding(Record[Idx++]));
291}
292
293void PCHDeclReader::VisitObjCCompatibleAliasDecl(ObjCCompatibleAliasDecl *CAD) {
294 VisitNamedDecl(CAD);
295 CAD->setClassInterface(cast<ObjCInterfaceDecl>(Reader.GetDecl(Record[Idx++])));
296}
297
298void PCHDeclReader::VisitObjCPropertyDecl(ObjCPropertyDecl *D) {
299 VisitNamedDecl(D);
300 D->setType(Reader.GetType(Record[Idx++]));
301 // FIXME: stable encoding
302 D->setPropertyAttributes(
303 (ObjCPropertyDecl::PropertyAttributeKind)Record[Idx++]);
304 // FIXME: stable encoding
305 D->setPropertyImplementation(
306 (ObjCPropertyDecl::PropertyControl)Record[Idx++]);
307 D->setGetterName(Reader.ReadDeclarationName(Record, Idx).getObjCSelector());
308 D->setSetterName(Reader.ReadDeclarationName(Record, Idx).getObjCSelector());
309 D->setGetterMethodDecl(
310 cast_or_null<ObjCMethodDecl>(Reader.GetDecl(Record[Idx++])));
311 D->setSetterMethodDecl(
312 cast_or_null<ObjCMethodDecl>(Reader.GetDecl(Record[Idx++])));
313 D->setPropertyIvarDecl(
314 cast_or_null<ObjCIvarDecl>(Reader.GetDecl(Record[Idx++])));
315}
316
317void PCHDeclReader::VisitObjCImplDecl(ObjCImplDecl *D) {
318 VisitNamedDecl(D);
319 D->setClassInterface(
320 cast_or_null<ObjCInterfaceDecl>(Reader.GetDecl(Record[Idx++])));
321 D->setLocEnd(SourceLocation::getFromRawEncoding(Record[Idx++]));
322}
323
324void PCHDeclReader::VisitObjCCategoryImplDecl(ObjCCategoryImplDecl *D) {
325 VisitObjCImplDecl(D);
326 D->setIdentifier(Reader.GetIdentifierInfo(Record, Idx));
327}
328
329void PCHDeclReader::VisitObjCImplementationDecl(ObjCImplementationDecl *D) {
330 VisitObjCImplDecl(D);
331 D->setSuperClass(
332 cast_or_null<ObjCInterfaceDecl>(Reader.GetDecl(Record[Idx++])));
333}
334
335
336void PCHDeclReader::VisitObjCPropertyImplDecl(ObjCPropertyImplDecl *D) {
337 VisitDecl(D);
338 D->setAtLoc(SourceLocation::getFromRawEncoding(Record[Idx++]));
339 D->setPropertyDecl(
340 cast_or_null<ObjCPropertyDecl>(Reader.GetDecl(Record[Idx++])));
341 D->setPropertyIvarDecl(
342 cast_or_null<ObjCIvarDecl>(Reader.GetDecl(Record[Idx++])));
343}
344
345void PCHDeclReader::VisitFieldDecl(FieldDecl *FD) {
346 VisitValueDecl(FD);
347 FD->setMutable(Record[Idx++]);
348 if (Record[Idx++])
Chris Lattner3ef21962009-04-27 05:58:23 +0000349 FD->setBitWidth(Reader.ReadDeclExpr());
Chris Lattner283c0612009-04-27 05:27:42 +0000350}
351
352void PCHDeclReader::VisitVarDecl(VarDecl *VD) {
353 VisitValueDecl(VD);
354 VD->setStorageClass((VarDecl::StorageClass)Record[Idx++]);
355 VD->setThreadSpecified(Record[Idx++]);
356 VD->setCXXDirectInitializer(Record[Idx++]);
357 VD->setDeclaredInCondition(Record[Idx++]);
358 VD->setPreviousDeclaration(
359 cast_or_null<VarDecl>(Reader.GetDecl(Record[Idx++])));
360 VD->setTypeSpecStartLoc(SourceLocation::getFromRawEncoding(Record[Idx++]));
361 if (Record[Idx++])
Chris Lattner3ef21962009-04-27 05:58:23 +0000362 VD->setInit(Reader.ReadDeclExpr());
Chris Lattner283c0612009-04-27 05:27:42 +0000363}
364
365void PCHDeclReader::VisitImplicitParamDecl(ImplicitParamDecl *PD) {
366 VisitVarDecl(PD);
367}
368
369void PCHDeclReader::VisitParmVarDecl(ParmVarDecl *PD) {
370 VisitVarDecl(PD);
371 PD->setObjCDeclQualifier((Decl::ObjCDeclQualifier)Record[Idx++]);
372 // FIXME: default argument (C++ only)
373}
374
375void PCHDeclReader::VisitOriginalParmVarDecl(OriginalParmVarDecl *PD) {
376 VisitParmVarDecl(PD);
377 PD->setOriginalType(Reader.GetType(Record[Idx++]));
378}
379
380void PCHDeclReader::VisitFileScopeAsmDecl(FileScopeAsmDecl *AD) {
381 VisitDecl(AD);
Chris Lattner3ef21962009-04-27 05:58:23 +0000382 AD->setAsmString(cast<StringLiteral>(Reader.ReadDeclExpr()));
Chris Lattner283c0612009-04-27 05:27:42 +0000383}
384
385void PCHDeclReader::VisitBlockDecl(BlockDecl *BD) {
386 VisitDecl(BD);
Chris Lattner3ef21962009-04-27 05:58:23 +0000387 BD->setBody(cast_or_null<CompoundStmt>(Reader.ReadDeclStmt()));
Chris Lattner283c0612009-04-27 05:27:42 +0000388 unsigned NumParams = Record[Idx++];
389 llvm::SmallVector<ParmVarDecl *, 16> Params;
390 Params.reserve(NumParams);
391 for (unsigned I = 0; I != NumParams; ++I)
392 Params.push_back(cast<ParmVarDecl>(Reader.GetDecl(Record[Idx++])));
393 BD->setParams(Reader.getContext(), &Params[0], NumParams);
394}
395
396std::pair<uint64_t, uint64_t>
397PCHDeclReader::VisitDeclContext(DeclContext *DC) {
398 uint64_t LexicalOffset = Record[Idx++];
399 uint64_t VisibleOffset = Record[Idx++];
400 return std::make_pair(LexicalOffset, VisibleOffset);
401}
402
403//===----------------------------------------------------------------------===//
Chris Lattner99dccc82009-04-27 06:01:06 +0000404// Attribute Reading
Chris Lattner283c0612009-04-27 05:27:42 +0000405//===----------------------------------------------------------------------===//
406
Chris Lattner99dccc82009-04-27 06:01:06 +0000407/// \brief Reads attributes from the current stream position.
408Attr *PCHReader::ReadAttributes() {
409 unsigned Code = DeclsCursor.ReadCode();
410 assert(Code == llvm::bitc::UNABBREV_RECORD &&
411 "Expected unabbreviated record"); (void)Code;
412
413 RecordData Record;
414 unsigned Idx = 0;
415 unsigned RecCode = DeclsCursor.ReadRecord(Code, Record);
416 assert(RecCode == pch::DECL_ATTR && "Expected attribute record");
417 (void)RecCode;
418
419#define SIMPLE_ATTR(Name) \
420 case Attr::Name: \
421 New = ::new (Context) Name##Attr(); \
422 break
423
424#define STRING_ATTR(Name) \
425 case Attr::Name: \
426 New = ::new (Context) Name##Attr(ReadString(Record, Idx)); \
427 break
428
429#define UNSIGNED_ATTR(Name) \
430 case Attr::Name: \
431 New = ::new (Context) Name##Attr(Record[Idx++]); \
432 break
433
434 Attr *Attrs = 0;
435 while (Idx < Record.size()) {
436 Attr *New = 0;
437 Attr::Kind Kind = (Attr::Kind)Record[Idx++];
438 bool IsInherited = Record[Idx++];
439
440 switch (Kind) {
441 STRING_ATTR(Alias);
442 UNSIGNED_ATTR(Aligned);
443 SIMPLE_ATTR(AlwaysInline);
444 SIMPLE_ATTR(AnalyzerNoReturn);
445 STRING_ATTR(Annotate);
446 STRING_ATTR(AsmLabel);
447
448 case Attr::Blocks:
449 New = ::new (Context) BlocksAttr(
450 (BlocksAttr::BlocksAttrTypes)Record[Idx++]);
451 break;
452
453 case Attr::Cleanup:
454 New = ::new (Context) CleanupAttr(
455 cast<FunctionDecl>(GetDecl(Record[Idx++])));
456 break;
457
458 SIMPLE_ATTR(Const);
459 UNSIGNED_ATTR(Constructor);
460 SIMPLE_ATTR(DLLExport);
461 SIMPLE_ATTR(DLLImport);
462 SIMPLE_ATTR(Deprecated);
463 UNSIGNED_ATTR(Destructor);
464 SIMPLE_ATTR(FastCall);
465
466 case Attr::Format: {
467 std::string Type = ReadString(Record, Idx);
468 unsigned FormatIdx = Record[Idx++];
469 unsigned FirstArg = Record[Idx++];
470 New = ::new (Context) FormatAttr(Type, FormatIdx, FirstArg);
471 break;
472 }
473
474 SIMPLE_ATTR(GNUInline);
475
476 case Attr::IBOutletKind:
477 New = ::new (Context) IBOutletAttr();
478 break;
479
480 SIMPLE_ATTR(NoReturn);
481 SIMPLE_ATTR(NoThrow);
482 SIMPLE_ATTR(Nodebug);
483 SIMPLE_ATTR(Noinline);
484
485 case Attr::NonNull: {
486 unsigned Size = Record[Idx++];
487 llvm::SmallVector<unsigned, 16> ArgNums;
488 ArgNums.insert(ArgNums.end(), &Record[Idx], &Record[Idx] + Size);
489 Idx += Size;
490 New = ::new (Context) NonNullAttr(&ArgNums[0], Size);
491 break;
492 }
493
494 SIMPLE_ATTR(ObjCException);
495 SIMPLE_ATTR(ObjCNSObject);
496 SIMPLE_ATTR(ObjCOwnershipRetain);
497 SIMPLE_ATTR(ObjCOwnershipReturns);
498 SIMPLE_ATTR(Overloadable);
499 UNSIGNED_ATTR(Packed);
500 SIMPLE_ATTR(Pure);
501 UNSIGNED_ATTR(Regparm);
502 STRING_ATTR(Section);
503 SIMPLE_ATTR(StdCall);
504 SIMPLE_ATTR(TransparentUnion);
505 SIMPLE_ATTR(Unavailable);
506 SIMPLE_ATTR(Unused);
507 SIMPLE_ATTR(Used);
508
509 case Attr::Visibility:
510 New = ::new (Context) VisibilityAttr(
511 (VisibilityAttr::VisibilityTypes)Record[Idx++]);
512 break;
513
514 SIMPLE_ATTR(WarnUnusedResult);
515 SIMPLE_ATTR(Weak);
516 SIMPLE_ATTR(WeakImport);
517 }
518
519 assert(New && "Unable to decode attribute?");
520 New->setInherited(IsInherited);
521 New->setNext(Attrs);
522 Attrs = New;
523 }
524#undef UNSIGNED_ATTR
525#undef STRING_ATTR
526#undef SIMPLE_ATTR
527
528 // The list of attributes was built backwards. Reverse the list
529 // before returning it.
530 Attr *PrevAttr = 0, *NextAttr = 0;
531 while (Attrs) {
532 NextAttr = Attrs->getNext();
533 Attrs->setNext(PrevAttr);
534 PrevAttr = Attrs;
535 Attrs = NextAttr;
536 }
537
538 return PrevAttr;
539}
540
541//===----------------------------------------------------------------------===//
542// PCHReader Implementation
543//===----------------------------------------------------------------------===//
Chris Lattner283c0612009-04-27 05:27:42 +0000544
545/// \brief Note that we have loaded the declaration with the given
546/// Index.
547///
548/// This routine notes that this declaration has already been loaded,
549/// so that future GetDecl calls will return this declaration rather
550/// than trying to load a new declaration.
551inline void PCHReader::LoadedDecl(unsigned Index, Decl *D) {
552 assert(!DeclsLoaded[Index] && "Decl loaded twice?");
553 DeclsLoaded[Index] = D;
554}
555
556
557/// \brief Determine whether the consumer will be interested in seeing
558/// this declaration (via HandleTopLevelDecl).
559///
560/// This routine should return true for anything that might affect
561/// code generation, e.g., inline function definitions, Objective-C
562/// declarations with metadata, etc.
563static bool isConsumerInterestedIn(Decl *D) {
564 if (VarDecl *Var = dyn_cast<VarDecl>(D))
565 return Var->isFileVarDecl() && Var->getInit();
566 if (FunctionDecl *Func = dyn_cast<FunctionDecl>(D))
567 return Func->isThisDeclarationADefinition();
568 return isa<ObjCProtocolDecl>(D);
569}
570
571/// \brief Read the declaration at the given offset from the PCH file.
572Decl *PCHReader::ReadDeclRecord(uint64_t Offset, unsigned Index) {
573 // Keep track of where we are in the stream, then jump back there
574 // after reading this declaration.
Chris Lattner3ef21962009-04-27 05:58:23 +0000575 SavedStreamPosition SavedPosition(DeclsCursor);
Chris Lattner283c0612009-04-27 05:27:42 +0000576
Chris Lattner3ef21962009-04-27 05:58:23 +0000577 DeclsCursor.JumpToBit(Offset);
Chris Lattner283c0612009-04-27 05:27:42 +0000578 RecordData Record;
Chris Lattner3ef21962009-04-27 05:58:23 +0000579 unsigned Code = DeclsCursor.ReadCode();
Chris Lattner283c0612009-04-27 05:27:42 +0000580 unsigned Idx = 0;
581 PCHDeclReader Reader(*this, Record, Idx);
582
Chris Lattner3ef21962009-04-27 05:58:23 +0000583 Decl *D = 0;
584 switch ((pch::DeclCode)DeclsCursor.ReadRecord(Code, Record)) {
Chris Lattner283c0612009-04-27 05:27:42 +0000585 case pch::DECL_ATTR:
586 case pch::DECL_CONTEXT_LEXICAL:
587 case pch::DECL_CONTEXT_VISIBLE:
588 assert(false && "Record cannot be de-serialized with ReadDeclRecord");
589 break;
Chris Lattner283c0612009-04-27 05:27:42 +0000590 case pch::DECL_TRANSLATION_UNIT:
591 assert(Index == 0 && "Translation unit must be at index 0");
592 D = Context.getTranslationUnitDecl();
593 break;
Chris Lattner99dccc82009-04-27 06:01:06 +0000594 case pch::DECL_TYPEDEF:
Chris Lattner283c0612009-04-27 05:27:42 +0000595 D = TypedefDecl::Create(Context, 0, SourceLocation(), 0, QualType());
596 break;
Chris Lattner99dccc82009-04-27 06:01:06 +0000597 case pch::DECL_ENUM:
Chris Lattner283c0612009-04-27 05:27:42 +0000598 D = EnumDecl::Create(Context, 0, SourceLocation(), 0, 0);
599 break;
Chris Lattner99dccc82009-04-27 06:01:06 +0000600 case pch::DECL_RECORD:
601 D = RecordDecl::Create(Context, TagDecl::TK_struct, 0, SourceLocation(),
Chris Lattner283c0612009-04-27 05:27:42 +0000602 0, 0);
603 break;
Chris Lattner99dccc82009-04-27 06:01:06 +0000604 case pch::DECL_ENUM_CONSTANT:
Chris Lattner283c0612009-04-27 05:27:42 +0000605 D = EnumConstantDecl::Create(Context, 0, SourceLocation(), 0, QualType(),
606 0, llvm::APSInt());
607 break;
Chris Lattner99dccc82009-04-27 06:01:06 +0000608 case pch::DECL_FUNCTION:
Chris Lattner283c0612009-04-27 05:27:42 +0000609 D = FunctionDecl::Create(Context, 0, SourceLocation(), DeclarationName(),
610 QualType());
611 break;
Chris Lattner99dccc82009-04-27 06:01:06 +0000612 case pch::DECL_OBJC_METHOD:
Chris Lattner283c0612009-04-27 05:27:42 +0000613 D = ObjCMethodDecl::Create(Context, SourceLocation(), SourceLocation(),
614 Selector(), QualType(), 0);
615 break;
Chris Lattner99dccc82009-04-27 06:01:06 +0000616 case pch::DECL_OBJC_INTERFACE:
Chris Lattner283c0612009-04-27 05:27:42 +0000617 D = ObjCInterfaceDecl::Create(Context, 0, SourceLocation(), 0);
618 break;
Chris Lattner99dccc82009-04-27 06:01:06 +0000619 case pch::DECL_OBJC_IVAR:
Chris Lattner283c0612009-04-27 05:27:42 +0000620 D = ObjCIvarDecl::Create(Context, 0, SourceLocation(), 0, QualType(),
621 ObjCIvarDecl::None);
622 break;
Chris Lattner99dccc82009-04-27 06:01:06 +0000623 case pch::DECL_OBJC_PROTOCOL:
Chris Lattner283c0612009-04-27 05:27:42 +0000624 D = ObjCProtocolDecl::Create(Context, 0, SourceLocation(), 0);
625 break;
Chris Lattner99dccc82009-04-27 06:01:06 +0000626 case pch::DECL_OBJC_AT_DEFS_FIELD:
Chris Lattner283c0612009-04-27 05:27:42 +0000627 D = ObjCAtDefsFieldDecl::Create(Context, 0, SourceLocation(), 0,
628 QualType(), 0);
629 break;
Chris Lattner99dccc82009-04-27 06:01:06 +0000630 case pch::DECL_OBJC_CLASS:
Chris Lattner283c0612009-04-27 05:27:42 +0000631 D = ObjCClassDecl::Create(Context, 0, SourceLocation());
632 break;
Chris Lattner99dccc82009-04-27 06:01:06 +0000633 case pch::DECL_OBJC_FORWARD_PROTOCOL:
Chris Lattner283c0612009-04-27 05:27:42 +0000634 D = ObjCForwardProtocolDecl::Create(Context, 0, SourceLocation());
635 break;
Chris Lattner99dccc82009-04-27 06:01:06 +0000636 case pch::DECL_OBJC_CATEGORY:
Chris Lattner283c0612009-04-27 05:27:42 +0000637 D = ObjCCategoryDecl::Create(Context, 0, SourceLocation(), 0);
638 break;
Chris Lattner99dccc82009-04-27 06:01:06 +0000639 case pch::DECL_OBJC_CATEGORY_IMPL:
Chris Lattner283c0612009-04-27 05:27:42 +0000640 D = ObjCCategoryImplDecl::Create(Context, 0, SourceLocation(), 0, 0);
641 break;
Chris Lattner99dccc82009-04-27 06:01:06 +0000642 case pch::DECL_OBJC_IMPLEMENTATION:
Chris Lattner283c0612009-04-27 05:27:42 +0000643 D = ObjCImplementationDecl::Create(Context, 0, SourceLocation(), 0, 0);
644 break;
Chris Lattner99dccc82009-04-27 06:01:06 +0000645 case pch::DECL_OBJC_COMPATIBLE_ALIAS:
Chris Lattner283c0612009-04-27 05:27:42 +0000646 D = ObjCCompatibleAliasDecl::Create(Context, 0, SourceLocation(), 0, 0);
647 break;
Chris Lattner99dccc82009-04-27 06:01:06 +0000648 case pch::DECL_OBJC_PROPERTY:
Chris Lattner283c0612009-04-27 05:27:42 +0000649 D = ObjCPropertyDecl::Create(Context, 0, SourceLocation(), 0, QualType());
650 break;
Chris Lattner99dccc82009-04-27 06:01:06 +0000651 case pch::DECL_OBJC_PROPERTY_IMPL:
Chris Lattner283c0612009-04-27 05:27:42 +0000652 D = ObjCPropertyImplDecl::Create(Context, 0, SourceLocation(),
653 SourceLocation(), 0,
654 ObjCPropertyImplDecl::Dynamic, 0);
655 break;
Chris Lattner99dccc82009-04-27 06:01:06 +0000656 case pch::DECL_FIELD:
Chris Lattner283c0612009-04-27 05:27:42 +0000657 D = FieldDecl::Create(Context, 0, SourceLocation(), 0, QualType(), 0,
658 false);
659 break;
Chris Lattner283c0612009-04-27 05:27:42 +0000660 case pch::DECL_VAR:
661 D = VarDecl::Create(Context, 0, SourceLocation(), 0, QualType(),
662 VarDecl::None, SourceLocation());
663 break;
664
665 case pch::DECL_IMPLICIT_PARAM:
666 D = ImplicitParamDecl::Create(Context, 0, SourceLocation(), 0, QualType());
667 break;
668
Chris Lattner99dccc82009-04-27 06:01:06 +0000669 case pch::DECL_PARM_VAR:
Chris Lattner283c0612009-04-27 05:27:42 +0000670 D = ParmVarDecl::Create(Context, 0, SourceLocation(), 0, QualType(),
671 VarDecl::None, 0);
672 break;
Chris Lattner99dccc82009-04-27 06:01:06 +0000673 case pch::DECL_ORIGINAL_PARM_VAR:
Chris Lattner283c0612009-04-27 05:27:42 +0000674 D = OriginalParmVarDecl::Create(Context, 0, SourceLocation(), 0,
Chris Lattner99dccc82009-04-27 06:01:06 +0000675 QualType(), QualType(), VarDecl::None, 0);
Chris Lattner283c0612009-04-27 05:27:42 +0000676 break;
Chris Lattner99dccc82009-04-27 06:01:06 +0000677 case pch::DECL_FILE_SCOPE_ASM:
Chris Lattner283c0612009-04-27 05:27:42 +0000678 D = FileScopeAsmDecl::Create(Context, 0, SourceLocation(), 0);
679 break;
Chris Lattner99dccc82009-04-27 06:01:06 +0000680 case pch::DECL_BLOCK:
Chris Lattner283c0612009-04-27 05:27:42 +0000681 D = BlockDecl::Create(Context, 0, SourceLocation());
682 break;
683 }
Chris Lattner283c0612009-04-27 05:27:42 +0000684
685 assert(D && "Unknown declaration reading PCH file");
Chris Lattner99dccc82009-04-27 06:01:06 +0000686 LoadedDecl(Index, D);
687 Reader.Visit(D);
Chris Lattner283c0612009-04-27 05:27:42 +0000688
689 // If this declaration is also a declaration context, get the
690 // offsets for its tables of lexical and visible declarations.
691 if (DeclContext *DC = dyn_cast<DeclContext>(D)) {
692 std::pair<uint64_t, uint64_t> Offsets = Reader.VisitDeclContext(DC);
693 if (Offsets.first || Offsets.second) {
694 DC->setHasExternalLexicalStorage(Offsets.first != 0);
695 DC->setHasExternalVisibleStorage(Offsets.second != 0);
696 DeclContextOffsets[DC] = Offsets;
697 }
698 }
699 assert(Idx == Record.size());
700
701 // If we have deserialized a declaration that has a definition the
702 // AST consumer might need to know about, notify the consumer
703 // about that definition now or queue it for later.
704 if (isConsumerInterestedIn(D)) {
705 if (Consumer) {
706 DeclGroupRef DG(D);
707 Consumer->HandleTopLevelDecl(DG);
708 } else {
709 InterestingDecls.push_back(D);
710 }
711 }
712
713 return D;
714}
715