testenv.h 6.7 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202
  1. /* This file is part of the dynarmic project.
  2. * Copyright (c) 2018 MerryMage
  3. * SPDX-License-Identifier: 0BSD
  4. */
  5. #pragma once
  6. #include <array>
  7. #include <cstring>
  8. #include <map>
  9. #include <string>
  10. #include <vector>
  11. #include <mcl/assert.hpp>
  12. #include <mcl/stdint.hpp>
  13. #include "dynarmic/interface/A32/a32.h"
  14. template<typename InstructionType_, u32 infinite_loop_u32>
  15. class A32TestEnv : public Dynarmic::A32::UserCallbacks {
  16. public:
  17. using InstructionType = InstructionType_;
  18. using RegisterArray = std::array<u32, 16>;
  19. using ExtRegsArray = std::array<u32, 64>;
  20. #ifdef _MSC_VER
  21. # pragma warning(push)
  22. # pragma warning(disable : 4309) // C4309: 'static_cast': truncation of constant value
  23. #endif
  24. static constexpr InstructionType infinite_loop = static_cast<InstructionType>(infinite_loop_u32);
  25. #ifdef _MSC_VER
  26. # pragma warning(pop)
  27. #endif
  28. u64 ticks_left = 0;
  29. bool code_mem_modified_by_guest = false;
  30. std::vector<InstructionType> code_mem;
  31. std::map<u32, u8> modified_memory;
  32. std::vector<std::string> interrupts;
  33. void PadCodeMem() {
  34. do {
  35. code_mem.push_back(infinite_loop);
  36. } while (code_mem.size() % 2 != 0);
  37. }
  38. bool IsInCodeMem(u32 vaddr) const {
  39. return vaddr < sizeof(InstructionType) * code_mem.size();
  40. }
  41. std::optional<std::uint32_t> MemoryReadCode(u32 vaddr) override {
  42. if (IsInCodeMem(vaddr)) {
  43. u32 value;
  44. std::memcpy(&value, &code_mem[vaddr / sizeof(InstructionType)], sizeof(u32));
  45. return value;
  46. }
  47. return infinite_loop_u32; // B .
  48. }
  49. std::uint8_t MemoryRead8(u32 vaddr) override {
  50. if (IsInCodeMem(vaddr)) {
  51. return reinterpret_cast<u8*>(code_mem.data())[vaddr];
  52. }
  53. if (auto iter = modified_memory.find(vaddr); iter != modified_memory.end()) {
  54. return iter->second;
  55. }
  56. return static_cast<u8>(vaddr);
  57. }
  58. std::uint16_t MemoryRead16(u32 vaddr) override {
  59. return u16(MemoryRead8(vaddr)) | u16(MemoryRead8(vaddr + 1)) << 8;
  60. }
  61. std::uint32_t MemoryRead32(u32 vaddr) override {
  62. return u32(MemoryRead16(vaddr)) | u32(MemoryRead16(vaddr + 2)) << 16;
  63. }
  64. std::uint64_t MemoryRead64(u32 vaddr) override {
  65. return u64(MemoryRead32(vaddr)) | u64(MemoryRead32(vaddr + 4)) << 32;
  66. }
  67. void MemoryWrite8(u32 vaddr, std::uint8_t value) override {
  68. if (vaddr < code_mem.size() * sizeof(u32)) {
  69. code_mem_modified_by_guest = true;
  70. }
  71. modified_memory[vaddr] = value;
  72. }
  73. void MemoryWrite16(u32 vaddr, std::uint16_t value) override {
  74. MemoryWrite8(vaddr, static_cast<u8>(value));
  75. MemoryWrite8(vaddr + 1, static_cast<u8>(value >> 8));
  76. }
  77. void MemoryWrite32(u32 vaddr, std::uint32_t value) override {
  78. MemoryWrite16(vaddr, static_cast<u16>(value));
  79. MemoryWrite16(vaddr + 2, static_cast<u16>(value >> 16));
  80. }
  81. void MemoryWrite64(u32 vaddr, std::uint64_t value) override {
  82. MemoryWrite32(vaddr, static_cast<u32>(value));
  83. MemoryWrite32(vaddr + 4, static_cast<u32>(value >> 32));
  84. }
  85. void InterpreterFallback(u32 pc, size_t num_instructions) override { ASSERT_MSG(false, "InterpreterFallback({:08x}, {}) code = {:08x}", pc, num_instructions, *MemoryReadCode(pc)); }
  86. void CallSVC(std::uint32_t swi) override { ASSERT_MSG(false, "CallSVC({})", swi); }
  87. void ExceptionRaised(u32 pc, Dynarmic::A32::Exception /*exception*/) override { ASSERT_MSG(false, "ExceptionRaised({:08x}) code = {:08x}", pc, *MemoryReadCode(pc)); }
  88. void AddTicks(std::uint64_t ticks) override {
  89. if (ticks > ticks_left) {
  90. ticks_left = 0;
  91. return;
  92. }
  93. ticks_left -= ticks;
  94. }
  95. std::uint64_t GetTicksRemaining() override {
  96. return ticks_left;
  97. }
  98. };
  99. using ArmTestEnv = A32TestEnv<u32, 0xEAFFFFFE>;
  100. using ThumbTestEnv = A32TestEnv<u16, 0xE7FEE7FE>;
  101. class A32FastmemTestEnv final : public Dynarmic::A32::UserCallbacks {
  102. public:
  103. u64 ticks_left = 0;
  104. char* backing_memory = nullptr;
  105. explicit A32FastmemTestEnv(char* addr)
  106. : backing_memory(addr) {}
  107. template<typename T>
  108. T read(std::uint32_t vaddr) {
  109. T value;
  110. memcpy(&value, backing_memory + vaddr, sizeof(T));
  111. return value;
  112. }
  113. template<typename T>
  114. void write(std::uint32_t vaddr, const T& value) {
  115. memcpy(backing_memory + vaddr, &value, sizeof(T));
  116. }
  117. std::optional<std::uint32_t> MemoryReadCode(std::uint32_t vaddr) override {
  118. return read<std::uint32_t>(vaddr);
  119. }
  120. std::uint8_t MemoryRead8(std::uint32_t vaddr) override {
  121. return read<std::uint8_t>(vaddr);
  122. }
  123. std::uint16_t MemoryRead16(std::uint32_t vaddr) override {
  124. return read<std::uint16_t>(vaddr);
  125. }
  126. std::uint32_t MemoryRead32(std::uint32_t vaddr) override {
  127. return read<std::uint32_t>(vaddr);
  128. }
  129. std::uint64_t MemoryRead64(std::uint32_t vaddr) override {
  130. return read<std::uint64_t>(vaddr);
  131. }
  132. void MemoryWrite8(std::uint32_t vaddr, std::uint8_t value) override {
  133. write(vaddr, value);
  134. }
  135. void MemoryWrite16(std::uint32_t vaddr, std::uint16_t value) override {
  136. write(vaddr, value);
  137. }
  138. void MemoryWrite32(std::uint32_t vaddr, std::uint32_t value) override {
  139. write(vaddr, value);
  140. }
  141. void MemoryWrite64(std::uint32_t vaddr, std::uint64_t value) override {
  142. write(vaddr, value);
  143. }
  144. bool MemoryWriteExclusive8(std::uint32_t vaddr, std::uint8_t value, [[maybe_unused]] std::uint8_t expected) override {
  145. MemoryWrite8(vaddr, value);
  146. return true;
  147. }
  148. bool MemoryWriteExclusive16(std::uint32_t vaddr, std::uint16_t value, [[maybe_unused]] std::uint16_t expected) override {
  149. MemoryWrite16(vaddr, value);
  150. return true;
  151. }
  152. bool MemoryWriteExclusive32(std::uint32_t vaddr, std::uint32_t value, [[maybe_unused]] std::uint32_t expected) override {
  153. MemoryWrite32(vaddr, value);
  154. return true;
  155. }
  156. bool MemoryWriteExclusive64(std::uint32_t vaddr, std::uint64_t value, [[maybe_unused]] std::uint64_t expected) override {
  157. MemoryWrite64(vaddr, value);
  158. return true;
  159. }
  160. void InterpreterFallback(std::uint32_t pc, size_t num_instructions) override { ASSERT_MSG(false, "InterpreterFallback({:016x}, {})", pc, num_instructions); }
  161. void CallSVC(std::uint32_t swi) override { ASSERT_MSG(false, "CallSVC({})", swi); }
  162. void ExceptionRaised(std::uint32_t pc, Dynarmic::A32::Exception) override { ASSERT_MSG(false, "ExceptionRaised({:016x})", pc); }
  163. void AddTicks(std::uint64_t ticks) override {
  164. if (ticks > ticks_left) {
  165. ticks_left = 0;
  166. return;
  167. }
  168. ticks_left -= ticks;
  169. }
  170. std::uint64_t GetTicksRemaining() override {
  171. return ticks_left;
  172. }
  173. };