llvm/llvm/test/Transforms/DeadStoreElimination/nounwind-invoke.ll

; NOTE: Assertions have been autogenerated by utils/update_test_checks.py UTC_ARGS: --prefix-filecheck-ir-name abc
; RUN: opt -passes=dse -S < %s | FileCheck %s

; Make sure invokes are not removed as dead stores.
define void @test_nounwind_invoke() personality ptr @__gxx_personality_v0 {
; CHECK-LABEL: @test_nounwind_invoke(
; CHECK-NEXT:  bb:
; CHECK-NEXT:    [[TMP:%.*]] = alloca i32, align 4
; CHECK-NEXT:    invoke void @foo(ptr [[TMP]])
; CHECK-NEXT:    to label [[BB1:%.*]] unwind label [[BB2:%.*]]
; CHECK:       bb1:
; CHECK-NEXT:    call void @llvm.lifetime.end.p0(i64 4, ptr [[TMP]])
; CHECK-NEXT:    ret void
; CHECK:       bb2:
; CHECK-NEXT:    [[ABCTMP1:%.*]] = landingpad { ptr, i32 }
; CHECK-NEXT:    cleanup
; CHECK-NEXT:    resume { ptr, i32 } [[ABCTMP1]]
;
bb:
  %tmp = alloca i32, align 4
  ; 'foo' is 'argmemonly', meaning it can only write to memory pointed by %tmp.
  ; And this def is killed by 'call @llvm.lifetime.end.p0' in bb1 without
  ; being used elsewhere, becoming a dead store. But we shouldn't remove this
  ; because invokes are terminators and thus cannot be removed.
  invoke void @foo(ptr %tmp)
  to label %bb1 unwind label %bb2

bb1:                                              ; preds = %bb
  call void @llvm.lifetime.end.p0(i64 4, ptr %tmp)
  ret void

bb2:                                              ; preds = %bb
  %tmp1 = landingpad { ptr, i32 }
  cleanup
  resume { ptr, i32 } %tmp1
}

; Function Attrs: argmemonly nocallback nofree nosync nounwind willreturn
declare void @llvm.lifetime.end.p0(i64 immarg, ptr nocapture) #0
; Function Attrs: argmemonly nounwind willreturn
declare void @foo(ptr) #1
declare i32 @__gxx_personality_v0(...)

attributes #0 = { argmemonly nocallback nofree nosync nounwind willreturn }
attributes #1 = { argmemonly nounwind willreturn }