Use ImmutableCallSite for statepoint verification.

Patch by: Igor Laevsky

"This change generalizes statepoint verification to use ImmutableCallSite instead of CallInst. This will allow to easily implement invoke statepoint verification (in a following change)."

Differential Revision: http://reviews.llvm.org/D7308



git-svn-id: https://llvm.org/svn/llvm-project/llvm/trunk@228064 91177308-0d34-0410-b5e6-96231b3b80d8
This commit is contained in:
Philip Reames
2015-02-03 23:18:47 +00:00
parent 6daac09e80
commit c81fa9543f

View File

@@ -370,7 +370,7 @@ private:
const Value *V); const Value *V);
void VerifyConstantExprBitcastType(const ConstantExpr *CE); void VerifyConstantExprBitcastType(const ConstantExpr *CE);
void VerifyStatepoint(CallInst &CI); void VerifyStatepoint(ImmutableCallSite CS);
}; };
class DebugInfoVerifier : public VerifierSupport { class DebugInfoVerifier : public VerifierSupport {
public: public:
@@ -1047,26 +1047,26 @@ bool Verifier::VerifyAttributeCount(AttributeSet Attrs, unsigned Params) {
} }
/// \brief Verify that statepoint intrinsic is well formed. /// \brief Verify that statepoint intrinsic is well formed.
void Verifier::VerifyStatepoint(CallInst &CI) { void Verifier::VerifyStatepoint(ImmutableCallSite CS) {
assert(CI.getCalledFunction() && assert(CS.getCalledFunction() &&
CI.getCalledFunction()->getIntrinsicID() == CS.getCalledFunction()->getIntrinsicID() ==
Intrinsic::experimental_gc_statepoint); Intrinsic::experimental_gc_statepoint);
Assert1(!CI.doesNotAccessMemory() && const Instruction &CI = *CS.getInstruction();
!CI.onlyReadsMemory(),
Assert1(!CS.doesNotAccessMemory() &&
!CS.onlyReadsMemory(),
"gc.statepoint must read and write memory to preserve " "gc.statepoint must read and write memory to preserve "
"reordering restrictions required by safepoint semantics", &CI); "reordering restrictions required by safepoint semantics", &CI);
Assert1(!CI.isInlineAsm(),
"gc.statepoint support for inline assembly unimplemented", &CI);
const Value *Target = CI.getArgOperand(0); const Value *Target = CS.getArgument(0);
const PointerType *PT = dyn_cast<PointerType>(Target->getType()); const PointerType *PT = dyn_cast<PointerType>(Target->getType());
Assert2(PT && PT->getElementType()->isFunctionTy(), Assert2(PT && PT->getElementType()->isFunctionTy(),
"gc.statepoint callee must be of function pointer type", "gc.statepoint callee must be of function pointer type",
&CI, Target); &CI, Target);
FunctionType *TargetFuncType = cast<FunctionType>(PT->getElementType()); FunctionType *TargetFuncType = cast<FunctionType>(PT->getElementType());
const Value *NumCallArgsV = CI.getArgOperand(1); const Value *NumCallArgsV = CS.getArgument(1);
Assert1(isa<ConstantInt>(NumCallArgsV), Assert1(isa<ConstantInt>(NumCallArgsV),
"gc.statepoint number of arguments to underlying call " "gc.statepoint number of arguments to underlying call "
"must be constant integer", &CI); "must be constant integer", &CI);
@@ -1087,7 +1087,7 @@ void Verifier::VerifyStatepoint(CallInst &CI) {
Assert1(NumCallArgs == NumParams, Assert1(NumCallArgs == NumParams,
"gc.statepoint mismatch in number of call args", &CI); "gc.statepoint mismatch in number of call args", &CI);
const Value *Unused = CI.getArgOperand(2); const Value *Unused = CS.getArgument(2);
Assert1(isa<ConstantInt>(Unused) && Assert1(isa<ConstantInt>(Unused) &&
cast<ConstantInt>(Unused)->isNullValue(), cast<ConstantInt>(Unused)->isNullValue(),
"gc.statepoint parameter #3 must be zero", &CI); "gc.statepoint parameter #3 must be zero", &CI);
@@ -1096,13 +1096,13 @@ void Verifier::VerifyStatepoint(CallInst &CI) {
// the type of the wrapped callee. // the type of the wrapped callee.
for (int i = 0; i < NumParams; i++) { for (int i = 0; i < NumParams; i++) {
Type *ParamType = TargetFuncType->getParamType(i); Type *ParamType = TargetFuncType->getParamType(i);
Type *ArgType = CI.getArgOperand(3+i)->getType(); Type *ArgType = CS.getArgument(3+i)->getType();
Assert1(ArgType == ParamType, Assert1(ArgType == ParamType,
"gc.statepoint call argument does not match wrapped " "gc.statepoint call argument does not match wrapped "
"function type", &CI); "function type", &CI);
} }
const int EndCallArgsInx = 2+NumCallArgs; const int EndCallArgsInx = 2+NumCallArgs;
const Value *NumDeoptArgsV = CI.getArgOperand(EndCallArgsInx+1); const Value *NumDeoptArgsV = CS.getArgument(EndCallArgsInx+1);
Assert1(isa<ConstantInt>(NumDeoptArgsV), Assert1(isa<ConstantInt>(NumDeoptArgsV),
"gc.statepoint number of deoptimization arguments " "gc.statepoint number of deoptimization arguments "
"must be constant integer", &CI); "must be constant integer", &CI);
@@ -1111,13 +1111,13 @@ void Verifier::VerifyStatepoint(CallInst &CI) {
"gc.statepoint number of deoptimization arguments " "gc.statepoint number of deoptimization arguments "
"must be positive", &CI); "must be positive", &CI);
Assert1(4 + NumCallArgs + NumDeoptArgs <= (int)CI.getNumArgOperands(), Assert1(4 + NumCallArgs + NumDeoptArgs <= (int)CS.arg_size(),
"gc.statepoint too few arguments according to length fields", &CI); "gc.statepoint too few arguments according to length fields", &CI);
// Check that the only uses of this gc.statepoint are gc.result or // Check that the only uses of this gc.statepoint are gc.result or
// gc.relocate calls which are tied to this statepoint and thus part // gc.relocate calls which are tied to this statepoint and thus part
// of the same statepoint sequence // of the same statepoint sequence
for (User *U : CI.users()) { for (const User *U : CI.users()) {
const CallInst *Call = dyn_cast<const CallInst>(U); const CallInst *Call = dyn_cast<const CallInst>(U);
Assert2(Call, "illegal use of statepoint token", &CI, U); Assert2(Call, "illegal use of statepoint token", &CI, U);
if (!Call) continue; if (!Call) continue;
@@ -2727,7 +2727,10 @@ void Verifier::visitIntrinsicFunctionCall(Intrinsic::ID ID, CallInst &CI) {
} }
case Intrinsic::experimental_gc_statepoint: case Intrinsic::experimental_gc_statepoint:
VerifyStatepoint(CI); Assert1(!CI.isInlineAsm(),
"gc.statepoint support for inline assembly unimplemented", &CI);
VerifyStatepoint(ImmutableCallSite(&CI));
break; break;
case Intrinsic::experimental_gc_result_int: case Intrinsic::experimental_gc_result_int:
case Intrinsic::experimental_gc_result_float: case Intrinsic::experimental_gc_result_float: