diff options
author | Duncan Sands <baldrick@free.fr> | 2011-02-02 20:52:00 +0000 |
---|---|---|
committer | Duncan Sands <baldrick@free.fr> | 2011-02-02 20:52:00 +0000 |
commit | 5747abab10b00a7f84458cc3fb41715b4790239a (patch) | |
tree | 813765a7a5f7d04b89b4df8bdb48d26592b59f2b | |
parent | 09b2331cf7a03a8f372edfb5123048a86d8b4df7 (diff) | |
download | bcm5719-llvm-5747abab10b00a7f84458cc3fb41715b4790239a.tar.gz bcm5719-llvm-5747abab10b00a7f84458cc3fb41715b4790239a.zip |
Reenable the transform "(X*Y)/Y->X" when the multiplication is known not to
overflow (nsw flag), which was disabled because it breaks 254.gap. I have
informed the GAP authors of the mistake in their code, and arranged for the
testsuite to use -fwrapv when compiling this benchmark.
llvm-svn: 124746
-rw-r--r-- | llvm/lib/Analysis/InstructionSimplify.cpp | 10 | ||||
-rw-r--r-- | llvm/test/Transforms/InstSimplify/2010-12-20-Reassociate.ll | 4 |
2 files changed, 9 insertions, 5 deletions
diff --git a/llvm/lib/Analysis/InstructionSimplify.cpp b/llvm/lib/Analysis/InstructionSimplify.cpp index 6ca4dddcf36..4b9b648aa6c 100644 --- a/llvm/lib/Analysis/InstructionSimplify.cpp +++ b/llvm/lib/Analysis/InstructionSimplify.cpp @@ -798,11 +798,11 @@ static Value *SimplifyDiv(unsigned Opcode, Value *Op0, Value *Op1, Value *X = 0, *Y = 0; if (match(Op0, m_Mul(m_Value(X), m_Value(Y))) && (X == Op1 || Y == Op1)) { if (Y != Op1) std::swap(X, Y); // Ensure expression is (X * Y) / Y, Y = Op1 -// BinaryOperator *Mul = cast<BinaryOperator>(Op0); -// // If the Mul knows it does not overflow, then we are good to go. -// if ((isSigned && Mul->hasNoSignedWrap()) || -// (!isSigned && Mul->hasNoUnsignedWrap())) -// return X; + BinaryOperator *Mul = cast<BinaryOperator>(Op0); + // If the Mul knows it does not overflow, then we are good to go. + if ((isSigned && Mul->hasNoSignedWrap()) || + (!isSigned && Mul->hasNoUnsignedWrap())) + return X; // If X has the form X = A / Y then X * Y cannot overflow. if (BinaryOperator *Div = dyn_cast<BinaryOperator>(X)) if (Div->getOpcode() == Opcode && Div->getOperand(1) == Y) diff --git a/llvm/test/Transforms/InstSimplify/2010-12-20-Reassociate.ll b/llvm/test/Transforms/InstSimplify/2010-12-20-Reassociate.ll index c0ae257dcdc..928442ac56c 100644 --- a/llvm/test/Transforms/InstSimplify/2010-12-20-Reassociate.ll +++ b/llvm/test/Transforms/InstSimplify/2010-12-20-Reassociate.ll @@ -92,10 +92,12 @@ define i32 @sub3(i32 %x, i32 %y) { } define i32 @sdiv1(i32 %x, i32 %y) { +; CHECK: @sdiv1 ; (no overflow X * Y) / Y -> X %mul = mul nsw i32 %x, %y %r = sdiv i32 %mul, %y ret i32 %r +; CHECK: ret i32 %x } define i32 @sdiv2(i32 %x, i32 %y) { @@ -136,10 +138,12 @@ define i32 @sdiv5(i32 %x, i32 %y) { } define i32 @udiv1(i32 %x, i32 %y) { +; CHECK: @udiv1 ; (no overflow X * Y) / Y -> X %mul = mul nuw i32 %x, %y %r = udiv i32 %mul, %y ret i32 %r +; CHECK: ret i32 %x } define i32 @udiv2(i32 %x, i32 %y) { |