Index: lib/Support/APInt.cpp =================================================================== --- lib/Support/APInt.cpp +++ lib/Support/APInt.cpp @@ -1914,12 +1914,24 @@ } APInt APInt::umul_ov(const APInt &RHS, bool &Overflow) const { - APInt Res = *this * RHS; - - if (*this != 0 && RHS != 0) - Overflow = Res.udiv(RHS) != *this || Res.udiv(*this) != RHS; - else + unsigned LZ = countLeadingZeros() + RHS.countLeadingZeros(); + if (LZ >= BitWidth) { Overflow = false; + return operator*(RHS); + } + if (LZ + 2 <= BitWidth) { + Overflow = true; + return operator*(RHS); + } + + APInt Res = lshr(1) * RHS; + Overflow = Res.isNegative(); + Res <<= 1; + if ((*this)[0]) { + Res += RHS; + if (Res.ult(RHS)) + Overflow = true; + } return Res; } Index: unittests/ADT/APIntTest.cpp =================================================================== --- unittests/ADT/APIntTest.cpp +++ unittests/ADT/APIntTest.cpp @@ -2381,6 +2381,42 @@ } } +TEST(APIntTest, umul_ov) { + const std::pair Overflows[] = { + {0x8000000000000000, 2}, + {0x5555555555555556, 3}, + {4294967296, 4294967296}, + {4294967295, 4294967298}, + }; + const std::pair NonOverflows[] = { + {0x7fffffffffffffff, 2}, + {0x5555555555555555, 3}, + {4294967295, 4294967297}, + }; + + bool Overflow; + for (auto &X : Overflows) { + APInt A(64, X.first); + APInt B(64, X.second); + (void)A.umul_ov(B, Overflow); + EXPECT_TRUE(Overflow); + } + for (auto &X : NonOverflows) { + APInt A(64, X.first); + APInt B(64, X.second); + (void)A.umul_ov(B, Overflow); + EXPECT_FALSE(Overflow); + } + + for (unsigned Bits = 1; Bits <= 5; ++Bits) + for (unsigned A = 0; A != 1u << Bits; ++A) + for (unsigned B = 0; B != 1u << Bits; ++B) { + APInt C = APInt(Bits, A).umul_ov(APInt(Bits, B), Overflow); + APInt D = APInt(2 * Bits, A) * APInt(2 * Bits, B); + EXPECT_TRUE((C != D) == Overflow); + } +} + TEST(APIntTest, SolveQuadraticEquationWrap) { // Verify that "Solution" is the first non-negative integer that solves // Ax^2 + Bx + C = "0 or overflow", i.e. that it is a correct solution