| 
									
										
										
										
											2014-12-03 18:49:51 -05:00
										 |  |  | // Copyright 2014 Citra Emulator Project
 | 
					
						
							| 
									
										
										
										
											2014-12-16 21:38:14 -08:00
										 |  |  | // Licensed under GPLv2 or any later version
 | 
					
						
							| 
									
										
										
										
											2014-12-03 18:49:51 -05:00
										 |  |  | // Refer to the license.txt file included.
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-05-06 04:06:12 -03:00
										 |  |  | #include "common/assert.h"
 | 
					
						
							| 
									
										
										
										
											2017-05-21 00:11:36 -07:00
										 |  |  | #include "core/hle/kernel/errors.h"
 | 
					
						
							| 
									
										
										
										
											2014-12-03 18:49:51 -05:00
										 |  |  | #include "core/hle/kernel/kernel.h"
 | 
					
						
							| 
									
										
										
										
											2016-09-20 23:52:38 -07:00
										 |  |  | #include "core/hle/kernel/semaphore.h"
 | 
					
						
							| 
									
										
										
										
											2014-12-03 18:49:51 -05:00
										 |  |  | #include "core/hle/kernel/thread.h"
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | namespace Kernel { | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-09-18 18:01:46 -07:00
										 |  |  | Semaphore::Semaphore() {} | 
					
						
							|  |  |  | Semaphore::~Semaphore() {} | 
					
						
							| 
									
										
										
										
											2015-01-31 22:56:59 -02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-01-11 13:53:11 -02:00
										 |  |  | ResultVal<SharedPtr<Semaphore>> Semaphore::Create(s32 initial_count, s32 max_count, | 
					
						
							| 
									
										
										
										
											2016-09-18 09:38:01 +09:00
										 |  |  |                                                   std::string name) { | 
					
						
							| 
									
										
										
										
											2014-12-04 11:40:36 -05:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-12-12 20:46:52 -05:00
										 |  |  |     if (initial_count > max_count) | 
					
						
							| 
									
										
										
										
											2017-05-21 00:11:36 -07:00
										 |  |  |         return ERR_INVALID_COMBINATION_KERNEL; | 
					
						
							| 
									
										
										
										
											2014-12-12 20:46:52 -05:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-01-11 13:53:11 -02:00
										 |  |  |     SharedPtr<Semaphore> semaphore(new 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; | 
					
						
							| 
									
										
										
										
											2015-01-11 13:53:11 -02:00
										 |  |  |     semaphore->name = std::move(name); | 
					
						
							| 
									
										
										
										
											2014-12-03 18:49:51 -05:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-01-11 13:53:11 -02:00
										 |  |  |     return MakeResult<SharedPtr<Semaphore>>(std::move(semaphore)); | 
					
						
							| 
									
										
										
										
											2014-12-04 11:40:36 -05:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-01-01 16:53:22 -05:00
										 |  |  | bool Semaphore::ShouldWait(Thread* thread) const { | 
					
						
							| 
									
										
										
										
											2015-01-11 13:53:11 -02:00
										 |  |  |     return available_count <= 0; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-01-01 16:53:22 -05:00
										 |  |  | void Semaphore::Acquire(Thread* thread) { | 
					
						
							| 
									
										
										
										
											2017-01-01 19:07:37 -05:00
										 |  |  |     if (available_count <= 0) | 
					
						
							|  |  |  |         return; | 
					
						
							| 
									
										
										
										
											2015-01-11 13:53:11 -02:00
										 |  |  |     --available_count; | 
					
						
							|  |  |  | } | 
					
						
							| 
									
										
										
										
											2014-12-04 11:40:36 -05:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-01-11 13:53:11 -02:00
										 |  |  | ResultVal<s32> Semaphore::Release(s32 release_count) { | 
					
						
							|  |  |  |     if (max_count - available_count < release_count) | 
					
						
							| 
									
										
										
										
											2017-05-21 00:11:36 -07:00
										 |  |  |         return ERR_OUT_OF_RANGE_KERNEL; | 
					
						
							| 
									
										
										
										
											2014-12-04 11:40:36 -05:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-01-11 13:53:11 -02:00
										 |  |  |     s32 previous_count = available_count; | 
					
						
							|  |  |  |     available_count += release_count; | 
					
						
							| 
									
										
										
										
											2014-12-04 11:40:36 -05:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-06-07 23:39:37 -04:00
										 |  |  |     WakeupAllWaitingThreads(); | 
					
						
							| 
									
										
										
										
											2015-05-19 20:24:30 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-01-11 13:53:11 -02:00
										 |  |  |     return MakeResult<s32>(previous_count); | 
					
						
							| 
									
										
										
										
											2014-12-03 18:49:51 -05:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | } // namespace
 |