Chris Lattner | 698f925 | 2009-04-27 05:27:42 +0000 | [diff] [blame] | 1 | //===--- 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" |
Argyrios Kyrtzidis | d4a7e54 | 2009-08-19 01:28:35 +0000 | [diff] [blame] | 21 | #include "clang/AST/TypeLoc.h" |
Chris Lattner | 698f925 | 2009-04-27 05:27:42 +0000 | [diff] [blame] | 22 | using namespace clang; |
| 23 | |
Chris Lattner | 698f925 | 2009-04-27 05:27:42 +0000 | [diff] [blame] | 24 | |
| 25 | //===----------------------------------------------------------------------===// |
| 26 | // Declaration deserialization |
| 27 | //===----------------------------------------------------------------------===// |
| 28 | |
| 29 | namespace { |
| 30 | class PCHDeclReader : public DeclVisitor<PCHDeclReader, void> { |
| 31 | PCHReader &Reader; |
| 32 | const PCHReader::RecordData &Record; |
| 33 | unsigned &Idx; |
| 34 | |
| 35 | public: |
| 36 | PCHDeclReader(PCHReader &Reader, const PCHReader::RecordData &Record, |
| 37 | unsigned &Idx) |
| 38 | : Reader(Reader), Record(Record), Idx(Idx) { } |
| 39 | |
| 40 | void VisitDecl(Decl *D); |
| 41 | void VisitTranslationUnitDecl(TranslationUnitDecl *TU); |
| 42 | void VisitNamedDecl(NamedDecl *ND); |
| 43 | void VisitTypeDecl(TypeDecl *TD); |
| 44 | void VisitTypedefDecl(TypedefDecl *TD); |
| 45 | void VisitTagDecl(TagDecl *TD); |
| 46 | void VisitEnumDecl(EnumDecl *ED); |
| 47 | void VisitRecordDecl(RecordDecl *RD); |
| 48 | void VisitValueDecl(ValueDecl *VD); |
| 49 | void VisitEnumConstantDecl(EnumConstantDecl *ECD); |
Argyrios Kyrtzidis | d4a7e54 | 2009-08-19 01:28:35 +0000 | [diff] [blame] | 50 | void VisitDeclaratorDecl(DeclaratorDecl *DD); |
Chris Lattner | 698f925 | 2009-04-27 05:27:42 +0000 | [diff] [blame] | 51 | void VisitFunctionDecl(FunctionDecl *FD); |
| 52 | void VisitFieldDecl(FieldDecl *FD); |
| 53 | void VisitVarDecl(VarDecl *VD); |
| 54 | void VisitImplicitParamDecl(ImplicitParamDecl *PD); |
| 55 | void VisitParmVarDecl(ParmVarDecl *PD); |
| 56 | void VisitOriginalParmVarDecl(OriginalParmVarDecl *PD); |
| 57 | void VisitFileScopeAsmDecl(FileScopeAsmDecl *AD); |
| 58 | void VisitBlockDecl(BlockDecl *BD); |
| 59 | std::pair<uint64_t, uint64_t> VisitDeclContext(DeclContext *DC); |
| 60 | void VisitObjCMethodDecl(ObjCMethodDecl *D); |
| 61 | void VisitObjCContainerDecl(ObjCContainerDecl *D); |
| 62 | void VisitObjCInterfaceDecl(ObjCInterfaceDecl *D); |
| 63 | void VisitObjCIvarDecl(ObjCIvarDecl *D); |
| 64 | void VisitObjCProtocolDecl(ObjCProtocolDecl *D); |
| 65 | void VisitObjCAtDefsFieldDecl(ObjCAtDefsFieldDecl *D); |
| 66 | void VisitObjCClassDecl(ObjCClassDecl *D); |
| 67 | void VisitObjCForwardProtocolDecl(ObjCForwardProtocolDecl *D); |
| 68 | void VisitObjCCategoryDecl(ObjCCategoryDecl *D); |
| 69 | void VisitObjCImplDecl(ObjCImplDecl *D); |
| 70 | void VisitObjCCategoryImplDecl(ObjCCategoryImplDecl *D); |
| 71 | void VisitObjCImplementationDecl(ObjCImplementationDecl *D); |
| 72 | void VisitObjCCompatibleAliasDecl(ObjCCompatibleAliasDecl *D); |
| 73 | void VisitObjCPropertyDecl(ObjCPropertyDecl *D); |
| 74 | void VisitObjCPropertyImplDecl(ObjCPropertyImplDecl *D); |
| 75 | }; |
| 76 | } |
| 77 | |
| 78 | void PCHDeclReader::VisitDecl(Decl *D) { |
| 79 | D->setDeclContext(cast_or_null<DeclContext>(Reader.GetDecl(Record[Idx++]))); |
| 80 | D->setLexicalDeclContext( |
| 81 | cast_or_null<DeclContext>(Reader.GetDecl(Record[Idx++]))); |
| 82 | D->setLocation(SourceLocation::getFromRawEncoding(Record[Idx++])); |
| 83 | D->setInvalidDecl(Record[Idx++]); |
| 84 | if (Record[Idx++]) |
Argyrios Kyrtzidis | 40b598e | 2009-06-30 02:34:44 +0000 | [diff] [blame] | 85 | D->addAttr(Reader.ReadAttributes()); |
Chris Lattner | 698f925 | 2009-04-27 05:27:42 +0000 | [diff] [blame] | 86 | D->setImplicit(Record[Idx++]); |
Douglas Gregor | e0762c9 | 2009-06-19 23:52:42 +0000 | [diff] [blame] | 87 | D->setUsed(Record[Idx++]); |
Chris Lattner | 698f925 | 2009-04-27 05:27:42 +0000 | [diff] [blame] | 88 | D->setAccess((AccessSpecifier)Record[Idx++]); |
| 89 | } |
| 90 | |
| 91 | void PCHDeclReader::VisitTranslationUnitDecl(TranslationUnitDecl *TU) { |
| 92 | VisitDecl(TU); |
| 93 | } |
| 94 | |
| 95 | void PCHDeclReader::VisitNamedDecl(NamedDecl *ND) { |
| 96 | VisitDecl(ND); |
| 97 | ND->setDeclName(Reader.ReadDeclarationName(Record, Idx)); |
| 98 | } |
| 99 | |
| 100 | void PCHDeclReader::VisitTypeDecl(TypeDecl *TD) { |
| 101 | VisitNamedDecl(TD); |
| 102 | TD->setTypeForDecl(Reader.GetType(Record[Idx++]).getTypePtr()); |
| 103 | } |
| 104 | |
| 105 | void PCHDeclReader::VisitTypedefDecl(TypedefDecl *TD) { |
| 106 | // Note that we cannot use VisitTypeDecl here, because we need to |
| 107 | // set the underlying type of the typedef *before* we try to read |
| 108 | // the type associated with the TypedefDecl. |
| 109 | VisitNamedDecl(TD); |
| 110 | TD->setUnderlyingType(Reader.GetType(Record[Idx + 1])); |
| 111 | TD->setTypeForDecl(Reader.GetType(Record[Idx]).getTypePtr()); |
| 112 | Idx += 2; |
| 113 | } |
| 114 | |
| 115 | void PCHDeclReader::VisitTagDecl(TagDecl *TD) { |
| 116 | VisitTypeDecl(TD); |
Douglas Gregor | 8e9e9ef | 2009-07-29 23:36:44 +0000 | [diff] [blame] | 117 | TD->setPreviousDeclaration( |
| 118 | cast_or_null<TagDecl>(Reader.GetDecl(Record[Idx++]))); |
Chris Lattner | 698f925 | 2009-04-27 05:27:42 +0000 | [diff] [blame] | 119 | TD->setTagKind((TagDecl::TagKind)Record[Idx++]); |
| 120 | TD->setDefinition(Record[Idx++]); |
| 121 | TD->setTypedefForAnonDecl( |
| 122 | cast_or_null<TypedefDecl>(Reader.GetDecl(Record[Idx++]))); |
Argyrios Kyrtzidis | ad93a74 | 2009-07-14 03:18:02 +0000 | [diff] [blame] | 123 | TD->setRBraceLoc(SourceLocation::getFromRawEncoding(Record[Idx++])); |
Douglas Gregor | 741dd9a | 2009-07-21 14:46:17 +0000 | [diff] [blame] | 124 | TD->setTagKeywordLoc(SourceLocation::getFromRawEncoding(Record[Idx++])); |
Chris Lattner | 698f925 | 2009-04-27 05:27:42 +0000 | [diff] [blame] | 125 | } |
| 126 | |
| 127 | void PCHDeclReader::VisitEnumDecl(EnumDecl *ED) { |
| 128 | VisitTagDecl(ED); |
| 129 | ED->setIntegerType(Reader.GetType(Record[Idx++])); |
Douglas Gregor | 8dbc3c6 | 2009-05-27 17:20:35 +0000 | [diff] [blame] | 130 | // FIXME: C++ InstantiatedFrom |
Chris Lattner | 698f925 | 2009-04-27 05:27:42 +0000 | [diff] [blame] | 131 | } |
| 132 | |
| 133 | void PCHDeclReader::VisitRecordDecl(RecordDecl *RD) { |
| 134 | VisitTagDecl(RD); |
| 135 | RD->setHasFlexibleArrayMember(Record[Idx++]); |
| 136 | RD->setAnonymousStructOrUnion(Record[Idx++]); |
Fariborz Jahanian | 643b7df | 2009-07-08 16:37:44 +0000 | [diff] [blame] | 137 | RD->setHasObjectMember(Record[Idx++]); |
Chris Lattner | 698f925 | 2009-04-27 05:27:42 +0000 | [diff] [blame] | 138 | } |
| 139 | |
| 140 | void PCHDeclReader::VisitValueDecl(ValueDecl *VD) { |
| 141 | VisitNamedDecl(VD); |
| 142 | VD->setType(Reader.GetType(Record[Idx++])); |
| 143 | } |
| 144 | |
| 145 | void PCHDeclReader::VisitEnumConstantDecl(EnumConstantDecl *ECD) { |
| 146 | VisitValueDecl(ECD); |
| 147 | if (Record[Idx++]) |
Chris Lattner | da93061 | 2009-04-27 05:58:23 +0000 | [diff] [blame] | 148 | ECD->setInitExpr(Reader.ReadDeclExpr()); |
Chris Lattner | 698f925 | 2009-04-27 05:27:42 +0000 | [diff] [blame] | 149 | ECD->setInitVal(Reader.ReadAPSInt(Record, Idx)); |
| 150 | } |
| 151 | |
Argyrios Kyrtzidis | d4a7e54 | 2009-08-19 01:28:35 +0000 | [diff] [blame] | 152 | namespace { |
| 153 | |
| 154 | class TypeLocReader : public TypeLocVisitor<TypeLocReader> { |
| 155 | PCHReader &Reader; |
| 156 | const PCHReader::RecordData &Record; |
| 157 | unsigned &Idx; |
| 158 | |
| 159 | public: |
| 160 | TypeLocReader(PCHReader &Reader, const PCHReader::RecordData &Record, |
| 161 | unsigned &Idx) |
| 162 | : Reader(Reader), Record(Record), Idx(Idx) { } |
| 163 | |
| 164 | #define ABSTRACT_TYPELOC(CLASS) |
| 165 | #define TYPELOC(CLASS, PARENT, TYPE) \ |
| 166 | void Visit##CLASS(CLASS TyLoc); |
| 167 | #include "clang/AST/TypeLocNodes.def" |
| 168 | |
| 169 | void VisitTypeLoc(TypeLoc TyLoc) { |
| 170 | assert(0 && "A type loc wrapper was not handled!"); |
| 171 | } |
| 172 | }; |
| 173 | |
| 174 | } |
| 175 | |
| 176 | void TypeLocReader::VisitDefaultTypeSpecLoc(DefaultTypeSpecLoc TyLoc) { |
| 177 | TyLoc.setStartLoc(SourceLocation::getFromRawEncoding(Record[Idx++])); |
| 178 | } |
| 179 | void TypeLocReader::VisitTypedefLoc(TypedefLoc TyLoc) { |
| 180 | TyLoc.setNameLoc(SourceLocation::getFromRawEncoding(Record[Idx++])); |
| 181 | } |
| 182 | void TypeLocReader::VisitPointerLoc(PointerLoc TyLoc) { |
| 183 | TyLoc.setStarLoc(SourceLocation::getFromRawEncoding(Record[Idx++])); |
| 184 | } |
| 185 | void TypeLocReader::VisitBlockPointerLoc(BlockPointerLoc TyLoc) { |
| 186 | TyLoc.setCaretLoc(SourceLocation::getFromRawEncoding(Record[Idx++])); |
| 187 | } |
| 188 | void TypeLocReader::VisitMemberPointerLoc(MemberPointerLoc TyLoc) { |
| 189 | TyLoc.setStarLoc(SourceLocation::getFromRawEncoding(Record[Idx++])); |
| 190 | } |
| 191 | void TypeLocReader::VisitReferenceLoc(ReferenceLoc TyLoc) { |
| 192 | TyLoc.setAmpLoc(SourceLocation::getFromRawEncoding(Record[Idx++])); |
| 193 | } |
| 194 | void TypeLocReader::VisitFunctionLoc(FunctionLoc TyLoc) { |
| 195 | TyLoc.setLParenLoc(SourceLocation::getFromRawEncoding(Record[Idx++])); |
| 196 | TyLoc.setRParenLoc(SourceLocation::getFromRawEncoding(Record[Idx++])); |
| 197 | for (unsigned i = 0, e = TyLoc.getNumArgs(); i != e; ++i) |
| 198 | TyLoc.setArg(i, cast<ParmVarDecl>(Reader.GetDecl(Record[Idx++]))); |
| 199 | } |
| 200 | void TypeLocReader::VisitArrayLoc(ArrayLoc TyLoc) { |
| 201 | TyLoc.setLBracketLoc(SourceLocation::getFromRawEncoding(Record[Idx++])); |
| 202 | TyLoc.setRBracketLoc(SourceLocation::getFromRawEncoding(Record[Idx++])); |
| 203 | if (Record[Idx++]) |
| 204 | TyLoc.setSizeExpr(Reader.ReadDeclExpr()); |
| 205 | } |
| 206 | |
| 207 | void PCHDeclReader::VisitDeclaratorDecl(DeclaratorDecl *DD) { |
| 208 | VisitValueDecl(DD); |
| 209 | QualType InfoTy = Reader.GetType(Record[Idx++]); |
| 210 | if (InfoTy.isNull()) |
| 211 | return; |
| 212 | |
| 213 | DeclaratorInfo *DInfo = Reader.getContext()->CreateDeclaratorInfo(InfoTy); |
| 214 | TypeLocReader TLR(Reader, Record, Idx); |
| 215 | for (TypeLoc TL = DInfo->getTypeLoc(); !TL.isNull(); TL = TL.getNextTypeLoc()) |
| 216 | TLR.Visit(TL); |
| 217 | DD->setDeclaratorInfo(DInfo); |
| 218 | } |
| 219 | |
Chris Lattner | 698f925 | 2009-04-27 05:27:42 +0000 | [diff] [blame] | 220 | void PCHDeclReader::VisitFunctionDecl(FunctionDecl *FD) { |
Argyrios Kyrtzidis | d4a7e54 | 2009-08-19 01:28:35 +0000 | [diff] [blame] | 221 | VisitDeclaratorDecl(FD); |
Chris Lattner | 698f925 | 2009-04-27 05:27:42 +0000 | [diff] [blame] | 222 | if (Record[Idx++]) |
Chris Lattner | da93061 | 2009-04-27 05:58:23 +0000 | [diff] [blame] | 223 | FD->setLazyBody(Reader.getDeclsCursor().GetCurrentBitNo()); |
Chris Lattner | 698f925 | 2009-04-27 05:27:42 +0000 | [diff] [blame] | 224 | FD->setPreviousDeclaration( |
| 225 | cast_or_null<FunctionDecl>(Reader.GetDecl(Record[Idx++]))); |
| 226 | FD->setStorageClass((FunctionDecl::StorageClass)Record[Idx++]); |
| 227 | FD->setInline(Record[Idx++]); |
| 228 | FD->setC99InlineDefinition(Record[Idx++]); |
Anders Carlsson | 77b7f1d | 2009-05-14 22:15:41 +0000 | [diff] [blame] | 229 | FD->setVirtualAsWritten(Record[Idx++]); |
Chris Lattner | 698f925 | 2009-04-27 05:27:42 +0000 | [diff] [blame] | 230 | FD->setPure(Record[Idx++]); |
Anders Carlsson | a75e853 | 2009-05-14 21:46:00 +0000 | [diff] [blame] | 231 | FD->setHasInheritedPrototype(Record[Idx++]); |
| 232 | FD->setHasWrittenPrototype(Record[Idx++]); |
Chris Lattner | 698f925 | 2009-04-27 05:27:42 +0000 | [diff] [blame] | 233 | FD->setDeleted(Record[Idx++]); |
Argyrios Kyrtzidis | 8cff90e | 2009-06-20 08:09:34 +0000 | [diff] [blame] | 234 | FD->setLocEnd(SourceLocation::getFromRawEncoding(Record[Idx++])); |
Douglas Gregor | 1eee0e7 | 2009-05-14 21:06:31 +0000 | [diff] [blame] | 235 | // FIXME: C++ TemplateOrInstantiation |
Chris Lattner | 698f925 | 2009-04-27 05:27:42 +0000 | [diff] [blame] | 236 | unsigned NumParams = Record[Idx++]; |
| 237 | llvm::SmallVector<ParmVarDecl *, 16> Params; |
| 238 | Params.reserve(NumParams); |
| 239 | for (unsigned I = 0; I != NumParams; ++I) |
| 240 | Params.push_back(cast<ParmVarDecl>(Reader.GetDecl(Record[Idx++]))); |
Jay Foad | beaaccd | 2009-05-21 09:52:38 +0000 | [diff] [blame] | 241 | FD->setParams(*Reader.getContext(), Params.data(), NumParams); |
Chris Lattner | 698f925 | 2009-04-27 05:27:42 +0000 | [diff] [blame] | 242 | } |
| 243 | |
| 244 | void PCHDeclReader::VisitObjCMethodDecl(ObjCMethodDecl *MD) { |
| 245 | VisitNamedDecl(MD); |
| 246 | if (Record[Idx++]) { |
| 247 | // In practice, this won't be executed (since method definitions |
| 248 | // don't occur in header files). |
Chris Lattner | da93061 | 2009-04-27 05:58:23 +0000 | [diff] [blame] | 249 | MD->setBody(Reader.ReadDeclStmt()); |
Chris Lattner | 698f925 | 2009-04-27 05:27:42 +0000 | [diff] [blame] | 250 | MD->setSelfDecl(cast<ImplicitParamDecl>(Reader.GetDecl(Record[Idx++]))); |
| 251 | MD->setCmdDecl(cast<ImplicitParamDecl>(Reader.GetDecl(Record[Idx++]))); |
| 252 | } |
| 253 | MD->setInstanceMethod(Record[Idx++]); |
| 254 | MD->setVariadic(Record[Idx++]); |
| 255 | MD->setSynthesized(Record[Idx++]); |
| 256 | MD->setDeclImplementation((ObjCMethodDecl::ImplementationControl)Record[Idx++]); |
| 257 | MD->setObjCDeclQualifier((Decl::ObjCDeclQualifier)Record[Idx++]); |
| 258 | MD->setResultType(Reader.GetType(Record[Idx++])); |
| 259 | MD->setEndLoc(SourceLocation::getFromRawEncoding(Record[Idx++])); |
| 260 | unsigned NumParams = Record[Idx++]; |
| 261 | llvm::SmallVector<ParmVarDecl *, 16> Params; |
| 262 | Params.reserve(NumParams); |
| 263 | for (unsigned I = 0; I != NumParams; ++I) |
| 264 | Params.push_back(cast<ParmVarDecl>(Reader.GetDecl(Record[Idx++]))); |
Jay Foad | beaaccd | 2009-05-21 09:52:38 +0000 | [diff] [blame] | 265 | MD->setMethodParams(*Reader.getContext(), Params.data(), NumParams); |
Chris Lattner | 698f925 | 2009-04-27 05:27:42 +0000 | [diff] [blame] | 266 | } |
| 267 | |
| 268 | void PCHDeclReader::VisitObjCContainerDecl(ObjCContainerDecl *CD) { |
| 269 | VisitNamedDecl(CD); |
| 270 | CD->setAtEndLoc(SourceLocation::getFromRawEncoding(Record[Idx++])); |
| 271 | } |
| 272 | |
| 273 | void PCHDeclReader::VisitObjCInterfaceDecl(ObjCInterfaceDecl *ID) { |
| 274 | VisitObjCContainerDecl(ID); |
| 275 | ID->setTypeForDecl(Reader.GetType(Record[Idx++]).getTypePtr()); |
| 276 | ID->setSuperClass(cast_or_null<ObjCInterfaceDecl> |
| 277 | (Reader.GetDecl(Record[Idx++]))); |
| 278 | unsigned NumProtocols = Record[Idx++]; |
| 279 | llvm::SmallVector<ObjCProtocolDecl *, 16> Protocols; |
| 280 | Protocols.reserve(NumProtocols); |
| 281 | for (unsigned I = 0; I != NumProtocols; ++I) |
| 282 | Protocols.push_back(cast<ObjCProtocolDecl>(Reader.GetDecl(Record[Idx++]))); |
Jay Foad | beaaccd | 2009-05-21 09:52:38 +0000 | [diff] [blame] | 283 | ID->setProtocolList(Protocols.data(), NumProtocols, *Reader.getContext()); |
Chris Lattner | 698f925 | 2009-04-27 05:27:42 +0000 | [diff] [blame] | 284 | unsigned NumIvars = Record[Idx++]; |
| 285 | llvm::SmallVector<ObjCIvarDecl *, 16> IVars; |
| 286 | IVars.reserve(NumIvars); |
| 287 | for (unsigned I = 0; I != NumIvars; ++I) |
| 288 | IVars.push_back(cast<ObjCIvarDecl>(Reader.GetDecl(Record[Idx++]))); |
Jay Foad | beaaccd | 2009-05-21 09:52:38 +0000 | [diff] [blame] | 289 | ID->setIVarList(IVars.data(), NumIvars, *Reader.getContext()); |
Chris Lattner | 698f925 | 2009-04-27 05:27:42 +0000 | [diff] [blame] | 290 | ID->setCategoryList( |
| 291 | cast_or_null<ObjCCategoryDecl>(Reader.GetDecl(Record[Idx++]))); |
| 292 | ID->setForwardDecl(Record[Idx++]); |
| 293 | ID->setImplicitInterfaceDecl(Record[Idx++]); |
| 294 | ID->setClassLoc(SourceLocation::getFromRawEncoding(Record[Idx++])); |
| 295 | ID->setSuperClassLoc(SourceLocation::getFromRawEncoding(Record[Idx++])); |
Argyrios Kyrtzidis | c999f1f | 2009-07-18 00:33:23 +0000 | [diff] [blame] | 296 | ID->setLocEnd(SourceLocation::getFromRawEncoding(Record[Idx++])); |
Chris Lattner | 698f925 | 2009-04-27 05:27:42 +0000 | [diff] [blame] | 297 | } |
| 298 | |
| 299 | void PCHDeclReader::VisitObjCIvarDecl(ObjCIvarDecl *IVD) { |
| 300 | VisitFieldDecl(IVD); |
| 301 | IVD->setAccessControl((ObjCIvarDecl::AccessControl)Record[Idx++]); |
| 302 | } |
| 303 | |
| 304 | void PCHDeclReader::VisitObjCProtocolDecl(ObjCProtocolDecl *PD) { |
| 305 | VisitObjCContainerDecl(PD); |
| 306 | PD->setForwardDecl(Record[Idx++]); |
| 307 | PD->setLocEnd(SourceLocation::getFromRawEncoding(Record[Idx++])); |
| 308 | unsigned NumProtoRefs = Record[Idx++]; |
| 309 | llvm::SmallVector<ObjCProtocolDecl *, 16> ProtoRefs; |
| 310 | ProtoRefs.reserve(NumProtoRefs); |
| 311 | for (unsigned I = 0; I != NumProtoRefs; ++I) |
| 312 | ProtoRefs.push_back(cast<ObjCProtocolDecl>(Reader.GetDecl(Record[Idx++]))); |
Jay Foad | beaaccd | 2009-05-21 09:52:38 +0000 | [diff] [blame] | 313 | PD->setProtocolList(ProtoRefs.data(), NumProtoRefs, *Reader.getContext()); |
Chris Lattner | 698f925 | 2009-04-27 05:27:42 +0000 | [diff] [blame] | 314 | } |
| 315 | |
| 316 | void PCHDeclReader::VisitObjCAtDefsFieldDecl(ObjCAtDefsFieldDecl *FD) { |
| 317 | VisitFieldDecl(FD); |
| 318 | } |
| 319 | |
| 320 | void PCHDeclReader::VisitObjCClassDecl(ObjCClassDecl *CD) { |
| 321 | VisitDecl(CD); |
| 322 | unsigned NumClassRefs = Record[Idx++]; |
| 323 | llvm::SmallVector<ObjCInterfaceDecl *, 16> ClassRefs; |
| 324 | ClassRefs.reserve(NumClassRefs); |
| 325 | for (unsigned I = 0; I != NumClassRefs; ++I) |
| 326 | ClassRefs.push_back(cast<ObjCInterfaceDecl>(Reader.GetDecl(Record[Idx++]))); |
Douglas Gregor | 75fdb23 | 2009-05-22 22:45:36 +0000 | [diff] [blame] | 327 | CD->setClassList(*Reader.getContext(), ClassRefs.data(), NumClassRefs); |
Chris Lattner | 698f925 | 2009-04-27 05:27:42 +0000 | [diff] [blame] | 328 | } |
| 329 | |
| 330 | void PCHDeclReader::VisitObjCForwardProtocolDecl(ObjCForwardProtocolDecl *FPD) { |
| 331 | VisitDecl(FPD); |
| 332 | unsigned NumProtoRefs = Record[Idx++]; |
| 333 | llvm::SmallVector<ObjCProtocolDecl *, 16> ProtoRefs; |
| 334 | ProtoRefs.reserve(NumProtoRefs); |
| 335 | for (unsigned I = 0; I != NumProtoRefs; ++I) |
| 336 | ProtoRefs.push_back(cast<ObjCProtocolDecl>(Reader.GetDecl(Record[Idx++]))); |
Douglas Gregor | 75fdb23 | 2009-05-22 22:45:36 +0000 | [diff] [blame] | 337 | FPD->setProtocolList(ProtoRefs.data(), NumProtoRefs, *Reader.getContext()); |
Chris Lattner | 698f925 | 2009-04-27 05:27:42 +0000 | [diff] [blame] | 338 | } |
| 339 | |
| 340 | void PCHDeclReader::VisitObjCCategoryDecl(ObjCCategoryDecl *CD) { |
| 341 | VisitObjCContainerDecl(CD); |
| 342 | CD->setClassInterface(cast<ObjCInterfaceDecl>(Reader.GetDecl(Record[Idx++]))); |
| 343 | unsigned NumProtoRefs = Record[Idx++]; |
| 344 | llvm::SmallVector<ObjCProtocolDecl *, 16> ProtoRefs; |
| 345 | ProtoRefs.reserve(NumProtoRefs); |
| 346 | for (unsigned I = 0; I != NumProtoRefs; ++I) |
| 347 | ProtoRefs.push_back(cast<ObjCProtocolDecl>(Reader.GetDecl(Record[Idx++]))); |
Ted Kremenek | 66ef111 | 2009-05-22 22:34:23 +0000 | [diff] [blame] | 348 | CD->setProtocolList(ProtoRefs.data(), NumProtoRefs, *Reader.getContext()); |
Chris Lattner | 698f925 | 2009-04-27 05:27:42 +0000 | [diff] [blame] | 349 | CD->setNextClassCategory(cast_or_null<ObjCCategoryDecl>(Reader.GetDecl(Record[Idx++]))); |
| 350 | CD->setLocEnd(SourceLocation::getFromRawEncoding(Record[Idx++])); |
| 351 | } |
| 352 | |
| 353 | void PCHDeclReader::VisitObjCCompatibleAliasDecl(ObjCCompatibleAliasDecl *CAD) { |
| 354 | VisitNamedDecl(CAD); |
| 355 | CAD->setClassInterface(cast<ObjCInterfaceDecl>(Reader.GetDecl(Record[Idx++]))); |
| 356 | } |
| 357 | |
| 358 | void PCHDeclReader::VisitObjCPropertyDecl(ObjCPropertyDecl *D) { |
| 359 | VisitNamedDecl(D); |
| 360 | D->setType(Reader.GetType(Record[Idx++])); |
| 361 | // FIXME: stable encoding |
| 362 | D->setPropertyAttributes( |
| 363 | (ObjCPropertyDecl::PropertyAttributeKind)Record[Idx++]); |
| 364 | // FIXME: stable encoding |
| 365 | D->setPropertyImplementation( |
| 366 | (ObjCPropertyDecl::PropertyControl)Record[Idx++]); |
| 367 | D->setGetterName(Reader.ReadDeclarationName(Record, Idx).getObjCSelector()); |
| 368 | D->setSetterName(Reader.ReadDeclarationName(Record, Idx).getObjCSelector()); |
| 369 | D->setGetterMethodDecl( |
| 370 | cast_or_null<ObjCMethodDecl>(Reader.GetDecl(Record[Idx++]))); |
| 371 | D->setSetterMethodDecl( |
| 372 | cast_or_null<ObjCMethodDecl>(Reader.GetDecl(Record[Idx++]))); |
| 373 | D->setPropertyIvarDecl( |
| 374 | cast_or_null<ObjCIvarDecl>(Reader.GetDecl(Record[Idx++]))); |
| 375 | } |
| 376 | |
| 377 | void PCHDeclReader::VisitObjCImplDecl(ObjCImplDecl *D) { |
Argyrios Kyrtzidis | aecae62 | 2009-07-27 19:04:32 +0000 | [diff] [blame] | 378 | VisitObjCContainerDecl(D); |
Chris Lattner | 698f925 | 2009-04-27 05:27:42 +0000 | [diff] [blame] | 379 | D->setClassInterface( |
| 380 | cast_or_null<ObjCInterfaceDecl>(Reader.GetDecl(Record[Idx++]))); |
Chris Lattner | 698f925 | 2009-04-27 05:27:42 +0000 | [diff] [blame] | 381 | } |
| 382 | |
| 383 | void PCHDeclReader::VisitObjCCategoryImplDecl(ObjCCategoryImplDecl *D) { |
| 384 | VisitObjCImplDecl(D); |
| 385 | D->setIdentifier(Reader.GetIdentifierInfo(Record, Idx)); |
| 386 | } |
| 387 | |
| 388 | void PCHDeclReader::VisitObjCImplementationDecl(ObjCImplementationDecl *D) { |
| 389 | VisitObjCImplDecl(D); |
| 390 | D->setSuperClass( |
| 391 | cast_or_null<ObjCInterfaceDecl>(Reader.GetDecl(Record[Idx++]))); |
| 392 | } |
| 393 | |
| 394 | |
| 395 | void PCHDeclReader::VisitObjCPropertyImplDecl(ObjCPropertyImplDecl *D) { |
| 396 | VisitDecl(D); |
| 397 | D->setAtLoc(SourceLocation::getFromRawEncoding(Record[Idx++])); |
| 398 | D->setPropertyDecl( |
| 399 | cast_or_null<ObjCPropertyDecl>(Reader.GetDecl(Record[Idx++]))); |
| 400 | D->setPropertyIvarDecl( |
| 401 | cast_or_null<ObjCIvarDecl>(Reader.GetDecl(Record[Idx++]))); |
| 402 | } |
| 403 | |
| 404 | void PCHDeclReader::VisitFieldDecl(FieldDecl *FD) { |
Argyrios Kyrtzidis | d4a7e54 | 2009-08-19 01:28:35 +0000 | [diff] [blame] | 405 | VisitDeclaratorDecl(FD); |
Chris Lattner | 698f925 | 2009-04-27 05:27:42 +0000 | [diff] [blame] | 406 | FD->setMutable(Record[Idx++]); |
| 407 | if (Record[Idx++]) |
Chris Lattner | da93061 | 2009-04-27 05:58:23 +0000 | [diff] [blame] | 408 | FD->setBitWidth(Reader.ReadDeclExpr()); |
Chris Lattner | 698f925 | 2009-04-27 05:27:42 +0000 | [diff] [blame] | 409 | } |
| 410 | |
| 411 | void PCHDeclReader::VisitVarDecl(VarDecl *VD) { |
Argyrios Kyrtzidis | d4a7e54 | 2009-08-19 01:28:35 +0000 | [diff] [blame] | 412 | VisitDeclaratorDecl(VD); |
Chris Lattner | 698f925 | 2009-04-27 05:27:42 +0000 | [diff] [blame] | 413 | VD->setStorageClass((VarDecl::StorageClass)Record[Idx++]); |
| 414 | VD->setThreadSpecified(Record[Idx++]); |
| 415 | VD->setCXXDirectInitializer(Record[Idx++]); |
| 416 | VD->setDeclaredInCondition(Record[Idx++]); |
| 417 | VD->setPreviousDeclaration( |
| 418 | cast_or_null<VarDecl>(Reader.GetDecl(Record[Idx++]))); |
Chris Lattner | 698f925 | 2009-04-27 05:27:42 +0000 | [diff] [blame] | 419 | if (Record[Idx++]) |
Douglas Gregor | 78d1583 | 2009-05-26 18:54:04 +0000 | [diff] [blame] | 420 | VD->setInit(*Reader.getContext(), Reader.ReadDeclExpr()); |
Chris Lattner | 698f925 | 2009-04-27 05:27:42 +0000 | [diff] [blame] | 421 | } |
| 422 | |
| 423 | void PCHDeclReader::VisitImplicitParamDecl(ImplicitParamDecl *PD) { |
| 424 | VisitVarDecl(PD); |
| 425 | } |
| 426 | |
| 427 | void PCHDeclReader::VisitParmVarDecl(ParmVarDecl *PD) { |
| 428 | VisitVarDecl(PD); |
| 429 | PD->setObjCDeclQualifier((Decl::ObjCDeclQualifier)Record[Idx++]); |
Chris Lattner | 698f925 | 2009-04-27 05:27:42 +0000 | [diff] [blame] | 430 | } |
| 431 | |
| 432 | void PCHDeclReader::VisitOriginalParmVarDecl(OriginalParmVarDecl *PD) { |
| 433 | VisitParmVarDecl(PD); |
| 434 | PD->setOriginalType(Reader.GetType(Record[Idx++])); |
| 435 | } |
| 436 | |
| 437 | void PCHDeclReader::VisitFileScopeAsmDecl(FileScopeAsmDecl *AD) { |
| 438 | VisitDecl(AD); |
Chris Lattner | da93061 | 2009-04-27 05:58:23 +0000 | [diff] [blame] | 439 | AD->setAsmString(cast<StringLiteral>(Reader.ReadDeclExpr())); |
Chris Lattner | 698f925 | 2009-04-27 05:27:42 +0000 | [diff] [blame] | 440 | } |
| 441 | |
| 442 | void PCHDeclReader::VisitBlockDecl(BlockDecl *BD) { |
| 443 | VisitDecl(BD); |
Chris Lattner | da93061 | 2009-04-27 05:58:23 +0000 | [diff] [blame] | 444 | BD->setBody(cast_or_null<CompoundStmt>(Reader.ReadDeclStmt())); |
Chris Lattner | 698f925 | 2009-04-27 05:27:42 +0000 | [diff] [blame] | 445 | unsigned NumParams = Record[Idx++]; |
| 446 | llvm::SmallVector<ParmVarDecl *, 16> Params; |
| 447 | Params.reserve(NumParams); |
| 448 | for (unsigned I = 0; I != NumParams; ++I) |
| 449 | Params.push_back(cast<ParmVarDecl>(Reader.GetDecl(Record[Idx++]))); |
Douglas Gregor | 75fdb23 | 2009-05-22 22:45:36 +0000 | [diff] [blame] | 450 | BD->setParams(*Reader.getContext(), Params.data(), NumParams); |
Chris Lattner | 698f925 | 2009-04-27 05:27:42 +0000 | [diff] [blame] | 451 | } |
| 452 | |
| 453 | std::pair<uint64_t, uint64_t> |
| 454 | PCHDeclReader::VisitDeclContext(DeclContext *DC) { |
| 455 | uint64_t LexicalOffset = Record[Idx++]; |
| 456 | uint64_t VisibleOffset = Record[Idx++]; |
| 457 | return std::make_pair(LexicalOffset, VisibleOffset); |
| 458 | } |
| 459 | |
| 460 | //===----------------------------------------------------------------------===// |
Chris Lattner | 4e3fcc8 | 2009-04-27 06:01:06 +0000 | [diff] [blame] | 461 | // Attribute Reading |
Chris Lattner | 698f925 | 2009-04-27 05:27:42 +0000 | [diff] [blame] | 462 | //===----------------------------------------------------------------------===// |
| 463 | |
Chris Lattner | 4e3fcc8 | 2009-04-27 06:01:06 +0000 | [diff] [blame] | 464 | /// \brief Reads attributes from the current stream position. |
| 465 | Attr *PCHReader::ReadAttributes() { |
| 466 | unsigned Code = DeclsCursor.ReadCode(); |
| 467 | assert(Code == llvm::bitc::UNABBREV_RECORD && |
| 468 | "Expected unabbreviated record"); (void)Code; |
| 469 | |
| 470 | RecordData Record; |
| 471 | unsigned Idx = 0; |
| 472 | unsigned RecCode = DeclsCursor.ReadRecord(Code, Record); |
| 473 | assert(RecCode == pch::DECL_ATTR && "Expected attribute record"); |
| 474 | (void)RecCode; |
| 475 | |
| 476 | #define SIMPLE_ATTR(Name) \ |
| 477 | case Attr::Name: \ |
Chris Lattner | d1d64a0 | 2009-04-27 21:45:14 +0000 | [diff] [blame] | 478 | New = ::new (*Context) Name##Attr(); \ |
Chris Lattner | 4e3fcc8 | 2009-04-27 06:01:06 +0000 | [diff] [blame] | 479 | break |
| 480 | |
| 481 | #define STRING_ATTR(Name) \ |
| 482 | case Attr::Name: \ |
Chris Lattner | d1d64a0 | 2009-04-27 21:45:14 +0000 | [diff] [blame] | 483 | New = ::new (*Context) Name##Attr(ReadString(Record, Idx)); \ |
Chris Lattner | 4e3fcc8 | 2009-04-27 06:01:06 +0000 | [diff] [blame] | 484 | break |
| 485 | |
| 486 | #define UNSIGNED_ATTR(Name) \ |
| 487 | case Attr::Name: \ |
Chris Lattner | d1d64a0 | 2009-04-27 21:45:14 +0000 | [diff] [blame] | 488 | New = ::new (*Context) Name##Attr(Record[Idx++]); \ |
Chris Lattner | 4e3fcc8 | 2009-04-27 06:01:06 +0000 | [diff] [blame] | 489 | break |
| 490 | |
| 491 | Attr *Attrs = 0; |
| 492 | while (Idx < Record.size()) { |
| 493 | Attr *New = 0; |
| 494 | Attr::Kind Kind = (Attr::Kind)Record[Idx++]; |
| 495 | bool IsInherited = Record[Idx++]; |
| 496 | |
| 497 | switch (Kind) { |
| 498 | STRING_ATTR(Alias); |
| 499 | UNSIGNED_ATTR(Aligned); |
| 500 | SIMPLE_ATTR(AlwaysInline); |
| 501 | SIMPLE_ATTR(AnalyzerNoReturn); |
| 502 | STRING_ATTR(Annotate); |
| 503 | STRING_ATTR(AsmLabel); |
| 504 | |
| 505 | case Attr::Blocks: |
Chris Lattner | d1d64a0 | 2009-04-27 21:45:14 +0000 | [diff] [blame] | 506 | New = ::new (*Context) BlocksAttr( |
Chris Lattner | 4e3fcc8 | 2009-04-27 06:01:06 +0000 | [diff] [blame] | 507 | (BlocksAttr::BlocksAttrTypes)Record[Idx++]); |
| 508 | break; |
| 509 | |
| 510 | case Attr::Cleanup: |
Chris Lattner | d1d64a0 | 2009-04-27 21:45:14 +0000 | [diff] [blame] | 511 | New = ::new (*Context) CleanupAttr( |
Chris Lattner | 4e3fcc8 | 2009-04-27 06:01:06 +0000 | [diff] [blame] | 512 | cast<FunctionDecl>(GetDecl(Record[Idx++]))); |
| 513 | break; |
| 514 | |
| 515 | SIMPLE_ATTR(Const); |
| 516 | UNSIGNED_ATTR(Constructor); |
| 517 | SIMPLE_ATTR(DLLExport); |
| 518 | SIMPLE_ATTR(DLLImport); |
| 519 | SIMPLE_ATTR(Deprecated); |
| 520 | UNSIGNED_ATTR(Destructor); |
| 521 | SIMPLE_ATTR(FastCall); |
| 522 | |
| 523 | case Attr::Format: { |
| 524 | std::string Type = ReadString(Record, Idx); |
| 525 | unsigned FormatIdx = Record[Idx++]; |
| 526 | unsigned FirstArg = Record[Idx++]; |
Chris Lattner | d1d64a0 | 2009-04-27 21:45:14 +0000 | [diff] [blame] | 527 | New = ::new (*Context) FormatAttr(Type, FormatIdx, FirstArg); |
Chris Lattner | 4e3fcc8 | 2009-04-27 06:01:06 +0000 | [diff] [blame] | 528 | break; |
| 529 | } |
Fariborz Jahanian | 5b53005 | 2009-05-13 18:09:35 +0000 | [diff] [blame] | 530 | |
Fariborz Jahanian | 5b16092 | 2009-05-20 17:41:43 +0000 | [diff] [blame] | 531 | case Attr::FormatArg: { |
| 532 | unsigned FormatIdx = Record[Idx++]; |
| 533 | New = ::new (*Context) FormatArgAttr(FormatIdx); |
| 534 | break; |
| 535 | } |
| 536 | |
Fariborz Jahanian | 5b53005 | 2009-05-13 18:09:35 +0000 | [diff] [blame] | 537 | case Attr::Sentinel: { |
| 538 | int sentinel = Record[Idx++]; |
| 539 | int nullPos = Record[Idx++]; |
| 540 | New = ::new (*Context) SentinelAttr(sentinel, nullPos); |
| 541 | break; |
| 542 | } |
Chris Lattner | 4e3fcc8 | 2009-04-27 06:01:06 +0000 | [diff] [blame] | 543 | |
| 544 | SIMPLE_ATTR(GNUInline); |
| 545 | |
| 546 | case Attr::IBOutletKind: |
Chris Lattner | d1d64a0 | 2009-04-27 21:45:14 +0000 | [diff] [blame] | 547 | New = ::new (*Context) IBOutletAttr(); |
Chris Lattner | 4e3fcc8 | 2009-04-27 06:01:06 +0000 | [diff] [blame] | 548 | break; |
| 549 | |
Ryan Flynn | 76168e2 | 2009-08-09 20:07:29 +0000 | [diff] [blame] | 550 | SIMPLE_ATTR(Malloc); |
Mike Stump | 1feade8 | 2009-08-26 22:31:08 +0000 | [diff] [blame] | 551 | SIMPLE_ATTR(NoDebug); |
| 552 | SIMPLE_ATTR(NoInline); |
Chris Lattner | 4e3fcc8 | 2009-04-27 06:01:06 +0000 | [diff] [blame] | 553 | SIMPLE_ATTR(NoReturn); |
| 554 | SIMPLE_ATTR(NoThrow); |
Chris Lattner | 4e3fcc8 | 2009-04-27 06:01:06 +0000 | [diff] [blame] | 555 | |
| 556 | case Attr::NonNull: { |
| 557 | unsigned Size = Record[Idx++]; |
| 558 | llvm::SmallVector<unsigned, 16> ArgNums; |
| 559 | ArgNums.insert(ArgNums.end(), &Record[Idx], &Record[Idx] + Size); |
| 560 | Idx += Size; |
Douglas Gregor | 75fdb23 | 2009-05-22 22:45:36 +0000 | [diff] [blame] | 561 | New = ::new (*Context) NonNullAttr(ArgNums.data(), Size); |
Chris Lattner | 4e3fcc8 | 2009-04-27 06:01:06 +0000 | [diff] [blame] | 562 | break; |
| 563 | } |
Nate Begeman | 6f3d838 | 2009-06-26 06:32:41 +0000 | [diff] [blame] | 564 | |
| 565 | case Attr::ReqdWorkGroupSize: { |
| 566 | unsigned X = Record[Idx++]; |
| 567 | unsigned Y = Record[Idx++]; |
| 568 | unsigned Z = Record[Idx++]; |
| 569 | New = ::new (*Context) ReqdWorkGroupSizeAttr(X, Y, Z); |
| 570 | break; |
| 571 | } |
Chris Lattner | 4e3fcc8 | 2009-04-27 06:01:06 +0000 | [diff] [blame] | 572 | |
| 573 | SIMPLE_ATTR(ObjCException); |
| 574 | SIMPLE_ATTR(ObjCNSObject); |
Ted Kremenek | b71368d | 2009-05-09 02:44:38 +0000 | [diff] [blame] | 575 | SIMPLE_ATTR(CFReturnsRetained); |
| 576 | SIMPLE_ATTR(NSReturnsRetained); |
Chris Lattner | 4e3fcc8 | 2009-04-27 06:01:06 +0000 | [diff] [blame] | 577 | SIMPLE_ATTR(Overloadable); |
Anders Carlsson | a860e75 | 2009-08-08 18:23:56 +0000 | [diff] [blame] | 578 | SIMPLE_ATTR(Packed); |
| 579 | UNSIGNED_ATTR(PragmaPack); |
Chris Lattner | 4e3fcc8 | 2009-04-27 06:01:06 +0000 | [diff] [blame] | 580 | SIMPLE_ATTR(Pure); |
| 581 | UNSIGNED_ATTR(Regparm); |
| 582 | STRING_ATTR(Section); |
| 583 | SIMPLE_ATTR(StdCall); |
| 584 | SIMPLE_ATTR(TransparentUnion); |
| 585 | SIMPLE_ATTR(Unavailable); |
| 586 | SIMPLE_ATTR(Unused); |
| 587 | SIMPLE_ATTR(Used); |
| 588 | |
| 589 | case Attr::Visibility: |
Chris Lattner | d1d64a0 | 2009-04-27 21:45:14 +0000 | [diff] [blame] | 590 | New = ::new (*Context) VisibilityAttr( |
Chris Lattner | 4e3fcc8 | 2009-04-27 06:01:06 +0000 | [diff] [blame] | 591 | (VisibilityAttr::VisibilityTypes)Record[Idx++]); |
| 592 | break; |
| 593 | |
| 594 | SIMPLE_ATTR(WarnUnusedResult); |
| 595 | SIMPLE_ATTR(Weak); |
| 596 | SIMPLE_ATTR(WeakImport); |
| 597 | } |
| 598 | |
| 599 | assert(New && "Unable to decode attribute?"); |
| 600 | New->setInherited(IsInherited); |
| 601 | New->setNext(Attrs); |
| 602 | Attrs = New; |
| 603 | } |
| 604 | #undef UNSIGNED_ATTR |
| 605 | #undef STRING_ATTR |
| 606 | #undef SIMPLE_ATTR |
| 607 | |
| 608 | // The list of attributes was built backwards. Reverse the list |
| 609 | // before returning it. |
| 610 | Attr *PrevAttr = 0, *NextAttr = 0; |
| 611 | while (Attrs) { |
| 612 | NextAttr = Attrs->getNext(); |
| 613 | Attrs->setNext(PrevAttr); |
| 614 | PrevAttr = Attrs; |
| 615 | Attrs = NextAttr; |
| 616 | } |
| 617 | |
| 618 | return PrevAttr; |
| 619 | } |
| 620 | |
| 621 | //===----------------------------------------------------------------------===// |
| 622 | // PCHReader Implementation |
| 623 | //===----------------------------------------------------------------------===// |
Chris Lattner | 698f925 | 2009-04-27 05:27:42 +0000 | [diff] [blame] | 624 | |
| 625 | /// \brief Note that we have loaded the declaration with the given |
| 626 | /// Index. |
| 627 | /// |
| 628 | /// This routine notes that this declaration has already been loaded, |
| 629 | /// so that future GetDecl calls will return this declaration rather |
| 630 | /// than trying to load a new declaration. |
| 631 | inline void PCHReader::LoadedDecl(unsigned Index, Decl *D) { |
| 632 | assert(!DeclsLoaded[Index] && "Decl loaded twice?"); |
| 633 | DeclsLoaded[Index] = D; |
| 634 | } |
| 635 | |
| 636 | |
| 637 | /// \brief Determine whether the consumer will be interested in seeing |
| 638 | /// this declaration (via HandleTopLevelDecl). |
| 639 | /// |
| 640 | /// This routine should return true for anything that might affect |
| 641 | /// code generation, e.g., inline function definitions, Objective-C |
| 642 | /// declarations with metadata, etc. |
| 643 | static bool isConsumerInterestedIn(Decl *D) { |
| 644 | if (VarDecl *Var = dyn_cast<VarDecl>(D)) |
| 645 | return Var->isFileVarDecl() && Var->getInit(); |
| 646 | if (FunctionDecl *Func = dyn_cast<FunctionDecl>(D)) |
| 647 | return Func->isThisDeclarationADefinition(); |
| 648 | return isa<ObjCProtocolDecl>(D); |
| 649 | } |
| 650 | |
| 651 | /// \brief Read the declaration at the given offset from the PCH file. |
| 652 | Decl *PCHReader::ReadDeclRecord(uint64_t Offset, unsigned Index) { |
| 653 | // Keep track of where we are in the stream, then jump back there |
| 654 | // after reading this declaration. |
Chris Lattner | da93061 | 2009-04-27 05:58:23 +0000 | [diff] [blame] | 655 | SavedStreamPosition SavedPosition(DeclsCursor); |
Chris Lattner | 698f925 | 2009-04-27 05:27:42 +0000 | [diff] [blame] | 656 | |
Douglas Gregor | d89275b | 2009-07-06 18:54:52 +0000 | [diff] [blame] | 657 | // Note that we are loading a declaration record. |
| 658 | LoadingTypeOrDecl Loading(*this); |
| 659 | |
Chris Lattner | da93061 | 2009-04-27 05:58:23 +0000 | [diff] [blame] | 660 | DeclsCursor.JumpToBit(Offset); |
Chris Lattner | 698f925 | 2009-04-27 05:27:42 +0000 | [diff] [blame] | 661 | RecordData Record; |
Chris Lattner | da93061 | 2009-04-27 05:58:23 +0000 | [diff] [blame] | 662 | unsigned Code = DeclsCursor.ReadCode(); |
Chris Lattner | 698f925 | 2009-04-27 05:27:42 +0000 | [diff] [blame] | 663 | unsigned Idx = 0; |
| 664 | PCHDeclReader Reader(*this, Record, Idx); |
| 665 | |
Chris Lattner | da93061 | 2009-04-27 05:58:23 +0000 | [diff] [blame] | 666 | Decl *D = 0; |
| 667 | switch ((pch::DeclCode)DeclsCursor.ReadRecord(Code, Record)) { |
Chris Lattner | 698f925 | 2009-04-27 05:27:42 +0000 | [diff] [blame] | 668 | case pch::DECL_ATTR: |
| 669 | case pch::DECL_CONTEXT_LEXICAL: |
| 670 | case pch::DECL_CONTEXT_VISIBLE: |
| 671 | assert(false && "Record cannot be de-serialized with ReadDeclRecord"); |
| 672 | break; |
Chris Lattner | 698f925 | 2009-04-27 05:27:42 +0000 | [diff] [blame] | 673 | case pch::DECL_TRANSLATION_UNIT: |
| 674 | assert(Index == 0 && "Translation unit must be at index 0"); |
Chris Lattner | d1d64a0 | 2009-04-27 21:45:14 +0000 | [diff] [blame] | 675 | D = Context->getTranslationUnitDecl(); |
Chris Lattner | 698f925 | 2009-04-27 05:27:42 +0000 | [diff] [blame] | 676 | break; |
Chris Lattner | 4e3fcc8 | 2009-04-27 06:01:06 +0000 | [diff] [blame] | 677 | case pch::DECL_TYPEDEF: |
Chris Lattner | d1d64a0 | 2009-04-27 21:45:14 +0000 | [diff] [blame] | 678 | D = TypedefDecl::Create(*Context, 0, SourceLocation(), 0, QualType()); |
Chris Lattner | 698f925 | 2009-04-27 05:27:42 +0000 | [diff] [blame] | 679 | break; |
Chris Lattner | 4e3fcc8 | 2009-04-27 06:01:06 +0000 | [diff] [blame] | 680 | case pch::DECL_ENUM: |
Douglas Gregor | 741dd9a | 2009-07-21 14:46:17 +0000 | [diff] [blame] | 681 | D = EnumDecl::Create(*Context, 0, SourceLocation(), 0, SourceLocation(), 0); |
Chris Lattner | 698f925 | 2009-04-27 05:27:42 +0000 | [diff] [blame] | 682 | break; |
Chris Lattner | 4e3fcc8 | 2009-04-27 06:01:06 +0000 | [diff] [blame] | 683 | case pch::DECL_RECORD: |
Chris Lattner | d1d64a0 | 2009-04-27 21:45:14 +0000 | [diff] [blame] | 684 | D = RecordDecl::Create(*Context, TagDecl::TK_struct, 0, SourceLocation(), |
Douglas Gregor | 741dd9a | 2009-07-21 14:46:17 +0000 | [diff] [blame] | 685 | 0, SourceLocation(), 0); |
Chris Lattner | 698f925 | 2009-04-27 05:27:42 +0000 | [diff] [blame] | 686 | break; |
Chris Lattner | 4e3fcc8 | 2009-04-27 06:01:06 +0000 | [diff] [blame] | 687 | case pch::DECL_ENUM_CONSTANT: |
Chris Lattner | d1d64a0 | 2009-04-27 21:45:14 +0000 | [diff] [blame] | 688 | D = EnumConstantDecl::Create(*Context, 0, SourceLocation(), 0, QualType(), |
Chris Lattner | 698f925 | 2009-04-27 05:27:42 +0000 | [diff] [blame] | 689 | 0, llvm::APSInt()); |
| 690 | break; |
Chris Lattner | 4e3fcc8 | 2009-04-27 06:01:06 +0000 | [diff] [blame] | 691 | case pch::DECL_FUNCTION: |
Chris Lattner | d1d64a0 | 2009-04-27 21:45:14 +0000 | [diff] [blame] | 692 | D = FunctionDecl::Create(*Context, 0, SourceLocation(), DeclarationName(), |
Argyrios Kyrtzidis | a1d5662 | 2009-08-19 01:27:57 +0000 | [diff] [blame] | 693 | QualType(), 0); |
Chris Lattner | 698f925 | 2009-04-27 05:27:42 +0000 | [diff] [blame] | 694 | break; |
Chris Lattner | 4e3fcc8 | 2009-04-27 06:01:06 +0000 | [diff] [blame] | 695 | case pch::DECL_OBJC_METHOD: |
Chris Lattner | d1d64a0 | 2009-04-27 21:45:14 +0000 | [diff] [blame] | 696 | D = ObjCMethodDecl::Create(*Context, SourceLocation(), SourceLocation(), |
Chris Lattner | 698f925 | 2009-04-27 05:27:42 +0000 | [diff] [blame] | 697 | Selector(), QualType(), 0); |
| 698 | break; |
Chris Lattner | 4e3fcc8 | 2009-04-27 06:01:06 +0000 | [diff] [blame] | 699 | case pch::DECL_OBJC_INTERFACE: |
Chris Lattner | d1d64a0 | 2009-04-27 21:45:14 +0000 | [diff] [blame] | 700 | D = ObjCInterfaceDecl::Create(*Context, 0, SourceLocation(), 0); |
Chris Lattner | 698f925 | 2009-04-27 05:27:42 +0000 | [diff] [blame] | 701 | break; |
Chris Lattner | 4e3fcc8 | 2009-04-27 06:01:06 +0000 | [diff] [blame] | 702 | case pch::DECL_OBJC_IVAR: |
Argyrios Kyrtzidis | a1d5662 | 2009-08-19 01:27:57 +0000 | [diff] [blame] | 703 | D = ObjCIvarDecl::Create(*Context, 0, SourceLocation(), 0, QualType(), 0, |
Chris Lattner | 698f925 | 2009-04-27 05:27:42 +0000 | [diff] [blame] | 704 | ObjCIvarDecl::None); |
| 705 | break; |
Chris Lattner | 4e3fcc8 | 2009-04-27 06:01:06 +0000 | [diff] [blame] | 706 | case pch::DECL_OBJC_PROTOCOL: |
Chris Lattner | d1d64a0 | 2009-04-27 21:45:14 +0000 | [diff] [blame] | 707 | D = ObjCProtocolDecl::Create(*Context, 0, SourceLocation(), 0); |
Chris Lattner | 698f925 | 2009-04-27 05:27:42 +0000 | [diff] [blame] | 708 | break; |
Chris Lattner | 4e3fcc8 | 2009-04-27 06:01:06 +0000 | [diff] [blame] | 709 | case pch::DECL_OBJC_AT_DEFS_FIELD: |
Chris Lattner | d1d64a0 | 2009-04-27 21:45:14 +0000 | [diff] [blame] | 710 | D = ObjCAtDefsFieldDecl::Create(*Context, 0, SourceLocation(), 0, |
Chris Lattner | 698f925 | 2009-04-27 05:27:42 +0000 | [diff] [blame] | 711 | QualType(), 0); |
| 712 | break; |
Chris Lattner | 4e3fcc8 | 2009-04-27 06:01:06 +0000 | [diff] [blame] | 713 | case pch::DECL_OBJC_CLASS: |
Chris Lattner | d1d64a0 | 2009-04-27 21:45:14 +0000 | [diff] [blame] | 714 | D = ObjCClassDecl::Create(*Context, 0, SourceLocation()); |
Chris Lattner | 698f925 | 2009-04-27 05:27:42 +0000 | [diff] [blame] | 715 | break; |
Chris Lattner | 4e3fcc8 | 2009-04-27 06:01:06 +0000 | [diff] [blame] | 716 | case pch::DECL_OBJC_FORWARD_PROTOCOL: |
Chris Lattner | d1d64a0 | 2009-04-27 21:45:14 +0000 | [diff] [blame] | 717 | D = ObjCForwardProtocolDecl::Create(*Context, 0, SourceLocation()); |
Chris Lattner | 698f925 | 2009-04-27 05:27:42 +0000 | [diff] [blame] | 718 | break; |
Chris Lattner | 4e3fcc8 | 2009-04-27 06:01:06 +0000 | [diff] [blame] | 719 | case pch::DECL_OBJC_CATEGORY: |
Chris Lattner | d1d64a0 | 2009-04-27 21:45:14 +0000 | [diff] [blame] | 720 | D = ObjCCategoryDecl::Create(*Context, 0, SourceLocation(), 0); |
Chris Lattner | 698f925 | 2009-04-27 05:27:42 +0000 | [diff] [blame] | 721 | break; |
Chris Lattner | 4e3fcc8 | 2009-04-27 06:01:06 +0000 | [diff] [blame] | 722 | case pch::DECL_OBJC_CATEGORY_IMPL: |
Chris Lattner | d1d64a0 | 2009-04-27 21:45:14 +0000 | [diff] [blame] | 723 | D = ObjCCategoryImplDecl::Create(*Context, 0, SourceLocation(), 0, 0); |
Chris Lattner | 698f925 | 2009-04-27 05:27:42 +0000 | [diff] [blame] | 724 | break; |
Chris Lattner | 4e3fcc8 | 2009-04-27 06:01:06 +0000 | [diff] [blame] | 725 | case pch::DECL_OBJC_IMPLEMENTATION: |
Chris Lattner | d1d64a0 | 2009-04-27 21:45:14 +0000 | [diff] [blame] | 726 | D = ObjCImplementationDecl::Create(*Context, 0, SourceLocation(), 0, 0); |
Chris Lattner | 698f925 | 2009-04-27 05:27:42 +0000 | [diff] [blame] | 727 | break; |
Chris Lattner | 4e3fcc8 | 2009-04-27 06:01:06 +0000 | [diff] [blame] | 728 | case pch::DECL_OBJC_COMPATIBLE_ALIAS: |
Chris Lattner | d1d64a0 | 2009-04-27 21:45:14 +0000 | [diff] [blame] | 729 | D = ObjCCompatibleAliasDecl::Create(*Context, 0, SourceLocation(), 0, 0); |
Chris Lattner | 698f925 | 2009-04-27 05:27:42 +0000 | [diff] [blame] | 730 | break; |
Chris Lattner | 4e3fcc8 | 2009-04-27 06:01:06 +0000 | [diff] [blame] | 731 | case pch::DECL_OBJC_PROPERTY: |
Chris Lattner | d1d64a0 | 2009-04-27 21:45:14 +0000 | [diff] [blame] | 732 | D = ObjCPropertyDecl::Create(*Context, 0, SourceLocation(), 0, QualType()); |
Chris Lattner | 698f925 | 2009-04-27 05:27:42 +0000 | [diff] [blame] | 733 | break; |
Chris Lattner | 4e3fcc8 | 2009-04-27 06:01:06 +0000 | [diff] [blame] | 734 | case pch::DECL_OBJC_PROPERTY_IMPL: |
Chris Lattner | d1d64a0 | 2009-04-27 21:45:14 +0000 | [diff] [blame] | 735 | D = ObjCPropertyImplDecl::Create(*Context, 0, SourceLocation(), |
Chris Lattner | 698f925 | 2009-04-27 05:27:42 +0000 | [diff] [blame] | 736 | SourceLocation(), 0, |
| 737 | ObjCPropertyImplDecl::Dynamic, 0); |
| 738 | break; |
Chris Lattner | 4e3fcc8 | 2009-04-27 06:01:06 +0000 | [diff] [blame] | 739 | case pch::DECL_FIELD: |
Argyrios Kyrtzidis | a1d5662 | 2009-08-19 01:27:57 +0000 | [diff] [blame] | 740 | D = FieldDecl::Create(*Context, 0, SourceLocation(), 0, QualType(), 0, 0, |
Argyrios Kyrtzidis | a5d8200 | 2009-08-21 00:31:54 +0000 | [diff] [blame] | 741 | false); |
Chris Lattner | 698f925 | 2009-04-27 05:27:42 +0000 | [diff] [blame] | 742 | break; |
Chris Lattner | 698f925 | 2009-04-27 05:27:42 +0000 | [diff] [blame] | 743 | case pch::DECL_VAR: |
Argyrios Kyrtzidis | a1d5662 | 2009-08-19 01:27:57 +0000 | [diff] [blame] | 744 | D = VarDecl::Create(*Context, 0, SourceLocation(), 0, QualType(), 0, |
Argyrios Kyrtzidis | a5d8200 | 2009-08-21 00:31:54 +0000 | [diff] [blame] | 745 | VarDecl::None); |
Chris Lattner | 698f925 | 2009-04-27 05:27:42 +0000 | [diff] [blame] | 746 | break; |
| 747 | |
| 748 | case pch::DECL_IMPLICIT_PARAM: |
Chris Lattner | d1d64a0 | 2009-04-27 21:45:14 +0000 | [diff] [blame] | 749 | D = ImplicitParamDecl::Create(*Context, 0, SourceLocation(), 0, QualType()); |
Chris Lattner | 698f925 | 2009-04-27 05:27:42 +0000 | [diff] [blame] | 750 | break; |
| 751 | |
Chris Lattner | 4e3fcc8 | 2009-04-27 06:01:06 +0000 | [diff] [blame] | 752 | case pch::DECL_PARM_VAR: |
Argyrios Kyrtzidis | a1d5662 | 2009-08-19 01:27:57 +0000 | [diff] [blame] | 753 | D = ParmVarDecl::Create(*Context, 0, SourceLocation(), 0, QualType(), 0, |
Chris Lattner | 698f925 | 2009-04-27 05:27:42 +0000 | [diff] [blame] | 754 | VarDecl::None, 0); |
| 755 | break; |
Chris Lattner | 4e3fcc8 | 2009-04-27 06:01:06 +0000 | [diff] [blame] | 756 | case pch::DECL_ORIGINAL_PARM_VAR: |
Chris Lattner | d1d64a0 | 2009-04-27 21:45:14 +0000 | [diff] [blame] | 757 | D = OriginalParmVarDecl::Create(*Context, 0, SourceLocation(), 0, |
Argyrios Kyrtzidis | a1d5662 | 2009-08-19 01:27:57 +0000 | [diff] [blame] | 758 | QualType(),0, QualType(), VarDecl::None, 0); |
Chris Lattner | 698f925 | 2009-04-27 05:27:42 +0000 | [diff] [blame] | 759 | break; |
Chris Lattner | 4e3fcc8 | 2009-04-27 06:01:06 +0000 | [diff] [blame] | 760 | case pch::DECL_FILE_SCOPE_ASM: |
Chris Lattner | d1d64a0 | 2009-04-27 21:45:14 +0000 | [diff] [blame] | 761 | D = FileScopeAsmDecl::Create(*Context, 0, SourceLocation(), 0); |
Chris Lattner | 698f925 | 2009-04-27 05:27:42 +0000 | [diff] [blame] | 762 | break; |
Chris Lattner | 4e3fcc8 | 2009-04-27 06:01:06 +0000 | [diff] [blame] | 763 | case pch::DECL_BLOCK: |
Chris Lattner | d1d64a0 | 2009-04-27 21:45:14 +0000 | [diff] [blame] | 764 | D = BlockDecl::Create(*Context, 0, SourceLocation()); |
Chris Lattner | 698f925 | 2009-04-27 05:27:42 +0000 | [diff] [blame] | 765 | break; |
| 766 | } |
Chris Lattner | 698f925 | 2009-04-27 05:27:42 +0000 | [diff] [blame] | 767 | |
| 768 | assert(D && "Unknown declaration reading PCH file"); |
Chris Lattner | 4e3fcc8 | 2009-04-27 06:01:06 +0000 | [diff] [blame] | 769 | LoadedDecl(Index, D); |
| 770 | Reader.Visit(D); |
Chris Lattner | 698f925 | 2009-04-27 05:27:42 +0000 | [diff] [blame] | 771 | |
| 772 | // If this declaration is also a declaration context, get the |
| 773 | // offsets for its tables of lexical and visible declarations. |
| 774 | if (DeclContext *DC = dyn_cast<DeclContext>(D)) { |
| 775 | std::pair<uint64_t, uint64_t> Offsets = Reader.VisitDeclContext(DC); |
| 776 | if (Offsets.first || Offsets.second) { |
| 777 | DC->setHasExternalLexicalStorage(Offsets.first != 0); |
| 778 | DC->setHasExternalVisibleStorage(Offsets.second != 0); |
| 779 | DeclContextOffsets[DC] = Offsets; |
| 780 | } |
| 781 | } |
| 782 | assert(Idx == Record.size()); |
| 783 | |
| 784 | // If we have deserialized a declaration that has a definition the |
| 785 | // AST consumer might need to know about, notify the consumer |
| 786 | // about that definition now or queue it for later. |
| 787 | if (isConsumerInterestedIn(D)) { |
| 788 | if (Consumer) { |
| 789 | DeclGroupRef DG(D); |
| 790 | Consumer->HandleTopLevelDecl(DG); |
| 791 | } else { |
| 792 | InterestingDecls.push_back(D); |
| 793 | } |
| 794 | } |
| 795 | |
| 796 | return D; |
| 797 | } |
| 798 | |