; 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 }