mirror of
				https://git.suyu.dev/suyu/suyu
				synced 2025-10-30 15:39:02 -05:00 
			
		
		
		
	Merge pull request #3353 from FernandoS27/aries
System: Refactor CPU Core management and move ARMInterface and Schedulers to Kernel
This commit is contained in:
		| @@ -8,7 +8,6 @@ | ||||
| #include "common/assert.h" | ||||
| #include "common/common_types.h" | ||||
| #include "core/core.h" | ||||
| #include "core/core_cpu.h" | ||||
| #include "core/hle/kernel/address_arbiter.h" | ||||
| #include "core/hle/kernel/errors.h" | ||||
| #include "core/hle/kernel/scheduler.h" | ||||
|   | ||||
| @@ -3,13 +3,15 @@ | ||||
| // Refer to the license.txt file included. | ||||
|  | ||||
| #include <atomic> | ||||
| #include <functional> | ||||
| #include <memory> | ||||
| #include <mutex> | ||||
| #include <utility> | ||||
|  | ||||
| #include "common/assert.h" | ||||
| #include "common/logging/log.h" | ||||
|  | ||||
| #include "core/arm/arm_interface.h" | ||||
| #include "core/arm/exclusive_monitor.h" | ||||
| #include "core/core.h" | ||||
| #include "core/core_timing.h" | ||||
| #include "core/core_timing_util.h" | ||||
| @@ -17,6 +19,7 @@ | ||||
| #include "core/hle/kernel/errors.h" | ||||
| #include "core/hle/kernel/handle_table.h" | ||||
| #include "core/hle/kernel/kernel.h" | ||||
| #include "core/hle/kernel/physical_core.h" | ||||
| #include "core/hle/kernel/process.h" | ||||
| #include "core/hle/kernel/resource_limit.h" | ||||
| #include "core/hle/kernel/scheduler.h" | ||||
| @@ -98,6 +101,7 @@ struct KernelCore::Impl { | ||||
|     void Initialize(KernelCore& kernel) { | ||||
|         Shutdown(); | ||||
|  | ||||
|         InitializePhysicalCores(kernel); | ||||
|         InitializeSystemResourceLimit(kernel); | ||||
|         InitializeThreads(); | ||||
|         InitializePreemption(); | ||||
| @@ -121,6 +125,21 @@ struct KernelCore::Impl { | ||||
|         global_scheduler.Shutdown(); | ||||
|  | ||||
|         named_ports.clear(); | ||||
|  | ||||
|         for (auto& core : cores) { | ||||
|             core.Shutdown(); | ||||
|         } | ||||
|         cores.clear(); | ||||
|  | ||||
|         exclusive_monitor.reset(nullptr); | ||||
|     } | ||||
|  | ||||
|     void InitializePhysicalCores(KernelCore& kernel) { | ||||
|         exclusive_monitor = | ||||
|             Core::MakeExclusiveMonitor(system.Memory(), global_scheduler.CpuCoresCount()); | ||||
|         for (std::size_t i = 0; i < global_scheduler.CpuCoresCount(); i++) { | ||||
|             cores.emplace_back(system, kernel, i, *exclusive_monitor); | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     // Creates the default system resource limit | ||||
| @@ -186,6 +205,9 @@ struct KernelCore::Impl { | ||||
|     /// the ConnectToPort SVC. | ||||
|     NamedPortTable named_ports; | ||||
|  | ||||
|     std::unique_ptr<Core::ExclusiveMonitor> exclusive_monitor; | ||||
|     std::vector<Kernel::PhysicalCore> cores; | ||||
|  | ||||
|     // System context | ||||
|     Core::System& system; | ||||
| }; | ||||
| @@ -240,6 +262,34 @@ const Kernel::GlobalScheduler& KernelCore::GlobalScheduler() const { | ||||
|     return impl->global_scheduler; | ||||
| } | ||||
|  | ||||
| Kernel::PhysicalCore& KernelCore::PhysicalCore(std::size_t id) { | ||||
|     return impl->cores[id]; | ||||
| } | ||||
|  | ||||
| const Kernel::PhysicalCore& KernelCore::PhysicalCore(std::size_t id) const { | ||||
|     return impl->cores[id]; | ||||
| } | ||||
|  | ||||
| Core::ExclusiveMonitor& KernelCore::GetExclusiveMonitor() { | ||||
|     return *impl->exclusive_monitor; | ||||
| } | ||||
|  | ||||
| const Core::ExclusiveMonitor& KernelCore::GetExclusiveMonitor() const { | ||||
|     return *impl->exclusive_monitor; | ||||
| } | ||||
|  | ||||
| void KernelCore::InvalidateAllInstructionCaches() { | ||||
|     for (std::size_t i = 0; i < impl->global_scheduler.CpuCoresCount(); i++) { | ||||
|         PhysicalCore(i).ArmInterface().ClearInstructionCache(); | ||||
|     } | ||||
| } | ||||
|  | ||||
| void KernelCore::PrepareReschedule(std::size_t id) { | ||||
|     if (id < impl->global_scheduler.CpuCoresCount()) { | ||||
|         impl->cores[id].Stop(); | ||||
|     } | ||||
| } | ||||
|  | ||||
| void KernelCore::AddNamedPort(std::string name, std::shared_ptr<ClientPort> port) { | ||||
|     impl->named_ports.emplace(std::move(name), std::move(port)); | ||||
| } | ||||
|   | ||||
| @@ -11,8 +11,9 @@ | ||||
| #include "core/hle/kernel/object.h" | ||||
|  | ||||
| namespace Core { | ||||
| class ExclusiveMonitor; | ||||
| class System; | ||||
| } | ||||
| } // namespace Core | ||||
|  | ||||
| namespace Core::Timing { | ||||
| class CoreTiming; | ||||
| @@ -25,6 +26,7 @@ class AddressArbiter; | ||||
| class ClientPort; | ||||
| class GlobalScheduler; | ||||
| class HandleTable; | ||||
| class PhysicalCore; | ||||
| class Process; | ||||
| class ResourceLimit; | ||||
| class Thread; | ||||
| @@ -84,6 +86,21 @@ public: | ||||
|     /// Gets the sole instance of the global scheduler | ||||
|     const Kernel::GlobalScheduler& GlobalScheduler() const; | ||||
|  | ||||
|     /// Gets the an instance of the respective physical CPU core. | ||||
|     Kernel::PhysicalCore& PhysicalCore(std::size_t id); | ||||
|  | ||||
|     /// Gets the an instance of the respective physical CPU core. | ||||
|     const Kernel::PhysicalCore& PhysicalCore(std::size_t id) const; | ||||
|  | ||||
|     /// Stops execution of 'id' core, in order to reschedule a new thread. | ||||
|     void PrepareReschedule(std::size_t id); | ||||
|  | ||||
|     Core::ExclusiveMonitor& GetExclusiveMonitor(); | ||||
|  | ||||
|     const Core::ExclusiveMonitor& GetExclusiveMonitor() const; | ||||
|  | ||||
|     void InvalidateAllInstructionCaches(); | ||||
|  | ||||
|     /// Adds a port to the named port table | ||||
|     void AddNamedPort(std::string name, std::shared_ptr<ClientPort> port); | ||||
|  | ||||
|   | ||||
							
								
								
									
										52
									
								
								src/core/hle/kernel/physical_core.cpp
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										52
									
								
								src/core/hle/kernel/physical_core.cpp
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,52 @@ | ||||
| // Copyright 2020 yuzu Emulator Project | ||||
| // Licensed under GPLv2 or any later version | ||||
| // Refer to the license.txt file included. | ||||
|  | ||||
| #include "common/logging/log.h" | ||||
| #include "core/arm/arm_interface.h" | ||||
| #ifdef ARCHITECTURE_x86_64 | ||||
| #include "core/arm/dynarmic/arm_dynarmic.h" | ||||
| #endif | ||||
| #include "core/arm/exclusive_monitor.h" | ||||
| #include "core/arm/unicorn/arm_unicorn.h" | ||||
| #include "core/core.h" | ||||
| #include "core/hle/kernel/kernel.h" | ||||
| #include "core/hle/kernel/physical_core.h" | ||||
| #include "core/hle/kernel/scheduler.h" | ||||
| #include "core/hle/kernel/thread.h" | ||||
|  | ||||
| namespace Kernel { | ||||
|  | ||||
| PhysicalCore::PhysicalCore(Core::System& system, KernelCore& kernel, std::size_t id, | ||||
|                            Core::ExclusiveMonitor& exclusive_monitor) | ||||
|     : core_index{id}, kernel{kernel} { | ||||
| #ifdef ARCHITECTURE_x86_64 | ||||
|     arm_interface = std::make_shared<Core::ARM_Dynarmic>(system, exclusive_monitor, core_index); | ||||
| #else | ||||
|     arm_interface = std::make_shared<Core::ARM_Unicorn>(system); | ||||
|     LOG_WARNING(Core, "CPU JIT requested, but Dynarmic not available"); | ||||
| #endif | ||||
|  | ||||
|     scheduler = std::make_shared<Kernel::Scheduler>(system, *arm_interface, core_index); | ||||
| } | ||||
|  | ||||
| PhysicalCore::~PhysicalCore() = default; | ||||
|  | ||||
| void PhysicalCore::Run() { | ||||
|     arm_interface->Run(); | ||||
|     arm_interface->ClearExclusiveState(); | ||||
| } | ||||
|  | ||||
| void PhysicalCore::Step() { | ||||
|     arm_interface->Step(); | ||||
| } | ||||
|  | ||||
| void PhysicalCore::Stop() { | ||||
|     arm_interface->PrepareReschedule(); | ||||
| } | ||||
|  | ||||
| void PhysicalCore::Shutdown() { | ||||
|     scheduler->Shutdown(); | ||||
| } | ||||
|  | ||||
| } // namespace Kernel | ||||
							
								
								
									
										74
									
								
								src/core/hle/kernel/physical_core.h
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										74
									
								
								src/core/hle/kernel/physical_core.h
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,74 @@ | ||||
| // Copyright 2020 yuzu Emulator Project | ||||
| // Licensed under GPLv2 or any later version | ||||
| // Refer to the license.txt file included. | ||||
|  | ||||
| #pragma once | ||||
|  | ||||
| #include <cstddef> | ||||
| #include <memory> | ||||
|  | ||||
| namespace Kernel { | ||||
| class Scheduler; | ||||
| } // namespace Kernel | ||||
|  | ||||
| namespace Core { | ||||
| class ARM_Interface; | ||||
| class ExclusiveMonitor; | ||||
| class System; | ||||
| } // namespace Core | ||||
|  | ||||
| namespace Kernel { | ||||
|  | ||||
| class PhysicalCore { | ||||
| public: | ||||
|     PhysicalCore(Core::System& system, KernelCore& kernel, std::size_t id, | ||||
|                  Core::ExclusiveMonitor& exclusive_monitor); | ||||
|  | ||||
|     ~PhysicalCore(); | ||||
|  | ||||
|     /// Execute current jit state | ||||
|     void Run(); | ||||
|     /// Execute a single instruction in current jit. | ||||
|     void Step(); | ||||
|     /// Stop JIT execution/exit | ||||
|     void Stop(); | ||||
|  | ||||
|     // Shutdown this physical core. | ||||
|     void Shutdown(); | ||||
|  | ||||
|     Core::ARM_Interface& ArmInterface() { | ||||
|         return *arm_interface; | ||||
|     } | ||||
|  | ||||
|     const Core::ARM_Interface& ArmInterface() const { | ||||
|         return *arm_interface; | ||||
|     } | ||||
|  | ||||
|     bool IsMainCore() const { | ||||
|         return core_index == 0; | ||||
|     } | ||||
|  | ||||
|     bool IsSystemCore() const { | ||||
|         return core_index == 3; | ||||
|     } | ||||
|  | ||||
|     std::size_t CoreIndex() const { | ||||
|         return core_index; | ||||
|     } | ||||
|  | ||||
|     Kernel::Scheduler& Scheduler() { | ||||
|         return *scheduler; | ||||
|     } | ||||
|  | ||||
|     const Kernel::Scheduler& Scheduler() const { | ||||
|         return *scheduler; | ||||
|     } | ||||
|  | ||||
| private: | ||||
|     std::size_t core_index; | ||||
|     KernelCore& kernel; | ||||
|     std::shared_ptr<Core::ARM_Interface> arm_interface; | ||||
|     std::shared_ptr<Kernel::Scheduler> scheduler; | ||||
| }; | ||||
|  | ||||
| } // namespace Kernel | ||||
| @@ -14,7 +14,6 @@ | ||||
| #include "common/logging/log.h" | ||||
| #include "core/arm/arm_interface.h" | ||||
| #include "core/core.h" | ||||
| #include "core/core_cpu.h" | ||||
| #include "core/core_timing.h" | ||||
| #include "core/hle/kernel/kernel.h" | ||||
| #include "core/hle/kernel/process.h" | ||||
|   | ||||
| @@ -15,7 +15,7 @@ | ||||
| #include "common/string_util.h" | ||||
| #include "core/arm/exclusive_monitor.h" | ||||
| #include "core/core.h" | ||||
| #include "core/core_cpu.h" | ||||
| #include "core/core_manager.h" | ||||
| #include "core/core_timing.h" | ||||
| #include "core/core_timing_util.h" | ||||
| #include "core/hle/kernel/address_arbiter.h" | ||||
|   | ||||
| @@ -13,7 +13,6 @@ | ||||
| #include "common/thread_queue_list.h" | ||||
| #include "core/arm/arm_interface.h" | ||||
| #include "core/core.h" | ||||
| #include "core/core_cpu.h" | ||||
| #include "core/core_timing.h" | ||||
| #include "core/core_timing_util.h" | ||||
| #include "core/hle/kernel/errors.h" | ||||
| @@ -356,7 +355,7 @@ void Thread::SetActivity(ThreadActivity value) { | ||||
|         // Set status if not waiting | ||||
|         if (status == ThreadStatus::Ready || status == ThreadStatus::Running) { | ||||
|             SetStatus(ThreadStatus::Paused); | ||||
|             Core::System::GetInstance().CpuCore(processor_id).PrepareReschedule(); | ||||
|             kernel.PrepareReschedule(processor_id); | ||||
|         } | ||||
|     } else if (status == ThreadStatus::Paused) { | ||||
|         // Ready to reschedule | ||||
|   | ||||
| @@ -7,7 +7,6 @@ | ||||
| #include "common/common_types.h" | ||||
| #include "common/logging/log.h" | ||||
| #include "core/core.h" | ||||
| #include "core/core_cpu.h" | ||||
| #include "core/hle/kernel/kernel.h" | ||||
| #include "core/hle/kernel/object.h" | ||||
| #include "core/hle/kernel/process.h" | ||||
| @@ -96,7 +95,7 @@ void WaitObject::WakeupWaitingThread(std::shared_ptr<Thread> thread) { | ||||
|     } | ||||
|     if (resume) { | ||||
|         thread->ResumeFromWait(); | ||||
|         Core::System::GetInstance().PrepareReschedule(thread->GetProcessorID()); | ||||
|         kernel.PrepareReschedule(thread->GetProcessorID()); | ||||
|     } | ||||
| } | ||||
|  | ||||
|   | ||||
		Reference in New Issue
	
	Block a user
	 bunnei
					bunnei