llvm/llvm/test/Transforms/RewriteStatepointsForGC/invokes.ll

; RUN: opt < %s -S -passes=rewrite-statepoints-for-gc | FileCheck %s

declare ptr addrspace(1) @some_call(ptr addrspace(1))
declare i32 @personality_function()

define ptr addrspace(1) @test_basic(ptr addrspace(1) %obj, ptr addrspace(1) %obj1) gc "statepoint-example" personality ptr @personality_function {
; CHECK-LABEL: entry:
entry:
  ; CHECK: invoke
  ; CHECK: statepoint
  ; CHECK: some_call
  %ret_val = invoke ptr addrspace(1) @some_call(ptr addrspace(1) %obj)
               to label %normal_return unwind label %exceptional_return

; CHECK-LABEL: normal_return:
; CHECK: gc.result
; CHECK: ret ptr

normal_return:
  ret ptr addrspace(1) %ret_val

; CHECK-LABEL: exceptional_return:
; CHECK: landingpad
; CHECK: ret ptr

exceptional_return:
  %landing_pad4 = landingpad token
          cleanup
  ret ptr addrspace(1) %obj1
}

declare <4 x ptr addrspace(1)> @some_vector_call(<4 x ptr addrspace(1)>)

define <4 x ptr addrspace(1)> @test_basic_vector(<4 x ptr addrspace(1)> %objs, <4 x ptr addrspace(1)> %objs1) gc "statepoint-example" personality ptr @personality_function {
; CHECK-LABEL: @test_basic_vector
entry:
; CHECK: invoke{{.*}}llvm.experimental.gc.statepoint{{.*}}some_vector_call
  %ret_val = invoke <4 x ptr addrspace(1)> @some_vector_call(<4 x ptr addrspace(1)> %objs)
               to label %normal_return unwind label %exceptional_return

; CHECK-LABEL: normal_return:
; CHECK: gc.result
; CHECK: ret <4 x ptr addrspace(1)>

normal_return:
  ret <4 x ptr addrspace(1)> %ret_val

; CHECK-LABEL: exceptional_return:
; CHECK: landingpad
; CHECK: ret <4 x ptr addrspace(1)>

exceptional_return:
  %landing_pad4 = landingpad token
          cleanup
  ret <4 x ptr addrspace(1)> %objs1
}

define ptr addrspace(1) @test_two_invokes(ptr addrspace(1) %obj, ptr addrspace(1) %obj1) gc "statepoint-example" personality ptr @personality_function {
; CHECK-LABEL: entry:
entry:
  ; CHECK: invoke
  ; CHECK: statepoint
  ; CHECK: some_call
  %ret_val1 = invoke ptr addrspace(1) @some_call(ptr addrspace(1) %obj)
               to label %second_invoke unwind label %exceptional_return

; CHECK-LABEL: second_invoke:
second_invoke:
  ; CHECK: invoke
  ; CHECK: statepoint
  ; CHECK: some_call
  %ret_val2 = invoke ptr addrspace(1) @some_call(ptr addrspace(1) %ret_val1)
                to label %normal_return unwind label %exceptional_return

; CHECK-LABEL: normal_return:
normal_return:
  ; CHECK: gc.result
  ; CHECK: ret ptr
  ret ptr addrspace(1) %ret_val2

; CHECK: exceptional_return:
; CHECK: ret ptr

exceptional_return:
  %landing_pad4 = landingpad token
          cleanup
  ret ptr addrspace(1) %obj1
}

define ptr addrspace(1) @test_phi_node(i1 %cond, ptr addrspace(1) %obj) gc "statepoint-example" personality ptr @personality_function {
; CHECK-LABEL: @test_phi_node
; CHECK-LABEL: entry:
entry:
  br i1 %cond, label %left, label %right

left:
  %ret_val_left = invoke ptr addrspace(1) @some_call(ptr addrspace(1) %obj)
                    to label %merge unwind label %exceptional_return

right:
  %ret_val_right = invoke ptr addrspace(1) @some_call(ptr addrspace(1) %obj)
                     to label %merge unwind label %exceptional_return

; CHECK: merge[[A:[0-9]]]:
; CHECK: gc.result
; CHECK: br label %[[with_phi:merge[0-9]*]]

; CHECK: merge[[B:[0-9]]]:
; CHECK: gc.result
; CHECK: br label %[[with_phi]]

; CHECK: [[with_phi]]:
; CHECK: phi
; CHECK: ret ptr addrspace(1) %ret_val
merge:
  %ret_val = phi ptr addrspace(1) [%ret_val_left, %left], [%ret_val_right, %right]
  ret ptr addrspace(1) %ret_val

; CHECK-LABEL: exceptional_return:
; CHECK: ret ptr addrspace(1)

exceptional_return:
  %landing_pad4 = landingpad token
          cleanup
  ret ptr addrspace(1) %obj
}

declare void @do_safepoint()
define void @gc.safepoint_poll() {
; CHECK-LABEL: gc.safepoint_poll
; CHECK-LABEL: entry
; CHECK-NEXT: do_safepoint
; CHECK-NEXT: ret void
entry:
  call void @do_safepoint()
  ret void
}