diff options
author | Nick Lewycky <nicholas@mxc.ca> | 2008-02-18 22:48:05 +0000 |
---|---|---|
committer | Nick Lewycky <nicholas@mxc.ca> | 2008-02-18 22:48:05 +0000 |
commit | fefd0202c9e5f5a3b12d713d321fe9844ab79b7c (patch) | |
tree | 70f7edc63f331a91d84cc9694c1379e68cdd53c3 /llvm/lib | |
parent | dec89733a77205f7d2c69ce12ea30e4aa3acfcee (diff) | |
download | bcm5719-llvm-fefd0202c9e5f5a3b12d713d321fe9844ab79b7c.tar.gz bcm5719-llvm-fefd0202c9e5f5a3b12d713d321fe9844ab79b7c.zip |
Correctly fold divide-by-constant, even when faced with overflow.
llvm-svn: 47287
Diffstat (limited to 'llvm/lib')
-rw-r--r-- | llvm/lib/Transforms/Scalar/InstructionCombining.cpp | 29 |
1 files changed, 27 insertions, 2 deletions
diff --git a/llvm/lib/Transforms/Scalar/InstructionCombining.cpp b/llvm/lib/Transforms/Scalar/InstructionCombining.cpp index f89d1f2aeee..5582f511340 100644 --- a/llvm/lib/Transforms/Scalar/InstructionCombining.cpp +++ b/llvm/lib/Transforms/Scalar/InstructionCombining.cpp @@ -605,6 +605,28 @@ static ConstantInt *Subtract(ConstantInt *C1, ConstantInt *C2) { static ConstantInt *Multiply(ConstantInt *C1, ConstantInt *C2) { return ConstantInt::get(C1->getValue() * C2->getValue()); } +/// MultiplyOverflows - True if the multiply can not be expressed in an int +/// this size. +static bool MultiplyOverflows(ConstantInt *C1, ConstantInt *C2, bool sign) { + uint32_t W = C1->getBitWidth(); + APInt LHSExt = C1->getValue(), RHSExt = C2->getValue(); + if (sign) { + LHSExt.sext(W * 2); + RHSExt.sext(W * 2); + } else { + LHSExt.zext(W * 2); + RHSExt.zext(W * 2); + } + + APInt MulExt = LHSExt * RHSExt; + + if (sign) { + APInt Min = APInt::getSignedMinValue(W).sext(W * 2); + APInt Max = APInt::getSignedMaxValue(W).sext(W * 2); + return MulExt.slt(Min) || MulExt.sgt(Max); + } else + return MulExt.ugt(APInt::getLowBitsSet(W * 2, W)); +} /// ComputeMaskedBits - Determine which of the bits specified in Mask are /// known to be either zero or one and return them in the KnownZero/KnownOne @@ -2632,8 +2654,11 @@ Instruction *InstCombiner::commonIDivTransforms(BinaryOperator &I) { if (Instruction *LHS = dyn_cast<Instruction>(Op0)) if (Instruction::BinaryOps(LHS->getOpcode()) == I.getOpcode()) if (ConstantInt *LHSRHS = dyn_cast<ConstantInt>(LHS->getOperand(1))) { - return BinaryOperator::create(I.getOpcode(), LHS->getOperand(0), - Multiply(RHS, LHSRHS)); + if (MultiplyOverflows(RHS, LHSRHS, I.getOpcode()==Instruction::SDiv)) + return ReplaceInstUsesWith(I, Constant::getNullValue(I.getType())); + else + return BinaryOperator::create(I.getOpcode(), LHS->getOperand(0), + Multiply(RHS, LHSRHS)); } if (!RHS->isZero()) { // avoid X udiv 0 |