/* -*- 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; }