2018-01-27 23:36:55 +01:00
|
|
|
/* This file is part of the dynarmic project.
|
|
|
|
* Copyright (c) 2018 MerryMage
|
2020-04-23 16:25:11 +02:00
|
|
|
* SPDX-License-Identifier: 0BSD
|
2018-01-27 23:36:55 +01:00
|
|
|
*/
|
|
|
|
|
|
|
|
#pragma once
|
|
|
|
|
|
|
|
#include <array>
|
2018-01-28 00:42:30 +01:00
|
|
|
#include <cstring>
|
2018-01-27 23:36:55 +01:00
|
|
|
#include <map>
|
2018-08-15 14:41:29 +02:00
|
|
|
#include <string>
|
|
|
|
#include <vector>
|
2018-01-27 23:36:55 +01:00
|
|
|
|
2021-05-19 18:28:35 +02:00
|
|
|
#include "dynarmic/common/assert.h"
|
|
|
|
#include "dynarmic/common/common_types.h"
|
|
|
|
#include "dynarmic/interface/A32/a32.h"
|
2018-01-27 23:36:55 +01:00
|
|
|
|
2021-05-22 15:51:20 +02:00
|
|
|
template<typename InstructionType_, u32 infinite_loop_u32>
|
2018-01-27 23:36:55 +01:00
|
|
|
class A32TestEnv final : public Dynarmic::A32::UserCallbacks {
|
|
|
|
public:
|
2019-04-13 11:56:55 +02:00
|
|
|
using InstructionType = InstructionType_;
|
2018-07-13 20:19:22 +02:00
|
|
|
using RegisterArray = std::array<u32, 16>;
|
|
|
|
using ExtRegsArray = std::array<u32, 64>;
|
|
|
|
|
2021-02-06 23:31:49 +01:00
|
|
|
#ifdef _MSC_VER
|
2021-05-22 15:51:20 +02:00
|
|
|
# pragma warning(push)
|
|
|
|
# pragma warning(disable : 4309) // C4309: 'static_cast': truncation of constant value
|
2021-02-06 23:31:49 +01:00
|
|
|
#endif
|
2021-02-06 22:25:08 +01:00
|
|
|
static constexpr InstructionType infinite_loop = static_cast<InstructionType>(infinite_loop_u32);
|
2021-02-06 23:31:49 +01:00
|
|
|
#ifdef _MSC_VER
|
2021-05-22 15:51:20 +02:00
|
|
|
# pragma warning(pop)
|
2021-02-06 23:31:49 +01:00
|
|
|
#endif
|
2021-02-06 22:25:08 +01:00
|
|
|
|
2018-01-27 23:36:55 +01:00
|
|
|
u64 ticks_left = 0;
|
|
|
|
bool code_mem_modified_by_guest = false;
|
2019-03-02 22:18:42 +01:00
|
|
|
std::vector<InstructionType> code_mem;
|
2018-01-27 23:36:55 +01:00
|
|
|
std::map<u32, u8> modified_memory;
|
2018-08-11 12:39:53 +02:00
|
|
|
std::vector<std::string> interrupts;
|
2018-01-27 23:36:55 +01:00
|
|
|
|
2021-02-06 22:25:08 +01:00
|
|
|
void PadCodeMem() {
|
|
|
|
do {
|
|
|
|
code_mem.push_back(infinite_loop);
|
|
|
|
} while (code_mem.size() % 2 != 0);
|
|
|
|
}
|
|
|
|
|
2021-02-06 23:15:02 +01:00
|
|
|
bool IsInCodeMem(u32 vaddr) const {
|
|
|
|
return vaddr < sizeof(InstructionType) * code_mem.size();
|
|
|
|
}
|
|
|
|
|
2018-01-27 23:36:55 +01:00
|
|
|
std::uint32_t MemoryReadCode(u32 vaddr) override {
|
2021-02-06 23:15:02 +01:00
|
|
|
if (IsInCodeMem(vaddr)) {
|
2018-01-27 23:36:55 +01:00
|
|
|
u32 value;
|
2021-02-06 23:15:02 +01:00
|
|
|
std::memcpy(&value, &code_mem[vaddr / sizeof(InstructionType)], sizeof(u32));
|
2018-01-27 23:36:55 +01:00
|
|
|
return value;
|
|
|
|
}
|
2021-05-22 15:51:20 +02:00
|
|
|
return infinite_loop_u32; // B .
|
2018-01-27 23:36:55 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
std::uint8_t MemoryRead8(u32 vaddr) override {
|
2021-02-06 23:15:02 +01:00
|
|
|
if (IsInCodeMem(vaddr)) {
|
2018-01-27 23:36:55 +01:00
|
|
|
return reinterpret_cast<u8*>(code_mem.data())[vaddr];
|
|
|
|
}
|
|
|
|
if (auto iter = modified_memory.find(vaddr); iter != modified_memory.end()) {
|
|
|
|
return iter->second;
|
|
|
|
}
|
|
|
|
return static_cast<u8>(vaddr);
|
|
|
|
}
|
|
|
|
std::uint16_t MemoryRead16(u32 vaddr) override {
|
|
|
|
return u16(MemoryRead8(vaddr)) | u16(MemoryRead8(vaddr + 1)) << 8;
|
|
|
|
}
|
|
|
|
std::uint32_t MemoryRead32(u32 vaddr) override {
|
|
|
|
return u32(MemoryRead16(vaddr)) | u32(MemoryRead16(vaddr + 2)) << 16;
|
|
|
|
}
|
|
|
|
std::uint64_t MemoryRead64(u32 vaddr) override {
|
|
|
|
return u64(MemoryRead32(vaddr)) | u64(MemoryRead32(vaddr + 4)) << 32;
|
|
|
|
}
|
|
|
|
|
|
|
|
void MemoryWrite8(u32 vaddr, std::uint8_t value) override {
|
|
|
|
if (vaddr < code_mem.size() * sizeof(u32)) {
|
|
|
|
code_mem_modified_by_guest = true;
|
|
|
|
}
|
|
|
|
modified_memory[vaddr] = value;
|
|
|
|
}
|
|
|
|
void MemoryWrite16(u32 vaddr, std::uint16_t value) override {
|
|
|
|
MemoryWrite8(vaddr, static_cast<u8>(value));
|
|
|
|
MemoryWrite8(vaddr + 1, static_cast<u8>(value >> 8));
|
|
|
|
}
|
|
|
|
void MemoryWrite32(u32 vaddr, std::uint32_t value) override {
|
|
|
|
MemoryWrite16(vaddr, static_cast<u16>(value));
|
|
|
|
MemoryWrite16(vaddr + 2, static_cast<u16>(value >> 16));
|
|
|
|
}
|
|
|
|
void MemoryWrite64(u32 vaddr, std::uint64_t value) override {
|
|
|
|
MemoryWrite32(vaddr, static_cast<u32>(value));
|
|
|
|
MemoryWrite32(vaddr + 4, static_cast<u32>(value >> 32));
|
|
|
|
}
|
|
|
|
|
2019-07-27 20:56:18 +02:00
|
|
|
void InterpreterFallback(u32 pc, size_t num_instructions) override { ASSERT_MSG(false, "InterpreterFallback({:08x}, {}) code = {:08x}", pc, num_instructions, MemoryReadCode(pc)); }
|
2018-01-27 23:36:55 +01:00
|
|
|
|
2018-01-28 00:42:30 +01:00
|
|
|
void CallSVC(std::uint32_t swi) override { ASSERT_MSG(false, "CallSVC({})", swi); }
|
2018-01-27 23:36:55 +01:00
|
|
|
|
2021-02-06 22:25:08 +01:00
|
|
|
void ExceptionRaised(u32 pc, Dynarmic::A32::Exception /*exception*/) override { ASSERT_MSG(false, "ExceptionRaised({:08x}) code = {:08x}", pc, MemoryReadCode(pc)); }
|
2018-01-27 23:36:55 +01: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;
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
using ArmTestEnv = A32TestEnv<u32, 0xEAFFFFFE>;
|
|
|
|
using ThumbTestEnv = A32TestEnv<u16, 0xE7FEE7FE>;
|