From 2edb926880d2a03c4280405fc53c37ca5cf1e060 Mon Sep 17 00:00:00 2001 From: Chris Lattner Date: Wed, 13 Oct 2010 23:46:56 +0000 Subject: [PATCH] move logic for computing signed integer overflow when constant folding into APInt. llvm-svn: 116453 --- clang/lib/Lex/PPExpressions.cpp | 49 +++++++++++++++------------------ 1 file changed, 22 insertions(+), 27 deletions(-) diff --git a/clang/lib/Lex/PPExpressions.cpp b/clang/lib/Lex/PPExpressions.cpp index 163e869400aa..79ed8677da7a 100644 --- a/clang/lib/Lex/PPExpressions.cpp +++ b/clang/lib/Lex/PPExpressions.cpp @@ -519,7 +519,6 @@ static bool EvaluateDirectiveSubExpr(PPValue &LHS, unsigned MinPrec, RHS.Val.setIsUnsigned(Res.isUnsigned()); } - // FIXME: All of these should detect and report overflow?? bool Overflow = false; switch (Operator) { default: assert(0 && "Unknown operator token!"); @@ -534,9 +533,10 @@ static bool EvaluateDirectiveSubExpr(PPValue &LHS, unsigned MinPrec, break; case tok::slash: if (RHS.Val != 0) { - Res = LHS.Val / RHS.Val; - if (LHS.Val.isSigned()) // MININT/-1 --> overflow. - Overflow = LHS.Val.isMinSignedValue() && RHS.Val.isAllOnesValue(); + if (LHS.Val.isSigned()) + Res = llvm::APSInt(LHS.Val.sdiv_ov(RHS.Val, Overflow), false); + else + Res = LHS.Val / RHS.Val; } else if (ValueLive) { PP.Diag(OpLoc, diag::err_pp_division_by_zero) << LHS.getRange() << RHS.getRange(); @@ -545,23 +545,22 @@ static bool EvaluateDirectiveSubExpr(PPValue &LHS, unsigned MinPrec, break; case tok::star: - Res = LHS.Val * RHS.Val; - if (Res.isSigned() && LHS.Val != 0 && RHS.Val != 0) - Overflow = Res/RHS.Val != LHS.Val || Res/LHS.Val != RHS.Val; + if (Res.isSigned()) + Res = llvm::APSInt(LHS.Val.smul_ov(RHS.Val, Overflow), false); + else + Res = LHS.Val * RHS.Val; break; case tok::lessless: { // Determine whether overflow is about to happen. unsigned ShAmt = static_cast(RHS.Val.getLimitedValue()); - if (ShAmt >= LHS.Val.getBitWidth()) - Overflow = true, ShAmt = LHS.Val.getBitWidth()-1; - else if (LHS.isUnsigned()) - Overflow = false; - else if (LHS.Val.isNonNegative()) // Don't allow sign change. - Overflow = ShAmt >= LHS.Val.countLeadingZeros(); - else - Overflow = ShAmt >= LHS.Val.countLeadingOnes(); - - Res = LHS.Val << ShAmt; + if (LHS.isUnsigned()) { + Overflow = ShAmt >= LHS.Val.getBitWidth(); + if (Overflow) + ShAmt = LHS.Val.getBitWidth()-1; + Res = LHS.Val << ShAmt; + } else { + Res = llvm::APSInt(LHS.Val.sshl_ov(ShAmt, Overflow), false); + } break; } case tok::greatergreater: { @@ -573,20 +572,16 @@ static bool EvaluateDirectiveSubExpr(PPValue &LHS, unsigned MinPrec, break; } case tok::plus: - Res = LHS.Val + RHS.Val; if (LHS.isUnsigned()) - Overflow = false; - else if (LHS.Val.isNonNegative() == RHS.Val.isNonNegative() && - Res.isNonNegative() != LHS.Val.isNonNegative()) - Overflow = true; // Overflow for signed addition. + Res = LHS.Val + RHS.Val; + else + Res = llvm::APSInt(LHS.Val.sadd_ov(RHS.Val, Overflow), false); break; case tok::minus: - Res = LHS.Val - RHS.Val; if (LHS.isUnsigned()) - Overflow = false; - else if (LHS.Val.isNonNegative() != RHS.Val.isNonNegative() && - Res.isNonNegative() != LHS.Val.isNonNegative()) - Overflow = true; // Overflow for signed subtraction. + Res = LHS.Val - RHS.Val; + else + Res = llvm::APSInt(LHS.Val.ssub_ov(RHS.Val, Overflow), false); break; case tok::lessequal: Res = LHS.Val <= RHS.Val;