1 ; RUN: opt < %s -rewrite-statepoints-for-gc -spp-rematerialization-threshold=0 -S | FileCheck %s 2 3 4 declare void @foo() 5 6 declare void @use(...) "gc-leaf-function" 7 8 define i64 addrspace(1)* @test1(i64 addrspace(1)* %obj, i64 addrspace(1)* %obj2, i1 %condition) gc "statepoint-example" { 9 ; CHECK-LABEL: @test1 10 ; CHECK-DAG: %obj.relocated 11 ; CHECK-DAG: %obj2.relocated 12 entry: 13 call void @foo() [ "deopt"() ] 14 br label %joint 15 16 joint: ; preds = %joint2, %entry 17 ; CHECK-LABEL: joint: 18 ; CHECK: %phi1 = phi i64 addrspace(1)* [ %obj.relocated.casted, %entry ], [ %obj3, %joint2 ] 19 %phi1 = phi i64 addrspace(1)* [ %obj, %entry ], [ %obj3, %joint2 ] 20 br i1 %condition, label %use, label %joint2 21 22 use: ; preds = %joint 23 br label %joint2 24 25 joint2: ; preds = %use, %joint 26 ; CHECK-LABEL: joint2: 27 ; CHECK: %phi2 = phi i64 addrspace(1)* [ %obj.relocated.casted, %use ], [ %obj2.relocated.casted, %joint ] 28 ; CHECK: %obj3 = getelementptr i64, i64 addrspace(1)* %obj2.relocated.casted, i32 1 29 %phi2 = phi i64 addrspace(1)* [ %obj, %use ], [ %obj2, %joint ] 30 %obj3 = getelementptr i64, i64 addrspace(1)* %obj2, i32 1 31 br label %joint 32 } 33 34 declare i64 addrspace(1)* @generate_obj() "gc-leaf-function" 35 36 declare void @consume_obj(i64 addrspace(1)*) "gc-leaf-function" 37 38 declare i1 @rt() "gc-leaf-function" 39 40 define void @test2() gc "statepoint-example" { 41 ; CHECK-LABEL: @test2 42 entry: 43 %obj_init = call i64 addrspace(1)* @generate_obj() 44 %obj = getelementptr i64, i64 addrspace(1)* %obj_init, i32 42 45 br label %loop 46 47 loop: ; preds = %loop.backedge, %entry 48 ; CHECK: loop: 49 ; CHECK-DAG: [ %obj_init.relocated.casted, %loop.backedge ] 50 ; CHECK-DAG: [ %obj_init, %entry ] 51 ; CHECK-DAG: [ %obj.relocated.casted, %loop.backedge ] 52 ; CHECK-DAG: [ %obj, %entry ] 53 ; CHECK-NOT: %location = getelementptr i64, i64 addrspace(1)* %obj, i32 %index 54 %index = phi i32 [ 0, %entry ], [ %index.inc, %loop.backedge ] 55 %location = getelementptr i64, i64 addrspace(1)* %obj, i32 %index 56 call void @consume_obj(i64 addrspace(1)* %location) 57 %index.inc = add i32 %index, 1 58 %condition = call i1 @rt() 59 br i1 %condition, label %loop_x, label %loop_y 60 61 loop_x: ; preds = %loop 62 br label %loop.backedge 63 64 loop.backedge: ; preds = %loop_y, %loop_x 65 call void @do_safepoint() [ "deopt"() ] 66 br label %loop 67 68 loop_y: ; preds = %loop 69 br label %loop.backedge 70 } 71 72 declare void @some_call(i8 addrspace(1)*) "gc-leaf-function" 73 74 define void @relocate_merge(i1 %cnd, i8 addrspace(1)* %arg) gc "statepoint-example" { 75 ; CHECK-LABEL: @relocate_merge 76 77 bci_0: 78 br i1 %cnd, label %if_branch, label %else_branch 79 80 if_branch: ; preds = %bci_0 81 ; CHECK-LABEL: if_branch: 82 ; CHECK: gc.statepoint 83 ; CHECK: gc.relocate 84 call void @foo() [ "deopt"() ] 85 br label %join 86 87 else_branch: ; preds = %bci_0 88 ; CHECK-LABEL: else_branch: 89 ; CHECK: gc.statepoint 90 ; CHECK: gc.relocate 91 ; We need to end up with a single relocation phi updated from both paths 92 call void @foo() [ "deopt"() ] 93 br label %join 94 95 join: ; preds = %else_branch, %if_branch 96 ; CHECK-LABEL: join: 97 ; CHECK: phi i8 addrspace(1)* 98 ; CHECK-DAG: [ %arg.relocated, %if_branch ] 99 ; CHECK-DAG: [ %arg.relocated2, %else_branch ] 100 ; CHECK-NOT: phi 101 call void @some_call(i8 addrspace(1)* %arg) 102 ret void 103 } 104 105 ; Make sure a use in a statepoint gets properly relocated at a previous one. 106 ; This is basically just making sure that statepoints aren't accidentally 107 ; treated specially. 108 define void @test3(i64 addrspace(1)* %obj) gc "statepoint-example" { 109 ; CHECK-LABEL: @test3 110 ; CHECK: gc.statepoint 111 ; CHECK-NEXT: gc.relocate 112 ; CHECK-NEXT: bitcast 113 ; CHECK-NEXT: gc.statepoint 114 entry: 115 call void undef(i64 undef) [ "deopt"(i32 0, i32 -1, i32 0, i32 0, i32 0) ] 116 %0 = call i32 undef(i64 addrspace(1)* %obj) [ "deopt"(i32 0, i32 -1, i32 0, i32 0, i32 0) ] 117 ret void 118 } 119 120 ; Check specifically for the case where the result of a statepoint needs to 121 ; be relocated itself 122 define void @test4() gc "statepoint-example" { 123 ; CHECK-LABEL: @test4 124 ; CHECK: gc.statepoint 125 ; CHECK: gc.result 126 ; CHECK: gc.statepoint 127 ; CHECK: [[RELOCATED:%[^ ]+]] = call {{.*}}gc.relocate 128 ; CHECK: @use(i8 addrspace(1)* [[RELOCATED]]) 129 %1 = call i8 addrspace(1)* undef() [ "deopt"() ] 130 %2 = call i8 addrspace(1)* undef() [ "deopt"() ] 131 call void (...) @use(i8 addrspace(1)* %1) 132 unreachable 133 } 134 135 ; Test updating a phi where not all inputs are live to begin with 136 define void @test5(i8 addrspace(1)* %arg) gc "statepoint-example" { 137 ; CHECK-LABEL: test5 138 entry: 139 %0 = call i8 addrspace(1)* undef() [ "deopt"() ] 140 switch i32 undef, label %kill [ 141 i32 10, label %merge 142 i32 13, label %merge 143 ] 144 145 kill: ; preds = %entry 146 br label %merge 147 148 merge: ; preds = %kill, %entry, %entry 149 ; CHECK: merge: 150 ; CHECK: %test = phi i8 addrspace(1) 151 ; CHECK-DAG: [ null, %kill ] 152 ; CHECK-DAG: [ %arg.relocated, %entry ] 153 ; CHECK-DAG: [ %arg.relocated, %entry ] 154 %test = phi i8 addrspace(1)* [ null, %kill ], [ %arg, %entry ], [ %arg, %entry ] 155 call void (...) @use(i8 addrspace(1)* %test) 156 unreachable 157 } 158 159 ; Check to make sure we handle values live over an entry statepoint 160 define void @test6(i8 addrspace(1)* %arg1, i8 addrspace(1)* %arg2, i8 addrspace(1)* %arg3) gc "statepoint-example" { 161 ; CHECK-LABEL: @test6 162 entry: 163 br i1 undef, label %gc.safepoint_poll.exit2, label %do_safepoint 164 165 do_safepoint: ; preds = %entry 166 ; CHECK-LABEL: do_safepoint: 167 ; CHECK: gc.statepoint 168 ; CHECK: arg1.relocated = 169 ; CHECK: arg2.relocated = 170 ; CHECK: arg3.relocated = 171 call void @foo() [ "deopt"(i8 addrspace(1)* %arg1, i8 addrspace(1)* %arg2, i8 addrspace(1)* %arg3) ] 172 br label %gc.safepoint_poll.exit2 173 174 gc.safepoint_poll.exit2: ; preds = %do_safepoint, %entry 175 ; CHECK-LABEL: gc.safepoint_poll.exit2: 176 ; CHECK: phi i8 addrspace(1)* 177 ; CHECK-DAG: [ %arg3, %entry ] 178 ; CHECK-DAG: [ %arg3.relocated, %do_safepoint ] 179 ; CHECK: phi i8 addrspace(1)* 180 ; CHECK-DAG: [ %arg2, %entry ] 181 ; CHECK-DAG: [ %arg2.relocated, %do_safepoint ] 182 ; CHECK: phi i8 addrspace(1)* 183 ; CHECK-DAG: [ %arg1, %entry ] 184 ; CHECK-DAG: [ %arg1.relocated, %do_safepoint ] 185 call void (...) @use(i8 addrspace(1)* %arg1, i8 addrspace(1)* %arg2, i8 addrspace(1)* %arg3) 186 ret void 187 } 188 189 ; Check relocation in a loop nest where a relocation happens in the outer 190 ; but not the inner loop 191 define void @test_outer_loop(i8 addrspace(1)* %arg1, i8 addrspace(1)* %arg2, i1 %cmp) gc "statepoint-example" { 192 ; CHECK-LABEL: @test_outer_loop 193 194 bci_0: 195 br label %outer-loop 196 197 outer-loop: ; preds = %outer-inc, %bci_0 198 ; CHECK-LABEL: outer-loop: 199 ; CHECK: phi i8 addrspace(1)* [ %arg2, %bci_0 ], [ %arg2.relocated, %outer-inc ] 200 ; CHECK: phi i8 addrspace(1)* [ %arg1, %bci_0 ], [ %arg1.relocated, %outer-inc ] 201 br label %inner-loop 202 203 inner-loop: ; preds = %inner-loop, %outer-loop 204 br i1 %cmp, label %inner-loop, label %outer-inc 205 206 outer-inc: ; preds = %inner-loop 207 ; CHECK-LABEL: outer-inc: 208 ; CHECK: %arg1.relocated 209 ; CHECK: %arg2.relocated 210 call void @foo() [ "deopt"(i8 addrspace(1)* %arg1, i8 addrspace(1)* %arg2) ] 211 br label %outer-loop 212 } 213 214 ; Check that both inner and outer loops get phis when relocation is in 215 ; inner loop 216 define void @test_inner_loop(i8 addrspace(1)* %arg1, i8 addrspace(1)* %arg2, i1 %cmp) gc "statepoint-example" { 217 ; CHECK-LABEL: @test_inner_loop 218 219 bci_0: 220 br label %outer-loop 221 222 outer-loop: ; preds = %outer-inc, %bci_0 223 ; CHECK-LABEL: outer-loop: 224 ; CHECK: phi i8 addrspace(1)* [ %arg2, %bci_0 ], [ %arg2.relocated, %outer-inc ] 225 ; CHECK: phi i8 addrspace(1)* [ %arg1, %bci_0 ], [ %arg1.relocated, %outer-inc ] 226 br label %inner-loop 227 ; CHECK-LABEL: inner-loop 228 ; CHECK: phi i8 addrspace(1)* 229 ; CHECK-DAG: %outer-loop ] 230 ; CHECK-DAG: [ %arg2.relocated, %inner-loop ] 231 ; CHECK: phi i8 addrspace(1)* 232 ; CHECK-DAG: %outer-loop ] 233 ; CHECK-DAG: [ %arg1.relocated, %inner-loop ] 234 ; CHECK: gc.statepoint 235 ; CHECK: %arg1.relocated 236 ; CHECK: %arg2.relocated 237 238 inner-loop: ; preds = %inner-loop, %outer-loop 239 call void @foo() [ "deopt"(i8 addrspace(1)* %arg1, i8 addrspace(1)* %arg2) ] 240 br i1 %cmp, label %inner-loop, label %outer-inc 241 242 outer-inc: ; preds = %inner-loop 243 ; CHECK-LABEL: outer-inc: 244 ; This test shows why updating just those uses of the original value being 245 ; relocated dominated by the inserted relocation is not always sufficient. 246 br label %outer-loop 247 } 248 249 define i64 addrspace(1)* @test7(i64 addrspace(1)* %obj, i64 addrspace(1)* %obj2, i1 %condition) gc "statepoint-example" { 250 ; CHECK-LABEL: @test7 251 entry: 252 br i1 %condition, label %branch2, label %join 253 254 branch2: ; preds = %entry 255 br i1 %condition, label %callbb, label %join2 256 257 callbb: ; preds = %branch2 258 call void @foo() [ "deopt"(i32 0, i32 -1, i32 0, i32 0, i32 0) ] 259 br label %join 260 261 join: ; preds = %callbb, %entry 262 ; CHECK-LABEL: join: 263 ; CHECK: phi i64 addrspace(1)* [ %obj.relocated.casted, %callbb ], [ %obj, %entry ] 264 ; CHECK: phi i64 addrspace(1)* 265 ; CHECK-DAG: [ %obj, %entry ] 266 ; CHECK-DAG: [ %obj2.relocated.casted, %callbb ] 267 %phi1 = phi i64 addrspace(1)* [ %obj, %entry ], [ %obj2, %callbb ] 268 br label %join2 269 270 join2: ; preds = %join, %branch2 271 ; CHECK-LABEL: join2: 272 ; CHECK: phi2 = phi i64 addrspace(1)* 273 ; CHECK-DAG: %join ] 274 ; CHECK-DAG: [ %obj2, %branch2 ] 275 %phi2 = phi i64 addrspace(1)* [ %obj, %join ], [ %obj2, %branch2 ] 276 ret i64 addrspace(1)* %phi2 277 } 278 279 declare void @do_safepoint() 280