2018-01-13 18:03:34 +00:00
|
|
|
/* This file is part of the dynarmic project.
|
|
|
|
* Copyright (c) 2018 MerryMage
|
2020-04-23 15:25:11 +01:00
|
|
|
* SPDX-License-Identifier: 0BSD
|
2018-01-13 18:03:34 +00:00
|
|
|
*/
|
|
|
|
|
|
|
|
#pragma once
|
|
|
|
|
|
|
|
#include <array>
|
|
|
|
#include <map>
|
|
|
|
|
2021-05-19 17:28:35 +01:00
|
|
|
#include "dynarmic/common/assert.h"
|
|
|
|
#include "dynarmic/common/common_types.h"
|
|
|
|
#include "dynarmic/interface/A64/a64.h"
|
2018-01-13 18:03:34 +00:00
|
|
|
|
2018-01-24 15:52:49 +00:00
|
|
|
using Vector = Dynarmic::A64::Vector;
|
|
|
|
|
2020-11-12 16:15:33 +00:00
|
|
|
class A64TestEnv : public Dynarmic::A64::UserCallbacks {
|
2018-01-13 18:03:34 +00:00
|
|
|
public:
|
|
|
|
u64 ticks_left = 0;
|
2018-08-01 15:25:40 +01:00
|
|
|
|
2018-01-13 18:03:34 +00:00
|
|
|
bool code_mem_modified_by_guest = false;
|
2018-08-01 15:25:40 +01:00
|
|
|
u64 code_mem_start_address = 0;
|
|
|
|
std::vector<u32> code_mem;
|
|
|
|
|
2018-01-13 18:03:34 +00:00
|
|
|
std::map<u64, u8> modified_memory;
|
2018-07-15 11:58:53 +01:00
|
|
|
std::vector<std::string> interrupts;
|
2018-01-13 18:03:34 +00:00
|
|
|
|
2018-08-01 15:25:40 +01:00
|
|
|
bool IsInCodeMem(u64 vaddr) const {
|
|
|
|
return vaddr >= code_mem_start_address && vaddr < code_mem_start_address + code_mem.size() * 4;
|
|
|
|
}
|
|
|
|
|
2018-01-13 18:03:34 +00:00
|
|
|
std::uint32_t MemoryReadCode(u64 vaddr) override {
|
2018-08-01 15:25:40 +01:00
|
|
|
if (!IsInCodeMem(vaddr)) {
|
2021-05-22 14:51:20 +01:00
|
|
|
return 0x14000000; // B .
|
2018-01-13 18:03:34 +00:00
|
|
|
}
|
2018-08-01 15:25:40 +01:00
|
|
|
|
|
|
|
const size_t index = (vaddr - code_mem_start_address) / 4;
|
|
|
|
return code_mem[index];
|
2018-01-13 18:03:34 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
std::uint8_t MemoryRead8(u64 vaddr) override {
|
2018-08-01 15:25:40 +01:00
|
|
|
if (IsInCodeMem(vaddr)) {
|
|
|
|
return reinterpret_cast<u8*>(code_mem.data())[vaddr - code_mem_start_address];
|
2018-01-13 18:03:34 +00:00
|
|
|
}
|
|
|
|
if (auto iter = modified_memory.find(vaddr); iter != modified_memory.end()) {
|
|
|
|
return iter->second;
|
|
|
|
}
|
|
|
|
return static_cast<u8>(vaddr);
|
|
|
|
}
|
|
|
|
std::uint16_t MemoryRead16(u64 vaddr) override {
|
|
|
|
return u16(MemoryRead8(vaddr)) | u16(MemoryRead8(vaddr + 1)) << 8;
|
|
|
|
}
|
|
|
|
std::uint32_t MemoryRead32(u64 vaddr) override {
|
|
|
|
return u32(MemoryRead16(vaddr)) | u32(MemoryRead16(vaddr + 2)) << 16;
|
|
|
|
}
|
|
|
|
std::uint64_t MemoryRead64(u64 vaddr) override {
|
|
|
|
return u64(MemoryRead32(vaddr)) | u64(MemoryRead32(vaddr + 4)) << 32;
|
|
|
|
}
|
2018-01-24 15:55:59 +00:00
|
|
|
Vector MemoryRead128(u64 vaddr) override {
|
|
|
|
return {MemoryRead64(vaddr), MemoryRead64(vaddr + 8)};
|
|
|
|
}
|
2018-01-13 18:03:34 +00:00
|
|
|
|
|
|
|
void MemoryWrite8(u64 vaddr, std::uint8_t value) override {
|
2018-08-01 15:25:40 +01:00
|
|
|
if (IsInCodeMem(vaddr)) {
|
2018-01-13 18:03:34 +00:00
|
|
|
code_mem_modified_by_guest = true;
|
|
|
|
}
|
|
|
|
modified_memory[vaddr] = value;
|
|
|
|
}
|
|
|
|
void MemoryWrite16(u64 vaddr, std::uint16_t value) override {
|
|
|
|
MemoryWrite8(vaddr, static_cast<u8>(value));
|
|
|
|
MemoryWrite8(vaddr + 1, static_cast<u8>(value >> 8));
|
|
|
|
}
|
|
|
|
void MemoryWrite32(u64 vaddr, std::uint32_t value) override {
|
|
|
|
MemoryWrite16(vaddr, static_cast<u16>(value));
|
|
|
|
MemoryWrite16(vaddr + 2, static_cast<u16>(value >> 16));
|
|
|
|
}
|
|
|
|
void MemoryWrite64(u64 vaddr, std::uint64_t value) override {
|
|
|
|
MemoryWrite32(vaddr, static_cast<u32>(value));
|
|
|
|
MemoryWrite32(vaddr + 4, static_cast<u32>(value >> 32));
|
|
|
|
}
|
2018-01-24 15:55:59 +00:00
|
|
|
void MemoryWrite128(u64 vaddr, Vector value) override {
|
|
|
|
MemoryWrite64(vaddr, value[0]);
|
2018-01-25 23:56:57 +00:00
|
|
|
MemoryWrite64(vaddr + 8, value[1]);
|
2018-01-24 15:55:59 +00:00
|
|
|
}
|
2018-01-13 18:03:34 +00:00
|
|
|
|
2020-03-10 22:07:44 +00:00
|
|
|
bool MemoryWriteExclusive8(u64 vaddr, std::uint8_t value, [[maybe_unused]] std::uint8_t expected) override {
|
|
|
|
MemoryWrite8(vaddr, value);
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
bool MemoryWriteExclusive16(u64 vaddr, std::uint16_t value, [[maybe_unused]] std::uint16_t expected) override {
|
|
|
|
MemoryWrite16(vaddr, value);
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
bool MemoryWriteExclusive32(u64 vaddr, std::uint32_t value, [[maybe_unused]] std::uint32_t expected) override {
|
|
|
|
MemoryWrite32(vaddr, value);
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
bool MemoryWriteExclusive64(u64 vaddr, std::uint64_t value, [[maybe_unused]] std::uint64_t expected) override {
|
|
|
|
MemoryWrite64(vaddr, value);
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
bool MemoryWriteExclusive128(u64 vaddr, Vector value, [[maybe_unused]] Vector expected) override {
|
|
|
|
MemoryWrite128(vaddr, value);
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2018-01-27 23:42:30 +00:00
|
|
|
void InterpreterFallback(u64 pc, size_t num_instructions) override { ASSERT_MSG(false, "InterpreterFallback({:016x}, {})", pc, num_instructions); }
|
2018-01-13 18:03:34 +00:00
|
|
|
|
2018-01-27 23:42:30 +00:00
|
|
|
void CallSVC(std::uint32_t swi) override { ASSERT_MSG(false, "CallSVC({})", swi); }
|
2018-01-13 18:03:34 +00:00
|
|
|
|
2018-01-27 23:42:30 +00:00
|
|
|
void ExceptionRaised(u64 pc, Dynarmic::A64::Exception /*exception*/) override { ASSERT_MSG(false, "ExceptionRaised({:016x})", pc); }
|
2018-01-13 18:03:34 +00:00
|
|
|
|
|
|
|
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-02-20 16:54:10 +00:00
|
|
|
std::uint64_t GetCNTPCT() override {
|
2018-07-22 16:16:26 +01:00
|
|
|
return 0x10000000000 - ticks_left;
|
2018-02-20 16:54:10 +00:00
|
|
|
}
|
2018-01-13 18:03:34 +00:00
|
|
|
};
|
2020-06-21 18:29:55 +01:00
|
|
|
|
|
|
|
class A64FastmemTestEnv final : public Dynarmic::A64::UserCallbacks {
|
|
|
|
public:
|
|
|
|
u64 ticks_left = 0;
|
|
|
|
char* backing_memory = nullptr;
|
|
|
|
|
|
|
|
explicit A64FastmemTestEnv(char* addr) : backing_memory(addr) {}
|
|
|
|
|
|
|
|
template<typename T>
|
|
|
|
T read(u64 vaddr) {
|
|
|
|
T value;
|
|
|
|
memcpy(&value, backing_memory + vaddr, sizeof(T));
|
|
|
|
return value;
|
|
|
|
}
|
|
|
|
template<typename T>
|
|
|
|
void write(u64 vaddr, const T& value) {
|
|
|
|
memcpy(backing_memory + vaddr, &value, sizeof(T));
|
|
|
|
}
|
|
|
|
|
|
|
|
std::uint32_t MemoryReadCode(u64 vaddr) override {
|
|
|
|
return read<std::uint32_t>(vaddr);
|
|
|
|
}
|
|
|
|
|
|
|
|
std::uint8_t MemoryRead8(u64 vaddr) override {
|
|
|
|
return read<std::uint8_t>(vaddr);
|
|
|
|
}
|
|
|
|
std::uint16_t MemoryRead16(u64 vaddr) override {
|
|
|
|
return read<std::uint16_t>(vaddr);
|
|
|
|
}
|
|
|
|
std::uint32_t MemoryRead32(u64 vaddr) override {
|
|
|
|
return read<std::uint32_t>(vaddr);
|
|
|
|
}
|
|
|
|
std::uint64_t MemoryRead64(u64 vaddr) override {
|
|
|
|
return read<std::uint64_t>(vaddr);
|
|
|
|
}
|
|
|
|
Vector MemoryRead128(u64 vaddr) override {
|
|
|
|
return read<Vector>(vaddr);
|
|
|
|
}
|
|
|
|
|
|
|
|
void MemoryWrite8(u64 vaddr, std::uint8_t value) override {
|
|
|
|
write(vaddr, value);
|
|
|
|
}
|
|
|
|
void MemoryWrite16(u64 vaddr, std::uint16_t value) override {
|
|
|
|
write(vaddr, value);
|
|
|
|
}
|
|
|
|
void MemoryWrite32(u64 vaddr, std::uint32_t value) override {
|
|
|
|
write(vaddr, value);
|
|
|
|
}
|
|
|
|
void MemoryWrite64(u64 vaddr, std::uint64_t value) override {
|
|
|
|
write(vaddr, value);
|
|
|
|
}
|
|
|
|
void MemoryWrite128(u64 vaddr, Vector value) override {
|
|
|
|
write(vaddr, value);
|
|
|
|
}
|
|
|
|
|
|
|
|
bool MemoryWriteExclusive8(u64 vaddr, std::uint8_t value, [[maybe_unused]] std::uint8_t expected) override {
|
|
|
|
MemoryWrite8(vaddr, value);
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
bool MemoryWriteExclusive16(u64 vaddr, std::uint16_t value, [[maybe_unused]] std::uint16_t expected) override {
|
|
|
|
MemoryWrite16(vaddr, value);
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
bool MemoryWriteExclusive32(u64 vaddr, std::uint32_t value, [[maybe_unused]] std::uint32_t expected) override {
|
|
|
|
MemoryWrite32(vaddr, value);
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
bool MemoryWriteExclusive64(u64 vaddr, std::uint64_t value, [[maybe_unused]] std::uint64_t expected) override {
|
|
|
|
MemoryWrite64(vaddr, value);
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
bool MemoryWriteExclusive128(u64 vaddr, Vector value, [[maybe_unused]] Vector expected) override {
|
|
|
|
MemoryWrite128(vaddr, value);
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
void InterpreterFallback(u64 pc, size_t num_instructions) override { ASSERT_MSG(false, "InterpreterFallback({:016x}, {})", pc, num_instructions); }
|
|
|
|
|
|
|
|
void CallSVC(std::uint32_t swi) override { ASSERT_MSG(false, "CallSVC({})", swi); }
|
|
|
|
|
|
|
|
void ExceptionRaised(u64 pc, Dynarmic::A64::Exception) override { ASSERT_MSG(false, "ExceptionRaised({:016x})", pc); }
|
|
|
|
|
|
|
|
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;
|
|
|
|
}
|
|
|
|
std::uint64_t GetCNTPCT() override {
|
|
|
|
return 0x10000000000 - ticks_left;
|
|
|
|
}
|
|
|
|
};
|