| 
									
										
										
										
											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"
 | 
					
						
							| 
									
										
										
										
											2014-05-20 23:03:45 -04:00
										 |  |  | 
 | 
					
						
							|  |  |  | #include "core/hle/kernel/kernel.h"
 | 
					
						
							| 
									
										
										
										
											2014-05-29 23:31:01 -04:00
										 |  |  | #include "core/hle/kernel/mutex.h"
 | 
					
						
							| 
									
										
										
										
											2014-05-20 23:03:45 -04:00
										 |  |  | #include "core/hle/kernel/thread.h"
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | namespace Kernel { | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-12-07 15:44:21 -05:00
										 |  |  | /**
 | 
					
						
							|  |  |  |  * Resumes a thread waiting for the specified mutex | 
					
						
							|  |  |  |  * @param mutex The mutex that some thread is waiting on | 
					
						
							|  |  |  |  */ | 
					
						
							| 
									
										
										
										
											2015-01-11 01:20:49 -02:00
										 |  |  | static void ResumeWaitingThread(Mutex* mutex) { | 
					
						
							| 
									
										
										
										
											2015-01-22 23:12:19 -02:00
										 |  |  |     // Reset mutex lock thread handle, nothing is waiting
 | 
					
						
							| 
									
										
										
										
											2015-02-09 22:06:09 -05:00
										 |  |  |     mutex->lock_count = 0; | 
					
						
							| 
									
										
										
										
											2015-01-22 23:12:19 -02:00
										 |  |  |     mutex->holding_thread = nullptr; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-12-07 15:44:21 -05:00
										 |  |  |     // Find the next waiting thread for the mutex...
 | 
					
						
							| 
									
										
										
										
											2015-01-20 18:20:47 -05:00
										 |  |  |     auto next_thread = mutex->WakeupNextThread(); | 
					
						
							| 
									
										
										
										
											2015-01-14 19:22:50 -05:00
										 |  |  |     if (next_thread != nullptr) { | 
					
						
							| 
									
										
										
										
											2015-01-22 23:12:19 -02:00
										 |  |  |         mutex->Acquire(next_thread); | 
					
						
							| 
									
										
										
										
											2014-12-07 15:44:21 -05:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2014-05-20 23:03:45 -04:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											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) { | 
					
						
							|  |  |  |         ResumeWaitingThread(mtx.get()); | 
					
						
							| 
									
										
										
										
											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
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-01-31 22:56:59 -02:00
										 |  |  | Mutex::Mutex() {} | 
					
						
							|  |  |  | Mutex::~Mutex() {} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											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
										 |  |  | 
 | 
					
						
							|  |  |  |     // Acquire mutex with current thread if initialized as locked...
 | 
					
						
							| 
									
										
										
										
											2015-01-22 23:12:19 -02:00
										 |  |  |     if (initial_locked) | 
					
						
							|  |  |  |         mutex->Acquire(); | 
					
						
							| 
									
										
										
										
											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
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-01-20 18:16:45 -05:00
										 |  |  | bool Mutex::ShouldWait() { | 
					
						
							| 
									
										
										
										
											2015-04-03 18:40:16 -04:00
										 |  |  |     auto thread = GetCurrentThread(); | 
					
						
							|  |  |  |     bool wait = lock_count > 0 && holding_thread != thread; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     // If the holding thread of the mutex is lower priority than this thread, that thread should
 | 
					
						
							|  |  |  |     // temporarily inherit this thread's priority
 | 
					
						
							|  |  |  |     if (wait && thread->current_priority < holding_thread->current_priority) | 
					
						
							|  |  |  |         holding_thread->BoostPriority(thread->current_priority); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     return wait; | 
					
						
							| 
									
										
										
										
											2015-01-17 22:23:49 -05:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-01-20 18:16:45 -05:00
										 |  |  | void Mutex::Acquire() { | 
					
						
							| 
									
										
										
										
											2015-01-22 23:12:19 -02:00
										 |  |  |     Acquire(GetCurrentThread()); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-01-31 23:26:16 -02:00
										 |  |  | void Mutex::Acquire(SharedPtr<Thread> thread) { | 
					
						
							| 
									
										
										
										
											2015-01-20 17:16:47 -08:00
										 |  |  |     ASSERT_MSG(!ShouldWait(), "object unavailable!"); | 
					
						
							| 
									
										
										
										
											2015-01-22 23:12:19 -02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-02-09 22:06:09 -05:00
										 |  |  |     // Actually "acquire" the mutex only if we don't already have it...
 | 
					
						
							|  |  |  |     if (lock_count == 0) { | 
					
						
							|  |  |  |         thread->held_mutexes.insert(this); | 
					
						
							|  |  |  |         holding_thread = std::move(thread); | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											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() { | 
					
						
							| 
									
										
										
										
											2015-02-09 22:06:09 -05:00
										 |  |  |     // Only release if the mutex is held...
 | 
					
						
							|  |  |  |     if (lock_count > 0) { | 
					
						
							|  |  |  |         lock_count--; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         // Yield to the next thread only if we've fully released the mutex...
 | 
					
						
							|  |  |  |         if (lock_count == 0) { | 
					
						
							|  |  |  |             holding_thread->held_mutexes.erase(this); | 
					
						
							|  |  |  |             ResumeWaitingThread(this); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2015-05-19 20:24:30 -04:00
										 |  |  | 
 | 
					
						
							|  |  |  |     HLE::Reschedule(__func__); | 
					
						
							| 
									
										
										
										
											2014-12-07 15:57:28 -05:00
										 |  |  | } | 
					
						
							| 
									
										
										
										
											2015-01-20 18:16:45 -05:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-05-20 23:03:45 -04:00
										 |  |  | } // namespace
 |