diff options
| author | Bill Wendling <isanbard@gmail.com> | 2008-11-30 05:01:05 +0000 | 
|---|---|---|
| committer | Bill Wendling <isanbard@gmail.com> | 2008-11-30 05:01:05 +0000 | 
| commit | af200e9237cd81fbcba7080057d493941952835c (patch) | |
| tree | e3654c3ea8da486bbfad2de99d1585bfc1a632c4 /llvm | |
| parent | bd0f57821a7326716db85236ba93bd1c9dc44d28 (diff) | |
| download | bcm5719-llvm-af200e9237cd81fbcba7080057d493941952835c.tar.gz bcm5719-llvm-af200e9237cd81fbcba7080057d493941952835c.zip  | |
From Hacker's Delight:
"For signed integers, the determination of overflow of x*y is not so simple. If
x and y have the same sign, then overflow occurs iff xy > 2**31 - 1. If they
have opposite signs, then overflow occurs iff xy < -2**31."
In this case, x == -1.
llvm-svn: 60278
Diffstat (limited to 'llvm')
| -rw-r--r-- | llvm/lib/Transforms/Scalar/InstructionCombining.cpp | 18 | 
1 files changed, 10 insertions, 8 deletions
diff --git a/llvm/lib/Transforms/Scalar/InstructionCombining.cpp b/llvm/lib/Transforms/Scalar/InstructionCombining.cpp index 8461aa75e02..fce23739b66 100644 --- a/llvm/lib/Transforms/Scalar/InstructionCombining.cpp +++ b/llvm/lib/Transforms/Scalar/InstructionCombining.cpp @@ -2957,20 +2957,22 @@ Instruction *InstCombiner::visitSDiv(BinaryOperator &I) {        return BinaryOperator::CreateNeg(Op0);      ConstantInt *RHSNeg = cast<ConstantInt>(ConstantExpr::getNeg(RHS)); +    APInt RHSNegAPI(RHSNeg->getBitWidth(), RHSNeg->getSExtValue(), true); + +    APInt NegOne = -APInt(RHSNeg->getBitWidth(), 1, true); +    APInt TwoToExp(RHSNeg->getBitWidth(), 1 << (RHSNeg->getBitWidth() - 1), +                   true);      // -X/C -> X/-C, if and only if negation doesn't overflow. -    if ((RHS->getSExtValue() < 0 && -         RHS->getSExtValue() < RHSNeg->getSExtValue()) || -        (RHS->getSExtValue() > 0 && -         RHS->getSExtValue() > RHSNeg->getSExtValue())) { +    if ((RHS->getSExtValue() < 0 && RHSNegAPI.slt(TwoToExp - 1)) || +        (RHS->getSExtValue() > 0 && RHSNegAPI.sgt(TwoToExp * NegOne))) {        if (Value *LHSNeg = dyn_castNegVal(Op0)) {          if (ConstantInt *CI = dyn_cast<ConstantInt>(LHSNeg)) {            ConstantInt *CINeg = cast<ConstantInt>(ConstantExpr::getNeg(CI)); +          APInt CINegAPI(CINeg->getBitWidth(), CINeg->getSExtValue(), true); -          if ((CI->getSExtValue() < 0 && -               CI->getSExtValue() < CINeg->getSExtValue()) || -              (CI->getSExtValue() > 0 && -               CI->getSExtValue() > CINeg->getSExtValue())) +          if ((CI->getSExtValue() < 0 && CINegAPI.slt(TwoToExp - 1)) || +              (CI->getSExtValue() > 0 && CINegAPI.sgt(TwoToExp * NegOne)))              return BinaryOperator::CreateSDiv(LHSNeg,                                                ConstantExpr::getNeg(RHS));          }  | 

