mirror of
				https://github.com/PabloMK7/citra.git
				synced 2025-10-30 21:30:04 +00:00 
			
		
		
		
	Merge remote-tracking branch 'upstream/master' into feature/savestates-2
This commit is contained in:
		
						commit
						915c426dc9
					
				
					 11 changed files with 165 additions and 121 deletions
				
			
		|  | @ -32,7 +32,7 @@ matrix: | ||||||
|     - os: osx |     - os: osx | ||||||
|       env: NAME="macos build" |       env: NAME="macos build" | ||||||
|       sudo: false |       sudo: false | ||||||
|       osx_image: xcode10 |       osx_image: xcode10.2 | ||||||
|       install: "./.travis/macos/deps.sh" |       install: "./.travis/macos/deps.sh" | ||||||
|       script: "./.travis/macos/build.sh" |       script: "./.travis/macos/build.sh" | ||||||
|       after_success: "./.travis/macos/upload.sh" |       after_success: "./.travis/macos/upload.sh" | ||||||
|  |  | ||||||
							
								
								
									
										2
									
								
								externals/dynarmic
									
										
									
									
										vendored
									
									
								
							
							
						
						
									
										2
									
								
								externals/dynarmic
									
										
									
									
										vendored
									
									
								
							|  | @ -1 +1 @@ | ||||||
