@@ -1987,6 +1987,13 @@ void Verifier::visitInvokeInst(InvokeInst &II) {
1987
1987
Assert1 (II.getUnwindDest ()->isLandingPad (),
1988
1988
" The unwind destination does not have a landingpad instruction!" ,&II);
1989
1989
1990
+ if (Function *F = II.getCalledFunction ())
1991
+ // TODO: Ideally we should use visitIntrinsicFunction here. But it uses
1992
+ // CallInst as an input parameter. It not woth updating this whole
1993
+ // function only to support statepoint verification.
1994
+ if (F->getIntrinsicID () == Intrinsic::experimental_gc_statepoint)
1995
+ VerifyStatepoint (ImmutableCallSite (&II));
1996
+
1990
1997
visitTerminatorInst (II);
1991
1998
}
1992
1999
@@ -2486,7 +2493,8 @@ void Verifier::visitInstruction(Instruction &I) {
2486
2493
Assert1 (!F->isIntrinsic () || isa<CallInst>(I) ||
2487
2494
F->getIntrinsicID () == Intrinsic::donothing ||
2488
2495
F->getIntrinsicID () == Intrinsic::experimental_patchpoint_void ||
2489
- F->getIntrinsicID () == Intrinsic::experimental_patchpoint_i64,
2496
+ F->getIntrinsicID () == Intrinsic::experimental_patchpoint_i64 ||
2497
+ F->getIntrinsicID () == Intrinsic::experimental_gc_statepoint,
2490
2498
" Cannot invoke an intrinsinc other than"
2491
2499
" donothing or patchpoint" , &I);
2492
2500
Assert1 (F->getParent () == M, " Referencing function in another module!" ,
@@ -2901,14 +2909,46 @@ void Verifier::visitIntrinsicFunctionCall(Intrinsic::ID ID, CallInst &CI) {
2901
2909
break ;
2902
2910
}
2903
2911
case Intrinsic::experimental_gc_relocate: {
2904
- // Are we tied to a statepoint properly?
2905
- CallSite StatepointCS (CI.getArgOperand (0 ));
2906
- const Function *StatepointFn =
2907
- StatepointCS.getInstruction () ? StatepointCS.getCalledFunction () : nullptr ;
2908
- Assert2 (StatepointFn && StatepointFn->isDeclaration () &&
2909
- StatepointFn->getIntrinsicID () == Intrinsic::experimental_gc_statepoint,
2910
- " gc.relocate operand #1 must be from a statepoint" ,
2911
- &CI, CI.getArgOperand (0 ));
2912
+ Assert1 (CI.getNumArgOperands () == 3 , " wrong number of arguments" , &CI);
2913
+
2914
+ // Check that this relocate is correctly tied to the statepoint
2915
+
2916
+ // This is case for relocate on the unwinding path of an invoke statepoint
2917
+ if (ExtractValueInst *ExtractValue =
2918
+ dyn_cast<ExtractValueInst>(CI.getArgOperand (0 ))) {
2919
+ Assert1 (isa<LandingPadInst>(ExtractValue->getAggregateOperand ()),
2920
+ " gc relocate on unwind path incorrectly linked to the statepoint" ,
2921
+ &CI);
2922
+
2923
+ const BasicBlock *invokeBB =
2924
+ ExtractValue->getParent ()->getUniquePredecessor ();
2925
+
2926
+ // Landingpad relocates should have only one predecessor with invoke
2927
+ // statepoint terminator
2928
+ Assert1 (invokeBB,
2929
+ " safepoints should have unique landingpads" ,
2930
+ ExtractValue->getParent ());
2931
+ Assert1 (invokeBB->getTerminator (),
2932
+ " safepoint block should be well formed" ,
2933
+ invokeBB);
2934
+ Assert1 (isStatepoint (invokeBB->getTerminator ()),
2935
+ " gc relocate should be linked to a statepoint" ,
2936
+ invokeBB);
2937
+ }
2938
+ else {
2939
+ // In all other cases relocate should be tied to the statepoint directly.
2940
+ // This covers relocates on a normal return path of invoke statepoint and
2941
+ // relocates of a call statepoint
2942
+ auto Token = CI.getArgOperand (0 );
2943
+ Assert2 (isa<Instruction>(Token) && isStatepoint (cast<Instruction>(Token)),
2944
+ " gc relocate is incorrectly tied to the statepoint" ,
2945
+ &CI, Token);
2946
+ }
2947
+
2948
+ // Verify rest of the relocate arguments
2949
+
2950
+ GCRelocateOperands ops (&CI);
2951
+ ImmutableCallSite StatepointCS (ops.statepoint ());
2912
2952
2913
2953
// Both the base and derived must be piped through the safepoint
2914
2954
Value* Base = CI.getArgOperand (1 );
0 commit comments