| 
									
										
										
										
											2022-04-23 04:59:50 -04:00
										 |  |  | // SPDX-FileCopyrightText: Copyright 2021 yuzu Emulator Project
 | 
					
						
							|  |  |  | // SPDX-License-Identifier: GPL-2.0-or-later
 | 
					
						
							| 
									
										
										
										
											2014-05-09 22:11:18 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-06-05 22:35:36 -04:00
										 |  |  | #include <algorithm>
 | 
					
						
							| 
									
										
										
										
											2021-12-29 21:40:38 -08:00
										 |  |  | #include <atomic>
 | 
					
						
							| 
									
										
										
										
											2018-02-14 00:33:15 -05:00
										 |  |  | #include <cinttypes>
 | 
					
						
							| 
									
										
										
										
											2018-10-30 05:03:25 +01:00
										 |  |  | #include <optional>
 | 
					
						
							| 
									
										
										
										
											2014-08-17 23:03:22 -04:00
										 |  |  | #include <vector>
 | 
					
						
							| 
									
										
										
										
											2018-07-31 08:06:09 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-05-06 04:06:12 -03:00
										 |  |  | #include "common/assert.h"
 | 
					
						
							| 
									
										
										
										
											2021-01-20 13:42:27 -08:00
										 |  |  | #include "common/bit_util.h"
 | 
					
						
							| 
									
										
										
										
											2021-01-01 02:06:06 -08:00
										 |  |  | #include "common/common_funcs.h"
 | 
					
						
							| 
									
										
										
										
											2015-05-06 04:06:12 -03:00
										 |  |  | #include "common/common_types.h"
 | 
					
						
							| 
									
										
										
										
											2020-02-24 22:04:12 -04:00
										 |  |  | #include "common/fiber.h"
 | 
					
						
							| 
									
										
										
										
											2015-05-06 04:06:12 -03:00
										 |  |  | #include "common/logging/log.h"
 | 
					
						
							| 
									
										
										
										
											2021-08-14 02:14:19 -07:00
										 |  |  | #include "common/settings.h"
 | 
					
						
							| 
									
										
										
										
											2014-05-13 22:00:11 -04:00
										 |  |  | #include "core/core.h"
 | 
					
						
							| 
									
										
										
										
											2020-02-24 22:04:12 -04:00
										 |  |  | #include "core/cpu_manager.h"
 | 
					
						
							| 
									
										
										
										
											2020-02-11 19:56:24 -04:00
										 |  |  | #include "core/hardware_properties.h"
 | 
					
						
							| 
									
										
										
										
											2020-12-30 01:14:02 -08:00
										 |  |  | #include "core/hle/kernel/k_condition_variable.h"
 | 
					
						
							| 
									
										
										
										
											2021-04-24 02:40:31 -07:00
										 |  |  | #include "core/hle/kernel/k_handle_table.h"
 | 
					
						
							| 
									
										
										
										
											2021-02-12 16:02:35 -08:00
										 |  |  | #include "core/hle/kernel/k_memory_layout.h"
 | 
					
						
							| 
									
										
										
										
											2021-04-23 22:04:28 -07:00
										 |  |  | #include "core/hle/kernel/k_process.h"
 | 
					
						
							| 
									
										
										
										
											2021-01-30 20:40:49 +11:00
										 |  |  | #include "core/hle/kernel/k_resource_limit.h"
 | 
					
						
							| 
									
										
										
										
											2020-12-02 18:08:35 -08:00
										 |  |  | #include "core/hle/kernel/k_scheduler.h"
 | 
					
						
							| 
									
										
										
										
											2020-12-03 21:56:02 -08:00
										 |  |  | #include "core/hle/kernel/k_scoped_scheduler_lock_and_sleep.h"
 | 
					
						
							| 
									
										
										
										
											2021-12-23 00:03:39 -08:00
										 |  |  | #include "core/hle/kernel/k_system_control.h"
 | 
					
						
							| 
									
										
										
										
											2020-12-30 23:01:08 -08:00
										 |  |  | #include "core/hle/kernel/k_thread.h"
 | 
					
						
							| 
									
										
										
										
											2021-01-20 13:42:27 -08:00
										 |  |  | #include "core/hle/kernel/k_thread_queue.h"
 | 
					
						
							| 
									
										
										
										
											2022-01-14 16:36:10 -08:00
										 |  |  | #include "core/hle/kernel/k_worker_task_manager.h"
 | 
					
						
							| 
									
										
										
										
											2018-08-28 12:30:33 -04:00
										 |  |  | #include "core/hle/kernel/kernel.h"
 | 
					
						
							| 
									
										
										
										
											2023-02-18 16:26:48 -05:00
										 |  |  | #include "core/hle/kernel/svc.h"
 | 
					
						
							| 
									
										
										
										
											2021-01-01 02:06:06 -08:00
										 |  |  | #include "core/hle/kernel/svc_results.h"
 | 
					
						
							| 
									
										
										
										
											2022-09-05 18:19:30 -07:00
										 |  |  | #include "core/hle/kernel/svc_types.h"
 | 
					
						
							| 
									
										
										
										
											2014-10-23 01:20:01 -02:00
										 |  |  | #include "core/hle/result.h"
 | 
					
						
							| 
									
										
										
										
											2021-12-29 21:40:38 -08:00
										 |  |  | #include "core/memory.h"
 | 
					
						
							| 
									
										
										
										
											2014-05-09 22:11:18 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-07-15 13:13:31 -04:00
										 |  |  | #ifdef ARCHITECTURE_x86_64
 | 
					
						
							|  |  |  | #include "core/arm/dynarmic/arm_dynarmic_32.h"
 | 
					
						
							|  |  |  | #endif
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-01-20 13:42:27 -08:00
										 |  |  | namespace { | 
					
						
							| 
									
										
										
										
											2022-09-05 18:19:30 -07:00
										 |  |  | 
 | 
					
						
							|  |  |  | constexpr inline s32 TerminatingThreadPriority = Kernel::Svc::SystemThreadPriorityHighest - 1; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-01-20 13:42:27 -08:00
										 |  |  | static void ResetThreadContext32(Core::ARM_Interface::ThreadContext32& context, u32 stack_top, | 
					
						
							|  |  |  |                                  u32 entry_point, u32 arg) { | 
					
						
							|  |  |  |     context = {}; | 
					
						
							|  |  |  |     context.cpu_registers[0] = arg; | 
					
						
							|  |  |  |     context.cpu_registers[15] = entry_point; | 
					
						
							|  |  |  |     context.cpu_registers[13] = stack_top; | 
					
						
							| 
									
										
										
										
											2023-03-07 19:18:06 -05:00
										 |  |  |     context.fpscr = 0; | 
					
						
							| 
									
										
										
										
											2021-01-20 13:42:27 -08:00
										 |  |  | } | 
					
						
							| 
									
										
										
										
											2014-05-13 22:00:11 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-01-20 13:42:27 -08:00
										 |  |  | static void ResetThreadContext64(Core::ARM_Interface::ThreadContext64& context, VAddr stack_top, | 
					
						
							|  |  |  |                                  VAddr entry_point, u64 arg) { | 
					
						
							|  |  |  |     context = {}; | 
					
						
							|  |  |  |     context.cpu_registers[0] = arg; | 
					
						
							| 
									
										
										
										
											2021-12-23 00:03:39 -08:00
										 |  |  |     context.cpu_registers[18] = Kernel::KSystemControl::GenerateRandomU64() | 1; | 
					
						
							| 
									
										
										
										
											2021-01-20 13:42:27 -08:00
										 |  |  |     context.pc = entry_point; | 
					
						
							|  |  |  |     context.sp = stack_top; | 
					
						
							|  |  |  |     context.fpcr = 0; | 
					
						
							| 
									
										
										
										
											2023-03-07 19:18:06 -05:00
										 |  |  |     context.fpsr = 0; | 
					
						
							| 
									
										
										
										
											2020-02-11 17:36:39 -04:00
										 |  |  | } | 
					
						
							| 
									
										
										
										
											2021-01-20 13:42:27 -08:00
										 |  |  | } // namespace
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | namespace Kernel { | 
					
						
							| 
									
										
										
										
											2020-02-11 17:36:39 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-11-10 22:28:30 -08:00
										 |  |  | namespace { | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-12-29 21:40:38 -08:00
										 |  |  | struct ThreadLocalRegion { | 
					
						
							|  |  |  |     static constexpr std::size_t MessageBufferSize = 0x100; | 
					
						
							|  |  |  |     std::array<u32, MessageBufferSize / sizeof(u32)> message_buffer; | 
					
						
							|  |  |  |     std::atomic_uint16_t disable_count; | 
					
						
							|  |  |  |     std::atomic_uint16_t interrupt_flag; | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-11-10 22:28:30 -08:00
										 |  |  | class ThreadQueueImplForKThreadSleep final : public KThreadQueueWithoutEndWait { | 
					
						
							|  |  |  | public: | 
					
						
							| 
									
										
										
										
											2023-03-07 10:49:41 -05:00
										 |  |  |     explicit ThreadQueueImplForKThreadSleep(KernelCore& kernel) | 
					
						
							|  |  |  |         : KThreadQueueWithoutEndWait(kernel) {} | 
					
						
							| 
									
										
										
										
											2021-11-10 22:28:30 -08:00
										 |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | class ThreadQueueImplForKThreadSetProperty final : public KThreadQueue { | 
					
						
							|  |  |  | public: | 
					
						
							| 
									
										
										
										
											2023-03-07 10:49:41 -05:00
										 |  |  |     explicit ThreadQueueImplForKThreadSetProperty(KernelCore& kernel, KThread::WaiterList* wl) | 
					
						
							|  |  |  |         : KThreadQueue(kernel), m_wait_list(wl) {} | 
					
						
							| 
									
										
										
										
											2021-11-10 22:28:30 -08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-06-25 22:44:19 -05:00
										 |  |  |     void CancelWait(KThread* waiting_thread, Result wait_result, bool cancel_timer_task) override { | 
					
						
							| 
									
										
										
										
											2021-11-10 22:28:30 -08:00
										 |  |  |         // Remove the thread from the wait list.
 | 
					
						
							|  |  |  |         m_wait_list->erase(m_wait_list->iterator_to(*waiting_thread)); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         // Invoke the base cancel wait handler.
 | 
					
						
							|  |  |  |         KThreadQueue::CancelWait(waiting_thread, wait_result, cancel_timer_task); | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2021-11-21 02:29:53 -08:00
										 |  |  | 
 | 
					
						
							|  |  |  | private: | 
					
						
							|  |  |  |     KThread::WaiterList* m_wait_list; | 
					
						
							| 
									
										
										
										
											2021-11-10 22:28:30 -08:00
										 |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | } // namespace
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-03-07 10:49:41 -05:00
										 |  |  | KThread::KThread(KernelCore& kernel) | 
					
						
							|  |  |  |     : KAutoObjectWithSlabHeapAndContainer{kernel}, activity_pause_lock{kernel} {} | 
					
						
							| 
									
										
										
										
											2020-12-30 23:01:08 -08:00
										 |  |  | KThread::~KThread() = default; | 
					
						
							| 
									
										
										
										
											2015-01-31 19:22:40 -02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-06-25 22:44:19 -05:00
										 |  |  | Result KThread::Initialize(KThreadFunction func, uintptr_t arg, VAddr user_stack_top, s32 prio, | 
					
						
							|  |  |  |                            s32 virt_core, KProcess* owner, ThreadType type) { | 
					
						
							| 
									
										
										
										
											2021-01-20 13:42:27 -08:00
										 |  |  |     // Assert parameters are valid.
 | 
					
						
							| 
									
										
										
										
											2022-01-17 18:48:14 -08:00
										 |  |  |     ASSERT((type == ThreadType::Main) || (type == ThreadType::Dummy) || | 
					
						
							| 
									
										
										
										
											2021-01-20 13:42:27 -08:00
										 |  |  |            (Svc::HighestThreadPriority <= prio && prio <= Svc::LowestThreadPriority)); | 
					
						
							|  |  |  |     ASSERT((owner != nullptr) || (type != ThreadType::User)); | 
					
						
							|  |  |  |     ASSERT(0 <= virt_core && virt_core < static_cast<s32>(Common::BitSize<u64>())); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     // Convert the virtual core to a physical core.
 | 
					
						
							|  |  |  |     const s32 phys_core = Core::Hardware::VirtualToPhysicalCoreMap[virt_core]; | 
					
						
							|  |  |  |     ASSERT(0 <= phys_core && phys_core < static_cast<s32>(Core::Hardware::NUM_CPU_CORES)); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     // First, clear the TLS address.
 | 
					
						
							|  |  |  |     tls_address = {}; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     // Next, assert things based on the type.
 | 
					
						
							|  |  |  |     switch (type) { | 
					
						
							|  |  |  |     case ThreadType::Main: | 
					
						
							|  |  |  |         ASSERT(arg == 0); | 
					
						
							|  |  |  |         [[fallthrough]]; | 
					
						
							|  |  |  |     case ThreadType::HighPriority: | 
					
						
							|  |  |  |         [[fallthrough]]; | 
					
						
							| 
									
										
										
										
											2021-11-26 15:10:21 -08:00
										 |  |  |     case ThreadType::Dummy: | 
					
						
							|  |  |  |         [[fallthrough]]; | 
					
						
							| 
									
										
										
										
											2021-01-20 13:42:27 -08:00
										 |  |  |     case ThreadType::User: | 
					
						
							|  |  |  |         ASSERT(((owner == nullptr) || | 
					
						
							|  |  |  |                 (owner->GetCoreMask() | (1ULL << virt_core)) == owner->GetCoreMask())); | 
					
						
							|  |  |  |         ASSERT(((owner == nullptr) || | 
					
						
							|  |  |  |                 (owner->GetPriorityMask() | (1ULL << prio)) == owner->GetPriorityMask())); | 
					
						
							|  |  |  |         break; | 
					
						
							|  |  |  |     case ThreadType::Kernel: | 
					
						
							|  |  |  |         UNIMPLEMENTED(); | 
					
						
							|  |  |  |         break; | 
					
						
							|  |  |  |     default: | 
					
						
							| 
									
										
										
										
											2022-06-07 17:02:29 -04:00
										 |  |  |         ASSERT_MSG(false, "KThread::Initialize: Unknown ThreadType {}", static_cast<u32>(type)); | 
					
						
							| 
									
										
										
										
											2021-01-20 13:42:27 -08:00
										 |  |  |         break; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2022-01-17 16:41:06 -08:00
										 |  |  |     thread_type = type; | 
					
						
							| 
									
										
										
										
											2021-01-20 13:42:27 -08:00
										 |  |  | 
 | 
					
						
							|  |  |  |     // Set the ideal core ID and affinity mask.
 | 
					
						
							|  |  |  |     virtual_ideal_core_id = virt_core; | 
					
						
							|  |  |  |     physical_ideal_core_id = phys_core; | 
					
						
							| 
									
										
										
										
											2021-01-24 22:55:08 -08:00
										 |  |  |     virtual_affinity_mask = 1ULL << virt_core; | 
					
						
							| 
									
										
										
										
											2021-01-20 13:42:27 -08:00
										 |  |  |     physical_affinity_mask.SetAffinity(phys_core, true); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     // Set the thread state.
 | 
					
						
							| 
									
										
										
										
											2022-10-23 05:24:38 -04:00
										 |  |  |     thread_state = (type == ThreadType::Main || type == ThreadType::Dummy) | 
					
						
							|  |  |  |                        ? ThreadState::Runnable | 
					
						
							|  |  |  |                        : ThreadState::Initialized; | 
					
						
							| 
									
										
										
										
											2021-01-20 13:42:27 -08:00
										 |  |  | 
 | 
					
						
							|  |  |  |     // Set TLS address.
 | 
					
						
							|  |  |  |     tls_address = 0; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     // Set parent and condvar tree.
 | 
					
						
							|  |  |  |     parent = nullptr; | 
					
						
							|  |  |  |     condvar_tree = nullptr; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     // Set sync booleans.
 | 
					
						
							|  |  |  |     signaled = false; | 
					
						
							|  |  |  |     termination_requested = false; | 
					
						
							|  |  |  |     wait_cancelled = false; | 
					
						
							|  |  |  |     cancellable = false; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     // Set core ID and wait result.
 | 
					
						
							|  |  |  |     core_id = phys_core; | 
					
						
							| 
									
										
										
										
											2021-02-13 10:43:01 +11:00
										 |  |  |     wait_result = ResultNoSynchronizationObject; | 
					
						
							| 
									
										
										
										
											2021-01-20 13:42:27 -08:00
										 |  |  | 
 | 
					
						
							|  |  |  |     // Set priorities.
 | 
					
						
							|  |  |  |     priority = prio; | 
					
						
							|  |  |  |     base_priority = prio; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     // Initialize sleeping queue.
 | 
					
						
							| 
									
										
										
										
											2021-11-28 00:40:25 -08:00
										 |  |  |     wait_queue = nullptr; | 
					
						
							| 
									
										
										
										
											2021-01-20 13:42:27 -08:00
										 |  |  | 
 | 
					
						
							|  |  |  |     // Set suspend flags.
 | 
					
						
							|  |  |  |     suspend_request_flags = 0; | 
					
						
							|  |  |  |     suspend_allowed_flags = static_cast<u32>(ThreadState::SuspendFlagMask); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     // We're neither debug attached, nor are we nesting our priority inheritance.
 | 
					
						
							|  |  |  |     debug_attached = false; | 
					
						
							|  |  |  |     priority_inheritance_count = 0; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     // We haven't been scheduled, and we have done no light IPC.
 | 
					
						
							|  |  |  |     schedule_count = -1; | 
					
						
							|  |  |  |     last_scheduled_tick = 0; | 
					
						
							|  |  |  |     light_ipc_data = nullptr; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     // We're not waiting for a lock, and we haven't disabled migration.
 | 
					
						
							| 
									
										
										
										
											2023-02-23 15:49:42 -05:00
										 |  |  |     waiting_lock_info = nullptr; | 
					
						
							| 
									
										
										
										
											2021-01-20 13:42:27 -08:00
										 |  |  |     num_core_migration_disables = 0; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     // We have no waiters, but we do have an entrypoint.
 | 
					
						
							|  |  |  |     num_kernel_waiters = 0; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     // Set our current core id.
 | 
					
						
							|  |  |  |     current_core_id = phys_core; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     // We haven't released our resource limit hint, and we've spent no time on the cpu.
 | 
					
						
							|  |  |  |     resource_limit_release_hint = false; | 
					
						
							|  |  |  |     cpu_time = 0; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-06-01 10:54:44 -04:00
										 |  |  |     // Set debug context.
 | 
					
						
							|  |  |  |     stack_top = user_stack_top; | 
					
						
							|  |  |  |     argument = arg; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-01-20 13:42:27 -08:00
										 |  |  |     // Clear our stack parameters.
 | 
					
						
							|  |  |  |     std::memset(static_cast<void*>(std::addressof(GetStackParameters())), 0, | 
					
						
							|  |  |  |                 sizeof(StackParameters)); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     // Set parent, if relevant.
 | 
					
						
							|  |  |  |     if (owner != nullptr) { | 
					
						
							| 
									
										
										
										
											2021-05-29 00:31:04 -04:00
										 |  |  |         // Setup the TLS, if needed.
 | 
					
						
							|  |  |  |         if (type == ThreadType::User) { | 
					
						
							| 
									
										
										
										
											2022-03-11 17:14:17 -08:00
										 |  |  |             R_TRY(owner->CreateThreadLocalRegion(std::addressof(tls_address))); | 
					
						
							| 
									
										
										
										
											2021-05-29 00:31:04 -04:00
										 |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-01-20 13:42:27 -08:00
										 |  |  |         parent = owner; | 
					
						
							| 
									
										
										
										
											2021-04-02 23:24:20 -07:00
										 |  |  |         parent->Open(); | 
					
						
							| 
									
										
										
										
											2021-01-20 13:42:27 -08:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2020-03-03 13:02:50 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-01-20 13:42:27 -08:00
										 |  |  |     // Initialize thread context.
 | 
					
						
							|  |  |  |     ResetThreadContext64(thread_context_64, user_stack_top, func, arg); | 
					
						
							|  |  |  |     ResetThreadContext32(thread_context_32, static_cast<u32>(user_stack_top), | 
					
						
							|  |  |  |                          static_cast<u32>(func), static_cast<u32>(arg)); | 
					
						
							| 
									
										
										
										
											2020-03-03 13:02:50 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-01-20 13:42:27 -08:00
										 |  |  |     // Setup the stack parameters.
 | 
					
						
							|  |  |  |     StackParameters& sp = GetStackParameters(); | 
					
						
							|  |  |  |     sp.cur_thread = this; | 
					
						
							| 
									
										
										
										
											2022-06-15 20:53:49 -04:00
										 |  |  |     sp.disable_count = 1; | 
					
						
							| 
									
										
										
										
											2021-01-20 13:42:27 -08:00
										 |  |  |     SetInExceptionHandler(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     // Set thread ID.
 | 
					
						
							| 
									
										
										
										
											2023-03-07 10:49:41 -05:00
										 |  |  |     thread_id = m_kernel.CreateNewThreadID(); | 
					
						
							| 
									
										
										
										
											2021-01-20 13:42:27 -08:00
										 |  |  | 
 | 
					
						
							|  |  |  |     // We initialized!
 | 
					
						
							|  |  |  |     initialized = true; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     // Register ourselves with our parent process.
 | 
					
						
							|  |  |  |     if (parent != nullptr) { | 
					
						
							|  |  |  |         parent->RegisterThread(this); | 
					
						
							|  |  |  |         if (parent->IsSuspended()) { | 
					
						
							|  |  |  |             RequestSuspend(SuspendType::Process); | 
					
						
							| 
									
										
										
										
											2020-03-11 20:44:53 -04:00
										 |  |  |         } | 
					
						
							| 
									
										
										
										
											2020-03-03 13:02:50 -04:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2014-05-13 22:00:11 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-10-14 22:55:51 -07:00
										 |  |  |     R_SUCCEED(); | 
					
						
							| 
									
										
										
										
											2020-02-24 22:04:12 -04:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-06-25 22:44:19 -05:00
										 |  |  | Result KThread::InitializeThread(KThread* thread, KThreadFunction func, uintptr_t arg, | 
					
						
							|  |  |  |                                  VAddr user_stack_top, s32 prio, s32 core, KProcess* owner, | 
					
						
							| 
									
										
										
										
											2022-07-02 12:33:49 -04:00
										 |  |  |                                  ThreadType type, std::function<void()>&& init_func) { | 
					
						
							| 
									
										
										
										
											2021-01-20 13:42:27 -08:00
										 |  |  |     // Initialize the thread.
 | 
					
						
							|  |  |  |     R_TRY(thread->Initialize(func, arg, user_stack_top, prio, core, owner, type)); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-08-14 02:14:19 -07:00
										 |  |  |     // Initialize emulation parameters.
 | 
					
						
							| 
									
										
										
										
											2022-07-02 12:33:49 -04:00
										 |  |  |     thread->host_context = std::make_shared<Common::Fiber>(std::move(init_func)); | 
					
						
							| 
									
										
										
										
											2021-08-14 02:14:19 -07:00
										 |  |  |     thread->is_single_core = !Settings::values.use_multi_core.GetValue(); | 
					
						
							| 
									
										
										
										
											2021-04-02 23:24:20 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-10-14 22:55:51 -07:00
										 |  |  |     R_SUCCEED(); | 
					
						
							| 
									
										
										
										
											2020-02-25 12:40:33 -04:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-11-02 20:21:32 -04:00
										 |  |  | Result KThread::InitializeDummyThread(KThread* thread, KProcess* owner) { | 
					
						
							| 
									
										
										
										
											2022-07-09 18:47:32 -04:00
										 |  |  |     // Initialize the thread.
 | 
					
						
							| 
									
										
										
										
											2022-11-02 20:21:32 -04:00
										 |  |  |     R_TRY(thread->Initialize({}, {}, {}, DummyThreadPriority, 3, owner, ThreadType::Dummy)); | 
					
						
							| 
									
										
										
										
											2022-07-09 18:47:32 -04:00
										 |  |  | 
 | 
					
						
							|  |  |  |     // Initialize emulation parameters.
 | 
					
						
							|  |  |  |     thread->stack_parameters.disable_count = 0; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-10-14 22:55:51 -07:00
										 |  |  |     R_SUCCEED(); | 
					
						
							| 
									
										
										
										
											2021-04-02 23:24:20 -07:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-06-26 18:52:16 -04:00
										 |  |  | Result KThread::InitializeMainThread(Core::System& system, KThread* thread, s32 virt_core) { | 
					
						
							| 
									
										
										
										
											2022-10-14 22:55:51 -07:00
										 |  |  |     R_RETURN(InitializeThread(thread, {}, {}, {}, IdleThreadPriority, virt_core, {}, | 
					
						
							|  |  |  |                               ThreadType::Main, system.GetCpuManager().GetGuestActivateFunc())); | 
					
						
							| 
									
										
										
										
											2022-06-26 18:52:16 -04:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-06-25 22:44:19 -05:00
										 |  |  | Result KThread::InitializeIdleThread(Core::System& system, KThread* thread, s32 virt_core) { | 
					
						
							| 
									
										
										
										
											2022-10-14 22:55:51 -07:00
										 |  |  |     R_RETURN(InitializeThread(thread, {}, {}, {}, IdleThreadPriority, virt_core, {}, | 
					
						
							|  |  |  |                               ThreadType::Main, system.GetCpuManager().GetIdleThreadStartFunc())); | 
					
						
							| 
									
										
										
										
											2021-04-02 23:24:20 -07:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-06-25 22:44:19 -05:00
										 |  |  | Result KThread::InitializeHighPriorityThread(Core::System& system, KThread* thread, | 
					
						
							|  |  |  |                                              KThreadFunction func, uintptr_t arg, s32 virt_core) { | 
					
						
							| 
									
										
										
										
											2022-10-14 22:55:51 -07:00
										 |  |  |     R_RETURN(InitializeThread(thread, func, arg, {}, {}, virt_core, nullptr, | 
					
						
							|  |  |  |                               ThreadType::HighPriority, | 
					
						
							|  |  |  |                               system.GetCpuManager().GetShutdownThreadStartFunc())); | 
					
						
							| 
									
										
										
										
											2021-04-02 23:24:20 -07:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-06-25 22:44:19 -05:00
										 |  |  | Result KThread::InitializeUserThread(Core::System& system, KThread* thread, KThreadFunction func, | 
					
						
							|  |  |  |                                      uintptr_t arg, VAddr user_stack_top, s32 prio, s32 virt_core, | 
					
						
							|  |  |  |                                      KProcess* owner) { | 
					
						
							| 
									
										
										
										
											2021-04-02 23:24:20 -07:00
										 |  |  |     system.Kernel().GlobalSchedulerContext().AddThread(thread); | 
					
						
							| 
									
										
										
										
											2022-10-14 22:55:51 -07:00
										 |  |  |     R_RETURN(InitializeThread(thread, func, arg, user_stack_top, prio, virt_core, owner, | 
					
						
							|  |  |  |                               ThreadType::User, system.GetCpuManager().GetGuestThreadFunc())); | 
					
						
							| 
									
										
										
										
											2021-04-02 23:24:20 -07:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-02-18 16:26:48 -05:00
										 |  |  | Result KThread::InitializeServiceThread(Core::System& system, KThread* thread, | 
					
						
							|  |  |  |                                         std::function<void()>&& func, s32 prio, s32 virt_core, | 
					
						
							|  |  |  |                                         KProcess* owner) { | 
					
						
							|  |  |  |     system.Kernel().GlobalSchedulerContext().AddThread(thread); | 
					
						
							|  |  |  |     std::function<void()> func2{[&system, func{std::move(func)}] { | 
					
						
							|  |  |  |         // Similar to UserModeThreadStarter.
 | 
					
						
							|  |  |  |         system.Kernel().CurrentScheduler()->OnThreadStart(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         // Run the guest function.
 | 
					
						
							|  |  |  |         func(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         // Exit.
 | 
					
						
							|  |  |  |         Svc::ExitThread(system); | 
					
						
							|  |  |  |     }}; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     R_RETURN(InitializeThread(thread, {}, {}, {}, prio, virt_core, owner, ThreadType::HighPriority, | 
					
						
							|  |  |  |                               std::move(func2))); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-04-02 23:24:20 -07:00
										 |  |  | void KThread::PostDestroy(uintptr_t arg) { | 
					
						
							| 
									
										
										
										
											2021-04-23 22:04:28 -07:00
										 |  |  |     KProcess* owner = reinterpret_cast<KProcess*>(arg & ~1ULL); | 
					
						
							| 
									
										
										
										
											2021-04-02 23:24:20 -07:00
										 |  |  |     const bool resource_limit_release_hint = (arg & 1); | 
					
						
							|  |  |  |     const s64 hint_value = (resource_limit_release_hint ? 0 : 1); | 
					
						
							|  |  |  |     if (owner != nullptr) { | 
					
						
							| 
									
										
										
										
											2022-11-03 10:22:05 -04:00
										 |  |  |         owner->GetResourceLimit()->Release(LimitableResource::ThreadCountMax, 1, hint_value); | 
					
						
							| 
									
										
										
										
											2021-04-02 23:24:20 -07:00
										 |  |  |         owner->Close(); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-01-20 13:42:27 -08:00
										 |  |  | void KThread::Finalize() { | 
					
						
							|  |  |  |     // If the thread has an owner process, unregister it.
 | 
					
						
							|  |  |  |     if (parent != nullptr) { | 
					
						
							|  |  |  |         parent->UnregisterThread(this); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     // If the thread has a local region, delete it.
 | 
					
						
							|  |  |  |     if (tls_address != 0) { | 
					
						
							| 
									
										
										
										
											2022-03-11 17:14:17 -08:00
										 |  |  |         ASSERT(parent->DeleteThreadLocalRegion(tls_address).IsSuccess()); | 
					
						
							| 
									
										
										
										
											2021-01-20 13:42:27 -08:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     // Release any waiters.
 | 
					
						
							|  |  |  |     { | 
					
						
							| 
									
										
										
										
											2023-02-23 15:49:42 -05:00
										 |  |  |         ASSERT(waiting_lock_info == nullptr); | 
					
						
							| 
									
										
										
										
											2023-03-07 10:49:41 -05:00
										 |  |  |         KScopedSchedulerLock sl{m_kernel}; | 
					
						
							| 
									
										
										
										
											2021-01-20 13:42:27 -08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-02-23 15:49:42 -05:00
										 |  |  |         // Check that we have no kernel waiters.
 | 
					
						
							|  |  |  |         ASSERT(num_kernel_waiters == 0); | 
					
						
							| 
									
										
										
										
											2022-07-09 18:54:54 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-02-23 15:49:42 -05:00
										 |  |  |         auto it = held_lock_info_list.begin(); | 
					
						
							|  |  |  |         while (it != held_lock_info_list.end()) { | 
					
						
							|  |  |  |             // Get the lock info.
 | 
					
						
							|  |  |  |             auto* const lock_info = std::addressof(*it); | 
					
						
							| 
									
										
										
										
											2022-07-09 18:54:54 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-02-23 15:49:42 -05:00
										 |  |  |             // The lock shouldn't have a kernel waiter.
 | 
					
						
							|  |  |  |             ASSERT(!lock_info->GetIsKernelAddressKey()); | 
					
						
							| 
									
										
										
										
											2021-11-10 22:28:30 -08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-02-23 15:49:42 -05:00
										 |  |  |             // Remove all waiters.
 | 
					
						
							|  |  |  |             while (lock_info->GetWaiterCount() != 0) { | 
					
						
							|  |  |  |                 // Get the front waiter.
 | 
					
						
							|  |  |  |                 KThread* const waiter = lock_info->GetHighestPriorityWaiter(); | 
					
						
							| 
									
										
										
										
											2021-11-10 22:28:30 -08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-02-23 15:49:42 -05:00
										 |  |  |                 // Remove it from the lock.
 | 
					
						
							|  |  |  |                 if (lock_info->RemoveWaiter(waiter)) { | 
					
						
							|  |  |  |                     ASSERT(lock_info->GetWaiterCount() == 0); | 
					
						
							|  |  |  |                 } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |                 // Cancel the thread's wait.
 | 
					
						
							|  |  |  |                 waiter->CancelWait(ResultInvalidState, true); | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             // Remove the held lock from our list.
 | 
					
						
							|  |  |  |             it = held_lock_info_list.erase(it); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             // Free the lock info.
 | 
					
						
							| 
									
										
										
										
											2023-03-07 10:49:41 -05:00
										 |  |  |             LockWithPriorityInheritanceInfo::Free(m_kernel, lock_info); | 
					
						
							| 
									
										
										
										
											2021-01-20 13:42:27 -08:00
										 |  |  |         } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-03-11 00:14:13 -08:00
										 |  |  |     // Release host emulation members.
 | 
					
						
							|  |  |  |     host_context.reset(); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-04-03 23:22:07 -07:00
										 |  |  |     // Perform inherited finalization.
 | 
					
						
							| 
									
										
										
										
											2022-01-14 16:36:10 -08:00
										 |  |  |     KSynchronizationObject::Finalize(); | 
					
						
							| 
									
										
										
										
											2019-04-17 07:08:12 -04:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-01-20 13:42:27 -08:00
										 |  |  | bool KThread::IsSignaled() const { | 
					
						
							|  |  |  |     return signaled; | 
					
						
							| 
									
										
										
										
											2020-03-01 23:46:10 -05:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-11-10 22:28:30 -08:00
										 |  |  | void KThread::OnTimer() { | 
					
						
							| 
									
										
										
										
											2023-03-07 10:49:41 -05:00
										 |  |  |     ASSERT(m_kernel.GlobalSchedulerContext().IsLocked()); | 
					
						
							| 
									
										
										
										
											2021-01-20 13:42:27 -08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-11-10 22:28:30 -08:00
										 |  |  |     // If we're waiting, cancel the wait.
 | 
					
						
							| 
									
										
										
										
											2021-01-20 13:42:27 -08:00
										 |  |  |     if (GetState() == ThreadState::Waiting) { | 
					
						
							| 
									
										
										
										
											2021-11-28 00:40:25 -08:00
										 |  |  |         wait_queue->CancelWait(this, ResultTimedOut, false); | 
					
						
							| 
									
										
										
										
											2021-01-20 13:42:27 -08:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2016-09-02 08:53:42 -04:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-01-20 13:42:27 -08:00
										 |  |  | void KThread::StartTermination() { | 
					
						
							| 
									
										
										
										
											2023-03-07 10:49:41 -05:00
										 |  |  |     ASSERT(m_kernel.GlobalSchedulerContext().IsLocked()); | 
					
						
							| 
									
										
										
										
											2021-01-20 13:42:27 -08:00
										 |  |  | 
 | 
					
						
							|  |  |  |     // Release user exception and unpin, if relevant.
 | 
					
						
							|  |  |  |     if (parent != nullptr) { | 
					
						
							|  |  |  |         parent->ReleaseUserException(this); | 
					
						
							| 
									
										
										
										
											2023-03-07 10:49:41 -05:00
										 |  |  |         if (parent->GetPinnedThread(GetCurrentCoreId(m_kernel)) == this) { | 
					
						
							| 
									
										
										
										
											2021-12-29 21:40:38 -08:00
										 |  |  |             parent->UnpinCurrentThread(core_id); | 
					
						
							| 
									
										
										
										
											2021-01-20 13:42:27 -08:00
										 |  |  |         } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     // Set state to terminated.
 | 
					
						
							|  |  |  |     SetState(ThreadState::Terminated); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     // Clear the thread's status as running in parent.
 | 
					
						
							|  |  |  |     if (parent != nullptr) { | 
					
						
							|  |  |  |         parent->ClearRunningThread(this); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     // Signal.
 | 
					
						
							|  |  |  |     signaled = true; | 
					
						
							| 
									
										
										
										
											2021-11-09 19:02:11 -08:00
										 |  |  |     KSynchronizationObject::NotifyAvailable(); | 
					
						
							| 
									
										
										
										
											2021-01-20 13:42:27 -08:00
										 |  |  | 
 | 
					
						
							|  |  |  |     // Clear previous thread in KScheduler.
 | 
					
						
							| 
									
										
										
										
											2023-03-07 10:49:41 -05:00
										 |  |  |     KScheduler::ClearPreviousThread(m_kernel, this); | 
					
						
							| 
									
										
										
										
											2021-01-20 13:42:27 -08:00
										 |  |  | 
 | 
					
						
							|  |  |  |     // Register terminated dpc flag.
 | 
					
						
							|  |  |  |     RegisterDpc(DpcFlag::Terminated); | 
					
						
							| 
									
										
										
										
											2022-01-14 16:36:10 -08:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void KThread::FinishTermination() { | 
					
						
							|  |  |  |     // Ensure that the thread is not executing on any core.
 | 
					
						
							|  |  |  |     if (parent != nullptr) { | 
					
						
							|  |  |  |         for (std::size_t i = 0; i < static_cast<std::size_t>(Core::Hardware::NUM_CPU_CORES); ++i) { | 
					
						
							|  |  |  |             KThread* core_thread{}; | 
					
						
							|  |  |  |             do { | 
					
						
							| 
									
										
										
										
											2023-03-07 10:49:41 -05:00
										 |  |  |                 core_thread = m_kernel.Scheduler(i).GetSchedulerCurrentThread(); | 
					
						
							| 
									
										
										
										
											2022-01-14 16:36:10 -08:00
										 |  |  |             } while (core_thread == this); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2021-04-02 23:24:20 -07:00
										 |  |  | 
 | 
					
						
							|  |  |  |     // Close the thread.
 | 
					
						
							|  |  |  |     this->Close(); | 
					
						
							| 
									
										
										
										
											2020-02-24 22:04:12 -04:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-01-14 16:36:10 -08:00
										 |  |  | void KThread::DoWorkerTaskImpl() { | 
					
						
							|  |  |  |     // Finish the termination that was begun by Exit().
 | 
					
						
							|  |  |  |     this->FinishTermination(); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-12-29 21:40:38 -08:00
										 |  |  | void KThread::Pin(s32 current_core) { | 
					
						
							| 
									
										
										
										
											2023-03-07 10:49:41 -05:00
										 |  |  |     ASSERT(m_kernel.GlobalSchedulerContext().IsLocked()); | 
					
						
							| 
									
										
										
										
											2021-01-20 13:42:27 -08:00
										 |  |  | 
 | 
					
						
							|  |  |  |     // Set ourselves as pinned.
 | 
					
						
							|  |  |  |     GetStackParameters().is_pinned = true; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     // Disable core migration.
 | 
					
						
							|  |  |  |     ASSERT(num_core_migration_disables == 0); | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         ++num_core_migration_disables; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         // Save our ideal state to restore when we're unpinned.
 | 
					
						
							|  |  |  |         original_physical_ideal_core_id = physical_ideal_core_id; | 
					
						
							|  |  |  |         original_physical_affinity_mask = physical_affinity_mask; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         // Bind ourselves to this core.
 | 
					
						
							|  |  |  |         const s32 active_core = GetActiveCore(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         SetActiveCore(current_core); | 
					
						
							|  |  |  |         physical_ideal_core_id = current_core; | 
					
						
							|  |  |  |         physical_affinity_mask.SetAffinityMask(1ULL << current_core); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         if (active_core != current_core || physical_affinity_mask.GetAffinityMask() != | 
					
						
							|  |  |  |                                                original_physical_affinity_mask.GetAffinityMask()) { | 
					
						
							| 
									
										
										
										
											2023-03-07 10:49:41 -05:00
										 |  |  |             KScheduler::OnThreadAffinityMaskChanged(m_kernel, this, original_physical_affinity_mask, | 
					
						
							| 
									
										
										
										
											2021-01-20 13:42:27 -08:00
										 |  |  |                                                     active_core); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     // Disallow performing thread suspension.
 | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         // Update our allow flags.
 | 
					
						
							|  |  |  |         suspend_allowed_flags &= ~(1 << (static_cast<u32>(SuspendType::Thread) + | 
					
						
							|  |  |  |                                          static_cast<u32>(ThreadState::SuspendShift))); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         // Update our state.
 | 
					
						
							| 
									
										
										
										
											2022-01-14 16:31:47 -08:00
										 |  |  |         UpdateState(); | 
					
						
							| 
									
										
										
										
											2021-01-20 13:42:27 -08:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     // TODO(bunnei): Update our SVC access permissions.
 | 
					
						
							|  |  |  |     ASSERT(parent != nullptr); | 
					
						
							| 
									
										
										
										
											2020-02-24 22:04:12 -04:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-01-20 13:42:27 -08:00
										 |  |  | void KThread::Unpin() { | 
					
						
							| 
									
										
										
										
											2023-03-07 10:49:41 -05:00
										 |  |  |     ASSERT(m_kernel.GlobalSchedulerContext().IsLocked()); | 
					
						
							| 
									
										
										
										
											2021-01-20 13:42:27 -08:00
										 |  |  | 
 | 
					
						
							|  |  |  |     // Set ourselves as unpinned.
 | 
					
						
							|  |  |  |     GetStackParameters().is_pinned = false; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     // Enable core migration.
 | 
					
						
							|  |  |  |     ASSERT(num_core_migration_disables == 1); | 
					
						
							|  |  |  |     { | 
					
						
							| 
									
										
										
										
											2021-01-24 22:55:08 -08:00
										 |  |  |         num_core_migration_disables--; | 
					
						
							| 
									
										
										
										
											2021-01-20 13:42:27 -08:00
										 |  |  | 
 | 
					
						
							|  |  |  |         // Restore our original state.
 | 
					
						
							|  |  |  |         const KAffinityMask old_mask = physical_affinity_mask; | 
					
						
							| 
									
										
										
										
											2021-01-01 02:06:06 -08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-01-20 13:42:27 -08:00
										 |  |  |         physical_ideal_core_id = original_physical_ideal_core_id; | 
					
						
							|  |  |  |         physical_affinity_mask = original_physical_affinity_mask; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         if (physical_affinity_mask.GetAffinityMask() != old_mask.GetAffinityMask()) { | 
					
						
							|  |  |  |             const s32 active_core = GetActiveCore(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             if (!physical_affinity_mask.GetAffinity(active_core)) { | 
					
						
							|  |  |  |                 if (physical_ideal_core_id >= 0) { | 
					
						
							|  |  |  |                     SetActiveCore(physical_ideal_core_id); | 
					
						
							|  |  |  |                 } else { | 
					
						
							|  |  |  |                     SetActiveCore(static_cast<s32>( | 
					
						
							|  |  |  |                         Common::BitSize<u64>() - 1 - | 
					
						
							|  |  |  |                         std::countl_zero(physical_affinity_mask.GetAffinityMask()))); | 
					
						
							|  |  |  |                 } | 
					
						
							|  |  |  |             } | 
					
						
							| 
									
										
										
										
											2023-03-07 10:49:41 -05:00
										 |  |  |             KScheduler::OnThreadAffinityMaskChanged(m_kernel, this, old_mask, active_core); | 
					
						
							| 
									
										
										
										
											2020-02-24 22:04:12 -04:00
										 |  |  |         } | 
					
						
							| 
									
										
										
										
											2014-12-22 11:07:22 -02:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2014-05-20 21:02:35 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-01-20 13:42:27 -08:00
										 |  |  |     // Allow performing thread suspension (if termination hasn't been requested).
 | 
					
						
							| 
									
										
										
										
											2022-01-14 16:31:47 -08:00
										 |  |  |     if (!IsTerminationRequested()) { | 
					
						
							| 
									
										
										
										
											2021-01-20 13:42:27 -08:00
										 |  |  |         // Update our allow flags.
 | 
					
						
							| 
									
										
										
										
											2022-01-14 16:31:47 -08:00
										 |  |  |         suspend_allowed_flags |= (1 << (static_cast<u32>(SuspendType::Thread) + | 
					
						
							|  |  |  |                                         static_cast<u32>(ThreadState::SuspendShift))); | 
					
						
							| 
									
										
										
										
											2014-06-05 22:35:36 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-01-20 13:42:27 -08:00
										 |  |  |         // Update our state.
 | 
					
						
							| 
									
										
										
										
											2022-01-14 16:31:47 -08:00
										 |  |  |         UpdateState(); | 
					
						
							| 
									
										
										
										
											2021-01-20 13:42:27 -08:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2020-12-31 00:46:09 -08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-01-20 13:42:27 -08:00
										 |  |  |     // TODO(bunnei): Update our SVC access permissions.
 | 
					
						
							|  |  |  |     ASSERT(parent != nullptr); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     // Resume any threads that began waiting on us while we were pinned.
 | 
					
						
							|  |  |  |     for (auto it = pinned_waiter_list.begin(); it != pinned_waiter_list.end(); ++it) { | 
					
						
							| 
									
										
										
										
											2022-06-25 13:36:14 -04:00
										 |  |  |         it->EndWait(ResultSuccess); | 
					
						
							| 
									
										
										
										
											2021-01-20 13:42:27 -08:00
										 |  |  |     } | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-12-29 21:40:38 -08:00
										 |  |  | u16 KThread::GetUserDisableCount() const { | 
					
						
							|  |  |  |     if (!IsUserThread()) { | 
					
						
							|  |  |  |         // We only emulate TLS for user threads
 | 
					
						
							|  |  |  |         return {}; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-03-07 10:49:41 -05:00
										 |  |  |     auto& memory = m_kernel.System().Memory(); | 
					
						
							| 
									
										
										
										
											2021-12-29 21:40:38 -08:00
										 |  |  |     return memory.Read16(tls_address + offsetof(ThreadLocalRegion, disable_count)); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void KThread::SetInterruptFlag() { | 
					
						
							|  |  |  |     if (!IsUserThread()) { | 
					
						
							|  |  |  |         // We only emulate TLS for user threads
 | 
					
						
							|  |  |  |         return; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-03-07 10:49:41 -05:00
										 |  |  |     auto& memory = m_kernel.System().Memory(); | 
					
						
							| 
									
										
										
										
											2021-12-29 21:40:38 -08:00
										 |  |  |     memory.Write16(tls_address + offsetof(ThreadLocalRegion, interrupt_flag), 1); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void KThread::ClearInterruptFlag() { | 
					
						
							|  |  |  |     if (!IsUserThread()) { | 
					
						
							|  |  |  |         // We only emulate TLS for user threads
 | 
					
						
							|  |  |  |         return; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-03-07 10:49:41 -05:00
										 |  |  |     auto& memory = m_kernel.System().Memory(); | 
					
						
							| 
									
										
										
										
											2021-12-29 21:40:38 -08:00
										 |  |  |     memory.Write16(tls_address + offsetof(ThreadLocalRegion, interrupt_flag), 0); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-06-25 22:44:19 -05:00
										 |  |  | Result KThread::GetCoreMask(s32* out_ideal_core, u64* out_affinity_mask) { | 
					
						
							| 
									
										
										
										
											2023-03-07 10:49:41 -05:00
										 |  |  |     KScopedSchedulerLock sl{m_kernel}; | 
					
						
							| 
									
										
										
										
											2021-01-20 13:42:27 -08:00
										 |  |  | 
 | 
					
						
							|  |  |  |     // Get the virtual mask.
 | 
					
						
							|  |  |  |     *out_ideal_core = virtual_ideal_core_id; | 
					
						
							|  |  |  |     *out_affinity_mask = virtual_affinity_mask; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-10-14 22:55:51 -07:00
										 |  |  |     R_SUCCEED(); | 
					
						
							| 
									
										
										
										
											2021-01-20 13:42:27 -08:00
										 |  |  | } | 
					
						
							| 
									
										
										
										
											2020-12-31 00:46:09 -08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-06-25 22:44:19 -05:00
										 |  |  | Result KThread::GetPhysicalCoreMask(s32* out_ideal_core, u64* out_affinity_mask) { | 
					
						
							| 
									
										
										
										
											2023-03-07 10:49:41 -05:00
										 |  |  |     KScopedSchedulerLock sl{m_kernel}; | 
					
						
							| 
									
										
										
										
											2021-01-20 13:42:27 -08:00
										 |  |  |     ASSERT(num_core_migration_disables >= 0); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     // Select between core mask and original core mask.
 | 
					
						
							|  |  |  |     if (num_core_migration_disables == 0) { | 
					
						
							|  |  |  |         *out_ideal_core = physical_ideal_core_id; | 
					
						
							|  |  |  |         *out_affinity_mask = physical_affinity_mask.GetAffinityMask(); | 
					
						
							| 
									
										
										
										
											2020-02-24 22:04:12 -04:00
										 |  |  |     } else { | 
					
						
							| 
									
										
										
										
											2021-01-20 13:42:27 -08:00
										 |  |  |         *out_ideal_core = original_physical_ideal_core_id; | 
					
						
							|  |  |  |         *out_affinity_mask = original_physical_affinity_mask.GetAffinityMask(); | 
					
						
							| 
									
										
										
										
											2020-02-24 22:04:12 -04:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2020-11-03 19:54:53 -05:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-10-14 22:55:51 -07:00
										 |  |  |     R_SUCCEED(); | 
					
						
							| 
									
										
										
										
											2021-01-20 13:42:27 -08:00
										 |  |  | } | 
					
						
							| 
									
										
										
										
											2021-11-10 22:46:07 -08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-06-25 22:44:19 -05:00
										 |  |  | Result KThread::SetCoreMask(s32 core_id_, u64 v_affinity_mask) { | 
					
						
							| 
									
										
										
										
											2021-01-20 13:42:27 -08:00
										 |  |  |     ASSERT(parent != nullptr); | 
					
						
							|  |  |  |     ASSERT(v_affinity_mask != 0); | 
					
						
							| 
									
										
										
										
											2021-11-10 22:46:07 -08:00
										 |  |  |     KScopedLightLock lk(activity_pause_lock); | 
					
						
							| 
									
										
										
										
											2015-01-25 22:56:17 -08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-01-20 13:42:27 -08:00
										 |  |  |     // Set the core mask.
 | 
					
						
							|  |  |  |     u64 p_affinity_mask = 0; | 
					
						
							|  |  |  |     { | 
					
						
							| 
									
										
										
										
											2023-03-07 10:49:41 -05:00
										 |  |  |         KScopedSchedulerLock sl(m_kernel); | 
					
						
							| 
									
										
										
										
											2021-01-20 13:42:27 -08:00
										 |  |  |         ASSERT(num_core_migration_disables >= 0); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-11-10 22:46:07 -08:00
										 |  |  |         // If we're updating, set our ideal virtual core.
 | 
					
						
							|  |  |  |         if (core_id_ != Svc::IdealCoreNoUpdate) { | 
					
						
							|  |  |  |             virtual_ideal_core_id = core_id_; | 
					
						
							|  |  |  |         } else { | 
					
						
							|  |  |  |             // Preserve our ideal core id.
 | 
					
						
							|  |  |  |             core_id_ = virtual_ideal_core_id; | 
					
						
							|  |  |  |             R_UNLESS(((1ULL << core_id_) & v_affinity_mask) != 0, ResultInvalidCombination); | 
					
						
							| 
									
										
										
										
											2021-01-20 13:42:27 -08:00
										 |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-11-10 22:46:07 -08:00
										 |  |  |         // Set our affinity mask.
 | 
					
						
							| 
									
										
										
										
											2021-01-20 13:42:27 -08:00
										 |  |  |         virtual_affinity_mask = v_affinity_mask; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         // Translate the virtual core to a physical core.
 | 
					
						
							| 
									
										
										
										
											2021-11-10 22:46:07 -08:00
										 |  |  |         if (core_id_ >= 0) { | 
					
						
							|  |  |  |             core_id_ = Core::Hardware::VirtualToPhysicalCoreMap[core_id_]; | 
					
						
							| 
									
										
										
										
											2021-01-20 13:42:27 -08:00
										 |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         // Translate the virtual affinity mask to a physical one.
 | 
					
						
							|  |  |  |         while (v_affinity_mask != 0) { | 
					
						
							|  |  |  |             const u64 next = std::countr_zero(v_affinity_mask); | 
					
						
							|  |  |  |             v_affinity_mask &= ~(1ULL << next); | 
					
						
							|  |  |  |             p_affinity_mask |= (1ULL << Core::Hardware::VirtualToPhysicalCoreMap[next]); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         // If we haven't disabled migration, perform an affinity change.
 | 
					
						
							|  |  |  |         if (num_core_migration_disables == 0) { | 
					
						
							|  |  |  |             const KAffinityMask old_mask = physical_affinity_mask; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             // Set our new ideals.
 | 
					
						
							| 
									
										
										
										
											2021-11-10 22:46:07 -08:00
										 |  |  |             physical_ideal_core_id = core_id_; | 
					
						
							| 
									
										
										
										
											2021-01-20 13:42:27 -08:00
										 |  |  |             physical_affinity_mask.SetAffinityMask(p_affinity_mask); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             if (physical_affinity_mask.GetAffinityMask() != old_mask.GetAffinityMask()) { | 
					
						
							|  |  |  |                 const s32 active_core = GetActiveCore(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |                 if (active_core >= 0 && !physical_affinity_mask.GetAffinity(active_core)) { | 
					
						
							|  |  |  |                     const s32 new_core = static_cast<s32>( | 
					
						
							|  |  |  |                         physical_ideal_core_id >= 0 | 
					
						
							|  |  |  |                             ? physical_ideal_core_id | 
					
						
							|  |  |  |                             : Common::BitSize<u64>() - 1 - | 
					
						
							|  |  |  |                                   std::countl_zero(physical_affinity_mask.GetAffinityMask())); | 
					
						
							|  |  |  |                     SetActiveCore(new_core); | 
					
						
							|  |  |  |                 } | 
					
						
							| 
									
										
										
										
											2023-03-07 10:49:41 -05:00
										 |  |  |                 KScheduler::OnThreadAffinityMaskChanged(m_kernel, this, old_mask, active_core); | 
					
						
							| 
									
										
										
										
											2021-01-20 13:42:27 -08:00
										 |  |  |             } | 
					
						
							|  |  |  |         } else { | 
					
						
							|  |  |  |             // Otherwise, we edit the original affinity for restoration later.
 | 
					
						
							| 
									
										
										
										
											2021-11-10 22:46:07 -08:00
										 |  |  |             original_physical_ideal_core_id = core_id_; | 
					
						
							| 
									
										
										
										
											2021-01-20 13:42:27 -08:00
										 |  |  |             original_physical_affinity_mask.SetAffinityMask(p_affinity_mask); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     // Update the pinned waiter list.
 | 
					
						
							| 
									
										
										
										
											2023-03-07 10:49:41 -05:00
										 |  |  |     ThreadQueueImplForKThreadSetProperty wait_queue_(m_kernel, std::addressof(pinned_waiter_list)); | 
					
						
							| 
									
										
										
										
											2021-01-20 13:42:27 -08:00
										 |  |  |     { | 
					
						
							| 
									
										
										
										
											2021-01-24 22:55:08 -08:00
										 |  |  |         bool retry_update{}; | 
					
						
							| 
									
										
										
										
											2021-01-20 13:42:27 -08:00
										 |  |  |         do { | 
					
						
							|  |  |  |             // Lock the scheduler.
 | 
					
						
							| 
									
										
										
										
											2023-03-07 10:49:41 -05:00
										 |  |  |             KScopedSchedulerLock sl(m_kernel); | 
					
						
							| 
									
										
										
										
											2021-01-20 13:42:27 -08:00
										 |  |  | 
 | 
					
						
							|  |  |  |             // Don't do any further management if our termination has been requested.
 | 
					
						
							|  |  |  |             R_SUCCEED_IF(IsTerminationRequested()); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             // By default, we won't need to retry.
 | 
					
						
							|  |  |  |             retry_update = false; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             // Check if the thread is currently running.
 | 
					
						
							| 
									
										
										
										
											2021-01-24 22:55:08 -08:00
										 |  |  |             bool thread_is_current{}; | 
					
						
							| 
									
										
										
										
											2021-01-20 13:42:27 -08:00
										 |  |  |             s32 thread_core; | 
					
						
							|  |  |  |             for (thread_core = 0; thread_core < static_cast<s32>(Core::Hardware::NUM_CPU_CORES); | 
					
						
							|  |  |  |                  ++thread_core) { | 
					
						
							| 
									
										
										
										
											2023-03-07 10:49:41 -05:00
										 |  |  |                 if (m_kernel.Scheduler(thread_core).GetSchedulerCurrentThread() == this) { | 
					
						
							| 
									
										
										
										
											2021-01-20 13:42:27 -08:00
										 |  |  |                     thread_is_current = true; | 
					
						
							|  |  |  |                     break; | 
					
						
							|  |  |  |                 } | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             // If the thread is currently running, check whether it's no longer allowed under the
 | 
					
						
							|  |  |  |             // new mask.
 | 
					
						
							|  |  |  |             if (thread_is_current && ((1ULL << thread_core) & p_affinity_mask) == 0) { | 
					
						
							|  |  |  |                 // If the thread is pinned, we want to wait until it's not pinned.
 | 
					
						
							|  |  |  |                 if (GetStackParameters().is_pinned) { | 
					
						
							|  |  |  |                     // Verify that the current thread isn't terminating.
 | 
					
						
							| 
									
										
										
										
											2023-03-07 10:49:41 -05:00
										 |  |  |                     R_UNLESS(!GetCurrentThread(m_kernel).IsTerminationRequested(), | 
					
						
							| 
									
										
										
										
											2021-02-13 10:43:01 +11:00
										 |  |  |                              ResultTerminationRequested); | 
					
						
							| 
									
										
										
										
											2021-01-20 13:42:27 -08:00
										 |  |  | 
 | 
					
						
							|  |  |  |                     // Wait until the thread isn't pinned any more.
 | 
					
						
							| 
									
										
										
										
											2023-03-07 10:49:41 -05:00
										 |  |  |                     pinned_waiter_list.push_back(GetCurrentThread(m_kernel)); | 
					
						
							|  |  |  |                     GetCurrentThread(m_kernel).BeginWait(std::addressof(wait_queue_)); | 
					
						
							| 
									
										
										
										
											2021-01-20 13:42:27 -08:00
										 |  |  |                 } else { | 
					
						
							|  |  |  |                     // If the thread isn't pinned, release the scheduler lock and retry until it's
 | 
					
						
							|  |  |  |                     // not current.
 | 
					
						
							|  |  |  |                     retry_update = true; | 
					
						
							|  |  |  |                 } | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |         } while (retry_update); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-10-14 22:55:51 -07:00
										 |  |  |     R_SUCCEED(); | 
					
						
							| 
									
										
										
										
											2014-06-01 22:12:54 -04:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-01-20 13:42:27 -08:00
										 |  |  | void KThread::SetBasePriority(s32 value) { | 
					
						
							|  |  |  |     ASSERT(Svc::HighestThreadPriority <= value && value <= Svc::LowestThreadPriority); | 
					
						
							| 
									
										
										
										
											2020-12-30 01:14:02 -08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-03-07 10:49:41 -05:00
										 |  |  |     KScopedSchedulerLock sl{m_kernel}; | 
					
						
							| 
									
										
										
										
											2020-12-30 01:14:02 -08:00
										 |  |  | 
 | 
					
						
							|  |  |  |     // Change our base priority.
 | 
					
						
							| 
									
										
										
										
											2021-01-20 13:42:27 -08:00
										 |  |  |     base_priority = value; | 
					
						
							| 
									
										
										
										
											2020-12-30 01:14:02 -08:00
										 |  |  | 
 | 
					
						
							|  |  |  |     // Perform a priority restoration.
 | 
					
						
							| 
									
										
										
										
											2023-03-07 10:49:41 -05:00
										 |  |  |     RestorePriority(m_kernel, this); | 
					
						
							| 
									
										
										
										
											2014-06-01 22:12:54 -04:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-02-23 15:49:42 -05:00
										 |  |  | KThread* KThread::GetLockOwner() const { | 
					
						
							|  |  |  |     return waiting_lock_info != nullptr ? waiting_lock_info->GetOwner() : nullptr; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void KThread::IncreaseBasePriority(s32 priority_) { | 
					
						
							|  |  |  |     ASSERT(Svc::HighestThreadPriority <= priority_ && priority_ <= Svc::LowestThreadPriority); | 
					
						
							| 
									
										
										
										
											2023-03-07 10:49:41 -05:00
										 |  |  |     ASSERT(KScheduler::IsSchedulerLockedByCurrentThread(m_kernel)); | 
					
						
							| 
									
										
										
										
											2023-02-23 15:49:42 -05:00
										 |  |  |     ASSERT(!this->GetStackParameters().is_pinned); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     // Set our base priority.
 | 
					
						
							|  |  |  |     if (base_priority > priority_) { | 
					
						
							|  |  |  |         base_priority = priority_; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         // Perform a priority restoration.
 | 
					
						
							| 
									
										
										
										
											2023-03-07 10:49:41 -05:00
										 |  |  |         RestorePriority(m_kernel, this); | 
					
						
							| 
									
										
										
										
											2023-02-23 15:49:42 -05:00
										 |  |  |     } | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-01-20 13:42:27 -08:00
										 |  |  | void KThread::RequestSuspend(SuspendType type) { | 
					
						
							| 
									
										
										
										
											2023-03-07 10:49:41 -05:00
										 |  |  |     KScopedSchedulerLock sl{m_kernel}; | 
					
						
							| 
									
										
										
										
											2015-01-17 02:03:44 -05:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-01-20 13:42:27 -08:00
										 |  |  |     // Note the request in our flags.
 | 
					
						
							|  |  |  |     suspend_request_flags |= | 
					
						
							|  |  |  |         (1u << (static_cast<u32>(ThreadState::SuspendShift) + static_cast<u32>(type))); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     // Try to perform the suspend.
 | 
					
						
							|  |  |  |     TrySuspend(); | 
					
						
							| 
									
										
										
										
											2017-01-04 10:53:01 -05:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-01-20 13:42:27 -08:00
										 |  |  | void KThread::Resume(SuspendType type) { | 
					
						
							| 
									
										
										
										
											2023-03-07 10:49:41 -05:00
										 |  |  |     KScopedSchedulerLock sl{m_kernel}; | 
					
						
							| 
									
										
										
										
											2020-12-30 01:14:02 -08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-01-20 13:42:27 -08:00
										 |  |  |     // Clear the request in our flags.
 | 
					
						
							|  |  |  |     suspend_request_flags &= | 
					
						
							|  |  |  |         ~(1u << (static_cast<u32>(ThreadState::SuspendShift) + static_cast<u32>(type))); | 
					
						
							| 
									
										
										
										
											2021-01-10 14:29:02 -08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-01-20 13:42:27 -08:00
										 |  |  |     // Update our state.
 | 
					
						
							| 
									
										
										
										
											2022-01-14 16:31:47 -08:00
										 |  |  |     this->UpdateState(); | 
					
						
							| 
									
										
										
										
											2020-12-30 01:14:02 -08:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-01-20 13:42:27 -08:00
										 |  |  | void KThread::WaitCancel() { | 
					
						
							| 
									
										
										
										
											2023-03-07 10:49:41 -05:00
										 |  |  |     KScopedSchedulerLock sl{m_kernel}; | 
					
						
							| 
									
										
										
										
											2021-01-20 13:42:27 -08:00
										 |  |  | 
 | 
					
						
							|  |  |  |     // Check if we're waiting and cancellable.
 | 
					
						
							| 
									
										
										
										
											2021-11-10 22:28:30 -08:00
										 |  |  |     if (this->GetState() == ThreadState::Waiting && cancellable) { | 
					
						
							|  |  |  |         wait_cancelled = false; | 
					
						
							| 
									
										
										
										
											2021-11-28 00:40:25 -08:00
										 |  |  |         wait_queue->CancelWait(this, ResultCancelled, true); | 
					
						
							| 
									
										
										
										
											2021-01-20 13:42:27 -08:00
										 |  |  |     } else { | 
					
						
							|  |  |  |         // Otherwise, note that we cancelled a wait.
 | 
					
						
							|  |  |  |         wait_cancelled = true; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void KThread::TrySuspend() { | 
					
						
							| 
									
										
										
										
											2023-03-07 10:49:41 -05:00
										 |  |  |     ASSERT(m_kernel.GlobalSchedulerContext().IsLocked()); | 
					
						
							| 
									
										
										
										
											2021-01-20 13:42:27 -08:00
										 |  |  |     ASSERT(IsSuspendRequested()); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     // Ensure that we have no waiters.
 | 
					
						
							|  |  |  |     if (GetNumKernelWaiters() > 0) { | 
					
						
							|  |  |  |         return; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     ASSERT(GetNumKernelWaiters() == 0); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     // Perform the suspend.
 | 
					
						
							| 
									
										
										
										
											2022-01-14 16:31:47 -08:00
										 |  |  |     this->UpdateState(); | 
					
						
							| 
									
										
										
										
											2021-01-20 13:42:27 -08:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-01-14 16:31:47 -08:00
										 |  |  | void KThread::UpdateState() { | 
					
						
							| 
									
										
										
										
											2023-03-07 10:49:41 -05:00
										 |  |  |     ASSERT(m_kernel.GlobalSchedulerContext().IsLocked()); | 
					
						
							| 
									
										
										
										
											2021-01-20 13:42:27 -08:00
										 |  |  | 
 | 
					
						
							|  |  |  |     // Set our suspend flags in state.
 | 
					
						
							| 
									
										
										
										
											2022-04-11 20:57:32 -07:00
										 |  |  |     const ThreadState old_state = thread_state.load(std::memory_order_relaxed); | 
					
						
							| 
									
										
										
										
											2022-01-14 16:31:47 -08:00
										 |  |  |     const auto new_state = | 
					
						
							|  |  |  |         static_cast<ThreadState>(this->GetSuspendFlags()) | (old_state & ThreadState::Mask); | 
					
						
							| 
									
										
										
										
											2022-04-11 20:57:32 -07:00
										 |  |  |     thread_state.store(new_state, std::memory_order_relaxed); | 
					
						
							| 
									
										
										
										
											2021-01-20 13:42:27 -08:00
										 |  |  | 
 | 
					
						
							|  |  |  |     // Note the state change in scheduler.
 | 
					
						
							| 
									
										
										
										
											2022-01-14 16:31:47 -08:00
										 |  |  |     if (new_state != old_state) { | 
					
						
							| 
									
										
										
										
											2023-03-07 10:49:41 -05:00
										 |  |  |         KScheduler::OnThreadStateChanged(m_kernel, this, old_state); | 
					
						
							| 
									
										
										
										
											2022-01-14 16:31:47 -08:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2021-01-20 13:42:27 -08:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void KThread::Continue() { | 
					
						
							| 
									
										
										
										
											2023-03-07 10:49:41 -05:00
										 |  |  |     ASSERT(m_kernel.GlobalSchedulerContext().IsLocked()); | 
					
						
							| 
									
										
										
										
											2021-01-20 13:42:27 -08:00
										 |  |  | 
 | 
					
						
							|  |  |  |     // Clear our suspend flags in state.
 | 
					
						
							| 
									
										
										
										
											2022-04-11 20:57:32 -07:00
										 |  |  |     const ThreadState old_state = thread_state.load(std::memory_order_relaxed); | 
					
						
							|  |  |  |     thread_state.store(old_state & ThreadState::Mask, std::memory_order_relaxed); | 
					
						
							| 
									
										
										
										
											2021-01-20 13:42:27 -08:00
										 |  |  | 
 | 
					
						
							|  |  |  |     // Note the state change in scheduler.
 | 
					
						
							| 
									
										
										
										
											2023-03-07 10:49:41 -05:00
										 |  |  |     KScheduler::OnThreadStateChanged(m_kernel, this, old_state); | 
					
						
							| 
									
										
										
										
											2021-01-20 13:42:27 -08:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-03-07 19:18:06 -05:00
										 |  |  | void KThread::CloneFpuStatus() { | 
					
						
							|  |  |  |     // We shouldn't reach here when starting kernel threads.
 | 
					
						
							|  |  |  |     ASSERT(this->GetOwnerProcess() != nullptr); | 
					
						
							| 
									
										
										
										
											2023-03-07 10:49:41 -05:00
										 |  |  |     ASSERT(this->GetOwnerProcess() == GetCurrentProcessPointer(m_kernel)); | 
					
						
							| 
									
										
										
										
											2023-03-07 19:18:06 -05:00
										 |  |  | 
 | 
					
						
							|  |  |  |     if (this->GetOwnerProcess()->Is64BitProcess()) { | 
					
						
							|  |  |  |         // Clone FPSR and FPCR.
 | 
					
						
							|  |  |  |         ThreadContext64 cur_ctx{}; | 
					
						
							| 
									
										
										
										
											2023-03-07 10:49:41 -05:00
										 |  |  |         m_kernel.System().CurrentArmInterface().SaveContext(cur_ctx); | 
					
						
							| 
									
										
										
										
											2023-03-07 19:18:06 -05:00
										 |  |  | 
 | 
					
						
							|  |  |  |         this->GetContext64().fpcr = cur_ctx.fpcr; | 
					
						
							|  |  |  |         this->GetContext64().fpsr = cur_ctx.fpsr; | 
					
						
							|  |  |  |     } else { | 
					
						
							|  |  |  |         // Clone FPSCR.
 | 
					
						
							|  |  |  |         ThreadContext32 cur_ctx{}; | 
					
						
							| 
									
										
										
										
											2023-03-07 10:49:41 -05:00
										 |  |  |         m_kernel.System().CurrentArmInterface().SaveContext(cur_ctx); | 
					
						
							| 
									
										
										
										
											2023-03-07 19:18:06 -05:00
										 |  |  | 
 | 
					
						
							|  |  |  |         this->GetContext32().fpscr = cur_ctx.fpscr; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-06-25 22:44:19 -05:00
										 |  |  | Result KThread::SetActivity(Svc::ThreadActivity activity) { | 
					
						
							| 
									
										
										
										
											2021-01-20 13:42:27 -08:00
										 |  |  |     // Lock ourselves.
 | 
					
						
							|  |  |  |     KScopedLightLock lk(activity_pause_lock); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     // Set the activity.
 | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         // Lock the scheduler.
 | 
					
						
							| 
									
										
										
										
											2023-03-07 10:49:41 -05:00
										 |  |  |         KScopedSchedulerLock sl(m_kernel); | 
					
						
							| 
									
										
										
										
											2021-01-20 13:42:27 -08:00
										 |  |  | 
 | 
					
						
							|  |  |  |         // Verify our state.
 | 
					
						
							| 
									
										
										
										
											2021-11-10 22:28:30 -08:00
										 |  |  |         const auto cur_state = this->GetState(); | 
					
						
							| 
									
										
										
										
											2021-01-20 13:42:27 -08:00
										 |  |  |         R_UNLESS((cur_state == ThreadState::Waiting || cur_state == ThreadState::Runnable), | 
					
						
							| 
									
										
										
										
											2021-02-13 10:43:01 +11:00
										 |  |  |                  ResultInvalidState); | 
					
						
							| 
									
										
										
										
											2021-01-20 13:42:27 -08:00
										 |  |  | 
 | 
					
						
							|  |  |  |         // Either pause or resume.
 | 
					
						
							|  |  |  |         if (activity == Svc::ThreadActivity::Paused) { | 
					
						
							|  |  |  |             // Verify that we're not suspended.
 | 
					
						
							| 
									
										
										
										
											2021-11-10 22:28:30 -08:00
										 |  |  |             R_UNLESS(!this->IsSuspendRequested(SuspendType::Thread), ResultInvalidState); | 
					
						
							| 
									
										
										
										
											2021-01-20 13:42:27 -08:00
										 |  |  | 
 | 
					
						
							|  |  |  |             // Suspend.
 | 
					
						
							| 
									
										
										
										
											2021-11-10 22:28:30 -08:00
										 |  |  |             this->RequestSuspend(SuspendType::Thread); | 
					
						
							| 
									
										
										
										
											2021-01-20 13:42:27 -08:00
										 |  |  |         } else { | 
					
						
							|  |  |  |             ASSERT(activity == Svc::ThreadActivity::Runnable); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             // Verify that we're suspended.
 | 
					
						
							| 
									
										
										
										
											2021-11-10 22:28:30 -08:00
										 |  |  |             R_UNLESS(this->IsSuspendRequested(SuspendType::Thread), ResultInvalidState); | 
					
						
							| 
									
										
										
										
											2021-01-20 13:42:27 -08:00
										 |  |  | 
 | 
					
						
							|  |  |  |             // Resume.
 | 
					
						
							| 
									
										
										
										
											2021-11-10 22:28:30 -08:00
										 |  |  |             this->Resume(SuspendType::Thread); | 
					
						
							| 
									
										
										
										
											2021-01-20 13:42:27 -08:00
										 |  |  |         } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     // If the thread is now paused, update the pinned waiter list.
 | 
					
						
							|  |  |  |     if (activity == Svc::ThreadActivity::Paused) { | 
					
						
							| 
									
										
										
										
											2023-03-07 10:49:41 -05:00
										 |  |  |         ThreadQueueImplForKThreadSetProperty wait_queue_(m_kernel, | 
					
						
							| 
									
										
										
										
											2021-11-10 22:28:30 -08:00
										 |  |  |                                                          std::addressof(pinned_waiter_list)); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         bool thread_is_current; | 
					
						
							| 
									
										
										
										
											2021-01-20 13:42:27 -08:00
										 |  |  |         do { | 
					
						
							|  |  |  |             // Lock the scheduler.
 | 
					
						
							| 
									
										
										
										
											2023-03-07 10:49:41 -05:00
										 |  |  |             KScopedSchedulerLock sl(m_kernel); | 
					
						
							| 
									
										
										
										
											2021-01-20 13:42:27 -08:00
										 |  |  | 
 | 
					
						
							|  |  |  |             // Don't do any further management if our termination has been requested.
 | 
					
						
							| 
									
										
										
										
											2021-11-10 22:28:30 -08:00
										 |  |  |             R_SUCCEED_IF(this->IsTerminationRequested()); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             // By default, treat the thread as not current.
 | 
					
						
							|  |  |  |             thread_is_current = false; | 
					
						
							| 
									
										
										
										
											2021-01-20 13:42:27 -08:00
										 |  |  | 
 | 
					
						
							|  |  |  |             // Check whether the thread is pinned.
 | 
					
						
							| 
									
										
										
										
											2021-11-10 22:28:30 -08:00
										 |  |  |             if (this->GetStackParameters().is_pinned) { | 
					
						
							| 
									
										
										
										
											2021-01-20 13:42:27 -08:00
										 |  |  |                 // Verify that the current thread isn't terminating.
 | 
					
						
							| 
									
										
										
										
											2023-03-07 10:49:41 -05:00
										 |  |  |                 R_UNLESS(!GetCurrentThread(m_kernel).IsTerminationRequested(), | 
					
						
							| 
									
										
										
										
											2021-02-13 10:43:01 +11:00
										 |  |  |                          ResultTerminationRequested); | 
					
						
							| 
									
										
										
										
											2021-01-20 13:42:27 -08:00
										 |  |  | 
 | 
					
						
							|  |  |  |                 // Wait until the thread isn't pinned any more.
 | 
					
						
							| 
									
										
										
										
											2023-03-07 10:49:41 -05:00
										 |  |  |                 pinned_waiter_list.push_back(GetCurrentThread(m_kernel)); | 
					
						
							|  |  |  |                 GetCurrentThread(m_kernel).BeginWait(std::addressof(wait_queue_)); | 
					
						
							| 
									
										
										
										
											2021-01-20 13:42:27 -08:00
										 |  |  |             } else { | 
					
						
							|  |  |  |                 // Check if the thread is currently running.
 | 
					
						
							|  |  |  |                 // If it is, we'll need to retry.
 | 
					
						
							|  |  |  |                 for (auto i = 0; i < static_cast<s32>(Core::Hardware::NUM_CPU_CORES); ++i) { | 
					
						
							| 
									
										
										
										
											2023-03-07 10:49:41 -05:00
										 |  |  |                     if (m_kernel.Scheduler(i).GetSchedulerCurrentThread() == this) { | 
					
						
							| 
									
										
										
										
											2021-01-20 13:42:27 -08:00
										 |  |  |                         thread_is_current = true; | 
					
						
							|  |  |  |                         break; | 
					
						
							|  |  |  |                     } | 
					
						
							|  |  |  |                 } | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |         } while (thread_is_current); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-10-14 22:55:51 -07:00
										 |  |  |     R_SUCCEED(); | 
					
						
							| 
									
										
										
										
											2021-01-20 13:42:27 -08:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-06-25 22:44:19 -05:00
										 |  |  | Result KThread::GetThreadContext3(std::vector<u8>& out) { | 
					
						
							| 
									
										
										
										
											2021-01-20 13:42:27 -08:00
										 |  |  |     // Lock ourselves.
 | 
					
						
							|  |  |  |     KScopedLightLock lk{activity_pause_lock}; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     // Get the context.
 | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         // Lock the scheduler.
 | 
					
						
							| 
									
										
										
										
											2023-03-07 10:49:41 -05:00
										 |  |  |         KScopedSchedulerLock sl{m_kernel}; | 
					
						
							| 
									
										
										
										
											2021-01-20 13:42:27 -08:00
										 |  |  | 
 | 
					
						
							|  |  |  |         // Verify that we're suspended.
 | 
					
						
							| 
									
										
										
										
											2021-02-13 10:43:01 +11:00
										 |  |  |         R_UNLESS(IsSuspendRequested(SuspendType::Thread), ResultInvalidState); | 
					
						
							| 
									
										
										
										
											2021-01-20 13:42:27 -08:00
										 |  |  | 
 | 
					
						
							|  |  |  |         // If we're not terminating, get the thread's user context.
 | 
					
						
							|  |  |  |         if (!IsTerminationRequested()) { | 
					
						
							|  |  |  |             if (parent->Is64BitProcess()) { | 
					
						
							|  |  |  |                 // Mask away mode bits, interrupt bits, IL bit, and other reserved bits.
 | 
					
						
							|  |  |  |                 auto context = GetContext64(); | 
					
						
							|  |  |  |                 context.pstate &= 0xFF0FFE20; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |                 out.resize(sizeof(context)); | 
					
						
							|  |  |  |                 std::memcpy(out.data(), &context, sizeof(context)); | 
					
						
							|  |  |  |             } else { | 
					
						
							|  |  |  |                 // Mask away mode bits, interrupt bits, IL bit, and other reserved bits.
 | 
					
						
							|  |  |  |                 auto context = GetContext32(); | 
					
						
							|  |  |  |                 context.cpsr &= 0xFF0FFE20; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |                 out.resize(sizeof(context)); | 
					
						
							|  |  |  |                 std::memcpy(out.data(), &context, sizeof(context)); | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-10-14 22:55:51 -07:00
										 |  |  |     R_SUCCEED(); | 
					
						
							| 
									
										
										
										
											2021-01-20 13:42:27 -08:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-02-23 15:49:42 -05:00
										 |  |  | void KThread::AddHeldLock(LockWithPriorityInheritanceInfo* lock_info) { | 
					
						
							| 
									
										
										
										
											2023-03-07 10:49:41 -05:00
										 |  |  |     ASSERT(KScheduler::IsSchedulerLockedByCurrentThread(m_kernel)); | 
					
						
							| 
									
										
										
										
											2018-10-03 18:47:57 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-02-23 15:49:42 -05:00
										 |  |  |     // Set ourselves as the lock's owner.
 | 
					
						
							|  |  |  |     lock_info->SetOwner(this); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     // Add the lock to our held list.
 | 
					
						
							|  |  |  |     held_lock_info_list.push_front(*lock_info); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-02-23 20:32:03 -05:00
										 |  |  | KThread::LockWithPriorityInheritanceInfo* KThread::FindHeldLock(VAddr address_key_, | 
					
						
							|  |  |  |                                                                 bool is_kernel_address_key_) { | 
					
						
							| 
									
										
										
										
											2023-03-07 10:49:41 -05:00
										 |  |  |     ASSERT(KScheduler::IsSchedulerLockedByCurrentThread(m_kernel)); | 
					
						
							| 
									
										
										
										
											2023-02-23 15:49:42 -05:00
										 |  |  | 
 | 
					
						
							|  |  |  |     // Try to find an existing held lock.
 | 
					
						
							|  |  |  |     for (auto& held_lock : held_lock_info_list) { | 
					
						
							| 
									
										
										
										
											2023-02-23 20:32:03 -05:00
										 |  |  |         if (held_lock.GetAddressKey() == address_key_ && | 
					
						
							|  |  |  |             held_lock.GetIsKernelAddressKey() == is_kernel_address_key_) { | 
					
						
							| 
									
										
										
										
											2023-02-23 15:49:42 -05:00
										 |  |  |             return std::addressof(held_lock); | 
					
						
							| 
									
										
										
										
											2020-12-30 01:14:02 -08:00
										 |  |  |         } | 
					
						
							| 
									
										
										
										
											2019-03-29 17:01:46 -04:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-02-23 15:49:42 -05:00
										 |  |  |     return nullptr; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void KThread::AddWaiterImpl(KThread* thread) { | 
					
						
							| 
									
										
										
										
											2023-03-07 10:49:41 -05:00
										 |  |  |     ASSERT(KScheduler::IsSchedulerLockedByCurrentThread(m_kernel)); | 
					
						
							| 
									
										
										
										
											2023-02-23 15:49:42 -05:00
										 |  |  |     ASSERT(thread->GetConditionVariableTree() == nullptr); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     // Get the thread's address key.
 | 
					
						
							|  |  |  |     const auto address_key_ = thread->GetAddressKey(); | 
					
						
							|  |  |  |     const auto is_kernel_address_key_ = thread->GetIsKernelAddressKey(); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-12-30 01:14:02 -08:00
										 |  |  |     // Keep track of how many kernel waiters we have.
 | 
					
						
							| 
									
										
										
										
											2023-02-23 15:49:42 -05:00
										 |  |  |     if (is_kernel_address_key_) { | 
					
						
							| 
									
										
										
										
											2020-12-30 01:14:02 -08:00
										 |  |  |         ASSERT((num_kernel_waiters++) >= 0); | 
					
						
							| 
									
										
										
										
											2023-03-07 10:49:41 -05:00
										 |  |  |         KScheduler::SetSchedulerUpdateNeeded(m_kernel); | 
					
						
							| 
									
										
										
										
											2020-12-30 01:14:02 -08:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2020-12-28 13:16:43 -08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-02-23 15:49:42 -05:00
										 |  |  |     // Get the relevant lock info.
 | 
					
						
							| 
									
										
										
										
											2023-02-23 20:32:03 -05:00
										 |  |  |     auto* lock_info = this->FindHeldLock(address_key_, is_kernel_address_key_); | 
					
						
							| 
									
										
										
										
											2023-02-23 15:49:42 -05:00
										 |  |  |     if (lock_info == nullptr) { | 
					
						
							|  |  |  |         // Create a new lock for the address key.
 | 
					
						
							|  |  |  |         lock_info = | 
					
						
							| 
									
										
										
										
											2023-03-07 10:49:41 -05:00
										 |  |  |             LockWithPriorityInheritanceInfo::Create(m_kernel, address_key_, is_kernel_address_key_); | 
					
						
							| 
									
										
										
										
											2023-02-23 15:49:42 -05:00
										 |  |  | 
 | 
					
						
							|  |  |  |         // Add the new lock to our list.
 | 
					
						
							|  |  |  |         this->AddHeldLock(lock_info); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     // Add the thread as waiter to the lock info.
 | 
					
						
							|  |  |  |     lock_info->AddWaiter(thread); | 
					
						
							| 
									
										
										
										
											2018-10-03 18:47:57 -04:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-12-30 23:01:08 -08:00
										 |  |  | void KThread::RemoveWaiterImpl(KThread* thread) { | 
					
						
							| 
									
										
										
										
											2023-03-07 10:49:41 -05:00
										 |  |  |     ASSERT(KScheduler::IsSchedulerLockedByCurrentThread(m_kernel)); | 
					
						
							| 
									
										
										
										
											2020-12-30 01:14:02 -08:00
										 |  |  | 
 | 
					
						
							|  |  |  |     // Keep track of how many kernel waiters we have.
 | 
					
						
							| 
									
										
										
										
											2023-02-23 15:49:42 -05:00
										 |  |  |     if (thread->GetIsKernelAddressKey()) { | 
					
						
							| 
									
										
										
										
											2020-12-30 01:14:02 -08:00
										 |  |  |         ASSERT((num_kernel_waiters--) > 0); | 
					
						
							| 
									
										
										
										
											2023-03-07 10:49:41 -05:00
										 |  |  |         KScheduler::SetSchedulerUpdateNeeded(m_kernel); | 
					
						
							| 
									
										
										
										
											2018-08-12 16:35:27 -05:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-02-23 15:49:42 -05:00
										 |  |  |     // Get the info for the lock the thread is waiting on.
 | 
					
						
							|  |  |  |     auto* lock_info = thread->GetWaitingLockInfo(); | 
					
						
							|  |  |  |     ASSERT(lock_info->GetOwner() == this); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-12-30 01:14:02 -08:00
										 |  |  |     // Remove the waiter.
 | 
					
						
							| 
									
										
										
										
											2023-02-23 15:49:42 -05:00
										 |  |  |     if (lock_info->RemoveWaiter(thread)) { | 
					
						
							|  |  |  |         held_lock_info_list.erase(held_lock_info_list.iterator_to(*lock_info)); | 
					
						
							| 
									
										
										
										
											2023-03-07 10:49:41 -05:00
										 |  |  |         LockWithPriorityInheritanceInfo::Free(m_kernel, lock_info); | 
					
						
							| 
									
										
										
										
											2023-02-23 15:49:42 -05:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2020-12-30 01:14:02 -08:00
										 |  |  | } | 
					
						
							| 
									
										
										
										
											2018-08-12 16:35:27 -05:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-02-23 15:49:42 -05:00
										 |  |  | void KThread::RestorePriority(KernelCore& kernel, KThread* thread) { | 
					
						
							|  |  |  |     ASSERT(KScheduler::IsSchedulerLockedByCurrentThread(kernel)); | 
					
						
							| 
									
										
										
										
											2019-03-15 01:02:13 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-02-23 15:49:42 -05:00
										 |  |  |     while (thread != nullptr) { | 
					
						
							| 
									
										
										
										
											2020-12-30 01:14:02 -08:00
										 |  |  |         // We want to inherit priority where possible.
 | 
					
						
							|  |  |  |         s32 new_priority = thread->GetBasePriority(); | 
					
						
							| 
									
										
										
										
											2023-02-23 15:49:42 -05:00
										 |  |  |         for (const auto& held_lock : thread->held_lock_info_list) { | 
					
						
							|  |  |  |             new_priority = | 
					
						
							|  |  |  |                 std::min(new_priority, held_lock.GetHighestPriorityWaiter()->GetPriority()); | 
					
						
							| 
									
										
										
										
											2020-12-30 01:14:02 -08:00
										 |  |  |         } | 
					
						
							| 
									
										
										
										
											2019-03-15 01:02:13 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-12-30 01:14:02 -08:00
										 |  |  |         // If the priority we would inherit is not different from ours, don't do anything.
 | 
					
						
							|  |  |  |         if (new_priority == thread->GetPriority()) { | 
					
						
							|  |  |  |             return; | 
					
						
							|  |  |  |         } | 
					
						
							| 
									
										
										
										
											2018-04-20 20:15:16 -05:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-02-23 15:49:42 -05:00
										 |  |  |         // Get the owner of whatever lock this thread is waiting on.
 | 
					
						
							|  |  |  |         KThread* const lock_owner = thread->GetLockOwner(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         // If the thread is waiting on some lock, remove it as a waiter to prevent violating red
 | 
					
						
							|  |  |  |         // black tree invariants.
 | 
					
						
							|  |  |  |         if (lock_owner != nullptr) { | 
					
						
							|  |  |  |             lock_owner->RemoveWaiterImpl(thread); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-12-30 01:14:02 -08:00
										 |  |  |         // Ensure we don't violate condition variable red black tree invariants.
 | 
					
						
							|  |  |  |         if (auto* cv_tree = thread->GetConditionVariableTree(); cv_tree != nullptr) { | 
					
						
							| 
									
										
										
										
											2023-02-23 15:49:42 -05:00
										 |  |  |             BeforeUpdatePriority(kernel, cv_tree, thread); | 
					
						
							| 
									
										
										
										
											2020-12-30 01:14:02 -08:00
										 |  |  |         } | 
					
						
							| 
									
										
										
										
											2018-08-12 16:35:27 -05:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-12-30 01:14:02 -08:00
										 |  |  |         // Change the priority.
 | 
					
						
							|  |  |  |         const s32 old_priority = thread->GetPriority(); | 
					
						
							|  |  |  |         thread->SetPriority(new_priority); | 
					
						
							| 
									
										
										
										
											2019-03-15 01:02:13 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-12-30 01:14:02 -08:00
										 |  |  |         // Restore the condition variable, if relevant.
 | 
					
						
							|  |  |  |         if (auto* cv_tree = thread->GetConditionVariableTree(); cv_tree != nullptr) { | 
					
						
							| 
									
										
										
										
											2023-02-23 15:49:42 -05:00
										 |  |  |             AfterUpdatePriority(kernel, cv_tree, thread); | 
					
						
							| 
									
										
										
										
											2020-12-30 01:14:02 -08:00
										 |  |  |         } | 
					
						
							| 
									
										
										
										
											2018-08-12 16:35:27 -05:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-02-23 15:49:42 -05:00
										 |  |  |         // If we removed the thread from some lock's waiting list, add it back.
 | 
					
						
							|  |  |  |         if (lock_owner != nullptr) { | 
					
						
							|  |  |  |             lock_owner->AddWaiterImpl(thread); | 
					
						
							| 
									
										
										
										
											2019-03-14 21:51:03 -04:00
										 |  |  |         } | 
					
						
							| 
									
										
										
										
											2018-04-20 20:15:16 -05:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-02-23 15:49:42 -05:00
										 |  |  |         // Update the scheduler.
 | 
					
						
							|  |  |  |         KScheduler::OnThreadPriorityChanged(kernel, thread, old_priority); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         // Continue inheriting priority.
 | 
					
						
							| 
									
										
										
										
											2020-12-30 01:14:02 -08:00
										 |  |  |         thread = lock_owner; | 
					
						
							| 
									
										
										
										
											2019-03-14 21:51:03 -04:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2020-12-30 01:14:02 -08:00
										 |  |  | } | 
					
						
							| 
									
										
										
										
											2018-04-20 20:15:16 -05:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-12-30 23:01:08 -08:00
										 |  |  | void KThread::AddWaiter(KThread* thread) { | 
					
						
							| 
									
										
										
										
											2023-02-23 15:49:42 -05:00
										 |  |  |     this->AddWaiterImpl(thread); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     // If the thread has a higher priority than us, we should inherit.
 | 
					
						
							|  |  |  |     if (thread->GetPriority() < this->GetPriority()) { | 
					
						
							| 
									
										
										
										
											2023-03-07 10:49:41 -05:00
										 |  |  |         RestorePriority(m_kernel, this); | 
					
						
							| 
									
										
										
										
											2023-02-23 15:49:42 -05:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2020-12-30 01:14:02 -08:00
										 |  |  | } | 
					
						
							| 
									
										
										
										
											2019-11-14 20:13:18 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-12-30 23:01:08 -08:00
										 |  |  | void KThread::RemoveWaiter(KThread* thread) { | 
					
						
							| 
									
										
										
										
											2023-02-23 15:49:42 -05:00
										 |  |  |     this->RemoveWaiterImpl(thread); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     // If our priority is the same as the thread's (and we've inherited), we may need to restore to
 | 
					
						
							|  |  |  |     // lower priority.
 | 
					
						
							|  |  |  |     if (this->GetPriority() == thread->GetPriority() && | 
					
						
							|  |  |  |         this->GetPriority() < this->GetBasePriority()) { | 
					
						
							| 
									
										
										
										
											2023-03-07 10:49:41 -05:00
										 |  |  |         RestorePriority(m_kernel, this); | 
					
						
							| 
									
										
										
										
											2023-02-23 15:49:42 -05:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2020-12-30 01:14:02 -08:00
										 |  |  | } | 
					
						
							| 
									
										
										
										
											2018-04-20 20:15:16 -05:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-02-23 20:32:03 -05:00
										 |  |  | KThread* KThread::RemoveWaiterByKey(bool* out_has_waiters, VAddr key, bool is_kernel_address_key_) { | 
					
						
							| 
									
										
										
										
											2023-03-07 10:49:41 -05:00
										 |  |  |     ASSERT(KScheduler::IsSchedulerLockedByCurrentThread(m_kernel)); | 
					
						
							| 
									
										
										
										
											2019-11-14 20:13:18 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-02-23 15:49:42 -05:00
										 |  |  |     // Get the relevant lock info.
 | 
					
						
							| 
									
										
										
										
											2023-02-23 20:32:03 -05:00
										 |  |  |     auto* lock_info = this->FindHeldLock(key, is_kernel_address_key_); | 
					
						
							| 
									
										
										
										
											2023-02-23 15:49:42 -05:00
										 |  |  |     if (lock_info == nullptr) { | 
					
						
							|  |  |  |         *out_has_waiters = false; | 
					
						
							|  |  |  |         return nullptr; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2020-12-30 01:14:02 -08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-02-23 15:49:42 -05:00
										 |  |  |     // Remove the lock info from our held list.
 | 
					
						
							|  |  |  |     held_lock_info_list.erase(held_lock_info_list.iterator_to(*lock_info)); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     // Keep track of how many kernel waiters we have.
 | 
					
						
							|  |  |  |     if (lock_info->GetIsKernelAddressKey()) { | 
					
						
							|  |  |  |         num_kernel_waiters -= lock_info->GetWaiterCount(); | 
					
						
							|  |  |  |         ASSERT(num_kernel_waiters >= 0); | 
					
						
							| 
									
										
										
										
											2023-03-07 10:49:41 -05:00
										 |  |  |         KScheduler::SetSchedulerUpdateNeeded(m_kernel); | 
					
						
							| 
									
										
										
										
											2023-02-23 15:49:42 -05:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     ASSERT(lock_info->GetWaiterCount() > 0); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     // Remove the highest priority waiter from the lock to be the next owner.
 | 
					
						
							|  |  |  |     KThread* next_lock_owner = lock_info->GetHighestPriorityWaiter(); | 
					
						
							|  |  |  |     if (lock_info->RemoveWaiter(next_lock_owner)) { | 
					
						
							|  |  |  |         // The new owner was the only waiter.
 | 
					
						
							|  |  |  |         *out_has_waiters = false; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         // Free the lock info, since it has no waiters.
 | 
					
						
							| 
									
										
										
										
											2023-03-07 10:49:41 -05:00
										 |  |  |         LockWithPriorityInheritanceInfo::Free(m_kernel, lock_info); | 
					
						
							| 
									
										
										
										
											2023-02-23 15:49:42 -05:00
										 |  |  |     } else { | 
					
						
							|  |  |  |         // There are additional waiters on the lock.
 | 
					
						
							|  |  |  |         *out_has_waiters = true; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         // Add the lock to the new owner's held list.
 | 
					
						
							|  |  |  |         next_lock_owner->AddHeldLock(lock_info); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         // Keep track of any kernel waiters for the new owner.
 | 
					
						
							|  |  |  |         if (lock_info->GetIsKernelAddressKey()) { | 
					
						
							|  |  |  |             next_lock_owner->num_kernel_waiters += lock_info->GetWaiterCount(); | 
					
						
							|  |  |  |             ASSERT(next_lock_owner->num_kernel_waiters > 0); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             // NOTE: No need to set scheduler update needed, because we will have already done so
 | 
					
						
							|  |  |  |             // when removing earlier.
 | 
					
						
							| 
									
										
										
										
											2020-12-30 01:14:02 -08:00
										 |  |  |         } | 
					
						
							| 
									
										
										
										
											2019-03-15 01:02:13 -04:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-02-23 15:49:42 -05:00
										 |  |  |     // If our priority is the same as the next owner's (and we've inherited), we may need to restore
 | 
					
						
							|  |  |  |     // to lower priority.
 | 
					
						
							|  |  |  |     if (this->GetPriority() == next_lock_owner->GetPriority() && | 
					
						
							|  |  |  |         this->GetPriority() < this->GetBasePriority()) { | 
					
						
							| 
									
										
										
										
											2023-03-07 10:49:41 -05:00
										 |  |  |         RestorePriority(m_kernel, this); | 
					
						
							| 
									
										
										
										
											2023-02-23 15:49:42 -05:00
										 |  |  |         // NOTE: No need to restore priority on the next lock owner, because it was already the
 | 
					
						
							|  |  |  |         // highest priority waiter on the lock.
 | 
					
						
							| 
									
										
										
										
											2020-12-30 01:14:02 -08:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2019-03-15 01:02:13 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-02-23 15:49:42 -05:00
										 |  |  |     // Return the next lock owner.
 | 
					
						
							| 
									
										
										
										
											2020-12-30 01:14:02 -08:00
										 |  |  |     return next_lock_owner; | 
					
						
							| 
									
										
										
										
											2018-04-20 20:15:16 -05:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-06-25 22:44:19 -05:00
										 |  |  | Result KThread::Run() { | 
					
						
							| 
									
										
										
										
											2021-01-20 13:42:27 -08:00
										 |  |  |     while (true) { | 
					
						
							| 
									
										
										
										
											2023-03-07 10:49:41 -05:00
										 |  |  |         KScopedSchedulerLock lk{m_kernel}; | 
					
						
							| 
									
										
										
										
											2020-03-07 12:44:35 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-01-20 13:42:27 -08:00
										 |  |  |         // If either this thread or the current thread are requesting termination, note it.
 | 
					
						
							| 
									
										
										
										
											2021-02-13 10:43:01 +11:00
										 |  |  |         R_UNLESS(!IsTerminationRequested(), ResultTerminationRequested); | 
					
						
							| 
									
										
										
										
											2023-03-07 10:49:41 -05:00
										 |  |  |         R_UNLESS(!GetCurrentThread(m_kernel).IsTerminationRequested(), ResultTerminationRequested); | 
					
						
							| 
									
										
										
										
											2020-03-07 12:44:35 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-01-20 13:42:27 -08:00
										 |  |  |         // Ensure our thread state is correct.
 | 
					
						
							| 
									
										
										
										
											2021-02-13 10:43:01 +11:00
										 |  |  |         R_UNLESS(GetState() == ThreadState::Initialized, ResultInvalidState); | 
					
						
							| 
									
										
										
										
											2020-03-07 12:44:35 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-01-20 13:42:27 -08:00
										 |  |  |         // If the current thread has been asked to suspend, suspend it and retry.
 | 
					
						
							| 
									
										
										
										
											2023-03-07 10:49:41 -05:00
										 |  |  |         if (GetCurrentThread(m_kernel).IsSuspended()) { | 
					
						
							|  |  |  |             GetCurrentThread(m_kernel).UpdateState(); | 
					
						
							| 
									
										
										
										
											2021-01-20 13:42:27 -08:00
										 |  |  |             continue; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         // If we're not a kernel thread and we've been asked to suspend, suspend ourselves.
 | 
					
						
							| 
									
										
										
										
											2022-01-14 16:31:47 -08:00
										 |  |  |         if (KProcess* owner = this->GetOwnerProcess(); owner != nullptr) { | 
					
						
							|  |  |  |             if (IsUserThread() && IsSuspended()) { | 
					
						
							|  |  |  |                 this->UpdateState(); | 
					
						
							|  |  |  |             } | 
					
						
							| 
									
										
										
										
											2022-01-22 21:09:45 -08:00
										 |  |  |             owner->IncrementRunningThreadCount(); | 
					
						
							| 
									
										
										
										
											2021-01-20 13:42:27 -08:00
										 |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         // Set our state and finish.
 | 
					
						
							|  |  |  |         SetState(ThreadState::Runnable); | 
					
						
							| 
									
										
										
										
											2021-08-06 23:04:32 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-10-14 22:55:51 -07:00
										 |  |  |         R_SUCCEED(); | 
					
						
							| 
									
										
										
										
											2020-03-07 12:44:35 -04:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2021-01-20 13:42:27 -08:00
										 |  |  | } | 
					
						
							| 
									
										
										
										
											2018-12-03 12:25:27 -05:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-01-20 13:42:27 -08:00
										 |  |  | void KThread::Exit() { | 
					
						
							| 
									
										
										
										
											2023-03-07 10:49:41 -05:00
										 |  |  |     ASSERT(this == GetCurrentThreadPointer(m_kernel)); | 
					
						
							| 
									
										
										
										
											2021-01-20 13:42:27 -08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-01-22 21:09:45 -08:00
										 |  |  |     // Release the thread resource hint, running thread count from parent.
 | 
					
						
							| 
									
										
										
										
											2021-01-20 13:42:27 -08:00
										 |  |  |     if (parent != nullptr) { | 
					
						
							| 
									
										
										
										
											2022-11-03 10:22:05 -04:00
										 |  |  |         parent->GetResourceLimit()->Release(Kernel::LimitableResource::ThreadCountMax, 0, 1); | 
					
						
							| 
									
										
										
										
											2021-01-20 13:42:27 -08:00
										 |  |  |         resource_limit_release_hint = true; | 
					
						
							| 
									
										
										
										
											2022-01-22 21:09:45 -08:00
										 |  |  |         parent->DecrementRunningThreadCount(); | 
					
						
							| 
									
										
										
										
											2021-01-20 13:42:27 -08:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     // Perform termination.
 | 
					
						
							|  |  |  |     { | 
					
						
							| 
									
										
										
										
											2023-03-07 10:49:41 -05:00
										 |  |  |         KScopedSchedulerLock sl{m_kernel}; | 
					
						
							| 
									
										
										
										
											2021-01-20 13:42:27 -08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-01-14 16:31:47 -08:00
										 |  |  |         // Disallow all suspension.
 | 
					
						
							|  |  |  |         suspend_allowed_flags = 0; | 
					
						
							|  |  |  |         this->UpdateState(); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-01-20 13:42:27 -08:00
										 |  |  |         // Disallow all suspension.
 | 
					
						
							|  |  |  |         suspend_allowed_flags = 0; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         // Start termination.
 | 
					
						
							|  |  |  |         StartTermination(); | 
					
						
							| 
									
										
										
										
											2022-01-14 16:36:10 -08:00
										 |  |  | 
 | 
					
						
							|  |  |  |         // Register the thread as a work task.
 | 
					
						
							| 
									
										
										
										
											2023-03-07 10:49:41 -05:00
										 |  |  |         KWorkerTaskManager::AddTask(m_kernel, KWorkerTaskManager::WorkerType::Exit, this); | 
					
						
							| 
									
										
										
										
											2018-12-03 12:25:27 -05:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2022-06-26 18:52:16 -04:00
										 |  |  | 
 | 
					
						
							|  |  |  |     UNREACHABLE_MSG("KThread::Exit() would return"); | 
					
						
							| 
									
										
										
										
											2018-12-03 12:25:27 -05:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-09-05 18:19:30 -07:00
										 |  |  | Result KThread::Terminate() { | 
					
						
							| 
									
										
										
										
											2023-03-07 10:49:41 -05:00
										 |  |  |     ASSERT(this != GetCurrentThreadPointer(m_kernel)); | 
					
						
							| 
									
										
										
										
											2022-09-05 18:19:30 -07:00
										 |  |  | 
 | 
					
						
							|  |  |  |     // Request the thread terminate if it hasn't already.
 | 
					
						
							|  |  |  |     if (const auto new_state = this->RequestTerminate(); new_state != ThreadState::Terminated) { | 
					
						
							|  |  |  |         // If the thread isn't terminated, wait for it to terminate.
 | 
					
						
							|  |  |  |         s32 index; | 
					
						
							|  |  |  |         KSynchronizationObject* objects[] = {this}; | 
					
						
							| 
									
										
										
										
											2023-03-07 10:49:41 -05:00
										 |  |  |         R_TRY(KSynchronizationObject::Wait(m_kernel, std::addressof(index), objects, 1, | 
					
						
							| 
									
										
										
										
											2022-09-05 18:19:30 -07:00
										 |  |  |                                            Svc::WaitInfinite)); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-10-14 22:55:51 -07:00
										 |  |  |     R_SUCCEED(); | 
					
						
							| 
									
										
										
										
											2022-09-05 18:19:30 -07:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | ThreadState KThread::RequestTerminate() { | 
					
						
							| 
									
										
										
										
											2023-03-07 10:49:41 -05:00
										 |  |  |     ASSERT(this != GetCurrentThreadPointer(m_kernel)); | 
					
						
							| 
									
										
										
										
											2022-09-05 18:19:30 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-03-07 10:49:41 -05:00
										 |  |  |     KScopedSchedulerLock sl{m_kernel}; | 
					
						
							| 
									
										
										
										
											2022-09-05 18:19:30 -07:00
										 |  |  | 
 | 
					
						
							|  |  |  |     // Determine if this is the first termination request.
 | 
					
						
							|  |  |  |     const bool first_request = [&]() -> bool { | 
					
						
							|  |  |  |         // Perform an atomic compare-and-swap from false to true.
 | 
					
						
							|  |  |  |         bool expected = false; | 
					
						
							|  |  |  |         return termination_requested.compare_exchange_strong(expected, true); | 
					
						
							|  |  |  |     }(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     // If this is the first request, start termination procedure.
 | 
					
						
							|  |  |  |     if (first_request) { | 
					
						
							|  |  |  |         // If the thread is in initialized state, just change state to terminated.
 | 
					
						
							|  |  |  |         if (this->GetState() == ThreadState::Initialized) { | 
					
						
							|  |  |  |             thread_state = ThreadState::Terminated; | 
					
						
							|  |  |  |             return ThreadState::Terminated; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         // Register the terminating dpc.
 | 
					
						
							|  |  |  |         this->RegisterDpc(DpcFlag::Terminating); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         // If the thread is pinned, unpin it.
 | 
					
						
							|  |  |  |         if (this->GetStackParameters().is_pinned) { | 
					
						
							|  |  |  |             this->GetOwnerProcess()->UnpinThread(this); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         // If the thread is suspended, continue it.
 | 
					
						
							|  |  |  |         if (this->IsSuspended()) { | 
					
						
							|  |  |  |             suspend_allowed_flags = 0; | 
					
						
							|  |  |  |             this->UpdateState(); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         // Change the thread's priority to be higher than any system thread's.
 | 
					
						
							| 
									
										
										
										
											2023-02-23 15:49:42 -05:00
										 |  |  |         this->IncreaseBasePriority(TerminatingThreadPriority); | 
					
						
							| 
									
										
										
										
											2022-09-05 18:19:30 -07:00
										 |  |  | 
 | 
					
						
							|  |  |  |         // If the thread is runnable, send a termination interrupt to other cores.
 | 
					
						
							|  |  |  |         if (this->GetState() == ThreadState::Runnable) { | 
					
						
							| 
									
										
										
										
											2023-03-07 10:49:41 -05:00
										 |  |  |             if (const u64 core_mask = physical_affinity_mask.GetAffinityMask() & | 
					
						
							|  |  |  |                                       ~(1ULL << GetCurrentCoreId(m_kernel)); | 
					
						
							| 
									
										
										
										
											2022-09-05 18:19:30 -07:00
										 |  |  |                 core_mask != 0) { | 
					
						
							| 
									
										
										
										
											2023-03-07 10:49:41 -05:00
										 |  |  |                 Kernel::KInterruptManager::SendInterProcessorInterrupt(m_kernel, core_mask); | 
					
						
							| 
									
										
										
										
											2022-09-05 18:19:30 -07:00
										 |  |  |             } | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         // Wake up the thread.
 | 
					
						
							|  |  |  |         if (this->GetState() == ThreadState::Waiting) { | 
					
						
							|  |  |  |             wait_queue->CancelWait(this, ResultTerminationRequested, true); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     return this->GetState(); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-06-25 22:44:19 -05:00
										 |  |  | Result KThread::Sleep(s64 timeout) { | 
					
						
							| 
									
										
										
										
											2023-03-07 10:49:41 -05:00
										 |  |  |     ASSERT(!m_kernel.GlobalSchedulerContext().IsLocked()); | 
					
						
							|  |  |  |     ASSERT(this == GetCurrentThreadPointer(m_kernel)); | 
					
						
							| 
									
										
										
										
											2021-01-20 13:42:27 -08:00
										 |  |  |     ASSERT(timeout > 0); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-03-07 10:49:41 -05:00
										 |  |  |     ThreadQueueImplForKThreadSleep wait_queue_(m_kernel); | 
					
						
							| 
									
										
										
										
											2023-03-07 20:48:46 -05:00
										 |  |  |     KHardwareTimer* timer{}; | 
					
						
							| 
									
										
										
										
											2020-02-24 22:04:12 -04:00
										 |  |  |     { | 
					
						
							| 
									
										
										
										
											2021-01-20 13:42:27 -08:00
										 |  |  |         // Setup the scheduling lock and sleep.
 | 
					
						
							| 
									
										
										
										
											2023-03-07 10:49:41 -05:00
										 |  |  |         KScopedSchedulerLockAndSleep slp(m_kernel, std::addressof(timer), this, timeout); | 
					
						
							| 
									
										
										
										
											2021-01-20 13:42:27 -08:00
										 |  |  | 
 | 
					
						
							|  |  |  |         // Check if the thread should terminate.
 | 
					
						
							| 
									
										
										
										
											2021-11-10 22:28:30 -08:00
										 |  |  |         if (this->IsTerminationRequested()) { | 
					
						
							| 
									
										
										
										
											2021-01-20 13:42:27 -08:00
										 |  |  |             slp.CancelSleep(); | 
					
						
							| 
									
										
										
										
											2022-10-14 22:55:51 -07:00
										 |  |  |             R_THROW(ResultTerminationRequested); | 
					
						
							| 
									
										
										
										
											2021-01-20 13:42:27 -08:00
										 |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-11-10 22:28:30 -08:00
										 |  |  |         // Wait for the sleep to end.
 | 
					
						
							| 
									
										
										
										
											2023-03-07 20:48:46 -05:00
										 |  |  |         wait_queue_.SetHardwareTimer(timer); | 
					
						
							| 
									
										
										
										
											2021-11-28 00:40:25 -08:00
										 |  |  |         this->BeginWait(std::addressof(wait_queue_)); | 
					
						
							| 
									
										
										
										
											2021-01-10 14:29:02 -08:00
										 |  |  |         SetWaitReasonForDebugging(ThreadWaitReasonForDebugging::Sleep); | 
					
						
							| 
									
										
										
										
											2020-02-24 22:04:12 -04:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2019-03-15 23:28:29 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-10-14 22:55:51 -07:00
										 |  |  |     R_SUCCEED(); | 
					
						
							| 
									
										
										
										
											2019-03-15 23:28:29 -04:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-10-23 05:24:38 -04:00
										 |  |  | void KThread::RequestDummyThreadWait() { | 
					
						
							| 
									
										
										
										
											2023-03-07 10:49:41 -05:00
										 |  |  |     ASSERT(KScheduler::IsSchedulerLockedByCurrentThread(m_kernel)); | 
					
						
							| 
									
										
										
										
											2022-10-23 05:24:38 -04:00
										 |  |  |     ASSERT(this->IsDummyThread()); | 
					
						
							| 
									
										
										
										
											2022-01-21 17:10:11 -08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-10-23 05:24:38 -04:00
										 |  |  |     // We will block when the scheduler lock is released.
 | 
					
						
							|  |  |  |     dummy_thread_runnable.store(false); | 
					
						
							|  |  |  | } | 
					
						
							| 
									
										
										
										
											2022-01-21 17:10:11 -08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-10-23 05:24:38 -04:00
										 |  |  | void KThread::DummyThreadBeginWait() { | 
					
						
							| 
									
										
										
										
											2023-03-07 10:49:41 -05:00
										 |  |  |     if (!this->IsDummyThread() || m_kernel.IsPhantomModeForSingleCore()) { | 
					
						
							| 
									
										
										
										
											2022-10-30 18:44:29 -04:00
										 |  |  |         // Occurs in single core mode.
 | 
					
						
							|  |  |  |         return; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2022-07-11 10:13:13 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-10-23 05:24:38 -04:00
										 |  |  |     // Block until runnable is no longer false.
 | 
					
						
							|  |  |  |     dummy_thread_runnable.wait(false); | 
					
						
							| 
									
										
										
										
											2022-01-21 17:10:11 -08:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-10-23 05:24:38 -04:00
										 |  |  | void KThread::DummyThreadEndWait() { | 
					
						
							| 
									
										
										
										
											2023-03-07 10:49:41 -05:00
										 |  |  |     ASSERT(KScheduler::IsSchedulerLockedByCurrentThread(m_kernel)); | 
					
						
							| 
									
										
										
										
											2022-10-23 05:24:38 -04:00
										 |  |  |     ASSERT(this->IsDummyThread()); | 
					
						
							| 
									
										
										
										
											2022-01-21 17:10:11 -08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-04-11 20:57:32 -07:00
										 |  |  |     // Wake up the waiting thread.
 | 
					
						
							| 
									
										
										
										
											2022-10-23 05:24:38 -04:00
										 |  |  |     dummy_thread_runnable.store(true); | 
					
						
							|  |  |  |     dummy_thread_runnable.notify_one(); | 
					
						
							| 
									
										
										
										
											2022-01-21 17:10:11 -08:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-11-09 19:02:11 -08:00
										 |  |  | void KThread::BeginWait(KThreadQueue* queue) { | 
					
						
							|  |  |  |     // Set our state as waiting.
 | 
					
						
							|  |  |  |     SetState(ThreadState::Waiting); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     // Set our wait queue.
 | 
					
						
							| 
									
										
										
										
											2021-11-28 00:40:25 -08:00
										 |  |  |     wait_queue = queue; | 
					
						
							| 
									
										
										
										
											2021-11-09 19:02:11 -08:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-06-25 22:44:19 -05:00
										 |  |  | void KThread::NotifyAvailable(KSynchronizationObject* signaled_object, Result wait_result_) { | 
					
						
							| 
									
										
										
										
											2021-11-09 19:02:11 -08:00
										 |  |  |     // Lock the scheduler.
 | 
					
						
							| 
									
										
										
										
											2023-03-07 10:49:41 -05:00
										 |  |  |     KScopedSchedulerLock sl(m_kernel); | 
					
						
							| 
									
										
										
										
											2021-11-09 19:02:11 -08:00
										 |  |  | 
 | 
					
						
							|  |  |  |     // If we're waiting, notify our queue that we're available.
 | 
					
						
							|  |  |  |     if (GetState() == ThreadState::Waiting) { | 
					
						
							| 
									
										
										
										
											2021-11-28 00:40:25 -08:00
										 |  |  |         wait_queue->NotifyAvailable(this, signaled_object, wait_result_); | 
					
						
							| 
									
										
										
										
											2021-11-09 19:02:11 -08:00
										 |  |  |     } | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-06-25 22:44:19 -05:00
										 |  |  | void KThread::EndWait(Result wait_result_) { | 
					
						
							| 
									
										
										
										
											2021-11-09 19:02:11 -08:00
										 |  |  |     // Lock the scheduler.
 | 
					
						
							| 
									
										
										
										
											2023-03-07 10:49:41 -05:00
										 |  |  |     KScopedSchedulerLock sl(m_kernel); | 
					
						
							| 
									
										
										
										
											2021-11-09 19:02:11 -08:00
										 |  |  | 
 | 
					
						
							|  |  |  |     // If we're waiting, notify our queue that we're available.
 | 
					
						
							|  |  |  |     if (GetState() == ThreadState::Waiting) { | 
					
						
							| 
									
										
										
										
											2022-01-18 17:56:08 -08:00
										 |  |  |         if (wait_queue == nullptr) { | 
					
						
							|  |  |  |             // This should never happen, but avoid a hard crash below to get this logged.
 | 
					
						
							|  |  |  |             ASSERT_MSG(false, "wait_queue is nullptr!"); | 
					
						
							|  |  |  |             return; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-11-28 00:40:25 -08:00
										 |  |  |         wait_queue->EndWait(this, wait_result_); | 
					
						
							| 
									
										
										
										
											2021-11-09 19:02:11 -08:00
										 |  |  |     } | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-06-25 22:44:19 -05:00
										 |  |  | void KThread::CancelWait(Result wait_result_, bool cancel_timer_task) { | 
					
						
							| 
									
										
										
										
											2021-11-09 19:02:11 -08:00
										 |  |  |     // Lock the scheduler.
 | 
					
						
							| 
									
										
										
										
											2023-03-07 10:49:41 -05:00
										 |  |  |     KScopedSchedulerLock sl(m_kernel); | 
					
						
							| 
									
										
										
										
											2021-11-09 19:02:11 -08:00
										 |  |  | 
 | 
					
						
							|  |  |  |     // If we're waiting, notify our queue that we're available.
 | 
					
						
							|  |  |  |     if (GetState() == ThreadState::Waiting) { | 
					
						
							| 
									
										
										
										
											2021-11-28 00:40:25 -08:00
										 |  |  |         wait_queue->CancelWait(this, wait_result_, cancel_timer_task); | 
					
						
							| 
									
										
										
										
											2021-11-09 19:02:11 -08:00
										 |  |  |     } | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-01-20 13:42:27 -08:00
										 |  |  | void KThread::SetState(ThreadState state) { | 
					
						
							| 
									
										
										
										
											2023-03-07 10:49:41 -05:00
										 |  |  |     KScopedSchedulerLock sl{m_kernel}; | 
					
						
							| 
									
										
										
										
											2021-01-20 13:42:27 -08:00
										 |  |  | 
 | 
					
						
							|  |  |  |     // Clear debugging state
 | 
					
						
							|  |  |  |     SetMutexWaitAddressForDebugging({}); | 
					
						
							|  |  |  |     SetWaitReasonForDebugging({}); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-04-11 20:57:32 -07:00
										 |  |  |     const ThreadState old_state = thread_state.load(std::memory_order_relaxed); | 
					
						
							|  |  |  |     thread_state.store( | 
					
						
							|  |  |  |         static_cast<ThreadState>((old_state & ~ThreadState::Mask) | (state & ThreadState::Mask)), | 
					
						
							|  |  |  |         std::memory_order_relaxed); | 
					
						
							|  |  |  |     if (thread_state.load(std::memory_order_relaxed) != old_state) { | 
					
						
							| 
									
										
										
										
											2023-03-07 10:49:41 -05:00
										 |  |  |         KScheduler::OnThreadStateChanged(m_kernel, this, old_state); | 
					
						
							| 
									
										
										
										
											2021-01-20 13:42:27 -08:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2020-03-07 12:44:35 -04:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-03-05 17:08:17 -08:00
										 |  |  | std::shared_ptr<Common::Fiber>& KThread::GetHostContext() { | 
					
						
							|  |  |  |     return host_context; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-06-16 10:35:52 -04:00
										 |  |  | void SetCurrentThread(KernelCore& kernel, KThread* thread) { | 
					
						
							|  |  |  |     kernel.SetCurrentEmuThread(thread); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-01-20 13:42:27 -08:00
										 |  |  | KThread* GetCurrentThreadPointer(KernelCore& kernel) { | 
					
						
							| 
									
										
										
										
											2021-01-21 13:00:16 -08:00
										 |  |  |     return kernel.GetCurrentEmuThread(); | 
					
						
							| 
									
										
										
										
											2021-01-20 13:42:27 -08:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | KThread& GetCurrentThread(KernelCore& kernel) { | 
					
						
							|  |  |  |     return *GetCurrentThreadPointer(kernel); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-02-13 10:44:41 -05:00
										 |  |  | KProcess* GetCurrentProcessPointer(KernelCore& kernel) { | 
					
						
							|  |  |  |     return GetCurrentThread(kernel).GetOwnerProcess(); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | KProcess& GetCurrentProcess(KernelCore& kernel) { | 
					
						
							|  |  |  |     return *GetCurrentProcessPointer(kernel); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-01-20 13:42:27 -08:00
										 |  |  | s32 GetCurrentCoreId(KernelCore& kernel) { | 
					
						
							|  |  |  |     return GetCurrentThread(kernel).GetCurrentCore(); | 
					
						
							| 
									
										
										
										
											2019-03-29 17:01:46 -04:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-08-06 23:04:32 -07:00
										 |  |  | KScopedDisableDispatch::~KScopedDisableDispatch() { | 
					
						
							| 
									
										
										
										
											2021-11-25 20:46:17 -08:00
										 |  |  |     // If we are shutting down the kernel, none of this is relevant anymore.
 | 
					
						
							| 
									
										
										
										
											2023-03-07 10:49:41 -05:00
										 |  |  |     if (m_kernel.IsShuttingDown()) { | 
					
						
							| 
									
										
										
										
											2021-11-25 20:46:17 -08:00
										 |  |  |         return; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-03-07 10:49:41 -05:00
										 |  |  |     if (GetCurrentThread(m_kernel).GetDisableDispatchCount() <= 1) { | 
					
						
							|  |  |  |         auto* scheduler = m_kernel.CurrentScheduler(); | 
					
						
							| 
									
										
										
										
											2021-08-06 23:04:32 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-03-07 10:49:41 -05:00
										 |  |  |         if (scheduler && !m_kernel.IsPhantomModeForSingleCore()) { | 
					
						
							| 
									
										
										
										
											2021-08-06 23:04:32 -07:00
										 |  |  |             scheduler->RescheduleCurrentCore(); | 
					
						
							| 
									
										
										
										
											2022-07-11 10:13:13 -04:00
										 |  |  |         } else { | 
					
						
							| 
									
										
										
										
											2023-03-07 10:49:41 -05:00
										 |  |  |             KScheduler::RescheduleCurrentHLEThread(m_kernel); | 
					
						
							| 
									
										
										
										
											2021-08-06 23:04:32 -07:00
										 |  |  |         } | 
					
						
							|  |  |  |     } else { | 
					
						
							| 
									
										
										
										
											2023-03-07 10:49:41 -05:00
										 |  |  |         GetCurrentThread(m_kernel).EnableDispatch(); | 
					
						
							| 
									
										
										
										
											2021-08-06 23:04:32 -07:00
										 |  |  |     } | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-01-08 11:35:03 -05:00
										 |  |  | } // namespace Kernel
 |