IndVarSimplify: check if loop invariant expansion can trap

IndVarSimplify is willing to move divide instructions outside of their
loop bodies if they are invariant of the loop.  However, it may not be
safe to expand them if we do not know if they can trap.

Instead, check to see if it is not safe to expand the instruction and
skip the expansion.

This fixes PR16041.

Testcase by Rafael Ávila de Espíndola.


git-svn-id: https://llvm.org/svn/llvm-project/llvm/trunk@183239 91177308-0d34-0410-b5e6-96231b3b80d8
This commit is contained in:
David Majnemer
2013-06-04 17:51:58 +00:00
parent 35e7751af4
commit 5a57dbef33
2 changed files with 33 additions and 1 deletions

View File

@@ -532,7 +532,7 @@ void IndVarSimplify::RewriteLoopExitValues(Loop *L, SCEVExpander &Rewriter) {
// and varies predictably *inside* the loop. Evaluate the value it
// contains when the loop exits, if possible.
const SCEV *ExitValue = SE->getSCEVAtScope(Inst, L->getParentLoop());
if (!SE->isLoopInvariant(ExitValue, L))
if (!SE->isLoopInvariant(ExitValue, L) || !isSafeToExpand(ExitValue))
continue;
// Computing the value outside of the loop brings no benefit if :

View File

@@ -0,0 +1,32 @@
; RUN: opt -indvars -S < %s | FileCheck %s
@b = common global i32 0, align 4
define i32 @foo(i32 %x, i1 %y) {
bb0:
br label %bb1
bb1:
br i1 %y, label %bb14, label %bb8
bb8:
%i = phi i64 [ %i.next, %bb8 ], [ 0, %bb1 ]
%i.next = add i64 %i, 1
%div = udiv i32 1, %x
%c = icmp eq i64 %i.next, 6
br i1 %c, label %bb11, label %bb8
bb11:
br i1 %y, label %bb1, label %bb13
bb13:
store i32 %div, i32* @b, align 4
br label %bb14
bb14:
ret i32 0
}
; CHECK: @foo
; CHECK: bb8:
; CHECK: udiv