| 
									
										
										
										
											2017-05-29 15:45:30 -07:00
										 |  |  | // Copyright 2014 Citra Emulator Project
 | 
					
						
							|  |  |  | // Licensed under GPLv2 or any later version
 | 
					
						
							|  |  |  | // Refer to the license.txt file included.
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #include <algorithm>
 | 
					
						
							|  |  |  | #include "common/assert.h"
 | 
					
						
							|  |  |  | #include "common/logging/log.h"
 | 
					
						
							| 
									
										
										
										
											2018-08-01 22:40:00 -04:00
										 |  |  | #include "core/hle/kernel/object.h"
 | 
					
						
							| 
									
										
										
										
											2017-05-29 15:45:30 -07:00
										 |  |  | #include "core/hle/kernel/process.h"
 | 
					
						
							|  |  |  | #include "core/hle/kernel/thread.h"
 | 
					
						
							|  |  |  | #include "core/hle/kernel/timer.h"
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | namespace Kernel { | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-08-28 12:30:33 -04:00
										 |  |  | WaitObject::WaitObject(KernelCore& kernel) : Object{kernel} {} | 
					
						
							|  |  |  | WaitObject::~WaitObject() = default; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-05-29 15:45:30 -07:00
										 |  |  | void WaitObject::AddWaitingThread(SharedPtr<Thread> thread) { | 
					
						
							|  |  |  |     auto itr = std::find(waiting_threads.begin(), waiting_threads.end(), thread); | 
					
						
							|  |  |  |     if (itr == waiting_threads.end()) | 
					
						
							|  |  |  |         waiting_threads.push_back(std::move(thread)); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void WaitObject::RemoveWaitingThread(Thread* thread) { | 
					
						
							|  |  |  |     auto itr = std::find(waiting_threads.begin(), waiting_threads.end(), thread); | 
					
						
							|  |  |  |     // If a thread passed multiple handles to the same object,
 | 
					
						
							|  |  |  |     // the kernel might attempt to remove the thread from the object's
 | 
					
						
							|  |  |  |     // waiting threads list multiple times.
 | 
					
						
							|  |  |  |     if (itr != waiting_threads.end()) | 
					
						
							|  |  |  |         waiting_threads.erase(itr); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | SharedPtr<Thread> WaitObject::GetHighestPriorityReadyThread() { | 
					
						
							|  |  |  |     Thread* candidate = nullptr; | 
					
						
							| 
									
										
										
										
											2017-09-27 00:26:09 +01:00
										 |  |  |     u32 candidate_priority = THREADPRIO_LOWEST + 1; | 
					
						
							| 
									
										
										
										
											2017-05-29 15:45:30 -07:00
										 |  |  | 
 | 
					
						
							|  |  |  |     for (const auto& thread : waiting_threads) { | 
					
						
							| 
									
										
										
										
											2018-10-03 18:47:57 -04:00
										 |  |  |         const ThreadStatus thread_status = thread->GetStatus(); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-05-29 15:45:30 -07:00
										 |  |  |         // The list of waiting threads must not contain threads that are not waiting to be awakened.
 | 
					
						
							| 
									
										
										
										
											2018-10-03 18:47:57 -04:00
										 |  |  |         ASSERT_MSG(thread_status == ThreadStatus::WaitSynchAny || | 
					
						
							|  |  |  |                        thread_status == ThreadStatus::WaitSynchAll || | 
					
						
							|  |  |  |                        thread_status == ThreadStatus::WaitHLEEvent, | 
					
						
							| 
									
										
										
										
											2017-05-29 15:45:30 -07:00
										 |  |  |                    "Inconsistent thread statuses in waiting_threads"); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-10-03 18:47:57 -04:00
										 |  |  |         if (thread->GetPriority() >= candidate_priority) | 
					
						
							| 
									
										
										
										
											2017-05-29 15:45:30 -07:00
										 |  |  |             continue; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         if (ShouldWait(thread.get())) | 
					
						
							|  |  |  |             continue; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-07-19 21:39:05 -04:00
										 |  |  |         // A thread is ready to run if it's either in ThreadStatus::WaitSynchAny or
 | 
					
						
							|  |  |  |         // in ThreadStatus::WaitSynchAll and the rest of the objects it is waiting on are ready.
 | 
					
						
							| 
									
										
										
										
											2017-05-29 15:45:30 -07:00
										 |  |  |         bool ready_to_run = true; | 
					
						
							| 
									
										
										
										
											2018-10-03 18:47:57 -04:00
										 |  |  |         if (thread_status == ThreadStatus::WaitSynchAll) { | 
					
						
							|  |  |  |             ready_to_run = thread->AllWaitObjectsReady(); | 
					
						
							| 
									
										
										
										
											2017-05-29 15:45:30 -07:00
										 |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         if (ready_to_run) { | 
					
						
							|  |  |  |             candidate = thread.get(); | 
					
						
							| 
									
										
										
										
											2018-10-03 18:47:57 -04:00
										 |  |  |             candidate_priority = thread->GetPriority(); | 
					
						
							| 
									
										
										
										
											2017-05-29 15:45:30 -07:00
										 |  |  |         } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     return candidate; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-01-07 16:33:41 -05:00
										 |  |  | void WaitObject::WakeupWaitingThread(SharedPtr<Thread> thread) { | 
					
						
							| 
									
										
										
										
											2018-01-08 11:35:03 -05:00
										 |  |  |     ASSERT(!ShouldWait(thread.get())); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-01-07 16:33:41 -05:00
										 |  |  |     if (!thread) | 
					
						
							|  |  |  |         return; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (!thread->IsSleepingOnWaitAll()) { | 
					
						
							|  |  |  |         Acquire(thread.get()); | 
					
						
							|  |  |  |     } else { | 
					
						
							| 
									
										
										
										
											2018-10-03 18:47:57 -04:00
										 |  |  |         for (const auto& object : thread->GetWaitObjects()) { | 
					
						
							| 
									
										
										
										
											2018-01-08 11:35:03 -05:00
										 |  |  |             ASSERT(!object->ShouldWait(thread.get())); | 
					
						
							| 
									
										
										
										
											2018-01-07 16:33:41 -05:00
										 |  |  |             object->Acquire(thread.get()); | 
					
						
							| 
									
										
										
										
											2017-05-29 15:45:30 -07:00
										 |  |  |         } | 
					
						
							| 
									
										
										
										
											2018-01-07 16:33:41 -05:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2017-05-29 15:45:30 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-10-03 18:47:57 -04:00
										 |  |  |     const std::size_t index = thread->GetWaitObjectIndex(this); | 
					
						
							| 
									
										
										
										
											2017-09-28 11:53:32 -05:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-10-03 18:47:57 -04:00
										 |  |  |     for (const auto& object : thread->GetWaitObjects()) | 
					
						
							| 
									
										
										
										
											2018-01-07 16:33:41 -05:00
										 |  |  |         object->RemoveWaitingThread(thread.get()); | 
					
						
							| 
									
										
										
										
											2018-10-03 18:47:57 -04:00
										 |  |  |     thread->ClearWaitObjects(); | 
					
						
							| 
									
										
										
										
											2017-05-29 15:45:30 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-01-08 11:35:03 -05:00
										 |  |  |     thread->CancelWakeupTimer(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     bool resume = true; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-10-03 18:47:57 -04:00
										 |  |  |     if (thread->HasWakeupCallback()) | 
					
						
							|  |  |  |         resume = thread->InvokeWakeupCallback(ThreadWakeupReason::Signal, thread, this, index); | 
					
						
							| 
									
										
										
										
											2018-01-08 11:35:03 -05:00
										 |  |  | 
 | 
					
						
							|  |  |  |     if (resume) | 
					
						
							|  |  |  |         thread->ResumeFromWait(); | 
					
						
							| 
									
										
										
										
											2018-01-07 16:33:41 -05:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void WaitObject::WakeupAllWaitingThreads() { | 
					
						
							|  |  |  |     while (auto thread = GetHighestPriorityReadyThread()) { | 
					
						
							|  |  |  |         WakeupWaitingThread(thread); | 
					
						
							| 
									
										
										
										
											2017-05-29 15:45:30 -07:00
										 |  |  |     } | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | const std::vector<SharedPtr<Thread>>& WaitObject::GetWaitingThreads() const { | 
					
						
							|  |  |  |     return waiting_threads; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | } // namespace Kernel
 |