; RUN: opt < %s -passes=instcombine -S | FileCheck %s
@glob = global i32 0
declare ptr @llvm.stacksave()
declare void @llvm.stackrestore(ptr)
;; Test that llvm.stackrestore is removed when possible.
define ptr @test1(i32 %P) {
%tmp = call ptr @llvm.stacksave( )
call void @llvm.stackrestore( ptr %tmp ) ;; not restoring anything
%A = alloca i32, i32 %P
ret ptr %A
}
; CHECK-LABEL: define ptr @test1(
; CHECK-NOT: call void @llvm.stackrestore
; CHECK: ret ptr
define void @test2(ptr %X) {
call void @llvm.stackrestore( ptr %X ) ;; no allocas before return.
ret void
}
; CHECK-LABEL: define void @test2(
; CHECK-NOT: call void @llvm.stackrestore
; CHECK: ret void
define void @foo(i32 %size) nounwind {
entry:
%tmp118124 = icmp sgt i32 %size, 0 ; <i1> [#uses=1]
br i1 %tmp118124, label %bb.preheader, label %return
bb.preheader: ; preds = %entry
%tmp25 = add i32 %size, -1 ; <i32> [#uses=1]
%tmp125 = icmp slt i32 %size, 1 ; <i1> [#uses=1]
%smax = select i1 %tmp125, i32 1, i32 %size ; <i32> [#uses=1]
br label %bb
bb: ; preds = %bb, %bb.preheader
%i.0.reg2mem.0 = phi i32 [ 0, %bb.preheader ], [ %indvar.next, %bb ] ; <i32> [#uses=2]
%tmp = call ptr @llvm.stacksave( ) ; <ptr> [#uses=1]
%tmp23 = alloca i8, i32 %size ; <ptr> [#uses=2]
%tmp27 = getelementptr i8, ptr %tmp23, i32 %tmp25 ; <ptr> [#uses=1]
store i8 0, ptr %tmp27, align 1
%tmp28 = call ptr @llvm.stacksave( ) ; <ptr> [#uses=1]
%tmp52 = alloca i8, i32 %size ; <ptr> [#uses=1]
%tmp53 = call ptr @llvm.stacksave( ) ; <ptr> [#uses=1]
%tmp77 = alloca i8, i32 %size ; <ptr> [#uses=1]
%tmp78 = call ptr @llvm.stacksave( ) ; <ptr> [#uses=1]
%tmp102 = alloca i8, i32 %size ; <ptr> [#uses=1]
call void @bar( i32 %i.0.reg2mem.0, ptr %tmp23, ptr %tmp52, ptr %tmp77, ptr %tmp102, i32 %size ) nounwind
call void @llvm.stackrestore( ptr %tmp78 )
call void @llvm.stackrestore( ptr %tmp53 )
call void @llvm.stackrestore( ptr %tmp28 )
call void @llvm.stackrestore( ptr %tmp )
%indvar.next = add i32 %i.0.reg2mem.0, 1 ; <i32> [#uses=2]
%exitcond = icmp eq i32 %indvar.next, %smax ; <i1> [#uses=1]
br i1 %exitcond, label %return, label %bb
return: ; preds = %bb, %entry
ret void
}
; CHECK-LABEL: define void @foo(
; CHECK: %tmp = call ptr @llvm.stacksave.p0()
; CHECK: alloca i8
; CHECK-NOT: stacksave
; CHECK: call void @bar(
; CHECK-NEXT: call void @llvm.stackrestore.p0(ptr %tmp)
; CHECK: ret void
declare void @bar(i32, ptr, ptr, ptr, ptr, i32)
declare void @inalloca_callee(ptr inalloca(i32))
define void @test3(i32 %c) {
entry:
br label %loop
loop:
%i = phi i32 [0, %entry], [%i1, %loop]
%save1 = call ptr @llvm.stacksave()
%argmem = alloca inalloca i32
store i32 0, ptr %argmem
call void @inalloca_callee(ptr inalloca(i32) %argmem)
; This restore cannot be deleted, the restore below does not make it dead.
call void @llvm.stackrestore(ptr %save1)
; FIXME: We should be able to remove this save/restore pair, but we don't.
%save2 = call ptr @llvm.stacksave()
store i32 0, ptr @glob
call void @llvm.stackrestore(ptr %save2)
%i1 = add i32 1, %i
%done = icmp eq i32 %i1, %c
br i1 %done, label %loop, label %return
return:
ret void
}
; CHECK-LABEL: define void @test3(
; CHECK: loop:
; CHECK: %i = phi i32 [ 0, %entry ], [ %i1, %loop ]
; CHECK: %save1 = call ptr @llvm.stacksave.p0()
; CHECK: %argmem = alloca inalloca i32
; CHECK: store i32 0, ptr %argmem
; CHECK: call void @inalloca_callee(ptr {{.*}} inalloca(i32) %argmem)
; CHECK: call void @llvm.stackrestore.p0(ptr %save1)
; CHECK: br i1 %done, label %loop, label %return
; CHECK: ret void
define i32 @test4(i32 %m, ptr %a, ptr %b) {
entry:
br label %for.body
for.body:
%x.012 = phi i32 [ 0, %entry ], [ %add2, %for.body ]
%i.011 = phi i32 [ 0, %entry ], [ %inc, %for.body ]
%0 = call ptr @llvm.stacksave()
%load1 = load i32, ptr %a, align 4
%mul1 = mul nsw i32 %load1, %m
%add1 = add nsw i32 %mul1, %x.012
call void @llvm.stackrestore(ptr %0)
%load2 = load i32, ptr %b, align 4
%mul2 = mul nsw i32 %load2, %m
%add2 = add nsw i32 %mul2, %add1
call void @llvm.stackrestore(ptr %0)
%inc = add nuw nsw i32 %i.011, 1
%exitcond.not = icmp eq i32 %inc, 100
br i1 %exitcond.not, label %for.cond.cleanup, label %for.body
for.cond.cleanup:
ret i32 %add2
}
; CHECK-LABEL: define i32 @test4(
; CHECK-NOT: call void @llvm.stackrestore
; CHECK: ret i32 %add2