| Subproject commit 4e6848d1c9e8dadc70595c15b5589f8b14aad478 | Subproject commit b58048a5a88ad6184d64f16cfd2c5d63a1952e77 | ||||||
|  | @ -103,8 +103,6 @@ void RegistersWidget::OnEmulationStopping() { | ||||||
| 
 | 
 | ||||||
|     vfp_system_registers->child(0)->setText(1, QString{}); |     vfp_system_registers->child(0)->setText(1, QString{}); | ||||||
|     vfp_system_registers->child(1)->setText(1, QString{}); |     vfp_system_registers->child(1)->setText(1, QString{}); | ||||||
|     vfp_system_registers->child(2)->setText(1, QString{}); |  | ||||||
|     vfp_system_registers->child(3)->setText(1, QString{}); |  | ||||||
| 
 | 
 | ||||||
|     setEnabled(false); |     setEnabled(false); | ||||||
| } | } | ||||||
|  | @ -188,16 +186,12 @@ void RegistersWidget::CreateVFPSystemRegisterChildren() { | ||||||
| 
 | 
 | ||||||
|     vfp_system_registers->addChild(fpscr); |     vfp_system_registers->addChild(fpscr); | ||||||
|     vfp_system_registers->addChild(fpexc); |     vfp_system_registers->addChild(fpexc); | ||||||
|     vfp_system_registers->addChild(new QTreeWidgetItem(QStringList(QStringLiteral("FPINST")))); |  | ||||||
|     vfp_system_registers->addChild(new QTreeWidgetItem(QStringList(QStringLiteral("FPINST2")))); |  | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| void RegistersWidget::UpdateVFPSystemRegisterValues() { | void RegistersWidget::UpdateVFPSystemRegisterValues() { | ||||||
|     // Todo: handle all cores
 |     // Todo: handle all cores
 | ||||||
|     const u32 fpscr_val = Core::GetCore(0).GetVFPSystemReg(VFP_FPSCR); |     const u32 fpscr_val = Core::GetCore(0).GetVFPSystemReg(VFP_FPSCR); | ||||||
|     const u32 fpexc_val = Core::GetCore(0).GetVFPSystemReg(VFP_FPEXC); |     const u32 fpexc_val = Core::GetCore(0).GetVFPSystemReg(VFP_FPEXC); | ||||||
|     const u32 fpinst_val = Core::GetCore(0).GetVFPSystemReg(VFP_FPINST); |  | ||||||
|     const u32 fpinst2_val = Core::GetCore(0).GetVFPSystemReg(VFP_FPINST2); |  | ||||||
| 
 | 
 | ||||||
|     QTreeWidgetItem* const fpscr = vfp_system_registers->child(0); |     QTreeWidgetItem* const fpscr = vfp_system_registers->child(0); | ||||||
|     fpscr->setText(1, QStringLiteral("0x%1").arg(fpscr_val, 8, 16, QLatin1Char('0'))); |     fpscr->setText(1, QStringLiteral("0x%1").arg(fpscr_val, 8, 16, QLatin1Char('0'))); | ||||||
|  | @ -237,9 +231,4 @@ void RegistersWidget::UpdateVFPSystemRegisterValues() { | ||||||
|     fpexc->child(5)->setText(1, QString::number((fpexc_val >> 28) & 1)); |     fpexc->child(5)->setText(1, QString::number((fpexc_val >> 28) & 1)); | ||||||
|     fpexc->child(6)->setText(1, QString::number((fpexc_val >> 30) & 1)); |     fpexc->child(6)->setText(1, QString::number((fpexc_val >> 30) & 1)); | ||||||
|     fpexc->child(7)->setText(1, QString::number((fpexc_val >> 31) & 1)); |     fpexc->child(7)->setText(1, QString::number((fpexc_val >> 31) & 1)); | ||||||
| 
 |  | ||||||
|     vfp_system_registers->child(2)->setText( |  | ||||||
|         1, QStringLiteral("0x%1").arg(fpinst_val, 8, 16, QLatin1Char('0'))); |  | ||||||
|     vfp_system_registers->child(3)->setText( |  | ||||||
|         1, QStringLiteral("0x%1").arg(fpinst2_val, 8, 16, QLatin1Char('0'))); |  | ||||||
| } | } | ||||||
|  |  | ||||||
|  | @ -9,6 +9,7 @@ | ||||||
| #include <atomic> | #include <atomic> | ||||||
| #include <cstddef> | #include <cstddef> | ||||||
| #include <cstring> | #include <cstring> | ||||||
|  | #include <new> | ||||||
| #include <type_traits> | #include <type_traits> | ||||||
| #include <vector> | #include <vector> | ||||||
| #include "common/common_types.h" | #include "common/common_types.h" | ||||||
|  | @ -29,7 +30,7 @@ class RingBuffer { | ||||||
|     static_assert(capacity < std::numeric_limits<std::size_t>::max() / 2 / granularity); |     static_assert(capacity < std::numeric_limits<std::size_t>::max() / 2 / granularity); | ||||||
|     static_assert((capacity & (capacity - 1)) == 0, "capacity must be a power of two"); |     static_assert((capacity & (capacity - 1)) == 0, "capacity must be a power of two"); | ||||||
|     // Ensure lock-free.
 |     // Ensure lock-free.
 | ||||||
|     static_assert(std::atomic<std::size_t>::is_always_lock_free); |     static_assert(std::atomic_size_t::is_always_lock_free); | ||||||
| 
 | 
 | ||||||
| public: | public: | ||||||
|     /// Pushes slots into the ring buffer
 |     /// Pushes slots into the ring buffer
 | ||||||
|  | @ -100,10 +101,22 @@ public: | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
| private: | private: | ||||||
|     // It is important to align the below variables for performance reasons:
 |     // It is important to separate the below atomics for performance reasons:
 | ||||||
|     // Having them on the same cache-line would result in false-sharing between them.
 |     // Having them on the same cache-line would result in false-sharing between them.
 | ||||||
|     alignas(128) std::atomic<std::size_t> m_read_index{0}; |     // TODO: Remove this ifdef whenever clang and GCC support
 | ||||||
|     alignas(128) std::atomic<std::size_t> m_write_index{0}; |     //       std::hardware_destructive_interference_size.
 | ||||||
|  | #if defined(_MSC_VER) && _MSC_VER >= 1911 | ||||||
|  |     static constexpr std::size_t padding_size = | ||||||
|  |         std::hardware_destructive_interference_size - sizeof(std::atomic_size_t); | ||||||
|  | #else | ||||||
|  |     static constexpr std::size_t padding_size = 128 - sizeof(std::atomic_size_t); | ||||||
|  | #endif | ||||||
|  | 
 | ||||||
|  |     std::atomic_size_t m_read_index{0}; | ||||||
|  |     char padding1[padding_size]; | ||||||
|  | 
 | ||||||
|  |     std::atomic_size_t m_write_index{0}; | ||||||
|  |     char padding2[padding_size]; | ||||||
| 
 | 
 | ||||||
|     std::array<T, granularity * capacity> m_data; |     std::array<T, granularity * capacity> m_data; | ||||||
| }; | }; | ||||||
|  |  | ||||||
|  | @ -263,11 +263,23 @@ private: | ||||||
|             ar << r; |             ar << r; | ||||||
|         } |         } | ||||||
|         for (std::size_t i = 0; i < VFPSystemRegister::VFP_SYSTEM_REGISTER_COUNT; i++) { |         for (std::size_t i = 0; i < VFPSystemRegister::VFP_SYSTEM_REGISTER_COUNT; i++) { | ||||||
|             const auto r = GetVFPSystemReg(static_cast<VFPSystemRegister>(i)); |             const auto reg = static_cast<VFPSystemRegister>(i); | ||||||
|  |             u32 r = 0; | ||||||
|  |             switch (reg) { | ||||||
|  |             case VFP_FPSCR: | ||||||
|  |             case VFP_FPEXC: | ||||||
|  |                 r = GetVFPSystemReg(reg); | ||||||
|  |             } | ||||||
|             ar << r; |             ar << r; | ||||||
|         } |         } | ||||||
|         for (std::size_t i = 0; i < CP15Register::CP15_REGISTER_COUNT; i++) { |         for (std::size_t i = 0; i < CP15Register::CP15_REGISTER_COUNT; i++) { | ||||||
|             const auto r = GetCP15Register(static_cast<CP15Register>(i)); |             const auto reg = static_cast<CP15Register>(i); | ||||||
|  |             u32 r = 0; | ||||||
|  |             switch (reg) { | ||||||
|  |             case CP15_THREAD_UPRW: | ||||||
|  |             case CP15_THREAD_URO: | ||||||
|  |                 r = GetCP15Register(reg); | ||||||
|  |             } | ||||||
|             ar << r; |             ar << r; | ||||||
|         } |         } | ||||||
|     } |     } | ||||||
|  | @ -296,11 +308,21 @@ private: | ||||||
|         } |         } | ||||||
|         for (std::size_t i = 0; i < VFPSystemRegister::VFP_SYSTEM_REGISTER_COUNT; i++) { |         for (std::size_t i = 0; i < VFPSystemRegister::VFP_SYSTEM_REGISTER_COUNT; i++) { | ||||||
|             ar >> r; |             ar >> r; | ||||||
|             SetVFPSystemReg(static_cast<VFPSystemRegister>(i), r); |             const auto reg = static_cast<VFPSystemRegister>(i); | ||||||
|  |             switch (reg) { | ||||||
|  |             case VFP_FPSCR: | ||||||
|  |             case VFP_FPEXC: | ||||||
|  |                 SetVFPSystemReg(reg, r); | ||||||
|  |             } | ||||||
|         } |         } | ||||||
|         for (std::size_t i = 0; i < CP15Register::CP15_REGISTER_COUNT; i++) { |         for (std::size_t i = 0; i < CP15Register::CP15_REGISTER_COUNT; i++) { | ||||||
|             ar >> r; |             ar >> r; | ||||||
|             SetCP15Register(static_cast<CP15Register>(i), r); |             const auto reg = static_cast<CP15Register>(i); | ||||||
|  |             switch (reg) { | ||||||
|  |             case CP15_THREAD_UPRW: | ||||||
|  |             case CP15_THREAD_URO: | ||||||
|  |                 SetCP15Register(reg, r); | ||||||
|  |             } | ||||||
|         } |         } | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|  |  | ||||||
|  | @ -9,7 +9,6 @@ | ||||||
| #include "common/microprofile.h" | #include "common/microprofile.h" | ||||||
| #include "core/arm/dynarmic/arm_dynarmic.h" | #include "core/arm/dynarmic/arm_dynarmic.h" | ||||||
| #include "core/arm/dynarmic/arm_dynarmic_cp15.h" | #include "core/arm/dynarmic/arm_dynarmic_cp15.h" | ||||||
| #include "core/arm/dyncom/arm_dyncom_interpreter.h" |  | ||||||
| #include "core/core.h" | #include "core/core.h" | ||||||
| #include "core/core_timing.h" | #include "core/core_timing.h" | ||||||
| #include "core/gdbstub/gdbstub.h" | #include "core/gdbstub/gdbstub.h" | ||||||
|  | @ -102,24 +101,9 @@ public: | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|     void InterpreterFallback(VAddr pc, std::size_t num_instructions) override { |     void InterpreterFallback(VAddr pc, std::size_t num_instructions) override { | ||||||
|         parent.interpreter_state->Reg = parent.jit->Regs(); |         // Should never happen.
 | ||||||
|         parent.interpreter_state->Cpsr = parent.jit->Cpsr(); |         UNREACHABLE_MSG("InterpeterFallback reached with pc = 0x{:08x}, code = 0x{:08x}, num = {}", | ||||||
|         parent.interpreter_state->Reg[15] = pc; |                         pc, MemoryReadCode(pc), num_instructions); | ||||||
|         parent.interpreter_state->ExtReg = parent.jit->ExtRegs(); |  | ||||||
|         parent.interpreter_state->VFP[VFP_FPSCR] = parent.jit->Fpscr(); |  | ||||||
|         parent.interpreter_state->NumInstrsToExecute = num_instructions; |  | ||||||
| 
 |  | ||||||
|         InterpreterMainLoop(parent.interpreter_state.get()); |  | ||||||
| 
 |  | ||||||
|         bool is_thumb = (parent.interpreter_state->Cpsr & (1 << 5)) != 0; |  | ||||||
|         parent.interpreter_state->Reg[15] &= (is_thumb ? 0xFFFFFFFE : 0xFFFFFFFC); |  | ||||||
| 
 |  | ||||||
|         parent.jit->Regs() = parent.interpreter_state->Reg; |  | ||||||
|         parent.jit->SetCpsr(parent.interpreter_state->Cpsr); |  | ||||||
|         parent.jit->ExtRegs() = parent.interpreter_state->ExtReg; |  | ||||||
|         parent.jit->SetFpscr(parent.interpreter_state->VFP[VFP_FPSCR]); |  | ||||||
| 
 |  | ||||||
|         parent.interpreter_state->ServeBreak(); |  | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|     void CallSVC(std::uint32_t swi) override { |     void CallSVC(std::uint32_t swi) override { | ||||||
|  | @ -135,14 +119,18 @@ public: | ||||||
|             if (GDBStub::IsConnected()) { |             if (GDBStub::IsConnected()) { | ||||||
|                 parent.jit->HaltExecution(); |                 parent.jit->HaltExecution(); | ||||||
|                 parent.SetPC(pc); |                 parent.SetPC(pc); | ||||||
|                 Kernel::Thread* thread = |                 parent.ServeBreak(); | ||||||
|                     parent.system.Kernel().GetCurrentThreadManager().GetCurrentThread(); |  | ||||||
|                 parent.SaveContext(thread->context); |  | ||||||
|                 GDBStub::Break(); |  | ||||||
|                 GDBStub::SendTrap(thread, 5); |  | ||||||
|                 return; |                 return; | ||||||
|             } |             } | ||||||
|             break; |             break; | ||||||
|  |         case Dynarmic::A32::Exception::SendEvent: | ||||||
|  |         case Dynarmic::A32::Exception::SendEventLocal: | ||||||
|  |         case Dynarmic::A32::Exception::WaitForInterrupt: | ||||||
|  |         case Dynarmic::A32::Exception::WaitForEvent: | ||||||
|  |         case Dynarmic::A32::Exception::Yield: | ||||||
|  |         case Dynarmic::A32::Exception::PreloadData: | ||||||
|  |         case Dynarmic::A32::Exception::PreloadDataWithIntentToWrite: | ||||||
|  |             return; | ||||||
|         } |         } | ||||||
|         ASSERT_MSG(false, "ExceptionRaised(exception = {}, pc = {:08X}, code = {:08X})", |         ASSERT_MSG(false, "ExceptionRaised(exception = {}, pc = {:08X}, code = {:08X})", | ||||||
|                    static_cast<std::size_t>(exception), pc, MemoryReadCode(pc)); |                    static_cast<std::size_t>(exception), pc, MemoryReadCode(pc)); | ||||||
|  | @ -161,12 +149,10 @@ public: | ||||||
|     Memory::MemorySystem& memory; |     Memory::MemorySystem& memory; | ||||||
| }; | }; | ||||||
| 
 | 
 | ||||||
