diff options
| author | 2022-07-07 12:34:46 -0400 | |
|---|---|---|
| committer | 2022-07-14 22:47:18 -0400 | |
| commit | da07e13e0798a4ebd423595830f04e2234a03942 (patch) | |
| tree | 2e2a88eb2fd2a5ae6f070609040418f9c1df4c35 /src/core/cpu_manager.cpp | |
| parent | kernel: fix issues with single core mode (diff) | |
| download | yuzu-da07e13e0798a4ebd423595830f04e2234a03942.tar.gz yuzu-da07e13e0798a4ebd423595830f04e2234a03942.tar.xz yuzu-da07e13e0798a4ebd423595830f04e2234a03942.zip | |
kernel: fix single-core preemption points
Diffstat (limited to 'src/core/cpu_manager.cpp')
| -rw-r--r-- | src/core/cpu_manager.cpp | 42 |
1 files changed, 14 insertions, 28 deletions
diff --git a/src/core/cpu_manager.cpp b/src/core/cpu_manager.cpp index 838d6be21..9b1565ae1 100644 --- a/src/core/cpu_manager.cpp +++ b/src/core/cpu_manager.cpp | |||
| @@ -144,39 +144,25 @@ void CpuManager::SingleCoreRunIdleThread() { | |||
| 144 | } | 144 | } |
| 145 | 145 | ||
| 146 | void CpuManager::PreemptSingleCore(bool from_running_environment) { | 146 | void CpuManager::PreemptSingleCore(bool from_running_environment) { |
| 147 | { | 147 | auto& kernel = system.Kernel(); |
| 148 | auto& kernel = system.Kernel(); | ||
| 149 | auto& scheduler = kernel.Scheduler(current_core); | ||
| 150 | |||
| 151 | Kernel::KThread* current_thread = scheduler.GetSchedulerCurrentThread(); | ||
| 152 | if (idle_count >= 4 || from_running_environment) { | ||
| 153 | if (!from_running_environment) { | ||
| 154 | system.CoreTiming().Idle(); | ||
| 155 | idle_count = 0; | ||
| 156 | } | ||
| 157 | kernel.SetIsPhantomModeForSingleCore(true); | ||
| 158 | system.CoreTiming().Advance(); | ||
| 159 | kernel.SetIsPhantomModeForSingleCore(false); | ||
| 160 | } | ||
| 161 | current_core.store((current_core + 1) % Core::Hardware::NUM_CPU_CORES); | ||
| 162 | system.CoreTiming().ResetTicks(); | ||
| 163 | scheduler.Unload(scheduler.GetSchedulerCurrentThread()); | ||
| 164 | |||
| 165 | auto& next_scheduler = kernel.Scheduler(current_core); | ||
| 166 | 148 | ||
| 167 | // Disable dispatch. We're about to preempt this thread. | 149 | if (idle_count >= 4 || from_running_environment) { |
| 168 | Kernel::KScopedDisableDispatch dd{kernel}; | 150 | if (!from_running_environment) { |
| 169 | Common::Fiber::YieldTo(current_thread->GetHostContext(), *next_scheduler.GetSwitchFiber()); | 151 | system.CoreTiming().Idle(); |
| 152 | idle_count = 0; | ||
| 153 | } | ||
| 154 | kernel.SetIsPhantomModeForSingleCore(true); | ||
| 155 | system.CoreTiming().Advance(); | ||
| 156 | kernel.SetIsPhantomModeForSingleCore(false); | ||
| 170 | } | 157 | } |
| 158 | current_core.store((current_core + 1) % Core::Hardware::NUM_CPU_CORES); | ||
| 159 | system.CoreTiming().ResetTicks(); | ||
| 160 | kernel.Scheduler(current_core).PreemptSingleCore(); | ||
| 171 | 161 | ||
| 172 | // We've now been scheduled again, and we may have exchanged schedulers. | 162 | // We've now been scheduled again, and we may have exchanged schedulers. |
| 173 | // Reload the scheduler in case it's different. | 163 | // Reload the scheduler in case it's different. |
| 174 | { | 164 | if (!kernel.Scheduler(current_core).IsIdle()) { |
| 175 | auto& scheduler = system.Kernel().Scheduler(current_core); | 165 | idle_count = 0; |
| 176 | scheduler.Reload(scheduler.GetSchedulerCurrentThread()); | ||
| 177 | if (!scheduler.IsIdle()) { | ||
| 178 | idle_count = 0; | ||
| 179 | } | ||
| 180 | } | 166 | } |
| 181 | } | 167 | } |
| 182 | 168 | ||