| 
									
										
										
										
											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
										 |  |  | 
 | 
					
						
							|  |  |  | #pragma once
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-01-22 23:12:19 -02:00
										 |  |  | #include <string>
 | 
					
						
							| 
									
										
										
										
											2014-05-20 23:03:45 -04:00
										 |  |  | #include "common/common_types.h"
 | 
					
						
							| 
									
										
										
										
											2018-01-01 14:02:26 -05:00
										 |  |  | #include "common/swap.h"
 | 
					
						
							| 
									
										
										
										
											2014-05-20 23:03:45 -04:00
										 |  |  | #include "core/hle/kernel/kernel.h"
 | 
					
						
							| 
									
										
										
										
											2017-05-29 15:45:30 -07:00
										 |  |  | #include "core/hle/kernel/wait_object.h"
 | 
					
						
							| 
									
										
										
										
											2018-01-01 14:02:26 -05:00
										 |  |  | #include "core/hle/result.h"
 | 
					
						
							| 
									
										
										
										
											2014-05-20 23:03:45 -04:00
										 |  |  | 
 | 
					
						
							|  |  |  | namespace Kernel { | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-01-22 23:12:19 -02:00
										 |  |  | class Thread; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-01-27 02:40:21 -02:00
										 |  |  | class Mutex final : public WaitObject { | 
					
						
							| 
									
										
										
										
											2015-01-22 23:12:19 -02:00
										 |  |  | public: | 
					
						
							|  |  |  |     /**
 | 
					
						
							|  |  |  |      * Creates a mutex. | 
					
						
							| 
									
										
										
										
											2018-01-01 14:02:26 -05:00
										 |  |  |      * @param holding_thread Specifies a thread already holding the mutex. If not nullptr, this | 
					
						
							|  |  |  |      * thread will acquire the mutex. | 
					
						
							|  |  |  |      * @param guest_addr Address of the object tracking the mutex in guest memory. If specified, | 
					
						
							|  |  |  |      * this mutex will update the guest object when its state changes. | 
					
						
							| 
									
										
										
										
											2015-01-22 23:12:19 -02:00
										 |  |  |      * @param name Optional name of mutex | 
					
						
							|  |  |  |      * @return Pointer to new Mutex object | 
					
						
							|  |  |  |      */ | 
					
						
							| 
									
										
										
										
											2018-01-01 14:02:26 -05:00
										 |  |  |     static SharedPtr<Mutex> Create(SharedPtr<Kernel::Thread> holding_thread, VAddr guest_addr = 0, | 
					
						
							|  |  |  |                                    std::string name = "Unknown"); | 
					
						
							| 
									
										
										
										
											2015-01-22 23:12:19 -02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-09-18 09:38:01 +09:00
										 |  |  |     std::string GetTypeName() const override { | 
					
						
							|  |  |  |         return "Mutex"; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     std::string GetName() const override { | 
					
						
							|  |  |  |         return name; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2015-01-22 23:12:19 -02:00
										 |  |  | 
 | 
					
						
							|  |  |  |     static const HandleType HANDLE_TYPE = HandleType::Mutex; | 
					
						
							| 
									
										
										
										
											2016-09-18 09:38:01 +09:00
										 |  |  |     HandleType GetHandleType() const override { | 
					
						
							|  |  |  |         return HANDLE_TYPE; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2015-01-22 23:12:19 -02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-09-18 09:38:01 +09:00
										 |  |  |     int lock_count;                   ///< Number of times the mutex has been acquired
 | 
					
						
							| 
									
										
										
										
											2017-01-01 16:59:30 -05:00
										 |  |  |     u32 priority;                     ///< The priority of the mutex, used for priority inheritance.
 | 
					
						
							| 
									
										
										
										
											2016-09-18 09:38:01 +09:00
										 |  |  |     std::string name;                 ///< Name of mutex (optional)
 | 
					
						
							|  |  |  |     SharedPtr<Thread> holding_thread; ///< Thread that has acquired the mutex
 | 
					
						
							| 
									
										
										
										
											2018-01-01 14:02:26 -05:00
										 |  |  |     VAddr guest_addr;                 ///< Address of the guest mutex value
 | 
					
						
							| 
									
										
										
										
											2015-01-22 23:12:19 -02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-01-02 19:38:08 -05:00
										 |  |  |     /**
 | 
					
						
							|  |  |  |      * Elevate the mutex priority to the best priority | 
					
						
							|  |  |  |      * among the priorities of all its waiting threads. | 
					
						
							|  |  |  |      */ | 
					
						
							|  |  |  |     void UpdatePriority(); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-01-01 16:53:22 -05:00
										 |  |  |     bool ShouldWait(Thread* thread) const override; | 
					
						
							|  |  |  |     void Acquire(Thread* thread) override; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-01-01 16:59:30 -05:00
										 |  |  |     void AddWaitingThread(SharedPtr<Thread> thread) override; | 
					
						
							| 
									
										
										
										
											2017-01-02 13:53:10 -05:00
										 |  |  |     void RemoveWaitingThread(Thread* thread) override; | 
					
						
							| 
									
										
										
										
											2015-01-22 23:12:19 -02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-01-01 14:02:26 -05:00
										 |  |  |     /**
 | 
					
						
							|  |  |  |      * Attempts to release the mutex from the specified thread. | 
					
						
							|  |  |  |      * @param thread Thread that wants to release the mutex. | 
					
						
							|  |  |  |      * @returns The result code of the operation. | 
					
						
							|  |  |  |      */ | 
					
						
							|  |  |  |     ResultCode Release(Thread* thread); | 
					
						
							| 
									
										
										
										
											2015-01-22 23:12:19 -02:00
										 |  |  | 
 | 
					
						
							|  |  |  | private: | 
					
						
							| 
									
										
										
										
											2015-01-31 22:56:59 -02:00
										 |  |  |     Mutex(); | 
					
						
							|  |  |  |     ~Mutex() override; | 
					
						
							| 
									
										
										
										
											2018-01-01 14:02:26 -05:00
										 |  |  | 
 | 
					
						
							|  |  |  |     /// Object in guest memory used to track the mutex state
 | 
					
						
							|  |  |  |     union GuestState { | 
					
						
							|  |  |  |         u32_le raw; | 
					
						
							|  |  |  |         /// Handle of the thread that currently holds the mutex, 0 if available
 | 
					
						
							|  |  |  |         BitField<0, 30, u32_le> holding_thread_handle; | 
					
						
							|  |  |  |         /// 1 when there are threads waiting for this mutex, otherwise 0
 | 
					
						
							|  |  |  |         BitField<30, 1, u32_le> has_waiters; | 
					
						
							|  |  |  |     }; | 
					
						
							|  |  |  |     static_assert(sizeof(GuestState) == 4, "GuestState size is incorrect"); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     /// Updates the state of the object tracking this mutex in guest memory
 | 
					
						
							|  |  |  |     void UpdateGuestState(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     /// Verifies the state of the object tracking this mutex in guest memory
 | 
					
						
							|  |  |  |     void VerifyGuestState(); | 
					
						
							| 
									
										
										
										
											2015-01-22 23:12:19 -02:00
										 |  |  | }; | 
					
						
							| 
									
										
										
										
											2014-05-20 23:03:45 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-12-07 15:44:21 -05:00
										 |  |  | /**
 | 
					
						
							|  |  |  |  * Releases all the mutexes held by the specified thread | 
					
						
							|  |  |  |  * @param thread Thread that is holding the mutexes | 
					
						
							|  |  |  |  */ | 
					
						
							| 
									
										
										
										
											2015-01-20 18:33:23 -05:00
										 |  |  | void ReleaseThreadMutexes(Thread* thread); | 
					
						
							| 
									
										
										
										
											2014-12-07 15:44:21 -05:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-01-01 14:02:26 -05:00
										 |  |  | } // namespace Kernel
 |