blob: 03764d27ff157d9cc06ff7534b88d681df2d299c [file] [log] [blame]
Daniel Dunbar303e2c22008-08-11 02:45:11 +00001//===------- CGObjCMac.cpp - Interface to Apple Objective-C Runtime -------===//
2//
3// The LLVM Compiler Infrastructure
4//
5// This file is distributed under the University of Illinois Open Source
6// License. See LICENSE.TXT for details.
7//
8//===----------------------------------------------------------------------===//
9//
10// This provides Objective-C code generation targetting the Apple runtime.
11//
12//===----------------------------------------------------------------------===//
13
14#include "CGObjCRuntime.h"
Daniel Dunbar3ad53482008-08-11 21:35:06 +000015
Daniel Dunbar034299e2010-03-31 01:09:11 +000016#include "CGRecordLayout.h"
Daniel Dunbar3ad53482008-08-11 21:35:06 +000017#include "CodeGenModule.h"
Daniel Dunbara94ecd22008-08-16 03:19:19 +000018#include "CodeGenFunction.h"
John McCallbd309292010-07-06 01:34:17 +000019#include "CGException.h"
Daniel Dunbar8b8683f2008-08-12 00:12:39 +000020#include "clang/AST/ASTContext.h"
Daniel Dunbar221fa942008-08-11 04:54:23 +000021#include "clang/AST/Decl.h"
Daniel Dunbarb036db82008-08-13 03:21:16 +000022#include "clang/AST/DeclObjC.h"
Anders Carlsson15b73de2009-07-18 19:43:29 +000023#include "clang/AST/RecordLayout.h"
Chris Lattnerf0b64d72009-04-26 01:32:48 +000024#include "clang/AST/StmtObjC.h"
Daniel Dunbar3ad53482008-08-11 21:35:06 +000025#include "clang/Basic/LangOptions.h"
Chandler Carruth85098242010-06-15 23:19:56 +000026#include "clang/Frontend/CodeGenOptions.h"
Daniel Dunbar3ad53482008-08-11 21:35:06 +000027
Daniel Dunbar0b0dcd92009-02-24 07:47:38 +000028#include "llvm/Intrinsics.h"
Owen Andersonae86c192009-07-13 04:10:07 +000029#include "llvm/LLVMContext.h"
Daniel Dunbar8b8683f2008-08-12 00:12:39 +000030#include "llvm/Module.h"
Daniel Dunbarc475d422008-10-29 22:36:39 +000031#include "llvm/ADT/DenseSet.h"
Daniel Dunbard027a922009-09-07 00:20:42 +000032#include "llvm/ADT/SetVector.h"
33#include "llvm/ADT/SmallString.h"
Fariborz Jahanian751c1e72009-12-12 21:26:21 +000034#include "llvm/ADT/SmallPtrSet.h"
John McCall5c08ab92010-07-13 22:12:14 +000035#include "llvm/Support/CallSite.h"
Daniel Dunbard027a922009-09-07 00:20:42 +000036#include "llvm/Support/raw_ostream.h"
Daniel Dunbar7bd00bd2008-08-12 06:48:42 +000037#include "llvm/Target/TargetData.h"
Torok Edwindb714922009-08-24 13:25:12 +000038#include <cstdio>
Daniel Dunbar303e2c22008-08-11 02:45:11 +000039
40using namespace clang;
Daniel Dunbar41cf9de2008-09-09 01:06:48 +000041using namespace CodeGen;
Daniel Dunbar303e2c22008-08-11 02:45:11 +000042
Daniel Dunbar9fd114d2009-04-22 07:32:20 +000043// Common CGObjCRuntime functions, these don't belong here, but they
44// don't belong in CGObjCRuntime either so we will live with it for
45// now.
46
Daniel Dunbar0cec95f2009-05-03 08:55:17 +000047static uint64_t LookupFieldBitOffset(CodeGen::CodeGenModule &CGM,
48 const ObjCInterfaceDecl *OID,
Daniel Dunbar961202372009-05-03 12:57:56 +000049 const ObjCImplementationDecl *ID,
Daniel Dunbar0cec95f2009-05-03 08:55:17 +000050 const ObjCIvarDecl *Ivar) {
Daniel Dunbar8c7f9812010-04-02 21:14:02 +000051 const ObjCInterfaceDecl *Container = Ivar->getContainingInterface();
Daniel Dunbar80b4eef2009-05-03 13:15:50 +000052
Daniel Dunbar7e5aba42010-04-02 22:29:40 +000053 // FIXME: We should eliminate the need to have ObjCImplementationDecl passed
54 // in here; it should never be necessary because that should be the lexical
55 // decl context for the ivar.
Daniel Dunbar031d4d42010-04-02 15:43:29 +000056
Daniel Dunbar80b4eef2009-05-03 13:15:50 +000057 // If we know have an implementation (and the ivar is in it) then
58 // look up in the implementation layout.
Daniel Dunbar59e476b2009-08-03 17:06:42 +000059 const ASTRecordLayout *RL;
Daniel Dunbar80b4eef2009-05-03 13:15:50 +000060 if (ID && ID->getClassInterface() == Container)
61 RL = &CGM.getContext().getASTObjCImplementationLayout(ID);
62 else
63 RL = &CGM.getContext().getASTObjCInterfaceLayout(Container);
Daniel Dunbar8c7f9812010-04-02 21:14:02 +000064
65 // Compute field index.
66 //
67 // FIXME: The index here is closely tied to how ASTContext::getObjCLayout is
68 // implemented. This should be fixed to get the information from the layout
69 // directly.
70 unsigned Index = 0;
71 llvm::SmallVector<ObjCIvarDecl*, 16> Ivars;
72 CGM.getContext().ShallowCollectObjCIvars(Container, Ivars);
73 for (unsigned k = 0, e = Ivars.size(); k != e; ++k) {
74 if (Ivar == Ivars[k])
75 break;
76 ++Index;
77 }
78 assert(Index != Ivars.size() && "Ivar is not inside container!");
79
Daniel Dunbar80b4eef2009-05-03 13:15:50 +000080 return RL->getFieldOffset(Index);
Daniel Dunbar0cec95f2009-05-03 08:55:17 +000081}
82
83uint64_t CGObjCRuntime::ComputeIvarBaseOffset(CodeGen::CodeGenModule &CGM,
84 const ObjCInterfaceDecl *OID,
85 const ObjCIvarDecl *Ivar) {
Daniel Dunbar961202372009-05-03 12:57:56 +000086 return LookupFieldBitOffset(CGM, OID, 0, Ivar) / 8;
87}
88
89uint64_t CGObjCRuntime::ComputeIvarBaseOffset(CodeGen::CodeGenModule &CGM,
90 const ObjCImplementationDecl *OID,
91 const ObjCIvarDecl *Ivar) {
92 return LookupFieldBitOffset(CGM, OID->getClassInterface(), OID, Ivar) / 8;
Daniel Dunbar9fd114d2009-04-22 07:32:20 +000093}
94
95LValue CGObjCRuntime::EmitValueForIvarAtOffset(CodeGen::CodeGenFunction &CGF,
96 const ObjCInterfaceDecl *OID,
97 llvm::Value *BaseValue,
98 const ObjCIvarDecl *Ivar,
99 unsigned CVRQualifiers,
100 llvm::Value *Offset) {
Daniel Dunbar0cec95f2009-05-03 08:55:17 +0000101 // Compute (type*) ( (char *) BaseValue + Offset)
Benjamin Kramerabd5b902009-10-13 10:07:13 +0000102 const llvm::Type *I8Ptr = llvm::Type::getInt8PtrTy(CGF.getLLVMContext());
Daniel Dunbar0cec95f2009-05-03 08:55:17 +0000103 QualType IvarTy = Ivar->getType();
104 const llvm::Type *LTy = CGF.CGM.getTypes().ConvertTypeForMem(IvarTy);
Daniel Dunbar9fd114d2009-04-22 07:32:20 +0000105 llvm::Value *V = CGF.Builder.CreateBitCast(BaseValue, I8Ptr);
Daniel Dunbar9fd114d2009-04-22 07:32:20 +0000106 V = CGF.Builder.CreateGEP(V, Offset, "add.ptr");
Owen Anderson9793f0e2009-07-29 22:16:19 +0000107 V = CGF.Builder.CreateBitCast(V, llvm::PointerType::getUnqual(LTy));
Daniel Dunbar59e476b2009-08-03 17:06:42 +0000108
John McCall8ccfcb52009-09-24 19:53:00 +0000109 Qualifiers Quals = CGF.MakeQualifiers(IvarTy);
110 Quals.addCVRQualifiers(CVRQualifiers);
111
Daniel Dunbarb76c4cd2010-04-05 16:20:33 +0000112 if (!Ivar->isBitField())
113 return LValue::MakeAddr(V, Quals);
Daniel Dunbar961202372009-05-03 12:57:56 +0000114
Daniel Dunbarb76c4cd2010-04-05 16:20:33 +0000115 // We need to compute the bit offset for the bit-field, the offset is to the
116 // byte. Note, there is a subtle invariant here: we can only call this routine
117 // on non-synthesized ivars but we may be called for synthesized ivars.
118 // However, a synthesized ivar can never be a bit-field, so this is safe.
119 uint64_t BitOffset = LookupFieldBitOffset(CGF.CGM, OID, 0, Ivar) % 8;
120 uint64_t BitFieldSize =
121 Ivar->getBitWidth()->EvaluateAsInt(CGF.getContext()).getZExtValue();
Daniel Dunbar9fd114d2009-04-22 07:32:20 +0000122
Daniel Dunbardc406b82010-04-05 21:36:35 +0000123 // Allocate a new CGBitFieldInfo object to describe this access.
124 //
125 // FIXME: This is incredibly wasteful, these should be uniqued or part of some
126 // layout object. However, this is blocked on other cleanups to the
127 // Objective-C code, so for now we just live with allocating a bunch of these
128 // objects.
Daniel Dunbardc406b82010-04-05 21:36:35 +0000129
Daniel Dunbarb935b932010-04-13 20:58:55 +0000130 // We always construct a single, possibly unaligned, access for this case.
Daniel Dunbar9c78d632010-04-15 05:09:32 +0000131 CGBitFieldInfo::AccessInfo AI;
Daniel Dunbarb935b932010-04-13 20:58:55 +0000132 AI.FieldIndex = 0;
133 AI.FieldByteOffset = 0;
134 AI.FieldBitStart = BitOffset;
135 AI.AccessWidth = CGF.CGM.getContext().getTypeSize(IvarTy);
136 AI.AccessAlignment = 0;
137 AI.TargetBitOffset = 0;
138 AI.TargetBitWidth = BitFieldSize;
139
Daniel Dunbar9c78d632010-04-15 05:09:32 +0000140 CGBitFieldInfo *Info =
141 new (CGF.CGM.getContext()) CGBitFieldInfo(BitFieldSize, 1, &AI,
142 IvarTy->isSignedIntegerType());
143
Daniel Dunbarb76c4cd2010-04-05 16:20:33 +0000144 // FIXME: We need to set a very conservative alignment on this, or make sure
145 // that the runtime is doing the right thing.
Daniel Dunbar196ea442010-04-06 01:07:44 +0000146 return LValue::MakeBitfield(V, *Info, Quals.getCVRQualifiers());
Daniel Dunbar9fd114d2009-04-22 07:32:20 +0000147}
148
149///
150
Daniel Dunbar303e2c22008-08-11 02:45:11 +0000151namespace {
Daniel Dunbar8b8683f2008-08-12 00:12:39 +0000152
Daniel Dunbar59e476b2009-08-03 17:06:42 +0000153typedef std::vector<llvm::Constant*> ConstantVector;
Daniel Dunbareb1f9a22008-08-27 02:31:56 +0000154
Daniel Dunbar59e476b2009-08-03 17:06:42 +0000155// FIXME: We should find a nicer way to make the labels for metadata, string
156// concatenation is lame.
Daniel Dunbarb036db82008-08-13 03:21:16 +0000157
Fariborz Jahanian279eda62009-01-21 22:04:16 +0000158class ObjCCommonTypesHelper {
Owen Anderson170229f2009-07-14 23:10:40 +0000159protected:
160 llvm::LLVMContext &VMContext;
Daniel Dunbar59e476b2009-08-03 17:06:42 +0000161
Fariborz Jahanian5d5ed2d2009-05-11 19:25:47 +0000162private:
163 llvm::Constant *getMessageSendFn() const {
164 // id objc_msgSend (id, SEL, ...)
165 std::vector<const llvm::Type*> Params;
166 Params.push_back(ObjectPtrTy);
167 Params.push_back(SelectorPtrTy);
168 return
Daniel Dunbar59e476b2009-08-03 17:06:42 +0000169 CGM.CreateRuntimeFunction(llvm::FunctionType::get(ObjectPtrTy,
170 Params, true),
171 "objc_msgSend");
Fariborz Jahanian5d5ed2d2009-05-11 19:25:47 +0000172 }
Daniel Dunbar59e476b2009-08-03 17:06:42 +0000173
Fariborz Jahanian5d5ed2d2009-05-11 19:25:47 +0000174 llvm::Constant *getMessageSendStretFn() const {
175 // id objc_msgSend_stret (id, SEL, ...)
176 std::vector<const llvm::Type*> Params;
177 Params.push_back(ObjectPtrTy);
178 Params.push_back(SelectorPtrTy);
179 return
Owen Anderson41a75022009-08-13 21:57:51 +0000180 CGM.CreateRuntimeFunction(llvm::FunctionType::get(llvm::Type::getVoidTy(VMContext),
Daniel Dunbar59e476b2009-08-03 17:06:42 +0000181 Params, true),
182 "objc_msgSend_stret");
183
Fariborz Jahanian5d5ed2d2009-05-11 19:25:47 +0000184 }
Daniel Dunbar59e476b2009-08-03 17:06:42 +0000185
Fariborz Jahanian5d5ed2d2009-05-11 19:25:47 +0000186 llvm::Constant *getMessageSendFpretFn() const {
187 // FIXME: This should be long double on x86_64?
188 // [double | long double] objc_msgSend_fpret(id self, SEL op, ...)
189 std::vector<const llvm::Type*> Params;
190 Params.push_back(ObjectPtrTy);
191 Params.push_back(SelectorPtrTy);
192 return
Owen Anderson41a75022009-08-13 21:57:51 +0000193 CGM.CreateRuntimeFunction(llvm::FunctionType::get(
194 llvm::Type::getDoubleTy(VMContext),
Daniel Dunbar59e476b2009-08-03 17:06:42 +0000195 Params,
196 true),
197 "objc_msgSend_fpret");
198
Fariborz Jahanian5d5ed2d2009-05-11 19:25:47 +0000199 }
Daniel Dunbar59e476b2009-08-03 17:06:42 +0000200
Fariborz Jahanian5d5ed2d2009-05-11 19:25:47 +0000201 llvm::Constant *getMessageSendSuperFn() const {
202 // id objc_msgSendSuper(struct objc_super *super, SEL op, ...)
203 const char *SuperName = "objc_msgSendSuper";
204 std::vector<const llvm::Type*> Params;
205 Params.push_back(SuperPtrTy);
206 Params.push_back(SelectorPtrTy);
Owen Anderson9793f0e2009-07-29 22:16:19 +0000207 return CGM.CreateRuntimeFunction(llvm::FunctionType::get(ObjectPtrTy,
Fariborz Jahanian5d5ed2d2009-05-11 19:25:47 +0000208 Params, true),
209 SuperName);
210 }
Daniel Dunbar59e476b2009-08-03 17:06:42 +0000211
Fariborz Jahanian5d5ed2d2009-05-11 19:25:47 +0000212 llvm::Constant *getMessageSendSuperFn2() const {
213 // id objc_msgSendSuper2(struct objc_super *super, SEL op, ...)
214 const char *SuperName = "objc_msgSendSuper2";
215 std::vector<const llvm::Type*> Params;
216 Params.push_back(SuperPtrTy);
217 Params.push_back(SelectorPtrTy);
Owen Anderson9793f0e2009-07-29 22:16:19 +0000218 return CGM.CreateRuntimeFunction(llvm::FunctionType::get(ObjectPtrTy,
Fariborz Jahanian5d5ed2d2009-05-11 19:25:47 +0000219 Params, true),
220 SuperName);
221 }
Daniel Dunbar59e476b2009-08-03 17:06:42 +0000222
Fariborz Jahanian5d5ed2d2009-05-11 19:25:47 +0000223 llvm::Constant *getMessageSendSuperStretFn() const {
224 // void objc_msgSendSuper_stret(void * stretAddr, struct objc_super *super,
225 // SEL op, ...)
226 std::vector<const llvm::Type*> Params;
227 Params.push_back(Int8PtrTy);
228 Params.push_back(SuperPtrTy);
229 Params.push_back(SelectorPtrTy);
Owen Anderson170229f2009-07-14 23:10:40 +0000230 return CGM.CreateRuntimeFunction(
Owen Anderson41a75022009-08-13 21:57:51 +0000231 llvm::FunctionType::get(llvm::Type::getVoidTy(VMContext),
Daniel Dunbar59e476b2009-08-03 17:06:42 +0000232 Params, true),
233 "objc_msgSendSuper_stret");
Fariborz Jahanian5d5ed2d2009-05-11 19:25:47 +0000234 }
Daniel Dunbar59e476b2009-08-03 17:06:42 +0000235
Fariborz Jahanian5d5ed2d2009-05-11 19:25:47 +0000236 llvm::Constant *getMessageSendSuperStretFn2() const {
237 // void objc_msgSendSuper2_stret(void * stretAddr, struct objc_super *super,
238 // SEL op, ...)
239 std::vector<const llvm::Type*> Params;
240 Params.push_back(Int8PtrTy);
241 Params.push_back(SuperPtrTy);
242 Params.push_back(SelectorPtrTy);
Owen Anderson170229f2009-07-14 23:10:40 +0000243 return CGM.CreateRuntimeFunction(
Owen Anderson41a75022009-08-13 21:57:51 +0000244 llvm::FunctionType::get(llvm::Type::getVoidTy(VMContext),
Daniel Dunbar59e476b2009-08-03 17:06:42 +0000245 Params, true),
246 "objc_msgSendSuper2_stret");
Fariborz Jahanian5d5ed2d2009-05-11 19:25:47 +0000247 }
Daniel Dunbar59e476b2009-08-03 17:06:42 +0000248
Fariborz Jahanian5d5ed2d2009-05-11 19:25:47 +0000249 llvm::Constant *getMessageSendSuperFpretFn() const {
250 // There is no objc_msgSendSuper_fpret? How can that work?
251 return getMessageSendSuperFn();
252 }
Daniel Dunbar59e476b2009-08-03 17:06:42 +0000253
Fariborz Jahanian5d5ed2d2009-05-11 19:25:47 +0000254 llvm::Constant *getMessageSendSuperFpretFn2() const {
255 // There is no objc_msgSendSuper_fpret? How can that work?
256 return getMessageSendSuperFn2();
257 }
Daniel Dunbar59e476b2009-08-03 17:06:42 +0000258
Fariborz Jahanian279eda62009-01-21 22:04:16 +0000259protected:
260 CodeGen::CodeGenModule &CGM;
Daniel Dunbar59e476b2009-08-03 17:06:42 +0000261
Daniel Dunbar8b8683f2008-08-12 00:12:39 +0000262public:
Fariborz Jahanianaedcfa42009-03-23 19:10:40 +0000263 const llvm::Type *ShortTy, *IntTy, *LongTy, *LongLongTy;
Daniel Dunbar22d82ed2008-08-21 04:36:09 +0000264 const llvm::Type *Int8PtrTy;
Daniel Dunbar59e476b2009-08-03 17:06:42 +0000265
Daniel Dunbar5d715592008-08-12 05:28:47 +0000266 /// ObjectPtrTy - LLVM type for object handles (typeof(id))
267 const llvm::Type *ObjectPtrTy;
Daniel Dunbar59e476b2009-08-03 17:06:42 +0000268
Fariborz Jahanian406b1172008-11-18 20:18:11 +0000269 /// PtrObjectPtrTy - LLVM type for id *
270 const llvm::Type *PtrObjectPtrTy;
Daniel Dunbar59e476b2009-08-03 17:06:42 +0000271
Daniel Dunbar7bd00bd2008-08-12 06:48:42 +0000272 /// SelectorPtrTy - LLVM type for selector handles (typeof(SEL))
Daniel Dunbar5d715592008-08-12 05:28:47 +0000273 const llvm::Type *SelectorPtrTy;
Daniel Dunbarb036db82008-08-13 03:21:16 +0000274 /// ProtocolPtrTy - LLVM type for external protocol handles
275 /// (typeof(Protocol))
276 const llvm::Type *ExternalProtocolPtrTy;
Daniel Dunbar59e476b2009-08-03 17:06:42 +0000277
Daniel Dunbarc722b852008-08-30 03:02:31 +0000278 // SuperCTy - clang type for struct objc_super.
279 QualType SuperCTy;
280 // SuperPtrCTy - clang type for struct objc_super *.
281 QualType SuperPtrCTy;
Daniel Dunbar59e476b2009-08-03 17:06:42 +0000282
Daniel Dunbarf6397fe2008-08-23 04:28:29 +0000283 /// SuperTy - LLVM type for struct objc_super.
284 const llvm::StructType *SuperTy;
Daniel Dunbar97ff50d2008-08-23 09:25:55 +0000285 /// SuperPtrTy - LLVM type for struct objc_super *.
286 const llvm::Type *SuperPtrTy;
Daniel Dunbar59e476b2009-08-03 17:06:42 +0000287
Fariborz Jahanianb15a3d52009-01-22 23:02:58 +0000288 /// PropertyTy - LLVM type for struct objc_property (struct _prop_t
289 /// in GCC parlance).
290 const llvm::StructType *PropertyTy;
Daniel Dunbar59e476b2009-08-03 17:06:42 +0000291
Fariborz Jahanianb15a3d52009-01-22 23:02:58 +0000292 /// PropertyListTy - LLVM type for struct objc_property_list
293 /// (_prop_list_t in GCC parlance).
294 const llvm::StructType *PropertyListTy;
295 /// PropertyListPtrTy - LLVM type for struct objc_property_list*.
296 const llvm::Type *PropertyListPtrTy;
Daniel Dunbar59e476b2009-08-03 17:06:42 +0000297
Fariborz Jahanianb15a3d52009-01-22 23:02:58 +0000298 // MethodTy - LLVM type for struct objc_method.
299 const llvm::StructType *MethodTy;
Daniel Dunbar59e476b2009-08-03 17:06:42 +0000300
Fariborz Jahanian0232c052009-01-23 01:46:23 +0000301 /// CacheTy - LLVM type for struct objc_cache.
302 const llvm::Type *CacheTy;
303 /// CachePtrTy - LLVM type for struct objc_cache *.
304 const llvm::Type *CachePtrTy;
Daniel Dunbar59e476b2009-08-03 17:06:42 +0000305
Chris Lattnerce8754e2009-04-22 02:44:54 +0000306 llvm::Constant *getGetPropertyFn() {
307 CodeGen::CodeGenTypes &Types = CGM.getTypes();
308 ASTContext &Ctx = CGM.getContext();
309 // id objc_getProperty (id, SEL, ptrdiff_t, bool)
John McCall2da83a32010-02-26 00:48:12 +0000310 llvm::SmallVector<CanQualType,4> Params;
311 CanQualType IdType = Ctx.getCanonicalParamType(Ctx.getObjCIdType());
312 CanQualType SelType = Ctx.getCanonicalParamType(Ctx.getObjCSelType());
Chris Lattnerce8754e2009-04-22 02:44:54 +0000313 Params.push_back(IdType);
314 Params.push_back(SelType);
315 Params.push_back(Ctx.LongTy);
316 Params.push_back(Ctx.BoolTy);
317 const llvm::FunctionType *FTy =
John McCallab26cfa2010-02-05 21:31:56 +0000318 Types.GetFunctionType(Types.getFunctionInfo(IdType, Params,
Rafael Espindolac50c27c2010-03-30 20:24:48 +0000319 FunctionType::ExtInfo()),
320 false);
Chris Lattnerce8754e2009-04-22 02:44:54 +0000321 return CGM.CreateRuntimeFunction(FTy, "objc_getProperty");
322 }
Daniel Dunbar59e476b2009-08-03 17:06:42 +0000323
Chris Lattnerce8754e2009-04-22 02:44:54 +0000324 llvm::Constant *getSetPropertyFn() {
325 CodeGen::CodeGenTypes &Types = CGM.getTypes();
326 ASTContext &Ctx = CGM.getContext();
327 // void objc_setProperty (id, SEL, ptrdiff_t, id, bool, bool)
John McCall2da83a32010-02-26 00:48:12 +0000328 llvm::SmallVector<CanQualType,6> Params;
329 CanQualType IdType = Ctx.getCanonicalParamType(Ctx.getObjCIdType());
330 CanQualType SelType = Ctx.getCanonicalParamType(Ctx.getObjCSelType());
Chris Lattnerce8754e2009-04-22 02:44:54 +0000331 Params.push_back(IdType);
332 Params.push_back(SelType);
333 Params.push_back(Ctx.LongTy);
334 Params.push_back(IdType);
335 Params.push_back(Ctx.BoolTy);
336 Params.push_back(Ctx.BoolTy);
337 const llvm::FunctionType *FTy =
John McCallab26cfa2010-02-05 21:31:56 +0000338 Types.GetFunctionType(Types.getFunctionInfo(Ctx.VoidTy, Params,
Rafael Espindolac50c27c2010-03-30 20:24:48 +0000339 FunctionType::ExtInfo()),
340 false);
Chris Lattnerce8754e2009-04-22 02:44:54 +0000341 return CGM.CreateRuntimeFunction(FTy, "objc_setProperty");
342 }
Daniel Dunbar59e476b2009-08-03 17:06:42 +0000343
Fariborz Jahanian5a8c2032010-04-12 18:18:10 +0000344
345 llvm::Constant *getCopyStructFn() {
346 CodeGen::CodeGenTypes &Types = CGM.getTypes();
347 ASTContext &Ctx = CGM.getContext();
348 // void objc_copyStruct (void *, const void *, size_t, bool, bool)
349 llvm::SmallVector<CanQualType,5> Params;
350 Params.push_back(Ctx.VoidPtrTy);
351 Params.push_back(Ctx.VoidPtrTy);
352 Params.push_back(Ctx.LongTy);
353 Params.push_back(Ctx.BoolTy);
354 Params.push_back(Ctx.BoolTy);
355 const llvm::FunctionType *FTy =
356 Types.GetFunctionType(Types.getFunctionInfo(Ctx.VoidTy, Params,
357 FunctionType::ExtInfo()),
358 false);
359 return CGM.CreateRuntimeFunction(FTy, "objc_copyStruct");
360 }
361
Chris Lattnerce8754e2009-04-22 02:44:54 +0000362 llvm::Constant *getEnumerationMutationFn() {
Daniel Dunbar9d82da42009-07-11 20:32:50 +0000363 CodeGen::CodeGenTypes &Types = CGM.getTypes();
364 ASTContext &Ctx = CGM.getContext();
Chris Lattnerce8754e2009-04-22 02:44:54 +0000365 // void objc_enumerationMutation (id)
John McCall2da83a32010-02-26 00:48:12 +0000366 llvm::SmallVector<CanQualType,1> Params;
367 Params.push_back(Ctx.getCanonicalParamType(Ctx.getObjCIdType()));
Daniel Dunbar9d82da42009-07-11 20:32:50 +0000368 const llvm::FunctionType *FTy =
John McCallab26cfa2010-02-05 21:31:56 +0000369 Types.GetFunctionType(Types.getFunctionInfo(Ctx.VoidTy, Params,
Rafael Espindolac50c27c2010-03-30 20:24:48 +0000370 FunctionType::ExtInfo()),
371 false);
Chris Lattnerce8754e2009-04-22 02:44:54 +0000372 return CGM.CreateRuntimeFunction(FTy, "objc_enumerationMutation");
373 }
Daniel Dunbar59e476b2009-08-03 17:06:42 +0000374
Fariborz Jahanianeee54df2009-01-22 00:37:21 +0000375 /// GcReadWeakFn -- LLVM objc_read_weak (id *src) function.
Chris Lattnerce8754e2009-04-22 02:44:54 +0000376 llvm::Constant *getGcReadWeakFn() {
377 // id objc_read_weak (id *)
378 std::vector<const llvm::Type*> Args;
379 Args.push_back(ObjectPtrTy->getPointerTo());
Daniel Dunbar59e476b2009-08-03 17:06:42 +0000380 llvm::FunctionType *FTy =
Owen Anderson9793f0e2009-07-29 22:16:19 +0000381 llvm::FunctionType::get(ObjectPtrTy, Args, false);
Chris Lattnerce8754e2009-04-22 02:44:54 +0000382 return CGM.CreateRuntimeFunction(FTy, "objc_read_weak");
Daniel Dunbar59e476b2009-08-03 17:06:42 +0000383 }
384
Fariborz Jahanianeee54df2009-01-22 00:37:21 +0000385 /// GcAssignWeakFn -- LLVM objc_assign_weak function.
Chris Lattner6fdd57c2009-04-17 22:12:36 +0000386 llvm::Constant *getGcAssignWeakFn() {
387 // id objc_assign_weak (id, id *)
388 std::vector<const llvm::Type*> Args(1, ObjectPtrTy);
389 Args.push_back(ObjectPtrTy->getPointerTo());
390 llvm::FunctionType *FTy =
Owen Anderson9793f0e2009-07-29 22:16:19 +0000391 llvm::FunctionType::get(ObjectPtrTy, Args, false);
Chris Lattner6fdd57c2009-04-17 22:12:36 +0000392 return CGM.CreateRuntimeFunction(FTy, "objc_assign_weak");
393 }
Daniel Dunbar59e476b2009-08-03 17:06:42 +0000394
Fariborz Jahanianeee54df2009-01-22 00:37:21 +0000395 /// GcAssignGlobalFn -- LLVM objc_assign_global function.
Chris Lattner0a696a422009-04-22 02:38:11 +0000396 llvm::Constant *getGcAssignGlobalFn() {
397 // id objc_assign_global(id, id *)
398 std::vector<const llvm::Type*> Args(1, ObjectPtrTy);
399 Args.push_back(ObjectPtrTy->getPointerTo());
Owen Anderson170229f2009-07-14 23:10:40 +0000400 llvm::FunctionType *FTy =
Owen Anderson9793f0e2009-07-29 22:16:19 +0000401 llvm::FunctionType::get(ObjectPtrTy, Args, false);
Chris Lattner0a696a422009-04-22 02:38:11 +0000402 return CGM.CreateRuntimeFunction(FTy, "objc_assign_global");
403 }
Daniel Dunbar59e476b2009-08-03 17:06:42 +0000404
Fariborz Jahanian217af242010-07-20 20:30:03 +0000405 /// GcAssignThreadLocalFn -- LLVM objc_assign_threadlocal function.
406 llvm::Constant *getGcAssignThreadLocalFn() {
407 // id objc_assign_threadlocal(id src, id * dest)
408 std::vector<const llvm::Type*> Args(1, ObjectPtrTy);
409 Args.push_back(ObjectPtrTy->getPointerTo());
410 llvm::FunctionType *FTy =
411 llvm::FunctionType::get(ObjectPtrTy, Args, false);
412 return CGM.CreateRuntimeFunction(FTy, "objc_assign_threadlocal");
413 }
414
Fariborz Jahanianeee54df2009-01-22 00:37:21 +0000415 /// GcAssignIvarFn -- LLVM objc_assign_ivar function.
Chris Lattner0a696a422009-04-22 02:38:11 +0000416 llvm::Constant *getGcAssignIvarFn() {
Fariborz Jahanian7a95d722009-09-24 22:25:38 +0000417 // id objc_assign_ivar(id, id *, ptrdiff_t)
Chris Lattner0a696a422009-04-22 02:38:11 +0000418 std::vector<const llvm::Type*> Args(1, ObjectPtrTy);
419 Args.push_back(ObjectPtrTy->getPointerTo());
Fariborz Jahanian7a95d722009-09-24 22:25:38 +0000420 Args.push_back(LongTy);
Owen Anderson170229f2009-07-14 23:10:40 +0000421 llvm::FunctionType *FTy =
Owen Anderson9793f0e2009-07-29 22:16:19 +0000422 llvm::FunctionType::get(ObjectPtrTy, Args, false);
Chris Lattner0a696a422009-04-22 02:38:11 +0000423 return CGM.CreateRuntimeFunction(FTy, "objc_assign_ivar");
424 }
Daniel Dunbar59e476b2009-08-03 17:06:42 +0000425
Fariborz Jahanian5f21d2f2009-07-08 01:18:33 +0000426 /// GcMemmoveCollectableFn -- LLVM objc_memmove_collectable function.
427 llvm::Constant *GcMemmoveCollectableFn() {
428 // void *objc_memmove_collectable(void *dst, const void *src, size_t size)
429 std::vector<const llvm::Type*> Args(1, Int8PtrTy);
430 Args.push_back(Int8PtrTy);
431 Args.push_back(LongTy);
Owen Anderson9793f0e2009-07-29 22:16:19 +0000432 llvm::FunctionType *FTy = llvm::FunctionType::get(Int8PtrTy, Args, false);
Fariborz Jahanian5f21d2f2009-07-08 01:18:33 +0000433 return CGM.CreateRuntimeFunction(FTy, "objc_memmove_collectable");
434 }
Daniel Dunbar59e476b2009-08-03 17:06:42 +0000435
Fariborz Jahanianeee54df2009-01-22 00:37:21 +0000436 /// GcAssignStrongCastFn -- LLVM objc_assign_strongCast function.
Chris Lattner0a696a422009-04-22 02:38:11 +0000437 llvm::Constant *getGcAssignStrongCastFn() {
Fariborz Jahanian217af242010-07-20 20:30:03 +0000438 // id objc_assign_strongCast(id, id *)
Chris Lattner0a696a422009-04-22 02:38:11 +0000439 std::vector<const llvm::Type*> Args(1, ObjectPtrTy);
440 Args.push_back(ObjectPtrTy->getPointerTo());
Owen Anderson170229f2009-07-14 23:10:40 +0000441 llvm::FunctionType *FTy =
Owen Anderson9793f0e2009-07-29 22:16:19 +0000442 llvm::FunctionType::get(ObjectPtrTy, Args, false);
Chris Lattner0a696a422009-04-22 02:38:11 +0000443 return CGM.CreateRuntimeFunction(FTy, "objc_assign_strongCast");
444 }
Anders Carlsson9ab53d12009-02-16 22:59:18 +0000445
446 /// ExceptionThrowFn - LLVM objc_exception_throw function.
Chris Lattner0a696a422009-04-22 02:38:11 +0000447 llvm::Constant *getExceptionThrowFn() {
448 // void objc_exception_throw(id)
449 std::vector<const llvm::Type*> Args(1, ObjectPtrTy);
450 llvm::FunctionType *FTy =
Owen Anderson41a75022009-08-13 21:57:51 +0000451 llvm::FunctionType::get(llvm::Type::getVoidTy(VMContext), Args, false);
Chris Lattner0a696a422009-04-22 02:38:11 +0000452 return CGM.CreateRuntimeFunction(FTy, "objc_exception_throw");
453 }
Daniel Dunbar59e476b2009-08-03 17:06:42 +0000454
Fariborz Jahanian3336de12010-05-28 17:34:43 +0000455 /// ExceptionRethrowFn - LLVM objc_exception_rethrow function.
456 llvm::Constant *getExceptionRethrowFn() {
457 // void objc_exception_rethrow(void)
458 std::vector<const llvm::Type*> Args;
459 llvm::FunctionType *FTy =
460 llvm::FunctionType::get(llvm::Type::getVoidTy(VMContext), Args, true);
461 return CGM.CreateRuntimeFunction(FTy, "objc_exception_rethrow");
462 }
463
Daniel Dunbar94ceb612009-02-24 01:43:46 +0000464 /// SyncEnterFn - LLVM object_sync_enter function.
Chris Lattnerdcceee72009-04-06 16:53:45 +0000465 llvm::Constant *getSyncEnterFn() {
466 // void objc_sync_enter (id)
467 std::vector<const llvm::Type*> Args(1, ObjectPtrTy);
468 llvm::FunctionType *FTy =
Owen Anderson41a75022009-08-13 21:57:51 +0000469 llvm::FunctionType::get(llvm::Type::getVoidTy(VMContext), Args, false);
Chris Lattnerdcceee72009-04-06 16:53:45 +0000470 return CGM.CreateRuntimeFunction(FTy, "objc_sync_enter");
471 }
Daniel Dunbar59e476b2009-08-03 17:06:42 +0000472
Daniel Dunbar94ceb612009-02-24 01:43:46 +0000473 /// SyncExitFn - LLVM object_sync_exit function.
Chris Lattner0a696a422009-04-22 02:38:11 +0000474 llvm::Constant *getSyncExitFn() {
475 // void objc_sync_exit (id)
476 std::vector<const llvm::Type*> Args(1, ObjectPtrTy);
477 llvm::FunctionType *FTy =
Owen Anderson41a75022009-08-13 21:57:51 +0000478 llvm::FunctionType::get(llvm::Type::getVoidTy(VMContext), Args, false);
Chris Lattner0a696a422009-04-22 02:38:11 +0000479 return CGM.CreateRuntimeFunction(FTy, "objc_sync_exit");
480 }
Daniel Dunbar59e476b2009-08-03 17:06:42 +0000481
Fariborz Jahanian5d5ed2d2009-05-11 19:25:47 +0000482 llvm::Constant *getSendFn(bool IsSuper) const {
483 return IsSuper ? getMessageSendSuperFn() : getMessageSendFn();
484 }
Daniel Dunbar59e476b2009-08-03 17:06:42 +0000485
Fariborz Jahanian5d5ed2d2009-05-11 19:25:47 +0000486 llvm::Constant *getSendFn2(bool IsSuper) const {
487 return IsSuper ? getMessageSendSuperFn2() : getMessageSendFn();
488 }
Daniel Dunbar59e476b2009-08-03 17:06:42 +0000489
Fariborz Jahanian5d5ed2d2009-05-11 19:25:47 +0000490 llvm::Constant *getSendStretFn(bool IsSuper) const {
491 return IsSuper ? getMessageSendSuperStretFn() : getMessageSendStretFn();
492 }
Daniel Dunbar59e476b2009-08-03 17:06:42 +0000493
Fariborz Jahanian5d5ed2d2009-05-11 19:25:47 +0000494 llvm::Constant *getSendStretFn2(bool IsSuper) const {
495 return IsSuper ? getMessageSendSuperStretFn2() : getMessageSendStretFn();
496 }
Daniel Dunbar59e476b2009-08-03 17:06:42 +0000497
Fariborz Jahanian5d5ed2d2009-05-11 19:25:47 +0000498 llvm::Constant *getSendFpretFn(bool IsSuper) const {
499 return IsSuper ? getMessageSendSuperFpretFn() : getMessageSendFpretFn();
500 }
Daniel Dunbar59e476b2009-08-03 17:06:42 +0000501
Fariborz Jahanian5d5ed2d2009-05-11 19:25:47 +0000502 llvm::Constant *getSendFpretFn2(bool IsSuper) const {
503 return IsSuper ? getMessageSendSuperFpretFn2() : getMessageSendFpretFn();
504 }
Daniel Dunbar59e476b2009-08-03 17:06:42 +0000505
Fariborz Jahanian279eda62009-01-21 22:04:16 +0000506 ObjCCommonTypesHelper(CodeGen::CodeGenModule &cgm);
507 ~ObjCCommonTypesHelper(){}
508};
Daniel Dunbarf6397fe2008-08-23 04:28:29 +0000509
Fariborz Jahanian279eda62009-01-21 22:04:16 +0000510/// ObjCTypesHelper - Helper class that encapsulates lazy
511/// construction of varies types used during ObjC generation.
512class ObjCTypesHelper : public ObjCCommonTypesHelper {
Fariborz Jahanian279eda62009-01-21 22:04:16 +0000513public:
Daniel Dunbar7bd00bd2008-08-12 06:48:42 +0000514 /// SymtabTy - LLVM type for struct objc_symtab.
515 const llvm::StructType *SymtabTy;
Daniel Dunbar22d82ed2008-08-21 04:36:09 +0000516 /// SymtabPtrTy - LLVM type for struct objc_symtab *.
517 const llvm::Type *SymtabPtrTy;
Daniel Dunbar7bd00bd2008-08-12 06:48:42 +0000518 /// ModuleTy - LLVM type for struct objc_module.
519 const llvm::StructType *ModuleTy;
Daniel Dunbarcb515c82008-08-12 03:39:23 +0000520
Daniel Dunbarb036db82008-08-13 03:21:16 +0000521 /// ProtocolTy - LLVM type for struct objc_protocol.
522 const llvm::StructType *ProtocolTy;
523 /// ProtocolPtrTy - LLVM type for struct objc_protocol *.
524 const llvm::Type *ProtocolPtrTy;
525 /// ProtocolExtensionTy - LLVM type for struct
526 /// objc_protocol_extension.
527 const llvm::StructType *ProtocolExtensionTy;
528 /// ProtocolExtensionTy - LLVM type for struct
529 /// objc_protocol_extension *.
530 const llvm::Type *ProtocolExtensionPtrTy;
531 /// MethodDescriptionTy - LLVM type for struct
532 /// objc_method_description.
533 const llvm::StructType *MethodDescriptionTy;
534 /// MethodDescriptionListTy - LLVM type for struct
535 /// objc_method_description_list.
536 const llvm::StructType *MethodDescriptionListTy;
537 /// MethodDescriptionListPtrTy - LLVM type for struct
538 /// objc_method_description_list *.
539 const llvm::Type *MethodDescriptionListPtrTy;
Daniel Dunbarb036db82008-08-13 03:21:16 +0000540 /// ProtocolListTy - LLVM type for struct objc_property_list.
541 const llvm::Type *ProtocolListTy;
542 /// ProtocolListPtrTy - LLVM type for struct objc_property_list*.
543 const llvm::Type *ProtocolListPtrTy;
Daniel Dunbar938a77f2008-08-22 20:34:54 +0000544 /// CategoryTy - LLVM type for struct objc_category.
545 const llvm::StructType *CategoryTy;
Daniel Dunbar22d82ed2008-08-21 04:36:09 +0000546 /// ClassTy - LLVM type for struct objc_class.
547 const llvm::StructType *ClassTy;
548 /// ClassPtrTy - LLVM type for struct objc_class *.
549 const llvm::Type *ClassPtrTy;
550 /// ClassExtensionTy - LLVM type for struct objc_class_ext.
551 const llvm::StructType *ClassExtensionTy;
552 /// ClassExtensionPtrTy - LLVM type for struct objc_class_ext *.
553 const llvm::Type *ClassExtensionPtrTy;
Daniel Dunbar22d82ed2008-08-21 04:36:09 +0000554 // IvarTy - LLVM type for struct objc_ivar.
555 const llvm::StructType *IvarTy;
556 /// IvarListTy - LLVM type for struct objc_ivar_list.
557 const llvm::Type *IvarListTy;
558 /// IvarListPtrTy - LLVM type for struct objc_ivar_list *.
559 const llvm::Type *IvarListPtrTy;
Daniel Dunbar22d82ed2008-08-21 04:36:09 +0000560 /// MethodListTy - LLVM type for struct objc_method_list.
561 const llvm::Type *MethodListTy;
562 /// MethodListPtrTy - LLVM type for struct objc_method_list *.
563 const llvm::Type *MethodListPtrTy;
Daniel Dunbar59e476b2009-08-03 17:06:42 +0000564
Anders Carlsson9ff22482008-09-09 10:10:21 +0000565 /// ExceptionDataTy - LLVM type for struct _objc_exception_data.
566 const llvm::Type *ExceptionDataTy;
Daniel Dunbar59e476b2009-08-03 17:06:42 +0000567
Anders Carlsson9ff22482008-09-09 10:10:21 +0000568 /// ExceptionTryEnterFn - LLVM objc_exception_try_enter function.
Chris Lattnerc6406db2009-04-22 02:26:14 +0000569 llvm::Constant *getExceptionTryEnterFn() {
570 std::vector<const llvm::Type*> Params;
Owen Anderson9793f0e2009-07-29 22:16:19 +0000571 Params.push_back(llvm::PointerType::getUnqual(ExceptionDataTy));
Owen Anderson170229f2009-07-14 23:10:40 +0000572 return CGM.CreateRuntimeFunction(
Owen Anderson41a75022009-08-13 21:57:51 +0000573 llvm::FunctionType::get(llvm::Type::getVoidTy(VMContext),
Daniel Dunbar59e476b2009-08-03 17:06:42 +0000574 Params, false),
575 "objc_exception_try_enter");
Chris Lattnerc6406db2009-04-22 02:26:14 +0000576 }
Anders Carlsson9ff22482008-09-09 10:10:21 +0000577
578 /// ExceptionTryExitFn - LLVM objc_exception_try_exit function.
Chris Lattnerc6406db2009-04-22 02:26:14 +0000579 llvm::Constant *getExceptionTryExitFn() {
580 std::vector<const llvm::Type*> Params;
Owen Anderson9793f0e2009-07-29 22:16:19 +0000581 Params.push_back(llvm::PointerType::getUnqual(ExceptionDataTy));
Owen Anderson170229f2009-07-14 23:10:40 +0000582 return CGM.CreateRuntimeFunction(
Owen Anderson41a75022009-08-13 21:57:51 +0000583 llvm::FunctionType::get(llvm::Type::getVoidTy(VMContext),
Daniel Dunbar59e476b2009-08-03 17:06:42 +0000584 Params, false),
585 "objc_exception_try_exit");
Chris Lattnerc6406db2009-04-22 02:26:14 +0000586 }
Anders Carlsson9ff22482008-09-09 10:10:21 +0000587
588 /// ExceptionExtractFn - LLVM objc_exception_extract function.
Chris Lattnerc6406db2009-04-22 02:26:14 +0000589 llvm::Constant *getExceptionExtractFn() {
590 std::vector<const llvm::Type*> Params;
Owen Anderson9793f0e2009-07-29 22:16:19 +0000591 Params.push_back(llvm::PointerType::getUnqual(ExceptionDataTy));
592 return CGM.CreateRuntimeFunction(llvm::FunctionType::get(ObjectPtrTy,
Chris Lattnerc6406db2009-04-22 02:26:14 +0000593 Params, false),
594 "objc_exception_extract");
Daniel Dunbar59e476b2009-08-03 17:06:42 +0000595
Chris Lattnerc6406db2009-04-22 02:26:14 +0000596 }
Daniel Dunbar59e476b2009-08-03 17:06:42 +0000597
Anders Carlsson9ff22482008-09-09 10:10:21 +0000598 /// ExceptionMatchFn - LLVM objc_exception_match function.
Chris Lattnerc6406db2009-04-22 02:26:14 +0000599 llvm::Constant *getExceptionMatchFn() {
600 std::vector<const llvm::Type*> Params;
601 Params.push_back(ClassPtrTy);
602 Params.push_back(ObjectPtrTy);
Daniel Dunbar59e476b2009-08-03 17:06:42 +0000603 return CGM.CreateRuntimeFunction(
Owen Anderson41a75022009-08-13 21:57:51 +0000604 llvm::FunctionType::get(llvm::Type::getInt32Ty(VMContext),
Daniel Dunbar59e476b2009-08-03 17:06:42 +0000605 Params, false),
606 "objc_exception_match");
607
Chris Lattnerc6406db2009-04-22 02:26:14 +0000608 }
Daniel Dunbar59e476b2009-08-03 17:06:42 +0000609
Anders Carlsson9ff22482008-09-09 10:10:21 +0000610 /// SetJmpFn - LLVM _setjmp function.
Chris Lattnerc6406db2009-04-22 02:26:14 +0000611 llvm::Constant *getSetJmpFn() {
612 std::vector<const llvm::Type*> Params;
Benjamin Kramerabd5b902009-10-13 10:07:13 +0000613 Params.push_back(llvm::Type::getInt32PtrTy(VMContext));
Chris Lattnerc6406db2009-04-22 02:26:14 +0000614 return
Owen Anderson41a75022009-08-13 21:57:51 +0000615 CGM.CreateRuntimeFunction(llvm::FunctionType::get(llvm::Type::getInt32Ty(VMContext),
Chris Lattnerc6406db2009-04-22 02:26:14 +0000616 Params, false),
617 "_setjmp");
Daniel Dunbar59e476b2009-08-03 17:06:42 +0000618
Chris Lattnerc6406db2009-04-22 02:26:14 +0000619 }
Daniel Dunbar59e476b2009-08-03 17:06:42 +0000620
Daniel Dunbar8b8683f2008-08-12 00:12:39 +0000621public:
622 ObjCTypesHelper(CodeGen::CodeGenModule &cgm);
Fariborz Jahanian279eda62009-01-21 22:04:16 +0000623 ~ObjCTypesHelper() {}
Daniel Dunbar8b8683f2008-08-12 00:12:39 +0000624};
625
Fariborz Jahanianb15a3d52009-01-22 23:02:58 +0000626/// ObjCNonFragileABITypesHelper - will have all types needed by objective-c's
Fariborz Jahanian279eda62009-01-21 22:04:16 +0000627/// modern abi
Fariborz Jahanianb15a3d52009-01-22 23:02:58 +0000628class ObjCNonFragileABITypesHelper : public ObjCCommonTypesHelper {
Fariborz Jahaniane4dc35d2009-02-04 20:42:28 +0000629public:
Daniel Dunbar59e476b2009-08-03 17:06:42 +0000630
Fariborz Jahanian0232c052009-01-23 01:46:23 +0000631 // MethodListnfABITy - LLVM for struct _method_list_t
632 const llvm::StructType *MethodListnfABITy;
Daniel Dunbar59e476b2009-08-03 17:06:42 +0000633
Fariborz Jahanian0232c052009-01-23 01:46:23 +0000634 // MethodListnfABIPtrTy - LLVM for struct _method_list_t*
635 const llvm::Type *MethodListnfABIPtrTy;
Daniel Dunbar59e476b2009-08-03 17:06:42 +0000636
Fariborz Jahanian0232c052009-01-23 01:46:23 +0000637 // ProtocolnfABITy = LLVM for struct _protocol_t
638 const llvm::StructType *ProtocolnfABITy;
Daniel Dunbar59e476b2009-08-03 17:06:42 +0000639
Daniel Dunbar8de90f02009-02-15 07:36:20 +0000640 // ProtocolnfABIPtrTy = LLVM for struct _protocol_t*
641 const llvm::Type *ProtocolnfABIPtrTy;
642
Fariborz Jahanian0232c052009-01-23 01:46:23 +0000643 // ProtocolListnfABITy - LLVM for struct _objc_protocol_list
644 const llvm::StructType *ProtocolListnfABITy;
Daniel Dunbar59e476b2009-08-03 17:06:42 +0000645
Fariborz Jahanian0232c052009-01-23 01:46:23 +0000646 // ProtocolListnfABIPtrTy - LLVM for struct _objc_protocol_list*
647 const llvm::Type *ProtocolListnfABIPtrTy;
Daniel Dunbar59e476b2009-08-03 17:06:42 +0000648
Fariborz Jahanian0232c052009-01-23 01:46:23 +0000649 // ClassnfABITy - LLVM for struct _class_t
650 const llvm::StructType *ClassnfABITy;
Daniel Dunbar59e476b2009-08-03 17:06:42 +0000651
Fariborz Jahanian71394042009-01-23 23:53:38 +0000652 // ClassnfABIPtrTy - LLVM for struct _class_t*
653 const llvm::Type *ClassnfABIPtrTy;
Daniel Dunbar59e476b2009-08-03 17:06:42 +0000654
Fariborz Jahanian0232c052009-01-23 01:46:23 +0000655 // IvarnfABITy - LLVM for struct _ivar_t
656 const llvm::StructType *IvarnfABITy;
Daniel Dunbar59e476b2009-08-03 17:06:42 +0000657
Fariborz Jahanian0232c052009-01-23 01:46:23 +0000658 // IvarListnfABITy - LLVM for struct _ivar_list_t
659 const llvm::StructType *IvarListnfABITy;
Daniel Dunbar59e476b2009-08-03 17:06:42 +0000660
Fariborz Jahanian0232c052009-01-23 01:46:23 +0000661 // IvarListnfABIPtrTy = LLVM for struct _ivar_list_t*
662 const llvm::Type *IvarListnfABIPtrTy;
Daniel Dunbar59e476b2009-08-03 17:06:42 +0000663
Fariborz Jahanian0232c052009-01-23 01:46:23 +0000664 // ClassRonfABITy - LLVM for struct _class_ro_t
665 const llvm::StructType *ClassRonfABITy;
Daniel Dunbar59e476b2009-08-03 17:06:42 +0000666
Fariborz Jahanian0232c052009-01-23 01:46:23 +0000667 // ImpnfABITy - LLVM for id (*)(id, SEL, ...)
668 const llvm::Type *ImpnfABITy;
Daniel Dunbar59e476b2009-08-03 17:06:42 +0000669
Fariborz Jahanian0232c052009-01-23 01:46:23 +0000670 // CategorynfABITy - LLVM for struct _category_t
671 const llvm::StructType *CategorynfABITy;
Daniel Dunbar59e476b2009-08-03 17:06:42 +0000672
Fariborz Jahanian82c72e12009-02-03 23:49:23 +0000673 // New types for nonfragile abi messaging.
Daniel Dunbar59e476b2009-08-03 17:06:42 +0000674
Fariborz Jahanian82c72e12009-02-03 23:49:23 +0000675 // MessageRefTy - LLVM for:
676 // struct _message_ref_t {
677 // IMP messenger;
678 // SEL name;
679 // };
680 const llvm::StructType *MessageRefTy;
Fariborz Jahaniane4dc35d2009-02-04 20:42:28 +0000681 // MessageRefCTy - clang type for struct _message_ref_t
682 QualType MessageRefCTy;
Daniel Dunbar59e476b2009-08-03 17:06:42 +0000683
Fariborz Jahanian82c72e12009-02-03 23:49:23 +0000684 // MessageRefPtrTy - LLVM for struct _message_ref_t*
685 const llvm::Type *MessageRefPtrTy;
Fariborz Jahaniane4dc35d2009-02-04 20:42:28 +0000686 // MessageRefCPtrTy - clang type for struct _message_ref_t*
687 QualType MessageRefCPtrTy;
Daniel Dunbar59e476b2009-08-03 17:06:42 +0000688
Fariborz Jahanian4e87c832009-02-05 01:13:09 +0000689 // MessengerTy - Type of the messenger (shown as IMP above)
690 const llvm::FunctionType *MessengerTy;
Daniel Dunbar59e476b2009-08-03 17:06:42 +0000691
Fariborz Jahanian82c72e12009-02-03 23:49:23 +0000692 // SuperMessageRefTy - LLVM for:
693 // struct _super_message_ref_t {
694 // SUPER_IMP messenger;
695 // SEL name;
696 // };
697 const llvm::StructType *SuperMessageRefTy;
Daniel Dunbar59e476b2009-08-03 17:06:42 +0000698
Fariborz Jahanian82c72e12009-02-03 23:49:23 +0000699 // SuperMessageRefPtrTy - LLVM for struct _super_message_ref_t*
700 const llvm::Type *SuperMessageRefPtrTy;
Daniel Dunbar0b0dcd92009-02-24 07:47:38 +0000701
Chris Lattner2dfdb3e2009-04-22 02:53:24 +0000702 llvm::Constant *getMessageSendFixupFn() {
703 // id objc_msgSend_fixup(id, struct message_ref_t*, ...)
704 std::vector<const llvm::Type*> Params;
705 Params.push_back(ObjectPtrTy);
706 Params.push_back(MessageRefPtrTy);
Owen Anderson9793f0e2009-07-29 22:16:19 +0000707 return CGM.CreateRuntimeFunction(llvm::FunctionType::get(ObjectPtrTy,
Chris Lattner2dfdb3e2009-04-22 02:53:24 +0000708 Params, true),
709 "objc_msgSend_fixup");
710 }
Daniel Dunbar59e476b2009-08-03 17:06:42 +0000711
Chris Lattner2dfdb3e2009-04-22 02:53:24 +0000712 llvm::Constant *getMessageSendFpretFixupFn() {
713 // id objc_msgSend_fpret_fixup(id, struct message_ref_t*, ...)
714 std::vector<const llvm::Type*> Params;
715 Params.push_back(ObjectPtrTy);
716 Params.push_back(MessageRefPtrTy);
Owen Anderson9793f0e2009-07-29 22:16:19 +0000717 return CGM.CreateRuntimeFunction(llvm::FunctionType::get(ObjectPtrTy,
Chris Lattner2dfdb3e2009-04-22 02:53:24 +0000718 Params, true),
719 "objc_msgSend_fpret_fixup");
720 }
Daniel Dunbar59e476b2009-08-03 17:06:42 +0000721
Chris Lattner2dfdb3e2009-04-22 02:53:24 +0000722 llvm::Constant *getMessageSendStretFixupFn() {
723 // id objc_msgSend_stret_fixup(id, struct message_ref_t*, ...)
724 std::vector<const llvm::Type*> Params;
725 Params.push_back(ObjectPtrTy);
726 Params.push_back(MessageRefPtrTy);
Owen Anderson9793f0e2009-07-29 22:16:19 +0000727 return CGM.CreateRuntimeFunction(llvm::FunctionType::get(ObjectPtrTy,
Chris Lattner2dfdb3e2009-04-22 02:53:24 +0000728 Params, true),
729 "objc_msgSend_stret_fixup");
730 }
Daniel Dunbar59e476b2009-08-03 17:06:42 +0000731
Chris Lattner2dfdb3e2009-04-22 02:53:24 +0000732 llvm::Constant *getMessageSendIdFixupFn() {
733 // id objc_msgSendId_fixup(id, struct message_ref_t*, ...)
734 std::vector<const llvm::Type*> Params;
735 Params.push_back(ObjectPtrTy);
736 Params.push_back(MessageRefPtrTy);
Owen Anderson9793f0e2009-07-29 22:16:19 +0000737 return CGM.CreateRuntimeFunction(llvm::FunctionType::get(ObjectPtrTy,
Chris Lattner2dfdb3e2009-04-22 02:53:24 +0000738 Params, true),
739 "objc_msgSendId_fixup");
740 }
Daniel Dunbar59e476b2009-08-03 17:06:42 +0000741
Chris Lattner2dfdb3e2009-04-22 02:53:24 +0000742 llvm::Constant *getMessageSendIdStretFixupFn() {
743 // id objc_msgSendId_stret_fixup(id, struct message_ref_t*, ...)
744 std::vector<const llvm::Type*> Params;
745 Params.push_back(ObjectPtrTy);
746 Params.push_back(MessageRefPtrTy);
Owen Anderson9793f0e2009-07-29 22:16:19 +0000747 return CGM.CreateRuntimeFunction(llvm::FunctionType::get(ObjectPtrTy,
Chris Lattner2dfdb3e2009-04-22 02:53:24 +0000748 Params, true),
749 "objc_msgSendId_stret_fixup");
750 }
751 llvm::Constant *getMessageSendSuper2FixupFn() {
Daniel Dunbar59e476b2009-08-03 17:06:42 +0000752 // id objc_msgSendSuper2_fixup (struct objc_super *,
Chris Lattner2dfdb3e2009-04-22 02:53:24 +0000753 // struct _super_message_ref_t*, ...)
754 std::vector<const llvm::Type*> Params;
755 Params.push_back(SuperPtrTy);
756 Params.push_back(SuperMessageRefPtrTy);
Owen Anderson9793f0e2009-07-29 22:16:19 +0000757 return CGM.CreateRuntimeFunction(llvm::FunctionType::get(ObjectPtrTy,
Chris Lattner2dfdb3e2009-04-22 02:53:24 +0000758 Params, true),
759 "objc_msgSendSuper2_fixup");
760 }
Daniel Dunbar59e476b2009-08-03 17:06:42 +0000761
Chris Lattner2dfdb3e2009-04-22 02:53:24 +0000762 llvm::Constant *getMessageSendSuper2StretFixupFn() {
Daniel Dunbar59e476b2009-08-03 17:06:42 +0000763 // id objc_msgSendSuper2_stret_fixup(struct objc_super *,
Chris Lattner2dfdb3e2009-04-22 02:53:24 +0000764 // struct _super_message_ref_t*, ...)
765 std::vector<const llvm::Type*> Params;
766 Params.push_back(SuperPtrTy);
767 Params.push_back(SuperMessageRefPtrTy);
Owen Anderson9793f0e2009-07-29 22:16:19 +0000768 return CGM.CreateRuntimeFunction(llvm::FunctionType::get(ObjectPtrTy,
Chris Lattner2dfdb3e2009-04-22 02:53:24 +0000769 Params, true),
770 "objc_msgSendSuper2_stret_fixup");
771 }
Daniel Dunbar59e476b2009-08-03 17:06:42 +0000772
773
774
Daniel Dunbar0b0dcd92009-02-24 07:47:38 +0000775 /// EHPersonalityPtr - LLVM value for an i8* to the Objective-C
776 /// exception personality function.
Chris Lattnerdcceee72009-04-06 16:53:45 +0000777 llvm::Value *getEHPersonalityPtr() {
Daniel Dunbar59e476b2009-08-03 17:06:42 +0000778 llvm::Constant *Personality =
Owen Anderson41a75022009-08-13 21:57:51 +0000779 CGM.CreateRuntimeFunction(llvm::FunctionType::get(llvm::Type::getInt32Ty(VMContext),
Chris Lattnerdcceee72009-04-06 16:53:45 +0000780 true),
Daniel Dunbar59e476b2009-08-03 17:06:42 +0000781 "__objc_personality_v0");
Owen Andersonade90fd2009-07-29 18:54:39 +0000782 return llvm::ConstantExpr::getBitCast(Personality, Int8PtrTy);
Chris Lattnerdcceee72009-04-06 16:53:45 +0000783 }
Daniel Dunbar0b0dcd92009-02-24 07:47:38 +0000784
Chris Lattnera7c00b42009-04-22 02:15:23 +0000785 llvm::Constant *getUnwindResumeOrRethrowFn() {
786 std::vector<const llvm::Type*> Params;
787 Params.push_back(Int8PtrTy);
Owen Anderson170229f2009-07-14 23:10:40 +0000788 return CGM.CreateRuntimeFunction(
Owen Anderson41a75022009-08-13 21:57:51 +0000789 llvm::FunctionType::get(llvm::Type::getVoidTy(VMContext),
Daniel Dunbar59e476b2009-08-03 17:06:42 +0000790 Params, false),
Daniel Dunbar3241d402010-02-10 18:49:11 +0000791 (CGM.getLangOptions().SjLjExceptions ? "_Unwind_SjLj_Resume" :
792 "_Unwind_Resume_or_Rethrow"));
Chris Lattnera7c00b42009-04-22 02:15:23 +0000793 }
Daniel Dunbar59e476b2009-08-03 17:06:42 +0000794
Chris Lattnera7c00b42009-04-22 02:15:23 +0000795 llvm::Constant *getObjCEndCatchFn() {
Owen Anderson41a75022009-08-13 21:57:51 +0000796 return CGM.CreateRuntimeFunction(llvm::FunctionType::get(llvm::Type::getVoidTy(VMContext),
Chris Lattner3dd1b4b2009-07-01 04:13:52 +0000797 false),
Chris Lattnera7c00b42009-04-22 02:15:23 +0000798 "objc_end_catch");
Daniel Dunbar59e476b2009-08-03 17:06:42 +0000799
Chris Lattnera7c00b42009-04-22 02:15:23 +0000800 }
Daniel Dunbar59e476b2009-08-03 17:06:42 +0000801
Chris Lattnera7c00b42009-04-22 02:15:23 +0000802 llvm::Constant *getObjCBeginCatchFn() {
803 std::vector<const llvm::Type*> Params;
804 Params.push_back(Int8PtrTy);
Owen Anderson9793f0e2009-07-29 22:16:19 +0000805 return CGM.CreateRuntimeFunction(llvm::FunctionType::get(Int8PtrTy,
Chris Lattnera7c00b42009-04-22 02:15:23 +0000806 Params, false),
807 "objc_begin_catch");
808 }
Daniel Dunbarb1559a42009-03-01 04:46:24 +0000809
810 const llvm::StructType *EHTypeTy;
811 const llvm::Type *EHTypePtrTy;
Daniel Dunbar0b0dcd92009-02-24 07:47:38 +0000812
Fariborz Jahanianb15a3d52009-01-22 23:02:58 +0000813 ObjCNonFragileABITypesHelper(CodeGen::CodeGenModule &cgm);
814 ~ObjCNonFragileABITypesHelper(){}
Fariborz Jahanian279eda62009-01-21 22:04:16 +0000815};
Daniel Dunbar59e476b2009-08-03 17:06:42 +0000816
Fariborz Jahanian279eda62009-01-21 22:04:16 +0000817class CGObjCCommonMac : public CodeGen::CGObjCRuntime {
Fariborz Jahaniancbaf73c2009-03-11 20:59:05 +0000818public:
819 // FIXME - accessibility
Fariborz Jahanian524bb202009-03-10 16:22:08 +0000820 class GC_IVAR {
Fariborz Jahanian3b0f8862009-03-11 00:07:04 +0000821 public:
Daniel Dunbar7b89ace2009-05-03 13:44:42 +0000822 unsigned ivar_bytepos;
823 unsigned ivar_size;
824 GC_IVAR(unsigned bytepos = 0, unsigned size = 0)
Daniel Dunbar59e476b2009-08-03 17:06:42 +0000825 : ivar_bytepos(bytepos), ivar_size(size) {}
Daniel Dunbar22b0ada2009-04-23 01:29:05 +0000826
827 // Allow sorting based on byte pos.
828 bool operator<(const GC_IVAR &b) const {
829 return ivar_bytepos < b.ivar_bytepos;
830 }
Fariborz Jahanian524bb202009-03-10 16:22:08 +0000831 };
Daniel Dunbar59e476b2009-08-03 17:06:42 +0000832
Fariborz Jahaniancbaf73c2009-03-11 20:59:05 +0000833 class SKIP_SCAN {
Daniel Dunbar7b89ace2009-05-03 13:44:42 +0000834 public:
835 unsigned skip;
836 unsigned scan;
Daniel Dunbar59e476b2009-08-03 17:06:42 +0000837 SKIP_SCAN(unsigned _skip = 0, unsigned _scan = 0)
Daniel Dunbar7b89ace2009-05-03 13:44:42 +0000838 : skip(_skip), scan(_scan) {}
Fariborz Jahaniancbaf73c2009-03-11 20:59:05 +0000839 };
Daniel Dunbar59e476b2009-08-03 17:06:42 +0000840
Fariborz Jahanian279eda62009-01-21 22:04:16 +0000841protected:
842 CodeGen::CodeGenModule &CGM;
Owen Andersonae86c192009-07-13 04:10:07 +0000843 llvm::LLVMContext &VMContext;
Fariborz Jahanian279eda62009-01-21 22:04:16 +0000844 // FIXME! May not be needing this after all.
Daniel Dunbar8b8683f2008-08-12 00:12:39 +0000845 unsigned ObjCABI;
Daniel Dunbar59e476b2009-08-03 17:06:42 +0000846
Fariborz Jahaniancbaf73c2009-03-11 20:59:05 +0000847 // gc ivar layout bitmap calculation helper caches.
848 llvm::SmallVector<GC_IVAR, 16> SkipIvars;
849 llvm::SmallVector<GC_IVAR, 16> IvarsInfo;
Daniel Dunbar59e476b2009-08-03 17:06:42 +0000850
Daniel Dunbarc61d0e92008-08-25 06:02:07 +0000851 /// LazySymbols - Symbols to generate a lazy reference for. See
852 /// DefinedSymbols and FinishModule().
Daniel Dunbard027a922009-09-07 00:20:42 +0000853 llvm::SetVector<IdentifierInfo*> LazySymbols;
Daniel Dunbar59e476b2009-08-03 17:06:42 +0000854
Daniel Dunbarc61d0e92008-08-25 06:02:07 +0000855 /// DefinedSymbols - External symbols which are defined by this
856 /// module. The symbols in this list and LazySymbols are used to add
857 /// special linker symbols which ensure that Objective-C modules are
858 /// linked properly.
Daniel Dunbard027a922009-09-07 00:20:42 +0000859 llvm::SetVector<IdentifierInfo*> DefinedSymbols;
Daniel Dunbar59e476b2009-08-03 17:06:42 +0000860
Daniel Dunbar7bd00bd2008-08-12 06:48:42 +0000861 /// ClassNames - uniqued class names.
Daniel Dunbarb036db82008-08-13 03:21:16 +0000862 llvm::DenseMap<IdentifierInfo*, llvm::GlobalVariable*> ClassNames;
Daniel Dunbar59e476b2009-08-03 17:06:42 +0000863
Daniel Dunbarcb515c82008-08-12 03:39:23 +0000864 /// MethodVarNames - uniqued method variable names.
865 llvm::DenseMap<Selector, llvm::GlobalVariable*> MethodVarNames;
Daniel Dunbar59e476b2009-08-03 17:06:42 +0000866
Fariborz Jahanian9adb2e62010-06-21 22:05:18 +0000867 /// DefinedCategoryNames - list of category names in form Class_Category.
868 llvm::SetVector<std::string> DefinedCategoryNames;
869
Daniel Dunbarb036db82008-08-13 03:21:16 +0000870 /// MethodVarTypes - uniqued method type signatures. We have to use
871 /// a StringMap here because have no other unique reference.
872 llvm::StringMap<llvm::GlobalVariable*> MethodVarTypes;
Daniel Dunbar59e476b2009-08-03 17:06:42 +0000873
Daniel Dunbar3c76cb52008-08-26 21:51:14 +0000874 /// MethodDefinitions - map of methods which have been defined in
875 /// this translation unit.
876 llvm::DenseMap<const ObjCMethodDecl*, llvm::Function*> MethodDefinitions;
Daniel Dunbar59e476b2009-08-03 17:06:42 +0000877
Daniel Dunbar80a840b2008-08-23 00:19:03 +0000878 /// PropertyNames - uniqued method variable names.
879 llvm::DenseMap<IdentifierInfo*, llvm::GlobalVariable*> PropertyNames;
Daniel Dunbar59e476b2009-08-03 17:06:42 +0000880
Daniel Dunbar22d82ed2008-08-21 04:36:09 +0000881 /// ClassReferences - uniqued class references.
882 llvm::DenseMap<IdentifierInfo*, llvm::GlobalVariable*> ClassReferences;
Daniel Dunbar59e476b2009-08-03 17:06:42 +0000883
Daniel Dunbarcb515c82008-08-12 03:39:23 +0000884 /// SelectorReferences - uniqued selector references.
885 llvm::DenseMap<Selector, llvm::GlobalVariable*> SelectorReferences;
Daniel Dunbar59e476b2009-08-03 17:06:42 +0000886
Daniel Dunbarb036db82008-08-13 03:21:16 +0000887 /// Protocols - Protocols for which an objc_protocol structure has
888 /// been emitted. Forward declarations are handled by creating an
889 /// empty structure whose initializer is filled in when/if defined.
890 llvm::DenseMap<IdentifierInfo*, llvm::GlobalVariable*> Protocols;
Daniel Dunbar59e476b2009-08-03 17:06:42 +0000891
Daniel Dunbarc475d422008-10-29 22:36:39 +0000892 /// DefinedProtocols - Protocols which have actually been
893 /// defined. We should not need this, see FIXME in GenerateProtocol.
894 llvm::DenseSet<IdentifierInfo*> DefinedProtocols;
Daniel Dunbar59e476b2009-08-03 17:06:42 +0000895
Daniel Dunbar22d82ed2008-08-21 04:36:09 +0000896 /// DefinedClasses - List of defined classes.
897 std::vector<llvm::GlobalValue*> DefinedClasses;
Daniel Dunbar9a017d72009-05-15 22:33:15 +0000898
899 /// DefinedNonLazyClasses - List of defined "non-lazy" classes.
900 std::vector<llvm::GlobalValue*> DefinedNonLazyClasses;
Daniel Dunbar59e476b2009-08-03 17:06:42 +0000901
Daniel Dunbar22d82ed2008-08-21 04:36:09 +0000902 /// DefinedCategories - List of defined categories.
903 std::vector<llvm::GlobalValue*> DefinedCategories;
Daniel Dunbar59e476b2009-08-03 17:06:42 +0000904
Daniel Dunbar9a017d72009-05-15 22:33:15 +0000905 /// DefinedNonLazyCategories - List of defined "non-lazy" categories.
906 std::vector<llvm::GlobalValue*> DefinedNonLazyCategories;
Daniel Dunbar59e476b2009-08-03 17:06:42 +0000907
Fariborz Jahanian0b1ccdc2009-01-21 23:34:32 +0000908 /// GetNameForMethod - Return a name for the given method.
909 /// \param[out] NameOut - The return value.
910 void GetNameForMethod(const ObjCMethodDecl *OMD,
911 const ObjCContainerDecl *CD,
Daniel Dunbard2386812009-10-19 01:21:19 +0000912 llvm::SmallVectorImpl<char> &NameOut);
Daniel Dunbar59e476b2009-08-03 17:06:42 +0000913
Fariborz Jahanian0b1ccdc2009-01-21 23:34:32 +0000914 /// GetMethodVarName - Return a unique constant for the given
915 /// selector's name. The return value has type char *.
916 llvm::Constant *GetMethodVarName(Selector Sel);
917 llvm::Constant *GetMethodVarName(IdentifierInfo *Ident);
918 llvm::Constant *GetMethodVarName(const std::string &Name);
Daniel Dunbar59e476b2009-08-03 17:06:42 +0000919
Fariborz Jahanian0b1ccdc2009-01-21 23:34:32 +0000920 /// GetMethodVarType - Return a unique constant for the given
921 /// selector's name. The return value has type char *.
Daniel Dunbar59e476b2009-08-03 17:06:42 +0000922
Fariborz Jahanian0b1ccdc2009-01-21 23:34:32 +0000923 // FIXME: This is a horrible name.
924 llvm::Constant *GetMethodVarType(const ObjCMethodDecl *D);
Daniel Dunbarf5c18462009-04-20 06:54:31 +0000925 llvm::Constant *GetMethodVarType(const FieldDecl *D);
Daniel Dunbar59e476b2009-08-03 17:06:42 +0000926
Fariborz Jahanian0b1ccdc2009-01-21 23:34:32 +0000927 /// GetPropertyName - Return a unique constant for the given
928 /// name. The return value has type char *.
929 llvm::Constant *GetPropertyName(IdentifierInfo *Ident);
Daniel Dunbar59e476b2009-08-03 17:06:42 +0000930
Fariborz Jahanian0b1ccdc2009-01-21 23:34:32 +0000931 // FIXME: This can be dropped once string functions are unified.
932 llvm::Constant *GetPropertyTypeString(const ObjCPropertyDecl *PD,
933 const Decl *Container);
Daniel Dunbar59e476b2009-08-03 17:06:42 +0000934
Fariborz Jahanian9e3ad522009-01-24 20:21:50 +0000935 /// GetClassName - Return a unique constant for the given selector's
936 /// name. The return value has type char *.
937 llvm::Constant *GetClassName(IdentifierInfo *Ident);
Daniel Dunbar59e476b2009-08-03 17:06:42 +0000938
Fariborz Jahanianc559f3f2009-03-05 22:39:55 +0000939 /// BuildIvarLayout - Builds ivar layout bitmap for the class
940 /// implementation for the __strong or __weak case.
941 ///
Fariborz Jahanian1bf72882009-03-12 22:50:49 +0000942 llvm::Constant *BuildIvarLayout(const ObjCImplementationDecl *OI,
943 bool ForStrongLayout);
Daniel Dunbar59e476b2009-08-03 17:06:42 +0000944
Daniel Dunbar15bd8882009-05-03 14:10:34 +0000945 void BuildAggrIvarRecordLayout(const RecordType *RT,
Daniel Dunbar59e476b2009-08-03 17:06:42 +0000946 unsigned int BytePos, bool ForStrongLayout,
947 bool &HasUnion);
Daniel Dunbar36e2a1e2009-05-03 21:05:10 +0000948 void BuildAggrIvarLayout(const ObjCImplementationDecl *OI,
Fariborz Jahanian1bf72882009-03-12 22:50:49 +0000949 const llvm::StructLayout *Layout,
Fariborz Jahanian524bb202009-03-10 16:22:08 +0000950 const RecordDecl *RD,
Chris Lattner5b36ddb2009-03-31 08:48:01 +0000951 const llvm::SmallVectorImpl<FieldDecl*> &RecFields,
Fariborz Jahanianc559f3f2009-03-05 22:39:55 +0000952 unsigned int BytePos, bool ForStrongLayout,
Fariborz Jahaniana123b642009-04-24 16:17:09 +0000953 bool &HasUnion);
Fariborz Jahanian1bf72882009-03-12 22:50:49 +0000954
Fariborz Jahanian01dff422009-03-05 19:17:31 +0000955 /// GetIvarLayoutName - Returns a unique constant for the given
956 /// ivar layout bitmap.
957 llvm::Constant *GetIvarLayoutName(IdentifierInfo *Ident,
958 const ObjCCommonTypesHelper &ObjCTypes);
Daniel Dunbar59e476b2009-08-03 17:06:42 +0000959
Fariborz Jahanian066347e2009-01-28 22:18:42 +0000960 /// EmitPropertyList - Emit the given property list. The return
961 /// value has type PropertyListPtrTy.
Daniel Dunbar349e6fb2009-10-18 20:48:59 +0000962 llvm::Constant *EmitPropertyList(llvm::Twine Name,
Daniel Dunbar59e476b2009-08-03 17:06:42 +0000963 const Decl *Container,
Fariborz Jahanian066347e2009-01-28 22:18:42 +0000964 const ObjCContainerDecl *OCD,
965 const ObjCCommonTypesHelper &ObjCTypes);
Daniel Dunbar59e476b2009-08-03 17:06:42 +0000966
Fariborz Jahanian751c1e72009-12-12 21:26:21 +0000967 /// PushProtocolProperties - Push protocol's property on the input stack.
968 void PushProtocolProperties(llvm::SmallPtrSet<const IdentifierInfo*, 16> &PropertySet,
969 std::vector<llvm::Constant*> &Properties,
970 const Decl *Container,
971 const ObjCProtocolDecl *PROTO,
972 const ObjCCommonTypesHelper &ObjCTypes);
973
Fariborz Jahanian56b3b772009-01-29 19:24:30 +0000974 /// GetProtocolRef - Return a reference to the internal protocol
975 /// description, creating an empty one if it has not been
976 /// defined. The return value has type ProtocolPtrTy.
977 llvm::Constant *GetProtocolRef(const ObjCProtocolDecl *PD);
Fariborz Jahanian67726212009-03-08 20:18:37 +0000978
Daniel Dunbar30c65362009-03-09 20:09:19 +0000979 /// CreateMetadataVar - Create a global variable with internal
980 /// linkage for use by the Objective-C runtime.
981 ///
982 /// This is a convenience wrapper which not only creates the
983 /// variable, but also sets the section and alignment and adds the
Chris Lattnerf56501c2009-07-17 23:57:13 +0000984 /// global to the "llvm.used" list.
Daniel Dunbar463cc8a2009-03-09 20:50:13 +0000985 ///
986 /// \param Name - The variable name.
987 /// \param Init - The variable initializer; this is also used to
988 /// define the type of the variable.
989 /// \param Section - The section the variable should go into, or 0.
990 /// \param Align - The alignment for the variable, or 0.
991 /// \param AddToUsed - Whether the variable should be added to
Daniel Dunbar4527d302009-04-14 17:42:51 +0000992 /// "llvm.used".
Daniel Dunbar349e6fb2009-10-18 20:48:59 +0000993 llvm::GlobalVariable *CreateMetadataVar(llvm::Twine Name,
Daniel Dunbar30c65362009-03-09 20:09:19 +0000994 llvm::Constant *Init,
995 const char *Section,
Daniel Dunbar463cc8a2009-03-09 20:50:13 +0000996 unsigned Align,
997 bool AddToUsed);
Daniel Dunbar30c65362009-03-09 20:09:19 +0000998
Fariborz Jahanian5d5ed2d2009-05-11 19:25:47 +0000999 CodeGen::RValue EmitLegacyMessageSend(CodeGen::CodeGenFunction &CGF,
John McCall78a15112010-05-22 01:48:05 +00001000 ReturnValueSlot Return,
Fariborz Jahanian5d5ed2d2009-05-11 19:25:47 +00001001 QualType ResultType,
1002 llvm::Value *Sel,
1003 llvm::Value *Arg0,
1004 QualType Arg0Ty,
1005 bool IsSuper,
1006 const CallArgList &CallArgs,
Daniel Dunbaraff9fca2009-09-17 04:01:22 +00001007 const ObjCMethodDecl *OMD,
Fariborz Jahanian5d5ed2d2009-05-11 19:25:47 +00001008 const ObjCCommonTypesHelper &ObjCTypes);
Daniel Dunbarf5c18462009-04-20 06:54:31 +00001009
Daniel Dunbar5e639272010-04-25 20:39:01 +00001010 /// EmitImageInfo - Emit the image info marker used to encode some module
1011 /// level information.
1012 void EmitImageInfo();
1013
Fariborz Jahanian279eda62009-01-21 22:04:16 +00001014public:
Owen Andersonae86c192009-07-13 04:10:07 +00001015 CGObjCCommonMac(CodeGen::CodeGenModule &cgm) :
Mike Stump11289f42009-09-09 15:08:12 +00001016 CGM(cgm), VMContext(cgm.getLLVMContext()) { }
Daniel Dunbar59e476b2009-08-03 17:06:42 +00001017
David Chisnall481e3a82010-01-23 02:40:42 +00001018 virtual llvm::Constant *GenerateConstantString(const StringLiteral *SL);
Daniel Dunbar59e476b2009-08-03 17:06:42 +00001019
Fariborz Jahanian99113fd2009-01-26 21:38:32 +00001020 virtual llvm::Function *GenerateMethod(const ObjCMethodDecl *OMD,
1021 const ObjCContainerDecl *CD=0);
Daniel Dunbar59e476b2009-08-03 17:06:42 +00001022
Fariborz Jahanian56b3b772009-01-29 19:24:30 +00001023 virtual void GenerateProtocol(const ObjCProtocolDecl *PD);
Daniel Dunbar59e476b2009-08-03 17:06:42 +00001024
Fariborz Jahanian56b3b772009-01-29 19:24:30 +00001025 /// GetOrEmitProtocol - Get the protocol object for the given
1026 /// declaration, emitting it if necessary. The return value has type
1027 /// ProtocolPtrTy.
1028 virtual llvm::Constant *GetOrEmitProtocol(const ObjCProtocolDecl *PD)=0;
Daniel Dunbar59e476b2009-08-03 17:06:42 +00001029
Fariborz Jahanian56b3b772009-01-29 19:24:30 +00001030 /// GetOrEmitProtocolRef - Get a forward reference to the protocol
1031 /// object for the given declaration, emitting it if needed. These
1032 /// forward references will be filled in with empty bodies if no
1033 /// definition is seen. The return value has type ProtocolPtrTy.
1034 virtual llvm::Constant *GetOrEmitProtocolRef(const ObjCProtocolDecl *PD)=0;
Fariborz Jahanian279eda62009-01-21 22:04:16 +00001035};
Daniel Dunbar59e476b2009-08-03 17:06:42 +00001036
Fariborz Jahanian279eda62009-01-21 22:04:16 +00001037class CGObjCMac : public CGObjCCommonMac {
1038private:
1039 ObjCTypesHelper ObjCTypes;
Daniel Dunbar3ad53482008-08-11 21:35:06 +00001040
Daniel Dunbar7bd00bd2008-08-12 06:48:42 +00001041 /// EmitModuleInfo - Another marker encoding module level
Daniel Dunbar59e476b2009-08-03 17:06:42 +00001042 /// information.
Daniel Dunbar7bd00bd2008-08-12 06:48:42 +00001043 void EmitModuleInfo();
1044
Daniel Dunbar22d82ed2008-08-21 04:36:09 +00001045 /// EmitModuleSymols - Emit module symbols, the list of defined
1046 /// classes and categories. The result has type SymtabPtrTy.
Daniel Dunbar7bd00bd2008-08-12 06:48:42 +00001047 llvm::Constant *EmitModuleSymbols();
1048
Daniel Dunbar3ad53482008-08-11 21:35:06 +00001049 /// FinishModule - Write out global data structures at the end of
1050 /// processing a translation unit.
1051 void FinishModule();
Daniel Dunbarb036db82008-08-13 03:21:16 +00001052
Daniel Dunbar22d82ed2008-08-21 04:36:09 +00001053 /// EmitClassExtension - Generate the class extension structure used
1054 /// to store the weak ivar layout and properties. The return value
1055 /// has type ClassExtensionPtrTy.
1056 llvm::Constant *EmitClassExtension(const ObjCImplementationDecl *ID);
1057
1058 /// EmitClassRef - Return a Value*, of type ObjCTypes.ClassPtrTy,
1059 /// for the given class.
Daniel Dunbar59e476b2009-08-03 17:06:42 +00001060 llvm::Value *EmitClassRef(CGBuilderTy &Builder,
Daniel Dunbar22d82ed2008-08-21 04:36:09 +00001061 const ObjCInterfaceDecl *ID);
Fariborz Jahanianeb80c982009-11-12 20:14:24 +00001062
1063 /// EmitSuperClassRef - Emits reference to class's main metadata class.
1064 llvm::Value *EmitSuperClassRef(const ObjCInterfaceDecl *ID);
Daniel Dunbar22d82ed2008-08-21 04:36:09 +00001065
Daniel Dunbar97ff50d2008-08-23 09:25:55 +00001066 CodeGen::RValue EmitMessageSend(CodeGen::CodeGenFunction &CGF,
John McCall78a15112010-05-22 01:48:05 +00001067 ReturnValueSlot Return,
Daniel Dunbar4b8c6db2008-08-30 05:35:15 +00001068 QualType ResultType,
1069 Selector Sel,
Daniel Dunbar97ff50d2008-08-23 09:25:55 +00001070 llvm::Value *Arg0,
Daniel Dunbarc722b852008-08-30 03:02:31 +00001071 QualType Arg0Ty,
1072 bool IsSuper,
1073 const CallArgList &CallArgs);
Daniel Dunbar97ff50d2008-08-23 09:25:55 +00001074
Daniel Dunbar22d82ed2008-08-21 04:36:09 +00001075 /// EmitIvarList - Emit the ivar list for the given
1076 /// implementation. If ForClass is true the list of class ivars
1077 /// (i.e. metaclass ivars) is emitted, otherwise the list of
1078 /// interface ivars will be emitted. The return value has type
1079 /// IvarListPtrTy.
1080 llvm::Constant *EmitIvarList(const ObjCImplementationDecl *ID,
Fariborz Jahanianb042a592009-01-28 19:12:34 +00001081 bool ForClass);
Daniel Dunbar59e476b2009-08-03 17:06:42 +00001082
Daniel Dunbarca8531a2008-08-25 08:19:24 +00001083 /// EmitMetaClass - Emit a forward reference to the class structure
1084 /// for the metaclass of the given interface. The return value has
1085 /// type ClassPtrTy.
1086 llvm::Constant *EmitMetaClassRef(const ObjCInterfaceDecl *ID);
1087
Daniel Dunbar22d82ed2008-08-21 04:36:09 +00001088 /// EmitMetaClass - Emit a class structure for the metaclass of the
Daniel Dunbarca8531a2008-08-25 08:19:24 +00001089 /// given implementation. The return value has type ClassPtrTy.
Daniel Dunbar22d82ed2008-08-21 04:36:09 +00001090 llvm::Constant *EmitMetaClass(const ObjCImplementationDecl *ID,
1091 llvm::Constant *Protocols,
Daniel Dunbareb1f9a22008-08-27 02:31:56 +00001092 const ConstantVector &Methods);
Daniel Dunbar59e476b2009-08-03 17:06:42 +00001093
Daniel Dunbareb1f9a22008-08-27 02:31:56 +00001094 llvm::Constant *GetMethodConstant(const ObjCMethodDecl *MD);
Daniel Dunbar59e476b2009-08-03 17:06:42 +00001095
Daniel Dunbareb1f9a22008-08-27 02:31:56 +00001096 llvm::Constant *GetMethodDescriptionConstant(const ObjCMethodDecl *MD);
Daniel Dunbar22d82ed2008-08-21 04:36:09 +00001097
1098 /// EmitMethodList - Emit the method list for the given
Daniel Dunbar89654ee2008-08-26 08:29:31 +00001099 /// implementation. The return value has type MethodListPtrTy.
Daniel Dunbar349e6fb2009-10-18 20:48:59 +00001100 llvm::Constant *EmitMethodList(llvm::Twine Name,
Daniel Dunbar938a77f2008-08-22 20:34:54 +00001101 const char *Section,
Daniel Dunbareb1f9a22008-08-27 02:31:56 +00001102 const ConstantVector &Methods);
Daniel Dunbar22d82ed2008-08-21 04:36:09 +00001103
1104 /// EmitMethodDescList - Emit a method description list for a list of
Daniel Dunbar59e476b2009-08-03 17:06:42 +00001105 /// method declarations.
Daniel Dunbarb036db82008-08-13 03:21:16 +00001106 /// - TypeName: The name for the type containing the methods.
1107 /// - IsProtocol: True iff these methods are for a protocol.
1108 /// - ClassMethds: True iff these are class methods.
1109 /// - Required: When true, only "required" methods are
1110 /// listed. Similarly, when false only "optional" methods are
1111 /// listed. For classes this should always be true.
1112 /// - begin, end: The method list to output.
1113 ///
1114 /// The return value has type MethodDescriptionListPtrTy.
Daniel Dunbar349e6fb2009-10-18 20:48:59 +00001115 llvm::Constant *EmitMethodDescList(llvm::Twine Name,
Daniel Dunbareb1f9a22008-08-27 02:31:56 +00001116 const char *Section,
1117 const ConstantVector &Methods);
Daniel Dunbarb036db82008-08-13 03:21:16 +00001118
Daniel Dunbarc475d422008-10-29 22:36:39 +00001119 /// GetOrEmitProtocol - Get the protocol object for the given
1120 /// declaration, emitting it if necessary. The return value has type
1121 /// ProtocolPtrTy.
Fariborz Jahanian56b3b772009-01-29 19:24:30 +00001122 virtual llvm::Constant *GetOrEmitProtocol(const ObjCProtocolDecl *PD);
Daniel Dunbarc475d422008-10-29 22:36:39 +00001123
1124 /// GetOrEmitProtocolRef - Get a forward reference to the protocol
1125 /// object for the given declaration, emitting it if needed. These
1126 /// forward references will be filled in with empty bodies if no
1127 /// definition is seen. The return value has type ProtocolPtrTy.
Fariborz Jahanian56b3b772009-01-29 19:24:30 +00001128 virtual llvm::Constant *GetOrEmitProtocolRef(const ObjCProtocolDecl *PD);
Daniel Dunbarc475d422008-10-29 22:36:39 +00001129
Daniel Dunbarb036db82008-08-13 03:21:16 +00001130 /// EmitProtocolExtension - Generate the protocol extension
1131 /// structure used to store optional instance and class methods, and
1132 /// protocol properties. The return value has type
1133 /// ProtocolExtensionPtrTy.
Daniel Dunbareb1f9a22008-08-27 02:31:56 +00001134 llvm::Constant *
1135 EmitProtocolExtension(const ObjCProtocolDecl *PD,
1136 const ConstantVector &OptInstanceMethods,
1137 const ConstantVector &OptClassMethods);
Daniel Dunbarb036db82008-08-13 03:21:16 +00001138
1139 /// EmitProtocolList - Generate the list of referenced
1140 /// protocols. The return value has type ProtocolListPtrTy.
Daniel Dunbar349e6fb2009-10-18 20:48:59 +00001141 llvm::Constant *EmitProtocolList(llvm::Twine Name,
Daniel Dunbardec75f82008-08-21 21:57:41 +00001142 ObjCProtocolDecl::protocol_iterator begin,
1143 ObjCProtocolDecl::protocol_iterator end);
Daniel Dunbarb036db82008-08-13 03:21:16 +00001144
Daniel Dunbar22d82ed2008-08-21 04:36:09 +00001145 /// EmitSelector - Return a Value*, of type ObjCTypes.SelectorPtrTy,
1146 /// for the given selector.
Fariborz Jahanian9240f3d2010-06-17 19:56:20 +00001147 llvm::Value *EmitSelector(CGBuilderTy &Builder, Selector Sel,
1148 bool lval=false);
Daniel Dunbar59e476b2009-08-03 17:06:42 +00001149
1150public:
Daniel Dunbar303e2c22008-08-11 02:45:11 +00001151 CGObjCMac(CodeGen::CodeGenModule &cgm);
Daniel Dunbar303e2c22008-08-11 02:45:11 +00001152
Fariborz Jahanian71394042009-01-23 23:53:38 +00001153 virtual llvm::Function *ModuleInitFunction();
Daniel Dunbar59e476b2009-08-03 17:06:42 +00001154
Daniel Dunbar97db84c2008-08-23 03:46:30 +00001155 virtual CodeGen::RValue GenerateMessageSend(CodeGen::CodeGenFunction &CGF,
John McCall78a15112010-05-22 01:48:05 +00001156 ReturnValueSlot Return,
Daniel Dunbar4b8c6db2008-08-30 05:35:15 +00001157 QualType ResultType,
1158 Selector Sel,
Daniel Dunbarca8531a2008-08-25 08:19:24 +00001159 llvm::Value *Receiver,
Fariborz Jahanianf3648b82009-05-05 21:36:57 +00001160 const CallArgList &CallArgs,
David Chisnall01aa4672010-04-28 19:33:36 +00001161 const ObjCInterfaceDecl *Class,
Fariborz Jahanianf3648b82009-05-05 21:36:57 +00001162 const ObjCMethodDecl *Method);
Daniel Dunbar303e2c22008-08-11 02:45:11 +00001163
Daniel Dunbar59e476b2009-08-03 17:06:42 +00001164 virtual CodeGen::RValue
Daniel Dunbar97db84c2008-08-23 03:46:30 +00001165 GenerateMessageSendSuper(CodeGen::CodeGenFunction &CGF,
John McCall78a15112010-05-22 01:48:05 +00001166 ReturnValueSlot Return,
Daniel Dunbar4b8c6db2008-08-30 05:35:15 +00001167 QualType ResultType,
1168 Selector Sel,
Daniel Dunbarca8531a2008-08-25 08:19:24 +00001169 const ObjCInterfaceDecl *Class,
Fariborz Jahanianbac73ac2009-02-28 20:07:56 +00001170 bool isCategoryImpl,
Daniel Dunbarca8531a2008-08-25 08:19:24 +00001171 llvm::Value *Receiver,
Daniel Dunbarc722b852008-08-30 03:02:31 +00001172 bool IsClassMessage,
Daniel Dunbaraff9fca2009-09-17 04:01:22 +00001173 const CallArgList &CallArgs,
1174 const ObjCMethodDecl *Method);
Daniel Dunbar59e476b2009-08-03 17:06:42 +00001175
Daniel Dunbarcb463852008-11-01 01:53:16 +00001176 virtual llvm::Value *GetClass(CGBuilderTy &Builder,
Daniel Dunbar22d82ed2008-08-21 04:36:09 +00001177 const ObjCInterfaceDecl *ID);
Daniel Dunbar303e2c22008-08-11 02:45:11 +00001178
Fariborz Jahanian9240f3d2010-06-17 19:56:20 +00001179 virtual llvm::Value *GetSelector(CGBuilderTy &Builder, Selector Sel,
1180 bool lval = false);
Fariborz Jahanianf3648b82009-05-05 21:36:57 +00001181
1182 /// The NeXT/Apple runtimes do not support typed selectors; just emit an
1183 /// untyped one.
1184 virtual llvm::Value *GetSelector(CGBuilderTy &Builder,
1185 const ObjCMethodDecl *Method);
1186
Daniel Dunbar92992502008-08-15 22:20:32 +00001187 virtual void GenerateCategory(const ObjCCategoryImplDecl *CMD);
Daniel Dunbar303e2c22008-08-11 02:45:11 +00001188
Daniel Dunbar92992502008-08-15 22:20:32 +00001189 virtual void GenerateClass(const ObjCImplementationDecl *ClassDecl);
Daniel Dunbar303e2c22008-08-11 02:45:11 +00001190
Daniel Dunbarcb463852008-11-01 01:53:16 +00001191 virtual llvm::Value *GenerateProtocolRef(CGBuilderTy &Builder,
Daniel Dunbar89da6ad2008-08-13 00:59:25 +00001192 const ObjCProtocolDecl *PD);
Daniel Dunbar59e476b2009-08-03 17:06:42 +00001193
Chris Lattnerd4808922009-03-22 21:03:39 +00001194 virtual llvm::Constant *GetPropertyGetFunction();
1195 virtual llvm::Constant *GetPropertySetFunction();
Fariborz Jahanian5a8c2032010-04-12 18:18:10 +00001196 virtual llvm::Constant *GetCopyStructFunction();
Chris Lattnerd4808922009-03-22 21:03:39 +00001197 virtual llvm::Constant *EnumerationMutationFunction();
Daniel Dunbar59e476b2009-08-03 17:06:42 +00001198
John McCallbd309292010-07-06 01:34:17 +00001199 virtual void EmitTryStmt(CodeGen::CodeGenFunction &CGF,
1200 const ObjCAtTryStmt &S);
1201 virtual void EmitSynchronizedStmt(CodeGen::CodeGenFunction &CGF,
1202 const ObjCAtSynchronizedStmt &S);
1203 void EmitTryOrSynchronizedStmt(CodeGen::CodeGenFunction &CGF, const Stmt &S);
Anders Carlsson1963b0c2008-09-09 10:04:29 +00001204 virtual void EmitThrowStmt(CodeGen::CodeGenFunction &CGF,
1205 const ObjCAtThrowStmt &S);
Fariborz Jahanian83f45b552008-11-18 22:37:34 +00001206 virtual llvm::Value * EmitObjCWeakRead(CodeGen::CodeGenFunction &CGF,
Daniel Dunbar59e476b2009-08-03 17:06:42 +00001207 llvm::Value *AddrWeakObj);
Fariborz Jahanian83f45b552008-11-18 22:37:34 +00001208 virtual void EmitObjCWeakAssign(CodeGen::CodeGenFunction &CGF,
Daniel Dunbar59e476b2009-08-03 17:06:42 +00001209 llvm::Value *src, llvm::Value *dst);
Fariborz Jahaniand7db9642008-11-19 00:59:10 +00001210 virtual void EmitObjCGlobalAssign(CodeGen::CodeGenFunction &CGF,
Fariborz Jahanian217af242010-07-20 20:30:03 +00001211 llvm::Value *src, llvm::Value *dest,
1212 bool threadlocal = false);
Fariborz Jahaniane881b532008-11-20 19:23:36 +00001213 virtual void EmitObjCIvarAssign(CodeGen::CodeGenFunction &CGF,
Fariborz Jahanian7a95d722009-09-24 22:25:38 +00001214 llvm::Value *src, llvm::Value *dest,
1215 llvm::Value *ivarOffset);
Fariborz Jahaniand7db9642008-11-19 00:59:10 +00001216 virtual void EmitObjCStrongCastAssign(CodeGen::CodeGenFunction &CGF,
1217 llvm::Value *src, llvm::Value *dest);
Fariborz Jahanian5f21d2f2009-07-08 01:18:33 +00001218 virtual void EmitGCMemmoveCollectable(CodeGen::CodeGenFunction &CGF,
1219 llvm::Value *dest, llvm::Value *src,
Fariborz Jahanian021510e2010-06-15 22:44:06 +00001220 llvm::Value *size);
Daniel Dunbar59e476b2009-08-03 17:06:42 +00001221
Fariborz Jahanian712bfa62009-02-03 19:03:09 +00001222 virtual LValue EmitObjCValueForIvar(CodeGen::CodeGenFunction &CGF,
1223 QualType ObjectTy,
1224 llvm::Value *BaseValue,
1225 const ObjCIvarDecl *Ivar,
Fariborz Jahanian712bfa62009-02-03 19:03:09 +00001226 unsigned CVRQualifiers);
Fariborz Jahanian21fc74c2009-02-10 19:02:04 +00001227 virtual llvm::Value *EmitIvarOffset(CodeGen::CodeGenFunction &CGF,
Daniel Dunbar722f4242009-04-22 05:08:15 +00001228 const ObjCInterfaceDecl *Interface,
Fariborz Jahanian21fc74c2009-02-10 19:02:04 +00001229 const ObjCIvarDecl *Ivar);
Daniel Dunbar303e2c22008-08-11 02:45:11 +00001230};
Daniel Dunbar59e476b2009-08-03 17:06:42 +00001231
Fariborz Jahanianb15a3d52009-01-22 23:02:58 +00001232class CGObjCNonFragileABIMac : public CGObjCCommonMac {
Fariborz Jahanian279eda62009-01-21 22:04:16 +00001233private:
Fariborz Jahanianb15a3d52009-01-22 23:02:58 +00001234 ObjCNonFragileABITypesHelper ObjCTypes;
Fariborz Jahanian71394042009-01-23 23:53:38 +00001235 llvm::GlobalVariable* ObjCEmptyCacheVar;
1236 llvm::GlobalVariable* ObjCEmptyVtableVar;
Daniel Dunbar59e476b2009-08-03 17:06:42 +00001237
Daniel Dunbar508a7dd2009-04-18 08:51:00 +00001238 /// SuperClassReferences - uniqued super class references.
1239 llvm::DenseMap<IdentifierInfo*, llvm::GlobalVariable*> SuperClassReferences;
Daniel Dunbar59e476b2009-08-03 17:06:42 +00001240
Fariborz Jahanian6b7cd6e2009-02-06 20:09:23 +00001241 /// MetaClassReferences - uniqued meta class references.
1242 llvm::DenseMap<IdentifierInfo*, llvm::GlobalVariable*> MetaClassReferences;
Daniel Dunbarb1559a42009-03-01 04:46:24 +00001243
1244 /// EHTypeReferences - uniqued class ehtype references.
1245 llvm::DenseMap<IdentifierInfo*, llvm::GlobalVariable*> EHTypeReferences;
Daniel Dunbar59e476b2009-08-03 17:06:42 +00001246
Fariborz Jahaniane4128642009-05-12 20:06:41 +00001247 /// NonLegacyDispatchMethods - List of methods for which we do *not* generate
Fariborz Jahanian5d5ed2d2009-05-11 19:25:47 +00001248 /// legacy messaging dispatch.
Fariborz Jahaniane4128642009-05-12 20:06:41 +00001249 llvm::DenseSet<Selector> NonLegacyDispatchMethods;
Daniel Dunbar59e476b2009-08-03 17:06:42 +00001250
Fariborz Jahanian67260552009-11-17 21:37:35 +00001251 /// DefinedMetaClasses - List of defined meta-classes.
1252 std::vector<llvm::GlobalValue*> DefinedMetaClasses;
1253
Fariborz Jahanian5d5ed2d2009-05-11 19:25:47 +00001254 /// LegacyDispatchedSelector - Returns true if SEL is not in the list of
Fariborz Jahaniane4128642009-05-12 20:06:41 +00001255 /// NonLegacyDispatchMethods; false otherwise.
Fariborz Jahanian5d5ed2d2009-05-11 19:25:47 +00001256 bool LegacyDispatchedSelector(Selector Sel);
Daniel Dunbar59e476b2009-08-03 17:06:42 +00001257
Fariborz Jahanian71394042009-01-23 23:53:38 +00001258 /// FinishNonFragileABIModule - Write out global data structures at the end of
1259 /// processing a translation unit.
1260 void FinishNonFragileABIModule();
Daniel Dunbar8f28d012009-04-08 04:21:03 +00001261
Daniel Dunbar19573e72009-05-15 21:48:48 +00001262 /// AddModuleClassList - Add the given list of class pointers to the
1263 /// module with the provided symbol and section names.
1264 void AddModuleClassList(const std::vector<llvm::GlobalValue*> &Container,
1265 const char *SymbolName,
1266 const char *SectionName);
1267
Daniel Dunbar59e476b2009-08-03 17:06:42 +00001268 llvm::GlobalVariable * BuildClassRoTInitializer(unsigned flags,
1269 unsigned InstanceStart,
1270 unsigned InstanceSize,
1271 const ObjCImplementationDecl *ID);
Fariborz Jahanian4723fb72009-01-24 21:21:53 +00001272 llvm::GlobalVariable * BuildClassMetaData(std::string &ClassName,
Daniel Dunbar59e476b2009-08-03 17:06:42 +00001273 llvm::Constant *IsAGV,
Fariborz Jahanian4723fb72009-01-24 21:21:53 +00001274 llvm::Constant *SuperClassGV,
Fariborz Jahanian82208252009-01-31 00:59:10 +00001275 llvm::Constant *ClassRoGV,
1276 bool HiddenVisibility);
Daniel Dunbar59e476b2009-08-03 17:06:42 +00001277
Fariborz Jahanian99113fd2009-01-26 21:38:32 +00001278 llvm::Constant *GetMethodConstant(const ObjCMethodDecl *MD);
Daniel Dunbar59e476b2009-08-03 17:06:42 +00001279
Fariborz Jahaniand9c28b82009-01-30 00:46:37 +00001280 llvm::Constant *GetMethodDescriptionConstant(const ObjCMethodDecl *MD);
Daniel Dunbar59e476b2009-08-03 17:06:42 +00001281
Fariborz Jahanian99113fd2009-01-26 21:38:32 +00001282 /// EmitMethodList - Emit the method list for the given
1283 /// implementation. The return value has type MethodListnfABITy.
Daniel Dunbar349e6fb2009-10-18 20:48:59 +00001284 llvm::Constant *EmitMethodList(llvm::Twine Name,
Fariborz Jahanian99113fd2009-01-26 21:38:32 +00001285 const char *Section,
Fariborz Jahanian7415caa2009-01-27 19:38:51 +00001286 const ConstantVector &Methods);
1287 /// EmitIvarList - Emit the ivar list for the given
1288 /// implementation. If ForClass is true the list of class ivars
1289 /// (i.e. metaclass ivars) is emitted, otherwise the list of
1290 /// interface ivars will be emitted. The return value has type
1291 /// IvarListnfABIPtrTy.
1292 llvm::Constant *EmitIvarList(const ObjCImplementationDecl *ID);
Daniel Dunbar59e476b2009-08-03 17:06:42 +00001293
Fariborz Jahanian4e7ae062009-02-10 20:21:06 +00001294 llvm::Constant *EmitIvarOffsetVar(const ObjCInterfaceDecl *ID,
Fariborz Jahanian3d3426f2009-01-28 01:36:42 +00001295 const ObjCIvarDecl *Ivar,
Fariborz Jahanian40a4bcd2009-01-28 01:05:23 +00001296 unsigned long int offset);
Daniel Dunbar59e476b2009-08-03 17:06:42 +00001297
Fariborz Jahanian56b3b772009-01-29 19:24:30 +00001298 /// GetOrEmitProtocol - Get the protocol object for the given
1299 /// declaration, emitting it if necessary. The return value has type
1300 /// ProtocolPtrTy.
1301 virtual llvm::Constant *GetOrEmitProtocol(const ObjCProtocolDecl *PD);
Daniel Dunbar59e476b2009-08-03 17:06:42 +00001302
Fariborz Jahanian56b3b772009-01-29 19:24:30 +00001303 /// GetOrEmitProtocolRef - Get a forward reference to the protocol
1304 /// object for the given declaration, emitting it if needed. These
1305 /// forward references will be filled in with empty bodies if no
1306 /// definition is seen. The return value has type ProtocolPtrTy.
1307 virtual llvm::Constant *GetOrEmitProtocolRef(const ObjCProtocolDecl *PD);
Daniel Dunbar59e476b2009-08-03 17:06:42 +00001308
Fariborz Jahanian56b3b772009-01-29 19:24:30 +00001309 /// EmitProtocolList - Generate the list of referenced
1310 /// protocols. The return value has type ProtocolListPtrTy.
Daniel Dunbar349e6fb2009-10-18 20:48:59 +00001311 llvm::Constant *EmitProtocolList(llvm::Twine Name,
Fariborz Jahanian56b3b772009-01-29 19:24:30 +00001312 ObjCProtocolDecl::protocol_iterator begin,
Fariborz Jahanian3d9296e2009-02-04 00:22:57 +00001313 ObjCProtocolDecl::protocol_iterator end);
Daniel Dunbar59e476b2009-08-03 17:06:42 +00001314
Fariborz Jahanian3d9296e2009-02-04 00:22:57 +00001315 CodeGen::RValue EmitMessageSend(CodeGen::CodeGenFunction &CGF,
John McCall78a15112010-05-22 01:48:05 +00001316 ReturnValueSlot Return,
Fariborz Jahanian3d9296e2009-02-04 00:22:57 +00001317 QualType ResultType,
1318 Selector Sel,
Fariborz Jahaniane4dc35d2009-02-04 20:42:28 +00001319 llvm::Value *Receiver,
Fariborz Jahanian3d9296e2009-02-04 00:22:57 +00001320 QualType Arg0Ty,
1321 bool IsSuper,
1322 const CallArgList &CallArgs);
Daniel Dunbarc6928bb2009-03-01 04:40:10 +00001323
1324 /// GetClassGlobal - Return the global variable for the Objective-C
1325 /// class of the given name.
Fariborz Jahanian899e7eb2009-04-14 18:41:56 +00001326 llvm::GlobalVariable *GetClassGlobal(const std::string &Name);
Daniel Dunbar59e476b2009-08-03 17:06:42 +00001327
Fariborz Jahanian33f66e62009-02-05 20:41:40 +00001328 /// EmitClassRef - Return a Value*, of type ObjCTypes.ClassPtrTy,
Daniel Dunbar508a7dd2009-04-18 08:51:00 +00001329 /// for the given class reference.
Daniel Dunbar59e476b2009-08-03 17:06:42 +00001330 llvm::Value *EmitClassRef(CGBuilderTy &Builder,
Daniel Dunbar508a7dd2009-04-18 08:51:00 +00001331 const ObjCInterfaceDecl *ID);
Daniel Dunbar59e476b2009-08-03 17:06:42 +00001332
Daniel Dunbar508a7dd2009-04-18 08:51:00 +00001333 /// EmitSuperClassRef - Return a Value*, of type ObjCTypes.ClassPtrTy,
1334 /// for the given super class reference.
Daniel Dunbar59e476b2009-08-03 17:06:42 +00001335 llvm::Value *EmitSuperClassRef(CGBuilderTy &Builder,
1336 const ObjCInterfaceDecl *ID);
1337
Fariborz Jahanian6b7cd6e2009-02-06 20:09:23 +00001338 /// EmitMetaClassRef - Return a Value * of the address of _class_t
1339 /// meta-data
Daniel Dunbar59e476b2009-08-03 17:06:42 +00001340 llvm::Value *EmitMetaClassRef(CGBuilderTy &Builder,
Fariborz Jahanian6b7cd6e2009-02-06 20:09:23 +00001341 const ObjCInterfaceDecl *ID);
1342
Fariborz Jahanian4e7ae062009-02-10 20:21:06 +00001343 /// ObjCIvarOffsetVariable - Returns the ivar offset variable for
1344 /// the given ivar.
1345 ///
Daniel Dunbara1060522009-04-19 00:31:15 +00001346 llvm::GlobalVariable * ObjCIvarOffsetVariable(
Daniel Dunbar59e476b2009-08-03 17:06:42 +00001347 const ObjCInterfaceDecl *ID,
1348 const ObjCIvarDecl *Ivar);
1349
Fariborz Jahanian74b77222009-02-11 20:51:17 +00001350 /// EmitSelector - Return a Value*, of type ObjCTypes.SelectorPtrTy,
1351 /// for the given selector.
Fariborz Jahanian9240f3d2010-06-17 19:56:20 +00001352 llvm::Value *EmitSelector(CGBuilderTy &Builder, Selector Sel,
1353 bool lval=false);
Daniel Dunbarb1559a42009-03-01 04:46:24 +00001354
Daniel Dunbar8f28d012009-04-08 04:21:03 +00001355 /// GetInterfaceEHType - Get the cached ehtype for the given Objective-C
Daniel Dunbarb1559a42009-03-01 04:46:24 +00001356 /// interface. The return value has type EHTypePtrTy.
Daniel Dunbar8f28d012009-04-08 04:21:03 +00001357 llvm::Value *GetInterfaceEHType(const ObjCInterfaceDecl *ID,
1358 bool ForDefinition);
Daniel Dunbar15894b72009-04-07 05:48:37 +00001359
Daniel Dunbar59e476b2009-08-03 17:06:42 +00001360 const char *getMetaclassSymbolPrefix() const {
Daniel Dunbar15894b72009-04-07 05:48:37 +00001361 return "OBJC_METACLASS_$_";
1362 }
Daniel Dunbar59e476b2009-08-03 17:06:42 +00001363
Daniel Dunbar15894b72009-04-07 05:48:37 +00001364 const char *getClassSymbolPrefix() const {
1365 return "OBJC_CLASS_$_";
1366 }
1367
Daniel Dunbar961202372009-05-03 12:57:56 +00001368 void GetClassSizeInfo(const ObjCImplementationDecl *OID,
Daniel Dunbar554fd792009-04-19 23:41:48 +00001369 uint32_t &InstanceStart,
1370 uint32_t &InstanceSize);
Daniel Dunbar59e476b2009-08-03 17:06:42 +00001371
Fariborz Jahaniane4128642009-05-12 20:06:41 +00001372 // Shamelessly stolen from Analysis/CFRefCount.cpp
Daniel Dunbar9a017d72009-05-15 22:33:15 +00001373 Selector GetNullarySelector(const char* name) const {
Fariborz Jahaniane4128642009-05-12 20:06:41 +00001374 IdentifierInfo* II = &CGM.getContext().Idents.get(name);
1375 return CGM.getContext().Selectors.getSelector(0, &II);
1376 }
Daniel Dunbar59e476b2009-08-03 17:06:42 +00001377
Daniel Dunbar9a017d72009-05-15 22:33:15 +00001378 Selector GetUnarySelector(const char* name) const {
Fariborz Jahaniane4128642009-05-12 20:06:41 +00001379 IdentifierInfo* II = &CGM.getContext().Idents.get(name);
1380 return CGM.getContext().Selectors.getSelector(1, &II);
1381 }
Daniel Dunbar554fd792009-04-19 23:41:48 +00001382
Daniel Dunbar9a017d72009-05-15 22:33:15 +00001383 /// ImplementationIsNonLazy - Check whether the given category or
1384 /// class implementation is "non-lazy".
Fariborz Jahaniana6bed832009-05-21 01:03:45 +00001385 bool ImplementationIsNonLazy(const ObjCImplDecl *OD) const;
Daniel Dunbar9a017d72009-05-15 22:33:15 +00001386
Fariborz Jahanian279eda62009-01-21 22:04:16 +00001387public:
Fariborz Jahanianb15a3d52009-01-22 23:02:58 +00001388 CGObjCNonFragileABIMac(CodeGen::CodeGenModule &cgm);
Fariborz Jahanian71394042009-01-23 23:53:38 +00001389 // FIXME. All stubs for now!
1390 virtual llvm::Function *ModuleInitFunction();
Daniel Dunbar59e476b2009-08-03 17:06:42 +00001391
Fariborz Jahanian71394042009-01-23 23:53:38 +00001392 virtual CodeGen::RValue GenerateMessageSend(CodeGen::CodeGenFunction &CGF,
John McCall78a15112010-05-22 01:48:05 +00001393 ReturnValueSlot Return,
Fariborz Jahanian71394042009-01-23 23:53:38 +00001394 QualType ResultType,
1395 Selector Sel,
1396 llvm::Value *Receiver,
Fariborz Jahanianf3648b82009-05-05 21:36:57 +00001397 const CallArgList &CallArgs,
David Chisnall01aa4672010-04-28 19:33:36 +00001398 const ObjCInterfaceDecl *Class,
Fariborz Jahanianf3648b82009-05-05 21:36:57 +00001399 const ObjCMethodDecl *Method);
Daniel Dunbar59e476b2009-08-03 17:06:42 +00001400
1401 virtual CodeGen::RValue
Fariborz Jahanian71394042009-01-23 23:53:38 +00001402 GenerateMessageSendSuper(CodeGen::CodeGenFunction &CGF,
John McCall78a15112010-05-22 01:48:05 +00001403 ReturnValueSlot Return,
Fariborz Jahanian71394042009-01-23 23:53:38 +00001404 QualType ResultType,
1405 Selector Sel,
1406 const ObjCInterfaceDecl *Class,
Fariborz Jahanianbac73ac2009-02-28 20:07:56 +00001407 bool isCategoryImpl,
Fariborz Jahanian71394042009-01-23 23:53:38 +00001408 llvm::Value *Receiver,
1409 bool IsClassMessage,
Daniel Dunbaraff9fca2009-09-17 04:01:22 +00001410 const CallArgList &CallArgs,
1411 const ObjCMethodDecl *Method);
Daniel Dunbar59e476b2009-08-03 17:06:42 +00001412
Fariborz Jahanian71394042009-01-23 23:53:38 +00001413 virtual llvm::Value *GetClass(CGBuilderTy &Builder,
Fariborz Jahanian33f66e62009-02-05 20:41:40 +00001414 const ObjCInterfaceDecl *ID);
Daniel Dunbar59e476b2009-08-03 17:06:42 +00001415
Fariborz Jahanian9240f3d2010-06-17 19:56:20 +00001416 virtual llvm::Value *GetSelector(CGBuilderTy &Builder, Selector Sel,
1417 bool lvalue = false)
1418 { return EmitSelector(Builder, Sel, lvalue); }
Fariborz Jahanianf3648b82009-05-05 21:36:57 +00001419
1420 /// The NeXT/Apple runtimes do not support typed selectors; just emit an
1421 /// untyped one.
1422 virtual llvm::Value *GetSelector(CGBuilderTy &Builder,
1423 const ObjCMethodDecl *Method)
1424 { return EmitSelector(Builder, Method->getSelector()); }
Daniel Dunbar59e476b2009-08-03 17:06:42 +00001425
Fariborz Jahanian0c8d0602009-01-26 18:32:24 +00001426 virtual void GenerateCategory(const ObjCCategoryImplDecl *CMD);
Daniel Dunbar59e476b2009-08-03 17:06:42 +00001427
Fariborz Jahanian71394042009-01-23 23:53:38 +00001428 virtual void GenerateClass(const ObjCImplementationDecl *ClassDecl);
Fariborz Jahanian71394042009-01-23 23:53:38 +00001429 virtual llvm::Value *GenerateProtocolRef(CGBuilderTy &Builder,
Fariborz Jahanian097feda2009-01-30 18:58:59 +00001430 const ObjCProtocolDecl *PD);
Daniel Dunbar59e476b2009-08-03 17:06:42 +00001431
1432 virtual llvm::Constant *GetPropertyGetFunction() {
Chris Lattnerce8754e2009-04-22 02:44:54 +00001433 return ObjCTypes.getGetPropertyFn();
Fariborz Jahanian21fc74c2009-02-10 19:02:04 +00001434 }
Daniel Dunbar59e476b2009-08-03 17:06:42 +00001435 virtual llvm::Constant *GetPropertySetFunction() {
1436 return ObjCTypes.getSetPropertyFn();
Fariborz Jahanian21fc74c2009-02-10 19:02:04 +00001437 }
Fariborz Jahanian5a8c2032010-04-12 18:18:10 +00001438
1439 virtual llvm::Constant *GetCopyStructFunction() {
1440 return ObjCTypes.getCopyStructFn();
1441 }
1442
Chris Lattnerd4808922009-03-22 21:03:39 +00001443 virtual llvm::Constant *EnumerationMutationFunction() {
Chris Lattnerce8754e2009-04-22 02:44:54 +00001444 return ObjCTypes.getEnumerationMutationFn();
Daniel Dunbard73ea8162009-02-16 18:48:45 +00001445 }
Daniel Dunbar59e476b2009-08-03 17:06:42 +00001446
John McCallbd309292010-07-06 01:34:17 +00001447 virtual void EmitTryStmt(CodeGen::CodeGenFunction &CGF,
1448 const ObjCAtTryStmt &S);
1449 virtual void EmitSynchronizedStmt(CodeGen::CodeGenFunction &CGF,
1450 const ObjCAtSynchronizedStmt &S);
Fariborz Jahanian71394042009-01-23 23:53:38 +00001451 virtual void EmitThrowStmt(CodeGen::CodeGenFunction &CGF,
Anders Carlsson9ab53d12009-02-16 22:59:18 +00001452 const ObjCAtThrowStmt &S);
Fariborz Jahanian71394042009-01-23 23:53:38 +00001453 virtual llvm::Value * EmitObjCWeakRead(CodeGen::CodeGenFunction &CGF,
Fariborz Jahanian06292952009-02-16 22:52:32 +00001454 llvm::Value *AddrWeakObj);
Fariborz Jahanian71394042009-01-23 23:53:38 +00001455 virtual void EmitObjCWeakAssign(CodeGen::CodeGenFunction &CGF,
Fariborz Jahanian06292952009-02-16 22:52:32 +00001456 llvm::Value *src, llvm::Value *dst);
Fariborz Jahanian71394042009-01-23 23:53:38 +00001457 virtual void EmitObjCGlobalAssign(CodeGen::CodeGenFunction &CGF,
Fariborz Jahanian217af242010-07-20 20:30:03 +00001458 llvm::Value *src, llvm::Value *dest,
1459 bool threadlocal = false);
Fariborz Jahanian71394042009-01-23 23:53:38 +00001460 virtual void EmitObjCIvarAssign(CodeGen::CodeGenFunction &CGF,
Fariborz Jahanian7a95d722009-09-24 22:25:38 +00001461 llvm::Value *src, llvm::Value *dest,
1462 llvm::Value *ivarOffset);
Fariborz Jahanian71394042009-01-23 23:53:38 +00001463 virtual void EmitObjCStrongCastAssign(CodeGen::CodeGenFunction &CGF,
Fariborz Jahanian06292952009-02-16 22:52:32 +00001464 llvm::Value *src, llvm::Value *dest);
Fariborz Jahanian5f21d2f2009-07-08 01:18:33 +00001465 virtual void EmitGCMemmoveCollectable(CodeGen::CodeGenFunction &CGF,
1466 llvm::Value *dest, llvm::Value *src,
Fariborz Jahanian021510e2010-06-15 22:44:06 +00001467 llvm::Value *size);
Fariborz Jahanian712bfa62009-02-03 19:03:09 +00001468 virtual LValue EmitObjCValueForIvar(CodeGen::CodeGenFunction &CGF,
1469 QualType ObjectTy,
1470 llvm::Value *BaseValue,
1471 const ObjCIvarDecl *Ivar,
Fariborz Jahanian712bfa62009-02-03 19:03:09 +00001472 unsigned CVRQualifiers);
Fariborz Jahanian21fc74c2009-02-10 19:02:04 +00001473 virtual llvm::Value *EmitIvarOffset(CodeGen::CodeGenFunction &CGF,
Daniel Dunbar722f4242009-04-22 05:08:15 +00001474 const ObjCInterfaceDecl *Interface,
Fariborz Jahanian21fc74c2009-02-10 19:02:04 +00001475 const ObjCIvarDecl *Ivar);
Fariborz Jahanian279eda62009-01-21 22:04:16 +00001476};
Daniel Dunbar59e476b2009-08-03 17:06:42 +00001477
Daniel Dunbar303e2c22008-08-11 02:45:11 +00001478} // end anonymous namespace
Daniel Dunbar8b8683f2008-08-12 00:12:39 +00001479
1480/* *** Helper Functions *** */
1481
1482/// getConstantGEP() - Help routine to construct simple GEPs.
Owen Anderson170229f2009-07-14 23:10:40 +00001483static llvm::Constant *getConstantGEP(llvm::LLVMContext &VMContext,
Daniel Dunbar59e476b2009-08-03 17:06:42 +00001484 llvm::Constant *C,
Daniel Dunbar8b8683f2008-08-12 00:12:39 +00001485 unsigned idx0,
1486 unsigned idx1) {
1487 llvm::Value *Idxs[] = {
Owen Anderson41a75022009-08-13 21:57:51 +00001488 llvm::ConstantInt::get(llvm::Type::getInt32Ty(VMContext), idx0),
1489 llvm::ConstantInt::get(llvm::Type::getInt32Ty(VMContext), idx1)
Daniel Dunbar8b8683f2008-08-12 00:12:39 +00001490 };
Owen Andersonade90fd2009-07-29 18:54:39 +00001491 return llvm::ConstantExpr::getGetElementPtr(C, Idxs, 2);
Daniel Dunbar8b8683f2008-08-12 00:12:39 +00001492}
1493
Daniel Dunbar8f28d012009-04-08 04:21:03 +00001494/// hasObjCExceptionAttribute - Return true if this class or any super
1495/// class has the __objc_exception__ attribute.
Daniel Dunbar59e476b2009-08-03 17:06:42 +00001496static bool hasObjCExceptionAttribute(ASTContext &Context,
Douglas Gregor78bd61f2009-06-18 16:11:24 +00001497 const ObjCInterfaceDecl *OID) {
Argyrios Kyrtzidisb4b64ca2009-06-30 02:34:44 +00001498 if (OID->hasAttr<ObjCExceptionAttr>())
Daniel Dunbar8f28d012009-04-08 04:21:03 +00001499 return true;
1500 if (const ObjCInterfaceDecl *Super = OID->getSuperClass())
Douglas Gregor78bd61f2009-06-18 16:11:24 +00001501 return hasObjCExceptionAttribute(Context, Super);
Daniel Dunbar8f28d012009-04-08 04:21:03 +00001502 return false;
1503}
1504
Daniel Dunbar8b8683f2008-08-12 00:12:39 +00001505/* *** CGObjCMac Public Interface *** */
Daniel Dunbar59e476b2009-08-03 17:06:42 +00001506
Fariborz Jahanian279eda62009-01-21 22:04:16 +00001507CGObjCMac::CGObjCMac(CodeGen::CodeGenModule &cgm) : CGObjCCommonMac(cgm),
Mike Stump11289f42009-09-09 15:08:12 +00001508 ObjCTypes(cgm) {
Fariborz Jahanian279eda62009-01-21 22:04:16 +00001509 ObjCABI = 1;
Daniel Dunbar59e476b2009-08-03 17:06:42 +00001510 EmitImageInfo();
Daniel Dunbar303e2c22008-08-11 02:45:11 +00001511}
1512
Daniel Dunbar7c6d3a72008-08-16 00:25:02 +00001513/// GetClass - Return a reference to the class for the given interface
1514/// decl.
Daniel Dunbarcb463852008-11-01 01:53:16 +00001515llvm::Value *CGObjCMac::GetClass(CGBuilderTy &Builder,
Daniel Dunbar22d82ed2008-08-21 04:36:09 +00001516 const ObjCInterfaceDecl *ID) {
1517 return EmitClassRef(Builder, ID);
Daniel Dunbar303e2c22008-08-11 02:45:11 +00001518}
1519
1520/// GetSelector - Return the pointer to the unique'd string for this selector.
Fariborz Jahanian9240f3d2010-06-17 19:56:20 +00001521llvm::Value *CGObjCMac::GetSelector(CGBuilderTy &Builder, Selector Sel,
1522 bool lval) {
1523 return EmitSelector(Builder, Sel, lval);
Daniel Dunbar303e2c22008-08-11 02:45:11 +00001524}
Fariborz Jahanianf3648b82009-05-05 21:36:57 +00001525llvm::Value *CGObjCMac::GetSelector(CGBuilderTy &Builder, const ObjCMethodDecl
Daniel Dunbar59e476b2009-08-03 17:06:42 +00001526 *Method) {
Fariborz Jahanianf3648b82009-05-05 21:36:57 +00001527 return EmitSelector(Builder, Method->getSelector());
1528}
Daniel Dunbar303e2c22008-08-11 02:45:11 +00001529
Daniel Dunbar8b8683f2008-08-12 00:12:39 +00001530/// Generate a constant CFString object.
Daniel Dunbar59e476b2009-08-03 17:06:42 +00001531/*
1532 struct __builtin_CFString {
1533 const int *isa; // point to __CFConstantStringClassReference
1534 int flags;
1535 const char *str;
1536 long length;
1537 };
Daniel Dunbar8b8683f2008-08-12 00:12:39 +00001538*/
1539
Fariborz Jahanian63408e82010-04-22 20:26:39 +00001540/// or Generate a constant NSString object.
1541/*
1542 struct __builtin_NSString {
1543 const int *isa; // point to __NSConstantStringClassReference
1544 const char *str;
1545 unsigned int length;
1546 };
1547*/
1548
Fariborz Jahanian71394042009-01-23 23:53:38 +00001549llvm::Constant *CGObjCCommonMac::GenerateConstantString(
David Chisnall481e3a82010-01-23 02:40:42 +00001550 const StringLiteral *SL) {
Fariborz Jahanian63408e82010-04-22 20:26:39 +00001551 return (CGM.getLangOptions().NoConstantCFStrings == 0 ?
1552 CGM.GetAddrOfConstantCFString(SL) :
1553 CGM.GetAddrOfConstantNSString(SL));
Daniel Dunbar303e2c22008-08-11 02:45:11 +00001554}
1555
1556/// Generates a message send where the super is the receiver. This is
1557/// a message send to self with special delivery semantics indicating
1558/// which class's method should be called.
Daniel Dunbar97db84c2008-08-23 03:46:30 +00001559CodeGen::RValue
1560CGObjCMac::GenerateMessageSendSuper(CodeGen::CodeGenFunction &CGF,
John McCall78a15112010-05-22 01:48:05 +00001561 ReturnValueSlot Return,
Daniel Dunbar4b8c6db2008-08-30 05:35:15 +00001562 QualType ResultType,
1563 Selector Sel,
Daniel Dunbarca8531a2008-08-25 08:19:24 +00001564 const ObjCInterfaceDecl *Class,
Fariborz Jahanianbac73ac2009-02-28 20:07:56 +00001565 bool isCategoryImpl,
Daniel Dunbarca8531a2008-08-25 08:19:24 +00001566 llvm::Value *Receiver,
Daniel Dunbarc722b852008-08-30 03:02:31 +00001567 bool IsClassMessage,
Daniel Dunbaraff9fca2009-09-17 04:01:22 +00001568 const CodeGen::CallArgList &CallArgs,
1569 const ObjCMethodDecl *Method) {
Daniel Dunbarf6397fe2008-08-23 04:28:29 +00001570 // Create and init a super structure; this is a (receiver, class)
1571 // pair we will pass to objc_msgSendSuper.
Daniel Dunbar59e476b2009-08-03 17:06:42 +00001572 llvm::Value *ObjCSuper =
Daniel Dunbarf6397fe2008-08-23 04:28:29 +00001573 CGF.Builder.CreateAlloca(ObjCTypes.SuperTy, 0, "objc_super");
Daniel Dunbar59e476b2009-08-03 17:06:42 +00001574 llvm::Value *ReceiverAsObject =
Daniel Dunbarf6397fe2008-08-23 04:28:29 +00001575 CGF.Builder.CreateBitCast(Receiver, ObjCTypes.ObjectPtrTy);
Daniel Dunbar59e476b2009-08-03 17:06:42 +00001576 CGF.Builder.CreateStore(ReceiverAsObject,
Daniel Dunbarf6397fe2008-08-23 04:28:29 +00001577 CGF.Builder.CreateStructGEP(ObjCSuper, 0));
Daniel Dunbarf6397fe2008-08-23 04:28:29 +00001578
Daniel Dunbarca8531a2008-08-25 08:19:24 +00001579 // If this is a class message the metaclass is passed as the target.
1580 llvm::Value *Target;
1581 if (IsClassMessage) {
Fariborz Jahanianbac73ac2009-02-28 20:07:56 +00001582 if (isCategoryImpl) {
1583 // Message sent to 'super' in a class method defined in a category
1584 // implementation requires an odd treatment.
1585 // If we are in a class method, we must retrieve the
1586 // _metaclass_ for the current class, pointed at by
1587 // the class's "isa" pointer. The following assumes that
1588 // isa" is the first ivar in a class (which it must be).
1589 Target = EmitClassRef(CGF.Builder, Class->getSuperClass());
1590 Target = CGF.Builder.CreateStructGEP(Target, 0);
1591 Target = CGF.Builder.CreateLoad(Target);
Mike Stump658fe022009-07-30 22:28:39 +00001592 } else {
Fariborz Jahanianbac73ac2009-02-28 20:07:56 +00001593 llvm::Value *MetaClassPtr = EmitMetaClassRef(Class);
1594 llvm::Value *SuperPtr = CGF.Builder.CreateStructGEP(MetaClassPtr, 1);
1595 llvm::Value *Super = CGF.Builder.CreateLoad(SuperPtr);
1596 Target = Super;
Daniel Dunbar59e476b2009-08-03 17:06:42 +00001597 }
Fariborz Jahanianda2efb02009-11-14 02:18:31 +00001598 }
1599 else if (isCategoryImpl)
1600 Target = EmitClassRef(CGF.Builder, Class->getSuperClass());
1601 else {
Fariborz Jahanianeb80c982009-11-12 20:14:24 +00001602 llvm::Value *ClassPtr = EmitSuperClassRef(Class);
1603 ClassPtr = CGF.Builder.CreateStructGEP(ClassPtr, 1);
1604 Target = CGF.Builder.CreateLoad(ClassPtr);
Daniel Dunbarca8531a2008-08-25 08:19:24 +00001605 }
Mike Stump18bb9282009-05-16 07:57:57 +00001606 // FIXME: We shouldn't need to do this cast, rectify the ASTContext and
1607 // ObjCTypes types.
Daniel Dunbar59e476b2009-08-03 17:06:42 +00001608 const llvm::Type *ClassTy =
Daniel Dunbarc722b852008-08-30 03:02:31 +00001609 CGM.getTypes().ConvertType(CGF.getContext().getObjCClassType());
Daniel Dunbarc475d422008-10-29 22:36:39 +00001610 Target = CGF.Builder.CreateBitCast(Target, ClassTy);
Daniel Dunbar59e476b2009-08-03 17:06:42 +00001611 CGF.Builder.CreateStore(Target,
Daniel Dunbarca8531a2008-08-25 08:19:24 +00001612 CGF.Builder.CreateStructGEP(ObjCSuper, 1));
John McCall78a15112010-05-22 01:48:05 +00001613 return EmitLegacyMessageSend(CGF, Return, ResultType,
Fariborz Jahanian5d5ed2d2009-05-11 19:25:47 +00001614 EmitSelector(CGF.Builder, Sel),
1615 ObjCSuper, ObjCTypes.SuperPtrCTy,
Daniel Dunbaraff9fca2009-09-17 04:01:22 +00001616 true, CallArgs, Method, ObjCTypes);
Daniel Dunbar303e2c22008-08-11 02:45:11 +00001617}
Daniel Dunbar59e476b2009-08-03 17:06:42 +00001618
1619/// Generate code for a message send expression.
Daniel Dunbar97db84c2008-08-23 03:46:30 +00001620CodeGen::RValue CGObjCMac::GenerateMessageSend(CodeGen::CodeGenFunction &CGF,
John McCall78a15112010-05-22 01:48:05 +00001621 ReturnValueSlot Return,
Daniel Dunbar4b8c6db2008-08-30 05:35:15 +00001622 QualType ResultType,
1623 Selector Sel,
Daniel Dunbarca8531a2008-08-25 08:19:24 +00001624 llvm::Value *Receiver,
Fariborz Jahanianf3648b82009-05-05 21:36:57 +00001625 const CallArgList &CallArgs,
David Chisnall01aa4672010-04-28 19:33:36 +00001626 const ObjCInterfaceDecl *Class,
Fariborz Jahanianf3648b82009-05-05 21:36:57 +00001627 const ObjCMethodDecl *Method) {
John McCall78a15112010-05-22 01:48:05 +00001628 return EmitLegacyMessageSend(CGF, Return, ResultType,
Fariborz Jahanian5d5ed2d2009-05-11 19:25:47 +00001629 EmitSelector(CGF.Builder, Sel),
1630 Receiver, CGF.getContext().getObjCIdType(),
Daniel Dunbaraff9fca2009-09-17 04:01:22 +00001631 false, CallArgs, Method, ObjCTypes);
Daniel Dunbar97ff50d2008-08-23 09:25:55 +00001632}
1633
Daniel Dunbaraff9fca2009-09-17 04:01:22 +00001634CodeGen::RValue
1635CGObjCCommonMac::EmitLegacyMessageSend(CodeGen::CodeGenFunction &CGF,
John McCall78a15112010-05-22 01:48:05 +00001636 ReturnValueSlot Return,
Daniel Dunbaraff9fca2009-09-17 04:01:22 +00001637 QualType ResultType,
1638 llvm::Value *Sel,
1639 llvm::Value *Arg0,
1640 QualType Arg0Ty,
1641 bool IsSuper,
1642 const CallArgList &CallArgs,
1643 const ObjCMethodDecl *Method,
1644 const ObjCCommonTypesHelper &ObjCTypes) {
Daniel Dunbarc722b852008-08-30 03:02:31 +00001645 CallArgList ActualArgs;
Fariborz Jahanian969bc682009-04-24 21:07:43 +00001646 if (!IsSuper)
1647 Arg0 = CGF.Builder.CreateBitCast(Arg0, ObjCTypes.ObjectPtrTy, "tmp");
Daniel Dunbar41cf9de2008-09-09 01:06:48 +00001648 ActualArgs.push_back(std::make_pair(RValue::get(Arg0), Arg0Ty));
Fariborz Jahanian5d5ed2d2009-05-11 19:25:47 +00001649 ActualArgs.push_back(std::make_pair(RValue::get(Sel),
Daniel Dunbarc722b852008-08-30 03:02:31 +00001650 CGF.getContext().getObjCSelType()));
1651 ActualArgs.insert(ActualArgs.end(), CallArgs.begin(), CallArgs.end());
Daniel Dunbar59e476b2009-08-03 17:06:42 +00001652
Daniel Dunbarbf8c24a2009-02-02 23:23:47 +00001653 CodeGenTypes &Types = CGM.getTypes();
John McCallab26cfa2010-02-05 21:31:56 +00001654 const CGFunctionInfo &FnInfo = Types.getFunctionInfo(ResultType, ActualArgs,
Rafael Espindolac50c27c2010-03-30 20:24:48 +00001655 FunctionType::ExtInfo());
Daniel Dunbardf0e62d2009-09-17 04:01:40 +00001656 const llvm::FunctionType *FTy =
1657 Types.GetFunctionType(FnInfo, Method ? Method->isVariadic() : false);
Daniel Dunbar59e476b2009-08-03 17:06:42 +00001658
Anders Carlsson280e61f12010-06-21 20:59:55 +00001659 if (Method)
1660 assert(CGM.getContext().getCanonicalType(Method->getResultType()) ==
1661 CGM.getContext().getCanonicalType(ResultType) &&
1662 "Result type mismatch!");
1663
Fariborz Jahanian5d5ed2d2009-05-11 19:25:47 +00001664 llvm::Constant *Fn = NULL;
Daniel Dunbar6f2e8392010-07-14 23:39:36 +00001665 if (CGM.ReturnTypeUsesSRet(FnInfo)) {
Fariborz Jahanian5d5ed2d2009-05-11 19:25:47 +00001666 Fn = (ObjCABI == 2) ? ObjCTypes.getSendStretFn2(IsSuper)
Daniel Dunbar59e476b2009-08-03 17:06:42 +00001667 : ObjCTypes.getSendStretFn(IsSuper);
Daniel Dunbar6f2e8392010-07-14 23:39:36 +00001668 } else if (CGM.ReturnTypeUsesFPRet(ResultType)) {
1669 Fn = (ObjCABI == 2) ? ObjCTypes.getSendFpretFn2(IsSuper)
1670 : ObjCTypes.getSendFpretFn(IsSuper);
Daniel Dunbar3c683f5b2008-10-17 03:24:53 +00001671 } else {
Fariborz Jahanian5d5ed2d2009-05-11 19:25:47 +00001672 Fn = (ObjCABI == 2) ? ObjCTypes.getSendFn2(IsSuper)
Daniel Dunbar59e476b2009-08-03 17:06:42 +00001673 : ObjCTypes.getSendFn(IsSuper);
Daniel Dunbar3c683f5b2008-10-17 03:24:53 +00001674 }
Daniel Dunbar6f2e8392010-07-14 23:39:36 +00001675 Fn = llvm::ConstantExpr::getBitCast(Fn, llvm::PointerType::getUnqual(FTy));
John McCall78a15112010-05-22 01:48:05 +00001676 return CGF.EmitCall(FnInfo, Fn, Return, ActualArgs);
Daniel Dunbar303e2c22008-08-11 02:45:11 +00001677}
1678
Daniel Dunbar59e476b2009-08-03 17:06:42 +00001679llvm::Value *CGObjCMac::GenerateProtocolRef(CGBuilderTy &Builder,
Daniel Dunbar89da6ad2008-08-13 00:59:25 +00001680 const ObjCProtocolDecl *PD) {
Daniel Dunbar7050c552008-09-04 04:33:15 +00001681 // FIXME: I don't understand why gcc generates this, or where it is
Mike Stump18bb9282009-05-16 07:57:57 +00001682 // resolved. Investigate. Its also wasteful to look this up over and over.
Daniel Dunbar7050c552008-09-04 04:33:15 +00001683 LazySymbols.insert(&CGM.getContext().Idents.get("Protocol"));
1684
Owen Andersonade90fd2009-07-29 18:54:39 +00001685 return llvm::ConstantExpr::getBitCast(GetProtocolRef(PD),
Daniel Dunbarb036db82008-08-13 03:21:16 +00001686 ObjCTypes.ExternalProtocolPtrTy);
Daniel Dunbar303e2c22008-08-11 02:45:11 +00001687}
1688
Fariborz Jahanian56b3b772009-01-29 19:24:30 +00001689void CGObjCCommonMac::GenerateProtocol(const ObjCProtocolDecl *PD) {
Mike Stump18bb9282009-05-16 07:57:57 +00001690 // FIXME: We shouldn't need this, the protocol decl should contain enough
1691 // information to tell us whether this was a declaration or a definition.
Daniel Dunbarc475d422008-10-29 22:36:39 +00001692 DefinedProtocols.insert(PD->getIdentifier());
1693
1694 // If we have generated a forward reference to this protocol, emit
1695 // it now. Otherwise do nothing, the protocol objects are lazily
1696 // emitted.
Daniel Dunbar59e476b2009-08-03 17:06:42 +00001697 if (Protocols.count(PD->getIdentifier()))
Daniel Dunbarc475d422008-10-29 22:36:39 +00001698 GetOrEmitProtocol(PD);
1699}
1700
Fariborz Jahanian56b3b772009-01-29 19:24:30 +00001701llvm::Constant *CGObjCCommonMac::GetProtocolRef(const ObjCProtocolDecl *PD) {
Daniel Dunbarc475d422008-10-29 22:36:39 +00001702 if (DefinedProtocols.count(PD->getIdentifier()))
1703 return GetOrEmitProtocol(PD);
1704 return GetOrEmitProtocolRef(PD);
1705}
1706
Daniel Dunbarb036db82008-08-13 03:21:16 +00001707/*
Daniel Dunbar59e476b2009-08-03 17:06:42 +00001708// APPLE LOCAL radar 4585769 - Objective-C 1.0 extensions
1709struct _objc_protocol {
1710struct _objc_protocol_extension *isa;
1711char *protocol_name;
1712struct _objc_protocol_list *protocol_list;
1713struct _objc__method_prototype_list *instance_methods;
1714struct _objc__method_prototype_list *class_methods
1715};
Daniel Dunbarb036db82008-08-13 03:21:16 +00001716
Daniel Dunbar59e476b2009-08-03 17:06:42 +00001717See EmitProtocolExtension().
Daniel Dunbarb036db82008-08-13 03:21:16 +00001718*/
Daniel Dunbarc475d422008-10-29 22:36:39 +00001719llvm::Constant *CGObjCMac::GetOrEmitProtocol(const ObjCProtocolDecl *PD) {
1720 llvm::GlobalVariable *&Entry = Protocols[PD->getIdentifier()];
1721
1722 // Early exit if a defining object has already been generated.
1723 if (Entry && Entry->hasInitializer())
1724 return Entry;
1725
Daniel Dunbarc61d0e92008-08-25 06:02:07 +00001726 // FIXME: I don't understand why gcc generates this, or where it is
Mike Stump18bb9282009-05-16 07:57:57 +00001727 // resolved. Investigate. Its also wasteful to look this up over and over.
Daniel Dunbarc61d0e92008-08-25 06:02:07 +00001728 LazySymbols.insert(&CGM.getContext().Idents.get("Protocol"));
1729
Daniel Dunbareb1f9a22008-08-27 02:31:56 +00001730 // Construct method lists.
1731 std::vector<llvm::Constant*> InstanceMethods, ClassMethods;
1732 std::vector<llvm::Constant*> OptInstanceMethods, OptClassMethods;
Daniel Dunbar59e476b2009-08-03 17:06:42 +00001733 for (ObjCProtocolDecl::instmeth_iterator
Argyrios Kyrtzidiscfbfe782009-06-30 02:36:12 +00001734 i = PD->instmeth_begin(), e = PD->instmeth_end(); i != e; ++i) {
Daniel Dunbareb1f9a22008-08-27 02:31:56 +00001735 ObjCMethodDecl *MD = *i;
1736 llvm::Constant *C = GetMethodDescriptionConstant(MD);
1737 if (MD->getImplementationControl() == ObjCMethodDecl::Optional) {
1738 OptInstanceMethods.push_back(C);
1739 } else {
1740 InstanceMethods.push_back(C);
Daniel Dunbar59e476b2009-08-03 17:06:42 +00001741 }
Daniel Dunbareb1f9a22008-08-27 02:31:56 +00001742 }
1743
Daniel Dunbar59e476b2009-08-03 17:06:42 +00001744 for (ObjCProtocolDecl::classmeth_iterator
Argyrios Kyrtzidiscfbfe782009-06-30 02:36:12 +00001745 i = PD->classmeth_begin(), e = PD->classmeth_end(); i != e; ++i) {
Daniel Dunbareb1f9a22008-08-27 02:31:56 +00001746 ObjCMethodDecl *MD = *i;
1747 llvm::Constant *C = GetMethodDescriptionConstant(MD);
1748 if (MD->getImplementationControl() == ObjCMethodDecl::Optional) {
1749 OptClassMethods.push_back(C);
1750 } else {
1751 ClassMethods.push_back(C);
Daniel Dunbar59e476b2009-08-03 17:06:42 +00001752 }
Daniel Dunbareb1f9a22008-08-27 02:31:56 +00001753 }
1754
Daniel Dunbarb036db82008-08-13 03:21:16 +00001755 std::vector<llvm::Constant*> Values(5);
Daniel Dunbareb1f9a22008-08-27 02:31:56 +00001756 Values[0] = EmitProtocolExtension(PD, OptInstanceMethods, OptClassMethods);
Daniel Dunbarb036db82008-08-13 03:21:16 +00001757 Values[1] = GetClassName(PD->getIdentifier());
Daniel Dunbar59e476b2009-08-03 17:06:42 +00001758 Values[2] =
Daniel Dunbar349e6fb2009-10-18 20:48:59 +00001759 EmitProtocolList("\01L_OBJC_PROTOCOL_REFS_" + PD->getName(),
Daniel Dunbardec75f82008-08-21 21:57:41 +00001760 PD->protocol_begin(),
1761 PD->protocol_end());
Daniel Dunbar59e476b2009-08-03 17:06:42 +00001762 Values[3] =
Daniel Dunbar349e6fb2009-10-18 20:48:59 +00001763 EmitMethodDescList("\01L_OBJC_PROTOCOL_INSTANCE_METHODS_" + PD->getName(),
Daniel Dunbareb1f9a22008-08-27 02:31:56 +00001764 "__OBJC,__cat_inst_meth,regular,no_dead_strip",
1765 InstanceMethods);
Daniel Dunbar59e476b2009-08-03 17:06:42 +00001766 Values[4] =
Daniel Dunbar349e6fb2009-10-18 20:48:59 +00001767 EmitMethodDescList("\01L_OBJC_PROTOCOL_CLASS_METHODS_" + PD->getName(),
Daniel Dunbareb1f9a22008-08-27 02:31:56 +00001768 "__OBJC,__cat_cls_meth,regular,no_dead_strip",
1769 ClassMethods);
Owen Anderson0e0189d2009-07-27 22:29:56 +00001770 llvm::Constant *Init = llvm::ConstantStruct::get(ObjCTypes.ProtocolTy,
Daniel Dunbarb036db82008-08-13 03:21:16 +00001771 Values);
Daniel Dunbar59e476b2009-08-03 17:06:42 +00001772
Daniel Dunbarb036db82008-08-13 03:21:16 +00001773 if (Entry) {
Daniel Dunbarc475d422008-10-29 22:36:39 +00001774 // Already created, fix the linkage and update the initializer.
1775 Entry->setLinkage(llvm::GlobalValue::InternalLinkage);
Daniel Dunbarb036db82008-08-13 03:21:16 +00001776 Entry->setInitializer(Init);
1777 } else {
Daniel Dunbar59e476b2009-08-03 17:06:42 +00001778 Entry =
Owen Andersonc10c8d32009-07-08 19:05:04 +00001779 new llvm::GlobalVariable(CGM.getModule(), ObjCTypes.ProtocolTy, false,
Daniel Dunbarb036db82008-08-13 03:21:16 +00001780 llvm::GlobalValue::InternalLinkage,
Daniel Dunbar59e476b2009-08-03 17:06:42 +00001781 Init,
Daniel Dunbar349e6fb2009-10-18 20:48:59 +00001782 "\01L_OBJC_PROTOCOL_" + PD->getName());
Daniel Dunbarb036db82008-08-13 03:21:16 +00001783 Entry->setSection("__OBJC,__protocol,regular,no_dead_strip");
Daniel Dunbarb036db82008-08-13 03:21:16 +00001784 // FIXME: Is this necessary? Why only for protocol?
1785 Entry->setAlignment(4);
1786 }
Chris Lattnerf56501c2009-07-17 23:57:13 +00001787 CGM.AddUsedGlobal(Entry);
Daniel Dunbarc475d422008-10-29 22:36:39 +00001788
1789 return Entry;
Daniel Dunbarb036db82008-08-13 03:21:16 +00001790}
1791
Daniel Dunbarc475d422008-10-29 22:36:39 +00001792llvm::Constant *CGObjCMac::GetOrEmitProtocolRef(const ObjCProtocolDecl *PD) {
Daniel Dunbarb036db82008-08-13 03:21:16 +00001793 llvm::GlobalVariable *&Entry = Protocols[PD->getIdentifier()];
1794
1795 if (!Entry) {
Daniel Dunbarc475d422008-10-29 22:36:39 +00001796 // We use the initializer as a marker of whether this is a forward
1797 // reference or not. At module finalization we add the empty
1798 // contents for protocols which were referenced but never defined.
Daniel Dunbar59e476b2009-08-03 17:06:42 +00001799 Entry =
Owen Andersonc10c8d32009-07-08 19:05:04 +00001800 new llvm::GlobalVariable(CGM.getModule(), ObjCTypes.ProtocolTy, false,
Daniel Dunbarc475d422008-10-29 22:36:39 +00001801 llvm::GlobalValue::ExternalLinkage,
1802 0,
Daniel Dunbar349e6fb2009-10-18 20:48:59 +00001803 "\01L_OBJC_PROTOCOL_" + PD->getName());
Daniel Dunbarb036db82008-08-13 03:21:16 +00001804 Entry->setSection("__OBJC,__protocol,regular,no_dead_strip");
Daniel Dunbarb036db82008-08-13 03:21:16 +00001805 // FIXME: Is this necessary? Why only for protocol?
1806 Entry->setAlignment(4);
1807 }
Daniel Dunbar59e476b2009-08-03 17:06:42 +00001808
Daniel Dunbarb036db82008-08-13 03:21:16 +00001809 return Entry;
1810}
1811
1812/*
1813 struct _objc_protocol_extension {
Daniel Dunbar59e476b2009-08-03 17:06:42 +00001814 uint32_t size;
1815 struct objc_method_description_list *optional_instance_methods;
1816 struct objc_method_description_list *optional_class_methods;
1817 struct objc_property_list *instance_properties;
Daniel Dunbarb036db82008-08-13 03:21:16 +00001818 };
1819*/
Daniel Dunbareb1f9a22008-08-27 02:31:56 +00001820llvm::Constant *
1821CGObjCMac::EmitProtocolExtension(const ObjCProtocolDecl *PD,
1822 const ConstantVector &OptInstanceMethods,
1823 const ConstantVector &OptClassMethods) {
Daniel Dunbar59e476b2009-08-03 17:06:42 +00001824 uint64_t Size =
Duncan Sandsc76fe8b2009-05-09 07:08:47 +00001825 CGM.getTargetData().getTypeAllocSize(ObjCTypes.ProtocolExtensionTy);
Daniel Dunbarb036db82008-08-13 03:21:16 +00001826 std::vector<llvm::Constant*> Values(4);
Owen Andersonb7a2fe62009-07-24 23:12:58 +00001827 Values[0] = llvm::ConstantInt::get(ObjCTypes.IntTy, Size);
Daniel Dunbar59e476b2009-08-03 17:06:42 +00001828 Values[1] =
1829 EmitMethodDescList("\01L_OBJC_PROTOCOL_INSTANCE_METHODS_OPT_"
Daniel Dunbar349e6fb2009-10-18 20:48:59 +00001830 + PD->getName(),
Daniel Dunbareb1f9a22008-08-27 02:31:56 +00001831 "__OBJC,__cat_inst_meth,regular,no_dead_strip",
1832 OptInstanceMethods);
Daniel Dunbar59e476b2009-08-03 17:06:42 +00001833 Values[2] =
Daniel Dunbar349e6fb2009-10-18 20:48:59 +00001834 EmitMethodDescList("\01L_OBJC_PROTOCOL_CLASS_METHODS_OPT_" + PD->getName(),
Daniel Dunbareb1f9a22008-08-27 02:31:56 +00001835 "__OBJC,__cat_cls_meth,regular,no_dead_strip",
1836 OptClassMethods);
Daniel Dunbar349e6fb2009-10-18 20:48:59 +00001837 Values[3] = EmitPropertyList("\01L_OBJC_$_PROP_PROTO_LIST_" + PD->getName(),
Fariborz Jahanian066347e2009-01-28 22:18:42 +00001838 0, PD, ObjCTypes);
Daniel Dunbarb036db82008-08-13 03:21:16 +00001839
Daniel Dunbar22d82ed2008-08-21 04:36:09 +00001840 // Return null if no extension bits are used.
Daniel Dunbar59e476b2009-08-03 17:06:42 +00001841 if (Values[1]->isNullValue() && Values[2]->isNullValue() &&
Daniel Dunbarb036db82008-08-13 03:21:16 +00001842 Values[3]->isNullValue())
Owen Anderson0b75f232009-07-31 20:28:54 +00001843 return llvm::Constant::getNullValue(ObjCTypes.ProtocolExtensionPtrTy);
Daniel Dunbarb036db82008-08-13 03:21:16 +00001844
Daniel Dunbar59e476b2009-08-03 17:06:42 +00001845 llvm::Constant *Init =
Owen Anderson0e0189d2009-07-27 22:29:56 +00001846 llvm::ConstantStruct::get(ObjCTypes.ProtocolExtensionTy, Values);
Daniel Dunbarb036db82008-08-13 03:21:16 +00001847
Daniel Dunbarc2d4b622009-03-09 21:49:58 +00001848 // No special section, but goes in llvm.used
Daniel Dunbar349e6fb2009-10-18 20:48:59 +00001849 return CreateMetadataVar("\01L_OBJC_PROTOCOLEXT_" + PD->getName(),
Daniel Dunbar59e476b2009-08-03 17:06:42 +00001850 Init,
Daniel Dunbarc2d4b622009-03-09 21:49:58 +00001851 0, 0, true);
Daniel Dunbarb036db82008-08-13 03:21:16 +00001852}
1853
1854/*
1855 struct objc_protocol_list {
Daniel Dunbar59e476b2009-08-03 17:06:42 +00001856 struct objc_protocol_list *next;
1857 long count;
1858 Protocol *list[];
Daniel Dunbarb036db82008-08-13 03:21:16 +00001859 };
1860*/
Daniel Dunbardec75f82008-08-21 21:57:41 +00001861llvm::Constant *
Daniel Dunbar349e6fb2009-10-18 20:48:59 +00001862CGObjCMac::EmitProtocolList(llvm::Twine Name,
Daniel Dunbardec75f82008-08-21 21:57:41 +00001863 ObjCProtocolDecl::protocol_iterator begin,
1864 ObjCProtocolDecl::protocol_iterator end) {
Daniel Dunbarb036db82008-08-13 03:21:16 +00001865 std::vector<llvm::Constant*> ProtocolRefs;
1866
Daniel Dunbardec75f82008-08-21 21:57:41 +00001867 for (; begin != end; ++begin)
1868 ProtocolRefs.push_back(GetProtocolRef(*begin));
Daniel Dunbarb036db82008-08-13 03:21:16 +00001869
1870 // Just return null for empty protocol lists
Daniel Dunbar59e476b2009-08-03 17:06:42 +00001871 if (ProtocolRefs.empty())
Owen Anderson0b75f232009-07-31 20:28:54 +00001872 return llvm::Constant::getNullValue(ObjCTypes.ProtocolListPtrTy);
Daniel Dunbarb036db82008-08-13 03:21:16 +00001873
Daniel Dunbar22d82ed2008-08-21 04:36:09 +00001874 // This list is null terminated.
Owen Anderson0b75f232009-07-31 20:28:54 +00001875 ProtocolRefs.push_back(llvm::Constant::getNullValue(ObjCTypes.ProtocolPtrTy));
Daniel Dunbarb036db82008-08-13 03:21:16 +00001876
1877 std::vector<llvm::Constant*> Values(3);
Daniel Dunbar22d82ed2008-08-21 04:36:09 +00001878 // This field is only used by the runtime.
Owen Anderson0b75f232009-07-31 20:28:54 +00001879 Values[0] = llvm::Constant::getNullValue(ObjCTypes.ProtocolListPtrTy);
Owen Andersonb7a2fe62009-07-24 23:12:58 +00001880 Values[1] = llvm::ConstantInt::get(ObjCTypes.LongTy,
Daniel Dunbar59e476b2009-08-03 17:06:42 +00001881 ProtocolRefs.size() - 1);
1882 Values[2] =
1883 llvm::ConstantArray::get(llvm::ArrayType::get(ObjCTypes.ProtocolPtrTy,
1884 ProtocolRefs.size()),
Daniel Dunbarb036db82008-08-13 03:21:16 +00001885 ProtocolRefs);
Daniel Dunbar59e476b2009-08-03 17:06:42 +00001886
Nick Lewycky41eaf0a2009-09-19 20:00:52 +00001887 llvm::Constant *Init = llvm::ConstantStruct::get(VMContext, Values, false);
Daniel Dunbar59e476b2009-08-03 17:06:42 +00001888 llvm::GlobalVariable *GV =
Daniel Dunbarc2d4b622009-03-09 21:49:58 +00001889 CreateMetadataVar(Name, Init, "__OBJC,__cat_cls_meth,regular,no_dead_strip",
Daniel Dunbarae333842009-03-09 22:18:41 +00001890 4, false);
Owen Andersonade90fd2009-07-29 18:54:39 +00001891 return llvm::ConstantExpr::getBitCast(GV, ObjCTypes.ProtocolListPtrTy);
Daniel Dunbarb036db82008-08-13 03:21:16 +00001892}
1893
Fariborz Jahanian751c1e72009-12-12 21:26:21 +00001894void CGObjCCommonMac::PushProtocolProperties(llvm::SmallPtrSet<const IdentifierInfo*, 16> &PropertySet,
1895 std::vector<llvm::Constant*> &Properties,
1896 const Decl *Container,
1897 const ObjCProtocolDecl *PROTO,
1898 const ObjCCommonTypesHelper &ObjCTypes) {
1899 std::vector<llvm::Constant*> Prop(2);
1900 for (ObjCProtocolDecl::protocol_iterator P = PROTO->protocol_begin(),
1901 E = PROTO->protocol_end(); P != E; ++P)
1902 PushProtocolProperties(PropertySet, Properties, Container, (*P), ObjCTypes);
1903 for (ObjCContainerDecl::prop_iterator I = PROTO->prop_begin(),
1904 E = PROTO->prop_end(); I != E; ++I) {
1905 const ObjCPropertyDecl *PD = *I;
1906 if (!PropertySet.insert(PD->getIdentifier()))
1907 continue;
1908 Prop[0] = GetPropertyName(PD->getIdentifier());
1909 Prop[1] = GetPropertyTypeString(PD, Container);
1910 Properties.push_back(llvm::ConstantStruct::get(ObjCTypes.PropertyTy, Prop));
1911 }
1912}
1913
Daniel Dunbarb036db82008-08-13 03:21:16 +00001914/*
Daniel Dunbar80a840b2008-08-23 00:19:03 +00001915 struct _objc_property {
Daniel Dunbar59e476b2009-08-03 17:06:42 +00001916 const char * const name;
1917 const char * const attributes;
Daniel Dunbar80a840b2008-08-23 00:19:03 +00001918 };
1919
1920 struct _objc_property_list {
Daniel Dunbar59e476b2009-08-03 17:06:42 +00001921 uint32_t entsize; // sizeof (struct _objc_property)
1922 uint32_t prop_count;
1923 struct _objc_property[prop_count];
Daniel Dunbar80a840b2008-08-23 00:19:03 +00001924 };
1925*/
Daniel Dunbar349e6fb2009-10-18 20:48:59 +00001926llvm::Constant *CGObjCCommonMac::EmitPropertyList(llvm::Twine Name,
Daniel Dunbar59e476b2009-08-03 17:06:42 +00001927 const Decl *Container,
1928 const ObjCContainerDecl *OCD,
1929 const ObjCCommonTypesHelper &ObjCTypes) {
Daniel Dunbar80a840b2008-08-23 00:19:03 +00001930 std::vector<llvm::Constant*> Properties, Prop(2);
Fariborz Jahanian751c1e72009-12-12 21:26:21 +00001931 llvm::SmallPtrSet<const IdentifierInfo*, 16> PropertySet;
Daniel Dunbar59e476b2009-08-03 17:06:42 +00001932 for (ObjCContainerDecl::prop_iterator I = OCD->prop_begin(),
1933 E = OCD->prop_end(); I != E; ++I) {
Steve Naroffba3dc382009-01-11 12:47:58 +00001934 const ObjCPropertyDecl *PD = *I;
Fariborz Jahanian751c1e72009-12-12 21:26:21 +00001935 PropertySet.insert(PD->getIdentifier());
Daniel Dunbar80a840b2008-08-23 00:19:03 +00001936 Prop[0] = GetPropertyName(PD->getIdentifier());
Daniel Dunbar4932b362008-08-28 04:38:10 +00001937 Prop[1] = GetPropertyTypeString(PD, Container);
Owen Anderson0e0189d2009-07-27 22:29:56 +00001938 Properties.push_back(llvm::ConstantStruct::get(ObjCTypes.PropertyTy,
Daniel Dunbar80a840b2008-08-23 00:19:03 +00001939 Prop));
1940 }
Fariborz Jahanian7966aff2010-06-22 16:33:55 +00001941 if (const ObjCInterfaceDecl *OID = dyn_cast<ObjCInterfaceDecl>(OCD)) {
Fariborz Jahanian751c1e72009-12-12 21:26:21 +00001942 for (ObjCInterfaceDecl::protocol_iterator P = OID->protocol_begin(),
1943 E = OID->protocol_end(); P != E; ++P)
Fariborz Jahanian7966aff2010-06-22 16:33:55 +00001944 PushProtocolProperties(PropertySet, Properties, Container, (*P),
1945 ObjCTypes);
1946 }
1947 else if (const ObjCCategoryDecl *CD = dyn_cast<ObjCCategoryDecl>(OCD)) {
1948 for (ObjCCategoryDecl::protocol_iterator P = CD->protocol_begin(),
1949 E = CD->protocol_end(); P != E; ++P)
1950 PushProtocolProperties(PropertySet, Properties, Container, (*P),
1951 ObjCTypes);
1952 }
Daniel Dunbar80a840b2008-08-23 00:19:03 +00001953
1954 // Return null for empty list.
1955 if (Properties.empty())
Owen Anderson0b75f232009-07-31 20:28:54 +00001956 return llvm::Constant::getNullValue(ObjCTypes.PropertyListPtrTy);
Daniel Dunbar80a840b2008-08-23 00:19:03 +00001957
Daniel Dunbar59e476b2009-08-03 17:06:42 +00001958 unsigned PropertySize =
Duncan Sandsc76fe8b2009-05-09 07:08:47 +00001959 CGM.getTargetData().getTypeAllocSize(ObjCTypes.PropertyTy);
Daniel Dunbar80a840b2008-08-23 00:19:03 +00001960 std::vector<llvm::Constant*> Values(3);
Owen Andersonb7a2fe62009-07-24 23:12:58 +00001961 Values[0] = llvm::ConstantInt::get(ObjCTypes.IntTy, PropertySize);
1962 Values[1] = llvm::ConstantInt::get(ObjCTypes.IntTy, Properties.size());
Daniel Dunbar59e476b2009-08-03 17:06:42 +00001963 llvm::ArrayType *AT = llvm::ArrayType::get(ObjCTypes.PropertyTy,
Daniel Dunbar80a840b2008-08-23 00:19:03 +00001964 Properties.size());
Owen Anderson47034e12009-07-28 18:33:04 +00001965 Values[2] = llvm::ConstantArray::get(AT, Properties);
Nick Lewycky41eaf0a2009-09-19 20:00:52 +00001966 llvm::Constant *Init = llvm::ConstantStruct::get(VMContext, Values, false);
Daniel Dunbar80a840b2008-08-23 00:19:03 +00001967
Daniel Dunbar59e476b2009-08-03 17:06:42 +00001968 llvm::GlobalVariable *GV =
1969 CreateMetadataVar(Name, Init,
1970 (ObjCABI == 2) ? "__DATA, __objc_const" :
Daniel Dunbarb25452a2009-04-15 02:56:18 +00001971 "__OBJC,__property,regular,no_dead_strip",
Daniel Dunbar59e476b2009-08-03 17:06:42 +00001972 (ObjCABI == 2) ? 8 : 4,
Daniel Dunbarb25452a2009-04-15 02:56:18 +00001973 true);
Owen Andersonade90fd2009-07-29 18:54:39 +00001974 return llvm::ConstantExpr::getBitCast(GV, ObjCTypes.PropertyListPtrTy);
Daniel Dunbar80a840b2008-08-23 00:19:03 +00001975}
1976
1977/*
Daniel Dunbarb036db82008-08-13 03:21:16 +00001978 struct objc_method_description_list {
Daniel Dunbar59e476b2009-08-03 17:06:42 +00001979 int count;
1980 struct objc_method_description list[];
Daniel Dunbarb036db82008-08-13 03:21:16 +00001981 };
1982*/
Daniel Dunbareb1f9a22008-08-27 02:31:56 +00001983llvm::Constant *
1984CGObjCMac::GetMethodDescriptionConstant(const ObjCMethodDecl *MD) {
1985 std::vector<llvm::Constant*> Desc(2);
Owen Anderson170229f2009-07-14 23:10:40 +00001986 Desc[0] =
Daniel Dunbar59e476b2009-08-03 17:06:42 +00001987 llvm::ConstantExpr::getBitCast(GetMethodVarName(MD->getSelector()),
1988 ObjCTypes.SelectorPtrTy);
Daniel Dunbareb1f9a22008-08-27 02:31:56 +00001989 Desc[1] = GetMethodVarType(MD);
Owen Anderson0e0189d2009-07-27 22:29:56 +00001990 return llvm::ConstantStruct::get(ObjCTypes.MethodDescriptionTy,
Daniel Dunbareb1f9a22008-08-27 02:31:56 +00001991 Desc);
1992}
Daniel Dunbarb036db82008-08-13 03:21:16 +00001993
Daniel Dunbar349e6fb2009-10-18 20:48:59 +00001994llvm::Constant *CGObjCMac::EmitMethodDescList(llvm::Twine Name,
Daniel Dunbareb1f9a22008-08-27 02:31:56 +00001995 const char *Section,
1996 const ConstantVector &Methods) {
Daniel Dunbarb036db82008-08-13 03:21:16 +00001997 // Return null for empty list.
1998 if (Methods.empty())
Owen Anderson0b75f232009-07-31 20:28:54 +00001999 return llvm::Constant::getNullValue(ObjCTypes.MethodDescriptionListPtrTy);
Daniel Dunbarb036db82008-08-13 03:21:16 +00002000
2001 std::vector<llvm::Constant*> Values(2);
Owen Andersonb7a2fe62009-07-24 23:12:58 +00002002 Values[0] = llvm::ConstantInt::get(ObjCTypes.IntTy, Methods.size());
Daniel Dunbar59e476b2009-08-03 17:06:42 +00002003 llvm::ArrayType *AT = llvm::ArrayType::get(ObjCTypes.MethodDescriptionTy,
Daniel Dunbarb036db82008-08-13 03:21:16 +00002004 Methods.size());
Owen Anderson47034e12009-07-28 18:33:04 +00002005 Values[1] = llvm::ConstantArray::get(AT, Methods);
Nick Lewycky41eaf0a2009-09-19 20:00:52 +00002006 llvm::Constant *Init = llvm::ConstantStruct::get(VMContext, Values, false);
Daniel Dunbarb036db82008-08-13 03:21:16 +00002007
Daniel Dunbarb25452a2009-04-15 02:56:18 +00002008 llvm::GlobalVariable *GV = CreateMetadataVar(Name, Init, Section, 4, true);
Daniel Dunbar59e476b2009-08-03 17:06:42 +00002009 return llvm::ConstantExpr::getBitCast(GV,
Daniel Dunbarb036db82008-08-13 03:21:16 +00002010 ObjCTypes.MethodDescriptionListPtrTy);
Daniel Dunbar303e2c22008-08-11 02:45:11 +00002011}
2012
Daniel Dunbar938a77f2008-08-22 20:34:54 +00002013/*
2014 struct _objc_category {
Daniel Dunbar59e476b2009-08-03 17:06:42 +00002015 char *category_name;
2016 char *class_name;
2017 struct _objc_method_list *instance_methods;
2018 struct _objc_method_list *class_methods;
2019 struct _objc_protocol_list *protocols;
2020 uint32_t size; // <rdar://4585769>
2021 struct _objc_property_list *instance_properties;
Daniel Dunbar938a77f2008-08-22 20:34:54 +00002022 };
Daniel Dunbar59e476b2009-08-03 17:06:42 +00002023*/
Daniel Dunbar92992502008-08-15 22:20:32 +00002024void CGObjCMac::GenerateCategory(const ObjCCategoryImplDecl *OCD) {
Duncan Sandsc76fe8b2009-05-09 07:08:47 +00002025 unsigned Size = CGM.getTargetData().getTypeAllocSize(ObjCTypes.CategoryTy);
Daniel Dunbar938a77f2008-08-22 20:34:54 +00002026
Mike Stump18bb9282009-05-16 07:57:57 +00002027 // FIXME: This is poor design, the OCD should have a pointer to the category
2028 // decl. Additionally, note that Category can be null for the @implementation
2029 // w/o an @interface case. Sema should just create one for us as it does for
2030 // @implementation so everyone else can live life under a clear blue sky.
Daniel Dunbar938a77f2008-08-22 20:34:54 +00002031 const ObjCInterfaceDecl *Interface = OCD->getClassInterface();
Daniel Dunbar59e476b2009-08-03 17:06:42 +00002032 const ObjCCategoryDecl *Category =
Daniel Dunbar28e76ca2008-08-26 23:03:11 +00002033 Interface->FindCategoryDeclaration(OCD->getIdentifier());
Daniel Dunbar349e6fb2009-10-18 20:48:59 +00002034
2035 llvm::SmallString<256> ExtName;
2036 llvm::raw_svector_ostream(ExtName) << Interface->getName() << '_'
2037 << OCD->getName();
Daniel Dunbar938a77f2008-08-22 20:34:54 +00002038
Daniel Dunbar3c76cb52008-08-26 21:51:14 +00002039 std::vector<llvm::Constant*> InstanceMethods, ClassMethods;
Daniel Dunbar59e476b2009-08-03 17:06:42 +00002040 for (ObjCCategoryImplDecl::instmeth_iterator
Argyrios Kyrtzidiscfbfe782009-06-30 02:36:12 +00002041 i = OCD->instmeth_begin(), e = OCD->instmeth_end(); i != e; ++i) {
Daniel Dunbar3c76cb52008-08-26 21:51:14 +00002042 // Instance methods should always be defined.
2043 InstanceMethods.push_back(GetMethodConstant(*i));
2044 }
Daniel Dunbar59e476b2009-08-03 17:06:42 +00002045 for (ObjCCategoryImplDecl::classmeth_iterator
Argyrios Kyrtzidiscfbfe782009-06-30 02:36:12 +00002046 i = OCD->classmeth_begin(), e = OCD->classmeth_end(); i != e; ++i) {
Daniel Dunbar3c76cb52008-08-26 21:51:14 +00002047 // Class methods should always be defined.
2048 ClassMethods.push_back(GetMethodConstant(*i));
2049 }
2050
Daniel Dunbar938a77f2008-08-22 20:34:54 +00002051 std::vector<llvm::Constant*> Values(7);
2052 Values[0] = GetClassName(OCD->getIdentifier());
2053 Values[1] = GetClassName(Interface->getIdentifier());
Fariborz Jahaniane55f8662009-04-29 20:40:05 +00002054 LazySymbols.insert(Interface->getIdentifier());
Daniel Dunbar59e476b2009-08-03 17:06:42 +00002055 Values[2] =
Daniel Dunbar349e6fb2009-10-18 20:48:59 +00002056 EmitMethodList("\01L_OBJC_CATEGORY_INSTANCE_METHODS_" + ExtName.str(),
Daniel Dunbar80a840b2008-08-23 00:19:03 +00002057 "__OBJC,__cat_inst_meth,regular,no_dead_strip",
Daniel Dunbar3c76cb52008-08-26 21:51:14 +00002058 InstanceMethods);
Daniel Dunbar59e476b2009-08-03 17:06:42 +00002059 Values[3] =
Daniel Dunbar349e6fb2009-10-18 20:48:59 +00002060 EmitMethodList("\01L_OBJC_CATEGORY_CLASS_METHODS_" + ExtName.str(),
Daniel Dunbarb25452a2009-04-15 02:56:18 +00002061 "__OBJC,__cat_cls_meth,regular,no_dead_strip",
Daniel Dunbar3c76cb52008-08-26 21:51:14 +00002062 ClassMethods);
Daniel Dunbareb1f9a22008-08-27 02:31:56 +00002063 if (Category) {
Daniel Dunbar59e476b2009-08-03 17:06:42 +00002064 Values[4] =
Daniel Dunbar349e6fb2009-10-18 20:48:59 +00002065 EmitProtocolList("\01L_OBJC_CATEGORY_PROTOCOLS_" + ExtName.str(),
Daniel Dunbareb1f9a22008-08-27 02:31:56 +00002066 Category->protocol_begin(),
2067 Category->protocol_end());
2068 } else {
Owen Anderson0b75f232009-07-31 20:28:54 +00002069 Values[4] = llvm::Constant::getNullValue(ObjCTypes.ProtocolListPtrTy);
Daniel Dunbareb1f9a22008-08-27 02:31:56 +00002070 }
Owen Andersonb7a2fe62009-07-24 23:12:58 +00002071 Values[5] = llvm::ConstantInt::get(ObjCTypes.IntTy, Size);
Daniel Dunbar28e76ca2008-08-26 23:03:11 +00002072
2073 // If there is no category @interface then there can be no properties.
2074 if (Category) {
Daniel Dunbar349e6fb2009-10-18 20:48:59 +00002075 Values[6] = EmitPropertyList("\01l_OBJC_$_PROP_LIST_" + ExtName.str(),
Fariborz Jahanian066347e2009-01-28 22:18:42 +00002076 OCD, Category, ObjCTypes);
Daniel Dunbar28e76ca2008-08-26 23:03:11 +00002077 } else {
Owen Anderson0b75f232009-07-31 20:28:54 +00002078 Values[6] = llvm::Constant::getNullValue(ObjCTypes.PropertyListPtrTy);
Daniel Dunbar28e76ca2008-08-26 23:03:11 +00002079 }
Daniel Dunbar59e476b2009-08-03 17:06:42 +00002080
Owen Anderson0e0189d2009-07-27 22:29:56 +00002081 llvm::Constant *Init = llvm::ConstantStruct::get(ObjCTypes.CategoryTy,
Daniel Dunbar938a77f2008-08-22 20:34:54 +00002082 Values);
2083
Daniel Dunbar59e476b2009-08-03 17:06:42 +00002084 llvm::GlobalVariable *GV =
Daniel Dunbar349e6fb2009-10-18 20:48:59 +00002085 CreateMetadataVar("\01L_OBJC_CATEGORY_" + ExtName.str(), Init,
Daniel Dunbarc2d4b622009-03-09 21:49:58 +00002086 "__OBJC,__category,regular,no_dead_strip",
Daniel Dunbarae333842009-03-09 22:18:41 +00002087 4, true);
Daniel Dunbar938a77f2008-08-22 20:34:54 +00002088 DefinedCategories.push_back(GV);
Fariborz Jahanian9adb2e62010-06-21 22:05:18 +00002089 DefinedCategoryNames.insert(ExtName.str());
Daniel Dunbar303e2c22008-08-11 02:45:11 +00002090}
2091
Daniel Dunbar22d82ed2008-08-21 04:36:09 +00002092// FIXME: Get from somewhere?
2093enum ClassFlags {
2094 eClassFlags_Factory = 0x00001,
2095 eClassFlags_Meta = 0x00002,
2096 // <rdr://5142207>
2097 eClassFlags_HasCXXStructors = 0x02000,
2098 eClassFlags_Hidden = 0x20000,
2099 eClassFlags_ABI2_Hidden = 0x00010,
2100 eClassFlags_ABI2_HasCXXStructors = 0x00004 // <rdr://4923634>
2101};
2102
Daniel Dunbar22d82ed2008-08-21 04:36:09 +00002103/*
2104 struct _objc_class {
Daniel Dunbar59e476b2009-08-03 17:06:42 +00002105 Class isa;
2106 Class super_class;
2107 const char *name;
2108 long version;
2109 long info;
2110 long instance_size;
2111 struct _objc_ivar_list *ivars;
2112 struct _objc_method_list *methods;
2113 struct _objc_cache *cache;
2114 struct _objc_protocol_list *protocols;
2115 // Objective-C 1.0 extensions (<rdr://4585769>)
2116 const char *ivar_layout;
2117 struct _objc_class_ext *ext;
Daniel Dunbar22d82ed2008-08-21 04:36:09 +00002118 };
2119
2120 See EmitClassExtension();
Daniel Dunbar59e476b2009-08-03 17:06:42 +00002121*/
Daniel Dunbar22d82ed2008-08-21 04:36:09 +00002122void CGObjCMac::GenerateClass(const ObjCImplementationDecl *ID) {
Daniel Dunbarc61d0e92008-08-25 06:02:07 +00002123 DefinedSymbols.insert(ID->getIdentifier());
2124
Chris Lattner86d7d912008-11-24 03:54:41 +00002125 std::string ClassName = ID->getNameAsString();
Daniel Dunbar22d82ed2008-08-21 04:36:09 +00002126 // FIXME: Gross
Daniel Dunbar59e476b2009-08-03 17:06:42 +00002127 ObjCInterfaceDecl *Interface =
Daniel Dunbar22d82ed2008-08-21 04:36:09 +00002128 const_cast<ObjCInterfaceDecl*>(ID->getClassInterface());
Daniel Dunbar59e476b2009-08-03 17:06:42 +00002129 llvm::Constant *Protocols =
Daniel Dunbar349e6fb2009-10-18 20:48:59 +00002130 EmitProtocolList("\01L_OBJC_CLASS_PROTOCOLS_" + ID->getName(),
Daniel Dunbardec75f82008-08-21 21:57:41 +00002131 Interface->protocol_begin(),
2132 Interface->protocol_end());
Daniel Dunbar22d82ed2008-08-21 04:36:09 +00002133 unsigned Flags = eClassFlags_Factory;
Fariborz Jahanian0dec1e02010-04-28 21:28:56 +00002134 if (ID->getNumIvarInitializers())
2135 Flags |= eClassFlags_HasCXXStructors;
Daniel Dunbar59e476b2009-08-03 17:06:42 +00002136 unsigned Size =
Daniel Dunbar12119b92009-05-03 10:46:44 +00002137 CGM.getContext().getASTObjCImplementationLayout(ID).getSize() / 8;
Daniel Dunbar22d82ed2008-08-21 04:36:09 +00002138
2139 // FIXME: Set CXX-structors flag.
Daniel Dunbarf5f359f2009-04-14 06:00:08 +00002140 if (CGM.getDeclVisibilityMode(ID->getClassInterface()) == LangOptions::Hidden)
Daniel Dunbar22d82ed2008-08-21 04:36:09 +00002141 Flags |= eClassFlags_Hidden;
2142
Daniel Dunbar3c76cb52008-08-26 21:51:14 +00002143 std::vector<llvm::Constant*> InstanceMethods, ClassMethods;
Daniel Dunbar59e476b2009-08-03 17:06:42 +00002144 for (ObjCImplementationDecl::instmeth_iterator
Argyrios Kyrtzidiscfbfe782009-06-30 02:36:12 +00002145 i = ID->instmeth_begin(), e = ID->instmeth_end(); i != e; ++i) {
Daniel Dunbar3c76cb52008-08-26 21:51:14 +00002146 // Instance methods should always be defined.
2147 InstanceMethods.push_back(GetMethodConstant(*i));
2148 }
Daniel Dunbar59e476b2009-08-03 17:06:42 +00002149 for (ObjCImplementationDecl::classmeth_iterator
Argyrios Kyrtzidiscfbfe782009-06-30 02:36:12 +00002150 i = ID->classmeth_begin(), e = ID->classmeth_end(); i != e; ++i) {
Daniel Dunbar3c76cb52008-08-26 21:51:14 +00002151 // Class methods should always be defined.
2152 ClassMethods.push_back(GetMethodConstant(*i));
2153 }
2154
Daniel Dunbar59e476b2009-08-03 17:06:42 +00002155 for (ObjCImplementationDecl::propimpl_iterator
Argyrios Kyrtzidiscfbfe782009-06-30 02:36:12 +00002156 i = ID->propimpl_begin(), e = ID->propimpl_end(); i != e; ++i) {
Daniel Dunbar3c76cb52008-08-26 21:51:14 +00002157 ObjCPropertyImplDecl *PID = *i;
2158
2159 if (PID->getPropertyImplementation() == ObjCPropertyImplDecl::Synthesize) {
2160 ObjCPropertyDecl *PD = PID->getPropertyDecl();
2161
2162 if (ObjCMethodDecl *MD = PD->getGetterMethodDecl())
2163 if (llvm::Constant *C = GetMethodConstant(MD))
2164 InstanceMethods.push_back(C);
2165 if (ObjCMethodDecl *MD = PD->getSetterMethodDecl())
2166 if (llvm::Constant *C = GetMethodConstant(MD))
2167 InstanceMethods.push_back(C);
2168 }
2169 }
2170
Daniel Dunbar22d82ed2008-08-21 04:36:09 +00002171 std::vector<llvm::Constant*> Values(12);
Daniel Dunbarccf61832009-05-03 08:56:52 +00002172 Values[ 0] = EmitMetaClass(ID, Protocols, ClassMethods);
Daniel Dunbar22d82ed2008-08-21 04:36:09 +00002173 if (ObjCInterfaceDecl *Super = Interface->getSuperClass()) {
Daniel Dunbarc61d0e92008-08-25 06:02:07 +00002174 // Record a reference to the super class.
2175 LazySymbols.insert(Super->getIdentifier());
2176
Daniel Dunbar59e476b2009-08-03 17:06:42 +00002177 Values[ 1] =
Owen Andersonade90fd2009-07-29 18:54:39 +00002178 llvm::ConstantExpr::getBitCast(GetClassName(Super->getIdentifier()),
Daniel Dunbar22d82ed2008-08-21 04:36:09 +00002179 ObjCTypes.ClassPtrTy);
2180 } else {
Owen Anderson0b75f232009-07-31 20:28:54 +00002181 Values[ 1] = llvm::Constant::getNullValue(ObjCTypes.ClassPtrTy);
Daniel Dunbar22d82ed2008-08-21 04:36:09 +00002182 }
2183 Values[ 2] = GetClassName(ID->getIdentifier());
2184 // Version is always 0.
Owen Andersonb7a2fe62009-07-24 23:12:58 +00002185 Values[ 3] = llvm::ConstantInt::get(ObjCTypes.LongTy, 0);
2186 Values[ 4] = llvm::ConstantInt::get(ObjCTypes.LongTy, Flags);
2187 Values[ 5] = llvm::ConstantInt::get(ObjCTypes.LongTy, Size);
Fariborz Jahanianb042a592009-01-28 19:12:34 +00002188 Values[ 6] = EmitIvarList(ID, false);
Daniel Dunbar59e476b2009-08-03 17:06:42 +00002189 Values[ 7] =
Daniel Dunbar349e6fb2009-10-18 20:48:59 +00002190 EmitMethodList("\01L_OBJC_INSTANCE_METHODS_" + ID->getName(),
Daniel Dunbar80a840b2008-08-23 00:19:03 +00002191 "__OBJC,__inst_meth,regular,no_dead_strip",
Daniel Dunbar3c76cb52008-08-26 21:51:14 +00002192 InstanceMethods);
Daniel Dunbar22d82ed2008-08-21 04:36:09 +00002193 // cache is always NULL.
Owen Anderson0b75f232009-07-31 20:28:54 +00002194 Values[ 8] = llvm::Constant::getNullValue(ObjCTypes.CachePtrTy);
Daniel Dunbar22d82ed2008-08-21 04:36:09 +00002195 Values[ 9] = Protocols;
Daniel Dunbar59e476b2009-08-03 17:06:42 +00002196 Values[10] = BuildIvarLayout(ID, true);
Daniel Dunbar22d82ed2008-08-21 04:36:09 +00002197 Values[11] = EmitClassExtension(ID);
Owen Anderson0e0189d2009-07-27 22:29:56 +00002198 llvm::Constant *Init = llvm::ConstantStruct::get(ObjCTypes.ClassTy,
Daniel Dunbar22d82ed2008-08-21 04:36:09 +00002199 Values);
Fariborz Jahanianeb80c982009-11-12 20:14:24 +00002200 std::string Name("\01L_OBJC_CLASS_");
2201 Name += ClassName;
2202 const char *Section = "__OBJC,__class,regular,no_dead_strip";
2203 // Check for a forward reference.
2204 llvm::GlobalVariable *GV = CGM.getModule().getGlobalVariable(Name);
2205 if (GV) {
2206 assert(GV->getType()->getElementType() == ObjCTypes.ClassTy &&
2207 "Forward metaclass reference has incorrect type.");
2208 GV->setLinkage(llvm::GlobalValue::InternalLinkage);
2209 GV->setInitializer(Init);
2210 GV->setSection(Section);
2211 GV->setAlignment(4);
2212 CGM.AddUsedGlobal(GV);
2213 }
2214 else
2215 GV = CreateMetadataVar(Name, Init, Section, 4, true);
Daniel Dunbar22d82ed2008-08-21 04:36:09 +00002216 DefinedClasses.push_back(GV);
2217}
2218
2219llvm::Constant *CGObjCMac::EmitMetaClass(const ObjCImplementationDecl *ID,
2220 llvm::Constant *Protocols,
Daniel Dunbareb1f9a22008-08-27 02:31:56 +00002221 const ConstantVector &Methods) {
Daniel Dunbar22d82ed2008-08-21 04:36:09 +00002222 unsigned Flags = eClassFlags_Meta;
Duncan Sandsc76fe8b2009-05-09 07:08:47 +00002223 unsigned Size = CGM.getTargetData().getTypeAllocSize(ObjCTypes.ClassTy);
Daniel Dunbar22d82ed2008-08-21 04:36:09 +00002224
Daniel Dunbarf5f359f2009-04-14 06:00:08 +00002225 if (CGM.getDeclVisibilityMode(ID->getClassInterface()) == LangOptions::Hidden)
Daniel Dunbar22d82ed2008-08-21 04:36:09 +00002226 Flags |= eClassFlags_Hidden;
Daniel Dunbar59e476b2009-08-03 17:06:42 +00002227
Daniel Dunbar22d82ed2008-08-21 04:36:09 +00002228 std::vector<llvm::Constant*> Values(12);
2229 // The isa for the metaclass is the root of the hierarchy.
2230 const ObjCInterfaceDecl *Root = ID->getClassInterface();
2231 while (const ObjCInterfaceDecl *Super = Root->getSuperClass())
2232 Root = Super;
Daniel Dunbar59e476b2009-08-03 17:06:42 +00002233 Values[ 0] =
Owen Andersonade90fd2009-07-29 18:54:39 +00002234 llvm::ConstantExpr::getBitCast(GetClassName(Root->getIdentifier()),
Daniel Dunbar22d82ed2008-08-21 04:36:09 +00002235 ObjCTypes.ClassPtrTy);
Daniel Dunbar938a77f2008-08-22 20:34:54 +00002236 // The super class for the metaclass is emitted as the name of the
2237 // super class. The runtime fixes this up to point to the
2238 // *metaclass* for the super class.
Daniel Dunbar22d82ed2008-08-21 04:36:09 +00002239 if (ObjCInterfaceDecl *Super = ID->getClassInterface()->getSuperClass()) {
Daniel Dunbar59e476b2009-08-03 17:06:42 +00002240 Values[ 1] =
Owen Andersonade90fd2009-07-29 18:54:39 +00002241 llvm::ConstantExpr::getBitCast(GetClassName(Super->getIdentifier()),
Daniel Dunbar22d82ed2008-08-21 04:36:09 +00002242 ObjCTypes.ClassPtrTy);
2243 } else {
Owen Anderson0b75f232009-07-31 20:28:54 +00002244 Values[ 1] = llvm::Constant::getNullValue(ObjCTypes.ClassPtrTy);
Daniel Dunbar22d82ed2008-08-21 04:36:09 +00002245 }
2246 Values[ 2] = GetClassName(ID->getIdentifier());
2247 // Version is always 0.
Owen Andersonb7a2fe62009-07-24 23:12:58 +00002248 Values[ 3] = llvm::ConstantInt::get(ObjCTypes.LongTy, 0);
2249 Values[ 4] = llvm::ConstantInt::get(ObjCTypes.LongTy, Flags);
2250 Values[ 5] = llvm::ConstantInt::get(ObjCTypes.LongTy, Size);
Fariborz Jahanianb042a592009-01-28 19:12:34 +00002251 Values[ 6] = EmitIvarList(ID, true);
Daniel Dunbar59e476b2009-08-03 17:06:42 +00002252 Values[ 7] =
Chris Lattnerf3d3fae2008-11-24 05:29:24 +00002253 EmitMethodList("\01L_OBJC_CLASS_METHODS_" + ID->getNameAsString(),
Daniel Dunbarb25452a2009-04-15 02:56:18 +00002254 "__OBJC,__cls_meth,regular,no_dead_strip",
Daniel Dunbar3c76cb52008-08-26 21:51:14 +00002255 Methods);
Daniel Dunbar22d82ed2008-08-21 04:36:09 +00002256 // cache is always NULL.
Owen Anderson0b75f232009-07-31 20:28:54 +00002257 Values[ 8] = llvm::Constant::getNullValue(ObjCTypes.CachePtrTy);
Daniel Dunbar22d82ed2008-08-21 04:36:09 +00002258 Values[ 9] = Protocols;
2259 // ivar_layout for metaclass is always NULL.
Owen Anderson0b75f232009-07-31 20:28:54 +00002260 Values[10] = llvm::Constant::getNullValue(ObjCTypes.Int8PtrTy);
Daniel Dunbar22d82ed2008-08-21 04:36:09 +00002261 // The class extension is always unused for metaclasses.
Owen Anderson0b75f232009-07-31 20:28:54 +00002262 Values[11] = llvm::Constant::getNullValue(ObjCTypes.ClassExtensionPtrTy);
Owen Anderson0e0189d2009-07-27 22:29:56 +00002263 llvm::Constant *Init = llvm::ConstantStruct::get(ObjCTypes.ClassTy,
Daniel Dunbar22d82ed2008-08-21 04:36:09 +00002264 Values);
2265
Daniel Dunbarca8531a2008-08-25 08:19:24 +00002266 std::string Name("\01L_OBJC_METACLASS_");
Chris Lattner86d7d912008-11-24 03:54:41 +00002267 Name += ID->getNameAsCString();
Daniel Dunbarca8531a2008-08-25 08:19:24 +00002268
2269 // Check for a forward reference.
2270 llvm::GlobalVariable *GV = CGM.getModule().getGlobalVariable(Name);
2271 if (GV) {
2272 assert(GV->getType()->getElementType() == ObjCTypes.ClassTy &&
2273 "Forward metaclass reference has incorrect type.");
2274 GV->setLinkage(llvm::GlobalValue::InternalLinkage);
2275 GV->setInitializer(Init);
2276 } else {
Owen Andersonc10c8d32009-07-08 19:05:04 +00002277 GV = new llvm::GlobalVariable(CGM.getModule(), ObjCTypes.ClassTy, false,
Daniel Dunbarca8531a2008-08-25 08:19:24 +00002278 llvm::GlobalValue::InternalLinkage,
Owen Andersonc10c8d32009-07-08 19:05:04 +00002279 Init, Name);
Daniel Dunbarca8531a2008-08-25 08:19:24 +00002280 }
Daniel Dunbar22d82ed2008-08-21 04:36:09 +00002281 GV->setSection("__OBJC,__meta_class,regular,no_dead_strip");
Daniel Dunbarae333842009-03-09 22:18:41 +00002282 GV->setAlignment(4);
Chris Lattnerf56501c2009-07-17 23:57:13 +00002283 CGM.AddUsedGlobal(GV);
Daniel Dunbar22d82ed2008-08-21 04:36:09 +00002284
2285 return GV;
2286}
2287
Daniel Dunbar59e476b2009-08-03 17:06:42 +00002288llvm::Constant *CGObjCMac::EmitMetaClassRef(const ObjCInterfaceDecl *ID) {
Chris Lattnerf3d3fae2008-11-24 05:29:24 +00002289 std::string Name = "\01L_OBJC_METACLASS_" + ID->getNameAsString();
Daniel Dunbarca8531a2008-08-25 08:19:24 +00002290
Mike Stump18bb9282009-05-16 07:57:57 +00002291 // FIXME: Should we look these up somewhere other than the module. Its a bit
2292 // silly since we only generate these while processing an implementation, so
2293 // exactly one pointer would work if know when we entered/exitted an
2294 // implementation block.
Daniel Dunbarca8531a2008-08-25 08:19:24 +00002295
2296 // Check for an existing forward reference.
Fariborz Jahanian475831b2009-01-07 20:11:22 +00002297 // Previously, metaclass with internal linkage may have been defined.
2298 // pass 'true' as 2nd argument so it is returned.
Daniel Dunbar59e476b2009-08-03 17:06:42 +00002299 if (llvm::GlobalVariable *GV = CGM.getModule().getGlobalVariable(Name,
2300 true)) {
Daniel Dunbarca8531a2008-08-25 08:19:24 +00002301 assert(GV->getType()->getElementType() == ObjCTypes.ClassTy &&
2302 "Forward metaclass reference has incorrect type.");
2303 return GV;
2304 } else {
2305 // Generate as an external reference to keep a consistent
2306 // module. This will be patched up when we emit the metaclass.
Owen Andersonc10c8d32009-07-08 19:05:04 +00002307 return new llvm::GlobalVariable(CGM.getModule(), ObjCTypes.ClassTy, false,
Daniel Dunbarca8531a2008-08-25 08:19:24 +00002308 llvm::GlobalValue::ExternalLinkage,
2309 0,
Owen Andersonc10c8d32009-07-08 19:05:04 +00002310 Name);
Daniel Dunbarca8531a2008-08-25 08:19:24 +00002311 }
2312}
2313
Fariborz Jahanianeb80c982009-11-12 20:14:24 +00002314llvm::Value *CGObjCMac::EmitSuperClassRef(const ObjCInterfaceDecl *ID) {
2315 std::string Name = "\01L_OBJC_CLASS_" + ID->getNameAsString();
2316
2317 if (llvm::GlobalVariable *GV = CGM.getModule().getGlobalVariable(Name,
2318 true)) {
2319 assert(GV->getType()->getElementType() == ObjCTypes.ClassTy &&
2320 "Forward class metadata reference has incorrect type.");
2321 return GV;
2322 } else {
2323 return new llvm::GlobalVariable(CGM.getModule(), ObjCTypes.ClassTy, false,
2324 llvm::GlobalValue::ExternalLinkage,
2325 0,
2326 Name);
2327 }
2328}
2329
Daniel Dunbar22d82ed2008-08-21 04:36:09 +00002330/*
2331 struct objc_class_ext {
Daniel Dunbar59e476b2009-08-03 17:06:42 +00002332 uint32_t size;
2333 const char *weak_ivar_layout;
2334 struct _objc_property_list *properties;
Daniel Dunbar22d82ed2008-08-21 04:36:09 +00002335 };
2336*/
2337llvm::Constant *
2338CGObjCMac::EmitClassExtension(const ObjCImplementationDecl *ID) {
Daniel Dunbar59e476b2009-08-03 17:06:42 +00002339 uint64_t Size =
Duncan Sandsc76fe8b2009-05-09 07:08:47 +00002340 CGM.getTargetData().getTypeAllocSize(ObjCTypes.ClassExtensionTy);
Daniel Dunbar22d82ed2008-08-21 04:36:09 +00002341
2342 std::vector<llvm::Constant*> Values(3);
Owen Andersonb7a2fe62009-07-24 23:12:58 +00002343 Values[0] = llvm::ConstantInt::get(ObjCTypes.IntTy, Size);
Fariborz Jahanian6df69862009-04-22 23:00:43 +00002344 Values[1] = BuildIvarLayout(ID, false);
Daniel Dunbar349e6fb2009-10-18 20:48:59 +00002345 Values[2] = EmitPropertyList("\01l_OBJC_$_PROP_LIST_" + ID->getName(),
Fariborz Jahanian066347e2009-01-28 22:18:42 +00002346 ID, ID->getClassInterface(), ObjCTypes);
Daniel Dunbar22d82ed2008-08-21 04:36:09 +00002347
2348 // Return null if no extension bits are used.
2349 if (Values[1]->isNullValue() && Values[2]->isNullValue())
Owen Anderson0b75f232009-07-31 20:28:54 +00002350 return llvm::Constant::getNullValue(ObjCTypes.ClassExtensionPtrTy);
Daniel Dunbar22d82ed2008-08-21 04:36:09 +00002351
Daniel Dunbar59e476b2009-08-03 17:06:42 +00002352 llvm::Constant *Init =
Owen Anderson0e0189d2009-07-27 22:29:56 +00002353 llvm::ConstantStruct::get(ObjCTypes.ClassExtensionTy, Values);
Daniel Dunbar349e6fb2009-10-18 20:48:59 +00002354 return CreateMetadataVar("\01L_OBJC_CLASSEXT_" + ID->getName(),
Daniel Dunbar59e476b2009-08-03 17:06:42 +00002355 Init, "__OBJC,__class_ext,regular,no_dead_strip",
Daniel Dunbarb25452a2009-04-15 02:56:18 +00002356 4, true);
Daniel Dunbar22d82ed2008-08-21 04:36:09 +00002357}
2358
2359/*
2360 struct objc_ivar {
Daniel Dunbar59e476b2009-08-03 17:06:42 +00002361 char *ivar_name;
2362 char *ivar_type;
2363 int ivar_offset;
Daniel Dunbar22d82ed2008-08-21 04:36:09 +00002364 };
2365
2366 struct objc_ivar_list {
Daniel Dunbar59e476b2009-08-03 17:06:42 +00002367 int ivar_count;
2368 struct objc_ivar list[count];
Daniel Dunbar22d82ed2008-08-21 04:36:09 +00002369 };
Daniel Dunbar59e476b2009-08-03 17:06:42 +00002370*/
Daniel Dunbar22d82ed2008-08-21 04:36:09 +00002371llvm::Constant *CGObjCMac::EmitIvarList(const ObjCImplementationDecl *ID,
Fariborz Jahanianb042a592009-01-28 19:12:34 +00002372 bool ForClass) {
Daniel Dunbar22d82ed2008-08-21 04:36:09 +00002373 std::vector<llvm::Constant*> Ivars, Ivar(3);
2374
2375 // When emitting the root class GCC emits ivar entries for the
2376 // actual class structure. It is not clear if we need to follow this
2377 // behavior; for now lets try and get away with not doing it. If so,
2378 // the cleanest solution would be to make up an ObjCInterfaceDecl
2379 // for the class.
2380 if (ForClass)
Owen Anderson0b75f232009-07-31 20:28:54 +00002381 return llvm::Constant::getNullValue(ObjCTypes.IvarListPtrTy);
Daniel Dunbar59e476b2009-08-03 17:06:42 +00002382
2383 ObjCInterfaceDecl *OID =
Fariborz Jahanianb042a592009-01-28 19:12:34 +00002384 const_cast<ObjCInterfaceDecl*>(ID->getClassInterface());
Daniel Dunbar59e476b2009-08-03 17:06:42 +00002385
Daniel Dunbarf5c18462009-04-20 06:54:31 +00002386 llvm::SmallVector<ObjCIvarDecl*, 16> OIvars;
Fariborz Jahanian7c809592009-06-04 01:19:09 +00002387 CGM.getContext().ShallowCollectObjCIvars(OID, OIvars);
Daniel Dunbar59e476b2009-08-03 17:06:42 +00002388
Daniel Dunbarf5c18462009-04-20 06:54:31 +00002389 for (unsigned i = 0, e = OIvars.size(); i != e; ++i) {
2390 ObjCIvarDecl *IVD = OIvars[i];
Fariborz Jahanian7c809592009-06-04 01:19:09 +00002391 // Ignore unnamed bit-fields.
2392 if (!IVD->getDeclName())
Daniel Dunbar59e476b2009-08-03 17:06:42 +00002393 continue;
Daniel Dunbar725dc2c2009-04-22 08:22:17 +00002394 Ivar[0] = GetMethodVarName(IVD->getIdentifier());
2395 Ivar[1] = GetMethodVarType(IVD);
Daniel Dunbar59e476b2009-08-03 17:06:42 +00002396 Ivar[2] = llvm::ConstantInt::get(ObjCTypes.IntTy,
Daniel Dunbar9fd114d2009-04-22 07:32:20 +00002397 ComputeIvarBaseOffset(CGM, OID, IVD));
Owen Anderson0e0189d2009-07-27 22:29:56 +00002398 Ivars.push_back(llvm::ConstantStruct::get(ObjCTypes.IvarTy, Ivar));
Daniel Dunbar22d82ed2008-08-21 04:36:09 +00002399 }
2400
2401 // Return null for empty list.
2402 if (Ivars.empty())
Owen Anderson0b75f232009-07-31 20:28:54 +00002403 return llvm::Constant::getNullValue(ObjCTypes.IvarListPtrTy);
Daniel Dunbar22d82ed2008-08-21 04:36:09 +00002404
2405 std::vector<llvm::Constant*> Values(2);
Owen Andersonb7a2fe62009-07-24 23:12:58 +00002406 Values[0] = llvm::ConstantInt::get(ObjCTypes.IntTy, Ivars.size());
Owen Anderson9793f0e2009-07-29 22:16:19 +00002407 llvm::ArrayType *AT = llvm::ArrayType::get(ObjCTypes.IvarTy,
Daniel Dunbar22d82ed2008-08-21 04:36:09 +00002408 Ivars.size());
Owen Anderson47034e12009-07-28 18:33:04 +00002409 Values[1] = llvm::ConstantArray::get(AT, Ivars);
Nick Lewycky41eaf0a2009-09-19 20:00:52 +00002410 llvm::Constant *Init = llvm::ConstantStruct::get(VMContext, Values, false);
Daniel Dunbar22d82ed2008-08-21 04:36:09 +00002411
Daniel Dunbarc2d4b622009-03-09 21:49:58 +00002412 llvm::GlobalVariable *GV;
2413 if (ForClass)
Daniel Dunbar349e6fb2009-10-18 20:48:59 +00002414 GV = CreateMetadataVar("\01L_OBJC_CLASS_VARIABLES_" + ID->getName(),
Daniel Dunbar59e476b2009-08-03 17:06:42 +00002415 Init, "__OBJC,__class_vars,regular,no_dead_strip",
Daniel Dunbarae333842009-03-09 22:18:41 +00002416 4, true);
Daniel Dunbarc2d4b622009-03-09 21:49:58 +00002417 else
Daniel Dunbar349e6fb2009-10-18 20:48:59 +00002418 GV = CreateMetadataVar("\01L_OBJC_INSTANCE_VARIABLES_" + ID->getName(),
Daniel Dunbarc2d4b622009-03-09 21:49:58 +00002419 Init, "__OBJC,__instance_vars,regular,no_dead_strip",
Daniel Dunbarb25452a2009-04-15 02:56:18 +00002420 4, true);
Owen Andersonade90fd2009-07-29 18:54:39 +00002421 return llvm::ConstantExpr::getBitCast(GV, ObjCTypes.IvarListPtrTy);
Daniel Dunbar22d82ed2008-08-21 04:36:09 +00002422}
2423
2424/*
2425 struct objc_method {
Daniel Dunbar59e476b2009-08-03 17:06:42 +00002426 SEL method_name;
2427 char *method_types;
2428 void *method;
Daniel Dunbar22d82ed2008-08-21 04:36:09 +00002429 };
Daniel Dunbar59e476b2009-08-03 17:06:42 +00002430
Daniel Dunbar22d82ed2008-08-21 04:36:09 +00002431 struct objc_method_list {
Daniel Dunbar59e476b2009-08-03 17:06:42 +00002432 struct objc_method_list *obsolete;
2433 int count;
2434 struct objc_method methods_list[count];
Daniel Dunbar22d82ed2008-08-21 04:36:09 +00002435 };
2436*/
Daniel Dunbar3c76cb52008-08-26 21:51:14 +00002437
2438/// GetMethodConstant - Return a struct objc_method constant for the
2439/// given method if it has been defined. The result is null if the
2440/// method has not been defined. The return value has type MethodPtrTy.
Daniel Dunbareb1f9a22008-08-27 02:31:56 +00002441llvm::Constant *CGObjCMac::GetMethodConstant(const ObjCMethodDecl *MD) {
Daniel Dunbar3c76cb52008-08-26 21:51:14 +00002442 // FIXME: Use DenseMap::lookup
2443 llvm::Function *Fn = MethodDefinitions[MD];
2444 if (!Fn)
2445 return 0;
Daniel Dunbar59e476b2009-08-03 17:06:42 +00002446
Daniel Dunbar3c76cb52008-08-26 21:51:14 +00002447 std::vector<llvm::Constant*> Method(3);
Daniel Dunbar59e476b2009-08-03 17:06:42 +00002448 Method[0] =
Owen Andersonade90fd2009-07-29 18:54:39 +00002449 llvm::ConstantExpr::getBitCast(GetMethodVarName(MD->getSelector()),
Daniel Dunbar3c76cb52008-08-26 21:51:14 +00002450 ObjCTypes.SelectorPtrTy);
2451 Method[1] = GetMethodVarType(MD);
Owen Andersonade90fd2009-07-29 18:54:39 +00002452 Method[2] = llvm::ConstantExpr::getBitCast(Fn, ObjCTypes.Int8PtrTy);
Owen Anderson0e0189d2009-07-27 22:29:56 +00002453 return llvm::ConstantStruct::get(ObjCTypes.MethodTy, Method);
Daniel Dunbar3c76cb52008-08-26 21:51:14 +00002454}
2455
Daniel Dunbar349e6fb2009-10-18 20:48:59 +00002456llvm::Constant *CGObjCMac::EmitMethodList(llvm::Twine Name,
Daniel Dunbar938a77f2008-08-22 20:34:54 +00002457 const char *Section,
Daniel Dunbareb1f9a22008-08-27 02:31:56 +00002458 const ConstantVector &Methods) {
Daniel Dunbar22d82ed2008-08-21 04:36:09 +00002459 // Return null for empty list.
2460 if (Methods.empty())
Owen Anderson0b75f232009-07-31 20:28:54 +00002461 return llvm::Constant::getNullValue(ObjCTypes.MethodListPtrTy);
Daniel Dunbar22d82ed2008-08-21 04:36:09 +00002462
2463 std::vector<llvm::Constant*> Values(3);
Owen Anderson0b75f232009-07-31 20:28:54 +00002464 Values[0] = llvm::Constant::getNullValue(ObjCTypes.Int8PtrTy);
Owen Andersonb7a2fe62009-07-24 23:12:58 +00002465 Values[1] = llvm::ConstantInt::get(ObjCTypes.IntTy, Methods.size());
Owen Anderson9793f0e2009-07-29 22:16:19 +00002466 llvm::ArrayType *AT = llvm::ArrayType::get(ObjCTypes.MethodTy,
Daniel Dunbar22d82ed2008-08-21 04:36:09 +00002467 Methods.size());
Owen Anderson47034e12009-07-28 18:33:04 +00002468 Values[2] = llvm::ConstantArray::get(AT, Methods);
Nick Lewycky41eaf0a2009-09-19 20:00:52 +00002469 llvm::Constant *Init = llvm::ConstantStruct::get(VMContext, Values, false);
Daniel Dunbar22d82ed2008-08-21 04:36:09 +00002470
Daniel Dunbarb25452a2009-04-15 02:56:18 +00002471 llvm::GlobalVariable *GV = CreateMetadataVar(Name, Init, Section, 4, true);
Owen Andersonade90fd2009-07-29 18:54:39 +00002472 return llvm::ConstantExpr::getBitCast(GV,
Daniel Dunbar22d82ed2008-08-21 04:36:09 +00002473 ObjCTypes.MethodListPtrTy);
Daniel Dunbara94ecd22008-08-16 03:19:19 +00002474}
2475
Fariborz Jahanian99113fd2009-01-26 21:38:32 +00002476llvm::Function *CGObjCCommonMac::GenerateMethod(const ObjCMethodDecl *OMD,
Daniel Dunbar59e476b2009-08-03 17:06:42 +00002477 const ObjCContainerDecl *CD) {
Daniel Dunbard2386812009-10-19 01:21:19 +00002478 llvm::SmallString<256> Name;
Fariborz Jahanian0196a1c2009-01-10 21:06:09 +00002479 GetNameForMethod(OMD, CD, Name);
Daniel Dunbara94ecd22008-08-16 03:19:19 +00002480
Daniel Dunbarbf8c24a2009-02-02 23:23:47 +00002481 CodeGenTypes &Types = CGM.getTypes();
Daniel Dunbar7a95ca32008-09-10 04:01:49 +00002482 const llvm::FunctionType *MethodTy =
Daniel Dunbarbf8c24a2009-02-02 23:23:47 +00002483 Types.GetFunctionType(Types.getFunctionInfo(OMD), OMD->isVariadic());
Daniel Dunbar59e476b2009-08-03 17:06:42 +00002484 llvm::Function *Method =
Daniel Dunbar7a95ca32008-09-10 04:01:49 +00002485 llvm::Function::Create(MethodTy,
Daniel Dunbara94ecd22008-08-16 03:19:19 +00002486 llvm::GlobalValue::InternalLinkage,
Daniel Dunbard2386812009-10-19 01:21:19 +00002487 Name.str(),
Daniel Dunbara94ecd22008-08-16 03:19:19 +00002488 &CGM.getModule());
Daniel Dunbar3c76cb52008-08-26 21:51:14 +00002489 MethodDefinitions.insert(std::make_pair(OMD, Method));
Daniel Dunbara94ecd22008-08-16 03:19:19 +00002490
Daniel Dunbara94ecd22008-08-16 03:19:19 +00002491 return Method;
Daniel Dunbar303e2c22008-08-11 02:45:11 +00002492}
2493
Daniel Dunbar30c65362009-03-09 20:09:19 +00002494llvm::GlobalVariable *
Daniel Dunbar349e6fb2009-10-18 20:48:59 +00002495CGObjCCommonMac::CreateMetadataVar(llvm::Twine Name,
Daniel Dunbar30c65362009-03-09 20:09:19 +00002496 llvm::Constant *Init,
2497 const char *Section,
Daniel Dunbar463cc8a2009-03-09 20:50:13 +00002498 unsigned Align,
2499 bool AddToUsed) {
Daniel Dunbar30c65362009-03-09 20:09:19 +00002500 const llvm::Type *Ty = Init->getType();
Daniel Dunbar59e476b2009-08-03 17:06:42 +00002501 llvm::GlobalVariable *GV =
Owen Andersonc10c8d32009-07-08 19:05:04 +00002502 new llvm::GlobalVariable(CGM.getModule(), Ty, false,
Chris Lattnerf56501c2009-07-17 23:57:13 +00002503 llvm::GlobalValue::InternalLinkage, Init, Name);
Daniel Dunbar30c65362009-03-09 20:09:19 +00002504 if (Section)
2505 GV->setSection(Section);
Daniel Dunbar463cc8a2009-03-09 20:50:13 +00002506 if (Align)
2507 GV->setAlignment(Align);
2508 if (AddToUsed)
Chris Lattnerf56501c2009-07-17 23:57:13 +00002509 CGM.AddUsedGlobal(GV);
Daniel Dunbar30c65362009-03-09 20:09:19 +00002510 return GV;
2511}
2512
Daniel Dunbar59e476b2009-08-03 17:06:42 +00002513llvm::Function *CGObjCMac::ModuleInitFunction() {
Daniel Dunbar3ad53482008-08-11 21:35:06 +00002514 // Abuse this interface function as a place to finalize.
2515 FinishModule();
Daniel Dunbar303e2c22008-08-11 02:45:11 +00002516 return NULL;
2517}
2518
Chris Lattnerd4808922009-03-22 21:03:39 +00002519llvm::Constant *CGObjCMac::GetPropertyGetFunction() {
Chris Lattnerce8754e2009-04-22 02:44:54 +00002520 return ObjCTypes.getGetPropertyFn();
Daniel Dunbara91c3e02008-09-24 03:38:44 +00002521}
2522
Chris Lattnerd4808922009-03-22 21:03:39 +00002523llvm::Constant *CGObjCMac::GetPropertySetFunction() {
Chris Lattnerce8754e2009-04-22 02:44:54 +00002524 return ObjCTypes.getSetPropertyFn();
Daniel Dunbara91c3e02008-09-24 03:38:44 +00002525}
2526
Fariborz Jahanian5a8c2032010-04-12 18:18:10 +00002527llvm::Constant *CGObjCMac::GetCopyStructFunction() {
2528 return ObjCTypes.getCopyStructFn();
2529}
2530
Chris Lattnerd4808922009-03-22 21:03:39 +00002531llvm::Constant *CGObjCMac::EnumerationMutationFunction() {
Chris Lattnerce8754e2009-04-22 02:44:54 +00002532 return ObjCTypes.getEnumerationMutationFn();
Anders Carlsson3f35a262008-08-31 04:05:03 +00002533}
2534
John McCallbd309292010-07-06 01:34:17 +00002535void CGObjCMac::EmitTryStmt(CodeGenFunction &CGF, const ObjCAtTryStmt &S) {
2536 return EmitTryOrSynchronizedStmt(CGF, S);
2537}
2538
2539void CGObjCMac::EmitSynchronizedStmt(CodeGenFunction &CGF,
2540 const ObjCAtSynchronizedStmt &S) {
2541 return EmitTryOrSynchronizedStmt(CGF, S);
2542}
2543
Daniel Dunbar59e476b2009-08-03 17:06:42 +00002544/*
Daniel Dunbard3dcb4f82008-09-28 01:03:14 +00002545
Daniel Dunbar59e476b2009-08-03 17:06:42 +00002546 Objective-C setjmp-longjmp (sjlj) Exception Handling
2547 --
Daniel Dunbard3dcb4f82008-09-28 01:03:14 +00002548
John McCallbd309292010-07-06 01:34:17 +00002549 A catch buffer is a setjmp buffer plus:
2550 - a pointer to the exception that was caught
2551 - a pointer to the previous exception data buffer
2552 - two pointers of reserved storage
2553 Therefore catch buffers form a stack, with a pointer to the top
2554 of the stack kept in thread-local storage.
2555
2556 objc_exception_try_enter pushes a catch buffer onto the EH stack.
2557 objc_exception_try_exit pops the given catch buffer, which is
2558 required to be the top of the EH stack.
2559 objc_exception_throw pops the top of the EH stack, writes the
2560 thrown exception into the appropriate field, and longjmps
2561 to the setjmp buffer. It crashes the process (with a printf
2562 and an abort()) if there are no catch buffers on the stack.
2563 objc_exception_extract just reads the exception pointer out of the
2564 catch buffer.
2565
2566 There's no reason an implementation couldn't use a light-weight
2567 setjmp here --- something like __builtin_setjmp, but API-compatible
2568 with the heavyweight setjmp. This will be more important if we ever
2569 want to implement correct ObjC/C++ exception interactions for the
2570 fragile ABI.
2571
2572 Note that for this use of setjmp/longjmp to be correct, we may need
2573 to mark some local variables volatile: if a non-volatile local
2574 variable is modified between the setjmp and the longjmp, it has
2575 indeterminate value. For the purposes of LLVM IR, it may be
2576 sufficient to make loads and stores within the @try (to variables
2577 declared outside the @try) volatile. This is necessary for
2578 optimized correctness, but is not currently being done; this is
2579 being tracked as rdar://problem/8160285
2580
Daniel Dunbar59e476b2009-08-03 17:06:42 +00002581 The basic framework for a @try-catch-finally is as follows:
2582 {
Daniel Dunbard3dcb4f82008-09-28 01:03:14 +00002583 objc_exception_data d;
2584 id _rethrow = null;
Anders Carlssonda0e4562009-02-07 21:26:04 +00002585 bool _call_try_exit = true;
Daniel Dunbar59e476b2009-08-03 17:06:42 +00002586
Daniel Dunbard3dcb4f82008-09-28 01:03:14 +00002587 objc_exception_try_enter(&d);
2588 if (!setjmp(d.jmp_buf)) {
Daniel Dunbar59e476b2009-08-03 17:06:42 +00002589 ... try body ...
Daniel Dunbard3dcb4f82008-09-28 01:03:14 +00002590 } else {
Daniel Dunbar59e476b2009-08-03 17:06:42 +00002591 // exception path
2592 id _caught = objc_exception_extract(&d);
2593
2594 // enter new try scope for handlers
2595 if (!setjmp(d.jmp_buf)) {
2596 ... match exception and execute catch blocks ...
2597
2598 // fell off end, rethrow.
2599 _rethrow = _caught;
2600 ... jump-through-finally to finally_rethrow ...
2601 } else {
2602 // exception in catch block
2603 _rethrow = objc_exception_extract(&d);
2604 _call_try_exit = false;
2605 ... jump-through-finally to finally_rethrow ...
2606 }
Daniel Dunbard3dcb4f82008-09-28 01:03:14 +00002607 }
Daniel Dunbar2efd5382008-09-30 01:06:03 +00002608 ... jump-through-finally to finally_end ...
Daniel Dunbard3dcb4f82008-09-28 01:03:14 +00002609
Daniel Dunbar59e476b2009-08-03 17:06:42 +00002610 finally:
Anders Carlssonda0e4562009-02-07 21:26:04 +00002611 if (_call_try_exit)
Daniel Dunbar59e476b2009-08-03 17:06:42 +00002612 objc_exception_try_exit(&d);
Anders Carlssonda0e4562009-02-07 21:26:04 +00002613
Daniel Dunbard3dcb4f82008-09-28 01:03:14 +00002614 ... finally block ....
Daniel Dunbar2efd5382008-09-30 01:06:03 +00002615 ... dispatch to finally destination ...
2616
Daniel Dunbar59e476b2009-08-03 17:06:42 +00002617 finally_rethrow:
Daniel Dunbar2efd5382008-09-30 01:06:03 +00002618 objc_exception_throw(_rethrow);
2619
Daniel Dunbar59e476b2009-08-03 17:06:42 +00002620 finally_end:
2621 }
Daniel Dunbard3dcb4f82008-09-28 01:03:14 +00002622
Daniel Dunbar59e476b2009-08-03 17:06:42 +00002623 This framework differs slightly from the one gcc uses, in that gcc
2624 uses _rethrow to determine if objc_exception_try_exit should be called
2625 and if the object should be rethrown. This breaks in the face of
2626 throwing nil and introduces unnecessary branches.
Daniel Dunbard3dcb4f82008-09-28 01:03:14 +00002627
Daniel Dunbar59e476b2009-08-03 17:06:42 +00002628 We specialize this framework for a few particular circumstances:
Daniel Dunbard3dcb4f82008-09-28 01:03:14 +00002629
Daniel Dunbar59e476b2009-08-03 17:06:42 +00002630 - If there are no catch blocks, then we avoid emitting the second
2631 exception handling context.
Daniel Dunbard3dcb4f82008-09-28 01:03:14 +00002632
Daniel Dunbar59e476b2009-08-03 17:06:42 +00002633 - If there is a catch-all catch block (i.e. @catch(...) or @catch(id
2634 e)) we avoid emitting the code to rethrow an uncaught exception.
Daniel Dunbard3dcb4f82008-09-28 01:03:14 +00002635
Daniel Dunbar59e476b2009-08-03 17:06:42 +00002636 - FIXME: If there is no @finally block we can do a few more
2637 simplifications.
Daniel Dunbard3dcb4f82008-09-28 01:03:14 +00002638
Daniel Dunbar59e476b2009-08-03 17:06:42 +00002639 Rethrows and Jumps-Through-Finally
2640 --
Daniel Dunbard3dcb4f82008-09-28 01:03:14 +00002641
John McCallbd309292010-07-06 01:34:17 +00002642 '@throw;' is supported by pushing the currently-caught exception
2643 onto ObjCEHStack while the @catch blocks are emitted.
Daniel Dunbard3dcb4f82008-09-28 01:03:14 +00002644
John McCallbd309292010-07-06 01:34:17 +00002645 Branches through the @finally block are handled with an ordinary
2646 normal cleanup. We do not register an EH cleanup; fragile-ABI ObjC
2647 exceptions are not compatible with C++ exceptions, and this is
2648 hardly the only place where this will go wrong.
Daniel Dunbar2efd5382008-09-30 01:06:03 +00002649
John McCallbd309292010-07-06 01:34:17 +00002650 @synchronized(expr) { stmt; } is emitted as if it were:
2651 id synch_value = expr;
2652 objc_sync_enter(synch_value);
2653 @try { stmt; } @finally { objc_sync_exit(synch_value); }
Daniel Dunbard3dcb4f82008-09-28 01:03:14 +00002654*/
2655
Fariborz Jahanianc2ad6dc2008-11-21 00:49:24 +00002656void CGObjCMac::EmitTryOrSynchronizedStmt(CodeGen::CodeGenFunction &CGF,
2657 const Stmt &S) {
2658 bool isTry = isa<ObjCAtTryStmt>(S);
John McCallbd309292010-07-06 01:34:17 +00002659
2660 // A destination for the fall-through edges of the catch handlers to
2661 // jump to.
2662 CodeGenFunction::JumpDest FinallyEnd =
2663 CGF.getJumpDestInCurrentScope("finally.end");
2664
2665 // A destination for the rethrow edge of the catch handlers to jump
2666 // to.
2667 CodeGenFunction::JumpDest FinallyRethrow =
2668 CGF.getJumpDestInCurrentScope("finally.rethrow");
Daniel Dunbar59e476b2009-08-03 17:06:42 +00002669
Daniel Dunbar94ceb612009-02-24 01:43:46 +00002670 // For @synchronized, call objc_sync_enter(sync.expr). The
2671 // evaluation of the expression must occur before we enter the
2672 // @synchronized. We can safely avoid a temp here because jumps into
2673 // @synchronized are illegal & this will dominate uses.
2674 llvm::Value *SyncArg = 0;
2675 if (!isTry) {
Daniel Dunbar59e476b2009-08-03 17:06:42 +00002676 SyncArg =
Daniel Dunbar94ceb612009-02-24 01:43:46 +00002677 CGF.EmitScalarExpr(cast<ObjCAtSynchronizedStmt>(S).getSynchExpr());
2678 SyncArg = CGF.Builder.CreateBitCast(SyncArg, ObjCTypes.ObjectPtrTy);
John McCallbd309292010-07-06 01:34:17 +00002679 CGF.Builder.CreateCall(ObjCTypes.getSyncEnterFn(), SyncArg)
2680 ->setDoesNotThrow();
Daniel Dunbar94ceb612009-02-24 01:43:46 +00002681 }
Daniel Dunbar2efd5382008-09-30 01:06:03 +00002682
Daniel Dunbar2efd5382008-09-30 01:06:03 +00002683 // Allocate memory for the exception data and rethrow pointer.
Anders Carlsson4f1c7c32008-09-09 17:59:25 +00002684 llvm::Value *ExceptionData = CGF.CreateTempAlloca(ObjCTypes.ExceptionDataTy,
2685 "exceptiondata.ptr");
Daniel Dunbar59e476b2009-08-03 17:06:42 +00002686 llvm::Value *RethrowPtr = CGF.CreateTempAlloca(ObjCTypes.ObjectPtrTy,
Daniel Dunbarc475d422008-10-29 22:36:39 +00002687 "_rethrow");
John McCallbd309292010-07-06 01:34:17 +00002688
2689 // Create a flag indicating whether the cleanup needs to call
2690 // objc_exception_try_exit. This is true except when
2691 // - no catches match and we're branching through the cleanup
2692 // just to rethrow the exception, or
2693 // - a catch matched and we're falling out of the catch handler.
2694 llvm::Value *CallTryExitVar = CGF.CreateTempAlloca(CGF.Builder.getInt1Ty(),
Anders Carlssonda0e4562009-02-07 21:26:04 +00002695 "_call_try_exit");
Daniel Dunbar59e476b2009-08-03 17:06:42 +00002696 CGF.Builder.CreateStore(llvm::ConstantInt::getTrue(VMContext),
John McCallbd309292010-07-06 01:34:17 +00002697 CallTryExitVar);
Daniel Dunbar59e476b2009-08-03 17:06:42 +00002698
John McCallbd309292010-07-06 01:34:17 +00002699 // Push a normal cleanup to leave the try scope.
2700 {
John McCall2b7fc382010-07-13 20:32:21 +00002701 CodeGenFunction::CleanupBlock FinallyScope(CGF, NormalCleanup);
Daniel Dunbar2efd5382008-09-30 01:06:03 +00002702
John McCallbd309292010-07-06 01:34:17 +00002703 // Check whether we need to call objc_exception_try_exit.
2704 // In optimized code, this branch will always be folded.
2705 llvm::BasicBlock *FinallyCallExit =
2706 CGF.createBasicBlock("finally.call_exit");
2707 llvm::BasicBlock *FinallyNoCallExit =
2708 CGF.createBasicBlock("finally.no_call_exit");
2709 CGF.Builder.CreateCondBr(CGF.Builder.CreateLoad(CallTryExitVar),
2710 FinallyCallExit, FinallyNoCallExit);
2711
2712 CGF.EmitBlock(FinallyCallExit);
2713 CGF.Builder.CreateCall(ObjCTypes.getExceptionTryExitFn(), ExceptionData)
2714 ->setDoesNotThrow();
2715
2716 CGF.EmitBlock(FinallyNoCallExit);
2717
2718 if (isTry) {
2719 if (const ObjCAtFinallyStmt* FinallyStmt =
2720 cast<ObjCAtTryStmt>(S).getFinallyStmt())
2721 CGF.EmitStmt(FinallyStmt->getFinallyBody());
2722
2723 // ~CleanupBlock requires there to be an exit block.
2724 CGF.EnsureInsertPoint();
2725 } else {
2726 // Emit objc_sync_exit(expr); as finally's sole statement for
2727 // @synchronized.
2728 CGF.Builder.CreateCall(ObjCTypes.getSyncExitFn(), SyncArg)
2729 ->setDoesNotThrow();
2730 }
2731 }
2732
2733 // Enter a try block:
2734 // - Call objc_exception_try_enter to push ExceptionData on top of
2735 // the EH stack.
2736 CGF.Builder.CreateCall(ObjCTypes.getExceptionTryEnterFn(), ExceptionData)
2737 ->setDoesNotThrow();
2738
2739 // - Call setjmp on the exception data buffer.
2740 llvm::Constant *Zero = llvm::ConstantInt::get(CGF.Builder.getInt32Ty(), 0);
2741 llvm::Value *GEPIndexes[] = { Zero, Zero, Zero };
2742 llvm::Value *SetJmpBuffer =
2743 CGF.Builder.CreateGEP(ExceptionData, GEPIndexes, GEPIndexes+3, "setjmp_buffer");
2744 llvm::CallInst *SetJmpResult =
2745 CGF.Builder.CreateCall(ObjCTypes.getSetJmpFn(), SetJmpBuffer, "setjmp_result");
2746 SetJmpResult->setDoesNotThrow();
2747
2748 // If setjmp returned 0, enter the protected block; otherwise,
2749 // branch to the handler.
Daniel Dunbar75283ff2008-11-11 02:29:29 +00002750 llvm::BasicBlock *TryBlock = CGF.createBasicBlock("try");
2751 llvm::BasicBlock *TryHandler = CGF.createBasicBlock("try.handler");
John McCallbd309292010-07-06 01:34:17 +00002752 llvm::Value *DidCatch =
2753 CGF.Builder.CreateIsNull(SetJmpResult, "did_catch_exception");
2754 CGF.Builder.CreateCondBr(DidCatch, TryBlock, TryHandler);
Anders Carlsson4f1c7c32008-09-09 17:59:25 +00002755
John McCallbd309292010-07-06 01:34:17 +00002756 // Emit the protected block.
Anders Carlsson4f1c7c32008-09-09 17:59:25 +00002757 CGF.EmitBlock(TryBlock);
Daniel Dunbar59e476b2009-08-03 17:06:42 +00002758 CGF.EmitStmt(isTry ? cast<ObjCAtTryStmt>(S).getTryBody()
John McCallbd309292010-07-06 01:34:17 +00002759 : cast<ObjCAtSynchronizedStmt>(S).getSynchBody());
Anders Carlssonbfee7e92009-02-09 20:38:58 +00002760 CGF.EmitBranchThroughCleanup(FinallyEnd);
Daniel Dunbar59e476b2009-08-03 17:06:42 +00002761
John McCallbd309292010-07-06 01:34:17 +00002762 // Emit the exception handler block.
Daniel Dunbar7f086782008-09-27 23:30:04 +00002763 CGF.EmitBlock(TryHandler);
Daniel Dunbarb22ff592008-09-27 07:03:52 +00002764
2765 // Retrieve the exception object. We may emit multiple blocks but
2766 // nothing can cross this so the value is already in SSA form.
John McCallbd309292010-07-06 01:34:17 +00002767 llvm::CallInst *Caught =
Chris Lattnerc6406db2009-04-22 02:26:14 +00002768 CGF.Builder.CreateCall(ObjCTypes.getExceptionExtractFn(),
2769 ExceptionData, "caught");
John McCallbd309292010-07-06 01:34:17 +00002770 Caught->setDoesNotThrow();
2771
2772 // Remember the exception to rethrow.
2773 CGF.Builder.CreateStore(Caught, RethrowPtr);
2774
2775 // Note: at this point, objc_exception_throw already popped the
2776 // catch handler, so anything that branches to the cleanup needs
2777 // to set CallTryExitVar to false.
2778
2779 // For a @synchronized (or a @try with no catches), just branch
2780 // through the cleanup to the rethrow block.
2781 if (!isTry || !cast<ObjCAtTryStmt>(S).getNumCatchStmts()) {
2782 // Tell the cleanup not to re-pop the exit.
Daniel Dunbar59e476b2009-08-03 17:06:42 +00002783 CGF.Builder.CreateStore(llvm::ConstantInt::getFalse(VMContext),
John McCallbd309292010-07-06 01:34:17 +00002784 CallTryExitVar);
2785
Anders Carlssonbfee7e92009-02-09 20:38:58 +00002786 CGF.EmitBranchThroughCleanup(FinallyRethrow);
John McCallbd309292010-07-06 01:34:17 +00002787
2788 // Otherwise, we have to match against the caught exceptions.
2789 } else {
2790 // Push the exception to rethrow onto the EH value stack for the
2791 // benefit of any @throws in the handlers.
2792 CGF.ObjCEHValueStack.push_back(Caught);
2793
Douglas Gregor96c79492010-04-23 22:50:49 +00002794 const ObjCAtTryStmt* AtTryStmt = cast<ObjCAtTryStmt>(&S);
2795
Daniel Dunbarb22ff592008-09-27 07:03:52 +00002796 // Enter a new exception try block (in case a @catch block throws
John McCallbd309292010-07-06 01:34:17 +00002797 // an exception). Now CallTryExitVar (currently true) is back in
2798 // synch with reality.
2799 CGF.Builder.CreateCall(ObjCTypes.getExceptionTryEnterFn(), ExceptionData)
2800 ->setDoesNotThrow();
Daniel Dunbar59e476b2009-08-03 17:06:42 +00002801
John McCallbd309292010-07-06 01:34:17 +00002802 llvm::CallInst *SetJmpResult =
2803 CGF.Builder.CreateCall(ObjCTypes.getSetJmpFn(), SetJmpBuffer,
2804 "setjmp.result");
2805 SetJmpResult->setDoesNotThrow();
2806
2807 llvm::Value *Threw =
2808 CGF.Builder.CreateIsNotNull(SetJmpResult, "did_catch_exception");
Anders Carlsson4f1c7c32008-09-09 17:59:25 +00002809
Daniel Dunbar75283ff2008-11-11 02:29:29 +00002810 llvm::BasicBlock *CatchBlock = CGF.createBasicBlock("catch");
John McCallbd309292010-07-06 01:34:17 +00002811 llvm::BasicBlock *CatchHandler = CGF.createBasicBlock("catch_for_catch");
Daniel Dunbar7f086782008-09-27 23:30:04 +00002812 CGF.Builder.CreateCondBr(Threw, CatchHandler, CatchBlock);
Daniel Dunbar59e476b2009-08-03 17:06:42 +00002813
Anders Carlsson4f1c7c32008-09-09 17:59:25 +00002814 CGF.EmitBlock(CatchBlock);
Daniel Dunbar59e476b2009-08-03 17:06:42 +00002815
Daniel Dunbarb22ff592008-09-27 07:03:52 +00002816 // Handle catch list. As a special case we check if everything is
2817 // matched and avoid generating code for falling off the end if
2818 // so.
2819 bool AllMatched = false;
Douglas Gregor96c79492010-04-23 22:50:49 +00002820 for (unsigned I = 0, N = AtTryStmt->getNumCatchStmts(); I != N; ++I) {
2821 const ObjCAtCatchStmt *CatchStmt = AtTryStmt->getCatchStmt(I);
Anders Carlsson4f1c7c32008-09-09 17:59:25 +00002822
Douglas Gregor46a572b2010-04-26 16:46:50 +00002823 const VarDecl *CatchParam = CatchStmt->getCatchParamDecl();
Steve Naroff7cae42b2009-07-10 23:34:53 +00002824 const ObjCObjectPointerType *OPT = 0;
Daniel Dunbar523208f2008-09-27 07:36:24 +00002825
Anders Carlsson4f1c7c32008-09-09 17:59:25 +00002826 // catch(...) always matches.
Daniel Dunbarb22ff592008-09-27 07:03:52 +00002827 if (!CatchParam) {
2828 AllMatched = true;
2829 } else {
John McCall9dd450b2009-09-21 23:43:11 +00002830 OPT = CatchParam->getType()->getAs<ObjCObjectPointerType>();
Daniel Dunbar59e476b2009-08-03 17:06:42 +00002831
John McCallbd309292010-07-06 01:34:17 +00002832 // catch(id e) always matches under this ABI, since only
2833 // ObjC exceptions end up here in the first place.
Daniel Dunbar86919f42008-09-27 22:21:14 +00002834 // FIXME: For the time being we also match id<X>; this should
2835 // be rejected by Sema instead.
Eli Friedman55179ca2009-07-11 00:57:02 +00002836 if (OPT && (OPT->isObjCIdType() || OPT->isObjCQualifiedIdType()))
Daniel Dunbarb22ff592008-09-27 07:03:52 +00002837 AllMatched = true;
Anders Carlsson4f1c7c32008-09-09 17:59:25 +00002838 }
Daniel Dunbar59e476b2009-08-03 17:06:42 +00002839
John McCallbd309292010-07-06 01:34:17 +00002840 // If this is a catch-all, we don't need to test anything.
Daniel Dunbar59e476b2009-08-03 17:06:42 +00002841 if (AllMatched) {
John McCallbd309292010-07-06 01:34:17 +00002842 CodeGenFunction::RunCleanupsScope CatchVarCleanups(CGF);
2843
Anders Carlsson9396a892008-09-11 09:15:33 +00002844 if (CatchParam) {
Steve Naroff371b8fb2009-03-03 19:52:17 +00002845 CGF.EmitLocalBlockVarDecl(*CatchParam);
Daniel Dunbar5c7e3932008-11-11 23:11:34 +00002846 assert(CGF.HaveInsertPoint() && "DeclStmt destroyed insert point?");
John McCallbd309292010-07-06 01:34:17 +00002847
2848 // These types work out because ConvertType(id) == i8*.
Steve Naroff371b8fb2009-03-03 19:52:17 +00002849 CGF.Builder.CreateStore(Caught, CGF.GetAddrOfLocalVar(CatchParam));
Anders Carlsson9396a892008-09-11 09:15:33 +00002850 }
Daniel Dunbar59e476b2009-08-03 17:06:42 +00002851
Anders Carlsson9396a892008-09-11 09:15:33 +00002852 CGF.EmitStmt(CatchStmt->getCatchBody());
John McCallbd309292010-07-06 01:34:17 +00002853
2854 // The scope of the catch variable ends right here.
2855 CatchVarCleanups.ForceCleanup();
2856
Anders Carlssonbfee7e92009-02-09 20:38:58 +00002857 CGF.EmitBranchThroughCleanup(FinallyEnd);
Anders Carlsson4f1c7c32008-09-09 17:59:25 +00002858 break;
2859 }
Daniel Dunbar59e476b2009-08-03 17:06:42 +00002860
Steve Naroff7cae42b2009-07-10 23:34:53 +00002861 assert(OPT && "Unexpected non-object pointer type in @catch");
John McCall96fa4842010-05-17 21:00:27 +00002862 const ObjCObjectType *ObjTy = OPT->getObjectType();
John McCallbd309292010-07-06 01:34:17 +00002863
2864 // FIXME: @catch (Class c) ?
John McCall96fa4842010-05-17 21:00:27 +00002865 ObjCInterfaceDecl *IDecl = ObjTy->getInterface();
2866 assert(IDecl && "Catch parameter must have Objective-C type!");
Anders Carlsson4f1c7c32008-09-09 17:59:25 +00002867
2868 // Check if the @catch block matches the exception object.
John McCall96fa4842010-05-17 21:00:27 +00002869 llvm::Value *Class = EmitClassRef(CGF.Builder, IDecl);
Daniel Dunbar59e476b2009-08-03 17:06:42 +00002870
John McCallbd309292010-07-06 01:34:17 +00002871 llvm::CallInst *Match =
Chris Lattnerc6406db2009-04-22 02:26:14 +00002872 CGF.Builder.CreateCall2(ObjCTypes.getExceptionMatchFn(),
2873 Class, Caught, "match");
John McCallbd309292010-07-06 01:34:17 +00002874 Match->setDoesNotThrow();
Daniel Dunbar59e476b2009-08-03 17:06:42 +00002875
John McCallbd309292010-07-06 01:34:17 +00002876 llvm::BasicBlock *MatchedBlock = CGF.createBasicBlock("match");
2877 llvm::BasicBlock *NextCatchBlock = CGF.createBasicBlock("catch.next");
Daniel Dunbar59e476b2009-08-03 17:06:42 +00002878
2879 CGF.Builder.CreateCondBr(CGF.Builder.CreateIsNotNull(Match, "matched"),
Daniel Dunbar7f086782008-09-27 23:30:04 +00002880 MatchedBlock, NextCatchBlock);
Daniel Dunbar59e476b2009-08-03 17:06:42 +00002881
Anders Carlsson4f1c7c32008-09-09 17:59:25 +00002882 // Emit the @catch block.
2883 CGF.EmitBlock(MatchedBlock);
John McCallbd309292010-07-06 01:34:17 +00002884
2885 // Collect any cleanups for the catch variable. The scope lasts until
2886 // the end of the catch body.
2887 CodeGenFunction::RunCleanupsScope CatchVarCleanups(CGF);
2888
Steve Naroff371b8fb2009-03-03 19:52:17 +00002889 CGF.EmitLocalBlockVarDecl(*CatchParam);
Daniel Dunbar5c7e3932008-11-11 23:11:34 +00002890 assert(CGF.HaveInsertPoint() && "DeclStmt destroyed insert point?");
Daniel Dunbard3dcb4f82008-09-28 01:03:14 +00002891
John McCallbd309292010-07-06 01:34:17 +00002892 // Initialize the catch variable.
Daniel Dunbar59e476b2009-08-03 17:06:42 +00002893 llvm::Value *Tmp =
2894 CGF.Builder.CreateBitCast(Caught,
2895 CGF.ConvertType(CatchParam->getType()),
Daniel Dunbard3dcb4f82008-09-28 01:03:14 +00002896 "tmp");
Steve Naroff371b8fb2009-03-03 19:52:17 +00002897 CGF.Builder.CreateStore(Tmp, CGF.GetAddrOfLocalVar(CatchParam));
Daniel Dunbar59e476b2009-08-03 17:06:42 +00002898
Anders Carlsson9396a892008-09-11 09:15:33 +00002899 CGF.EmitStmt(CatchStmt->getCatchBody());
John McCallbd309292010-07-06 01:34:17 +00002900
2901 // We're done with the catch variable.
2902 CatchVarCleanups.ForceCleanup();
2903
Anders Carlssonbfee7e92009-02-09 20:38:58 +00002904 CGF.EmitBranchThroughCleanup(FinallyEnd);
Daniel Dunbar59e476b2009-08-03 17:06:42 +00002905
Anders Carlsson4f1c7c32008-09-09 17:59:25 +00002906 CGF.EmitBlock(NextCatchBlock);
2907 }
2908
John McCallbd309292010-07-06 01:34:17 +00002909 CGF.ObjCEHValueStack.pop_back();
2910
Daniel Dunbarb22ff592008-09-27 07:03:52 +00002911 if (!AllMatched) {
2912 // None of the handlers caught the exception, so store it to be
2913 // rethrown at the end of the @finally block.
2914 CGF.Builder.CreateStore(Caught, RethrowPtr);
Anders Carlssonbfee7e92009-02-09 20:38:58 +00002915 CGF.EmitBranchThroughCleanup(FinallyRethrow);
Daniel Dunbarb22ff592008-09-27 07:03:52 +00002916 }
Daniel Dunbar59e476b2009-08-03 17:06:42 +00002917
Daniel Dunbarb22ff592008-09-27 07:03:52 +00002918 // Emit the exception handler for the @catch blocks.
Daniel Dunbar59e476b2009-08-03 17:06:42 +00002919 CGF.EmitBlock(CatchHandler);
John McCallbd309292010-07-06 01:34:17 +00002920
2921 // Rethrow the new exception, not the old one.
2922 Caught = CGF.Builder.CreateCall(ObjCTypes.getExceptionExtractFn(),
2923 ExceptionData);
2924 Caught->setDoesNotThrow();
Anders Carlsson4f1c7c32008-09-09 17:59:25 +00002925 CGF.Builder.CreateStore(Caught, RethrowPtr);
John McCallbd309292010-07-06 01:34:17 +00002926
2927 // Don't pop the catch handler; the throw already did.
Daniel Dunbar59e476b2009-08-03 17:06:42 +00002928 CGF.Builder.CreateStore(llvm::ConstantInt::getFalse(VMContext),
John McCallbd309292010-07-06 01:34:17 +00002929 CallTryExitVar);
Anders Carlssonbfee7e92009-02-09 20:38:58 +00002930 CGF.EmitBranchThroughCleanup(FinallyRethrow);
Anders Carlsson4f1c7c32008-09-09 17:59:25 +00002931 }
Daniel Dunbar59e476b2009-08-03 17:06:42 +00002932
John McCallbd309292010-07-06 01:34:17 +00002933 // Pop the cleanup.
2934 CGF.PopCleanupBlock();
2935 CGF.EmitBlock(FinallyEnd.Block);
Anders Carlssonbfee7e92009-02-09 20:38:58 +00002936
John McCallbd309292010-07-06 01:34:17 +00002937 // Emit the rethrow block.
2938 CGF.Builder.ClearInsertionPoint();
2939 CGF.EmitBlock(FinallyRethrow.Block, true);
2940 if (CGF.HaveInsertPoint()) {
2941 CGF.Builder.CreateCall(ObjCTypes.getExceptionThrowFn(),
2942 CGF.Builder.CreateLoad(RethrowPtr))
2943 ->setDoesNotThrow();
2944 CGF.Builder.CreateUnreachable();
Fariborz Jahaniane2caaaa2008-11-21 19:21:53 +00002945 }
Anders Carlsson4f1c7c32008-09-09 17:59:25 +00002946
John McCallbd309292010-07-06 01:34:17 +00002947 CGF.Builder.SetInsertPoint(FinallyEnd.Block);
Anders Carlsson1963b0c2008-09-09 10:04:29 +00002948}
2949
2950void CGObjCMac::EmitThrowStmt(CodeGen::CodeGenFunction &CGF,
Daniel Dunbar2efd5382008-09-30 01:06:03 +00002951 const ObjCAtThrowStmt &S) {
Anders Carlssone005aa12008-09-09 16:16:55 +00002952 llvm::Value *ExceptionAsObject;
Daniel Dunbar59e476b2009-08-03 17:06:42 +00002953
Anders Carlssone005aa12008-09-09 16:16:55 +00002954 if (const Expr *ThrowExpr = S.getThrowExpr()) {
2955 llvm::Value *Exception = CGF.EmitScalarExpr(ThrowExpr);
Daniel Dunbar59e476b2009-08-03 17:06:42 +00002956 ExceptionAsObject =
Anders Carlssone005aa12008-09-09 16:16:55 +00002957 CGF.Builder.CreateBitCast(Exception, ObjCTypes.ObjectPtrTy, "tmp");
2958 } else {
Daniel Dunbar59e476b2009-08-03 17:06:42 +00002959 assert((!CGF.ObjCEHValueStack.empty() && CGF.ObjCEHValueStack.back()) &&
Daniel Dunbard3dcb4f82008-09-28 01:03:14 +00002960 "Unexpected rethrow outside @catch block.");
Anders Carlssonbf8a1be2009-02-07 21:37:21 +00002961 ExceptionAsObject = CGF.ObjCEHValueStack.back();
Anders Carlssone005aa12008-09-09 16:16:55 +00002962 }
Daniel Dunbar59e476b2009-08-03 17:06:42 +00002963
John McCallbd309292010-07-06 01:34:17 +00002964 CGF.Builder.CreateCall(ObjCTypes.getExceptionThrowFn(), ExceptionAsObject)
2965 ->setDoesNotReturn();
Anders Carlsson4f1c7c32008-09-09 17:59:25 +00002966 CGF.Builder.CreateUnreachable();
Daniel Dunbar5c7e3932008-11-11 23:11:34 +00002967
2968 // Clear the insertion point to indicate we are in unreachable code.
2969 CGF.Builder.ClearInsertionPoint();
Anders Carlsson1963b0c2008-09-09 10:04:29 +00002970}
2971
Fariborz Jahanian83f45b552008-11-18 22:37:34 +00002972/// EmitObjCWeakRead - Code gen for loading value of a __weak
Fariborz Jahanianf5125d12008-11-18 21:45:40 +00002973/// object: objc_read_weak (id *src)
2974///
Fariborz Jahanian83f45b552008-11-18 22:37:34 +00002975llvm::Value * CGObjCMac::EmitObjCWeakRead(CodeGen::CodeGenFunction &CGF,
Mike Stump11289f42009-09-09 15:08:12 +00002976 llvm::Value *AddrWeakObj) {
Eli Friedmana374b682009-03-07 03:57:15 +00002977 const llvm::Type* DestTy =
Daniel Dunbar59e476b2009-08-03 17:06:42 +00002978 cast<llvm::PointerType>(AddrWeakObj->getType())->getElementType();
2979 AddrWeakObj = CGF.Builder.CreateBitCast(AddrWeakObj,
2980 ObjCTypes.PtrObjectPtrTy);
Chris Lattnerce8754e2009-04-22 02:44:54 +00002981 llvm::Value *read_weak = CGF.Builder.CreateCall(ObjCTypes.getGcReadWeakFn(),
Fariborz Jahanian83f45b552008-11-18 22:37:34 +00002982 AddrWeakObj, "weakread");
Eli Friedmana374b682009-03-07 03:57:15 +00002983 read_weak = CGF.Builder.CreateBitCast(read_weak, DestTy);
Fariborz Jahanianf5125d12008-11-18 21:45:40 +00002984 return read_weak;
2985}
2986
Fariborz Jahanian83f45b552008-11-18 22:37:34 +00002987/// EmitObjCWeakAssign - Code gen for assigning to a __weak object.
2988/// objc_assign_weak (id src, id *dst)
2989///
2990void CGObjCMac::EmitObjCWeakAssign(CodeGen::CodeGenFunction &CGF,
Mike Stump11289f42009-09-09 15:08:12 +00002991 llvm::Value *src, llvm::Value *dst) {
Fariborz Jahanianaedcfa42009-03-23 19:10:40 +00002992 const llvm::Type * SrcTy = src->getType();
2993 if (!isa<llvm::PointerType>(SrcTy)) {
Duncan Sandsc76fe8b2009-05-09 07:08:47 +00002994 unsigned Size = CGM.getTargetData().getTypeAllocSize(SrcTy);
Fariborz Jahanianaedcfa42009-03-23 19:10:40 +00002995 assert(Size <= 8 && "does not support size > 8");
2996 src = (Size == 4) ? CGF.Builder.CreateBitCast(src, ObjCTypes.IntTy)
Daniel Dunbar59e476b2009-08-03 17:06:42 +00002997 : CGF.Builder.CreateBitCast(src, ObjCTypes.LongLongTy);
Fariborz Jahanian1b074a32009-03-13 00:42:52 +00002998 src = CGF.Builder.CreateIntToPtr(src, ObjCTypes.Int8PtrTy);
2999 }
Fariborz Jahanian50a12702008-11-19 17:34:06 +00003000 src = CGF.Builder.CreateBitCast(src, ObjCTypes.ObjectPtrTy);
3001 dst = CGF.Builder.CreateBitCast(dst, ObjCTypes.PtrObjectPtrTy);
Chris Lattner6fdd57c2009-04-17 22:12:36 +00003002 CGF.Builder.CreateCall2(ObjCTypes.getGcAssignWeakFn(),
Fariborz Jahanian83f45b552008-11-18 22:37:34 +00003003 src, dst, "weakassign");
3004 return;
3005}
3006
Fariborz Jahaniand7db9642008-11-19 00:59:10 +00003007/// EmitObjCGlobalAssign - Code gen for assigning to a __strong object.
3008/// objc_assign_global (id src, id *dst)
3009///
3010void CGObjCMac::EmitObjCGlobalAssign(CodeGen::CodeGenFunction &CGF,
Fariborz Jahanian217af242010-07-20 20:30:03 +00003011 llvm::Value *src, llvm::Value *dst,
3012 bool threadlocal) {
Fariborz Jahanianaedcfa42009-03-23 19:10:40 +00003013 const llvm::Type * SrcTy = src->getType();
3014 if (!isa<llvm::PointerType>(SrcTy)) {
Duncan Sandsc76fe8b2009-05-09 07:08:47 +00003015 unsigned Size = CGM.getTargetData().getTypeAllocSize(SrcTy);
Fariborz Jahanianaedcfa42009-03-23 19:10:40 +00003016 assert(Size <= 8 && "does not support size > 8");
3017 src = (Size == 4) ? CGF.Builder.CreateBitCast(src, ObjCTypes.IntTy)
Daniel Dunbar59e476b2009-08-03 17:06:42 +00003018 : CGF.Builder.CreateBitCast(src, ObjCTypes.LongLongTy);
Fariborz Jahanian1b074a32009-03-13 00:42:52 +00003019 src = CGF.Builder.CreateIntToPtr(src, ObjCTypes.Int8PtrTy);
3020 }
Fariborz Jahanian50a12702008-11-19 17:34:06 +00003021 src = CGF.Builder.CreateBitCast(src, ObjCTypes.ObjectPtrTy);
3022 dst = CGF.Builder.CreateBitCast(dst, ObjCTypes.PtrObjectPtrTy);
Fariborz Jahanian217af242010-07-20 20:30:03 +00003023 if (!threadlocal)
3024 CGF.Builder.CreateCall2(ObjCTypes.getGcAssignGlobalFn(),
3025 src, dst, "globalassign");
3026 else
3027 CGF.Builder.CreateCall2(ObjCTypes.getGcAssignThreadLocalFn(),
3028 src, dst, "threadlocalassign");
Fariborz Jahaniand7db9642008-11-19 00:59:10 +00003029 return;
3030}
3031
Fariborz Jahaniane881b532008-11-20 19:23:36 +00003032/// EmitObjCIvarAssign - Code gen for assigning to a __strong object.
Fariborz Jahanian7a95d722009-09-24 22:25:38 +00003033/// objc_assign_ivar (id src, id *dst, ptrdiff_t ivaroffset)
Fariborz Jahaniane881b532008-11-20 19:23:36 +00003034///
3035void CGObjCMac::EmitObjCIvarAssign(CodeGen::CodeGenFunction &CGF,
Fariborz Jahanian7a95d722009-09-24 22:25:38 +00003036 llvm::Value *src, llvm::Value *dst,
3037 llvm::Value *ivarOffset) {
3038 assert(ivarOffset && "EmitObjCIvarAssign - ivarOffset is NULL");
Fariborz Jahanianaedcfa42009-03-23 19:10:40 +00003039 const llvm::Type * SrcTy = src->getType();
3040 if (!isa<llvm::PointerType>(SrcTy)) {
Duncan Sandsc76fe8b2009-05-09 07:08:47 +00003041 unsigned Size = CGM.getTargetData().getTypeAllocSize(SrcTy);
Fariborz Jahanianaedcfa42009-03-23 19:10:40 +00003042 assert(Size <= 8 && "does not support size > 8");
3043 src = (Size == 4) ? CGF.Builder.CreateBitCast(src, ObjCTypes.IntTy)
Daniel Dunbar59e476b2009-08-03 17:06:42 +00003044 : CGF.Builder.CreateBitCast(src, ObjCTypes.LongLongTy);
Fariborz Jahanian1b074a32009-03-13 00:42:52 +00003045 src = CGF.Builder.CreateIntToPtr(src, ObjCTypes.Int8PtrTy);
3046 }
Fariborz Jahaniane881b532008-11-20 19:23:36 +00003047 src = CGF.Builder.CreateBitCast(src, ObjCTypes.ObjectPtrTy);
3048 dst = CGF.Builder.CreateBitCast(dst, ObjCTypes.PtrObjectPtrTy);
Fariborz Jahanian7a95d722009-09-24 22:25:38 +00003049 CGF.Builder.CreateCall3(ObjCTypes.getGcAssignIvarFn(),
3050 src, dst, ivarOffset);
Fariborz Jahaniane881b532008-11-20 19:23:36 +00003051 return;
3052}
3053
Fariborz Jahaniand7db9642008-11-19 00:59:10 +00003054/// EmitObjCStrongCastAssign - Code gen for assigning to a __strong cast object.
3055/// objc_assign_strongCast (id src, id *dst)
3056///
3057void CGObjCMac::EmitObjCStrongCastAssign(CodeGen::CodeGenFunction &CGF,
Mike Stump11289f42009-09-09 15:08:12 +00003058 llvm::Value *src, llvm::Value *dst) {
Fariborz Jahanianaedcfa42009-03-23 19:10:40 +00003059 const llvm::Type * SrcTy = src->getType();
3060 if (!isa<llvm::PointerType>(SrcTy)) {
Duncan Sandsc76fe8b2009-05-09 07:08:47 +00003061 unsigned Size = CGM.getTargetData().getTypeAllocSize(SrcTy);
Fariborz Jahanianaedcfa42009-03-23 19:10:40 +00003062 assert(Size <= 8 && "does not support size > 8");
3063 src = (Size == 4) ? CGF.Builder.CreateBitCast(src, ObjCTypes.IntTy)
Daniel Dunbar59e476b2009-08-03 17:06:42 +00003064 : CGF.Builder.CreateBitCast(src, ObjCTypes.LongLongTy);
Fariborz Jahanian1b074a32009-03-13 00:42:52 +00003065 src = CGF.Builder.CreateIntToPtr(src, ObjCTypes.Int8PtrTy);
3066 }
Fariborz Jahanian50a12702008-11-19 17:34:06 +00003067 src = CGF.Builder.CreateBitCast(src, ObjCTypes.ObjectPtrTy);
3068 dst = CGF.Builder.CreateBitCast(dst, ObjCTypes.PtrObjectPtrTy);
Chris Lattner0a696a422009-04-22 02:38:11 +00003069 CGF.Builder.CreateCall2(ObjCTypes.getGcAssignStrongCastFn(),
Fariborz Jahaniand7db9642008-11-19 00:59:10 +00003070 src, dst, "weakassign");
3071 return;
3072}
3073
Fariborz Jahanian5f21d2f2009-07-08 01:18:33 +00003074void CGObjCMac::EmitGCMemmoveCollectable(CodeGen::CodeGenFunction &CGF,
Daniel Dunbar59e476b2009-08-03 17:06:42 +00003075 llvm::Value *DestPtr,
3076 llvm::Value *SrcPtr,
Fariborz Jahanian021510e2010-06-15 22:44:06 +00003077 llvm::Value *size) {
Fariborz Jahanian5f21d2f2009-07-08 01:18:33 +00003078 SrcPtr = CGF.Builder.CreateBitCast(SrcPtr, ObjCTypes.Int8PtrTy);
3079 DestPtr = CGF.Builder.CreateBitCast(DestPtr, ObjCTypes.Int8PtrTy);
Fariborz Jahanian5f21d2f2009-07-08 01:18:33 +00003080 CGF.Builder.CreateCall3(ObjCTypes.GcMemmoveCollectableFn(),
Fariborz Jahanian021510e2010-06-15 22:44:06 +00003081 DestPtr, SrcPtr, size);
Fariborz Jahanian5f21d2f2009-07-08 01:18:33 +00003082 return;
3083}
3084
Fariborz Jahanian9f84b782009-02-02 20:02:29 +00003085/// EmitObjCValueForIvar - Code Gen for ivar reference.
3086///
Fariborz Jahanian712bfa62009-02-03 19:03:09 +00003087LValue CGObjCMac::EmitObjCValueForIvar(CodeGen::CodeGenFunction &CGF,
3088 QualType ObjectTy,
3089 llvm::Value *BaseValue,
3090 const ObjCIvarDecl *Ivar,
Fariborz Jahanian712bfa62009-02-03 19:03:09 +00003091 unsigned CVRQualifiers) {
John McCall8b07ec22010-05-15 11:32:37 +00003092 const ObjCInterfaceDecl *ID =
3093 ObjectTy->getAs<ObjCObjectType>()->getInterface();
Daniel Dunbar9fd114d2009-04-22 07:32:20 +00003094 return EmitValueForIvarAtOffset(CGF, ID, BaseValue, Ivar, CVRQualifiers,
3095 EmitIvarOffset(CGF, ID, Ivar));
Fariborz Jahanian9f84b782009-02-02 20:02:29 +00003096}
3097
Fariborz Jahanian21fc74c2009-02-10 19:02:04 +00003098llvm::Value *CGObjCMac::EmitIvarOffset(CodeGen::CodeGenFunction &CGF,
Daniel Dunbar722f4242009-04-22 05:08:15 +00003099 const ObjCInterfaceDecl *Interface,
Fariborz Jahanian21fc74c2009-02-10 19:02:04 +00003100 const ObjCIvarDecl *Ivar) {
Daniel Dunbar9fd114d2009-04-22 07:32:20 +00003101 uint64_t Offset = ComputeIvarBaseOffset(CGM, Interface, Ivar);
Owen Andersonb7a2fe62009-07-24 23:12:58 +00003102 return llvm::ConstantInt::get(
Daniel Dunbar59e476b2009-08-03 17:06:42 +00003103 CGM.getTypes().ConvertType(CGM.getContext().LongTy),
3104 Offset);
Fariborz Jahanian21fc74c2009-02-10 19:02:04 +00003105}
3106
Daniel Dunbar3ad53482008-08-11 21:35:06 +00003107/* *** Private Interface *** */
3108
3109/// EmitImageInfo - Emit the image info marker used to encode some module
3110/// level information.
3111///
3112/// See: <rdr://4810609&4810587&4810587>
3113/// struct IMAGE_INFO {
3114/// unsigned version;
3115/// unsigned flags;
3116/// };
3117enum ImageInfoFlags {
Daniel Dunbar5e639272010-04-25 20:39:01 +00003118 eImageInfo_FixAndContinue = (1 << 0),
Daniel Dunbar59e476b2009-08-03 17:06:42 +00003119 eImageInfo_GarbageCollected = (1 << 1),
3120 eImageInfo_GCOnly = (1 << 2),
Daniel Dunbar75e909f2009-04-20 07:11:47 +00003121 eImageInfo_OptimizedByDyld = (1 << 3), // FIXME: When is this set.
3122
Daniel Dunbar5e639272010-04-25 20:39:01 +00003123 // A flag indicating that the module has no instances of a @synthesize of a
3124 // superclass variable. <rdar://problem/6803242>
Daniel Dunbar75e909f2009-04-20 07:11:47 +00003125 eImageInfo_CorrectedSynthesize = (1 << 4)
Daniel Dunbar3ad53482008-08-11 21:35:06 +00003126};
3127
Daniel Dunbar5e639272010-04-25 20:39:01 +00003128void CGObjCCommonMac::EmitImageInfo() {
Daniel Dunbar3ad53482008-08-11 21:35:06 +00003129 unsigned version = 0; // Version is unused?
3130 unsigned flags = 0;
3131
3132 // FIXME: Fix and continue?
3133 if (CGM.getLangOptions().getGCMode() != LangOptions::NonGC)
3134 flags |= eImageInfo_GarbageCollected;
3135 if (CGM.getLangOptions().getGCMode() == LangOptions::GCOnly)
3136 flags |= eImageInfo_GCOnly;
Daniel Dunbar59e476b2009-08-03 17:06:42 +00003137
Daniel Dunbar75e909f2009-04-20 07:11:47 +00003138 // We never allow @synthesize of a superclass property.
3139 flags |= eImageInfo_CorrectedSynthesize;
Daniel Dunbar3ad53482008-08-11 21:35:06 +00003140
Chris Lattner5e016ae2010-06-27 07:15:29 +00003141 const llvm::Type *Int32Ty = llvm::Type::getInt32Ty(VMContext);
3142
Daniel Dunbar3ad53482008-08-11 21:35:06 +00003143 // Emitted as int[2];
3144 llvm::Constant *values[2] = {
Chris Lattner5e016ae2010-06-27 07:15:29 +00003145 llvm::ConstantInt::get(Int32Ty, version),
3146 llvm::ConstantInt::get(Int32Ty, flags)
Daniel Dunbar3ad53482008-08-11 21:35:06 +00003147 };
Chris Lattner5e016ae2010-06-27 07:15:29 +00003148 llvm::ArrayType *AT = llvm::ArrayType::get(Int32Ty, 2);
Daniel Dunbarc2d4b622009-03-09 21:49:58 +00003149
3150 const char *Section;
3151 if (ObjCABI == 1)
3152 Section = "__OBJC, __image_info,regular";
3153 else
3154 Section = "__DATA, __objc_imageinfo, regular, no_dead_strip";
Daniel Dunbar59e476b2009-08-03 17:06:42 +00003155 llvm::GlobalVariable *GV =
Daniel Dunbarc2d4b622009-03-09 21:49:58 +00003156 CreateMetadataVar("\01L_OBJC_IMAGE_INFO",
Owen Anderson47034e12009-07-28 18:33:04 +00003157 llvm::ConstantArray::get(AT, values, 2),
Daniel Dunbarc2d4b622009-03-09 21:49:58 +00003158 Section,
3159 0,
3160 true);
3161 GV->setConstant(true);
Daniel Dunbar3ad53482008-08-11 21:35:06 +00003162}
3163
Daniel Dunbar7bd00bd2008-08-12 06:48:42 +00003164
3165// struct objc_module {
3166// unsigned long version;
3167// unsigned long size;
3168// const char *name;
3169// Symtab symtab;
3170// };
3171
3172// FIXME: Get from somewhere
3173static const int ModuleVersion = 7;
3174
3175void CGObjCMac::EmitModuleInfo() {
Duncan Sandsc76fe8b2009-05-09 07:08:47 +00003176 uint64_t Size = CGM.getTargetData().getTypeAllocSize(ObjCTypes.ModuleTy);
Daniel Dunbar59e476b2009-08-03 17:06:42 +00003177
Daniel Dunbar7bd00bd2008-08-12 06:48:42 +00003178 std::vector<llvm::Constant*> Values(4);
Owen Andersonb7a2fe62009-07-24 23:12:58 +00003179 Values[0] = llvm::ConstantInt::get(ObjCTypes.LongTy, ModuleVersion);
3180 Values[1] = llvm::ConstantInt::get(ObjCTypes.LongTy, Size);
Daniel Dunbar92992502008-08-15 22:20:32 +00003181 // This used to be the filename, now it is unused. <rdr://4327263>
Daniel Dunbarb036db82008-08-13 03:21:16 +00003182 Values[2] = GetClassName(&CGM.getContext().Idents.get(""));
Daniel Dunbar7bd00bd2008-08-12 06:48:42 +00003183 Values[3] = EmitModuleSymbols();
Daniel Dunbar59e476b2009-08-03 17:06:42 +00003184 CreateMetadataVar("\01L_OBJC_MODULES",
Owen Anderson0e0189d2009-07-27 22:29:56 +00003185 llvm::ConstantStruct::get(ObjCTypes.ModuleTy, Values),
Daniel Dunbarc2d4b622009-03-09 21:49:58 +00003186 "__OBJC,__module_info,regular,no_dead_strip",
Daniel Dunbarae333842009-03-09 22:18:41 +00003187 4, true);
Daniel Dunbar7bd00bd2008-08-12 06:48:42 +00003188}
3189
3190llvm::Constant *CGObjCMac::EmitModuleSymbols() {
Daniel Dunbar22d82ed2008-08-21 04:36:09 +00003191 unsigned NumClasses = DefinedClasses.size();
3192 unsigned NumCategories = DefinedCategories.size();
3193
Daniel Dunbarc61d0e92008-08-25 06:02:07 +00003194 // Return null if no symbols were defined.
3195 if (!NumClasses && !NumCategories)
Owen Anderson0b75f232009-07-31 20:28:54 +00003196 return llvm::Constant::getNullValue(ObjCTypes.SymtabPtrTy);
Daniel Dunbarc61d0e92008-08-25 06:02:07 +00003197
3198 std::vector<llvm::Constant*> Values(5);
Owen Andersonb7a2fe62009-07-24 23:12:58 +00003199 Values[0] = llvm::ConstantInt::get(ObjCTypes.LongTy, 0);
Owen Anderson0b75f232009-07-31 20:28:54 +00003200 Values[1] = llvm::Constant::getNullValue(ObjCTypes.SelectorPtrTy);
Owen Andersonb7a2fe62009-07-24 23:12:58 +00003201 Values[2] = llvm::ConstantInt::get(ObjCTypes.ShortTy, NumClasses);
3202 Values[3] = llvm::ConstantInt::get(ObjCTypes.ShortTy, NumCategories);
Daniel Dunbar22d82ed2008-08-21 04:36:09 +00003203
Daniel Dunbar938a77f2008-08-22 20:34:54 +00003204 // The runtime expects exactly the list of defined classes followed
3205 // by the list of defined categories, in a single array.
Daniel Dunbar22d82ed2008-08-21 04:36:09 +00003206 std::vector<llvm::Constant*> Symbols(NumClasses + NumCategories);
Daniel Dunbar938a77f2008-08-22 20:34:54 +00003207 for (unsigned i=0; i<NumClasses; i++)
Owen Andersonade90fd2009-07-29 18:54:39 +00003208 Symbols[i] = llvm::ConstantExpr::getBitCast(DefinedClasses[i],
Daniel Dunbar938a77f2008-08-22 20:34:54 +00003209 ObjCTypes.Int8PtrTy);
3210 for (unsigned i=0; i<NumCategories; i++)
Daniel Dunbar59e476b2009-08-03 17:06:42 +00003211 Symbols[NumClasses + i] =
Owen Andersonade90fd2009-07-29 18:54:39 +00003212 llvm::ConstantExpr::getBitCast(DefinedCategories[i],
Daniel Dunbar938a77f2008-08-22 20:34:54 +00003213 ObjCTypes.Int8PtrTy);
3214
Daniel Dunbar59e476b2009-08-03 17:06:42 +00003215 Values[4] =
Owen Anderson9793f0e2009-07-29 22:16:19 +00003216 llvm::ConstantArray::get(llvm::ArrayType::get(ObjCTypes.Int8PtrTy,
Daniel Dunbar22d82ed2008-08-21 04:36:09 +00003217 NumClasses + NumCategories),
3218 Symbols);
3219
Nick Lewycky41eaf0a2009-09-19 20:00:52 +00003220 llvm::Constant *Init = llvm::ConstantStruct::get(VMContext, Values, false);
Daniel Dunbar22d82ed2008-08-21 04:36:09 +00003221
Daniel Dunbar7bd00bd2008-08-12 06:48:42 +00003222 llvm::GlobalVariable *GV =
Daniel Dunbarc2d4b622009-03-09 21:49:58 +00003223 CreateMetadataVar("\01L_OBJC_SYMBOLS", Init,
3224 "__OBJC,__symbols,regular,no_dead_strip",
Daniel Dunbarb25452a2009-04-15 02:56:18 +00003225 4, true);
Owen Andersonade90fd2009-07-29 18:54:39 +00003226 return llvm::ConstantExpr::getBitCast(GV, ObjCTypes.SymtabPtrTy);
Daniel Dunbar22d82ed2008-08-21 04:36:09 +00003227}
3228
Daniel Dunbar59e476b2009-08-03 17:06:42 +00003229llvm::Value *CGObjCMac::EmitClassRef(CGBuilderTy &Builder,
Daniel Dunbar22d82ed2008-08-21 04:36:09 +00003230 const ObjCInterfaceDecl *ID) {
Daniel Dunbarc61d0e92008-08-25 06:02:07 +00003231 LazySymbols.insert(ID->getIdentifier());
3232
Daniel Dunbar22d82ed2008-08-21 04:36:09 +00003233 llvm::GlobalVariable *&Entry = ClassReferences[ID->getIdentifier()];
Daniel Dunbar59e476b2009-08-03 17:06:42 +00003234
Daniel Dunbar22d82ed2008-08-21 04:36:09 +00003235 if (!Entry) {
Daniel Dunbar59e476b2009-08-03 17:06:42 +00003236 llvm::Constant *Casted =
Owen Andersonade90fd2009-07-29 18:54:39 +00003237 llvm::ConstantExpr::getBitCast(GetClassName(ID->getIdentifier()),
Daniel Dunbar22d82ed2008-08-21 04:36:09 +00003238 ObjCTypes.ClassPtrTy);
Daniel Dunbar59e476b2009-08-03 17:06:42 +00003239 Entry =
Daniel Dunbarc2d4b622009-03-09 21:49:58 +00003240 CreateMetadataVar("\01L_OBJC_CLASS_REFERENCES_", Casted,
3241 "__OBJC,__cls_refs,literal_pointers,no_dead_strip",
Daniel Dunbarb25452a2009-04-15 02:56:18 +00003242 4, true);
Daniel Dunbar22d82ed2008-08-21 04:36:09 +00003243 }
3244
Daniel Dunbarc76493a2009-11-29 21:23:36 +00003245 return Builder.CreateLoad(Entry, "tmp");
Daniel Dunbar7bd00bd2008-08-12 06:48:42 +00003246}
3247
Fariborz Jahanian9240f3d2010-06-17 19:56:20 +00003248llvm::Value *CGObjCMac::EmitSelector(CGBuilderTy &Builder, Selector Sel,
3249 bool lvalue) {
Daniel Dunbarcb515c82008-08-12 03:39:23 +00003250 llvm::GlobalVariable *&Entry = SelectorReferences[Sel];
Daniel Dunbar59e476b2009-08-03 17:06:42 +00003251
Daniel Dunbarcb515c82008-08-12 03:39:23 +00003252 if (!Entry) {
Daniel Dunbar59e476b2009-08-03 17:06:42 +00003253 llvm::Constant *Casted =
Owen Andersonade90fd2009-07-29 18:54:39 +00003254 llvm::ConstantExpr::getBitCast(GetMethodVarName(Sel),
Daniel Dunbarcb515c82008-08-12 03:39:23 +00003255 ObjCTypes.SelectorPtrTy);
Daniel Dunbar59e476b2009-08-03 17:06:42 +00003256 Entry =
Daniel Dunbarc2d4b622009-03-09 21:49:58 +00003257 CreateMetadataVar("\01L_OBJC_SELECTOR_REFERENCES_", Casted,
3258 "__OBJC,__message_refs,literal_pointers,no_dead_strip",
Daniel Dunbarb25452a2009-04-15 02:56:18 +00003259 4, true);
Daniel Dunbarcb515c82008-08-12 03:39:23 +00003260 }
3261
Fariborz Jahanian9240f3d2010-06-17 19:56:20 +00003262 if (lvalue)
3263 return Entry;
Daniel Dunbarc76493a2009-11-29 21:23:36 +00003264 return Builder.CreateLoad(Entry, "tmp");
Daniel Dunbarcb515c82008-08-12 03:39:23 +00003265}
3266
Fariborz Jahanian9e3ad522009-01-24 20:21:50 +00003267llvm::Constant *CGObjCCommonMac::GetClassName(IdentifierInfo *Ident) {
Daniel Dunbarb036db82008-08-13 03:21:16 +00003268 llvm::GlobalVariable *&Entry = ClassNames[Ident];
Daniel Dunbar7bd00bd2008-08-12 06:48:42 +00003269
Daniel Dunbarc2d4b622009-03-09 21:49:58 +00003270 if (!Entry)
Daniel Dunbar59e476b2009-08-03 17:06:42 +00003271 Entry = CreateMetadataVar("\01L_OBJC_CLASS_NAME_",
Daniel Dunbar349e6fb2009-10-18 20:48:59 +00003272 llvm::ConstantArray::get(VMContext,
3273 Ident->getNameStart()),
Jim Grosbach71d963e2010-07-20 16:20:26 +00003274 ((ObjCABI == 2) ?
3275 "__TEXT,__objc_classname,cstring_literals" :
3276 "__TEXT,__cstring,cstring_literals"),
Daniel Dunbar3241fae2009-04-14 23:14:47 +00003277 1, true);
Daniel Dunbar7bd00bd2008-08-12 06:48:42 +00003278
Owen Anderson170229f2009-07-14 23:10:40 +00003279 return getConstantGEP(VMContext, Entry, 0, 0);
Daniel Dunbar7bd00bd2008-08-12 06:48:42 +00003280}
3281
Fariborz Jahanian01dff422009-03-05 19:17:31 +00003282/// GetIvarLayoutName - Returns a unique constant for the given
3283/// ivar layout bitmap.
3284llvm::Constant *CGObjCCommonMac::GetIvarLayoutName(IdentifierInfo *Ident,
Daniel Dunbar59e476b2009-08-03 17:06:42 +00003285 const ObjCCommonTypesHelper &ObjCTypes) {
Owen Anderson0b75f232009-07-31 20:28:54 +00003286 return llvm::Constant::getNullValue(ObjCTypes.Int8PtrTy);
Fariborz Jahanian01dff422009-03-05 19:17:31 +00003287}
3288
John McCall8ccfcb52009-09-24 19:53:00 +00003289static Qualifiers::GC GetGCAttrTypeForType(ASTContext &Ctx, QualType FQT) {
Daniel Dunbar7abf83c2009-05-03 13:55:09 +00003290 if (FQT.isObjCGCStrong())
John McCall8ccfcb52009-09-24 19:53:00 +00003291 return Qualifiers::Strong;
Daniel Dunbar7abf83c2009-05-03 13:55:09 +00003292
3293 if (FQT.isObjCGCWeak())
John McCall8ccfcb52009-09-24 19:53:00 +00003294 return Qualifiers::Weak;
Daniel Dunbar7abf83c2009-05-03 13:55:09 +00003295
Fariborz Jahanian10e9bff2009-09-11 17:39:05 +00003296 if (FQT->isObjCObjectPointerType() || FQT->isBlockPointerType())
John McCall8ccfcb52009-09-24 19:53:00 +00003297 return Qualifiers::Strong;
Daniel Dunbar7abf83c2009-05-03 13:55:09 +00003298
Ted Kremenekc23c7e62009-07-29 21:53:49 +00003299 if (const PointerType *PT = FQT->getAs<PointerType>())
Daniel Dunbar15bd8882009-05-03 14:10:34 +00003300 return GetGCAttrTypeForType(Ctx, PT->getPointeeType());
Daniel Dunbar7abf83c2009-05-03 13:55:09 +00003301
John McCall8ccfcb52009-09-24 19:53:00 +00003302 return Qualifiers::GCNone;
Daniel Dunbar7abf83c2009-05-03 13:55:09 +00003303}
3304
Daniel Dunbar15bd8882009-05-03 14:10:34 +00003305void CGObjCCommonMac::BuildAggrIvarRecordLayout(const RecordType *RT,
Daniel Dunbar59e476b2009-08-03 17:06:42 +00003306 unsigned int BytePos,
Daniel Dunbar15bd8882009-05-03 14:10:34 +00003307 bool ForStrongLayout,
3308 bool &HasUnion) {
3309 const RecordDecl *RD = RT->getDecl();
3310 // FIXME - Use iterator.
Argyrios Kyrtzidiscfbfe782009-06-30 02:36:12 +00003311 llvm::SmallVector<FieldDecl*, 16> Fields(RD->field_begin(), RD->field_end());
Daniel Dunbar15bd8882009-05-03 14:10:34 +00003312 const llvm::Type *Ty = CGM.getTypes().ConvertType(QualType(RT, 0));
Daniel Dunbar59e476b2009-08-03 17:06:42 +00003313 const llvm::StructLayout *RecLayout =
Daniel Dunbar15bd8882009-05-03 14:10:34 +00003314 CGM.getTargetData().getStructLayout(cast<llvm::StructType>(Ty));
Daniel Dunbar59e476b2009-08-03 17:06:42 +00003315
Daniel Dunbar15bd8882009-05-03 14:10:34 +00003316 BuildAggrIvarLayout(0, RecLayout, RD, Fields, BytePos,
3317 ForStrongLayout, HasUnion);
3318}
3319
Daniel Dunbar36e2a1e2009-05-03 21:05:10 +00003320void CGObjCCommonMac::BuildAggrIvarLayout(const ObjCImplementationDecl *OI,
Daniel Dunbar59e476b2009-08-03 17:06:42 +00003321 const llvm::StructLayout *Layout,
3322 const RecordDecl *RD,
Chris Lattner5b36ddb2009-03-31 08:48:01 +00003323 const llvm::SmallVectorImpl<FieldDecl*> &RecFields,
Daniel Dunbar59e476b2009-08-03 17:06:42 +00003324 unsigned int BytePos, bool ForStrongLayout,
3325 bool &HasUnion) {
Fariborz Jahanian3b0f8862009-03-11 00:07:04 +00003326 bool IsUnion = (RD && RD->isUnion());
3327 uint64_t MaxUnionIvarSize = 0;
3328 uint64_t MaxSkippedUnionIvarSize = 0;
3329 FieldDecl *MaxField = 0;
3330 FieldDecl *MaxSkippedField = 0;
Fariborz Jahanianf5fec022009-04-21 18:33:06 +00003331 FieldDecl *LastFieldBitfield = 0;
Daniel Dunbar5b743912009-05-03 23:31:46 +00003332 uint64_t MaxFieldOffset = 0;
3333 uint64_t MaxSkippedFieldOffset = 0;
3334 uint64_t LastBitfieldOffset = 0;
Daniel Dunbar59e476b2009-08-03 17:06:42 +00003335
Fariborz Jahanian524bb202009-03-10 16:22:08 +00003336 if (RecFields.empty())
3337 return;
Chris Lattner5b36ddb2009-03-31 08:48:01 +00003338 unsigned WordSizeInBits = CGM.getContext().Target.getPointerWidth(0);
3339 unsigned ByteSizeInBits = CGM.getContext().Target.getCharWidth();
3340
Chris Lattner5b36ddb2009-03-31 08:48:01 +00003341 for (unsigned i = 0, e = RecFields.size(); i != e; ++i) {
Fariborz Jahanian524bb202009-03-10 16:22:08 +00003342 FieldDecl *Field = RecFields[i];
Daniel Dunbar62b10702009-05-03 23:35:23 +00003343 uint64_t FieldOffset;
Anders Carlssone2c6baf2009-07-24 17:23:54 +00003344 if (RD) {
Daniel Dunbard51c1a62010-04-14 17:02:21 +00003345 // Note that 'i' here is actually the field index inside RD of Field,
3346 // although this dependency is hidden.
3347 const ASTRecordLayout &RL = CGM.getContext().getASTRecordLayout(RD);
3348 FieldOffset = RL.getFieldOffset(i) / 8;
Anders Carlssone2c6baf2009-07-24 17:23:54 +00003349 } else
Daniel Dunbar62b10702009-05-03 23:35:23 +00003350 FieldOffset = ComputeIvarBaseOffset(CGM, OI, cast<ObjCIvarDecl>(Field));
Daniel Dunbar94f46dc2009-05-03 14:17:18 +00003351
Fariborz Jahanian524bb202009-03-10 16:22:08 +00003352 // Skip over unnamed or bitfields
Fariborz Jahanianf5fec022009-04-21 18:33:06 +00003353 if (!Field->getIdentifier() || Field->isBitField()) {
3354 LastFieldBitfield = Field;
Daniel Dunbar5b743912009-05-03 23:31:46 +00003355 LastBitfieldOffset = FieldOffset;
Fariborz Jahanian524bb202009-03-10 16:22:08 +00003356 continue;
Fariborz Jahanianf5fec022009-04-21 18:33:06 +00003357 }
Daniel Dunbar94f46dc2009-05-03 14:17:18 +00003358
Fariborz Jahanianf5fec022009-04-21 18:33:06 +00003359 LastFieldBitfield = 0;
Fariborz Jahanian524bb202009-03-10 16:22:08 +00003360 QualType FQT = Field->getType();
Fariborz Jahanianf909f922009-03-25 22:36:49 +00003361 if (FQT->isRecordType() || FQT->isUnionType()) {
Fariborz Jahanian524bb202009-03-10 16:22:08 +00003362 if (FQT->isUnionType())
3363 HasUnion = true;
Fariborz Jahanian3b0f8862009-03-11 00:07:04 +00003364
Daniel Dunbar59e476b2009-08-03 17:06:42 +00003365 BuildAggrIvarRecordLayout(FQT->getAs<RecordType>(),
Daniel Dunbar94f46dc2009-05-03 14:17:18 +00003366 BytePos + FieldOffset,
Daniel Dunbar15bd8882009-05-03 14:10:34 +00003367 ForStrongLayout, HasUnion);
Fariborz Jahanian524bb202009-03-10 16:22:08 +00003368 continue;
3369 }
Daniel Dunbar59e476b2009-08-03 17:06:42 +00003370
Chris Lattner5b36ddb2009-03-31 08:48:01 +00003371 if (const ArrayType *Array = CGM.getContext().getAsArrayType(FQT)) {
Daniel Dunbar59e476b2009-08-03 17:06:42 +00003372 const ConstantArrayType *CArray =
Daniel Dunbar7abf83c2009-05-03 13:55:09 +00003373 dyn_cast_or_null<ConstantArrayType>(Array);
Fariborz Jahanianf5fec022009-04-21 18:33:06 +00003374 uint64_t ElCount = CArray->getSize().getZExtValue();
Daniel Dunbar7abf83c2009-05-03 13:55:09 +00003375 assert(CArray && "only array with known element size is supported");
Fariborz Jahanian3b0f8862009-03-11 00:07:04 +00003376 FQT = CArray->getElementType();
Fariborz Jahanianf909f922009-03-25 22:36:49 +00003377 while (const ArrayType *Array = CGM.getContext().getAsArrayType(FQT)) {
3378 const ConstantArrayType *CArray =
Daniel Dunbar7abf83c2009-05-03 13:55:09 +00003379 dyn_cast_or_null<ConstantArrayType>(Array);
Fariborz Jahanianf5fec022009-04-21 18:33:06 +00003380 ElCount *= CArray->getSize().getZExtValue();
Fariborz Jahanianf909f922009-03-25 22:36:49 +00003381 FQT = CArray->getElementType();
3382 }
Daniel Dunbar59e476b2009-08-03 17:06:42 +00003383
3384 assert(!FQT->isUnionType() &&
Fariborz Jahanian3b0f8862009-03-11 00:07:04 +00003385 "layout for array of unions not supported");
3386 if (FQT->isRecordType()) {
Fariborz Jahaniana123b642009-04-24 16:17:09 +00003387 int OldIndex = IvarsInfo.size() - 1;
3388 int OldSkIndex = SkipIvars.size() -1;
Daniel Dunbar59e476b2009-08-03 17:06:42 +00003389
Ted Kremenekc23c7e62009-07-29 21:53:49 +00003390 const RecordType *RT = FQT->getAs<RecordType>();
Daniel Dunbar94f46dc2009-05-03 14:17:18 +00003391 BuildAggrIvarRecordLayout(RT, BytePos + FieldOffset,
Daniel Dunbar15bd8882009-05-03 14:10:34 +00003392 ForStrongLayout, HasUnion);
Daniel Dunbar59e476b2009-08-03 17:06:42 +00003393
Fariborz Jahanian3b0f8862009-03-11 00:07:04 +00003394 // Replicate layout information for each array element. Note that
3395 // one element is already done.
3396 uint64_t ElIx = 1;
Daniel Dunbar59e476b2009-08-03 17:06:42 +00003397 for (int FirstIndex = IvarsInfo.size() - 1,
3398 FirstSkIndex = SkipIvars.size() - 1 ;ElIx < ElCount; ElIx++) {
Fariborz Jahanian1bf72882009-03-12 22:50:49 +00003399 uint64_t Size = CGM.getContext().getTypeSize(RT)/ByteSizeInBits;
Daniel Dunbar7b89ace2009-05-03 13:44:42 +00003400 for (int i = OldIndex+1; i <= FirstIndex; ++i)
3401 IvarsInfo.push_back(GC_IVAR(IvarsInfo[i].ivar_bytepos + Size*ElIx,
3402 IvarsInfo[i].ivar_size));
3403 for (int i = OldSkIndex+1; i <= FirstSkIndex; ++i)
3404 SkipIvars.push_back(GC_IVAR(SkipIvars[i].ivar_bytepos + Size*ElIx,
3405 SkipIvars[i].ivar_size));
Fariborz Jahanian3b0f8862009-03-11 00:07:04 +00003406 }
3407 continue;
3408 }
Fariborz Jahanian524bb202009-03-10 16:22:08 +00003409 }
Fariborz Jahanian3b0f8862009-03-11 00:07:04 +00003410 // At this point, we are done with Record/Union and array there of.
3411 // For other arrays we are down to its element type.
John McCall8ccfcb52009-09-24 19:53:00 +00003412 Qualifiers::GC GCAttr = GetGCAttrTypeForType(CGM.getContext(), FQT);
Daniel Dunbar7abf83c2009-05-03 13:55:09 +00003413
Daniel Dunbar7b89ace2009-05-03 13:44:42 +00003414 unsigned FieldSize = CGM.getContext().getTypeSize(Field->getType());
John McCall8ccfcb52009-09-24 19:53:00 +00003415 if ((ForStrongLayout && GCAttr == Qualifiers::Strong)
3416 || (!ForStrongLayout && GCAttr == Qualifiers::Weak)) {
Daniel Dunbar22007d32009-05-03 13:32:01 +00003417 if (IsUnion) {
Daniel Dunbar7b89ace2009-05-03 13:44:42 +00003418 uint64_t UnionIvarSize = FieldSize / WordSizeInBits;
Daniel Dunbar22007d32009-05-03 13:32:01 +00003419 if (UnionIvarSize > MaxUnionIvarSize) {
Fariborz Jahanian3b0f8862009-03-11 00:07:04 +00003420 MaxUnionIvarSize = UnionIvarSize;
3421 MaxField = Field;
Daniel Dunbar5b743912009-05-03 23:31:46 +00003422 MaxFieldOffset = FieldOffset;
Fariborz Jahanian3b0f8862009-03-11 00:07:04 +00003423 }
Daniel Dunbar22007d32009-05-03 13:32:01 +00003424 } else {
Daniel Dunbar94f46dc2009-05-03 14:17:18 +00003425 IvarsInfo.push_back(GC_IVAR(BytePos + FieldOffset,
Daniel Dunbar7b89ace2009-05-03 13:44:42 +00003426 FieldSize / WordSizeInBits));
Fariborz Jahanian3b0f8862009-03-11 00:07:04 +00003427 }
Daniel Dunbar59e476b2009-08-03 17:06:42 +00003428 } else if ((ForStrongLayout &&
John McCall8ccfcb52009-09-24 19:53:00 +00003429 (GCAttr == Qualifiers::GCNone || GCAttr == Qualifiers::Weak))
3430 || (!ForStrongLayout && GCAttr != Qualifiers::Weak)) {
Daniel Dunbar22007d32009-05-03 13:32:01 +00003431 if (IsUnion) {
Mike Stump18bb9282009-05-16 07:57:57 +00003432 // FIXME: Why the asymmetry? We divide by word size in bits on other
3433 // side.
Daniel Dunbar7b89ace2009-05-03 13:44:42 +00003434 uint64_t UnionIvarSize = FieldSize;
Daniel Dunbar22007d32009-05-03 13:32:01 +00003435 if (UnionIvarSize > MaxSkippedUnionIvarSize) {
Fariborz Jahanian3b0f8862009-03-11 00:07:04 +00003436 MaxSkippedUnionIvarSize = UnionIvarSize;
3437 MaxSkippedField = Field;
Daniel Dunbar5b743912009-05-03 23:31:46 +00003438 MaxSkippedFieldOffset = FieldOffset;
Fariborz Jahanian3b0f8862009-03-11 00:07:04 +00003439 }
Daniel Dunbar22007d32009-05-03 13:32:01 +00003440 } else {
Daniel Dunbar7b89ace2009-05-03 13:44:42 +00003441 // FIXME: Why the asymmetry, we divide by byte size in bits here?
Daniel Dunbar94f46dc2009-05-03 14:17:18 +00003442 SkipIvars.push_back(GC_IVAR(BytePos + FieldOffset,
Daniel Dunbar7b89ace2009-05-03 13:44:42 +00003443 FieldSize / ByteSizeInBits));
Fariborz Jahanian3b0f8862009-03-11 00:07:04 +00003444 }
3445 }
3446 }
Daniel Dunbar15bd8882009-05-03 14:10:34 +00003447
Fariborz Jahanianf5fec022009-04-21 18:33:06 +00003448 if (LastFieldBitfield) {
3449 // Last field was a bitfield. Must update skip info.
Fariborz Jahanianf5fec022009-04-21 18:33:06 +00003450 Expr *BitWidth = LastFieldBitfield->getBitWidth();
3451 uint64_t BitFieldSize =
Eli Friedman1c4a1752009-04-26 19:19:15 +00003452 BitWidth->EvaluateAsInt(CGM.getContext()).getZExtValue();
Daniel Dunbar22007d32009-05-03 13:32:01 +00003453 GC_IVAR skivar;
Daniel Dunbar5b743912009-05-03 23:31:46 +00003454 skivar.ivar_bytepos = BytePos + LastBitfieldOffset;
Daniel Dunbar59e476b2009-08-03 17:06:42 +00003455 skivar.ivar_size = (BitFieldSize / ByteSizeInBits)
3456 + ((BitFieldSize % ByteSizeInBits) != 0);
3457 SkipIvars.push_back(skivar);
Fariborz Jahanianf5fec022009-04-21 18:33:06 +00003458 }
Daniel Dunbar59e476b2009-08-03 17:06:42 +00003459
Daniel Dunbar7b89ace2009-05-03 13:44:42 +00003460 if (MaxField)
Daniel Dunbar59e476b2009-08-03 17:06:42 +00003461 IvarsInfo.push_back(GC_IVAR(BytePos + MaxFieldOffset,
Daniel Dunbar7b89ace2009-05-03 13:44:42 +00003462 MaxUnionIvarSize));
3463 if (MaxSkippedField)
Daniel Dunbar5b743912009-05-03 23:31:46 +00003464 SkipIvars.push_back(GC_IVAR(BytePos + MaxSkippedFieldOffset,
Daniel Dunbar7b89ace2009-05-03 13:44:42 +00003465 MaxSkippedUnionIvarSize));
Fariborz Jahanianc559f3f2009-03-05 22:39:55 +00003466}
3467
3468/// BuildIvarLayout - Builds ivar layout bitmap for the class
3469/// implementation for the __strong or __weak case.
Daniel Dunbar59e476b2009-08-03 17:06:42 +00003470/// The layout map displays which words in ivar list must be skipped
3471/// and which must be scanned by GC (see below). String is built of bytes.
3472/// Each byte is divided up in two nibbles (4-bit each). Left nibble is count
Fariborz Jahanianc559f3f2009-03-05 22:39:55 +00003473/// of words to skip and right nibble is count of words to scan. So, each
Daniel Dunbar59e476b2009-08-03 17:06:42 +00003474/// nibble represents up to 15 workds to skip or scan. Skipping the rest is
Fariborz Jahanianc559f3f2009-03-05 22:39:55 +00003475/// represented by a 0x00 byte which also ends the string.
3476/// 1. when ForStrongLayout is true, following ivars are scanned:
3477/// - id, Class
3478/// - object *
3479/// - __strong anything
Daniel Dunbar59e476b2009-08-03 17:06:42 +00003480///
Fariborz Jahanianc559f3f2009-03-05 22:39:55 +00003481/// 2. When ForStrongLayout is false, following ivars are scanned:
3482/// - __weak anything
3483///
Fariborz Jahanian524bb202009-03-10 16:22:08 +00003484llvm::Constant *CGObjCCommonMac::BuildIvarLayout(
Daniel Dunbar59e476b2009-08-03 17:06:42 +00003485 const ObjCImplementationDecl *OMD,
3486 bool ForStrongLayout) {
Fariborz Jahanianc559f3f2009-03-05 22:39:55 +00003487 bool hasUnion = false;
Daniel Dunbar59e476b2009-08-03 17:06:42 +00003488
Fariborz Jahaniancbaf73c2009-03-11 20:59:05 +00003489 unsigned int WordsToScan, WordsToSkip;
Benjamin Kramerabd5b902009-10-13 10:07:13 +00003490 const llvm::Type *PtrTy = llvm::Type::getInt8PtrTy(VMContext);
Fariborz Jahanian1bf72882009-03-12 22:50:49 +00003491 if (CGM.getLangOptions().getGCMode() == LangOptions::NonGC)
Owen Anderson0b75f232009-07-31 20:28:54 +00003492 return llvm::Constant::getNullValue(PtrTy);
Daniel Dunbar59e476b2009-08-03 17:06:42 +00003493
Chris Lattner5b36ddb2009-03-31 08:48:01 +00003494 llvm::SmallVector<FieldDecl*, 32> RecFields;
Fariborz Jahanian1bf72882009-03-12 22:50:49 +00003495 const ObjCInterfaceDecl *OI = OMD->getClassInterface();
Fariborz Jahanianc559f3f2009-03-05 22:39:55 +00003496 CGM.getContext().CollectObjCIvars(OI, RecFields);
Daniel Dunbar59e476b2009-08-03 17:06:42 +00003497
Daniel Dunbar98ba9642009-05-04 04:10:48 +00003498 // Add this implementations synthesized ivars.
Fariborz Jahanian0f44d812009-05-12 18:14:29 +00003499 llvm::SmallVector<ObjCIvarDecl*, 16> Ivars;
Fariborz Jahanianaef66222010-02-19 00:31:17 +00003500 CGM.getContext().CollectNonClassIvars(OI, Ivars);
Fariborz Jahanian0f44d812009-05-12 18:14:29 +00003501 for (unsigned k = 0, e = Ivars.size(); k != e; ++k)
3502 RecFields.push_back(cast<FieldDecl>(Ivars[k]));
Daniel Dunbar59e476b2009-08-03 17:06:42 +00003503
Fariborz Jahanianc559f3f2009-03-05 22:39:55 +00003504 if (RecFields.empty())
Owen Anderson0b75f232009-07-31 20:28:54 +00003505 return llvm::Constant::getNullValue(PtrTy);
Daniel Dunbar59e476b2009-08-03 17:06:42 +00003506
3507 SkipIvars.clear();
Fariborz Jahanian1bf72882009-03-12 22:50:49 +00003508 IvarsInfo.clear();
Daniel Dunbar59e476b2009-08-03 17:06:42 +00003509
Daniel Dunbar36e2a1e2009-05-03 21:05:10 +00003510 BuildAggrIvarLayout(OMD, 0, 0, RecFields, 0, ForStrongLayout, hasUnion);
Fariborz Jahaniana123b642009-04-24 16:17:09 +00003511 if (IvarsInfo.empty())
Owen Anderson0b75f232009-07-31 20:28:54 +00003512 return llvm::Constant::getNullValue(PtrTy);
Daniel Dunbar59e476b2009-08-03 17:06:42 +00003513
Fariborz Jahaniancbaf73c2009-03-11 20:59:05 +00003514 // Sort on byte position in case we encounterred a union nested in
3515 // the ivar list.
Fariborz Jahanian1bf72882009-03-12 22:50:49 +00003516 if (hasUnion && !IvarsInfo.empty())
Daniel Dunbar22b0ada2009-04-23 01:29:05 +00003517 std::sort(IvarsInfo.begin(), IvarsInfo.end());
Fariborz Jahanian1bf72882009-03-12 22:50:49 +00003518 if (hasUnion && !SkipIvars.empty())
Daniel Dunbar22b0ada2009-04-23 01:29:05 +00003519 std::sort(SkipIvars.begin(), SkipIvars.end());
Daniel Dunbar59e476b2009-08-03 17:06:42 +00003520
Fariborz Jahaniancbaf73c2009-03-11 20:59:05 +00003521 // Build the string of skip/scan nibbles
Fariborz Jahaniance5676572009-04-24 17:15:27 +00003522 llvm::SmallVector<SKIP_SCAN, 32> SkipScanIvars;
Daniel Dunbar59e476b2009-08-03 17:06:42 +00003523 unsigned int WordSize =
Duncan Sandsc76fe8b2009-05-09 07:08:47 +00003524 CGM.getTypes().getTargetData().getTypeAllocSize(PtrTy);
Fariborz Jahaniancbaf73c2009-03-11 20:59:05 +00003525 if (IvarsInfo[0].ivar_bytepos == 0) {
3526 WordsToSkip = 0;
3527 WordsToScan = IvarsInfo[0].ivar_size;
Daniel Dunbard22aa4a2009-05-03 23:21:22 +00003528 } else {
Fariborz Jahaniancbaf73c2009-03-11 20:59:05 +00003529 WordsToSkip = IvarsInfo[0].ivar_bytepos/WordSize;
3530 WordsToScan = IvarsInfo[0].ivar_size;
3531 }
Daniel Dunbard22aa4a2009-05-03 23:21:22 +00003532 for (unsigned int i=1, Last=IvarsInfo.size(); i != Last; i++) {
Daniel Dunbar59e476b2009-08-03 17:06:42 +00003533 unsigned int TailPrevGCObjC =
Fariborz Jahaniancbaf73c2009-03-11 20:59:05 +00003534 IvarsInfo[i-1].ivar_bytepos + IvarsInfo[i-1].ivar_size * WordSize;
Daniel Dunbard22aa4a2009-05-03 23:21:22 +00003535 if (IvarsInfo[i].ivar_bytepos == TailPrevGCObjC) {
Fariborz Jahaniancbaf73c2009-03-11 20:59:05 +00003536 // consecutive 'scanned' object pointers.
3537 WordsToScan += IvarsInfo[i].ivar_size;
Daniel Dunbard22aa4a2009-05-03 23:21:22 +00003538 } else {
Fariborz Jahaniancbaf73c2009-03-11 20:59:05 +00003539 // Skip over 'gc'able object pointer which lay over each other.
3540 if (TailPrevGCObjC > IvarsInfo[i].ivar_bytepos)
3541 continue;
3542 // Must skip over 1 or more words. We save current skip/scan values
3543 // and start a new pair.
Fariborz Jahanian1bf72882009-03-12 22:50:49 +00003544 SKIP_SCAN SkScan;
3545 SkScan.skip = WordsToSkip;
3546 SkScan.scan = WordsToScan;
Fariborz Jahaniana123b642009-04-24 16:17:09 +00003547 SkipScanIvars.push_back(SkScan);
Daniel Dunbar59e476b2009-08-03 17:06:42 +00003548
Fariborz Jahaniancbaf73c2009-03-11 20:59:05 +00003549 // Skip the hole.
Fariborz Jahanian1bf72882009-03-12 22:50:49 +00003550 SkScan.skip = (IvarsInfo[i].ivar_bytepos - TailPrevGCObjC) / WordSize;
3551 SkScan.scan = 0;
Fariborz Jahaniana123b642009-04-24 16:17:09 +00003552 SkipScanIvars.push_back(SkScan);
Fariborz Jahaniancbaf73c2009-03-11 20:59:05 +00003553 WordsToSkip = 0;
3554 WordsToScan = IvarsInfo[i].ivar_size;
3555 }
3556 }
Daniel Dunbard22aa4a2009-05-03 23:21:22 +00003557 if (WordsToScan > 0) {
Fariborz Jahanian1bf72882009-03-12 22:50:49 +00003558 SKIP_SCAN SkScan;
3559 SkScan.skip = WordsToSkip;
3560 SkScan.scan = WordsToScan;
Fariborz Jahaniana123b642009-04-24 16:17:09 +00003561 SkipScanIvars.push_back(SkScan);
Fariborz Jahaniancbaf73c2009-03-11 20:59:05 +00003562 }
Daniel Dunbar59e476b2009-08-03 17:06:42 +00003563
Daniel Dunbard22aa4a2009-05-03 23:21:22 +00003564 if (!SkipIvars.empty()) {
Fariborz Jahaniana123b642009-04-24 16:17:09 +00003565 unsigned int LastIndex = SkipIvars.size()-1;
Daniel Dunbar59e476b2009-08-03 17:06:42 +00003566 int LastByteSkipped =
3567 SkipIvars[LastIndex].ivar_bytepos + SkipIvars[LastIndex].ivar_size;
Fariborz Jahaniana123b642009-04-24 16:17:09 +00003568 LastIndex = IvarsInfo.size()-1;
Daniel Dunbar59e476b2009-08-03 17:06:42 +00003569 int LastByteScanned =
3570 IvarsInfo[LastIndex].ivar_bytepos +
3571 IvarsInfo[LastIndex].ivar_size * WordSize;
Fariborz Jahaniancbaf73c2009-03-11 20:59:05 +00003572 // Compute number of bytes to skip at the tail end of the last ivar scanned.
Benjamin Kramerd20ef752009-12-25 15:43:36 +00003573 if (LastByteSkipped > LastByteScanned) {
Fariborz Jahaniancbaf73c2009-03-11 20:59:05 +00003574 unsigned int TotalWords = (LastByteSkipped + (WordSize -1)) / WordSize;
Fariborz Jahanian1bf72882009-03-12 22:50:49 +00003575 SKIP_SCAN SkScan;
3576 SkScan.skip = TotalWords - (LastByteScanned/WordSize);
3577 SkScan.scan = 0;
Fariborz Jahaniana123b642009-04-24 16:17:09 +00003578 SkipScanIvars.push_back(SkScan);
Fariborz Jahaniancbaf73c2009-03-11 20:59:05 +00003579 }
3580 }
3581 // Mini optimization of nibbles such that an 0xM0 followed by 0x0N is produced
3582 // as 0xMN.
Fariborz Jahaniana123b642009-04-24 16:17:09 +00003583 int SkipScan = SkipScanIvars.size()-1;
Daniel Dunbard22aa4a2009-05-03 23:21:22 +00003584 for (int i = 0; i <= SkipScan; i++) {
Fariborz Jahaniancbaf73c2009-03-11 20:59:05 +00003585 if ((i < SkipScan) && SkipScanIvars[i].skip && SkipScanIvars[i].scan == 0
3586 && SkipScanIvars[i+1].skip == 0 && SkipScanIvars[i+1].scan) {
3587 // 0xM0 followed by 0x0N detected.
3588 SkipScanIvars[i].scan = SkipScanIvars[i+1].scan;
3589 for (int j = i+1; j < SkipScan; j++)
3590 SkipScanIvars[j] = SkipScanIvars[j+1];
3591 --SkipScan;
3592 }
3593 }
Daniel Dunbar59e476b2009-08-03 17:06:42 +00003594
Fariborz Jahaniancbaf73c2009-03-11 20:59:05 +00003595 // Generate the string.
3596 std::string BitMap;
Daniel Dunbard22aa4a2009-05-03 23:21:22 +00003597 for (int i = 0; i <= SkipScan; i++) {
Fariborz Jahaniancbaf73c2009-03-11 20:59:05 +00003598 unsigned char byte;
3599 unsigned int skip_small = SkipScanIvars[i].skip % 0xf;
3600 unsigned int scan_small = SkipScanIvars[i].scan % 0xf;
3601 unsigned int skip_big = SkipScanIvars[i].skip / 0xf;
3602 unsigned int scan_big = SkipScanIvars[i].scan / 0xf;
3603
Fariborz Jahaniancbaf73c2009-03-11 20:59:05 +00003604 // first skip big.
3605 for (unsigned int ix = 0; ix < skip_big; ix++)
3606 BitMap += (unsigned char)(0xf0);
Daniel Dunbar59e476b2009-08-03 17:06:42 +00003607
Fariborz Jahaniancbaf73c2009-03-11 20:59:05 +00003608 // next (skip small, scan)
Daniel Dunbard22aa4a2009-05-03 23:21:22 +00003609 if (skip_small) {
Fariborz Jahaniancbaf73c2009-03-11 20:59:05 +00003610 byte = skip_small << 4;
Daniel Dunbard22aa4a2009-05-03 23:21:22 +00003611 if (scan_big > 0) {
Fariborz Jahaniancbaf73c2009-03-11 20:59:05 +00003612 byte |= 0xf;
3613 --scan_big;
Daniel Dunbard22aa4a2009-05-03 23:21:22 +00003614 } else if (scan_small) {
Fariborz Jahaniancbaf73c2009-03-11 20:59:05 +00003615 byte |= scan_small;
3616 scan_small = 0;
3617 }
3618 BitMap += byte;
3619 }
3620 // next scan big
3621 for (unsigned int ix = 0; ix < scan_big; ix++)
3622 BitMap += (unsigned char)(0x0f);
3623 // last scan small
Daniel Dunbard22aa4a2009-05-03 23:21:22 +00003624 if (scan_small) {
Fariborz Jahaniancbaf73c2009-03-11 20:59:05 +00003625 byte = scan_small;
3626 BitMap += byte;
3627 }
3628 }
3629 // null terminate string.
Fariborz Jahanianf909f922009-03-25 22:36:49 +00003630 unsigned char zero = 0;
3631 BitMap += zero;
Daniel Dunbar59e476b2009-08-03 17:06:42 +00003632
Fariborz Jahanian80c9ce22009-04-20 22:03:45 +00003633 if (CGM.getLangOptions().ObjCGCBitmapPrint) {
Daniel Dunbar59e476b2009-08-03 17:06:42 +00003634 printf("\n%s ivar layout for class '%s': ",
Fariborz Jahanian80c9ce22009-04-20 22:03:45 +00003635 ForStrongLayout ? "strong" : "weak",
3636 OMD->getClassInterface()->getNameAsCString());
3637 const unsigned char *s = (unsigned char*)BitMap.c_str();
3638 for (unsigned i = 0; i < BitMap.size(); i++)
3639 if (!(s[i] & 0xf0))
3640 printf("0x0%x%s", s[i], s[i] != 0 ? ", " : "");
3641 else
3642 printf("0x%x%s", s[i], s[i] != 0 ? ", " : "");
3643 printf("\n");
3644 }
Daniel Dunbar59e476b2009-08-03 17:06:42 +00003645 llvm::GlobalVariable * Entry =
3646 CreateMetadataVar("\01L_OBJC_CLASS_NAME_",
Owen Anderson41a75022009-08-13 21:57:51 +00003647 llvm::ConstantArray::get(VMContext, BitMap.c_str()),
Jim Grosbach71d963e2010-07-20 16:20:26 +00003648 ((ObjCABI == 2) ?
3649 "__TEXT,__objc_classname,cstring_literals" :
3650 "__TEXT,__cstring,cstring_literals"),
Daniel Dunbar59e476b2009-08-03 17:06:42 +00003651 1, true);
3652 return getConstantGEP(VMContext, Entry, 0, 0);
Fariborz Jahanianc559f3f2009-03-05 22:39:55 +00003653}
3654
Fariborz Jahanian0b1ccdc2009-01-21 23:34:32 +00003655llvm::Constant *CGObjCCommonMac::GetMethodVarName(Selector Sel) {
Daniel Dunbarcb515c82008-08-12 03:39:23 +00003656 llvm::GlobalVariable *&Entry = MethodVarNames[Sel];
3657
Daniel Dunbarc2d4b622009-03-09 21:49:58 +00003658 // FIXME: Avoid std::string copying.
3659 if (!Entry)
Daniel Dunbar59e476b2009-08-03 17:06:42 +00003660 Entry = CreateMetadataVar("\01L_OBJC_METH_VAR_NAME_",
Owen Anderson41a75022009-08-13 21:57:51 +00003661 llvm::ConstantArray::get(VMContext, Sel.getAsString()),
Jim Grosbach71d963e2010-07-20 16:20:26 +00003662 ((ObjCABI == 2) ?
3663 "__TEXT,__objc_methname,cstring_literals" :
3664 "__TEXT,__cstring,cstring_literals"),
Daniel Dunbar3241fae2009-04-14 23:14:47 +00003665 1, true);
Daniel Dunbarcb515c82008-08-12 03:39:23 +00003666
Owen Anderson170229f2009-07-14 23:10:40 +00003667 return getConstantGEP(VMContext, Entry, 0, 0);
Daniel Dunbarb036db82008-08-13 03:21:16 +00003668}
3669
Daniel Dunbar22d82ed2008-08-21 04:36:09 +00003670// FIXME: Merge into a single cstring creation function.
Fariborz Jahanian0b1ccdc2009-01-21 23:34:32 +00003671llvm::Constant *CGObjCCommonMac::GetMethodVarName(IdentifierInfo *ID) {
Daniel Dunbar22d82ed2008-08-21 04:36:09 +00003672 return GetMethodVarName(CGM.getContext().Selectors.getNullarySelector(ID));
3673}
3674
3675// FIXME: Merge into a single cstring creation function.
Fariborz Jahanian0b1ccdc2009-01-21 23:34:32 +00003676llvm::Constant *CGObjCCommonMac::GetMethodVarName(const std::string &Name) {
Daniel Dunbar22d82ed2008-08-21 04:36:09 +00003677 return GetMethodVarName(&CGM.getContext().Idents.get(Name));
3678}
3679
Daniel Dunbarf5c18462009-04-20 06:54:31 +00003680llvm::Constant *CGObjCCommonMac::GetMethodVarType(const FieldDecl *Field) {
Devang Patel4b6e4bb2009-03-04 18:21:39 +00003681 std::string TypeStr;
3682 CGM.getContext().getObjCEncodingForType(Field->getType(), TypeStr, Field);
3683
3684 llvm::GlobalVariable *&Entry = MethodVarTypes[TypeStr];
Daniel Dunbarb036db82008-08-13 03:21:16 +00003685
Daniel Dunbarc2d4b622009-03-09 21:49:58 +00003686 if (!Entry)
3687 Entry = CreateMetadataVar("\01L_OBJC_METH_VAR_TYPE_",
Owen Anderson41a75022009-08-13 21:57:51 +00003688 llvm::ConstantArray::get(VMContext, TypeStr),
Jim Grosbach71d963e2010-07-20 16:20:26 +00003689 ((ObjCABI == 2) ?
3690 "__TEXT,__objc_methtype,cstring_literals" :
3691 "__TEXT,__cstring,cstring_literals"),
Daniel Dunbar3241fae2009-04-14 23:14:47 +00003692 1, true);
Daniel Dunbar59e476b2009-08-03 17:06:42 +00003693
Owen Anderson170229f2009-07-14 23:10:40 +00003694 return getConstantGEP(VMContext, Entry, 0, 0);
Daniel Dunbarcb515c82008-08-12 03:39:23 +00003695}
3696
Fariborz Jahanian0b1ccdc2009-01-21 23:34:32 +00003697llvm::Constant *CGObjCCommonMac::GetMethodVarType(const ObjCMethodDecl *D) {
Daniel Dunbar22d82ed2008-08-21 04:36:09 +00003698 std::string TypeStr;
Daniel Dunbar3c76cb52008-08-26 21:51:14 +00003699 CGM.getContext().getObjCEncodingForMethodDecl(const_cast<ObjCMethodDecl*>(D),
3700 TypeStr);
Devang Patel4b6e4bb2009-03-04 18:21:39 +00003701
3702 llvm::GlobalVariable *&Entry = MethodVarTypes[TypeStr];
3703
Daniel Dunbar3241fae2009-04-14 23:14:47 +00003704 if (!Entry)
3705 Entry = CreateMetadataVar("\01L_OBJC_METH_VAR_TYPE_",
Owen Anderson41a75022009-08-13 21:57:51 +00003706 llvm::ConstantArray::get(VMContext, TypeStr),
Jim Grosbach71d963e2010-07-20 16:20:26 +00003707 ((ObjCABI == 2) ?
3708 "__TEXT,__objc_methtype,cstring_literals" :
3709 "__TEXT,__cstring,cstring_literals"),
Daniel Dunbar3241fae2009-04-14 23:14:47 +00003710 1, true);
Devang Patel4b6e4bb2009-03-04 18:21:39 +00003711
Owen Anderson170229f2009-07-14 23:10:40 +00003712 return getConstantGEP(VMContext, Entry, 0, 0);
Daniel Dunbar22d82ed2008-08-21 04:36:09 +00003713}
3714
Daniel Dunbar80a840b2008-08-23 00:19:03 +00003715// FIXME: Merge into a single cstring creation function.
Fariborz Jahanian0b1ccdc2009-01-21 23:34:32 +00003716llvm::Constant *CGObjCCommonMac::GetPropertyName(IdentifierInfo *Ident) {
Daniel Dunbar80a840b2008-08-23 00:19:03 +00003717 llvm::GlobalVariable *&Entry = PropertyNames[Ident];
Daniel Dunbar59e476b2009-08-03 17:06:42 +00003718
Daniel Dunbarc2d4b622009-03-09 21:49:58 +00003719 if (!Entry)
Daniel Dunbar59e476b2009-08-03 17:06:42 +00003720 Entry = CreateMetadataVar("\01L_OBJC_PROP_NAME_ATTR_",
Daniel Dunbar349e6fb2009-10-18 20:48:59 +00003721 llvm::ConstantArray::get(VMContext,
3722 Ident->getNameStart()),
Daniel Dunbarc2d4b622009-03-09 21:49:58 +00003723 "__TEXT,__cstring,cstring_literals",
Daniel Dunbar3241fae2009-04-14 23:14:47 +00003724 1, true);
Daniel Dunbar80a840b2008-08-23 00:19:03 +00003725
Owen Anderson170229f2009-07-14 23:10:40 +00003726 return getConstantGEP(VMContext, Entry, 0, 0);
Daniel Dunbar80a840b2008-08-23 00:19:03 +00003727}
3728
3729// FIXME: Merge into a single cstring creation function.
Daniel Dunbar4932b362008-08-28 04:38:10 +00003730// FIXME: This Decl should be more precise.
Daniel Dunbarc2d4b622009-03-09 21:49:58 +00003731llvm::Constant *
Daniel Dunbar59e476b2009-08-03 17:06:42 +00003732CGObjCCommonMac::GetPropertyTypeString(const ObjCPropertyDecl *PD,
3733 const Decl *Container) {
Daniel Dunbar4932b362008-08-28 04:38:10 +00003734 std::string TypeStr;
3735 CGM.getContext().getObjCEncodingForPropertyDecl(PD, Container, TypeStr);
Daniel Dunbar80a840b2008-08-23 00:19:03 +00003736 return GetPropertyName(&CGM.getContext().Idents.get(TypeStr));
3737}
3738
Daniel Dunbar59e476b2009-08-03 17:06:42 +00003739void CGObjCCommonMac::GetNameForMethod(const ObjCMethodDecl *D,
Fariborz Jahanian0b1ccdc2009-01-21 23:34:32 +00003740 const ObjCContainerDecl *CD,
Daniel Dunbard2386812009-10-19 01:21:19 +00003741 llvm::SmallVectorImpl<char> &Name) {
3742 llvm::raw_svector_ostream OS(Name);
Fariborz Jahanian0196a1c2009-01-10 21:06:09 +00003743 assert (CD && "Missing container decl in GetNameForMethod");
Daniel Dunbard2386812009-10-19 01:21:19 +00003744 OS << '\01' << (D->isInstanceMethod() ? '-' : '+')
3745 << '[' << CD->getName();
Daniel Dunbar59e476b2009-08-03 17:06:42 +00003746 if (const ObjCCategoryImplDecl *CID =
Daniel Dunbard2386812009-10-19 01:21:19 +00003747 dyn_cast<ObjCCategoryImplDecl>(D->getDeclContext()))
Benjamin Kramerb11416d2010-04-17 09:33:03 +00003748 OS << '(' << CID << ')';
Daniel Dunbard2386812009-10-19 01:21:19 +00003749 OS << ' ' << D->getSelector().getAsString() << ']';
Daniel Dunbara94ecd22008-08-16 03:19:19 +00003750}
3751
Daniel Dunbar3ad53482008-08-11 21:35:06 +00003752void CGObjCMac::FinishModule() {
Daniel Dunbar7bd00bd2008-08-12 06:48:42 +00003753 EmitModuleInfo();
3754
Daniel Dunbarc475d422008-10-29 22:36:39 +00003755 // Emit the dummy bodies for any protocols which were referenced but
3756 // never defined.
Daniel Dunbar59e476b2009-08-03 17:06:42 +00003757 for (llvm::DenseMap<IdentifierInfo*, llvm::GlobalVariable*>::iterator
Chris Lattnerf56501c2009-07-17 23:57:13 +00003758 I = Protocols.begin(), e = Protocols.end(); I != e; ++I) {
3759 if (I->second->hasInitializer())
Daniel Dunbarc475d422008-10-29 22:36:39 +00003760 continue;
Daniel Dunbar7bd00bd2008-08-12 06:48:42 +00003761
Daniel Dunbarc475d422008-10-29 22:36:39 +00003762 std::vector<llvm::Constant*> Values(5);
Owen Anderson0b75f232009-07-31 20:28:54 +00003763 Values[0] = llvm::Constant::getNullValue(ObjCTypes.ProtocolExtensionPtrTy);
Chris Lattnerf56501c2009-07-17 23:57:13 +00003764 Values[1] = GetClassName(I->first);
Owen Anderson0b75f232009-07-31 20:28:54 +00003765 Values[2] = llvm::Constant::getNullValue(ObjCTypes.ProtocolListPtrTy);
Daniel Dunbarc475d422008-10-29 22:36:39 +00003766 Values[3] = Values[4] =
Owen Anderson0b75f232009-07-31 20:28:54 +00003767 llvm::Constant::getNullValue(ObjCTypes.MethodDescriptionListPtrTy);
Chris Lattnerf56501c2009-07-17 23:57:13 +00003768 I->second->setLinkage(llvm::GlobalValue::InternalLinkage);
Owen Anderson0e0189d2009-07-27 22:29:56 +00003769 I->second->setInitializer(llvm::ConstantStruct::get(ObjCTypes.ProtocolTy,
Daniel Dunbarc475d422008-10-29 22:36:39 +00003770 Values));
Chris Lattnerf56501c2009-07-17 23:57:13 +00003771 CGM.AddUsedGlobal(I->second);
Daniel Dunbarc475d422008-10-29 22:36:39 +00003772 }
3773
Daniel Dunbarc61d0e92008-08-25 06:02:07 +00003774 // Add assembler directives to add lazy undefined symbol references
3775 // for classes which are referenced but not defined. This is
3776 // important for correct linker interaction.
Daniel Dunbard027a922009-09-07 00:20:42 +00003777 //
3778 // FIXME: It would be nice if we had an LLVM construct for this.
3779 if (!LazySymbols.empty() || !DefinedSymbols.empty()) {
3780 llvm::SmallString<256> Asm;
3781 Asm += CGM.getModule().getModuleInlineAsm();
3782 if (!Asm.empty() && Asm.back() != '\n')
3783 Asm += '\n';
Daniel Dunbarc61d0e92008-08-25 06:02:07 +00003784
Daniel Dunbard027a922009-09-07 00:20:42 +00003785 llvm::raw_svector_ostream OS(Asm);
Daniel Dunbard027a922009-09-07 00:20:42 +00003786 for (llvm::SetVector<IdentifierInfo*>::iterator I = DefinedSymbols.begin(),
3787 e = DefinedSymbols.end(); I != e; ++I)
Daniel Dunbar07d07852009-10-18 21:17:35 +00003788 OS << "\t.objc_class_name_" << (*I)->getName() << "=0\n"
3789 << "\t.globl .objc_class_name_" << (*I)->getName() << "\n";
Chris Lattnera299f2c2010-04-17 18:26:20 +00003790 for (llvm::SetVector<IdentifierInfo*>::iterator I = LazySymbols.begin(),
Fariborz Jahanian9adb2e62010-06-21 22:05:18 +00003791 e = LazySymbols.end(); I != e; ++I) {
Chris Lattnera299f2c2010-04-17 18:26:20 +00003792 OS << "\t.lazy_reference .objc_class_name_" << (*I)->getName() << "\n";
Fariborz Jahanian9adb2e62010-06-21 22:05:18 +00003793 }
3794
3795 for (size_t i = 0; i < DefinedCategoryNames.size(); ++i) {
3796 OS << "\t.objc_category_name_" << DefinedCategoryNames[i] << "=0\n"
3797 << "\t.globl .objc_category_name_" << DefinedCategoryNames[i] << "\n";
3798 }
Chris Lattnera299f2c2010-04-17 18:26:20 +00003799
Daniel Dunbard027a922009-09-07 00:20:42 +00003800 CGM.getModule().setModuleInlineAsm(OS.str());
Daniel Dunbarc61d0e92008-08-25 06:02:07 +00003801 }
Daniel Dunbar3ad53482008-08-11 21:35:06 +00003802}
3803
Daniel Dunbar59e476b2009-08-03 17:06:42 +00003804CGObjCNonFragileABIMac::CGObjCNonFragileABIMac(CodeGen::CodeGenModule &cgm)
Fariborz Jahanian279eda62009-01-21 22:04:16 +00003805 : CGObjCCommonMac(cgm),
Mike Stump11289f42009-09-09 15:08:12 +00003806 ObjCTypes(cgm) {
Fariborz Jahanian71394042009-01-23 23:53:38 +00003807 ObjCEmptyCacheVar = ObjCEmptyVtableVar = NULL;
Fariborz Jahanian279eda62009-01-21 22:04:16 +00003808 ObjCABI = 2;
3809}
3810
Daniel Dunbar3ad53482008-08-11 21:35:06 +00003811/* *** */
3812
Fariborz Jahanian279eda62009-01-21 22:04:16 +00003813ObjCCommonTypesHelper::ObjCCommonTypesHelper(CodeGen::CodeGenModule &cgm)
Mike Stump11289f42009-09-09 15:08:12 +00003814 : VMContext(cgm.getLLVMContext()), CGM(cgm) {
Daniel Dunbar7bd00bd2008-08-12 06:48:42 +00003815 CodeGen::CodeGenTypes &Types = CGM.getTypes();
3816 ASTContext &Ctx = CGM.getContext();
Daniel Dunbar59e476b2009-08-03 17:06:42 +00003817
Daniel Dunbar22d82ed2008-08-21 04:36:09 +00003818 ShortTy = Types.ConvertType(Ctx.ShortTy);
Daniel Dunbarb036db82008-08-13 03:21:16 +00003819 IntTy = Types.ConvertType(Ctx.IntTy);
Daniel Dunbar7bd00bd2008-08-12 06:48:42 +00003820 LongTy = Types.ConvertType(Ctx.LongTy);
Fariborz Jahanianaedcfa42009-03-23 19:10:40 +00003821 LongLongTy = Types.ConvertType(Ctx.LongLongTy);
Benjamin Kramerabd5b902009-10-13 10:07:13 +00003822 Int8PtrTy = llvm::Type::getInt8PtrTy(VMContext);
Daniel Dunbar59e476b2009-08-03 17:06:42 +00003823
Daniel Dunbar7bd00bd2008-08-12 06:48:42 +00003824 ObjectPtrTy = Types.ConvertType(Ctx.getObjCIdType());
Owen Anderson9793f0e2009-07-29 22:16:19 +00003825 PtrObjectPtrTy = llvm::PointerType::getUnqual(ObjectPtrTy);
Daniel Dunbar7bd00bd2008-08-12 06:48:42 +00003826 SelectorPtrTy = Types.ConvertType(Ctx.getObjCSelType());
Daniel Dunbar59e476b2009-08-03 17:06:42 +00003827
Mike Stump18bb9282009-05-16 07:57:57 +00003828 // FIXME: It would be nice to unify this with the opaque type, so that the IR
3829 // comes out a bit cleaner.
Daniel Dunbarb036db82008-08-13 03:21:16 +00003830 const llvm::Type *T = Types.ConvertType(Ctx.getObjCProtoType());
Owen Anderson9793f0e2009-07-29 22:16:19 +00003831 ExternalProtocolPtrTy = llvm::PointerType::getUnqual(T);
Daniel Dunbar59e476b2009-08-03 17:06:42 +00003832
Fariborz Jahanian279eda62009-01-21 22:04:16 +00003833 // I'm not sure I like this. The implicit coordination is a bit
3834 // gross. We should solve this in a reasonable fashion because this
3835 // is a pretty common task (match some runtime data structure with
3836 // an LLVM data structure).
Daniel Dunbar59e476b2009-08-03 17:06:42 +00003837
Fariborz Jahanian279eda62009-01-21 22:04:16 +00003838 // FIXME: This is leaked.
3839 // FIXME: Merge with rewriter code?
Daniel Dunbar59e476b2009-08-03 17:06:42 +00003840
Fariborz Jahanian279eda62009-01-21 22:04:16 +00003841 // struct _objc_super {
3842 // id self;
3843 // Class cls;
3844 // }
Abramo Bagnara6150c882010-05-11 21:36:43 +00003845 RecordDecl *RD = RecordDecl::Create(Ctx, TTK_Struct,
Daniel Dunbar0c005372010-04-29 16:29:11 +00003846 Ctx.getTranslationUnitDecl(),
Fariborz Jahanian279eda62009-01-21 22:04:16 +00003847 SourceLocation(),
Daniel Dunbar59e476b2009-08-03 17:06:42 +00003848 &Ctx.Idents.get("_objc_super"));
Argyrios Kyrtzidiscfbfe782009-06-30 02:36:12 +00003849 RD->addDecl(FieldDecl::Create(Ctx, RD, SourceLocation(), 0,
Argyrios Kyrtzidis60ed5602009-08-19 01:27:57 +00003850 Ctx.getObjCIdType(), 0, 0, false));
Daniel Dunbar59e476b2009-08-03 17:06:42 +00003851 RD->addDecl(FieldDecl::Create(Ctx, RD, SourceLocation(), 0,
Argyrios Kyrtzidis60ed5602009-08-19 01:27:57 +00003852 Ctx.getObjCClassType(), 0, 0, false));
Douglas Gregord5058122010-02-11 01:19:42 +00003853 RD->completeDefinition();
Daniel Dunbar59e476b2009-08-03 17:06:42 +00003854
Fariborz Jahanian279eda62009-01-21 22:04:16 +00003855 SuperCTy = Ctx.getTagDeclType(RD);
3856 SuperPtrCTy = Ctx.getPointerType(SuperCTy);
Daniel Dunbar59e476b2009-08-03 17:06:42 +00003857
Fariborz Jahanian279eda62009-01-21 22:04:16 +00003858 SuperTy = cast<llvm::StructType>(Types.ConvertType(SuperCTy));
Daniel Dunbar59e476b2009-08-03 17:06:42 +00003859 SuperPtrTy = llvm::PointerType::getUnqual(SuperTy);
3860
Fariborz Jahanianb15a3d52009-01-22 23:02:58 +00003861 // struct _prop_t {
3862 // char *name;
Daniel Dunbar59e476b2009-08-03 17:06:42 +00003863 // char *attributes;
Fariborz Jahanianb15a3d52009-01-22 23:02:58 +00003864 // }
Owen Anderson758428f2009-08-05 23:18:46 +00003865 PropertyTy = llvm::StructType::get(VMContext, Int8PtrTy, Int8PtrTy, NULL);
Daniel Dunbar59e476b2009-08-03 17:06:42 +00003866 CGM.getModule().addTypeName("struct._prop_t",
Fariborz Jahanianb15a3d52009-01-22 23:02:58 +00003867 PropertyTy);
Daniel Dunbar59e476b2009-08-03 17:06:42 +00003868
Fariborz Jahanianb15a3d52009-01-22 23:02:58 +00003869 // struct _prop_list_t {
Fariborz Jahanian0232c052009-01-23 01:46:23 +00003870 // uint32_t entsize; // sizeof(struct _prop_t)
Fariborz Jahanianb15a3d52009-01-22 23:02:58 +00003871 // uint32_t count_of_properties;
Fariborz Jahanian0232c052009-01-23 01:46:23 +00003872 // struct _prop_t prop_list[count_of_properties];
Fariborz Jahanianb15a3d52009-01-22 23:02:58 +00003873 // }
Owen Anderson758428f2009-08-05 23:18:46 +00003874 PropertyListTy = llvm::StructType::get(VMContext, IntTy,
Fariborz Jahanianb15a3d52009-01-22 23:02:58 +00003875 IntTy,
Owen Anderson9793f0e2009-07-29 22:16:19 +00003876 llvm::ArrayType::get(PropertyTy, 0),
Fariborz Jahanianb15a3d52009-01-22 23:02:58 +00003877 NULL);
Daniel Dunbar59e476b2009-08-03 17:06:42 +00003878 CGM.getModule().addTypeName("struct._prop_list_t",
Fariborz Jahanianb15a3d52009-01-22 23:02:58 +00003879 PropertyListTy);
3880 // struct _prop_list_t *
Owen Anderson9793f0e2009-07-29 22:16:19 +00003881 PropertyListPtrTy = llvm::PointerType::getUnqual(PropertyListTy);
Daniel Dunbar59e476b2009-08-03 17:06:42 +00003882
Fariborz Jahanianb15a3d52009-01-22 23:02:58 +00003883 // struct _objc_method {
3884 // SEL _cmd;
3885 // char *method_type;
3886 // char *_imp;
3887 // }
Owen Anderson758428f2009-08-05 23:18:46 +00003888 MethodTy = llvm::StructType::get(VMContext, SelectorPtrTy,
Fariborz Jahanianb15a3d52009-01-22 23:02:58 +00003889 Int8PtrTy,
3890 Int8PtrTy,
3891 NULL);
3892 CGM.getModule().addTypeName("struct._objc_method", MethodTy);
Daniel Dunbar59e476b2009-08-03 17:06:42 +00003893
Fariborz Jahanian0232c052009-01-23 01:46:23 +00003894 // struct _objc_cache *
Owen Andersonc36edfe2009-08-13 23:27:53 +00003895 CacheTy = llvm::OpaqueType::get(VMContext);
Fariborz Jahanian0232c052009-01-23 01:46:23 +00003896 CGM.getModule().addTypeName("struct._objc_cache", CacheTy);
Owen Anderson9793f0e2009-07-29 22:16:19 +00003897 CachePtrTy = llvm::PointerType::getUnqual(CacheTy);
Fariborz Jahanian279eda62009-01-21 22:04:16 +00003898}
Daniel Dunbar7bd00bd2008-08-12 06:48:42 +00003899
Daniel Dunbar59e476b2009-08-03 17:06:42 +00003900ObjCTypesHelper::ObjCTypesHelper(CodeGen::CodeGenModule &cgm)
Mike Stump11289f42009-09-09 15:08:12 +00003901 : ObjCCommonTypesHelper(cgm) {
Fariborz Jahanian4b4c8262009-01-21 00:39:53 +00003902 // struct _objc_method_description {
3903 // SEL name;
3904 // char *types;
3905 // }
Daniel Dunbar59e476b2009-08-03 17:06:42 +00003906 MethodDescriptionTy =
Owen Anderson758428f2009-08-05 23:18:46 +00003907 llvm::StructType::get(VMContext, SelectorPtrTy,
Daniel Dunbar22d82ed2008-08-21 04:36:09 +00003908 Int8PtrTy,
Daniel Dunbarb036db82008-08-13 03:21:16 +00003909 NULL);
Daniel Dunbar59e476b2009-08-03 17:06:42 +00003910 CGM.getModule().addTypeName("struct._objc_method_description",
Daniel Dunbarb036db82008-08-13 03:21:16 +00003911 MethodDescriptionTy);
3912
Fariborz Jahanian4b4c8262009-01-21 00:39:53 +00003913 // struct _objc_method_description_list {
3914 // int count;
3915 // struct _objc_method_description[1];
3916 // }
Daniel Dunbar59e476b2009-08-03 17:06:42 +00003917 MethodDescriptionListTy =
Owen Anderson758428f2009-08-05 23:18:46 +00003918 llvm::StructType::get(VMContext, IntTy,
Owen Anderson9793f0e2009-07-29 22:16:19 +00003919 llvm::ArrayType::get(MethodDescriptionTy, 0),
Daniel Dunbarb036db82008-08-13 03:21:16 +00003920 NULL);
Daniel Dunbar59e476b2009-08-03 17:06:42 +00003921 CGM.getModule().addTypeName("struct._objc_method_description_list",
Daniel Dunbarb036db82008-08-13 03:21:16 +00003922 MethodDescriptionListTy);
Daniel Dunbar59e476b2009-08-03 17:06:42 +00003923
Fariborz Jahanian4b4c8262009-01-21 00:39:53 +00003924 // struct _objc_method_description_list *
Daniel Dunbar59e476b2009-08-03 17:06:42 +00003925 MethodDescriptionListPtrTy =
Owen Anderson9793f0e2009-07-29 22:16:19 +00003926 llvm::PointerType::getUnqual(MethodDescriptionListTy);
Daniel Dunbarb036db82008-08-13 03:21:16 +00003927
Daniel Dunbarb036db82008-08-13 03:21:16 +00003928 // Protocol description structures
3929
Fariborz Jahanian4b4c8262009-01-21 00:39:53 +00003930 // struct _objc_protocol_extension {
3931 // uint32_t size; // sizeof(struct _objc_protocol_extension)
3932 // struct _objc_method_description_list *optional_instance_methods;
3933 // struct _objc_method_description_list *optional_class_methods;
3934 // struct _objc_property_list *instance_properties;
3935 // }
Daniel Dunbar59e476b2009-08-03 17:06:42 +00003936 ProtocolExtensionTy =
Owen Anderson758428f2009-08-05 23:18:46 +00003937 llvm::StructType::get(VMContext, IntTy,
Fariborz Jahanian4b4c8262009-01-21 00:39:53 +00003938 MethodDescriptionListPtrTy,
3939 MethodDescriptionListPtrTy,
Daniel Dunbarb036db82008-08-13 03:21:16 +00003940 PropertyListPtrTy,
3941 NULL);
Daniel Dunbar59e476b2009-08-03 17:06:42 +00003942 CGM.getModule().addTypeName("struct._objc_protocol_extension",
Daniel Dunbarb036db82008-08-13 03:21:16 +00003943 ProtocolExtensionTy);
Daniel Dunbar59e476b2009-08-03 17:06:42 +00003944
Fariborz Jahanian4b4c8262009-01-21 00:39:53 +00003945 // struct _objc_protocol_extension *
Owen Anderson9793f0e2009-07-29 22:16:19 +00003946 ProtocolExtensionPtrTy = llvm::PointerType::getUnqual(ProtocolExtensionTy);
Daniel Dunbarb036db82008-08-13 03:21:16 +00003947
Daniel Dunbarc475d422008-10-29 22:36:39 +00003948 // Handle recursive construction of Protocol and ProtocolList types
Daniel Dunbarb036db82008-08-13 03:21:16 +00003949
Owen Andersonc36edfe2009-08-13 23:27:53 +00003950 llvm::PATypeHolder ProtocolTyHolder = llvm::OpaqueType::get(VMContext);
3951 llvm::PATypeHolder ProtocolListTyHolder = llvm::OpaqueType::get(VMContext);
Daniel Dunbarb036db82008-08-13 03:21:16 +00003952
Daniel Dunbar59e476b2009-08-03 17:06:42 +00003953 const llvm::Type *T =
Mike Stump11289f42009-09-09 15:08:12 +00003954 llvm::StructType::get(VMContext,
Owen Anderson758428f2009-08-05 23:18:46 +00003955 llvm::PointerType::getUnqual(ProtocolListTyHolder),
Fariborz Jahanian279eda62009-01-21 22:04:16 +00003956 LongTy,
Owen Anderson9793f0e2009-07-29 22:16:19 +00003957 llvm::ArrayType::get(ProtocolTyHolder, 0),
Fariborz Jahanian279eda62009-01-21 22:04:16 +00003958 NULL);
Daniel Dunbarb036db82008-08-13 03:21:16 +00003959 cast<llvm::OpaqueType>(ProtocolListTyHolder.get())->refineAbstractTypeTo(T);
3960
Fariborz Jahanian4b4c8262009-01-21 00:39:53 +00003961 // struct _objc_protocol {
3962 // struct _objc_protocol_extension *isa;
3963 // char *protocol_name;
3964 // struct _objc_protocol **_objc_protocol_list;
3965 // struct _objc_method_description_list *instance_methods;
3966 // struct _objc_method_description_list *class_methods;
3967 // }
Owen Anderson758428f2009-08-05 23:18:46 +00003968 T = llvm::StructType::get(VMContext, ProtocolExtensionPtrTy,
Daniel Dunbar22d82ed2008-08-21 04:36:09 +00003969 Int8PtrTy,
Daniel Dunbar59e476b2009-08-03 17:06:42 +00003970 llvm::PointerType::getUnqual(ProtocolListTyHolder),
Daniel Dunbarb036db82008-08-13 03:21:16 +00003971 MethodDescriptionListPtrTy,
3972 MethodDescriptionListPtrTy,
3973 NULL);
3974 cast<llvm::OpaqueType>(ProtocolTyHolder.get())->refineAbstractTypeTo(T);
3975
3976 ProtocolListTy = cast<llvm::StructType>(ProtocolListTyHolder.get());
Daniel Dunbar59e476b2009-08-03 17:06:42 +00003977 CGM.getModule().addTypeName("struct._objc_protocol_list",
Daniel Dunbarb036db82008-08-13 03:21:16 +00003978 ProtocolListTy);
Fariborz Jahanian4b4c8262009-01-21 00:39:53 +00003979 // struct _objc_protocol_list *
Owen Anderson9793f0e2009-07-29 22:16:19 +00003980 ProtocolListPtrTy = llvm::PointerType::getUnqual(ProtocolListTy);
Daniel Dunbarb036db82008-08-13 03:21:16 +00003981
3982 ProtocolTy = cast<llvm::StructType>(ProtocolTyHolder.get());
Fariborz Jahanian4b4c8262009-01-21 00:39:53 +00003983 CGM.getModule().addTypeName("struct._objc_protocol", ProtocolTy);
Owen Anderson9793f0e2009-07-29 22:16:19 +00003984 ProtocolPtrTy = llvm::PointerType::getUnqual(ProtocolTy);
Daniel Dunbar22d82ed2008-08-21 04:36:09 +00003985
3986 // Class description structures
3987
Fariborz Jahanian4b4c8262009-01-21 00:39:53 +00003988 // struct _objc_ivar {
3989 // char *ivar_name;
3990 // char *ivar_type;
3991 // int ivar_offset;
3992 // }
Owen Anderson758428f2009-08-05 23:18:46 +00003993 IvarTy = llvm::StructType::get(VMContext, Int8PtrTy,
Daniel Dunbar59e476b2009-08-03 17:06:42 +00003994 Int8PtrTy,
3995 IntTy,
Daniel Dunbar22d82ed2008-08-21 04:36:09 +00003996 NULL);
3997 CGM.getModule().addTypeName("struct._objc_ivar", IvarTy);
3998
Fariborz Jahanian4b4c8262009-01-21 00:39:53 +00003999 // struct _objc_ivar_list *
Owen Andersonc36edfe2009-08-13 23:27:53 +00004000 IvarListTy = llvm::OpaqueType::get(VMContext);
Daniel Dunbar22d82ed2008-08-21 04:36:09 +00004001 CGM.getModule().addTypeName("struct._objc_ivar_list", IvarListTy);
Owen Anderson9793f0e2009-07-29 22:16:19 +00004002 IvarListPtrTy = llvm::PointerType::getUnqual(IvarListTy);
Daniel Dunbar22d82ed2008-08-21 04:36:09 +00004003
Fariborz Jahanian4b4c8262009-01-21 00:39:53 +00004004 // struct _objc_method_list *
Owen Andersonc36edfe2009-08-13 23:27:53 +00004005 MethodListTy = llvm::OpaqueType::get(VMContext);
Daniel Dunbar22d82ed2008-08-21 04:36:09 +00004006 CGM.getModule().addTypeName("struct._objc_method_list", MethodListTy);
Owen Anderson9793f0e2009-07-29 22:16:19 +00004007 MethodListPtrTy = llvm::PointerType::getUnqual(MethodListTy);
Daniel Dunbar22d82ed2008-08-21 04:36:09 +00004008
Fariborz Jahanian4b4c8262009-01-21 00:39:53 +00004009 // struct _objc_class_extension *
Daniel Dunbar59e476b2009-08-03 17:06:42 +00004010 ClassExtensionTy =
Owen Anderson758428f2009-08-05 23:18:46 +00004011 llvm::StructType::get(VMContext, IntTy,
Daniel Dunbar22d82ed2008-08-21 04:36:09 +00004012 Int8PtrTy,
4013 PropertyListPtrTy,
4014 NULL);
4015 CGM.getModule().addTypeName("struct._objc_class_extension", ClassExtensionTy);
Owen Anderson9793f0e2009-07-29 22:16:19 +00004016 ClassExtensionPtrTy = llvm::PointerType::getUnqual(ClassExtensionTy);
Daniel Dunbar22d82ed2008-08-21 04:36:09 +00004017
Owen Andersonc36edfe2009-08-13 23:27:53 +00004018 llvm::PATypeHolder ClassTyHolder = llvm::OpaqueType::get(VMContext);
Daniel Dunbar22d82ed2008-08-21 04:36:09 +00004019
Fariborz Jahanian4b4c8262009-01-21 00:39:53 +00004020 // struct _objc_class {
4021 // Class isa;
4022 // Class super_class;
4023 // char *name;
4024 // long version;
4025 // long info;
4026 // long instance_size;
4027 // struct _objc_ivar_list *ivars;
4028 // struct _objc_method_list *methods;
4029 // struct _objc_cache *cache;
4030 // struct _objc_protocol_list *protocols;
4031 // char *ivar_layout;
4032 // struct _objc_class_ext *ext;
4033 // };
Owen Anderson758428f2009-08-05 23:18:46 +00004034 T = llvm::StructType::get(VMContext,
4035 llvm::PointerType::getUnqual(ClassTyHolder),
Owen Anderson9793f0e2009-07-29 22:16:19 +00004036 llvm::PointerType::getUnqual(ClassTyHolder),
Daniel Dunbar22d82ed2008-08-21 04:36:09 +00004037 Int8PtrTy,
4038 LongTy,
4039 LongTy,
4040 LongTy,
4041 IvarListPtrTy,
4042 MethodListPtrTy,
4043 CachePtrTy,
4044 ProtocolListPtrTy,
4045 Int8PtrTy,
4046 ClassExtensionPtrTy,
4047 NULL);
4048 cast<llvm::OpaqueType>(ClassTyHolder.get())->refineAbstractTypeTo(T);
Daniel Dunbar59e476b2009-08-03 17:06:42 +00004049
Daniel Dunbar22d82ed2008-08-21 04:36:09 +00004050 ClassTy = cast<llvm::StructType>(ClassTyHolder.get());
4051 CGM.getModule().addTypeName("struct._objc_class", ClassTy);
Owen Anderson9793f0e2009-07-29 22:16:19 +00004052 ClassPtrTy = llvm::PointerType::getUnqual(ClassTy);
Daniel Dunbar22d82ed2008-08-21 04:36:09 +00004053
Fariborz Jahanian4b4c8262009-01-21 00:39:53 +00004054 // struct _objc_category {
4055 // char *category_name;
4056 // char *class_name;
4057 // struct _objc_method_list *instance_method;
4058 // struct _objc_method_list *class_method;
4059 // uint32_t size; // sizeof(struct _objc_category)
4060 // struct _objc_property_list *instance_properties;// category's @property
4061 // }
Owen Anderson758428f2009-08-05 23:18:46 +00004062 CategoryTy = llvm::StructType::get(VMContext, Int8PtrTy,
Daniel Dunbar938a77f2008-08-22 20:34:54 +00004063 Int8PtrTy,
4064 MethodListPtrTy,
4065 MethodListPtrTy,
4066 ProtocolListPtrTy,
4067 IntTy,
4068 PropertyListPtrTy,
4069 NULL);
4070 CGM.getModule().addTypeName("struct._objc_category", CategoryTy);
4071
Daniel Dunbar22d82ed2008-08-21 04:36:09 +00004072 // Global metadata structures
4073
Fariborz Jahanian4b4c8262009-01-21 00:39:53 +00004074 // struct _objc_symtab {
4075 // long sel_ref_cnt;
4076 // SEL *refs;
4077 // short cls_def_cnt;
4078 // short cat_def_cnt;
4079 // char *defs[cls_def_cnt + cat_def_cnt];
4080 // }
Owen Anderson758428f2009-08-05 23:18:46 +00004081 SymtabTy = llvm::StructType::get(VMContext, LongTy,
Daniel Dunbar22d82ed2008-08-21 04:36:09 +00004082 SelectorPtrTy,
4083 ShortTy,
4084 ShortTy,
Owen Anderson9793f0e2009-07-29 22:16:19 +00004085 llvm::ArrayType::get(Int8PtrTy, 0),
Daniel Dunbar22d82ed2008-08-21 04:36:09 +00004086 NULL);
4087 CGM.getModule().addTypeName("struct._objc_symtab", SymtabTy);
Owen Anderson9793f0e2009-07-29 22:16:19 +00004088 SymtabPtrTy = llvm::PointerType::getUnqual(SymtabTy);
Daniel Dunbar22d82ed2008-08-21 04:36:09 +00004089
Fariborz Jahanianeee54df2009-01-22 00:37:21 +00004090 // struct _objc_module {
4091 // long version;
4092 // long size; // sizeof(struct _objc_module)
4093 // char *name;
4094 // struct _objc_symtab* symtab;
4095 // }
Daniel Dunbar59e476b2009-08-03 17:06:42 +00004096 ModuleTy =
Owen Anderson758428f2009-08-05 23:18:46 +00004097 llvm::StructType::get(VMContext, LongTy,
Daniel Dunbar22d82ed2008-08-21 04:36:09 +00004098 LongTy,
4099 Int8PtrTy,
4100 SymtabPtrTy,
4101 NULL);
4102 CGM.getModule().addTypeName("struct._objc_module", ModuleTy);
Daniel Dunbar97ff50d2008-08-23 09:25:55 +00004103
Daniel Dunbar59e476b2009-08-03 17:06:42 +00004104
Mike Stump18bb9282009-05-16 07:57:57 +00004105 // FIXME: This is the size of the setjmp buffer and should be target
4106 // specific. 18 is what's used on 32-bit X86.
Anders Carlsson9ff22482008-09-09 10:10:21 +00004107 uint64_t SetJmpBufferSize = 18;
Daniel Dunbar59e476b2009-08-03 17:06:42 +00004108
Anders Carlsson9ff22482008-09-09 10:10:21 +00004109 // Exceptions
Owen Anderson9793f0e2009-07-29 22:16:19 +00004110 const llvm::Type *StackPtrTy = llvm::ArrayType::get(
Benjamin Kramerabd5b902009-10-13 10:07:13 +00004111 llvm::Type::getInt8PtrTy(VMContext), 4);
Daniel Dunbar59e476b2009-08-03 17:06:42 +00004112
4113 ExceptionDataTy =
Chris Lattner5e016ae2010-06-27 07:15:29 +00004114 llvm::StructType::get(VMContext,
4115 llvm::ArrayType::get(llvm::Type::getInt32Ty(VMContext),
4116 SetJmpBufferSize),
Anders Carlsson9ff22482008-09-09 10:10:21 +00004117 StackPtrTy, NULL);
Daniel Dunbar59e476b2009-08-03 17:06:42 +00004118 CGM.getModule().addTypeName("struct._objc_exception_data",
Anders Carlsson9ff22482008-09-09 10:10:21 +00004119 ExceptionDataTy);
4120
Daniel Dunbar8b8683f2008-08-12 00:12:39 +00004121}
4122
Daniel Dunbar59e476b2009-08-03 17:06:42 +00004123ObjCNonFragileABITypesHelper::ObjCNonFragileABITypesHelper(CodeGen::CodeGenModule &cgm)
Mike Stump11289f42009-09-09 15:08:12 +00004124 : ObjCCommonTypesHelper(cgm) {
Fariborz Jahanianb15a3d52009-01-22 23:02:58 +00004125 // struct _method_list_t {
4126 // uint32_t entsize; // sizeof(struct _objc_method)
4127 // uint32_t method_count;
4128 // struct _objc_method method_list[method_count];
4129 // }
Owen Anderson758428f2009-08-05 23:18:46 +00004130 MethodListnfABITy = llvm::StructType::get(VMContext, IntTy,
Fariborz Jahanian0232c052009-01-23 01:46:23 +00004131 IntTy,
Owen Anderson9793f0e2009-07-29 22:16:19 +00004132 llvm::ArrayType::get(MethodTy, 0),
Fariborz Jahanian0232c052009-01-23 01:46:23 +00004133 NULL);
4134 CGM.getModule().addTypeName("struct.__method_list_t",
4135 MethodListnfABITy);
4136 // struct method_list_t *
Owen Anderson9793f0e2009-07-29 22:16:19 +00004137 MethodListnfABIPtrTy = llvm::PointerType::getUnqual(MethodListnfABITy);
Daniel Dunbar59e476b2009-08-03 17:06:42 +00004138
Fariborz Jahanianb15a3d52009-01-22 23:02:58 +00004139 // struct _protocol_t {
4140 // id isa; // NULL
4141 // const char * const protocol_name;
Fariborz Jahanian0232c052009-01-23 01:46:23 +00004142 // const struct _protocol_list_t * protocol_list; // super protocols
Fariborz Jahanianb15a3d52009-01-22 23:02:58 +00004143 // const struct method_list_t * const instance_methods;
4144 // const struct method_list_t * const class_methods;
4145 // const struct method_list_t *optionalInstanceMethods;
4146 // const struct method_list_t *optionalClassMethods;
Fariborz Jahanian0232c052009-01-23 01:46:23 +00004147 // const struct _prop_list_t * properties;
Fariborz Jahanianb15a3d52009-01-22 23:02:58 +00004148 // const uint32_t size; // sizeof(struct _protocol_t)
4149 // const uint32_t flags; // = 0
4150 // }
Daniel Dunbar59e476b2009-08-03 17:06:42 +00004151
Fariborz Jahanian0232c052009-01-23 01:46:23 +00004152 // Holder for struct _protocol_list_t *
Owen Andersonc36edfe2009-08-13 23:27:53 +00004153 llvm::PATypeHolder ProtocolListTyHolder = llvm::OpaqueType::get(VMContext);
Daniel Dunbar59e476b2009-08-03 17:06:42 +00004154
Owen Anderson758428f2009-08-05 23:18:46 +00004155 ProtocolnfABITy = llvm::StructType::get(VMContext, ObjectPtrTy,
Fariborz Jahanian0232c052009-01-23 01:46:23 +00004156 Int8PtrTy,
Owen Anderson9793f0e2009-07-29 22:16:19 +00004157 llvm::PointerType::getUnqual(
Fariborz Jahanian0232c052009-01-23 01:46:23 +00004158 ProtocolListTyHolder),
4159 MethodListnfABIPtrTy,
4160 MethodListnfABIPtrTy,
4161 MethodListnfABIPtrTy,
4162 MethodListnfABIPtrTy,
4163 PropertyListPtrTy,
4164 IntTy,
4165 IntTy,
4166 NULL);
4167 CGM.getModule().addTypeName("struct._protocol_t",
4168 ProtocolnfABITy);
Daniel Dunbar8de90f02009-02-15 07:36:20 +00004169
4170 // struct _protocol_t*
Owen Anderson9793f0e2009-07-29 22:16:19 +00004171 ProtocolnfABIPtrTy = llvm::PointerType::getUnqual(ProtocolnfABITy);
Daniel Dunbar59e476b2009-08-03 17:06:42 +00004172
Fariborz Jahanian56b3b772009-01-29 19:24:30 +00004173 // struct _protocol_list_t {
Fariborz Jahanian0232c052009-01-23 01:46:23 +00004174 // long protocol_count; // Note, this is 32/64 bit
Daniel Dunbar8de90f02009-02-15 07:36:20 +00004175 // struct _protocol_t *[protocol_count];
Fariborz Jahanianb15a3d52009-01-22 23:02:58 +00004176 // }
Owen Anderson758428f2009-08-05 23:18:46 +00004177 ProtocolListnfABITy = llvm::StructType::get(VMContext, LongTy,
Owen Anderson9793f0e2009-07-29 22:16:19 +00004178 llvm::ArrayType::get(
Daniel Dunbar8de90f02009-02-15 07:36:20 +00004179 ProtocolnfABIPtrTy, 0),
Fariborz Jahanian0232c052009-01-23 01:46:23 +00004180 NULL);
4181 CGM.getModule().addTypeName("struct._objc_protocol_list",
4182 ProtocolListnfABITy);
Daniel Dunbar8de90f02009-02-15 07:36:20 +00004183 cast<llvm::OpaqueType>(ProtocolListTyHolder.get())->refineAbstractTypeTo(
Daniel Dunbar59e476b2009-08-03 17:06:42 +00004184 ProtocolListnfABITy);
4185
Fariborz Jahanian0232c052009-01-23 01:46:23 +00004186 // struct _objc_protocol_list*
Owen Anderson9793f0e2009-07-29 22:16:19 +00004187 ProtocolListnfABIPtrTy = llvm::PointerType::getUnqual(ProtocolListnfABITy);
Daniel Dunbar59e476b2009-08-03 17:06:42 +00004188
Fariborz Jahanianb15a3d52009-01-22 23:02:58 +00004189 // struct _ivar_t {
4190 // unsigned long int *offset; // pointer to ivar offset location
4191 // char *name;
4192 // char *type;
4193 // uint32_t alignment;
4194 // uint32_t size;
4195 // }
Mike Stump11289f42009-09-09 15:08:12 +00004196 IvarnfABITy = llvm::StructType::get(VMContext,
Owen Anderson758428f2009-08-05 23:18:46 +00004197 llvm::PointerType::getUnqual(LongTy),
Fariborz Jahanian0232c052009-01-23 01:46:23 +00004198 Int8PtrTy,
4199 Int8PtrTy,
4200 IntTy,
4201 IntTy,
4202 NULL);
4203 CGM.getModule().addTypeName("struct._ivar_t", IvarnfABITy);
Daniel Dunbar59e476b2009-08-03 17:06:42 +00004204
Fariborz Jahanianb15a3d52009-01-22 23:02:58 +00004205 // struct _ivar_list_t {
4206 // uint32 entsize; // sizeof(struct _ivar_t)
4207 // uint32 count;
4208 // struct _iver_t list[count];
4209 // }
Owen Anderson758428f2009-08-05 23:18:46 +00004210 IvarListnfABITy = llvm::StructType::get(VMContext, IntTy,
Fariborz Jahanian9e3ad522009-01-24 20:21:50 +00004211 IntTy,
Owen Anderson9793f0e2009-07-29 22:16:19 +00004212 llvm::ArrayType::get(
Daniel Dunbar59e476b2009-08-03 17:06:42 +00004213 IvarnfABITy, 0),
Fariborz Jahanian9e3ad522009-01-24 20:21:50 +00004214 NULL);
4215 CGM.getModule().addTypeName("struct._ivar_list_t", IvarListnfABITy);
Daniel Dunbar59e476b2009-08-03 17:06:42 +00004216
Owen Anderson9793f0e2009-07-29 22:16:19 +00004217 IvarListnfABIPtrTy = llvm::PointerType::getUnqual(IvarListnfABITy);
Daniel Dunbar59e476b2009-08-03 17:06:42 +00004218
Fariborz Jahanian0232c052009-01-23 01:46:23 +00004219 // struct _class_ro_t {
Fariborz Jahanianb15a3d52009-01-22 23:02:58 +00004220 // uint32_t const flags;
4221 // uint32_t const instanceStart;
4222 // uint32_t const instanceSize;
4223 // uint32_t const reserved; // only when building for 64bit targets
4224 // const uint8_t * const ivarLayout;
4225 // const char *const name;
4226 // const struct _method_list_t * const baseMethods;
4227 // const struct _objc_protocol_list *const baseProtocols;
4228 // const struct _ivar_list_t *const ivars;
4229 // const uint8_t * const weakIvarLayout;
4230 // const struct _prop_list_t * const properties;
4231 // }
Daniel Dunbar59e476b2009-08-03 17:06:42 +00004232
Fariborz Jahanian0232c052009-01-23 01:46:23 +00004233 // FIXME. Add 'reserved' field in 64bit abi mode!
Owen Anderson758428f2009-08-05 23:18:46 +00004234 ClassRonfABITy = llvm::StructType::get(VMContext, IntTy,
Fariborz Jahanian0232c052009-01-23 01:46:23 +00004235 IntTy,
4236 IntTy,
4237 Int8PtrTy,
4238 Int8PtrTy,
4239 MethodListnfABIPtrTy,
4240 ProtocolListnfABIPtrTy,
4241 IvarListnfABIPtrTy,
4242 Int8PtrTy,
4243 PropertyListPtrTy,
4244 NULL);
4245 CGM.getModule().addTypeName("struct._class_ro_t",
4246 ClassRonfABITy);
Daniel Dunbar59e476b2009-08-03 17:06:42 +00004247
Fariborz Jahanian0232c052009-01-23 01:46:23 +00004248 // ImpnfABITy - LLVM for id (*)(id, SEL, ...)
4249 std::vector<const llvm::Type*> Params;
4250 Params.push_back(ObjectPtrTy);
4251 Params.push_back(SelectorPtrTy);
Owen Anderson9793f0e2009-07-29 22:16:19 +00004252 ImpnfABITy = llvm::PointerType::getUnqual(
Daniel Dunbar59e476b2009-08-03 17:06:42 +00004253 llvm::FunctionType::get(ObjectPtrTy, Params, false));
4254
Fariborz Jahanian0232c052009-01-23 01:46:23 +00004255 // struct _class_t {
4256 // struct _class_t *isa;
4257 // struct _class_t * const superclass;
4258 // void *cache;
4259 // IMP *vtable;
4260 // struct class_ro_t *ro;
4261 // }
Daniel Dunbar59e476b2009-08-03 17:06:42 +00004262
Owen Andersonc36edfe2009-08-13 23:27:53 +00004263 llvm::PATypeHolder ClassTyHolder = llvm::OpaqueType::get(VMContext);
Owen Anderson170229f2009-07-14 23:10:40 +00004264 ClassnfABITy =
Owen Anderson758428f2009-08-05 23:18:46 +00004265 llvm::StructType::get(VMContext,
4266 llvm::PointerType::getUnqual(ClassTyHolder),
Daniel Dunbar59e476b2009-08-03 17:06:42 +00004267 llvm::PointerType::getUnqual(ClassTyHolder),
4268 CachePtrTy,
4269 llvm::PointerType::getUnqual(ImpnfABITy),
4270 llvm::PointerType::getUnqual(ClassRonfABITy),
4271 NULL);
Fariborz Jahanian0232c052009-01-23 01:46:23 +00004272 CGM.getModule().addTypeName("struct._class_t", ClassnfABITy);
4273
4274 cast<llvm::OpaqueType>(ClassTyHolder.get())->refineAbstractTypeTo(
Daniel Dunbar59e476b2009-08-03 17:06:42 +00004275 ClassnfABITy);
4276
Fariborz Jahanian71394042009-01-23 23:53:38 +00004277 // LLVM for struct _class_t *
Owen Anderson9793f0e2009-07-29 22:16:19 +00004278 ClassnfABIPtrTy = llvm::PointerType::getUnqual(ClassnfABITy);
Daniel Dunbar59e476b2009-08-03 17:06:42 +00004279
Fariborz Jahanian0232c052009-01-23 01:46:23 +00004280 // struct _category_t {
4281 // const char * const name;
4282 // struct _class_t *const cls;
4283 // const struct _method_list_t * const instance_methods;
4284 // const struct _method_list_t * const class_methods;
4285 // const struct _protocol_list_t * const protocols;
4286 // const struct _prop_list_t * const properties;
Fariborz Jahanian5a63e4c2009-01-23 17:41:22 +00004287 // }
Owen Anderson758428f2009-08-05 23:18:46 +00004288 CategorynfABITy = llvm::StructType::get(VMContext, Int8PtrTy,
Fariborz Jahanian71394042009-01-23 23:53:38 +00004289 ClassnfABIPtrTy,
Fariborz Jahanian5a63e4c2009-01-23 17:41:22 +00004290 MethodListnfABIPtrTy,
4291 MethodListnfABIPtrTy,
4292 ProtocolListnfABIPtrTy,
4293 PropertyListPtrTy,
4294 NULL);
4295 CGM.getModule().addTypeName("struct._category_t", CategorynfABITy);
Daniel Dunbar59e476b2009-08-03 17:06:42 +00004296
Fariborz Jahanian82c72e12009-02-03 23:49:23 +00004297 // New types for nonfragile abi messaging.
Fariborz Jahaniane4dc35d2009-02-04 20:42:28 +00004298 CodeGen::CodeGenTypes &Types = CGM.getTypes();
4299 ASTContext &Ctx = CGM.getContext();
Daniel Dunbar59e476b2009-08-03 17:06:42 +00004300
Fariborz Jahanian82c72e12009-02-03 23:49:23 +00004301 // MessageRefTy - LLVM for:
4302 // struct _message_ref_t {
4303 // IMP messenger;
4304 // SEL name;
4305 // };
Daniel Dunbar59e476b2009-08-03 17:06:42 +00004306
Fariborz Jahaniane4dc35d2009-02-04 20:42:28 +00004307 // First the clang type for struct _message_ref_t
Abramo Bagnara6150c882010-05-11 21:36:43 +00004308 RecordDecl *RD = RecordDecl::Create(Ctx, TTK_Struct,
Daniel Dunbar0c005372010-04-29 16:29:11 +00004309 Ctx.getTranslationUnitDecl(),
Fariborz Jahaniane4dc35d2009-02-04 20:42:28 +00004310 SourceLocation(),
4311 &Ctx.Idents.get("_message_ref_t"));
Argyrios Kyrtzidiscfbfe782009-06-30 02:36:12 +00004312 RD->addDecl(FieldDecl::Create(Ctx, RD, SourceLocation(), 0,
Argyrios Kyrtzidis60ed5602009-08-19 01:27:57 +00004313 Ctx.VoidPtrTy, 0, 0, false));
Argyrios Kyrtzidiscfbfe782009-06-30 02:36:12 +00004314 RD->addDecl(FieldDecl::Create(Ctx, RD, SourceLocation(), 0,
Argyrios Kyrtzidis60ed5602009-08-19 01:27:57 +00004315 Ctx.getObjCSelType(), 0, 0, false));
Douglas Gregord5058122010-02-11 01:19:42 +00004316 RD->completeDefinition();
Daniel Dunbar59e476b2009-08-03 17:06:42 +00004317
Fariborz Jahaniane4dc35d2009-02-04 20:42:28 +00004318 MessageRefCTy = Ctx.getTagDeclType(RD);
4319 MessageRefCPtrTy = Ctx.getPointerType(MessageRefCTy);
4320 MessageRefTy = cast<llvm::StructType>(Types.ConvertType(MessageRefCTy));
Daniel Dunbar59e476b2009-08-03 17:06:42 +00004321
Fariborz Jahanian82c72e12009-02-03 23:49:23 +00004322 // MessageRefPtrTy - LLVM for struct _message_ref_t*
Owen Anderson9793f0e2009-07-29 22:16:19 +00004323 MessageRefPtrTy = llvm::PointerType::getUnqual(MessageRefTy);
Daniel Dunbar59e476b2009-08-03 17:06:42 +00004324
Fariborz Jahanian82c72e12009-02-03 23:49:23 +00004325 // SuperMessageRefTy - LLVM for:
4326 // struct _super_message_ref_t {
4327 // SUPER_IMP messenger;
4328 // SEL name;
4329 // };
Owen Anderson758428f2009-08-05 23:18:46 +00004330 SuperMessageRefTy = llvm::StructType::get(VMContext, ImpnfABITy,
Fariborz Jahanian82c72e12009-02-03 23:49:23 +00004331 SelectorPtrTy,
4332 NULL);
4333 CGM.getModule().addTypeName("struct._super_message_ref_t", SuperMessageRefTy);
Daniel Dunbar59e476b2009-08-03 17:06:42 +00004334
Fariborz Jahanian82c72e12009-02-03 23:49:23 +00004335 // SuperMessageRefPtrTy - LLVM for struct _super_message_ref_t*
Daniel Dunbar59e476b2009-08-03 17:06:42 +00004336 SuperMessageRefPtrTy = llvm::PointerType::getUnqual(SuperMessageRefTy);
4337
Daniel Dunbarb1559a42009-03-01 04:46:24 +00004338
4339 // struct objc_typeinfo {
4340 // const void** vtable; // objc_ehtype_vtable + 2
4341 // const char* name; // c++ typeinfo string
4342 // Class cls;
4343 // };
Owen Anderson758428f2009-08-05 23:18:46 +00004344 EHTypeTy = llvm::StructType::get(VMContext,
4345 llvm::PointerType::getUnqual(Int8PtrTy),
Daniel Dunbarb1559a42009-03-01 04:46:24 +00004346 Int8PtrTy,
4347 ClassnfABIPtrTy,
4348 NULL);
Daniel Dunbar76b7acc2009-03-02 06:08:11 +00004349 CGM.getModule().addTypeName("struct._objc_typeinfo", EHTypeTy);
Owen Anderson9793f0e2009-07-29 22:16:19 +00004350 EHTypePtrTy = llvm::PointerType::getUnqual(EHTypeTy);
Daniel Dunbar8b8683f2008-08-12 00:12:39 +00004351}
4352
Daniel Dunbar59e476b2009-08-03 17:06:42 +00004353llvm::Function *CGObjCNonFragileABIMac::ModuleInitFunction() {
Fariborz Jahanian71394042009-01-23 23:53:38 +00004354 FinishNonFragileABIModule();
Daniel Dunbar59e476b2009-08-03 17:06:42 +00004355
Fariborz Jahanian71394042009-01-23 23:53:38 +00004356 return NULL;
4357}
4358
Daniel Dunbar59e476b2009-08-03 17:06:42 +00004359void CGObjCNonFragileABIMac::AddModuleClassList(const
4360 std::vector<llvm::GlobalValue*>
4361 &Container,
Daniel Dunbar19573e72009-05-15 21:48:48 +00004362 const char *SymbolName,
4363 const char *SectionName) {
4364 unsigned NumClasses = Container.size();
Daniel Dunbar59e476b2009-08-03 17:06:42 +00004365
Daniel Dunbar19573e72009-05-15 21:48:48 +00004366 if (!NumClasses)
4367 return;
Daniel Dunbar59e476b2009-08-03 17:06:42 +00004368
Daniel Dunbar19573e72009-05-15 21:48:48 +00004369 std::vector<llvm::Constant*> Symbols(NumClasses);
4370 for (unsigned i=0; i<NumClasses; i++)
Owen Andersonade90fd2009-07-29 18:54:39 +00004371 Symbols[i] = llvm::ConstantExpr::getBitCast(Container[i],
Daniel Dunbar19573e72009-05-15 21:48:48 +00004372 ObjCTypes.Int8PtrTy);
Daniel Dunbar59e476b2009-08-03 17:06:42 +00004373 llvm::Constant* Init =
Owen Anderson9793f0e2009-07-29 22:16:19 +00004374 llvm::ConstantArray::get(llvm::ArrayType::get(ObjCTypes.Int8PtrTy,
Daniel Dunbar19573e72009-05-15 21:48:48 +00004375 NumClasses),
4376 Symbols);
Daniel Dunbar59e476b2009-08-03 17:06:42 +00004377
Daniel Dunbar19573e72009-05-15 21:48:48 +00004378 llvm::GlobalVariable *GV =
Owen Andersonc10c8d32009-07-08 19:05:04 +00004379 new llvm::GlobalVariable(CGM.getModule(), Init->getType(), false,
Daniel Dunbar19573e72009-05-15 21:48:48 +00004380 llvm::GlobalValue::InternalLinkage,
4381 Init,
Owen Andersonc10c8d32009-07-08 19:05:04 +00004382 SymbolName);
Daniel Dunbar710cb202010-04-25 20:39:32 +00004383 GV->setAlignment(CGM.getTargetData().getABITypeAlignment(Init->getType()));
Daniel Dunbar19573e72009-05-15 21:48:48 +00004384 GV->setSection(SectionName);
Chris Lattnerf56501c2009-07-17 23:57:13 +00004385 CGM.AddUsedGlobal(GV);
Daniel Dunbar19573e72009-05-15 21:48:48 +00004386}
Daniel Dunbar59e476b2009-08-03 17:06:42 +00004387
Fariborz Jahanian71394042009-01-23 23:53:38 +00004388void CGObjCNonFragileABIMac::FinishNonFragileABIModule() {
4389 // nonfragile abi has no module definition.
Daniel Dunbar59e476b2009-08-03 17:06:42 +00004390
Daniel Dunbar19573e72009-05-15 21:48:48 +00004391 // Build list of all implemented class addresses in array
Fariborz Jahanian279abd32009-01-30 20:55:31 +00004392 // L_OBJC_LABEL_CLASS_$.
Daniel Dunbar59e476b2009-08-03 17:06:42 +00004393 AddModuleClassList(DefinedClasses,
Daniel Dunbar19573e72009-05-15 21:48:48 +00004394 "\01L_OBJC_LABEL_CLASS_$",
4395 "__DATA, __objc_classlist, regular, no_dead_strip");
Fariborz Jahanian67260552009-11-17 21:37:35 +00004396
Fariborz Jahanian67260552009-11-17 21:37:35 +00004397 for (unsigned i = 0; i < DefinedClasses.size(); i++) {
4398 llvm::GlobalValue *IMPLGV = DefinedClasses[i];
4399 if (IMPLGV->getLinkage() != llvm::GlobalValue::ExternalWeakLinkage)
4400 continue;
4401 IMPLGV->setLinkage(llvm::GlobalValue::ExternalLinkage);
Fariborz Jahanian67260552009-11-17 21:37:35 +00004402 }
4403
Fariborz Jahaniana6227fd2009-12-01 18:25:24 +00004404 for (unsigned i = 0; i < DefinedMetaClasses.size(); i++) {
4405 llvm::GlobalValue *IMPLGV = DefinedMetaClasses[i];
4406 if (IMPLGV->getLinkage() != llvm::GlobalValue::ExternalWeakLinkage)
4407 continue;
4408 IMPLGV->setLinkage(llvm::GlobalValue::ExternalLinkage);
4409 }
Fariborz Jahanian67260552009-11-17 21:37:35 +00004410
Daniel Dunbar59e476b2009-08-03 17:06:42 +00004411 AddModuleClassList(DefinedNonLazyClasses,
Daniel Dunbar9a017d72009-05-15 22:33:15 +00004412 "\01L_OBJC_LABEL_NONLAZY_CLASS_$",
4413 "__DATA, __objc_nlclslist, regular, no_dead_strip");
Daniel Dunbar59e476b2009-08-03 17:06:42 +00004414
Fariborz Jahanian279abd32009-01-30 20:55:31 +00004415 // Build list of all implemented category addresses in array
4416 // L_OBJC_LABEL_CATEGORY_$.
Daniel Dunbar59e476b2009-08-03 17:06:42 +00004417 AddModuleClassList(DefinedCategories,
Daniel Dunbar19573e72009-05-15 21:48:48 +00004418 "\01L_OBJC_LABEL_CATEGORY_$",
4419 "__DATA, __objc_catlist, regular, no_dead_strip");
Daniel Dunbar59e476b2009-08-03 17:06:42 +00004420 AddModuleClassList(DefinedNonLazyCategories,
Daniel Dunbar9a017d72009-05-15 22:33:15 +00004421 "\01L_OBJC_LABEL_NONLAZY_CATEGORY_$",
4422 "__DATA, __objc_nlcatlist, regular, no_dead_strip");
Daniel Dunbar59e476b2009-08-03 17:06:42 +00004423
Daniel Dunbar5e639272010-04-25 20:39:01 +00004424 EmitImageInfo();
Fariborz Jahanian71394042009-01-23 23:53:38 +00004425}
4426
Fariborz Jahanian5d5ed2d2009-05-11 19:25:47 +00004427/// LegacyDispatchedSelector - Returns true if SEL is not in the list of
Fariborz Jahaniane4128642009-05-12 20:06:41 +00004428/// NonLegacyDispatchMethods; false otherwise. What this means is that
Daniel Dunbar59e476b2009-08-03 17:06:42 +00004429/// except for the 19 selectors in the list, we generate 32bit-style
Fariborz Jahanian5d5ed2d2009-05-11 19:25:47 +00004430/// message dispatch call for all the rest.
4431///
4432bool CGObjCNonFragileABIMac::LegacyDispatchedSelector(Selector Sel) {
Daniel Dunbarfca18c1b42010-04-24 17:56:46 +00004433 switch (CGM.getCodeGenOpts().getObjCDispatchMethod()) {
4434 default:
4435 assert(0 && "Invalid dispatch method!");
4436 case CodeGenOptions::Legacy:
Daniel Dunbarca5e3eb2010-02-01 21:07:33 +00004437 return true;
Daniel Dunbarfca18c1b42010-04-24 17:56:46 +00004438 case CodeGenOptions::NonLegacy:
Fariborz Jahaniandfb39832010-04-19 17:53:30 +00004439 return false;
Daniel Dunbarfca18c1b42010-04-24 17:56:46 +00004440 case CodeGenOptions::Mixed:
4441 break;
4442 }
4443
4444 // If so, see whether this selector is in the white-list of things which must
4445 // use the new dispatch convention. We lazily build a dense set for this.
Fariborz Jahaniane4128642009-05-12 20:06:41 +00004446 if (NonLegacyDispatchMethods.empty()) {
4447 NonLegacyDispatchMethods.insert(GetNullarySelector("alloc"));
4448 NonLegacyDispatchMethods.insert(GetNullarySelector("class"));
4449 NonLegacyDispatchMethods.insert(GetNullarySelector("self"));
4450 NonLegacyDispatchMethods.insert(GetNullarySelector("isFlipped"));
4451 NonLegacyDispatchMethods.insert(GetNullarySelector("length"));
4452 NonLegacyDispatchMethods.insert(GetNullarySelector("count"));
4453 NonLegacyDispatchMethods.insert(GetNullarySelector("retain"));
4454 NonLegacyDispatchMethods.insert(GetNullarySelector("release"));
4455 NonLegacyDispatchMethods.insert(GetNullarySelector("autorelease"));
4456 NonLegacyDispatchMethods.insert(GetNullarySelector("hash"));
Daniel Dunbar59e476b2009-08-03 17:06:42 +00004457
Fariborz Jahaniane4128642009-05-12 20:06:41 +00004458 NonLegacyDispatchMethods.insert(GetUnarySelector("allocWithZone"));
4459 NonLegacyDispatchMethods.insert(GetUnarySelector("isKindOfClass"));
4460 NonLegacyDispatchMethods.insert(GetUnarySelector("respondsToSelector"));
4461 NonLegacyDispatchMethods.insert(GetUnarySelector("objectForKey"));
4462 NonLegacyDispatchMethods.insert(GetUnarySelector("objectAtIndex"));
4463 NonLegacyDispatchMethods.insert(GetUnarySelector("isEqualToString"));
4464 NonLegacyDispatchMethods.insert(GetUnarySelector("isEqual"));
4465 NonLegacyDispatchMethods.insert(GetUnarySelector("addObject"));
Daniel Dunbar59e476b2009-08-03 17:06:42 +00004466 // "countByEnumeratingWithState:objects:count"
Fariborz Jahanian18801362009-05-13 16:19:02 +00004467 IdentifierInfo *KeyIdents[] = {
Daniel Dunbar59e476b2009-08-03 17:06:42 +00004468 &CGM.getContext().Idents.get("countByEnumeratingWithState"),
4469 &CGM.getContext().Idents.get("objects"),
4470 &CGM.getContext().Idents.get("count")
Fariborz Jahanian18801362009-05-13 16:19:02 +00004471 };
4472 NonLegacyDispatchMethods.insert(
4473 CGM.getContext().Selectors.getSelector(3, KeyIdents));
Fariborz Jahanian5d5ed2d2009-05-11 19:25:47 +00004474 }
Daniel Dunbarfca18c1b42010-04-24 17:56:46 +00004475
Fariborz Jahaniane4128642009-05-12 20:06:41 +00004476 return (NonLegacyDispatchMethods.count(Sel) == 0);
Fariborz Jahanian5d5ed2d2009-05-11 19:25:47 +00004477}
4478
Fariborz Jahanian9e3ad522009-01-24 20:21:50 +00004479// Metadata flags
4480enum MetaDataDlags {
4481 CLS = 0x0,
4482 CLS_META = 0x1,
4483 CLS_ROOT = 0x2,
Fariborz Jahanian99113fd2009-01-26 21:38:32 +00004484 OBJC2_CLS_HIDDEN = 0x10,
Fariborz Jahanian9e3ad522009-01-24 20:21:50 +00004485 CLS_EXCEPTION = 0x20
4486};
4487/// BuildClassRoTInitializer - generate meta-data for:
4488/// struct _class_ro_t {
4489/// uint32_t const flags;
4490/// uint32_t const instanceStart;
4491/// uint32_t const instanceSize;
4492/// uint32_t const reserved; // only when building for 64bit targets
4493/// const uint8_t * const ivarLayout;
4494/// const char *const name;
4495/// const struct _method_list_t * const baseMethods;
Fariborz Jahanian56b3b772009-01-29 19:24:30 +00004496/// const struct _protocol_list_t *const baseProtocols;
Fariborz Jahanian9e3ad522009-01-24 20:21:50 +00004497/// const struct _ivar_list_t *const ivars;
4498/// const uint8_t * const weakIvarLayout;
4499/// const struct _prop_list_t * const properties;
4500/// }
4501///
4502llvm::GlobalVariable * CGObjCNonFragileABIMac::BuildClassRoTInitializer(
Daniel Dunbar59e476b2009-08-03 17:06:42 +00004503 unsigned flags,
4504 unsigned InstanceStart,
4505 unsigned InstanceSize,
4506 const ObjCImplementationDecl *ID) {
Fariborz Jahanian9e3ad522009-01-24 20:21:50 +00004507 std::string ClassName = ID->getNameAsString();
4508 std::vector<llvm::Constant*> Values(10); // 11 for 64bit targets!
Owen Andersonb7a2fe62009-07-24 23:12:58 +00004509 Values[ 0] = llvm::ConstantInt::get(ObjCTypes.IntTy, flags);
4510 Values[ 1] = llvm::ConstantInt::get(ObjCTypes.IntTy, InstanceStart);
4511 Values[ 2] = llvm::ConstantInt::get(ObjCTypes.IntTy, InstanceSize);
Fariborz Jahanian9e3ad522009-01-24 20:21:50 +00004512 // FIXME. For 64bit targets add 0 here.
Daniel Dunbar59e476b2009-08-03 17:06:42 +00004513 Values[ 3] = (flags & CLS_META) ? GetIvarLayoutName(0, ObjCTypes)
4514 : BuildIvarLayout(ID, true);
Fariborz Jahanian9e3ad522009-01-24 20:21:50 +00004515 Values[ 4] = GetClassName(ID->getIdentifier());
Fariborz Jahanian99113fd2009-01-26 21:38:32 +00004516 // const struct _method_list_t * const baseMethods;
4517 std::vector<llvm::Constant*> Methods;
4518 std::string MethodListName("\01l_OBJC_$_");
4519 if (flags & CLS_META) {
4520 MethodListName += "CLASS_METHODS_" + ID->getNameAsString();
Daniel Dunbar59e476b2009-08-03 17:06:42 +00004521 for (ObjCImplementationDecl::classmeth_iterator
Argyrios Kyrtzidiscfbfe782009-06-30 02:36:12 +00004522 i = ID->classmeth_begin(), e = ID->classmeth_end(); i != e; ++i) {
Fariborz Jahanian99113fd2009-01-26 21:38:32 +00004523 // Class methods should always be defined.
4524 Methods.push_back(GetMethodConstant(*i));
4525 }
4526 } else {
4527 MethodListName += "INSTANCE_METHODS_" + ID->getNameAsString();
Daniel Dunbar59e476b2009-08-03 17:06:42 +00004528 for (ObjCImplementationDecl::instmeth_iterator
Argyrios Kyrtzidiscfbfe782009-06-30 02:36:12 +00004529 i = ID->instmeth_begin(), e = ID->instmeth_end(); i != e; ++i) {
Fariborz Jahanian99113fd2009-01-26 21:38:32 +00004530 // Instance methods should always be defined.
4531 Methods.push_back(GetMethodConstant(*i));
4532 }
Daniel Dunbar59e476b2009-08-03 17:06:42 +00004533 for (ObjCImplementationDecl::propimpl_iterator
Argyrios Kyrtzidiscfbfe782009-06-30 02:36:12 +00004534 i = ID->propimpl_begin(), e = ID->propimpl_end(); i != e; ++i) {
Fariborz Jahaniand27a8202009-01-28 22:46:49 +00004535 ObjCPropertyImplDecl *PID = *i;
Daniel Dunbar59e476b2009-08-03 17:06:42 +00004536
Fariborz Jahaniand27a8202009-01-28 22:46:49 +00004537 if (PID->getPropertyImplementation() == ObjCPropertyImplDecl::Synthesize){
4538 ObjCPropertyDecl *PD = PID->getPropertyDecl();
Daniel Dunbar59e476b2009-08-03 17:06:42 +00004539
Fariborz Jahaniand27a8202009-01-28 22:46:49 +00004540 if (ObjCMethodDecl *MD = PD->getGetterMethodDecl())
4541 if (llvm::Constant *C = GetMethodConstant(MD))
4542 Methods.push_back(C);
4543 if (ObjCMethodDecl *MD = PD->getSetterMethodDecl())
4544 if (llvm::Constant *C = GetMethodConstant(MD))
4545 Methods.push_back(C);
4546 }
4547 }
Fariborz Jahanian99113fd2009-01-26 21:38:32 +00004548 }
Daniel Dunbar59e476b2009-08-03 17:06:42 +00004549 Values[ 5] = EmitMethodList(MethodListName,
4550 "__DATA, __objc_const", Methods);
4551
Fariborz Jahanian56b3b772009-01-29 19:24:30 +00004552 const ObjCInterfaceDecl *OID = ID->getClassInterface();
4553 assert(OID && "CGObjCNonFragileABIMac::BuildClassRoTInitializer");
Daniel Dunbar59e476b2009-08-03 17:06:42 +00004554 Values[ 6] = EmitProtocolList("\01l_OBJC_CLASS_PROTOCOLS_$_"
Daniel Dunbar349e6fb2009-10-18 20:48:59 +00004555 + OID->getName(),
Fariborz Jahanian56b3b772009-01-29 19:24:30 +00004556 OID->protocol_begin(),
4557 OID->protocol_end());
Daniel Dunbar59e476b2009-08-03 17:06:42 +00004558
Fariborz Jahanian7415caa2009-01-27 19:38:51 +00004559 if (flags & CLS_META)
Owen Anderson0b75f232009-07-31 20:28:54 +00004560 Values[ 7] = llvm::Constant::getNullValue(ObjCTypes.IvarListnfABIPtrTy);
Fariborz Jahanian7415caa2009-01-27 19:38:51 +00004561 else
4562 Values[ 7] = EmitIvarList(ID);
Daniel Dunbar59e476b2009-08-03 17:06:42 +00004563 Values[ 8] = (flags & CLS_META) ? GetIvarLayoutName(0, ObjCTypes)
4564 : BuildIvarLayout(ID, false);
Fariborz Jahanian066347e2009-01-28 22:18:42 +00004565 if (flags & CLS_META)
Owen Anderson0b75f232009-07-31 20:28:54 +00004566 Values[ 9] = llvm::Constant::getNullValue(ObjCTypes.PropertyListPtrTy);
Fariborz Jahanian066347e2009-01-28 22:18:42 +00004567 else
Daniel Dunbar349e6fb2009-10-18 20:48:59 +00004568 Values[ 9] = EmitPropertyList("\01l_OBJC_$_PROP_LIST_" + ID->getName(),
4569 ID, ID->getClassInterface(), ObjCTypes);
Owen Anderson0e0189d2009-07-27 22:29:56 +00004570 llvm::Constant *Init = llvm::ConstantStruct::get(ObjCTypes.ClassRonfABITy,
Fariborz Jahanian9e3ad522009-01-24 20:21:50 +00004571 Values);
4572 llvm::GlobalVariable *CLASS_RO_GV =
Daniel Dunbar59e476b2009-08-03 17:06:42 +00004573 new llvm::GlobalVariable(CGM.getModule(), ObjCTypes.ClassRonfABITy, false,
4574 llvm::GlobalValue::InternalLinkage,
4575 Init,
4576 (flags & CLS_META) ?
4577 std::string("\01l_OBJC_METACLASS_RO_$_")+ClassName :
4578 std::string("\01l_OBJC_CLASS_RO_$_")+ClassName);
Fariborz Jahanianc22f2362009-01-31 02:43:27 +00004579 CLASS_RO_GV->setAlignment(
Daniel Dunbar710cb202010-04-25 20:39:32 +00004580 CGM.getTargetData().getABITypeAlignment(ObjCTypes.ClassRonfABITy));
Fariborz Jahanian40a4bcd2009-01-28 01:05:23 +00004581 CLASS_RO_GV->setSection("__DATA, __objc_const");
Fariborz Jahanian9e3ad522009-01-24 20:21:50 +00004582 return CLASS_RO_GV;
Fariborz Jahanian2612e142009-01-26 22:58:07 +00004583
Fariborz Jahanian9e3ad522009-01-24 20:21:50 +00004584}
4585
4586/// BuildClassMetaData - This routine defines that to-level meta-data
4587/// for the given ClassName for:
4588/// struct _class_t {
4589/// struct _class_t *isa;
4590/// struct _class_t * const superclass;
4591/// void *cache;
4592/// IMP *vtable;
4593/// struct class_ro_t *ro;
4594/// }
4595///
Fariborz Jahanian4723fb72009-01-24 21:21:53 +00004596llvm::GlobalVariable * CGObjCNonFragileABIMac::BuildClassMetaData(
Daniel Dunbar59e476b2009-08-03 17:06:42 +00004597 std::string &ClassName,
4598 llvm::Constant *IsAGV,
4599 llvm::Constant *SuperClassGV,
4600 llvm::Constant *ClassRoGV,
4601 bool HiddenVisibility) {
Fariborz Jahanian9e3ad522009-01-24 20:21:50 +00004602 std::vector<llvm::Constant*> Values(5);
4603 Values[0] = IsAGV;
Daniel Dunbar59e476b2009-08-03 17:06:42 +00004604 Values[1] = SuperClassGV;
4605 if (!Values[1])
4606 Values[1] = llvm::Constant::getNullValue(ObjCTypes.ClassnfABIPtrTy);
Fariborz Jahanian9e3ad522009-01-24 20:21:50 +00004607 Values[2] = ObjCEmptyCacheVar; // &ObjCEmptyCacheVar
4608 Values[3] = ObjCEmptyVtableVar; // &ObjCEmptyVtableVar
4609 Values[4] = ClassRoGV; // &CLASS_RO_GV
Daniel Dunbar59e476b2009-08-03 17:06:42 +00004610 llvm::Constant *Init = llvm::ConstantStruct::get(ObjCTypes.ClassnfABITy,
Fariborz Jahanian9e3ad522009-01-24 20:21:50 +00004611 Values);
Daniel Dunbarc6928bb2009-03-01 04:40:10 +00004612 llvm::GlobalVariable *GV = GetClassGlobal(ClassName);
4613 GV->setInitializer(Init);
Fariborz Jahanian04087232009-01-31 01:07:39 +00004614 GV->setSection("__DATA, __objc_data");
Fariborz Jahanianc22f2362009-01-31 02:43:27 +00004615 GV->setAlignment(
Daniel Dunbar710cb202010-04-25 20:39:32 +00004616 CGM.getTargetData().getABITypeAlignment(ObjCTypes.ClassnfABITy));
Fariborz Jahanian82208252009-01-31 00:59:10 +00004617 if (HiddenVisibility)
4618 GV->setVisibility(llvm::GlobalValue::HiddenVisibility);
Fariborz Jahanian4723fb72009-01-24 21:21:53 +00004619 return GV;
Fariborz Jahanian9e3ad522009-01-24 20:21:50 +00004620}
4621
Daniel Dunbar59e476b2009-08-03 17:06:42 +00004622bool
Fariborz Jahaniana6bed832009-05-21 01:03:45 +00004623CGObjCNonFragileABIMac::ImplementationIsNonLazy(const ObjCImplDecl *OD) const {
Argyrios Kyrtzidiscfbfe782009-06-30 02:36:12 +00004624 return OD->getClassMethod(GetNullarySelector("load")) != 0;
Daniel Dunbar9a017d72009-05-15 22:33:15 +00004625}
4626
Daniel Dunbar961202372009-05-03 12:57:56 +00004627void CGObjCNonFragileABIMac::GetClassSizeInfo(const ObjCImplementationDecl *OID,
Daniel Dunbar554fd792009-04-19 23:41:48 +00004628 uint32_t &InstanceStart,
4629 uint32_t &InstanceSize) {
Daniel Dunbar59e476b2009-08-03 17:06:42 +00004630 const ASTRecordLayout &RL =
Daniel Dunbar9252ee12009-05-04 21:26:30 +00004631 CGM.getContext().getASTObjCImplementationLayout(OID);
Daniel Dunbar59e476b2009-08-03 17:06:42 +00004632
Daniel Dunbar9b042e02009-05-04 23:23:09 +00004633 // InstanceSize is really instance end.
Anders Carlsson27b50132009-07-18 21:26:44 +00004634 InstanceSize = llvm::RoundUpToAlignment(RL.getDataSize(), 8) / 8;
Daniel Dunbar9b042e02009-05-04 23:23:09 +00004635
4636 // If there are no fields, the start is the same as the end.
4637 if (!RL.getFieldCount())
4638 InstanceStart = InstanceSize;
4639 else
4640 InstanceStart = RL.getFieldOffset(0) / 8;
Daniel Dunbar554fd792009-04-19 23:41:48 +00004641}
4642
Fariborz Jahanian71394042009-01-23 23:53:38 +00004643void CGObjCNonFragileABIMac::GenerateClass(const ObjCImplementationDecl *ID) {
4644 std::string ClassName = ID->getNameAsString();
4645 if (!ObjCEmptyCacheVar) {
4646 ObjCEmptyCacheVar = new llvm::GlobalVariable(
Daniel Dunbar59e476b2009-08-03 17:06:42 +00004647 CGM.getModule(),
4648 ObjCTypes.CacheTy,
4649 false,
4650 llvm::GlobalValue::ExternalLinkage,
4651 0,
4652 "_objc_empty_cache");
4653
Fariborz Jahanian71394042009-01-23 23:53:38 +00004654 ObjCEmptyVtableVar = new llvm::GlobalVariable(
Daniel Dunbar59e476b2009-08-03 17:06:42 +00004655 CGM.getModule(),
4656 ObjCTypes.ImpnfABITy,
4657 false,
4658 llvm::GlobalValue::ExternalLinkage,
4659 0,
4660 "_objc_empty_vtable");
Fariborz Jahanian71394042009-01-23 23:53:38 +00004661 }
Daniel Dunbar59e476b2009-08-03 17:06:42 +00004662 assert(ID->getClassInterface() &&
Fariborz Jahanian99113fd2009-01-26 21:38:32 +00004663 "CGObjCNonFragileABIMac::GenerateClass - class is 0");
Daniel Dunbare3f5cfc2009-04-20 20:18:54 +00004664 // FIXME: Is this correct (that meta class size is never computed)?
Daniel Dunbar59e476b2009-08-03 17:06:42 +00004665 uint32_t InstanceStart =
Duncan Sandsc76fe8b2009-05-09 07:08:47 +00004666 CGM.getTargetData().getTypeAllocSize(ObjCTypes.ClassnfABITy);
Fariborz Jahanian9e3ad522009-01-24 20:21:50 +00004667 uint32_t InstanceSize = InstanceStart;
4668 uint32_t flags = CLS_META;
Daniel Dunbar15894b72009-04-07 05:48:37 +00004669 std::string ObjCMetaClassName(getMetaclassSymbolPrefix());
4670 std::string ObjCClassName(getClassSymbolPrefix());
Daniel Dunbar59e476b2009-08-03 17:06:42 +00004671
Fariborz Jahanian9e3ad522009-01-24 20:21:50 +00004672 llvm::GlobalVariable *SuperClassGV, *IsAGV;
Daniel Dunbar59e476b2009-08-03 17:06:42 +00004673
4674 bool classIsHidden =
Daniel Dunbarf5f359f2009-04-14 06:00:08 +00004675 CGM.getDeclVisibilityMode(ID->getClassInterface()) == LangOptions::Hidden;
Fariborz Jahanian82208252009-01-31 00:59:10 +00004676 if (classIsHidden)
Fariborz Jahanian99113fd2009-01-26 21:38:32 +00004677 flags |= OBJC2_CLS_HIDDEN;
Fariborz Jahanian0dec1e02010-04-28 21:28:56 +00004678 if (ID->getNumIvarInitializers())
4679 flags |= eClassFlags_ABI2_HasCXXStructors;
Fariborz Jahanian99113fd2009-01-26 21:38:32 +00004680 if (!ID->getClassInterface()->getSuperClass()) {
Fariborz Jahanian9e3ad522009-01-24 20:21:50 +00004681 // class is root
4682 flags |= CLS_ROOT;
Daniel Dunbarc6928bb2009-03-01 04:40:10 +00004683 SuperClassGV = GetClassGlobal(ObjCClassName + ClassName);
Fariborz Jahanian899e7eb2009-04-14 18:41:56 +00004684 IsAGV = GetClassGlobal(ObjCMetaClassName + ClassName);
Fariborz Jahanian9e3ad522009-01-24 20:21:50 +00004685 } else {
Fariborz Jahanian4723fb72009-01-24 21:21:53 +00004686 // Has a root. Current class is not a root.
Fariborz Jahanian03b300b2009-02-26 18:23:47 +00004687 const ObjCInterfaceDecl *Root = ID->getClassInterface();
4688 while (const ObjCInterfaceDecl *Super = Root->getSuperClass())
4689 Root = Super;
Fariborz Jahanian899e7eb2009-04-14 18:41:56 +00004690 IsAGV = GetClassGlobal(ObjCMetaClassName + Root->getNameAsString());
Fariborz Jahaniana6227fd2009-12-01 18:25:24 +00004691 if (Root->hasAttr<WeakImportAttr>())
4692 IsAGV->setLinkage(llvm::GlobalValue::ExternalWeakLinkage);
Fariborz Jahanian03b300b2009-02-26 18:23:47 +00004693 // work on super class metadata symbol.
Daniel Dunbar59e476b2009-08-03 17:06:42 +00004694 std::string SuperClassName =
Fariborz Jahaniana6227fd2009-12-01 18:25:24 +00004695 ObjCMetaClassName +
4696 ID->getClassInterface()->getSuperClass()->getNameAsString();
Fariborz Jahanian899e7eb2009-04-14 18:41:56 +00004697 SuperClassGV = GetClassGlobal(SuperClassName);
Fariborz Jahanian67260552009-11-17 21:37:35 +00004698 if (ID->getClassInterface()->getSuperClass()->hasAttr<WeakImportAttr>())
4699 SuperClassGV->setLinkage(llvm::GlobalValue::ExternalWeakLinkage);
Fariborz Jahanian9e3ad522009-01-24 20:21:50 +00004700 }
4701 llvm::GlobalVariable *CLASS_RO_GV = BuildClassRoTInitializer(flags,
4702 InstanceStart,
4703 InstanceSize,ID);
Fariborz Jahanian4723fb72009-01-24 21:21:53 +00004704 std::string TClassName = ObjCMetaClassName + ClassName;
Daniel Dunbar59e476b2009-08-03 17:06:42 +00004705 llvm::GlobalVariable *MetaTClass =
Fariborz Jahanian82208252009-01-31 00:59:10 +00004706 BuildClassMetaData(TClassName, IsAGV, SuperClassGV, CLASS_RO_GV,
4707 classIsHidden);
Fariborz Jahanian67260552009-11-17 21:37:35 +00004708 DefinedMetaClasses.push_back(MetaTClass);
Daniel Dunbar15894b72009-04-07 05:48:37 +00004709
Fariborz Jahanian4723fb72009-01-24 21:21:53 +00004710 // Metadata for the class
4711 flags = CLS;
Fariborz Jahanian82208252009-01-31 00:59:10 +00004712 if (classIsHidden)
Fariborz Jahanian99113fd2009-01-26 21:38:32 +00004713 flags |= OBJC2_CLS_HIDDEN;
Fariborz Jahanian0dec1e02010-04-28 21:28:56 +00004714 if (ID->getNumIvarInitializers())
4715 flags |= eClassFlags_ABI2_HasCXXStructors;
Daniel Dunbar8f28d012009-04-08 04:21:03 +00004716
Douglas Gregor78bd61f2009-06-18 16:11:24 +00004717 if (hasObjCExceptionAttribute(CGM.getContext(), ID->getClassInterface()))
Daniel Dunbar8f28d012009-04-08 04:21:03 +00004718 flags |= CLS_EXCEPTION;
4719
Fariborz Jahanian99113fd2009-01-26 21:38:32 +00004720 if (!ID->getClassInterface()->getSuperClass()) {
Fariborz Jahanian4723fb72009-01-24 21:21:53 +00004721 flags |= CLS_ROOT;
4722 SuperClassGV = 0;
Chris Lattnerb433b272009-04-19 06:02:28 +00004723 } else {
Fariborz Jahanian4723fb72009-01-24 21:21:53 +00004724 // Has a root. Current class is not a root.
Fariborz Jahanian03b300b2009-02-26 18:23:47 +00004725 std::string RootClassName =
Fariborz Jahanian4723fb72009-01-24 21:21:53 +00004726 ID->getClassInterface()->getSuperClass()->getNameAsString();
Daniel Dunbarc6928bb2009-03-01 04:40:10 +00004727 SuperClassGV = GetClassGlobal(ObjCClassName + RootClassName);
Fariborz Jahanian67260552009-11-17 21:37:35 +00004728 if (ID->getClassInterface()->getSuperClass()->hasAttr<WeakImportAttr>())
4729 SuperClassGV->setLinkage(llvm::GlobalValue::ExternalWeakLinkage);
Fariborz Jahanian4723fb72009-01-24 21:21:53 +00004730 }
Daniel Dunbar961202372009-05-03 12:57:56 +00004731 GetClassSizeInfo(ID, InstanceStart, InstanceSize);
Fariborz Jahanian4723fb72009-01-24 21:21:53 +00004732 CLASS_RO_GV = BuildClassRoTInitializer(flags,
Fariborz Jahaniana887e632009-01-24 23:43:01 +00004733 InstanceStart,
Daniel Dunbar59e476b2009-08-03 17:06:42 +00004734 InstanceSize,
Fariborz Jahaniana887e632009-01-24 23:43:01 +00004735 ID);
Daniel Dunbar59e476b2009-08-03 17:06:42 +00004736
Fariborz Jahanian4723fb72009-01-24 21:21:53 +00004737 TClassName = ObjCClassName + ClassName;
Daniel Dunbar59e476b2009-08-03 17:06:42 +00004738 llvm::GlobalVariable *ClassMD =
Fariborz Jahanian82208252009-01-31 00:59:10 +00004739 BuildClassMetaData(TClassName, MetaTClass, SuperClassGV, CLASS_RO_GV,
4740 classIsHidden);
Fariborz Jahanian279abd32009-01-30 20:55:31 +00004741 DefinedClasses.push_back(ClassMD);
Daniel Dunbar8f28d012009-04-08 04:21:03 +00004742
Daniel Dunbar9a017d72009-05-15 22:33:15 +00004743 // Determine if this class is also "non-lazy".
4744 if (ImplementationIsNonLazy(ID))
4745 DefinedNonLazyClasses.push_back(ClassMD);
4746
Daniel Dunbar8f28d012009-04-08 04:21:03 +00004747 // Force the definition of the EHType if necessary.
4748 if (flags & CLS_EXCEPTION)
4749 GetInterfaceEHType(ID->getClassInterface(), true);
Fariborz Jahanian71394042009-01-23 23:53:38 +00004750}
4751
Fariborz Jahanian097feda2009-01-30 18:58:59 +00004752/// GenerateProtocolRef - This routine is called to generate code for
4753/// a protocol reference expression; as in:
4754/// @code
4755/// @protocol(Proto1);
4756/// @endcode
4757/// It generates a weak reference to l_OBJC_PROTOCOL_REFERENCE_$_Proto1
4758/// which will hold address of the protocol meta-data.
4759///
4760llvm::Value *CGObjCNonFragileABIMac::GenerateProtocolRef(CGBuilderTy &Builder,
Daniel Dunbar59e476b2009-08-03 17:06:42 +00004761 const ObjCProtocolDecl *PD) {
4762
Fariborz Jahanian464423d2009-04-10 18:47:34 +00004763 // This routine is called for @protocol only. So, we must build definition
4764 // of protocol's meta-data (not a reference to it!)
4765 //
Daniel Dunbar59e476b2009-08-03 17:06:42 +00004766 llvm::Constant *Init =
4767 llvm::ConstantExpr::getBitCast(GetOrEmitProtocol(PD),
4768 ObjCTypes.ExternalProtocolPtrTy);
4769
Fariborz Jahanian097feda2009-01-30 18:58:59 +00004770 std::string ProtocolName("\01l_OBJC_PROTOCOL_REFERENCE_$_");
4771 ProtocolName += PD->getNameAsCString();
Daniel Dunbar59e476b2009-08-03 17:06:42 +00004772
Fariborz Jahanian097feda2009-01-30 18:58:59 +00004773 llvm::GlobalVariable *PTGV = CGM.getModule().getGlobalVariable(ProtocolName);
4774 if (PTGV)
Daniel Dunbarc76493a2009-11-29 21:23:36 +00004775 return Builder.CreateLoad(PTGV, "tmp");
Fariborz Jahanian097feda2009-01-30 18:58:59 +00004776 PTGV = new llvm::GlobalVariable(
Daniel Dunbar59e476b2009-08-03 17:06:42 +00004777 CGM.getModule(),
4778 Init->getType(), false,
4779 llvm::GlobalValue::WeakAnyLinkage,
4780 Init,
4781 ProtocolName);
Fariborz Jahanian097feda2009-01-30 18:58:59 +00004782 PTGV->setSection("__DATA, __objc_protorefs, coalesced, no_dead_strip");
4783 PTGV->setVisibility(llvm::GlobalValue::HiddenVisibility);
Chris Lattnerf56501c2009-07-17 23:57:13 +00004784 CGM.AddUsedGlobal(PTGV);
Daniel Dunbarc76493a2009-11-29 21:23:36 +00004785 return Builder.CreateLoad(PTGV, "tmp");
Fariborz Jahanian097feda2009-01-30 18:58:59 +00004786}
4787
Fariborz Jahanian0c8d0602009-01-26 18:32:24 +00004788/// GenerateCategory - Build metadata for a category implementation.
4789/// struct _category_t {
4790/// const char * const name;
4791/// struct _class_t *const cls;
4792/// const struct _method_list_t * const instance_methods;
4793/// const struct _method_list_t * const class_methods;
4794/// const struct _protocol_list_t * const protocols;
4795/// const struct _prop_list_t * const properties;
4796/// }
4797///
Daniel Dunbar9a017d72009-05-15 22:33:15 +00004798void CGObjCNonFragileABIMac::GenerateCategory(const ObjCCategoryImplDecl *OCD) {
Fariborz Jahanian0c8d0602009-01-26 18:32:24 +00004799 const ObjCInterfaceDecl *Interface = OCD->getClassInterface();
Fariborz Jahanian2612e142009-01-26 22:58:07 +00004800 const char *Prefix = "\01l_OBJC_$_CATEGORY_";
Daniel Dunbar59e476b2009-08-03 17:06:42 +00004801 std::string ExtCatName(Prefix + Interface->getNameAsString()+
4802 "_$_" + OCD->getNameAsString());
4803 std::string ExtClassName(getClassSymbolPrefix() +
Daniel Dunbar15894b72009-04-07 05:48:37 +00004804 Interface->getNameAsString());
Daniel Dunbar59e476b2009-08-03 17:06:42 +00004805
Fariborz Jahanian0c8d0602009-01-26 18:32:24 +00004806 std::vector<llvm::Constant*> Values(6);
4807 Values[0] = GetClassName(OCD->getIdentifier());
4808 // meta-class entry symbol
Daniel Dunbarc6928bb2009-03-01 04:40:10 +00004809 llvm::GlobalVariable *ClassGV = GetClassGlobal(ExtClassName);
Fariborz Jahanian3ad8dcf2009-11-17 22:02:21 +00004810 if (Interface->hasAttr<WeakImportAttr>())
4811 ClassGV->setLinkage(llvm::GlobalValue::ExternalWeakLinkage);
4812
Fariborz Jahanian0c8d0602009-01-26 18:32:24 +00004813 Values[1] = ClassGV;
Fariborz Jahanian2612e142009-01-26 22:58:07 +00004814 std::vector<llvm::Constant*> Methods;
4815 std::string MethodListName(Prefix);
Daniel Dunbar59e476b2009-08-03 17:06:42 +00004816 MethodListName += "INSTANCE_METHODS_" + Interface->getNameAsString() +
Fariborz Jahanian2612e142009-01-26 22:58:07 +00004817 "_$_" + OCD->getNameAsString();
Daniel Dunbar59e476b2009-08-03 17:06:42 +00004818
4819 for (ObjCCategoryImplDecl::instmeth_iterator
Argyrios Kyrtzidiscfbfe782009-06-30 02:36:12 +00004820 i = OCD->instmeth_begin(), e = OCD->instmeth_end(); i != e; ++i) {
Fariborz Jahanian2612e142009-01-26 22:58:07 +00004821 // Instance methods should always be defined.
4822 Methods.push_back(GetMethodConstant(*i));
4823 }
Daniel Dunbar59e476b2009-08-03 17:06:42 +00004824
4825 Values[2] = EmitMethodList(MethodListName,
4826 "__DATA, __objc_const",
Fariborz Jahanian2612e142009-01-26 22:58:07 +00004827 Methods);
4828
4829 MethodListName = Prefix;
4830 MethodListName += "CLASS_METHODS_" + Interface->getNameAsString() + "_$_" +
4831 OCD->getNameAsString();
4832 Methods.clear();
Daniel Dunbar59e476b2009-08-03 17:06:42 +00004833 for (ObjCCategoryImplDecl::classmeth_iterator
Argyrios Kyrtzidiscfbfe782009-06-30 02:36:12 +00004834 i = OCD->classmeth_begin(), e = OCD->classmeth_end(); i != e; ++i) {
Fariborz Jahanian2612e142009-01-26 22:58:07 +00004835 // Class methods should always be defined.
4836 Methods.push_back(GetMethodConstant(*i));
4837 }
Daniel Dunbar59e476b2009-08-03 17:06:42 +00004838
4839 Values[3] = EmitMethodList(MethodListName,
4840 "__DATA, __objc_const",
Fariborz Jahanian2612e142009-01-26 22:58:07 +00004841 Methods);
Daniel Dunbar59e476b2009-08-03 17:06:42 +00004842 const ObjCCategoryDecl *Category =
Fariborz Jahanian066347e2009-01-28 22:18:42 +00004843 Interface->FindCategoryDeclaration(OCD->getIdentifier());
Fariborz Jahaniand8fc1052009-02-13 17:52:22 +00004844 if (Category) {
Daniel Dunbar349e6fb2009-10-18 20:48:59 +00004845 llvm::SmallString<256> ExtName;
4846 llvm::raw_svector_ostream(ExtName) << Interface->getName() << "_$_"
4847 << OCD->getName();
Fariborz Jahaniand8fc1052009-02-13 17:52:22 +00004848 Values[4] = EmitProtocolList("\01l_OBJC_CATEGORY_PROTOCOLS_$_"
Daniel Dunbar349e6fb2009-10-18 20:48:59 +00004849 + Interface->getName() + "_$_"
4850 + Category->getName(),
Fariborz Jahaniand8fc1052009-02-13 17:52:22 +00004851 Category->protocol_begin(),
4852 Category->protocol_end());
Daniel Dunbar349e6fb2009-10-18 20:48:59 +00004853 Values[5] = EmitPropertyList("\01l_OBJC_$_PROP_LIST_" + ExtName.str(),
4854 OCD, Category, ObjCTypes);
Mike Stump658fe022009-07-30 22:28:39 +00004855 } else {
Owen Anderson0b75f232009-07-31 20:28:54 +00004856 Values[4] = llvm::Constant::getNullValue(ObjCTypes.ProtocolListnfABIPtrTy);
4857 Values[5] = llvm::Constant::getNullValue(ObjCTypes.PropertyListPtrTy);
Fariborz Jahaniand8fc1052009-02-13 17:52:22 +00004858 }
Daniel Dunbar59e476b2009-08-03 17:06:42 +00004859
4860 llvm::Constant *Init =
4861 llvm::ConstantStruct::get(ObjCTypes.CategorynfABITy,
Fariborz Jahanian0c8d0602009-01-26 18:32:24 +00004862 Values);
4863 llvm::GlobalVariable *GCATV
Daniel Dunbar59e476b2009-08-03 17:06:42 +00004864 = new llvm::GlobalVariable(CGM.getModule(), ObjCTypes.CategorynfABITy,
Fariborz Jahanian0c8d0602009-01-26 18:32:24 +00004865 false,
4866 llvm::GlobalValue::InternalLinkage,
4867 Init,
Owen Andersonc10c8d32009-07-08 19:05:04 +00004868 ExtCatName);
Fariborz Jahanianc22f2362009-01-31 02:43:27 +00004869 GCATV->setAlignment(
Daniel Dunbar710cb202010-04-25 20:39:32 +00004870 CGM.getTargetData().getABITypeAlignment(ObjCTypes.CategorynfABITy));
Fariborz Jahanian40a4bcd2009-01-28 01:05:23 +00004871 GCATV->setSection("__DATA, __objc_const");
Chris Lattnerf56501c2009-07-17 23:57:13 +00004872 CGM.AddUsedGlobal(GCATV);
Fariborz Jahanian0c8d0602009-01-26 18:32:24 +00004873 DefinedCategories.push_back(GCATV);
Daniel Dunbar9a017d72009-05-15 22:33:15 +00004874
4875 // Determine if this category is also "non-lazy".
4876 if (ImplementationIsNonLazy(OCD))
4877 DefinedNonLazyCategories.push_back(GCATV);
Fariborz Jahanian0c8d0602009-01-26 18:32:24 +00004878}
Fariborz Jahanian99113fd2009-01-26 21:38:32 +00004879
4880/// GetMethodConstant - Return a struct objc_method constant for the
4881/// given method if it has been defined. The result is null if the
4882/// method has not been defined. The return value has type MethodPtrTy.
4883llvm::Constant *CGObjCNonFragileABIMac::GetMethodConstant(
Daniel Dunbar59e476b2009-08-03 17:06:42 +00004884 const ObjCMethodDecl *MD) {
Fariborz Jahanian99113fd2009-01-26 21:38:32 +00004885 // FIXME: Use DenseMap::lookup
4886 llvm::Function *Fn = MethodDefinitions[MD];
4887 if (!Fn)
4888 return 0;
Daniel Dunbar59e476b2009-08-03 17:06:42 +00004889
Fariborz Jahanian99113fd2009-01-26 21:38:32 +00004890 std::vector<llvm::Constant*> Method(3);
Daniel Dunbar59e476b2009-08-03 17:06:42 +00004891 Method[0] =
Owen Andersonade90fd2009-07-29 18:54:39 +00004892 llvm::ConstantExpr::getBitCast(GetMethodVarName(MD->getSelector()),
Daniel Dunbar59e476b2009-08-03 17:06:42 +00004893 ObjCTypes.SelectorPtrTy);
Fariborz Jahanian99113fd2009-01-26 21:38:32 +00004894 Method[1] = GetMethodVarType(MD);
Owen Andersonade90fd2009-07-29 18:54:39 +00004895 Method[2] = llvm::ConstantExpr::getBitCast(Fn, ObjCTypes.Int8PtrTy);
Owen Anderson0e0189d2009-07-27 22:29:56 +00004896 return llvm::ConstantStruct::get(ObjCTypes.MethodTy, Method);
Fariborz Jahanian99113fd2009-01-26 21:38:32 +00004897}
4898
4899/// EmitMethodList - Build meta-data for method declarations
4900/// struct _method_list_t {
4901/// uint32_t entsize; // sizeof(struct _objc_method)
4902/// uint32_t method_count;
4903/// struct _objc_method method_list[method_count];
4904/// }
4905///
Daniel Dunbar349e6fb2009-10-18 20:48:59 +00004906llvm::Constant *CGObjCNonFragileABIMac::EmitMethodList(llvm::Twine Name,
4907 const char *Section,
4908 const ConstantVector &Methods) {
Fariborz Jahanian99113fd2009-01-26 21:38:32 +00004909 // Return null for empty list.
4910 if (Methods.empty())
Owen Anderson0b75f232009-07-31 20:28:54 +00004911 return llvm::Constant::getNullValue(ObjCTypes.MethodListnfABIPtrTy);
Daniel Dunbar59e476b2009-08-03 17:06:42 +00004912
Fariborz Jahanian99113fd2009-01-26 21:38:32 +00004913 std::vector<llvm::Constant*> Values(3);
4914 // sizeof(struct _objc_method)
Duncan Sandsc76fe8b2009-05-09 07:08:47 +00004915 unsigned Size = CGM.getTargetData().getTypeAllocSize(ObjCTypes.MethodTy);
Owen Andersonb7a2fe62009-07-24 23:12:58 +00004916 Values[0] = llvm::ConstantInt::get(ObjCTypes.IntTy, Size);
Fariborz Jahanian99113fd2009-01-26 21:38:32 +00004917 // method_count
Owen Andersonb7a2fe62009-07-24 23:12:58 +00004918 Values[1] = llvm::ConstantInt::get(ObjCTypes.IntTy, Methods.size());
Owen Anderson9793f0e2009-07-29 22:16:19 +00004919 llvm::ArrayType *AT = llvm::ArrayType::get(ObjCTypes.MethodTy,
Fariborz Jahanian99113fd2009-01-26 21:38:32 +00004920 Methods.size());
Owen Anderson47034e12009-07-28 18:33:04 +00004921 Values[2] = llvm::ConstantArray::get(AT, Methods);
Nick Lewycky41eaf0a2009-09-19 20:00:52 +00004922 llvm::Constant *Init = llvm::ConstantStruct::get(VMContext, Values, false);
Daniel Dunbar59e476b2009-08-03 17:06:42 +00004923
Fariborz Jahanian99113fd2009-01-26 21:38:32 +00004924 llvm::GlobalVariable *GV =
Owen Andersonc10c8d32009-07-08 19:05:04 +00004925 new llvm::GlobalVariable(CGM.getModule(), Init->getType(), false,
Fariborz Jahanian99113fd2009-01-26 21:38:32 +00004926 llvm::GlobalValue::InternalLinkage,
4927 Init,
Owen Andersonc10c8d32009-07-08 19:05:04 +00004928 Name);
Fariborz Jahanianc22f2362009-01-31 02:43:27 +00004929 GV->setAlignment(
Daniel Dunbar710cb202010-04-25 20:39:32 +00004930 CGM.getTargetData().getABITypeAlignment(Init->getType()));
Fariborz Jahanian99113fd2009-01-26 21:38:32 +00004931 GV->setSection(Section);
Chris Lattnerf56501c2009-07-17 23:57:13 +00004932 CGM.AddUsedGlobal(GV);
Owen Andersonade90fd2009-07-29 18:54:39 +00004933 return llvm::ConstantExpr::getBitCast(GV,
Fariborz Jahanian99113fd2009-01-26 21:38:32 +00004934 ObjCTypes.MethodListnfABIPtrTy);
4935}
Fariborz Jahanian7415caa2009-01-27 19:38:51 +00004936
Fariborz Jahanian4e7ae062009-02-10 20:21:06 +00004937/// ObjCIvarOffsetVariable - Returns the ivar offset variable for
4938/// the given ivar.
Daniel Dunbar8c7f9812010-04-02 21:14:02 +00004939llvm::GlobalVariable *
4940CGObjCNonFragileABIMac::ObjCIvarOffsetVariable(const ObjCInterfaceDecl *ID,
4941 const ObjCIvarDecl *Ivar) {
4942 const ObjCInterfaceDecl *Container = Ivar->getContainingInterface();
Daniel Dunbar3f86b9c2009-05-05 00:36:57 +00004943 std::string Name = "OBJC_IVAR_$_" + Container->getNameAsString() +
Douglas Gregorbcced4e2009-04-09 21:40:53 +00004944 '.' + Ivar->getNameAsString();
Daniel Dunbar59e476b2009-08-03 17:06:42 +00004945 llvm::GlobalVariable *IvarOffsetGV =
Fariborz Jahanian4e7ae062009-02-10 20:21:06 +00004946 CGM.getModule().getGlobalVariable(Name);
4947 if (!IvarOffsetGV)
Daniel Dunbar59e476b2009-08-03 17:06:42 +00004948 IvarOffsetGV =
Owen Andersonc10c8d32009-07-08 19:05:04 +00004949 new llvm::GlobalVariable(CGM.getModule(), ObjCTypes.LongTy,
Fariborz Jahanian4e7ae062009-02-10 20:21:06 +00004950 false,
4951 llvm::GlobalValue::ExternalLinkage,
4952 0,
Owen Andersonc10c8d32009-07-08 19:05:04 +00004953 Name);
Fariborz Jahanian4e7ae062009-02-10 20:21:06 +00004954 return IvarOffsetGV;
4955}
4956
Daniel Dunbar8c7f9812010-04-02 21:14:02 +00004957llvm::Constant *
4958CGObjCNonFragileABIMac::EmitIvarOffsetVar(const ObjCInterfaceDecl *ID,
4959 const ObjCIvarDecl *Ivar,
4960 unsigned long int Offset) {
Daniel Dunbarbf90b332009-04-19 00:44:02 +00004961 llvm::GlobalVariable *IvarOffsetGV = ObjCIvarOffsetVariable(ID, Ivar);
Daniel Dunbar59e476b2009-08-03 17:06:42 +00004962 IvarOffsetGV->setInitializer(llvm::ConstantInt::get(ObjCTypes.LongTy,
Daniel Dunbarbf90b332009-04-19 00:44:02 +00004963 Offset));
Fariborz Jahanianc22f2362009-01-31 02:43:27 +00004964 IvarOffsetGV->setAlignment(
Daniel Dunbar710cb202010-04-25 20:39:32 +00004965 CGM.getTargetData().getABITypeAlignment(ObjCTypes.LongTy));
Daniel Dunbarbf90b332009-04-19 00:44:02 +00004966
Mike Stump18bb9282009-05-16 07:57:57 +00004967 // FIXME: This matches gcc, but shouldn't the visibility be set on the use as
4968 // well (i.e., in ObjCIvarOffsetVariable).
Daniel Dunbarbf90b332009-04-19 00:44:02 +00004969 if (Ivar->getAccessControl() == ObjCIvarDecl::Private ||
4970 Ivar->getAccessControl() == ObjCIvarDecl::Package ||
4971 CGM.getDeclVisibilityMode(ID) == LangOptions::Hidden)
Fariborz Jahanian3d3426f2009-01-28 01:36:42 +00004972 IvarOffsetGV->setVisibility(llvm::GlobalValue::HiddenVisibility);
Daniel Dunbarf5f359f2009-04-14 06:00:08 +00004973 else
Fariborz Jahanianbc3c77b2009-04-06 18:30:00 +00004974 IvarOffsetGV->setVisibility(llvm::GlobalValue::DefaultVisibility);
Fariborz Jahanian40a4bcd2009-01-28 01:05:23 +00004975 IvarOffsetGV->setSection("__DATA, __objc_const");
Fariborz Jahanianc88a70d2009-02-03 00:09:52 +00004976 return IvarOffsetGV;
Fariborz Jahanian40a4bcd2009-01-28 01:05:23 +00004977}
4978
Fariborz Jahanian7415caa2009-01-27 19:38:51 +00004979/// EmitIvarList - Emit the ivar list for the given
Daniel Dunbar508a7dd2009-04-18 08:51:00 +00004980/// implementation. The return value has type
Fariborz Jahanian7415caa2009-01-27 19:38:51 +00004981/// IvarListnfABIPtrTy.
4982/// struct _ivar_t {
4983/// unsigned long int *offset; // pointer to ivar offset location
4984/// char *name;
4985/// char *type;
4986/// uint32_t alignment;
4987/// uint32_t size;
4988/// }
4989/// struct _ivar_list_t {
4990/// uint32 entsize; // sizeof(struct _ivar_t)
4991/// uint32 count;
4992/// struct _iver_t list[count];
4993/// }
4994///
Daniel Dunbarf5c18462009-04-20 06:54:31 +00004995
Fariborz Jahanian7415caa2009-01-27 19:38:51 +00004996llvm::Constant *CGObjCNonFragileABIMac::EmitIvarList(
Daniel Dunbar59e476b2009-08-03 17:06:42 +00004997 const ObjCImplementationDecl *ID) {
4998
Fariborz Jahanian7415caa2009-01-27 19:38:51 +00004999 std::vector<llvm::Constant*> Ivars, Ivar(5);
Daniel Dunbar59e476b2009-08-03 17:06:42 +00005000
Fariborz Jahanian7415caa2009-01-27 19:38:51 +00005001 const ObjCInterfaceDecl *OID = ID->getClassInterface();
5002 assert(OID && "CGObjCNonFragileABIMac::EmitIvarList - null interface");
Daniel Dunbar59e476b2009-08-03 17:06:42 +00005003
Fariborz Jahanian40a4bcd2009-01-28 01:05:23 +00005004 // FIXME. Consolidate this with similar code in GenerateClass.
Daniel Dunbar59e476b2009-08-03 17:06:42 +00005005
Daniel Dunbarae032262009-04-20 00:33:43 +00005006 // Collect declared and synthesized ivars in a small vector.
Fariborz Jahanian63a224a2009-03-31 18:11:23 +00005007 llvm::SmallVector<ObjCIvarDecl*, 16> OIvars;
Fariborz Jahanian7c809592009-06-04 01:19:09 +00005008 CGM.getContext().ShallowCollectObjCIvars(OID, OIvars);
Daniel Dunbar59e476b2009-08-03 17:06:42 +00005009
Daniel Dunbarf5c18462009-04-20 06:54:31 +00005010 for (unsigned i = 0, e = OIvars.size(); i != e; ++i) {
5011 ObjCIvarDecl *IVD = OIvars[i];
Fariborz Jahanian7c809592009-06-04 01:19:09 +00005012 // Ignore unnamed bit-fields.
5013 if (!IVD->getDeclName())
5014 continue;
Daniel Dunbar59e476b2009-08-03 17:06:42 +00005015 Ivar[0] = EmitIvarOffsetVar(ID->getClassInterface(), IVD,
Daniel Dunbar961202372009-05-03 12:57:56 +00005016 ComputeIvarBaseOffset(CGM, ID, IVD));
Daniel Dunbar725dc2c2009-04-22 08:22:17 +00005017 Ivar[1] = GetMethodVarName(IVD->getIdentifier());
5018 Ivar[2] = GetMethodVarType(IVD);
Fariborz Jahanian7415caa2009-01-27 19:38:51 +00005019 const llvm::Type *FieldTy =
Daniel Dunbar725dc2c2009-04-22 08:22:17 +00005020 CGM.getTypes().ConvertTypeForMem(IVD->getType());
Duncan Sandsc76fe8b2009-05-09 07:08:47 +00005021 unsigned Size = CGM.getTargetData().getTypeAllocSize(FieldTy);
Fariborz Jahanian7415caa2009-01-27 19:38:51 +00005022 unsigned Align = CGM.getContext().getPreferredTypeAlign(
Daniel Dunbar59e476b2009-08-03 17:06:42 +00005023 IVD->getType().getTypePtr()) >> 3;
Fariborz Jahanian7415caa2009-01-27 19:38:51 +00005024 Align = llvm::Log2_32(Align);
Owen Andersonb7a2fe62009-07-24 23:12:58 +00005025 Ivar[3] = llvm::ConstantInt::get(ObjCTypes.IntTy, Align);
Daniel Dunbarae032262009-04-20 00:33:43 +00005026 // NOTE. Size of a bitfield does not match gcc's, because of the
5027 // way bitfields are treated special in each. But I am told that
5028 // 'size' for bitfield ivars is ignored by the runtime so it does
5029 // not matter. If it matters, there is enough info to get the
5030 // bitfield right!
Owen Andersonb7a2fe62009-07-24 23:12:58 +00005031 Ivar[4] = llvm::ConstantInt::get(ObjCTypes.IntTy, Size);
Owen Anderson0e0189d2009-07-27 22:29:56 +00005032 Ivars.push_back(llvm::ConstantStruct::get(ObjCTypes.IvarnfABITy, Ivar));
Fariborz Jahanian7415caa2009-01-27 19:38:51 +00005033 }
5034 // Return null for empty list.
5035 if (Ivars.empty())
Owen Anderson0b75f232009-07-31 20:28:54 +00005036 return llvm::Constant::getNullValue(ObjCTypes.IvarListnfABIPtrTy);
Fariborz Jahanian7415caa2009-01-27 19:38:51 +00005037 std::vector<llvm::Constant*> Values(3);
Duncan Sandsc76fe8b2009-05-09 07:08:47 +00005038 unsigned Size = CGM.getTargetData().getTypeAllocSize(ObjCTypes.IvarnfABITy);
Owen Andersonb7a2fe62009-07-24 23:12:58 +00005039 Values[0] = llvm::ConstantInt::get(ObjCTypes.IntTy, Size);
5040 Values[1] = llvm::ConstantInt::get(ObjCTypes.IntTy, Ivars.size());
Owen Anderson9793f0e2009-07-29 22:16:19 +00005041 llvm::ArrayType *AT = llvm::ArrayType::get(ObjCTypes.IvarnfABITy,
Fariborz Jahanian7415caa2009-01-27 19:38:51 +00005042 Ivars.size());
Owen Anderson47034e12009-07-28 18:33:04 +00005043 Values[2] = llvm::ConstantArray::get(AT, Ivars);
Nick Lewycky41eaf0a2009-09-19 20:00:52 +00005044 llvm::Constant *Init = llvm::ConstantStruct::get(VMContext, Values, false);
Fariborz Jahanian7415caa2009-01-27 19:38:51 +00005045 const char *Prefix = "\01l_OBJC_$_INSTANCE_VARIABLES_";
5046 llvm::GlobalVariable *GV =
Owen Andersonc10c8d32009-07-08 19:05:04 +00005047 new llvm::GlobalVariable(CGM.getModule(), Init->getType(), false,
Fariborz Jahanian7415caa2009-01-27 19:38:51 +00005048 llvm::GlobalValue::InternalLinkage,
5049 Init,
Daniel Dunbar349e6fb2009-10-18 20:48:59 +00005050 Prefix + OID->getName());
Fariborz Jahanianc22f2362009-01-31 02:43:27 +00005051 GV->setAlignment(
Daniel Dunbar710cb202010-04-25 20:39:32 +00005052 CGM.getTargetData().getABITypeAlignment(Init->getType()));
Fariborz Jahanian40a4bcd2009-01-28 01:05:23 +00005053 GV->setSection("__DATA, __objc_const");
Daniel Dunbar59e476b2009-08-03 17:06:42 +00005054
Chris Lattnerf56501c2009-07-17 23:57:13 +00005055 CGM.AddUsedGlobal(GV);
Owen Andersonade90fd2009-07-29 18:54:39 +00005056 return llvm::ConstantExpr::getBitCast(GV, ObjCTypes.IvarListnfABIPtrTy);
Fariborz Jahanian56b3b772009-01-29 19:24:30 +00005057}
5058
5059llvm::Constant *CGObjCNonFragileABIMac::GetOrEmitProtocolRef(
Daniel Dunbar59e476b2009-08-03 17:06:42 +00005060 const ObjCProtocolDecl *PD) {
Fariborz Jahanian56b3b772009-01-29 19:24:30 +00005061 llvm::GlobalVariable *&Entry = Protocols[PD->getIdentifier()];
Daniel Dunbar59e476b2009-08-03 17:06:42 +00005062
Fariborz Jahanian56b3b772009-01-29 19:24:30 +00005063 if (!Entry) {
5064 // We use the initializer as a marker of whether this is a forward
5065 // reference or not. At module finalization we add the empty
5066 // contents for protocols which were referenced but never defined.
Daniel Dunbar59e476b2009-08-03 17:06:42 +00005067 Entry =
5068 new llvm::GlobalVariable(CGM.getModule(), ObjCTypes.ProtocolnfABITy, false,
5069 llvm::GlobalValue::ExternalLinkage,
5070 0,
Daniel Dunbar349e6fb2009-10-18 20:48:59 +00005071 "\01l_OBJC_PROTOCOL_$_" + PD->getName());
Fariborz Jahanian56b3b772009-01-29 19:24:30 +00005072 Entry->setSection("__DATA,__datacoal_nt,coalesced");
Fariborz Jahanian56b3b772009-01-29 19:24:30 +00005073 }
Daniel Dunbar59e476b2009-08-03 17:06:42 +00005074
Fariborz Jahanian56b3b772009-01-29 19:24:30 +00005075 return Entry;
5076}
5077
5078/// GetOrEmitProtocol - Generate the protocol meta-data:
5079/// @code
5080/// struct _protocol_t {
5081/// id isa; // NULL
5082/// const char * const protocol_name;
5083/// const struct _protocol_list_t * protocol_list; // super protocols
5084/// const struct method_list_t * const instance_methods;
5085/// const struct method_list_t * const class_methods;
5086/// const struct method_list_t *optionalInstanceMethods;
5087/// const struct method_list_t *optionalClassMethods;
5088/// const struct _prop_list_t * properties;
5089/// const uint32_t size; // sizeof(struct _protocol_t)
5090/// const uint32_t flags; // = 0
5091/// }
5092/// @endcode
5093///
5094
5095llvm::Constant *CGObjCNonFragileABIMac::GetOrEmitProtocol(
Daniel Dunbar59e476b2009-08-03 17:06:42 +00005096 const ObjCProtocolDecl *PD) {
Fariborz Jahanian56b3b772009-01-29 19:24:30 +00005097 llvm::GlobalVariable *&Entry = Protocols[PD->getIdentifier()];
Daniel Dunbar59e476b2009-08-03 17:06:42 +00005098
Fariborz Jahanian56b3b772009-01-29 19:24:30 +00005099 // Early exit if a defining object has already been generated.
5100 if (Entry && Entry->hasInitializer())
5101 return Entry;
5102
Fariborz Jahanian56b3b772009-01-29 19:24:30 +00005103 // Construct method lists.
5104 std::vector<llvm::Constant*> InstanceMethods, ClassMethods;
5105 std::vector<llvm::Constant*> OptInstanceMethods, OptClassMethods;
Daniel Dunbar59e476b2009-08-03 17:06:42 +00005106 for (ObjCProtocolDecl::instmeth_iterator
Argyrios Kyrtzidiscfbfe782009-06-30 02:36:12 +00005107 i = PD->instmeth_begin(), e = PD->instmeth_end(); i != e; ++i) {
Fariborz Jahanian56b3b772009-01-29 19:24:30 +00005108 ObjCMethodDecl *MD = *i;
Fariborz Jahaniand9c28b82009-01-30 00:46:37 +00005109 llvm::Constant *C = GetMethodDescriptionConstant(MD);
Fariborz Jahanian56b3b772009-01-29 19:24:30 +00005110 if (MD->getImplementationControl() == ObjCMethodDecl::Optional) {
5111 OptInstanceMethods.push_back(C);
5112 } else {
5113 InstanceMethods.push_back(C);
Daniel Dunbar59e476b2009-08-03 17:06:42 +00005114 }
Fariborz Jahanian56b3b772009-01-29 19:24:30 +00005115 }
Daniel Dunbar59e476b2009-08-03 17:06:42 +00005116
5117 for (ObjCProtocolDecl::classmeth_iterator
Argyrios Kyrtzidiscfbfe782009-06-30 02:36:12 +00005118 i = PD->classmeth_begin(), e = PD->classmeth_end(); i != e; ++i) {
Fariborz Jahanian56b3b772009-01-29 19:24:30 +00005119 ObjCMethodDecl *MD = *i;
Fariborz Jahaniand9c28b82009-01-30 00:46:37 +00005120 llvm::Constant *C = GetMethodDescriptionConstant(MD);
Fariborz Jahanian56b3b772009-01-29 19:24:30 +00005121 if (MD->getImplementationControl() == ObjCMethodDecl::Optional) {
5122 OptClassMethods.push_back(C);
5123 } else {
5124 ClassMethods.push_back(C);
Daniel Dunbar59e476b2009-08-03 17:06:42 +00005125 }
Fariborz Jahanian56b3b772009-01-29 19:24:30 +00005126 }
Daniel Dunbar59e476b2009-08-03 17:06:42 +00005127
Fariborz Jahanian56b3b772009-01-29 19:24:30 +00005128 std::vector<llvm::Constant*> Values(10);
5129 // isa is NULL
Owen Anderson0b75f232009-07-31 20:28:54 +00005130 Values[0] = llvm::Constant::getNullValue(ObjCTypes.ObjectPtrTy);
Fariborz Jahanian56b3b772009-01-29 19:24:30 +00005131 Values[1] = GetClassName(PD->getIdentifier());
Daniel Dunbar349e6fb2009-10-18 20:48:59 +00005132 Values[2] = EmitProtocolList("\01l_OBJC_$_PROTOCOL_REFS_" + PD->getName(),
5133 PD->protocol_begin(),
5134 PD->protocol_end());
Daniel Dunbar59e476b2009-08-03 17:06:42 +00005135
Fariborz Jahaniand9c28b82009-01-30 00:46:37 +00005136 Values[3] = EmitMethodList("\01l_OBJC_$_PROTOCOL_INSTANCE_METHODS_"
Daniel Dunbar349e6fb2009-10-18 20:48:59 +00005137 + PD->getName(),
Fariborz Jahanian56b3b772009-01-29 19:24:30 +00005138 "__DATA, __objc_const",
5139 InstanceMethods);
Daniel Dunbar59e476b2009-08-03 17:06:42 +00005140 Values[4] = EmitMethodList("\01l_OBJC_$_PROTOCOL_CLASS_METHODS_"
Daniel Dunbar349e6fb2009-10-18 20:48:59 +00005141 + PD->getName(),
Fariborz Jahanian56b3b772009-01-29 19:24:30 +00005142 "__DATA, __objc_const",
5143 ClassMethods);
Fariborz Jahaniand9c28b82009-01-30 00:46:37 +00005144 Values[5] = EmitMethodList("\01l_OBJC_$_PROTOCOL_INSTANCE_METHODS_OPT_"
Daniel Dunbar349e6fb2009-10-18 20:48:59 +00005145 + PD->getName(),
Fariborz Jahanian56b3b772009-01-29 19:24:30 +00005146 "__DATA, __objc_const",
5147 OptInstanceMethods);
Daniel Dunbar59e476b2009-08-03 17:06:42 +00005148 Values[6] = EmitMethodList("\01l_OBJC_$_PROTOCOL_CLASS_METHODS_OPT_"
Daniel Dunbar349e6fb2009-10-18 20:48:59 +00005149 + PD->getName(),
Fariborz Jahanian56b3b772009-01-29 19:24:30 +00005150 "__DATA, __objc_const",
5151 OptClassMethods);
Daniel Dunbar349e6fb2009-10-18 20:48:59 +00005152 Values[7] = EmitPropertyList("\01l_OBJC_$_PROP_LIST_" + PD->getName(),
Fariborz Jahanian56b3b772009-01-29 19:24:30 +00005153 0, PD, ObjCTypes);
Daniel Dunbar59e476b2009-08-03 17:06:42 +00005154 uint32_t Size =
Duncan Sandsc76fe8b2009-05-09 07:08:47 +00005155 CGM.getTargetData().getTypeAllocSize(ObjCTypes.ProtocolnfABITy);
Owen Andersonb7a2fe62009-07-24 23:12:58 +00005156 Values[8] = llvm::ConstantInt::get(ObjCTypes.IntTy, Size);
Owen Anderson0b75f232009-07-31 20:28:54 +00005157 Values[9] = llvm::Constant::getNullValue(ObjCTypes.IntTy);
Owen Anderson0e0189d2009-07-27 22:29:56 +00005158 llvm::Constant *Init = llvm::ConstantStruct::get(ObjCTypes.ProtocolnfABITy,
Fariborz Jahanian56b3b772009-01-29 19:24:30 +00005159 Values);
Daniel Dunbar59e476b2009-08-03 17:06:42 +00005160
Fariborz Jahanian56b3b772009-01-29 19:24:30 +00005161 if (Entry) {
5162 // Already created, fix the linkage and update the initializer.
Mike Stumpa6ca3342009-03-07 16:33:28 +00005163 Entry->setLinkage(llvm::GlobalValue::WeakAnyLinkage);
Fariborz Jahanian56b3b772009-01-29 19:24:30 +00005164 Entry->setInitializer(Init);
5165 } else {
Daniel Dunbar59e476b2009-08-03 17:06:42 +00005166 Entry =
Daniel Dunbar349e6fb2009-10-18 20:48:59 +00005167 new llvm::GlobalVariable(CGM.getModule(), ObjCTypes.ProtocolnfABITy,
5168 false, llvm::GlobalValue::WeakAnyLinkage, Init,
5169 "\01l_OBJC_PROTOCOL_$_" + PD->getName());
Fariborz Jahanianc22f2362009-01-31 02:43:27 +00005170 Entry->setAlignment(
Daniel Dunbar710cb202010-04-25 20:39:32 +00005171 CGM.getTargetData().getABITypeAlignment(ObjCTypes.ProtocolnfABITy));
Fariborz Jahanian56b3b772009-01-29 19:24:30 +00005172 Entry->setSection("__DATA,__datacoal_nt,coalesced");
Fariborz Jahanian56b3b772009-01-29 19:24:30 +00005173 }
Fariborz Jahanian61cd4b52009-01-29 20:10:59 +00005174 Entry->setVisibility(llvm::GlobalValue::HiddenVisibility);
Chris Lattnerf56501c2009-07-17 23:57:13 +00005175 CGM.AddUsedGlobal(Entry);
5176
Fariborz Jahanian61cd4b52009-01-29 20:10:59 +00005177 // Use this protocol meta-data to build protocol list table in section
5178 // __DATA, __objc_protolist
Daniel Dunbar349e6fb2009-10-18 20:48:59 +00005179 llvm::GlobalVariable *PTGV =
5180 new llvm::GlobalVariable(CGM.getModule(), ObjCTypes.ProtocolnfABIPtrTy,
5181 false, llvm::GlobalValue::WeakAnyLinkage, Entry,
5182 "\01l_OBJC_LABEL_PROTOCOL_$_" + PD->getName());
Fariborz Jahanianc22f2362009-01-31 02:43:27 +00005183 PTGV->setAlignment(
Daniel Dunbar710cb202010-04-25 20:39:32 +00005184 CGM.getTargetData().getABITypeAlignment(ObjCTypes.ProtocolnfABIPtrTy));
Daniel Dunbarb25452a2009-04-15 02:56:18 +00005185 PTGV->setSection("__DATA, __objc_protolist, coalesced, no_dead_strip");
Fariborz Jahanian61cd4b52009-01-29 20:10:59 +00005186 PTGV->setVisibility(llvm::GlobalValue::HiddenVisibility);
Chris Lattnerf56501c2009-07-17 23:57:13 +00005187 CGM.AddUsedGlobal(PTGV);
Fariborz Jahanian56b3b772009-01-29 19:24:30 +00005188 return Entry;
5189}
5190
5191/// EmitProtocolList - Generate protocol list meta-data:
5192/// @code
5193/// struct _protocol_list_t {
5194/// long protocol_count; // Note, this is 32/64 bit
5195/// struct _protocol_t[protocol_count];
5196/// }
5197/// @endcode
5198///
5199llvm::Constant *
Daniel Dunbar349e6fb2009-10-18 20:48:59 +00005200CGObjCNonFragileABIMac::EmitProtocolList(llvm::Twine Name,
5201 ObjCProtocolDecl::protocol_iterator begin,
5202 ObjCProtocolDecl::protocol_iterator end) {
Fariborz Jahanian56b3b772009-01-29 19:24:30 +00005203 std::vector<llvm::Constant*> ProtocolRefs;
Daniel Dunbar59e476b2009-08-03 17:06:42 +00005204
Fariborz Jahanian56b3b772009-01-29 19:24:30 +00005205 // Just return null for empty protocol lists
Daniel Dunbar59e476b2009-08-03 17:06:42 +00005206 if (begin == end)
Owen Anderson0b75f232009-07-31 20:28:54 +00005207 return llvm::Constant::getNullValue(ObjCTypes.ProtocolListnfABIPtrTy);
Daniel Dunbar59e476b2009-08-03 17:06:42 +00005208
Daniel Dunbar8de90f02009-02-15 07:36:20 +00005209 // FIXME: We shouldn't need to do this lookup here, should we?
Daniel Dunbar349e6fb2009-10-18 20:48:59 +00005210 llvm::SmallString<256> TmpName;
5211 Name.toVector(TmpName);
5212 llvm::GlobalVariable *GV =
5213 CGM.getModule().getGlobalVariable(TmpName.str(), true);
Fariborz Jahanian56b3b772009-01-29 19:24:30 +00005214 if (GV)
Daniel Dunbar349e6fb2009-10-18 20:48:59 +00005215 return llvm::ConstantExpr::getBitCast(GV, ObjCTypes.ProtocolListnfABIPtrTy);
Daniel Dunbar59e476b2009-08-03 17:06:42 +00005216
Daniel Dunbar8de90f02009-02-15 07:36:20 +00005217 for (; begin != end; ++begin)
5218 ProtocolRefs.push_back(GetProtocolRef(*begin)); // Implemented???
5219
Fariborz Jahanian56b3b772009-01-29 19:24:30 +00005220 // This list is null terminated.
Owen Anderson0b75f232009-07-31 20:28:54 +00005221 ProtocolRefs.push_back(llvm::Constant::getNullValue(
Daniel Dunbar59e476b2009-08-03 17:06:42 +00005222 ObjCTypes.ProtocolnfABIPtrTy));
5223
Fariborz Jahanian56b3b772009-01-29 19:24:30 +00005224 std::vector<llvm::Constant*> Values(2);
Owen Anderson170229f2009-07-14 23:10:40 +00005225 Values[0] =
Owen Andersonb7a2fe62009-07-24 23:12:58 +00005226 llvm::ConstantInt::get(ObjCTypes.LongTy, ProtocolRefs.size() - 1);
Daniel Dunbar59e476b2009-08-03 17:06:42 +00005227 Values[1] =
Owen Anderson47034e12009-07-28 18:33:04 +00005228 llvm::ConstantArray::get(
Owen Anderson9793f0e2009-07-29 22:16:19 +00005229 llvm::ArrayType::get(ObjCTypes.ProtocolnfABIPtrTy,
Daniel Dunbar59e476b2009-08-03 17:06:42 +00005230 ProtocolRefs.size()),
5231 ProtocolRefs);
5232
Nick Lewycky41eaf0a2009-09-19 20:00:52 +00005233 llvm::Constant *Init = llvm::ConstantStruct::get(VMContext, Values, false);
Owen Andersonc10c8d32009-07-08 19:05:04 +00005234 GV = new llvm::GlobalVariable(CGM.getModule(), Init->getType(), false,
Fariborz Jahanian56b3b772009-01-29 19:24:30 +00005235 llvm::GlobalValue::InternalLinkage,
5236 Init,
Owen Andersonc10c8d32009-07-08 19:05:04 +00005237 Name);
Fariborz Jahanian56b3b772009-01-29 19:24:30 +00005238 GV->setSection("__DATA, __objc_const");
Fariborz Jahanianc22f2362009-01-31 02:43:27 +00005239 GV->setAlignment(
Daniel Dunbar710cb202010-04-25 20:39:32 +00005240 CGM.getTargetData().getABITypeAlignment(Init->getType()));
Chris Lattnerf56501c2009-07-17 23:57:13 +00005241 CGM.AddUsedGlobal(GV);
Daniel Dunbar59e476b2009-08-03 17:06:42 +00005242 return llvm::ConstantExpr::getBitCast(GV,
Daniel Dunbar8de90f02009-02-15 07:36:20 +00005243 ObjCTypes.ProtocolListnfABIPtrTy);
Fariborz Jahanian56b3b772009-01-29 19:24:30 +00005244}
5245
Fariborz Jahaniand9c28b82009-01-30 00:46:37 +00005246/// GetMethodDescriptionConstant - This routine build following meta-data:
5247/// struct _objc_method {
5248/// SEL _cmd;
5249/// char *method_type;
5250/// char *_imp;
5251/// }
5252
5253llvm::Constant *
5254CGObjCNonFragileABIMac::GetMethodDescriptionConstant(const ObjCMethodDecl *MD) {
5255 std::vector<llvm::Constant*> Desc(3);
Owen Anderson170229f2009-07-14 23:10:40 +00005256 Desc[0] =
Daniel Dunbar59e476b2009-08-03 17:06:42 +00005257 llvm::ConstantExpr::getBitCast(GetMethodVarName(MD->getSelector()),
5258 ObjCTypes.SelectorPtrTy);
Fariborz Jahaniand9c28b82009-01-30 00:46:37 +00005259 Desc[1] = GetMethodVarType(MD);
Fariborz Jahanian097feda2009-01-30 18:58:59 +00005260 // Protocol methods have no implementation. So, this entry is always NULL.
Owen Anderson0b75f232009-07-31 20:28:54 +00005261 Desc[2] = llvm::Constant::getNullValue(ObjCTypes.Int8PtrTy);
Owen Anderson0e0189d2009-07-27 22:29:56 +00005262 return llvm::ConstantStruct::get(ObjCTypes.MethodTy, Desc);
Fariborz Jahaniand9c28b82009-01-30 00:46:37 +00005263}
Fariborz Jahanianc88a70d2009-02-03 00:09:52 +00005264
5265/// EmitObjCValueForIvar - Code Gen for nonfragile ivar reference.
5266/// This code gen. amounts to generating code for:
5267/// @code
5268/// (type *)((char *)base + _OBJC_IVAR_$_.ivar;
5269/// @encode
Daniel Dunbar59e476b2009-08-03 17:06:42 +00005270///
Fariborz Jahanian712bfa62009-02-03 19:03:09 +00005271LValue CGObjCNonFragileABIMac::EmitObjCValueForIvar(
John McCall96fa4842010-05-17 21:00:27 +00005272 CodeGen::CodeGenFunction &CGF,
5273 QualType ObjectTy,
5274 llvm::Value *BaseValue,
5275 const ObjCIvarDecl *Ivar,
5276 unsigned CVRQualifiers) {
5277 ObjCInterfaceDecl *ID = ObjectTy->getAs<ObjCObjectType>()->getInterface();
Daniel Dunbar9fd114d2009-04-22 07:32:20 +00005278 return EmitValueForIvarAtOffset(CGF, ID, BaseValue, Ivar, CVRQualifiers,
5279 EmitIvarOffset(CGF, ID, Ivar));
Fariborz Jahanianc88a70d2009-02-03 00:09:52 +00005280}
5281
Fariborz Jahanian21fc74c2009-02-10 19:02:04 +00005282llvm::Value *CGObjCNonFragileABIMac::EmitIvarOffset(
Daniel Dunbar59e476b2009-08-03 17:06:42 +00005283 CodeGen::CodeGenFunction &CGF,
5284 const ObjCInterfaceDecl *Interface,
5285 const ObjCIvarDecl *Ivar) {
Daniel Dunbarc76493a2009-11-29 21:23:36 +00005286 return CGF.Builder.CreateLoad(ObjCIvarOffsetVariable(Interface, Ivar),"ivar");
Fariborz Jahanian21fc74c2009-02-10 19:02:04 +00005287}
5288
Fariborz Jahanian3d9296e2009-02-04 00:22:57 +00005289CodeGen::RValue CGObjCNonFragileABIMac::EmitMessageSend(
Daniel Dunbar59e476b2009-08-03 17:06:42 +00005290 CodeGen::CodeGenFunction &CGF,
John McCall78a15112010-05-22 01:48:05 +00005291 ReturnValueSlot Return,
Daniel Dunbar59e476b2009-08-03 17:06:42 +00005292 QualType ResultType,
5293 Selector Sel,
5294 llvm::Value *Receiver,
5295 QualType Arg0Ty,
5296 bool IsSuper,
5297 const CallArgList &CallArgs) {
Mike Stump18bb9282009-05-16 07:57:57 +00005298 // FIXME. Even though IsSuper is passes. This function doese not handle calls
5299 // to 'super' receivers.
Fariborz Jahaniane4dc35d2009-02-04 20:42:28 +00005300 CodeGenTypes &Types = CGM.getTypes();
Fariborz Jahanian6b7cd6e2009-02-06 20:09:23 +00005301 llvm::Value *Arg0 = Receiver;
5302 if (!IsSuper)
5303 Arg0 = CGF.Builder.CreateBitCast(Arg0, ObjCTypes.ObjectPtrTy, "tmp");
Daniel Dunbar59e476b2009-08-03 17:06:42 +00005304
Fariborz Jahaniane4dc35d2009-02-04 20:42:28 +00005305 // Find the message function name.
Mike Stump18bb9282009-05-16 07:57:57 +00005306 // FIXME. This is too much work to get the ABI-specific result type needed to
5307 // find the message name.
John McCall2da83a32010-02-26 00:48:12 +00005308 const CGFunctionInfo &FnInfo
Rafael Espindolac50c27c2010-03-30 20:24:48 +00005309 = Types.getFunctionInfo(ResultType, CallArgList(),
5310 FunctionType::ExtInfo());
Fariborz Jahaniane29b4f02009-04-30 23:08:58 +00005311 llvm::Constant *Fn = 0;
Fariborz Jahaniane4dc35d2009-02-04 20:42:28 +00005312 std::string Name("\01l_");
Daniel Dunbar6f2e8392010-07-14 23:39:36 +00005313 if (CGM.ReturnTypeUsesSRet(FnInfo)) {
Fariborz Jahanian90655412009-02-05 18:00:27 +00005314#if 0
5315 // unlike what is documented. gcc never generates this API!!
Fariborz Jahaniane4dc35d2009-02-04 20:42:28 +00005316 if (Receiver->getType() == ObjCTypes.ObjectPtrTy) {
Chris Lattner2dfdb3e2009-04-22 02:53:24 +00005317 Fn = ObjCTypes.getMessageSendIdStretFixupFn();
Fariborz Jahaniane4dc35d2009-02-04 20:42:28 +00005318 // FIXME. Is there a better way of getting these names.
5319 // They are available in RuntimeFunctions vector pair.
5320 Name += "objc_msgSendId_stret_fixup";
Mike Stump658fe022009-07-30 22:28:39 +00005321 } else
Fariborz Jahanian90655412009-02-05 18:00:27 +00005322#endif
Daniel Dunbar59e476b2009-08-03 17:06:42 +00005323 if (IsSuper) {
Chris Lattner2dfdb3e2009-04-22 02:53:24 +00005324 Fn = ObjCTypes.getMessageSendSuper2StretFixupFn();
Fariborz Jahanian6b7cd6e2009-02-06 20:09:23 +00005325 Name += "objc_msgSendSuper2_stret_fixup";
Daniel Dunbar59e476b2009-08-03 17:06:42 +00005326 } else {
5327 Fn = ObjCTypes.getMessageSendStretFixupFn();
5328 Name += "objc_msgSend_stret_fixup";
5329 }
Daniel Dunbar6f2e8392010-07-14 23:39:36 +00005330 } else if (!IsSuper && CGM.ReturnTypeUsesFPRet(ResultType)) {
5331 Fn = ObjCTypes.getMessageSendFpretFixupFn();
5332 Name += "objc_msgSend_fpret_fixup";
Mike Stump658fe022009-07-30 22:28:39 +00005333 } else {
Fariborz Jahanian90655412009-02-05 18:00:27 +00005334#if 0
5335// unlike what is documented. gcc never generates this API!!
Fariborz Jahaniane4dc35d2009-02-04 20:42:28 +00005336 if (Receiver->getType() == ObjCTypes.ObjectPtrTy) {
Chris Lattner2dfdb3e2009-04-22 02:53:24 +00005337 Fn = ObjCTypes.getMessageSendIdFixupFn();
Fariborz Jahaniane4dc35d2009-02-04 20:42:28 +00005338 Name += "objc_msgSendId_fixup";
Daniel Dunbar59e476b2009-08-03 17:06:42 +00005339 } else
Fariborz Jahanian90655412009-02-05 18:00:27 +00005340#endif
Daniel Dunbar59e476b2009-08-03 17:06:42 +00005341 if (IsSuper) {
Chris Lattner2dfdb3e2009-04-22 02:53:24 +00005342 Fn = ObjCTypes.getMessageSendSuper2FixupFn();
Fariborz Jahanian6b7cd6e2009-02-06 20:09:23 +00005343 Name += "objc_msgSendSuper2_fixup";
Daniel Dunbar59e476b2009-08-03 17:06:42 +00005344 } else {
5345 Fn = ObjCTypes.getMessageSendFixupFn();
5346 Name += "objc_msgSend_fixup";
5347 }
Fariborz Jahaniane4dc35d2009-02-04 20:42:28 +00005348 }
Fariborz Jahaniane29b4f02009-04-30 23:08:58 +00005349 assert(Fn && "CGObjCNonFragileABIMac::EmitMessageSend");
Fariborz Jahaniane4dc35d2009-02-04 20:42:28 +00005350 Name += '_';
5351 std::string SelName(Sel.getAsString());
5352 // Replace all ':' in selector name with '_' ouch!
Mike Stump11289f42009-09-09 15:08:12 +00005353 for (unsigned i = 0; i < SelName.size(); i++)
Fariborz Jahaniane4dc35d2009-02-04 20:42:28 +00005354 if (SelName[i] == ':')
5355 SelName[i] = '_';
5356 Name += SelName;
5357 llvm::GlobalVariable *GV = CGM.getModule().getGlobalVariable(Name);
5358 if (!GV) {
Daniel Dunbare60aa052009-04-15 19:03:14 +00005359 // Build message ref table entry.
Fariborz Jahaniane4dc35d2009-02-04 20:42:28 +00005360 std::vector<llvm::Constant*> Values(2);
5361 Values[0] = Fn;
5362 Values[1] = GetMethodVarName(Sel);
Nick Lewycky41eaf0a2009-09-19 20:00:52 +00005363 llvm::Constant *Init = llvm::ConstantStruct::get(VMContext, Values, false);
Owen Andersonc10c8d32009-07-08 19:05:04 +00005364 GV = new llvm::GlobalVariable(CGM.getModule(), Init->getType(), false,
Mike Stumpa6ca3342009-03-07 16:33:28 +00005365 llvm::GlobalValue::WeakAnyLinkage,
Fariborz Jahaniane4dc35d2009-02-04 20:42:28 +00005366 Init,
Owen Andersonc10c8d32009-07-08 19:05:04 +00005367 Name);
Fariborz Jahaniane4dc35d2009-02-04 20:42:28 +00005368 GV->setVisibility(llvm::GlobalValue::HiddenVisibility);
Daniel Dunbar24645c92009-04-15 19:04:46 +00005369 GV->setAlignment(16);
Fariborz Jahaniane4dc35d2009-02-04 20:42:28 +00005370 GV->setSection("__DATA, __objc_msgrefs, coalesced");
Fariborz Jahaniane4dc35d2009-02-04 20:42:28 +00005371 }
5372 llvm::Value *Arg1 = CGF.Builder.CreateBitCast(GV, ObjCTypes.MessageRefPtrTy);
Daniel Dunbar59e476b2009-08-03 17:06:42 +00005373
Fariborz Jahaniane4dc35d2009-02-04 20:42:28 +00005374 CallArgList ActualArgs;
5375 ActualArgs.push_back(std::make_pair(RValue::get(Arg0), Arg0Ty));
Daniel Dunbar59e476b2009-08-03 17:06:42 +00005376 ActualArgs.push_back(std::make_pair(RValue::get(Arg1),
Fariborz Jahaniane4dc35d2009-02-04 20:42:28 +00005377 ObjCTypes.MessageRefCPtrTy));
5378 ActualArgs.insert(ActualArgs.end(), CallArgs.begin(), CallArgs.end());
John McCallab26cfa2010-02-05 21:31:56 +00005379 const CGFunctionInfo &FnInfo1 = Types.getFunctionInfo(ResultType, ActualArgs,
Rafael Espindolac50c27c2010-03-30 20:24:48 +00005380 FunctionType::ExtInfo());
Fariborz Jahanian4e87c832009-02-05 01:13:09 +00005381 llvm::Value *Callee = CGF.Builder.CreateStructGEP(Arg1, 0);
5382 Callee = CGF.Builder.CreateLoad(Callee);
Fariborz Jahanian35afdfc2009-02-14 21:25:36 +00005383 const llvm::FunctionType *FTy = Types.GetFunctionType(FnInfo1, true);
Fariborz Jahanian4e87c832009-02-05 01:13:09 +00005384 Callee = CGF.Builder.CreateBitCast(Callee,
Owen Anderson9793f0e2009-07-29 22:16:19 +00005385 llvm::PointerType::getUnqual(FTy));
John McCall78a15112010-05-22 01:48:05 +00005386 return CGF.EmitCall(FnInfo1, Callee, Return, ActualArgs);
Fariborz Jahanian3d9296e2009-02-04 00:22:57 +00005387}
5388
5389/// Generate code for a message send expression in the nonfragile abi.
Daniel Dunbaraff9fca2009-09-17 04:01:22 +00005390CodeGen::RValue
5391CGObjCNonFragileABIMac::GenerateMessageSend(CodeGen::CodeGenFunction &CGF,
John McCall78a15112010-05-22 01:48:05 +00005392 ReturnValueSlot Return,
Daniel Dunbaraff9fca2009-09-17 04:01:22 +00005393 QualType ResultType,
5394 Selector Sel,
5395 llvm::Value *Receiver,
Daniel Dunbaraff9fca2009-09-17 04:01:22 +00005396 const CallArgList &CallArgs,
David Chisnall01aa4672010-04-28 19:33:36 +00005397 const ObjCInterfaceDecl *Class,
Daniel Dunbaraff9fca2009-09-17 04:01:22 +00005398 const ObjCMethodDecl *Method) {
Fariborz Jahanian5d5ed2d2009-05-11 19:25:47 +00005399 return LegacyDispatchedSelector(Sel)
John McCall78a15112010-05-22 01:48:05 +00005400 ? EmitLegacyMessageSend(CGF, Return, ResultType,
5401 EmitSelector(CGF.Builder, Sel),
Daniel Dunbar59e476b2009-08-03 17:06:42 +00005402 Receiver, CGF.getContext().getObjCIdType(),
Daniel Dunbaraff9fca2009-09-17 04:01:22 +00005403 false, CallArgs, Method, ObjCTypes)
John McCall78a15112010-05-22 01:48:05 +00005404 : EmitMessageSend(CGF, Return, ResultType, Sel,
Daniel Dunbar59e476b2009-08-03 17:06:42 +00005405 Receiver, CGF.getContext().getObjCIdType(),
5406 false, CallArgs);
Fariborz Jahanian3d9296e2009-02-04 00:22:57 +00005407}
5408
Daniel Dunbarc6928bb2009-03-01 04:40:10 +00005409llvm::GlobalVariable *
Fariborz Jahanian899e7eb2009-04-14 18:41:56 +00005410CGObjCNonFragileABIMac::GetClassGlobal(const std::string &Name) {
Daniel Dunbarc6928bb2009-03-01 04:40:10 +00005411 llvm::GlobalVariable *GV = CGM.getModule().getGlobalVariable(Name);
5412
Daniel Dunbara6468342009-03-02 05:18:14 +00005413 if (!GV) {
Daniel Dunbar59e476b2009-08-03 17:06:42 +00005414 GV = new llvm::GlobalVariable(CGM.getModule(), ObjCTypes.ClassnfABITy,
Owen Andersonc10c8d32009-07-08 19:05:04 +00005415 false, llvm::GlobalValue::ExternalLinkage,
5416 0, Name);
Daniel Dunbarc6928bb2009-03-01 04:40:10 +00005417 }
5418
5419 return GV;
5420}
5421
Daniel Dunbar59e476b2009-08-03 17:06:42 +00005422llvm::Value *CGObjCNonFragileABIMac::EmitClassRef(CGBuilderTy &Builder,
5423 const ObjCInterfaceDecl *ID) {
Fariborz Jahanian33f66e62009-02-05 20:41:40 +00005424 llvm::GlobalVariable *&Entry = ClassReferences[ID->getIdentifier()];
Daniel Dunbar59e476b2009-08-03 17:06:42 +00005425
Fariborz Jahanian33f66e62009-02-05 20:41:40 +00005426 if (!Entry) {
Daniel Dunbar15894b72009-04-07 05:48:37 +00005427 std::string ClassName(getClassSymbolPrefix() + ID->getNameAsString());
Daniel Dunbarc6928bb2009-03-01 04:40:10 +00005428 llvm::GlobalVariable *ClassGV = GetClassGlobal(ClassName);
Daniel Dunbar59e476b2009-08-03 17:06:42 +00005429 Entry =
5430 new llvm::GlobalVariable(CGM.getModule(), ObjCTypes.ClassnfABIPtrTy,
Owen Andersonc10c8d32009-07-08 19:05:04 +00005431 false, llvm::GlobalValue::InternalLinkage,
Daniel Dunbar59e476b2009-08-03 17:06:42 +00005432 ClassGV,
Owen Andersonc10c8d32009-07-08 19:05:04 +00005433 "\01L_OBJC_CLASSLIST_REFERENCES_$_");
Fariborz Jahanian33f66e62009-02-05 20:41:40 +00005434 Entry->setAlignment(
Daniel Dunbar710cb202010-04-25 20:39:32 +00005435 CGM.getTargetData().getABITypeAlignment(
Daniel Dunbar59e476b2009-08-03 17:06:42 +00005436 ObjCTypes.ClassnfABIPtrTy));
Daniel Dunbar508a7dd2009-04-18 08:51:00 +00005437 Entry->setSection("__DATA, __objc_classrefs, regular, no_dead_strip");
Chris Lattnerf56501c2009-07-17 23:57:13 +00005438 CGM.AddUsedGlobal(Entry);
Daniel Dunbar508a7dd2009-04-18 08:51:00 +00005439 }
Daniel Dunbar59e476b2009-08-03 17:06:42 +00005440
Daniel Dunbarc76493a2009-11-29 21:23:36 +00005441 return Builder.CreateLoad(Entry, "tmp");
Daniel Dunbar508a7dd2009-04-18 08:51:00 +00005442}
Fariborz Jahanian33f66e62009-02-05 20:41:40 +00005443
Daniel Dunbar508a7dd2009-04-18 08:51:00 +00005444llvm::Value *
Daniel Dunbar59e476b2009-08-03 17:06:42 +00005445CGObjCNonFragileABIMac::EmitSuperClassRef(CGBuilderTy &Builder,
Daniel Dunbar508a7dd2009-04-18 08:51:00 +00005446 const ObjCInterfaceDecl *ID) {
5447 llvm::GlobalVariable *&Entry = SuperClassReferences[ID->getIdentifier()];
Daniel Dunbar59e476b2009-08-03 17:06:42 +00005448
Daniel Dunbar508a7dd2009-04-18 08:51:00 +00005449 if (!Entry) {
5450 std::string ClassName(getClassSymbolPrefix() + ID->getNameAsString());
5451 llvm::GlobalVariable *ClassGV = GetClassGlobal(ClassName);
Daniel Dunbar59e476b2009-08-03 17:06:42 +00005452 Entry =
5453 new llvm::GlobalVariable(CGM.getModule(), ObjCTypes.ClassnfABIPtrTy,
Owen Andersonc10c8d32009-07-08 19:05:04 +00005454 false, llvm::GlobalValue::InternalLinkage,
Daniel Dunbar59e476b2009-08-03 17:06:42 +00005455 ClassGV,
Owen Andersonc10c8d32009-07-08 19:05:04 +00005456 "\01L_OBJC_CLASSLIST_SUP_REFS_$_");
Daniel Dunbar508a7dd2009-04-18 08:51:00 +00005457 Entry->setAlignment(
Daniel Dunbar710cb202010-04-25 20:39:32 +00005458 CGM.getTargetData().getABITypeAlignment(
Daniel Dunbar59e476b2009-08-03 17:06:42 +00005459 ObjCTypes.ClassnfABIPtrTy));
Daniel Dunbar508a7dd2009-04-18 08:51:00 +00005460 Entry->setSection("__DATA, __objc_superrefs, regular, no_dead_strip");
Chris Lattnerf56501c2009-07-17 23:57:13 +00005461 CGM.AddUsedGlobal(Entry);
Fariborz Jahanian33f66e62009-02-05 20:41:40 +00005462 }
Daniel Dunbar59e476b2009-08-03 17:06:42 +00005463
Daniel Dunbarc76493a2009-11-29 21:23:36 +00005464 return Builder.CreateLoad(Entry, "tmp");
Fariborz Jahanian33f66e62009-02-05 20:41:40 +00005465}
5466
Fariborz Jahanian6b7cd6e2009-02-06 20:09:23 +00005467/// EmitMetaClassRef - Return a Value * of the address of _class_t
5468/// meta-data
5469///
Daniel Dunbar59e476b2009-08-03 17:06:42 +00005470llvm::Value *CGObjCNonFragileABIMac::EmitMetaClassRef(CGBuilderTy &Builder,
5471 const ObjCInterfaceDecl *ID) {
Fariborz Jahanian6b7cd6e2009-02-06 20:09:23 +00005472 llvm::GlobalVariable * &Entry = MetaClassReferences[ID->getIdentifier()];
5473 if (Entry)
Daniel Dunbarc76493a2009-11-29 21:23:36 +00005474 return Builder.CreateLoad(Entry, "tmp");
Daniel Dunbar59e476b2009-08-03 17:06:42 +00005475
Daniel Dunbar15894b72009-04-07 05:48:37 +00005476 std::string MetaClassName(getMetaclassSymbolPrefix() + ID->getNameAsString());
Fariborz Jahanian899e7eb2009-04-14 18:41:56 +00005477 llvm::GlobalVariable *MetaClassGV = GetClassGlobal(MetaClassName);
Daniel Dunbar59e476b2009-08-03 17:06:42 +00005478 Entry =
Owen Andersonc10c8d32009-07-08 19:05:04 +00005479 new llvm::GlobalVariable(CGM.getModule(), ObjCTypes.ClassnfABIPtrTy, false,
Fariborz Jahanian6b7cd6e2009-02-06 20:09:23 +00005480 llvm::GlobalValue::InternalLinkage,
Daniel Dunbar59e476b2009-08-03 17:06:42 +00005481 MetaClassGV,
Owen Andersonc10c8d32009-07-08 19:05:04 +00005482 "\01L_OBJC_CLASSLIST_SUP_REFS_$_");
Fariborz Jahanian6b7cd6e2009-02-06 20:09:23 +00005483 Entry->setAlignment(
Daniel Dunbar710cb202010-04-25 20:39:32 +00005484 CGM.getTargetData().getABITypeAlignment(
Daniel Dunbar59e476b2009-08-03 17:06:42 +00005485 ObjCTypes.ClassnfABIPtrTy));
5486
Daniel Dunbare60aa052009-04-15 19:03:14 +00005487 Entry->setSection("__DATA, __objc_superrefs, regular, no_dead_strip");
Chris Lattnerf56501c2009-07-17 23:57:13 +00005488 CGM.AddUsedGlobal(Entry);
Daniel Dunbar59e476b2009-08-03 17:06:42 +00005489
Daniel Dunbarc76493a2009-11-29 21:23:36 +00005490 return Builder.CreateLoad(Entry, "tmp");
Fariborz Jahanian6b7cd6e2009-02-06 20:09:23 +00005491}
5492
Fariborz Jahanian33f66e62009-02-05 20:41:40 +00005493/// GetClass - Return a reference to the class for the given interface
5494/// decl.
5495llvm::Value *CGObjCNonFragileABIMac::GetClass(CGBuilderTy &Builder,
5496 const ObjCInterfaceDecl *ID) {
Fariborz Jahanian95ace552009-11-17 22:42:00 +00005497 if (ID->hasAttr<WeakImportAttr>()) {
5498 std::string ClassName(getClassSymbolPrefix() + ID->getNameAsString());
5499 llvm::GlobalVariable *ClassGV = GetClassGlobal(ClassName);
5500 ClassGV->setLinkage(llvm::GlobalValue::ExternalWeakLinkage);
5501 }
5502
Fariborz Jahanian33f66e62009-02-05 20:41:40 +00005503 return EmitClassRef(Builder, ID);
5504}
5505
Fariborz Jahanian6b7cd6e2009-02-06 20:09:23 +00005506/// Generates a message send where the super is the receiver. This is
5507/// a message send to self with special delivery semantics indicating
5508/// which class's method should be called.
5509CodeGen::RValue
5510CGObjCNonFragileABIMac::GenerateMessageSendSuper(CodeGen::CodeGenFunction &CGF,
John McCall78a15112010-05-22 01:48:05 +00005511 ReturnValueSlot Return,
Daniel Dunbar59e476b2009-08-03 17:06:42 +00005512 QualType ResultType,
5513 Selector Sel,
5514 const ObjCInterfaceDecl *Class,
5515 bool isCategoryImpl,
5516 llvm::Value *Receiver,
5517 bool IsClassMessage,
Daniel Dunbaraff9fca2009-09-17 04:01:22 +00005518 const CodeGen::CallArgList &CallArgs,
5519 const ObjCMethodDecl *Method) {
Fariborz Jahanian6b7cd6e2009-02-06 20:09:23 +00005520 // ...
5521 // Create and init a super structure; this is a (receiver, class)
5522 // pair we will pass to objc_msgSendSuper.
5523 llvm::Value *ObjCSuper =
5524 CGF.Builder.CreateAlloca(ObjCTypes.SuperTy, 0, "objc_super");
Daniel Dunbar59e476b2009-08-03 17:06:42 +00005525
Fariborz Jahanian6b7cd6e2009-02-06 20:09:23 +00005526 llvm::Value *ReceiverAsObject =
5527 CGF.Builder.CreateBitCast(Receiver, ObjCTypes.ObjectPtrTy);
5528 CGF.Builder.CreateStore(ReceiverAsObject,
5529 CGF.Builder.CreateStructGEP(ObjCSuper, 0));
Daniel Dunbar59e476b2009-08-03 17:06:42 +00005530
Fariborz Jahanian6b7cd6e2009-02-06 20:09:23 +00005531 // If this is a class message the metaclass is passed as the target.
Fariborz Jahanianbac73ac2009-02-28 20:07:56 +00005532 llvm::Value *Target;
5533 if (IsClassMessage) {
5534 if (isCategoryImpl) {
5535 // Message sent to "super' in a class method defined in
5536 // a category implementation.
Daniel Dunbar508a7dd2009-04-18 08:51:00 +00005537 Target = EmitClassRef(CGF.Builder, Class);
Fariborz Jahanianbac73ac2009-02-28 20:07:56 +00005538 Target = CGF.Builder.CreateStructGEP(Target, 0);
5539 Target = CGF.Builder.CreateLoad(Target);
Mike Stump658fe022009-07-30 22:28:39 +00005540 } else
Fariborz Jahanianbac73ac2009-02-28 20:07:56 +00005541 Target = EmitMetaClassRef(CGF.Builder, Class);
Mike Stump658fe022009-07-30 22:28:39 +00005542 } else
Daniel Dunbar508a7dd2009-04-18 08:51:00 +00005543 Target = EmitSuperClassRef(CGF.Builder, Class);
Daniel Dunbar59e476b2009-08-03 17:06:42 +00005544
Mike Stump18bb9282009-05-16 07:57:57 +00005545 // FIXME: We shouldn't need to do this cast, rectify the ASTContext and
5546 // ObjCTypes types.
Fariborz Jahanian6b7cd6e2009-02-06 20:09:23 +00005547 const llvm::Type *ClassTy =
5548 CGM.getTypes().ConvertType(CGF.getContext().getObjCClassType());
5549 Target = CGF.Builder.CreateBitCast(Target, ClassTy);
5550 CGF.Builder.CreateStore(Target,
5551 CGF.Builder.CreateStructGEP(ObjCSuper, 1));
Daniel Dunbar59e476b2009-08-03 17:06:42 +00005552
Fariborz Jahanian5d5ed2d2009-05-11 19:25:47 +00005553 return (LegacyDispatchedSelector(Sel))
John McCall78a15112010-05-22 01:48:05 +00005554 ? EmitLegacyMessageSend(CGF, Return, ResultType,
5555 EmitSelector(CGF.Builder, Sel),
Daniel Dunbar59e476b2009-08-03 17:06:42 +00005556 ObjCSuper, ObjCTypes.SuperPtrCTy,
Daniel Dunbaraff9fca2009-09-17 04:01:22 +00005557 true, CallArgs, Method, ObjCTypes)
John McCall78a15112010-05-22 01:48:05 +00005558 : EmitMessageSend(CGF, Return, ResultType, Sel,
Daniel Dunbar59e476b2009-08-03 17:06:42 +00005559 ObjCSuper, ObjCTypes.SuperPtrCTy,
5560 true, CallArgs);
Fariborz Jahanian6b7cd6e2009-02-06 20:09:23 +00005561}
Fariborz Jahanian74b77222009-02-11 20:51:17 +00005562
Daniel Dunbar59e476b2009-08-03 17:06:42 +00005563llvm::Value *CGObjCNonFragileABIMac::EmitSelector(CGBuilderTy &Builder,
Fariborz Jahanian9240f3d2010-06-17 19:56:20 +00005564 Selector Sel, bool lval) {
Fariborz Jahanian74b77222009-02-11 20:51:17 +00005565 llvm::GlobalVariable *&Entry = SelectorReferences[Sel];
Daniel Dunbar59e476b2009-08-03 17:06:42 +00005566
Fariborz Jahanian74b77222009-02-11 20:51:17 +00005567 if (!Entry) {
Daniel Dunbar59e476b2009-08-03 17:06:42 +00005568 llvm::Constant *Casted =
5569 llvm::ConstantExpr::getBitCast(GetMethodVarName(Sel),
5570 ObjCTypes.SelectorPtrTy);
5571 Entry =
5572 new llvm::GlobalVariable(CGM.getModule(), ObjCTypes.SelectorPtrTy, false,
5573 llvm::GlobalValue::InternalLinkage,
5574 Casted, "\01L_OBJC_SELECTOR_REFERENCES_");
Fariborz Jahanian5d5ed2d2009-05-11 19:25:47 +00005575 Entry->setSection("__DATA, __objc_selrefs, literal_pointers, no_dead_strip");
Chris Lattnerf56501c2009-07-17 23:57:13 +00005576 CGM.AddUsedGlobal(Entry);
Fariborz Jahanian74b77222009-02-11 20:51:17 +00005577 }
Daniel Dunbar59e476b2009-08-03 17:06:42 +00005578
Fariborz Jahanian9240f3d2010-06-17 19:56:20 +00005579 if (lval)
5580 return Entry;
Daniel Dunbarc76493a2009-11-29 21:23:36 +00005581 return Builder.CreateLoad(Entry, "tmp");
Fariborz Jahanian74b77222009-02-11 20:51:17 +00005582}
Fariborz Jahanian06292952009-02-16 22:52:32 +00005583/// EmitObjCIvarAssign - Code gen for assigning to a __strong object.
Fariborz Jahanian7a95d722009-09-24 22:25:38 +00005584/// objc_assign_ivar (id src, id *dst, ptrdiff_t)
Fariborz Jahanian06292952009-02-16 22:52:32 +00005585///
5586void CGObjCNonFragileABIMac::EmitObjCIvarAssign(CodeGen::CodeGenFunction &CGF,
Mike Stump11289f42009-09-09 15:08:12 +00005587 llvm::Value *src,
Fariborz Jahanian7a95d722009-09-24 22:25:38 +00005588 llvm::Value *dst,
5589 llvm::Value *ivarOffset) {
Fariborz Jahanianaedcfa42009-03-23 19:10:40 +00005590 const llvm::Type * SrcTy = src->getType();
5591 if (!isa<llvm::PointerType>(SrcTy)) {
Duncan Sandsc76fe8b2009-05-09 07:08:47 +00005592 unsigned Size = CGM.getTargetData().getTypeAllocSize(SrcTy);
Fariborz Jahanianaedcfa42009-03-23 19:10:40 +00005593 assert(Size <= 8 && "does not support size > 8");
5594 src = (Size == 4 ? CGF.Builder.CreateBitCast(src, ObjCTypes.IntTy)
5595 : CGF.Builder.CreateBitCast(src, ObjCTypes.LongTy));
Fariborz Jahanian1b074a32009-03-13 00:42:52 +00005596 src = CGF.Builder.CreateIntToPtr(src, ObjCTypes.Int8PtrTy);
5597 }
Fariborz Jahanian06292952009-02-16 22:52:32 +00005598 src = CGF.Builder.CreateBitCast(src, ObjCTypes.ObjectPtrTy);
5599 dst = CGF.Builder.CreateBitCast(dst, ObjCTypes.PtrObjectPtrTy);
Fariborz Jahanian7a95d722009-09-24 22:25:38 +00005600 CGF.Builder.CreateCall3(ObjCTypes.getGcAssignIvarFn(),
5601 src, dst, ivarOffset);
Fariborz Jahanian06292952009-02-16 22:52:32 +00005602 return;
5603}
5604
5605/// EmitObjCStrongCastAssign - Code gen for assigning to a __strong cast object.
5606/// objc_assign_strongCast (id src, id *dst)
5607///
5608void CGObjCNonFragileABIMac::EmitObjCStrongCastAssign(
Daniel Dunbar59e476b2009-08-03 17:06:42 +00005609 CodeGen::CodeGenFunction &CGF,
Mike Stump11289f42009-09-09 15:08:12 +00005610 llvm::Value *src, llvm::Value *dst) {
Fariborz Jahanianaedcfa42009-03-23 19:10:40 +00005611 const llvm::Type * SrcTy = src->getType();
5612 if (!isa<llvm::PointerType>(SrcTy)) {
Duncan Sandsc76fe8b2009-05-09 07:08:47 +00005613 unsigned Size = CGM.getTargetData().getTypeAllocSize(SrcTy);
Fariborz Jahanianaedcfa42009-03-23 19:10:40 +00005614 assert(Size <= 8 && "does not support size > 8");
5615 src = (Size == 4 ? CGF.Builder.CreateBitCast(src, ObjCTypes.IntTy)
Daniel Dunbar59e476b2009-08-03 17:06:42 +00005616 : CGF.Builder.CreateBitCast(src, ObjCTypes.LongTy));
Fariborz Jahanian1b074a32009-03-13 00:42:52 +00005617 src = CGF.Builder.CreateIntToPtr(src, ObjCTypes.Int8PtrTy);
5618 }
Fariborz Jahanian06292952009-02-16 22:52:32 +00005619 src = CGF.Builder.CreateBitCast(src, ObjCTypes.ObjectPtrTy);
5620 dst = CGF.Builder.CreateBitCast(dst, ObjCTypes.PtrObjectPtrTy);
Chris Lattner0a696a422009-04-22 02:38:11 +00005621 CGF.Builder.CreateCall2(ObjCTypes.getGcAssignStrongCastFn(),
Fariborz Jahanian06292952009-02-16 22:52:32 +00005622 src, dst, "weakassign");
5623 return;
5624}
5625
Fariborz Jahanian5f21d2f2009-07-08 01:18:33 +00005626void CGObjCNonFragileABIMac::EmitGCMemmoveCollectable(
Daniel Dunbar59e476b2009-08-03 17:06:42 +00005627 CodeGen::CodeGenFunction &CGF,
5628 llvm::Value *DestPtr,
5629 llvm::Value *SrcPtr,
Fariborz Jahanian021510e2010-06-15 22:44:06 +00005630 llvm::Value *Size) {
Fariborz Jahanian5f21d2f2009-07-08 01:18:33 +00005631 SrcPtr = CGF.Builder.CreateBitCast(SrcPtr, ObjCTypes.Int8PtrTy);
5632 DestPtr = CGF.Builder.CreateBitCast(DestPtr, ObjCTypes.Int8PtrTy);
Fariborz Jahanian5f21d2f2009-07-08 01:18:33 +00005633 CGF.Builder.CreateCall3(ObjCTypes.GcMemmoveCollectableFn(),
Fariborz Jahanian021510e2010-06-15 22:44:06 +00005634 DestPtr, SrcPtr, Size);
Fariborz Jahanian5f21d2f2009-07-08 01:18:33 +00005635 return;
5636}
5637
Fariborz Jahanian06292952009-02-16 22:52:32 +00005638/// EmitObjCWeakRead - Code gen for loading value of a __weak
5639/// object: objc_read_weak (id *src)
5640///
5641llvm::Value * CGObjCNonFragileABIMac::EmitObjCWeakRead(
Daniel Dunbar59e476b2009-08-03 17:06:42 +00005642 CodeGen::CodeGenFunction &CGF,
Mike Stump11289f42009-09-09 15:08:12 +00005643 llvm::Value *AddrWeakObj) {
Eli Friedmana374b682009-03-07 03:57:15 +00005644 const llvm::Type* DestTy =
Daniel Dunbar59e476b2009-08-03 17:06:42 +00005645 cast<llvm::PointerType>(AddrWeakObj->getType())->getElementType();
5646 AddrWeakObj = CGF.Builder.CreateBitCast(AddrWeakObj, ObjCTypes.PtrObjectPtrTy);
Chris Lattnerce8754e2009-04-22 02:44:54 +00005647 llvm::Value *read_weak = CGF.Builder.CreateCall(ObjCTypes.getGcReadWeakFn(),
Fariborz Jahanian06292952009-02-16 22:52:32 +00005648 AddrWeakObj, "weakread");
Eli Friedmana374b682009-03-07 03:57:15 +00005649 read_weak = CGF.Builder.CreateBitCast(read_weak, DestTy);
Fariborz Jahanian06292952009-02-16 22:52:32 +00005650 return read_weak;
5651}
5652
5653/// EmitObjCWeakAssign - Code gen for assigning to a __weak object.
5654/// objc_assign_weak (id src, id *dst)
5655///
5656void CGObjCNonFragileABIMac::EmitObjCWeakAssign(CodeGen::CodeGenFunction &CGF,
Mike Stump11289f42009-09-09 15:08:12 +00005657 llvm::Value *src, llvm::Value *dst) {
Fariborz Jahanianaedcfa42009-03-23 19:10:40 +00005658 const llvm::Type * SrcTy = src->getType();
5659 if (!isa<llvm::PointerType>(SrcTy)) {
Duncan Sandsc76fe8b2009-05-09 07:08:47 +00005660 unsigned Size = CGM.getTargetData().getTypeAllocSize(SrcTy);
Fariborz Jahanianaedcfa42009-03-23 19:10:40 +00005661 assert(Size <= 8 && "does not support size > 8");
5662 src = (Size == 4 ? CGF.Builder.CreateBitCast(src, ObjCTypes.IntTy)
5663 : CGF.Builder.CreateBitCast(src, ObjCTypes.LongTy));
Fariborz Jahanian1b074a32009-03-13 00:42:52 +00005664 src = CGF.Builder.CreateIntToPtr(src, ObjCTypes.Int8PtrTy);
5665 }
Fariborz Jahanian06292952009-02-16 22:52:32 +00005666 src = CGF.Builder.CreateBitCast(src, ObjCTypes.ObjectPtrTy);
5667 dst = CGF.Builder.CreateBitCast(dst, ObjCTypes.PtrObjectPtrTy);
Chris Lattner6fdd57c2009-04-17 22:12:36 +00005668 CGF.Builder.CreateCall2(ObjCTypes.getGcAssignWeakFn(),
Fariborz Jahanian06292952009-02-16 22:52:32 +00005669 src, dst, "weakassign");
5670 return;
5671}
5672
5673/// EmitObjCGlobalAssign - Code gen for assigning to a __strong object.
5674/// objc_assign_global (id src, id *dst)
5675///
5676void CGObjCNonFragileABIMac::EmitObjCGlobalAssign(CodeGen::CodeGenFunction &CGF,
Fariborz Jahanian217af242010-07-20 20:30:03 +00005677 llvm::Value *src, llvm::Value *dst,
5678 bool threadlocal) {
Fariborz Jahanianaedcfa42009-03-23 19:10:40 +00005679 const llvm::Type * SrcTy = src->getType();
5680 if (!isa<llvm::PointerType>(SrcTy)) {
Duncan Sandsc76fe8b2009-05-09 07:08:47 +00005681 unsigned Size = CGM.getTargetData().getTypeAllocSize(SrcTy);
Fariborz Jahanianaedcfa42009-03-23 19:10:40 +00005682 assert(Size <= 8 && "does not support size > 8");
5683 src = (Size == 4 ? CGF.Builder.CreateBitCast(src, ObjCTypes.IntTy)
5684 : CGF.Builder.CreateBitCast(src, ObjCTypes.LongTy));
Fariborz Jahanian1b074a32009-03-13 00:42:52 +00005685 src = CGF.Builder.CreateIntToPtr(src, ObjCTypes.Int8PtrTy);
5686 }
Fariborz Jahanian06292952009-02-16 22:52:32 +00005687 src = CGF.Builder.CreateBitCast(src, ObjCTypes.ObjectPtrTy);
5688 dst = CGF.Builder.CreateBitCast(dst, ObjCTypes.PtrObjectPtrTy);
Fariborz Jahanian217af242010-07-20 20:30:03 +00005689 if (!threadlocal)
5690 CGF.Builder.CreateCall2(ObjCTypes.getGcAssignGlobalFn(),
5691 src, dst, "globalassign");
5692 else
5693 CGF.Builder.CreateCall2(ObjCTypes.getGcAssignThreadLocalFn(),
5694 src, dst, "threadlocalassign");
Fariborz Jahanian06292952009-02-16 22:52:32 +00005695 return;
5696}
Fariborz Jahanian74b77222009-02-11 20:51:17 +00005697
Daniel Dunbar59e476b2009-08-03 17:06:42 +00005698void
John McCallbd309292010-07-06 01:34:17 +00005699CGObjCNonFragileABIMac::EmitSynchronizedStmt(CodeGen::CodeGenFunction &CGF,
5700 const ObjCAtSynchronizedStmt &S) {
5701 // Evaluate the lock operand. This should dominate the cleanup.
5702 llvm::Value *SyncArg = CGF.EmitScalarExpr(S.getSynchExpr());
Daniel Dunbar0b0dcd92009-02-24 07:47:38 +00005703
John McCallbd309292010-07-06 01:34:17 +00005704 // Acquire the lock.
5705 SyncArg = CGF.Builder.CreateBitCast(SyncArg, ObjCTypes.ObjectPtrTy);
5706 CGF.Builder.CreateCall(ObjCTypes.getSyncEnterFn(), SyncArg)
5707 ->setDoesNotThrow();
5708
5709 // Register an all-paths cleanup to release the lock.
5710 {
John McCall2b7fc382010-07-13 20:32:21 +00005711 CodeGenFunction::CleanupBlock ReleaseScope(CGF, NormalAndEHCleanup);
John McCallbd309292010-07-06 01:34:17 +00005712
5713 CGF.Builder.CreateCall(ObjCTypes.getSyncExitFn(), SyncArg)
5714 ->setDoesNotThrow();
Daniel Dunbar0b0dcd92009-02-24 07:47:38 +00005715 }
5716
John McCallbd309292010-07-06 01:34:17 +00005717 // Emit the body of the statement.
5718 CGF.EmitStmt(S.getSynchBody());
Daniel Dunbar0b0dcd92009-02-24 07:47:38 +00005719
John McCallbd309292010-07-06 01:34:17 +00005720 // Pop the lock-release cleanup.
5721 CGF.PopCleanupBlock();
5722}
Daniel Dunbar0b0dcd92009-02-24 07:47:38 +00005723
John McCallbd309292010-07-06 01:34:17 +00005724namespace {
5725 struct CatchHandler {
5726 const VarDecl *Variable;
5727 const Stmt *Body;
5728 llvm::BasicBlock *Block;
5729 llvm::Value *TypeInfo;
5730 };
John McCall5c08ab92010-07-13 22:12:14 +00005731
5732 struct CallObjCEndCatch : EHScopeStack::LazyCleanup {
5733 CallObjCEndCatch(bool MightThrow, llvm::Value *Fn) :
5734 MightThrow(MightThrow), Fn(Fn) {}
5735 bool MightThrow;
5736 llvm::Value *Fn;
5737
5738 void Emit(CodeGenFunction &CGF, bool IsForEH) {
5739 if (!MightThrow) {
5740 CGF.Builder.CreateCall(Fn)->setDoesNotThrow();
5741 return;
5742 }
5743
5744 CGF.EmitCallOrInvoke(Fn, 0, 0);
5745 }
5746 };
John McCallbd309292010-07-06 01:34:17 +00005747}
Daniel Dunbar59e476b2009-08-03 17:06:42 +00005748
John McCallbd309292010-07-06 01:34:17 +00005749void CGObjCNonFragileABIMac::EmitTryStmt(CodeGen::CodeGenFunction &CGF,
5750 const ObjCAtTryStmt &S) {
5751 // Jump destination for falling out of catch bodies.
5752 CodeGenFunction::JumpDest Cont;
5753 if (S.getNumCatchStmts())
5754 Cont = CGF.getJumpDestInCurrentScope("eh.cont");
Daniel Dunbar76b7acc2009-03-02 06:08:11 +00005755
John McCallbd309292010-07-06 01:34:17 +00005756 CodeGenFunction::FinallyInfo FinallyInfo;
5757 if (const ObjCAtFinallyStmt *Finally = S.getFinallyStmt())
5758 FinallyInfo = CGF.EnterFinallyBlock(Finally->getFinallyBody(),
5759 ObjCTypes.getObjCBeginCatchFn(),
5760 ObjCTypes.getObjCEndCatchFn(),
5761 ObjCTypes.getExceptionRethrowFn());
Daniel Dunbar76b7acc2009-03-02 06:08:11 +00005762
John McCallbd309292010-07-06 01:34:17 +00005763 llvm::SmallVector<CatchHandler, 8> Handlers;
Daniel Dunbar76b7acc2009-03-02 06:08:11 +00005764
John McCallbd309292010-07-06 01:34:17 +00005765 // Enter the catch, if there is one.
5766 if (S.getNumCatchStmts()) {
5767 for (unsigned I = 0, N = S.getNumCatchStmts(); I != N; ++I) {
5768 const ObjCAtCatchStmt *CatchStmt = S.getCatchStmt(I);
Douglas Gregor46a572b2010-04-26 16:46:50 +00005769 const VarDecl *CatchDecl = CatchStmt->getCatchParamDecl();
Daniel Dunbar76b7acc2009-03-02 06:08:11 +00005770
John McCallbd309292010-07-06 01:34:17 +00005771 Handlers.push_back(CatchHandler());
5772 CatchHandler &Handler = Handlers.back();
5773 Handler.Variable = CatchDecl;
5774 Handler.Body = CatchStmt->getCatchBody();
5775 Handler.Block = CGF.createBasicBlock("catch");
5776
5777 // @catch(...) always matches.
Douglas Gregor96c79492010-04-23 22:50:49 +00005778 if (!CatchDecl) {
John McCallbd309292010-07-06 01:34:17 +00005779 Handler.TypeInfo = 0; // catch-all
5780 // Don't consider any other catches.
Douglas Gregor96c79492010-04-23 22:50:49 +00005781 break;
5782 }
Daniel Dunbar76b7acc2009-03-02 06:08:11 +00005783
John McCallbd309292010-07-06 01:34:17 +00005784 // There's a particular fixed type info for 'id'.
Douglas Gregor96c79492010-04-23 22:50:49 +00005785 if (CatchDecl->getType()->isObjCIdType() ||
5786 CatchDecl->getType()->isObjCQualifiedIdType()) {
5787 llvm::Value *IDEHType =
5788 CGM.getModule().getGlobalVariable("OBJC_EHTYPE_id");
5789 if (!IDEHType)
5790 IDEHType =
5791 new llvm::GlobalVariable(CGM.getModule(), ObjCTypes.EHTypeTy,
5792 false,
5793 llvm::GlobalValue::ExternalLinkage,
5794 0, "OBJC_EHTYPE_id");
John McCallbd309292010-07-06 01:34:17 +00005795 Handler.TypeInfo = IDEHType;
Douglas Gregor96c79492010-04-23 22:50:49 +00005796 } else {
5797 // All other types should be Objective-C interface pointer types.
5798 const ObjCObjectPointerType *PT =
5799 CatchDecl->getType()->getAs<ObjCObjectPointerType>();
5800 assert(PT && "Invalid @catch type.");
5801 const ObjCInterfaceType *IT = PT->getInterfaceType();
5802 assert(IT && "Invalid @catch type.");
John McCallbd309292010-07-06 01:34:17 +00005803 Handler.TypeInfo = GetInterfaceEHType(IT->getDecl(), false);
Daniel Dunbar76b7acc2009-03-02 06:08:11 +00005804 }
5805 }
John McCallbd309292010-07-06 01:34:17 +00005806
5807 EHCatchScope *Catch = CGF.EHStack.pushCatch(Handlers.size());
5808 for (unsigned I = 0, E = Handlers.size(); I != E; ++I)
5809 Catch->setHandler(I, Handlers[I].TypeInfo, Handlers[I].Block);
Daniel Dunbar76b7acc2009-03-02 06:08:11 +00005810 }
John McCallbd309292010-07-06 01:34:17 +00005811
5812 // Emit the try body.
5813 CGF.EmitStmt(S.getTryBody());
Daniel Dunbar76b7acc2009-03-02 06:08:11 +00005814
John McCallbd309292010-07-06 01:34:17 +00005815 // Leave the try.
5816 if (S.getNumCatchStmts())
5817 CGF.EHStack.popCatch();
Daniel Dunbar59e476b2009-08-03 17:06:42 +00005818
John McCallbd309292010-07-06 01:34:17 +00005819 // Remember where we were.
5820 CGBuilderTy::InsertPoint SavedIP = CGF.Builder.saveAndClearIP();
Daniel Dunbar76b7acc2009-03-02 06:08:11 +00005821
John McCallbd309292010-07-06 01:34:17 +00005822 // Emit the handlers.
5823 for (unsigned I = 0, E = Handlers.size(); I != E; ++I) {
5824 CatchHandler &Handler = Handlers[I];
Daniel Dunbar76b7acc2009-03-02 06:08:11 +00005825
John McCallbd309292010-07-06 01:34:17 +00005826 CGF.EmitBlock(Handler.Block);
5827 llvm::Value *RawExn = CGF.Builder.CreateLoad(CGF.getExceptionSlot());
Daniel Dunbar76b7acc2009-03-02 06:08:11 +00005828
John McCallbd309292010-07-06 01:34:17 +00005829 // Enter the catch.
5830 llvm::CallInst *Exn =
5831 CGF.Builder.CreateCall(ObjCTypes.getObjCBeginCatchFn(), RawExn,
5832 "exn.adjusted");
5833 Exn->setDoesNotThrow();
Daniel Dunbar76b7acc2009-03-02 06:08:11 +00005834
John McCallbd309292010-07-06 01:34:17 +00005835 // Add a cleanup to leave the catch.
John McCall5c08ab92010-07-13 22:12:14 +00005836 bool EndCatchMightThrow = (Handler.Variable == 0);
5837 CGF.EHStack.pushLazyCleanup<CallObjCEndCatch>(NormalAndEHCleanup,
5838 EndCatchMightThrow,
5839 ObjCTypes.getObjCEndCatchFn());
Daniel Dunbar59e476b2009-08-03 17:06:42 +00005840
John McCallbd309292010-07-06 01:34:17 +00005841 // Bind the catch parameter if it exists.
5842 if (const VarDecl *CatchParam = Handler.Variable) {
5843 const llvm::Type *CatchType = CGF.ConvertType(CatchParam->getType());
5844 llvm::Value *CastExn = CGF.Builder.CreateBitCast(Exn, CatchType);
Daniel Dunbar76b7acc2009-03-02 06:08:11 +00005845
John McCallbd309292010-07-06 01:34:17 +00005846 CGF.EmitLocalBlockVarDecl(*CatchParam);
5847 CGF.Builder.CreateStore(CastExn, CGF.GetAddrOfLocalVar(CatchParam));
Daniel Dunbar76b7acc2009-03-02 06:08:11 +00005848 }
Daniel Dunbar76b7acc2009-03-02 06:08:11 +00005849
John McCallbd309292010-07-06 01:34:17 +00005850 CGF.ObjCEHValueStack.push_back(Exn);
5851 CGF.EmitStmt(Handler.Body);
5852 CGF.ObjCEHValueStack.pop_back();
Daniel Dunbar0b0dcd92009-02-24 07:47:38 +00005853
John McCallbd309292010-07-06 01:34:17 +00005854 // Leave the earlier cleanup.
5855 CGF.PopCleanupBlock();
Daniel Dunbar0b0dcd92009-02-24 07:47:38 +00005856
John McCallbd309292010-07-06 01:34:17 +00005857 CGF.EmitBranchThroughCleanup(Cont);
5858 }
Daniel Dunbar0b0dcd92009-02-24 07:47:38 +00005859
John McCallbd309292010-07-06 01:34:17 +00005860 // Go back to the try-statement fallthrough.
5861 CGF.Builder.restoreIP(SavedIP);
Daniel Dunbar0b0dcd92009-02-24 07:47:38 +00005862
John McCallbd309292010-07-06 01:34:17 +00005863 // Pop out of the normal cleanup on the finally.
5864 if (S.getFinallyStmt())
5865 CGF.ExitFinallyBlock(FinallyInfo);
Daniel Dunbar0b0dcd92009-02-24 07:47:38 +00005866
John McCallbd309292010-07-06 01:34:17 +00005867 if (Cont.Block)
5868 CGF.EmitBlock(Cont.Block);
Daniel Dunbar0b0dcd92009-02-24 07:47:38 +00005869}
5870
Anders Carlsson9ab53d12009-02-16 22:59:18 +00005871/// EmitThrowStmt - Generate code for a throw statement.
5872void CGObjCNonFragileABIMac::EmitThrowStmt(CodeGen::CodeGenFunction &CGF,
5873 const ObjCAtThrowStmt &S) {
Daniel Dunbar59e476b2009-08-03 17:06:42 +00005874 llvm::Value *Exception;
Fariborz Jahanian3336de12010-05-28 17:34:43 +00005875 llvm::Constant *FunctionThrowOrRethrow;
Anders Carlsson9ab53d12009-02-16 22:59:18 +00005876 if (const Expr *ThrowExpr = S.getThrowExpr()) {
Daniel Dunbar76b7acc2009-03-02 06:08:11 +00005877 Exception = CGF.EmitScalarExpr(ThrowExpr);
Fariborz Jahanian3336de12010-05-28 17:34:43 +00005878 FunctionThrowOrRethrow = ObjCTypes.getExceptionThrowFn();
Anders Carlsson9ab53d12009-02-16 22:59:18 +00005879 } else {
Daniel Dunbar59e476b2009-08-03 17:06:42 +00005880 assert((!CGF.ObjCEHValueStack.empty() && CGF.ObjCEHValueStack.back()) &&
Daniel Dunbar76b7acc2009-03-02 06:08:11 +00005881 "Unexpected rethrow outside @catch block.");
5882 Exception = CGF.ObjCEHValueStack.back();
Fariborz Jahanian3336de12010-05-28 17:34:43 +00005883 FunctionThrowOrRethrow = ObjCTypes.getExceptionRethrowFn();
Anders Carlsson9ab53d12009-02-16 22:59:18 +00005884 }
5885
Daniel Dunbar59e476b2009-08-03 17:06:42 +00005886 llvm::Value *ExceptionAsObject =
Daniel Dunbar76b7acc2009-03-02 06:08:11 +00005887 CGF.Builder.CreateBitCast(Exception, ObjCTypes.ObjectPtrTy, "tmp");
5888 llvm::BasicBlock *InvokeDest = CGF.getInvokeDest();
5889 if (InvokeDest) {
Fariborz Jahanian3336de12010-05-28 17:34:43 +00005890 CGF.Builder.CreateInvoke(FunctionThrowOrRethrow,
John McCallbd309292010-07-06 01:34:17 +00005891 CGF.getUnreachableBlock(), InvokeDest,
Daniel Dunbar76b7acc2009-03-02 06:08:11 +00005892 &ExceptionAsObject, &ExceptionAsObject + 1);
John McCallbd309292010-07-06 01:34:17 +00005893 } else {
5894 CGF.Builder.CreateCall(FunctionThrowOrRethrow, ExceptionAsObject)
5895 ->setDoesNotReturn();
5896 CGF.Builder.CreateUnreachable();
5897 }
Daniel Dunbar76b7acc2009-03-02 06:08:11 +00005898
Anders Carlsson9ab53d12009-02-16 22:59:18 +00005899 // Clear the insertion point to indicate we are in unreachable code.
5900 CGF.Builder.ClearInsertionPoint();
5901}
Daniel Dunbarb1559a42009-03-01 04:46:24 +00005902
5903llvm::Value *
Daniel Dunbar59e476b2009-08-03 17:06:42 +00005904CGObjCNonFragileABIMac::GetInterfaceEHType(const ObjCInterfaceDecl *ID,
Daniel Dunbar8f28d012009-04-08 04:21:03 +00005905 bool ForDefinition) {
Daniel Dunbarb1559a42009-03-01 04:46:24 +00005906 llvm::GlobalVariable * &Entry = EHTypeReferences[ID->getIdentifier()];
Daniel Dunbarb1559a42009-03-01 04:46:24 +00005907
Daniel Dunbar8f28d012009-04-08 04:21:03 +00005908 // If we don't need a definition, return the entry if found or check
5909 // if we use an external reference.
5910 if (!ForDefinition) {
5911 if (Entry)
5912 return Entry;
Daniel Dunbard7beeea2009-04-07 06:43:45 +00005913
Daniel Dunbar8f28d012009-04-08 04:21:03 +00005914 // If this type (or a super class) has the __objc_exception__
5915 // attribute, emit an external reference.
Douglas Gregor78bd61f2009-06-18 16:11:24 +00005916 if (hasObjCExceptionAttribute(CGM.getContext(), ID))
Daniel Dunbar59e476b2009-08-03 17:06:42 +00005917 return Entry =
Owen Andersonc10c8d32009-07-08 19:05:04 +00005918 new llvm::GlobalVariable(CGM.getModule(), ObjCTypes.EHTypeTy, false,
Daniel Dunbar8f28d012009-04-08 04:21:03 +00005919 llvm::GlobalValue::ExternalLinkage,
Daniel Dunbar59e476b2009-08-03 17:06:42 +00005920 0,
Daniel Dunbar349e6fb2009-10-18 20:48:59 +00005921 ("OBJC_EHTYPE_$_" +
Daniel Dunbar07d07852009-10-18 21:17:35 +00005922 ID->getIdentifier()->getName()));
Daniel Dunbar8f28d012009-04-08 04:21:03 +00005923 }
Daniel Dunbar59e476b2009-08-03 17:06:42 +00005924
Daniel Dunbar8f28d012009-04-08 04:21:03 +00005925 // Otherwise we need to either make a new entry or fill in the
5926 // initializer.
5927 assert((!Entry || !Entry->hasInitializer()) && "Duplicate EHType definition");
Daniel Dunbar15894b72009-04-07 05:48:37 +00005928 std::string ClassName(getClassSymbolPrefix() + ID->getNameAsString());
Daniel Dunbarb1559a42009-03-01 04:46:24 +00005929 std::string VTableName = "objc_ehtype_vtable";
Daniel Dunbar59e476b2009-08-03 17:06:42 +00005930 llvm::GlobalVariable *VTableGV =
Daniel Dunbarb1559a42009-03-01 04:46:24 +00005931 CGM.getModule().getGlobalVariable(VTableName);
5932 if (!VTableGV)
Daniel Dunbar59e476b2009-08-03 17:06:42 +00005933 VTableGV = new llvm::GlobalVariable(CGM.getModule(), ObjCTypes.Int8PtrTy,
5934 false,
Daniel Dunbarb1559a42009-03-01 04:46:24 +00005935 llvm::GlobalValue::ExternalLinkage,
Owen Andersonc10c8d32009-07-08 19:05:04 +00005936 0, VTableName);
Daniel Dunbarb1559a42009-03-01 04:46:24 +00005937
Chris Lattner5e016ae2010-06-27 07:15:29 +00005938 llvm::Value *VTableIdx =
5939 llvm::ConstantInt::get(llvm::Type::getInt32Ty(VMContext), 2);
Daniel Dunbarb1559a42009-03-01 04:46:24 +00005940
5941 std::vector<llvm::Constant*> Values(3);
Owen Andersonade90fd2009-07-29 18:54:39 +00005942 Values[0] = llvm::ConstantExpr::getGetElementPtr(VTableGV, &VTableIdx, 1);
Daniel Dunbarb1559a42009-03-01 04:46:24 +00005943 Values[1] = GetClassName(ID->getIdentifier());
Fariborz Jahanian899e7eb2009-04-14 18:41:56 +00005944 Values[2] = GetClassGlobal(ClassName);
Owen Anderson170229f2009-07-14 23:10:40 +00005945 llvm::Constant *Init =
Owen Anderson0e0189d2009-07-27 22:29:56 +00005946 llvm::ConstantStruct::get(ObjCTypes.EHTypeTy, Values);
Daniel Dunbarb1559a42009-03-01 04:46:24 +00005947
Daniel Dunbar8f28d012009-04-08 04:21:03 +00005948 if (Entry) {
5949 Entry->setInitializer(Init);
5950 } else {
Owen Andersonc10c8d32009-07-08 19:05:04 +00005951 Entry = new llvm::GlobalVariable(CGM.getModule(), ObjCTypes.EHTypeTy, false,
Daniel Dunbar8f28d012009-04-08 04:21:03 +00005952 llvm::GlobalValue::WeakAnyLinkage,
Daniel Dunbar59e476b2009-08-03 17:06:42 +00005953 Init,
Daniel Dunbar349e6fb2009-10-18 20:48:59 +00005954 ("OBJC_EHTYPE_$_" +
Daniel Dunbar07d07852009-10-18 21:17:35 +00005955 ID->getIdentifier()->getName()));
Daniel Dunbar8f28d012009-04-08 04:21:03 +00005956 }
5957
Daniel Dunbarf5f359f2009-04-14 06:00:08 +00005958 if (CGM.getLangOptions().getVisibilityMode() == LangOptions::Hidden)
Daniel Dunbar15894b72009-04-07 05:48:37 +00005959 Entry->setVisibility(llvm::GlobalValue::HiddenVisibility);
Daniel Dunbar710cb202010-04-25 20:39:32 +00005960 Entry->setAlignment(CGM.getTargetData().getABITypeAlignment(
5961 ObjCTypes.EHTypeTy));
Daniel Dunbar8f28d012009-04-08 04:21:03 +00005962
5963 if (ForDefinition) {
5964 Entry->setSection("__DATA,__objc_const");
5965 Entry->setLinkage(llvm::GlobalValue::ExternalLinkage);
5966 } else {
5967 Entry->setSection("__DATA,__datacoal_nt,coalesced");
5968 }
Daniel Dunbarb1559a42009-03-01 04:46:24 +00005969
5970 return Entry;
5971}
Daniel Dunbar59e476b2009-08-03 17:06:42 +00005972
Daniel Dunbar8b8683f2008-08-12 00:12:39 +00005973/* *** */
5974
Daniel Dunbarb036db82008-08-13 03:21:16 +00005975CodeGen::CGObjCRuntime *
5976CodeGen::CreateMacObjCRuntime(CodeGen::CodeGenModule &CGM) {
Daniel Dunbar303e2c22008-08-11 02:45:11 +00005977 return new CGObjCMac(CGM);
5978}
Fariborz Jahanian279eda62009-01-21 22:04:16 +00005979
5980CodeGen::CGObjCRuntime *
Fariborz Jahanianb15a3d52009-01-22 23:02:58 +00005981CodeGen::CreateMacNonFragileABIObjCRuntime(CodeGen::CodeGenModule &CGM) {
Fariborz Jahanian71394042009-01-23 23:53:38 +00005982 return new CGObjCNonFragileABIMac(CGM);
Fariborz Jahanian279eda62009-01-21 22:04:16 +00005983}