2018-01-06 22:15:25 +01:00
|
|
|
/* This file is part of the dynarmic project.
|
2018-01-07 01:11:57 +01:00
|
|
|
* Copyright (c) 2018 MerryMage
|
2018-01-06 22:15:25 +01:00
|
|
|
* This software may be used and distributed according to the terms of the GNU
|
|
|
|
* General Public License version 2 or any later version.
|
|
|
|
*/
|
|
|
|
|
|
|
|
#include <array>
|
|
|
|
|
|
|
|
#include <catch.hpp>
|
|
|
|
|
|
|
|
#include <dynarmic/A64/a64.h>
|
|
|
|
|
|
|
|
#include "common/assert.h"
|
|
|
|
#include "common/common_types.h"
|
|
|
|
|
|
|
|
class TestEnv final : public Dynarmic::A64::UserCallbacks {
|
2018-01-07 01:11:57 +01:00
|
|
|
public:
|
2018-01-06 22:15:25 +01:00
|
|
|
u64 ticks_left = 0;
|
|
|
|
std::array<u32, 3000> code_mem{};
|
|
|
|
|
|
|
|
std::uint32_t MemoryReadCode(u64 vaddr) override {
|
|
|
|
if (vaddr < code_mem.size() * sizeof(u32)) {
|
|
|
|
size_t index = vaddr / sizeof(u32);
|
|
|
|
return code_mem[index];
|
|
|
|
}
|
2018-01-07 01:11:57 +01:00
|
|
|
return 0x14000000; // B .
|
2018-01-06 22:15:25 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
std::uint8_t MemoryRead8(u64 vaddr) override { ASSERT_MSG(false, "MemoryRead8(%llx)", vaddr); }
|
|
|
|
std::uint16_t MemoryRead16(u64 vaddr) override { ASSERT_MSG(false, "MemoryRead16(%llx)", vaddr); }
|
|
|
|
std::uint32_t MemoryRead32(u64 vaddr) override { ASSERT_MSG(false, "MemoryRead32(%llx)", vaddr); }
|
|
|
|
std::uint64_t MemoryRead64(u64 vaddr) override { ASSERT_MSG(false, "MemoryRead64(%llx)", vaddr); }
|
|
|
|
|
|
|
|
void MemoryWrite8(u64 vaddr, std::uint8_t value) override { ASSERT_MSG(false, "MemoryWrite8(%llx, %hhi)", vaddr, value); }
|
|
|
|
void MemoryWrite16(u64 vaddr, std::uint16_t value) override { ASSERT_MSG(false, "MemoryWrite16(%llx, %hi)", vaddr, value); }
|
|
|
|
void MemoryWrite32(u64 vaddr, std::uint32_t value) override { ASSERT_MSG(false, "MemoryWrite32(%llx, %i)", vaddr, value); }
|
|
|
|
void MemoryWrite64(u64 vaddr, std::uint64_t value) override { ASSERT_MSG(false, "MemoryWrite64(%llx, %lli)", vaddr, value); }
|
|
|
|
|
|
|
|
void InterpreterFallback(u64 pc, size_t num_instructions) override { ASSERT_MSG(false, "InterpreterFallback(%llx, %zu)", pc, num_instructions); }
|
|
|
|
|
|
|
|
void CallSVC(std::uint32_t swi) override { ASSERT_MSG(false, "CallSVC(%u)", swi); }
|
|
|
|
|
|
|
|
void AddTicks(std::uint64_t ticks) override {
|
|
|
|
if (ticks > ticks_left) {
|
|
|
|
ticks_left = 0;
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
ticks_left -= ticks;
|
|
|
|
}
|
|
|
|
std::uint64_t GetTicksRemaining() override {
|
|
|
|
return ticks_left;
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
2018-01-07 01:11:57 +01:00
|
|
|
TEST_CASE("A64: ADD", "[a64]") {
|
2018-01-06 22:15:25 +01:00
|
|
|
TestEnv env;
|
|
|
|
Dynarmic::A64::Jit jit{Dynarmic::A64::UserConfig{&env}};
|
2018-01-07 01:11:57 +01:00
|
|
|
|
|
|
|
env.code_mem[0] = 0x8b020020; // ADD X0, X1, X2
|
|
|
|
env.code_mem[1] = 0x14000000; // B .
|
|
|
|
|
|
|
|
jit.SetRegister(0, 0);
|
|
|
|
jit.SetRegister(1, 1);
|
|
|
|
jit.SetRegister(2, 2);
|
|
|
|
jit.SetPC(0);
|
|
|
|
|
|
|
|
env.ticks_left = 2;
|
|
|
|
jit.Run();
|
|
|
|
|
|
|
|
REQUIRE(jit.GetRegister(0) == 3);
|
|
|
|
REQUIRE(jit.GetRegister(1) == 1);
|
|
|
|
REQUIRE(jit.GetRegister(2) == 2);
|
|
|
|
REQUIRE(jit.GetPC() == 4);
|
|
|
|
}
|
2018-01-07 13:52:12 +01:00
|
|
|
|
|
|
|
TEST_CASE("A64: AND", "[a64]") {
|
|
|
|
TestEnv env;
|
|
|
|
Dynarmic::A64::Jit jit{Dynarmic::A64::UserConfig{&env}};
|
|
|
|
|
|
|
|
env.code_mem[0] = 0x8a020020; // AND X0, X1, X2
|
|
|
|
env.code_mem[1] = 0x14000000; // B .
|
|
|
|
|
|
|
|
jit.SetRegister(0, 0);
|
|
|
|
jit.SetRegister(1, 1);
|
|
|
|
jit.SetRegister(2, 3);
|
|
|
|
jit.SetPC(0);
|
|
|
|
|
|
|
|
env.ticks_left = 2;
|
|
|
|
jit.Run();
|
|
|
|
|
|
|
|
REQUIRE(jit.GetRegister(0) == 1);
|
|
|
|
REQUIRE(jit.GetRegister(1) == 1);
|
|
|
|
REQUIRE(jit.GetRegister(2) == 3);
|
|
|
|
REQUIRE(jit.GetPC() == 4);
|
|
|
|
}
|
|
|
|
|
|
|
|
TEST_CASE("A64: Bitmasks", "[a64]") {
|
|
|
|
TestEnv env;
|
|
|
|
Dynarmic::A64::Jit jit{Dynarmic::A64::UserConfig{&env}};
|
|
|
|
|
|
|
|
env.code_mem[0] = 0x3200c3e0; // ORR W0, WZR, #0x01010101
|
|
|
|
env.code_mem[1] = 0x320c8fe1; // ORR W1, WZR, #0x00F000F0
|
|
|
|
env.code_mem[2] = 0x320003e2; // ORR W2, WZR, #1
|
|
|
|
env.code_mem[3] = 0x14000000; // B .
|
|
|
|
|
|
|
|
jit.SetPC(0);
|
|
|
|
|
|
|
|
env.ticks_left = 4;
|
|
|
|
jit.Run();
|
|
|
|
|
|
|
|
REQUIRE(jit.GetRegister(0) == 0x01010101);
|
|
|
|
REQUIRE(jit.GetRegister(1) == 0x00F000F0);
|
|
|
|
REQUIRE(jit.GetRegister(2) == 1);
|
|
|
|
REQUIRE(jit.GetPC() == 12);
|
|
|
|
}
|