blob: 3a80330a2fd40ae3f5aca9980f5540a0b69c0929 [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
John McCall87bb5822010-07-31 23:20:56 +000028#include "llvm/InlineAsm.h"
29#include "llvm/IntrinsicInst.h"
Owen Anderson69243822009-07-13 04:10:07 +000030#include "llvm/LLVMContext.h"
Daniel Dunbarbbce49b2008-08-12 00:12:39 +000031#include "llvm/Module.h"
Daniel Dunbar0c0e7a62008-10-29 22:36:39 +000032#include "llvm/ADT/DenseSet.h"
Daniel Dunbar33063492009-09-07 00:20:42 +000033#include "llvm/ADT/SetVector.h"
34#include "llvm/ADT/SmallString.h"
Fariborz Jahanian191dcd72009-12-12 21:26:21 +000035#include "llvm/ADT/SmallPtrSet.h"
John McCall8e3f8612010-07-13 22:12:14 +000036#include "llvm/Support/CallSite.h"
Daniel Dunbar33063492009-09-07 00:20:42 +000037#include "llvm/Support/raw_ostream.h"
Daniel Dunbar4e2d7d02008-08-12 06:48:42 +000038#include "llvm/Target/TargetData.h"
Torok Edwinf42e4a62009-08-24 13:25:12 +000039#include <cstdio>
Daniel Dunbarc17a4d32008-08-11 02:45:11 +000040
41using namespace clang;
Daniel Dunbar46f45b92008-09-09 01:06:48 +000042using namespace CodeGen;
Daniel Dunbarc17a4d32008-08-11 02:45:11 +000043
Daniel Dunbar97776872009-04-22 07:32:20 +000044// Common CGObjCRuntime functions, these don't belong here, but they
45// don't belong in CGObjCRuntime either so we will live with it for
46// now.
47
Daniel Dunbar1d7e5392009-05-03 08:55:17 +000048static uint64_t LookupFieldBitOffset(CodeGen::CodeGenModule &CGM,
49 const ObjCInterfaceDecl *OID,
Daniel Dunbar9f89f2b2009-05-03 12:57:56 +000050 const ObjCImplementationDecl *ID,
Daniel Dunbar1d7e5392009-05-03 08:55:17 +000051 const ObjCIvarDecl *Ivar) {
Daniel Dunbare83be122010-04-02 21:14:02 +000052 const ObjCInterfaceDecl *Container = Ivar->getContainingInterface();
Daniel Dunbar532d4da2009-05-03 13:15:50 +000053
Daniel Dunbar61ac1d22010-04-02 22:29:40 +000054 // FIXME: We should eliminate the need to have ObjCImplementationDecl passed
55 // in here; it should never be necessary because that should be the lexical
56 // decl context for the ivar.
Daniel Dunbar3a2c80f2010-04-02 15:43:29 +000057
Daniel Dunbar532d4da2009-05-03 13:15:50 +000058 // If we know have an implementation (and the ivar is in it) then
59 // look up in the implementation layout.
Daniel Dunbar6bff2512009-08-03 17:06:42 +000060 const ASTRecordLayout *RL;
Daniel Dunbar532d4da2009-05-03 13:15:50 +000061 if (ID && ID->getClassInterface() == Container)
62 RL = &CGM.getContext().getASTObjCImplementationLayout(ID);
63 else
64 RL = &CGM.getContext().getASTObjCInterfaceLayout(Container);
Daniel Dunbare83be122010-04-02 21:14:02 +000065
66 // Compute field index.
67 //
68 // FIXME: The index here is closely tied to how ASTContext::getObjCLayout is
69 // implemented. This should be fixed to get the information from the layout
70 // directly.
71 unsigned Index = 0;
72 llvm::SmallVector<ObjCIvarDecl*, 16> Ivars;
73 CGM.getContext().ShallowCollectObjCIvars(Container, Ivars);
74 for (unsigned k = 0, e = Ivars.size(); k != e; ++k) {
75 if (Ivar == Ivars[k])
76 break;
77 ++Index;
78 }
79 assert(Index != Ivars.size() && "Ivar is not inside container!");
80
Daniel Dunbar532d4da2009-05-03 13:15:50 +000081 return RL->getFieldOffset(Index);
Daniel Dunbar1d7e5392009-05-03 08:55:17 +000082}
83
84uint64_t CGObjCRuntime::ComputeIvarBaseOffset(CodeGen::CodeGenModule &CGM,
85 const ObjCInterfaceDecl *OID,
86 const ObjCIvarDecl *Ivar) {
Daniel Dunbar9f89f2b2009-05-03 12:57:56 +000087 return LookupFieldBitOffset(CGM, OID, 0, Ivar) / 8;
88}
89
90uint64_t CGObjCRuntime::ComputeIvarBaseOffset(CodeGen::CodeGenModule &CGM,
91 const ObjCImplementationDecl *OID,
92 const ObjCIvarDecl *Ivar) {
93 return LookupFieldBitOffset(CGM, OID->getClassInterface(), OID, Ivar) / 8;
Daniel Dunbar97776872009-04-22 07:32:20 +000094}
95
96LValue CGObjCRuntime::EmitValueForIvarAtOffset(CodeGen::CodeGenFunction &CGF,
97 const ObjCInterfaceDecl *OID,
98 llvm::Value *BaseValue,
99 const ObjCIvarDecl *Ivar,
100 unsigned CVRQualifiers,
101 llvm::Value *Offset) {
Daniel Dunbar1d7e5392009-05-03 08:55:17 +0000102 // Compute (type*) ( (char *) BaseValue + Offset)
Benjamin Kramer3c0ef8c2009-10-13 10:07:13 +0000103 const llvm::Type *I8Ptr = llvm::Type::getInt8PtrTy(CGF.getLLVMContext());
Daniel Dunbar1d7e5392009-05-03 08:55:17 +0000104 QualType IvarTy = Ivar->getType();
105 const llvm::Type *LTy = CGF.CGM.getTypes().ConvertTypeForMem(IvarTy);
Daniel Dunbar97776872009-04-22 07:32:20 +0000106 llvm::Value *V = CGF.Builder.CreateBitCast(BaseValue, I8Ptr);
Daniel Dunbar97776872009-04-22 07:32:20 +0000107 V = CGF.Builder.CreateGEP(V, Offset, "add.ptr");
Owen Anderson96e0fc72009-07-29 22:16:19 +0000108 V = CGF.Builder.CreateBitCast(V, llvm::PointerType::getUnqual(LTy));
Daniel Dunbar6bff2512009-08-03 17:06:42 +0000109
John McCall0953e762009-09-24 19:53:00 +0000110 Qualifiers Quals = CGF.MakeQualifiers(IvarTy);
111 Quals.addCVRQualifiers(CVRQualifiers);
112
Daniel Dunbar56229f52010-04-05 16:20:33 +0000113 if (!Ivar->isBitField())
114 return LValue::MakeAddr(V, Quals);
Daniel Dunbar9f89f2b2009-05-03 12:57:56 +0000115
Daniel Dunbar56229f52010-04-05 16:20:33 +0000116 // We need to compute the bit offset for the bit-field, the offset is to the
117 // byte. Note, there is a subtle invariant here: we can only call this routine
118 // on non-synthesized ivars but we may be called for synthesized ivars.
119 // However, a synthesized ivar can never be a bit-field, so this is safe.
120 uint64_t BitOffset = LookupFieldBitOffset(CGF.CGM, OID, 0, Ivar) % 8;
121 uint64_t BitFieldSize =
122 Ivar->getBitWidth()->EvaluateAsInt(CGF.getContext()).getZExtValue();
Daniel Dunbar97776872009-04-22 07:32:20 +0000123
Daniel Dunbarf0fe5bc2010-04-05 21:36:35 +0000124 // Allocate a new CGBitFieldInfo object to describe this access.
125 //
126 // FIXME: This is incredibly wasteful, these should be uniqued or part of some
127 // layout object. However, this is blocked on other cleanups to the
128 // Objective-C code, so for now we just live with allocating a bunch of these
129 // objects.
Daniel Dunbarf0fe5bc2010-04-05 21:36:35 +0000130
Daniel Dunbarab970f92010-04-13 20:58:55 +0000131 // We always construct a single, possibly unaligned, access for this case.
Daniel Dunbar2df25692010-04-15 05:09:32 +0000132 CGBitFieldInfo::AccessInfo AI;
Daniel Dunbarab970f92010-04-13 20:58:55 +0000133 AI.FieldIndex = 0;
134 AI.FieldByteOffset = 0;
135 AI.FieldBitStart = BitOffset;
136 AI.AccessWidth = CGF.CGM.getContext().getTypeSize(IvarTy);
137 AI.AccessAlignment = 0;
138 AI.TargetBitOffset = 0;
139 AI.TargetBitWidth = BitFieldSize;
140
Daniel Dunbar2df25692010-04-15 05:09:32 +0000141 CGBitFieldInfo *Info =
142 new (CGF.CGM.getContext()) CGBitFieldInfo(BitFieldSize, 1, &AI,
143 IvarTy->isSignedIntegerType());
144
Daniel Dunbar56229f52010-04-05 16:20:33 +0000145 // FIXME: We need to set a very conservative alignment on this, or make sure
146 // that the runtime is doing the right thing.
Daniel Dunbarefbf4872010-04-06 01:07:44 +0000147 return LValue::MakeBitfield(V, *Info, Quals.getCVRQualifiers());
Daniel Dunbar97776872009-04-22 07:32:20 +0000148}
149
150///
151
Daniel Dunbarc17a4d32008-08-11 02:45:11 +0000152namespace {
Daniel Dunbarbbce49b2008-08-12 00:12:39 +0000153
Daniel Dunbar6bff2512009-08-03 17:06:42 +0000154typedef std::vector<llvm::Constant*> ConstantVector;
Daniel Dunbarae226fa2008-08-27 02:31:56 +0000155
Daniel Dunbar6bff2512009-08-03 17:06:42 +0000156// FIXME: We should find a nicer way to make the labels for metadata, string
157// concatenation is lame.
Daniel Dunbar6efc0c52008-08-13 03:21:16 +0000158
Fariborz Jahanianee0af742009-01-21 22:04:16 +0000159class ObjCCommonTypesHelper {
Owen Andersona1cf15f2009-07-14 23:10:40 +0000160protected:
161 llvm::LLVMContext &VMContext;
Daniel Dunbar6bff2512009-08-03 17:06:42 +0000162
Fariborz Jahaniand0f8a8d2009-05-11 19:25:47 +0000163private:
164 llvm::Constant *getMessageSendFn() const {
165 // id objc_msgSend (id, SEL, ...)
166 std::vector<const llvm::Type*> Params;
167 Params.push_back(ObjectPtrTy);
168 Params.push_back(SelectorPtrTy);
169 return
Daniel Dunbar6bff2512009-08-03 17:06:42 +0000170 CGM.CreateRuntimeFunction(llvm::FunctionType::get(ObjectPtrTy,
171 Params, true),
172 "objc_msgSend");
Fariborz Jahaniand0f8a8d2009-05-11 19:25:47 +0000173 }
Daniel Dunbar6bff2512009-08-03 17:06:42 +0000174
Fariborz Jahaniand0f8a8d2009-05-11 19:25:47 +0000175 llvm::Constant *getMessageSendStretFn() const {
176 // id objc_msgSend_stret (id, SEL, ...)
177 std::vector<const llvm::Type*> Params;
178 Params.push_back(ObjectPtrTy);
179 Params.push_back(SelectorPtrTy);
180 return
Owen Anderson0032b272009-08-13 21:57:51 +0000181 CGM.CreateRuntimeFunction(llvm::FunctionType::get(llvm::Type::getVoidTy(VMContext),
Daniel Dunbar6bff2512009-08-03 17:06:42 +0000182 Params, true),
183 "objc_msgSend_stret");
184
Fariborz Jahaniand0f8a8d2009-05-11 19:25:47 +0000185 }
Daniel Dunbar6bff2512009-08-03 17:06:42 +0000186
Fariborz Jahaniand0f8a8d2009-05-11 19:25:47 +0000187 llvm::Constant *getMessageSendFpretFn() const {
188 // FIXME: This should be long double on x86_64?
189 // [double | long double] objc_msgSend_fpret(id self, SEL op, ...)
190 std::vector<const llvm::Type*> Params;
191 Params.push_back(ObjectPtrTy);
192 Params.push_back(SelectorPtrTy);
193 return
Owen Anderson0032b272009-08-13 21:57:51 +0000194 CGM.CreateRuntimeFunction(llvm::FunctionType::get(
195 llvm::Type::getDoubleTy(VMContext),
Daniel Dunbar6bff2512009-08-03 17:06:42 +0000196 Params,
197 true),
198 "objc_msgSend_fpret");
199
Fariborz Jahaniand0f8a8d2009-05-11 19:25:47 +0000200 }
Daniel Dunbar6bff2512009-08-03 17:06:42 +0000201
Fariborz Jahaniand0f8a8d2009-05-11 19:25:47 +0000202 llvm::Constant *getMessageSendSuperFn() const {
203 // id objc_msgSendSuper(struct objc_super *super, SEL op, ...)
204 const char *SuperName = "objc_msgSendSuper";
205 std::vector<const llvm::Type*> Params;
206 Params.push_back(SuperPtrTy);
207 Params.push_back(SelectorPtrTy);
Owen Anderson96e0fc72009-07-29 22:16:19 +0000208 return CGM.CreateRuntimeFunction(llvm::FunctionType::get(ObjectPtrTy,
Fariborz Jahaniand0f8a8d2009-05-11 19:25:47 +0000209 Params, true),
210 SuperName);
211 }
Daniel Dunbar6bff2512009-08-03 17:06:42 +0000212
Fariborz Jahaniand0f8a8d2009-05-11 19:25:47 +0000213 llvm::Constant *getMessageSendSuperFn2() const {
214 // id objc_msgSendSuper2(struct objc_super *super, SEL op, ...)
215 const char *SuperName = "objc_msgSendSuper2";
216 std::vector<const llvm::Type*> Params;
217 Params.push_back(SuperPtrTy);
218 Params.push_back(SelectorPtrTy);
Owen Anderson96e0fc72009-07-29 22:16:19 +0000219 return CGM.CreateRuntimeFunction(llvm::FunctionType::get(ObjectPtrTy,
Fariborz Jahaniand0f8a8d2009-05-11 19:25:47 +0000220 Params, true),
221 SuperName);
222 }
Daniel Dunbar6bff2512009-08-03 17:06:42 +0000223
Fariborz Jahaniand0f8a8d2009-05-11 19:25:47 +0000224 llvm::Constant *getMessageSendSuperStretFn() const {
225 // void objc_msgSendSuper_stret(void * stretAddr, struct objc_super *super,
226 // SEL op, ...)
227 std::vector<const llvm::Type*> Params;
228 Params.push_back(Int8PtrTy);
229 Params.push_back(SuperPtrTy);
230 Params.push_back(SelectorPtrTy);
Owen Andersona1cf15f2009-07-14 23:10:40 +0000231 return CGM.CreateRuntimeFunction(
Owen Anderson0032b272009-08-13 21:57:51 +0000232 llvm::FunctionType::get(llvm::Type::getVoidTy(VMContext),
Daniel Dunbar6bff2512009-08-03 17:06:42 +0000233 Params, true),
234 "objc_msgSendSuper_stret");
Fariborz Jahaniand0f8a8d2009-05-11 19:25:47 +0000235 }
Daniel Dunbar6bff2512009-08-03 17:06:42 +0000236
Fariborz Jahaniand0f8a8d2009-05-11 19:25:47 +0000237 llvm::Constant *getMessageSendSuperStretFn2() const {
238 // void objc_msgSendSuper2_stret(void * stretAddr, struct objc_super *super,
239 // SEL op, ...)
240 std::vector<const llvm::Type*> Params;
241 Params.push_back(Int8PtrTy);
242 Params.push_back(SuperPtrTy);
243 Params.push_back(SelectorPtrTy);
Owen Andersona1cf15f2009-07-14 23:10:40 +0000244 return CGM.CreateRuntimeFunction(
Owen Anderson0032b272009-08-13 21:57:51 +0000245 llvm::FunctionType::get(llvm::Type::getVoidTy(VMContext),
Daniel Dunbar6bff2512009-08-03 17:06:42 +0000246 Params, true),
247 "objc_msgSendSuper2_stret");
Fariborz Jahaniand0f8a8d2009-05-11 19:25:47 +0000248 }
Daniel Dunbar6bff2512009-08-03 17:06:42 +0000249
Fariborz Jahaniand0f8a8d2009-05-11 19:25:47 +0000250 llvm::Constant *getMessageSendSuperFpretFn() const {
251 // There is no objc_msgSendSuper_fpret? How can that work?
252 return getMessageSendSuperFn();
253 }
Daniel Dunbar6bff2512009-08-03 17:06:42 +0000254
Fariborz Jahaniand0f8a8d2009-05-11 19:25:47 +0000255 llvm::Constant *getMessageSendSuperFpretFn2() const {
256 // There is no objc_msgSendSuper_fpret? How can that work?
257 return getMessageSendSuperFn2();
258 }
Daniel Dunbar6bff2512009-08-03 17:06:42 +0000259
Fariborz Jahanianee0af742009-01-21 22:04:16 +0000260protected:
261 CodeGen::CodeGenModule &CGM;
Daniel Dunbar6bff2512009-08-03 17:06:42 +0000262
Daniel Dunbarbbce49b2008-08-12 00:12:39 +0000263public:
Fariborz Jahanian0a855d02009-03-23 19:10:40 +0000264 const llvm::Type *ShortTy, *IntTy, *LongTy, *LongLongTy;
Daniel Dunbar27f9d772008-08-21 04:36:09 +0000265 const llvm::Type *Int8PtrTy;
Daniel Dunbar6bff2512009-08-03 17:06:42 +0000266
Daniel Dunbar2bedbf82008-08-12 05:28:47 +0000267 /// ObjectPtrTy - LLVM type for object handles (typeof(id))
268 const llvm::Type *ObjectPtrTy;
Daniel Dunbar6bff2512009-08-03 17:06:42 +0000269
Fariborz Jahanian6d657c42008-11-18 20:18:11 +0000270 /// PtrObjectPtrTy - LLVM type for id *
271 const llvm::Type *PtrObjectPtrTy;
Daniel Dunbar6bff2512009-08-03 17:06:42 +0000272
Daniel Dunbar4e2d7d02008-08-12 06:48:42 +0000273 /// SelectorPtrTy - LLVM type for selector handles (typeof(SEL))
Daniel Dunbar2bedbf82008-08-12 05:28:47 +0000274 const llvm::Type *SelectorPtrTy;
Daniel Dunbar6efc0c52008-08-13 03:21:16 +0000275 /// ProtocolPtrTy - LLVM type for external protocol handles
276 /// (typeof(Protocol))
277 const llvm::Type *ExternalProtocolPtrTy;
Daniel Dunbar6bff2512009-08-03 17:06:42 +0000278
Daniel Dunbar19cd87e2008-08-30 03:02:31 +0000279 // SuperCTy - clang type for struct objc_super.
280 QualType SuperCTy;
281 // SuperPtrCTy - clang type for struct objc_super *.
282 QualType SuperPtrCTy;
Daniel Dunbar6bff2512009-08-03 17:06:42 +0000283
Daniel Dunbare8b470d2008-08-23 04:28:29 +0000284 /// SuperTy - LLVM type for struct objc_super.
285 const llvm::StructType *SuperTy;
Daniel Dunbar14c80b72008-08-23 09:25:55 +0000286 /// SuperPtrTy - LLVM type for struct objc_super *.
287 const llvm::Type *SuperPtrTy;
Daniel Dunbar6bff2512009-08-03 17:06:42 +0000288
Fariborz Jahanian30bc5712009-01-22 23:02:58 +0000289 /// PropertyTy - LLVM type for struct objc_property (struct _prop_t
290 /// in GCC parlance).
291 const llvm::StructType *PropertyTy;
Daniel Dunbar6bff2512009-08-03 17:06:42 +0000292
Fariborz Jahanian30bc5712009-01-22 23:02:58 +0000293 /// PropertyListTy - LLVM type for struct objc_property_list
294 /// (_prop_list_t in GCC parlance).
295 const llvm::StructType *PropertyListTy;
296 /// PropertyListPtrTy - LLVM type for struct objc_property_list*.
297 const llvm::Type *PropertyListPtrTy;
Daniel Dunbar6bff2512009-08-03 17:06:42 +0000298
Fariborz Jahanian30bc5712009-01-22 23:02:58 +0000299 // MethodTy - LLVM type for struct objc_method.
300 const llvm::StructType *MethodTy;
Daniel Dunbar6bff2512009-08-03 17:06:42 +0000301
Fariborz Jahaniand55b6fc2009-01-23 01:46:23 +0000302 /// CacheTy - LLVM type for struct objc_cache.
303 const llvm::Type *CacheTy;
304 /// CachePtrTy - LLVM type for struct objc_cache *.
305 const llvm::Type *CachePtrTy;
Daniel Dunbar6bff2512009-08-03 17:06:42 +0000306
Chris Lattner72db6c32009-04-22 02:44:54 +0000307 llvm::Constant *getGetPropertyFn() {
308 CodeGen::CodeGenTypes &Types = CGM.getTypes();
309 ASTContext &Ctx = CGM.getContext();
310 // id objc_getProperty (id, SEL, ptrdiff_t, bool)
John McCallead608a2010-02-26 00:48:12 +0000311 llvm::SmallVector<CanQualType,4> Params;
312 CanQualType IdType = Ctx.getCanonicalParamType(Ctx.getObjCIdType());
313 CanQualType SelType = Ctx.getCanonicalParamType(Ctx.getObjCSelType());
Chris Lattner72db6c32009-04-22 02:44:54 +0000314 Params.push_back(IdType);
315 Params.push_back(SelType);
316 Params.push_back(Ctx.LongTy);
317 Params.push_back(Ctx.BoolTy);
318 const llvm::FunctionType *FTy =
John McCall04a67a62010-02-05 21:31:56 +0000319 Types.GetFunctionType(Types.getFunctionInfo(IdType, Params,
Rafael Espindola264ba482010-03-30 20:24:48 +0000320 FunctionType::ExtInfo()),
321 false);
Chris Lattner72db6c32009-04-22 02:44:54 +0000322 return CGM.CreateRuntimeFunction(FTy, "objc_getProperty");
323 }
Daniel Dunbar6bff2512009-08-03 17:06:42 +0000324
Chris Lattner72db6c32009-04-22 02:44:54 +0000325 llvm::Constant *getSetPropertyFn() {
326 CodeGen::CodeGenTypes &Types = CGM.getTypes();
327 ASTContext &Ctx = CGM.getContext();
328 // void objc_setProperty (id, SEL, ptrdiff_t, id, bool, bool)
John McCallead608a2010-02-26 00:48:12 +0000329 llvm::SmallVector<CanQualType,6> Params;
330 CanQualType IdType = Ctx.getCanonicalParamType(Ctx.getObjCIdType());
331 CanQualType SelType = Ctx.getCanonicalParamType(Ctx.getObjCSelType());
Chris Lattner72db6c32009-04-22 02:44:54 +0000332 Params.push_back(IdType);
333 Params.push_back(SelType);
334 Params.push_back(Ctx.LongTy);
335 Params.push_back(IdType);
336 Params.push_back(Ctx.BoolTy);
337 Params.push_back(Ctx.BoolTy);
338 const llvm::FunctionType *FTy =
John McCall04a67a62010-02-05 21:31:56 +0000339 Types.GetFunctionType(Types.getFunctionInfo(Ctx.VoidTy, Params,
Rafael Espindola264ba482010-03-30 20:24:48 +0000340 FunctionType::ExtInfo()),
341 false);
Chris Lattner72db6c32009-04-22 02:44:54 +0000342 return CGM.CreateRuntimeFunction(FTy, "objc_setProperty");
343 }
Daniel Dunbar6bff2512009-08-03 17:06:42 +0000344
Fariborz Jahanian6cc59062010-04-12 18:18:10 +0000345
346 llvm::Constant *getCopyStructFn() {
347 CodeGen::CodeGenTypes &Types = CGM.getTypes();
348 ASTContext &Ctx = CGM.getContext();
349 // void objc_copyStruct (void *, const void *, size_t, bool, bool)
350 llvm::SmallVector<CanQualType,5> Params;
351 Params.push_back(Ctx.VoidPtrTy);
352 Params.push_back(Ctx.VoidPtrTy);
353 Params.push_back(Ctx.LongTy);
354 Params.push_back(Ctx.BoolTy);
355 Params.push_back(Ctx.BoolTy);
356 const llvm::FunctionType *FTy =
357 Types.GetFunctionType(Types.getFunctionInfo(Ctx.VoidTy, Params,
358 FunctionType::ExtInfo()),
359 false);
360 return CGM.CreateRuntimeFunction(FTy, "objc_copyStruct");
361 }
362
Chris Lattner72db6c32009-04-22 02:44:54 +0000363 llvm::Constant *getEnumerationMutationFn() {
Daniel Dunbarc1ab9002009-07-11 20:32:50 +0000364 CodeGen::CodeGenTypes &Types = CGM.getTypes();
365 ASTContext &Ctx = CGM.getContext();
Chris Lattner72db6c32009-04-22 02:44:54 +0000366 // void objc_enumerationMutation (id)
John McCallead608a2010-02-26 00:48:12 +0000367 llvm::SmallVector<CanQualType,1> Params;
368 Params.push_back(Ctx.getCanonicalParamType(Ctx.getObjCIdType()));
Daniel Dunbarc1ab9002009-07-11 20:32:50 +0000369 const llvm::FunctionType *FTy =
John McCall04a67a62010-02-05 21:31:56 +0000370 Types.GetFunctionType(Types.getFunctionInfo(Ctx.VoidTy, Params,
Rafael Espindola264ba482010-03-30 20:24:48 +0000371 FunctionType::ExtInfo()),
372 false);
Chris Lattner72db6c32009-04-22 02:44:54 +0000373 return CGM.CreateRuntimeFunction(FTy, "objc_enumerationMutation");
374 }
Daniel Dunbar6bff2512009-08-03 17:06:42 +0000375
Fariborz Jahaniandb286862009-01-22 00:37:21 +0000376 /// GcReadWeakFn -- LLVM objc_read_weak (id *src) function.
Chris Lattner72db6c32009-04-22 02:44:54 +0000377 llvm::Constant *getGcReadWeakFn() {
378 // id objc_read_weak (id *)
379 std::vector<const llvm::Type*> Args;
380 Args.push_back(ObjectPtrTy->getPointerTo());
Daniel Dunbar6bff2512009-08-03 17:06:42 +0000381 llvm::FunctionType *FTy =
Owen Anderson96e0fc72009-07-29 22:16:19 +0000382 llvm::FunctionType::get(ObjectPtrTy, Args, false);
Chris Lattner72db6c32009-04-22 02:44:54 +0000383 return CGM.CreateRuntimeFunction(FTy, "objc_read_weak");
Daniel Dunbar6bff2512009-08-03 17:06:42 +0000384 }
385
Fariborz Jahaniandb286862009-01-22 00:37:21 +0000386 /// GcAssignWeakFn -- LLVM objc_assign_weak function.
Chris Lattner96508e12009-04-17 22:12:36 +0000387 llvm::Constant *getGcAssignWeakFn() {
388 // id objc_assign_weak (id, id *)
389 std::vector<const llvm::Type*> Args(1, ObjectPtrTy);
390 Args.push_back(ObjectPtrTy->getPointerTo());
391 llvm::FunctionType *FTy =
Owen Anderson96e0fc72009-07-29 22:16:19 +0000392 llvm::FunctionType::get(ObjectPtrTy, Args, false);
Chris Lattner96508e12009-04-17 22:12:36 +0000393 return CGM.CreateRuntimeFunction(FTy, "objc_assign_weak");
394 }
Daniel Dunbar6bff2512009-08-03 17:06:42 +0000395
Fariborz Jahaniandb286862009-01-22 00:37:21 +0000396 /// GcAssignGlobalFn -- LLVM objc_assign_global function.
Chris Lattnerbbccd612009-04-22 02:38:11 +0000397 llvm::Constant *getGcAssignGlobalFn() {
398 // id objc_assign_global(id, id *)
399 std::vector<const llvm::Type*> Args(1, ObjectPtrTy);
400 Args.push_back(ObjectPtrTy->getPointerTo());
Owen Andersona1cf15f2009-07-14 23:10:40 +0000401 llvm::FunctionType *FTy =
Owen Anderson96e0fc72009-07-29 22:16:19 +0000402 llvm::FunctionType::get(ObjectPtrTy, Args, false);
Chris Lattnerbbccd612009-04-22 02:38:11 +0000403 return CGM.CreateRuntimeFunction(FTy, "objc_assign_global");
404 }
Daniel Dunbar6bff2512009-08-03 17:06:42 +0000405
Fariborz Jahanian021a7a62010-07-20 20:30:03 +0000406 /// GcAssignThreadLocalFn -- LLVM objc_assign_threadlocal function.
407 llvm::Constant *getGcAssignThreadLocalFn() {
408 // id objc_assign_threadlocal(id src, id * dest)
409 std::vector<const llvm::Type*> Args(1, ObjectPtrTy);
410 Args.push_back(ObjectPtrTy->getPointerTo());
411 llvm::FunctionType *FTy =
412 llvm::FunctionType::get(ObjectPtrTy, Args, false);
413 return CGM.CreateRuntimeFunction(FTy, "objc_assign_threadlocal");
414 }
415
Fariborz Jahaniandb286862009-01-22 00:37:21 +0000416 /// GcAssignIvarFn -- LLVM objc_assign_ivar function.
Chris Lattnerbbccd612009-04-22 02:38:11 +0000417 llvm::Constant *getGcAssignIvarFn() {
Fariborz Jahanian6c7a1f32009-09-24 22:25:38 +0000418 // id objc_assign_ivar(id, id *, ptrdiff_t)
Chris Lattnerbbccd612009-04-22 02:38:11 +0000419 std::vector<const llvm::Type*> Args(1, ObjectPtrTy);
420 Args.push_back(ObjectPtrTy->getPointerTo());
Fariborz Jahanian6c7a1f32009-09-24 22:25:38 +0000421 Args.push_back(LongTy);
Owen Andersona1cf15f2009-07-14 23:10:40 +0000422 llvm::FunctionType *FTy =
Owen Anderson96e0fc72009-07-29 22:16:19 +0000423 llvm::FunctionType::get(ObjectPtrTy, Args, false);
Chris Lattnerbbccd612009-04-22 02:38:11 +0000424 return CGM.CreateRuntimeFunction(FTy, "objc_assign_ivar");
425 }
Daniel Dunbar6bff2512009-08-03 17:06:42 +0000426
Fariborz Jahanian082b02e2009-07-08 01:18:33 +0000427 /// GcMemmoveCollectableFn -- LLVM objc_memmove_collectable function.
428 llvm::Constant *GcMemmoveCollectableFn() {
429 // void *objc_memmove_collectable(void *dst, const void *src, size_t size)
430 std::vector<const llvm::Type*> Args(1, Int8PtrTy);
431 Args.push_back(Int8PtrTy);
432 Args.push_back(LongTy);
Owen Anderson96e0fc72009-07-29 22:16:19 +0000433 llvm::FunctionType *FTy = llvm::FunctionType::get(Int8PtrTy, Args, false);
Fariborz Jahanian082b02e2009-07-08 01:18:33 +0000434 return CGM.CreateRuntimeFunction(FTy, "objc_memmove_collectable");
435 }
Daniel Dunbar6bff2512009-08-03 17:06:42 +0000436
Fariborz Jahaniandb286862009-01-22 00:37:21 +0000437 /// GcAssignStrongCastFn -- LLVM objc_assign_strongCast function.
Chris Lattnerbbccd612009-04-22 02:38:11 +0000438 llvm::Constant *getGcAssignStrongCastFn() {
Fariborz Jahanian021a7a62010-07-20 20:30:03 +0000439 // id objc_assign_strongCast(id, id *)
Chris Lattnerbbccd612009-04-22 02:38:11 +0000440 std::vector<const llvm::Type*> Args(1, ObjectPtrTy);
441 Args.push_back(ObjectPtrTy->getPointerTo());
Owen Andersona1cf15f2009-07-14 23:10:40 +0000442 llvm::FunctionType *FTy =
Owen Anderson96e0fc72009-07-29 22:16:19 +0000443 llvm::FunctionType::get(ObjectPtrTy, Args, false);
Chris Lattnerbbccd612009-04-22 02:38:11 +0000444 return CGM.CreateRuntimeFunction(FTy, "objc_assign_strongCast");
445 }
Anders Carlssonf57c5b22009-02-16 22:59:18 +0000446
447 /// ExceptionThrowFn - LLVM objc_exception_throw function.
Chris Lattnerbbccd612009-04-22 02:38:11 +0000448 llvm::Constant *getExceptionThrowFn() {
449 // void objc_exception_throw(id)
450 std::vector<const llvm::Type*> Args(1, ObjectPtrTy);
451 llvm::FunctionType *FTy =
Owen Anderson0032b272009-08-13 21:57:51 +0000452 llvm::FunctionType::get(llvm::Type::getVoidTy(VMContext), Args, false);
Chris Lattnerbbccd612009-04-22 02:38:11 +0000453 return CGM.CreateRuntimeFunction(FTy, "objc_exception_throw");
454 }
Daniel Dunbar6bff2512009-08-03 17:06:42 +0000455
Fariborz Jahanian69677ea2010-05-28 17:34:43 +0000456 /// ExceptionRethrowFn - LLVM objc_exception_rethrow function.
457 llvm::Constant *getExceptionRethrowFn() {
458 // void objc_exception_rethrow(void)
459 std::vector<const llvm::Type*> Args;
460 llvm::FunctionType *FTy =
461 llvm::FunctionType::get(llvm::Type::getVoidTy(VMContext), Args, true);
462 return CGM.CreateRuntimeFunction(FTy, "objc_exception_rethrow");
463 }
464
Daniel Dunbar1c566672009-02-24 01:43:46 +0000465 /// SyncEnterFn - LLVM object_sync_enter function.
Chris Lattnerb02e53b2009-04-06 16:53:45 +0000466 llvm::Constant *getSyncEnterFn() {
467 // void objc_sync_enter (id)
468 std::vector<const llvm::Type*> Args(1, ObjectPtrTy);
469 llvm::FunctionType *FTy =
Owen Anderson0032b272009-08-13 21:57:51 +0000470 llvm::FunctionType::get(llvm::Type::getVoidTy(VMContext), Args, false);
Chris Lattnerb02e53b2009-04-06 16:53:45 +0000471 return CGM.CreateRuntimeFunction(FTy, "objc_sync_enter");
472 }
Daniel Dunbar6bff2512009-08-03 17:06:42 +0000473
Daniel Dunbar1c566672009-02-24 01:43:46 +0000474 /// SyncExitFn - LLVM object_sync_exit function.
Chris Lattnerbbccd612009-04-22 02:38:11 +0000475 llvm::Constant *getSyncExitFn() {
476 // void objc_sync_exit (id)
477 std::vector<const llvm::Type*> Args(1, ObjectPtrTy);
478 llvm::FunctionType *FTy =
Owen Anderson0032b272009-08-13 21:57:51 +0000479 llvm::FunctionType::get(llvm::Type::getVoidTy(VMContext), Args, false);
Chris Lattnerbbccd612009-04-22 02:38:11 +0000480 return CGM.CreateRuntimeFunction(FTy, "objc_sync_exit");
481 }
Daniel Dunbar6bff2512009-08-03 17:06:42 +0000482
Fariborz Jahaniand0f8a8d2009-05-11 19:25:47 +0000483 llvm::Constant *getSendFn(bool IsSuper) const {
484 return IsSuper ? getMessageSendSuperFn() : getMessageSendFn();
485 }
Daniel Dunbar6bff2512009-08-03 17:06:42 +0000486
Fariborz Jahaniand0f8a8d2009-05-11 19:25:47 +0000487 llvm::Constant *getSendFn2(bool IsSuper) const {
488 return IsSuper ? getMessageSendSuperFn2() : getMessageSendFn();
489 }
Daniel Dunbar6bff2512009-08-03 17:06:42 +0000490
Fariborz Jahaniand0f8a8d2009-05-11 19:25:47 +0000491 llvm::Constant *getSendStretFn(bool IsSuper) const {
492 return IsSuper ? getMessageSendSuperStretFn() : getMessageSendStretFn();
493 }
Daniel Dunbar6bff2512009-08-03 17:06:42 +0000494
Fariborz Jahaniand0f8a8d2009-05-11 19:25:47 +0000495 llvm::Constant *getSendStretFn2(bool IsSuper) const {
496 return IsSuper ? getMessageSendSuperStretFn2() : getMessageSendStretFn();
497 }
Daniel Dunbar6bff2512009-08-03 17:06:42 +0000498
Fariborz Jahaniand0f8a8d2009-05-11 19:25:47 +0000499 llvm::Constant *getSendFpretFn(bool IsSuper) const {
500 return IsSuper ? getMessageSendSuperFpretFn() : getMessageSendFpretFn();
501 }
Daniel Dunbar6bff2512009-08-03 17:06:42 +0000502
Fariborz Jahaniand0f8a8d2009-05-11 19:25:47 +0000503 llvm::Constant *getSendFpretFn2(bool IsSuper) const {
504 return IsSuper ? getMessageSendSuperFpretFn2() : getMessageSendFpretFn();
505 }
Daniel Dunbar6bff2512009-08-03 17:06:42 +0000506
Fariborz Jahanianee0af742009-01-21 22:04:16 +0000507 ObjCCommonTypesHelper(CodeGen::CodeGenModule &cgm);
508 ~ObjCCommonTypesHelper(){}
509};
Daniel Dunbare8b470d2008-08-23 04:28:29 +0000510
Fariborz Jahanianee0af742009-01-21 22:04:16 +0000511/// ObjCTypesHelper - Helper class that encapsulates lazy
512/// construction of varies types used during ObjC generation.
513class ObjCTypesHelper : public ObjCCommonTypesHelper {
Fariborz Jahanianee0af742009-01-21 22:04:16 +0000514public:
Daniel Dunbar4e2d7d02008-08-12 06:48:42 +0000515 /// SymtabTy - LLVM type for struct objc_symtab.
516 const llvm::StructType *SymtabTy;
Daniel Dunbar27f9d772008-08-21 04:36:09 +0000517 /// SymtabPtrTy - LLVM type for struct objc_symtab *.
518 const llvm::Type *SymtabPtrTy;
Daniel Dunbar4e2d7d02008-08-12 06:48:42 +0000519 /// ModuleTy - LLVM type for struct objc_module.
520 const llvm::StructType *ModuleTy;
Daniel Dunbar259d93d2008-08-12 03:39:23 +0000521
Daniel Dunbar6efc0c52008-08-13 03:21:16 +0000522 /// ProtocolTy - LLVM type for struct objc_protocol.
523 const llvm::StructType *ProtocolTy;
524 /// ProtocolPtrTy - LLVM type for struct objc_protocol *.
525 const llvm::Type *ProtocolPtrTy;
526 /// ProtocolExtensionTy - LLVM type for struct
527 /// objc_protocol_extension.
528 const llvm::StructType *ProtocolExtensionTy;
529 /// ProtocolExtensionTy - LLVM type for struct
530 /// objc_protocol_extension *.
531 const llvm::Type *ProtocolExtensionPtrTy;
532 /// MethodDescriptionTy - LLVM type for struct
533 /// objc_method_description.
534 const llvm::StructType *MethodDescriptionTy;
535 /// MethodDescriptionListTy - LLVM type for struct
536 /// objc_method_description_list.
537 const llvm::StructType *MethodDescriptionListTy;
538 /// MethodDescriptionListPtrTy - LLVM type for struct
539 /// objc_method_description_list *.
540 const llvm::Type *MethodDescriptionListPtrTy;
Daniel Dunbar6efc0c52008-08-13 03:21:16 +0000541 /// ProtocolListTy - LLVM type for struct objc_property_list.
542 const llvm::Type *ProtocolListTy;
543 /// ProtocolListPtrTy - LLVM type for struct objc_property_list*.
544 const llvm::Type *ProtocolListPtrTy;
Daniel Dunbar86e253a2008-08-22 20:34:54 +0000545 /// CategoryTy - LLVM type for struct objc_category.
546 const llvm::StructType *CategoryTy;
Daniel Dunbar27f9d772008-08-21 04:36:09 +0000547 /// ClassTy - LLVM type for struct objc_class.
548 const llvm::StructType *ClassTy;
549 /// ClassPtrTy - LLVM type for struct objc_class *.
550 const llvm::Type *ClassPtrTy;
551 /// ClassExtensionTy - LLVM type for struct objc_class_ext.
552 const llvm::StructType *ClassExtensionTy;
553 /// ClassExtensionPtrTy - LLVM type for struct objc_class_ext *.
554 const llvm::Type *ClassExtensionPtrTy;
Daniel Dunbar27f9d772008-08-21 04:36:09 +0000555 // IvarTy - LLVM type for struct objc_ivar.
556 const llvm::StructType *IvarTy;
557 /// IvarListTy - LLVM type for struct objc_ivar_list.
558 const llvm::Type *IvarListTy;
559 /// IvarListPtrTy - LLVM type for struct objc_ivar_list *.
560 const llvm::Type *IvarListPtrTy;
Daniel Dunbar27f9d772008-08-21 04:36:09 +0000561 /// MethodListTy - LLVM type for struct objc_method_list.
562 const llvm::Type *MethodListTy;
563 /// MethodListPtrTy - LLVM type for struct objc_method_list *.
564 const llvm::Type *MethodListPtrTy;
Daniel Dunbar6bff2512009-08-03 17:06:42 +0000565
Anders Carlsson124526b2008-09-09 10:10:21 +0000566 /// ExceptionDataTy - LLVM type for struct _objc_exception_data.
567 const llvm::Type *ExceptionDataTy;
Daniel Dunbar6bff2512009-08-03 17:06:42 +0000568
Anders Carlsson124526b2008-09-09 10:10:21 +0000569 /// ExceptionTryEnterFn - LLVM objc_exception_try_enter function.
Chris Lattner34b02a12009-04-22 02:26:14 +0000570 llvm::Constant *getExceptionTryEnterFn() {
571 std::vector<const llvm::Type*> Params;
Owen Anderson96e0fc72009-07-29 22:16:19 +0000572 Params.push_back(llvm::PointerType::getUnqual(ExceptionDataTy));
Owen Andersona1cf15f2009-07-14 23:10:40 +0000573 return CGM.CreateRuntimeFunction(
Owen Anderson0032b272009-08-13 21:57:51 +0000574 llvm::FunctionType::get(llvm::Type::getVoidTy(VMContext),
Daniel Dunbar6bff2512009-08-03 17:06:42 +0000575 Params, false),
576 "objc_exception_try_enter");
Chris Lattner34b02a12009-04-22 02:26:14 +0000577 }
Anders Carlsson124526b2008-09-09 10:10:21 +0000578
579 /// ExceptionTryExitFn - LLVM objc_exception_try_exit function.
Chris Lattner34b02a12009-04-22 02:26:14 +0000580 llvm::Constant *getExceptionTryExitFn() {
581 std::vector<const llvm::Type*> Params;
Owen Anderson96e0fc72009-07-29 22:16:19 +0000582 Params.push_back(llvm::PointerType::getUnqual(ExceptionDataTy));
Owen Andersona1cf15f2009-07-14 23:10:40 +0000583 return CGM.CreateRuntimeFunction(
Owen Anderson0032b272009-08-13 21:57:51 +0000584 llvm::FunctionType::get(llvm::Type::getVoidTy(VMContext),
Daniel Dunbar6bff2512009-08-03 17:06:42 +0000585 Params, false),
586 "objc_exception_try_exit");
Chris Lattner34b02a12009-04-22 02:26:14 +0000587 }
Anders Carlsson124526b2008-09-09 10:10:21 +0000588
589 /// ExceptionExtractFn - LLVM objc_exception_extract function.
Chris Lattner34b02a12009-04-22 02:26:14 +0000590 llvm::Constant *getExceptionExtractFn() {
591 std::vector<const llvm::Type*> Params;
Owen Anderson96e0fc72009-07-29 22:16:19 +0000592 Params.push_back(llvm::PointerType::getUnqual(ExceptionDataTy));
593 return CGM.CreateRuntimeFunction(llvm::FunctionType::get(ObjectPtrTy,
Chris Lattner34b02a12009-04-22 02:26:14 +0000594 Params, false),
595 "objc_exception_extract");
Daniel Dunbar6bff2512009-08-03 17:06:42 +0000596
Chris Lattner34b02a12009-04-22 02:26:14 +0000597 }
Daniel Dunbar6bff2512009-08-03 17:06:42 +0000598
Anders Carlsson124526b2008-09-09 10:10:21 +0000599 /// ExceptionMatchFn - LLVM objc_exception_match function.
Chris Lattner34b02a12009-04-22 02:26:14 +0000600 llvm::Constant *getExceptionMatchFn() {
601 std::vector<const llvm::Type*> Params;
602 Params.push_back(ClassPtrTy);
603 Params.push_back(ObjectPtrTy);
Daniel Dunbar6bff2512009-08-03 17:06:42 +0000604 return CGM.CreateRuntimeFunction(
Owen Anderson0032b272009-08-13 21:57:51 +0000605 llvm::FunctionType::get(llvm::Type::getInt32Ty(VMContext),
Daniel Dunbar6bff2512009-08-03 17:06:42 +0000606 Params, false),
607 "objc_exception_match");
608
Chris Lattner34b02a12009-04-22 02:26:14 +0000609 }
Daniel Dunbar6bff2512009-08-03 17:06:42 +0000610
Anders Carlsson124526b2008-09-09 10:10:21 +0000611 /// SetJmpFn - LLVM _setjmp function.
Chris Lattner34b02a12009-04-22 02:26:14 +0000612 llvm::Constant *getSetJmpFn() {
613 std::vector<const llvm::Type*> Params;
Benjamin Kramer3c0ef8c2009-10-13 10:07:13 +0000614 Params.push_back(llvm::Type::getInt32PtrTy(VMContext));
Chris Lattner34b02a12009-04-22 02:26:14 +0000615 return
Owen Anderson0032b272009-08-13 21:57:51 +0000616 CGM.CreateRuntimeFunction(llvm::FunctionType::get(llvm::Type::getInt32Ty(VMContext),
Chris Lattner34b02a12009-04-22 02:26:14 +0000617 Params, false),
618 "_setjmp");
Daniel Dunbar6bff2512009-08-03 17:06:42 +0000619
Chris Lattner34b02a12009-04-22 02:26:14 +0000620 }
Daniel Dunbar6bff2512009-08-03 17:06:42 +0000621
Daniel Dunbarbbce49b2008-08-12 00:12:39 +0000622public:
623 ObjCTypesHelper(CodeGen::CodeGenModule &cgm);
Fariborz Jahanianee0af742009-01-21 22:04:16 +0000624 ~ObjCTypesHelper() {}
Daniel Dunbarbbce49b2008-08-12 00:12:39 +0000625};
626
Fariborz Jahanian30bc5712009-01-22 23:02:58 +0000627/// ObjCNonFragileABITypesHelper - will have all types needed by objective-c's
Fariborz Jahanianee0af742009-01-21 22:04:16 +0000628/// modern abi
Fariborz Jahanian30bc5712009-01-22 23:02:58 +0000629class ObjCNonFragileABITypesHelper : public ObjCCommonTypesHelper {
Fariborz Jahanian83a8a752009-02-04 20:42:28 +0000630public:
Daniel Dunbar6bff2512009-08-03 17:06:42 +0000631
Fariborz Jahaniand55b6fc2009-01-23 01:46:23 +0000632 // MethodListnfABITy - LLVM for struct _method_list_t
633 const llvm::StructType *MethodListnfABITy;
Daniel Dunbar6bff2512009-08-03 17:06:42 +0000634
Fariborz Jahaniand55b6fc2009-01-23 01:46:23 +0000635 // MethodListnfABIPtrTy - LLVM for struct _method_list_t*
636 const llvm::Type *MethodListnfABIPtrTy;
Daniel Dunbar6bff2512009-08-03 17:06:42 +0000637
Fariborz Jahaniand55b6fc2009-01-23 01:46:23 +0000638 // ProtocolnfABITy = LLVM for struct _protocol_t
639 const llvm::StructType *ProtocolnfABITy;
Daniel Dunbar6bff2512009-08-03 17:06:42 +0000640
Daniel Dunbar948e2582009-02-15 07:36:20 +0000641 // ProtocolnfABIPtrTy = LLVM for struct _protocol_t*
642 const llvm::Type *ProtocolnfABIPtrTy;
643
Fariborz Jahaniand55b6fc2009-01-23 01:46:23 +0000644 // ProtocolListnfABITy - LLVM for struct _objc_protocol_list
645 const llvm::StructType *ProtocolListnfABITy;
Daniel Dunbar6bff2512009-08-03 17:06:42 +0000646
Fariborz Jahaniand55b6fc2009-01-23 01:46:23 +0000647 // ProtocolListnfABIPtrTy - LLVM for struct _objc_protocol_list*
648 const llvm::Type *ProtocolListnfABIPtrTy;
Daniel Dunbar6bff2512009-08-03 17:06:42 +0000649
Fariborz Jahaniand55b6fc2009-01-23 01:46:23 +0000650 // ClassnfABITy - LLVM for struct _class_t
651 const llvm::StructType *ClassnfABITy;
Daniel Dunbar6bff2512009-08-03 17:06:42 +0000652
Fariborz Jahanianaa23b572009-01-23 23:53:38 +0000653 // ClassnfABIPtrTy - LLVM for struct _class_t*
654 const llvm::Type *ClassnfABIPtrTy;
Daniel Dunbar6bff2512009-08-03 17:06:42 +0000655
Fariborz Jahaniand55b6fc2009-01-23 01:46:23 +0000656 // IvarnfABITy - LLVM for struct _ivar_t
657 const llvm::StructType *IvarnfABITy;
Daniel Dunbar6bff2512009-08-03 17:06:42 +0000658
Fariborz Jahaniand55b6fc2009-01-23 01:46:23 +0000659 // IvarListnfABITy - LLVM for struct _ivar_list_t
660 const llvm::StructType *IvarListnfABITy;
Daniel Dunbar6bff2512009-08-03 17:06:42 +0000661
Fariborz Jahaniand55b6fc2009-01-23 01:46:23 +0000662 // IvarListnfABIPtrTy = LLVM for struct _ivar_list_t*
663 const llvm::Type *IvarListnfABIPtrTy;
Daniel Dunbar6bff2512009-08-03 17:06:42 +0000664
Fariborz Jahaniand55b6fc2009-01-23 01:46:23 +0000665 // ClassRonfABITy - LLVM for struct _class_ro_t
666 const llvm::StructType *ClassRonfABITy;
Daniel Dunbar6bff2512009-08-03 17:06:42 +0000667
Fariborz Jahaniand55b6fc2009-01-23 01:46:23 +0000668 // ImpnfABITy - LLVM for id (*)(id, SEL, ...)
669 const llvm::Type *ImpnfABITy;
Daniel Dunbar6bff2512009-08-03 17:06:42 +0000670
Fariborz Jahaniand55b6fc2009-01-23 01:46:23 +0000671 // CategorynfABITy - LLVM for struct _category_t
672 const llvm::StructType *CategorynfABITy;
Daniel Dunbar6bff2512009-08-03 17:06:42 +0000673
Fariborz Jahanian2e4672b2009-02-03 23:49:23 +0000674 // New types for nonfragile abi messaging.
Daniel Dunbar6bff2512009-08-03 17:06:42 +0000675
Fariborz Jahanian2e4672b2009-02-03 23:49:23 +0000676 // MessageRefTy - LLVM for:
677 // struct _message_ref_t {
678 // IMP messenger;
679 // SEL name;
680 // };
681 const llvm::StructType *MessageRefTy;
Fariborz Jahanian83a8a752009-02-04 20:42:28 +0000682 // MessageRefCTy - clang type for struct _message_ref_t
683 QualType MessageRefCTy;
Daniel Dunbar6bff2512009-08-03 17:06:42 +0000684
Fariborz Jahanian2e4672b2009-02-03 23:49:23 +0000685 // MessageRefPtrTy - LLVM for struct _message_ref_t*
686 const llvm::Type *MessageRefPtrTy;
Fariborz Jahanian83a8a752009-02-04 20:42:28 +0000687 // MessageRefCPtrTy - clang type for struct _message_ref_t*
688 QualType MessageRefCPtrTy;
Daniel Dunbar6bff2512009-08-03 17:06:42 +0000689
Fariborz Jahanianef163782009-02-05 01:13:09 +0000690 // MessengerTy - Type of the messenger (shown as IMP above)
691 const llvm::FunctionType *MessengerTy;
Daniel Dunbar6bff2512009-08-03 17:06:42 +0000692
Fariborz Jahanian2e4672b2009-02-03 23:49:23 +0000693 // SuperMessageRefTy - LLVM for:
694 // struct _super_message_ref_t {
695 // SUPER_IMP messenger;
696 // SEL name;
697 // };
698 const llvm::StructType *SuperMessageRefTy;
Daniel Dunbar6bff2512009-08-03 17:06:42 +0000699
Fariborz Jahanian2e4672b2009-02-03 23:49:23 +0000700 // SuperMessageRefPtrTy - LLVM for struct _super_message_ref_t*
701 const llvm::Type *SuperMessageRefPtrTy;
Daniel Dunbar8ecbaf22009-02-24 07:47:38 +0000702
Chris Lattner1c02f862009-04-22 02:53:24 +0000703 llvm::Constant *getMessageSendFixupFn() {
704 // id objc_msgSend_fixup(id, struct message_ref_t*, ...)
705 std::vector<const llvm::Type*> Params;
706 Params.push_back(ObjectPtrTy);
707 Params.push_back(MessageRefPtrTy);
Owen Anderson96e0fc72009-07-29 22:16:19 +0000708 return CGM.CreateRuntimeFunction(llvm::FunctionType::get(ObjectPtrTy,
Chris Lattner1c02f862009-04-22 02:53:24 +0000709 Params, true),
710 "objc_msgSend_fixup");
711 }
Daniel Dunbar6bff2512009-08-03 17:06:42 +0000712
Chris Lattner1c02f862009-04-22 02:53:24 +0000713 llvm::Constant *getMessageSendFpretFixupFn() {
714 // id objc_msgSend_fpret_fixup(id, struct message_ref_t*, ...)
715 std::vector<const llvm::Type*> Params;
716 Params.push_back(ObjectPtrTy);
717 Params.push_back(MessageRefPtrTy);
Owen Anderson96e0fc72009-07-29 22:16:19 +0000718 return CGM.CreateRuntimeFunction(llvm::FunctionType::get(ObjectPtrTy,
Chris Lattner1c02f862009-04-22 02:53:24 +0000719 Params, true),
720 "objc_msgSend_fpret_fixup");
721 }
Daniel Dunbar6bff2512009-08-03 17:06:42 +0000722
Chris Lattner1c02f862009-04-22 02:53:24 +0000723 llvm::Constant *getMessageSendStretFixupFn() {
724 // id objc_msgSend_stret_fixup(id, struct message_ref_t*, ...)
725 std::vector<const llvm::Type*> Params;
726 Params.push_back(ObjectPtrTy);
727 Params.push_back(MessageRefPtrTy);
Owen Anderson96e0fc72009-07-29 22:16:19 +0000728 return CGM.CreateRuntimeFunction(llvm::FunctionType::get(ObjectPtrTy,
Chris Lattner1c02f862009-04-22 02:53:24 +0000729 Params, true),
730 "objc_msgSend_stret_fixup");
731 }
Daniel Dunbar6bff2512009-08-03 17:06:42 +0000732
Chris Lattner1c02f862009-04-22 02:53:24 +0000733 llvm::Constant *getMessageSendIdFixupFn() {
734 // id objc_msgSendId_fixup(id, struct message_ref_t*, ...)
735 std::vector<const llvm::Type*> Params;
736 Params.push_back(ObjectPtrTy);
737 Params.push_back(MessageRefPtrTy);
Owen Anderson96e0fc72009-07-29 22:16:19 +0000738 return CGM.CreateRuntimeFunction(llvm::FunctionType::get(ObjectPtrTy,
Chris Lattner1c02f862009-04-22 02:53:24 +0000739 Params, true),
740 "objc_msgSendId_fixup");
741 }
Daniel Dunbar6bff2512009-08-03 17:06:42 +0000742
Chris Lattner1c02f862009-04-22 02:53:24 +0000743 llvm::Constant *getMessageSendIdStretFixupFn() {
744 // id objc_msgSendId_stret_fixup(id, struct message_ref_t*, ...)
745 std::vector<const llvm::Type*> Params;
746 Params.push_back(ObjectPtrTy);
747 Params.push_back(MessageRefPtrTy);
Owen Anderson96e0fc72009-07-29 22:16:19 +0000748 return CGM.CreateRuntimeFunction(llvm::FunctionType::get(ObjectPtrTy,
Chris Lattner1c02f862009-04-22 02:53:24 +0000749 Params, true),
750 "objc_msgSendId_stret_fixup");
751 }
752 llvm::Constant *getMessageSendSuper2FixupFn() {
Daniel Dunbar6bff2512009-08-03 17:06:42 +0000753 // id objc_msgSendSuper2_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_fixup");
761 }
Daniel Dunbar6bff2512009-08-03 17:06:42 +0000762
Chris Lattner1c02f862009-04-22 02:53:24 +0000763 llvm::Constant *getMessageSendSuper2StretFixupFn() {
Daniel Dunbar6bff2512009-08-03 17:06:42 +0000764 // id objc_msgSendSuper2_stret_fixup(struct objc_super *,
Chris Lattner1c02f862009-04-22 02:53:24 +0000765 // struct _super_message_ref_t*, ...)
766 std::vector<const llvm::Type*> Params;
767 Params.push_back(SuperPtrTy);
768 Params.push_back(SuperMessageRefPtrTy);
Owen Anderson96e0fc72009-07-29 22:16:19 +0000769 return CGM.CreateRuntimeFunction(llvm::FunctionType::get(ObjectPtrTy,
Chris Lattner1c02f862009-04-22 02:53:24 +0000770 Params, true),
771 "objc_msgSendSuper2_stret_fixup");
772 }
Daniel Dunbar6bff2512009-08-03 17:06:42 +0000773
774
775
Daniel Dunbar8ecbaf22009-02-24 07:47:38 +0000776 /// EHPersonalityPtr - LLVM value for an i8* to the Objective-C
777 /// exception personality function.
Chris Lattnerb02e53b2009-04-06 16:53:45 +0000778 llvm::Value *getEHPersonalityPtr() {
Daniel Dunbar6bff2512009-08-03 17:06:42 +0000779 llvm::Constant *Personality =
Owen Anderson0032b272009-08-13 21:57:51 +0000780 CGM.CreateRuntimeFunction(llvm::FunctionType::get(llvm::Type::getInt32Ty(VMContext),
Chris Lattnerb02e53b2009-04-06 16:53:45 +0000781 true),
Daniel Dunbar6bff2512009-08-03 17:06:42 +0000782 "__objc_personality_v0");
Owen Anderson3c4972d2009-07-29 18:54:39 +0000783 return llvm::ConstantExpr::getBitCast(Personality, Int8PtrTy);
Chris Lattnerb02e53b2009-04-06 16:53:45 +0000784 }
Daniel Dunbar8ecbaf22009-02-24 07:47:38 +0000785
Chris Lattner8a569112009-04-22 02:15:23 +0000786 llvm::Constant *getUnwindResumeOrRethrowFn() {
787 std::vector<const llvm::Type*> Params;
788 Params.push_back(Int8PtrTy);
Owen Andersona1cf15f2009-07-14 23:10:40 +0000789 return CGM.CreateRuntimeFunction(
Owen Anderson0032b272009-08-13 21:57:51 +0000790 llvm::FunctionType::get(llvm::Type::getVoidTy(VMContext),
Daniel Dunbar6bff2512009-08-03 17:06:42 +0000791 Params, false),
Daniel Dunbarb2987d12010-02-10 18:49:11 +0000792 (CGM.getLangOptions().SjLjExceptions ? "_Unwind_SjLj_Resume" :
793 "_Unwind_Resume_or_Rethrow"));
Chris Lattner8a569112009-04-22 02:15:23 +0000794 }
Daniel Dunbar6bff2512009-08-03 17:06:42 +0000795
Chris Lattner8a569112009-04-22 02:15:23 +0000796 llvm::Constant *getObjCEndCatchFn() {
Owen Anderson0032b272009-08-13 21:57:51 +0000797 return CGM.CreateRuntimeFunction(llvm::FunctionType::get(llvm::Type::getVoidTy(VMContext),
Chris Lattnerb59761b2009-07-01 04:13:52 +0000798 false),
Chris Lattner8a569112009-04-22 02:15:23 +0000799 "objc_end_catch");
Daniel Dunbar6bff2512009-08-03 17:06:42 +0000800
Chris Lattner8a569112009-04-22 02:15:23 +0000801 }
Daniel Dunbar6bff2512009-08-03 17:06:42 +0000802
Chris Lattner8a569112009-04-22 02:15:23 +0000803 llvm::Constant *getObjCBeginCatchFn() {
804 std::vector<const llvm::Type*> Params;
805 Params.push_back(Int8PtrTy);
Owen Anderson96e0fc72009-07-29 22:16:19 +0000806 return CGM.CreateRuntimeFunction(llvm::FunctionType::get(Int8PtrTy,
Chris Lattner8a569112009-04-22 02:15:23 +0000807 Params, false),
808 "objc_begin_catch");
809 }
Daniel Dunbare588b992009-03-01 04:46:24 +0000810
811 const llvm::StructType *EHTypeTy;
812 const llvm::Type *EHTypePtrTy;
Daniel Dunbar8ecbaf22009-02-24 07:47:38 +0000813
Fariborz Jahanian30bc5712009-01-22 23:02:58 +0000814 ObjCNonFragileABITypesHelper(CodeGen::CodeGenModule &cgm);
815 ~ObjCNonFragileABITypesHelper(){}
Fariborz Jahanianee0af742009-01-21 22:04:16 +0000816};
Daniel Dunbar6bff2512009-08-03 17:06:42 +0000817
Fariborz Jahanianee0af742009-01-21 22:04:16 +0000818class CGObjCCommonMac : public CodeGen::CGObjCRuntime {
Fariborz Jahanian9397e1d2009-03-11 20:59:05 +0000819public:
820 // FIXME - accessibility
Fariborz Jahaniana5a10c32009-03-10 16:22:08 +0000821 class GC_IVAR {
Fariborz Jahanian820e0202009-03-11 00:07:04 +0000822 public:
Daniel Dunbar8b2926c2009-05-03 13:44:42 +0000823 unsigned ivar_bytepos;
824 unsigned ivar_size;
825 GC_IVAR(unsigned bytepos = 0, unsigned size = 0)
Daniel Dunbar6bff2512009-08-03 17:06:42 +0000826 : ivar_bytepos(bytepos), ivar_size(size) {}
Daniel Dunbar0941b492009-04-23 01:29:05 +0000827
828 // Allow sorting based on byte pos.
829 bool operator<(const GC_IVAR &b) const {
830 return ivar_bytepos < b.ivar_bytepos;
831 }
Fariborz Jahaniana5a10c32009-03-10 16:22:08 +0000832 };
Daniel Dunbar6bff2512009-08-03 17:06:42 +0000833
Fariborz Jahanian9397e1d2009-03-11 20:59:05 +0000834 class SKIP_SCAN {
Daniel Dunbar8b2926c2009-05-03 13:44:42 +0000835 public:
836 unsigned skip;
837 unsigned scan;
Daniel Dunbar6bff2512009-08-03 17:06:42 +0000838 SKIP_SCAN(unsigned _skip = 0, unsigned _scan = 0)
Daniel Dunbar8b2926c2009-05-03 13:44:42 +0000839 : skip(_skip), scan(_scan) {}
Fariborz Jahanian9397e1d2009-03-11 20:59:05 +0000840 };
Daniel Dunbar6bff2512009-08-03 17:06:42 +0000841
Fariborz Jahanianee0af742009-01-21 22:04:16 +0000842protected:
843 CodeGen::CodeGenModule &CGM;
Owen Anderson69243822009-07-13 04:10:07 +0000844 llvm::LLVMContext &VMContext;
Fariborz Jahanianee0af742009-01-21 22:04:16 +0000845 // FIXME! May not be needing this after all.
Daniel Dunbarbbce49b2008-08-12 00:12:39 +0000846 unsigned ObjCABI;
Daniel Dunbar6bff2512009-08-03 17:06:42 +0000847
Fariborz Jahanian9397e1d2009-03-11 20:59:05 +0000848 // gc ivar layout bitmap calculation helper caches.
849 llvm::SmallVector<GC_IVAR, 16> SkipIvars;
850 llvm::SmallVector<GC_IVAR, 16> IvarsInfo;
Daniel Dunbar6bff2512009-08-03 17:06:42 +0000851
Daniel Dunbar242d4dc2008-08-25 06:02:07 +0000852 /// LazySymbols - Symbols to generate a lazy reference for. See
853 /// DefinedSymbols and FinishModule().
Daniel Dunbar33063492009-09-07 00:20:42 +0000854 llvm::SetVector<IdentifierInfo*> LazySymbols;
Daniel Dunbar6bff2512009-08-03 17:06:42 +0000855
Daniel Dunbar242d4dc2008-08-25 06:02:07 +0000856 /// DefinedSymbols - External symbols which are defined by this
857 /// module. The symbols in this list and LazySymbols are used to add
858 /// special linker symbols which ensure that Objective-C modules are
859 /// linked properly.
Daniel Dunbar33063492009-09-07 00:20:42 +0000860 llvm::SetVector<IdentifierInfo*> DefinedSymbols;
Daniel Dunbar6bff2512009-08-03 17:06:42 +0000861
Daniel Dunbar4e2d7d02008-08-12 06:48:42 +0000862 /// ClassNames - uniqued class names.
Daniel Dunbar6efc0c52008-08-13 03:21:16 +0000863 llvm::DenseMap<IdentifierInfo*, llvm::GlobalVariable*> ClassNames;
Daniel Dunbar6bff2512009-08-03 17:06:42 +0000864
Daniel Dunbar259d93d2008-08-12 03:39:23 +0000865 /// MethodVarNames - uniqued method variable names.
866 llvm::DenseMap<Selector, llvm::GlobalVariable*> MethodVarNames;
Daniel Dunbar6bff2512009-08-03 17:06:42 +0000867
Fariborz Jahanianb9c5b3d2010-06-21 22:05:18 +0000868 /// DefinedCategoryNames - list of category names in form Class_Category.
869 llvm::SetVector<std::string> DefinedCategoryNames;
870
Daniel Dunbar6efc0c52008-08-13 03:21:16 +0000871 /// MethodVarTypes - uniqued method type signatures. We have to use
872 /// a StringMap here because have no other unique reference.
873 llvm::StringMap<llvm::GlobalVariable*> MethodVarTypes;
Daniel Dunbar6bff2512009-08-03 17:06:42 +0000874
Daniel Dunbarc45ef602008-08-26 21:51:14 +0000875 /// MethodDefinitions - map of methods which have been defined in
876 /// this translation unit.
877 llvm::DenseMap<const ObjCMethodDecl*, llvm::Function*> MethodDefinitions;
Daniel Dunbar6bff2512009-08-03 17:06:42 +0000878
Daniel Dunbarc8ef5512008-08-23 00:19:03 +0000879 /// PropertyNames - uniqued method variable names.
880 llvm::DenseMap<IdentifierInfo*, llvm::GlobalVariable*> PropertyNames;
Daniel Dunbar6bff2512009-08-03 17:06:42 +0000881
Daniel Dunbar27f9d772008-08-21 04:36:09 +0000882 /// ClassReferences - uniqued class references.
883 llvm::DenseMap<IdentifierInfo*, llvm::GlobalVariable*> ClassReferences;
Daniel Dunbar6bff2512009-08-03 17:06:42 +0000884
Daniel Dunbar259d93d2008-08-12 03:39:23 +0000885 /// SelectorReferences - uniqued selector references.
886 llvm::DenseMap<Selector, llvm::GlobalVariable*> SelectorReferences;
Daniel Dunbar6bff2512009-08-03 17:06:42 +0000887
Daniel Dunbar6efc0c52008-08-13 03:21:16 +0000888 /// Protocols - Protocols for which an objc_protocol structure has
889 /// been emitted. Forward declarations are handled by creating an
890 /// empty structure whose initializer is filled in when/if defined.
891 llvm::DenseMap<IdentifierInfo*, llvm::GlobalVariable*> Protocols;
Daniel Dunbar6bff2512009-08-03 17:06:42 +0000892
Daniel Dunbar0c0e7a62008-10-29 22:36:39 +0000893 /// DefinedProtocols - Protocols which have actually been
894 /// defined. We should not need this, see FIXME in GenerateProtocol.
895 llvm::DenseSet<IdentifierInfo*> DefinedProtocols;
Daniel Dunbar6bff2512009-08-03 17:06:42 +0000896
Daniel Dunbar27f9d772008-08-21 04:36:09 +0000897 /// DefinedClasses - List of defined classes.
898 std::vector<llvm::GlobalValue*> DefinedClasses;
Daniel Dunbar74d4b122009-05-15 22:33:15 +0000899
900 /// DefinedNonLazyClasses - List of defined "non-lazy" classes.
901 std::vector<llvm::GlobalValue*> DefinedNonLazyClasses;
Daniel Dunbar6bff2512009-08-03 17:06:42 +0000902
Daniel Dunbar27f9d772008-08-21 04:36:09 +0000903 /// DefinedCategories - List of defined categories.
904 std::vector<llvm::GlobalValue*> DefinedCategories;
Daniel Dunbar6bff2512009-08-03 17:06:42 +0000905
Daniel Dunbar74d4b122009-05-15 22:33:15 +0000906 /// DefinedNonLazyCategories - List of defined "non-lazy" categories.
907 std::vector<llvm::GlobalValue*> DefinedNonLazyCategories;
Daniel Dunbar6bff2512009-08-03 17:06:42 +0000908
Fariborz Jahanian56210f72009-01-21 23:34:32 +0000909 /// GetNameForMethod - Return a name for the given method.
910 /// \param[out] NameOut - The return value.
911 void GetNameForMethod(const ObjCMethodDecl *OMD,
912 const ObjCContainerDecl *CD,
Daniel Dunbarc575ce72009-10-19 01:21:19 +0000913 llvm::SmallVectorImpl<char> &NameOut);
Daniel Dunbar6bff2512009-08-03 17:06:42 +0000914
Fariborz Jahanian56210f72009-01-21 23:34:32 +0000915 /// GetMethodVarName - Return a unique constant for the given
916 /// selector's name. The return value has type char *.
917 llvm::Constant *GetMethodVarName(Selector Sel);
918 llvm::Constant *GetMethodVarName(IdentifierInfo *Ident);
919 llvm::Constant *GetMethodVarName(const std::string &Name);
Daniel Dunbar6bff2512009-08-03 17:06:42 +0000920
Fariborz Jahanian56210f72009-01-21 23:34:32 +0000921 /// GetMethodVarType - Return a unique constant for the given
922 /// selector's name. The return value has type char *.
Daniel Dunbar6bff2512009-08-03 17:06:42 +0000923
Fariborz Jahanian56210f72009-01-21 23:34:32 +0000924 // FIXME: This is a horrible name.
925 llvm::Constant *GetMethodVarType(const ObjCMethodDecl *D);
Daniel Dunbar3e5f0d82009-04-20 06:54:31 +0000926 llvm::Constant *GetMethodVarType(const FieldDecl *D);
Daniel Dunbar6bff2512009-08-03 17:06:42 +0000927
Fariborz Jahanian56210f72009-01-21 23:34:32 +0000928 /// GetPropertyName - Return a unique constant for the given
929 /// name. The return value has type char *.
930 llvm::Constant *GetPropertyName(IdentifierInfo *Ident);
Daniel Dunbar6bff2512009-08-03 17:06:42 +0000931
Fariborz Jahanian56210f72009-01-21 23:34:32 +0000932 // FIXME: This can be dropped once string functions are unified.
933 llvm::Constant *GetPropertyTypeString(const ObjCPropertyDecl *PD,
934 const Decl *Container);
Daniel Dunbar6bff2512009-08-03 17:06:42 +0000935
Fariborz Jahanian058a1b72009-01-24 20:21:50 +0000936 /// GetClassName - Return a unique constant for the given selector's
937 /// name. The return value has type char *.
938 llvm::Constant *GetClassName(IdentifierInfo *Ident);
Daniel Dunbar6bff2512009-08-03 17:06:42 +0000939
Argyrios Kyrtzidis9d50c062010-08-09 10:54:20 +0000940 llvm::Function *GetMethodDefinition(const ObjCMethodDecl *MD);
941
Fariborz Jahaniand61a50a2009-03-05 22:39:55 +0000942 /// BuildIvarLayout - Builds ivar layout bitmap for the class
943 /// implementation for the __strong or __weak case.
944 ///
Fariborz Jahanianc8ce9c82009-03-12 22:50:49 +0000945 llvm::Constant *BuildIvarLayout(const ObjCImplementationDecl *OI,
946 bool ForStrongLayout);
Fariborz Jahanian93ce50d2010-08-04 23:55:24 +0000947
Fariborz Jahanianb8fd2eb2010-08-05 00:19:48 +0000948 llvm::Constant *BuildIvarLayoutBitmap(std::string &BitMap);
Daniel Dunbar6bff2512009-08-03 17:06:42 +0000949
Daniel Dunbard58edcb2009-05-03 14:10:34 +0000950 void BuildAggrIvarRecordLayout(const RecordType *RT,
Daniel Dunbar6bff2512009-08-03 17:06:42 +0000951 unsigned int BytePos, bool ForStrongLayout,
952 bool &HasUnion);
Daniel Dunbar5a5a8032009-05-03 21:05:10 +0000953 void BuildAggrIvarLayout(const ObjCImplementationDecl *OI,
Fariborz Jahanianc8ce9c82009-03-12 22:50:49 +0000954 const llvm::StructLayout *Layout,
Fariborz Jahaniana5a10c32009-03-10 16:22:08 +0000955 const RecordDecl *RD,
Chris Lattnerf1690852009-03-31 08:48:01 +0000956 const llvm::SmallVectorImpl<FieldDecl*> &RecFields,
Fariborz Jahaniand61a50a2009-03-05 22:39:55 +0000957 unsigned int BytePos, bool ForStrongLayout,
Fariborz Jahanian81adc052009-04-24 16:17:09 +0000958 bool &HasUnion);
Fariborz Jahanianc8ce9c82009-03-12 22:50:49 +0000959
Fariborz Jahaniand80d81b2009-03-05 19:17:31 +0000960 /// GetIvarLayoutName - Returns a unique constant for the given
961 /// ivar layout bitmap.
962 llvm::Constant *GetIvarLayoutName(IdentifierInfo *Ident,
963 const ObjCCommonTypesHelper &ObjCTypes);
Daniel Dunbar6bff2512009-08-03 17:06:42 +0000964
Fariborz Jahanian5de14dc2009-01-28 22:18:42 +0000965 /// EmitPropertyList - Emit the given property list. The return
966 /// value has type PropertyListPtrTy.
Daniel Dunbar9c29bf52009-10-18 20:48:59 +0000967 llvm::Constant *EmitPropertyList(llvm::Twine Name,
Daniel Dunbar6bff2512009-08-03 17:06:42 +0000968 const Decl *Container,
Fariborz Jahanian5de14dc2009-01-28 22:18:42 +0000969 const ObjCContainerDecl *OCD,
970 const ObjCCommonTypesHelper &ObjCTypes);
Daniel Dunbar6bff2512009-08-03 17:06:42 +0000971
Fariborz Jahanian191dcd72009-12-12 21:26:21 +0000972 /// PushProtocolProperties - Push protocol's property on the input stack.
973 void PushProtocolProperties(llvm::SmallPtrSet<const IdentifierInfo*, 16> &PropertySet,
974 std::vector<llvm::Constant*> &Properties,
975 const Decl *Container,
976 const ObjCProtocolDecl *PROTO,
977 const ObjCCommonTypesHelper &ObjCTypes);
978
Fariborz Jahanianda320092009-01-29 19:24:30 +0000979 /// GetProtocolRef - Return a reference to the internal protocol
980 /// description, creating an empty one if it has not been
981 /// defined. The return value has type ProtocolPtrTy.
982 llvm::Constant *GetProtocolRef(const ObjCProtocolDecl *PD);
Fariborz Jahanianb21f07e2009-03-08 20:18:37 +0000983
Daniel Dunbarfd65d372009-03-09 20:09:19 +0000984 /// CreateMetadataVar - Create a global variable with internal
985 /// linkage for use by the Objective-C runtime.
986 ///
987 /// This is a convenience wrapper which not only creates the
988 /// variable, but also sets the section and alignment and adds the
Chris Lattnerad64e022009-07-17 23:57:13 +0000989 /// global to the "llvm.used" list.
Daniel Dunbar35bd7632009-03-09 20:50:13 +0000990 ///
991 /// \param Name - The variable name.
992 /// \param Init - The variable initializer; this is also used to
993 /// define the type of the variable.
994 /// \param Section - The section the variable should go into, or 0.
995 /// \param Align - The alignment for the variable, or 0.
996 /// \param AddToUsed - Whether the variable should be added to
Daniel Dunbarc1583062009-04-14 17:42:51 +0000997 /// "llvm.used".
Daniel Dunbar9c29bf52009-10-18 20:48:59 +0000998 llvm::GlobalVariable *CreateMetadataVar(llvm::Twine Name,
Daniel Dunbarfd65d372009-03-09 20:09:19 +0000999 llvm::Constant *Init,
1000 const char *Section,
Daniel Dunbar35bd7632009-03-09 20:50:13 +00001001 unsigned Align,
1002 bool AddToUsed);
Daniel Dunbarfd65d372009-03-09 20:09:19 +00001003
Fariborz Jahaniand0f8a8d2009-05-11 19:25:47 +00001004 CodeGen::RValue EmitLegacyMessageSend(CodeGen::CodeGenFunction &CGF,
John McCallef072fd2010-05-22 01:48:05 +00001005 ReturnValueSlot Return,
Fariborz Jahaniand0f8a8d2009-05-11 19:25:47 +00001006 QualType ResultType,
1007 llvm::Value *Sel,
1008 llvm::Value *Arg0,
1009 QualType Arg0Ty,
1010 bool IsSuper,
1011 const CallArgList &CallArgs,
Daniel Dunbard6c93d72009-09-17 04:01:22 +00001012 const ObjCMethodDecl *OMD,
Fariborz Jahaniand0f8a8d2009-05-11 19:25:47 +00001013 const ObjCCommonTypesHelper &ObjCTypes);
Daniel Dunbar3e5f0d82009-04-20 06:54:31 +00001014
Daniel Dunbarfce176b2010-04-25 20:39:01 +00001015 /// EmitImageInfo - Emit the image info marker used to encode some module
1016 /// level information.
1017 void EmitImageInfo();
1018
Fariborz Jahanianee0af742009-01-21 22:04:16 +00001019public:
Owen Anderson69243822009-07-13 04:10:07 +00001020 CGObjCCommonMac(CodeGen::CodeGenModule &cgm) :
Mike Stump1eb44332009-09-09 15:08:12 +00001021 CGM(cgm), VMContext(cgm.getLLVMContext()) { }
Daniel Dunbar6bff2512009-08-03 17:06:42 +00001022
David Chisnall0d13f6f2010-01-23 02:40:42 +00001023 virtual llvm::Constant *GenerateConstantString(const StringLiteral *SL);
Daniel Dunbar6bff2512009-08-03 17:06:42 +00001024
Fariborz Jahanian493dab72009-01-26 21:38:32 +00001025 virtual llvm::Function *GenerateMethod(const ObjCMethodDecl *OMD,
1026 const ObjCContainerDecl *CD=0);
Daniel Dunbar6bff2512009-08-03 17:06:42 +00001027
Fariborz Jahanianda320092009-01-29 19:24:30 +00001028 virtual void GenerateProtocol(const ObjCProtocolDecl *PD);
Daniel Dunbar6bff2512009-08-03 17:06:42 +00001029
Fariborz Jahanianda320092009-01-29 19:24:30 +00001030 /// GetOrEmitProtocol - Get the protocol object for the given
1031 /// declaration, emitting it if necessary. The return value has type
1032 /// ProtocolPtrTy.
1033 virtual llvm::Constant *GetOrEmitProtocol(const ObjCProtocolDecl *PD)=0;
Daniel Dunbar6bff2512009-08-03 17:06:42 +00001034
Fariborz Jahanianda320092009-01-29 19:24:30 +00001035 /// GetOrEmitProtocolRef - Get a forward reference to the protocol
1036 /// object for the given declaration, emitting it if needed. These
1037 /// forward references will be filled in with empty bodies if no
1038 /// definition is seen. The return value has type ProtocolPtrTy.
1039 virtual llvm::Constant *GetOrEmitProtocolRef(const ObjCProtocolDecl *PD)=0;
Fariborz Jahanian89ecd412010-08-04 16:57:49 +00001040 virtual llvm::Constant *GCBlockLayout(CodeGen::CodeGenFunction &CGF,
1041 const llvm::SmallVectorImpl<const BlockDeclRefExpr *> &);
1042
Fariborz Jahanianee0af742009-01-21 22:04:16 +00001043};
Daniel Dunbar6bff2512009-08-03 17:06:42 +00001044
Fariborz Jahanianee0af742009-01-21 22:04:16 +00001045class CGObjCMac : public CGObjCCommonMac {
1046private:
1047 ObjCTypesHelper ObjCTypes;
Daniel Dunbarf77ac862008-08-11 21:35:06 +00001048
Daniel Dunbar4e2d7d02008-08-12 06:48:42 +00001049 /// EmitModuleInfo - Another marker encoding module level
Daniel Dunbar6bff2512009-08-03 17:06:42 +00001050 /// information.
Daniel Dunbar4e2d7d02008-08-12 06:48:42 +00001051 void EmitModuleInfo();
1052
Daniel Dunbar27f9d772008-08-21 04:36:09 +00001053 /// EmitModuleSymols - Emit module symbols, the list of defined
1054 /// classes and categories. The result has type SymtabPtrTy.
Daniel Dunbar4e2d7d02008-08-12 06:48:42 +00001055 llvm::Constant *EmitModuleSymbols();
1056
Daniel Dunbarf77ac862008-08-11 21:35:06 +00001057 /// FinishModule - Write out global data structures at the end of
1058 /// processing a translation unit.
1059 void FinishModule();
Daniel Dunbar6efc0c52008-08-13 03:21:16 +00001060
Daniel Dunbar27f9d772008-08-21 04:36:09 +00001061 /// EmitClassExtension - Generate the class extension structure used
1062 /// to store the weak ivar layout and properties. The return value
1063 /// has type ClassExtensionPtrTy.
1064 llvm::Constant *EmitClassExtension(const ObjCImplementationDecl *ID);
1065
1066 /// EmitClassRef - Return a Value*, of type ObjCTypes.ClassPtrTy,
1067 /// for the given class.
Daniel Dunbar6bff2512009-08-03 17:06:42 +00001068 llvm::Value *EmitClassRef(CGBuilderTy &Builder,
Daniel Dunbar27f9d772008-08-21 04:36:09 +00001069 const ObjCInterfaceDecl *ID);
Fariborz Jahanianb0069ee2009-11-12 20:14:24 +00001070
1071 /// EmitSuperClassRef - Emits reference to class's main metadata class.
1072 llvm::Value *EmitSuperClassRef(const ObjCInterfaceDecl *ID);
Daniel Dunbar27f9d772008-08-21 04:36:09 +00001073
Daniel Dunbar14c80b72008-08-23 09:25:55 +00001074 CodeGen::RValue EmitMessageSend(CodeGen::CodeGenFunction &CGF,
John McCallef072fd2010-05-22 01:48:05 +00001075 ReturnValueSlot Return,
Daniel Dunbar7f8ea5c2008-08-30 05:35:15 +00001076 QualType ResultType,
1077 Selector Sel,
Daniel Dunbar14c80b72008-08-23 09:25:55 +00001078 llvm::Value *Arg0,
Daniel Dunbar19cd87e2008-08-30 03:02:31 +00001079 QualType Arg0Ty,
1080 bool IsSuper,
1081 const CallArgList &CallArgs);
Daniel Dunbar14c80b72008-08-23 09:25:55 +00001082
Daniel Dunbar27f9d772008-08-21 04:36:09 +00001083 /// EmitIvarList - Emit the ivar list for the given
1084 /// implementation. If ForClass is true the list of class ivars
1085 /// (i.e. metaclass ivars) is emitted, otherwise the list of
1086 /// interface ivars will be emitted. The return value has type
1087 /// IvarListPtrTy.
1088 llvm::Constant *EmitIvarList(const ObjCImplementationDecl *ID,
Fariborz Jahanian46b86c62009-01-28 19:12:34 +00001089 bool ForClass);
Daniel Dunbar6bff2512009-08-03 17:06:42 +00001090
Daniel Dunbarf56f1912008-08-25 08:19:24 +00001091 /// EmitMetaClass - Emit a forward reference to the class structure
1092 /// for the metaclass of the given interface. The return value has
1093 /// type ClassPtrTy.
1094 llvm::Constant *EmitMetaClassRef(const ObjCInterfaceDecl *ID);
1095
Daniel Dunbar27f9d772008-08-21 04:36:09 +00001096 /// EmitMetaClass - Emit a class structure for the metaclass of the
Daniel Dunbarf56f1912008-08-25 08:19:24 +00001097 /// given implementation. The return value has type ClassPtrTy.
Daniel Dunbar27f9d772008-08-21 04:36:09 +00001098 llvm::Constant *EmitMetaClass(const ObjCImplementationDecl *ID,
1099 llvm::Constant *Protocols,
Daniel Dunbarae226fa2008-08-27 02:31:56 +00001100 const ConstantVector &Methods);
Daniel Dunbar6bff2512009-08-03 17:06:42 +00001101
Daniel Dunbarae226fa2008-08-27 02:31:56 +00001102 llvm::Constant *GetMethodConstant(const ObjCMethodDecl *MD);
Daniel Dunbar6bff2512009-08-03 17:06:42 +00001103
Daniel Dunbarae226fa2008-08-27 02:31:56 +00001104 llvm::Constant *GetMethodDescriptionConstant(const ObjCMethodDecl *MD);
Daniel Dunbar27f9d772008-08-21 04:36:09 +00001105
1106 /// EmitMethodList - Emit the method list for the given
Daniel Dunbaraf05bb92008-08-26 08:29:31 +00001107 /// implementation. The return value has type MethodListPtrTy.
Daniel Dunbar9c29bf52009-10-18 20:48:59 +00001108 llvm::Constant *EmitMethodList(llvm::Twine Name,
Daniel Dunbar86e253a2008-08-22 20:34:54 +00001109 const char *Section,
Daniel Dunbarae226fa2008-08-27 02:31:56 +00001110 const ConstantVector &Methods);
Daniel Dunbar27f9d772008-08-21 04:36:09 +00001111
1112 /// EmitMethodDescList - Emit a method description list for a list of
Daniel Dunbar6bff2512009-08-03 17:06:42 +00001113 /// method declarations.
Daniel Dunbar6efc0c52008-08-13 03:21:16 +00001114 /// - TypeName: The name for the type containing the methods.
1115 /// - IsProtocol: True iff these methods are for a protocol.
1116 /// - ClassMethds: True iff these are class methods.
1117 /// - Required: When true, only "required" methods are
1118 /// listed. Similarly, when false only "optional" methods are
1119 /// listed. For classes this should always be true.
1120 /// - begin, end: The method list to output.
1121 ///
1122 /// The return value has type MethodDescriptionListPtrTy.
Daniel Dunbar9c29bf52009-10-18 20:48:59 +00001123 llvm::Constant *EmitMethodDescList(llvm::Twine Name,
Daniel Dunbarae226fa2008-08-27 02:31:56 +00001124 const char *Section,
1125 const ConstantVector &Methods);
Daniel Dunbar6efc0c52008-08-13 03:21:16 +00001126
Daniel Dunbar0c0e7a62008-10-29 22:36:39 +00001127 /// GetOrEmitProtocol - Get the protocol object for the given
1128 /// declaration, emitting it if necessary. The return value has type
1129 /// ProtocolPtrTy.
Fariborz Jahanianda320092009-01-29 19:24:30 +00001130 virtual llvm::Constant *GetOrEmitProtocol(const ObjCProtocolDecl *PD);
Daniel Dunbar0c0e7a62008-10-29 22:36:39 +00001131
1132 /// GetOrEmitProtocolRef - Get a forward reference to the protocol
1133 /// object for the given declaration, emitting it if needed. These
1134 /// forward references will be filled in with empty bodies if no
1135 /// definition is seen. The return value has type ProtocolPtrTy.
Fariborz Jahanianda320092009-01-29 19:24:30 +00001136 virtual llvm::Constant *GetOrEmitProtocolRef(const ObjCProtocolDecl *PD);
Daniel Dunbar0c0e7a62008-10-29 22:36:39 +00001137
Daniel Dunbar6efc0c52008-08-13 03:21:16 +00001138 /// EmitProtocolExtension - Generate the protocol extension
1139 /// structure used to store optional instance and class methods, and
1140 /// protocol properties. The return value has type
1141 /// ProtocolExtensionPtrTy.
Daniel Dunbarae226fa2008-08-27 02:31:56 +00001142 llvm::Constant *
1143 EmitProtocolExtension(const ObjCProtocolDecl *PD,
1144 const ConstantVector &OptInstanceMethods,
1145 const ConstantVector &OptClassMethods);
Daniel Dunbar6efc0c52008-08-13 03:21:16 +00001146
1147 /// EmitProtocolList - Generate the list of referenced
1148 /// protocols. The return value has type ProtocolListPtrTy.
Daniel Dunbar9c29bf52009-10-18 20:48:59 +00001149 llvm::Constant *EmitProtocolList(llvm::Twine Name,
Daniel Dunbardbc933702008-08-21 21:57:41 +00001150 ObjCProtocolDecl::protocol_iterator begin,
1151 ObjCProtocolDecl::protocol_iterator end);
Daniel Dunbar6efc0c52008-08-13 03:21:16 +00001152
Daniel Dunbar27f9d772008-08-21 04:36:09 +00001153 /// EmitSelector - Return a Value*, of type ObjCTypes.SelectorPtrTy,
1154 /// for the given selector.
Fariborz Jahanian03b29602010-06-17 19:56:20 +00001155 llvm::Value *EmitSelector(CGBuilderTy &Builder, Selector Sel,
1156 bool lval=false);
Daniel Dunbar6bff2512009-08-03 17:06:42 +00001157
1158public:
Daniel Dunbarc17a4d32008-08-11 02:45:11 +00001159 CGObjCMac(CodeGen::CodeGenModule &cgm);
Daniel Dunbarc17a4d32008-08-11 02:45:11 +00001160
Fariborz Jahanianaa23b572009-01-23 23:53:38 +00001161 virtual llvm::Function *ModuleInitFunction();
Daniel Dunbar6bff2512009-08-03 17:06:42 +00001162
Daniel Dunbar8f2926b2008-08-23 03:46:30 +00001163 virtual CodeGen::RValue GenerateMessageSend(CodeGen::CodeGenFunction &CGF,
John McCallef072fd2010-05-22 01:48:05 +00001164 ReturnValueSlot Return,
Daniel Dunbar7f8ea5c2008-08-30 05:35:15 +00001165 QualType ResultType,
1166 Selector Sel,
Daniel Dunbarf56f1912008-08-25 08:19:24 +00001167 llvm::Value *Receiver,
Fariborz Jahaniandf9ccc62009-05-05 21:36:57 +00001168 const CallArgList &CallArgs,
David Chisnallc6cd5fd2010-04-28 19:33:36 +00001169 const ObjCInterfaceDecl *Class,
Fariborz Jahaniandf9ccc62009-05-05 21:36:57 +00001170 const ObjCMethodDecl *Method);
Daniel Dunbarc17a4d32008-08-11 02:45:11 +00001171
Daniel Dunbar6bff2512009-08-03 17:06:42 +00001172 virtual CodeGen::RValue
Daniel Dunbar8f2926b2008-08-23 03:46:30 +00001173 GenerateMessageSendSuper(CodeGen::CodeGenFunction &CGF,
John McCallef072fd2010-05-22 01:48:05 +00001174 ReturnValueSlot Return,
Daniel Dunbar7f8ea5c2008-08-30 05:35:15 +00001175 QualType ResultType,
1176 Selector Sel,
Daniel Dunbarf56f1912008-08-25 08:19:24 +00001177 const ObjCInterfaceDecl *Class,
Fariborz Jahanian7ce77922009-02-28 20:07:56 +00001178 bool isCategoryImpl,
Daniel Dunbarf56f1912008-08-25 08:19:24 +00001179 llvm::Value *Receiver,
Daniel Dunbar19cd87e2008-08-30 03:02:31 +00001180 bool IsClassMessage,
Daniel Dunbard6c93d72009-09-17 04:01:22 +00001181 const CallArgList &CallArgs,
1182 const ObjCMethodDecl *Method);
Daniel Dunbar6bff2512009-08-03 17:06:42 +00001183
Daniel Dunbar45d196b2008-11-01 01:53:16 +00001184 virtual llvm::Value *GetClass(CGBuilderTy &Builder,
Daniel Dunbar27f9d772008-08-21 04:36:09 +00001185 const ObjCInterfaceDecl *ID);
Daniel Dunbarc17a4d32008-08-11 02:45:11 +00001186
Fariborz Jahanian03b29602010-06-17 19:56:20 +00001187 virtual llvm::Value *GetSelector(CGBuilderTy &Builder, Selector Sel,
1188 bool lval = false);
Fariborz Jahaniandf9ccc62009-05-05 21:36:57 +00001189
1190 /// The NeXT/Apple runtimes do not support typed selectors; just emit an
1191 /// untyped one.
1192 virtual llvm::Value *GetSelector(CGBuilderTy &Builder,
1193 const ObjCMethodDecl *Method);
1194
John McCall5a180392010-07-24 00:37:23 +00001195 virtual llvm::Constant *GetEHType(QualType T);
1196
Daniel Dunbar7ded7f42008-08-15 22:20:32 +00001197 virtual void GenerateCategory(const ObjCCategoryImplDecl *CMD);
Daniel Dunbarc17a4d32008-08-11 02:45:11 +00001198
Daniel Dunbar7ded7f42008-08-15 22:20:32 +00001199 virtual void GenerateClass(const ObjCImplementationDecl *ClassDecl);
Daniel Dunbarc17a4d32008-08-11 02:45:11 +00001200
Daniel Dunbar45d196b2008-11-01 01:53:16 +00001201 virtual llvm::Value *GenerateProtocolRef(CGBuilderTy &Builder,
Daniel Dunbaraf2f62c2008-08-13 00:59:25 +00001202 const ObjCProtocolDecl *PD);
Daniel Dunbar6bff2512009-08-03 17:06:42 +00001203
Chris Lattner74391b42009-03-22 21:03:39 +00001204 virtual llvm::Constant *GetPropertyGetFunction();
1205 virtual llvm::Constant *GetPropertySetFunction();
Fariborz Jahanian6cc59062010-04-12 18:18:10 +00001206 virtual llvm::Constant *GetCopyStructFunction();
Chris Lattner74391b42009-03-22 21:03:39 +00001207 virtual llvm::Constant *EnumerationMutationFunction();
Daniel Dunbar6bff2512009-08-03 17:06:42 +00001208
John McCallf1549f62010-07-06 01:34:17 +00001209 virtual void EmitTryStmt(CodeGen::CodeGenFunction &CGF,
1210 const ObjCAtTryStmt &S);
1211 virtual void EmitSynchronizedStmt(CodeGen::CodeGenFunction &CGF,
1212 const ObjCAtSynchronizedStmt &S);
1213 void EmitTryOrSynchronizedStmt(CodeGen::CodeGenFunction &CGF, const Stmt &S);
Anders Carlsson64d5d6c2008-09-09 10:04:29 +00001214 virtual void EmitThrowStmt(CodeGen::CodeGenFunction &CGF,
1215 const ObjCAtThrowStmt &S);
Fariborz Jahanian3e283e32008-11-18 22:37:34 +00001216 virtual llvm::Value * EmitObjCWeakRead(CodeGen::CodeGenFunction &CGF,
Daniel Dunbar6bff2512009-08-03 17:06:42 +00001217 llvm::Value *AddrWeakObj);
Fariborz Jahanian3e283e32008-11-18 22:37:34 +00001218 virtual void EmitObjCWeakAssign(CodeGen::CodeGenFunction &CGF,
Daniel Dunbar6bff2512009-08-03 17:06:42 +00001219 llvm::Value *src, llvm::Value *dst);
Fariborz Jahanian58626502008-11-19 00:59:10 +00001220 virtual void EmitObjCGlobalAssign(CodeGen::CodeGenFunction &CGF,
Fariborz Jahanian021a7a62010-07-20 20:30:03 +00001221 llvm::Value *src, llvm::Value *dest,
1222 bool threadlocal = false);
Fariborz Jahanian7eda8362008-11-20 19:23:36 +00001223 virtual void EmitObjCIvarAssign(CodeGen::CodeGenFunction &CGF,
Fariborz Jahanian6c7a1f32009-09-24 22:25:38 +00001224 llvm::Value *src, llvm::Value *dest,
1225 llvm::Value *ivarOffset);
Fariborz Jahanian58626502008-11-19 00:59:10 +00001226 virtual void EmitObjCStrongCastAssign(CodeGen::CodeGenFunction &CGF,
1227 llvm::Value *src, llvm::Value *dest);
Fariborz Jahanian082b02e2009-07-08 01:18:33 +00001228 virtual void EmitGCMemmoveCollectable(CodeGen::CodeGenFunction &CGF,
1229 llvm::Value *dest, llvm::Value *src,
Fariborz Jahanian55bcace2010-06-15 22:44:06 +00001230 llvm::Value *size);
Daniel Dunbar6bff2512009-08-03 17:06:42 +00001231
Fariborz Jahanian598d3f62009-02-03 19:03:09 +00001232 virtual LValue EmitObjCValueForIvar(CodeGen::CodeGenFunction &CGF,
1233 QualType ObjectTy,
1234 llvm::Value *BaseValue,
1235 const ObjCIvarDecl *Ivar,
Fariborz Jahanian598d3f62009-02-03 19:03:09 +00001236 unsigned CVRQualifiers);
Fariborz Jahanianf63aa3f2009-02-10 19:02:04 +00001237 virtual llvm::Value *EmitIvarOffset(CodeGen::CodeGenFunction &CGF,
Daniel Dunbar2a031922009-04-22 05:08:15 +00001238 const ObjCInterfaceDecl *Interface,
Fariborz Jahanianf63aa3f2009-02-10 19:02:04 +00001239 const ObjCIvarDecl *Ivar);
Daniel Dunbarc17a4d32008-08-11 02:45:11 +00001240};
Daniel Dunbar6bff2512009-08-03 17:06:42 +00001241
Fariborz Jahanian30bc5712009-01-22 23:02:58 +00001242class CGObjCNonFragileABIMac : public CGObjCCommonMac {
Fariborz Jahanianee0af742009-01-21 22:04:16 +00001243private:
Fariborz Jahanian30bc5712009-01-22 23:02:58 +00001244 ObjCNonFragileABITypesHelper ObjCTypes;
Fariborz Jahanianaa23b572009-01-23 23:53:38 +00001245 llvm::GlobalVariable* ObjCEmptyCacheVar;
1246 llvm::GlobalVariable* ObjCEmptyVtableVar;
Daniel Dunbar6bff2512009-08-03 17:06:42 +00001247
Daniel Dunbar11394522009-04-18 08:51:00 +00001248 /// SuperClassReferences - uniqued super class references.
1249 llvm::DenseMap<IdentifierInfo*, llvm::GlobalVariable*> SuperClassReferences;
Daniel Dunbar6bff2512009-08-03 17:06:42 +00001250
Fariborz Jahanian7a06aae2009-02-06 20:09:23 +00001251 /// MetaClassReferences - uniqued meta class references.
1252 llvm::DenseMap<IdentifierInfo*, llvm::GlobalVariable*> MetaClassReferences;
Daniel Dunbare588b992009-03-01 04:46:24 +00001253
1254 /// EHTypeReferences - uniqued class ehtype references.
1255 llvm::DenseMap<IdentifierInfo*, llvm::GlobalVariable*> EHTypeReferences;
Daniel Dunbar6bff2512009-08-03 17:06:42 +00001256
Fariborz Jahanian4523eb02009-05-12 20:06:41 +00001257 /// NonLegacyDispatchMethods - List of methods for which we do *not* generate
Fariborz Jahaniand0f8a8d2009-05-11 19:25:47 +00001258 /// legacy messaging dispatch.
Fariborz Jahanian4523eb02009-05-12 20:06:41 +00001259 llvm::DenseSet<Selector> NonLegacyDispatchMethods;
Daniel Dunbar6bff2512009-08-03 17:06:42 +00001260
Fariborz Jahaniana03d0dd2009-11-17 21:37:35 +00001261 /// DefinedMetaClasses - List of defined meta-classes.
1262 std::vector<llvm::GlobalValue*> DefinedMetaClasses;
1263
Fariborz Jahaniand0f8a8d2009-05-11 19:25:47 +00001264 /// LegacyDispatchedSelector - Returns true if SEL is not in the list of
Fariborz Jahanian4523eb02009-05-12 20:06:41 +00001265 /// NonLegacyDispatchMethods; false otherwise.
Fariborz Jahaniand0f8a8d2009-05-11 19:25:47 +00001266 bool LegacyDispatchedSelector(Selector Sel);
Daniel Dunbar6bff2512009-08-03 17:06:42 +00001267
Fariborz Jahanianaa23b572009-01-23 23:53:38 +00001268 /// FinishNonFragileABIModule - Write out global data structures at the end of
1269 /// processing a translation unit.
1270 void FinishNonFragileABIModule();
Daniel Dunbar8158a2f2009-04-08 04:21:03 +00001271
Daniel Dunbar463b8762009-05-15 21:48:48 +00001272 /// AddModuleClassList - Add the given list of class pointers to the
1273 /// module with the provided symbol and section names.
1274 void AddModuleClassList(const std::vector<llvm::GlobalValue*> &Container,
1275 const char *SymbolName,
1276 const char *SectionName);
1277
Daniel Dunbar6bff2512009-08-03 17:06:42 +00001278 llvm::GlobalVariable * BuildClassRoTInitializer(unsigned flags,
1279 unsigned InstanceStart,
1280 unsigned InstanceSize,
1281 const ObjCImplementationDecl *ID);
Fariborz Jahanian84394a52009-01-24 21:21:53 +00001282 llvm::GlobalVariable * BuildClassMetaData(std::string &ClassName,
Daniel Dunbar6bff2512009-08-03 17:06:42 +00001283 llvm::Constant *IsAGV,
Fariborz Jahanian84394a52009-01-24 21:21:53 +00001284 llvm::Constant *SuperClassGV,
Fariborz Jahaniancf555162009-01-31 00:59:10 +00001285 llvm::Constant *ClassRoGV,
1286 bool HiddenVisibility);
Daniel Dunbar6bff2512009-08-03 17:06:42 +00001287
Fariborz Jahanian493dab72009-01-26 21:38:32 +00001288 llvm::Constant *GetMethodConstant(const ObjCMethodDecl *MD);
Daniel Dunbar6bff2512009-08-03 17:06:42 +00001289
Fariborz Jahanian3819a0b2009-01-30 00:46:37 +00001290 llvm::Constant *GetMethodDescriptionConstant(const ObjCMethodDecl *MD);
Daniel Dunbar6bff2512009-08-03 17:06:42 +00001291
Fariborz Jahanian493dab72009-01-26 21:38:32 +00001292 /// EmitMethodList - Emit the method list for the given
1293 /// implementation. The return value has type MethodListnfABITy.
Daniel Dunbar9c29bf52009-10-18 20:48:59 +00001294 llvm::Constant *EmitMethodList(llvm::Twine Name,
Fariborz Jahanian493dab72009-01-26 21:38:32 +00001295 const char *Section,
Fariborz Jahanian98abf4b2009-01-27 19:38:51 +00001296 const ConstantVector &Methods);
1297 /// EmitIvarList - Emit the ivar list for the given
1298 /// implementation. If ForClass is true the list of class ivars
1299 /// (i.e. metaclass ivars) is emitted, otherwise the list of
1300 /// interface ivars will be emitted. The return value has type
1301 /// IvarListnfABIPtrTy.
1302 llvm::Constant *EmitIvarList(const ObjCImplementationDecl *ID);
Daniel Dunbar6bff2512009-08-03 17:06:42 +00001303
Fariborz Jahanianed157d32009-02-10 20:21:06 +00001304 llvm::Constant *EmitIvarOffsetVar(const ObjCInterfaceDecl *ID,
Fariborz Jahanian2fa5a272009-01-28 01:36:42 +00001305 const ObjCIvarDecl *Ivar,
Fariborz Jahanian1bf0afb2009-01-28 01:05:23 +00001306 unsigned long int offset);
Daniel Dunbar6bff2512009-08-03 17:06:42 +00001307
Fariborz Jahanianda320092009-01-29 19:24:30 +00001308 /// GetOrEmitProtocol - Get the protocol object for the given
1309 /// declaration, emitting it if necessary. The return value has type
1310 /// ProtocolPtrTy.
1311 virtual llvm::Constant *GetOrEmitProtocol(const ObjCProtocolDecl *PD);
Daniel Dunbar6bff2512009-08-03 17:06:42 +00001312
Fariborz Jahanianda320092009-01-29 19:24:30 +00001313 /// GetOrEmitProtocolRef - Get a forward reference to the protocol
1314 /// object for the given declaration, emitting it if needed. These
1315 /// forward references will be filled in with empty bodies if no
1316 /// definition is seen. The return value has type ProtocolPtrTy.
1317 virtual llvm::Constant *GetOrEmitProtocolRef(const ObjCProtocolDecl *PD);
Daniel Dunbar6bff2512009-08-03 17:06:42 +00001318
Fariborz Jahanianda320092009-01-29 19:24:30 +00001319 /// EmitProtocolList - Generate the list of referenced
1320 /// protocols. The return value has type ProtocolListPtrTy.
Daniel Dunbar9c29bf52009-10-18 20:48:59 +00001321 llvm::Constant *EmitProtocolList(llvm::Twine Name,
Fariborz Jahanianda320092009-01-29 19:24:30 +00001322 ObjCProtocolDecl::protocol_iterator begin,
Fariborz Jahanian46551122009-02-04 00:22:57 +00001323 ObjCProtocolDecl::protocol_iterator end);
Daniel Dunbar6bff2512009-08-03 17:06:42 +00001324
Fariborz Jahanian46551122009-02-04 00:22:57 +00001325 CodeGen::RValue EmitMessageSend(CodeGen::CodeGenFunction &CGF,
John McCallef072fd2010-05-22 01:48:05 +00001326 ReturnValueSlot Return,
Fariborz Jahanian46551122009-02-04 00:22:57 +00001327 QualType ResultType,
1328 Selector Sel,
Fariborz Jahanian83a8a752009-02-04 20:42:28 +00001329 llvm::Value *Receiver,
Fariborz Jahanian46551122009-02-04 00:22:57 +00001330 QualType Arg0Ty,
1331 bool IsSuper,
1332 const CallArgList &CallArgs);
Daniel Dunbar5a7379a2009-03-01 04:40:10 +00001333
1334 /// GetClassGlobal - Return the global variable for the Objective-C
1335 /// class of the given name.
Fariborz Jahanian0f902942009-04-14 18:41:56 +00001336 llvm::GlobalVariable *GetClassGlobal(const std::string &Name);
Daniel Dunbar6bff2512009-08-03 17:06:42 +00001337
Fariborz Jahanian0e81f4b2009-02-05 20:41:40 +00001338 /// EmitClassRef - Return a Value*, of type ObjCTypes.ClassPtrTy,
Daniel Dunbar11394522009-04-18 08:51:00 +00001339 /// for the given class reference.
Daniel Dunbar6bff2512009-08-03 17:06:42 +00001340 llvm::Value *EmitClassRef(CGBuilderTy &Builder,
Daniel Dunbar11394522009-04-18 08:51:00 +00001341 const ObjCInterfaceDecl *ID);
Daniel Dunbar6bff2512009-08-03 17:06:42 +00001342
Daniel Dunbar11394522009-04-18 08:51:00 +00001343 /// EmitSuperClassRef - Return a Value*, of type ObjCTypes.ClassPtrTy,
1344 /// for the given super class reference.
Daniel Dunbar6bff2512009-08-03 17:06:42 +00001345 llvm::Value *EmitSuperClassRef(CGBuilderTy &Builder,
1346 const ObjCInterfaceDecl *ID);
1347
Fariborz Jahanian7a06aae2009-02-06 20:09:23 +00001348 /// EmitMetaClassRef - Return a Value * of the address of _class_t
1349 /// meta-data
Daniel Dunbar6bff2512009-08-03 17:06:42 +00001350 llvm::Value *EmitMetaClassRef(CGBuilderTy &Builder,
Fariborz Jahanian7a06aae2009-02-06 20:09:23 +00001351 const ObjCInterfaceDecl *ID);
1352
Fariborz Jahanianed157d32009-02-10 20:21:06 +00001353 /// ObjCIvarOffsetVariable - Returns the ivar offset variable for
1354 /// the given ivar.
1355 ///
Daniel Dunbar5e88bea2009-04-19 00:31:15 +00001356 llvm::GlobalVariable * ObjCIvarOffsetVariable(
Daniel Dunbar6bff2512009-08-03 17:06:42 +00001357 const ObjCInterfaceDecl *ID,
1358 const ObjCIvarDecl *Ivar);
1359
Fariborz Jahanian26cc89f2009-02-11 20:51:17 +00001360 /// EmitSelector - Return a Value*, of type ObjCTypes.SelectorPtrTy,
1361 /// for the given selector.
Fariborz Jahanian03b29602010-06-17 19:56:20 +00001362 llvm::Value *EmitSelector(CGBuilderTy &Builder, Selector Sel,
1363 bool lval=false);
Daniel Dunbare588b992009-03-01 04:46:24 +00001364
Daniel Dunbar8158a2f2009-04-08 04:21:03 +00001365 /// GetInterfaceEHType - Get the cached ehtype for the given Objective-C
Daniel Dunbare588b992009-03-01 04:46:24 +00001366 /// interface. The return value has type EHTypePtrTy.
John McCall5a180392010-07-24 00:37:23 +00001367 llvm::Constant *GetInterfaceEHType(const ObjCInterfaceDecl *ID,
Daniel Dunbar8158a2f2009-04-08 04:21:03 +00001368 bool ForDefinition);
Daniel Dunbar6ab187a2009-04-07 05:48:37 +00001369
Daniel Dunbar6bff2512009-08-03 17:06:42 +00001370 const char *getMetaclassSymbolPrefix() const {
Daniel Dunbar6ab187a2009-04-07 05:48:37 +00001371 return "OBJC_METACLASS_$_";
1372 }
Daniel Dunbar6bff2512009-08-03 17:06:42 +00001373
Daniel Dunbar6ab187a2009-04-07 05:48:37 +00001374 const char *getClassSymbolPrefix() const {
1375 return "OBJC_CLASS_$_";
1376 }
1377
Daniel Dunbar9f89f2b2009-05-03 12:57:56 +00001378 void GetClassSizeInfo(const ObjCImplementationDecl *OID,
Daniel Dunbarb02532a2009-04-19 23:41:48 +00001379 uint32_t &InstanceStart,
1380 uint32_t &InstanceSize);
Daniel Dunbar6bff2512009-08-03 17:06:42 +00001381
Fariborz Jahanian4523eb02009-05-12 20:06:41 +00001382 // Shamelessly stolen from Analysis/CFRefCount.cpp
Daniel Dunbar74d4b122009-05-15 22:33:15 +00001383 Selector GetNullarySelector(const char* name) const {
Fariborz Jahanian4523eb02009-05-12 20:06:41 +00001384 IdentifierInfo* II = &CGM.getContext().Idents.get(name);
1385 return CGM.getContext().Selectors.getSelector(0, &II);
1386 }
Daniel Dunbar6bff2512009-08-03 17:06:42 +00001387
Daniel Dunbar74d4b122009-05-15 22:33:15 +00001388 Selector GetUnarySelector(const char* name) const {
Fariborz Jahanian4523eb02009-05-12 20:06:41 +00001389 IdentifierInfo* II = &CGM.getContext().Idents.get(name);
1390 return CGM.getContext().Selectors.getSelector(1, &II);
1391 }
Daniel Dunbarb02532a2009-04-19 23:41:48 +00001392
Daniel Dunbar74d4b122009-05-15 22:33:15 +00001393 /// ImplementationIsNonLazy - Check whether the given category or
1394 /// class implementation is "non-lazy".
Fariborz Jahanianecfbdcb2009-05-21 01:03:45 +00001395 bool ImplementationIsNonLazy(const ObjCImplDecl *OD) const;
Daniel Dunbar74d4b122009-05-15 22:33:15 +00001396
Fariborz Jahanianee0af742009-01-21 22:04:16 +00001397public:
Fariborz Jahanian30bc5712009-01-22 23:02:58 +00001398 CGObjCNonFragileABIMac(CodeGen::CodeGenModule &cgm);
Fariborz Jahanianaa23b572009-01-23 23:53:38 +00001399 // FIXME. All stubs for now!
1400 virtual llvm::Function *ModuleInitFunction();
Daniel Dunbar6bff2512009-08-03 17:06:42 +00001401
Fariborz Jahanianaa23b572009-01-23 23:53:38 +00001402 virtual CodeGen::RValue GenerateMessageSend(CodeGen::CodeGenFunction &CGF,
John McCallef072fd2010-05-22 01:48:05 +00001403 ReturnValueSlot Return,
Fariborz Jahanianaa23b572009-01-23 23:53:38 +00001404 QualType ResultType,
1405 Selector Sel,
1406 llvm::Value *Receiver,
Fariborz Jahaniandf9ccc62009-05-05 21:36:57 +00001407 const CallArgList &CallArgs,
David Chisnallc6cd5fd2010-04-28 19:33:36 +00001408 const ObjCInterfaceDecl *Class,
Fariborz Jahaniandf9ccc62009-05-05 21:36:57 +00001409 const ObjCMethodDecl *Method);
Daniel Dunbar6bff2512009-08-03 17:06:42 +00001410
1411 virtual CodeGen::RValue
Fariborz Jahanianaa23b572009-01-23 23:53:38 +00001412 GenerateMessageSendSuper(CodeGen::CodeGenFunction &CGF,
John McCallef072fd2010-05-22 01:48:05 +00001413 ReturnValueSlot Return,
Fariborz Jahanianaa23b572009-01-23 23:53:38 +00001414 QualType ResultType,
1415 Selector Sel,
1416 const ObjCInterfaceDecl *Class,
Fariborz Jahanian7ce77922009-02-28 20:07:56 +00001417 bool isCategoryImpl,
Fariborz Jahanianaa23b572009-01-23 23:53:38 +00001418 llvm::Value *Receiver,
1419 bool IsClassMessage,
Daniel Dunbard6c93d72009-09-17 04:01:22 +00001420 const CallArgList &CallArgs,
1421 const ObjCMethodDecl *Method);
Daniel Dunbar6bff2512009-08-03 17:06:42 +00001422
Fariborz Jahanianaa23b572009-01-23 23:53:38 +00001423 virtual llvm::Value *GetClass(CGBuilderTy &Builder,
Fariborz Jahanian0e81f4b2009-02-05 20:41:40 +00001424 const ObjCInterfaceDecl *ID);
Daniel Dunbar6bff2512009-08-03 17:06:42 +00001425
Fariborz Jahanian03b29602010-06-17 19:56:20 +00001426 virtual llvm::Value *GetSelector(CGBuilderTy &Builder, Selector Sel,
1427 bool lvalue = false)
1428 { return EmitSelector(Builder, Sel, lvalue); }
Fariborz Jahaniandf9ccc62009-05-05 21:36:57 +00001429
1430 /// The NeXT/Apple runtimes do not support typed selectors; just emit an
1431 /// untyped one.
1432 virtual llvm::Value *GetSelector(CGBuilderTy &Builder,
1433 const ObjCMethodDecl *Method)
1434 { return EmitSelector(Builder, Method->getSelector()); }
Daniel Dunbar6bff2512009-08-03 17:06:42 +00001435
Fariborz Jahanianeb062d92009-01-26 18:32:24 +00001436 virtual void GenerateCategory(const ObjCCategoryImplDecl *CMD);
Daniel Dunbar6bff2512009-08-03 17:06:42 +00001437
Fariborz Jahanianaa23b572009-01-23 23:53:38 +00001438 virtual void GenerateClass(const ObjCImplementationDecl *ClassDecl);
Fariborz Jahanianaa23b572009-01-23 23:53:38 +00001439 virtual llvm::Value *GenerateProtocolRef(CGBuilderTy &Builder,
Fariborz Jahanian8cfd3972009-01-30 18:58:59 +00001440 const ObjCProtocolDecl *PD);
Daniel Dunbar6bff2512009-08-03 17:06:42 +00001441
John McCall5a180392010-07-24 00:37:23 +00001442 virtual llvm::Constant *GetEHType(QualType T);
1443
Daniel Dunbar6bff2512009-08-03 17:06:42 +00001444 virtual llvm::Constant *GetPropertyGetFunction() {
Chris Lattner72db6c32009-04-22 02:44:54 +00001445 return ObjCTypes.getGetPropertyFn();
Fariborz Jahanianf63aa3f2009-02-10 19:02:04 +00001446 }
Daniel Dunbar6bff2512009-08-03 17:06:42 +00001447 virtual llvm::Constant *GetPropertySetFunction() {
1448 return ObjCTypes.getSetPropertyFn();
Fariborz Jahanianf63aa3f2009-02-10 19:02:04 +00001449 }
Fariborz Jahanian6cc59062010-04-12 18:18:10 +00001450
1451 virtual llvm::Constant *GetCopyStructFunction() {
1452 return ObjCTypes.getCopyStructFn();
1453 }
1454
Chris Lattner74391b42009-03-22 21:03:39 +00001455 virtual llvm::Constant *EnumerationMutationFunction() {
Chris Lattner72db6c32009-04-22 02:44:54 +00001456 return ObjCTypes.getEnumerationMutationFn();
Daniel Dunbar28ed0842009-02-16 18:48:45 +00001457 }
Daniel Dunbar6bff2512009-08-03 17:06:42 +00001458
John McCallf1549f62010-07-06 01:34:17 +00001459 virtual void EmitTryStmt(CodeGen::CodeGenFunction &CGF,
1460 const ObjCAtTryStmt &S);
1461 virtual void EmitSynchronizedStmt(CodeGen::CodeGenFunction &CGF,
1462 const ObjCAtSynchronizedStmt &S);
Fariborz Jahanianaa23b572009-01-23 23:53:38 +00001463 virtual void EmitThrowStmt(CodeGen::CodeGenFunction &CGF,
Anders Carlssonf57c5b22009-02-16 22:59:18 +00001464 const ObjCAtThrowStmt &S);
Fariborz Jahanianaa23b572009-01-23 23:53:38 +00001465 virtual llvm::Value * EmitObjCWeakRead(CodeGen::CodeGenFunction &CGF,
Fariborz Jahanian6948aea2009-02-16 22:52:32 +00001466 llvm::Value *AddrWeakObj);
Fariborz Jahanianaa23b572009-01-23 23:53:38 +00001467 virtual void EmitObjCWeakAssign(CodeGen::CodeGenFunction &CGF,
Fariborz Jahanian6948aea2009-02-16 22:52:32 +00001468 llvm::Value *src, llvm::Value *dst);
Fariborz Jahanianaa23b572009-01-23 23:53:38 +00001469 virtual void EmitObjCGlobalAssign(CodeGen::CodeGenFunction &CGF,
Fariborz Jahanian021a7a62010-07-20 20:30:03 +00001470 llvm::Value *src, llvm::Value *dest,
1471 bool threadlocal = false);
Fariborz Jahanianaa23b572009-01-23 23:53:38 +00001472 virtual void EmitObjCIvarAssign(CodeGen::CodeGenFunction &CGF,
Fariborz Jahanian6c7a1f32009-09-24 22:25:38 +00001473 llvm::Value *src, llvm::Value *dest,
1474 llvm::Value *ivarOffset);
Fariborz Jahanianaa23b572009-01-23 23:53:38 +00001475 virtual void EmitObjCStrongCastAssign(CodeGen::CodeGenFunction &CGF,
Fariborz Jahanian6948aea2009-02-16 22:52:32 +00001476 llvm::Value *src, llvm::Value *dest);
Fariborz Jahanian082b02e2009-07-08 01:18:33 +00001477 virtual void EmitGCMemmoveCollectable(CodeGen::CodeGenFunction &CGF,
1478 llvm::Value *dest, llvm::Value *src,
Fariborz Jahanian55bcace2010-06-15 22:44:06 +00001479 llvm::Value *size);
Fariborz Jahanian598d3f62009-02-03 19:03:09 +00001480 virtual LValue EmitObjCValueForIvar(CodeGen::CodeGenFunction &CGF,
1481 QualType ObjectTy,
1482 llvm::Value *BaseValue,
1483 const ObjCIvarDecl *Ivar,
Fariborz Jahanian598d3f62009-02-03 19:03:09 +00001484 unsigned CVRQualifiers);
Fariborz Jahanianf63aa3f2009-02-10 19:02:04 +00001485 virtual llvm::Value *EmitIvarOffset(CodeGen::CodeGenFunction &CGF,
Daniel Dunbar2a031922009-04-22 05:08:15 +00001486 const ObjCInterfaceDecl *Interface,
Fariborz Jahanianf63aa3f2009-02-10 19:02:04 +00001487 const ObjCIvarDecl *Ivar);
Fariborz Jahanianee0af742009-01-21 22:04:16 +00001488};
Daniel Dunbar6bff2512009-08-03 17:06:42 +00001489
Daniel Dunbarc17a4d32008-08-11 02:45:11 +00001490} // end anonymous namespace
Daniel Dunbarbbce49b2008-08-12 00:12:39 +00001491
1492/* *** Helper Functions *** */
1493
1494/// getConstantGEP() - Help routine to construct simple GEPs.
Owen Andersona1cf15f2009-07-14 23:10:40 +00001495static llvm::Constant *getConstantGEP(llvm::LLVMContext &VMContext,
Daniel Dunbar6bff2512009-08-03 17:06:42 +00001496 llvm::Constant *C,
Daniel Dunbarbbce49b2008-08-12 00:12:39 +00001497 unsigned idx0,
1498 unsigned idx1) {
1499 llvm::Value *Idxs[] = {
Owen Anderson0032b272009-08-13 21:57:51 +00001500 llvm::ConstantInt::get(llvm::Type::getInt32Ty(VMContext), idx0),
1501 llvm::ConstantInt::get(llvm::Type::getInt32Ty(VMContext), idx1)
Daniel Dunbarbbce49b2008-08-12 00:12:39 +00001502 };
Owen Anderson3c4972d2009-07-29 18:54:39 +00001503 return llvm::ConstantExpr::getGetElementPtr(C, Idxs, 2);
Daniel Dunbarbbce49b2008-08-12 00:12:39 +00001504}
1505
Daniel Dunbar8158a2f2009-04-08 04:21:03 +00001506/// hasObjCExceptionAttribute - Return true if this class or any super
1507/// class has the __objc_exception__ attribute.
Daniel Dunbar6bff2512009-08-03 17:06:42 +00001508static bool hasObjCExceptionAttribute(ASTContext &Context,
Douglas Gregor68584ed2009-06-18 16:11:24 +00001509 const ObjCInterfaceDecl *OID) {
Argyrios Kyrtzidis40b598e2009-06-30 02:34:44 +00001510 if (OID->hasAttr<ObjCExceptionAttr>())
Daniel Dunbar8158a2f2009-04-08 04:21:03 +00001511 return true;
1512 if (const ObjCInterfaceDecl *Super = OID->getSuperClass())
Douglas Gregor68584ed2009-06-18 16:11:24 +00001513 return hasObjCExceptionAttribute(Context, Super);
Daniel Dunbar8158a2f2009-04-08 04:21:03 +00001514 return false;
1515}
1516
Daniel Dunbarbbce49b2008-08-12 00:12:39 +00001517/* *** CGObjCMac Public Interface *** */
Daniel Dunbar6bff2512009-08-03 17:06:42 +00001518
Fariborz Jahanianee0af742009-01-21 22:04:16 +00001519CGObjCMac::CGObjCMac(CodeGen::CodeGenModule &cgm) : CGObjCCommonMac(cgm),
Mike Stump1eb44332009-09-09 15:08:12 +00001520 ObjCTypes(cgm) {
Fariborz Jahanianee0af742009-01-21 22:04:16 +00001521 ObjCABI = 1;
Daniel Dunbar6bff2512009-08-03 17:06:42 +00001522 EmitImageInfo();
Daniel Dunbarc17a4d32008-08-11 02:45:11 +00001523}
1524
Daniel Dunbarddb2a3d2008-08-16 00:25:02 +00001525/// GetClass - Return a reference to the class for the given interface
1526/// decl.
Daniel Dunbar45d196b2008-11-01 01:53:16 +00001527llvm::Value *CGObjCMac::GetClass(CGBuilderTy &Builder,
Daniel Dunbar27f9d772008-08-21 04:36:09 +00001528 const ObjCInterfaceDecl *ID) {
1529 return EmitClassRef(Builder, ID);
Daniel Dunbarc17a4d32008-08-11 02:45:11 +00001530}
1531
1532/// GetSelector - Return the pointer to the unique'd string for this selector.
Fariborz Jahanian03b29602010-06-17 19:56:20 +00001533llvm::Value *CGObjCMac::GetSelector(CGBuilderTy &Builder, Selector Sel,
1534 bool lval) {
1535 return EmitSelector(Builder, Sel, lval);
Daniel Dunbarc17a4d32008-08-11 02:45:11 +00001536}
Fariborz Jahaniandf9ccc62009-05-05 21:36:57 +00001537llvm::Value *CGObjCMac::GetSelector(CGBuilderTy &Builder, const ObjCMethodDecl
Daniel Dunbar6bff2512009-08-03 17:06:42 +00001538 *Method) {
Fariborz Jahaniandf9ccc62009-05-05 21:36:57 +00001539 return EmitSelector(Builder, Method->getSelector());
1540}
Daniel Dunbarc17a4d32008-08-11 02:45:11 +00001541
John McCall5a180392010-07-24 00:37:23 +00001542llvm::Constant *CGObjCMac::GetEHType(QualType T) {
1543 llvm_unreachable("asking for catch type for ObjC type in fragile runtime");
1544 return 0;
1545}
1546
Daniel Dunbarbbce49b2008-08-12 00:12:39 +00001547/// Generate a constant CFString object.
Daniel Dunbar6bff2512009-08-03 17:06:42 +00001548/*
1549 struct __builtin_CFString {
1550 const int *isa; // point to __CFConstantStringClassReference
1551 int flags;
1552 const char *str;
1553 long length;
1554 };
Daniel Dunbarbbce49b2008-08-12 00:12:39 +00001555*/
1556
Fariborz Jahanian33e982b2010-04-22 20:26:39 +00001557/// or Generate a constant NSString object.
1558/*
1559 struct __builtin_NSString {
1560 const int *isa; // point to __NSConstantStringClassReference
1561 const char *str;
1562 unsigned int length;
1563 };
1564*/
1565
Fariborz Jahanianaa23b572009-01-23 23:53:38 +00001566llvm::Constant *CGObjCCommonMac::GenerateConstantString(
David Chisnall0d13f6f2010-01-23 02:40:42 +00001567 const StringLiteral *SL) {
Fariborz Jahanian33e982b2010-04-22 20:26:39 +00001568 return (CGM.getLangOptions().NoConstantCFStrings == 0 ?
1569 CGM.GetAddrOfConstantCFString(SL) :
1570 CGM.GetAddrOfConstantNSString(SL));
Daniel Dunbarc17a4d32008-08-11 02:45:11 +00001571}
1572
1573/// Generates a message send where the super is the receiver. This is
1574/// a message send to self with special delivery semantics indicating
1575/// which class's method should be called.
Daniel Dunbar8f2926b2008-08-23 03:46:30 +00001576CodeGen::RValue
1577CGObjCMac::GenerateMessageSendSuper(CodeGen::CodeGenFunction &CGF,
John McCallef072fd2010-05-22 01:48:05 +00001578 ReturnValueSlot Return,
Daniel Dunbar7f8ea5c2008-08-30 05:35:15 +00001579 QualType ResultType,
1580 Selector Sel,
Daniel Dunbarf56f1912008-08-25 08:19:24 +00001581 const ObjCInterfaceDecl *Class,
Fariborz Jahanian7ce77922009-02-28 20:07:56 +00001582 bool isCategoryImpl,
Daniel Dunbarf56f1912008-08-25 08:19:24 +00001583 llvm::Value *Receiver,
Daniel Dunbar19cd87e2008-08-30 03:02:31 +00001584 bool IsClassMessage,
Daniel Dunbard6c93d72009-09-17 04:01:22 +00001585 const CodeGen::CallArgList &CallArgs,
1586 const ObjCMethodDecl *Method) {
Daniel Dunbare8b470d2008-08-23 04:28:29 +00001587 // Create and init a super structure; this is a (receiver, class)
1588 // pair we will pass to objc_msgSendSuper.
Daniel Dunbar6bff2512009-08-03 17:06:42 +00001589 llvm::Value *ObjCSuper =
Daniel Dunbare8b470d2008-08-23 04:28:29 +00001590 CGF.Builder.CreateAlloca(ObjCTypes.SuperTy, 0, "objc_super");
Daniel Dunbar6bff2512009-08-03 17:06:42 +00001591 llvm::Value *ReceiverAsObject =
Daniel Dunbare8b470d2008-08-23 04:28:29 +00001592 CGF.Builder.CreateBitCast(Receiver, ObjCTypes.ObjectPtrTy);
Daniel Dunbar6bff2512009-08-03 17:06:42 +00001593 CGF.Builder.CreateStore(ReceiverAsObject,
Daniel Dunbare8b470d2008-08-23 04:28:29 +00001594 CGF.Builder.CreateStructGEP(ObjCSuper, 0));
Daniel Dunbare8b470d2008-08-23 04:28:29 +00001595
Daniel Dunbarf56f1912008-08-25 08:19:24 +00001596 // If this is a class message the metaclass is passed as the target.
1597 llvm::Value *Target;
1598 if (IsClassMessage) {
Fariborz Jahanian7ce77922009-02-28 20:07:56 +00001599 if (isCategoryImpl) {
1600 // Message sent to 'super' in a class method defined in a category
1601 // implementation requires an odd treatment.
1602 // If we are in a class method, we must retrieve the
1603 // _metaclass_ for the current class, pointed at by
1604 // the class's "isa" pointer. The following assumes that
1605 // isa" is the first ivar in a class (which it must be).
1606 Target = EmitClassRef(CGF.Builder, Class->getSuperClass());
1607 Target = CGF.Builder.CreateStructGEP(Target, 0);
1608 Target = CGF.Builder.CreateLoad(Target);
Mike Stumpb3589f42009-07-30 22:28:39 +00001609 } else {
Fariborz Jahanian7ce77922009-02-28 20:07:56 +00001610 llvm::Value *MetaClassPtr = EmitMetaClassRef(Class);
1611 llvm::Value *SuperPtr = CGF.Builder.CreateStructGEP(MetaClassPtr, 1);
1612 llvm::Value *Super = CGF.Builder.CreateLoad(SuperPtr);
1613 Target = Super;
Daniel Dunbar6bff2512009-08-03 17:06:42 +00001614 }
Fariborz Jahanian182f2682009-11-14 02:18:31 +00001615 }
1616 else if (isCategoryImpl)
1617 Target = EmitClassRef(CGF.Builder, Class->getSuperClass());
1618 else {
Fariborz Jahanianb0069ee2009-11-12 20:14:24 +00001619 llvm::Value *ClassPtr = EmitSuperClassRef(Class);
1620 ClassPtr = CGF.Builder.CreateStructGEP(ClassPtr, 1);
1621 Target = CGF.Builder.CreateLoad(ClassPtr);
Daniel Dunbarf56f1912008-08-25 08:19:24 +00001622 }
Mike Stumpf5408fe2009-05-16 07:57:57 +00001623 // FIXME: We shouldn't need to do this cast, rectify the ASTContext and
1624 // ObjCTypes types.
Daniel Dunbar6bff2512009-08-03 17:06:42 +00001625 const llvm::Type *ClassTy =
Daniel Dunbar19cd87e2008-08-30 03:02:31 +00001626 CGM.getTypes().ConvertType(CGF.getContext().getObjCClassType());
Daniel Dunbar0c0e7a62008-10-29 22:36:39 +00001627 Target = CGF.Builder.CreateBitCast(Target, ClassTy);
Daniel Dunbar6bff2512009-08-03 17:06:42 +00001628 CGF.Builder.CreateStore(Target,
Daniel Dunbarf56f1912008-08-25 08:19:24 +00001629 CGF.Builder.CreateStructGEP(ObjCSuper, 1));
John McCallef072fd2010-05-22 01:48:05 +00001630 return EmitLegacyMessageSend(CGF, Return, ResultType,
Fariborz Jahaniand0f8a8d2009-05-11 19:25:47 +00001631 EmitSelector(CGF.Builder, Sel),
1632 ObjCSuper, ObjCTypes.SuperPtrCTy,
Daniel Dunbard6c93d72009-09-17 04:01:22 +00001633 true, CallArgs, Method, ObjCTypes);
Daniel Dunbarc17a4d32008-08-11 02:45:11 +00001634}
Daniel Dunbar6bff2512009-08-03 17:06:42 +00001635
1636/// Generate code for a message send expression.
Daniel Dunbar8f2926b2008-08-23 03:46:30 +00001637CodeGen::RValue CGObjCMac::GenerateMessageSend(CodeGen::CodeGenFunction &CGF,
John McCallef072fd2010-05-22 01:48:05 +00001638 ReturnValueSlot Return,
Daniel Dunbar7f8ea5c2008-08-30 05:35:15 +00001639 QualType ResultType,
1640 Selector Sel,
Daniel Dunbarf56f1912008-08-25 08:19:24 +00001641 llvm::Value *Receiver,
Fariborz Jahaniandf9ccc62009-05-05 21:36:57 +00001642 const CallArgList &CallArgs,
David Chisnallc6cd5fd2010-04-28 19:33:36 +00001643 const ObjCInterfaceDecl *Class,
Fariborz Jahaniandf9ccc62009-05-05 21:36:57 +00001644 const ObjCMethodDecl *Method) {
John McCallef072fd2010-05-22 01:48:05 +00001645 return EmitLegacyMessageSend(CGF, Return, ResultType,
Fariborz Jahaniand0f8a8d2009-05-11 19:25:47 +00001646 EmitSelector(CGF.Builder, Sel),
1647 Receiver, CGF.getContext().getObjCIdType(),
Daniel Dunbard6c93d72009-09-17 04:01:22 +00001648 false, CallArgs, Method, ObjCTypes);
Daniel Dunbar14c80b72008-08-23 09:25:55 +00001649}
1650
Daniel Dunbard6c93d72009-09-17 04:01:22 +00001651CodeGen::RValue
1652CGObjCCommonMac::EmitLegacyMessageSend(CodeGen::CodeGenFunction &CGF,
John McCallef072fd2010-05-22 01:48:05 +00001653 ReturnValueSlot Return,
Daniel Dunbard6c93d72009-09-17 04:01:22 +00001654 QualType ResultType,
1655 llvm::Value *Sel,
1656 llvm::Value *Arg0,
1657 QualType Arg0Ty,
1658 bool IsSuper,
1659 const CallArgList &CallArgs,
1660 const ObjCMethodDecl *Method,
1661 const ObjCCommonTypesHelper &ObjCTypes) {
Daniel Dunbar19cd87e2008-08-30 03:02:31 +00001662 CallArgList ActualArgs;
Fariborz Jahaniand019d962009-04-24 21:07:43 +00001663 if (!IsSuper)
1664 Arg0 = CGF.Builder.CreateBitCast(Arg0, ObjCTypes.ObjectPtrTy, "tmp");
Daniel Dunbar46f45b92008-09-09 01:06:48 +00001665 ActualArgs.push_back(std::make_pair(RValue::get(Arg0), Arg0Ty));
Fariborz Jahaniand0f8a8d2009-05-11 19:25:47 +00001666 ActualArgs.push_back(std::make_pair(RValue::get(Sel),
Daniel Dunbar19cd87e2008-08-30 03:02:31 +00001667 CGF.getContext().getObjCSelType()));
1668 ActualArgs.insert(ActualArgs.end(), CallArgs.begin(), CallArgs.end());
Daniel Dunbar6bff2512009-08-03 17:06:42 +00001669
Daniel Dunbar541b63b2009-02-02 23:23:47 +00001670 CodeGenTypes &Types = CGM.getTypes();
John McCall04a67a62010-02-05 21:31:56 +00001671 const CGFunctionInfo &FnInfo = Types.getFunctionInfo(ResultType, ActualArgs,
Rafael Espindola264ba482010-03-30 20:24:48 +00001672 FunctionType::ExtInfo());
Daniel Dunbar67939662009-09-17 04:01:40 +00001673 const llvm::FunctionType *FTy =
1674 Types.GetFunctionType(FnInfo, Method ? Method->isVariadic() : false);
Daniel Dunbar6bff2512009-08-03 17:06:42 +00001675
Anders Carlsson7e70fb22010-06-21 20:59:55 +00001676 if (Method)
1677 assert(CGM.getContext().getCanonicalType(Method->getResultType()) ==
1678 CGM.getContext().getCanonicalType(ResultType) &&
1679 "Result type mismatch!");
1680
Fariborz Jahaniand0f8a8d2009-05-11 19:25:47 +00001681 llvm::Constant *Fn = NULL;
Daniel Dunbardacf9dd2010-07-14 23:39:36 +00001682 if (CGM.ReturnTypeUsesSRet(FnInfo)) {
Fariborz Jahaniand0f8a8d2009-05-11 19:25:47 +00001683 Fn = (ObjCABI == 2) ? ObjCTypes.getSendStretFn2(IsSuper)
Daniel Dunbar6bff2512009-08-03 17:06:42 +00001684 : ObjCTypes.getSendStretFn(IsSuper);
Daniel Dunbardacf9dd2010-07-14 23:39:36 +00001685 } else if (CGM.ReturnTypeUsesFPRet(ResultType)) {
1686 Fn = (ObjCABI == 2) ? ObjCTypes.getSendFpretFn2(IsSuper)
1687 : ObjCTypes.getSendFpretFn(IsSuper);
Daniel Dunbar5669e572008-10-17 03:24:53 +00001688 } else {
Fariborz Jahaniand0f8a8d2009-05-11 19:25:47 +00001689 Fn = (ObjCABI == 2) ? ObjCTypes.getSendFn2(IsSuper)
Daniel Dunbar6bff2512009-08-03 17:06:42 +00001690 : ObjCTypes.getSendFn(IsSuper);
Daniel Dunbar5669e572008-10-17 03:24:53 +00001691 }
Daniel Dunbardacf9dd2010-07-14 23:39:36 +00001692 Fn = llvm::ConstantExpr::getBitCast(Fn, llvm::PointerType::getUnqual(FTy));
John McCallef072fd2010-05-22 01:48:05 +00001693 return CGF.EmitCall(FnInfo, Fn, Return, ActualArgs);
Daniel Dunbarc17a4d32008-08-11 02:45:11 +00001694}
1695
Fariborz Jahanian93ce50d2010-08-04 23:55:24 +00001696static Qualifiers::GC GetGCAttrTypeForType(ASTContext &Ctx, QualType FQT) {
1697 if (FQT.isObjCGCStrong())
1698 return Qualifiers::Strong;
1699
1700 if (FQT.isObjCGCWeak())
1701 return Qualifiers::Weak;
1702
1703 if (FQT->isObjCObjectPointerType() || FQT->isBlockPointerType())
1704 return Qualifiers::Strong;
1705
1706 if (const PointerType *PT = FQT->getAs<PointerType>())
1707 return GetGCAttrTypeForType(Ctx, PT->getPointeeType());
1708
1709 return Qualifiers::GCNone;
1710}
1711
Fariborz Jahanian89ecd412010-08-04 16:57:49 +00001712llvm::Constant *CGObjCCommonMac::GCBlockLayout(CodeGen::CodeGenFunction &CGF,
Fariborz Jahanian44034db2010-08-04 18:44:59 +00001713 const llvm::SmallVectorImpl<const BlockDeclRefExpr *> &DeclRefs) {
1714 llvm::Constant *NullPtr =
1715 llvm::Constant::getNullValue(llvm::Type::getInt8PtrTy(VMContext));
Fariborz Jahanian5af12352010-08-05 15:52:12 +00001716 if ((CGM.getLangOptions().getGCMode() == LangOptions::NonGC) ||
1717 DeclRefs.empty())
Fariborz Jahanian44034db2010-08-04 18:44:59 +00001718 return NullPtr;
Fariborz Jahaniana1f024c2010-08-06 16:28:55 +00001719 bool hasUnion = false;
Fariborz Jahanian93ce50d2010-08-04 23:55:24 +00001720 SkipIvars.clear();
1721 IvarsInfo.clear();
1722 unsigned WordSizeInBits = CGM.getContext().Target.getPointerWidth(0);
1723 unsigned ByteSizeInBits = CGM.getContext().Target.getCharWidth();
1724
1725 for (size_t i = 0; i < DeclRefs.size(); ++i) {
1726 const BlockDeclRefExpr *BDRE = DeclRefs[i];
1727 const ValueDecl *VD = BDRE->getDecl();
1728 CharUnits Offset = CGF.BlockDecls[VD];
1729 uint64_t FieldOffset = Offset.getQuantity();
1730 QualType Ty = VD->getType();
1731 assert(!Ty->isArrayType() &&
1732 "Array block variable should have been caught");
Fariborz Jahaniana1f024c2010-08-06 16:28:55 +00001733 if ((Ty->isRecordType() || Ty->isUnionType()) && !BDRE->isByRef()) {
Fariborz Jahaniane1a48982010-08-05 21:00:25 +00001734 BuildAggrIvarRecordLayout(Ty->getAs<RecordType>(),
1735 FieldOffset,
1736 true,
Fariborz Jahaniana1f024c2010-08-06 16:28:55 +00001737 hasUnion);
Fariborz Jahaniane1a48982010-08-05 21:00:25 +00001738 continue;
1739 }
Fariborz Jahaniana1f024c2010-08-06 16:28:55 +00001740
Fariborz Jahanian93ce50d2010-08-04 23:55:24 +00001741 Qualifiers::GC GCAttr = GetGCAttrTypeForType(CGM.getContext(), Ty);
1742 unsigned FieldSize = CGM.getContext().getTypeSize(Ty);
1743 // __block variables are passed by their descriptior address. So, size
1744 // must reflect this.
1745 if (BDRE->isByRef())
1746 FieldSize = WordSizeInBits;
1747 if (GCAttr == Qualifiers::Strong || BDRE->isByRef())
1748 IvarsInfo.push_back(GC_IVAR(FieldOffset,
1749 FieldSize / WordSizeInBits));
1750 else if (GCAttr == Qualifiers::GCNone || GCAttr == Qualifiers::Weak)
1751 SkipIvars.push_back(GC_IVAR(FieldOffset,
1752 FieldSize / ByteSizeInBits));
1753 }
1754
1755 if (IvarsInfo.empty())
1756 return NullPtr;
Fariborz Jahaniana1f024c2010-08-06 16:28:55 +00001757 // Sort on byte position in case we encounterred a union nested in
1758 // block variable type's aggregate type.
1759 if (hasUnion && !IvarsInfo.empty())
1760 std::sort(IvarsInfo.begin(), IvarsInfo.end());
1761 if (hasUnion && !SkipIvars.empty())
1762 std::sort(SkipIvars.begin(), SkipIvars.end());
Fariborz Jahanian93ce50d2010-08-04 23:55:24 +00001763
1764 std::string BitMap;
Fariborz Jahanianb8fd2eb2010-08-05 00:19:48 +00001765 llvm::Constant *C = BuildIvarLayoutBitmap(BitMap);
Fariborz Jahanian93ce50d2010-08-04 23:55:24 +00001766 if (CGM.getLangOptions().ObjCGCBitmapPrint) {
1767 printf("\n block variable layout for block: ");
1768 const unsigned char *s = (unsigned char*)BitMap.c_str();
1769 for (unsigned i = 0; i < BitMap.size(); i++)
1770 if (!(s[i] & 0xf0))
1771 printf("0x0%x%s", s[i], s[i] != 0 ? ", " : "");
1772 else
1773 printf("0x%x%s", s[i], s[i] != 0 ? ", " : "");
1774 printf("\n");
1775 }
1776
1777 return C;
Fariborz Jahanian89ecd412010-08-04 16:57:49 +00001778}
1779
Daniel Dunbar6bff2512009-08-03 17:06:42 +00001780llvm::Value *CGObjCMac::GenerateProtocolRef(CGBuilderTy &Builder,
Daniel Dunbaraf2f62c2008-08-13 00:59:25 +00001781 const ObjCProtocolDecl *PD) {
Daniel Dunbarc67876d2008-09-04 04:33:15 +00001782 // FIXME: I don't understand why gcc generates this, or where it is
Mike Stumpf5408fe2009-05-16 07:57:57 +00001783 // resolved. Investigate. Its also wasteful to look this up over and over.
Daniel Dunbarc67876d2008-09-04 04:33:15 +00001784 LazySymbols.insert(&CGM.getContext().Idents.get("Protocol"));
1785
Owen Anderson3c4972d2009-07-29 18:54:39 +00001786 return llvm::ConstantExpr::getBitCast(GetProtocolRef(PD),
Daniel Dunbar6efc0c52008-08-13 03:21:16 +00001787 ObjCTypes.ExternalProtocolPtrTy);
Daniel Dunbarc17a4d32008-08-11 02:45:11 +00001788}
1789
Fariborz Jahanianda320092009-01-29 19:24:30 +00001790void CGObjCCommonMac::GenerateProtocol(const ObjCProtocolDecl *PD) {
Mike Stumpf5408fe2009-05-16 07:57:57 +00001791 // FIXME: We shouldn't need this, the protocol decl should contain enough
1792 // information to tell us whether this was a declaration or a definition.
Daniel Dunbar0c0e7a62008-10-29 22:36:39 +00001793 DefinedProtocols.insert(PD->getIdentifier());
1794
1795 // If we have generated a forward reference to this protocol, emit
1796 // it now. Otherwise do nothing, the protocol objects are lazily
1797 // emitted.
Daniel Dunbar6bff2512009-08-03 17:06:42 +00001798 if (Protocols.count(PD->getIdentifier()))
Daniel Dunbar0c0e7a62008-10-29 22:36:39 +00001799 GetOrEmitProtocol(PD);
1800}
1801
Fariborz Jahanianda320092009-01-29 19:24:30 +00001802llvm::Constant *CGObjCCommonMac::GetProtocolRef(const ObjCProtocolDecl *PD) {
Daniel Dunbar0c0e7a62008-10-29 22:36:39 +00001803 if (DefinedProtocols.count(PD->getIdentifier()))
1804 return GetOrEmitProtocol(PD);
1805 return GetOrEmitProtocolRef(PD);
1806}
1807
Daniel Dunbar6efc0c52008-08-13 03:21:16 +00001808/*
Daniel Dunbar6bff2512009-08-03 17:06:42 +00001809// APPLE LOCAL radar 4585769 - Objective-C 1.0 extensions
1810struct _objc_protocol {
1811struct _objc_protocol_extension *isa;
1812char *protocol_name;
1813struct _objc_protocol_list *protocol_list;
1814struct _objc__method_prototype_list *instance_methods;
1815struct _objc__method_prototype_list *class_methods
1816};
Daniel Dunbar6efc0c52008-08-13 03:21:16 +00001817
Daniel Dunbar6bff2512009-08-03 17:06:42 +00001818See EmitProtocolExtension().
Daniel Dunbar6efc0c52008-08-13 03:21:16 +00001819*/
Daniel Dunbar0c0e7a62008-10-29 22:36:39 +00001820llvm::Constant *CGObjCMac::GetOrEmitProtocol(const ObjCProtocolDecl *PD) {
1821 llvm::GlobalVariable *&Entry = Protocols[PD->getIdentifier()];
1822
1823 // Early exit if a defining object has already been generated.
1824 if (Entry && Entry->hasInitializer())
1825 return Entry;
1826
Daniel Dunbar242d4dc2008-08-25 06:02:07 +00001827 // FIXME: I don't understand why gcc generates this, or where it is
Mike Stumpf5408fe2009-05-16 07:57:57 +00001828 // resolved. Investigate. Its also wasteful to look this up over and over.
Daniel Dunbar242d4dc2008-08-25 06:02:07 +00001829 LazySymbols.insert(&CGM.getContext().Idents.get("Protocol"));
1830
Daniel Dunbarae226fa2008-08-27 02:31:56 +00001831 // Construct method lists.
1832 std::vector<llvm::Constant*> InstanceMethods, ClassMethods;
1833 std::vector<llvm::Constant*> OptInstanceMethods, OptClassMethods;
Daniel Dunbar6bff2512009-08-03 17:06:42 +00001834 for (ObjCProtocolDecl::instmeth_iterator
Argyrios Kyrtzidis17945a02009-06-30 02:36:12 +00001835 i = PD->instmeth_begin(), e = PD->instmeth_end(); i != e; ++i) {
Daniel Dunbarae226fa2008-08-27 02:31:56 +00001836 ObjCMethodDecl *MD = *i;
1837 llvm::Constant *C = GetMethodDescriptionConstant(MD);
1838 if (MD->getImplementationControl() == ObjCMethodDecl::Optional) {
1839 OptInstanceMethods.push_back(C);
1840 } else {
1841 InstanceMethods.push_back(C);
Daniel Dunbar6bff2512009-08-03 17:06:42 +00001842 }
Daniel Dunbarae226fa2008-08-27 02:31:56 +00001843 }
1844
Daniel Dunbar6bff2512009-08-03 17:06:42 +00001845 for (ObjCProtocolDecl::classmeth_iterator
Argyrios Kyrtzidis17945a02009-06-30 02:36:12 +00001846 i = PD->classmeth_begin(), e = PD->classmeth_end(); i != e; ++i) {
Daniel Dunbarae226fa2008-08-27 02:31:56 +00001847 ObjCMethodDecl *MD = *i;
1848 llvm::Constant *C = GetMethodDescriptionConstant(MD);
1849 if (MD->getImplementationControl() == ObjCMethodDecl::Optional) {
1850 OptClassMethods.push_back(C);
1851 } else {
1852 ClassMethods.push_back(C);
Daniel Dunbar6bff2512009-08-03 17:06:42 +00001853 }
Daniel Dunbarae226fa2008-08-27 02:31:56 +00001854 }
1855
Daniel Dunbar6efc0c52008-08-13 03:21:16 +00001856 std::vector<llvm::Constant*> Values(5);
Daniel Dunbarae226fa2008-08-27 02:31:56 +00001857 Values[0] = EmitProtocolExtension(PD, OptInstanceMethods, OptClassMethods);
Daniel Dunbar6efc0c52008-08-13 03:21:16 +00001858 Values[1] = GetClassName(PD->getIdentifier());
Daniel Dunbar6bff2512009-08-03 17:06:42 +00001859 Values[2] =
Daniel Dunbar9c29bf52009-10-18 20:48:59 +00001860 EmitProtocolList("\01L_OBJC_PROTOCOL_REFS_" + PD->getName(),
Daniel Dunbardbc933702008-08-21 21:57:41 +00001861 PD->protocol_begin(),
1862 PD->protocol_end());
Daniel Dunbar6bff2512009-08-03 17:06:42 +00001863 Values[3] =
Daniel Dunbar9c29bf52009-10-18 20:48:59 +00001864 EmitMethodDescList("\01L_OBJC_PROTOCOL_INSTANCE_METHODS_" + PD->getName(),
Daniel Dunbarae226fa2008-08-27 02:31:56 +00001865 "__OBJC,__cat_inst_meth,regular,no_dead_strip",
1866 InstanceMethods);
Daniel Dunbar6bff2512009-08-03 17:06:42 +00001867 Values[4] =
Daniel Dunbar9c29bf52009-10-18 20:48:59 +00001868 EmitMethodDescList("\01L_OBJC_PROTOCOL_CLASS_METHODS_" + PD->getName(),
Daniel Dunbarae226fa2008-08-27 02:31:56 +00001869 "__OBJC,__cat_cls_meth,regular,no_dead_strip",
1870 ClassMethods);
Owen Anderson08e25242009-07-27 22:29:56 +00001871 llvm::Constant *Init = llvm::ConstantStruct::get(ObjCTypes.ProtocolTy,
Daniel Dunbar6efc0c52008-08-13 03:21:16 +00001872 Values);
Daniel Dunbar6bff2512009-08-03 17:06:42 +00001873
Daniel Dunbar6efc0c52008-08-13 03:21:16 +00001874 if (Entry) {
Daniel Dunbar0c0e7a62008-10-29 22:36:39 +00001875 // Already created, fix the linkage and update the initializer.
1876 Entry->setLinkage(llvm::GlobalValue::InternalLinkage);
Daniel Dunbar6efc0c52008-08-13 03:21:16 +00001877 Entry->setInitializer(Init);
1878 } else {
Daniel Dunbar6bff2512009-08-03 17:06:42 +00001879 Entry =
Owen Anderson1c431b32009-07-08 19:05:04 +00001880 new llvm::GlobalVariable(CGM.getModule(), ObjCTypes.ProtocolTy, false,
Daniel Dunbar6efc0c52008-08-13 03:21:16 +00001881 llvm::GlobalValue::InternalLinkage,
Daniel Dunbar6bff2512009-08-03 17:06:42 +00001882 Init,
Daniel Dunbar9c29bf52009-10-18 20:48:59 +00001883 "\01L_OBJC_PROTOCOL_" + PD->getName());
Daniel Dunbar6efc0c52008-08-13 03:21:16 +00001884 Entry->setSection("__OBJC,__protocol,regular,no_dead_strip");
Daniel Dunbar6efc0c52008-08-13 03:21:16 +00001885 // FIXME: Is this necessary? Why only for protocol?
1886 Entry->setAlignment(4);
1887 }
Chris Lattnerad64e022009-07-17 23:57:13 +00001888 CGM.AddUsedGlobal(Entry);
Daniel Dunbar0c0e7a62008-10-29 22:36:39 +00001889
1890 return Entry;
Daniel Dunbar6efc0c52008-08-13 03:21:16 +00001891}
1892
Daniel Dunbar0c0e7a62008-10-29 22:36:39 +00001893llvm::Constant *CGObjCMac::GetOrEmitProtocolRef(const ObjCProtocolDecl *PD) {
Daniel Dunbar6efc0c52008-08-13 03:21:16 +00001894 llvm::GlobalVariable *&Entry = Protocols[PD->getIdentifier()];
1895
1896 if (!Entry) {
Daniel Dunbar0c0e7a62008-10-29 22:36:39 +00001897 // We use the initializer as a marker of whether this is a forward
1898 // reference or not. At module finalization we add the empty
1899 // contents for protocols which were referenced but never defined.
Daniel Dunbar6bff2512009-08-03 17:06:42 +00001900 Entry =
Owen Anderson1c431b32009-07-08 19:05:04 +00001901 new llvm::GlobalVariable(CGM.getModule(), ObjCTypes.ProtocolTy, false,
Daniel Dunbar0c0e7a62008-10-29 22:36:39 +00001902 llvm::GlobalValue::ExternalLinkage,
1903 0,
Daniel Dunbar9c29bf52009-10-18 20:48:59 +00001904 "\01L_OBJC_PROTOCOL_" + PD->getName());
Daniel Dunbar6efc0c52008-08-13 03:21:16 +00001905 Entry->setSection("__OBJC,__protocol,regular,no_dead_strip");
Daniel Dunbar6efc0c52008-08-13 03:21:16 +00001906 // FIXME: Is this necessary? Why only for protocol?
1907 Entry->setAlignment(4);
1908 }
Daniel Dunbar6bff2512009-08-03 17:06:42 +00001909
Daniel Dunbar6efc0c52008-08-13 03:21:16 +00001910 return Entry;
1911}
1912
1913/*
1914 struct _objc_protocol_extension {
Daniel Dunbar6bff2512009-08-03 17:06:42 +00001915 uint32_t size;
1916 struct objc_method_description_list *optional_instance_methods;
1917 struct objc_method_description_list *optional_class_methods;
1918 struct objc_property_list *instance_properties;
Daniel Dunbar6efc0c52008-08-13 03:21:16 +00001919 };
1920*/
Daniel Dunbarae226fa2008-08-27 02:31:56 +00001921llvm::Constant *
1922CGObjCMac::EmitProtocolExtension(const ObjCProtocolDecl *PD,
1923 const ConstantVector &OptInstanceMethods,
1924 const ConstantVector &OptClassMethods) {
Daniel Dunbar6bff2512009-08-03 17:06:42 +00001925 uint64_t Size =
Duncan Sands9408c452009-05-09 07:08:47 +00001926 CGM.getTargetData().getTypeAllocSize(ObjCTypes.ProtocolExtensionTy);
Daniel Dunbar6efc0c52008-08-13 03:21:16 +00001927 std::vector<llvm::Constant*> Values(4);
Owen Anderson4a28d5d2009-07-24 23:12:58 +00001928 Values[0] = llvm::ConstantInt::get(ObjCTypes.IntTy, Size);
Daniel Dunbar6bff2512009-08-03 17:06:42 +00001929 Values[1] =
1930 EmitMethodDescList("\01L_OBJC_PROTOCOL_INSTANCE_METHODS_OPT_"
Daniel Dunbar9c29bf52009-10-18 20:48:59 +00001931 + PD->getName(),
Daniel Dunbarae226fa2008-08-27 02:31:56 +00001932 "__OBJC,__cat_inst_meth,regular,no_dead_strip",
1933 OptInstanceMethods);
Daniel Dunbar6bff2512009-08-03 17:06:42 +00001934 Values[2] =
Daniel Dunbar9c29bf52009-10-18 20:48:59 +00001935 EmitMethodDescList("\01L_OBJC_PROTOCOL_CLASS_METHODS_OPT_" + PD->getName(),
Daniel Dunbarae226fa2008-08-27 02:31:56 +00001936 "__OBJC,__cat_cls_meth,regular,no_dead_strip",
1937 OptClassMethods);
Daniel Dunbar9c29bf52009-10-18 20:48:59 +00001938 Values[3] = EmitPropertyList("\01L_OBJC_$_PROP_PROTO_LIST_" + PD->getName(),
Fariborz Jahanian5de14dc2009-01-28 22:18:42 +00001939 0, PD, ObjCTypes);
Daniel Dunbar6efc0c52008-08-13 03:21:16 +00001940
Daniel Dunbar27f9d772008-08-21 04:36:09 +00001941 // Return null if no extension bits are used.
Daniel Dunbar6bff2512009-08-03 17:06:42 +00001942 if (Values[1]->isNullValue() && Values[2]->isNullValue() &&
Daniel Dunbar6efc0c52008-08-13 03:21:16 +00001943 Values[3]->isNullValue())
Owen Andersonc9c88b42009-07-31 20:28:54 +00001944 return llvm::Constant::getNullValue(ObjCTypes.ProtocolExtensionPtrTy);
Daniel Dunbar6efc0c52008-08-13 03:21:16 +00001945
Daniel Dunbar6bff2512009-08-03 17:06:42 +00001946 llvm::Constant *Init =
Owen Anderson08e25242009-07-27 22:29:56 +00001947 llvm::ConstantStruct::get(ObjCTypes.ProtocolExtensionTy, Values);
Daniel Dunbar6efc0c52008-08-13 03:21:16 +00001948
Daniel Dunbar63c5b502009-03-09 21:49:58 +00001949 // No special section, but goes in llvm.used
Daniel Dunbar9c29bf52009-10-18 20:48:59 +00001950 return CreateMetadataVar("\01L_OBJC_PROTOCOLEXT_" + PD->getName(),
Daniel Dunbar6bff2512009-08-03 17:06:42 +00001951 Init,
Daniel Dunbar63c5b502009-03-09 21:49:58 +00001952 0, 0, true);
Daniel Dunbar6efc0c52008-08-13 03:21:16 +00001953}
1954
1955/*
1956 struct objc_protocol_list {
Daniel Dunbar6bff2512009-08-03 17:06:42 +00001957 struct objc_protocol_list *next;
1958 long count;
1959 Protocol *list[];
Daniel Dunbar6efc0c52008-08-13 03:21:16 +00001960 };
1961*/
Daniel Dunbardbc933702008-08-21 21:57:41 +00001962llvm::Constant *
Daniel Dunbar9c29bf52009-10-18 20:48:59 +00001963CGObjCMac::EmitProtocolList(llvm::Twine Name,
Daniel Dunbardbc933702008-08-21 21:57:41 +00001964 ObjCProtocolDecl::protocol_iterator begin,
1965 ObjCProtocolDecl::protocol_iterator end) {
Daniel Dunbar6efc0c52008-08-13 03:21:16 +00001966 std::vector<llvm::Constant*> ProtocolRefs;
1967
Daniel Dunbardbc933702008-08-21 21:57:41 +00001968 for (; begin != end; ++begin)
1969 ProtocolRefs.push_back(GetProtocolRef(*begin));
Daniel Dunbar6efc0c52008-08-13 03:21:16 +00001970
1971 // Just return null for empty protocol lists
Daniel Dunbar6bff2512009-08-03 17:06:42 +00001972 if (ProtocolRefs.empty())
Owen Andersonc9c88b42009-07-31 20:28:54 +00001973 return llvm::Constant::getNullValue(ObjCTypes.ProtocolListPtrTy);
Daniel Dunbar6efc0c52008-08-13 03:21:16 +00001974
Daniel Dunbar27f9d772008-08-21 04:36:09 +00001975 // This list is null terminated.
Owen Andersonc9c88b42009-07-31 20:28:54 +00001976 ProtocolRefs.push_back(llvm::Constant::getNullValue(ObjCTypes.ProtocolPtrTy));
Daniel Dunbar6efc0c52008-08-13 03:21:16 +00001977
1978 std::vector<llvm::Constant*> Values(3);
Daniel Dunbar27f9d772008-08-21 04:36:09 +00001979 // This field is only used by the runtime.
Owen Andersonc9c88b42009-07-31 20:28:54 +00001980 Values[0] = llvm::Constant::getNullValue(ObjCTypes.ProtocolListPtrTy);
Owen Anderson4a28d5d2009-07-24 23:12:58 +00001981 Values[1] = llvm::ConstantInt::get(ObjCTypes.LongTy,
Daniel Dunbar6bff2512009-08-03 17:06:42 +00001982 ProtocolRefs.size() - 1);
1983 Values[2] =
1984 llvm::ConstantArray::get(llvm::ArrayType::get(ObjCTypes.ProtocolPtrTy,
1985 ProtocolRefs.size()),
Daniel Dunbar6efc0c52008-08-13 03:21:16 +00001986 ProtocolRefs);
Daniel Dunbar6bff2512009-08-03 17:06:42 +00001987
Nick Lewycky0d36dd22009-09-19 20:00:52 +00001988 llvm::Constant *Init = llvm::ConstantStruct::get(VMContext, Values, false);
Daniel Dunbar6bff2512009-08-03 17:06:42 +00001989 llvm::GlobalVariable *GV =
Daniel Dunbar63c5b502009-03-09 21:49:58 +00001990 CreateMetadataVar(Name, Init, "__OBJC,__cat_cls_meth,regular,no_dead_strip",
Daniel Dunbar58a29122009-03-09 22:18:41 +00001991 4, false);
Owen Anderson3c4972d2009-07-29 18:54:39 +00001992 return llvm::ConstantExpr::getBitCast(GV, ObjCTypes.ProtocolListPtrTy);
Daniel Dunbar6efc0c52008-08-13 03:21:16 +00001993}
1994
Fariborz Jahanian191dcd72009-12-12 21:26:21 +00001995void CGObjCCommonMac::PushProtocolProperties(llvm::SmallPtrSet<const IdentifierInfo*, 16> &PropertySet,
1996 std::vector<llvm::Constant*> &Properties,
1997 const Decl *Container,
1998 const ObjCProtocolDecl *PROTO,
1999 const ObjCCommonTypesHelper &ObjCTypes) {
2000 std::vector<llvm::Constant*> Prop(2);
2001 for (ObjCProtocolDecl::protocol_iterator P = PROTO->protocol_begin(),
2002 E = PROTO->protocol_end(); P != E; ++P)
2003 PushProtocolProperties(PropertySet, Properties, Container, (*P), ObjCTypes);
2004 for (ObjCContainerDecl::prop_iterator I = PROTO->prop_begin(),
2005 E = PROTO->prop_end(); I != E; ++I) {
2006 const ObjCPropertyDecl *PD = *I;
2007 if (!PropertySet.insert(PD->getIdentifier()))
2008 continue;
2009 Prop[0] = GetPropertyName(PD->getIdentifier());
2010 Prop[1] = GetPropertyTypeString(PD, Container);
2011 Properties.push_back(llvm::ConstantStruct::get(ObjCTypes.PropertyTy, Prop));
2012 }
2013}
2014
Daniel Dunbar6efc0c52008-08-13 03:21:16 +00002015/*
Daniel Dunbarc8ef5512008-08-23 00:19:03 +00002016 struct _objc_property {
Daniel Dunbar6bff2512009-08-03 17:06:42 +00002017 const char * const name;
2018 const char * const attributes;
Daniel Dunbarc8ef5512008-08-23 00:19:03 +00002019 };
2020
2021 struct _objc_property_list {
Daniel Dunbar6bff2512009-08-03 17:06:42 +00002022 uint32_t entsize; // sizeof (struct _objc_property)
2023 uint32_t prop_count;
2024 struct _objc_property[prop_count];
Daniel Dunbarc8ef5512008-08-23 00:19:03 +00002025 };
2026*/
Daniel Dunbar9c29bf52009-10-18 20:48:59 +00002027llvm::Constant *CGObjCCommonMac::EmitPropertyList(llvm::Twine Name,
Daniel Dunbar6bff2512009-08-03 17:06:42 +00002028 const Decl *Container,
2029 const ObjCContainerDecl *OCD,
2030 const ObjCCommonTypesHelper &ObjCTypes) {
Daniel Dunbarc8ef5512008-08-23 00:19:03 +00002031 std::vector<llvm::Constant*> Properties, Prop(2);
Fariborz Jahanian191dcd72009-12-12 21:26:21 +00002032 llvm::SmallPtrSet<const IdentifierInfo*, 16> PropertySet;
Daniel Dunbar6bff2512009-08-03 17:06:42 +00002033 for (ObjCContainerDecl::prop_iterator I = OCD->prop_begin(),
2034 E = OCD->prop_end(); I != E; ++I) {
Steve Naroff93983f82009-01-11 12:47:58 +00002035 const ObjCPropertyDecl *PD = *I;
Fariborz Jahanian191dcd72009-12-12 21:26:21 +00002036 PropertySet.insert(PD->getIdentifier());
Daniel Dunbarc8ef5512008-08-23 00:19:03 +00002037 Prop[0] = GetPropertyName(PD->getIdentifier());
Daniel Dunbarc56f34a2008-08-28 04:38:10 +00002038 Prop[1] = GetPropertyTypeString(PD, Container);
Owen Anderson08e25242009-07-27 22:29:56 +00002039 Properties.push_back(llvm::ConstantStruct::get(ObjCTypes.PropertyTy,
Daniel Dunbarc8ef5512008-08-23 00:19:03 +00002040 Prop));
2041 }
Fariborz Jahanian6afbdf52010-06-22 16:33:55 +00002042 if (const ObjCInterfaceDecl *OID = dyn_cast<ObjCInterfaceDecl>(OCD)) {
Fariborz Jahanian191dcd72009-12-12 21:26:21 +00002043 for (ObjCInterfaceDecl::protocol_iterator P = OID->protocol_begin(),
2044 E = OID->protocol_end(); P != E; ++P)
Fariborz Jahanian6afbdf52010-06-22 16:33:55 +00002045 PushProtocolProperties(PropertySet, Properties, Container, (*P),
2046 ObjCTypes);
2047 }
2048 else if (const ObjCCategoryDecl *CD = dyn_cast<ObjCCategoryDecl>(OCD)) {
2049 for (ObjCCategoryDecl::protocol_iterator P = CD->protocol_begin(),
2050 E = CD->protocol_end(); P != E; ++P)
2051 PushProtocolProperties(PropertySet, Properties, Container, (*P),
2052 ObjCTypes);
2053 }
Daniel Dunbarc8ef5512008-08-23 00:19:03 +00002054
2055 // Return null for empty list.
2056 if (Properties.empty())
Owen Andersonc9c88b42009-07-31 20:28:54 +00002057 return llvm::Constant::getNullValue(ObjCTypes.PropertyListPtrTy);
Daniel Dunbarc8ef5512008-08-23 00:19:03 +00002058
Daniel Dunbar6bff2512009-08-03 17:06:42 +00002059 unsigned PropertySize =
Duncan Sands9408c452009-05-09 07:08:47 +00002060 CGM.getTargetData().getTypeAllocSize(ObjCTypes.PropertyTy);
Daniel Dunbarc8ef5512008-08-23 00:19:03 +00002061 std::vector<llvm::Constant*> Values(3);
Owen Anderson4a28d5d2009-07-24 23:12:58 +00002062 Values[0] = llvm::ConstantInt::get(ObjCTypes.IntTy, PropertySize);
2063 Values[1] = llvm::ConstantInt::get(ObjCTypes.IntTy, Properties.size());
Daniel Dunbar6bff2512009-08-03 17:06:42 +00002064 llvm::ArrayType *AT = llvm::ArrayType::get(ObjCTypes.PropertyTy,
Daniel Dunbarc8ef5512008-08-23 00:19:03 +00002065 Properties.size());
Owen Anderson7db6d832009-07-28 18:33:04 +00002066 Values[2] = llvm::ConstantArray::get(AT, Properties);
Nick Lewycky0d36dd22009-09-19 20:00:52 +00002067 llvm::Constant *Init = llvm::ConstantStruct::get(VMContext, Values, false);
Daniel Dunbarc8ef5512008-08-23 00:19:03 +00002068
Daniel Dunbar6bff2512009-08-03 17:06:42 +00002069 llvm::GlobalVariable *GV =
2070 CreateMetadataVar(Name, Init,
2071 (ObjCABI == 2) ? "__DATA, __objc_const" :
Daniel Dunbar0bf21992009-04-15 02:56:18 +00002072 "__OBJC,__property,regular,no_dead_strip",
Daniel Dunbar6bff2512009-08-03 17:06:42 +00002073 (ObjCABI == 2) ? 8 : 4,
Daniel Dunbar0bf21992009-04-15 02:56:18 +00002074 true);
Owen Anderson3c4972d2009-07-29 18:54:39 +00002075 return llvm::ConstantExpr::getBitCast(GV, ObjCTypes.PropertyListPtrTy);
Daniel Dunbarc8ef5512008-08-23 00:19:03 +00002076}
2077
2078/*
Daniel Dunbar6efc0c52008-08-13 03:21:16 +00002079 struct objc_method_description_list {
Daniel Dunbar6bff2512009-08-03 17:06:42 +00002080 int count;
2081 struct objc_method_description list[];
Daniel Dunbar6efc0c52008-08-13 03:21:16 +00002082 };
2083*/
Daniel Dunbarae226fa2008-08-27 02:31:56 +00002084llvm::Constant *
2085CGObjCMac::GetMethodDescriptionConstant(const ObjCMethodDecl *MD) {
2086 std::vector<llvm::Constant*> Desc(2);
Owen Andersona1cf15f2009-07-14 23:10:40 +00002087 Desc[0] =
Daniel Dunbar6bff2512009-08-03 17:06:42 +00002088 llvm::ConstantExpr::getBitCast(GetMethodVarName(MD->getSelector()),
2089 ObjCTypes.SelectorPtrTy);
Daniel Dunbarae226fa2008-08-27 02:31:56 +00002090 Desc[1] = GetMethodVarType(MD);
Owen Anderson08e25242009-07-27 22:29:56 +00002091 return llvm::ConstantStruct::get(ObjCTypes.MethodDescriptionTy,
Daniel Dunbarae226fa2008-08-27 02:31:56 +00002092 Desc);
2093}
Daniel Dunbar6efc0c52008-08-13 03:21:16 +00002094
Daniel Dunbar9c29bf52009-10-18 20:48:59 +00002095llvm::Constant *CGObjCMac::EmitMethodDescList(llvm::Twine Name,
Daniel Dunbarae226fa2008-08-27 02:31:56 +00002096 const char *Section,
2097 const ConstantVector &Methods) {
Daniel Dunbar6efc0c52008-08-13 03:21:16 +00002098 // Return null for empty list.
2099 if (Methods.empty())
Owen Andersonc9c88b42009-07-31 20:28:54 +00002100 return llvm::Constant::getNullValue(ObjCTypes.MethodDescriptionListPtrTy);
Daniel Dunbar6efc0c52008-08-13 03:21:16 +00002101
2102 std::vector<llvm::Constant*> Values(2);
Owen Anderson4a28d5d2009-07-24 23:12:58 +00002103 Values[0] = llvm::ConstantInt::get(ObjCTypes.IntTy, Methods.size());
Daniel Dunbar6bff2512009-08-03 17:06:42 +00002104 llvm::ArrayType *AT = llvm::ArrayType::get(ObjCTypes.MethodDescriptionTy,
Daniel Dunbar6efc0c52008-08-13 03:21:16 +00002105 Methods.size());
Owen Anderson7db6d832009-07-28 18:33:04 +00002106 Values[1] = llvm::ConstantArray::get(AT, Methods);
Nick Lewycky0d36dd22009-09-19 20:00:52 +00002107 llvm::Constant *Init = llvm::ConstantStruct::get(VMContext, Values, false);
Daniel Dunbar6efc0c52008-08-13 03:21:16 +00002108
Daniel Dunbar0bf21992009-04-15 02:56:18 +00002109 llvm::GlobalVariable *GV = CreateMetadataVar(Name, Init, Section, 4, true);
Daniel Dunbar6bff2512009-08-03 17:06:42 +00002110 return llvm::ConstantExpr::getBitCast(GV,
Daniel Dunbar6efc0c52008-08-13 03:21:16 +00002111 ObjCTypes.MethodDescriptionListPtrTy);
Daniel Dunbarc17a4d32008-08-11 02:45:11 +00002112}
2113
Daniel Dunbar86e253a2008-08-22 20:34:54 +00002114/*
2115 struct _objc_category {
Daniel Dunbar6bff2512009-08-03 17:06:42 +00002116 char *category_name;
2117 char *class_name;
2118 struct _objc_method_list *instance_methods;
2119 struct _objc_method_list *class_methods;
2120 struct _objc_protocol_list *protocols;
2121 uint32_t size; // <rdar://4585769>
2122 struct _objc_property_list *instance_properties;
Daniel Dunbar86e253a2008-08-22 20:34:54 +00002123 };
Daniel Dunbar6bff2512009-08-03 17:06:42 +00002124*/
Daniel Dunbar7ded7f42008-08-15 22:20:32 +00002125void CGObjCMac::GenerateCategory(const ObjCCategoryImplDecl *OCD) {
Duncan Sands9408c452009-05-09 07:08:47 +00002126 unsigned Size = CGM.getTargetData().getTypeAllocSize(ObjCTypes.CategoryTy);
Daniel Dunbar86e253a2008-08-22 20:34:54 +00002127
Mike Stumpf5408fe2009-05-16 07:57:57 +00002128 // FIXME: This is poor design, the OCD should have a pointer to the category
2129 // decl. Additionally, note that Category can be null for the @implementation
2130 // w/o an @interface case. Sema should just create one for us as it does for
2131 // @implementation so everyone else can live life under a clear blue sky.
Daniel Dunbar86e253a2008-08-22 20:34:54 +00002132 const ObjCInterfaceDecl *Interface = OCD->getClassInterface();
Daniel Dunbar6bff2512009-08-03 17:06:42 +00002133 const ObjCCategoryDecl *Category =
Daniel Dunbar86e2f402008-08-26 23:03:11 +00002134 Interface->FindCategoryDeclaration(OCD->getIdentifier());
Daniel Dunbar9c29bf52009-10-18 20:48:59 +00002135
2136 llvm::SmallString<256> ExtName;
2137 llvm::raw_svector_ostream(ExtName) << Interface->getName() << '_'
2138 << OCD->getName();
Daniel Dunbar86e253a2008-08-22 20:34:54 +00002139
Daniel Dunbarc45ef602008-08-26 21:51:14 +00002140 std::vector<llvm::Constant*> InstanceMethods, ClassMethods;
Daniel Dunbar6bff2512009-08-03 17:06:42 +00002141 for (ObjCCategoryImplDecl::instmeth_iterator
Argyrios Kyrtzidis17945a02009-06-30 02:36:12 +00002142 i = OCD->instmeth_begin(), e = OCD->instmeth_end(); i != e; ++i) {
Daniel Dunbarc45ef602008-08-26 21:51:14 +00002143 // Instance methods should always be defined.
2144 InstanceMethods.push_back(GetMethodConstant(*i));
2145 }
Daniel Dunbar6bff2512009-08-03 17:06:42 +00002146 for (ObjCCategoryImplDecl::classmeth_iterator
Argyrios Kyrtzidis17945a02009-06-30 02:36:12 +00002147 i = OCD->classmeth_begin(), e = OCD->classmeth_end(); i != e; ++i) {
Daniel Dunbarc45ef602008-08-26 21:51:14 +00002148 // Class methods should always be defined.
2149 ClassMethods.push_back(GetMethodConstant(*i));
2150 }
2151
Daniel Dunbar86e253a2008-08-22 20:34:54 +00002152 std::vector<llvm::Constant*> Values(7);
2153 Values[0] = GetClassName(OCD->getIdentifier());
2154 Values[1] = GetClassName(Interface->getIdentifier());
Fariborz Jahanian679cd7f2009-04-29 20:40:05 +00002155 LazySymbols.insert(Interface->getIdentifier());
Daniel Dunbar6bff2512009-08-03 17:06:42 +00002156 Values[2] =
Daniel Dunbar9c29bf52009-10-18 20:48:59 +00002157 EmitMethodList("\01L_OBJC_CATEGORY_INSTANCE_METHODS_" + ExtName.str(),
Daniel Dunbarc8ef5512008-08-23 00:19:03 +00002158 "__OBJC,__cat_inst_meth,regular,no_dead_strip",
Daniel Dunbarc45ef602008-08-26 21:51:14 +00002159 InstanceMethods);
Daniel Dunbar6bff2512009-08-03 17:06:42 +00002160 Values[3] =
Daniel Dunbar9c29bf52009-10-18 20:48:59 +00002161 EmitMethodList("\01L_OBJC_CATEGORY_CLASS_METHODS_" + ExtName.str(),
Daniel Dunbar0bf21992009-04-15 02:56:18 +00002162 "__OBJC,__cat_cls_meth,regular,no_dead_strip",
Daniel Dunbarc45ef602008-08-26 21:51:14 +00002163 ClassMethods);
Daniel Dunbarae226fa2008-08-27 02:31:56 +00002164 if (Category) {
Daniel Dunbar6bff2512009-08-03 17:06:42 +00002165 Values[4] =
Daniel Dunbar9c29bf52009-10-18 20:48:59 +00002166 EmitProtocolList("\01L_OBJC_CATEGORY_PROTOCOLS_" + ExtName.str(),
Daniel Dunbarae226fa2008-08-27 02:31:56 +00002167 Category->protocol_begin(),
2168 Category->protocol_end());
2169 } else {
Owen Andersonc9c88b42009-07-31 20:28:54 +00002170 Values[4] = llvm::Constant::getNullValue(ObjCTypes.ProtocolListPtrTy);
Daniel Dunbarae226fa2008-08-27 02:31:56 +00002171 }
Owen Anderson4a28d5d2009-07-24 23:12:58 +00002172 Values[5] = llvm::ConstantInt::get(ObjCTypes.IntTy, Size);
Daniel Dunbar86e2f402008-08-26 23:03:11 +00002173
2174 // If there is no category @interface then there can be no properties.
2175 if (Category) {
Daniel Dunbar9c29bf52009-10-18 20:48:59 +00002176 Values[6] = EmitPropertyList("\01l_OBJC_$_PROP_LIST_" + ExtName.str(),
Fariborz Jahanian5de14dc2009-01-28 22:18:42 +00002177 OCD, Category, ObjCTypes);
Daniel Dunbar86e2f402008-08-26 23:03:11 +00002178 } else {
Owen Andersonc9c88b42009-07-31 20:28:54 +00002179 Values[6] = llvm::Constant::getNullValue(ObjCTypes.PropertyListPtrTy);
Daniel Dunbar86e2f402008-08-26 23:03:11 +00002180 }
Daniel Dunbar6bff2512009-08-03 17:06:42 +00002181
Owen Anderson08e25242009-07-27 22:29:56 +00002182 llvm::Constant *Init = llvm::ConstantStruct::get(ObjCTypes.CategoryTy,
Daniel Dunbar86e253a2008-08-22 20:34:54 +00002183 Values);
2184
Daniel Dunbar6bff2512009-08-03 17:06:42 +00002185 llvm::GlobalVariable *GV =
Daniel Dunbar9c29bf52009-10-18 20:48:59 +00002186 CreateMetadataVar("\01L_OBJC_CATEGORY_" + ExtName.str(), Init,
Daniel Dunbar63c5b502009-03-09 21:49:58 +00002187 "__OBJC,__category,regular,no_dead_strip",
Daniel Dunbar58a29122009-03-09 22:18:41 +00002188 4, true);
Daniel Dunbar86e253a2008-08-22 20:34:54 +00002189 DefinedCategories.push_back(GV);
Fariborz Jahanianb9c5b3d2010-06-21 22:05:18 +00002190 DefinedCategoryNames.insert(ExtName.str());
Daniel Dunbarc17a4d32008-08-11 02:45:11 +00002191}
2192
Daniel Dunbar27f9d772008-08-21 04:36:09 +00002193// FIXME: Get from somewhere?
2194enum ClassFlags {
2195 eClassFlags_Factory = 0x00001,
2196 eClassFlags_Meta = 0x00002,
2197 // <rdr://5142207>
2198 eClassFlags_HasCXXStructors = 0x02000,
2199 eClassFlags_Hidden = 0x20000,
2200 eClassFlags_ABI2_Hidden = 0x00010,
2201 eClassFlags_ABI2_HasCXXStructors = 0x00004 // <rdr://4923634>
2202};
2203
Daniel Dunbar27f9d772008-08-21 04:36:09 +00002204/*
2205 struct _objc_class {
Daniel Dunbar6bff2512009-08-03 17:06:42 +00002206 Class isa;
2207 Class super_class;
2208 const char *name;
2209 long version;
2210 long info;
2211 long instance_size;
2212 struct _objc_ivar_list *ivars;
2213 struct _objc_method_list *methods;
2214 struct _objc_cache *cache;
2215 struct _objc_protocol_list *protocols;
2216 // Objective-C 1.0 extensions (<rdr://4585769>)
2217 const char *ivar_layout;
2218 struct _objc_class_ext *ext;
Daniel Dunbar27f9d772008-08-21 04:36:09 +00002219 };
2220
2221 See EmitClassExtension();
Daniel Dunbar6bff2512009-08-03 17:06:42 +00002222*/
Daniel Dunbar27f9d772008-08-21 04:36:09 +00002223void CGObjCMac::GenerateClass(const ObjCImplementationDecl *ID) {
Daniel Dunbar242d4dc2008-08-25 06:02:07 +00002224 DefinedSymbols.insert(ID->getIdentifier());
2225
Chris Lattner8ec03f52008-11-24 03:54:41 +00002226 std::string ClassName = ID->getNameAsString();
Daniel Dunbar27f9d772008-08-21 04:36:09 +00002227 // FIXME: Gross
Daniel Dunbar6bff2512009-08-03 17:06:42 +00002228 ObjCInterfaceDecl *Interface =
Daniel Dunbar27f9d772008-08-21 04:36:09 +00002229 const_cast<ObjCInterfaceDecl*>(ID->getClassInterface());
Daniel Dunbar6bff2512009-08-03 17:06:42 +00002230 llvm::Constant *Protocols =
Daniel Dunbar9c29bf52009-10-18 20:48:59 +00002231 EmitProtocolList("\01L_OBJC_CLASS_PROTOCOLS_" + ID->getName(),
Daniel Dunbardbc933702008-08-21 21:57:41 +00002232 Interface->protocol_begin(),
2233 Interface->protocol_end());
Daniel Dunbar27f9d772008-08-21 04:36:09 +00002234 unsigned Flags = eClassFlags_Factory;
Fariborz Jahanian109dfc62010-04-28 21:28:56 +00002235 if (ID->getNumIvarInitializers())
2236 Flags |= eClassFlags_HasCXXStructors;
Daniel Dunbar6bff2512009-08-03 17:06:42 +00002237 unsigned Size =
Daniel Dunbar2bebbf02009-05-03 10:46:44 +00002238 CGM.getContext().getASTObjCImplementationLayout(ID).getSize() / 8;
Daniel Dunbar27f9d772008-08-21 04:36:09 +00002239
2240 // FIXME: Set CXX-structors flag.
Daniel Dunbar04d40782009-04-14 06:00:08 +00002241 if (CGM.getDeclVisibilityMode(ID->getClassInterface()) == LangOptions::Hidden)
Daniel Dunbar27f9d772008-08-21 04:36:09 +00002242 Flags |= eClassFlags_Hidden;
2243
Daniel Dunbarc45ef602008-08-26 21:51:14 +00002244 std::vector<llvm::Constant*> InstanceMethods, ClassMethods;
Daniel Dunbar6bff2512009-08-03 17:06:42 +00002245 for (ObjCImplementationDecl::instmeth_iterator
Argyrios Kyrtzidis17945a02009-06-30 02:36:12 +00002246 i = ID->instmeth_begin(), e = ID->instmeth_end(); i != e; ++i) {
Daniel Dunbarc45ef602008-08-26 21:51:14 +00002247 // Instance methods should always be defined.
2248 InstanceMethods.push_back(GetMethodConstant(*i));
2249 }
Daniel Dunbar6bff2512009-08-03 17:06:42 +00002250 for (ObjCImplementationDecl::classmeth_iterator
Argyrios Kyrtzidis17945a02009-06-30 02:36:12 +00002251 i = ID->classmeth_begin(), e = ID->classmeth_end(); i != e; ++i) {
Daniel Dunbarc45ef602008-08-26 21:51:14 +00002252 // Class methods should always be defined.
2253 ClassMethods.push_back(GetMethodConstant(*i));
2254 }
2255
Daniel Dunbar6bff2512009-08-03 17:06:42 +00002256 for (ObjCImplementationDecl::propimpl_iterator
Argyrios Kyrtzidis17945a02009-06-30 02:36:12 +00002257 i = ID->propimpl_begin(), e = ID->propimpl_end(); i != e; ++i) {
Daniel Dunbarc45ef602008-08-26 21:51:14 +00002258 ObjCPropertyImplDecl *PID = *i;
2259
2260 if (PID->getPropertyImplementation() == ObjCPropertyImplDecl::Synthesize) {
2261 ObjCPropertyDecl *PD = PID->getPropertyDecl();
2262
2263 if (ObjCMethodDecl *MD = PD->getGetterMethodDecl())
2264 if (llvm::Constant *C = GetMethodConstant(MD))
2265 InstanceMethods.push_back(C);
2266 if (ObjCMethodDecl *MD = PD->getSetterMethodDecl())
2267 if (llvm::Constant *C = GetMethodConstant(MD))
2268 InstanceMethods.push_back(C);
2269 }
2270 }
2271
Daniel Dunbar27f9d772008-08-21 04:36:09 +00002272 std::vector<llvm::Constant*> Values(12);
Daniel Dunbar5384b092009-05-03 08:56:52 +00002273 Values[ 0] = EmitMetaClass(ID, Protocols, ClassMethods);
Daniel Dunbar27f9d772008-08-21 04:36:09 +00002274 if (ObjCInterfaceDecl *Super = Interface->getSuperClass()) {
Daniel Dunbar242d4dc2008-08-25 06:02:07 +00002275 // Record a reference to the super class.
2276 LazySymbols.insert(Super->getIdentifier());
2277
Daniel Dunbar6bff2512009-08-03 17:06:42 +00002278 Values[ 1] =
Owen Anderson3c4972d2009-07-29 18:54:39 +00002279 llvm::ConstantExpr::getBitCast(GetClassName(Super->getIdentifier()),
Daniel Dunbar27f9d772008-08-21 04:36:09 +00002280 ObjCTypes.ClassPtrTy);
2281 } else {
Owen Andersonc9c88b42009-07-31 20:28:54 +00002282 Values[ 1] = llvm::Constant::getNullValue(ObjCTypes.ClassPtrTy);
Daniel Dunbar27f9d772008-08-21 04:36:09 +00002283 }
2284 Values[ 2] = GetClassName(ID->getIdentifier());
2285 // Version is always 0.
Owen Anderson4a28d5d2009-07-24 23:12:58 +00002286 Values[ 3] = llvm::ConstantInt::get(ObjCTypes.LongTy, 0);
2287 Values[ 4] = llvm::ConstantInt::get(ObjCTypes.LongTy, Flags);
2288 Values[ 5] = llvm::ConstantInt::get(ObjCTypes.LongTy, Size);
Fariborz Jahanian46b86c62009-01-28 19:12:34 +00002289 Values[ 6] = EmitIvarList(ID, false);
Daniel Dunbar6bff2512009-08-03 17:06:42 +00002290 Values[ 7] =
Daniel Dunbar9c29bf52009-10-18 20:48:59 +00002291 EmitMethodList("\01L_OBJC_INSTANCE_METHODS_" + ID->getName(),
Daniel Dunbarc8ef5512008-08-23 00:19:03 +00002292 "__OBJC,__inst_meth,regular,no_dead_strip",
Daniel Dunbarc45ef602008-08-26 21:51:14 +00002293 InstanceMethods);
Daniel Dunbar27f9d772008-08-21 04:36:09 +00002294 // cache is always NULL.
Owen Andersonc9c88b42009-07-31 20:28:54 +00002295 Values[ 8] = llvm::Constant::getNullValue(ObjCTypes.CachePtrTy);
Daniel Dunbar27f9d772008-08-21 04:36:09 +00002296 Values[ 9] = Protocols;
Daniel Dunbar6bff2512009-08-03 17:06:42 +00002297 Values[10] = BuildIvarLayout(ID, true);
Daniel Dunbar27f9d772008-08-21 04:36:09 +00002298 Values[11] = EmitClassExtension(ID);
Owen Anderson08e25242009-07-27 22:29:56 +00002299 llvm::Constant *Init = llvm::ConstantStruct::get(ObjCTypes.ClassTy,
Daniel Dunbar27f9d772008-08-21 04:36:09 +00002300 Values);
Fariborz Jahanianb0069ee2009-11-12 20:14:24 +00002301 std::string Name("\01L_OBJC_CLASS_");
2302 Name += ClassName;
2303 const char *Section = "__OBJC,__class,regular,no_dead_strip";
2304 // Check for a forward reference.
2305 llvm::GlobalVariable *GV = CGM.getModule().getGlobalVariable(Name);
2306 if (GV) {
2307 assert(GV->getType()->getElementType() == ObjCTypes.ClassTy &&
2308 "Forward metaclass reference has incorrect type.");
2309 GV->setLinkage(llvm::GlobalValue::InternalLinkage);
2310 GV->setInitializer(Init);
2311 GV->setSection(Section);
2312 GV->setAlignment(4);
2313 CGM.AddUsedGlobal(GV);
2314 }
2315 else
2316 GV = CreateMetadataVar(Name, Init, Section, 4, true);
Daniel Dunbar27f9d772008-08-21 04:36:09 +00002317 DefinedClasses.push_back(GV);
2318}
2319
2320llvm::Constant *CGObjCMac::EmitMetaClass(const ObjCImplementationDecl *ID,
2321 llvm::Constant *Protocols,
Daniel Dunbarae226fa2008-08-27 02:31:56 +00002322 const ConstantVector &Methods) {
Daniel Dunbar27f9d772008-08-21 04:36:09 +00002323 unsigned Flags = eClassFlags_Meta;
Duncan Sands9408c452009-05-09 07:08:47 +00002324 unsigned Size = CGM.getTargetData().getTypeAllocSize(ObjCTypes.ClassTy);
Daniel Dunbar27f9d772008-08-21 04:36:09 +00002325
Daniel Dunbar04d40782009-04-14 06:00:08 +00002326 if (CGM.getDeclVisibilityMode(ID->getClassInterface()) == LangOptions::Hidden)
Daniel Dunbar27f9d772008-08-21 04:36:09 +00002327 Flags |= eClassFlags_Hidden;
Daniel Dunbar6bff2512009-08-03 17:06:42 +00002328
Daniel Dunbar27f9d772008-08-21 04:36:09 +00002329 std::vector<llvm::Constant*> Values(12);
2330 // The isa for the metaclass is the root of the hierarchy.
2331 const ObjCInterfaceDecl *Root = ID->getClassInterface();
2332 while (const ObjCInterfaceDecl *Super = Root->getSuperClass())
2333 Root = Super;
Daniel Dunbar6bff2512009-08-03 17:06:42 +00002334 Values[ 0] =
Owen Anderson3c4972d2009-07-29 18:54:39 +00002335 llvm::ConstantExpr::getBitCast(GetClassName(Root->getIdentifier()),
Daniel Dunbar27f9d772008-08-21 04:36:09 +00002336 ObjCTypes.ClassPtrTy);
Daniel Dunbar86e253a2008-08-22 20:34:54 +00002337 // The super class for the metaclass is emitted as the name of the
2338 // super class. The runtime fixes this up to point to the
2339 // *metaclass* for the super class.
Daniel Dunbar27f9d772008-08-21 04:36:09 +00002340 if (ObjCInterfaceDecl *Super = ID->getClassInterface()->getSuperClass()) {
Daniel Dunbar6bff2512009-08-03 17:06:42 +00002341 Values[ 1] =
Owen Anderson3c4972d2009-07-29 18:54:39 +00002342 llvm::ConstantExpr::getBitCast(GetClassName(Super->getIdentifier()),
Daniel Dunbar27f9d772008-08-21 04:36:09 +00002343 ObjCTypes.ClassPtrTy);
2344 } else {
Owen Andersonc9c88b42009-07-31 20:28:54 +00002345 Values[ 1] = llvm::Constant::getNullValue(ObjCTypes.ClassPtrTy);
Daniel Dunbar27f9d772008-08-21 04:36:09 +00002346 }
2347 Values[ 2] = GetClassName(ID->getIdentifier());
2348 // Version is always 0.
Owen Anderson4a28d5d2009-07-24 23:12:58 +00002349 Values[ 3] = llvm::ConstantInt::get(ObjCTypes.LongTy, 0);
2350 Values[ 4] = llvm::ConstantInt::get(ObjCTypes.LongTy, Flags);
2351 Values[ 5] = llvm::ConstantInt::get(ObjCTypes.LongTy, Size);
Fariborz Jahanian46b86c62009-01-28 19:12:34 +00002352 Values[ 6] = EmitIvarList(ID, true);
Daniel Dunbar6bff2512009-08-03 17:06:42 +00002353 Values[ 7] =
Chris Lattnerd9d22dd2008-11-24 05:29:24 +00002354 EmitMethodList("\01L_OBJC_CLASS_METHODS_" + ID->getNameAsString(),
Daniel Dunbar0bf21992009-04-15 02:56:18 +00002355 "__OBJC,__cls_meth,regular,no_dead_strip",
Daniel Dunbarc45ef602008-08-26 21:51:14 +00002356 Methods);
Daniel Dunbar27f9d772008-08-21 04:36:09 +00002357 // cache is always NULL.
Owen Andersonc9c88b42009-07-31 20:28:54 +00002358 Values[ 8] = llvm::Constant::getNullValue(ObjCTypes.CachePtrTy);
Daniel Dunbar27f9d772008-08-21 04:36:09 +00002359 Values[ 9] = Protocols;
2360 // ivar_layout for metaclass is always NULL.
Owen Andersonc9c88b42009-07-31 20:28:54 +00002361 Values[10] = llvm::Constant::getNullValue(ObjCTypes.Int8PtrTy);
Daniel Dunbar27f9d772008-08-21 04:36:09 +00002362 // The class extension is always unused for metaclasses.
Owen Andersonc9c88b42009-07-31 20:28:54 +00002363 Values[11] = llvm::Constant::getNullValue(ObjCTypes.ClassExtensionPtrTy);
Owen Anderson08e25242009-07-27 22:29:56 +00002364 llvm::Constant *Init = llvm::ConstantStruct::get(ObjCTypes.ClassTy,
Daniel Dunbar27f9d772008-08-21 04:36:09 +00002365 Values);
2366
Daniel Dunbarf56f1912008-08-25 08:19:24 +00002367 std::string Name("\01L_OBJC_METACLASS_");
Chris Lattner8ec03f52008-11-24 03:54:41 +00002368 Name += ID->getNameAsCString();
Daniel Dunbarf56f1912008-08-25 08:19:24 +00002369
2370 // Check for a forward reference.
2371 llvm::GlobalVariable *GV = CGM.getModule().getGlobalVariable(Name);
2372 if (GV) {
2373 assert(GV->getType()->getElementType() == ObjCTypes.ClassTy &&
2374 "Forward metaclass reference has incorrect type.");
2375 GV->setLinkage(llvm::GlobalValue::InternalLinkage);
2376 GV->setInitializer(Init);
2377 } else {
Owen Anderson1c431b32009-07-08 19:05:04 +00002378 GV = new llvm::GlobalVariable(CGM.getModule(), ObjCTypes.ClassTy, false,
Daniel Dunbarf56f1912008-08-25 08:19:24 +00002379 llvm::GlobalValue::InternalLinkage,
Owen Anderson1c431b32009-07-08 19:05:04 +00002380 Init, Name);
Daniel Dunbarf56f1912008-08-25 08:19:24 +00002381 }
Daniel Dunbar27f9d772008-08-21 04:36:09 +00002382 GV->setSection("__OBJC,__meta_class,regular,no_dead_strip");
Daniel Dunbar58a29122009-03-09 22:18:41 +00002383 GV->setAlignment(4);
Chris Lattnerad64e022009-07-17 23:57:13 +00002384 CGM.AddUsedGlobal(GV);
Daniel Dunbar27f9d772008-08-21 04:36:09 +00002385
2386 return GV;
2387}
2388
Daniel Dunbar6bff2512009-08-03 17:06:42 +00002389llvm::Constant *CGObjCMac::EmitMetaClassRef(const ObjCInterfaceDecl *ID) {
Chris Lattnerd9d22dd2008-11-24 05:29:24 +00002390 std::string Name = "\01L_OBJC_METACLASS_" + ID->getNameAsString();
Daniel Dunbarf56f1912008-08-25 08:19:24 +00002391
Mike Stumpf5408fe2009-05-16 07:57:57 +00002392 // FIXME: Should we look these up somewhere other than the module. Its a bit
2393 // silly since we only generate these while processing an implementation, so
2394 // exactly one pointer would work if know when we entered/exitted an
2395 // implementation block.
Daniel Dunbarf56f1912008-08-25 08:19:24 +00002396
2397 // Check for an existing forward reference.
Fariborz Jahanianb0d27942009-01-07 20:11:22 +00002398 // Previously, metaclass with internal linkage may have been defined.
2399 // pass 'true' as 2nd argument so it is returned.
Daniel Dunbar6bff2512009-08-03 17:06:42 +00002400 if (llvm::GlobalVariable *GV = CGM.getModule().getGlobalVariable(Name,
2401 true)) {
Daniel Dunbarf56f1912008-08-25 08:19:24 +00002402 assert(GV->getType()->getElementType() == ObjCTypes.ClassTy &&
2403 "Forward metaclass reference has incorrect type.");
2404 return GV;
2405 } else {
2406 // Generate as an external reference to keep a consistent
2407 // module. This will be patched up when we emit the metaclass.
Owen Anderson1c431b32009-07-08 19:05:04 +00002408 return new llvm::GlobalVariable(CGM.getModule(), ObjCTypes.ClassTy, false,
Daniel Dunbarf56f1912008-08-25 08:19:24 +00002409 llvm::GlobalValue::ExternalLinkage,
2410 0,
Owen Anderson1c431b32009-07-08 19:05:04 +00002411 Name);
Daniel Dunbarf56f1912008-08-25 08:19:24 +00002412 }
2413}
2414
Fariborz Jahanianb0069ee2009-11-12 20:14:24 +00002415llvm::Value *CGObjCMac::EmitSuperClassRef(const ObjCInterfaceDecl *ID) {
2416 std::string Name = "\01L_OBJC_CLASS_" + ID->getNameAsString();
2417
2418 if (llvm::GlobalVariable *GV = CGM.getModule().getGlobalVariable(Name,
2419 true)) {
2420 assert(GV->getType()->getElementType() == ObjCTypes.ClassTy &&
2421 "Forward class metadata reference has incorrect type.");
2422 return GV;
2423 } else {
2424 return new llvm::GlobalVariable(CGM.getModule(), ObjCTypes.ClassTy, false,
2425 llvm::GlobalValue::ExternalLinkage,
2426 0,
2427 Name);
2428 }
2429}
2430
Daniel Dunbar27f9d772008-08-21 04:36:09 +00002431/*
2432 struct objc_class_ext {
Daniel Dunbar6bff2512009-08-03 17:06:42 +00002433 uint32_t size;
2434 const char *weak_ivar_layout;
2435 struct _objc_property_list *properties;
Daniel Dunbar27f9d772008-08-21 04:36:09 +00002436 };
2437*/
2438llvm::Constant *
2439CGObjCMac::EmitClassExtension(const ObjCImplementationDecl *ID) {
Daniel Dunbar6bff2512009-08-03 17:06:42 +00002440 uint64_t Size =
Duncan Sands9408c452009-05-09 07:08:47 +00002441 CGM.getTargetData().getTypeAllocSize(ObjCTypes.ClassExtensionTy);
Daniel Dunbar27f9d772008-08-21 04:36:09 +00002442
2443 std::vector<llvm::Constant*> Values(3);
Owen Anderson4a28d5d2009-07-24 23:12:58 +00002444 Values[0] = llvm::ConstantInt::get(ObjCTypes.IntTy, Size);
Fariborz Jahanianc71303d2009-04-22 23:00:43 +00002445 Values[1] = BuildIvarLayout(ID, false);
Daniel Dunbar9c29bf52009-10-18 20:48:59 +00002446 Values[2] = EmitPropertyList("\01l_OBJC_$_PROP_LIST_" + ID->getName(),
Fariborz Jahanian5de14dc2009-01-28 22:18:42 +00002447 ID, ID->getClassInterface(), ObjCTypes);
Daniel Dunbar27f9d772008-08-21 04:36:09 +00002448
2449 // Return null if no extension bits are used.
2450 if (Values[1]->isNullValue() && Values[2]->isNullValue())
Owen Andersonc9c88b42009-07-31 20:28:54 +00002451 return llvm::Constant::getNullValue(ObjCTypes.ClassExtensionPtrTy);
Daniel Dunbar27f9d772008-08-21 04:36:09 +00002452
Daniel Dunbar6bff2512009-08-03 17:06:42 +00002453 llvm::Constant *Init =
Owen Anderson08e25242009-07-27 22:29:56 +00002454 llvm::ConstantStruct::get(ObjCTypes.ClassExtensionTy, Values);
Daniel Dunbar9c29bf52009-10-18 20:48:59 +00002455 return CreateMetadataVar("\01L_OBJC_CLASSEXT_" + ID->getName(),
Daniel Dunbar6bff2512009-08-03 17:06:42 +00002456 Init, "__OBJC,__class_ext,regular,no_dead_strip",
Daniel Dunbar0bf21992009-04-15 02:56:18 +00002457 4, true);
Daniel Dunbar27f9d772008-08-21 04:36:09 +00002458}
2459
2460/*
2461 struct objc_ivar {
Daniel Dunbar6bff2512009-08-03 17:06:42 +00002462 char *ivar_name;
2463 char *ivar_type;
2464 int ivar_offset;
Daniel Dunbar27f9d772008-08-21 04:36:09 +00002465 };
2466
2467 struct objc_ivar_list {
Daniel Dunbar6bff2512009-08-03 17:06:42 +00002468 int ivar_count;
2469 struct objc_ivar list[count];
Daniel Dunbar27f9d772008-08-21 04:36:09 +00002470 };
Daniel Dunbar6bff2512009-08-03 17:06:42 +00002471*/
Daniel Dunbar27f9d772008-08-21 04:36:09 +00002472llvm::Constant *CGObjCMac::EmitIvarList(const ObjCImplementationDecl *ID,
Fariborz Jahanian46b86c62009-01-28 19:12:34 +00002473 bool ForClass) {
Daniel Dunbar27f9d772008-08-21 04:36:09 +00002474 std::vector<llvm::Constant*> Ivars, Ivar(3);
2475
2476 // When emitting the root class GCC emits ivar entries for the
2477 // actual class structure. It is not clear if we need to follow this
2478 // behavior; for now lets try and get away with not doing it. If so,
2479 // the cleanest solution would be to make up an ObjCInterfaceDecl
2480 // for the class.
2481 if (ForClass)
Owen Andersonc9c88b42009-07-31 20:28:54 +00002482 return llvm::Constant::getNullValue(ObjCTypes.IvarListPtrTy);
Daniel Dunbar6bff2512009-08-03 17:06:42 +00002483
2484 ObjCInterfaceDecl *OID =
Fariborz Jahanian46b86c62009-01-28 19:12:34 +00002485 const_cast<ObjCInterfaceDecl*>(ID->getClassInterface());
Daniel Dunbar6bff2512009-08-03 17:06:42 +00002486
Daniel Dunbar3e5f0d82009-04-20 06:54:31 +00002487 llvm::SmallVector<ObjCIvarDecl*, 16> OIvars;
Fariborz Jahanian8e6ac1d2009-06-04 01:19:09 +00002488 CGM.getContext().ShallowCollectObjCIvars(OID, OIvars);
Daniel Dunbar6bff2512009-08-03 17:06:42 +00002489
Daniel Dunbar3e5f0d82009-04-20 06:54:31 +00002490 for (unsigned i = 0, e = OIvars.size(); i != e; ++i) {
2491 ObjCIvarDecl *IVD = OIvars[i];
Fariborz Jahanian8e6ac1d2009-06-04 01:19:09 +00002492 // Ignore unnamed bit-fields.
2493 if (!IVD->getDeclName())
Daniel Dunbar6bff2512009-08-03 17:06:42 +00002494 continue;
Daniel Dunbar3fea0c02009-04-22 08:22:17 +00002495 Ivar[0] = GetMethodVarName(IVD->getIdentifier());
2496 Ivar[1] = GetMethodVarType(IVD);
Daniel Dunbar6bff2512009-08-03 17:06:42 +00002497 Ivar[2] = llvm::ConstantInt::get(ObjCTypes.IntTy,
Daniel Dunbar97776872009-04-22 07:32:20 +00002498 ComputeIvarBaseOffset(CGM, OID, IVD));
Owen Anderson08e25242009-07-27 22:29:56 +00002499 Ivars.push_back(llvm::ConstantStruct::get(ObjCTypes.IvarTy, Ivar));
Daniel Dunbar27f9d772008-08-21 04:36:09 +00002500 }
2501
2502 // Return null for empty list.
2503 if (Ivars.empty())
Owen Andersonc9c88b42009-07-31 20:28:54 +00002504 return llvm::Constant::getNullValue(ObjCTypes.IvarListPtrTy);
Daniel Dunbar27f9d772008-08-21 04:36:09 +00002505
2506 std::vector<llvm::Constant*> Values(2);
Owen Anderson4a28d5d2009-07-24 23:12:58 +00002507 Values[0] = llvm::ConstantInt::get(ObjCTypes.IntTy, Ivars.size());
Owen Anderson96e0fc72009-07-29 22:16:19 +00002508 llvm::ArrayType *AT = llvm::ArrayType::get(ObjCTypes.IvarTy,
Daniel Dunbar27f9d772008-08-21 04:36:09 +00002509 Ivars.size());
Owen Anderson7db6d832009-07-28 18:33:04 +00002510 Values[1] = llvm::ConstantArray::get(AT, Ivars);
Nick Lewycky0d36dd22009-09-19 20:00:52 +00002511 llvm::Constant *Init = llvm::ConstantStruct::get(VMContext, Values, false);
Daniel Dunbar27f9d772008-08-21 04:36:09 +00002512
Daniel Dunbar63c5b502009-03-09 21:49:58 +00002513 llvm::GlobalVariable *GV;
2514 if (ForClass)
Daniel Dunbar9c29bf52009-10-18 20:48:59 +00002515 GV = CreateMetadataVar("\01L_OBJC_CLASS_VARIABLES_" + ID->getName(),
Daniel Dunbar6bff2512009-08-03 17:06:42 +00002516 Init, "__OBJC,__class_vars,regular,no_dead_strip",
Daniel Dunbar58a29122009-03-09 22:18:41 +00002517 4, true);
Daniel Dunbar63c5b502009-03-09 21:49:58 +00002518 else
Daniel Dunbar9c29bf52009-10-18 20:48:59 +00002519 GV = CreateMetadataVar("\01L_OBJC_INSTANCE_VARIABLES_" + ID->getName(),
Daniel Dunbar63c5b502009-03-09 21:49:58 +00002520 Init, "__OBJC,__instance_vars,regular,no_dead_strip",
Daniel Dunbar0bf21992009-04-15 02:56:18 +00002521 4, true);
Owen Anderson3c4972d2009-07-29 18:54:39 +00002522 return llvm::ConstantExpr::getBitCast(GV, ObjCTypes.IvarListPtrTy);
Daniel Dunbar27f9d772008-08-21 04:36:09 +00002523}
2524
2525/*
2526 struct objc_method {
Daniel Dunbar6bff2512009-08-03 17:06:42 +00002527 SEL method_name;
2528 char *method_types;
2529 void *method;
Daniel Dunbar27f9d772008-08-21 04:36:09 +00002530 };
Daniel Dunbar6bff2512009-08-03 17:06:42 +00002531
Daniel Dunbar27f9d772008-08-21 04:36:09 +00002532 struct objc_method_list {
Daniel Dunbar6bff2512009-08-03 17:06:42 +00002533 struct objc_method_list *obsolete;
2534 int count;
2535 struct objc_method methods_list[count];
Daniel Dunbar27f9d772008-08-21 04:36:09 +00002536 };
2537*/
Daniel Dunbarc45ef602008-08-26 21:51:14 +00002538
2539/// GetMethodConstant - Return a struct objc_method constant for the
2540/// given method if it has been defined. The result is null if the
2541/// method has not been defined. The return value has type MethodPtrTy.
Daniel Dunbarae226fa2008-08-27 02:31:56 +00002542llvm::Constant *CGObjCMac::GetMethodConstant(const ObjCMethodDecl *MD) {
Argyrios Kyrtzidis9d50c062010-08-09 10:54:20 +00002543 llvm::Function *Fn = GetMethodDefinition(MD);
Daniel Dunbarc45ef602008-08-26 21:51:14 +00002544 if (!Fn)
2545 return 0;
Daniel Dunbar6bff2512009-08-03 17:06:42 +00002546
Daniel Dunbarc45ef602008-08-26 21:51:14 +00002547 std::vector<llvm::Constant*> Method(3);
Daniel Dunbar6bff2512009-08-03 17:06:42 +00002548 Method[0] =
Owen Anderson3c4972d2009-07-29 18:54:39 +00002549 llvm::ConstantExpr::getBitCast(GetMethodVarName(MD->getSelector()),
Daniel Dunbarc45ef602008-08-26 21:51:14 +00002550 ObjCTypes.SelectorPtrTy);
2551 Method[1] = GetMethodVarType(MD);
Owen Anderson3c4972d2009-07-29 18:54:39 +00002552 Method[2] = llvm::ConstantExpr::getBitCast(Fn, ObjCTypes.Int8PtrTy);
Owen Anderson08e25242009-07-27 22:29:56 +00002553 return llvm::ConstantStruct::get(ObjCTypes.MethodTy, Method);
Daniel Dunbarc45ef602008-08-26 21:51:14 +00002554}
2555
Daniel Dunbar9c29bf52009-10-18 20:48:59 +00002556llvm::Constant *CGObjCMac::EmitMethodList(llvm::Twine Name,
Daniel Dunbar86e253a2008-08-22 20:34:54 +00002557 const char *Section,
Daniel Dunbarae226fa2008-08-27 02:31:56 +00002558 const ConstantVector &Methods) {
Daniel Dunbar27f9d772008-08-21 04:36:09 +00002559 // Return null for empty list.
2560 if (Methods.empty())
Owen Andersonc9c88b42009-07-31 20:28:54 +00002561 return llvm::Constant::getNullValue(ObjCTypes.MethodListPtrTy);
Daniel Dunbar27f9d772008-08-21 04:36:09 +00002562
2563 std::vector<llvm::Constant*> Values(3);
Owen Andersonc9c88b42009-07-31 20:28:54 +00002564 Values[0] = llvm::Constant::getNullValue(ObjCTypes.Int8PtrTy);
Owen Anderson4a28d5d2009-07-24 23:12:58 +00002565 Values[1] = llvm::ConstantInt::get(ObjCTypes.IntTy, Methods.size());
Owen Anderson96e0fc72009-07-29 22:16:19 +00002566 llvm::ArrayType *AT = llvm::ArrayType::get(ObjCTypes.MethodTy,
Daniel Dunbar27f9d772008-08-21 04:36:09 +00002567 Methods.size());
Owen Anderson7db6d832009-07-28 18:33:04 +00002568 Values[2] = llvm::ConstantArray::get(AT, Methods);
Nick Lewycky0d36dd22009-09-19 20:00:52 +00002569 llvm::Constant *Init = llvm::ConstantStruct::get(VMContext, Values, false);
Daniel Dunbar27f9d772008-08-21 04:36:09 +00002570
Daniel Dunbar0bf21992009-04-15 02:56:18 +00002571 llvm::GlobalVariable *GV = CreateMetadataVar(Name, Init, Section, 4, true);
Owen Anderson3c4972d2009-07-29 18:54:39 +00002572 return llvm::ConstantExpr::getBitCast(GV,
Daniel Dunbar27f9d772008-08-21 04:36:09 +00002573 ObjCTypes.MethodListPtrTy);
Daniel Dunbarb7ec2462008-08-16 03:19:19 +00002574}
2575
Fariborz Jahanian493dab72009-01-26 21:38:32 +00002576llvm::Function *CGObjCCommonMac::GenerateMethod(const ObjCMethodDecl *OMD,
Daniel Dunbar6bff2512009-08-03 17:06:42 +00002577 const ObjCContainerDecl *CD) {
Daniel Dunbarc575ce72009-10-19 01:21:19 +00002578 llvm::SmallString<256> Name;
Fariborz Jahanian679a5022009-01-10 21:06:09 +00002579 GetNameForMethod(OMD, CD, Name);
Daniel Dunbarb7ec2462008-08-16 03:19:19 +00002580
Daniel Dunbar541b63b2009-02-02 23:23:47 +00002581 CodeGenTypes &Types = CGM.getTypes();
Daniel Dunbar45c25ba2008-09-10 04:01:49 +00002582 const llvm::FunctionType *MethodTy =
Daniel Dunbar541b63b2009-02-02 23:23:47 +00002583 Types.GetFunctionType(Types.getFunctionInfo(OMD), OMD->isVariadic());
Daniel Dunbar6bff2512009-08-03 17:06:42 +00002584 llvm::Function *Method =
Daniel Dunbar45c25ba2008-09-10 04:01:49 +00002585 llvm::Function::Create(MethodTy,
Daniel Dunbarb7ec2462008-08-16 03:19:19 +00002586 llvm::GlobalValue::InternalLinkage,
Daniel Dunbarc575ce72009-10-19 01:21:19 +00002587 Name.str(),
Daniel Dunbarb7ec2462008-08-16 03:19:19 +00002588 &CGM.getModule());
Daniel Dunbarc45ef602008-08-26 21:51:14 +00002589 MethodDefinitions.insert(std::make_pair(OMD, Method));
Daniel Dunbarb7ec2462008-08-16 03:19:19 +00002590
Daniel Dunbarb7ec2462008-08-16 03:19:19 +00002591 return Method;
Daniel Dunbarc17a4d32008-08-11 02:45:11 +00002592}
2593
Daniel Dunbarfd65d372009-03-09 20:09:19 +00002594llvm::GlobalVariable *
Daniel Dunbar9c29bf52009-10-18 20:48:59 +00002595CGObjCCommonMac::CreateMetadataVar(llvm::Twine Name,
Daniel Dunbarfd65d372009-03-09 20:09:19 +00002596 llvm::Constant *Init,
2597 const char *Section,
Daniel Dunbar35bd7632009-03-09 20:50:13 +00002598 unsigned Align,
2599 bool AddToUsed) {
Daniel Dunbarfd65d372009-03-09 20:09:19 +00002600 const llvm::Type *Ty = Init->getType();
Daniel Dunbar6bff2512009-08-03 17:06:42 +00002601 llvm::GlobalVariable *GV =
Owen Anderson1c431b32009-07-08 19:05:04 +00002602 new llvm::GlobalVariable(CGM.getModule(), Ty, false,
Chris Lattnerad64e022009-07-17 23:57:13 +00002603 llvm::GlobalValue::InternalLinkage, Init, Name);
Daniel Dunbarfd65d372009-03-09 20:09:19 +00002604 if (Section)
2605 GV->setSection(Section);
Daniel Dunbar35bd7632009-03-09 20:50:13 +00002606 if (Align)
2607 GV->setAlignment(Align);
2608 if (AddToUsed)
Chris Lattnerad64e022009-07-17 23:57:13 +00002609 CGM.AddUsedGlobal(GV);
Daniel Dunbarfd65d372009-03-09 20:09:19 +00002610 return GV;
2611}
2612
Daniel Dunbar6bff2512009-08-03 17:06:42 +00002613llvm::Function *CGObjCMac::ModuleInitFunction() {
Daniel Dunbarf77ac862008-08-11 21:35:06 +00002614 // Abuse this interface function as a place to finalize.
2615 FinishModule();
Daniel Dunbarc17a4d32008-08-11 02:45:11 +00002616 return NULL;
2617}
2618
Chris Lattner74391b42009-03-22 21:03:39 +00002619llvm::Constant *CGObjCMac::GetPropertyGetFunction() {
Chris Lattner72db6c32009-04-22 02:44:54 +00002620 return ObjCTypes.getGetPropertyFn();
Daniel Dunbar49f66022008-09-24 03:38:44 +00002621}
2622
Chris Lattner74391b42009-03-22 21:03:39 +00002623llvm::Constant *CGObjCMac::GetPropertySetFunction() {
Chris Lattner72db6c32009-04-22 02:44:54 +00002624 return ObjCTypes.getSetPropertyFn();
Daniel Dunbar49f66022008-09-24 03:38:44 +00002625}
2626
Fariborz Jahanian6cc59062010-04-12 18:18:10 +00002627llvm::Constant *CGObjCMac::GetCopyStructFunction() {
2628 return ObjCTypes.getCopyStructFn();
2629}
2630
Chris Lattner74391b42009-03-22 21:03:39 +00002631llvm::Constant *CGObjCMac::EnumerationMutationFunction() {
Chris Lattner72db6c32009-04-22 02:44:54 +00002632 return ObjCTypes.getEnumerationMutationFn();
Anders Carlsson2abd89c2008-08-31 04:05:03 +00002633}
2634
John McCallf1549f62010-07-06 01:34:17 +00002635void CGObjCMac::EmitTryStmt(CodeGenFunction &CGF, const ObjCAtTryStmt &S) {
2636 return EmitTryOrSynchronizedStmt(CGF, S);
2637}
2638
2639void CGObjCMac::EmitSynchronizedStmt(CodeGenFunction &CGF,
2640 const ObjCAtSynchronizedStmt &S) {
2641 return EmitTryOrSynchronizedStmt(CGF, S);
2642}
2643
John McCallcc505292010-07-21 06:59:36 +00002644namespace {
John McCall1f0fca52010-07-21 07:22:38 +00002645 struct PerformFragileFinally : EHScopeStack::Cleanup {
John McCallcc505292010-07-21 06:59:36 +00002646 const Stmt &S;
John McCall0b251722010-08-04 05:59:32 +00002647 llvm::Value *SyncArgSlot;
John McCallcc505292010-07-21 06:59:36 +00002648 llvm::Value *CallTryExitVar;
2649 llvm::Value *ExceptionData;
2650 ObjCTypesHelper &ObjCTypes;
2651 PerformFragileFinally(const Stmt *S,
John McCall0b251722010-08-04 05:59:32 +00002652 llvm::Value *SyncArgSlot,
John McCallcc505292010-07-21 06:59:36 +00002653 llvm::Value *CallTryExitVar,
2654 llvm::Value *ExceptionData,
2655 ObjCTypesHelper *ObjCTypes)
John McCall0b251722010-08-04 05:59:32 +00002656 : S(*S), SyncArgSlot(SyncArgSlot), CallTryExitVar(CallTryExitVar),
John McCallcc505292010-07-21 06:59:36 +00002657 ExceptionData(ExceptionData), ObjCTypes(*ObjCTypes) {}
2658
2659 void Emit(CodeGenFunction &CGF, bool IsForEH) {
2660 // Check whether we need to call objc_exception_try_exit.
2661 // In optimized code, this branch will always be folded.
2662 llvm::BasicBlock *FinallyCallExit =
2663 CGF.createBasicBlock("finally.call_exit");
2664 llvm::BasicBlock *FinallyNoCallExit =
2665 CGF.createBasicBlock("finally.no_call_exit");
2666 CGF.Builder.CreateCondBr(CGF.Builder.CreateLoad(CallTryExitVar),
2667 FinallyCallExit, FinallyNoCallExit);
2668
2669 CGF.EmitBlock(FinallyCallExit);
2670 CGF.Builder.CreateCall(ObjCTypes.getExceptionTryExitFn(), ExceptionData)
2671 ->setDoesNotThrow();
2672
2673 CGF.EmitBlock(FinallyNoCallExit);
2674
2675 if (isa<ObjCAtTryStmt>(S)) {
2676 if (const ObjCAtFinallyStmt* FinallyStmt =
2677 cast<ObjCAtTryStmt>(S).getFinallyStmt())
2678 CGF.EmitStmt(FinallyStmt->getFinallyBody());
2679
2680 // Currently, the end of the cleanup must always exist.
2681 CGF.EnsureInsertPoint();
2682 } else {
2683 // Emit objc_sync_exit(expr); as finally's sole statement for
2684 // @synchronized.
John McCall0b251722010-08-04 05:59:32 +00002685 llvm::Value *SyncArg = CGF.Builder.CreateLoad(SyncArgSlot);
John McCallcc505292010-07-21 06:59:36 +00002686 CGF.Builder.CreateCall(ObjCTypes.getSyncExitFn(), SyncArg)
2687 ->setDoesNotThrow();
2688 }
2689 }
2690 };
John McCall87bb5822010-07-31 23:20:56 +00002691
2692 class FragileHazards {
2693 CodeGenFunction &CGF;
2694 llvm::SmallVector<llvm::Value*, 20> Locals;
2695 llvm::DenseSet<llvm::BasicBlock*> BlocksBeforeTry;
2696
2697 llvm::InlineAsm *ReadHazard;
2698 llvm::InlineAsm *WriteHazard;
2699
2700 llvm::FunctionType *GetAsmFnType();
2701
2702 void collectLocals();
2703 void emitReadHazard(CGBuilderTy &Builder);
2704
2705 public:
2706 FragileHazards(CodeGenFunction &CGF);
John McCall0b251722010-08-04 05:59:32 +00002707
John McCall87bb5822010-07-31 23:20:56 +00002708 void emitWriteHazard();
John McCall0b251722010-08-04 05:59:32 +00002709 void emitHazardsInNewBlocks();
John McCall87bb5822010-07-31 23:20:56 +00002710 };
2711}
2712
2713/// Create the fragile-ABI read and write hazards based on the current
2714/// state of the function, which is presumed to be immediately prior
2715/// to a @try block. These hazards are used to maintain correct
2716/// semantics in the face of optimization and the fragile ABI's
2717/// cavalier use of setjmp/longjmp.
2718FragileHazards::FragileHazards(CodeGenFunction &CGF) : CGF(CGF) {
2719 collectLocals();
2720
2721 if (Locals.empty()) return;
2722
2723 // Collect all the blocks in the function.
2724 for (llvm::Function::iterator
2725 I = CGF.CurFn->begin(), E = CGF.CurFn->end(); I != E; ++I)
2726 BlocksBeforeTry.insert(&*I);
2727
2728 llvm::FunctionType *AsmFnTy = GetAsmFnType();
2729
2730 // Create a read hazard for the allocas. This inhibits dead-store
2731 // optimizations and forces the values to memory. This hazard is
2732 // inserted before any 'throwing' calls in the protected scope to
2733 // reflect the possibility that the variables might be read from the
2734 // catch block if the call throws.
2735 {
2736 std::string Constraint;
2737 for (unsigned I = 0, E = Locals.size(); I != E; ++I) {
2738 if (I) Constraint += ',';
2739 Constraint += "*m";
2740 }
2741
2742 ReadHazard = llvm::InlineAsm::get(AsmFnTy, "", Constraint, true, false);
2743 }
2744
2745 // Create a write hazard for the allocas. This inhibits folding
2746 // loads across the hazard. This hazard is inserted at the
2747 // beginning of the catch path to reflect the possibility that the
2748 // variables might have been written within the protected scope.
2749 {
2750 std::string Constraint;
2751 for (unsigned I = 0, E = Locals.size(); I != E; ++I) {
2752 if (I) Constraint += ',';
2753 Constraint += "=*m";
2754 }
2755
2756 WriteHazard = llvm::InlineAsm::get(AsmFnTy, "", Constraint, true, false);
2757 }
2758}
2759
2760/// Emit a write hazard at the current location.
2761void FragileHazards::emitWriteHazard() {
2762 if (Locals.empty()) return;
2763
2764 CGF.Builder.CreateCall(WriteHazard, Locals.begin(), Locals.end())
2765 ->setDoesNotThrow();
2766}
2767
John McCall87bb5822010-07-31 23:20:56 +00002768void FragileHazards::emitReadHazard(CGBuilderTy &Builder) {
2769 assert(!Locals.empty());
2770 Builder.CreateCall(ReadHazard, Locals.begin(), Locals.end())
2771 ->setDoesNotThrow();
2772}
2773
2774/// Emit read hazards in all the protected blocks, i.e. all the blocks
2775/// which have been inserted since the beginning of the try.
John McCall0b251722010-08-04 05:59:32 +00002776void FragileHazards::emitHazardsInNewBlocks() {
John McCall87bb5822010-07-31 23:20:56 +00002777 if (Locals.empty()) return;
2778
2779 CGBuilderTy Builder(CGF.getLLVMContext());
2780
2781 // Iterate through all blocks, skipping those prior to the try.
2782 for (llvm::Function::iterator
2783 FI = CGF.CurFn->begin(), FE = CGF.CurFn->end(); FI != FE; ++FI) {
2784 llvm::BasicBlock &BB = *FI;
2785 if (BlocksBeforeTry.count(&BB)) continue;
2786
2787 // Walk through all the calls in the block.
2788 for (llvm::BasicBlock::iterator
2789 BI = BB.begin(), BE = BB.end(); BI != BE; ++BI) {
2790 llvm::Instruction &I = *BI;
2791
2792 // Ignore instructions that aren't non-intrinsic calls.
2793 // These are the only calls that can possibly call longjmp.
2794 if (!isa<llvm::CallInst>(I) && !isa<llvm::InvokeInst>(I)) continue;
2795 if (isa<llvm::IntrinsicInst>(I))
2796 continue;
2797
2798 // Ignore call sites marked nounwind. This may be questionable,
2799 // since 'nounwind' doesn't necessarily mean 'does not call longjmp'.
2800 llvm::CallSite CS(&I);
2801 if (CS.doesNotThrow()) continue;
2802
John McCall0b251722010-08-04 05:59:32 +00002803 // Insert a read hazard before the call. This will ensure that
2804 // any writes to the locals are performed before making the
2805 // call. If the call throws, then this is sufficient to
2806 // guarantee correctness as long as it doesn't also write to any
2807 // locals.
John McCall87bb5822010-07-31 23:20:56 +00002808 Builder.SetInsertPoint(&BB, BI);
2809 emitReadHazard(Builder);
2810 }
2811 }
2812}
2813
2814static void addIfPresent(llvm::DenseSet<llvm::Value*> &S, llvm::Value *V) {
2815 if (V) S.insert(V);
2816}
2817
2818void FragileHazards::collectLocals() {
2819 // Compute a set of allocas to ignore.
2820 llvm::DenseSet<llvm::Value*> AllocasToIgnore;
2821 addIfPresent(AllocasToIgnore, CGF.ReturnValue);
2822 addIfPresent(AllocasToIgnore, CGF.NormalCleanupDest);
2823 addIfPresent(AllocasToIgnore, CGF.EHCleanupDest);
2824
2825 // Collect all the allocas currently in the function. This is
2826 // probably way too aggressive.
2827 llvm::BasicBlock &Entry = CGF.CurFn->getEntryBlock();
2828 for (llvm::BasicBlock::iterator
2829 I = Entry.begin(), E = Entry.end(); I != E; ++I)
2830 if (isa<llvm::AllocaInst>(*I) && !AllocasToIgnore.count(&*I))
2831 Locals.push_back(&*I);
2832}
2833
2834llvm::FunctionType *FragileHazards::GetAsmFnType() {
2835 std::vector<const llvm::Type *> Tys(Locals.size());
2836 for (unsigned I = 0, E = Locals.size(); I != E; ++I)
2837 Tys[I] = Locals[I]->getType();
2838 return llvm::FunctionType::get(CGF.Builder.getVoidTy(), Tys, false);
John McCallcc505292010-07-21 06:59:36 +00002839}
2840
Daniel Dunbar6bff2512009-08-03 17:06:42 +00002841/*
Daniel Dunbar18ccc772008-09-28 01:03:14 +00002842
Daniel Dunbar6bff2512009-08-03 17:06:42 +00002843 Objective-C setjmp-longjmp (sjlj) Exception Handling
2844 --
Daniel Dunbar18ccc772008-09-28 01:03:14 +00002845
John McCallf1549f62010-07-06 01:34:17 +00002846 A catch buffer is a setjmp buffer plus:
2847 - a pointer to the exception that was caught
2848 - a pointer to the previous exception data buffer
2849 - two pointers of reserved storage
2850 Therefore catch buffers form a stack, with a pointer to the top
2851 of the stack kept in thread-local storage.
2852
2853 objc_exception_try_enter pushes a catch buffer onto the EH stack.
2854 objc_exception_try_exit pops the given catch buffer, which is
2855 required to be the top of the EH stack.
2856 objc_exception_throw pops the top of the EH stack, writes the
2857 thrown exception into the appropriate field, and longjmps
2858 to the setjmp buffer. It crashes the process (with a printf
2859 and an abort()) if there are no catch buffers on the stack.
2860 objc_exception_extract just reads the exception pointer out of the
2861 catch buffer.
2862
2863 There's no reason an implementation couldn't use a light-weight
2864 setjmp here --- something like __builtin_setjmp, but API-compatible
2865 with the heavyweight setjmp. This will be more important if we ever
2866 want to implement correct ObjC/C++ exception interactions for the
2867 fragile ABI.
2868
2869 Note that for this use of setjmp/longjmp to be correct, we may need
2870 to mark some local variables volatile: if a non-volatile local
2871 variable is modified between the setjmp and the longjmp, it has
2872 indeterminate value. For the purposes of LLVM IR, it may be
2873 sufficient to make loads and stores within the @try (to variables
2874 declared outside the @try) volatile. This is necessary for
2875 optimized correctness, but is not currently being done; this is
2876 being tracked as rdar://problem/8160285
2877
Daniel Dunbar6bff2512009-08-03 17:06:42 +00002878 The basic framework for a @try-catch-finally is as follows:
2879 {
Daniel Dunbar18ccc772008-09-28 01:03:14 +00002880 objc_exception_data d;
2881 id _rethrow = null;
Anders Carlsson190d00e2009-02-07 21:26:04 +00002882 bool _call_try_exit = true;
Daniel Dunbar6bff2512009-08-03 17:06:42 +00002883
Daniel Dunbar18ccc772008-09-28 01:03:14 +00002884 objc_exception_try_enter(&d);
2885 if (!setjmp(d.jmp_buf)) {
Daniel Dunbar6bff2512009-08-03 17:06:42 +00002886 ... try body ...
Daniel Dunbar18ccc772008-09-28 01:03:14 +00002887 } else {
Daniel Dunbar6bff2512009-08-03 17:06:42 +00002888 // exception path
2889 id _caught = objc_exception_extract(&d);
2890
2891 // enter new try scope for handlers
2892 if (!setjmp(d.jmp_buf)) {
2893 ... match exception and execute catch blocks ...
2894
2895 // fell off end, rethrow.
2896 _rethrow = _caught;
2897 ... jump-through-finally to finally_rethrow ...
2898 } else {
2899 // exception in catch block
2900 _rethrow = objc_exception_extract(&d);
2901 _call_try_exit = false;
2902 ... jump-through-finally to finally_rethrow ...
2903 }
Daniel Dunbar18ccc772008-09-28 01:03:14 +00002904 }
Daniel Dunbar898d5082008-09-30 01:06:03 +00002905 ... jump-through-finally to finally_end ...
Daniel Dunbar18ccc772008-09-28 01:03:14 +00002906
Daniel Dunbar6bff2512009-08-03 17:06:42 +00002907 finally:
Anders Carlsson190d00e2009-02-07 21:26:04 +00002908 if (_call_try_exit)
Daniel Dunbar6bff2512009-08-03 17:06:42 +00002909 objc_exception_try_exit(&d);
Anders Carlsson190d00e2009-02-07 21:26:04 +00002910
Daniel Dunbar18ccc772008-09-28 01:03:14 +00002911 ... finally block ....
Daniel Dunbar898d5082008-09-30 01:06:03 +00002912 ... dispatch to finally destination ...
2913
Daniel Dunbar6bff2512009-08-03 17:06:42 +00002914 finally_rethrow:
Daniel Dunbar898d5082008-09-30 01:06:03 +00002915 objc_exception_throw(_rethrow);
2916
Daniel Dunbar6bff2512009-08-03 17:06:42 +00002917 finally_end:
2918 }
Daniel Dunbar18ccc772008-09-28 01:03:14 +00002919
Daniel Dunbar6bff2512009-08-03 17:06:42 +00002920 This framework differs slightly from the one gcc uses, in that gcc
2921 uses _rethrow to determine if objc_exception_try_exit should be called
2922 and if the object should be rethrown. This breaks in the face of
2923 throwing nil and introduces unnecessary branches.
Daniel Dunbar18ccc772008-09-28 01:03:14 +00002924
Daniel Dunbar6bff2512009-08-03 17:06:42 +00002925 We specialize this framework for a few particular circumstances:
Daniel Dunbar18ccc772008-09-28 01:03:14 +00002926
Daniel Dunbar6bff2512009-08-03 17:06:42 +00002927 - If there are no catch blocks, then we avoid emitting the second
2928 exception handling context.
Daniel Dunbar18ccc772008-09-28 01:03:14 +00002929
Daniel Dunbar6bff2512009-08-03 17:06:42 +00002930 - If there is a catch-all catch block (i.e. @catch(...) or @catch(id
2931 e)) we avoid emitting the code to rethrow an uncaught exception.
Daniel Dunbar18ccc772008-09-28 01:03:14 +00002932
Daniel Dunbar6bff2512009-08-03 17:06:42 +00002933 - FIXME: If there is no @finally block we can do a few more
2934 simplifications.
Daniel Dunbar18ccc772008-09-28 01:03:14 +00002935
Daniel Dunbar6bff2512009-08-03 17:06:42 +00002936 Rethrows and Jumps-Through-Finally
2937 --
Daniel Dunbar18ccc772008-09-28 01:03:14 +00002938
John McCallf1549f62010-07-06 01:34:17 +00002939 '@throw;' is supported by pushing the currently-caught exception
2940 onto ObjCEHStack while the @catch blocks are emitted.
Daniel Dunbar18ccc772008-09-28 01:03:14 +00002941
John McCallf1549f62010-07-06 01:34:17 +00002942 Branches through the @finally block are handled with an ordinary
2943 normal cleanup. We do not register an EH cleanup; fragile-ABI ObjC
2944 exceptions are not compatible with C++ exceptions, and this is
2945 hardly the only place where this will go wrong.
Daniel Dunbar898d5082008-09-30 01:06:03 +00002946
John McCallf1549f62010-07-06 01:34:17 +00002947 @synchronized(expr) { stmt; } is emitted as if it were:
2948 id synch_value = expr;
2949 objc_sync_enter(synch_value);
2950 @try { stmt; } @finally { objc_sync_exit(synch_value); }
Daniel Dunbar18ccc772008-09-28 01:03:14 +00002951*/
2952
Fariborz Jahanianbd71be42008-11-21 00:49:24 +00002953void CGObjCMac::EmitTryOrSynchronizedStmt(CodeGen::CodeGenFunction &CGF,
2954 const Stmt &S) {
2955 bool isTry = isa<ObjCAtTryStmt>(S);
John McCallf1549f62010-07-06 01:34:17 +00002956
2957 // A destination for the fall-through edges of the catch handlers to
2958 // jump to.
2959 CodeGenFunction::JumpDest FinallyEnd =
2960 CGF.getJumpDestInCurrentScope("finally.end");
2961
2962 // A destination for the rethrow edge of the catch handlers to jump
2963 // to.
2964 CodeGenFunction::JumpDest FinallyRethrow =
2965 CGF.getJumpDestInCurrentScope("finally.rethrow");
Daniel Dunbar6bff2512009-08-03 17:06:42 +00002966
Daniel Dunbar1c566672009-02-24 01:43:46 +00002967 // For @synchronized, call objc_sync_enter(sync.expr). The
2968 // evaluation of the expression must occur before we enter the
John McCall0b251722010-08-04 05:59:32 +00002969 // @synchronized. We can't avoid a temp here because we need the
2970 // value to be preserved. If the backend ever does liveness
2971 // correctly after setjmp, this will be unnecessary.
2972 llvm::Value *SyncArgSlot = 0;
Daniel Dunbar1c566672009-02-24 01:43:46 +00002973 if (!isTry) {
John McCall0b251722010-08-04 05:59:32 +00002974 llvm::Value *SyncArg =
Daniel Dunbar1c566672009-02-24 01:43:46 +00002975 CGF.EmitScalarExpr(cast<ObjCAtSynchronizedStmt>(S).getSynchExpr());
2976 SyncArg = CGF.Builder.CreateBitCast(SyncArg, ObjCTypes.ObjectPtrTy);
John McCallf1549f62010-07-06 01:34:17 +00002977 CGF.Builder.CreateCall(ObjCTypes.getSyncEnterFn(), SyncArg)
2978 ->setDoesNotThrow();
John McCall0b251722010-08-04 05:59:32 +00002979
2980 SyncArgSlot = CGF.CreateTempAlloca(SyncArg->getType(), "sync.arg");
2981 CGF.Builder.CreateStore(SyncArg, SyncArgSlot);
Daniel Dunbar1c566672009-02-24 01:43:46 +00002982 }
Daniel Dunbar898d5082008-09-30 01:06:03 +00002983
John McCall0b251722010-08-04 05:59:32 +00002984 // Allocate memory for the setjmp buffer. This needs to be kept
2985 // live throughout the try and catch blocks.
2986 llvm::Value *ExceptionData = CGF.CreateTempAlloca(ObjCTypes.ExceptionDataTy,
2987 "exceptiondata.ptr");
2988
John McCall87bb5822010-07-31 23:20:56 +00002989 // Create the fragile hazards. Note that this will not capture any
2990 // of the allocas required for exception processing, but will
2991 // capture the current basic block (which extends all the way to the
2992 // setjmp call) as "before the @try".
2993 FragileHazards Hazards(CGF);
2994
John McCallf1549f62010-07-06 01:34:17 +00002995 // Create a flag indicating whether the cleanup needs to call
2996 // objc_exception_try_exit. This is true except when
2997 // - no catches match and we're branching through the cleanup
2998 // just to rethrow the exception, or
2999 // - a catch matched and we're falling out of the catch handler.
John McCall0b251722010-08-04 05:59:32 +00003000 // The setjmp-safety rule here is that we should always store to this
3001 // variable in a place that dominates the branch through the cleanup
3002 // without passing through any setjmps.
John McCallf1549f62010-07-06 01:34:17 +00003003 llvm::Value *CallTryExitVar = CGF.CreateTempAlloca(CGF.Builder.getInt1Ty(),
Anders Carlsson190d00e2009-02-07 21:26:04 +00003004 "_call_try_exit");
Daniel Dunbar6bff2512009-08-03 17:06:42 +00003005
John McCallf1549f62010-07-06 01:34:17 +00003006 // Push a normal cleanup to leave the try scope.
John McCall1f0fca52010-07-21 07:22:38 +00003007 CGF.EHStack.pushCleanup<PerformFragileFinally>(NormalCleanup, &S,
John McCall0b251722010-08-04 05:59:32 +00003008 SyncArgSlot,
John McCall1f0fca52010-07-21 07:22:38 +00003009 CallTryExitVar,
3010 ExceptionData,
3011 &ObjCTypes);
John McCallf1549f62010-07-06 01:34:17 +00003012
3013 // Enter a try block:
3014 // - Call objc_exception_try_enter to push ExceptionData on top of
3015 // the EH stack.
3016 CGF.Builder.CreateCall(ObjCTypes.getExceptionTryEnterFn(), ExceptionData)
3017 ->setDoesNotThrow();
3018
3019 // - Call setjmp on the exception data buffer.
3020 llvm::Constant *Zero = llvm::ConstantInt::get(CGF.Builder.getInt32Ty(), 0);
3021 llvm::Value *GEPIndexes[] = { Zero, Zero, Zero };
3022 llvm::Value *SetJmpBuffer =
3023 CGF.Builder.CreateGEP(ExceptionData, GEPIndexes, GEPIndexes+3, "setjmp_buffer");
3024 llvm::CallInst *SetJmpResult =
3025 CGF.Builder.CreateCall(ObjCTypes.getSetJmpFn(), SetJmpBuffer, "setjmp_result");
3026 SetJmpResult->setDoesNotThrow();
3027
3028 // If setjmp returned 0, enter the protected block; otherwise,
3029 // branch to the handler.
Daniel Dunbar55e87422008-11-11 02:29:29 +00003030 llvm::BasicBlock *TryBlock = CGF.createBasicBlock("try");
3031 llvm::BasicBlock *TryHandler = CGF.createBasicBlock("try.handler");
John McCallf1549f62010-07-06 01:34:17 +00003032 llvm::Value *DidCatch =
3033 CGF.Builder.CreateIsNull(SetJmpResult, "did_catch_exception");
3034 CGF.Builder.CreateCondBr(DidCatch, TryBlock, TryHandler);
Anders Carlsson80f25672008-09-09 17:59:25 +00003035
John McCallf1549f62010-07-06 01:34:17 +00003036 // Emit the protected block.
Anders Carlsson80f25672008-09-09 17:59:25 +00003037 CGF.EmitBlock(TryBlock);
John McCall0b251722010-08-04 05:59:32 +00003038 CGF.Builder.CreateStore(CGF.Builder.getTrue(), CallTryExitVar);
Daniel Dunbar6bff2512009-08-03 17:06:42 +00003039 CGF.EmitStmt(isTry ? cast<ObjCAtTryStmt>(S).getTryBody()
John McCallf1549f62010-07-06 01:34:17 +00003040 : cast<ObjCAtSynchronizedStmt>(S).getSynchBody());
John McCall0b251722010-08-04 05:59:32 +00003041
3042 CGBuilderTy::InsertPoint TryFallthroughIP = CGF.Builder.saveAndClearIP();
Daniel Dunbar6bff2512009-08-03 17:06:42 +00003043
John McCallf1549f62010-07-06 01:34:17 +00003044 // Emit the exception handler block.
Daniel Dunbare4b5ee02008-09-27 23:30:04 +00003045 CGF.EmitBlock(TryHandler);
Daniel Dunbar55e40722008-09-27 07:03:52 +00003046
John McCall87bb5822010-07-31 23:20:56 +00003047 // Don't optimize loads of the in-scope locals across this point.
3048 Hazards.emitWriteHazard();
3049
John McCallf1549f62010-07-06 01:34:17 +00003050 // For a @synchronized (or a @try with no catches), just branch
3051 // through the cleanup to the rethrow block.
3052 if (!isTry || !cast<ObjCAtTryStmt>(S).getNumCatchStmts()) {
3053 // Tell the cleanup not to re-pop the exit.
John McCall0b251722010-08-04 05:59:32 +00003054 CGF.Builder.CreateStore(CGF.Builder.getFalse(), CallTryExitVar);
Anders Carlssonf3a79a92009-02-09 20:38:58 +00003055 CGF.EmitBranchThroughCleanup(FinallyRethrow);
John McCallf1549f62010-07-06 01:34:17 +00003056
3057 // Otherwise, we have to match against the caught exceptions.
3058 } else {
John McCall0b251722010-08-04 05:59:32 +00003059 // Retrieve the exception object. We may emit multiple blocks but
3060 // nothing can cross this so the value is already in SSA form.
3061 llvm::CallInst *Caught =
3062 CGF.Builder.CreateCall(ObjCTypes.getExceptionExtractFn(),
3063 ExceptionData, "caught");
3064 Caught->setDoesNotThrow();
3065
John McCallf1549f62010-07-06 01:34:17 +00003066 // Push the exception to rethrow onto the EH value stack for the
3067 // benefit of any @throws in the handlers.
3068 CGF.ObjCEHValueStack.push_back(Caught);
3069
Douglas Gregor8f5e3dd2010-04-23 22:50:49 +00003070 const ObjCAtTryStmt* AtTryStmt = cast<ObjCAtTryStmt>(&S);
Daniel Dunbar6bff2512009-08-03 17:06:42 +00003071
John McCall0b251722010-08-04 05:59:32 +00003072 bool HasFinally = (AtTryStmt->getFinallyStmt() != 0);
John McCallf1549f62010-07-06 01:34:17 +00003073
John McCall0b251722010-08-04 05:59:32 +00003074 llvm::BasicBlock *CatchBlock = 0;
3075 llvm::BasicBlock *CatchHandler = 0;
3076 if (HasFinally) {
3077 // Enter a new exception try block (in case a @catch block
3078 // throws an exception).
3079 CGF.Builder.CreateCall(ObjCTypes.getExceptionTryEnterFn(), ExceptionData)
3080 ->setDoesNotThrow();
Anders Carlsson80f25672008-09-09 17:59:25 +00003081
John McCall0b251722010-08-04 05:59:32 +00003082 llvm::CallInst *SetJmpResult =
3083 CGF.Builder.CreateCall(ObjCTypes.getSetJmpFn(), SetJmpBuffer,
3084 "setjmp.result");
3085 SetJmpResult->setDoesNotThrow();
Daniel Dunbar6bff2512009-08-03 17:06:42 +00003086
John McCall0b251722010-08-04 05:59:32 +00003087 llvm::Value *Threw =
3088 CGF.Builder.CreateIsNotNull(SetJmpResult, "did_catch_exception");
3089
3090 CatchBlock = CGF.createBasicBlock("catch");
3091 CatchHandler = CGF.createBasicBlock("catch_for_catch");
3092 CGF.Builder.CreateCondBr(Threw, CatchHandler, CatchBlock);
3093
3094 CGF.EmitBlock(CatchBlock);
3095 }
3096
3097 CGF.Builder.CreateStore(CGF.Builder.getInt1(HasFinally), CallTryExitVar);
Daniel Dunbar6bff2512009-08-03 17:06:42 +00003098
Daniel Dunbar55e40722008-09-27 07:03:52 +00003099 // Handle catch list. As a special case we check if everything is
3100 // matched and avoid generating code for falling off the end if
3101 // so.
3102 bool AllMatched = false;
Douglas Gregor8f5e3dd2010-04-23 22:50:49 +00003103 for (unsigned I = 0, N = AtTryStmt->getNumCatchStmts(); I != N; ++I) {
3104 const ObjCAtCatchStmt *CatchStmt = AtTryStmt->getCatchStmt(I);
Anders Carlsson80f25672008-09-09 17:59:25 +00003105
Douglas Gregorc00d8e12010-04-26 16:46:50 +00003106 const VarDecl *CatchParam = CatchStmt->getCatchParamDecl();
Steve Naroff14108da2009-07-10 23:34:53 +00003107 const ObjCObjectPointerType *OPT = 0;
Daniel Dunbar129271a2008-09-27 07:36:24 +00003108
Anders Carlsson80f25672008-09-09 17:59:25 +00003109 // catch(...) always matches.
Daniel Dunbar55e40722008-09-27 07:03:52 +00003110 if (!CatchParam) {
3111 AllMatched = true;
3112 } else {
John McCall183700f2009-09-21 23:43:11 +00003113 OPT = CatchParam->getType()->getAs<ObjCObjectPointerType>();
Daniel Dunbar6bff2512009-08-03 17:06:42 +00003114
John McCallf1549f62010-07-06 01:34:17 +00003115 // catch(id e) always matches under this ABI, since only
3116 // ObjC exceptions end up here in the first place.
Daniel Dunbar97f61d12008-09-27 22:21:14 +00003117 // FIXME: For the time being we also match id<X>; this should
3118 // be rejected by Sema instead.
Eli Friedman818e96f2009-07-11 00:57:02 +00003119 if (OPT && (OPT->isObjCIdType() || OPT->isObjCQualifiedIdType()))
Daniel Dunbar55e40722008-09-27 07:03:52 +00003120 AllMatched = true;
Anders Carlsson80f25672008-09-09 17:59:25 +00003121 }
Daniel Dunbar6bff2512009-08-03 17:06:42 +00003122
John McCallf1549f62010-07-06 01:34:17 +00003123 // If this is a catch-all, we don't need to test anything.
Daniel Dunbar6bff2512009-08-03 17:06:42 +00003124 if (AllMatched) {
John McCallf1549f62010-07-06 01:34:17 +00003125 CodeGenFunction::RunCleanupsScope CatchVarCleanups(CGF);
3126
Anders Carlssondde0a942008-09-11 09:15:33 +00003127 if (CatchParam) {
Steve Naroff7ba138a2009-03-03 19:52:17 +00003128 CGF.EmitLocalBlockVarDecl(*CatchParam);
Daniel Dunbara448fb22008-11-11 23:11:34 +00003129 assert(CGF.HaveInsertPoint() && "DeclStmt destroyed insert point?");
John McCallf1549f62010-07-06 01:34:17 +00003130
3131 // These types work out because ConvertType(id) == i8*.
Steve Naroff7ba138a2009-03-03 19:52:17 +00003132 CGF.Builder.CreateStore(Caught, CGF.GetAddrOfLocalVar(CatchParam));
Anders Carlssondde0a942008-09-11 09:15:33 +00003133 }
Daniel Dunbar6bff2512009-08-03 17:06:42 +00003134
Anders Carlssondde0a942008-09-11 09:15:33 +00003135 CGF.EmitStmt(CatchStmt->getCatchBody());
John McCallf1549f62010-07-06 01:34:17 +00003136
3137 // The scope of the catch variable ends right here.
3138 CatchVarCleanups.ForceCleanup();
3139
Anders Carlssonf3a79a92009-02-09 20:38:58 +00003140 CGF.EmitBranchThroughCleanup(FinallyEnd);
Anders Carlsson80f25672008-09-09 17:59:25 +00003141 break;
3142 }
Daniel Dunbar6bff2512009-08-03 17:06:42 +00003143
Steve Naroff14108da2009-07-10 23:34:53 +00003144 assert(OPT && "Unexpected non-object pointer type in @catch");
John McCall506b57e2010-05-17 21:00:27 +00003145 const ObjCObjectType *ObjTy = OPT->getObjectType();
John McCallf1549f62010-07-06 01:34:17 +00003146
3147 // FIXME: @catch (Class c) ?
John McCall506b57e2010-05-17 21:00:27 +00003148 ObjCInterfaceDecl *IDecl = ObjTy->getInterface();
3149 assert(IDecl && "Catch parameter must have Objective-C type!");
Anders Carlsson80f25672008-09-09 17:59:25 +00003150
3151 // Check if the @catch block matches the exception object.
John McCall506b57e2010-05-17 21:00:27 +00003152 llvm::Value *Class = EmitClassRef(CGF.Builder, IDecl);
Daniel Dunbar6bff2512009-08-03 17:06:42 +00003153
John McCallf1549f62010-07-06 01:34:17 +00003154 llvm::CallInst *Match =
Chris Lattner34b02a12009-04-22 02:26:14 +00003155 CGF.Builder.CreateCall2(ObjCTypes.getExceptionMatchFn(),
3156 Class, Caught, "match");
John McCallf1549f62010-07-06 01:34:17 +00003157 Match->setDoesNotThrow();
Daniel Dunbar6bff2512009-08-03 17:06:42 +00003158
John McCallf1549f62010-07-06 01:34:17 +00003159 llvm::BasicBlock *MatchedBlock = CGF.createBasicBlock("match");
3160 llvm::BasicBlock *NextCatchBlock = CGF.createBasicBlock("catch.next");
Daniel Dunbar6bff2512009-08-03 17:06:42 +00003161
3162 CGF.Builder.CreateCondBr(CGF.Builder.CreateIsNotNull(Match, "matched"),
Daniel Dunbare4b5ee02008-09-27 23:30:04 +00003163 MatchedBlock, NextCatchBlock);
Daniel Dunbar6bff2512009-08-03 17:06:42 +00003164
Anders Carlsson80f25672008-09-09 17:59:25 +00003165 // Emit the @catch block.
3166 CGF.EmitBlock(MatchedBlock);
John McCallf1549f62010-07-06 01:34:17 +00003167
3168 // Collect any cleanups for the catch variable. The scope lasts until
3169 // the end of the catch body.
John McCall0b251722010-08-04 05:59:32 +00003170 CodeGenFunction::RunCleanupsScope CatchVarCleanups(CGF);
John McCallf1549f62010-07-06 01:34:17 +00003171
Steve Naroff7ba138a2009-03-03 19:52:17 +00003172 CGF.EmitLocalBlockVarDecl(*CatchParam);
Daniel Dunbara448fb22008-11-11 23:11:34 +00003173 assert(CGF.HaveInsertPoint() && "DeclStmt destroyed insert point?");
Daniel Dunbar18ccc772008-09-28 01:03:14 +00003174
John McCallf1549f62010-07-06 01:34:17 +00003175 // Initialize the catch variable.
Daniel Dunbar6bff2512009-08-03 17:06:42 +00003176 llvm::Value *Tmp =
3177 CGF.Builder.CreateBitCast(Caught,
3178 CGF.ConvertType(CatchParam->getType()),
Daniel Dunbar18ccc772008-09-28 01:03:14 +00003179 "tmp");
Steve Naroff7ba138a2009-03-03 19:52:17 +00003180 CGF.Builder.CreateStore(Tmp, CGF.GetAddrOfLocalVar(CatchParam));
Daniel Dunbar6bff2512009-08-03 17:06:42 +00003181
Anders Carlssondde0a942008-09-11 09:15:33 +00003182 CGF.EmitStmt(CatchStmt->getCatchBody());
John McCallf1549f62010-07-06 01:34:17 +00003183
3184 // We're done with the catch variable.
3185 CatchVarCleanups.ForceCleanup();
3186
Anders Carlssonf3a79a92009-02-09 20:38:58 +00003187 CGF.EmitBranchThroughCleanup(FinallyEnd);
Daniel Dunbar6bff2512009-08-03 17:06:42 +00003188
Anders Carlsson80f25672008-09-09 17:59:25 +00003189 CGF.EmitBlock(NextCatchBlock);
3190 }
3191
John McCallf1549f62010-07-06 01:34:17 +00003192 CGF.ObjCEHValueStack.pop_back();
3193
John McCall0b251722010-08-04 05:59:32 +00003194 // If nothing wanted anything to do with the caught exception,
3195 // kill the extract call.
3196 if (Caught->use_empty())
3197 Caught->eraseFromParent();
3198
3199 if (!AllMatched)
3200 CGF.EmitBranchThroughCleanup(FinallyRethrow);
3201
3202 if (HasFinally) {
3203 // Emit the exception handler for the @catch blocks.
3204 CGF.EmitBlock(CatchHandler);
3205
3206 // In theory we might now need a write hazard, but actually it's
3207 // unnecessary because there's no local-accessing code between
3208 // the try's write hazard and here.
3209 //Hazards.emitWriteHazard();
3210
3211 // Don't pop the catch handler; the throw already did.
3212 CGF.Builder.CreateStore(CGF.Builder.getFalse(), CallTryExitVar);
Anders Carlssonf3a79a92009-02-09 20:38:58 +00003213 CGF.EmitBranchThroughCleanup(FinallyRethrow);
Daniel Dunbar55e40722008-09-27 07:03:52 +00003214 }
Anders Carlsson80f25672008-09-09 17:59:25 +00003215 }
Daniel Dunbar6bff2512009-08-03 17:06:42 +00003216
John McCall87bb5822010-07-31 23:20:56 +00003217 // Insert read hazards as required in the new blocks.
John McCall0b251722010-08-04 05:59:32 +00003218 Hazards.emitHazardsInNewBlocks();
John McCall87bb5822010-07-31 23:20:56 +00003219
John McCallf1549f62010-07-06 01:34:17 +00003220 // Pop the cleanup.
John McCall0b251722010-08-04 05:59:32 +00003221 CGF.Builder.restoreIP(TryFallthroughIP);
3222 if (CGF.HaveInsertPoint())
3223 CGF.Builder.CreateStore(CGF.Builder.getTrue(), CallTryExitVar);
John McCallf1549f62010-07-06 01:34:17 +00003224 CGF.PopCleanupBlock();
John McCall0b251722010-08-04 05:59:32 +00003225 CGF.EmitBlock(FinallyEnd.getBlock(), true);
Anders Carlssonf3a79a92009-02-09 20:38:58 +00003226
John McCallf1549f62010-07-06 01:34:17 +00003227 // Emit the rethrow block.
John McCall87bb5822010-07-31 23:20:56 +00003228 CGBuilderTy::InsertPoint SavedIP = CGF.Builder.saveAndClearIP();
John McCallff8e1152010-07-23 21:56:41 +00003229 CGF.EmitBlock(FinallyRethrow.getBlock(), true);
John McCallf1549f62010-07-06 01:34:17 +00003230 if (CGF.HaveInsertPoint()) {
John McCall0b251722010-08-04 05:59:32 +00003231 // Just look in the buffer for the exception to throw.
3232 llvm::CallInst *Caught =
3233 CGF.Builder.CreateCall(ObjCTypes.getExceptionExtractFn(),
3234 ExceptionData);
3235 Caught->setDoesNotThrow();
3236
3237 CGF.Builder.CreateCall(ObjCTypes.getExceptionThrowFn(), Caught)
John McCallf1549f62010-07-06 01:34:17 +00003238 ->setDoesNotThrow();
3239 CGF.Builder.CreateUnreachable();
Fariborz Jahanianf2878e52008-11-21 19:21:53 +00003240 }
Anders Carlsson80f25672008-09-09 17:59:25 +00003241
John McCall87bb5822010-07-31 23:20:56 +00003242 CGF.Builder.restoreIP(SavedIP);
Anders Carlsson64d5d6c2008-09-09 10:04:29 +00003243}
3244
3245void CGObjCMac::EmitThrowStmt(CodeGen::CodeGenFunction &CGF,
Daniel Dunbar898d5082008-09-30 01:06:03 +00003246 const ObjCAtThrowStmt &S) {
Anders Carlsson2b1e3112008-09-09 16:16:55 +00003247 llvm::Value *ExceptionAsObject;
Daniel Dunbar6bff2512009-08-03 17:06:42 +00003248
Anders Carlsson2b1e3112008-09-09 16:16:55 +00003249 if (const Expr *ThrowExpr = S.getThrowExpr()) {
3250 llvm::Value *Exception = CGF.EmitScalarExpr(ThrowExpr);
Daniel Dunbar6bff2512009-08-03 17:06:42 +00003251 ExceptionAsObject =
Anders Carlsson2b1e3112008-09-09 16:16:55 +00003252 CGF.Builder.CreateBitCast(Exception, ObjCTypes.ObjectPtrTy, "tmp");
3253 } else {
Daniel Dunbar6bff2512009-08-03 17:06:42 +00003254 assert((!CGF.ObjCEHValueStack.empty() && CGF.ObjCEHValueStack.back()) &&
Daniel Dunbar18ccc772008-09-28 01:03:14 +00003255 "Unexpected rethrow outside @catch block.");
Anders Carlsson273558f2009-02-07 21:37:21 +00003256 ExceptionAsObject = CGF.ObjCEHValueStack.back();
Anders Carlsson2b1e3112008-09-09 16:16:55 +00003257 }
Daniel Dunbar6bff2512009-08-03 17:06:42 +00003258
John McCallf1549f62010-07-06 01:34:17 +00003259 CGF.Builder.CreateCall(ObjCTypes.getExceptionThrowFn(), ExceptionAsObject)
3260 ->setDoesNotReturn();
Anders Carlsson80f25672008-09-09 17:59:25 +00003261 CGF.Builder.CreateUnreachable();
Daniel Dunbara448fb22008-11-11 23:11:34 +00003262
3263 // Clear the insertion point to indicate we are in unreachable code.
3264 CGF.Builder.ClearInsertionPoint();
Anders Carlsson64d5d6c2008-09-09 10:04:29 +00003265}
3266
Fariborz Jahanian3e283e32008-11-18 22:37:34 +00003267/// EmitObjCWeakRead - Code gen for loading value of a __weak
Fariborz Jahanian6dc23172008-11-18 21:45:40 +00003268/// object: objc_read_weak (id *src)
3269///
Fariborz Jahanian3e283e32008-11-18 22:37:34 +00003270llvm::Value * CGObjCMac::EmitObjCWeakRead(CodeGen::CodeGenFunction &CGF,
Mike Stump1eb44332009-09-09 15:08:12 +00003271 llvm::Value *AddrWeakObj) {
Eli Friedman8339b352009-03-07 03:57:15 +00003272 const llvm::Type* DestTy =
Daniel Dunbar6bff2512009-08-03 17:06:42 +00003273 cast<llvm::PointerType>(AddrWeakObj->getType())->getElementType();
3274 AddrWeakObj = CGF.Builder.CreateBitCast(AddrWeakObj,
3275 ObjCTypes.PtrObjectPtrTy);
Chris Lattner72db6c32009-04-22 02:44:54 +00003276 llvm::Value *read_weak = CGF.Builder.CreateCall(ObjCTypes.getGcReadWeakFn(),
Fariborz Jahanian3e283e32008-11-18 22:37:34 +00003277 AddrWeakObj, "weakread");
Eli Friedman8339b352009-03-07 03:57:15 +00003278 read_weak = CGF.Builder.CreateBitCast(read_weak, DestTy);
Fariborz Jahanian6dc23172008-11-18 21:45:40 +00003279 return read_weak;
3280}
3281
Fariborz Jahanian3e283e32008-11-18 22:37:34 +00003282/// EmitObjCWeakAssign - Code gen for assigning to a __weak object.
3283/// objc_assign_weak (id src, id *dst)
3284///
3285void CGObjCMac::EmitObjCWeakAssign(CodeGen::CodeGenFunction &CGF,
Mike Stump1eb44332009-09-09 15:08:12 +00003286 llvm::Value *src, llvm::Value *dst) {
Fariborz Jahanian0a855d02009-03-23 19:10:40 +00003287 const llvm::Type * SrcTy = src->getType();
3288 if (!isa<llvm::PointerType>(SrcTy)) {
Duncan Sands9408c452009-05-09 07:08:47 +00003289 unsigned Size = CGM.getTargetData().getTypeAllocSize(SrcTy);
Fariborz Jahanian0a855d02009-03-23 19:10:40 +00003290 assert(Size <= 8 && "does not support size > 8");
3291 src = (Size == 4) ? CGF.Builder.CreateBitCast(src, ObjCTypes.IntTy)
Daniel Dunbar6bff2512009-08-03 17:06:42 +00003292 : CGF.Builder.CreateBitCast(src, ObjCTypes.LongLongTy);
Fariborz Jahanian3b8a6522009-03-13 00:42:52 +00003293 src = CGF.Builder.CreateIntToPtr(src, ObjCTypes.Int8PtrTy);
3294 }
Fariborz Jahaniandbd32c22008-11-19 17:34:06 +00003295 src = CGF.Builder.CreateBitCast(src, ObjCTypes.ObjectPtrTy);
3296 dst = CGF.Builder.CreateBitCast(dst, ObjCTypes.PtrObjectPtrTy);
Chris Lattner96508e12009-04-17 22:12:36 +00003297 CGF.Builder.CreateCall2(ObjCTypes.getGcAssignWeakFn(),
Fariborz Jahanian3e283e32008-11-18 22:37:34 +00003298 src, dst, "weakassign");
3299 return;
3300}
3301
Fariborz Jahanian58626502008-11-19 00:59:10 +00003302/// EmitObjCGlobalAssign - Code gen for assigning to a __strong object.
3303/// objc_assign_global (id src, id *dst)
3304///
3305void CGObjCMac::EmitObjCGlobalAssign(CodeGen::CodeGenFunction &CGF,
Fariborz Jahanian021a7a62010-07-20 20:30:03 +00003306 llvm::Value *src, llvm::Value *dst,
3307 bool threadlocal) {
Fariborz Jahanian0a855d02009-03-23 19:10:40 +00003308 const llvm::Type * SrcTy = src->getType();
3309 if (!isa<llvm::PointerType>(SrcTy)) {
Duncan Sands9408c452009-05-09 07:08:47 +00003310 unsigned Size = CGM.getTargetData().getTypeAllocSize(SrcTy);
Fariborz Jahanian0a855d02009-03-23 19:10:40 +00003311 assert(Size <= 8 && "does not support size > 8");
3312 src = (Size == 4) ? CGF.Builder.CreateBitCast(src, ObjCTypes.IntTy)
Daniel Dunbar6bff2512009-08-03 17:06:42 +00003313 : CGF.Builder.CreateBitCast(src, ObjCTypes.LongLongTy);
Fariborz Jahanian3b8a6522009-03-13 00:42:52 +00003314 src = CGF.Builder.CreateIntToPtr(src, ObjCTypes.Int8PtrTy);
3315 }
Fariborz Jahaniandbd32c22008-11-19 17:34:06 +00003316 src = CGF.Builder.CreateBitCast(src, ObjCTypes.ObjectPtrTy);
3317 dst = CGF.Builder.CreateBitCast(dst, ObjCTypes.PtrObjectPtrTy);
Fariborz Jahanian021a7a62010-07-20 20:30:03 +00003318 if (!threadlocal)
3319 CGF.Builder.CreateCall2(ObjCTypes.getGcAssignGlobalFn(),
3320 src, dst, "globalassign");
3321 else
3322 CGF.Builder.CreateCall2(ObjCTypes.getGcAssignThreadLocalFn(),
3323 src, dst, "threadlocalassign");
Fariborz Jahanian58626502008-11-19 00:59:10 +00003324 return;
3325}
3326
Fariborz Jahanian7eda8362008-11-20 19:23:36 +00003327/// EmitObjCIvarAssign - Code gen for assigning to a __strong object.
Fariborz Jahanian6c7a1f32009-09-24 22:25:38 +00003328/// objc_assign_ivar (id src, id *dst, ptrdiff_t ivaroffset)
Fariborz Jahanian7eda8362008-11-20 19:23:36 +00003329///
3330void CGObjCMac::EmitObjCIvarAssign(CodeGen::CodeGenFunction &CGF,
Fariborz Jahanian6c7a1f32009-09-24 22:25:38 +00003331 llvm::Value *src, llvm::Value *dst,
3332 llvm::Value *ivarOffset) {
3333 assert(ivarOffset && "EmitObjCIvarAssign - ivarOffset is NULL");
Fariborz Jahanian0a855d02009-03-23 19:10:40 +00003334 const llvm::Type * SrcTy = src->getType();
3335 if (!isa<llvm::PointerType>(SrcTy)) {
Duncan Sands9408c452009-05-09 07:08:47 +00003336 unsigned Size = CGM.getTargetData().getTypeAllocSize(SrcTy);
Fariborz Jahanian0a855d02009-03-23 19:10:40 +00003337 assert(Size <= 8 && "does not support size > 8");
3338 src = (Size == 4) ? CGF.Builder.CreateBitCast(src, ObjCTypes.IntTy)
Daniel Dunbar6bff2512009-08-03 17:06:42 +00003339 : CGF.Builder.CreateBitCast(src, ObjCTypes.LongLongTy);
Fariborz Jahanian3b8a6522009-03-13 00:42:52 +00003340 src = CGF.Builder.CreateIntToPtr(src, ObjCTypes.Int8PtrTy);
3341 }
Fariborz Jahanian7eda8362008-11-20 19:23:36 +00003342 src = CGF.Builder.CreateBitCast(src, ObjCTypes.ObjectPtrTy);
3343 dst = CGF.Builder.CreateBitCast(dst, ObjCTypes.PtrObjectPtrTy);
Fariborz Jahanian6c7a1f32009-09-24 22:25:38 +00003344 CGF.Builder.CreateCall3(ObjCTypes.getGcAssignIvarFn(),
3345 src, dst, ivarOffset);
Fariborz Jahanian7eda8362008-11-20 19:23:36 +00003346 return;
3347}
3348
Fariborz Jahanian58626502008-11-19 00:59:10 +00003349/// EmitObjCStrongCastAssign - Code gen for assigning to a __strong cast object.
3350/// objc_assign_strongCast (id src, id *dst)
3351///
3352void CGObjCMac::EmitObjCStrongCastAssign(CodeGen::CodeGenFunction &CGF,
Mike Stump1eb44332009-09-09 15:08:12 +00003353 llvm::Value *src, llvm::Value *dst) {
Fariborz Jahanian0a855d02009-03-23 19:10:40 +00003354 const llvm::Type * SrcTy = src->getType();
3355 if (!isa<llvm::PointerType>(SrcTy)) {
Duncan Sands9408c452009-05-09 07:08:47 +00003356 unsigned Size = CGM.getTargetData().getTypeAllocSize(SrcTy);
Fariborz Jahanian0a855d02009-03-23 19:10:40 +00003357 assert(Size <= 8 && "does not support size > 8");
3358 src = (Size == 4) ? CGF.Builder.CreateBitCast(src, ObjCTypes.IntTy)
Daniel Dunbar6bff2512009-08-03 17:06:42 +00003359 : CGF.Builder.CreateBitCast(src, ObjCTypes.LongLongTy);
Fariborz Jahanian3b8a6522009-03-13 00:42:52 +00003360 src = CGF.Builder.CreateIntToPtr(src, ObjCTypes.Int8PtrTy);
3361 }
Fariborz Jahaniandbd32c22008-11-19 17:34:06 +00003362 src = CGF.Builder.CreateBitCast(src, ObjCTypes.ObjectPtrTy);
3363 dst = CGF.Builder.CreateBitCast(dst, ObjCTypes.PtrObjectPtrTy);
Chris Lattnerbbccd612009-04-22 02:38:11 +00003364 CGF.Builder.CreateCall2(ObjCTypes.getGcAssignStrongCastFn(),
Fariborz Jahanian58626502008-11-19 00:59:10 +00003365 src, dst, "weakassign");
3366 return;
3367}
3368
Fariborz Jahanian082b02e2009-07-08 01:18:33 +00003369void CGObjCMac::EmitGCMemmoveCollectable(CodeGen::CodeGenFunction &CGF,
Daniel Dunbar6bff2512009-08-03 17:06:42 +00003370 llvm::Value *DestPtr,
3371 llvm::Value *SrcPtr,
Fariborz Jahanian55bcace2010-06-15 22:44:06 +00003372 llvm::Value *size) {
Fariborz Jahanian082b02e2009-07-08 01:18:33 +00003373 SrcPtr = CGF.Builder.CreateBitCast(SrcPtr, ObjCTypes.Int8PtrTy);
3374 DestPtr = CGF.Builder.CreateBitCast(DestPtr, ObjCTypes.Int8PtrTy);
Fariborz Jahanian082b02e2009-07-08 01:18:33 +00003375 CGF.Builder.CreateCall3(ObjCTypes.GcMemmoveCollectableFn(),
Fariborz Jahanian55bcace2010-06-15 22:44:06 +00003376 DestPtr, SrcPtr, size);
Fariborz Jahanian082b02e2009-07-08 01:18:33 +00003377 return;
3378}
3379
Fariborz Jahanian0bb20362009-02-02 20:02:29 +00003380/// EmitObjCValueForIvar - Code Gen for ivar reference.
3381///
Fariborz Jahanian598d3f62009-02-03 19:03:09 +00003382LValue CGObjCMac::EmitObjCValueForIvar(CodeGen::CodeGenFunction &CGF,
3383 QualType ObjectTy,
3384 llvm::Value *BaseValue,
3385 const ObjCIvarDecl *Ivar,
Fariborz Jahanian598d3f62009-02-03 19:03:09 +00003386 unsigned CVRQualifiers) {
John McCallc12c5bb2010-05-15 11:32:37 +00003387 const ObjCInterfaceDecl *ID =
3388 ObjectTy->getAs<ObjCObjectType>()->getInterface();
Daniel Dunbar97776872009-04-22 07:32:20 +00003389 return EmitValueForIvarAtOffset(CGF, ID, BaseValue, Ivar, CVRQualifiers,
3390 EmitIvarOffset(CGF, ID, Ivar));
Fariborz Jahanian0bb20362009-02-02 20:02:29 +00003391}
3392
Fariborz Jahanianf63aa3f2009-02-10 19:02:04 +00003393llvm::Value *CGObjCMac::EmitIvarOffset(CodeGen::CodeGenFunction &CGF,
Daniel Dunbar2a031922009-04-22 05:08:15 +00003394 const ObjCInterfaceDecl *Interface,
Fariborz Jahanianf63aa3f2009-02-10 19:02:04 +00003395 const ObjCIvarDecl *Ivar) {
Daniel Dunbar97776872009-04-22 07:32:20 +00003396 uint64_t Offset = ComputeIvarBaseOffset(CGM, Interface, Ivar);
Owen Anderson4a28d5d2009-07-24 23:12:58 +00003397 return llvm::ConstantInt::get(
Daniel Dunbar6bff2512009-08-03 17:06:42 +00003398 CGM.getTypes().ConvertType(CGM.getContext().LongTy),
3399 Offset);
Fariborz Jahanianf63aa3f2009-02-10 19:02:04 +00003400}
3401
Daniel Dunbarf77ac862008-08-11 21:35:06 +00003402/* *** Private Interface *** */
3403
3404/// EmitImageInfo - Emit the image info marker used to encode some module
3405/// level information.
3406///
3407/// See: <rdr://4810609&4810587&4810587>
3408/// struct IMAGE_INFO {
3409/// unsigned version;
3410/// unsigned flags;
3411/// };
3412enum ImageInfoFlags {
Daniel Dunbarfce176b2010-04-25 20:39:01 +00003413 eImageInfo_FixAndContinue = (1 << 0),
Daniel Dunbar6bff2512009-08-03 17:06:42 +00003414 eImageInfo_GarbageCollected = (1 << 1),
3415 eImageInfo_GCOnly = (1 << 2),
Daniel Dunbarc7c6dc02009-04-20 07:11:47 +00003416 eImageInfo_OptimizedByDyld = (1 << 3), // FIXME: When is this set.
3417
Daniel Dunbarfce176b2010-04-25 20:39:01 +00003418 // A flag indicating that the module has no instances of a @synthesize of a
3419 // superclass variable. <rdar://problem/6803242>
Daniel Dunbarc7c6dc02009-04-20 07:11:47 +00003420 eImageInfo_CorrectedSynthesize = (1 << 4)
Daniel Dunbarf77ac862008-08-11 21:35:06 +00003421};
3422
Daniel Dunbarfce176b2010-04-25 20:39:01 +00003423void CGObjCCommonMac::EmitImageInfo() {
Daniel Dunbarf77ac862008-08-11 21:35:06 +00003424 unsigned version = 0; // Version is unused?
3425 unsigned flags = 0;
3426
3427 // FIXME: Fix and continue?
3428 if (CGM.getLangOptions().getGCMode() != LangOptions::NonGC)
3429 flags |= eImageInfo_GarbageCollected;
3430 if (CGM.getLangOptions().getGCMode() == LangOptions::GCOnly)
3431 flags |= eImageInfo_GCOnly;
Daniel Dunbar6bff2512009-08-03 17:06:42 +00003432
Daniel Dunbarc7c6dc02009-04-20 07:11:47 +00003433 // We never allow @synthesize of a superclass property.
3434 flags |= eImageInfo_CorrectedSynthesize;
Daniel Dunbarf77ac862008-08-11 21:35:06 +00003435
Chris Lattner77b89b82010-06-27 07:15:29 +00003436 const llvm::Type *Int32Ty = llvm::Type::getInt32Ty(VMContext);
3437
Daniel Dunbarf77ac862008-08-11 21:35:06 +00003438 // Emitted as int[2];
3439 llvm::Constant *values[2] = {
Chris Lattner77b89b82010-06-27 07:15:29 +00003440 llvm::ConstantInt::get(Int32Ty, version),
3441 llvm::ConstantInt::get(Int32Ty, flags)
Daniel Dunbarf77ac862008-08-11 21:35:06 +00003442 };
Chris Lattner77b89b82010-06-27 07:15:29 +00003443 llvm::ArrayType *AT = llvm::ArrayType::get(Int32Ty, 2);
Daniel Dunbar63c5b502009-03-09 21:49:58 +00003444
3445 const char *Section;
3446 if (ObjCABI == 1)
3447 Section = "__OBJC, __image_info,regular";
3448 else
3449 Section = "__DATA, __objc_imageinfo, regular, no_dead_strip";
Daniel Dunbar6bff2512009-08-03 17:06:42 +00003450 llvm::GlobalVariable *GV =
Daniel Dunbar63c5b502009-03-09 21:49:58 +00003451 CreateMetadataVar("\01L_OBJC_IMAGE_INFO",
Owen Anderson7db6d832009-07-28 18:33:04 +00003452 llvm::ConstantArray::get(AT, values, 2),
Daniel Dunbar63c5b502009-03-09 21:49:58 +00003453 Section,
3454 0,
3455 true);
3456 GV->setConstant(true);
Daniel Dunbarf77ac862008-08-11 21:35:06 +00003457}
3458
Daniel Dunbar4e2d7d02008-08-12 06:48:42 +00003459
3460// struct objc_module {
3461// unsigned long version;
3462// unsigned long size;
3463// const char *name;
3464// Symtab symtab;
3465// };
3466
3467// FIXME: Get from somewhere
3468static const int ModuleVersion = 7;
3469
3470void CGObjCMac::EmitModuleInfo() {
Duncan Sands9408c452009-05-09 07:08:47 +00003471 uint64_t Size = CGM.getTargetData().getTypeAllocSize(ObjCTypes.ModuleTy);
Daniel Dunbar6bff2512009-08-03 17:06:42 +00003472
Daniel Dunbar4e2d7d02008-08-12 06:48:42 +00003473 std::vector<llvm::Constant*> Values(4);
Owen Anderson4a28d5d2009-07-24 23:12:58 +00003474 Values[0] = llvm::ConstantInt::get(ObjCTypes.LongTy, ModuleVersion);
3475 Values[1] = llvm::ConstantInt::get(ObjCTypes.LongTy, Size);
Daniel Dunbar7ded7f42008-08-15 22:20:32 +00003476 // This used to be the filename, now it is unused. <rdr://4327263>
Daniel Dunbar6efc0c52008-08-13 03:21:16 +00003477 Values[2] = GetClassName(&CGM.getContext().Idents.get(""));
Daniel Dunbar4e2d7d02008-08-12 06:48:42 +00003478 Values[3] = EmitModuleSymbols();
Daniel Dunbar6bff2512009-08-03 17:06:42 +00003479 CreateMetadataVar("\01L_OBJC_MODULES",
Owen Anderson08e25242009-07-27 22:29:56 +00003480 llvm::ConstantStruct::get(ObjCTypes.ModuleTy, Values),
Daniel Dunbar63c5b502009-03-09 21:49:58 +00003481 "__OBJC,__module_info,regular,no_dead_strip",
Daniel Dunbar58a29122009-03-09 22:18:41 +00003482 4, true);
Daniel Dunbar4e2d7d02008-08-12 06:48:42 +00003483}
3484
3485llvm::Constant *CGObjCMac::EmitModuleSymbols() {
Daniel Dunbar27f9d772008-08-21 04:36:09 +00003486 unsigned NumClasses = DefinedClasses.size();
3487 unsigned NumCategories = DefinedCategories.size();
3488
Daniel Dunbar242d4dc2008-08-25 06:02:07 +00003489 // Return null if no symbols were defined.
3490 if (!NumClasses && !NumCategories)
Owen Andersonc9c88b42009-07-31 20:28:54 +00003491 return llvm::Constant::getNullValue(ObjCTypes.SymtabPtrTy);
Daniel Dunbar242d4dc2008-08-25 06:02:07 +00003492
3493 std::vector<llvm::Constant*> Values(5);
Owen Anderson4a28d5d2009-07-24 23:12:58 +00003494 Values[0] = llvm::ConstantInt::get(ObjCTypes.LongTy, 0);
Owen Andersonc9c88b42009-07-31 20:28:54 +00003495 Values[1] = llvm::Constant::getNullValue(ObjCTypes.SelectorPtrTy);
Owen Anderson4a28d5d2009-07-24 23:12:58 +00003496 Values[2] = llvm::ConstantInt::get(ObjCTypes.ShortTy, NumClasses);
3497 Values[3] = llvm::ConstantInt::get(ObjCTypes.ShortTy, NumCategories);
Daniel Dunbar27f9d772008-08-21 04:36:09 +00003498
Daniel Dunbar86e253a2008-08-22 20:34:54 +00003499 // The runtime expects exactly the list of defined classes followed
3500 // by the list of defined categories, in a single array.
Daniel Dunbar27f9d772008-08-21 04:36:09 +00003501 std::vector<llvm::Constant*> Symbols(NumClasses + NumCategories);
Daniel Dunbar86e253a2008-08-22 20:34:54 +00003502 for (unsigned i=0; i<NumClasses; i++)
Owen Anderson3c4972d2009-07-29 18:54:39 +00003503 Symbols[i] = llvm::ConstantExpr::getBitCast(DefinedClasses[i],
Daniel Dunbar86e253a2008-08-22 20:34:54 +00003504 ObjCTypes.Int8PtrTy);
3505 for (unsigned i=0; i<NumCategories; i++)
Daniel Dunbar6bff2512009-08-03 17:06:42 +00003506 Symbols[NumClasses + i] =
Owen Anderson3c4972d2009-07-29 18:54:39 +00003507 llvm::ConstantExpr::getBitCast(DefinedCategories[i],
Daniel Dunbar86e253a2008-08-22 20:34:54 +00003508 ObjCTypes.Int8PtrTy);
3509
Daniel Dunbar6bff2512009-08-03 17:06:42 +00003510 Values[4] =
Owen Anderson96e0fc72009-07-29 22:16:19 +00003511 llvm::ConstantArray::get(llvm::ArrayType::get(ObjCTypes.Int8PtrTy,
Daniel Dunbar27f9d772008-08-21 04:36:09 +00003512 NumClasses + NumCategories),
3513 Symbols);
3514
Nick Lewycky0d36dd22009-09-19 20:00:52 +00003515 llvm::Constant *Init = llvm::ConstantStruct::get(VMContext, Values, false);
Daniel Dunbar27f9d772008-08-21 04:36:09 +00003516
Daniel Dunbar4e2d7d02008-08-12 06:48:42 +00003517 llvm::GlobalVariable *GV =
Daniel Dunbar63c5b502009-03-09 21:49:58 +00003518 CreateMetadataVar("\01L_OBJC_SYMBOLS", Init,
3519 "__OBJC,__symbols,regular,no_dead_strip",
Daniel Dunbar0bf21992009-04-15 02:56:18 +00003520 4, true);
Owen Anderson3c4972d2009-07-29 18:54:39 +00003521 return llvm::ConstantExpr::getBitCast(GV, ObjCTypes.SymtabPtrTy);
Daniel Dunbar27f9d772008-08-21 04:36:09 +00003522}
3523
Daniel Dunbar6bff2512009-08-03 17:06:42 +00003524llvm::Value *CGObjCMac::EmitClassRef(CGBuilderTy &Builder,
Daniel Dunbar27f9d772008-08-21 04:36:09 +00003525 const ObjCInterfaceDecl *ID) {
Daniel Dunbar242d4dc2008-08-25 06:02:07 +00003526 LazySymbols.insert(ID->getIdentifier());
3527
Daniel Dunbar27f9d772008-08-21 04:36:09 +00003528 llvm::GlobalVariable *&Entry = ClassReferences[ID->getIdentifier()];
Daniel Dunbar6bff2512009-08-03 17:06:42 +00003529
Daniel Dunbar27f9d772008-08-21 04:36:09 +00003530 if (!Entry) {
Daniel Dunbar6bff2512009-08-03 17:06:42 +00003531 llvm::Constant *Casted =
Owen Anderson3c4972d2009-07-29 18:54:39 +00003532 llvm::ConstantExpr::getBitCast(GetClassName(ID->getIdentifier()),
Daniel Dunbar27f9d772008-08-21 04:36:09 +00003533 ObjCTypes.ClassPtrTy);
Daniel Dunbar6bff2512009-08-03 17:06:42 +00003534 Entry =
Daniel Dunbar63c5b502009-03-09 21:49:58 +00003535 CreateMetadataVar("\01L_OBJC_CLASS_REFERENCES_", Casted,
3536 "__OBJC,__cls_refs,literal_pointers,no_dead_strip",
Daniel Dunbar0bf21992009-04-15 02:56:18 +00003537 4, true);
Daniel Dunbar27f9d772008-08-21 04:36:09 +00003538 }
3539
Daniel Dunbar2da84ff2009-11-29 21:23:36 +00003540 return Builder.CreateLoad(Entry, "tmp");
Daniel Dunbar4e2d7d02008-08-12 06:48:42 +00003541}
3542
Fariborz Jahanian03b29602010-06-17 19:56:20 +00003543llvm::Value *CGObjCMac::EmitSelector(CGBuilderTy &Builder, Selector Sel,
3544 bool lvalue) {
Daniel Dunbar259d93d2008-08-12 03:39:23 +00003545 llvm::GlobalVariable *&Entry = SelectorReferences[Sel];
Daniel Dunbar6bff2512009-08-03 17:06:42 +00003546
Daniel Dunbar259d93d2008-08-12 03:39:23 +00003547 if (!Entry) {
Daniel Dunbar6bff2512009-08-03 17:06:42 +00003548 llvm::Constant *Casted =
Owen Anderson3c4972d2009-07-29 18:54:39 +00003549 llvm::ConstantExpr::getBitCast(GetMethodVarName(Sel),
Daniel Dunbar259d93d2008-08-12 03:39:23 +00003550 ObjCTypes.SelectorPtrTy);
Daniel Dunbar6bff2512009-08-03 17:06:42 +00003551 Entry =
Daniel Dunbar63c5b502009-03-09 21:49:58 +00003552 CreateMetadataVar("\01L_OBJC_SELECTOR_REFERENCES_", Casted,
3553 "__OBJC,__message_refs,literal_pointers,no_dead_strip",
Daniel Dunbar0bf21992009-04-15 02:56:18 +00003554 4, true);
Daniel Dunbar259d93d2008-08-12 03:39:23 +00003555 }
3556
Fariborz Jahanian03b29602010-06-17 19:56:20 +00003557 if (lvalue)
3558 return Entry;
Daniel Dunbar2da84ff2009-11-29 21:23:36 +00003559 return Builder.CreateLoad(Entry, "tmp");
Daniel Dunbar259d93d2008-08-12 03:39:23 +00003560}
3561
Fariborz Jahanian058a1b72009-01-24 20:21:50 +00003562llvm::Constant *CGObjCCommonMac::GetClassName(IdentifierInfo *Ident) {
Daniel Dunbar6efc0c52008-08-13 03:21:16 +00003563 llvm::GlobalVariable *&Entry = ClassNames[Ident];
Daniel Dunbar4e2d7d02008-08-12 06:48:42 +00003564
Daniel Dunbar63c5b502009-03-09 21:49:58 +00003565 if (!Entry)
Daniel Dunbar6bff2512009-08-03 17:06:42 +00003566 Entry = CreateMetadataVar("\01L_OBJC_CLASS_NAME_",
Daniel Dunbar9c29bf52009-10-18 20:48:59 +00003567 llvm::ConstantArray::get(VMContext,
3568 Ident->getNameStart()),
Daniel Dunbar6633e3c2010-07-29 22:57:21 +00003569 "__TEXT,__cstring,cstring_literals",
Daniel Dunbarb90bb002009-04-14 23:14:47 +00003570 1, true);
Daniel Dunbar4e2d7d02008-08-12 06:48:42 +00003571
Owen Andersona1cf15f2009-07-14 23:10:40 +00003572 return getConstantGEP(VMContext, Entry, 0, 0);
Daniel Dunbar4e2d7d02008-08-12 06:48:42 +00003573}
3574
Argyrios Kyrtzidis9d50c062010-08-09 10:54:20 +00003575llvm::Function *CGObjCCommonMac::GetMethodDefinition(const ObjCMethodDecl *MD) {
3576 llvm::DenseMap<const ObjCMethodDecl*, llvm::Function*>::iterator
3577 I = MethodDefinitions.find(MD);
3578 if (I != MethodDefinitions.end())
3579 return I->second;
3580
3581 if (MD->hasBody() && MD->getPCHLevel() > 0) {
3582 // MD isn't emitted yet because it comes from PCH.
3583 CGM.EmitTopLevelDecl(const_cast<ObjCMethodDecl*>(MD));
3584 assert(MethodDefinitions[MD] && "EmitTopLevelDecl didn't emit the method!");
3585 return MethodDefinitions[MD];
3586 }
3587
3588 return NULL;
3589}
3590
Fariborz Jahaniand80d81b2009-03-05 19:17:31 +00003591/// GetIvarLayoutName - Returns a unique constant for the given
3592/// ivar layout bitmap.
3593llvm::Constant *CGObjCCommonMac::GetIvarLayoutName(IdentifierInfo *Ident,
Daniel Dunbar6bff2512009-08-03 17:06:42 +00003594 const ObjCCommonTypesHelper &ObjCTypes) {
Owen Andersonc9c88b42009-07-31 20:28:54 +00003595 return llvm::Constant::getNullValue(ObjCTypes.Int8PtrTy);
Fariborz Jahaniand80d81b2009-03-05 19:17:31 +00003596}
3597
Daniel Dunbard58edcb2009-05-03 14:10:34 +00003598void CGObjCCommonMac::BuildAggrIvarRecordLayout(const RecordType *RT,
Daniel Dunbar6bff2512009-08-03 17:06:42 +00003599 unsigned int BytePos,
Daniel Dunbard58edcb2009-05-03 14:10:34 +00003600 bool ForStrongLayout,
3601 bool &HasUnion) {
3602 const RecordDecl *RD = RT->getDecl();
3603 // FIXME - Use iterator.
Argyrios Kyrtzidis17945a02009-06-30 02:36:12 +00003604 llvm::SmallVector<FieldDecl*, 16> Fields(RD->field_begin(), RD->field_end());
Daniel Dunbard58edcb2009-05-03 14:10:34 +00003605 const llvm::Type *Ty = CGM.getTypes().ConvertType(QualType(RT, 0));
Daniel Dunbar6bff2512009-08-03 17:06:42 +00003606 const llvm::StructLayout *RecLayout =
Daniel Dunbard58edcb2009-05-03 14:10:34 +00003607 CGM.getTargetData().getStructLayout(cast<llvm::StructType>(Ty));
Daniel Dunbar6bff2512009-08-03 17:06:42 +00003608
Daniel Dunbard58edcb2009-05-03 14:10:34 +00003609 BuildAggrIvarLayout(0, RecLayout, RD, Fields, BytePos,
3610 ForStrongLayout, HasUnion);
3611}
3612
Daniel Dunbar5a5a8032009-05-03 21:05:10 +00003613void CGObjCCommonMac::BuildAggrIvarLayout(const ObjCImplementationDecl *OI,
Daniel Dunbar6bff2512009-08-03 17:06:42 +00003614 const llvm::StructLayout *Layout,
3615 const RecordDecl *RD,
Chris Lattnerf1690852009-03-31 08:48:01 +00003616 const llvm::SmallVectorImpl<FieldDecl*> &RecFields,
Daniel Dunbar6bff2512009-08-03 17:06:42 +00003617 unsigned int BytePos, bool ForStrongLayout,
3618 bool &HasUnion) {
Fariborz Jahanian820e0202009-03-11 00:07:04 +00003619 bool IsUnion = (RD && RD->isUnion());
3620 uint64_t MaxUnionIvarSize = 0;
3621 uint64_t MaxSkippedUnionIvarSize = 0;
3622 FieldDecl *MaxField = 0;
3623 FieldDecl *MaxSkippedField = 0;
Fariborz Jahanian7fb16272009-04-21 18:33:06 +00003624 FieldDecl *LastFieldBitfield = 0;
Daniel Dunbar900c1982009-05-03 23:31:46 +00003625 uint64_t MaxFieldOffset = 0;
3626 uint64_t MaxSkippedFieldOffset = 0;
3627 uint64_t LastBitfieldOffset = 0;
Daniel Dunbar6bff2512009-08-03 17:06:42 +00003628
Fariborz Jahaniana5a10c32009-03-10 16:22:08 +00003629 if (RecFields.empty())
3630 return;
Chris Lattnerf1690852009-03-31 08:48:01 +00003631 unsigned WordSizeInBits = CGM.getContext().Target.getPointerWidth(0);
3632 unsigned ByteSizeInBits = CGM.getContext().Target.getCharWidth();
3633
Chris Lattnerf1690852009-03-31 08:48:01 +00003634 for (unsigned i = 0, e = RecFields.size(); i != e; ++i) {
Fariborz Jahaniana5a10c32009-03-10 16:22:08 +00003635 FieldDecl *Field = RecFields[i];
Daniel Dunbare05cc982009-05-03 23:35:23 +00003636 uint64_t FieldOffset;
Anders Carlssonfc514ab2009-07-24 17:23:54 +00003637 if (RD) {
Daniel Dunbarf8f8eba2010-04-14 17:02:21 +00003638 // Note that 'i' here is actually the field index inside RD of Field,
3639 // although this dependency is hidden.
3640 const ASTRecordLayout &RL = CGM.getContext().getASTRecordLayout(RD);
3641 FieldOffset = RL.getFieldOffset(i) / 8;
Anders Carlssonfc514ab2009-07-24 17:23:54 +00003642 } else
Daniel Dunbare05cc982009-05-03 23:35:23 +00003643 FieldOffset = ComputeIvarBaseOffset(CGM, OI, cast<ObjCIvarDecl>(Field));
Daniel Dunbar25d583e2009-05-03 14:17:18 +00003644
Fariborz Jahaniana5a10c32009-03-10 16:22:08 +00003645 // Skip over unnamed or bitfields
Fariborz Jahanian7fb16272009-04-21 18:33:06 +00003646 if (!Field->getIdentifier() || Field->isBitField()) {
3647 LastFieldBitfield = Field;
Daniel Dunbar900c1982009-05-03 23:31:46 +00003648 LastBitfieldOffset = FieldOffset;
Fariborz Jahaniana5a10c32009-03-10 16:22:08 +00003649 continue;
Fariborz Jahanian7fb16272009-04-21 18:33:06 +00003650 }
Daniel Dunbar25d583e2009-05-03 14:17:18 +00003651
Fariborz Jahanian7fb16272009-04-21 18:33:06 +00003652 LastFieldBitfield = 0;
Fariborz Jahaniana5a10c32009-03-10 16:22:08 +00003653 QualType FQT = Field->getType();
Fariborz Jahanian667423a2009-03-25 22:36:49 +00003654 if (FQT->isRecordType() || FQT->isUnionType()) {
Fariborz Jahaniana5a10c32009-03-10 16:22:08 +00003655 if (FQT->isUnionType())
3656 HasUnion = true;
Fariborz Jahanian820e0202009-03-11 00:07:04 +00003657
Daniel Dunbar6bff2512009-08-03 17:06:42 +00003658 BuildAggrIvarRecordLayout(FQT->getAs<RecordType>(),
Daniel Dunbar25d583e2009-05-03 14:17:18 +00003659 BytePos + FieldOffset,
Daniel Dunbard58edcb2009-05-03 14:10:34 +00003660 ForStrongLayout, HasUnion);
Fariborz Jahaniana5a10c32009-03-10 16:22:08 +00003661 continue;
3662 }
Daniel Dunbar6bff2512009-08-03 17:06:42 +00003663
Chris Lattnerf1690852009-03-31 08:48:01 +00003664 if (const ArrayType *Array = CGM.getContext().getAsArrayType(FQT)) {
Daniel Dunbar6bff2512009-08-03 17:06:42 +00003665 const ConstantArrayType *CArray =
Daniel Dunbar5e563dd2009-05-03 13:55:09 +00003666 dyn_cast_or_null<ConstantArrayType>(Array);
Fariborz Jahanian7fb16272009-04-21 18:33:06 +00003667 uint64_t ElCount = CArray->getSize().getZExtValue();
Daniel Dunbar5e563dd2009-05-03 13:55:09 +00003668 assert(CArray && "only array with known element size is supported");
Fariborz Jahanian820e0202009-03-11 00:07:04 +00003669 FQT = CArray->getElementType();
Fariborz Jahanian667423a2009-03-25 22:36:49 +00003670 while (const ArrayType *Array = CGM.getContext().getAsArrayType(FQT)) {
3671 const ConstantArrayType *CArray =
Daniel Dunbar5e563dd2009-05-03 13:55:09 +00003672 dyn_cast_or_null<ConstantArrayType>(Array);
Fariborz Jahanian7fb16272009-04-21 18:33:06 +00003673 ElCount *= CArray->getSize().getZExtValue();
Fariborz Jahanian667423a2009-03-25 22:36:49 +00003674 FQT = CArray->getElementType();
3675 }
Daniel Dunbar6bff2512009-08-03 17:06:42 +00003676
3677 assert(!FQT->isUnionType() &&
Fariborz Jahanian820e0202009-03-11 00:07:04 +00003678 "layout for array of unions not supported");
3679 if (FQT->isRecordType()) {
Fariborz Jahanian81adc052009-04-24 16:17:09 +00003680 int OldIndex = IvarsInfo.size() - 1;
3681 int OldSkIndex = SkipIvars.size() -1;
Daniel Dunbar6bff2512009-08-03 17:06:42 +00003682
Ted Kremenek6217b802009-07-29 21:53:49 +00003683 const RecordType *RT = FQT->getAs<RecordType>();
Daniel Dunbar25d583e2009-05-03 14:17:18 +00003684 BuildAggrIvarRecordLayout(RT, BytePos + FieldOffset,
Daniel Dunbard58edcb2009-05-03 14:10:34 +00003685 ForStrongLayout, HasUnion);
Daniel Dunbar6bff2512009-08-03 17:06:42 +00003686
Fariborz Jahanian820e0202009-03-11 00:07:04 +00003687 // Replicate layout information for each array element. Note that
3688 // one element is already done.
3689 uint64_t ElIx = 1;
Daniel Dunbar6bff2512009-08-03 17:06:42 +00003690 for (int FirstIndex = IvarsInfo.size() - 1,
3691 FirstSkIndex = SkipIvars.size() - 1 ;ElIx < ElCount; ElIx++) {
Fariborz Jahanianc8ce9c82009-03-12 22:50:49 +00003692 uint64_t Size = CGM.getContext().getTypeSize(RT)/ByteSizeInBits;
Daniel Dunbar8b2926c2009-05-03 13:44:42 +00003693 for (int i = OldIndex+1; i <= FirstIndex; ++i)
3694 IvarsInfo.push_back(GC_IVAR(IvarsInfo[i].ivar_bytepos + Size*ElIx,
3695 IvarsInfo[i].ivar_size));
3696 for (int i = OldSkIndex+1; i <= FirstSkIndex; ++i)
3697 SkipIvars.push_back(GC_IVAR(SkipIvars[i].ivar_bytepos + Size*ElIx,
3698 SkipIvars[i].ivar_size));
Fariborz Jahanian820e0202009-03-11 00:07:04 +00003699 }
3700 continue;
3701 }
Fariborz Jahaniana5a10c32009-03-10 16:22:08 +00003702 }
Fariborz Jahanian820e0202009-03-11 00:07:04 +00003703 // At this point, we are done with Record/Union and array there of.
3704 // For other arrays we are down to its element type.
John McCall0953e762009-09-24 19:53:00 +00003705 Qualifiers::GC GCAttr = GetGCAttrTypeForType(CGM.getContext(), FQT);
Daniel Dunbar5e563dd2009-05-03 13:55:09 +00003706
Daniel Dunbar8b2926c2009-05-03 13:44:42 +00003707 unsigned FieldSize = CGM.getContext().getTypeSize(Field->getType());
John McCall0953e762009-09-24 19:53:00 +00003708 if ((ForStrongLayout && GCAttr == Qualifiers::Strong)
3709 || (!ForStrongLayout && GCAttr == Qualifiers::Weak)) {
Daniel Dunbar487993b2009-05-03 13:32:01 +00003710 if (IsUnion) {
Daniel Dunbar8b2926c2009-05-03 13:44:42 +00003711 uint64_t UnionIvarSize = FieldSize / WordSizeInBits;
Daniel Dunbar487993b2009-05-03 13:32:01 +00003712 if (UnionIvarSize > MaxUnionIvarSize) {
Fariborz Jahanian820e0202009-03-11 00:07:04 +00003713 MaxUnionIvarSize = UnionIvarSize;
3714 MaxField = Field;
Daniel Dunbar900c1982009-05-03 23:31:46 +00003715 MaxFieldOffset = FieldOffset;
Fariborz Jahanian820e0202009-03-11 00:07:04 +00003716 }
Daniel Dunbar487993b2009-05-03 13:32:01 +00003717 } else {
Daniel Dunbar25d583e2009-05-03 14:17:18 +00003718 IvarsInfo.push_back(GC_IVAR(BytePos + FieldOffset,
Daniel Dunbar8b2926c2009-05-03 13:44:42 +00003719 FieldSize / WordSizeInBits));
Fariborz Jahanian820e0202009-03-11 00:07:04 +00003720 }
Daniel Dunbar6bff2512009-08-03 17:06:42 +00003721 } else if ((ForStrongLayout &&
John McCall0953e762009-09-24 19:53:00 +00003722 (GCAttr == Qualifiers::GCNone || GCAttr == Qualifiers::Weak))
3723 || (!ForStrongLayout && GCAttr != Qualifiers::Weak)) {
Daniel Dunbar487993b2009-05-03 13:32:01 +00003724 if (IsUnion) {
Mike Stumpf5408fe2009-05-16 07:57:57 +00003725 // FIXME: Why the asymmetry? We divide by word size in bits on other
3726 // side.
Daniel Dunbar8b2926c2009-05-03 13:44:42 +00003727 uint64_t UnionIvarSize = FieldSize;
Daniel Dunbar487993b2009-05-03 13:32:01 +00003728 if (UnionIvarSize > MaxSkippedUnionIvarSize) {
Fariborz Jahanian820e0202009-03-11 00:07:04 +00003729 MaxSkippedUnionIvarSize = UnionIvarSize;
3730 MaxSkippedField = Field;
Daniel Dunbar900c1982009-05-03 23:31:46 +00003731 MaxSkippedFieldOffset = FieldOffset;
Fariborz Jahanian820e0202009-03-11 00:07:04 +00003732 }
Daniel Dunbar487993b2009-05-03 13:32:01 +00003733 } else {
Daniel Dunbar8b2926c2009-05-03 13:44:42 +00003734 // FIXME: Why the asymmetry, we divide by byte size in bits here?
Daniel Dunbar25d583e2009-05-03 14:17:18 +00003735 SkipIvars.push_back(GC_IVAR(BytePos + FieldOffset,
Daniel Dunbar8b2926c2009-05-03 13:44:42 +00003736 FieldSize / ByteSizeInBits));
Fariborz Jahanian820e0202009-03-11 00:07:04 +00003737 }
3738 }
3739 }
Daniel Dunbard58edcb2009-05-03 14:10:34 +00003740
Fariborz Jahanian7fb16272009-04-21 18:33:06 +00003741 if (LastFieldBitfield) {
3742 // Last field was a bitfield. Must update skip info.
Fariborz Jahanian7fb16272009-04-21 18:33:06 +00003743 Expr *BitWidth = LastFieldBitfield->getBitWidth();
3744 uint64_t BitFieldSize =
Eli Friedman9a901bb2009-04-26 19:19:15 +00003745 BitWidth->EvaluateAsInt(CGM.getContext()).getZExtValue();
Daniel Dunbar487993b2009-05-03 13:32:01 +00003746 GC_IVAR skivar;
Daniel Dunbar900c1982009-05-03 23:31:46 +00003747 skivar.ivar_bytepos = BytePos + LastBitfieldOffset;
Daniel Dunbar6bff2512009-08-03 17:06:42 +00003748 skivar.ivar_size = (BitFieldSize / ByteSizeInBits)
3749 + ((BitFieldSize % ByteSizeInBits) != 0);
3750 SkipIvars.push_back(skivar);
Fariborz Jahanian7fb16272009-04-21 18:33:06 +00003751 }
Daniel Dunbar6bff2512009-08-03 17:06:42 +00003752
Daniel Dunbar8b2926c2009-05-03 13:44:42 +00003753 if (MaxField)
Daniel Dunbar6bff2512009-08-03 17:06:42 +00003754 IvarsInfo.push_back(GC_IVAR(BytePos + MaxFieldOffset,
Daniel Dunbar8b2926c2009-05-03 13:44:42 +00003755 MaxUnionIvarSize));
3756 if (MaxSkippedField)
Daniel Dunbar900c1982009-05-03 23:31:46 +00003757 SkipIvars.push_back(GC_IVAR(BytePos + MaxSkippedFieldOffset,
Daniel Dunbar8b2926c2009-05-03 13:44:42 +00003758 MaxSkippedUnionIvarSize));
Fariborz Jahaniand61a50a2009-03-05 22:39:55 +00003759}
3760
Fariborz Jahanianb8fd2eb2010-08-05 00:19:48 +00003761/// BuildIvarLayoutBitmap - This routine is the horsework for doing all
3762/// the computations and returning the layout bitmap (for ivar or blocks) in
3763/// the given argument BitMap string container. Routine reads
3764/// two containers, IvarsInfo and SkipIvars which are assumed to be
3765/// filled already by the caller.
3766llvm::Constant *CGObjCCommonMac::BuildIvarLayoutBitmap(std::string& BitMap) {
Fariborz Jahanian9397e1d2009-03-11 20:59:05 +00003767 unsigned int WordsToScan, WordsToSkip;
Benjamin Kramer3c0ef8c2009-10-13 10:07:13 +00003768 const llvm::Type *PtrTy = llvm::Type::getInt8PtrTy(VMContext);
Fariborz Jahanian93ce50d2010-08-04 23:55:24 +00003769
Fariborz Jahanian9397e1d2009-03-11 20:59:05 +00003770 // Build the string of skip/scan nibbles
Fariborz Jahanian8c2f2d12009-04-24 17:15:27 +00003771 llvm::SmallVector<SKIP_SCAN, 32> SkipScanIvars;
Daniel Dunbar6bff2512009-08-03 17:06:42 +00003772 unsigned int WordSize =
Fariborz Jahanian93ce50d2010-08-04 23:55:24 +00003773 CGM.getTypes().getTargetData().getTypeAllocSize(PtrTy);
Fariborz Jahanian9397e1d2009-03-11 20:59:05 +00003774 if (IvarsInfo[0].ivar_bytepos == 0) {
3775 WordsToSkip = 0;
3776 WordsToScan = IvarsInfo[0].ivar_size;
Daniel Dunbar31682fd2009-05-03 23:21:22 +00003777 } else {
Fariborz Jahanian9397e1d2009-03-11 20:59:05 +00003778 WordsToSkip = IvarsInfo[0].ivar_bytepos/WordSize;
3779 WordsToScan = IvarsInfo[0].ivar_size;
3780 }
Daniel Dunbar31682fd2009-05-03 23:21:22 +00003781 for (unsigned int i=1, Last=IvarsInfo.size(); i != Last; i++) {
Daniel Dunbar6bff2512009-08-03 17:06:42 +00003782 unsigned int TailPrevGCObjC =
Fariborz Jahanian93ce50d2010-08-04 23:55:24 +00003783 IvarsInfo[i-1].ivar_bytepos + IvarsInfo[i-1].ivar_size * WordSize;
Daniel Dunbar31682fd2009-05-03 23:21:22 +00003784 if (IvarsInfo[i].ivar_bytepos == TailPrevGCObjC) {
Fariborz Jahanian9397e1d2009-03-11 20:59:05 +00003785 // consecutive 'scanned' object pointers.
3786 WordsToScan += IvarsInfo[i].ivar_size;
Daniel Dunbar31682fd2009-05-03 23:21:22 +00003787 } else {
Fariborz Jahanian9397e1d2009-03-11 20:59:05 +00003788 // Skip over 'gc'able object pointer which lay over each other.
3789 if (TailPrevGCObjC > IvarsInfo[i].ivar_bytepos)
3790 continue;
3791 // Must skip over 1 or more words. We save current skip/scan values
3792 // and start a new pair.
Fariborz Jahanianc8ce9c82009-03-12 22:50:49 +00003793 SKIP_SCAN SkScan;
3794 SkScan.skip = WordsToSkip;
3795 SkScan.scan = WordsToScan;
Fariborz Jahanian81adc052009-04-24 16:17:09 +00003796 SkipScanIvars.push_back(SkScan);
Fariborz Jahanian93ce50d2010-08-04 23:55:24 +00003797
Fariborz Jahanian9397e1d2009-03-11 20:59:05 +00003798 // Skip the hole.
Fariborz Jahanianc8ce9c82009-03-12 22:50:49 +00003799 SkScan.skip = (IvarsInfo[i].ivar_bytepos - TailPrevGCObjC) / WordSize;
3800 SkScan.scan = 0;
Fariborz Jahanian81adc052009-04-24 16:17:09 +00003801 SkipScanIvars.push_back(SkScan);
Fariborz Jahanian9397e1d2009-03-11 20:59:05 +00003802 WordsToSkip = 0;
3803 WordsToScan = IvarsInfo[i].ivar_size;
3804 }
3805 }
Daniel Dunbar31682fd2009-05-03 23:21:22 +00003806 if (WordsToScan > 0) {
Fariborz Jahanianc8ce9c82009-03-12 22:50:49 +00003807 SKIP_SCAN SkScan;
3808 SkScan.skip = WordsToSkip;
3809 SkScan.scan = WordsToScan;
Fariborz Jahanian81adc052009-04-24 16:17:09 +00003810 SkipScanIvars.push_back(SkScan);
Fariborz Jahanian9397e1d2009-03-11 20:59:05 +00003811 }
Fariborz Jahanian93ce50d2010-08-04 23:55:24 +00003812
Daniel Dunbar31682fd2009-05-03 23:21:22 +00003813 if (!SkipIvars.empty()) {
Fariborz Jahanian81adc052009-04-24 16:17:09 +00003814 unsigned int LastIndex = SkipIvars.size()-1;
Daniel Dunbar6bff2512009-08-03 17:06:42 +00003815 int LastByteSkipped =
Fariborz Jahanian93ce50d2010-08-04 23:55:24 +00003816 SkipIvars[LastIndex].ivar_bytepos + SkipIvars[LastIndex].ivar_size;
Fariborz Jahanian81adc052009-04-24 16:17:09 +00003817 LastIndex = IvarsInfo.size()-1;
Daniel Dunbar6bff2512009-08-03 17:06:42 +00003818 int LastByteScanned =
Fariborz Jahanian93ce50d2010-08-04 23:55:24 +00003819 IvarsInfo[LastIndex].ivar_bytepos +
3820 IvarsInfo[LastIndex].ivar_size * WordSize;
Fariborz Jahanian9397e1d2009-03-11 20:59:05 +00003821 // Compute number of bytes to skip at the tail end of the last ivar scanned.
Benjamin Kramer54d76db2009-12-25 15:43:36 +00003822 if (LastByteSkipped > LastByteScanned) {
Fariborz Jahanian9397e1d2009-03-11 20:59:05 +00003823 unsigned int TotalWords = (LastByteSkipped + (WordSize -1)) / WordSize;
Fariborz Jahanianc8ce9c82009-03-12 22:50:49 +00003824 SKIP_SCAN SkScan;
3825 SkScan.skip = TotalWords - (LastByteScanned/WordSize);
3826 SkScan.scan = 0;
Fariborz Jahanian81adc052009-04-24 16:17:09 +00003827 SkipScanIvars.push_back(SkScan);
Fariborz Jahanian9397e1d2009-03-11 20:59:05 +00003828 }
3829 }
3830 // Mini optimization of nibbles such that an 0xM0 followed by 0x0N is produced
3831 // as 0xMN.
Fariborz Jahanian81adc052009-04-24 16:17:09 +00003832 int SkipScan = SkipScanIvars.size()-1;
Daniel Dunbar31682fd2009-05-03 23:21:22 +00003833 for (int i = 0; i <= SkipScan; i++) {
Fariborz Jahanian9397e1d2009-03-11 20:59:05 +00003834 if ((i < SkipScan) && SkipScanIvars[i].skip && SkipScanIvars[i].scan == 0
3835 && SkipScanIvars[i+1].skip == 0 && SkipScanIvars[i+1].scan) {
3836 // 0xM0 followed by 0x0N detected.
3837 SkipScanIvars[i].scan = SkipScanIvars[i+1].scan;
3838 for (int j = i+1; j < SkipScan; j++)
3839 SkipScanIvars[j] = SkipScanIvars[j+1];
3840 --SkipScan;
3841 }
3842 }
Fariborz Jahanian93ce50d2010-08-04 23:55:24 +00003843
Fariborz Jahanian9397e1d2009-03-11 20:59:05 +00003844 // Generate the string.
Daniel Dunbar31682fd2009-05-03 23:21:22 +00003845 for (int i = 0; i <= SkipScan; i++) {
Fariborz Jahanian9397e1d2009-03-11 20:59:05 +00003846 unsigned char byte;
3847 unsigned int skip_small = SkipScanIvars[i].skip % 0xf;
3848 unsigned int scan_small = SkipScanIvars[i].scan % 0xf;
3849 unsigned int skip_big = SkipScanIvars[i].skip / 0xf;
3850 unsigned int scan_big = SkipScanIvars[i].scan / 0xf;
Fariborz Jahanian93ce50d2010-08-04 23:55:24 +00003851
Fariborz Jahanian9397e1d2009-03-11 20:59:05 +00003852 // first skip big.
3853 for (unsigned int ix = 0; ix < skip_big; ix++)
3854 BitMap += (unsigned char)(0xf0);
Fariborz Jahanian93ce50d2010-08-04 23:55:24 +00003855
Fariborz Jahanian9397e1d2009-03-11 20:59:05 +00003856 // next (skip small, scan)
Daniel Dunbar31682fd2009-05-03 23:21:22 +00003857 if (skip_small) {
Fariborz Jahanian9397e1d2009-03-11 20:59:05 +00003858 byte = skip_small << 4;
Daniel Dunbar31682fd2009-05-03 23:21:22 +00003859 if (scan_big > 0) {
Fariborz Jahanian9397e1d2009-03-11 20:59:05 +00003860 byte |= 0xf;
3861 --scan_big;
Daniel Dunbar31682fd2009-05-03 23:21:22 +00003862 } else if (scan_small) {
Fariborz Jahanian9397e1d2009-03-11 20:59:05 +00003863 byte |= scan_small;
3864 scan_small = 0;
3865 }
3866 BitMap += byte;
3867 }
3868 // next scan big
3869 for (unsigned int ix = 0; ix < scan_big; ix++)
3870 BitMap += (unsigned char)(0x0f);
3871 // last scan small
Daniel Dunbar31682fd2009-05-03 23:21:22 +00003872 if (scan_small) {
Fariborz Jahanian9397e1d2009-03-11 20:59:05 +00003873 byte = scan_small;
3874 BitMap += byte;
3875 }
3876 }
3877 // null terminate string.
Fariborz Jahanian667423a2009-03-25 22:36:49 +00003878 unsigned char zero = 0;
3879 BitMap += zero;
Fariborz Jahanian93ce50d2010-08-04 23:55:24 +00003880
3881 llvm::GlobalVariable * Entry =
3882 CreateMetadataVar("\01L_OBJC_CLASS_NAME_",
3883 llvm::ConstantArray::get(VMContext, BitMap.c_str()),
3884 "__TEXT,__cstring,cstring_literals",
3885 1, true);
3886 return getConstantGEP(VMContext, Entry, 0, 0);
3887}
Daniel Dunbar6bff2512009-08-03 17:06:42 +00003888
Fariborz Jahanian93ce50d2010-08-04 23:55:24 +00003889/// BuildIvarLayout - Builds ivar layout bitmap for the class
3890/// implementation for the __strong or __weak case.
3891/// The layout map displays which words in ivar list must be skipped
3892/// and which must be scanned by GC (see below). String is built of bytes.
3893/// Each byte is divided up in two nibbles (4-bit each). Left nibble is count
3894/// of words to skip and right nibble is count of words to scan. So, each
3895/// nibble represents up to 15 workds to skip or scan. Skipping the rest is
3896/// represented by a 0x00 byte which also ends the string.
3897/// 1. when ForStrongLayout is true, following ivars are scanned:
3898/// - id, Class
3899/// - object *
3900/// - __strong anything
3901///
3902/// 2. When ForStrongLayout is false, following ivars are scanned:
3903/// - __weak anything
3904///
3905llvm::Constant *CGObjCCommonMac::BuildIvarLayout(
3906 const ObjCImplementationDecl *OMD,
3907 bool ForStrongLayout) {
3908 bool hasUnion = false;
3909
3910 const llvm::Type *PtrTy = llvm::Type::getInt8PtrTy(VMContext);
3911 if (CGM.getLangOptions().getGCMode() == LangOptions::NonGC)
3912 return llvm::Constant::getNullValue(PtrTy);
3913
3914 llvm::SmallVector<FieldDecl*, 32> RecFields;
3915 const ObjCInterfaceDecl *OI = OMD->getClassInterface();
3916 CGM.getContext().CollectObjCIvars(OI, RecFields);
3917
3918 // Add this implementations synthesized ivars.
3919 llvm::SmallVector<ObjCIvarDecl*, 16> Ivars;
3920 CGM.getContext().CollectNonClassIvars(OI, Ivars);
3921 for (unsigned k = 0, e = Ivars.size(); k != e; ++k)
3922 RecFields.push_back(cast<FieldDecl>(Ivars[k]));
3923
3924 if (RecFields.empty())
3925 return llvm::Constant::getNullValue(PtrTy);
3926
3927 SkipIvars.clear();
3928 IvarsInfo.clear();
3929
3930 BuildAggrIvarLayout(OMD, 0, 0, RecFields, 0, ForStrongLayout, hasUnion);
3931 if (IvarsInfo.empty())
3932 return llvm::Constant::getNullValue(PtrTy);
Fariborz Jahanianb8fd2eb2010-08-05 00:19:48 +00003933 // Sort on byte position in case we encounterred a union nested in
3934 // the ivar list.
3935 if (hasUnion && !IvarsInfo.empty())
3936 std::sort(IvarsInfo.begin(), IvarsInfo.end());
3937 if (hasUnion && !SkipIvars.empty())
3938 std::sort(SkipIvars.begin(), SkipIvars.end());
3939
Fariborz Jahanian93ce50d2010-08-04 23:55:24 +00003940 std::string BitMap;
Fariborz Jahanianb8fd2eb2010-08-05 00:19:48 +00003941 llvm::Constant *C = BuildIvarLayoutBitmap(BitMap);
Fariborz Jahanian93ce50d2010-08-04 23:55:24 +00003942
3943 if (CGM.getLangOptions().ObjCGCBitmapPrint) {
Daniel Dunbar6bff2512009-08-03 17:06:42 +00003944 printf("\n%s ivar layout for class '%s': ",
Fariborz Jahanian3d2ad662009-04-20 22:03:45 +00003945 ForStrongLayout ? "strong" : "weak",
3946 OMD->getClassInterface()->getNameAsCString());
3947 const unsigned char *s = (unsigned char*)BitMap.c_str();
3948 for (unsigned i = 0; i < BitMap.size(); i++)
3949 if (!(s[i] & 0xf0))
3950 printf("0x0%x%s", s[i], s[i] != 0 ? ", " : "");
3951 else
3952 printf("0x%x%s", s[i], s[i] != 0 ? ", " : "");
3953 printf("\n");
3954 }
Fariborz Jahanian93ce50d2010-08-04 23:55:24 +00003955 return C;
Fariborz Jahaniand61a50a2009-03-05 22:39:55 +00003956}
3957
Fariborz Jahanian56210f72009-01-21 23:34:32 +00003958llvm::Constant *CGObjCCommonMac::GetMethodVarName(Selector Sel) {
Daniel Dunbar259d93d2008-08-12 03:39:23 +00003959 llvm::GlobalVariable *&Entry = MethodVarNames[Sel];
3960
Daniel Dunbar63c5b502009-03-09 21:49:58 +00003961 // FIXME: Avoid std::string copying.
3962 if (!Entry)
Daniel Dunbar6bff2512009-08-03 17:06:42 +00003963 Entry = CreateMetadataVar("\01L_OBJC_METH_VAR_NAME_",
Owen Anderson0032b272009-08-13 21:57:51 +00003964 llvm::ConstantArray::get(VMContext, Sel.getAsString()),
Daniel Dunbar6633e3c2010-07-29 22:57:21 +00003965 "__TEXT,__cstring,cstring_literals",
Daniel Dunbarb90bb002009-04-14 23:14:47 +00003966 1, true);
Daniel Dunbar259d93d2008-08-12 03:39:23 +00003967
Owen Andersona1cf15f2009-07-14 23:10:40 +00003968 return getConstantGEP(VMContext, Entry, 0, 0);
Daniel Dunbar6efc0c52008-08-13 03:21:16 +00003969}
3970
Daniel Dunbar27f9d772008-08-21 04:36:09 +00003971// FIXME: Merge into a single cstring creation function.
Fariborz Jahanian56210f72009-01-21 23:34:32 +00003972llvm::Constant *CGObjCCommonMac::GetMethodVarName(IdentifierInfo *ID) {
Daniel Dunbar27f9d772008-08-21 04:36:09 +00003973 return GetMethodVarName(CGM.getContext().Selectors.getNullarySelector(ID));
3974}
3975
3976// FIXME: Merge into a single cstring creation function.
Fariborz Jahanian56210f72009-01-21 23:34:32 +00003977llvm::Constant *CGObjCCommonMac::GetMethodVarName(const std::string &Name) {
Daniel Dunbar27f9d772008-08-21 04:36:09 +00003978 return GetMethodVarName(&CGM.getContext().Idents.get(Name));
3979}
3980
Daniel Dunbar3e5f0d82009-04-20 06:54:31 +00003981llvm::Constant *CGObjCCommonMac::GetMethodVarType(const FieldDecl *Field) {
Devang Patel7794bb82009-03-04 18:21:39 +00003982 std::string TypeStr;
3983 CGM.getContext().getObjCEncodingForType(Field->getType(), TypeStr, Field);
3984
3985 llvm::GlobalVariable *&Entry = MethodVarTypes[TypeStr];
Daniel Dunbar6efc0c52008-08-13 03:21:16 +00003986
Daniel Dunbar63c5b502009-03-09 21:49:58 +00003987 if (!Entry)
3988 Entry = CreateMetadataVar("\01L_OBJC_METH_VAR_TYPE_",
Owen Anderson0032b272009-08-13 21:57:51 +00003989 llvm::ConstantArray::get(VMContext, TypeStr),
Daniel Dunbar6633e3c2010-07-29 22:57:21 +00003990 "__TEXT,__cstring,cstring_literals",
Daniel Dunbarb90bb002009-04-14 23:14:47 +00003991 1, true);
Daniel Dunbar6bff2512009-08-03 17:06:42 +00003992
Owen Andersona1cf15f2009-07-14 23:10:40 +00003993 return getConstantGEP(VMContext, Entry, 0, 0);
Daniel Dunbar259d93d2008-08-12 03:39:23 +00003994}
3995
Fariborz Jahanian56210f72009-01-21 23:34:32 +00003996llvm::Constant *CGObjCCommonMac::GetMethodVarType(const ObjCMethodDecl *D) {
Daniel Dunbar27f9d772008-08-21 04:36:09 +00003997 std::string TypeStr;
Daniel Dunbarc45ef602008-08-26 21:51:14 +00003998 CGM.getContext().getObjCEncodingForMethodDecl(const_cast<ObjCMethodDecl*>(D),
3999 TypeStr);
Devang Patel7794bb82009-03-04 18:21:39 +00004000
4001 llvm::GlobalVariable *&Entry = MethodVarTypes[TypeStr];
4002
Daniel Dunbarb90bb002009-04-14 23:14:47 +00004003 if (!Entry)
4004 Entry = CreateMetadataVar("\01L_OBJC_METH_VAR_TYPE_",
Owen Anderson0032b272009-08-13 21:57:51 +00004005 llvm::ConstantArray::get(VMContext, TypeStr),
Daniel Dunbar6633e3c2010-07-29 22:57:21 +00004006 "__TEXT,__cstring,cstring_literals",
Daniel Dunbarb90bb002009-04-14 23:14:47 +00004007 1, true);
Devang Patel7794bb82009-03-04 18:21:39 +00004008
Owen Andersona1cf15f2009-07-14 23:10:40 +00004009 return getConstantGEP(VMContext, Entry, 0, 0);
Daniel Dunbar27f9d772008-08-21 04:36:09 +00004010}
4011
Daniel Dunbarc8ef5512008-08-23 00:19:03 +00004012// FIXME: Merge into a single cstring creation function.
Fariborz Jahanian56210f72009-01-21 23:34:32 +00004013llvm::Constant *CGObjCCommonMac::GetPropertyName(IdentifierInfo *Ident) {
Daniel Dunbarc8ef5512008-08-23 00:19:03 +00004014 llvm::GlobalVariable *&Entry = PropertyNames[Ident];
Daniel Dunbar6bff2512009-08-03 17:06:42 +00004015
Daniel Dunbar63c5b502009-03-09 21:49:58 +00004016 if (!Entry)
Daniel Dunbar6bff2512009-08-03 17:06:42 +00004017 Entry = CreateMetadataVar("\01L_OBJC_PROP_NAME_ATTR_",
Daniel Dunbar9c29bf52009-10-18 20:48:59 +00004018 llvm::ConstantArray::get(VMContext,
4019 Ident->getNameStart()),
Daniel Dunbar63c5b502009-03-09 21:49:58 +00004020 "__TEXT,__cstring,cstring_literals",
Daniel Dunbarb90bb002009-04-14 23:14:47 +00004021 1, true);
Daniel Dunbarc8ef5512008-08-23 00:19:03 +00004022
Owen Andersona1cf15f2009-07-14 23:10:40 +00004023 return getConstantGEP(VMContext, Entry, 0, 0);
Daniel Dunbarc8ef5512008-08-23 00:19:03 +00004024}
4025
4026// FIXME: Merge into a single cstring creation function.
Daniel Dunbarc56f34a2008-08-28 04:38:10 +00004027// FIXME: This Decl should be more precise.
Daniel Dunbar63c5b502009-03-09 21:49:58 +00004028llvm::Constant *
Daniel Dunbar6bff2512009-08-03 17:06:42 +00004029CGObjCCommonMac::GetPropertyTypeString(const ObjCPropertyDecl *PD,
4030 const Decl *Container) {
Daniel Dunbarc56f34a2008-08-28 04:38:10 +00004031 std::string TypeStr;
4032 CGM.getContext().getObjCEncodingForPropertyDecl(PD, Container, TypeStr);
Daniel Dunbarc8ef5512008-08-23 00:19:03 +00004033 return GetPropertyName(&CGM.getContext().Idents.get(TypeStr));
4034}
4035
Daniel Dunbar6bff2512009-08-03 17:06:42 +00004036void CGObjCCommonMac::GetNameForMethod(const ObjCMethodDecl *D,
Fariborz Jahanian56210f72009-01-21 23:34:32 +00004037 const ObjCContainerDecl *CD,
Daniel Dunbarc575ce72009-10-19 01:21:19 +00004038 llvm::SmallVectorImpl<char> &Name) {
4039 llvm::raw_svector_ostream OS(Name);
Fariborz Jahanian679a5022009-01-10 21:06:09 +00004040 assert (CD && "Missing container decl in GetNameForMethod");
Daniel Dunbarc575ce72009-10-19 01:21:19 +00004041 OS << '\01' << (D->isInstanceMethod() ? '-' : '+')
4042 << '[' << CD->getName();
Daniel Dunbar6bff2512009-08-03 17:06:42 +00004043 if (const ObjCCategoryImplDecl *CID =
Daniel Dunbarc575ce72009-10-19 01:21:19 +00004044 dyn_cast<ObjCCategoryImplDecl>(D->getDeclContext()))
Benjamin Kramer900fc632010-04-17 09:33:03 +00004045 OS << '(' << CID << ')';
Daniel Dunbarc575ce72009-10-19 01:21:19 +00004046 OS << ' ' << D->getSelector().getAsString() << ']';
Daniel Dunbarb7ec2462008-08-16 03:19:19 +00004047}
4048
Daniel Dunbarf77ac862008-08-11 21:35:06 +00004049void CGObjCMac::FinishModule() {
Daniel Dunbar4e2d7d02008-08-12 06:48:42 +00004050 EmitModuleInfo();
4051
Daniel Dunbar0c0e7a62008-10-29 22:36:39 +00004052 // Emit the dummy bodies for any protocols which were referenced but
4053 // never defined.
Daniel Dunbar6bff2512009-08-03 17:06:42 +00004054 for (llvm::DenseMap<IdentifierInfo*, llvm::GlobalVariable*>::iterator
Chris Lattnerad64e022009-07-17 23:57:13 +00004055 I = Protocols.begin(), e = Protocols.end(); I != e; ++I) {
4056 if (I->second->hasInitializer())
Daniel Dunbar0c0e7a62008-10-29 22:36:39 +00004057 continue;
Daniel Dunbar4e2d7d02008-08-12 06:48:42 +00004058
Daniel Dunbar0c0e7a62008-10-29 22:36:39 +00004059 std::vector<llvm::Constant*> Values(5);
Owen Andersonc9c88b42009-07-31 20:28:54 +00004060 Values[0] = llvm::Constant::getNullValue(ObjCTypes.ProtocolExtensionPtrTy);
Chris Lattnerad64e022009-07-17 23:57:13 +00004061 Values[1] = GetClassName(I->first);
Owen Andersonc9c88b42009-07-31 20:28:54 +00004062 Values[2] = llvm::Constant::getNullValue(ObjCTypes.ProtocolListPtrTy);
Daniel Dunbar0c0e7a62008-10-29 22:36:39 +00004063 Values[3] = Values[4] =
Owen Andersonc9c88b42009-07-31 20:28:54 +00004064 llvm::Constant::getNullValue(ObjCTypes.MethodDescriptionListPtrTy);
Chris Lattnerad64e022009-07-17 23:57:13 +00004065 I->second->setLinkage(llvm::GlobalValue::InternalLinkage);
Owen Anderson08e25242009-07-27 22:29:56 +00004066 I->second->setInitializer(llvm::ConstantStruct::get(ObjCTypes.ProtocolTy,
Daniel Dunbar0c0e7a62008-10-29 22:36:39 +00004067 Values));
Chris Lattnerad64e022009-07-17 23:57:13 +00004068 CGM.AddUsedGlobal(I->second);
Daniel Dunbar0c0e7a62008-10-29 22:36:39 +00004069 }
4070
Daniel Dunbar242d4dc2008-08-25 06:02:07 +00004071 // Add assembler directives to add lazy undefined symbol references
4072 // for classes which are referenced but not defined. This is
4073 // important for correct linker interaction.
Daniel Dunbar33063492009-09-07 00:20:42 +00004074 //
4075 // FIXME: It would be nice if we had an LLVM construct for this.
4076 if (!LazySymbols.empty() || !DefinedSymbols.empty()) {
4077 llvm::SmallString<256> Asm;
4078 Asm += CGM.getModule().getModuleInlineAsm();
4079 if (!Asm.empty() && Asm.back() != '\n')
4080 Asm += '\n';
Daniel Dunbar242d4dc2008-08-25 06:02:07 +00004081
Daniel Dunbar33063492009-09-07 00:20:42 +00004082 llvm::raw_svector_ostream OS(Asm);
Daniel Dunbar33063492009-09-07 00:20:42 +00004083 for (llvm::SetVector<IdentifierInfo*>::iterator I = DefinedSymbols.begin(),
4084 e = DefinedSymbols.end(); I != e; ++I)
Daniel Dunbar01eb9b92009-10-18 21:17:35 +00004085 OS << "\t.objc_class_name_" << (*I)->getName() << "=0\n"
4086 << "\t.globl .objc_class_name_" << (*I)->getName() << "\n";
Chris Lattnerafd5eda2010-04-17 18:26:20 +00004087 for (llvm::SetVector<IdentifierInfo*>::iterator I = LazySymbols.begin(),
Fariborz Jahanianb9c5b3d2010-06-21 22:05:18 +00004088 e = LazySymbols.end(); I != e; ++I) {
Chris Lattnerafd5eda2010-04-17 18:26:20 +00004089 OS << "\t.lazy_reference .objc_class_name_" << (*I)->getName() << "\n";
Fariborz Jahanianb9c5b3d2010-06-21 22:05:18 +00004090 }
4091
4092 for (size_t i = 0; i < DefinedCategoryNames.size(); ++i) {
4093 OS << "\t.objc_category_name_" << DefinedCategoryNames[i] << "=0\n"
4094 << "\t.globl .objc_category_name_" << DefinedCategoryNames[i] << "\n";
4095 }
Chris Lattnerafd5eda2010-04-17 18:26:20 +00004096
Daniel Dunbar33063492009-09-07 00:20:42 +00004097 CGM.getModule().setModuleInlineAsm(OS.str());
Daniel Dunbar242d4dc2008-08-25 06:02:07 +00004098 }
Daniel Dunbarf77ac862008-08-11 21:35:06 +00004099}
4100
Daniel Dunbar6bff2512009-08-03 17:06:42 +00004101CGObjCNonFragileABIMac::CGObjCNonFragileABIMac(CodeGen::CodeGenModule &cgm)
Fariborz Jahanianee0af742009-01-21 22:04:16 +00004102 : CGObjCCommonMac(cgm),
Mike Stump1eb44332009-09-09 15:08:12 +00004103 ObjCTypes(cgm) {
Fariborz Jahanianaa23b572009-01-23 23:53:38 +00004104 ObjCEmptyCacheVar = ObjCEmptyVtableVar = NULL;
Fariborz Jahanianee0af742009-01-21 22:04:16 +00004105 ObjCABI = 2;
4106}
4107
Daniel Dunbarf77ac862008-08-11 21:35:06 +00004108/* *** */
4109
Fariborz Jahanianee0af742009-01-21 22:04:16 +00004110ObjCCommonTypesHelper::ObjCCommonTypesHelper(CodeGen::CodeGenModule &cgm)
Mike Stump1eb44332009-09-09 15:08:12 +00004111 : VMContext(cgm.getLLVMContext()), CGM(cgm) {
Daniel Dunbar4e2d7d02008-08-12 06:48:42 +00004112 CodeGen::CodeGenTypes &Types = CGM.getTypes();
4113 ASTContext &Ctx = CGM.getContext();
Daniel Dunbar6bff2512009-08-03 17:06:42 +00004114
Daniel Dunbar27f9d772008-08-21 04:36:09 +00004115 ShortTy = Types.ConvertType(Ctx.ShortTy);
Daniel Dunbar6efc0c52008-08-13 03:21:16 +00004116 IntTy = Types.ConvertType(Ctx.IntTy);
Daniel Dunbar4e2d7d02008-08-12 06:48:42 +00004117 LongTy = Types.ConvertType(Ctx.LongTy);
Fariborz Jahanian0a855d02009-03-23 19:10:40 +00004118 LongLongTy = Types.ConvertType(Ctx.LongLongTy);
Benjamin Kramer3c0ef8c2009-10-13 10:07:13 +00004119 Int8PtrTy = llvm::Type::getInt8PtrTy(VMContext);
Daniel Dunbar6bff2512009-08-03 17:06:42 +00004120
Daniel Dunbar4e2d7d02008-08-12 06:48:42 +00004121 ObjectPtrTy = Types.ConvertType(Ctx.getObjCIdType());
Owen Anderson96e0fc72009-07-29 22:16:19 +00004122 PtrObjectPtrTy = llvm::PointerType::getUnqual(ObjectPtrTy);
Daniel Dunbar4e2d7d02008-08-12 06:48:42 +00004123 SelectorPtrTy = Types.ConvertType(Ctx.getObjCSelType());
Daniel Dunbar6bff2512009-08-03 17:06:42 +00004124
Mike Stumpf5408fe2009-05-16 07:57:57 +00004125 // FIXME: It would be nice to unify this with the opaque type, so that the IR
4126 // comes out a bit cleaner.
Daniel Dunbar6efc0c52008-08-13 03:21:16 +00004127 const llvm::Type *T = Types.ConvertType(Ctx.getObjCProtoType());
Owen Anderson96e0fc72009-07-29 22:16:19 +00004128 ExternalProtocolPtrTy = llvm::PointerType::getUnqual(T);
Daniel Dunbar6bff2512009-08-03 17:06:42 +00004129
Fariborz Jahanianee0af742009-01-21 22:04:16 +00004130 // I'm not sure I like this. The implicit coordination is a bit
4131 // gross. We should solve this in a reasonable fashion because this
4132 // is a pretty common task (match some runtime data structure with
4133 // an LLVM data structure).
Daniel Dunbar6bff2512009-08-03 17:06:42 +00004134
Fariborz Jahanianee0af742009-01-21 22:04:16 +00004135 // FIXME: This is leaked.
4136 // FIXME: Merge with rewriter code?
Daniel Dunbar6bff2512009-08-03 17:06:42 +00004137
Fariborz Jahanianee0af742009-01-21 22:04:16 +00004138 // struct _objc_super {
4139 // id self;
4140 // Class cls;
4141 // }
Abramo Bagnara465d41b2010-05-11 21:36:43 +00004142 RecordDecl *RD = RecordDecl::Create(Ctx, TTK_Struct,
Daniel Dunbardaa3ac52010-04-29 16:29:11 +00004143 Ctx.getTranslationUnitDecl(),
Fariborz Jahanianee0af742009-01-21 22:04:16 +00004144 SourceLocation(),
Daniel Dunbar6bff2512009-08-03 17:06:42 +00004145 &Ctx.Idents.get("_objc_super"));
Argyrios Kyrtzidis17945a02009-06-30 02:36:12 +00004146 RD->addDecl(FieldDecl::Create(Ctx, RD, SourceLocation(), 0,
Argyrios Kyrtzidisa1d56622009-08-19 01:27:57 +00004147 Ctx.getObjCIdType(), 0, 0, false));
Daniel Dunbar6bff2512009-08-03 17:06:42 +00004148 RD->addDecl(FieldDecl::Create(Ctx, RD, SourceLocation(), 0,
Argyrios Kyrtzidisa1d56622009-08-19 01:27:57 +00004149 Ctx.getObjCClassType(), 0, 0, false));
Douglas Gregor838db382010-02-11 01:19:42 +00004150 RD->completeDefinition();
Daniel Dunbar6bff2512009-08-03 17:06:42 +00004151
Fariborz Jahanianee0af742009-01-21 22:04:16 +00004152 SuperCTy = Ctx.getTagDeclType(RD);
4153 SuperPtrCTy = Ctx.getPointerType(SuperCTy);
Daniel Dunbar6bff2512009-08-03 17:06:42 +00004154
Fariborz Jahanianee0af742009-01-21 22:04:16 +00004155 SuperTy = cast<llvm::StructType>(Types.ConvertType(SuperCTy));
Daniel Dunbar6bff2512009-08-03 17:06:42 +00004156 SuperPtrTy = llvm::PointerType::getUnqual(SuperTy);
4157
Fariborz Jahanian30bc5712009-01-22 23:02:58 +00004158 // struct _prop_t {
4159 // char *name;
Daniel Dunbar6bff2512009-08-03 17:06:42 +00004160 // char *attributes;
Fariborz Jahanian30bc5712009-01-22 23:02:58 +00004161 // }
Owen Anderson47a434f2009-08-05 23:18:46 +00004162 PropertyTy = llvm::StructType::get(VMContext, Int8PtrTy, Int8PtrTy, NULL);
Daniel Dunbar6bff2512009-08-03 17:06:42 +00004163 CGM.getModule().addTypeName("struct._prop_t",
Fariborz Jahanian30bc5712009-01-22 23:02:58 +00004164 PropertyTy);
Daniel Dunbar6bff2512009-08-03 17:06:42 +00004165
Fariborz Jahanian30bc5712009-01-22 23:02:58 +00004166 // struct _prop_list_t {
Fariborz Jahaniand55b6fc2009-01-23 01:46:23 +00004167 // uint32_t entsize; // sizeof(struct _prop_t)
Fariborz Jahanian30bc5712009-01-22 23:02:58 +00004168 // uint32_t count_of_properties;
Fariborz Jahaniand55b6fc2009-01-23 01:46:23 +00004169 // struct _prop_t prop_list[count_of_properties];
Fariborz Jahanian30bc5712009-01-22 23:02:58 +00004170 // }
Owen Anderson47a434f2009-08-05 23:18:46 +00004171 PropertyListTy = llvm::StructType::get(VMContext, IntTy,
Fariborz Jahanian30bc5712009-01-22 23:02:58 +00004172 IntTy,
Owen Anderson96e0fc72009-07-29 22:16:19 +00004173 llvm::ArrayType::get(PropertyTy, 0),
Fariborz Jahanian30bc5712009-01-22 23:02:58 +00004174 NULL);
Daniel Dunbar6bff2512009-08-03 17:06:42 +00004175 CGM.getModule().addTypeName("struct._prop_list_t",
Fariborz Jahanian30bc5712009-01-22 23:02:58 +00004176 PropertyListTy);
4177 // struct _prop_list_t *
Owen Anderson96e0fc72009-07-29 22:16:19 +00004178 PropertyListPtrTy = llvm::PointerType::getUnqual(PropertyListTy);
Daniel Dunbar6bff2512009-08-03 17:06:42 +00004179
Fariborz Jahanian30bc5712009-01-22 23:02:58 +00004180 // struct _objc_method {
4181 // SEL _cmd;
4182 // char *method_type;
4183 // char *_imp;
4184 // }
Owen Anderson47a434f2009-08-05 23:18:46 +00004185 MethodTy = llvm::StructType::get(VMContext, SelectorPtrTy,
Fariborz Jahanian30bc5712009-01-22 23:02:58 +00004186 Int8PtrTy,
4187 Int8PtrTy,
4188 NULL);
4189 CGM.getModule().addTypeName("struct._objc_method", MethodTy);
Daniel Dunbar6bff2512009-08-03 17:06:42 +00004190
Fariborz Jahaniand55b6fc2009-01-23 01:46:23 +00004191 // struct _objc_cache *
Owen Anderson8c8f69e2009-08-13 23:27:53 +00004192 CacheTy = llvm::OpaqueType::get(VMContext);
Fariborz Jahaniand55b6fc2009-01-23 01:46:23 +00004193 CGM.getModule().addTypeName("struct._objc_cache", CacheTy);
Owen Anderson96e0fc72009-07-29 22:16:19 +00004194 CachePtrTy = llvm::PointerType::getUnqual(CacheTy);
Fariborz Jahanianee0af742009-01-21 22:04:16 +00004195}
Daniel Dunbar4e2d7d02008-08-12 06:48:42 +00004196
Daniel Dunbar6bff2512009-08-03 17:06:42 +00004197ObjCTypesHelper::ObjCTypesHelper(CodeGen::CodeGenModule &cgm)
Mike Stump1eb44332009-09-09 15:08:12 +00004198 : ObjCCommonTypesHelper(cgm) {
Fariborz Jahanian10a42312009-01-21 00:39:53 +00004199 // struct _objc_method_description {
4200 // SEL name;
4201 // char *types;
4202 // }
Daniel Dunbar6bff2512009-08-03 17:06:42 +00004203 MethodDescriptionTy =
Owen Anderson47a434f2009-08-05 23:18:46 +00004204 llvm::StructType::get(VMContext, SelectorPtrTy,
Daniel Dunbar27f9d772008-08-21 04:36:09 +00004205 Int8PtrTy,
Daniel Dunbar6efc0c52008-08-13 03:21:16 +00004206 NULL);
Daniel Dunbar6bff2512009-08-03 17:06:42 +00004207 CGM.getModule().addTypeName("struct._objc_method_description",
Daniel Dunbar6efc0c52008-08-13 03:21:16 +00004208 MethodDescriptionTy);
4209
Fariborz Jahanian10a42312009-01-21 00:39:53 +00004210 // struct _objc_method_description_list {
4211 // int count;
4212 // struct _objc_method_description[1];
4213 // }
Daniel Dunbar6bff2512009-08-03 17:06:42 +00004214 MethodDescriptionListTy =
Owen Anderson47a434f2009-08-05 23:18:46 +00004215 llvm::StructType::get(VMContext, IntTy,
Owen Anderson96e0fc72009-07-29 22:16:19 +00004216 llvm::ArrayType::get(MethodDescriptionTy, 0),
Daniel Dunbar6efc0c52008-08-13 03:21:16 +00004217 NULL);
Daniel Dunbar6bff2512009-08-03 17:06:42 +00004218 CGM.getModule().addTypeName("struct._objc_method_description_list",
Daniel Dunbar6efc0c52008-08-13 03:21:16 +00004219 MethodDescriptionListTy);
Daniel Dunbar6bff2512009-08-03 17:06:42 +00004220
Fariborz Jahanian10a42312009-01-21 00:39:53 +00004221 // struct _objc_method_description_list *
Daniel Dunbar6bff2512009-08-03 17:06:42 +00004222 MethodDescriptionListPtrTy =
Owen Anderson96e0fc72009-07-29 22:16:19 +00004223 llvm::PointerType::getUnqual(MethodDescriptionListTy);
Daniel Dunbar6efc0c52008-08-13 03:21:16 +00004224
Daniel Dunbar6efc0c52008-08-13 03:21:16 +00004225 // Protocol description structures
4226
Fariborz Jahanian10a42312009-01-21 00:39:53 +00004227 // struct _objc_protocol_extension {
4228 // uint32_t size; // sizeof(struct _objc_protocol_extension)
4229 // struct _objc_method_description_list *optional_instance_methods;
4230 // struct _objc_method_description_list *optional_class_methods;
4231 // struct _objc_property_list *instance_properties;
4232 // }
Daniel Dunbar6bff2512009-08-03 17:06:42 +00004233 ProtocolExtensionTy =
Owen Anderson47a434f2009-08-05 23:18:46 +00004234 llvm::StructType::get(VMContext, IntTy,
Fariborz Jahanian10a42312009-01-21 00:39:53 +00004235 MethodDescriptionListPtrTy,
4236 MethodDescriptionListPtrTy,
Daniel Dunbar6efc0c52008-08-13 03:21:16 +00004237 PropertyListPtrTy,
4238 NULL);
Daniel Dunbar6bff2512009-08-03 17:06:42 +00004239 CGM.getModule().addTypeName("struct._objc_protocol_extension",
Daniel Dunbar6efc0c52008-08-13 03:21:16 +00004240 ProtocolExtensionTy);
Daniel Dunbar6bff2512009-08-03 17:06:42 +00004241
Fariborz Jahanian10a42312009-01-21 00:39:53 +00004242 // struct _objc_protocol_extension *
Owen Anderson96e0fc72009-07-29 22:16:19 +00004243 ProtocolExtensionPtrTy = llvm::PointerType::getUnqual(ProtocolExtensionTy);
Daniel Dunbar6efc0c52008-08-13 03:21:16 +00004244
Daniel Dunbar0c0e7a62008-10-29 22:36:39 +00004245 // Handle recursive construction of Protocol and ProtocolList types
Daniel Dunbar6efc0c52008-08-13 03:21:16 +00004246
Owen Anderson8c8f69e2009-08-13 23:27:53 +00004247 llvm::PATypeHolder ProtocolTyHolder = llvm::OpaqueType::get(VMContext);
4248 llvm::PATypeHolder ProtocolListTyHolder = llvm::OpaqueType::get(VMContext);
Daniel Dunbar6efc0c52008-08-13 03:21:16 +00004249
Daniel Dunbar6bff2512009-08-03 17:06:42 +00004250 const llvm::Type *T =
Mike Stump1eb44332009-09-09 15:08:12 +00004251 llvm::StructType::get(VMContext,
Owen Anderson47a434f2009-08-05 23:18:46 +00004252 llvm::PointerType::getUnqual(ProtocolListTyHolder),
Fariborz Jahanianee0af742009-01-21 22:04:16 +00004253 LongTy,
Owen Anderson96e0fc72009-07-29 22:16:19 +00004254 llvm::ArrayType::get(ProtocolTyHolder, 0),
Fariborz Jahanianee0af742009-01-21 22:04:16 +00004255 NULL);
Daniel Dunbar6efc0c52008-08-13 03:21:16 +00004256 cast<llvm::OpaqueType>(ProtocolListTyHolder.get())->refineAbstractTypeTo(T);
4257
Fariborz Jahanian10a42312009-01-21 00:39:53 +00004258 // struct _objc_protocol {
4259 // struct _objc_protocol_extension *isa;
4260 // char *protocol_name;
4261 // struct _objc_protocol **_objc_protocol_list;
4262 // struct _objc_method_description_list *instance_methods;
4263 // struct _objc_method_description_list *class_methods;
4264 // }
Owen Anderson47a434f2009-08-05 23:18:46 +00004265 T = llvm::StructType::get(VMContext, ProtocolExtensionPtrTy,
Daniel Dunbar27f9d772008-08-21 04:36:09 +00004266 Int8PtrTy,
Daniel Dunbar6bff2512009-08-03 17:06:42 +00004267 llvm::PointerType::getUnqual(ProtocolListTyHolder),
Daniel Dunbar6efc0c52008-08-13 03:21:16 +00004268 MethodDescriptionListPtrTy,
4269 MethodDescriptionListPtrTy,
4270 NULL);
4271 cast<llvm::OpaqueType>(ProtocolTyHolder.get())->refineAbstractTypeTo(T);
4272
4273 ProtocolListTy = cast<llvm::StructType>(ProtocolListTyHolder.get());
Daniel Dunbar6bff2512009-08-03 17:06:42 +00004274 CGM.getModule().addTypeName("struct._objc_protocol_list",
Daniel Dunbar6efc0c52008-08-13 03:21:16 +00004275 ProtocolListTy);
Fariborz Jahanian10a42312009-01-21 00:39:53 +00004276 // struct _objc_protocol_list *
Owen Anderson96e0fc72009-07-29 22:16:19 +00004277 ProtocolListPtrTy = llvm::PointerType::getUnqual(ProtocolListTy);
Daniel Dunbar6efc0c52008-08-13 03:21:16 +00004278
4279 ProtocolTy = cast<llvm::StructType>(ProtocolTyHolder.get());
Fariborz Jahanian10a42312009-01-21 00:39:53 +00004280 CGM.getModule().addTypeName("struct._objc_protocol", ProtocolTy);
Owen Anderson96e0fc72009-07-29 22:16:19 +00004281 ProtocolPtrTy = llvm::PointerType::getUnqual(ProtocolTy);
Daniel Dunbar27f9d772008-08-21 04:36:09 +00004282
4283 // Class description structures
4284
Fariborz Jahanian10a42312009-01-21 00:39:53 +00004285 // struct _objc_ivar {
4286 // char *ivar_name;
4287 // char *ivar_type;
4288 // int ivar_offset;
4289 // }
Owen Anderson47a434f2009-08-05 23:18:46 +00004290 IvarTy = llvm::StructType::get(VMContext, Int8PtrTy,
Daniel Dunbar6bff2512009-08-03 17:06:42 +00004291 Int8PtrTy,
4292 IntTy,
Daniel Dunbar27f9d772008-08-21 04:36:09 +00004293 NULL);
4294 CGM.getModule().addTypeName("struct._objc_ivar", IvarTy);
4295
Fariborz Jahanian10a42312009-01-21 00:39:53 +00004296 // struct _objc_ivar_list *
Owen Anderson8c8f69e2009-08-13 23:27:53 +00004297 IvarListTy = llvm::OpaqueType::get(VMContext);
Daniel Dunbar27f9d772008-08-21 04:36:09 +00004298 CGM.getModule().addTypeName("struct._objc_ivar_list", IvarListTy);
Owen Anderson96e0fc72009-07-29 22:16:19 +00004299 IvarListPtrTy = llvm::PointerType::getUnqual(IvarListTy);
Daniel Dunbar27f9d772008-08-21 04:36:09 +00004300
Fariborz Jahanian10a42312009-01-21 00:39:53 +00004301 // struct _objc_method_list *
Owen Anderson8c8f69e2009-08-13 23:27:53 +00004302 MethodListTy = llvm::OpaqueType::get(VMContext);
Daniel Dunbar27f9d772008-08-21 04:36:09 +00004303 CGM.getModule().addTypeName("struct._objc_method_list", MethodListTy);
Owen Anderson96e0fc72009-07-29 22:16:19 +00004304 MethodListPtrTy = llvm::PointerType::getUnqual(MethodListTy);
Daniel Dunbar27f9d772008-08-21 04:36:09 +00004305
Fariborz Jahanian10a42312009-01-21 00:39:53 +00004306 // struct _objc_class_extension *
Daniel Dunbar6bff2512009-08-03 17:06:42 +00004307 ClassExtensionTy =
Owen Anderson47a434f2009-08-05 23:18:46 +00004308 llvm::StructType::get(VMContext, IntTy,
Daniel Dunbar27f9d772008-08-21 04:36:09 +00004309 Int8PtrTy,
4310 PropertyListPtrTy,
4311 NULL);
4312 CGM.getModule().addTypeName("struct._objc_class_extension", ClassExtensionTy);
Owen Anderson96e0fc72009-07-29 22:16:19 +00004313 ClassExtensionPtrTy = llvm::PointerType::getUnqual(ClassExtensionTy);
Daniel Dunbar27f9d772008-08-21 04:36:09 +00004314
Owen Anderson8c8f69e2009-08-13 23:27:53 +00004315 llvm::PATypeHolder ClassTyHolder = llvm::OpaqueType::get(VMContext);
Daniel Dunbar27f9d772008-08-21 04:36:09 +00004316
Fariborz Jahanian10a42312009-01-21 00:39:53 +00004317 // struct _objc_class {
4318 // Class isa;
4319 // Class super_class;
4320 // char *name;
4321 // long version;
4322 // long info;
4323 // long instance_size;
4324 // struct _objc_ivar_list *ivars;
4325 // struct _objc_method_list *methods;
4326 // struct _objc_cache *cache;
4327 // struct _objc_protocol_list *protocols;
4328 // char *ivar_layout;
4329 // struct _objc_class_ext *ext;
4330 // };
Owen Anderson47a434f2009-08-05 23:18:46 +00004331 T = llvm::StructType::get(VMContext,
4332 llvm::PointerType::getUnqual(ClassTyHolder),
Owen Anderson96e0fc72009-07-29 22:16:19 +00004333 llvm::PointerType::getUnqual(ClassTyHolder),
Daniel Dunbar27f9d772008-08-21 04:36:09 +00004334 Int8PtrTy,
4335 LongTy,
4336 LongTy,
4337 LongTy,
4338 IvarListPtrTy,
4339 MethodListPtrTy,
4340 CachePtrTy,
4341 ProtocolListPtrTy,
4342 Int8PtrTy,
4343 ClassExtensionPtrTy,
4344 NULL);
4345 cast<llvm::OpaqueType>(ClassTyHolder.get())->refineAbstractTypeTo(T);
Daniel Dunbar6bff2512009-08-03 17:06:42 +00004346
Daniel Dunbar27f9d772008-08-21 04:36:09 +00004347 ClassTy = cast<llvm::StructType>(ClassTyHolder.get());
4348 CGM.getModule().addTypeName("struct._objc_class", ClassTy);
Owen Anderson96e0fc72009-07-29 22:16:19 +00004349 ClassPtrTy = llvm::PointerType::getUnqual(ClassTy);
Daniel Dunbar27f9d772008-08-21 04:36:09 +00004350
Fariborz Jahanian10a42312009-01-21 00:39:53 +00004351 // struct _objc_category {
4352 // char *category_name;
4353 // char *class_name;
4354 // struct _objc_method_list *instance_method;
4355 // struct _objc_method_list *class_method;
4356 // uint32_t size; // sizeof(struct _objc_category)
4357 // struct _objc_property_list *instance_properties;// category's @property
4358 // }
Owen Anderson47a434f2009-08-05 23:18:46 +00004359 CategoryTy = llvm::StructType::get(VMContext, Int8PtrTy,
Daniel Dunbar86e253a2008-08-22 20:34:54 +00004360 Int8PtrTy,
4361 MethodListPtrTy,
4362 MethodListPtrTy,
4363 ProtocolListPtrTy,
4364 IntTy,
4365 PropertyListPtrTy,
4366 NULL);
4367 CGM.getModule().addTypeName("struct._objc_category", CategoryTy);
4368
Daniel Dunbar27f9d772008-08-21 04:36:09 +00004369 // Global metadata structures
4370
Fariborz Jahanian10a42312009-01-21 00:39:53 +00004371 // struct _objc_symtab {
4372 // long sel_ref_cnt;
4373 // SEL *refs;
4374 // short cls_def_cnt;
4375 // short cat_def_cnt;
4376 // char *defs[cls_def_cnt + cat_def_cnt];
4377 // }
Owen Anderson47a434f2009-08-05 23:18:46 +00004378 SymtabTy = llvm::StructType::get(VMContext, LongTy,
Daniel Dunbar27f9d772008-08-21 04:36:09 +00004379 SelectorPtrTy,
4380 ShortTy,
4381 ShortTy,
Owen Anderson96e0fc72009-07-29 22:16:19 +00004382 llvm::ArrayType::get(Int8PtrTy, 0),
Daniel Dunbar27f9d772008-08-21 04:36:09 +00004383 NULL);
4384 CGM.getModule().addTypeName("struct._objc_symtab", SymtabTy);
Owen Anderson96e0fc72009-07-29 22:16:19 +00004385 SymtabPtrTy = llvm::PointerType::getUnqual(SymtabTy);
Daniel Dunbar27f9d772008-08-21 04:36:09 +00004386
Fariborz Jahaniandb286862009-01-22 00:37:21 +00004387 // struct _objc_module {
4388 // long version;
4389 // long size; // sizeof(struct _objc_module)
4390 // char *name;
4391 // struct _objc_symtab* symtab;
4392 // }
Daniel Dunbar6bff2512009-08-03 17:06:42 +00004393 ModuleTy =
Owen Anderson47a434f2009-08-05 23:18:46 +00004394 llvm::StructType::get(VMContext, LongTy,
Daniel Dunbar27f9d772008-08-21 04:36:09 +00004395 LongTy,
4396 Int8PtrTy,
4397 SymtabPtrTy,
4398 NULL);
4399 CGM.getModule().addTypeName("struct._objc_module", ModuleTy);
Daniel Dunbar14c80b72008-08-23 09:25:55 +00004400
Daniel Dunbar6bff2512009-08-03 17:06:42 +00004401
Mike Stumpf5408fe2009-05-16 07:57:57 +00004402 // FIXME: This is the size of the setjmp buffer and should be target
4403 // specific. 18 is what's used on 32-bit X86.
Anders Carlsson124526b2008-09-09 10:10:21 +00004404 uint64_t SetJmpBufferSize = 18;
Daniel Dunbar6bff2512009-08-03 17:06:42 +00004405
Anders Carlsson124526b2008-09-09 10:10:21 +00004406 // Exceptions
Owen Anderson96e0fc72009-07-29 22:16:19 +00004407 const llvm::Type *StackPtrTy = llvm::ArrayType::get(
Benjamin Kramer3c0ef8c2009-10-13 10:07:13 +00004408 llvm::Type::getInt8PtrTy(VMContext), 4);
Daniel Dunbar6bff2512009-08-03 17:06:42 +00004409
4410 ExceptionDataTy =
Chris Lattner77b89b82010-06-27 07:15:29 +00004411 llvm::StructType::get(VMContext,
4412 llvm::ArrayType::get(llvm::Type::getInt32Ty(VMContext),
4413 SetJmpBufferSize),
Anders Carlsson124526b2008-09-09 10:10:21 +00004414 StackPtrTy, NULL);
Daniel Dunbar6bff2512009-08-03 17:06:42 +00004415 CGM.getModule().addTypeName("struct._objc_exception_data",
Anders Carlsson124526b2008-09-09 10:10:21 +00004416 ExceptionDataTy);
4417
Daniel Dunbarbbce49b2008-08-12 00:12:39 +00004418}
4419
Daniel Dunbar6bff2512009-08-03 17:06:42 +00004420ObjCNonFragileABITypesHelper::ObjCNonFragileABITypesHelper(CodeGen::CodeGenModule &cgm)
Mike Stump1eb44332009-09-09 15:08:12 +00004421 : ObjCCommonTypesHelper(cgm) {
Fariborz Jahanian30bc5712009-01-22 23:02:58 +00004422 // struct _method_list_t {
4423 // uint32_t entsize; // sizeof(struct _objc_method)
4424 // uint32_t method_count;
4425 // struct _objc_method method_list[method_count];
4426 // }
Owen Anderson47a434f2009-08-05 23:18:46 +00004427 MethodListnfABITy = llvm::StructType::get(VMContext, IntTy,
Fariborz Jahaniand55b6fc2009-01-23 01:46:23 +00004428 IntTy,
Owen Anderson96e0fc72009-07-29 22:16:19 +00004429 llvm::ArrayType::get(MethodTy, 0),
Fariborz Jahaniand55b6fc2009-01-23 01:46:23 +00004430 NULL);
4431 CGM.getModule().addTypeName("struct.__method_list_t",
4432 MethodListnfABITy);
4433 // struct method_list_t *
Owen Anderson96e0fc72009-07-29 22:16:19 +00004434 MethodListnfABIPtrTy = llvm::PointerType::getUnqual(MethodListnfABITy);
Daniel Dunbar6bff2512009-08-03 17:06:42 +00004435
Fariborz Jahanian30bc5712009-01-22 23:02:58 +00004436 // struct _protocol_t {
4437 // id isa; // NULL
4438 // const char * const protocol_name;
Fariborz Jahaniand55b6fc2009-01-23 01:46:23 +00004439 // const struct _protocol_list_t * protocol_list; // super protocols
Fariborz Jahanian30bc5712009-01-22 23:02:58 +00004440 // const struct method_list_t * const instance_methods;
4441 // const struct method_list_t * const class_methods;
4442 // const struct method_list_t *optionalInstanceMethods;
4443 // const struct method_list_t *optionalClassMethods;
Fariborz Jahaniand55b6fc2009-01-23 01:46:23 +00004444 // const struct _prop_list_t * properties;
Fariborz Jahanian30bc5712009-01-22 23:02:58 +00004445 // const uint32_t size; // sizeof(struct _protocol_t)
4446 // const uint32_t flags; // = 0
4447 // }
Daniel Dunbar6bff2512009-08-03 17:06:42 +00004448
Fariborz Jahaniand55b6fc2009-01-23 01:46:23 +00004449 // Holder for struct _protocol_list_t *
Owen Anderson8c8f69e2009-08-13 23:27:53 +00004450 llvm::PATypeHolder ProtocolListTyHolder = llvm::OpaqueType::get(VMContext);
Daniel Dunbar6bff2512009-08-03 17:06:42 +00004451
Owen Anderson47a434f2009-08-05 23:18:46 +00004452 ProtocolnfABITy = llvm::StructType::get(VMContext, ObjectPtrTy,
Fariborz Jahaniand55b6fc2009-01-23 01:46:23 +00004453 Int8PtrTy,
Owen Anderson96e0fc72009-07-29 22:16:19 +00004454 llvm::PointerType::getUnqual(
Fariborz Jahaniand55b6fc2009-01-23 01:46:23 +00004455 ProtocolListTyHolder),
4456 MethodListnfABIPtrTy,
4457 MethodListnfABIPtrTy,
4458 MethodListnfABIPtrTy,
4459 MethodListnfABIPtrTy,
4460 PropertyListPtrTy,
4461 IntTy,
4462 IntTy,
4463 NULL);
4464 CGM.getModule().addTypeName("struct._protocol_t",
4465 ProtocolnfABITy);
Daniel Dunbar948e2582009-02-15 07:36:20 +00004466
4467 // struct _protocol_t*
Owen Anderson96e0fc72009-07-29 22:16:19 +00004468 ProtocolnfABIPtrTy = llvm::PointerType::getUnqual(ProtocolnfABITy);
Daniel Dunbar6bff2512009-08-03 17:06:42 +00004469
Fariborz Jahanianda320092009-01-29 19:24:30 +00004470 // struct _protocol_list_t {
Fariborz Jahaniand55b6fc2009-01-23 01:46:23 +00004471 // long protocol_count; // Note, this is 32/64 bit
Daniel Dunbar948e2582009-02-15 07:36:20 +00004472 // struct _protocol_t *[protocol_count];
Fariborz Jahanian30bc5712009-01-22 23:02:58 +00004473 // }
Owen Anderson47a434f2009-08-05 23:18:46 +00004474 ProtocolListnfABITy = llvm::StructType::get(VMContext, LongTy,
Owen Anderson96e0fc72009-07-29 22:16:19 +00004475 llvm::ArrayType::get(
Daniel Dunbar948e2582009-02-15 07:36:20 +00004476 ProtocolnfABIPtrTy, 0),
Fariborz Jahaniand55b6fc2009-01-23 01:46:23 +00004477 NULL);
4478 CGM.getModule().addTypeName("struct._objc_protocol_list",
4479 ProtocolListnfABITy);
Daniel Dunbar948e2582009-02-15 07:36:20 +00004480 cast<llvm::OpaqueType>(ProtocolListTyHolder.get())->refineAbstractTypeTo(
Daniel Dunbar6bff2512009-08-03 17:06:42 +00004481 ProtocolListnfABITy);
4482
Fariborz Jahaniand55b6fc2009-01-23 01:46:23 +00004483 // struct _objc_protocol_list*
Owen Anderson96e0fc72009-07-29 22:16:19 +00004484 ProtocolListnfABIPtrTy = llvm::PointerType::getUnqual(ProtocolListnfABITy);
Daniel Dunbar6bff2512009-08-03 17:06:42 +00004485
Fariborz Jahanian30bc5712009-01-22 23:02:58 +00004486 // struct _ivar_t {
4487 // unsigned long int *offset; // pointer to ivar offset location
4488 // char *name;
4489 // char *type;
4490 // uint32_t alignment;
4491 // uint32_t size;
4492 // }
Mike Stump1eb44332009-09-09 15:08:12 +00004493 IvarnfABITy = llvm::StructType::get(VMContext,
Owen Anderson47a434f2009-08-05 23:18:46 +00004494 llvm::PointerType::getUnqual(LongTy),
Fariborz Jahaniand55b6fc2009-01-23 01:46:23 +00004495 Int8PtrTy,
4496 Int8PtrTy,
4497 IntTy,
4498 IntTy,
4499 NULL);
4500 CGM.getModule().addTypeName("struct._ivar_t", IvarnfABITy);
Daniel Dunbar6bff2512009-08-03 17:06:42 +00004501
Fariborz Jahanian30bc5712009-01-22 23:02:58 +00004502 // struct _ivar_list_t {
4503 // uint32 entsize; // sizeof(struct _ivar_t)
4504 // uint32 count;
4505 // struct _iver_t list[count];
4506 // }
Owen Anderson47a434f2009-08-05 23:18:46 +00004507 IvarListnfABITy = llvm::StructType::get(VMContext, IntTy,
Fariborz Jahanian058a1b72009-01-24 20:21:50 +00004508 IntTy,
Owen Anderson96e0fc72009-07-29 22:16:19 +00004509 llvm::ArrayType::get(
Daniel Dunbar6bff2512009-08-03 17:06:42 +00004510 IvarnfABITy, 0),
Fariborz Jahanian058a1b72009-01-24 20:21:50 +00004511 NULL);
4512 CGM.getModule().addTypeName("struct._ivar_list_t", IvarListnfABITy);
Daniel Dunbar6bff2512009-08-03 17:06:42 +00004513
Owen Anderson96e0fc72009-07-29 22:16:19 +00004514 IvarListnfABIPtrTy = llvm::PointerType::getUnqual(IvarListnfABITy);
Daniel Dunbar6bff2512009-08-03 17:06:42 +00004515
Fariborz Jahaniand55b6fc2009-01-23 01:46:23 +00004516 // struct _class_ro_t {
Fariborz Jahanian30bc5712009-01-22 23:02:58 +00004517 // uint32_t const flags;
4518 // uint32_t const instanceStart;
4519 // uint32_t const instanceSize;
4520 // uint32_t const reserved; // only when building for 64bit targets
4521 // const uint8_t * const ivarLayout;
4522 // const char *const name;
4523 // const struct _method_list_t * const baseMethods;
4524 // const struct _objc_protocol_list *const baseProtocols;
4525 // const struct _ivar_list_t *const ivars;
4526 // const uint8_t * const weakIvarLayout;
4527 // const struct _prop_list_t * const properties;
4528 // }
Daniel Dunbar6bff2512009-08-03 17:06:42 +00004529
Fariborz Jahaniand55b6fc2009-01-23 01:46:23 +00004530 // FIXME. Add 'reserved' field in 64bit abi mode!
Owen Anderson47a434f2009-08-05 23:18:46 +00004531 ClassRonfABITy = llvm::StructType::get(VMContext, IntTy,
Fariborz Jahaniand55b6fc2009-01-23 01:46:23 +00004532 IntTy,
4533 IntTy,
4534 Int8PtrTy,
4535 Int8PtrTy,
4536 MethodListnfABIPtrTy,
4537 ProtocolListnfABIPtrTy,
4538 IvarListnfABIPtrTy,
4539 Int8PtrTy,
4540 PropertyListPtrTy,
4541 NULL);
4542 CGM.getModule().addTypeName("struct._class_ro_t",
4543 ClassRonfABITy);
Daniel Dunbar6bff2512009-08-03 17:06:42 +00004544
Fariborz Jahaniand55b6fc2009-01-23 01:46:23 +00004545 // ImpnfABITy - LLVM for id (*)(id, SEL, ...)
4546 std::vector<const llvm::Type*> Params;
4547 Params.push_back(ObjectPtrTy);
4548 Params.push_back(SelectorPtrTy);
Owen Anderson96e0fc72009-07-29 22:16:19 +00004549 ImpnfABITy = llvm::PointerType::getUnqual(
Daniel Dunbar6bff2512009-08-03 17:06:42 +00004550 llvm::FunctionType::get(ObjectPtrTy, Params, false));
4551
Fariborz Jahaniand55b6fc2009-01-23 01:46:23 +00004552 // struct _class_t {
4553 // struct _class_t *isa;
4554 // struct _class_t * const superclass;
4555 // void *cache;
4556 // IMP *vtable;
4557 // struct class_ro_t *ro;
4558 // }
Daniel Dunbar6bff2512009-08-03 17:06:42 +00004559
Owen Anderson8c8f69e2009-08-13 23:27:53 +00004560 llvm::PATypeHolder ClassTyHolder = llvm::OpaqueType::get(VMContext);
Owen Andersona1cf15f2009-07-14 23:10:40 +00004561 ClassnfABITy =
Owen Anderson47a434f2009-08-05 23:18:46 +00004562 llvm::StructType::get(VMContext,
4563 llvm::PointerType::getUnqual(ClassTyHolder),
Daniel Dunbar6bff2512009-08-03 17:06:42 +00004564 llvm::PointerType::getUnqual(ClassTyHolder),
4565 CachePtrTy,
4566 llvm::PointerType::getUnqual(ImpnfABITy),
4567 llvm::PointerType::getUnqual(ClassRonfABITy),
4568 NULL);
Fariborz Jahaniand55b6fc2009-01-23 01:46:23 +00004569 CGM.getModule().addTypeName("struct._class_t", ClassnfABITy);
4570
4571 cast<llvm::OpaqueType>(ClassTyHolder.get())->refineAbstractTypeTo(
Daniel Dunbar6bff2512009-08-03 17:06:42 +00004572 ClassnfABITy);
4573
Fariborz Jahanianaa23b572009-01-23 23:53:38 +00004574 // LLVM for struct _class_t *
Owen Anderson96e0fc72009-07-29 22:16:19 +00004575 ClassnfABIPtrTy = llvm::PointerType::getUnqual(ClassnfABITy);
Daniel Dunbar6bff2512009-08-03 17:06:42 +00004576
Fariborz Jahaniand55b6fc2009-01-23 01:46:23 +00004577 // struct _category_t {
4578 // const char * const name;
4579 // struct _class_t *const cls;
4580 // const struct _method_list_t * const instance_methods;
4581 // const struct _method_list_t * const class_methods;
4582 // const struct _protocol_list_t * const protocols;
4583 // const struct _prop_list_t * const properties;
Fariborz Jahanian45c2ba02009-01-23 17:41:22 +00004584 // }
Owen Anderson47a434f2009-08-05 23:18:46 +00004585 CategorynfABITy = llvm::StructType::get(VMContext, Int8PtrTy,
Fariborz Jahanianaa23b572009-01-23 23:53:38 +00004586 ClassnfABIPtrTy,
Fariborz Jahanian45c2ba02009-01-23 17:41:22 +00004587 MethodListnfABIPtrTy,
4588 MethodListnfABIPtrTy,
4589 ProtocolListnfABIPtrTy,
4590 PropertyListPtrTy,
4591 NULL);
4592 CGM.getModule().addTypeName("struct._category_t", CategorynfABITy);
Daniel Dunbar6bff2512009-08-03 17:06:42 +00004593
Fariborz Jahanian2e4672b2009-02-03 23:49:23 +00004594 // New types for nonfragile abi messaging.
Fariborz Jahanian83a8a752009-02-04 20:42:28 +00004595 CodeGen::CodeGenTypes &Types = CGM.getTypes();
4596 ASTContext &Ctx = CGM.getContext();
Daniel Dunbar6bff2512009-08-03 17:06:42 +00004597
Fariborz Jahanian2e4672b2009-02-03 23:49:23 +00004598 // MessageRefTy - LLVM for:
4599 // struct _message_ref_t {
4600 // IMP messenger;
4601 // SEL name;
4602 // };
Daniel Dunbar6bff2512009-08-03 17:06:42 +00004603
Fariborz Jahanian83a8a752009-02-04 20:42:28 +00004604 // First the clang type for struct _message_ref_t
Abramo Bagnara465d41b2010-05-11 21:36:43 +00004605 RecordDecl *RD = RecordDecl::Create(Ctx, TTK_Struct,
Daniel Dunbardaa3ac52010-04-29 16:29:11 +00004606 Ctx.getTranslationUnitDecl(),
Fariborz Jahanian83a8a752009-02-04 20:42:28 +00004607 SourceLocation(),
4608 &Ctx.Idents.get("_message_ref_t"));
Argyrios Kyrtzidis17945a02009-06-30 02:36:12 +00004609 RD->addDecl(FieldDecl::Create(Ctx, RD, SourceLocation(), 0,
Argyrios Kyrtzidisa1d56622009-08-19 01:27:57 +00004610 Ctx.VoidPtrTy, 0, 0, false));
Argyrios Kyrtzidis17945a02009-06-30 02:36:12 +00004611 RD->addDecl(FieldDecl::Create(Ctx, RD, SourceLocation(), 0,
Argyrios Kyrtzidisa1d56622009-08-19 01:27:57 +00004612 Ctx.getObjCSelType(), 0, 0, false));
Douglas Gregor838db382010-02-11 01:19:42 +00004613 RD->completeDefinition();
Daniel Dunbar6bff2512009-08-03 17:06:42 +00004614
Fariborz Jahanian83a8a752009-02-04 20:42:28 +00004615 MessageRefCTy = Ctx.getTagDeclType(RD);
4616 MessageRefCPtrTy = Ctx.getPointerType(MessageRefCTy);
4617 MessageRefTy = cast<llvm::StructType>(Types.ConvertType(MessageRefCTy));
Daniel Dunbar6bff2512009-08-03 17:06:42 +00004618
Fariborz Jahanian2e4672b2009-02-03 23:49:23 +00004619 // MessageRefPtrTy - LLVM for struct _message_ref_t*
Owen Anderson96e0fc72009-07-29 22:16:19 +00004620 MessageRefPtrTy = llvm::PointerType::getUnqual(MessageRefTy);
Daniel Dunbar6bff2512009-08-03 17:06:42 +00004621
Fariborz Jahanian2e4672b2009-02-03 23:49:23 +00004622 // SuperMessageRefTy - LLVM for:
4623 // struct _super_message_ref_t {
4624 // SUPER_IMP messenger;
4625 // SEL name;
4626 // };
Owen Anderson47a434f2009-08-05 23:18:46 +00004627 SuperMessageRefTy = llvm::StructType::get(VMContext, ImpnfABITy,
Fariborz Jahanian2e4672b2009-02-03 23:49:23 +00004628 SelectorPtrTy,
4629 NULL);
4630 CGM.getModule().addTypeName("struct._super_message_ref_t", SuperMessageRefTy);
Daniel Dunbar6bff2512009-08-03 17:06:42 +00004631
Fariborz Jahanian2e4672b2009-02-03 23:49:23 +00004632 // SuperMessageRefPtrTy - LLVM for struct _super_message_ref_t*
Daniel Dunbar6bff2512009-08-03 17:06:42 +00004633 SuperMessageRefPtrTy = llvm::PointerType::getUnqual(SuperMessageRefTy);
4634
Daniel Dunbare588b992009-03-01 04:46:24 +00004635
4636 // struct objc_typeinfo {
4637 // const void** vtable; // objc_ehtype_vtable + 2
4638 // const char* name; // c++ typeinfo string
4639 // Class cls;
4640 // };
Owen Anderson47a434f2009-08-05 23:18:46 +00004641 EHTypeTy = llvm::StructType::get(VMContext,
4642 llvm::PointerType::getUnqual(Int8PtrTy),
Daniel Dunbare588b992009-03-01 04:46:24 +00004643 Int8PtrTy,
4644 ClassnfABIPtrTy,
4645 NULL);
Daniel Dunbar4ff36842009-03-02 06:08:11 +00004646 CGM.getModule().addTypeName("struct._objc_typeinfo", EHTypeTy);
Owen Anderson96e0fc72009-07-29 22:16:19 +00004647 EHTypePtrTy = llvm::PointerType::getUnqual(EHTypeTy);
Daniel Dunbarbbce49b2008-08-12 00:12:39 +00004648}
4649
Daniel Dunbar6bff2512009-08-03 17:06:42 +00004650llvm::Function *CGObjCNonFragileABIMac::ModuleInitFunction() {
Fariborz Jahanianaa23b572009-01-23 23:53:38 +00004651 FinishNonFragileABIModule();
Daniel Dunbar6bff2512009-08-03 17:06:42 +00004652
Fariborz Jahanianaa23b572009-01-23 23:53:38 +00004653 return NULL;
4654}
4655
Daniel Dunbar6bff2512009-08-03 17:06:42 +00004656void CGObjCNonFragileABIMac::AddModuleClassList(const
4657 std::vector<llvm::GlobalValue*>
4658 &Container,
Daniel Dunbar463b8762009-05-15 21:48:48 +00004659 const char *SymbolName,
4660 const char *SectionName) {
4661 unsigned NumClasses = Container.size();
Daniel Dunbar6bff2512009-08-03 17:06:42 +00004662
Daniel Dunbar463b8762009-05-15 21:48:48 +00004663 if (!NumClasses)
4664 return;
Daniel Dunbar6bff2512009-08-03 17:06:42 +00004665
Daniel Dunbar463b8762009-05-15 21:48:48 +00004666 std::vector<llvm::Constant*> Symbols(NumClasses);
4667 for (unsigned i=0; i<NumClasses; i++)
Owen Anderson3c4972d2009-07-29 18:54:39 +00004668 Symbols[i] = llvm::ConstantExpr::getBitCast(Container[i],
Daniel Dunbar463b8762009-05-15 21:48:48 +00004669 ObjCTypes.Int8PtrTy);
Daniel Dunbar6bff2512009-08-03 17:06:42 +00004670 llvm::Constant* Init =
Owen Anderson96e0fc72009-07-29 22:16:19 +00004671 llvm::ConstantArray::get(llvm::ArrayType::get(ObjCTypes.Int8PtrTy,
Daniel Dunbar463b8762009-05-15 21:48:48 +00004672 NumClasses),
4673 Symbols);
Daniel Dunbar6bff2512009-08-03 17:06:42 +00004674
Daniel Dunbar463b8762009-05-15 21:48:48 +00004675 llvm::GlobalVariable *GV =
Owen Anderson1c431b32009-07-08 19:05:04 +00004676 new llvm::GlobalVariable(CGM.getModule(), Init->getType(), false,
Daniel Dunbar463b8762009-05-15 21:48:48 +00004677 llvm::GlobalValue::InternalLinkage,
4678 Init,
Owen Anderson1c431b32009-07-08 19:05:04 +00004679 SymbolName);
Daniel Dunbar4b429ae2010-04-25 20:39:32 +00004680 GV->setAlignment(CGM.getTargetData().getABITypeAlignment(Init->getType()));
Daniel Dunbar463b8762009-05-15 21:48:48 +00004681 GV->setSection(SectionName);
Chris Lattnerad64e022009-07-17 23:57:13 +00004682 CGM.AddUsedGlobal(GV);
Daniel Dunbar463b8762009-05-15 21:48:48 +00004683}
Daniel Dunbar6bff2512009-08-03 17:06:42 +00004684
Fariborz Jahanianaa23b572009-01-23 23:53:38 +00004685void CGObjCNonFragileABIMac::FinishNonFragileABIModule() {
4686 // nonfragile abi has no module definition.
Daniel Dunbar6bff2512009-08-03 17:06:42 +00004687
Daniel Dunbar463b8762009-05-15 21:48:48 +00004688 // Build list of all implemented class addresses in array
Fariborz Jahanianf87a0cc2009-01-30 20:55:31 +00004689 // L_OBJC_LABEL_CLASS_$.
Daniel Dunbar6bff2512009-08-03 17:06:42 +00004690 AddModuleClassList(DefinedClasses,
Daniel Dunbar463b8762009-05-15 21:48:48 +00004691 "\01L_OBJC_LABEL_CLASS_$",
4692 "__DATA, __objc_classlist, regular, no_dead_strip");
Fariborz Jahaniana03d0dd2009-11-17 21:37:35 +00004693
Fariborz Jahaniana03d0dd2009-11-17 21:37:35 +00004694 for (unsigned i = 0; i < DefinedClasses.size(); i++) {
4695 llvm::GlobalValue *IMPLGV = DefinedClasses[i];
4696 if (IMPLGV->getLinkage() != llvm::GlobalValue::ExternalWeakLinkage)
4697 continue;
4698 IMPLGV->setLinkage(llvm::GlobalValue::ExternalLinkage);
Fariborz Jahaniana03d0dd2009-11-17 21:37:35 +00004699 }
4700
Fariborz Jahanian0e93d252009-12-01 18:25:24 +00004701 for (unsigned i = 0; i < DefinedMetaClasses.size(); i++) {
4702 llvm::GlobalValue *IMPLGV = DefinedMetaClasses[i];
4703 if (IMPLGV->getLinkage() != llvm::GlobalValue::ExternalWeakLinkage)
4704 continue;
4705 IMPLGV->setLinkage(llvm::GlobalValue::ExternalLinkage);
4706 }
Fariborz Jahaniana03d0dd2009-11-17 21:37:35 +00004707
Daniel Dunbar6bff2512009-08-03 17:06:42 +00004708 AddModuleClassList(DefinedNonLazyClasses,
Daniel Dunbar74d4b122009-05-15 22:33:15 +00004709 "\01L_OBJC_LABEL_NONLAZY_CLASS_$",
4710 "__DATA, __objc_nlclslist, regular, no_dead_strip");
Daniel Dunbar6bff2512009-08-03 17:06:42 +00004711
Fariborz Jahanianf87a0cc2009-01-30 20:55:31 +00004712 // Build list of all implemented category addresses in array
4713 // L_OBJC_LABEL_CATEGORY_$.
Daniel Dunbar6bff2512009-08-03 17:06:42 +00004714 AddModuleClassList(DefinedCategories,
Daniel Dunbar463b8762009-05-15 21:48:48 +00004715 "\01L_OBJC_LABEL_CATEGORY_$",
4716 "__DATA, __objc_catlist, regular, no_dead_strip");
Daniel Dunbar6bff2512009-08-03 17:06:42 +00004717 AddModuleClassList(DefinedNonLazyCategories,
Daniel Dunbar74d4b122009-05-15 22:33:15 +00004718 "\01L_OBJC_LABEL_NONLAZY_CATEGORY_$",
4719 "__DATA, __objc_nlcatlist, regular, no_dead_strip");
Daniel Dunbar6bff2512009-08-03 17:06:42 +00004720
Daniel Dunbarfce176b2010-04-25 20:39:01 +00004721 EmitImageInfo();
Fariborz Jahanianaa23b572009-01-23 23:53:38 +00004722}
4723
Fariborz Jahaniand0f8a8d2009-05-11 19:25:47 +00004724/// LegacyDispatchedSelector - Returns true if SEL is not in the list of
Fariborz Jahanian4523eb02009-05-12 20:06:41 +00004725/// NonLegacyDispatchMethods; false otherwise. What this means is that
Daniel Dunbar6bff2512009-08-03 17:06:42 +00004726/// except for the 19 selectors in the list, we generate 32bit-style
Fariborz Jahaniand0f8a8d2009-05-11 19:25:47 +00004727/// message dispatch call for all the rest.
4728///
4729bool CGObjCNonFragileABIMac::LegacyDispatchedSelector(Selector Sel) {
Daniel Dunbarf643b9b2010-04-24 17:56:46 +00004730 switch (CGM.getCodeGenOpts().getObjCDispatchMethod()) {
4731 default:
4732 assert(0 && "Invalid dispatch method!");
4733 case CodeGenOptions::Legacy:
Daniel Dunbar2feefe82010-02-01 21:07:33 +00004734 return true;
Daniel Dunbarf643b9b2010-04-24 17:56:46 +00004735 case CodeGenOptions::NonLegacy:
Fariborz Jahanian776dbf92010-04-19 17:53:30 +00004736 return false;
Daniel Dunbarf643b9b2010-04-24 17:56:46 +00004737 case CodeGenOptions::Mixed:
4738 break;
4739 }
4740
4741 // If so, see whether this selector is in the white-list of things which must
4742 // use the new dispatch convention. We lazily build a dense set for this.
Fariborz Jahanian4523eb02009-05-12 20:06:41 +00004743 if (NonLegacyDispatchMethods.empty()) {
4744 NonLegacyDispatchMethods.insert(GetNullarySelector("alloc"));
4745 NonLegacyDispatchMethods.insert(GetNullarySelector("class"));
4746 NonLegacyDispatchMethods.insert(GetNullarySelector("self"));
4747 NonLegacyDispatchMethods.insert(GetNullarySelector("isFlipped"));
4748 NonLegacyDispatchMethods.insert(GetNullarySelector("length"));
4749 NonLegacyDispatchMethods.insert(GetNullarySelector("count"));
4750 NonLegacyDispatchMethods.insert(GetNullarySelector("retain"));
4751 NonLegacyDispatchMethods.insert(GetNullarySelector("release"));
4752 NonLegacyDispatchMethods.insert(GetNullarySelector("autorelease"));
4753 NonLegacyDispatchMethods.insert(GetNullarySelector("hash"));
Daniel Dunbar6bff2512009-08-03 17:06:42 +00004754
Fariborz Jahanian4523eb02009-05-12 20:06:41 +00004755 NonLegacyDispatchMethods.insert(GetUnarySelector("allocWithZone"));
4756 NonLegacyDispatchMethods.insert(GetUnarySelector("isKindOfClass"));
4757 NonLegacyDispatchMethods.insert(GetUnarySelector("respondsToSelector"));
4758 NonLegacyDispatchMethods.insert(GetUnarySelector("objectForKey"));
4759 NonLegacyDispatchMethods.insert(GetUnarySelector("objectAtIndex"));
4760 NonLegacyDispatchMethods.insert(GetUnarySelector("isEqualToString"));
4761 NonLegacyDispatchMethods.insert(GetUnarySelector("isEqual"));
4762 NonLegacyDispatchMethods.insert(GetUnarySelector("addObject"));
Daniel Dunbar6bff2512009-08-03 17:06:42 +00004763 // "countByEnumeratingWithState:objects:count"
Fariborz Jahanianbe53be42009-05-13 16:19:02 +00004764 IdentifierInfo *KeyIdents[] = {
Daniel Dunbar6bff2512009-08-03 17:06:42 +00004765 &CGM.getContext().Idents.get("countByEnumeratingWithState"),
4766 &CGM.getContext().Idents.get("objects"),
4767 &CGM.getContext().Idents.get("count")
Fariborz Jahanianbe53be42009-05-13 16:19:02 +00004768 };
4769 NonLegacyDispatchMethods.insert(
4770 CGM.getContext().Selectors.getSelector(3, KeyIdents));
Fariborz Jahaniand0f8a8d2009-05-11 19:25:47 +00004771 }
Daniel Dunbarf643b9b2010-04-24 17:56:46 +00004772
Fariborz Jahanian4523eb02009-05-12 20:06:41 +00004773 return (NonLegacyDispatchMethods.count(Sel) == 0);
Fariborz Jahaniand0f8a8d2009-05-11 19:25:47 +00004774}
4775
Fariborz Jahanian058a1b72009-01-24 20:21:50 +00004776// Metadata flags
4777enum MetaDataDlags {
4778 CLS = 0x0,
4779 CLS_META = 0x1,
4780 CLS_ROOT = 0x2,
Fariborz Jahanian493dab72009-01-26 21:38:32 +00004781 OBJC2_CLS_HIDDEN = 0x10,
Fariborz Jahanian058a1b72009-01-24 20:21:50 +00004782 CLS_EXCEPTION = 0x20
4783};
4784/// BuildClassRoTInitializer - generate meta-data for:
4785/// struct _class_ro_t {
4786/// uint32_t const flags;
4787/// uint32_t const instanceStart;
4788/// uint32_t const instanceSize;
4789/// uint32_t const reserved; // only when building for 64bit targets
4790/// const uint8_t * const ivarLayout;
4791/// const char *const name;
4792/// const struct _method_list_t * const baseMethods;
Fariborz Jahanianda320092009-01-29 19:24:30 +00004793/// const struct _protocol_list_t *const baseProtocols;
Fariborz Jahanian058a1b72009-01-24 20:21:50 +00004794/// const struct _ivar_list_t *const ivars;
4795/// const uint8_t * const weakIvarLayout;
4796/// const struct _prop_list_t * const properties;
4797/// }
4798///
4799llvm::GlobalVariable * CGObjCNonFragileABIMac::BuildClassRoTInitializer(
Daniel Dunbar6bff2512009-08-03 17:06:42 +00004800 unsigned flags,
4801 unsigned InstanceStart,
4802 unsigned InstanceSize,
4803 const ObjCImplementationDecl *ID) {
Fariborz Jahanian058a1b72009-01-24 20:21:50 +00004804 std::string ClassName = ID->getNameAsString();
4805 std::vector<llvm::Constant*> Values(10); // 11 for 64bit targets!
Owen Anderson4a28d5d2009-07-24 23:12:58 +00004806 Values[ 0] = llvm::ConstantInt::get(ObjCTypes.IntTy, flags);
4807 Values[ 1] = llvm::ConstantInt::get(ObjCTypes.IntTy, InstanceStart);
4808 Values[ 2] = llvm::ConstantInt::get(ObjCTypes.IntTy, InstanceSize);
Fariborz Jahanian058a1b72009-01-24 20:21:50 +00004809 // FIXME. For 64bit targets add 0 here.
Daniel Dunbar6bff2512009-08-03 17:06:42 +00004810 Values[ 3] = (flags & CLS_META) ? GetIvarLayoutName(0, ObjCTypes)
4811 : BuildIvarLayout(ID, true);
Fariborz Jahanian058a1b72009-01-24 20:21:50 +00004812 Values[ 4] = GetClassName(ID->getIdentifier());
Fariborz Jahanian493dab72009-01-26 21:38:32 +00004813 // const struct _method_list_t * const baseMethods;
4814 std::vector<llvm::Constant*> Methods;
4815 std::string MethodListName("\01l_OBJC_$_");
4816 if (flags & CLS_META) {
4817 MethodListName += "CLASS_METHODS_" + ID->getNameAsString();
Daniel Dunbar6bff2512009-08-03 17:06:42 +00004818 for (ObjCImplementationDecl::classmeth_iterator
Argyrios Kyrtzidis17945a02009-06-30 02:36:12 +00004819 i = ID->classmeth_begin(), e = ID->classmeth_end(); i != e; ++i) {
Fariborz Jahanian493dab72009-01-26 21:38:32 +00004820 // Class methods should always be defined.
4821 Methods.push_back(GetMethodConstant(*i));
4822 }
4823 } else {
4824 MethodListName += "INSTANCE_METHODS_" + ID->getNameAsString();
Daniel Dunbar6bff2512009-08-03 17:06:42 +00004825 for (ObjCImplementationDecl::instmeth_iterator
Argyrios Kyrtzidis17945a02009-06-30 02:36:12 +00004826 i = ID->instmeth_begin(), e = ID->instmeth_end(); i != e; ++i) {
Fariborz Jahanian493dab72009-01-26 21:38:32 +00004827 // Instance methods should always be defined.
4828 Methods.push_back(GetMethodConstant(*i));
4829 }
Daniel Dunbar6bff2512009-08-03 17:06:42 +00004830 for (ObjCImplementationDecl::propimpl_iterator
Argyrios Kyrtzidis17945a02009-06-30 02:36:12 +00004831 i = ID->propimpl_begin(), e = ID->propimpl_end(); i != e; ++i) {
Fariborz Jahanian939abce2009-01-28 22:46:49 +00004832 ObjCPropertyImplDecl *PID = *i;
Daniel Dunbar6bff2512009-08-03 17:06:42 +00004833
Fariborz Jahanian939abce2009-01-28 22:46:49 +00004834 if (PID->getPropertyImplementation() == ObjCPropertyImplDecl::Synthesize){
4835 ObjCPropertyDecl *PD = PID->getPropertyDecl();
Daniel Dunbar6bff2512009-08-03 17:06:42 +00004836
Fariborz Jahanian939abce2009-01-28 22:46:49 +00004837 if (ObjCMethodDecl *MD = PD->getGetterMethodDecl())
4838 if (llvm::Constant *C = GetMethodConstant(MD))
4839 Methods.push_back(C);
4840 if (ObjCMethodDecl *MD = PD->getSetterMethodDecl())
4841 if (llvm::Constant *C = GetMethodConstant(MD))
4842 Methods.push_back(C);
4843 }
4844 }
Fariborz Jahanian493dab72009-01-26 21:38:32 +00004845 }
Daniel Dunbar6bff2512009-08-03 17:06:42 +00004846 Values[ 5] = EmitMethodList(MethodListName,
4847 "__DATA, __objc_const", Methods);
4848
Fariborz Jahanianda320092009-01-29 19:24:30 +00004849 const ObjCInterfaceDecl *OID = ID->getClassInterface();
4850 assert(OID && "CGObjCNonFragileABIMac::BuildClassRoTInitializer");
Daniel Dunbar6bff2512009-08-03 17:06:42 +00004851 Values[ 6] = EmitProtocolList("\01l_OBJC_CLASS_PROTOCOLS_$_"
Daniel Dunbar9c29bf52009-10-18 20:48:59 +00004852 + OID->getName(),
Fariborz Jahanianda320092009-01-29 19:24:30 +00004853 OID->protocol_begin(),
4854 OID->protocol_end());
Daniel Dunbar6bff2512009-08-03 17:06:42 +00004855
Fariborz Jahanian98abf4b2009-01-27 19:38:51 +00004856 if (flags & CLS_META)
Owen Andersonc9c88b42009-07-31 20:28:54 +00004857 Values[ 7] = llvm::Constant::getNullValue(ObjCTypes.IvarListnfABIPtrTy);
Fariborz Jahanian98abf4b2009-01-27 19:38:51 +00004858 else
4859 Values[ 7] = EmitIvarList(ID);
Daniel Dunbar6bff2512009-08-03 17:06:42 +00004860 Values[ 8] = (flags & CLS_META) ? GetIvarLayoutName(0, ObjCTypes)
4861 : BuildIvarLayout(ID, false);
Fariborz Jahanian5de14dc2009-01-28 22:18:42 +00004862 if (flags & CLS_META)
Owen Andersonc9c88b42009-07-31 20:28:54 +00004863 Values[ 9] = llvm::Constant::getNullValue(ObjCTypes.PropertyListPtrTy);
Fariborz Jahanian5de14dc2009-01-28 22:18:42 +00004864 else
Daniel Dunbar9c29bf52009-10-18 20:48:59 +00004865 Values[ 9] = EmitPropertyList("\01l_OBJC_$_PROP_LIST_" + ID->getName(),
4866 ID, ID->getClassInterface(), ObjCTypes);
Owen Anderson08e25242009-07-27 22:29:56 +00004867 llvm::Constant *Init = llvm::ConstantStruct::get(ObjCTypes.ClassRonfABITy,
Fariborz Jahanian058a1b72009-01-24 20:21:50 +00004868 Values);
4869 llvm::GlobalVariable *CLASS_RO_GV =
Daniel Dunbar6bff2512009-08-03 17:06:42 +00004870 new llvm::GlobalVariable(CGM.getModule(), ObjCTypes.ClassRonfABITy, false,
4871 llvm::GlobalValue::InternalLinkage,
4872 Init,
4873 (flags & CLS_META) ?
4874 std::string("\01l_OBJC_METACLASS_RO_$_")+ClassName :
4875 std::string("\01l_OBJC_CLASS_RO_$_")+ClassName);
Fariborz Jahanian09796d62009-01-31 02:43:27 +00004876 CLASS_RO_GV->setAlignment(
Daniel Dunbar4b429ae2010-04-25 20:39:32 +00004877 CGM.getTargetData().getABITypeAlignment(ObjCTypes.ClassRonfABITy));
Fariborz Jahanian1bf0afb2009-01-28 01:05:23 +00004878 CLASS_RO_GV->setSection("__DATA, __objc_const");
Fariborz Jahanian058a1b72009-01-24 20:21:50 +00004879 return CLASS_RO_GV;
Fariborz Jahanianf6317dd2009-01-26 22:58:07 +00004880
Fariborz Jahanian058a1b72009-01-24 20:21:50 +00004881}
4882
4883/// BuildClassMetaData - This routine defines that to-level meta-data
4884/// for the given ClassName for:
4885/// struct _class_t {
4886/// struct _class_t *isa;
4887/// struct _class_t * const superclass;
4888/// void *cache;
4889/// IMP *vtable;
4890/// struct class_ro_t *ro;
4891/// }
4892///
Fariborz Jahanian84394a52009-01-24 21:21:53 +00004893llvm::GlobalVariable * CGObjCNonFragileABIMac::BuildClassMetaData(
Daniel Dunbar6bff2512009-08-03 17:06:42 +00004894 std::string &ClassName,
4895 llvm::Constant *IsAGV,
4896 llvm::Constant *SuperClassGV,
4897 llvm::Constant *ClassRoGV,
4898 bool HiddenVisibility) {
Fariborz Jahanian058a1b72009-01-24 20:21:50 +00004899 std::vector<llvm::Constant*> Values(5);
4900 Values[0] = IsAGV;
Daniel Dunbar6bff2512009-08-03 17:06:42 +00004901 Values[1] = SuperClassGV;
4902 if (!Values[1])
4903 Values[1] = llvm::Constant::getNullValue(ObjCTypes.ClassnfABIPtrTy);
Fariborz Jahanian058a1b72009-01-24 20:21:50 +00004904 Values[2] = ObjCEmptyCacheVar; // &ObjCEmptyCacheVar
4905 Values[3] = ObjCEmptyVtableVar; // &ObjCEmptyVtableVar
4906 Values[4] = ClassRoGV; // &CLASS_RO_GV
Daniel Dunbar6bff2512009-08-03 17:06:42 +00004907 llvm::Constant *Init = llvm::ConstantStruct::get(ObjCTypes.ClassnfABITy,
Fariborz Jahanian058a1b72009-01-24 20:21:50 +00004908 Values);
Daniel Dunbar5a7379a2009-03-01 04:40:10 +00004909 llvm::GlobalVariable *GV = GetClassGlobal(ClassName);
4910 GV->setInitializer(Init);
Fariborz Jahaniandd0db2a2009-01-31 01:07:39 +00004911 GV->setSection("__DATA, __objc_data");
Fariborz Jahanian09796d62009-01-31 02:43:27 +00004912 GV->setAlignment(
Daniel Dunbar4b429ae2010-04-25 20:39:32 +00004913 CGM.getTargetData().getABITypeAlignment(ObjCTypes.ClassnfABITy));
Fariborz Jahaniancf555162009-01-31 00:59:10 +00004914 if (HiddenVisibility)
4915 GV->setVisibility(llvm::GlobalValue::HiddenVisibility);
Fariborz Jahanian84394a52009-01-24 21:21:53 +00004916 return GV;
Fariborz Jahanian058a1b72009-01-24 20:21:50 +00004917}
4918
Daniel Dunbar6bff2512009-08-03 17:06:42 +00004919bool
Fariborz Jahanianecfbdcb2009-05-21 01:03:45 +00004920CGObjCNonFragileABIMac::ImplementationIsNonLazy(const ObjCImplDecl *OD) const {
Argyrios Kyrtzidis17945a02009-06-30 02:36:12 +00004921 return OD->getClassMethod(GetNullarySelector("load")) != 0;
Daniel Dunbar74d4b122009-05-15 22:33:15 +00004922}
4923
Daniel Dunbar9f89f2b2009-05-03 12:57:56 +00004924void CGObjCNonFragileABIMac::GetClassSizeInfo(const ObjCImplementationDecl *OID,
Daniel Dunbarb02532a2009-04-19 23:41:48 +00004925 uint32_t &InstanceStart,
4926 uint32_t &InstanceSize) {
Daniel Dunbar6bff2512009-08-03 17:06:42 +00004927 const ASTRecordLayout &RL =
Daniel Dunbarb4c79e02009-05-04 21:26:30 +00004928 CGM.getContext().getASTObjCImplementationLayout(OID);
Daniel Dunbar6bff2512009-08-03 17:06:42 +00004929
Daniel Dunbar6e8575b2009-05-04 23:23:09 +00004930 // InstanceSize is really instance end.
Anders Carlsson243a6852009-07-18 21:26:44 +00004931 InstanceSize = llvm::RoundUpToAlignment(RL.getDataSize(), 8) / 8;
Daniel Dunbar6e8575b2009-05-04 23:23:09 +00004932
4933 // If there are no fields, the start is the same as the end.
4934 if (!RL.getFieldCount())
4935 InstanceStart = InstanceSize;
4936 else
4937 InstanceStart = RL.getFieldOffset(0) / 8;
Daniel Dunbarb02532a2009-04-19 23:41:48 +00004938}
4939
Fariborz Jahanianaa23b572009-01-23 23:53:38 +00004940void CGObjCNonFragileABIMac::GenerateClass(const ObjCImplementationDecl *ID) {
4941 std::string ClassName = ID->getNameAsString();
4942 if (!ObjCEmptyCacheVar) {
4943 ObjCEmptyCacheVar = new llvm::GlobalVariable(
Daniel Dunbar6bff2512009-08-03 17:06:42 +00004944 CGM.getModule(),
4945 ObjCTypes.CacheTy,
4946 false,
4947 llvm::GlobalValue::ExternalLinkage,
4948 0,
4949 "_objc_empty_cache");
4950
Fariborz Jahanianaa23b572009-01-23 23:53:38 +00004951 ObjCEmptyVtableVar = new llvm::GlobalVariable(
Daniel Dunbar6bff2512009-08-03 17:06:42 +00004952 CGM.getModule(),
4953 ObjCTypes.ImpnfABITy,
4954 false,
4955 llvm::GlobalValue::ExternalLinkage,
4956 0,
4957 "_objc_empty_vtable");
Fariborz Jahanianaa23b572009-01-23 23:53:38 +00004958 }
Daniel Dunbar6bff2512009-08-03 17:06:42 +00004959 assert(ID->getClassInterface() &&
Fariborz Jahanian493dab72009-01-26 21:38:32 +00004960 "CGObjCNonFragileABIMac::GenerateClass - class is 0");
Daniel Dunbar6c1aac82009-04-20 20:18:54 +00004961 // FIXME: Is this correct (that meta class size is never computed)?
Daniel Dunbar6bff2512009-08-03 17:06:42 +00004962 uint32_t InstanceStart =
Duncan Sands9408c452009-05-09 07:08:47 +00004963 CGM.getTargetData().getTypeAllocSize(ObjCTypes.ClassnfABITy);
Fariborz Jahanian058a1b72009-01-24 20:21:50 +00004964 uint32_t InstanceSize = InstanceStart;
4965 uint32_t flags = CLS_META;
Daniel Dunbar6ab187a2009-04-07 05:48:37 +00004966 std::string ObjCMetaClassName(getMetaclassSymbolPrefix());
4967 std::string ObjCClassName(getClassSymbolPrefix());
Daniel Dunbar6bff2512009-08-03 17:06:42 +00004968
Fariborz Jahanian058a1b72009-01-24 20:21:50 +00004969 llvm::GlobalVariable *SuperClassGV, *IsAGV;
Daniel Dunbar6bff2512009-08-03 17:06:42 +00004970
4971 bool classIsHidden =
Daniel Dunbar04d40782009-04-14 06:00:08 +00004972 CGM.getDeclVisibilityMode(ID->getClassInterface()) == LangOptions::Hidden;
Fariborz Jahaniancf555162009-01-31 00:59:10 +00004973 if (classIsHidden)
Fariborz Jahanian493dab72009-01-26 21:38:32 +00004974 flags |= OBJC2_CLS_HIDDEN;
Fariborz Jahanian109dfc62010-04-28 21:28:56 +00004975 if (ID->getNumIvarInitializers())
4976 flags |= eClassFlags_ABI2_HasCXXStructors;
Fariborz Jahanian493dab72009-01-26 21:38:32 +00004977 if (!ID->getClassInterface()->getSuperClass()) {
Fariborz Jahanian058a1b72009-01-24 20:21:50 +00004978 // class is root
4979 flags |= CLS_ROOT;
Daniel Dunbar5a7379a2009-03-01 04:40:10 +00004980 SuperClassGV = GetClassGlobal(ObjCClassName + ClassName);
Fariborz Jahanian0f902942009-04-14 18:41:56 +00004981 IsAGV = GetClassGlobal(ObjCMetaClassName + ClassName);
Fariborz Jahanian058a1b72009-01-24 20:21:50 +00004982 } else {
Fariborz Jahanian84394a52009-01-24 21:21:53 +00004983 // Has a root. Current class is not a root.
Fariborz Jahanianfab98c42009-02-26 18:23:47 +00004984 const ObjCInterfaceDecl *Root = ID->getClassInterface();
4985 while (const ObjCInterfaceDecl *Super = Root->getSuperClass())
4986 Root = Super;
Fariborz Jahanian0f902942009-04-14 18:41:56 +00004987 IsAGV = GetClassGlobal(ObjCMetaClassName + Root->getNameAsString());
Fariborz Jahanian0e93d252009-12-01 18:25:24 +00004988 if (Root->hasAttr<WeakImportAttr>())
4989 IsAGV->setLinkage(llvm::GlobalValue::ExternalWeakLinkage);
Fariborz Jahanianfab98c42009-02-26 18:23:47 +00004990 // work on super class metadata symbol.
Daniel Dunbar6bff2512009-08-03 17:06:42 +00004991 std::string SuperClassName =
Fariborz Jahanian0e93d252009-12-01 18:25:24 +00004992 ObjCMetaClassName +
4993 ID->getClassInterface()->getSuperClass()->getNameAsString();
Fariborz Jahanian0f902942009-04-14 18:41:56 +00004994 SuperClassGV = GetClassGlobal(SuperClassName);
Fariborz Jahaniana03d0dd2009-11-17 21:37:35 +00004995 if (ID->getClassInterface()->getSuperClass()->hasAttr<WeakImportAttr>())
4996 SuperClassGV->setLinkage(llvm::GlobalValue::ExternalWeakLinkage);
Fariborz Jahanian058a1b72009-01-24 20:21:50 +00004997 }
4998 llvm::GlobalVariable *CLASS_RO_GV = BuildClassRoTInitializer(flags,
4999 InstanceStart,
5000 InstanceSize,ID);
Fariborz Jahanian84394a52009-01-24 21:21:53 +00005001 std::string TClassName = ObjCMetaClassName + ClassName;
Daniel Dunbar6bff2512009-08-03 17:06:42 +00005002 llvm::GlobalVariable *MetaTClass =
Fariborz Jahaniancf555162009-01-31 00:59:10 +00005003 BuildClassMetaData(TClassName, IsAGV, SuperClassGV, CLASS_RO_GV,
5004 classIsHidden);
Fariborz Jahaniana03d0dd2009-11-17 21:37:35 +00005005 DefinedMetaClasses.push_back(MetaTClass);
Daniel Dunbar6ab187a2009-04-07 05:48:37 +00005006
Fariborz Jahanian84394a52009-01-24 21:21:53 +00005007 // Metadata for the class
5008 flags = CLS;
Fariborz Jahaniancf555162009-01-31 00:59:10 +00005009 if (classIsHidden)
Fariborz Jahanian493dab72009-01-26 21:38:32 +00005010 flags |= OBJC2_CLS_HIDDEN;
Fariborz Jahanian109dfc62010-04-28 21:28:56 +00005011 if (ID->getNumIvarInitializers())
5012 flags |= eClassFlags_ABI2_HasCXXStructors;
Daniel Dunbar8158a2f2009-04-08 04:21:03 +00005013
Douglas Gregor68584ed2009-06-18 16:11:24 +00005014 if (hasObjCExceptionAttribute(CGM.getContext(), ID->getClassInterface()))
Daniel Dunbar8158a2f2009-04-08 04:21:03 +00005015 flags |= CLS_EXCEPTION;
5016
Fariborz Jahanian493dab72009-01-26 21:38:32 +00005017 if (!ID->getClassInterface()->getSuperClass()) {
Fariborz Jahanian84394a52009-01-24 21:21:53 +00005018 flags |= CLS_ROOT;
5019 SuperClassGV = 0;
Chris Lattnerb7b58b12009-04-19 06:02:28 +00005020 } else {
Fariborz Jahanian84394a52009-01-24 21:21:53 +00005021 // Has a root. Current class is not a root.
Fariborz Jahanianfab98c42009-02-26 18:23:47 +00005022 std::string RootClassName =
Fariborz Jahanian84394a52009-01-24 21:21:53 +00005023 ID->getClassInterface()->getSuperClass()->getNameAsString();
Daniel Dunbar5a7379a2009-03-01 04:40:10 +00005024 SuperClassGV = GetClassGlobal(ObjCClassName + RootClassName);
Fariborz Jahaniana03d0dd2009-11-17 21:37:35 +00005025 if (ID->getClassInterface()->getSuperClass()->hasAttr<WeakImportAttr>())
5026 SuperClassGV->setLinkage(llvm::GlobalValue::ExternalWeakLinkage);
Fariborz Jahanian84394a52009-01-24 21:21:53 +00005027 }
Daniel Dunbar9f89f2b2009-05-03 12:57:56 +00005028 GetClassSizeInfo(ID, InstanceStart, InstanceSize);
Fariborz Jahanian84394a52009-01-24 21:21:53 +00005029 CLASS_RO_GV = BuildClassRoTInitializer(flags,
Fariborz Jahanianf6a077e2009-01-24 23:43:01 +00005030 InstanceStart,
Daniel Dunbar6bff2512009-08-03 17:06:42 +00005031 InstanceSize,
Fariborz Jahanianf6a077e2009-01-24 23:43:01 +00005032 ID);
Daniel Dunbar6bff2512009-08-03 17:06:42 +00005033
Fariborz Jahanian84394a52009-01-24 21:21:53 +00005034 TClassName = ObjCClassName + ClassName;
Daniel Dunbar6bff2512009-08-03 17:06:42 +00005035 llvm::GlobalVariable *ClassMD =
Fariborz Jahaniancf555162009-01-31 00:59:10 +00005036 BuildClassMetaData(TClassName, MetaTClass, SuperClassGV, CLASS_RO_GV,
5037 classIsHidden);
Fariborz Jahanianf87a0cc2009-01-30 20:55:31 +00005038 DefinedClasses.push_back(ClassMD);
Daniel Dunbar8158a2f2009-04-08 04:21:03 +00005039
Daniel Dunbar74d4b122009-05-15 22:33:15 +00005040 // Determine if this class is also "non-lazy".
5041 if (ImplementationIsNonLazy(ID))
5042 DefinedNonLazyClasses.push_back(ClassMD);
5043
Daniel Dunbar8158a2f2009-04-08 04:21:03 +00005044 // Force the definition of the EHType if necessary.
5045 if (flags & CLS_EXCEPTION)
5046 GetInterfaceEHType(ID->getClassInterface(), true);
Fariborz Jahanianaa23b572009-01-23 23:53:38 +00005047}
5048
Fariborz Jahanian8cfd3972009-01-30 18:58:59 +00005049/// GenerateProtocolRef - This routine is called to generate code for
5050/// a protocol reference expression; as in:
5051/// @code
5052/// @protocol(Proto1);
5053/// @endcode
5054/// It generates a weak reference to l_OBJC_PROTOCOL_REFERENCE_$_Proto1
5055/// which will hold address of the protocol meta-data.
5056///
5057llvm::Value *CGObjCNonFragileABIMac::GenerateProtocolRef(CGBuilderTy &Builder,
Daniel Dunbar6bff2512009-08-03 17:06:42 +00005058 const ObjCProtocolDecl *PD) {
5059
Fariborz Jahanian960cd062009-04-10 18:47:34 +00005060 // This routine is called for @protocol only. So, we must build definition
5061 // of protocol's meta-data (not a reference to it!)
5062 //
Daniel Dunbar6bff2512009-08-03 17:06:42 +00005063 llvm::Constant *Init =
5064 llvm::ConstantExpr::getBitCast(GetOrEmitProtocol(PD),
5065 ObjCTypes.ExternalProtocolPtrTy);
5066
Fariborz Jahanian8cfd3972009-01-30 18:58:59 +00005067 std::string ProtocolName("\01l_OBJC_PROTOCOL_REFERENCE_$_");
5068 ProtocolName += PD->getNameAsCString();
Daniel Dunbar6bff2512009-08-03 17:06:42 +00005069
Fariborz Jahanian8cfd3972009-01-30 18:58:59 +00005070 llvm::GlobalVariable *PTGV = CGM.getModule().getGlobalVariable(ProtocolName);
5071 if (PTGV)
Daniel Dunbar2da84ff2009-11-29 21:23:36 +00005072 return Builder.CreateLoad(PTGV, "tmp");
Fariborz Jahanian8cfd3972009-01-30 18:58:59 +00005073 PTGV = new llvm::GlobalVariable(
Daniel Dunbar6bff2512009-08-03 17:06:42 +00005074 CGM.getModule(),
5075 Init->getType(), false,
5076 llvm::GlobalValue::WeakAnyLinkage,
5077 Init,
5078 ProtocolName);
Fariborz Jahanian8cfd3972009-01-30 18:58:59 +00005079 PTGV->setSection("__DATA, __objc_protorefs, coalesced, no_dead_strip");
5080 PTGV->setVisibility(llvm::GlobalValue::HiddenVisibility);
Chris Lattnerad64e022009-07-17 23:57:13 +00005081 CGM.AddUsedGlobal(PTGV);
Daniel Dunbar2da84ff2009-11-29 21:23:36 +00005082 return Builder.CreateLoad(PTGV, "tmp");
Fariborz Jahanian8cfd3972009-01-30 18:58:59 +00005083}
5084
Fariborz Jahanianeb062d92009-01-26 18:32:24 +00005085/// GenerateCategory - Build metadata for a category implementation.
5086/// struct _category_t {
5087/// const char * const name;
5088/// struct _class_t *const cls;
5089/// const struct _method_list_t * const instance_methods;
5090/// const struct _method_list_t * const class_methods;
5091/// const struct _protocol_list_t * const protocols;
5092/// const struct _prop_list_t * const properties;
5093/// }
5094///
Daniel Dunbar74d4b122009-05-15 22:33:15 +00005095void CGObjCNonFragileABIMac::GenerateCategory(const ObjCCategoryImplDecl *OCD) {
Fariborz Jahanianeb062d92009-01-26 18:32:24 +00005096 const ObjCInterfaceDecl *Interface = OCD->getClassInterface();
Fariborz Jahanianf6317dd2009-01-26 22:58:07 +00005097 const char *Prefix = "\01l_OBJC_$_CATEGORY_";
Daniel Dunbar6bff2512009-08-03 17:06:42 +00005098 std::string ExtCatName(Prefix + Interface->getNameAsString()+
5099 "_$_" + OCD->getNameAsString());
5100 std::string ExtClassName(getClassSymbolPrefix() +
Daniel Dunbar6ab187a2009-04-07 05:48:37 +00005101 Interface->getNameAsString());
Daniel Dunbar6bff2512009-08-03 17:06:42 +00005102
Fariborz Jahanianeb062d92009-01-26 18:32:24 +00005103 std::vector<llvm::Constant*> Values(6);
5104 Values[0] = GetClassName(OCD->getIdentifier());
5105 // meta-class entry symbol
Daniel Dunbar5a7379a2009-03-01 04:40:10 +00005106 llvm::GlobalVariable *ClassGV = GetClassGlobal(ExtClassName);
Fariborz Jahanian2cdcc4c2009-11-17 22:02:21 +00005107 if (Interface->hasAttr<WeakImportAttr>())
5108 ClassGV->setLinkage(llvm::GlobalValue::ExternalWeakLinkage);
5109
Fariborz Jahanianeb062d92009-01-26 18:32:24 +00005110 Values[1] = ClassGV;
Fariborz Jahanianf6317dd2009-01-26 22:58:07 +00005111 std::vector<llvm::Constant*> Methods;
5112 std::string MethodListName(Prefix);
Daniel Dunbar6bff2512009-08-03 17:06:42 +00005113 MethodListName += "INSTANCE_METHODS_" + Interface->getNameAsString() +
Fariborz Jahanianf6317dd2009-01-26 22:58:07 +00005114 "_$_" + OCD->getNameAsString();
Daniel Dunbar6bff2512009-08-03 17:06:42 +00005115
5116 for (ObjCCategoryImplDecl::instmeth_iterator
Argyrios Kyrtzidis17945a02009-06-30 02:36:12 +00005117 i = OCD->instmeth_begin(), e = OCD->instmeth_end(); i != e; ++i) {
Fariborz Jahanianf6317dd2009-01-26 22:58:07 +00005118 // Instance methods should always be defined.
5119 Methods.push_back(GetMethodConstant(*i));
5120 }
Daniel Dunbar6bff2512009-08-03 17:06:42 +00005121
5122 Values[2] = EmitMethodList(MethodListName,
5123 "__DATA, __objc_const",
Fariborz Jahanianf6317dd2009-01-26 22:58:07 +00005124 Methods);
5125
5126 MethodListName = Prefix;
5127 MethodListName += "CLASS_METHODS_" + Interface->getNameAsString() + "_$_" +
5128 OCD->getNameAsString();
5129 Methods.clear();
Daniel Dunbar6bff2512009-08-03 17:06:42 +00005130 for (ObjCCategoryImplDecl::classmeth_iterator
Argyrios Kyrtzidis17945a02009-06-30 02:36:12 +00005131 i = OCD->classmeth_begin(), e = OCD->classmeth_end(); i != e; ++i) {
Fariborz Jahanianf6317dd2009-01-26 22:58:07 +00005132 // Class methods should always be defined.
5133 Methods.push_back(GetMethodConstant(*i));
5134 }
Daniel Dunbar6bff2512009-08-03 17:06:42 +00005135
5136 Values[3] = EmitMethodList(MethodListName,
5137 "__DATA, __objc_const",
Fariborz Jahanianf6317dd2009-01-26 22:58:07 +00005138 Methods);
Daniel Dunbar6bff2512009-08-03 17:06:42 +00005139 const ObjCCategoryDecl *Category =
Fariborz Jahanian5de14dc2009-01-28 22:18:42 +00005140 Interface->FindCategoryDeclaration(OCD->getIdentifier());
Fariborz Jahanian943ed6f2009-02-13 17:52:22 +00005141 if (Category) {
Daniel Dunbar9c29bf52009-10-18 20:48:59 +00005142 llvm::SmallString<256> ExtName;
5143 llvm::raw_svector_ostream(ExtName) << Interface->getName() << "_$_"
5144 << OCD->getName();
Fariborz Jahanian943ed6f2009-02-13 17:52:22 +00005145 Values[4] = EmitProtocolList("\01l_OBJC_CATEGORY_PROTOCOLS_$_"
Daniel Dunbar9c29bf52009-10-18 20:48:59 +00005146 + Interface->getName() + "_$_"
5147 + Category->getName(),
Fariborz Jahanian943ed6f2009-02-13 17:52:22 +00005148 Category->protocol_begin(),
5149 Category->protocol_end());
Daniel Dunbar9c29bf52009-10-18 20:48:59 +00005150 Values[5] = EmitPropertyList("\01l_OBJC_$_PROP_LIST_" + ExtName.str(),
5151 OCD, Category, ObjCTypes);
Mike Stumpb3589f42009-07-30 22:28:39 +00005152 } else {
Owen Andersonc9c88b42009-07-31 20:28:54 +00005153 Values[4] = llvm::Constant::getNullValue(ObjCTypes.ProtocolListnfABIPtrTy);
5154 Values[5] = llvm::Constant::getNullValue(ObjCTypes.PropertyListPtrTy);
Fariborz Jahanian943ed6f2009-02-13 17:52:22 +00005155 }
Daniel Dunbar6bff2512009-08-03 17:06:42 +00005156
5157 llvm::Constant *Init =
5158 llvm::ConstantStruct::get(ObjCTypes.CategorynfABITy,
Fariborz Jahanianeb062d92009-01-26 18:32:24 +00005159 Values);
5160 llvm::GlobalVariable *GCATV
Daniel Dunbar6bff2512009-08-03 17:06:42 +00005161 = new llvm::GlobalVariable(CGM.getModule(), ObjCTypes.CategorynfABITy,
Fariborz Jahanianeb062d92009-01-26 18:32:24 +00005162 false,
5163 llvm::GlobalValue::InternalLinkage,
5164 Init,
Owen Anderson1c431b32009-07-08 19:05:04 +00005165 ExtCatName);
Fariborz Jahanian09796d62009-01-31 02:43:27 +00005166 GCATV->setAlignment(
Daniel Dunbar4b429ae2010-04-25 20:39:32 +00005167 CGM.getTargetData().getABITypeAlignment(ObjCTypes.CategorynfABITy));
Fariborz Jahanian1bf0afb2009-01-28 01:05:23 +00005168 GCATV->setSection("__DATA, __objc_const");
Chris Lattnerad64e022009-07-17 23:57:13 +00005169 CGM.AddUsedGlobal(GCATV);
Fariborz Jahanianeb062d92009-01-26 18:32:24 +00005170 DefinedCategories.push_back(GCATV);
Daniel Dunbar74d4b122009-05-15 22:33:15 +00005171
5172 // Determine if this category is also "non-lazy".
5173 if (ImplementationIsNonLazy(OCD))
5174 DefinedNonLazyCategories.push_back(GCATV);
Fariborz Jahanianeb062d92009-01-26 18:32:24 +00005175}
Fariborz Jahanian493dab72009-01-26 21:38:32 +00005176
5177/// GetMethodConstant - Return a struct objc_method constant for the
5178/// given method if it has been defined. The result is null if the
5179/// method has not been defined. The return value has type MethodPtrTy.
5180llvm::Constant *CGObjCNonFragileABIMac::GetMethodConstant(
Daniel Dunbar6bff2512009-08-03 17:06:42 +00005181 const ObjCMethodDecl *MD) {
Argyrios Kyrtzidis9d50c062010-08-09 10:54:20 +00005182 llvm::Function *Fn = GetMethodDefinition(MD);
Fariborz Jahanian493dab72009-01-26 21:38:32 +00005183 if (!Fn)
5184 return 0;
Daniel Dunbar6bff2512009-08-03 17:06:42 +00005185
Fariborz Jahanian493dab72009-01-26 21:38:32 +00005186 std::vector<llvm::Constant*> Method(3);
Daniel Dunbar6bff2512009-08-03 17:06:42 +00005187 Method[0] =
Owen Anderson3c4972d2009-07-29 18:54:39 +00005188 llvm::ConstantExpr::getBitCast(GetMethodVarName(MD->getSelector()),
Daniel Dunbar6bff2512009-08-03 17:06:42 +00005189 ObjCTypes.SelectorPtrTy);
Fariborz Jahanian493dab72009-01-26 21:38:32 +00005190 Method[1] = GetMethodVarType(MD);
Owen Anderson3c4972d2009-07-29 18:54:39 +00005191 Method[2] = llvm::ConstantExpr::getBitCast(Fn, ObjCTypes.Int8PtrTy);
Owen Anderson08e25242009-07-27 22:29:56 +00005192 return llvm::ConstantStruct::get(ObjCTypes.MethodTy, Method);
Fariborz Jahanian493dab72009-01-26 21:38:32 +00005193}
5194
5195/// EmitMethodList - Build meta-data for method declarations
5196/// struct _method_list_t {
5197/// uint32_t entsize; // sizeof(struct _objc_method)
5198/// uint32_t method_count;
5199/// struct _objc_method method_list[method_count];
5200/// }
5201///
Daniel Dunbar9c29bf52009-10-18 20:48:59 +00005202llvm::Constant *CGObjCNonFragileABIMac::EmitMethodList(llvm::Twine Name,
5203 const char *Section,
5204 const ConstantVector &Methods) {
Fariborz Jahanian493dab72009-01-26 21:38:32 +00005205 // Return null for empty list.
5206 if (Methods.empty())
Owen Andersonc9c88b42009-07-31 20:28:54 +00005207 return llvm::Constant::getNullValue(ObjCTypes.MethodListnfABIPtrTy);
Daniel Dunbar6bff2512009-08-03 17:06:42 +00005208
Fariborz Jahanian493dab72009-01-26 21:38:32 +00005209 std::vector<llvm::Constant*> Values(3);
5210 // sizeof(struct _objc_method)
Duncan Sands9408c452009-05-09 07:08:47 +00005211 unsigned Size = CGM.getTargetData().getTypeAllocSize(ObjCTypes.MethodTy);
Owen Anderson4a28d5d2009-07-24 23:12:58 +00005212 Values[0] = llvm::ConstantInt::get(ObjCTypes.IntTy, Size);
Fariborz Jahanian493dab72009-01-26 21:38:32 +00005213 // method_count
Owen Anderson4a28d5d2009-07-24 23:12:58 +00005214 Values[1] = llvm::ConstantInt::get(ObjCTypes.IntTy, Methods.size());
Owen Anderson96e0fc72009-07-29 22:16:19 +00005215 llvm::ArrayType *AT = llvm::ArrayType::get(ObjCTypes.MethodTy,
Fariborz Jahanian493dab72009-01-26 21:38:32 +00005216 Methods.size());
Owen Anderson7db6d832009-07-28 18:33:04 +00005217 Values[2] = llvm::ConstantArray::get(AT, Methods);
Nick Lewycky0d36dd22009-09-19 20:00:52 +00005218 llvm::Constant *Init = llvm::ConstantStruct::get(VMContext, Values, false);
Daniel Dunbar6bff2512009-08-03 17:06:42 +00005219
Fariborz Jahanian493dab72009-01-26 21:38:32 +00005220 llvm::GlobalVariable *GV =
Owen Anderson1c431b32009-07-08 19:05:04 +00005221 new llvm::GlobalVariable(CGM.getModule(), Init->getType(), false,
Fariborz Jahanian493dab72009-01-26 21:38:32 +00005222 llvm::GlobalValue::InternalLinkage,
5223 Init,
Owen Anderson1c431b32009-07-08 19:05:04 +00005224 Name);
Fariborz Jahanian09796d62009-01-31 02:43:27 +00005225 GV->setAlignment(
Daniel Dunbar4b429ae2010-04-25 20:39:32 +00005226 CGM.getTargetData().getABITypeAlignment(Init->getType()));
Fariborz Jahanian493dab72009-01-26 21:38:32 +00005227 GV->setSection(Section);
Chris Lattnerad64e022009-07-17 23:57:13 +00005228 CGM.AddUsedGlobal(GV);
Owen Anderson3c4972d2009-07-29 18:54:39 +00005229 return llvm::ConstantExpr::getBitCast(GV,
Fariborz Jahanian493dab72009-01-26 21:38:32 +00005230 ObjCTypes.MethodListnfABIPtrTy);
5231}
Fariborz Jahanian98abf4b2009-01-27 19:38:51 +00005232
Fariborz Jahanianed157d32009-02-10 20:21:06 +00005233/// ObjCIvarOffsetVariable - Returns the ivar offset variable for
5234/// the given ivar.
Daniel Dunbare83be122010-04-02 21:14:02 +00005235llvm::GlobalVariable *
5236CGObjCNonFragileABIMac::ObjCIvarOffsetVariable(const ObjCInterfaceDecl *ID,
5237 const ObjCIvarDecl *Ivar) {
5238 const ObjCInterfaceDecl *Container = Ivar->getContainingInterface();
Daniel Dunbara81419d2009-05-05 00:36:57 +00005239 std::string Name = "OBJC_IVAR_$_" + Container->getNameAsString() +
Douglas Gregor6ab35242009-04-09 21:40:53 +00005240 '.' + Ivar->getNameAsString();
Daniel Dunbar6bff2512009-08-03 17:06:42 +00005241 llvm::GlobalVariable *IvarOffsetGV =
Fariborz Jahanianed157d32009-02-10 20:21:06 +00005242 CGM.getModule().getGlobalVariable(Name);
5243 if (!IvarOffsetGV)
Daniel Dunbar6bff2512009-08-03 17:06:42 +00005244 IvarOffsetGV =
Owen Anderson1c431b32009-07-08 19:05:04 +00005245 new llvm::GlobalVariable(CGM.getModule(), ObjCTypes.LongTy,
Fariborz Jahanianed157d32009-02-10 20:21:06 +00005246 false,
5247 llvm::GlobalValue::ExternalLinkage,
5248 0,
Owen Anderson1c431b32009-07-08 19:05:04 +00005249 Name);
Fariborz Jahanianed157d32009-02-10 20:21:06 +00005250 return IvarOffsetGV;
5251}
5252
Daniel Dunbare83be122010-04-02 21:14:02 +00005253llvm::Constant *
5254CGObjCNonFragileABIMac::EmitIvarOffsetVar(const ObjCInterfaceDecl *ID,
5255 const ObjCIvarDecl *Ivar,
5256 unsigned long int Offset) {
Daniel Dunbar737c5022009-04-19 00:44:02 +00005257 llvm::GlobalVariable *IvarOffsetGV = ObjCIvarOffsetVariable(ID, Ivar);
Daniel Dunbar6bff2512009-08-03 17:06:42 +00005258 IvarOffsetGV->setInitializer(llvm::ConstantInt::get(ObjCTypes.LongTy,
Daniel Dunbar737c5022009-04-19 00:44:02 +00005259 Offset));
Fariborz Jahanian09796d62009-01-31 02:43:27 +00005260 IvarOffsetGV->setAlignment(
Daniel Dunbar4b429ae2010-04-25 20:39:32 +00005261 CGM.getTargetData().getABITypeAlignment(ObjCTypes.LongTy));
Daniel Dunbar737c5022009-04-19 00:44:02 +00005262
Mike Stumpf5408fe2009-05-16 07:57:57 +00005263 // FIXME: This matches gcc, but shouldn't the visibility be set on the use as
5264 // well (i.e., in ObjCIvarOffsetVariable).
Daniel Dunbar737c5022009-04-19 00:44:02 +00005265 if (Ivar->getAccessControl() == ObjCIvarDecl::Private ||
5266 Ivar->getAccessControl() == ObjCIvarDecl::Package ||
5267 CGM.getDeclVisibilityMode(ID) == LangOptions::Hidden)
Fariborz Jahanian2fa5a272009-01-28 01:36:42 +00005268 IvarOffsetGV->setVisibility(llvm::GlobalValue::HiddenVisibility);
Daniel Dunbar04d40782009-04-14 06:00:08 +00005269 else
Fariborz Jahanian77c9fd22009-04-06 18:30:00 +00005270 IvarOffsetGV->setVisibility(llvm::GlobalValue::DefaultVisibility);
Fariborz Jahanian1bf0afb2009-01-28 01:05:23 +00005271 IvarOffsetGV->setSection("__DATA, __objc_const");
Fariborz Jahanian45012a72009-02-03 00:09:52 +00005272 return IvarOffsetGV;
Fariborz Jahanian1bf0afb2009-01-28 01:05:23 +00005273}
5274
Fariborz Jahanian98abf4b2009-01-27 19:38:51 +00005275/// EmitIvarList - Emit the ivar list for the given
Daniel Dunbar11394522009-04-18 08:51:00 +00005276/// implementation. The return value has type
Fariborz Jahanian98abf4b2009-01-27 19:38:51 +00005277/// IvarListnfABIPtrTy.
5278/// struct _ivar_t {
5279/// unsigned long int *offset; // pointer to ivar offset location
5280/// char *name;
5281/// char *type;
5282/// uint32_t alignment;
5283/// uint32_t size;
5284/// }
5285/// struct _ivar_list_t {
5286/// uint32 entsize; // sizeof(struct _ivar_t)
5287/// uint32 count;
5288/// struct _iver_t list[count];
5289/// }
5290///
Daniel Dunbar3e5f0d82009-04-20 06:54:31 +00005291
Fariborz Jahanian98abf4b2009-01-27 19:38:51 +00005292llvm::Constant *CGObjCNonFragileABIMac::EmitIvarList(
Daniel Dunbar6bff2512009-08-03 17:06:42 +00005293 const ObjCImplementationDecl *ID) {
5294
Fariborz Jahanian98abf4b2009-01-27 19:38:51 +00005295 std::vector<llvm::Constant*> Ivars, Ivar(5);
Daniel Dunbar6bff2512009-08-03 17:06:42 +00005296
Fariborz Jahanian98abf4b2009-01-27 19:38:51 +00005297 const ObjCInterfaceDecl *OID = ID->getClassInterface();
5298 assert(OID && "CGObjCNonFragileABIMac::EmitIvarList - null interface");
Daniel Dunbar6bff2512009-08-03 17:06:42 +00005299
Fariborz Jahanian1bf0afb2009-01-28 01:05:23 +00005300 // FIXME. Consolidate this with similar code in GenerateClass.
Daniel Dunbar6bff2512009-08-03 17:06:42 +00005301
Daniel Dunbar91636d62009-04-20 00:33:43 +00005302 // Collect declared and synthesized ivars in a small vector.
Fariborz Jahanian18191882009-03-31 18:11:23 +00005303 llvm::SmallVector<ObjCIvarDecl*, 16> OIvars;
Fariborz Jahanian8e6ac1d2009-06-04 01:19:09 +00005304 CGM.getContext().ShallowCollectObjCIvars(OID, OIvars);
Daniel Dunbar6bff2512009-08-03 17:06:42 +00005305
Daniel Dunbar3e5f0d82009-04-20 06:54:31 +00005306 for (unsigned i = 0, e = OIvars.size(); i != e; ++i) {
5307 ObjCIvarDecl *IVD = OIvars[i];
Fariborz Jahanian8e6ac1d2009-06-04 01:19:09 +00005308 // Ignore unnamed bit-fields.
5309 if (!IVD->getDeclName())
5310 continue;
Daniel Dunbar6bff2512009-08-03 17:06:42 +00005311 Ivar[0] = EmitIvarOffsetVar(ID->getClassInterface(), IVD,
Daniel Dunbar9f89f2b2009-05-03 12:57:56 +00005312 ComputeIvarBaseOffset(CGM, ID, IVD));
Daniel Dunbar3fea0c02009-04-22 08:22:17 +00005313 Ivar[1] = GetMethodVarName(IVD->getIdentifier());
5314 Ivar[2] = GetMethodVarType(IVD);
Fariborz Jahanian98abf4b2009-01-27 19:38:51 +00005315 const llvm::Type *FieldTy =
Daniel Dunbar3fea0c02009-04-22 08:22:17 +00005316 CGM.getTypes().ConvertTypeForMem(IVD->getType());
Duncan Sands9408c452009-05-09 07:08:47 +00005317 unsigned Size = CGM.getTargetData().getTypeAllocSize(FieldTy);
Fariborz Jahanian98abf4b2009-01-27 19:38:51 +00005318 unsigned Align = CGM.getContext().getPreferredTypeAlign(
Daniel Dunbar6bff2512009-08-03 17:06:42 +00005319 IVD->getType().getTypePtr()) >> 3;
Fariborz Jahanian98abf4b2009-01-27 19:38:51 +00005320 Align = llvm::Log2_32(Align);
Owen Anderson4a28d5d2009-07-24 23:12:58 +00005321 Ivar[3] = llvm::ConstantInt::get(ObjCTypes.IntTy, Align);
Daniel Dunbar91636d62009-04-20 00:33:43 +00005322 // NOTE. Size of a bitfield does not match gcc's, because of the
5323 // way bitfields are treated special in each. But I am told that
5324 // 'size' for bitfield ivars is ignored by the runtime so it does
5325 // not matter. If it matters, there is enough info to get the
5326 // bitfield right!
Owen Anderson4a28d5d2009-07-24 23:12:58 +00005327 Ivar[4] = llvm::ConstantInt::get(ObjCTypes.IntTy, Size);
Owen Anderson08e25242009-07-27 22:29:56 +00005328 Ivars.push_back(llvm::ConstantStruct::get(ObjCTypes.IvarnfABITy, Ivar));
Fariborz Jahanian98abf4b2009-01-27 19:38:51 +00005329 }
5330 // Return null for empty list.
5331 if (Ivars.empty())
Owen Andersonc9c88b42009-07-31 20:28:54 +00005332 return llvm::Constant::getNullValue(ObjCTypes.IvarListnfABIPtrTy);
Fariborz Jahanian98abf4b2009-01-27 19:38:51 +00005333 std::vector<llvm::Constant*> Values(3);
Duncan Sands9408c452009-05-09 07:08:47 +00005334 unsigned Size = CGM.getTargetData().getTypeAllocSize(ObjCTypes.IvarnfABITy);
Owen Anderson4a28d5d2009-07-24 23:12:58 +00005335 Values[0] = llvm::ConstantInt::get(ObjCTypes.IntTy, Size);
5336 Values[1] = llvm::ConstantInt::get(ObjCTypes.IntTy, Ivars.size());
Owen Anderson96e0fc72009-07-29 22:16:19 +00005337 llvm::ArrayType *AT = llvm::ArrayType::get(ObjCTypes.IvarnfABITy,
Fariborz Jahanian98abf4b2009-01-27 19:38:51 +00005338 Ivars.size());
Owen Anderson7db6d832009-07-28 18:33:04 +00005339 Values[2] = llvm::ConstantArray::get(AT, Ivars);
Nick Lewycky0d36dd22009-09-19 20:00:52 +00005340 llvm::Constant *Init = llvm::ConstantStruct::get(VMContext, Values, false);
Fariborz Jahanian98abf4b2009-01-27 19:38:51 +00005341 const char *Prefix = "\01l_OBJC_$_INSTANCE_VARIABLES_";
5342 llvm::GlobalVariable *GV =
Owen Anderson1c431b32009-07-08 19:05:04 +00005343 new llvm::GlobalVariable(CGM.getModule(), Init->getType(), false,
Fariborz Jahanian98abf4b2009-01-27 19:38:51 +00005344 llvm::GlobalValue::InternalLinkage,
5345 Init,
Daniel Dunbar9c29bf52009-10-18 20:48:59 +00005346 Prefix + OID->getName());
Fariborz Jahanian09796d62009-01-31 02:43:27 +00005347 GV->setAlignment(
Daniel Dunbar4b429ae2010-04-25 20:39:32 +00005348 CGM.getTargetData().getABITypeAlignment(Init->getType()));
Fariborz Jahanian1bf0afb2009-01-28 01:05:23 +00005349 GV->setSection("__DATA, __objc_const");
Daniel Dunbar6bff2512009-08-03 17:06:42 +00005350
Chris Lattnerad64e022009-07-17 23:57:13 +00005351 CGM.AddUsedGlobal(GV);
Owen Anderson3c4972d2009-07-29 18:54:39 +00005352 return llvm::ConstantExpr::getBitCast(GV, ObjCTypes.IvarListnfABIPtrTy);
Fariborz Jahanianda320092009-01-29 19:24:30 +00005353}
5354
5355llvm::Constant *CGObjCNonFragileABIMac::GetOrEmitProtocolRef(
Daniel Dunbar6bff2512009-08-03 17:06:42 +00005356 const ObjCProtocolDecl *PD) {
Fariborz Jahanianda320092009-01-29 19:24:30 +00005357 llvm::GlobalVariable *&Entry = Protocols[PD->getIdentifier()];
Daniel Dunbar6bff2512009-08-03 17:06:42 +00005358
Fariborz Jahanianda320092009-01-29 19:24:30 +00005359 if (!Entry) {
5360 // We use the initializer as a marker of whether this is a forward
5361 // reference or not. At module finalization we add the empty
5362 // contents for protocols which were referenced but never defined.
Daniel Dunbar6bff2512009-08-03 17:06:42 +00005363 Entry =
5364 new llvm::GlobalVariable(CGM.getModule(), ObjCTypes.ProtocolnfABITy, false,
5365 llvm::GlobalValue::ExternalLinkage,
5366 0,
Daniel Dunbar9c29bf52009-10-18 20:48:59 +00005367 "\01l_OBJC_PROTOCOL_$_" + PD->getName());
Fariborz Jahanianda320092009-01-29 19:24:30 +00005368 Entry->setSection("__DATA,__datacoal_nt,coalesced");
Fariborz Jahanianda320092009-01-29 19:24:30 +00005369 }
Daniel Dunbar6bff2512009-08-03 17:06:42 +00005370
Fariborz Jahanianda320092009-01-29 19:24:30 +00005371 return Entry;
5372}
5373
5374/// GetOrEmitProtocol - Generate the protocol meta-data:
5375/// @code
5376/// struct _protocol_t {
5377/// id isa; // NULL
5378/// const char * const protocol_name;
5379/// const struct _protocol_list_t * protocol_list; // super protocols
5380/// const struct method_list_t * const instance_methods;
5381/// const struct method_list_t * const class_methods;
5382/// const struct method_list_t *optionalInstanceMethods;
5383/// const struct method_list_t *optionalClassMethods;
5384/// const struct _prop_list_t * properties;
5385/// const uint32_t size; // sizeof(struct _protocol_t)
5386/// const uint32_t flags; // = 0
5387/// }
5388/// @endcode
5389///
5390
5391llvm::Constant *CGObjCNonFragileABIMac::GetOrEmitProtocol(
Daniel Dunbar6bff2512009-08-03 17:06:42 +00005392 const ObjCProtocolDecl *PD) {
Fariborz Jahanianda320092009-01-29 19:24:30 +00005393 llvm::GlobalVariable *&Entry = Protocols[PD->getIdentifier()];
Daniel Dunbar6bff2512009-08-03 17:06:42 +00005394
Fariborz Jahanianda320092009-01-29 19:24:30 +00005395 // Early exit if a defining object has already been generated.
5396 if (Entry && Entry->hasInitializer())
5397 return Entry;
5398
Fariborz Jahanianda320092009-01-29 19:24:30 +00005399 // Construct method lists.
5400 std::vector<llvm::Constant*> InstanceMethods, ClassMethods;
5401 std::vector<llvm::Constant*> OptInstanceMethods, OptClassMethods;
Daniel Dunbar6bff2512009-08-03 17:06:42 +00005402 for (ObjCProtocolDecl::instmeth_iterator
Argyrios Kyrtzidis17945a02009-06-30 02:36:12 +00005403 i = PD->instmeth_begin(), e = PD->instmeth_end(); i != e; ++i) {
Fariborz Jahanianda320092009-01-29 19:24:30 +00005404 ObjCMethodDecl *MD = *i;
Fariborz Jahanian3819a0b2009-01-30 00:46:37 +00005405 llvm::Constant *C = GetMethodDescriptionConstant(MD);
Fariborz Jahanianda320092009-01-29 19:24:30 +00005406 if (MD->getImplementationControl() == ObjCMethodDecl::Optional) {
5407 OptInstanceMethods.push_back(C);
5408 } else {
5409 InstanceMethods.push_back(C);
Daniel Dunbar6bff2512009-08-03 17:06:42 +00005410 }
Fariborz Jahanianda320092009-01-29 19:24:30 +00005411 }
Daniel Dunbar6bff2512009-08-03 17:06:42 +00005412
5413 for (ObjCProtocolDecl::classmeth_iterator
Argyrios Kyrtzidis17945a02009-06-30 02:36:12 +00005414 i = PD->classmeth_begin(), e = PD->classmeth_end(); i != e; ++i) {
Fariborz Jahanianda320092009-01-29 19:24:30 +00005415 ObjCMethodDecl *MD = *i;
Fariborz Jahanian3819a0b2009-01-30 00:46:37 +00005416 llvm::Constant *C = GetMethodDescriptionConstant(MD);
Fariborz Jahanianda320092009-01-29 19:24:30 +00005417 if (MD->getImplementationControl() == ObjCMethodDecl::Optional) {
5418 OptClassMethods.push_back(C);
5419 } else {
5420 ClassMethods.push_back(C);
Daniel Dunbar6bff2512009-08-03 17:06:42 +00005421 }
Fariborz Jahanianda320092009-01-29 19:24:30 +00005422 }
Daniel Dunbar6bff2512009-08-03 17:06:42 +00005423
Fariborz Jahanianda320092009-01-29 19:24:30 +00005424 std::vector<llvm::Constant*> Values(10);
5425 // isa is NULL
Owen Andersonc9c88b42009-07-31 20:28:54 +00005426 Values[0] = llvm::Constant::getNullValue(ObjCTypes.ObjectPtrTy);
Fariborz Jahanianda320092009-01-29 19:24:30 +00005427 Values[1] = GetClassName(PD->getIdentifier());
Daniel Dunbar9c29bf52009-10-18 20:48:59 +00005428 Values[2] = EmitProtocolList("\01l_OBJC_$_PROTOCOL_REFS_" + PD->getName(),
5429 PD->protocol_begin(),
5430 PD->protocol_end());
Daniel Dunbar6bff2512009-08-03 17:06:42 +00005431
Fariborz Jahanian3819a0b2009-01-30 00:46:37 +00005432 Values[3] = EmitMethodList("\01l_OBJC_$_PROTOCOL_INSTANCE_METHODS_"
Daniel Dunbar9c29bf52009-10-18 20:48:59 +00005433 + PD->getName(),
Fariborz Jahanianda320092009-01-29 19:24:30 +00005434 "__DATA, __objc_const",
5435 InstanceMethods);
Daniel Dunbar6bff2512009-08-03 17:06:42 +00005436 Values[4] = EmitMethodList("\01l_OBJC_$_PROTOCOL_CLASS_METHODS_"
Daniel Dunbar9c29bf52009-10-18 20:48:59 +00005437 + PD->getName(),
Fariborz Jahanianda320092009-01-29 19:24:30 +00005438 "__DATA, __objc_const",
5439 ClassMethods);
Fariborz Jahanian3819a0b2009-01-30 00:46:37 +00005440 Values[5] = EmitMethodList("\01l_OBJC_$_PROTOCOL_INSTANCE_METHODS_OPT_"
Daniel Dunbar9c29bf52009-10-18 20:48:59 +00005441 + PD->getName(),
Fariborz Jahanianda320092009-01-29 19:24:30 +00005442 "__DATA, __objc_const",
5443 OptInstanceMethods);
Daniel Dunbar6bff2512009-08-03 17:06:42 +00005444 Values[6] = EmitMethodList("\01l_OBJC_$_PROTOCOL_CLASS_METHODS_OPT_"
Daniel Dunbar9c29bf52009-10-18 20:48:59 +00005445 + PD->getName(),
Fariborz Jahanianda320092009-01-29 19:24:30 +00005446 "__DATA, __objc_const",
5447 OptClassMethods);
Daniel Dunbar9c29bf52009-10-18 20:48:59 +00005448 Values[7] = EmitPropertyList("\01l_OBJC_$_PROP_LIST_" + PD->getName(),
Fariborz Jahanianda320092009-01-29 19:24:30 +00005449 0, PD, ObjCTypes);
Daniel Dunbar6bff2512009-08-03 17:06:42 +00005450 uint32_t Size =
Duncan Sands9408c452009-05-09 07:08:47 +00005451 CGM.getTargetData().getTypeAllocSize(ObjCTypes.ProtocolnfABITy);
Owen Anderson4a28d5d2009-07-24 23:12:58 +00005452 Values[8] = llvm::ConstantInt::get(ObjCTypes.IntTy, Size);
Owen Andersonc9c88b42009-07-31 20:28:54 +00005453 Values[9] = llvm::Constant::getNullValue(ObjCTypes.IntTy);
Owen Anderson08e25242009-07-27 22:29:56 +00005454 llvm::Constant *Init = llvm::ConstantStruct::get(ObjCTypes.ProtocolnfABITy,
Fariborz Jahanianda320092009-01-29 19:24:30 +00005455 Values);
Daniel Dunbar6bff2512009-08-03 17:06:42 +00005456
Fariborz Jahanianda320092009-01-29 19:24:30 +00005457 if (Entry) {
5458 // Already created, fix the linkage and update the initializer.
Mike Stump286acbd2009-03-07 16:33:28 +00005459 Entry->setLinkage(llvm::GlobalValue::WeakAnyLinkage);
Fariborz Jahanianda320092009-01-29 19:24:30 +00005460 Entry->setInitializer(Init);
5461 } else {
Daniel Dunbar6bff2512009-08-03 17:06:42 +00005462 Entry =
Daniel Dunbar9c29bf52009-10-18 20:48:59 +00005463 new llvm::GlobalVariable(CGM.getModule(), ObjCTypes.ProtocolnfABITy,
5464 false, llvm::GlobalValue::WeakAnyLinkage, Init,
5465 "\01l_OBJC_PROTOCOL_$_" + PD->getName());
Fariborz Jahanian09796d62009-01-31 02:43:27 +00005466 Entry->setAlignment(
Daniel Dunbar4b429ae2010-04-25 20:39:32 +00005467 CGM.getTargetData().getABITypeAlignment(ObjCTypes.ProtocolnfABITy));
Fariborz Jahanianda320092009-01-29 19:24:30 +00005468 Entry->setSection("__DATA,__datacoal_nt,coalesced");
Fariborz Jahanianda320092009-01-29 19:24:30 +00005469 }
Fariborz Jahanian8448c2c2009-01-29 20:10:59 +00005470 Entry->setVisibility(llvm::GlobalValue::HiddenVisibility);
Chris Lattnerad64e022009-07-17 23:57:13 +00005471 CGM.AddUsedGlobal(Entry);
5472
Fariborz Jahanian8448c2c2009-01-29 20:10:59 +00005473 // Use this protocol meta-data to build protocol list table in section
5474 // __DATA, __objc_protolist
Daniel Dunbar9c29bf52009-10-18 20:48:59 +00005475 llvm::GlobalVariable *PTGV =
5476 new llvm::GlobalVariable(CGM.getModule(), ObjCTypes.ProtocolnfABIPtrTy,
5477 false, llvm::GlobalValue::WeakAnyLinkage, Entry,
5478 "\01l_OBJC_LABEL_PROTOCOL_$_" + PD->getName());
Fariborz Jahanian09796d62009-01-31 02:43:27 +00005479 PTGV->setAlignment(
Daniel Dunbar4b429ae2010-04-25 20:39:32 +00005480 CGM.getTargetData().getABITypeAlignment(ObjCTypes.ProtocolnfABIPtrTy));
Daniel Dunbar0bf21992009-04-15 02:56:18 +00005481 PTGV->setSection("__DATA, __objc_protolist, coalesced, no_dead_strip");
Fariborz Jahanian8448c2c2009-01-29 20:10:59 +00005482 PTGV->setVisibility(llvm::GlobalValue::HiddenVisibility);
Chris Lattnerad64e022009-07-17 23:57:13 +00005483 CGM.AddUsedGlobal(PTGV);
Fariborz Jahanianda320092009-01-29 19:24:30 +00005484 return Entry;
5485}
5486
5487/// EmitProtocolList - Generate protocol list meta-data:
5488/// @code
5489/// struct _protocol_list_t {
5490/// long protocol_count; // Note, this is 32/64 bit
5491/// struct _protocol_t[protocol_count];
5492/// }
5493/// @endcode
5494///
5495llvm::Constant *
Daniel Dunbar9c29bf52009-10-18 20:48:59 +00005496CGObjCNonFragileABIMac::EmitProtocolList(llvm::Twine Name,
5497 ObjCProtocolDecl::protocol_iterator begin,
5498 ObjCProtocolDecl::protocol_iterator end) {
Fariborz Jahanianda320092009-01-29 19:24:30 +00005499 std::vector<llvm::Constant*> ProtocolRefs;
Daniel Dunbar6bff2512009-08-03 17:06:42 +00005500
Fariborz Jahanianda320092009-01-29 19:24:30 +00005501 // Just return null for empty protocol lists
Daniel Dunbar6bff2512009-08-03 17:06:42 +00005502 if (begin == end)
Owen Andersonc9c88b42009-07-31 20:28:54 +00005503 return llvm::Constant::getNullValue(ObjCTypes.ProtocolListnfABIPtrTy);
Daniel Dunbar6bff2512009-08-03 17:06:42 +00005504
Daniel Dunbar948e2582009-02-15 07:36:20 +00005505 // FIXME: We shouldn't need to do this lookup here, should we?
Daniel Dunbar9c29bf52009-10-18 20:48:59 +00005506 llvm::SmallString<256> TmpName;
5507 Name.toVector(TmpName);
5508 llvm::GlobalVariable *GV =
5509 CGM.getModule().getGlobalVariable(TmpName.str(), true);
Fariborz Jahanianda320092009-01-29 19:24:30 +00005510 if (GV)
Daniel Dunbar9c29bf52009-10-18 20:48:59 +00005511 return llvm::ConstantExpr::getBitCast(GV, ObjCTypes.ProtocolListnfABIPtrTy);
Daniel Dunbar6bff2512009-08-03 17:06:42 +00005512
Daniel Dunbar948e2582009-02-15 07:36:20 +00005513 for (; begin != end; ++begin)
5514 ProtocolRefs.push_back(GetProtocolRef(*begin)); // Implemented???
5515
Fariborz Jahanianda320092009-01-29 19:24:30 +00005516 // This list is null terminated.
Owen Andersonc9c88b42009-07-31 20:28:54 +00005517 ProtocolRefs.push_back(llvm::Constant::getNullValue(
Daniel Dunbar6bff2512009-08-03 17:06:42 +00005518 ObjCTypes.ProtocolnfABIPtrTy));
5519
Fariborz Jahanianda320092009-01-29 19:24:30 +00005520 std::vector<llvm::Constant*> Values(2);
Owen Andersona1cf15f2009-07-14 23:10:40 +00005521 Values[0] =
Owen Anderson4a28d5d2009-07-24 23:12:58 +00005522 llvm::ConstantInt::get(ObjCTypes.LongTy, ProtocolRefs.size() - 1);
Daniel Dunbar6bff2512009-08-03 17:06:42 +00005523 Values[1] =
Owen Anderson7db6d832009-07-28 18:33:04 +00005524 llvm::ConstantArray::get(
Owen Anderson96e0fc72009-07-29 22:16:19 +00005525 llvm::ArrayType::get(ObjCTypes.ProtocolnfABIPtrTy,
Daniel Dunbar6bff2512009-08-03 17:06:42 +00005526 ProtocolRefs.size()),
5527 ProtocolRefs);
5528
Nick Lewycky0d36dd22009-09-19 20:00:52 +00005529 llvm::Constant *Init = llvm::ConstantStruct::get(VMContext, Values, false);
Owen Anderson1c431b32009-07-08 19:05:04 +00005530 GV = new llvm::GlobalVariable(CGM.getModule(), Init->getType(), false,
Fariborz Jahanianda320092009-01-29 19:24:30 +00005531 llvm::GlobalValue::InternalLinkage,
5532 Init,
Owen Anderson1c431b32009-07-08 19:05:04 +00005533 Name);
Fariborz Jahanianda320092009-01-29 19:24:30 +00005534 GV->setSection("__DATA, __objc_const");
Fariborz Jahanian09796d62009-01-31 02:43:27 +00005535 GV->setAlignment(
Daniel Dunbar4b429ae2010-04-25 20:39:32 +00005536 CGM.getTargetData().getABITypeAlignment(Init->getType()));
Chris Lattnerad64e022009-07-17 23:57:13 +00005537 CGM.AddUsedGlobal(GV);
Daniel Dunbar6bff2512009-08-03 17:06:42 +00005538 return llvm::ConstantExpr::getBitCast(GV,
Daniel Dunbar948e2582009-02-15 07:36:20 +00005539 ObjCTypes.ProtocolListnfABIPtrTy);
Fariborz Jahanianda320092009-01-29 19:24:30 +00005540}
5541
Fariborz Jahanian3819a0b2009-01-30 00:46:37 +00005542/// GetMethodDescriptionConstant - This routine build following meta-data:
5543/// struct _objc_method {
5544/// SEL _cmd;
5545/// char *method_type;
5546/// char *_imp;
5547/// }
5548
5549llvm::Constant *
5550CGObjCNonFragileABIMac::GetMethodDescriptionConstant(const ObjCMethodDecl *MD) {
5551 std::vector<llvm::Constant*> Desc(3);
Owen Andersona1cf15f2009-07-14 23:10:40 +00005552 Desc[0] =
Daniel Dunbar6bff2512009-08-03 17:06:42 +00005553 llvm::ConstantExpr::getBitCast(GetMethodVarName(MD->getSelector()),
5554 ObjCTypes.SelectorPtrTy);
Fariborz Jahanian3819a0b2009-01-30 00:46:37 +00005555 Desc[1] = GetMethodVarType(MD);
Fariborz Jahanian8cfd3972009-01-30 18:58:59 +00005556 // Protocol methods have no implementation. So, this entry is always NULL.
Owen Andersonc9c88b42009-07-31 20:28:54 +00005557 Desc[2] = llvm::Constant::getNullValue(ObjCTypes.Int8PtrTy);
Owen Anderson08e25242009-07-27 22:29:56 +00005558 return llvm::ConstantStruct::get(ObjCTypes.MethodTy, Desc);
Fariborz Jahanian3819a0b2009-01-30 00:46:37 +00005559}
Fariborz Jahanian45012a72009-02-03 00:09:52 +00005560
5561/// EmitObjCValueForIvar - Code Gen for nonfragile ivar reference.
5562/// This code gen. amounts to generating code for:
5563/// @code
5564/// (type *)((char *)base + _OBJC_IVAR_$_.ivar;
5565/// @encode
Daniel Dunbar6bff2512009-08-03 17:06:42 +00005566///
Fariborz Jahanian598d3f62009-02-03 19:03:09 +00005567LValue CGObjCNonFragileABIMac::EmitObjCValueForIvar(
John McCall506b57e2010-05-17 21:00:27 +00005568 CodeGen::CodeGenFunction &CGF,
5569 QualType ObjectTy,
5570 llvm::Value *BaseValue,
5571 const ObjCIvarDecl *Ivar,
5572 unsigned CVRQualifiers) {
5573 ObjCInterfaceDecl *ID = ObjectTy->getAs<ObjCObjectType>()->getInterface();
Daniel Dunbar97776872009-04-22 07:32:20 +00005574 return EmitValueForIvarAtOffset(CGF, ID, BaseValue, Ivar, CVRQualifiers,
5575 EmitIvarOffset(CGF, ID, Ivar));
Fariborz Jahanian45012a72009-02-03 00:09:52 +00005576}
5577
Fariborz Jahanianf63aa3f2009-02-10 19:02:04 +00005578llvm::Value *CGObjCNonFragileABIMac::EmitIvarOffset(
Daniel Dunbar6bff2512009-08-03 17:06:42 +00005579 CodeGen::CodeGenFunction &CGF,
5580 const ObjCInterfaceDecl *Interface,
5581 const ObjCIvarDecl *Ivar) {
Daniel Dunbar2da84ff2009-11-29 21:23:36 +00005582 return CGF.Builder.CreateLoad(ObjCIvarOffsetVariable(Interface, Ivar),"ivar");
Fariborz Jahanianf63aa3f2009-02-10 19:02:04 +00005583}
5584
Fariborz Jahanian46551122009-02-04 00:22:57 +00005585CodeGen::RValue CGObjCNonFragileABIMac::EmitMessageSend(
Daniel Dunbar6bff2512009-08-03 17:06:42 +00005586 CodeGen::CodeGenFunction &CGF,
John McCallef072fd2010-05-22 01:48:05 +00005587 ReturnValueSlot Return,
Daniel Dunbar6bff2512009-08-03 17:06:42 +00005588 QualType ResultType,
5589 Selector Sel,
5590 llvm::Value *Receiver,
5591 QualType Arg0Ty,
5592 bool IsSuper,
5593 const CallArgList &CallArgs) {
Mike Stumpf5408fe2009-05-16 07:57:57 +00005594 // FIXME. Even though IsSuper is passes. This function doese not handle calls
5595 // to 'super' receivers.
Fariborz Jahanian83a8a752009-02-04 20:42:28 +00005596 CodeGenTypes &Types = CGM.getTypes();
Fariborz Jahanian7a06aae2009-02-06 20:09:23 +00005597 llvm::Value *Arg0 = Receiver;
5598 if (!IsSuper)
5599 Arg0 = CGF.Builder.CreateBitCast(Arg0, ObjCTypes.ObjectPtrTy, "tmp");
Daniel Dunbar6bff2512009-08-03 17:06:42 +00005600
Fariborz Jahanian83a8a752009-02-04 20:42:28 +00005601 // Find the message function name.
Mike Stumpf5408fe2009-05-16 07:57:57 +00005602 // FIXME. This is too much work to get the ABI-specific result type needed to
5603 // find the message name.
John McCallead608a2010-02-26 00:48:12 +00005604 const CGFunctionInfo &FnInfo
Rafael Espindola264ba482010-03-30 20:24:48 +00005605 = Types.getFunctionInfo(ResultType, CallArgList(),
5606 FunctionType::ExtInfo());
Fariborz Jahanian70b51c72009-04-30 23:08:58 +00005607 llvm::Constant *Fn = 0;
Fariborz Jahanian83a8a752009-02-04 20:42:28 +00005608 std::string Name("\01l_");
Daniel Dunbardacf9dd2010-07-14 23:39:36 +00005609 if (CGM.ReturnTypeUsesSRet(FnInfo)) {
Fariborz Jahanianc1708522009-02-05 18:00:27 +00005610#if 0
5611 // unlike what is documented. gcc never generates this API!!
Fariborz Jahanian83a8a752009-02-04 20:42:28 +00005612 if (Receiver->getType() == ObjCTypes.ObjectPtrTy) {
Chris Lattner1c02f862009-04-22 02:53:24 +00005613 Fn = ObjCTypes.getMessageSendIdStretFixupFn();
Fariborz Jahanian83a8a752009-02-04 20:42:28 +00005614 // FIXME. Is there a better way of getting these names.
5615 // They are available in RuntimeFunctions vector pair.
5616 Name += "objc_msgSendId_stret_fixup";
Mike Stumpb3589f42009-07-30 22:28:39 +00005617 } else
Fariborz Jahanianc1708522009-02-05 18:00:27 +00005618#endif
Daniel Dunbar6bff2512009-08-03 17:06:42 +00005619 if (IsSuper) {
Chris Lattner1c02f862009-04-22 02:53:24 +00005620 Fn = ObjCTypes.getMessageSendSuper2StretFixupFn();
Fariborz Jahanian7a06aae2009-02-06 20:09:23 +00005621 Name += "objc_msgSendSuper2_stret_fixup";
Daniel Dunbar6bff2512009-08-03 17:06:42 +00005622 } else {
5623 Fn = ObjCTypes.getMessageSendStretFixupFn();
5624 Name += "objc_msgSend_stret_fixup";
5625 }
Daniel Dunbardacf9dd2010-07-14 23:39:36 +00005626 } else if (!IsSuper && CGM.ReturnTypeUsesFPRet(ResultType)) {
5627 Fn = ObjCTypes.getMessageSendFpretFixupFn();
5628 Name += "objc_msgSend_fpret_fixup";
Mike Stumpb3589f42009-07-30 22:28:39 +00005629 } else {
Fariborz Jahanianc1708522009-02-05 18:00:27 +00005630#if 0
5631// unlike what is documented. gcc never generates this API!!
Fariborz Jahanian83a8a752009-02-04 20:42:28 +00005632 if (Receiver->getType() == ObjCTypes.ObjectPtrTy) {
Chris Lattner1c02f862009-04-22 02:53:24 +00005633 Fn = ObjCTypes.getMessageSendIdFixupFn();
Fariborz Jahanian83a8a752009-02-04 20:42:28 +00005634 Name += "objc_msgSendId_fixup";
Daniel Dunbar6bff2512009-08-03 17:06:42 +00005635 } else
Fariborz Jahanianc1708522009-02-05 18:00:27 +00005636#endif
Daniel Dunbar6bff2512009-08-03 17:06:42 +00005637 if (IsSuper) {
Chris Lattner1c02f862009-04-22 02:53:24 +00005638 Fn = ObjCTypes.getMessageSendSuper2FixupFn();
Fariborz Jahanian7a06aae2009-02-06 20:09:23 +00005639 Name += "objc_msgSendSuper2_fixup";
Daniel Dunbar6bff2512009-08-03 17:06:42 +00005640 } else {
5641 Fn = ObjCTypes.getMessageSendFixupFn();
5642 Name += "objc_msgSend_fixup";
5643 }
Fariborz Jahanian83a8a752009-02-04 20:42:28 +00005644 }
Fariborz Jahanian70b51c72009-04-30 23:08:58 +00005645 assert(Fn && "CGObjCNonFragileABIMac::EmitMessageSend");
Fariborz Jahanian83a8a752009-02-04 20:42:28 +00005646 Name += '_';
5647 std::string SelName(Sel.getAsString());
5648 // Replace all ':' in selector name with '_' ouch!
Mike Stump1eb44332009-09-09 15:08:12 +00005649 for (unsigned i = 0; i < SelName.size(); i++)
Fariborz Jahanian83a8a752009-02-04 20:42:28 +00005650 if (SelName[i] == ':')
5651 SelName[i] = '_';
5652 Name += SelName;
5653 llvm::GlobalVariable *GV = CGM.getModule().getGlobalVariable(Name);
5654 if (!GV) {
Daniel Dunbar33af70f2009-04-15 19:03:14 +00005655 // Build message ref table entry.
Fariborz Jahanian83a8a752009-02-04 20:42:28 +00005656 std::vector<llvm::Constant*> Values(2);
5657 Values[0] = Fn;
5658 Values[1] = GetMethodVarName(Sel);
Nick Lewycky0d36dd22009-09-19 20:00:52 +00005659 llvm::Constant *Init = llvm::ConstantStruct::get(VMContext, Values, false);
Owen Anderson1c431b32009-07-08 19:05:04 +00005660 GV = new llvm::GlobalVariable(CGM.getModule(), Init->getType(), false,
Mike Stump286acbd2009-03-07 16:33:28 +00005661 llvm::GlobalValue::WeakAnyLinkage,
Fariborz Jahanian83a8a752009-02-04 20:42:28 +00005662 Init,
Owen Anderson1c431b32009-07-08 19:05:04 +00005663 Name);
Fariborz Jahanian83a8a752009-02-04 20:42:28 +00005664 GV->setVisibility(llvm::GlobalValue::HiddenVisibility);
Daniel Dunbarf59c1a62009-04-15 19:04:46 +00005665 GV->setAlignment(16);
Fariborz Jahanian83a8a752009-02-04 20:42:28 +00005666 GV->setSection("__DATA, __objc_msgrefs, coalesced");
Fariborz Jahanian83a8a752009-02-04 20:42:28 +00005667 }
5668 llvm::Value *Arg1 = CGF.Builder.CreateBitCast(GV, ObjCTypes.MessageRefPtrTy);
Daniel Dunbar6bff2512009-08-03 17:06:42 +00005669
Fariborz Jahanian83a8a752009-02-04 20:42:28 +00005670 CallArgList ActualArgs;
5671 ActualArgs.push_back(std::make_pair(RValue::get(Arg0), Arg0Ty));
Daniel Dunbar6bff2512009-08-03 17:06:42 +00005672 ActualArgs.push_back(std::make_pair(RValue::get(Arg1),
Fariborz Jahanian83a8a752009-02-04 20:42:28 +00005673 ObjCTypes.MessageRefCPtrTy));
5674 ActualArgs.insert(ActualArgs.end(), CallArgs.begin(), CallArgs.end());
John McCall04a67a62010-02-05 21:31:56 +00005675 const CGFunctionInfo &FnInfo1 = Types.getFunctionInfo(ResultType, ActualArgs,
Rafael Espindola264ba482010-03-30 20:24:48 +00005676 FunctionType::ExtInfo());
Fariborz Jahanianef163782009-02-05 01:13:09 +00005677 llvm::Value *Callee = CGF.Builder.CreateStructGEP(Arg1, 0);
5678 Callee = CGF.Builder.CreateLoad(Callee);
Fariborz Jahanian3ab75bd2009-02-14 21:25:36 +00005679 const llvm::FunctionType *FTy = Types.GetFunctionType(FnInfo1, true);
Fariborz Jahanianef163782009-02-05 01:13:09 +00005680 Callee = CGF.Builder.CreateBitCast(Callee,
Owen Anderson96e0fc72009-07-29 22:16:19 +00005681 llvm::PointerType::getUnqual(FTy));
John McCallef072fd2010-05-22 01:48:05 +00005682 return CGF.EmitCall(FnInfo1, Callee, Return, ActualArgs);
Fariborz Jahanian46551122009-02-04 00:22:57 +00005683}
5684
5685/// Generate code for a message send expression in the nonfragile abi.
Daniel Dunbard6c93d72009-09-17 04:01:22 +00005686CodeGen::RValue
5687CGObjCNonFragileABIMac::GenerateMessageSend(CodeGen::CodeGenFunction &CGF,
John McCallef072fd2010-05-22 01:48:05 +00005688 ReturnValueSlot Return,
Daniel Dunbard6c93d72009-09-17 04:01:22 +00005689 QualType ResultType,
5690 Selector Sel,
5691 llvm::Value *Receiver,
Daniel Dunbard6c93d72009-09-17 04:01:22 +00005692 const CallArgList &CallArgs,
David Chisnallc6cd5fd2010-04-28 19:33:36 +00005693 const ObjCInterfaceDecl *Class,
Daniel Dunbard6c93d72009-09-17 04:01:22 +00005694 const ObjCMethodDecl *Method) {
Fariborz Jahaniand0f8a8d2009-05-11 19:25:47 +00005695 return LegacyDispatchedSelector(Sel)
John McCallef072fd2010-05-22 01:48:05 +00005696 ? EmitLegacyMessageSend(CGF, Return, ResultType,
5697 EmitSelector(CGF.Builder, Sel),
Daniel Dunbar6bff2512009-08-03 17:06:42 +00005698 Receiver, CGF.getContext().getObjCIdType(),
Daniel Dunbard6c93d72009-09-17 04:01:22 +00005699 false, CallArgs, Method, ObjCTypes)
John McCallef072fd2010-05-22 01:48:05 +00005700 : EmitMessageSend(CGF, Return, ResultType, Sel,
Daniel Dunbar6bff2512009-08-03 17:06:42 +00005701 Receiver, CGF.getContext().getObjCIdType(),
5702 false, CallArgs);
Fariborz Jahanian46551122009-02-04 00:22:57 +00005703}
5704
Daniel Dunbar5a7379a2009-03-01 04:40:10 +00005705llvm::GlobalVariable *
Fariborz Jahanian0f902942009-04-14 18:41:56 +00005706CGObjCNonFragileABIMac::GetClassGlobal(const std::string &Name) {
Daniel Dunbar5a7379a2009-03-01 04:40:10 +00005707 llvm::GlobalVariable *GV = CGM.getModule().getGlobalVariable(Name);
5708
Daniel Dunbardfff2302009-03-02 05:18:14 +00005709 if (!GV) {
Daniel Dunbar6bff2512009-08-03 17:06:42 +00005710 GV = new llvm::GlobalVariable(CGM.getModule(), ObjCTypes.ClassnfABITy,
Owen Anderson1c431b32009-07-08 19:05:04 +00005711 false, llvm::GlobalValue::ExternalLinkage,
5712 0, Name);
Daniel Dunbar5a7379a2009-03-01 04:40:10 +00005713 }
5714
5715 return GV;
5716}
5717
Daniel Dunbar6bff2512009-08-03 17:06:42 +00005718llvm::Value *CGObjCNonFragileABIMac::EmitClassRef(CGBuilderTy &Builder,
5719 const ObjCInterfaceDecl *ID) {
Fariborz Jahanian0e81f4b2009-02-05 20:41:40 +00005720 llvm::GlobalVariable *&Entry = ClassReferences[ID->getIdentifier()];
Daniel Dunbar6bff2512009-08-03 17:06:42 +00005721
Fariborz Jahanian0e81f4b2009-02-05 20:41:40 +00005722 if (!Entry) {
Daniel Dunbar6ab187a2009-04-07 05:48:37 +00005723 std::string ClassName(getClassSymbolPrefix() + ID->getNameAsString());
Daniel Dunbar5a7379a2009-03-01 04:40:10 +00005724 llvm::GlobalVariable *ClassGV = GetClassGlobal(ClassName);
Daniel Dunbar6bff2512009-08-03 17:06:42 +00005725 Entry =
5726 new llvm::GlobalVariable(CGM.getModule(), ObjCTypes.ClassnfABIPtrTy,
Owen Anderson1c431b32009-07-08 19:05:04 +00005727 false, llvm::GlobalValue::InternalLinkage,
Daniel Dunbar6bff2512009-08-03 17:06:42 +00005728 ClassGV,
Owen Anderson1c431b32009-07-08 19:05:04 +00005729 "\01L_OBJC_CLASSLIST_REFERENCES_$_");
Fariborz Jahanian0e81f4b2009-02-05 20:41:40 +00005730 Entry->setAlignment(
Daniel Dunbar4b429ae2010-04-25 20:39:32 +00005731 CGM.getTargetData().getABITypeAlignment(
Daniel Dunbar6bff2512009-08-03 17:06:42 +00005732 ObjCTypes.ClassnfABIPtrTy));
Daniel Dunbar11394522009-04-18 08:51:00 +00005733 Entry->setSection("__DATA, __objc_classrefs, regular, no_dead_strip");
Chris Lattnerad64e022009-07-17 23:57:13 +00005734 CGM.AddUsedGlobal(Entry);
Daniel Dunbar11394522009-04-18 08:51:00 +00005735 }
Daniel Dunbar6bff2512009-08-03 17:06:42 +00005736
Daniel Dunbar2da84ff2009-11-29 21:23:36 +00005737 return Builder.CreateLoad(Entry, "tmp");
Daniel Dunbar11394522009-04-18 08:51:00 +00005738}
Fariborz Jahanian0e81f4b2009-02-05 20:41:40 +00005739
Daniel Dunbar11394522009-04-18 08:51:00 +00005740llvm::Value *
Daniel Dunbar6bff2512009-08-03 17:06:42 +00005741CGObjCNonFragileABIMac::EmitSuperClassRef(CGBuilderTy &Builder,
Daniel Dunbar11394522009-04-18 08:51:00 +00005742 const ObjCInterfaceDecl *ID) {
5743 llvm::GlobalVariable *&Entry = SuperClassReferences[ID->getIdentifier()];
Daniel Dunbar6bff2512009-08-03 17:06:42 +00005744
Daniel Dunbar11394522009-04-18 08:51:00 +00005745 if (!Entry) {
5746 std::string ClassName(getClassSymbolPrefix() + ID->getNameAsString());
5747 llvm::GlobalVariable *ClassGV = GetClassGlobal(ClassName);
Daniel Dunbar6bff2512009-08-03 17:06:42 +00005748 Entry =
5749 new llvm::GlobalVariable(CGM.getModule(), ObjCTypes.ClassnfABIPtrTy,
Owen Anderson1c431b32009-07-08 19:05:04 +00005750 false, llvm::GlobalValue::InternalLinkage,
Daniel Dunbar6bff2512009-08-03 17:06:42 +00005751 ClassGV,
Owen Anderson1c431b32009-07-08 19:05:04 +00005752 "\01L_OBJC_CLASSLIST_SUP_REFS_$_");
Daniel Dunbar11394522009-04-18 08:51:00 +00005753 Entry->setAlignment(
Daniel Dunbar4b429ae2010-04-25 20:39:32 +00005754 CGM.getTargetData().getABITypeAlignment(
Daniel Dunbar6bff2512009-08-03 17:06:42 +00005755 ObjCTypes.ClassnfABIPtrTy));
Daniel Dunbar11394522009-04-18 08:51:00 +00005756 Entry->setSection("__DATA, __objc_superrefs, regular, no_dead_strip");
Chris Lattnerad64e022009-07-17 23:57:13 +00005757 CGM.AddUsedGlobal(Entry);
Fariborz Jahanian0e81f4b2009-02-05 20:41:40 +00005758 }
Daniel Dunbar6bff2512009-08-03 17:06:42 +00005759
Daniel Dunbar2da84ff2009-11-29 21:23:36 +00005760 return Builder.CreateLoad(Entry, "tmp");
Fariborz Jahanian0e81f4b2009-02-05 20:41:40 +00005761}
5762
Fariborz Jahanian7a06aae2009-02-06 20:09:23 +00005763/// EmitMetaClassRef - Return a Value * of the address of _class_t
5764/// meta-data
5765///
Daniel Dunbar6bff2512009-08-03 17:06:42 +00005766llvm::Value *CGObjCNonFragileABIMac::EmitMetaClassRef(CGBuilderTy &Builder,
5767 const ObjCInterfaceDecl *ID) {
Fariborz Jahanian7a06aae2009-02-06 20:09:23 +00005768 llvm::GlobalVariable * &Entry = MetaClassReferences[ID->getIdentifier()];
5769 if (Entry)
Daniel Dunbar2da84ff2009-11-29 21:23:36 +00005770 return Builder.CreateLoad(Entry, "tmp");
Daniel Dunbar6bff2512009-08-03 17:06:42 +00005771
Daniel Dunbar6ab187a2009-04-07 05:48:37 +00005772 std::string MetaClassName(getMetaclassSymbolPrefix() + ID->getNameAsString());
Fariborz Jahanian0f902942009-04-14 18:41:56 +00005773 llvm::GlobalVariable *MetaClassGV = GetClassGlobal(MetaClassName);
Daniel Dunbar6bff2512009-08-03 17:06:42 +00005774 Entry =
Owen Anderson1c431b32009-07-08 19:05:04 +00005775 new llvm::GlobalVariable(CGM.getModule(), ObjCTypes.ClassnfABIPtrTy, false,
Fariborz Jahanian7a06aae2009-02-06 20:09:23 +00005776 llvm::GlobalValue::InternalLinkage,
Daniel Dunbar6bff2512009-08-03 17:06:42 +00005777 MetaClassGV,
Owen Anderson1c431b32009-07-08 19:05:04 +00005778 "\01L_OBJC_CLASSLIST_SUP_REFS_$_");
Fariborz Jahanian7a06aae2009-02-06 20:09:23 +00005779 Entry->setAlignment(
Daniel Dunbar4b429ae2010-04-25 20:39:32 +00005780 CGM.getTargetData().getABITypeAlignment(
Daniel Dunbar6bff2512009-08-03 17:06:42 +00005781 ObjCTypes.ClassnfABIPtrTy));
5782
Daniel Dunbar33af70f2009-04-15 19:03:14 +00005783 Entry->setSection("__DATA, __objc_superrefs, regular, no_dead_strip");
Chris Lattnerad64e022009-07-17 23:57:13 +00005784 CGM.AddUsedGlobal(Entry);
Daniel Dunbar6bff2512009-08-03 17:06:42 +00005785
Daniel Dunbar2da84ff2009-11-29 21:23:36 +00005786 return Builder.CreateLoad(Entry, "tmp");
Fariborz Jahanian7a06aae2009-02-06 20:09:23 +00005787}
5788
Fariborz Jahanian0e81f4b2009-02-05 20:41:40 +00005789/// GetClass - Return a reference to the class for the given interface
5790/// decl.
5791llvm::Value *CGObjCNonFragileABIMac::GetClass(CGBuilderTy &Builder,
5792 const ObjCInterfaceDecl *ID) {
Fariborz Jahanian269f8bc2009-11-17 22:42:00 +00005793 if (ID->hasAttr<WeakImportAttr>()) {
5794 std::string ClassName(getClassSymbolPrefix() + ID->getNameAsString());
5795 llvm::GlobalVariable *ClassGV = GetClassGlobal(ClassName);
5796 ClassGV->setLinkage(llvm::GlobalValue::ExternalWeakLinkage);
5797 }
5798
Fariborz Jahanian0e81f4b2009-02-05 20:41:40 +00005799 return EmitClassRef(Builder, ID);
5800}
5801
Fariborz Jahanian7a06aae2009-02-06 20:09:23 +00005802/// Generates a message send where the super is the receiver. This is
5803/// a message send to self with special delivery semantics indicating
5804/// which class's method should be called.
5805CodeGen::RValue
5806CGObjCNonFragileABIMac::GenerateMessageSendSuper(CodeGen::CodeGenFunction &CGF,
John McCallef072fd2010-05-22 01:48:05 +00005807 ReturnValueSlot Return,
Daniel Dunbar6bff2512009-08-03 17:06:42 +00005808 QualType ResultType,
5809 Selector Sel,
5810 const ObjCInterfaceDecl *Class,
5811 bool isCategoryImpl,
5812 llvm::Value *Receiver,
5813 bool IsClassMessage,
Daniel Dunbard6c93d72009-09-17 04:01:22 +00005814 const CodeGen::CallArgList &CallArgs,
5815 const ObjCMethodDecl *Method) {
Fariborz Jahanian7a06aae2009-02-06 20:09:23 +00005816 // ...
5817 // Create and init a super structure; this is a (receiver, class)
5818 // pair we will pass to objc_msgSendSuper.
5819 llvm::Value *ObjCSuper =
5820 CGF.Builder.CreateAlloca(ObjCTypes.SuperTy, 0, "objc_super");
Daniel Dunbar6bff2512009-08-03 17:06:42 +00005821
Fariborz Jahanian7a06aae2009-02-06 20:09:23 +00005822 llvm::Value *ReceiverAsObject =
5823 CGF.Builder.CreateBitCast(Receiver, ObjCTypes.ObjectPtrTy);
5824 CGF.Builder.CreateStore(ReceiverAsObject,
5825 CGF.Builder.CreateStructGEP(ObjCSuper, 0));
Daniel Dunbar6bff2512009-08-03 17:06:42 +00005826
Fariborz Jahanian7a06aae2009-02-06 20:09:23 +00005827 // If this is a class message the metaclass is passed as the target.
Fariborz Jahanian7ce77922009-02-28 20:07:56 +00005828 llvm::Value *Target;
5829 if (IsClassMessage) {
5830 if (isCategoryImpl) {
5831 // Message sent to "super' in a class method defined in
5832 // a category implementation.
Daniel Dunbar11394522009-04-18 08:51:00 +00005833 Target = EmitClassRef(CGF.Builder, Class);
Fariborz Jahanian7ce77922009-02-28 20:07:56 +00005834 Target = CGF.Builder.CreateStructGEP(Target, 0);
5835 Target = CGF.Builder.CreateLoad(Target);
Mike Stumpb3589f42009-07-30 22:28:39 +00005836 } else
Fariborz Jahanian7ce77922009-02-28 20:07:56 +00005837 Target = EmitMetaClassRef(CGF.Builder, Class);
Mike Stumpb3589f42009-07-30 22:28:39 +00005838 } else
Daniel Dunbar11394522009-04-18 08:51:00 +00005839 Target = EmitSuperClassRef(CGF.Builder, Class);
Daniel Dunbar6bff2512009-08-03 17:06:42 +00005840
Mike Stumpf5408fe2009-05-16 07:57:57 +00005841 // FIXME: We shouldn't need to do this cast, rectify the ASTContext and
5842 // ObjCTypes types.
Fariborz Jahanian7a06aae2009-02-06 20:09:23 +00005843 const llvm::Type *ClassTy =
5844 CGM.getTypes().ConvertType(CGF.getContext().getObjCClassType());
5845 Target = CGF.Builder.CreateBitCast(Target, ClassTy);
5846 CGF.Builder.CreateStore(Target,
5847 CGF.Builder.CreateStructGEP(ObjCSuper, 1));
Daniel Dunbar6bff2512009-08-03 17:06:42 +00005848
Fariborz Jahaniand0f8a8d2009-05-11 19:25:47 +00005849 return (LegacyDispatchedSelector(Sel))
John McCallef072fd2010-05-22 01:48:05 +00005850 ? EmitLegacyMessageSend(CGF, Return, ResultType,
5851 EmitSelector(CGF.Builder, Sel),
Daniel Dunbar6bff2512009-08-03 17:06:42 +00005852 ObjCSuper, ObjCTypes.SuperPtrCTy,
Daniel Dunbard6c93d72009-09-17 04:01:22 +00005853 true, CallArgs, Method, ObjCTypes)
John McCallef072fd2010-05-22 01:48:05 +00005854 : EmitMessageSend(CGF, Return, ResultType, Sel,
Daniel Dunbar6bff2512009-08-03 17:06:42 +00005855 ObjCSuper, ObjCTypes.SuperPtrCTy,
5856 true, CallArgs);
Fariborz Jahanian7a06aae2009-02-06 20:09:23 +00005857}
Fariborz Jahanian26cc89f2009-02-11 20:51:17 +00005858
Daniel Dunbar6bff2512009-08-03 17:06:42 +00005859llvm::Value *CGObjCNonFragileABIMac::EmitSelector(CGBuilderTy &Builder,
Fariborz Jahanian03b29602010-06-17 19:56:20 +00005860 Selector Sel, bool lval) {
Fariborz Jahanian26cc89f2009-02-11 20:51:17 +00005861 llvm::GlobalVariable *&Entry = SelectorReferences[Sel];
Daniel Dunbar6bff2512009-08-03 17:06:42 +00005862
Fariborz Jahanian26cc89f2009-02-11 20:51:17 +00005863 if (!Entry) {
Daniel Dunbar6bff2512009-08-03 17:06:42 +00005864 llvm::Constant *Casted =
5865 llvm::ConstantExpr::getBitCast(GetMethodVarName(Sel),
5866 ObjCTypes.SelectorPtrTy);
5867 Entry =
5868 new llvm::GlobalVariable(CGM.getModule(), ObjCTypes.SelectorPtrTy, false,
5869 llvm::GlobalValue::InternalLinkage,
5870 Casted, "\01L_OBJC_SELECTOR_REFERENCES_");
Fariborz Jahaniand0f8a8d2009-05-11 19:25:47 +00005871 Entry->setSection("__DATA, __objc_selrefs, literal_pointers, no_dead_strip");
Chris Lattnerad64e022009-07-17 23:57:13 +00005872 CGM.AddUsedGlobal(Entry);
Fariborz Jahanian26cc89f2009-02-11 20:51:17 +00005873 }
Daniel Dunbar6bff2512009-08-03 17:06:42 +00005874
Fariborz Jahanian03b29602010-06-17 19:56:20 +00005875 if (lval)
5876 return Entry;
Daniel Dunbar2da84ff2009-11-29 21:23:36 +00005877 return Builder.CreateLoad(Entry, "tmp");
Fariborz Jahanian26cc89f2009-02-11 20:51:17 +00005878}
Fariborz Jahanian6948aea2009-02-16 22:52:32 +00005879/// EmitObjCIvarAssign - Code gen for assigning to a __strong object.
Fariborz Jahanian6c7a1f32009-09-24 22:25:38 +00005880/// objc_assign_ivar (id src, id *dst, ptrdiff_t)
Fariborz Jahanian6948aea2009-02-16 22:52:32 +00005881///
5882void CGObjCNonFragileABIMac::EmitObjCIvarAssign(CodeGen::CodeGenFunction &CGF,
Mike Stump1eb44332009-09-09 15:08:12 +00005883 llvm::Value *src,
Fariborz Jahanian6c7a1f32009-09-24 22:25:38 +00005884 llvm::Value *dst,
5885 llvm::Value *ivarOffset) {
Fariborz Jahanian0a855d02009-03-23 19:10:40 +00005886 const llvm::Type * SrcTy = src->getType();
5887 if (!isa<llvm::PointerType>(SrcTy)) {
Duncan Sands9408c452009-05-09 07:08:47 +00005888 unsigned Size = CGM.getTargetData().getTypeAllocSize(SrcTy);
Fariborz Jahanian0a855d02009-03-23 19:10:40 +00005889 assert(Size <= 8 && "does not support size > 8");
5890 src = (Size == 4 ? CGF.Builder.CreateBitCast(src, ObjCTypes.IntTy)
5891 : CGF.Builder.CreateBitCast(src, ObjCTypes.LongTy));
Fariborz Jahanian3b8a6522009-03-13 00:42:52 +00005892 src = CGF.Builder.CreateIntToPtr(src, ObjCTypes.Int8PtrTy);
5893 }
Fariborz Jahanian6948aea2009-02-16 22:52:32 +00005894 src = CGF.Builder.CreateBitCast(src, ObjCTypes.ObjectPtrTy);
5895 dst = CGF.Builder.CreateBitCast(dst, ObjCTypes.PtrObjectPtrTy);
Fariborz Jahanian6c7a1f32009-09-24 22:25:38 +00005896 CGF.Builder.CreateCall3(ObjCTypes.getGcAssignIvarFn(),
5897 src, dst, ivarOffset);
Fariborz Jahanian6948aea2009-02-16 22:52:32 +00005898 return;
5899}
5900
5901/// EmitObjCStrongCastAssign - Code gen for assigning to a __strong cast object.
5902/// objc_assign_strongCast (id src, id *dst)
5903///
5904void CGObjCNonFragileABIMac::EmitObjCStrongCastAssign(
Daniel Dunbar6bff2512009-08-03 17:06:42 +00005905 CodeGen::CodeGenFunction &CGF,
Mike Stump1eb44332009-09-09 15:08:12 +00005906 llvm::Value *src, llvm::Value *dst) {
Fariborz Jahanian0a855d02009-03-23 19:10:40 +00005907 const llvm::Type * SrcTy = src->getType();
5908 if (!isa<llvm::PointerType>(SrcTy)) {
Duncan Sands9408c452009-05-09 07:08:47 +00005909 unsigned Size = CGM.getTargetData().getTypeAllocSize(SrcTy);
Fariborz Jahanian0a855d02009-03-23 19:10:40 +00005910 assert(Size <= 8 && "does not support size > 8");
5911 src = (Size == 4 ? CGF.Builder.CreateBitCast(src, ObjCTypes.IntTy)
Daniel Dunbar6bff2512009-08-03 17:06:42 +00005912 : CGF.Builder.CreateBitCast(src, ObjCTypes.LongTy));
Fariborz Jahanian3b8a6522009-03-13 00:42:52 +00005913 src = CGF.Builder.CreateIntToPtr(src, ObjCTypes.Int8PtrTy);
5914 }
Fariborz Jahanian6948aea2009-02-16 22:52:32 +00005915 src = CGF.Builder.CreateBitCast(src, ObjCTypes.ObjectPtrTy);
5916 dst = CGF.Builder.CreateBitCast(dst, ObjCTypes.PtrObjectPtrTy);
Chris Lattnerbbccd612009-04-22 02:38:11 +00005917 CGF.Builder.CreateCall2(ObjCTypes.getGcAssignStrongCastFn(),
Fariborz Jahanian6948aea2009-02-16 22:52:32 +00005918 src, dst, "weakassign");
5919 return;
5920}
5921
Fariborz Jahanian082b02e2009-07-08 01:18:33 +00005922void CGObjCNonFragileABIMac::EmitGCMemmoveCollectable(
Daniel Dunbar6bff2512009-08-03 17:06:42 +00005923 CodeGen::CodeGenFunction &CGF,
5924 llvm::Value *DestPtr,
5925 llvm::Value *SrcPtr,
Fariborz Jahanian55bcace2010-06-15 22:44:06 +00005926 llvm::Value *Size) {
Fariborz Jahanian082b02e2009-07-08 01:18:33 +00005927 SrcPtr = CGF.Builder.CreateBitCast(SrcPtr, ObjCTypes.Int8PtrTy);
5928 DestPtr = CGF.Builder.CreateBitCast(DestPtr, ObjCTypes.Int8PtrTy);
Fariborz Jahanian082b02e2009-07-08 01:18:33 +00005929 CGF.Builder.CreateCall3(ObjCTypes.GcMemmoveCollectableFn(),
Fariborz Jahanian55bcace2010-06-15 22:44:06 +00005930 DestPtr, SrcPtr, Size);
Fariborz Jahanian082b02e2009-07-08 01:18:33 +00005931 return;
5932}
5933
Fariborz Jahanian6948aea2009-02-16 22:52:32 +00005934/// EmitObjCWeakRead - Code gen for loading value of a __weak
5935/// object: objc_read_weak (id *src)
5936///
5937llvm::Value * CGObjCNonFragileABIMac::EmitObjCWeakRead(
Daniel Dunbar6bff2512009-08-03 17:06:42 +00005938 CodeGen::CodeGenFunction &CGF,
Mike Stump1eb44332009-09-09 15:08:12 +00005939 llvm::Value *AddrWeakObj) {
Eli Friedman8339b352009-03-07 03:57:15 +00005940 const llvm::Type* DestTy =
Daniel Dunbar6bff2512009-08-03 17:06:42 +00005941 cast<llvm::PointerType>(AddrWeakObj->getType())->getElementType();
5942 AddrWeakObj = CGF.Builder.CreateBitCast(AddrWeakObj, ObjCTypes.PtrObjectPtrTy);
Chris Lattner72db6c32009-04-22 02:44:54 +00005943 llvm::Value *read_weak = CGF.Builder.CreateCall(ObjCTypes.getGcReadWeakFn(),
Fariborz Jahanian6948aea2009-02-16 22:52:32 +00005944 AddrWeakObj, "weakread");
Eli Friedman8339b352009-03-07 03:57:15 +00005945 read_weak = CGF.Builder.CreateBitCast(read_weak, DestTy);
Fariborz Jahanian6948aea2009-02-16 22:52:32 +00005946 return read_weak;
5947}
5948
5949/// EmitObjCWeakAssign - Code gen for assigning to a __weak object.
5950/// objc_assign_weak (id src, id *dst)
5951///
5952void CGObjCNonFragileABIMac::EmitObjCWeakAssign(CodeGen::CodeGenFunction &CGF,
Mike Stump1eb44332009-09-09 15:08:12 +00005953 llvm::Value *src, llvm::Value *dst) {
Fariborz Jahanian0a855d02009-03-23 19:10:40 +00005954 const llvm::Type * SrcTy = src->getType();
5955 if (!isa<llvm::PointerType>(SrcTy)) {
Duncan Sands9408c452009-05-09 07:08:47 +00005956 unsigned Size = CGM.getTargetData().getTypeAllocSize(SrcTy);
Fariborz Jahanian0a855d02009-03-23 19:10:40 +00005957 assert(Size <= 8 && "does not support size > 8");
5958 src = (Size == 4 ? CGF.Builder.CreateBitCast(src, ObjCTypes.IntTy)
5959 : CGF.Builder.CreateBitCast(src, ObjCTypes.LongTy));
Fariborz Jahanian3b8a6522009-03-13 00:42:52 +00005960 src = CGF.Builder.CreateIntToPtr(src, ObjCTypes.Int8PtrTy);
5961 }
Fariborz Jahanian6948aea2009-02-16 22:52:32 +00005962 src = CGF.Builder.CreateBitCast(src, ObjCTypes.ObjectPtrTy);
5963 dst = CGF.Builder.CreateBitCast(dst, ObjCTypes.PtrObjectPtrTy);
Chris Lattner96508e12009-04-17 22:12:36 +00005964 CGF.Builder.CreateCall2(ObjCTypes.getGcAssignWeakFn(),
Fariborz Jahanian6948aea2009-02-16 22:52:32 +00005965 src, dst, "weakassign");
5966 return;
5967}
5968
5969/// EmitObjCGlobalAssign - Code gen for assigning to a __strong object.
5970/// objc_assign_global (id src, id *dst)
5971///
5972void CGObjCNonFragileABIMac::EmitObjCGlobalAssign(CodeGen::CodeGenFunction &CGF,
Fariborz Jahanian021a7a62010-07-20 20:30:03 +00005973 llvm::Value *src, llvm::Value *dst,
5974 bool threadlocal) {
Fariborz Jahanian0a855d02009-03-23 19:10:40 +00005975 const llvm::Type * SrcTy = src->getType();
5976 if (!isa<llvm::PointerType>(SrcTy)) {
Duncan Sands9408c452009-05-09 07:08:47 +00005977 unsigned Size = CGM.getTargetData().getTypeAllocSize(SrcTy);
Fariborz Jahanian0a855d02009-03-23 19:10:40 +00005978 assert(Size <= 8 && "does not support size > 8");
5979 src = (Size == 4 ? CGF.Builder.CreateBitCast(src, ObjCTypes.IntTy)
5980 : CGF.Builder.CreateBitCast(src, ObjCTypes.LongTy));
Fariborz Jahanian3b8a6522009-03-13 00:42:52 +00005981 src = CGF.Builder.CreateIntToPtr(src, ObjCTypes.Int8PtrTy);
5982 }
Fariborz Jahanian6948aea2009-02-16 22:52:32 +00005983 src = CGF.Builder.CreateBitCast(src, ObjCTypes.ObjectPtrTy);
5984 dst = CGF.Builder.CreateBitCast(dst, ObjCTypes.PtrObjectPtrTy);
Fariborz Jahanian021a7a62010-07-20 20:30:03 +00005985 if (!threadlocal)
5986 CGF.Builder.CreateCall2(ObjCTypes.getGcAssignGlobalFn(),
5987 src, dst, "globalassign");
5988 else
5989 CGF.Builder.CreateCall2(ObjCTypes.getGcAssignThreadLocalFn(),
5990 src, dst, "threadlocalassign");
Fariborz Jahanian6948aea2009-02-16 22:52:32 +00005991 return;
5992}
Fariborz Jahanian26cc89f2009-02-11 20:51:17 +00005993
John McCall740e8072010-07-21 00:41:47 +00005994namespace {
John McCall1f0fca52010-07-21 07:22:38 +00005995 struct CallSyncExit : EHScopeStack::Cleanup {
John McCall740e8072010-07-21 00:41:47 +00005996 llvm::Value *SyncExitFn;
5997 llvm::Value *SyncArg;
5998 CallSyncExit(llvm::Value *SyncExitFn, llvm::Value *SyncArg)
5999 : SyncExitFn(SyncExitFn), SyncArg(SyncArg) {}
6000
6001 void Emit(CodeGenFunction &CGF, bool IsForEHCleanup) {
6002 CGF.Builder.CreateCall(SyncExitFn, SyncArg)->setDoesNotThrow();
6003 }
6004 };
6005}
6006
Daniel Dunbar6bff2512009-08-03 17:06:42 +00006007void
John McCallf1549f62010-07-06 01:34:17 +00006008CGObjCNonFragileABIMac::EmitSynchronizedStmt(CodeGen::CodeGenFunction &CGF,
6009 const ObjCAtSynchronizedStmt &S) {
6010 // Evaluate the lock operand. This should dominate the cleanup.
6011 llvm::Value *SyncArg = CGF.EmitScalarExpr(S.getSynchExpr());
Daniel Dunbar8ecbaf22009-02-24 07:47:38 +00006012
John McCallf1549f62010-07-06 01:34:17 +00006013 // Acquire the lock.
6014 SyncArg = CGF.Builder.CreateBitCast(SyncArg, ObjCTypes.ObjectPtrTy);
6015 CGF.Builder.CreateCall(ObjCTypes.getSyncEnterFn(), SyncArg)
6016 ->setDoesNotThrow();
6017
6018 // Register an all-paths cleanup to release the lock.
John McCall1f0fca52010-07-21 07:22:38 +00006019 CGF.EHStack.pushCleanup<CallSyncExit>(NormalAndEHCleanup,
6020 ObjCTypes.getSyncExitFn(),
6021 SyncArg);
Daniel Dunbar8ecbaf22009-02-24 07:47:38 +00006022
John McCallf1549f62010-07-06 01:34:17 +00006023 // Emit the body of the statement.
6024 CGF.EmitStmt(S.getSynchBody());
Daniel Dunbar8ecbaf22009-02-24 07:47:38 +00006025
John McCallf1549f62010-07-06 01:34:17 +00006026 // Pop the lock-release cleanup.
6027 CGF.PopCleanupBlock();
6028}
Daniel Dunbar8ecbaf22009-02-24 07:47:38 +00006029
John McCallf1549f62010-07-06 01:34:17 +00006030namespace {
6031 struct CatchHandler {
6032 const VarDecl *Variable;
6033 const Stmt *Body;
6034 llvm::BasicBlock *Block;
6035 llvm::Value *TypeInfo;
6036 };
John McCall8e3f8612010-07-13 22:12:14 +00006037
John McCall1f0fca52010-07-21 07:22:38 +00006038 struct CallObjCEndCatch : EHScopeStack::Cleanup {
John McCall8e3f8612010-07-13 22:12:14 +00006039 CallObjCEndCatch(bool MightThrow, llvm::Value *Fn) :
6040 MightThrow(MightThrow), Fn(Fn) {}
6041 bool MightThrow;
6042 llvm::Value *Fn;
6043
6044 void Emit(CodeGenFunction &CGF, bool IsForEH) {
6045 if (!MightThrow) {
6046 CGF.Builder.CreateCall(Fn)->setDoesNotThrow();
6047 return;
6048 }
6049
6050 CGF.EmitCallOrInvoke(Fn, 0, 0);
6051 }
6052 };
John McCallf1549f62010-07-06 01:34:17 +00006053}
Daniel Dunbar6bff2512009-08-03 17:06:42 +00006054
John McCall5a180392010-07-24 00:37:23 +00006055llvm::Constant *
6056CGObjCNonFragileABIMac::GetEHType(QualType T) {
6057 // There's a particular fixed type info for 'id'.
6058 if (T->isObjCIdType() ||
6059 T->isObjCQualifiedIdType()) {
6060 llvm::Constant *IDEHType =
6061 CGM.getModule().getGlobalVariable("OBJC_EHTYPE_id");
6062 if (!IDEHType)
6063 IDEHType =
6064 new llvm::GlobalVariable(CGM.getModule(), ObjCTypes.EHTypeTy,
6065 false,
6066 llvm::GlobalValue::ExternalLinkage,
6067 0, "OBJC_EHTYPE_id");
6068 return IDEHType;
6069 }
6070
6071 // All other types should be Objective-C interface pointer types.
6072 const ObjCObjectPointerType *PT =
6073 T->getAs<ObjCObjectPointerType>();
6074 assert(PT && "Invalid @catch type.");
6075 const ObjCInterfaceType *IT = PT->getInterfaceType();
6076 assert(IT && "Invalid @catch type.");
6077 return GetInterfaceEHType(IT->getDecl(), false);
6078}
6079
John McCallf1549f62010-07-06 01:34:17 +00006080void CGObjCNonFragileABIMac::EmitTryStmt(CodeGen::CodeGenFunction &CGF,
6081 const ObjCAtTryStmt &S) {
6082 // Jump destination for falling out of catch bodies.
6083 CodeGenFunction::JumpDest Cont;
6084 if (S.getNumCatchStmts())
6085 Cont = CGF.getJumpDestInCurrentScope("eh.cont");
Daniel Dunbar4ff36842009-03-02 06:08:11 +00006086
John McCallf1549f62010-07-06 01:34:17 +00006087 CodeGenFunction::FinallyInfo FinallyInfo;
6088 if (const ObjCAtFinallyStmt *Finally = S.getFinallyStmt())
6089 FinallyInfo = CGF.EnterFinallyBlock(Finally->getFinallyBody(),
6090 ObjCTypes.getObjCBeginCatchFn(),
6091 ObjCTypes.getObjCEndCatchFn(),
6092 ObjCTypes.getExceptionRethrowFn());
Daniel Dunbar4ff36842009-03-02 06:08:11 +00006093
John McCallf1549f62010-07-06 01:34:17 +00006094 llvm::SmallVector<CatchHandler, 8> Handlers;
Daniel Dunbar4ff36842009-03-02 06:08:11 +00006095
John McCallf1549f62010-07-06 01:34:17 +00006096 // Enter the catch, if there is one.
6097 if (S.getNumCatchStmts()) {
6098 for (unsigned I = 0, N = S.getNumCatchStmts(); I != N; ++I) {
6099 const ObjCAtCatchStmt *CatchStmt = S.getCatchStmt(I);
Douglas Gregorc00d8e12010-04-26 16:46:50 +00006100 const VarDecl *CatchDecl = CatchStmt->getCatchParamDecl();
Daniel Dunbar4ff36842009-03-02 06:08:11 +00006101
John McCallf1549f62010-07-06 01:34:17 +00006102 Handlers.push_back(CatchHandler());
6103 CatchHandler &Handler = Handlers.back();
6104 Handler.Variable = CatchDecl;
6105 Handler.Body = CatchStmt->getCatchBody();
6106 Handler.Block = CGF.createBasicBlock("catch");
6107
6108 // @catch(...) always matches.
Douglas Gregor8f5e3dd2010-04-23 22:50:49 +00006109 if (!CatchDecl) {
John McCallf1549f62010-07-06 01:34:17 +00006110 Handler.TypeInfo = 0; // catch-all
6111 // Don't consider any other catches.
Douglas Gregor8f5e3dd2010-04-23 22:50:49 +00006112 break;
6113 }
Daniel Dunbar4ff36842009-03-02 06:08:11 +00006114
John McCall5a180392010-07-24 00:37:23 +00006115 Handler.TypeInfo = GetEHType(CatchDecl->getType());
Daniel Dunbar4ff36842009-03-02 06:08:11 +00006116 }
John McCallf1549f62010-07-06 01:34:17 +00006117
6118 EHCatchScope *Catch = CGF.EHStack.pushCatch(Handlers.size());
6119 for (unsigned I = 0, E = Handlers.size(); I != E; ++I)
6120 Catch->setHandler(I, Handlers[I].TypeInfo, Handlers[I].Block);
Daniel Dunbar4ff36842009-03-02 06:08:11 +00006121 }
John McCallf1549f62010-07-06 01:34:17 +00006122
6123 // Emit the try body.
6124 CGF.EmitStmt(S.getTryBody());
Daniel Dunbar4ff36842009-03-02 06:08:11 +00006125
John McCallf1549f62010-07-06 01:34:17 +00006126 // Leave the try.
6127 if (S.getNumCatchStmts())
6128 CGF.EHStack.popCatch();
Daniel Dunbar6bff2512009-08-03 17:06:42 +00006129
John McCallf1549f62010-07-06 01:34:17 +00006130 // Remember where we were.
6131 CGBuilderTy::InsertPoint SavedIP = CGF.Builder.saveAndClearIP();
Daniel Dunbar4ff36842009-03-02 06:08:11 +00006132
John McCallf1549f62010-07-06 01:34:17 +00006133 // Emit the handlers.
6134 for (unsigned I = 0, E = Handlers.size(); I != E; ++I) {
6135 CatchHandler &Handler = Handlers[I];
Daniel Dunbar4ff36842009-03-02 06:08:11 +00006136
John McCallf1549f62010-07-06 01:34:17 +00006137 CGF.EmitBlock(Handler.Block);
6138 llvm::Value *RawExn = CGF.Builder.CreateLoad(CGF.getExceptionSlot());
Daniel Dunbar4ff36842009-03-02 06:08:11 +00006139
John McCallf1549f62010-07-06 01:34:17 +00006140 // Enter the catch.
6141 llvm::CallInst *Exn =
6142 CGF.Builder.CreateCall(ObjCTypes.getObjCBeginCatchFn(), RawExn,
6143 "exn.adjusted");
6144 Exn->setDoesNotThrow();
Daniel Dunbar4ff36842009-03-02 06:08:11 +00006145
John McCallf1549f62010-07-06 01:34:17 +00006146 // Add a cleanup to leave the catch.
John McCall8e3f8612010-07-13 22:12:14 +00006147 bool EndCatchMightThrow = (Handler.Variable == 0);
John McCall1f0fca52010-07-21 07:22:38 +00006148 CGF.EHStack.pushCleanup<CallObjCEndCatch>(NormalAndEHCleanup,
6149 EndCatchMightThrow,
6150 ObjCTypes.getObjCEndCatchFn());
Daniel Dunbar6bff2512009-08-03 17:06:42 +00006151
John McCallf1549f62010-07-06 01:34:17 +00006152 // Bind the catch parameter if it exists.
6153 if (const VarDecl *CatchParam = Handler.Variable) {
6154 const llvm::Type *CatchType = CGF.ConvertType(CatchParam->getType());
6155 llvm::Value *CastExn = CGF.Builder.CreateBitCast(Exn, CatchType);
Daniel Dunbar4ff36842009-03-02 06:08:11 +00006156
John McCallf1549f62010-07-06 01:34:17 +00006157 CGF.EmitLocalBlockVarDecl(*CatchParam);
6158 CGF.Builder.CreateStore(CastExn, CGF.GetAddrOfLocalVar(CatchParam));
Daniel Dunbar4ff36842009-03-02 06:08:11 +00006159 }
Daniel Dunbar4ff36842009-03-02 06:08:11 +00006160
John McCallf1549f62010-07-06 01:34:17 +00006161 CGF.ObjCEHValueStack.push_back(Exn);
6162 CGF.EmitStmt(Handler.Body);
6163 CGF.ObjCEHValueStack.pop_back();
Daniel Dunbar8ecbaf22009-02-24 07:47:38 +00006164
John McCallf1549f62010-07-06 01:34:17 +00006165 // Leave the earlier cleanup.
6166 CGF.PopCleanupBlock();
Daniel Dunbar8ecbaf22009-02-24 07:47:38 +00006167
John McCallf1549f62010-07-06 01:34:17 +00006168 CGF.EmitBranchThroughCleanup(Cont);
6169 }
Daniel Dunbar8ecbaf22009-02-24 07:47:38 +00006170
John McCallf1549f62010-07-06 01:34:17 +00006171 // Go back to the try-statement fallthrough.
6172 CGF.Builder.restoreIP(SavedIP);
Daniel Dunbar8ecbaf22009-02-24 07:47:38 +00006173
John McCallf1549f62010-07-06 01:34:17 +00006174 // Pop out of the normal cleanup on the finally.
6175 if (S.getFinallyStmt())
6176 CGF.ExitFinallyBlock(FinallyInfo);
Daniel Dunbar8ecbaf22009-02-24 07:47:38 +00006177
John McCallff8e1152010-07-23 21:56:41 +00006178 if (Cont.isValid())
6179 CGF.EmitBlock(Cont.getBlock());
Daniel Dunbar8ecbaf22009-02-24 07:47:38 +00006180}
6181
Anders Carlssonf57c5b22009-02-16 22:59:18 +00006182/// EmitThrowStmt - Generate code for a throw statement.
6183void CGObjCNonFragileABIMac::EmitThrowStmt(CodeGen::CodeGenFunction &CGF,
6184 const ObjCAtThrowStmt &S) {
Daniel Dunbar6bff2512009-08-03 17:06:42 +00006185 llvm::Value *Exception;
Fariborz Jahanian69677ea2010-05-28 17:34:43 +00006186 llvm::Constant *FunctionThrowOrRethrow;
Anders Carlssonf57c5b22009-02-16 22:59:18 +00006187 if (const Expr *ThrowExpr = S.getThrowExpr()) {
Daniel Dunbar4ff36842009-03-02 06:08:11 +00006188 Exception = CGF.EmitScalarExpr(ThrowExpr);
Fariborz Jahanian69677ea2010-05-28 17:34:43 +00006189 FunctionThrowOrRethrow = ObjCTypes.getExceptionThrowFn();
Anders Carlssonf57c5b22009-02-16 22:59:18 +00006190 } else {
Daniel Dunbar6bff2512009-08-03 17:06:42 +00006191 assert((!CGF.ObjCEHValueStack.empty() && CGF.ObjCEHValueStack.back()) &&
Daniel Dunbar4ff36842009-03-02 06:08:11 +00006192 "Unexpected rethrow outside @catch block.");
6193 Exception = CGF.ObjCEHValueStack.back();
Fariborz Jahanian69677ea2010-05-28 17:34:43 +00006194 FunctionThrowOrRethrow = ObjCTypes.getExceptionRethrowFn();
Anders Carlssonf57c5b22009-02-16 22:59:18 +00006195 }
6196
Daniel Dunbar6bff2512009-08-03 17:06:42 +00006197 llvm::Value *ExceptionAsObject =
Daniel Dunbar4ff36842009-03-02 06:08:11 +00006198 CGF.Builder.CreateBitCast(Exception, ObjCTypes.ObjectPtrTy, "tmp");
6199 llvm::BasicBlock *InvokeDest = CGF.getInvokeDest();
6200 if (InvokeDest) {
Fariborz Jahanian69677ea2010-05-28 17:34:43 +00006201 CGF.Builder.CreateInvoke(FunctionThrowOrRethrow,
John McCallf1549f62010-07-06 01:34:17 +00006202 CGF.getUnreachableBlock(), InvokeDest,
Daniel Dunbar4ff36842009-03-02 06:08:11 +00006203 &ExceptionAsObject, &ExceptionAsObject + 1);
John McCallf1549f62010-07-06 01:34:17 +00006204 } else {
6205 CGF.Builder.CreateCall(FunctionThrowOrRethrow, ExceptionAsObject)
6206 ->setDoesNotReturn();
6207 CGF.Builder.CreateUnreachable();
6208 }
Daniel Dunbar4ff36842009-03-02 06:08:11 +00006209
Anders Carlssonf57c5b22009-02-16 22:59:18 +00006210 // Clear the insertion point to indicate we are in unreachable code.
6211 CGF.Builder.ClearInsertionPoint();
6212}
Daniel Dunbare588b992009-03-01 04:46:24 +00006213
John McCall5a180392010-07-24 00:37:23 +00006214llvm::Constant *
Daniel Dunbar6bff2512009-08-03 17:06:42 +00006215CGObjCNonFragileABIMac::GetInterfaceEHType(const ObjCInterfaceDecl *ID,
Daniel Dunbar8158a2f2009-04-08 04:21:03 +00006216 bool ForDefinition) {
Daniel Dunbare588b992009-03-01 04:46:24 +00006217 llvm::GlobalVariable * &Entry = EHTypeReferences[ID->getIdentifier()];
Daniel Dunbare588b992009-03-01 04:46:24 +00006218
Daniel Dunbar8158a2f2009-04-08 04:21:03 +00006219 // If we don't need a definition, return the entry if found or check
6220 // if we use an external reference.
6221 if (!ForDefinition) {
6222 if (Entry)
6223 return Entry;
Daniel Dunbar7e075cb2009-04-07 06:43:45 +00006224
Daniel Dunbar8158a2f2009-04-08 04:21:03 +00006225 // If this type (or a super class) has the __objc_exception__
6226 // attribute, emit an external reference.
Douglas Gregor68584ed2009-06-18 16:11:24 +00006227 if (hasObjCExceptionAttribute(CGM.getContext(), ID))
Daniel Dunbar6bff2512009-08-03 17:06:42 +00006228 return Entry =
Owen Anderson1c431b32009-07-08 19:05:04 +00006229 new llvm::GlobalVariable(CGM.getModule(), ObjCTypes.EHTypeTy, false,
Daniel Dunbar8158a2f2009-04-08 04:21:03 +00006230 llvm::GlobalValue::ExternalLinkage,
Daniel Dunbar6bff2512009-08-03 17:06:42 +00006231 0,
Daniel Dunbar9c29bf52009-10-18 20:48:59 +00006232 ("OBJC_EHTYPE_$_" +
Daniel Dunbar01eb9b92009-10-18 21:17:35 +00006233 ID->getIdentifier()->getName()));
Daniel Dunbar8158a2f2009-04-08 04:21:03 +00006234 }
Daniel Dunbar6bff2512009-08-03 17:06:42 +00006235
Daniel Dunbar8158a2f2009-04-08 04:21:03 +00006236 // Otherwise we need to either make a new entry or fill in the
6237 // initializer.
6238 assert((!Entry || !Entry->hasInitializer()) && "Duplicate EHType definition");
Daniel Dunbar6ab187a2009-04-07 05:48:37 +00006239 std::string ClassName(getClassSymbolPrefix() + ID->getNameAsString());
Daniel Dunbare588b992009-03-01 04:46:24 +00006240 std::string VTableName = "objc_ehtype_vtable";
Daniel Dunbar6bff2512009-08-03 17:06:42 +00006241 llvm::GlobalVariable *VTableGV =
Daniel Dunbare588b992009-03-01 04:46:24 +00006242 CGM.getModule().getGlobalVariable(VTableName);
6243 if (!VTableGV)
Daniel Dunbar6bff2512009-08-03 17:06:42 +00006244 VTableGV = new llvm::GlobalVariable(CGM.getModule(), ObjCTypes.Int8PtrTy,
6245 false,
Daniel Dunbare588b992009-03-01 04:46:24 +00006246 llvm::GlobalValue::ExternalLinkage,
Owen Anderson1c431b32009-07-08 19:05:04 +00006247 0, VTableName);
Daniel Dunbare588b992009-03-01 04:46:24 +00006248
Chris Lattner77b89b82010-06-27 07:15:29 +00006249 llvm::Value *VTableIdx =
6250 llvm::ConstantInt::get(llvm::Type::getInt32Ty(VMContext), 2);
Daniel Dunbare588b992009-03-01 04:46:24 +00006251
6252 std::vector<llvm::Constant*> Values(3);
Owen Anderson3c4972d2009-07-29 18:54:39 +00006253 Values[0] = llvm::ConstantExpr::getGetElementPtr(VTableGV, &VTableIdx, 1);
Daniel Dunbare588b992009-03-01 04:46:24 +00006254 Values[1] = GetClassName(ID->getIdentifier());
Fariborz Jahanian0f902942009-04-14 18:41:56 +00006255 Values[2] = GetClassGlobal(ClassName);
Owen Andersona1cf15f2009-07-14 23:10:40 +00006256 llvm::Constant *Init =
Owen Anderson08e25242009-07-27 22:29:56 +00006257 llvm::ConstantStruct::get(ObjCTypes.EHTypeTy, Values);
Daniel Dunbare588b992009-03-01 04:46:24 +00006258
Daniel Dunbar8158a2f2009-04-08 04:21:03 +00006259 if (Entry) {
6260 Entry->setInitializer(Init);
6261 } else {
Owen Anderson1c431b32009-07-08 19:05:04 +00006262 Entry = new llvm::GlobalVariable(CGM.getModule(), ObjCTypes.EHTypeTy, false,
Daniel Dunbar8158a2f2009-04-08 04:21:03 +00006263 llvm::GlobalValue::WeakAnyLinkage,
Daniel Dunbar6bff2512009-08-03 17:06:42 +00006264 Init,
Daniel Dunbar9c29bf52009-10-18 20:48:59 +00006265 ("OBJC_EHTYPE_$_" +
Daniel Dunbar01eb9b92009-10-18 21:17:35 +00006266 ID->getIdentifier()->getName()));
Daniel Dunbar8158a2f2009-04-08 04:21:03 +00006267 }
6268
Daniel Dunbar04d40782009-04-14 06:00:08 +00006269 if (CGM.getLangOptions().getVisibilityMode() == LangOptions::Hidden)
Daniel Dunbar6ab187a2009-04-07 05:48:37 +00006270 Entry->setVisibility(llvm::GlobalValue::HiddenVisibility);
Daniel Dunbar4b429ae2010-04-25 20:39:32 +00006271 Entry->setAlignment(CGM.getTargetData().getABITypeAlignment(
6272 ObjCTypes.EHTypeTy));
Daniel Dunbar8158a2f2009-04-08 04:21:03 +00006273
6274 if (ForDefinition) {
6275 Entry->setSection("__DATA,__objc_const");
6276 Entry->setLinkage(llvm::GlobalValue::ExternalLinkage);
6277 } else {
6278 Entry->setSection("__DATA,__datacoal_nt,coalesced");
6279 }
Daniel Dunbare588b992009-03-01 04:46:24 +00006280
6281 return Entry;
6282}
Daniel Dunbar6bff2512009-08-03 17:06:42 +00006283
Daniel Dunbarbbce49b2008-08-12 00:12:39 +00006284/* *** */
6285
Daniel Dunbar6efc0c52008-08-13 03:21:16 +00006286CodeGen::CGObjCRuntime *
6287CodeGen::CreateMacObjCRuntime(CodeGen::CodeGenModule &CGM) {
Daniel Dunbarc17a4d32008-08-11 02:45:11 +00006288 return new CGObjCMac(CGM);
6289}
Fariborz Jahanianee0af742009-01-21 22:04:16 +00006290
6291CodeGen::CGObjCRuntime *
Fariborz Jahanian30bc5712009-01-22 23:02:58 +00006292CodeGen::CreateMacNonFragileABIObjCRuntime(CodeGen::CodeGenModule &CGM) {
Fariborz Jahanianaa23b572009-01-23 23:53:38 +00006293 return new CGObjCNonFragileABIMac(CGM);
Fariborz Jahanianee0af742009-01-21 22:04:16 +00006294}