blob: 78d5aba4128e75bdc5193bf2dd009914054f9521 [file] [log] [blame]
Daniel Dunbar303e2c22008-08-11 02:45:11 +00001//===----- CGObjCRuntime.h - Interface to ObjC Runtimes ---------*- C++ -*-===//
Chris Lattnerb7256cd2008-03-01 08:50:34 +00002//
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 an abstract class for Objective-C code generation. Concrete
11// subclasses of this implement code generation for specific Objective-C
12// runtime libraries.
13//
14//===----------------------------------------------------------------------===//
15
Benjamin Kramer2f5db8b2014-08-13 16:25:19 +000016#ifndef LLVM_CLANG_LIB_CODEGEN_CGOBJCRUNTIME_H
17#define LLVM_CLANG_LIB_CODEGEN_CGOBJCRUNTIME_H
Daniel Dunbarcb463852008-11-01 01:53:16 +000018#include "CGBuilder.h"
Daniel Dunbar41cf9de2008-09-09 01:06:48 +000019#include "CGCall.h"
David Chisnall93ce0182018-08-10 12:53:13 +000020#include "CGCleanup.h"
Daniel Dunbarcb463852008-11-01 01:53:16 +000021#include "CGValue.h"
Chandler Carruth3a022472012-12-04 09:13:33 +000022#include "clang/AST/DeclObjC.h"
23#include "clang/Basic/IdentifierTable.h" // Selector
Daniel Dunbar97db84c2008-08-23 03:46:30 +000024
Chris Lattnerb7256cd2008-03-01 08:50:34 +000025namespace llvm {
Chris Lattnerb7256cd2008-03-01 08:50:34 +000026 class Constant;
Daniel Dunbar9fd114d2009-04-22 07:32:20 +000027 class Function;
28 class Module;
29 class StructLayout;
Daniel Dunbar7b4dfc82009-04-22 09:39:34 +000030 class StructType;
Chris Lattnerb7256cd2008-03-01 08:50:34 +000031 class Type;
32 class Value;
Chris Lattnerb7256cd2008-03-01 08:50:34 +000033}
34
35namespace clang {
Daniel Dunbar41cf9de2008-09-09 01:06:48 +000036namespace CodeGen {
37 class CodeGenFunction;
38}
Daniel Dunbar97db84c2008-08-23 03:46:30 +000039
Daniel Dunbar9fd114d2009-04-22 07:32:20 +000040 class FieldDecl;
Anders Carlsson1963b0c2008-09-09 10:04:29 +000041 class ObjCAtTryStmt;
42 class ObjCAtThrowStmt;
Chris Lattnere132e242008-11-15 21:26:17 +000043 class ObjCAtSynchronizedStmt;
Fariborz Jahanian0196a1c2009-01-10 21:06:09 +000044 class ObjCContainerDecl;
Daniel Dunbar92992502008-08-15 22:20:32 +000045 class ObjCCategoryImplDecl;
46 class ObjCImplementationDecl;
Daniel Dunbar7c6d3a72008-08-16 00:25:02 +000047 class ObjCInterfaceDecl;
Daniel Dunbar97db84c2008-08-23 03:46:30 +000048 class ObjCMessageExpr;
Daniel Dunbar92992502008-08-15 22:20:32 +000049 class ObjCMethodDecl;
Daniel Dunbar89da6ad2008-08-13 00:59:25 +000050 class ObjCProtocolDecl;
Chris Lattner6d522c02008-06-26 04:37:12 +000051 class Selector;
Fariborz Jahanian9f84b782009-02-02 20:02:29 +000052 class ObjCIvarDecl;
Steve Naroff66afeb52009-03-31 16:53:37 +000053 class ObjCStringLiteral;
Fariborz Jahanianc05349e2010-08-04 16:57:49 +000054 class BlockDeclRefExpr;
Daniel Dunbar89da6ad2008-08-13 00:59:25 +000055
Chris Lattnerb7256cd2008-03-01 08:50:34 +000056namespace CodeGen {
Chris Lattner87ab27d2008-06-26 04:19:03 +000057 class CodeGenModule;
John McCall351762c2011-02-07 10:33:21 +000058 class CGBlockInfo;
Chris Lattnerb7256cd2008-03-01 08:50:34 +000059
Mike Stump18bb9282009-05-16 07:57:57 +000060// FIXME: Several methods should be pure virtual but aren't to avoid the
61// partially-implemented subclass breaking.
Anton Korobeynikov1200aca2008-06-01 14:13:53 +000062
63/// Implements runtime-specific code generation functions.
Chris Lattnerb7256cd2008-03-01 08:50:34 +000064class CGObjCRuntime {
Daniel Dunbardc406b82010-04-05 21:36:35 +000065protected:
John McCalla729c622012-02-17 03:33:10 +000066 CodeGen::CodeGenModule &CGM;
67 CGObjCRuntime(CodeGen::CodeGenModule &CGM) : CGM(CGM) {}
68
Daniel Dunbar9fd114d2009-04-22 07:32:20 +000069 // Utility functions for unified ivar access. These need to
70 // eventually be folded into other places (the structure layout
71 // code).
72
Daniel Dunbar7b4dfc82009-04-22 09:39:34 +000073 /// Compute an offset to the given ivar, suitable for passing to
74 /// EmitValueForIvarAtOffset. Note that the correct handling of
75 /// bit-fields is carefully coordinated by these two, use caution!
Daniel Dunbar961202372009-05-03 12:57:56 +000076 ///
77 /// The latter overload is suitable for computing the offset of a
78 /// sythesized ivar.
Eli Friedman8cbca202012-11-06 22:15:52 +000079 uint64_t ComputeIvarBaseOffset(CodeGen::CodeGenModule &CGM,
80 const ObjCInterfaceDecl *OID,
81 const ObjCIvarDecl *Ivar);
82 uint64_t ComputeIvarBaseOffset(CodeGen::CodeGenModule &CGM,
83 const ObjCImplementationDecl *OID,
84 const ObjCIvarDecl *Ivar);
Daniel Dunbar9fd114d2009-04-22 07:32:20 +000085
86 LValue EmitValueForIvarAtOffset(CodeGen::CodeGenFunction &CGF,
87 const ObjCInterfaceDecl *OID,
88 llvm::Value *BaseValue,
89 const ObjCIvarDecl *Ivar,
90 unsigned CVRQualifiers,
Mike Stump11289f42009-09-09 15:08:12 +000091 llvm::Value *Offset);
David Chisnalld3858d62011-03-25 11:57:33 +000092 /// Emits a try / catch statement. This function is intended to be called by
93 /// subclasses, and provides a generic mechanism for generating these, which
James Dennettbe302452012-06-15 22:10:14 +000094 /// should be usable by all runtimes. The caller must provide the functions
95 /// to call when entering and exiting a \@catch() block, and the function
96 /// used to rethrow exceptions. If the begin and end catch functions are
97 /// NULL, then the function assumes that the EH personality function provides
98 /// the thrown object directly.
David Chisnalld3858d62011-03-25 11:57:33 +000099 void EmitTryCatchStmt(CodeGenFunction &CGF,
100 const ObjCAtTryStmt &S,
David Chisnall3fe89562011-05-23 22:33:28 +0000101 llvm::Constant *beginCatchFn,
102 llvm::Constant *endCatchFn,
103 llvm::Constant *exceptionRethrowFn);
John McCall17f02752015-10-30 00:56:02 +0000104
105 void EmitInitOfCatchParam(CodeGenFunction &CGF, llvm::Value *exn,
106 const VarDecl *paramDecl);
107
James Dennettbe302452012-06-15 22:10:14 +0000108 /// Emits an \@synchronize() statement, using the \p syncEnterFn and
109 /// \p syncExitFn arguments as the functions called to lock and unlock
110 /// the object. This function can be called by subclasses that use
111 /// zero-cost exception handling.
David Chisnalld3858d62011-03-25 11:57:33 +0000112 void EmitAtSynchronizedStmt(CodeGenFunction &CGF,
113 const ObjCAtSynchronizedStmt &S,
114 llvm::Function *syncEnterFn,
115 llvm::Function *syncExitFn);
Daniel Dunbar48595de2008-08-11 16:50:21 +0000116
Chris Lattnerb7256cd2008-03-01 08:50:34 +0000117public:
118 virtual ~CGObjCRuntime();
Daniel Dunbar92992502008-08-15 22:20:32 +0000119
120 /// Generate the function required to register all Objective-C components in
121 /// this compilation unit with the runtime library.
122 virtual llvm::Function *ModuleInitFunction() = 0;
123
John McCall7f416cc2015-09-08 08:05:57 +0000124 /// Get a selector for the specified name and type values.
125 /// The result should have the LLVM type for ASTContext::getObjCSelType().
126 virtual llvm::Value *GetSelector(CodeGenFunction &CGF, Selector Sel) = 0;
127
128 /// Get the address of a selector for the specified name and type values.
129 /// This is a rarely-used language extension, but sadly it exists.
130 ///
131 /// The result should have the LLVM type for a pointer to
Daniel Dunbar92992502008-08-15 22:20:32 +0000132 /// ASTContext::getObjCSelType().
John McCall7f416cc2015-09-08 08:05:57 +0000133 virtual Address GetAddrOfSelector(CodeGenFunction &CGF, Selector Sel) = 0;
Daniel Dunbar92992502008-08-15 22:20:32 +0000134
Mike Stump11289f42009-09-09 15:08:12 +0000135 /// Get a typed selector.
John McCall882987f2013-02-28 19:01:20 +0000136 virtual llvm::Value *GetSelector(CodeGenFunction &CGF,
Fariborz Jahanianf3648b82009-05-05 21:36:57 +0000137 const ObjCMethodDecl *Method) = 0;
138
John McCall2ca705e2010-07-24 00:37:23 +0000139 /// Get the type constant to catch for the given ObjC pointer type.
140 /// This is used externally to implement catching ObjC types in C++.
141 /// Runtimes which don't support this should add the appropriate
142 /// error to Sema.
Fariborz Jahanian831f0fc2011-06-23 19:00:08 +0000143 virtual llvm::Constant *GetEHType(QualType T) = 0;
John McCall2ca705e2010-07-24 00:37:23 +0000144
David Chisnall93ce0182018-08-10 12:53:13 +0000145 virtual CatchTypeInfo getCatchAllTypeInfo() { return { nullptr, 0 }; }
146
Daniel Dunbar92992502008-08-15 22:20:32 +0000147 /// Generate a constant string object.
John McCall7f416cc2015-09-08 08:05:57 +0000148 virtual ConstantAddress GenerateConstantString(const StringLiteral *) = 0;
Fangrui Song6907ce22018-07-30 19:24:48 +0000149
Daniel Dunbar92992502008-08-15 22:20:32 +0000150 /// Generate a category. A category contains a list of methods (and
151 /// accompanying metadata) and a list of protocols.
152 virtual void GenerateCategory(const ObjCCategoryImplDecl *OCD) = 0;
153
Chris Lattner57540c52011-04-15 05:22:18 +0000154 /// Generate a class structure for this class.
Daniel Dunbar92992502008-08-15 22:20:32 +0000155 virtual void GenerateClass(const ObjCImplementationDecl *OID) = 0;
Mike Stump11289f42009-09-09 15:08:12 +0000156
David Chisnall92d436b2012-01-31 18:59:20 +0000157 /// Register an class alias.
158 virtual void RegisterAlias(const ObjCCompatibleAliasDecl *OAD) = 0;
159
Mike Stump11289f42009-09-09 15:08:12 +0000160 /// Generate an Objective-C message send operation.
Daniel Dunbaraff9fca2009-09-17 04:01:22 +0000161 ///
162 /// \param Method - The method being called, this may be null if synthesizing
163 /// a property setter or getter.
Mike Stump11289f42009-09-09 15:08:12 +0000164 virtual CodeGen::RValue
Daniel Dunbar97db84c2008-08-23 03:46:30 +0000165 GenerateMessageSend(CodeGen::CodeGenFunction &CGF,
John McCall78a15112010-05-22 01:48:05 +0000166 ReturnValueSlot ReturnSlot,
Daniel Dunbar4b8c6db2008-08-30 05:35:15 +0000167 QualType ResultType,
168 Selector Sel,
Daniel Dunbarca8531a2008-08-25 08:19:24 +0000169 llvm::Value *Receiver,
Fariborz Jahanianf3648b82009-05-05 21:36:57 +0000170 const CallArgList &CallArgs,
Craig Topper8a13c412014-05-21 05:09:00 +0000171 const ObjCInterfaceDecl *Class = nullptr,
172 const ObjCMethodDecl *Method = nullptr) = 0;
Daniel Dunbar89da6ad2008-08-13 00:59:25 +0000173
Daniel Dunbar92992502008-08-15 22:20:32 +0000174 /// Generate an Objective-C message send operation to the super
Daniel Dunbarca8531a2008-08-25 08:19:24 +0000175 /// class initiated in a method for Class and with the given Self
176 /// object.
Daniel Dunbaraff9fca2009-09-17 04:01:22 +0000177 ///
178 /// \param Method - The method being called, this may be null if synthesizing
179 /// a property setter or getter.
Daniel Dunbar97db84c2008-08-23 03:46:30 +0000180 virtual CodeGen::RValue
181 GenerateMessageSendSuper(CodeGen::CodeGenFunction &CGF,
John McCall78a15112010-05-22 01:48:05 +0000182 ReturnValueSlot ReturnSlot,
Daniel Dunbar4b8c6db2008-08-30 05:35:15 +0000183 QualType ResultType,
184 Selector Sel,
Daniel Dunbarca8531a2008-08-25 08:19:24 +0000185 const ObjCInterfaceDecl *Class,
Fariborz Jahanianbac73ac2009-02-28 20:07:56 +0000186 bool isCategoryImpl,
Daniel Dunbarca8531a2008-08-25 08:19:24 +0000187 llvm::Value *Self,
Daniel Dunbarc722b852008-08-30 03:02:31 +0000188 bool IsClassMessage,
Daniel Dunbaraff9fca2009-09-17 04:01:22 +0000189 const CallArgList &CallArgs,
Craig Topper8a13c412014-05-21 05:09:00 +0000190 const ObjCMethodDecl *Method = nullptr) = 0;
Daniel Dunbar6630e102008-08-12 05:08:18 +0000191
192 /// Emit the code to return the named protocol as an object, as in a
James Dennett1355bd12012-06-11 06:19:40 +0000193 /// \@protocol expression.
John McCall882987f2013-02-28 19:01:20 +0000194 virtual llvm::Value *GenerateProtocolRef(CodeGenFunction &CGF,
Daniel Dunbar92992502008-08-15 22:20:32 +0000195 const ObjCProtocolDecl *OPD) = 0;
Daniel Dunbar6630e102008-08-12 05:08:18 +0000196
Mike Stump11289f42009-09-09 15:08:12 +0000197 /// Generate the named protocol. Protocols contain method metadata but no
198 /// implementations.
Daniel Dunbar92992502008-08-15 22:20:32 +0000199 virtual void GenerateProtocol(const ObjCProtocolDecl *OPD) = 0;
Daniel Dunbar89da6ad2008-08-13 00:59:25 +0000200
Daniel Dunbar92992502008-08-15 22:20:32 +0000201 /// Generate a function preamble for a method with the specified
Mike Stump11289f42009-09-09 15:08:12 +0000202 /// types.
Daniel Dunbar92992502008-08-15 22:20:32 +0000203
Mike Stump18bb9282009-05-16 07:57:57 +0000204 // FIXME: Current this just generates the Function definition, but really this
205 // should also be generating the loads of the parameters, as the runtime
206 // should have full control over how parameters are passed.
Mike Stump11289f42009-09-09 15:08:12 +0000207 virtual llvm::Function *GenerateMethod(const ObjCMethodDecl *OMD,
Fariborz Jahanian0196a1c2009-01-10 21:06:09 +0000208 const ObjCContainerDecl *CD) = 0;
Daniel Dunbar89da6ad2008-08-13 00:59:25 +0000209
Daniel Dunbara91c3e02008-09-24 03:38:44 +0000210 /// Return the runtime function for getting properties.
Chris Lattnerd4808922009-03-22 21:03:39 +0000211 virtual llvm::Constant *GetPropertyGetFunction() = 0;
Mike Stump11289f42009-09-09 15:08:12 +0000212
Daniel Dunbara91c3e02008-09-24 03:38:44 +0000213 /// Return the runtime function for setting properties.
Chris Lattnerd4808922009-03-22 21:03:39 +0000214 virtual llvm::Constant *GetPropertySetFunction() = 0;
Daniel Dunbara91c3e02008-09-24 03:38:44 +0000215
Ted Kremeneke65b0862012-03-06 20:05:56 +0000216 /// Return the runtime function for optimized setting properties.
Fangrui Song6907ce22018-07-30 19:24:48 +0000217 virtual llvm::Constant *GetOptimizedPropertySetFunction(bool atomic,
Ted Kremeneke65b0862012-03-06 20:05:56 +0000218 bool copy) = 0;
219
David Chisnall168b80f2010-12-26 22:13:16 +0000220 // API for atomic copying of qualified aggregates in getter.
221 virtual llvm::Constant *GetGetStructFunction() = 0;
222 // API for atomic copying of qualified aggregates in setter.
223 virtual llvm::Constant *GetSetStructFunction() = 0;
David Chisnall0d75e062012-12-17 18:54:24 +0000224 /// API for atomic copying of qualified aggregates with non-trivial copy
225 /// assignment (c++) in setter.
226 virtual llvm::Constant *GetCppAtomicObjectSetFunction() = 0;
227 /// API for atomic copying of qualified aggregates with non-trivial copy
228 /// assignment (c++) in getter.
229 virtual llvm::Constant *GetCppAtomicObjectGetFunction() = 0;
Fangrui Song6907ce22018-07-30 19:24:48 +0000230
Daniel Dunbar7c6d3a72008-08-16 00:25:02 +0000231 /// GetClass - Return a reference to the class for the given
232 /// interface decl.
John McCall882987f2013-02-28 19:01:20 +0000233 virtual llvm::Value *GetClass(CodeGenFunction &CGF,
Daniel Dunbar7c6d3a72008-08-16 00:25:02 +0000234 const ObjCInterfaceDecl *OID) = 0;
Fangrui Song6907ce22018-07-30 19:24:48 +0000235
236
John McCall882987f2013-02-28 19:01:20 +0000237 virtual llvm::Value *EmitNSAutoreleasePoolClassRef(CodeGenFunction &CGF) {
David Blaikie83d382b2011-09-23 05:06:16 +0000238 llvm_unreachable("autoreleasepool unsupported in this ABI");
John McCall31168b02011-06-15 23:02:42 +0000239 }
Fangrui Song6907ce22018-07-30 19:24:48 +0000240
Anders Carlsson3f35a262008-08-31 04:05:03 +0000241 /// EnumerationMutationFunction - Return the function that's called by the
242 /// compiler when a mutation is detected during foreach iteration.
Chris Lattnerd4808922009-03-22 21:03:39 +0000243 virtual llvm::Constant *EnumerationMutationFunction() = 0;
Mike Stump11289f42009-09-09 15:08:12 +0000244
John McCallbd309292010-07-06 01:34:17 +0000245 virtual void EmitSynchronizedStmt(CodeGen::CodeGenFunction &CGF,
246 const ObjCAtSynchronizedStmt &S) = 0;
247 virtual void EmitTryStmt(CodeGen::CodeGenFunction &CGF,
248 const ObjCAtTryStmt &S) = 0;
Anders Carlsson1963b0c2008-09-09 10:04:29 +0000249 virtual void EmitThrowStmt(CodeGen::CodeGenFunction &CGF,
Fariborz Jahanian1eab0522013-01-10 19:02:56 +0000250 const ObjCAtThrowStmt &S,
251 bool ClearInsertionPoint=true) = 0;
Daniel Dunbarbe9dae82009-04-21 00:49:20 +0000252 virtual llvm::Value *EmitObjCWeakRead(CodeGen::CodeGenFunction &CGF,
John McCall7f416cc2015-09-08 08:05:57 +0000253 Address AddrWeakObj) = 0;
Fariborz Jahanian83f45b552008-11-18 22:37:34 +0000254 virtual void EmitObjCWeakAssign(CodeGen::CodeGenFunction &CGF,
John McCall7f416cc2015-09-08 08:05:57 +0000255 llvm::Value *src, Address dest) = 0;
Fariborz Jahaniand7db9642008-11-19 00:59:10 +0000256 virtual void EmitObjCGlobalAssign(CodeGen::CodeGenFunction &CGF,
John McCall7f416cc2015-09-08 08:05:57 +0000257 llvm::Value *src, Address dest,
Fariborz Jahanian217af242010-07-20 20:30:03 +0000258 bool threadlocal=false) = 0;
Fariborz Jahaniane881b532008-11-20 19:23:36 +0000259 virtual void EmitObjCIvarAssign(CodeGen::CodeGenFunction &CGF,
John McCall7f416cc2015-09-08 08:05:57 +0000260 llvm::Value *src, Address dest,
Fariborz Jahanian7a95d722009-09-24 22:25:38 +0000261 llvm::Value *ivarOffset) = 0;
Fariborz Jahaniand7db9642008-11-19 00:59:10 +0000262 virtual void EmitObjCStrongCastAssign(CodeGen::CodeGenFunction &CGF,
John McCall7f416cc2015-09-08 08:05:57 +0000263 llvm::Value *src, Address dest) = 0;
Mike Stump11289f42009-09-09 15:08:12 +0000264
Fariborz Jahanian712bfa62009-02-03 19:03:09 +0000265 virtual LValue EmitObjCValueForIvar(CodeGen::CodeGenFunction &CGF,
266 QualType ObjectTy,
267 llvm::Value *BaseValue,
268 const ObjCIvarDecl *Ivar,
Fariborz Jahanian712bfa62009-02-03 19:03:09 +0000269 unsigned CVRQualifiers) = 0;
Fariborz Jahanian21fc74c2009-02-10 19:02:04 +0000270 virtual llvm::Value *EmitIvarOffset(CodeGen::CodeGenFunction &CGF,
Daniel Dunbar722f4242009-04-22 05:08:15 +0000271 const ObjCInterfaceDecl *Interface,
Fariborz Jahanian21fc74c2009-02-10 19:02:04 +0000272 const ObjCIvarDecl *Ivar) = 0;
Fariborz Jahanian5f21d2f2009-07-08 01:18:33 +0000273 virtual void EmitGCMemmoveCollectable(CodeGen::CodeGenFunction &CGF,
John McCall7f416cc2015-09-08 08:05:57 +0000274 Address DestPtr,
275 Address SrcPtr,
Fariborz Jahanian021510e2010-06-15 22:44:06 +0000276 llvm::Value *Size) = 0;
John McCall351762c2011-02-07 10:33:21 +0000277 virtual llvm::Constant *BuildGCBlockLayout(CodeGen::CodeGenModule &CGM,
278 const CodeGen::CGBlockInfo &blockInfo) = 0;
Fariborz Jahanian0c58ce92012-10-27 21:10:38 +0000279 virtual llvm::Constant *BuildRCBlockLayout(CodeGen::CodeGenModule &CGM,
280 const CodeGen::CGBlockInfo &blockInfo) = 0;
Vedant Kumar2f5bb1152015-12-21 20:21:15 +0000281
282 /// Returns an i8* which points to the byref layout information.
Fariborz Jahaniana9d44642012-11-14 17:15:51 +0000283 virtual llvm::Constant *BuildByrefLayout(CodeGen::CodeGenModule &CGM,
284 QualType T) = 0;
Vedant Kumar2f5bb1152015-12-21 20:21:15 +0000285
John McCalla729c622012-02-17 03:33:10 +0000286 struct MessageSendInfo {
287 const CGFunctionInfo &CallInfo;
288 llvm::PointerType *MessengerType;
289
290 MessageSendInfo(const CGFunctionInfo &callInfo,
291 llvm::PointerType *messengerType)
292 : CallInfo(callInfo), MessengerType(messengerType) {}
293 };
294
295 MessageSendInfo getMessageSendInfo(const ObjCMethodDecl *method,
296 QualType resultType,
297 CallArgList &callArgs);
Eli Friedman85937482012-11-06 23:40:48 +0000298
299 // FIXME: This probably shouldn't be here, but the code to compute
300 // it is here.
301 unsigned ComputeBitfieldBitOffset(CodeGen::CodeGenModule &CGM,
302 const ObjCInterfaceDecl *ID,
303 const ObjCIvarDecl *Ivar);
Chris Lattnerb7256cd2008-03-01 08:50:34 +0000304};
305
Mike Stump11289f42009-09-09 15:08:12 +0000306/// Creates an instance of an Objective-C runtime class.
Chris Lattner4bd55962008-03-30 23:03:07 +0000307//TODO: This should include some way of selecting which runtime to target.
Daniel Dunbar303e2c22008-08-11 02:45:11 +0000308CGObjCRuntime *CreateGNUObjCRuntime(CodeGenModule &CGM);
309CGObjCRuntime *CreateMacObjCRuntime(CodeGenModule &CGM);
Alexander Kornienkoab9db512015-06-22 23:07:51 +0000310}
311}
Chris Lattnerb7256cd2008-03-01 08:50:34 +0000312#endif