forked from OSchip/llvm-project
131 lines
3.6 KiB
LLVM
131 lines
3.6 KiB
LLVM
; RUN: opt -S -licm < %s | FileCheck %s
|
|
; RUN: opt -aa-pipeline=basic-aa -passes='require<aa>,require<targetir>,require<scalar-evolution>,require<opt-remark-emit>,loop(licm)' -S %s | FileCheck %s
|
|
|
|
declare void @use_nothrow(i64 %a) nounwind
|
|
declare void @use(i64 %a)
|
|
declare void @maythrow()
|
|
|
|
define void @nothrow(i64 %x, i64 %y, i1* %cond) {
|
|
; CHECK-LABEL: nothrow
|
|
; CHECK-LABEL: entry
|
|
; CHECK: %div = udiv i64 %x, %y
|
|
; CHECK-LABEL: loop
|
|
; CHECK: call void @use_nothrow(i64 %div)
|
|
entry:
|
|
br label %loop
|
|
|
|
loop: ; preds = %entry, %for.inc
|
|
%div = udiv i64 %x, %y
|
|
br label %loop2
|
|
|
|
loop2:
|
|
call void @use_nothrow(i64 %div)
|
|
br label %loop
|
|
}
|
|
|
|
; The udiv is guarantee to execute if the loop is
|
|
define void @throw_header_after(i64 %x, i64 %y, i1* %cond) {
|
|
; CHECK-LABEL: throw_header_after
|
|
; CHECK: %div = udiv i64 %x, %y
|
|
; CHECK-LABEL: loop
|
|
; CHECK: call void @use(i64 %div)
|
|
entry:
|
|
br label %loop
|
|
|
|
loop: ; preds = %entry, %for.inc
|
|
%div = udiv i64 %x, %y
|
|
call void @use(i64 %div)
|
|
br label %loop
|
|
}
|
|
define void @throw_header_after_rec(i64* %xp, i64* %yp, i1* %cond) {
|
|
; CHECK-LABEL: throw_header_after_rec
|
|
; CHECK: %x = load i64, i64* %xp
|
|
; CHECK: %y = load i64, i64* %yp
|
|
; CHECK: %div = udiv i64 %x, %y
|
|
; CHECK-LABEL: loop
|
|
; CHECK: call void @use(i64 %div)
|
|
entry:
|
|
br label %loop
|
|
|
|
loop: ; preds = %entry, %for.inc
|
|
%x = load i64, i64* %xp
|
|
%y = load i64, i64* %yp
|
|
%div = udiv i64 %x, %y
|
|
call void @use(i64 %div) readonly
|
|
br label %loop
|
|
}
|
|
|
|
; Similiar to the above, but the hoistable instruction (%y in this case)
|
|
; happens not to be the first instruction in the block.
|
|
define void @throw_header_after_nonfirst(i64* %xp, i64* %yp, i1* %cond) {
|
|
; CHECK-LABEL: throw_header_after_nonfirst
|
|
; CHECK: %y = load i64, i64* %yp
|
|
; CHECK-LABEL: loop
|
|
; CHECK: %x = load i64, i64* %gep
|
|
; CHECK: %div = udiv i64 %x, %y
|
|
; CHECK: call void @use(i64 %div)
|
|
entry:
|
|
br label %loop
|
|
|
|
loop: ; preds = %entry, %for.inc
|
|
%iv = phi i64 [0, %entry], [%div, %loop]
|
|
%gep = getelementptr i64, i64* %xp, i64 %iv
|
|
%x = load i64, i64* %gep
|
|
%y = load i64, i64* %yp
|
|
%div = udiv i64 %x, %y
|
|
call void @use(i64 %div) readonly
|
|
br label %loop
|
|
}
|
|
|
|
; Negative test
|
|
define void @throw_header_before(i64 %x, i64 %y, i1* %cond) {
|
|
; CHECK-LABEL: throw_header_before
|
|
; CHECK-LABEL: loop
|
|
; CHECK: %div = udiv i64 %x, %y
|
|
; CHECK: call void @use(i64 %div)
|
|
entry:
|
|
br label %loop
|
|
|
|
loop: ; preds = %entry, %for.inc
|
|
call void @maythrow()
|
|
%div = udiv i64 %x, %y
|
|
call void @use(i64 %div)
|
|
br label %loop
|
|
}
|
|
|
|
; The header is known no throw, but the loop is not. We can
|
|
; still lift out of the header.
|
|
define void @nothrow_header(i64 %x, i64 %y, i1 %cond) {
|
|
; CHECK-LABEL: nothrow_header
|
|
; CHECK-LABEL: entry
|
|
; CHECK: %div = udiv i64 %x, %y
|
|
; CHECK-LABEL: loop
|
|
; CHECK: call void @use(i64 %div)
|
|
entry:
|
|
br label %loop
|
|
loop: ; preds = %entry, %for.inc
|
|
%div = udiv i64 %x, %y
|
|
br i1 %cond, label %loop-if, label %exit
|
|
loop-if:
|
|
call void @use(i64 %div)
|
|
br label %loop
|
|
exit:
|
|
ret void
|
|
}
|
|
; Negative test - can't move out of throwing block
|
|
define void @nothrow_header_neg(i64 %x, i64 %y, i1 %cond) {
|
|
; CHECK-LABEL: nothrow_header_neg
|
|
; CHECK-LABEL: entry
|
|
; CHECK-LABEL: loop
|
|
; CHECK: %div = udiv i64 %x, %y
|
|
; CHECK: call void @use(i64 %div)
|
|
entry:
|
|
br label %loop
|
|
loop: ; preds = %entry, %for.inc
|
|
br label %loop-if
|
|
loop-if:
|
|
%div = udiv i64 %x, %y
|
|
call void @use(i64 %div)
|
|
br label %loop
|
|
}
|