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