blob: c5ce073d323f79813955439dd8c7af0f243e10ee [file]
// Copyright 2026 The Dawn & Tint Authors
//
// Redistribution and use in source and binary forms, with or without
// modification, are permitted provided that the following conditions are met:
//
// 1. Redistributions of source code must retain the above copyright notice, this
// list of conditions and the following disclaimer.
//
// 2. Redistributions in binary form must reproduce the above copyright notice,
// this list of conditions and the following disclaimer in the documentation
// and/or other materials provided with the distribution.
//
// 3. Neither the name of the copyright holder nor the names of its
// contributors may be used to endorse or promote products derived from
// this software without specific prior written permission.
//
// THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS"
// AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
// IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE
// DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT HOLDER OR CONTRIBUTORS BE LIABLE
// FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
// DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR
// SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER
// CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY,
// OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
// OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
#include "src/utils/numeric.h"
#include <gtest/gtest.h>
#include "src/utils/typed_integer.h"
namespace dawn {
namespace {
TEST(NumericTest, IsCastAlwaysInRange) {
static_assert(kIsCastAlwaysInRange<uint32_t, uint64_t>);
static_assert(!kIsCastAlwaysInRange<int32_t, uint32_t>);
static_assert(!kIsCastAlwaysInRange<uint32_t, int32_t>);
static_assert(!kIsCastAlwaysInRange<uint64_t, uint32_t>);
}
// Death tests
// Name "*DeathTest" per https://google.github.io/googletest/advanced.html#death-test-naming
#ifdef DAWN_ENABLE_ASSERTS
#define EXPECT_DCHECK(statement, matcher) EXPECT_DEATH(statement, matcher)
#else
#define EXPECT_DCHECK(statement, matcher) statement
#endif
// Test for checked cast between various types.
template <typename T32, typename T64>
void CheckedCastTestPair() {
using Prim32 = UnderlyingType<T32>;
using Prim64 = UnderlyingType<T64>;
// No-ops
checked_cast<T32>(T32{std::numeric_limits<Prim32>::max()});
checked_cast<T64>(T64{std::numeric_limits<Prim64>::max()});
// Widening
checked_cast<T64>(T32{std::numeric_limits<Prim32>::max()});
// Narrowing
EXPECT_DEATH(checked_cast<T32>(T64{Prim64{std::numeric_limits<Prim32>::max()} + 1}), "");
EXPECT_DEATH(checked_cast<T32>(T64{Prim64{std::numeric_limits<Prim32>::max()} * 2}), "");
EXPECT_DEATH(checked_cast<T32>(T64{std::numeric_limits<Prim64>::max()}), "");
}
template <typename U32, typename I32, typename U64, typename I64>
void CheckedCastTest() {
static_assert(std::is_same_v<UnderlyingType<U32>, uint32_t>);
static_assert(std::is_same_v<UnderlyingType<U64>, uint64_t>);
static_assert(std::is_same_v<UnderlyingType<I32>, int32_t>);
static_assert(std::is_same_v<UnderlyingType<I64>, int64_t>);
CheckedCastTestPair<U32, U64>();
CheckedCastTestPair<U32, I64>();
CheckedCastTestPair<I32, U64>();
CheckedCastTestPair<I32, I64>();
}
TEST(NumericDeathTest, CheckedCast) {
using TypedU32 = TypedInteger<struct TypedU32T, uint32_t>;
using TypedU64 = TypedInteger<struct TypedU64T, uint64_t>;
using TypedI32 = TypedInteger<struct TypedI32T, int32_t>;
using TypedI64 = TypedInteger<struct TypedI64T, int64_t>;
enum class EnumU32 : uint32_t {};
enum class EnumU64 : uint64_t {};
enum class EnumI32 : int32_t {};
enum class EnumI64 : int64_t {};
// primitive <-> primitive
CheckedCastTest<uint32_t, int32_t, uint64_t, int64_t>();
// primitive <-> TypedInteger
CheckedCastTest<uint32_t, int32_t, TypedU64, TypedI64>();
CheckedCastTest<TypedU32, TypedI32, uint64_t, int64_t>();
// TypedInteger <-> TypedInteger
CheckedCastTest<TypedU32, TypedI32, TypedU64, TypedI64>();
// primitive <-> enum class
CheckedCastTest<uint32_t, int32_t, EnumU64, EnumI64>();
CheckedCastTest<EnumU32, EnumI32, uint64_t, int64_t>();
// enum class <-> enum class
CheckedCastTest<EnumU32, EnumI32, EnumU64, EnumI64>();
// TypedInteger <-> enum class
CheckedCastTest<TypedU32, TypedI32, EnumU64, EnumI64>();
CheckedCastTest<EnumU32, EnumI32, EnumU64, EnumI64>();
// Basic test that dchecked_cast is checked in debug but not release.
// Not exhaustive like the cases above because those are very slow.
EXPECT_DCHECK(dchecked_cast<uint32_t>(uint64_t{std::numeric_limits<uint32_t>::max()} + 1), "");
EXPECT_DCHECK(dchecked_cast<int32_t>(uint32_t{std::numeric_limits<int32_t>::max()} + 1), "");
}
} // anonymous namespace
} // namespace dawn