From 84a41387ae6e5305540eb3c77c9643eb85ea660a Mon Sep 17 00:00:00 2001 From: Redstone1024 <2824517378@qq.com> Date: Wed, 27 Nov 2024 16:58:39 +0800 Subject: [PATCH] feat(numeric): add fixed-width literals, bit math functions and the corresponding testing --- .../Source/Private/Testing/NumericTesting.cpp | 127 ++++++++++ Redcraft.Utility/Source/Public/Numeric/Bit.h | 228 ++++++++++++++++++ .../Source/Public/Numeric/Limits.h | 1 + .../Source/Public/Numeric/Literal.h | 133 ++++++++++ .../Source/Public/Numeric/Numeric.h | 7 + .../Source/Public/Testing/NumericTesting.h | 19 ++ 6 files changed, 515 insertions(+) create mode 100644 Redcraft.Utility/Source/Private/Testing/NumericTesting.cpp create mode 100644 Redcraft.Utility/Source/Public/Numeric/Bit.h create mode 100644 Redcraft.Utility/Source/Public/Numeric/Literal.h create mode 100644 Redcraft.Utility/Source/Public/Numeric/Numeric.h create mode 100644 Redcraft.Utility/Source/Public/Testing/NumericTesting.h diff --git a/Redcraft.Utility/Source/Private/Testing/NumericTesting.cpp b/Redcraft.Utility/Source/Private/Testing/NumericTesting.cpp new file mode 100644 index 0000000..91074dc --- /dev/null +++ b/Redcraft.Utility/Source/Private/Testing/NumericTesting.cpp @@ -0,0 +1,127 @@ +#include "Testing/NumericTesting.h" + +#include "Numeric/Numeric.h" +#include "Miscellaneous/AssertionMacros.h" + +NAMESPACE_REDCRAFT_BEGIN +NAMESPACE_MODULE_BEGIN(Redcraft) +NAMESPACE_MODULE_BEGIN(Utility) + +NAMESPACE_BEGIN(Testing) + +void TestNumeric() +{ + TestLiteral(); + TestBit(); +} + +void TestLiteral() +{ +// always_check((CSameAs)); + always_check((CSameAs)); + always_check((CSameAs)); + always_check((CSameAs)); + + always_check((CSameAs)); + always_check((CSameAs)); + always_check((CSameAs)); + always_check((CSameAs)); + + always_check((CSameAs)); + always_check((CSameAs)); + + always_check((CSameAs)); + always_check((CSameAs)); +} + +void TestBit() +{ + always_check(Math::ByteSwap(0x00 ) == 0x00 ); + always_check(Math::ByteSwap(0x0011 ) == 0x1100 ); + always_check(Math::ByteSwap(0x00112233 ) == 0x33221100 ); + always_check(Math::ByteSwap(0x0011223344556677) == 0x7766554433221100); + + always_check(Math::IsSingleBit(0b0000u) == false); + always_check(Math::IsSingleBit(0b0001u) == true ); + always_check(Math::IsSingleBit(0b0010u) == true ); + always_check(Math::IsSingleBit(0b0011u) == false); + always_check(Math::IsSingleBit(0b0100u) == true ); + always_check(Math::IsSingleBit(0b0101u) == false); + always_check(Math::IsSingleBit(0b0110u) == false); + always_check(Math::IsSingleBit(0b0111u) == false); + always_check(Math::IsSingleBit(0b1000u) == true ); + always_check(Math::IsSingleBit(0b1001u) == false); + + always_check(Math::CountAllZero(0b00000000u8) == 8); + always_check(Math::CountAllZero(0b11111111u8) == 0); + always_check(Math::CountAllZero(0b00011101u8) == 4); + + always_check(Math::CountAllOne(0b00000000u8) == 0); + always_check(Math::CountAllOne(0b11111111u8) == 8); + always_check(Math::CountAllOne(0b00011101u8) == 4); + + always_check(Math::CountLeftZero(0b00000000u8) == 8); + always_check(Math::CountLeftZero(0b11111111u8) == 0); + always_check(Math::CountLeftZero(0b00011100u8) == 3); + + always_check(Math::CountLeftOne(0b00000000u8) == 0); + always_check(Math::CountLeftOne(0b11111111u8) == 8); + always_check(Math::CountLeftOne(0b11100011u8) == 3); + + always_check(Math::CountRightZero(0b00000000u8) == 8); + always_check(Math::CountRightZero(0b11111111u8) == 0); + always_check(Math::CountRightZero(0b00011100u8) == 2); + + always_check(Math::CountRightOne(0b00000000u8) == 0); + always_check(Math::CountRightOne(0b11111111u8) == 8); + always_check(Math::CountRightOne(0b11100011u8) == 2); + + always_check(Math::BitWidth(0b0000u) == 0); + always_check(Math::BitWidth(0b0001u) == 1); + always_check(Math::BitWidth(0b0010u) == 2); + always_check(Math::BitWidth(0b0011u) == 2); + always_check(Math::BitWidth(0b0100u) == 3); + always_check(Math::BitWidth(0b0101u) == 3); + always_check(Math::BitWidth(0b0110u) == 3); + always_check(Math::BitWidth(0b0111u) == 3); + + always_check(Math::BitCeil(0b00000000u) == 0b00000001u); + always_check(Math::BitCeil(0b00000001u) == 0b00000001u); + always_check(Math::BitCeil(0b00000010u) == 0b00000010u); + always_check(Math::BitCeil(0b00000011u) == 0b00000100u); + always_check(Math::BitCeil(0b00000100u) == 0b00000100u); + always_check(Math::BitCeil(0b00000101u) == 0b00001000u); + always_check(Math::BitCeil(0b00000110u) == 0b00001000u); + always_check(Math::BitCeil(0b00000111u) == 0b00001000u); + always_check(Math::BitCeil(0b00001000u) == 0b00001000u); + always_check(Math::BitCeil(0b00001001u) == 0b00010000u); + + always_check(Math::BitFloor(0b00000000u) == 0b00000000u); + always_check(Math::BitFloor(0b00000001u) == 0b00000001u); + always_check(Math::BitFloor(0b00000010u) == 0b00000010u); + always_check(Math::BitFloor(0b00000011u) == 0b00000010u); + always_check(Math::BitFloor(0b00000100u) == 0b00000100u); + always_check(Math::BitFloor(0b00000101u) == 0b00000100u); + always_check(Math::BitFloor(0b00000110u) == 0b00000100u); + always_check(Math::BitFloor(0b00000111u) == 0b00000100u); + always_check(Math::BitFloor(0b00001000u) == 0b00001000u); + always_check(Math::BitFloor(0b00001001u) == 0b00001000u); + + always_check(Math::RotateLeft(0b00011101u8, 0) == 0b00011101u8); + always_check(Math::RotateLeft(0b00011101u8, 1) == 0b00111010u8); + always_check(Math::RotateLeft(0b00011101u8, 4) == 0b11010001u8); + always_check(Math::RotateLeft(0b00011101u8, 9) == 0b00111010u8); + always_check(Math::RotateLeft(0b00011101u8, -1) == 0b10001110u8); + + always_check(Math::RotateRight(0b00011101u8, 0) == 0b00011101u8); + always_check(Math::RotateRight(0b00011101u8, 1) == 0b10001110u8); + always_check(Math::RotateRight(0b00011101u8, 4) == 0b11010001u8); + always_check(Math::RotateRight(0b00011101u8, 9) == 0b10001110u8); + always_check(Math::RotateRight(0b00011101u8, -1) == 0b00111010u8); +} + +NAMESPACE_END(Testing) + +NAMESPACE_MODULE_END(Utility) +NAMESPACE_MODULE_END(Redcraft) +NAMESPACE_REDCRAFT_END diff --git a/Redcraft.Utility/Source/Public/Numeric/Bit.h b/Redcraft.Utility/Source/Public/Numeric/Bit.h new file mode 100644 index 0000000..ed68a29 --- /dev/null +++ b/Redcraft.Utility/Source/Public/Numeric/Bit.h @@ -0,0 +1,228 @@ +#pragma once + +#include "CoreTypes.h" +#include "Numeric/Limits.h" +#include "TypeTraits/TypeTraits.h" + +#include + +NAMESPACE_REDCRAFT_BEGIN +NAMESPACE_MODULE_BEGIN(Redcraft) +NAMESPACE_MODULE_BEGIN(Utility) + +NAMESPACE_BEGIN(Math) + +template requires (sizeof(T) == sizeof(U)) +FORCEINLINE constexpr T BitCast(const U& Value) +{ + return __builtin_bit_cast(T, Value); +} + +template +FORCEINLINE constexpr T ByteSwap(T Value) +{ + static_assert(sizeof(T) <= 16, "ByteSwap only works with T up to 128 bits."); + + if constexpr (sizeof(T) == 1) return Value; + +# if PLATFORM_COMPILER_MSVC + { + if constexpr (sizeof(T) == 2) return _byteswap_ushort(Value); + if constexpr (sizeof(T) == 4) return _byteswap_ulong(Value); + if constexpr (sizeof(T) == 8) return _byteswap_uint64(Value); + } +# elif PLATFORM_COMPILER_CLANG || PLATFORM_COMPILER_GCC + { + if constexpr (sizeof(T) == 2) return __builtin_bswap16(Value); + if constexpr (sizeof(T) == 4) return __builtin_bswap32(Value); + if constexpr (sizeof(T) == 8) return __builtin_bswap64(Value); + } +# else + { + if constexpr (sizeof(T) == 2) return (Value << 8) | (Value >> 8); // AB -> BA + + if constexpr (sizeof(T) == 4) + { + T Result = 0; + + Result = ((Result << 8) & 0xFF00FF00u32) | ((Result >> 8) & 0x00FF00FFu32); // ABCD -> BADC + + return (Result << 16) | (Result >> 16); + } + + if constexpr (sizeof(T) == 8) + { + T Result = Value; + + Result = ((Result << 8) & 0xFF00FF00FF00FF00u64) | ((Result >> 8) & 0x00FF00FF00FF00FFu64); // ABCDEFGH -> BADCFEHG + Result = ((Result << 16) & 0xFFFF0000FFFF0000u64) | ((Result >> 16) & 0x0000FFFF0000FFFFu64); // BADCFEHG -> DCBAHGFE + + return (Result << 32) | (Result >> 32); + } + } +# endif + +# if PLATFORM_HAS_INT128 + { + if constexpr (sizeof(T) == 16) + { + T Result = Value; + + Result = ((Result << 8) & 0xFF00FF00FF00FF00FF00FF00FF00FF00u128) | ((Result >> 8) & 0x00FF00FF00FF00FF00FF00FF00FF00FFu128); // ABCDEFGHIJKLMNOP -> BADCFEHGJILKMONP + Result = ((Result << 16) & 0xFFFF0000FFFF0000FFFF0000FFFF0000u128) | ((Result >> 16) & 0x0000FFFF0000FFFF0000FFFF0000FFFFu128); // BADCFEHGJILKMONP -> DCBAHGFEJIKLNOPM + Result = ((Result << 32) & 0xFFFFFFFF00000000FFFFFFFF00000000u128) | ((Result >> 32) & 0x00000000FFFFFFFF00000000FFFFFFFFu128); // DCBAHGFEJIKLNOPM -> HGFEDCBAKJILMOPN + + return (Result << 64) | (Result >> 64); + } + } +# endif + + return 0; +} + +template +FORCEINLINE constexpr bool IsSingleBit(T Value) +{ + if constexpr (CSameAs) return Value; + + else return Value && !(Value & (Value - 1)); +} + +template +FORCEINLINE constexpr uint CountAllZero(T Value) +{ + if constexpr (CSameAs) return Value ? 0 : 1; + + else return static_cast(TNumericLimits::Digits - NAMESPACE_STD::popcount(Value)); +} + +template +FORCEINLINE constexpr uint CountAllOne(T Value) +{ + if constexpr (CSameAs) return Value ? 1 : 0; + + else return static_cast(NAMESPACE_STD::popcount(Value)); +} + +template +FORCEINLINE constexpr uint CountLeftZero(T Value) +{ + if constexpr (CSameAs) return Value ? 0 : 1; + + else return static_cast(NAMESPACE_STD::countl_zero(Value)); +} + +template +FORCEINLINE constexpr uint CountLeftOne(T Value) +{ + return Math::CountLeftZero(~Value); +} + +template +FORCEINLINE constexpr uint CountRightZero(T Value) +{ + if constexpr (CSameAs) return Value ? 0 : 1; + + else return static_cast(NAMESPACE_STD::countr_zero(Value)); +} + +template +FORCEINLINE constexpr uint CountRightOne(T Value) +{ + return Math::CountRightZero(~Value); +} + +template +FORCEINLINE constexpr T BitWidth(T Value) +{ + return TNumericLimits::Digits - Math::CountLeftZero(Value); +} + +template +FORCEINLINE constexpr T BitCeil(T Value) +{ + if (Value <= 1u) return static_cast(1); + + return static_cast(1) << Math::BitWidth(static_cast(Value - 1)); +} + +template +FORCEINLINE constexpr T BitFloor(T Value) +{ + if (Value == 0u) return static_cast(0); + + return static_cast(1) << (Math::BitWidth(static_cast(Value)) - 1); +} + +template FORCEINLINE constexpr T RotateLeft (T Value, int Offset); +template FORCEINLINE constexpr T RotateRight(T Value, int Offset); + +template +FORCEINLINE constexpr T RotateLeft(T Value, int Offset) +{ + if constexpr (CSameAs) return Value; + + else + { + if (Offset >= 0) + { + const auto Remainder = Offset % TNumericLimits::Digits; + + return static_cast((Value << Remainder) | (Value >> (TNumericLimits::Digits - Remainder))); + } + + return Math::RotateRight(Value, -Offset); + } +} + +template +FORCEINLINE constexpr T RotateRight(T Value, int Offset) +{ + if constexpr (CSameAs) return Value; + + else + { + if (Offset >= 0) + { + const auto Remainder = Offset % TNumericLimits::Digits; + + return static_cast((Value >> Remainder) | (Value << (TNumericLimits::Digits - Remainder))); + } + return Math::RotateLeft(Value, -Offset); + } +} + +enum class EEndian +{ +#if defined(__BYTE_ORDER__) && defined(__ORDER_LITTLE_ENDIAN__) && defined(__ORDER_BIG_ENDIAN__) + + Little = __ORDER_LITTLE_ENDIAN__, + Big = __ORDER_BIG_ENDIAN__, + Native = __BYTE_ORDER__, + +#elif PLATFORM_LITTLE_ENDIAN + + Little, + Big, + Native = Little, + +#elif PLATFORM_BIG_ENDIAN + + Little, + Big, + Native = Big, + +#else + + Little, + Big, + Native, + +#endif +}; + +NAMESPACE_END(Math) + +NAMESPACE_MODULE_END(Utility) +NAMESPACE_MODULE_END(Redcraft) +NAMESPACE_REDCRAFT_END diff --git a/Redcraft.Utility/Source/Public/Numeric/Limits.h b/Redcraft.Utility/Source/Public/Numeric/Limits.h index d53c60c..78622e4 100644 --- a/Redcraft.Utility/Source/Public/Numeric/Limits.h +++ b/Redcraft.Utility/Source/Public/Numeric/Limits.h @@ -3,6 +3,7 @@ #include "CoreTypes.h" #include "TypeTraits/TypeTraits.h" +#include #include NAMESPACE_REDCRAFT_BEGIN diff --git a/Redcraft.Utility/Source/Public/Numeric/Literal.h b/Redcraft.Utility/Source/Public/Numeric/Literal.h new file mode 100644 index 0000000..d8490b0 --- /dev/null +++ b/Redcraft.Utility/Source/Public/Numeric/Literal.h @@ -0,0 +1,133 @@ +#pragma once + +#include "CoreTypes.h" + +NAMESPACE_REDCRAFT_BEGIN +NAMESPACE_MODULE_BEGIN(Redcraft) +NAMESPACE_MODULE_BEGIN(Utility) + +#if PLATFORM_COMPILER_MSVC +# pragma warning(push) +# pragma warning(disable : 4455) +#elif PLATFORM_COMPILER_GCC || PLATFORM_COMPILER_CLANG +# pragma GCC diagnostic push +# pragma GCC diagnostic ignored "-Wliteral-suffix" +#endif + +FORCEINLINE constexpr int8 operator""i8 (unsigned long long int Value) { return static_cast(Value); } +FORCEINLINE constexpr int16 operator""i16(unsigned long long int Value) { return static_cast(Value); } +FORCEINLINE constexpr int32 operator""i32(unsigned long long int Value) { return static_cast(Value); } +FORCEINLINE constexpr int64 operator""i64(unsigned long long int Value) { return static_cast(Value); } + +FORCEINLINE constexpr int8 operator""I8 (unsigned long long int Value) { return static_cast(Value); } +FORCEINLINE constexpr int16 operator""I16(unsigned long long int Value) { return static_cast(Value); } +FORCEINLINE constexpr int32 operator""I32(unsigned long long int Value) { return static_cast(Value); } +FORCEINLINE constexpr int64 operator""I64(unsigned long long int Value) { return static_cast(Value); } + +FORCEINLINE constexpr uint8 operator""u8 (unsigned long long int Value) { return static_cast(Value); } +FORCEINLINE constexpr uint16 operator""u16(unsigned long long int Value) { return static_cast(Value); } +FORCEINLINE constexpr uint32 operator""u32(unsigned long long int Value) { return static_cast(Value); } +FORCEINLINE constexpr uint64 operator""u64(unsigned long long int Value) { return static_cast(Value); } + +FORCEINLINE constexpr uint8 operator""U8 (unsigned long long int Value) { return static_cast(Value); } +FORCEINLINE constexpr uint16 operator""U16(unsigned long long int Value) { return static_cast(Value); } +FORCEINLINE constexpr uint32 operator""U32(unsigned long long int Value) { return static_cast(Value); } +FORCEINLINE constexpr uint64 operator""U64(unsigned long long int Value) { return static_cast(Value); } + +#if PLATFORM_HAS_INT128 + +FORCEINLINE constexpr int128 operator""i128(const char* Str); +FORCEINLINE constexpr uint128 operator""u128(const char* Str); + +FORCEINLINE constexpr int128 operator""I128(const char* Str) { return operator""i128(Str); } +FORCEINLINE constexpr uint128 operator""U128(const char* Str) { return operator""u128(Str); } + +FORCEINLINE constexpr int128 operator""i128(const char* Str) +{ + return static_cast(operator""u128(Str)); +} + +FORCEINLINE constexpr uint128 operator""u128(const char* Str) +{ + constexpr uint8 DigitFromChar[] = + { + 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, + 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, + 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, + 0x00, 0x01, 0x02, 0x03, 0x04, 0x05, 0x06, 0x07, 0x08, 0x09, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, + 0xFF, 0x0A, 0x0B, 0x0C, 0x0D, 0x0E, 0x0F, 0x10, 0x11, 0x12, 0x13, 0x14, 0x15, 0x16, 0x17, 0x18, + 0x19, 0x1A, 0x1B, 0x1C, 0x1D, 0x1E, 0x1F, 0x20, 0x21, 0x22, 0x23, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, + 0xFF, 0x0A, 0x0B, 0x0C, 0x0D, 0x0E, 0x0F, 0x10, 0x11, 0x12, 0x13, 0x14, 0x15, 0x16, 0x17, 0x18, + 0x19, 0x1A, 0x1B, 0x1C, 0x1D, 0x1E, 0x1F, 0x20, 0x21, 0x22, 0x23, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, + 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, + 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, + 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, + 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, + 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, + 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, + 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, + 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, + }; + + uint128 Result = 0; + + uint Base = 10; + + size_t BeginIndex = 0; + + if (Str[0] == '0') + { + if (Str[1] == 'x' || Str[1] == 'X') + { + Base = 16; + BeginIndex += 2; + } + else if (Str[1] == 'b' || Str[1] == 'B') + { + Base = 2; + BeginIndex += 2; + } + else Base = 8; + } + + for (size_t I = BeginIndex; Str[I]; ++I) + { + Result = Result * Base + DigitFromChar[Str[I]]; + } + + return Result; +} + +#endif + +#if PLATFORM_HAS_INT128 +FORCEINLINE constexpr intmax operator""imax(const char* Str) { return operator""i128(Str); } +FORCEINLINE constexpr intmax operator""IMAX(const char* Str) { return operator""I128(Str); } +FORCEINLINE constexpr uintmax operator""umax(const char* Str) { return operator""u128(Str); } +FORCEINLINE constexpr uintmax operator""UMAX(const char* Str) { return operator""U128(Str); } +#else +FORCEINLINE constexpr intmax operator""imax(unsigned long long int Value) { return operator""i64(Value); } +FORCEINLINE constexpr intmax operator""IMAX(unsigned long long int Value) { return operator""I64(Value); } +FORCEINLINE constexpr uintmax operator""umax(unsigned long long int Value) { return operator""u64(Value); } +FORCEINLINE constexpr uintmax operator""UMAX(unsigned long long int Value) { return operator""U64(Value); } +#endif + +#ifndef __STDCPP_FLOAT32_T__ +FORCEINLINE constexpr float32 operator""f32(long double Value) { return static_cast(Value); } +FORCEINLINE constexpr float32 operator""F32(long double Value) { return static_cast(Value); } +#endif + +#ifndef __STDCPP_FLOAT64_T__ +FORCEINLINE constexpr float64 operator""f64(long double Value) { return static_cast(Value); } +FORCEINLINE constexpr float64 operator""F64(long double Value) { return static_cast(Value); } +#endif + +#if PLATFORM_COMPILER_MSVC +# pragma warning(pop) +#elif PLATFORM_COMPILER_GCC || PLATFORM_COMPILER_CLANG +# pragma GCC diagnostic pop +#endif + +NAMESPACE_MODULE_END(Utility) +NAMESPACE_MODULE_END(Redcraft) +NAMESPACE_REDCRAFT_END diff --git a/Redcraft.Utility/Source/Public/Numeric/Numeric.h b/Redcraft.Utility/Source/Public/Numeric/Numeric.h new file mode 100644 index 0000000..d2b2139 --- /dev/null +++ b/Redcraft.Utility/Source/Public/Numeric/Numeric.h @@ -0,0 +1,7 @@ +#pragma once + +#include "CoreTypes.h" +#include "Numeric/Literal.h" +#include "Numeric/Limits.h" +#include "Numeric/Numbers.h" +#include "Numeric/Bit.h" diff --git a/Redcraft.Utility/Source/Public/Testing/NumericTesting.h b/Redcraft.Utility/Source/Public/Testing/NumericTesting.h new file mode 100644 index 0000000..4f48614 --- /dev/null +++ b/Redcraft.Utility/Source/Public/Testing/NumericTesting.h @@ -0,0 +1,19 @@ +#pragma once + +#include "CoreTypes.h" + +NAMESPACE_REDCRAFT_BEGIN +NAMESPACE_MODULE_BEGIN(Redcraft) +NAMESPACE_MODULE_BEGIN(Utility) + +NAMESPACE_BEGIN(Testing) + +REDCRAFTUTILITY_API void TestNumeric(); +REDCRAFTUTILITY_API void TestLiteral(); +REDCRAFTUTILITY_API void TestBit(); + +NAMESPACE_END(Testing) + +NAMESPACE_MODULE_END(Utility) +NAMESPACE_MODULE_END(Redcraft) +NAMESPACE_REDCRAFT_END