testenv.h 7.5 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225
  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 <map>
  8. #include <mcl/assert.hpp>
  9. #include <mcl/stdint.hpp>
  10. #include "dynarmic/interface/A64/a64.h"
  11. using Vector = Dynarmic::A64::Vector;
  12. class A64TestEnv : public Dynarmic::A64::UserCallbacks {
  13. public:
  14. u64 ticks_left = 0;
  15. bool code_mem_modified_by_guest = false;
  16. u64 code_mem_start_address = 0;
  17. std::vector<u32> code_mem;
  18. std::map<u64, u8> modified_memory;
  19. std::vector<std::string> interrupts;
  20. bool IsInCodeMem(u64 vaddr) const {
  21. return vaddr >= code_mem_start_address && vaddr < code_mem_start_address + code_mem.size() * 4;
  22. }
  23. std::optional<std::uint32_t> MemoryReadCode(u64 vaddr) override {
  24. if (!IsInCodeMem(vaddr)) {
  25. return 0x14000000; // B .
  26. }
  27. const size_t index = (vaddr - code_mem_start_address) / 4;
  28. return code_mem[index];
  29. }
  30. std::uint8_t MemoryRead8(u64 vaddr) override {
  31. if (IsInCodeMem(vaddr)) {
  32. return reinterpret_cast<u8*>(code_mem.data())[vaddr - code_mem_start_address];
  33. }
  34. if (auto iter = modified_memory.find(vaddr); iter != modified_memory.end()) {
  35. return iter->second;
  36. }
  37. return static_cast<u8>(vaddr);
  38. }
  39. std::uint16_t MemoryRead16(u64 vaddr) override {
  40. return u16(MemoryRead8(vaddr)) | u16(MemoryRead8(vaddr + 1)) << 8;
  41. }
  42. std::uint32_t MemoryRead32(u64 vaddr) override {
  43. return u32(MemoryRead16(vaddr)) | u32(MemoryRead16(vaddr + 2)) << 16;
  44. }
  45. std::uint64_t MemoryRead64(u64 vaddr) override {
  46. return u64(MemoryRead32(vaddr)) | u64(MemoryRead32(vaddr + 4)) << 32;
  47. }
  48. Vector MemoryRead128(u64 vaddr) override {
  49. return {MemoryRead64(vaddr), MemoryRead64(vaddr + 8)};
  50. }
  51. void MemoryWrite8(u64 vaddr, std::uint8_t value) override {
  52. if (IsInCodeMem(vaddr)) {
  53. code_mem_modified_by_guest = true;
  54. }
  55. modified_memory[vaddr] = value;
  56. }
  57. void MemoryWrite16(u64 vaddr, std::uint16_t value) override {
  58. MemoryWrite8(vaddr, static_cast<u8>(value));
  59. MemoryWrite8(vaddr + 1, static_cast<u8>(value >> 8));
  60. }
  61. void MemoryWrite32(u64 vaddr, std::uint32_t value) override {
  62. MemoryWrite16(vaddr, static_cast<u16>(value));
  63. MemoryWrite16(vaddr + 2, static_cast<u16>(value >> 16));
  64. }
  65. void MemoryWrite64(u64 vaddr, std::uint64_t value) override {
  66. MemoryWrite32(vaddr, static_cast<u32>(value));
  67. MemoryWrite32(vaddr + 4, static_cast<u32>(value >> 32));
  68. }
  69. void MemoryWrite128(u64 vaddr, Vector value) override {
  70. MemoryWrite64(vaddr, value[0]);
  71. MemoryWrite64(vaddr + 8, value[1]);
  72. }
  73. bool MemoryWriteExclusive8(u64 vaddr, std::uint8_t value, [[maybe_unused]] std::uint8_t expected) override {
  74. MemoryWrite8(vaddr, value);
  75. return true;
  76. }
  77. bool MemoryWriteExclusive16(u64 vaddr, std::uint16_t value, [[maybe_unused]] std::uint16_t expected) override {
  78. MemoryWrite16(vaddr, value);
  79. return true;
  80. }
  81. bool MemoryWriteExclusive32(u64 vaddr, std::uint32_t value, [[maybe_unused]] std::uint32_t expected) override {
  82. MemoryWrite32(vaddr, value);
  83. return true;
  84. }
  85. bool MemoryWriteExclusive64(u64 vaddr, std::uint64_t value, [[maybe_unused]] std::uint64_t expected) override {
  86. MemoryWrite64(vaddr, value);
  87. return true;
  88. }
  89. bool MemoryWriteExclusive128(u64 vaddr, Vector value, [[maybe_unused]] Vector expected) override {
  90. MemoryWrite128(vaddr, value);
  91. return true;
  92. }
  93. void InterpreterFallback(u64 pc, size_t num_instructions) override { ASSERT_MSG(false, "InterpreterFallback({:016x}, {})", pc, num_instructions); }
  94. void CallSVC(std::uint32_t swi) override { ASSERT_MSG(false, "CallSVC({})", swi); }
  95. void ExceptionRaised(u64 pc, Dynarmic::A64::Exception /*exception*/) override { ASSERT_MSG(false, "ExceptionRaised({:016x})", pc); }
  96. void AddTicks(std::uint64_t ticks) override {
  97. if (ticks > ticks_left) {
  98. ticks_left = 0;
  99. return;
  100. }
  101. ticks_left -= ticks;
  102. }
  103. std::uint64_t GetTicksRemaining() override {
  104. return ticks_left;
  105. }
  106. std::uint64_t GetCNTPCT() override {
  107. return 0x10000000000 - ticks_left;
  108. }
  109. };
  110. class A64FastmemTestEnv final : public Dynarmic::A64::UserCallbacks {
  111. public:
  112. u64 ticks_left = 0;
  113. char* backing_memory = nullptr;
  114. explicit A64FastmemTestEnv(char* addr)
  115. : backing_memory(addr) {}
  116. template<typename T>
  117. T read(u64 vaddr) {
  118. T value;
  119. memcpy(&value, backing_memory + vaddr, sizeof(T));
  120. return value;
  121. }
  122. template<typename T>
  123. void write(u64 vaddr, const T& value) {
  124. memcpy(backing_memory + vaddr, &value, sizeof(T));
  125. }
  126. std::optional<std::uint32_t> MemoryReadCode(u64 vaddr) override {
  127. return read<std::uint32_t>(vaddr);
  128. }
  129. std::uint8_t MemoryRead8(u64 vaddr) override {
  130. return read<std::uint8_t>(vaddr);
  131. }
  132. std::uint16_t MemoryRead16(u64 vaddr) override {
  133. return read<std::uint16_t>(vaddr);
  134. }
  135. std::uint32_t MemoryRead32(u64 vaddr) override {
  136. return read<std::uint32_t>(vaddr);
  137. }
  138. std::uint64_t MemoryRead64(u64 vaddr) override {
  139. return read<std::uint64_t>(vaddr);
  140. }
  141. Vector MemoryRead128(u64 vaddr) override {
  142. return read<Vector>(vaddr);
  143. }
  144. void MemoryWrite8(u64 vaddr, std::uint8_t value) override {
  145. write(vaddr, value);
  146. }
  147. void MemoryWrite16(u64 vaddr, std::uint16_t value) override {
  148. write(vaddr, value);
  149. }
  150. void MemoryWrite32(u64 vaddr, std::uint32_t value) override {
  151. write(vaddr, value);
  152. }
  153. void MemoryWrite64(u64 vaddr, std::uint64_t value) override {
  154. write(vaddr, value);
  155. }
  156. void MemoryWrite128(u64 vaddr, Vector value) override {
  157. write(vaddr, value);
  158. }
  159. bool MemoryWriteExclusive8(u64 vaddr, std::uint8_t value, [[maybe_unused]] std::uint8_t expected) override {
  160. MemoryWrite8(vaddr, value);
  161. return true;
  162. }
  163. bool MemoryWriteExclusive16(u64 vaddr, std::uint16_t value, [[maybe_unused]] std::uint16_t expected) override {
  164. MemoryWrite16(vaddr, value);
  165. return true;
  166. }
  167. bool MemoryWriteExclusive32(u64 vaddr, std::uint32_t value, [[maybe_unused]] std::uint32_t expected) override {
  168. MemoryWrite32(vaddr, value);
  169. return true;
  170. }
  171. bool MemoryWriteExclusive64(u64 vaddr, std::uint64_t value, [[maybe_unused]] std::uint64_t expected) override {
  172. MemoryWrite64(vaddr, value);
  173. return true;
  174. }
  175. bool MemoryWriteExclusive128(u64 vaddr, Vector value, [[maybe_unused]] Vector expected) override {
  176. MemoryWrite128(vaddr, value);
  177. return true;
  178. }
  179. void InterpreterFallback(u64 pc, size_t num_instructions) override { ASSERT_MSG(false, "InterpreterFallback({:016x}, {})", pc, num_instructions); }
  180. void CallSVC(std::uint32_t swi) override { ASSERT_MSG(false, "CallSVC({})", swi); }
  181. void ExceptionRaised(u64 pc, Dynarmic::A64::Exception) override { ASSERT_MSG(false, "ExceptionRaised({:016x})", pc); }
  182. void AddTicks(std::uint64_t ticks) override {
  183. if (ticks > ticks_left) {
  184. ticks_left = 0;
  185. return;
  186. }
  187. ticks_left -= ticks;
  188. }
  189. std::uint64_t GetTicksRemaining() override {
  190. return ticks_left;
  191. }
  192. std::uint64_t GetCNTPCT() override {
  193. return 0x10000000000 - ticks_left;
  194. }
  195. };