| 
									
										
										
										
											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
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-12-22 04:32:03 -02:00
										 |  |  | #include <string>
 | 
					
						
							|  |  |  | #include <vector>
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-01-31 19:22:40 -02:00
										 |  |  | #include <boost/container/flat_set.hpp>
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-05-09 22:11:18 -04:00
										 |  |  | #include "common/common_types.h"
 | 
					
						
							| 
									
										
										
										
											2014-12-03 00:46:34 -05:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-12-22 04:32:03 -02:00
										 |  |  | #include "core/core.h"
 | 
					
						
							| 
									
										
										
										
											2014-12-03 00:46:34 -05:00
										 |  |  | #include "core/mem_map.h"
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-05-13 22:29:31 -04:00
										 |  |  | #include "core/hle/kernel/kernel.h"
 | 
					
						
							| 
									
										
										
										
											2014-10-23 01:20:01 -02:00
										 |  |  | #include "core/hle/result.h"
 | 
					
						
							| 
									
										
										
										
											2014-05-09 22:11:18 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-05-17 00:56:00 -04:00
										 |  |  | enum ThreadPriority { | 
					
						
							| 
									
										
										
										
											2014-05-20 21:02:35 -04:00
										 |  |  |     THREADPRIO_HIGHEST      = 0,    ///< Highest thread priority
 | 
					
						
							|  |  |  |     THREADPRIO_DEFAULT      = 16,   ///< Default thread priority for userland apps
 | 
					
						
							|  |  |  |     THREADPRIO_LOW          = 31,   ///< Low range of thread priority for userland apps
 | 
					
						
							|  |  |  |     THREADPRIO_LOWEST       = 63,   ///< Thread priority max checked by svcCreateThread
 | 
					
						
							| 
									
										
										
										
											2014-05-17 00:56:00 -04:00
										 |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | enum ThreadProcessorId { | 
					
						
							| 
									
										
										
										
											2014-05-20 21:02:35 -04:00
										 |  |  |     THREADPROCESSORID_0     = 0xFFFFFFFE,   ///< Enables core appcode
 | 
					
						
							|  |  |  |     THREADPROCESSORID_1     = 0xFFFFFFFD,   ///< Enables core syscore
 | 
					
						
							|  |  |  |     THREADPROCESSORID_ALL   = 0xFFFFFFFC,   ///< Enables both cores
 | 
					
						
							| 
									
										
										
										
											2014-05-17 00:56:00 -04:00
										 |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-05-22 18:50:36 -04:00
										 |  |  | enum ThreadStatus { | 
					
						
							|  |  |  |     THREADSTATUS_RUNNING        = 1, | 
					
						
							|  |  |  |     THREADSTATUS_READY          = 2, | 
					
						
							|  |  |  |     THREADSTATUS_WAIT           = 4, | 
					
						
							|  |  |  |     THREADSTATUS_SUSPEND        = 8, | 
					
						
							|  |  |  |     THREADSTATUS_DORMANT        = 16, | 
					
						
							|  |  |  |     THREADSTATUS_DEAD           = 32, | 
					
						
							|  |  |  |     THREADSTATUS_WAITSUSPEND    = THREADSTATUS_WAIT | THREADSTATUS_SUSPEND | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-05-20 19:37:46 -04:00
										 |  |  | namespace Kernel { | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-01-31 19:22:40 -02:00
										 |  |  | class Mutex; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-01-27 02:40:21 -02:00
										 |  |  | class Thread final : public WaitObject { | 
					
						
							| 
									
										
										
										
											2014-12-22 04:32:03 -02:00
										 |  |  | public: | 
					
						
							| 
									
										
										
										
											2014-12-29 10:55:30 -02:00
										 |  |  |     static ResultVal<SharedPtr<Thread>> Create(std::string name, VAddr entry_point, s32 priority, | 
					
						
							|  |  |  |         u32 arg, s32 processor_id, VAddr stack_top, u32 stack_size); | 
					
						
							| 
									
										
										
										
											2014-12-22 11:07:22 -02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-12-22 04:32:03 -02:00
										 |  |  |     std::string GetName() const override { return name; } | 
					
						
							|  |  |  |     std::string GetTypeName() const override { return "Thread"; } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     static const HandleType HANDLE_TYPE = HandleType::Thread; | 
					
						
							|  |  |  |     HandleType GetHandleType() const override { return HANDLE_TYPE; } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     inline bool IsRunning() const { return (status & THREADSTATUS_RUNNING) != 0; } | 
					
						
							|  |  |  |     inline bool IsStopped() const { return (status & THREADSTATUS_DORMANT) != 0; } | 
					
						
							|  |  |  |     inline bool IsReady() const { return (status & THREADSTATUS_READY) != 0; } | 
					
						
							|  |  |  |     inline bool IsWaiting() const { return (status & THREADSTATUS_WAIT) != 0; } | 
					
						
							|  |  |  |     inline bool IsSuspended() const { return (status & THREADSTATUS_SUSPEND) != 0; } | 
					
						
							|  |  |  |     inline bool IsIdle() const { return idle; } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-01-20 18:16:45 -05:00
										 |  |  |     bool ShouldWait() override; | 
					
						
							|  |  |  |     void Acquire() override; | 
					
						
							| 
									
										
										
										
											2014-12-22 04:32:03 -02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-12-22 11:07:22 -02:00
										 |  |  |     s32 GetPriority() const { return current_priority; } | 
					
						
							|  |  |  |     void SetPriority(s32 priority); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     u32 GetThreadId() const { return thread_id; } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     void Stop(const char* reason); | 
					
						
							| 
									
										
										
										
											2015-01-17 02:03:44 -05:00
										 |  |  |      | 
					
						
							|  |  |  |     /**
 | 
					
						
							| 
									
										
										
										
											2015-01-18 13:56:40 -05:00
										 |  |  |      * Release an acquired wait object | 
					
						
							|  |  |  |      * @param wait_object WaitObject to release | 
					
						
							| 
									
										
										
										
											2015-01-17 02:03:44 -05:00
										 |  |  |      */ | 
					
						
							| 
									
										
										
										
											2015-01-18 13:56:40 -05:00
										 |  |  |     void ReleaseWaitObject(WaitObject* wait_object); | 
					
						
							| 
									
										
										
										
											2015-01-17 02:03:44 -05:00
										 |  |  | 
 | 
					
						
							|  |  |  |     /// Resumes a thread from waiting by marking it as "ready"
 | 
					
						
							| 
									
										
										
										
											2014-12-22 11:07:22 -02:00
										 |  |  |     void ResumeFromWait(); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-01-30 23:07:54 -02:00
										 |  |  |     /**
 | 
					
						
							|  |  |  |     * Schedules an event to wake up the specified thread after the specified delay. | 
					
						
							|  |  |  |     * @param nanoseconds The time this thread will be allowed to sleep for. | 
					
						
							|  |  |  |     */ | 
					
						
							|  |  |  |     void WakeAfterDelay(s64 nanoseconds); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-01-17 02:03:44 -05:00
										 |  |  |     /**
 | 
					
						
							| 
									
										
										
										
											2015-01-20 20:53:52 -05:00
										 |  |  |      * Sets the result after the thread awakens (from either WaitSynchronization SVC) | 
					
						
							|  |  |  |      * @param result Value to set to the returned result | 
					
						
							| 
									
										
										
										
											2015-01-17 02:03:44 -05:00
										 |  |  |      */ | 
					
						
							| 
									
										
										
										
											2015-01-20 20:53:52 -05:00
										 |  |  |     void SetWaitSynchronizationResult(ResultCode result); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     /**
 | 
					
						
							|  |  |  |      * Sets the output parameter value after the thread awakens (from WaitSynchronizationN SVC only) | 
					
						
							|  |  |  |      * @param output Value to set to the output parameter | 
					
						
							|  |  |  |      */ | 
					
						
							|  |  |  |     void SetWaitSynchronizationOutput(s32 output); | 
					
						
							| 
									
										
										
										
											2015-01-17 02:03:44 -05:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-12-22 04:32:03 -02:00
										 |  |  |     Core::ThreadContext context; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     u32 thread_id; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     u32 status; | 
					
						
							|  |  |  |     u32 entry_point; | 
					
						
							|  |  |  |     u32 stack_top; | 
					
						
							|  |  |  |     u32 stack_size; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     s32 initial_priority; | 
					
						
							|  |  |  |     s32 current_priority; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     s32 processor_id; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-01-31 19:22:40 -02:00
										 |  |  |     /// Mutexes currently held by this thread, which will be released when it exits.
 | 
					
						
							|  |  |  |     boost::container::flat_set<SharedPtr<Mutex>> held_mutexes; | 
					
						
							| 
									
										
										
										
											2015-01-18 13:25:51 -05:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-01-31 19:22:40 -02:00
										 |  |  |     std::vector<SharedPtr<WaitObject>> wait_objects; ///< Objects that the thread is waiting on
 | 
					
						
							| 
									
										
										
										
											2015-01-20 20:53:52 -05:00
										 |  |  |     VAddr wait_address;     ///< If waiting on an AddressArbiter, this is the arbitration address
 | 
					
						
							|  |  |  |     bool wait_all;          ///< True if the thread is waiting on all objects before resuming
 | 
					
						
							|  |  |  |     bool wait_set_output;   ///< True if the output parameter should be set on thread wakeup
 | 
					
						
							| 
									
										
										
										
											2014-12-22 04:32:03 -02:00
										 |  |  | 
 | 
					
						
							|  |  |  |     std::string name; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     /// Whether this thread is intended to never actually be executed, i.e. always idle
 | 
					
						
							|  |  |  |     bool idle = false; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | private: | 
					
						
							| 
									
										
										
										
											2015-01-31 19:22:40 -02:00
										 |  |  |     Thread(); | 
					
						
							| 
									
										
										
										
											2015-01-31 22:56:59 -02:00
										 |  |  |     ~Thread() override; | 
					
						
							| 
									
										
										
										
											2015-01-31 14:23:09 -02:00
										 |  |  | 
 | 
					
						
							|  |  |  |     /// Handle used as userdata to reference this object when inserting into the CoreTiming queue.
 | 
					
						
							|  |  |  |     Handle callback_handle; | 
					
						
							| 
									
										
										
										
											2014-12-22 04:32:03 -02:00
										 |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-01-31 23:26:16 -02:00
										 |  |  | extern SharedPtr<Thread> g_main_thread; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-05-15 18:27:08 -04:00
										 |  |  | /// Sets up the primary application thread
 | 
					
						
							| 
									
										
										
										
											2014-12-29 10:55:30 -02:00
										 |  |  | SharedPtr<Thread> SetupMainThread(s32 priority, u32 stack_size); | 
					
						
							| 
									
										
										
										
											2014-05-13 22:00:11 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-05-19 22:19:48 -04:00
										 |  |  | /// Reschedules to the next available thread (call after current thread is suspended)
 | 
					
						
							| 
									
										
										
										
											2014-05-22 19:32:45 -04:00
										 |  |  | void Reschedule(); | 
					
						
							| 
									
										
										
										
											2014-05-19 22:19:48 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-07-06 22:48:19 -04:00
										 |  |  | /// Arbitrate the highest priority thread that is waiting
 | 
					
						
							| 
									
										
										
										
											2015-01-18 13:56:40 -05:00
										 |  |  | Thread* ArbitrateHighestPriorityThread(u32 address); | 
					
						
							| 
									
										
										
										
											2014-07-06 22:48:19 -04:00
										 |  |  | 
 | 
					
						
							|  |  |  | /// Arbitrate all threads currently waiting...
 | 
					
						
							| 
									
										
										
										
											2015-01-18 13:56:40 -05:00
										 |  |  | void ArbitrateAllThreads(u32 address); | 
					
						
							| 
									
										
										
										
											2014-07-06 22:48:19 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-12-22 04:32:03 -02:00
										 |  |  | /// Gets the current thread
 | 
					
						
							|  |  |  | Thread* GetCurrentThread(); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-01-18 13:56:40 -05:00
										 |  |  | /// Waits the current thread on a sleep
 | 
					
						
							|  |  |  | void WaitCurrentThread_Sleep(); | 
					
						
							| 
									
										
										
										
											2014-06-05 22:35:36 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-01-07 16:40:08 -05:00
										 |  |  | /**
 | 
					
						
							| 
									
										
										
										
											2015-01-17 02:03:44 -05:00
										 |  |  |  * Waits the current thread from a WaitSynchronization call | 
					
						
							|  |  |  |  * @param wait_object Kernel object that we are waiting on | 
					
						
							| 
									
										
										
										
											2015-01-20 20:53:52 -05:00
										 |  |  |  * @param wait_set_output If true, set the output parameter on thread wakeup (for WaitSynchronizationN only) | 
					
						
							| 
									
										
										
										
											2015-01-18 13:25:51 -05:00
										 |  |  |  * @param wait_all If true, wait on all objects before resuming (for WaitSynchronizationN only) | 
					
						
							| 
									
										
										
										
											2015-01-07 16:40:08 -05:00
										 |  |  |  */ | 
					
						
							| 
									
										
										
										
											2015-01-20 20:53:52 -05:00
										 |  |  | void WaitCurrentThread_WaitSynchronization(SharedPtr<WaitObject> wait_object, bool wait_set_output, bool wait_all); | 
					
						
							| 
									
										
										
										
											2015-01-07 16:40:08 -05:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-12-03 00:46:34 -05:00
										 |  |  | /**
 | 
					
						
							| 
									
										
										
										
											2015-01-17 02:03:44 -05:00
										 |  |  |  * Waits the current thread from an ArbitrateAddress call | 
					
						
							| 
									
										
										
										
											2014-12-03 00:46:34 -05:00
										 |  |  |  * @param wait_address Arbitration address used to resume from wait | 
					
						
							|  |  |  |  */ | 
					
						
							| 
									
										
										
										
											2015-01-18 13:56:40 -05:00
										 |  |  | void WaitCurrentThread_ArbitrateAddress(VAddr wait_address); | 
					
						
							| 
									
										
										
										
											2014-05-20 19:37:46 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-01-07 10:10:58 -05:00
										 |  |  | /**
 | 
					
						
							|  |  |  |  * Sets up the idle thread, this is a thread that is intended to never execute instructions, | 
					
						
							|  |  |  |  * only to advance the timing. It is scheduled when there are no other ready threads in the thread queue | 
					
						
							|  |  |  |  * and will try to yield on every call. | 
					
						
							|  |  |  |  * @returns The handle of the idle thread | 
					
						
							|  |  |  |  */ | 
					
						
							| 
									
										
										
										
											2015-01-30 23:07:54 -02:00
										 |  |  | SharedPtr<Thread> SetupIdleThread(); | 
					
						
							| 
									
										
										
										
											2015-01-18 13:25:51 -05:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-05-20 19:37:46 -04:00
										 |  |  | /// Initialize threading
 | 
					
						
							|  |  |  | void ThreadingInit(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /// Shutdown threading
 | 
					
						
							|  |  |  | void ThreadingShutdown(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | } // namespace
 |