DeclObjC.h revision 1d784b277cdfd4eba03680715d2a082b3f28d295
1980e508ca70d6de75d2abfd96b4681fc98bb2698Steve Naroff//===--- DeclObjC.h - Classes for representing declarations -----*- C++ -*-===//
2980e508ca70d6de75d2abfd96b4681fc98bb2698Steve Naroff//
3980e508ca70d6de75d2abfd96b4681fc98bb2698Steve Naroff//                     The LLVM Compiler Infrastructure
4980e508ca70d6de75d2abfd96b4681fc98bb2698Steve Naroff//
50bc735ffcfb223c0186419547abaa5c84482663eChris Lattner// This file is distributed under the University of Illinois Open Source
60bc735ffcfb223c0186419547abaa5c84482663eChris Lattner// License. See LICENSE.TXT for details.
7980e508ca70d6de75d2abfd96b4681fc98bb2698Steve Naroff//
8980e508ca70d6de75d2abfd96b4681fc98bb2698Steve Naroff//===----------------------------------------------------------------------===//
9980e508ca70d6de75d2abfd96b4681fc98bb2698Steve Naroff//
10980e508ca70d6de75d2abfd96b4681fc98bb2698Steve Naroff//  This file defines the DeclObjC interface and subclasses.
11980e508ca70d6de75d2abfd96b4681fc98bb2698Steve Naroff//
12980e508ca70d6de75d2abfd96b4681fc98bb2698Steve Naroff//===----------------------------------------------------------------------===//
13980e508ca70d6de75d2abfd96b4681fc98bb2698Steve Naroff
14980e508ca70d6de75d2abfd96b4681fc98bb2698Steve Naroff#ifndef LLVM_CLANG_AST_DECLOBJC_H
15980e508ca70d6de75d2abfd96b4681fc98bb2698Steve Naroff#define LLVM_CLANG_AST_DECLOBJC_H
16980e508ca70d6de75d2abfd96b4681fc98bb2698Steve Naroff
17980e508ca70d6de75d2abfd96b4681fc98bb2698Steve Naroff#include "clang/AST/Decl.h"
18491306a83c4f0f49f95a3bcbca8580cb98a91c7aArgyrios Kyrtzidis#include "clang/AST/SelectorLocationsKind.h"
196ea8e2152e1ba93b4c80e7268403a582896dc3dcAnders Carlsson#include "llvm/ADT/STLExtras.h"
20980e508ca70d6de75d2abfd96b4681fc98bb2698Steve Naroff
21980e508ca70d6de75d2abfd96b4681fc98bb2698Steve Naroffnamespace clang {
22980e508ca70d6de75d2abfd96b4681fc98bb2698Steve Naroffclass Expr;
23980e508ca70d6de75d2abfd96b4681fc98bb2698Steve Naroffclass Stmt;
24980e508ca70d6de75d2abfd96b4681fc98bb2698Steve Naroffclass FunctionDecl;
2560f8c868ffb346b78451a3eccaecd0461d2ae498Fariborz Jahanianclass RecordDecl;
26a526c5c67e5a0473c340903ee542ce570119665fTed Kremenekclass ObjCIvarDecl;
27a526c5c67e5a0473c340903ee542ce570119665fTed Kremenekclass ObjCMethodDecl;
28a526c5c67e5a0473c340903ee542ce570119665fTed Kremenekclass ObjCProtocolDecl;
29a526c5c67e5a0473c340903ee542ce570119665fTed Kremenekclass ObjCCategoryDecl;
30a526c5c67e5a0473c340903ee542ce570119665fTed Kremenekclass ObjCPropertyDecl;
31f624f8186d8fe474350051c6d3f00b2c204fbeaeFariborz Jahanianclass ObjCPropertyImplDecl;
32cbb67480094b3bcb5b715acd827cbad55e2a204cSean Huntclass CXXCtorInitializer;
3368835718c4125f2f66740cd04de7088645ec695dChris Lattner
34793ccfd646d0388e06c587e962a18fa723b72f02Chris Lattnerclass ObjCListBase {
35793ccfd646d0388e06c587e962a18fa723b72f02Chris Lattner  void operator=(const ObjCListBase &);     // DO NOT IMPLEMENT
36793ccfd646d0388e06c587e962a18fa723b72f02Chris Lattner  ObjCListBase(const ObjCListBase&);        // DO NOT IMPLEMENT
37793ccfd646d0388e06c587e962a18fa723b72f02Chris Lattnerprotected:
38793ccfd646d0388e06c587e962a18fa723b72f02Chris Lattner  /// List is an array of pointers to objects that are not owned by this object.
39793ccfd646d0388e06c587e962a18fa723b72f02Chris Lattner  void **List;
403db6cae19c236fe2cef343c90105ce7cca7de965Chris Lattner  unsigned NumElts;
41793ccfd646d0388e06c587e962a18fa723b72f02Chris Lattner
423db6cae19c236fe2cef343c90105ce7cca7de965Chris Lattnerpublic:
43793ccfd646d0388e06c587e962a18fa723b72f02Chris Lattner  ObjCListBase() : List(0), NumElts(0) {}
44793ccfd646d0388e06c587e962a18fa723b72f02Chris Lattner  unsigned size() const { return NumElts; }
45793ccfd646d0388e06c587e962a18fa723b72f02Chris Lattner  bool empty() const { return NumElts == 0; }
461eb4433ac451dc16f4133a88af2d002ac26c58efMike Stump
47793ccfd646d0388e06c587e962a18fa723b72f02Chris Lattnerprotected:
4838af2deb27cdfa1a95bde96e30dab15dce53fcefChris Lattner  void set(void *const* InList, unsigned Elts, ASTContext &Ctx);
49793ccfd646d0388e06c587e962a18fa723b72f02Chris Lattner};
501eb4433ac451dc16f4133a88af2d002ac26c58efMike Stump
511eb4433ac451dc16f4133a88af2d002ac26c58efMike Stump
52793ccfd646d0388e06c587e962a18fa723b72f02Chris Lattner/// ObjCList - This is a simple template class used to hold various lists of
53793ccfd646d0388e06c587e962a18fa723b72f02Chris Lattner/// decls etc, which is heavily used by the ObjC front-end.  This only use case
54793ccfd646d0388e06c587e962a18fa723b72f02Chris Lattner/// this supports is setting the list all at once and then reading elements out
55793ccfd646d0388e06c587e962a18fa723b72f02Chris Lattner/// of it.
56793ccfd646d0388e06c587e962a18fa723b72f02Chris Lattnertemplate <typename T>
57793ccfd646d0388e06c587e962a18fa723b72f02Chris Lattnerclass ObjCList : public ObjCListBase {
58793ccfd646d0388e06c587e962a18fa723b72f02Chris Lattnerpublic:
5938af2deb27cdfa1a95bde96e30dab15dce53fcefChris Lattner  void set(T* const* InList, unsigned Elts, ASTContext &Ctx) {
6038af2deb27cdfa1a95bde96e30dab15dce53fcefChris Lattner    ObjCListBase::set(reinterpret_cast<void*const*>(InList), Elts, Ctx);
61793ccfd646d0388e06c587e962a18fa723b72f02Chris Lattner  }
621eb4433ac451dc16f4133a88af2d002ac26c58efMike Stump
63793ccfd646d0388e06c587e962a18fa723b72f02Chris Lattner  typedef T* const * iterator;
64793ccfd646d0388e06c587e962a18fa723b72f02Chris Lattner  iterator begin() const { return (iterator)List; }
65793ccfd646d0388e06c587e962a18fa723b72f02Chris Lattner  iterator end() const { return (iterator)List+NumElts; }
661eb4433ac451dc16f4133a88af2d002ac26c58efMike Stump
67793ccfd646d0388e06c587e962a18fa723b72f02Chris Lattner  T* operator[](unsigned Idx) const {
68793ccfd646d0388e06c587e962a18fa723b72f02Chris Lattner    assert(Idx < NumElts && "Invalid access");
69793ccfd646d0388e06c587e962a18fa723b72f02Chris Lattner    return (T*)List[Idx];
703db6cae19c236fe2cef343c90105ce7cca7de965Chris Lattner  }
713db6cae19c236fe2cef343c90105ce7cca7de965Chris Lattner};
723db6cae19c236fe2cef343c90105ce7cca7de965Chris Lattner
7318df52bbb5d28ca082064d31ae7558dbdae52377Douglas Gregor/// \brief A list of Objective-C protocols, along with the source
7418df52bbb5d28ca082064d31ae7558dbdae52377Douglas Gregor/// locations at which they were referenced.
7518df52bbb5d28ca082064d31ae7558dbdae52377Douglas Gregorclass ObjCProtocolList : public ObjCList<ObjCProtocolDecl> {
7618df52bbb5d28ca082064d31ae7558dbdae52377Douglas Gregor  SourceLocation *Locations;
7718df52bbb5d28ca082064d31ae7558dbdae52377Douglas Gregor
7818df52bbb5d28ca082064d31ae7558dbdae52377Douglas Gregor  using ObjCList<ObjCProtocolDecl>::set;
7918df52bbb5d28ca082064d31ae7558dbdae52377Douglas Gregor
8018df52bbb5d28ca082064d31ae7558dbdae52377Douglas Gregorpublic:
8118df52bbb5d28ca082064d31ae7558dbdae52377Douglas Gregor  ObjCProtocolList() : ObjCList<ObjCProtocolDecl>(), Locations(0) { }
8218df52bbb5d28ca082064d31ae7558dbdae52377Douglas Gregor
8318df52bbb5d28ca082064d31ae7558dbdae52377Douglas Gregor  typedef const SourceLocation *loc_iterator;
8418df52bbb5d28ca082064d31ae7558dbdae52377Douglas Gregor  loc_iterator loc_begin() const { return Locations; }
8518df52bbb5d28ca082064d31ae7558dbdae52377Douglas Gregor  loc_iterator loc_end() const { return Locations + size(); }
8618df52bbb5d28ca082064d31ae7558dbdae52377Douglas Gregor
87ba243b59a1074e0962f6abfa3bb9aa984eac1245David Blaikie  void set(ObjCProtocolDecl* const* InList, unsigned Elts,
8818df52bbb5d28ca082064d31ae7558dbdae52377Douglas Gregor           const SourceLocation *Locs, ASTContext &Ctx);
8918df52bbb5d28ca082064d31ae7558dbdae52377Douglas Gregor};
901eb4433ac451dc16f4133a88af2d002ac26c58efMike Stump
9158dbdeb69c063f82d644504fc638120198f7fad2Steve Naroff
92a526c5c67e5a0473c340903ee542ce570119665fTed Kremenek/// ObjCMethodDecl - Represents an instance or class method declaration.
9358dbdeb69c063f82d644504fc638120198f7fad2Steve Naroff/// ObjC methods can be declared within 4 contexts: class interfaces,
9458dbdeb69c063f82d644504fc638120198f7fad2Steve Naroff/// categories, protocols, and class implementations. While C++ member
951eb4433ac451dc16f4133a88af2d002ac26c58efMike Stump/// functions leverage C syntax, Objective-C method syntax is modeled after
961eb4433ac451dc16f4133a88af2d002ac26c58efMike Stump/// Smalltalk (using colons to specify argument types/expressions).
9758dbdeb69c063f82d644504fc638120198f7fad2Steve Naroff/// Here are some brief examples:
9858dbdeb69c063f82d644504fc638120198f7fad2Steve Naroff///
9958dbdeb69c063f82d644504fc638120198f7fad2Steve Naroff/// Setter/getter instance methods:
10058dbdeb69c063f82d644504fc638120198f7fad2Steve Naroff/// - (void)setMenu:(NSMenu *)menu;
1011eb4433ac451dc16f4133a88af2d002ac26c58efMike Stump/// - (NSMenu *)menu;
1021eb4433ac451dc16f4133a88af2d002ac26c58efMike Stump///
10358dbdeb69c063f82d644504fc638120198f7fad2Steve Naroff/// Instance method that takes 2 NSView arguments:
10458dbdeb69c063f82d644504fc638120198f7fad2Steve Naroff/// - (void)replaceSubview:(NSView *)oldView with:(NSView *)newView;
10558dbdeb69c063f82d644504fc638120198f7fad2Steve Naroff///
10658dbdeb69c063f82d644504fc638120198f7fad2Steve Naroff/// Getter class method:
10758dbdeb69c063f82d644504fc638120198f7fad2Steve Naroff/// + (NSMenu *)defaultMenu;
10858dbdeb69c063f82d644504fc638120198f7fad2Steve Naroff///
10958dbdeb69c063f82d644504fc638120198f7fad2Steve Naroff/// A selector represents a unique name for a method. The selector names for
11058dbdeb69c063f82d644504fc638120198f7fad2Steve Naroff/// the above methods are setMenu:, menu, replaceSubview:with:, and defaultMenu.
11158dbdeb69c063f82d644504fc638120198f7fad2Steve Naroff///
1124afa39deaa245592977136d367251ee2c173dd8dDouglas Gregorclass ObjCMethodDecl : public NamedDecl, public DeclContext {
11358dbdeb69c063f82d644504fc638120198f7fad2Steve Naroffpublic:
11458dbdeb69c063f82d644504fc638120198f7fad2Steve Naroff  enum ImplementationControl { None, Required, Optional };
11558dbdeb69c063f82d644504fc638120198f7fad2Steve Naroffprivate:
11685f3d76c0ecfdefcf83ea44a57b7a16119c8a045John McCall  // The conventional meaning of this method; an ObjCMethodFamily.
11785f3d76c0ecfdefcf83ea44a57b7a16119c8a045John McCall  // This is not serialized; instead, it is computed on demand and
11885f3d76c0ecfdefcf83ea44a57b7a16119c8a045John McCall  // cached.
11985f3d76c0ecfdefcf83ea44a57b7a16119c8a045John McCall  mutable unsigned Family : ObjCMethodFamilyBitWidth;
12085f3d76c0ecfdefcf83ea44a57b7a16119c8a045John McCall
12158dbdeb69c063f82d644504fc638120198f7fad2Steve Naroff  /// instance (true) or class (false) method.
12285f3d76c0ecfdefcf83ea44a57b7a16119c8a045John McCall  unsigned IsInstance : 1;
12385f3d76c0ecfdefcf83ea44a57b7a16119c8a045John McCall  unsigned IsVariadic : 1;
1241eb4433ac451dc16f4133a88af2d002ac26c58efMike Stump
1254607034e621aa77378ec75249d1e9eaf5de49b6aFariborz Jahanian  // Synthesized declaration method for a property setter/getter
12685f3d76c0ecfdefcf83ea44a57b7a16119c8a045John McCall  unsigned IsSynthesized : 1;
127ba243b59a1074e0962f6abfa3bb9aa984eac1245David Blaikie
1283fe104154dd2e8ffb351142d74f308938b5c99bfFariborz Jahanian  // Method has a definition.
12985f3d76c0ecfdefcf83ea44a57b7a16119c8a045John McCall  unsigned IsDefined : 1;
1301eb4433ac451dc16f4133a88af2d002ac26c58efMike Stump
13172b2625aa67c8213acaf4bf6209b67859d60e2cfArgyrios Kyrtzidis  /// \brief Method redeclaration in the same interface.
1323a919e7110407ae7609bb6edc57aac16a5990661Argyrios Kyrtzidis  unsigned IsRedeclaration : 1;
1333a919e7110407ae7609bb6edc57aac16a5990661Argyrios Kyrtzidis
13472b2625aa67c8213acaf4bf6209b67859d60e2cfArgyrios Kyrtzidis  /// \brief Is redeclared in the same interface.
13572b2625aa67c8213acaf4bf6209b67859d60e2cfArgyrios Kyrtzidis  mutable unsigned HasRedeclaration : 1;
13672b2625aa67c8213acaf4bf6209b67859d60e2cfArgyrios Kyrtzidis
137ca63fa00786e51c207c829f4182f11a6c6b552beTed Kremenek  // NOTE: VC++ treats enums as signed, avoid using ImplementationControl enum
13858dbdeb69c063f82d644504fc638120198f7fad2Steve Naroff  /// @required/@optional
139ca63fa00786e51c207c829f4182f11a6c6b552beTed Kremenek  unsigned DeclImplementation : 2;
1401eb4433ac451dc16f4133a88af2d002ac26c58efMike Stump
141ca63fa00786e51c207c829f4182f11a6c6b552beTed Kremenek  // NOTE: VC++ treats enums as signed, avoid using the ObjCDeclQualifier enum
14258dbdeb69c063f82d644504fc638120198f7fad2Steve Naroff  /// in, inout, etc.
143ca63fa00786e51c207c829f4182f11a6c6b552beTed Kremenek  unsigned objcDeclQualifier : 6;
1441eb4433ac451dc16f4133a88af2d002ac26c58efMike Stump
145926df6cfabf3eaa4afc990c097fa4619b76a9b57Douglas Gregor  /// \brief Indicates whether this method has a related result type.
146926df6cfabf3eaa4afc990c097fa4619b76a9b57Douglas Gregor  unsigned RelatedResultType : 1;
147ba243b59a1074e0962f6abfa3bb9aa984eac1245David Blaikie
148491306a83c4f0f49f95a3bcbca8580cb98a91c7aArgyrios Kyrtzidis  /// \brief Whether the locations of the selector identifiers are in a
149491306a83c4f0f49f95a3bcbca8580cb98a91c7aArgyrios Kyrtzidis  /// "standard" position, a enum SelectorLocationsKind.
150491306a83c4f0f49f95a3bcbca8580cb98a91c7aArgyrios Kyrtzidis  unsigned SelLocsKind : 2;
1517732cc9c0fdc97a2f8cce4e5933d8103213d1aefFariborz Jahanian
1524bc1cb6aa635a5bf8fae99bf69c56c724c1e786cDouglas Gregor  // Result type of this method.
15358dbdeb69c063f82d644504fc638120198f7fad2Steve Naroff  QualType MethodDeclType;
154ba243b59a1074e0962f6abfa3bb9aa984eac1245David Blaikie
1554bc1cb6aa635a5bf8fae99bf69c56c724c1e786cDouglas Gregor  // Type source information for the result type.
1564bc1cb6aa635a5bf8fae99bf69c56c724c1e786cDouglas Gregor  TypeSourceInfo *ResultTInfo;
1574bc1cb6aa635a5bf8fae99bf69c56c724c1e786cDouglas Gregor
158491306a83c4f0f49f95a3bcbca8580cb98a91c7aArgyrios Kyrtzidis  /// \brief Array of ParmVarDecls for the formal parameters of this method
159491306a83c4f0f49f95a3bcbca8580cb98a91c7aArgyrios Kyrtzidis  /// and optionally followed by selector locations.
160491306a83c4f0f49f95a3bcbca8580cb98a91c7aArgyrios Kyrtzidis  void *ParamsAndSelLocs;
161491306a83c4f0f49f95a3bcbca8580cb98a91c7aArgyrios Kyrtzidis  unsigned NumParams;
1621eb4433ac451dc16f4133a88af2d002ac26c58efMike Stump
16358dbdeb69c063f82d644504fc638120198f7fad2Steve Naroff  /// List of attributes for this method declaration.
164a2e85ada1dfef36201a31f6646bc4ea3bd76a89aArgyrios Kyrtzidis  SourceLocation EndLoc; // the location of the ';' or '}'.
1651eb4433ac451dc16f4133a88af2d002ac26c58efMike Stump
16658dbdeb69c063f82d644504fc638120198f7fad2Steve Naroff  // The following are only used for method definitions, null otherwise.
16758dbdeb69c063f82d644504fc638120198f7fad2Steve Naroff  // FIXME: space savings opportunity, consider a sub-class.
16858dbdeb69c063f82d644504fc638120198f7fad2Steve Naroff  Stmt *Body;
169451318c08a6342c10b8986060386fd9274418437Daniel Dunbar
170451318c08a6342c10b8986060386fd9274418437Daniel Dunbar  /// SelfDecl - Decl for the implicit self parameter. This is lazily
171451318c08a6342c10b8986060386fd9274418437Daniel Dunbar  /// constructed by createImplicitParams.
1724111024be81e7c0525e42dadcc126d27e5bf2425Chris Lattner  ImplicitParamDecl *SelfDecl;
173451318c08a6342c10b8986060386fd9274418437Daniel Dunbar  /// CmdDecl - Decl for the implicit _cmd parameter. This is lazily
174451318c08a6342c10b8986060386fd9274418437Daniel Dunbar  /// constructed by createImplicitParams.
1754111024be81e7c0525e42dadcc126d27e5bf2425Chris Lattner  ImplicitParamDecl *CmdDecl;
1761eb4433ac451dc16f4133a88af2d002ac26c58efMike Stump
177491306a83c4f0f49f95a3bcbca8580cb98a91c7aArgyrios Kyrtzidis  SelectorLocationsKind getSelLocsKind() const {
178491306a83c4f0f49f95a3bcbca8580cb98a91c7aArgyrios Kyrtzidis    return (SelectorLocationsKind)SelLocsKind;
179491306a83c4f0f49f95a3bcbca8580cb98a91c7aArgyrios Kyrtzidis  }
180491306a83c4f0f49f95a3bcbca8580cb98a91c7aArgyrios Kyrtzidis  bool hasStandardSelLocs() const {
181491306a83c4f0f49f95a3bcbca8580cb98a91c7aArgyrios Kyrtzidis    return getSelLocsKind() != SelLoc_NonStandard;
182491306a83c4f0f49f95a3bcbca8580cb98a91c7aArgyrios Kyrtzidis  }
183491306a83c4f0f49f95a3bcbca8580cb98a91c7aArgyrios Kyrtzidis
184491306a83c4f0f49f95a3bcbca8580cb98a91c7aArgyrios Kyrtzidis  /// \brief Get a pointer to the stored selector identifiers locations array.
185491306a83c4f0f49f95a3bcbca8580cb98a91c7aArgyrios Kyrtzidis  /// No locations will be stored if HasStandardSelLocs is true.
186491306a83c4f0f49f95a3bcbca8580cb98a91c7aArgyrios Kyrtzidis  SourceLocation *getStoredSelLocs() {
187491306a83c4f0f49f95a3bcbca8580cb98a91c7aArgyrios Kyrtzidis    return reinterpret_cast<SourceLocation*>(getParams() + NumParams);
188491306a83c4f0f49f95a3bcbca8580cb98a91c7aArgyrios Kyrtzidis  }
189491306a83c4f0f49f95a3bcbca8580cb98a91c7aArgyrios Kyrtzidis  const SourceLocation *getStoredSelLocs() const {
190491306a83c4f0f49f95a3bcbca8580cb98a91c7aArgyrios Kyrtzidis    return reinterpret_cast<const SourceLocation*>(getParams() + NumParams);
191491306a83c4f0f49f95a3bcbca8580cb98a91c7aArgyrios Kyrtzidis  }
192491306a83c4f0f49f95a3bcbca8580cb98a91c7aArgyrios Kyrtzidis
193491306a83c4f0f49f95a3bcbca8580cb98a91c7aArgyrios Kyrtzidis  /// \brief Get a pointer to the stored selector identifiers locations array.
194491306a83c4f0f49f95a3bcbca8580cb98a91c7aArgyrios Kyrtzidis  /// No locations will be stored if HasStandardSelLocs is true.
195491306a83c4f0f49f95a3bcbca8580cb98a91c7aArgyrios Kyrtzidis  ParmVarDecl **getParams() {
196491306a83c4f0f49f95a3bcbca8580cb98a91c7aArgyrios Kyrtzidis    return reinterpret_cast<ParmVarDecl **>(ParamsAndSelLocs);
197491306a83c4f0f49f95a3bcbca8580cb98a91c7aArgyrios Kyrtzidis  }
198491306a83c4f0f49f95a3bcbca8580cb98a91c7aArgyrios Kyrtzidis  const ParmVarDecl *const *getParams() const {
199491306a83c4f0f49f95a3bcbca8580cb98a91c7aArgyrios Kyrtzidis    return reinterpret_cast<const ParmVarDecl *const *>(ParamsAndSelLocs);
200491306a83c4f0f49f95a3bcbca8580cb98a91c7aArgyrios Kyrtzidis  }
201491306a83c4f0f49f95a3bcbca8580cb98a91c7aArgyrios Kyrtzidis
202491306a83c4f0f49f95a3bcbca8580cb98a91c7aArgyrios Kyrtzidis  /// \brief Get the number of stored selector identifiers locations.
203491306a83c4f0f49f95a3bcbca8580cb98a91c7aArgyrios Kyrtzidis  /// No locations will be stored if HasStandardSelLocs is true.
204491306a83c4f0f49f95a3bcbca8580cb98a91c7aArgyrios Kyrtzidis  unsigned getNumStoredSelLocs() const {
205491306a83c4f0f49f95a3bcbca8580cb98a91c7aArgyrios Kyrtzidis    if (hasStandardSelLocs())
206491306a83c4f0f49f95a3bcbca8580cb98a91c7aArgyrios Kyrtzidis      return 0;
207491306a83c4f0f49f95a3bcbca8580cb98a91c7aArgyrios Kyrtzidis    return getNumSelectorLocs();
208491306a83c4f0f49f95a3bcbca8580cb98a91c7aArgyrios Kyrtzidis  }
209491306a83c4f0f49f95a3bcbca8580cb98a91c7aArgyrios Kyrtzidis
210491306a83c4f0f49f95a3bcbca8580cb98a91c7aArgyrios Kyrtzidis  void setParamsAndSelLocs(ASTContext &C,
211491306a83c4f0f49f95a3bcbca8580cb98a91c7aArgyrios Kyrtzidis                           ArrayRef<ParmVarDecl*> Params,
212491306a83c4f0f49f95a3bcbca8580cb98a91c7aArgyrios Kyrtzidis                           ArrayRef<SourceLocation> SelLocs);
213491306a83c4f0f49f95a3bcbca8580cb98a91c7aArgyrios Kyrtzidis
214a526c5c67e5a0473c340903ee542ce570119665fTed Kremenek  ObjCMethodDecl(SourceLocation beginLoc, SourceLocation endLoc,
21558dbdeb69c063f82d644504fc638120198f7fad2Steve Naroff                 Selector SelInfo, QualType T,
2164bc1cb6aa635a5bf8fae99bf69c56c724c1e786cDouglas Gregor                 TypeSourceInfo *ResultTInfo,
2170701bbb228dfd87e1fe82a0a4b7b9facfecb43daSteve Naroff                 DeclContext *contextDecl,
218f6414927e67e27d9324d8d179c5f7ea620443924Daniel Dunbar                 bool isInstance = true,
21958dbdeb69c063f82d644504fc638120198f7fad2Steve Naroff                 bool isVariadic = false,
2204607034e621aa77378ec75249d1e9eaf5de49b6aFariborz Jahanian                 bool isSynthesized = false,
22175cf3e86d33ce810c12084126385371b335c30baArgyrios Kyrtzidis                 bool isImplicitlyDeclared = false,
2223fe104154dd2e8ffb351142d74f308938b5c99bfFariborz Jahanian                 bool isDefined = false,
2237732cc9c0fdc97a2f8cce4e5933d8103213d1aefFariborz Jahanian                 ImplementationControl impControl = None,
224da92a7f91cf88f49e02050919676f7fb8e3bdff8Argyrios Kyrtzidis                 bool HasRelatedResultType = false)
2254afa39deaa245592977136d367251ee2c173dd8dDouglas Gregor  : NamedDecl(ObjCMethod, contextDecl, beginLoc, SelInfo),
22685f3d76c0ecfdefcf83ea44a57b7a16119c8a045John McCall    DeclContext(ObjCMethod), Family(InvalidObjCMethodFamily),
22758dbdeb69c063f82d644504fc638120198f7fad2Steve Naroff    IsInstance(isInstance), IsVariadic(isVariadic),
2284607034e621aa77378ec75249d1e9eaf5de49b6aFariborz Jahanian    IsSynthesized(isSynthesized),
22972b2625aa67c8213acaf4bf6209b67859d60e2cfArgyrios Kyrtzidis    IsDefined(isDefined), IsRedeclaration(0), HasRedeclaration(0),
23058dbdeb69c063f82d644504fc638120198f7fad2Steve Naroff    DeclImplementation(impControl), objcDeclQualifier(OBJC_TQ_None),
231da92a7f91cf88f49e02050919676f7fb8e3bdff8Argyrios Kyrtzidis    RelatedResultType(HasRelatedResultType),
232491306a83c4f0f49f95a3bcbca8580cb98a91c7aArgyrios Kyrtzidis    SelLocsKind(SelLoc_StandardNoSpace),
233926df6cfabf3eaa4afc990c097fa4619b76a9b57Douglas Gregor    MethodDeclType(T), ResultTInfo(ResultTInfo),
234491306a83c4f0f49f95a3bcbca8580cb98a91c7aArgyrios Kyrtzidis    ParamsAndSelLocs(0), NumParams(0),
23575cf3e86d33ce810c12084126385371b335c30baArgyrios Kyrtzidis    EndLoc(endLoc), Body(0), SelfDecl(0), CmdDecl(0) {
23675cf3e86d33ce810c12084126385371b335c30baArgyrios Kyrtzidis    setImplicit(isImplicitlyDeclared);
23775cf3e86d33ce810c12084126385371b335c30baArgyrios Kyrtzidis  }
2388a779314870760848e61da2c428a78971fe3f1c3Ted Kremenek
23957ea6bee79cc60ba20c7886b453f40f380dce1b1Argyrios Kyrtzidis  /// \brief A definition will return its interface declaration.
24057ea6bee79cc60ba20c7886b453f40f380dce1b1Argyrios Kyrtzidis  /// An interface declaration will return its definition.
24157ea6bee79cc60ba20c7886b453f40f380dce1b1Argyrios Kyrtzidis  /// Otherwise it will return itself.
242da2142f2e2b3a02ee6eb5de9f9e6ed6f7eb5a0c0Douglas Gregor  virtual ObjCMethodDecl *getNextRedeclaration();
243da2142f2e2b3a02ee6eb5de9f9e6ed6f7eb5a0c0Douglas Gregor
2446c4ae5de0c356777446f823b573821fb95560d91Chris Lattnerpublic:
2450ed844b04ea4387caa4e1cf3dc375d269657536bChris Lattner  static ObjCMethodDecl *Create(ASTContext &C,
2461eb4433ac451dc16f4133a88af2d002ac26c58efMike Stump                                SourceLocation beginLoc,
24711d77169555480ee0a04c6e5bc390d8fde41175dArgyrios Kyrtzidis                                SourceLocation endLoc,
24811d77169555480ee0a04c6e5bc390d8fde41175dArgyrios Kyrtzidis                                Selector SelInfo,
249ba243b59a1074e0962f6abfa3bb9aa984eac1245David Blaikie                                QualType T,
2504bc1cb6aa635a5bf8fae99bf69c56c724c1e786cDouglas Gregor                                TypeSourceInfo *ResultTInfo,
2514bc1cb6aa635a5bf8fae99bf69c56c724c1e786cDouglas Gregor                                DeclContext *contextDecl,
252f6414927e67e27d9324d8d179c5f7ea620443924Daniel Dunbar                                bool isInstance = true,
2536c4ae5de0c356777446f823b573821fb95560d91Chris Lattner                                bool isVariadic = false,
2544607034e621aa77378ec75249d1e9eaf5de49b6aFariborz Jahanian                                bool isSynthesized = false,
25575cf3e86d33ce810c12084126385371b335c30baArgyrios Kyrtzidis                                bool isImplicitlyDeclared = false,
2563fe104154dd2e8ffb351142d74f308938b5c99bfFariborz Jahanian                                bool isDefined = false,
2577732cc9c0fdc97a2f8cce4e5933d8103213d1aefFariborz Jahanian                                ImplementationControl impControl = None,
258da92a7f91cf88f49e02050919676f7fb8e3bdff8Argyrios Kyrtzidis                                bool HasRelatedResultType = false);
259e7f9d301a10b4b3223e91d9be4362b44cba0a212Argyrios Kyrtzidis
260da2142f2e2b3a02ee6eb5de9f9e6ed6f7eb5a0c0Douglas Gregor  virtual ObjCMethodDecl *getCanonicalDecl();
26113e635c55eb23f736d38abf9c954801bd6482db4Ted Kremenek  const ObjCMethodDecl *getCanonicalDecl() const {
26213e635c55eb23f736d38abf9c954801bd6482db4Ted Kremenek    return const_cast<ObjCMethodDecl*>(this)->getCanonicalDecl();
26313e635c55eb23f736d38abf9c954801bd6482db4Ted Kremenek  }
264e7f9d301a10b4b3223e91d9be4362b44cba0a212Argyrios Kyrtzidis
265ca63fa00786e51c207c829f4182f11a6c6b552beTed Kremenek  ObjCDeclQualifier getObjCDeclQualifier() const {
266ca63fa00786e51c207c829f4182f11a6c6b552beTed Kremenek    return ObjCDeclQualifier(objcDeclQualifier);
267ca63fa00786e51c207c829f4182f11a6c6b552beTed Kremenek  }
268a526c5c67e5a0473c340903ee542ce570119665fTed Kremenek  void setObjCDeclQualifier(ObjCDeclQualifier QV) { objcDeclQualifier = QV; }
2691eb4433ac451dc16f4133a88af2d002ac26c58efMike Stump
270926df6cfabf3eaa4afc990c097fa4619b76a9b57Douglas Gregor  /// \brief Determine whether this method has a result type that is related
271926df6cfabf3eaa4afc990c097fa4619b76a9b57Douglas Gregor  /// to the message receiver's type.
272926df6cfabf3eaa4afc990c097fa4619b76a9b57Douglas Gregor  bool hasRelatedResultType() const { return RelatedResultType; }
273ba243b59a1074e0962f6abfa3bb9aa984eac1245David Blaikie
274926df6cfabf3eaa4afc990c097fa4619b76a9b57Douglas Gregor  /// \brief Note whether this method has a related result type.
275926df6cfabf3eaa4afc990c097fa4619b76a9b57Douglas Gregor  void SetRelatedResultType(bool RRT = true) { RelatedResultType = RRT; }
2763a919e7110407ae7609bb6edc57aac16a5990661Argyrios Kyrtzidis
2773a919e7110407ae7609bb6edc57aac16a5990661Argyrios Kyrtzidis  /// \brief True if this is a method redeclaration in the same interface.
2783a919e7110407ae7609bb6edc57aac16a5990661Argyrios Kyrtzidis  bool isRedeclaration() const { return IsRedeclaration; }
2793a919e7110407ae7609bb6edc57aac16a5990661Argyrios Kyrtzidis  void setAsRedeclaration(const ObjCMethodDecl *PrevMethod);
280ba243b59a1074e0962f6abfa3bb9aa984eac1245David Blaikie
28158dbdeb69c063f82d644504fc638120198f7fad2Steve Naroff  // Location information, modeled after the Stmt API.
28258dbdeb69c063f82d644504fc638120198f7fad2Steve Naroff  SourceLocation getLocStart() const { return getLocation(); }
28358dbdeb69c063f82d644504fc638120198f7fad2Steve Naroff  SourceLocation getLocEnd() const { return EndLoc; }
28453c9d8a4b8f0a76cb9dd2fdd8c433ccf110f2eecSteve Naroff  void setEndLoc(SourceLocation Loc) { EndLoc = Loc; }
285da2142f2e2b3a02ee6eb5de9f9e6ed6f7eb5a0c0Douglas Gregor  virtual SourceRange getSourceRange() const {
2861eb4433ac451dc16f4133a88af2d002ac26c58efMike Stump    return SourceRange(getLocation(), EndLoc);
2879776ba0d844cf9f6888e871e3fd246ae782f76f4Daniel Dunbar  }
2881eb4433ac451dc16f4133a88af2d002ac26c58efMike Stump
289491306a83c4f0f49f95a3bcbca8580cb98a91c7aArgyrios Kyrtzidis  SourceLocation getSelectorStartLoc() const { return getSelectorLoc(0); }
290491306a83c4f0f49f95a3bcbca8580cb98a91c7aArgyrios Kyrtzidis  SourceLocation getSelectorLoc(unsigned Index) const {
291491306a83c4f0f49f95a3bcbca8580cb98a91c7aArgyrios Kyrtzidis    assert(Index < getNumSelectorLocs() && "Index out of range!");
292491306a83c4f0f49f95a3bcbca8580cb98a91c7aArgyrios Kyrtzidis    if (hasStandardSelLocs())
293491306a83c4f0f49f95a3bcbca8580cb98a91c7aArgyrios Kyrtzidis      return getStandardSelectorLoc(Index, getSelector(),
294491306a83c4f0f49f95a3bcbca8580cb98a91c7aArgyrios Kyrtzidis                                   getSelLocsKind() == SelLoc_StandardWithSpace,
295491306a83c4f0f49f95a3bcbca8580cb98a91c7aArgyrios Kyrtzidis                      llvm::makeArrayRef(const_cast<ParmVarDecl**>(getParams()),
296491306a83c4f0f49f95a3bcbca8580cb98a91c7aArgyrios Kyrtzidis                                         NumParams),
297491306a83c4f0f49f95a3bcbca8580cb98a91c7aArgyrios Kyrtzidis                                   EndLoc);
298491306a83c4f0f49f95a3bcbca8580cb98a91c7aArgyrios Kyrtzidis    return getStoredSelLocs()[Index];
299491306a83c4f0f49f95a3bcbca8580cb98a91c7aArgyrios Kyrtzidis  }
300491306a83c4f0f49f95a3bcbca8580cb98a91c7aArgyrios Kyrtzidis
301491306a83c4f0f49f95a3bcbca8580cb98a91c7aArgyrios Kyrtzidis  void getSelectorLocs(SmallVectorImpl<SourceLocation> &SelLocs) const;
302491306a83c4f0f49f95a3bcbca8580cb98a91c7aArgyrios Kyrtzidis
303491306a83c4f0f49f95a3bcbca8580cb98a91c7aArgyrios Kyrtzidis  unsigned getNumSelectorLocs() const {
304491306a83c4f0f49f95a3bcbca8580cb98a91c7aArgyrios Kyrtzidis    if (isImplicit())
305491306a83c4f0f49f95a3bcbca8580cb98a91c7aArgyrios Kyrtzidis      return 0;
306491306a83c4f0f49f95a3bcbca8580cb98a91c7aArgyrios Kyrtzidis    Selector Sel = getSelector();
307491306a83c4f0f49f95a3bcbca8580cb98a91c7aArgyrios Kyrtzidis    if (Sel.isUnarySelector())
308491306a83c4f0f49f95a3bcbca8580cb98a91c7aArgyrios Kyrtzidis      return 1;
309491306a83c4f0f49f95a3bcbca8580cb98a91c7aArgyrios Kyrtzidis    return Sel.getNumArgs();
310491306a83c4f0f49f95a3bcbca8580cb98a91c7aArgyrios Kyrtzidis  }
311491306a83c4f0f49f95a3bcbca8580cb98a91c7aArgyrios Kyrtzidis
3125619688510185081cbb4621d703daf7ee24cf39aChris Lattner  ObjCInterfaceDecl *getClassInterface();
3135619688510185081cbb4621d703daf7ee24cf39aChris Lattner  const ObjCInterfaceDecl *getClassInterface() const {
3145619688510185081cbb4621d703daf7ee24cf39aChris Lattner    return const_cast<ObjCMethodDecl*>(this)->getClassInterface();
315e0def7589a8afa8a6acef13476bb3f882c104b91Chris Lattner  }
3161eb4433ac451dc16f4133a88af2d002ac26c58efMike Stump
3172e1cd4264d363ca869bf37ef160902f211d21b8cDouglas Gregor  Selector getSelector() const { return getDeclName().getObjCSelector(); }
3183a082d81006e7a2e01a6e431a22e21c78490ff8fAnders Carlsson
31958dbdeb69c063f82d644504fc638120198f7fad2Steve Naroff  QualType getResultType() const { return MethodDeclType; }
32053c9d8a4b8f0a76cb9dd2fdd8c433ccf110f2eecSteve Naroff  void setResultType(QualType T) { MethodDeclType = T; }
3211eb4433ac451dc16f4133a88af2d002ac26c58efMike Stump
322ba243b59a1074e0962f6abfa3bb9aa984eac1245David Blaikie  /// \brief Determine the type of an expression that sends a message to this
3235291c3cec0dbe8ad1d8e7e67e93af2b1586d5400Douglas Gregor  /// function.
3245291c3cec0dbe8ad1d8e7e67e93af2b1586d5400Douglas Gregor  QualType getSendResultType() const {
3256398235d7890a81b785ea5af3b6e66d86bf184ccDouglas Gregor    return getResultType().getNonLValueExprType(getASTContext());
3265291c3cec0dbe8ad1d8e7e67e93af2b1586d5400Douglas Gregor  }
327ba243b59a1074e0962f6abfa3bb9aa984eac1245David Blaikie
3284bc1cb6aa635a5bf8fae99bf69c56c724c1e786cDouglas Gregor  TypeSourceInfo *getResultTypeSourceInfo() const { return ResultTInfo; }
3294bc1cb6aa635a5bf8fae99bf69c56c724c1e786cDouglas Gregor  void setResultTypeSourceInfo(TypeSourceInfo *TInfo) { ResultTInfo = TInfo; }
3304bc1cb6aa635a5bf8fae99bf69c56c724c1e786cDouglas Gregor
331d57f635d520e8cb5b93d3d770ff58db06c62de54Chris Lattner  // Iterator access to formal parameters.
332491306a83c4f0f49f95a3bcbca8580cb98a91c7aArgyrios Kyrtzidis  unsigned param_size() const { return NumParams; }
333491306a83c4f0f49f95a3bcbca8580cb98a91c7aArgyrios Kyrtzidis  typedef const ParmVarDecl *const *param_const_iterator;
334491306a83c4f0f49f95a3bcbca8580cb98a91c7aArgyrios Kyrtzidis  typedef ParmVarDecl *const *param_iterator;
335491306a83c4f0f49f95a3bcbca8580cb98a91c7aArgyrios Kyrtzidis  param_const_iterator param_begin() const { return getParams(); }
336491306a83c4f0f49f95a3bcbca8580cb98a91c7aArgyrios Kyrtzidis  param_const_iterator param_end() const { return getParams() + NumParams; }
337491306a83c4f0f49f95a3bcbca8580cb98a91c7aArgyrios Kyrtzidis  param_iterator param_begin() { return getParams(); }
338491306a83c4f0f49f95a3bcbca8580cb98a91c7aArgyrios Kyrtzidis  param_iterator param_end() { return getParams() + NumParams; }
3397732cc9c0fdc97a2f8cce4e5933d8103213d1aefFariborz Jahanian  // This method returns and of the parameters which are part of the selector
3407732cc9c0fdc97a2f8cce4e5933d8103213d1aefFariborz Jahanian  // name mangling requirements.
341ba243b59a1074e0962f6abfa3bb9aa984eac1245David Blaikie  param_const_iterator sel_param_end() const {
342ba243b59a1074e0962f6abfa3bb9aa984eac1245David Blaikie    return param_begin() + getSelector().getNumArgs();
3437732cc9c0fdc97a2f8cce4e5933d8103213d1aefFariborz Jahanian  }
34489951a86b594513c2a013532ed45d197413b1087Chris Lattner
3459dd0065e61ea4b48b19eee550704ce964e64e946Argyrios Kyrtzidis  /// \brief Sets the method's parameters and selector source locations.
3469dd0065e61ea4b48b19eee550704ce964e64e946Argyrios Kyrtzidis  /// If the method is implicit (not coming from source) \arg SelLocs is
3479dd0065e61ea4b48b19eee550704ce964e64e946Argyrios Kyrtzidis  /// ignored.
348491306a83c4f0f49f95a3bcbca8580cb98a91c7aArgyrios Kyrtzidis  void setMethodParams(ASTContext &C,
349491306a83c4f0f49f95a3bcbca8580cb98a91c7aArgyrios Kyrtzidis                       ArrayRef<ParmVarDecl*> Params,
3509dd0065e61ea4b48b19eee550704ce964e64e946Argyrios Kyrtzidis                       ArrayRef<SourceLocation> SelLocs =
3519dd0065e61ea4b48b19eee550704ce964e64e946Argyrios Kyrtzidis                           ArrayRef<SourceLocation>());
3524111024be81e7c0525e42dadcc126d27e5bf2425Chris Lattner
3536ea8e2152e1ba93b4c80e7268403a582896dc3dcAnders Carlsson  // Iterator access to parameter types.
3546ea8e2152e1ba93b4c80e7268403a582896dc3dcAnders Carlsson  typedef std::const_mem_fun_t<QualType, ParmVarDecl> deref_fun;
355491306a83c4f0f49f95a3bcbca8580cb98a91c7aArgyrios Kyrtzidis  typedef llvm::mapped_iterator<param_const_iterator, deref_fun>
356491306a83c4f0f49f95a3bcbca8580cb98a91c7aArgyrios Kyrtzidis      arg_type_iterator;
3576ea8e2152e1ba93b4c80e7268403a582896dc3dcAnders Carlsson
3586ea8e2152e1ba93b4c80e7268403a582896dc3dcAnders Carlsson  arg_type_iterator arg_type_begin() const {
3596ea8e2152e1ba93b4c80e7268403a582896dc3dcAnders Carlsson    return llvm::map_iterator(param_begin(), deref_fun(&ParmVarDecl::getType));
3606ea8e2152e1ba93b4c80e7268403a582896dc3dcAnders Carlsson  }
3616ea8e2152e1ba93b4c80e7268403a582896dc3dcAnders Carlsson  arg_type_iterator arg_type_end() const {
3626ea8e2152e1ba93b4c80e7268403a582896dc3dcAnders Carlsson    return llvm::map_iterator(param_end(), deref_fun(&ParmVarDecl::getType));
3636ea8e2152e1ba93b4c80e7268403a582896dc3dcAnders Carlsson  }
3641eb4433ac451dc16f4133a88af2d002ac26c58efMike Stump
365451318c08a6342c10b8986060386fd9274418437Daniel Dunbar  /// createImplicitParams - Used to lazily create the self and cmd
366451318c08a6342c10b8986060386fd9274418437Daniel Dunbar  /// implict parameters. This must be called prior to using getSelfDecl()
367451318c08a6342c10b8986060386fd9274418437Daniel Dunbar  /// or getCmdDecl(). The call is ignored if the implicit paramters
368451318c08a6342c10b8986060386fd9274418437Daniel Dunbar  /// have already been created.
369fef30b55230064d334a669a065a1c9acdb87cdfeFariborz Jahanian  void createImplicitParams(ASTContext &Context, const ObjCInterfaceDecl *ID);
370451318c08a6342c10b8986060386fd9274418437Daniel Dunbar
3714111024be81e7c0525e42dadcc126d27e5bf2425Chris Lattner  ImplicitParamDecl * getSelfDecl() const { return SelfDecl; }
37253c9d8a4b8f0a76cb9dd2fdd8c433ccf110f2eecSteve Naroff  void setSelfDecl(ImplicitParamDecl *SD) { SelfDecl = SD; }
3734111024be81e7c0525e42dadcc126d27e5bf2425Chris Lattner  ImplicitParamDecl * getCmdDecl() const { return CmdDecl; }
37453c9d8a4b8f0a76cb9dd2fdd8c433ccf110f2eecSteve Naroff  void setCmdDecl(ImplicitParamDecl *CD) { CmdDecl = CD; }
3751eb4433ac451dc16f4133a88af2d002ac26c58efMike Stump
37685f3d76c0ecfdefcf83ea44a57b7a16119c8a045John McCall  /// Determines the family of this method.
37785f3d76c0ecfdefcf83ea44a57b7a16119c8a045John McCall  ObjCMethodFamily getMethodFamily() const;
37885f3d76c0ecfdefcf83ea44a57b7a16119c8a045John McCall
379f8d49f64ef6ab7e632717a31631fc289aab69428Douglas Gregor  bool isInstanceMethod() const { return IsInstance; }
38053c9d8a4b8f0a76cb9dd2fdd8c433ccf110f2eecSteve Naroff  void setInstanceMethod(bool isInst) { IsInstance = isInst; }
38158dbdeb69c063f82d644504fc638120198f7fad2Steve Naroff  bool isVariadic() const { return IsVariadic; }
38253c9d8a4b8f0a76cb9dd2fdd8c433ccf110f2eecSteve Naroff  void setVariadic(bool isVar) { IsVariadic = isVar; }
3831eb4433ac451dc16f4133a88af2d002ac26c58efMike Stump
384f8d49f64ef6ab7e632717a31631fc289aab69428Douglas Gregor  bool isClassMethod() const { return !IsInstance; }
385f8d49f64ef6ab7e632717a31631fc289aab69428Douglas Gregor
3864607034e621aa77378ec75249d1e9eaf5de49b6aFariborz Jahanian  bool isSynthesized() const { return IsSynthesized; }
38753c9d8a4b8f0a76cb9dd2fdd8c433ccf110f2eecSteve Naroff  void setSynthesized(bool isSynth) { IsSynthesized = isSynth; }
388ba243b59a1074e0962f6abfa3bb9aa984eac1245David Blaikie
3893fe104154dd2e8ffb351142d74f308938b5c99bfFariborz Jahanian  bool isDefined() const { return IsDefined; }
3903fe104154dd2e8ffb351142d74f308938b5c99bfFariborz Jahanian  void setDefined(bool isDefined) { IsDefined = isDefined; }
3911eb4433ac451dc16f4133a88af2d002ac26c58efMike Stump
39258dbdeb69c063f82d644504fc638120198f7fad2Steve Naroff  // Related to protocols declared in  @protocol
3931eb4433ac451dc16f4133a88af2d002ac26c58efMike Stump  void setDeclImplementation(ImplementationControl ic) {
3941eb4433ac451dc16f4133a88af2d002ac26c58efMike Stump    DeclImplementation = ic;
39558dbdeb69c063f82d644504fc638120198f7fad2Steve Naroff  }
3961eb4433ac451dc16f4133a88af2d002ac26c58efMike Stump  ImplementationControl getImplementationControl() const {
3971eb4433ac451dc16f4133a88af2d002ac26c58efMike Stump    return ImplementationControl(DeclImplementation);
39858dbdeb69c063f82d644504fc638120198f7fad2Steve Naroff  }
399792481eec23d8c1aa92173be589e2ae9d02514a5Ted Kremenek
400da2142f2e2b3a02ee6eb5de9f9e6ed6f7eb5a0c0Douglas Gregor  virtual Stmt *getBody() const {
4011eb4433ac451dc16f4133a88af2d002ac26c58efMike Stump    return (Stmt*) Body;
4027297134f128423fce2e88f92421ed135bded7d4eDouglas Gregor  }
4036fb0aee4f9dc261bbec72e1283ad8dc0557a6d96Argyrios Kyrtzidis  CompoundStmt *getCompoundBody() { return (CompoundStmt*)Body; }
404d3a413d3b8eb39bcee5944bc545d9997c1abe492Sebastian Redl  void setBody(Stmt *B) { Body = B; }
40558dbdeb69c063f82d644504fc638120198f7fad2Steve Naroff
40666570b230941651245accbc5680b60e904eb993cArgyrios Kyrtzidis  /// \brief Returns whether this specific method is a definition.
40766570b230941651245accbc5680b60e904eb993cArgyrios Kyrtzidis  bool isThisDeclarationADefinition() const { return Body; }
40866570b230941651245accbc5680b60e904eb993cArgyrios Kyrtzidis
40958dbdeb69c063f82d644504fc638120198f7fad2Steve Naroff  // Implement isa/cast/dyncast/etc.
41080cd64a8450d8e2c079dc134d9711cd45ba89d63John McCall  static bool classof(const Decl *D) { return classofKind(D->getKind()); }
411a526c5c67e5a0473c340903ee542ce570119665fTed Kremenek  static bool classof(const ObjCMethodDecl *D) { return true; }
41280cd64a8450d8e2c079dc134d9711cd45ba89d63John McCall  static bool classofKind(Kind K) { return K == ObjCMethod; }
41342220c5432c141d47cc8ce786e472b49dc907378Argyrios Kyrtzidis  static DeclContext *castToDeclContext(const ObjCMethodDecl *D) {
41442220c5432c141d47cc8ce786e472b49dc907378Argyrios Kyrtzidis    return static_cast<DeclContext *>(const_cast<ObjCMethodDecl*>(D));
41542220c5432c141d47cc8ce786e472b49dc907378Argyrios Kyrtzidis  }
41642220c5432c141d47cc8ce786e472b49dc907378Argyrios Kyrtzidis  static ObjCMethodDecl *castFromDeclContext(const DeclContext *DC) {
41742220c5432c141d47cc8ce786e472b49dc907378Argyrios Kyrtzidis    return static_cast<ObjCMethodDecl *>(const_cast<DeclContext*>(DC));
41842220c5432c141d47cc8ce786e472b49dc907378Argyrios Kyrtzidis  }
419491306a83c4f0f49f95a3bcbca8580cb98a91c7aArgyrios Kyrtzidis
420491306a83c4f0f49f95a3bcbca8580cb98a91c7aArgyrios Kyrtzidis  friend class ASTDeclReader;
421491306a83c4f0f49f95a3bcbca8580cb98a91c7aArgyrios Kyrtzidis  friend class ASTDeclWriter;
42258dbdeb69c063f82d644504fc638120198f7fad2Steve Naroff};
423e4f039e01e797a38bc97bf22aff9832ecd18ff5fSteve Naroff
424e4f039e01e797a38bc97bf22aff9832ecd18ff5fSteve Naroff/// ObjCContainerDecl - Represents a container for method declarations.
425aecae629269fae3bf484baf1d109e9a89d14eeadArgyrios Kyrtzidis/// Current sub-classes are ObjCInterfaceDecl, ObjCCategoryDecl,
4261eb4433ac451dc16f4133a88af2d002ac26c58efMike Stump/// ObjCProtocolDecl, and ObjCImplDecl.
427e4f039e01e797a38bc97bf22aff9832ecd18ff5fSteve Naroff///
4284afa39deaa245592977136d367251ee2c173dd8dDouglas Gregorclass ObjCContainerDecl : public NamedDecl, public DeclContext {
42999ba9e3bd70671f3441fb974895f226a83ce0e66David Blaikie  virtual void anchor();
43099ba9e3bd70671f3441fb974895f226a83ce0e66David Blaikie
4311711fc91efb36d131f7ba771f73f0154dc1abd1fArgyrios Kyrtzidis  SourceLocation AtStart;
4321711fc91efb36d131f7ba771f73f0154dc1abd1fArgyrios Kyrtzidis
433782f2f52b78d8ca785110398a7f7b56b830b9ac7Ted Kremenek  // These two locations in the range mark the end of the method container.
434782f2f52b78d8ca785110398a7f7b56b830b9ac7Ted Kremenek  // The first points to the '@' token, and the second to the 'end' token.
435782f2f52b78d8ca785110398a7f7b56b830b9ac7Ted Kremenek  SourceRange AtEnd;
436e4f039e01e797a38bc97bf22aff9832ecd18ff5fSteve Naroffpublic:
437e4f039e01e797a38bc97bf22aff9832ecd18ff5fSteve Naroff
4381711fc91efb36d131f7ba771f73f0154dc1abd1fArgyrios Kyrtzidis  ObjCContainerDecl(Kind DK, DeclContext *DC,
4391711fc91efb36d131f7ba771f73f0154dc1abd1fArgyrios Kyrtzidis                    IdentifierInfo *Id, SourceLocation nameLoc,
4401711fc91efb36d131f7ba771f73f0154dc1abd1fArgyrios Kyrtzidis                    SourceLocation atStartLoc)
4411711fc91efb36d131f7ba771f73f0154dc1abd1fArgyrios Kyrtzidis    : NamedDecl(DK, DC, nameLoc, Id), DeclContext(DK), AtStart(atStartLoc) {}
442e4f039e01e797a38bc97bf22aff9832ecd18ff5fSteve Naroff
44393983f8fa120330bf212bfde7e65da2709fb3be8Steve Naroff  // Iterator access to properties.
44493983f8fa120330bf212bfde7e65da2709fb3be8Steve Naroff  typedef specific_decl_iterator<ObjCPropertyDecl> prop_iterator;
4451eb4433ac451dc16f4133a88af2d002ac26c58efMike Stump  prop_iterator prop_begin() const {
44617945a0f64fe03ff6ec0c2146005a87636e3ac12Argyrios Kyrtzidis    return prop_iterator(decls_begin());
44793983f8fa120330bf212bfde7e65da2709fb3be8Steve Naroff  }
4481eb4433ac451dc16f4133a88af2d002ac26c58efMike Stump  prop_iterator prop_end() const {
44917945a0f64fe03ff6ec0c2146005a87636e3ac12Argyrios Kyrtzidis    return prop_iterator(decls_end());
45009c4719788a5cea09897525e528fa00420f1677bSteve Naroff  }
4511eb4433ac451dc16f4133a88af2d002ac26c58efMike Stump
4520701bbb228dfd87e1fe82a0a4b7b9facfecb43daSteve Naroff  // Iterator access to instance/class methods.
453f8d49f64ef6ab7e632717a31631fc289aab69428Douglas Gregor  typedef specific_decl_iterator<ObjCMethodDecl> method_iterator;
4541eb4433ac451dc16f4133a88af2d002ac26c58efMike Stump  method_iterator meth_begin() const {
45517945a0f64fe03ff6ec0c2146005a87636e3ac12Argyrios Kyrtzidis    return method_iterator(decls_begin());
4567ed4faca5162b3ab85be7f7e57aa40e6ec170971Ted Kremenek  }
4571eb4433ac451dc16f4133a88af2d002ac26c58efMike Stump  method_iterator meth_end() const {
45817945a0f64fe03ff6ec0c2146005a87636e3ac12Argyrios Kyrtzidis    return method_iterator(decls_end());
4597ed4faca5162b3ab85be7f7e57aa40e6ec170971Ted Kremenek  }
4600701bbb228dfd87e1fe82a0a4b7b9facfecb43daSteve Naroff
4611eb4433ac451dc16f4133a88af2d002ac26c58efMike Stump  typedef filtered_decl_iterator<ObjCMethodDecl,
4621eb4433ac451dc16f4133a88af2d002ac26c58efMike Stump                                 &ObjCMethodDecl::isInstanceMethod>
463669c9a28fa4be35e6b6322aa7f2f3b2968189b80Douglas Gregor    instmeth_iterator;
46417945a0f64fe03ff6ec0c2146005a87636e3ac12Argyrios Kyrtzidis  instmeth_iterator instmeth_begin() const {
46517945a0f64fe03ff6ec0c2146005a87636e3ac12Argyrios Kyrtzidis    return instmeth_iterator(decls_begin());
466e4f039e01e797a38bc97bf22aff9832ecd18ff5fSteve Naroff  }
46717945a0f64fe03ff6ec0c2146005a87636e3ac12Argyrios Kyrtzidis  instmeth_iterator instmeth_end() const {
46817945a0f64fe03ff6ec0c2146005a87636e3ac12Argyrios Kyrtzidis    return instmeth_iterator(decls_end());
469e4f039e01e797a38bc97bf22aff9832ecd18ff5fSteve Naroff  }
470e4f039e01e797a38bc97bf22aff9832ecd18ff5fSteve Naroff
4711eb4433ac451dc16f4133a88af2d002ac26c58efMike Stump  typedef filtered_decl_iterator<ObjCMethodDecl,
4721eb4433ac451dc16f4133a88af2d002ac26c58efMike Stump                                 &ObjCMethodDecl::isClassMethod>
473669c9a28fa4be35e6b6322aa7f2f3b2968189b80Douglas Gregor    classmeth_iterator;
47417945a0f64fe03ff6ec0c2146005a87636e3ac12Argyrios Kyrtzidis  classmeth_iterator classmeth_begin() const {
47517945a0f64fe03ff6ec0c2146005a87636e3ac12Argyrios Kyrtzidis    return classmeth_iterator(decls_begin());
476e4f039e01e797a38bc97bf22aff9832ecd18ff5fSteve Naroff  }
47717945a0f64fe03ff6ec0c2146005a87636e3ac12Argyrios Kyrtzidis  classmeth_iterator classmeth_end() const {
47817945a0f64fe03ff6ec0c2146005a87636e3ac12Argyrios Kyrtzidis    return classmeth_iterator(decls_end());
4790701bbb228dfd87e1fe82a0a4b7b9facfecb43daSteve Naroff  }
4800701bbb228dfd87e1fe82a0a4b7b9facfecb43daSteve Naroff
4810701bbb228dfd87e1fe82a0a4b7b9facfecb43daSteve Naroff  // Get the local instance/class method declared in this interface.
482467c0b165072689ef87fe8d9cd47a5b63485bcdcArgyrios Kyrtzidis  ObjCMethodDecl *getMethod(Selector Sel, bool isInstance) const;
483467c0b165072689ef87fe8d9cd47a5b63485bcdcArgyrios Kyrtzidis  ObjCMethodDecl *getInstanceMethod(Selector Sel) const {
484467c0b165072689ef87fe8d9cd47a5b63485bcdcArgyrios Kyrtzidis    return getMethod(Sel, true/*isInstance*/);
48553df12d1ba68dbd071d067f8236c16fba815aad5Chris Lattner  }
486467c0b165072689ef87fe8d9cd47a5b63485bcdcArgyrios Kyrtzidis  ObjCMethodDecl *getClassMethod(Selector Sel) const {
487467c0b165072689ef87fe8d9cd47a5b63485bcdcArgyrios Kyrtzidis    return getMethod(Sel, false/*isInstance*/);
488467c0b165072689ef87fe8d9cd47a5b63485bcdcArgyrios Kyrtzidis  }
489467c0b165072689ef87fe8d9cd47a5b63485bcdcArgyrios Kyrtzidis  ObjCIvarDecl *getIvarDecl(IdentifierInfo *Id) const;
4901eb4433ac451dc16f4133a88af2d002ac26c58efMike Stump
49117945a0f64fe03ff6ec0c2146005a87636e3ac12Argyrios Kyrtzidis  ObjCPropertyDecl *FindPropertyDeclaration(IdentifierInfo *PropertyId) const;
49293983f8fa120330bf212bfde7e65da2709fb3be8Steve Naroff
4931711fc91efb36d131f7ba771f73f0154dc1abd1fArgyrios Kyrtzidis  SourceLocation getAtStartLoc() const { return AtStart; }
4941711fc91efb36d131f7ba771f73f0154dc1abd1fArgyrios Kyrtzidis  void setAtStartLoc(SourceLocation Loc) { AtStart = Loc; }
4951711fc91efb36d131f7ba771f73f0154dc1abd1fArgyrios Kyrtzidis
496e4f039e01e797a38bc97bf22aff9832ecd18ff5fSteve Naroff  // Marks the end of the container.
497782f2f52b78d8ca785110398a7f7b56b830b9ac7Ted Kremenek  SourceRange getAtEndRange() const {
498782f2f52b78d8ca785110398a7f7b56b830b9ac7Ted Kremenek    return AtEnd;
499782f2f52b78d8ca785110398a7f7b56b830b9ac7Ted Kremenek  }
500782f2f52b78d8ca785110398a7f7b56b830b9ac7Ted Kremenek  void setAtEndRange(SourceRange atEnd) {
501782f2f52b78d8ca785110398a7f7b56b830b9ac7Ted Kremenek    AtEnd = atEnd;
502782f2f52b78d8ca785110398a7f7b56b830b9ac7Ted Kremenek  }
503ddfd4c9eda34765b08fae4cb31ad5a365face107Argyrios Kyrtzidis
504da2142f2e2b3a02ee6eb5de9f9e6ed6f7eb5a0c0Douglas Gregor  virtual SourceRange getSourceRange() const {
5051711fc91efb36d131f7ba771f73f0154dc1abd1fArgyrios Kyrtzidis    return SourceRange(AtStart, getAtEndRange().getEnd());
506ddfd4c9eda34765b08fae4cb31ad5a365face107Argyrios Kyrtzidis  }
5071eb4433ac451dc16f4133a88af2d002ac26c58efMike Stump
50809c4719788a5cea09897525e528fa00420f1677bSteve Naroff  // Implement isa/cast/dyncast/etc.
50980cd64a8450d8e2c079dc134d9711cd45ba89d63John McCall  static bool classof(const Decl *D) { return classofKind(D->getKind()); }
51009c4719788a5cea09897525e528fa00420f1677bSteve Naroff  static bool classof(const ObjCContainerDecl *D) { return true; }
51180cd64a8450d8e2c079dc134d9711cd45ba89d63John McCall  static bool classofKind(Kind K) {
5129a55591af3e5506b95a9718e15380129fbfc5ebcSean Hunt    return K >= firstObjCContainer &&
5139a55591af3e5506b95a9718e15380129fbfc5ebcSean Hunt           K <= lastObjCContainer;
51480cd64a8450d8e2c079dc134d9711cd45ba89d63John McCall  }
51509c4719788a5cea09897525e528fa00420f1677bSteve Naroff
51609c4719788a5cea09897525e528fa00420f1677bSteve Naroff  static DeclContext *castToDeclContext(const ObjCContainerDecl *D) {
51709c4719788a5cea09897525e528fa00420f1677bSteve Naroff    return static_cast<DeclContext *>(const_cast<ObjCContainerDecl*>(D));
51809c4719788a5cea09897525e528fa00420f1677bSteve Naroff  }
51909c4719788a5cea09897525e528fa00420f1677bSteve Naroff  static ObjCContainerDecl *castFromDeclContext(const DeclContext *DC) {
52009c4719788a5cea09897525e528fa00420f1677bSteve Naroff    return static_cast<ObjCContainerDecl *>(const_cast<DeclContext*>(DC));
52109c4719788a5cea09897525e528fa00420f1677bSteve Naroff  }
522e4f039e01e797a38bc97bf22aff9832ecd18ff5fSteve Naroff};
523e4f039e01e797a38bc97bf22aff9832ecd18ff5fSteve Naroff
524a526c5c67e5a0473c340903ee542ce570119665fTed Kremenek/// ObjCInterfaceDecl - Represents an ObjC class declaration. For example:
5250c6b6243d3efd958c17943130e2a773653511edcSteve Naroff///
5260c6b6243d3efd958c17943130e2a773653511edcSteve Naroff///   // MostPrimitive declares no super class (not particularly useful).
5271eb4433ac451dc16f4133a88af2d002ac26c58efMike Stump///   @interface MostPrimitive
5280c6b6243d3efd958c17943130e2a773653511edcSteve Naroff///     // no instance variables or methods.
5290c6b6243d3efd958c17943130e2a773653511edcSteve Naroff///   @end
5300c6b6243d3efd958c17943130e2a773653511edcSteve Naroff///
5311eb4433ac451dc16f4133a88af2d002ac26c58efMike Stump///   // NSResponder inherits from NSObject & implements NSCoding (a protocol).
5320c6b6243d3efd958c17943130e2a773653511edcSteve Naroff///   @interface NSResponder : NSObject <NSCoding>
533a526c5c67e5a0473c340903ee542ce570119665fTed Kremenek///   { // instance variables are represented by ObjCIvarDecl.
5340c6b6243d3efd958c17943130e2a773653511edcSteve Naroff///     id nextResponder; // nextResponder instance variable.
5350c6b6243d3efd958c17943130e2a773653511edcSteve Naroff///   }
5360c6b6243d3efd958c17943130e2a773653511edcSteve Naroff///   - (NSResponder *)nextResponder; // return a pointer to NSResponder.
5370c6b6243d3efd958c17943130e2a773653511edcSteve Naroff///   - (void)mouseMoved:(NSEvent *)theEvent; // return void, takes a pointer
5380c6b6243d3efd958c17943130e2a773653511edcSteve Naroff///   @end                                    // to an NSEvent.
5390c6b6243d3efd958c17943130e2a773653511edcSteve Naroff///
5400c6b6243d3efd958c17943130e2a773653511edcSteve Naroff///   Unlike C/C++, forward class declarations are accomplished with @class.
5410c6b6243d3efd958c17943130e2a773653511edcSteve Naroff///   Unlike C/C++, @class allows for a list of classes to be forward declared.
5420c6b6243d3efd958c17943130e2a773653511edcSteve Naroff///   Unlike C++, ObjC is a single-rooted class model. In Cocoa, classes
5430c6b6243d3efd958c17943130e2a773653511edcSteve Naroff///   typically inherit from NSObject (an exception is NSProxy).
5440c6b6243d3efd958c17943130e2a773653511edcSteve Naroff///
54553df7a1d34f21d8f2309311d1067d463e9064c60Douglas Gregorclass ObjCInterfaceDecl : public ObjCContainerDecl
54653df7a1d34f21d8f2309311d1067d463e9064c60Douglas Gregor                        , public Redeclarable<ObjCInterfaceDecl> {
54799ba9e3bd70671f3441fb974895f226a83ce0e66David Blaikie  virtual void anchor();
54899ba9e3bd70671f3441fb974895f226a83ce0e66David Blaikie
5493110251f13981689f384eb3c0aba2afffea18d9dSteve Naroff  /// TypeForDecl - This indicates the Type object that represents this
5503110251f13981689f384eb3c0aba2afffea18d9dSteve Naroff  /// TypeDecl.  It is a cache maintained by ASTContext::getObjCInterfaceType
551f4c7371fb1d3cebcfb40abad4537bb82515704eaJohn McCall  mutable const Type *TypeForDecl;
5523110251f13981689f384eb3c0aba2afffea18d9dSteve Naroff  friend class ASTContext;
5532e5c15be82f362611c5928ce853d0685ff98c766Douglas Gregor
5542e5c15be82f362611c5928ce853d0685ff98c766Douglas Gregor  struct DefinitionData {
55526fec63b14565e9e2d8c9935b276b99be950444aDouglas Gregor    /// \brief The definition of this class, for quick access from any
55626fec63b14565e9e2d8c9935b276b99be950444aDouglas Gregor    /// declaration.
55726fec63b14565e9e2d8c9935b276b99be950444aDouglas Gregor    ObjCInterfaceDecl *Definition;
55826fec63b14565e9e2d8c9935b276b99be950444aDouglas Gregor
5592e5c15be82f362611c5928ce853d0685ff98c766Douglas Gregor    /// Class's super class.
5602e5c15be82f362611c5928ce853d0685ff98c766Douglas Gregor    ObjCInterfaceDecl *SuperClass;
5611eb4433ac451dc16f4133a88af2d002ac26c58efMike Stump
5622e5c15be82f362611c5928ce853d0685ff98c766Douglas Gregor    /// Protocols referenced in the @interface  declaration
5632e5c15be82f362611c5928ce853d0685ff98c766Douglas Gregor    ObjCProtocolList ReferencedProtocols;
5641eb4433ac451dc16f4133a88af2d002ac26c58efMike Stump
5652e5c15be82f362611c5928ce853d0685ff98c766Douglas Gregor    /// Protocols reference in both the @interface and class extensions.
5662e5c15be82f362611c5928ce853d0685ff98c766Douglas Gregor    ObjCList<ObjCProtocolDecl> AllReferencedProtocols;
567ba243b59a1074e0962f6abfa3bb9aa984eac1245David Blaikie
5682e5c15be82f362611c5928ce853d0685ff98c766Douglas Gregor    /// \brief List of categories and class extensions defined for this class.
5692e5c15be82f362611c5928ce853d0685ff98c766Douglas Gregor    ///
5702e5c15be82f362611c5928ce853d0685ff98c766Douglas Gregor    /// Categories are stored as a linked list in the AST, since the categories
5712e5c15be82f362611c5928ce853d0685ff98c766Douglas Gregor    /// and class extensions come long after the initial interface declaration,
5722e5c15be82f362611c5928ce853d0685ff98c766Douglas Gregor    /// and we avoid dynamically-resized arrays in the AST wherever possible.
5732e5c15be82f362611c5928ce853d0685ff98c766Douglas Gregor    ObjCCategoryDecl *CategoryList;
5741eb4433ac451dc16f4133a88af2d002ac26c58efMike Stump
5752e5c15be82f362611c5928ce853d0685ff98c766Douglas Gregor    /// IvarList - List of all ivars defined by this class; including class
5762e5c15be82f362611c5928ce853d0685ff98c766Douglas Gregor    /// extensions and implementation. This list is built lazily.
5772e5c15be82f362611c5928ce853d0685ff98c766Douglas Gregor    ObjCIvarDecl *IvarList;
578ba243b59a1074e0962f6abfa3bb9aa984eac1245David Blaikie
5792e5c15be82f362611c5928ce853d0685ff98c766Douglas Gregor    /// \brief Indicates that the contents of this Objective-C class will be
5802e5c15be82f362611c5928ce853d0685ff98c766Douglas Gregor    /// completed by the external AST source when required.
5812e5c15be82f362611c5928ce853d0685ff98c766Douglas Gregor    mutable bool ExternallyCompleted : 1;
5822e5c15be82f362611c5928ce853d0685ff98c766Douglas Gregor
58305c272fed899b8d3142cf080e86a235fc6168862Douglas Gregor    /// \brief The location of the superclass, if any.
58405c272fed899b8d3142cf080e86a235fc6168862Douglas Gregor    SourceLocation SuperClassLoc;
585161794732195881c33305f701f6e58721998541fDouglas Gregor
58605c272fed899b8d3142cf080e86a235fc6168862Douglas Gregor    /// \brief The location of the last location in this declaration, before
58705c272fed899b8d3142cf080e86a235fc6168862Douglas Gregor    /// the properties/methods. For example, this will be the '>', '}', or
58805c272fed899b8d3142cf080e86a235fc6168862Douglas Gregor    /// identifier,
58905c272fed899b8d3142cf080e86a235fc6168862Douglas Gregor    SourceLocation EndLoc;
59005c272fed899b8d3142cf080e86a235fc6168862Douglas Gregor
59126fec63b14565e9e2d8c9935b276b99be950444aDouglas Gregor    DefinitionData() : Definition(), SuperClass(), CategoryList(), IvarList(),
592161794732195881c33305f701f6e58721998541fDouglas Gregor                       ExternallyCompleted() { }
5932e5c15be82f362611c5928ce853d0685ff98c766Douglas Gregor  };
5942e5c15be82f362611c5928ce853d0685ff98c766Douglas Gregor
5952e5c15be82f362611c5928ce853d0685ff98c766Douglas Gregor  ObjCInterfaceDecl(DeclContext *DC, SourceLocation atLoc, IdentifierInfo *Id,
59637f953f021c67e3b97f1ef38e1ef3cb08bd9d146Douglas Gregor                    SourceLocation CLoc, ObjCInterfaceDecl *PrevDecl,
59737f953f021c67e3b97f1ef38e1ef3cb08bd9d146Douglas Gregor                    bool isInternal);
5982e5c15be82f362611c5928ce853d0685ff98c766Douglas Gregor
5992e5c15be82f362611c5928ce853d0685ff98c766Douglas Gregor  void LoadExternalDefinition() const;
6002e5c15be82f362611c5928ce853d0685ff98c766Douglas Gregor
6012e5c15be82f362611c5928ce853d0685ff98c766Douglas Gregor  /// \brief Contains a pointer to the data associated with this class,
6022e5c15be82f362611c5928ce853d0685ff98c766Douglas Gregor  /// which will be NULL if this class has not yet been defined.
60326fec63b14565e9e2d8c9935b276b99be950444aDouglas Gregor  DefinitionData *Data;
6042e5c15be82f362611c5928ce853d0685ff98c766Douglas Gregor
6052e5c15be82f362611c5928ce853d0685ff98c766Douglas Gregor  DefinitionData &data() const {
60626fec63b14565e9e2d8c9935b276b99be950444aDouglas Gregor    assert(Data != 0 && "Declaration has no definition!");
60726fec63b14565e9e2d8c9935b276b99be950444aDouglas Gregor    return *Data;
6082e5c15be82f362611c5928ce853d0685ff98c766Douglas Gregor  }
6091eb4433ac451dc16f4133a88af2d002ac26c58efMike Stump
6102e5c15be82f362611c5928ce853d0685ff98c766Douglas Gregor  /// \brief Allocate the definition data for this class.
6112e5c15be82f362611c5928ce853d0685ff98c766Douglas Gregor  void allocateDefinitionData();
6122e5c15be82f362611c5928ce853d0685ff98c766Douglas Gregor
61353df7a1d34f21d8f2309311d1067d463e9064c60Douglas Gregor  typedef Redeclarable<ObjCInterfaceDecl> redeclarable_base;
61453df7a1d34f21d8f2309311d1067d463e9064c60Douglas Gregor  virtual ObjCInterfaceDecl *getNextRedeclaration() {
61553df7a1d34f21d8f2309311d1067d463e9064c60Douglas Gregor    return RedeclLink.getNext();
61653df7a1d34f21d8f2309311d1067d463e9064c60Douglas Gregor  }
61753df7a1d34f21d8f2309311d1067d463e9064c60Douglas Gregor
6180e77ba0bf769e2e5a4a93c079f241b02aeb3ef93Chris Lattnerpublic:
619d04341000d35c8808a72838b057eed7bf13b7661Douglas Gregor  static ObjCInterfaceDecl *Create(ASTContext &C, DeclContext *DC,
6200ed844b04ea4387caa4e1cf3dc375d269657536bChris Lattner                                   SourceLocation atLoc,
6211eb4433ac451dc16f4133a88af2d002ac26c58efMike Stump                                   IdentifierInfo *Id,
6220af550115df1f57f17a4f125ff0e8b34820c65d1Douglas Gregor                                   ObjCInterfaceDecl *PrevDecl,
623deacbdca554298ccdf636f19c6094a8825ec6b34Douglas Gregor                                   SourceLocation ClassLoc = SourceLocation(),
6240e77ba0bf769e2e5a4a93c079f241b02aeb3ef93Chris Lattner                                   bool isInternal = false);
625d1cf3ff6c7e34fce764293cd2900fce99a60ed69Argyrios Kyrtzidis
6260af550115df1f57f17a4f125ff0e8b34820c65d1Douglas Gregor  static ObjCInterfaceDecl *CreateEmpty(ASTContext &C);
6270af550115df1f57f17a4f125ff0e8b34820c65d1Douglas Gregor
628d1cf3ff6c7e34fce764293cd2900fce99a60ed69Argyrios Kyrtzidis  virtual SourceRange getSourceRange() const {
6297723fec9b45b7258c0eddf4cbfd0d335348f5edcDouglas Gregor    if (isThisDeclarationADefinition())
6307723fec9b45b7258c0eddf4cbfd0d335348f5edcDouglas Gregor      return ObjCContainerDecl::getSourceRange();
6317723fec9b45b7258c0eddf4cbfd0d335348f5edcDouglas Gregor
6327723fec9b45b7258c0eddf4cbfd0d335348f5edcDouglas Gregor    return SourceRange(getAtStartLoc(), getLocation());
633d1cf3ff6c7e34fce764293cd2900fce99a60ed69Argyrios Kyrtzidis  }
634ba243b59a1074e0962f6abfa3bb9aa984eac1245David Blaikie
63526ac3f30ecef21749c00a4b1a08dd15d772dd5aaDouglas Gregor  /// \brief Indicate that this Objective-C class is complete, but that
63626ac3f30ecef21749c00a4b1a08dd15d772dd5aaDouglas Gregor  /// the external AST source will be responsible for filling in its contents
63726ac3f30ecef21749c00a4b1a08dd15d772dd5aaDouglas Gregor  /// when a complete class is required.
63826ac3f30ecef21749c00a4b1a08dd15d772dd5aaDouglas Gregor  void setExternallyCompleted();
639ba243b59a1074e0962f6abfa3bb9aa984eac1245David Blaikie
64018df52bbb5d28ca082064d31ae7558dbdae52377Douglas Gregor  const ObjCProtocolList &getReferencedProtocols() const {
6412e5c15be82f362611c5928ce853d0685ff98c766Douglas Gregor    if (data().ExternallyCompleted)
64226ac3f30ecef21749c00a4b1a08dd15d772dd5aaDouglas Gregor      LoadExternalDefinition();
643ba243b59a1074e0962f6abfa3bb9aa984eac1245David Blaikie
6442e5c15be82f362611c5928ce853d0685ff98c766Douglas Gregor    return data().ReferencedProtocols;
6457ed9e0f97f4645edc5d4670385b985ea4c617ce7Fariborz Jahanian  }
6468a1d722f13df383600f36d77f842957c8adb5f1bArgyrios Kyrtzidis
6478a1d722f13df383600f36d77f842957c8adb5f1bArgyrios Kyrtzidis  ObjCImplementationDecl *getImplementation() const;
6488a1d722f13df383600f36d77f842957c8adb5f1bArgyrios Kyrtzidis  void setImplementation(ObjCImplementationDecl *ImplD);
6498a1d722f13df383600f36d77f842957c8adb5f1bArgyrios Kyrtzidis
650559c0c4bbecc017aab0716d546c4fefbcc194687Fariborz Jahanian  ObjCCategoryDecl *FindCategoryDeclaration(IdentifierInfo *CategoryId) const;
6511eb4433ac451dc16f4133a88af2d002ac26c58efMike Stump
6521cb35dd4840d21cec58648361180d5688446a9caArgyrios Kyrtzidis  // Get the local instance/class method declared in a category.
6531cb35dd4840d21cec58648361180d5688446a9caArgyrios Kyrtzidis  ObjCMethodDecl *getCategoryInstanceMethod(Selector Sel) const;
6541cb35dd4840d21cec58648361180d5688446a9caArgyrios Kyrtzidis  ObjCMethodDecl *getCategoryClassMethod(Selector Sel) const;
6551cb35dd4840d21cec58648361180d5688446a9caArgyrios Kyrtzidis  ObjCMethodDecl *getCategoryMethod(Selector Sel, bool isInstance) const {
6561eb4433ac451dc16f4133a88af2d002ac26c58efMike Stump    return isInstance ? getInstanceMethod(Sel)
6571cb35dd4840d21cec58648361180d5688446a9caArgyrios Kyrtzidis                      : getClassMethod(Sel);
6581cb35dd4840d21cec58648361180d5688446a9caArgyrios Kyrtzidis  }
6593db6cae19c236fe2cef343c90105ce7cca7de965Chris Lattner
66018df52bbb5d28ca082064d31ae7558dbdae52377Douglas Gregor  typedef ObjCProtocolList::iterator protocol_iterator;
661ba243b59a1074e0962f6abfa3bb9aa984eac1245David Blaikie
66253b9441b5a81a24fa1f66f3f6416f1e36baa9c2fTed Kremenek  protocol_iterator protocol_begin() const {
6632e5c15be82f362611c5928ce853d0685ff98c766Douglas Gregor    // FIXME: Should make sure no callers ever do this.
6642e5c15be82f362611c5928ce853d0685ff98c766Douglas Gregor    if (!hasDefinition())
6652e5c15be82f362611c5928ce853d0685ff98c766Douglas Gregor      return protocol_iterator();
6662e5c15be82f362611c5928ce853d0685ff98c766Douglas Gregor
6672e5c15be82f362611c5928ce853d0685ff98c766Douglas Gregor    if (data().ExternallyCompleted)
66826ac3f30ecef21749c00a4b1a08dd15d772dd5aaDouglas Gregor      LoadExternalDefinition();
66926ac3f30ecef21749c00a4b1a08dd15d772dd5aaDouglas Gregor
6702e5c15be82f362611c5928ce853d0685ff98c766Douglas Gregor    return data().ReferencedProtocols.begin();
67153b9441b5a81a24fa1f66f3f6416f1e36baa9c2fTed Kremenek  }
67253b9441b5a81a24fa1f66f3f6416f1e36baa9c2fTed Kremenek  protocol_iterator protocol_end() const {
6732e5c15be82f362611c5928ce853d0685ff98c766Douglas Gregor    // FIXME: Should make sure no callers ever do this.
6742e5c15be82f362611c5928ce853d0685ff98c766Douglas Gregor    if (!hasDefinition())
6752e5c15be82f362611c5928ce853d0685ff98c766Douglas Gregor      return protocol_iterator();
6762e5c15be82f362611c5928ce853d0685ff98c766Douglas Gregor
6772e5c15be82f362611c5928ce853d0685ff98c766Douglas Gregor    if (data().ExternallyCompleted)
67826ac3f30ecef21749c00a4b1a08dd15d772dd5aaDouglas Gregor      LoadExternalDefinition();
67926ac3f30ecef21749c00a4b1a08dd15d772dd5aaDouglas Gregor
6802e5c15be82f362611c5928ce853d0685ff98c766Douglas Gregor    return data().ReferencedProtocols.end();
68153b9441b5a81a24fa1f66f3f6416f1e36baa9c2fTed Kremenek  }
68253b9441b5a81a24fa1f66f3f6416f1e36baa9c2fTed Kremenek
68318df52bbb5d28ca082064d31ae7558dbdae52377Douglas Gregor  typedef ObjCProtocolList::loc_iterator protocol_loc_iterator;
68453b9441b5a81a24fa1f66f3f6416f1e36baa9c2fTed Kremenek
685ba243b59a1074e0962f6abfa3bb9aa984eac1245David Blaikie  protocol_loc_iterator protocol_loc_begin() const {
6862e5c15be82f362611c5928ce853d0685ff98c766Douglas Gregor    // FIXME: Should make sure no callers ever do this.
6872e5c15be82f362611c5928ce853d0685ff98c766Douglas Gregor    if (!hasDefinition())
6882e5c15be82f362611c5928ce853d0685ff98c766Douglas Gregor      return protocol_loc_iterator();
6892e5c15be82f362611c5928ce853d0685ff98c766Douglas Gregor
6902e5c15be82f362611c5928ce853d0685ff98c766Douglas Gregor    if (data().ExternallyCompleted)
69126ac3f30ecef21749c00a4b1a08dd15d772dd5aaDouglas Gregor      LoadExternalDefinition();
69226ac3f30ecef21749c00a4b1a08dd15d772dd5aaDouglas Gregor
6932e5c15be82f362611c5928ce853d0685ff98c766Douglas Gregor    return data().ReferencedProtocols.loc_begin();
69418df52bbb5d28ca082064d31ae7558dbdae52377Douglas Gregor  }
69553b9441b5a81a24fa1f66f3f6416f1e36baa9c2fTed Kremenek
696ba243b59a1074e0962f6abfa3bb9aa984eac1245David Blaikie  protocol_loc_iterator protocol_loc_end() const {
6972e5c15be82f362611c5928ce853d0685ff98c766Douglas Gregor    // FIXME: Should make sure no callers ever do this.
6982e5c15be82f362611c5928ce853d0685ff98c766Douglas Gregor    if (!hasDefinition())
6992e5c15be82f362611c5928ce853d0685ff98c766Douglas Gregor      return protocol_loc_iterator();
7002e5c15be82f362611c5928ce853d0685ff98c766Douglas Gregor
7012e5c15be82f362611c5928ce853d0685ff98c766Douglas Gregor    if (data().ExternallyCompleted)
70226ac3f30ecef21749c00a4b1a08dd15d772dd5aaDouglas Gregor      LoadExternalDefinition();
70326ac3f30ecef21749c00a4b1a08dd15d772dd5aaDouglas Gregor
7042e5c15be82f362611c5928ce853d0685ff98c766Douglas Gregor    return data().ReferencedProtocols.loc_end();
70518df52bbb5d28ca082064d31ae7558dbdae52377Douglas Gregor  }
706ba243b59a1074e0962f6abfa3bb9aa984eac1245David Blaikie
70753b9441b5a81a24fa1f66f3f6416f1e36baa9c2fTed Kremenek  typedef ObjCList<ObjCProtocolDecl>::iterator all_protocol_iterator;
708ba243b59a1074e0962f6abfa3bb9aa984eac1245David Blaikie
70953b9441b5a81a24fa1f66f3f6416f1e36baa9c2fTed Kremenek  all_protocol_iterator all_referenced_protocol_begin() const {
7102e5c15be82f362611c5928ce853d0685ff98c766Douglas Gregor    // FIXME: Should make sure no callers ever do this.
7112e5c15be82f362611c5928ce853d0685ff98c766Douglas Gregor    if (!hasDefinition())
7122e5c15be82f362611c5928ce853d0685ff98c766Douglas Gregor      return all_protocol_iterator();
7132e5c15be82f362611c5928ce853d0685ff98c766Douglas Gregor
7142e5c15be82f362611c5928ce853d0685ff98c766Douglas Gregor    if (data().ExternallyCompleted)
71526ac3f30ecef21749c00a4b1a08dd15d772dd5aaDouglas Gregor      LoadExternalDefinition();
71626ac3f30ecef21749c00a4b1a08dd15d772dd5aaDouglas Gregor
7172e5c15be82f362611c5928ce853d0685ff98c766Douglas Gregor    return data().AllReferencedProtocols.empty()
7182e5c15be82f362611c5928ce853d0685ff98c766Douglas Gregor             ? protocol_begin()
7192e5c15be82f362611c5928ce853d0685ff98c766Douglas Gregor             : data().AllReferencedProtocols.begin();
72053b9441b5a81a24fa1f66f3f6416f1e36baa9c2fTed Kremenek  }
72153b9441b5a81a24fa1f66f3f6416f1e36baa9c2fTed Kremenek  all_protocol_iterator all_referenced_protocol_end() const {
7222e5c15be82f362611c5928ce853d0685ff98c766Douglas Gregor    // FIXME: Should make sure no callers ever do this.
7232e5c15be82f362611c5928ce853d0685ff98c766Douglas Gregor    if (!hasDefinition())
7242e5c15be82f362611c5928ce853d0685ff98c766Douglas Gregor      return all_protocol_iterator();
7252e5c15be82f362611c5928ce853d0685ff98c766Douglas Gregor
7262e5c15be82f362611c5928ce853d0685ff98c766Douglas Gregor    if (data().ExternallyCompleted)
72726ac3f30ecef21749c00a4b1a08dd15d772dd5aaDouglas Gregor      LoadExternalDefinition();
72826ac3f30ecef21749c00a4b1a08dd15d772dd5aaDouglas Gregor
7292e5c15be82f362611c5928ce853d0685ff98c766Douglas Gregor    return data().AllReferencedProtocols.empty()
7302e5c15be82f362611c5928ce853d0685ff98c766Douglas Gregor             ? protocol_end()
7312e5c15be82f362611c5928ce853d0685ff98c766Douglas Gregor             : data().AllReferencedProtocols.end();
73253b9441b5a81a24fa1f66f3f6416f1e36baa9c2fTed Kremenek  }
733291be393aa33759e6e34b6429c5ffa206ba50115Douglas Gregor
73411062e11236b7bc689dad150e8b490fd6b063ec3Fariborz Jahanian  typedef specific_decl_iterator<ObjCIvarDecl> ivar_iterator;
73553b9441b5a81a24fa1f66f3f6416f1e36baa9c2fTed Kremenek
7362e5c15be82f362611c5928ce853d0685ff98c766Douglas Gregor  ivar_iterator ivar_begin() const {
7372e5c15be82f362611c5928ce853d0685ff98c766Douglas Gregor    if (const ObjCInterfaceDecl *Def = getDefinition())
7382e5c15be82f362611c5928ce853d0685ff98c766Douglas Gregor      return ivar_iterator(Def->decls_begin());
7392e5c15be82f362611c5928ce853d0685ff98c766Douglas Gregor
7402e5c15be82f362611c5928ce853d0685ff98c766Douglas Gregor    // FIXME: Should make sure no callers ever do this.
7412e5c15be82f362611c5928ce853d0685ff98c766Douglas Gregor    return ivar_iterator();
7422e5c15be82f362611c5928ce853d0685ff98c766Douglas Gregor  }
7432e5c15be82f362611c5928ce853d0685ff98c766Douglas Gregor  ivar_iterator ivar_end() const {
7442e5c15be82f362611c5928ce853d0685ff98c766Douglas Gregor    if (const ObjCInterfaceDecl *Def = getDefinition())
7452e5c15be82f362611c5928ce853d0685ff98c766Douglas Gregor      return ivar_iterator(Def->decls_end());
7462e5c15be82f362611c5928ce853d0685ff98c766Douglas Gregor
7472e5c15be82f362611c5928ce853d0685ff98c766Douglas Gregor    // FIXME: Should make sure no callers ever do this.
7482e5c15be82f362611c5928ce853d0685ff98c766Douglas Gregor    return ivar_iterator();
7492e5c15be82f362611c5928ce853d0685ff98c766Douglas Gregor  }
75053b9441b5a81a24fa1f66f3f6416f1e36baa9c2fTed Kremenek
75111062e11236b7bc689dad150e8b490fd6b063ec3Fariborz Jahanian  unsigned ivar_size() const {
75211062e11236b7bc689dad150e8b490fd6b063ec3Fariborz Jahanian    return std::distance(ivar_begin(), ivar_end());
75311062e11236b7bc689dad150e8b490fd6b063ec3Fariborz Jahanian  }
754ba243b59a1074e0962f6abfa3bb9aa984eac1245David Blaikie
75511062e11236b7bc689dad150e8b490fd6b063ec3Fariborz Jahanian  bool ivar_empty() const { return ivar_begin() == ivar_end(); }
756ba243b59a1074e0962f6abfa3bb9aa984eac1245David Blaikie
757db8264e4c5ffd7af6fbad4ca4306bd382bb02691Jordy Rose  ObjCIvarDecl *all_declared_ivar_begin();
758db8264e4c5ffd7af6fbad4ca4306bd382bb02691Jordy Rose  const ObjCIvarDecl *all_declared_ivar_begin() const {
759db8264e4c5ffd7af6fbad4ca4306bd382bb02691Jordy Rose    // Even though this modifies IvarList, it's conceptually const:
760db8264e4c5ffd7af6fbad4ca4306bd382bb02691Jordy Rose    // the ivar chain is essentially a cached property of ObjCInterfaceDecl.
761db8264e4c5ffd7af6fbad4ca4306bd382bb02691Jordy Rose    return const_cast<ObjCInterfaceDecl *>(this)->all_declared_ivar_begin();
762db8264e4c5ffd7af6fbad4ca4306bd382bb02691Jordy Rose  }
7632e5c15be82f362611c5928ce853d0685ff98c766Douglas Gregor  void setIvarList(ObjCIvarDecl *ivar) { data().IvarList = ivar; }
764ba243b59a1074e0962f6abfa3bb9aa984eac1245David Blaikie
76538af2deb27cdfa1a95bde96e30dab15dce53fcefChris Lattner  /// setProtocolList - Set the list of protocols that this interface
766b752f289026ad8e5f44851b20e009a27ed61eefcChris Lattner  /// implements.
76738af2deb27cdfa1a95bde96e30dab15dce53fcefChris Lattner  void setProtocolList(ObjCProtocolDecl *const* List, unsigned Num,
76818df52bbb5d28ca082064d31ae7558dbdae52377Douglas Gregor                       const SourceLocation *Locs, ASTContext &C) {
7692e5c15be82f362611c5928ce853d0685ff98c766Douglas Gregor    data().ReferencedProtocols.set(List, Num, Locs, C);
7703db6cae19c236fe2cef343c90105ce7cca7de965Chris Lattner  }
7711eb4433ac451dc16f4133a88af2d002ac26c58efMike Stump
772339798eae1eb61c50ca68766ed028c0a16d0a284Fariborz Jahanian  /// mergeClassExtensionProtocolList - Merge class extension's protocol list
773339798eae1eb61c50ca68766ed028c0a16d0a284Fariborz Jahanian  /// into the protocol list for this class.
774ba243b59a1074e0962f6abfa3bb9aa984eac1245David Blaikie  void mergeClassExtensionProtocolList(ObjCProtocolDecl *const* List,
77518df52bbb5d28ca082064d31ae7558dbdae52377Douglas Gregor                                       unsigned Num,
77618df52bbb5d28ca082064d31ae7558dbdae52377Douglas Gregor                                       ASTContext &C);
777339798eae1eb61c50ca68766ed028c0a16d0a284Fariborz Jahanian
77853df7a1d34f21d8f2309311d1067d463e9064c60Douglas Gregor  /// \brief Determine whether this particular declaration of this class is
77953df7a1d34f21d8f2309311d1067d463e9064c60Douglas Gregor  /// actually also a definition.
78026fec63b14565e9e2d8c9935b276b99be950444aDouglas Gregor  bool isThisDeclarationADefinition() const {
7817723fec9b45b7258c0eddf4cbfd0d335348f5edcDouglas Gregor    return Data && Data->Definition == this;
78226fec63b14565e9e2d8c9935b276b99be950444aDouglas Gregor  }
78353df7a1d34f21d8f2309311d1067d463e9064c60Douglas Gregor
7842e5c15be82f362611c5928ce853d0685ff98c766Douglas Gregor  /// \brief Determine whether this class has been defined.
78526fec63b14565e9e2d8c9935b276b99be950444aDouglas Gregor  bool hasDefinition() const { return Data; }
78653df7a1d34f21d8f2309311d1067d463e9064c60Douglas Gregor
7872e5c15be82f362611c5928ce853d0685ff98c766Douglas Gregor  /// \brief Retrieve the definition of this class, or NULL if this class
7882e5c15be82f362611c5928ce853d0685ff98c766Douglas Gregor  /// has been forward-declared (with @class) but not yet defined (with
7892e5c15be82f362611c5928ce853d0685ff98c766Douglas Gregor  /// @interface).
7902e5c15be82f362611c5928ce853d0685ff98c766Douglas Gregor  ObjCInterfaceDecl *getDefinition() {
79126fec63b14565e9e2d8c9935b276b99be950444aDouglas Gregor    return hasDefinition()? Data->Definition : 0;
7922e5c15be82f362611c5928ce853d0685ff98c766Douglas Gregor  }
793ad834d534e9a5db3d3baa09593775f83ceaff1f2Argyrios Kyrtzidis
7942e5c15be82f362611c5928ce853d0685ff98c766Douglas Gregor  /// \brief Retrieve the definition of this class, or NULL if this class
7952e5c15be82f362611c5928ce853d0685ff98c766Douglas Gregor  /// has been forward-declared (with @class) but not yet defined (with
7962e5c15be82f362611c5928ce853d0685ff98c766Douglas Gregor  /// @interface).
7972e5c15be82f362611c5928ce853d0685ff98c766Douglas Gregor  const ObjCInterfaceDecl *getDefinition() const {
79826fec63b14565e9e2d8c9935b276b99be950444aDouglas Gregor    return hasDefinition()? Data->Definition : 0;
7992e5c15be82f362611c5928ce853d0685ff98c766Douglas Gregor  }
8001eb4433ac451dc16f4133a88af2d002ac26c58efMike Stump
8012e5c15be82f362611c5928ce853d0685ff98c766Douglas Gregor  /// \brief Starts the definition of this Objective-C class, taking it from
8022e5c15be82f362611c5928ce853d0685ff98c766Douglas Gregor  /// a forward declaration (@class) to a definition (@interface).
8032e5c15be82f362611c5928ce853d0685ff98c766Douglas Gregor  void startDefinition();
8042e5c15be82f362611c5928ce853d0685ff98c766Douglas Gregor
805ba243b59a1074e0962f6abfa3bb9aa984eac1245David Blaikie  ObjCInterfaceDecl *getSuperClass() const {
8062e5c15be82f362611c5928ce853d0685ff98c766Douglas Gregor    // FIXME: Should make sure no callers ever do this.
8072e5c15be82f362611c5928ce853d0685ff98c766Douglas Gregor    if (!hasDefinition())
8082e5c15be82f362611c5928ce853d0685ff98c766Douglas Gregor      return 0;
8092e5c15be82f362611c5928ce853d0685ff98c766Douglas Gregor
8102e5c15be82f362611c5928ce853d0685ff98c766Douglas Gregor    if (data().ExternallyCompleted)
81126ac3f30ecef21749c00a4b1a08dd15d772dd5aaDouglas Gregor      LoadExternalDefinition();
81226ac3f30ecef21749c00a4b1a08dd15d772dd5aaDouglas Gregor
8132e5c15be82f362611c5928ce853d0685ff98c766Douglas Gregor    return data().SuperClass;
81426ac3f30ecef21749c00a4b1a08dd15d772dd5aaDouglas Gregor  }
815ba243b59a1074e0962f6abfa3bb9aa984eac1245David Blaikie
8162e5c15be82f362611c5928ce853d0685ff98c766Douglas Gregor  void setSuperClass(ObjCInterfaceDecl * superCls) {
817712ef874534ee1bef41d1aa4664ae36148ec8b12Fariborz Jahanian    data().SuperClass =
818712ef874534ee1bef41d1aa4664ae36148ec8b12Fariborz Jahanian      (superCls && superCls->hasDefinition()) ? superCls->getDefinition()
819712ef874534ee1bef41d1aa4664ae36148ec8b12Fariborz Jahanian                                              : superCls;
8202e5c15be82f362611c5928ce853d0685ff98c766Douglas Gregor  }
8211eb4433ac451dc16f4133a88af2d002ac26c58efMike Stump
822ba243b59a1074e0962f6abfa3bb9aa984eac1245David Blaikie  ObjCCategoryDecl* getCategoryList() const {
8232e5c15be82f362611c5928ce853d0685ff98c766Douglas Gregor    // FIXME: Should make sure no callers ever do this.
8242e5c15be82f362611c5928ce853d0685ff98c766Douglas Gregor    if (!hasDefinition())
8252e5c15be82f362611c5928ce853d0685ff98c766Douglas Gregor      return 0;
8262e5c15be82f362611c5928ce853d0685ff98c766Douglas Gregor
8272e5c15be82f362611c5928ce853d0685ff98c766Douglas Gregor    if (data().ExternallyCompleted)
82826ac3f30ecef21749c00a4b1a08dd15d772dd5aaDouglas Gregor      LoadExternalDefinition();
82926ac3f30ecef21749c00a4b1a08dd15d772dd5aaDouglas Gregor
8302e5c15be82f362611c5928ce853d0685ff98c766Douglas Gregor    return data().CategoryList;
83126ac3f30ecef21749c00a4b1a08dd15d772dd5aaDouglas Gregor  }
832ba243b59a1074e0962f6abfa3bb9aa984eac1245David Blaikie
8331eb4433ac451dc16f4133a88af2d002ac26c58efMike Stump  void setCategoryList(ObjCCategoryDecl *category) {
8342e5c15be82f362611c5928ce853d0685ff98c766Douglas Gregor    data().CategoryList = category;
835980e508ca70d6de75d2abfd96b4681fc98bb2698Steve Naroff  }
836ba243b59a1074e0962f6abfa3bb9aa984eac1245David Blaikie
83780aa1cd7973561889e51c1c152c8990a8de9c953Fariborz Jahanian  ObjCCategoryDecl* getFirstClassExtension() const;
83837cafb077ad5b170acae77e566638603011ef4c0Ted Kremenek
83937cafb077ad5b170acae77e566638603011ef4c0Ted Kremenek  ObjCPropertyDecl
84037cafb077ad5b170acae77e566638603011ef4c0Ted Kremenek    *FindPropertyVisibleInPrimaryClass(IdentifierInfo *PropertyId) const;
84137cafb077ad5b170acae77e566638603011ef4c0Ted Kremenek
84253efc251792bf2c9c5f295bd3507facc51a1fe7eChris Lattner  /// isSuperClassOf - Return true if this class is the specified class or is a
84353efc251792bf2c9c5f295bd3507facc51a1fe7eChris Lattner  /// super class of the specified interface class.
84453efc251792bf2c9c5f295bd3507facc51a1fe7eChris Lattner  bool isSuperClassOf(const ObjCInterfaceDecl *I) const {
84553efc251792bf2c9c5f295bd3507facc51a1fe7eChris Lattner    // If RHS is derived from LHS it is OK; else it is not OK.
84653efc251792bf2c9c5f295bd3507facc51a1fe7eChris Lattner    while (I != NULL) {
84760ef308e51c71b760d7f598c1b763ceb7b768148Douglas Gregor      if (declaresSameEntity(this, I))
84853efc251792bf2c9c5f295bd3507facc51a1fe7eChris Lattner        return true;
8492e5c15be82f362611c5928ce853d0685ff98c766Douglas Gregor
85053efc251792bf2c9c5f295bd3507facc51a1fe7eChris Lattner      I = I->getSuperClass();
85153efc251792bf2c9c5f295bd3507facc51a1fe7eChris Lattner    }
85253efc251792bf2c9c5f295bd3507facc51a1fe7eChris Lattner    return false;
85353efc251792bf2c9c5f295bd3507facc51a1fe7eChris Lattner  }
8541eb4433ac451dc16f4133a88af2d002ac26c58efMike Stump
8557263feeb367ab55af7e9a6fd701148b1b8264dbaFariborz Jahanian  /// isArcWeakrefUnavailable - Checks for a class or one of its super classes
8567263feeb367ab55af7e9a6fd701148b1b8264dbaFariborz Jahanian  /// to be incompatible with __weak references. Returns true if it is.
8577263feeb367ab55af7e9a6fd701148b1b8264dbaFariborz Jahanian  bool isArcWeakrefUnavailable() const {
8587263feeb367ab55af7e9a6fd701148b1b8264dbaFariborz Jahanian    const ObjCInterfaceDecl *Class = this;
8597263feeb367ab55af7e9a6fd701148b1b8264dbaFariborz Jahanian    while (Class) {
8607263feeb367ab55af7e9a6fd701148b1b8264dbaFariborz Jahanian      if (Class->hasAttr<ArcWeakrefUnavailableAttr>())
8617263feeb367ab55af7e9a6fd701148b1b8264dbaFariborz Jahanian        return true;
8627263feeb367ab55af7e9a6fd701148b1b8264dbaFariborz Jahanian      Class = Class->getSuperClass();
8637263feeb367ab55af7e9a6fd701148b1b8264dbaFariborz Jahanian   }
864ba243b59a1074e0962f6abfa3bb9aa984eac1245David Blaikie   return false;
8657263feeb367ab55af7e9a6fd701148b1b8264dbaFariborz Jahanian  }
8667263feeb367ab55af7e9a6fd701148b1b8264dbaFariborz Jahanian
86717945a0f64fe03ff6ec0c2146005a87636e3ac12Argyrios Kyrtzidis  ObjCIvarDecl *lookupInstanceVariable(IdentifierInfo *IVarName,
86868a057b4292f5ff814ec8da53f6cda8cdcfbd2aeChris Lattner                                       ObjCInterfaceDecl *&ClassDeclared);
86917945a0f64fe03ff6ec0c2146005a87636e3ac12Argyrios Kyrtzidis  ObjCIvarDecl *lookupInstanceVariable(IdentifierInfo *IVarName) {
87068a057b4292f5ff814ec8da53f6cda8cdcfbd2aeChris Lattner    ObjCInterfaceDecl *ClassDeclared;
87117945a0f64fe03ff6ec0c2146005a87636e3ac12Argyrios Kyrtzidis    return lookupInstanceVariable(IVarName, ClassDeclared);
87268a057b4292f5ff814ec8da53f6cda8cdcfbd2aeChris Lattner  }
87368a057b4292f5ff814ec8da53f6cda8cdcfbd2aeChris Lattner
87494a5c3334bba3cc8cd1da85ba1118bc2c080add9Steve Naroff  // Lookup a method. First, we search locally. If a method isn't
87594a5c3334bba3cc8cd1da85ba1118bc2c080add9Steve Naroff  // found, we search referenced protocols and class categories.
876aa5420c1e36ab8e0e4bb87239d8b73a3a8ce75dbArgyrios Kyrtzidis  ObjCMethodDecl *lookupMethod(Selector Sel, bool isInstance) const;
877aa5420c1e36ab8e0e4bb87239d8b73a3a8ce75dbArgyrios Kyrtzidis  ObjCMethodDecl *lookupInstanceMethod(Selector Sel) const {
878aa5420c1e36ab8e0e4bb87239d8b73a3a8ce75dbArgyrios Kyrtzidis    return lookupMethod(Sel, true/*isInstance*/);
879aa5420c1e36ab8e0e4bb87239d8b73a3a8ce75dbArgyrios Kyrtzidis  }
880aa5420c1e36ab8e0e4bb87239d8b73a3a8ce75dbArgyrios Kyrtzidis  ObjCMethodDecl *lookupClassMethod(Selector Sel) const {
881aa5420c1e36ab8e0e4bb87239d8b73a3a8ce75dbArgyrios Kyrtzidis    return lookupMethod(Sel, false/*isInstance*/);
882aa5420c1e36ab8e0e4bb87239d8b73a3a8ce75dbArgyrios Kyrtzidis  }
883cd1876207f5564beba74e4b2524b664bdba0ba9fFariborz Jahanian  ObjCInterfaceDecl *lookupInheritedClass(const IdentifierInfo *ICName);
884ba243b59a1074e0962f6abfa3bb9aa984eac1245David Blaikie
885d789d3d985e28c9404e62157af46dcb7331920e0Steve Naroff  // Lookup a method in the classes implementation hierarchy.
88674b2756bc1f1f5f7c189996fe7e4cd3efef70263Fariborz Jahanian  ObjCMethodDecl *lookupPrivateMethod(const Selector &Sel, bool Instance=true);
88760fcceeedbfc8b4a99cb942e2bc5aeb9e2f92a1fSteve Naroff
88805c272fed899b8d3142cf080e86a235fc6168862Douglas Gregor  SourceLocation getEndOfDefinitionLoc() const {
88905c272fed899b8d3142cf080e86a235fc6168862Douglas Gregor    if (!hasDefinition())
89005c272fed899b8d3142cf080e86a235fc6168862Douglas Gregor      return getLocation();
89105c272fed899b8d3142cf080e86a235fc6168862Douglas Gregor
89205c272fed899b8d3142cf080e86a235fc6168862Douglas Gregor    return data().EndLoc;
89305c272fed899b8d3142cf080e86a235fc6168862Douglas Gregor  }
89405c272fed899b8d3142cf080e86a235fc6168862Douglas Gregor
89505c272fed899b8d3142cf080e86a235fc6168862Douglas Gregor  void setEndOfDefinitionLoc(SourceLocation LE) { data().EndLoc = LE; }
8961eb4433ac451dc16f4133a88af2d002ac26c58efMike Stump
8972e5c15be82f362611c5928ce853d0685ff98c766Douglas Gregor  void setSuperClassLoc(SourceLocation Loc) { data().SuperClassLoc = Loc; }
8982e5c15be82f362611c5928ce853d0685ff98c766Douglas Gregor  SourceLocation getSuperClassLoc() const { return data().SuperClassLoc; }
8991eb4433ac451dc16f4133a88af2d002ac26c58efMike Stump
90028e71cf851b73a67604735a9a95aef800b144e2eSteve Naroff  /// isImplicitInterfaceDecl - check that this is an implicitly declared
901a526c5c67e5a0473c340903ee542ce570119665fTed Kremenek  /// ObjCInterfaceDecl node. This is for legacy objective-c @implementation
9024b6df3fa953267c5d755628c8afd818bb571e579Fariborz Jahanian  /// declaration without an @interface declaration.
90397bbab2df74cbfe221fb20454738d607a41f3ca4Fariborz Jahanian  bool isImplicitInterfaceDecl() const {
90497bbab2df74cbfe221fb20454738d607a41f3ca4Fariborz Jahanian    return hasDefinition() ? Data->Definition->isImplicit() : isImplicit();
90597bbab2df74cbfe221fb20454738d607a41f3ca4Fariborz Jahanian  }
9061eb4433ac451dc16f4133a88af2d002ac26c58efMike Stump
9070fd8904c5f71a11d29f67716c3ebdf7ad1c855fbFariborz Jahanian  /// ClassImplementsProtocol - Checks that 'lProto' protocol
9080fd8904c5f71a11d29f67716c3ebdf7ad1c855fbFariborz Jahanian  /// has been implemented in IDecl class, its super class or categories (if
9090fd8904c5f71a11d29f67716c3ebdf7ad1c855fbFariborz Jahanian  /// lookupCategory is true).
9100fd8904c5f71a11d29f67716c3ebdf7ad1c855fbFariborz Jahanian  bool ClassImplementsProtocol(ObjCProtocolDecl *lProto,
9110fd8904c5f71a11d29f67716c3ebdf7ad1c855fbFariborz Jahanian                               bool lookupCategory,
9120fd8904c5f71a11d29f67716c3ebdf7ad1c855fbFariborz Jahanian                               bool RHSIsQualifiedID = false);
9131eb4433ac451dc16f4133a88af2d002ac26c58efMike Stump
91453df7a1d34f21d8f2309311d1067d463e9064c60Douglas Gregor  typedef redeclarable_base::redecl_iterator redecl_iterator;
91553df7a1d34f21d8f2309311d1067d463e9064c60Douglas Gregor  redecl_iterator redecls_begin() const {
91653df7a1d34f21d8f2309311d1067d463e9064c60Douglas Gregor    return redeclarable_base::redecls_begin();
91753df7a1d34f21d8f2309311d1067d463e9064c60Douglas Gregor  }
91853df7a1d34f21d8f2309311d1067d463e9064c60Douglas Gregor  redecl_iterator redecls_end() const {
91953df7a1d34f21d8f2309311d1067d463e9064c60Douglas Gregor    return redeclarable_base::redecls_end();
92053df7a1d34f21d8f2309311d1067d463e9064c60Douglas Gregor  }
92153df7a1d34f21d8f2309311d1067d463e9064c60Douglas Gregor
92253df7a1d34f21d8f2309311d1067d463e9064c60Douglas Gregor  /// Retrieves the canonical declaration of this Objective-C class.
92353df7a1d34f21d8f2309311d1067d463e9064c60Douglas Gregor  ObjCInterfaceDecl *getCanonicalDecl() {
92453df7a1d34f21d8f2309311d1067d463e9064c60Douglas Gregor    return getFirstDeclaration();
92553df7a1d34f21d8f2309311d1067d463e9064c60Douglas Gregor  }
92653df7a1d34f21d8f2309311d1067d463e9064c60Douglas Gregor  const ObjCInterfaceDecl *getCanonicalDecl() const {
92753df7a1d34f21d8f2309311d1067d463e9064c60Douglas Gregor    return getFirstDeclaration();
92853df7a1d34f21d8f2309311d1067d463e9064c60Douglas Gregor  }
92953df7a1d34f21d8f2309311d1067d463e9064c60Douglas Gregor
93033feeb019a5742b286eededd5446ec0fe87c5a61Steve Naroff  // Low-level accessor
931f4c7371fb1d3cebcfb40abad4537bb82515704eaJohn McCall  const Type *getTypeForDecl() const { return TypeForDecl; }
932f4c7371fb1d3cebcfb40abad4537bb82515704eaJohn McCall  void setTypeForDecl(const Type *TD) const { TypeForDecl = TD; }
93333feeb019a5742b286eededd5446ec0fe87c5a61Steve Naroff
93480cd64a8450d8e2c079dc134d9711cd45ba89d63John McCall  static bool classof(const Decl *D) { return classofKind(D->getKind()); }
935a526c5c67e5a0473c340903ee542ce570119665fTed Kremenek  static bool classof(const ObjCInterfaceDecl *D) { return true; }
93680cd64a8450d8e2c079dc134d9711cd45ba89d63John McCall  static bool classofKind(Kind K) { return K == ObjCInterface; }
93753b9441b5a81a24fa1f66f3f6416f1e36baa9c2fTed Kremenek
938fc529f7fcafe7da0b8a32621e13685891e8ce52aDouglas Gregor  friend class ASTReader;
93953b9441b5a81a24fa1f66f3f6416f1e36baa9c2fTed Kremenek  friend class ASTDeclReader;
94053b9441b5a81a24fa1f66f3f6416f1e36baa9c2fTed Kremenek  friend class ASTDeclWriter;
941980e508ca70d6de75d2abfd96b4681fc98bb2698Steve Naroff};
942980e508ca70d6de75d2abfd96b4681fc98bb2698Steve Naroff
943a526c5c67e5a0473c340903ee542ce570119665fTed Kremenek/// ObjCIvarDecl - Represents an ObjC instance variable. In general, ObjC
9440c6b6243d3efd958c17943130e2a773653511edcSteve Naroff/// instance variables are identical to C. The only exception is Objective-C
9450c6b6243d3efd958c17943130e2a773653511edcSteve Naroff/// supports C++ style access control. For example:
9460c6b6243d3efd958c17943130e2a773653511edcSteve Naroff///
9470c6b6243d3efd958c17943130e2a773653511edcSteve Naroff///   @interface IvarExample : NSObject
9480c6b6243d3efd958c17943130e2a773653511edcSteve Naroff///   {
949f079570fcad0d0053e75ebae29c883ec4276e020Ted Kremenek///     id defaultToProtected;
9500c6b6243d3efd958c17943130e2a773653511edcSteve Naroff///   @public:
9510c6b6243d3efd958c17943130e2a773653511edcSteve Naroff///     id canBePublic; // same as C++.
9520c6b6243d3efd958c17943130e2a773653511edcSteve Naroff///   @protected:
9530c6b6243d3efd958c17943130e2a773653511edcSteve Naroff///     id canBeProtected; // same as C++.
9540c6b6243d3efd958c17943130e2a773653511edcSteve Naroff///   @package:
9550c6b6243d3efd958c17943130e2a773653511edcSteve Naroff///     id canBePackage; // framework visibility (not available in C++).
9560c6b6243d3efd958c17943130e2a773653511edcSteve Naroff///   }
9570c6b6243d3efd958c17943130e2a773653511edcSteve Naroff///
958a526c5c67e5a0473c340903ee542ce570119665fTed Kremenekclass ObjCIvarDecl : public FieldDecl {
95999ba9e3bd70671f3441fb974895f226a83ce0e66David Blaikie  virtual void anchor();
96099ba9e3bd70671f3441fb974895f226a83ce0e66David Blaikie
9610e77ba0bf769e2e5a4a93c079f241b02aeb3ef93Chris Lattnerpublic:
962980e508ca70d6de75d2abfd96b4681fc98bb2698Steve Naroff  enum AccessControl {
963980e508ca70d6de75d2abfd96b4681fc98bb2698Steve Naroff    None, Private, Protected, Public, Package
964980e508ca70d6de75d2abfd96b4681fc98bb2698Steve Naroff  };
9651eb4433ac451dc16f4133a88af2d002ac26c58efMike Stump
966b8db21d4df5fbb6ce1ace6411b82d3d623d789deTed Kremenekprivate:
967ff676cb48fe8bf7be2feaa251dc7c5fb15af4730Abramo Bagnara  ObjCIvarDecl(ObjCContainerDecl *DC, SourceLocation StartLoc,
968ff676cb48fe8bf7be2feaa251dc7c5fb15af4730Abramo Bagnara               SourceLocation IdLoc, IdentifierInfo *Id,
969ad51e74030a59a8aa4ef0ebca1d7a701602ef53bFariborz Jahanian               QualType T, TypeSourceInfo *TInfo, AccessControl ac, Expr *BW,
970ad51e74030a59a8aa4ef0ebca1d7a701602ef53bFariborz Jahanian               bool synthesized)
971ff676cb48fe8bf7be2feaa251dc7c5fb15af4730Abramo Bagnara    : FieldDecl(ObjCIvar, DC, StartLoc, IdLoc, Id, T, TInfo, BW,
9727a614d8380297fcd2bc23986241905d97222948cRichard Smith                /*Mutable=*/false, /*HasInit=*/false),
973ff676cb48fe8bf7be2feaa251dc7c5fb15af4730Abramo Bagnara      NextIvar(0), DeclAccess(ac), Synthesized(synthesized) {}
9741eb4433ac451dc16f4133a88af2d002ac26c58efMike Stump
975b8db21d4df5fbb6ce1ace6411b82d3d623d789deTed Kremenekpublic:
976a06549226f45d5b72169a3d054415616dd1014a2Daniel Dunbar  static ObjCIvarDecl *Create(ASTContext &C, ObjCContainerDecl *DC,
977ff676cb48fe8bf7be2feaa251dc7c5fb15af4730Abramo Bagnara                              SourceLocation StartLoc, SourceLocation IdLoc,
978ff676cb48fe8bf7be2feaa251dc7c5fb15af4730Abramo Bagnara                              IdentifierInfo *Id, QualType T,
979a93c934af4fbf97cbe8e649d82e68ccacfe57c95John McCall                              TypeSourceInfo *TInfo,
980ad51e74030a59a8aa4ef0ebca1d7a701602ef53bFariborz Jahanian                              AccessControl ac, Expr *BW = NULL,
981ad51e74030a59a8aa4ef0ebca1d7a701602ef53bFariborz Jahanian                              bool synthesized=false);
9821eb4433ac451dc16f4133a88af2d002ac26c58efMike Stump
98327a961a6adab85cfcf7e48485bbec9237719ae96Daniel Dunbar  /// \brief Return the class interface that this ivar is logically contained
98427a961a6adab85cfcf7e48485bbec9237719ae96Daniel Dunbar  /// in; this is either the interface where the ivar was declared, or the
98527a961a6adab85cfcf7e48485bbec9237719ae96Daniel Dunbar  /// interface the ivar is conceptually a part of in the case of synthesized
98627a961a6adab85cfcf7e48485bbec9237719ae96Daniel Dunbar  /// ivars.
98727a961a6adab85cfcf7e48485bbec9237719ae96Daniel Dunbar  const ObjCInterfaceDecl *getContainingInterface() const;
988ba243b59a1074e0962f6abfa3bb9aa984eac1245David Blaikie
9892c18bb7c9fca66c30b6eabbdcbc6399d24a54fa9Fariborz Jahanian  ObjCIvarDecl *getNextIvar() { return NextIvar; }
990db8264e4c5ffd7af6fbad4ca4306bd382bb02691Jordy Rose  const ObjCIvarDecl *getNextIvar() const { return NextIvar; }
9912c18bb7c9fca66c30b6eabbdcbc6399d24a54fa9Fariborz Jahanian  void setNextIvar(ObjCIvarDecl *ivar) { NextIvar = ivar; }
99227a961a6adab85cfcf7e48485bbec9237719ae96Daniel Dunbar
993980e508ca70d6de75d2abfd96b4681fc98bb2698Steve Naroff  void setAccessControl(AccessControl ac) { DeclAccess = ac; }
994f079570fcad0d0053e75ebae29c883ec4276e020Ted Kremenek
995ca63fa00786e51c207c829f4182f11a6c6b552beTed Kremenek  AccessControl getAccessControl() const { return AccessControl(DeclAccess); }
996f079570fcad0d0053e75ebae29c883ec4276e020Ted Kremenek
997f079570fcad0d0053e75ebae29c883ec4276e020Ted Kremenek  AccessControl getCanonicalAccessControl() const {
998f079570fcad0d0053e75ebae29c883ec4276e020Ted Kremenek    return DeclAccess == None ? Protected : AccessControl(DeclAccess);
999f079570fcad0d0053e75ebae29c883ec4276e020Ted Kremenek  }
10001eb4433ac451dc16f4133a88af2d002ac26c58efMike Stump
1001ac0021ba802e193e0f9f8207768c7862c7603bc0Fariborz Jahanian  void setSynthesize(bool synth) { Synthesized = synth; }
1002ad51e74030a59a8aa4ef0ebca1d7a701602ef53bFariborz Jahanian  bool getSynthesize() const { return Synthesized; }
1003ba243b59a1074e0962f6abfa3bb9aa984eac1245David Blaikie
1004980e508ca70d6de75d2abfd96b4681fc98bb2698Steve Naroff  // Implement isa/cast/dyncast/etc.
100580cd64a8450d8e2c079dc134d9711cd45ba89d63John McCall  static bool classof(const Decl *D) { return classofKind(D->getKind()); }
1006a526c5c67e5a0473c340903ee542ce570119665fTed Kremenek  static bool classof(const ObjCIvarDecl *D) { return true; }
100780cd64a8450d8e2c079dc134d9711cd45ba89d63John McCall  static bool classofKind(Kind K) { return K == ObjCIvar; }
1008980e508ca70d6de75d2abfd96b4681fc98bb2698Steve Naroffprivate:
1009ba243b59a1074e0962f6abfa3bb9aa984eac1245David Blaikie  /// NextIvar - Next Ivar in the list of ivars declared in class; class's
10102c18bb7c9fca66c30b6eabbdcbc6399d24a54fa9Fariborz Jahanian  /// extensions and class's implementation
10112c18bb7c9fca66c30b6eabbdcbc6399d24a54fa9Fariborz Jahanian  ObjCIvarDecl *NextIvar;
1012ba243b59a1074e0962f6abfa3bb9aa984eac1245David Blaikie
1013ca63fa00786e51c207c829f4182f11a6c6b552beTed Kremenek  // NOTE: VC++ treats enums as signed, avoid using the AccessControl enum
1014ca63fa00786e51c207c829f4182f11a6c6b552beTed Kremenek  unsigned DeclAccess : 3;
1015ad51e74030a59a8aa4ef0ebca1d7a701602ef53bFariborz Jahanian  unsigned Synthesized : 1;
1016980e508ca70d6de75d2abfd96b4681fc98bb2698Steve Naroff};
1017980e508ca70d6de75d2abfd96b4681fc98bb2698Steve Naroff
10181eb4433ac451dc16f4133a88af2d002ac26c58efMike Stump
101901e6779faca1e3a3164c697d6e2dfee0881a6981Ted Kremenek/// ObjCAtDefsFieldDecl - Represents a field declaration created by an
102001e6779faca1e3a3164c697d6e2dfee0881a6981Ted Kremenek///  @defs(...).
102101e6779faca1e3a3164c697d6e2dfee0881a6981Ted Kremenekclass ObjCAtDefsFieldDecl : public FieldDecl {
102299ba9e3bd70671f3441fb974895f226a83ce0e66David Blaikie  virtual void anchor();
1023ff676cb48fe8bf7be2feaa251dc7c5fb15af4730Abramo Bagnara  ObjCAtDefsFieldDecl(DeclContext *DC, SourceLocation StartLoc,
1024ff676cb48fe8bf7be2feaa251dc7c5fb15af4730Abramo Bagnara                      SourceLocation IdLoc, IdentifierInfo *Id,
102501e6779faca1e3a3164c697d6e2dfee0881a6981Ted Kremenek                      QualType T, Expr *BW)
1026ff676cb48fe8bf7be2feaa251dc7c5fb15af4730Abramo Bagnara    : FieldDecl(ObjCAtDefsField, DC, StartLoc, IdLoc, Id, T,
1027a93c934af4fbf97cbe8e649d82e68ccacfe57c95John McCall                /*TInfo=*/0, // FIXME: Do ObjCAtDefs have declarators ?
10287a614d8380297fcd2bc23986241905d97222948cRichard Smith                BW, /*Mutable=*/false, /*HasInit=*/false) {}
10291eb4433ac451dc16f4133a88af2d002ac26c58efMike Stump
103001e6779faca1e3a3164c697d6e2dfee0881a6981Ted Kremenekpublic:
103144b4321feab46299d3f5cfd404680884752a0fcfDouglas Gregor  static ObjCAtDefsFieldDecl *Create(ASTContext &C, DeclContext *DC,
1032ff676cb48fe8bf7be2feaa251dc7c5fb15af4730Abramo Bagnara                                     SourceLocation StartLoc,
1033ff676cb48fe8bf7be2feaa251dc7c5fb15af4730Abramo Bagnara                                     SourceLocation IdLoc, IdentifierInfo *Id,
1034ff676cb48fe8bf7be2feaa251dc7c5fb15af4730Abramo Bagnara                                     QualType T, Expr *BW);
10351eb4433ac451dc16f4133a88af2d002ac26c58efMike Stump
103601e6779faca1e3a3164c697d6e2dfee0881a6981Ted Kremenek  // Implement isa/cast/dyncast/etc.
103780cd64a8450d8e2c079dc134d9711cd45ba89d63John McCall  static bool classof(const Decl *D) { return classofKind(D->getKind()); }
103801e6779faca1e3a3164c697d6e2dfee0881a6981Ted Kremenek  static bool classof(const ObjCAtDefsFieldDecl *D) { return true; }
103980cd64a8450d8e2c079dc134d9711cd45ba89d63John McCall  static bool classofKind(Kind K) { return K == ObjCAtDefsField; }
104001e6779faca1e3a3164c697d6e2dfee0881a6981Ted Kremenek};
1041980e508ca70d6de75d2abfd96b4681fc98bb2698Steve Naroff
1042a526c5c67e5a0473c340903ee542ce570119665fTed Kremenek/// ObjCProtocolDecl - Represents a protocol declaration. ObjC protocols
10431eb4433ac451dc16f4133a88af2d002ac26c58efMike Stump/// declare a pure abstract type (i.e no instance variables are permitted).
1044fc8f0e14ad142ed811e90fbd9a30e419e301c717Chris Lattner/// Protocols originally drew inspiration from C++ pure virtual functions (a C++
10450c6b6243d3efd958c17943130e2a773653511edcSteve Naroff/// feature with nice semantics and lousy syntax:-). Here is an example:
10460c6b6243d3efd958c17943130e2a773653511edcSteve Naroff///
1047eca7be6b7ebd93682eeaab2c71d59f2995dacdccChris Lattner/// @protocol NSDraggingInfo <refproto1, refproto2>
10480c6b6243d3efd958c17943130e2a773653511edcSteve Naroff/// - (NSWindow *)draggingDestinationWindow;
10490c6b6243d3efd958c17943130e2a773653511edcSteve Naroff/// - (NSImage *)draggedImage;
10500c6b6243d3efd958c17943130e2a773653511edcSteve Naroff/// @end
10510c6b6243d3efd958c17943130e2a773653511edcSteve Naroff///
1052eca7be6b7ebd93682eeaab2c71d59f2995dacdccChris Lattner/// This says that NSDraggingInfo requires two methods and requires everything
1053eca7be6b7ebd93682eeaab2c71d59f2995dacdccChris Lattner/// that the two "referenced protocols" 'refproto1' and 'refproto2' require as
1054eca7be6b7ebd93682eeaab2c71d59f2995dacdccChris Lattner/// well.
1055eca7be6b7ebd93682eeaab2c71d59f2995dacdccChris Lattner///
10560c6b6243d3efd958c17943130e2a773653511edcSteve Naroff/// @interface ImplementsNSDraggingInfo : NSObject <NSDraggingInfo>
10570c6b6243d3efd958c17943130e2a773653511edcSteve Naroff/// @end
10580c6b6243d3efd958c17943130e2a773653511edcSteve Naroff///
1059a526c5c67e5a0473c340903ee542ce570119665fTed Kremenek/// ObjC protocols inspired Java interfaces. Unlike Java, ObjC classes and
10600c6b6243d3efd958c17943130e2a773653511edcSteve Naroff/// protocols are in distinct namespaces. For example, Cocoa defines both
10611eb4433ac451dc16f4133a88af2d002ac26c58efMike Stump/// an NSObject protocol and class (which isn't allowed in Java). As a result,
10620c6b6243d3efd958c17943130e2a773653511edcSteve Naroff/// protocols are referenced using angle brackets as follows:
10630c6b6243d3efd958c17943130e2a773653511edcSteve Naroff///
10640c6b6243d3efd958c17943130e2a773653511edcSteve Naroff/// id <NSDraggingInfo> anyObjectThatImplementsNSDraggingInfo;
10650c6b6243d3efd958c17943130e2a773653511edcSteve Naroff///
10661d784b277cdfd4eba03680715d2a082b3f28d295Douglas Gregorclass ObjCProtocolDecl : public ObjCContainerDecl,
10671d784b277cdfd4eba03680715d2a082b3f28d295Douglas Gregor                         public Redeclarable<ObjCProtocolDecl> {
106899ba9e3bd70671f3441fb974895f226a83ce0e66David Blaikie  virtual void anchor();
106999ba9e3bd70671f3441fb974895f226a83ce0e66David Blaikie
10705e2a1ff9f28d2eab256d2553e76a9c9d54693875Douglas Gregor  struct DefinitionData {
10711d784b277cdfd4eba03680715d2a082b3f28d295Douglas Gregor    // \brief The declaration that defines this protocol.
10721d784b277cdfd4eba03680715d2a082b3f28d295Douglas Gregor    ObjCProtocolDecl *Definition;
10731d784b277cdfd4eba03680715d2a082b3f28d295Douglas Gregor
10745e2a1ff9f28d2eab256d2553e76a9c9d54693875Douglas Gregor    /// Referenced protocols
10755e2a1ff9f28d2eab256d2553e76a9c9d54693875Douglas Gregor    ObjCProtocolList ReferencedProtocols;
10765e2a1ff9f28d2eab256d2553e76a9c9d54693875Douglas Gregor  };
10775e2a1ff9f28d2eab256d2553e76a9c9d54693875Douglas Gregor
10785e2a1ff9f28d2eab256d2553e76a9c9d54693875Douglas Gregor  DefinitionData *Data;
10791eb4433ac451dc16f4133a88af2d002ac26c58efMike Stump
1080b05d7b20171bbd2feb14b059f39332cbe1bf1014Argyrios Kyrtzidis  bool InitiallyForwardDecl : 1;
1081b05d7b20171bbd2feb14b059f39332cbe1bf1014Argyrios Kyrtzidis  bool isForwardProtoDecl : 1; // declared with @protocol.
10821eb4433ac451dc16f4133a88af2d002ac26c58efMike Stump
1083423cb565abc681b770fb4b9b4bc24d398c98157bSteve Naroff  SourceLocation EndLoc; // marks the '>' or identifier.
10841eb4433ac451dc16f4133a88af2d002ac26c58efMike Stump
10855e2a1ff9f28d2eab256d2553e76a9c9d54693875Douglas Gregor  DefinitionData &data() const {
10865e2a1ff9f28d2eab256d2553e76a9c9d54693875Douglas Gregor    assert(Data && "Objective-C protocol has no definition!");
10875e2a1ff9f28d2eab256d2553e76a9c9d54693875Douglas Gregor    return *Data;
10885e2a1ff9f28d2eab256d2553e76a9c9d54693875Douglas Gregor  }
10895e2a1ff9f28d2eab256d2553e76a9c9d54693875Douglas Gregor
10901711fc91efb36d131f7ba771f73f0154dc1abd1fArgyrios Kyrtzidis  ObjCProtocolDecl(DeclContext *DC, IdentifierInfo *Id,
1091b05d7b20171bbd2feb14b059f39332cbe1bf1014Argyrios Kyrtzidis                   SourceLocation nameLoc, SourceLocation atStartLoc,
1092b05d7b20171bbd2feb14b059f39332cbe1bf1014Argyrios Kyrtzidis                   bool isForwardDecl)
10931711fc91efb36d131f7ba771f73f0154dc1abd1fArgyrios Kyrtzidis    : ObjCContainerDecl(ObjCProtocol, DC, Id, nameLoc, atStartLoc),
10945e2a1ff9f28d2eab256d2553e76a9c9d54693875Douglas Gregor      Data(0),
1095b05d7b20171bbd2feb14b059f39332cbe1bf1014Argyrios Kyrtzidis      InitiallyForwardDecl(isForwardDecl),
1096b05d7b20171bbd2feb14b059f39332cbe1bf1014Argyrios Kyrtzidis      isForwardProtoDecl(isForwardDecl) {
1097cca59d77c4b84fd2da268018dbaf9431a621e75bChris Lattner  }
10981eb4433ac451dc16f4133a88af2d002ac26c58efMike Stump
10995e2a1ff9f28d2eab256d2553e76a9c9d54693875Douglas Gregor  void allocateDefinitionData();
11001d784b277cdfd4eba03680715d2a082b3f28d295Douglas Gregor
11011d784b277cdfd4eba03680715d2a082b3f28d295Douglas Gregor  typedef Redeclarable<ObjCProtocolDecl> redeclarable_base;
11021d784b277cdfd4eba03680715d2a082b3f28d295Douglas Gregor  virtual ObjCProtocolDecl *getNextRedeclaration() {
11031d784b277cdfd4eba03680715d2a082b3f28d295Douglas Gregor    return RedeclLink.getNext();
11041d784b277cdfd4eba03680715d2a082b3f28d295Douglas Gregor  }
11051d784b277cdfd4eba03680715d2a082b3f28d295Douglas Gregor
1106cca59d77c4b84fd2da268018dbaf9431a621e75bChris Lattnerpublic:
11071eb4433ac451dc16f4133a88af2d002ac26c58efMike Stump  static ObjCProtocolDecl *Create(ASTContext &C, DeclContext *DC,
11081711fc91efb36d131f7ba771f73f0154dc1abd1fArgyrios Kyrtzidis                                  IdentifierInfo *Id,
11091711fc91efb36d131f7ba771f73f0154dc1abd1fArgyrios Kyrtzidis                                  SourceLocation nameLoc,
1110b05d7b20171bbd2feb14b059f39332cbe1bf1014Argyrios Kyrtzidis                                  SourceLocation atStartLoc,
1111b05d7b20171bbd2feb14b059f39332cbe1bf1014Argyrios Kyrtzidis                                  bool isForwardDecl);
1112cca59d77c4b84fd2da268018dbaf9431a621e75bChris Lattner
111318df52bbb5d28ca082064d31ae7558dbdae52377Douglas Gregor  const ObjCProtocolList &getReferencedProtocols() const {
11145e2a1ff9f28d2eab256d2553e76a9c9d54693875Douglas Gregor    assert(hasDefinition() && "No definition available!");
11155e2a1ff9f28d2eab256d2553e76a9c9d54693875Douglas Gregor    return data().ReferencedProtocols;
1116980e508ca70d6de75d2abfd96b4681fc98bb2698Steve Naroff  }
111718df52bbb5d28ca082064d31ae7558dbdae52377Douglas Gregor  typedef ObjCProtocolList::iterator protocol_iterator;
11185e2a1ff9f28d2eab256d2553e76a9c9d54693875Douglas Gregor  protocol_iterator protocol_begin() const {
11195e2a1ff9f28d2eab256d2553e76a9c9d54693875Douglas Gregor    if (!hasDefinition())
11205e2a1ff9f28d2eab256d2553e76a9c9d54693875Douglas Gregor      return protocol_iterator();
11215e2a1ff9f28d2eab256d2553e76a9c9d54693875Douglas Gregor
11225e2a1ff9f28d2eab256d2553e76a9c9d54693875Douglas Gregor    return data().ReferencedProtocols.begin();
11235e2a1ff9f28d2eab256d2553e76a9c9d54693875Douglas Gregor  }
11245e2a1ff9f28d2eab256d2553e76a9c9d54693875Douglas Gregor  protocol_iterator protocol_end() const {
11255e2a1ff9f28d2eab256d2553e76a9c9d54693875Douglas Gregor    if (!hasDefinition())
11265e2a1ff9f28d2eab256d2553e76a9c9d54693875Douglas Gregor      return protocol_iterator();
11275e2a1ff9f28d2eab256d2553e76a9c9d54693875Douglas Gregor
11285e2a1ff9f28d2eab256d2553e76a9c9d54693875Douglas Gregor    return data().ReferencedProtocols.end();
11295e2a1ff9f28d2eab256d2553e76a9c9d54693875Douglas Gregor  }
113018df52bbb5d28ca082064d31ae7558dbdae52377Douglas Gregor  typedef ObjCProtocolList::loc_iterator protocol_loc_iterator;
1131ba243b59a1074e0962f6abfa3bb9aa984eac1245David Blaikie  protocol_loc_iterator protocol_loc_begin() const {
11325e2a1ff9f28d2eab256d2553e76a9c9d54693875Douglas Gregor    if (!hasDefinition())
11335e2a1ff9f28d2eab256d2553e76a9c9d54693875Douglas Gregor      return protocol_loc_iterator();
11345e2a1ff9f28d2eab256d2553e76a9c9d54693875Douglas Gregor
11355e2a1ff9f28d2eab256d2553e76a9c9d54693875Douglas Gregor    return data().ReferencedProtocols.loc_begin();
113618df52bbb5d28ca082064d31ae7558dbdae52377Douglas Gregor  }
1137ba243b59a1074e0962f6abfa3bb9aa984eac1245David Blaikie  protocol_loc_iterator protocol_loc_end() const {
11385e2a1ff9f28d2eab256d2553e76a9c9d54693875Douglas Gregor    if (!hasDefinition())
11395e2a1ff9f28d2eab256d2553e76a9c9d54693875Douglas Gregor      return protocol_loc_iterator();
11405e2a1ff9f28d2eab256d2553e76a9c9d54693875Douglas Gregor
11415e2a1ff9f28d2eab256d2553e76a9c9d54693875Douglas Gregor    return data().ReferencedProtocols.loc_end();
11425e2a1ff9f28d2eab256d2553e76a9c9d54693875Douglas Gregor  }
11435e2a1ff9f28d2eab256d2553e76a9c9d54693875Douglas Gregor  unsigned protocol_size() const {
11445e2a1ff9f28d2eab256d2553e76a9c9d54693875Douglas Gregor    if (!hasDefinition())
11455e2a1ff9f28d2eab256d2553e76a9c9d54693875Douglas Gregor      return 0;
11465e2a1ff9f28d2eab256d2553e76a9c9d54693875Douglas Gregor
11475e2a1ff9f28d2eab256d2553e76a9c9d54693875Douglas Gregor    return data().ReferencedProtocols.size();
114818df52bbb5d28ca082064d31ae7558dbdae52377Douglas Gregor  }
11491eb4433ac451dc16f4133a88af2d002ac26c58efMike Stump
115038af2deb27cdfa1a95bde96e30dab15dce53fcefChris Lattner  /// setProtocolList - Set the list of protocols that this interface
1151780f329cb011bff0da5763e2e9744eec093d0509Chris Lattner  /// implements.
115238af2deb27cdfa1a95bde96e30dab15dce53fcefChris Lattner  void setProtocolList(ObjCProtocolDecl *const*List, unsigned Num,
115318df52bbb5d28ca082064d31ae7558dbdae52377Douglas Gregor                       const SourceLocation *Locs, ASTContext &C) {
11545e2a1ff9f28d2eab256d2553e76a9c9d54693875Douglas Gregor    assert(Data && "Protocol is not defined");
11555e2a1ff9f28d2eab256d2553e76a9c9d54693875Douglas Gregor    data().ReferencedProtocols.set(List, Num, Locs, C);
1156aebf0cba02c014ac8b19d615c654248e0e93779fFariborz Jahanian  }
11571eb4433ac451dc16f4133a88af2d002ac26c58efMike Stump
115891b0b0cf6b537cbcbca0038c7032f87161a41d31Steve Naroff  ObjCProtocolDecl *lookupProtocolNamed(IdentifierInfo *PName);
11591eb4433ac451dc16f4133a88af2d002ac26c58efMike Stump
116094a5c3334bba3cc8cd1da85ba1118bc2c080add9Steve Naroff  // Lookup a method. First, we search locally. If a method isn't
116194a5c3334bba3cc8cd1da85ba1118bc2c080add9Steve Naroff  // found, we search referenced protocols and class categories.
1162094e2bb6730d63e0f6919e4839522a43b7644181Argyrios Kyrtzidis  ObjCMethodDecl *lookupMethod(Selector Sel, bool isInstance) const;
1163094e2bb6730d63e0f6919e4839522a43b7644181Argyrios Kyrtzidis  ObjCMethodDecl *lookupInstanceMethod(Selector Sel) const {
1164094e2bb6730d63e0f6919e4839522a43b7644181Argyrios Kyrtzidis    return lookupMethod(Sel, true/*isInstance*/);
1165094e2bb6730d63e0f6919e4839522a43b7644181Argyrios Kyrtzidis  }
1166094e2bb6730d63e0f6919e4839522a43b7644181Argyrios Kyrtzidis  ObjCMethodDecl *lookupClassMethod(Selector Sel) const {
1167094e2bb6730d63e0f6919e4839522a43b7644181Argyrios Kyrtzidis    return lookupMethod(Sel, false/*isInstance*/);
1168094e2bb6730d63e0f6919e4839522a43b7644181Argyrios Kyrtzidis  }
1169b05d7b20171bbd2feb14b059f39332cbe1bf1014Argyrios Kyrtzidis
11705e2a1ff9f28d2eab256d2553e76a9c9d54693875Douglas Gregor  /// \brief Determine whether this protocol has a definition.
11715e2a1ff9f28d2eab256d2553e76a9c9d54693875Douglas Gregor  bool hasDefinition() const { return Data != 0; }
11725e2a1ff9f28d2eab256d2553e76a9c9d54693875Douglas Gregor
11735e2a1ff9f28d2eab256d2553e76a9c9d54693875Douglas Gregor  /// \brief Retrieve the definition of this protocol, if any.
11745e2a1ff9f28d2eab256d2553e76a9c9d54693875Douglas Gregor  ObjCProtocolDecl *getDefinition() {
11751d784b277cdfd4eba03680715d2a082b3f28d295Douglas Gregor    return Data? Data->Definition : 0;
11765e2a1ff9f28d2eab256d2553e76a9c9d54693875Douglas Gregor  }
11775e2a1ff9f28d2eab256d2553e76a9c9d54693875Douglas Gregor
11785e2a1ff9f28d2eab256d2553e76a9c9d54693875Douglas Gregor  /// \brief Retrieve the definition of this protocol, if any.
11795e2a1ff9f28d2eab256d2553e76a9c9d54693875Douglas Gregor  const ObjCProtocolDecl *getDefinition() const {
11801d784b277cdfd4eba03680715d2a082b3f28d295Douglas Gregor    return Data? Data->Definition : 0;
11815e2a1ff9f28d2eab256d2553e76a9c9d54693875Douglas Gregor  }
11825e2a1ff9f28d2eab256d2553e76a9c9d54693875Douglas Gregor
11835e2a1ff9f28d2eab256d2553e76a9c9d54693875Douglas Gregor  /// \brief Determine whether this particular declaration is also the
11845e2a1ff9f28d2eab256d2553e76a9c9d54693875Douglas Gregor  /// definition.
11855e2a1ff9f28d2eab256d2553e76a9c9d54693875Douglas Gregor  bool isThisDeclarationADefinition() const {
11865e2a1ff9f28d2eab256d2553e76a9c9d54693875Douglas Gregor    return getDefinition() == this;
11875e2a1ff9f28d2eab256d2553e76a9c9d54693875Douglas Gregor  }
11885e2a1ff9f28d2eab256d2553e76a9c9d54693875Douglas Gregor
11895e2a1ff9f28d2eab256d2553e76a9c9d54693875Douglas Gregor  /// \brief Starts the definition of this Objective-C protocol.
11905e2a1ff9f28d2eab256d2553e76a9c9d54693875Douglas Gregor  void startDefinition();
11915e2a1ff9f28d2eab256d2553e76a9c9d54693875Douglas Gregor
1192b05d7b20171bbd2feb14b059f39332cbe1bf1014Argyrios Kyrtzidis  /// \brief True if it was initially a forward reference.
1193b05d7b20171bbd2feb14b059f39332cbe1bf1014Argyrios Kyrtzidis  /// Differs with \see isForwardDecl in that \see isForwardDecl will change to
1194b05d7b20171bbd2feb14b059f39332cbe1bf1014Argyrios Kyrtzidis  /// false when we see the definition, but this will remain true.
1195b05d7b20171bbd2feb14b059f39332cbe1bf1014Argyrios Kyrtzidis  bool isInitiallyForwardDecl() const { return InitiallyForwardDecl; }
1196ba243b59a1074e0962f6abfa3bb9aa984eac1245David Blaikie
1197ad834d534e9a5db3d3baa09593775f83ceaff1f2Argyrios Kyrtzidis  void completedForwardDecl();
1198980e508ca70d6de75d2abfd96b4681fc98bb2698Steve Naroff
11991eb4433ac451dc16f4133a88af2d002ac26c58efMike Stump  // Location information, modeled after the Stmt API.
12001711fc91efb36d131f7ba771f73f0154dc1abd1fArgyrios Kyrtzidis  SourceLocation getLocStart() const { return getAtStartLoc(); } // '@'protocol
1201423cb565abc681b770fb4b9b4bc24d398c98157bSteve Naroff  SourceLocation getLocEnd() const { return EndLoc; }
12027177dee8aee4b432911c91f1b788963bec0cac9fDaniel Dunbar  void setLocEnd(SourceLocation LE) { EndLoc = LE; }
12031eb4433ac451dc16f4133a88af2d002ac26c58efMike Stump
12041d784b277cdfd4eba03680715d2a082b3f28d295Douglas Gregor  typedef redeclarable_base::redecl_iterator redecl_iterator;
12051d784b277cdfd4eba03680715d2a082b3f28d295Douglas Gregor  redecl_iterator redecls_begin() const {
12061d784b277cdfd4eba03680715d2a082b3f28d295Douglas Gregor    return redeclarable_base::redecls_begin();
12071d784b277cdfd4eba03680715d2a082b3f28d295Douglas Gregor  }
12081d784b277cdfd4eba03680715d2a082b3f28d295Douglas Gregor  redecl_iterator redecls_end() const {
12091d784b277cdfd4eba03680715d2a082b3f28d295Douglas Gregor    return redeclarable_base::redecls_end();
12101d784b277cdfd4eba03680715d2a082b3f28d295Douglas Gregor  }
12111d784b277cdfd4eba03680715d2a082b3f28d295Douglas Gregor
12123fc73ee0c613715ebce78e30b4d050ea715a007dDouglas Gregor  /// Retrieves the canonical declaration of this Objective-C protocol.
12133fc73ee0c613715ebce78e30b4d050ea715a007dDouglas Gregor  ObjCProtocolDecl *getCanonicalDecl() {
12141d784b277cdfd4eba03680715d2a082b3f28d295Douglas Gregor    return getFirstDeclaration();
12153fc73ee0c613715ebce78e30b4d050ea715a007dDouglas Gregor  }
12163fc73ee0c613715ebce78e30b4d050ea715a007dDouglas Gregor  const ObjCProtocolDecl *getCanonicalDecl() const {
12171d784b277cdfd4eba03680715d2a082b3f28d295Douglas Gregor    return getFirstDeclaration();
12183fc73ee0c613715ebce78e30b4d050ea715a007dDouglas Gregor  }
12193fc73ee0c613715ebce78e30b4d050ea715a007dDouglas Gregor
122080cd64a8450d8e2c079dc134d9711cd45ba89d63John McCall  static bool classof(const Decl *D) { return classofKind(D->getKind()); }
1221a526c5c67e5a0473c340903ee542ce570119665fTed Kremenek  static bool classof(const ObjCProtocolDecl *D) { return true; }
122280cd64a8450d8e2c079dc134d9711cd45ba89d63John McCall  static bool classofKind(Kind K) { return K == ObjCProtocol; }
1223b05d7b20171bbd2feb14b059f39332cbe1bf1014Argyrios Kyrtzidis
12245e2a1ff9f28d2eab256d2553e76a9c9d54693875Douglas Gregor  friend class ASTReader;
1225b05d7b20171bbd2feb14b059f39332cbe1bf1014Argyrios Kyrtzidis  friend class ASTDeclReader;
1226b05d7b20171bbd2feb14b059f39332cbe1bf1014Argyrios Kyrtzidis  friend class ASTDeclWriter;
1227980e508ca70d6de75d2abfd96b4681fc98bb2698Steve Naroff};
12281eb4433ac451dc16f4133a88af2d002ac26c58efMike Stump
1229a526c5c67e5a0473c340903ee542ce570119665fTed Kremenek/// ObjCForwardProtocolDecl - Specifies a list of forward protocol declarations.
123006ae8d68ef258ccd40c9cd1ce762eaae6f3d4432Steve Naroff/// For example:
12311eb4433ac451dc16f4133a88af2d002ac26c58efMike Stump///
12320c6b6243d3efd958c17943130e2a773653511edcSteve Naroff/// @protocol NSTextInput, NSChangeSpelling, NSDraggingInfo;
12331eb4433ac451dc16f4133a88af2d002ac26c58efMike Stump///
12344afa39deaa245592977136d367251ee2c173dd8dDouglas Gregorclass ObjCForwardProtocolDecl : public Decl {
123599ba9e3bd70671f3441fb974895f226a83ce0e66David Blaikie  virtual void anchor();
123699ba9e3bd70671f3441fb974895f226a83ce0e66David Blaikie
123718df52bbb5d28ca082064d31ae7558dbdae52377Douglas Gregor  ObjCProtocolList ReferencedProtocols;
12381eb4433ac451dc16f4133a88af2d002ac26c58efMike Stump
1239d04341000d35c8808a72838b057eed7bf13b7661Douglas Gregor  ObjCForwardProtocolDecl(DeclContext *DC, SourceLocation L,
124038af2deb27cdfa1a95bde96e30dab15dce53fcefChris Lattner                          ObjCProtocolDecl *const *Elts, unsigned nElts,
124118df52bbb5d28ca082064d31ae7558dbdae52377Douglas Gregor                          const SourceLocation *Locs, ASTContext &C);
12421eb4433ac451dc16f4133a88af2d002ac26c58efMike Stump
124361f9d41036e30ff80130f99b31c0626e3ef057ccChris Lattnerpublic:
1244d04341000d35c8808a72838b057eed7bf13b7661Douglas Gregor  static ObjCForwardProtocolDecl *Create(ASTContext &C, DeclContext *DC,
12451eb4433ac451dc16f4133a88af2d002ac26c58efMike Stump                                         SourceLocation L,
124618df52bbb5d28ca082064d31ae7558dbdae52377Douglas Gregor                                         ObjCProtocolDecl *const *Elts,
124718df52bbb5d28ca082064d31ae7558dbdae52377Douglas Gregor                                         unsigned Num,
124818df52bbb5d28ca082064d31ae7558dbdae52377Douglas Gregor                                         const SourceLocation *Locs);
124918df52bbb5d28ca082064d31ae7558dbdae52377Douglas Gregor
125018df52bbb5d28ca082064d31ae7558dbdae52377Douglas Gregor  static ObjCForwardProtocolDecl *Create(ASTContext &C, DeclContext *DC,
125118df52bbb5d28ca082064d31ae7558dbdae52377Douglas Gregor                                         SourceLocation L) {
125218df52bbb5d28ca082064d31ae7558dbdae52377Douglas Gregor    return Create(C, DC, L, 0, 0, 0);
125318df52bbb5d28ca082064d31ae7558dbdae52377Douglas Gregor  }
125461f9d41036e30ff80130f99b31c0626e3ef057ccChris Lattner
125518df52bbb5d28ca082064d31ae7558dbdae52377Douglas Gregor  typedef ObjCProtocolList::iterator protocol_iterator;
125630833f8d77c08f8f16371776fde85a9fde3d9b6eSteve Naroff  protocol_iterator protocol_begin() const {return ReferencedProtocols.begin();}
125730833f8d77c08f8f16371776fde85a9fde3d9b6eSteve Naroff  protocol_iterator protocol_end() const { return ReferencedProtocols.end(); }
125818df52bbb5d28ca082064d31ae7558dbdae52377Douglas Gregor  typedef ObjCProtocolList::loc_iterator protocol_loc_iterator;
1259ba243b59a1074e0962f6abfa3bb9aa984eac1245David Blaikie  protocol_loc_iterator protocol_loc_begin() const {
1260ba243b59a1074e0962f6abfa3bb9aa984eac1245David Blaikie    return ReferencedProtocols.loc_begin();
126118df52bbb5d28ca082064d31ae7558dbdae52377Douglas Gregor  }
1262ba243b59a1074e0962f6abfa3bb9aa984eac1245David Blaikie  protocol_loc_iterator protocol_loc_end() const {
1263ba243b59a1074e0962f6abfa3bb9aa984eac1245David Blaikie    return ReferencedProtocols.loc_end();
126418df52bbb5d28ca082064d31ae7558dbdae52377Douglas Gregor  }
126518df52bbb5d28ca082064d31ae7558dbdae52377Douglas Gregor
126630833f8d77c08f8f16371776fde85a9fde3d9b6eSteve Naroff  unsigned protocol_size() const { return ReferencedProtocols.size(); }
126730833f8d77c08f8f16371776fde85a9fde3d9b6eSteve Naroff
126830833f8d77c08f8f16371776fde85a9fde3d9b6eSteve Naroff  /// setProtocolList - Set the list of forward protocols.
126930833f8d77c08f8f16371776fde85a9fde3d9b6eSteve Naroff  void setProtocolList(ObjCProtocolDecl *const*List, unsigned Num,
127018df52bbb5d28ca082064d31ae7558dbdae52377Douglas Gregor                       const SourceLocation *Locs, ASTContext &C) {
127118df52bbb5d28ca082064d31ae7558dbdae52377Douglas Gregor    ReferencedProtocols.set(List, Num, Locs, C);
127230833f8d77c08f8f16371776fde85a9fde3d9b6eSteve Naroff  }
127380cd64a8450d8e2c079dc134d9711cd45ba89d63John McCall  static bool classof(const Decl *D) { return classofKind(D->getKind()); }
1274a526c5c67e5a0473c340903ee542ce570119665fTed Kremenek  static bool classof(const ObjCForwardProtocolDecl *D) { return true; }
127580cd64a8450d8e2c079dc134d9711cd45ba89d63John McCall  static bool classofKind(Kind K) { return K == ObjCForwardProtocol; }
1276980e508ca70d6de75d2abfd96b4681fc98bb2698Steve Naroff};
1277980e508ca70d6de75d2abfd96b4681fc98bb2698Steve Naroff
1278a526c5c67e5a0473c340903ee542ce570119665fTed Kremenek/// ObjCCategoryDecl - Represents a category declaration. A category allows
12790c6b6243d3efd958c17943130e2a773653511edcSteve Naroff/// you to add methods to an existing class (without subclassing or modifying
12801eb4433ac451dc16f4133a88af2d002ac26c58efMike Stump/// the original class interface or implementation:-). Categories don't allow
12810c6b6243d3efd958c17943130e2a773653511edcSteve Naroff/// you to add instance data. The following example adds "myMethod" to all
12820c6b6243d3efd958c17943130e2a773653511edcSteve Naroff/// NSView's within a process:
12830c6b6243d3efd958c17943130e2a773653511edcSteve Naroff///
12840c6b6243d3efd958c17943130e2a773653511edcSteve Naroff/// @interface NSView (MyViewMethods)
12850c6b6243d3efd958c17943130e2a773653511edcSteve Naroff/// - myMethod;
12860c6b6243d3efd958c17943130e2a773653511edcSteve Naroff/// @end
12870c6b6243d3efd958c17943130e2a773653511edcSteve Naroff///
128833ced0b8550f3e7169f326944731ee02e9338659Douglas Gregor/// Categories also allow you to split the implementation of a class across
12890c6b6243d3efd958c17943130e2a773653511edcSteve Naroff/// several files (a feature more naturally supported in C++).
12900c6b6243d3efd958c17943130e2a773653511edcSteve Naroff///
12910c6b6243d3efd958c17943130e2a773653511edcSteve Naroff/// Categories were originally inspired by dynamic languages such as Common
12921eb4433ac451dc16f4133a88af2d002ac26c58efMike Stump/// Lisp and Smalltalk.  More traditional class-based languages (C++, Java)
12930c6b6243d3efd958c17943130e2a773653511edcSteve Naroff/// don't support this level of dynamism, which is both powerful and dangerous.
12940c6b6243d3efd958c17943130e2a773653511edcSteve Naroff///
1295e4f039e01e797a38bc97bf22aff9832ecd18ff5fSteve Naroffclass ObjCCategoryDecl : public ObjCContainerDecl {
129699ba9e3bd70671f3441fb974895f226a83ce0e66David Blaikie  virtual void anchor();
129799ba9e3bd70671f3441fb974895f226a83ce0e66David Blaikie
1298980e508ca70d6de75d2abfd96b4681fc98bb2698Steve Naroff  /// Interface belonging to this category
1299a526c5c67e5a0473c340903ee542ce570119665fTed Kremenek  ObjCInterfaceDecl *ClassInterface;
13001eb4433ac451dc16f4133a88af2d002ac26c58efMike Stump
130168c82cf61228102aba1194efef222fa1478af2a8Chris Lattner  /// referenced protocols in this category.
130218df52bbb5d28ca082064d31ae7558dbdae52377Douglas Gregor  ObjCProtocolList ReferencedProtocols;
13031eb4433ac451dc16f4133a88af2d002ac26c58efMike Stump
1304a8ff9f455d94d9609766cfd5186b6e21dc2102f1Chris Lattner  /// Next category belonging to this class.
1305a8ff9f455d94d9609766cfd5186b6e21dc2102f1Chris Lattner  /// FIXME: this should not be a singly-linked list.  Move storage elsewhere.
1306a526c5c67e5a0473c340903ee542ce570119665fTed Kremenek  ObjCCategoryDecl *NextClassCategory;
13071eb4433ac451dc16f4133a88af2d002ac26c58efMike Stump
1308000835d0b04345c0014c603fe6339b3bc154050eFariborz Jahanian  /// true of class extension has at least one bitfield ivar.
1309000835d0b04345c0014c603fe6339b3bc154050eFariborz Jahanian  bool HasSynthBitfield : 1;
13103db211b617c5073aa70eb25d37ed44ae0dca17c4Douglas Gregor
13113db211b617c5073aa70eb25d37ed44ae0dca17c4Douglas Gregor  /// \brief The location of the category name in this declaration.
13123db211b617c5073aa70eb25d37ed44ae0dca17c4Douglas Gregor  SourceLocation CategoryNameLoc;
13131eb4433ac451dc16f4133a88af2d002ac26c58efMike Stump
1314ba243b59a1074e0962f6abfa3bb9aa984eac1245David Blaikie  ObjCCategoryDecl(DeclContext *DC, SourceLocation AtLoc,
13153db211b617c5073aa70eb25d37ed44ae0dca17c4Douglas Gregor                   SourceLocation ClassNameLoc, SourceLocation CategoryNameLoc,
1316955fadbdfecfa24a590febe66a86519096787f2dArgyrios Kyrtzidis                   IdentifierInfo *Id, ObjCInterfaceDecl *IDecl)
13171711fc91efb36d131f7ba771f73f0154dc1abd1fArgyrios Kyrtzidis    : ObjCContainerDecl(ObjCCategory, DC, Id, ClassNameLoc, AtLoc),
1318955fadbdfecfa24a590febe66a86519096787f2dArgyrios Kyrtzidis      ClassInterface(IDecl), NextClassCategory(0), HasSynthBitfield(false),
13191711fc91efb36d131f7ba771f73f0154dc1abd1fArgyrios Kyrtzidis      CategoryNameLoc(CategoryNameLoc) {
1320a906135721c350435319347d2672bbb3bf494f91Chris Lattner  }
132161f9d41036e30ff80130f99b31c0626e3ef057ccChris Lattnerpublic:
13221eb4433ac451dc16f4133a88af2d002ac26c58efMike Stump
1323d04341000d35c8808a72838b057eed7bf13b7661Douglas Gregor  static ObjCCategoryDecl *Create(ASTContext &C, DeclContext *DC,
1324ba243b59a1074e0962f6abfa3bb9aa984eac1245David Blaikie                                  SourceLocation AtLoc,
13253db211b617c5073aa70eb25d37ed44ae0dca17c4Douglas Gregor                                  SourceLocation ClassNameLoc,
13263db211b617c5073aa70eb25d37ed44ae0dca17c4Douglas Gregor                                  SourceLocation CategoryNameLoc,
1327955fadbdfecfa24a590febe66a86519096787f2dArgyrios Kyrtzidis                                  IdentifierInfo *Id,
1328955fadbdfecfa24a590febe66a86519096787f2dArgyrios Kyrtzidis                                  ObjCInterfaceDecl *IDecl);
1329955fadbdfecfa24a590febe66a86519096787f2dArgyrios Kyrtzidis  static ObjCCategoryDecl *Create(ASTContext &C, EmptyShell Empty);
13301eb4433ac451dc16f4133a88af2d002ac26c58efMike Stump
1331e0def7589a8afa8a6acef13476bb3f882c104b91Chris Lattner  ObjCInterfaceDecl *getClassInterface() { return ClassInterface; }
1332e0def7589a8afa8a6acef13476bb3f882c104b91Chris Lattner  const ObjCInterfaceDecl *getClassInterface() const { return ClassInterface; }
13338a1d722f13df383600f36d77f842957c8adb5f1bArgyrios Kyrtzidis
13348a1d722f13df383600f36d77f842957c8adb5f1bArgyrios Kyrtzidis  ObjCCategoryImplDecl *getImplementation() const;
13358a1d722f13df383600f36d77f842957c8adb5f1bArgyrios Kyrtzidis  void setImplementation(ObjCCategoryImplDecl *ImplD);
13368a1d722f13df383600f36d77f842957c8adb5f1bArgyrios Kyrtzidis
133738af2deb27cdfa1a95bde96e30dab15dce53fcefChris Lattner  /// setProtocolList - Set the list of protocols that this interface
1338f7b2c98c16dfb2261ea57d40a1d5bc4738e73175Chris Lattner  /// implements.
133938af2deb27cdfa1a95bde96e30dab15dce53fcefChris Lattner  void setProtocolList(ObjCProtocolDecl *const*List, unsigned Num,
134018df52bbb5d28ca082064d31ae7558dbdae52377Douglas Gregor                       const SourceLocation *Locs, ASTContext &C) {
134118df52bbb5d28ca082064d31ae7558dbdae52377Douglas Gregor    ReferencedProtocols.set(List, Num, Locs, C);
1342780f329cb011bff0da5763e2e9744eec093d0509Chris Lattner  }
13431eb4433ac451dc16f4133a88af2d002ac26c58efMike Stump
134418df52bbb5d28ca082064d31ae7558dbdae52377Douglas Gregor  const ObjCProtocolList &getReferencedProtocols() const {
1345780f329cb011bff0da5763e2e9744eec093d0509Chris Lattner    return ReferencedProtocols;
13468f3fde00ad4d4f943321e338b914ae4740711c84Fariborz Jahanian  }
13471eb4433ac451dc16f4133a88af2d002ac26c58efMike Stump
134818df52bbb5d28ca082064d31ae7558dbdae52377Douglas Gregor  typedef ObjCProtocolList::iterator protocol_iterator;
1349780f329cb011bff0da5763e2e9744eec093d0509Chris Lattner  protocol_iterator protocol_begin() const {return ReferencedProtocols.begin();}
1350780f329cb011bff0da5763e2e9744eec093d0509Chris Lattner  protocol_iterator protocol_end() const { return ReferencedProtocols.end(); }
135130833f8d77c08f8f16371776fde85a9fde3d9b6eSteve Naroff  unsigned protocol_size() const { return ReferencedProtocols.size(); }
135218df52bbb5d28ca082064d31ae7558dbdae52377Douglas Gregor  typedef ObjCProtocolList::loc_iterator protocol_loc_iterator;
1353ba243b59a1074e0962f6abfa3bb9aa984eac1245David Blaikie  protocol_loc_iterator protocol_loc_begin() const {
1354ba243b59a1074e0962f6abfa3bb9aa984eac1245David Blaikie    return ReferencedProtocols.loc_begin();
135518df52bbb5d28ca082064d31ae7558dbdae52377Douglas Gregor  }
1356ba243b59a1074e0962f6abfa3bb9aa984eac1245David Blaikie  protocol_loc_iterator protocol_loc_end() const {
1357ba243b59a1074e0962f6abfa3bb9aa984eac1245David Blaikie    return ReferencedProtocols.loc_end();
135818df52bbb5d28ca082064d31ae7558dbdae52377Douglas Gregor  }
13591eb4433ac451dc16f4133a88af2d002ac26c58efMike Stump
1360a526c5c67e5a0473c340903ee542ce570119665fTed Kremenek  ObjCCategoryDecl *getNextClassCategory() const { return NextClassCategory; }
13613db211b617c5073aa70eb25d37ed44ae0dca17c4Douglas Gregor
136225760611365be23556b32332f8a66ae21ea93ecfFariborz Jahanian  bool IsClassExtension() const { return getIdentifier() == 0; }
136380aa1cd7973561889e51c1c152c8990a8de9c953Fariborz Jahanian  const ObjCCategoryDecl *getNextClassExtension() const;
1364ba243b59a1074e0962f6abfa3bb9aa984eac1245David Blaikie
1365000835d0b04345c0014c603fe6339b3bc154050eFariborz Jahanian  bool hasSynthBitfield() const { return HasSynthBitfield; }
1366000835d0b04345c0014c603fe6339b3bc154050eFariborz Jahanian  void setHasSynthBitfield (bool val) { HasSynthBitfield = val; }
1367ba243b59a1074e0962f6abfa3bb9aa984eac1245David Blaikie
13680e5ad255729ee86b8ed57e659029008984517cdeFariborz Jahanian  typedef specific_decl_iterator<ObjCIvarDecl> ivar_iterator;
13690e5ad255729ee86b8ed57e659029008984517cdeFariborz Jahanian  ivar_iterator ivar_begin() const {
13700e5ad255729ee86b8ed57e659029008984517cdeFariborz Jahanian    return ivar_iterator(decls_begin());
13710e5ad255729ee86b8ed57e659029008984517cdeFariborz Jahanian  }
13720e5ad255729ee86b8ed57e659029008984517cdeFariborz Jahanian  ivar_iterator ivar_end() const {
13730e5ad255729ee86b8ed57e659029008984517cdeFariborz Jahanian    return ivar_iterator(decls_end());
13740e5ad255729ee86b8ed57e659029008984517cdeFariborz Jahanian  }
13750e5ad255729ee86b8ed57e659029008984517cdeFariborz Jahanian  unsigned ivar_size() const {
13760e5ad255729ee86b8ed57e659029008984517cdeFariborz Jahanian    return std::distance(ivar_begin(), ivar_end());
13770e5ad255729ee86b8ed57e659029008984517cdeFariborz Jahanian  }
13780e5ad255729ee86b8ed57e659029008984517cdeFariborz Jahanian  bool ivar_empty() const {
13790e5ad255729ee86b8ed57e659029008984517cdeFariborz Jahanian    return ivar_begin() == ivar_end();
13800e5ad255729ee86b8ed57e659029008984517cdeFariborz Jahanian  }
13813db211b617c5073aa70eb25d37ed44ae0dca17c4Douglas Gregor
13823db211b617c5073aa70eb25d37ed44ae0dca17c4Douglas Gregor  SourceLocation getCategoryNameLoc() const { return CategoryNameLoc; }
13833db211b617c5073aa70eb25d37ed44ae0dca17c4Douglas Gregor  void setCategoryNameLoc(SourceLocation Loc) { CategoryNameLoc = Loc; }
13843db211b617c5073aa70eb25d37ed44ae0dca17c4Douglas Gregor
138580cd64a8450d8e2c079dc134d9711cd45ba89d63John McCall  static bool classof(const Decl *D) { return classofKind(D->getKind()); }
1386a526c5c67e5a0473c340903ee542ce570119665fTed Kremenek  static bool classof(const ObjCCategoryDecl *D) { return true; }
138780cd64a8450d8e2c079dc134d9711cd45ba89d63John McCall  static bool classofKind(Kind K) { return K == ObjCCategory; }
1388955fadbdfecfa24a590febe66a86519096787f2dArgyrios Kyrtzidis
1389955fadbdfecfa24a590febe66a86519096787f2dArgyrios Kyrtzidis  friend class ASTDeclReader;
1390955fadbdfecfa24a590febe66a86519096787f2dArgyrios Kyrtzidis  friend class ASTDeclWriter;
1391980e508ca70d6de75d2abfd96b4681fc98bb2698Steve Naroff};
13920c6b6243d3efd958c17943130e2a773653511edcSteve Naroff
1393aecae629269fae3bf484baf1d109e9a89d14eeadArgyrios Kyrtzidisclass ObjCImplDecl : public ObjCContainerDecl {
139499ba9e3bd70671f3441fb974895f226a83ce0e66David Blaikie  virtual void anchor();
139599ba9e3bd70671f3441fb974895f226a83ce0e66David Blaikie
13960d69b8cc8e90a9364771837cb42d7031b4cbb984Steve Naroff  /// Class interface for this class/category implementation
1397a526c5c67e5a0473c340903ee542ce570119665fTed Kremenek  ObjCInterfaceDecl *ClassInterface;
13981eb4433ac451dc16f4133a88af2d002ac26c58efMike Stump
13993aa1861bd8b5121e53379b1a00f9d6ad8dead4f6Chris Lattnerprotected:
14001711fc91efb36d131f7ba771f73f0154dc1abd1fArgyrios Kyrtzidis  ObjCImplDecl(Kind DK, DeclContext *DC,
14011711fc91efb36d131f7ba771f73f0154dc1abd1fArgyrios Kyrtzidis               ObjCInterfaceDecl *classInterface,
14021711fc91efb36d131f7ba771f73f0154dc1abd1fArgyrios Kyrtzidis               SourceLocation nameLoc, SourceLocation atStartLoc)
14031711fc91efb36d131f7ba771f73f0154dc1abd1fArgyrios Kyrtzidis    : ObjCContainerDecl(DK, DC,
14041711fc91efb36d131f7ba771f73f0154dc1abd1fArgyrios Kyrtzidis                        classInterface? classInterface->getIdentifier() : 0,
14051711fc91efb36d131f7ba771f73f0154dc1abd1fArgyrios Kyrtzidis                        nameLoc, atStartLoc),
1406aecae629269fae3bf484baf1d109e9a89d14eeadArgyrios Kyrtzidis      ClassInterface(classInterface) {}
14071eb4433ac451dc16f4133a88af2d002ac26c58efMike Stump
140875c9cae5f85c72cbb1649e93849e16ede3f07522Chris Lattnerpublic:
1409e0def7589a8afa8a6acef13476bb3f882c104b91Chris Lattner  const ObjCInterfaceDecl *getClassInterface() const { return ClassInterface; }
1410e0def7589a8afa8a6acef13476bb3f882c104b91Chris Lattner  ObjCInterfaceDecl *getClassInterface() { return ClassInterface; }
14118a1d722f13df383600f36d77f842957c8adb5f1bArgyrios Kyrtzidis  void setClassInterface(ObjCInterfaceDecl *IFace);
14122c2d43c557beca1b4ba4bd743f33978aecb46a97Douglas Gregor
14131eb4433ac451dc16f4133a88af2d002ac26c58efMike Stump  void addInstanceMethod(ObjCMethodDecl *method) {
14142c2d43c557beca1b4ba4bd743f33978aecb46a97Douglas Gregor    // FIXME: Context should be set correctly before we get here.
1415653f1b1bf293a9bd96fd4dd6372e779cc7af1597Douglas Gregor    method->setLexicalDeclContext(this);
14161eb4433ac451dc16f4133a88af2d002ac26c58efMike Stump    addDecl(method);
1417e1e6c0d5c79c0ee7ed62fef47a19aa7ecef40db4Steve Naroff  }
14181eb4433ac451dc16f4133a88af2d002ac26c58efMike Stump  void addClassMethod(ObjCMethodDecl *method) {
14192c2d43c557beca1b4ba4bd743f33978aecb46a97Douglas Gregor    // FIXME: Context should be set correctly before we get here.
1420653f1b1bf293a9bd96fd4dd6372e779cc7af1597Douglas Gregor    method->setLexicalDeclContext(this);
14211eb4433ac451dc16f4133a88af2d002ac26c58efMike Stump    addDecl(method);
142253df12d1ba68dbd071d067f8236c16fba815aad5Chris Lattner  }
14231eb4433ac451dc16f4133a88af2d002ac26c58efMike Stump
142417945a0f64fe03ff6ec0c2146005a87636e3ac12Argyrios Kyrtzidis  void addPropertyImplementation(ObjCPropertyImplDecl *property);
14251eb4433ac451dc16f4133a88af2d002ac26c58efMike Stump
142617945a0f64fe03ff6ec0c2146005a87636e3ac12Argyrios Kyrtzidis  ObjCPropertyImplDecl *FindPropertyImplDecl(IdentifierInfo *propertyId) const;
142717945a0f64fe03ff6ec0c2146005a87636e3ac12Argyrios Kyrtzidis  ObjCPropertyImplDecl *FindPropertyImplIvarDecl(IdentifierInfo *ivarId) const;
1428653f1b1bf293a9bd96fd4dd6372e779cc7af1597Douglas Gregor
1429653f1b1bf293a9bd96fd4dd6372e779cc7af1597Douglas Gregor  // Iterator access to properties.
1430653f1b1bf293a9bd96fd4dd6372e779cc7af1597Douglas Gregor  typedef specific_decl_iterator<ObjCPropertyImplDecl> propimpl_iterator;
14311eb4433ac451dc16f4133a88af2d002ac26c58efMike Stump  propimpl_iterator propimpl_begin() const {
143217945a0f64fe03ff6ec0c2146005a87636e3ac12Argyrios Kyrtzidis    return propimpl_iterator(decls_begin());
1433559c0c4bbecc017aab0716d546c4fefbcc194687Fariborz Jahanian  }
14341eb4433ac451dc16f4133a88af2d002ac26c58efMike Stump  propimpl_iterator propimpl_end() const {
143517945a0f64fe03ff6ec0c2146005a87636e3ac12Argyrios Kyrtzidis    return propimpl_iterator(decls_end());
1436559c0c4bbecc017aab0716d546c4fefbcc194687Fariborz Jahanian  }
1437653f1b1bf293a9bd96fd4dd6372e779cc7af1597Douglas Gregor
143880cd64a8450d8e2c079dc134d9711cd45ba89d63John McCall  static bool classof(const Decl *D) { return classofKind(D->getKind()); }
1439bfb498d0996ef049efe9476f2802976fd145cd60Argyrios Kyrtzidis  static bool classof(const ObjCImplDecl *D) { return true; }
144080cd64a8450d8e2c079dc134d9711cd45ba89d63John McCall  static bool classofKind(Kind K) {
14419a55591af3e5506b95a9718e15380129fbfc5ebcSean Hunt    return K >= firstObjCImpl && K <= lastObjCImpl;
144280cd64a8450d8e2c079dc134d9711cd45ba89d63John McCall  }
14433aa1861bd8b5121e53379b1a00f9d6ad8dead4f6Chris Lattner};
14441eb4433ac451dc16f4133a88af2d002ac26c58efMike Stump
14451eb4433ac451dc16f4133a88af2d002ac26c58efMike Stump/// ObjCCategoryImplDecl - An object of this class encapsulates a category
14461eb4433ac451dc16f4133a88af2d002ac26c58efMike Stump/// @implementation declaration. If a category class has declaration of a
14471eb4433ac451dc16f4133a88af2d002ac26c58efMike Stump/// property, its implementation must be specified in the category's
14483aa1861bd8b5121e53379b1a00f9d6ad8dead4f6Chris Lattner/// @implementation declaration. Example:
14493aa1861bd8b5121e53379b1a00f9d6ad8dead4f6Chris Lattner/// @interface I @end
14503aa1861bd8b5121e53379b1a00f9d6ad8dead4f6Chris Lattner/// @interface I(CATEGORY)
14513aa1861bd8b5121e53379b1a00f9d6ad8dead4f6Chris Lattner///    @property int p1, d1;
14523aa1861bd8b5121e53379b1a00f9d6ad8dead4f6Chris Lattner/// @end
14533aa1861bd8b5121e53379b1a00f9d6ad8dead4f6Chris Lattner/// @implementation I(CATEGORY)
14543aa1861bd8b5121e53379b1a00f9d6ad8dead4f6Chris Lattner///  @dynamic p1,d1;
14553aa1861bd8b5121e53379b1a00f9d6ad8dead4f6Chris Lattner/// @end
14563aa1861bd8b5121e53379b1a00f9d6ad8dead4f6Chris Lattner///
14573aa1861bd8b5121e53379b1a00f9d6ad8dead4f6Chris Lattner/// ObjCCategoryImplDecl
14583aa1861bd8b5121e53379b1a00f9d6ad8dead4f6Chris Lattnerclass ObjCCategoryImplDecl : public ObjCImplDecl {
145999ba9e3bd70671f3441fb974895f226a83ce0e66David Blaikie  virtual void anchor();
146099ba9e3bd70671f3441fb974895f226a83ce0e66David Blaikie
14613aa1861bd8b5121e53379b1a00f9d6ad8dead4f6Chris Lattner  // Category name
14623aa1861bd8b5121e53379b1a00f9d6ad8dead4f6Chris Lattner  IdentifierInfo *Id;
14633aa1861bd8b5121e53379b1a00f9d6ad8dead4f6Chris Lattner
1464c6994005dc9f677c831b8e90bdab483cc2197c29Argyrios Kyrtzidis  // Category name location
1465c6994005dc9f677c831b8e90bdab483cc2197c29Argyrios Kyrtzidis  SourceLocation CategoryNameLoc;
1466c6994005dc9f677c831b8e90bdab483cc2197c29Argyrios Kyrtzidis
14671711fc91efb36d131f7ba771f73f0154dc1abd1fArgyrios Kyrtzidis  ObjCCategoryImplDecl(DeclContext *DC, IdentifierInfo *Id,
14681711fc91efb36d131f7ba771f73f0154dc1abd1fArgyrios Kyrtzidis                       ObjCInterfaceDecl *classInterface,
1469c6994005dc9f677c831b8e90bdab483cc2197c29Argyrios Kyrtzidis                       SourceLocation nameLoc, SourceLocation atStartLoc,
1470c6994005dc9f677c831b8e90bdab483cc2197c29Argyrios Kyrtzidis                       SourceLocation CategoryNameLoc)
14711711fc91efb36d131f7ba771f73f0154dc1abd1fArgyrios Kyrtzidis    : ObjCImplDecl(ObjCCategoryImpl, DC, classInterface, nameLoc, atStartLoc),
1472c6994005dc9f677c831b8e90bdab483cc2197c29Argyrios Kyrtzidis      Id(Id), CategoryNameLoc(CategoryNameLoc) {}
14733aa1861bd8b5121e53379b1a00f9d6ad8dead4f6Chris Lattnerpublic:
14743aa1861bd8b5121e53379b1a00f9d6ad8dead4f6Chris Lattner  static ObjCCategoryImplDecl *Create(ASTContext &C, DeclContext *DC,
14751711fc91efb36d131f7ba771f73f0154dc1abd1fArgyrios Kyrtzidis                                      IdentifierInfo *Id,
14761711fc91efb36d131f7ba771f73f0154dc1abd1fArgyrios Kyrtzidis                                      ObjCInterfaceDecl *classInterface,
14771711fc91efb36d131f7ba771f73f0154dc1abd1fArgyrios Kyrtzidis                                      SourceLocation nameLoc,
1478c6994005dc9f677c831b8e90bdab483cc2197c29Argyrios Kyrtzidis                                      SourceLocation atStartLoc,
1479c6994005dc9f677c831b8e90bdab483cc2197c29Argyrios Kyrtzidis                                      SourceLocation CategoryNameLoc);
14801eb4433ac451dc16f4133a88af2d002ac26c58efMike Stump
14810d69b8cc8e90a9364771837cb42d7031b4cbb984Steve Naroff  /// getIdentifier - Get the identifier that names the category
14823aa1861bd8b5121e53379b1a00f9d6ad8dead4f6Chris Lattner  /// interface associated with this implementation.
14830d69b8cc8e90a9364771837cb42d7031b4cbb984Steve Naroff  /// FIXME: This is a bad API, we are overriding the NamedDecl::getIdentifier()
14840d69b8cc8e90a9364771837cb42d7031b4cbb984Steve Naroff  /// to mean something different. For example:
1485ba243b59a1074e0962f6abfa3bb9aa984eac1245David Blaikie  /// ((NamedDecl *)SomeCategoryImplDecl)->getIdentifier()
1486ba243b59a1074e0962f6abfa3bb9aa984eac1245David Blaikie  /// returns the class interface name, whereas
1487ba243b59a1074e0962f6abfa3bb9aa984eac1245David Blaikie  /// ((ObjCCategoryImplDecl *)SomeCategoryImplDecl)->getIdentifier()
14880d69b8cc8e90a9364771837cb42d7031b4cbb984Steve Naroff  /// returns the category name.
14891eb4433ac451dc16f4133a88af2d002ac26c58efMike Stump  IdentifierInfo *getIdentifier() const {
14901eb4433ac451dc16f4133a88af2d002ac26c58efMike Stump    return Id;
14913aa1861bd8b5121e53379b1a00f9d6ad8dead4f6Chris Lattner  }
149210b0e1fa3aabd8877dbbc0df1f2414e04afd5fddDouglas Gregor  void setIdentifier(IdentifierInfo *II) { Id = II; }
14931eb4433ac451dc16f4133a88af2d002ac26c58efMike Stump
14940d69b8cc8e90a9364771837cb42d7031b4cbb984Steve Naroff  ObjCCategoryDecl *getCategoryDecl() const;
149510b0e1fa3aabd8877dbbc0df1f2414e04afd5fddDouglas Gregor
1496c6994005dc9f677c831b8e90bdab483cc2197c29Argyrios Kyrtzidis  SourceLocation getCategoryNameLoc() const { return CategoryNameLoc; }
1497c6994005dc9f677c831b8e90bdab483cc2197c29Argyrios Kyrtzidis
1498b5217efa5be2fd6be48d207267c8bcda6bf9206cDaniel Dunbar  /// getName - Get the name of identifier for the class interface associated
1499b5217efa5be2fd6be48d207267c8bcda6bf9206cDaniel Dunbar  /// with this implementation as a StringRef.
1500b5217efa5be2fd6be48d207267c8bcda6bf9206cDaniel Dunbar  //
1501b5217efa5be2fd6be48d207267c8bcda6bf9206cDaniel Dunbar  // FIXME: This is a bad API, we are overriding the NamedDecl::getName, to mean
1502b5217efa5be2fd6be48d207267c8bcda6bf9206cDaniel Dunbar  // something different.
1503686775deca8b8685eb90801495880e3abdd844c2Chris Lattner  StringRef getName() const {
1504b5217efa5be2fd6be48d207267c8bcda6bf9206cDaniel Dunbar    return Id ? Id->getNameStart() : "";
1505b5217efa5be2fd6be48d207267c8bcda6bf9206cDaniel Dunbar  }
1506b5217efa5be2fd6be48d207267c8bcda6bf9206cDaniel Dunbar
15073aa1861bd8b5121e53379b1a00f9d6ad8dead4f6Chris Lattner  /// getNameAsCString - Get the name of identifier for the class
15083aa1861bd8b5121e53379b1a00f9d6ad8dead4f6Chris Lattner  /// interface associated with this implementation as a C string
15093aa1861bd8b5121e53379b1a00f9d6ad8dead4f6Chris Lattner  /// (const char*).
15107fe60650c1133ee74a3395cf1063690e274bb7acDaniel Dunbar  //
1511b5217efa5be2fd6be48d207267c8bcda6bf9206cDaniel Dunbar  // FIXME: Deprecated, move clients to getName().
15123aa1861bd8b5121e53379b1a00f9d6ad8dead4f6Chris Lattner  const char *getNameAsCString() const {
15137fe60650c1133ee74a3395cf1063690e274bb7acDaniel Dunbar    return Id ? Id->getNameStart() : "";
15143aa1861bd8b5121e53379b1a00f9d6ad8dead4f6Chris Lattner  }
15151eb4433ac451dc16f4133a88af2d002ac26c58efMike Stump
15163aa1861bd8b5121e53379b1a00f9d6ad8dead4f6Chris Lattner  /// @brief Get the name of the class associated with this interface.
15177fe60650c1133ee74a3395cf1063690e274bb7acDaniel Dunbar  //
1518b5217efa5be2fd6be48d207267c8bcda6bf9206cDaniel Dunbar  // FIXME: Deprecated, move clients to getName().
15193aa1861bd8b5121e53379b1a00f9d6ad8dead4f6Chris Lattner  std::string getNameAsString() const {
1520b5217efa5be2fd6be48d207267c8bcda6bf9206cDaniel Dunbar    return getName();
15213aa1861bd8b5121e53379b1a00f9d6ad8dead4f6Chris Lattner  }
15221eb4433ac451dc16f4133a88af2d002ac26c58efMike Stump
152380cd64a8450d8e2c079dc134d9711cd45ba89d63John McCall  static bool classof(const Decl *D) { return classofKind(D->getKind()); }
1524a526c5c67e5a0473c340903ee542ce570119665fTed Kremenek  static bool classof(const ObjCCategoryImplDecl *D) { return true; }
152580cd64a8450d8e2c079dc134d9711cd45ba89d63John McCall  static bool classofKind(Kind K) { return K == ObjCCategoryImpl;}
1526c6994005dc9f677c831b8e90bdab483cc2197c29Argyrios Kyrtzidis
1527c6994005dc9f677c831b8e90bdab483cc2197c29Argyrios Kyrtzidis  friend class ASTDeclReader;
1528c6994005dc9f677c831b8e90bdab483cc2197c29Argyrios Kyrtzidis  friend class ASTDeclWriter;
15298f3fde00ad4d4f943321e338b914ae4740711c84Fariborz Jahanian};
15308f3fde00ad4d4f943321e338b914ae4740711c84Fariborz Jahanian
15318cc488fefb2fb04bc8d5398da29f0182f97934cfChris Lattnerraw_ostream &operator<<(raw_ostream &OS,
1532900fc6388e803868a34b9483510c345e9b49d7ebBenjamin Kramer                              const ObjCCategoryImplDecl *CID);
1533900fc6388e803868a34b9483510c345e9b49d7ebBenjamin Kramer
1534a526c5c67e5a0473c340903ee542ce570119665fTed Kremenek/// ObjCImplementationDecl - Represents a class definition - this is where
15350c6b6243d3efd958c17943130e2a773653511edcSteve Naroff/// method definitions are specified. For example:
15360c6b6243d3efd958c17943130e2a773653511edcSteve Naroff///
153798abf4bd3526a00a0e5cf71a9462c181f97b1c81Fariborz Jahanian/// @code
15380c6b6243d3efd958c17943130e2a773653511edcSteve Naroff/// @implementation MyClass
15390c6b6243d3efd958c17943130e2a773653511edcSteve Naroff/// - (void)myMethod { /* do something */ }
15400c6b6243d3efd958c17943130e2a773653511edcSteve Naroff/// @end
154198abf4bd3526a00a0e5cf71a9462c181f97b1c81Fariborz Jahanian/// @endcode
15420c6b6243d3efd958c17943130e2a773653511edcSteve Naroff///
15431eb4433ac451dc16f4133a88af2d002ac26c58efMike Stump/// Typically, instance variables are specified in the class interface,
1544ec0d7a6f4b0699cc9960e6d9fee0f957c64d1cf9Douglas Gregor/// *not* in the implementation. Nevertheless (for legacy reasons), we
154553df12d1ba68dbd071d067f8236c16fba815aad5Chris Lattner/// allow instance variables to be specified in the implementation.  When
154653df12d1ba68dbd071d067f8236c16fba815aad5Chris Lattner/// specified, they need to be *identical* to the interface.
15470c6b6243d3efd958c17943130e2a773653511edcSteve Naroff///
15481eb4433ac451dc16f4133a88af2d002ac26c58efMike Stumpclass ObjCImplementationDecl : public ObjCImplDecl {
154999ba9e3bd70671f3441fb974895f226a83ce0e66David Blaikie  virtual void anchor();
1550980e508ca70d6de75d2abfd96b4681fc98bb2698Steve Naroff  /// Implementation Class's super class.
1551a526c5c67e5a0473c340903ee542ce570119665fTed Kremenek  ObjCInterfaceDecl *SuperClass;
1552e4498c6d66a8f472cba29b6158a2e86dfc60d0efFariborz Jahanian  /// Support for ivar initialization.
1553e4498c6d66a8f472cba29b6158a2e86dfc60d0efFariborz Jahanian  /// IvarInitializers - The arguments used to initialize the ivars
1554cbb67480094b3bcb5b715acd827cbad55e2a204cSean Hunt  CXXCtorInitializer **IvarInitializers;
1555e4498c6d66a8f472cba29b6158a2e86dfc60d0efFariborz Jahanian  unsigned NumIvarInitializers;
1556f85e193739c953358c865005855253af4f68a497John McCall
1557f85e193739c953358c865005855253af4f68a497John McCall  /// true if class has a .cxx_[construct,destruct] method.
1558f85e193739c953358c865005855253af4f68a497John McCall  bool HasCXXStructors : 1;
1559ba243b59a1074e0962f6abfa3bb9aa984eac1245David Blaikie
1560000835d0b04345c0014c603fe6339b3bc154050eFariborz Jahanian  /// true of class extension has at least one bitfield ivar.
1561000835d0b04345c0014c603fe6339b3bc154050eFariborz Jahanian  bool HasSynthBitfield : 1;
1562ba243b59a1074e0962f6abfa3bb9aa984eac1245David Blaikie
15631711fc91efb36d131f7ba771f73f0154dc1abd1fArgyrios Kyrtzidis  ObjCImplementationDecl(DeclContext *DC,
1564a526c5c67e5a0473c340903ee542ce570119665fTed Kremenek                         ObjCInterfaceDecl *classInterface,
15651711fc91efb36d131f7ba771f73f0154dc1abd1fArgyrios Kyrtzidis                         ObjCInterfaceDecl *superDecl,
15661711fc91efb36d131f7ba771f73f0154dc1abd1fArgyrios Kyrtzidis                         SourceLocation nameLoc, SourceLocation atStartLoc)
15671711fc91efb36d131f7ba771f73f0154dc1abd1fArgyrios Kyrtzidis    : ObjCImplDecl(ObjCImplementation, DC, classInterface, nameLoc, atStartLoc),
1568000835d0b04345c0014c603fe6339b3bc154050eFariborz Jahanian       SuperClass(superDecl), IvarInitializers(0), NumIvarInitializers(0),
1569f85e193739c953358c865005855253af4f68a497John McCall       HasCXXStructors(false), HasSynthBitfield(false) {}
15701eb4433ac451dc16f4133a88af2d002ac26c58efMike Stumppublic:
15711eb4433ac451dc16f4133a88af2d002ac26c58efMike Stump  static ObjCImplementationDecl *Create(ASTContext &C, DeclContext *DC,
157275c9cae5f85c72cbb1649e93849e16ede3f07522Chris Lattner                                        ObjCInterfaceDecl *classInterface,
15731711fc91efb36d131f7ba771f73f0154dc1abd1fArgyrios Kyrtzidis                                        ObjCInterfaceDecl *superDecl,
15741711fc91efb36d131f7ba771f73f0154dc1abd1fArgyrios Kyrtzidis                                        SourceLocation nameLoc,
15751711fc91efb36d131f7ba771f73f0154dc1abd1fArgyrios Kyrtzidis                                        SourceLocation atStartLoc);
1576ba243b59a1074e0962f6abfa3bb9aa984eac1245David Blaikie
1577e4498c6d66a8f472cba29b6158a2e86dfc60d0efFariborz Jahanian  /// init_iterator - Iterates through the ivar initializer list.
1578cbb67480094b3bcb5b715acd827cbad55e2a204cSean Hunt  typedef CXXCtorInitializer **init_iterator;
1579ba243b59a1074e0962f6abfa3bb9aa984eac1245David Blaikie
1580e4498c6d66a8f472cba29b6158a2e86dfc60d0efFariborz Jahanian  /// init_const_iterator - Iterates through the ivar initializer list.
1581cbb67480094b3bcb5b715acd827cbad55e2a204cSean Hunt  typedef CXXCtorInitializer * const * init_const_iterator;
1582ba243b59a1074e0962f6abfa3bb9aa984eac1245David Blaikie
1583e4498c6d66a8f472cba29b6158a2e86dfc60d0efFariborz Jahanian  /// init_begin() - Retrieve an iterator to the first initializer.
1584e4498c6d66a8f472cba29b6158a2e86dfc60d0efFariborz Jahanian  init_iterator       init_begin()       { return IvarInitializers; }
1585e4498c6d66a8f472cba29b6158a2e86dfc60d0efFariborz Jahanian  /// begin() - Retrieve an iterator to the first initializer.
1586e4498c6d66a8f472cba29b6158a2e86dfc60d0efFariborz Jahanian  init_const_iterator init_begin() const { return IvarInitializers; }
1587ba243b59a1074e0962f6abfa3bb9aa984eac1245David Blaikie
1588e4498c6d66a8f472cba29b6158a2e86dfc60d0efFariborz Jahanian  /// init_end() - Retrieve an iterator past the last initializer.
1589e4498c6d66a8f472cba29b6158a2e86dfc60d0efFariborz Jahanian  init_iterator       init_end()       {
1590e4498c6d66a8f472cba29b6158a2e86dfc60d0efFariborz Jahanian    return IvarInitializers + NumIvarInitializers;
1591e4498c6d66a8f472cba29b6158a2e86dfc60d0efFariborz Jahanian  }
1592e4498c6d66a8f472cba29b6158a2e86dfc60d0efFariborz Jahanian  /// end() - Retrieve an iterator past the last initializer.
1593e4498c6d66a8f472cba29b6158a2e86dfc60d0efFariborz Jahanian  init_const_iterator init_end() const {
1594e4498c6d66a8f472cba29b6158a2e86dfc60d0efFariborz Jahanian    return IvarInitializers + NumIvarInitializers;
1595e4498c6d66a8f472cba29b6158a2e86dfc60d0efFariborz Jahanian  }
1596e4498c6d66a8f472cba29b6158a2e86dfc60d0efFariborz Jahanian  /// getNumArgs - Number of ivars which must be initialized.
1597e4498c6d66a8f472cba29b6158a2e86dfc60d0efFariborz Jahanian  unsigned getNumIvarInitializers() const {
1598e4498c6d66a8f472cba29b6158a2e86dfc60d0efFariborz Jahanian    return NumIvarInitializers;
1599e4498c6d66a8f472cba29b6158a2e86dfc60d0efFariborz Jahanian  }
1600ba243b59a1074e0962f6abfa3bb9aa984eac1245David Blaikie
1601e4498c6d66a8f472cba29b6158a2e86dfc60d0efFariborz Jahanian  void setNumIvarInitializers(unsigned numNumIvarInitializers) {
1602e4498c6d66a8f472cba29b6158a2e86dfc60d0efFariborz Jahanian    NumIvarInitializers = numNumIvarInitializers;
1603e4498c6d66a8f472cba29b6158a2e86dfc60d0efFariborz Jahanian  }
1604ba243b59a1074e0962f6abfa3bb9aa984eac1245David Blaikie
1605e4498c6d66a8f472cba29b6158a2e86dfc60d0efFariborz Jahanian  void setIvarInitializers(ASTContext &C,
1606cbb67480094b3bcb5b715acd827cbad55e2a204cSean Hunt                           CXXCtorInitializer ** initializers,
1607e4498c6d66a8f472cba29b6158a2e86dfc60d0efFariborz Jahanian                           unsigned numInitializers);
1608f85e193739c953358c865005855253af4f68a497John McCall
1609f85e193739c953358c865005855253af4f68a497John McCall  bool hasCXXStructors() const { return HasCXXStructors; }
1610f85e193739c953358c865005855253af4f68a497John McCall  void setHasCXXStructors(bool val) { HasCXXStructors = val; }
1611ba243b59a1074e0962f6abfa3bb9aa984eac1245David Blaikie
1612000835d0b04345c0014c603fe6339b3bc154050eFariborz Jahanian  bool hasSynthBitfield() const { return HasSynthBitfield; }
1613000835d0b04345c0014c603fe6339b3bc154050eFariborz Jahanian  void setHasSynthBitfield (bool val) { HasSynthBitfield = val; }
1614ba243b59a1074e0962f6abfa3bb9aa984eac1245David Blaikie
16154afa39deaa245592977136d367251ee2c173dd8dDouglas Gregor  /// getIdentifier - Get the identifier that names the class
16164afa39deaa245592977136d367251ee2c173dd8dDouglas Gregor  /// interface associated with this implementation.
16171eb4433ac451dc16f4133a88af2d002ac26c58efMike Stump  IdentifierInfo *getIdentifier() const {
16181eb4433ac451dc16f4133a88af2d002ac26c58efMike Stump    return getClassInterface()->getIdentifier();
16194afa39deaa245592977136d367251ee2c173dd8dDouglas Gregor  }
16204afa39deaa245592977136d367251ee2c173dd8dDouglas Gregor
1621d0c10e20d5ba8c1a8a077db128c03eddc3158673Daniel Dunbar  /// getName - Get the name of identifier for the class interface associated
1622d0c10e20d5ba8c1a8a077db128c03eddc3158673Daniel Dunbar  /// with this implementation as a StringRef.
1623d0c10e20d5ba8c1a8a077db128c03eddc3158673Daniel Dunbar  //
1624d0c10e20d5ba8c1a8a077db128c03eddc3158673Daniel Dunbar  // FIXME: This is a bad API, we are overriding the NamedDecl::getName, to mean
1625d0c10e20d5ba8c1a8a077db128c03eddc3158673Daniel Dunbar  // something different.
1626686775deca8b8685eb90801495880e3abdd844c2Chris Lattner  StringRef getName() const {
1627d0c10e20d5ba8c1a8a077db128c03eddc3158673Daniel Dunbar    assert(getIdentifier() && "Name is not a simple identifier");
1628d0c10e20d5ba8c1a8a077db128c03eddc3158673Daniel Dunbar    return getIdentifier()->getName();
1629d0c10e20d5ba8c1a8a077db128c03eddc3158673Daniel Dunbar  }
1630d0c10e20d5ba8c1a8a077db128c03eddc3158673Daniel Dunbar
16314afa39deaa245592977136d367251ee2c173dd8dDouglas Gregor  /// getNameAsCString - Get the name of identifier for the class
16324afa39deaa245592977136d367251ee2c173dd8dDouglas Gregor  /// interface associated with this implementation as a C string
16334afa39deaa245592977136d367251ee2c173dd8dDouglas Gregor  /// (const char*).
16347fe60650c1133ee74a3395cf1063690e274bb7acDaniel Dunbar  //
16357fe60650c1133ee74a3395cf1063690e274bb7acDaniel Dunbar  // FIXME: Move to StringRef API.
16364afa39deaa245592977136d367251ee2c173dd8dDouglas Gregor  const char *getNameAsCString() const {
1637d0c10e20d5ba8c1a8a077db128c03eddc3158673Daniel Dunbar    return getName().data();
16384afa39deaa245592977136d367251ee2c173dd8dDouglas Gregor  }
16394afa39deaa245592977136d367251ee2c173dd8dDouglas Gregor
16404afa39deaa245592977136d367251ee2c173dd8dDouglas Gregor  /// @brief Get the name of the class associated with this interface.
16417fe60650c1133ee74a3395cf1063690e274bb7acDaniel Dunbar  //
16427fe60650c1133ee74a3395cf1063690e274bb7acDaniel Dunbar  // FIXME: Move to StringRef API.
16434afa39deaa245592977136d367251ee2c173dd8dDouglas Gregor  std::string getNameAsString() const {
1644d0c10e20d5ba8c1a8a077db128c03eddc3158673Daniel Dunbar    return getName();
16454afa39deaa245592977136d367251ee2c173dd8dDouglas Gregor  }
16464afa39deaa245592977136d367251ee2c173dd8dDouglas Gregor
1647e0def7589a8afa8a6acef13476bb3f882c104b91Chris Lattner  const ObjCInterfaceDecl *getSuperClass() const { return SuperClass; }
1648e0def7589a8afa8a6acef13476bb3f882c104b91Chris Lattner  ObjCInterfaceDecl *getSuperClass() { return SuperClass; }
16491eb4433ac451dc16f4133a88af2d002ac26c58efMike Stump
1650f3a7af901231535191d14cb524d58f22907ac4d2Chris Lattner  void setSuperClass(ObjCInterfaceDecl * superCls) { SuperClass = superCls; }
16511eb4433ac451dc16f4133a88af2d002ac26c58efMike Stump
16528f36aba016c2d236a90f9ecf0a66904209202202Douglas Gregor  typedef specific_decl_iterator<ObjCIvarDecl> ivar_iterator;
16531eb4433ac451dc16f4133a88af2d002ac26c58efMike Stump  ivar_iterator ivar_begin() const {
16541eb4433ac451dc16f4133a88af2d002ac26c58efMike Stump    return ivar_iterator(decls_begin());
16558f36aba016c2d236a90f9ecf0a66904209202202Douglas Gregor  }
16561eb4433ac451dc16f4133a88af2d002ac26c58efMike Stump  ivar_iterator ivar_end() const {
165717945a0f64fe03ff6ec0c2146005a87636e3ac12Argyrios Kyrtzidis    return ivar_iterator(decls_end());
16588f36aba016c2d236a90f9ecf0a66904209202202Douglas Gregor  }
16591eb4433ac451dc16f4133a88af2d002ac26c58efMike Stump  unsigned ivar_size() const {
166017945a0f64fe03ff6ec0c2146005a87636e3ac12Argyrios Kyrtzidis    return std::distance(ivar_begin(), ivar_end());
16618f36aba016c2d236a90f9ecf0a66904209202202Douglas Gregor  }
16621eb4433ac451dc16f4133a88af2d002ac26c58efMike Stump  bool ivar_empty() const {
166317945a0f64fe03ff6ec0c2146005a87636e3ac12Argyrios Kyrtzidis    return ivar_begin() == ivar_end();
16648f36aba016c2d236a90f9ecf0a66904209202202Douglas Gregor  }
16651eb4433ac451dc16f4133a88af2d002ac26c58efMike Stump
166680cd64a8450d8e2c079dc134d9711cd45ba89d63John McCall  static bool classof(const Decl *D) { return classofKind(D->getKind()); }
1667a526c5c67e5a0473c340903ee542ce570119665fTed Kremenek  static bool classof(const ObjCImplementationDecl *D) { return true; }
166880cd64a8450d8e2c079dc134d9711cd45ba89d63John McCall  static bool classofKind(Kind K) { return K == ObjCImplementation; }
16699d50c0635fb213b2a1857e3f8488580f0dab2f98Argyrios Kyrtzidis
1670d527cc06d78fe5afa5f20105b51697637eb02c56Sebastian Redl  friend class ASTDeclReader;
16713397c5570369f19b2d6c52e898f708d75ceede1fSebastian Redl  friend class ASTDeclWriter;
1672980e508ca70d6de75d2abfd96b4681fc98bb2698Steve Naroff};
1673243b64b0001172405ff803c61bdcaa8e98ec1552Fariborz Jahanian
16748cc488fefb2fb04bc8d5398da29f0182f97934cfChris Lattnerraw_ostream &operator<<(raw_ostream &OS,
1675900fc6388e803868a34b9483510c345e9b49d7ebBenjamin Kramer                              const ObjCImplementationDecl *ID);
1676900fc6388e803868a34b9483510c345e9b49d7ebBenjamin Kramer
16771eb4433ac451dc16f4133a88af2d002ac26c58efMike Stump/// ObjCCompatibleAliasDecl - Represents alias of a class. This alias is
1678243b64b0001172405ff803c61bdcaa8e98ec1552Fariborz Jahanian/// declared as @compatibility_alias alias class.
16794afa39deaa245592977136d367251ee2c173dd8dDouglas Gregorclass ObjCCompatibleAliasDecl : public NamedDecl {
168099ba9e3bd70671f3441fb974895f226a83ce0e66David Blaikie  virtual void anchor();
1681243b64b0001172405ff803c61bdcaa8e98ec1552Fariborz Jahanian  /// Class that this is an alias of.
1682a526c5c67e5a0473c340903ee542ce570119665fTed Kremenek  ObjCInterfaceDecl *AliasedClass;
16831eb4433ac451dc16f4133a88af2d002ac26c58efMike Stump
1684d04341000d35c8808a72838b057eed7bf13b7661Douglas Gregor  ObjCCompatibleAliasDecl(DeclContext *DC, SourceLocation L, IdentifierInfo *Id,
1685e8043c39176e7f253fbd92982b077eca6bf2fd59Steve Naroff                          ObjCInterfaceDecl* aliasedClass)
16864afa39deaa245592977136d367251ee2c173dd8dDouglas Gregor    : NamedDecl(ObjCCompatibleAlias, DC, L, Id), AliasedClass(aliasedClass) {}
1687f8d17a59167d9c2026506ed8813ea434d93b662aChris Lattnerpublic:
1688d04341000d35c8808a72838b057eed7bf13b7661Douglas Gregor  static ObjCCompatibleAliasDecl *Create(ASTContext &C, DeclContext *DC,
16890ed844b04ea4387caa4e1cf3dc375d269657536bChris Lattner                                         SourceLocation L, IdentifierInfo *Id,
1690f8d17a59167d9c2026506ed8813ea434d93b662aChris Lattner                                         ObjCInterfaceDecl* aliasedClass);
1691f8d17a59167d9c2026506ed8813ea434d93b662aChris Lattner
1692f8d17a59167d9c2026506ed8813ea434d93b662aChris Lattner  const ObjCInterfaceDecl *getClassInterface() const { return AliasedClass; }
1693f8d17a59167d9c2026506ed8813ea434d93b662aChris Lattner  ObjCInterfaceDecl *getClassInterface() { return AliasedClass; }
169430833f8d77c08f8f16371776fde85a9fde3d9b6eSteve Naroff  void setClassInterface(ObjCInterfaceDecl *D) { AliasedClass = D; }
16951eb4433ac451dc16f4133a88af2d002ac26c58efMike Stump
169680cd64a8450d8e2c079dc134d9711cd45ba89d63John McCall  static bool classof(const Decl *D) { return classofKind(D->getKind()); }
1697a526c5c67e5a0473c340903ee542ce570119665fTed Kremenek  static bool classof(const ObjCCompatibleAliasDecl *D) { return true; }
169880cd64a8450d8e2c079dc134d9711cd45ba89d63John McCall  static bool classofKind(Kind K) { return K == ObjCCompatibleAlias; }
16991eb4433ac451dc16f4133a88af2d002ac26c58efMike Stump
1700243b64b0001172405ff803c61bdcaa8e98ec1552Fariborz Jahanian};
17011de1e74541c25c1a7b721f1c3991ea34c8403420Fariborz Jahanian
17021de1e74541c25c1a7b721f1c3991ea34c8403420Fariborz Jahanian/// ObjCPropertyDecl - Represents one property declaration in an interface.
17031de1e74541c25c1a7b721f1c3991ea34c8403420Fariborz Jahanian/// For example:
17041de1e74541c25c1a7b721f1c3991ea34c8403420Fariborz Jahanian/// @property (assign, readwrite) int MyProperty;
17051de1e74541c25c1a7b721f1c3991ea34c8403420Fariborz Jahanian///
17064afa39deaa245592977136d367251ee2c173dd8dDouglas Gregorclass ObjCPropertyDecl : public NamedDecl {
170799ba9e3bd70671f3441fb974895f226a83ce0e66David Blaikie  virtual void anchor();
170882a5fe3d1cf204b672cdab24d72275b6ad2c3527Fariborz Jahanianpublic:
1709a5674258f5e6f74f1c0ed3ece4d64fbb1b9afb11Chris Lattner  enum PropertyAttributeKind {
17101eb4433ac451dc16f4133a88af2d002ac26c58efMike Stump    OBJC_PR_noattr    = 0x00,
17111eb4433ac451dc16f4133a88af2d002ac26c58efMike Stump    OBJC_PR_readonly  = 0x01,
1712a5674258f5e6f74f1c0ed3ece4d64fbb1b9afb11Chris Lattner    OBJC_PR_getter    = 0x02,
17131eb4433ac451dc16f4133a88af2d002ac26c58efMike Stump    OBJC_PR_assign    = 0x04,
17141eb4433ac451dc16f4133a88af2d002ac26c58efMike Stump    OBJC_PR_readwrite = 0x08,
1715a5674258f5e6f74f1c0ed3ece4d64fbb1b9afb11Chris Lattner    OBJC_PR_retain    = 0x10,
17161eb4433ac451dc16f4133a88af2d002ac26c58efMike Stump    OBJC_PR_copy      = 0x20,
1717a5674258f5e6f74f1c0ed3ece4d64fbb1b9afb11Chris Lattner    OBJC_PR_nonatomic = 0x40,
171845937ae10a0f70f74508165aab4f2b63e18ea747Fariborz Jahanian    OBJC_PR_setter    = 0x80,
1719f85e193739c953358c865005855253af4f68a497John McCall    OBJC_PR_atomic    = 0x100,
1720f85e193739c953358c865005855253af4f68a497John McCall    OBJC_PR_weak      = 0x200,
1721f85e193739c953358c865005855253af4f68a497John McCall    OBJC_PR_strong    = 0x400,
17229f3480bab8fcb6547978f8ad9e2b8f468e3658eeArgyrios Kyrtzidis    OBJC_PR_unsafe_unretained = 0x800
17239f3480bab8fcb6547978f8ad9e2b8f468e3658eeArgyrios Kyrtzidis    // Adding a property should change NumPropertyAttrsBits
17249f3480bab8fcb6547978f8ad9e2b8f468e3658eeArgyrios Kyrtzidis  };
17250a68dc7f04be1542ce249ff4adb169453698ad91Argyrios Kyrtzidis
17269f3480bab8fcb6547978f8ad9e2b8f468e3658eeArgyrios Kyrtzidis  enum {
17270a68dc7f04be1542ce249ff4adb169453698ad91Argyrios Kyrtzidis    /// \brief Number of bits fitting all the property attributes.
17289f3480bab8fcb6547978f8ad9e2b8f468e3658eeArgyrios Kyrtzidis    NumPropertyAttrsBits = 12
1729a5674258f5e6f74f1c0ed3ece4d64fbb1b9afb11Chris Lattner  };
1730af05bb9073319d8381b71c4325188853fd4b8ed6Daniel Dunbar
17313a02b44e3948f7762dbfba94b7961281ca29d022Fariborz Jahanian  enum SetterKind { Assign, Retain, Copy, Weak };
173246b55e56d029aec699fc2701e43d70264da9ecd8Fariborz Jahanian  enum PropertyControl { None, Required, Optional };
173382a5fe3d1cf204b672cdab24d72275b6ad2c3527Fariborz Jahanianprivate:
1734d8265b838360578032020757d9a2a84c86457edcFariborz Jahanian  SourceLocation AtLoc;   // location of @property
173583a230c83a54190366138c1a4f4310ef838b88fcJohn McCall  TypeSourceInfo *DeclType;
17369f3480bab8fcb6547978f8ad9e2b8f468e3658eeArgyrios Kyrtzidis  unsigned PropertyAttributes : NumPropertyAttrsBits;
17379f3480bab8fcb6547978f8ad9e2b8f468e3658eeArgyrios Kyrtzidis  unsigned PropertyAttributesAsWritten : NumPropertyAttrsBits;
173846b55e56d029aec699fc2701e43d70264da9ecd8Fariborz Jahanian  // @required/@optional
173946b55e56d029aec699fc2701e43d70264da9ecd8Fariborz Jahanian  unsigned PropertyImplementation : 2;
17401eb4433ac451dc16f4133a88af2d002ac26c58efMike Stump
17415251e130a23d997f7c0dfdc250cdc41f179e5bedFariborz Jahanian  Selector GetterName;    // getter name of NULL if no getter
17425251e130a23d997f7c0dfdc250cdc41f179e5bedFariborz Jahanian  Selector SetterName;    // setter name of NULL if no setter
17431eb4433ac451dc16f4133a88af2d002ac26c58efMike Stump
174433de3f0333ca0b5274291b8d76c86758c0484691Fariborz Jahanian  ObjCMethodDecl *GetterMethodDecl; // Declaration of getter instance method
174533de3f0333ca0b5274291b8d76c86758c0484691Fariborz Jahanian  ObjCMethodDecl *SetterMethodDecl; // Declaration of setter instance method
1746af3e72285238369c2ea4ebd40a1c9a87bd3eabb7Fariborz Jahanian  ObjCIvarDecl *PropertyIvarDecl;   // Synthesize ivar for this property
174733de3f0333ca0b5274291b8d76c86758c0484691Fariborz Jahanian
17481eb4433ac451dc16f4133a88af2d002ac26c58efMike Stump  ObjCPropertyDecl(DeclContext *DC, SourceLocation L, IdentifierInfo *Id,
174983a230c83a54190366138c1a4f4310ef838b88fcJohn McCall                   SourceLocation AtLocation, TypeSourceInfo *T)
1750d0502407c1b41b2ace326f355d7b7a6876246223Fariborz Jahanian    : NamedDecl(ObjCProperty, DC, L, Id), AtLoc(AtLocation), DeclType(T),
1751ba243b59a1074e0962f6abfa3bb9aa984eac1245David Blaikie      PropertyAttributes(OBJC_PR_noattr),
175280aa1cd7973561889e51c1c152c8990a8de9c953Fariborz Jahanian      PropertyAttributesAsWritten(OBJC_PR_noattr),
175380aa1cd7973561889e51c1c152c8990a8de9c953Fariborz Jahanian      PropertyImplementation(None),
17541eb4433ac451dc16f4133a88af2d002ac26c58efMike Stump      GetterName(Selector()),
175533de3f0333ca0b5274291b8d76c86758c0484691Fariborz Jahanian      SetterName(Selector()),
1756af3e72285238369c2ea4ebd40a1c9a87bd3eabb7Fariborz Jahanian      GetterMethodDecl(0), SetterMethodDecl(0) , PropertyIvarDecl(0) {}
1757f8d17a59167d9c2026506ed8813ea434d93b662aChris Lattnerpublic:
17581eb4433ac451dc16f4133a88af2d002ac26c58efMike Stump  static ObjCPropertyDecl *Create(ASTContext &C, DeclContext *DC,
17591eb4433ac451dc16f4133a88af2d002ac26c58efMike Stump                                  SourceLocation L,
1760d0502407c1b41b2ace326f355d7b7a6876246223Fariborz Jahanian                                  IdentifierInfo *Id, SourceLocation AtLocation,
176183a230c83a54190366138c1a4f4310ef838b88fcJohn McCall                                  TypeSourceInfo *T,
176246b55e56d029aec699fc2701e43d70264da9ecd8Fariborz Jahanian                                  PropertyControl propControl = None);
1763d0502407c1b41b2ace326f355d7b7a6876246223Fariborz Jahanian  SourceLocation getAtLoc() const { return AtLoc; }
1764d0502407c1b41b2ace326f355d7b7a6876246223Fariborz Jahanian  void setAtLoc(SourceLocation L) { AtLoc = L; }
1765ba243b59a1074e0962f6abfa3bb9aa984eac1245David Blaikie
176683a230c83a54190366138c1a4f4310ef838b88fcJohn McCall  TypeSourceInfo *getTypeSourceInfo() const { return DeclType; }
176783a230c83a54190366138c1a4f4310ef838b88fcJohn McCall  QualType getType() const { return DeclType->getType(); }
176883a230c83a54190366138c1a4f4310ef838b88fcJohn McCall  void setType(TypeSourceInfo *T) { DeclType = T; }
176970e5a14c6076d63833c62d1d6d628c26309897c1Douglas Gregor
1770a5674258f5e6f74f1c0ed3ece4d64fbb1b9afb11Chris Lattner  PropertyAttributeKind getPropertyAttributes() const {
1771f4af5154571e0c5eadb19df10e65464766ef6683Chris Lattner    return PropertyAttributeKind(PropertyAttributes);
1772f4af5154571e0c5eadb19df10e65464766ef6683Chris Lattner  }
17731eb4433ac451dc16f4133a88af2d002ac26c58efMike Stump  void setPropertyAttributes(PropertyAttributeKind PRVal) {
1774a5674258f5e6f74f1c0ed3ece4d64fbb1b9afb11Chris Lattner    PropertyAttributes |= PRVal;
177582a5fe3d1cf204b672cdab24d72275b6ad2c3527Fariborz Jahanian  }
1776394d33f1f602f7681032a659dff5bb09061ee510Daniel Dunbar
177780aa1cd7973561889e51c1c152c8990a8de9c953Fariborz Jahanian  PropertyAttributeKind getPropertyAttributesAsWritten() const {
177880aa1cd7973561889e51c1c152c8990a8de9c953Fariborz Jahanian    return PropertyAttributeKind(PropertyAttributesAsWritten);
177980aa1cd7973561889e51c1c152c8990a8de9c953Fariborz Jahanian  }
17800a68dc7f04be1542ce249ff4adb169453698ad91Argyrios Kyrtzidis
17810a68dc7f04be1542ce249ff4adb169453698ad91Argyrios Kyrtzidis  bool hasWrittenStorageAttribute() const {
17820a68dc7f04be1542ce249ff4adb169453698ad91Argyrios Kyrtzidis    return PropertyAttributesAsWritten & (OBJC_PR_assign | OBJC_PR_copy |
17830a68dc7f04be1542ce249ff4adb169453698ad91Argyrios Kyrtzidis        OBJC_PR_unsafe_unretained | OBJC_PR_retain | OBJC_PR_strong |
17840a68dc7f04be1542ce249ff4adb169453698ad91Argyrios Kyrtzidis        OBJC_PR_weak);
17850a68dc7f04be1542ce249ff4adb169453698ad91Argyrios Kyrtzidis  }
1786ba243b59a1074e0962f6abfa3bb9aa984eac1245David Blaikie
178780aa1cd7973561889e51c1c152c8990a8de9c953Fariborz Jahanian  void setPropertyAttributesAsWritten(PropertyAttributeKind PRVal) {
178880aa1cd7973561889e51c1c152c8990a8de9c953Fariborz Jahanian    PropertyAttributesAsWritten = PRVal;
178980aa1cd7973561889e51c1c152c8990a8de9c953Fariborz Jahanian  }
1790ba243b59a1074e0962f6abfa3bb9aa984eac1245David Blaikie
17918cf0bb3c2a798ce3acacaac2d3178648cd4c65c6Fariborz Jahanian void makeitReadWriteAttribute(void) {
17928cf0bb3c2a798ce3acacaac2d3178648cd4c65c6Fariborz Jahanian    PropertyAttributes &= ~OBJC_PR_readonly;
17938cf0bb3c2a798ce3acacaac2d3178648cd4c65c6Fariborz Jahanian    PropertyAttributes |= OBJC_PR_readwrite;
17941eb4433ac451dc16f4133a88af2d002ac26c58efMike Stump }
17958cf0bb3c2a798ce3acacaac2d3178648cd4c65c6Fariborz Jahanian
1796af05bb9073319d8381b71c4325188853fd4b8ed6Daniel Dunbar  // Helper methods for accessing attributes.
1797af05bb9073319d8381b71c4325188853fd4b8ed6Daniel Dunbar
1798af05bb9073319d8381b71c4325188853fd4b8ed6Daniel Dunbar  /// isReadOnly - Return true iff the property has a setter.
1799394d33f1f602f7681032a659dff5bb09061ee510Daniel Dunbar  bool isReadOnly() const {
1800394d33f1f602f7681032a659dff5bb09061ee510Daniel Dunbar    return (PropertyAttributes & OBJC_PR_readonly);
1801394d33f1f602f7681032a659dff5bb09061ee510Daniel Dunbar  }
1802af05bb9073319d8381b71c4325188853fd4b8ed6Daniel Dunbar
1803265941bc308d65cc270d5c4de5806f37ce405606John McCall  /// isAtomic - Return true if the property is atomic.
1804265941bc308d65cc270d5c4de5806f37ce405606John McCall  bool isAtomic() const {
1805265941bc308d65cc270d5c4de5806f37ce405606John McCall    return (PropertyAttributes & OBJC_PR_atomic);
1806265941bc308d65cc270d5c4de5806f37ce405606John McCall  }
1807265941bc308d65cc270d5c4de5806f37ce405606John McCall
1808265941bc308d65cc270d5c4de5806f37ce405606John McCall  /// isRetaining - Return true if the property retains its value.
1809265941bc308d65cc270d5c4de5806f37ce405606John McCall  bool isRetaining() const {
1810265941bc308d65cc270d5c4de5806f37ce405606John McCall    return (PropertyAttributes &
1811265941bc308d65cc270d5c4de5806f37ce405606John McCall            (OBJC_PR_retain | OBJC_PR_strong | OBJC_PR_copy));
1812265941bc308d65cc270d5c4de5806f37ce405606John McCall  }
1813265941bc308d65cc270d5c4de5806f37ce405606John McCall
1814af05bb9073319d8381b71c4325188853fd4b8ed6Daniel Dunbar  /// getSetterKind - Return the method used for doing assignment in
1815af05bb9073319d8381b71c4325188853fd4b8ed6Daniel Dunbar  /// the property setter. This is only valid if the property has been
1816af05bb9073319d8381b71c4325188853fd4b8ed6Daniel Dunbar  /// defined to have a setter.
1817af05bb9073319d8381b71c4325188853fd4b8ed6Daniel Dunbar  SetterKind getSetterKind() const {
18186dc6f008994472cf4da321855e8c51c39720f3edJohn McCall    if (PropertyAttributes & OBJC_PR_strong)
18196dc6f008994472cf4da321855e8c51c39720f3edJohn McCall      return getType()->isBlockPointerType() ? Copy : Retain;
18206dc6f008994472cf4da321855e8c51c39720f3edJohn McCall    if (PropertyAttributes & OBJC_PR_retain)
1821af05bb9073319d8381b71c4325188853fd4b8ed6Daniel Dunbar      return Retain;
1822af05bb9073319d8381b71c4325188853fd4b8ed6Daniel Dunbar    if (PropertyAttributes & OBJC_PR_copy)
1823af05bb9073319d8381b71c4325188853fd4b8ed6Daniel Dunbar      return Copy;
18243a02b44e3948f7762dbfba94b7961281ca29d022Fariborz Jahanian    if (PropertyAttributes & OBJC_PR_weak)
18253a02b44e3948f7762dbfba94b7961281ca29d022Fariborz Jahanian      return Weak;
1826af05bb9073319d8381b71c4325188853fd4b8ed6Daniel Dunbar    return Assign;
1827af05bb9073319d8381b71c4325188853fd4b8ed6Daniel Dunbar  }
1828af05bb9073319d8381b71c4325188853fd4b8ed6Daniel Dunbar
18295251e130a23d997f7c0dfdc250cdc41f179e5bedFariborz Jahanian  Selector getGetterName() const { return GetterName; }
18305251e130a23d997f7c0dfdc250cdc41f179e5bedFariborz Jahanian  void setGetterName(Selector Sel) { GetterName = Sel; }
18311eb4433ac451dc16f4133a88af2d002ac26c58efMike Stump
18325251e130a23d997f7c0dfdc250cdc41f179e5bedFariborz Jahanian  Selector getSetterName() const { return SetterName; }
18335251e130a23d997f7c0dfdc250cdc41f179e5bedFariborz Jahanian  void setSetterName(Selector Sel) { SetterName = Sel; }
18341eb4433ac451dc16f4133a88af2d002ac26c58efMike Stump
183533de3f0333ca0b5274291b8d76c86758c0484691Fariborz Jahanian  ObjCMethodDecl *getGetterMethodDecl() const { return GetterMethodDecl; }
183633de3f0333ca0b5274291b8d76c86758c0484691Fariborz Jahanian  void setGetterMethodDecl(ObjCMethodDecl *gDecl) { GetterMethodDecl = gDecl; }
183733de3f0333ca0b5274291b8d76c86758c0484691Fariborz Jahanian
183833de3f0333ca0b5274291b8d76c86758c0484691Fariborz Jahanian  ObjCMethodDecl *getSetterMethodDecl() const { return SetterMethodDecl; }
183933de3f0333ca0b5274291b8d76c86758c0484691Fariborz Jahanian  void setSetterMethodDecl(ObjCMethodDecl *gDecl) { SetterMethodDecl = gDecl; }
18401eb4433ac451dc16f4133a88af2d002ac26c58efMike Stump
184146b55e56d029aec699fc2701e43d70264da9ecd8Fariborz Jahanian  // Related to @optional/@required declared in @protocol
184246b55e56d029aec699fc2701e43d70264da9ecd8Fariborz Jahanian  void setPropertyImplementation(PropertyControl pc) {
184346b55e56d029aec699fc2701e43d70264da9ecd8Fariborz Jahanian    PropertyImplementation = pc;
184446b55e56d029aec699fc2701e43d70264da9ecd8Fariborz Jahanian  }
184546b55e56d029aec699fc2701e43d70264da9ecd8Fariborz Jahanian  PropertyControl getPropertyImplementation() const {
184646b55e56d029aec699fc2701e43d70264da9ecd8Fariborz Jahanian    return PropertyControl(PropertyImplementation);
18471eb4433ac451dc16f4133a88af2d002ac26c58efMike Stump  }
18481eb4433ac451dc16f4133a88af2d002ac26c58efMike Stump
1849af3e72285238369c2ea4ebd40a1c9a87bd3eabb7Fariborz Jahanian  void setPropertyIvarDecl(ObjCIvarDecl *Ivar) {
1850af3e72285238369c2ea4ebd40a1c9a87bd3eabb7Fariborz Jahanian    PropertyIvarDecl = Ivar;
1851af3e72285238369c2ea4ebd40a1c9a87bd3eabb7Fariborz Jahanian  }
1852af3e72285238369c2ea4ebd40a1c9a87bd3eabb7Fariborz Jahanian  ObjCIvarDecl *getPropertyIvarDecl() const {
1853af3e72285238369c2ea4ebd40a1c9a87bd3eabb7Fariborz Jahanian    return PropertyIvarDecl;
1854af3e72285238369c2ea4ebd40a1c9a87bd3eabb7Fariborz Jahanian  }
18551eb4433ac451dc16f4133a88af2d002ac26c58efMike Stump
1856da2142f2e2b3a02ee6eb5de9f9e6ed6f7eb5a0c0Douglas Gregor  virtual SourceRange getSourceRange() const {
1857e3c60a7ce9e0f42c7ca2344b33203266aceca1dbDouglas Gregor    return SourceRange(AtLoc, getLocation());
1858e3c60a7ce9e0f42c7ca2344b33203266aceca1dbDouglas Gregor  }
1859e3c60a7ce9e0f42c7ca2344b33203266aceca1dbDouglas Gregor
18609f550ff05d496e6b9480e5619a21d9da0c9e27c1Ted Kremenek  /// Lookup a property by name in the specified DeclContext.
1861de09d0c9694f01a99870a8825266d44a29ebb325Ted Kremenek  static ObjCPropertyDecl *findPropertyDecl(const DeclContext *DC,
18629f550ff05d496e6b9480e5619a21d9da0c9e27c1Ted Kremenek                                            IdentifierInfo *propertyID);
18639f550ff05d496e6b9480e5619a21d9da0c9e27c1Ted Kremenek
186480cd64a8450d8e2c079dc134d9711cd45ba89d63John McCall  static bool classof(const Decl *D) { return classofKind(D->getKind()); }
1865a526c5c67e5a0473c340903ee542ce570119665fTed Kremenek  static bool classof(const ObjCPropertyDecl *D) { return true; }
186680cd64a8450d8e2c079dc134d9711cd45ba89d63John McCall  static bool classofKind(Kind K) { return K == ObjCProperty; }
186782a5fe3d1cf204b672cdab24d72275b6ad2c3527Fariborz Jahanian};
1868980e508ca70d6de75d2abfd96b4681fc98bb2698Steve Naroff
18691eb4433ac451dc16f4133a88af2d002ac26c58efMike Stump/// ObjCPropertyImplDecl - Represents implementation declaration of a property
187061d46159af2a740207de8dc024211d531ae290d9Fariborz Jahanian/// in a class or category implementation block. For example:
187161d46159af2a740207de8dc024211d531ae290d9Fariborz Jahanian/// @synthesize prop1 = ivar1;
187261d46159af2a740207de8dc024211d531ae290d9Fariborz Jahanian///
18734afa39deaa245592977136d367251ee2c173dd8dDouglas Gregorclass ObjCPropertyImplDecl : public Decl {
187461d46159af2a740207de8dc024211d531ae290d9Fariborz Jahanianpublic:
18759f0afd4e79601d9982072ff9318e6f9a982c770eDaniel Dunbar  enum Kind {
18769f0afd4e79601d9982072ff9318e6f9a982c770eDaniel Dunbar    Synthesize,
18779f0afd4e79601d9982072ff9318e6f9a982c770eDaniel Dunbar    Dynamic
187861d46159af2a740207de8dc024211d531ae290d9Fariborz Jahanian  };
187961d46159af2a740207de8dc024211d531ae290d9Fariborz Jahanianprivate:
1880559c0c4bbecc017aab0716d546c4fefbcc194687Fariborz Jahanian  SourceLocation AtLoc;   // location of @synthesize or @dynamic
1881ba243b59a1074e0962f6abfa3bb9aa984eac1245David Blaikie
1882a4ffd85a6684e42f900aad5459e58ad91bb88755Douglas Gregor  /// \brief For @synthesize, the location of the ivar, if it was written in
1883a4ffd85a6684e42f900aad5459e58ad91bb88755Douglas Gregor  /// the source code.
1884a4ffd85a6684e42f900aad5459e58ad91bb88755Douglas Gregor  ///
1885a4ffd85a6684e42f900aad5459e58ad91bb88755Douglas Gregor  /// \code
1886a4ffd85a6684e42f900aad5459e58ad91bb88755Douglas Gregor  /// @synthesize int a = b
1887a4ffd85a6684e42f900aad5459e58ad91bb88755Douglas Gregor  /// \endcode
1888a4ffd85a6684e42f900aad5459e58ad91bb88755Douglas Gregor  SourceLocation IvarLoc;
1889ba243b59a1074e0962f6abfa3bb9aa984eac1245David Blaikie
189061d46159af2a740207de8dc024211d531ae290d9Fariborz Jahanian  /// Property declaration being implemented
189161d46159af2a740207de8dc024211d531ae290d9Fariborz Jahanian  ObjCPropertyDecl *PropertyDecl;
1892be57c3a3fef0776fca57ad88b2db263f37b074c4Ted Kremenek
189361d46159af2a740207de8dc024211d531ae290d9Fariborz Jahanian  /// Null for @dynamic. Required for @synthesize.
189461d46159af2a740207de8dc024211d531ae290d9Fariborz Jahanian  ObjCIvarDecl *PropertyIvarDecl;
1895ba243b59a1074e0962f6abfa3bb9aa984eac1245David Blaikie
189617cb326cb62a59f53d92236394af40eaae4eddbdFariborz Jahanian  /// Null for @dynamic. Non-null if property must be copy-constructed in getter
189717cb326cb62a59f53d92236394af40eaae4eddbdFariborz Jahanian  Expr *GetterCXXConstructor;
1898ba243b59a1074e0962f6abfa3bb9aa984eac1245David Blaikie
189917cb326cb62a59f53d92236394af40eaae4eddbdFariborz Jahanian  /// Null for @dynamic. Non-null if property has assignment operator to call
190017cb326cb62a59f53d92236394af40eaae4eddbdFariborz Jahanian  /// in Setter synthesis.
190117cb326cb62a59f53d92236394af40eaae4eddbdFariborz Jahanian  Expr *SetterCXXAssignment;
1902be57c3a3fef0776fca57ad88b2db263f37b074c4Ted Kremenek
1903d04341000d35c8808a72838b057eed7bf13b7661Douglas Gregor  ObjCPropertyImplDecl(DeclContext *DC, SourceLocation atLoc, SourceLocation L,
19041eb4433ac451dc16f4133a88af2d002ac26c58efMike Stump                       ObjCPropertyDecl *property,
19051eb4433ac451dc16f4133a88af2d002ac26c58efMike Stump                       Kind PK,
1906a4ffd85a6684e42f900aad5459e58ad91bb88755Douglas Gregor                       ObjCIvarDecl *ivarDecl,
1907a4ffd85a6684e42f900aad5459e58ad91bb88755Douglas Gregor                       SourceLocation ivarLoc)
19081eb4433ac451dc16f4133a88af2d002ac26c58efMike Stump    : Decl(ObjCPropertyImpl, DC, L), AtLoc(atLoc),
1909ba243b59a1074e0962f6abfa3bb9aa984eac1245David Blaikie      IvarLoc(ivarLoc), PropertyDecl(property), PropertyIvarDecl(ivarDecl),
191017cb326cb62a59f53d92236394af40eaae4eddbdFariborz Jahanian      GetterCXXConstructor(0), SetterCXXAssignment(0) {
19119f0afd4e79601d9982072ff9318e6f9a982c770eDaniel Dunbar    assert (PK == Dynamic || PropertyIvarDecl);
19129f0afd4e79601d9982072ff9318e6f9a982c770eDaniel Dunbar  }
19131eb4433ac451dc16f4133a88af2d002ac26c58efMike Stump
19149f0afd4e79601d9982072ff9318e6f9a982c770eDaniel Dunbarpublic:
1915d04341000d35c8808a72838b057eed7bf13b7661Douglas Gregor  static ObjCPropertyImplDecl *Create(ASTContext &C, DeclContext *DC,
19161eb4433ac451dc16f4133a88af2d002ac26c58efMike Stump                                      SourceLocation atLoc, SourceLocation L,
19171eb4433ac451dc16f4133a88af2d002ac26c58efMike Stump                                      ObjCPropertyDecl *property,
19181eb4433ac451dc16f4133a88af2d002ac26c58efMike Stump                                      Kind PK,
1919a4ffd85a6684e42f900aad5459e58ad91bb88755Douglas Gregor                                      ObjCIvarDecl *ivarDecl,
1920a4ffd85a6684e42f900aad5459e58ad91bb88755Douglas Gregor                                      SourceLocation ivarLoc);
192161d46159af2a740207de8dc024211d531ae290d9Fariborz Jahanian
1922da2142f2e2b3a02ee6eb5de9f9e6ed6f7eb5a0c0Douglas Gregor  virtual SourceRange getSourceRange() const;
1923ba243b59a1074e0962f6abfa3bb9aa984eac1245David Blaikie
1924d40910b581b09c937a8c1fdcde9b8ec724398fb9Steve Naroff  SourceLocation getLocStart() const { return AtLoc; }
19258818c4fb69cb2a4eec94b217a90f94f9e075296eDouglas Gregor  void setAtLoc(SourceLocation Loc) { AtLoc = Loc; }
1926d40910b581b09c937a8c1fdcde9b8ec724398fb9Steve Naroff
1927be57c3a3fef0776fca57ad88b2db263f37b074c4Ted Kremenek  ObjCPropertyDecl *getPropertyDecl() const {
1928be57c3a3fef0776fca57ad88b2db263f37b074c4Ted Kremenek    return PropertyDecl;
1929be57c3a3fef0776fca57ad88b2db263f37b074c4Ted Kremenek  }
19308818c4fb69cb2a4eec94b217a90f94f9e075296eDouglas Gregor  void setPropertyDecl(ObjCPropertyDecl *Prop) { PropertyDecl = Prop; }
19318818c4fb69cb2a4eec94b217a90f94f9e075296eDouglas Gregor
19329f0afd4e79601d9982072ff9318e6f9a982c770eDaniel Dunbar  Kind getPropertyImplementation() const {
19339f0afd4e79601d9982072ff9318e6f9a982c770eDaniel Dunbar    return PropertyIvarDecl ? Synthesize : Dynamic;
1934be57c3a3fef0776fca57ad88b2db263f37b074c4Ted Kremenek  }
19351eb4433ac451dc16f4133a88af2d002ac26c58efMike Stump
1936af05bb9073319d8381b71c4325188853fd4b8ed6Daniel Dunbar  ObjCIvarDecl *getPropertyIvarDecl() const {
1937be57c3a3fef0776fca57ad88b2db263f37b074c4Ted Kremenek    return PropertyIvarDecl;
1938be57c3a3fef0776fca57ad88b2db263f37b074c4Ted Kremenek  }
1939a4ffd85a6684e42f900aad5459e58ad91bb88755Douglas Gregor  SourceLocation getPropertyIvarDeclLoc() const { return IvarLoc; }
1940ba243b59a1074e0962f6abfa3bb9aa984eac1245David Blaikie
1941a4ffd85a6684e42f900aad5459e58ad91bb88755Douglas Gregor  void setPropertyIvarDecl(ObjCIvarDecl *Ivar,
1942ba243b59a1074e0962f6abfa3bb9aa984eac1245David Blaikie                           SourceLocation IvarLoc) {
1943ba243b59a1074e0962f6abfa3bb9aa984eac1245David Blaikie    PropertyIvarDecl = Ivar;
1944a4ffd85a6684e42f900aad5459e58ad91bb88755Douglas Gregor    this->IvarLoc = IvarLoc;
1945a4ffd85a6684e42f900aad5459e58ad91bb88755Douglas Gregor  }
1946ba243b59a1074e0962f6abfa3bb9aa984eac1245David Blaikie
194717cb326cb62a59f53d92236394af40eaae4eddbdFariborz Jahanian  Expr *getGetterCXXConstructor() const {
194817cb326cb62a59f53d92236394af40eaae4eddbdFariborz Jahanian    return GetterCXXConstructor;
194917cb326cb62a59f53d92236394af40eaae4eddbdFariborz Jahanian  }
195017cb326cb62a59f53d92236394af40eaae4eddbdFariborz Jahanian  void setGetterCXXConstructor(Expr *getterCXXConstructor) {
195117cb326cb62a59f53d92236394af40eaae4eddbdFariborz Jahanian    GetterCXXConstructor = getterCXXConstructor;
195217cb326cb62a59f53d92236394af40eaae4eddbdFariborz Jahanian  }
19538818c4fb69cb2a4eec94b217a90f94f9e075296eDouglas Gregor
195417cb326cb62a59f53d92236394af40eaae4eddbdFariborz Jahanian  Expr *getSetterCXXAssignment() const {
195517cb326cb62a59f53d92236394af40eaae4eddbdFariborz Jahanian    return SetterCXXAssignment;
195617cb326cb62a59f53d92236394af40eaae4eddbdFariborz Jahanian  }
195717cb326cb62a59f53d92236394af40eaae4eddbdFariborz Jahanian  void setSetterCXXAssignment(Expr *setterCXXAssignment) {
195817cb326cb62a59f53d92236394af40eaae4eddbdFariborz Jahanian    SetterCXXAssignment = setterCXXAssignment;
195917cb326cb62a59f53d92236394af40eaae4eddbdFariborz Jahanian  }
1960ba243b59a1074e0962f6abfa3bb9aa984eac1245David Blaikie
196180cd64a8450d8e2c079dc134d9711cd45ba89d63John McCall  static bool classof(const Decl *D) { return classofKind(D->getKind()); }
19621eb4433ac451dc16f4133a88af2d002ac26c58efMike Stump  static bool classof(const ObjCPropertyImplDecl *D) { return true; }
196380cd64a8450d8e2c079dc134d9711cd45ba89d63John McCall  static bool classofKind(Decl::Kind K) { return K == ObjCPropertyImpl; }
1964ba243b59a1074e0962f6abfa3bb9aa984eac1245David Blaikie
1965a4ffd85a6684e42f900aad5459e58ad91bb88755Douglas Gregor  friend class ASTDeclReader;
196661d46159af2a740207de8dc024211d531ae290d9Fariborz Jahanian};
196761d46159af2a740207de8dc024211d531ae290d9Fariborz Jahanian
1968980e508ca70d6de75d2abfd96b4681fc98bb2698Steve Naroff}  // end namespace clang
1969980e508ca70d6de75d2abfd96b4681fc98bb2698Steve Naroff#endif
1970