blob: 2992867cbc1273fc08e638850e8943d5f1b47b4b [file] [log] [blame]
Daniel Dunbarc17a4d32008-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 Dunbarf77ac862008-08-11 21:35:06 +000015
Daniel Dunbar198bcb42010-03-31 01:09:11 +000016#include "CGRecordLayout.h"
Daniel Dunbarf77ac862008-08-11 21:35:06 +000017#include "CodeGenModule.h"
Daniel Dunbarb7ec2462008-08-16 03:19:19 +000018#include "CodeGenFunction.h"
John McCallf1549f62010-07-06 01:34:17 +000019#include "CGException.h"
Daniel Dunbarbbce49b2008-08-12 00:12:39 +000020#include "clang/AST/ASTContext.h"
Daniel Dunbare91593e2008-08-11 04:54:23 +000021#include "clang/AST/Decl.h"
Daniel Dunbar6efc0c52008-08-13 03:21:16 +000022#include "clang/AST/DeclObjC.h"
Anders Carlsson19cc4ab2009-07-18 19:43:29 +000023#include "clang/AST/RecordLayout.h"
Chris Lattner16f00492009-04-26 01:32:48 +000024#include "clang/AST/StmtObjC.h"
Daniel Dunbarf77ac862008-08-11 21:35:06 +000025#include "clang/Basic/LangOptions.h"
Chandler Carruth06057ce2010-06-15 23:19:56 +000026#include "clang/Frontend/CodeGenOptions.h"
Daniel Dunbarf77ac862008-08-11 21:35:06 +000027
Daniel Dunbar8ecbaf22009-02-24 07:47:38 +000028#include "llvm/Intrinsics.h"
Owen Anderson69243822009-07-13 04:10:07 +000029#include "llvm/LLVMContext.h"
Daniel Dunbarbbce49b2008-08-12 00:12:39 +000030#include "llvm/Module.h"
Daniel Dunbar0c0e7a62008-10-29 22:36:39 +000031#include "llvm/ADT/DenseSet.h"
Daniel Dunbar33063492009-09-07 00:20:42 +000032#include "llvm/ADT/SetVector.h"
33#include "llvm/ADT/SmallString.h"
Fariborz Jahanian191dcd72009-12-12 21:26:21 +000034#include "llvm/ADT/SmallPtrSet.h"
John McCall8e3f8612010-07-13 22:12:14 +000035#include "llvm/Support/CallSite.h"
Daniel Dunbar33063492009-09-07 00:20:42 +000036#include "llvm/Support/raw_ostream.h"
Daniel Dunbar4e2d7d02008-08-12 06:48:42 +000037#include "llvm/Target/TargetData.h"
Torok Edwinf42e4a62009-08-24 13:25:12 +000038#include <cstdio>
Daniel Dunbarc17a4d32008-08-11 02:45:11 +000039
40using namespace clang;
Daniel Dunbar46f45b92008-09-09 01:06:48 +000041using namespace CodeGen;
Daniel Dunbarc17a4d32008-08-11 02:45:11 +000042
Daniel Dunbar97776872009-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 Dunbar1d7e5392009-05-03 08:55:17 +000047static uint64_t LookupFieldBitOffset(CodeGen::CodeGenModule &CGM,
48 const ObjCInterfaceDecl *OID,
Daniel Dunbar9f89f2b2009-05-03 12:57:56 +000049 const ObjCImplementationDecl *ID,
Daniel Dunbar1d7e5392009-05-03 08:55:17 +000050 const ObjCIvarDecl *Ivar) {
Daniel Dunbare83be122010-04-02 21:14:02 +000051 const ObjCInterfaceDecl *Container = Ivar->getContainingInterface();
Daniel Dunbar532d4da2009-05-03 13:15:50 +000052
Daniel Dunbar61ac1d22010-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 Dunbar3a2c80f2010-04-02 15:43:29 +000056
Daniel Dunbar532d4da2009-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 Dunbar6bff2512009-08-03 17:06:42 +000059 const ASTRecordLayout *RL;
Daniel Dunbar532d4da2009-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 Dunbare83be122010-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 Dunbar532d4da2009-05-03 13:15:50 +000080 return RL->getFieldOffset(Index);
Daniel Dunbar1d7e5392009-05-03 08:55:17 +000081}
82
83uint64_t CGObjCRuntime::ComputeIvarBaseOffset(CodeGen::CodeGenModule &CGM,
84 const ObjCInterfaceDecl *OID,
85 const ObjCIvarDecl *Ivar) {
Daniel Dunbar9f89f2b2009-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 Dunbar97776872009-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 Dunbar1d7e5392009-05-03 08:55:17 +0000101 // Compute (type*) ( (char *) BaseValue + Offset)
Benjamin Kramer3c0ef8c2009-10-13 10:07:13 +0000102 const llvm::Type *I8Ptr = llvm::Type::getInt8PtrTy(CGF.getLLVMContext());
Daniel Dunbar1d7e5392009-05-03 08:55:17 +0000103 QualType IvarTy = Ivar->getType();
104 const llvm::Type *LTy = CGF.CGM.getTypes().ConvertTypeForMem(IvarTy);
Daniel Dunbar97776872009-04-22 07:32:20 +0000105 llvm::Value *V = CGF.Builder.CreateBitCast(BaseValue, I8Ptr);
Daniel Dunbar97776872009-04-22 07:32:20 +0000106 V = CGF.Builder.CreateGEP(V, Offset, "add.ptr");
Owen Anderson96e0fc72009-07-29 22:16:19 +0000107 V = CGF.Builder.CreateBitCast(V, llvm::PointerType::getUnqual(LTy));
Daniel Dunbar6bff2512009-08-03 17:06:42 +0000108
John McCall0953e762009-09-24 19:53:00 +0000109 Qualifiers Quals = CGF.MakeQualifiers(IvarTy);
110 Quals.addCVRQualifiers(CVRQualifiers);
111
Daniel Dunbar56229f52010-04-05 16:20:33 +0000112 if (!Ivar->isBitField())
113 return LValue::MakeAddr(V, Quals);
Daniel Dunbar9f89f2b2009-05-03 12:57:56 +0000114
Daniel Dunbar56229f52010-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 Dunbar97776872009-04-22 07:32:20 +0000122
Daniel Dunbarf0fe5bc2010-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 Dunbarf0fe5bc2010-04-05 21:36:35 +0000129
Daniel Dunbarab970f92010-04-13 20:58:55 +0000130 // We always construct a single, possibly unaligned, access for this case.
Daniel Dunbar2df25692010-04-15 05:09:32 +0000131 CGBitFieldInfo::AccessInfo AI;
Daniel Dunbarab970f92010-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 Dunbar2df25692010-04-15 05:09:32 +0000140 CGBitFieldInfo *Info =
141 new (CGF.CGM.getContext()) CGBitFieldInfo(BitFieldSize, 1, &AI,
142 IvarTy->isSignedIntegerType());
143
Daniel Dunbar56229f52010-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 Dunbarefbf4872010-04-06 01:07:44 +0000146 return LValue::MakeBitfield(V, *Info, Quals.getCVRQualifiers());
Daniel Dunbar97776872009-04-22 07:32:20 +0000147}
148
149///
150
Daniel Dunbarc17a4d32008-08-11 02:45:11 +0000151namespace {
Daniel Dunbarbbce49b2008-08-12 00:12:39 +0000152
Daniel Dunbar6bff2512009-08-03 17:06:42 +0000153typedef std::vector<llvm::Constant*> ConstantVector;
Daniel Dunbarae226fa2008-08-27 02:31:56 +0000154
Daniel Dunbar6bff2512009-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 Dunbar6efc0c52008-08-13 03:21:16 +0000157
Fariborz Jahanianee0af742009-01-21 22:04:16 +0000158class ObjCCommonTypesHelper {
Owen Andersona1cf15f2009-07-14 23:10:40 +0000159protected:
160 llvm::LLVMContext &VMContext;
Daniel Dunbar6bff2512009-08-03 17:06:42 +0000161
Fariborz Jahaniand0f8a8d2009-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 Dunbar6bff2512009-08-03 17:06:42 +0000169 CGM.CreateRuntimeFunction(llvm::FunctionType::get(ObjectPtrTy,
170 Params, true),
171 "objc_msgSend");
Fariborz Jahaniand0f8a8d2009-05-11 19:25:47 +0000172 }
Daniel Dunbar6bff2512009-08-03 17:06:42 +0000173
Fariborz Jahaniand0f8a8d2009-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 Anderson0032b272009-08-13 21:57:51 +0000180 CGM.CreateRuntimeFunction(llvm::FunctionType::get(llvm::Type::getVoidTy(VMContext),
Daniel Dunbar6bff2512009-08-03 17:06:42 +0000181 Params, true),
182 "objc_msgSend_stret");
183
Fariborz Jahaniand0f8a8d2009-05-11 19:25:47 +0000184 }
Daniel Dunbar6bff2512009-08-03 17:06:42 +0000185
Fariborz Jahaniand0f8a8d2009-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 Anderson0032b272009-08-13 21:57:51 +0000193 CGM.CreateRuntimeFunction(llvm::FunctionType::get(
194 llvm::Type::getDoubleTy(VMContext),
Daniel Dunbar6bff2512009-08-03 17:06:42 +0000195 Params,
196 true),
197 "objc_msgSend_fpret");
198
Fariborz Jahaniand0f8a8d2009-05-11 19:25:47 +0000199 }
Daniel Dunbar6bff2512009-08-03 17:06:42 +0000200
Fariborz Jahaniand0f8a8d2009-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 Anderson96e0fc72009-07-29 22:16:19 +0000207 return CGM.CreateRuntimeFunction(llvm::FunctionType::get(ObjectPtrTy,
Fariborz Jahaniand0f8a8d2009-05-11 19:25:47 +0000208 Params, true),
209 SuperName);
210 }
Daniel Dunbar6bff2512009-08-03 17:06:42 +0000211
Fariborz Jahaniand0f8a8d2009-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 Anderson96e0fc72009-07-29 22:16:19 +0000218 return CGM.CreateRuntimeFunction(llvm::FunctionType::get(ObjectPtrTy,
Fariborz Jahaniand0f8a8d2009-05-11 19:25:47 +0000219 Params, true),
220 SuperName);
221 }
Daniel Dunbar6bff2512009-08-03 17:06:42 +0000222
Fariborz Jahaniand0f8a8d2009-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 Andersona1cf15f2009-07-14 23:10:40 +0000230 return CGM.CreateRuntimeFunction(
Owen Anderson0032b272009-08-13 21:57:51 +0000231 llvm::FunctionType::get(llvm::Type::getVoidTy(VMContext),
Daniel Dunbar6bff2512009-08-03 17:06:42 +0000232 Params, true),
233 "objc_msgSendSuper_stret");
Fariborz Jahaniand0f8a8d2009-05-11 19:25:47 +0000234 }
Daniel Dunbar6bff2512009-08-03 17:06:42 +0000235
Fariborz Jahaniand0f8a8d2009-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 Andersona1cf15f2009-07-14 23:10:40 +0000243 return CGM.CreateRuntimeFunction(
Owen Anderson0032b272009-08-13 21:57:51 +0000244 llvm::FunctionType::get(llvm::Type::getVoidTy(VMContext),
Daniel Dunbar6bff2512009-08-03 17:06:42 +0000245 Params, true),
246 "objc_msgSendSuper2_stret");
Fariborz Jahaniand0f8a8d2009-05-11 19:25:47 +0000247 }
Daniel Dunbar6bff2512009-08-03 17:06:42 +0000248
Fariborz Jahaniand0f8a8d2009-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 Dunbar6bff2512009-08-03 17:06:42 +0000253
Fariborz Jahaniand0f8a8d2009-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 Dunbar6bff2512009-08-03 17:06:42 +0000258
Fariborz Jahanianee0af742009-01-21 22:04:16 +0000259protected:
260 CodeGen::CodeGenModule &CGM;
Daniel Dunbar6bff2512009-08-03 17:06:42 +0000261
Daniel Dunbarbbce49b2008-08-12 00:12:39 +0000262public:
Fariborz Jahanian0a855d02009-03-23 19:10:40 +0000263 const llvm::Type *ShortTy, *IntTy, *LongTy, *LongLongTy;
Daniel Dunbar27f9d772008-08-21 04:36:09 +0000264 const llvm::Type *Int8PtrTy;
Daniel Dunbar6bff2512009-08-03 17:06:42 +0000265
Daniel Dunbar2bedbf82008-08-12 05:28:47 +0000266 /// ObjectPtrTy - LLVM type for object handles (typeof(id))
267 const llvm::Type *ObjectPtrTy;
Daniel Dunbar6bff2512009-08-03 17:06:42 +0000268
Fariborz Jahanian6d657c42008-11-18 20:18:11 +0000269 /// PtrObjectPtrTy - LLVM type for id *
270 const llvm::Type *PtrObjectPtrTy;
Daniel Dunbar6bff2512009-08-03 17:06:42 +0000271
Daniel Dunbar4e2d7d02008-08-12 06:48:42 +0000272 /// SelectorPtrTy - LLVM type for selector handles (typeof(SEL))
Daniel Dunbar2bedbf82008-08-12 05:28:47 +0000273 const llvm::Type *SelectorPtrTy;
Daniel Dunbar6efc0c52008-08-13 03:21:16 +0000274 /// ProtocolPtrTy - LLVM type for external protocol handles
275 /// (typeof(Protocol))
276 const llvm::Type *ExternalProtocolPtrTy;
Daniel Dunbar6bff2512009-08-03 17:06:42 +0000277
Daniel Dunbar19cd87e2008-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 Dunbar6bff2512009-08-03 17:06:42 +0000282
Daniel Dunbare8b470d2008-08-23 04:28:29 +0000283 /// SuperTy - LLVM type for struct objc_super.
284 const llvm::StructType *SuperTy;
Daniel Dunbar14c80b72008-08-23 09:25:55 +0000285 /// SuperPtrTy - LLVM type for struct objc_super *.
286 const llvm::Type *SuperPtrTy;
Daniel Dunbar6bff2512009-08-03 17:06:42 +0000287
Fariborz Jahanian30bc5712009-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 Dunbar6bff2512009-08-03 17:06:42 +0000291
Fariborz Jahanian30bc5712009-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 Dunbar6bff2512009-08-03 17:06:42 +0000297
Fariborz Jahanian30bc5712009-01-22 23:02:58 +0000298 // MethodTy - LLVM type for struct objc_method.
299 const llvm::StructType *MethodTy;
Daniel Dunbar6bff2512009-08-03 17:06:42 +0000300
Fariborz Jahaniand55b6fc2009-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 Dunbar6bff2512009-08-03 17:06:42 +0000305
Chris Lattner72db6c32009-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 McCallead608a2010-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 Lattner72db6c32009-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 McCall04a67a62010-02-05 21:31:56 +0000318 Types.GetFunctionType(Types.getFunctionInfo(IdType, Params,
Rafael Espindola264ba482010-03-30 20:24:48 +0000319 FunctionType::ExtInfo()),
320 false);
Chris Lattner72db6c32009-04-22 02:44:54 +0000321 return CGM.CreateRuntimeFunction(FTy, "objc_getProperty");
322 }
Daniel Dunbar6bff2512009-08-03 17:06:42 +0000323
Chris Lattner72db6c32009-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 McCallead608a2010-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 Lattner72db6c32009-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 McCall04a67a62010-02-05 21:31:56 +0000338 Types.GetFunctionType(Types.getFunctionInfo(Ctx.VoidTy, Params,
Rafael Espindola264ba482010-03-30 20:24:48 +0000339 FunctionType::ExtInfo()),
340 false);
Chris Lattner72db6c32009-04-22 02:44:54 +0000341 return CGM.CreateRuntimeFunction(FTy, "objc_setProperty");
342 }
Daniel Dunbar6bff2512009-08-03 17:06:42 +0000343
Fariborz Jahanian6cc59062010-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 Lattner72db6c32009-04-22 02:44:54 +0000362 llvm::Constant *getEnumerationMutationFn() {
Daniel Dunbarc1ab9002009-07-11 20:32:50 +0000363 CodeGen::CodeGenTypes &Types = CGM.getTypes();
364 ASTContext &Ctx = CGM.getContext();
Chris Lattner72db6c32009-04-22 02:44:54 +0000365 // void objc_enumerationMutation (id)
John McCallead608a2010-02-26 00:48:12 +0000366 llvm::SmallVector<CanQualType,1> Params;
367 Params.push_back(Ctx.getCanonicalParamType(Ctx.getObjCIdType()));
Daniel Dunbarc1ab9002009-07-11 20:32:50 +0000368 const llvm::FunctionType *FTy =
John McCall04a67a62010-02-05 21:31:56 +0000369 Types.GetFunctionType(Types.getFunctionInfo(Ctx.VoidTy, Params,
Rafael Espindola264ba482010-03-30 20:24:48 +0000370 FunctionType::ExtInfo()),
371 false);
Chris Lattner72db6c32009-04-22 02:44:54 +0000372 return CGM.CreateRuntimeFunction(FTy, "objc_enumerationMutation");
373 }
Daniel Dunbar6bff2512009-08-03 17:06:42 +0000374
Fariborz Jahaniandb286862009-01-22 00:37:21 +0000375 /// GcReadWeakFn -- LLVM objc_read_weak (id *src) function.
Chris Lattner72db6c32009-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 Dunbar6bff2512009-08-03 17:06:42 +0000380 llvm::FunctionType *FTy =
Owen Anderson96e0fc72009-07-29 22:16:19 +0000381 llvm::FunctionType::get(ObjectPtrTy, Args, false);
Chris Lattner72db6c32009-04-22 02:44:54 +0000382 return CGM.CreateRuntimeFunction(FTy, "objc_read_weak");
Daniel Dunbar6bff2512009-08-03 17:06:42 +0000383 }
384
Fariborz Jahaniandb286862009-01-22 00:37:21 +0000385 /// GcAssignWeakFn -- LLVM objc_assign_weak function.
Chris Lattner96508e12009-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 Anderson96e0fc72009-07-29 22:16:19 +0000391 llvm::FunctionType::get(ObjectPtrTy, Args, false);
Chris Lattner96508e12009-04-17 22:12:36 +0000392 return CGM.CreateRuntimeFunction(FTy, "objc_assign_weak");
393 }
Daniel Dunbar6bff2512009-08-03 17:06:42 +0000394
Fariborz Jahaniandb286862009-01-22 00:37:21 +0000395 /// GcAssignGlobalFn -- LLVM objc_assign_global function.
Chris Lattnerbbccd612009-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 Andersona1cf15f2009-07-14 23:10:40 +0000400 llvm::FunctionType *FTy =
Owen Anderson96e0fc72009-07-29 22:16:19 +0000401 llvm::FunctionType::get(ObjectPtrTy, Args, false);
Chris Lattnerbbccd612009-04-22 02:38:11 +0000402 return CGM.CreateRuntimeFunction(FTy, "objc_assign_global");
403 }
Daniel Dunbar6bff2512009-08-03 17:06:42 +0000404
Fariborz Jahaniandb286862009-01-22 00:37:21 +0000405 /// GcAssignIvarFn -- LLVM objc_assign_ivar function.
Chris Lattnerbbccd612009-04-22 02:38:11 +0000406 llvm::Constant *getGcAssignIvarFn() {
Fariborz Jahanian6c7a1f32009-09-24 22:25:38 +0000407 // id objc_assign_ivar(id, id *, ptrdiff_t)
Chris Lattnerbbccd612009-04-22 02:38:11 +0000408 std::vector<const llvm::Type*> Args(1, ObjectPtrTy);
409 Args.push_back(ObjectPtrTy->getPointerTo());
Fariborz Jahanian6c7a1f32009-09-24 22:25:38 +0000410 Args.push_back(LongTy);
Owen Andersona1cf15f2009-07-14 23:10:40 +0000411 llvm::FunctionType *FTy =
Owen Anderson96e0fc72009-07-29 22:16:19 +0000412 llvm::FunctionType::get(ObjectPtrTy, Args, false);
Chris Lattnerbbccd612009-04-22 02:38:11 +0000413 return CGM.CreateRuntimeFunction(FTy, "objc_assign_ivar");
414 }
Daniel Dunbar6bff2512009-08-03 17:06:42 +0000415
Fariborz Jahanian082b02e2009-07-08 01:18:33 +0000416 /// GcMemmoveCollectableFn -- LLVM objc_memmove_collectable function.
417 llvm::Constant *GcMemmoveCollectableFn() {
418 // void *objc_memmove_collectable(void *dst, const void *src, size_t size)
419 std::vector<const llvm::Type*> Args(1, Int8PtrTy);
420 Args.push_back(Int8PtrTy);
421 Args.push_back(LongTy);
Owen Anderson96e0fc72009-07-29 22:16:19 +0000422 llvm::FunctionType *FTy = llvm::FunctionType::get(Int8PtrTy, Args, false);
Fariborz Jahanian082b02e2009-07-08 01:18:33 +0000423 return CGM.CreateRuntimeFunction(FTy, "objc_memmove_collectable");
424 }
Daniel Dunbar6bff2512009-08-03 17:06:42 +0000425
Fariborz Jahaniandb286862009-01-22 00:37:21 +0000426 /// GcAssignStrongCastFn -- LLVM objc_assign_strongCast function.
Chris Lattnerbbccd612009-04-22 02:38:11 +0000427 llvm::Constant *getGcAssignStrongCastFn() {
428 // id objc_assign_global(id, id *)
429 std::vector<const llvm::Type*> Args(1, ObjectPtrTy);
430 Args.push_back(ObjectPtrTy->getPointerTo());
Owen Andersona1cf15f2009-07-14 23:10:40 +0000431 llvm::FunctionType *FTy =
Owen Anderson96e0fc72009-07-29 22:16:19 +0000432 llvm::FunctionType::get(ObjectPtrTy, Args, false);
Chris Lattnerbbccd612009-04-22 02:38:11 +0000433 return CGM.CreateRuntimeFunction(FTy, "objc_assign_strongCast");
434 }
Anders Carlssonf57c5b22009-02-16 22:59:18 +0000435
436 /// ExceptionThrowFn - LLVM objc_exception_throw function.
Chris Lattnerbbccd612009-04-22 02:38:11 +0000437 llvm::Constant *getExceptionThrowFn() {
438 // void objc_exception_throw(id)
439 std::vector<const llvm::Type*> Args(1, ObjectPtrTy);
440 llvm::FunctionType *FTy =
Owen Anderson0032b272009-08-13 21:57:51 +0000441 llvm::FunctionType::get(llvm::Type::getVoidTy(VMContext), Args, false);
Chris Lattnerbbccd612009-04-22 02:38:11 +0000442 return CGM.CreateRuntimeFunction(FTy, "objc_exception_throw");
443 }
Daniel Dunbar6bff2512009-08-03 17:06:42 +0000444
Fariborz Jahanian69677ea2010-05-28 17:34:43 +0000445 /// ExceptionRethrowFn - LLVM objc_exception_rethrow function.
446 llvm::Constant *getExceptionRethrowFn() {
447 // void objc_exception_rethrow(void)
448 std::vector<const llvm::Type*> Args;
449 llvm::FunctionType *FTy =
450 llvm::FunctionType::get(llvm::Type::getVoidTy(VMContext), Args, true);
451 return CGM.CreateRuntimeFunction(FTy, "objc_exception_rethrow");
452 }
453
Daniel Dunbar1c566672009-02-24 01:43:46 +0000454 /// SyncEnterFn - LLVM object_sync_enter function.
Chris Lattnerb02e53b2009-04-06 16:53:45 +0000455 llvm::Constant *getSyncEnterFn() {
456 // void objc_sync_enter (id)
457 std::vector<const llvm::Type*> Args(1, ObjectPtrTy);
458 llvm::FunctionType *FTy =
Owen Anderson0032b272009-08-13 21:57:51 +0000459 llvm::FunctionType::get(llvm::Type::getVoidTy(VMContext), Args, false);
Chris Lattnerb02e53b2009-04-06 16:53:45 +0000460 return CGM.CreateRuntimeFunction(FTy, "objc_sync_enter");
461 }
Daniel Dunbar6bff2512009-08-03 17:06:42 +0000462
Daniel Dunbar1c566672009-02-24 01:43:46 +0000463 /// SyncExitFn - LLVM object_sync_exit function.
Chris Lattnerbbccd612009-04-22 02:38:11 +0000464 llvm::Constant *getSyncExitFn() {
465 // void objc_sync_exit (id)
466 std::vector<const llvm::Type*> Args(1, ObjectPtrTy);
467 llvm::FunctionType *FTy =
Owen Anderson0032b272009-08-13 21:57:51 +0000468 llvm::FunctionType::get(llvm::Type::getVoidTy(VMContext), Args, false);
Chris Lattnerbbccd612009-04-22 02:38:11 +0000469 return CGM.CreateRuntimeFunction(FTy, "objc_sync_exit");
470 }
Daniel Dunbar6bff2512009-08-03 17:06:42 +0000471
Fariborz Jahaniand0f8a8d2009-05-11 19:25:47 +0000472 llvm::Constant *getSendFn(bool IsSuper) const {
473 return IsSuper ? getMessageSendSuperFn() : getMessageSendFn();
474 }
Daniel Dunbar6bff2512009-08-03 17:06:42 +0000475
Fariborz Jahaniand0f8a8d2009-05-11 19:25:47 +0000476 llvm::Constant *getSendFn2(bool IsSuper) const {
477 return IsSuper ? getMessageSendSuperFn2() : getMessageSendFn();
478 }
Daniel Dunbar6bff2512009-08-03 17:06:42 +0000479
Fariborz Jahaniand0f8a8d2009-05-11 19:25:47 +0000480 llvm::Constant *getSendStretFn(bool IsSuper) const {
481 return IsSuper ? getMessageSendSuperStretFn() : getMessageSendStretFn();
482 }
Daniel Dunbar6bff2512009-08-03 17:06:42 +0000483
Fariborz Jahaniand0f8a8d2009-05-11 19:25:47 +0000484 llvm::Constant *getSendStretFn2(bool IsSuper) const {
485 return IsSuper ? getMessageSendSuperStretFn2() : getMessageSendStretFn();
486 }
Daniel Dunbar6bff2512009-08-03 17:06:42 +0000487
Fariborz Jahaniand0f8a8d2009-05-11 19:25:47 +0000488 llvm::Constant *getSendFpretFn(bool IsSuper) const {
489 return IsSuper ? getMessageSendSuperFpretFn() : getMessageSendFpretFn();
490 }
Daniel Dunbar6bff2512009-08-03 17:06:42 +0000491
Fariborz Jahaniand0f8a8d2009-05-11 19:25:47 +0000492 llvm::Constant *getSendFpretFn2(bool IsSuper) const {
493 return IsSuper ? getMessageSendSuperFpretFn2() : getMessageSendFpretFn();
494 }
Daniel Dunbar6bff2512009-08-03 17:06:42 +0000495
Fariborz Jahanianee0af742009-01-21 22:04:16 +0000496 ObjCCommonTypesHelper(CodeGen::CodeGenModule &cgm);
497 ~ObjCCommonTypesHelper(){}
498};
Daniel Dunbare8b470d2008-08-23 04:28:29 +0000499
Fariborz Jahanianee0af742009-01-21 22:04:16 +0000500/// ObjCTypesHelper - Helper class that encapsulates lazy
501/// construction of varies types used during ObjC generation.
502class ObjCTypesHelper : public ObjCCommonTypesHelper {
Fariborz Jahanianee0af742009-01-21 22:04:16 +0000503public:
Daniel Dunbar4e2d7d02008-08-12 06:48:42 +0000504 /// SymtabTy - LLVM type for struct objc_symtab.
505 const llvm::StructType *SymtabTy;
Daniel Dunbar27f9d772008-08-21 04:36:09 +0000506 /// SymtabPtrTy - LLVM type for struct objc_symtab *.
507 const llvm::Type *SymtabPtrTy;
Daniel Dunbar4e2d7d02008-08-12 06:48:42 +0000508 /// ModuleTy - LLVM type for struct objc_module.
509 const llvm::StructType *ModuleTy;
Daniel Dunbar259d93d2008-08-12 03:39:23 +0000510
Daniel Dunbar6efc0c52008-08-13 03:21:16 +0000511 /// ProtocolTy - LLVM type for struct objc_protocol.
512 const llvm::StructType *ProtocolTy;
513 /// ProtocolPtrTy - LLVM type for struct objc_protocol *.
514 const llvm::Type *ProtocolPtrTy;
515 /// ProtocolExtensionTy - LLVM type for struct
516 /// objc_protocol_extension.
517 const llvm::StructType *ProtocolExtensionTy;
518 /// ProtocolExtensionTy - LLVM type for struct
519 /// objc_protocol_extension *.
520 const llvm::Type *ProtocolExtensionPtrTy;
521 /// MethodDescriptionTy - LLVM type for struct
522 /// objc_method_description.
523 const llvm::StructType *MethodDescriptionTy;
524 /// MethodDescriptionListTy - LLVM type for struct
525 /// objc_method_description_list.
526 const llvm::StructType *MethodDescriptionListTy;
527 /// MethodDescriptionListPtrTy - LLVM type for struct
528 /// objc_method_description_list *.
529 const llvm::Type *MethodDescriptionListPtrTy;
Daniel Dunbar6efc0c52008-08-13 03:21:16 +0000530 /// ProtocolListTy - LLVM type for struct objc_property_list.
531 const llvm::Type *ProtocolListTy;
532 /// ProtocolListPtrTy - LLVM type for struct objc_property_list*.
533 const llvm::Type *ProtocolListPtrTy;
Daniel Dunbar86e253a2008-08-22 20:34:54 +0000534 /// CategoryTy - LLVM type for struct objc_category.
535 const llvm::StructType *CategoryTy;
Daniel Dunbar27f9d772008-08-21 04:36:09 +0000536 /// ClassTy - LLVM type for struct objc_class.
537 const llvm::StructType *ClassTy;
538 /// ClassPtrTy - LLVM type for struct objc_class *.
539 const llvm::Type *ClassPtrTy;
540 /// ClassExtensionTy - LLVM type for struct objc_class_ext.
541 const llvm::StructType *ClassExtensionTy;
542 /// ClassExtensionPtrTy - LLVM type for struct objc_class_ext *.
543 const llvm::Type *ClassExtensionPtrTy;
Daniel Dunbar27f9d772008-08-21 04:36:09 +0000544 // IvarTy - LLVM type for struct objc_ivar.
545 const llvm::StructType *IvarTy;
546 /// IvarListTy - LLVM type for struct objc_ivar_list.
547 const llvm::Type *IvarListTy;
548 /// IvarListPtrTy - LLVM type for struct objc_ivar_list *.
549 const llvm::Type *IvarListPtrTy;
Daniel Dunbar27f9d772008-08-21 04:36:09 +0000550 /// MethodListTy - LLVM type for struct objc_method_list.
551 const llvm::Type *MethodListTy;
552 /// MethodListPtrTy - LLVM type for struct objc_method_list *.
553 const llvm::Type *MethodListPtrTy;
Daniel Dunbar6bff2512009-08-03 17:06:42 +0000554
Anders Carlsson124526b2008-09-09 10:10:21 +0000555 /// ExceptionDataTy - LLVM type for struct _objc_exception_data.
556 const llvm::Type *ExceptionDataTy;
Daniel Dunbar6bff2512009-08-03 17:06:42 +0000557
Anders Carlsson124526b2008-09-09 10:10:21 +0000558 /// ExceptionTryEnterFn - LLVM objc_exception_try_enter function.
Chris Lattner34b02a12009-04-22 02:26:14 +0000559 llvm::Constant *getExceptionTryEnterFn() {
560 std::vector<const llvm::Type*> Params;
Owen Anderson96e0fc72009-07-29 22:16:19 +0000561 Params.push_back(llvm::PointerType::getUnqual(ExceptionDataTy));
Owen Andersona1cf15f2009-07-14 23:10:40 +0000562 return CGM.CreateRuntimeFunction(
Owen Anderson0032b272009-08-13 21:57:51 +0000563 llvm::FunctionType::get(llvm::Type::getVoidTy(VMContext),
Daniel Dunbar6bff2512009-08-03 17:06:42 +0000564 Params, false),
565 "objc_exception_try_enter");
Chris Lattner34b02a12009-04-22 02:26:14 +0000566 }
Anders Carlsson124526b2008-09-09 10:10:21 +0000567
568 /// ExceptionTryExitFn - LLVM objc_exception_try_exit function.
Chris Lattner34b02a12009-04-22 02:26:14 +0000569 llvm::Constant *getExceptionTryExitFn() {
570 std::vector<const llvm::Type*> Params;
Owen Anderson96e0fc72009-07-29 22:16:19 +0000571 Params.push_back(llvm::PointerType::getUnqual(ExceptionDataTy));
Owen Andersona1cf15f2009-07-14 23:10:40 +0000572 return CGM.CreateRuntimeFunction(
Owen Anderson0032b272009-08-13 21:57:51 +0000573 llvm::FunctionType::get(llvm::Type::getVoidTy(VMContext),
Daniel Dunbar6bff2512009-08-03 17:06:42 +0000574 Params, false),
575 "objc_exception_try_exit");
Chris Lattner34b02a12009-04-22 02:26:14 +0000576 }
Anders Carlsson124526b2008-09-09 10:10:21 +0000577
578 /// ExceptionExtractFn - LLVM objc_exception_extract function.
Chris Lattner34b02a12009-04-22 02:26:14 +0000579 llvm::Constant *getExceptionExtractFn() {
580 std::vector<const llvm::Type*> Params;
Owen Anderson96e0fc72009-07-29 22:16:19 +0000581 Params.push_back(llvm::PointerType::getUnqual(ExceptionDataTy));
582 return CGM.CreateRuntimeFunction(llvm::FunctionType::get(ObjectPtrTy,
Chris Lattner34b02a12009-04-22 02:26:14 +0000583 Params, false),
584 "objc_exception_extract");
Daniel Dunbar6bff2512009-08-03 17:06:42 +0000585
Chris Lattner34b02a12009-04-22 02:26:14 +0000586 }
Daniel Dunbar6bff2512009-08-03 17:06:42 +0000587
Anders Carlsson124526b2008-09-09 10:10:21 +0000588 /// ExceptionMatchFn - LLVM objc_exception_match function.
Chris Lattner34b02a12009-04-22 02:26:14 +0000589 llvm::Constant *getExceptionMatchFn() {
590 std::vector<const llvm::Type*> Params;
591 Params.push_back(ClassPtrTy);
592 Params.push_back(ObjectPtrTy);
Daniel Dunbar6bff2512009-08-03 17:06:42 +0000593 return CGM.CreateRuntimeFunction(
Owen Anderson0032b272009-08-13 21:57:51 +0000594 llvm::FunctionType::get(llvm::Type::getInt32Ty(VMContext),
Daniel Dunbar6bff2512009-08-03 17:06:42 +0000595 Params, false),
596 "objc_exception_match");
597
Chris Lattner34b02a12009-04-22 02:26:14 +0000598 }
Daniel Dunbar6bff2512009-08-03 17:06:42 +0000599
Anders Carlsson124526b2008-09-09 10:10:21 +0000600 /// SetJmpFn - LLVM _setjmp function.
Chris Lattner34b02a12009-04-22 02:26:14 +0000601 llvm::Constant *getSetJmpFn() {
602 std::vector<const llvm::Type*> Params;
Benjamin Kramer3c0ef8c2009-10-13 10:07:13 +0000603 Params.push_back(llvm::Type::getInt32PtrTy(VMContext));
Chris Lattner34b02a12009-04-22 02:26:14 +0000604 return
Owen Anderson0032b272009-08-13 21:57:51 +0000605 CGM.CreateRuntimeFunction(llvm::FunctionType::get(llvm::Type::getInt32Ty(VMContext),
Chris Lattner34b02a12009-04-22 02:26:14 +0000606 Params, false),
607 "_setjmp");
Daniel Dunbar6bff2512009-08-03 17:06:42 +0000608
Chris Lattner34b02a12009-04-22 02:26:14 +0000609 }
Daniel Dunbar6bff2512009-08-03 17:06:42 +0000610
Daniel Dunbarbbce49b2008-08-12 00:12:39 +0000611public:
612 ObjCTypesHelper(CodeGen::CodeGenModule &cgm);
Fariborz Jahanianee0af742009-01-21 22:04:16 +0000613 ~ObjCTypesHelper() {}
Daniel Dunbarbbce49b2008-08-12 00:12:39 +0000614};
615
Fariborz Jahanian30bc5712009-01-22 23:02:58 +0000616/// ObjCNonFragileABITypesHelper - will have all types needed by objective-c's
Fariborz Jahanianee0af742009-01-21 22:04:16 +0000617/// modern abi
Fariborz Jahanian30bc5712009-01-22 23:02:58 +0000618class ObjCNonFragileABITypesHelper : public ObjCCommonTypesHelper {
Fariborz Jahanian83a8a752009-02-04 20:42:28 +0000619public:
Daniel Dunbar6bff2512009-08-03 17:06:42 +0000620
Fariborz Jahaniand55b6fc2009-01-23 01:46:23 +0000621 // MethodListnfABITy - LLVM for struct _method_list_t
622 const llvm::StructType *MethodListnfABITy;
Daniel Dunbar6bff2512009-08-03 17:06:42 +0000623
Fariborz Jahaniand55b6fc2009-01-23 01:46:23 +0000624 // MethodListnfABIPtrTy - LLVM for struct _method_list_t*
625 const llvm::Type *MethodListnfABIPtrTy;
Daniel Dunbar6bff2512009-08-03 17:06:42 +0000626
Fariborz Jahaniand55b6fc2009-01-23 01:46:23 +0000627 // ProtocolnfABITy = LLVM for struct _protocol_t
628 const llvm::StructType *ProtocolnfABITy;
Daniel Dunbar6bff2512009-08-03 17:06:42 +0000629
Daniel Dunbar948e2582009-02-15 07:36:20 +0000630 // ProtocolnfABIPtrTy = LLVM for struct _protocol_t*
631 const llvm::Type *ProtocolnfABIPtrTy;
632
Fariborz Jahaniand55b6fc2009-01-23 01:46:23 +0000633 // ProtocolListnfABITy - LLVM for struct _objc_protocol_list
634 const llvm::StructType *ProtocolListnfABITy;
Daniel Dunbar6bff2512009-08-03 17:06:42 +0000635
Fariborz Jahaniand55b6fc2009-01-23 01:46:23 +0000636 // ProtocolListnfABIPtrTy - LLVM for struct _objc_protocol_list*
637 const llvm::Type *ProtocolListnfABIPtrTy;
Daniel Dunbar6bff2512009-08-03 17:06:42 +0000638
Fariborz Jahaniand55b6fc2009-01-23 01:46:23 +0000639 // ClassnfABITy - LLVM for struct _class_t
640 const llvm::StructType *ClassnfABITy;
Daniel Dunbar6bff2512009-08-03 17:06:42 +0000641
Fariborz Jahanianaa23b572009-01-23 23:53:38 +0000642 // ClassnfABIPtrTy - LLVM for struct _class_t*
643 const llvm::Type *ClassnfABIPtrTy;
Daniel Dunbar6bff2512009-08-03 17:06:42 +0000644
Fariborz Jahaniand55b6fc2009-01-23 01:46:23 +0000645 // IvarnfABITy - LLVM for struct _ivar_t
646 const llvm::StructType *IvarnfABITy;
Daniel Dunbar6bff2512009-08-03 17:06:42 +0000647
Fariborz Jahaniand55b6fc2009-01-23 01:46:23 +0000648 // IvarListnfABITy - LLVM for struct _ivar_list_t
649 const llvm::StructType *IvarListnfABITy;
Daniel Dunbar6bff2512009-08-03 17:06:42 +0000650
Fariborz Jahaniand55b6fc2009-01-23 01:46:23 +0000651 // IvarListnfABIPtrTy = LLVM for struct _ivar_list_t*
652 const llvm::Type *IvarListnfABIPtrTy;
Daniel Dunbar6bff2512009-08-03 17:06:42 +0000653
Fariborz Jahaniand55b6fc2009-01-23 01:46:23 +0000654 // ClassRonfABITy - LLVM for struct _class_ro_t
655 const llvm::StructType *ClassRonfABITy;
Daniel Dunbar6bff2512009-08-03 17:06:42 +0000656
Fariborz Jahaniand55b6fc2009-01-23 01:46:23 +0000657 // ImpnfABITy - LLVM for id (*)(id, SEL, ...)
658 const llvm::Type *ImpnfABITy;
Daniel Dunbar6bff2512009-08-03 17:06:42 +0000659
Fariborz Jahaniand55b6fc2009-01-23 01:46:23 +0000660 // CategorynfABITy - LLVM for struct _category_t
661 const llvm::StructType *CategorynfABITy;
Daniel Dunbar6bff2512009-08-03 17:06:42 +0000662
Fariborz Jahanian2e4672b2009-02-03 23:49:23 +0000663 // New types for nonfragile abi messaging.
Daniel Dunbar6bff2512009-08-03 17:06:42 +0000664
Fariborz Jahanian2e4672b2009-02-03 23:49:23 +0000665 // MessageRefTy - LLVM for:
666 // struct _message_ref_t {
667 // IMP messenger;
668 // SEL name;
669 // };
670 const llvm::StructType *MessageRefTy;
Fariborz Jahanian83a8a752009-02-04 20:42:28 +0000671 // MessageRefCTy - clang type for struct _message_ref_t
672 QualType MessageRefCTy;
Daniel Dunbar6bff2512009-08-03 17:06:42 +0000673
Fariborz Jahanian2e4672b2009-02-03 23:49:23 +0000674 // MessageRefPtrTy - LLVM for struct _message_ref_t*
675 const llvm::Type *MessageRefPtrTy;
Fariborz Jahanian83a8a752009-02-04 20:42:28 +0000676 // MessageRefCPtrTy - clang type for struct _message_ref_t*
677 QualType MessageRefCPtrTy;
Daniel Dunbar6bff2512009-08-03 17:06:42 +0000678
Fariborz Jahanianef163782009-02-05 01:13:09 +0000679 // MessengerTy - Type of the messenger (shown as IMP above)
680 const llvm::FunctionType *MessengerTy;
Daniel Dunbar6bff2512009-08-03 17:06:42 +0000681
Fariborz Jahanian2e4672b2009-02-03 23:49:23 +0000682 // SuperMessageRefTy - LLVM for:
683 // struct _super_message_ref_t {
684 // SUPER_IMP messenger;
685 // SEL name;
686 // };
687 const llvm::StructType *SuperMessageRefTy;
Daniel Dunbar6bff2512009-08-03 17:06:42 +0000688
Fariborz Jahanian2e4672b2009-02-03 23:49:23 +0000689 // SuperMessageRefPtrTy - LLVM for struct _super_message_ref_t*
690 const llvm::Type *SuperMessageRefPtrTy;
Daniel Dunbar8ecbaf22009-02-24 07:47:38 +0000691
Chris Lattner1c02f862009-04-22 02:53:24 +0000692 llvm::Constant *getMessageSendFixupFn() {
693 // id objc_msgSend_fixup(id, struct message_ref_t*, ...)
694 std::vector<const llvm::Type*> Params;
695 Params.push_back(ObjectPtrTy);
696 Params.push_back(MessageRefPtrTy);
Owen Anderson96e0fc72009-07-29 22:16:19 +0000697 return CGM.CreateRuntimeFunction(llvm::FunctionType::get(ObjectPtrTy,
Chris Lattner1c02f862009-04-22 02:53:24 +0000698 Params, true),
699 "objc_msgSend_fixup");
700 }
Daniel Dunbar6bff2512009-08-03 17:06:42 +0000701
Chris Lattner1c02f862009-04-22 02:53:24 +0000702 llvm::Constant *getMessageSendFpretFixupFn() {
703 // id objc_msgSend_fpret_fixup(id, struct message_ref_t*, ...)
704 std::vector<const llvm::Type*> Params;
705 Params.push_back(ObjectPtrTy);
706 Params.push_back(MessageRefPtrTy);
Owen Anderson96e0fc72009-07-29 22:16:19 +0000707 return CGM.CreateRuntimeFunction(llvm::FunctionType::get(ObjectPtrTy,
Chris Lattner1c02f862009-04-22 02:53:24 +0000708 Params, true),
709 "objc_msgSend_fpret_fixup");
710 }
Daniel Dunbar6bff2512009-08-03 17:06:42 +0000711
Chris Lattner1c02f862009-04-22 02:53:24 +0000712 llvm::Constant *getMessageSendStretFixupFn() {
713 // id objc_msgSend_stret_fixup(id, struct message_ref_t*, ...)
714 std::vector<const llvm::Type*> Params;
715 Params.push_back(ObjectPtrTy);
716 Params.push_back(MessageRefPtrTy);
Owen Anderson96e0fc72009-07-29 22:16:19 +0000717 return CGM.CreateRuntimeFunction(llvm::FunctionType::get(ObjectPtrTy,
Chris Lattner1c02f862009-04-22 02:53:24 +0000718 Params, true),
719 "objc_msgSend_stret_fixup");
720 }
Daniel Dunbar6bff2512009-08-03 17:06:42 +0000721
Chris Lattner1c02f862009-04-22 02:53:24 +0000722 llvm::Constant *getMessageSendIdFixupFn() {
723 // id objc_msgSendId_fixup(id, struct message_ref_t*, ...)
724 std::vector<const llvm::Type*> Params;
725 Params.push_back(ObjectPtrTy);
726 Params.push_back(MessageRefPtrTy);
Owen Anderson96e0fc72009-07-29 22:16:19 +0000727 return CGM.CreateRuntimeFunction(llvm::FunctionType::get(ObjectPtrTy,
Chris Lattner1c02f862009-04-22 02:53:24 +0000728 Params, true),
729 "objc_msgSendId_fixup");
730 }
Daniel Dunbar6bff2512009-08-03 17:06:42 +0000731
Chris Lattner1c02f862009-04-22 02:53:24 +0000732 llvm::Constant *getMessageSendIdStretFixupFn() {
733 // id objc_msgSendId_stret_fixup(id, struct message_ref_t*, ...)
734 std::vector<const llvm::Type*> Params;
735 Params.push_back(ObjectPtrTy);
736 Params.push_back(MessageRefPtrTy);
Owen Anderson96e0fc72009-07-29 22:16:19 +0000737 return CGM.CreateRuntimeFunction(llvm::FunctionType::get(ObjectPtrTy,
Chris Lattner1c02f862009-04-22 02:53:24 +0000738 Params, true),
739 "objc_msgSendId_stret_fixup");
740 }
741 llvm::Constant *getMessageSendSuper2FixupFn() {
Daniel Dunbar6bff2512009-08-03 17:06:42 +0000742 // id objc_msgSendSuper2_fixup (struct objc_super *,
Chris Lattner1c02f862009-04-22 02:53:24 +0000743 // struct _super_message_ref_t*, ...)
744 std::vector<const llvm::Type*> Params;
745 Params.push_back(SuperPtrTy);
746 Params.push_back(SuperMessageRefPtrTy);
Owen Anderson96e0fc72009-07-29 22:16:19 +0000747 return CGM.CreateRuntimeFunction(llvm::FunctionType::get(ObjectPtrTy,
Chris Lattner1c02f862009-04-22 02:53:24 +0000748 Params, true),
749 "objc_msgSendSuper2_fixup");
750 }
Daniel Dunbar6bff2512009-08-03 17:06:42 +0000751
Chris Lattner1c02f862009-04-22 02:53:24 +0000752 llvm::Constant *getMessageSendSuper2StretFixupFn() {
Daniel Dunbar6bff2512009-08-03 17:06:42 +0000753 // id objc_msgSendSuper2_stret_fixup(struct objc_super *,
Chris Lattner1c02f862009-04-22 02:53:24 +0000754 // struct _super_message_ref_t*, ...)
755 std::vector<const llvm::Type*> Params;
756 Params.push_back(SuperPtrTy);
757 Params.push_back(SuperMessageRefPtrTy);
Owen Anderson96e0fc72009-07-29 22:16:19 +0000758 return CGM.CreateRuntimeFunction(llvm::FunctionType::get(ObjectPtrTy,
Chris Lattner1c02f862009-04-22 02:53:24 +0000759 Params, true),
760 "objc_msgSendSuper2_stret_fixup");
761 }
Daniel Dunbar6bff2512009-08-03 17:06:42 +0000762
763
764
Daniel Dunbar8ecbaf22009-02-24 07:47:38 +0000765 /// EHPersonalityPtr - LLVM value for an i8* to the Objective-C
766 /// exception personality function.
Chris Lattnerb02e53b2009-04-06 16:53:45 +0000767 llvm::Value *getEHPersonalityPtr() {
Daniel Dunbar6bff2512009-08-03 17:06:42 +0000768 llvm::Constant *Personality =
Owen Anderson0032b272009-08-13 21:57:51 +0000769 CGM.CreateRuntimeFunction(llvm::FunctionType::get(llvm::Type::getInt32Ty(VMContext),
Chris Lattnerb02e53b2009-04-06 16:53:45 +0000770 true),
Daniel Dunbar6bff2512009-08-03 17:06:42 +0000771 "__objc_personality_v0");
Owen Anderson3c4972d2009-07-29 18:54:39 +0000772 return llvm::ConstantExpr::getBitCast(Personality, Int8PtrTy);
Chris Lattnerb02e53b2009-04-06 16:53:45 +0000773 }
Daniel Dunbar8ecbaf22009-02-24 07:47:38 +0000774
Chris Lattner8a569112009-04-22 02:15:23 +0000775 llvm::Constant *getUnwindResumeOrRethrowFn() {
776 std::vector<const llvm::Type*> Params;
777 Params.push_back(Int8PtrTy);
Owen Andersona1cf15f2009-07-14 23:10:40 +0000778 return CGM.CreateRuntimeFunction(
Owen Anderson0032b272009-08-13 21:57:51 +0000779 llvm::FunctionType::get(llvm::Type::getVoidTy(VMContext),
Daniel Dunbar6bff2512009-08-03 17:06:42 +0000780 Params, false),
Daniel Dunbarb2987d12010-02-10 18:49:11 +0000781 (CGM.getLangOptions().SjLjExceptions ? "_Unwind_SjLj_Resume" :
782 "_Unwind_Resume_or_Rethrow"));
Chris Lattner8a569112009-04-22 02:15:23 +0000783 }
Daniel Dunbar6bff2512009-08-03 17:06:42 +0000784
Chris Lattner8a569112009-04-22 02:15:23 +0000785 llvm::Constant *getObjCEndCatchFn() {
Owen Anderson0032b272009-08-13 21:57:51 +0000786 return CGM.CreateRuntimeFunction(llvm::FunctionType::get(llvm::Type::getVoidTy(VMContext),
Chris Lattnerb59761b2009-07-01 04:13:52 +0000787 false),
Chris Lattner8a569112009-04-22 02:15:23 +0000788 "objc_end_catch");
Daniel Dunbar6bff2512009-08-03 17:06:42 +0000789
Chris Lattner8a569112009-04-22 02:15:23 +0000790 }
Daniel Dunbar6bff2512009-08-03 17:06:42 +0000791
Chris Lattner8a569112009-04-22 02:15:23 +0000792 llvm::Constant *getObjCBeginCatchFn() {
793 std::vector<const llvm::Type*> Params;
794 Params.push_back(Int8PtrTy);
Owen Anderson96e0fc72009-07-29 22:16:19 +0000795 return CGM.CreateRuntimeFunction(llvm::FunctionType::get(Int8PtrTy,
Chris Lattner8a569112009-04-22 02:15:23 +0000796 Params, false),
797 "objc_begin_catch");
798 }
Daniel Dunbare588b992009-03-01 04:46:24 +0000799
800 const llvm::StructType *EHTypeTy;
801 const llvm::Type *EHTypePtrTy;
Daniel Dunbar8ecbaf22009-02-24 07:47:38 +0000802
Fariborz Jahanian30bc5712009-01-22 23:02:58 +0000803 ObjCNonFragileABITypesHelper(CodeGen::CodeGenModule &cgm);
804 ~ObjCNonFragileABITypesHelper(){}
Fariborz Jahanianee0af742009-01-21 22:04:16 +0000805};
Daniel Dunbar6bff2512009-08-03 17:06:42 +0000806
Fariborz Jahanianee0af742009-01-21 22:04:16 +0000807class CGObjCCommonMac : public CodeGen::CGObjCRuntime {
Fariborz Jahanian9397e1d2009-03-11 20:59:05 +0000808public:
809 // FIXME - accessibility
Fariborz Jahaniana5a10c32009-03-10 16:22:08 +0000810 class GC_IVAR {
Fariborz Jahanian820e0202009-03-11 00:07:04 +0000811 public:
Daniel Dunbar8b2926c2009-05-03 13:44:42 +0000812 unsigned ivar_bytepos;
813 unsigned ivar_size;
814 GC_IVAR(unsigned bytepos = 0, unsigned size = 0)
Daniel Dunbar6bff2512009-08-03 17:06:42 +0000815 : ivar_bytepos(bytepos), ivar_size(size) {}
Daniel Dunbar0941b492009-04-23 01:29:05 +0000816
817 // Allow sorting based on byte pos.
818 bool operator<(const GC_IVAR &b) const {
819 return ivar_bytepos < b.ivar_bytepos;
820 }
Fariborz Jahaniana5a10c32009-03-10 16:22:08 +0000821 };
Daniel Dunbar6bff2512009-08-03 17:06:42 +0000822
Fariborz Jahanian9397e1d2009-03-11 20:59:05 +0000823 class SKIP_SCAN {
Daniel Dunbar8b2926c2009-05-03 13:44:42 +0000824 public:
825 unsigned skip;
826 unsigned scan;
Daniel Dunbar6bff2512009-08-03 17:06:42 +0000827 SKIP_SCAN(unsigned _skip = 0, unsigned _scan = 0)
Daniel Dunbar8b2926c2009-05-03 13:44:42 +0000828 : skip(_skip), scan(_scan) {}
Fariborz Jahanian9397e1d2009-03-11 20:59:05 +0000829 };
Daniel Dunbar6bff2512009-08-03 17:06:42 +0000830
Fariborz Jahanianee0af742009-01-21 22:04:16 +0000831protected:
832 CodeGen::CodeGenModule &CGM;
Owen Anderson69243822009-07-13 04:10:07 +0000833 llvm::LLVMContext &VMContext;
Fariborz Jahanianee0af742009-01-21 22:04:16 +0000834 // FIXME! May not be needing this after all.
Daniel Dunbarbbce49b2008-08-12 00:12:39 +0000835 unsigned ObjCABI;
Daniel Dunbar6bff2512009-08-03 17:06:42 +0000836
Fariborz Jahanian9397e1d2009-03-11 20:59:05 +0000837 // gc ivar layout bitmap calculation helper caches.
838 llvm::SmallVector<GC_IVAR, 16> SkipIvars;
839 llvm::SmallVector<GC_IVAR, 16> IvarsInfo;
Daniel Dunbar6bff2512009-08-03 17:06:42 +0000840
Daniel Dunbar242d4dc2008-08-25 06:02:07 +0000841 /// LazySymbols - Symbols to generate a lazy reference for. See
842 /// DefinedSymbols and FinishModule().
Daniel Dunbar33063492009-09-07 00:20:42 +0000843 llvm::SetVector<IdentifierInfo*> LazySymbols;
Daniel Dunbar6bff2512009-08-03 17:06:42 +0000844
Daniel Dunbar242d4dc2008-08-25 06:02:07 +0000845 /// DefinedSymbols - External symbols which are defined by this
846 /// module. The symbols in this list and LazySymbols are used to add
847 /// special linker symbols which ensure that Objective-C modules are
848 /// linked properly.
Daniel Dunbar33063492009-09-07 00:20:42 +0000849 llvm::SetVector<IdentifierInfo*> DefinedSymbols;
Daniel Dunbar6bff2512009-08-03 17:06:42 +0000850
Daniel Dunbar4e2d7d02008-08-12 06:48:42 +0000851 /// ClassNames - uniqued class names.
Daniel Dunbar6efc0c52008-08-13 03:21:16 +0000852 llvm::DenseMap<IdentifierInfo*, llvm::GlobalVariable*> ClassNames;
Daniel Dunbar6bff2512009-08-03 17:06:42 +0000853
Daniel Dunbar259d93d2008-08-12 03:39:23 +0000854 /// MethodVarNames - uniqued method variable names.
855 llvm::DenseMap<Selector, llvm::GlobalVariable*> MethodVarNames;
Daniel Dunbar6bff2512009-08-03 17:06:42 +0000856
Fariborz Jahanianb9c5b3d2010-06-21 22:05:18 +0000857 /// DefinedCategoryNames - list of category names in form Class_Category.
858 llvm::SetVector<std::string> DefinedCategoryNames;
859
Daniel Dunbar6efc0c52008-08-13 03:21:16 +0000860 /// MethodVarTypes - uniqued method type signatures. We have to use
861 /// a StringMap here because have no other unique reference.
862 llvm::StringMap<llvm::GlobalVariable*> MethodVarTypes;
Daniel Dunbar6bff2512009-08-03 17:06:42 +0000863
Daniel Dunbarc45ef602008-08-26 21:51:14 +0000864 /// MethodDefinitions - map of methods which have been defined in
865 /// this translation unit.
866 llvm::DenseMap<const ObjCMethodDecl*, llvm::Function*> MethodDefinitions;
Daniel Dunbar6bff2512009-08-03 17:06:42 +0000867
Daniel Dunbarc8ef5512008-08-23 00:19:03 +0000868 /// PropertyNames - uniqued method variable names.
869 llvm::DenseMap<IdentifierInfo*, llvm::GlobalVariable*> PropertyNames;
Daniel Dunbar6bff2512009-08-03 17:06:42 +0000870
Daniel Dunbar27f9d772008-08-21 04:36:09 +0000871 /// ClassReferences - uniqued class references.
872 llvm::DenseMap<IdentifierInfo*, llvm::GlobalVariable*> ClassReferences;
Daniel Dunbar6bff2512009-08-03 17:06:42 +0000873
Daniel Dunbar259d93d2008-08-12 03:39:23 +0000874 /// SelectorReferences - uniqued selector references.
875 llvm::DenseMap<Selector, llvm::GlobalVariable*> SelectorReferences;
Daniel Dunbar6bff2512009-08-03 17:06:42 +0000876
Daniel Dunbar6efc0c52008-08-13 03:21:16 +0000877 /// Protocols - Protocols for which an objc_protocol structure has
878 /// been emitted. Forward declarations are handled by creating an
879 /// empty structure whose initializer is filled in when/if defined.
880 llvm::DenseMap<IdentifierInfo*, llvm::GlobalVariable*> Protocols;
Daniel Dunbar6bff2512009-08-03 17:06:42 +0000881
Daniel Dunbar0c0e7a62008-10-29 22:36:39 +0000882 /// DefinedProtocols - Protocols which have actually been
883 /// defined. We should not need this, see FIXME in GenerateProtocol.
884 llvm::DenseSet<IdentifierInfo*> DefinedProtocols;
Daniel Dunbar6bff2512009-08-03 17:06:42 +0000885
Daniel Dunbar27f9d772008-08-21 04:36:09 +0000886 /// DefinedClasses - List of defined classes.
887 std::vector<llvm::GlobalValue*> DefinedClasses;
Daniel Dunbar74d4b122009-05-15 22:33:15 +0000888
889 /// DefinedNonLazyClasses - List of defined "non-lazy" classes.
890 std::vector<llvm::GlobalValue*> DefinedNonLazyClasses;
Daniel Dunbar6bff2512009-08-03 17:06:42 +0000891
Daniel Dunbar27f9d772008-08-21 04:36:09 +0000892 /// DefinedCategories - List of defined categories.
893 std::vector<llvm::GlobalValue*> DefinedCategories;
Daniel Dunbar6bff2512009-08-03 17:06:42 +0000894
Daniel Dunbar74d4b122009-05-15 22:33:15 +0000895 /// DefinedNonLazyCategories - List of defined "non-lazy" categories.
896 std::vector<llvm::GlobalValue*> DefinedNonLazyCategories;
Daniel Dunbar6bff2512009-08-03 17:06:42 +0000897
Fariborz Jahanian56210f72009-01-21 23:34:32 +0000898 /// GetNameForMethod - Return a name for the given method.
899 /// \param[out] NameOut - The return value.
900 void GetNameForMethod(const ObjCMethodDecl *OMD,
901 const ObjCContainerDecl *CD,
Daniel Dunbarc575ce72009-10-19 01:21:19 +0000902 llvm::SmallVectorImpl<char> &NameOut);
Daniel Dunbar6bff2512009-08-03 17:06:42 +0000903
Fariborz Jahanian56210f72009-01-21 23:34:32 +0000904 /// GetMethodVarName - Return a unique constant for the given
905 /// selector's name. The return value has type char *.
906 llvm::Constant *GetMethodVarName(Selector Sel);
907 llvm::Constant *GetMethodVarName(IdentifierInfo *Ident);
908 llvm::Constant *GetMethodVarName(const std::string &Name);
Daniel Dunbar6bff2512009-08-03 17:06:42 +0000909
Fariborz Jahanian56210f72009-01-21 23:34:32 +0000910 /// GetMethodVarType - Return a unique constant for the given
911 /// selector's name. The return value has type char *.
Daniel Dunbar6bff2512009-08-03 17:06:42 +0000912
Fariborz Jahanian56210f72009-01-21 23:34:32 +0000913 // FIXME: This is a horrible name.
914 llvm::Constant *GetMethodVarType(const ObjCMethodDecl *D);
Daniel Dunbar3e5f0d82009-04-20 06:54:31 +0000915 llvm::Constant *GetMethodVarType(const FieldDecl *D);
Daniel Dunbar6bff2512009-08-03 17:06:42 +0000916
Fariborz Jahanian56210f72009-01-21 23:34:32 +0000917 /// GetPropertyName - Return a unique constant for the given
918 /// name. The return value has type char *.
919 llvm::Constant *GetPropertyName(IdentifierInfo *Ident);
Daniel Dunbar6bff2512009-08-03 17:06:42 +0000920
Fariborz Jahanian56210f72009-01-21 23:34:32 +0000921 // FIXME: This can be dropped once string functions are unified.
922 llvm::Constant *GetPropertyTypeString(const ObjCPropertyDecl *PD,
923 const Decl *Container);
Daniel Dunbar6bff2512009-08-03 17:06:42 +0000924
Fariborz Jahanian058a1b72009-01-24 20:21:50 +0000925 /// GetClassName - Return a unique constant for the given selector's
926 /// name. The return value has type char *.
927 llvm::Constant *GetClassName(IdentifierInfo *Ident);
Daniel Dunbar6bff2512009-08-03 17:06:42 +0000928
Fariborz Jahaniand61a50a2009-03-05 22:39:55 +0000929 /// BuildIvarLayout - Builds ivar layout bitmap for the class
930 /// implementation for the __strong or __weak case.
931 ///
Fariborz Jahanianc8ce9c82009-03-12 22:50:49 +0000932 llvm::Constant *BuildIvarLayout(const ObjCImplementationDecl *OI,
933 bool ForStrongLayout);
Daniel Dunbar6bff2512009-08-03 17:06:42 +0000934
Daniel Dunbard58edcb2009-05-03 14:10:34 +0000935 void BuildAggrIvarRecordLayout(const RecordType *RT,
Daniel Dunbar6bff2512009-08-03 17:06:42 +0000936 unsigned int BytePos, bool ForStrongLayout,
937 bool &HasUnion);
Daniel Dunbar5a5a8032009-05-03 21:05:10 +0000938 void BuildAggrIvarLayout(const ObjCImplementationDecl *OI,
Fariborz Jahanianc8ce9c82009-03-12 22:50:49 +0000939 const llvm::StructLayout *Layout,
Fariborz Jahaniana5a10c32009-03-10 16:22:08 +0000940 const RecordDecl *RD,
Chris Lattnerf1690852009-03-31 08:48:01 +0000941 const llvm::SmallVectorImpl<FieldDecl*> &RecFields,
Fariborz Jahaniand61a50a2009-03-05 22:39:55 +0000942 unsigned int BytePos, bool ForStrongLayout,
Fariborz Jahanian81adc052009-04-24 16:17:09 +0000943 bool &HasUnion);
Fariborz Jahanianc8ce9c82009-03-12 22:50:49 +0000944
Fariborz Jahaniand80d81b2009-03-05 19:17:31 +0000945 /// GetIvarLayoutName - Returns a unique constant for the given
946 /// ivar layout bitmap.
947 llvm::Constant *GetIvarLayoutName(IdentifierInfo *Ident,
948 const ObjCCommonTypesHelper &ObjCTypes);
Daniel Dunbar6bff2512009-08-03 17:06:42 +0000949
Fariborz Jahanian5de14dc2009-01-28 22:18:42 +0000950 /// EmitPropertyList - Emit the given property list. The return
951 /// value has type PropertyListPtrTy.
Daniel Dunbar9c29bf52009-10-18 20:48:59 +0000952 llvm::Constant *EmitPropertyList(llvm::Twine Name,
Daniel Dunbar6bff2512009-08-03 17:06:42 +0000953 const Decl *Container,
Fariborz Jahanian5de14dc2009-01-28 22:18:42 +0000954 const ObjCContainerDecl *OCD,
955 const ObjCCommonTypesHelper &ObjCTypes);
Daniel Dunbar6bff2512009-08-03 17:06:42 +0000956
Fariborz Jahanian191dcd72009-12-12 21:26:21 +0000957 /// PushProtocolProperties - Push protocol's property on the input stack.
958 void PushProtocolProperties(llvm::SmallPtrSet<const IdentifierInfo*, 16> &PropertySet,
959 std::vector<llvm::Constant*> &Properties,
960 const Decl *Container,
961 const ObjCProtocolDecl *PROTO,
962 const ObjCCommonTypesHelper &ObjCTypes);
963
Fariborz Jahanianda320092009-01-29 19:24:30 +0000964 /// GetProtocolRef - Return a reference to the internal protocol
965 /// description, creating an empty one if it has not been
966 /// defined. The return value has type ProtocolPtrTy.
967 llvm::Constant *GetProtocolRef(const ObjCProtocolDecl *PD);
Fariborz Jahanianb21f07e2009-03-08 20:18:37 +0000968
Daniel Dunbarfd65d372009-03-09 20:09:19 +0000969 /// CreateMetadataVar - Create a global variable with internal
970 /// linkage for use by the Objective-C runtime.
971 ///
972 /// This is a convenience wrapper which not only creates the
973 /// variable, but also sets the section and alignment and adds the
Chris Lattnerad64e022009-07-17 23:57:13 +0000974 /// global to the "llvm.used" list.
Daniel Dunbar35bd7632009-03-09 20:50:13 +0000975 ///
976 /// \param Name - The variable name.
977 /// \param Init - The variable initializer; this is also used to
978 /// define the type of the variable.
979 /// \param Section - The section the variable should go into, or 0.
980 /// \param Align - The alignment for the variable, or 0.
981 /// \param AddToUsed - Whether the variable should be added to
Daniel Dunbarc1583062009-04-14 17:42:51 +0000982 /// "llvm.used".
Daniel Dunbar9c29bf52009-10-18 20:48:59 +0000983 llvm::GlobalVariable *CreateMetadataVar(llvm::Twine Name,
Daniel Dunbarfd65d372009-03-09 20:09:19 +0000984 llvm::Constant *Init,
985 const char *Section,
Daniel Dunbar35bd7632009-03-09 20:50:13 +0000986 unsigned Align,
987 bool AddToUsed);
Daniel Dunbarfd65d372009-03-09 20:09:19 +0000988
Fariborz Jahaniand0f8a8d2009-05-11 19:25:47 +0000989 CodeGen::RValue EmitLegacyMessageSend(CodeGen::CodeGenFunction &CGF,
John McCallef072fd2010-05-22 01:48:05 +0000990 ReturnValueSlot Return,
Fariborz Jahaniand0f8a8d2009-05-11 19:25:47 +0000991 QualType ResultType,
992 llvm::Value *Sel,
993 llvm::Value *Arg0,
994 QualType Arg0Ty,
995 bool IsSuper,
996 const CallArgList &CallArgs,
Daniel Dunbard6c93d72009-09-17 04:01:22 +0000997 const ObjCMethodDecl *OMD,
Fariborz Jahaniand0f8a8d2009-05-11 19:25:47 +0000998 const ObjCCommonTypesHelper &ObjCTypes);
Daniel Dunbar3e5f0d82009-04-20 06:54:31 +0000999
Daniel Dunbarfce176b2010-04-25 20:39:01 +00001000 /// EmitImageInfo - Emit the image info marker used to encode some module
1001 /// level information.
1002 void EmitImageInfo();
1003
Fariborz Jahanianee0af742009-01-21 22:04:16 +00001004public:
Owen Anderson69243822009-07-13 04:10:07 +00001005 CGObjCCommonMac(CodeGen::CodeGenModule &cgm) :
Mike Stump1eb44332009-09-09 15:08:12 +00001006 CGM(cgm), VMContext(cgm.getLLVMContext()) { }
Daniel Dunbar6bff2512009-08-03 17:06:42 +00001007
David Chisnall0d13f6f2010-01-23 02:40:42 +00001008 virtual llvm::Constant *GenerateConstantString(const StringLiteral *SL);
Daniel Dunbar6bff2512009-08-03 17:06:42 +00001009
Fariborz Jahanian493dab72009-01-26 21:38:32 +00001010 virtual llvm::Function *GenerateMethod(const ObjCMethodDecl *OMD,
1011 const ObjCContainerDecl *CD=0);
Daniel Dunbar6bff2512009-08-03 17:06:42 +00001012
Fariborz Jahanianda320092009-01-29 19:24:30 +00001013 virtual void GenerateProtocol(const ObjCProtocolDecl *PD);
Daniel Dunbar6bff2512009-08-03 17:06:42 +00001014
Fariborz Jahanianda320092009-01-29 19:24:30 +00001015 /// GetOrEmitProtocol - Get the protocol object for the given
1016 /// declaration, emitting it if necessary. The return value has type
1017 /// ProtocolPtrTy.
1018 virtual llvm::Constant *GetOrEmitProtocol(const ObjCProtocolDecl *PD)=0;
Daniel Dunbar6bff2512009-08-03 17:06:42 +00001019
Fariborz Jahanianda320092009-01-29 19:24:30 +00001020 /// GetOrEmitProtocolRef - Get a forward reference to the protocol
1021 /// object for the given declaration, emitting it if needed. These
1022 /// forward references will be filled in with empty bodies if no
1023 /// definition is seen. The return value has type ProtocolPtrTy.
1024 virtual llvm::Constant *GetOrEmitProtocolRef(const ObjCProtocolDecl *PD)=0;
Fariborz Jahanianee0af742009-01-21 22:04:16 +00001025};
Daniel Dunbar6bff2512009-08-03 17:06:42 +00001026
Fariborz Jahanianee0af742009-01-21 22:04:16 +00001027class CGObjCMac : public CGObjCCommonMac {
1028private:
1029 ObjCTypesHelper ObjCTypes;
Daniel Dunbarf77ac862008-08-11 21:35:06 +00001030
Daniel Dunbar4e2d7d02008-08-12 06:48:42 +00001031 /// EmitModuleInfo - Another marker encoding module level
Daniel Dunbar6bff2512009-08-03 17:06:42 +00001032 /// information.
Daniel Dunbar4e2d7d02008-08-12 06:48:42 +00001033 void EmitModuleInfo();
1034
Daniel Dunbar27f9d772008-08-21 04:36:09 +00001035 /// EmitModuleSymols - Emit module symbols, the list of defined
1036 /// classes and categories. The result has type SymtabPtrTy.
Daniel Dunbar4e2d7d02008-08-12 06:48:42 +00001037 llvm::Constant *EmitModuleSymbols();
1038
Daniel Dunbarf77ac862008-08-11 21:35:06 +00001039 /// FinishModule - Write out global data structures at the end of
1040 /// processing a translation unit.
1041 void FinishModule();
Daniel Dunbar6efc0c52008-08-13 03:21:16 +00001042
Daniel Dunbar27f9d772008-08-21 04:36:09 +00001043 /// EmitClassExtension - Generate the class extension structure used
1044 /// to store the weak ivar layout and properties. The return value
1045 /// has type ClassExtensionPtrTy.
1046 llvm::Constant *EmitClassExtension(const ObjCImplementationDecl *ID);
1047
1048 /// EmitClassRef - Return a Value*, of type ObjCTypes.ClassPtrTy,
1049 /// for the given class.
Daniel Dunbar6bff2512009-08-03 17:06:42 +00001050 llvm::Value *EmitClassRef(CGBuilderTy &Builder,
Daniel Dunbar27f9d772008-08-21 04:36:09 +00001051 const ObjCInterfaceDecl *ID);
Fariborz Jahanianb0069ee2009-11-12 20:14:24 +00001052
1053 /// EmitSuperClassRef - Emits reference to class's main metadata class.
1054 llvm::Value *EmitSuperClassRef(const ObjCInterfaceDecl *ID);
Daniel Dunbar27f9d772008-08-21 04:36:09 +00001055
Daniel Dunbar14c80b72008-08-23 09:25:55 +00001056 CodeGen::RValue EmitMessageSend(CodeGen::CodeGenFunction &CGF,
John McCallef072fd2010-05-22 01:48:05 +00001057 ReturnValueSlot Return,
Daniel Dunbar7f8ea5c2008-08-30 05:35:15 +00001058 QualType ResultType,
1059 Selector Sel,
Daniel Dunbar14c80b72008-08-23 09:25:55 +00001060 llvm::Value *Arg0,
Daniel Dunbar19cd87e2008-08-30 03:02:31 +00001061 QualType Arg0Ty,
1062 bool IsSuper,
1063 const CallArgList &CallArgs);
Daniel Dunbar14c80b72008-08-23 09:25:55 +00001064
Daniel Dunbar27f9d772008-08-21 04:36:09 +00001065 /// EmitIvarList - Emit the ivar list for the given
1066 /// implementation. If ForClass is true the list of class ivars
1067 /// (i.e. metaclass ivars) is emitted, otherwise the list of
1068 /// interface ivars will be emitted. The return value has type
1069 /// IvarListPtrTy.
1070 llvm::Constant *EmitIvarList(const ObjCImplementationDecl *ID,
Fariborz Jahanian46b86c62009-01-28 19:12:34 +00001071 bool ForClass);
Daniel Dunbar6bff2512009-08-03 17:06:42 +00001072
Daniel Dunbarf56f1912008-08-25 08:19:24 +00001073 /// EmitMetaClass - Emit a forward reference to the class structure
1074 /// for the metaclass of the given interface. The return value has
1075 /// type ClassPtrTy.
1076 llvm::Constant *EmitMetaClassRef(const ObjCInterfaceDecl *ID);
1077
Daniel Dunbar27f9d772008-08-21 04:36:09 +00001078 /// EmitMetaClass - Emit a class structure for the metaclass of the
Daniel Dunbarf56f1912008-08-25 08:19:24 +00001079 /// given implementation. The return value has type ClassPtrTy.
Daniel Dunbar27f9d772008-08-21 04:36:09 +00001080 llvm::Constant *EmitMetaClass(const ObjCImplementationDecl *ID,
1081 llvm::Constant *Protocols,
Daniel Dunbarae226fa2008-08-27 02:31:56 +00001082 const ConstantVector &Methods);
Daniel Dunbar6bff2512009-08-03 17:06:42 +00001083
Daniel Dunbarae226fa2008-08-27 02:31:56 +00001084 llvm::Constant *GetMethodConstant(const ObjCMethodDecl *MD);
Daniel Dunbar6bff2512009-08-03 17:06:42 +00001085
Daniel Dunbarae226fa2008-08-27 02:31:56 +00001086 llvm::Constant *GetMethodDescriptionConstant(const ObjCMethodDecl *MD);
Daniel Dunbar27f9d772008-08-21 04:36:09 +00001087
1088 /// EmitMethodList - Emit the method list for the given
Daniel Dunbaraf05bb92008-08-26 08:29:31 +00001089 /// implementation. The return value has type MethodListPtrTy.
Daniel Dunbar9c29bf52009-10-18 20:48:59 +00001090 llvm::Constant *EmitMethodList(llvm::Twine Name,
Daniel Dunbar86e253a2008-08-22 20:34:54 +00001091 const char *Section,
Daniel Dunbarae226fa2008-08-27 02:31:56 +00001092 const ConstantVector &Methods);
Daniel Dunbar27f9d772008-08-21 04:36:09 +00001093
1094 /// EmitMethodDescList - Emit a method description list for a list of
Daniel Dunbar6bff2512009-08-03 17:06:42 +00001095 /// method declarations.
Daniel Dunbar6efc0c52008-08-13 03:21:16 +00001096 /// - TypeName: The name for the type containing the methods.
1097 /// - IsProtocol: True iff these methods are for a protocol.
1098 /// - ClassMethds: True iff these are class methods.
1099 /// - Required: When true, only "required" methods are
1100 /// listed. Similarly, when false only "optional" methods are
1101 /// listed. For classes this should always be true.
1102 /// - begin, end: The method list to output.
1103 ///
1104 /// The return value has type MethodDescriptionListPtrTy.
Daniel Dunbar9c29bf52009-10-18 20:48:59 +00001105 llvm::Constant *EmitMethodDescList(llvm::Twine Name,
Daniel Dunbarae226fa2008-08-27 02:31:56 +00001106 const char *Section,
1107 const ConstantVector &Methods);
Daniel Dunbar6efc0c52008-08-13 03:21:16 +00001108
Daniel Dunbar0c0e7a62008-10-29 22:36:39 +00001109 /// GetOrEmitProtocol - Get the protocol object for the given
1110 /// declaration, emitting it if necessary. The return value has type
1111 /// ProtocolPtrTy.
Fariborz Jahanianda320092009-01-29 19:24:30 +00001112 virtual llvm::Constant *GetOrEmitProtocol(const ObjCProtocolDecl *PD);
Daniel Dunbar0c0e7a62008-10-29 22:36:39 +00001113
1114 /// GetOrEmitProtocolRef - Get a forward reference to the protocol
1115 /// object for the given declaration, emitting it if needed. These
1116 /// forward references will be filled in with empty bodies if no
1117 /// definition is seen. The return value has type ProtocolPtrTy.
Fariborz Jahanianda320092009-01-29 19:24:30 +00001118 virtual llvm::Constant *GetOrEmitProtocolRef(const ObjCProtocolDecl *PD);
Daniel Dunbar0c0e7a62008-10-29 22:36:39 +00001119
Daniel Dunbar6efc0c52008-08-13 03:21:16 +00001120 /// EmitProtocolExtension - Generate the protocol extension
1121 /// structure used to store optional instance and class methods, and
1122 /// protocol properties. The return value has type
1123 /// ProtocolExtensionPtrTy.
Daniel Dunbarae226fa2008-08-27 02:31:56 +00001124 llvm::Constant *
1125 EmitProtocolExtension(const ObjCProtocolDecl *PD,
1126 const ConstantVector &OptInstanceMethods,
1127 const ConstantVector &OptClassMethods);
Daniel Dunbar6efc0c52008-08-13 03:21:16 +00001128
1129 /// EmitProtocolList - Generate the list of referenced
1130 /// protocols. The return value has type ProtocolListPtrTy.
Daniel Dunbar9c29bf52009-10-18 20:48:59 +00001131 llvm::Constant *EmitProtocolList(llvm::Twine Name,
Daniel Dunbardbc933702008-08-21 21:57:41 +00001132 ObjCProtocolDecl::protocol_iterator begin,
1133 ObjCProtocolDecl::protocol_iterator end);
Daniel Dunbar6efc0c52008-08-13 03:21:16 +00001134
Daniel Dunbar27f9d772008-08-21 04:36:09 +00001135 /// EmitSelector - Return a Value*, of type ObjCTypes.SelectorPtrTy,
1136 /// for the given selector.
Fariborz Jahanian03b29602010-06-17 19:56:20 +00001137 llvm::Value *EmitSelector(CGBuilderTy &Builder, Selector Sel,
1138 bool lval=false);
Daniel Dunbar6bff2512009-08-03 17:06:42 +00001139
1140public:
Daniel Dunbarc17a4d32008-08-11 02:45:11 +00001141 CGObjCMac(CodeGen::CodeGenModule &cgm);
Daniel Dunbarc17a4d32008-08-11 02:45:11 +00001142
Fariborz Jahanianaa23b572009-01-23 23:53:38 +00001143 virtual llvm::Function *ModuleInitFunction();
Daniel Dunbar6bff2512009-08-03 17:06:42 +00001144
Daniel Dunbar8f2926b2008-08-23 03:46:30 +00001145 virtual CodeGen::RValue GenerateMessageSend(CodeGen::CodeGenFunction &CGF,
John McCallef072fd2010-05-22 01:48:05 +00001146 ReturnValueSlot Return,
Daniel Dunbar7f8ea5c2008-08-30 05:35:15 +00001147 QualType ResultType,
1148 Selector Sel,
Daniel Dunbarf56f1912008-08-25 08:19:24 +00001149 llvm::Value *Receiver,
Fariborz Jahaniandf9ccc62009-05-05 21:36:57 +00001150 const CallArgList &CallArgs,
David Chisnallc6cd5fd2010-04-28 19:33:36 +00001151 const ObjCInterfaceDecl *Class,
Fariborz Jahaniandf9ccc62009-05-05 21:36:57 +00001152 const ObjCMethodDecl *Method);
Daniel Dunbarc17a4d32008-08-11 02:45:11 +00001153
Daniel Dunbar6bff2512009-08-03 17:06:42 +00001154 virtual CodeGen::RValue
Daniel Dunbar8f2926b2008-08-23 03:46:30 +00001155 GenerateMessageSendSuper(CodeGen::CodeGenFunction &CGF,
John McCallef072fd2010-05-22 01:48:05 +00001156 ReturnValueSlot Return,
Daniel Dunbar7f8ea5c2008-08-30 05:35:15 +00001157 QualType ResultType,
1158 Selector Sel,
Daniel Dunbarf56f1912008-08-25 08:19:24 +00001159 const ObjCInterfaceDecl *Class,
Fariborz Jahanian7ce77922009-02-28 20:07:56 +00001160 bool isCategoryImpl,
Daniel Dunbarf56f1912008-08-25 08:19:24 +00001161 llvm::Value *Receiver,
Daniel Dunbar19cd87e2008-08-30 03:02:31 +00001162 bool IsClassMessage,
Daniel Dunbard6c93d72009-09-17 04:01:22 +00001163 const CallArgList &CallArgs,
1164 const ObjCMethodDecl *Method);
Daniel Dunbar6bff2512009-08-03 17:06:42 +00001165
Daniel Dunbar45d196b2008-11-01 01:53:16 +00001166 virtual llvm::Value *GetClass(CGBuilderTy &Builder,
Daniel Dunbar27f9d772008-08-21 04:36:09 +00001167 const ObjCInterfaceDecl *ID);
Daniel Dunbarc17a4d32008-08-11 02:45:11 +00001168
Fariborz Jahanian03b29602010-06-17 19:56:20 +00001169 virtual llvm::Value *GetSelector(CGBuilderTy &Builder, Selector Sel,
1170 bool lval = false);
Fariborz Jahaniandf9ccc62009-05-05 21:36:57 +00001171
1172 /// The NeXT/Apple runtimes do not support typed selectors; just emit an
1173 /// untyped one.
1174 virtual llvm::Value *GetSelector(CGBuilderTy &Builder,
1175 const ObjCMethodDecl *Method);
1176
Daniel Dunbar7ded7f42008-08-15 22:20:32 +00001177 virtual void GenerateCategory(const ObjCCategoryImplDecl *CMD);
Daniel Dunbarc17a4d32008-08-11 02:45:11 +00001178
Daniel Dunbar7ded7f42008-08-15 22:20:32 +00001179 virtual void GenerateClass(const ObjCImplementationDecl *ClassDecl);
Daniel Dunbarc17a4d32008-08-11 02:45:11 +00001180
Daniel Dunbar45d196b2008-11-01 01:53:16 +00001181 virtual llvm::Value *GenerateProtocolRef(CGBuilderTy &Builder,
Daniel Dunbaraf2f62c2008-08-13 00:59:25 +00001182 const ObjCProtocolDecl *PD);
Daniel Dunbar6bff2512009-08-03 17:06:42 +00001183
Chris Lattner74391b42009-03-22 21:03:39 +00001184 virtual llvm::Constant *GetPropertyGetFunction();
1185 virtual llvm::Constant *GetPropertySetFunction();
Fariborz Jahanian6cc59062010-04-12 18:18:10 +00001186 virtual llvm::Constant *GetCopyStructFunction();
Chris Lattner74391b42009-03-22 21:03:39 +00001187 virtual llvm::Constant *EnumerationMutationFunction();
Daniel Dunbar6bff2512009-08-03 17:06:42 +00001188
John McCallf1549f62010-07-06 01:34:17 +00001189 virtual void EmitTryStmt(CodeGen::CodeGenFunction &CGF,
1190 const ObjCAtTryStmt &S);
1191 virtual void EmitSynchronizedStmt(CodeGen::CodeGenFunction &CGF,
1192 const ObjCAtSynchronizedStmt &S);
1193 void EmitTryOrSynchronizedStmt(CodeGen::CodeGenFunction &CGF, const Stmt &S);
Anders Carlsson64d5d6c2008-09-09 10:04:29 +00001194 virtual void EmitThrowStmt(CodeGen::CodeGenFunction &CGF,
1195 const ObjCAtThrowStmt &S);
Fariborz Jahanian3e283e32008-11-18 22:37:34 +00001196 virtual llvm::Value * EmitObjCWeakRead(CodeGen::CodeGenFunction &CGF,
Daniel Dunbar6bff2512009-08-03 17:06:42 +00001197 llvm::Value *AddrWeakObj);
Fariborz Jahanian3e283e32008-11-18 22:37:34 +00001198 virtual void EmitObjCWeakAssign(CodeGen::CodeGenFunction &CGF,
Daniel Dunbar6bff2512009-08-03 17:06:42 +00001199 llvm::Value *src, llvm::Value *dst);
Fariborz Jahanian58626502008-11-19 00:59:10 +00001200 virtual void EmitObjCGlobalAssign(CodeGen::CodeGenFunction &CGF,
1201 llvm::Value *src, llvm::Value *dest);
Fariborz Jahanian7eda8362008-11-20 19:23:36 +00001202 virtual void EmitObjCIvarAssign(CodeGen::CodeGenFunction &CGF,
Fariborz Jahanian6c7a1f32009-09-24 22:25:38 +00001203 llvm::Value *src, llvm::Value *dest,
1204 llvm::Value *ivarOffset);
Fariborz Jahanian58626502008-11-19 00:59:10 +00001205 virtual void EmitObjCStrongCastAssign(CodeGen::CodeGenFunction &CGF,
1206 llvm::Value *src, llvm::Value *dest);
Fariborz Jahanian082b02e2009-07-08 01:18:33 +00001207 virtual void EmitGCMemmoveCollectable(CodeGen::CodeGenFunction &CGF,
1208 llvm::Value *dest, llvm::Value *src,
Fariborz Jahanian55bcace2010-06-15 22:44:06 +00001209 llvm::Value *size);
Daniel Dunbar6bff2512009-08-03 17:06:42 +00001210
Fariborz Jahanian598d3f62009-02-03 19:03:09 +00001211 virtual LValue EmitObjCValueForIvar(CodeGen::CodeGenFunction &CGF,
1212 QualType ObjectTy,
1213 llvm::Value *BaseValue,
1214 const ObjCIvarDecl *Ivar,
Fariborz Jahanian598d3f62009-02-03 19:03:09 +00001215 unsigned CVRQualifiers);
Fariborz Jahanianf63aa3f2009-02-10 19:02:04 +00001216 virtual llvm::Value *EmitIvarOffset(CodeGen::CodeGenFunction &CGF,
Daniel Dunbar2a031922009-04-22 05:08:15 +00001217 const ObjCInterfaceDecl *Interface,
Fariborz Jahanianf63aa3f2009-02-10 19:02:04 +00001218 const ObjCIvarDecl *Ivar);
Daniel Dunbarc17a4d32008-08-11 02:45:11 +00001219};
Daniel Dunbar6bff2512009-08-03 17:06:42 +00001220
Fariborz Jahanian30bc5712009-01-22 23:02:58 +00001221class CGObjCNonFragileABIMac : public CGObjCCommonMac {
Fariborz Jahanianee0af742009-01-21 22:04:16 +00001222private:
Fariborz Jahanian30bc5712009-01-22 23:02:58 +00001223 ObjCNonFragileABITypesHelper ObjCTypes;
Fariborz Jahanianaa23b572009-01-23 23:53:38 +00001224 llvm::GlobalVariable* ObjCEmptyCacheVar;
1225 llvm::GlobalVariable* ObjCEmptyVtableVar;
Daniel Dunbar6bff2512009-08-03 17:06:42 +00001226
Daniel Dunbar11394522009-04-18 08:51:00 +00001227 /// SuperClassReferences - uniqued super class references.
1228 llvm::DenseMap<IdentifierInfo*, llvm::GlobalVariable*> SuperClassReferences;
Daniel Dunbar6bff2512009-08-03 17:06:42 +00001229
Fariborz Jahanian7a06aae2009-02-06 20:09:23 +00001230 /// MetaClassReferences - uniqued meta class references.
1231 llvm::DenseMap<IdentifierInfo*, llvm::GlobalVariable*> MetaClassReferences;
Daniel Dunbare588b992009-03-01 04:46:24 +00001232
1233 /// EHTypeReferences - uniqued class ehtype references.
1234 llvm::DenseMap<IdentifierInfo*, llvm::GlobalVariable*> EHTypeReferences;
Daniel Dunbar6bff2512009-08-03 17:06:42 +00001235
Fariborz Jahanian4523eb02009-05-12 20:06:41 +00001236 /// NonLegacyDispatchMethods - List of methods for which we do *not* generate
Fariborz Jahaniand0f8a8d2009-05-11 19:25:47 +00001237 /// legacy messaging dispatch.
Fariborz Jahanian4523eb02009-05-12 20:06:41 +00001238 llvm::DenseSet<Selector> NonLegacyDispatchMethods;
Daniel Dunbar6bff2512009-08-03 17:06:42 +00001239
Fariborz Jahaniana03d0dd2009-11-17 21:37:35 +00001240 /// DefinedMetaClasses - List of defined meta-classes.
1241 std::vector<llvm::GlobalValue*> DefinedMetaClasses;
1242
Fariborz Jahaniand0f8a8d2009-05-11 19:25:47 +00001243 /// LegacyDispatchedSelector - Returns true if SEL is not in the list of
Fariborz Jahanian4523eb02009-05-12 20:06:41 +00001244 /// NonLegacyDispatchMethods; false otherwise.
Fariborz Jahaniand0f8a8d2009-05-11 19:25:47 +00001245 bool LegacyDispatchedSelector(Selector Sel);
Daniel Dunbar6bff2512009-08-03 17:06:42 +00001246
Fariborz Jahanianaa23b572009-01-23 23:53:38 +00001247 /// FinishNonFragileABIModule - Write out global data structures at the end of
1248 /// processing a translation unit.
1249 void FinishNonFragileABIModule();
Daniel Dunbar8158a2f2009-04-08 04:21:03 +00001250
Daniel Dunbar463b8762009-05-15 21:48:48 +00001251 /// AddModuleClassList - Add the given list of class pointers to the
1252 /// module with the provided symbol and section names.
1253 void AddModuleClassList(const std::vector<llvm::GlobalValue*> &Container,
1254 const char *SymbolName,
1255 const char *SectionName);
1256
Daniel Dunbar6bff2512009-08-03 17:06:42 +00001257 llvm::GlobalVariable * BuildClassRoTInitializer(unsigned flags,
1258 unsigned InstanceStart,
1259 unsigned InstanceSize,
1260 const ObjCImplementationDecl *ID);
Fariborz Jahanian84394a52009-01-24 21:21:53 +00001261 llvm::GlobalVariable * BuildClassMetaData(std::string &ClassName,
Daniel Dunbar6bff2512009-08-03 17:06:42 +00001262 llvm::Constant *IsAGV,
Fariborz Jahanian84394a52009-01-24 21:21:53 +00001263 llvm::Constant *SuperClassGV,
Fariborz Jahaniancf555162009-01-31 00:59:10 +00001264 llvm::Constant *ClassRoGV,
1265 bool HiddenVisibility);
Daniel Dunbar6bff2512009-08-03 17:06:42 +00001266
Fariborz Jahanian493dab72009-01-26 21:38:32 +00001267 llvm::Constant *GetMethodConstant(const ObjCMethodDecl *MD);
Daniel Dunbar6bff2512009-08-03 17:06:42 +00001268
Fariborz Jahanian3819a0b2009-01-30 00:46:37 +00001269 llvm::Constant *GetMethodDescriptionConstant(const ObjCMethodDecl *MD);
Daniel Dunbar6bff2512009-08-03 17:06:42 +00001270
Fariborz Jahanian493dab72009-01-26 21:38:32 +00001271 /// EmitMethodList - Emit the method list for the given
1272 /// implementation. The return value has type MethodListnfABITy.
Daniel Dunbar9c29bf52009-10-18 20:48:59 +00001273 llvm::Constant *EmitMethodList(llvm::Twine Name,
Fariborz Jahanian493dab72009-01-26 21:38:32 +00001274 const char *Section,
Fariborz Jahanian98abf4b2009-01-27 19:38:51 +00001275 const ConstantVector &Methods);
1276 /// EmitIvarList - Emit the ivar list for the given
1277 /// implementation. If ForClass is true the list of class ivars
1278 /// (i.e. metaclass ivars) is emitted, otherwise the list of
1279 /// interface ivars will be emitted. The return value has type
1280 /// IvarListnfABIPtrTy.
1281 llvm::Constant *EmitIvarList(const ObjCImplementationDecl *ID);
Daniel Dunbar6bff2512009-08-03 17:06:42 +00001282
Fariborz Jahanianed157d32009-02-10 20:21:06 +00001283 llvm::Constant *EmitIvarOffsetVar(const ObjCInterfaceDecl *ID,
Fariborz Jahanian2fa5a272009-01-28 01:36:42 +00001284 const ObjCIvarDecl *Ivar,
Fariborz Jahanian1bf0afb2009-01-28 01:05:23 +00001285 unsigned long int offset);
Daniel Dunbar6bff2512009-08-03 17:06:42 +00001286
Fariborz Jahanianda320092009-01-29 19:24:30 +00001287 /// GetOrEmitProtocol - Get the protocol object for the given
1288 /// declaration, emitting it if necessary. The return value has type
1289 /// ProtocolPtrTy.
1290 virtual llvm::Constant *GetOrEmitProtocol(const ObjCProtocolDecl *PD);
Daniel Dunbar6bff2512009-08-03 17:06:42 +00001291
Fariborz Jahanianda320092009-01-29 19:24:30 +00001292 /// GetOrEmitProtocolRef - Get a forward reference to the protocol
1293 /// object for the given declaration, emitting it if needed. These
1294 /// forward references will be filled in with empty bodies if no
1295 /// definition is seen. The return value has type ProtocolPtrTy.
1296 virtual llvm::Constant *GetOrEmitProtocolRef(const ObjCProtocolDecl *PD);
Daniel Dunbar6bff2512009-08-03 17:06:42 +00001297
Fariborz Jahanianda320092009-01-29 19:24:30 +00001298 /// EmitProtocolList - Generate the list of referenced
1299 /// protocols. The return value has type ProtocolListPtrTy.
Daniel Dunbar9c29bf52009-10-18 20:48:59 +00001300 llvm::Constant *EmitProtocolList(llvm::Twine Name,
Fariborz Jahanianda320092009-01-29 19:24:30 +00001301 ObjCProtocolDecl::protocol_iterator begin,
Fariborz Jahanian46551122009-02-04 00:22:57 +00001302 ObjCProtocolDecl::protocol_iterator end);
Daniel Dunbar6bff2512009-08-03 17:06:42 +00001303
Fariborz Jahanian46551122009-02-04 00:22:57 +00001304 CodeGen::RValue EmitMessageSend(CodeGen::CodeGenFunction &CGF,
John McCallef072fd2010-05-22 01:48:05 +00001305 ReturnValueSlot Return,
Fariborz Jahanian46551122009-02-04 00:22:57 +00001306 QualType ResultType,
1307 Selector Sel,
Fariborz Jahanian83a8a752009-02-04 20:42:28 +00001308 llvm::Value *Receiver,
Fariborz Jahanian46551122009-02-04 00:22:57 +00001309 QualType Arg0Ty,
1310 bool IsSuper,
1311 const CallArgList &CallArgs);
Daniel Dunbar5a7379a2009-03-01 04:40:10 +00001312
1313 /// GetClassGlobal - Return the global variable for the Objective-C
1314 /// class of the given name.
Fariborz Jahanian0f902942009-04-14 18:41:56 +00001315 llvm::GlobalVariable *GetClassGlobal(const std::string &Name);
Daniel Dunbar6bff2512009-08-03 17:06:42 +00001316
Fariborz Jahanian0e81f4b2009-02-05 20:41:40 +00001317 /// EmitClassRef - Return a Value*, of type ObjCTypes.ClassPtrTy,
Daniel Dunbar11394522009-04-18 08:51:00 +00001318 /// for the given class reference.
Daniel Dunbar6bff2512009-08-03 17:06:42 +00001319 llvm::Value *EmitClassRef(CGBuilderTy &Builder,
Daniel Dunbar11394522009-04-18 08:51:00 +00001320 const ObjCInterfaceDecl *ID);
Daniel Dunbar6bff2512009-08-03 17:06:42 +00001321
Daniel Dunbar11394522009-04-18 08:51:00 +00001322 /// EmitSuperClassRef - Return a Value*, of type ObjCTypes.ClassPtrTy,
1323 /// for the given super class reference.
Daniel Dunbar6bff2512009-08-03 17:06:42 +00001324 llvm::Value *EmitSuperClassRef(CGBuilderTy &Builder,
1325 const ObjCInterfaceDecl *ID);
1326
Fariborz Jahanian7a06aae2009-02-06 20:09:23 +00001327 /// EmitMetaClassRef - Return a Value * of the address of _class_t
1328 /// meta-data
Daniel Dunbar6bff2512009-08-03 17:06:42 +00001329 llvm::Value *EmitMetaClassRef(CGBuilderTy &Builder,
Fariborz Jahanian7a06aae2009-02-06 20:09:23 +00001330 const ObjCInterfaceDecl *ID);
1331
Fariborz Jahanianed157d32009-02-10 20:21:06 +00001332 /// ObjCIvarOffsetVariable - Returns the ivar offset variable for
1333 /// the given ivar.
1334 ///
Daniel Dunbar5e88bea2009-04-19 00:31:15 +00001335 llvm::GlobalVariable * ObjCIvarOffsetVariable(
Daniel Dunbar6bff2512009-08-03 17:06:42 +00001336 const ObjCInterfaceDecl *ID,
1337 const ObjCIvarDecl *Ivar);
1338
Fariborz Jahanian26cc89f2009-02-11 20:51:17 +00001339 /// EmitSelector - Return a Value*, of type ObjCTypes.SelectorPtrTy,
1340 /// for the given selector.
Fariborz Jahanian03b29602010-06-17 19:56:20 +00001341 llvm::Value *EmitSelector(CGBuilderTy &Builder, Selector Sel,
1342 bool lval=false);
Daniel Dunbare588b992009-03-01 04:46:24 +00001343
Daniel Dunbar8158a2f2009-04-08 04:21:03 +00001344 /// GetInterfaceEHType - Get the cached ehtype for the given Objective-C
Daniel Dunbare588b992009-03-01 04:46:24 +00001345 /// interface. The return value has type EHTypePtrTy.
Daniel Dunbar8158a2f2009-04-08 04:21:03 +00001346 llvm::Value *GetInterfaceEHType(const ObjCInterfaceDecl *ID,
1347 bool ForDefinition);
Daniel Dunbar6ab187a2009-04-07 05:48:37 +00001348
Daniel Dunbar6bff2512009-08-03 17:06:42 +00001349 const char *getMetaclassSymbolPrefix() const {
Daniel Dunbar6ab187a2009-04-07 05:48:37 +00001350 return "OBJC_METACLASS_$_";
1351 }
Daniel Dunbar6bff2512009-08-03 17:06:42 +00001352
Daniel Dunbar6ab187a2009-04-07 05:48:37 +00001353 const char *getClassSymbolPrefix() const {
1354 return "OBJC_CLASS_$_";
1355 }
1356
Daniel Dunbar9f89f2b2009-05-03 12:57:56 +00001357 void GetClassSizeInfo(const ObjCImplementationDecl *OID,
Daniel Dunbarb02532a2009-04-19 23:41:48 +00001358 uint32_t &InstanceStart,
1359 uint32_t &InstanceSize);
Daniel Dunbar6bff2512009-08-03 17:06:42 +00001360
Fariborz Jahanian4523eb02009-05-12 20:06:41 +00001361 // Shamelessly stolen from Analysis/CFRefCount.cpp
Daniel Dunbar74d4b122009-05-15 22:33:15 +00001362 Selector GetNullarySelector(const char* name) const {
Fariborz Jahanian4523eb02009-05-12 20:06:41 +00001363 IdentifierInfo* II = &CGM.getContext().Idents.get(name);
1364 return CGM.getContext().Selectors.getSelector(0, &II);
1365 }
Daniel Dunbar6bff2512009-08-03 17:06:42 +00001366
Daniel Dunbar74d4b122009-05-15 22:33:15 +00001367 Selector GetUnarySelector(const char* name) const {
Fariborz Jahanian4523eb02009-05-12 20:06:41 +00001368 IdentifierInfo* II = &CGM.getContext().Idents.get(name);
1369 return CGM.getContext().Selectors.getSelector(1, &II);
1370 }
Daniel Dunbarb02532a2009-04-19 23:41:48 +00001371
Daniel Dunbar74d4b122009-05-15 22:33:15 +00001372 /// ImplementationIsNonLazy - Check whether the given category or
1373 /// class implementation is "non-lazy".
Fariborz Jahanianecfbdcb2009-05-21 01:03:45 +00001374 bool ImplementationIsNonLazy(const ObjCImplDecl *OD) const;
Daniel Dunbar74d4b122009-05-15 22:33:15 +00001375
Fariborz Jahanianee0af742009-01-21 22:04:16 +00001376public:
Fariborz Jahanian30bc5712009-01-22 23:02:58 +00001377 CGObjCNonFragileABIMac(CodeGen::CodeGenModule &cgm);
Fariborz Jahanianaa23b572009-01-23 23:53:38 +00001378 // FIXME. All stubs for now!
1379 virtual llvm::Function *ModuleInitFunction();
Daniel Dunbar6bff2512009-08-03 17:06:42 +00001380
Fariborz Jahanianaa23b572009-01-23 23:53:38 +00001381 virtual CodeGen::RValue GenerateMessageSend(CodeGen::CodeGenFunction &CGF,
John McCallef072fd2010-05-22 01:48:05 +00001382 ReturnValueSlot Return,
Fariborz Jahanianaa23b572009-01-23 23:53:38 +00001383 QualType ResultType,
1384 Selector Sel,
1385 llvm::Value *Receiver,
Fariborz Jahaniandf9ccc62009-05-05 21:36:57 +00001386 const CallArgList &CallArgs,
David Chisnallc6cd5fd2010-04-28 19:33:36 +00001387 const ObjCInterfaceDecl *Class,
Fariborz Jahaniandf9ccc62009-05-05 21:36:57 +00001388 const ObjCMethodDecl *Method);
Daniel Dunbar6bff2512009-08-03 17:06:42 +00001389
1390 virtual CodeGen::RValue
Fariborz Jahanianaa23b572009-01-23 23:53:38 +00001391 GenerateMessageSendSuper(CodeGen::CodeGenFunction &CGF,
John McCallef072fd2010-05-22 01:48:05 +00001392 ReturnValueSlot Return,
Fariborz Jahanianaa23b572009-01-23 23:53:38 +00001393 QualType ResultType,
1394 Selector Sel,
1395 const ObjCInterfaceDecl *Class,
Fariborz Jahanian7ce77922009-02-28 20:07:56 +00001396 bool isCategoryImpl,
Fariborz Jahanianaa23b572009-01-23 23:53:38 +00001397 llvm::Value *Receiver,
1398 bool IsClassMessage,
Daniel Dunbard6c93d72009-09-17 04:01:22 +00001399 const CallArgList &CallArgs,
1400 const ObjCMethodDecl *Method);
Daniel Dunbar6bff2512009-08-03 17:06:42 +00001401
Fariborz Jahanianaa23b572009-01-23 23:53:38 +00001402 virtual llvm::Value *GetClass(CGBuilderTy &Builder,
Fariborz Jahanian0e81f4b2009-02-05 20:41:40 +00001403 const ObjCInterfaceDecl *ID);
Daniel Dunbar6bff2512009-08-03 17:06:42 +00001404
Fariborz Jahanian03b29602010-06-17 19:56:20 +00001405 virtual llvm::Value *GetSelector(CGBuilderTy &Builder, Selector Sel,
1406 bool lvalue = false)
1407 { return EmitSelector(Builder, Sel, lvalue); }
Fariborz Jahaniandf9ccc62009-05-05 21:36:57 +00001408
1409 /// The NeXT/Apple runtimes do not support typed selectors; just emit an
1410 /// untyped one.
1411 virtual llvm::Value *GetSelector(CGBuilderTy &Builder,
1412 const ObjCMethodDecl *Method)
1413 { return EmitSelector(Builder, Method->getSelector()); }
Daniel Dunbar6bff2512009-08-03 17:06:42 +00001414
Fariborz Jahanianeb062d92009-01-26 18:32:24 +00001415 virtual void GenerateCategory(const ObjCCategoryImplDecl *CMD);
Daniel Dunbar6bff2512009-08-03 17:06:42 +00001416
Fariborz Jahanianaa23b572009-01-23 23:53:38 +00001417 virtual void GenerateClass(const ObjCImplementationDecl *ClassDecl);
Fariborz Jahanianaa23b572009-01-23 23:53:38 +00001418 virtual llvm::Value *GenerateProtocolRef(CGBuilderTy &Builder,
Fariborz Jahanian8cfd3972009-01-30 18:58:59 +00001419 const ObjCProtocolDecl *PD);
Daniel Dunbar6bff2512009-08-03 17:06:42 +00001420
1421 virtual llvm::Constant *GetPropertyGetFunction() {
Chris Lattner72db6c32009-04-22 02:44:54 +00001422 return ObjCTypes.getGetPropertyFn();
Fariborz Jahanianf63aa3f2009-02-10 19:02:04 +00001423 }
Daniel Dunbar6bff2512009-08-03 17:06:42 +00001424 virtual llvm::Constant *GetPropertySetFunction() {
1425 return ObjCTypes.getSetPropertyFn();
Fariborz Jahanianf63aa3f2009-02-10 19:02:04 +00001426 }
Fariborz Jahanian6cc59062010-04-12 18:18:10 +00001427
1428 virtual llvm::Constant *GetCopyStructFunction() {
1429 return ObjCTypes.getCopyStructFn();
1430 }
1431
Chris Lattner74391b42009-03-22 21:03:39 +00001432 virtual llvm::Constant *EnumerationMutationFunction() {
Chris Lattner72db6c32009-04-22 02:44:54 +00001433 return ObjCTypes.getEnumerationMutationFn();
Daniel Dunbar28ed0842009-02-16 18:48:45 +00001434 }
Daniel Dunbar6bff2512009-08-03 17:06:42 +00001435
John McCallf1549f62010-07-06 01:34:17 +00001436 virtual void EmitTryStmt(CodeGen::CodeGenFunction &CGF,
1437 const ObjCAtTryStmt &S);
1438 virtual void EmitSynchronizedStmt(CodeGen::CodeGenFunction &CGF,
1439 const ObjCAtSynchronizedStmt &S);
Fariborz Jahanianaa23b572009-01-23 23:53:38 +00001440 virtual void EmitThrowStmt(CodeGen::CodeGenFunction &CGF,
Anders Carlssonf57c5b22009-02-16 22:59:18 +00001441 const ObjCAtThrowStmt &S);
Fariborz Jahanianaa23b572009-01-23 23:53:38 +00001442 virtual llvm::Value * EmitObjCWeakRead(CodeGen::CodeGenFunction &CGF,
Fariborz Jahanian6948aea2009-02-16 22:52:32 +00001443 llvm::Value *AddrWeakObj);
Fariborz Jahanianaa23b572009-01-23 23:53:38 +00001444 virtual void EmitObjCWeakAssign(CodeGen::CodeGenFunction &CGF,
Fariborz Jahanian6948aea2009-02-16 22:52:32 +00001445 llvm::Value *src, llvm::Value *dst);
Fariborz Jahanianaa23b572009-01-23 23:53:38 +00001446 virtual void EmitObjCGlobalAssign(CodeGen::CodeGenFunction &CGF,
Fariborz Jahanian6948aea2009-02-16 22:52:32 +00001447 llvm::Value *src, llvm::Value *dest);
Fariborz Jahanianaa23b572009-01-23 23:53:38 +00001448 virtual void EmitObjCIvarAssign(CodeGen::CodeGenFunction &CGF,
Fariborz Jahanian6c7a1f32009-09-24 22:25:38 +00001449 llvm::Value *src, llvm::Value *dest,
1450 llvm::Value *ivarOffset);
Fariborz Jahanianaa23b572009-01-23 23:53:38 +00001451 virtual void EmitObjCStrongCastAssign(CodeGen::CodeGenFunction &CGF,
Fariborz Jahanian6948aea2009-02-16 22:52:32 +00001452 llvm::Value *src, llvm::Value *dest);
Fariborz Jahanian082b02e2009-07-08 01:18:33 +00001453 virtual void EmitGCMemmoveCollectable(CodeGen::CodeGenFunction &CGF,
1454 llvm::Value *dest, llvm::Value *src,
Fariborz Jahanian55bcace2010-06-15 22:44:06 +00001455 llvm::Value *size);
Fariborz Jahanian598d3f62009-02-03 19:03:09 +00001456 virtual LValue EmitObjCValueForIvar(CodeGen::CodeGenFunction &CGF,
1457 QualType ObjectTy,
1458 llvm::Value *BaseValue,
1459 const ObjCIvarDecl *Ivar,
Fariborz Jahanian598d3f62009-02-03 19:03:09 +00001460 unsigned CVRQualifiers);
Fariborz Jahanianf63aa3f2009-02-10 19:02:04 +00001461 virtual llvm::Value *EmitIvarOffset(CodeGen::CodeGenFunction &CGF,
Daniel Dunbar2a031922009-04-22 05:08:15 +00001462 const ObjCInterfaceDecl *Interface,
Fariborz Jahanianf63aa3f2009-02-10 19:02:04 +00001463 const ObjCIvarDecl *Ivar);
Fariborz Jahanianee0af742009-01-21 22:04:16 +00001464};
Daniel Dunbar6bff2512009-08-03 17:06:42 +00001465
Daniel Dunbarc17a4d32008-08-11 02:45:11 +00001466} // end anonymous namespace
Daniel Dunbarbbce49b2008-08-12 00:12:39 +00001467
1468/* *** Helper Functions *** */
1469
1470/// getConstantGEP() - Help routine to construct simple GEPs.
Owen Andersona1cf15f2009-07-14 23:10:40 +00001471static llvm::Constant *getConstantGEP(llvm::LLVMContext &VMContext,
Daniel Dunbar6bff2512009-08-03 17:06:42 +00001472 llvm::Constant *C,
Daniel Dunbarbbce49b2008-08-12 00:12:39 +00001473 unsigned idx0,
1474 unsigned idx1) {
1475 llvm::Value *Idxs[] = {
Owen Anderson0032b272009-08-13 21:57:51 +00001476 llvm::ConstantInt::get(llvm::Type::getInt32Ty(VMContext), idx0),
1477 llvm::ConstantInt::get(llvm::Type::getInt32Ty(VMContext), idx1)
Daniel Dunbarbbce49b2008-08-12 00:12:39 +00001478 };
Owen Anderson3c4972d2009-07-29 18:54:39 +00001479 return llvm::ConstantExpr::getGetElementPtr(C, Idxs, 2);
Daniel Dunbarbbce49b2008-08-12 00:12:39 +00001480}
1481
Daniel Dunbar8158a2f2009-04-08 04:21:03 +00001482/// hasObjCExceptionAttribute - Return true if this class or any super
1483/// class has the __objc_exception__ attribute.
Daniel Dunbar6bff2512009-08-03 17:06:42 +00001484static bool hasObjCExceptionAttribute(ASTContext &Context,
Douglas Gregor68584ed2009-06-18 16:11:24 +00001485 const ObjCInterfaceDecl *OID) {
Argyrios Kyrtzidis40b598e2009-06-30 02:34:44 +00001486 if (OID->hasAttr<ObjCExceptionAttr>())
Daniel Dunbar8158a2f2009-04-08 04:21:03 +00001487 return true;
1488 if (const ObjCInterfaceDecl *Super = OID->getSuperClass())
Douglas Gregor68584ed2009-06-18 16:11:24 +00001489 return hasObjCExceptionAttribute(Context, Super);
Daniel Dunbar8158a2f2009-04-08 04:21:03 +00001490 return false;
1491}
1492
Daniel Dunbarbbce49b2008-08-12 00:12:39 +00001493/* *** CGObjCMac Public Interface *** */
Daniel Dunbar6bff2512009-08-03 17:06:42 +00001494
Fariborz Jahanianee0af742009-01-21 22:04:16 +00001495CGObjCMac::CGObjCMac(CodeGen::CodeGenModule &cgm) : CGObjCCommonMac(cgm),
Mike Stump1eb44332009-09-09 15:08:12 +00001496 ObjCTypes(cgm) {
Fariborz Jahanianee0af742009-01-21 22:04:16 +00001497 ObjCABI = 1;
Daniel Dunbar6bff2512009-08-03 17:06:42 +00001498 EmitImageInfo();
Daniel Dunbarc17a4d32008-08-11 02:45:11 +00001499}
1500
Daniel Dunbarddb2a3d2008-08-16 00:25:02 +00001501/// GetClass - Return a reference to the class for the given interface
1502/// decl.
Daniel Dunbar45d196b2008-11-01 01:53:16 +00001503llvm::Value *CGObjCMac::GetClass(CGBuilderTy &Builder,
Daniel Dunbar27f9d772008-08-21 04:36:09 +00001504 const ObjCInterfaceDecl *ID) {
1505 return EmitClassRef(Builder, ID);
Daniel Dunbarc17a4d32008-08-11 02:45:11 +00001506}
1507
1508/// GetSelector - Return the pointer to the unique'd string for this selector.
Fariborz Jahanian03b29602010-06-17 19:56:20 +00001509llvm::Value *CGObjCMac::GetSelector(CGBuilderTy &Builder, Selector Sel,
1510 bool lval) {
1511 return EmitSelector(Builder, Sel, lval);
Daniel Dunbarc17a4d32008-08-11 02:45:11 +00001512}
Fariborz Jahaniandf9ccc62009-05-05 21:36:57 +00001513llvm::Value *CGObjCMac::GetSelector(CGBuilderTy &Builder, const ObjCMethodDecl
Daniel Dunbar6bff2512009-08-03 17:06:42 +00001514 *Method) {
Fariborz Jahaniandf9ccc62009-05-05 21:36:57 +00001515 return EmitSelector(Builder, Method->getSelector());
1516}
Daniel Dunbarc17a4d32008-08-11 02:45:11 +00001517
Daniel Dunbarbbce49b2008-08-12 00:12:39 +00001518/// Generate a constant CFString object.
Daniel Dunbar6bff2512009-08-03 17:06:42 +00001519/*
1520 struct __builtin_CFString {
1521 const int *isa; // point to __CFConstantStringClassReference
1522 int flags;
1523 const char *str;
1524 long length;
1525 };
Daniel Dunbarbbce49b2008-08-12 00:12:39 +00001526*/
1527
Fariborz Jahanian33e982b2010-04-22 20:26:39 +00001528/// or Generate a constant NSString object.
1529/*
1530 struct __builtin_NSString {
1531 const int *isa; // point to __NSConstantStringClassReference
1532 const char *str;
1533 unsigned int length;
1534 };
1535*/
1536
Fariborz Jahanianaa23b572009-01-23 23:53:38 +00001537llvm::Constant *CGObjCCommonMac::GenerateConstantString(
David Chisnall0d13f6f2010-01-23 02:40:42 +00001538 const StringLiteral *SL) {
Fariborz Jahanian33e982b2010-04-22 20:26:39 +00001539 return (CGM.getLangOptions().NoConstantCFStrings == 0 ?
1540 CGM.GetAddrOfConstantCFString(SL) :
1541 CGM.GetAddrOfConstantNSString(SL));
Daniel Dunbarc17a4d32008-08-11 02:45:11 +00001542}
1543
1544/// Generates a message send where the super is the receiver. This is
1545/// a message send to self with special delivery semantics indicating
1546/// which class's method should be called.
Daniel Dunbar8f2926b2008-08-23 03:46:30 +00001547CodeGen::RValue
1548CGObjCMac::GenerateMessageSendSuper(CodeGen::CodeGenFunction &CGF,
John McCallef072fd2010-05-22 01:48:05 +00001549 ReturnValueSlot Return,
Daniel Dunbar7f8ea5c2008-08-30 05:35:15 +00001550 QualType ResultType,
1551 Selector Sel,
Daniel Dunbarf56f1912008-08-25 08:19:24 +00001552 const ObjCInterfaceDecl *Class,
Fariborz Jahanian7ce77922009-02-28 20:07:56 +00001553 bool isCategoryImpl,
Daniel Dunbarf56f1912008-08-25 08:19:24 +00001554 llvm::Value *Receiver,
Daniel Dunbar19cd87e2008-08-30 03:02:31 +00001555 bool IsClassMessage,
Daniel Dunbard6c93d72009-09-17 04:01:22 +00001556 const CodeGen::CallArgList &CallArgs,
1557 const ObjCMethodDecl *Method) {
Daniel Dunbare8b470d2008-08-23 04:28:29 +00001558 // Create and init a super structure; this is a (receiver, class)
1559 // pair we will pass to objc_msgSendSuper.
Daniel Dunbar6bff2512009-08-03 17:06:42 +00001560 llvm::Value *ObjCSuper =
Daniel Dunbare8b470d2008-08-23 04:28:29 +00001561 CGF.Builder.CreateAlloca(ObjCTypes.SuperTy, 0, "objc_super");
Daniel Dunbar6bff2512009-08-03 17:06:42 +00001562 llvm::Value *ReceiverAsObject =
Daniel Dunbare8b470d2008-08-23 04:28:29 +00001563 CGF.Builder.CreateBitCast(Receiver, ObjCTypes.ObjectPtrTy);
Daniel Dunbar6bff2512009-08-03 17:06:42 +00001564 CGF.Builder.CreateStore(ReceiverAsObject,
Daniel Dunbare8b470d2008-08-23 04:28:29 +00001565 CGF.Builder.CreateStructGEP(ObjCSuper, 0));
Daniel Dunbare8b470d2008-08-23 04:28:29 +00001566
Daniel Dunbarf56f1912008-08-25 08:19:24 +00001567 // If this is a class message the metaclass is passed as the target.
1568 llvm::Value *Target;
1569 if (IsClassMessage) {
Fariborz Jahanian7ce77922009-02-28 20:07:56 +00001570 if (isCategoryImpl) {
1571 // Message sent to 'super' in a class method defined in a category
1572 // implementation requires an odd treatment.
1573 // If we are in a class method, we must retrieve the
1574 // _metaclass_ for the current class, pointed at by
1575 // the class's "isa" pointer. The following assumes that
1576 // isa" is the first ivar in a class (which it must be).
1577 Target = EmitClassRef(CGF.Builder, Class->getSuperClass());
1578 Target = CGF.Builder.CreateStructGEP(Target, 0);
1579 Target = CGF.Builder.CreateLoad(Target);
Mike Stumpb3589f42009-07-30 22:28:39 +00001580 } else {
Fariborz Jahanian7ce77922009-02-28 20:07:56 +00001581 llvm::Value *MetaClassPtr = EmitMetaClassRef(Class);
1582 llvm::Value *SuperPtr = CGF.Builder.CreateStructGEP(MetaClassPtr, 1);
1583 llvm::Value *Super = CGF.Builder.CreateLoad(SuperPtr);
1584 Target = Super;
Daniel Dunbar6bff2512009-08-03 17:06:42 +00001585 }
Fariborz Jahanian182f2682009-11-14 02:18:31 +00001586 }
1587 else if (isCategoryImpl)
1588 Target = EmitClassRef(CGF.Builder, Class->getSuperClass());
1589 else {
Fariborz Jahanianb0069ee2009-11-12 20:14:24 +00001590 llvm::Value *ClassPtr = EmitSuperClassRef(Class);
1591 ClassPtr = CGF.Builder.CreateStructGEP(ClassPtr, 1);
1592 Target = CGF.Builder.CreateLoad(ClassPtr);
Daniel Dunbarf56f1912008-08-25 08:19:24 +00001593 }
Mike Stumpf5408fe2009-05-16 07:57:57 +00001594 // FIXME: We shouldn't need to do this cast, rectify the ASTContext and
1595 // ObjCTypes types.
Daniel Dunbar6bff2512009-08-03 17:06:42 +00001596 const llvm::Type *ClassTy =
Daniel Dunbar19cd87e2008-08-30 03:02:31 +00001597 CGM.getTypes().ConvertType(CGF.getContext().getObjCClassType());
Daniel Dunbar0c0e7a62008-10-29 22:36:39 +00001598 Target = CGF.Builder.CreateBitCast(Target, ClassTy);
Daniel Dunbar6bff2512009-08-03 17:06:42 +00001599 CGF.Builder.CreateStore(Target,
Daniel Dunbarf56f1912008-08-25 08:19:24 +00001600 CGF.Builder.CreateStructGEP(ObjCSuper, 1));
John McCallef072fd2010-05-22 01:48:05 +00001601 return EmitLegacyMessageSend(CGF, Return, ResultType,
Fariborz Jahaniand0f8a8d2009-05-11 19:25:47 +00001602 EmitSelector(CGF.Builder, Sel),
1603 ObjCSuper, ObjCTypes.SuperPtrCTy,
Daniel Dunbard6c93d72009-09-17 04:01:22 +00001604 true, CallArgs, Method, ObjCTypes);
Daniel Dunbarc17a4d32008-08-11 02:45:11 +00001605}
Daniel Dunbar6bff2512009-08-03 17:06:42 +00001606
1607/// Generate code for a message send expression.
Daniel Dunbar8f2926b2008-08-23 03:46:30 +00001608CodeGen::RValue CGObjCMac::GenerateMessageSend(CodeGen::CodeGenFunction &CGF,
John McCallef072fd2010-05-22 01:48:05 +00001609 ReturnValueSlot Return,
Daniel Dunbar7f8ea5c2008-08-30 05:35:15 +00001610 QualType ResultType,
1611 Selector Sel,
Daniel Dunbarf56f1912008-08-25 08:19:24 +00001612 llvm::Value *Receiver,
Fariborz Jahaniandf9ccc62009-05-05 21:36:57 +00001613 const CallArgList &CallArgs,
David Chisnallc6cd5fd2010-04-28 19:33:36 +00001614 const ObjCInterfaceDecl *Class,
Fariborz Jahaniandf9ccc62009-05-05 21:36:57 +00001615 const ObjCMethodDecl *Method) {
John McCallef072fd2010-05-22 01:48:05 +00001616 return EmitLegacyMessageSend(CGF, Return, ResultType,
Fariborz Jahaniand0f8a8d2009-05-11 19:25:47 +00001617 EmitSelector(CGF.Builder, Sel),
1618 Receiver, CGF.getContext().getObjCIdType(),
Daniel Dunbard6c93d72009-09-17 04:01:22 +00001619 false, CallArgs, Method, ObjCTypes);
Daniel Dunbar14c80b72008-08-23 09:25:55 +00001620}
1621
Daniel Dunbard6c93d72009-09-17 04:01:22 +00001622CodeGen::RValue
1623CGObjCCommonMac::EmitLegacyMessageSend(CodeGen::CodeGenFunction &CGF,
John McCallef072fd2010-05-22 01:48:05 +00001624 ReturnValueSlot Return,
Daniel Dunbard6c93d72009-09-17 04:01:22 +00001625 QualType ResultType,
1626 llvm::Value *Sel,
1627 llvm::Value *Arg0,
1628 QualType Arg0Ty,
1629 bool IsSuper,
1630 const CallArgList &CallArgs,
1631 const ObjCMethodDecl *Method,
1632 const ObjCCommonTypesHelper &ObjCTypes) {
Daniel Dunbar19cd87e2008-08-30 03:02:31 +00001633 CallArgList ActualArgs;
Fariborz Jahaniand019d962009-04-24 21:07:43 +00001634 if (!IsSuper)
1635 Arg0 = CGF.Builder.CreateBitCast(Arg0, ObjCTypes.ObjectPtrTy, "tmp");
Daniel Dunbar46f45b92008-09-09 01:06:48 +00001636 ActualArgs.push_back(std::make_pair(RValue::get(Arg0), Arg0Ty));
Fariborz Jahaniand0f8a8d2009-05-11 19:25:47 +00001637 ActualArgs.push_back(std::make_pair(RValue::get(Sel),
Daniel Dunbar19cd87e2008-08-30 03:02:31 +00001638 CGF.getContext().getObjCSelType()));
1639 ActualArgs.insert(ActualArgs.end(), CallArgs.begin(), CallArgs.end());
Daniel Dunbar6bff2512009-08-03 17:06:42 +00001640
Daniel Dunbar541b63b2009-02-02 23:23:47 +00001641 CodeGenTypes &Types = CGM.getTypes();
John McCall04a67a62010-02-05 21:31:56 +00001642 const CGFunctionInfo &FnInfo = Types.getFunctionInfo(ResultType, ActualArgs,
Rafael Espindola264ba482010-03-30 20:24:48 +00001643 FunctionType::ExtInfo());
Daniel Dunbar67939662009-09-17 04:01:40 +00001644 const llvm::FunctionType *FTy =
1645 Types.GetFunctionType(FnInfo, Method ? Method->isVariadic() : false);
Daniel Dunbar6bff2512009-08-03 17:06:42 +00001646
Anders Carlsson7e70fb22010-06-21 20:59:55 +00001647 if (Method)
1648 assert(CGM.getContext().getCanonicalType(Method->getResultType()) ==
1649 CGM.getContext().getCanonicalType(ResultType) &&
1650 "Result type mismatch!");
1651
Fariborz Jahaniand0f8a8d2009-05-11 19:25:47 +00001652 llvm::Constant *Fn = NULL;
Daniel Dunbar88b53962009-02-02 22:03:45 +00001653 if (CGM.ReturnTypeUsesSret(FnInfo)) {
Fariborz Jahaniand0f8a8d2009-05-11 19:25:47 +00001654 Fn = (ObjCABI == 2) ? ObjCTypes.getSendStretFn2(IsSuper)
Daniel Dunbar6bff2512009-08-03 17:06:42 +00001655 : ObjCTypes.getSendStretFn(IsSuper);
Douglas Gregor0c293ea2010-06-22 23:07:26 +00001656 } else if (ResultType->isRealFloatingType()) {
Fariborz Jahaniand0f8a8d2009-05-11 19:25:47 +00001657 if (ObjCABI == 2) {
John McCall183700f2009-09-21 23:43:11 +00001658 if (const BuiltinType *BT = ResultType->getAs<BuiltinType>()) {
Fariborz Jahaniand0f8a8d2009-05-11 19:25:47 +00001659 BuiltinType::Kind k = BT->getKind();
1660 Fn = (k == BuiltinType::LongDouble) ? ObjCTypes.getSendFpretFn2(IsSuper)
Daniel Dunbar6bff2512009-08-03 17:06:42 +00001661 : ObjCTypes.getSendFn2(IsSuper);
Daniel Dunbar42f963d2009-06-10 04:38:50 +00001662 } else {
1663 Fn = ObjCTypes.getSendFn2(IsSuper);
Fariborz Jahaniand0f8a8d2009-05-11 19:25:47 +00001664 }
Mike Stumpb3589f42009-07-30 22:28:39 +00001665 } else
Mike Stumpf5408fe2009-05-16 07:57:57 +00001666 // FIXME. This currently matches gcc's API for x86-32. May need to change
1667 // for others if we have their API.
Fariborz Jahaniand0f8a8d2009-05-11 19:25:47 +00001668 Fn = ObjCTypes.getSendFpretFn(IsSuper);
Daniel Dunbar5669e572008-10-17 03:24:53 +00001669 } else {
Fariborz Jahaniand0f8a8d2009-05-11 19:25:47 +00001670 Fn = (ObjCABI == 2) ? ObjCTypes.getSendFn2(IsSuper)
Daniel Dunbar6bff2512009-08-03 17:06:42 +00001671 : ObjCTypes.getSendFn(IsSuper);
Daniel Dunbar5669e572008-10-17 03:24:53 +00001672 }
Fariborz Jahaniand0f8a8d2009-05-11 19:25:47 +00001673 assert(Fn && "EmitLegacyMessageSend - unknown API");
Owen Anderson3c4972d2009-07-29 18:54:39 +00001674 Fn = llvm::ConstantExpr::getBitCast(Fn,
Daniel Dunbar6bff2512009-08-03 17:06:42 +00001675 llvm::PointerType::getUnqual(FTy));
John McCallef072fd2010-05-22 01:48:05 +00001676 return CGF.EmitCall(FnInfo, Fn, Return, ActualArgs);
Daniel Dunbarc17a4d32008-08-11 02:45:11 +00001677}
1678
Daniel Dunbar6bff2512009-08-03 17:06:42 +00001679llvm::Value *CGObjCMac::GenerateProtocolRef(CGBuilderTy &Builder,
Daniel Dunbaraf2f62c2008-08-13 00:59:25 +00001680 const ObjCProtocolDecl *PD) {
Daniel Dunbarc67876d2008-09-04 04:33:15 +00001681 // FIXME: I don't understand why gcc generates this, or where it is
Mike Stumpf5408fe2009-05-16 07:57:57 +00001682 // resolved. Investigate. Its also wasteful to look this up over and over.
Daniel Dunbarc67876d2008-09-04 04:33:15 +00001683 LazySymbols.insert(&CGM.getContext().Idents.get("Protocol"));
1684
Owen Anderson3c4972d2009-07-29 18:54:39 +00001685 return llvm::ConstantExpr::getBitCast(GetProtocolRef(PD),
Daniel Dunbar6efc0c52008-08-13 03:21:16 +00001686 ObjCTypes.ExternalProtocolPtrTy);
Daniel Dunbarc17a4d32008-08-11 02:45:11 +00001687}
1688
Fariborz Jahanianda320092009-01-29 19:24:30 +00001689void CGObjCCommonMac::GenerateProtocol(const ObjCProtocolDecl *PD) {
Mike Stumpf5408fe2009-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 Dunbar0c0e7a62008-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 Dunbar6bff2512009-08-03 17:06:42 +00001697 if (Protocols.count(PD->getIdentifier()))
Daniel Dunbar0c0e7a62008-10-29 22:36:39 +00001698 GetOrEmitProtocol(PD);
1699}
1700
Fariborz Jahanianda320092009-01-29 19:24:30 +00001701llvm::Constant *CGObjCCommonMac::GetProtocolRef(const ObjCProtocolDecl *PD) {
Daniel Dunbar0c0e7a62008-10-29 22:36:39 +00001702 if (DefinedProtocols.count(PD->getIdentifier()))
1703 return GetOrEmitProtocol(PD);
1704 return GetOrEmitProtocolRef(PD);
1705}
1706
Daniel Dunbar6efc0c52008-08-13 03:21:16 +00001707/*
Daniel Dunbar6bff2512009-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 Dunbar6efc0c52008-08-13 03:21:16 +00001716
Daniel Dunbar6bff2512009-08-03 17:06:42 +00001717See EmitProtocolExtension().
Daniel Dunbar6efc0c52008-08-13 03:21:16 +00001718*/
Daniel Dunbar0c0e7a62008-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 Dunbar242d4dc2008-08-25 06:02:07 +00001726 // FIXME: I don't understand why gcc generates this, or where it is
Mike Stumpf5408fe2009-05-16 07:57:57 +00001727 // resolved. Investigate. Its also wasteful to look this up over and over.
Daniel Dunbar242d4dc2008-08-25 06:02:07 +00001728 LazySymbols.insert(&CGM.getContext().Idents.get("Protocol"));
1729
Daniel Dunbarae226fa2008-08-27 02:31:56 +00001730 // Construct method lists.
1731 std::vector<llvm::Constant*> InstanceMethods, ClassMethods;
1732 std::vector<llvm::Constant*> OptInstanceMethods, OptClassMethods;
Daniel Dunbar6bff2512009-08-03 17:06:42 +00001733 for (ObjCProtocolDecl::instmeth_iterator
Argyrios Kyrtzidis17945a02009-06-30 02:36:12 +00001734 i = PD->instmeth_begin(), e = PD->instmeth_end(); i != e; ++i) {
Daniel Dunbarae226fa2008-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 Dunbar6bff2512009-08-03 17:06:42 +00001741 }
Daniel Dunbarae226fa2008-08-27 02:31:56 +00001742 }
1743
Daniel Dunbar6bff2512009-08-03 17:06:42 +00001744 for (ObjCProtocolDecl::classmeth_iterator
Argyrios Kyrtzidis17945a02009-06-30 02:36:12 +00001745 i = PD->classmeth_begin(), e = PD->classmeth_end(); i != e; ++i) {
Daniel Dunbarae226fa2008-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 Dunbar6bff2512009-08-03 17:06:42 +00001752 }
Daniel Dunbarae226fa2008-08-27 02:31:56 +00001753 }
1754
Daniel Dunbar6efc0c52008-08-13 03:21:16 +00001755 std::vector<llvm::Constant*> Values(5);
Daniel Dunbarae226fa2008-08-27 02:31:56 +00001756 Values[0] = EmitProtocolExtension(PD, OptInstanceMethods, OptClassMethods);
Daniel Dunbar6efc0c52008-08-13 03:21:16 +00001757 Values[1] = GetClassName(PD->getIdentifier());
Daniel Dunbar6bff2512009-08-03 17:06:42 +00001758 Values[2] =
Daniel Dunbar9c29bf52009-10-18 20:48:59 +00001759 EmitProtocolList("\01L_OBJC_PROTOCOL_REFS_" + PD->getName(),
Daniel Dunbardbc933702008-08-21 21:57:41 +00001760 PD->protocol_begin(),
1761 PD->protocol_end());
Daniel Dunbar6bff2512009-08-03 17:06:42 +00001762 Values[3] =
Daniel Dunbar9c29bf52009-10-18 20:48:59 +00001763 EmitMethodDescList("\01L_OBJC_PROTOCOL_INSTANCE_METHODS_" + PD->getName(),
Daniel Dunbarae226fa2008-08-27 02:31:56 +00001764 "__OBJC,__cat_inst_meth,regular,no_dead_strip",
1765 InstanceMethods);
Daniel Dunbar6bff2512009-08-03 17:06:42 +00001766 Values[4] =
Daniel Dunbar9c29bf52009-10-18 20:48:59 +00001767 EmitMethodDescList("\01L_OBJC_PROTOCOL_CLASS_METHODS_" + PD->getName(),
Daniel Dunbarae226fa2008-08-27 02:31:56 +00001768 "__OBJC,__cat_cls_meth,regular,no_dead_strip",
1769 ClassMethods);
Owen Anderson08e25242009-07-27 22:29:56 +00001770 llvm::Constant *Init = llvm::ConstantStruct::get(ObjCTypes.ProtocolTy,
Daniel Dunbar6efc0c52008-08-13 03:21:16 +00001771 Values);
Daniel Dunbar6bff2512009-08-03 17:06:42 +00001772
Daniel Dunbar6efc0c52008-08-13 03:21:16 +00001773 if (Entry) {
Daniel Dunbar0c0e7a62008-10-29 22:36:39 +00001774 // Already created, fix the linkage and update the initializer.
1775 Entry->setLinkage(llvm::GlobalValue::InternalLinkage);
Daniel Dunbar6efc0c52008-08-13 03:21:16 +00001776 Entry->setInitializer(Init);
1777 } else {
Daniel Dunbar6bff2512009-08-03 17:06:42 +00001778 Entry =
Owen Anderson1c431b32009-07-08 19:05:04 +00001779 new llvm::GlobalVariable(CGM.getModule(), ObjCTypes.ProtocolTy, false,
Daniel Dunbar6efc0c52008-08-13 03:21:16 +00001780 llvm::GlobalValue::InternalLinkage,
Daniel Dunbar6bff2512009-08-03 17:06:42 +00001781 Init,
Daniel Dunbar9c29bf52009-10-18 20:48:59 +00001782 "\01L_OBJC_PROTOCOL_" + PD->getName());
Daniel Dunbar6efc0c52008-08-13 03:21:16 +00001783 Entry->setSection("__OBJC,__protocol,regular,no_dead_strip");
Daniel Dunbar6efc0c52008-08-13 03:21:16 +00001784 // FIXME: Is this necessary? Why only for protocol?
1785 Entry->setAlignment(4);
1786 }
Chris Lattnerad64e022009-07-17 23:57:13 +00001787 CGM.AddUsedGlobal(Entry);
Daniel Dunbar0c0e7a62008-10-29 22:36:39 +00001788
1789 return Entry;
Daniel Dunbar6efc0c52008-08-13 03:21:16 +00001790}
1791
Daniel Dunbar0c0e7a62008-10-29 22:36:39 +00001792llvm::Constant *CGObjCMac::GetOrEmitProtocolRef(const ObjCProtocolDecl *PD) {
Daniel Dunbar6efc0c52008-08-13 03:21:16 +00001793 llvm::GlobalVariable *&Entry = Protocols[PD->getIdentifier()];
1794
1795 if (!Entry) {
Daniel Dunbar0c0e7a62008-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 Dunbar6bff2512009-08-03 17:06:42 +00001799 Entry =
Owen Anderson1c431b32009-07-08 19:05:04 +00001800 new llvm::GlobalVariable(CGM.getModule(), ObjCTypes.ProtocolTy, false,
Daniel Dunbar0c0e7a62008-10-29 22:36:39 +00001801 llvm::GlobalValue::ExternalLinkage,
1802 0,
Daniel Dunbar9c29bf52009-10-18 20:48:59 +00001803 "\01L_OBJC_PROTOCOL_" + PD->getName());
Daniel Dunbar6efc0c52008-08-13 03:21:16 +00001804 Entry->setSection("__OBJC,__protocol,regular,no_dead_strip");
Daniel Dunbar6efc0c52008-08-13 03:21:16 +00001805 // FIXME: Is this necessary? Why only for protocol?
1806 Entry->setAlignment(4);
1807 }
Daniel Dunbar6bff2512009-08-03 17:06:42 +00001808
Daniel Dunbar6efc0c52008-08-13 03:21:16 +00001809 return Entry;
1810}
1811
1812/*
1813 struct _objc_protocol_extension {
Daniel Dunbar6bff2512009-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 Dunbar6efc0c52008-08-13 03:21:16 +00001818 };
1819*/
Daniel Dunbarae226fa2008-08-27 02:31:56 +00001820llvm::Constant *
1821CGObjCMac::EmitProtocolExtension(const ObjCProtocolDecl *PD,
1822 const ConstantVector &OptInstanceMethods,
1823 const ConstantVector &OptClassMethods) {
Daniel Dunbar6bff2512009-08-03 17:06:42 +00001824 uint64_t Size =
Duncan Sands9408c452009-05-09 07:08:47 +00001825 CGM.getTargetData().getTypeAllocSize(ObjCTypes.ProtocolExtensionTy);
Daniel Dunbar6efc0c52008-08-13 03:21:16 +00001826 std::vector<llvm::Constant*> Values(4);
Owen Anderson4a28d5d2009-07-24 23:12:58 +00001827 Values[0] = llvm::ConstantInt::get(ObjCTypes.IntTy, Size);
Daniel Dunbar6bff2512009-08-03 17:06:42 +00001828 Values[1] =
1829 EmitMethodDescList("\01L_OBJC_PROTOCOL_INSTANCE_METHODS_OPT_"
Daniel Dunbar9c29bf52009-10-18 20:48:59 +00001830 + PD->getName(),
Daniel Dunbarae226fa2008-08-27 02:31:56 +00001831 "__OBJC,__cat_inst_meth,regular,no_dead_strip",
1832 OptInstanceMethods);
Daniel Dunbar6bff2512009-08-03 17:06:42 +00001833 Values[2] =
Daniel Dunbar9c29bf52009-10-18 20:48:59 +00001834 EmitMethodDescList("\01L_OBJC_PROTOCOL_CLASS_METHODS_OPT_" + PD->getName(),
Daniel Dunbarae226fa2008-08-27 02:31:56 +00001835 "__OBJC,__cat_cls_meth,regular,no_dead_strip",
1836 OptClassMethods);
Daniel Dunbar9c29bf52009-10-18 20:48:59 +00001837 Values[3] = EmitPropertyList("\01L_OBJC_$_PROP_PROTO_LIST_" + PD->getName(),
Fariborz Jahanian5de14dc2009-01-28 22:18:42 +00001838 0, PD, ObjCTypes);
Daniel Dunbar6efc0c52008-08-13 03:21:16 +00001839
Daniel Dunbar27f9d772008-08-21 04:36:09 +00001840 // Return null if no extension bits are used.
Daniel Dunbar6bff2512009-08-03 17:06:42 +00001841 if (Values[1]->isNullValue() && Values[2]->isNullValue() &&
Daniel Dunbar6efc0c52008-08-13 03:21:16 +00001842 Values[3]->isNullValue())
Owen Andersonc9c88b42009-07-31 20:28:54 +00001843 return llvm::Constant::getNullValue(ObjCTypes.ProtocolExtensionPtrTy);
Daniel Dunbar6efc0c52008-08-13 03:21:16 +00001844
Daniel Dunbar6bff2512009-08-03 17:06:42 +00001845 llvm::Constant *Init =
Owen Anderson08e25242009-07-27 22:29:56 +00001846 llvm::ConstantStruct::get(ObjCTypes.ProtocolExtensionTy, Values);
Daniel Dunbar6efc0c52008-08-13 03:21:16 +00001847
Daniel Dunbar63c5b502009-03-09 21:49:58 +00001848 // No special section, but goes in llvm.used
Daniel Dunbar9c29bf52009-10-18 20:48:59 +00001849 return CreateMetadataVar("\01L_OBJC_PROTOCOLEXT_" + PD->getName(),
Daniel Dunbar6bff2512009-08-03 17:06:42 +00001850 Init,
Daniel Dunbar63c5b502009-03-09 21:49:58 +00001851 0, 0, true);
Daniel Dunbar6efc0c52008-08-13 03:21:16 +00001852}
1853
1854/*
1855 struct objc_protocol_list {
Daniel Dunbar6bff2512009-08-03 17:06:42 +00001856 struct objc_protocol_list *next;
1857 long count;
1858 Protocol *list[];
Daniel Dunbar6efc0c52008-08-13 03:21:16 +00001859 };
1860*/
Daniel Dunbardbc933702008-08-21 21:57:41 +00001861llvm::Constant *
Daniel Dunbar9c29bf52009-10-18 20:48:59 +00001862CGObjCMac::EmitProtocolList(llvm::Twine Name,
Daniel Dunbardbc933702008-08-21 21:57:41 +00001863 ObjCProtocolDecl::protocol_iterator begin,
1864 ObjCProtocolDecl::protocol_iterator end) {
Daniel Dunbar6efc0c52008-08-13 03:21:16 +00001865 std::vector<llvm::Constant*> ProtocolRefs;
1866
Daniel Dunbardbc933702008-08-21 21:57:41 +00001867 for (; begin != end; ++begin)
1868 ProtocolRefs.push_back(GetProtocolRef(*begin));
Daniel Dunbar6efc0c52008-08-13 03:21:16 +00001869
1870 // Just return null for empty protocol lists
Daniel Dunbar6bff2512009-08-03 17:06:42 +00001871 if (ProtocolRefs.empty())
Owen Andersonc9c88b42009-07-31 20:28:54 +00001872 return llvm::Constant::getNullValue(ObjCTypes.ProtocolListPtrTy);
Daniel Dunbar6efc0c52008-08-13 03:21:16 +00001873
Daniel Dunbar27f9d772008-08-21 04:36:09 +00001874 // This list is null terminated.
Owen Andersonc9c88b42009-07-31 20:28:54 +00001875 ProtocolRefs.push_back(llvm::Constant::getNullValue(ObjCTypes.ProtocolPtrTy));
Daniel Dunbar6efc0c52008-08-13 03:21:16 +00001876
1877 std::vector<llvm::Constant*> Values(3);
Daniel Dunbar27f9d772008-08-21 04:36:09 +00001878 // This field is only used by the runtime.
Owen Andersonc9c88b42009-07-31 20:28:54 +00001879 Values[0] = llvm::Constant::getNullValue(ObjCTypes.ProtocolListPtrTy);
Owen Anderson4a28d5d2009-07-24 23:12:58 +00001880 Values[1] = llvm::ConstantInt::get(ObjCTypes.LongTy,
Daniel Dunbar6bff2512009-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 Dunbar6efc0c52008-08-13 03:21:16 +00001885 ProtocolRefs);
Daniel Dunbar6bff2512009-08-03 17:06:42 +00001886
Nick Lewycky0d36dd22009-09-19 20:00:52 +00001887 llvm::Constant *Init = llvm::ConstantStruct::get(VMContext, Values, false);
Daniel Dunbar6bff2512009-08-03 17:06:42 +00001888 llvm::GlobalVariable *GV =
Daniel Dunbar63c5b502009-03-09 21:49:58 +00001889 CreateMetadataVar(Name, Init, "__OBJC,__cat_cls_meth,regular,no_dead_strip",
Daniel Dunbar58a29122009-03-09 22:18:41 +00001890 4, false);
Owen Anderson3c4972d2009-07-29 18:54:39 +00001891 return llvm::ConstantExpr::getBitCast(GV, ObjCTypes.ProtocolListPtrTy);
Daniel Dunbar6efc0c52008-08-13 03:21:16 +00001892}
1893
Fariborz Jahanian191dcd72009-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 Dunbar6efc0c52008-08-13 03:21:16 +00001914/*
Daniel Dunbarc8ef5512008-08-23 00:19:03 +00001915 struct _objc_property {
Daniel Dunbar6bff2512009-08-03 17:06:42 +00001916 const char * const name;
1917 const char * const attributes;
Daniel Dunbarc8ef5512008-08-23 00:19:03 +00001918 };
1919
1920 struct _objc_property_list {
Daniel Dunbar6bff2512009-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 Dunbarc8ef5512008-08-23 00:19:03 +00001924 };
1925*/
Daniel Dunbar9c29bf52009-10-18 20:48:59 +00001926llvm::Constant *CGObjCCommonMac::EmitPropertyList(llvm::Twine Name,
Daniel Dunbar6bff2512009-08-03 17:06:42 +00001927 const Decl *Container,
1928 const ObjCContainerDecl *OCD,
1929 const ObjCCommonTypesHelper &ObjCTypes) {
Daniel Dunbarc8ef5512008-08-23 00:19:03 +00001930 std::vector<llvm::Constant*> Properties, Prop(2);
Fariborz Jahanian191dcd72009-12-12 21:26:21 +00001931 llvm::SmallPtrSet<const IdentifierInfo*, 16> PropertySet;
Daniel Dunbar6bff2512009-08-03 17:06:42 +00001932 for (ObjCContainerDecl::prop_iterator I = OCD->prop_begin(),
1933 E = OCD->prop_end(); I != E; ++I) {
Steve Naroff93983f82009-01-11 12:47:58 +00001934 const ObjCPropertyDecl *PD = *I;
Fariborz Jahanian191dcd72009-12-12 21:26:21 +00001935 PropertySet.insert(PD->getIdentifier());
Daniel Dunbarc8ef5512008-08-23 00:19:03 +00001936 Prop[0] = GetPropertyName(PD->getIdentifier());
Daniel Dunbarc56f34a2008-08-28 04:38:10 +00001937 Prop[1] = GetPropertyTypeString(PD, Container);
Owen Anderson08e25242009-07-27 22:29:56 +00001938 Properties.push_back(llvm::ConstantStruct::get(ObjCTypes.PropertyTy,
Daniel Dunbarc8ef5512008-08-23 00:19:03 +00001939 Prop));
1940 }
Fariborz Jahanian6afbdf52010-06-22 16:33:55 +00001941 if (const ObjCInterfaceDecl *OID = dyn_cast<ObjCInterfaceDecl>(OCD)) {
Fariborz Jahanian191dcd72009-12-12 21:26:21 +00001942 for (ObjCInterfaceDecl::protocol_iterator P = OID->protocol_begin(),
1943 E = OID->protocol_end(); P != E; ++P)
Fariborz Jahanian6afbdf52010-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 Dunbarc8ef5512008-08-23 00:19:03 +00001953
1954 // Return null for empty list.
1955 if (Properties.empty())
Owen Andersonc9c88b42009-07-31 20:28:54 +00001956 return llvm::Constant::getNullValue(ObjCTypes.PropertyListPtrTy);
Daniel Dunbarc8ef5512008-08-23 00:19:03 +00001957
Daniel Dunbar6bff2512009-08-03 17:06:42 +00001958 unsigned PropertySize =
Duncan Sands9408c452009-05-09 07:08:47 +00001959 CGM.getTargetData().getTypeAllocSize(ObjCTypes.PropertyTy);
Daniel Dunbarc8ef5512008-08-23 00:19:03 +00001960 std::vector<llvm::Constant*> Values(3);
Owen Anderson4a28d5d2009-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 Dunbar6bff2512009-08-03 17:06:42 +00001963 llvm::ArrayType *AT = llvm::ArrayType::get(ObjCTypes.PropertyTy,
Daniel Dunbarc8ef5512008-08-23 00:19:03 +00001964 Properties.size());
Owen Anderson7db6d832009-07-28 18:33:04 +00001965 Values[2] = llvm::ConstantArray::get(AT, Properties);
Nick Lewycky0d36dd22009-09-19 20:00:52 +00001966 llvm::Constant *Init = llvm::ConstantStruct::get(VMContext, Values, false);
Daniel Dunbarc8ef5512008-08-23 00:19:03 +00001967
Daniel Dunbar6bff2512009-08-03 17:06:42 +00001968 llvm::GlobalVariable *GV =
1969 CreateMetadataVar(Name, Init,
1970 (ObjCABI == 2) ? "__DATA, __objc_const" :
Daniel Dunbar0bf21992009-04-15 02:56:18 +00001971 "__OBJC,__property,regular,no_dead_strip",
Daniel Dunbar6bff2512009-08-03 17:06:42 +00001972 (ObjCABI == 2) ? 8 : 4,
Daniel Dunbar0bf21992009-04-15 02:56:18 +00001973 true);
Owen Anderson3c4972d2009-07-29 18:54:39 +00001974 return llvm::ConstantExpr::getBitCast(GV, ObjCTypes.PropertyListPtrTy);
Daniel Dunbarc8ef5512008-08-23 00:19:03 +00001975}
1976
1977/*
Daniel Dunbar6efc0c52008-08-13 03:21:16 +00001978 struct objc_method_description_list {
Daniel Dunbar6bff2512009-08-03 17:06:42 +00001979 int count;
1980 struct objc_method_description list[];
Daniel Dunbar6efc0c52008-08-13 03:21:16 +00001981 };
1982*/
Daniel Dunbarae226fa2008-08-27 02:31:56 +00001983llvm::Constant *
1984CGObjCMac::GetMethodDescriptionConstant(const ObjCMethodDecl *MD) {
1985 std::vector<llvm::Constant*> Desc(2);
Owen Andersona1cf15f2009-07-14 23:10:40 +00001986 Desc[0] =
Daniel Dunbar6bff2512009-08-03 17:06:42 +00001987 llvm::ConstantExpr::getBitCast(GetMethodVarName(MD->getSelector()),
1988 ObjCTypes.SelectorPtrTy);
Daniel Dunbarae226fa2008-08-27 02:31:56 +00001989 Desc[1] = GetMethodVarType(MD);
Owen Anderson08e25242009-07-27 22:29:56 +00001990 return llvm::ConstantStruct::get(ObjCTypes.MethodDescriptionTy,
Daniel Dunbarae226fa2008-08-27 02:31:56 +00001991 Desc);
1992}
Daniel Dunbar6efc0c52008-08-13 03:21:16 +00001993
Daniel Dunbar9c29bf52009-10-18 20:48:59 +00001994llvm::Constant *CGObjCMac::EmitMethodDescList(llvm::Twine Name,
Daniel Dunbarae226fa2008-08-27 02:31:56 +00001995 const char *Section,
1996 const ConstantVector &Methods) {
Daniel Dunbar6efc0c52008-08-13 03:21:16 +00001997 // Return null for empty list.
1998 if (Methods.empty())
Owen Andersonc9c88b42009-07-31 20:28:54 +00001999 return llvm::Constant::getNullValue(ObjCTypes.MethodDescriptionListPtrTy);
Daniel Dunbar6efc0c52008-08-13 03:21:16 +00002000
2001 std::vector<llvm::Constant*> Values(2);
Owen Anderson4a28d5d2009-07-24 23:12:58 +00002002 Values[0] = llvm::ConstantInt::get(ObjCTypes.IntTy, Methods.size());
Daniel Dunbar6bff2512009-08-03 17:06:42 +00002003 llvm::ArrayType *AT = llvm::ArrayType::get(ObjCTypes.MethodDescriptionTy,
Daniel Dunbar6efc0c52008-08-13 03:21:16 +00002004 Methods.size());
Owen Anderson7db6d832009-07-28 18:33:04 +00002005 Values[1] = llvm::ConstantArray::get(AT, Methods);
Nick Lewycky0d36dd22009-09-19 20:00:52 +00002006 llvm::Constant *Init = llvm::ConstantStruct::get(VMContext, Values, false);
Daniel Dunbar6efc0c52008-08-13 03:21:16 +00002007
Daniel Dunbar0bf21992009-04-15 02:56:18 +00002008 llvm::GlobalVariable *GV = CreateMetadataVar(Name, Init, Section, 4, true);
Daniel Dunbar6bff2512009-08-03 17:06:42 +00002009 return llvm::ConstantExpr::getBitCast(GV,
Daniel Dunbar6efc0c52008-08-13 03:21:16 +00002010 ObjCTypes.MethodDescriptionListPtrTy);
Daniel Dunbarc17a4d32008-08-11 02:45:11 +00002011}
2012
Daniel Dunbar86e253a2008-08-22 20:34:54 +00002013/*
2014 struct _objc_category {
Daniel Dunbar6bff2512009-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 Dunbar86e253a2008-08-22 20:34:54 +00002022 };
Daniel Dunbar6bff2512009-08-03 17:06:42 +00002023*/
Daniel Dunbar7ded7f42008-08-15 22:20:32 +00002024void CGObjCMac::GenerateCategory(const ObjCCategoryImplDecl *OCD) {
Duncan Sands9408c452009-05-09 07:08:47 +00002025 unsigned Size = CGM.getTargetData().getTypeAllocSize(ObjCTypes.CategoryTy);
Daniel Dunbar86e253a2008-08-22 20:34:54 +00002026
Mike Stumpf5408fe2009-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 Dunbar86e253a2008-08-22 20:34:54 +00002031 const ObjCInterfaceDecl *Interface = OCD->getClassInterface();
Daniel Dunbar6bff2512009-08-03 17:06:42 +00002032 const ObjCCategoryDecl *Category =
Daniel Dunbar86e2f402008-08-26 23:03:11 +00002033 Interface->FindCategoryDeclaration(OCD->getIdentifier());
Daniel Dunbar9c29bf52009-10-18 20:48:59 +00002034
2035 llvm::SmallString<256> ExtName;
2036 llvm::raw_svector_ostream(ExtName) << Interface->getName() << '_'
2037 << OCD->getName();
Daniel Dunbar86e253a2008-08-22 20:34:54 +00002038
Daniel Dunbarc45ef602008-08-26 21:51:14 +00002039 std::vector<llvm::Constant*> InstanceMethods, ClassMethods;
Daniel Dunbar6bff2512009-08-03 17:06:42 +00002040 for (ObjCCategoryImplDecl::instmeth_iterator
Argyrios Kyrtzidis17945a02009-06-30 02:36:12 +00002041 i = OCD->instmeth_begin(), e = OCD->instmeth_end(); i != e; ++i) {
Daniel Dunbarc45ef602008-08-26 21:51:14 +00002042 // Instance methods should always be defined.
2043 InstanceMethods.push_back(GetMethodConstant(*i));
2044 }
Daniel Dunbar6bff2512009-08-03 17:06:42 +00002045 for (ObjCCategoryImplDecl::classmeth_iterator
Argyrios Kyrtzidis17945a02009-06-30 02:36:12 +00002046 i = OCD->classmeth_begin(), e = OCD->classmeth_end(); i != e; ++i) {
Daniel Dunbarc45ef602008-08-26 21:51:14 +00002047 // Class methods should always be defined.
2048 ClassMethods.push_back(GetMethodConstant(*i));
2049 }
2050
Daniel Dunbar86e253a2008-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 Jahanian679cd7f2009-04-29 20:40:05 +00002054 LazySymbols.insert(Interface->getIdentifier());
Daniel Dunbar6bff2512009-08-03 17:06:42 +00002055 Values[2] =
Daniel Dunbar9c29bf52009-10-18 20:48:59 +00002056 EmitMethodList("\01L_OBJC_CATEGORY_INSTANCE_METHODS_" + ExtName.str(),
Daniel Dunbarc8ef5512008-08-23 00:19:03 +00002057 "__OBJC,__cat_inst_meth,regular,no_dead_strip",
Daniel Dunbarc45ef602008-08-26 21:51:14 +00002058 InstanceMethods);
Daniel Dunbar6bff2512009-08-03 17:06:42 +00002059 Values[3] =
Daniel Dunbar9c29bf52009-10-18 20:48:59 +00002060 EmitMethodList("\01L_OBJC_CATEGORY_CLASS_METHODS_" + ExtName.str(),
Daniel Dunbar0bf21992009-04-15 02:56:18 +00002061 "__OBJC,__cat_cls_meth,regular,no_dead_strip",
Daniel Dunbarc45ef602008-08-26 21:51:14 +00002062 ClassMethods);
Daniel Dunbarae226fa2008-08-27 02:31:56 +00002063 if (Category) {
Daniel Dunbar6bff2512009-08-03 17:06:42 +00002064 Values[4] =
Daniel Dunbar9c29bf52009-10-18 20:48:59 +00002065 EmitProtocolList("\01L_OBJC_CATEGORY_PROTOCOLS_" + ExtName.str(),
Daniel Dunbarae226fa2008-08-27 02:31:56 +00002066 Category->protocol_begin(),
2067 Category->protocol_end());
2068 } else {
Owen Andersonc9c88b42009-07-31 20:28:54 +00002069 Values[4] = llvm::Constant::getNullValue(ObjCTypes.ProtocolListPtrTy);
Daniel Dunbarae226fa2008-08-27 02:31:56 +00002070 }
Owen Anderson4a28d5d2009-07-24 23:12:58 +00002071 Values[5] = llvm::ConstantInt::get(ObjCTypes.IntTy, Size);
Daniel Dunbar86e2f402008-08-26 23:03:11 +00002072
2073 // If there is no category @interface then there can be no properties.
2074 if (Category) {
Daniel Dunbar9c29bf52009-10-18 20:48:59 +00002075 Values[6] = EmitPropertyList("\01l_OBJC_$_PROP_LIST_" + ExtName.str(),
Fariborz Jahanian5de14dc2009-01-28 22:18:42 +00002076 OCD, Category, ObjCTypes);
Daniel Dunbar86e2f402008-08-26 23:03:11 +00002077 } else {
Owen Andersonc9c88b42009-07-31 20:28:54 +00002078 Values[6] = llvm::Constant::getNullValue(ObjCTypes.PropertyListPtrTy);
Daniel Dunbar86e2f402008-08-26 23:03:11 +00002079 }
Daniel Dunbar6bff2512009-08-03 17:06:42 +00002080
Owen Anderson08e25242009-07-27 22:29:56 +00002081 llvm::Constant *Init = llvm::ConstantStruct::get(ObjCTypes.CategoryTy,
Daniel Dunbar86e253a2008-08-22 20:34:54 +00002082 Values);
2083
Daniel Dunbar6bff2512009-08-03 17:06:42 +00002084 llvm::GlobalVariable *GV =
Daniel Dunbar9c29bf52009-10-18 20:48:59 +00002085 CreateMetadataVar("\01L_OBJC_CATEGORY_" + ExtName.str(), Init,
Daniel Dunbar63c5b502009-03-09 21:49:58 +00002086 "__OBJC,__category,regular,no_dead_strip",
Daniel Dunbar58a29122009-03-09 22:18:41 +00002087 4, true);
Daniel Dunbar86e253a2008-08-22 20:34:54 +00002088 DefinedCategories.push_back(GV);
Fariborz Jahanianb9c5b3d2010-06-21 22:05:18 +00002089 DefinedCategoryNames.insert(ExtName.str());
Daniel Dunbarc17a4d32008-08-11 02:45:11 +00002090}
2091
Daniel Dunbar27f9d772008-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 Dunbar27f9d772008-08-21 04:36:09 +00002103/*
2104 struct _objc_class {
Daniel Dunbar6bff2512009-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 Dunbar27f9d772008-08-21 04:36:09 +00002118 };
2119
2120 See EmitClassExtension();
Daniel Dunbar6bff2512009-08-03 17:06:42 +00002121*/
Daniel Dunbar27f9d772008-08-21 04:36:09 +00002122void CGObjCMac::GenerateClass(const ObjCImplementationDecl *ID) {
Daniel Dunbar242d4dc2008-08-25 06:02:07 +00002123 DefinedSymbols.insert(ID->getIdentifier());
2124
Chris Lattner8ec03f52008-11-24 03:54:41 +00002125 std::string ClassName = ID->getNameAsString();
Daniel Dunbar27f9d772008-08-21 04:36:09 +00002126 // FIXME: Gross
Daniel Dunbar6bff2512009-08-03 17:06:42 +00002127 ObjCInterfaceDecl *Interface =
Daniel Dunbar27f9d772008-08-21 04:36:09 +00002128 const_cast<ObjCInterfaceDecl*>(ID->getClassInterface());
Daniel Dunbar6bff2512009-08-03 17:06:42 +00002129 llvm::Constant *Protocols =
Daniel Dunbar9c29bf52009-10-18 20:48:59 +00002130 EmitProtocolList("\01L_OBJC_CLASS_PROTOCOLS_" + ID->getName(),
Daniel Dunbardbc933702008-08-21 21:57:41 +00002131 Interface->protocol_begin(),
2132 Interface->protocol_end());
Daniel Dunbar27f9d772008-08-21 04:36:09 +00002133 unsigned Flags = eClassFlags_Factory;
Fariborz Jahanian109dfc62010-04-28 21:28:56 +00002134 if (ID->getNumIvarInitializers())
2135 Flags |= eClassFlags_HasCXXStructors;
Daniel Dunbar6bff2512009-08-03 17:06:42 +00002136 unsigned Size =
Daniel Dunbar2bebbf02009-05-03 10:46:44 +00002137 CGM.getContext().getASTObjCImplementationLayout(ID).getSize() / 8;
Daniel Dunbar27f9d772008-08-21 04:36:09 +00002138
2139 // FIXME: Set CXX-structors flag.
Daniel Dunbar04d40782009-04-14 06:00:08 +00002140 if (CGM.getDeclVisibilityMode(ID->getClassInterface()) == LangOptions::Hidden)
Daniel Dunbar27f9d772008-08-21 04:36:09 +00002141 Flags |= eClassFlags_Hidden;
2142
Daniel Dunbarc45ef602008-08-26 21:51:14 +00002143 std::vector<llvm::Constant*> InstanceMethods, ClassMethods;
Daniel Dunbar6bff2512009-08-03 17:06:42 +00002144 for (ObjCImplementationDecl::instmeth_iterator
Argyrios Kyrtzidis17945a02009-06-30 02:36:12 +00002145 i = ID->instmeth_begin(), e = ID->instmeth_end(); i != e; ++i) {
Daniel Dunbarc45ef602008-08-26 21:51:14 +00002146 // Instance methods should always be defined.
2147 InstanceMethods.push_back(GetMethodConstant(*i));
2148 }
Daniel Dunbar6bff2512009-08-03 17:06:42 +00002149 for (ObjCImplementationDecl::classmeth_iterator
Argyrios Kyrtzidis17945a02009-06-30 02:36:12 +00002150 i = ID->classmeth_begin(), e = ID->classmeth_end(); i != e; ++i) {
Daniel Dunbarc45ef602008-08-26 21:51:14 +00002151 // Class methods should always be defined.
2152 ClassMethods.push_back(GetMethodConstant(*i));
2153 }
2154
Daniel Dunbar6bff2512009-08-03 17:06:42 +00002155 for (ObjCImplementationDecl::propimpl_iterator
Argyrios Kyrtzidis17945a02009-06-30 02:36:12 +00002156 i = ID->propimpl_begin(), e = ID->propimpl_end(); i != e; ++i) {
Daniel Dunbarc45ef602008-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 Dunbar27f9d772008-08-21 04:36:09 +00002171 std::vector<llvm::Constant*> Values(12);
Daniel Dunbar5384b092009-05-03 08:56:52 +00002172 Values[ 0] = EmitMetaClass(ID, Protocols, ClassMethods);
Daniel Dunbar27f9d772008-08-21 04:36:09 +00002173 if (ObjCInterfaceDecl *Super = Interface->getSuperClass()) {
Daniel Dunbar242d4dc2008-08-25 06:02:07 +00002174 // Record a reference to the super class.
2175 LazySymbols.insert(Super->getIdentifier());
2176
Daniel Dunbar6bff2512009-08-03 17:06:42 +00002177 Values[ 1] =
Owen Anderson3c4972d2009-07-29 18:54:39 +00002178 llvm::ConstantExpr::getBitCast(GetClassName(Super->getIdentifier()),
Daniel Dunbar27f9d772008-08-21 04:36:09 +00002179 ObjCTypes.ClassPtrTy);
2180 } else {
Owen Andersonc9c88b42009-07-31 20:28:54 +00002181 Values[ 1] = llvm::Constant::getNullValue(ObjCTypes.ClassPtrTy);
Daniel Dunbar27f9d772008-08-21 04:36:09 +00002182 }
2183 Values[ 2] = GetClassName(ID->getIdentifier());
2184 // Version is always 0.
Owen Anderson4a28d5d2009-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 Jahanian46b86c62009-01-28 19:12:34 +00002188 Values[ 6] = EmitIvarList(ID, false);
Daniel Dunbar6bff2512009-08-03 17:06:42 +00002189 Values[ 7] =
Daniel Dunbar9c29bf52009-10-18 20:48:59 +00002190 EmitMethodList("\01L_OBJC_INSTANCE_METHODS_" + ID->getName(),
Daniel Dunbarc8ef5512008-08-23 00:19:03 +00002191 "__OBJC,__inst_meth,regular,no_dead_strip",
Daniel Dunbarc45ef602008-08-26 21:51:14 +00002192 InstanceMethods);
Daniel Dunbar27f9d772008-08-21 04:36:09 +00002193 // cache is always NULL.
Owen Andersonc9c88b42009-07-31 20:28:54 +00002194 Values[ 8] = llvm::Constant::getNullValue(ObjCTypes.CachePtrTy);
Daniel Dunbar27f9d772008-08-21 04:36:09 +00002195 Values[ 9] = Protocols;
Daniel Dunbar6bff2512009-08-03 17:06:42 +00002196 Values[10] = BuildIvarLayout(ID, true);
Daniel Dunbar27f9d772008-08-21 04:36:09 +00002197 Values[11] = EmitClassExtension(ID);
Owen Anderson08e25242009-07-27 22:29:56 +00002198 llvm::Constant *Init = llvm::ConstantStruct::get(ObjCTypes.ClassTy,
Daniel Dunbar27f9d772008-08-21 04:36:09 +00002199 Values);
Fariborz Jahanianb0069ee2009-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 Dunbar27f9d772008-08-21 04:36:09 +00002216 DefinedClasses.push_back(GV);
2217}
2218
2219llvm::Constant *CGObjCMac::EmitMetaClass(const ObjCImplementationDecl *ID,
2220 llvm::Constant *Protocols,
Daniel Dunbarae226fa2008-08-27 02:31:56 +00002221 const ConstantVector &Methods) {
Daniel Dunbar27f9d772008-08-21 04:36:09 +00002222 unsigned Flags = eClassFlags_Meta;
Duncan Sands9408c452009-05-09 07:08:47 +00002223 unsigned Size = CGM.getTargetData().getTypeAllocSize(ObjCTypes.ClassTy);
Daniel Dunbar27f9d772008-08-21 04:36:09 +00002224
Daniel Dunbar04d40782009-04-14 06:00:08 +00002225 if (CGM.getDeclVisibilityMode(ID->getClassInterface()) == LangOptions::Hidden)
Daniel Dunbar27f9d772008-08-21 04:36:09 +00002226 Flags |= eClassFlags_Hidden;
Daniel Dunbar6bff2512009-08-03 17:06:42 +00002227
Daniel Dunbar27f9d772008-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 Dunbar6bff2512009-08-03 17:06:42 +00002233 Values[ 0] =
Owen Anderson3c4972d2009-07-29 18:54:39 +00002234 llvm::ConstantExpr::getBitCast(GetClassName(Root->getIdentifier()),
Daniel Dunbar27f9d772008-08-21 04:36:09 +00002235 ObjCTypes.ClassPtrTy);
Daniel Dunbar86e253a2008-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 Dunbar27f9d772008-08-21 04:36:09 +00002239 if (ObjCInterfaceDecl *Super = ID->getClassInterface()->getSuperClass()) {
Daniel Dunbar6bff2512009-08-03 17:06:42 +00002240 Values[ 1] =
Owen Anderson3c4972d2009-07-29 18:54:39 +00002241 llvm::ConstantExpr::getBitCast(GetClassName(Super->getIdentifier()),
Daniel Dunbar27f9d772008-08-21 04:36:09 +00002242 ObjCTypes.ClassPtrTy);
2243 } else {
Owen Andersonc9c88b42009-07-31 20:28:54 +00002244 Values[ 1] = llvm::Constant::getNullValue(ObjCTypes.ClassPtrTy);
Daniel Dunbar27f9d772008-08-21 04:36:09 +00002245 }
2246 Values[ 2] = GetClassName(ID->getIdentifier());
2247 // Version is always 0.
Owen Anderson4a28d5d2009-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 Jahanian46b86c62009-01-28 19:12:34 +00002251 Values[ 6] = EmitIvarList(ID, true);
Daniel Dunbar6bff2512009-08-03 17:06:42 +00002252 Values[ 7] =
Chris Lattnerd9d22dd2008-11-24 05:29:24 +00002253 EmitMethodList("\01L_OBJC_CLASS_METHODS_" + ID->getNameAsString(),
Daniel Dunbar0bf21992009-04-15 02:56:18 +00002254 "__OBJC,__cls_meth,regular,no_dead_strip",
Daniel Dunbarc45ef602008-08-26 21:51:14 +00002255 Methods);
Daniel Dunbar27f9d772008-08-21 04:36:09 +00002256 // cache is always NULL.
Owen Andersonc9c88b42009-07-31 20:28:54 +00002257 Values[ 8] = llvm::Constant::getNullValue(ObjCTypes.CachePtrTy);
Daniel Dunbar27f9d772008-08-21 04:36:09 +00002258 Values[ 9] = Protocols;
2259 // ivar_layout for metaclass is always NULL.
Owen Andersonc9c88b42009-07-31 20:28:54 +00002260 Values[10] = llvm::Constant::getNullValue(ObjCTypes.Int8PtrTy);
Daniel Dunbar27f9d772008-08-21 04:36:09 +00002261 // The class extension is always unused for metaclasses.
Owen Andersonc9c88b42009-07-31 20:28:54 +00002262 Values[11] = llvm::Constant::getNullValue(ObjCTypes.ClassExtensionPtrTy);
Owen Anderson08e25242009-07-27 22:29:56 +00002263 llvm::Constant *Init = llvm::ConstantStruct::get(ObjCTypes.ClassTy,
Daniel Dunbar27f9d772008-08-21 04:36:09 +00002264 Values);
2265
Daniel Dunbarf56f1912008-08-25 08:19:24 +00002266 std::string Name("\01L_OBJC_METACLASS_");
Chris Lattner8ec03f52008-11-24 03:54:41 +00002267 Name += ID->getNameAsCString();
Daniel Dunbarf56f1912008-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 Anderson1c431b32009-07-08 19:05:04 +00002277 GV = new llvm::GlobalVariable(CGM.getModule(), ObjCTypes.ClassTy, false,
Daniel Dunbarf56f1912008-08-25 08:19:24 +00002278 llvm::GlobalValue::InternalLinkage,
Owen Anderson1c431b32009-07-08 19:05:04 +00002279 Init, Name);
Daniel Dunbarf56f1912008-08-25 08:19:24 +00002280 }
Daniel Dunbar27f9d772008-08-21 04:36:09 +00002281 GV->setSection("__OBJC,__meta_class,regular,no_dead_strip");
Daniel Dunbar58a29122009-03-09 22:18:41 +00002282 GV->setAlignment(4);
Chris Lattnerad64e022009-07-17 23:57:13 +00002283 CGM.AddUsedGlobal(GV);
Daniel Dunbar27f9d772008-08-21 04:36:09 +00002284
2285 return GV;
2286}
2287
Daniel Dunbar6bff2512009-08-03 17:06:42 +00002288llvm::Constant *CGObjCMac::EmitMetaClassRef(const ObjCInterfaceDecl *ID) {
Chris Lattnerd9d22dd2008-11-24 05:29:24 +00002289 std::string Name = "\01L_OBJC_METACLASS_" + ID->getNameAsString();
Daniel Dunbarf56f1912008-08-25 08:19:24 +00002290
Mike Stumpf5408fe2009-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 Dunbarf56f1912008-08-25 08:19:24 +00002295
2296 // Check for an existing forward reference.
Fariborz Jahanianb0d27942009-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 Dunbar6bff2512009-08-03 17:06:42 +00002299 if (llvm::GlobalVariable *GV = CGM.getModule().getGlobalVariable(Name,
2300 true)) {
Daniel Dunbarf56f1912008-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 Anderson1c431b32009-07-08 19:05:04 +00002307 return new llvm::GlobalVariable(CGM.getModule(), ObjCTypes.ClassTy, false,
Daniel Dunbarf56f1912008-08-25 08:19:24 +00002308 llvm::GlobalValue::ExternalLinkage,
2309 0,
Owen Anderson1c431b32009-07-08 19:05:04 +00002310 Name);
Daniel Dunbarf56f1912008-08-25 08:19:24 +00002311 }
2312}
2313
Fariborz Jahanianb0069ee2009-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 Dunbar27f9d772008-08-21 04:36:09 +00002330/*
2331 struct objc_class_ext {
Daniel Dunbar6bff2512009-08-03 17:06:42 +00002332 uint32_t size;
2333 const char *weak_ivar_layout;
2334 struct _objc_property_list *properties;
Daniel Dunbar27f9d772008-08-21 04:36:09 +00002335 };
2336*/
2337llvm::Constant *
2338CGObjCMac::EmitClassExtension(const ObjCImplementationDecl *ID) {
Daniel Dunbar6bff2512009-08-03 17:06:42 +00002339 uint64_t Size =
Duncan Sands9408c452009-05-09 07:08:47 +00002340 CGM.getTargetData().getTypeAllocSize(ObjCTypes.ClassExtensionTy);
Daniel Dunbar27f9d772008-08-21 04:36:09 +00002341
2342 std::vector<llvm::Constant*> Values(3);
Owen Anderson4a28d5d2009-07-24 23:12:58 +00002343 Values[0] = llvm::ConstantInt::get(ObjCTypes.IntTy, Size);
Fariborz Jahanianc71303d2009-04-22 23:00:43 +00002344 Values[1] = BuildIvarLayout(ID, false);
Daniel Dunbar9c29bf52009-10-18 20:48:59 +00002345 Values[2] = EmitPropertyList("\01l_OBJC_$_PROP_LIST_" + ID->getName(),
Fariborz Jahanian5de14dc2009-01-28 22:18:42 +00002346 ID, ID->getClassInterface(), ObjCTypes);
Daniel Dunbar27f9d772008-08-21 04:36:09 +00002347
2348 // Return null if no extension bits are used.
2349 if (Values[1]->isNullValue() && Values[2]->isNullValue())
Owen Andersonc9c88b42009-07-31 20:28:54 +00002350 return llvm::Constant::getNullValue(ObjCTypes.ClassExtensionPtrTy);
Daniel Dunbar27f9d772008-08-21 04:36:09 +00002351
Daniel Dunbar6bff2512009-08-03 17:06:42 +00002352 llvm::Constant *Init =
Owen Anderson08e25242009-07-27 22:29:56 +00002353 llvm::ConstantStruct::get(ObjCTypes.ClassExtensionTy, Values);
Daniel Dunbar9c29bf52009-10-18 20:48:59 +00002354 return CreateMetadataVar("\01L_OBJC_CLASSEXT_" + ID->getName(),
Daniel Dunbar6bff2512009-08-03 17:06:42 +00002355 Init, "__OBJC,__class_ext,regular,no_dead_strip",
Daniel Dunbar0bf21992009-04-15 02:56:18 +00002356 4, true);
Daniel Dunbar27f9d772008-08-21 04:36:09 +00002357}
2358
2359/*
2360 struct objc_ivar {
Daniel Dunbar6bff2512009-08-03 17:06:42 +00002361 char *ivar_name;
2362 char *ivar_type;
2363 int ivar_offset;
Daniel Dunbar27f9d772008-08-21 04:36:09 +00002364 };
2365
2366 struct objc_ivar_list {
Daniel Dunbar6bff2512009-08-03 17:06:42 +00002367 int ivar_count;
2368 struct objc_ivar list[count];
Daniel Dunbar27f9d772008-08-21 04:36:09 +00002369 };
Daniel Dunbar6bff2512009-08-03 17:06:42 +00002370*/
Daniel Dunbar27f9d772008-08-21 04:36:09 +00002371llvm::Constant *CGObjCMac::EmitIvarList(const ObjCImplementationDecl *ID,
Fariborz Jahanian46b86c62009-01-28 19:12:34 +00002372 bool ForClass) {
Daniel Dunbar27f9d772008-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 Andersonc9c88b42009-07-31 20:28:54 +00002381 return llvm::Constant::getNullValue(ObjCTypes.IvarListPtrTy);
Daniel Dunbar6bff2512009-08-03 17:06:42 +00002382
2383 ObjCInterfaceDecl *OID =
Fariborz Jahanian46b86c62009-01-28 19:12:34 +00002384 const_cast<ObjCInterfaceDecl*>(ID->getClassInterface());
Daniel Dunbar6bff2512009-08-03 17:06:42 +00002385
Daniel Dunbar3e5f0d82009-04-20 06:54:31 +00002386 llvm::SmallVector<ObjCIvarDecl*, 16> OIvars;
Fariborz Jahanian8e6ac1d2009-06-04 01:19:09 +00002387 CGM.getContext().ShallowCollectObjCIvars(OID, OIvars);
Daniel Dunbar6bff2512009-08-03 17:06:42 +00002388
Daniel Dunbar3e5f0d82009-04-20 06:54:31 +00002389 for (unsigned i = 0, e = OIvars.size(); i != e; ++i) {
2390 ObjCIvarDecl *IVD = OIvars[i];
Fariborz Jahanian8e6ac1d2009-06-04 01:19:09 +00002391 // Ignore unnamed bit-fields.
2392 if (!IVD->getDeclName())
Daniel Dunbar6bff2512009-08-03 17:06:42 +00002393 continue;
Daniel Dunbar3fea0c02009-04-22 08:22:17 +00002394 Ivar[0] = GetMethodVarName(IVD->getIdentifier());
2395 Ivar[1] = GetMethodVarType(IVD);
Daniel Dunbar6bff2512009-08-03 17:06:42 +00002396 Ivar[2] = llvm::ConstantInt::get(ObjCTypes.IntTy,
Daniel Dunbar97776872009-04-22 07:32:20 +00002397 ComputeIvarBaseOffset(CGM, OID, IVD));
Owen Anderson08e25242009-07-27 22:29:56 +00002398 Ivars.push_back(llvm::ConstantStruct::get(ObjCTypes.IvarTy, Ivar));
Daniel Dunbar27f9d772008-08-21 04:36:09 +00002399 }
2400
2401 // Return null for empty list.
2402 if (Ivars.empty())
Owen Andersonc9c88b42009-07-31 20:28:54 +00002403 return llvm::Constant::getNullValue(ObjCTypes.IvarListPtrTy);
Daniel Dunbar27f9d772008-08-21 04:36:09 +00002404
2405 std::vector<llvm::Constant*> Values(2);
Owen Anderson4a28d5d2009-07-24 23:12:58 +00002406 Values[0] = llvm::ConstantInt::get(ObjCTypes.IntTy, Ivars.size());
Owen Anderson96e0fc72009-07-29 22:16:19 +00002407 llvm::ArrayType *AT = llvm::ArrayType::get(ObjCTypes.IvarTy,
Daniel Dunbar27f9d772008-08-21 04:36:09 +00002408 Ivars.size());
Owen Anderson7db6d832009-07-28 18:33:04 +00002409 Values[1] = llvm::ConstantArray::get(AT, Ivars);
Nick Lewycky0d36dd22009-09-19 20:00:52 +00002410 llvm::Constant *Init = llvm::ConstantStruct::get(VMContext, Values, false);
Daniel Dunbar27f9d772008-08-21 04:36:09 +00002411
Daniel Dunbar63c5b502009-03-09 21:49:58 +00002412 llvm::GlobalVariable *GV;
2413 if (ForClass)
Daniel Dunbar9c29bf52009-10-18 20:48:59 +00002414 GV = CreateMetadataVar("\01L_OBJC_CLASS_VARIABLES_" + ID->getName(),
Daniel Dunbar6bff2512009-08-03 17:06:42 +00002415 Init, "__OBJC,__class_vars,regular,no_dead_strip",
Daniel Dunbar58a29122009-03-09 22:18:41 +00002416 4, true);
Daniel Dunbar63c5b502009-03-09 21:49:58 +00002417 else
Daniel Dunbar9c29bf52009-10-18 20:48:59 +00002418 GV = CreateMetadataVar("\01L_OBJC_INSTANCE_VARIABLES_" + ID->getName(),
Daniel Dunbar63c5b502009-03-09 21:49:58 +00002419 Init, "__OBJC,__instance_vars,regular,no_dead_strip",
Daniel Dunbar0bf21992009-04-15 02:56:18 +00002420 4, true);
Owen Anderson3c4972d2009-07-29 18:54:39 +00002421 return llvm::ConstantExpr::getBitCast(GV, ObjCTypes.IvarListPtrTy);
Daniel Dunbar27f9d772008-08-21 04:36:09 +00002422}
2423
2424/*
2425 struct objc_method {
Daniel Dunbar6bff2512009-08-03 17:06:42 +00002426 SEL method_name;
2427 char *method_types;
2428 void *method;
Daniel Dunbar27f9d772008-08-21 04:36:09 +00002429 };
Daniel Dunbar6bff2512009-08-03 17:06:42 +00002430
Daniel Dunbar27f9d772008-08-21 04:36:09 +00002431 struct objc_method_list {
Daniel Dunbar6bff2512009-08-03 17:06:42 +00002432 struct objc_method_list *obsolete;
2433 int count;
2434 struct objc_method methods_list[count];
Daniel Dunbar27f9d772008-08-21 04:36:09 +00002435 };
2436*/
Daniel Dunbarc45ef602008-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 Dunbarae226fa2008-08-27 02:31:56 +00002441llvm::Constant *CGObjCMac::GetMethodConstant(const ObjCMethodDecl *MD) {
Daniel Dunbarc45ef602008-08-26 21:51:14 +00002442 // FIXME: Use DenseMap::lookup
2443 llvm::Function *Fn = MethodDefinitions[MD];
2444 if (!Fn)
2445 return 0;
Daniel Dunbar6bff2512009-08-03 17:06:42 +00002446
Daniel Dunbarc45ef602008-08-26 21:51:14 +00002447 std::vector<llvm::Constant*> Method(3);
Daniel Dunbar6bff2512009-08-03 17:06:42 +00002448 Method[0] =
Owen Anderson3c4972d2009-07-29 18:54:39 +00002449 llvm::ConstantExpr::getBitCast(GetMethodVarName(MD->getSelector()),
Daniel Dunbarc45ef602008-08-26 21:51:14 +00002450 ObjCTypes.SelectorPtrTy);
2451 Method[1] = GetMethodVarType(MD);
Owen Anderson3c4972d2009-07-29 18:54:39 +00002452 Method[2] = llvm::ConstantExpr::getBitCast(Fn, ObjCTypes.Int8PtrTy);
Owen Anderson08e25242009-07-27 22:29:56 +00002453 return llvm::ConstantStruct::get(ObjCTypes.MethodTy, Method);
Daniel Dunbarc45ef602008-08-26 21:51:14 +00002454}
2455
Daniel Dunbar9c29bf52009-10-18 20:48:59 +00002456llvm::Constant *CGObjCMac::EmitMethodList(llvm::Twine Name,
Daniel Dunbar86e253a2008-08-22 20:34:54 +00002457 const char *Section,
Daniel Dunbarae226fa2008-08-27 02:31:56 +00002458 const ConstantVector &Methods) {
Daniel Dunbar27f9d772008-08-21 04:36:09 +00002459 // Return null for empty list.
2460 if (Methods.empty())
Owen Andersonc9c88b42009-07-31 20:28:54 +00002461 return llvm::Constant::getNullValue(ObjCTypes.MethodListPtrTy);
Daniel Dunbar27f9d772008-08-21 04:36:09 +00002462
2463 std::vector<llvm::Constant*> Values(3);
Owen Andersonc9c88b42009-07-31 20:28:54 +00002464 Values[0] = llvm::Constant::getNullValue(ObjCTypes.Int8PtrTy);
Owen Anderson4a28d5d2009-07-24 23:12:58 +00002465 Values[1] = llvm::ConstantInt::get(ObjCTypes.IntTy, Methods.size());
Owen Anderson96e0fc72009-07-29 22:16:19 +00002466 llvm::ArrayType *AT = llvm::ArrayType::get(ObjCTypes.MethodTy,
Daniel Dunbar27f9d772008-08-21 04:36:09 +00002467 Methods.size());
Owen Anderson7db6d832009-07-28 18:33:04 +00002468 Values[2] = llvm::ConstantArray::get(AT, Methods);
Nick Lewycky0d36dd22009-09-19 20:00:52 +00002469 llvm::Constant *Init = llvm::ConstantStruct::get(VMContext, Values, false);
Daniel Dunbar27f9d772008-08-21 04:36:09 +00002470
Daniel Dunbar0bf21992009-04-15 02:56:18 +00002471 llvm::GlobalVariable *GV = CreateMetadataVar(Name, Init, Section, 4, true);
Owen Anderson3c4972d2009-07-29 18:54:39 +00002472 return llvm::ConstantExpr::getBitCast(GV,
Daniel Dunbar27f9d772008-08-21 04:36:09 +00002473 ObjCTypes.MethodListPtrTy);
Daniel Dunbarb7ec2462008-08-16 03:19:19 +00002474}
2475
Fariborz Jahanian493dab72009-01-26 21:38:32 +00002476llvm::Function *CGObjCCommonMac::GenerateMethod(const ObjCMethodDecl *OMD,
Daniel Dunbar6bff2512009-08-03 17:06:42 +00002477 const ObjCContainerDecl *CD) {
Daniel Dunbarc575ce72009-10-19 01:21:19 +00002478 llvm::SmallString<256> Name;
Fariborz Jahanian679a5022009-01-10 21:06:09 +00002479 GetNameForMethod(OMD, CD, Name);
Daniel Dunbarb7ec2462008-08-16 03:19:19 +00002480
Daniel Dunbar541b63b2009-02-02 23:23:47 +00002481 CodeGenTypes &Types = CGM.getTypes();
Daniel Dunbar45c25ba2008-09-10 04:01:49 +00002482 const llvm::FunctionType *MethodTy =
Daniel Dunbar541b63b2009-02-02 23:23:47 +00002483 Types.GetFunctionType(Types.getFunctionInfo(OMD), OMD->isVariadic());
Daniel Dunbar6bff2512009-08-03 17:06:42 +00002484 llvm::Function *Method =
Daniel Dunbar45c25ba2008-09-10 04:01:49 +00002485 llvm::Function::Create(MethodTy,
Daniel Dunbarb7ec2462008-08-16 03:19:19 +00002486 llvm::GlobalValue::InternalLinkage,
Daniel Dunbarc575ce72009-10-19 01:21:19 +00002487 Name.str(),
Daniel Dunbarb7ec2462008-08-16 03:19:19 +00002488 &CGM.getModule());
Daniel Dunbarc45ef602008-08-26 21:51:14 +00002489 MethodDefinitions.insert(std::make_pair(OMD, Method));
Daniel Dunbarb7ec2462008-08-16 03:19:19 +00002490
Daniel Dunbarb7ec2462008-08-16 03:19:19 +00002491 return Method;
Daniel Dunbarc17a4d32008-08-11 02:45:11 +00002492}
2493
Daniel Dunbarfd65d372009-03-09 20:09:19 +00002494llvm::GlobalVariable *
Daniel Dunbar9c29bf52009-10-18 20:48:59 +00002495CGObjCCommonMac::CreateMetadataVar(llvm::Twine Name,
Daniel Dunbarfd65d372009-03-09 20:09:19 +00002496 llvm::Constant *Init,
2497 const char *Section,
Daniel Dunbar35bd7632009-03-09 20:50:13 +00002498 unsigned Align,
2499 bool AddToUsed) {
Daniel Dunbarfd65d372009-03-09 20:09:19 +00002500 const llvm::Type *Ty = Init->getType();
Daniel Dunbar6bff2512009-08-03 17:06:42 +00002501 llvm::GlobalVariable *GV =
Owen Anderson1c431b32009-07-08 19:05:04 +00002502 new llvm::GlobalVariable(CGM.getModule(), Ty, false,
Chris Lattnerad64e022009-07-17 23:57:13 +00002503 llvm::GlobalValue::InternalLinkage, Init, Name);
Daniel Dunbarfd65d372009-03-09 20:09:19 +00002504 if (Section)
2505 GV->setSection(Section);
Daniel Dunbar35bd7632009-03-09 20:50:13 +00002506 if (Align)
2507 GV->setAlignment(Align);
2508 if (AddToUsed)
Chris Lattnerad64e022009-07-17 23:57:13 +00002509 CGM.AddUsedGlobal(GV);
Daniel Dunbarfd65d372009-03-09 20:09:19 +00002510 return GV;
2511}
2512
Daniel Dunbar6bff2512009-08-03 17:06:42 +00002513llvm::Function *CGObjCMac::ModuleInitFunction() {
Daniel Dunbarf77ac862008-08-11 21:35:06 +00002514 // Abuse this interface function as a place to finalize.
2515 FinishModule();
Daniel Dunbarc17a4d32008-08-11 02:45:11 +00002516 return NULL;
2517}
2518
Chris Lattner74391b42009-03-22 21:03:39 +00002519llvm::Constant *CGObjCMac::GetPropertyGetFunction() {
Chris Lattner72db6c32009-04-22 02:44:54 +00002520 return ObjCTypes.getGetPropertyFn();
Daniel Dunbar49f66022008-09-24 03:38:44 +00002521}
2522
Chris Lattner74391b42009-03-22 21:03:39 +00002523llvm::Constant *CGObjCMac::GetPropertySetFunction() {
Chris Lattner72db6c32009-04-22 02:44:54 +00002524 return ObjCTypes.getSetPropertyFn();
Daniel Dunbar49f66022008-09-24 03:38:44 +00002525}
2526
Fariborz Jahanian6cc59062010-04-12 18:18:10 +00002527llvm::Constant *CGObjCMac::GetCopyStructFunction() {
2528 return ObjCTypes.getCopyStructFn();
2529}
2530
Chris Lattner74391b42009-03-22 21:03:39 +00002531llvm::Constant *CGObjCMac::EnumerationMutationFunction() {
Chris Lattner72db6c32009-04-22 02:44:54 +00002532 return ObjCTypes.getEnumerationMutationFn();
Anders Carlsson2abd89c2008-08-31 04:05:03 +00002533}
2534
John McCallf1549f62010-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 Dunbar6bff2512009-08-03 17:06:42 +00002544/*
Daniel Dunbar18ccc772008-09-28 01:03:14 +00002545
Daniel Dunbar6bff2512009-08-03 17:06:42 +00002546 Objective-C setjmp-longjmp (sjlj) Exception Handling
2547 --
Daniel Dunbar18ccc772008-09-28 01:03:14 +00002548
John McCallf1549f62010-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 Dunbar6bff2512009-08-03 17:06:42 +00002581 The basic framework for a @try-catch-finally is as follows:
2582 {
Daniel Dunbar18ccc772008-09-28 01:03:14 +00002583 objc_exception_data d;
2584 id _rethrow = null;
Anders Carlsson190d00e2009-02-07 21:26:04 +00002585 bool _call_try_exit = true;
Daniel Dunbar6bff2512009-08-03 17:06:42 +00002586
Daniel Dunbar18ccc772008-09-28 01:03:14 +00002587 objc_exception_try_enter(&d);
2588 if (!setjmp(d.jmp_buf)) {
Daniel Dunbar6bff2512009-08-03 17:06:42 +00002589 ... try body ...
Daniel Dunbar18ccc772008-09-28 01:03:14 +00002590 } else {
Daniel Dunbar6bff2512009-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 Dunbar18ccc772008-09-28 01:03:14 +00002607 }
Daniel Dunbar898d5082008-09-30 01:06:03 +00002608 ... jump-through-finally to finally_end ...
Daniel Dunbar18ccc772008-09-28 01:03:14 +00002609
Daniel Dunbar6bff2512009-08-03 17:06:42 +00002610 finally:
Anders Carlsson190d00e2009-02-07 21:26:04 +00002611 if (_call_try_exit)
Daniel Dunbar6bff2512009-08-03 17:06:42 +00002612 objc_exception_try_exit(&d);
Anders Carlsson190d00e2009-02-07 21:26:04 +00002613
Daniel Dunbar18ccc772008-09-28 01:03:14 +00002614 ... finally block ....
Daniel Dunbar898d5082008-09-30 01:06:03 +00002615 ... dispatch to finally destination ...
2616
Daniel Dunbar6bff2512009-08-03 17:06:42 +00002617 finally_rethrow:
Daniel Dunbar898d5082008-09-30 01:06:03 +00002618 objc_exception_throw(_rethrow);
2619
Daniel Dunbar6bff2512009-08-03 17:06:42 +00002620 finally_end:
2621 }
Daniel Dunbar18ccc772008-09-28 01:03:14 +00002622
Daniel Dunbar6bff2512009-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 Dunbar18ccc772008-09-28 01:03:14 +00002627
Daniel Dunbar6bff2512009-08-03 17:06:42 +00002628 We specialize this framework for a few particular circumstances:
Daniel Dunbar18ccc772008-09-28 01:03:14 +00002629
Daniel Dunbar6bff2512009-08-03 17:06:42 +00002630 - If there are no catch blocks, then we avoid emitting the second
2631 exception handling context.
Daniel Dunbar18ccc772008-09-28 01:03:14 +00002632
Daniel Dunbar6bff2512009-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 Dunbar18ccc772008-09-28 01:03:14 +00002635
Daniel Dunbar6bff2512009-08-03 17:06:42 +00002636 - FIXME: If there is no @finally block we can do a few more
2637 simplifications.
Daniel Dunbar18ccc772008-09-28 01:03:14 +00002638
Daniel Dunbar6bff2512009-08-03 17:06:42 +00002639 Rethrows and Jumps-Through-Finally
2640 --
Daniel Dunbar18ccc772008-09-28 01:03:14 +00002641
John McCallf1549f62010-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 Dunbar18ccc772008-09-28 01:03:14 +00002644
John McCallf1549f62010-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 Dunbar898d5082008-09-30 01:06:03 +00002649
John McCallf1549f62010-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 Dunbar18ccc772008-09-28 01:03:14 +00002654*/
2655
Fariborz Jahanianbd71be42008-11-21 00:49:24 +00002656void CGObjCMac::EmitTryOrSynchronizedStmt(CodeGen::CodeGenFunction &CGF,
2657 const Stmt &S) {
2658 bool isTry = isa<ObjCAtTryStmt>(S);
John McCallf1549f62010-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 Dunbar6bff2512009-08-03 17:06:42 +00002669
Daniel Dunbar1c566672009-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 Dunbar6bff2512009-08-03 17:06:42 +00002676 SyncArg =
Daniel Dunbar1c566672009-02-24 01:43:46 +00002677 CGF.EmitScalarExpr(cast<ObjCAtSynchronizedStmt>(S).getSynchExpr());
2678 SyncArg = CGF.Builder.CreateBitCast(SyncArg, ObjCTypes.ObjectPtrTy);
John McCallf1549f62010-07-06 01:34:17 +00002679 CGF.Builder.CreateCall(ObjCTypes.getSyncEnterFn(), SyncArg)
2680 ->setDoesNotThrow();
Daniel Dunbar1c566672009-02-24 01:43:46 +00002681 }
Daniel Dunbar898d5082008-09-30 01:06:03 +00002682
Daniel Dunbar898d5082008-09-30 01:06:03 +00002683 // Allocate memory for the exception data and rethrow pointer.
Anders Carlsson80f25672008-09-09 17:59:25 +00002684 llvm::Value *ExceptionData = CGF.CreateTempAlloca(ObjCTypes.ExceptionDataTy,
2685 "exceptiondata.ptr");
Daniel Dunbar6bff2512009-08-03 17:06:42 +00002686 llvm::Value *RethrowPtr = CGF.CreateTempAlloca(ObjCTypes.ObjectPtrTy,
Daniel Dunbar0c0e7a62008-10-29 22:36:39 +00002687 "_rethrow");
John McCallf1549f62010-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 Carlsson190d00e2009-02-07 21:26:04 +00002695 "_call_try_exit");
Daniel Dunbar6bff2512009-08-03 17:06:42 +00002696 CGF.Builder.CreateStore(llvm::ConstantInt::getTrue(VMContext),
John McCallf1549f62010-07-06 01:34:17 +00002697 CallTryExitVar);
Daniel Dunbar6bff2512009-08-03 17:06:42 +00002698
John McCallf1549f62010-07-06 01:34:17 +00002699 // Push a normal cleanup to leave the try scope.
2700 {
John McCallda65ea82010-07-13 20:32:21 +00002701 CodeGenFunction::CleanupBlock FinallyScope(CGF, NormalCleanup);
Daniel Dunbar898d5082008-09-30 01:06:03 +00002702
John McCallf1549f62010-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 Dunbar55e87422008-11-11 02:29:29 +00002750 llvm::BasicBlock *TryBlock = CGF.createBasicBlock("try");
2751 llvm::BasicBlock *TryHandler = CGF.createBasicBlock("try.handler");
John McCallf1549f62010-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 Carlsson80f25672008-09-09 17:59:25 +00002755
John McCallf1549f62010-07-06 01:34:17 +00002756 // Emit the protected block.
Anders Carlsson80f25672008-09-09 17:59:25 +00002757 CGF.EmitBlock(TryBlock);
Daniel Dunbar6bff2512009-08-03 17:06:42 +00002758 CGF.EmitStmt(isTry ? cast<ObjCAtTryStmt>(S).getTryBody()
John McCallf1549f62010-07-06 01:34:17 +00002759 : cast<ObjCAtSynchronizedStmt>(S).getSynchBody());
Anders Carlssonf3a79a92009-02-09 20:38:58 +00002760 CGF.EmitBranchThroughCleanup(FinallyEnd);
Daniel Dunbar6bff2512009-08-03 17:06:42 +00002761
John McCallf1549f62010-07-06 01:34:17 +00002762 // Emit the exception handler block.
Daniel Dunbare4b5ee02008-09-27 23:30:04 +00002763 CGF.EmitBlock(TryHandler);
Daniel Dunbar55e40722008-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 McCallf1549f62010-07-06 01:34:17 +00002767 llvm::CallInst *Caught =
Chris Lattner34b02a12009-04-22 02:26:14 +00002768 CGF.Builder.CreateCall(ObjCTypes.getExceptionExtractFn(),
2769 ExceptionData, "caught");
John McCallf1549f62010-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 Dunbar6bff2512009-08-03 17:06:42 +00002783 CGF.Builder.CreateStore(llvm::ConstantInt::getFalse(VMContext),
John McCallf1549f62010-07-06 01:34:17 +00002784 CallTryExitVar);
2785
Anders Carlssonf3a79a92009-02-09 20:38:58 +00002786 CGF.EmitBranchThroughCleanup(FinallyRethrow);
John McCallf1549f62010-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 Gregor8f5e3dd2010-04-23 22:50:49 +00002794 const ObjCAtTryStmt* AtTryStmt = cast<ObjCAtTryStmt>(&S);
2795
Daniel Dunbar55e40722008-09-27 07:03:52 +00002796 // Enter a new exception try block (in case a @catch block throws
John McCallf1549f62010-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 Dunbar6bff2512009-08-03 17:06:42 +00002801
John McCallf1549f62010-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 Carlsson80f25672008-09-09 17:59:25 +00002809
Daniel Dunbar55e87422008-11-11 02:29:29 +00002810 llvm::BasicBlock *CatchBlock = CGF.createBasicBlock("catch");
John McCallf1549f62010-07-06 01:34:17 +00002811 llvm::BasicBlock *CatchHandler = CGF.createBasicBlock("catch_for_catch");
Daniel Dunbare4b5ee02008-09-27 23:30:04 +00002812 CGF.Builder.CreateCondBr(Threw, CatchHandler, CatchBlock);
Daniel Dunbar6bff2512009-08-03 17:06:42 +00002813
Anders Carlsson80f25672008-09-09 17:59:25 +00002814 CGF.EmitBlock(CatchBlock);
Daniel Dunbar6bff2512009-08-03 17:06:42 +00002815
Daniel Dunbar55e40722008-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 Gregor8f5e3dd2010-04-23 22:50:49 +00002820 for (unsigned I = 0, N = AtTryStmt->getNumCatchStmts(); I != N; ++I) {
2821 const ObjCAtCatchStmt *CatchStmt = AtTryStmt->getCatchStmt(I);
Anders Carlsson80f25672008-09-09 17:59:25 +00002822
Douglas Gregorc00d8e12010-04-26 16:46:50 +00002823 const VarDecl *CatchParam = CatchStmt->getCatchParamDecl();
Steve Naroff14108da2009-07-10 23:34:53 +00002824 const ObjCObjectPointerType *OPT = 0;
Daniel Dunbar129271a2008-09-27 07:36:24 +00002825
Anders Carlsson80f25672008-09-09 17:59:25 +00002826 // catch(...) always matches.
Daniel Dunbar55e40722008-09-27 07:03:52 +00002827 if (!CatchParam) {
2828 AllMatched = true;
2829 } else {
John McCall183700f2009-09-21 23:43:11 +00002830 OPT = CatchParam->getType()->getAs<ObjCObjectPointerType>();
Daniel Dunbar6bff2512009-08-03 17:06:42 +00002831
John McCallf1549f62010-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 Dunbar97f61d12008-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 Friedman818e96f2009-07-11 00:57:02 +00002836 if (OPT && (OPT->isObjCIdType() || OPT->isObjCQualifiedIdType()))
Daniel Dunbar55e40722008-09-27 07:03:52 +00002837 AllMatched = true;
Anders Carlsson80f25672008-09-09 17:59:25 +00002838 }
Daniel Dunbar6bff2512009-08-03 17:06:42 +00002839
John McCallf1549f62010-07-06 01:34:17 +00002840 // If this is a catch-all, we don't need to test anything.
Daniel Dunbar6bff2512009-08-03 17:06:42 +00002841 if (AllMatched) {
John McCallf1549f62010-07-06 01:34:17 +00002842 CodeGenFunction::RunCleanupsScope CatchVarCleanups(CGF);
2843
Anders Carlssondde0a942008-09-11 09:15:33 +00002844 if (CatchParam) {
Steve Naroff7ba138a2009-03-03 19:52:17 +00002845 CGF.EmitLocalBlockVarDecl(*CatchParam);
Daniel Dunbara448fb22008-11-11 23:11:34 +00002846 assert(CGF.HaveInsertPoint() && "DeclStmt destroyed insert point?");
John McCallf1549f62010-07-06 01:34:17 +00002847
2848 // These types work out because ConvertType(id) == i8*.
Steve Naroff7ba138a2009-03-03 19:52:17 +00002849 CGF.Builder.CreateStore(Caught, CGF.GetAddrOfLocalVar(CatchParam));
Anders Carlssondde0a942008-09-11 09:15:33 +00002850 }
Daniel Dunbar6bff2512009-08-03 17:06:42 +00002851
Anders Carlssondde0a942008-09-11 09:15:33 +00002852 CGF.EmitStmt(CatchStmt->getCatchBody());
John McCallf1549f62010-07-06 01:34:17 +00002853
2854 // The scope of the catch variable ends right here.
2855 CatchVarCleanups.ForceCleanup();
2856
Anders Carlssonf3a79a92009-02-09 20:38:58 +00002857 CGF.EmitBranchThroughCleanup(FinallyEnd);
Anders Carlsson80f25672008-09-09 17:59:25 +00002858 break;
2859 }
Daniel Dunbar6bff2512009-08-03 17:06:42 +00002860
Steve Naroff14108da2009-07-10 23:34:53 +00002861 assert(OPT && "Unexpected non-object pointer type in @catch");
John McCall506b57e2010-05-17 21:00:27 +00002862 const ObjCObjectType *ObjTy = OPT->getObjectType();
John McCallf1549f62010-07-06 01:34:17 +00002863
2864 // FIXME: @catch (Class c) ?
John McCall506b57e2010-05-17 21:00:27 +00002865 ObjCInterfaceDecl *IDecl = ObjTy->getInterface();
2866 assert(IDecl && "Catch parameter must have Objective-C type!");
Anders Carlsson80f25672008-09-09 17:59:25 +00002867
2868 // Check if the @catch block matches the exception object.
John McCall506b57e2010-05-17 21:00:27 +00002869 llvm::Value *Class = EmitClassRef(CGF.Builder, IDecl);
Daniel Dunbar6bff2512009-08-03 17:06:42 +00002870
John McCallf1549f62010-07-06 01:34:17 +00002871 llvm::CallInst *Match =
Chris Lattner34b02a12009-04-22 02:26:14 +00002872 CGF.Builder.CreateCall2(ObjCTypes.getExceptionMatchFn(),
2873 Class, Caught, "match");
John McCallf1549f62010-07-06 01:34:17 +00002874 Match->setDoesNotThrow();
Daniel Dunbar6bff2512009-08-03 17:06:42 +00002875
John McCallf1549f62010-07-06 01:34:17 +00002876 llvm::BasicBlock *MatchedBlock = CGF.createBasicBlock("match");
2877 llvm::BasicBlock *NextCatchBlock = CGF.createBasicBlock("catch.next");
Daniel Dunbar6bff2512009-08-03 17:06:42 +00002878
2879 CGF.Builder.CreateCondBr(CGF.Builder.CreateIsNotNull(Match, "matched"),
Daniel Dunbare4b5ee02008-09-27 23:30:04 +00002880 MatchedBlock, NextCatchBlock);
Daniel Dunbar6bff2512009-08-03 17:06:42 +00002881
Anders Carlsson80f25672008-09-09 17:59:25 +00002882 // Emit the @catch block.
2883 CGF.EmitBlock(MatchedBlock);
John McCallf1549f62010-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 Naroff7ba138a2009-03-03 19:52:17 +00002889 CGF.EmitLocalBlockVarDecl(*CatchParam);
Daniel Dunbara448fb22008-11-11 23:11:34 +00002890 assert(CGF.HaveInsertPoint() && "DeclStmt destroyed insert point?");
Daniel Dunbar18ccc772008-09-28 01:03:14 +00002891
John McCallf1549f62010-07-06 01:34:17 +00002892 // Initialize the catch variable.
Daniel Dunbar6bff2512009-08-03 17:06:42 +00002893 llvm::Value *Tmp =
2894 CGF.Builder.CreateBitCast(Caught,
2895 CGF.ConvertType(CatchParam->getType()),
Daniel Dunbar18ccc772008-09-28 01:03:14 +00002896 "tmp");
Steve Naroff7ba138a2009-03-03 19:52:17 +00002897 CGF.Builder.CreateStore(Tmp, CGF.GetAddrOfLocalVar(CatchParam));
Daniel Dunbar6bff2512009-08-03 17:06:42 +00002898
Anders Carlssondde0a942008-09-11 09:15:33 +00002899 CGF.EmitStmt(CatchStmt->getCatchBody());
John McCallf1549f62010-07-06 01:34:17 +00002900
2901 // We're done with the catch variable.
2902 CatchVarCleanups.ForceCleanup();
2903
Anders Carlssonf3a79a92009-02-09 20:38:58 +00002904 CGF.EmitBranchThroughCleanup(FinallyEnd);
Daniel Dunbar6bff2512009-08-03 17:06:42 +00002905
Anders Carlsson80f25672008-09-09 17:59:25 +00002906 CGF.EmitBlock(NextCatchBlock);
2907 }
2908
John McCallf1549f62010-07-06 01:34:17 +00002909 CGF.ObjCEHValueStack.pop_back();
2910
Daniel Dunbar55e40722008-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 Carlssonf3a79a92009-02-09 20:38:58 +00002915 CGF.EmitBranchThroughCleanup(FinallyRethrow);
Daniel Dunbar55e40722008-09-27 07:03:52 +00002916 }
Daniel Dunbar6bff2512009-08-03 17:06:42 +00002917
Daniel Dunbar55e40722008-09-27 07:03:52 +00002918 // Emit the exception handler for the @catch blocks.
Daniel Dunbar6bff2512009-08-03 17:06:42 +00002919 CGF.EmitBlock(CatchHandler);
John McCallf1549f62010-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 Carlsson80f25672008-09-09 17:59:25 +00002925 CGF.Builder.CreateStore(Caught, RethrowPtr);
John McCallf1549f62010-07-06 01:34:17 +00002926
2927 // Don't pop the catch handler; the throw already did.
Daniel Dunbar6bff2512009-08-03 17:06:42 +00002928 CGF.Builder.CreateStore(llvm::ConstantInt::getFalse(VMContext),
John McCallf1549f62010-07-06 01:34:17 +00002929 CallTryExitVar);
Anders Carlssonf3a79a92009-02-09 20:38:58 +00002930 CGF.EmitBranchThroughCleanup(FinallyRethrow);
Anders Carlsson80f25672008-09-09 17:59:25 +00002931 }
Daniel Dunbar6bff2512009-08-03 17:06:42 +00002932
John McCallf1549f62010-07-06 01:34:17 +00002933 // Pop the cleanup.
2934 CGF.PopCleanupBlock();
2935 CGF.EmitBlock(FinallyEnd.Block);
Anders Carlssonf3a79a92009-02-09 20:38:58 +00002936
John McCallf1549f62010-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 Jahanianf2878e52008-11-21 19:21:53 +00002945 }
Anders Carlsson80f25672008-09-09 17:59:25 +00002946
John McCallf1549f62010-07-06 01:34:17 +00002947 CGF.Builder.SetInsertPoint(FinallyEnd.Block);
Anders Carlsson64d5d6c2008-09-09 10:04:29 +00002948}
2949
2950void CGObjCMac::EmitThrowStmt(CodeGen::CodeGenFunction &CGF,
Daniel Dunbar898d5082008-09-30 01:06:03 +00002951 const ObjCAtThrowStmt &S) {
Anders Carlsson2b1e3112008-09-09 16:16:55 +00002952 llvm::Value *ExceptionAsObject;
Daniel Dunbar6bff2512009-08-03 17:06:42 +00002953
Anders Carlsson2b1e3112008-09-09 16:16:55 +00002954 if (const Expr *ThrowExpr = S.getThrowExpr()) {
2955 llvm::Value *Exception = CGF.EmitScalarExpr(ThrowExpr);
Daniel Dunbar6bff2512009-08-03 17:06:42 +00002956 ExceptionAsObject =
Anders Carlsson2b1e3112008-09-09 16:16:55 +00002957 CGF.Builder.CreateBitCast(Exception, ObjCTypes.ObjectPtrTy, "tmp");
2958 } else {
Daniel Dunbar6bff2512009-08-03 17:06:42 +00002959 assert((!CGF.ObjCEHValueStack.empty() && CGF.ObjCEHValueStack.back()) &&
Daniel Dunbar18ccc772008-09-28 01:03:14 +00002960 "Unexpected rethrow outside @catch block.");
Anders Carlsson273558f2009-02-07 21:37:21 +00002961 ExceptionAsObject = CGF.ObjCEHValueStack.back();
Anders Carlsson2b1e3112008-09-09 16:16:55 +00002962 }
Daniel Dunbar6bff2512009-08-03 17:06:42 +00002963
John McCallf1549f62010-07-06 01:34:17 +00002964 CGF.Builder.CreateCall(ObjCTypes.getExceptionThrowFn(), ExceptionAsObject)
2965 ->setDoesNotReturn();
Anders Carlsson80f25672008-09-09 17:59:25 +00002966 CGF.Builder.CreateUnreachable();
Daniel Dunbara448fb22008-11-11 23:11:34 +00002967
2968 // Clear the insertion point to indicate we are in unreachable code.
2969 CGF.Builder.ClearInsertionPoint();
Anders Carlsson64d5d6c2008-09-09 10:04:29 +00002970}
2971
Fariborz Jahanian3e283e32008-11-18 22:37:34 +00002972/// EmitObjCWeakRead - Code gen for loading value of a __weak
Fariborz Jahanian6dc23172008-11-18 21:45:40 +00002973/// object: objc_read_weak (id *src)
2974///
Fariborz Jahanian3e283e32008-11-18 22:37:34 +00002975llvm::Value * CGObjCMac::EmitObjCWeakRead(CodeGen::CodeGenFunction &CGF,
Mike Stump1eb44332009-09-09 15:08:12 +00002976 llvm::Value *AddrWeakObj) {
Eli Friedman8339b352009-03-07 03:57:15 +00002977 const llvm::Type* DestTy =
Daniel Dunbar6bff2512009-08-03 17:06:42 +00002978 cast<llvm::PointerType>(AddrWeakObj->getType())->getElementType();
2979 AddrWeakObj = CGF.Builder.CreateBitCast(AddrWeakObj,
2980 ObjCTypes.PtrObjectPtrTy);
Chris Lattner72db6c32009-04-22 02:44:54 +00002981 llvm::Value *read_weak = CGF.Builder.CreateCall(ObjCTypes.getGcReadWeakFn(),
Fariborz Jahanian3e283e32008-11-18 22:37:34 +00002982 AddrWeakObj, "weakread");
Eli Friedman8339b352009-03-07 03:57:15 +00002983 read_weak = CGF.Builder.CreateBitCast(read_weak, DestTy);
Fariborz Jahanian6dc23172008-11-18 21:45:40 +00002984 return read_weak;
2985}
2986
Fariborz Jahanian3e283e32008-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 Stump1eb44332009-09-09 15:08:12 +00002991 llvm::Value *src, llvm::Value *dst) {
Fariborz Jahanian0a855d02009-03-23 19:10:40 +00002992 const llvm::Type * SrcTy = src->getType();
2993 if (!isa<llvm::PointerType>(SrcTy)) {
Duncan Sands9408c452009-05-09 07:08:47 +00002994 unsigned Size = CGM.getTargetData().getTypeAllocSize(SrcTy);
Fariborz Jahanian0a855d02009-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 Dunbar6bff2512009-08-03 17:06:42 +00002997 : CGF.Builder.CreateBitCast(src, ObjCTypes.LongLongTy);
Fariborz Jahanian3b8a6522009-03-13 00:42:52 +00002998 src = CGF.Builder.CreateIntToPtr(src, ObjCTypes.Int8PtrTy);
2999 }
Fariborz Jahaniandbd32c22008-11-19 17:34:06 +00003000 src = CGF.Builder.CreateBitCast(src, ObjCTypes.ObjectPtrTy);
3001 dst = CGF.Builder.CreateBitCast(dst, ObjCTypes.PtrObjectPtrTy);
Chris Lattner96508e12009-04-17 22:12:36 +00003002 CGF.Builder.CreateCall2(ObjCTypes.getGcAssignWeakFn(),
Fariborz Jahanian3e283e32008-11-18 22:37:34 +00003003 src, dst, "weakassign");
3004 return;
3005}
3006
Fariborz Jahanian58626502008-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,
Mike Stump1eb44332009-09-09 15:08:12 +00003011 llvm::Value *src, llvm::Value *dst) {
Fariborz Jahanian0a855d02009-03-23 19:10:40 +00003012 const llvm::Type * SrcTy = src->getType();
3013 if (!isa<llvm::PointerType>(SrcTy)) {
Duncan Sands9408c452009-05-09 07:08:47 +00003014 unsigned Size = CGM.getTargetData().getTypeAllocSize(SrcTy);
Fariborz Jahanian0a855d02009-03-23 19:10:40 +00003015 assert(Size <= 8 && "does not support size > 8");
3016 src = (Size == 4) ? CGF.Builder.CreateBitCast(src, ObjCTypes.IntTy)
Daniel Dunbar6bff2512009-08-03 17:06:42 +00003017 : CGF.Builder.CreateBitCast(src, ObjCTypes.LongLongTy);
Fariborz Jahanian3b8a6522009-03-13 00:42:52 +00003018 src = CGF.Builder.CreateIntToPtr(src, ObjCTypes.Int8PtrTy);
3019 }
Fariborz Jahaniandbd32c22008-11-19 17:34:06 +00003020 src = CGF.Builder.CreateBitCast(src, ObjCTypes.ObjectPtrTy);
3021 dst = CGF.Builder.CreateBitCast(dst, ObjCTypes.PtrObjectPtrTy);
Chris Lattnerbbccd612009-04-22 02:38:11 +00003022 CGF.Builder.CreateCall2(ObjCTypes.getGcAssignGlobalFn(),
Fariborz Jahanian58626502008-11-19 00:59:10 +00003023 src, dst, "globalassign");
3024 return;
3025}
3026
Fariborz Jahanian7eda8362008-11-20 19:23:36 +00003027/// EmitObjCIvarAssign - Code gen for assigning to a __strong object.
Fariborz Jahanian6c7a1f32009-09-24 22:25:38 +00003028/// objc_assign_ivar (id src, id *dst, ptrdiff_t ivaroffset)
Fariborz Jahanian7eda8362008-11-20 19:23:36 +00003029///
3030void CGObjCMac::EmitObjCIvarAssign(CodeGen::CodeGenFunction &CGF,
Fariborz Jahanian6c7a1f32009-09-24 22:25:38 +00003031 llvm::Value *src, llvm::Value *dst,
3032 llvm::Value *ivarOffset) {
3033 assert(ivarOffset && "EmitObjCIvarAssign - ivarOffset is NULL");
Fariborz Jahanian0a855d02009-03-23 19:10:40 +00003034 const llvm::Type * SrcTy = src->getType();
3035 if (!isa<llvm::PointerType>(SrcTy)) {
Duncan Sands9408c452009-05-09 07:08:47 +00003036 unsigned Size = CGM.getTargetData().getTypeAllocSize(SrcTy);
Fariborz Jahanian0a855d02009-03-23 19:10:40 +00003037 assert(Size <= 8 && "does not support size > 8");
3038 src = (Size == 4) ? CGF.Builder.CreateBitCast(src, ObjCTypes.IntTy)
Daniel Dunbar6bff2512009-08-03 17:06:42 +00003039 : CGF.Builder.CreateBitCast(src, ObjCTypes.LongLongTy);
Fariborz Jahanian3b8a6522009-03-13 00:42:52 +00003040 src = CGF.Builder.CreateIntToPtr(src, ObjCTypes.Int8PtrTy);
3041 }
Fariborz Jahanian7eda8362008-11-20 19:23:36 +00003042 src = CGF.Builder.CreateBitCast(src, ObjCTypes.ObjectPtrTy);
3043 dst = CGF.Builder.CreateBitCast(dst, ObjCTypes.PtrObjectPtrTy);
Fariborz Jahanian6c7a1f32009-09-24 22:25:38 +00003044 CGF.Builder.CreateCall3(ObjCTypes.getGcAssignIvarFn(),
3045 src, dst, ivarOffset);
Fariborz Jahanian7eda8362008-11-20 19:23:36 +00003046 return;
3047}
3048
Fariborz Jahanian58626502008-11-19 00:59:10 +00003049/// EmitObjCStrongCastAssign - Code gen for assigning to a __strong cast object.
3050/// objc_assign_strongCast (id src, id *dst)
3051///
3052void CGObjCMac::EmitObjCStrongCastAssign(CodeGen::CodeGenFunction &CGF,
Mike Stump1eb44332009-09-09 15:08:12 +00003053 llvm::Value *src, llvm::Value *dst) {
Fariborz Jahanian0a855d02009-03-23 19:10:40 +00003054 const llvm::Type * SrcTy = src->getType();
3055 if (!isa<llvm::PointerType>(SrcTy)) {
Duncan Sands9408c452009-05-09 07:08:47 +00003056 unsigned Size = CGM.getTargetData().getTypeAllocSize(SrcTy);
Fariborz Jahanian0a855d02009-03-23 19:10:40 +00003057 assert(Size <= 8 && "does not support size > 8");
3058 src = (Size == 4) ? CGF.Builder.CreateBitCast(src, ObjCTypes.IntTy)
Daniel Dunbar6bff2512009-08-03 17:06:42 +00003059 : CGF.Builder.CreateBitCast(src, ObjCTypes.LongLongTy);
Fariborz Jahanian3b8a6522009-03-13 00:42:52 +00003060 src = CGF.Builder.CreateIntToPtr(src, ObjCTypes.Int8PtrTy);
3061 }
Fariborz Jahaniandbd32c22008-11-19 17:34:06 +00003062 src = CGF.Builder.CreateBitCast(src, ObjCTypes.ObjectPtrTy);
3063 dst = CGF.Builder.CreateBitCast(dst, ObjCTypes.PtrObjectPtrTy);
Chris Lattnerbbccd612009-04-22 02:38:11 +00003064 CGF.Builder.CreateCall2(ObjCTypes.getGcAssignStrongCastFn(),
Fariborz Jahanian58626502008-11-19 00:59:10 +00003065 src, dst, "weakassign");
3066 return;
3067}
3068
Fariborz Jahanian082b02e2009-07-08 01:18:33 +00003069void CGObjCMac::EmitGCMemmoveCollectable(CodeGen::CodeGenFunction &CGF,
Daniel Dunbar6bff2512009-08-03 17:06:42 +00003070 llvm::Value *DestPtr,
3071 llvm::Value *SrcPtr,
Fariborz Jahanian55bcace2010-06-15 22:44:06 +00003072 llvm::Value *size) {
Fariborz Jahanian082b02e2009-07-08 01:18:33 +00003073 SrcPtr = CGF.Builder.CreateBitCast(SrcPtr, ObjCTypes.Int8PtrTy);
3074 DestPtr = CGF.Builder.CreateBitCast(DestPtr, ObjCTypes.Int8PtrTy);
Fariborz Jahanian082b02e2009-07-08 01:18:33 +00003075 CGF.Builder.CreateCall3(ObjCTypes.GcMemmoveCollectableFn(),
Fariborz Jahanian55bcace2010-06-15 22:44:06 +00003076 DestPtr, SrcPtr, size);
Fariborz Jahanian082b02e2009-07-08 01:18:33 +00003077 return;
3078}
3079
Fariborz Jahanian0bb20362009-02-02 20:02:29 +00003080/// EmitObjCValueForIvar - Code Gen for ivar reference.
3081///
Fariborz Jahanian598d3f62009-02-03 19:03:09 +00003082LValue CGObjCMac::EmitObjCValueForIvar(CodeGen::CodeGenFunction &CGF,
3083 QualType ObjectTy,
3084 llvm::Value *BaseValue,
3085 const ObjCIvarDecl *Ivar,
Fariborz Jahanian598d3f62009-02-03 19:03:09 +00003086 unsigned CVRQualifiers) {
John McCallc12c5bb2010-05-15 11:32:37 +00003087 const ObjCInterfaceDecl *ID =
3088 ObjectTy->getAs<ObjCObjectType>()->getInterface();
Daniel Dunbar97776872009-04-22 07:32:20 +00003089 return EmitValueForIvarAtOffset(CGF, ID, BaseValue, Ivar, CVRQualifiers,
3090 EmitIvarOffset(CGF, ID, Ivar));
Fariborz Jahanian0bb20362009-02-02 20:02:29 +00003091}
3092
Fariborz Jahanianf63aa3f2009-02-10 19:02:04 +00003093llvm::Value *CGObjCMac::EmitIvarOffset(CodeGen::CodeGenFunction &CGF,
Daniel Dunbar2a031922009-04-22 05:08:15 +00003094 const ObjCInterfaceDecl *Interface,
Fariborz Jahanianf63aa3f2009-02-10 19:02:04 +00003095 const ObjCIvarDecl *Ivar) {
Daniel Dunbar97776872009-04-22 07:32:20 +00003096 uint64_t Offset = ComputeIvarBaseOffset(CGM, Interface, Ivar);
Owen Anderson4a28d5d2009-07-24 23:12:58 +00003097 return llvm::ConstantInt::get(
Daniel Dunbar6bff2512009-08-03 17:06:42 +00003098 CGM.getTypes().ConvertType(CGM.getContext().LongTy),
3099 Offset);
Fariborz Jahanianf63aa3f2009-02-10 19:02:04 +00003100}
3101
Daniel Dunbarf77ac862008-08-11 21:35:06 +00003102/* *** Private Interface *** */
3103
3104/// EmitImageInfo - Emit the image info marker used to encode some module
3105/// level information.
3106///
3107/// See: <rdr://4810609&4810587&4810587>
3108/// struct IMAGE_INFO {
3109/// unsigned version;
3110/// unsigned flags;
3111/// };
3112enum ImageInfoFlags {
Daniel Dunbarfce176b2010-04-25 20:39:01 +00003113 eImageInfo_FixAndContinue = (1 << 0),
Daniel Dunbar6bff2512009-08-03 17:06:42 +00003114 eImageInfo_GarbageCollected = (1 << 1),
3115 eImageInfo_GCOnly = (1 << 2),
Daniel Dunbarc7c6dc02009-04-20 07:11:47 +00003116 eImageInfo_OptimizedByDyld = (1 << 3), // FIXME: When is this set.
3117
Daniel Dunbarfce176b2010-04-25 20:39:01 +00003118 // A flag indicating that the module has no instances of a @synthesize of a
3119 // superclass variable. <rdar://problem/6803242>
Daniel Dunbarc7c6dc02009-04-20 07:11:47 +00003120 eImageInfo_CorrectedSynthesize = (1 << 4)
Daniel Dunbarf77ac862008-08-11 21:35:06 +00003121};
3122
Daniel Dunbarfce176b2010-04-25 20:39:01 +00003123void CGObjCCommonMac::EmitImageInfo() {
Daniel Dunbarf77ac862008-08-11 21:35:06 +00003124 unsigned version = 0; // Version is unused?
3125 unsigned flags = 0;
3126
3127 // FIXME: Fix and continue?
3128 if (CGM.getLangOptions().getGCMode() != LangOptions::NonGC)
3129 flags |= eImageInfo_GarbageCollected;
3130 if (CGM.getLangOptions().getGCMode() == LangOptions::GCOnly)
3131 flags |= eImageInfo_GCOnly;
Daniel Dunbar6bff2512009-08-03 17:06:42 +00003132
Daniel Dunbarc7c6dc02009-04-20 07:11:47 +00003133 // We never allow @synthesize of a superclass property.
3134 flags |= eImageInfo_CorrectedSynthesize;
Daniel Dunbarf77ac862008-08-11 21:35:06 +00003135
Chris Lattner77b89b82010-06-27 07:15:29 +00003136 const llvm::Type *Int32Ty = llvm::Type::getInt32Ty(VMContext);
3137
Daniel Dunbarf77ac862008-08-11 21:35:06 +00003138 // Emitted as int[2];
3139 llvm::Constant *values[2] = {
Chris Lattner77b89b82010-06-27 07:15:29 +00003140 llvm::ConstantInt::get(Int32Ty, version),
3141 llvm::ConstantInt::get(Int32Ty, flags)
Daniel Dunbarf77ac862008-08-11 21:35:06 +00003142 };
Chris Lattner77b89b82010-06-27 07:15:29 +00003143 llvm::ArrayType *AT = llvm::ArrayType::get(Int32Ty, 2);
Daniel Dunbar63c5b502009-03-09 21:49:58 +00003144
3145 const char *Section;
3146 if (ObjCABI == 1)
3147 Section = "__OBJC, __image_info,regular";
3148 else
3149 Section = "__DATA, __objc_imageinfo, regular, no_dead_strip";
Daniel Dunbar6bff2512009-08-03 17:06:42 +00003150 llvm::GlobalVariable *GV =
Daniel Dunbar63c5b502009-03-09 21:49:58 +00003151 CreateMetadataVar("\01L_OBJC_IMAGE_INFO",
Owen Anderson7db6d832009-07-28 18:33:04 +00003152 llvm::ConstantArray::get(AT, values, 2),
Daniel Dunbar63c5b502009-03-09 21:49:58 +00003153 Section,
3154 0,
3155 true);
3156 GV->setConstant(true);
Daniel Dunbarf77ac862008-08-11 21:35:06 +00003157}
3158
Daniel Dunbar4e2d7d02008-08-12 06:48:42 +00003159
3160// struct objc_module {
3161// unsigned long version;
3162// unsigned long size;
3163// const char *name;
3164// Symtab symtab;
3165// };
3166
3167// FIXME: Get from somewhere
3168static const int ModuleVersion = 7;
3169
3170void CGObjCMac::EmitModuleInfo() {
Duncan Sands9408c452009-05-09 07:08:47 +00003171 uint64_t Size = CGM.getTargetData().getTypeAllocSize(ObjCTypes.ModuleTy);
Daniel Dunbar6bff2512009-08-03 17:06:42 +00003172
Daniel Dunbar4e2d7d02008-08-12 06:48:42 +00003173 std::vector<llvm::Constant*> Values(4);
Owen Anderson4a28d5d2009-07-24 23:12:58 +00003174 Values[0] = llvm::ConstantInt::get(ObjCTypes.LongTy, ModuleVersion);
3175 Values[1] = llvm::ConstantInt::get(ObjCTypes.LongTy, Size);
Daniel Dunbar7ded7f42008-08-15 22:20:32 +00003176 // This used to be the filename, now it is unused. <rdr://4327263>
Daniel Dunbar6efc0c52008-08-13 03:21:16 +00003177 Values[2] = GetClassName(&CGM.getContext().Idents.get(""));
Daniel Dunbar4e2d7d02008-08-12 06:48:42 +00003178 Values[3] = EmitModuleSymbols();
Daniel Dunbar6bff2512009-08-03 17:06:42 +00003179 CreateMetadataVar("\01L_OBJC_MODULES",
Owen Anderson08e25242009-07-27 22:29:56 +00003180 llvm::ConstantStruct::get(ObjCTypes.ModuleTy, Values),
Daniel Dunbar63c5b502009-03-09 21:49:58 +00003181 "__OBJC,__module_info,regular,no_dead_strip",
Daniel Dunbar58a29122009-03-09 22:18:41 +00003182 4, true);
Daniel Dunbar4e2d7d02008-08-12 06:48:42 +00003183}
3184
3185llvm::Constant *CGObjCMac::EmitModuleSymbols() {
Daniel Dunbar27f9d772008-08-21 04:36:09 +00003186 unsigned NumClasses = DefinedClasses.size();
3187 unsigned NumCategories = DefinedCategories.size();
3188
Daniel Dunbar242d4dc2008-08-25 06:02:07 +00003189 // Return null if no symbols were defined.
3190 if (!NumClasses && !NumCategories)
Owen Andersonc9c88b42009-07-31 20:28:54 +00003191 return llvm::Constant::getNullValue(ObjCTypes.SymtabPtrTy);
Daniel Dunbar242d4dc2008-08-25 06:02:07 +00003192
3193 std::vector<llvm::Constant*> Values(5);
Owen Anderson4a28d5d2009-07-24 23:12:58 +00003194 Values[0] = llvm::ConstantInt::get(ObjCTypes.LongTy, 0);
Owen Andersonc9c88b42009-07-31 20:28:54 +00003195 Values[1] = llvm::Constant::getNullValue(ObjCTypes.SelectorPtrTy);
Owen Anderson4a28d5d2009-07-24 23:12:58 +00003196 Values[2] = llvm::ConstantInt::get(ObjCTypes.ShortTy, NumClasses);
3197 Values[3] = llvm::ConstantInt::get(ObjCTypes.ShortTy, NumCategories);
Daniel Dunbar27f9d772008-08-21 04:36:09 +00003198
Daniel Dunbar86e253a2008-08-22 20:34:54 +00003199 // The runtime expects exactly the list of defined classes followed
3200 // by the list of defined categories, in a single array.
Daniel Dunbar27f9d772008-08-21 04:36:09 +00003201 std::vector<llvm::Constant*> Symbols(NumClasses + NumCategories);
Daniel Dunbar86e253a2008-08-22 20:34:54 +00003202 for (unsigned i=0; i<NumClasses; i++)
Owen Anderson3c4972d2009-07-29 18:54:39 +00003203 Symbols[i] = llvm::ConstantExpr::getBitCast(DefinedClasses[i],
Daniel Dunbar86e253a2008-08-22 20:34:54 +00003204 ObjCTypes.Int8PtrTy);
3205 for (unsigned i=0; i<NumCategories; i++)
Daniel Dunbar6bff2512009-08-03 17:06:42 +00003206 Symbols[NumClasses + i] =
Owen Anderson3c4972d2009-07-29 18:54:39 +00003207 llvm::ConstantExpr::getBitCast(DefinedCategories[i],
Daniel Dunbar86e253a2008-08-22 20:34:54 +00003208 ObjCTypes.Int8PtrTy);
3209
Daniel Dunbar6bff2512009-08-03 17:06:42 +00003210 Values[4] =
Owen Anderson96e0fc72009-07-29 22:16:19 +00003211 llvm::ConstantArray::get(llvm::ArrayType::get(ObjCTypes.Int8PtrTy,
Daniel Dunbar27f9d772008-08-21 04:36:09 +00003212 NumClasses + NumCategories),
3213 Symbols);
3214
Nick Lewycky0d36dd22009-09-19 20:00:52 +00003215 llvm::Constant *Init = llvm::ConstantStruct::get(VMContext, Values, false);
Daniel Dunbar27f9d772008-08-21 04:36:09 +00003216
Daniel Dunbar4e2d7d02008-08-12 06:48:42 +00003217 llvm::GlobalVariable *GV =
Daniel Dunbar63c5b502009-03-09 21:49:58 +00003218 CreateMetadataVar("\01L_OBJC_SYMBOLS", Init,
3219 "__OBJC,__symbols,regular,no_dead_strip",
Daniel Dunbar0bf21992009-04-15 02:56:18 +00003220 4, true);
Owen Anderson3c4972d2009-07-29 18:54:39 +00003221 return llvm::ConstantExpr::getBitCast(GV, ObjCTypes.SymtabPtrTy);
Daniel Dunbar27f9d772008-08-21 04:36:09 +00003222}
3223
Daniel Dunbar6bff2512009-08-03 17:06:42 +00003224llvm::Value *CGObjCMac::EmitClassRef(CGBuilderTy &Builder,
Daniel Dunbar27f9d772008-08-21 04:36:09 +00003225 const ObjCInterfaceDecl *ID) {
Daniel Dunbar242d4dc2008-08-25 06:02:07 +00003226 LazySymbols.insert(ID->getIdentifier());
3227
Daniel Dunbar27f9d772008-08-21 04:36:09 +00003228 llvm::GlobalVariable *&Entry = ClassReferences[ID->getIdentifier()];
Daniel Dunbar6bff2512009-08-03 17:06:42 +00003229
Daniel Dunbar27f9d772008-08-21 04:36:09 +00003230 if (!Entry) {
Daniel Dunbar6bff2512009-08-03 17:06:42 +00003231 llvm::Constant *Casted =
Owen Anderson3c4972d2009-07-29 18:54:39 +00003232 llvm::ConstantExpr::getBitCast(GetClassName(ID->getIdentifier()),
Daniel Dunbar27f9d772008-08-21 04:36:09 +00003233 ObjCTypes.ClassPtrTy);
Daniel Dunbar6bff2512009-08-03 17:06:42 +00003234 Entry =
Daniel Dunbar63c5b502009-03-09 21:49:58 +00003235 CreateMetadataVar("\01L_OBJC_CLASS_REFERENCES_", Casted,
3236 "__OBJC,__cls_refs,literal_pointers,no_dead_strip",
Daniel Dunbar0bf21992009-04-15 02:56:18 +00003237 4, true);
Daniel Dunbar27f9d772008-08-21 04:36:09 +00003238 }
3239
Daniel Dunbar2da84ff2009-11-29 21:23:36 +00003240 return Builder.CreateLoad(Entry, "tmp");
Daniel Dunbar4e2d7d02008-08-12 06:48:42 +00003241}
3242
Fariborz Jahanian03b29602010-06-17 19:56:20 +00003243llvm::Value *CGObjCMac::EmitSelector(CGBuilderTy &Builder, Selector Sel,
3244 bool lvalue) {
Daniel Dunbar259d93d2008-08-12 03:39:23 +00003245 llvm::GlobalVariable *&Entry = SelectorReferences[Sel];
Daniel Dunbar6bff2512009-08-03 17:06:42 +00003246
Daniel Dunbar259d93d2008-08-12 03:39:23 +00003247 if (!Entry) {
Daniel Dunbar6bff2512009-08-03 17:06:42 +00003248 llvm::Constant *Casted =
Owen Anderson3c4972d2009-07-29 18:54:39 +00003249 llvm::ConstantExpr::getBitCast(GetMethodVarName(Sel),
Daniel Dunbar259d93d2008-08-12 03:39:23 +00003250 ObjCTypes.SelectorPtrTy);
Daniel Dunbar6bff2512009-08-03 17:06:42 +00003251 Entry =
Daniel Dunbar63c5b502009-03-09 21:49:58 +00003252 CreateMetadataVar("\01L_OBJC_SELECTOR_REFERENCES_", Casted,
3253 "__OBJC,__message_refs,literal_pointers,no_dead_strip",
Daniel Dunbar0bf21992009-04-15 02:56:18 +00003254 4, true);
Daniel Dunbar259d93d2008-08-12 03:39:23 +00003255 }
3256
Fariborz Jahanian03b29602010-06-17 19:56:20 +00003257 if (lvalue)
3258 return Entry;
Daniel Dunbar2da84ff2009-11-29 21:23:36 +00003259 return Builder.CreateLoad(Entry, "tmp");
Daniel Dunbar259d93d2008-08-12 03:39:23 +00003260}
3261
Fariborz Jahanian058a1b72009-01-24 20:21:50 +00003262llvm::Constant *CGObjCCommonMac::GetClassName(IdentifierInfo *Ident) {
Daniel Dunbar6efc0c52008-08-13 03:21:16 +00003263 llvm::GlobalVariable *&Entry = ClassNames[Ident];
Daniel Dunbar4e2d7d02008-08-12 06:48:42 +00003264
Daniel Dunbar63c5b502009-03-09 21:49:58 +00003265 if (!Entry)
Daniel Dunbar6bff2512009-08-03 17:06:42 +00003266 Entry = CreateMetadataVar("\01L_OBJC_CLASS_NAME_",
Daniel Dunbar9c29bf52009-10-18 20:48:59 +00003267 llvm::ConstantArray::get(VMContext,
3268 Ident->getNameStart()),
Daniel Dunbar63c5b502009-03-09 21:49:58 +00003269 "__TEXT,__cstring,cstring_literals",
Daniel Dunbarb90bb002009-04-14 23:14:47 +00003270 1, true);
Daniel Dunbar4e2d7d02008-08-12 06:48:42 +00003271
Owen Andersona1cf15f2009-07-14 23:10:40 +00003272 return getConstantGEP(VMContext, Entry, 0, 0);
Daniel Dunbar4e2d7d02008-08-12 06:48:42 +00003273}
3274
Fariborz Jahaniand80d81b2009-03-05 19:17:31 +00003275/// GetIvarLayoutName - Returns a unique constant for the given
3276/// ivar layout bitmap.
3277llvm::Constant *CGObjCCommonMac::GetIvarLayoutName(IdentifierInfo *Ident,
Daniel Dunbar6bff2512009-08-03 17:06:42 +00003278 const ObjCCommonTypesHelper &ObjCTypes) {
Owen Andersonc9c88b42009-07-31 20:28:54 +00003279 return llvm::Constant::getNullValue(ObjCTypes.Int8PtrTy);
Fariborz Jahaniand80d81b2009-03-05 19:17:31 +00003280}
3281
John McCall0953e762009-09-24 19:53:00 +00003282static Qualifiers::GC GetGCAttrTypeForType(ASTContext &Ctx, QualType FQT) {
Daniel Dunbar5e563dd2009-05-03 13:55:09 +00003283 if (FQT.isObjCGCStrong())
John McCall0953e762009-09-24 19:53:00 +00003284 return Qualifiers::Strong;
Daniel Dunbar5e563dd2009-05-03 13:55:09 +00003285
3286 if (FQT.isObjCGCWeak())
John McCall0953e762009-09-24 19:53:00 +00003287 return Qualifiers::Weak;
Daniel Dunbar5e563dd2009-05-03 13:55:09 +00003288
Fariborz Jahanian039e6a12009-09-11 17:39:05 +00003289 if (FQT->isObjCObjectPointerType() || FQT->isBlockPointerType())
John McCall0953e762009-09-24 19:53:00 +00003290 return Qualifiers::Strong;
Daniel Dunbar5e563dd2009-05-03 13:55:09 +00003291
Ted Kremenek6217b802009-07-29 21:53:49 +00003292 if (const PointerType *PT = FQT->getAs<PointerType>())
Daniel Dunbard58edcb2009-05-03 14:10:34 +00003293 return GetGCAttrTypeForType(Ctx, PT->getPointeeType());
Daniel Dunbar5e563dd2009-05-03 13:55:09 +00003294
John McCall0953e762009-09-24 19:53:00 +00003295 return Qualifiers::GCNone;
Daniel Dunbar5e563dd2009-05-03 13:55:09 +00003296}
3297
Daniel Dunbard58edcb2009-05-03 14:10:34 +00003298void CGObjCCommonMac::BuildAggrIvarRecordLayout(const RecordType *RT,
Daniel Dunbar6bff2512009-08-03 17:06:42 +00003299 unsigned int BytePos,
Daniel Dunbard58edcb2009-05-03 14:10:34 +00003300 bool ForStrongLayout,
3301 bool &HasUnion) {
3302 const RecordDecl *RD = RT->getDecl();
3303 // FIXME - Use iterator.
Argyrios Kyrtzidis17945a02009-06-30 02:36:12 +00003304 llvm::SmallVector<FieldDecl*, 16> Fields(RD->field_begin(), RD->field_end());
Daniel Dunbard58edcb2009-05-03 14:10:34 +00003305 const llvm::Type *Ty = CGM.getTypes().ConvertType(QualType(RT, 0));
Daniel Dunbar6bff2512009-08-03 17:06:42 +00003306 const llvm::StructLayout *RecLayout =
Daniel Dunbard58edcb2009-05-03 14:10:34 +00003307 CGM.getTargetData().getStructLayout(cast<llvm::StructType>(Ty));
Daniel Dunbar6bff2512009-08-03 17:06:42 +00003308
Daniel Dunbard58edcb2009-05-03 14:10:34 +00003309 BuildAggrIvarLayout(0, RecLayout, RD, Fields, BytePos,
3310 ForStrongLayout, HasUnion);
3311}
3312
Daniel Dunbar5a5a8032009-05-03 21:05:10 +00003313void CGObjCCommonMac::BuildAggrIvarLayout(const ObjCImplementationDecl *OI,
Daniel Dunbar6bff2512009-08-03 17:06:42 +00003314 const llvm::StructLayout *Layout,
3315 const RecordDecl *RD,
Chris Lattnerf1690852009-03-31 08:48:01 +00003316 const llvm::SmallVectorImpl<FieldDecl*> &RecFields,
Daniel Dunbar6bff2512009-08-03 17:06:42 +00003317 unsigned int BytePos, bool ForStrongLayout,
3318 bool &HasUnion) {
Fariborz Jahanian820e0202009-03-11 00:07:04 +00003319 bool IsUnion = (RD && RD->isUnion());
3320 uint64_t MaxUnionIvarSize = 0;
3321 uint64_t MaxSkippedUnionIvarSize = 0;
3322 FieldDecl *MaxField = 0;
3323 FieldDecl *MaxSkippedField = 0;
Fariborz Jahanian7fb16272009-04-21 18:33:06 +00003324 FieldDecl *LastFieldBitfield = 0;
Daniel Dunbar900c1982009-05-03 23:31:46 +00003325 uint64_t MaxFieldOffset = 0;
3326 uint64_t MaxSkippedFieldOffset = 0;
3327 uint64_t LastBitfieldOffset = 0;
Daniel Dunbar6bff2512009-08-03 17:06:42 +00003328
Fariborz Jahaniana5a10c32009-03-10 16:22:08 +00003329 if (RecFields.empty())
3330 return;
Chris Lattnerf1690852009-03-31 08:48:01 +00003331 unsigned WordSizeInBits = CGM.getContext().Target.getPointerWidth(0);
3332 unsigned ByteSizeInBits = CGM.getContext().Target.getCharWidth();
3333
Chris Lattnerf1690852009-03-31 08:48:01 +00003334 for (unsigned i = 0, e = RecFields.size(); i != e; ++i) {
Fariborz Jahaniana5a10c32009-03-10 16:22:08 +00003335 FieldDecl *Field = RecFields[i];
Daniel Dunbare05cc982009-05-03 23:35:23 +00003336 uint64_t FieldOffset;
Anders Carlssonfc514ab2009-07-24 17:23:54 +00003337 if (RD) {
Daniel Dunbarf8f8eba2010-04-14 17:02:21 +00003338 // Note that 'i' here is actually the field index inside RD of Field,
3339 // although this dependency is hidden.
3340 const ASTRecordLayout &RL = CGM.getContext().getASTRecordLayout(RD);
3341 FieldOffset = RL.getFieldOffset(i) / 8;
Anders Carlssonfc514ab2009-07-24 17:23:54 +00003342 } else
Daniel Dunbare05cc982009-05-03 23:35:23 +00003343 FieldOffset = ComputeIvarBaseOffset(CGM, OI, cast<ObjCIvarDecl>(Field));
Daniel Dunbar25d583e2009-05-03 14:17:18 +00003344
Fariborz Jahaniana5a10c32009-03-10 16:22:08 +00003345 // Skip over unnamed or bitfields
Fariborz Jahanian7fb16272009-04-21 18:33:06 +00003346 if (!Field->getIdentifier() || Field->isBitField()) {
3347 LastFieldBitfield = Field;
Daniel Dunbar900c1982009-05-03 23:31:46 +00003348 LastBitfieldOffset = FieldOffset;
Fariborz Jahaniana5a10c32009-03-10 16:22:08 +00003349 continue;
Fariborz Jahanian7fb16272009-04-21 18:33:06 +00003350 }
Daniel Dunbar25d583e2009-05-03 14:17:18 +00003351
Fariborz Jahanian7fb16272009-04-21 18:33:06 +00003352 LastFieldBitfield = 0;
Fariborz Jahaniana5a10c32009-03-10 16:22:08 +00003353 QualType FQT = Field->getType();
Fariborz Jahanian667423a2009-03-25 22:36:49 +00003354 if (FQT->isRecordType() || FQT->isUnionType()) {
Fariborz Jahaniana5a10c32009-03-10 16:22:08 +00003355 if (FQT->isUnionType())
3356 HasUnion = true;
Fariborz Jahanian820e0202009-03-11 00:07:04 +00003357
Daniel Dunbar6bff2512009-08-03 17:06:42 +00003358 BuildAggrIvarRecordLayout(FQT->getAs<RecordType>(),
Daniel Dunbar25d583e2009-05-03 14:17:18 +00003359 BytePos + FieldOffset,
Daniel Dunbard58edcb2009-05-03 14:10:34 +00003360 ForStrongLayout, HasUnion);
Fariborz Jahaniana5a10c32009-03-10 16:22:08 +00003361 continue;
3362 }
Daniel Dunbar6bff2512009-08-03 17:06:42 +00003363
Chris Lattnerf1690852009-03-31 08:48:01 +00003364 if (const ArrayType *Array = CGM.getContext().getAsArrayType(FQT)) {
Daniel Dunbar6bff2512009-08-03 17:06:42 +00003365 const ConstantArrayType *CArray =
Daniel Dunbar5e563dd2009-05-03 13:55:09 +00003366 dyn_cast_or_null<ConstantArrayType>(Array);
Fariborz Jahanian7fb16272009-04-21 18:33:06 +00003367 uint64_t ElCount = CArray->getSize().getZExtValue();
Daniel Dunbar5e563dd2009-05-03 13:55:09 +00003368 assert(CArray && "only array with known element size is supported");
Fariborz Jahanian820e0202009-03-11 00:07:04 +00003369 FQT = CArray->getElementType();
Fariborz Jahanian667423a2009-03-25 22:36:49 +00003370 while (const ArrayType *Array = CGM.getContext().getAsArrayType(FQT)) {
3371 const ConstantArrayType *CArray =
Daniel Dunbar5e563dd2009-05-03 13:55:09 +00003372 dyn_cast_or_null<ConstantArrayType>(Array);
Fariborz Jahanian7fb16272009-04-21 18:33:06 +00003373 ElCount *= CArray->getSize().getZExtValue();
Fariborz Jahanian667423a2009-03-25 22:36:49 +00003374 FQT = CArray->getElementType();
3375 }
Daniel Dunbar6bff2512009-08-03 17:06:42 +00003376
3377 assert(!FQT->isUnionType() &&
Fariborz Jahanian820e0202009-03-11 00:07:04 +00003378 "layout for array of unions not supported");
3379 if (FQT->isRecordType()) {
Fariborz Jahanian81adc052009-04-24 16:17:09 +00003380 int OldIndex = IvarsInfo.size() - 1;
3381 int OldSkIndex = SkipIvars.size() -1;
Daniel Dunbar6bff2512009-08-03 17:06:42 +00003382
Ted Kremenek6217b802009-07-29 21:53:49 +00003383 const RecordType *RT = FQT->getAs<RecordType>();
Daniel Dunbar25d583e2009-05-03 14:17:18 +00003384 BuildAggrIvarRecordLayout(RT, BytePos + FieldOffset,
Daniel Dunbard58edcb2009-05-03 14:10:34 +00003385 ForStrongLayout, HasUnion);
Daniel Dunbar6bff2512009-08-03 17:06:42 +00003386
Fariborz Jahanian820e0202009-03-11 00:07:04 +00003387 // Replicate layout information for each array element. Note that
3388 // one element is already done.
3389 uint64_t ElIx = 1;
Daniel Dunbar6bff2512009-08-03 17:06:42 +00003390 for (int FirstIndex = IvarsInfo.size() - 1,
3391 FirstSkIndex = SkipIvars.size() - 1 ;ElIx < ElCount; ElIx++) {
Fariborz Jahanianc8ce9c82009-03-12 22:50:49 +00003392 uint64_t Size = CGM.getContext().getTypeSize(RT)/ByteSizeInBits;
Daniel Dunbar8b2926c2009-05-03 13:44:42 +00003393 for (int i = OldIndex+1; i <= FirstIndex; ++i)
3394 IvarsInfo.push_back(GC_IVAR(IvarsInfo[i].ivar_bytepos + Size*ElIx,
3395 IvarsInfo[i].ivar_size));
3396 for (int i = OldSkIndex+1; i <= FirstSkIndex; ++i)
3397 SkipIvars.push_back(GC_IVAR(SkipIvars[i].ivar_bytepos + Size*ElIx,
3398 SkipIvars[i].ivar_size));
Fariborz Jahanian820e0202009-03-11 00:07:04 +00003399 }
3400 continue;
3401 }
Fariborz Jahaniana5a10c32009-03-10 16:22:08 +00003402 }
Fariborz Jahanian820e0202009-03-11 00:07:04 +00003403 // At this point, we are done with Record/Union and array there of.
3404 // For other arrays we are down to its element type.
John McCall0953e762009-09-24 19:53:00 +00003405 Qualifiers::GC GCAttr = GetGCAttrTypeForType(CGM.getContext(), FQT);
Daniel Dunbar5e563dd2009-05-03 13:55:09 +00003406
Daniel Dunbar8b2926c2009-05-03 13:44:42 +00003407 unsigned FieldSize = CGM.getContext().getTypeSize(Field->getType());
John McCall0953e762009-09-24 19:53:00 +00003408 if ((ForStrongLayout && GCAttr == Qualifiers::Strong)
3409 || (!ForStrongLayout && GCAttr == Qualifiers::Weak)) {
Daniel Dunbar487993b2009-05-03 13:32:01 +00003410 if (IsUnion) {
Daniel Dunbar8b2926c2009-05-03 13:44:42 +00003411 uint64_t UnionIvarSize = FieldSize / WordSizeInBits;
Daniel Dunbar487993b2009-05-03 13:32:01 +00003412 if (UnionIvarSize > MaxUnionIvarSize) {
Fariborz Jahanian820e0202009-03-11 00:07:04 +00003413 MaxUnionIvarSize = UnionIvarSize;
3414 MaxField = Field;
Daniel Dunbar900c1982009-05-03 23:31:46 +00003415 MaxFieldOffset = FieldOffset;
Fariborz Jahanian820e0202009-03-11 00:07:04 +00003416 }
Daniel Dunbar487993b2009-05-03 13:32:01 +00003417 } else {
Daniel Dunbar25d583e2009-05-03 14:17:18 +00003418 IvarsInfo.push_back(GC_IVAR(BytePos + FieldOffset,
Daniel Dunbar8b2926c2009-05-03 13:44:42 +00003419 FieldSize / WordSizeInBits));
Fariborz Jahanian820e0202009-03-11 00:07:04 +00003420 }
Daniel Dunbar6bff2512009-08-03 17:06:42 +00003421 } else if ((ForStrongLayout &&
John McCall0953e762009-09-24 19:53:00 +00003422 (GCAttr == Qualifiers::GCNone || GCAttr == Qualifiers::Weak))
3423 || (!ForStrongLayout && GCAttr != Qualifiers::Weak)) {
Daniel Dunbar487993b2009-05-03 13:32:01 +00003424 if (IsUnion) {
Mike Stumpf5408fe2009-05-16 07:57:57 +00003425 // FIXME: Why the asymmetry? We divide by word size in bits on other
3426 // side.
Daniel Dunbar8b2926c2009-05-03 13:44:42 +00003427 uint64_t UnionIvarSize = FieldSize;
Daniel Dunbar487993b2009-05-03 13:32:01 +00003428 if (UnionIvarSize > MaxSkippedUnionIvarSize) {
Fariborz Jahanian820e0202009-03-11 00:07:04 +00003429 MaxSkippedUnionIvarSize = UnionIvarSize;
3430 MaxSkippedField = Field;
Daniel Dunbar900c1982009-05-03 23:31:46 +00003431 MaxSkippedFieldOffset = FieldOffset;
Fariborz Jahanian820e0202009-03-11 00:07:04 +00003432 }
Daniel Dunbar487993b2009-05-03 13:32:01 +00003433 } else {
Daniel Dunbar8b2926c2009-05-03 13:44:42 +00003434 // FIXME: Why the asymmetry, we divide by byte size in bits here?
Daniel Dunbar25d583e2009-05-03 14:17:18 +00003435 SkipIvars.push_back(GC_IVAR(BytePos + FieldOffset,
Daniel Dunbar8b2926c2009-05-03 13:44:42 +00003436 FieldSize / ByteSizeInBits));
Fariborz Jahanian820e0202009-03-11 00:07:04 +00003437 }
3438 }
3439 }
Daniel Dunbard58edcb2009-05-03 14:10:34 +00003440
Fariborz Jahanian7fb16272009-04-21 18:33:06 +00003441 if (LastFieldBitfield) {
3442 // Last field was a bitfield. Must update skip info.
Fariborz Jahanian7fb16272009-04-21 18:33:06 +00003443 Expr *BitWidth = LastFieldBitfield->getBitWidth();
3444 uint64_t BitFieldSize =
Eli Friedman9a901bb2009-04-26 19:19:15 +00003445 BitWidth->EvaluateAsInt(CGM.getContext()).getZExtValue();
Daniel Dunbar487993b2009-05-03 13:32:01 +00003446 GC_IVAR skivar;
Daniel Dunbar900c1982009-05-03 23:31:46 +00003447 skivar.ivar_bytepos = BytePos + LastBitfieldOffset;
Daniel Dunbar6bff2512009-08-03 17:06:42 +00003448 skivar.ivar_size = (BitFieldSize / ByteSizeInBits)
3449 + ((BitFieldSize % ByteSizeInBits) != 0);
3450 SkipIvars.push_back(skivar);
Fariborz Jahanian7fb16272009-04-21 18:33:06 +00003451 }
Daniel Dunbar6bff2512009-08-03 17:06:42 +00003452
Daniel Dunbar8b2926c2009-05-03 13:44:42 +00003453 if (MaxField)
Daniel Dunbar6bff2512009-08-03 17:06:42 +00003454 IvarsInfo.push_back(GC_IVAR(BytePos + MaxFieldOffset,
Daniel Dunbar8b2926c2009-05-03 13:44:42 +00003455 MaxUnionIvarSize));
3456 if (MaxSkippedField)
Daniel Dunbar900c1982009-05-03 23:31:46 +00003457 SkipIvars.push_back(GC_IVAR(BytePos + MaxSkippedFieldOffset,
Daniel Dunbar8b2926c2009-05-03 13:44:42 +00003458 MaxSkippedUnionIvarSize));
Fariborz Jahaniand61a50a2009-03-05 22:39:55 +00003459}
3460
3461/// BuildIvarLayout - Builds ivar layout bitmap for the class
3462/// implementation for the __strong or __weak case.
Daniel Dunbar6bff2512009-08-03 17:06:42 +00003463/// The layout map displays which words in ivar list must be skipped
3464/// and which must be scanned by GC (see below). String is built of bytes.
3465/// Each byte is divided up in two nibbles (4-bit each). Left nibble is count
Fariborz Jahaniand61a50a2009-03-05 22:39:55 +00003466/// of words to skip and right nibble is count of words to scan. So, each
Daniel Dunbar6bff2512009-08-03 17:06:42 +00003467/// nibble represents up to 15 workds to skip or scan. Skipping the rest is
Fariborz Jahaniand61a50a2009-03-05 22:39:55 +00003468/// represented by a 0x00 byte which also ends the string.
3469/// 1. when ForStrongLayout is true, following ivars are scanned:
3470/// - id, Class
3471/// - object *
3472/// - __strong anything
Daniel Dunbar6bff2512009-08-03 17:06:42 +00003473///
Fariborz Jahaniand61a50a2009-03-05 22:39:55 +00003474/// 2. When ForStrongLayout is false, following ivars are scanned:
3475/// - __weak anything
3476///
Fariborz Jahaniana5a10c32009-03-10 16:22:08 +00003477llvm::Constant *CGObjCCommonMac::BuildIvarLayout(
Daniel Dunbar6bff2512009-08-03 17:06:42 +00003478 const ObjCImplementationDecl *OMD,
3479 bool ForStrongLayout) {
Fariborz Jahaniand61a50a2009-03-05 22:39:55 +00003480 bool hasUnion = false;
Daniel Dunbar6bff2512009-08-03 17:06:42 +00003481
Fariborz Jahanian9397e1d2009-03-11 20:59:05 +00003482 unsigned int WordsToScan, WordsToSkip;
Benjamin Kramer3c0ef8c2009-10-13 10:07:13 +00003483 const llvm::Type *PtrTy = llvm::Type::getInt8PtrTy(VMContext);
Fariborz Jahanianc8ce9c82009-03-12 22:50:49 +00003484 if (CGM.getLangOptions().getGCMode() == LangOptions::NonGC)
Owen Andersonc9c88b42009-07-31 20:28:54 +00003485 return llvm::Constant::getNullValue(PtrTy);
Daniel Dunbar6bff2512009-08-03 17:06:42 +00003486
Chris Lattnerf1690852009-03-31 08:48:01 +00003487 llvm::SmallVector<FieldDecl*, 32> RecFields;
Fariborz Jahanianc8ce9c82009-03-12 22:50:49 +00003488 const ObjCInterfaceDecl *OI = OMD->getClassInterface();
Fariborz Jahaniand61a50a2009-03-05 22:39:55 +00003489 CGM.getContext().CollectObjCIvars(OI, RecFields);
Daniel Dunbar6bff2512009-08-03 17:06:42 +00003490
Daniel Dunbar37153282009-05-04 04:10:48 +00003491 // Add this implementations synthesized ivars.
Fariborz Jahanian98200742009-05-12 18:14:29 +00003492 llvm::SmallVector<ObjCIvarDecl*, 16> Ivars;
Fariborz Jahanian11062e12010-02-19 00:31:17 +00003493 CGM.getContext().CollectNonClassIvars(OI, Ivars);
Fariborz Jahanian98200742009-05-12 18:14:29 +00003494 for (unsigned k = 0, e = Ivars.size(); k != e; ++k)
3495 RecFields.push_back(cast<FieldDecl>(Ivars[k]));
Daniel Dunbar6bff2512009-08-03 17:06:42 +00003496
Fariborz Jahaniand61a50a2009-03-05 22:39:55 +00003497 if (RecFields.empty())
Owen Andersonc9c88b42009-07-31 20:28:54 +00003498 return llvm::Constant::getNullValue(PtrTy);
Daniel Dunbar6bff2512009-08-03 17:06:42 +00003499
3500 SkipIvars.clear();
Fariborz Jahanianc8ce9c82009-03-12 22:50:49 +00003501 IvarsInfo.clear();
Daniel Dunbar6bff2512009-08-03 17:06:42 +00003502
Daniel Dunbar5a5a8032009-05-03 21:05:10 +00003503 BuildAggrIvarLayout(OMD, 0, 0, RecFields, 0, ForStrongLayout, hasUnion);
Fariborz Jahanian81adc052009-04-24 16:17:09 +00003504 if (IvarsInfo.empty())
Owen Andersonc9c88b42009-07-31 20:28:54 +00003505 return llvm::Constant::getNullValue(PtrTy);
Daniel Dunbar6bff2512009-08-03 17:06:42 +00003506
Fariborz Jahanian9397e1d2009-03-11 20:59:05 +00003507 // Sort on byte position in case we encounterred a union nested in
3508 // the ivar list.
Fariborz Jahanianc8ce9c82009-03-12 22:50:49 +00003509 if (hasUnion && !IvarsInfo.empty())
Daniel Dunbar0941b492009-04-23 01:29:05 +00003510 std::sort(IvarsInfo.begin(), IvarsInfo.end());
Fariborz Jahanianc8ce9c82009-03-12 22:50:49 +00003511 if (hasUnion && !SkipIvars.empty())
Daniel Dunbar0941b492009-04-23 01:29:05 +00003512 std::sort(SkipIvars.begin(), SkipIvars.end());
Daniel Dunbar6bff2512009-08-03 17:06:42 +00003513
Fariborz Jahanian9397e1d2009-03-11 20:59:05 +00003514 // Build the string of skip/scan nibbles
Fariborz Jahanian8c2f2d12009-04-24 17:15:27 +00003515 llvm::SmallVector<SKIP_SCAN, 32> SkipScanIvars;
Daniel Dunbar6bff2512009-08-03 17:06:42 +00003516 unsigned int WordSize =
Duncan Sands9408c452009-05-09 07:08:47 +00003517 CGM.getTypes().getTargetData().getTypeAllocSize(PtrTy);
Fariborz Jahanian9397e1d2009-03-11 20:59:05 +00003518 if (IvarsInfo[0].ivar_bytepos == 0) {
3519 WordsToSkip = 0;
3520 WordsToScan = IvarsInfo[0].ivar_size;
Daniel Dunbar31682fd2009-05-03 23:21:22 +00003521 } else {
Fariborz Jahanian9397e1d2009-03-11 20:59:05 +00003522 WordsToSkip = IvarsInfo[0].ivar_bytepos/WordSize;
3523 WordsToScan = IvarsInfo[0].ivar_size;
3524 }
Daniel Dunbar31682fd2009-05-03 23:21:22 +00003525 for (unsigned int i=1, Last=IvarsInfo.size(); i != Last; i++) {
Daniel Dunbar6bff2512009-08-03 17:06:42 +00003526 unsigned int TailPrevGCObjC =
Fariborz Jahanian9397e1d2009-03-11 20:59:05 +00003527 IvarsInfo[i-1].ivar_bytepos + IvarsInfo[i-1].ivar_size * WordSize;
Daniel Dunbar31682fd2009-05-03 23:21:22 +00003528 if (IvarsInfo[i].ivar_bytepos == TailPrevGCObjC) {
Fariborz Jahanian9397e1d2009-03-11 20:59:05 +00003529 // consecutive 'scanned' object pointers.
3530 WordsToScan += IvarsInfo[i].ivar_size;
Daniel Dunbar31682fd2009-05-03 23:21:22 +00003531 } else {
Fariborz Jahanian9397e1d2009-03-11 20:59:05 +00003532 // Skip over 'gc'able object pointer which lay over each other.
3533 if (TailPrevGCObjC > IvarsInfo[i].ivar_bytepos)
3534 continue;
3535 // Must skip over 1 or more words. We save current skip/scan values
3536 // and start a new pair.
Fariborz Jahanianc8ce9c82009-03-12 22:50:49 +00003537 SKIP_SCAN SkScan;
3538 SkScan.skip = WordsToSkip;
3539 SkScan.scan = WordsToScan;
Fariborz Jahanian81adc052009-04-24 16:17:09 +00003540 SkipScanIvars.push_back(SkScan);
Daniel Dunbar6bff2512009-08-03 17:06:42 +00003541
Fariborz Jahanian9397e1d2009-03-11 20:59:05 +00003542 // Skip the hole.
Fariborz Jahanianc8ce9c82009-03-12 22:50:49 +00003543 SkScan.skip = (IvarsInfo[i].ivar_bytepos - TailPrevGCObjC) / WordSize;
3544 SkScan.scan = 0;
Fariborz Jahanian81adc052009-04-24 16:17:09 +00003545 SkipScanIvars.push_back(SkScan);
Fariborz Jahanian9397e1d2009-03-11 20:59:05 +00003546 WordsToSkip = 0;
3547 WordsToScan = IvarsInfo[i].ivar_size;
3548 }
3549 }
Daniel Dunbar31682fd2009-05-03 23:21:22 +00003550 if (WordsToScan > 0) {
Fariborz Jahanianc8ce9c82009-03-12 22:50:49 +00003551 SKIP_SCAN SkScan;
3552 SkScan.skip = WordsToSkip;
3553 SkScan.scan = WordsToScan;
Fariborz Jahanian81adc052009-04-24 16:17:09 +00003554 SkipScanIvars.push_back(SkScan);
Fariborz Jahanian9397e1d2009-03-11 20:59:05 +00003555 }
Daniel Dunbar6bff2512009-08-03 17:06:42 +00003556
Daniel Dunbar31682fd2009-05-03 23:21:22 +00003557 if (!SkipIvars.empty()) {
Fariborz Jahanian81adc052009-04-24 16:17:09 +00003558 unsigned int LastIndex = SkipIvars.size()-1;
Daniel Dunbar6bff2512009-08-03 17:06:42 +00003559 int LastByteSkipped =
3560 SkipIvars[LastIndex].ivar_bytepos + SkipIvars[LastIndex].ivar_size;
Fariborz Jahanian81adc052009-04-24 16:17:09 +00003561 LastIndex = IvarsInfo.size()-1;
Daniel Dunbar6bff2512009-08-03 17:06:42 +00003562 int LastByteScanned =
3563 IvarsInfo[LastIndex].ivar_bytepos +
3564 IvarsInfo[LastIndex].ivar_size * WordSize;
Fariborz Jahanian9397e1d2009-03-11 20:59:05 +00003565 // Compute number of bytes to skip at the tail end of the last ivar scanned.
Benjamin Kramer54d76db2009-12-25 15:43:36 +00003566 if (LastByteSkipped > LastByteScanned) {
Fariborz Jahanian9397e1d2009-03-11 20:59:05 +00003567 unsigned int TotalWords = (LastByteSkipped + (WordSize -1)) / WordSize;
Fariborz Jahanianc8ce9c82009-03-12 22:50:49 +00003568 SKIP_SCAN SkScan;
3569 SkScan.skip = TotalWords - (LastByteScanned/WordSize);
3570 SkScan.scan = 0;
Fariborz Jahanian81adc052009-04-24 16:17:09 +00003571 SkipScanIvars.push_back(SkScan);
Fariborz Jahanian9397e1d2009-03-11 20:59:05 +00003572 }
3573 }
3574 // Mini optimization of nibbles such that an 0xM0 followed by 0x0N is produced
3575 // as 0xMN.
Fariborz Jahanian81adc052009-04-24 16:17:09 +00003576 int SkipScan = SkipScanIvars.size()-1;
Daniel Dunbar31682fd2009-05-03 23:21:22 +00003577 for (int i = 0; i <= SkipScan; i++) {
Fariborz Jahanian9397e1d2009-03-11 20:59:05 +00003578 if ((i < SkipScan) && SkipScanIvars[i].skip && SkipScanIvars[i].scan == 0
3579 && SkipScanIvars[i+1].skip == 0 && SkipScanIvars[i+1].scan) {
3580 // 0xM0 followed by 0x0N detected.
3581 SkipScanIvars[i].scan = SkipScanIvars[i+1].scan;
3582 for (int j = i+1; j < SkipScan; j++)
3583 SkipScanIvars[j] = SkipScanIvars[j+1];
3584 --SkipScan;
3585 }
3586 }
Daniel Dunbar6bff2512009-08-03 17:06:42 +00003587
Fariborz Jahanian9397e1d2009-03-11 20:59:05 +00003588 // Generate the string.
3589 std::string BitMap;
Daniel Dunbar31682fd2009-05-03 23:21:22 +00003590 for (int i = 0; i <= SkipScan; i++) {
Fariborz Jahanian9397e1d2009-03-11 20:59:05 +00003591 unsigned char byte;
3592 unsigned int skip_small = SkipScanIvars[i].skip % 0xf;
3593 unsigned int scan_small = SkipScanIvars[i].scan % 0xf;
3594 unsigned int skip_big = SkipScanIvars[i].skip / 0xf;
3595 unsigned int scan_big = SkipScanIvars[i].scan / 0xf;
3596
Fariborz Jahanian9397e1d2009-03-11 20:59:05 +00003597 // first skip big.
3598 for (unsigned int ix = 0; ix < skip_big; ix++)
3599 BitMap += (unsigned char)(0xf0);
Daniel Dunbar6bff2512009-08-03 17:06:42 +00003600
Fariborz Jahanian9397e1d2009-03-11 20:59:05 +00003601 // next (skip small, scan)
Daniel Dunbar31682fd2009-05-03 23:21:22 +00003602 if (skip_small) {
Fariborz Jahanian9397e1d2009-03-11 20:59:05 +00003603 byte = skip_small << 4;
Daniel Dunbar31682fd2009-05-03 23:21:22 +00003604 if (scan_big > 0) {
Fariborz Jahanian9397e1d2009-03-11 20:59:05 +00003605 byte |= 0xf;
3606 --scan_big;
Daniel Dunbar31682fd2009-05-03 23:21:22 +00003607 } else if (scan_small) {
Fariborz Jahanian9397e1d2009-03-11 20:59:05 +00003608 byte |= scan_small;
3609 scan_small = 0;
3610 }
3611 BitMap += byte;
3612 }
3613 // next scan big
3614 for (unsigned int ix = 0; ix < scan_big; ix++)
3615 BitMap += (unsigned char)(0x0f);
3616 // last scan small
Daniel Dunbar31682fd2009-05-03 23:21:22 +00003617 if (scan_small) {
Fariborz Jahanian9397e1d2009-03-11 20:59:05 +00003618 byte = scan_small;
3619 BitMap += byte;
3620 }
3621 }
3622 // null terminate string.
Fariborz Jahanian667423a2009-03-25 22:36:49 +00003623 unsigned char zero = 0;
3624 BitMap += zero;
Daniel Dunbar6bff2512009-08-03 17:06:42 +00003625
Fariborz Jahanian3d2ad662009-04-20 22:03:45 +00003626 if (CGM.getLangOptions().ObjCGCBitmapPrint) {
Daniel Dunbar6bff2512009-08-03 17:06:42 +00003627 printf("\n%s ivar layout for class '%s': ",
Fariborz Jahanian3d2ad662009-04-20 22:03:45 +00003628 ForStrongLayout ? "strong" : "weak",
3629 OMD->getClassInterface()->getNameAsCString());
3630 const unsigned char *s = (unsigned char*)BitMap.c_str();
3631 for (unsigned i = 0; i < BitMap.size(); i++)
3632 if (!(s[i] & 0xf0))
3633 printf("0x0%x%s", s[i], s[i] != 0 ? ", " : "");
3634 else
3635 printf("0x%x%s", s[i], s[i] != 0 ? ", " : "");
3636 printf("\n");
3637 }
Daniel Dunbar6bff2512009-08-03 17:06:42 +00003638 llvm::GlobalVariable * Entry =
3639 CreateMetadataVar("\01L_OBJC_CLASS_NAME_",
Owen Anderson0032b272009-08-13 21:57:51 +00003640 llvm::ConstantArray::get(VMContext, BitMap.c_str()),
Daniel Dunbar6bff2512009-08-03 17:06:42 +00003641 "__TEXT,__cstring,cstring_literals",
3642 1, true);
3643 return getConstantGEP(VMContext, Entry, 0, 0);
Fariborz Jahaniand61a50a2009-03-05 22:39:55 +00003644}
3645
Fariborz Jahanian56210f72009-01-21 23:34:32 +00003646llvm::Constant *CGObjCCommonMac::GetMethodVarName(Selector Sel) {
Daniel Dunbar259d93d2008-08-12 03:39:23 +00003647 llvm::GlobalVariable *&Entry = MethodVarNames[Sel];
3648
Daniel Dunbar63c5b502009-03-09 21:49:58 +00003649 // FIXME: Avoid std::string copying.
3650 if (!Entry)
Daniel Dunbar6bff2512009-08-03 17:06:42 +00003651 Entry = CreateMetadataVar("\01L_OBJC_METH_VAR_NAME_",
Owen Anderson0032b272009-08-13 21:57:51 +00003652 llvm::ConstantArray::get(VMContext, Sel.getAsString()),
Daniel Dunbar63c5b502009-03-09 21:49:58 +00003653 "__TEXT,__cstring,cstring_literals",
Daniel Dunbarb90bb002009-04-14 23:14:47 +00003654 1, true);
Daniel Dunbar259d93d2008-08-12 03:39:23 +00003655
Owen Andersona1cf15f2009-07-14 23:10:40 +00003656 return getConstantGEP(VMContext, Entry, 0, 0);
Daniel Dunbar6efc0c52008-08-13 03:21:16 +00003657}
3658
Daniel Dunbar27f9d772008-08-21 04:36:09 +00003659// FIXME: Merge into a single cstring creation function.
Fariborz Jahanian56210f72009-01-21 23:34:32 +00003660llvm::Constant *CGObjCCommonMac::GetMethodVarName(IdentifierInfo *ID) {
Daniel Dunbar27f9d772008-08-21 04:36:09 +00003661 return GetMethodVarName(CGM.getContext().Selectors.getNullarySelector(ID));
3662}
3663
3664// FIXME: Merge into a single cstring creation function.
Fariborz Jahanian56210f72009-01-21 23:34:32 +00003665llvm::Constant *CGObjCCommonMac::GetMethodVarName(const std::string &Name) {
Daniel Dunbar27f9d772008-08-21 04:36:09 +00003666 return GetMethodVarName(&CGM.getContext().Idents.get(Name));
3667}
3668
Daniel Dunbar3e5f0d82009-04-20 06:54:31 +00003669llvm::Constant *CGObjCCommonMac::GetMethodVarType(const FieldDecl *Field) {
Devang Patel7794bb82009-03-04 18:21:39 +00003670 std::string TypeStr;
3671 CGM.getContext().getObjCEncodingForType(Field->getType(), TypeStr, Field);
3672
3673 llvm::GlobalVariable *&Entry = MethodVarTypes[TypeStr];
Daniel Dunbar6efc0c52008-08-13 03:21:16 +00003674
Daniel Dunbar63c5b502009-03-09 21:49:58 +00003675 if (!Entry)
3676 Entry = CreateMetadataVar("\01L_OBJC_METH_VAR_TYPE_",
Owen Anderson0032b272009-08-13 21:57:51 +00003677 llvm::ConstantArray::get(VMContext, TypeStr),
Daniel Dunbar63c5b502009-03-09 21:49:58 +00003678 "__TEXT,__cstring,cstring_literals",
Daniel Dunbarb90bb002009-04-14 23:14:47 +00003679 1, true);
Daniel Dunbar6bff2512009-08-03 17:06:42 +00003680
Owen Andersona1cf15f2009-07-14 23:10:40 +00003681 return getConstantGEP(VMContext, Entry, 0, 0);
Daniel Dunbar259d93d2008-08-12 03:39:23 +00003682}
3683
Fariborz Jahanian56210f72009-01-21 23:34:32 +00003684llvm::Constant *CGObjCCommonMac::GetMethodVarType(const ObjCMethodDecl *D) {
Daniel Dunbar27f9d772008-08-21 04:36:09 +00003685 std::string TypeStr;
Daniel Dunbarc45ef602008-08-26 21:51:14 +00003686 CGM.getContext().getObjCEncodingForMethodDecl(const_cast<ObjCMethodDecl*>(D),
3687 TypeStr);
Devang Patel7794bb82009-03-04 18:21:39 +00003688
3689 llvm::GlobalVariable *&Entry = MethodVarTypes[TypeStr];
3690
Daniel Dunbarb90bb002009-04-14 23:14:47 +00003691 if (!Entry)
3692 Entry = CreateMetadataVar("\01L_OBJC_METH_VAR_TYPE_",
Owen Anderson0032b272009-08-13 21:57:51 +00003693 llvm::ConstantArray::get(VMContext, TypeStr),
Daniel Dunbarb90bb002009-04-14 23:14:47 +00003694 "__TEXT,__cstring,cstring_literals",
3695 1, true);
Devang Patel7794bb82009-03-04 18:21:39 +00003696
Owen Andersona1cf15f2009-07-14 23:10:40 +00003697 return getConstantGEP(VMContext, Entry, 0, 0);
Daniel Dunbar27f9d772008-08-21 04:36:09 +00003698}
3699
Daniel Dunbarc8ef5512008-08-23 00:19:03 +00003700// FIXME: Merge into a single cstring creation function.
Fariborz Jahanian56210f72009-01-21 23:34:32 +00003701llvm::Constant *CGObjCCommonMac::GetPropertyName(IdentifierInfo *Ident) {
Daniel Dunbarc8ef5512008-08-23 00:19:03 +00003702 llvm::GlobalVariable *&Entry = PropertyNames[Ident];
Daniel Dunbar6bff2512009-08-03 17:06:42 +00003703
Daniel Dunbar63c5b502009-03-09 21:49:58 +00003704 if (!Entry)
Daniel Dunbar6bff2512009-08-03 17:06:42 +00003705 Entry = CreateMetadataVar("\01L_OBJC_PROP_NAME_ATTR_",
Daniel Dunbar9c29bf52009-10-18 20:48:59 +00003706 llvm::ConstantArray::get(VMContext,
3707 Ident->getNameStart()),
Daniel Dunbar63c5b502009-03-09 21:49:58 +00003708 "__TEXT,__cstring,cstring_literals",
Daniel Dunbarb90bb002009-04-14 23:14:47 +00003709 1, true);
Daniel Dunbarc8ef5512008-08-23 00:19:03 +00003710
Owen Andersona1cf15f2009-07-14 23:10:40 +00003711 return getConstantGEP(VMContext, Entry, 0, 0);
Daniel Dunbarc8ef5512008-08-23 00:19:03 +00003712}
3713
3714// FIXME: Merge into a single cstring creation function.
Daniel Dunbarc56f34a2008-08-28 04:38:10 +00003715// FIXME: This Decl should be more precise.
Daniel Dunbar63c5b502009-03-09 21:49:58 +00003716llvm::Constant *
Daniel Dunbar6bff2512009-08-03 17:06:42 +00003717CGObjCCommonMac::GetPropertyTypeString(const ObjCPropertyDecl *PD,
3718 const Decl *Container) {
Daniel Dunbarc56f34a2008-08-28 04:38:10 +00003719 std::string TypeStr;
3720 CGM.getContext().getObjCEncodingForPropertyDecl(PD, Container, TypeStr);
Daniel Dunbarc8ef5512008-08-23 00:19:03 +00003721 return GetPropertyName(&CGM.getContext().Idents.get(TypeStr));
3722}
3723
Daniel Dunbar6bff2512009-08-03 17:06:42 +00003724void CGObjCCommonMac::GetNameForMethod(const ObjCMethodDecl *D,
Fariborz Jahanian56210f72009-01-21 23:34:32 +00003725 const ObjCContainerDecl *CD,
Daniel Dunbarc575ce72009-10-19 01:21:19 +00003726 llvm::SmallVectorImpl<char> &Name) {
3727 llvm::raw_svector_ostream OS(Name);
Fariborz Jahanian679a5022009-01-10 21:06:09 +00003728 assert (CD && "Missing container decl in GetNameForMethod");
Daniel Dunbarc575ce72009-10-19 01:21:19 +00003729 OS << '\01' << (D->isInstanceMethod() ? '-' : '+')
3730 << '[' << CD->getName();
Daniel Dunbar6bff2512009-08-03 17:06:42 +00003731 if (const ObjCCategoryImplDecl *CID =
Daniel Dunbarc575ce72009-10-19 01:21:19 +00003732 dyn_cast<ObjCCategoryImplDecl>(D->getDeclContext()))
Benjamin Kramer900fc632010-04-17 09:33:03 +00003733 OS << '(' << CID << ')';
Daniel Dunbarc575ce72009-10-19 01:21:19 +00003734 OS << ' ' << D->getSelector().getAsString() << ']';
Daniel Dunbarb7ec2462008-08-16 03:19:19 +00003735}
3736
Daniel Dunbarf77ac862008-08-11 21:35:06 +00003737void CGObjCMac::FinishModule() {
Daniel Dunbar4e2d7d02008-08-12 06:48:42 +00003738 EmitModuleInfo();
3739
Daniel Dunbar0c0e7a62008-10-29 22:36:39 +00003740 // Emit the dummy bodies for any protocols which were referenced but
3741 // never defined.
Daniel Dunbar6bff2512009-08-03 17:06:42 +00003742 for (llvm::DenseMap<IdentifierInfo*, llvm::GlobalVariable*>::iterator
Chris Lattnerad64e022009-07-17 23:57:13 +00003743 I = Protocols.begin(), e = Protocols.end(); I != e; ++I) {
3744 if (I->second->hasInitializer())
Daniel Dunbar0c0e7a62008-10-29 22:36:39 +00003745 continue;
Daniel Dunbar4e2d7d02008-08-12 06:48:42 +00003746
Daniel Dunbar0c0e7a62008-10-29 22:36:39 +00003747 std::vector<llvm::Constant*> Values(5);
Owen Andersonc9c88b42009-07-31 20:28:54 +00003748 Values[0] = llvm::Constant::getNullValue(ObjCTypes.ProtocolExtensionPtrTy);
Chris Lattnerad64e022009-07-17 23:57:13 +00003749 Values[1] = GetClassName(I->first);
Owen Andersonc9c88b42009-07-31 20:28:54 +00003750 Values[2] = llvm::Constant::getNullValue(ObjCTypes.ProtocolListPtrTy);
Daniel Dunbar0c0e7a62008-10-29 22:36:39 +00003751 Values[3] = Values[4] =
Owen Andersonc9c88b42009-07-31 20:28:54 +00003752 llvm::Constant::getNullValue(ObjCTypes.MethodDescriptionListPtrTy);
Chris Lattnerad64e022009-07-17 23:57:13 +00003753 I->second->setLinkage(llvm::GlobalValue::InternalLinkage);
Owen Anderson08e25242009-07-27 22:29:56 +00003754 I->second->setInitializer(llvm::ConstantStruct::get(ObjCTypes.ProtocolTy,
Daniel Dunbar0c0e7a62008-10-29 22:36:39 +00003755 Values));
Chris Lattnerad64e022009-07-17 23:57:13 +00003756 CGM.AddUsedGlobal(I->second);
Daniel Dunbar0c0e7a62008-10-29 22:36:39 +00003757 }
3758
Daniel Dunbar242d4dc2008-08-25 06:02:07 +00003759 // Add assembler directives to add lazy undefined symbol references
3760 // for classes which are referenced but not defined. This is
3761 // important for correct linker interaction.
Daniel Dunbar33063492009-09-07 00:20:42 +00003762 //
3763 // FIXME: It would be nice if we had an LLVM construct for this.
3764 if (!LazySymbols.empty() || !DefinedSymbols.empty()) {
3765 llvm::SmallString<256> Asm;
3766 Asm += CGM.getModule().getModuleInlineAsm();
3767 if (!Asm.empty() && Asm.back() != '\n')
3768 Asm += '\n';
Daniel Dunbar242d4dc2008-08-25 06:02:07 +00003769
Daniel Dunbar33063492009-09-07 00:20:42 +00003770 llvm::raw_svector_ostream OS(Asm);
Daniel Dunbar33063492009-09-07 00:20:42 +00003771 for (llvm::SetVector<IdentifierInfo*>::iterator I = DefinedSymbols.begin(),
3772 e = DefinedSymbols.end(); I != e; ++I)
Daniel Dunbar01eb9b92009-10-18 21:17:35 +00003773 OS << "\t.objc_class_name_" << (*I)->getName() << "=0\n"
3774 << "\t.globl .objc_class_name_" << (*I)->getName() << "\n";
Chris Lattnerafd5eda2010-04-17 18:26:20 +00003775 for (llvm::SetVector<IdentifierInfo*>::iterator I = LazySymbols.begin(),
Fariborz Jahanianb9c5b3d2010-06-21 22:05:18 +00003776 e = LazySymbols.end(); I != e; ++I) {
Chris Lattnerafd5eda2010-04-17 18:26:20 +00003777 OS << "\t.lazy_reference .objc_class_name_" << (*I)->getName() << "\n";
Fariborz Jahanianb9c5b3d2010-06-21 22:05:18 +00003778 }
3779
3780 for (size_t i = 0; i < DefinedCategoryNames.size(); ++i) {
3781 OS << "\t.objc_category_name_" << DefinedCategoryNames[i] << "=0\n"
3782 << "\t.globl .objc_category_name_" << DefinedCategoryNames[i] << "\n";
3783 }
Chris Lattnerafd5eda2010-04-17 18:26:20 +00003784
Daniel Dunbar33063492009-09-07 00:20:42 +00003785 CGM.getModule().setModuleInlineAsm(OS.str());
Daniel Dunbar242d4dc2008-08-25 06:02:07 +00003786 }
Daniel Dunbarf77ac862008-08-11 21:35:06 +00003787}
3788
Daniel Dunbar6bff2512009-08-03 17:06:42 +00003789CGObjCNonFragileABIMac::CGObjCNonFragileABIMac(CodeGen::CodeGenModule &cgm)
Fariborz Jahanianee0af742009-01-21 22:04:16 +00003790 : CGObjCCommonMac(cgm),
Mike Stump1eb44332009-09-09 15:08:12 +00003791 ObjCTypes(cgm) {
Fariborz Jahanianaa23b572009-01-23 23:53:38 +00003792 ObjCEmptyCacheVar = ObjCEmptyVtableVar = NULL;
Fariborz Jahanianee0af742009-01-21 22:04:16 +00003793 ObjCABI = 2;
3794}
3795
Daniel Dunbarf77ac862008-08-11 21:35:06 +00003796/* *** */
3797
Fariborz Jahanianee0af742009-01-21 22:04:16 +00003798ObjCCommonTypesHelper::ObjCCommonTypesHelper(CodeGen::CodeGenModule &cgm)
Mike Stump1eb44332009-09-09 15:08:12 +00003799 : VMContext(cgm.getLLVMContext()), CGM(cgm) {
Daniel Dunbar4e2d7d02008-08-12 06:48:42 +00003800 CodeGen::CodeGenTypes &Types = CGM.getTypes();
3801 ASTContext &Ctx = CGM.getContext();
Daniel Dunbar6bff2512009-08-03 17:06:42 +00003802
Daniel Dunbar27f9d772008-08-21 04:36:09 +00003803 ShortTy = Types.ConvertType(Ctx.ShortTy);
Daniel Dunbar6efc0c52008-08-13 03:21:16 +00003804 IntTy = Types.ConvertType(Ctx.IntTy);
Daniel Dunbar4e2d7d02008-08-12 06:48:42 +00003805 LongTy = Types.ConvertType(Ctx.LongTy);
Fariborz Jahanian0a855d02009-03-23 19:10:40 +00003806 LongLongTy = Types.ConvertType(Ctx.LongLongTy);
Benjamin Kramer3c0ef8c2009-10-13 10:07:13 +00003807 Int8PtrTy = llvm::Type::getInt8PtrTy(VMContext);
Daniel Dunbar6bff2512009-08-03 17:06:42 +00003808
Daniel Dunbar4e2d7d02008-08-12 06:48:42 +00003809 ObjectPtrTy = Types.ConvertType(Ctx.getObjCIdType());
Owen Anderson96e0fc72009-07-29 22:16:19 +00003810 PtrObjectPtrTy = llvm::PointerType::getUnqual(ObjectPtrTy);
Daniel Dunbar4e2d7d02008-08-12 06:48:42 +00003811 SelectorPtrTy = Types.ConvertType(Ctx.getObjCSelType());
Daniel Dunbar6bff2512009-08-03 17:06:42 +00003812
Mike Stumpf5408fe2009-05-16 07:57:57 +00003813 // FIXME: It would be nice to unify this with the opaque type, so that the IR
3814 // comes out a bit cleaner.
Daniel Dunbar6efc0c52008-08-13 03:21:16 +00003815 const llvm::Type *T = Types.ConvertType(Ctx.getObjCProtoType());
Owen Anderson96e0fc72009-07-29 22:16:19 +00003816 ExternalProtocolPtrTy = llvm::PointerType::getUnqual(T);
Daniel Dunbar6bff2512009-08-03 17:06:42 +00003817
Fariborz Jahanianee0af742009-01-21 22:04:16 +00003818 // I'm not sure I like this. The implicit coordination is a bit
3819 // gross. We should solve this in a reasonable fashion because this
3820 // is a pretty common task (match some runtime data structure with
3821 // an LLVM data structure).
Daniel Dunbar6bff2512009-08-03 17:06:42 +00003822
Fariborz Jahanianee0af742009-01-21 22:04:16 +00003823 // FIXME: This is leaked.
3824 // FIXME: Merge with rewriter code?
Daniel Dunbar6bff2512009-08-03 17:06:42 +00003825
Fariborz Jahanianee0af742009-01-21 22:04:16 +00003826 // struct _objc_super {
3827 // id self;
3828 // Class cls;
3829 // }
Abramo Bagnara465d41b2010-05-11 21:36:43 +00003830 RecordDecl *RD = RecordDecl::Create(Ctx, TTK_Struct,
Daniel Dunbardaa3ac52010-04-29 16:29:11 +00003831 Ctx.getTranslationUnitDecl(),
Fariborz Jahanianee0af742009-01-21 22:04:16 +00003832 SourceLocation(),
Daniel Dunbar6bff2512009-08-03 17:06:42 +00003833 &Ctx.Idents.get("_objc_super"));
Argyrios Kyrtzidis17945a02009-06-30 02:36:12 +00003834 RD->addDecl(FieldDecl::Create(Ctx, RD, SourceLocation(), 0,
Argyrios Kyrtzidisa1d56622009-08-19 01:27:57 +00003835 Ctx.getObjCIdType(), 0, 0, false));
Daniel Dunbar6bff2512009-08-03 17:06:42 +00003836 RD->addDecl(FieldDecl::Create(Ctx, RD, SourceLocation(), 0,
Argyrios Kyrtzidisa1d56622009-08-19 01:27:57 +00003837 Ctx.getObjCClassType(), 0, 0, false));
Douglas Gregor838db382010-02-11 01:19:42 +00003838 RD->completeDefinition();
Daniel Dunbar6bff2512009-08-03 17:06:42 +00003839
Fariborz Jahanianee0af742009-01-21 22:04:16 +00003840 SuperCTy = Ctx.getTagDeclType(RD);
3841 SuperPtrCTy = Ctx.getPointerType(SuperCTy);
Daniel Dunbar6bff2512009-08-03 17:06:42 +00003842
Fariborz Jahanianee0af742009-01-21 22:04:16 +00003843 SuperTy = cast<llvm::StructType>(Types.ConvertType(SuperCTy));
Daniel Dunbar6bff2512009-08-03 17:06:42 +00003844 SuperPtrTy = llvm::PointerType::getUnqual(SuperTy);
3845
Fariborz Jahanian30bc5712009-01-22 23:02:58 +00003846 // struct _prop_t {
3847 // char *name;
Daniel Dunbar6bff2512009-08-03 17:06:42 +00003848 // char *attributes;
Fariborz Jahanian30bc5712009-01-22 23:02:58 +00003849 // }
Owen Anderson47a434f2009-08-05 23:18:46 +00003850 PropertyTy = llvm::StructType::get(VMContext, Int8PtrTy, Int8PtrTy, NULL);
Daniel Dunbar6bff2512009-08-03 17:06:42 +00003851 CGM.getModule().addTypeName("struct._prop_t",
Fariborz Jahanian30bc5712009-01-22 23:02:58 +00003852 PropertyTy);
Daniel Dunbar6bff2512009-08-03 17:06:42 +00003853
Fariborz Jahanian30bc5712009-01-22 23:02:58 +00003854 // struct _prop_list_t {
Fariborz Jahaniand55b6fc2009-01-23 01:46:23 +00003855 // uint32_t entsize; // sizeof(struct _prop_t)
Fariborz Jahanian30bc5712009-01-22 23:02:58 +00003856 // uint32_t count_of_properties;
Fariborz Jahaniand55b6fc2009-01-23 01:46:23 +00003857 // struct _prop_t prop_list[count_of_properties];
Fariborz Jahanian30bc5712009-01-22 23:02:58 +00003858 // }
Owen Anderson47a434f2009-08-05 23:18:46 +00003859 PropertyListTy = llvm::StructType::get(VMContext, IntTy,
Fariborz Jahanian30bc5712009-01-22 23:02:58 +00003860 IntTy,
Owen Anderson96e0fc72009-07-29 22:16:19 +00003861 llvm::ArrayType::get(PropertyTy, 0),
Fariborz Jahanian30bc5712009-01-22 23:02:58 +00003862 NULL);
Daniel Dunbar6bff2512009-08-03 17:06:42 +00003863 CGM.getModule().addTypeName("struct._prop_list_t",
Fariborz Jahanian30bc5712009-01-22 23:02:58 +00003864 PropertyListTy);
3865 // struct _prop_list_t *
Owen Anderson96e0fc72009-07-29 22:16:19 +00003866 PropertyListPtrTy = llvm::PointerType::getUnqual(PropertyListTy);
Daniel Dunbar6bff2512009-08-03 17:06:42 +00003867
Fariborz Jahanian30bc5712009-01-22 23:02:58 +00003868 // struct _objc_method {
3869 // SEL _cmd;
3870 // char *method_type;
3871 // char *_imp;
3872 // }
Owen Anderson47a434f2009-08-05 23:18:46 +00003873 MethodTy = llvm::StructType::get(VMContext, SelectorPtrTy,
Fariborz Jahanian30bc5712009-01-22 23:02:58 +00003874 Int8PtrTy,
3875 Int8PtrTy,
3876 NULL);
3877 CGM.getModule().addTypeName("struct._objc_method", MethodTy);
Daniel Dunbar6bff2512009-08-03 17:06:42 +00003878
Fariborz Jahaniand55b6fc2009-01-23 01:46:23 +00003879 // struct _objc_cache *
Owen Anderson8c8f69e2009-08-13 23:27:53 +00003880 CacheTy = llvm::OpaqueType::get(VMContext);
Fariborz Jahaniand55b6fc2009-01-23 01:46:23 +00003881 CGM.getModule().addTypeName("struct._objc_cache", CacheTy);
Owen Anderson96e0fc72009-07-29 22:16:19 +00003882 CachePtrTy = llvm::PointerType::getUnqual(CacheTy);
Fariborz Jahanianee0af742009-01-21 22:04:16 +00003883}
Daniel Dunbar4e2d7d02008-08-12 06:48:42 +00003884
Daniel Dunbar6bff2512009-08-03 17:06:42 +00003885ObjCTypesHelper::ObjCTypesHelper(CodeGen::CodeGenModule &cgm)
Mike Stump1eb44332009-09-09 15:08:12 +00003886 : ObjCCommonTypesHelper(cgm) {
Fariborz Jahanian10a42312009-01-21 00:39:53 +00003887 // struct _objc_method_description {
3888 // SEL name;
3889 // char *types;
3890 // }
Daniel Dunbar6bff2512009-08-03 17:06:42 +00003891 MethodDescriptionTy =
Owen Anderson47a434f2009-08-05 23:18:46 +00003892 llvm::StructType::get(VMContext, SelectorPtrTy,
Daniel Dunbar27f9d772008-08-21 04:36:09 +00003893 Int8PtrTy,
Daniel Dunbar6efc0c52008-08-13 03:21:16 +00003894 NULL);
Daniel Dunbar6bff2512009-08-03 17:06:42 +00003895 CGM.getModule().addTypeName("struct._objc_method_description",
Daniel Dunbar6efc0c52008-08-13 03:21:16 +00003896 MethodDescriptionTy);
3897
Fariborz Jahanian10a42312009-01-21 00:39:53 +00003898 // struct _objc_method_description_list {
3899 // int count;
3900 // struct _objc_method_description[1];
3901 // }
Daniel Dunbar6bff2512009-08-03 17:06:42 +00003902 MethodDescriptionListTy =
Owen Anderson47a434f2009-08-05 23:18:46 +00003903 llvm::StructType::get(VMContext, IntTy,
Owen Anderson96e0fc72009-07-29 22:16:19 +00003904 llvm::ArrayType::get(MethodDescriptionTy, 0),
Daniel Dunbar6efc0c52008-08-13 03:21:16 +00003905 NULL);
Daniel Dunbar6bff2512009-08-03 17:06:42 +00003906 CGM.getModule().addTypeName("struct._objc_method_description_list",
Daniel Dunbar6efc0c52008-08-13 03:21:16 +00003907 MethodDescriptionListTy);
Daniel Dunbar6bff2512009-08-03 17:06:42 +00003908
Fariborz Jahanian10a42312009-01-21 00:39:53 +00003909 // struct _objc_method_description_list *
Daniel Dunbar6bff2512009-08-03 17:06:42 +00003910 MethodDescriptionListPtrTy =
Owen Anderson96e0fc72009-07-29 22:16:19 +00003911 llvm::PointerType::getUnqual(MethodDescriptionListTy);
Daniel Dunbar6efc0c52008-08-13 03:21:16 +00003912
Daniel Dunbar6efc0c52008-08-13 03:21:16 +00003913 // Protocol description structures
3914
Fariborz Jahanian10a42312009-01-21 00:39:53 +00003915 // struct _objc_protocol_extension {
3916 // uint32_t size; // sizeof(struct _objc_protocol_extension)
3917 // struct _objc_method_description_list *optional_instance_methods;
3918 // struct _objc_method_description_list *optional_class_methods;
3919 // struct _objc_property_list *instance_properties;
3920 // }
Daniel Dunbar6bff2512009-08-03 17:06:42 +00003921 ProtocolExtensionTy =
Owen Anderson47a434f2009-08-05 23:18:46 +00003922 llvm::StructType::get(VMContext, IntTy,
Fariborz Jahanian10a42312009-01-21 00:39:53 +00003923 MethodDescriptionListPtrTy,
3924 MethodDescriptionListPtrTy,
Daniel Dunbar6efc0c52008-08-13 03:21:16 +00003925 PropertyListPtrTy,
3926 NULL);
Daniel Dunbar6bff2512009-08-03 17:06:42 +00003927 CGM.getModule().addTypeName("struct._objc_protocol_extension",
Daniel Dunbar6efc0c52008-08-13 03:21:16 +00003928 ProtocolExtensionTy);
Daniel Dunbar6bff2512009-08-03 17:06:42 +00003929
Fariborz Jahanian10a42312009-01-21 00:39:53 +00003930 // struct _objc_protocol_extension *
Owen Anderson96e0fc72009-07-29 22:16:19 +00003931 ProtocolExtensionPtrTy = llvm::PointerType::getUnqual(ProtocolExtensionTy);
Daniel Dunbar6efc0c52008-08-13 03:21:16 +00003932
Daniel Dunbar0c0e7a62008-10-29 22:36:39 +00003933 // Handle recursive construction of Protocol and ProtocolList types
Daniel Dunbar6efc0c52008-08-13 03:21:16 +00003934
Owen Anderson8c8f69e2009-08-13 23:27:53 +00003935 llvm::PATypeHolder ProtocolTyHolder = llvm::OpaqueType::get(VMContext);
3936 llvm::PATypeHolder ProtocolListTyHolder = llvm::OpaqueType::get(VMContext);
Daniel Dunbar6efc0c52008-08-13 03:21:16 +00003937
Daniel Dunbar6bff2512009-08-03 17:06:42 +00003938 const llvm::Type *T =
Mike Stump1eb44332009-09-09 15:08:12 +00003939 llvm::StructType::get(VMContext,
Owen Anderson47a434f2009-08-05 23:18:46 +00003940 llvm::PointerType::getUnqual(ProtocolListTyHolder),
Fariborz Jahanianee0af742009-01-21 22:04:16 +00003941 LongTy,
Owen Anderson96e0fc72009-07-29 22:16:19 +00003942 llvm::ArrayType::get(ProtocolTyHolder, 0),
Fariborz Jahanianee0af742009-01-21 22:04:16 +00003943 NULL);
Daniel Dunbar6efc0c52008-08-13 03:21:16 +00003944 cast<llvm::OpaqueType>(ProtocolListTyHolder.get())->refineAbstractTypeTo(T);
3945
Fariborz Jahanian10a42312009-01-21 00:39:53 +00003946 // struct _objc_protocol {
3947 // struct _objc_protocol_extension *isa;
3948 // char *protocol_name;
3949 // struct _objc_protocol **_objc_protocol_list;
3950 // struct _objc_method_description_list *instance_methods;
3951 // struct _objc_method_description_list *class_methods;
3952 // }
Owen Anderson47a434f2009-08-05 23:18:46 +00003953 T = llvm::StructType::get(VMContext, ProtocolExtensionPtrTy,
Daniel Dunbar27f9d772008-08-21 04:36:09 +00003954 Int8PtrTy,
Daniel Dunbar6bff2512009-08-03 17:06:42 +00003955 llvm::PointerType::getUnqual(ProtocolListTyHolder),
Daniel Dunbar6efc0c52008-08-13 03:21:16 +00003956 MethodDescriptionListPtrTy,
3957 MethodDescriptionListPtrTy,
3958 NULL);
3959 cast<llvm::OpaqueType>(ProtocolTyHolder.get())->refineAbstractTypeTo(T);
3960
3961 ProtocolListTy = cast<llvm::StructType>(ProtocolListTyHolder.get());
Daniel Dunbar6bff2512009-08-03 17:06:42 +00003962 CGM.getModule().addTypeName("struct._objc_protocol_list",
Daniel Dunbar6efc0c52008-08-13 03:21:16 +00003963 ProtocolListTy);
Fariborz Jahanian10a42312009-01-21 00:39:53 +00003964 // struct _objc_protocol_list *
Owen Anderson96e0fc72009-07-29 22:16:19 +00003965 ProtocolListPtrTy = llvm::PointerType::getUnqual(ProtocolListTy);
Daniel Dunbar6efc0c52008-08-13 03:21:16 +00003966
3967 ProtocolTy = cast<llvm::StructType>(ProtocolTyHolder.get());
Fariborz Jahanian10a42312009-01-21 00:39:53 +00003968 CGM.getModule().addTypeName("struct._objc_protocol", ProtocolTy);
Owen Anderson96e0fc72009-07-29 22:16:19 +00003969 ProtocolPtrTy = llvm::PointerType::getUnqual(ProtocolTy);
Daniel Dunbar27f9d772008-08-21 04:36:09 +00003970
3971 // Class description structures
3972
Fariborz Jahanian10a42312009-01-21 00:39:53 +00003973 // struct _objc_ivar {
3974 // char *ivar_name;
3975 // char *ivar_type;
3976 // int ivar_offset;
3977 // }
Owen Anderson47a434f2009-08-05 23:18:46 +00003978 IvarTy = llvm::StructType::get(VMContext, Int8PtrTy,
Daniel Dunbar6bff2512009-08-03 17:06:42 +00003979 Int8PtrTy,
3980 IntTy,
Daniel Dunbar27f9d772008-08-21 04:36:09 +00003981 NULL);
3982 CGM.getModule().addTypeName("struct._objc_ivar", IvarTy);
3983
Fariborz Jahanian10a42312009-01-21 00:39:53 +00003984 // struct _objc_ivar_list *
Owen Anderson8c8f69e2009-08-13 23:27:53 +00003985 IvarListTy = llvm::OpaqueType::get(VMContext);
Daniel Dunbar27f9d772008-08-21 04:36:09 +00003986 CGM.getModule().addTypeName("struct._objc_ivar_list", IvarListTy);
Owen Anderson96e0fc72009-07-29 22:16:19 +00003987 IvarListPtrTy = llvm::PointerType::getUnqual(IvarListTy);
Daniel Dunbar27f9d772008-08-21 04:36:09 +00003988
Fariborz Jahanian10a42312009-01-21 00:39:53 +00003989 // struct _objc_method_list *
Owen Anderson8c8f69e2009-08-13 23:27:53 +00003990 MethodListTy = llvm::OpaqueType::get(VMContext);
Daniel Dunbar27f9d772008-08-21 04:36:09 +00003991 CGM.getModule().addTypeName("struct._objc_method_list", MethodListTy);
Owen Anderson96e0fc72009-07-29 22:16:19 +00003992 MethodListPtrTy = llvm::PointerType::getUnqual(MethodListTy);
Daniel Dunbar27f9d772008-08-21 04:36:09 +00003993
Fariborz Jahanian10a42312009-01-21 00:39:53 +00003994 // struct _objc_class_extension *
Daniel Dunbar6bff2512009-08-03 17:06:42 +00003995 ClassExtensionTy =
Owen Anderson47a434f2009-08-05 23:18:46 +00003996 llvm::StructType::get(VMContext, IntTy,
Daniel Dunbar27f9d772008-08-21 04:36:09 +00003997 Int8PtrTy,
3998 PropertyListPtrTy,
3999 NULL);
4000 CGM.getModule().addTypeName("struct._objc_class_extension", ClassExtensionTy);
Owen Anderson96e0fc72009-07-29 22:16:19 +00004001 ClassExtensionPtrTy = llvm::PointerType::getUnqual(ClassExtensionTy);
Daniel Dunbar27f9d772008-08-21 04:36:09 +00004002
Owen Anderson8c8f69e2009-08-13 23:27:53 +00004003 llvm::PATypeHolder ClassTyHolder = llvm::OpaqueType::get(VMContext);
Daniel Dunbar27f9d772008-08-21 04:36:09 +00004004
Fariborz Jahanian10a42312009-01-21 00:39:53 +00004005 // struct _objc_class {
4006 // Class isa;
4007 // Class super_class;
4008 // char *name;
4009 // long version;
4010 // long info;
4011 // long instance_size;
4012 // struct _objc_ivar_list *ivars;
4013 // struct _objc_method_list *methods;
4014 // struct _objc_cache *cache;
4015 // struct _objc_protocol_list *protocols;
4016 // char *ivar_layout;
4017 // struct _objc_class_ext *ext;
4018 // };
Owen Anderson47a434f2009-08-05 23:18:46 +00004019 T = llvm::StructType::get(VMContext,
4020 llvm::PointerType::getUnqual(ClassTyHolder),
Owen Anderson96e0fc72009-07-29 22:16:19 +00004021 llvm::PointerType::getUnqual(ClassTyHolder),
Daniel Dunbar27f9d772008-08-21 04:36:09 +00004022 Int8PtrTy,
4023 LongTy,
4024 LongTy,
4025 LongTy,
4026 IvarListPtrTy,
4027 MethodListPtrTy,
4028 CachePtrTy,
4029 ProtocolListPtrTy,
4030 Int8PtrTy,
4031 ClassExtensionPtrTy,
4032 NULL);
4033 cast<llvm::OpaqueType>(ClassTyHolder.get())->refineAbstractTypeTo(T);
Daniel Dunbar6bff2512009-08-03 17:06:42 +00004034
Daniel Dunbar27f9d772008-08-21 04:36:09 +00004035 ClassTy = cast<llvm::StructType>(ClassTyHolder.get());
4036 CGM.getModule().addTypeName("struct._objc_class", ClassTy);
Owen Anderson96e0fc72009-07-29 22:16:19 +00004037 ClassPtrTy = llvm::PointerType::getUnqual(ClassTy);
Daniel Dunbar27f9d772008-08-21 04:36:09 +00004038
Fariborz Jahanian10a42312009-01-21 00:39:53 +00004039 // struct _objc_category {
4040 // char *category_name;
4041 // char *class_name;
4042 // struct _objc_method_list *instance_method;
4043 // struct _objc_method_list *class_method;
4044 // uint32_t size; // sizeof(struct _objc_category)
4045 // struct _objc_property_list *instance_properties;// category's @property
4046 // }
Owen Anderson47a434f2009-08-05 23:18:46 +00004047 CategoryTy = llvm::StructType::get(VMContext, Int8PtrTy,
Daniel Dunbar86e253a2008-08-22 20:34:54 +00004048 Int8PtrTy,
4049 MethodListPtrTy,
4050 MethodListPtrTy,
4051 ProtocolListPtrTy,
4052 IntTy,
4053 PropertyListPtrTy,
4054 NULL);
4055 CGM.getModule().addTypeName("struct._objc_category", CategoryTy);
4056
Daniel Dunbar27f9d772008-08-21 04:36:09 +00004057 // Global metadata structures
4058
Fariborz Jahanian10a42312009-01-21 00:39:53 +00004059 // struct _objc_symtab {
4060 // long sel_ref_cnt;
4061 // SEL *refs;
4062 // short cls_def_cnt;
4063 // short cat_def_cnt;
4064 // char *defs[cls_def_cnt + cat_def_cnt];
4065 // }
Owen Anderson47a434f2009-08-05 23:18:46 +00004066 SymtabTy = llvm::StructType::get(VMContext, LongTy,
Daniel Dunbar27f9d772008-08-21 04:36:09 +00004067 SelectorPtrTy,
4068 ShortTy,
4069 ShortTy,
Owen Anderson96e0fc72009-07-29 22:16:19 +00004070 llvm::ArrayType::get(Int8PtrTy, 0),
Daniel Dunbar27f9d772008-08-21 04:36:09 +00004071 NULL);
4072 CGM.getModule().addTypeName("struct._objc_symtab", SymtabTy);
Owen Anderson96e0fc72009-07-29 22:16:19 +00004073 SymtabPtrTy = llvm::PointerType::getUnqual(SymtabTy);
Daniel Dunbar27f9d772008-08-21 04:36:09 +00004074
Fariborz Jahaniandb286862009-01-22 00:37:21 +00004075 // struct _objc_module {
4076 // long version;
4077 // long size; // sizeof(struct _objc_module)
4078 // char *name;
4079 // struct _objc_symtab* symtab;
4080 // }
Daniel Dunbar6bff2512009-08-03 17:06:42 +00004081 ModuleTy =
Owen Anderson47a434f2009-08-05 23:18:46 +00004082 llvm::StructType::get(VMContext, LongTy,
Daniel Dunbar27f9d772008-08-21 04:36:09 +00004083 LongTy,
4084 Int8PtrTy,
4085 SymtabPtrTy,
4086 NULL);
4087 CGM.getModule().addTypeName("struct._objc_module", ModuleTy);
Daniel Dunbar14c80b72008-08-23 09:25:55 +00004088
Daniel Dunbar6bff2512009-08-03 17:06:42 +00004089
Mike Stumpf5408fe2009-05-16 07:57:57 +00004090 // FIXME: This is the size of the setjmp buffer and should be target
4091 // specific. 18 is what's used on 32-bit X86.
Anders Carlsson124526b2008-09-09 10:10:21 +00004092 uint64_t SetJmpBufferSize = 18;
Daniel Dunbar6bff2512009-08-03 17:06:42 +00004093
Anders Carlsson124526b2008-09-09 10:10:21 +00004094 // Exceptions
Owen Anderson96e0fc72009-07-29 22:16:19 +00004095 const llvm::Type *StackPtrTy = llvm::ArrayType::get(
Benjamin Kramer3c0ef8c2009-10-13 10:07:13 +00004096 llvm::Type::getInt8PtrTy(VMContext), 4);
Daniel Dunbar6bff2512009-08-03 17:06:42 +00004097
4098 ExceptionDataTy =
Chris Lattner77b89b82010-06-27 07:15:29 +00004099 llvm::StructType::get(VMContext,
4100 llvm::ArrayType::get(llvm::Type::getInt32Ty(VMContext),
4101 SetJmpBufferSize),
Anders Carlsson124526b2008-09-09 10:10:21 +00004102 StackPtrTy, NULL);
Daniel Dunbar6bff2512009-08-03 17:06:42 +00004103 CGM.getModule().addTypeName("struct._objc_exception_data",
Anders Carlsson124526b2008-09-09 10:10:21 +00004104 ExceptionDataTy);
4105
Daniel Dunbarbbce49b2008-08-12 00:12:39 +00004106}
4107
Daniel Dunbar6bff2512009-08-03 17:06:42 +00004108ObjCNonFragileABITypesHelper::ObjCNonFragileABITypesHelper(CodeGen::CodeGenModule &cgm)
Mike Stump1eb44332009-09-09 15:08:12 +00004109 : ObjCCommonTypesHelper(cgm) {
Fariborz Jahanian30bc5712009-01-22 23:02:58 +00004110 // struct _method_list_t {
4111 // uint32_t entsize; // sizeof(struct _objc_method)
4112 // uint32_t method_count;
4113 // struct _objc_method method_list[method_count];
4114 // }
Owen Anderson47a434f2009-08-05 23:18:46 +00004115 MethodListnfABITy = llvm::StructType::get(VMContext, IntTy,
Fariborz Jahaniand55b6fc2009-01-23 01:46:23 +00004116 IntTy,
Owen Anderson96e0fc72009-07-29 22:16:19 +00004117 llvm::ArrayType::get(MethodTy, 0),
Fariborz Jahaniand55b6fc2009-01-23 01:46:23 +00004118 NULL);
4119 CGM.getModule().addTypeName("struct.__method_list_t",
4120 MethodListnfABITy);
4121 // struct method_list_t *
Owen Anderson96e0fc72009-07-29 22:16:19 +00004122 MethodListnfABIPtrTy = llvm::PointerType::getUnqual(MethodListnfABITy);
Daniel Dunbar6bff2512009-08-03 17:06:42 +00004123
Fariborz Jahanian30bc5712009-01-22 23:02:58 +00004124 // struct _protocol_t {
4125 // id isa; // NULL
4126 // const char * const protocol_name;
Fariborz Jahaniand55b6fc2009-01-23 01:46:23 +00004127 // const struct _protocol_list_t * protocol_list; // super protocols
Fariborz Jahanian30bc5712009-01-22 23:02:58 +00004128 // const struct method_list_t * const instance_methods;
4129 // const struct method_list_t * const class_methods;
4130 // const struct method_list_t *optionalInstanceMethods;
4131 // const struct method_list_t *optionalClassMethods;
Fariborz Jahaniand55b6fc2009-01-23 01:46:23 +00004132 // const struct _prop_list_t * properties;
Fariborz Jahanian30bc5712009-01-22 23:02:58 +00004133 // const uint32_t size; // sizeof(struct _protocol_t)
4134 // const uint32_t flags; // = 0
4135 // }
Daniel Dunbar6bff2512009-08-03 17:06:42 +00004136
Fariborz Jahaniand55b6fc2009-01-23 01:46:23 +00004137 // Holder for struct _protocol_list_t *
Owen Anderson8c8f69e2009-08-13 23:27:53 +00004138 llvm::PATypeHolder ProtocolListTyHolder = llvm::OpaqueType::get(VMContext);
Daniel Dunbar6bff2512009-08-03 17:06:42 +00004139
Owen Anderson47a434f2009-08-05 23:18:46 +00004140 ProtocolnfABITy = llvm::StructType::get(VMContext, ObjectPtrTy,
Fariborz Jahaniand55b6fc2009-01-23 01:46:23 +00004141 Int8PtrTy,
Owen Anderson96e0fc72009-07-29 22:16:19 +00004142 llvm::PointerType::getUnqual(
Fariborz Jahaniand55b6fc2009-01-23 01:46:23 +00004143 ProtocolListTyHolder),
4144 MethodListnfABIPtrTy,
4145 MethodListnfABIPtrTy,
4146 MethodListnfABIPtrTy,
4147 MethodListnfABIPtrTy,
4148 PropertyListPtrTy,
4149 IntTy,
4150 IntTy,
4151 NULL);
4152 CGM.getModule().addTypeName("struct._protocol_t",
4153 ProtocolnfABITy);
Daniel Dunbar948e2582009-02-15 07:36:20 +00004154
4155 // struct _protocol_t*
Owen Anderson96e0fc72009-07-29 22:16:19 +00004156 ProtocolnfABIPtrTy = llvm::PointerType::getUnqual(ProtocolnfABITy);
Daniel Dunbar6bff2512009-08-03 17:06:42 +00004157
Fariborz Jahanianda320092009-01-29 19:24:30 +00004158 // struct _protocol_list_t {
Fariborz Jahaniand55b6fc2009-01-23 01:46:23 +00004159 // long protocol_count; // Note, this is 32/64 bit
Daniel Dunbar948e2582009-02-15 07:36:20 +00004160 // struct _protocol_t *[protocol_count];
Fariborz Jahanian30bc5712009-01-22 23:02:58 +00004161 // }
Owen Anderson47a434f2009-08-05 23:18:46 +00004162 ProtocolListnfABITy = llvm::StructType::get(VMContext, LongTy,
Owen Anderson96e0fc72009-07-29 22:16:19 +00004163 llvm::ArrayType::get(
Daniel Dunbar948e2582009-02-15 07:36:20 +00004164 ProtocolnfABIPtrTy, 0),
Fariborz Jahaniand55b6fc2009-01-23 01:46:23 +00004165 NULL);
4166 CGM.getModule().addTypeName("struct._objc_protocol_list",
4167 ProtocolListnfABITy);
Daniel Dunbar948e2582009-02-15 07:36:20 +00004168 cast<llvm::OpaqueType>(ProtocolListTyHolder.get())->refineAbstractTypeTo(
Daniel Dunbar6bff2512009-08-03 17:06:42 +00004169 ProtocolListnfABITy);
4170
Fariborz Jahaniand55b6fc2009-01-23 01:46:23 +00004171 // struct _objc_protocol_list*
Owen Anderson96e0fc72009-07-29 22:16:19 +00004172 ProtocolListnfABIPtrTy = llvm::PointerType::getUnqual(ProtocolListnfABITy);
Daniel Dunbar6bff2512009-08-03 17:06:42 +00004173
Fariborz Jahanian30bc5712009-01-22 23:02:58 +00004174 // struct _ivar_t {
4175 // unsigned long int *offset; // pointer to ivar offset location
4176 // char *name;
4177 // char *type;
4178 // uint32_t alignment;
4179 // uint32_t size;
4180 // }
Mike Stump1eb44332009-09-09 15:08:12 +00004181 IvarnfABITy = llvm::StructType::get(VMContext,
Owen Anderson47a434f2009-08-05 23:18:46 +00004182 llvm::PointerType::getUnqual(LongTy),
Fariborz Jahaniand55b6fc2009-01-23 01:46:23 +00004183 Int8PtrTy,
4184 Int8PtrTy,
4185 IntTy,
4186 IntTy,
4187 NULL);
4188 CGM.getModule().addTypeName("struct._ivar_t", IvarnfABITy);
Daniel Dunbar6bff2512009-08-03 17:06:42 +00004189
Fariborz Jahanian30bc5712009-01-22 23:02:58 +00004190 // struct _ivar_list_t {
4191 // uint32 entsize; // sizeof(struct _ivar_t)
4192 // uint32 count;
4193 // struct _iver_t list[count];
4194 // }
Owen Anderson47a434f2009-08-05 23:18:46 +00004195 IvarListnfABITy = llvm::StructType::get(VMContext, IntTy,
Fariborz Jahanian058a1b72009-01-24 20:21:50 +00004196 IntTy,
Owen Anderson96e0fc72009-07-29 22:16:19 +00004197 llvm::ArrayType::get(
Daniel Dunbar6bff2512009-08-03 17:06:42 +00004198 IvarnfABITy, 0),
Fariborz Jahanian058a1b72009-01-24 20:21:50 +00004199 NULL);
4200 CGM.getModule().addTypeName("struct._ivar_list_t", IvarListnfABITy);
Daniel Dunbar6bff2512009-08-03 17:06:42 +00004201
Owen Anderson96e0fc72009-07-29 22:16:19 +00004202 IvarListnfABIPtrTy = llvm::PointerType::getUnqual(IvarListnfABITy);
Daniel Dunbar6bff2512009-08-03 17:06:42 +00004203
Fariborz Jahaniand55b6fc2009-01-23 01:46:23 +00004204 // struct _class_ro_t {
Fariborz Jahanian30bc5712009-01-22 23:02:58 +00004205 // uint32_t const flags;
4206 // uint32_t const instanceStart;
4207 // uint32_t const instanceSize;
4208 // uint32_t const reserved; // only when building for 64bit targets
4209 // const uint8_t * const ivarLayout;
4210 // const char *const name;
4211 // const struct _method_list_t * const baseMethods;
4212 // const struct _objc_protocol_list *const baseProtocols;
4213 // const struct _ivar_list_t *const ivars;
4214 // const uint8_t * const weakIvarLayout;
4215 // const struct _prop_list_t * const properties;
4216 // }
Daniel Dunbar6bff2512009-08-03 17:06:42 +00004217
Fariborz Jahaniand55b6fc2009-01-23 01:46:23 +00004218 // FIXME. Add 'reserved' field in 64bit abi mode!
Owen Anderson47a434f2009-08-05 23:18:46 +00004219 ClassRonfABITy = llvm::StructType::get(VMContext, IntTy,
Fariborz Jahaniand55b6fc2009-01-23 01:46:23 +00004220 IntTy,
4221 IntTy,
4222 Int8PtrTy,
4223 Int8PtrTy,
4224 MethodListnfABIPtrTy,
4225 ProtocolListnfABIPtrTy,
4226 IvarListnfABIPtrTy,
4227 Int8PtrTy,
4228 PropertyListPtrTy,
4229 NULL);
4230 CGM.getModule().addTypeName("struct._class_ro_t",
4231 ClassRonfABITy);
Daniel Dunbar6bff2512009-08-03 17:06:42 +00004232
Fariborz Jahaniand55b6fc2009-01-23 01:46:23 +00004233 // ImpnfABITy - LLVM for id (*)(id, SEL, ...)
4234 std::vector<const llvm::Type*> Params;
4235 Params.push_back(ObjectPtrTy);
4236 Params.push_back(SelectorPtrTy);
Owen Anderson96e0fc72009-07-29 22:16:19 +00004237 ImpnfABITy = llvm::PointerType::getUnqual(
Daniel Dunbar6bff2512009-08-03 17:06:42 +00004238 llvm::FunctionType::get(ObjectPtrTy, Params, false));
4239
Fariborz Jahaniand55b6fc2009-01-23 01:46:23 +00004240 // struct _class_t {
4241 // struct _class_t *isa;
4242 // struct _class_t * const superclass;
4243 // void *cache;
4244 // IMP *vtable;
4245 // struct class_ro_t *ro;
4246 // }
Daniel Dunbar6bff2512009-08-03 17:06:42 +00004247
Owen Anderson8c8f69e2009-08-13 23:27:53 +00004248 llvm::PATypeHolder ClassTyHolder = llvm::OpaqueType::get(VMContext);
Owen Andersona1cf15f2009-07-14 23:10:40 +00004249 ClassnfABITy =
Owen Anderson47a434f2009-08-05 23:18:46 +00004250 llvm::StructType::get(VMContext,
4251 llvm::PointerType::getUnqual(ClassTyHolder),
Daniel Dunbar6bff2512009-08-03 17:06:42 +00004252 llvm::PointerType::getUnqual(ClassTyHolder),
4253 CachePtrTy,
4254 llvm::PointerType::getUnqual(ImpnfABITy),
4255 llvm::PointerType::getUnqual(ClassRonfABITy),
4256 NULL);
Fariborz Jahaniand55b6fc2009-01-23 01:46:23 +00004257 CGM.getModule().addTypeName("struct._class_t", ClassnfABITy);
4258
4259 cast<llvm::OpaqueType>(ClassTyHolder.get())->refineAbstractTypeTo(
Daniel Dunbar6bff2512009-08-03 17:06:42 +00004260 ClassnfABITy);
4261
Fariborz Jahanianaa23b572009-01-23 23:53:38 +00004262 // LLVM for struct _class_t *
Owen Anderson96e0fc72009-07-29 22:16:19 +00004263 ClassnfABIPtrTy = llvm::PointerType::getUnqual(ClassnfABITy);
Daniel Dunbar6bff2512009-08-03 17:06:42 +00004264
Fariborz Jahaniand55b6fc2009-01-23 01:46:23 +00004265 // struct _category_t {
4266 // const char * const name;
4267 // struct _class_t *const cls;
4268 // const struct _method_list_t * const instance_methods;
4269 // const struct _method_list_t * const class_methods;
4270 // const struct _protocol_list_t * const protocols;
4271 // const struct _prop_list_t * const properties;
Fariborz Jahanian45c2ba02009-01-23 17:41:22 +00004272 // }
Owen Anderson47a434f2009-08-05 23:18:46 +00004273 CategorynfABITy = llvm::StructType::get(VMContext, Int8PtrTy,
Fariborz Jahanianaa23b572009-01-23 23:53:38 +00004274 ClassnfABIPtrTy,
Fariborz Jahanian45c2ba02009-01-23 17:41:22 +00004275 MethodListnfABIPtrTy,
4276 MethodListnfABIPtrTy,
4277 ProtocolListnfABIPtrTy,
4278 PropertyListPtrTy,
4279 NULL);
4280 CGM.getModule().addTypeName("struct._category_t", CategorynfABITy);
Daniel Dunbar6bff2512009-08-03 17:06:42 +00004281
Fariborz Jahanian2e4672b2009-02-03 23:49:23 +00004282 // New types for nonfragile abi messaging.
Fariborz Jahanian83a8a752009-02-04 20:42:28 +00004283 CodeGen::CodeGenTypes &Types = CGM.getTypes();
4284 ASTContext &Ctx = CGM.getContext();
Daniel Dunbar6bff2512009-08-03 17:06:42 +00004285
Fariborz Jahanian2e4672b2009-02-03 23:49:23 +00004286 // MessageRefTy - LLVM for:
4287 // struct _message_ref_t {
4288 // IMP messenger;
4289 // SEL name;
4290 // };
Daniel Dunbar6bff2512009-08-03 17:06:42 +00004291
Fariborz Jahanian83a8a752009-02-04 20:42:28 +00004292 // First the clang type for struct _message_ref_t
Abramo Bagnara465d41b2010-05-11 21:36:43 +00004293 RecordDecl *RD = RecordDecl::Create(Ctx, TTK_Struct,
Daniel Dunbardaa3ac52010-04-29 16:29:11 +00004294 Ctx.getTranslationUnitDecl(),
Fariborz Jahanian83a8a752009-02-04 20:42:28 +00004295 SourceLocation(),
4296 &Ctx.Idents.get("_message_ref_t"));
Argyrios Kyrtzidis17945a02009-06-30 02:36:12 +00004297 RD->addDecl(FieldDecl::Create(Ctx, RD, SourceLocation(), 0,
Argyrios Kyrtzidisa1d56622009-08-19 01:27:57 +00004298 Ctx.VoidPtrTy, 0, 0, false));
Argyrios Kyrtzidis17945a02009-06-30 02:36:12 +00004299 RD->addDecl(FieldDecl::Create(Ctx, RD, SourceLocation(), 0,
Argyrios Kyrtzidisa1d56622009-08-19 01:27:57 +00004300 Ctx.getObjCSelType(), 0, 0, false));
Douglas Gregor838db382010-02-11 01:19:42 +00004301 RD->completeDefinition();
Daniel Dunbar6bff2512009-08-03 17:06:42 +00004302
Fariborz Jahanian83a8a752009-02-04 20:42:28 +00004303 MessageRefCTy = Ctx.getTagDeclType(RD);
4304 MessageRefCPtrTy = Ctx.getPointerType(MessageRefCTy);
4305 MessageRefTy = cast<llvm::StructType>(Types.ConvertType(MessageRefCTy));
Daniel Dunbar6bff2512009-08-03 17:06:42 +00004306
Fariborz Jahanian2e4672b2009-02-03 23:49:23 +00004307 // MessageRefPtrTy - LLVM for struct _message_ref_t*
Owen Anderson96e0fc72009-07-29 22:16:19 +00004308 MessageRefPtrTy = llvm::PointerType::getUnqual(MessageRefTy);
Daniel Dunbar6bff2512009-08-03 17:06:42 +00004309
Fariborz Jahanian2e4672b2009-02-03 23:49:23 +00004310 // SuperMessageRefTy - LLVM for:
4311 // struct _super_message_ref_t {
4312 // SUPER_IMP messenger;
4313 // SEL name;
4314 // };
Owen Anderson47a434f2009-08-05 23:18:46 +00004315 SuperMessageRefTy = llvm::StructType::get(VMContext, ImpnfABITy,
Fariborz Jahanian2e4672b2009-02-03 23:49:23 +00004316 SelectorPtrTy,
4317 NULL);
4318 CGM.getModule().addTypeName("struct._super_message_ref_t", SuperMessageRefTy);
Daniel Dunbar6bff2512009-08-03 17:06:42 +00004319
Fariborz Jahanian2e4672b2009-02-03 23:49:23 +00004320 // SuperMessageRefPtrTy - LLVM for struct _super_message_ref_t*
Daniel Dunbar6bff2512009-08-03 17:06:42 +00004321 SuperMessageRefPtrTy = llvm::PointerType::getUnqual(SuperMessageRefTy);
4322
Daniel Dunbare588b992009-03-01 04:46:24 +00004323
4324 // struct objc_typeinfo {
4325 // const void** vtable; // objc_ehtype_vtable + 2
4326 // const char* name; // c++ typeinfo string
4327 // Class cls;
4328 // };
Owen Anderson47a434f2009-08-05 23:18:46 +00004329 EHTypeTy = llvm::StructType::get(VMContext,
4330 llvm::PointerType::getUnqual(Int8PtrTy),
Daniel Dunbare588b992009-03-01 04:46:24 +00004331 Int8PtrTy,
4332 ClassnfABIPtrTy,
4333 NULL);
Daniel Dunbar4ff36842009-03-02 06:08:11 +00004334 CGM.getModule().addTypeName("struct._objc_typeinfo", EHTypeTy);
Owen Anderson96e0fc72009-07-29 22:16:19 +00004335 EHTypePtrTy = llvm::PointerType::getUnqual(EHTypeTy);
Daniel Dunbarbbce49b2008-08-12 00:12:39 +00004336}
4337
Daniel Dunbar6bff2512009-08-03 17:06:42 +00004338llvm::Function *CGObjCNonFragileABIMac::ModuleInitFunction() {
Fariborz Jahanianaa23b572009-01-23 23:53:38 +00004339 FinishNonFragileABIModule();
Daniel Dunbar6bff2512009-08-03 17:06:42 +00004340
Fariborz Jahanianaa23b572009-01-23 23:53:38 +00004341 return NULL;
4342}
4343
Daniel Dunbar6bff2512009-08-03 17:06:42 +00004344void CGObjCNonFragileABIMac::AddModuleClassList(const
4345 std::vector<llvm::GlobalValue*>
4346 &Container,
Daniel Dunbar463b8762009-05-15 21:48:48 +00004347 const char *SymbolName,
4348 const char *SectionName) {
4349 unsigned NumClasses = Container.size();
Daniel Dunbar6bff2512009-08-03 17:06:42 +00004350
Daniel Dunbar463b8762009-05-15 21:48:48 +00004351 if (!NumClasses)
4352 return;
Daniel Dunbar6bff2512009-08-03 17:06:42 +00004353
Daniel Dunbar463b8762009-05-15 21:48:48 +00004354 std::vector<llvm::Constant*> Symbols(NumClasses);
4355 for (unsigned i=0; i<NumClasses; i++)
Owen Anderson3c4972d2009-07-29 18:54:39 +00004356 Symbols[i] = llvm::ConstantExpr::getBitCast(Container[i],
Daniel Dunbar463b8762009-05-15 21:48:48 +00004357 ObjCTypes.Int8PtrTy);
Daniel Dunbar6bff2512009-08-03 17:06:42 +00004358 llvm::Constant* Init =
Owen Anderson96e0fc72009-07-29 22:16:19 +00004359 llvm::ConstantArray::get(llvm::ArrayType::get(ObjCTypes.Int8PtrTy,
Daniel Dunbar463b8762009-05-15 21:48:48 +00004360 NumClasses),
4361 Symbols);
Daniel Dunbar6bff2512009-08-03 17:06:42 +00004362
Daniel Dunbar463b8762009-05-15 21:48:48 +00004363 llvm::GlobalVariable *GV =
Owen Anderson1c431b32009-07-08 19:05:04 +00004364 new llvm::GlobalVariable(CGM.getModule(), Init->getType(), false,
Daniel Dunbar463b8762009-05-15 21:48:48 +00004365 llvm::GlobalValue::InternalLinkage,
4366 Init,
Owen Anderson1c431b32009-07-08 19:05:04 +00004367 SymbolName);
Daniel Dunbar4b429ae2010-04-25 20:39:32 +00004368 GV->setAlignment(CGM.getTargetData().getABITypeAlignment(Init->getType()));
Daniel Dunbar463b8762009-05-15 21:48:48 +00004369 GV->setSection(SectionName);
Chris Lattnerad64e022009-07-17 23:57:13 +00004370 CGM.AddUsedGlobal(GV);
Daniel Dunbar463b8762009-05-15 21:48:48 +00004371}
Daniel Dunbar6bff2512009-08-03 17:06:42 +00004372
Fariborz Jahanianaa23b572009-01-23 23:53:38 +00004373void CGObjCNonFragileABIMac::FinishNonFragileABIModule() {
4374 // nonfragile abi has no module definition.
Daniel Dunbar6bff2512009-08-03 17:06:42 +00004375
Daniel Dunbar463b8762009-05-15 21:48:48 +00004376 // Build list of all implemented class addresses in array
Fariborz Jahanianf87a0cc2009-01-30 20:55:31 +00004377 // L_OBJC_LABEL_CLASS_$.
Daniel Dunbar6bff2512009-08-03 17:06:42 +00004378 AddModuleClassList(DefinedClasses,
Daniel Dunbar463b8762009-05-15 21:48:48 +00004379 "\01L_OBJC_LABEL_CLASS_$",
4380 "__DATA, __objc_classlist, regular, no_dead_strip");
Fariborz Jahaniana03d0dd2009-11-17 21:37:35 +00004381
Fariborz Jahaniana03d0dd2009-11-17 21:37:35 +00004382 for (unsigned i = 0; i < DefinedClasses.size(); i++) {
4383 llvm::GlobalValue *IMPLGV = DefinedClasses[i];
4384 if (IMPLGV->getLinkage() != llvm::GlobalValue::ExternalWeakLinkage)
4385 continue;
4386 IMPLGV->setLinkage(llvm::GlobalValue::ExternalLinkage);
Fariborz Jahaniana03d0dd2009-11-17 21:37:35 +00004387 }
4388
Fariborz Jahanian0e93d252009-12-01 18:25:24 +00004389 for (unsigned i = 0; i < DefinedMetaClasses.size(); i++) {
4390 llvm::GlobalValue *IMPLGV = DefinedMetaClasses[i];
4391 if (IMPLGV->getLinkage() != llvm::GlobalValue::ExternalWeakLinkage)
4392 continue;
4393 IMPLGV->setLinkage(llvm::GlobalValue::ExternalLinkage);
4394 }
Fariborz Jahaniana03d0dd2009-11-17 21:37:35 +00004395
Daniel Dunbar6bff2512009-08-03 17:06:42 +00004396 AddModuleClassList(DefinedNonLazyClasses,
Daniel Dunbar74d4b122009-05-15 22:33:15 +00004397 "\01L_OBJC_LABEL_NONLAZY_CLASS_$",
4398 "__DATA, __objc_nlclslist, regular, no_dead_strip");
Daniel Dunbar6bff2512009-08-03 17:06:42 +00004399
Fariborz Jahanianf87a0cc2009-01-30 20:55:31 +00004400 // Build list of all implemented category addresses in array
4401 // L_OBJC_LABEL_CATEGORY_$.
Daniel Dunbar6bff2512009-08-03 17:06:42 +00004402 AddModuleClassList(DefinedCategories,
Daniel Dunbar463b8762009-05-15 21:48:48 +00004403 "\01L_OBJC_LABEL_CATEGORY_$",
4404 "__DATA, __objc_catlist, regular, no_dead_strip");
Daniel Dunbar6bff2512009-08-03 17:06:42 +00004405 AddModuleClassList(DefinedNonLazyCategories,
Daniel Dunbar74d4b122009-05-15 22:33:15 +00004406 "\01L_OBJC_LABEL_NONLAZY_CATEGORY_$",
4407 "__DATA, __objc_nlcatlist, regular, no_dead_strip");
Daniel Dunbar6bff2512009-08-03 17:06:42 +00004408
Daniel Dunbarfce176b2010-04-25 20:39:01 +00004409 EmitImageInfo();
Fariborz Jahanianaa23b572009-01-23 23:53:38 +00004410}
4411
Fariborz Jahaniand0f8a8d2009-05-11 19:25:47 +00004412/// LegacyDispatchedSelector - Returns true if SEL is not in the list of
Fariborz Jahanian4523eb02009-05-12 20:06:41 +00004413/// NonLegacyDispatchMethods; false otherwise. What this means is that
Daniel Dunbar6bff2512009-08-03 17:06:42 +00004414/// except for the 19 selectors in the list, we generate 32bit-style
Fariborz Jahaniand0f8a8d2009-05-11 19:25:47 +00004415/// message dispatch call for all the rest.
4416///
4417bool CGObjCNonFragileABIMac::LegacyDispatchedSelector(Selector Sel) {
Daniel Dunbarf643b9b2010-04-24 17:56:46 +00004418 switch (CGM.getCodeGenOpts().getObjCDispatchMethod()) {
4419 default:
4420 assert(0 && "Invalid dispatch method!");
4421 case CodeGenOptions::Legacy:
Daniel Dunbar2feefe82010-02-01 21:07:33 +00004422 return true;
Daniel Dunbarf643b9b2010-04-24 17:56:46 +00004423 case CodeGenOptions::NonLegacy:
Fariborz Jahanian776dbf92010-04-19 17:53:30 +00004424 return false;
Daniel Dunbarf643b9b2010-04-24 17:56:46 +00004425 case CodeGenOptions::Mixed:
4426 break;
4427 }
4428
4429 // If so, see whether this selector is in the white-list of things which must
4430 // use the new dispatch convention. We lazily build a dense set for this.
Fariborz Jahanian4523eb02009-05-12 20:06:41 +00004431 if (NonLegacyDispatchMethods.empty()) {
4432 NonLegacyDispatchMethods.insert(GetNullarySelector("alloc"));
4433 NonLegacyDispatchMethods.insert(GetNullarySelector("class"));
4434 NonLegacyDispatchMethods.insert(GetNullarySelector("self"));
4435 NonLegacyDispatchMethods.insert(GetNullarySelector("isFlipped"));
4436 NonLegacyDispatchMethods.insert(GetNullarySelector("length"));
4437 NonLegacyDispatchMethods.insert(GetNullarySelector("count"));
4438 NonLegacyDispatchMethods.insert(GetNullarySelector("retain"));
4439 NonLegacyDispatchMethods.insert(GetNullarySelector("release"));
4440 NonLegacyDispatchMethods.insert(GetNullarySelector("autorelease"));
4441 NonLegacyDispatchMethods.insert(GetNullarySelector("hash"));
Daniel Dunbar6bff2512009-08-03 17:06:42 +00004442
Fariborz Jahanian4523eb02009-05-12 20:06:41 +00004443 NonLegacyDispatchMethods.insert(GetUnarySelector("allocWithZone"));
4444 NonLegacyDispatchMethods.insert(GetUnarySelector("isKindOfClass"));
4445 NonLegacyDispatchMethods.insert(GetUnarySelector("respondsToSelector"));
4446 NonLegacyDispatchMethods.insert(GetUnarySelector("objectForKey"));
4447 NonLegacyDispatchMethods.insert(GetUnarySelector("objectAtIndex"));
4448 NonLegacyDispatchMethods.insert(GetUnarySelector("isEqualToString"));
4449 NonLegacyDispatchMethods.insert(GetUnarySelector("isEqual"));
4450 NonLegacyDispatchMethods.insert(GetUnarySelector("addObject"));
Daniel Dunbar6bff2512009-08-03 17:06:42 +00004451 // "countByEnumeratingWithState:objects:count"
Fariborz Jahanianbe53be42009-05-13 16:19:02 +00004452 IdentifierInfo *KeyIdents[] = {
Daniel Dunbar6bff2512009-08-03 17:06:42 +00004453 &CGM.getContext().Idents.get("countByEnumeratingWithState"),
4454 &CGM.getContext().Idents.get("objects"),
4455 &CGM.getContext().Idents.get("count")
Fariborz Jahanianbe53be42009-05-13 16:19:02 +00004456 };
4457 NonLegacyDispatchMethods.insert(
4458 CGM.getContext().Selectors.getSelector(3, KeyIdents));
Fariborz Jahaniand0f8a8d2009-05-11 19:25:47 +00004459 }
Daniel Dunbarf643b9b2010-04-24 17:56:46 +00004460
Fariborz Jahanian4523eb02009-05-12 20:06:41 +00004461 return (NonLegacyDispatchMethods.count(Sel) == 0);
Fariborz Jahaniand0f8a8d2009-05-11 19:25:47 +00004462}
4463
Fariborz Jahanian058a1b72009-01-24 20:21:50 +00004464// Metadata flags
4465enum MetaDataDlags {
4466 CLS = 0x0,
4467 CLS_META = 0x1,
4468 CLS_ROOT = 0x2,
Fariborz Jahanian493dab72009-01-26 21:38:32 +00004469 OBJC2_CLS_HIDDEN = 0x10,
Fariborz Jahanian058a1b72009-01-24 20:21:50 +00004470 CLS_EXCEPTION = 0x20
4471};
4472/// BuildClassRoTInitializer - generate meta-data for:
4473/// struct _class_ro_t {
4474/// uint32_t const flags;
4475/// uint32_t const instanceStart;
4476/// uint32_t const instanceSize;
4477/// uint32_t const reserved; // only when building for 64bit targets
4478/// const uint8_t * const ivarLayout;
4479/// const char *const name;
4480/// const struct _method_list_t * const baseMethods;
Fariborz Jahanianda320092009-01-29 19:24:30 +00004481/// const struct _protocol_list_t *const baseProtocols;
Fariborz Jahanian058a1b72009-01-24 20:21:50 +00004482/// const struct _ivar_list_t *const ivars;
4483/// const uint8_t * const weakIvarLayout;
4484/// const struct _prop_list_t * const properties;
4485/// }
4486///
4487llvm::GlobalVariable * CGObjCNonFragileABIMac::BuildClassRoTInitializer(
Daniel Dunbar6bff2512009-08-03 17:06:42 +00004488 unsigned flags,
4489 unsigned InstanceStart,
4490 unsigned InstanceSize,
4491 const ObjCImplementationDecl *ID) {
Fariborz Jahanian058a1b72009-01-24 20:21:50 +00004492 std::string ClassName = ID->getNameAsString();
4493 std::vector<llvm::Constant*> Values(10); // 11 for 64bit targets!
Owen Anderson4a28d5d2009-07-24 23:12:58 +00004494 Values[ 0] = llvm::ConstantInt::get(ObjCTypes.IntTy, flags);
4495 Values[ 1] = llvm::ConstantInt::get(ObjCTypes.IntTy, InstanceStart);
4496 Values[ 2] = llvm::ConstantInt::get(ObjCTypes.IntTy, InstanceSize);
Fariborz Jahanian058a1b72009-01-24 20:21:50 +00004497 // FIXME. For 64bit targets add 0 here.
Daniel Dunbar6bff2512009-08-03 17:06:42 +00004498 Values[ 3] = (flags & CLS_META) ? GetIvarLayoutName(0, ObjCTypes)
4499 : BuildIvarLayout(ID, true);
Fariborz Jahanian058a1b72009-01-24 20:21:50 +00004500 Values[ 4] = GetClassName(ID->getIdentifier());
Fariborz Jahanian493dab72009-01-26 21:38:32 +00004501 // const struct _method_list_t * const baseMethods;
4502 std::vector<llvm::Constant*> Methods;
4503 std::string MethodListName("\01l_OBJC_$_");
4504 if (flags & CLS_META) {
4505 MethodListName += "CLASS_METHODS_" + ID->getNameAsString();
Daniel Dunbar6bff2512009-08-03 17:06:42 +00004506 for (ObjCImplementationDecl::classmeth_iterator
Argyrios Kyrtzidis17945a02009-06-30 02:36:12 +00004507 i = ID->classmeth_begin(), e = ID->classmeth_end(); i != e; ++i) {
Fariborz Jahanian493dab72009-01-26 21:38:32 +00004508 // Class methods should always be defined.
4509 Methods.push_back(GetMethodConstant(*i));
4510 }
4511 } else {
4512 MethodListName += "INSTANCE_METHODS_" + ID->getNameAsString();
Daniel Dunbar6bff2512009-08-03 17:06:42 +00004513 for (ObjCImplementationDecl::instmeth_iterator
Argyrios Kyrtzidis17945a02009-06-30 02:36:12 +00004514 i = ID->instmeth_begin(), e = ID->instmeth_end(); i != e; ++i) {
Fariborz Jahanian493dab72009-01-26 21:38:32 +00004515 // Instance methods should always be defined.
4516 Methods.push_back(GetMethodConstant(*i));
4517 }
Daniel Dunbar6bff2512009-08-03 17:06:42 +00004518 for (ObjCImplementationDecl::propimpl_iterator
Argyrios Kyrtzidis17945a02009-06-30 02:36:12 +00004519 i = ID->propimpl_begin(), e = ID->propimpl_end(); i != e; ++i) {
Fariborz Jahanian939abce2009-01-28 22:46:49 +00004520 ObjCPropertyImplDecl *PID = *i;
Daniel Dunbar6bff2512009-08-03 17:06:42 +00004521
Fariborz Jahanian939abce2009-01-28 22:46:49 +00004522 if (PID->getPropertyImplementation() == ObjCPropertyImplDecl::Synthesize){
4523 ObjCPropertyDecl *PD = PID->getPropertyDecl();
Daniel Dunbar6bff2512009-08-03 17:06:42 +00004524
Fariborz Jahanian939abce2009-01-28 22:46:49 +00004525 if (ObjCMethodDecl *MD = PD->getGetterMethodDecl())
4526 if (llvm::Constant *C = GetMethodConstant(MD))
4527 Methods.push_back(C);
4528 if (ObjCMethodDecl *MD = PD->getSetterMethodDecl())
4529 if (llvm::Constant *C = GetMethodConstant(MD))
4530 Methods.push_back(C);
4531 }
4532 }
Fariborz Jahanian493dab72009-01-26 21:38:32 +00004533 }
Daniel Dunbar6bff2512009-08-03 17:06:42 +00004534 Values[ 5] = EmitMethodList(MethodListName,
4535 "__DATA, __objc_const", Methods);
4536
Fariborz Jahanianda320092009-01-29 19:24:30 +00004537 const ObjCInterfaceDecl *OID = ID->getClassInterface();
4538 assert(OID && "CGObjCNonFragileABIMac::BuildClassRoTInitializer");
Daniel Dunbar6bff2512009-08-03 17:06:42 +00004539 Values[ 6] = EmitProtocolList("\01l_OBJC_CLASS_PROTOCOLS_$_"
Daniel Dunbar9c29bf52009-10-18 20:48:59 +00004540 + OID->getName(),
Fariborz Jahanianda320092009-01-29 19:24:30 +00004541 OID->protocol_begin(),
4542 OID->protocol_end());
Daniel Dunbar6bff2512009-08-03 17:06:42 +00004543
Fariborz Jahanian98abf4b2009-01-27 19:38:51 +00004544 if (flags & CLS_META)
Owen Andersonc9c88b42009-07-31 20:28:54 +00004545 Values[ 7] = llvm::Constant::getNullValue(ObjCTypes.IvarListnfABIPtrTy);
Fariborz Jahanian98abf4b2009-01-27 19:38:51 +00004546 else
4547 Values[ 7] = EmitIvarList(ID);
Daniel Dunbar6bff2512009-08-03 17:06:42 +00004548 Values[ 8] = (flags & CLS_META) ? GetIvarLayoutName(0, ObjCTypes)
4549 : BuildIvarLayout(ID, false);
Fariborz Jahanian5de14dc2009-01-28 22:18:42 +00004550 if (flags & CLS_META)
Owen Andersonc9c88b42009-07-31 20:28:54 +00004551 Values[ 9] = llvm::Constant::getNullValue(ObjCTypes.PropertyListPtrTy);
Fariborz Jahanian5de14dc2009-01-28 22:18:42 +00004552 else
Daniel Dunbar9c29bf52009-10-18 20:48:59 +00004553 Values[ 9] = EmitPropertyList("\01l_OBJC_$_PROP_LIST_" + ID->getName(),
4554 ID, ID->getClassInterface(), ObjCTypes);
Owen Anderson08e25242009-07-27 22:29:56 +00004555 llvm::Constant *Init = llvm::ConstantStruct::get(ObjCTypes.ClassRonfABITy,
Fariborz Jahanian058a1b72009-01-24 20:21:50 +00004556 Values);
4557 llvm::GlobalVariable *CLASS_RO_GV =
Daniel Dunbar6bff2512009-08-03 17:06:42 +00004558 new llvm::GlobalVariable(CGM.getModule(), ObjCTypes.ClassRonfABITy, false,
4559 llvm::GlobalValue::InternalLinkage,
4560 Init,
4561 (flags & CLS_META) ?
4562 std::string("\01l_OBJC_METACLASS_RO_$_")+ClassName :
4563 std::string("\01l_OBJC_CLASS_RO_$_")+ClassName);
Fariborz Jahanian09796d62009-01-31 02:43:27 +00004564 CLASS_RO_GV->setAlignment(
Daniel Dunbar4b429ae2010-04-25 20:39:32 +00004565 CGM.getTargetData().getABITypeAlignment(ObjCTypes.ClassRonfABITy));
Fariborz Jahanian1bf0afb2009-01-28 01:05:23 +00004566 CLASS_RO_GV->setSection("__DATA, __objc_const");
Fariborz Jahanian058a1b72009-01-24 20:21:50 +00004567 return CLASS_RO_GV;
Fariborz Jahanianf6317dd2009-01-26 22:58:07 +00004568
Fariborz Jahanian058a1b72009-01-24 20:21:50 +00004569}
4570
4571/// BuildClassMetaData - This routine defines that to-level meta-data
4572/// for the given ClassName for:
4573/// struct _class_t {
4574/// struct _class_t *isa;
4575/// struct _class_t * const superclass;
4576/// void *cache;
4577/// IMP *vtable;
4578/// struct class_ro_t *ro;
4579/// }
4580///
Fariborz Jahanian84394a52009-01-24 21:21:53 +00004581llvm::GlobalVariable * CGObjCNonFragileABIMac::BuildClassMetaData(
Daniel Dunbar6bff2512009-08-03 17:06:42 +00004582 std::string &ClassName,
4583 llvm::Constant *IsAGV,
4584 llvm::Constant *SuperClassGV,
4585 llvm::Constant *ClassRoGV,
4586 bool HiddenVisibility) {
Fariborz Jahanian058a1b72009-01-24 20:21:50 +00004587 std::vector<llvm::Constant*> Values(5);
4588 Values[0] = IsAGV;
Daniel Dunbar6bff2512009-08-03 17:06:42 +00004589 Values[1] = SuperClassGV;
4590 if (!Values[1])
4591 Values[1] = llvm::Constant::getNullValue(ObjCTypes.ClassnfABIPtrTy);
Fariborz Jahanian058a1b72009-01-24 20:21:50 +00004592 Values[2] = ObjCEmptyCacheVar; // &ObjCEmptyCacheVar
4593 Values[3] = ObjCEmptyVtableVar; // &ObjCEmptyVtableVar
4594 Values[4] = ClassRoGV; // &CLASS_RO_GV
Daniel Dunbar6bff2512009-08-03 17:06:42 +00004595 llvm::Constant *Init = llvm::ConstantStruct::get(ObjCTypes.ClassnfABITy,
Fariborz Jahanian058a1b72009-01-24 20:21:50 +00004596 Values);
Daniel Dunbar5a7379a2009-03-01 04:40:10 +00004597 llvm::GlobalVariable *GV = GetClassGlobal(ClassName);
4598 GV->setInitializer(Init);
Fariborz Jahaniandd0db2a2009-01-31 01:07:39 +00004599 GV->setSection("__DATA, __objc_data");
Fariborz Jahanian09796d62009-01-31 02:43:27 +00004600 GV->setAlignment(
Daniel Dunbar4b429ae2010-04-25 20:39:32 +00004601 CGM.getTargetData().getABITypeAlignment(ObjCTypes.ClassnfABITy));
Fariborz Jahaniancf555162009-01-31 00:59:10 +00004602 if (HiddenVisibility)
4603 GV->setVisibility(llvm::GlobalValue::HiddenVisibility);
Fariborz Jahanian84394a52009-01-24 21:21:53 +00004604 return GV;
Fariborz Jahanian058a1b72009-01-24 20:21:50 +00004605}
4606
Daniel Dunbar6bff2512009-08-03 17:06:42 +00004607bool
Fariborz Jahanianecfbdcb2009-05-21 01:03:45 +00004608CGObjCNonFragileABIMac::ImplementationIsNonLazy(const ObjCImplDecl *OD) const {
Argyrios Kyrtzidis17945a02009-06-30 02:36:12 +00004609 return OD->getClassMethod(GetNullarySelector("load")) != 0;
Daniel Dunbar74d4b122009-05-15 22:33:15 +00004610}
4611
Daniel Dunbar9f89f2b2009-05-03 12:57:56 +00004612void CGObjCNonFragileABIMac::GetClassSizeInfo(const ObjCImplementationDecl *OID,
Daniel Dunbarb02532a2009-04-19 23:41:48 +00004613 uint32_t &InstanceStart,
4614 uint32_t &InstanceSize) {
Daniel Dunbar6bff2512009-08-03 17:06:42 +00004615 const ASTRecordLayout &RL =
Daniel Dunbarb4c79e02009-05-04 21:26:30 +00004616 CGM.getContext().getASTObjCImplementationLayout(OID);
Daniel Dunbar6bff2512009-08-03 17:06:42 +00004617
Daniel Dunbar6e8575b2009-05-04 23:23:09 +00004618 // InstanceSize is really instance end.
Anders Carlsson243a6852009-07-18 21:26:44 +00004619 InstanceSize = llvm::RoundUpToAlignment(RL.getDataSize(), 8) / 8;
Daniel Dunbar6e8575b2009-05-04 23:23:09 +00004620
4621 // If there are no fields, the start is the same as the end.
4622 if (!RL.getFieldCount())
4623 InstanceStart = InstanceSize;
4624 else
4625 InstanceStart = RL.getFieldOffset(0) / 8;
Daniel Dunbarb02532a2009-04-19 23:41:48 +00004626}
4627
Fariborz Jahanianaa23b572009-01-23 23:53:38 +00004628void CGObjCNonFragileABIMac::GenerateClass(const ObjCImplementationDecl *ID) {
4629 std::string ClassName = ID->getNameAsString();
4630 if (!ObjCEmptyCacheVar) {
4631 ObjCEmptyCacheVar = new llvm::GlobalVariable(
Daniel Dunbar6bff2512009-08-03 17:06:42 +00004632 CGM.getModule(),
4633 ObjCTypes.CacheTy,
4634 false,
4635 llvm::GlobalValue::ExternalLinkage,
4636 0,
4637 "_objc_empty_cache");
4638
Fariborz Jahanianaa23b572009-01-23 23:53:38 +00004639 ObjCEmptyVtableVar = new llvm::GlobalVariable(
Daniel Dunbar6bff2512009-08-03 17:06:42 +00004640 CGM.getModule(),
4641 ObjCTypes.ImpnfABITy,
4642 false,
4643 llvm::GlobalValue::ExternalLinkage,
4644 0,
4645 "_objc_empty_vtable");
Fariborz Jahanianaa23b572009-01-23 23:53:38 +00004646 }
Daniel Dunbar6bff2512009-08-03 17:06:42 +00004647 assert(ID->getClassInterface() &&
Fariborz Jahanian493dab72009-01-26 21:38:32 +00004648 "CGObjCNonFragileABIMac::GenerateClass - class is 0");
Daniel Dunbar6c1aac82009-04-20 20:18:54 +00004649 // FIXME: Is this correct (that meta class size is never computed)?
Daniel Dunbar6bff2512009-08-03 17:06:42 +00004650 uint32_t InstanceStart =
Duncan Sands9408c452009-05-09 07:08:47 +00004651 CGM.getTargetData().getTypeAllocSize(ObjCTypes.ClassnfABITy);
Fariborz Jahanian058a1b72009-01-24 20:21:50 +00004652 uint32_t InstanceSize = InstanceStart;
4653 uint32_t flags = CLS_META;
Daniel Dunbar6ab187a2009-04-07 05:48:37 +00004654 std::string ObjCMetaClassName(getMetaclassSymbolPrefix());
4655 std::string ObjCClassName(getClassSymbolPrefix());
Daniel Dunbar6bff2512009-08-03 17:06:42 +00004656
Fariborz Jahanian058a1b72009-01-24 20:21:50 +00004657 llvm::GlobalVariable *SuperClassGV, *IsAGV;
Daniel Dunbar6bff2512009-08-03 17:06:42 +00004658
4659 bool classIsHidden =
Daniel Dunbar04d40782009-04-14 06:00:08 +00004660 CGM.getDeclVisibilityMode(ID->getClassInterface()) == LangOptions::Hidden;
Fariborz Jahaniancf555162009-01-31 00:59:10 +00004661 if (classIsHidden)
Fariborz Jahanian493dab72009-01-26 21:38:32 +00004662 flags |= OBJC2_CLS_HIDDEN;
Fariborz Jahanian109dfc62010-04-28 21:28:56 +00004663 if (ID->getNumIvarInitializers())
4664 flags |= eClassFlags_ABI2_HasCXXStructors;
Fariborz Jahanian493dab72009-01-26 21:38:32 +00004665 if (!ID->getClassInterface()->getSuperClass()) {
Fariborz Jahanian058a1b72009-01-24 20:21:50 +00004666 // class is root
4667 flags |= CLS_ROOT;
Daniel Dunbar5a7379a2009-03-01 04:40:10 +00004668 SuperClassGV = GetClassGlobal(ObjCClassName + ClassName);
Fariborz Jahanian0f902942009-04-14 18:41:56 +00004669 IsAGV = GetClassGlobal(ObjCMetaClassName + ClassName);
Fariborz Jahanian058a1b72009-01-24 20:21:50 +00004670 } else {
Fariborz Jahanian84394a52009-01-24 21:21:53 +00004671 // Has a root. Current class is not a root.
Fariborz Jahanianfab98c42009-02-26 18:23:47 +00004672 const ObjCInterfaceDecl *Root = ID->getClassInterface();
4673 while (const ObjCInterfaceDecl *Super = Root->getSuperClass())
4674 Root = Super;
Fariborz Jahanian0f902942009-04-14 18:41:56 +00004675 IsAGV = GetClassGlobal(ObjCMetaClassName + Root->getNameAsString());
Fariborz Jahanian0e93d252009-12-01 18:25:24 +00004676 if (Root->hasAttr<WeakImportAttr>())
4677 IsAGV->setLinkage(llvm::GlobalValue::ExternalWeakLinkage);
Fariborz Jahanianfab98c42009-02-26 18:23:47 +00004678 // work on super class metadata symbol.
Daniel Dunbar6bff2512009-08-03 17:06:42 +00004679 std::string SuperClassName =
Fariborz Jahanian0e93d252009-12-01 18:25:24 +00004680 ObjCMetaClassName +
4681 ID->getClassInterface()->getSuperClass()->getNameAsString();
Fariborz Jahanian0f902942009-04-14 18:41:56 +00004682 SuperClassGV = GetClassGlobal(SuperClassName);
Fariborz Jahaniana03d0dd2009-11-17 21:37:35 +00004683 if (ID->getClassInterface()->getSuperClass()->hasAttr<WeakImportAttr>())
4684 SuperClassGV->setLinkage(llvm::GlobalValue::ExternalWeakLinkage);
Fariborz Jahanian058a1b72009-01-24 20:21:50 +00004685 }
4686 llvm::GlobalVariable *CLASS_RO_GV = BuildClassRoTInitializer(flags,
4687 InstanceStart,
4688 InstanceSize,ID);
Fariborz Jahanian84394a52009-01-24 21:21:53 +00004689 std::string TClassName = ObjCMetaClassName + ClassName;
Daniel Dunbar6bff2512009-08-03 17:06:42 +00004690 llvm::GlobalVariable *MetaTClass =
Fariborz Jahaniancf555162009-01-31 00:59:10 +00004691 BuildClassMetaData(TClassName, IsAGV, SuperClassGV, CLASS_RO_GV,
4692 classIsHidden);
Fariborz Jahaniana03d0dd2009-11-17 21:37:35 +00004693 DefinedMetaClasses.push_back(MetaTClass);
Daniel Dunbar6ab187a2009-04-07 05:48:37 +00004694
Fariborz Jahanian84394a52009-01-24 21:21:53 +00004695 // Metadata for the class
4696 flags = CLS;
Fariborz Jahaniancf555162009-01-31 00:59:10 +00004697 if (classIsHidden)
Fariborz Jahanian493dab72009-01-26 21:38:32 +00004698 flags |= OBJC2_CLS_HIDDEN;
Fariborz Jahanian109dfc62010-04-28 21:28:56 +00004699 if (ID->getNumIvarInitializers())
4700 flags |= eClassFlags_ABI2_HasCXXStructors;
Daniel Dunbar8158a2f2009-04-08 04:21:03 +00004701
Douglas Gregor68584ed2009-06-18 16:11:24 +00004702 if (hasObjCExceptionAttribute(CGM.getContext(), ID->getClassInterface()))
Daniel Dunbar8158a2f2009-04-08 04:21:03 +00004703 flags |= CLS_EXCEPTION;
4704
Fariborz Jahanian493dab72009-01-26 21:38:32 +00004705 if (!ID->getClassInterface()->getSuperClass()) {
Fariborz Jahanian84394a52009-01-24 21:21:53 +00004706 flags |= CLS_ROOT;
4707 SuperClassGV = 0;
Chris Lattnerb7b58b12009-04-19 06:02:28 +00004708 } else {
Fariborz Jahanian84394a52009-01-24 21:21:53 +00004709 // Has a root. Current class is not a root.
Fariborz Jahanianfab98c42009-02-26 18:23:47 +00004710 std::string RootClassName =
Fariborz Jahanian84394a52009-01-24 21:21:53 +00004711 ID->getClassInterface()->getSuperClass()->getNameAsString();
Daniel Dunbar5a7379a2009-03-01 04:40:10 +00004712 SuperClassGV = GetClassGlobal(ObjCClassName + RootClassName);
Fariborz Jahaniana03d0dd2009-11-17 21:37:35 +00004713 if (ID->getClassInterface()->getSuperClass()->hasAttr<WeakImportAttr>())
4714 SuperClassGV->setLinkage(llvm::GlobalValue::ExternalWeakLinkage);
Fariborz Jahanian84394a52009-01-24 21:21:53 +00004715 }
Daniel Dunbar9f89f2b2009-05-03 12:57:56 +00004716 GetClassSizeInfo(ID, InstanceStart, InstanceSize);
Fariborz Jahanian84394a52009-01-24 21:21:53 +00004717 CLASS_RO_GV = BuildClassRoTInitializer(flags,
Fariborz Jahanianf6a077e2009-01-24 23:43:01 +00004718 InstanceStart,
Daniel Dunbar6bff2512009-08-03 17:06:42 +00004719 InstanceSize,
Fariborz Jahanianf6a077e2009-01-24 23:43:01 +00004720 ID);
Daniel Dunbar6bff2512009-08-03 17:06:42 +00004721
Fariborz Jahanian84394a52009-01-24 21:21:53 +00004722 TClassName = ObjCClassName + ClassName;
Daniel Dunbar6bff2512009-08-03 17:06:42 +00004723 llvm::GlobalVariable *ClassMD =
Fariborz Jahaniancf555162009-01-31 00:59:10 +00004724 BuildClassMetaData(TClassName, MetaTClass, SuperClassGV, CLASS_RO_GV,
4725 classIsHidden);
Fariborz Jahanianf87a0cc2009-01-30 20:55:31 +00004726 DefinedClasses.push_back(ClassMD);
Daniel Dunbar8158a2f2009-04-08 04:21:03 +00004727
Daniel Dunbar74d4b122009-05-15 22:33:15 +00004728 // Determine if this class is also "non-lazy".
4729 if (ImplementationIsNonLazy(ID))
4730 DefinedNonLazyClasses.push_back(ClassMD);
4731
Daniel Dunbar8158a2f2009-04-08 04:21:03 +00004732 // Force the definition of the EHType if necessary.
4733 if (flags & CLS_EXCEPTION)
4734 GetInterfaceEHType(ID->getClassInterface(), true);
Fariborz Jahanianaa23b572009-01-23 23:53:38 +00004735}
4736
Fariborz Jahanian8cfd3972009-01-30 18:58:59 +00004737/// GenerateProtocolRef - This routine is called to generate code for
4738/// a protocol reference expression; as in:
4739/// @code
4740/// @protocol(Proto1);
4741/// @endcode
4742/// It generates a weak reference to l_OBJC_PROTOCOL_REFERENCE_$_Proto1
4743/// which will hold address of the protocol meta-data.
4744///
4745llvm::Value *CGObjCNonFragileABIMac::GenerateProtocolRef(CGBuilderTy &Builder,
Daniel Dunbar6bff2512009-08-03 17:06:42 +00004746 const ObjCProtocolDecl *PD) {
4747
Fariborz Jahanian960cd062009-04-10 18:47:34 +00004748 // This routine is called for @protocol only. So, we must build definition
4749 // of protocol's meta-data (not a reference to it!)
4750 //
Daniel Dunbar6bff2512009-08-03 17:06:42 +00004751 llvm::Constant *Init =
4752 llvm::ConstantExpr::getBitCast(GetOrEmitProtocol(PD),
4753 ObjCTypes.ExternalProtocolPtrTy);
4754
Fariborz Jahanian8cfd3972009-01-30 18:58:59 +00004755 std::string ProtocolName("\01l_OBJC_PROTOCOL_REFERENCE_$_");
4756 ProtocolName += PD->getNameAsCString();
Daniel Dunbar6bff2512009-08-03 17:06:42 +00004757
Fariborz Jahanian8cfd3972009-01-30 18:58:59 +00004758 llvm::GlobalVariable *PTGV = CGM.getModule().getGlobalVariable(ProtocolName);
4759 if (PTGV)
Daniel Dunbar2da84ff2009-11-29 21:23:36 +00004760 return Builder.CreateLoad(PTGV, "tmp");
Fariborz Jahanian8cfd3972009-01-30 18:58:59 +00004761 PTGV = new llvm::GlobalVariable(
Daniel Dunbar6bff2512009-08-03 17:06:42 +00004762 CGM.getModule(),
4763 Init->getType(), false,
4764 llvm::GlobalValue::WeakAnyLinkage,
4765 Init,
4766 ProtocolName);
Fariborz Jahanian8cfd3972009-01-30 18:58:59 +00004767 PTGV->setSection("__DATA, __objc_protorefs, coalesced, no_dead_strip");
4768 PTGV->setVisibility(llvm::GlobalValue::HiddenVisibility);
Chris Lattnerad64e022009-07-17 23:57:13 +00004769 CGM.AddUsedGlobal(PTGV);
Daniel Dunbar2da84ff2009-11-29 21:23:36 +00004770 return Builder.CreateLoad(PTGV, "tmp");
Fariborz Jahanian8cfd3972009-01-30 18:58:59 +00004771}
4772
Fariborz Jahanianeb062d92009-01-26 18:32:24 +00004773/// GenerateCategory - Build metadata for a category implementation.
4774/// struct _category_t {
4775/// const char * const name;
4776/// struct _class_t *const cls;
4777/// const struct _method_list_t * const instance_methods;
4778/// const struct _method_list_t * const class_methods;
4779/// const struct _protocol_list_t * const protocols;
4780/// const struct _prop_list_t * const properties;
4781/// }
4782///
Daniel Dunbar74d4b122009-05-15 22:33:15 +00004783void CGObjCNonFragileABIMac::GenerateCategory(const ObjCCategoryImplDecl *OCD) {
Fariborz Jahanianeb062d92009-01-26 18:32:24 +00004784 const ObjCInterfaceDecl *Interface = OCD->getClassInterface();
Fariborz Jahanianf6317dd2009-01-26 22:58:07 +00004785 const char *Prefix = "\01l_OBJC_$_CATEGORY_";
Daniel Dunbar6bff2512009-08-03 17:06:42 +00004786 std::string ExtCatName(Prefix + Interface->getNameAsString()+
4787 "_$_" + OCD->getNameAsString());
4788 std::string ExtClassName(getClassSymbolPrefix() +
Daniel Dunbar6ab187a2009-04-07 05:48:37 +00004789 Interface->getNameAsString());
Daniel Dunbar6bff2512009-08-03 17:06:42 +00004790
Fariborz Jahanianeb062d92009-01-26 18:32:24 +00004791 std::vector<llvm::Constant*> Values(6);
4792 Values[0] = GetClassName(OCD->getIdentifier());
4793 // meta-class entry symbol
Daniel Dunbar5a7379a2009-03-01 04:40:10 +00004794 llvm::GlobalVariable *ClassGV = GetClassGlobal(ExtClassName);
Fariborz Jahanian2cdcc4c2009-11-17 22:02:21 +00004795 if (Interface->hasAttr<WeakImportAttr>())
4796 ClassGV->setLinkage(llvm::GlobalValue::ExternalWeakLinkage);
4797
Fariborz Jahanianeb062d92009-01-26 18:32:24 +00004798 Values[1] = ClassGV;
Fariborz Jahanianf6317dd2009-01-26 22:58:07 +00004799 std::vector<llvm::Constant*> Methods;
4800 std::string MethodListName(Prefix);
Daniel Dunbar6bff2512009-08-03 17:06:42 +00004801 MethodListName += "INSTANCE_METHODS_" + Interface->getNameAsString() +
Fariborz Jahanianf6317dd2009-01-26 22:58:07 +00004802 "_$_" + OCD->getNameAsString();
Daniel Dunbar6bff2512009-08-03 17:06:42 +00004803
4804 for (ObjCCategoryImplDecl::instmeth_iterator
Argyrios Kyrtzidis17945a02009-06-30 02:36:12 +00004805 i = OCD->instmeth_begin(), e = OCD->instmeth_end(); i != e; ++i) {
Fariborz Jahanianf6317dd2009-01-26 22:58:07 +00004806 // Instance methods should always be defined.
4807 Methods.push_back(GetMethodConstant(*i));
4808 }
Daniel Dunbar6bff2512009-08-03 17:06:42 +00004809
4810 Values[2] = EmitMethodList(MethodListName,
4811 "__DATA, __objc_const",
Fariborz Jahanianf6317dd2009-01-26 22:58:07 +00004812 Methods);
4813
4814 MethodListName = Prefix;
4815 MethodListName += "CLASS_METHODS_" + Interface->getNameAsString() + "_$_" +
4816 OCD->getNameAsString();
4817 Methods.clear();
Daniel Dunbar6bff2512009-08-03 17:06:42 +00004818 for (ObjCCategoryImplDecl::classmeth_iterator
Argyrios Kyrtzidis17945a02009-06-30 02:36:12 +00004819 i = OCD->classmeth_begin(), e = OCD->classmeth_end(); i != e; ++i) {
Fariborz Jahanianf6317dd2009-01-26 22:58:07 +00004820 // Class methods should always be defined.
4821 Methods.push_back(GetMethodConstant(*i));
4822 }
Daniel Dunbar6bff2512009-08-03 17:06:42 +00004823
4824 Values[3] = EmitMethodList(MethodListName,
4825 "__DATA, __objc_const",
Fariborz Jahanianf6317dd2009-01-26 22:58:07 +00004826 Methods);
Daniel Dunbar6bff2512009-08-03 17:06:42 +00004827 const ObjCCategoryDecl *Category =
Fariborz Jahanian5de14dc2009-01-28 22:18:42 +00004828 Interface->FindCategoryDeclaration(OCD->getIdentifier());
Fariborz Jahanian943ed6f2009-02-13 17:52:22 +00004829 if (Category) {
Daniel Dunbar9c29bf52009-10-18 20:48:59 +00004830 llvm::SmallString<256> ExtName;
4831 llvm::raw_svector_ostream(ExtName) << Interface->getName() << "_$_"
4832 << OCD->getName();
Fariborz Jahanian943ed6f2009-02-13 17:52:22 +00004833 Values[4] = EmitProtocolList("\01l_OBJC_CATEGORY_PROTOCOLS_$_"
Daniel Dunbar9c29bf52009-10-18 20:48:59 +00004834 + Interface->getName() + "_$_"
4835 + Category->getName(),
Fariborz Jahanian943ed6f2009-02-13 17:52:22 +00004836 Category->protocol_begin(),
4837 Category->protocol_end());
Daniel Dunbar9c29bf52009-10-18 20:48:59 +00004838 Values[5] = EmitPropertyList("\01l_OBJC_$_PROP_LIST_" + ExtName.str(),
4839 OCD, Category, ObjCTypes);
Mike Stumpb3589f42009-07-30 22:28:39 +00004840 } else {
Owen Andersonc9c88b42009-07-31 20:28:54 +00004841 Values[4] = llvm::Constant::getNullValue(ObjCTypes.ProtocolListnfABIPtrTy);
4842 Values[5] = llvm::Constant::getNullValue(ObjCTypes.PropertyListPtrTy);
Fariborz Jahanian943ed6f2009-02-13 17:52:22 +00004843 }
Daniel Dunbar6bff2512009-08-03 17:06:42 +00004844
4845 llvm::Constant *Init =
4846 llvm::ConstantStruct::get(ObjCTypes.CategorynfABITy,
Fariborz Jahanianeb062d92009-01-26 18:32:24 +00004847 Values);
4848 llvm::GlobalVariable *GCATV
Daniel Dunbar6bff2512009-08-03 17:06:42 +00004849 = new llvm::GlobalVariable(CGM.getModule(), ObjCTypes.CategorynfABITy,
Fariborz Jahanianeb062d92009-01-26 18:32:24 +00004850 false,
4851 llvm::GlobalValue::InternalLinkage,
4852 Init,
Owen Anderson1c431b32009-07-08 19:05:04 +00004853 ExtCatName);
Fariborz Jahanian09796d62009-01-31 02:43:27 +00004854 GCATV->setAlignment(
Daniel Dunbar4b429ae2010-04-25 20:39:32 +00004855 CGM.getTargetData().getABITypeAlignment(ObjCTypes.CategorynfABITy));
Fariborz Jahanian1bf0afb2009-01-28 01:05:23 +00004856 GCATV->setSection("__DATA, __objc_const");
Chris Lattnerad64e022009-07-17 23:57:13 +00004857 CGM.AddUsedGlobal(GCATV);
Fariborz Jahanianeb062d92009-01-26 18:32:24 +00004858 DefinedCategories.push_back(GCATV);
Daniel Dunbar74d4b122009-05-15 22:33:15 +00004859
4860 // Determine if this category is also "non-lazy".
4861 if (ImplementationIsNonLazy(OCD))
4862 DefinedNonLazyCategories.push_back(GCATV);
Fariborz Jahanianeb062d92009-01-26 18:32:24 +00004863}
Fariborz Jahanian493dab72009-01-26 21:38:32 +00004864
4865/// GetMethodConstant - Return a struct objc_method constant for the
4866/// given method if it has been defined. The result is null if the
4867/// method has not been defined. The return value has type MethodPtrTy.
4868llvm::Constant *CGObjCNonFragileABIMac::GetMethodConstant(
Daniel Dunbar6bff2512009-08-03 17:06:42 +00004869 const ObjCMethodDecl *MD) {
Fariborz Jahanian493dab72009-01-26 21:38:32 +00004870 // FIXME: Use DenseMap::lookup
4871 llvm::Function *Fn = MethodDefinitions[MD];
4872 if (!Fn)
4873 return 0;
Daniel Dunbar6bff2512009-08-03 17:06:42 +00004874
Fariborz Jahanian493dab72009-01-26 21:38:32 +00004875 std::vector<llvm::Constant*> Method(3);
Daniel Dunbar6bff2512009-08-03 17:06:42 +00004876 Method[0] =
Owen Anderson3c4972d2009-07-29 18:54:39 +00004877 llvm::ConstantExpr::getBitCast(GetMethodVarName(MD->getSelector()),
Daniel Dunbar6bff2512009-08-03 17:06:42 +00004878 ObjCTypes.SelectorPtrTy);
Fariborz Jahanian493dab72009-01-26 21:38:32 +00004879 Method[1] = GetMethodVarType(MD);
Owen Anderson3c4972d2009-07-29 18:54:39 +00004880 Method[2] = llvm::ConstantExpr::getBitCast(Fn, ObjCTypes.Int8PtrTy);
Owen Anderson08e25242009-07-27 22:29:56 +00004881 return llvm::ConstantStruct::get(ObjCTypes.MethodTy, Method);
Fariborz Jahanian493dab72009-01-26 21:38:32 +00004882}
4883
4884/// EmitMethodList - Build meta-data for method declarations
4885/// struct _method_list_t {
4886/// uint32_t entsize; // sizeof(struct _objc_method)
4887/// uint32_t method_count;
4888/// struct _objc_method method_list[method_count];
4889/// }
4890///
Daniel Dunbar9c29bf52009-10-18 20:48:59 +00004891llvm::Constant *CGObjCNonFragileABIMac::EmitMethodList(llvm::Twine Name,
4892 const char *Section,
4893 const ConstantVector &Methods) {
Fariborz Jahanian493dab72009-01-26 21:38:32 +00004894 // Return null for empty list.
4895 if (Methods.empty())
Owen Andersonc9c88b42009-07-31 20:28:54 +00004896 return llvm::Constant::getNullValue(ObjCTypes.MethodListnfABIPtrTy);
Daniel Dunbar6bff2512009-08-03 17:06:42 +00004897
Fariborz Jahanian493dab72009-01-26 21:38:32 +00004898 std::vector<llvm::Constant*> Values(3);
4899 // sizeof(struct _objc_method)
Duncan Sands9408c452009-05-09 07:08:47 +00004900 unsigned Size = CGM.getTargetData().getTypeAllocSize(ObjCTypes.MethodTy);
Owen Anderson4a28d5d2009-07-24 23:12:58 +00004901 Values[0] = llvm::ConstantInt::get(ObjCTypes.IntTy, Size);
Fariborz Jahanian493dab72009-01-26 21:38:32 +00004902 // method_count
Owen Anderson4a28d5d2009-07-24 23:12:58 +00004903 Values[1] = llvm::ConstantInt::get(ObjCTypes.IntTy, Methods.size());
Owen Anderson96e0fc72009-07-29 22:16:19 +00004904 llvm::ArrayType *AT = llvm::ArrayType::get(ObjCTypes.MethodTy,
Fariborz Jahanian493dab72009-01-26 21:38:32 +00004905 Methods.size());
Owen Anderson7db6d832009-07-28 18:33:04 +00004906 Values[2] = llvm::ConstantArray::get(AT, Methods);
Nick Lewycky0d36dd22009-09-19 20:00:52 +00004907 llvm::Constant *Init = llvm::ConstantStruct::get(VMContext, Values, false);
Daniel Dunbar6bff2512009-08-03 17:06:42 +00004908
Fariborz Jahanian493dab72009-01-26 21:38:32 +00004909 llvm::GlobalVariable *GV =
Owen Anderson1c431b32009-07-08 19:05:04 +00004910 new llvm::GlobalVariable(CGM.getModule(), Init->getType(), false,
Fariborz Jahanian493dab72009-01-26 21:38:32 +00004911 llvm::GlobalValue::InternalLinkage,
4912 Init,
Owen Anderson1c431b32009-07-08 19:05:04 +00004913 Name);
Fariborz Jahanian09796d62009-01-31 02:43:27 +00004914 GV->setAlignment(
Daniel Dunbar4b429ae2010-04-25 20:39:32 +00004915 CGM.getTargetData().getABITypeAlignment(Init->getType()));
Fariborz Jahanian493dab72009-01-26 21:38:32 +00004916 GV->setSection(Section);
Chris Lattnerad64e022009-07-17 23:57:13 +00004917 CGM.AddUsedGlobal(GV);
Owen Anderson3c4972d2009-07-29 18:54:39 +00004918 return llvm::ConstantExpr::getBitCast(GV,
Fariborz Jahanian493dab72009-01-26 21:38:32 +00004919 ObjCTypes.MethodListnfABIPtrTy);
4920}
Fariborz Jahanian98abf4b2009-01-27 19:38:51 +00004921
Fariborz Jahanianed157d32009-02-10 20:21:06 +00004922/// ObjCIvarOffsetVariable - Returns the ivar offset variable for
4923/// the given ivar.
Daniel Dunbare83be122010-04-02 21:14:02 +00004924llvm::GlobalVariable *
4925CGObjCNonFragileABIMac::ObjCIvarOffsetVariable(const ObjCInterfaceDecl *ID,
4926 const ObjCIvarDecl *Ivar) {
4927 const ObjCInterfaceDecl *Container = Ivar->getContainingInterface();
Daniel Dunbara81419d2009-05-05 00:36:57 +00004928 std::string Name = "OBJC_IVAR_$_" + Container->getNameAsString() +
Douglas Gregor6ab35242009-04-09 21:40:53 +00004929 '.' + Ivar->getNameAsString();
Daniel Dunbar6bff2512009-08-03 17:06:42 +00004930 llvm::GlobalVariable *IvarOffsetGV =
Fariborz Jahanianed157d32009-02-10 20:21:06 +00004931 CGM.getModule().getGlobalVariable(Name);
4932 if (!IvarOffsetGV)
Daniel Dunbar6bff2512009-08-03 17:06:42 +00004933 IvarOffsetGV =
Owen Anderson1c431b32009-07-08 19:05:04 +00004934 new llvm::GlobalVariable(CGM.getModule(), ObjCTypes.LongTy,
Fariborz Jahanianed157d32009-02-10 20:21:06 +00004935 false,
4936 llvm::GlobalValue::ExternalLinkage,
4937 0,
Owen Anderson1c431b32009-07-08 19:05:04 +00004938 Name);
Fariborz Jahanianed157d32009-02-10 20:21:06 +00004939 return IvarOffsetGV;
4940}
4941
Daniel Dunbare83be122010-04-02 21:14:02 +00004942llvm::Constant *
4943CGObjCNonFragileABIMac::EmitIvarOffsetVar(const ObjCInterfaceDecl *ID,
4944 const ObjCIvarDecl *Ivar,
4945 unsigned long int Offset) {
Daniel Dunbar737c5022009-04-19 00:44:02 +00004946 llvm::GlobalVariable *IvarOffsetGV = ObjCIvarOffsetVariable(ID, Ivar);
Daniel Dunbar6bff2512009-08-03 17:06:42 +00004947 IvarOffsetGV->setInitializer(llvm::ConstantInt::get(ObjCTypes.LongTy,
Daniel Dunbar737c5022009-04-19 00:44:02 +00004948 Offset));
Fariborz Jahanian09796d62009-01-31 02:43:27 +00004949 IvarOffsetGV->setAlignment(
Daniel Dunbar4b429ae2010-04-25 20:39:32 +00004950 CGM.getTargetData().getABITypeAlignment(ObjCTypes.LongTy));
Daniel Dunbar737c5022009-04-19 00:44:02 +00004951
Mike Stumpf5408fe2009-05-16 07:57:57 +00004952 // FIXME: This matches gcc, but shouldn't the visibility be set on the use as
4953 // well (i.e., in ObjCIvarOffsetVariable).
Daniel Dunbar737c5022009-04-19 00:44:02 +00004954 if (Ivar->getAccessControl() == ObjCIvarDecl::Private ||
4955 Ivar->getAccessControl() == ObjCIvarDecl::Package ||
4956 CGM.getDeclVisibilityMode(ID) == LangOptions::Hidden)
Fariborz Jahanian2fa5a272009-01-28 01:36:42 +00004957 IvarOffsetGV->setVisibility(llvm::GlobalValue::HiddenVisibility);
Daniel Dunbar04d40782009-04-14 06:00:08 +00004958 else
Fariborz Jahanian77c9fd22009-04-06 18:30:00 +00004959 IvarOffsetGV->setVisibility(llvm::GlobalValue::DefaultVisibility);
Fariborz Jahanian1bf0afb2009-01-28 01:05:23 +00004960 IvarOffsetGV->setSection("__DATA, __objc_const");
Fariborz Jahanian45012a72009-02-03 00:09:52 +00004961 return IvarOffsetGV;
Fariborz Jahanian1bf0afb2009-01-28 01:05:23 +00004962}
4963
Fariborz Jahanian98abf4b2009-01-27 19:38:51 +00004964/// EmitIvarList - Emit the ivar list for the given
Daniel Dunbar11394522009-04-18 08:51:00 +00004965/// implementation. The return value has type
Fariborz Jahanian98abf4b2009-01-27 19:38:51 +00004966/// IvarListnfABIPtrTy.
4967/// struct _ivar_t {
4968/// unsigned long int *offset; // pointer to ivar offset location
4969/// char *name;
4970/// char *type;
4971/// uint32_t alignment;
4972/// uint32_t size;
4973/// }
4974/// struct _ivar_list_t {
4975/// uint32 entsize; // sizeof(struct _ivar_t)
4976/// uint32 count;
4977/// struct _iver_t list[count];
4978/// }
4979///
Daniel Dunbar3e5f0d82009-04-20 06:54:31 +00004980
Fariborz Jahanian98abf4b2009-01-27 19:38:51 +00004981llvm::Constant *CGObjCNonFragileABIMac::EmitIvarList(
Daniel Dunbar6bff2512009-08-03 17:06:42 +00004982 const ObjCImplementationDecl *ID) {
4983
Fariborz Jahanian98abf4b2009-01-27 19:38:51 +00004984 std::vector<llvm::Constant*> Ivars, Ivar(5);
Daniel Dunbar6bff2512009-08-03 17:06:42 +00004985
Fariborz Jahanian98abf4b2009-01-27 19:38:51 +00004986 const ObjCInterfaceDecl *OID = ID->getClassInterface();
4987 assert(OID && "CGObjCNonFragileABIMac::EmitIvarList - null interface");
Daniel Dunbar6bff2512009-08-03 17:06:42 +00004988
Fariborz Jahanian1bf0afb2009-01-28 01:05:23 +00004989 // FIXME. Consolidate this with similar code in GenerateClass.
Daniel Dunbar6bff2512009-08-03 17:06:42 +00004990
Daniel Dunbar91636d62009-04-20 00:33:43 +00004991 // Collect declared and synthesized ivars in a small vector.
Fariborz Jahanian18191882009-03-31 18:11:23 +00004992 llvm::SmallVector<ObjCIvarDecl*, 16> OIvars;
Fariborz Jahanian8e6ac1d2009-06-04 01:19:09 +00004993 CGM.getContext().ShallowCollectObjCIvars(OID, OIvars);
Daniel Dunbar6bff2512009-08-03 17:06:42 +00004994
Daniel Dunbar3e5f0d82009-04-20 06:54:31 +00004995 for (unsigned i = 0, e = OIvars.size(); i != e; ++i) {
4996 ObjCIvarDecl *IVD = OIvars[i];
Fariborz Jahanian8e6ac1d2009-06-04 01:19:09 +00004997 // Ignore unnamed bit-fields.
4998 if (!IVD->getDeclName())
4999 continue;
Daniel Dunbar6bff2512009-08-03 17:06:42 +00005000 Ivar[0] = EmitIvarOffsetVar(ID->getClassInterface(), IVD,
Daniel Dunbar9f89f2b2009-05-03 12:57:56 +00005001 ComputeIvarBaseOffset(CGM, ID, IVD));
Daniel Dunbar3fea0c02009-04-22 08:22:17 +00005002 Ivar[1] = GetMethodVarName(IVD->getIdentifier());
5003 Ivar[2] = GetMethodVarType(IVD);
Fariborz Jahanian98abf4b2009-01-27 19:38:51 +00005004 const llvm::Type *FieldTy =
Daniel Dunbar3fea0c02009-04-22 08:22:17 +00005005 CGM.getTypes().ConvertTypeForMem(IVD->getType());
Duncan Sands9408c452009-05-09 07:08:47 +00005006 unsigned Size = CGM.getTargetData().getTypeAllocSize(FieldTy);
Fariborz Jahanian98abf4b2009-01-27 19:38:51 +00005007 unsigned Align = CGM.getContext().getPreferredTypeAlign(
Daniel Dunbar6bff2512009-08-03 17:06:42 +00005008 IVD->getType().getTypePtr()) >> 3;
Fariborz Jahanian98abf4b2009-01-27 19:38:51 +00005009 Align = llvm::Log2_32(Align);
Owen Anderson4a28d5d2009-07-24 23:12:58 +00005010 Ivar[3] = llvm::ConstantInt::get(ObjCTypes.IntTy, Align);
Daniel Dunbar91636d62009-04-20 00:33:43 +00005011 // NOTE. Size of a bitfield does not match gcc's, because of the
5012 // way bitfields are treated special in each. But I am told that
5013 // 'size' for bitfield ivars is ignored by the runtime so it does
5014 // not matter. If it matters, there is enough info to get the
5015 // bitfield right!
Owen Anderson4a28d5d2009-07-24 23:12:58 +00005016 Ivar[4] = llvm::ConstantInt::get(ObjCTypes.IntTy, Size);
Owen Anderson08e25242009-07-27 22:29:56 +00005017 Ivars.push_back(llvm::ConstantStruct::get(ObjCTypes.IvarnfABITy, Ivar));
Fariborz Jahanian98abf4b2009-01-27 19:38:51 +00005018 }
5019 // Return null for empty list.
5020 if (Ivars.empty())
Owen Andersonc9c88b42009-07-31 20:28:54 +00005021 return llvm::Constant::getNullValue(ObjCTypes.IvarListnfABIPtrTy);
Fariborz Jahanian98abf4b2009-01-27 19:38:51 +00005022 std::vector<llvm::Constant*> Values(3);
Duncan Sands9408c452009-05-09 07:08:47 +00005023 unsigned Size = CGM.getTargetData().getTypeAllocSize(ObjCTypes.IvarnfABITy);
Owen Anderson4a28d5d2009-07-24 23:12:58 +00005024 Values[0] = llvm::ConstantInt::get(ObjCTypes.IntTy, Size);
5025 Values[1] = llvm::ConstantInt::get(ObjCTypes.IntTy, Ivars.size());
Owen Anderson96e0fc72009-07-29 22:16:19 +00005026 llvm::ArrayType *AT = llvm::ArrayType::get(ObjCTypes.IvarnfABITy,
Fariborz Jahanian98abf4b2009-01-27 19:38:51 +00005027 Ivars.size());
Owen Anderson7db6d832009-07-28 18:33:04 +00005028 Values[2] = llvm::ConstantArray::get(AT, Ivars);
Nick Lewycky0d36dd22009-09-19 20:00:52 +00005029 llvm::Constant *Init = llvm::ConstantStruct::get(VMContext, Values, false);
Fariborz Jahanian98abf4b2009-01-27 19:38:51 +00005030 const char *Prefix = "\01l_OBJC_$_INSTANCE_VARIABLES_";
5031 llvm::GlobalVariable *GV =
Owen Anderson1c431b32009-07-08 19:05:04 +00005032 new llvm::GlobalVariable(CGM.getModule(), Init->getType(), false,
Fariborz Jahanian98abf4b2009-01-27 19:38:51 +00005033 llvm::GlobalValue::InternalLinkage,
5034 Init,
Daniel Dunbar9c29bf52009-10-18 20:48:59 +00005035 Prefix + OID->getName());
Fariborz Jahanian09796d62009-01-31 02:43:27 +00005036 GV->setAlignment(
Daniel Dunbar4b429ae2010-04-25 20:39:32 +00005037 CGM.getTargetData().getABITypeAlignment(Init->getType()));
Fariborz Jahanian1bf0afb2009-01-28 01:05:23 +00005038 GV->setSection("__DATA, __objc_const");
Daniel Dunbar6bff2512009-08-03 17:06:42 +00005039
Chris Lattnerad64e022009-07-17 23:57:13 +00005040 CGM.AddUsedGlobal(GV);
Owen Anderson3c4972d2009-07-29 18:54:39 +00005041 return llvm::ConstantExpr::getBitCast(GV, ObjCTypes.IvarListnfABIPtrTy);
Fariborz Jahanianda320092009-01-29 19:24:30 +00005042}
5043
5044llvm::Constant *CGObjCNonFragileABIMac::GetOrEmitProtocolRef(
Daniel Dunbar6bff2512009-08-03 17:06:42 +00005045 const ObjCProtocolDecl *PD) {
Fariborz Jahanianda320092009-01-29 19:24:30 +00005046 llvm::GlobalVariable *&Entry = Protocols[PD->getIdentifier()];
Daniel Dunbar6bff2512009-08-03 17:06:42 +00005047
Fariborz Jahanianda320092009-01-29 19:24:30 +00005048 if (!Entry) {
5049 // We use the initializer as a marker of whether this is a forward
5050 // reference or not. At module finalization we add the empty
5051 // contents for protocols which were referenced but never defined.
Daniel Dunbar6bff2512009-08-03 17:06:42 +00005052 Entry =
5053 new llvm::GlobalVariable(CGM.getModule(), ObjCTypes.ProtocolnfABITy, false,
5054 llvm::GlobalValue::ExternalLinkage,
5055 0,
Daniel Dunbar9c29bf52009-10-18 20:48:59 +00005056 "\01l_OBJC_PROTOCOL_$_" + PD->getName());
Fariborz Jahanianda320092009-01-29 19:24:30 +00005057 Entry->setSection("__DATA,__datacoal_nt,coalesced");
Fariborz Jahanianda320092009-01-29 19:24:30 +00005058 }
Daniel Dunbar6bff2512009-08-03 17:06:42 +00005059
Fariborz Jahanianda320092009-01-29 19:24:30 +00005060 return Entry;
5061}
5062
5063/// GetOrEmitProtocol - Generate the protocol meta-data:
5064/// @code
5065/// struct _protocol_t {
5066/// id isa; // NULL
5067/// const char * const protocol_name;
5068/// const struct _protocol_list_t * protocol_list; // super protocols
5069/// const struct method_list_t * const instance_methods;
5070/// const struct method_list_t * const class_methods;
5071/// const struct method_list_t *optionalInstanceMethods;
5072/// const struct method_list_t *optionalClassMethods;
5073/// const struct _prop_list_t * properties;
5074/// const uint32_t size; // sizeof(struct _protocol_t)
5075/// const uint32_t flags; // = 0
5076/// }
5077/// @endcode
5078///
5079
5080llvm::Constant *CGObjCNonFragileABIMac::GetOrEmitProtocol(
Daniel Dunbar6bff2512009-08-03 17:06:42 +00005081 const ObjCProtocolDecl *PD) {
Fariborz Jahanianda320092009-01-29 19:24:30 +00005082 llvm::GlobalVariable *&Entry = Protocols[PD->getIdentifier()];
Daniel Dunbar6bff2512009-08-03 17:06:42 +00005083
Fariborz Jahanianda320092009-01-29 19:24:30 +00005084 // Early exit if a defining object has already been generated.
5085 if (Entry && Entry->hasInitializer())
5086 return Entry;
5087
Fariborz Jahanianda320092009-01-29 19:24:30 +00005088 // Construct method lists.
5089 std::vector<llvm::Constant*> InstanceMethods, ClassMethods;
5090 std::vector<llvm::Constant*> OptInstanceMethods, OptClassMethods;
Daniel Dunbar6bff2512009-08-03 17:06:42 +00005091 for (ObjCProtocolDecl::instmeth_iterator
Argyrios Kyrtzidis17945a02009-06-30 02:36:12 +00005092 i = PD->instmeth_begin(), e = PD->instmeth_end(); i != e; ++i) {
Fariborz Jahanianda320092009-01-29 19:24:30 +00005093 ObjCMethodDecl *MD = *i;
Fariborz Jahanian3819a0b2009-01-30 00:46:37 +00005094 llvm::Constant *C = GetMethodDescriptionConstant(MD);
Fariborz Jahanianda320092009-01-29 19:24:30 +00005095 if (MD->getImplementationControl() == ObjCMethodDecl::Optional) {
5096 OptInstanceMethods.push_back(C);
5097 } else {
5098 InstanceMethods.push_back(C);
Daniel Dunbar6bff2512009-08-03 17:06:42 +00005099 }
Fariborz Jahanianda320092009-01-29 19:24:30 +00005100 }
Daniel Dunbar6bff2512009-08-03 17:06:42 +00005101
5102 for (ObjCProtocolDecl::classmeth_iterator
Argyrios Kyrtzidis17945a02009-06-30 02:36:12 +00005103 i = PD->classmeth_begin(), e = PD->classmeth_end(); i != e; ++i) {
Fariborz Jahanianda320092009-01-29 19:24:30 +00005104 ObjCMethodDecl *MD = *i;
Fariborz Jahanian3819a0b2009-01-30 00:46:37 +00005105 llvm::Constant *C = GetMethodDescriptionConstant(MD);
Fariborz Jahanianda320092009-01-29 19:24:30 +00005106 if (MD->getImplementationControl() == ObjCMethodDecl::Optional) {
5107 OptClassMethods.push_back(C);
5108 } else {
5109 ClassMethods.push_back(C);
Daniel Dunbar6bff2512009-08-03 17:06:42 +00005110 }
Fariborz Jahanianda320092009-01-29 19:24:30 +00005111 }
Daniel Dunbar6bff2512009-08-03 17:06:42 +00005112
Fariborz Jahanianda320092009-01-29 19:24:30 +00005113 std::vector<llvm::Constant*> Values(10);
5114 // isa is NULL
Owen Andersonc9c88b42009-07-31 20:28:54 +00005115 Values[0] = llvm::Constant::getNullValue(ObjCTypes.ObjectPtrTy);
Fariborz Jahanianda320092009-01-29 19:24:30 +00005116 Values[1] = GetClassName(PD->getIdentifier());
Daniel Dunbar9c29bf52009-10-18 20:48:59 +00005117 Values[2] = EmitProtocolList("\01l_OBJC_$_PROTOCOL_REFS_" + PD->getName(),
5118 PD->protocol_begin(),
5119 PD->protocol_end());
Daniel Dunbar6bff2512009-08-03 17:06:42 +00005120
Fariborz Jahanian3819a0b2009-01-30 00:46:37 +00005121 Values[3] = EmitMethodList("\01l_OBJC_$_PROTOCOL_INSTANCE_METHODS_"
Daniel Dunbar9c29bf52009-10-18 20:48:59 +00005122 + PD->getName(),
Fariborz Jahanianda320092009-01-29 19:24:30 +00005123 "__DATA, __objc_const",
5124 InstanceMethods);
Daniel Dunbar6bff2512009-08-03 17:06:42 +00005125 Values[4] = EmitMethodList("\01l_OBJC_$_PROTOCOL_CLASS_METHODS_"
Daniel Dunbar9c29bf52009-10-18 20:48:59 +00005126 + PD->getName(),
Fariborz Jahanianda320092009-01-29 19:24:30 +00005127 "__DATA, __objc_const",
5128 ClassMethods);
Fariborz Jahanian3819a0b2009-01-30 00:46:37 +00005129 Values[5] = EmitMethodList("\01l_OBJC_$_PROTOCOL_INSTANCE_METHODS_OPT_"
Daniel Dunbar9c29bf52009-10-18 20:48:59 +00005130 + PD->getName(),
Fariborz Jahanianda320092009-01-29 19:24:30 +00005131 "__DATA, __objc_const",
5132 OptInstanceMethods);
Daniel Dunbar6bff2512009-08-03 17:06:42 +00005133 Values[6] = EmitMethodList("\01l_OBJC_$_PROTOCOL_CLASS_METHODS_OPT_"
Daniel Dunbar9c29bf52009-10-18 20:48:59 +00005134 + PD->getName(),
Fariborz Jahanianda320092009-01-29 19:24:30 +00005135 "__DATA, __objc_const",
5136 OptClassMethods);
Daniel Dunbar9c29bf52009-10-18 20:48:59 +00005137 Values[7] = EmitPropertyList("\01l_OBJC_$_PROP_LIST_" + PD->getName(),
Fariborz Jahanianda320092009-01-29 19:24:30 +00005138 0, PD, ObjCTypes);
Daniel Dunbar6bff2512009-08-03 17:06:42 +00005139 uint32_t Size =
Duncan Sands9408c452009-05-09 07:08:47 +00005140 CGM.getTargetData().getTypeAllocSize(ObjCTypes.ProtocolnfABITy);
Owen Anderson4a28d5d2009-07-24 23:12:58 +00005141 Values[8] = llvm::ConstantInt::get(ObjCTypes.IntTy, Size);
Owen Andersonc9c88b42009-07-31 20:28:54 +00005142 Values[9] = llvm::Constant::getNullValue(ObjCTypes.IntTy);
Owen Anderson08e25242009-07-27 22:29:56 +00005143 llvm::Constant *Init = llvm::ConstantStruct::get(ObjCTypes.ProtocolnfABITy,
Fariborz Jahanianda320092009-01-29 19:24:30 +00005144 Values);
Daniel Dunbar6bff2512009-08-03 17:06:42 +00005145
Fariborz Jahanianda320092009-01-29 19:24:30 +00005146 if (Entry) {
5147 // Already created, fix the linkage and update the initializer.
Mike Stump286acbd2009-03-07 16:33:28 +00005148 Entry->setLinkage(llvm::GlobalValue::WeakAnyLinkage);
Fariborz Jahanianda320092009-01-29 19:24:30 +00005149 Entry->setInitializer(Init);
5150 } else {
Daniel Dunbar6bff2512009-08-03 17:06:42 +00005151 Entry =
Daniel Dunbar9c29bf52009-10-18 20:48:59 +00005152 new llvm::GlobalVariable(CGM.getModule(), ObjCTypes.ProtocolnfABITy,
5153 false, llvm::GlobalValue::WeakAnyLinkage, Init,
5154 "\01l_OBJC_PROTOCOL_$_" + PD->getName());
Fariborz Jahanian09796d62009-01-31 02:43:27 +00005155 Entry->setAlignment(
Daniel Dunbar4b429ae2010-04-25 20:39:32 +00005156 CGM.getTargetData().getABITypeAlignment(ObjCTypes.ProtocolnfABITy));
Fariborz Jahanianda320092009-01-29 19:24:30 +00005157 Entry->setSection("__DATA,__datacoal_nt,coalesced");
Fariborz Jahanianda320092009-01-29 19:24:30 +00005158 }
Fariborz Jahanian8448c2c2009-01-29 20:10:59 +00005159 Entry->setVisibility(llvm::GlobalValue::HiddenVisibility);
Chris Lattnerad64e022009-07-17 23:57:13 +00005160 CGM.AddUsedGlobal(Entry);
5161
Fariborz Jahanian8448c2c2009-01-29 20:10:59 +00005162 // Use this protocol meta-data to build protocol list table in section
5163 // __DATA, __objc_protolist
Daniel Dunbar9c29bf52009-10-18 20:48:59 +00005164 llvm::GlobalVariable *PTGV =
5165 new llvm::GlobalVariable(CGM.getModule(), ObjCTypes.ProtocolnfABIPtrTy,
5166 false, llvm::GlobalValue::WeakAnyLinkage, Entry,
5167 "\01l_OBJC_LABEL_PROTOCOL_$_" + PD->getName());
Fariborz Jahanian09796d62009-01-31 02:43:27 +00005168 PTGV->setAlignment(
Daniel Dunbar4b429ae2010-04-25 20:39:32 +00005169 CGM.getTargetData().getABITypeAlignment(ObjCTypes.ProtocolnfABIPtrTy));
Daniel Dunbar0bf21992009-04-15 02:56:18 +00005170 PTGV->setSection("__DATA, __objc_protolist, coalesced, no_dead_strip");
Fariborz Jahanian8448c2c2009-01-29 20:10:59 +00005171 PTGV->setVisibility(llvm::GlobalValue::HiddenVisibility);
Chris Lattnerad64e022009-07-17 23:57:13 +00005172 CGM.AddUsedGlobal(PTGV);
Fariborz Jahanianda320092009-01-29 19:24:30 +00005173 return Entry;
5174}
5175
5176/// EmitProtocolList - Generate protocol list meta-data:
5177/// @code
5178/// struct _protocol_list_t {
5179/// long protocol_count; // Note, this is 32/64 bit
5180/// struct _protocol_t[protocol_count];
5181/// }
5182/// @endcode
5183///
5184llvm::Constant *
Daniel Dunbar9c29bf52009-10-18 20:48:59 +00005185CGObjCNonFragileABIMac::EmitProtocolList(llvm::Twine Name,
5186 ObjCProtocolDecl::protocol_iterator begin,
5187 ObjCProtocolDecl::protocol_iterator end) {
Fariborz Jahanianda320092009-01-29 19:24:30 +00005188 std::vector<llvm::Constant*> ProtocolRefs;
Daniel Dunbar6bff2512009-08-03 17:06:42 +00005189
Fariborz Jahanianda320092009-01-29 19:24:30 +00005190 // Just return null for empty protocol lists
Daniel Dunbar6bff2512009-08-03 17:06:42 +00005191 if (begin == end)
Owen Andersonc9c88b42009-07-31 20:28:54 +00005192 return llvm::Constant::getNullValue(ObjCTypes.ProtocolListnfABIPtrTy);
Daniel Dunbar6bff2512009-08-03 17:06:42 +00005193
Daniel Dunbar948e2582009-02-15 07:36:20 +00005194 // FIXME: We shouldn't need to do this lookup here, should we?
Daniel Dunbar9c29bf52009-10-18 20:48:59 +00005195 llvm::SmallString<256> TmpName;
5196 Name.toVector(TmpName);
5197 llvm::GlobalVariable *GV =
5198 CGM.getModule().getGlobalVariable(TmpName.str(), true);
Fariborz Jahanianda320092009-01-29 19:24:30 +00005199 if (GV)
Daniel Dunbar9c29bf52009-10-18 20:48:59 +00005200 return llvm::ConstantExpr::getBitCast(GV, ObjCTypes.ProtocolListnfABIPtrTy);
Daniel Dunbar6bff2512009-08-03 17:06:42 +00005201
Daniel Dunbar948e2582009-02-15 07:36:20 +00005202 for (; begin != end; ++begin)
5203 ProtocolRefs.push_back(GetProtocolRef(*begin)); // Implemented???
5204
Fariborz Jahanianda320092009-01-29 19:24:30 +00005205 // This list is null terminated.
Owen Andersonc9c88b42009-07-31 20:28:54 +00005206 ProtocolRefs.push_back(llvm::Constant::getNullValue(
Daniel Dunbar6bff2512009-08-03 17:06:42 +00005207 ObjCTypes.ProtocolnfABIPtrTy));
5208
Fariborz Jahanianda320092009-01-29 19:24:30 +00005209 std::vector<llvm::Constant*> Values(2);
Owen Andersona1cf15f2009-07-14 23:10:40 +00005210 Values[0] =
Owen Anderson4a28d5d2009-07-24 23:12:58 +00005211 llvm::ConstantInt::get(ObjCTypes.LongTy, ProtocolRefs.size() - 1);
Daniel Dunbar6bff2512009-08-03 17:06:42 +00005212 Values[1] =
Owen Anderson7db6d832009-07-28 18:33:04 +00005213 llvm::ConstantArray::get(
Owen Anderson96e0fc72009-07-29 22:16:19 +00005214 llvm::ArrayType::get(ObjCTypes.ProtocolnfABIPtrTy,
Daniel Dunbar6bff2512009-08-03 17:06:42 +00005215 ProtocolRefs.size()),
5216 ProtocolRefs);
5217
Nick Lewycky0d36dd22009-09-19 20:00:52 +00005218 llvm::Constant *Init = llvm::ConstantStruct::get(VMContext, Values, false);
Owen Anderson1c431b32009-07-08 19:05:04 +00005219 GV = new llvm::GlobalVariable(CGM.getModule(), Init->getType(), false,
Fariborz Jahanianda320092009-01-29 19:24:30 +00005220 llvm::GlobalValue::InternalLinkage,
5221 Init,
Owen Anderson1c431b32009-07-08 19:05:04 +00005222 Name);
Fariborz Jahanianda320092009-01-29 19:24:30 +00005223 GV->setSection("__DATA, __objc_const");
Fariborz Jahanian09796d62009-01-31 02:43:27 +00005224 GV->setAlignment(
Daniel Dunbar4b429ae2010-04-25 20:39:32 +00005225 CGM.getTargetData().getABITypeAlignment(Init->getType()));
Chris Lattnerad64e022009-07-17 23:57:13 +00005226 CGM.AddUsedGlobal(GV);
Daniel Dunbar6bff2512009-08-03 17:06:42 +00005227 return llvm::ConstantExpr::getBitCast(GV,
Daniel Dunbar948e2582009-02-15 07:36:20 +00005228 ObjCTypes.ProtocolListnfABIPtrTy);
Fariborz Jahanianda320092009-01-29 19:24:30 +00005229}
5230
Fariborz Jahanian3819a0b2009-01-30 00:46:37 +00005231/// GetMethodDescriptionConstant - This routine build following meta-data:
5232/// struct _objc_method {
5233/// SEL _cmd;
5234/// char *method_type;
5235/// char *_imp;
5236/// }
5237
5238llvm::Constant *
5239CGObjCNonFragileABIMac::GetMethodDescriptionConstant(const ObjCMethodDecl *MD) {
5240 std::vector<llvm::Constant*> Desc(3);
Owen Andersona1cf15f2009-07-14 23:10:40 +00005241 Desc[0] =
Daniel Dunbar6bff2512009-08-03 17:06:42 +00005242 llvm::ConstantExpr::getBitCast(GetMethodVarName(MD->getSelector()),
5243 ObjCTypes.SelectorPtrTy);
Fariborz Jahanian3819a0b2009-01-30 00:46:37 +00005244 Desc[1] = GetMethodVarType(MD);
Fariborz Jahanian8cfd3972009-01-30 18:58:59 +00005245 // Protocol methods have no implementation. So, this entry is always NULL.
Owen Andersonc9c88b42009-07-31 20:28:54 +00005246 Desc[2] = llvm::Constant::getNullValue(ObjCTypes.Int8PtrTy);
Owen Anderson08e25242009-07-27 22:29:56 +00005247 return llvm::ConstantStruct::get(ObjCTypes.MethodTy, Desc);
Fariborz Jahanian3819a0b2009-01-30 00:46:37 +00005248}
Fariborz Jahanian45012a72009-02-03 00:09:52 +00005249
5250/// EmitObjCValueForIvar - Code Gen for nonfragile ivar reference.
5251/// This code gen. amounts to generating code for:
5252/// @code
5253/// (type *)((char *)base + _OBJC_IVAR_$_.ivar;
5254/// @encode
Daniel Dunbar6bff2512009-08-03 17:06:42 +00005255///
Fariborz Jahanian598d3f62009-02-03 19:03:09 +00005256LValue CGObjCNonFragileABIMac::EmitObjCValueForIvar(
John McCall506b57e2010-05-17 21:00:27 +00005257 CodeGen::CodeGenFunction &CGF,
5258 QualType ObjectTy,
5259 llvm::Value *BaseValue,
5260 const ObjCIvarDecl *Ivar,
5261 unsigned CVRQualifiers) {
5262 ObjCInterfaceDecl *ID = ObjectTy->getAs<ObjCObjectType>()->getInterface();
Daniel Dunbar97776872009-04-22 07:32:20 +00005263 return EmitValueForIvarAtOffset(CGF, ID, BaseValue, Ivar, CVRQualifiers,
5264 EmitIvarOffset(CGF, ID, Ivar));
Fariborz Jahanian45012a72009-02-03 00:09:52 +00005265}
5266
Fariborz Jahanianf63aa3f2009-02-10 19:02:04 +00005267llvm::Value *CGObjCNonFragileABIMac::EmitIvarOffset(
Daniel Dunbar6bff2512009-08-03 17:06:42 +00005268 CodeGen::CodeGenFunction &CGF,
5269 const ObjCInterfaceDecl *Interface,
5270 const ObjCIvarDecl *Ivar) {
Daniel Dunbar2da84ff2009-11-29 21:23:36 +00005271 return CGF.Builder.CreateLoad(ObjCIvarOffsetVariable(Interface, Ivar),"ivar");
Fariborz Jahanianf63aa3f2009-02-10 19:02:04 +00005272}
5273
Fariborz Jahanian46551122009-02-04 00:22:57 +00005274CodeGen::RValue CGObjCNonFragileABIMac::EmitMessageSend(
Daniel Dunbar6bff2512009-08-03 17:06:42 +00005275 CodeGen::CodeGenFunction &CGF,
John McCallef072fd2010-05-22 01:48:05 +00005276 ReturnValueSlot Return,
Daniel Dunbar6bff2512009-08-03 17:06:42 +00005277 QualType ResultType,
5278 Selector Sel,
5279 llvm::Value *Receiver,
5280 QualType Arg0Ty,
5281 bool IsSuper,
5282 const CallArgList &CallArgs) {
Mike Stumpf5408fe2009-05-16 07:57:57 +00005283 // FIXME. Even though IsSuper is passes. This function doese not handle calls
5284 // to 'super' receivers.
Fariborz Jahanian83a8a752009-02-04 20:42:28 +00005285 CodeGenTypes &Types = CGM.getTypes();
Fariborz Jahanian7a06aae2009-02-06 20:09:23 +00005286 llvm::Value *Arg0 = Receiver;
5287 if (!IsSuper)
5288 Arg0 = CGF.Builder.CreateBitCast(Arg0, ObjCTypes.ObjectPtrTy, "tmp");
Daniel Dunbar6bff2512009-08-03 17:06:42 +00005289
Fariborz Jahanian83a8a752009-02-04 20:42:28 +00005290 // Find the message function name.
Mike Stumpf5408fe2009-05-16 07:57:57 +00005291 // FIXME. This is too much work to get the ABI-specific result type needed to
5292 // find the message name.
John McCallead608a2010-02-26 00:48:12 +00005293 const CGFunctionInfo &FnInfo
Rafael Espindola264ba482010-03-30 20:24:48 +00005294 = Types.getFunctionInfo(ResultType, CallArgList(),
5295 FunctionType::ExtInfo());
Fariborz Jahanian70b51c72009-04-30 23:08:58 +00005296 llvm::Constant *Fn = 0;
Fariborz Jahanian83a8a752009-02-04 20:42:28 +00005297 std::string Name("\01l_");
5298 if (CGM.ReturnTypeUsesSret(FnInfo)) {
Fariborz Jahanianc1708522009-02-05 18:00:27 +00005299#if 0
5300 // unlike what is documented. gcc never generates this API!!
Fariborz Jahanian83a8a752009-02-04 20:42:28 +00005301 if (Receiver->getType() == ObjCTypes.ObjectPtrTy) {
Chris Lattner1c02f862009-04-22 02:53:24 +00005302 Fn = ObjCTypes.getMessageSendIdStretFixupFn();
Fariborz Jahanian83a8a752009-02-04 20:42:28 +00005303 // FIXME. Is there a better way of getting these names.
5304 // They are available in RuntimeFunctions vector pair.
5305 Name += "objc_msgSendId_stret_fixup";
Mike Stumpb3589f42009-07-30 22:28:39 +00005306 } else
Fariborz Jahanianc1708522009-02-05 18:00:27 +00005307#endif
Daniel Dunbar6bff2512009-08-03 17:06:42 +00005308 if (IsSuper) {
Chris Lattner1c02f862009-04-22 02:53:24 +00005309 Fn = ObjCTypes.getMessageSendSuper2StretFixupFn();
Fariborz Jahanian7a06aae2009-02-06 20:09:23 +00005310 Name += "objc_msgSendSuper2_stret_fixup";
Daniel Dunbar6bff2512009-08-03 17:06:42 +00005311 } else {
5312 Fn = ObjCTypes.getMessageSendStretFixupFn();
5313 Name += "objc_msgSend_stret_fixup";
5314 }
Douglas Gregor0c293ea2010-06-22 23:07:26 +00005315 } else if (!IsSuper && ResultType->isRealFloatingType()) {
Daniel Dunbarc0183e82009-06-26 18:32:06 +00005316 if (ResultType->isSpecificBuiltinType(BuiltinType::LongDouble)) {
5317 Fn = ObjCTypes.getMessageSendFpretFixupFn();
5318 Name += "objc_msgSend_fpret_fixup";
Mike Stumpb3589f42009-07-30 22:28:39 +00005319 } else {
Daniel Dunbarc0183e82009-06-26 18:32:06 +00005320 Fn = ObjCTypes.getMessageSendFixupFn();
5321 Name += "objc_msgSend_fixup";
Fariborz Jahanian5b2bad02009-04-30 16:31:11 +00005322 }
Mike Stumpb3589f42009-07-30 22:28:39 +00005323 } else {
Fariborz Jahanianc1708522009-02-05 18:00:27 +00005324#if 0
5325// unlike what is documented. gcc never generates this API!!
Fariborz Jahanian83a8a752009-02-04 20:42:28 +00005326 if (Receiver->getType() == ObjCTypes.ObjectPtrTy) {
Chris Lattner1c02f862009-04-22 02:53:24 +00005327 Fn = ObjCTypes.getMessageSendIdFixupFn();
Fariborz Jahanian83a8a752009-02-04 20:42:28 +00005328 Name += "objc_msgSendId_fixup";
Daniel Dunbar6bff2512009-08-03 17:06:42 +00005329 } else
Fariborz Jahanianc1708522009-02-05 18:00:27 +00005330#endif
Daniel Dunbar6bff2512009-08-03 17:06:42 +00005331 if (IsSuper) {
Chris Lattner1c02f862009-04-22 02:53:24 +00005332 Fn = ObjCTypes.getMessageSendSuper2FixupFn();
Fariborz Jahanian7a06aae2009-02-06 20:09:23 +00005333 Name += "objc_msgSendSuper2_fixup";
Daniel Dunbar6bff2512009-08-03 17:06:42 +00005334 } else {
5335 Fn = ObjCTypes.getMessageSendFixupFn();
5336 Name += "objc_msgSend_fixup";
5337 }
Fariborz Jahanian83a8a752009-02-04 20:42:28 +00005338 }
Fariborz Jahanian70b51c72009-04-30 23:08:58 +00005339 assert(Fn && "CGObjCNonFragileABIMac::EmitMessageSend");
Fariborz Jahanian83a8a752009-02-04 20:42:28 +00005340 Name += '_';
5341 std::string SelName(Sel.getAsString());
5342 // Replace all ':' in selector name with '_' ouch!
Mike Stump1eb44332009-09-09 15:08:12 +00005343 for (unsigned i = 0; i < SelName.size(); i++)
Fariborz Jahanian83a8a752009-02-04 20:42:28 +00005344 if (SelName[i] == ':')
5345 SelName[i] = '_';
5346 Name += SelName;
5347 llvm::GlobalVariable *GV = CGM.getModule().getGlobalVariable(Name);
5348 if (!GV) {
Daniel Dunbar33af70f2009-04-15 19:03:14 +00005349 // Build message ref table entry.
Fariborz Jahanian83a8a752009-02-04 20:42:28 +00005350 std::vector<llvm::Constant*> Values(2);
5351 Values[0] = Fn;
5352 Values[1] = GetMethodVarName(Sel);
Nick Lewycky0d36dd22009-09-19 20:00:52 +00005353 llvm::Constant *Init = llvm::ConstantStruct::get(VMContext, Values, false);
Owen Anderson1c431b32009-07-08 19:05:04 +00005354 GV = new llvm::GlobalVariable(CGM.getModule(), Init->getType(), false,
Mike Stump286acbd2009-03-07 16:33:28 +00005355 llvm::GlobalValue::WeakAnyLinkage,
Fariborz Jahanian83a8a752009-02-04 20:42:28 +00005356 Init,
Owen Anderson1c431b32009-07-08 19:05:04 +00005357 Name);
Fariborz Jahanian83a8a752009-02-04 20:42:28 +00005358 GV->setVisibility(llvm::GlobalValue::HiddenVisibility);
Daniel Dunbarf59c1a62009-04-15 19:04:46 +00005359 GV->setAlignment(16);
Fariborz Jahanian83a8a752009-02-04 20:42:28 +00005360 GV->setSection("__DATA, __objc_msgrefs, coalesced");
Fariborz Jahanian83a8a752009-02-04 20:42:28 +00005361 }
5362 llvm::Value *Arg1 = CGF.Builder.CreateBitCast(GV, ObjCTypes.MessageRefPtrTy);
Daniel Dunbar6bff2512009-08-03 17:06:42 +00005363
Fariborz Jahanian83a8a752009-02-04 20:42:28 +00005364 CallArgList ActualArgs;
5365 ActualArgs.push_back(std::make_pair(RValue::get(Arg0), Arg0Ty));
Daniel Dunbar6bff2512009-08-03 17:06:42 +00005366 ActualArgs.push_back(std::make_pair(RValue::get(Arg1),
Fariborz Jahanian83a8a752009-02-04 20:42:28 +00005367 ObjCTypes.MessageRefCPtrTy));
5368 ActualArgs.insert(ActualArgs.end(), CallArgs.begin(), CallArgs.end());
John McCall04a67a62010-02-05 21:31:56 +00005369 const CGFunctionInfo &FnInfo1 = Types.getFunctionInfo(ResultType, ActualArgs,
Rafael Espindola264ba482010-03-30 20:24:48 +00005370 FunctionType::ExtInfo());
Fariborz Jahanianef163782009-02-05 01:13:09 +00005371 llvm::Value *Callee = CGF.Builder.CreateStructGEP(Arg1, 0);
5372 Callee = CGF.Builder.CreateLoad(Callee);
Fariborz Jahanian3ab75bd2009-02-14 21:25:36 +00005373 const llvm::FunctionType *FTy = Types.GetFunctionType(FnInfo1, true);
Fariborz Jahanianef163782009-02-05 01:13:09 +00005374 Callee = CGF.Builder.CreateBitCast(Callee,
Owen Anderson96e0fc72009-07-29 22:16:19 +00005375 llvm::PointerType::getUnqual(FTy));
John McCallef072fd2010-05-22 01:48:05 +00005376 return CGF.EmitCall(FnInfo1, Callee, Return, ActualArgs);
Fariborz Jahanian46551122009-02-04 00:22:57 +00005377}
5378
5379/// Generate code for a message send expression in the nonfragile abi.
Daniel Dunbard6c93d72009-09-17 04:01:22 +00005380CodeGen::RValue
5381CGObjCNonFragileABIMac::GenerateMessageSend(CodeGen::CodeGenFunction &CGF,
John McCallef072fd2010-05-22 01:48:05 +00005382 ReturnValueSlot Return,
Daniel Dunbard6c93d72009-09-17 04:01:22 +00005383 QualType ResultType,
5384 Selector Sel,
5385 llvm::Value *Receiver,
Daniel Dunbard6c93d72009-09-17 04:01:22 +00005386 const CallArgList &CallArgs,
David Chisnallc6cd5fd2010-04-28 19:33:36 +00005387 const ObjCInterfaceDecl *Class,
Daniel Dunbard6c93d72009-09-17 04:01:22 +00005388 const ObjCMethodDecl *Method) {
Fariborz Jahaniand0f8a8d2009-05-11 19:25:47 +00005389 return LegacyDispatchedSelector(Sel)
John McCallef072fd2010-05-22 01:48:05 +00005390 ? EmitLegacyMessageSend(CGF, Return, ResultType,
5391 EmitSelector(CGF.Builder, Sel),
Daniel Dunbar6bff2512009-08-03 17:06:42 +00005392 Receiver, CGF.getContext().getObjCIdType(),
Daniel Dunbard6c93d72009-09-17 04:01:22 +00005393 false, CallArgs, Method, ObjCTypes)
John McCallef072fd2010-05-22 01:48:05 +00005394 : EmitMessageSend(CGF, Return, ResultType, Sel,
Daniel Dunbar6bff2512009-08-03 17:06:42 +00005395 Receiver, CGF.getContext().getObjCIdType(),
5396 false, CallArgs);
Fariborz Jahanian46551122009-02-04 00:22:57 +00005397}
5398
Daniel Dunbar5a7379a2009-03-01 04:40:10 +00005399llvm::GlobalVariable *
Fariborz Jahanian0f902942009-04-14 18:41:56 +00005400CGObjCNonFragileABIMac::GetClassGlobal(const std::string &Name) {
Daniel Dunbar5a7379a2009-03-01 04:40:10 +00005401 llvm::GlobalVariable *GV = CGM.getModule().getGlobalVariable(Name);
5402
Daniel Dunbardfff2302009-03-02 05:18:14 +00005403 if (!GV) {
Daniel Dunbar6bff2512009-08-03 17:06:42 +00005404 GV = new llvm::GlobalVariable(CGM.getModule(), ObjCTypes.ClassnfABITy,
Owen Anderson1c431b32009-07-08 19:05:04 +00005405 false, llvm::GlobalValue::ExternalLinkage,
5406 0, Name);
Daniel Dunbar5a7379a2009-03-01 04:40:10 +00005407 }
5408
5409 return GV;
5410}
5411
Daniel Dunbar6bff2512009-08-03 17:06:42 +00005412llvm::Value *CGObjCNonFragileABIMac::EmitClassRef(CGBuilderTy &Builder,
5413 const ObjCInterfaceDecl *ID) {
Fariborz Jahanian0e81f4b2009-02-05 20:41:40 +00005414 llvm::GlobalVariable *&Entry = ClassReferences[ID->getIdentifier()];
Daniel Dunbar6bff2512009-08-03 17:06:42 +00005415
Fariborz Jahanian0e81f4b2009-02-05 20:41:40 +00005416 if (!Entry) {
Daniel Dunbar6ab187a2009-04-07 05:48:37 +00005417 std::string ClassName(getClassSymbolPrefix() + ID->getNameAsString());
Daniel Dunbar5a7379a2009-03-01 04:40:10 +00005418 llvm::GlobalVariable *ClassGV = GetClassGlobal(ClassName);
Daniel Dunbar6bff2512009-08-03 17:06:42 +00005419 Entry =
5420 new llvm::GlobalVariable(CGM.getModule(), ObjCTypes.ClassnfABIPtrTy,
Owen Anderson1c431b32009-07-08 19:05:04 +00005421 false, llvm::GlobalValue::InternalLinkage,
Daniel Dunbar6bff2512009-08-03 17:06:42 +00005422 ClassGV,
Owen Anderson1c431b32009-07-08 19:05:04 +00005423 "\01L_OBJC_CLASSLIST_REFERENCES_$_");
Fariborz Jahanian0e81f4b2009-02-05 20:41:40 +00005424 Entry->setAlignment(
Daniel Dunbar4b429ae2010-04-25 20:39:32 +00005425 CGM.getTargetData().getABITypeAlignment(
Daniel Dunbar6bff2512009-08-03 17:06:42 +00005426 ObjCTypes.ClassnfABIPtrTy));
Daniel Dunbar11394522009-04-18 08:51:00 +00005427 Entry->setSection("__DATA, __objc_classrefs, regular, no_dead_strip");
Chris Lattnerad64e022009-07-17 23:57:13 +00005428 CGM.AddUsedGlobal(Entry);
Daniel Dunbar11394522009-04-18 08:51:00 +00005429 }
Daniel Dunbar6bff2512009-08-03 17:06:42 +00005430
Daniel Dunbar2da84ff2009-11-29 21:23:36 +00005431 return Builder.CreateLoad(Entry, "tmp");
Daniel Dunbar11394522009-04-18 08:51:00 +00005432}
Fariborz Jahanian0e81f4b2009-02-05 20:41:40 +00005433
Daniel Dunbar11394522009-04-18 08:51:00 +00005434llvm::Value *
Daniel Dunbar6bff2512009-08-03 17:06:42 +00005435CGObjCNonFragileABIMac::EmitSuperClassRef(CGBuilderTy &Builder,
Daniel Dunbar11394522009-04-18 08:51:00 +00005436 const ObjCInterfaceDecl *ID) {
5437 llvm::GlobalVariable *&Entry = SuperClassReferences[ID->getIdentifier()];
Daniel Dunbar6bff2512009-08-03 17:06:42 +00005438
Daniel Dunbar11394522009-04-18 08:51:00 +00005439 if (!Entry) {
5440 std::string ClassName(getClassSymbolPrefix() + ID->getNameAsString());
5441 llvm::GlobalVariable *ClassGV = GetClassGlobal(ClassName);
Daniel Dunbar6bff2512009-08-03 17:06:42 +00005442 Entry =
5443 new llvm::GlobalVariable(CGM.getModule(), ObjCTypes.ClassnfABIPtrTy,
Owen Anderson1c431b32009-07-08 19:05:04 +00005444 false, llvm::GlobalValue::InternalLinkage,
Daniel Dunbar6bff2512009-08-03 17:06:42 +00005445 ClassGV,
Owen Anderson1c431b32009-07-08 19:05:04 +00005446 "\01L_OBJC_CLASSLIST_SUP_REFS_$_");
Daniel Dunbar11394522009-04-18 08:51:00 +00005447 Entry->setAlignment(
Daniel Dunbar4b429ae2010-04-25 20:39:32 +00005448 CGM.getTargetData().getABITypeAlignment(
Daniel Dunbar6bff2512009-08-03 17:06:42 +00005449 ObjCTypes.ClassnfABIPtrTy));
Daniel Dunbar11394522009-04-18 08:51:00 +00005450 Entry->setSection("__DATA, __objc_superrefs, regular, no_dead_strip");
Chris Lattnerad64e022009-07-17 23:57:13 +00005451 CGM.AddUsedGlobal(Entry);
Fariborz Jahanian0e81f4b2009-02-05 20:41:40 +00005452 }
Daniel Dunbar6bff2512009-08-03 17:06:42 +00005453
Daniel Dunbar2da84ff2009-11-29 21:23:36 +00005454 return Builder.CreateLoad(Entry, "tmp");
Fariborz Jahanian0e81f4b2009-02-05 20:41:40 +00005455}
5456
Fariborz Jahanian7a06aae2009-02-06 20:09:23 +00005457/// EmitMetaClassRef - Return a Value * of the address of _class_t
5458/// meta-data
5459///
Daniel Dunbar6bff2512009-08-03 17:06:42 +00005460llvm::Value *CGObjCNonFragileABIMac::EmitMetaClassRef(CGBuilderTy &Builder,
5461 const ObjCInterfaceDecl *ID) {
Fariborz Jahanian7a06aae2009-02-06 20:09:23 +00005462 llvm::GlobalVariable * &Entry = MetaClassReferences[ID->getIdentifier()];
5463 if (Entry)
Daniel Dunbar2da84ff2009-11-29 21:23:36 +00005464 return Builder.CreateLoad(Entry, "tmp");
Daniel Dunbar6bff2512009-08-03 17:06:42 +00005465
Daniel Dunbar6ab187a2009-04-07 05:48:37 +00005466 std::string MetaClassName(getMetaclassSymbolPrefix() + ID->getNameAsString());
Fariborz Jahanian0f902942009-04-14 18:41:56 +00005467 llvm::GlobalVariable *MetaClassGV = GetClassGlobal(MetaClassName);
Daniel Dunbar6bff2512009-08-03 17:06:42 +00005468 Entry =
Owen Anderson1c431b32009-07-08 19:05:04 +00005469 new llvm::GlobalVariable(CGM.getModule(), ObjCTypes.ClassnfABIPtrTy, false,
Fariborz Jahanian7a06aae2009-02-06 20:09:23 +00005470 llvm::GlobalValue::InternalLinkage,
Daniel Dunbar6bff2512009-08-03 17:06:42 +00005471 MetaClassGV,
Owen Anderson1c431b32009-07-08 19:05:04 +00005472 "\01L_OBJC_CLASSLIST_SUP_REFS_$_");
Fariborz Jahanian7a06aae2009-02-06 20:09:23 +00005473 Entry->setAlignment(
Daniel Dunbar4b429ae2010-04-25 20:39:32 +00005474 CGM.getTargetData().getABITypeAlignment(
Daniel Dunbar6bff2512009-08-03 17:06:42 +00005475 ObjCTypes.ClassnfABIPtrTy));
5476
Daniel Dunbar33af70f2009-04-15 19:03:14 +00005477 Entry->setSection("__DATA, __objc_superrefs, regular, no_dead_strip");
Chris Lattnerad64e022009-07-17 23:57:13 +00005478 CGM.AddUsedGlobal(Entry);
Daniel Dunbar6bff2512009-08-03 17:06:42 +00005479
Daniel Dunbar2da84ff2009-11-29 21:23:36 +00005480 return Builder.CreateLoad(Entry, "tmp");
Fariborz Jahanian7a06aae2009-02-06 20:09:23 +00005481}
5482
Fariborz Jahanian0e81f4b2009-02-05 20:41:40 +00005483/// GetClass - Return a reference to the class for the given interface
5484/// decl.
5485llvm::Value *CGObjCNonFragileABIMac::GetClass(CGBuilderTy &Builder,
5486 const ObjCInterfaceDecl *ID) {
Fariborz Jahanian269f8bc2009-11-17 22:42:00 +00005487 if (ID->hasAttr<WeakImportAttr>()) {
5488 std::string ClassName(getClassSymbolPrefix() + ID->getNameAsString());
5489 llvm::GlobalVariable *ClassGV = GetClassGlobal(ClassName);
5490 ClassGV->setLinkage(llvm::GlobalValue::ExternalWeakLinkage);
5491 }
5492
Fariborz Jahanian0e81f4b2009-02-05 20:41:40 +00005493 return EmitClassRef(Builder, ID);
5494}
5495
Fariborz Jahanian7a06aae2009-02-06 20:09:23 +00005496/// Generates a message send where the super is the receiver. This is
5497/// a message send to self with special delivery semantics indicating
5498/// which class's method should be called.
5499CodeGen::RValue
5500CGObjCNonFragileABIMac::GenerateMessageSendSuper(CodeGen::CodeGenFunction &CGF,
John McCallef072fd2010-05-22 01:48:05 +00005501 ReturnValueSlot Return,
Daniel Dunbar6bff2512009-08-03 17:06:42 +00005502 QualType ResultType,
5503 Selector Sel,
5504 const ObjCInterfaceDecl *Class,
5505 bool isCategoryImpl,
5506 llvm::Value *Receiver,
5507 bool IsClassMessage,
Daniel Dunbard6c93d72009-09-17 04:01:22 +00005508 const CodeGen::CallArgList &CallArgs,
5509 const ObjCMethodDecl *Method) {
Fariborz Jahanian7a06aae2009-02-06 20:09:23 +00005510 // ...
5511 // Create and init a super structure; this is a (receiver, class)
5512 // pair we will pass to objc_msgSendSuper.
5513 llvm::Value *ObjCSuper =
5514 CGF.Builder.CreateAlloca(ObjCTypes.SuperTy, 0, "objc_super");
Daniel Dunbar6bff2512009-08-03 17:06:42 +00005515
Fariborz Jahanian7a06aae2009-02-06 20:09:23 +00005516 llvm::Value *ReceiverAsObject =
5517 CGF.Builder.CreateBitCast(Receiver, ObjCTypes.ObjectPtrTy);
5518 CGF.Builder.CreateStore(ReceiverAsObject,
5519 CGF.Builder.CreateStructGEP(ObjCSuper, 0));
Daniel Dunbar6bff2512009-08-03 17:06:42 +00005520
Fariborz Jahanian7a06aae2009-02-06 20:09:23 +00005521 // If this is a class message the metaclass is passed as the target.
Fariborz Jahanian7ce77922009-02-28 20:07:56 +00005522 llvm::Value *Target;
5523 if (IsClassMessage) {
5524 if (isCategoryImpl) {
5525 // Message sent to "super' in a class method defined in
5526 // a category implementation.
Daniel Dunbar11394522009-04-18 08:51:00 +00005527 Target = EmitClassRef(CGF.Builder, Class);
Fariborz Jahanian7ce77922009-02-28 20:07:56 +00005528 Target = CGF.Builder.CreateStructGEP(Target, 0);
5529 Target = CGF.Builder.CreateLoad(Target);
Mike Stumpb3589f42009-07-30 22:28:39 +00005530 } else
Fariborz Jahanian7ce77922009-02-28 20:07:56 +00005531 Target = EmitMetaClassRef(CGF.Builder, Class);
Mike Stumpb3589f42009-07-30 22:28:39 +00005532 } else
Daniel Dunbar11394522009-04-18 08:51:00 +00005533 Target = EmitSuperClassRef(CGF.Builder, Class);
Daniel Dunbar6bff2512009-08-03 17:06:42 +00005534
Mike Stumpf5408fe2009-05-16 07:57:57 +00005535 // FIXME: We shouldn't need to do this cast, rectify the ASTContext and
5536 // ObjCTypes types.
Fariborz Jahanian7a06aae2009-02-06 20:09:23 +00005537 const llvm::Type *ClassTy =
5538 CGM.getTypes().ConvertType(CGF.getContext().getObjCClassType());
5539 Target = CGF.Builder.CreateBitCast(Target, ClassTy);
5540 CGF.Builder.CreateStore(Target,
5541 CGF.Builder.CreateStructGEP(ObjCSuper, 1));
Daniel Dunbar6bff2512009-08-03 17:06:42 +00005542
Fariborz Jahaniand0f8a8d2009-05-11 19:25:47 +00005543 return (LegacyDispatchedSelector(Sel))
John McCallef072fd2010-05-22 01:48:05 +00005544 ? EmitLegacyMessageSend(CGF, Return, ResultType,
5545 EmitSelector(CGF.Builder, Sel),
Daniel Dunbar6bff2512009-08-03 17:06:42 +00005546 ObjCSuper, ObjCTypes.SuperPtrCTy,
Daniel Dunbard6c93d72009-09-17 04:01:22 +00005547 true, CallArgs, Method, ObjCTypes)
John McCallef072fd2010-05-22 01:48:05 +00005548 : EmitMessageSend(CGF, Return, ResultType, Sel,
Daniel Dunbar6bff2512009-08-03 17:06:42 +00005549 ObjCSuper, ObjCTypes.SuperPtrCTy,
5550 true, CallArgs);
Fariborz Jahanian7a06aae2009-02-06 20:09:23 +00005551}
Fariborz Jahanian26cc89f2009-02-11 20:51:17 +00005552
Daniel Dunbar6bff2512009-08-03 17:06:42 +00005553llvm::Value *CGObjCNonFragileABIMac::EmitSelector(CGBuilderTy &Builder,
Fariborz Jahanian03b29602010-06-17 19:56:20 +00005554 Selector Sel, bool lval) {
Fariborz Jahanian26cc89f2009-02-11 20:51:17 +00005555 llvm::GlobalVariable *&Entry = SelectorReferences[Sel];
Daniel Dunbar6bff2512009-08-03 17:06:42 +00005556
Fariborz Jahanian26cc89f2009-02-11 20:51:17 +00005557 if (!Entry) {
Daniel Dunbar6bff2512009-08-03 17:06:42 +00005558 llvm::Constant *Casted =
5559 llvm::ConstantExpr::getBitCast(GetMethodVarName(Sel),
5560 ObjCTypes.SelectorPtrTy);
5561 Entry =
5562 new llvm::GlobalVariable(CGM.getModule(), ObjCTypes.SelectorPtrTy, false,
5563 llvm::GlobalValue::InternalLinkage,
5564 Casted, "\01L_OBJC_SELECTOR_REFERENCES_");
Fariborz Jahaniand0f8a8d2009-05-11 19:25:47 +00005565 Entry->setSection("__DATA, __objc_selrefs, literal_pointers, no_dead_strip");
Chris Lattnerad64e022009-07-17 23:57:13 +00005566 CGM.AddUsedGlobal(Entry);
Fariborz Jahanian26cc89f2009-02-11 20:51:17 +00005567 }
Daniel Dunbar6bff2512009-08-03 17:06:42 +00005568
Fariborz Jahanian03b29602010-06-17 19:56:20 +00005569 if (lval)
5570 return Entry;
Daniel Dunbar2da84ff2009-11-29 21:23:36 +00005571 return Builder.CreateLoad(Entry, "tmp");
Fariborz Jahanian26cc89f2009-02-11 20:51:17 +00005572}
Fariborz Jahanian6948aea2009-02-16 22:52:32 +00005573/// EmitObjCIvarAssign - Code gen for assigning to a __strong object.
Fariborz Jahanian6c7a1f32009-09-24 22:25:38 +00005574/// objc_assign_ivar (id src, id *dst, ptrdiff_t)
Fariborz Jahanian6948aea2009-02-16 22:52:32 +00005575///
5576void CGObjCNonFragileABIMac::EmitObjCIvarAssign(CodeGen::CodeGenFunction &CGF,
Mike Stump1eb44332009-09-09 15:08:12 +00005577 llvm::Value *src,
Fariborz Jahanian6c7a1f32009-09-24 22:25:38 +00005578 llvm::Value *dst,
5579 llvm::Value *ivarOffset) {
Fariborz Jahanian0a855d02009-03-23 19:10:40 +00005580 const llvm::Type * SrcTy = src->getType();
5581 if (!isa<llvm::PointerType>(SrcTy)) {
Duncan Sands9408c452009-05-09 07:08:47 +00005582 unsigned Size = CGM.getTargetData().getTypeAllocSize(SrcTy);
Fariborz Jahanian0a855d02009-03-23 19:10:40 +00005583 assert(Size <= 8 && "does not support size > 8");
5584 src = (Size == 4 ? CGF.Builder.CreateBitCast(src, ObjCTypes.IntTy)
5585 : CGF.Builder.CreateBitCast(src, ObjCTypes.LongTy));
Fariborz Jahanian3b8a6522009-03-13 00:42:52 +00005586 src = CGF.Builder.CreateIntToPtr(src, ObjCTypes.Int8PtrTy);
5587 }
Fariborz Jahanian6948aea2009-02-16 22:52:32 +00005588 src = CGF.Builder.CreateBitCast(src, ObjCTypes.ObjectPtrTy);
5589 dst = CGF.Builder.CreateBitCast(dst, ObjCTypes.PtrObjectPtrTy);
Fariborz Jahanian6c7a1f32009-09-24 22:25:38 +00005590 CGF.Builder.CreateCall3(ObjCTypes.getGcAssignIvarFn(),
5591 src, dst, ivarOffset);
Fariborz Jahanian6948aea2009-02-16 22:52:32 +00005592 return;
5593}
5594
5595/// EmitObjCStrongCastAssign - Code gen for assigning to a __strong cast object.
5596/// objc_assign_strongCast (id src, id *dst)
5597///
5598void CGObjCNonFragileABIMac::EmitObjCStrongCastAssign(
Daniel Dunbar6bff2512009-08-03 17:06:42 +00005599 CodeGen::CodeGenFunction &CGF,
Mike Stump1eb44332009-09-09 15:08:12 +00005600 llvm::Value *src, llvm::Value *dst) {
Fariborz Jahanian0a855d02009-03-23 19:10:40 +00005601 const llvm::Type * SrcTy = src->getType();
5602 if (!isa<llvm::PointerType>(SrcTy)) {
Duncan Sands9408c452009-05-09 07:08:47 +00005603 unsigned Size = CGM.getTargetData().getTypeAllocSize(SrcTy);
Fariborz Jahanian0a855d02009-03-23 19:10:40 +00005604 assert(Size <= 8 && "does not support size > 8");
5605 src = (Size == 4 ? CGF.Builder.CreateBitCast(src, ObjCTypes.IntTy)
Daniel Dunbar6bff2512009-08-03 17:06:42 +00005606 : CGF.Builder.CreateBitCast(src, ObjCTypes.LongTy));
Fariborz Jahanian3b8a6522009-03-13 00:42:52 +00005607 src = CGF.Builder.CreateIntToPtr(src, ObjCTypes.Int8PtrTy);
5608 }
Fariborz Jahanian6948aea2009-02-16 22:52:32 +00005609 src = CGF.Builder.CreateBitCast(src, ObjCTypes.ObjectPtrTy);
5610 dst = CGF.Builder.CreateBitCast(dst, ObjCTypes.PtrObjectPtrTy);
Chris Lattnerbbccd612009-04-22 02:38:11 +00005611 CGF.Builder.CreateCall2(ObjCTypes.getGcAssignStrongCastFn(),
Fariborz Jahanian6948aea2009-02-16 22:52:32 +00005612 src, dst, "weakassign");
5613 return;
5614}
5615
Fariborz Jahanian082b02e2009-07-08 01:18:33 +00005616void CGObjCNonFragileABIMac::EmitGCMemmoveCollectable(
Daniel Dunbar6bff2512009-08-03 17:06:42 +00005617 CodeGen::CodeGenFunction &CGF,
5618 llvm::Value *DestPtr,
5619 llvm::Value *SrcPtr,
Fariborz Jahanian55bcace2010-06-15 22:44:06 +00005620 llvm::Value *Size) {
Fariborz Jahanian082b02e2009-07-08 01:18:33 +00005621 SrcPtr = CGF.Builder.CreateBitCast(SrcPtr, ObjCTypes.Int8PtrTy);
5622 DestPtr = CGF.Builder.CreateBitCast(DestPtr, ObjCTypes.Int8PtrTy);
Fariborz Jahanian082b02e2009-07-08 01:18:33 +00005623 CGF.Builder.CreateCall3(ObjCTypes.GcMemmoveCollectableFn(),
Fariborz Jahanian55bcace2010-06-15 22:44:06 +00005624 DestPtr, SrcPtr, Size);
Fariborz Jahanian082b02e2009-07-08 01:18:33 +00005625 return;
5626}
5627
Fariborz Jahanian6948aea2009-02-16 22:52:32 +00005628/// EmitObjCWeakRead - Code gen for loading value of a __weak
5629/// object: objc_read_weak (id *src)
5630///
5631llvm::Value * CGObjCNonFragileABIMac::EmitObjCWeakRead(
Daniel Dunbar6bff2512009-08-03 17:06:42 +00005632 CodeGen::CodeGenFunction &CGF,
Mike Stump1eb44332009-09-09 15:08:12 +00005633 llvm::Value *AddrWeakObj) {
Eli Friedman8339b352009-03-07 03:57:15 +00005634 const llvm::Type* DestTy =
Daniel Dunbar6bff2512009-08-03 17:06:42 +00005635 cast<llvm::PointerType>(AddrWeakObj->getType())->getElementType();
5636 AddrWeakObj = CGF.Builder.CreateBitCast(AddrWeakObj, ObjCTypes.PtrObjectPtrTy);
Chris Lattner72db6c32009-04-22 02:44:54 +00005637 llvm::Value *read_weak = CGF.Builder.CreateCall(ObjCTypes.getGcReadWeakFn(),
Fariborz Jahanian6948aea2009-02-16 22:52:32 +00005638 AddrWeakObj, "weakread");
Eli Friedman8339b352009-03-07 03:57:15 +00005639 read_weak = CGF.Builder.CreateBitCast(read_weak, DestTy);
Fariborz Jahanian6948aea2009-02-16 22:52:32 +00005640 return read_weak;
5641}
5642
5643/// EmitObjCWeakAssign - Code gen for assigning to a __weak object.
5644/// objc_assign_weak (id src, id *dst)
5645///
5646void CGObjCNonFragileABIMac::EmitObjCWeakAssign(CodeGen::CodeGenFunction &CGF,
Mike Stump1eb44332009-09-09 15:08:12 +00005647 llvm::Value *src, llvm::Value *dst) {
Fariborz Jahanian0a855d02009-03-23 19:10:40 +00005648 const llvm::Type * SrcTy = src->getType();
5649 if (!isa<llvm::PointerType>(SrcTy)) {
Duncan Sands9408c452009-05-09 07:08:47 +00005650 unsigned Size = CGM.getTargetData().getTypeAllocSize(SrcTy);
Fariborz Jahanian0a855d02009-03-23 19:10:40 +00005651 assert(Size <= 8 && "does not support size > 8");
5652 src = (Size == 4 ? CGF.Builder.CreateBitCast(src, ObjCTypes.IntTy)
5653 : CGF.Builder.CreateBitCast(src, ObjCTypes.LongTy));
Fariborz Jahanian3b8a6522009-03-13 00:42:52 +00005654 src = CGF.Builder.CreateIntToPtr(src, ObjCTypes.Int8PtrTy);
5655 }
Fariborz Jahanian6948aea2009-02-16 22:52:32 +00005656 src = CGF.Builder.CreateBitCast(src, ObjCTypes.ObjectPtrTy);
5657 dst = CGF.Builder.CreateBitCast(dst, ObjCTypes.PtrObjectPtrTy);
Chris Lattner96508e12009-04-17 22:12:36 +00005658 CGF.Builder.CreateCall2(ObjCTypes.getGcAssignWeakFn(),
Fariborz Jahanian6948aea2009-02-16 22:52:32 +00005659 src, dst, "weakassign");
5660 return;
5661}
5662
5663/// EmitObjCGlobalAssign - Code gen for assigning to a __strong object.
5664/// objc_assign_global (id src, id *dst)
5665///
5666void CGObjCNonFragileABIMac::EmitObjCGlobalAssign(CodeGen::CodeGenFunction &CGF,
Mike Stump1eb44332009-09-09 15:08:12 +00005667 llvm::Value *src, llvm::Value *dst) {
Fariborz Jahanian0a855d02009-03-23 19:10:40 +00005668 const llvm::Type * SrcTy = src->getType();
5669 if (!isa<llvm::PointerType>(SrcTy)) {
Duncan Sands9408c452009-05-09 07:08:47 +00005670 unsigned Size = CGM.getTargetData().getTypeAllocSize(SrcTy);
Fariborz Jahanian0a855d02009-03-23 19:10:40 +00005671 assert(Size <= 8 && "does not support size > 8");
5672 src = (Size == 4 ? CGF.Builder.CreateBitCast(src, ObjCTypes.IntTy)
5673 : CGF.Builder.CreateBitCast(src, ObjCTypes.LongTy));
Fariborz Jahanian3b8a6522009-03-13 00:42:52 +00005674 src = CGF.Builder.CreateIntToPtr(src, ObjCTypes.Int8PtrTy);
5675 }
Fariborz Jahanian6948aea2009-02-16 22:52:32 +00005676 src = CGF.Builder.CreateBitCast(src, ObjCTypes.ObjectPtrTy);
5677 dst = CGF.Builder.CreateBitCast(dst, ObjCTypes.PtrObjectPtrTy);
Chris Lattnerbbccd612009-04-22 02:38:11 +00005678 CGF.Builder.CreateCall2(ObjCTypes.getGcAssignGlobalFn(),
Fariborz Jahanian6948aea2009-02-16 22:52:32 +00005679 src, dst, "globalassign");
5680 return;
5681}
Fariborz Jahanian26cc89f2009-02-11 20:51:17 +00005682
Daniel Dunbar6bff2512009-08-03 17:06:42 +00005683void
John McCallf1549f62010-07-06 01:34:17 +00005684CGObjCNonFragileABIMac::EmitSynchronizedStmt(CodeGen::CodeGenFunction &CGF,
5685 const ObjCAtSynchronizedStmt &S) {
5686 // Evaluate the lock operand. This should dominate the cleanup.
5687 llvm::Value *SyncArg = CGF.EmitScalarExpr(S.getSynchExpr());
Daniel Dunbar8ecbaf22009-02-24 07:47:38 +00005688
John McCallf1549f62010-07-06 01:34:17 +00005689 // Acquire the lock.
5690 SyncArg = CGF.Builder.CreateBitCast(SyncArg, ObjCTypes.ObjectPtrTy);
5691 CGF.Builder.CreateCall(ObjCTypes.getSyncEnterFn(), SyncArg)
5692 ->setDoesNotThrow();
5693
5694 // Register an all-paths cleanup to release the lock.
5695 {
John McCallda65ea82010-07-13 20:32:21 +00005696 CodeGenFunction::CleanupBlock ReleaseScope(CGF, NormalAndEHCleanup);
John McCallf1549f62010-07-06 01:34:17 +00005697
5698 CGF.Builder.CreateCall(ObjCTypes.getSyncExitFn(), SyncArg)
5699 ->setDoesNotThrow();
Daniel Dunbar8ecbaf22009-02-24 07:47:38 +00005700 }
5701
John McCallf1549f62010-07-06 01:34:17 +00005702 // Emit the body of the statement.
5703 CGF.EmitStmt(S.getSynchBody());
Daniel Dunbar8ecbaf22009-02-24 07:47:38 +00005704
John McCallf1549f62010-07-06 01:34:17 +00005705 // Pop the lock-release cleanup.
5706 CGF.PopCleanupBlock();
5707}
Daniel Dunbar8ecbaf22009-02-24 07:47:38 +00005708
John McCallf1549f62010-07-06 01:34:17 +00005709namespace {
5710 struct CatchHandler {
5711 const VarDecl *Variable;
5712 const Stmt *Body;
5713 llvm::BasicBlock *Block;
5714 llvm::Value *TypeInfo;
5715 };
John McCall8e3f8612010-07-13 22:12:14 +00005716
5717 struct CallObjCEndCatch : EHScopeStack::LazyCleanup {
5718 CallObjCEndCatch(bool MightThrow, llvm::Value *Fn) :
5719 MightThrow(MightThrow), Fn(Fn) {}
5720 bool MightThrow;
5721 llvm::Value *Fn;
5722
5723 void Emit(CodeGenFunction &CGF, bool IsForEH) {
5724 if (!MightThrow) {
5725 CGF.Builder.CreateCall(Fn)->setDoesNotThrow();
5726 return;
5727 }
5728
5729 CGF.EmitCallOrInvoke(Fn, 0, 0);
5730 }
5731 };
John McCallf1549f62010-07-06 01:34:17 +00005732}
Daniel Dunbar6bff2512009-08-03 17:06:42 +00005733
John McCallf1549f62010-07-06 01:34:17 +00005734void CGObjCNonFragileABIMac::EmitTryStmt(CodeGen::CodeGenFunction &CGF,
5735 const ObjCAtTryStmt &S) {
5736 // Jump destination for falling out of catch bodies.
5737 CodeGenFunction::JumpDest Cont;
5738 if (S.getNumCatchStmts())
5739 Cont = CGF.getJumpDestInCurrentScope("eh.cont");
Daniel Dunbar4ff36842009-03-02 06:08:11 +00005740
John McCallf1549f62010-07-06 01:34:17 +00005741 CodeGenFunction::FinallyInfo FinallyInfo;
5742 if (const ObjCAtFinallyStmt *Finally = S.getFinallyStmt())
5743 FinallyInfo = CGF.EnterFinallyBlock(Finally->getFinallyBody(),
5744 ObjCTypes.getObjCBeginCatchFn(),
5745 ObjCTypes.getObjCEndCatchFn(),
5746 ObjCTypes.getExceptionRethrowFn());
Daniel Dunbar4ff36842009-03-02 06:08:11 +00005747
John McCallf1549f62010-07-06 01:34:17 +00005748 llvm::SmallVector<CatchHandler, 8> Handlers;
Daniel Dunbar4ff36842009-03-02 06:08:11 +00005749
John McCallf1549f62010-07-06 01:34:17 +00005750 // Enter the catch, if there is one.
5751 if (S.getNumCatchStmts()) {
5752 for (unsigned I = 0, N = S.getNumCatchStmts(); I != N; ++I) {
5753 const ObjCAtCatchStmt *CatchStmt = S.getCatchStmt(I);
Douglas Gregorc00d8e12010-04-26 16:46:50 +00005754 const VarDecl *CatchDecl = CatchStmt->getCatchParamDecl();
Daniel Dunbar4ff36842009-03-02 06:08:11 +00005755
John McCallf1549f62010-07-06 01:34:17 +00005756 Handlers.push_back(CatchHandler());
5757 CatchHandler &Handler = Handlers.back();
5758 Handler.Variable = CatchDecl;
5759 Handler.Body = CatchStmt->getCatchBody();
5760 Handler.Block = CGF.createBasicBlock("catch");
5761
5762 // @catch(...) always matches.
Douglas Gregor8f5e3dd2010-04-23 22:50:49 +00005763 if (!CatchDecl) {
John McCallf1549f62010-07-06 01:34:17 +00005764 Handler.TypeInfo = 0; // catch-all
5765 // Don't consider any other catches.
Douglas Gregor8f5e3dd2010-04-23 22:50:49 +00005766 break;
5767 }
Daniel Dunbar4ff36842009-03-02 06:08:11 +00005768
John McCallf1549f62010-07-06 01:34:17 +00005769 // There's a particular fixed type info for 'id'.
Douglas Gregor8f5e3dd2010-04-23 22:50:49 +00005770 if (CatchDecl->getType()->isObjCIdType() ||
5771 CatchDecl->getType()->isObjCQualifiedIdType()) {
5772 llvm::Value *IDEHType =
5773 CGM.getModule().getGlobalVariable("OBJC_EHTYPE_id");
5774 if (!IDEHType)
5775 IDEHType =
5776 new llvm::GlobalVariable(CGM.getModule(), ObjCTypes.EHTypeTy,
5777 false,
5778 llvm::GlobalValue::ExternalLinkage,
5779 0, "OBJC_EHTYPE_id");
John McCallf1549f62010-07-06 01:34:17 +00005780 Handler.TypeInfo = IDEHType;
Douglas Gregor8f5e3dd2010-04-23 22:50:49 +00005781 } else {
5782 // All other types should be Objective-C interface pointer types.
5783 const ObjCObjectPointerType *PT =
5784 CatchDecl->getType()->getAs<ObjCObjectPointerType>();
5785 assert(PT && "Invalid @catch type.");
5786 const ObjCInterfaceType *IT = PT->getInterfaceType();
5787 assert(IT && "Invalid @catch type.");
John McCallf1549f62010-07-06 01:34:17 +00005788 Handler.TypeInfo = GetInterfaceEHType(IT->getDecl(), false);
Daniel Dunbar4ff36842009-03-02 06:08:11 +00005789 }
5790 }
John McCallf1549f62010-07-06 01:34:17 +00005791
5792 EHCatchScope *Catch = CGF.EHStack.pushCatch(Handlers.size());
5793 for (unsigned I = 0, E = Handlers.size(); I != E; ++I)
5794 Catch->setHandler(I, Handlers[I].TypeInfo, Handlers[I].Block);
Daniel Dunbar4ff36842009-03-02 06:08:11 +00005795 }
John McCallf1549f62010-07-06 01:34:17 +00005796
5797 // Emit the try body.
5798 CGF.EmitStmt(S.getTryBody());
Daniel Dunbar4ff36842009-03-02 06:08:11 +00005799
John McCallf1549f62010-07-06 01:34:17 +00005800 // Leave the try.
5801 if (S.getNumCatchStmts())
5802 CGF.EHStack.popCatch();
Daniel Dunbar6bff2512009-08-03 17:06:42 +00005803
John McCallf1549f62010-07-06 01:34:17 +00005804 // Remember where we were.
5805 CGBuilderTy::InsertPoint SavedIP = CGF.Builder.saveAndClearIP();
Daniel Dunbar4ff36842009-03-02 06:08:11 +00005806
John McCallf1549f62010-07-06 01:34:17 +00005807 // Emit the handlers.
5808 for (unsigned I = 0, E = Handlers.size(); I != E; ++I) {
5809 CatchHandler &Handler = Handlers[I];
Daniel Dunbar4ff36842009-03-02 06:08:11 +00005810
John McCallf1549f62010-07-06 01:34:17 +00005811 CGF.EmitBlock(Handler.Block);
5812 llvm::Value *RawExn = CGF.Builder.CreateLoad(CGF.getExceptionSlot());
Daniel Dunbar4ff36842009-03-02 06:08:11 +00005813
John McCallf1549f62010-07-06 01:34:17 +00005814 // Enter the catch.
5815 llvm::CallInst *Exn =
5816 CGF.Builder.CreateCall(ObjCTypes.getObjCBeginCatchFn(), RawExn,
5817 "exn.adjusted");
5818 Exn->setDoesNotThrow();
Daniel Dunbar4ff36842009-03-02 06:08:11 +00005819
John McCallf1549f62010-07-06 01:34:17 +00005820 // Add a cleanup to leave the catch.
John McCall8e3f8612010-07-13 22:12:14 +00005821 bool EndCatchMightThrow = (Handler.Variable == 0);
5822 CGF.EHStack.pushLazyCleanup<CallObjCEndCatch>(NormalAndEHCleanup,
5823 EndCatchMightThrow,
5824 ObjCTypes.getObjCEndCatchFn());
Daniel Dunbar6bff2512009-08-03 17:06:42 +00005825
John McCallf1549f62010-07-06 01:34:17 +00005826 // Bind the catch parameter if it exists.
5827 if (const VarDecl *CatchParam = Handler.Variable) {
5828 const llvm::Type *CatchType = CGF.ConvertType(CatchParam->getType());
5829 llvm::Value *CastExn = CGF.Builder.CreateBitCast(Exn, CatchType);
Daniel Dunbar4ff36842009-03-02 06:08:11 +00005830
John McCallf1549f62010-07-06 01:34:17 +00005831 CGF.EmitLocalBlockVarDecl(*CatchParam);
5832 CGF.Builder.CreateStore(CastExn, CGF.GetAddrOfLocalVar(CatchParam));
Daniel Dunbar4ff36842009-03-02 06:08:11 +00005833 }
Daniel Dunbar4ff36842009-03-02 06:08:11 +00005834
John McCallf1549f62010-07-06 01:34:17 +00005835 CGF.ObjCEHValueStack.push_back(Exn);
5836 CGF.EmitStmt(Handler.Body);
5837 CGF.ObjCEHValueStack.pop_back();
Daniel Dunbar8ecbaf22009-02-24 07:47:38 +00005838
John McCallf1549f62010-07-06 01:34:17 +00005839 // Leave the earlier cleanup.
5840 CGF.PopCleanupBlock();
Daniel Dunbar8ecbaf22009-02-24 07:47:38 +00005841
John McCallf1549f62010-07-06 01:34:17 +00005842 CGF.EmitBranchThroughCleanup(Cont);
5843 }
Daniel Dunbar8ecbaf22009-02-24 07:47:38 +00005844
John McCallf1549f62010-07-06 01:34:17 +00005845 // Go back to the try-statement fallthrough.
5846 CGF.Builder.restoreIP(SavedIP);
Daniel Dunbar8ecbaf22009-02-24 07:47:38 +00005847
John McCallf1549f62010-07-06 01:34:17 +00005848 // Pop out of the normal cleanup on the finally.
5849 if (S.getFinallyStmt())
5850 CGF.ExitFinallyBlock(FinallyInfo);
Daniel Dunbar8ecbaf22009-02-24 07:47:38 +00005851
John McCallf1549f62010-07-06 01:34:17 +00005852 if (Cont.Block)
5853 CGF.EmitBlock(Cont.Block);
Daniel Dunbar8ecbaf22009-02-24 07:47:38 +00005854}
5855
Anders Carlssonf57c5b22009-02-16 22:59:18 +00005856/// EmitThrowStmt - Generate code for a throw statement.
5857void CGObjCNonFragileABIMac::EmitThrowStmt(CodeGen::CodeGenFunction &CGF,
5858 const ObjCAtThrowStmt &S) {
Daniel Dunbar6bff2512009-08-03 17:06:42 +00005859 llvm::Value *Exception;
Fariborz Jahanian69677ea2010-05-28 17:34:43 +00005860 llvm::Constant *FunctionThrowOrRethrow;
Anders Carlssonf57c5b22009-02-16 22:59:18 +00005861 if (const Expr *ThrowExpr = S.getThrowExpr()) {
Daniel Dunbar4ff36842009-03-02 06:08:11 +00005862 Exception = CGF.EmitScalarExpr(ThrowExpr);
Fariborz Jahanian69677ea2010-05-28 17:34:43 +00005863 FunctionThrowOrRethrow = ObjCTypes.getExceptionThrowFn();
Anders Carlssonf57c5b22009-02-16 22:59:18 +00005864 } else {
Daniel Dunbar6bff2512009-08-03 17:06:42 +00005865 assert((!CGF.ObjCEHValueStack.empty() && CGF.ObjCEHValueStack.back()) &&
Daniel Dunbar4ff36842009-03-02 06:08:11 +00005866 "Unexpected rethrow outside @catch block.");
5867 Exception = CGF.ObjCEHValueStack.back();
Fariborz Jahanian69677ea2010-05-28 17:34:43 +00005868 FunctionThrowOrRethrow = ObjCTypes.getExceptionRethrowFn();
Anders Carlssonf57c5b22009-02-16 22:59:18 +00005869 }
5870
Daniel Dunbar6bff2512009-08-03 17:06:42 +00005871 llvm::Value *ExceptionAsObject =
Daniel Dunbar4ff36842009-03-02 06:08:11 +00005872 CGF.Builder.CreateBitCast(Exception, ObjCTypes.ObjectPtrTy, "tmp");
5873 llvm::BasicBlock *InvokeDest = CGF.getInvokeDest();
5874 if (InvokeDest) {
Fariborz Jahanian69677ea2010-05-28 17:34:43 +00005875 CGF.Builder.CreateInvoke(FunctionThrowOrRethrow,
John McCallf1549f62010-07-06 01:34:17 +00005876 CGF.getUnreachableBlock(), InvokeDest,
Daniel Dunbar4ff36842009-03-02 06:08:11 +00005877 &ExceptionAsObject, &ExceptionAsObject + 1);
John McCallf1549f62010-07-06 01:34:17 +00005878 } else {
5879 CGF.Builder.CreateCall(FunctionThrowOrRethrow, ExceptionAsObject)
5880 ->setDoesNotReturn();
5881 CGF.Builder.CreateUnreachable();
5882 }
Daniel Dunbar4ff36842009-03-02 06:08:11 +00005883
Anders Carlssonf57c5b22009-02-16 22:59:18 +00005884 // Clear the insertion point to indicate we are in unreachable code.
5885 CGF.Builder.ClearInsertionPoint();
5886}
Daniel Dunbare588b992009-03-01 04:46:24 +00005887
5888llvm::Value *
Daniel Dunbar6bff2512009-08-03 17:06:42 +00005889CGObjCNonFragileABIMac::GetInterfaceEHType(const ObjCInterfaceDecl *ID,
Daniel Dunbar8158a2f2009-04-08 04:21:03 +00005890 bool ForDefinition) {
Daniel Dunbare588b992009-03-01 04:46:24 +00005891 llvm::GlobalVariable * &Entry = EHTypeReferences[ID->getIdentifier()];
Daniel Dunbare588b992009-03-01 04:46:24 +00005892
Daniel Dunbar8158a2f2009-04-08 04:21:03 +00005893 // If we don't need a definition, return the entry if found or check
5894 // if we use an external reference.
5895 if (!ForDefinition) {
5896 if (Entry)
5897 return Entry;
Daniel Dunbar7e075cb2009-04-07 06:43:45 +00005898
Daniel Dunbar8158a2f2009-04-08 04:21:03 +00005899 // If this type (or a super class) has the __objc_exception__
5900 // attribute, emit an external reference.
Douglas Gregor68584ed2009-06-18 16:11:24 +00005901 if (hasObjCExceptionAttribute(CGM.getContext(), ID))
Daniel Dunbar6bff2512009-08-03 17:06:42 +00005902 return Entry =
Owen Anderson1c431b32009-07-08 19:05:04 +00005903 new llvm::GlobalVariable(CGM.getModule(), ObjCTypes.EHTypeTy, false,
Daniel Dunbar8158a2f2009-04-08 04:21:03 +00005904 llvm::GlobalValue::ExternalLinkage,
Daniel Dunbar6bff2512009-08-03 17:06:42 +00005905 0,
Daniel Dunbar9c29bf52009-10-18 20:48:59 +00005906 ("OBJC_EHTYPE_$_" +
Daniel Dunbar01eb9b92009-10-18 21:17:35 +00005907 ID->getIdentifier()->getName()));
Daniel Dunbar8158a2f2009-04-08 04:21:03 +00005908 }
Daniel Dunbar6bff2512009-08-03 17:06:42 +00005909
Daniel Dunbar8158a2f2009-04-08 04:21:03 +00005910 // Otherwise we need to either make a new entry or fill in the
5911 // initializer.
5912 assert((!Entry || !Entry->hasInitializer()) && "Duplicate EHType definition");
Daniel Dunbar6ab187a2009-04-07 05:48:37 +00005913 std::string ClassName(getClassSymbolPrefix() + ID->getNameAsString());
Daniel Dunbare588b992009-03-01 04:46:24 +00005914 std::string VTableName = "objc_ehtype_vtable";
Daniel Dunbar6bff2512009-08-03 17:06:42 +00005915 llvm::GlobalVariable *VTableGV =
Daniel Dunbare588b992009-03-01 04:46:24 +00005916 CGM.getModule().getGlobalVariable(VTableName);
5917 if (!VTableGV)
Daniel Dunbar6bff2512009-08-03 17:06:42 +00005918 VTableGV = new llvm::GlobalVariable(CGM.getModule(), ObjCTypes.Int8PtrTy,
5919 false,
Daniel Dunbare588b992009-03-01 04:46:24 +00005920 llvm::GlobalValue::ExternalLinkage,
Owen Anderson1c431b32009-07-08 19:05:04 +00005921 0, VTableName);
Daniel Dunbare588b992009-03-01 04:46:24 +00005922
Chris Lattner77b89b82010-06-27 07:15:29 +00005923 llvm::Value *VTableIdx =
5924 llvm::ConstantInt::get(llvm::Type::getInt32Ty(VMContext), 2);
Daniel Dunbare588b992009-03-01 04:46:24 +00005925
5926 std::vector<llvm::Constant*> Values(3);
Owen Anderson3c4972d2009-07-29 18:54:39 +00005927 Values[0] = llvm::ConstantExpr::getGetElementPtr(VTableGV, &VTableIdx, 1);
Daniel Dunbare588b992009-03-01 04:46:24 +00005928 Values[1] = GetClassName(ID->getIdentifier());
Fariborz Jahanian0f902942009-04-14 18:41:56 +00005929 Values[2] = GetClassGlobal(ClassName);
Owen Andersona1cf15f2009-07-14 23:10:40 +00005930 llvm::Constant *Init =
Owen Anderson08e25242009-07-27 22:29:56 +00005931 llvm::ConstantStruct::get(ObjCTypes.EHTypeTy, Values);
Daniel Dunbare588b992009-03-01 04:46:24 +00005932
Daniel Dunbar8158a2f2009-04-08 04:21:03 +00005933 if (Entry) {
5934 Entry->setInitializer(Init);
5935 } else {
Owen Anderson1c431b32009-07-08 19:05:04 +00005936 Entry = new llvm::GlobalVariable(CGM.getModule(), ObjCTypes.EHTypeTy, false,
Daniel Dunbar8158a2f2009-04-08 04:21:03 +00005937 llvm::GlobalValue::WeakAnyLinkage,
Daniel Dunbar6bff2512009-08-03 17:06:42 +00005938 Init,
Daniel Dunbar9c29bf52009-10-18 20:48:59 +00005939 ("OBJC_EHTYPE_$_" +
Daniel Dunbar01eb9b92009-10-18 21:17:35 +00005940 ID->getIdentifier()->getName()));
Daniel Dunbar8158a2f2009-04-08 04:21:03 +00005941 }
5942
Daniel Dunbar04d40782009-04-14 06:00:08 +00005943 if (CGM.getLangOptions().getVisibilityMode() == LangOptions::Hidden)
Daniel Dunbar6ab187a2009-04-07 05:48:37 +00005944 Entry->setVisibility(llvm::GlobalValue::HiddenVisibility);
Daniel Dunbar4b429ae2010-04-25 20:39:32 +00005945 Entry->setAlignment(CGM.getTargetData().getABITypeAlignment(
5946 ObjCTypes.EHTypeTy));
Daniel Dunbar8158a2f2009-04-08 04:21:03 +00005947
5948 if (ForDefinition) {
5949 Entry->setSection("__DATA,__objc_const");
5950 Entry->setLinkage(llvm::GlobalValue::ExternalLinkage);
5951 } else {
5952 Entry->setSection("__DATA,__datacoal_nt,coalesced");
5953 }
Daniel Dunbare588b992009-03-01 04:46:24 +00005954
5955 return Entry;
5956}
Daniel Dunbar6bff2512009-08-03 17:06:42 +00005957
Daniel Dunbarbbce49b2008-08-12 00:12:39 +00005958/* *** */
5959
Daniel Dunbar6efc0c52008-08-13 03:21:16 +00005960CodeGen::CGObjCRuntime *
5961CodeGen::CreateMacObjCRuntime(CodeGen::CodeGenModule &CGM) {
Daniel Dunbarc17a4d32008-08-11 02:45:11 +00005962 return new CGObjCMac(CGM);
5963}
Fariborz Jahanianee0af742009-01-21 22:04:16 +00005964
5965CodeGen::CGObjCRuntime *
Fariborz Jahanian30bc5712009-01-22 23:02:58 +00005966CodeGen::CreateMacNonFragileABIObjCRuntime(CodeGen::CodeGenModule &CGM) {
Fariborz Jahanianaa23b572009-01-23 23:53:38 +00005967 return new CGObjCNonFragileABIMac(CGM);
Fariborz Jahanianee0af742009-01-21 22:04:16 +00005968}