Home | History | Annotate | Download | only in deopt-bundles
      1 ; RUN: opt %s -rewrite-statepoints-for-gc -rs4gc-use-deopt-bundles -spp-rematerialization-threshold=0 -S 2>&1 | 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