| 
									
										
										
										
											2018-05-01 22:21:38 -04:00
										 |  |  | // Copyright 2018 yuzu emulator team
 | 
					
						
							|  |  |  | // Licensed under GPLv2 or any later version
 | 
					
						
							|  |  |  | // Refer to the license.txt file included.
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-05-02 21:26:14 -04:00
										 |  |  | #include <condition_variable>
 | 
					
						
							|  |  |  | #include <mutex>
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-05-01 22:21:38 -04:00
										 |  |  | #include "common/logging/log.h"
 | 
					
						
							|  |  |  | #ifdef ARCHITECTURE_x86_64
 | 
					
						
							|  |  |  | #include "core/arm/dynarmic/arm_dynarmic.h"
 | 
					
						
							|  |  |  | #endif
 | 
					
						
							| 
									
										
										
										
											2018-09-17 18:15:09 -04:00
										 |  |  | #include "core/arm/exclusive_monitor.h"
 | 
					
						
							| 
									
										
										
										
											2018-05-01 22:21:38 -04:00
										 |  |  | #include "core/arm/unicorn/arm_unicorn.h"
 | 
					
						
							| 
									
										
										
										
											2019-03-04 16:02:59 -05:00
										 |  |  | #include "core/core.h"
 | 
					
						
							| 
									
										
										
										
											2018-05-01 22:21:38 -04:00
										 |  |  | #include "core/core_cpu.h"
 | 
					
						
							|  |  |  | #include "core/core_timing.h"
 | 
					
						
							|  |  |  | #include "core/hle/kernel/scheduler.h"
 | 
					
						
							|  |  |  | #include "core/hle/kernel/thread.h"
 | 
					
						
							| 
									
										
										
										
											2018-08-12 17:50:44 -05:00
										 |  |  | #include "core/hle/lock.h"
 | 
					
						
							| 
									
										
										
										
											2018-05-01 22:21:38 -04:00
										 |  |  | #include "core/settings.h"
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | namespace Core { | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-05-03 00:16:12 -04:00
										 |  |  | void CpuBarrier::NotifyEnd() { | 
					
						
							| 
									
										
										
										
											2019-04-01 12:29:59 -04:00
										 |  |  |     std::unique_lock lock{mutex}; | 
					
						
							| 
									
										
										
										
											2018-05-03 00:16:12 -04:00
										 |  |  |     end = true; | 
					
						
							|  |  |  |     condition.notify_all(); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | bool CpuBarrier::Rendezvous() { | 
					
						
							| 
									
										
										
										
											2018-05-03 00:34:54 -04:00
										 |  |  |     if (!Settings::values.use_multi_core) { | 
					
						
							|  |  |  |         // Meaningless when running in single-core mode
 | 
					
						
							|  |  |  |         return true; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (!end) { | 
					
						
							| 
									
										
										
										
											2019-04-01 12:29:59 -04:00
										 |  |  |         std::unique_lock lock{mutex}; | 
					
						
							| 
									
										
										
										
											2018-05-03 00:16:12 -04:00
										 |  |  | 
 | 
					
						
							|  |  |  |         --cores_waiting; | 
					
						
							|  |  |  |         if (!cores_waiting) { | 
					
						
							|  |  |  |             cores_waiting = NUM_CPU_CORES; | 
					
						
							|  |  |  |             condition.notify_all(); | 
					
						
							|  |  |  |             return true; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         condition.wait(lock); | 
					
						
							|  |  |  |         return true; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2018-05-03 00:34:54 -04:00
										 |  |  | 
 | 
					
						
							|  |  |  |     return false; | 
					
						
							| 
									
										
										
										
											2018-05-03 00:16:12 -04:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-03-04 16:02:59 -05:00
										 |  |  | Cpu::Cpu(System& system, ExclusiveMonitor& exclusive_monitor, CpuBarrier& cpu_barrier, | 
					
						
							|  |  |  |          std::size_t core_index) | 
					
						
							| 
									
										
										
										
											2019-03-29 17:09:10 -04:00
										 |  |  |     : cpu_barrier{cpu_barrier}, global_scheduler{system.GlobalScheduler()}, | 
					
						
							|  |  |  |       core_timing{system.CoreTiming()}, core_index{core_index} { | 
					
						
							| 
									
										
										
										
											2018-05-01 22:21:38 -04:00
										 |  |  | #ifdef ARCHITECTURE_x86_64
 | 
					
						
							| 
									
										
										
										
											2019-07-11 05:52:38 -04:00
										 |  |  |     arm_interface = std::make_unique<ARM_Dynarmic>(system, exclusive_monitor, core_index); | 
					
						
							| 
									
										
										
										
											2018-05-01 22:21:38 -04:00
										 |  |  | #else
 | 
					
						
							| 
									
										
										
										
											2019-07-11 05:52:38 -04:00
										 |  |  |     arm_interface = std::make_unique<ARM_Unicorn>(system); | 
					
						
							|  |  |  |     LOG_WARNING(Core, "CPU JIT requested, but Dynarmic not available"); | 
					
						
							| 
									
										
										
										
											2018-05-01 22:21:38 -04:00
										 |  |  | #endif
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-03-29 17:09:10 -04:00
										 |  |  |     scheduler = std::make_unique<Kernel::Scheduler>(system, *arm_interface, core_index); | 
					
						
							| 
									
										
										
										
											2018-05-01 22:21:38 -04:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-09-17 18:15:09 -04:00
										 |  |  | Cpu::~Cpu() = default; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-11-26 17:39:57 -05:00
										 |  |  | std::unique_ptr<ExclusiveMonitor> Cpu::MakeExclusiveMonitor( | 
					
						
							|  |  |  |     [[maybe_unused]] Memory::Memory& memory, [[maybe_unused]] std::size_t num_cores) { | 
					
						
							| 
									
										
										
										
											2018-07-03 14:28:46 +01:00
										 |  |  | #ifdef ARCHITECTURE_x86_64
 | 
					
						
							| 
									
										
										
										
											2019-11-26 17:39:57 -05:00
										 |  |  |     return std::make_unique<DynarmicExclusiveMonitor>(memory, num_cores); | 
					
						
							| 
									
										
										
										
											2018-07-03 14:28:46 +01:00
										 |  |  | #else
 | 
					
						
							| 
									
										
										
										
											2019-07-11 05:52:38 -04:00
										 |  |  |     // TODO(merry): Passthrough exclusive monitor
 | 
					
						
							|  |  |  |     return nullptr; | 
					
						
							| 
									
										
										
										
											2018-07-03 14:28:46 +01:00
										 |  |  | #endif
 | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-05-01 22:21:38 -04:00
										 |  |  | void Cpu::RunLoop(bool tight_loop) { | 
					
						
							| 
									
										
										
										
											2018-05-02 21:26:14 -04:00
										 |  |  |     // Wait for all other CPU cores to complete the previous slice, such that they run in lock-step
 | 
					
						
							| 
									
										
										
										
											2018-10-15 08:42:06 -04:00
										 |  |  |     if (!cpu_barrier.Rendezvous()) { | 
					
						
							| 
									
										
										
										
											2018-05-03 00:16:12 -04:00
										 |  |  |         // If rendezvous failed, session has been killed
 | 
					
						
							|  |  |  |         return; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2018-05-02 21:26:14 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-03-29 17:09:10 -04:00
										 |  |  |     Reschedule(); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-05-01 22:21:38 -04:00
										 |  |  |     // If we don't have a currently active thread then don't execute instructions,
 | 
					
						
							|  |  |  |     // instead advance to the next event and try to yield to the next thread
 | 
					
						
							|  |  |  |     if (Kernel::GetCurrentThread() == nullptr) { | 
					
						
							| 
									
										
										
										
											2018-07-02 10:13:26 -06:00
										 |  |  |         LOG_TRACE(Core, "Core-{} idling", core_index); | 
					
						
							| 
									
										
										
										
											2019-09-09 21:37:29 -04:00
										 |  |  |         core_timing.Idle(); | 
					
						
							| 
									
										
										
										
											2018-05-01 22:21:38 -04:00
										 |  |  |     } else { | 
					
						
							|  |  |  |         if (tight_loop) { | 
					
						
							|  |  |  |             arm_interface->Run(); | 
					
						
							|  |  |  |         } else { | 
					
						
							|  |  |  |             arm_interface->Step(); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2019-03-29 17:09:10 -04:00
										 |  |  |     core_timing.Advance(); | 
					
						
							| 
									
										
										
										
											2018-05-01 22:21:38 -04:00
										 |  |  | 
 | 
					
						
							|  |  |  |     Reschedule(); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void Cpu::SingleStep() { | 
					
						
							|  |  |  |     return RunLoop(false); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void Cpu::PrepareReschedule() { | 
					
						
							|  |  |  |     arm_interface->PrepareReschedule(); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void Cpu::Reschedule() { | 
					
						
							| 
									
										
										
										
											2018-08-12 17:50:44 -05:00
										 |  |  |     // Lock the global kernel mutex when we manipulate the HLE state
 | 
					
						
							| 
									
										
										
										
											2019-06-19 09:11:18 -04:00
										 |  |  |     std::lock_guard lock(HLE::g_hle_lock); | 
					
						
							| 
									
										
										
										
											2019-03-29 17:09:10 -04:00
										 |  |  | 
 | 
					
						
							|  |  |  |     global_scheduler.SelectThread(core_index); | 
					
						
							|  |  |  |     scheduler->TryDoContextSwitch(); | 
					
						
							| 
									
										
										
										
											2018-05-01 22:21:38 -04:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-10-12 08:21:51 -04:00
										 |  |  | void Cpu::Shutdown() { | 
					
						
							|  |  |  |     scheduler->Shutdown(); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-05-01 22:21:38 -04:00
										 |  |  | } // namespace Core
 |