| 
									
										
										
										
											2014-12-03 18:49:51 -05:00
										 |  |  | // Copyright 2014 Citra Emulator Project
 | 
					
						
							|  |  |  | // Licensed under GPLv2+
 | 
					
						
							|  |  |  | // Refer to the license.txt file included.
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-12-04 11:40:36 -05:00
										 |  |  | #include <queue>
 | 
					
						
							| 
									
										
										
										
											2014-12-03 18:49:51 -05:00
										 |  |  | 
 | 
					
						
							|  |  |  | #include "common/common.h"
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #include "core/hle/kernel/kernel.h"
 | 
					
						
							|  |  |  | #include "core/hle/kernel/semaphore.h"
 | 
					
						
							|  |  |  | #include "core/hle/kernel/thread.h"
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | namespace Kernel { | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | class Semaphore : public Object { | 
					
						
							|  |  |  | public: | 
					
						
							|  |  |  |     std::string GetTypeName() const override { return "Semaphore"; } | 
					
						
							|  |  |  |     std::string GetName() const override { return name; } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     static Kernel::HandleType GetStaticHandleType() { return Kernel::HandleType::Semaphore; } | 
					
						
							|  |  |  |     Kernel::HandleType GetHandleType() const override { return Kernel::HandleType::Semaphore; } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-12-19 22:16:34 -05:00
										 |  |  |     s32 max_count;                              ///< Maximum number of simultaneous holders the semaphore can have
 | 
					
						
							|  |  |  |     s32 available_count;                        ///< Number of free slots left in the semaphore
 | 
					
						
							| 
									
										
										
										
											2014-12-04 11:40:36 -05:00
										 |  |  |     std::queue<Handle> waiting_threads;         ///< Threads that are waiting for the semaphore
 | 
					
						
							| 
									
										
										
										
											2014-12-03 18:49:51 -05:00
										 |  |  |     std::string name;                           ///< Name of semaphore (optional)
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-12-04 11:40:36 -05:00
										 |  |  |     /**
 | 
					
						
							| 
									
										
										
										
											2014-12-12 20:46:52 -05:00
										 |  |  |      * Tests whether a semaphore still has free slots | 
					
						
							|  |  |  |      * @return Whether the semaphore is available | 
					
						
							| 
									
										
										
										
											2014-12-04 11:40:36 -05:00
										 |  |  |      */ | 
					
						
							| 
									
										
										
										
											2014-12-12 20:46:52 -05:00
										 |  |  |     bool IsAvailable() const { | 
					
						
							| 
									
										
										
										
											2014-12-12 22:22:11 -05:00
										 |  |  |         return available_count > 0; | 
					
						
							| 
									
										
										
										
											2014-12-04 11:40:36 -05:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-12-03 18:49:51 -05:00
										 |  |  |     ResultVal<bool> WaitSynchronization() override { | 
					
						
							| 
									
										
										
										
											2014-12-13 10:29:11 -05:00
										 |  |  |         bool wait = !IsAvailable(); | 
					
						
							| 
									
										
										
										
											2014-12-03 18:49:51 -05:00
										 |  |  | 
 | 
					
						
							|  |  |  |         if (wait) { | 
					
						
							|  |  |  |             Kernel::WaitCurrentThread(WAITTYPE_SEMA, GetHandle()); | 
					
						
							| 
									
										
										
										
											2014-12-04 11:40:36 -05:00
										 |  |  |             waiting_threads.push(GetCurrentThreadHandle()); | 
					
						
							| 
									
										
										
										
											2014-12-03 18:49:51 -05:00
										 |  |  |         } else { | 
					
						
							| 
									
										
										
										
											2014-12-12 22:22:11 -05:00
										 |  |  |             --available_count; | 
					
						
							| 
									
										
										
										
											2014-12-03 18:49:51 -05:00
										 |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         return MakeResult<bool>(wait); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | ////////////////////////////////////////////////////////////////////////////////////////////////////
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-12-19 22:16:34 -05:00
										 |  |  | ResultCode CreateSemaphore(Handle* handle, s32 initial_count,  | 
					
						
							|  |  |  |     s32 max_count, const std::string& name) { | 
					
						
							| 
									
										
										
										
											2014-12-04 11:40:36 -05:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-12-12 20:46:52 -05:00
										 |  |  |     if (initial_count > max_count) | 
					
						
							|  |  |  |         return ResultCode(ErrorDescription::InvalidCombination, ErrorModule::Kernel, | 
					
						
							|  |  |  |                           ErrorSummary::WrongArgument, ErrorLevel::Permanent); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-12-03 18:49:51 -05:00
										 |  |  |     Semaphore* semaphore = new Semaphore; | 
					
						
							| 
									
										
										
										
											2014-12-12 20:46:52 -05:00
										 |  |  |     *handle = g_object_pool.Create(semaphore); | 
					
						
							| 
									
										
										
										
											2014-12-03 18:49:51 -05:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-12-04 11:55:13 -05:00
										 |  |  |     // When the semaphore is created, some slots are reserved for other threads,
 | 
					
						
							|  |  |  |     // and the rest is reserved for the caller thread
 | 
					
						
							| 
									
										
										
										
											2014-12-12 20:46:52 -05:00
										 |  |  |     semaphore->max_count = max_count; | 
					
						
							| 
									
										
										
										
											2014-12-12 22:22:11 -05:00
										 |  |  |     semaphore->available_count = initial_count; | 
					
						
							| 
									
										
										
										
											2014-12-03 18:49:51 -05:00
										 |  |  |     semaphore->name = name; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-12-04 11:40:36 -05:00
										 |  |  |     return RESULT_SUCCESS; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | ResultCode ReleaseSemaphore(s32* count, Handle handle, s32 release_count) { | 
					
						
							|  |  |  |     Semaphore* semaphore = g_object_pool.Get<Semaphore>(handle); | 
					
						
							|  |  |  |     if (semaphore == nullptr) | 
					
						
							|  |  |  |         return InvalidHandle(ErrorModule::Kernel); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-12-12 22:22:11 -05:00
										 |  |  |     if (semaphore->max_count - semaphore->available_count < release_count) | 
					
						
							| 
									
										
										
										
											2014-12-04 11:40:36 -05:00
										 |  |  |         return ResultCode(ErrorDescription::OutOfRange, ErrorModule::Kernel,  | 
					
						
							|  |  |  |                           ErrorSummary::InvalidArgument, ErrorLevel::Permanent); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-12-12 22:22:11 -05:00
										 |  |  |     *count = semaphore->available_count; | 
					
						
							|  |  |  |     semaphore->available_count += release_count; | 
					
						
							| 
									
										
										
										
											2014-12-04 11:40:36 -05:00
										 |  |  | 
 | 
					
						
							|  |  |  |     // Notify some of the threads that the semaphore has been released
 | 
					
						
							|  |  |  |     // stop once the semaphore is full again or there are no more waiting threads
 | 
					
						
							| 
									
										
										
										
											2014-12-12 20:46:52 -05:00
										 |  |  |     while (!semaphore->waiting_threads.empty() && semaphore->IsAvailable()) { | 
					
						
							| 
									
										
										
										
											2014-12-04 11:40:36 -05:00
										 |  |  |         Kernel::ResumeThreadFromWait(semaphore->waiting_threads.front()); | 
					
						
							|  |  |  |         semaphore->waiting_threads.pop(); | 
					
						
							| 
									
										
										
										
											2014-12-12 22:22:11 -05:00
										 |  |  |         --semaphore->available_count; | 
					
						
							| 
									
										
										
										
											2014-12-04 11:40:36 -05:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     return RESULT_SUCCESS; | 
					
						
							| 
									
										
										
										
											2014-12-03 18:49:51 -05:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | } // namespace
 |