| 
									
										
										
										
											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"
 | 
					
						
							| 
									
										
										
										
											2018-12-31 18:09:41 -05:00
										 |  |  | #include "common/common_types.h"
 | 
					
						
							| 
									
										
										
										
											2017-05-29 15:45:30 -07:00
										 |  |  | #include "common/logging/log.h"
 | 
					
						
							| 
									
										
										
										
											2019-03-29 17:13:00 -04:00
										 |  |  | #include "core/core.h"
 | 
					
						
							| 
									
										
										
										
											2019-09-11 12:47:37 -04:00
										 |  |  | #include "core/hle/kernel/kernel.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"
 | 
					
						
							| 
									
										
										
										
											2020-02-11 17:36:39 -04:00
										 |  |  | #include "core/hle/kernel/synchronization.h"
 | 
					
						
							| 
									
										
										
										
											2020-02-11 10:46:25 -04:00
										 |  |  | #include "core/hle/kernel/synchronization_object.h"
 | 
					
						
							| 
									
										
										
										
											2017-05-29 15:45:30 -07:00
										 |  |  | #include "core/hle/kernel/thread.h"
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | namespace Kernel { | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-02-11 10:46:25 -04:00
										 |  |  | SynchronizationObject::SynchronizationObject(KernelCore& kernel) : Object{kernel} {} | 
					
						
							|  |  |  | SynchronizationObject::~SynchronizationObject() = default; | 
					
						
							| 
									
										
										
										
											2018-08-28 12:30:33 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-02-11 17:36:39 -04:00
										 |  |  | void SynchronizationObject::Signal() { | 
					
						
							|  |  |  |     kernel.Synchronization().SignalObject(*this); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-02-11 10:46:25 -04:00
										 |  |  | void SynchronizationObject::AddWaitingThread(std::shared_ptr<Thread> thread) { | 
					
						
							| 
									
										
										
										
											2017-05-29 15:45:30 -07:00
										 |  |  |     auto itr = std::find(waiting_threads.begin(), waiting_threads.end(), thread); | 
					
						
							|  |  |  |     if (itr == waiting_threads.end()) | 
					
						
							|  |  |  |         waiting_threads.push_back(std::move(thread)); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-02-11 10:46:25 -04:00
										 |  |  | void SynchronizationObject::RemoveWaitingThread(std::shared_ptr<Thread> thread) { | 
					
						
							| 
									
										
										
										
											2017-05-29 15:45:30 -07:00
										 |  |  |     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); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-02-11 10:46:25 -04:00
										 |  |  | std::shared_ptr<Thread> SynchronizationObject::GetHighestPriorityReadyThread() const { | 
					
						
							| 
									
										
										
										
											2017-05-29 15:45:30 -07:00
										 |  |  |     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.
 | 
					
						
							| 
									
										
										
										
											2019-04-17 07:21:19 -04:00
										 |  |  |         ASSERT_MSG(thread_status == ThreadStatus::WaitSynch || | 
					
						
							| 
									
										
										
										
											2018-10-03 18:47:57 -04:00
										 |  |  |                        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; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-07-03 23:21:48 -04:00
										 |  |  |         candidate = thread.get(); | 
					
						
							|  |  |  |         candidate_priority = thread->GetPriority(); | 
					
						
							| 
									
										
										
										
											2017-05-29 15:45:30 -07:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-11-24 20:15:51 -05:00
										 |  |  |     return SharedFrom(candidate); | 
					
						
							| 
									
										
										
										
											2017-05-29 15:45:30 -07:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-02-11 10:46:25 -04:00
										 |  |  | void SynchronizationObject::WakeupWaitingThread(std::shared_ptr<Thread> thread) { | 
					
						
							| 
									
										
										
										
											2018-01-08 11:35:03 -05:00
										 |  |  |     ASSERT(!ShouldWait(thread.get())); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-17 07:21:19 -04:00
										 |  |  |     if (!thread) { | 
					
						
							| 
									
										
										
										
											2018-01-07 16:33:41 -05:00
										 |  |  |         return; | 
					
						
							| 
									
										
										
										
											2019-04-17 07:21:19 -04:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2018-01-07 16:33:41 -05:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-17 07:21:19 -04:00
										 |  |  |     if (thread->IsSleepingOnWait()) { | 
					
						
							| 
									
										
										
										
											2020-02-11 10:46:25 -04:00
										 |  |  |         for (const auto& object : thread->GetSynchronizationObjects()) { | 
					
						
							| 
									
										
										
										
											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
										 |  |  |         } | 
					
						
							| 
									
										
										
										
											2019-04-17 07:21:19 -04:00
										 |  |  |     } else { | 
					
						
							|  |  |  |         Acquire(thread.get()); | 
					
						
							| 
									
										
										
										
											2018-01-07 16:33:41 -05:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2017-05-29 15:45:30 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-02-11 10:46:25 -04:00
										 |  |  |     const std::size_t index = thread->GetSynchronizationObjectIndex(SharedFrom(this)); | 
					
						
							| 
									
										
										
										
											2017-09-28 11:53:32 -05:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-02-11 10:46:25 -04:00
										 |  |  |     thread->ClearSynchronizationObjects(); | 
					
						
							| 
									
										
										
										
											2017-05-29 15:45:30 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-01-08 11:35:03 -05:00
										 |  |  |     thread->CancelWakeupTimer(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     bool resume = true; | 
					
						
							| 
									
										
										
										
											2019-04-17 07:21:19 -04:00
										 |  |  |     if (thread->HasWakeupCallback()) { | 
					
						
							| 
									
										
										
										
											2019-11-24 20:15:51 -05:00
										 |  |  |         resume = thread->InvokeWakeupCallback(ThreadWakeupReason::Signal, thread, SharedFrom(this), | 
					
						
							|  |  |  |                                               index); | 
					
						
							| 
									
										
										
										
											2019-04-17 07:21:19 -04:00
										 |  |  |     } | 
					
						
							|  |  |  |     if (resume) { | 
					
						
							| 
									
										
										
										
											2018-01-08 11:35:03 -05:00
										 |  |  |         thread->ResumeFromWait(); | 
					
						
							| 
									
										
										
										
											2020-01-26 14:23:46 -04:00
										 |  |  |         kernel.PrepareReschedule(thread->GetProcessorID()); | 
					
						
							| 
									
										
										
										
											2019-04-17 07:21:19 -04:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2018-01-07 16:33:41 -05:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-02-11 10:46:25 -04:00
										 |  |  | void SynchronizationObject::WakeupAllWaitingThreads() { | 
					
						
							| 
									
										
										
										
											2018-01-07 16:33:41 -05:00
										 |  |  |     while (auto thread = GetHighestPriorityReadyThread()) { | 
					
						
							|  |  |  |         WakeupWaitingThread(thread); | 
					
						
							| 
									
										
										
										
											2017-05-29 15:45:30 -07:00
										 |  |  |     } | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-02-11 10:46:25 -04:00
										 |  |  | const std::vector<std::shared_ptr<Thread>>& SynchronizationObject::GetWaitingThreads() const { | 
					
						
							| 
									
										
										
										
											2017-05-29 15:45:30 -07:00
										 |  |  |     return waiting_threads; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | } // namespace Kernel
 |