DeclObjC.h revision b048c9835969c4f7fe06264748be18ed4b442116
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"
18c7229c338c21ef26b01ef3ecf9eec4fd373fa9ecChris Lattner#include "clang/Basic/IdentifierTable.h"
19980e508ca70d6de75d2abfd96b4681fc98bb2698Steve Naroff
20980e508ca70d6de75d2abfd96b4681fc98bb2698Steve Naroffnamespace clang {
21980e508ca70d6de75d2abfd96b4681fc98bb2698Steve Naroffclass Expr;
22980e508ca70d6de75d2abfd96b4681fc98bb2698Steve Naroffclass Stmt;
23980e508ca70d6de75d2abfd96b4681fc98bb2698Steve Naroffclass FunctionDecl;
24980e508ca70d6de75d2abfd96b4681fc98bb2698Steve Naroffclass AttributeList;
25a526c5c67e5a0473c340903ee542ce570119665fTed Kremenekclass ObjCIvarDecl;
26a526c5c67e5a0473c340903ee542ce570119665fTed Kremenekclass ObjCMethodDecl;
27a526c5c67e5a0473c340903ee542ce570119665fTed Kremenekclass ObjCProtocolDecl;
28a526c5c67e5a0473c340903ee542ce570119665fTed Kremenekclass ObjCCategoryDecl;
29a526c5c67e5a0473c340903ee542ce570119665fTed Kremenekclass ObjCPropertyDecl;
3058dbdeb69c063f82d644504fc638120198f7fad2Steve Naroff
31a526c5c67e5a0473c340903ee542ce570119665fTed Kremenek/// ObjCMethodDecl - Represents an instance or class method declaration.
3258dbdeb69c063f82d644504fc638120198f7fad2Steve Naroff/// ObjC methods can be declared within 4 contexts: class interfaces,
3358dbdeb69c063f82d644504fc638120198f7fad2Steve Naroff/// categories, protocols, and class implementations. While C++ member
3458dbdeb69c063f82d644504fc638120198f7fad2Steve Naroff/// functions leverage C syntax, Objective-C method syntax is modeled after
3558dbdeb69c063f82d644504fc638120198f7fad2Steve Naroff/// Smalltalk (using colons to specify argument types/expressions).
3658dbdeb69c063f82d644504fc638120198f7fad2Steve Naroff/// Here are some brief examples:
3758dbdeb69c063f82d644504fc638120198f7fad2Steve Naroff///
3858dbdeb69c063f82d644504fc638120198f7fad2Steve Naroff/// Setter/getter instance methods:
3958dbdeb69c063f82d644504fc638120198f7fad2Steve Naroff/// - (void)setMenu:(NSMenu *)menu;
4058dbdeb69c063f82d644504fc638120198f7fad2Steve Naroff/// - (NSMenu *)menu;
4158dbdeb69c063f82d644504fc638120198f7fad2Steve Naroff///
4258dbdeb69c063f82d644504fc638120198f7fad2Steve Naroff/// Instance method that takes 2 NSView arguments:
4358dbdeb69c063f82d644504fc638120198f7fad2Steve Naroff/// - (void)replaceSubview:(NSView *)oldView with:(NSView *)newView;
4458dbdeb69c063f82d644504fc638120198f7fad2Steve Naroff///
4558dbdeb69c063f82d644504fc638120198f7fad2Steve Naroff/// Getter class method:
4658dbdeb69c063f82d644504fc638120198f7fad2Steve Naroff/// + (NSMenu *)defaultMenu;
4758dbdeb69c063f82d644504fc638120198f7fad2Steve Naroff///
4858dbdeb69c063f82d644504fc638120198f7fad2Steve Naroff/// A selector represents a unique name for a method. The selector names for
4958dbdeb69c063f82d644504fc638120198f7fad2Steve Naroff/// the above methods are setMenu:, menu, replaceSubview:with:, and defaultMenu.
5058dbdeb69c063f82d644504fc638120198f7fad2Steve Naroff///
51b048c9835969c4f7fe06264748be18ed4b442116Chris Lattnerclass ObjCMethodDecl : public Decl, public DeclContext {
5258dbdeb69c063f82d644504fc638120198f7fad2Steve Naroffpublic:
5358dbdeb69c063f82d644504fc638120198f7fad2Steve Naroff  enum ImplementationControl { None, Required, Optional };
5458dbdeb69c063f82d644504fc638120198f7fad2Steve Naroffprivate:
5558dbdeb69c063f82d644504fc638120198f7fad2Steve Naroff  /// Bitfields must be first fields in this class so they pack with those
5658dbdeb69c063f82d644504fc638120198f7fad2Steve Naroff  /// declared in class Decl.
5758dbdeb69c063f82d644504fc638120198f7fad2Steve Naroff  /// instance (true) or class (false) method.
5858dbdeb69c063f82d644504fc638120198f7fad2Steve Naroff  bool IsInstance : 1;
5958dbdeb69c063f82d644504fc638120198f7fad2Steve Naroff  bool IsVariadic : 1;
6058dbdeb69c063f82d644504fc638120198f7fad2Steve Naroff
61ca63fa00786e51c207c829f4182f11a6c6b552beTed Kremenek  // NOTE: VC++ treats enums as signed, avoid using ImplementationControl enum
6258dbdeb69c063f82d644504fc638120198f7fad2Steve Naroff  /// @required/@optional
63ca63fa00786e51c207c829f4182f11a6c6b552beTed Kremenek  unsigned DeclImplementation : 2;
6458dbdeb69c063f82d644504fc638120198f7fad2Steve Naroff
65ca63fa00786e51c207c829f4182f11a6c6b552beTed Kremenek  // NOTE: VC++ treats enums as signed, avoid using the ObjCDeclQualifier enum
6658dbdeb69c063f82d644504fc638120198f7fad2Steve Naroff  /// in, inout, etc.
67ca63fa00786e51c207c829f4182f11a6c6b552beTed Kremenek  unsigned objcDeclQualifier : 6;
6858dbdeb69c063f82d644504fc638120198f7fad2Steve Naroff
6958dbdeb69c063f82d644504fc638120198f7fad2Steve Naroff  // Context this method is declared in.
7058dbdeb69c063f82d644504fc638120198f7fad2Steve Naroff  NamedDecl *MethodContext;
7158dbdeb69c063f82d644504fc638120198f7fad2Steve Naroff
7258dbdeb69c063f82d644504fc638120198f7fad2Steve Naroff  // A unigue name for this method.
7358dbdeb69c063f82d644504fc638120198f7fad2Steve Naroff  Selector SelName;
7458dbdeb69c063f82d644504fc638120198f7fad2Steve Naroff
7558dbdeb69c063f82d644504fc638120198f7fad2Steve Naroff  // Type of this method.
7658dbdeb69c063f82d644504fc638120198f7fad2Steve Naroff  QualType MethodDeclType;
7758dbdeb69c063f82d644504fc638120198f7fad2Steve Naroff  /// ParamInfo - new[]'d array of pointers to VarDecls for the formal
7858dbdeb69c063f82d644504fc638120198f7fad2Steve Naroff  /// parameters of this Method.  This is null if there are no formals.
7958dbdeb69c063f82d644504fc638120198f7fad2Steve Naroff  ParmVarDecl **ParamInfo;
8058cce3b0dcbdcc95b7e713795834b4cb2c8a008aChris Lattner  unsigned NumMethodParams;
8158dbdeb69c063f82d644504fc638120198f7fad2Steve Naroff
8258dbdeb69c063f82d644504fc638120198f7fad2Steve Naroff  /// List of attributes for this method declaration.
8358dbdeb69c063f82d644504fc638120198f7fad2Steve Naroff  AttributeList *MethodAttrs;
8458dbdeb69c063f82d644504fc638120198f7fad2Steve Naroff
8558dbdeb69c063f82d644504fc638120198f7fad2Steve Naroff  SourceLocation EndLoc; // the location of the ';' or '{'.
8658dbdeb69c063f82d644504fc638120198f7fad2Steve Naroff
8758dbdeb69c063f82d644504fc638120198f7fad2Steve Naroff  // The following are only used for method definitions, null otherwise.
8858dbdeb69c063f82d644504fc638120198f7fad2Steve Naroff  // FIXME: space savings opportunity, consider a sub-class.
8958dbdeb69c063f82d644504fc638120198f7fad2Steve Naroff  Stmt *Body;
9058dbdeb69c063f82d644504fc638120198f7fad2Steve Naroff  ParmVarDecl *SelfDecl;
916c4ae5de0c356777446f823b573821fb95560d91Chris Lattner
92a526c5c67e5a0473c340903ee542ce570119665fTed Kremenek  ObjCMethodDecl(SourceLocation beginLoc, SourceLocation endLoc,
9358dbdeb69c063f82d644504fc638120198f7fad2Steve Naroff                 Selector SelInfo, QualType T,
9458dbdeb69c063f82d644504fc638120198f7fad2Steve Naroff                 Decl *contextDecl,
9558dbdeb69c063f82d644504fc638120198f7fad2Steve Naroff                 AttributeList *M = 0, bool isInstance = true,
9658dbdeb69c063f82d644504fc638120198f7fad2Steve Naroff                 bool isVariadic = false,
9758cce3b0dcbdcc95b7e713795834b4cb2c8a008aChris Lattner                 ImplementationControl impControl = None)
98a526c5c67e5a0473c340903ee542ce570119665fTed Kremenek  : Decl(ObjCMethod, beginLoc),
99b048c9835969c4f7fe06264748be18ed4b442116Chris Lattner    DeclContext(ObjCMethod),
10058dbdeb69c063f82d644504fc638120198f7fad2Steve Naroff    IsInstance(isInstance), IsVariadic(isVariadic),
10158dbdeb69c063f82d644504fc638120198f7fad2Steve Naroff    DeclImplementation(impControl), objcDeclQualifier(OBJC_TQ_None),
10258dbdeb69c063f82d644504fc638120198f7fad2Steve Naroff    MethodContext(static_cast<NamedDecl*>(contextDecl)),
10358dbdeb69c063f82d644504fc638120198f7fad2Steve Naroff    SelName(SelInfo), MethodDeclType(T),
104b06fa3b86951b9f179c99c3768331536c32e902dChris Lattner    ParamInfo(0), NumMethodParams(0),
10558dbdeb69c063f82d644504fc638120198f7fad2Steve Naroff    MethodAttrs(M), EndLoc(endLoc), Body(0), SelfDecl(0) {}
106b06fa3b86951b9f179c99c3768331536c32e902dChris Lattner  virtual ~ObjCMethodDecl();
1076c4ae5de0c356777446f823b573821fb95560d91Chris Lattnerpublic:
1086c4ae5de0c356777446f823b573821fb95560d91Chris Lattner
1090ed844b04ea4387caa4e1cf3dc375d269657536bChris Lattner  static ObjCMethodDecl *Create(ASTContext &C,
1100ed844b04ea4387caa4e1cf3dc375d269657536bChris Lattner                                SourceLocation beginLoc,
1116c4ae5de0c356777446f823b573821fb95560d91Chris Lattner                                SourceLocation endLoc, Selector SelInfo,
1126c4ae5de0c356777446f823b573821fb95560d91Chris Lattner                                QualType T, Decl *contextDecl,
1136c4ae5de0c356777446f823b573821fb95560d91Chris Lattner                                AttributeList *M = 0, bool isInstance = true,
1146c4ae5de0c356777446f823b573821fb95560d91Chris Lattner                                bool isVariadic = false,
115b06fa3b86951b9f179c99c3768331536c32e902dChris Lattner                                ImplementationControl impControl = None);
11658dbdeb69c063f82d644504fc638120198f7fad2Steve Naroff
117ca63fa00786e51c207c829f4182f11a6c6b552beTed Kremenek  ObjCDeclQualifier getObjCDeclQualifier() const {
118ca63fa00786e51c207c829f4182f11a6c6b552beTed Kremenek    return ObjCDeclQualifier(objcDeclQualifier);
119ca63fa00786e51c207c829f4182f11a6c6b552beTed Kremenek  }
120a526c5c67e5a0473c340903ee542ce570119665fTed Kremenek  void setObjCDeclQualifier(ObjCDeclQualifier QV) { objcDeclQualifier = QV; }
12158dbdeb69c063f82d644504fc638120198f7fad2Steve Naroff
12258dbdeb69c063f82d644504fc638120198f7fad2Steve Naroff  // Location information, modeled after the Stmt API.
12358dbdeb69c063f82d644504fc638120198f7fad2Steve Naroff  SourceLocation getLocStart() const { return getLocation(); }
12458dbdeb69c063f82d644504fc638120198f7fad2Steve Naroff  SourceLocation getLocEnd() const { return EndLoc; }
12558dbdeb69c063f82d644504fc638120198f7fad2Steve Naroff
12658dbdeb69c063f82d644504fc638120198f7fad2Steve Naroff  NamedDecl *getMethodContext() const { return MethodContext; }
12758dbdeb69c063f82d644504fc638120198f7fad2Steve Naroff
128e0def7589a8afa8a6acef13476bb3f882c104b91Chris Lattner  const ObjCInterfaceDecl *getClassInterface() const;
129e0def7589a8afa8a6acef13476bb3f882c104b91Chris Lattner  ObjCInterfaceDecl *getClassInterface() {
130e0def7589a8afa8a6acef13476bb3f882c104b91Chris Lattner    return (ObjCInterfaceDecl*)((ObjCMethodDecl*)this)->getClassInterface();
131e0def7589a8afa8a6acef13476bb3f882c104b91Chris Lattner  }
13258dbdeb69c063f82d644504fc638120198f7fad2Steve Naroff
13358dbdeb69c063f82d644504fc638120198f7fad2Steve Naroff  Selector getSelector() const { return SelName; }
134faf5e779d16bb4590f2a97e1c7ded255eddd90f3Fariborz Jahanian  unsigned getSynthesizedMethodSize() const;
13558dbdeb69c063f82d644504fc638120198f7fad2Steve Naroff  QualType getResultType() const { return MethodDeclType; }
13658dbdeb69c063f82d644504fc638120198f7fad2Steve Naroff
137d57f635d520e8cb5b93d3d770ff58db06c62de54Chris Lattner  // Iterator access to formal parameters.
13858cce3b0dcbdcc95b7e713795834b4cb2c8a008aChris Lattner  unsigned param_size() const { return NumMethodParams; }
139d57f635d520e8cb5b93d3d770ff58db06c62de54Chris Lattner  typedef ParmVarDecl **param_iterator;
140d57f635d520e8cb5b93d3d770ff58db06c62de54Chris Lattner  typedef ParmVarDecl * const *param_const_iterator;
141d57f635d520e8cb5b93d3d770ff58db06c62de54Chris Lattner  param_iterator param_begin() { return ParamInfo; }
142d57f635d520e8cb5b93d3d770ff58db06c62de54Chris Lattner  param_iterator param_end() { return ParamInfo+param_size(); }
143d57f635d520e8cb5b93d3d770ff58db06c62de54Chris Lattner  param_const_iterator param_begin() const { return ParamInfo; }
144d57f635d520e8cb5b93d3d770ff58db06c62de54Chris Lattner  param_const_iterator param_end() const { return ParamInfo+param_size(); }
145d57f635d520e8cb5b93d3d770ff58db06c62de54Chris Lattner
14658cce3b0dcbdcc95b7e713795834b4cb2c8a008aChris Lattner  unsigned getNumParams() const { return NumMethodParams; }
14758cce3b0dcbdcc95b7e713795834b4cb2c8a008aChris Lattner  ParmVarDecl *getParamDecl(unsigned i) const {
14858dbdeb69c063f82d644504fc638120198f7fad2Steve Naroff    assert(i < getNumParams() && "Illegal param #");
14958dbdeb69c063f82d644504fc638120198f7fad2Steve Naroff    return ParamInfo[i];
15058dbdeb69c063f82d644504fc638120198f7fad2Steve Naroff  }
1512338d58a905191c4205d4f73affd0bdaa13493b4Fariborz Jahanian  void setParamDecl(int i, ParmVarDecl *pDecl) {
1522338d58a905191c4205d4f73affd0bdaa13493b4Fariborz Jahanian    ParamInfo[i] = pDecl;
1532338d58a905191c4205d4f73affd0bdaa13493b4Fariborz Jahanian  }
15458dbdeb69c063f82d644504fc638120198f7fad2Steve Naroff  void setMethodParams(ParmVarDecl **NewParamInfo, unsigned NumParams);
15558dbdeb69c063f82d644504fc638120198f7fad2Steve Naroff
15658dbdeb69c063f82d644504fc638120198f7fad2Steve Naroff  AttributeList *getMethodAttrs() const {return MethodAttrs;}
15758dbdeb69c063f82d644504fc638120198f7fad2Steve Naroff  bool isInstance() const { return IsInstance; }
15858dbdeb69c063f82d644504fc638120198f7fad2Steve Naroff  bool isVariadic() const { return IsVariadic; }
15958dbdeb69c063f82d644504fc638120198f7fad2Steve Naroff
16058dbdeb69c063f82d644504fc638120198f7fad2Steve Naroff  // Related to protocols declared in  @protocol
16158dbdeb69c063f82d644504fc638120198f7fad2Steve Naroff  void setDeclImplementation(ImplementationControl ic) {
16258dbdeb69c063f82d644504fc638120198f7fad2Steve Naroff    DeclImplementation = ic;
16358dbdeb69c063f82d644504fc638120198f7fad2Steve Naroff  }
16458dbdeb69c063f82d644504fc638120198f7fad2Steve Naroff  ImplementationControl getImplementationControl() const {
165ca63fa00786e51c207c829f4182f11a6c6b552beTed Kremenek    return ImplementationControl(DeclImplementation);
16658dbdeb69c063f82d644504fc638120198f7fad2Steve Naroff  }
167e0def7589a8afa8a6acef13476bb3f882c104b91Chris Lattner  Stmt *getBody() { return Body; }
168e0def7589a8afa8a6acef13476bb3f882c104b91Chris Lattner  const Stmt *getBody() const { return Body; }
16958dbdeb69c063f82d644504fc638120198f7fad2Steve Naroff  void setBody(Stmt *B) { Body = B; }
17058dbdeb69c063f82d644504fc638120198f7fad2Steve Naroff
171e0def7589a8afa8a6acef13476bb3f882c104b91Chris Lattner  const ParmVarDecl *getSelfDecl() const { return SelfDecl; }
172e0def7589a8afa8a6acef13476bb3f882c104b91Chris Lattner  ParmVarDecl *getSelfDecl() { return SelfDecl; }
17358dbdeb69c063f82d644504fc638120198f7fad2Steve Naroff  void setSelfDecl(ParmVarDecl *PVD) { SelfDecl = PVD; }
17458dbdeb69c063f82d644504fc638120198f7fad2Steve Naroff
17558dbdeb69c063f82d644504fc638120198f7fad2Steve Naroff  // Implement isa/cast/dyncast/etc.
176a526c5c67e5a0473c340903ee542ce570119665fTed Kremenek  static bool classof(const Decl *D) { return D->getKind() == ObjCMethod; }
177a526c5c67e5a0473c340903ee542ce570119665fTed Kremenek  static bool classof(const ObjCMethodDecl *D) { return true; }
17858dbdeb69c063f82d644504fc638120198f7fad2Steve Naroff};
179b245a33e6f562377946dc9cb5694a2d3cb8a98a3Fariborz Jahanian
180a526c5c67e5a0473c340903ee542ce570119665fTed Kremenek/// ObjCInterfaceDecl - Represents an ObjC class declaration. For example:
1810c6b6243d3efd958c17943130e2a773653511edcSteve Naroff///
1820c6b6243d3efd958c17943130e2a773653511edcSteve Naroff///   // MostPrimitive declares no super class (not particularly useful).
1830c6b6243d3efd958c17943130e2a773653511edcSteve Naroff///   @interface MostPrimitive
1840c6b6243d3efd958c17943130e2a773653511edcSteve Naroff///     // no instance variables or methods.
1850c6b6243d3efd958c17943130e2a773653511edcSteve Naroff///   @end
1860c6b6243d3efd958c17943130e2a773653511edcSteve Naroff///
187fd5de471478a507dd2495c4ea9bcab801ea5fa65Chris Lattner///   // NSResponder inherits from NSObject & implements NSCoding (a protocol).
1880c6b6243d3efd958c17943130e2a773653511edcSteve Naroff///   @interface NSResponder : NSObject <NSCoding>
189a526c5c67e5a0473c340903ee542ce570119665fTed Kremenek///   { // instance variables are represented by ObjCIvarDecl.
1900c6b6243d3efd958c17943130e2a773653511edcSteve Naroff///     id nextResponder; // nextResponder instance variable.
1910c6b6243d3efd958c17943130e2a773653511edcSteve Naroff///   }
1920c6b6243d3efd958c17943130e2a773653511edcSteve Naroff///   - (NSResponder *)nextResponder; // return a pointer to NSResponder.
1930c6b6243d3efd958c17943130e2a773653511edcSteve Naroff///   - (void)mouseMoved:(NSEvent *)theEvent; // return void, takes a pointer
1940c6b6243d3efd958c17943130e2a773653511edcSteve Naroff///   @end                                    // to an NSEvent.
1950c6b6243d3efd958c17943130e2a773653511edcSteve Naroff///
1960c6b6243d3efd958c17943130e2a773653511edcSteve Naroff///   Unlike C/C++, forward class declarations are accomplished with @class.
1970c6b6243d3efd958c17943130e2a773653511edcSteve Naroff///   Unlike C/C++, @class allows for a list of classes to be forward declared.
1980c6b6243d3efd958c17943130e2a773653511edcSteve Naroff///   Unlike C++, ObjC is a single-rooted class model. In Cocoa, classes
1990c6b6243d3efd958c17943130e2a773653511edcSteve Naroff///   typically inherit from NSObject (an exception is NSProxy).
2000c6b6243d3efd958c17943130e2a773653511edcSteve Naroff///
201b048c9835969c4f7fe06264748be18ed4b442116Chris Lattnerclass ObjCInterfaceDecl : public NamedDecl, public DeclContext {
2023110251f13981689f384eb3c0aba2afffea18d9dSteve Naroff  /// TypeForDecl - This indicates the Type object that represents this
2033110251f13981689f384eb3c0aba2afffea18d9dSteve Naroff  /// TypeDecl.  It is a cache maintained by ASTContext::getObjCInterfaceType
2043110251f13981689f384eb3c0aba2afffea18d9dSteve Naroff  Type *TypeForDecl;
2053110251f13981689f384eb3c0aba2afffea18d9dSteve Naroff  friend class ASTContext;
206980e508ca70d6de75d2abfd96b4681fc98bb2698Steve Naroff
207980e508ca70d6de75d2abfd96b4681fc98bb2698Steve Naroff  /// Class's super class.
208a526c5c67e5a0473c340903ee542ce570119665fTed Kremenek  ObjCInterfaceDecl *SuperClass;
209980e508ca70d6de75d2abfd96b4681fc98bb2698Steve Naroff
210980e508ca70d6de75d2abfd96b4681fc98bb2698Steve Naroff  /// Protocols referenced in interface header declaration
211a526c5c67e5a0473c340903ee542ce570119665fTed Kremenek  ObjCProtocolDecl **ReferencedProtocols;  // Null if none
212c395bda57a0b2f6d8433a462caa780a7dcb7307bFariborz Jahanian  unsigned NumReferencedProtocols;  // 0 if none
213980e508ca70d6de75d2abfd96b4681fc98bb2698Steve Naroff
214980e508ca70d6de75d2abfd96b4681fc98bb2698Steve Naroff  /// Ivars/NumIvars - This is a new[]'d array of pointers to Decls.
215a526c5c67e5a0473c340903ee542ce570119665fTed Kremenek  ObjCIvarDecl **Ivars;   // Null if not defined.
216f3a7af901231535191d14cb524d58f22907ac4d2Chris Lattner  unsigned NumIvars;      // 0 if none.
217980e508ca70d6de75d2abfd96b4681fc98bb2698Steve Naroff
218980e508ca70d6de75d2abfd96b4681fc98bb2698Steve Naroff  /// instance methods
219a526c5c67e5a0473c340903ee542ce570119665fTed Kremenek  ObjCMethodDecl **InstanceMethods;  // Null if not defined
220b5feb35360526da42ad7e3fa7ef3d44a5992b83cChris Lattner  unsigned NumInstanceMethods;  // 0 if none.
221980e508ca70d6de75d2abfd96b4681fc98bb2698Steve Naroff
222980e508ca70d6de75d2abfd96b4681fc98bb2698Steve Naroff  /// class methods
223a526c5c67e5a0473c340903ee542ce570119665fTed Kremenek  ObjCMethodDecl **ClassMethods;  // Null if not defined
22462db2f4214c1589082960f12c9cb8924fe0cf8c5Chris Lattner  unsigned NumClassMethods;  // 0 if none
225980e508ca70d6de75d2abfd96b4681fc98bb2698Steve Naroff
226980e508ca70d6de75d2abfd96b4681fc98bb2698Steve Naroff  /// List of categories defined for this class.
227a526c5c67e5a0473c340903ee542ce570119665fTed Kremenek  ObjCCategoryDecl *CategoryList;
22882a5fe3d1cf204b672cdab24d72275b6ad2c3527Fariborz Jahanian
22982a5fe3d1cf204b672cdab24d72275b6ad2c3527Fariborz Jahanian  /// class properties
230a526c5c67e5a0473c340903ee542ce570119665fTed Kremenek  ObjCPropertyDecl **PropertyDecl;  // Null if no property
23155d13b4d5530a14d5baa72adab32ae78ba256cafChris Lattner  unsigned NumPropertyDecl;  // 0 if none.
232980e508ca70d6de75d2abfd96b4681fc98bb2698Steve Naroff
2333a3ca1b35a7121aea0bf465a192dce748465e10fFariborz Jahanian  bool ForwardDecl:1; // declared with @class.
2343a3ca1b35a7121aea0bf465a192dce748465e10fFariborz Jahanian  bool InternalInterface:1; // true - no @interface for @implementation
23560fcceeedbfc8b4a99cb942e2bc5aeb9e2f92a1fSteve Naroff
236f908a87299d278164540f90b5b6e6cab7b14fb41Steve Naroff  SourceLocation EndLoc; // marks the '>', '}', or identifier.
237f908a87299d278164540f90b5b6e6cab7b14fb41Steve Naroff  SourceLocation AtEndLoc; // marks the end of the entire interface.
2380e77ba0bf769e2e5a4a93c079f241b02aeb3ef93Chris Lattner
2390ed844b04ea4387caa4e1cf3dc375d269657536bChris Lattner  ObjCInterfaceDecl(SourceLocation atLoc,
2400ed844b04ea4387caa4e1cf3dc375d269657536bChris Lattner                    unsigned numRefProtos,
241cca59d77c4b84fd2da268018dbaf9431a621e75bChris Lattner                    IdentifierInfo *Id, bool FD, bool isInternal)
242b048c9835969c4f7fe06264748be18ed4b442116Chris Lattner    : NamedDecl(ObjCInterface, atLoc, Id), DeclContext(ObjCInterface),
2430ed844b04ea4387caa4e1cf3dc375d269657536bChris Lattner      TypeForDecl(0), SuperClass(0),
244c395bda57a0b2f6d8433a462caa780a7dcb7307bFariborz Jahanian      ReferencedProtocols(0), NumReferencedProtocols(0), Ivars(0),
245f3a7af901231535191d14cb524d58f22907ac4d2Chris Lattner      NumIvars(0),
246b5feb35360526da42ad7e3fa7ef3d44a5992b83cChris Lattner      InstanceMethods(0), NumInstanceMethods(0),
24762db2f4214c1589082960f12c9cb8924fe0cf8c5Chris Lattner      ClassMethods(0), NumClassMethods(0),
248f4af5154571e0c5eadb19df10e65464766ef6683Chris Lattner      CategoryList(0), PropertyDecl(0), NumPropertyDecl(0),
24982a5fe3d1cf204b672cdab24d72275b6ad2c3527Fariborz Jahanian      ForwardDecl(FD), InternalInterface(isInternal) {
250980e508ca70d6de75d2abfd96b4681fc98bb2698Steve Naroff        AllocIntfRefProtocols(numRefProtos);
251980e508ca70d6de75d2abfd96b4681fc98bb2698Steve Naroff      }
2520e77ba0bf769e2e5a4a93c079f241b02aeb3ef93Chris Lattnerpublic:
2530e77ba0bf769e2e5a4a93c079f241b02aeb3ef93Chris Lattner
2540ed844b04ea4387caa4e1cf3dc375d269657536bChris Lattner  static ObjCInterfaceDecl *Create(ASTContext &C,
2550ed844b04ea4387caa4e1cf3dc375d269657536bChris Lattner                                   SourceLocation atLoc,
2560e77ba0bf769e2e5a4a93c079f241b02aeb3ef93Chris Lattner                                   unsigned numRefProtos, IdentifierInfo *Id,
2570e77ba0bf769e2e5a4a93c079f241b02aeb3ef93Chris Lattner                                   bool ForwardDecl = false,
2580e77ba0bf769e2e5a4a93c079f241b02aeb3ef93Chris Lattner                                   bool isInternal = false);
259a5997c4f36e0f5aa44623a5b1e3b914760d1ec68Steve Naroff
260a5997c4f36e0f5aa44623a5b1e3b914760d1ec68Steve Naroff  // This is necessary when converting a forward declaration to a definition.
261980e508ca70d6de75d2abfd96b4681fc98bb2698Steve Naroff  void AllocIntfRefProtocols(unsigned numRefProtos) {
262980e508ca70d6de75d2abfd96b4681fc98bb2698Steve Naroff    if (numRefProtos) {
263a526c5c67e5a0473c340903ee542ce570119665fTed Kremenek      ReferencedProtocols = new ObjCProtocolDecl*[numRefProtos];
2647ed9e0f97f4645edc5d4670385b985ea4c617ce7Fariborz Jahanian      memset(ReferencedProtocols, '\0',
265a526c5c67e5a0473c340903ee542ce570119665fTed Kremenek             numRefProtos*sizeof(ObjCProtocolDecl*));
2667ed9e0f97f4645edc5d4670385b985ea4c617ce7Fariborz Jahanian      NumReferencedProtocols = numRefProtos;
267980e508ca70d6de75d2abfd96b4681fc98bb2698Steve Naroff    }
268980e508ca70d6de75d2abfd96b4681fc98bb2698Steve Naroff  }
269980e508ca70d6de75d2abfd96b4681fc98bb2698Steve Naroff
270a526c5c67e5a0473c340903ee542ce570119665fTed Kremenek  ObjCProtocolDecl **getReferencedProtocols() const {
2717ed9e0f97f4645edc5d4670385b985ea4c617ce7Fariborz Jahanian    return ReferencedProtocols;
2727ed9e0f97f4645edc5d4670385b985ea4c617ce7Fariborz Jahanian  }
273c395bda57a0b2f6d8433a462caa780a7dcb7307bFariborz Jahanian  unsigned getNumIntfRefProtocols() const { return NumReferencedProtocols; }
274980e508ca70d6de75d2abfd96b4681fc98bb2698Steve Naroff
275a526c5c67e5a0473c340903ee542ce570119665fTed Kremenek  typedef ObjCIvarDecl * const *ivar_iterator;
276be6df088d69bca0e99c7845a6cd8c1ca85034f31Chris Lattner  ivar_iterator ivar_begin() const { return Ivars; }
2775564e07af5e62875f3b83fc2e5a8823588b5adeaChris Lattner  ivar_iterator ivar_end() const { return Ivars + ivar_size();}
278f3a7af901231535191d14cb524d58f22907ac4d2Chris Lattner  unsigned ivar_size() const { return NumIvars; }
279be6df088d69bca0e99c7845a6cd8c1ca85034f31Chris Lattner
280b5feb35360526da42ad7e3fa7ef3d44a5992b83cChris Lattner  unsigned getNumInstanceMethods() const { return NumInstanceMethods; }
28162db2f4214c1589082960f12c9cb8924fe0cf8c5Chris Lattner  unsigned getNumClassMethods() const { return NumClassMethods; }
282980e508ca70d6de75d2abfd96b4681fc98bb2698Steve Naroff
283a526c5c67e5a0473c340903ee542ce570119665fTed Kremenek  typedef ObjCMethodDecl * const * instmeth_iterator;
2844c52509ffb9fdd6c93aa7b50812e316f1d920a26Chris Lattner  instmeth_iterator instmeth_begin() const { return InstanceMethods; }
2854c52509ffb9fdd6c93aa7b50812e316f1d920a26Chris Lattner  instmeth_iterator instmeth_end() const {
286b5feb35360526da42ad7e3fa7ef3d44a5992b83cChris Lattner    return InstanceMethods+NumInstanceMethods;
2874c52509ffb9fdd6c93aa7b50812e316f1d920a26Chris Lattner  }
2884c52509ffb9fdd6c93aa7b50812e316f1d920a26Chris Lattner
289a526c5c67e5a0473c340903ee542ce570119665fTed Kremenek  typedef ObjCMethodDecl * const * classmeth_iterator;
2904c52509ffb9fdd6c93aa7b50812e316f1d920a26Chris Lattner  classmeth_iterator classmeth_begin() const { return ClassMethods; }
2914c52509ffb9fdd6c93aa7b50812e316f1d920a26Chris Lattner  classmeth_iterator classmeth_end() const {
29262db2f4214c1589082960f12c9cb8924fe0cf8c5Chris Lattner    return ClassMethods+NumClassMethods;
2934c52509ffb9fdd6c93aa7b50812e316f1d920a26Chris Lattner  }
2944c52509ffb9fdd6c93aa7b50812e316f1d920a26Chris Lattner
295a526c5c67e5a0473c340903ee542ce570119665fTed Kremenek  void addInstanceVariablesToClass(ObjCIvarDecl **ivars, unsigned numIvars,
29660fcceeedbfc8b4a99cb942e2bc5aeb9e2f92a1fSteve Naroff                                   SourceLocation RBracLoc);
297980e508ca70d6de75d2abfd96b4681fc98bb2698Steve Naroff
298a526c5c67e5a0473c340903ee542ce570119665fTed Kremenek  void addMethods(ObjCMethodDecl **insMethods, unsigned numInsMembers,
299a526c5c67e5a0473c340903ee542ce570119665fTed Kremenek                  ObjCMethodDecl **clsMethods, unsigned numClsMembers,
30060fcceeedbfc8b4a99cb942e2bc5aeb9e2f92a1fSteve Naroff                  SourceLocation AtEnd);
301980e508ca70d6de75d2abfd96b4681fc98bb2698Steve Naroff
30255d13b4d5530a14d5baa72adab32ae78ba256cafChris Lattner  void addProperties(ObjCPropertyDecl **Properties, unsigned NumProperties);
30355d13b4d5530a14d5baa72adab32ae78ba256cafChris Lattner
30455d13b4d5530a14d5baa72adab32ae78ba256cafChris Lattner
305768f26ee5892cd63ff0335a15d71a2385ba7c5eaSteve Naroff  bool isForwardDecl() const { return ForwardDecl; }
306768f26ee5892cd63ff0335a15d71a2385ba7c5eaSteve Naroff  void setForwardDecl(bool val) { ForwardDecl = val; }
307980e508ca70d6de75d2abfd96b4681fc98bb2698Steve Naroff
308a526c5c67e5a0473c340903ee542ce570119665fTed Kremenek  void setIntfRefProtocols(unsigned idx, ObjCProtocolDecl *OID) {
3097ed9e0f97f4645edc5d4670385b985ea4c617ce7Fariborz Jahanian    assert((idx < NumReferencedProtocols) && "index out of range");
3107ed9e0f97f4645edc5d4670385b985ea4c617ce7Fariborz Jahanian    ReferencedProtocols[idx] = OID;
311980e508ca70d6de75d2abfd96b4681fc98bb2698Steve Naroff  }
312980e508ca70d6de75d2abfd96b4681fc98bb2698Steve Naroff
313a526c5c67e5a0473c340903ee542ce570119665fTed Kremenek  ObjCInterfaceDecl *getSuperClass() const { return SuperClass; }
314a526c5c67e5a0473c340903ee542ce570119665fTed Kremenek  void setSuperClass(ObjCInterfaceDecl * superCls) { SuperClass = superCls; }
315980e508ca70d6de75d2abfd96b4681fc98bb2698Steve Naroff
316a526c5c67e5a0473c340903ee542ce570119665fTed Kremenek  ObjCCategoryDecl* getCategoryList() const { return CategoryList; }
317a526c5c67e5a0473c340903ee542ce570119665fTed Kremenek  void setCategoryList(ObjCCategoryDecl *category) {
3183d58138992b9bc7b34aaa680f3ddf3971292eb7dSteve Naroff         CategoryList = category;
319980e508ca70d6de75d2abfd96b4681fc98bb2698Steve Naroff  }
320a526c5c67e5a0473c340903ee542ce570119665fTed Kremenek  ObjCIvarDecl *lookupInstanceVariable(IdentifierInfo *ivarName,
321a526c5c67e5a0473c340903ee542ce570119665fTed Kremenek                                       ObjCInterfaceDecl *&clsDeclared);
322c81c8144a661a49d7b9dae8d2080dee2e43186ecChris Lattner
32358dbdeb69c063f82d644504fc638120198f7fad2Steve Naroff  // Get the local instance method declared in this interface.
324a526c5c67e5a0473c340903ee542ce570119665fTed Kremenek  ObjCMethodDecl *getInstanceMethod(Selector &Sel) {
32558dbdeb69c063f82d644504fc638120198f7fad2Steve Naroff    for (instmeth_iterator I = instmeth_begin(), E = instmeth_end();
3260e77ba0bf769e2e5a4a93c079f241b02aeb3ef93Chris Lattner         I != E; ++I) {
32758dbdeb69c063f82d644504fc638120198f7fad2Steve Naroff      if ((*I)->getSelector() == Sel)
32858dbdeb69c063f82d644504fc638120198f7fad2Steve Naroff        return *I;
32958dbdeb69c063f82d644504fc638120198f7fad2Steve Naroff    }
3300e77ba0bf769e2e5a4a93c079f241b02aeb3ef93Chris Lattner    return 0;
33158dbdeb69c063f82d644504fc638120198f7fad2Steve Naroff  }
33258dbdeb69c063f82d644504fc638120198f7fad2Steve Naroff  // Get the local class method declared in this interface.
333a526c5c67e5a0473c340903ee542ce570119665fTed Kremenek  ObjCMethodDecl *getClassMethod(Selector &Sel) {
33458dbdeb69c063f82d644504fc638120198f7fad2Steve Naroff    for (classmeth_iterator I = classmeth_begin(), E = classmeth_end();
3350e77ba0bf769e2e5a4a93c079f241b02aeb3ef93Chris Lattner         I != E; ++I) {
33658dbdeb69c063f82d644504fc638120198f7fad2Steve Naroff      if ((*I)->getSelector() == Sel)
33758dbdeb69c063f82d644504fc638120198f7fad2Steve Naroff        return *I;
33858dbdeb69c063f82d644504fc638120198f7fad2Steve Naroff    }
3390e77ba0bf769e2e5a4a93c079f241b02aeb3ef93Chris Lattner    return 0;
34058dbdeb69c063f82d644504fc638120198f7fad2Steve Naroff  }
34194a5c3334bba3cc8cd1da85ba1118bc2c080add9Steve Naroff  // Lookup a method. First, we search locally. If a method isn't
34294a5c3334bba3cc8cd1da85ba1118bc2c080add9Steve Naroff  // found, we search referenced protocols and class categories.
343a526c5c67e5a0473c340903ee542ce570119665fTed Kremenek  ObjCMethodDecl *lookupInstanceMethod(Selector Sel);
344a526c5c67e5a0473c340903ee542ce570119665fTed Kremenek  ObjCMethodDecl *lookupClassMethod(Selector Sel);
34560fcceeedbfc8b4a99cb942e2bc5aeb9e2f92a1fSteve Naroff
346f908a87299d278164540f90b5b6e6cab7b14fb41Steve Naroff  // Location information, modeled after the Stmt API.
34760fcceeedbfc8b4a99cb942e2bc5aeb9e2f92a1fSteve Naroff  SourceLocation getLocStart() const { return getLocation(); } // '@'interface
348f908a87299d278164540f90b5b6e6cab7b14fb41Steve Naroff  SourceLocation getLocEnd() const { return EndLoc; }
349f908a87299d278164540f90b5b6e6cab7b14fb41Steve Naroff  void setLocEnd(SourceLocation LE) { EndLoc = LE; };
35060fcceeedbfc8b4a99cb942e2bc5aeb9e2f92a1fSteve Naroff
35160fcceeedbfc8b4a99cb942e2bc5aeb9e2f92a1fSteve Naroff  // We also need to record the @end location.
352f908a87299d278164540f90b5b6e6cab7b14fb41Steve Naroff  SourceLocation getAtEndLoc() const { return AtEndLoc; }
35382a5fe3d1cf204b672cdab24d72275b6ad2c3527Fariborz Jahanian
354f4af5154571e0c5eadb19df10e65464766ef6683Chris Lattner  unsigned getNumPropertyDecl() const { return NumPropertyDecl; }
35582a5fe3d1cf204b672cdab24d72275b6ad2c3527Fariborz Jahanian
35655d13b4d5530a14d5baa72adab32ae78ba256cafChris Lattner  ObjCPropertyDecl * const * getPropertyDecl() const { return PropertyDecl; }
357a526c5c67e5a0473c340903ee542ce570119665fTed Kremenek  ObjCPropertyDecl **getPropertyDecl() { return PropertyDecl; }
358f908a87299d278164540f90b5b6e6cab7b14fb41Steve Naroff
3594b6df3fa953267c5d755628c8afd818bb571e579Fariborz Jahanian  /// ImplicitInterfaceDecl - check that this is an implicitely declared
360a526c5c67e5a0473c340903ee542ce570119665fTed Kremenek  /// ObjCInterfaceDecl node. This is for legacy objective-c @implementation
3614b6df3fa953267c5d755628c8afd818bb571e579Fariborz Jahanian  /// declaration without an @interface declaration.
3623a3ca1b35a7121aea0bf465a192dce748465e10fFariborz Jahanian  bool ImplicitInterfaceDecl() const { return InternalInterface; }
3634b6df3fa953267c5d755628c8afd818bb571e579Fariborz Jahanian
364a526c5c67e5a0473c340903ee542ce570119665fTed Kremenek  static bool classof(const Decl *D) { return D->getKind() == ObjCInterface; }
365a526c5c67e5a0473c340903ee542ce570119665fTed Kremenek  static bool classof(const ObjCInterfaceDecl *D) { return true; }
366980e508ca70d6de75d2abfd96b4681fc98bb2698Steve Naroff};
367980e508ca70d6de75d2abfd96b4681fc98bb2698Steve Naroff
368a526c5c67e5a0473c340903ee542ce570119665fTed Kremenek/// ObjCIvarDecl - Represents an ObjC instance variable. In general, ObjC
3690c6b6243d3efd958c17943130e2a773653511edcSteve Naroff/// instance variables are identical to C. The only exception is Objective-C
3700c6b6243d3efd958c17943130e2a773653511edcSteve Naroff/// supports C++ style access control. For example:
3710c6b6243d3efd958c17943130e2a773653511edcSteve Naroff///
3720c6b6243d3efd958c17943130e2a773653511edcSteve Naroff///   @interface IvarExample : NSObject
3730c6b6243d3efd958c17943130e2a773653511edcSteve Naroff///   {
3740c6b6243d3efd958c17943130e2a773653511edcSteve Naroff///     id defaultToPrivate; // same as C++.
3750c6b6243d3efd958c17943130e2a773653511edcSteve Naroff///   @public:
3760c6b6243d3efd958c17943130e2a773653511edcSteve Naroff///     id canBePublic; // same as C++.
3770c6b6243d3efd958c17943130e2a773653511edcSteve Naroff///   @protected:
3780c6b6243d3efd958c17943130e2a773653511edcSteve Naroff///     id canBeProtected; // same as C++.
3790c6b6243d3efd958c17943130e2a773653511edcSteve Naroff///   @package:
3800c6b6243d3efd958c17943130e2a773653511edcSteve Naroff///     id canBePackage; // framework visibility (not available in C++).
3810c6b6243d3efd958c17943130e2a773653511edcSteve Naroff///   }
3820c6b6243d3efd958c17943130e2a773653511edcSteve Naroff///
383a526c5c67e5a0473c340903ee542ce570119665fTed Kremenekclass ObjCIvarDecl : public FieldDecl {
384b048c9835969c4f7fe06264748be18ed4b442116Chris Lattner  ObjCIvarDecl(SourceLocation L, IdentifierInfo *Id, QualType T)
385b048c9835969c4f7fe06264748be18ed4b442116Chris Lattner    : FieldDecl(ObjCIvar, L, Id, T) {}
3860e77ba0bf769e2e5a4a93c079f241b02aeb3ef93Chris Lattnerpublic:
387b048c9835969c4f7fe06264748be18ed4b442116Chris Lattner  static ObjCIvarDecl *Create(ASTContext &C, SourceLocation L,
3880e77ba0bf769e2e5a4a93c079f241b02aeb3ef93Chris Lattner                              IdentifierInfo *Id, QualType T);
389980e508ca70d6de75d2abfd96b4681fc98bb2698Steve Naroff
390980e508ca70d6de75d2abfd96b4681fc98bb2698Steve Naroff  enum AccessControl {
391980e508ca70d6de75d2abfd96b4681fc98bb2698Steve Naroff    None, Private, Protected, Public, Package
392980e508ca70d6de75d2abfd96b4681fc98bb2698Steve Naroff  };
393980e508ca70d6de75d2abfd96b4681fc98bb2698Steve Naroff  void setAccessControl(AccessControl ac) { DeclAccess = ac; }
394ca63fa00786e51c207c829f4182f11a6c6b552beTed Kremenek  AccessControl getAccessControl() const { return AccessControl(DeclAccess); }
395980e508ca70d6de75d2abfd96b4681fc98bb2698Steve Naroff
396980e508ca70d6de75d2abfd96b4681fc98bb2698Steve Naroff  // Implement isa/cast/dyncast/etc.
397a526c5c67e5a0473c340903ee542ce570119665fTed Kremenek  static bool classof(const Decl *D) { return D->getKind() == ObjCIvar; }
398a526c5c67e5a0473c340903ee542ce570119665fTed Kremenek  static bool classof(const ObjCIvarDecl *D) { return true; }
399980e508ca70d6de75d2abfd96b4681fc98bb2698Steve Naroffprivate:
400ca63fa00786e51c207c829f4182f11a6c6b552beTed Kremenek  // NOTE: VC++ treats enums as signed, avoid using the AccessControl enum
401ca63fa00786e51c207c829f4182f11a6c6b552beTed Kremenek  unsigned DeclAccess : 3;
402980e508ca70d6de75d2abfd96b4681fc98bb2698Steve Naroff};
403980e508ca70d6de75d2abfd96b4681fc98bb2698Steve Naroff
404980e508ca70d6de75d2abfd96b4681fc98bb2698Steve Naroff
405a526c5c67e5a0473c340903ee542ce570119665fTed Kremenek/// ObjCProtocolDecl - Represents a protocol declaration. ObjC protocols
4060c6b6243d3efd958c17943130e2a773653511edcSteve Naroff/// declare a pure abstract type (i.e no instance variables are permitted).
4070c6b6243d3efd958c17943130e2a773653511edcSteve Naroff/// Protocols orginally drew inspiration from C++ pure virtual functions (a C++
4080c6b6243d3efd958c17943130e2a773653511edcSteve Naroff/// feature with nice semantics and lousy syntax:-). Here is an example:
4090c6b6243d3efd958c17943130e2a773653511edcSteve Naroff///
4100c6b6243d3efd958c17943130e2a773653511edcSteve Naroff/// @protocol NSDraggingInfo
4110c6b6243d3efd958c17943130e2a773653511edcSteve Naroff/// - (NSWindow *)draggingDestinationWindow;
4120c6b6243d3efd958c17943130e2a773653511edcSteve Naroff/// - (NSImage *)draggedImage;
4130c6b6243d3efd958c17943130e2a773653511edcSteve Naroff/// @end
4140c6b6243d3efd958c17943130e2a773653511edcSteve Naroff///
4150c6b6243d3efd958c17943130e2a773653511edcSteve Naroff/// @interface ImplementsNSDraggingInfo : NSObject <NSDraggingInfo>
4160c6b6243d3efd958c17943130e2a773653511edcSteve Naroff/// @end
4170c6b6243d3efd958c17943130e2a773653511edcSteve Naroff///
418a526c5c67e5a0473c340903ee542ce570119665fTed Kremenek/// ObjC protocols inspired Java interfaces. Unlike Java, ObjC classes and
4190c6b6243d3efd958c17943130e2a773653511edcSteve Naroff/// protocols are in distinct namespaces. For example, Cocoa defines both
4200c6b6243d3efd958c17943130e2a773653511edcSteve Naroff/// an NSObject protocol and class (which isn't allowed in Java). As a result,
4210c6b6243d3efd958c17943130e2a773653511edcSteve Naroff/// protocols are referenced using angle brackets as follows:
4220c6b6243d3efd958c17943130e2a773653511edcSteve Naroff///
4230c6b6243d3efd958c17943130e2a773653511edcSteve Naroff/// id <NSDraggingInfo> anyObjectThatImplementsNSDraggingInfo;
4240c6b6243d3efd958c17943130e2a773653511edcSteve Naroff///
425a526c5c67e5a0473c340903ee542ce570119665fTed Kremenekclass ObjCProtocolDecl : public NamedDecl {
426980e508ca70d6de75d2abfd96b4681fc98bb2698Steve Naroff  /// referenced protocols
427a526c5c67e5a0473c340903ee542ce570119665fTed Kremenek  ObjCProtocolDecl **ReferencedProtocols;  // Null if none
428c395bda57a0b2f6d8433a462caa780a7dcb7307bFariborz Jahanian  unsigned NumReferencedProtocols;  // 0 if none
429980e508ca70d6de75d2abfd96b4681fc98bb2698Steve Naroff
430980e508ca70d6de75d2abfd96b4681fc98bb2698Steve Naroff  /// protocol instance methods
431a526c5c67e5a0473c340903ee542ce570119665fTed Kremenek  ObjCMethodDecl **InstanceMethods;  // Null if not defined
43262db2f4214c1589082960f12c9cb8924fe0cf8c5Chris Lattner  unsigned NumInstanceMethods;  // 0 if none
433980e508ca70d6de75d2abfd96b4681fc98bb2698Steve Naroff
434980e508ca70d6de75d2abfd96b4681fc98bb2698Steve Naroff  /// protocol class methods
435a526c5c67e5a0473c340903ee542ce570119665fTed Kremenek  ObjCMethodDecl **ClassMethods;  // Null if not defined
43662db2f4214c1589082960f12c9cb8924fe0cf8c5Chris Lattner  unsigned NumClassMethods;  // 0 if none
437980e508ca70d6de75d2abfd96b4681fc98bb2698Steve Naroff
438980e508ca70d6de75d2abfd96b4681fc98bb2698Steve Naroff  bool isForwardProtoDecl; // declared with @protocol.
439423cb565abc681b770fb4b9b4bc24d398c98157bSteve Naroff
440423cb565abc681b770fb4b9b4bc24d398c98157bSteve Naroff  SourceLocation EndLoc; // marks the '>' or identifier.
441423cb565abc681b770fb4b9b4bc24d398c98157bSteve Naroff  SourceLocation AtEndLoc; // marks the end of the entire interface.
442cca59d77c4b84fd2da268018dbaf9431a621e75bChris Lattner
443c858105d41602a2dadb2efbc1af80a7b791ebac3Chris Lattner  ObjCProtocolDecl(SourceLocation L, unsigned numRefProtos, IdentifierInfo *Id)
444a526c5c67e5a0473c340903ee542ce570119665fTed Kremenek    : NamedDecl(ObjCProtocol, L, Id),
445c395bda57a0b2f6d8433a462caa780a7dcb7307bFariborz Jahanian      ReferencedProtocols(0), NumReferencedProtocols(0),
44662db2f4214c1589082960f12c9cb8924fe0cf8c5Chris Lattner      InstanceMethods(0), NumInstanceMethods(0),
44762db2f4214c1589082960f12c9cb8924fe0cf8c5Chris Lattner      ClassMethods(0), NumClassMethods(0),
448c858105d41602a2dadb2efbc1af80a7b791ebac3Chris Lattner      isForwardProtoDecl(true) {
449cca59d77c4b84fd2da268018dbaf9431a621e75bChris Lattner    AllocReferencedProtocols(numRefProtos);
450cca59d77c4b84fd2da268018dbaf9431a621e75bChris Lattner  }
451cca59d77c4b84fd2da268018dbaf9431a621e75bChris Lattnerpublic:
452cca59d77c4b84fd2da268018dbaf9431a621e75bChris Lattner  static ObjCProtocolDecl *Create(ASTContext &C, SourceLocation L,
453c858105d41602a2dadb2efbc1af80a7b791ebac3Chris Lattner                                  unsigned numRefProtos, IdentifierInfo *Id);
454cca59d77c4b84fd2da268018dbaf9431a621e75bChris Lattner
455980e508ca70d6de75d2abfd96b4681fc98bb2698Steve Naroff  void AllocReferencedProtocols(unsigned numRefProtos) {
456980e508ca70d6de75d2abfd96b4681fc98bb2698Steve Naroff    if (numRefProtos) {
457a526c5c67e5a0473c340903ee542ce570119665fTed Kremenek      ReferencedProtocols = new ObjCProtocolDecl*[numRefProtos];
458980e508ca70d6de75d2abfd96b4681fc98bb2698Steve Naroff      memset(ReferencedProtocols, '\0',
459a526c5c67e5a0473c340903ee542ce570119665fTed Kremenek             numRefProtos*sizeof(ObjCProtocolDecl*));
460980e508ca70d6de75d2abfd96b4681fc98bb2698Steve Naroff      NumReferencedProtocols = numRefProtos;
461980e508ca70d6de75d2abfd96b4681fc98bb2698Steve Naroff    }
462980e508ca70d6de75d2abfd96b4681fc98bb2698Steve Naroff  }
463a526c5c67e5a0473c340903ee542ce570119665fTed Kremenek  void addMethods(ObjCMethodDecl **insMethods, unsigned numInsMembers,
464a526c5c67e5a0473c340903ee542ce570119665fTed Kremenek                  ObjCMethodDecl **clsMethods, unsigned numClsMembers,
46560fcceeedbfc8b4a99cb942e2bc5aeb9e2f92a1fSteve Naroff                  SourceLocation AtEndLoc);
466980e508ca70d6de75d2abfd96b4681fc98bb2698Steve Naroff
467a526c5c67e5a0473c340903ee542ce570119665fTed Kremenek  void setReferencedProtocols(unsigned idx, ObjCProtocolDecl *OID) {
468980e508ca70d6de75d2abfd96b4681fc98bb2698Steve Naroff    assert((idx < NumReferencedProtocols) && "index out of range");
469980e508ca70d6de75d2abfd96b4681fc98bb2698Steve Naroff    ReferencedProtocols[idx] = OID;
470980e508ca70d6de75d2abfd96b4681fc98bb2698Steve Naroff  }
471980e508ca70d6de75d2abfd96b4681fc98bb2698Steve Naroff
472a526c5c67e5a0473c340903ee542ce570119665fTed Kremenek  ObjCProtocolDecl** getReferencedProtocols() const {
473980e508ca70d6de75d2abfd96b4681fc98bb2698Steve Naroff    return ReferencedProtocols;
474980e508ca70d6de75d2abfd96b4681fc98bb2698Steve Naroff  }
475c395bda57a0b2f6d8433a462caa780a7dcb7307bFariborz Jahanian  unsigned getNumReferencedProtocols() const { return NumReferencedProtocols; }
476c858105d41602a2dadb2efbc1af80a7b791ebac3Chris Lattner  unsigned getNumInstanceMethods() const { return NumInstanceMethods; }
47762db2f4214c1589082960f12c9cb8924fe0cf8c5Chris Lattner  unsigned getNumClassMethods() const { return NumClassMethods; }
47858dbdeb69c063f82d644504fc638120198f7fad2Steve Naroff
479a526c5c67e5a0473c340903ee542ce570119665fTed Kremenek  typedef ObjCMethodDecl * const * instmeth_iterator;
48058dbdeb69c063f82d644504fc638120198f7fad2Steve Naroff  instmeth_iterator instmeth_begin() const { return InstanceMethods; }
48158dbdeb69c063f82d644504fc638120198f7fad2Steve Naroff  instmeth_iterator instmeth_end() const {
48262db2f4214c1589082960f12c9cb8924fe0cf8c5Chris Lattner    return InstanceMethods+NumInstanceMethods;
48358dbdeb69c063f82d644504fc638120198f7fad2Steve Naroff  }
48458dbdeb69c063f82d644504fc638120198f7fad2Steve Naroff
485a526c5c67e5a0473c340903ee542ce570119665fTed Kremenek  typedef ObjCMethodDecl * const * classmeth_iterator;
48658dbdeb69c063f82d644504fc638120198f7fad2Steve Naroff  classmeth_iterator classmeth_begin() const { return ClassMethods; }
48758dbdeb69c063f82d644504fc638120198f7fad2Steve Naroff  classmeth_iterator classmeth_end() const {
48862db2f4214c1589082960f12c9cb8924fe0cf8c5Chris Lattner    return ClassMethods+NumClassMethods;
48958dbdeb69c063f82d644504fc638120198f7fad2Steve Naroff  }
49058dbdeb69c063f82d644504fc638120198f7fad2Steve Naroff
49158dbdeb69c063f82d644504fc638120198f7fad2Steve Naroff  // Get the local instance method declared in this interface.
49262db2f4214c1589082960f12c9cb8924fe0cf8c5Chris Lattner  ObjCMethodDecl *getInstanceMethod(Selector Sel) {
49362db2f4214c1589082960f12c9cb8924fe0cf8c5Chris Lattner    for (instmeth_iterator I = instmeth_begin(), E = instmeth_end();
49462db2f4214c1589082960f12c9cb8924fe0cf8c5Chris Lattner         I != E; ++I) {
49558dbdeb69c063f82d644504fc638120198f7fad2Steve Naroff      if ((*I)->getSelector() == Sel)
49658dbdeb69c063f82d644504fc638120198f7fad2Steve Naroff        return *I;
49758dbdeb69c063f82d644504fc638120198f7fad2Steve Naroff    }
49862db2f4214c1589082960f12c9cb8924fe0cf8c5Chris Lattner    return 0;
49958dbdeb69c063f82d644504fc638120198f7fad2Steve Naroff  }
50058dbdeb69c063f82d644504fc638120198f7fad2Steve Naroff  // Get the local class method declared in this interface.
50162db2f4214c1589082960f12c9cb8924fe0cf8c5Chris Lattner  ObjCMethodDecl *getClassMethod(Selector Sel) {
50258dbdeb69c063f82d644504fc638120198f7fad2Steve Naroff    for (classmeth_iterator I = classmeth_begin(), E = classmeth_end();
50362db2f4214c1589082960f12c9cb8924fe0cf8c5Chris Lattner         I != E; ++I) {
50458dbdeb69c063f82d644504fc638120198f7fad2Steve Naroff      if ((*I)->getSelector() == Sel)
50558dbdeb69c063f82d644504fc638120198f7fad2Steve Naroff        return *I;
50658dbdeb69c063f82d644504fc638120198f7fad2Steve Naroff    }
50762db2f4214c1589082960f12c9cb8924fe0cf8c5Chris Lattner    return 0;
50858dbdeb69c063f82d644504fc638120198f7fad2Steve Naroff  }
509980e508ca70d6de75d2abfd96b4681fc98bb2698Steve Naroff
51094a5c3334bba3cc8cd1da85ba1118bc2c080add9Steve Naroff  // Lookup a method. First, we search locally. If a method isn't
51194a5c3334bba3cc8cd1da85ba1118bc2c080add9Steve Naroff  // found, we search referenced protocols and class categories.
512a526c5c67e5a0473c340903ee542ce570119665fTed Kremenek  ObjCMethodDecl *lookupInstanceMethod(Selector Sel);
513a526c5c67e5a0473c340903ee542ce570119665fTed Kremenek  ObjCMethodDecl *lookupClassMethod(Selector Sel);
5147dd82836dca87cf828ce994f161b53a34f6cdb7eFariborz Jahanian
515768f26ee5892cd63ff0335a15d71a2385ba7c5eaSteve Naroff  bool isForwardDecl() const { return isForwardProtoDecl; }
516768f26ee5892cd63ff0335a15d71a2385ba7c5eaSteve Naroff  void setForwardDecl(bool val) { isForwardProtoDecl = val; }
517980e508ca70d6de75d2abfd96b4681fc98bb2698Steve Naroff
518423cb565abc681b770fb4b9b4bc24d398c98157bSteve Naroff  // Location information, modeled after the Stmt API.
519423cb565abc681b770fb4b9b4bc24d398c98157bSteve Naroff  SourceLocation getLocStart() const { return getLocation(); } // '@'protocol
520423cb565abc681b770fb4b9b4bc24d398c98157bSteve Naroff  SourceLocation getLocEnd() const { return EndLoc; }
521423cb565abc681b770fb4b9b4bc24d398c98157bSteve Naroff  void setLocEnd(SourceLocation LE) { EndLoc = LE; };
522423cb565abc681b770fb4b9b4bc24d398c98157bSteve Naroff
523423cb565abc681b770fb4b9b4bc24d398c98157bSteve Naroff  // We also need to record the @end location.
524423cb565abc681b770fb4b9b4bc24d398c98157bSteve Naroff  SourceLocation getAtEndLoc() const { return AtEndLoc; }
525423cb565abc681b770fb4b9b4bc24d398c98157bSteve Naroff
526a526c5c67e5a0473c340903ee542ce570119665fTed Kremenek  static bool classof(const Decl *D) { return D->getKind() == ObjCProtocol; }
527a526c5c67e5a0473c340903ee542ce570119665fTed Kremenek  static bool classof(const ObjCProtocolDecl *D) { return true; }
528980e508ca70d6de75d2abfd96b4681fc98bb2698Steve Naroff};
529980e508ca70d6de75d2abfd96b4681fc98bb2698Steve Naroff
530a526c5c67e5a0473c340903ee542ce570119665fTed Kremenek/// ObjCClassDecl - Specifies a list of forward class declarations. For example:
53106ae8d68ef258ccd40c9cd1ce762eaae6f3d4432Steve Naroff///
53206ae8d68ef258ccd40c9cd1ce762eaae6f3d4432Steve Naroff/// @class NSCursor, NSImage, NSPasteboard, NSWindow;
5330c6b6243d3efd958c17943130e2a773653511edcSteve Naroff///
534a526c5c67e5a0473c340903ee542ce570119665fTed Kremenekclass ObjCClassDecl : public Decl {
535a526c5c67e5a0473c340903ee542ce570119665fTed Kremenek  ObjCInterfaceDecl **ForwardDecls;
5367e620729fcb5a0042107f999dcf524f7357ce819Chris Lattner  unsigned NumForwardDecls;
53761f9d41036e30ff80130f99b31c0626e3ef057ccChris Lattner
538a526c5c67e5a0473c340903ee542ce570119665fTed Kremenek  ObjCClassDecl(SourceLocation L, ObjCInterfaceDecl **Elts, unsigned nElts)
539a526c5c67e5a0473c340903ee542ce570119665fTed Kremenek    : Decl(ObjCClass, L) {
54006ae8d68ef258ccd40c9cd1ce762eaae6f3d4432Steve Naroff    if (nElts) {
541a526c5c67e5a0473c340903ee542ce570119665fTed Kremenek      ForwardDecls = new ObjCInterfaceDecl*[nElts];
542a526c5c67e5a0473c340903ee542ce570119665fTed Kremenek      memcpy(ForwardDecls, Elts, nElts*sizeof(ObjCInterfaceDecl*));
5437e620729fcb5a0042107f999dcf524f7357ce819Chris Lattner    } else {
5447e620729fcb5a0042107f999dcf524f7357ce819Chris Lattner      ForwardDecls = 0;
54506ae8d68ef258ccd40c9cd1ce762eaae6f3d4432Steve Naroff    }
54606ae8d68ef258ccd40c9cd1ce762eaae6f3d4432Steve Naroff    NumForwardDecls = nElts;
54706ae8d68ef258ccd40c9cd1ce762eaae6f3d4432Steve Naroff  }
54861f9d41036e30ff80130f99b31c0626e3ef057ccChris Lattnerpublic:
54961f9d41036e30ff80130f99b31c0626e3ef057ccChris Lattner  static ObjCClassDecl *Create(ASTContext &C, SourceLocation L,
55061f9d41036e30ff80130f99b31c0626e3ef057ccChris Lattner                               ObjCInterfaceDecl **Elts, unsigned nElts);
55161f9d41036e30ff80130f99b31c0626e3ef057ccChris Lattner
552a526c5c67e5a0473c340903ee542ce570119665fTed Kremenek  void setInterfaceDecl(unsigned idx, ObjCInterfaceDecl *OID) {
5537e620729fcb5a0042107f999dcf524f7357ce819Chris Lattner    assert(idx < NumForwardDecls && "index out of range");
55406ae8d68ef258ccd40c9cd1ce762eaae6f3d4432Steve Naroff    ForwardDecls[idx] = OID;
55506ae8d68ef258ccd40c9cd1ce762eaae6f3d4432Steve Naroff  }
556a526c5c67e5a0473c340903ee542ce570119665fTed Kremenek  ObjCInterfaceDecl** getForwardDecls() const { return ForwardDecls; }
5577e15891fc89256fc013bd1003676ad3197b85c25Steve Naroff  int getNumForwardDecls() const { return NumForwardDecls; }
5587e15891fc89256fc013bd1003676ad3197b85c25Steve Naroff
559a526c5c67e5a0473c340903ee542ce570119665fTed Kremenek  static bool classof(const Decl *D) { return D->getKind() == ObjCClass; }
560a526c5c67e5a0473c340903ee542ce570119665fTed Kremenek  static bool classof(const ObjCClassDecl *D) { return true; }
56106ae8d68ef258ccd40c9cd1ce762eaae6f3d4432Steve Naroff};
56206ae8d68ef258ccd40c9cd1ce762eaae6f3d4432Steve Naroff
563a526c5c67e5a0473c340903ee542ce570119665fTed Kremenek/// ObjCForwardProtocolDecl - Specifies a list of forward protocol declarations.
56406ae8d68ef258ccd40c9cd1ce762eaae6f3d4432Steve Naroff/// For example:
56506ae8d68ef258ccd40c9cd1ce762eaae6f3d4432Steve Naroff///
5660c6b6243d3efd958c17943130e2a773653511edcSteve Naroff/// @protocol NSTextInput, NSChangeSpelling, NSDraggingInfo;
5670c6b6243d3efd958c17943130e2a773653511edcSteve Naroff///
568a526c5c67e5a0473c340903ee542ce570119665fTed Kremenekclass ObjCForwardProtocolDecl : public Decl {
569a526c5c67e5a0473c340903ee542ce570119665fTed Kremenek  ObjCProtocolDecl **ReferencedProtocols;
5709fa5e65d08aee1875c5f2a841c8b0b4069bd00e5Chris Lattner  unsigned NumReferencedProtocols;
57161f9d41036e30ff80130f99b31c0626e3ef057ccChris Lattner
5720ed844b04ea4387caa4e1cf3dc375d269657536bChris Lattner  ObjCForwardProtocolDecl(SourceLocation L,
573a526c5c67e5a0473c340903ee542ce570119665fTed Kremenek                          ObjCProtocolDecl **Elts, unsigned nElts)
574a526c5c67e5a0473c340903ee542ce570119665fTed Kremenek  : Decl(ObjCForwardProtocol, L) {
575b97de3eddf0a0f7ff97204e7def6b7e0aa8953f2Chris Lattner    NumReferencedProtocols = nElts;
57606ae8d68ef258ccd40c9cd1ce762eaae6f3d4432Steve Naroff    if (nElts) {
577a526c5c67e5a0473c340903ee542ce570119665fTed Kremenek      ReferencedProtocols = new ObjCProtocolDecl*[nElts];
578a526c5c67e5a0473c340903ee542ce570119665fTed Kremenek      memcpy(ReferencedProtocols, Elts, nElts*sizeof(ObjCProtocolDecl*));
5799fa5e65d08aee1875c5f2a841c8b0b4069bd00e5Chris Lattner    } else {
5809fa5e65d08aee1875c5f2a841c8b0b4069bd00e5Chris Lattner      ReferencedProtocols = 0;
581980e508ca70d6de75d2abfd96b4681fc98bb2698Steve Naroff    }
58206ae8d68ef258ccd40c9cd1ce762eaae6f3d4432Steve Naroff  }
58361f9d41036e30ff80130f99b31c0626e3ef057ccChris Lattnerpublic:
58461f9d41036e30ff80130f99b31c0626e3ef057ccChris Lattner  static ObjCForwardProtocolDecl *Create(ASTContext &C, SourceLocation L,
58561f9d41036e30ff80130f99b31c0626e3ef057ccChris Lattner                                         ObjCProtocolDecl **Elts, unsigned Num);
58661f9d41036e30ff80130f99b31c0626e3ef057ccChris Lattner
58761f9d41036e30ff80130f99b31c0626e3ef057ccChris Lattner
588a526c5c67e5a0473c340903ee542ce570119665fTed Kremenek  void setForwardProtocolDecl(unsigned idx, ObjCProtocolDecl *OID) {
5899fa5e65d08aee1875c5f2a841c8b0b4069bd00e5Chris Lattner    assert(idx < NumReferencedProtocols && "index out of range");
5907ed9e0f97f4645edc5d4670385b985ea4c617ce7Fariborz Jahanian    ReferencedProtocols[idx] = OID;
59106ae8d68ef258ccd40c9cd1ce762eaae6f3d4432Steve Naroff  }
5929fa5e65d08aee1875c5f2a841c8b0b4069bd00e5Chris Lattner
5939fa5e65d08aee1875c5f2a841c8b0b4069bd00e5Chris Lattner  unsigned getNumForwardDecls() const { return NumReferencedProtocols; }
5949fa5e65d08aee1875c5f2a841c8b0b4069bd00e5Chris Lattner
595a526c5c67e5a0473c340903ee542ce570119665fTed Kremenek  ObjCProtocolDecl *getForwardProtocolDecl(unsigned idx) {
5969fa5e65d08aee1875c5f2a841c8b0b4069bd00e5Chris Lattner    assert(idx < NumReferencedProtocols && "index out of range");
5979fa5e65d08aee1875c5f2a841c8b0b4069bd00e5Chris Lattner    return ReferencedProtocols[idx];
5989fa5e65d08aee1875c5f2a841c8b0b4069bd00e5Chris Lattner  }
599a526c5c67e5a0473c340903ee542ce570119665fTed Kremenek  const ObjCProtocolDecl *getForwardProtocolDecl(unsigned idx) const {
6009fa5e65d08aee1875c5f2a841c8b0b4069bd00e5Chris Lattner    assert(idx < NumReferencedProtocols && "index out of range");
6019fa5e65d08aee1875c5f2a841c8b0b4069bd00e5Chris Lattner    return ReferencedProtocols[idx];
6029fa5e65d08aee1875c5f2a841c8b0b4069bd00e5Chris Lattner  }
6039fa5e65d08aee1875c5f2a841c8b0b4069bd00e5Chris Lattner
60406ae8d68ef258ccd40c9cd1ce762eaae6f3d4432Steve Naroff  static bool classof(const Decl *D) {
605a526c5c67e5a0473c340903ee542ce570119665fTed Kremenek    return D->getKind() == ObjCForwardProtocol;
60606ae8d68ef258ccd40c9cd1ce762eaae6f3d4432Steve Naroff  }
607a526c5c67e5a0473c340903ee542ce570119665fTed Kremenek  static bool classof(const ObjCForwardProtocolDecl *D) { return true; }
608980e508ca70d6de75d2abfd96b4681fc98bb2698Steve Naroff};
609980e508ca70d6de75d2abfd96b4681fc98bb2698Steve Naroff
610a526c5c67e5a0473c340903ee542ce570119665fTed Kremenek/// ObjCCategoryDecl - Represents a category declaration. A category allows
6110c6b6243d3efd958c17943130e2a773653511edcSteve Naroff/// you to add methods to an existing class (without subclassing or modifying
6120c6b6243d3efd958c17943130e2a773653511edcSteve Naroff/// the original class interface or implementation:-). Categories don't allow
6130c6b6243d3efd958c17943130e2a773653511edcSteve Naroff/// you to add instance data. The following example adds "myMethod" to all
6140c6b6243d3efd958c17943130e2a773653511edcSteve Naroff/// NSView's within a process:
6150c6b6243d3efd958c17943130e2a773653511edcSteve Naroff///
6160c6b6243d3efd958c17943130e2a773653511edcSteve Naroff/// @interface NSView (MyViewMethods)
6170c6b6243d3efd958c17943130e2a773653511edcSteve Naroff/// - myMethod;
6180c6b6243d3efd958c17943130e2a773653511edcSteve Naroff/// @end
6190c6b6243d3efd958c17943130e2a773653511edcSteve Naroff///
6200c6b6243d3efd958c17943130e2a773653511edcSteve Naroff/// Cateogries also allow you to split the implementation of a class across
6210c6b6243d3efd958c17943130e2a773653511edcSteve Naroff/// several files (a feature more naturally supported in C++).
6220c6b6243d3efd958c17943130e2a773653511edcSteve Naroff///
6230c6b6243d3efd958c17943130e2a773653511edcSteve Naroff/// Categories were originally inspired by dynamic languages such as Common
62468c82cf61228102aba1194efef222fa1478af2a8Chris Lattner/// Lisp and Smalltalk.  More traditional class-based languages (C++, Java)
6250c6b6243d3efd958c17943130e2a773653511edcSteve Naroff/// don't support this level of dynamism, which is both powerful and dangerous.
6260c6b6243d3efd958c17943130e2a773653511edcSteve Naroff///
627a526c5c67e5a0473c340903ee542ce570119665fTed Kremenekclass ObjCCategoryDecl : public NamedDecl {
628980e508ca70d6de75d2abfd96b4681fc98bb2698Steve Naroff  /// Interface belonging to this category
629a526c5c67e5a0473c340903ee542ce570119665fTed Kremenek  ObjCInterfaceDecl *ClassInterface;
630980e508ca70d6de75d2abfd96b4681fc98bb2698Steve Naroff
63168c82cf61228102aba1194efef222fa1478af2a8Chris Lattner  /// referenced protocols in this category.
632a526c5c67e5a0473c340903ee542ce570119665fTed Kremenek  ObjCProtocolDecl **ReferencedProtocols;  // Null if none
633c395bda57a0b2f6d8433a462caa780a7dcb7307bFariborz Jahanian  unsigned NumReferencedProtocols;  // 0 if none
634980e508ca70d6de75d2abfd96b4681fc98bb2698Steve Naroff
635980e508ca70d6de75d2abfd96b4681fc98bb2698Steve Naroff  /// category instance methods
636a526c5c67e5a0473c340903ee542ce570119665fTed Kremenek  ObjCMethodDecl **InstanceMethods;  // Null if not defined
637c858105d41602a2dadb2efbc1af80a7b791ebac3Chris Lattner  unsigned NumInstanceMethods;  // 0 if none
638980e508ca70d6de75d2abfd96b4681fc98bb2698Steve Naroff
639980e508ca70d6de75d2abfd96b4681fc98bb2698Steve Naroff  /// category class methods
640a526c5c67e5a0473c340903ee542ce570119665fTed Kremenek  ObjCMethodDecl **ClassMethods;  // Null if not defined
641a906135721c350435319347d2672bbb3bf494f91Chris Lattner  unsigned NumClassMethods;  // 0 if not defined
642980e508ca70d6de75d2abfd96b4681fc98bb2698Steve Naroff
643980e508ca70d6de75d2abfd96b4681fc98bb2698Steve Naroff  /// Next category belonging to this class
644a526c5c67e5a0473c340903ee542ce570119665fTed Kremenek  ObjCCategoryDecl *NextClassCategory;
645ca3adf7e8cac8c9fbaf592b1e5c2be6f082de7baFariborz Jahanian
646423cb565abc681b770fb4b9b4bc24d398c98157bSteve Naroff  SourceLocation EndLoc; // marks the '>' or identifier.
647423cb565abc681b770fb4b9b4bc24d398c98157bSteve Naroff  SourceLocation AtEndLoc; // marks the end of the entire interface.
64861f9d41036e30ff80130f99b31c0626e3ef057ccChris Lattner
64968c82cf61228102aba1194efef222fa1478af2a8Chris Lattner  ObjCCategoryDecl(SourceLocation L, IdentifierInfo *Id)
650a526c5c67e5a0473c340903ee542ce570119665fTed Kremenek    : NamedDecl(ObjCCategory, L, Id),
651c395bda57a0b2f6d8433a462caa780a7dcb7307bFariborz Jahanian      ClassInterface(0), ReferencedProtocols(0), NumReferencedProtocols(0),
652c858105d41602a2dadb2efbc1af80a7b791ebac3Chris Lattner      InstanceMethods(0), NumInstanceMethods(0),
653a906135721c350435319347d2672bbb3bf494f91Chris Lattner      ClassMethods(0), NumClassMethods(0),
654e5ab7f31054e87ee812830480a828a762cd9eb73Chris Lattner      NextClassCategory(0) {
655a906135721c350435319347d2672bbb3bf494f91Chris Lattner  }
65661f9d41036e30ff80130f99b31c0626e3ef057ccChris Lattnerpublic:
65761f9d41036e30ff80130f99b31c0626e3ef057ccChris Lattner
6580ed844b04ea4387caa4e1cf3dc375d269657536bChris Lattner  static ObjCCategoryDecl *Create(ASTContext &C,
6590ed844b04ea4387caa4e1cf3dc375d269657536bChris Lattner                                  SourceLocation L, IdentifierInfo *Id);
66061f9d41036e30ff80130f99b31c0626e3ef057ccChris Lattner
661e0def7589a8afa8a6acef13476bb3f882c104b91Chris Lattner  ObjCInterfaceDecl *getClassInterface() { return ClassInterface; }
662e0def7589a8afa8a6acef13476bb3f882c104b91Chris Lattner  const ObjCInterfaceDecl *getClassInterface() const { return ClassInterface; }
663a526c5c67e5a0473c340903ee542ce570119665fTed Kremenek  void setClassInterface(ObjCInterfaceDecl *IDecl) { ClassInterface = IDecl; }
664980e508ca70d6de75d2abfd96b4681fc98bb2698Steve Naroff
66568c82cf61228102aba1194efef222fa1478af2a8Chris Lattner  void setReferencedProtocolList(ObjCProtocolDecl **List, unsigned NumRPs);
66668c82cf61228102aba1194efef222fa1478af2a8Chris Lattner
667a526c5c67e5a0473c340903ee542ce570119665fTed Kremenek  void setCatReferencedProtocols(unsigned idx, ObjCProtocolDecl *OID) {
6687ed9e0f97f4645edc5d4670385b985ea4c617ce7Fariborz Jahanian    assert((idx < NumReferencedProtocols) && "index out of range");
6697ed9e0f97f4645edc5d4670385b985ea4c617ce7Fariborz Jahanian    ReferencedProtocols[idx] = OID;
670980e508ca70d6de75d2abfd96b4681fc98bb2698Steve Naroff  }
671980e508ca70d6de75d2abfd96b4681fc98bb2698Steve Naroff
672a526c5c67e5a0473c340903ee542ce570119665fTed Kremenek  ObjCProtocolDecl **getReferencedProtocols() const {
6737ed9e0f97f4645edc5d4670385b985ea4c617ce7Fariborz Jahanian    return ReferencedProtocols;
6748f3fde00ad4d4f943321e338b914ae4740711c84Fariborz Jahanian  }
675c395bda57a0b2f6d8433a462caa780a7dcb7307bFariborz Jahanian  unsigned getNumReferencedProtocols() const { return NumReferencedProtocols; }
676c858105d41602a2dadb2efbc1af80a7b791ebac3Chris Lattner  unsigned getNumInstanceMethods() const { return NumInstanceMethods; }
677a906135721c350435319347d2672bbb3bf494f91Chris Lattner  unsigned getNumClassMethods() const { return NumClassMethods; }
67858dbdeb69c063f82d644504fc638120198f7fad2Steve Naroff
679a526c5c67e5a0473c340903ee542ce570119665fTed Kremenek  typedef ObjCMethodDecl * const * instmeth_iterator;
68058dbdeb69c063f82d644504fc638120198f7fad2Steve Naroff  instmeth_iterator instmeth_begin() const { return InstanceMethods; }
68158dbdeb69c063f82d644504fc638120198f7fad2Steve Naroff  instmeth_iterator instmeth_end() const {
682c858105d41602a2dadb2efbc1af80a7b791ebac3Chris Lattner    return InstanceMethods+NumInstanceMethods;
68358dbdeb69c063f82d644504fc638120198f7fad2Steve Naroff  }
68458dbdeb69c063f82d644504fc638120198f7fad2Steve Naroff
685a526c5c67e5a0473c340903ee542ce570119665fTed Kremenek  typedef ObjCMethodDecl * const * classmeth_iterator;
68658dbdeb69c063f82d644504fc638120198f7fad2Steve Naroff  classmeth_iterator classmeth_begin() const { return ClassMethods; }
68758dbdeb69c063f82d644504fc638120198f7fad2Steve Naroff  classmeth_iterator classmeth_end() const {
688a906135721c350435319347d2672bbb3bf494f91Chris Lattner    return ClassMethods+NumClassMethods;
68958dbdeb69c063f82d644504fc638120198f7fad2Steve Naroff  }
69058dbdeb69c063f82d644504fc638120198f7fad2Steve Naroff
69158dbdeb69c063f82d644504fc638120198f7fad2Steve Naroff  // Get the local instance method declared in this interface.
69268c82cf61228102aba1194efef222fa1478af2a8Chris Lattner  ObjCMethodDecl *getInstanceMethod(Selector Sel) {
69358dbdeb69c063f82d644504fc638120198f7fad2Steve Naroff    for (instmeth_iterator I = instmeth_begin(), E = instmeth_end();
69468c82cf61228102aba1194efef222fa1478af2a8Chris Lattner         I != E; ++I) {
69558dbdeb69c063f82d644504fc638120198f7fad2Steve Naroff      if ((*I)->getSelector() == Sel)
69658dbdeb69c063f82d644504fc638120198f7fad2Steve Naroff        return *I;
69758dbdeb69c063f82d644504fc638120198f7fad2Steve Naroff    }
69868c82cf61228102aba1194efef222fa1478af2a8Chris Lattner    return 0;
69958dbdeb69c063f82d644504fc638120198f7fad2Steve Naroff  }
70058dbdeb69c063f82d644504fc638120198f7fad2Steve Naroff  // Get the local class method declared in this interface.
70168c82cf61228102aba1194efef222fa1478af2a8Chris Lattner  ObjCMethodDecl *getClassMethod(Selector Sel) {
70258dbdeb69c063f82d644504fc638120198f7fad2Steve Naroff    for (classmeth_iterator I = classmeth_begin(), E = classmeth_end();
70368c82cf61228102aba1194efef222fa1478af2a8Chris Lattner         I != E; ++I) {
70458dbdeb69c063f82d644504fc638120198f7fad2Steve Naroff      if ((*I)->getSelector() == Sel)
70558dbdeb69c063f82d644504fc638120198f7fad2Steve Naroff        return *I;
70658dbdeb69c063f82d644504fc638120198f7fad2Steve Naroff    }
70768c82cf61228102aba1194efef222fa1478af2a8Chris Lattner    return 0;
70858dbdeb69c063f82d644504fc638120198f7fad2Steve Naroff  }
7098f3fde00ad4d4f943321e338b914ae4740711c84Fariborz Jahanian
710a526c5c67e5a0473c340903ee542ce570119665fTed Kremenek  void addMethods(ObjCMethodDecl **insMethods, unsigned numInsMembers,
711a526c5c67e5a0473c340903ee542ce570119665fTed Kremenek                  ObjCMethodDecl **clsMethods, unsigned numClsMembers,
71260fcceeedbfc8b4a99cb942e2bc5aeb9e2f92a1fSteve Naroff                  SourceLocation AtEndLoc);
713980e508ca70d6de75d2abfd96b4681fc98bb2698Steve Naroff
714a526c5c67e5a0473c340903ee542ce570119665fTed Kremenek  ObjCCategoryDecl *getNextClassCategory() const { return NextClassCategory; }
715980e508ca70d6de75d2abfd96b4681fc98bb2698Steve Naroff  void insertNextClassCategory() {
7163d58138992b9bc7b34aaa680f3ddf3971292eb7dSteve Naroff    NextClassCategory = ClassInterface->getCategoryList();
7173d58138992b9bc7b34aaa680f3ddf3971292eb7dSteve Naroff    ClassInterface->setCategoryList(this);
718980e508ca70d6de75d2abfd96b4681fc98bb2698Steve Naroff  }
719423cb565abc681b770fb4b9b4bc24d398c98157bSteve Naroff  // Location information, modeled after the Stmt API.
720423cb565abc681b770fb4b9b4bc24d398c98157bSteve Naroff  SourceLocation getLocStart() const { return getLocation(); } // '@'interface
721423cb565abc681b770fb4b9b4bc24d398c98157bSteve Naroff  SourceLocation getLocEnd() const { return EndLoc; }
722423cb565abc681b770fb4b9b4bc24d398c98157bSteve Naroff  void setLocEnd(SourceLocation LE) { EndLoc = LE; };
723423cb565abc681b770fb4b9b4bc24d398c98157bSteve Naroff
724423cb565abc681b770fb4b9b4bc24d398c98157bSteve Naroff  // We also need to record the @end location.
725423cb565abc681b770fb4b9b4bc24d398c98157bSteve Naroff  SourceLocation getAtEndLoc() const { return AtEndLoc; }
726ca3adf7e8cac8c9fbaf592b1e5c2be6f082de7baFariborz Jahanian
727a526c5c67e5a0473c340903ee542ce570119665fTed Kremenek  static bool classof(const Decl *D) { return D->getKind() == ObjCCategory; }
728a526c5c67e5a0473c340903ee542ce570119665fTed Kremenek  static bool classof(const ObjCCategoryDecl *D) { return true; }
729980e508ca70d6de75d2abfd96b4681fc98bb2698Steve Naroff};
7300c6b6243d3efd958c17943130e2a773653511edcSteve Naroff
731a526c5c67e5a0473c340903ee542ce570119665fTed Kremenek/// ObjCCategoryImplDecl - An object of this class encapsulates a category
7328f3fde00ad4d4f943321e338b914ae4740711c84Fariborz Jahanian/// @implementation declaration.
733a526c5c67e5a0473c340903ee542ce570119665fTed Kremenekclass ObjCCategoryImplDecl : public NamedDecl {
7348f3fde00ad4d4f943321e338b914ae4740711c84Fariborz Jahanian  /// Class interface for this category implementation
735a526c5c67e5a0473c340903ee542ce570119665fTed Kremenek  ObjCInterfaceDecl *ClassInterface;
7368f3fde00ad4d4f943321e338b914ae4740711c84Fariborz Jahanian
737e1e6c0d5c79c0ee7ed62fef47a19aa7ecef40db4Steve Naroff  /// implemented instance methods
738a526c5c67e5a0473c340903ee542ce570119665fTed Kremenek  llvm::SmallVector<ObjCMethodDecl*, 32> InstanceMethods;
7398f3fde00ad4d4f943321e338b914ae4740711c84Fariborz Jahanian
740e1e6c0d5c79c0ee7ed62fef47a19aa7ecef40db4Steve Naroff  /// implemented class methods
741a526c5c67e5a0473c340903ee542ce570119665fTed Kremenek  llvm::SmallVector<ObjCMethodDecl*, 32> ClassMethods;
742e1e6c0d5c79c0ee7ed62fef47a19aa7ecef40db4Steve Naroff
743e1e6c0d5c79c0ee7ed62fef47a19aa7ecef40db4Steve Naroff  SourceLocation EndLoc;
74475c9cae5f85c72cbb1649e93849e16ede3f07522Chris Lattner
745a526c5c67e5a0473c340903ee542ce570119665fTed Kremenek  ObjCCategoryImplDecl(SourceLocation L, IdentifierInfo *Id,
746a526c5c67e5a0473c340903ee542ce570119665fTed Kremenek                       ObjCInterfaceDecl *classInterface)
747a526c5c67e5a0473c340903ee542ce570119665fTed Kremenek    : NamedDecl(ObjCCategoryImpl, L, Id), ClassInterface(classInterface) {}
74875c9cae5f85c72cbb1649e93849e16ede3f07522Chris Lattnerpublic:
7490ed844b04ea4387caa4e1cf3dc375d269657536bChris Lattner  static ObjCCategoryImplDecl *Create(ASTContext &C,
7500ed844b04ea4387caa4e1cf3dc375d269657536bChris Lattner                                      SourceLocation L, IdentifierInfo *Id,
75175c9cae5f85c72cbb1649e93849e16ede3f07522Chris Lattner                                      ObjCInterfaceDecl *classInterface);
7528f3fde00ad4d4f943321e338b914ae4740711c84Fariborz Jahanian
753e0def7589a8afa8a6acef13476bb3f882c104b91Chris Lattner  const ObjCInterfaceDecl *getClassInterface() const { return ClassInterface; }
754e0def7589a8afa8a6acef13476bb3f882c104b91Chris Lattner  ObjCInterfaceDecl *getClassInterface() { return ClassInterface; }
7558f3fde00ad4d4f943321e338b914ae4740711c84Fariborz Jahanian
756ab4c4d5e5ececa77aae7e291fafcba3049319cdcChris Lattner  unsigned getNumInstanceMethods() const { return InstanceMethods.size(); }
757ab4c4d5e5ececa77aae7e291fafcba3049319cdcChris Lattner  unsigned getNumClassMethods() const { return ClassMethods.size(); }
758e1e6c0d5c79c0ee7ed62fef47a19aa7ecef40db4Steve Naroff
759a526c5c67e5a0473c340903ee542ce570119665fTed Kremenek  void addInstanceMethod(ObjCMethodDecl *method) {
760e1e6c0d5c79c0ee7ed62fef47a19aa7ecef40db4Steve Naroff    InstanceMethods.push_back(method);
761e1e6c0d5c79c0ee7ed62fef47a19aa7ecef40db4Steve Naroff  }
762a526c5c67e5a0473c340903ee542ce570119665fTed Kremenek  void addClassMethod(ObjCMethodDecl *method) {
763e1e6c0d5c79c0ee7ed62fef47a19aa7ecef40db4Steve Naroff    ClassMethods.push_back(method);
764e1e6c0d5c79c0ee7ed62fef47a19aa7ecef40db4Steve Naroff  }
76594a5c3334bba3cc8cd1da85ba1118bc2c080add9Steve Naroff  // Get the instance method definition for this implementation.
766a526c5c67e5a0473c340903ee542ce570119665fTed Kremenek  ObjCMethodDecl *getInstanceMethod(Selector Sel);
76794a5c3334bba3cc8cd1da85ba1118bc2c080add9Steve Naroff
76894a5c3334bba3cc8cd1da85ba1118bc2c080add9Steve Naroff  // Get the class method definition for this implementation.
769a526c5c67e5a0473c340903ee542ce570119665fTed Kremenek  ObjCMethodDecl *getClassMethod(Selector Sel);
770e1e6c0d5c79c0ee7ed62fef47a19aa7ecef40db4Steve Naroff
771a526c5c67e5a0473c340903ee542ce570119665fTed Kremenek  typedef llvm::SmallVector<ObjCMethodDecl*, 32>::const_iterator
772ab4c4d5e5ececa77aae7e291fafcba3049319cdcChris Lattner    instmeth_iterator;
773ab4c4d5e5ececa77aae7e291fafcba3049319cdcChris Lattner  instmeth_iterator instmeth_begin() const { return InstanceMethods.begin(); }
774ab4c4d5e5ececa77aae7e291fafcba3049319cdcChris Lattner  instmeth_iterator instmeth_end() const { return InstanceMethods.end(); }
775ab4c4d5e5ececa77aae7e291fafcba3049319cdcChris Lattner
776a526c5c67e5a0473c340903ee542ce570119665fTed Kremenek  typedef llvm::SmallVector<ObjCMethodDecl*, 32>::const_iterator
777ab4c4d5e5ececa77aae7e291fafcba3049319cdcChris Lattner    classmeth_iterator;
778ab4c4d5e5ececa77aae7e291fafcba3049319cdcChris Lattner  classmeth_iterator classmeth_begin() const { return ClassMethods.begin(); }
779ab4c4d5e5ececa77aae7e291fafcba3049319cdcChris Lattner  classmeth_iterator classmeth_end() const { return ClassMethods.end(); }
780ab4c4d5e5ececa77aae7e291fafcba3049319cdcChris Lattner
781ab4c4d5e5ececa77aae7e291fafcba3049319cdcChris Lattner
782e1e6c0d5c79c0ee7ed62fef47a19aa7ecef40db4Steve Naroff  // Location information, modeled after the Stmt API.
783e1e6c0d5c79c0ee7ed62fef47a19aa7ecef40db4Steve Naroff  SourceLocation getLocStart() const { return getLocation(); }
784e1e6c0d5c79c0ee7ed62fef47a19aa7ecef40db4Steve Naroff  SourceLocation getLocEnd() const { return EndLoc; }
785e1e6c0d5c79c0ee7ed62fef47a19aa7ecef40db4Steve Naroff  void setLocEnd(SourceLocation LE) { EndLoc = LE; };
786e1e6c0d5c79c0ee7ed62fef47a19aa7ecef40db4Steve Naroff
787a526c5c67e5a0473c340903ee542ce570119665fTed Kremenek  static bool classof(const Decl *D) { return D->getKind() == ObjCCategoryImpl;}
788a526c5c67e5a0473c340903ee542ce570119665fTed Kremenek  static bool classof(const ObjCCategoryImplDecl *D) { return true; }
7898f3fde00ad4d4f943321e338b914ae4740711c84Fariborz Jahanian};
7908f3fde00ad4d4f943321e338b914ae4740711c84Fariborz Jahanian
791a526c5c67e5a0473c340903ee542ce570119665fTed Kremenek/// ObjCImplementationDecl - Represents a class definition - this is where
7920c6b6243d3efd958c17943130e2a773653511edcSteve Naroff/// method definitions are specified. For example:
7930c6b6243d3efd958c17943130e2a773653511edcSteve Naroff///
7940c6b6243d3efd958c17943130e2a773653511edcSteve Naroff/// @implementation MyClass
7950c6b6243d3efd958c17943130e2a773653511edcSteve Naroff/// - (void)myMethod { /* do something */ }
7960c6b6243d3efd958c17943130e2a773653511edcSteve Naroff/// @end
7970c6b6243d3efd958c17943130e2a773653511edcSteve Naroff///
7980c6b6243d3efd958c17943130e2a773653511edcSteve Naroff/// Typically, instance variables are specified in the class interface,
7990c6b6243d3efd958c17943130e2a773653511edcSteve Naroff/// *not* in the implemenentation. Nevertheless (for legacy reasons), we
8000c6b6243d3efd958c17943130e2a773653511edcSteve Naroff/// allow instance variables to be specified in the implementation. When
8010c6b6243d3efd958c17943130e2a773653511edcSteve Naroff/// specified, they need to be *identical* to the interface. Now that we
8020c6b6243d3efd958c17943130e2a773653511edcSteve Naroff/// have support for non-fragile ivars in ObjC 2.0, we can consider removing
8030c6b6243d3efd958c17943130e2a773653511edcSteve Naroff/// the legacy semantics and allow developers to move private ivar declarations
8040c6b6243d3efd958c17943130e2a773653511edcSteve Naroff/// from the class interface to the class implementation (but I digress:-)
8050c6b6243d3efd958c17943130e2a773653511edcSteve Naroff///
806a526c5c67e5a0473c340903ee542ce570119665fTed Kremenekclass ObjCImplementationDecl : public NamedDecl {
807f4d331dd922f92478ebf30e808c0ca97ce49418bFariborz Jahanian  /// Class interface for this category implementation
808a526c5c67e5a0473c340903ee542ce570119665fTed Kremenek  ObjCInterfaceDecl *ClassInterface;
809f4d331dd922f92478ebf30e808c0ca97ce49418bFariborz Jahanian
810980e508ca70d6de75d2abfd96b4681fc98bb2698Steve Naroff  /// Implementation Class's super class.
811a526c5c67e5a0473c340903ee542ce570119665fTed Kremenek  ObjCInterfaceDecl *SuperClass;
812980e508ca70d6de75d2abfd96b4681fc98bb2698Steve Naroff
813980e508ca70d6de75d2abfd96b4681fc98bb2698Steve Naroff  /// Optional Ivars/NumIvars - This is a new[]'d array of pointers to Decls.
814a526c5c67e5a0473c340903ee542ce570119665fTed Kremenek  ObjCIvarDecl **Ivars;   // Null if not specified
815f3a7af901231535191d14cb524d58f22907ac4d2Chris Lattner  unsigned NumIvars;      // 0 if none.
8160416fb9f379b49abb3eb0c1cb2ca75107e5a71d1Steve Naroff
817980e508ca70d6de75d2abfd96b4681fc98bb2698Steve Naroff  /// implemented instance methods
818a526c5c67e5a0473c340903ee542ce570119665fTed Kremenek  llvm::SmallVector<ObjCMethodDecl*, 32> InstanceMethods;
8190416fb9f379b49abb3eb0c1cb2ca75107e5a71d1Steve Naroff
820980e508ca70d6de75d2abfd96b4681fc98bb2698Steve Naroff  /// implemented class methods
821a526c5c67e5a0473c340903ee542ce570119665fTed Kremenek  llvm::SmallVector<ObjCMethodDecl*, 32> ClassMethods;
8220416fb9f379b49abb3eb0c1cb2ca75107e5a71d1Steve Naroff
8230416fb9f379b49abb3eb0c1cb2ca75107e5a71d1Steve Naroff  SourceLocation EndLoc;
82475c9cae5f85c72cbb1649e93849e16ede3f07522Chris Lattner
825a526c5c67e5a0473c340903ee542ce570119665fTed Kremenek  ObjCImplementationDecl(SourceLocation L, IdentifierInfo *Id,
826a526c5c67e5a0473c340903ee542ce570119665fTed Kremenek                         ObjCInterfaceDecl *classInterface,
827a526c5c67e5a0473c340903ee542ce570119665fTed Kremenek                         ObjCInterfaceDecl *superDecl)
828a526c5c67e5a0473c340903ee542ce570119665fTed Kremenek    : NamedDecl(ObjCImplementation, L, Id),
8290416fb9f379b49abb3eb0c1cb2ca75107e5a71d1Steve Naroff      ClassInterface(classInterface), SuperClass(superDecl),
830f3a7af901231535191d14cb524d58f22907ac4d2Chris Lattner      Ivars(0), NumIvars(0) {}
83175c9cae5f85c72cbb1649e93849e16ede3f07522Chris Lattnerpublic:
8320ed844b04ea4387caa4e1cf3dc375d269657536bChris Lattner  static ObjCImplementationDecl *Create(ASTContext &C,
8330ed844b04ea4387caa4e1cf3dc375d269657536bChris Lattner                                        SourceLocation L, IdentifierInfo *Id,
83475c9cae5f85c72cbb1649e93849e16ede3f07522Chris Lattner                                        ObjCInterfaceDecl *classInterface,
83575c9cae5f85c72cbb1649e93849e16ede3f07522Chris Lattner                                        ObjCInterfaceDecl *superDecl);
83675c9cae5f85c72cbb1649e93849e16ede3f07522Chris Lattner
837980e508ca70d6de75d2abfd96b4681fc98bb2698Steve Naroff
838a526c5c67e5a0473c340903ee542ce570119665fTed Kremenek  void ObjCAddInstanceVariablesToClassImpl(ObjCIvarDecl **ivars,
839980e508ca70d6de75d2abfd96b4681fc98bb2698Steve Naroff                                           unsigned numIvars);
840980e508ca70d6de75d2abfd96b4681fc98bb2698Steve Naroff
841a526c5c67e5a0473c340903ee542ce570119665fTed Kremenek  void addInstanceMethod(ObjCMethodDecl *method) {
8420416fb9f379b49abb3eb0c1cb2ca75107e5a71d1Steve Naroff    InstanceMethods.push_back(method);
8430416fb9f379b49abb3eb0c1cb2ca75107e5a71d1Steve Naroff  }
844a526c5c67e5a0473c340903ee542ce570119665fTed Kremenek  void addClassMethod(ObjCMethodDecl *method) {
8450416fb9f379b49abb3eb0c1cb2ca75107e5a71d1Steve Naroff    ClassMethods.push_back(method);
8460416fb9f379b49abb3eb0c1cb2ca75107e5a71d1Steve Naroff  }
8470416fb9f379b49abb3eb0c1cb2ca75107e5a71d1Steve Naroff  // Location information, modeled after the Stmt API.
8480416fb9f379b49abb3eb0c1cb2ca75107e5a71d1Steve Naroff  SourceLocation getLocStart() const { return getLocation(); }
8490416fb9f379b49abb3eb0c1cb2ca75107e5a71d1Steve Naroff  SourceLocation getLocEnd() const { return EndLoc; }
8500416fb9f379b49abb3eb0c1cb2ca75107e5a71d1Steve Naroff  void setLocEnd(SourceLocation LE) { EndLoc = LE; };
8510416fb9f379b49abb3eb0c1cb2ca75107e5a71d1Steve Naroff
852e0def7589a8afa8a6acef13476bb3f882c104b91Chris Lattner  const ObjCInterfaceDecl *getClassInterface() const { return ClassInterface; }
853e0def7589a8afa8a6acef13476bb3f882c104b91Chris Lattner  ObjCInterfaceDecl *getClassInterface() { return ClassInterface; }
854e0def7589a8afa8a6acef13476bb3f882c104b91Chris Lattner  const ObjCInterfaceDecl *getSuperClass() const { return SuperClass; }
855e0def7589a8afa8a6acef13476bb3f882c104b91Chris Lattner  ObjCInterfaceDecl *getSuperClass() { return SuperClass; }
856980e508ca70d6de75d2abfd96b4681fc98bb2698Steve Naroff
857f3a7af901231535191d14cb524d58f22907ac4d2Chris Lattner  void setSuperClass(ObjCInterfaceDecl * superCls) { SuperClass = superCls; }
858980e508ca70d6de75d2abfd96b4681fc98bb2698Steve Naroff
859b5feb35360526da42ad7e3fa7ef3d44a5992b83cChris Lattner  unsigned getNumInstanceMethods() const { return InstanceMethods.size(); }
86062db2f4214c1589082960f12c9cb8924fe0cf8c5Chris Lattner  unsigned getNumClassMethods() const { return ClassMethods.size(); }
861c43d868355374d48296ad3be2c9c536698a5e9a8Steve Naroff
862a526c5c67e5a0473c340903ee542ce570119665fTed Kremenek  typedef llvm::SmallVector<ObjCMethodDecl*, 32>::const_iterator
8630157c5144513438bb74aebf50d18f95df4104acbChris Lattner       instmeth_iterator;
8640157c5144513438bb74aebf50d18f95df4104acbChris Lattner  instmeth_iterator instmeth_begin() const { return InstanceMethods.begin(); }
8650157c5144513438bb74aebf50d18f95df4104acbChris Lattner  instmeth_iterator instmeth_end() const { return InstanceMethods.end(); }
8660157c5144513438bb74aebf50d18f95df4104acbChris Lattner
867a526c5c67e5a0473c340903ee542ce570119665fTed Kremenek  typedef llvm::SmallVector<ObjCMethodDecl*, 32>::const_iterator
8680157c5144513438bb74aebf50d18f95df4104acbChris Lattner    classmeth_iterator;
8690157c5144513438bb74aebf50d18f95df4104acbChris Lattner  classmeth_iterator classmeth_begin() const { return ClassMethods.begin(); }
8700157c5144513438bb74aebf50d18f95df4104acbChris Lattner  classmeth_iterator classmeth_end() const { return ClassMethods.end(); }
8710157c5144513438bb74aebf50d18f95df4104acbChris Lattner
87294a5c3334bba3cc8cd1da85ba1118bc2c080add9Steve Naroff  // Get the instance method definition for this implementation.
873a526c5c67e5a0473c340903ee542ce570119665fTed Kremenek  ObjCMethodDecl *getInstanceMethod(Selector Sel);
8740157c5144513438bb74aebf50d18f95df4104acbChris Lattner
87594a5c3334bba3cc8cd1da85ba1118bc2c080add9Steve Naroff  // Get the class method definition for this implementation.
876a526c5c67e5a0473c340903ee542ce570119665fTed Kremenek  ObjCMethodDecl *getClassMethod(Selector Sel);
8770157c5144513438bb74aebf50d18f95df4104acbChris Lattner
878a526c5c67e5a0473c340903ee542ce570119665fTed Kremenek  typedef ObjCIvarDecl * const *ivar_iterator;
8790157c5144513438bb74aebf50d18f95df4104acbChris Lattner  ivar_iterator ivar_begin() const { return Ivars; }
880f3a7af901231535191d14cb524d58f22907ac4d2Chris Lattner  ivar_iterator ivar_end() const { return Ivars+NumIvars; }
881f3a7af901231535191d14cb524d58f22907ac4d2Chris Lattner  unsigned ivar_size() const { return NumIvars; }
882f3a7af901231535191d14cb524d58f22907ac4d2Chris Lattner  bool ivar_empty() const { return NumIvars == 0; }
8830157c5144513438bb74aebf50d18f95df4104acbChris Lattner
884980e508ca70d6de75d2abfd96b4681fc98bb2698Steve Naroff  static bool classof(const Decl *D) {
885a526c5c67e5a0473c340903ee542ce570119665fTed Kremenek    return D->getKind() == ObjCImplementation;
886980e508ca70d6de75d2abfd96b4681fc98bb2698Steve Naroff  }
887a526c5c67e5a0473c340903ee542ce570119665fTed Kremenek  static bool classof(const ObjCImplementationDecl *D) { return true; }
888980e508ca70d6de75d2abfd96b4681fc98bb2698Steve Naroff};
889243b64b0001172405ff803c61bdcaa8e98ec1552Fariborz Jahanian
890a526c5c67e5a0473c340903ee542ce570119665fTed Kremenek/// ObjCCompatibleAliasDecl - Represents alias of a class. This alias is
891243b64b0001172405ff803c61bdcaa8e98ec1552Fariborz Jahanian/// declared as @compatibility_alias alias class.
892e8043c39176e7f253fbd92982b077eca6bf2fd59Steve Naroffclass ObjCCompatibleAliasDecl : public NamedDecl {
893243b64b0001172405ff803c61bdcaa8e98ec1552Fariborz Jahanian  /// Class that this is an alias of.
894a526c5c67e5a0473c340903ee542ce570119665fTed Kremenek  ObjCInterfaceDecl *AliasedClass;
895243b64b0001172405ff803c61bdcaa8e98ec1552Fariborz Jahanian
896a526c5c67e5a0473c340903ee542ce570119665fTed Kremenek  ObjCCompatibleAliasDecl(SourceLocation L, IdentifierInfo *Id,
897e8043c39176e7f253fbd92982b077eca6bf2fd59Steve Naroff                          ObjCInterfaceDecl* aliasedClass)
898e8043c39176e7f253fbd92982b077eca6bf2fd59Steve Naroff    : NamedDecl(ObjCCompatibleAlias, L, Id), AliasedClass(aliasedClass) {}
899f8d17a59167d9c2026506ed8813ea434d93b662aChris Lattnerpublic:
9000ed844b04ea4387caa4e1cf3dc375d269657536bChris Lattner  static ObjCCompatibleAliasDecl *Create(ASTContext &C,
9010ed844b04ea4387caa4e1cf3dc375d269657536bChris Lattner                                         SourceLocation L, IdentifierInfo *Id,
902f8d17a59167d9c2026506ed8813ea434d93b662aChris Lattner                                         ObjCInterfaceDecl* aliasedClass);
903f8d17a59167d9c2026506ed8813ea434d93b662aChris Lattner
904f8d17a59167d9c2026506ed8813ea434d93b662aChris Lattner  const ObjCInterfaceDecl *getClassInterface() const { return AliasedClass; }
905f8d17a59167d9c2026506ed8813ea434d93b662aChris Lattner  ObjCInterfaceDecl *getClassInterface() { return AliasedClass; }
906980e508ca70d6de75d2abfd96b4681fc98bb2698Steve Naroff
907243b64b0001172405ff803c61bdcaa8e98ec1552Fariborz Jahanian  static bool classof(const Decl *D) {
9088a934233d1582b5bde9d270bc0705aa81e471a79Chris Lattner    return D->getKind() == ObjCCompatibleAlias;
909243b64b0001172405ff803c61bdcaa8e98ec1552Fariborz Jahanian  }
910a526c5c67e5a0473c340903ee542ce570119665fTed Kremenek  static bool classof(const ObjCCompatibleAliasDecl *D) { return true; }
911243b64b0001172405ff803c61bdcaa8e98ec1552Fariborz Jahanian
912243b64b0001172405ff803c61bdcaa8e98ec1552Fariborz Jahanian};
91382a5fe3d1cf204b672cdab24d72275b6ad2c3527Fariborz Jahanian
914a526c5c67e5a0473c340903ee542ce570119665fTed Kremenekclass ObjCPropertyDecl : public Decl {
91582a5fe3d1cf204b672cdab24d72275b6ad2c3527Fariborz Jahanianpublic:
916a5674258f5e6f74f1c0ed3ece4d64fbb1b9afb11Chris Lattner  enum PropertyAttributeKind {
917a5674258f5e6f74f1c0ed3ece4d64fbb1b9afb11Chris Lattner    OBJC_PR_noattr    = 0x00,
918a5674258f5e6f74f1c0ed3ece4d64fbb1b9afb11Chris Lattner    OBJC_PR_readonly  = 0x01,
919a5674258f5e6f74f1c0ed3ece4d64fbb1b9afb11Chris Lattner    OBJC_PR_getter    = 0x02,
920a5674258f5e6f74f1c0ed3ece4d64fbb1b9afb11Chris Lattner    OBJC_PR_assign    = 0x04,
921a5674258f5e6f74f1c0ed3ece4d64fbb1b9afb11Chris Lattner    OBJC_PR_readwrite = 0x08,
922a5674258f5e6f74f1c0ed3ece4d64fbb1b9afb11Chris Lattner    OBJC_PR_retain    = 0x10,
923a5674258f5e6f74f1c0ed3ece4d64fbb1b9afb11Chris Lattner    OBJC_PR_copy      = 0x20,
924a5674258f5e6f74f1c0ed3ece4d64fbb1b9afb11Chris Lattner    OBJC_PR_nonatomic = 0x40,
925a5674258f5e6f74f1c0ed3ece4d64fbb1b9afb11Chris Lattner    OBJC_PR_setter    = 0x80
926a5674258f5e6f74f1c0ed3ece4d64fbb1b9afb11Chris Lattner  };
92782a5fe3d1cf204b672cdab24d72275b6ad2c3527Fariborz Jahanianprivate:
92882a5fe3d1cf204b672cdab24d72275b6ad2c3527Fariborz Jahanian  // List of property name declarations
92982a5fe3d1cf204b672cdab24d72275b6ad2c3527Fariborz Jahanian  // FIXME: Property is not an ivar.
930a526c5c67e5a0473c340903ee542ce570119665fTed Kremenek  ObjCIvarDecl **PropertyDecls;
931f4af5154571e0c5eadb19df10e65464766ef6683Chris Lattner  unsigned NumPropertyDecls;
932ca63fa00786e51c207c829f4182f11a6c6b552beTed Kremenek  unsigned PropertyAttributes : 8;
93382a5fe3d1cf204b672cdab24d72275b6ad2c3527Fariborz Jahanian
93482a5fe3d1cf204b672cdab24d72275b6ad2c3527Fariborz Jahanian  IdentifierInfo *GetterName;    // getter name of NULL if no getter
93582a5fe3d1cf204b672cdab24d72275b6ad2c3527Fariborz Jahanian  IdentifierInfo *SetterName;    // setter name of NULL if no setter
93682a5fe3d1cf204b672cdab24d72275b6ad2c3527Fariborz Jahanian
937a526c5c67e5a0473c340903ee542ce570119665fTed Kremenek  ObjCPropertyDecl(SourceLocation L)
938f4af5154571e0c5eadb19df10e65464766ef6683Chris Lattner    : Decl(PropertyDecl, L), PropertyDecls(0), NumPropertyDecls(0),
939f8d17a59167d9c2026506ed8813ea434d93b662aChris Lattner      PropertyAttributes(OBJC_PR_noattr), GetterName(0), SetterName(0) {}
940f8d17a59167d9c2026506ed8813ea434d93b662aChris Lattnerpublic:
941f8d17a59167d9c2026506ed8813ea434d93b662aChris Lattner  static ObjCPropertyDecl *Create(ASTContext &C, SourceLocation L);
94282a5fe3d1cf204b672cdab24d72275b6ad2c3527Fariborz Jahanian
943a5674258f5e6f74f1c0ed3ece4d64fbb1b9afb11Chris Lattner  typedef ObjCIvarDecl * const *propdecl_iterator;
944a5674258f5e6f74f1c0ed3ece4d64fbb1b9afb11Chris Lattner  propdecl_iterator propdecl_begin() const { return PropertyDecls; }
945a5674258f5e6f74f1c0ed3ece4d64fbb1b9afb11Chris Lattner  propdecl_iterator propdecl_end() const {
946a5674258f5e6f74f1c0ed3ece4d64fbb1b9afb11Chris Lattner    return PropertyDecls+NumPropertyDecls;
947a5674258f5e6f74f1c0ed3ece4d64fbb1b9afb11Chris Lattner  }
948a5674258f5e6f74f1c0ed3ece4d64fbb1b9afb11Chris Lattner  unsigned propdecl_size() const { return NumPropertyDecls; }
949a5674258f5e6f74f1c0ed3ece4d64fbb1b9afb11Chris Lattner  bool propdecl_empty() const { return NumPropertyDecls == 0; }
950f4af5154571e0c5eadb19df10e65464766ef6683Chris Lattner
951a5674258f5e6f74f1c0ed3ece4d64fbb1b9afb11Chris Lattner  /// setPropertyDeclLists - Set the property decl list to the specified array
952a5674258f5e6f74f1c0ed3ece4d64fbb1b9afb11Chris Lattner  /// of decls.
953f4af5154571e0c5eadb19df10e65464766ef6683Chris Lattner  void setPropertyDeclLists(ObjCIvarDecl **Properties, unsigned NumProp);
95482a5fe3d1cf204b672cdab24d72275b6ad2c3527Fariborz Jahanian
955a5674258f5e6f74f1c0ed3ece4d64fbb1b9afb11Chris Lattner  PropertyAttributeKind getPropertyAttributes() const {
956f4af5154571e0c5eadb19df10e65464766ef6683Chris Lattner    return PropertyAttributeKind(PropertyAttributes);
957f4af5154571e0c5eadb19df10e65464766ef6683Chris Lattner  }
958564576b225386cbff375351597dd5e2a92872d38Fariborz Jahanian  void setPropertyAttributes(PropertyAttributeKind PRVal) {
959a5674258f5e6f74f1c0ed3ece4d64fbb1b9afb11Chris Lattner    PropertyAttributes |= PRVal;
96082a5fe3d1cf204b672cdab24d72275b6ad2c3527Fariborz Jahanian  }
96182a5fe3d1cf204b672cdab24d72275b6ad2c3527Fariborz Jahanian
962a5674258f5e6f74f1c0ed3ece4d64fbb1b9afb11Chris Lattner  IdentifierInfo *getGetterName() const { return GetterName; }
96382a5fe3d1cf204b672cdab24d72275b6ad2c3527Fariborz Jahanian  void setGetterName(IdentifierInfo *Id) { GetterName = Id; }
96482a5fe3d1cf204b672cdab24d72275b6ad2c3527Fariborz Jahanian
965a5674258f5e6f74f1c0ed3ece4d64fbb1b9afb11Chris Lattner  IdentifierInfo *getSetterName() const { return SetterName; }
96682a5fe3d1cf204b672cdab24d72275b6ad2c3527Fariborz Jahanian  void setSetterName(IdentifierInfo *Id) { SetterName = Id; }
96782a5fe3d1cf204b672cdab24d72275b6ad2c3527Fariborz Jahanian
96882a5fe3d1cf204b672cdab24d72275b6ad2c3527Fariborz Jahanian  static bool classof(const Decl *D) {
96982a5fe3d1cf204b672cdab24d72275b6ad2c3527Fariborz Jahanian    return D->getKind() == PropertyDecl;
97082a5fe3d1cf204b672cdab24d72275b6ad2c3527Fariborz Jahanian  }
971a526c5c67e5a0473c340903ee542ce570119665fTed Kremenek  static bool classof(const ObjCPropertyDecl *D) { return true; }
97282a5fe3d1cf204b672cdab24d72275b6ad2c3527Fariborz Jahanian};
973980e508ca70d6de75d2abfd96b4681fc98bb2698Steve Naroff
974980e508ca70d6de75d2abfd96b4681fc98bb2698Steve Naroff}  // end namespace clang
975980e508ca70d6de75d2abfd96b4681fc98bb2698Steve Naroff#endif
976