2013-07-11 02:49:00 +08:00
|
|
|
//===- ARCRuntimeEntryPoints.h - ObjC ARC Optimization --*- C++ -*---------===//
|
2013-07-06 09:39:18 +08:00
|
|
|
//
|
|
|
|
// The LLVM Compiler Infrastructure
|
|
|
|
//
|
|
|
|
// This file is distributed under the University of Illinois Open Source
|
|
|
|
// License. See LICENSE.TXT for details.
|
|
|
|
//
|
|
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
/// \file
|
|
|
|
/// This file contains a class ARCRuntimeEntryPoints for use in
|
|
|
|
/// creating/managing references to entry points to the arc objective c runtime.
|
|
|
|
///
|
|
|
|
/// WARNING: This file knows about certain library functions. It recognizes them
|
|
|
|
/// by name, and hardwires knowledge of their semantics.
|
|
|
|
///
|
|
|
|
/// WARNING: This file knows about how certain Objective-C library functions are
|
|
|
|
/// used. Naive LLVM IR transformations which would otherwise be
|
|
|
|
/// behavior-preserving may break these assumptions.
|
|
|
|
///
|
|
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
|
2014-08-14 00:26:38 +08:00
|
|
|
#ifndef LLVM_LIB_TRANSFORMS_OBJCARC_ARCRUNTIMEENTRYPOINTS_H
|
|
|
|
#define LLVM_LIB_TRANSFORMS_OBJCARC_ARCRUNTIMEENTRYPOINTS_H
|
2013-07-06 09:39:18 +08:00
|
|
|
|
|
|
|
#include "ObjCARC.h"
|
|
|
|
|
|
|
|
namespace llvm {
|
|
|
|
namespace objcarc {
|
|
|
|
|
2015-03-16 15:02:24 +08:00
|
|
|
enum class ARCRuntimeEntryPointKind {
|
|
|
|
AutoreleaseRV,
|
|
|
|
Release,
|
|
|
|
Retain,
|
|
|
|
RetainBlock,
|
|
|
|
Autorelease,
|
|
|
|
StoreStrong,
|
|
|
|
RetainRV,
|
|
|
|
RetainAutorelease,
|
|
|
|
RetainAutoreleaseRV,
|
|
|
|
};
|
|
|
|
|
2013-07-06 09:39:18 +08:00
|
|
|
/// Declarations for ObjC runtime functions and constants. These are initialized
|
|
|
|
/// lazily to avoid cluttering up the Module with unused declarations.
|
2013-09-12 02:05:11 +08:00
|
|
|
class ARCRuntimeEntryPoints {
|
2013-07-06 09:39:18 +08:00
|
|
|
public:
|
2014-04-28 12:05:08 +08:00
|
|
|
ARCRuntimeEntryPoints() : TheModule(nullptr),
|
|
|
|
AutoreleaseRV(nullptr),
|
|
|
|
Release(nullptr),
|
|
|
|
Retain(nullptr),
|
|
|
|
RetainBlock(nullptr),
|
|
|
|
Autorelease(nullptr),
|
|
|
|
StoreStrong(nullptr),
|
|
|
|
RetainRV(nullptr),
|
|
|
|
RetainAutorelease(nullptr),
|
|
|
|
RetainAutoreleaseRV(nullptr) { }
|
2013-07-06 09:39:18 +08:00
|
|
|
|
2015-03-16 15:02:27 +08:00
|
|
|
void init(Module *M) {
|
2013-07-06 09:57:32 +08:00
|
|
|
TheModule = M;
|
2014-04-28 12:05:08 +08:00
|
|
|
AutoreleaseRV = nullptr;
|
|
|
|
Release = nullptr;
|
|
|
|
Retain = nullptr;
|
|
|
|
RetainBlock = nullptr;
|
|
|
|
Autorelease = nullptr;
|
|
|
|
StoreStrong = nullptr;
|
|
|
|
RetainRV = nullptr;
|
|
|
|
RetainAutorelease = nullptr;
|
|
|
|
RetainAutoreleaseRV = nullptr;
|
2013-07-06 09:39:18 +08:00
|
|
|
}
|
|
|
|
|
2015-03-16 15:02:24 +08:00
|
|
|
Constant *get(ARCRuntimeEntryPointKind kind) {
|
2014-04-28 12:05:08 +08:00
|
|
|
assert(TheModule != nullptr && "Not initialized.");
|
2013-07-06 09:39:18 +08:00
|
|
|
|
2015-03-16 15:02:24 +08:00
|
|
|
switch (kind) {
|
|
|
|
case ARCRuntimeEntryPointKind::AutoreleaseRV:
|
2013-07-06 09:39:18 +08:00
|
|
|
return getI8XRetI8XEntryPoint(AutoreleaseRV,
|
|
|
|
"objc_autoreleaseReturnValue", true);
|
2015-03-16 15:02:24 +08:00
|
|
|
case ARCRuntimeEntryPointKind::Release:
|
2013-07-06 09:39:18 +08:00
|
|
|
return getVoidRetI8XEntryPoint(Release, "objc_release");
|
2015-03-16 15:02:24 +08:00
|
|
|
case ARCRuntimeEntryPointKind::Retain:
|
2013-07-06 09:39:18 +08:00
|
|
|
return getI8XRetI8XEntryPoint(Retain, "objc_retain", true);
|
2015-03-16 15:02:24 +08:00
|
|
|
case ARCRuntimeEntryPointKind::RetainBlock:
|
2013-07-06 09:39:18 +08:00
|
|
|
return getI8XRetI8XEntryPoint(RetainBlock, "objc_retainBlock", false);
|
2015-03-16 15:02:24 +08:00
|
|
|
case ARCRuntimeEntryPointKind::Autorelease:
|
2013-07-06 09:39:18 +08:00
|
|
|
return getI8XRetI8XEntryPoint(Autorelease, "objc_autorelease", true);
|
2015-03-16 15:02:24 +08:00
|
|
|
case ARCRuntimeEntryPointKind::StoreStrong:
|
2013-07-06 09:39:18 +08:00
|
|
|
return getI8XRetI8XXI8XEntryPoint(StoreStrong, "objc_storeStrong");
|
2015-03-16 15:02:24 +08:00
|
|
|
case ARCRuntimeEntryPointKind::RetainRV:
|
2013-07-06 10:18:56 +08:00
|
|
|
return getI8XRetI8XEntryPoint(RetainRV,
|
|
|
|
"objc_retainAutoreleasedReturnValue", true);
|
2015-03-16 15:02:24 +08:00
|
|
|
case ARCRuntimeEntryPointKind::RetainAutorelease:
|
2013-07-06 09:39:18 +08:00
|
|
|
return getI8XRetI8XEntryPoint(RetainAutorelease, "objc_retainAutorelease",
|
|
|
|
true);
|
2015-03-16 15:02:24 +08:00
|
|
|
case ARCRuntimeEntryPointKind::RetainAutoreleaseRV:
|
2013-07-06 09:39:18 +08:00
|
|
|
return getI8XRetI8XEntryPoint(RetainAutoreleaseRV,
|
|
|
|
"objc_retainAutoreleaseReturnValue", true);
|
|
|
|
}
|
2013-07-06 10:18:56 +08:00
|
|
|
|
|
|
|
llvm_unreachable("Switch should be a covered switch.");
|
2013-07-06 09:39:18 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
private:
|
|
|
|
/// Cached reference to the module which we will insert declarations into.
|
2013-07-06 09:57:32 +08:00
|
|
|
Module *TheModule;
|
2013-07-06 09:39:18 +08:00
|
|
|
|
|
|
|
/// Declaration for ObjC runtime function objc_autoreleaseReturnValue.
|
|
|
|
Constant *AutoreleaseRV;
|
|
|
|
/// Declaration for ObjC runtime function objc_release.
|
|
|
|
Constant *Release;
|
|
|
|
/// Declaration for ObjC runtime function objc_retain.
|
|
|
|
Constant *Retain;
|
|
|
|
/// Declaration for ObjC runtime function objc_retainBlock.
|
|
|
|
Constant *RetainBlock;
|
|
|
|
/// Declaration for ObjC runtime function objc_autorelease.
|
|
|
|
Constant *Autorelease;
|
|
|
|
/// Declaration for objc_storeStrong().
|
|
|
|
Constant *StoreStrong;
|
|
|
|
/// Declaration for objc_retainAutoreleasedReturnValue().
|
|
|
|
Constant *RetainRV;
|
|
|
|
/// Declaration for objc_retainAutorelease().
|
|
|
|
Constant *RetainAutorelease;
|
|
|
|
/// Declaration for objc_retainAutoreleaseReturnValue().
|
|
|
|
Constant *RetainAutoreleaseRV;
|
|
|
|
|
|
|
|
Constant *getVoidRetI8XEntryPoint(Constant *&Decl,
|
|
|
|
const char *Name) {
|
|
|
|
if (Decl)
|
|
|
|
return Decl;
|
|
|
|
|
2013-07-06 09:57:32 +08:00
|
|
|
LLVMContext &C = TheModule->getContext();
|
2013-07-06 09:39:18 +08:00
|
|
|
Type *Params[] = { PointerType::getUnqual(Type::getInt8Ty(C)) };
|
|
|
|
AttributeSet Attr =
|
2013-07-06 09:57:32 +08:00
|
|
|
AttributeSet().addAttribute(C, AttributeSet::FunctionIndex,
|
2013-07-06 09:39:18 +08:00
|
|
|
Attribute::NoUnwind);
|
|
|
|
FunctionType *Fty = FunctionType::get(Type::getVoidTy(C), Params,
|
|
|
|
/*isVarArg=*/false);
|
2013-07-06 09:57:32 +08:00
|
|
|
return Decl = TheModule->getOrInsertFunction(Name, Fty, Attr);
|
2013-07-06 09:39:18 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
Constant *getI8XRetI8XEntryPoint(Constant *& Decl,
|
|
|
|
const char *Name,
|
|
|
|
bool NoUnwind = false) {
|
|
|
|
if (Decl)
|
|
|
|
return Decl;
|
|
|
|
|
2013-07-06 09:57:32 +08:00
|
|
|
LLVMContext &C = TheModule->getContext();
|
2013-07-06 09:39:18 +08:00
|
|
|
Type *I8X = PointerType::getUnqual(Type::getInt8Ty(C));
|
|
|
|
Type *Params[] = { I8X };
|
|
|
|
FunctionType *Fty = FunctionType::get(I8X, Params, /*isVarArg=*/false);
|
|
|
|
AttributeSet Attr = AttributeSet();
|
|
|
|
|
|
|
|
if (NoUnwind)
|
2013-07-06 09:57:32 +08:00
|
|
|
Attr = Attr.addAttribute(C, AttributeSet::FunctionIndex,
|
2013-07-06 09:39:18 +08:00
|
|
|
Attribute::NoUnwind);
|
|
|
|
|
2013-07-06 09:57:32 +08:00
|
|
|
return Decl = TheModule->getOrInsertFunction(Name, Fty, Attr);
|
2013-07-06 09:39:18 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
Constant *getI8XRetI8XXI8XEntryPoint(Constant *&Decl,
|
|
|
|
const char *Name) {
|
|
|
|
if (Decl)
|
|
|
|
return Decl;
|
|
|
|
|
2013-07-06 09:57:32 +08:00
|
|
|
LLVMContext &C = TheModule->getContext();
|
2013-07-06 09:39:18 +08:00
|
|
|
Type *I8X = PointerType::getUnqual(Type::getInt8Ty(C));
|
|
|
|
Type *I8XX = PointerType::getUnqual(I8X);
|
|
|
|
Type *Params[] = { I8XX, I8X };
|
|
|
|
|
|
|
|
AttributeSet Attr =
|
2013-07-06 09:57:32 +08:00
|
|
|
AttributeSet().addAttribute(C, AttributeSet::FunctionIndex,
|
2013-07-06 09:39:18 +08:00
|
|
|
Attribute::NoUnwind);
|
2013-07-06 09:57:32 +08:00
|
|
|
Attr = Attr.addAttribute(C, 1, Attribute::NoCapture);
|
2013-07-06 09:39:18 +08:00
|
|
|
|
|
|
|
FunctionType *Fty = FunctionType::get(Type::getVoidTy(C), Params,
|
|
|
|
/*isVarArg=*/false);
|
|
|
|
|
2013-07-06 09:57:32 +08:00
|
|
|
return Decl = TheModule->getOrInsertFunction(Name, Fty, Attr);
|
2013-07-06 09:39:18 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
}; // class ARCRuntimeEntryPoints
|
|
|
|
|
|
|
|
} // namespace objcarc
|
|
|
|
} // namespace llvm
|
|
|
|
|
2014-08-14 00:26:38 +08:00
|
|
|
#endif
|