| 
									
										
										
										
											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"
 | 
					
						
							| 
									
										
										
										
											2018-01-01 14:02:26 -05:00
										 |  |  | #include "core/hle/kernel/handle_table.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"
 | 
					
						
							| 
									
										
										
										
											2018-01-01 14:02:26 -05:00
										 |  |  | #include "core/hle/kernel/object_address_table.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) { | 
					
						
							| 
									
										
										
										
											2018-01-08 14:12:03 -05:00
										 |  |  |         mtx->SetHasWaiters(false); | 
					
						
							|  |  |  |         mtx->SetHoldingThread(nullptr); | 
					
						
							| 
									
										
										
										
											2017-01-01 16:59:30 -05:00
										 |  |  |         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
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-01-01 14:02:26 -05:00
										 |  |  | SharedPtr<Mutex> Mutex::Create(SharedPtr<Kernel::Thread> holding_thread, VAddr guest_addr, | 
					
						
							|  |  |  |                                std::string name) { | 
					
						
							| 
									
										
										
										
											2015-01-22 23:12:19 -02:00
										 |  |  |     SharedPtr<Mutex> mutex(new Mutex); | 
					
						
							| 
									
										
										
										
											2014-05-20 23:03:45 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-01-01 14:02:26 -05:00
										 |  |  |     mutex->guest_addr = guest_addr; | 
					
						
							| 
									
										
										
										
											2015-01-22 23:12:19 -02:00
										 |  |  |     mutex->name = std::move(name); | 
					
						
							| 
									
										
										
										
											2014-05-20 23:03:45 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-01-01 14:02:26 -05:00
										 |  |  |     // If mutex was initialized with a holding thread, acquire it by the holding thread
 | 
					
						
							|  |  |  |     if (holding_thread) { | 
					
						
							|  |  |  |         mutex->Acquire(holding_thread.get()); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     // Mutexes are referenced by guest address, so track this in the kernel
 | 
					
						
							|  |  |  |     g_object_address_table.Insert(guest_addr, mutex); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											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 { | 
					
						
							| 
									
										
										
										
											2018-01-08 14:12:03 -05:00
										 |  |  |     auto holding_thread = GetHoldingThread(); | 
					
						
							|  |  |  |     return holding_thread != nullptr && 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
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-01-08 14:12:03 -05:00
										 |  |  |     priority = thread->current_priority; | 
					
						
							|  |  |  |     thread->held_mutexes.insert(this); | 
					
						
							|  |  |  |     SetHoldingThread(thread); | 
					
						
							|  |  |  |     thread->UpdatePriority(); | 
					
						
							|  |  |  |     Core::System::GetInstance().PrepareReschedule(); | 
					
						
							| 
									
										
										
										
											2015-01-22 23:12:19 -02:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-01-01 14:02:26 -05:00
										 |  |  | ResultCode Mutex::Release(Thread* thread) { | 
					
						
							| 
									
										
										
										
											2018-01-08 14:12:03 -05:00
										 |  |  |     auto holding_thread = GetHoldingThread(); | 
					
						
							|  |  |  |     ASSERT(holding_thread); | 
					
						
							| 
									
										
										
										
											2018-01-01 14:02:26 -05:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-01-08 14:12:03 -05:00
										 |  |  |     // We can only release the mutex if it's held by the calling thread.
 | 
					
						
							|  |  |  |     ASSERT(thread == holding_thread); | 
					
						
							| 
									
										
										
										
											2018-01-01 14:02:26 -05:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-01-08 14:12:03 -05:00
										 |  |  |     holding_thread->held_mutexes.erase(this); | 
					
						
							|  |  |  |     holding_thread->UpdatePriority(); | 
					
						
							|  |  |  |     SetHoldingThread(nullptr); | 
					
						
							| 
									
										
										
										
											2018-02-05 21:54:10 -05:00
										 |  |  |     SetHasWaiters(!GetWaitingThreads().empty()); | 
					
						
							| 
									
										
										
										
											2018-01-08 14:12:03 -05:00
										 |  |  |     WakeupAllWaitingThreads(); | 
					
						
							|  |  |  |     Core::System::GetInstance().PrepareReschedule(); | 
					
						
							| 
									
										
										
										
											2018-01-01 14:02:26 -05:00
										 |  |  | 
 | 
					
						
							|  |  |  |     return RESULT_SUCCESS; | 
					
						
							| 
									
										
										
										
											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); | 
					
						
							| 
									
										
										
										
											2018-01-08 14:12:03 -05:00
										 |  |  |     SetHasWaiters(true); | 
					
						
							| 
									
										
										
										
											2017-01-02 19:38:08 -05:00
										 |  |  |     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); | 
					
						
							| 
									
										
										
										
											2018-01-08 14:12:03 -05:00
										 |  |  |     if (!GetHasWaiters()) | 
					
						
							|  |  |  |         SetHasWaiters(!GetWaitingThreads().empty()); | 
					
						
							| 
									
										
										
										
											2017-01-02 19:38:08 -05:00
										 |  |  |     UpdatePriority(); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void Mutex::UpdatePriority() { | 
					
						
							| 
									
										
										
										
											2018-01-08 14:12:03 -05:00
										 |  |  |     if (!GetHoldingThread()) | 
					
						
							| 
									
										
										
										
											2017-01-02 19:38:08 -05:00
										 |  |  |         return; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-09-27 00:26:09 +01:00
										 |  |  |     u32 best_priority = THREADPRIO_LOWEST; | 
					
						
							| 
									
										
										
										
											2017-01-02 19:38:08 -05:00
										 |  |  |     for (auto& waiter : GetWaitingThreads()) { | 
					
						
							|  |  |  |         if (waiter->current_priority < best_priority) | 
					
						
							|  |  |  |             best_priority = waiter->current_priority; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (best_priority != priority) { | 
					
						
							|  |  |  |         priority = best_priority; | 
					
						
							| 
									
										
										
										
											2018-01-08 14:12:03 -05:00
										 |  |  |         GetHoldingThread()->UpdatePriority(); | 
					
						
							| 
									
										
										
										
											2017-01-02 19:38:08 -05:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2017-01-01 16:59:30 -05:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-01-08 14:12:03 -05:00
										 |  |  | Handle Mutex::GetOwnerHandle() const { | 
					
						
							|  |  |  |     GuestState guest_state{Memory::Read32(guest_addr)}; | 
					
						
							|  |  |  |     return guest_state.holding_thread_handle; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | SharedPtr<Thread> Mutex::GetHoldingThread() const { | 
					
						
							|  |  |  |     GuestState guest_state{Memory::Read32(guest_addr)}; | 
					
						
							|  |  |  |     return g_handle_table.Get<Thread>(guest_state.holding_thread_handle); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void Mutex::SetHoldingThread(SharedPtr<Thread> thread) { | 
					
						
							| 
									
										
										
										
											2018-01-01 14:02:26 -05:00
										 |  |  |     GuestState guest_state{Memory::Read32(guest_addr)}; | 
					
						
							| 
									
										
										
										
											2018-01-08 14:12:03 -05:00
										 |  |  |     guest_state.holding_thread_handle.Assign(thread ? thread->guest_handle : 0); | 
					
						
							| 
									
										
										
										
											2018-01-01 14:02:26 -05:00
										 |  |  |     Memory::Write32(guest_addr, guest_state.raw); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-01-08 14:12:03 -05:00
										 |  |  | bool Mutex::GetHasWaiters() const { | 
					
						
							| 
									
										
										
										
											2018-01-01 14:02:26 -05:00
										 |  |  |     GuestState guest_state{Memory::Read32(guest_addr)}; | 
					
						
							| 
									
										
										
										
											2018-01-08 14:12:03 -05:00
										 |  |  |     return guest_state.has_waiters != 0; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void Mutex::SetHasWaiters(bool has_waiters) { | 
					
						
							|  |  |  |     GuestState guest_state{Memory::Read32(guest_addr)}; | 
					
						
							|  |  |  |     guest_state.has_waiters.Assign(has_waiters ? 1 : 0); | 
					
						
							|  |  |  |     Memory::Write32(guest_addr, guest_state.raw); | 
					
						
							| 
									
										
										
										
											2018-01-01 14:02:26 -05:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | } // namespace Kernel
 |