2008-08-11 10:45:11 +08:00
|
|
|
//===------- CGObjCMac.cpp - Interface to Apple Objective-C Runtime -------===//
|
|
|
|
//
|
|
|
|
// The LLVM Compiler Infrastructure
|
|
|
|
//
|
|
|
|
// This file is distributed under the University of Illinois Open Source
|
|
|
|
// License. See LICENSE.TXT for details.
|
|
|
|
//
|
|
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
//
|
|
|
|
// This provides Objective-C code generation targetting the Apple runtime.
|
|
|
|
//
|
|
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
|
|
|
|
#include "CGObjCRuntime.h"
|
2008-08-12 05:35:06 +08:00
|
|
|
|
|
|
|
#include "CodeGenModule.h"
|
2008-08-16 11:19:19 +08:00
|
|
|
#include "CodeGenFunction.h"
|
2008-08-12 08:12:39 +08:00
|
|
|
#include "clang/AST/ASTContext.h"
|
2008-08-11 12:54:23 +08:00
|
|
|
#include "clang/AST/Decl.h"
|
2008-08-13 11:21:16 +08:00
|
|
|
#include "clang/AST/DeclObjC.h"
|
2008-08-12 05:35:06 +08:00
|
|
|
#include "clang/Basic/LangOptions.h"
|
|
|
|
|
2008-08-12 08:12:39 +08:00
|
|
|
#include "llvm/Module.h"
|
2008-08-11 10:45:11 +08:00
|
|
|
#include "llvm/Support/IRBuilder.h"
|
2008-08-12 14:48:42 +08:00
|
|
|
#include "llvm/Target/TargetData.h"
|
2008-08-16 11:19:19 +08:00
|
|
|
#include <sstream>
|
2008-08-11 10:45:11 +08:00
|
|
|
|
|
|
|
using namespace clang;
|
|
|
|
|
|
|
|
namespace {
|
2008-08-12 08:12:39 +08:00
|
|
|
|
2008-08-13 11:21:16 +08:00
|
|
|
// FIXME: We should find a nicer way to make the labels for
|
|
|
|
// metadata, string concatenation is lame.
|
|
|
|
|
2008-08-12 08:12:39 +08:00
|
|
|
/// ObjCTypesHelper - Helper class that encapsulates lazy
|
|
|
|
/// construction of varies types used during ObjC generation.
|
|
|
|
class ObjCTypesHelper {
|
|
|
|
private:
|
|
|
|
CodeGen::CodeGenModule &CGM;
|
|
|
|
|
|
|
|
const llvm::StructType *CFStringType;
|
|
|
|
llvm::Constant *CFConstantStringClassReference;
|
2008-08-12 11:39:23 +08:00
|
|
|
llvm::Function *MessageSendFn;
|
|
|
|
|
2008-08-12 08:12:39 +08:00
|
|
|
public:
|
2008-08-13 11:21:16 +08:00
|
|
|
const llvm::Type *IntTy, *LongTy;
|
2008-08-12 11:39:23 +08:00
|
|
|
|
2008-08-12 13:28:47 +08:00
|
|
|
/// ObjectPtrTy - LLVM type for object handles (typeof(id))
|
|
|
|
const llvm::Type *ObjectPtrTy;
|
2008-08-12 14:48:42 +08:00
|
|
|
/// SelectorPtrTy - LLVM type for selector handles (typeof(SEL))
|
2008-08-12 13:28:47 +08:00
|
|
|
const llvm::Type *SelectorPtrTy;
|
2008-08-13 11:21:16 +08:00
|
|
|
/// ProtocolPtrTy - LLVM type for external protocol handles
|
|
|
|
/// (typeof(Protocol))
|
|
|
|
const llvm::Type *ExternalProtocolPtrTy;
|
2008-08-12 14:48:42 +08:00
|
|
|
|
|
|
|
/// SymtabTy - LLVM type for struct objc_symtab.
|
|
|
|
const llvm::StructType *SymtabTy;
|
|
|
|
/// ModuleTy - LLVM type for struct objc_module.
|
|
|
|
const llvm::StructType *ModuleTy;
|
2008-08-12 11:39:23 +08:00
|
|
|
|
2008-08-13 11:21:16 +08:00
|
|
|
/// ProtocolTy - LLVM type for struct objc_protocol.
|
|
|
|
const llvm::StructType *ProtocolTy;
|
|
|
|
/// ProtocolPtrTy - LLVM type for struct objc_protocol *.
|
|
|
|
const llvm::Type *ProtocolPtrTy;
|
|
|
|
/// ProtocolExtensionTy - LLVM type for struct
|
|
|
|
/// objc_protocol_extension.
|
|
|
|
const llvm::StructType *ProtocolExtensionTy;
|
|
|
|
/// ProtocolExtensionTy - LLVM type for struct
|
|
|
|
/// objc_protocol_extension *.
|
|
|
|
const llvm::Type *ProtocolExtensionPtrTy;
|
|
|
|
/// MethodDescriptionTy - LLVM type for struct
|
|
|
|
/// objc_method_description.
|
|
|
|
const llvm::StructType *MethodDescriptionTy;
|
|
|
|
/// MethodDescriptionListTy - LLVM type for struct
|
|
|
|
/// objc_method_description_list.
|
|
|
|
const llvm::StructType *MethodDescriptionListTy;
|
|
|
|
/// MethodDescriptionListPtrTy - LLVM type for struct
|
|
|
|
/// objc_method_description_list *.
|
|
|
|
const llvm::Type *MethodDescriptionListPtrTy;
|
|
|
|
/// PropertyListTy - LLVM type for struct objc_property_list.
|
|
|
|
const llvm::Type *PropertyListTy;
|
|
|
|
/// PropertyListPtrTy - LLVM type for struct objc_property_list*.
|
|
|
|
const llvm::Type *PropertyListPtrTy;
|
|
|
|
/// ProtocolListTy - LLVM type for struct objc_property_list.
|
|
|
|
const llvm::Type *ProtocolListTy;
|
|
|
|
/// ProtocolListPtrTy - LLVM type for struct objc_property_list*.
|
|
|
|
const llvm::Type *ProtocolListPtrTy;
|
|
|
|
|
2008-08-12 08:12:39 +08:00
|
|
|
public:
|
|
|
|
ObjCTypesHelper(CodeGen::CodeGenModule &cgm);
|
|
|
|
~ObjCTypesHelper();
|
|
|
|
|
|
|
|
llvm::Constant *getCFConstantStringClassReference();
|
|
|
|
const llvm::StructType *getCFStringType();
|
2008-08-12 11:39:23 +08:00
|
|
|
llvm::Function *getMessageSendFn();
|
2008-08-12 08:12:39 +08:00
|
|
|
};
|
|
|
|
|
2008-08-11 10:45:11 +08:00
|
|
|
class CGObjCMac : public CodeGen::CGObjCRuntime {
|
|
|
|
private:
|
2008-08-12 08:12:39 +08:00
|
|
|
CodeGen::CodeGenModule &CGM;
|
|
|
|
ObjCTypesHelper ObjCTypes;
|
|
|
|
/// ObjCABI - FIXME: Not sure yet.
|
|
|
|
unsigned ObjCABI;
|
2008-08-11 10:45:11 +08:00
|
|
|
|
2008-08-12 14:48:42 +08:00
|
|
|
/// ClassNames - uniqued class names.
|
2008-08-13 11:21:16 +08:00
|
|
|
llvm::DenseMap<IdentifierInfo*, llvm::GlobalVariable*> ClassNames;
|
2008-08-12 14:48:42 +08:00
|
|
|
|
2008-08-12 11:39:23 +08:00
|
|
|
/// MethodVarNames - uniqued method variable names.
|
|
|
|
llvm::DenseMap<Selector, llvm::GlobalVariable*> MethodVarNames;
|
|
|
|
|
2008-08-13 11:21:16 +08:00
|
|
|
/// MethodVarTypes - uniqued method type signatures. We have to use
|
|
|
|
/// a StringMap here because have no other unique reference.
|
|
|
|
llvm::StringMap<llvm::GlobalVariable*> MethodVarTypes;
|
|
|
|
|
2008-08-12 11:39:23 +08:00
|
|
|
/// SelectorReferences - uniqued selector references.
|
|
|
|
llvm::DenseMap<Selector, llvm::GlobalVariable*> SelectorReferences;
|
|
|
|
|
2008-08-13 11:21:16 +08:00
|
|
|
/// Protocols - Protocols for which an objc_protocol structure has
|
|
|
|
/// been emitted. Forward declarations are handled by creating an
|
|
|
|
/// empty structure whose initializer is filled in when/if defined.
|
|
|
|
llvm::DenseMap<IdentifierInfo*, llvm::GlobalVariable*> Protocols;
|
|
|
|
|
|
|
|
/// UsedGlobals - List of globals to pack into the llvm.used metadata
|
2008-08-12 05:35:06 +08:00
|
|
|
/// to prevent them from being clobbered.
|
2008-08-12 14:48:42 +08:00
|
|
|
std::vector<llvm::GlobalVariable*> UsedGlobals;
|
2008-08-12 05:35:06 +08:00
|
|
|
|
|
|
|
/// EmitImageInfo - Emit the image info marker used to encode some module
|
|
|
|
/// level information.
|
|
|
|
void EmitImageInfo();
|
|
|
|
|
2008-08-12 14:48:42 +08:00
|
|
|
/// EmitModuleInfo - Another marker encoding module level
|
|
|
|
/// information.
|
|
|
|
|
|
|
|
// FIXME: Not sure yet of the difference between this and
|
|
|
|
// IMAGE_INFO. otool looks at this before printing out Obj-C info
|
|
|
|
// though...
|
|
|
|
void EmitModuleInfo();
|
|
|
|
|
|
|
|
/// EmitModuleSymols - Emit module symbols, the result is a constant
|
|
|
|
/// of type pointer-to SymtabTy. // FIXME: Describe more completely
|
|
|
|
/// once known.
|
|
|
|
llvm::Constant *EmitModuleSymbols();
|
|
|
|
|
2008-08-12 05:35:06 +08:00
|
|
|
/// FinishModule - Write out global data structures at the end of
|
|
|
|
/// processing a translation unit.
|
|
|
|
void FinishModule();
|
2008-08-13 11:21:16 +08:00
|
|
|
|
|
|
|
/// EmitMethodList - Emit a method description list for a list of
|
|
|
|
/// method declarations.
|
|
|
|
/// - TypeName: The name for the type containing the methods.
|
|
|
|
/// - IsProtocol: True iff these methods are for a protocol.
|
|
|
|
/// - ClassMethds: True iff these are class methods.
|
|
|
|
/// - Required: When true, only "required" methods are
|
|
|
|
/// listed. Similarly, when false only "optional" methods are
|
|
|
|
/// listed. For classes this should always be true.
|
|
|
|
/// - begin, end: The method list to output.
|
|
|
|
///
|
|
|
|
/// The return value has type MethodDescriptionListPtrTy.
|
|
|
|
llvm::Constant *EmitMethodList(const std::string &TypeName,
|
|
|
|
bool IsProtocol,
|
|
|
|
bool ClassMethods,
|
|
|
|
bool Required,
|
|
|
|
ObjCMethodDecl * const *begin,
|
|
|
|
ObjCMethodDecl * const *end);
|
|
|
|
|
|
|
|
/// EmitProtocolExtension - Generate the protocol extension
|
|
|
|
/// structure used to store optional instance and class methods, and
|
|
|
|
/// protocol properties. The return value has type
|
|
|
|
/// ProtocolExtensionPtrTy.
|
|
|
|
llvm::Constant *EmitProtocolExtension(const ObjCProtocolDecl *PD);
|
|
|
|
|
|
|
|
/// EmitProtocolList - Generate the list of referenced
|
|
|
|
/// protocols. The return value has type ProtocolListPtrTy.
|
|
|
|
llvm::Constant *EmitProtocolList(const ObjCProtocolDecl *PD);
|
|
|
|
|
|
|
|
/// GetProtocolRef - Return a reference to the internal protocol
|
|
|
|
/// description, creating an empty one if it has not been
|
|
|
|
/// defined. The return value has type pointer-to ProtocolTy.
|
|
|
|
llvm::GlobalVariable *GetProtocolRef(const ObjCProtocolDecl *PD);
|
|
|
|
|
2008-08-12 11:39:23 +08:00
|
|
|
/// EmitSelector - Return a Value*, of type ObjCTypes.SelectorPtrTy,
|
|
|
|
/// for the given selector.
|
|
|
|
llvm::Value *EmitSelector(llvm::IRBuilder<> &Builder, Selector Sel);
|
|
|
|
|
2008-08-12 14:48:42 +08:00
|
|
|
/// GetClassName - Return a unique constant for the given selector's
|
|
|
|
/// name.
|
2008-08-13 11:21:16 +08:00
|
|
|
llvm::Constant *GetClassName(IdentifierInfo *Ident);
|
2008-08-12 14:48:42 +08:00
|
|
|
|
2008-08-12 11:39:23 +08:00
|
|
|
/// GetMethodVarName - Return a unique constant for the given
|
2008-08-13 11:21:16 +08:00
|
|
|
/// selector's name. This returns a constant i8* to the start of
|
|
|
|
/// the name.
|
2008-08-12 11:39:23 +08:00
|
|
|
llvm::Constant *GetMethodVarName(Selector Sel);
|
2008-08-12 05:35:06 +08:00
|
|
|
|
2008-08-13 11:21:16 +08:00
|
|
|
/// GetMethodVarType - Return a unique constant for the given
|
|
|
|
/// selector's name. This returns a constant i8* to the start of
|
|
|
|
/// the name.
|
|
|
|
llvm::Constant *GetMethodVarType(ObjCMethodDecl *D);
|
|
|
|
|
2008-08-16 11:19:19 +08:00
|
|
|
/// GetNameForMethod - Return a name for the given method.
|
|
|
|
/// \param[out] NameOut - The return value.
|
|
|
|
void GetNameForMethod(const ObjCMethodDecl *OMD,
|
|
|
|
std::string &NameOut);
|
|
|
|
|
2008-08-11 10:45:11 +08:00
|
|
|
public:
|
|
|
|
CGObjCMac(CodeGen::CodeGenModule &cgm);
|
2008-08-12 08:12:39 +08:00
|
|
|
virtual llvm::Constant *GenerateConstantString(const std::string &String);
|
2008-08-11 10:45:11 +08:00
|
|
|
|
|
|
|
virtual llvm::Value *GenerateMessageSend(llvm::IRBuilder<> &Builder,
|
|
|
|
const llvm::Type *ReturnTy,
|
|
|
|
llvm::Value *Receiver,
|
|
|
|
Selector Sel,
|
|
|
|
llvm::Value** ArgV,
|
|
|
|
unsigned ArgC);
|
|
|
|
|
2008-08-16 08:25:02 +08:00
|
|
|
virtual llvm::Value *
|
|
|
|
GenerateMessageSendSuper(llvm::IRBuilder<> &Builder,
|
|
|
|
const llvm::Type *ReturnTy,
|
|
|
|
const ObjCInterfaceDecl *SuperClass,
|
|
|
|
llvm::Value *Receiver,
|
|
|
|
Selector Sel,
|
|
|
|
llvm::Value** ArgV,
|
|
|
|
unsigned ArgC);
|
|
|
|
|
|
|
|
virtual llvm::Value *GetClass(llvm::IRBuilder<> &Builder,
|
|
|
|
const ObjCInterfaceDecl *SuperClass);
|
2008-08-11 10:45:11 +08:00
|
|
|
|
|
|
|
virtual llvm::Value *GetSelector(llvm::IRBuilder<> &Builder, Selector Sel);
|
|
|
|
|
2008-08-16 06:20:32 +08:00
|
|
|
virtual llvm::Function *GenerateMethod(const ObjCMethodDecl *OMD);
|
|
|
|
|
|
|
|
virtual void GenerateCategory(const ObjCCategoryImplDecl *CMD);
|
|
|
|
|
|
|
|
virtual void GenerateClass(const ObjCImplementationDecl *ClassDecl);
|
2008-08-11 10:45:11 +08:00
|
|
|
|
|
|
|
virtual llvm::Value *GenerateProtocolRef(llvm::IRBuilder<> &Builder,
|
2008-08-13 08:59:25 +08:00
|
|
|
const ObjCProtocolDecl *PD);
|
2008-08-11 10:45:11 +08:00
|
|
|
|
2008-08-13 08:59:25 +08:00
|
|
|
virtual void GenerateProtocol(const ObjCProtocolDecl *PD);
|
2008-08-11 10:45:11 +08:00
|
|
|
|
|
|
|
virtual llvm::Function *ModuleInitFunction();
|
|
|
|
};
|
|
|
|
} // end anonymous namespace
|
2008-08-12 08:12:39 +08:00
|
|
|
|
|
|
|
/* *** Helper Functions *** */
|
|
|
|
|
|
|
|
/// getConstantGEP() - Help routine to construct simple GEPs.
|
|
|
|
static llvm::Constant *getConstantGEP(llvm::Constant *C,
|
|
|
|
unsigned idx0,
|
|
|
|
unsigned idx1) {
|
|
|
|
llvm::Value *Idxs[] = {
|
|
|
|
llvm::ConstantInt::get(llvm::Type::Int32Ty, idx0),
|
|
|
|
llvm::ConstantInt::get(llvm::Type::Int32Ty, idx1)
|
|
|
|
};
|
|
|
|
return llvm::ConstantExpr::getGetElementPtr(C, Idxs, 2);
|
|
|
|
}
|
|
|
|
|
|
|
|
/* *** CGObjCMac Public Interface *** */
|
2008-08-11 10:45:11 +08:00
|
|
|
|
2008-08-12 08:12:39 +08:00
|
|
|
CGObjCMac::CGObjCMac(CodeGen::CodeGenModule &cgm)
|
|
|
|
: CGM(cgm),
|
|
|
|
ObjCTypes(cgm),
|
|
|
|
ObjCABI(1)
|
|
|
|
{
|
|
|
|
// FIXME: How does this get set in GCC? And what does it even mean?
|
|
|
|
if (ObjCTypes.LongTy != CGM.getTypes().ConvertType(CGM.getContext().IntTy))
|
|
|
|
ObjCABI = 2;
|
|
|
|
|
|
|
|
EmitImageInfo();
|
2008-08-11 10:45:11 +08:00
|
|
|
}
|
|
|
|
|
2008-08-16 08:25:02 +08:00
|
|
|
/// GetClass - Return a reference to the class for the given interface
|
|
|
|
/// decl.
|
|
|
|
llvm::Value *CGObjCMac::GetClass(llvm::IRBuilder<> &Builder,
|
|
|
|
const ObjCInterfaceDecl *OID) {
|
2008-08-11 10:45:11 +08:00
|
|
|
assert(0 && "Cannot lookup classes on Mac runtime.");
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
/// GetSelector - Return the pointer to the unique'd string for this selector.
|
|
|
|
llvm::Value *CGObjCMac::GetSelector(llvm::IRBuilder<> &Builder, Selector Sel) {
|
2008-08-12 11:39:23 +08:00
|
|
|
return EmitSelector(Builder, Sel);
|
2008-08-11 10:45:11 +08:00
|
|
|
}
|
|
|
|
|
2008-08-12 08:12:39 +08:00
|
|
|
/// Generate a constant CFString object.
|
|
|
|
/*
|
|
|
|
struct __builtin_CFString {
|
|
|
|
const int *isa; // point to __CFConstantStringClassReference
|
|
|
|
int flags;
|
|
|
|
const char *str;
|
|
|
|
long length;
|
|
|
|
};
|
|
|
|
*/
|
|
|
|
|
|
|
|
llvm::Constant *CGObjCMac::GenerateConstantString(const std::string &String) {
|
|
|
|
// FIXME: I have no idea what this constant is (it is a magic
|
|
|
|
// constant in GCC as well). Most likely the encoding of the string
|
|
|
|
// and at least one part of it relates to UTF-16. Is this just the
|
|
|
|
// code for UTF-8? Where is this handled for us?
|
|
|
|
// See: <rdr://2996215>
|
|
|
|
unsigned flags = 0x07c8;
|
|
|
|
|
|
|
|
// FIXME: Use some machinery to unique this. We can't reuse the CGM
|
|
|
|
// one since we put them in a different section.
|
|
|
|
llvm::Constant *StringC = llvm::ConstantArray::get(String);
|
|
|
|
llvm::Constant *StringGV =
|
|
|
|
new llvm::GlobalVariable(StringC->getType(), true,
|
|
|
|
llvm::GlobalValue::InternalLinkage,
|
|
|
|
StringC, ".str", &CGM.getModule());
|
|
|
|
llvm::Constant *Values[4] = {
|
|
|
|
ObjCTypes.getCFConstantStringClassReference(),
|
|
|
|
llvm::ConstantInt::get(llvm::Type::Int32Ty, flags),
|
|
|
|
getConstantGEP(StringGV, 0, 0), // Decay array -> ptr
|
|
|
|
llvm::ConstantInt::get(ObjCTypes.LongTy, String.size())
|
|
|
|
};
|
|
|
|
|
|
|
|
llvm::Constant *CFStringC =
|
|
|
|
llvm::ConstantStruct::get(ObjCTypes.getCFStringType(),
|
|
|
|
std::vector<llvm::Constant*>(Values, Values+4));
|
|
|
|
|
|
|
|
llvm::GlobalVariable *CFStringGV =
|
|
|
|
new llvm::GlobalVariable(CFStringC->getType(), true,
|
|
|
|
llvm::GlobalValue::InternalLinkage,
|
|
|
|
CFStringC, "",
|
|
|
|
&CGM.getModule());
|
|
|
|
|
|
|
|
CFStringGV->setSection("__DATA, __cfstring");
|
|
|
|
|
|
|
|
return CFStringGV;
|
2008-08-11 10:45:11 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
/// Generates a message send where the super is the receiver. This is
|
|
|
|
/// a message send to self with special delivery semantics indicating
|
|
|
|
/// which class's method should be called.
|
2008-08-16 08:25:02 +08:00
|
|
|
llvm::Value *
|
|
|
|
CGObjCMac::GenerateMessageSendSuper(llvm::IRBuilder<> &Builder,
|
|
|
|
const llvm::Type *ReturnTy,
|
|
|
|
const ObjCInterfaceDecl *SuperClass,
|
|
|
|
llvm::Value *Receiver,
|
|
|
|
Selector Sel,
|
|
|
|
llvm::Value** ArgV,
|
|
|
|
unsigned ArgC) {
|
2008-08-11 10:45:11 +08:00
|
|
|
assert(0 && "Cannot generate message send to super for Mac runtime.");
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Generate code for a message send expression.
|
|
|
|
llvm::Value *CGObjCMac::GenerateMessageSend(llvm::IRBuilder<> &Builder,
|
|
|
|
const llvm::Type *ReturnTy,
|
|
|
|
llvm::Value *Receiver,
|
|
|
|
Selector Sel,
|
|
|
|
llvm::Value** ArgV,
|
2008-08-12 13:28:47 +08:00
|
|
|
unsigned ArgC) {
|
|
|
|
llvm::Function *F = ObjCTypes.getMessageSendFn();
|
|
|
|
llvm::Value **Args = new llvm::Value*[ArgC+2];
|
|
|
|
Args[0] = Builder.CreateBitCast(Receiver, ObjCTypes.ObjectPtrTy, "tmp");
|
|
|
|
Args[1] = EmitSelector(Builder, Sel);
|
|
|
|
std::copy(ArgV, ArgV+ArgC, Args+2);
|
2008-08-16 06:20:32 +08:00
|
|
|
|
|
|
|
std::vector<const llvm::Type*> Params;
|
|
|
|
Params.push_back(ObjCTypes.ObjectPtrTy);
|
|
|
|
Params.push_back(ObjCTypes.SelectorPtrTy);
|
|
|
|
llvm::FunctionType *CallFTy = llvm::FunctionType::get(ReturnTy,
|
|
|
|
Params,
|
|
|
|
true);
|
|
|
|
llvm::Type *PCallFTy = llvm::PointerType::getUnqual(CallFTy);
|
|
|
|
llvm::Constant *C = llvm::ConstantExpr::getBitCast(F, PCallFTy);
|
|
|
|
llvm::CallInst *CI = Builder.CreateCall(C,
|
|
|
|
Args, Args+ArgC+2, "tmp");
|
2008-08-12 13:28:47 +08:00
|
|
|
delete[] Args;
|
|
|
|
return Builder.CreateBitCast(CI, ReturnTy, "tmp");
|
2008-08-11 10:45:11 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
llvm::Value *CGObjCMac::GenerateProtocolRef(llvm::IRBuilder<> &Builder,
|
2008-08-13 08:59:25 +08:00
|
|
|
const ObjCProtocolDecl *PD) {
|
2008-08-13 11:21:16 +08:00
|
|
|
return llvm::ConstantExpr::getBitCast(GetProtocolRef(PD),
|
|
|
|
ObjCTypes.ExternalProtocolPtrTy);
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
// APPLE LOCAL radar 4585769 - Objective-C 1.0 extensions
|
|
|
|
struct _objc_protocol {
|
|
|
|
struct _objc_protocol_extension *isa;
|
|
|
|
char *protocol_name;
|
|
|
|
struct _objc_protocol_list *protocol_list;
|
|
|
|
struct _objc__method_prototype_list *instance_methods;
|
|
|
|
struct _objc__method_prototype_list *class_methods
|
|
|
|
};
|
|
|
|
|
|
|
|
See EmitProtocolExtension().
|
|
|
|
*/
|
|
|
|
void CGObjCMac::GenerateProtocol(const ObjCProtocolDecl *PD) {
|
|
|
|
const char *ProtocolName = PD->getName();
|
|
|
|
|
|
|
|
std::vector<llvm::Constant*> Values(5);
|
|
|
|
Values[0] = EmitProtocolExtension(PD);
|
|
|
|
Values[1] = GetClassName(PD->getIdentifier());
|
|
|
|
Values[2] = EmitProtocolList(PD);
|
|
|
|
Values[3] = EmitMethodList(ProtocolName,
|
|
|
|
true, // IsProtocol
|
|
|
|
false, // ClassMethods
|
|
|
|
true, // Required
|
|
|
|
PD->instmeth_begin(),
|
|
|
|
PD->instmeth_end());
|
|
|
|
Values[4] = EmitMethodList(ProtocolName,
|
|
|
|
true, // IsProtocol
|
|
|
|
true, // ClassMethods
|
|
|
|
true, // Required
|
|
|
|
PD->classmeth_begin(),
|
|
|
|
PD->classmeth_end());
|
|
|
|
llvm::Constant *Init = llvm::ConstantStruct::get(ObjCTypes.ProtocolTy,
|
|
|
|
Values);
|
|
|
|
|
|
|
|
llvm::GlobalVariable *&Entry = Protocols[PD->getIdentifier()];
|
|
|
|
if (Entry) {
|
|
|
|
// Already created, just update the initializer
|
|
|
|
Entry->setInitializer(Init);
|
|
|
|
} else {
|
|
|
|
Entry =
|
|
|
|
new llvm::GlobalVariable(ObjCTypes.ProtocolTy, false,
|
|
|
|
llvm::GlobalValue::InternalLinkage,
|
|
|
|
Init,
|
|
|
|
std::string("\01L_OBJC_PROTOCOL_")+ProtocolName,
|
|
|
|
&CGM.getModule());
|
|
|
|
Entry->setSection("__OBJC,__protocol,regular,no_dead_strip");
|
|
|
|
UsedGlobals.push_back(Entry);
|
|
|
|
// FIXME: Is this necessary? Why only for protocol?
|
|
|
|
Entry->setAlignment(4);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
llvm::GlobalVariable *CGObjCMac::GetProtocolRef(const ObjCProtocolDecl *PD) {
|
|
|
|
llvm::GlobalVariable *&Entry = Protocols[PD->getIdentifier()];
|
|
|
|
|
|
|
|
if (!Entry) {
|
|
|
|
std::vector<llvm::Constant*> Values(5);
|
|
|
|
Values[0] = llvm::Constant::getNullValue(ObjCTypes.ProtocolExtensionPtrTy);
|
|
|
|
Values[1] = GetClassName(PD->getIdentifier());
|
|
|
|
Values[2] = llvm::Constant::getNullValue(ObjCTypes.ProtocolListPtrTy);
|
|
|
|
Values[3] = Values[4] =
|
|
|
|
llvm::Constant::getNullValue(ObjCTypes.MethodDescriptionListPtrTy);
|
|
|
|
llvm::Constant *Init = llvm::ConstantStruct::get(ObjCTypes.ProtocolTy,
|
|
|
|
Values);
|
|
|
|
|
|
|
|
Entry =
|
|
|
|
new llvm::GlobalVariable(ObjCTypes.ProtocolTy, false,
|
|
|
|
llvm::GlobalValue::InternalLinkage,
|
|
|
|
Init,
|
|
|
|
std::string("\01L_OBJC_PROTOCOL_")+PD->getName(),
|
|
|
|
&CGM.getModule());
|
|
|
|
Entry->setSection("__OBJC,__protocol,regular,no_dead_strip");
|
|
|
|
UsedGlobals.push_back(Entry);
|
|
|
|
// FIXME: Is this necessary? Why only for protocol?
|
|
|
|
Entry->setAlignment(4);
|
|
|
|
}
|
|
|
|
|
|
|
|
return Entry;
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
struct _objc_protocol_extension {
|
|
|
|
uint32_t size;
|
|
|
|
struct objc_method_description_list *optional_instance_methods;
|
|
|
|
struct objc_method_description_list *optional_class_methods;
|
|
|
|
struct objc_property_list *instance_properties;
|
|
|
|
};
|
|
|
|
*/
|
|
|
|
llvm::Constant *CGObjCMac::EmitProtocolExtension(const ObjCProtocolDecl *PD) {
|
|
|
|
uint64_t Size =
|
|
|
|
CGM.getTargetData().getABITypeSize(ObjCTypes.ProtocolExtensionTy);
|
|
|
|
std::vector<llvm::Constant*> Values(4);
|
|
|
|
Values[0] = llvm::ConstantInt::get(ObjCTypes.IntTy, Size);
|
|
|
|
Values[1] = EmitMethodList(PD->getName(),
|
|
|
|
true, // IsProtocol
|
|
|
|
false, // ClassMethods
|
|
|
|
false, // Required
|
|
|
|
PD->instmeth_begin(),
|
|
|
|
PD->instmeth_end());
|
|
|
|
Values[2] = EmitMethodList(PD->getName(),
|
|
|
|
true, // IsProtocol
|
|
|
|
true, // ClassMethods
|
|
|
|
false, // Required
|
|
|
|
PD->classmeth_begin(),
|
|
|
|
PD->classmeth_end());
|
|
|
|
Values[3] = llvm::Constant::getNullValue(ObjCTypes.PropertyListPtrTy);
|
|
|
|
assert(!PD->getNumPropertyDecl() &&
|
|
|
|
"Cannot emit Obj-C protocol properties for NeXT runtime.");
|
|
|
|
|
|
|
|
// Return null if no extension bits are used
|
|
|
|
if (Values[1]->isNullValue() && Values[2]->isNullValue() &&
|
|
|
|
Values[3]->isNullValue())
|
|
|
|
return llvm::Constant::getNullValue(ObjCTypes.ProtocolExtensionPtrTy);
|
|
|
|
|
|
|
|
llvm::Constant *Init =
|
|
|
|
llvm::ConstantStruct::get(ObjCTypes.ProtocolExtensionTy, Values);
|
|
|
|
llvm::GlobalVariable *GV =
|
|
|
|
new llvm::GlobalVariable(ObjCTypes.ProtocolExtensionTy, false,
|
|
|
|
llvm::GlobalValue::InternalLinkage,
|
|
|
|
Init,
|
|
|
|
(std::string("\01L_OBJC_PROTOCOLEXT_") +
|
|
|
|
PD->getName()),
|
|
|
|
&CGM.getModule());
|
|
|
|
// No special section, but goes in llvm.used
|
|
|
|
UsedGlobals.push_back(GV);
|
|
|
|
|
|
|
|
return GV;
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
struct objc_protocol_list {
|
|
|
|
struct objc_protocol_list *next;
|
|
|
|
long count;
|
|
|
|
Protocol *list[];
|
|
|
|
};
|
|
|
|
*/
|
|
|
|
llvm::Constant *CGObjCMac::EmitProtocolList(const ObjCProtocolDecl *PD) {
|
|
|
|
std::vector<llvm::Constant*> ProtocolRefs;
|
|
|
|
|
|
|
|
for (ObjCProtocolDecl::protocol_iterator i = PD->protocol_begin(),
|
|
|
|
e = PD->protocol_end(); i != e; ++i)
|
|
|
|
ProtocolRefs.push_back(GetProtocolRef(*i));
|
|
|
|
|
|
|
|
// Just return null for empty protocol lists
|
|
|
|
if (ProtocolRefs.empty())
|
|
|
|
return llvm::Constant::getNullValue(ObjCTypes.ProtocolListPtrTy);
|
|
|
|
|
|
|
|
// This list is null terminated?
|
|
|
|
ProtocolRefs.push_back(llvm::Constant::getNullValue(ObjCTypes.ProtocolPtrTy));
|
|
|
|
|
|
|
|
std::vector<llvm::Constant*> Values(3);
|
|
|
|
// XXX: What is this for?
|
|
|
|
Values[0] = llvm::Constant::getNullValue(ObjCTypes.ProtocolListPtrTy);
|
|
|
|
Values[1] = llvm::ConstantInt::get(ObjCTypes.LongTy, ProtocolRefs.size() - 1);
|
|
|
|
Values[2] =
|
|
|
|
llvm::ConstantArray::get(llvm::ArrayType::get(ObjCTypes.ProtocolPtrTy,
|
|
|
|
ProtocolRefs.size()),
|
|
|
|
ProtocolRefs);
|
|
|
|
|
|
|
|
llvm::Constant *Init = llvm::ConstantStruct::get(Values);
|
|
|
|
llvm::GlobalVariable *GV =
|
|
|
|
new llvm::GlobalVariable(Init->getType(), false,
|
|
|
|
llvm::GlobalValue::InternalLinkage,
|
|
|
|
Init,
|
|
|
|
(std::string("\01L_OBJC_PROTOCOL_REFS_") +
|
|
|
|
PD->getName()),
|
|
|
|
&CGM.getModule());
|
|
|
|
GV->setSection("__OBJC,__cat_cls_meth,regular,no_dead_strip");
|
|
|
|
return llvm::ConstantExpr::getBitCast(GV, ObjCTypes.ProtocolListPtrTy);
|
2008-08-11 10:45:11 +08:00
|
|
|
}
|
|
|
|
|
2008-08-13 11:21:16 +08:00
|
|
|
/*
|
|
|
|
struct objc_method_description_list {
|
|
|
|
int count;
|
|
|
|
struct objc_method_description list[];
|
|
|
|
};
|
|
|
|
*/
|
|
|
|
llvm::Constant *CGObjCMac::EmitMethodList(const std::string &TypeName,
|
|
|
|
bool IsProtocol,
|
|
|
|
bool ClassMethods,
|
|
|
|
bool Required,
|
|
|
|
ObjCMethodDecl * const *begin,
|
|
|
|
ObjCMethodDecl * const *end) {
|
|
|
|
std::vector<llvm::Constant*> Methods, Desc(2);
|
|
|
|
for (; begin != end; ++begin) {
|
|
|
|
ObjCMethodDecl *D = *begin;
|
|
|
|
bool IsRequired = D->getImplementationControl() != ObjCMethodDecl::Optional;
|
|
|
|
|
|
|
|
// Skip if this method is required and we are outputting optional
|
|
|
|
// methods, or vice versa.
|
|
|
|
if (Required != IsRequired)
|
|
|
|
continue;
|
|
|
|
|
|
|
|
Desc[0] = llvm::ConstantExpr::getBitCast(GetMethodVarName(D->getSelector()),
|
|
|
|
ObjCTypes.SelectorPtrTy);
|
|
|
|
Desc[1] = GetMethodVarType(D);
|
|
|
|
Methods.push_back(llvm::ConstantStruct::get(ObjCTypes.MethodDescriptionTy,
|
|
|
|
Desc));
|
|
|
|
}
|
|
|
|
|
|
|
|
// Return null for empty list.
|
|
|
|
if (Methods.empty())
|
|
|
|
return llvm::Constant::getNullValue(ObjCTypes.MethodDescriptionListPtrTy);
|
|
|
|
|
|
|
|
std::vector<llvm::Constant*> Values(2);
|
|
|
|
Values[0] = llvm::ConstantInt::get(ObjCTypes.IntTy, Methods.size());
|
|
|
|
llvm::ArrayType *AT = llvm::ArrayType::get(ObjCTypes.MethodDescriptionTy,
|
|
|
|
Methods.size());
|
|
|
|
Values[1] = llvm::ConstantArray::get(AT, Methods);
|
|
|
|
llvm::Constant *Init = llvm::ConstantStruct::get(Values);
|
|
|
|
|
|
|
|
char Prefix[256];
|
|
|
|
sprintf(Prefix, "\01L_OBJC_%s%sMETHODS_%s",
|
|
|
|
IsProtocol ? "PROTOCOL_" : "",
|
|
|
|
ClassMethods ? "CLASS_" : "INSTANCE_",
|
|
|
|
!Required ? "OPT_" : "");
|
|
|
|
llvm::GlobalVariable *GV =
|
|
|
|
new llvm::GlobalVariable(Init->getType(), false,
|
|
|
|
llvm::GlobalValue::InternalLinkage,
|
|
|
|
Init,
|
|
|
|
std::string(Prefix) + TypeName,
|
|
|
|
&CGM.getModule());
|
|
|
|
if (ClassMethods) {
|
|
|
|
GV->setSection("__OBJC,__cat_cls_meth,regular,no_dead_strip");
|
|
|
|
} else {
|
|
|
|
GV->setSection("__OBJC,__cat_inst_meth,regular,no_dead_strip");
|
|
|
|
}
|
|
|
|
UsedGlobals.push_back(GV);
|
|
|
|
return llvm::ConstantExpr::getBitCast(GV,
|
|
|
|
ObjCTypes.MethodDescriptionListPtrTy);
|
2008-08-11 10:45:11 +08:00
|
|
|
}
|
|
|
|
|
2008-08-16 06:20:32 +08:00
|
|
|
void CGObjCMac::GenerateCategory(const ObjCCategoryImplDecl *OCD) {
|
2008-08-11 10:45:11 +08:00
|
|
|
assert(0 && "Cannot generate category for Mac runtime.");
|
|
|
|
}
|
|
|
|
|
2008-08-16 06:20:32 +08:00
|
|
|
void CGObjCMac::GenerateClass(const ObjCImplementationDecl *ClassDecl) {
|
2008-08-16 11:19:19 +08:00
|
|
|
//assert(0 && "Cannot generate class for Mac runtime.");
|
|
|
|
}
|
|
|
|
|
|
|
|
llvm::Function *CGObjCMac::GenerateMethod(const ObjCMethodDecl *OMD) {
|
|
|
|
const llvm::Type *ReturnTy =
|
|
|
|
CGM.getTypes().ConvertReturnType(OMD->getResultType());
|
|
|
|
const llvm::Type *SelfTy =
|
|
|
|
CGM.getTypes().ConvertType(OMD->getSelfDecl()->getType());
|
|
|
|
|
|
|
|
std::vector<const llvm::Type*> ArgTys;
|
|
|
|
ArgTys.reserve(1 + 2 + OMD->param_size());
|
|
|
|
|
|
|
|
// FIXME: This is not something we should have to be dealing with
|
|
|
|
// here.
|
|
|
|
bool useStructRet =
|
|
|
|
CodeGen::CodeGenFunction::hasAggregateLLVMType(OMD->getResultType());
|
|
|
|
if (useStructRet) {
|
|
|
|
ArgTys.push_back(llvm::PointerType::getUnqual(ReturnTy));
|
|
|
|
ReturnTy = llvm::Type::VoidTy;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Implicit arguments
|
|
|
|
ArgTys.push_back(SelfTy);
|
|
|
|
ArgTys.push_back(ObjCTypes.SelectorPtrTy);
|
|
|
|
|
|
|
|
for (ObjCMethodDecl::param_const_iterator
|
|
|
|
i = OMD->param_begin(), e = OMD->param_end();
|
|
|
|
i != e; ++i) {
|
|
|
|
const llvm::Type *Ty = CGM.getTypes().ConvertType((*i)->getType());
|
|
|
|
if (Ty->isFirstClassType()) {
|
|
|
|
ArgTys.push_back(Ty);
|
|
|
|
} else {
|
|
|
|
ArgTys.push_back(llvm::PointerType::getUnqual(Ty));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
std::string Name;
|
|
|
|
GetNameForMethod(OMD, Name);
|
|
|
|
|
|
|
|
llvm::Function *Method =
|
|
|
|
llvm::Function::Create(llvm::FunctionType::get(ReturnTy,
|
|
|
|
ArgTys,
|
|
|
|
OMD->isVariadic()),
|
|
|
|
llvm::GlobalValue::InternalLinkage,
|
|
|
|
Name,
|
|
|
|
&CGM.getModule());
|
|
|
|
|
|
|
|
if (useStructRet)
|
|
|
|
Method->addParamAttr(1, llvm::ParamAttr::StructRet);
|
|
|
|
|
|
|
|
return Method;
|
2008-08-11 10:45:11 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
llvm::Function *CGObjCMac::ModuleInitFunction() {
|
2008-08-12 05:35:06 +08:00
|
|
|
// Abuse this interface function as a place to finalize.
|
|
|
|
FinishModule();
|
|
|
|
|
2008-08-11 10:45:11 +08:00
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
2008-08-12 05:35:06 +08:00
|
|
|
/* *** Private Interface *** */
|
|
|
|
|
|
|
|
/// EmitImageInfo - Emit the image info marker used to encode some module
|
|
|
|
/// level information.
|
|
|
|
///
|
|
|
|
/// See: <rdr://4810609&4810587&4810587>
|
|
|
|
/// struct IMAGE_INFO {
|
|
|
|
/// unsigned version;
|
|
|
|
/// unsigned flags;
|
|
|
|
/// };
|
|
|
|
enum ImageInfoFlags {
|
|
|
|
eImageInfo_FixAndContinue = (1 << 0), // FIXME: Not sure what this implies
|
|
|
|
eImageInfo_GarbageCollected = (1 << 1),
|
|
|
|
eImageInfo_GCOnly = (1 << 2)
|
|
|
|
};
|
|
|
|
|
|
|
|
void CGObjCMac::EmitImageInfo() {
|
|
|
|
unsigned version = 0; // Version is unused?
|
|
|
|
unsigned flags = 0;
|
|
|
|
|
|
|
|
// FIXME: Fix and continue?
|
|
|
|
if (CGM.getLangOptions().getGCMode() != LangOptions::NonGC)
|
|
|
|
flags |= eImageInfo_GarbageCollected;
|
|
|
|
if (CGM.getLangOptions().getGCMode() == LangOptions::GCOnly)
|
|
|
|
flags |= eImageInfo_GCOnly;
|
|
|
|
|
|
|
|
// Emitted as int[2];
|
|
|
|
llvm::Constant *values[2] = {
|
|
|
|
llvm::ConstantInt::get(llvm::Type::Int32Ty, version),
|
|
|
|
llvm::ConstantInt::get(llvm::Type::Int32Ty, flags)
|
|
|
|
};
|
|
|
|
llvm::ArrayType *AT = llvm::ArrayType::get(llvm::Type::Int32Ty, 2);
|
2008-08-12 14:48:42 +08:00
|
|
|
llvm::GlobalVariable *GV =
|
2008-08-12 05:35:06 +08:00
|
|
|
new llvm::GlobalVariable(AT, true,
|
|
|
|
llvm::GlobalValue::InternalLinkage,
|
|
|
|
llvm::ConstantArray::get(AT, values, 2),
|
|
|
|
"\01L_OBJC_IMAGE_INFO",
|
|
|
|
&CGM.getModule());
|
|
|
|
|
2008-08-12 08:12:39 +08:00
|
|
|
if (ObjCABI == 1) {
|
|
|
|
GV->setSection("__OBJC, __image_info,regular");
|
|
|
|
} else {
|
|
|
|
GV->setSection("__DATA, __objc_imageinfo, regular, no_dead_strip");
|
|
|
|
}
|
2008-08-12 05:35:06 +08:00
|
|
|
|
|
|
|
UsedGlobals.push_back(GV);
|
|
|
|
}
|
|
|
|
|
2008-08-12 14:48:42 +08:00
|
|
|
|
|
|
|
// struct objc_module {
|
|
|
|
// unsigned long version;
|
|
|
|
// unsigned long size;
|
|
|
|
// const char *name;
|
|
|
|
// Symtab symtab;
|
|
|
|
// };
|
|
|
|
|
|
|
|
// FIXME: Get from somewhere
|
|
|
|
static const int ModuleVersion = 7;
|
|
|
|
|
|
|
|
void CGObjCMac::EmitModuleInfo() {
|
|
|
|
uint64_t Size = CGM.getTargetData().getABITypeSize(ObjCTypes.ModuleTy);
|
|
|
|
|
|
|
|
std::vector<llvm::Constant*> Values(4);
|
|
|
|
Values[0] = llvm::ConstantInt::get(ObjCTypes.LongTy, ModuleVersion);
|
|
|
|
Values[1] = llvm::ConstantInt::get(ObjCTypes.LongTy, Size);
|
2008-08-16 06:20:32 +08:00
|
|
|
// This used to be the filename, now it is unused. <rdr://4327263>
|
2008-08-13 11:21:16 +08:00
|
|
|
Values[2] = GetClassName(&CGM.getContext().Idents.get(""));
|
2008-08-12 14:48:42 +08:00
|
|
|
Values[3] = EmitModuleSymbols();
|
|
|
|
|
|
|
|
llvm::GlobalVariable *GV =
|
|
|
|
new llvm::GlobalVariable(ObjCTypes.ModuleTy, false,
|
|
|
|
llvm::GlobalValue::InternalLinkage,
|
|
|
|
llvm::ConstantStruct::get(ObjCTypes.ModuleTy,
|
|
|
|
Values),
|
2008-08-13 11:21:16 +08:00
|
|
|
"\01L_OBJC_MODULES",
|
2008-08-12 14:48:42 +08:00
|
|
|
&CGM.getModule());
|
|
|
|
GV->setSection("__OBJC,__module_info,regular,no_dead_strip");
|
|
|
|
UsedGlobals.push_back(GV);
|
|
|
|
}
|
|
|
|
|
|
|
|
llvm::Constant *CGObjCMac::EmitModuleSymbols() {
|
|
|
|
// FIXME: Is this ever used?
|
|
|
|
llvm::GlobalVariable *GV =
|
|
|
|
new llvm::GlobalVariable(ObjCTypes.SymtabTy, false,
|
|
|
|
llvm::GlobalValue::InternalLinkage,
|
|
|
|
llvm::Constant::getNullValue(ObjCTypes.SymtabTy),
|
|
|
|
"\01L_OBJC_SYMBOLS",
|
|
|
|
&CGM.getModule());
|
|
|
|
GV->setSection("__OBJC,__symbols,regular,no_dead_strip");
|
|
|
|
UsedGlobals.push_back(GV);
|
|
|
|
return GV;
|
|
|
|
}
|
|
|
|
|
2008-08-12 11:39:23 +08:00
|
|
|
llvm::Value *CGObjCMac::EmitSelector(llvm::IRBuilder<> &Builder, Selector Sel) {
|
|
|
|
llvm::GlobalVariable *&Entry = SelectorReferences[Sel];
|
|
|
|
|
|
|
|
if (!Entry) {
|
|
|
|
llvm::Constant *Casted =
|
|
|
|
llvm::ConstantExpr::getBitCast(GetMethodVarName(Sel),
|
|
|
|
ObjCTypes.SelectorPtrTy);
|
|
|
|
Entry =
|
|
|
|
new llvm::GlobalVariable(ObjCTypes.SelectorPtrTy, false,
|
|
|
|
llvm::GlobalValue::InternalLinkage,
|
|
|
|
Casted, "\01L_OBJC_SELECTOR_REFERENCES_",
|
|
|
|
&CGM.getModule());
|
|
|
|
Entry->setSection("__OBJC,__message_refs,literal_pointers,no_dead_strip");
|
|
|
|
UsedGlobals.push_back(Entry);
|
|
|
|
}
|
|
|
|
|
|
|
|
return Builder.CreateLoad(Entry, false, "tmp");
|
|
|
|
}
|
|
|
|
|
2008-08-13 11:21:16 +08:00
|
|
|
llvm::Constant *CGObjCMac::GetClassName(IdentifierInfo *Ident) {
|
|
|
|
llvm::GlobalVariable *&Entry = ClassNames[Ident];
|
2008-08-12 14:48:42 +08:00
|
|
|
|
|
|
|
if (!Entry) {
|
2008-08-13 11:21:16 +08:00
|
|
|
llvm::Constant *C = llvm::ConstantArray::get(Ident->getName());
|
2008-08-12 14:48:42 +08:00
|
|
|
Entry =
|
2008-08-13 11:21:16 +08:00
|
|
|
new llvm::GlobalVariable(C->getType(), false,
|
2008-08-12 14:48:42 +08:00
|
|
|
llvm::GlobalValue::InternalLinkage,
|
|
|
|
C, "\01L_OBJC_CLASS_NAME_",
|
|
|
|
&CGM.getModule());
|
|
|
|
Entry->setSection("__TEXT,__cstring,cstring_literals");
|
|
|
|
UsedGlobals.push_back(Entry);
|
|
|
|
}
|
|
|
|
|
2008-08-13 11:21:16 +08:00
|
|
|
return getConstantGEP(Entry, 0, 0);
|
2008-08-12 14:48:42 +08:00
|
|
|
}
|
|
|
|
|
2008-08-12 11:39:23 +08:00
|
|
|
llvm::Constant *CGObjCMac::GetMethodVarName(Selector Sel) {
|
|
|
|
llvm::GlobalVariable *&Entry = MethodVarNames[Sel];
|
|
|
|
|
|
|
|
if (!Entry) {
|
|
|
|
llvm::Constant *C = llvm::ConstantArray::get(Sel.getName());
|
|
|
|
Entry =
|
2008-08-13 11:21:16 +08:00
|
|
|
new llvm::GlobalVariable(C->getType(), false,
|
2008-08-12 11:39:23 +08:00
|
|
|
llvm::GlobalValue::InternalLinkage,
|
|
|
|
C, "\01L_OBJC_METH_VAR_NAME_",
|
|
|
|
&CGM.getModule());
|
|
|
|
Entry->setSection("__TEXT,__cstring,cstring_literals");
|
|
|
|
UsedGlobals.push_back(Entry);
|
|
|
|
}
|
|
|
|
|
2008-08-13 11:21:16 +08:00
|
|
|
return getConstantGEP(Entry, 0, 0);
|
|
|
|
}
|
|
|
|
|
|
|
|
llvm::Constant *CGObjCMac::GetMethodVarType(ObjCMethodDecl *D) {
|
|
|
|
std::string TypeStr;
|
|
|
|
CGM.getContext().getObjCEncodingForMethodDecl(D, TypeStr);
|
|
|
|
llvm::GlobalVariable *&Entry = MethodVarTypes[TypeStr];
|
|
|
|
|
|
|
|
if (!Entry) {
|
|
|
|
llvm::Constant *C = llvm::ConstantArray::get(TypeStr);
|
|
|
|
Entry =
|
|
|
|
new llvm::GlobalVariable(C->getType(), false,
|
|
|
|
llvm::GlobalValue::InternalLinkage,
|
|
|
|
C, "\01L_OBJC_METH_VAR_TYPE_",
|
|
|
|
&CGM.getModule());
|
|
|
|
Entry->setSection("__TEXT,__cstring,cstring_literals");
|
|
|
|
UsedGlobals.push_back(Entry);
|
|
|
|
}
|
|
|
|
|
|
|
|
return getConstantGEP(Entry, 0, 0);
|
2008-08-12 11:39:23 +08:00
|
|
|
}
|
|
|
|
|
2008-08-16 11:19:19 +08:00
|
|
|
void CGObjCMac::GetNameForMethod(const ObjCMethodDecl *D,
|
|
|
|
std::string &NameOut) {
|
|
|
|
// FIXME: Find the mangling GCC uses.
|
|
|
|
std::stringstream s;
|
|
|
|
s << (D->isInstance() ? "-" : "+");
|
|
|
|
s << "[";
|
|
|
|
s << D->getClassInterface()->getName();
|
|
|
|
s << " ";
|
|
|
|
s << D->getSelector().getName();
|
|
|
|
s << "]";
|
|
|
|
NameOut = s.str();
|
|
|
|
}
|
|
|
|
|
2008-08-12 05:35:06 +08:00
|
|
|
void CGObjCMac::FinishModule() {
|
2008-08-12 14:48:42 +08:00
|
|
|
EmitModuleInfo();
|
|
|
|
|
2008-08-12 05:35:06 +08:00
|
|
|
std::vector<llvm::Constant*> Used;
|
2008-08-12 14:48:42 +08:00
|
|
|
|
2008-08-12 05:35:06 +08:00
|
|
|
llvm::Type *I8Ptr = llvm::PointerType::getUnqual(llvm::Type::Int8Ty);
|
2008-08-12 14:48:42 +08:00
|
|
|
for (std::vector<llvm::GlobalVariable*>::iterator i = UsedGlobals.begin(),
|
2008-08-12 05:35:06 +08:00
|
|
|
e = UsedGlobals.end(); i != e; ++i) {
|
|
|
|
Used.push_back(llvm::ConstantExpr::getBitCast(*i, I8Ptr));
|
|
|
|
}
|
|
|
|
|
|
|
|
llvm::ArrayType *AT = llvm::ArrayType::get(I8Ptr, Used.size());
|
|
|
|
llvm::GlobalValue *GV =
|
|
|
|
new llvm::GlobalVariable(AT, false,
|
|
|
|
llvm::GlobalValue::AppendingLinkage,
|
|
|
|
llvm::ConstantArray::get(AT, Used),
|
|
|
|
"llvm.used",
|
|
|
|
&CGM.getModule());
|
|
|
|
|
|
|
|
GV->setSection("llvm.metadata");
|
|
|
|
}
|
|
|
|
|
|
|
|
/* *** */
|
|
|
|
|
2008-08-12 08:12:39 +08:00
|
|
|
ObjCTypesHelper::ObjCTypesHelper(CodeGen::CodeGenModule &cgm)
|
|
|
|
: CGM(cgm),
|
|
|
|
CFStringType(0),
|
|
|
|
CFConstantStringClassReference(0),
|
2008-08-12 14:48:42 +08:00
|
|
|
MessageSendFn(0)
|
2008-08-12 08:12:39 +08:00
|
|
|
{
|
2008-08-12 14:48:42 +08:00
|
|
|
CodeGen::CodeGenTypes &Types = CGM.getTypes();
|
|
|
|
ASTContext &Ctx = CGM.getContext();
|
2008-08-13 11:21:16 +08:00
|
|
|
|
|
|
|
IntTy = Types.ConvertType(Ctx.IntTy);
|
2008-08-12 14:48:42 +08:00
|
|
|
LongTy = Types.ConvertType(Ctx.LongTy);
|
|
|
|
ObjectPtrTy = Types.ConvertType(Ctx.getObjCIdType());
|
|
|
|
SelectorPtrTy = Types.ConvertType(Ctx.getObjCSelType());
|
2008-08-13 11:21:16 +08:00
|
|
|
|
|
|
|
// FIXME: It would be nice to unify this with the opaque type, so
|
|
|
|
// that the IR comes out a bit cleaner.
|
|
|
|
const llvm::Type *T = Types.ConvertType(Ctx.getObjCProtoType());
|
|
|
|
ExternalProtocolPtrTy = llvm::PointerType::getUnqual(T);
|
2008-08-12 14:48:42 +08:00
|
|
|
|
|
|
|
SymtabTy = llvm::StructType::get(LongTy,
|
|
|
|
SelectorPtrTy,
|
|
|
|
Types.ConvertType(Ctx.ShortTy),
|
|
|
|
Types.ConvertType(Ctx.ShortTy),
|
|
|
|
NULL);
|
|
|
|
CGM.getModule().addTypeName("struct._objc_symtab", SymtabTy);
|
|
|
|
|
|
|
|
ModuleTy =
|
|
|
|
llvm::StructType::get(LongTy,
|
|
|
|
LongTy,
|
|
|
|
llvm::PointerType::getUnqual(llvm::Type::Int8Ty),
|
|
|
|
llvm::PointerType::getUnqual(SymtabTy),
|
|
|
|
NULL);
|
|
|
|
CGM.getModule().addTypeName("struct._objc_module", ModuleTy);
|
2008-08-13 11:21:16 +08:00
|
|
|
|
|
|
|
MethodDescriptionTy =
|
|
|
|
llvm::StructType::get(SelectorPtrTy,
|
|
|
|
llvm::PointerType::getUnqual(llvm::Type::Int8Ty),
|
|
|
|
NULL);
|
|
|
|
CGM.getModule().addTypeName("struct._objc_method_description",
|
|
|
|
MethodDescriptionTy);
|
|
|
|
|
|
|
|
MethodDescriptionListTy =
|
|
|
|
llvm::StructType::get(IntTy,
|
|
|
|
llvm::ArrayType::get(MethodDescriptionTy, 0),
|
|
|
|
NULL);
|
|
|
|
CGM.getModule().addTypeName("struct._objc_method_description_list",
|
|
|
|
MethodDescriptionListTy);
|
|
|
|
MethodDescriptionListPtrTy =
|
|
|
|
llvm::PointerType::getUnqual(MethodDescriptionListTy);
|
|
|
|
|
|
|
|
PropertyListTy = llvm::OpaqueType::get();
|
|
|
|
CGM.getModule().addTypeName("struct._objc_property_list",
|
|
|
|
PropertyListTy);
|
|
|
|
PropertyListPtrTy = llvm::PointerType::getUnqual(PropertyListTy);
|
|
|
|
|
|
|
|
// Protocol description structures
|
|
|
|
|
|
|
|
ProtocolExtensionTy =
|
|
|
|
llvm::StructType::get(Types.ConvertType(Ctx.IntTy),
|
|
|
|
llvm::PointerType::getUnqual(MethodDescriptionListTy),
|
|
|
|
llvm::PointerType::getUnqual(MethodDescriptionListTy),
|
|
|
|
PropertyListPtrTy,
|
|
|
|
NULL);
|
|
|
|
CGM.getModule().addTypeName("struct._objc_protocol_extension",
|
|
|
|
ProtocolExtensionTy);
|
|
|
|
ProtocolExtensionPtrTy = llvm::PointerType::getUnqual(ProtocolExtensionTy);
|
|
|
|
|
|
|
|
// Handle recursive construction of Protocl and ProtocolList types
|
|
|
|
|
|
|
|
llvm::PATypeHolder ProtocolTyHolder = llvm::OpaqueType::get();
|
|
|
|
llvm::PATypeHolder ProtocolListTyHolder = llvm::OpaqueType::get();
|
|
|
|
|
|
|
|
T = llvm::StructType::get(llvm::PointerType::getUnqual(ProtocolListTyHolder),
|
|
|
|
LongTy,
|
|
|
|
llvm::ArrayType::get(ProtocolTyHolder, 0),
|
|
|
|
NULL);
|
|
|
|
cast<llvm::OpaqueType>(ProtocolListTyHolder.get())->refineAbstractTypeTo(T);
|
|
|
|
|
|
|
|
T = llvm::StructType::get(llvm::PointerType::getUnqual(ProtocolExtensionTy),
|
|
|
|
llvm::PointerType::getUnqual(llvm::Type::Int8Ty),
|
|
|
|
llvm::PointerType::getUnqual(ProtocolListTyHolder),
|
|
|
|
MethodDescriptionListPtrTy,
|
|
|
|
MethodDescriptionListPtrTy,
|
|
|
|
NULL);
|
|
|
|
cast<llvm::OpaqueType>(ProtocolTyHolder.get())->refineAbstractTypeTo(T);
|
|
|
|
|
|
|
|
ProtocolListTy = cast<llvm::StructType>(ProtocolListTyHolder.get());
|
|
|
|
CGM.getModule().addTypeName("struct._objc_protocol_list",
|
|
|
|
ProtocolListTy);
|
|
|
|
ProtocolListPtrTy = llvm::PointerType::getUnqual(ProtocolListTy);
|
|
|
|
|
|
|
|
ProtocolTy = cast<llvm::StructType>(ProtocolTyHolder.get());
|
|
|
|
CGM.getModule().addTypeName("struct.__objc_protocol", ProtocolTy);
|
|
|
|
ProtocolPtrTy = llvm::PointerType::getUnqual(ProtocolTy);
|
2008-08-12 08:12:39 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
ObjCTypesHelper::~ObjCTypesHelper() {
|
|
|
|
}
|
|
|
|
|
|
|
|
const llvm::StructType *ObjCTypesHelper::getCFStringType() {
|
|
|
|
if (!CFStringType) {
|
|
|
|
CFStringType =
|
|
|
|
llvm::StructType::get(llvm::PointerType::getUnqual(llvm::Type::Int32Ty),
|
|
|
|
llvm::Type::Int32Ty,
|
|
|
|
llvm::PointerType::getUnqual(llvm::Type::Int8Ty),
|
|
|
|
LongTy,
|
|
|
|
NULL);
|
|
|
|
|
|
|
|
CGM.getModule().addTypeName("struct.__builtin_CFString", CFStringType);
|
|
|
|
}
|
|
|
|
|
|
|
|
return CFStringType;
|
|
|
|
}
|
|
|
|
|
|
|
|
llvm::Constant *ObjCTypesHelper::getCFConstantStringClassReference() {
|
|
|
|
if (!CFConstantStringClassReference) {
|
|
|
|
llvm::GlobalValue *GV =
|
|
|
|
new llvm::GlobalVariable(llvm::ArrayType::get(llvm::Type::Int32Ty, 0),
|
|
|
|
false,
|
|
|
|
llvm::GlobalValue::ExternalLinkage,
|
|
|
|
0, "__CFConstantStringClassReference",
|
|
|
|
&CGM.getModule());
|
|
|
|
|
|
|
|
// Decay to pointer.
|
|
|
|
CFConstantStringClassReference = getConstantGEP(GV, 0, 0);
|
|
|
|
}
|
|
|
|
|
|
|
|
return CFConstantStringClassReference;
|
|
|
|
}
|
|
|
|
|
2008-08-12 11:39:23 +08:00
|
|
|
llvm::Function *ObjCTypesHelper::getMessageSendFn() {
|
|
|
|
if (!MessageSendFn) {
|
|
|
|
std::vector<const llvm::Type*> Params;
|
|
|
|
Params.push_back(ObjectPtrTy);
|
|
|
|
Params.push_back(SelectorPtrTy);
|
|
|
|
MessageSendFn = llvm::Function::Create(llvm::FunctionType::get(ObjectPtrTy,
|
|
|
|
Params,
|
|
|
|
true),
|
|
|
|
llvm::Function::ExternalLinkage,
|
|
|
|
"objc_msgSend",
|
|
|
|
&CGM.getModule());
|
|
|
|
}
|
|
|
|
|
|
|
|
return MessageSendFn;
|
|
|
|
}
|
|
|
|
|
2008-08-12 08:12:39 +08:00
|
|
|
/* *** */
|
|
|
|
|
2008-08-13 11:21:16 +08:00
|
|
|
CodeGen::CGObjCRuntime *
|
|
|
|
CodeGen::CreateMacObjCRuntime(CodeGen::CodeGenModule &CGM) {
|
2008-08-11 10:45:11 +08:00
|
|
|
return new CGObjCMac(CGM);
|
|
|
|
}
|