mirror of
https://github.com/RPCS3/llvm-mirror.git
synced 2024-11-22 18:54:02 +01:00
[IRBuilder] Add gc.statepoint related methods to IRBuilder
Summary: This adds some more routines to `IRBuilder` around creating calls and invokes to `gc.statepoint`. These will be used later. Reviewers: reames, swaroop.sridhar Subscribers: sanjoy, llvm-commits Differential Revision: http://reviews.llvm.org/D13371 llvm-svn: 249596
This commit is contained in:
parent
a61c0e284b
commit
f6945cd657
@ -25,6 +25,7 @@
|
|||||||
#include "llvm/IR/GlobalVariable.h"
|
#include "llvm/IR/GlobalVariable.h"
|
||||||
#include "llvm/IR/Instructions.h"
|
#include "llvm/IR/Instructions.h"
|
||||||
#include "llvm/IR/Intrinsics.h"
|
#include "llvm/IR/Intrinsics.h"
|
||||||
|
#include "llvm/IR/Statepoint.h"
|
||||||
#include "llvm/IR/LLVMContext.h"
|
#include "llvm/IR/LLVMContext.h"
|
||||||
#include "llvm/IR/Operator.h"
|
#include "llvm/IR/Operator.h"
|
||||||
#include "llvm/IR/ValueHandle.h"
|
#include "llvm/IR/ValueHandle.h"
|
||||||
@ -445,6 +446,16 @@ public:
|
|||||||
ArrayRef<Value *> GCArgs,
|
ArrayRef<Value *> GCArgs,
|
||||||
const Twine &Name = "");
|
const Twine &Name = "");
|
||||||
|
|
||||||
|
/// \brief Create a call to the experimental.gc.statepoint intrinsic to
|
||||||
|
/// start a new statepoint sequence.
|
||||||
|
CallInst *CreateGCStatepointCall(uint64_t ID, uint32_t NumPatchBytes,
|
||||||
|
Value *ActualCallee, StatepointFlags Flags,
|
||||||
|
ArrayRef<Use> CallArgs,
|
||||||
|
ArrayRef<Use> TransitionArgs,
|
||||||
|
ArrayRef<Use> DeoptArgs,
|
||||||
|
ArrayRef<Value *> GCArgs,
|
||||||
|
const Twine &Name = "");
|
||||||
|
|
||||||
// \brief Conveninence function for the common case when CallArgs are filled
|
// \brief Conveninence function for the common case when CallArgs are filled
|
||||||
// in using makeArrayRef(CS.arg_begin(), CS.arg_end()); Use needs to be
|
// in using makeArrayRef(CS.arg_begin(), CS.arg_end()); Use needs to be
|
||||||
// .get()'ed to get the Value pointer.
|
// .get()'ed to get the Value pointer.
|
||||||
@ -463,6 +474,15 @@ public:
|
|||||||
ArrayRef<Value *> DeoptArgs,
|
ArrayRef<Value *> DeoptArgs,
|
||||||
ArrayRef<Value *> GCArgs, const Twine &Name = "");
|
ArrayRef<Value *> GCArgs, const Twine &Name = "");
|
||||||
|
|
||||||
|
/// brief Create an invoke to the experimental.gc.statepoint intrinsic to
|
||||||
|
/// start a new statepoint sequence.
|
||||||
|
InvokeInst *CreateGCStatepointInvoke(
|
||||||
|
uint64_t ID, uint32_t NumPatchBytes, Value *ActualInvokee,
|
||||||
|
BasicBlock *NormalDest, BasicBlock *UnwindDest, StatepointFlags Flags,
|
||||||
|
ArrayRef<Use> InvokeArgs, ArrayRef<Use> TransitionArgs,
|
||||||
|
ArrayRef<Use> DeoptArgs, ArrayRef<Value *> GCArgs,
|
||||||
|
const Twine &Name = "");
|
||||||
|
|
||||||
// Conveninence function for the common case when CallArgs are filled in using
|
// Conveninence function for the common case when CallArgs are filled in using
|
||||||
// makeArrayRef(CS.arg_begin(), CS.arg_end()); Use needs to be .get()'ed to
|
// makeArrayRef(CS.arg_begin(), CS.arg_end()); Use needs to be .get()'ed to
|
||||||
// get the Value *.
|
// get the Value *.
|
||||||
|
@ -247,18 +247,21 @@ CallInst *IRBuilderBase::CreateMaskedIntrinsic(Intrinsic::ID Id,
|
|||||||
return createCallHelper(TheFn, Ops, this, Name);
|
return createCallHelper(TheFn, Ops, this, Name);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
template <typename T0, typename T1, typename T2, typename T3>
|
||||||
static std::vector<Value *>
|
static std::vector<Value *>
|
||||||
getStatepointArgs(IRBuilderBase &B, uint64_t ID, uint32_t NumPatchBytes,
|
getStatepointArgs(IRBuilderBase &B, uint64_t ID, uint32_t NumPatchBytes,
|
||||||
Value *ActualCallee, ArrayRef<Value *> CallArgs,
|
Value *ActualCallee, StatepointFlags Flags,
|
||||||
ArrayRef<Value *> DeoptArgs, ArrayRef<Value *> GCArgs) {
|
ArrayRef<T0> CallArgs, ArrayRef<T1> TransitionArgs,
|
||||||
|
ArrayRef<T2> DeoptArgs, ArrayRef<T3> GCArgs) {
|
||||||
std::vector<Value *> Args;
|
std::vector<Value *> Args;
|
||||||
Args.push_back(B.getInt64(ID));
|
Args.push_back(B.getInt64(ID));
|
||||||
Args.push_back(B.getInt32(NumPatchBytes));
|
Args.push_back(B.getInt32(NumPatchBytes));
|
||||||
Args.push_back(ActualCallee);
|
Args.push_back(ActualCallee);
|
||||||
Args.push_back(B.getInt32(CallArgs.size()));
|
Args.push_back(B.getInt32(CallArgs.size()));
|
||||||
Args.push_back(B.getInt32((unsigned)StatepointFlags::None));
|
Args.push_back(B.getInt32((unsigned)Flags));
|
||||||
Args.insert(Args.end(), CallArgs.begin(), CallArgs.end());
|
Args.insert(Args.end(), CallArgs.begin(), CallArgs.end());
|
||||||
Args.push_back(B.getInt32(0 /* no transition args */));
|
Args.push_back(B.getInt32(TransitionArgs.size()));
|
||||||
|
Args.insert(Args.end(), TransitionArgs.begin(), TransitionArgs.end());
|
||||||
Args.push_back(B.getInt32(DeoptArgs.size()));
|
Args.push_back(B.getInt32(DeoptArgs.size()));
|
||||||
Args.insert(Args.end(), DeoptArgs.begin(), DeoptArgs.end());
|
Args.insert(Args.end(), DeoptArgs.begin(), DeoptArgs.end());
|
||||||
Args.insert(Args.end(), GCArgs.begin(), GCArgs.end());
|
Args.insert(Args.end(), GCArgs.begin(), GCArgs.end());
|
||||||
@ -266,36 +269,78 @@ getStatepointArgs(IRBuilderBase &B, uint64_t ID, uint32_t NumPatchBytes,
|
|||||||
return Args;
|
return Args;
|
||||||
}
|
}
|
||||||
|
|
||||||
CallInst *IRBuilderBase::CreateGCStatepointCall(
|
template <typename T0, typename T1, typename T2, typename T3>
|
||||||
uint64_t ID, uint32_t NumPatchBytes, Value *ActualCallee,
|
static CallInst *CreateGCStatepointCallCommon(
|
||||||
ArrayRef<Value *> CallArgs, ArrayRef<Value *> DeoptArgs,
|
IRBuilderBase *Builder, uint64_t ID, uint32_t NumPatchBytes,
|
||||||
ArrayRef<Value *> GCArgs, const Twine &Name) {
|
Value *ActualCallee, StatepointFlags Flags, ArrayRef<T0> CallArgs,
|
||||||
|
ArrayRef<T1> TransitionArgs, ArrayRef<T2> DeoptArgs, ArrayRef<T3> GCArgs,
|
||||||
|
const Twine &Name) {
|
||||||
// Extract out the type of the callee.
|
// Extract out the type of the callee.
|
||||||
PointerType *FuncPtrType = cast<PointerType>(ActualCallee->getType());
|
PointerType *FuncPtrType = cast<PointerType>(ActualCallee->getType());
|
||||||
assert(isa<FunctionType>(FuncPtrType->getElementType()) &&
|
assert(isa<FunctionType>(FuncPtrType->getElementType()) &&
|
||||||
"actual callee must be a callable value");
|
"actual callee must be a callable value");
|
||||||
|
|
||||||
Module *M = BB->getParent()->getParent();
|
Module *M = Builder->GetInsertBlock()->getParent()->getParent();
|
||||||
// Fill in the one generic type'd argument (the function is also vararg)
|
// Fill in the one generic type'd argument (the function is also vararg)
|
||||||
Type *ArgTypes[] = { FuncPtrType };
|
Type *ArgTypes[] = { FuncPtrType };
|
||||||
Function *FnStatepoint =
|
Function *FnStatepoint =
|
||||||
Intrinsic::getDeclaration(M, Intrinsic::experimental_gc_statepoint,
|
Intrinsic::getDeclaration(M, Intrinsic::experimental_gc_statepoint,
|
||||||
ArgTypes);
|
ArgTypes);
|
||||||
|
|
||||||
std::vector<llvm::Value *> Args = getStatepointArgs(
|
std::vector<llvm::Value *> Args =
|
||||||
*this, ID, NumPatchBytes, ActualCallee, CallArgs, DeoptArgs, GCArgs);
|
getStatepointArgs(*Builder, ID, NumPatchBytes, ActualCallee, Flags,
|
||||||
return createCallHelper(FnStatepoint, Args, this, Name);
|
CallArgs, TransitionArgs, DeoptArgs, GCArgs);
|
||||||
|
return createCallHelper(FnStatepoint, Args, Builder, Name);
|
||||||
|
}
|
||||||
|
|
||||||
|
CallInst *IRBuilderBase::CreateGCStatepointCall(
|
||||||
|
uint64_t ID, uint32_t NumPatchBytes, Value *ActualCallee,
|
||||||
|
ArrayRef<Value *> CallArgs, ArrayRef<Value *> DeoptArgs,
|
||||||
|
ArrayRef<Value *> GCArgs, const Twine &Name) {
|
||||||
|
return CreateGCStatepointCallCommon<Value *, Value *, Value *, Value *>(
|
||||||
|
this, ID, NumPatchBytes, ActualCallee, StatepointFlags::None, CallArgs,
|
||||||
|
None /* No Transition Args */, DeoptArgs, GCArgs, Name);
|
||||||
|
}
|
||||||
|
|
||||||
|
CallInst *IRBuilderBase::CreateGCStatepointCall(
|
||||||
|
uint64_t ID, uint32_t NumPatchBytes, Value *ActualCallee,
|
||||||
|
StatepointFlags Flags, ArrayRef<Use> CallArgs, ArrayRef<Use> TransitionArgs,
|
||||||
|
ArrayRef<Use> DeoptArgs, ArrayRef<Value *> GCArgs, const Twine &Name) {
|
||||||
|
return CreateGCStatepointCallCommon<Use, Use, Use, Value *>(
|
||||||
|
this, ID, NumPatchBytes, ActualCallee, Flags, CallArgs, TransitionArgs,
|
||||||
|
DeoptArgs, GCArgs, Name);
|
||||||
}
|
}
|
||||||
|
|
||||||
CallInst *IRBuilderBase::CreateGCStatepointCall(
|
CallInst *IRBuilderBase::CreateGCStatepointCall(
|
||||||
uint64_t ID, uint32_t NumPatchBytes, Value *ActualCallee,
|
uint64_t ID, uint32_t NumPatchBytes, Value *ActualCallee,
|
||||||
ArrayRef<Use> CallArgs, ArrayRef<Value *> DeoptArgs,
|
ArrayRef<Use> CallArgs, ArrayRef<Value *> DeoptArgs,
|
||||||
ArrayRef<Value *> GCArgs, const Twine &Name) {
|
ArrayRef<Value *> GCArgs, const Twine &Name) {
|
||||||
std::vector<Value *> VCallArgs;
|
return CreateGCStatepointCallCommon<Use, Value *, Value *, Value *>(
|
||||||
for (auto &U : CallArgs)
|
this, ID, NumPatchBytes, ActualCallee, StatepointFlags::None, CallArgs,
|
||||||
VCallArgs.push_back(U.get());
|
None, DeoptArgs, GCArgs, Name);
|
||||||
return CreateGCStatepointCall(ID, NumPatchBytes, ActualCallee, VCallArgs,
|
}
|
||||||
DeoptArgs, GCArgs, Name);
|
|
||||||
|
template <typename T0, typename T1, typename T2, typename T3>
|
||||||
|
static InvokeInst *CreateGCStatepointInvokeCommon(
|
||||||
|
IRBuilderBase *Builder, uint64_t ID, uint32_t NumPatchBytes,
|
||||||
|
Value *ActualInvokee, BasicBlock *NormalDest, BasicBlock *UnwindDest,
|
||||||
|
StatepointFlags Flags, ArrayRef<T0> InvokeArgs, ArrayRef<T1> TransitionArgs,
|
||||||
|
ArrayRef<T2> DeoptArgs, ArrayRef<T3> GCArgs, const Twine &Name) {
|
||||||
|
// Extract out the type of the callee.
|
||||||
|
PointerType *FuncPtrType = cast<PointerType>(ActualInvokee->getType());
|
||||||
|
assert(isa<FunctionType>(FuncPtrType->getElementType()) &&
|
||||||
|
"actual callee must be a callable value");
|
||||||
|
|
||||||
|
Module *M = Builder->GetInsertBlock()->getParent()->getParent();
|
||||||
|
// Fill in the one generic type'd argument (the function is also vararg)
|
||||||
|
Function *FnStatepoint = Intrinsic::getDeclaration(
|
||||||
|
M, Intrinsic::experimental_gc_statepoint, {FuncPtrType});
|
||||||
|
|
||||||
|
std::vector<llvm::Value *> Args =
|
||||||
|
getStatepointArgs(*Builder, ID, NumPatchBytes, ActualInvokee, Flags,
|
||||||
|
InvokeArgs, TransitionArgs, DeoptArgs, GCArgs);
|
||||||
|
return createInvokeHelper(FnStatepoint, NormalDest, UnwindDest, Args, Builder,
|
||||||
|
Name);
|
||||||
}
|
}
|
||||||
|
|
||||||
InvokeInst *IRBuilderBase::CreateGCStatepointInvoke(
|
InvokeInst *IRBuilderBase::CreateGCStatepointInvoke(
|
||||||
@ -303,32 +348,29 @@ InvokeInst *IRBuilderBase::CreateGCStatepointInvoke(
|
|||||||
BasicBlock *NormalDest, BasicBlock *UnwindDest,
|
BasicBlock *NormalDest, BasicBlock *UnwindDest,
|
||||||
ArrayRef<Value *> InvokeArgs, ArrayRef<Value *> DeoptArgs,
|
ArrayRef<Value *> InvokeArgs, ArrayRef<Value *> DeoptArgs,
|
||||||
ArrayRef<Value *> GCArgs, const Twine &Name) {
|
ArrayRef<Value *> GCArgs, const Twine &Name) {
|
||||||
// Extract out the type of the callee.
|
return CreateGCStatepointInvokeCommon<Value *, Value *, Value *, Value *>(
|
||||||
PointerType *FuncPtrType = cast<PointerType>(ActualInvokee->getType());
|
this, ID, NumPatchBytes, ActualInvokee, NormalDest, UnwindDest,
|
||||||
assert(isa<FunctionType>(FuncPtrType->getElementType()) &&
|
StatepointFlags::None, InvokeArgs, None /* No Transition Args*/,
|
||||||
"actual callee must be a callable value");
|
DeoptArgs, GCArgs, Name);
|
||||||
|
}
|
||||||
|
|
||||||
Module *M = BB->getParent()->getParent();
|
InvokeInst *IRBuilderBase::CreateGCStatepointInvoke(
|
||||||
// Fill in the one generic type'd argument (the function is also vararg)
|
uint64_t ID, uint32_t NumPatchBytes, Value *ActualInvokee,
|
||||||
Function *FnStatepoint = Intrinsic::getDeclaration(
|
BasicBlock *NormalDest, BasicBlock *UnwindDest, StatepointFlags Flags,
|
||||||
M, Intrinsic::experimental_gc_statepoint, {FuncPtrType});
|
ArrayRef<Use> InvokeArgs, ArrayRef<Use> TransitionArgs,
|
||||||
|
ArrayRef<Use> DeoptArgs, ArrayRef<Value *> GCArgs, const Twine &Name) {
|
||||||
std::vector<llvm::Value *> Args = getStatepointArgs(
|
return CreateGCStatepointInvokeCommon<Use, Use, Use, Value *>(
|
||||||
*this, ID, NumPatchBytes, ActualInvokee, InvokeArgs, DeoptArgs, GCArgs);
|
this, ID, NumPatchBytes, ActualInvokee, NormalDest, UnwindDest, Flags,
|
||||||
return createInvokeHelper(FnStatepoint, NormalDest, UnwindDest, Args, this,
|
InvokeArgs, TransitionArgs, DeoptArgs, GCArgs, Name);
|
||||||
Name);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
InvokeInst *IRBuilderBase::CreateGCStatepointInvoke(
|
InvokeInst *IRBuilderBase::CreateGCStatepointInvoke(
|
||||||
uint64_t ID, uint32_t NumPatchBytes, Value *ActualInvokee,
|
uint64_t ID, uint32_t NumPatchBytes, Value *ActualInvokee,
|
||||||
BasicBlock *NormalDest, BasicBlock *UnwindDest, ArrayRef<Use> InvokeArgs,
|
BasicBlock *NormalDest, BasicBlock *UnwindDest, ArrayRef<Use> InvokeArgs,
|
||||||
ArrayRef<Value *> DeoptArgs, ArrayRef<Value *> GCArgs, const Twine &Name) {
|
ArrayRef<Value *> DeoptArgs, ArrayRef<Value *> GCArgs, const Twine &Name) {
|
||||||
std::vector<Value *> VCallArgs;
|
return CreateGCStatepointInvokeCommon<Use, Value *, Value *, Value *>(
|
||||||
for (auto &U : InvokeArgs)
|
this, ID, NumPatchBytes, ActualInvokee, NormalDest, UnwindDest,
|
||||||
VCallArgs.push_back(U.get());
|
StatepointFlags::None, InvokeArgs, None, DeoptArgs, GCArgs, Name);
|
||||||
return CreateGCStatepointInvoke(ID, NumPatchBytes, ActualInvokee, NormalDest,
|
|
||||||
UnwindDest, VCallArgs, DeoptArgs, GCArgs,
|
|
||||||
Name);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
CallInst *IRBuilderBase::CreateGCResult(Instruction *Statepoint,
|
CallInst *IRBuilderBase::CreateGCResult(Instruction *Statepoint,
|
||||||
|
Loading…
Reference in New Issue
Block a user