| ARM_Dynarmic::ARM_Dynarmic(Core::System* system, Memory::MemorySystem& memory, | ARM_Dynarmic::ARM_Dynarmic(Core::System* system, Memory::MemorySystem& memory, u32 id, | ||||||
|                            PrivilegeMode initial_mode, u32 id, |  | ||||||
|                            std::shared_ptr<Core::Timing::Timer> timer) |                            std::shared_ptr<Core::Timing::Timer> timer) | ||||||
|     : ARM_Interface(id, timer), system(*system), memory(memory), |     : ARM_Interface(id, timer), system(*system), memory(memory), | ||||||
|       cb(std::make_unique<DynarmicUserCallbacks>(*this)) { |       cb(std::make_unique<DynarmicUserCallbacks>(*this)) { | ||||||
|     interpreter_state = std::make_shared<ARMul_State>(system, memory, initial_mode); |  | ||||||
|     SetPageTable(memory.GetCurrentPageTable()); |     SetPageTable(memory.GetCurrentPageTable()); | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
|  | @ -182,7 +168,11 @@ void ARM_Dynarmic::Run() { | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| void ARM_Dynarmic::Step() { | void ARM_Dynarmic::Step() { | ||||||
|     cb->InterpreterFallback(jit->Regs()[15], 1); |     jit->Step(); | ||||||
|  | 
 | ||||||
|  |     if (GDBStub::IsConnected()) { | ||||||
|  |         ServeBreak(); | ||||||
|  |     } | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| void ARM_Dynarmic::SetPC(u32 pc) { | void ARM_Dynarmic::SetPC(u32 pc) { | ||||||
|  | @ -210,21 +200,25 @@ void ARM_Dynarmic::SetVFPReg(int index, u32 value) { | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| u32 ARM_Dynarmic::GetVFPSystemReg(VFPSystemRegister reg) const { | u32 ARM_Dynarmic::GetVFPSystemReg(VFPSystemRegister reg) const { | ||||||
|     if (reg == VFP_FPSCR) { |     switch (reg) { | ||||||
|  |     case VFP_FPSCR: | ||||||
|         return jit->Fpscr(); |         return jit->Fpscr(); | ||||||
|  |     case VFP_FPEXC: | ||||||
|  |         return fpexc; | ||||||
|     } |     } | ||||||
| 
 |     UNREACHABLE_MSG("Unknown VFP system register: {}", static_cast<size_t>(reg)); | ||||||
|     // Dynarmic does not implement and/or expose other VFP registers, fallback to interpreter state
 |  | ||||||
|     return interpreter_state->VFP[reg]; |  | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| void ARM_Dynarmic::SetVFPSystemReg(VFPSystemRegister reg, u32 value) { | void ARM_Dynarmic::SetVFPSystemReg(VFPSystemRegister reg, u32 value) { | ||||||
|     if (reg == VFP_FPSCR) { |     switch (reg) { | ||||||
|  |     case VFP_FPSCR: | ||||||
|         jit->SetFpscr(value); |         jit->SetFpscr(value); | ||||||
|  |         return; | ||||||
|  |     case VFP_FPEXC: | ||||||
|  |         fpexc = value; | ||||||
|  |         return; | ||||||
|     } |     } | ||||||
| 
 |     UNREACHABLE_MSG("Unknown VFP system register: {}", static_cast<size_t>(reg)); | ||||||
|     // Dynarmic does not implement and/or expose other VFP registers, fallback to interpreter state
 |  | ||||||
|     interpreter_state->VFP[reg] = value; |  | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| u32 ARM_Dynarmic::GetCPSR() const { | u32 ARM_Dynarmic::GetCPSR() const { | ||||||
|  | @ -236,11 +230,25 @@ void ARM_Dynarmic::SetCPSR(u32 cpsr) { | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| u32 ARM_Dynarmic::GetCP15Register(CP15Register reg) const { | u32 ARM_Dynarmic::GetCP15Register(CP15Register reg) const { | ||||||
|     return interpreter_state->CP15[reg]; |     switch (reg) { | ||||||
|  |     case CP15_THREAD_UPRW: | ||||||
|  |         return cp15_state.cp15_thread_uprw; | ||||||
|  |     case CP15_THREAD_URO: | ||||||
|  |         return cp15_state.cp15_thread_uro; | ||||||
|  |     } | ||||||
|  |     UNREACHABLE_MSG("Unknown CP15 register: {}", static_cast<size_t>(reg)); | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| void ARM_Dynarmic::SetCP15Register(CP15Register reg, u32 value) { | void ARM_Dynarmic::SetCP15Register(CP15Register reg, u32 value) { | ||||||
|     interpreter_state->CP15[reg] = value; |     switch (reg) { | ||||||
|  |     case CP15_THREAD_UPRW: | ||||||
|  |         cp15_state.cp15_thread_uprw = value; | ||||||
|  |         return; | ||||||
|  |     case CP15_THREAD_URO: | ||||||
|  |         cp15_state.cp15_thread_uro = value; | ||||||
|  |         return; | ||||||
|  |     } | ||||||
|  |     UNREACHABLE_MSG("Unknown CP15 register: {}", static_cast<size_t>(reg)); | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| std::unique_ptr<ARM_Interface::ThreadContext> ARM_Dynarmic::NewContext() const { | std::unique_ptr<ARM_Interface::ThreadContext> ARM_Dynarmic::NewContext() const { | ||||||
|  | @ -252,7 +260,7 @@ void ARM_Dynarmic::SaveContext(const std::unique_ptr<ThreadContext>& arg) { | ||||||
|     ASSERT(ctx); |     ASSERT(ctx); | ||||||
| 
 | 
 | ||||||
|     jit->SaveContext(ctx->ctx); |     jit->SaveContext(ctx->ctx); | ||||||
|     ctx->fpexc = interpreter_state->VFP[VFP_FPEXC]; |     ctx->fpexc = fpexc; | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| void ARM_Dynarmic::LoadContext(const std::unique_ptr<ThreadContext>& arg) { | void ARM_Dynarmic::LoadContext(const std::unique_ptr<ThreadContext>& arg) { | ||||||
|  | @ -260,7 +268,7 @@ void ARM_Dynarmic::LoadContext(const std::unique_ptr<ThreadContext>& arg) { | ||||||
|     ASSERT(ctx); |     ASSERT(ctx); | ||||||
| 
 | 
 | ||||||
|     jit->LoadContext(ctx->ctx); |     jit->LoadContext(ctx->ctx); | ||||||
|     interpreter_state->VFP[VFP_FPEXC] = ctx->fpexc; |     fpexc = ctx->fpexc; | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| void ARM_Dynarmic::PrepareReschedule() { | void ARM_Dynarmic::PrepareReschedule() { | ||||||
|  | @ -270,11 +278,9 @@ void ARM_Dynarmic::PrepareReschedule() { | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| void ARM_Dynarmic::ClearInstructionCache() { | void ARM_Dynarmic::ClearInstructionCache() { | ||||||
|     // TODO: Clear interpreter cache when appropriate.
 |  | ||||||
|     for (const auto& j : jits) { |     for (const auto& j : jits) { | ||||||
|         j.second->ClearCache(); |         j.second->ClearCache(); | ||||||
|     } |     } | ||||||
|     interpreter_state->instruction_cache.clear(); |  | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| void ARM_Dynarmic::InvalidateCacheRange(u32 start_address, std::size_t length) { | void ARM_Dynarmic::InvalidateCacheRange(u32 start_address, std::size_t length) { | ||||||
|  | @ -305,11 +311,18 @@ void ARM_Dynarmic::SetPageTable(const std::shared_ptr<Memory::PageTable>& page_t | ||||||
|     jits.emplace(current_page_table, std::move(new_jit)); |     jits.emplace(current_page_table, std::move(new_jit)); | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
|  | void ARM_Dynarmic::ServeBreak() { | ||||||
|  |     Kernel::Thread* thread = system.Kernel().GetCurrentThreadManager().GetCurrentThread(); | ||||||
|  |     SaveContext(thread->context); | ||||||
|  |     GDBStub::Break(); | ||||||
|  |     GDBStub::SendTrap(thread, 5); | ||||||
|  | } | ||||||
|  | 
 | ||||||
| std::unique_ptr<Dynarmic::A32::Jit> ARM_Dynarmic::MakeJit() { | std::unique_ptr<Dynarmic::A32::Jit> ARM_Dynarmic::MakeJit() { | ||||||
|     Dynarmic::A32::UserConfig config; |     Dynarmic::A32::UserConfig config; | ||||||
|     config.callbacks = cb.get(); |     config.callbacks = cb.get(); | ||||||
|     config.page_table = ¤t_page_table->GetPointerArray(); |     config.page_table = ¤t_page_table->GetPointerArray(); | ||||||
|     config.coprocessors[15] = std::make_shared<DynarmicCP15>(interpreter_state); |     config.coprocessors[15] = std::make_shared<DynarmicCP15>(cp15_state); | ||||||
|     config.define_unpredictable_behaviour = true; |     config.define_unpredictable_behaviour = true; | ||||||
|     return std::make_unique<Dynarmic::A32::Jit>(config); |     return std::make_unique<Dynarmic::A32::Jit>(config); | ||||||
| } | } | ||||||
|  |  | ||||||
|  | @ -9,7 +9,7 @@ | ||||||
| #include <dynarmic/A32/a32.h> | #include <dynarmic/A32/a32.h> | ||||||
| #include "common/common_types.h" | #include "common/common_types.h" | ||||||
| #include "core/arm/arm_interface.h" | #include "core/arm/arm_interface.h" | ||||||
| #include "core/arm/skyeye_common/armstate.h" | #include "core/arm/dynarmic/arm_dynarmic_cp15.h" | ||||||
| 
 | 
 | ||||||
| namespace Memory { | namespace Memory { | ||||||
| struct PageTable; | struct PageTable; | ||||||
|  | @ -24,8 +24,8 @@ class DynarmicUserCallbacks; | ||||||
| 
 | 
 | ||||||
| class ARM_Dynarmic final : public ARM_Interface { | class ARM_Dynarmic final : public ARM_Interface { | ||||||
| public: | public: | ||||||
|     ARM_Dynarmic(Core::System* system, Memory::MemorySystem& memory, PrivilegeMode initial_mode, |     ARM_Dynarmic(Core::System* system, Memory::MemorySystem& memory, u32 id, | ||||||
|                  u32 id, std::shared_ptr<Core::Timing::Timer> timer); |                  std::shared_ptr<Core::Timing::Timer> timer); | ||||||
|     ~ARM_Dynarmic() override; |     ~ARM_Dynarmic() override; | ||||||
| 
 | 
 | ||||||
|     void Run() override; |     void Run() override; | ||||||
|  | @ -59,14 +59,18 @@ protected: | ||||||
|     std::shared_ptr<Memory::PageTable> GetPageTable() const override; |     std::shared_ptr<Memory::PageTable> GetPageTable() const override; | ||||||
| 
 | 
 | ||||||
| private: | private: | ||||||
|  |     void ServeBreak(); | ||||||
|  | 
 | ||||||
|     friend class DynarmicUserCallbacks; |     friend class DynarmicUserCallbacks; | ||||||
|     Core::System& system; |     Core::System& system; | ||||||
|     Memory::MemorySystem& memory; |     Memory::MemorySystem& memory; | ||||||
|     std::unique_ptr<DynarmicUserCallbacks> cb; |     std::unique_ptr<DynarmicUserCallbacks> cb; | ||||||
|     std::unique_ptr<Dynarmic::A32::Jit> MakeJit(); |     std::unique_ptr<Dynarmic::A32::Jit> MakeJit(); | ||||||
| 
 | 
 | ||||||
|  |     u32 fpexc = 0; | ||||||
|  |     CP15State cp15_state; | ||||||
|  | 
 | ||||||
|     Dynarmic::A32::Jit* jit = nullptr; |     Dynarmic::A32::Jit* jit = nullptr; | ||||||
|     std::shared_ptr<Memory::PageTable> current_page_table = nullptr; |     std::shared_ptr<Memory::PageTable> current_page_table = nullptr; | ||||||
|     std::map<std::shared_ptr<Memory::PageTable>, std::unique_ptr<Dynarmic::A32::Jit>> jits; |     std::map<std::shared_ptr<Memory::PageTable>, std::unique_ptr<Dynarmic::A32::Jit>> jits; | ||||||
|     std::shared_ptr<ARMul_State> interpreter_state; |  | ||||||
| }; | }; | ||||||
|  |  | ||||||
|  | @ -10,14 +10,14 @@ using Callback = Dynarmic::A32::Coprocessor::Callback; | ||||||
| using CallbackOrAccessOneWord = Dynarmic::A32::Coprocessor::CallbackOrAccessOneWord; | using CallbackOrAccessOneWord = Dynarmic::A32::Coprocessor::CallbackOrAccessOneWord; | ||||||
| using CallbackOrAccessTwoWords = Dynarmic::A32::Coprocessor::CallbackOrAccessTwoWords; | using CallbackOrAccessTwoWords = Dynarmic::A32::Coprocessor::CallbackOrAccessTwoWords; | ||||||
| 
 | 
 | ||||||
| DynarmicCP15::DynarmicCP15(const std::shared_ptr<ARMul_State>& state) : interpreter_state(state) {} | DynarmicCP15::DynarmicCP15(CP15State& state) : state(state) {} | ||||||
| 
 | 
 | ||||||
| DynarmicCP15::~DynarmicCP15() = default; | DynarmicCP15::~DynarmicCP15() = default; | ||||||
| 
 | 
 | ||||||
| boost::optional<Callback> DynarmicCP15::CompileInternalOperation(bool two, unsigned opc1, | std::optional<Callback> DynarmicCP15::CompileInternalOperation(bool two, unsigned opc1, | ||||||
|                                                                CoprocReg CRd, CoprocReg CRn, |                                                                CoprocReg CRd, CoprocReg CRn, | ||||||
|                                                                CoprocReg CRm, unsigned opc2) { |                                                                CoprocReg CRm, unsigned opc2) { | ||||||
|     return boost::none; |     return std::nullopt; | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| CallbackOrAccessOneWord DynarmicCP15::CompileSendOneWord(bool two, unsigned opc1, CoprocReg CRn, | CallbackOrAccessOneWord DynarmicCP15::CompileSendOneWord(bool two, unsigned opc1, CoprocReg CRn, | ||||||
|  | @ -26,31 +26,31 @@ CallbackOrAccessOneWord DynarmicCP15::CompileSendOneWord(bool two, unsigned opc1 | ||||||
| 
 | 
 | ||||||
|     if (!two && CRn == CoprocReg::C7 && opc1 == 0 && CRm == CoprocReg::C5 && opc2 == 4) { |     if (!two && CRn == CoprocReg::C7 && opc1 == 0 && CRm == CoprocReg::C5 && opc2 == 4) { | ||||||
|         // This is a dummy write, we ignore the value written here.
 |         // This is a dummy write, we ignore the value written here.
 | ||||||
|         return &interpreter_state->CP15[CP15_FLUSH_PREFETCH_BUFFER]; |         return &state.cp15_flush_prefetch_buffer; | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|     if (!two && CRn == CoprocReg::C7 && opc1 == 0 && CRm == CoprocReg::C10) { |     if (!two && CRn == CoprocReg::C7 && opc1 == 0 && CRm == CoprocReg::C10) { | ||||||
|         switch (opc2) { |         switch (opc2) { | ||||||
|         case 4: |         case 4: | ||||||
|             // This is a dummy write, we ignore the value written here.
 |             // This is a dummy write, we ignore the value written here.
 | ||||||
|             return &interpreter_state->CP15[CP15_DATA_SYNC_BARRIER]; |             return &state.cp15_data_sync_barrier; | ||||||
|         case 5: |         case 5: | ||||||
|             // This is a dummy write, we ignore the value written here.
 |             // This is a dummy write, we ignore the value written here.
 | ||||||
|             return &interpreter_state->CP15[CP15_DATA_MEMORY_BARRIER]; |             return &state.cp15_data_memory_barrier; | ||||||
|         default: |         default: | ||||||
|             return boost::blank{}; |             return std::monostate{}; | ||||||
|         } |         } | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|     if (!two && CRn == CoprocReg::C13 && opc1 == 0 && CRm == CoprocReg::C0 && opc2 == 2) { |     if (!two && CRn == CoprocReg::C13 && opc1 == 0 && CRm == CoprocReg::C0 && opc2 == 2) { | ||||||
|         return &interpreter_state->CP15[CP15_THREAD_UPRW]; |         return &state.cp15_thread_uprw; | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|     return boost::blank{}; |     return std::monostate{}; | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| CallbackOrAccessTwoWords DynarmicCP15::CompileSendTwoWords(bool two, unsigned opc, CoprocReg CRm) { | CallbackOrAccessTwoWords DynarmicCP15::CompileSendTwoWords(bool two, unsigned opc, CoprocReg CRm) { | ||||||
|     return boost::blank{}; |     return std::monostate{}; | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| CallbackOrAccessOneWord DynarmicCP15::CompileGetOneWord(bool two, unsigned opc1, CoprocReg CRn, | CallbackOrAccessOneWord DynarmicCP15::CompileGetOneWord(bool two, unsigned opc1, CoprocReg CRn, | ||||||
|  | @ -60,29 +60,27 @@ CallbackOrAccessOneWord DynarmicCP15::CompileGetOneWord(bool two, unsigned opc1, | ||||||
|     if (!two && CRn == CoprocReg::C13 && opc1 == 0 && CRm == CoprocReg::C0) { |     if (!two && CRn == CoprocReg::C13 && opc1 == 0 && CRm == CoprocReg::C0) { | ||||||
|         switch (opc2) { |         switch (opc2) { | ||||||
|         case 2: |         case 2: | ||||||
|             return &interpreter_state->CP15[CP15_THREAD_UPRW]; |             return &state.cp15_thread_uprw; | ||||||
|         case 3: |         case 3: | ||||||
|             return &interpreter_state->CP15[CP15_THREAD_URO]; |             return &state.cp15_thread_uro; | ||||||
|         default: |         default: | ||||||
|             return boost::blank{}; |             return std::monostate{}; | ||||||
|         } |         } | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|     return boost::blank{}; |     return std::monostate{}; | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| CallbackOrAccessTwoWords DynarmicCP15::CompileGetTwoWords(bool two, unsigned opc, CoprocReg CRm) { | CallbackOrAccessTwoWords DynarmicCP15::CompileGetTwoWords(bool two, unsigned opc, CoprocReg CRm) { | ||||||
|     return boost::blank{}; |     return std::monostate{}; | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| boost::optional<Callback> DynarmicCP15::CompileLoadWords(bool two, bool long_transfer, | std::optional<Callback> DynarmicCP15::CompileLoadWords(bool two, bool long_transfer, CoprocReg CRd, | ||||||
|                                                          CoprocReg CRd, |                                                        std::optional<u8> option) { | ||||||
|                                                          boost::optional<u8> option) { |     return std::nullopt; | ||||||
|     return boost::none; |  | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| boost::optional<Callback> DynarmicCP15::CompileStoreWords(bool two, bool long_transfer, | std::optional<Callback> DynarmicCP15::CompileStoreWords(bool two, bool long_transfer, CoprocReg CRd, | ||||||
|                                                           CoprocReg CRd, |                                                         std::optional<u8> option) { | ||||||
|                                                           boost::optional<u8> option) { |     return std::nullopt; | ||||||
|     return boost::none; |  | ||||||
| } | } | ||||||
|  |  | ||||||
|  | @ -8,16 +8,22 @@ | ||||||
| #include <dynarmic/A32/coprocessor.h> | #include <dynarmic/A32/coprocessor.h> | ||||||
| #include "common/common_types.h" | #include "common/common_types.h" | ||||||
| 
 | 
 | ||||||
| struct ARMul_State; | struct CP15State { | ||||||
|  |     u32 cp15_thread_uprw = 0; | ||||||
|  |     u32 cp15_thread_uro = 0; | ||||||
|  |     u32 cp15_flush_prefetch_buffer = 0; ///< dummy value
 | ||||||
|  |     u32 cp15_data_sync_barrier = 0;     ///< dummy value
 | ||||||
|  |     u32 cp15_data_memory_barrier = 0;   ///< dummy value
 | ||||||
|  | }; | ||||||
| 
 | 
 | ||||||
| class DynarmicCP15 final : public Dynarmic::A32::Coprocessor { | class DynarmicCP15 final : public Dynarmic::A32::Coprocessor { | ||||||
| public: | public: | ||||||
|     using CoprocReg = Dynarmic::A32::CoprocReg; |     using CoprocReg = Dynarmic::A32::CoprocReg; | ||||||
| 
 | 
 | ||||||
|     explicit DynarmicCP15(const std::shared_ptr<ARMul_State>&); |     explicit DynarmicCP15(CP15State&); | ||||||
|     ~DynarmicCP15() override; |     ~DynarmicCP15() override; | ||||||
| 
 | 
 | ||||||
|     boost::optional<Callback> CompileInternalOperation(bool two, unsigned opc1, CoprocReg CRd, |     std::optional<Callback> CompileInternalOperation(bool two, unsigned opc1, CoprocReg CRd, | ||||||
|                                                      CoprocReg CRn, CoprocReg CRm, |                                                      CoprocReg CRn, CoprocReg CRm, | ||||||
|                                                      unsigned opc2) override; |                                                      unsigned opc2) override; | ||||||
|     CallbackOrAccessOneWord CompileSendOneWord(bool two, unsigned opc1, CoprocReg CRn, |     CallbackOrAccessOneWord CompileSendOneWord(bool two, unsigned opc1, CoprocReg CRn, | ||||||
|  | @ -26,11 +32,11 @@ public: | ||||||
|     CallbackOrAccessOneWord CompileGetOneWord(bool two, unsigned opc1, CoprocReg CRn, CoprocReg CRm, |     CallbackOrAccessOneWord CompileGetOneWord(bool two, unsigned opc1, CoprocReg CRn, CoprocReg CRm, | ||||||
|                                               unsigned opc2) override; |                                               unsigned opc2) override; | ||||||
|     CallbackOrAccessTwoWords CompileGetTwoWords(bool two, unsigned opc, CoprocReg CRm) override; |     CallbackOrAccessTwoWords CompileGetTwoWords(bool two, unsigned opc, CoprocReg CRm) override; | ||||||
|     boost::optional<Callback> CompileLoadWords(bool two, bool long_transfer, CoprocReg CRd, |     std::optional<Callback> CompileLoadWords(bool two, bool long_transfer, CoprocReg CRd, | ||||||
|                                                boost::optional<u8> option) override; |                                              std::optional<u8> option) override; | ||||||
|     boost::optional<Callback> CompileStoreWords(bool two, bool long_transfer, CoprocReg CRd, |     std::optional<Callback> CompileStoreWords(bool two, bool long_transfer, CoprocReg CRd, | ||||||
|                                                 boost::optional<u8> option) override; |                                               std::optional<u8> option) override; | ||||||
| 
 | 
 | ||||||
| private: | private: | ||||||
|     std::shared_ptr<ARMul_State> interpreter_state; |     CP15State& state; | ||||||
| }; | }; | ||||||
|  |  | ||||||
|  | @ -242,7 +242,11 @@ System::ResultStatus System::Load(Frontend::EmuWindow& emu_window, const std::st | ||||||
|     ASSERT(system_mode.first); |     ASSERT(system_mode.first); | ||||||
|     auto n3ds_mode = app_loader->LoadKernelN3dsMode(); |     auto n3ds_mode = app_loader->LoadKernelN3dsMode(); | ||||||
|     ASSERT(n3ds_mode.first); |     ASSERT(n3ds_mode.first); | ||||||
|     ResultStatus init_result{Init(emu_window, *system_mode.first, *n3ds_mode.first)}; |     u32 num_cores = 2; | ||||||
|  |     if (Settings::values.is_new_3ds) { | ||||||
|  |         num_cores = 4; | ||||||
|  |     } | ||||||
|  |     ResultStatus init_result{Init(emu_window, *system_mode.first, *n3ds_mode.first, num_cores)}; | ||||||
|     if (init_result != ResultStatus::Success) { |     if (init_result != ResultStatus::Success) { | ||||||
|         LOG_CRITICAL(Core, "Failed to initialize system (Error {})!", |         LOG_CRITICAL(Core, "Failed to initialize system (Error {})!", | ||||||
|                      static_cast<u32>(init_result)); |                      static_cast<u32>(init_result)); | ||||||
|  | @ -315,14 +319,10 @@ void System::Reschedule() { | ||||||
|     } |     } | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| System::ResultStatus System::Init(Frontend::EmuWindow& emu_window, u32 system_mode, u8 n3ds_mode) { | System::ResultStatus System::Init(Frontend::EmuWindow& emu_window, u32 system_mode, u8 n3ds_mode, | ||||||
|  |                                   u32 num_cores) { | ||||||
|     LOG_DEBUG(HW_Memory, "initialized OK"); |     LOG_DEBUG(HW_Memory, "initialized OK"); | ||||||
| 
 | 
 | ||||||
|     u32 num_cores = 2; |  | ||||||
|     if (Settings::values.is_new_3ds) { |  | ||||||
|         num_cores = 4; |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     memory = std::make_unique<Memory::MemorySystem>(); |     memory = std::make_unique<Memory::MemorySystem>(); | ||||||
| 
 | 
 | ||||||
|     timing = std::make_unique<Timing>(num_cores, Settings::values.cpu_clock_percentage); |     timing = std::make_unique<Timing>(num_cores, Settings::values.cpu_clock_percentage); | ||||||
|  | @ -334,7 +334,7 @@ System::ResultStatus System::Init(Frontend::EmuWindow& emu_window, u32 system_mo | ||||||
| #ifdef ARCHITECTURE_x86_64 | #ifdef ARCHITECTURE_x86_64 | ||||||
|         for (u32 i = 0; i < num_cores; ++i) { |         for (u32 i = 0; i < num_cores; ++i) { | ||||||
|             cpu_cores.push_back( |             cpu_cores.push_back( | ||||||
|                 std::make_shared<ARM_Dynarmic>(this, *memory, USER32MODE, i, timing->GetTimer(i))); |                 std::make_shared<ARM_Dynarmic>(this, *memory, i, timing->GetTimer(i))); | ||||||
|         } |         } | ||||||
| #else | #else | ||||||
|         for (u32 i = 0; i < num_cores; ++i) { |         for (u32 i = 0; i < num_cores; ++i) { | ||||||
|  | @ -536,6 +536,13 @@ void System::Reset() { | ||||||
| 
 | 
 | ||||||
| template <class Archive> | template <class Archive> | ||||||
| void System::serialize(Archive& ar, const unsigned int file_version) { | void System::serialize(Archive& ar, const unsigned int file_version) { | ||||||
|  | 
 | ||||||
|  |     u32 num_cores; | ||||||
|  |     if (Archive::is_saving::value) { | ||||||
|  |         num_cores = this->GetNumCores(); | ||||||
|  |     } | ||||||
|  |     ar& num_cores; | ||||||
|  | 
 | ||||||
|     if (Archive::is_loading::value) { |     if (Archive::is_loading::value) { | ||||||
|         // When loading, we want to make sure any lingering state gets cleared out before we begin.
 |         // When loading, we want to make sure any lingering state gets cleared out before we begin.
 | ||||||
|         // Shutdown, but persist a few things between loads...
 |         // Shutdown, but persist a few things between loads...
 | ||||||
|  | @ -544,17 +551,9 @@ void System::serialize(Archive& ar, const unsigned int file_version) { | ||||||
|         // Re-initialize everything like it was before
 |         // Re-initialize everything like it was before
 | ||||||
|         auto system_mode = this->app_loader->LoadKernelSystemMode(); |         auto system_mode = this->app_loader->LoadKernelSystemMode(); | ||||||
|         auto n3ds_mode = this->app_loader->LoadKernelN3dsMode(); |         auto n3ds_mode = this->app_loader->LoadKernelN3dsMode(); | ||||||
|         Init(*m_emu_window, *system_mode.first, *n3ds_mode.first); |         Init(*m_emu_window, *system_mode.first, *n3ds_mode.first, num_cores); | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|     u32 num_cores; |  | ||||||
|     if (Archive::is_saving::value) { |  | ||||||
|         num_cores = this->GetNumCores(); |  | ||||||
|     } |  | ||||||
|     ar& num_cores; |  | ||||||
|     if (num_cores != this->GetNumCores()) { |  | ||||||
|         throw std::runtime_error("Wrong N3DS mode"); |  | ||||||
|     } |  | ||||||
|     // flush on save, don't flush on load
 |     // flush on save, don't flush on load
 | ||||||
|     bool should_flush = !Archive::is_loading::value; |     bool should_flush = !Archive::is_loading::value; | ||||||
|     Memory::RasterizerClearAll(should_flush); |     Memory::RasterizerClearAll(should_flush); | ||||||
|  |  | ||||||
|  | @ -315,7 +315,7 @@ private: | ||||||
|      * @param system_mode The system mode. |      * @param system_mode The system mode. | ||||||
|      * @return ResultStatus code, indicating if the operation succeeded. |      * @return ResultStatus code, indicating if the operation succeeded. | ||||||
|      */ |      */ | ||||||
|     ResultStatus Init(Frontend::EmuWindow& emu_window, u32 system_mode, u8 n3ds_mode); |     ResultStatus Init(Frontend::EmuWindow& emu_window, u32 system_mode, u8 n3ds_mode, u32 num_cores); | ||||||
| 
 | 
 | ||||||
|     /// Reschedule the core emulation
 |     /// Reschedule the core emulation
 | ||||||
|     void Reschedule(); |     void Reschedule(); | ||||||
|  |  | ||||||
		Loading…
	
	Add table
		Add a link
		
	
		Reference in a new issue