| 
									
										
										
										
											2014-05-20 23:03:45 -04:00
										 |  |  | // Copyright 2014 Citra Emulator Project
 | 
					
						
							| 
									
										
										
										
											2014-12-16 21:38:14 -08:00
										 |  |  | // Licensed under GPLv2 or any later version
 | 
					
						
							| 
									
										
										
										
											2014-11-19 08:49:13 +00:00
										 |  |  | // Refer to the license.txt file included.
 | 
					
						
							| 
									
										
										
										
											2014-05-20 23:03:45 -04:00
										 |  |  | 
 | 
					
						
							|  |  |  | #include <map>
 | 
					
						
							|  |  |  | #include <vector>
 | 
					
						
							| 
									
										
										
										
											2015-01-31 19:22:40 -02:00
										 |  |  | #include <boost/range/algorithm_ext/erase.hpp>
 | 
					
						
							| 
									
										
										
										
											2015-05-06 04:06:12 -03:00
										 |  |  | #include "common/assert.h"
 | 
					
						
							| 
									
										
										
										
											2017-01-01 16:59:30 -05:00
										 |  |  | #include "core/core.h"
 | 
					
						
							| 
									
										
										
										
											2014-05-20 23:03:45 -04:00
										 |  |  | #include "core/hle/kernel/kernel.h"
 | 
					
						
							| 
									
										
										
										
											2016-09-20 23:52:38 -07:00
										 |  |  | #include "core/hle/kernel/mutex.h"
 | 
					
						
							| 
									
										
										
										
											2014-05-20 23:03:45 -04:00
										 |  |  | #include "core/hle/kernel/thread.h"
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | namespace Kernel { | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-01-20 18:33:23 -05:00
										 |  |  | void ReleaseThreadMutexes(Thread* thread) { | 
					
						
							| 
									
										
										
										
											2015-01-31 19:22:40 -02:00
										 |  |  |     for (auto& mtx : thread->held_mutexes) { | 
					
						
							| 
									
										
										
										
											2017-01-01 16:59:30 -05:00
										 |  |  |         mtx->lock_count = 0; | 
					
						
							|  |  |  |         mtx->holding_thread = nullptr; | 
					
						
							|  |  |  |         mtx->WakeupAllWaitingThreads(); | 
					
						
							| 
									
										
										
										
											2014-12-07 15:44:21 -05:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2015-01-31 19:22:40 -02:00
										 |  |  |     thread->held_mutexes.clear(); | 
					
						
							| 
									
										
										
										
											2014-12-07 15:44:21 -05:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-09-18 18:01:46 -07:00
										 |  |  | Mutex::Mutex() {} | 
					
						
							|  |  |  | Mutex::~Mutex() {} | 
					
						
							| 
									
										
										
										
											2015-01-31 22:56:59 -02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-02-01 00:14:40 -02:00
										 |  |  | SharedPtr<Mutex> Mutex::Create(bool initial_locked, std::string name) { | 
					
						
							| 
									
										
										
										
											2015-01-22 23:12:19 -02:00
										 |  |  |     SharedPtr<Mutex> mutex(new Mutex); | 
					
						
							| 
									
										
										
										
											2014-05-20 23:03:45 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-02-09 22:06:09 -05:00
										 |  |  |     mutex->lock_count = 0; | 
					
						
							| 
									
										
										
										
											2015-01-22 23:12:19 -02:00
										 |  |  |     mutex->name = std::move(name); | 
					
						
							| 
									
										
										
										
											2015-01-20 18:33:23 -05:00
										 |  |  |     mutex->holding_thread = nullptr; | 
					
						
							| 
									
										
										
										
											2014-05-20 23:03:45 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-01-01 16:53:22 -05:00
										 |  |  |     // Acquire mutex with current thread if initialized as locked
 | 
					
						
							| 
									
										
										
										
											2015-01-22 23:12:19 -02:00
										 |  |  |     if (initial_locked) | 
					
						
							| 
									
										
										
										
											2017-01-01 16:53:22 -05:00
										 |  |  |         mutex->Acquire(GetCurrentThread()); | 
					
						
							| 
									
										
										
										
											2014-05-20 23:03:45 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-02-01 00:14:40 -02:00
										 |  |  |     return mutex; | 
					
						
							| 
									
										
										
										
											2014-05-20 23:03:45 -04:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-01-01 16:53:22 -05:00
										 |  |  | bool Mutex::ShouldWait(Thread* thread) const { | 
					
						
							|  |  |  |     return lock_count > 0 && thread != holding_thread; | 
					
						
							| 
									
										
										
										
											2015-01-22 23:12:19 -02:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-01-01 16:53:22 -05:00
										 |  |  | void Mutex::Acquire(Thread* thread) { | 
					
						
							|  |  |  |     ASSERT_MSG(!ShouldWait(thread), "object unavailable!"); | 
					
						
							| 
									
										
										
										
											2015-01-22 23:12:19 -02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-01-01 16:59:30 -05:00
										 |  |  |     // Actually "acquire" the mutex only if we don't already have it
 | 
					
						
							| 
									
										
										
										
											2015-02-09 22:06:09 -05:00
										 |  |  |     if (lock_count == 0) { | 
					
						
							| 
									
										
										
										
											2017-01-01 16:59:30 -05:00
										 |  |  |         priority = thread->current_priority; | 
					
						
							| 
									
										
										
										
											2015-02-09 22:06:09 -05:00
										 |  |  |         thread->held_mutexes.insert(this); | 
					
						
							| 
									
										
										
										
											2017-01-01 16:59:30 -05:00
										 |  |  |         holding_thread = thread; | 
					
						
							| 
									
										
										
										
											2017-01-02 19:38:08 -05:00
										 |  |  |         thread->UpdatePriority(); | 
					
						
							| 
									
										
										
										
											2017-01-01 16:59:30 -05:00
										 |  |  |         Core::System::GetInstance().PrepareReschedule(); | 
					
						
							| 
									
										
										
										
											2015-02-09 22:06:09 -05:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2015-01-22 23:12:19 -02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-02-09 22:06:09 -05:00
										 |  |  |     lock_count++; | 
					
						
							| 
									
										
										
										
											2015-01-22 23:12:19 -02:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void Mutex::Release() { | 
					
						
							| 
									
										
										
										
											2017-01-01 16:59:30 -05:00
										 |  |  |     // Only release if the mutex is held
 | 
					
						
							| 
									
										
										
										
											2015-02-09 22:06:09 -05:00
										 |  |  |     if (lock_count > 0) { | 
					
						
							|  |  |  |         lock_count--; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-01-01 16:59:30 -05:00
										 |  |  |         // Yield to the next thread only if we've fully released the mutex
 | 
					
						
							| 
									
										
										
										
											2015-02-09 22:06:09 -05:00
										 |  |  |         if (lock_count == 0) { | 
					
						
							|  |  |  |             holding_thread->held_mutexes.erase(this); | 
					
						
							| 
									
										
										
										
											2017-01-02 19:38:08 -05:00
										 |  |  |             holding_thread->UpdatePriority(); | 
					
						
							| 
									
										
										
										
											2017-01-01 16:59:30 -05:00
										 |  |  |             holding_thread = nullptr; | 
					
						
							|  |  |  |             WakeupAllWaitingThreads(); | 
					
						
							| 
									
										
										
										
											2017-01-01 11:57:02 -05:00
										 |  |  |             Core::System::GetInstance().PrepareReschedule(); | 
					
						
							| 
									
										
										
										
											2015-02-09 22:06:09 -05:00
										 |  |  |         } | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2014-12-07 15:57:28 -05:00
										 |  |  | } | 
					
						
							| 
									
										
										
										
											2015-01-20 18:16:45 -05:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-01-01 16:59:30 -05:00
										 |  |  | void Mutex::AddWaitingThread(SharedPtr<Thread> thread) { | 
					
						
							|  |  |  |     WaitObject::AddWaitingThread(thread); | 
					
						
							| 
									
										
										
										
											2017-01-02 19:38:08 -05:00
										 |  |  |     thread->pending_mutexes.insert(this); | 
					
						
							|  |  |  |     UpdatePriority(); | 
					
						
							| 
									
										
										
										
											2017-01-02 13:53:10 -05:00
										 |  |  | } | 
					
						
							| 
									
										
										
										
											2017-01-01 16:59:30 -05:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-01-02 13:53:10 -05:00
										 |  |  | void Mutex::RemoveWaitingThread(Thread* thread) { | 
					
						
							|  |  |  |     WaitObject::RemoveWaitingThread(thread); | 
					
						
							| 
									
										
										
										
											2017-01-02 19:38:08 -05:00
										 |  |  |     thread->pending_mutexes.erase(this); | 
					
						
							|  |  |  |     UpdatePriority(); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void Mutex::UpdatePriority() { | 
					
						
							|  |  |  |     if (!holding_thread) | 
					
						
							|  |  |  |         return; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     s32 best_priority = THREADPRIO_LOWEST; | 
					
						
							|  |  |  |     for (auto& waiter : GetWaitingThreads()) { | 
					
						
							|  |  |  |         if (waiter->current_priority < best_priority) | 
					
						
							|  |  |  |             best_priority = waiter->current_priority; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (best_priority != priority) { | 
					
						
							|  |  |  |         priority = best_priority; | 
					
						
							|  |  |  |         holding_thread->UpdatePriority(); | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2017-01-01 16:59:30 -05:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-05-20 23:03:45 -04:00
										 |  |  | } // namespace
 |