| 
									
										
										
										
											2018-01-13 16:22:39 -05:00
										 |  |  | // Copyright 2018 yuzu emulator team
 | 
					
						
							| 
									
										
										
										
											2016-09-01 23:07:14 -04:00
										 |  |  | // Licensed under GPLv2 or any later version
 | 
					
						
							|  |  |  | // Refer to the license.txt file included.
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-01-13 22:34:15 +00:00
										 |  |  | #include <cinttypes>
 | 
					
						
							| 
									
										
										
										
											2018-01-09 21:33:46 +00:00
										 |  |  | #include <memory>
 | 
					
						
							|  |  |  | #include <dynarmic/A64/a64.h>
 | 
					
						
							|  |  |  | #include <dynarmic/A64/config.h>
 | 
					
						
							| 
									
										
										
										
											2018-02-21 20:48:22 +00:00
										 |  |  | #include "common/logging/log.h"
 | 
					
						
							| 
									
										
										
										
											2018-09-04 11:02:59 +02:00
										 |  |  | #include "common/microprofile.h"
 | 
					
						
							| 
									
										
										
										
											2016-09-20 23:52:38 -07:00
										 |  |  | #include "core/arm/dynarmic/arm_dynarmic.h"
 | 
					
						
							| 
									
										
										
										
											2018-03-13 17:49:59 -04:00
										 |  |  | #include "core/core.h"
 | 
					
						
							| 
									
										
										
										
											2018-08-31 12:21:34 -04:00
										 |  |  | #include "core/core_cpu.h"
 | 
					
						
							| 
									
										
										
										
											2018-01-09 21:33:46 +00:00
										 |  |  | #include "core/core_timing.h"
 | 
					
						
							| 
									
										
										
										
											2018-09-19 20:40:31 +01:00
										 |  |  | #include "core/gdbstub/gdbstub.h"
 | 
					
						
							| 
									
										
										
										
											2018-07-31 08:06:09 -04:00
										 |  |  | #include "core/hle/kernel/process.h"
 | 
					
						
							| 
									
										
										
										
											2018-01-09 21:33:46 +00:00
										 |  |  | #include "core/hle/kernel/svc.h"
 | 
					
						
							| 
									
										
										
										
											2018-09-20 19:28:48 -04:00
										 |  |  | #include "core/hle/kernel/vm_manager.h"
 | 
					
						
							| 
									
										
										
										
											2018-01-09 21:33:46 +00:00
										 |  |  | #include "core/memory.h"
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-08-24 21:43:32 -04:00
										 |  |  | namespace Core { | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-02-09 00:04:05 +00:00
										 |  |  | using Vector = Dynarmic::A64::Vector; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-01-09 21:33:46 +00:00
										 |  |  | class ARM_Dynarmic_Callbacks : public Dynarmic::A64::UserCallbacks { | 
					
						
							|  |  |  | public: | 
					
						
							|  |  |  |     explicit ARM_Dynarmic_Callbacks(ARM_Dynarmic& parent) : parent(parent) {} | 
					
						
							|  |  |  |     ~ARM_Dynarmic_Callbacks() = default; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-01-13 22:34:15 +00:00
										 |  |  |     u8 MemoryRead8(u64 vaddr) override { | 
					
						
							| 
									
										
										
										
											2018-01-09 21:33:46 +00:00
										 |  |  |         return Memory::Read8(vaddr); | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2018-01-13 22:34:15 +00:00
										 |  |  |     u16 MemoryRead16(u64 vaddr) override { | 
					
						
							| 
									
										
										
										
											2018-01-09 21:33:46 +00:00
										 |  |  |         return Memory::Read16(vaddr); | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2018-01-13 22:34:15 +00:00
										 |  |  |     u32 MemoryRead32(u64 vaddr) override { | 
					
						
							| 
									
										
										
										
											2018-01-09 21:33:46 +00:00
										 |  |  |         return Memory::Read32(vaddr); | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2018-01-13 22:34:15 +00:00
										 |  |  |     u64 MemoryRead64(u64 vaddr) override { | 
					
						
							| 
									
										
										
										
											2018-01-09 21:33:46 +00:00
										 |  |  |         return Memory::Read64(vaddr); | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2018-02-09 00:04:05 +00:00
										 |  |  |     Vector MemoryRead128(u64 vaddr) override { | 
					
						
							|  |  |  |         return {Memory::Read64(vaddr), Memory::Read64(vaddr + 8)}; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2018-01-09 21:33:46 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-01-13 22:34:15 +00:00
										 |  |  |     void MemoryWrite8(u64 vaddr, u8 value) override { | 
					
						
							| 
									
										
										
										
											2018-01-09 21:33:46 +00:00
										 |  |  |         Memory::Write8(vaddr, value); | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2018-01-13 22:34:15 +00:00
										 |  |  |     void MemoryWrite16(u64 vaddr, u16 value) override { | 
					
						
							| 
									
										
										
										
											2018-01-09 21:33:46 +00:00
										 |  |  |         Memory::Write16(vaddr, value); | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2018-01-13 22:34:15 +00:00
										 |  |  |     void MemoryWrite32(u64 vaddr, u32 value) override { | 
					
						
							| 
									
										
										
										
											2018-01-09 21:33:46 +00:00
										 |  |  |         Memory::Write32(vaddr, value); | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2018-01-13 22:34:15 +00:00
										 |  |  |     void MemoryWrite64(u64 vaddr, u64 value) override { | 
					
						
							| 
									
										
										
										
											2018-01-09 21:33:46 +00:00
										 |  |  |         Memory::Write64(vaddr, value); | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2018-02-09 00:04:05 +00:00
										 |  |  |     void MemoryWrite128(u64 vaddr, Vector value) override { | 
					
						
							|  |  |  |         Memory::Write64(vaddr, value[0]); | 
					
						
							|  |  |  |         Memory::Write64(vaddr + 8, value[1]); | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2018-01-09 21:33:46 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-09-15 15:21:06 +02:00
										 |  |  |     void InterpreterFallback(u64 pc, std::size_t num_instructions) override { | 
					
						
							| 
									
										
										
										
											2018-07-02 10:13:26 -06:00
										 |  |  |         LOG_INFO(Core_ARM, "Unicorn fallback @ 0x{:X} for {} instructions (instr = {:08X})", pc, | 
					
						
							| 
									
										
										
										
											2018-07-02 10:20:50 -06:00
										 |  |  |                  num_instructions, MemoryReadCode(pc)); | 
					
						
							| 
									
										
										
										
											2018-02-21 20:48:22 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-01-09 21:33:46 +00:00
										 |  |  |         ARM_Interface::ThreadContext ctx; | 
					
						
							|  |  |  |         parent.SaveContext(ctx); | 
					
						
							|  |  |  |         parent.inner_unicorn.LoadContext(ctx); | 
					
						
							| 
									
										
										
										
											2018-01-19 18:01:41 -05:00
										 |  |  |         parent.inner_unicorn.ExecuteInstructions(static_cast<int>(num_instructions)); | 
					
						
							| 
									
										
										
										
											2018-01-09 21:33:46 +00:00
										 |  |  |         parent.inner_unicorn.SaveContext(ctx); | 
					
						
							|  |  |  |         parent.LoadContext(ctx); | 
					
						
							|  |  |  |         num_interpreted_instructions += num_instructions; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-02-09 00:04:05 +00:00
										 |  |  |     void ExceptionRaised(u64 pc, Dynarmic::A64::Exception exception) override { | 
					
						
							| 
									
										
											  
											
												dynarmic: Update to 6b4c6b0
6b4c6b0 impl: Update PC when raising exception
7a1313a A64: Implement FDIV (vector)
b2d781d system: Raise exception for YIELD, WFE, WFI, SEV, SEVL
b277bf5 Correct FPSR and FPCR
7673933 A64: Implement USHL
8d0e558 A64: Implement UCVTF (vector, integer), scalar variant
da9a4f8 A64: Partially implement FCVTZU (scalar, fixed-point) and FCVTZS (scalar, fixed-point)
7479684 A64: Implement system register TPIDR_EL0
0fd75fd A64: Implement system registers FPCR and FPSR
31e370c A64: Implement system register CNTPCT_EL0
9a88fd3 A64: Implement system register CTR_EL0
1d16896 A64: Implement NEG (vector)
3184edf IR: Add IR instruction ZeroVector
31f8fbc emit_x64_floating_point: Add maybe_unused to preprocess parameter
567eb1a A64: Implement FMINNM (scalar)
c6d8fa1 A64: Implement FMAXNM (scalar)
616056d constant_pool: Add frame parameter
a3747cb A64: Implement ADDP (scalar)
5cd5d9f reg_alloc: Only exchange GPRs
dd0452a A64: Implement DUP (element), scalar variant
e5732ea emit_x64_floating_point: Correct FP{Max,Min}{32,64} implementations for -0/+0
40eb9c3 A64: Implement FMAX (scalar), FMIN (scalar)
7cef39b fuzz_with_unicorn: QEMU's implementation of FCVT is incorrect
826dce2 travis: Switch unicorn repository
9605f28 a64/config: Allow NaN emulation accuracy to be set
e9435bc a64_emit_x64: Add conf to A64EmitContext
30b596d fuzz_with_unicorn: Explicitly test floating point instructions
be292a8 A64: Implement FSQRT (scalar)
3c42d48 backend_x64: Accurately handle NaNs
4aefed0 fuzz_with_unicorn: Print AArch64 disassembly
											
										 
											2018-02-21 20:51:54 +00:00
										 |  |  |         switch (exception) { | 
					
						
							|  |  |  |         case Dynarmic::A64::Exception::WaitForInterrupt: | 
					
						
							|  |  |  |         case Dynarmic::A64::Exception::WaitForEvent: | 
					
						
							|  |  |  |         case Dynarmic::A64::Exception::SendEvent: | 
					
						
							|  |  |  |         case Dynarmic::A64::Exception::SendEventLocal: | 
					
						
							|  |  |  |         case Dynarmic::A64::Exception::Yield: | 
					
						
							|  |  |  |             return; | 
					
						
							| 
									
										
										
										
											2018-09-19 20:40:31 +01:00
										 |  |  |         case Dynarmic::A64::Exception::Breakpoint: | 
					
						
							|  |  |  |             if (GDBStub::IsServerEnabled()) { | 
					
						
							| 
									
										
										
										
											2018-09-20 19:12:42 +01:00
										 |  |  |                 parent.jit->HaltExecution(); | 
					
						
							| 
									
										
										
										
											2018-09-19 20:40:31 +01:00
										 |  |  |                 parent.SetPC(pc); | 
					
						
							|  |  |  |                 Kernel::Thread* thread = Kernel::GetCurrentThread(); | 
					
						
							| 
									
										
										
										
											2018-10-03 18:47:57 -04:00
										 |  |  |                 parent.SaveContext(thread->GetContext()); | 
					
						
							| 
									
										
										
										
											2018-09-19 20:40:31 +01:00
										 |  |  |                 GDBStub::Break(); | 
					
						
							|  |  |  |                 GDBStub::SendTrap(thread, 5); | 
					
						
							|  |  |  |                 return; | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |             [[fallthrough]]; | 
					
						
							| 
									
										
											  
											
												dynarmic: Update to 6b4c6b0
6b4c6b0 impl: Update PC when raising exception
7a1313a A64: Implement FDIV (vector)
b2d781d system: Raise exception for YIELD, WFE, WFI, SEV, SEVL
b277bf5 Correct FPSR and FPCR
7673933 A64: Implement USHL
8d0e558 A64: Implement UCVTF (vector, integer), scalar variant
da9a4f8 A64: Partially implement FCVTZU (scalar, fixed-point) and FCVTZS (scalar, fixed-point)
7479684 A64: Implement system register TPIDR_EL0
0fd75fd A64: Implement system registers FPCR and FPSR
31e370c A64: Implement system register CNTPCT_EL0
9a88fd3 A64: Implement system register CTR_EL0
1d16896 A64: Implement NEG (vector)
3184edf IR: Add IR instruction ZeroVector
31f8fbc emit_x64_floating_point: Add maybe_unused to preprocess parameter
567eb1a A64: Implement FMINNM (scalar)
c6d8fa1 A64: Implement FMAXNM (scalar)
616056d constant_pool: Add frame parameter
a3747cb A64: Implement ADDP (scalar)
5cd5d9f reg_alloc: Only exchange GPRs
dd0452a A64: Implement DUP (element), scalar variant
e5732ea emit_x64_floating_point: Correct FP{Max,Min}{32,64} implementations for -0/+0
40eb9c3 A64: Implement FMAX (scalar), FMIN (scalar)
7cef39b fuzz_with_unicorn: QEMU's implementation of FCVT is incorrect
826dce2 travis: Switch unicorn repository
9605f28 a64/config: Allow NaN emulation accuracy to be set
e9435bc a64_emit_x64: Add conf to A64EmitContext
30b596d fuzz_with_unicorn: Explicitly test floating point instructions
be292a8 A64: Implement FSQRT (scalar)
3c42d48 backend_x64: Accurately handle NaNs
4aefed0 fuzz_with_unicorn: Print AArch64 disassembly
											
										 
											2018-02-21 20:51:54 +00:00
										 |  |  |         default: | 
					
						
							| 
									
										
										
										
											2018-04-27 07:54:05 -04:00
										 |  |  |             ASSERT_MSG(false, "ExceptionRaised(exception = {}, pc = {:X})", | 
					
						
							| 
									
										
										
										
											2018-09-15 15:21:06 +02:00
										 |  |  |                        static_cast<std::size_t>(exception), pc); | 
					
						
							| 
									
										
											  
											
												dynarmic: Update to 6b4c6b0
6b4c6b0 impl: Update PC when raising exception
7a1313a A64: Implement FDIV (vector)
b2d781d system: Raise exception for YIELD, WFE, WFI, SEV, SEVL
b277bf5 Correct FPSR and FPCR
7673933 A64: Implement USHL
8d0e558 A64: Implement UCVTF (vector, integer), scalar variant
da9a4f8 A64: Partially implement FCVTZU (scalar, fixed-point) and FCVTZS (scalar, fixed-point)
7479684 A64: Implement system register TPIDR_EL0
0fd75fd A64: Implement system registers FPCR and FPSR
31e370c A64: Implement system register CNTPCT_EL0
9a88fd3 A64: Implement system register CTR_EL0
1d16896 A64: Implement NEG (vector)
3184edf IR: Add IR instruction ZeroVector
31f8fbc emit_x64_floating_point: Add maybe_unused to preprocess parameter
567eb1a A64: Implement FMINNM (scalar)
c6d8fa1 A64: Implement FMAXNM (scalar)
616056d constant_pool: Add frame parameter
a3747cb A64: Implement ADDP (scalar)
5cd5d9f reg_alloc: Only exchange GPRs
dd0452a A64: Implement DUP (element), scalar variant
e5732ea emit_x64_floating_point: Correct FP{Max,Min}{32,64} implementations for -0/+0
40eb9c3 A64: Implement FMAX (scalar), FMIN (scalar)
7cef39b fuzz_with_unicorn: QEMU's implementation of FCVT is incorrect
826dce2 travis: Switch unicorn repository
9605f28 a64/config: Allow NaN emulation accuracy to be set
e9435bc a64_emit_x64: Add conf to A64EmitContext
30b596d fuzz_with_unicorn: Explicitly test floating point instructions
be292a8 A64: Implement FSQRT (scalar)
3c42d48 backend_x64: Accurately handle NaNs
4aefed0 fuzz_with_unicorn: Print AArch64 disassembly
											
										 
											2018-02-21 20:51:54 +00:00
										 |  |  |         } | 
					
						
							| 
									
										
										
										
											2018-01-13 22:34:15 +00:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     void CallSVC(u32 swi) override { | 
					
						
							| 
									
										
										
										
											2018-01-09 21:33:46 +00:00
										 |  |  |         Kernel::CallSVC(swi); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-01-13 22:34:15 +00:00
										 |  |  |     void AddTicks(u64 ticks) override { | 
					
						
							| 
									
										
										
										
											2018-08-12 20:41:28 -05:00
										 |  |  |         // Divide the number of ticks by the amount of CPU cores. TODO(Subv): This yields only a
 | 
					
						
							|  |  |  |         // rough approximation of the amount of executed ticks in the system, it may be thrown off
 | 
					
						
							|  |  |  |         // if not all cores are doing a similar amount of work. Instead of doing this, we should
 | 
					
						
							|  |  |  |         // device a way so that timing is consistent across all cores without increasing the ticks 4
 | 
					
						
							|  |  |  |         // times.
 | 
					
						
							|  |  |  |         u64 amortized_ticks = (ticks - num_interpreted_instructions) / Core::NUM_CPU_CORES; | 
					
						
							|  |  |  |         // Always execute at least one tick.
 | 
					
						
							|  |  |  |         amortized_ticks = std::max<u64>(amortized_ticks, 1); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         CoreTiming::AddTicks(amortized_ticks); | 
					
						
							| 
									
										
										
										
											2018-03-24 09:02:19 +00:00
										 |  |  |         num_interpreted_instructions = 0; | 
					
						
							| 
									
										
										
										
											2018-01-09 21:33:46 +00:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2018-01-13 22:34:15 +00:00
										 |  |  |     u64 GetTicksRemaining() override { | 
					
						
							| 
									
										
										
										
											2018-03-24 09:02:19 +00:00
										 |  |  |         return std::max(CoreTiming::GetDowncount(), 0); | 
					
						
							| 
									
										
										
										
											2018-01-09 21:33:46 +00:00
										 |  |  |     } | 
					
						
							| 
									
										
											  
											
												dynarmic: Update to 6b4c6b0
6b4c6b0 impl: Update PC when raising exception
7a1313a A64: Implement FDIV (vector)
b2d781d system: Raise exception for YIELD, WFE, WFI, SEV, SEVL
b277bf5 Correct FPSR and FPCR
7673933 A64: Implement USHL
8d0e558 A64: Implement UCVTF (vector, integer), scalar variant
da9a4f8 A64: Partially implement FCVTZU (scalar, fixed-point) and FCVTZS (scalar, fixed-point)
7479684 A64: Implement system register TPIDR_EL0
0fd75fd A64: Implement system registers FPCR and FPSR
31e370c A64: Implement system register CNTPCT_EL0
9a88fd3 A64: Implement system register CTR_EL0
1d16896 A64: Implement NEG (vector)
3184edf IR: Add IR instruction ZeroVector
31f8fbc emit_x64_floating_point: Add maybe_unused to preprocess parameter
567eb1a A64: Implement FMINNM (scalar)
c6d8fa1 A64: Implement FMAXNM (scalar)
616056d constant_pool: Add frame parameter
a3747cb A64: Implement ADDP (scalar)
5cd5d9f reg_alloc: Only exchange GPRs
dd0452a A64: Implement DUP (element), scalar variant
e5732ea emit_x64_floating_point: Correct FP{Max,Min}{32,64} implementations for -0/+0
40eb9c3 A64: Implement FMAX (scalar), FMIN (scalar)
7cef39b fuzz_with_unicorn: QEMU's implementation of FCVT is incorrect
826dce2 travis: Switch unicorn repository
9605f28 a64/config: Allow NaN emulation accuracy to be set
e9435bc a64_emit_x64: Add conf to A64EmitContext
30b596d fuzz_with_unicorn: Explicitly test floating point instructions
be292a8 A64: Implement FSQRT (scalar)
3c42d48 backend_x64: Accurately handle NaNs
4aefed0 fuzz_with_unicorn: Print AArch64 disassembly
											
										 
											2018-02-21 20:51:54 +00:00
										 |  |  |     u64 GetCNTPCT() override { | 
					
						
							|  |  |  |         return CoreTiming::GetTicks(); | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2018-01-09 21:33:46 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |     ARM_Dynarmic& parent; | 
					
						
							| 
									
										
										
										
											2018-09-15 15:21:06 +02:00
										 |  |  |     std::size_t num_interpreted_instructions = 0; | 
					
						
							| 
									
										
										
										
											2018-02-12 21:53:32 +00:00
										 |  |  |     u64 tpidrro_el0 = 0; | 
					
						
							| 
									
										
											  
											
												dynarmic: Update to 6b4c6b0
6b4c6b0 impl: Update PC when raising exception
7a1313a A64: Implement FDIV (vector)
b2d781d system: Raise exception for YIELD, WFE, WFI, SEV, SEVL
b277bf5 Correct FPSR and FPCR
7673933 A64: Implement USHL
8d0e558 A64: Implement UCVTF (vector, integer), scalar variant
da9a4f8 A64: Partially implement FCVTZU (scalar, fixed-point) and FCVTZS (scalar, fixed-point)
7479684 A64: Implement system register TPIDR_EL0
0fd75fd A64: Implement system registers FPCR and FPSR
31e370c A64: Implement system register CNTPCT_EL0
9a88fd3 A64: Implement system register CTR_EL0
1d16896 A64: Implement NEG (vector)
3184edf IR: Add IR instruction ZeroVector
31f8fbc emit_x64_floating_point: Add maybe_unused to preprocess parameter
567eb1a A64: Implement FMINNM (scalar)
c6d8fa1 A64: Implement FMAXNM (scalar)
616056d constant_pool: Add frame parameter
a3747cb A64: Implement ADDP (scalar)
5cd5d9f reg_alloc: Only exchange GPRs
dd0452a A64: Implement DUP (element), scalar variant
e5732ea emit_x64_floating_point: Correct FP{Max,Min}{32,64} implementations for -0/+0
40eb9c3 A64: Implement FMAX (scalar), FMIN (scalar)
7cef39b fuzz_with_unicorn: QEMU's implementation of FCVT is incorrect
826dce2 travis: Switch unicorn repository
9605f28 a64/config: Allow NaN emulation accuracy to be set
e9435bc a64_emit_x64: Add conf to A64EmitContext
30b596d fuzz_with_unicorn: Explicitly test floating point instructions
be292a8 A64: Implement FSQRT (scalar)
3c42d48 backend_x64: Accurately handle NaNs
4aefed0 fuzz_with_unicorn: Print AArch64 disassembly
											
										 
											2018-02-21 20:51:54 +00:00
										 |  |  |     u64 tpidr_el0 = 0; | 
					
						
							| 
									
										
										
										
											2018-01-09 21:33:46 +00:00
										 |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-07-23 23:19:35 -04:00
										 |  |  | std::unique_ptr<Dynarmic::A64::Jit> ARM_Dynarmic::MakeJit() const { | 
					
						
							| 
									
										
										
										
											2018-10-10 00:42:10 -04:00
										 |  |  |     auto* current_process = Core::CurrentProcess(); | 
					
						
							| 
									
										
										
										
											2018-09-29 18:47:00 -04:00
										 |  |  |     auto** const page_table = current_process->VMManager().page_table.pointers.data(); | 
					
						
							| 
									
										
										
										
											2018-02-12 21:53:32 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |     Dynarmic::A64::UserConfig config; | 
					
						
							| 
									
										
										
										
											2018-07-03 14:28:46 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  |     // Callbacks
 | 
					
						
							| 
									
										
										
										
											2018-02-12 21:53:32 +00:00
										 |  |  |     config.callbacks = cb.get(); | 
					
						
							| 
									
										
										
										
											2018-07-03 14:28:46 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  |     // Memory
 | 
					
						
							|  |  |  |     config.page_table = reinterpret_cast<void**>(page_table); | 
					
						
							| 
									
										
										
										
											2018-09-29 18:47:00 -04:00
										 |  |  |     config.page_table_address_space_bits = current_process->VMManager().GetAddressSpaceWidth(); | 
					
						
							| 
									
										
										
										
											2018-07-03 14:28:46 +01:00
										 |  |  |     config.silently_mirror_page_table = false; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     // Multi-process state
 | 
					
						
							|  |  |  |     config.processor_id = core_index; | 
					
						
							| 
									
										
										
										
											2018-10-15 08:53:01 -04:00
										 |  |  |     config.global_monitor = &exclusive_monitor.monitor; | 
					
						
							| 
									
										
										
										
											2018-07-03 14:28:46 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  |     // System registers
 | 
					
						
							| 
									
										
										
										
											2018-02-12 21:53:32 +00:00
										 |  |  |     config.tpidrro_el0 = &cb->tpidrro_el0; | 
					
						
							| 
									
										
											  
											
												dynarmic: Update to 6b4c6b0
6b4c6b0 impl: Update PC when raising exception
7a1313a A64: Implement FDIV (vector)
b2d781d system: Raise exception for YIELD, WFE, WFI, SEV, SEVL
b277bf5 Correct FPSR and FPCR
7673933 A64: Implement USHL
8d0e558 A64: Implement UCVTF (vector, integer), scalar variant
da9a4f8 A64: Partially implement FCVTZU (scalar, fixed-point) and FCVTZS (scalar, fixed-point)
7479684 A64: Implement system register TPIDR_EL0
0fd75fd A64: Implement system registers FPCR and FPSR
31e370c A64: Implement system register CNTPCT_EL0
9a88fd3 A64: Implement system register CTR_EL0
1d16896 A64: Implement NEG (vector)
3184edf IR: Add IR instruction ZeroVector
31f8fbc emit_x64_floating_point: Add maybe_unused to preprocess parameter
567eb1a A64: Implement FMINNM (scalar)
c6d8fa1 A64: Implement FMAXNM (scalar)
616056d constant_pool: Add frame parameter
a3747cb A64: Implement ADDP (scalar)
5cd5d9f reg_alloc: Only exchange GPRs
dd0452a A64: Implement DUP (element), scalar variant
e5732ea emit_x64_floating_point: Correct FP{Max,Min}{32,64} implementations for -0/+0
40eb9c3 A64: Implement FMAX (scalar), FMIN (scalar)
7cef39b fuzz_with_unicorn: QEMU's implementation of FCVT is incorrect
826dce2 travis: Switch unicorn repository
9605f28 a64/config: Allow NaN emulation accuracy to be set
e9435bc a64_emit_x64: Add conf to A64EmitContext
30b596d fuzz_with_unicorn: Explicitly test floating point instructions
be292a8 A64: Implement FSQRT (scalar)
3c42d48 backend_x64: Accurately handle NaNs
4aefed0 fuzz_with_unicorn: Print AArch64 disassembly
											
										 
											2018-02-21 20:51:54 +00:00
										 |  |  |     config.tpidr_el0 = &cb->tpidr_el0; | 
					
						
							| 
									
										
										
										
											2018-02-12 21:53:32 +00:00
										 |  |  |     config.dczid_el0 = 4; | 
					
						
							| 
									
										
											  
											
												dynarmic: Update to 6b4c6b0
6b4c6b0 impl: Update PC when raising exception
7a1313a A64: Implement FDIV (vector)
b2d781d system: Raise exception for YIELD, WFE, WFI, SEV, SEVL
b277bf5 Correct FPSR and FPCR
7673933 A64: Implement USHL
8d0e558 A64: Implement UCVTF (vector, integer), scalar variant
da9a4f8 A64: Partially implement FCVTZU (scalar, fixed-point) and FCVTZS (scalar, fixed-point)
7479684 A64: Implement system register TPIDR_EL0
0fd75fd A64: Implement system registers FPCR and FPSR
31e370c A64: Implement system register CNTPCT_EL0
9a88fd3 A64: Implement system register CTR_EL0
1d16896 A64: Implement NEG (vector)
3184edf IR: Add IR instruction ZeroVector
31f8fbc emit_x64_floating_point: Add maybe_unused to preprocess parameter
567eb1a A64: Implement FMINNM (scalar)
c6d8fa1 A64: Implement FMAXNM (scalar)
616056d constant_pool: Add frame parameter
a3747cb A64: Implement ADDP (scalar)
5cd5d9f reg_alloc: Only exchange GPRs
dd0452a A64: Implement DUP (element), scalar variant
e5732ea emit_x64_floating_point: Correct FP{Max,Min}{32,64} implementations for -0/+0
40eb9c3 A64: Implement FMAX (scalar), FMIN (scalar)
7cef39b fuzz_with_unicorn: QEMU's implementation of FCVT is incorrect
826dce2 travis: Switch unicorn repository
9605f28 a64/config: Allow NaN emulation accuracy to be set
e9435bc a64_emit_x64: Add conf to A64EmitContext
30b596d fuzz_with_unicorn: Explicitly test floating point instructions
be292a8 A64: Implement FSQRT (scalar)
3c42d48 backend_x64: Accurately handle NaNs
4aefed0 fuzz_with_unicorn: Print AArch64 disassembly
											
										 
											2018-02-21 20:51:54 +00:00
										 |  |  |     config.ctr_el0 = 0x8444c004; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-08-16 10:12:20 +01:00
										 |  |  |     // Unpredictable instructions
 | 
					
						
							|  |  |  |     config.define_unpredictable_behaviour = true; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-02-09 00:04:05 +00:00
										 |  |  |     return std::make_unique<Dynarmic::A64::Jit>(config); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-09-04 11:02:59 +02:00
										 |  |  | MICROPROFILE_DEFINE(ARM_Jit_Dynarmic, "ARM JIT", "Dynarmic", MP_RGB(255, 64, 64)); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-02-14 18:47:48 +01:00
										 |  |  | void ARM_Dynarmic::Run() { | 
					
						
							| 
									
										
										
										
											2018-09-04 11:02:59 +02:00
										 |  |  |     MICROPROFILE_SCOPE(ARM_Jit_Dynarmic); | 
					
						
							| 
									
										
										
										
											2018-02-14 18:47:48 +01:00
										 |  |  |     ASSERT(Memory::GetCurrentPageTable() == current_page_table); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     jit->Run(); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void ARM_Dynarmic::Step() { | 
					
						
							|  |  |  |     cb->InterpreterFallback(jit->GetPC(), 1); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-10-15 08:53:01 -04:00
										 |  |  | ARM_Dynarmic::ARM_Dynarmic(ExclusiveMonitor& exclusive_monitor, std::size_t core_index) | 
					
						
							| 
									
										
										
										
											2018-07-30 23:46:07 -04:00
										 |  |  |     : cb(std::make_unique<ARM_Dynarmic_Callbacks>(*this)), core_index{core_index}, | 
					
						
							| 
									
										
										
										
											2018-10-15 08:53:01 -04:00
										 |  |  |       exclusive_monitor{dynamic_cast<DynarmicExclusiveMonitor&>(exclusive_monitor)} { | 
					
						
							| 
									
										
										
										
											2018-09-22 21:17:43 -04:00
										 |  |  |     ThreadContext ctx{}; | 
					
						
							| 
									
										
										
										
											2018-01-09 21:33:46 +00:00
										 |  |  |     inner_unicorn.SaveContext(ctx); | 
					
						
							| 
									
										
										
										
											2018-02-14 18:47:48 +01:00
										 |  |  |     PageTableChanged(); | 
					
						
							| 
									
										
										
										
											2018-07-30 23:46:07 -04:00
										 |  |  |     LoadContext(ctx); | 
					
						
							| 
									
										
										
										
											2018-01-09 21:33:46 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | ARM_Dynarmic::~ARM_Dynarmic() = default; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-09-15 15:21:06 +02:00
										 |  |  | void ARM_Dynarmic::MapBackingMemory(u64 address, std::size_t size, u8* memory, | 
					
						
							| 
									
										
										
										
											2018-01-09 21:33:46 +00:00
										 |  |  |                                     Kernel::VMAPermission perms) { | 
					
						
							|  |  |  |     inner_unicorn.MapBackingMemory(address, size, memory, perms); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-09-15 15:21:06 +02:00
										 |  |  | void ARM_Dynarmic::UnmapMemory(u64 address, std::size_t size) { | 
					
						
							| 
									
										
										
										
											2018-03-16 18:22:14 -04:00
										 |  |  |     inner_unicorn.UnmapMemory(address, size); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-01-09 21:33:46 +00:00
										 |  |  | void ARM_Dynarmic::SetPC(u64 pc) { | 
					
						
							| 
									
										
										
										
											2018-02-09 00:04:05 +00:00
										 |  |  |     jit->SetPC(pc); | 
					
						
							| 
									
										
										
										
											2016-09-01 23:07:14 -04:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-08-28 21:09:42 -04:00
										 |  |  | u64 ARM_Dynarmic::GetPC() const { | 
					
						
							| 
									
										
										
										
											2018-02-09 00:04:05 +00:00
										 |  |  |     return jit->GetPC(); | 
					
						
							| 
									
										
										
										
											2016-09-01 23:07:14 -04:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-01-09 21:33:46 +00:00
										 |  |  | u64 ARM_Dynarmic::GetReg(int index) const { | 
					
						
							| 
									
										
										
										
											2018-02-09 00:04:05 +00:00
										 |  |  |     return jit->GetRegister(index); | 
					
						
							| 
									
										
										
										
											2016-09-01 23:07:14 -04:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-01-09 21:33:46 +00:00
										 |  |  | void ARM_Dynarmic::SetReg(int index, u64 value) { | 
					
						
							| 
									
										
										
										
											2018-02-09 00:04:05 +00:00
										 |  |  |     jit->SetRegister(index, value); | 
					
						
							| 
									
										
										
										
											2016-09-01 23:07:14 -04:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-09-18 02:49:40 -04:00
										 |  |  | u128 ARM_Dynarmic::GetVectorReg(int index) const { | 
					
						
							| 
									
										
										
										
											2018-02-09 00:04:05 +00:00
										 |  |  |     return jit->GetVector(index); | 
					
						
							| 
									
										
										
										
											2016-09-01 23:07:14 -04:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-09-18 02:49:40 -04:00
										 |  |  | void ARM_Dynarmic::SetVectorReg(int index, u128 value) { | 
					
						
							| 
									
										
										
										
											2018-02-09 00:04:05 +00:00
										 |  |  |     jit->SetVector(index, value); | 
					
						
							| 
									
										
										
										
											2016-09-01 23:07:14 -04:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-09-18 02:49:40 -04:00
										 |  |  | u32 ARM_Dynarmic::GetPSTATE() const { | 
					
						
							| 
									
										
										
										
											2018-02-09 00:04:05 +00:00
										 |  |  |     return jit->GetPstate(); | 
					
						
							| 
									
										
										
										
											2016-09-01 23:07:14 -04:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-09-18 02:49:40 -04:00
										 |  |  | void ARM_Dynarmic::SetPSTATE(u32 pstate) { | 
					
						
							|  |  |  |     jit->SetPstate(pstate); | 
					
						
							| 
									
										
										
										
											2016-09-01 23:07:14 -04:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-01-09 21:33:46 +00:00
										 |  |  | u64 ARM_Dynarmic::GetTlsAddress() const { | 
					
						
							| 
									
										
										
										
											2018-02-12 21:53:32 +00:00
										 |  |  |     return cb->tpidrro_el0; | 
					
						
							| 
									
										
										
										
											2017-09-30 14:16:39 -04:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-07-31 07:58:24 -04:00
										 |  |  | void ARM_Dynarmic::SetTlsAddress(VAddr address) { | 
					
						
							| 
									
										
										
										
											2018-02-12 21:53:32 +00:00
										 |  |  |     cb->tpidrro_el0 = address; | 
					
						
							| 
									
										
										
										
											2017-09-30 14:16:39 -04:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-07-20 19:57:45 -05:00
										 |  |  | u64 ARM_Dynarmic::GetTPIDR_EL0() const { | 
					
						
							|  |  |  |     return cb->tpidr_el0; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void ARM_Dynarmic::SetTPIDR_EL0(u64 value) { | 
					
						
							|  |  |  |     cb->tpidr_el0 = value; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-07-31 07:56:57 -04:00
										 |  |  | void ARM_Dynarmic::SaveContext(ThreadContext& ctx) { | 
					
						
							| 
									
										
										
										
											2018-02-09 00:04:05 +00:00
										 |  |  |     ctx.cpu_registers = jit->GetRegisters(); | 
					
						
							|  |  |  |     ctx.sp = jit->GetSP(); | 
					
						
							|  |  |  |     ctx.pc = jit->GetPC(); | 
					
						
							| 
									
										
										
										
											2018-09-18 02:49:40 -04:00
										 |  |  |     ctx.pstate = jit->GetPstate(); | 
					
						
							|  |  |  |     ctx.vector_registers = jit->GetVectors(); | 
					
						
							|  |  |  |     ctx.fpcr = jit->GetFpcr(); | 
					
						
							| 
									
										
										
										
											2018-09-29 17:58:26 -04:00
										 |  |  |     ctx.fpsr = jit->GetFpsr(); | 
					
						
							|  |  |  |     ctx.tpidr = cb->tpidr_el0; | 
					
						
							| 
									
										
										
										
											2016-09-01 23:07:14 -04:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-07-31 07:56:57 -04:00
										 |  |  | void ARM_Dynarmic::LoadContext(const ThreadContext& ctx) { | 
					
						
							| 
									
										
										
										
											2018-02-09 00:04:05 +00:00
										 |  |  |     jit->SetRegisters(ctx.cpu_registers); | 
					
						
							|  |  |  |     jit->SetSP(ctx.sp); | 
					
						
							|  |  |  |     jit->SetPC(ctx.pc); | 
					
						
							| 
									
										
										
										
											2018-09-29 17:58:26 -04:00
										 |  |  |     jit->SetPstate(ctx.pstate); | 
					
						
							| 
									
										
										
										
											2018-09-18 02:49:40 -04:00
										 |  |  |     jit->SetVectors(ctx.vector_registers); | 
					
						
							| 
									
										
										
										
											2018-09-29 17:58:26 -04:00
										 |  |  |     jit->SetFpcr(ctx.fpcr); | 
					
						
							|  |  |  |     jit->SetFpsr(ctx.fpsr); | 
					
						
							|  |  |  |     SetTPIDR_EL0(ctx.tpidr); | 
					
						
							| 
									
										
										
										
											2016-09-01 23:07:14 -04:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void ARM_Dynarmic::PrepareReschedule() { | 
					
						
							| 
									
										
										
										
											2018-08-13 13:59:01 +01:00
										 |  |  |     jit->HaltExecution(); | 
					
						
							| 
									
										
										
										
											2016-09-01 23:07:14 -04:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void ARM_Dynarmic::ClearInstructionCache() { | 
					
						
							| 
									
										
										
										
											2018-02-09 00:04:05 +00:00
										 |  |  |     jit->ClearCache(); | 
					
						
							| 
									
										
										
										
											2016-09-01 23:07:14 -04:00
										 |  |  | } | 
					
						
							| 
									
										
										
										
											2017-09-24 22:44:13 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-07-16 11:24:00 +01:00
										 |  |  | void ARM_Dynarmic::ClearExclusiveState() { | 
					
						
							|  |  |  |     jit->ClearExclusiveState(); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-09-24 22:44:13 +01:00
										 |  |  | void ARM_Dynarmic::PageTableChanged() { | 
					
						
							| 
									
										
										
										
											2018-07-03 14:28:46 +01:00
										 |  |  |     jit = MakeJit(); | 
					
						
							| 
									
										
										
										
											2018-02-14 18:47:48 +01:00
										 |  |  |     current_page_table = Memory::GetCurrentPageTable(); | 
					
						
							| 
									
										
										
										
											2017-09-24 22:44:13 +01:00
										 |  |  | } | 
					
						
							| 
									
										
										
										
											2018-07-03 14:28:46 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-09-15 15:21:06 +02:00
										 |  |  | DynarmicExclusiveMonitor::DynarmicExclusiveMonitor(std::size_t core_count) : monitor(core_count) {} | 
					
						
							| 
									
										
										
										
											2018-07-03 14:28:46 +01:00
										 |  |  | DynarmicExclusiveMonitor::~DynarmicExclusiveMonitor() = default; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-09-15 15:21:06 +02:00
										 |  |  | void DynarmicExclusiveMonitor::SetExclusive(std::size_t core_index, VAddr addr) { | 
					
						
							| 
									
										
										
										
											2018-07-03 14:28:46 +01:00
										 |  |  |     // Size doesn't actually matter.
 | 
					
						
							|  |  |  |     monitor.Mark(core_index, addr, 16); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void DynarmicExclusiveMonitor::ClearExclusive() { | 
					
						
							|  |  |  |     monitor.Clear(); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-09-15 15:21:06 +02:00
										 |  |  | bool DynarmicExclusiveMonitor::ExclusiveWrite8(std::size_t core_index, VAddr vaddr, u8 value) { | 
					
						
							| 
									
										
										
										
											2018-07-03 14:28:46 +01:00
										 |  |  |     return monitor.DoExclusiveOperation(core_index, vaddr, 1, | 
					
						
							|  |  |  |                                         [&] { Memory::Write8(vaddr, value); }); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-09-15 15:21:06 +02:00
										 |  |  | bool DynarmicExclusiveMonitor::ExclusiveWrite16(std::size_t core_index, VAddr vaddr, u16 value) { | 
					
						
							| 
									
										
										
										
											2018-07-03 14:28:46 +01:00
										 |  |  |     return monitor.DoExclusiveOperation(core_index, vaddr, 2, | 
					
						
							|  |  |  |                                         [&] { Memory::Write16(vaddr, value); }); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-09-15 15:21:06 +02:00
										 |  |  | bool DynarmicExclusiveMonitor::ExclusiveWrite32(std::size_t core_index, VAddr vaddr, u32 value) { | 
					
						
							| 
									
										
										
										
											2018-07-03 14:28:46 +01:00
										 |  |  |     return monitor.DoExclusiveOperation(core_index, vaddr, 4, | 
					
						
							|  |  |  |                                         [&] { Memory::Write32(vaddr, value); }); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-09-15 15:21:06 +02:00
										 |  |  | bool DynarmicExclusiveMonitor::ExclusiveWrite64(std::size_t core_index, VAddr vaddr, u64 value) { | 
					
						
							| 
									
										
										
										
											2018-07-03 14:28:46 +01:00
										 |  |  |     return monitor.DoExclusiveOperation(core_index, vaddr, 8, | 
					
						
							|  |  |  |                                         [&] { Memory::Write64(vaddr, value); }); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-09-15 15:21:06 +02:00
										 |  |  | bool DynarmicExclusiveMonitor::ExclusiveWrite128(std::size_t core_index, VAddr vaddr, u128 value) { | 
					
						
							| 
									
										
										
										
											2018-07-03 14:28:46 +01:00
										 |  |  |     return monitor.DoExclusiveOperation(core_index, vaddr, 16, [&] { | 
					
						
							| 
									
										
										
										
											2018-09-18 03:54:05 -04:00
										 |  |  |         Memory::Write64(vaddr + 0, value[0]); | 
					
						
							|  |  |  |         Memory::Write64(vaddr + 8, value[1]); | 
					
						
							| 
									
										
										
										
											2018-07-03 14:28:46 +01:00
										 |  |  |     }); | 
					
						
							|  |  |  | } | 
					
						
							| 
									
										
										
										
											2018-08-24 21:43:32 -04:00
										 |  |  | 
 | 
					
						
							|  |  |  | } // namespace Core
 |