From 18df0664639a905a70a8ab4a5d771d5f8d3224d1 Mon Sep 17 00:00:00 2001 From: Wunk Date: Mon, 11 Mar 2024 15:53:49 -0700 Subject: [PATCH] Add shader unit-testing (#457) * Initialize catch-2 based unit tests * Add nihstro submodule Enabled only during testing to help with assembling shaders in-code. * Implement `ADD` instruction unit-test * Add arithmetic/logical instruction unit tests * Add embedded catch2 submodule Will use the host catch2 if available. --- .gitmodules | 6 ++ CMakeLists.txt | 26 +++++ tests/dynapica.cpp | 229 ++++++++++++++++++++++++++++++++++++++++++++ third_party/Catch2 | 1 + third_party/nihstro | 1 + 5 files changed, 263 insertions(+) create mode 100644 tests/dynapica.cpp create mode 160000 third_party/Catch2 create mode 160000 third_party/nihstro diff --git a/.gitmodules b/.gitmodules index 6c69fe14..25980054 100644 --- a/.gitmodules +++ b/.gitmodules @@ -61,3 +61,9 @@ [submodule "third_party/dynarmic"] path = third_party/dynarmic url = https://github.com/Panda3DS-emu/dynarmic +[submodule "third_party/nihstro"] + path = third_party/nihstro + url = https://github.com/neobrain/nihstro.git +[submodule "third_party/Catch2"] + path = third_party/Catch2 + url = https://github.com/catchorg/Catch2.git diff --git a/CMakeLists.txt b/CMakeLists.txt index d8a69acb..629601ab 100644 --- a/CMakeLists.txt +++ b/CMakeLists.txt @@ -33,6 +33,7 @@ option(GPU_DEBUG_INFO "Enable additional GPU debugging info" OFF) option(ENABLE_OPENGL "Enable OpenGL rendering backend" ON) option(ENABLE_VULKAN "Enable Vulkan rendering backend" ON) option(ENABLE_LTO "Enable link-time optimization" OFF) +option(ENABLE_TESTS "Compile unit-tests" OFF) option(ENABLE_USER_BUILD "Make a user-facing build. These builds have various assertions disabled, LTO, and more" OFF) option(ENABLE_HTTP_SERVER "Enable HTTP server. Used for Discord bot support" OFF) option(ENABLE_DISCORD_RPC "Compile with Discord RPC support (disabled by default)" ON) @@ -520,3 +521,28 @@ endif() if(ENABLE_LTO OR ENABLE_USER_BUILD) set_target_properties(Alber PROPERTIES INTERPROCEDURAL_OPTIMIZATION TRUE) endif() + +if(ENABLE_TESTS) + enable_testing() + + find_package(Catch2 3) + if(NOT Catch2_FOUND) + add_subdirectory(third_party/Catch2) + endif() + + add_library(nihstro-headers INTERFACE) + target_include_directories(nihstro-headers SYSTEM INTERFACE ./third_party/nihstro/include) + + add_executable(AlberTests + tests/dynapica.cpp + ) + target_link_libraries( + AlberTests + PRIVATE + Catch2::Catch2WithMain + AlberCore + nihstro-headers + ) + + add_test(AlberTests AlberTests) +endif() \ No newline at end of file diff --git a/tests/dynapica.cpp b/tests/dynapica.cpp new file mode 100644 index 00000000..32e7bdb0 --- /dev/null +++ b/tests/dynapica.cpp @@ -0,0 +1,229 @@ +#include + +#include +#include +#include +#include +#include +#include + +using namespace Floats; +static const nihstro::SourceRegister input0 = nihstro::SourceRegister::MakeInput(0); +static const nihstro::SourceRegister input1 = nihstro::SourceRegister::MakeInput(1); +static const nihstro::DestRegister output0 = nihstro::DestRegister::MakeOutput(0); + +static std::unique_ptr assembleVertexShader(std::initializer_list code) { + const auto shaderBinary = nihstro::InlineAsm::CompileToRawBinary(code); + auto newShader = std::make_unique(ShaderType::Vertex); + newShader->reset(); + + for (const nihstro::Instruction& instruction : shaderBinary.program) { + newShader->uploadWord(instruction.hex); + } + for (const nihstro::SwizzlePattern& swizzle : shaderBinary.swizzle_table) { + newShader->uploadDescriptor(swizzle.hex); + } + newShader->finalize(); + return newShader; +} + +class VertexShaderTest { + private: + std::unique_ptr shader; + + public: + explicit VertexShaderTest(std::initializer_list code) : shader(assembleVertexShader(code)) {} + + // Multiple inputs, singular scalar output + float runScalar(std::initializer_list inputs) { + usize inputIndex = 0; + for (const float& input : inputs) { + const std::array input_vec = std::array{f24::fromFloat32(input), f24::zero(), f24::zero(), f24::zero()}; + shader->inputs[inputIndex++] = input_vec; + } + shader->run(); + return shader->outputs[0][0]; + } + + static std::unique_ptr assembleTest(std::initializer_list code) { + return std::make_unique(code); + } +}; + +TEST_CASE("ADD", "[shader][vertex]") { + const auto shader = VertexShaderTest::assembleTest({ + {nihstro::OpCode::Id::ADD, output0, input0, input1}, + {nihstro::OpCode::Id::END}, + }); + + REQUIRE(shader->runScalar({+1.0f, -1.0f}) == +0.0f); + REQUIRE(shader->runScalar({+0.0f, -0.0f}) == -0.0f); + REQUIRE(std::isnan(shader->runScalar({+INFINITY, -INFINITY}))); + REQUIRE(std::isinf(shader->runScalar({INFINITY, +1.0f}))); + REQUIRE(std::isinf(shader->runScalar({INFINITY, -1.0f}))); +} + +TEST_CASE("MUL", "[shader][vertex]") { + const auto shader = VertexShaderTest::assembleTest({ + {nihstro::OpCode::Id::MUL, output0, input0, input1}, + {nihstro::OpCode::Id::END}, + }); + + REQUIRE(shader->runScalar({+1.0f, -1.0f}) == -1.0f); + REQUIRE(shader->runScalar({-1.0f, +1.0f}) == -1.0f); + REQUIRE(shader->runScalar({INFINITY, 0.0f}) == 0.0f); + REQUIRE(shader->runScalar({+INFINITY, +INFINITY}) == INFINITY); + REQUIRE(shader->runScalar({+INFINITY, -INFINITY}) == -INFINITY); + REQUIRE(std::isnan(shader->runScalar({NAN, 0.0f}))); +} + +TEST_CASE("RCP", "[shader][vertex]") { + const auto shader = VertexShaderTest::assembleTest({ + {nihstro::OpCode::Id::RCP, output0, input0}, + {nihstro::OpCode::Id::END}, + }); + + // REQUIRE(shader->RunScalar({-0.0f}) == INFINITY); // Violates IEEE + REQUIRE(shader->runScalar({0.0f}) == INFINITY); + REQUIRE(shader->runScalar({INFINITY}) == 0.0f); + REQUIRE(std::isnan(shader->runScalar({NAN}))); + + REQUIRE(shader->runScalar({16.0f}) == Catch::Approx(0.0625f).margin(0.001f)); + REQUIRE(shader->runScalar({8.0f}) == Catch::Approx(0.125f).margin(0.001f)); + REQUIRE(shader->runScalar({4.0f}) == Catch::Approx(0.25f).margin(0.001f)); + REQUIRE(shader->runScalar({2.0f}) == Catch::Approx(0.5f).margin(0.001f)); + REQUIRE(shader->runScalar({1.0f}) == Catch::Approx(1.0f).margin(0.001f)); + REQUIRE(shader->runScalar({0.5f}) == Catch::Approx(2.0f).margin(0.001f)); + REQUIRE(shader->runScalar({0.25f}) == Catch::Approx(4.0f).margin(0.001f)); + REQUIRE(shader->runScalar({0.125f}) == Catch::Approx(8.0f).margin(0.002f)); + REQUIRE(shader->runScalar({0.0625f}) == Catch::Approx(16.0f).margin(0.004f)); +} + +TEST_CASE("RSQ", "[shader][vertex]") { + const auto shader = VertexShaderTest::assembleTest({ + {nihstro::OpCode::Id::RSQ, output0, input0}, + {nihstro::OpCode::Id::END}, + }); + + REQUIRE(shader->runScalar({-0.0f}) == INFINITY); + REQUIRE(shader->runScalar({INFINITY}) == 0.0f); + REQUIRE(std::isnan(shader->runScalar({-2.0f}))); + REQUIRE(std::isnan(shader->runScalar({-INFINITY}))); + REQUIRE(std::isnan(shader->runScalar({NAN}))); + REQUIRE(shader->runScalar({16.0f}) == Catch::Approx(0.25f).margin(0.001f)); + REQUIRE(shader->runScalar({8.0f}) == Catch::Approx(1.0f / std::sqrt(8.0f)).margin(0.001f)); + REQUIRE(shader->runScalar({4.0f}) == Catch::Approx(0.5f).margin(0.001f)); + REQUIRE(shader->runScalar({2.0f}) == Catch::Approx(1.0f / std::sqrt(2.0f)).margin(0.001f)); + REQUIRE(shader->runScalar({1.0f}) == Catch::Approx(1.0f).margin(0.001f)); + REQUIRE(shader->runScalar({0.5f}) == Catch::Approx(1.0f / std::sqrt(0.5f)).margin(0.001f)); + REQUIRE(shader->runScalar({0.25f}) == Catch::Approx(2.0f).margin(0.001f)); + REQUIRE(shader->runScalar({0.125f}) == Catch::Approx(1.0 / std::sqrt(0.125)).margin(0.002f)); + REQUIRE(shader->runScalar({0.0625f}) == Catch::Approx(4.0f).margin(0.004f)); +} + +TEST_CASE("LG2", "[shader][vertex]") { + const auto shader = VertexShaderTest::assembleTest({ + {nihstro::OpCode::Id::LG2, output0, input0}, + {nihstro::OpCode::Id::END}, + }); + + REQUIRE(std::isnan(shader->runScalar({NAN}))); + REQUIRE(std::isnan(shader->runScalar({-1.f}))); + REQUIRE(std::isinf(shader->runScalar({0.f}))); + REQUIRE(shader->runScalar({4.f}) == Catch::Approx(2.f)); + REQUIRE(shader->runScalar({64.f}) == Catch::Approx(6.f)); + REQUIRE(shader->runScalar({1.e24f}) == Catch::Approx(79.7262742773f)); +} + +TEST_CASE("EX2", "[shader][vertex]") { + const auto shader = VertexShaderTest::assembleTest({ + {nihstro::OpCode::Id::EX2, output0, input0}, + {nihstro::OpCode::Id::END}, + }); + + REQUIRE(std::isnan(shader->runScalar({NAN}))); + REQUIRE(shader->runScalar({-800.f}) == Catch::Approx(0.f)); + REQUIRE(shader->runScalar({0.f}) == Catch::Approx(1.f)); + REQUIRE(shader->runScalar({2.f}) == Catch::Approx(4.f)); + REQUIRE(shader->runScalar({6.f}) == Catch::Approx(64.f)); + REQUIRE(shader->runScalar({79.7262742773f}) == Catch::Approx(1.e24f)); + REQUIRE(std::isinf(shader->runScalar({800.f}))); +} + +TEST_CASE("MAX", "[shader][vertex]") { + const auto shader = VertexShaderTest::assembleTest({ + {nihstro::OpCode::Id::MAX, output0, input0, input1}, + {nihstro::OpCode::Id::END}, + }); + + REQUIRE(shader->runScalar({1.0f, 0.0f}) == 1.0f); + REQUIRE(shader->runScalar({0.0f, 1.0f}) == 1.0f); + REQUIRE(shader->runScalar({0.0f, +INFINITY}) == +INFINITY); + REQUIRE(shader->runScalar({0.0f, -INFINITY}) == -INFINITY); + REQUIRE(shader->runScalar({NAN, 0.0f}) == 0.0f); + REQUIRE(shader->runScalar({-INFINITY, +INFINITY}) == +INFINITY); + REQUIRE(std::isnan(shader->runScalar({0.0f, NAN}))); +} + +TEST_CASE("MIN", "[shader][vertex]") { + const auto shader = VertexShaderTest::assembleTest({ + {nihstro::OpCode::Id::MIN, output0, input0, input1}, + {nihstro::OpCode::Id::END}, + }); + + REQUIRE(shader->runScalar({1.0f, 0.0f}) == 0.0f); + REQUIRE(shader->runScalar({0.0f, 1.0f}) == 0.0f); + REQUIRE(shader->runScalar({0.0f, +INFINITY}) == 0.0f); + REQUIRE(shader->runScalar({0.0f, -INFINITY}) == -INFINITY); + REQUIRE(shader->runScalar({NAN, 0.0f}) == 0.0f); + REQUIRE(shader->runScalar({-INFINITY, +INFINITY}) == -INFINITY); + REQUIRE(std::isnan(shader->runScalar({0.0f, NAN}))); +} + +TEST_CASE("SGE", "[shader][vertex]") { + const auto shader = VertexShaderTest::assembleTest({ + {nihstro::OpCode::Id::SGE, output0, input0, input1}, + {nihstro::OpCode::Id::END}, + }); + + REQUIRE(shader->runScalar({INFINITY, 0.0f}) == 1.0f); + REQUIRE(shader->runScalar({0.0f, INFINITY}) == 0.0f); + REQUIRE(shader->runScalar({NAN, 0.0f}) == 0.0f); + REQUIRE(shader->runScalar({0.0f, NAN}) == 0.0f); + REQUIRE(shader->runScalar({+INFINITY, +INFINITY}) == 1.0f); + REQUIRE(shader->runScalar({+INFINITY, -INFINITY}) == 1.0f); + REQUIRE(shader->runScalar({-INFINITY, +INFINITY}) == 0.0f); + REQUIRE(shader->runScalar({+1.0f, -1.0f}) == 1.0f); + REQUIRE(shader->runScalar({-1.0f, +1.0f}) == 0.0f); +} + +TEST_CASE("SLT", "[shader][vertex]") { + const auto shader = VertexShaderTest::assembleTest({ + {nihstro::OpCode::Id::SLT, output0, input0, input1}, + {nihstro::OpCode::Id::END}, + }); + + REQUIRE(shader->runScalar({INFINITY, 0.0f}) == 0.0f); + REQUIRE(shader->runScalar({0.0f, INFINITY}) == 1.0f); + REQUIRE(shader->runScalar({NAN, 0.0f}) == 0.0f); + REQUIRE(shader->runScalar({0.0f, NAN}) == 0.0f); + REQUIRE(shader->runScalar({+INFINITY, +INFINITY}) == 0.0f); + REQUIRE(shader->runScalar({+INFINITY, -INFINITY}) == 0.0f); + REQUIRE(shader->runScalar({-INFINITY, +INFINITY}) == 1.0f); + REQUIRE(shader->runScalar({+1.0f, -1.0f}) == 0.0f); + REQUIRE(shader->runScalar({-1.0f, +1.0f}) == 1.0f); +} + +TEST_CASE("FLR", "[shader][vertex]") { + const auto shader = VertexShaderTest::assembleTest({ + {nihstro::OpCode::Id::FLR, output0, input0}, + {nihstro::OpCode::Id::END}, + }); + + REQUIRE(shader->runScalar({0.5}) == 0.0f); + REQUIRE(shader->runScalar({-0.5}) == -1.0f); + REQUIRE(shader->runScalar({1.5}) == 1.0f); + REQUIRE(shader->runScalar({-1.5}) == -2.0f); + REQUIRE(std::isnan(shader->runScalar({NAN}))); + REQUIRE(std::isinf(shader->runScalar({INFINITY}))); +} \ No newline at end of file diff --git a/third_party/Catch2 b/third_party/Catch2 new file mode 160000 index 00000000..4acc5182 --- /dev/null +++ b/third_party/Catch2 @@ -0,0 +1 @@ +Subproject commit 4acc51828f7f93f3b2058a63f54d112af4034503 diff --git a/third_party/nihstro b/third_party/nihstro new file mode 160000 index 00000000..e924e21b --- /dev/null +++ b/third_party/nihstro @@ -0,0 +1 @@ +Subproject commit e924e21b1da60170f0f0a4e5a073cb7d579969c0