1 //===--- DeclObjC.cpp - ObjC Declaration AST Node Implementation ----------===// 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 Objective-C related Decl classes. 11 // 12 //===----------------------------------------------------------------------===// 13 14 #include "clang/AST/DeclObjC.h" 15 #include "clang/AST/ASTContext.h" 16 #include "clang/AST/Stmt.h" 17 #include "clang/AST/ASTMutationListener.h" 18 #include "llvm/ADT/STLExtras.h" 19 using namespace clang; 20 21 //===----------------------------------------------------------------------===// 22 // ObjCListBase 23 //===----------------------------------------------------------------------===// 24 25 void ObjCListBase::set(void *const* InList, unsigned Elts, ASTContext &Ctx) { 26 List = 0; 27 if (Elts == 0) return; // Setting to an empty list is a noop. 28 29 30 List = new (Ctx) void*[Elts]; 31 NumElts = Elts; 32 memcpy(List, InList, sizeof(void*)*Elts); 33 } 34 35 void ObjCProtocolList::set(ObjCProtocolDecl* const* InList, unsigned Elts, 36 const SourceLocation *Locs, ASTContext &Ctx) { 37 if (Elts == 0) 38 return; 39 40 Locations = new (Ctx) SourceLocation[Elts]; 41 memcpy(Locations, Locs, sizeof(SourceLocation) * Elts); 42 set(InList, Elts, Ctx); 43 } 44 45 //===----------------------------------------------------------------------===// 46 // ObjCInterfaceDecl 47 //===----------------------------------------------------------------------===// 48 49 /// getIvarDecl - This method looks up an ivar in this ContextDecl. 50 /// 51 ObjCIvarDecl * 52 ObjCContainerDecl::getIvarDecl(IdentifierInfo *Id) const { 53 lookup_const_iterator Ivar, IvarEnd; 54 for (llvm::tie(Ivar, IvarEnd) = lookup(Id); Ivar != IvarEnd; ++Ivar) { 55 if (ObjCIvarDecl *ivar = dyn_cast<ObjCIvarDecl>(*Ivar)) 56 return ivar; 57 } 58 return 0; 59 } 60 61 // Get the local instance/class method declared in this interface. 62 ObjCMethodDecl * 63 ObjCContainerDecl::getMethod(Selector Sel, bool isInstance) const { 64 // Since instance & class methods can have the same name, the loop below 65 // ensures we get the correct method. 66 // 67 // @interface Whatever 68 // - (int) class_method; 69 // + (float) class_method; 70 // @end 71 // 72 lookup_const_iterator Meth, MethEnd; 73 for (llvm::tie(Meth, MethEnd) = lookup(Sel); Meth != MethEnd; ++Meth) { 74 ObjCMethodDecl *MD = dyn_cast<ObjCMethodDecl>(*Meth); 75 if (MD && MD->isInstanceMethod() == isInstance) 76 return MD; 77 } 78 return 0; 79 } 80 81 ObjCPropertyDecl * 82 ObjCPropertyDecl::findPropertyDecl(const DeclContext *DC, 83 IdentifierInfo *propertyID) { 84 85 DeclContext::lookup_const_iterator I, E; 86 llvm::tie(I, E) = DC->lookup(propertyID); 87 for ( ; I != E; ++I) 88 if (ObjCPropertyDecl *PD = dyn_cast<ObjCPropertyDecl>(*I)) 89 return PD; 90 91 return 0; 92 } 93 94 /// FindPropertyDeclaration - Finds declaration of the property given its name 95 /// in 'PropertyId' and returns it. It returns 0, if not found. 96 ObjCPropertyDecl * 97 ObjCContainerDecl::FindPropertyDeclaration(IdentifierInfo *PropertyId) const { 98 99 if (ObjCPropertyDecl *PD = 100 ObjCPropertyDecl::findPropertyDecl(cast<DeclContext>(this), PropertyId)) 101 return PD; 102 103 switch (getKind()) { 104 default: 105 break; 106 case Decl::ObjCProtocol: { 107 const ObjCProtocolDecl *PID = cast<ObjCProtocolDecl>(this); 108 for (ObjCProtocolDecl::protocol_iterator I = PID->protocol_begin(), 109 E = PID->protocol_end(); I != E; ++I) 110 if (ObjCPropertyDecl *P = (*I)->FindPropertyDeclaration(PropertyId)) 111 return P; 112 break; 113 } 114 case Decl::ObjCInterface: { 115 const ObjCInterfaceDecl *OID = cast<ObjCInterfaceDecl>(this); 116 // Look through categories. 117 for (ObjCCategoryDecl *Cat = OID->getCategoryList(); 118 Cat; Cat = Cat->getNextClassCategory()) 119 if (!Cat->IsClassExtension()) 120 if (ObjCPropertyDecl *P = Cat->FindPropertyDeclaration(PropertyId)) 121 return P; 122 123 // Look through protocols. 124 for (ObjCInterfaceDecl::all_protocol_iterator 125 I = OID->all_referenced_protocol_begin(), 126 E = OID->all_referenced_protocol_end(); I != E; ++I) 127 if (ObjCPropertyDecl *P = (*I)->FindPropertyDeclaration(PropertyId)) 128 return P; 129 130 // Finally, check the super class. 131 if (const ObjCInterfaceDecl *superClass = OID->getSuperClass()) 132 return superClass->FindPropertyDeclaration(PropertyId); 133 break; 134 } 135 case Decl::ObjCCategory: { 136 const ObjCCategoryDecl *OCD = cast<ObjCCategoryDecl>(this); 137 // Look through protocols. 138 if (!OCD->IsClassExtension()) 139 for (ObjCCategoryDecl::protocol_iterator 140 I = OCD->protocol_begin(), E = OCD->protocol_end(); I != E; ++I) 141 if (ObjCPropertyDecl *P = (*I)->FindPropertyDeclaration(PropertyId)) 142 return P; 143 144 break; 145 } 146 } 147 return 0; 148 } 149 150 /// FindPropertyVisibleInPrimaryClass - Finds declaration of the property 151 /// with name 'PropertyId' in the primary class; including those in protocols 152 /// (direct or indirect) used by the primary class. 153 /// 154 ObjCPropertyDecl * 155 ObjCInterfaceDecl::FindPropertyVisibleInPrimaryClass( 156 IdentifierInfo *PropertyId) const { 157 if (ExternallyCompleted) 158 LoadExternalDefinition(); 159 160 if (ObjCPropertyDecl *PD = 161 ObjCPropertyDecl::findPropertyDecl(cast<DeclContext>(this), PropertyId)) 162 return PD; 163 164 // Look through protocols. 165 for (ObjCInterfaceDecl::all_protocol_iterator 166 I = all_referenced_protocol_begin(), 167 E = all_referenced_protocol_end(); I != E; ++I) 168 if (ObjCPropertyDecl *P = (*I)->FindPropertyDeclaration(PropertyId)) 169 return P; 170 171 return 0; 172 } 173 174 void ObjCInterfaceDecl::mergeClassExtensionProtocolList( 175 ObjCProtocolDecl *const* ExtList, unsigned ExtNum, 176 ASTContext &C) 177 { 178 if (ExternallyCompleted) 179 LoadExternalDefinition(); 180 181 if (AllReferencedProtocols.empty() && ReferencedProtocols.empty()) { 182 AllReferencedProtocols.set(ExtList, ExtNum, C); 183 return; 184 } 185 186 // Check for duplicate protocol in class's protocol list. 187 // This is O(n*m). But it is extremely rare and number of protocols in 188 // class or its extension are very few. 189 SmallVector<ObjCProtocolDecl*, 8> ProtocolRefs; 190 for (unsigned i = 0; i < ExtNum; i++) { 191 bool protocolExists = false; 192 ObjCProtocolDecl *ProtoInExtension = ExtList[i]; 193 for (all_protocol_iterator 194 p = all_referenced_protocol_begin(), 195 e = all_referenced_protocol_end(); p != e; ++p) { 196 ObjCProtocolDecl *Proto = (*p); 197 if (C.ProtocolCompatibleWithProtocol(ProtoInExtension, Proto)) { 198 protocolExists = true; 199 break; 200 } 201 } 202 // Do we want to warn on a protocol in extension class which 203 // already exist in the class? Probably not. 204 if (!protocolExists) 205 ProtocolRefs.push_back(ProtoInExtension); 206 } 207 208 if (ProtocolRefs.empty()) 209 return; 210 211 // Merge ProtocolRefs into class's protocol list; 212 for (all_protocol_iterator p = all_referenced_protocol_begin(), 213 e = all_referenced_protocol_end(); p != e; ++p) { 214 ProtocolRefs.push_back(*p); 215 } 216 217 AllReferencedProtocols.set(ProtocolRefs.data(), ProtocolRefs.size(), C); 218 } 219 220 void ObjCInterfaceDecl::completedForwardDecl() { 221 assert(isForwardDecl() && "Only valid to call for forward refs"); 222 ForwardDecl = false; 223 if (ASTMutationListener *L = getASTContext().getASTMutationListener()) 224 L->CompletedObjCForwardRef(this); 225 } 226 227 /// getFirstClassExtension - Find first class extension of the given class. 228 ObjCCategoryDecl* ObjCInterfaceDecl::getFirstClassExtension() const { 229 for (ObjCCategoryDecl *CDecl = getCategoryList(); CDecl; 230 CDecl = CDecl->getNextClassCategory()) 231 if (CDecl->IsClassExtension()) 232 return CDecl; 233 return 0; 234 } 235 236 /// getNextClassCategory - Find next class extension in list of categories. 237 const ObjCCategoryDecl* ObjCCategoryDecl::getNextClassExtension() const { 238 for (const ObjCCategoryDecl *CDecl = getNextClassCategory(); CDecl; 239 CDecl = CDecl->getNextClassCategory()) 240 if (CDecl->IsClassExtension()) 241 return CDecl; 242 return 0; 243 } 244 245 ObjCIvarDecl *ObjCInterfaceDecl::lookupInstanceVariable(IdentifierInfo *ID, 246 ObjCInterfaceDecl *&clsDeclared) { 247 if (ExternallyCompleted) 248 LoadExternalDefinition(); 249 250 ObjCInterfaceDecl* ClassDecl = this; 251 while (ClassDecl != NULL) { 252 if (ObjCIvarDecl *I = ClassDecl->getIvarDecl(ID)) { 253 clsDeclared = ClassDecl; 254 return I; 255 } 256 for (const ObjCCategoryDecl *CDecl = ClassDecl->getFirstClassExtension(); 257 CDecl; CDecl = CDecl->getNextClassExtension()) { 258 if (ObjCIvarDecl *I = CDecl->getIvarDecl(ID)) { 259 clsDeclared = ClassDecl; 260 return I; 261 } 262 } 263 264 ClassDecl = ClassDecl->getSuperClass(); 265 } 266 return NULL; 267 } 268 269 /// lookupInheritedClass - This method returns ObjCInterfaceDecl * of the super 270 /// class whose name is passed as argument. If it is not one of the super classes 271 /// the it returns NULL. 272 ObjCInterfaceDecl *ObjCInterfaceDecl::lookupInheritedClass( 273 const IdentifierInfo*ICName) { 274 if (ExternallyCompleted) 275 LoadExternalDefinition(); 276 277 ObjCInterfaceDecl* ClassDecl = this; 278 while (ClassDecl != NULL) { 279 if (ClassDecl->getIdentifier() == ICName) 280 return ClassDecl; 281 ClassDecl = ClassDecl->getSuperClass(); 282 } 283 return NULL; 284 } 285 286 /// lookupMethod - This method returns an instance/class method by looking in 287 /// the class, its categories, and its super classes (using a linear search). 288 ObjCMethodDecl *ObjCInterfaceDecl::lookupMethod(Selector Sel, 289 bool isInstance) const { 290 const ObjCInterfaceDecl* ClassDecl = this; 291 ObjCMethodDecl *MethodDecl = 0; 292 293 if (ExternallyCompleted) 294 LoadExternalDefinition(); 295 296 while (ClassDecl != NULL) { 297 if ((MethodDecl = ClassDecl->getMethod(Sel, isInstance))) 298 return MethodDecl; 299 300 // Didn't find one yet - look through protocols. 301 const ObjCList<ObjCProtocolDecl> &Protocols = 302 ClassDecl->getReferencedProtocols(); 303 for (ObjCList<ObjCProtocolDecl>::iterator I = Protocols.begin(), 304 E = Protocols.end(); I != E; ++I) 305 if ((MethodDecl = (*I)->lookupMethod(Sel, isInstance))) 306 return MethodDecl; 307 308 // Didn't find one yet - now look through categories. 309 ObjCCategoryDecl *CatDecl = ClassDecl->getCategoryList(); 310 while (CatDecl) { 311 if ((MethodDecl = CatDecl->getMethod(Sel, isInstance))) 312 return MethodDecl; 313 314 // Didn't find one yet - look through protocols. 315 const ObjCList<ObjCProtocolDecl> &Protocols = 316 CatDecl->getReferencedProtocols(); 317 for (ObjCList<ObjCProtocolDecl>::iterator I = Protocols.begin(), 318 E = Protocols.end(); I != E; ++I) 319 if ((MethodDecl = (*I)->lookupMethod(Sel, isInstance))) 320 return MethodDecl; 321 CatDecl = CatDecl->getNextClassCategory(); 322 } 323 ClassDecl = ClassDecl->getSuperClass(); 324 } 325 return NULL; 326 } 327 328 ObjCMethodDecl *ObjCInterfaceDecl::lookupPrivateMethod( 329 const Selector &Sel, 330 bool Instance) { 331 if (ExternallyCompleted) 332 LoadExternalDefinition(); 333 334 ObjCMethodDecl *Method = 0; 335 if (ObjCImplementationDecl *ImpDecl = getImplementation()) 336 Method = Instance ? ImpDecl->getInstanceMethod(Sel) 337 : ImpDecl->getClassMethod(Sel); 338 339 if (!Method && getSuperClass()) 340 return getSuperClass()->lookupPrivateMethod(Sel, Instance); 341 return Method; 342 } 343 344 //===----------------------------------------------------------------------===// 345 // ObjCMethodDecl 346 //===----------------------------------------------------------------------===// 347 348 ObjCMethodDecl *ObjCMethodDecl::Create(ASTContext &C, 349 SourceLocation beginLoc, 350 SourceLocation endLoc, 351 Selector SelInfo, QualType T, 352 TypeSourceInfo *ResultTInfo, 353 DeclContext *contextDecl, 354 bool isInstance, 355 bool isVariadic, 356 bool isSynthesized, 357 bool isImplicitlyDeclared, 358 bool isDefined, 359 ImplementationControl impControl, 360 bool HasRelatedResultType) { 361 return new (C) ObjCMethodDecl(beginLoc, endLoc, 362 SelInfo, T, ResultTInfo, contextDecl, 363 isInstance, 364 isVariadic, isSynthesized, isImplicitlyDeclared, 365 isDefined, 366 impControl, 367 HasRelatedResultType); 368 } 369 370 void ObjCMethodDecl::setAsRedeclaration(const ObjCMethodDecl *PrevMethod) { 371 assert(PrevMethod); 372 getASTContext().setObjCMethodRedeclaration(PrevMethod, this); 373 IsRedeclaration = true; 374 PrevMethod->HasRedeclaration = true; 375 } 376 377 void ObjCMethodDecl::setParamsAndSelLocs(ASTContext &C, 378 ArrayRef<ParmVarDecl*> Params, 379 ArrayRef<SourceLocation> SelLocs) { 380 ParamsAndSelLocs = 0; 381 NumParams = Params.size(); 382 if (Params.empty() && SelLocs.empty()) 383 return; 384 385 unsigned Size = sizeof(ParmVarDecl *) * NumParams + 386 sizeof(SourceLocation) * SelLocs.size(); 387 ParamsAndSelLocs = C.Allocate(Size); 388 std::copy(Params.begin(), Params.end(), getParams()); 389 std::copy(SelLocs.begin(), SelLocs.end(), getStoredSelLocs()); 390 } 391 392 void ObjCMethodDecl::getSelectorLocs( 393 SmallVectorImpl<SourceLocation> &SelLocs) const { 394 for (unsigned i = 0, e = getNumSelectorLocs(); i != e; ++i) 395 SelLocs.push_back(getSelectorLoc(i)); 396 } 397 398 void ObjCMethodDecl::setMethodParams(ASTContext &C, 399 ArrayRef<ParmVarDecl*> Params, 400 ArrayRef<SourceLocation> SelLocs) { 401 assert((!SelLocs.empty() || isImplicit()) && 402 "No selector locs for non-implicit method"); 403 if (isImplicit()) 404 return setParamsAndSelLocs(C, Params, ArrayRef<SourceLocation>()); 405 406 SelLocsKind = hasStandardSelectorLocs(getSelector(), SelLocs, Params, EndLoc); 407 if (SelLocsKind != SelLoc_NonStandard) 408 return setParamsAndSelLocs(C, Params, ArrayRef<SourceLocation>()); 409 410 setParamsAndSelLocs(C, Params, SelLocs); 411 } 412 413 /// \brief A definition will return its interface declaration. 414 /// An interface declaration will return its definition. 415 /// Otherwise it will return itself. 416 ObjCMethodDecl *ObjCMethodDecl::getNextRedeclaration() { 417 ASTContext &Ctx = getASTContext(); 418 ObjCMethodDecl *Redecl = 0; 419 if (HasRedeclaration) 420 Redecl = const_cast<ObjCMethodDecl*>(Ctx.getObjCMethodRedeclaration(this)); 421 if (Redecl) 422 return Redecl; 423 424 Decl *CtxD = cast<Decl>(getDeclContext()); 425 426 if (ObjCInterfaceDecl *IFD = dyn_cast<ObjCInterfaceDecl>(CtxD)) { 427 if (ObjCImplementationDecl *ImplD = Ctx.getObjCImplementation(IFD)) 428 Redecl = ImplD->getMethod(getSelector(), isInstanceMethod()); 429 430 } else if (ObjCCategoryDecl *CD = dyn_cast<ObjCCategoryDecl>(CtxD)) { 431 if (ObjCCategoryImplDecl *ImplD = Ctx.getObjCImplementation(CD)) 432 Redecl = ImplD->getMethod(getSelector(), isInstanceMethod()); 433 434 } else if (ObjCImplementationDecl *ImplD = 435 dyn_cast<ObjCImplementationDecl>(CtxD)) { 436 if (ObjCInterfaceDecl *IFD = ImplD->getClassInterface()) 437 Redecl = IFD->getMethod(getSelector(), isInstanceMethod()); 438 439 } else if (ObjCCategoryImplDecl *CImplD = 440 dyn_cast<ObjCCategoryImplDecl>(CtxD)) { 441 if (ObjCCategoryDecl *CatD = CImplD->getCategoryDecl()) 442 Redecl = CatD->getMethod(getSelector(), isInstanceMethod()); 443 } 444 445 if (!Redecl && isRedeclaration()) { 446 // This is the last redeclaration, go back to the first method. 447 return cast<ObjCContainerDecl>(CtxD)->getMethod(getSelector(), 448 isInstanceMethod()); 449 } 450 451 return Redecl ? Redecl : this; 452 } 453 454 ObjCMethodDecl *ObjCMethodDecl::getCanonicalDecl() { 455 Decl *CtxD = cast<Decl>(getDeclContext()); 456 457 if (ObjCImplementationDecl *ImplD = dyn_cast<ObjCImplementationDecl>(CtxD)) { 458 if (ObjCInterfaceDecl *IFD = ImplD->getClassInterface()) 459 if (ObjCMethodDecl *MD = IFD->getMethod(getSelector(), 460 isInstanceMethod())) 461 return MD; 462 463 } else if (ObjCCategoryImplDecl *CImplD = 464 dyn_cast<ObjCCategoryImplDecl>(CtxD)) { 465 if (ObjCCategoryDecl *CatD = CImplD->getCategoryDecl()) 466 if (ObjCMethodDecl *MD = CatD->getMethod(getSelector(), 467 isInstanceMethod())) 468 return MD; 469 } 470 471 if (isRedeclaration()) 472 return cast<ObjCContainerDecl>(CtxD)->getMethod(getSelector(), 473 isInstanceMethod()); 474 475 return this; 476 } 477 478 ObjCMethodFamily ObjCMethodDecl::getMethodFamily() const { 479 ObjCMethodFamily family = static_cast<ObjCMethodFamily>(Family); 480 if (family != static_cast<unsigned>(InvalidObjCMethodFamily)) 481 return family; 482 483 // Check for an explicit attribute. 484 if (const ObjCMethodFamilyAttr *attr = getAttr<ObjCMethodFamilyAttr>()) { 485 // The unfortunate necessity of mapping between enums here is due 486 // to the attributes framework. 487 switch (attr->getFamily()) { 488 case ObjCMethodFamilyAttr::OMF_None: family = OMF_None; break; 489 case ObjCMethodFamilyAttr::OMF_alloc: family = OMF_alloc; break; 490 case ObjCMethodFamilyAttr::OMF_copy: family = OMF_copy; break; 491 case ObjCMethodFamilyAttr::OMF_init: family = OMF_init; break; 492 case ObjCMethodFamilyAttr::OMF_mutableCopy: family = OMF_mutableCopy; break; 493 case ObjCMethodFamilyAttr::OMF_new: family = OMF_new; break; 494 } 495 Family = static_cast<unsigned>(family); 496 return family; 497 } 498 499 family = getSelector().getMethodFamily(); 500 switch (family) { 501 case OMF_None: break; 502 503 // init only has a conventional meaning for an instance method, and 504 // it has to return an object. 505 case OMF_init: 506 if (!isInstanceMethod() || !getResultType()->isObjCObjectPointerType()) 507 family = OMF_None; 508 break; 509 510 // alloc/copy/new have a conventional meaning for both class and 511 // instance methods, but they require an object return. 512 case OMF_alloc: 513 case OMF_copy: 514 case OMF_mutableCopy: 515 case OMF_new: 516 if (!getResultType()->isObjCObjectPointerType()) 517 family = OMF_None; 518 break; 519 520 // These selectors have a conventional meaning only for instance methods. 521 case OMF_dealloc: 522 case OMF_finalize: 523 case OMF_retain: 524 case OMF_release: 525 case OMF_autorelease: 526 case OMF_retainCount: 527 case OMF_self: 528 if (!isInstanceMethod()) 529 family = OMF_None; 530 break; 531 532 case OMF_performSelector: 533 if (!isInstanceMethod() || 534 !getResultType()->isObjCIdType()) 535 family = OMF_None; 536 else { 537 unsigned noParams = param_size(); 538 if (noParams < 1 || noParams > 3) 539 family = OMF_None; 540 else { 541 ObjCMethodDecl::arg_type_iterator it = arg_type_begin(); 542 QualType ArgT = (*it); 543 if (!ArgT->isObjCSelType()) { 544 family = OMF_None; 545 break; 546 } 547 while (--noParams) { 548 it++; 549 ArgT = (*it); 550 if (!ArgT->isObjCIdType()) { 551 family = OMF_None; 552 break; 553 } 554 } 555 } 556 } 557 break; 558 559 } 560 561 // Cache the result. 562 Family = static_cast<unsigned>(family); 563 return family; 564 } 565 566 void ObjCMethodDecl::createImplicitParams(ASTContext &Context, 567 const ObjCInterfaceDecl *OID) { 568 QualType selfTy; 569 if (isInstanceMethod()) { 570 // There may be no interface context due to error in declaration 571 // of the interface (which has been reported). Recover gracefully. 572 if (OID) { 573 selfTy = Context.getObjCInterfaceType(OID); 574 selfTy = Context.getObjCObjectPointerType(selfTy); 575 } else { 576 selfTy = Context.getObjCIdType(); 577 } 578 } else // we have a factory method. 579 selfTy = Context.getObjCClassType(); 580 581 bool selfIsPseudoStrong = false; 582 bool selfIsConsumed = false; 583 584 if (Context.getLangOptions().ObjCAutoRefCount) { 585 if (isInstanceMethod()) { 586 selfIsConsumed = hasAttr<NSConsumesSelfAttr>(); 587 588 // 'self' is always __strong. It's actually pseudo-strong except 589 // in init methods (or methods labeled ns_consumes_self), though. 590 Qualifiers qs; 591 qs.setObjCLifetime(Qualifiers::OCL_Strong); 592 selfTy = Context.getQualifiedType(selfTy, qs); 593 594 // In addition, 'self' is const unless this is an init method. 595 if (getMethodFamily() != OMF_init && !selfIsConsumed) { 596 selfTy = selfTy.withConst(); 597 selfIsPseudoStrong = true; 598 } 599 } 600 else { 601 assert(isClassMethod()); 602 // 'self' is always const in class methods. 603 selfTy = selfTy.withConst(); 604 selfIsPseudoStrong = true; 605 } 606 } 607 608 ImplicitParamDecl *self 609 = ImplicitParamDecl::Create(Context, this, SourceLocation(), 610 &Context.Idents.get("self"), selfTy); 611 setSelfDecl(self); 612 613 if (selfIsConsumed) 614 self->addAttr(new (Context) NSConsumedAttr(SourceLocation(), Context)); 615 616 if (selfIsPseudoStrong) 617 self->setARCPseudoStrong(true); 618 619 setCmdDecl(ImplicitParamDecl::Create(Context, this, SourceLocation(), 620 &Context.Idents.get("_cmd"), 621 Context.getObjCSelType())); 622 } 623 624 ObjCInterfaceDecl *ObjCMethodDecl::getClassInterface() { 625 if (ObjCInterfaceDecl *ID = dyn_cast<ObjCInterfaceDecl>(getDeclContext())) 626 return ID; 627 if (ObjCCategoryDecl *CD = dyn_cast<ObjCCategoryDecl>(getDeclContext())) 628 return CD->getClassInterface(); 629 if (ObjCImplDecl *IMD = dyn_cast<ObjCImplDecl>(getDeclContext())) 630 return IMD->getClassInterface(); 631 632 assert(!isa<ObjCProtocolDecl>(getDeclContext()) && "It's a protocol method"); 633 llvm_unreachable("unknown method context"); 634 } 635 636 //===----------------------------------------------------------------------===// 637 // ObjCInterfaceDecl 638 //===----------------------------------------------------------------------===// 639 640 ObjCInterfaceDecl *ObjCInterfaceDecl::Create(ASTContext &C, 641 DeclContext *DC, 642 SourceLocation atLoc, 643 IdentifierInfo *Id, 644 SourceLocation ClassLoc, 645 bool ForwardDecl, bool isInternal){ 646 return new (C) ObjCInterfaceDecl(DC, atLoc, Id, ClassLoc, ForwardDecl, 647 isInternal); 648 } 649 650 ObjCInterfaceDecl:: 651 ObjCInterfaceDecl(DeclContext *DC, SourceLocation atLoc, IdentifierInfo *Id, 652 SourceLocation CLoc, bool FD, bool isInternal) 653 : ObjCContainerDecl(ObjCInterface, DC, Id, CLoc, atLoc), 654 TypeForDecl(0), SuperClass(0), 655 CategoryList(0), IvarList(0), 656 InitiallyForwardDecl(FD), ForwardDecl(FD), 657 ExternallyCompleted(false) { 658 setImplicit(isInternal); 659 } 660 661 void ObjCInterfaceDecl::LoadExternalDefinition() const { 662 assert(ExternallyCompleted && "Class is not externally completed"); 663 ExternallyCompleted = false; 664 getASTContext().getExternalSource()->CompleteType( 665 const_cast<ObjCInterfaceDecl *>(this)); 666 } 667 668 void ObjCInterfaceDecl::setExternallyCompleted() { 669 assert(getASTContext().getExternalSource() && 670 "Class can't be externally completed without an external source"); 671 assert(!ForwardDecl && 672 "Forward declarations can't be externally completed"); 673 ExternallyCompleted = true; 674 } 675 676 ObjCImplementationDecl *ObjCInterfaceDecl::getImplementation() const { 677 if (ExternallyCompleted) 678 LoadExternalDefinition(); 679 680 return getASTContext().getObjCImplementation( 681 const_cast<ObjCInterfaceDecl*>(this)); 682 } 683 684 void ObjCInterfaceDecl::setImplementation(ObjCImplementationDecl *ImplD) { 685 getASTContext().setObjCImplementation(this, ImplD); 686 } 687 688 /// all_declared_ivar_begin - return first ivar declared in this class, 689 /// its extensions and its implementation. Lazily build the list on first 690 /// access. 691 ObjCIvarDecl *ObjCInterfaceDecl::all_declared_ivar_begin() { 692 if (IvarList) 693 return IvarList; 694 695 ObjCIvarDecl *curIvar = 0; 696 if (!ivar_empty()) { 697 ObjCInterfaceDecl::ivar_iterator I = ivar_begin(), E = ivar_end(); 698 IvarList = (*I); ++I; 699 for (curIvar = IvarList; I != E; curIvar = *I, ++I) 700 curIvar->setNextIvar(*I); 701 } 702 703 for (const ObjCCategoryDecl *CDecl = getFirstClassExtension(); CDecl; 704 CDecl = CDecl->getNextClassExtension()) { 705 if (!CDecl->ivar_empty()) { 706 ObjCCategoryDecl::ivar_iterator I = CDecl->ivar_begin(), 707 E = CDecl->ivar_end(); 708 if (!IvarList) { 709 IvarList = (*I); ++I; 710 curIvar = IvarList; 711 } 712 for ( ;I != E; curIvar = *I, ++I) 713 curIvar->setNextIvar(*I); 714 } 715 } 716 717 if (ObjCImplementationDecl *ImplDecl = getImplementation()) { 718 if (!ImplDecl->ivar_empty()) { 719 ObjCImplementationDecl::ivar_iterator I = ImplDecl->ivar_begin(), 720 E = ImplDecl->ivar_end(); 721 if (!IvarList) { 722 IvarList = (*I); ++I; 723 curIvar = IvarList; 724 } 725 for ( ;I != E; curIvar = *I, ++I) 726 curIvar->setNextIvar(*I); 727 } 728 } 729 return IvarList; 730 } 731 732 /// FindCategoryDeclaration - Finds category declaration in the list of 733 /// categories for this class and returns it. Name of the category is passed 734 /// in 'CategoryId'. If category not found, return 0; 735 /// 736 ObjCCategoryDecl * 737 ObjCInterfaceDecl::FindCategoryDeclaration(IdentifierInfo *CategoryId) const { 738 if (ExternallyCompleted) 739 LoadExternalDefinition(); 740 741 for (ObjCCategoryDecl *Category = getCategoryList(); 742 Category; Category = Category->getNextClassCategory()) 743 if (Category->getIdentifier() == CategoryId) 744 return Category; 745 return 0; 746 } 747 748 ObjCMethodDecl * 749 ObjCInterfaceDecl::getCategoryInstanceMethod(Selector Sel) const { 750 for (ObjCCategoryDecl *Category = getCategoryList(); 751 Category; Category = Category->getNextClassCategory()) 752 if (ObjCCategoryImplDecl *Impl = Category->getImplementation()) 753 if (ObjCMethodDecl *MD = Impl->getInstanceMethod(Sel)) 754 return MD; 755 return 0; 756 } 757 758 ObjCMethodDecl *ObjCInterfaceDecl::getCategoryClassMethod(Selector Sel) const { 759 for (ObjCCategoryDecl *Category = getCategoryList(); 760 Category; Category = Category->getNextClassCategory()) 761 if (ObjCCategoryImplDecl *Impl = Category->getImplementation()) 762 if (ObjCMethodDecl *MD = Impl->getClassMethod(Sel)) 763 return MD; 764 return 0; 765 } 766 767 /// ClassImplementsProtocol - Checks that 'lProto' protocol 768 /// has been implemented in IDecl class, its super class or categories (if 769 /// lookupCategory is true). 770 bool ObjCInterfaceDecl::ClassImplementsProtocol(ObjCProtocolDecl *lProto, 771 bool lookupCategory, 772 bool RHSIsQualifiedID) { 773 ObjCInterfaceDecl *IDecl = this; 774 // 1st, look up the class. 775 const ObjCList<ObjCProtocolDecl> &Protocols = 776 IDecl->getReferencedProtocols(); 777 778 for (ObjCList<ObjCProtocolDecl>::iterator PI = Protocols.begin(), 779 E = Protocols.end(); PI != E; ++PI) { 780 if (getASTContext().ProtocolCompatibleWithProtocol(lProto, *PI)) 781 return true; 782 // This is dubious and is added to be compatible with gcc. In gcc, it is 783 // also allowed assigning a protocol-qualified 'id' type to a LHS object 784 // when protocol in qualified LHS is in list of protocols in the rhs 'id' 785 // object. This IMO, should be a bug. 786 // FIXME: Treat this as an extension, and flag this as an error when GCC 787 // extensions are not enabled. 788 if (RHSIsQualifiedID && 789 getASTContext().ProtocolCompatibleWithProtocol(*PI, lProto)) 790 return true; 791 } 792 793 // 2nd, look up the category. 794 if (lookupCategory) 795 for (ObjCCategoryDecl *CDecl = IDecl->getCategoryList(); CDecl; 796 CDecl = CDecl->getNextClassCategory()) { 797 for (ObjCCategoryDecl::protocol_iterator PI = CDecl->protocol_begin(), 798 E = CDecl->protocol_end(); PI != E; ++PI) 799 if (getASTContext().ProtocolCompatibleWithProtocol(lProto, *PI)) 800 return true; 801 } 802 803 // 3rd, look up the super class(s) 804 if (IDecl->getSuperClass()) 805 return 806 IDecl->getSuperClass()->ClassImplementsProtocol(lProto, lookupCategory, 807 RHSIsQualifiedID); 808 809 return false; 810 } 811 812 //===----------------------------------------------------------------------===// 813 // ObjCIvarDecl 814 //===----------------------------------------------------------------------===// 815 816 ObjCIvarDecl *ObjCIvarDecl::Create(ASTContext &C, ObjCContainerDecl *DC, 817 SourceLocation StartLoc, 818 SourceLocation IdLoc, IdentifierInfo *Id, 819 QualType T, TypeSourceInfo *TInfo, 820 AccessControl ac, Expr *BW, 821 bool synthesized) { 822 if (DC) { 823 // Ivar's can only appear in interfaces, implementations (via synthesized 824 // properties), and class extensions (via direct declaration, or synthesized 825 // properties). 826 // 827 // FIXME: This should really be asserting this: 828 // (isa<ObjCCategoryDecl>(DC) && 829 // cast<ObjCCategoryDecl>(DC)->IsClassExtension())) 830 // but unfortunately we sometimes place ivars into non-class extension 831 // categories on error. This breaks an AST invariant, and should not be 832 // fixed. 833 assert((isa<ObjCInterfaceDecl>(DC) || isa<ObjCImplementationDecl>(DC) || 834 isa<ObjCCategoryDecl>(DC)) && 835 "Invalid ivar decl context!"); 836 // Once a new ivar is created in any of class/class-extension/implementation 837 // decl contexts, the previously built IvarList must be rebuilt. 838 ObjCInterfaceDecl *ID = dyn_cast<ObjCInterfaceDecl>(DC); 839 if (!ID) { 840 if (ObjCImplementationDecl *IM = dyn_cast<ObjCImplementationDecl>(DC)) { 841 ID = IM->getClassInterface(); 842 if (BW) 843 IM->setHasSynthBitfield(true); 844 } else { 845 ObjCCategoryDecl *CD = cast<ObjCCategoryDecl>(DC); 846 ID = CD->getClassInterface(); 847 if (BW) 848 CD->setHasSynthBitfield(true); 849 } 850 } 851 ID->setIvarList(0); 852 } 853 854 return new (C) ObjCIvarDecl(DC, StartLoc, IdLoc, Id, T, TInfo, 855 ac, BW, synthesized); 856 } 857 858 const ObjCInterfaceDecl *ObjCIvarDecl::getContainingInterface() const { 859 const ObjCContainerDecl *DC = cast<ObjCContainerDecl>(getDeclContext()); 860 861 switch (DC->getKind()) { 862 default: 863 case ObjCCategoryImpl: 864 case ObjCProtocol: 865 llvm_unreachable("invalid ivar container!"); 866 867 // Ivars can only appear in class extension categories. 868 case ObjCCategory: { 869 const ObjCCategoryDecl *CD = cast<ObjCCategoryDecl>(DC); 870 assert(CD->IsClassExtension() && "invalid container for ivar!"); 871 return CD->getClassInterface(); 872 } 873 874 case ObjCImplementation: 875 return cast<ObjCImplementationDecl>(DC)->getClassInterface(); 876 877 case ObjCInterface: 878 return cast<ObjCInterfaceDecl>(DC); 879 } 880 } 881 882 //===----------------------------------------------------------------------===// 883 // ObjCAtDefsFieldDecl 884 //===----------------------------------------------------------------------===// 885 886 ObjCAtDefsFieldDecl 887 *ObjCAtDefsFieldDecl::Create(ASTContext &C, DeclContext *DC, 888 SourceLocation StartLoc, SourceLocation IdLoc, 889 IdentifierInfo *Id, QualType T, Expr *BW) { 890 return new (C) ObjCAtDefsFieldDecl(DC, StartLoc, IdLoc, Id, T, BW); 891 } 892 893 //===----------------------------------------------------------------------===// 894 // ObjCProtocolDecl 895 //===----------------------------------------------------------------------===// 896 897 ObjCProtocolDecl *ObjCProtocolDecl::Create(ASTContext &C, DeclContext *DC, 898 IdentifierInfo *Id, 899 SourceLocation nameLoc, 900 SourceLocation atStartLoc, 901 bool isForwardDecl) { 902 return new (C) ObjCProtocolDecl(DC, Id, nameLoc, atStartLoc, isForwardDecl); 903 } 904 905 ObjCProtocolDecl *ObjCProtocolDecl::lookupProtocolNamed(IdentifierInfo *Name) { 906 ObjCProtocolDecl *PDecl = this; 907 908 if (Name == getIdentifier()) 909 return PDecl; 910 911 for (protocol_iterator I = protocol_begin(), E = protocol_end(); I != E; ++I) 912 if ((PDecl = (*I)->lookupProtocolNamed(Name))) 913 return PDecl; 914 915 return NULL; 916 } 917 918 // lookupMethod - Lookup a instance/class method in the protocol and protocols 919 // it inherited. 920 ObjCMethodDecl *ObjCProtocolDecl::lookupMethod(Selector Sel, 921 bool isInstance) const { 922 ObjCMethodDecl *MethodDecl = NULL; 923 924 if ((MethodDecl = getMethod(Sel, isInstance))) 925 return MethodDecl; 926 927 for (protocol_iterator I = protocol_begin(), E = protocol_end(); I != E; ++I) 928 if ((MethodDecl = (*I)->lookupMethod(Sel, isInstance))) 929 return MethodDecl; 930 return NULL; 931 } 932 933 void ObjCProtocolDecl::completedForwardDecl() { 934 assert(isForwardDecl() && "Only valid to call for forward refs"); 935 isForwardProtoDecl = false; 936 if (ASTMutationListener *L = getASTContext().getASTMutationListener()) 937 L->CompletedObjCForwardRef(this); 938 } 939 940 //===----------------------------------------------------------------------===// 941 // ObjCClassDecl 942 //===----------------------------------------------------------------------===// 943 944 ObjCClassDecl::ObjCClassDecl(DeclContext *DC, SourceLocation L, 945 ObjCInterfaceDecl *const Elt, 946 const SourceLocation Loc, 947 ASTContext &C) 948 : Decl(ObjCClass, DC, L) { 949 setClass(C, Elt, Loc); 950 } 951 952 ObjCClassDecl *ObjCClassDecl::Create(ASTContext &C, DeclContext *DC, 953 SourceLocation L, 954 ObjCInterfaceDecl *const Elt, 955 const SourceLocation Loc) { 956 return new (C) ObjCClassDecl(DC, L, Elt, Loc, C); 957 } 958 959 void ObjCClassDecl::setClass(ASTContext &C, ObjCInterfaceDecl*const Cls, 960 const SourceLocation Loc) { 961 962 ForwardDecl = (ObjCClassRef*) C.Allocate(sizeof(ObjCClassRef), 963 llvm::alignOf<ObjCClassRef>()); 964 new (ForwardDecl) ObjCClassRef(Cls, Loc); 965 } 966 967 SourceRange ObjCClassDecl::getSourceRange() const { 968 // FIXME: We should include the semicolon 969 return SourceRange(getLocation(), ForwardDecl->getLocation()); 970 } 971 972 //===----------------------------------------------------------------------===// 973 // ObjCForwardProtocolDecl 974 //===----------------------------------------------------------------------===// 975 976 ObjCForwardProtocolDecl:: 977 ObjCForwardProtocolDecl(DeclContext *DC, SourceLocation L, 978 ObjCProtocolDecl *const *Elts, unsigned nElts, 979 const SourceLocation *Locs, ASTContext &C) 980 : Decl(ObjCForwardProtocol, DC, L) { 981 ReferencedProtocols.set(Elts, nElts, Locs, C); 982 } 983 984 985 ObjCForwardProtocolDecl * 986 ObjCForwardProtocolDecl::Create(ASTContext &C, DeclContext *DC, 987 SourceLocation L, 988 ObjCProtocolDecl *const *Elts, 989 unsigned NumElts, 990 const SourceLocation *Locs) { 991 return new (C) ObjCForwardProtocolDecl(DC, L, Elts, NumElts, Locs, C); 992 } 993 994 //===----------------------------------------------------------------------===// 995 // ObjCCategoryDecl 996 //===----------------------------------------------------------------------===// 997 998 ObjCCategoryDecl *ObjCCategoryDecl::Create(ASTContext &C, DeclContext *DC, 999 SourceLocation AtLoc, 1000 SourceLocation ClassNameLoc, 1001 SourceLocation CategoryNameLoc, 1002 IdentifierInfo *Id, 1003 ObjCInterfaceDecl *IDecl) { 1004 ObjCCategoryDecl *CatDecl = new (C) ObjCCategoryDecl(DC, AtLoc, ClassNameLoc, 1005 CategoryNameLoc, Id, 1006 IDecl); 1007 if (IDecl) { 1008 // Link this category into its class's category list. 1009 CatDecl->NextClassCategory = IDecl->getCategoryList(); 1010 IDecl->setCategoryList(CatDecl); 1011 if (ASTMutationListener *L = C.getASTMutationListener()) 1012 L->AddedObjCCategoryToInterface(CatDecl, IDecl); 1013 } 1014 1015 return CatDecl; 1016 } 1017 1018 ObjCCategoryDecl *ObjCCategoryDecl::Create(ASTContext &C, EmptyShell Empty) { 1019 return new (C) ObjCCategoryDecl(0, SourceLocation(), SourceLocation(), 1020 SourceLocation(), 0, 0); 1021 } 1022 1023 ObjCCategoryImplDecl *ObjCCategoryDecl::getImplementation() const { 1024 return getASTContext().getObjCImplementation( 1025 const_cast<ObjCCategoryDecl*>(this)); 1026 } 1027 1028 void ObjCCategoryDecl::setImplementation(ObjCCategoryImplDecl *ImplD) { 1029 getASTContext().setObjCImplementation(this, ImplD); 1030 } 1031 1032 1033 //===----------------------------------------------------------------------===// 1034 // ObjCCategoryImplDecl 1035 //===----------------------------------------------------------------------===// 1036 1037 ObjCCategoryImplDecl * 1038 ObjCCategoryImplDecl::Create(ASTContext &C, DeclContext *DC, 1039 IdentifierInfo *Id, 1040 ObjCInterfaceDecl *ClassInterface, 1041 SourceLocation nameLoc, 1042 SourceLocation atStartLoc) { 1043 return new (C) ObjCCategoryImplDecl(DC, Id, ClassInterface, 1044 nameLoc, atStartLoc); 1045 } 1046 1047 ObjCCategoryDecl *ObjCCategoryImplDecl::getCategoryDecl() const { 1048 // The class interface might be NULL if we are working with invalid code. 1049 if (const ObjCInterfaceDecl *ID = getClassInterface()) 1050 return ID->FindCategoryDeclaration(getIdentifier()); 1051 return 0; 1052 } 1053 1054 1055 void ObjCImplDecl::addPropertyImplementation(ObjCPropertyImplDecl *property) { 1056 // FIXME: The context should be correct before we get here. 1057 property->setLexicalDeclContext(this); 1058 addDecl(property); 1059 } 1060 1061 void ObjCImplDecl::setClassInterface(ObjCInterfaceDecl *IFace) { 1062 ASTContext &Ctx = getASTContext(); 1063 1064 if (ObjCImplementationDecl *ImplD 1065 = dyn_cast_or_null<ObjCImplementationDecl>(this)) { 1066 if (IFace) 1067 Ctx.setObjCImplementation(IFace, ImplD); 1068 1069 } else if (ObjCCategoryImplDecl *ImplD = 1070 dyn_cast_or_null<ObjCCategoryImplDecl>(this)) { 1071 if (ObjCCategoryDecl *CD = IFace->FindCategoryDeclaration(getIdentifier())) 1072 Ctx.setObjCImplementation(CD, ImplD); 1073 } 1074 1075 ClassInterface = IFace; 1076 } 1077 1078 /// FindPropertyImplIvarDecl - This method lookup the ivar in the list of 1079 /// properties implemented in this category @implementation block and returns 1080 /// the implemented property that uses it. 1081 /// 1082 ObjCPropertyImplDecl *ObjCImplDecl:: 1083 FindPropertyImplIvarDecl(IdentifierInfo *ivarId) const { 1084 for (propimpl_iterator i = propimpl_begin(), e = propimpl_end(); i != e; ++i){ 1085 ObjCPropertyImplDecl *PID = *i; 1086 if (PID->getPropertyIvarDecl() && 1087 PID->getPropertyIvarDecl()->getIdentifier() == ivarId) 1088 return PID; 1089 } 1090 return 0; 1091 } 1092 1093 /// FindPropertyImplDecl - This method looks up a previous ObjCPropertyImplDecl 1094 /// added to the list of those properties @synthesized/@dynamic in this 1095 /// category @implementation block. 1096 /// 1097 ObjCPropertyImplDecl *ObjCImplDecl:: 1098 FindPropertyImplDecl(IdentifierInfo *Id) const { 1099 for (propimpl_iterator i = propimpl_begin(), e = propimpl_end(); i != e; ++i){ 1100 ObjCPropertyImplDecl *PID = *i; 1101 if (PID->getPropertyDecl()->getIdentifier() == Id) 1102 return PID; 1103 } 1104 return 0; 1105 } 1106 1107 raw_ostream &clang::operator<<(raw_ostream &OS, 1108 const ObjCCategoryImplDecl *CID) { 1109 OS << CID->getName(); 1110 return OS; 1111 } 1112 1113 //===----------------------------------------------------------------------===// 1114 // ObjCImplementationDecl 1115 //===----------------------------------------------------------------------===// 1116 1117 ObjCImplementationDecl * 1118 ObjCImplementationDecl::Create(ASTContext &C, DeclContext *DC, 1119 ObjCInterfaceDecl *ClassInterface, 1120 ObjCInterfaceDecl *SuperDecl, 1121 SourceLocation nameLoc, 1122 SourceLocation atStartLoc) { 1123 return new (C) ObjCImplementationDecl(DC, ClassInterface, SuperDecl, 1124 nameLoc, atStartLoc); 1125 } 1126 1127 void ObjCImplementationDecl::setIvarInitializers(ASTContext &C, 1128 CXXCtorInitializer ** initializers, 1129 unsigned numInitializers) { 1130 if (numInitializers > 0) { 1131 NumIvarInitializers = numInitializers; 1132 CXXCtorInitializer **ivarInitializers = 1133 new (C) CXXCtorInitializer*[NumIvarInitializers]; 1134 memcpy(ivarInitializers, initializers, 1135 numInitializers * sizeof(CXXCtorInitializer*)); 1136 IvarInitializers = ivarInitializers; 1137 } 1138 } 1139 1140 raw_ostream &clang::operator<<(raw_ostream &OS, 1141 const ObjCImplementationDecl *ID) { 1142 OS << ID->getName(); 1143 return OS; 1144 } 1145 1146 //===----------------------------------------------------------------------===// 1147 // ObjCCompatibleAliasDecl 1148 //===----------------------------------------------------------------------===// 1149 1150 ObjCCompatibleAliasDecl * 1151 ObjCCompatibleAliasDecl::Create(ASTContext &C, DeclContext *DC, 1152 SourceLocation L, 1153 IdentifierInfo *Id, 1154 ObjCInterfaceDecl* AliasedClass) { 1155 return new (C) ObjCCompatibleAliasDecl(DC, L, Id, AliasedClass); 1156 } 1157 1158 //===----------------------------------------------------------------------===// 1159 // ObjCPropertyDecl 1160 //===----------------------------------------------------------------------===// 1161 1162 ObjCPropertyDecl *ObjCPropertyDecl::Create(ASTContext &C, DeclContext *DC, 1163 SourceLocation L, 1164 IdentifierInfo *Id, 1165 SourceLocation AtLoc, 1166 TypeSourceInfo *T, 1167 PropertyControl propControl) { 1168 return new (C) ObjCPropertyDecl(DC, L, Id, AtLoc, T); 1169 } 1170 1171 //===----------------------------------------------------------------------===// 1172 // ObjCPropertyImplDecl 1173 //===----------------------------------------------------------------------===// 1174 1175 ObjCPropertyImplDecl *ObjCPropertyImplDecl::Create(ASTContext &C, 1176 DeclContext *DC, 1177 SourceLocation atLoc, 1178 SourceLocation L, 1179 ObjCPropertyDecl *property, 1180 Kind PK, 1181 ObjCIvarDecl *ivar, 1182 SourceLocation ivarLoc) { 1183 return new (C) ObjCPropertyImplDecl(DC, atLoc, L, property, PK, ivar, 1184 ivarLoc); 1185 } 1186 1187 SourceRange ObjCPropertyImplDecl::getSourceRange() const { 1188 SourceLocation EndLoc = getLocation(); 1189 if (IvarLoc.isValid()) 1190 EndLoc = IvarLoc; 1191 1192 return SourceRange(AtLoc, EndLoc); 1193 } 1194