diff --git a/clang/include/clang/StaticAnalyzer/Core/PathSensitive/APSIntType.h b/clang/include/clang/StaticAnalyzer/Core/PathSensitive/APSIntType.h --- a/clang/include/clang/StaticAnalyzer/Core/PathSensitive/APSIntType.h +++ b/clang/include/clang/StaticAnalyzer/Core/PathSensitive/APSIntType.h @@ -21,8 +21,8 @@ bool IsUnsigned; public: - APSIntType(uint32_t Width, bool Unsigned) - : BitWidth(Width), IsUnsigned(Unsigned) {} + constexpr APSIntType(uint32_t Width, bool Unsigned) + : BitWidth(Width), IsUnsigned(Unsigned) {} /* implicit */ APSIntType(const llvm::APSInt &Value) : BitWidth(Value.getBitWidth()), IsUnsigned(Value.isUnsigned()) {} diff --git a/clang/include/clang/StaticAnalyzer/Core/PathSensitive/RangedConstraintManager.h b/clang/include/clang/StaticAnalyzer/Core/PathSensitive/RangedConstraintManager.h --- a/clang/include/clang/StaticAnalyzer/Core/PathSensitive/RangedConstraintManager.h +++ b/clang/include/clang/StaticAnalyzer/Core/PathSensitive/RangedConstraintManager.h @@ -237,6 +237,29 @@ /// Complexity: O(N) /// where N = size(What) RangeSet negate(RangeSet What); + /// Performs promotions, truncations and conversions of the given set. + /// + /// This function is optimized for each of the six cast cases: + /// - noop + /// - conversion + /// - truncation + /// - truncation-conversion + /// - promotion + /// - promotion-conversion + /// + /// NOTE: This function is NOT self-inverse for truncations, because of + /// the higher bits loss: + /// - castTo(castTo(OrigRangeOfInt, char), int) != OrigRangeOfInt. + /// - castTo(castTo(OrigRangeOfChar, int), char) == OrigRangeOfChar. + /// But it is self-inverse for all the rest casts. + /// + /// Complexity: + /// - Noop O(1); + /// - Truncation O(N^2); + /// - Another case O(N); + /// where N = size(What) + RangeSet castTo(RangeSet What, APSIntType Ty); + RangeSet castTo(RangeSet What, QualType T); /// Return associated value factory. BasicValueFactory &getValueFactory() const { return ValueFactory; } @@ -252,6 +275,22 @@ /// containers are persistent (created via BasicValueFactory::getValue). ContainerType unite(const ContainerType &LHS, const ContainerType &RHS); + /// This is a helper function for `castTo` method. Implies not to be used + /// separately. + /// Performs a truncation case of a cast operation. + ContainerType truncateTo(RangeSet What, APSIntType Ty); + + /// This is a helper function for `castTo` method. Implies not to be used + /// separately. + /// Performs a conversion case and a promotion-conversion case for signeds + /// of a cast operation. + ContainerType convertTo(RangeSet What, APSIntType Ty); + + /// This is a helper function for `castTo` method. Implies not to be used + /// separately. + /// Performs a promotion for unsigneds only. + ContainerType promoteTo(RangeSet What, APSIntType Ty); + // Many operations include producing new APSInt values and that's why // we need this factory. BasicValueFactory &ValueFactory; @@ -303,6 +342,10 @@ /// Complexity: O(1) const llvm::APSInt &getMaxValue() const; + bool isUnsigned() const; + uint32_t getBitWidth() const; + APSIntType getAPSIntType() const; + /// Test whether the given point is contained by any of the ranges. /// /// Complexity: O(logN) diff --git a/clang/lib/StaticAnalyzer/Core/RangeConstraintManager.cpp b/clang/lib/StaticAnalyzer/Core/RangeConstraintManager.cpp --- a/clang/lib/StaticAnalyzer/Core/RangeConstraintManager.cpp +++ b/clang/lib/StaticAnalyzer/Core/RangeConstraintManager.cpp @@ -353,6 +353,21 @@ return std::prev(end())->To(); } +bool clang::ento::RangeSet::isUnsigned() const { + assert(!isEmpty()); + return begin()->From().isUnsigned(); +} + +uint32_t clang::ento::RangeSet::getBitWidth() const { + assert(!isEmpty()); + return begin()->From().getBitWidth(); +} + +APSIntType clang::ento::RangeSet::getAPSIntType() const { + assert(!isEmpty()); + return APSIntType(begin()->From()); +} + bool RangeSet::containsImpl(llvm::APSInt &Point) const { if (isEmpty() || !pin(Point)) return false; @@ -655,6 +670,181 @@ return makePersistent(std::move(Result)); } +// Convert range set to the given integral type using truncation and promotion. +// This works similar to APSIntType::apply function but for the range set. +RangeSet RangeSet::Factory::castTo(RangeSet What, APSIntType Ty) { + // Set is empty or NOOP (aka cast to the same type). + if (What.isEmpty() || What.getAPSIntType() == Ty) + return What; + + const bool IsConversion = What.isUnsigned() != Ty.isUnsigned(); + const bool IsTruncation = What.getBitWidth() > Ty.getBitWidth(); + const bool IsPromotion = What.getBitWidth() < Ty.getBitWidth(); + + if (IsTruncation) + return makePersistent(truncateTo(What, Ty)); + + // Here we handle 2 cases: + // - IsConversion && !IsPromotion. + // In this case we handle changing a sign with same bitwidth: char -> uchar, + // uint -> int. Here we convert negatives to positives and positives which + // is out of range to negatives. We use convertTo function for that. + // - IsConversion && IsPromotion && !What.isUnsigned(). + // In this case we handle changing a sign from signeds to unsigneds with + // higher bitwidth: char -> uint, int-> uint64. The point is that we also + // need convert negatives to positives and use convertTo function as well. + // For example, we don't need such a convertion when converting unsigned to + // signed with higher bitwidth, because all the values of unsigned is valid + // for the such signed. + if (IsConversion && (!IsPromotion || !What.isUnsigned())) + return makePersistent(convertTo(What, Ty)); + + assert(IsPromotion && "Only promotion operation from unsigneds left."); + return makePersistent(promoteTo(What, Ty)); +} + +RangeSet RangeSet::Factory::castTo(RangeSet What, QualType T) { + assert(T->isIntegralOrEnumerationType() && "T shall be an integral type."); + return castTo(What, ValueFactory.getAPSIntType(T)); +} + +RangeSet::ContainerType RangeSet::Factory::truncateTo(RangeSet What, + APSIntType Ty) { + using llvm::APInt; + using llvm::APSInt; + ContainerType Result; + ContainerType Dummy; + // CastRangeSize is an amount of all possible values of cast type. + // Example: `char` has 256 values; `short` has 65536 values. + // But in fact we use `amount of values` - 1, because + // we can't keep `amount of values of UINT64` inside uint64_t. + // E.g. 256 is an amount of all possible values of `char` and we can't keep + // it inside `char`. + // And it's OK, it's enough to do correct calculations. + uint64_t CastRangeSize = APInt::getMaxValue(Ty.getBitWidth()).getZExtValue(); + for (const Range &R : What) { + // Get bounds of the given range. + APSInt FromInt = R.From(); + APSInt ToInt = R.To(); + // CurrentRangeSize is an amount of all possible values of the current + // range minus one. + uint64_t CurrentRangeSize = (ToInt - FromInt).getZExtValue(); + // This is an optimization for a specific case when this Range covers + // the whole range of the target type. + Dummy.clear(); + if (CurrentRangeSize >= CastRangeSize) { + Dummy.emplace_back(ValueFactory.getMinValue(Ty), + ValueFactory.getMaxValue(Ty)); + Result = std::move(Dummy); + break; + } + // Cast the bounds. + Ty.apply(FromInt); + Ty.apply(ToInt); + const APSInt &PersistentFrom = ValueFactory.getValue(FromInt); + const APSInt &PersistentTo = ValueFactory.getValue(ToInt); + if (FromInt > ToInt) { + Dummy.emplace_back(ValueFactory.getMinValue(Ty), PersistentTo); + Dummy.emplace_back(PersistentFrom, ValueFactory.getMaxValue(Ty)); + } else + Dummy.emplace_back(PersistentFrom, PersistentTo); + // Every range retrieved after truncation potentialy has garbage values. + // So, we have to unite every next range with the previouses. + Result = unite(Result, Dummy); + } + + return Result; +} + +// Divide the convertion into two phases (presented as loops here). +// First phase(loop) works when casted values go in ascending order. +// E.g. char{1,3,5,127} -> uint{1,3,5,127} +// Interrupt the first phase and go to second one when casted values start +// go in descending order. That means that we crossed over the middle of +// the type value set (aka 0 for signeds and MAX/2+1 for unsigneds). +// For instance: +// 1: uchar{1,3,5,128,255} -> char{1,3,5,-128,-1} +// Here we put {1,3,5} to one array and {-128, -1} to another +// 2: char{-128,-127,-1,0,1,2} -> uchar{128,129,255,0,1,3} +// Here we put {128,129,255} to one array and {0,1,3} to another. +// After that we unite both arrays. +// NOTE: We don't just concatenate the arrays, because they may have +// adjacent ranges, e.g.: +// 1: char(-128, 127) -> uchar -> arr1(128, 255), arr2(0, 127) -> +// unite -> uchar(0, 255) +// 2: uchar(0, 1)U(254, 255) -> char -> arr1(0, 1), arr2(-2, -1) -> +// unite -> uchar(-2, 1) +RangeSet::ContainerType RangeSet::Factory::convertTo(RangeSet What, + APSIntType Ty) { + using llvm::APInt; + using llvm::APSInt; + using Bounds = std::pair; + ContainerType AscendArray; + ContainerType DescendArray; + auto CastRange = [Ty, &VF = ValueFactory](const Range &R) -> Bounds { + // Get bounds of the given range. + APSInt FromInt = R.From(); + APSInt ToInt = R.To(); + // Cast the bounds. + Ty.apply(FromInt); + Ty.apply(ToInt); + return {VF.getValue(FromInt), VF.getValue(ToInt)}; + }; + // Phase 1. Fill the first array. + APSInt LastConvertedInt = Ty.getMinValue(); + const auto *It = What.begin(); + const auto *E = What.end(); + while (It != E) { + Bounds NewBounds = CastRange(*(It++)); + // If values stop going acsending order, go to the second phase(loop). + if (NewBounds.first < LastConvertedInt) { + DescendArray.emplace_back(NewBounds.first, NewBounds.second); + break; + } + // If the range contains a midpoint, then split the range. + // E.g. char(-5, 5) -> uchar(251, 5) + // Here we shall add a range (251, 255) to the first array and (0, 5) to the + // second one. + if (NewBounds.first > NewBounds.second) { + DescendArray.emplace_back(ValueFactory.getMinValue(Ty), NewBounds.second); + AscendArray.emplace_back(NewBounds.first, ValueFactory.getMaxValue(Ty)); + } else + // Values are going acsending order. + AscendArray.emplace_back(NewBounds.first, NewBounds.second); + LastConvertedInt = NewBounds.first; + } + // Phase 2. Fill the second array. + while (It != E) { + Bounds NewBounds = CastRange(*(It++)); + DescendArray.emplace_back(NewBounds.first, NewBounds.second); + } + // Unite both arrays. + return unite(AscendArray, DescendArray); +} + +/// Promotion from unsigneds to signeds/unsigneds left. +RangeSet::ContainerType RangeSet::Factory::promoteTo(RangeSet What, + APSIntType Ty) { + ContainerType Result; + // We definitely know the size of the result set. + Result.reserve(What.size()); + + // Each unsigned value fits every larger type without any changes, + // whether the larger type is signed or unsigned. So just promote and push + // back each range one by one. + for (const Range &R : What) { + // Get bounds of the given range. + llvm::APSInt FromInt = R.From(); + llvm::APSInt ToInt = R.To(); + // Cast the bounds. + Ty.apply(FromInt); + Ty.apply(ToInt); + Result.emplace_back(ValueFactory.getValue(FromInt), + ValueFactory.getValue(ToInt)); + } + return Result; +} + RangeSet RangeSet::Factory::deletePoint(RangeSet From, const llvm::APSInt &Point) { if (!From.contains(Point)) diff --git a/clang/unittests/StaticAnalyzer/RangeSetTest.cpp b/clang/unittests/StaticAnalyzer/RangeSetTest.cpp --- a/clang/unittests/StaticAnalyzer/RangeSetTest.cpp +++ b/clang/unittests/StaticAnalyzer/RangeSetTest.cpp @@ -40,12 +40,18 @@ const Range &R) { return OS << toString(R); } +LLVM_ATTRIBUTE_UNUSED static std::ostream &operator<<(std::ostream &OS, + APSIntType Ty) { + return OS << (Ty.isUnsigned() ? "u" : "s") << Ty.getBitWidth(); +} } // namespace ento } // namespace clang namespace { +template constexpr bool is_signed_v = std::is_signed::value; + template struct TestValues { static constexpr T MIN = std::numeric_limits::min(); static constexpr T MAX = std::numeric_limits::max(); @@ -53,7 +59,7 @@ // which unary minus does not affect on, // e.g. int8/int32(0), uint8(128), uint32(2147483648). static constexpr T MID = - std::is_signed::value ? 0 : ~(static_cast(-1) / static_cast(2)); + is_signed_v ? 0 : ~(static_cast(-1) / static_cast(2)); static constexpr T A = MID - (MAX - MID) / 3 * 2; static constexpr T B = MID - (MAX - MID) / 3; static constexpr T C = -B; @@ -61,8 +67,40 @@ static_assert(MIN < A && A < B && B < MID && MID < C && C < D && D < MAX, "Values shall be in an ascending order"); + // Clear bits in low bytes by the given amount. + template + static constexpr T ClearLowBytes = + static_cast(static_cast(Value) + << ((Bytes >= CHAR_BIT) ? 0 : Bytes) * CHAR_BIT); + + template + static constexpr T TruncZeroOf = ClearLowBytes; + + // Random number with active bits in every byte. 0xAAAA'AAAA + static constexpr T XAAA = static_cast( + 0b10101010'10101010'10101010'10101010'10101010'10101010'10101010'10101010); + template + static constexpr T XAAATruncZeroOf = TruncZeroOf; // 0xAAAA'AB00 + + // Random number with active bits in every byte. 0x5555'5555 + static constexpr T X555 = static_cast( + 0b01010101'01010101'01010101'01010101'01010101'01010101'01010101'01010101); + template + static constexpr T X555TruncZeroOf = TruncZeroOf; // 0x5555'5600 + + // Numbers for ranges with the same bits in the lowest byte. + // 0xAAAA'AA2A + static constexpr T FromA = ClearLowBytes + 42; + static constexpr T ToA = FromA + 2; // 0xAAAA'AA2C + // 0x5555'552A + static constexpr T FromB = ClearLowBytes + 42; + static constexpr T ToB = FromB + 2; // 0x5555'552C }; +template +static constexpr APSIntType APSIntTy = + APSIntType(sizeof(T) * CHAR_BIT, !is_signed_v); + template class RangeSetTest : public testing::Test { public: // Init block @@ -74,21 +112,24 @@ // End init block using Self = RangeSetTest; - using RawRange = std::pair; - using RawRangeSet = std::initializer_list; - - const llvm::APSInt &from(BaseType X) { - static llvm::APSInt Base{sizeof(BaseType) * CHAR_BIT, - std::is_unsigned::value}; - Base = X; - return BVF.getValue(Base); + template using RawRangeT = std::pair; + template + using RawRangeSetT = std::initializer_list>; + using RawRange = RawRangeT; + using RawRangeSet = RawRangeSetT; + + template const llvm::APSInt &from(T X) { + static llvm::APSInt Int = APSIntTy.getZeroValue(); + Int = X; + return BVF.getValue(Int); } - Range from(const RawRange &Init) { + template Range from(const RawRangeT &Init) { return Range(from(Init.first), from(Init.second)); } - RangeSet from(const RawRangeSet &Init) { + template + RangeSet from(RawRangeSetT Init, APSIntType Ty = APSIntTy) { RangeSet RangeSet = F.getEmptySet(); for (const auto &Raw : Init) { RangeSet = F.add(RangeSet, from(Raw)); @@ -211,9 +252,20 @@ RawRangeSet RawExpected) { wrap(&Self::checkDeleteImpl, Point, RawFrom, RawExpected); } -}; -} // namespace + void checkCastToImpl(RangeSet What, APSIntType Ty, RangeSet Expected) { + RangeSet Result = F.castTo(What, Ty); + EXPECT_EQ(Result, Expected) + << "while casting " << toString(What) << " to " << Ty; + } + + template + void checkCastTo(RawRangeSetT What, RawRangeSetT Expected) { + static constexpr APSIntType FromTy = APSIntTy; + static constexpr APSIntType ToTy = APSIntTy; + this->checkCastToImpl(from(What, FromTy), ToTy, from(Expected, ToTy)); + } +}; using IntTypes = ::testing::Types; @@ -594,3 +646,437 @@ {{MIN, MIN}, {A, C}, {C + 2, D}, {MAX - 1, MAX}}); // clang-format on } + +template struct CastType { + using FromType = From; + using ToType = To; +}; + +template +class RangeSetCastToNoopTest : public RangeSetTest {}; +template +class RangeSetCastToPromotionTest + : public RangeSetTest {}; +template +class RangeSetCastToTruncationTest + : public RangeSetTest {}; +template +class RangeSetCastToConversionTest + : public RangeSetTest {}; +template +class RangeSetCastToPromotionConversionTest + : public RangeSetTest {}; +template +class RangeSetCastToTruncationConversionTest + : public RangeSetTest {}; + +using NoopCastTypes = + ::testing::Types, CastType, + CastType, CastType, + CastType, CastType, + CastType, CastType>; + +using PromotionCastTypes = + ::testing::Types, CastType, + CastType, CastType, + CastType, CastType, + CastType, CastType, + CastType, CastType, + CastType, CastType>; + +using TruncationCastTypes = + ::testing::Types, CastType, + CastType, CastType, + CastType, CastType, + CastType, CastType, + CastType, CastType, + CastType, CastType>; + +using ConversionCastTypes = + ::testing::Types, CastType, + CastType, CastType, + CastType, CastType, + CastType, CastType>; + +using PromotionConversionCastTypes = + ::testing::Types, CastType, + CastType, CastType, + CastType, CastType, + CastType, CastType, + CastType, CastType, + CastType, CastType>; + +using TruncationConversionCastTypes = + ::testing::Types, CastType, + CastType, CastType, + CastType, CastType, + CastType, CastType, + CastType, CastType, + CastType, CastType>; + +TYPED_TEST_SUITE(RangeSetCastToNoopTest, NoopCastTypes); +TYPED_TEST_SUITE(RangeSetCastToPromotionTest, PromotionCastTypes); +TYPED_TEST_SUITE(RangeSetCastToTruncationTest, TruncationCastTypes); +TYPED_TEST_SUITE(RangeSetCastToConversionTest, ConversionCastTypes); +TYPED_TEST_SUITE(RangeSetCastToPromotionConversionTest, + PromotionConversionCastTypes); +TYPED_TEST_SUITE(RangeSetCastToTruncationConversionTest, + TruncationConversionCastTypes); + +TYPED_TEST(RangeSetCastToNoopTest, RangeSetCastToNoopTest) { + // Just to reduce the verbosity. + using F = typename TypeParam::FromType; // From + using T = typename TypeParam::ToType; // To + + using TV = TestValues; + constexpr auto MIN = TV::MIN; + constexpr auto MAX = TV::MAX; + constexpr auto MID = TV::MID; + constexpr auto B = TV::B; + constexpr auto C = TV::C; + // One point + this->template checkCastTo({{MIN, MIN}}, {{MIN, MIN}}); + this->template checkCastTo({{MAX, MAX}}, {{MAX, MAX}}); + this->template checkCastTo({{MID, MID}}, {{MID, MID}}); + this->template checkCastTo({{B, B}}, {{B, B}}); + this->template checkCastTo({{C, C}}, {{C, C}}); + // Two points + this->template checkCastTo({{MIN, MIN}, {MAX, MAX}}, + {{MIN, MIN}, {MAX, MAX}}); + this->template checkCastTo({{MIN, MIN}, {B, B}}, {{MIN, MIN}, {B, B}}); + this->template checkCastTo({{MID, MID}, {MAX, MAX}}, + {{MID, MID}, {MAX, MAX}}); + this->template checkCastTo({{C, C}, {MAX, MAX}}, {{C, C}, {MAX, MAX}}); + this->template checkCastTo({{MID, MID}, {C, C}}, {{MID, MID}, {C, C}}); + this->template checkCastTo({{B, B}, {MID, MID}}, {{B, B}, {MID, MID}}); + this->template checkCastTo({{B, B}, {C, C}}, {{B, B}, {C, C}}); + // One range + this->template checkCastTo({{MIN, MAX}}, {{MIN, MAX}}); + this->template checkCastTo({{MIN, MID}}, {{MIN, MID}}); + this->template checkCastTo({{MID, MAX}}, {{MID, MAX}}); + this->template checkCastTo({{B, MAX}}, {{B, MAX}}); + this->template checkCastTo({{C, MAX}}, {{C, MAX}}); + this->template checkCastTo({{MIN, C}}, {{MIN, C}}); + this->template checkCastTo({{MIN, B}}, {{MIN, B}}); + this->template checkCastTo({{B, C}}, {{B, C}}); + // Two ranges + this->template checkCastTo({{MIN, B}, {C, MAX}}, {{MIN, B}, {C, MAX}}); + this->template checkCastTo({{B, MID}, {C, MAX}}, {{B, MID}, {C, MAX}}); + this->template checkCastTo({{MIN, B}, {MID, C}}, {{MIN, B}, {MID, C}}); +} + +TYPED_TEST(RangeSetCastToPromotionTest, Test) { + // Just to reduce the verbosity. + using F = typename TypeParam::FromType; // From + using T = typename TypeParam::ToType; // To + + using TV = TestValues; + constexpr auto MIN = TV::MIN; + constexpr auto MAX = TV::MAX; + constexpr auto MID = TV::MID; + constexpr auto B = TV::B; + constexpr auto C = TV::C; + // One point + this->template checkCastTo({{MIN, MIN}}, {{MIN, MIN}}); + this->template checkCastTo({{MAX, MAX}}, {{MAX, MAX}}); + this->template checkCastTo({{MID, MID}}, {{MID, MID}}); + this->template checkCastTo({{B, B}}, {{B, B}}); + this->template checkCastTo({{C, C}}, {{C, C}}); + // Two points + this->template checkCastTo({{MIN, MIN}, {MAX, MAX}}, + {{MIN, MIN}, {MAX, MAX}}); + this->template checkCastTo({{MIN, MIN}, {B, B}}, {{MIN, MIN}, {B, B}}); + this->template checkCastTo({{MID, MID}, {MAX, MAX}}, + {{MID, MID}, {MAX, MAX}}); + this->template checkCastTo({{C, C}, {MAX, MAX}}, {{C, C}, {MAX, MAX}}); + this->template checkCastTo({{MID, MID}, {C, C}}, {{MID, MID}, {C, C}}); + this->template checkCastTo({{B, B}, {MID, MID}}, {{B, B}, {MID, MID}}); + this->template checkCastTo({{B, B}, {C, C}}, {{B, B}, {C, C}}); + // One range + this->template checkCastTo({{MIN, MAX}}, {{MIN, MAX}}); + this->template checkCastTo({{MIN, MID}}, {{MIN, MID}}); + this->template checkCastTo({{MID, MAX}}, {{MID, MAX}}); + this->template checkCastTo({{B, MAX}}, {{B, MAX}}); + this->template checkCastTo({{C, MAX}}, {{C, MAX}}); + this->template checkCastTo({{MIN, C}}, {{MIN, C}}); + this->template checkCastTo({{MIN, B}}, {{MIN, B}}); + this->template checkCastTo({{B, C}}, {{B, C}}); + // Two ranges + this->template checkCastTo({{MIN, B}, {C, MAX}}, {{MIN, B}, {C, MAX}}); + this->template checkCastTo({{B, MID}, {C, MAX}}, {{B, MID}, {C, MAX}}); + this->template checkCastTo({{MIN, B}, {MID, C}}, {{MIN, B}, {MID, C}}); +} + +TYPED_TEST(RangeSetCastToTruncationTest, Test) { + // Just to reduce the verbosity. + using F = typename TypeParam::FromType; // From + using T = typename TypeParam::ToType; // To + + using TV = TestValues; + constexpr auto MIN = TV::MIN; + constexpr auto MAX = TV::MAX; + constexpr auto MID = TV::MID; + constexpr auto B = TV::B; + constexpr auto C = TV::C; + // One point + // + // NOTE: We can't use ToMIN, ToMAX, ... everywhere. That would be incorrect: + // int16(-32768, 32767) -> int8(-128, 127), + // aka (MIN, MAX) -> (ToMIN, ToMAX) // OK. + // int16(-32768, -32768) -> int8(-128, -128), + // aka (MIN, MIN) -> (ToMIN, ToMIN) // NOK. + // int16(-32768,-32768) -> int8(0, 0), + // aka (MIN, MIN) -> ((int8)MIN, (int8)MIN) // OK. + this->template checkCastTo({{MIN, MIN}}, {{MIN, MIN}}); + this->template checkCastTo({{MAX, MAX}}, {{MAX, MAX}}); + this->template checkCastTo({{MID, MID}}, {{MID, MID}}); + this->template checkCastTo({{B, B}}, {{B, B}}); + this->template checkCastTo({{C, C}}, {{C, C}}); + // Two points + // Use `if constexpr` here. + if (is_signed_v) { + this->template checkCastTo({{MIN, MIN}, {MAX, MAX}}, {{MAX, MIN}}); + this->template checkCastTo({{MID, MID}, {MAX, MAX}}, {{MAX, MID}}); + } else { + this->template checkCastTo({{MIN, MIN}, {MAX, MAX}}, + {{MIN, MIN}, {MAX, MAX}}); + this->template checkCastTo({{MID, MID}, {MAX, MAX}}, + {{MID, MID}, {MAX, MAX}}); + } + this->template checkCastTo({{MIN, MIN}, {B, B}}, {{MIN, MIN}, {B, B}}); + this->template checkCastTo({{C, C}, {MAX, MAX}}, {{C, C}, {MAX, MAX}}); + this->template checkCastTo({{MID, MID}, {C, C}}, {{MID, MID}, {C, C}}); + this->template checkCastTo({{B, B}, {MID, MID}}, {{B, B}, {MID, MID}}); + this->template checkCastTo({{B, B}, {C, C}}, {{B, B}, {C, C}}); + // One range + constexpr auto ToMIN = TestValues::MIN; + constexpr auto ToMAX = TestValues::MAX; + this->template checkCastTo({{MIN, MAX}}, {{ToMIN, ToMAX}}); + this->template checkCastTo({{MIN, MID}}, {{ToMIN, ToMAX}}); + this->template checkCastTo({{MID, MAX}}, {{ToMIN, ToMAX}}); + this->template checkCastTo({{B, MAX}}, {{ToMIN, ToMAX}}); + this->template checkCastTo({{C, MAX}}, {{ToMIN, ToMAX}}); + this->template checkCastTo({{MIN, C}}, {{ToMIN, ToMAX}}); + this->template checkCastTo({{MIN, B}}, {{ToMIN, ToMAX}}); + this->template checkCastTo({{B, C}}, {{ToMIN, ToMAX}}); + // Two ranges + this->template checkCastTo({{MIN, B}, {C, MAX}}, {{ToMIN, ToMAX}}); + this->template checkCastTo({{B, MID}, {C, MAX}}, {{ToMIN, ToMAX}}); + this->template checkCastTo({{MIN, B}, {MID, C}}, {{ToMIN, ToMAX}}); + constexpr auto XAAA = TV::XAAA; + constexpr auto X555 = TV::X555; + constexpr auto ZA = TV::template XAAATruncZeroOf; + constexpr auto Z5 = TV::template X555TruncZeroOf; + this->template checkCastTo({{XAAA, ZA}, {X555, Z5}}, + {{ToMIN, 0}, {X555, ToMAX}}); + // Use `if constexpr` here. + if (is_signed_v) { + // One range + this->template checkCastTo({{XAAA, ZA}}, {{XAAA, 0}}); + // Two ranges + this->template checkCastTo({{XAAA, ZA}, {1, 42}}, {{XAAA, 42}}); + } else { + // One range + this->template checkCastTo({{XAAA, ZA}}, {{0, 0}, {XAAA, ToMAX}}); + // Two ranges + this->template checkCastTo({{1, 42}, {XAAA, ZA}}, + {{0, 42}, {XAAA, ToMAX}}); + } + constexpr auto FromA = TV::FromA; + constexpr auto ToA = TV::ToA; + constexpr auto FromB = TV::FromB; + constexpr auto ToB = TV::ToB; + // int16 -> int8 + // (0x00'01, 0x00'05)U(0xFF'01, 0xFF'05) casts to + // (0x01, 0x05)U(0x01, 0x05) unites to + // (0x01, 0x05) + this->template checkCastTo({{FromA, ToA}, {FromB, ToB}}, + {{FromA, ToA}}); +} + +TYPED_TEST(RangeSetCastToConversionTest, Test) { + // Just to reduce the verbosity. + using F = typename TypeParam::FromType; // From + using T = typename TypeParam::ToType; // To + + using TV = TestValues; + constexpr auto MIN = TV::MIN; + constexpr auto MAX = TV::MAX; + constexpr auto MID = TV::MID; + constexpr auto B = TV::B; + constexpr auto C = TV::C; + // One point + this->template checkCastTo({{MIN, MIN}}, {{MIN, MIN}}); + this->template checkCastTo({{MAX, MAX}}, {{MAX, MAX}}); + this->template checkCastTo({{MID, MID}}, {{MID, MID}}); + this->template checkCastTo({{B, B}}, {{B, B}}); + this->template checkCastTo({{C, C}}, {{C, C}}); + // Two points + this->template checkCastTo({{MIN, MIN}, {MAX, MAX}}, {{MAX, MIN}}); + this->template checkCastTo({{MID, MID}, {MAX, MAX}}, + {{MID, MID}, {MAX, MAX}}); + this->template checkCastTo({{MIN, MIN}, {B, B}}, {{MIN, MIN}, {B, B}}); + this->template checkCastTo({{C, C}, {MAX, MAX}}, {{C, C}, {MAX, MAX}}); + this->template checkCastTo({{MID, MID}, {C, C}}, {{MID, MID}, {C, C}}); + this->template checkCastTo({{B, B}, {MID, MID}}, {{B, B}, {MID, MID}}); + this->template checkCastTo({{B, B}, {C, C}}, {{B, B}, {C, C}}); + // One range + constexpr auto ToMIN = TestValues::MIN; + constexpr auto ToMAX = TestValues::MAX; + this->template checkCastTo({{MIN, MAX}}, {{ToMIN, ToMAX}}); + this->template checkCastTo({{MIN, MID}}, + {{ToMIN, ToMIN}, {MIN, ToMAX}}); + this->template checkCastTo({{MID, MAX}}, {{MID, MAX}}); + this->template checkCastTo({{B, MAX}}, {{ToMIN, MAX}, {B, ToMAX}}); + this->template checkCastTo({{C, MAX}}, {{C, MAX}}); + this->template checkCastTo({{MIN, C}}, {{ToMIN, C}, {MIN, ToMAX}}); + this->template checkCastTo({{MIN, B}}, {{MIN, B}}); + this->template checkCastTo({{B, C}}, {{ToMIN, C}, {B, ToMAX}}); + // Two ranges + this->template checkCastTo({{MIN, B}, {C, MAX}}, {{C, B}}); + this->template checkCastTo({{B, MID}, {C, MAX}}, + {{MID, MID}, {C, MAX}, {B, ToMAX}}); + this->template checkCastTo({{MIN, B}, {MID, C}}, {{MID, C}, {MIN, B}}); +} + +TYPED_TEST(RangeSetCastToPromotionConversionTest, Test) { + // Just to reduce the verbosity. + using F = typename TypeParam::FromType; // From + using T = typename TypeParam::ToType; // To + + using TV = TestValues; + constexpr auto MIN = TV::MIN; + constexpr auto MAX = TV::MAX; + constexpr auto MID = TV::MID; + constexpr auto B = TV::B; + constexpr auto C = TV::C; + // One point + this->template checkCastTo({{MIN, MIN}}, {{MIN, MIN}}); + this->template checkCastTo({{MAX, MAX}}, {{MAX, MAX}}); + this->template checkCastTo({{MID, MID}}, {{MID, MID}}); + this->template checkCastTo({{B, B}}, {{B, B}}); + this->template checkCastTo({{C, C}}, {{C, C}}); + // Two points + this->template checkCastTo({{MIN, MIN}, {MAX, MAX}}, + {{MAX, MAX}, {MIN, MIN}}); + this->template checkCastTo({{MIN, MIN}, {B, B}}, {{MIN, MIN}, {B, B}}); + this->template checkCastTo({{MID, MID}, {MAX, MAX}}, + {{MID, MID}, {MAX, MAX}}); + this->template checkCastTo({{C, C}, {MAX, MAX}}, {{C, C}, {MAX, MAX}}); + this->template checkCastTo({{MID, MID}, {C, C}}, {{MID, MID}, {C, C}}); + this->template checkCastTo({{B, B}, {MID, MID}}, {{B, B}, {MID, MID}}); + this->template checkCastTo({{B, B}, {C, C}}, {{B, B}, {C, C}}); + + // Use `if constexpr` here. + if (is_signed_v) { + // One range + this->template checkCastTo({{MIN, MAX}}, {{0, MAX}, {MIN, -1}}); + this->template checkCastTo({{MIN, MID}}, {{0, 0}, {MIN, -1}}); + this->template checkCastTo({{MID, MAX}}, {{0, MAX}}); + this->template checkCastTo({{B, MAX}}, {{0, MAX}, {B, -1}}); + this->template checkCastTo({{C, MAX}}, {{C, MAX}}); + this->template checkCastTo({{MIN, C}}, {{0, C}, {MIN, -1}}); + this->template checkCastTo({{MIN, B}}, {{MIN, B}}); + this->template checkCastTo({{B, C}}, {{0, C}, {B, -1}}); + // Two ranges + this->template checkCastTo({{MIN, B}, {C, MAX}}, + {{C, MAX}, {MIN, B}}); + this->template checkCastTo({{B, MID}, {C, MAX}}, + {{0, 0}, {C, MAX}, {B, -1}}); + this->template checkCastTo({{MIN, B}, {MID, C}}, {{0, C}, {MIN, B}}); + } else { + // One range + this->template checkCastTo({{MIN, MAX}}, {{MIN, MAX}}); + this->template checkCastTo({{MIN, MID}}, {{MIN, MID}}); + this->template checkCastTo({{MID, MAX}}, {{MID, MAX}}); + this->template checkCastTo({{B, MAX}}, {{B, MAX}}); + this->template checkCastTo({{C, MAX}}, {{C, MAX}}); + this->template checkCastTo({{MIN, C}}, {{MIN, C}}); + this->template checkCastTo({{MIN, B}}, {{MIN, B}}); + this->template checkCastTo({{B, C}}, {{B, C}}); + // Two ranges + this->template checkCastTo({{MIN, B}, {C, MAX}}, + {{MIN, B}, {C, MAX}}); + this->template checkCastTo({{B, MID}, {C, MAX}}, + {{B, MID}, {C, MAX}}); + this->template checkCastTo({{MIN, B}, {MID, C}}, + {{MIN, B}, {MID, C}}); + } +} + +TYPED_TEST(RangeSetCastToTruncationConversionTest, Test) { + // Just to reduce the verbosity. + using F = typename TypeParam::FromType; // From + using T = typename TypeParam::ToType; // To + + using TV = TestValues; + constexpr auto MIN = TV::MIN; + constexpr auto MAX = TV::MAX; + constexpr auto MID = TV::MID; + constexpr auto B = TV::B; + constexpr auto C = TV::C; + // One point + this->template checkCastTo({{MIN, MIN}}, {{MIN, MIN}}); + this->template checkCastTo({{MAX, MAX}}, {{MAX, MAX}}); + this->template checkCastTo({{MID, MID}}, {{MID, MID}}); + this->template checkCastTo({{B, B}}, {{B, B}}); + this->template checkCastTo({{C, C}}, {{C, C}}); + // Two points + // Use `if constexpr` here. + if (is_signed_v) { + this->template checkCastTo({{MIN, MIN}, {MAX, MAX}}, + {{MIN, MIN}, {MAX, MAX}}); + this->template checkCastTo({{MID, MID}, {MAX, MAX}}, + {{MID, MID}, {MAX, MAX}}); + } else { + this->template checkCastTo({{MIN, MIN}, {MAX, MAX}}, {{MAX, MIN}}); + this->template checkCastTo({{MID, MID}, {MAX, MAX}}, {{MAX, MIN}}); + } + this->template checkCastTo({{MIN, MIN}, {B, B}}, {{MIN, MIN}, {B, B}}); + this->template checkCastTo({{C, C}, {MAX, MAX}}, {{C, C}, {MAX, MAX}}); + this->template checkCastTo({{MID, MID}, {C, C}}, {{MID, MID}, {C, C}}); + this->template checkCastTo({{B, B}, {MID, MID}}, {{B, B}, {MID, MID}}); + this->template checkCastTo({{B, B}, {C, C}}, {{B, B}, {C, C}}); + // One range + constexpr auto ToMIN = TestValues::MIN; + constexpr auto ToMAX = TestValues::MAX; + this->template checkCastTo({{MIN, MAX}}, {{ToMIN, ToMAX}}); + this->template checkCastTo({{MIN, MID}}, {{ToMIN, ToMAX}}); + this->template checkCastTo({{MID, MAX}}, {{ToMIN, ToMAX}}); + this->template checkCastTo({{B, MAX}}, {{ToMIN, ToMAX}}); + this->template checkCastTo({{C, MAX}}, {{ToMIN, ToMAX}}); + this->template checkCastTo({{MIN, C}}, {{ToMIN, ToMAX}}); + this->template checkCastTo({{MIN, B}}, {{ToMIN, ToMAX}}); + this->template checkCastTo({{B, C}}, {{ToMIN, ToMAX}}); + // Two ranges + this->template checkCastTo({{MIN, B}, {C, MAX}}, {{ToMIN, ToMAX}}); + this->template checkCastTo({{B, MID}, {C, MAX}}, {{ToMIN, ToMAX}}); + this->template checkCastTo({{MIN, B}, {MID, C}}, {{ToMIN, ToMAX}}); + constexpr auto XAAA = TV::XAAA; + constexpr auto X555 = TV::X555; + constexpr auto ZA = TV::template XAAATruncZeroOf; + constexpr auto Z5 = TV::template X555TruncZeroOf; + this->template checkCastTo({{XAAA, ZA}, {X555, Z5}}, + {{ToMIN, 0}, {X555, ToMAX}}); + // Use `if constexpr` here. + if (is_signed_v) { + // One range + this->template checkCastTo({{XAAA, ZA}}, {{0, 0}, {XAAA, ToMAX}}); + // Two ranges + this->template checkCastTo({{XAAA, ZA}, {1, 42}}, + {{0, 42}, {XAAA, ToMAX}}); + } else { + // One range + this->template checkCastTo({{XAAA, ZA}}, {{XAAA, 0}}); + // Two ranges + this->template checkCastTo({{1, 42}, {XAAA, ZA}}, {{XAAA, 42}}); + } + constexpr auto FromA = TV::FromA; + constexpr auto ToA = TV::ToA; + constexpr auto FromB = TV::FromB; + constexpr auto ToB = TV::ToB; + this->template checkCastTo({{FromA, ToA}, {FromB, ToB}}, + {{FromA, ToA}}); +} + +} // namespace