Index: llvm/include/llvm/ADT/APFloat.h =================================================================== --- llvm/include/llvm/ADT/APFloat.h +++ llvm/include/llvm/ADT/APFloat.h @@ -393,6 +393,10 @@ /// magnitude in the current semantics. bool isSmallest() const; + /// Returns if this is the smallest (by magnitude) normalized finite number in + /// the given semantics. + bool isSmallestNormalized() const; + /// Returns true if and only if the number has the largest possible finite /// magnitude in the current semantics. bool isLargest() const; @@ -517,6 +521,7 @@ bool isSignificandAllOnesExceptLSB() const; /// Return true if the significand excluding the integral bit is all zeros. bool isSignificandAllZeros() const; + bool isSignificandAllZerosExceptMSB() const; /// @} @@ -694,6 +699,7 @@ bool isDenormal() const; bool isSmallest() const; + bool isSmallestNormalized() const; bool isLargest() const; bool isInteger() const; @@ -1245,6 +1251,10 @@ bool isInteger() const { APFLOAT_DISPATCH_ON_SEMANTICS(isInteger()); } bool isIEEE() const { return usesLayout(getSemantics()); } + bool isSmallestNormalized() const { + APFLOAT_DISPATCH_ON_SEMANTICS(isSmallestNormalized()); + } + APFloat &operator=(const APFloat &RHS) = default; APFloat &operator=(APFloat &&RHS) = default; Index: llvm/include/llvm/IR/PatternMatch.h =================================================================== --- llvm/include/llvm/IR/PatternMatch.h +++ llvm/include/llvm/IR/PatternMatch.h @@ -625,6 +625,14 @@ /// For vectors, this includes constants with undefined elements. inline cstfp_pred_ty m_Inf() { return cstfp_pred_ty(); } +struct is_smallest_normalized { + bool isValue(const APFloat &C) { return C.isSmallestNormalized(); } +}; + +inline cstfp_pred_ty m_SmallestNormalized() { + return cstfp_pred_ty(); +} + struct is_noninf { bool isValue(const APFloat &C) { return !C.isInfinity(); } }; Index: llvm/lib/Support/APFloat.cpp =================================================================== --- llvm/lib/Support/APFloat.cpp +++ llvm/lib/Support/APFloat.cpp @@ -883,6 +883,11 @@ significandMSB() == 0; } +bool IEEEFloat::isSmallestNormalized() const { + return getCategory() == fcNormal && exponent == semantics->minExponent && + isSignificandAllZerosExceptMSB(); +} + bool IEEEFloat::isSignificandAllOnes() const { // Test if the significand excluding the integral bit is all ones. This allows // us to test for binade boundaries. @@ -955,6 +960,21 @@ return true; } +bool IEEEFloat::isSignificandAllZerosExceptMSB() const { + const integerPart *Parts = significandParts(); + const unsigned PartCount = partCountForBits(semantics->precision); + + for (unsigned i = 0; i < PartCount - 1; i++) { + if (Parts[i]) + return false; + } + + const unsigned NumHighBits = + PartCount * integerPartWidth - semantics->precision + 1; + return Parts[PartCount - 1] == integerPart(1) + << (integerPartWidth - NumHighBits); +} + bool IEEEFloat::isLargest() const { if (semantics->nonFiniteBehavior == fltNonfiniteBehavior::NanOnly) { // The largest number by magnitude in our format will be the floating point @@ -4993,6 +5013,14 @@ return Tmp.compare(*this) == cmpEqual; } +bool DoubleAPFloat::isSmallestNormalized() const { + if (!Floats[1].isPosZero()) + return false; + uint64_t SignBit = static_cast(isNegative()) << 63; + APFloat LoHalf(semIEEEdouble, APInt(64, 0x0360000000000000ull | SignBit)); + return Floats[0] == LoHalf; +} + bool DoubleAPFloat::isLargest() const { if (getCategory() != fcNormal) return false; Index: llvm/unittests/ADT/APFloatTest.cpp =================================================================== --- llvm/unittests/ADT/APFloatTest.cpp +++ llvm/unittests/ADT/APFloatTest.cpp @@ -664,6 +664,57 @@ } } +TEST(APFloatTest, IsSmallestNormalized) { + for (unsigned I = 0; I != APFloat::S_MaxSemantics + 1; ++I) { + const fltSemantics &Semantics = + APFloat::EnumToSemantics(static_cast(I)); + + EXPECT_FALSE(APFloat::getZero(Semantics, false).isSmallestNormalized()); + EXPECT_FALSE(APFloat::getZero(Semantics, true).isSmallestNormalized()); + + EXPECT_FALSE(APFloat::getInf(Semantics, false).isSmallestNormalized()); + EXPECT_FALSE(APFloat::getInf(Semantics, true).isSmallestNormalized()); + + EXPECT_FALSE(APFloat::getQNaN(Semantics).isSmallestNormalized()); + EXPECT_FALSE(APFloat::getSNaN(Semantics).isSmallestNormalized()); + + EXPECT_FALSE(APFloat::getLargest(Semantics).isSmallestNormalized()); + EXPECT_FALSE(APFloat::getLargest(Semantics, true).isSmallestNormalized()); + + EXPECT_FALSE(APFloat::getSmallest(Semantics).isSmallestNormalized()); + EXPECT_FALSE(APFloat::getSmallest(Semantics, true).isSmallestNormalized()); + + EXPECT_FALSE(APFloat::getAllOnesValue(Semantics).isSmallestNormalized()); + + APFloat PosSmallestNormalized = + APFloat::getSmallestNormalized(Semantics, false); + APFloat NegSmallestNormalized = + APFloat::getSmallestNormalized(Semantics, true); + EXPECT_TRUE(PosSmallestNormalized.isSmallestNormalized()); + EXPECT_TRUE(NegSmallestNormalized.isSmallestNormalized()); + + for (APFloat *Val : {&PosSmallestNormalized, &NegSmallestNormalized}) { + bool OldSign = Val->isNegative(); + + // Step down, make sure it's still not smallest normalized. + EXPECT_EQ(APFloat::opOK, Val->next(false)); + EXPECT_EQ(OldSign, Val->isNegative()); + EXPECT_FALSE(Val->isSmallestNormalized()); + EXPECT_EQ(OldSign, Val->isNegative()); + + // Step back up should restore it to being smallest normalized. + EXPECT_EQ(APFloat::opOK, Val->next(true)); + EXPECT_TRUE(Val->isSmallestNormalized()); + EXPECT_EQ(OldSign, Val->isNegative()); + + // Step beyond should no longer smallest normalized. + EXPECT_EQ(APFloat::opOK, Val->next(true)); + EXPECT_FALSE(Val->isSmallestNormalized()); + EXPECT_EQ(OldSign, Val->isNegative()); + } + } +} + TEST(APFloatTest, Zero) { EXPECT_EQ(0.0f, APFloat(0.0f).convertToFloat()); EXPECT_EQ(-0.0f, APFloat(-0.0f).convertToFloat()); Index: llvm/unittests/IR/PatternMatch.cpp =================================================================== --- llvm/unittests/IR/PatternMatch.cpp +++ llvm/unittests/IR/PatternMatch.cpp @@ -1519,6 +1519,19 @@ EXPECT_TRUE(match(CF32Pi, cstfp_pred_ty>())); EXPECT_FALSE(match(CF32Pi, cstfp_pred_ty>())); + APFloat PosSmallestNormal = + APFloat::getSmallestNormalized(APFloat::IEEEsingle()); + APFloat NegSmallestNormal = + APFloat::getSmallestNormalized(APFloat::IEEEsingle(), true); + Constant *CPosSmallestNormal = ConstantFP::get(F32Ty, PosSmallestNormal); + Constant *CNegSmallestNormal = ConstantFP::get(F32Ty, NegSmallestNormal); + + EXPECT_TRUE(match(CPosSmallestNormal, m_SmallestNormalized())); + EXPECT_TRUE(match(CNegSmallestNormal, m_SmallestNormalized())); + EXPECT_FALSE(match(CF32NaN, m_SmallestNormalized())); + EXPECT_FALSE(match(CF32Zero, m_SmallestNormalized())); + EXPECT_FALSE(match(CF32Pi, m_SmallestNormalized())); + auto FixedEC = ElementCount::getFixed(4); auto ScalableEC = ElementCount::getScalable(4);