| 
									
										
										
										
											2014-05-09 22:11:18 -04:00
										 |  |  | // Copyright 2014 Citra Emulator Project / PPSSPP 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-09 22:11:18 -04:00
										 |  |  | 
 | 
					
						
							|  |  |  | #pragma once
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-06-21 13:40:28 +01:00
										 |  |  | #include <algorithm>
 | 
					
						
							| 
									
										
										
										
											2014-08-19 00:20:56 -04:00
										 |  |  | #include <array>
 | 
					
						
							| 
									
										
										
										
											2015-06-21 13:40:28 +01:00
										 |  |  | #include <cstddef>
 | 
					
						
							| 
									
										
										
										
											2014-08-17 23:03:22 -04:00
										 |  |  | #include <string>
 | 
					
						
							| 
									
										
										
										
											2015-01-14 19:22:50 -05:00
										 |  |  | #include <vector>
 | 
					
						
							| 
									
										
										
										
											2016-09-21 00:21:23 +09:00
										 |  |  | #include <boost/smart_ptr/intrusive_ptr.hpp>
 | 
					
						
							| 
									
										
										
										
											2015-05-06 04:06:12 -03:00
										 |  |  | #include "common/common_types.h"
 | 
					
						
							| 
									
										
										
										
											2015-05-06 01:57:23 -03:00
										 |  |  | #include "core/hle/hle.h"
 | 
					
						
							| 
									
										
										
										
											2014-10-23 01:20:01 -02:00
										 |  |  | #include "core/hle/result.h"
 | 
					
						
							| 
									
										
										
										
											2014-05-09 22:11:18 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-05-20 18:13:25 -04:00
										 |  |  | namespace Kernel { | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-12-22 11:07:22 -02:00
										 |  |  | class Thread; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-12-21 10:04:08 -02:00
										 |  |  | // TODO: Verify code
 | 
					
						
							|  |  |  | const ResultCode ERR_OUT_OF_HANDLES(ErrorDescription::OutOfMemory, ErrorModule::Kernel, | 
					
						
							| 
									
										
										
										
											2016-09-18 09:38:01 +09:00
										 |  |  |                                     ErrorSummary::OutOfResource, ErrorLevel::Temporary); | 
					
						
							| 
									
										
										
										
											2014-12-21 10:04:08 -02:00
										 |  |  | // TOOD: Verify code
 | 
					
						
							| 
									
										
										
										
											2015-01-23 03:44:52 -02:00
										 |  |  | const ResultCode ERR_INVALID_HANDLE(ErrorDescription::InvalidHandle, ErrorModule::Kernel, | 
					
						
							| 
									
										
										
										
											2016-09-18 09:38:01 +09:00
										 |  |  |                                     ErrorSummary::InvalidArgument, ErrorLevel::Permanent); | 
					
						
							| 
									
										
										
										
											2014-12-20 03:04:36 -02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-12-21 10:04:08 -02:00
										 |  |  | enum KernelHandle : Handle { | 
					
						
							| 
									
										
										
										
											2016-09-18 09:38:01 +09:00
										 |  |  |     CurrentThread = 0xFFFF8000, | 
					
						
							|  |  |  |     CurrentProcess = 0xFFFF8001, | 
					
						
							| 
									
										
										
										
											2014-05-26 21:56:23 -04:00
										 |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-05-20 18:13:25 -04:00
										 |  |  | enum class HandleType : u32 { | 
					
						
							| 
									
										
										
										
											2016-09-18 09:38:01 +09:00
										 |  |  |     Unknown = 0, | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     Session = 2, | 
					
						
							|  |  |  |     Event = 3, | 
					
						
							|  |  |  |     Mutex = 4, | 
					
						
							|  |  |  |     SharedMemory = 5, | 
					
						
							|  |  |  |     Redirection = 6, | 
					
						
							|  |  |  |     Thread = 7, | 
					
						
							|  |  |  |     Process = 8, | 
					
						
							|  |  |  |     AddressArbiter = 9, | 
					
						
							|  |  |  |     Semaphore = 10, | 
					
						
							|  |  |  |     Timer = 11, | 
					
						
							|  |  |  |     ResourceLimit = 12, | 
					
						
							|  |  |  |     CodeSet = 13, | 
					
						
							|  |  |  |     ClientPort = 14, | 
					
						
							|  |  |  |     ServerPort = 15, | 
					
						
							| 
									
										
										
										
											2014-05-13 21:57:12 -04:00
										 |  |  | }; | 
					
						
							| 
									
										
										
										
											2014-11-19 08:49:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-05-15 18:26:28 -04:00
										 |  |  | enum { | 
					
						
							| 
									
										
										
										
											2016-09-18 09:38:01 +09:00
										 |  |  |     DEFAULT_STACK_SIZE = 0x4000, | 
					
						
							| 
									
										
										
										
											2014-05-15 18:26:28 -04:00
										 |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-05-25 13:36:15 +03:00
										 |  |  | enum class ResetType { | 
					
						
							|  |  |  |     OneShot, | 
					
						
							|  |  |  |     Sticky, | 
					
						
							|  |  |  |     Pulse, | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-05-20 18:13:25 -04:00
										 |  |  | class Object : NonCopyable { | 
					
						
							| 
									
										
										
										
											2014-05-09 22:11:18 -04:00
										 |  |  | public: | 
					
						
							| 
									
										
										
										
											2016-09-18 18:01:46 -07:00
										 |  |  |     virtual ~Object() {} | 
					
						
							| 
									
										
										
										
											2015-01-18 14:33:11 -05:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-01-31 14:55:40 -02:00
										 |  |  |     /// Returns a unique identifier for the object. For debugging purposes only.
 | 
					
						
							| 
									
										
										
										
											2016-09-18 09:38:01 +09:00
										 |  |  |     unsigned int GetObjectId() const { | 
					
						
							|  |  |  |         return object_id; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2015-01-31 14:55:40 -02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-09-18 09:38:01 +09:00
										 |  |  |     virtual std::string GetTypeName() const { | 
					
						
							|  |  |  |         return "[BAD KERNEL OBJECT TYPE]"; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     virtual std::string GetName() const { | 
					
						
							|  |  |  |         return "[UNKNOWN KERNEL OBJECT]"; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2014-05-20 18:13:25 -04:00
										 |  |  |     virtual Kernel::HandleType GetHandleType() const = 0; | 
					
						
							| 
									
										
										
										
											2014-05-26 22:12:46 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-05-26 22:17:49 -04:00
										 |  |  |     /**
 | 
					
						
							| 
									
										
										
										
											2015-01-18 14:33:11 -05:00
										 |  |  |      * Check if a thread can wait on the object | 
					
						
							|  |  |  |      * @return True if a thread can wait on the object, otherwise false | 
					
						
							| 
									
										
										
										
											2014-05-26 22:17:49 -04:00
										 |  |  |      */ | 
					
						
							| 
									
										
										
										
											2015-01-18 14:33:11 -05:00
										 |  |  |     bool IsWaitable() const { | 
					
						
							|  |  |  |         switch (GetHandleType()) { | 
					
						
							| 
									
										
										
										
											2015-01-18 20:40:53 -05:00
										 |  |  |         case HandleType::Session: | 
					
						
							| 
									
										
										
										
											2016-05-22 11:22:49 -05:00
										 |  |  |         case HandleType::ServerPort: | 
					
						
							| 
									
										
										
										
											2015-01-18 14:33:11 -05:00
										 |  |  |         case HandleType::Event: | 
					
						
							|  |  |  |         case HandleType::Mutex: | 
					
						
							|  |  |  |         case HandleType::Thread: | 
					
						
							|  |  |  |         case HandleType::Semaphore: | 
					
						
							|  |  |  |         case HandleType::Timer: | 
					
						
							|  |  |  |             return true; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         case HandleType::Unknown: | 
					
						
							|  |  |  |         case HandleType::SharedMemory: | 
					
						
							|  |  |  |         case HandleType::Redirection: | 
					
						
							|  |  |  |         case HandleType::Process: | 
					
						
							|  |  |  |         case HandleType::AddressArbiter: | 
					
						
							| 
									
										
										
										
											2015-05-21 20:54:49 +02:00
										 |  |  |         case HandleType::ResourceLimit: | 
					
						
							| 
									
										
										
										
											2015-07-12 18:14:26 -03:00
										 |  |  |         case HandleType::CodeSet: | 
					
						
							| 
									
										
										
										
											2016-05-22 11:22:49 -05:00
										 |  |  |         case HandleType::ClientPort: | 
					
						
							| 
									
										
										
										
											2015-01-18 14:33:11 -05:00
										 |  |  |             return false; | 
					
						
							|  |  |  |         } | 
					
						
							| 
									
										
										
										
											2014-12-05 23:53:49 -02:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2014-05-09 22:11:18 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-04-27 22:12:35 -04:00
										 |  |  | public: | 
					
						
							|  |  |  |     static unsigned int next_object_id; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-12-21 10:04:08 -02:00
										 |  |  | private: | 
					
						
							|  |  |  |     friend void intrusive_ptr_add_ref(Object*); | 
					
						
							|  |  |  |     friend void intrusive_ptr_release(Object*); | 
					
						
							| 
									
										
										
										
											2014-05-09 22:11:18 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-12-21 10:04:08 -02:00
										 |  |  |     unsigned int ref_count = 0; | 
					
						
							| 
									
										
										
										
											2015-01-31 14:55:40 -02:00
										 |  |  |     unsigned int object_id = next_object_id++; | 
					
						
							| 
									
										
										
										
											2014-12-21 10:04:08 -02:00
										 |  |  | }; | 
					
						
							| 
									
										
										
										
											2014-05-09 22:11:18 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-12-29 10:55:30 -02:00
										 |  |  | // Special functions used by boost::instrusive_ptr to do automatic ref-counting
 | 
					
						
							| 
									
										
										
										
											2014-12-21 10:04:08 -02:00
										 |  |  | inline void intrusive_ptr_add_ref(Object* object) { | 
					
						
							|  |  |  |     ++object->ref_count; | 
					
						
							|  |  |  | } | 
					
						
							| 
									
										
										
										
											2014-05-09 22:11:18 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-12-21 10:04:08 -02:00
										 |  |  | inline void intrusive_ptr_release(Object* object) { | 
					
						
							|  |  |  |     if (--object->ref_count == 0) { | 
					
						
							|  |  |  |         delete object; | 
					
						
							| 
									
										
										
										
											2014-11-18 08:27:16 -05:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2014-12-21 10:04:08 -02:00
										 |  |  | } | 
					
						
							| 
									
										
										
										
											2014-05-09 22:11:18 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-12-29 10:55:30 -02:00
										 |  |  | template <typename T> | 
					
						
							|  |  |  | using SharedPtr = boost::intrusive_ptr<T>; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-01-14 19:22:50 -05:00
										 |  |  | /// Class that represents a Kernel object that a thread can be waiting on
 | 
					
						
							|  |  |  | class WaitObject : public Object { | 
					
						
							|  |  |  | public: | 
					
						
							| 
									
										
										
										
											2015-01-18 14:33:11 -05:00
										 |  |  |     /**
 | 
					
						
							| 
									
										
										
										
											2015-01-20 17:41:12 -05:00
										 |  |  |      * Check if the current thread should wait until the object is available | 
					
						
							| 
									
										
										
										
											2015-01-18 14:33:11 -05:00
										 |  |  |      * @return True if the current thread should wait due to this object being unavailable | 
					
						
							|  |  |  |      */ | 
					
						
							| 
									
										
										
										
											2015-01-20 18:16:45 -05:00
										 |  |  |     virtual bool ShouldWait() = 0; | 
					
						
							| 
									
										
										
										
											2015-01-18 14:33:11 -05:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-01-20 18:16:45 -05:00
										 |  |  |     /// Acquire/lock the object if it is available
 | 
					
						
							|  |  |  |     virtual void Acquire() = 0; | 
					
						
							| 
									
										
										
										
											2015-01-18 14:33:11 -05:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-01-14 19:22:50 -05:00
										 |  |  |     /**
 | 
					
						
							|  |  |  |      * Add a thread to wait on this object | 
					
						
							|  |  |  |      * @param thread Pointer to thread to add | 
					
						
							|  |  |  |      */ | 
					
						
							| 
									
										
										
										
											2015-01-31 23:26:16 -02:00
										 |  |  |     void AddWaitingThread(SharedPtr<Thread> thread); | 
					
						
							| 
									
										
										
										
											2015-01-14 19:22:50 -05:00
										 |  |  | 
 | 
					
						
							|  |  |  |     /**
 | 
					
						
							| 
									
										
										
										
											2015-01-14 23:19:22 -05:00
										 |  |  |      * Removes a thread from waiting on this object (e.g. if it was resumed already) | 
					
						
							|  |  |  |      * @param thread Pointer to thread to remove | 
					
						
							|  |  |  |      */ | 
					
						
							| 
									
										
										
										
											2015-01-31 23:26:16 -02:00
										 |  |  |     void RemoveWaitingThread(Thread* thread); | 
					
						
							| 
									
										
										
										
											2015-01-14 23:19:22 -05:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-01-20 18:20:47 -05:00
										 |  |  |     /// Wake up all threads waiting on this object
 | 
					
						
							|  |  |  |     void WakeupAllWaitingThreads(); | 
					
						
							| 
									
										
										
										
											2015-01-14 19:22:50 -05:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-12-03 22:38:14 -05:00
										 |  |  |     /// Obtains the highest priority thread that is ready to run from this object's waiting list.
 | 
					
						
							|  |  |  |     SharedPtr<Thread> GetHighestPriorityReadyThread(); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-04-08 19:28:54 +03:00
										 |  |  |     /// Get a const reference to the waiting threads list for debug use
 | 
					
						
							|  |  |  |     const std::vector<SharedPtr<Thread>>& GetWaitingThreads() const; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-01-14 19:22:50 -05:00
										 |  |  | private: | 
					
						
							| 
									
										
										
										
											2015-01-31 23:26:16 -02:00
										 |  |  |     /// Threads waiting for this object to become available
 | 
					
						
							|  |  |  |     std::vector<SharedPtr<Thread>> waiting_threads; | 
					
						
							| 
									
										
										
										
											2015-01-14 19:22:50 -05:00
										 |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-12-21 10:04:08 -02:00
										 |  |  | /**
 | 
					
						
							|  |  |  |  * This class allows the creation of Handles, which are references to objects that can be tested | 
					
						
							|  |  |  |  * for validity and looked up. Here they are used to pass references to kernel objects to/from the | 
					
						
							|  |  |  |  * emulated process. it has been designed so that it follows the same handle format and has | 
					
						
							|  |  |  |  * approximately the same restrictions as the handle manager in the CTR-OS. | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * Handles contain two sub-fields: a slot index (bits 31:15) and a generation value (bits 14:0). | 
					
						
							|  |  |  |  * The slot index is used to index into the arrays in this class to access the data corresponding | 
					
						
							|  |  |  |  * to the Handle. | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * To prevent accidental use of a freed Handle whose slot has already been reused, a global counter | 
					
						
							|  |  |  |  * is kept and incremented every time a Handle is created. This is the Handle's "generation". The | 
					
						
							|  |  |  |  * value of the counter is stored into the Handle as well as in the handle table (in the | 
					
						
							|  |  |  |  * "generations" array). When looking up a handle, the Handle's generation must match with the | 
					
						
							|  |  |  |  * value stored on the class, otherwise the Handle is considered invalid. | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * To find free slots when allocating a Handle without needing to scan the entire object array, the | 
					
						
							|  |  |  |  * generations field of unallocated slots is re-purposed as a linked list of indices to free slots. | 
					
						
							|  |  |  |  * When a Handle is created, an index is popped off the list and used for the new Handle. When it | 
					
						
							|  |  |  |  * is destroyed, it is again pushed onto the list to be re-used by the next allocation. It is | 
					
						
							|  |  |  |  * likely that this allocation strategy differs from the one used in CTR-OS, but this hasn't been | 
					
						
							|  |  |  |  * verified and isn't likely to cause any problems. | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | class HandleTable final : NonCopyable { | 
					
						
							|  |  |  | public: | 
					
						
							|  |  |  |     HandleTable(); | 
					
						
							| 
									
										
										
										
											2014-05-09 22:11:18 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-12-21 10:04:08 -02:00
										 |  |  |     /**
 | 
					
						
							|  |  |  |      * Allocates a handle for the given object. | 
					
						
							|  |  |  |      * @return The created Handle or one of the following errors: | 
					
						
							|  |  |  |      *           - `ERR_OUT_OF_HANDLES`: the maximum number of handles has been exceeded. | 
					
						
							|  |  |  |      */ | 
					
						
							| 
									
										
										
										
											2014-12-29 10:55:30 -02:00
										 |  |  |     ResultVal<Handle> Create(SharedPtr<Object> obj); | 
					
						
							| 
									
										
										
										
											2014-12-20 03:04:36 -02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-12-21 10:04:08 -02:00
										 |  |  |     /**
 | 
					
						
							|  |  |  |      * Returns a new handle that points to the same object as the passed in handle. | 
					
						
							|  |  |  |      * @return The duplicated Handle or one of the following errors: | 
					
						
							|  |  |  |      *           - `ERR_INVALID_HANDLE`: an invalid handle was passed in. | 
					
						
							|  |  |  |      *           - Any errors returned by `Create()`. | 
					
						
							|  |  |  |      */ | 
					
						
							|  |  |  |     ResultVal<Handle> Duplicate(Handle handle); | 
					
						
							| 
									
										
										
										
											2014-05-09 22:11:18 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-12-21 10:04:08 -02:00
										 |  |  |     /**
 | 
					
						
							|  |  |  |      * Closes a handle, removing it from the table and decreasing the object's ref-count. | 
					
						
							|  |  |  |      * @return `RESULT_SUCCESS` or one of the following errors: | 
					
						
							|  |  |  |      *           - `ERR_INVALID_HANDLE`: an invalid handle was passed in. | 
					
						
							|  |  |  |      */ | 
					
						
							|  |  |  |     ResultCode Close(Handle handle); | 
					
						
							| 
									
										
										
										
											2014-12-20 03:04:36 -02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-12-21 10:04:08 -02:00
										 |  |  |     /// Checks if a handle is valid and points to an existing object.
 | 
					
						
							|  |  |  |     bool IsValid(Handle handle) const; | 
					
						
							| 
									
										
										
										
											2014-05-09 22:11:18 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-12-21 10:04:08 -02:00
										 |  |  |     /**
 | 
					
						
							|  |  |  |      * Looks up a handle. | 
					
						
							| 
									
										
										
										
											2015-01-18 14:33:11 -05:00
										 |  |  |      * @return Pointer to the looked-up object, or `nullptr` if the handle is not valid. | 
					
						
							| 
									
										
										
										
											2014-12-21 10:04:08 -02:00
										 |  |  |      */ | 
					
						
							| 
									
										
										
										
											2014-12-29 10:55:30 -02:00
										 |  |  |     SharedPtr<Object> GetGeneric(Handle handle) const; | 
					
						
							| 
									
										
										
										
											2014-05-09 22:11:18 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-12-21 10:04:08 -02:00
										 |  |  |     /**
 | 
					
						
							|  |  |  |      * Looks up a handle while verifying its type. | 
					
						
							| 
									
										
										
										
											2015-01-18 14:33:11 -05:00
										 |  |  |      * @return Pointer to the looked-up object, or `nullptr` if the handle is not valid or its | 
					
						
							|  |  |  |      *         type differs from the handle type `T::HANDLE_TYPE`. | 
					
						
							| 
									
										
										
										
											2014-12-21 10:04:08 -02:00
										 |  |  |      */ | 
					
						
							|  |  |  |     template <class T> | 
					
						
							| 
									
										
										
										
											2014-12-29 10:55:30 -02:00
										 |  |  |     SharedPtr<T> Get(Handle handle) const { | 
					
						
							|  |  |  |         SharedPtr<Object> object = GetGeneric(handle); | 
					
						
							| 
									
										
										
										
											2014-12-21 10:04:08 -02:00
										 |  |  |         if (object != nullptr && object->GetHandleType() == T::HANDLE_TYPE) { | 
					
						
							| 
									
										
										
										
											2014-12-29 10:55:30 -02:00
										 |  |  |             return boost::static_pointer_cast<T>(std::move(object)); | 
					
						
							| 
									
										
										
										
											2014-05-09 22:11:18 -04:00
										 |  |  |         } | 
					
						
							| 
									
										
										
										
											2014-12-21 10:04:08 -02:00
										 |  |  |         return nullptr; | 
					
						
							| 
									
										
										
										
											2014-05-09 22:11:18 -04:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-01-18 14:33:11 -05:00
										 |  |  |     /**
 | 
					
						
							|  |  |  |      * Looks up a handle while verifying that it is an object that a thread can wait on | 
					
						
							|  |  |  |      * @return Pointer to the looked-up object, or `nullptr` if the handle is not valid or it is | 
					
						
							|  |  |  |      *         not a waitable object. | 
					
						
							|  |  |  |      */ | 
					
						
							|  |  |  |     SharedPtr<WaitObject> GetWaitObject(Handle handle) const { | 
					
						
							|  |  |  |         SharedPtr<Object> object = GetGeneric(handle); | 
					
						
							|  |  |  |         if (object != nullptr && object->IsWaitable()) { | 
					
						
							|  |  |  |             return boost::static_pointer_cast<WaitObject>(std::move(object)); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         return nullptr; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-12-21 10:04:08 -02:00
										 |  |  |     /// Closes all handles held in this table.
 | 
					
						
							| 
									
										
										
										
											2014-05-09 22:11:18 -04:00
										 |  |  |     void Clear(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | private: | 
					
						
							| 
									
										
										
										
											2014-12-21 10:04:08 -02:00
										 |  |  |     /**
 | 
					
						
							|  |  |  |      * This is the maximum limit of handles allowed per process in CTR-OS. It can be further | 
					
						
							|  |  |  |      * reduced by ExHeader values, but this is not emulated here. | 
					
						
							|  |  |  |      */ | 
					
						
							|  |  |  |     static const size_t MAX_COUNT = 4096; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-09-18 09:38:01 +09:00
										 |  |  |     static u16 GetSlot(Handle handle) { | 
					
						
							|  |  |  |         return handle >> 15; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     static u16 GetGeneration(Handle handle) { | 
					
						
							|  |  |  |         return handle & 0x7FFF; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2014-12-21 10:04:08 -02:00
										 |  |  | 
 | 
					
						
							|  |  |  |     /// Stores the Object referenced by the handle or null if the slot is empty.
 | 
					
						
							| 
									
										
										
										
											2014-12-29 10:55:30 -02:00
										 |  |  |     std::array<SharedPtr<Object>, MAX_COUNT> objects; | 
					
						
							| 
									
										
										
										
											2014-11-19 08:49:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-12-21 10:04:08 -02:00
										 |  |  |     /**
 | 
					
						
							|  |  |  |      * The value of `next_generation` when the handle was created, used to check for validity. For | 
					
						
							|  |  |  |      * empty slots, contains the index of the next free slot in the list. | 
					
						
							|  |  |  |      */ | 
					
						
							|  |  |  |     std::array<u16, MAX_COUNT> generations; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     /**
 | 
					
						
							|  |  |  |      * Global counter of the number of created handles. Stored in `generations` when a handle is | 
					
						
							|  |  |  |      * created, and wraps around to 1 when it hits 0x8000. | 
					
						
							|  |  |  |      */ | 
					
						
							|  |  |  |     u16 next_generation; | 
					
						
							| 
									
										
										
										
											2014-05-20 18:13:25 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-12-21 10:04:08 -02:00
										 |  |  |     /// Head of the free slots linked list.
 | 
					
						
							|  |  |  |     u16 next_free_slot; | 
					
						
							| 
									
										
										
										
											2014-05-09 22:11:18 -04:00
										 |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-12-13 21:16:13 -02:00
										 |  |  | extern HandleTable g_handle_table; | 
					
						
							| 
									
										
										
										
											2014-05-20 18:13:25 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-11-19 20:40:04 -05:00
										 |  |  | /// Initialize the kernel with the specified system mode.
 | 
					
						
							|  |  |  | void Init(u32 system_mode); | 
					
						
							| 
									
										
										
										
											2014-06-10 22:43:50 -04:00
										 |  |  | 
 | 
					
						
							|  |  |  | /// Shutdown the kernel
 | 
					
						
							|  |  |  | void Shutdown(); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-05-22 19:06:12 -04:00
										 |  |  | } // namespace
 |