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