From 26a029d407be480d791972afb5975cf62c9360a6 Mon Sep 17 00:00:00 2001 From: Daniel Baumann Date: Fri, 19 Apr 2024 02:47:55 +0200 Subject: Adding upstream version 124.0.1. Signed-off-by: Daniel Baumann --- mfbt/tests/TestCasting.cpp | 255 +++++++++++++++++++++++++++++++++++++++++++++ 1 file changed, 255 insertions(+) create mode 100644 mfbt/tests/TestCasting.cpp (limited to 'mfbt/tests/TestCasting.cpp') diff --git a/mfbt/tests/TestCasting.cpp b/mfbt/tests/TestCasting.cpp new file mode 100644 index 0000000000..9b040956c7 --- /dev/null +++ b/mfbt/tests/TestCasting.cpp @@ -0,0 +1,255 @@ +/* -*- Mode: C++; tab-width: 8; indent-tabs-mode: nil; c-basic-offset: 2 -*- */ +/* vim: set ts=8 sts=2 et sw=2 tw=80: */ +/* This Source Code Form is subject to the terms of the Mozilla Public + * License, v. 2.0. If a copy of the MPL was not distributed with this file, + * You can obtain one at http://mozilla.org/MPL/2.0/. */ + +#include "mozilla/Casting.h" +#include "mozilla/ThreadSafety.h" + +#include +#include +#include +#include + +using mozilla::AssertedCast; +using mozilla::BitwiseCast; +using mozilla::detail::IsInBounds; + +static const uint8_t floatMantissaBitsPlusOne = 24; +static const uint8_t doubleMantissaBitsPlusOne = 53; + +template +struct UintUlongBitwiseCast; + +template +struct UintUlongBitwiseCast { + static void test() { + MOZ_RELEASE_ASSERT(BitwiseCast(Uint(8675309)) == Ulong(8675309)); + } +}; + +template +struct UintUlongBitwiseCast { + static void test() {} +}; + +static void TestBitwiseCast() { + MOZ_RELEASE_ASSERT(BitwiseCast(int(8675309)) == int(8675309)); + UintUlongBitwiseCast::test(); +} + +static void TestSameSize() { + MOZ_RELEASE_ASSERT((IsInBounds(int16_t(0)))); + MOZ_RELEASE_ASSERT((IsInBounds(int16_t(INT16_MIN)))); + MOZ_RELEASE_ASSERT((IsInBounds(int16_t(INT16_MAX)))); + MOZ_RELEASE_ASSERT((IsInBounds(uint16_t(UINT16_MAX)))); + MOZ_RELEASE_ASSERT((IsInBounds(uint16_t(0)))); + MOZ_RELEASE_ASSERT((!IsInBounds(uint16_t(-1)))); + MOZ_RELEASE_ASSERT((!IsInBounds(int16_t(-1)))); + MOZ_RELEASE_ASSERT((IsInBounds(int16_t(INT16_MAX)))); + MOZ_RELEASE_ASSERT((!IsInBounds(int16_t(INT16_MIN)))); + MOZ_RELEASE_ASSERT((IsInBounds(int32_t(INT32_MAX)))); + MOZ_RELEASE_ASSERT((!IsInBounds(int32_t(INT32_MIN)))); +} + +static void TestToBiggerSize() { + MOZ_RELEASE_ASSERT((IsInBounds(int16_t(0)))); + MOZ_RELEASE_ASSERT((IsInBounds(int16_t(INT16_MIN)))); + MOZ_RELEASE_ASSERT((IsInBounds(int16_t(INT16_MAX)))); + MOZ_RELEASE_ASSERT((IsInBounds(uint16_t(UINT16_MAX)))); + MOZ_RELEASE_ASSERT((IsInBounds(uint16_t(0)))); + MOZ_RELEASE_ASSERT((IsInBounds(uint16_t(-1)))); + MOZ_RELEASE_ASSERT((!IsInBounds(int16_t(-1)))); + MOZ_RELEASE_ASSERT((IsInBounds(int16_t(INT16_MAX)))); + MOZ_RELEASE_ASSERT((!IsInBounds(int16_t(INT16_MIN)))); + MOZ_RELEASE_ASSERT((IsInBounds(int32_t(INT32_MAX)))); + MOZ_RELEASE_ASSERT((!IsInBounds(int32_t(INT32_MIN)))); +} + +static void TestToSmallerSize() { + MOZ_RELEASE_ASSERT((IsInBounds(int16_t(0)))); + MOZ_RELEASE_ASSERT((!IsInBounds(int16_t(INT16_MIN)))); + MOZ_RELEASE_ASSERT((!IsInBounds(int16_t(INT16_MAX)))); + MOZ_RELEASE_ASSERT((!IsInBounds(uint16_t(UINT16_MAX)))); + MOZ_RELEASE_ASSERT((IsInBounds(uint16_t(0)))); + MOZ_RELEASE_ASSERT((!IsInBounds(uint16_t(-1)))); + MOZ_RELEASE_ASSERT((!IsInBounds(int16_t(-1)))); + MOZ_RELEASE_ASSERT((!IsInBounds(int16_t(INT16_MAX)))); + MOZ_RELEASE_ASSERT((!IsInBounds(int16_t(INT16_MIN)))); + MOZ_RELEASE_ASSERT((!IsInBounds(int32_t(INT32_MAX)))); + MOZ_RELEASE_ASSERT((!IsInBounds(int32_t(INT32_MIN)))); + + // Boundary cases + MOZ_RELEASE_ASSERT((!IsInBounds(int64_t(INT32_MIN) - 1))); + MOZ_RELEASE_ASSERT((IsInBounds(int64_t(INT32_MIN)))); + MOZ_RELEASE_ASSERT((IsInBounds(int64_t(INT32_MIN) + 1))); + MOZ_RELEASE_ASSERT((IsInBounds(int64_t(INT32_MAX) - 1))); + MOZ_RELEASE_ASSERT((IsInBounds(int64_t(INT32_MAX)))); + MOZ_RELEASE_ASSERT((!IsInBounds(int64_t(INT32_MAX) + 1))); + + MOZ_RELEASE_ASSERT((!IsInBounds(int64_t(-1)))); + MOZ_RELEASE_ASSERT((IsInBounds(int64_t(0)))); + MOZ_RELEASE_ASSERT((IsInBounds(int64_t(1)))); + MOZ_RELEASE_ASSERT((IsInBounds(int64_t(UINT32_MAX) - 1))); + MOZ_RELEASE_ASSERT((IsInBounds(int64_t(UINT32_MAX)))); + MOZ_RELEASE_ASSERT((!IsInBounds(int64_t(UINT32_MAX) + 1))); +} + +template +void checkBoundariesFloating(In aEpsilon = {}, Out aIntegerOffset = {}) { + // Check the max value of the input float can't be represented as an integer. + // This is true for all floating point and integer width. + MOZ_RELEASE_ASSERT((!IsInBounds(std::numeric_limits::max()))); + // Check that the max value of the integer, as a float, minus an offset that + // depends on the magnitude, can be represented as an integer. + MOZ_RELEASE_ASSERT((IsInBounds( + static_cast(std::numeric_limits::max() - aIntegerOffset)))); + // Check that the max value of the integer, plus a number that depends on the + // magnitude of the number, can't be represented as this integer (because it + // becomes too big). + MOZ_RELEASE_ASSERT((!IsInBounds( + aEpsilon + static_cast(std::numeric_limits::max())))); + if constexpr (std::is_signed_v) { + // Same for negative numbers. + MOZ_RELEASE_ASSERT( + (!IsInBounds(std::numeric_limits::lowest()))); + MOZ_RELEASE_ASSERT((IsInBounds( + static_cast(std::numeric_limits::lowest())))); + MOZ_RELEASE_ASSERT((!IsInBounds( + static_cast(std::numeric_limits::lowest()) - aEpsilon))); + } else { + // Check for negative floats and unsigned integer types. + MOZ_RELEASE_ASSERT((!IsInBounds(static_cast(-1)))); + } +} + +void TestFloatConversion() { + MOZ_RELEASE_ASSERT((!IsInBounds(UINT64_MAX))); + MOZ_RELEASE_ASSERT((!IsInBounds(UINT32_MAX))); + MOZ_RELEASE_ASSERT((IsInBounds(UINT16_MAX))); + MOZ_RELEASE_ASSERT((IsInBounds(UINT8_MAX))); + + MOZ_RELEASE_ASSERT((!IsInBounds(INT64_MAX))); + MOZ_RELEASE_ASSERT((!IsInBounds(INT64_MIN))); + MOZ_RELEASE_ASSERT((!IsInBounds(INT32_MAX))); + MOZ_RELEASE_ASSERT((!IsInBounds(INT32_MIN))); + MOZ_RELEASE_ASSERT((IsInBounds(INT16_MAX))); + MOZ_RELEASE_ASSERT((IsInBounds(INT16_MIN))); + MOZ_RELEASE_ASSERT((IsInBounds(INT8_MAX))); + MOZ_RELEASE_ASSERT((IsInBounds(INT8_MIN))); + + MOZ_RELEASE_ASSERT((!IsInBounds(UINT64_MAX))); + MOZ_RELEASE_ASSERT((IsInBounds(UINT32_MAX))); + MOZ_RELEASE_ASSERT((IsInBounds(UINT16_MAX))); + MOZ_RELEASE_ASSERT((IsInBounds(UINT8_MAX))); + + MOZ_RELEASE_ASSERT((!IsInBounds(INT64_MAX))); + MOZ_RELEASE_ASSERT((!IsInBounds(INT64_MIN))); + MOZ_RELEASE_ASSERT((IsInBounds(INT32_MAX))); + MOZ_RELEASE_ASSERT((IsInBounds(INT32_MIN))); + MOZ_RELEASE_ASSERT((IsInBounds(INT16_MAX))); + MOZ_RELEASE_ASSERT((IsInBounds(INT16_MIN))); + MOZ_RELEASE_ASSERT((IsInBounds(INT8_MAX))); + MOZ_RELEASE_ASSERT((IsInBounds(INT8_MIN))); + + // Floor check + MOZ_RELEASE_ASSERT((IsInBounds(4.3))); + MOZ_RELEASE_ASSERT((AssertedCast(4.3f) == 4u)); + MOZ_RELEASE_ASSERT((IsInBounds(4.3))); + MOZ_RELEASE_ASSERT((AssertedCast(4.3f) == 4u)); + MOZ_RELEASE_ASSERT((IsInBounds(4.3))); + MOZ_RELEASE_ASSERT((AssertedCast(4.3f) == 4u)); + MOZ_RELEASE_ASSERT((IsInBounds(4.3))); + MOZ_RELEASE_ASSERT((AssertedCast(4.3f) == 4u)); + + MOZ_RELEASE_ASSERT((IsInBounds(4.3))); + MOZ_RELEASE_ASSERT((AssertedCast(4.3f) == 4u)); + MOZ_RELEASE_ASSERT((IsInBounds(4.3))); + MOZ_RELEASE_ASSERT((AssertedCast(4.3f) == 4u)); + MOZ_RELEASE_ASSERT((IsInBounds(4.3))); + MOZ_RELEASE_ASSERT((AssertedCast(4.3f) == 4u)); + MOZ_RELEASE_ASSERT((IsInBounds(4.3))); + MOZ_RELEASE_ASSERT((AssertedCast(4.3f) == 4u)); + + MOZ_RELEASE_ASSERT((IsInBounds(-4.3))); + MOZ_RELEASE_ASSERT((AssertedCast(-4.3f) == -4)); + MOZ_RELEASE_ASSERT((IsInBounds(-4.3))); + MOZ_RELEASE_ASSERT((AssertedCast(-4.3f) == -4)); + MOZ_RELEASE_ASSERT((IsInBounds(-4.3))); + MOZ_RELEASE_ASSERT((AssertedCast(-4.3f) == -4)); + MOZ_RELEASE_ASSERT((IsInBounds(-4.3))); + MOZ_RELEASE_ASSERT((AssertedCast(-4.3f) == -4)); + + // Bound check for float to unsigned integer conversion. The parameters are + // espilons and offsets allowing to check boundaries, that depend on the + // magnitude of the numbers. + checkBoundariesFloating(2049.); + checkBoundariesFloating(1.); + checkBoundariesFloating(1.); + checkBoundariesFloating(1.); + // Large number because of the lack of precision of floats at this magnitude + checkBoundariesFloating(1.1e12f); + checkBoundariesFloating(1.f, 128u); + checkBoundariesFloating(1.f); + checkBoundariesFloating(1.f); + + checkBoundariesFloating(1025.); + checkBoundariesFloating(1.); + checkBoundariesFloating(1.); + checkBoundariesFloating(1.); + // Large number because of the lack of precision of floats at this magnitude + checkBoundariesFloating(1.1e12f); + checkBoundariesFloating(256.f, 64u); + checkBoundariesFloating(1.f); + checkBoundariesFloating(1.f); + + // Integer to floating point, boundary cases + MOZ_RELEASE_ASSERT(!(IsInBounds( + int64_t(std::pow(2, floatMantissaBitsPlusOne)) + 1))); + MOZ_RELEASE_ASSERT((IsInBounds( + int64_t(std::pow(2, floatMantissaBitsPlusOne))))); + MOZ_RELEASE_ASSERT((IsInBounds( + int64_t(std::pow(2, floatMantissaBitsPlusOne)) - 1))); + + MOZ_RELEASE_ASSERT(!(IsInBounds( + int64_t(-std::pow(2, floatMantissaBitsPlusOne)) - 1))); + MOZ_RELEASE_ASSERT((IsInBounds( + int64_t(-std::pow(2, floatMantissaBitsPlusOne))))); + MOZ_RELEASE_ASSERT((IsInBounds( + int64_t(-std::pow(2, floatMantissaBitsPlusOne)) + 1))); + + MOZ_RELEASE_ASSERT(!(IsInBounds( + uint64_t(std::pow(2, doubleMantissaBitsPlusOne)) + 1))); + MOZ_RELEASE_ASSERT((IsInBounds( + uint64_t(std::pow(2, doubleMantissaBitsPlusOne))))); + MOZ_RELEASE_ASSERT((IsInBounds( + uint64_t(std::pow(2, doubleMantissaBitsPlusOne)) - 1))); + + MOZ_RELEASE_ASSERT(!(IsInBounds( + int64_t(-std::pow(2, doubleMantissaBitsPlusOne)) - 1))); + MOZ_RELEASE_ASSERT((IsInBounds( + int64_t(-std::pow(2, doubleMantissaBitsPlusOne))))); + MOZ_RELEASE_ASSERT((IsInBounds( + int64_t(-std::pow(2, doubleMantissaBitsPlusOne)) + 1))); + + MOZ_RELEASE_ASSERT(!(IsInBounds(UINT64_MAX))); + MOZ_RELEASE_ASSERT(!(IsInBounds(INT64_MAX))); + MOZ_RELEASE_ASSERT(!(IsInBounds(INT64_MIN))); + + MOZ_RELEASE_ASSERT( + !(IsInBounds(std::numeric_limits::max()))); + MOZ_RELEASE_ASSERT( + !(IsInBounds(-std::numeric_limits::max()))); +} + +int main() { + TestBitwiseCast(); + + TestSameSize(); + TestToBiggerSize(); + TestToSmallerSize(); + TestFloatConversion(); + + return 0; +} -- cgit v1.2.3