diff options
Diffstat (limited to 'externals/dynarmic/tests/decoder_tests.cpp')
-rw-r--r-- | externals/dynarmic/tests/decoder_tests.cpp | 75 |
1 files changed, 75 insertions, 0 deletions
diff --git a/externals/dynarmic/tests/decoder_tests.cpp b/externals/dynarmic/tests/decoder_tests.cpp new file mode 100644 index 0000000000..2028ac98cd --- /dev/null +++ b/externals/dynarmic/tests/decoder_tests.cpp @@ -0,0 +1,75 @@ +/* This file is part of the dynarmic project. + * Copyright (c) 2020 MerryMage + * SPDX-License-Identifier: 0BSD + */ + +#include <cstring> +#include <iomanip> +#include <iostream> + +#include <catch2/catch_test_macros.hpp> +#include <mcl/assert.hpp> + +#include "dynarmic/frontend/A32/decoder/asimd.h" +#include "dynarmic/frontend/A32/translate/impl/a32_translate_impl.h" +#include "dynarmic/interface/A32/config.h" +#include "dynarmic/ir/opcodes.h" + +using namespace Dynarmic; + +TEST_CASE("ASIMD Decoder: Ensure table order correctness", "[decode][a32][.]") { + const auto table = A32::GetASIMDDecodeTable<A32::TranslatorVisitor>(); + + const auto get_ir = [](const A32::ASIMDMatcher<A32::TranslatorVisitor>& matcher, u32 instruction) { + ASSERT(matcher.Matches(instruction)); + + const A32::LocationDescriptor location{0, {}, {}}; + IR::Block block{location}; + A32::TranslatorVisitor visitor{block, location, {}}; + matcher.call(visitor, instruction); + + return block; + }; + + const auto is_decode_error = [&get_ir](const A32::ASIMDMatcher<A32::TranslatorVisitor>& matcher, u32 instruction) { + const auto block = get_ir(matcher, instruction); + + for (const auto& ir_inst : block) { + if (ir_inst.GetOpcode() == IR::Opcode::A32ExceptionRaised) { + if (static_cast<A32::Exception>(ir_inst.GetArg(1).GetU64()) == A32::Exception::DecodeError) { + return true; + } + } + } + return false; + }; + + for (auto iter = table.cbegin(); iter != table.cend(); ++iter) { + if (std::strncmp(iter->GetName(), "UNALLOCATED", 11) == 0) { + continue; + } + + const u32 expect = iter->GetExpected(); + const u32 mask = iter->GetMask(); + u32 x = 0; + do { + const u32 instruction = expect | x; + + const bool iserr = is_decode_error(*iter, instruction); + const auto alternative = std::find_if(table.cbegin(), iter, [instruction](const auto& m) { return m.Matches(instruction); }); + const bool altiserr = is_decode_error(*alternative, instruction); + + INFO("Instruction: " << std::hex << std::setfill('0') << std::setw(8) << instruction); + INFO("Expect: " << std::hex << std::setfill('0') << std::setw(8) << expect); + INFO("Fill: " << std::hex << std::setfill('0') << std::setw(8) << x); + INFO("Name: " << iter->GetName()); + INFO("iserr: " << iserr); + INFO("alternative: " << alternative->GetName()); + INFO("altiserr: " << altiserr); + + REQUIRE(((!iserr && alternative == iter) || (iserr && alternative != iter && !altiserr))); + + x = ((x | mask) + 1) & ~mask; + } while (x != 0); + } +} |