forked from eden-emu/eden
		
	core: hle: kernel: KThread: Integrate with KWorkerTask and implement DoWorkerTaskImpl.
- This is used to terminate a thread asynchronously after it has been exited. - This fixes a crash that can occur in Pokemon Sword/Shield because a thread is incorrectly closed on svcExitThread, then, the thread is destroyed on svcCloseHandle while it is still scheduled. - Instead, we now wait for the thread to no longer be scheduled on all cores before destroying it from KWorkerTaskManager, which is accurate to HOS behavior.
This commit is contained in:
		
							parent
							
								
									18969b5d8e
								
							
						
					
					
						commit
						b52516263d
					
				
					 2 changed files with 28 additions and 2 deletions
				
			
		|  | @ -30,6 +30,7 @@ | |||
| #include "core/hle/kernel/k_system_control.h" | ||||
| #include "core/hle/kernel/k_thread.h" | ||||
| #include "core/hle/kernel/k_thread_queue.h" | ||||
| #include "core/hle/kernel/k_worker_task_manager.h" | ||||
| #include "core/hle/kernel/kernel.h" | ||||
| #include "core/hle/kernel/svc_results.h" | ||||
| #include "core/hle/kernel/time_manager.h" | ||||
|  | @ -332,7 +333,7 @@ void KThread::Finalize() { | |||
|     } | ||||
| 
 | ||||
|     // Perform inherited finalization.
 | ||||
|     KAutoObjectWithSlabHeapAndContainer<KThread, KSynchronizationObject>::Finalize(); | ||||
|     KSynchronizationObject::Finalize(); | ||||
| } | ||||
| 
 | ||||
| bool KThread::IsSignaled() const { | ||||
|  | @ -376,11 +377,28 @@ void KThread::StartTermination() { | |||
| 
 | ||||
|     // Register terminated dpc flag.
 | ||||
|     RegisterDpc(DpcFlag::Terminated); | ||||
| } | ||||
| 
 | ||||
| void KThread::FinishTermination() { | ||||
|     // Ensure that the thread is not executing on any core.
 | ||||
|     if (parent != nullptr) { | ||||
|         for (std::size_t i = 0; i < static_cast<std::size_t>(Core::Hardware::NUM_CPU_CORES); ++i) { | ||||
|             KThread* core_thread{}; | ||||
|             do { | ||||
|                 core_thread = kernel.Scheduler(i).GetCurrentThread(); | ||||
|             } while (core_thread == this); | ||||
|         } | ||||
|     } | ||||
| 
 | ||||
|     // Close the thread.
 | ||||
|     this->Close(); | ||||
| } | ||||
| 
 | ||||
| void KThread::DoWorkerTaskImpl() { | ||||
|     // Finish the termination that was begun by Exit().
 | ||||
|     this->FinishTermination(); | ||||
| } | ||||
| 
 | ||||
| void KThread::Pin(s32 current_core) { | ||||
|     ASSERT(kernel.GlobalSchedulerContext().IsLocked()); | ||||
| 
 | ||||
|  | @ -1027,6 +1045,9 @@ void KThread::Exit() { | |||
| 
 | ||||
|         // Start termination.
 | ||||
|         StartTermination(); | ||||
| 
 | ||||
|         // Register the thread as a work task.
 | ||||
|         KWorkerTaskManager::AddTask(kernel, KWorkerTaskManager::WorkerType::Exit, this); | ||||
|     } | ||||
| } | ||||
| 
 | ||||
|  |  | |||
		Loading…
	
	Add table
		Add a link
		
	
		Reference in a new issue
	
	 bunnei
						bunnei