Tests for float16

This commit is contained in:
Vladyslav Baranovskyi
2024-11-05 21:42:39 +02:00
parent 2a27e3049f
commit 8066b2d059

139
tests/Math/Float16.cpp Normal file
View File

@@ -0,0 +1,139 @@
/*
* 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 https://mozilla.org/MPL/2.0/.
*/
#include <catch2/catch_all.hpp>
#include <cmath>
#include <limits>
#include "Math/Float16.hpp"
// using namespace OpenVulkano::Math;
namespace
{
bool almostEqual(float a, float b, float epsilon = 0.001f) { return std::fabs(a - b) < epsilon; }
}
TEST_CASE("Basic Conversions", "[float16]")
{
REQUIRE(almostEqual(static_cast<float>(fp16(0.0f)), 0.0f));
REQUIRE(almostEqual(static_cast<float>(fp16(-0.0f)), -0.0f));
REQUIRE(almostEqual(static_cast<float>(fp16(1.0f)), 1.0f));
REQUIRE(almostEqual(static_cast<float>(fp16(-1.0f)), -1.0f));
REQUIRE(almostEqual(static_cast<float>(fp16(0.5f)), 0.5f));
REQUIRE(almostEqual(static_cast<float>(fp16(-0.5f)), -0.5f));
REQUIRE(almostEqual(static_cast<float>(fp16(65504.0f)), 65504.0f));
REQUIRE(almostEqual(static_cast<float>(fp16(-65504.0f)), -65504.0f));
}
TEST_CASE("Infinity and NaN", "[float16]")
{
fp16 pos_inf = std::numeric_limits<fp16>::infinity();
fp16 neg_inf = -std::numeric_limits<fp16>::infinity();
fp16 nan_val = std::numeric_limits<fp16>::quiet_NaN();
REQUIRE(std::isinf(static_cast<float>(pos_inf)));
REQUIRE(std::isinf(static_cast<float>(neg_inf)));
REQUIRE(std::isnan(static_cast<float>(nan_val)));
REQUIRE(pos_inf > fp16(65504.0f));
REQUIRE(neg_inf < fp16(-65504.0f));
}
TEST_CASE("Denormalized Values", "[float16]")
{
fp16 min_denorm = std::numeric_limits<fp16>::denorm_min();
REQUIRE(almostEqual(static_cast<float>(min_denorm), 5.96e-8f));
fp16 denorm_value(1.0e-5f);
REQUIRE(almostEqual(static_cast<float>(denorm_value), 0.0f));
}
TEST_CASE("Special Cases", "[float16]")
{
fp16 zero(0.0f), neg_zero(-0.0f), epsilon_fp(std::numeric_limits<fp16>::epsilon());
fp16 min_normal(std::numeric_limits<fp16>::min());
fp16 max_fp(std::numeric_limits<fp16>::max());
REQUIRE(almostEqual(static_cast<float>(zero), 0.0f));
REQUIRE(almostEqual(static_cast<float>(neg_zero), 0.0f));
REQUIRE(almostEqual(static_cast<float>(epsilon_fp), 0.00097656f));
REQUIRE(almostEqual(static_cast<float>(min_normal), 6.10352e-5f));
REQUIRE(almostEqual(static_cast<float>(max_fp), 65504.0f));
}
TEST_CASE("Arithmetic Operations", "[float16]")
{
fp16 a(1.0f), b(2.0f), c(3.0f), d(4.0f), large(65504.0f), small(0.5f);
REQUIRE(almostEqual(static_cast<float>(a + b), 3.0f));
REQUIRE(almostEqual(static_cast<float>(b - a), 1.0f));
REQUIRE(almostEqual(static_cast<float>(a * b), 2.0f));
REQUIRE(almostEqual(static_cast<float>(d / c), 1.33333f));
REQUIRE(almostEqual(static_cast<float>(large * small), 32752.0f));
REQUIRE(std::isinf(static_cast<float>(large + large)));
REQUIRE(almostEqual(static_cast<float>(a - a), 0.0f));
}
TEST_CASE("Comparisons", "[float16]")
{
fp16 a(1.0f), b(2.0f), zero(0.0f), neg(-1.0f), inf = std::numeric_limits<fp16>::infinity();
REQUIRE(a < b);
REQUIRE(b > a);
REQUIRE(zero == -zero);
REQUIRE(a != neg);
REQUIRE(inf > b);
REQUIRE(!(inf < b));
}
TEST_CASE("Arithmetic with Special Values", "[float16]")
{
fp16 a(2.0f), inf = std::numeric_limits<fp16>::infinity(), nan = std::numeric_limits<fp16>::quiet_NaN();
REQUIRE(std::isinf(static_cast<float>(a + inf)));
REQUIRE(std::isinf(static_cast<float>(inf - a)));
REQUIRE(std::isnan(static_cast<float>(inf - inf)));
REQUIRE(std::isnan(static_cast<float>(inf / inf)));
REQUIRE(std::isnan(static_cast<float>(nan + a)));
REQUIRE(std::isnan(static_cast<float>(nan - a)));
REQUIRE(std::isnan(static_cast<float>(a * nan)));
REQUIRE(std::isnan(static_cast<float>(a / nan)));
}
TEST_CASE("Boundary Tests", "[float16]")
{
fp16 near_min(6.10352e-5f), near_max(65504.0f), tiny_pos(1.0e-8f), tiny_neg(-1.0e-8f);
REQUIRE(almostEqual(static_cast<float>(near_min), 6.10352e-5f));
REQUIRE(almostEqual(static_cast<float>(near_max), 65504.0f));
REQUIRE(almostEqual(static_cast<float>(tiny_pos), 0.0f));
REQUIRE(almostEqual(static_cast<float>(tiny_neg), 0.0f));
}
TEST_CASE("Edge Case Conversions", "[float16]")
{
fp16 max_convertible_fp(65504.0f), too_large(70000.0f), too_small(1.0e-8f);
REQUIRE(almostEqual(static_cast<float>(max_convertible_fp), 65504.0f));
REQUIRE(std::isinf(static_cast<float>(too_large)));
REQUIRE(almostEqual(static_cast<float>(too_small), 0.0f));
}
TEST_CASE("Special Comparison with Infinity and NaN", "[float16]")
{
fp16 inf = std::numeric_limits<fp16>::infinity();
fp16 neg_inf = -std::numeric_limits<fp16>::infinity();
fp16 nan = std::numeric_limits<fp16>::quiet_NaN();
fp16 one(1.0f);
REQUIRE(inf > one);
REQUIRE(neg_inf < -one);
REQUIRE(std::isnan(static_cast<float>(nan)));
REQUIRE(nan != nan);
}