diff options
Diffstat (limited to 'test/Transforms/RewriteStatepointsForGC/relocation.ll')
-rw-r--r-- | test/Transforms/RewriteStatepointsForGC/relocation.ll | 140 |
1 files changed, 62 insertions, 78 deletions
diff --git a/test/Transforms/RewriteStatepointsForGC/relocation.ll b/test/Transforms/RewriteStatepointsForGC/relocation.ll index deea377c5a28..eaa826c52dc2 100644 --- a/test/Transforms/RewriteStatepointsForGC/relocation.ll +++ b/test/Transforms/RewriteStatepointsForGC/relocation.ll @@ -1,27 +1,28 @@ -; RUN: opt %s -rewrite-statepoints-for-gc -spp-rematerialization-threshold=0 -S 2>&1 | FileCheck %s +; RUN: opt < %s -rewrite-statepoints-for-gc -spp-rematerialization-threshold=0 -S | FileCheck %s declare void @foo() -declare void @use(...) + +declare void @use(...) "gc-leaf-function" define i64 addrspace(1)* @test1(i64 addrspace(1)* %obj, i64 addrspace(1)* %obj2, i1 %condition) gc "statepoint-example" { -entry: ; CHECK-LABEL: @test1 ; CHECK-DAG: %obj.relocated ; CHECK-DAG: %obj2.relocated - %safepoint_token = call token (i64, i32, void ()*, i32, i32, ...) @llvm.experimental.gc.statepoint.p0f_isVoidf(i64 0, i32 0, void ()* @foo, i32 0, i32 0, i32 0, i32 0) +entry: + call void @foo() [ "deopt"() ] br label %joint -joint: +joint: ; preds = %joint2, %entry ; CHECK-LABEL: joint: ; CHECK: %phi1 = phi i64 addrspace(1)* [ %obj.relocated.casted, %entry ], [ %obj3, %joint2 ] %phi1 = phi i64 addrspace(1)* [ %obj, %entry ], [ %obj3, %joint2 ] br i1 %condition, label %use, label %joint2 -use: +use: ; preds = %joint br label %joint2 -joint2: +joint2: ; preds = %use, %joint ; CHECK-LABEL: joint2: ; CHECK: %phi2 = phi i64 addrspace(1)* [ %obj.relocated.casted, %use ], [ %obj2.relocated.casted, %joint ] ; CHECK: %obj3 = getelementptr i64, i64 addrspace(1)* %obj2.relocated.casted, i32 1 @@ -30,11 +31,11 @@ joint2: br label %joint } -declare i64 addrspace(1)* @generate_obj() +declare i64 addrspace(1)* @generate_obj() "gc-leaf-function" -declare void @consume_obj(i64 addrspace(1)*) +declare void @consume_obj(i64 addrspace(1)*) "gc-leaf-function" -declare i1 @rt() +declare i1 @rt() "gc-leaf-function" define void @test2() gc "statepoint-example" { ; CHECK-LABEL: @test2 @@ -43,60 +44,61 @@ entry: %obj = getelementptr i64, i64 addrspace(1)* %obj_init, i32 42 br label %loop -loop: +loop: ; preds = %loop.backedge, %entry ; CHECK: loop: ; CHECK-DAG: [ %obj_init.relocated.casted, %loop.backedge ] ; CHECK-DAG: [ %obj_init, %entry ] ; CHECK-DAG: [ %obj.relocated.casted, %loop.backedge ] ; CHECK-DAG: [ %obj, %entry ] - %index = phi i32 [ 0, %entry ], [ %index.inc, %loop.backedge ] ; CHECK-NOT: %location = getelementptr i64, i64 addrspace(1)* %obj, i32 %index + %index = phi i32 [ 0, %entry ], [ %index.inc, %loop.backedge ] %location = getelementptr i64, i64 addrspace(1)* %obj, i32 %index call void @consume_obj(i64 addrspace(1)* %location) %index.inc = add i32 %index, 1 %condition = call i1 @rt() br i1 %condition, label %loop_x, label %loop_y -loop_x: +loop_x: ; preds = %loop br label %loop.backedge -loop.backedge: - %safepoint_token = call token (i64, i32, void ()*, i32, i32, ...) @llvm.experimental.gc.statepoint.p0f_isVoidf(i64 0, i32 0, void ()* @do_safepoint, i32 0, i32 0, i32 0, i32 0) +loop.backedge: ; preds = %loop_y, %loop_x + call void @do_safepoint() [ "deopt"() ] br label %loop -loop_y: +loop_y: ; preds = %loop br label %loop.backedge } -declare void @some_call(i8 addrspace(1)*) +declare void @some_call(i8 addrspace(1)*) "gc-leaf-function" define void @relocate_merge(i1 %cnd, i8 addrspace(1)* %arg) gc "statepoint-example" { ; CHECK-LABEL: @relocate_merge + bci_0: br i1 %cnd, label %if_branch, label %else_branch -if_branch: +if_branch: ; preds = %bci_0 ; CHECK-LABEL: if_branch: ; CHECK: gc.statepoint ; CHECK: gc.relocate - %safepoint_token = call token (i64, i32, void ()*, i32, i32, ...) @llvm.experimental.gc.statepoint.p0f_isVoidf(i64 0, i32 0, void ()* @foo, i32 0, i32 0, i32 0, i32 0) + call void @foo() [ "deopt"() ] br label %join -else_branch: +else_branch: ; preds = %bci_0 ; CHECK-LABEL: else_branch: ; CHECK: gc.statepoint ; CHECK: gc.relocate - %safepoint_token1 = call token (i64, i32, void ()*, i32, i32, ...) @llvm.experimental.gc.statepoint.p0f_isVoidf(i64 0, i32 0, void ()* @foo, i32 0, i32 0, i32 0, i32 0) +; We need to end up with a single relocation phi updated from both paths + call void @foo() [ "deopt"() ] br label %join -join: -; We need to end up with a single relocation phi updated from both paths +join: ; preds = %else_branch, %if_branch ; CHECK-LABEL: join: ; CHECK: phi i8 addrspace(1)* ; CHECK-DAG: [ %arg.relocated, %if_branch ] -; CHECK-DAG: [ %arg.relocated4, %else_branch ] +; CHECK-DAG: [ %arg.relocated2, %else_branch ] ; CHECK-NOT: phi - call void (i8 addrspace(1)*) @some_call(i8 addrspace(1)* %arg) + call void @some_call(i8 addrspace(1)* %arg) ret void } @@ -104,14 +106,14 @@ join: ; This is basically just making sure that statepoints aren't accidentally ; treated specially. define void @test3(i64 addrspace(1)* %obj) gc "statepoint-example" { -entry: ; CHECK-LABEL: @test3 ; CHECK: gc.statepoint ; CHECK-NEXT: gc.relocate ; CHECK-NEXT: bitcast ; CHECK-NEXT: gc.statepoint - %safepoint_token = call token (i64, i32, void (i64)*, i32, i32, ...) @llvm.experimental.gc.statepoint.p0f_isVoidi64f(i64 0, i32 0, void (i64)* undef, i32 1, i32 0, i64 undef, i32 0, i32 5, i32 0, i32 -1, i32 0, i32 0, i32 0) - %safepoint_token1 = call token (i64, i32, i32 (i64 addrspace(1)*)*, i32, i32, ...) @llvm.experimental.gc.statepoint.p0f_i32p1i64f(i64 0, i32 0, i32 (i64 addrspace(1)*)* undef, i32 1, i32 0, i64 addrspace(1)* %obj, i32 0, i32 5, i32 0, i32 -1, i32 0, i32 0, i32 0) +entry: + call void undef(i64 undef) [ "deopt"(i32 0, i32 -1, i32 0, i32 0, i32 0) ] + %0 = call i32 undef(i64 addrspace(1)* %obj) [ "deopt"(i32 0, i32 -1, i32 0, i32 0, i32 0) ] ret void } @@ -122,30 +124,28 @@ define void @test4() gc "statepoint-example" { ; CHECK: gc.statepoint ; CHECK: gc.result ; CHECK: gc.statepoint -; CHECK: gc.relocate -; CHECK: @use(i8 addrspace(1)* %res.relocated) - %safepoint_token2 = tail call token (i64, i32, i8 addrspace(1)* ()*, i32, i32, ...) @llvm.experimental.gc.statepoint.p0f_p1i8f(i64 0, i32 0, i8 addrspace(1)* ()* undef, i32 0, i32 0, i32 0, i32 0) - %res = call i8 addrspace(1)* @llvm.experimental.gc.result.p1i8(token %safepoint_token2) - call token (i64, i32, i8 addrspace(1)* ()*, i32, i32, ...) @llvm.experimental.gc.statepoint.p0f_p1i8f(i64 0, i32 0, i8 addrspace(1)* ()* undef, i32 0, i32 0, i32 0, i32 0) - call void (...) @use(i8 addrspace(1)* %res) +; CHECK: [[RELOCATED:%[^ ]+]] = call {{.*}}gc.relocate +; CHECK: @use(i8 addrspace(1)* [[RELOCATED]]) + %1 = call i8 addrspace(1)* undef() [ "deopt"() ] + %2 = call i8 addrspace(1)* undef() [ "deopt"() ] + call void (...) @use(i8 addrspace(1)* %1) unreachable } - ; Test updating a phi where not all inputs are live to begin with define void @test5(i8 addrspace(1)* %arg) gc "statepoint-example" { ; CHECK-LABEL: test5 entry: - call token (i64, i32, i8 addrspace(1)* ()*, i32, i32, ...) @llvm.experimental.gc.statepoint.p0f_p1i8f(i64 0, i32 0, i8 addrspace(1)* ()* undef, i32 0, i32 0, i32 0, i32 0) + %0 = call i8 addrspace(1)* undef() [ "deopt"() ] switch i32 undef, label %kill [ i32 10, label %merge i32 13, label %merge ] -kill: +kill: ; preds = %entry br label %merge -merge: +merge: ; preds = %kill, %entry, %entry ; CHECK: merge: ; CHECK: %test = phi i8 addrspace(1) ; CHECK-DAG: [ null, %kill ] @@ -156,24 +156,22 @@ merge: unreachable } - ; Check to make sure we handle values live over an entry statepoint -define void @test6(i8 addrspace(1)* %arg1, i8 addrspace(1)* %arg2, - i8 addrspace(1)* %arg3) gc "statepoint-example" { +define void @test6(i8 addrspace(1)* %arg1, i8 addrspace(1)* %arg2, i8 addrspace(1)* %arg3) gc "statepoint-example" { ; CHECK-LABEL: @test6 entry: br i1 undef, label %gc.safepoint_poll.exit2, label %do_safepoint -do_safepoint: +do_safepoint: ; preds = %entry ; CHECK-LABEL: do_safepoint: ; CHECK: gc.statepoint ; CHECK: arg1.relocated = ; CHECK: arg2.relocated = ; CHECK: arg3.relocated = - call token (i64, i32, void ()*, i32, i32, ...) @llvm.experimental.gc.statepoint.p0f_isVoidf(i64 0, i32 0, void ()* @foo, i32 0, i32 0, i32 0, i32 3, i8 addrspace(1)* %arg1, i8 addrspace(1)* %arg2, i8 addrspace(1)* %arg3) + call void @foo() [ "deopt"(i8 addrspace(1)* %arg1, i8 addrspace(1)* %arg2, i8 addrspace(1)* %arg3) ] br label %gc.safepoint_poll.exit2 -gc.safepoint_poll.exit2: +gc.safepoint_poll.exit2: ; preds = %do_safepoint, %entry ; CHECK-LABEL: gc.safepoint_poll.exit2: ; CHECK: phi i8 addrspace(1)* ; CHECK-DAG: [ %arg3, %entry ] @@ -190,44 +188,42 @@ gc.safepoint_poll.exit2: ; Check relocation in a loop nest where a relocation happens in the outer ; but not the inner loop -define void @test_outer_loop(i8 addrspace(1)* %arg1, i8 addrspace(1)* %arg2, - i1 %cmp) gc "statepoint-example" { +define void @test_outer_loop(i8 addrspace(1)* %arg1, i8 addrspace(1)* %arg2, i1 %cmp) gc "statepoint-example" { ; CHECK-LABEL: @test_outer_loop + bci_0: br label %outer-loop -outer-loop: +outer-loop: ; preds = %outer-inc, %bci_0 ; CHECK-LABEL: outer-loop: ; CHECK: phi i8 addrspace(1)* [ %arg2, %bci_0 ], [ %arg2.relocated, %outer-inc ] ; CHECK: phi i8 addrspace(1)* [ %arg1, %bci_0 ], [ %arg1.relocated, %outer-inc ] br label %inner-loop -inner-loop: +inner-loop: ; preds = %inner-loop, %outer-loop br i1 %cmp, label %inner-loop, label %outer-inc -outer-inc: +outer-inc: ; preds = %inner-loop ; CHECK-LABEL: outer-inc: ; CHECK: %arg1.relocated ; CHECK: %arg2.relocated - %safepoint_token = call token (i64, i32, void ()*, i32, i32, ...) @llvm.experimental.gc.statepoint.p0f_isVoidf(i64 0, i32 0, void ()* @foo, i32 0, i32 0, i32 0, i32 2, i8 addrspace(1)* %arg1, i8 addrspace(1)* %arg2) + call void @foo() [ "deopt"(i8 addrspace(1)* %arg1, i8 addrspace(1)* %arg2) ] br label %outer-loop } ; Check that both inner and outer loops get phis when relocation is in ; inner loop -define void @test_inner_loop(i8 addrspace(1)* %arg1, i8 addrspace(1)* %arg2, - i1 %cmp) gc "statepoint-example" { +define void @test_inner_loop(i8 addrspace(1)* %arg1, i8 addrspace(1)* %arg2, i1 %cmp) gc "statepoint-example" { ; CHECK-LABEL: @test_inner_loop + bci_0: br label %outer-loop -outer-loop: +outer-loop: ; preds = %outer-inc, %bci_0 ; CHECK-LABEL: outer-loop: ; CHECK: phi i8 addrspace(1)* [ %arg2, %bci_0 ], [ %arg2.relocated, %outer-inc ] ; CHECK: phi i8 addrspace(1)* [ %arg1, %bci_0 ], [ %arg1.relocated, %outer-inc ] br label %inner-loop - -inner-loop: ; CHECK-LABEL: inner-loop ; CHECK: phi i8 addrspace(1)* ; CHECK-DAG: %outer-loop ] @@ -238,42 +234,40 @@ inner-loop: ; CHECK: gc.statepoint ; CHECK: %arg1.relocated ; CHECK: %arg2.relocated - %safepoint_token = call token (i64, i32, void ()*, i32, i32, ...) @llvm.experimental.gc.statepoint.p0f_isVoidf(i64 0, i32 0, void ()* @foo, i32 0, i32 0, i32 0, i32 2, i8 addrspace(1)* %arg1, i8 addrspace(1)* %arg2) + +inner-loop: ; preds = %inner-loop, %outer-loop + call void @foo() [ "deopt"(i8 addrspace(1)* %arg1, i8 addrspace(1)* %arg2) ] br i1 %cmp, label %inner-loop, label %outer-inc -outer-inc: +outer-inc: ; preds = %inner-loop ; CHECK-LABEL: outer-inc: +; This test shows why updating just those uses of the original value being +; relocated dominated by the inserted relocation is not always sufficient. br label %outer-loop } - -; This test shows why updating just those uses of the original value being -; relocated dominated by the inserted relocation is not always sufficient. define i64 addrspace(1)* @test7(i64 addrspace(1)* %obj, i64 addrspace(1)* %obj2, i1 %condition) gc "statepoint-example" { ; CHECK-LABEL: @test7 entry: br i1 %condition, label %branch2, label %join -branch2: +branch2: ; preds = %entry br i1 %condition, label %callbb, label %join2 -callbb: - %safepoint_token = call token (i64, i32, void ()*, i32, i32, ...) @llvm.experimental.gc.statepoint.p0f_isVoidf(i64 0, i32 0, void ()* @foo, i32 0, i32 0, i32 0, i32 5, i32 0, i32 -1, i32 0, i32 0, i32 0) +callbb: ; preds = %branch2 + call void @foo() [ "deopt"(i32 0, i32 -1, i32 0, i32 0, i32 0) ] br label %join -join: +join: ; preds = %callbb, %entry ; CHECK-LABEL: join: ; CHECK: phi i64 addrspace(1)* [ %obj.relocated.casted, %callbb ], [ %obj, %entry ] ; CHECK: phi i64 addrspace(1)* ; CHECK-DAG: [ %obj, %entry ] ; CHECK-DAG: [ %obj2.relocated.casted, %callbb ] - ; This is a phi outside the dominator region of the new defs inserted by - ; the safepoint, BUT we can't stop the search here or we miss the second - ; phi below. %phi1 = phi i64 addrspace(1)* [ %obj, %entry ], [ %obj2, %callbb ] br label %join2 -join2: +join2: ; preds = %join, %branch2 ; CHECK-LABEL: join2: ; CHECK: phi2 = phi i64 addrspace(1)* ; CHECK-DAG: %join ] @@ -282,14 +276,4 @@ join2: ret i64 addrspace(1)* %phi2 } - declare void @do_safepoint() - -declare token @llvm.experimental.gc.statepoint.p0f_isVoidf(i64, i32, void ()*, i32, i32, ...) -declare token @llvm.experimental.gc.statepoint.p0f_p1i8f(i64, i32, i8 addrspace(1)* ()*, i32, i32, ...) -declare token @llvm.experimental.gc.statepoint.p0f_isVoidi64f(i64, i32, void (i64)*, i32, i32, ...) -declare token @llvm.experimental.gc.statepoint.p0f_i32p1i64f(i64, i32, i32 (i64 addrspace(1)*)*, i32, i32, ...) -declare i8 addrspace(1)* @llvm.experimental.gc.result.p1i8(token) #3 - - - |