| 
									
										
										
										
											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
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-02-29 13:58:50 -04:00
										 |  |  | #include <array>
 | 
					
						
							| 
									
										
										
										
											2018-08-28 12:30:33 -04:00
										 |  |  | #include <atomic>
 | 
					
						
							| 
									
										
										
										
											2020-02-22 10:27:40 -04:00
										 |  |  | #include <bitset>
 | 
					
						
							| 
									
										
										
										
											2020-01-26 10:28:23 -04:00
										 |  |  | #include <functional>
 | 
					
						
							| 
									
										
										
										
											2018-08-28 12:30:33 -04:00
										 |  |  | #include <memory>
 | 
					
						
							| 
									
										
										
										
											2020-02-14 09:30:53 -04:00
										 |  |  | #include <thread>
 | 
					
						
							| 
									
										
										
										
											2020-12-15 00:41:48 -08:00
										 |  |  | #include <unordered_set>
 | 
					
						
							| 
									
										
										
										
											2018-08-28 12:30:33 -04:00
										 |  |  | #include <utility>
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #include "common/assert.h"
 | 
					
						
							|  |  |  | #include "common/logging/log.h"
 | 
					
						
							| 
									
										
										
										
											2020-03-12 16:48:43 -04:00
										 |  |  | #include "common/microprofile.h"
 | 
					
						
							| 
									
										
										
										
											2021-11-25 20:46:17 -08:00
										 |  |  | #include "common/scope_exit.h"
 | 
					
						
							| 
									
										
										
										
											2020-02-24 22:04:12 -04:00
										 |  |  | #include "common/thread.h"
 | 
					
						
							| 
									
										
										
										
											2020-12-29 16:39:04 -08:00
										 |  |  | #include "common/thread_worker.h"
 | 
					
						
							| 
									
										
										
										
											2020-01-25 18:55:32 -04:00
										 |  |  | #include "core/arm/arm_interface.h"
 | 
					
						
							|  |  |  | #include "core/arm/exclusive_monitor.h"
 | 
					
						
							| 
									
										
										
										
											2018-08-28 12:30:33 -04:00
										 |  |  | #include "core/core.h"
 | 
					
						
							|  |  |  | #include "core/core_timing.h"
 | 
					
						
							| 
									
										
										
										
											2020-02-24 22:04:12 -04:00
										 |  |  | #include "core/cpu_manager.h"
 | 
					
						
							| 
									
										
										
										
											2020-02-22 10:27:40 -04:00
										 |  |  | #include "core/hardware_properties.h"
 | 
					
						
							| 
									
										
										
										
											2021-04-03 19:11:46 -07:00
										 |  |  | #include "core/hle/kernel/init/init_slab_setup.h"
 | 
					
						
							| 
									
										
										
										
											2021-04-21 21:43:25 -07:00
										 |  |  | #include "core/hle/kernel/k_client_port.h"
 | 
					
						
							| 
									
										
										
										
											2022-09-16 23:33:47 -07:00
										 |  |  | #include "core/hle/kernel/k_dynamic_resource_manager.h"
 | 
					
						
							| 
									
										
										
										
											2021-04-24 02:40:31 -07:00
										 |  |  | #include "core/hle/kernel/k_handle_table.h"
 | 
					
						
							| 
									
										
										
										
											2022-12-18 16:50:02 -05:00
										 |  |  | #include "core/hle/kernel/k_hardware_timer.h"
 | 
					
						
							| 
									
										
										
										
											2021-02-12 16:02:35 -08:00
										 |  |  | #include "core/hle/kernel/k_memory_layout.h"
 | 
					
						
							| 
									
										
										
										
											2021-02-12 17:38:40 -08:00
										 |  |  | #include "core/hle/kernel/k_memory_manager.h"
 | 
					
						
							| 
									
										
										
										
											2023-02-16 23:16:08 -05:00
										 |  |  | #include "core/hle/kernel/k_object_name.h"
 | 
					
						
							| 
									
										
										
										
											2022-10-29 14:45:09 -07:00
										 |  |  | #include "core/hle/kernel/k_page_buffer.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"
 | 
					
						
							| 
									
										
										
										
											2023-02-18 16:26:48 -05:00
										 |  |  | #include "core/hle/kernel/k_scoped_resource_reservation.h"
 | 
					
						
							| 
									
										
										
										
											2021-02-05 23:14:31 -08:00
										 |  |  | #include "core/hle/kernel/k_shared_memory.h"
 | 
					
						
							| 
									
										
										
										
											2022-10-29 14:45:09 -07:00
										 |  |  | #include "core/hle/kernel/k_system_resource.h"
 | 
					
						
							| 
									
										
										
										
											2020-12-30 23:01:08 -08:00
										 |  |  | #include "core/hle/kernel/k_thread.h"
 | 
					
						
							| 
									
										
										
										
											2022-01-14 16:25:37 -08:00
										 |  |  | #include "core/hle/kernel/k_worker_task_manager.h"
 | 
					
						
							| 
									
										
										
										
											2016-09-20 23:52:38 -07:00
										 |  |  | #include "core/hle/kernel/kernel.h"
 | 
					
						
							| 
									
										
										
										
											2020-01-25 18:55:32 -04:00
										 |  |  | #include "core/hle/kernel/physical_core.h"
 | 
					
						
							| 
									
										
										
										
											2018-08-28 12:30:33 -04:00
										 |  |  | #include "core/hle/result.h"
 | 
					
						
							| 
									
										
										
										
											2023-02-18 16:26:48 -05:00
										 |  |  | #include "core/hle/service/server_manager.h"
 | 
					
						
							| 
									
										
										
										
											2021-05-10 15:57:59 -07:00
										 |  |  | #include "core/hle/service/sm/sm.h"
 | 
					
						
							| 
									
										
										
										
											2019-04-07 01:10:44 -04:00
										 |  |  | #include "core/memory.h"
 | 
					
						
							| 
									
										
										
										
											2014-05-09 22:11:18 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-03-12 16:48:43 -04:00
										 |  |  | MICROPROFILE_DEFINE(Kernel_SVC, "Kernel", "SVC", MP_RGB(70, 200, 70)); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-05-20 18:13:25 -04:00
										 |  |  | namespace Kernel { | 
					
						
							| 
									
										
										
										
											2014-05-09 22:11:18 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-08-28 12:30:33 -04:00
										 |  |  | struct KernelCore::Impl { | 
					
						
							| 
									
										
										
										
											2022-10-29 14:45:09 -07:00
										 |  |  |     static constexpr size_t ApplicationMemoryBlockSlabHeapSize = 20000; | 
					
						
							|  |  |  |     static constexpr size_t SystemMemoryBlockSlabHeapSize = 10000; | 
					
						
							|  |  |  |     static constexpr size_t BlockInfoSlabHeapSize = 4000; | 
					
						
							|  |  |  |     static constexpr size_t ReservedDynamicPageCount = 64; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-02-18 16:26:48 -05:00
										 |  |  |     explicit Impl(Core::System& system_, KernelCore& kernel_) : system{system_} {} | 
					
						
							| 
									
										
										
										
											2019-03-05 12:28:10 -05:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-05-08 12:11:36 -04:00
										 |  |  |     void SetMulticore(bool is_multi) { | 
					
						
							|  |  |  |         is_multicore = is_multi; | 
					
						
							| 
									
										
										
										
											2020-03-08 22:39:41 -04:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-03-05 12:28:10 -05:00
										 |  |  |     void Initialize(KernelCore& kernel) { | 
					
						
							| 
									
										
										
										
											2022-12-18 16:50:02 -05:00
										 |  |  |         hardware_timer = std::make_unique<Kernel::KHardwareTimer>(kernel); | 
					
						
							|  |  |  |         hardware_timer->Initialize(); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-03-11 17:24:37 -08:00
										 |  |  |         global_object_list_container = std::make_unique<KAutoObjectWithListContainer>(kernel); | 
					
						
							| 
									
										
										
										
											2021-01-21 13:00:16 -08:00
										 |  |  |         global_scheduler_context = std::make_unique<Kernel::GlobalSchedulerContext>(kernel); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-12-31 02:13:02 -08:00
										 |  |  |         is_phantom_mode_for_singlecore = false; | 
					
						
							| 
									
										
										
										
											2020-12-02 18:08:35 -08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-04-12 16:14:19 -04:00
										 |  |  |         // Derive the initial memory layout from the emulated board
 | 
					
						
							| 
									
										
										
										
											2021-05-04 21:35:42 -07:00
										 |  |  |         Init::InitializeSlabResourceCounts(kernel); | 
					
						
							| 
									
										
										
										
											2022-02-26 10:46:31 -08:00
										 |  |  |         DeriveInitialMemoryLayout(); | 
					
						
							| 
									
										
										
										
											2022-02-27 11:05:07 -08:00
										 |  |  |         Init::InitializeSlabHeaps(system, *memory_layout); | 
					
						
							| 
									
										
										
										
											2021-04-03 19:11:46 -07:00
										 |  |  | 
 | 
					
						
							|  |  |  |         // Initialize kernel memory and resources.
 | 
					
						
							| 
									
										
										
										
											2022-02-26 10:46:31 -08:00
										 |  |  |         InitializeSystemResourceLimit(kernel, system.CoreTiming()); | 
					
						
							|  |  |  |         InitializeMemoryLayout(); | 
					
						
							| 
									
										
										
										
											2022-06-13 18:36:30 -04:00
										 |  |  |         InitializeShutdownThreads(); | 
					
						
							| 
									
										
										
										
											2022-07-05 23:27:25 -04:00
										 |  |  |         InitializePhysicalCores(); | 
					
						
							| 
									
										
										
										
											2022-09-10 23:45:07 -07:00
										 |  |  |         InitializePreemption(kernel); | 
					
						
							| 
									
										
										
										
											2023-02-16 23:16:08 -05:00
										 |  |  |         InitializeGlobalData(kernel); | 
					
						
							| 
									
										
										
										
											2021-04-02 18:02:10 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-09-16 23:33:47 -07:00
										 |  |  |         // Initialize the Dynamic Slab Heaps.
 | 
					
						
							|  |  |  |         { | 
					
						
							|  |  |  |             const auto& pt_heap_region = memory_layout->GetPageTableHeapRegion(); | 
					
						
							|  |  |  |             ASSERT(pt_heap_region.GetEndAddress() != 0); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-10-29 14:45:09 -07:00
										 |  |  |             InitializeResourceManagers(kernel, pt_heap_region.GetAddress(), | 
					
						
							|  |  |  |                                        pt_heap_region.GetSize()); | 
					
						
							| 
									
										
										
										
											2022-09-16 23:33:47 -07:00
										 |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-04-29 21:16:09 -04:00
										 |  |  |         InitializeHackSharedMemory(kernel); | 
					
						
							| 
									
										
										
										
											2022-11-02 20:21:32 -04:00
										 |  |  |         RegisterHostThread(nullptr); | 
					
						
							| 
									
										
										
										
											2018-08-28 12:30:33 -04:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-01-02 18:29:03 -05:00
										 |  |  |     void TerminateAllProcesses() { | 
					
						
							|  |  |  |         std::scoped_lock lk{process_list_lock}; | 
					
						
							|  |  |  |         for (auto& process : process_list) { | 
					
						
							|  |  |  |             process->Terminate(); | 
					
						
							|  |  |  |             process->Close(); | 
					
						
							|  |  |  |             process = nullptr; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         process_list.clear(); | 
					
						
							| 
									
										
										
										
											2022-10-02 14:26:30 -07:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-08-28 12:30:33 -04:00
										 |  |  |     void Shutdown() { | 
					
						
							| 
									
										
										
										
											2021-11-25 20:46:17 -08:00
										 |  |  |         is_shutting_down.store(true, std::memory_order_relaxed); | 
					
						
							|  |  |  |         SCOPE_EXIT({ is_shutting_down.store(false, std::memory_order_relaxed); }); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-07-16 23:48:45 +01:00
										 |  |  |         CloseServices(); | 
					
						
							| 
									
										
										
										
											2020-12-29 16:39:04 -08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-01-02 18:29:03 -05:00
										 |  |  |         if (application_process) { | 
					
						
							|  |  |  |             application_process->Close(); | 
					
						
							|  |  |  |             application_process = nullptr; | 
					
						
							| 
									
										
										
										
											2024-01-02 17:12:16 -05:00
										 |  |  |         } | 
					
						
							| 
									
										
										
										
											2023-10-21 16:47:43 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-08-28 12:30:33 -04:00
										 |  |  |         next_object_id = 0; | 
					
						
							| 
									
										
										
										
											2023-10-21 16:47:43 -04:00
										 |  |  |         next_kernel_process_id = KProcess::InitialProcessIdMin; | 
					
						
							|  |  |  |         next_user_process_id = KProcess::ProcessIdMin; | 
					
						
							| 
									
										
										
										
											2018-08-28 12:30:33 -04:00
										 |  |  |         next_thread_id = 1; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-12-02 18:08:35 -08:00
										 |  |  |         preemption_event = nullptr; | 
					
						
							| 
									
										
										
										
											2019-10-12 08:21:51 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-04-23 17:00:46 -07:00
										 |  |  |         // Cleanup persistent kernel objects
 | 
					
						
							|  |  |  |         auto CleanupObject = [](KAutoObject* obj) { | 
					
						
							|  |  |  |             if (obj) { | 
					
						
							|  |  |  |                 obj->Close(); | 
					
						
							|  |  |  |                 obj = nullptr; | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |         }; | 
					
						
							|  |  |  |         CleanupObject(font_shared_mem); | 
					
						
							|  |  |  |         CleanupObject(irs_shared_mem); | 
					
						
							|  |  |  |         CleanupObject(time_shared_mem); | 
					
						
							| 
									
										
										
										
											2021-05-02 18:41:03 -05:00
										 |  |  |         CleanupObject(hidbus_shared_mem); | 
					
						
							| 
									
										
										
										
											2021-04-23 17:00:46 -07:00
										 |  |  |         CleanupObject(system_resource_limit); | 
					
						
							| 
									
										
										
										
											2021-02-20 20:51:11 -05:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-08-06 22:58:46 -07:00
										 |  |  |         for (u32 core_id = 0; core_id < Core::Hardware::NUM_CPU_CORES; core_id++) { | 
					
						
							| 
									
										
										
										
											2022-06-13 18:36:30 -04:00
										 |  |  |             if (shutdown_threads[core_id]) { | 
					
						
							|  |  |  |                 shutdown_threads[core_id]->Close(); | 
					
						
							|  |  |  |                 shutdown_threads[core_id] = nullptr; | 
					
						
							| 
									
										
										
										
											2021-08-06 22:58:46 -07:00
										 |  |  |             } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             schedulers[core_id].reset(); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-12-29 15:55:30 -08:00
										 |  |  |         // Next host thead ID to use, 0-3 IDs represent core threads, >3 represent others
 | 
					
						
							|  |  |  |         next_host_thread_id = Core::Hardware::NUM_CPU_CORES; | 
					
						
							| 
									
										
										
										
											2021-06-28 14:38:14 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-10-25 18:55:20 +08:00
										 |  |  |         // Close kernel objects that were not freed on shutdown
 | 
					
						
							|  |  |  |         { | 
					
						
							| 
									
										
										
										
											2022-04-07 19:31:32 +01:00
										 |  |  |             std::scoped_lock lk{registered_in_use_objects_lock}; | 
					
						
							| 
									
										
										
										
											2021-10-25 18:55:20 +08:00
										 |  |  |             if (registered_in_use_objects.size()) { | 
					
						
							| 
									
										
										
										
											2021-10-26 12:43:27 +08:00
										 |  |  |                 for (auto& object : registered_in_use_objects) { | 
					
						
							|  |  |  |                     object->Close(); | 
					
						
							| 
									
										
										
										
											2021-10-25 18:55:20 +08:00
										 |  |  |                 } | 
					
						
							|  |  |  |                 registered_in_use_objects.clear(); | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-06-28 14:38:14 -07:00
										 |  |  |         // Track kernel objects that were not freed on shutdown
 | 
					
						
							| 
									
										
										
										
											2021-07-02 15:19:04 -07:00
										 |  |  |         { | 
					
						
							| 
									
										
										
										
											2022-04-07 19:31:32 +01:00
										 |  |  |             std::scoped_lock lk{registered_objects_lock}; | 
					
						
							| 
									
										
										
										
											2021-07-02 15:19:04 -07:00
										 |  |  |             if (registered_objects.size()) { | 
					
						
							| 
									
										
										
										
											2022-03-11 17:25:43 -08:00
										 |  |  |                 LOG_DEBUG(Kernel, "{} kernel objects were dangling on shutdown!", | 
					
						
							|  |  |  |                           registered_objects.size()); | 
					
						
							| 
									
										
										
										
											2021-07-02 15:19:04 -07:00
										 |  |  |                 registered_objects.clear(); | 
					
						
							|  |  |  |             } | 
					
						
							| 
									
										
										
										
											2021-06-28 14:38:14 -07:00
										 |  |  |         } | 
					
						
							| 
									
										
										
										
											2022-03-11 17:24:37 -08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-02-16 23:16:08 -05:00
										 |  |  |         object_name_global_data.reset(); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-03-11 17:24:37 -08:00
										 |  |  |         // Ensure that the object list container is finalized and properly shutdown.
 | 
					
						
							|  |  |  |         global_object_list_container->Finalize(); | 
					
						
							|  |  |  |         global_object_list_container.reset(); | 
					
						
							| 
									
										
										
										
											2022-12-18 16:50:02 -05:00
										 |  |  | 
 | 
					
						
							|  |  |  |         hardware_timer->Finalize(); | 
					
						
							|  |  |  |         hardware_timer.reset(); | 
					
						
							| 
									
										
										
										
											2020-01-25 18:55:32 -04:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-07-16 23:48:45 +01:00
										 |  |  |     void CloseServices() { | 
					
						
							| 
									
										
										
										
											2023-02-18 16:26:48 -05:00
										 |  |  |         // Ensures all servers gracefully shutdown.
 | 
					
						
							|  |  |  |         std::scoped_lock lk{server_lock}; | 
					
						
							|  |  |  |         server_managers.clear(); | 
					
						
							| 
									
										
										
										
											2022-07-16 23:48:45 +01:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-01-30 18:19:21 -05:00
										 |  |  |     void InitializePhysicalCores() { | 
					
						
							| 
									
										
										
										
											2021-01-20 18:10:07 -08:00
										 |  |  |         for (u32 i = 0; i < Core::Hardware::NUM_CPU_CORES; i++) { | 
					
						
							| 
									
										
										
										
											2022-07-05 23:27:25 -04:00
										 |  |  |             const s32 core{static_cast<s32>(i)}; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-06-26 18:52:16 -04:00
										 |  |  |             schedulers[i] = std::make_unique<Kernel::KScheduler>(system.Kernel()); | 
					
						
							| 
									
										
										
										
											2023-11-28 14:30:39 -05:00
										 |  |  |             cores[i] = std::make_unique<Kernel::PhysicalCore>(system.Kernel(), i); | 
					
						
							| 
									
										
										
										
											2018-08-28 12:30:33 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-07-05 23:27:25 -04:00
										 |  |  |             auto* main_thread{Kernel::KThread::Create(system.Kernel())}; | 
					
						
							|  |  |  |             main_thread->SetCurrentCore(core); | 
					
						
							|  |  |  |             ASSERT(Kernel::KThread::InitializeMainThread(system, main_thread, core).IsSuccess()); | 
					
						
							| 
									
										
										
										
											2023-04-29 21:16:09 -04:00
										 |  |  |             KThread::Register(system.Kernel(), main_thread); | 
					
						
							| 
									
										
										
										
											2022-07-05 23:27:25 -04:00
										 |  |  | 
 | 
					
						
							|  |  |  |             auto* idle_thread{Kernel::KThread::Create(system.Kernel())}; | 
					
						
							|  |  |  |             idle_thread->SetCurrentCore(core); | 
					
						
							|  |  |  |             ASSERT(Kernel::KThread::InitializeIdleThread(system, idle_thread, core).IsSuccess()); | 
					
						
							| 
									
										
										
										
											2023-04-29 21:16:09 -04:00
										 |  |  |             KThread::Register(system.Kernel(), idle_thread); | 
					
						
							| 
									
										
										
										
											2022-07-05 23:27:25 -04:00
										 |  |  | 
 | 
					
						
							|  |  |  |             schedulers[i]->Initialize(main_thread, idle_thread, core); | 
					
						
							| 
									
										
										
										
											2020-02-24 22:04:12 -04:00
										 |  |  |         } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-11-19 12:54:06 -05:00
										 |  |  |     // Creates the default system resource limit
 | 
					
						
							| 
									
										
										
										
											2021-04-14 00:40:33 -04:00
										 |  |  |     void InitializeSystemResourceLimit(KernelCore& kernel, | 
					
						
							| 
									
										
										
										
											2022-02-26 10:46:31 -08:00
										 |  |  |                                        const Core::Timing::CoreTiming& core_timing) { | 
					
						
							| 
									
										
										
										
											2021-04-20 21:28:11 -07:00
										 |  |  |         system_resource_limit = KResourceLimit::Create(system.Kernel()); | 
					
						
							| 
									
										
										
										
											2023-08-14 18:12:06 -04:00
										 |  |  |         system_resource_limit->Initialize(); | 
					
						
							| 
									
										
										
										
											2023-04-29 21:16:09 -04:00
										 |  |  |         KResourceLimit::Register(kernel, system_resource_limit); | 
					
						
							| 
									
										
										
										
											2021-04-20 21:28:11 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-06-12 19:31:22 -04:00
										 |  |  |         const auto sizes{memory_layout->GetTotalAndKernelMemorySizes()}; | 
					
						
							|  |  |  |         const auto total_size{sizes.first}; | 
					
						
							|  |  |  |         const auto kernel_size{sizes.second}; | 
					
						
							| 
									
										
										
										
											2018-11-19 12:54:06 -05:00
										 |  |  | 
 | 
					
						
							|  |  |  |         // If setting the default system values fails, then something seriously wrong has occurred.
 | 
					
						
							| 
									
										
										
										
											2022-11-03 10:22:05 -04:00
										 |  |  |         ASSERT( | 
					
						
							|  |  |  |             system_resource_limit->SetLimitValue(LimitableResource::PhysicalMemoryMax, total_size) | 
					
						
							|  |  |  |                 .IsSuccess()); | 
					
						
							|  |  |  |         ASSERT(system_resource_limit->SetLimitValue(LimitableResource::ThreadCountMax, 800) | 
					
						
							| 
									
										
										
										
											2021-01-30 20:40:49 +11:00
										 |  |  |                    .IsSuccess()); | 
					
						
							| 
									
										
										
										
											2022-11-03 10:22:05 -04:00
										 |  |  |         ASSERT(system_resource_limit->SetLimitValue(LimitableResource::EventCountMax, 900) | 
					
						
							| 
									
										
										
										
											2018-11-19 12:54:06 -05:00
										 |  |  |                    .IsSuccess()); | 
					
						
							| 
									
										
										
										
											2022-11-03 10:22:05 -04:00
										 |  |  |         ASSERT(system_resource_limit->SetLimitValue(LimitableResource::TransferMemoryCountMax, 200) | 
					
						
							|  |  |  |                    .IsSuccess()); | 
					
						
							|  |  |  |         ASSERT(system_resource_limit->SetLimitValue(LimitableResource::SessionCountMax, 1133) | 
					
						
							|  |  |  |                    .IsSuccess()); | 
					
						
							|  |  |  |         system_resource_limit->Reserve(LimitableResource::PhysicalMemoryMax, kernel_size); | 
					
						
							| 
									
										
										
										
											2020-04-08 21:07:30 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-02-12 19:05:24 -05:00
										 |  |  |         // Reserve secure applet memory, introduced in firmware 5.0.0
 | 
					
						
							| 
									
										
										
										
											2021-06-23 14:18:27 -07:00
										 |  |  |         constexpr u64 secure_applet_memory_size{4_MiB}; | 
					
						
							| 
									
										
										
										
											2022-11-03 10:22:05 -04:00
										 |  |  |         ASSERT(system_resource_limit->Reserve(LimitableResource::PhysicalMemoryMax, | 
					
						
							| 
									
										
										
										
											2021-02-12 19:05:24 -05:00
										 |  |  |                                               secure_applet_memory_size)); | 
					
						
							| 
									
										
										
										
											2018-08-28 12:30:33 -04:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-02-24 22:04:12 -04:00
										 |  |  |     void InitializePreemption(KernelCore& kernel) { | 
					
						
							|  |  |  |         preemption_event = Core::Timing::CreateEvent( | 
					
						
							| 
									
										
										
										
											2022-07-10 06:59:40 +01:00
										 |  |  |             "PreemptionCallback", | 
					
						
							| 
									
										
										
										
											2023-12-23 13:58:09 -05:00
										 |  |  |             [this, &kernel](s64 time, | 
					
						
							| 
									
										
										
										
											2022-07-10 06:59:40 +01:00
										 |  |  |                             std::chrono::nanoseconds) -> std::optional<std::chrono::nanoseconds> { | 
					
						
							| 
									
										
										
										
											2020-02-24 22:04:12 -04:00
										 |  |  |                 { | 
					
						
							| 
									
										
										
										
											2020-12-03 22:26:42 -08:00
										 |  |  |                     KScopedSchedulerLock lock(kernel); | 
					
						
							| 
									
										
										
										
											2020-12-02 18:08:35 -08:00
										 |  |  |                     global_scheduler_context->PreemptThreads(); | 
					
						
							| 
									
										
										
										
											2020-02-24 22:04:12 -04:00
										 |  |  |                 } | 
					
						
							| 
									
										
										
										
											2022-07-10 06:59:40 +01:00
										 |  |  |                 return std::nullopt; | 
					
						
							| 
									
										
										
										
											2019-09-10 11:04:40 -04:00
										 |  |  |             }); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-11-26 18:16:54 +01:00
										 |  |  |         const auto time_interval = std::chrono::nanoseconds{std::chrono::milliseconds(10)}; | 
					
						
							| 
									
										
										
										
											2022-07-10 08:29:37 +01:00
										 |  |  |         system.CoreTiming().ScheduleLoopingEvent(time_interval, time_interval, preemption_event); | 
					
						
							| 
									
										
										
										
											2019-09-10 11:04:40 -04:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-03-17 21:26:04 -04:00
										 |  |  |     void InitializeResourceManagers(KernelCore& kernel, KVirtualAddress address, size_t size) { | 
					
						
							| 
									
										
										
										
											2022-10-29 14:45:09 -07:00
										 |  |  |         // Ensure that the buffer is suitable for our use.
 | 
					
						
							| 
									
										
										
										
											2023-03-17 21:26:04 -04:00
										 |  |  |         ASSERT(Common::IsAligned(GetInteger(address), PageSize)); | 
					
						
							| 
									
										
										
										
											2022-10-29 14:45:09 -07:00
										 |  |  |         ASSERT(Common::IsAligned(size, PageSize)); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         // Ensure that we have space for our reference counts.
 | 
					
						
							|  |  |  |         const size_t rc_size = | 
					
						
							|  |  |  |             Common::AlignUp(KPageTableSlabHeap::CalculateReferenceCountSize(size), PageSize); | 
					
						
							|  |  |  |         ASSERT(rc_size < size); | 
					
						
							|  |  |  |         size -= rc_size; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         // Initialize the resource managers' shared page manager.
 | 
					
						
							|  |  |  |         resource_manager_page_manager = std::make_unique<KDynamicPageManager>(); | 
					
						
							|  |  |  |         resource_manager_page_manager->Initialize( | 
					
						
							|  |  |  |             address, size, std::max<size_t>(PageSize, KPageBufferSlabHeap::BufferSize)); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         // Initialize the KPageBuffer slab heap.
 | 
					
						
							|  |  |  |         page_buffer_slab_heap.Initialize(system); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         // Initialize the fixed-size slabheaps.
 | 
					
						
							|  |  |  |         app_memory_block_heap = std::make_unique<KMemoryBlockSlabHeap>(); | 
					
						
							|  |  |  |         sys_memory_block_heap = std::make_unique<KMemoryBlockSlabHeap>(); | 
					
						
							|  |  |  |         block_info_heap = std::make_unique<KBlockInfoSlabHeap>(); | 
					
						
							|  |  |  |         app_memory_block_heap->Initialize(resource_manager_page_manager.get(), | 
					
						
							|  |  |  |                                           ApplicationMemoryBlockSlabHeapSize); | 
					
						
							|  |  |  |         sys_memory_block_heap->Initialize(resource_manager_page_manager.get(), | 
					
						
							|  |  |  |                                           SystemMemoryBlockSlabHeapSize); | 
					
						
							|  |  |  |         block_info_heap->Initialize(resource_manager_page_manager.get(), BlockInfoSlabHeapSize); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         // Reserve all but a fixed number of remaining pages for the page table heap.
 | 
					
						
							|  |  |  |         const size_t num_pt_pages = resource_manager_page_manager->GetCount() - | 
					
						
							|  |  |  |                                     resource_manager_page_manager->GetUsed() - | 
					
						
							|  |  |  |                                     ReservedDynamicPageCount; | 
					
						
							|  |  |  |         page_table_heap = std::make_unique<KPageTableSlabHeap>(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         // TODO(bunnei): Pass in address once we support kernel virtual memory allocations.
 | 
					
						
							|  |  |  |         page_table_heap->Initialize( | 
					
						
							|  |  |  |             resource_manager_page_manager.get(), num_pt_pages, | 
					
						
							|  |  |  |             /*GetPointer<KPageTableManager::RefCount>(address + size)*/ nullptr); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         // Setup the slab managers.
 | 
					
						
							|  |  |  |         KDynamicPageManager* const app_dynamic_page_manager = nullptr; | 
					
						
							|  |  |  |         KDynamicPageManager* const sys_dynamic_page_manager = | 
					
						
							|  |  |  |             /*KTargetSystem::IsDynamicResourceLimitsEnabled()*/ true | 
					
						
							|  |  |  |                 ? resource_manager_page_manager.get() | 
					
						
							|  |  |  |                 : nullptr; | 
					
						
							| 
									
										
										
										
											2022-09-16 23:33:47 -07:00
										 |  |  |         app_memory_block_manager = std::make_unique<KMemoryBlockSlabManager>(); | 
					
						
							| 
									
										
										
										
											2022-10-29 14:45:09 -07:00
										 |  |  |         sys_memory_block_manager = std::make_unique<KMemoryBlockSlabManager>(); | 
					
						
							|  |  |  |         app_block_info_manager = std::make_unique<KBlockInfoManager>(); | 
					
						
							|  |  |  |         sys_block_info_manager = std::make_unique<KBlockInfoManager>(); | 
					
						
							|  |  |  |         app_page_table_manager = std::make_unique<KPageTableManager>(); | 
					
						
							|  |  |  |         sys_page_table_manager = std::make_unique<KPageTableManager>(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         app_memory_block_manager->Initialize(app_dynamic_page_manager, app_memory_block_heap.get()); | 
					
						
							|  |  |  |         sys_memory_block_manager->Initialize(sys_dynamic_page_manager, sys_memory_block_heap.get()); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         app_block_info_manager->Initialize(app_dynamic_page_manager, block_info_heap.get()); | 
					
						
							|  |  |  |         sys_block_info_manager->Initialize(sys_dynamic_page_manager, block_info_heap.get()); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         app_page_table_manager->Initialize(app_dynamic_page_manager, page_table_heap.get()); | 
					
						
							|  |  |  |         sys_page_table_manager->Initialize(sys_dynamic_page_manager, page_table_heap.get()); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         // Check that we have the correct number of dynamic pages available.
 | 
					
						
							|  |  |  |         ASSERT(resource_manager_page_manager->GetCount() - | 
					
						
							|  |  |  |                    resource_manager_page_manager->GetUsed() == | 
					
						
							|  |  |  |                ReservedDynamicPageCount); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         // Create the system page table managers.
 | 
					
						
							|  |  |  |         app_system_resource = std::make_unique<KSystemResource>(kernel); | 
					
						
							|  |  |  |         sys_system_resource = std::make_unique<KSystemResource>(kernel); | 
					
						
							| 
									
										
										
										
											2023-10-21 16:47:43 -04:00
										 |  |  |         KAutoObject::Create(std::addressof(*app_system_resource)); | 
					
						
							|  |  |  |         KAutoObject::Create(std::addressof(*sys_system_resource)); | 
					
						
							| 
									
										
										
										
											2022-10-29 14:45:09 -07:00
										 |  |  | 
 | 
					
						
							|  |  |  |         // Set the managers for the system resources.
 | 
					
						
							|  |  |  |         app_system_resource->SetManagers(*app_memory_block_manager, *app_block_info_manager, | 
					
						
							|  |  |  |                                          *app_page_table_manager); | 
					
						
							|  |  |  |         sys_system_resource->SetManagers(*sys_memory_block_manager, *sys_block_info_manager, | 
					
						
							|  |  |  |                                          *sys_page_table_manager); | 
					
						
							| 
									
										
										
										
											2022-09-16 23:33:47 -07:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-06-13 18:36:30 -04:00
										 |  |  |     void InitializeShutdownThreads() { | 
					
						
							| 
									
										
										
										
											2021-05-01 23:24:51 -07:00
										 |  |  |         for (u32 core_id = 0; core_id < Core::Hardware::NUM_CPU_CORES; core_id++) { | 
					
						
							| 
									
										
										
										
											2022-06-13 18:36:30 -04:00
										 |  |  |             shutdown_threads[core_id] = KThread::Create(system.Kernel()); | 
					
						
							|  |  |  |             ASSERT(KThread::InitializeHighPriorityThread(system, shutdown_threads[core_id], {}, {}, | 
					
						
							| 
									
										
										
										
											2021-04-09 22:42:23 -07:00
										 |  |  |                                                          core_id) | 
					
						
							| 
									
										
										
										
											2021-04-03 19:11:46 -07:00
										 |  |  |                        .IsSuccess()); | 
					
						
							| 
									
										
										
										
											2023-04-29 21:16:09 -04:00
										 |  |  |             KThread::Register(system.Kernel(), shutdown_threads[core_id]); | 
					
						
							| 
									
										
										
										
											2020-02-24 22:04:12 -04:00
										 |  |  |         } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-02-16 23:16:08 -05:00
										 |  |  |     void InitializeGlobalData(KernelCore& kernel) { | 
					
						
							|  |  |  |         object_name_global_data = std::make_unique<KObjectNameGlobalData>(kernel); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-02-13 11:21:43 -05:00
										 |  |  |     void MakeApplicationProcess(KProcess* process) { | 
					
						
							|  |  |  |         application_process = process; | 
					
						
							| 
									
										
										
										
											2024-01-02 18:29:03 -05:00
										 |  |  |         application_process->Open(); | 
					
						
							| 
									
										
										
										
											2019-11-26 18:34:30 -05:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-02-10 20:43:06 -05:00
										 |  |  |     static inline thread_local u8 host_thread_id = UINT8_MAX; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     /// Sets the host thread ID for the caller.
 | 
					
						
							| 
									
										
										
										
											2023-10-12 09:16:22 -04:00
										 |  |  |     LTO_NOINLINE u32 SetHostThreadId(std::size_t core_id) { | 
					
						
							| 
									
										
										
										
											2023-02-10 20:43:06 -05:00
										 |  |  |         // This should only be called during core init.
 | 
					
						
							|  |  |  |         ASSERT(host_thread_id == UINT8_MAX); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         // The first four slots are reserved for CPU core threads
 | 
					
						
							|  |  |  |         ASSERT(core_id < Core::Hardware::NUM_CPU_CORES); | 
					
						
							|  |  |  |         host_thread_id = static_cast<u8>(core_id); | 
					
						
							| 
									
										
										
										
											2021-08-15 15:30:56 -05:00
										 |  |  |         return host_thread_id; | 
					
						
							| 
									
										
										
										
											2020-12-29 15:55:30 -08:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-02-10 20:43:06 -05:00
										 |  |  |     /// Gets the host thread ID for the caller
 | 
					
						
							| 
									
										
										
										
											2023-10-12 09:16:22 -04:00
										 |  |  |     LTO_NOINLINE u32 GetHostThreadId() const { | 
					
						
							| 
									
										
										
										
											2020-12-29 15:55:30 -08:00
										 |  |  |         return host_thread_id; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-01-21 13:00:16 -08:00
										 |  |  |     // Gets the dummy KThread for the caller, allocating a new one if this is the first time
 | 
					
						
							| 
									
										
										
										
											2023-10-12 09:16:22 -04:00
										 |  |  |     LTO_NOINLINE KThread* GetHostDummyThread(KThread* existing_thread) { | 
					
						
							|  |  |  |         const auto initialize{[](KThread* thread) LTO_NOINLINE { | 
					
						
							| 
									
										
										
										
											2022-11-02 20:21:32 -04:00
										 |  |  |             ASSERT(KThread::InitializeDummyThread(thread, nullptr).IsSuccess()); | 
					
						
							| 
									
										
										
										
											2022-01-18 17:59:54 -08:00
										 |  |  |             return thread; | 
					
						
							| 
									
										
										
										
											2023-03-10 17:04:50 -05:00
										 |  |  |         }}; | 
					
						
							| 
									
										
										
										
											2021-04-03 19:11:46 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-11-02 20:21:32 -04:00
										 |  |  |         thread_local KThread raw_thread{system.Kernel()}; | 
					
						
							| 
									
										
										
										
											2023-02-10 20:43:06 -05:00
										 |  |  |         thread_local KThread* thread = existing_thread ? existing_thread : initialize(&raw_thread); | 
					
						
							| 
									
										
										
										
											2022-03-12 03:06:57 -08:00
										 |  |  |         return thread; | 
					
						
							| 
									
										
										
										
											2021-01-21 13:00:16 -08:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-12-29 15:55:30 -08:00
										 |  |  |     /// Registers a CPU core thread by allocating a host thread ID for it
 | 
					
						
							| 
									
										
										
										
											2020-02-14 09:30:53 -04:00
										 |  |  |     void RegisterCoreThread(std::size_t core_id) { | 
					
						
							| 
									
										
										
										
											2020-12-29 15:55:30 -08:00
										 |  |  |         ASSERT(core_id < Core::Hardware::NUM_CPU_CORES); | 
					
						
							| 
									
										
										
										
											2023-02-10 20:43:06 -05:00
										 |  |  |         const auto this_id = SetHostThreadId(core_id); | 
					
						
							| 
									
										
										
										
											2020-03-08 22:39:41 -04:00
										 |  |  |         if (!is_multicore) { | 
					
						
							| 
									
										
										
										
											2020-10-13 18:00:25 -03:00
										 |  |  |             single_core_thread_id = this_id; | 
					
						
							| 
									
										
										
										
											2020-03-08 22:39:41 -04:00
										 |  |  |         } | 
					
						
							| 
									
										
										
										
											2020-02-14 09:30:53 -04:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-12-29 15:55:30 -08:00
										 |  |  |     /// Registers a new host thread by allocating a host thread ID for it
 | 
					
						
							| 
									
										
										
										
											2022-11-02 20:21:32 -04:00
										 |  |  |     void RegisterHostThread(KThread* existing_thread) { | 
					
						
							|  |  |  |         [[maybe_unused]] const auto dummy_thread = GetHostDummyThread(existing_thread); | 
					
						
							| 
									
										
										
										
											2020-10-13 18:00:25 -03:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-12-29 15:55:30 -08:00
										 |  |  |     [[nodiscard]] u32 GetCurrentHostThreadID() { | 
					
						
							|  |  |  |         const auto this_id = GetHostThreadId(); | 
					
						
							| 
									
										
										
										
											2020-10-13 18:00:25 -03:00
										 |  |  |         if (!is_multicore && single_core_thread_id == this_id) { | 
					
						
							|  |  |  |             return static_cast<u32>(system.GetCpuManager().CurrentCore()); | 
					
						
							| 
									
										
										
										
											2020-03-08 22:39:41 -04:00
										 |  |  |         } | 
					
						
							| 
									
										
										
										
											2020-12-29 15:55:30 -08:00
										 |  |  |         return this_id; | 
					
						
							| 
									
										
										
										
											2020-02-14 09:30:53 -04:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-10-30 20:02:33 -04:00
										 |  |  |     static inline thread_local bool is_phantom_mode_for_singlecore{false}; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-10-12 09:16:22 -04:00
										 |  |  |     LTO_NOINLINE bool IsPhantomModeForSingleCore() const { | 
					
						
							| 
									
										
										
										
											2020-12-31 02:13:02 -08:00
										 |  |  |         return is_phantom_mode_for_singlecore; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-10-12 09:16:22 -04:00
										 |  |  |     LTO_NOINLINE void SetIsPhantomModeForSingleCore(bool value) { | 
					
						
							| 
									
										
										
										
											2020-12-31 02:13:02 -08:00
										 |  |  |         ASSERT(!is_multicore); | 
					
						
							|  |  |  |         is_phantom_mode_for_singlecore = value; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-11-25 20:46:17 -08:00
										 |  |  |     bool IsShuttingDown() const { | 
					
						
							|  |  |  |         return is_shutting_down.load(std::memory_order_relaxed); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-06-16 10:35:52 -04:00
										 |  |  |     static inline thread_local KThread* current_thread{nullptr}; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-10-12 09:16:22 -04:00
										 |  |  |     LTO_NOINLINE KThread* GetCurrentEmuThread() { | 
					
						
							| 
									
										
										
										
											2023-02-10 20:43:06 -05:00
										 |  |  |         if (!current_thread) { | 
					
						
							|  |  |  |             current_thread = GetHostDummyThread(nullptr); | 
					
						
							| 
									
										
										
										
											2020-02-14 09:30:53 -04:00
										 |  |  |         } | 
					
						
							| 
									
										
										
										
											2022-06-16 10:35:52 -04:00
										 |  |  |         return current_thread; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-10-12 09:16:22 -04:00
										 |  |  |     LTO_NOINLINE void SetCurrentEmuThread(KThread* thread) { | 
					
						
							| 
									
										
										
										
											2022-06-16 10:35:52 -04:00
										 |  |  |         current_thread = thread; | 
					
						
							| 
									
										
										
										
											2020-02-14 09:30:53 -04:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-02-26 10:46:31 -08:00
										 |  |  |     void DeriveInitialMemoryLayout() { | 
					
						
							| 
									
										
										
										
											2022-02-27 11:05:07 -08:00
										 |  |  |         memory_layout = std::make_unique<KMemoryLayout>(); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-03-20 00:53:00 -07:00
										 |  |  |         // Insert the root region for the virtual memory tree, from which all other regions will
 | 
					
						
							|  |  |  |         // derive.
 | 
					
						
							| 
									
										
										
										
											2022-02-27 11:05:07 -08:00
										 |  |  |         memory_layout->GetVirtualMemoryRegionTree().InsertDirectly( | 
					
						
							| 
									
										
										
										
											2021-03-20 00:53:00 -07:00
										 |  |  |             KernelVirtualAddressSpaceBase, | 
					
						
							|  |  |  |             KernelVirtualAddressSpaceBase + KernelVirtualAddressSpaceSize - 1); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         // Insert the root region for the physical memory tree, from which all other regions will
 | 
					
						
							|  |  |  |         // derive.
 | 
					
						
							| 
									
										
										
										
											2022-02-27 11:05:07 -08:00
										 |  |  |         memory_layout->GetPhysicalMemoryRegionTree().InsertDirectly( | 
					
						
							| 
									
										
										
										
											2021-03-20 00:53:00 -07:00
										 |  |  |             KernelPhysicalAddressSpaceBase, | 
					
						
							|  |  |  |             KernelPhysicalAddressSpaceBase + KernelPhysicalAddressSpaceSize - 1); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         // Save start and end for ease of use.
 | 
					
						
							| 
									
										
										
										
											2023-03-17 21:26:04 -04:00
										 |  |  |         constexpr KVirtualAddress code_start_virt_addr = KernelVirtualAddressCodeBase; | 
					
						
							|  |  |  |         constexpr KVirtualAddress code_end_virt_addr = KernelVirtualAddressCodeEnd; | 
					
						
							| 
									
										
										
										
											2021-03-20 00:53:00 -07:00
										 |  |  | 
 | 
					
						
							|  |  |  |         // Setup the containing kernel region.
 | 
					
						
							| 
									
										
										
										
											2021-06-23 14:18:27 -07:00
										 |  |  |         constexpr size_t KernelRegionSize = 1_GiB; | 
					
						
							|  |  |  |         constexpr size_t KernelRegionAlign = 1_GiB; | 
					
						
							| 
									
										
										
										
											2023-03-17 21:26:04 -04:00
										 |  |  |         constexpr KVirtualAddress kernel_region_start = | 
					
						
							|  |  |  |             Common::AlignDown(GetInteger(code_start_virt_addr), KernelRegionAlign); | 
					
						
							| 
									
										
										
										
											2021-03-20 00:53:00 -07:00
										 |  |  |         size_t kernel_region_size = KernelRegionSize; | 
					
						
							|  |  |  |         if (!(kernel_region_start + KernelRegionSize - 1 <= KernelVirtualAddressSpaceLast)) { | 
					
						
							| 
									
										
										
										
											2023-03-17 21:26:04 -04:00
										 |  |  |             kernel_region_size = KernelVirtualAddressSpaceEnd - GetInteger(kernel_region_start); | 
					
						
							| 
									
										
										
										
											2021-03-20 00:53:00 -07:00
										 |  |  |         } | 
					
						
							| 
									
										
										
										
											2022-02-27 11:05:07 -08:00
										 |  |  |         ASSERT(memory_layout->GetVirtualMemoryRegionTree().Insert( | 
					
						
							| 
									
										
										
										
											2023-03-17 21:26:04 -04:00
										 |  |  |             GetInteger(kernel_region_start), kernel_region_size, KMemoryRegionType_Kernel)); | 
					
						
							| 
									
										
										
										
											2021-03-20 00:53:00 -07:00
										 |  |  | 
 | 
					
						
							|  |  |  |         // Setup the code region.
 | 
					
						
							|  |  |  |         constexpr size_t CodeRegionAlign = PageSize; | 
					
						
							| 
									
										
										
										
											2023-03-17 21:26:04 -04:00
										 |  |  |         constexpr KVirtualAddress code_region_start = | 
					
						
							|  |  |  |             Common::AlignDown(GetInteger(code_start_virt_addr), CodeRegionAlign); | 
					
						
							|  |  |  |         constexpr KVirtualAddress code_region_end = | 
					
						
							|  |  |  |             Common::AlignUp(GetInteger(code_end_virt_addr), CodeRegionAlign); | 
					
						
							| 
									
										
										
										
											2021-03-20 00:53:00 -07:00
										 |  |  |         constexpr size_t code_region_size = code_region_end - code_region_start; | 
					
						
							| 
									
										
										
										
											2022-02-27 11:05:07 -08:00
										 |  |  |         ASSERT(memory_layout->GetVirtualMemoryRegionTree().Insert( | 
					
						
							| 
									
										
										
										
											2023-03-17 21:26:04 -04:00
										 |  |  |             GetInteger(code_region_start), code_region_size, KMemoryRegionType_KernelCode)); | 
					
						
							| 
									
										
										
										
											2021-03-20 00:53:00 -07:00
										 |  |  | 
 | 
					
						
							|  |  |  |         // Setup board-specific device physical regions.
 | 
					
						
							| 
									
										
										
										
											2022-02-27 11:05:07 -08:00
										 |  |  |         Init::SetupDevicePhysicalMemoryRegions(*memory_layout); | 
					
						
							| 
									
										
										
										
											2021-03-20 00:53:00 -07:00
										 |  |  | 
 | 
					
						
							|  |  |  |         // Determine the amount of space needed for the misc region.
 | 
					
						
							|  |  |  |         size_t misc_region_needed_size; | 
					
						
							|  |  |  |         { | 
					
						
							|  |  |  |             // Each core has a one page stack for all three stack types (Main, Idle, Exception).
 | 
					
						
							|  |  |  |             misc_region_needed_size = Core::Hardware::NUM_CPU_CORES * (3 * (PageSize + PageSize)); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             // Account for each auto-map device.
 | 
					
						
							| 
									
										
										
										
											2022-02-27 11:05:07 -08:00
										 |  |  |             for (const auto& region : memory_layout->GetPhysicalMemoryRegionTree()) { | 
					
						
							| 
									
										
										
										
											2021-03-20 00:53:00 -07:00
										 |  |  |                 if (region.HasTypeAttribute(KMemoryRegionAttr_ShouldKernelMap)) { | 
					
						
							|  |  |  |                     // Check that the region is valid.
 | 
					
						
							|  |  |  |                     ASSERT(region.GetEndAddress() != 0); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |                     // Account for the region.
 | 
					
						
							|  |  |  |                     misc_region_needed_size += | 
					
						
							|  |  |  |                         PageSize + (Common::AlignUp(region.GetLastAddress(), PageSize) - | 
					
						
							|  |  |  |                                     Common::AlignDown(region.GetAddress(), PageSize)); | 
					
						
							|  |  |  |                 } | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             // Multiply the needed size by three, to account for the need for guard space.
 | 
					
						
							|  |  |  |             misc_region_needed_size *= 3; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         // Decide on the actual size for the misc region.
 | 
					
						
							|  |  |  |         constexpr size_t MiscRegionAlign = KernelAslrAlignment; | 
					
						
							| 
									
										
										
										
											2021-06-23 14:18:27 -07:00
										 |  |  |         constexpr size_t MiscRegionMinimumSize = 32_MiB; | 
					
						
							| 
									
										
										
										
											2021-03-20 00:53:00 -07:00
										 |  |  |         const size_t misc_region_size = Common::AlignUp( | 
					
						
							|  |  |  |             std::max(misc_region_needed_size, MiscRegionMinimumSize), MiscRegionAlign); | 
					
						
							|  |  |  |         ASSERT(misc_region_size > 0); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         // Setup the misc region.
 | 
					
						
							| 
									
										
										
										
											2023-03-17 21:26:04 -04:00
										 |  |  |         const KVirtualAddress misc_region_start = | 
					
						
							| 
									
										
										
										
											2022-02-27 11:05:07 -08:00
										 |  |  |             memory_layout->GetVirtualMemoryRegionTree().GetRandomAlignedRegion( | 
					
						
							| 
									
										
										
										
											2021-03-20 00:53:00 -07:00
										 |  |  |                 misc_region_size, MiscRegionAlign, KMemoryRegionType_Kernel); | 
					
						
							| 
									
										
										
										
											2022-02-27 11:05:07 -08:00
										 |  |  |         ASSERT(memory_layout->GetVirtualMemoryRegionTree().Insert( | 
					
						
							| 
									
										
										
										
											2023-03-17 21:26:04 -04:00
										 |  |  |             GetInteger(misc_region_start), misc_region_size, KMemoryRegionType_KernelMisc)); | 
					
						
							| 
									
										
										
										
											2021-03-20 00:53:00 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-10-29 14:23:14 -07:00
										 |  |  |         // Determine if we'll use extra thread resources.
 | 
					
						
							|  |  |  |         const bool use_extra_resources = KSystemControl::Init::ShouldIncreaseThreadResourceLimit(); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-03-20 00:53:00 -07:00
										 |  |  |         // Setup the stack region.
 | 
					
						
							| 
									
										
										
										
											2021-06-23 14:18:27 -07:00
										 |  |  |         constexpr size_t StackRegionSize = 14_MiB; | 
					
						
							| 
									
										
										
										
											2021-03-20 00:53:00 -07:00
										 |  |  |         constexpr size_t StackRegionAlign = KernelAslrAlignment; | 
					
						
							| 
									
										
										
										
											2023-03-17 21:26:04 -04:00
										 |  |  |         const KVirtualAddress stack_region_start = | 
					
						
							| 
									
										
										
										
											2022-02-27 11:05:07 -08:00
										 |  |  |             memory_layout->GetVirtualMemoryRegionTree().GetRandomAlignedRegion( | 
					
						
							| 
									
										
										
										
											2021-03-20 00:53:00 -07:00
										 |  |  |                 StackRegionSize, StackRegionAlign, KMemoryRegionType_Kernel); | 
					
						
							| 
									
										
										
										
											2022-02-27 11:05:07 -08:00
										 |  |  |         ASSERT(memory_layout->GetVirtualMemoryRegionTree().Insert( | 
					
						
							| 
									
										
										
										
											2023-03-17 21:26:04 -04:00
										 |  |  |             GetInteger(stack_region_start), StackRegionSize, KMemoryRegionType_KernelStack)); | 
					
						
							| 
									
										
										
										
											2021-03-20 00:53:00 -07:00
										 |  |  | 
 | 
					
						
							|  |  |  |         // Determine the size of the resource region.
 | 
					
						
							| 
									
										
										
										
											2022-10-29 14:23:14 -07:00
										 |  |  |         const size_t resource_region_size = | 
					
						
							|  |  |  |             memory_layout->GetResourceRegionSizeForInit(use_extra_resources); | 
					
						
							| 
									
										
										
										
											2021-03-20 00:53:00 -07:00
										 |  |  | 
 | 
					
						
							|  |  |  |         // Determine the size of the slab region.
 | 
					
						
							| 
									
										
										
										
											2021-04-03 19:11:46 -07:00
										 |  |  |         const size_t slab_region_size = | 
					
						
							| 
									
										
										
										
											2021-05-04 21:35:42 -07:00
										 |  |  |             Common::AlignUp(Init::CalculateTotalSlabHeapSize(system.Kernel()), PageSize); | 
					
						
							| 
									
										
										
										
											2021-03-20 00:53:00 -07:00
										 |  |  |         ASSERT(slab_region_size <= resource_region_size); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         // Setup the slab region.
 | 
					
						
							| 
									
										
										
										
											2023-03-17 21:26:04 -04:00
										 |  |  |         const KPhysicalAddress code_start_phys_addr = KernelPhysicalAddressCodeBase; | 
					
						
							|  |  |  |         const KPhysicalAddress code_end_phys_addr = code_start_phys_addr + code_region_size; | 
					
						
							|  |  |  |         const KPhysicalAddress slab_start_phys_addr = code_end_phys_addr; | 
					
						
							|  |  |  |         const KPhysicalAddress slab_end_phys_addr = slab_start_phys_addr + slab_region_size; | 
					
						
							| 
									
										
										
										
											2021-03-20 00:53:00 -07:00
										 |  |  |         constexpr size_t SlabRegionAlign = KernelAslrAlignment; | 
					
						
							|  |  |  |         const size_t slab_region_needed_size = | 
					
						
							| 
									
										
										
										
											2023-03-17 21:26:04 -04:00
										 |  |  |             Common::AlignUp(GetInteger(code_end_phys_addr) + slab_region_size, SlabRegionAlign) - | 
					
						
							|  |  |  |             Common::AlignDown(GetInteger(code_end_phys_addr), SlabRegionAlign); | 
					
						
							|  |  |  |         const KVirtualAddress slab_region_start = | 
					
						
							| 
									
										
										
										
											2022-02-27 11:05:07 -08:00
										 |  |  |             memory_layout->GetVirtualMemoryRegionTree().GetRandomAlignedRegion( | 
					
						
							| 
									
										
										
										
											2021-03-20 00:53:00 -07:00
										 |  |  |                 slab_region_needed_size, SlabRegionAlign, KMemoryRegionType_Kernel) + | 
					
						
							| 
									
										
										
										
											2023-03-17 21:26:04 -04:00
										 |  |  |             (GetInteger(code_end_phys_addr) % SlabRegionAlign); | 
					
						
							| 
									
										
										
										
											2022-02-27 11:05:07 -08:00
										 |  |  |         ASSERT(memory_layout->GetVirtualMemoryRegionTree().Insert( | 
					
						
							| 
									
										
										
										
											2023-03-17 21:26:04 -04:00
										 |  |  |             GetInteger(slab_region_start), slab_region_size, KMemoryRegionType_KernelSlab)); | 
					
						
							| 
									
										
										
										
											2021-03-20 00:53:00 -07:00
										 |  |  | 
 | 
					
						
							|  |  |  |         // Setup the temp region.
 | 
					
						
							| 
									
										
										
										
											2021-06-23 14:18:27 -07:00
										 |  |  |         constexpr size_t TempRegionSize = 128_MiB; | 
					
						
							| 
									
										
										
										
											2021-03-20 00:53:00 -07:00
										 |  |  |         constexpr size_t TempRegionAlign = KernelAslrAlignment; | 
					
						
							| 
									
										
										
										
											2023-03-17 21:26:04 -04:00
										 |  |  |         const KVirtualAddress temp_region_start = | 
					
						
							| 
									
										
										
										
											2022-02-27 11:05:07 -08:00
										 |  |  |             memory_layout->GetVirtualMemoryRegionTree().GetRandomAlignedRegion( | 
					
						
							| 
									
										
										
										
											2021-03-20 00:53:00 -07:00
										 |  |  |                 TempRegionSize, TempRegionAlign, KMemoryRegionType_Kernel); | 
					
						
							| 
									
										
										
										
											2023-03-17 21:26:04 -04:00
										 |  |  |         ASSERT(memory_layout->GetVirtualMemoryRegionTree().Insert( | 
					
						
							|  |  |  |             GetInteger(temp_region_start), TempRegionSize, KMemoryRegionType_KernelTemp)); | 
					
						
							| 
									
										
										
										
											2021-03-20 00:53:00 -07:00
										 |  |  | 
 | 
					
						
							|  |  |  |         // Automatically map in devices that have auto-map attributes.
 | 
					
						
							| 
									
										
										
										
											2022-02-27 11:05:07 -08:00
										 |  |  |         for (auto& region : memory_layout->GetPhysicalMemoryRegionTree()) { | 
					
						
							| 
									
										
										
										
											2021-03-20 00:53:00 -07:00
										 |  |  |             // We only care about kernel regions.
 | 
					
						
							|  |  |  |             if (!region.IsDerivedFrom(KMemoryRegionType_Kernel)) { | 
					
						
							|  |  |  |                 continue; | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             // Check whether we should map the region.
 | 
					
						
							|  |  |  |             if (!region.HasTypeAttribute(KMemoryRegionAttr_ShouldKernelMap)) { | 
					
						
							|  |  |  |                 continue; | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             // If this region has already been mapped, no need to consider it.
 | 
					
						
							|  |  |  |             if (region.HasTypeAttribute(KMemoryRegionAttr_DidKernelMap)) { | 
					
						
							|  |  |  |                 continue; | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             // Check that the region is valid.
 | 
					
						
							|  |  |  |             ASSERT(region.GetEndAddress() != 0); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             // Set the attribute to note we've mapped this region.
 | 
					
						
							|  |  |  |             region.SetTypeAttribute(KMemoryRegionAttr_DidKernelMap); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             // Create a virtual pair region and insert it into the tree.
 | 
					
						
							| 
									
										
										
										
											2023-03-17 21:26:04 -04:00
										 |  |  |             const KPhysicalAddress map_phys_addr = Common::AlignDown(region.GetAddress(), PageSize); | 
					
						
							| 
									
										
										
										
											2021-03-20 00:53:00 -07:00
										 |  |  |             const size_t map_size = | 
					
						
							| 
									
										
										
										
											2023-03-17 21:26:04 -04:00
										 |  |  |                 Common::AlignUp(region.GetEndAddress(), PageSize) - GetInteger(map_phys_addr); | 
					
						
							|  |  |  |             const KVirtualAddress map_virt_addr = | 
					
						
							| 
									
										
										
										
											2022-02-27 11:05:07 -08:00
										 |  |  |                 memory_layout->GetVirtualMemoryRegionTree().GetRandomAlignedRegionWithGuard( | 
					
						
							| 
									
										
										
										
											2021-03-20 00:53:00 -07:00
										 |  |  |                     map_size, PageSize, KMemoryRegionType_KernelMisc, PageSize); | 
					
						
							| 
									
										
										
										
											2022-02-27 11:05:07 -08:00
										 |  |  |             ASSERT(memory_layout->GetVirtualMemoryRegionTree().Insert( | 
					
						
							| 
									
										
										
										
											2023-03-17 21:26:04 -04:00
										 |  |  |                 GetInteger(map_virt_addr), map_size, KMemoryRegionType_KernelMiscMappedDevice)); | 
					
						
							|  |  |  |             region.SetPairAddress(GetInteger(map_virt_addr) + region.GetAddress() - | 
					
						
							|  |  |  |                                   GetInteger(map_phys_addr)); | 
					
						
							| 
									
										
										
										
											2021-03-20 00:53:00 -07:00
										 |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-02-27 11:05:07 -08:00
										 |  |  |         Init::SetupDramPhysicalMemoryRegions(*memory_layout); | 
					
						
							| 
									
										
										
										
											2021-03-20 00:53:00 -07:00
										 |  |  | 
 | 
					
						
							|  |  |  |         // Insert a physical region for the kernel code region.
 | 
					
						
							| 
									
										
										
										
											2022-02-27 11:05:07 -08:00
										 |  |  |         ASSERT(memory_layout->GetPhysicalMemoryRegionTree().Insert( | 
					
						
							| 
									
										
										
										
											2023-03-17 21:26:04 -04:00
										 |  |  |             GetInteger(code_start_phys_addr), code_region_size, KMemoryRegionType_DramKernelCode)); | 
					
						
							| 
									
										
										
										
											2021-03-20 00:53:00 -07:00
										 |  |  | 
 | 
					
						
							|  |  |  |         // Insert a physical region for the kernel slab region.
 | 
					
						
							| 
									
										
										
										
											2022-02-27 11:05:07 -08:00
										 |  |  |         ASSERT(memory_layout->GetPhysicalMemoryRegionTree().Insert( | 
					
						
							| 
									
										
										
										
											2023-03-17 21:26:04 -04:00
										 |  |  |             GetInteger(slab_start_phys_addr), slab_region_size, KMemoryRegionType_DramKernelSlab)); | 
					
						
							| 
									
										
										
										
											2021-03-20 00:53:00 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-10-11 12:55:12 -04:00
										 |  |  |         // Insert a physical region for the secure applet memory.
 | 
					
						
							|  |  |  |         const auto secure_applet_end_phys_addr = | 
					
						
							|  |  |  |             slab_end_phys_addr + KSystemControl::SecureAppletMemorySize; | 
					
						
							|  |  |  |         if constexpr (KSystemControl::SecureAppletMemorySize > 0) { | 
					
						
							|  |  |  |             ASSERT(memory_layout->GetPhysicalMemoryRegionTree().Insert( | 
					
						
							|  |  |  |                 GetInteger(slab_end_phys_addr), KSystemControl::SecureAppletMemorySize, | 
					
						
							|  |  |  |                 KMemoryRegionType_DramKernelSecureAppletMemory)); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         // Insert a physical region for the unknown debug2 region.
 | 
					
						
							|  |  |  |         constexpr size_t SecureUnknownRegionSize = 0; | 
					
						
							|  |  |  |         const size_t secure_unknown_size = SecureUnknownRegionSize; | 
					
						
							|  |  |  |         const auto secure_unknown_end_phys_addr = secure_applet_end_phys_addr + secure_unknown_size; | 
					
						
							| 
									
										
										
										
											2023-10-11 15:08:00 -04:00
										 |  |  |         if constexpr (SecureUnknownRegionSize > 0) { | 
					
						
							| 
									
										
										
										
											2023-10-11 12:55:12 -04:00
										 |  |  |             ASSERT(memory_layout->GetPhysicalMemoryRegionTree().Insert( | 
					
						
							|  |  |  |                 GetInteger(secure_applet_end_phys_addr), secure_unknown_size, | 
					
						
							|  |  |  |                 KMemoryRegionType_DramKernelSecureUnknown)); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-03-20 00:53:00 -07:00
										 |  |  |         // Determine size available for kernel page table heaps, requiring > 8 MB.
 | 
					
						
							| 
									
										
										
										
											2023-03-17 21:26:04 -04:00
										 |  |  |         const KPhysicalAddress resource_end_phys_addr = slab_start_phys_addr + resource_region_size; | 
					
						
							| 
									
										
										
										
											2023-10-11 12:55:12 -04:00
										 |  |  |         const size_t page_table_heap_size = resource_end_phys_addr - secure_unknown_end_phys_addr; | 
					
						
							| 
									
										
										
										
											2021-06-23 14:18:27 -07:00
										 |  |  |         ASSERT(page_table_heap_size / 4_MiB > 2); | 
					
						
							| 
									
										
										
										
											2021-03-20 00:53:00 -07:00
										 |  |  | 
 | 
					
						
							|  |  |  |         // Insert a physical region for the kernel page table heap region
 | 
					
						
							| 
									
										
										
										
											2022-02-27 11:05:07 -08:00
										 |  |  |         ASSERT(memory_layout->GetPhysicalMemoryRegionTree().Insert( | 
					
						
							| 
									
										
										
										
											2023-10-11 12:55:12 -04:00
										 |  |  |             GetInteger(secure_unknown_end_phys_addr), page_table_heap_size, | 
					
						
							| 
									
										
										
										
											2023-03-17 21:26:04 -04:00
										 |  |  |             KMemoryRegionType_DramKernelPtHeap)); | 
					
						
							| 
									
										
										
										
											2021-03-20 00:53:00 -07:00
										 |  |  | 
 | 
					
						
							|  |  |  |         // All DRAM regions that we haven't tagged by this point will be mapped under the linear
 | 
					
						
							|  |  |  |         // mapping. Tag them.
 | 
					
						
							| 
									
										
										
										
											2022-02-27 11:05:07 -08:00
										 |  |  |         for (auto& region : memory_layout->GetPhysicalMemoryRegionTree()) { | 
					
						
							| 
									
										
										
										
											2021-03-20 00:53:00 -07:00
										 |  |  |             if (region.GetType() == KMemoryRegionType_Dram) { | 
					
						
							|  |  |  |                 // Check that the region is valid.
 | 
					
						
							|  |  |  |                 ASSERT(region.GetEndAddress() != 0); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |                 // Set the linear map attribute.
 | 
					
						
							|  |  |  |                 region.SetTypeAttribute(KMemoryRegionAttr_LinearMapped); | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         // Get the linear region extents.
 | 
					
						
							|  |  |  |         const auto linear_extents = | 
					
						
							| 
									
										
										
										
											2022-02-27 11:05:07 -08:00
										 |  |  |             memory_layout->GetPhysicalMemoryRegionTree().GetDerivedRegionExtents( | 
					
						
							| 
									
										
										
										
											2021-03-20 00:53:00 -07:00
										 |  |  |                 KMemoryRegionAttr_LinearMapped); | 
					
						
							|  |  |  |         ASSERT(linear_extents.GetEndAddress() != 0); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         // Setup the linear mapping region.
 | 
					
						
							| 
									
										
										
										
											2021-06-23 14:18:27 -07:00
										 |  |  |         constexpr size_t LinearRegionAlign = 1_GiB; | 
					
						
							| 
									
										
										
										
											2023-03-17 21:26:04 -04:00
										 |  |  |         const KPhysicalAddress aligned_linear_phys_start = | 
					
						
							| 
									
										
										
										
											2021-03-20 00:53:00 -07:00
										 |  |  |             Common::AlignDown(linear_extents.GetAddress(), LinearRegionAlign); | 
					
						
							|  |  |  |         const size_t linear_region_size = | 
					
						
							|  |  |  |             Common::AlignUp(linear_extents.GetEndAddress(), LinearRegionAlign) - | 
					
						
							| 
									
										
										
										
											2023-03-17 21:26:04 -04:00
										 |  |  |             GetInteger(aligned_linear_phys_start); | 
					
						
							|  |  |  |         const KVirtualAddress linear_region_start = | 
					
						
							| 
									
										
										
										
											2022-02-27 11:05:07 -08:00
										 |  |  |             memory_layout->GetVirtualMemoryRegionTree().GetRandomAlignedRegionWithGuard( | 
					
						
							| 
									
										
										
										
											2021-03-20 00:53:00 -07:00
										 |  |  |                 linear_region_size, LinearRegionAlign, KMemoryRegionType_None, LinearRegionAlign); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-03-17 21:26:04 -04:00
										 |  |  |         const u64 linear_region_phys_to_virt_diff = | 
					
						
							|  |  |  |             GetInteger(linear_region_start) - GetInteger(aligned_linear_phys_start); | 
					
						
							| 
									
										
										
										
											2021-03-20 00:53:00 -07:00
										 |  |  | 
 | 
					
						
							|  |  |  |         // Map and create regions for all the linearly-mapped data.
 | 
					
						
							|  |  |  |         { | 
					
						
							| 
									
										
										
										
											2023-03-17 21:26:04 -04:00
										 |  |  |             KPhysicalAddress cur_phys_addr = 0; | 
					
						
							| 
									
										
										
										
											2021-03-20 00:53:00 -07:00
										 |  |  |             u64 cur_size = 0; | 
					
						
							| 
									
										
										
										
											2022-02-27 11:05:07 -08:00
										 |  |  |             for (auto& region : memory_layout->GetPhysicalMemoryRegionTree()) { | 
					
						
							| 
									
										
										
										
											2021-03-20 00:53:00 -07:00
										 |  |  |                 if (!region.HasTypeAttribute(KMemoryRegionAttr_LinearMapped)) { | 
					
						
							|  |  |  |                     continue; | 
					
						
							|  |  |  |                 } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |                 ASSERT(region.GetEndAddress() != 0); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |                 if (cur_size == 0) { | 
					
						
							|  |  |  |                     cur_phys_addr = region.GetAddress(); | 
					
						
							|  |  |  |                     cur_size = region.GetSize(); | 
					
						
							|  |  |  |                 } else if (cur_phys_addr + cur_size == region.GetAddress()) { | 
					
						
							|  |  |  |                     cur_size += region.GetSize(); | 
					
						
							|  |  |  |                 } else { | 
					
						
							|  |  |  |                     cur_phys_addr = region.GetAddress(); | 
					
						
							|  |  |  |                     cur_size = region.GetSize(); | 
					
						
							|  |  |  |                 } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-03-17 21:26:04 -04:00
										 |  |  |                 const KVirtualAddress region_virt_addr = | 
					
						
							| 
									
										
										
										
											2021-03-20 00:53:00 -07:00
										 |  |  |                     region.GetAddress() + linear_region_phys_to_virt_diff; | 
					
						
							| 
									
										
										
										
											2022-02-27 11:05:07 -08:00
										 |  |  |                 ASSERT(memory_layout->GetVirtualMemoryRegionTree().Insert( | 
					
						
							| 
									
										
										
										
											2023-03-17 21:26:04 -04:00
										 |  |  |                     GetInteger(region_virt_addr), region.GetSize(), | 
					
						
							| 
									
										
										
										
											2021-03-20 00:53:00 -07:00
										 |  |  |                     GetTypeForVirtualLinearMapping(region.GetType()))); | 
					
						
							| 
									
										
										
										
											2023-03-17 21:26:04 -04:00
										 |  |  |                 region.SetPairAddress(GetInteger(region_virt_addr)); | 
					
						
							| 
									
										
										
										
											2021-03-20 00:53:00 -07:00
										 |  |  | 
 | 
					
						
							|  |  |  |                 KMemoryRegion* virt_region = | 
					
						
							| 
									
										
										
										
											2023-03-17 21:26:04 -04:00
										 |  |  |                     memory_layout->GetVirtualMemoryRegionTree().FindModifiable( | 
					
						
							|  |  |  |                         GetInteger(region_virt_addr)); | 
					
						
							| 
									
										
										
										
											2021-03-20 00:53:00 -07:00
										 |  |  |                 ASSERT(virt_region != nullptr); | 
					
						
							|  |  |  |                 virt_region->SetPairAddress(region.GetAddress()); | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         // Insert regions for the initial page table region.
 | 
					
						
							| 
									
										
										
										
											2022-02-27 11:05:07 -08:00
										 |  |  |         ASSERT(memory_layout->GetPhysicalMemoryRegionTree().Insert( | 
					
						
							| 
									
										
										
										
											2023-03-17 21:26:04 -04:00
										 |  |  |             GetInteger(resource_end_phys_addr), KernelPageTableHeapSize, | 
					
						
							|  |  |  |             KMemoryRegionType_DramKernelInitPt)); | 
					
						
							| 
									
										
										
										
											2022-02-27 11:05:07 -08:00
										 |  |  |         ASSERT(memory_layout->GetVirtualMemoryRegionTree().Insert( | 
					
						
							| 
									
										
										
										
											2023-03-17 21:26:04 -04:00
										 |  |  |             GetInteger(resource_end_phys_addr) + linear_region_phys_to_virt_diff, | 
					
						
							|  |  |  |             KernelPageTableHeapSize, KMemoryRegionType_VirtualDramKernelInitPt)); | 
					
						
							| 
									
										
										
										
											2021-03-20 00:53:00 -07:00
										 |  |  | 
 | 
					
						
							|  |  |  |         // All linear-mapped DRAM regions that we haven't tagged by this point will be allocated to
 | 
					
						
							|  |  |  |         // some pool partition. Tag them.
 | 
					
						
							| 
									
										
										
										
											2022-02-27 11:05:07 -08:00
										 |  |  |         for (auto& region : memory_layout->GetPhysicalMemoryRegionTree()) { | 
					
						
							| 
									
										
										
										
											2021-03-20 00:53:00 -07:00
										 |  |  |             if (region.GetType() == (KMemoryRegionType_Dram | KMemoryRegionAttr_LinearMapped)) { | 
					
						
							|  |  |  |                 region.SetType(KMemoryRegionType_DramPoolPartition); | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         // Setup all other memory regions needed to arrange the pool partitions.
 | 
					
						
							| 
									
										
										
										
											2022-02-27 11:05:07 -08:00
										 |  |  |         Init::SetupPoolPartitionMemoryRegions(*memory_layout); | 
					
						
							| 
									
										
										
										
											2021-03-20 00:53:00 -07:00
										 |  |  | 
 | 
					
						
							|  |  |  |         // Cache all linear regions in their own trees for faster access, later.
 | 
					
						
							| 
									
										
										
										
											2022-02-27 11:05:07 -08:00
										 |  |  |         memory_layout->InitializeLinearMemoryRegionTrees(aligned_linear_phys_start, | 
					
						
							|  |  |  |                                                          linear_region_start); | 
					
						
							| 
									
										
										
										
											2021-03-23 18:47:16 -07:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-02-26 10:46:31 -08:00
										 |  |  |     void InitializeMemoryLayout() { | 
					
						
							|  |  |  |         // Initialize the memory manager.
 | 
					
						
							| 
									
										
										
										
											2022-01-11 21:57:01 -08:00
										 |  |  |         memory_manager = std::make_unique<KMemoryManager>(system); | 
					
						
							| 
									
										
										
										
											2022-02-27 11:05:07 -08:00
										 |  |  |         const auto& management_region = memory_layout->GetPoolManagementRegion(); | 
					
						
							| 
									
										
										
										
											2022-02-26 10:46:31 -08:00
										 |  |  |         ASSERT(management_region.GetEndAddress() != 0); | 
					
						
							|  |  |  |         memory_manager->Initialize(management_region.GetAddress(), management_region.GetSize()); | 
					
						
							| 
									
										
										
										
											2022-12-23 13:40:30 -05:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2021-03-20 00:53:00 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-04-29 21:16:09 -04:00
										 |  |  |     void InitializeHackSharedMemory(KernelCore& kernel) { | 
					
						
							| 
									
										
										
										
											2021-03-20 00:53:00 -07:00
										 |  |  |         // Setup memory regions for emulated processes
 | 
					
						
							|  |  |  |         // TODO(bunnei): These should not be hardcoded regions initialized within the kernel
 | 
					
						
							| 
									
										
										
										
											2020-04-08 21:06:37 -04:00
										 |  |  |         constexpr std::size_t font_size{0x1100000}; | 
					
						
							|  |  |  |         constexpr std::size_t irs_size{0x8000}; | 
					
						
							|  |  |  |         constexpr std::size_t time_size{0x1000}; | 
					
						
							| 
									
										
										
										
											2021-05-02 18:41:03 -05:00
										 |  |  |         constexpr std::size_t hidbus_size{0x1000}; | 
					
						
							| 
									
										
										
										
											2020-04-08 21:06:37 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-04-09 22:42:23 -07:00
										 |  |  |         font_shared_mem = KSharedMemory::Create(system.Kernel()); | 
					
						
							|  |  |  |         irs_shared_mem = KSharedMemory::Create(system.Kernel()); | 
					
						
							|  |  |  |         time_shared_mem = KSharedMemory::Create(system.Kernel()); | 
					
						
							| 
									
										
										
										
											2021-05-02 18:41:03 -05:00
										 |  |  |         hidbus_shared_mem = KSharedMemory::Create(system.Kernel()); | 
					
						
							| 
									
										
										
										
											2021-04-09 16:56:11 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-12-23 13:40:30 -05:00
										 |  |  |         font_shared_mem->Initialize(system.DeviceMemory(), nullptr, Svc::MemoryPermission::None, | 
					
						
							| 
									
										
										
										
											2023-03-06 20:34:25 -05:00
										 |  |  |                                     Svc::MemoryPermission::Read, font_size); | 
					
						
							| 
									
										
										
										
											2023-04-29 21:16:09 -04:00
										 |  |  |         KSharedMemory::Register(kernel, font_shared_mem); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-12-23 13:40:30 -05:00
										 |  |  |         irs_shared_mem->Initialize(system.DeviceMemory(), nullptr, Svc::MemoryPermission::None, | 
					
						
							| 
									
										
										
										
											2023-03-06 20:34:25 -05:00
										 |  |  |                                    Svc::MemoryPermission::Read, irs_size); | 
					
						
							| 
									
										
										
										
											2023-04-29 21:16:09 -04:00
										 |  |  |         KSharedMemory::Register(kernel, irs_shared_mem); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-12-23 13:40:30 -05:00
										 |  |  |         time_shared_mem->Initialize(system.DeviceMemory(), nullptr, Svc::MemoryPermission::None, | 
					
						
							| 
									
										
										
										
											2023-03-06 20:34:25 -05:00
										 |  |  |                                     Svc::MemoryPermission::Read, time_size); | 
					
						
							| 
									
										
										
										
											2023-04-29 21:16:09 -04:00
										 |  |  |         KSharedMemory::Register(kernel, time_shared_mem); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-12-23 13:40:30 -05:00
										 |  |  |         hidbus_shared_mem->Initialize(system.DeviceMemory(), nullptr, Svc::MemoryPermission::None, | 
					
						
							| 
									
										
										
										
											2023-03-06 20:34:25 -05:00
										 |  |  |                                       Svc::MemoryPermission::Read, hidbus_size); | 
					
						
							| 
									
										
										
										
											2023-04-29 21:16:09 -04:00
										 |  |  |         KSharedMemory::Register(kernel, hidbus_shared_mem); | 
					
						
							| 
									
										
										
										
											2021-04-12 16:14:19 -04:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2020-04-08 21:06:37 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-11-17 15:00:55 -08:00
										 |  |  |     std::mutex registered_objects_lock; | 
					
						
							|  |  |  |     std::mutex registered_in_use_objects_lock; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-08-28 12:30:33 -04:00
										 |  |  |     std::atomic<u32> next_object_id{0}; | 
					
						
							| 
									
										
										
										
											2023-10-21 16:47:43 -04:00
										 |  |  |     std::atomic<u64> next_kernel_process_id{KProcess::InitialProcessIdMin}; | 
					
						
							|  |  |  |     std::atomic<u64> next_user_process_id{KProcess::ProcessIdMin}; | 
					
						
							| 
									
										
										
										
											2018-12-18 22:37:01 -05:00
										 |  |  |     std::atomic<u64> next_thread_id{1}; | 
					
						
							| 
									
										
										
										
											2018-08-28 12:30:33 -04:00
										 |  |  | 
 | 
					
						
							|  |  |  |     // Lists all processes that exist in the current session.
 | 
					
						
							| 
									
										
										
										
											2024-01-02 17:12:16 -05:00
										 |  |  |     std::mutex process_list_lock; | 
					
						
							| 
									
										
										
										
											2021-04-23 22:04:28 -07:00
										 |  |  |     std::vector<KProcess*> process_list; | 
					
						
							| 
									
										
										
										
											2024-01-02 18:29:03 -05:00
										 |  |  |     KProcess* application_process{}; | 
					
						
							| 
									
										
										
										
											2020-12-02 18:08:35 -08:00
										 |  |  |     std::unique_ptr<Kernel::GlobalSchedulerContext> global_scheduler_context; | 
					
						
							| 
									
										
										
										
											2022-12-18 16:50:02 -05:00
										 |  |  |     std::unique_ptr<Kernel::KHardwareTimer> hardware_timer; | 
					
						
							| 
									
										
										
										
											2018-08-28 12:30:33 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-05-04 21:35:42 -07:00
										 |  |  |     Init::KSlabResourceCounts slab_resource_counts{}; | 
					
						
							| 
									
										
										
										
											2021-04-20 21:28:11 -07:00
										 |  |  |     KResourceLimit* system_resource_limit{}; | 
					
						
							| 
									
										
										
										
											2018-08-28 12:30:33 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-10-29 14:45:09 -07:00
										 |  |  |     KPageBufferSlabHeap page_buffer_slab_heap; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-11-26 21:48:56 -05:00
										 |  |  |     std::shared_ptr<Core::Timing::EventType> preemption_event; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-03-11 17:24:37 -08:00
										 |  |  |     std::unique_ptr<KAutoObjectWithListContainer> global_object_list_container; | 
					
						
							| 
									
										
										
										
											2021-04-03 19:11:46 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-02-16 23:16:08 -05:00
										 |  |  |     std::unique_ptr<KObjectNameGlobalData> object_name_global_data; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-06-28 14:38:14 -07:00
										 |  |  |     std::unordered_set<KAutoObject*> registered_objects; | 
					
						
							| 
									
										
										
										
											2021-10-25 18:55:20 +08:00
										 |  |  |     std::unordered_set<KAutoObject*> registered_in_use_objects; | 
					
						
							| 
									
										
										
										
											2019-03-05 12:28:10 -05:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-02-18 16:26:48 -05:00
										 |  |  |     std::mutex server_lock; | 
					
						
							|  |  |  |     std::vector<std::unique_ptr<Service::ServerManager>> server_managers; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-07-07 20:06:46 -04:00
										 |  |  |     std::array<std::unique_ptr<Kernel::PhysicalCore>, Core::Hardware::NUM_CPU_CORES> cores; | 
					
						
							| 
									
										
										
										
											2020-01-25 18:55:32 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-12-29 15:55:30 -08:00
										 |  |  |     // Next host thead ID to use, 0-3 IDs represent core threads, >3 represent others
 | 
					
						
							|  |  |  |     std::atomic<u32> next_host_thread_id{Core::Hardware::NUM_CPU_CORES}; | 
					
						
							| 
									
										
										
										
											2020-02-14 09:30:53 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-04-08 21:06:37 -04:00
										 |  |  |     // Kernel memory management
 | 
					
						
							| 
									
										
										
										
											2021-02-12 17:38:40 -08:00
										 |  |  |     std::unique_ptr<KMemoryManager> memory_manager; | 
					
						
							| 
									
										
										
										
											2020-04-08 21:06:37 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-10-29 14:45:09 -07:00
										 |  |  |     // Resource managers
 | 
					
						
							|  |  |  |     std::unique_ptr<KDynamicPageManager> resource_manager_page_manager; | 
					
						
							|  |  |  |     std::unique_ptr<KPageTableSlabHeap> page_table_heap; | 
					
						
							|  |  |  |     std::unique_ptr<KMemoryBlockSlabHeap> app_memory_block_heap; | 
					
						
							|  |  |  |     std::unique_ptr<KMemoryBlockSlabHeap> sys_memory_block_heap; | 
					
						
							|  |  |  |     std::unique_ptr<KBlockInfoSlabHeap> block_info_heap; | 
					
						
							|  |  |  |     std::unique_ptr<KPageTableManager> app_page_table_manager; | 
					
						
							|  |  |  |     std::unique_ptr<KPageTableManager> sys_page_table_manager; | 
					
						
							| 
									
										
										
										
											2022-09-16 23:33:47 -07:00
										 |  |  |     std::unique_ptr<KMemoryBlockSlabManager> app_memory_block_manager; | 
					
						
							| 
									
										
										
										
											2022-10-29 14:45:09 -07:00
										 |  |  |     std::unique_ptr<KMemoryBlockSlabManager> sys_memory_block_manager; | 
					
						
							|  |  |  |     std::unique_ptr<KBlockInfoManager> app_block_info_manager; | 
					
						
							|  |  |  |     std::unique_ptr<KBlockInfoManager> sys_block_info_manager; | 
					
						
							|  |  |  |     std::unique_ptr<KSystemResource> app_system_resource; | 
					
						
							|  |  |  |     std::unique_ptr<KSystemResource> sys_system_resource; | 
					
						
							| 
									
										
										
										
											2022-09-16 23:33:47 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-04-08 21:06:37 -04:00
										 |  |  |     // Shared memory for services
 | 
					
						
							| 
									
										
										
										
											2021-04-09 22:42:23 -07:00
										 |  |  |     Kernel::KSharedMemory* hid_shared_mem{}; | 
					
						
							|  |  |  |     Kernel::KSharedMemory* font_shared_mem{}; | 
					
						
							|  |  |  |     Kernel::KSharedMemory* irs_shared_mem{}; | 
					
						
							|  |  |  |     Kernel::KSharedMemory* time_shared_mem{}; | 
					
						
							| 
									
										
										
										
											2021-05-02 18:41:03 -05:00
										 |  |  |     Kernel::KSharedMemory* hidbus_shared_mem{}; | 
					
						
							| 
									
										
										
										
											2020-04-08 21:06:37 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-02-26 10:46:31 -08:00
										 |  |  |     // Memory layout
 | 
					
						
							| 
									
										
										
										
											2022-02-27 11:05:07 -08:00
										 |  |  |     std::unique_ptr<KMemoryLayout> memory_layout; | 
					
						
							| 
									
										
										
										
											2022-02-26 10:46:31 -08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-12-05 14:48:32 -05:00
										 |  |  |     std::array<KThread*, Core::Hardware::NUM_CPU_CORES> shutdown_threads{}; | 
					
						
							| 
									
										
										
										
											2020-12-02 18:08:35 -08:00
										 |  |  |     std::array<std::unique_ptr<Kernel::KScheduler>, Core::Hardware::NUM_CPU_CORES> schedulers{}; | 
					
						
							| 
									
										
										
										
											2020-02-24 22:04:12 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-03-08 22:39:41 -04:00
										 |  |  |     bool is_multicore{}; | 
					
						
							| 
									
										
										
										
											2021-11-25 20:46:17 -08:00
										 |  |  |     std::atomic_bool is_shutting_down{}; | 
					
						
							| 
									
										
										
										
											2020-12-29 15:55:30 -08:00
										 |  |  |     u32 single_core_thread_id{}; | 
					
						
							| 
									
										
										
										
											2020-03-08 22:39:41 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-03-12 16:48:43 -04:00
										 |  |  |     std::array<u64, Core::Hardware::NUM_CPU_CORES> svc_ticks{}; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-01-14 16:25:37 -08:00
										 |  |  |     KWorkerTaskManager worker_task_manager; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-03-05 12:28:10 -05:00
										 |  |  |     // System context
 | 
					
						
							|  |  |  |     Core::System& system; | 
					
						
							| 
									
										
										
										
											2018-08-28 12:30:33 -04:00
										 |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-02-13 22:04:10 -04:00
										 |  |  | KernelCore::KernelCore(Core::System& system) : impl{std::make_unique<Impl>(system, *this)} {} | 
					
						
							| 
									
										
										
										
											2021-04-23 17:04:33 -07:00
										 |  |  | KernelCore::~KernelCore() = default; | 
					
						
							| 
									
										
										
										
											2018-08-28 12:30:33 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-03-08 22:39:41 -04:00
										 |  |  | void KernelCore::SetMulticore(bool is_multicore) { | 
					
						
							|  |  |  |     impl->SetMulticore(is_multicore); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-03-05 12:28:10 -05:00
										 |  |  | void KernelCore::Initialize() { | 
					
						
							| 
									
										
										
										
											2021-04-09 23:16:13 -07:00
										 |  |  |     slab_heap_container = std::make_unique<SlabHeapContainer>(); | 
					
						
							| 
									
										
										
										
											2019-03-05 12:28:10 -05:00
										 |  |  |     impl->Initialize(*this); | 
					
						
							| 
									
										
										
										
											2018-08-28 12:30:33 -04:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void KernelCore::Shutdown() { | 
					
						
							|  |  |  |     impl->Shutdown(); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-07-16 23:48:45 +01:00
										 |  |  | void KernelCore::CloseServices() { | 
					
						
							|  |  |  |     impl->CloseServices(); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-04-20 21:28:11 -07:00
										 |  |  | const KResourceLimit* KernelCore::GetSystemResourceLimit() const { | 
					
						
							|  |  |  |     return impl->system_resource_limit; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | KResourceLimit* KernelCore::GetSystemResourceLimit() { | 
					
						
							| 
									
										
										
										
											2018-11-19 12:54:06 -05:00
										 |  |  |     return impl->system_resource_limit; | 
					
						
							| 
									
										
										
										
											2018-08-28 12:30:33 -04:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-04-23 22:04:28 -07:00
										 |  |  | void KernelCore::AppendNewProcess(KProcess* process) { | 
					
						
							| 
									
										
										
										
											2024-01-02 17:12:16 -05:00
										 |  |  |     process->Open(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     std::scoped_lock lk{impl->process_list_lock}; | 
					
						
							| 
									
										
										
										
											2021-04-03 22:22:36 -07:00
										 |  |  |     impl->process_list.push_back(process); | 
					
						
							| 
									
										
										
										
											2018-08-28 12:30:33 -04:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-01-02 17:12:16 -05:00
										 |  |  | void KernelCore::RemoveProcess(KProcess* process) { | 
					
						
							|  |  |  |     std::scoped_lock lk{impl->process_list_lock}; | 
					
						
							|  |  |  |     if (std::erase(impl->process_list, process)) { | 
					
						
							|  |  |  |         process->Close(); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-02-13 11:21:43 -05:00
										 |  |  | void KernelCore::MakeApplicationProcess(KProcess* process) { | 
					
						
							|  |  |  |     impl->MakeApplicationProcess(process); | 
					
						
							| 
									
										
										
										
											2018-09-06 20:34:51 -04:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-02-13 11:21:43 -05:00
										 |  |  | KProcess* KernelCore::ApplicationProcess() { | 
					
						
							|  |  |  |     return impl->application_process; | 
					
						
							| 
									
										
										
										
											2018-09-06 20:34:51 -04:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-02-13 11:21:43 -05:00
										 |  |  | const KProcess* KernelCore::ApplicationProcess() const { | 
					
						
							|  |  |  |     return impl->application_process; | 
					
						
							| 
									
										
										
										
											2018-09-06 20:34:51 -04:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-01-02 17:12:16 -05:00
										 |  |  | std::list<KScopedAutoObject<KProcess>> KernelCore::GetProcessList() { | 
					
						
							|  |  |  |     std::list<KScopedAutoObject<KProcess>> processes; | 
					
						
							|  |  |  |     std::scoped_lock lk{impl->process_list_lock}; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     for (auto* const process : impl->process_list) { | 
					
						
							|  |  |  |         processes.emplace_back(process); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     return processes; | 
					
						
							| 
									
										
										
										
											2019-03-20 15:03:52 -04:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-12-02 18:08:35 -08:00
										 |  |  | Kernel::GlobalSchedulerContext& KernelCore::GlobalSchedulerContext() { | 
					
						
							|  |  |  |     return *impl->global_scheduler_context; | 
					
						
							| 
									
										
										
										
											2019-03-29 17:02:57 -04:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-12-02 18:08:35 -08:00
										 |  |  | const Kernel::GlobalSchedulerContext& KernelCore::GlobalSchedulerContext() const { | 
					
						
							|  |  |  |     return *impl->global_scheduler_context; | 
					
						
							| 
									
										
										
										
											2019-03-29 17:02:57 -04:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-12-02 18:08:35 -08:00
										 |  |  | Kernel::KScheduler& KernelCore::Scheduler(std::size_t id) { | 
					
						
							| 
									
										
										
										
											2020-03-01 12:14:17 -04:00
										 |  |  |     return *impl->schedulers[id]; | 
					
						
							| 
									
										
										
										
											2020-02-13 22:04:10 -04:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-12-02 18:08:35 -08:00
										 |  |  | const Kernel::KScheduler& KernelCore::Scheduler(std::size_t id) const { | 
					
						
							| 
									
										
										
										
											2020-03-01 12:14:17 -04:00
										 |  |  |     return *impl->schedulers[id]; | 
					
						
							| 
									
										
										
										
											2020-02-13 22:04:10 -04:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-01-25 18:55:32 -04:00
										 |  |  | Kernel::PhysicalCore& KernelCore::PhysicalCore(std::size_t id) { | 
					
						
							| 
									
										
										
										
											2022-07-07 20:06:46 -04:00
										 |  |  |     return *impl->cores[id]; | 
					
						
							| 
									
										
										
										
											2020-01-25 18:55:32 -04:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | const Kernel::PhysicalCore& KernelCore::PhysicalCore(std::size_t id) const { | 
					
						
							| 
									
										
										
										
											2022-07-07 20:06:46 -04:00
										 |  |  |     return *impl->cores[id]; | 
					
						
							| 
									
										
										
										
											2020-01-25 18:55:32 -04:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-08-06 22:45:18 -07:00
										 |  |  | size_t KernelCore::CurrentPhysicalCoreIndex() const { | 
					
						
							|  |  |  |     const u32 core_id = impl->GetCurrentHostThreadID(); | 
					
						
							|  |  |  |     if (core_id >= Core::Hardware::NUM_CPU_CORES) { | 
					
						
							|  |  |  |         return Core::Hardware::NUM_CPU_CORES - 1; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     return core_id; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-02-24 22:04:12 -04:00
										 |  |  | Kernel::PhysicalCore& KernelCore::CurrentPhysicalCore() { | 
					
						
							| 
									
										
										
										
											2022-07-07 20:06:46 -04:00
										 |  |  |     return *impl->cores[CurrentPhysicalCoreIndex()]; | 
					
						
							| 
									
										
										
										
											2020-02-24 22:04:12 -04:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | const Kernel::PhysicalCore& KernelCore::CurrentPhysicalCore() const { | 
					
						
							| 
									
										
										
										
											2022-07-07 20:06:46 -04:00
										 |  |  |     return *impl->cores[CurrentPhysicalCoreIndex()]; | 
					
						
							| 
									
										
										
										
											2020-02-24 22:04:12 -04:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-12-02 18:08:35 -08:00
										 |  |  | Kernel::KScheduler* KernelCore::CurrentScheduler() { | 
					
						
							| 
									
										
										
										
											2023-02-10 20:43:06 -05:00
										 |  |  |     const u32 core_id = impl->GetCurrentHostThreadID(); | 
					
						
							| 
									
										
										
										
											2020-12-02 18:08:35 -08:00
										 |  |  |     if (core_id >= Core::Hardware::NUM_CPU_CORES) { | 
					
						
							|  |  |  |         // This is expected when called from not a guest thread
 | 
					
						
							|  |  |  |         return {}; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     return impl->schedulers[core_id].get(); | 
					
						
							| 
									
										
										
										
											2020-03-01 12:14:17 -04:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-12-18 16:50:02 -05:00
										 |  |  | Kernel::KHardwareTimer& KernelCore::HardwareTimer() { | 
					
						
							|  |  |  |     return *impl->hardware_timer; | 
					
						
							| 
									
										
										
										
											2020-02-14 10:56:27 -04:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-04-03 19:11:46 -07:00
										 |  |  | KAutoObjectWithListContainer& KernelCore::ObjectListContainer() { | 
					
						
							| 
									
										
										
										
											2022-03-11 17:24:37 -08:00
										 |  |  |     return *impl->global_object_list_container; | 
					
						
							| 
									
										
										
										
											2021-04-03 19:11:46 -07:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | const KAutoObjectWithListContainer& KernelCore::ObjectListContainer() const { | 
					
						
							| 
									
										
										
										
											2022-03-11 17:24:37 -08:00
										 |  |  |     return *impl->global_object_list_container; | 
					
						
							| 
									
										
										
										
											2021-04-03 19:11:46 -07:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-01-25 18:55:32 -04:00
										 |  |  | void KernelCore::PrepareReschedule(std::size_t id) { | 
					
						
							| 
									
										
										
										
											2020-03-01 12:14:17 -04:00
										 |  |  |     // TODO: Reimplement, this
 | 
					
						
							| 
									
										
										
										
											2020-01-25 18:55:32 -04:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-06-28 14:38:14 -07:00
										 |  |  | void KernelCore::RegisterKernelObject(KAutoObject* object) { | 
					
						
							| 
									
										
										
										
											2022-04-07 19:31:32 +01:00
										 |  |  |     std::scoped_lock lk{impl->registered_objects_lock}; | 
					
						
							| 
									
										
										
										
											2021-06-28 14:38:14 -07:00
										 |  |  |     impl->registered_objects.insert(object); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void KernelCore::UnregisterKernelObject(KAutoObject* object) { | 
					
						
							| 
									
										
										
										
											2022-04-07 19:31:32 +01:00
										 |  |  |     std::scoped_lock lk{impl->registered_objects_lock}; | 
					
						
							| 
									
										
										
										
											2021-06-28 14:38:14 -07:00
										 |  |  |     impl->registered_objects.erase(object); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-10-25 18:55:20 +08:00
										 |  |  | void KernelCore::RegisterInUseObject(KAutoObject* object) { | 
					
						
							| 
									
										
										
										
											2022-04-07 19:31:32 +01:00
										 |  |  |     std::scoped_lock lk{impl->registered_in_use_objects_lock}; | 
					
						
							| 
									
										
										
										
											2021-10-25 18:55:20 +08:00
										 |  |  |     impl->registered_in_use_objects.insert(object); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void KernelCore::UnregisterInUseObject(KAutoObject* object) { | 
					
						
							| 
									
										
										
										
											2022-04-07 19:31:32 +01:00
										 |  |  |     std::scoped_lock lk{impl->registered_in_use_objects_lock}; | 
					
						
							| 
									
										
										
										
											2021-10-25 18:55:20 +08:00
										 |  |  |     impl->registered_in_use_objects.erase(object); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-02-18 16:26:48 -05:00
										 |  |  | void KernelCore::RunServer(std::unique_ptr<Service::ServerManager>&& server_manager) { | 
					
						
							|  |  |  |     auto* manager = server_manager.get(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         std::scoped_lock lk{impl->server_lock}; | 
					
						
							|  |  |  |         if (impl->is_shutting_down) { | 
					
						
							|  |  |  |             return; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         impl->server_managers.emplace_back(std::move(server_manager)); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     manager->LoopProcess(); | 
					
						
							| 
									
										
										
										
											2018-09-02 11:58:58 -04:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-08-28 12:30:33 -04:00
										 |  |  | u32 KernelCore::CreateNewObjectID() { | 
					
						
							|  |  |  |     return impl->next_object_id++; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-12-18 22:37:01 -05:00
										 |  |  | u64 KernelCore::CreateNewThreadID() { | 
					
						
							| 
									
										
										
										
											2018-08-28 12:30:33 -04:00
										 |  |  |     return impl->next_thread_id++; | 
					
						
							| 
									
										
										
										
											2014-05-20 18:13:25 -04:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-06-10 00:28:33 -04:00
										 |  |  | u64 KernelCore::CreateNewKernelProcessID() { | 
					
						
							|  |  |  |     return impl->next_kernel_process_id++; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | u64 KernelCore::CreateNewUserProcessID() { | 
					
						
							|  |  |  |     return impl->next_user_process_id++; | 
					
						
							| 
									
										
										
										
											2018-08-28 12:30:33 -04:00
										 |  |  | } | 
					
						
							| 
									
										
										
										
											2015-08-05 21:26:52 -03:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-02-14 09:30:53 -04:00
										 |  |  | void KernelCore::RegisterCoreThread(std::size_t core_id) { | 
					
						
							|  |  |  |     impl->RegisterCoreThread(core_id); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-11-02 20:21:32 -04:00
										 |  |  | void KernelCore::RegisterHostThread(KThread* existing_thread) { | 
					
						
							|  |  |  |     impl->RegisterHostThread(existing_thread); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (existing_thread != nullptr) { | 
					
						
							|  |  |  |         ASSERT(GetCurrentEmuThread() == existing_thread); | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2020-02-14 09:30:53 -04:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-02-18 16:26:48 -05:00
										 |  |  | static std::jthread RunHostThreadFunc(KernelCore& kernel, KProcess* process, | 
					
						
							|  |  |  |                                       std::string&& thread_name, std::function<void()>&& func) { | 
					
						
							|  |  |  |     // Reserve a new thread from the process resource limit.
 | 
					
						
							|  |  |  |     KScopedResourceReservation thread_reservation(process, LimitableResource::ThreadCountMax); | 
					
						
							|  |  |  |     ASSERT(thread_reservation.Succeeded()); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     // Initialize the thread.
 | 
					
						
							|  |  |  |     KThread* thread = KThread::Create(kernel); | 
					
						
							|  |  |  |     ASSERT(R_SUCCEEDED(KThread::InitializeDummyThread(thread, process))); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     // Commit the thread reservation.
 | 
					
						
							|  |  |  |     thread_reservation.Commit(); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-04-29 21:16:09 -04:00
										 |  |  |     // Register the thread.
 | 
					
						
							|  |  |  |     KThread::Register(kernel, thread); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-02-18 16:26:48 -05:00
										 |  |  |     return std::jthread( | 
					
						
							| 
									
										
										
										
											2023-07-18 19:31:35 -04:00
										 |  |  |         [&kernel, thread, thread_name_{std::move(thread_name)}, func_{std::move(func)}] { | 
					
						
							| 
									
										
										
										
											2023-02-18 16:26:48 -05:00
										 |  |  |             // Set the thread name.
 | 
					
						
							| 
									
										
										
										
											2023-07-18 19:31:35 -04:00
										 |  |  |             Common::SetCurrentThreadName(thread_name_.c_str()); | 
					
						
							| 
									
										
										
										
											2023-02-18 16:26:48 -05:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-04-29 21:16:09 -04:00
										 |  |  |             // Set the thread as current.
 | 
					
						
							| 
									
										
										
										
											2023-02-18 16:26:48 -05:00
										 |  |  |             kernel.RegisterHostThread(thread); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             // Run the callback.
 | 
					
						
							| 
									
										
										
										
											2023-07-18 19:31:35 -04:00
										 |  |  |             func_(); | 
					
						
							| 
									
										
										
										
											2023-02-18 16:26:48 -05:00
										 |  |  | 
 | 
					
						
							|  |  |  |             // Close the thread.
 | 
					
						
							|  |  |  |             // This will free the process if it is the last reference.
 | 
					
						
							|  |  |  |             thread->Close(); | 
					
						
							|  |  |  |         }); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | std::jthread KernelCore::RunOnHostCoreProcess(std::string&& process_name, | 
					
						
							|  |  |  |                                               std::function<void()> func) { | 
					
						
							|  |  |  |     // Make a new process.
 | 
					
						
							|  |  |  |     KProcess* process = KProcess::Create(*this); | 
					
						
							| 
									
										
										
										
											2023-10-21 16:47:43 -04:00
										 |  |  |     ASSERT(R_SUCCEEDED( | 
					
						
							|  |  |  |         process->Initialize(Svc::CreateProcessParameter{}, GetSystemResourceLimit(), false))); | 
					
						
							| 
									
										
										
										
											2023-02-18 16:26:48 -05:00
										 |  |  | 
 | 
					
						
							|  |  |  |     // Ensure that we don't hold onto any extra references.
 | 
					
						
							|  |  |  |     SCOPE_EXIT({ process->Close(); }); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-04-29 21:16:09 -04:00
										 |  |  |     // Register the new process.
 | 
					
						
							|  |  |  |     KProcess::Register(*this, process); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-02-18 16:26:48 -05:00
										 |  |  |     // Run the host thread.
 | 
					
						
							|  |  |  |     return RunHostThreadFunc(*this, process, std::move(process_name), std::move(func)); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | std::jthread KernelCore::RunOnHostCoreThread(std::string&& thread_name, | 
					
						
							|  |  |  |                                              std::function<void()> func) { | 
					
						
							|  |  |  |     // Get the current process.
 | 
					
						
							|  |  |  |     KProcess* process = GetCurrentProcessPointer(*this); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     // Run the host thread.
 | 
					
						
							|  |  |  |     return RunHostThreadFunc(*this, process, std::move(thread_name), std::move(func)); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void KernelCore::RunOnGuestCoreProcess(std::string&& process_name, std::function<void()> func) { | 
					
						
							|  |  |  |     constexpr s32 ServiceThreadPriority = 16; | 
					
						
							|  |  |  |     constexpr s32 ServiceThreadCore = 3; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     // Make a new process.
 | 
					
						
							|  |  |  |     KProcess* process = KProcess::Create(*this); | 
					
						
							| 
									
										
										
										
											2023-10-21 16:47:43 -04:00
										 |  |  |     ASSERT(R_SUCCEEDED( | 
					
						
							|  |  |  |         process->Initialize(Svc::CreateProcessParameter{}, GetSystemResourceLimit(), false))); | 
					
						
							| 
									
										
										
										
											2023-02-18 16:26:48 -05:00
										 |  |  | 
 | 
					
						
							|  |  |  |     // Ensure that we don't hold onto any extra references.
 | 
					
						
							|  |  |  |     SCOPE_EXIT({ process->Close(); }); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-04-29 21:16:09 -04:00
										 |  |  |     // Register the new process.
 | 
					
						
							|  |  |  |     KProcess::Register(*this, process); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-02-18 16:26:48 -05:00
										 |  |  |     // Reserve a new thread from the process resource limit.
 | 
					
						
							|  |  |  |     KScopedResourceReservation thread_reservation(process, LimitableResource::ThreadCountMax); | 
					
						
							|  |  |  |     ASSERT(thread_reservation.Succeeded()); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     // Initialize the thread.
 | 
					
						
							|  |  |  |     KThread* thread = KThread::Create(*this); | 
					
						
							|  |  |  |     ASSERT(R_SUCCEEDED(KThread::InitializeServiceThread( | 
					
						
							|  |  |  |         System(), thread, std::move(func), ServiceThreadPriority, ServiceThreadCore, process))); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     // Commit the thread reservation.
 | 
					
						
							|  |  |  |     thread_reservation.Commit(); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-04-29 21:16:09 -04:00
										 |  |  |     // Register the new thread.
 | 
					
						
							|  |  |  |     KThread::Register(*this, thread); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-02-18 16:26:48 -05:00
										 |  |  |     // Begin running the thread.
 | 
					
						
							|  |  |  |     ASSERT(R_SUCCEEDED(thread->Run())); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-02-22 10:27:40 -04:00
										 |  |  | u32 KernelCore::GetCurrentHostThreadID() const { | 
					
						
							|  |  |  |     return impl->GetCurrentHostThreadID(); | 
					
						
							| 
									
										
										
										
											2018-08-28 12:30:33 -04:00
										 |  |  | } | 
					
						
							| 
									
										
										
										
											2015-08-05 21:26:52 -03:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-01-21 13:00:16 -08:00
										 |  |  | KThread* KernelCore::GetCurrentEmuThread() const { | 
					
						
							|  |  |  |     return impl->GetCurrentEmuThread(); | 
					
						
							| 
									
										
										
										
											2014-05-13 21:57:12 -04:00
										 |  |  | } | 
					
						
							| 
									
										
										
										
											2014-05-22 19:06:12 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-06-16 10:35:52 -04:00
										 |  |  | void KernelCore::SetCurrentEmuThread(KThread* thread) { | 
					
						
							|  |  |  |     impl->SetCurrentEmuThread(thread); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-02-16 23:16:08 -05:00
										 |  |  | KObjectNameGlobalData& KernelCore::ObjectNameGlobalData() { | 
					
						
							|  |  |  |     return *impl->object_name_global_data; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-02-12 17:38:40 -08:00
										 |  |  | KMemoryManager& KernelCore::MemoryManager() { | 
					
						
							| 
									
										
										
										
											2020-04-08 21:06:37 -04:00
										 |  |  |     return *impl->memory_manager; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-02-12 17:38:40 -08:00
										 |  |  | const KMemoryManager& KernelCore::MemoryManager() const { | 
					
						
							| 
									
										
										
										
											2020-04-08 21:06:37 -04:00
										 |  |  |     return *impl->memory_manager; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-02-10 21:03:39 -08:00
										 |  |  | KSystemResource& KernelCore::GetAppSystemResource() { | 
					
						
							|  |  |  |     return *impl->app_system_resource; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | const KSystemResource& KernelCore::GetAppSystemResource() const { | 
					
						
							|  |  |  |     return *impl->app_system_resource; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-10-29 14:45:09 -07:00
										 |  |  | KSystemResource& KernelCore::GetSystemSystemResource() { | 
					
						
							|  |  |  |     return *impl->sys_system_resource; | 
					
						
							| 
									
										
										
										
											2022-09-16 23:33:47 -07:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-10-29 14:45:09 -07:00
										 |  |  | const KSystemResource& KernelCore::GetSystemSystemResource() const { | 
					
						
							|  |  |  |     return *impl->sys_system_resource; | 
					
						
							| 
									
										
										
										
											2022-09-16 23:33:47 -07:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-02-05 23:14:31 -08:00
										 |  |  | Kernel::KSharedMemory& KernelCore::GetFontSharedMem() { | 
					
						
							| 
									
										
										
										
											2020-04-08 21:06:37 -04:00
										 |  |  |     return *impl->font_shared_mem; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-02-05 23:14:31 -08:00
										 |  |  | const Kernel::KSharedMemory& KernelCore::GetFontSharedMem() const { | 
					
						
							| 
									
										
										
										
											2020-04-08 21:06:37 -04:00
										 |  |  |     return *impl->font_shared_mem; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-02-05 23:14:31 -08:00
										 |  |  | Kernel::KSharedMemory& KernelCore::GetIrsSharedMem() { | 
					
						
							| 
									
										
										
										
											2020-04-08 21:06:37 -04:00
										 |  |  |     return *impl->irs_shared_mem; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-02-05 23:14:31 -08:00
										 |  |  | const Kernel::KSharedMemory& KernelCore::GetIrsSharedMem() const { | 
					
						
							| 
									
										
										
										
											2020-04-08 21:06:37 -04:00
										 |  |  |     return *impl->irs_shared_mem; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-02-05 23:14:31 -08:00
										 |  |  | Kernel::KSharedMemory& KernelCore::GetTimeSharedMem() { | 
					
						
							| 
									
										
										
										
											2020-04-08 21:06:37 -04:00
										 |  |  |     return *impl->time_shared_mem; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-02-05 23:14:31 -08:00
										 |  |  | const Kernel::KSharedMemory& KernelCore::GetTimeSharedMem() const { | 
					
						
							| 
									
										
										
										
											2020-04-08 21:06:37 -04:00
										 |  |  |     return *impl->time_shared_mem; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-05-02 18:41:03 -05:00
										 |  |  | Kernel::KSharedMemory& KernelCore::GetHidBusSharedMem() { | 
					
						
							|  |  |  |     return *impl->hidbus_shared_mem; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | const Kernel::KSharedMemory& KernelCore::GetHidBusSharedMem() const { | 
					
						
							|  |  |  |     return *impl->hidbus_shared_mem; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-02-14 16:57:20 -05:00
										 |  |  | void KernelCore::SuspendEmulation(bool suspended) { | 
					
						
							| 
									
										
										
										
											2022-06-13 18:36:30 -04:00
										 |  |  |     const bool should_suspend{exception_exited || suspended}; | 
					
						
							| 
									
										
										
										
											2024-02-14 16:57:20 -05:00
										 |  |  |     auto processes = GetProcessList(); | 
					
						
							| 
									
										
										
										
											2022-06-13 18:36:30 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-02-14 16:57:20 -05:00
										 |  |  |     for (auto& process : processes) { | 
					
						
							|  |  |  |         KScopedLightLock ll{process->GetListLock()}; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         for (auto& thread : process->GetThreadList()) { | 
					
						
							|  |  |  |             if (should_suspend) { | 
					
						
							|  |  |  |                 thread.RequestSuspend(SuspendType::System); | 
					
						
							|  |  |  |             } else { | 
					
						
							|  |  |  |                 thread.Resume(SuspendType::System); | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |         } | 
					
						
							| 
									
										
										
										
											2023-01-23 14:56:06 -05:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2022-11-02 20:08:19 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-02-14 16:57:20 -05:00
										 |  |  |     if (!should_suspend) { | 
					
						
							|  |  |  |         return; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2022-11-02 20:08:19 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-01-23 14:56:06 -05:00
										 |  |  |     // Wait for process execution to stop.
 | 
					
						
							| 
									
										
										
										
											2024-02-14 16:57:20 -05:00
										 |  |  |     // KernelCore::SuspendEmulation must be called from locked context,
 | 
					
						
							|  |  |  |     // or we could race another call, interfering with waiting.
 | 
					
						
							|  |  |  |     const auto TryWait = [&]() { | 
					
						
							| 
									
										
										
										
											2023-01-23 14:56:06 -05:00
										 |  |  |         KScopedSchedulerLock sl{*this}; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-02-14 16:57:20 -05:00
										 |  |  |         for (auto& process : processes) { | 
					
						
							|  |  |  |             for (auto i = 0; i < static_cast<s32>(Core::Hardware::NUM_CPU_CORES); ++i) { | 
					
						
							|  |  |  |                 if (Scheduler(i).GetSchedulerCurrentThread()->GetOwnerProcess() == | 
					
						
							|  |  |  |                     process.GetPointerUnsafe()) { | 
					
						
							|  |  |  |                     // A thread has not finished running yet.
 | 
					
						
							|  |  |  |                     // Continue waiting.
 | 
					
						
							|  |  |  |                     return false; | 
					
						
							|  |  |  |                 } | 
					
						
							| 
									
										
										
										
											2022-06-18 16:54:33 -04:00
										 |  |  |             } | 
					
						
							|  |  |  |         } | 
					
						
							| 
									
										
										
										
											2024-02-14 16:57:20 -05:00
										 |  |  | 
 | 
					
						
							|  |  |  |         return true; | 
					
						
							|  |  |  |     }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     while (!TryWait()) { | 
					
						
							|  |  |  |         // ...
 | 
					
						
							| 
									
										
										
										
											2022-06-13 18:36:30 -04:00
										 |  |  |     } | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void KernelCore::ShutdownCores() { | 
					
						
							| 
									
										
										
										
											2024-01-02 18:29:03 -05:00
										 |  |  |     impl->TerminateAllProcesses(); | 
					
						
							| 
									
										
										
										
											2023-10-21 16:47:43 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-07-05 23:27:25 -04:00
										 |  |  |     KScopedSchedulerLock lk{*this}; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-06-13 18:36:30 -04:00
										 |  |  |     for (auto* thread : impl->shutdown_threads) { | 
					
						
							|  |  |  |         void(thread->Run()); | 
					
						
							| 
									
										
										
										
											2020-02-24 22:04:12 -04:00
										 |  |  |     } | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-03-10 13:13:39 -04:00
										 |  |  | bool KernelCore::IsMulticore() const { | 
					
						
							|  |  |  |     return impl->is_multicore; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-11-25 20:46:17 -08:00
										 |  |  | bool KernelCore::IsShuttingDown() const { | 
					
						
							|  |  |  |     return impl->IsShuttingDown(); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-02-13 11:21:43 -05:00
										 |  |  | void KernelCore::ExceptionalExitApplication() { | 
					
						
							| 
									
										
										
										
											2020-02-24 22:04:12 -04:00
										 |  |  |     exception_exited = true; | 
					
						
							| 
									
										
										
										
											2024-02-14 16:57:20 -05:00
										 |  |  |     SuspendEmulation(true); | 
					
						
							| 
									
										
										
										
											2020-02-24 22:04:12 -04:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-03-12 16:48:43 -04:00
										 |  |  | void KernelCore::EnterSVCProfile() { | 
					
						
							| 
									
										
										
										
											2021-08-06 22:58:46 -07:00
										 |  |  |     impl->svc_ticks[CurrentPhysicalCoreIndex()] = MicroProfileEnter(MICROPROFILE_TOKEN(Kernel_SVC)); | 
					
						
							| 
									
										
										
										
											2020-03-12 16:48:43 -04:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void KernelCore::ExitSVCProfile() { | 
					
						
							| 
									
										
										
										
											2021-08-06 22:58:46 -07:00
										 |  |  |     MicroProfileLeave(MICROPROFILE_TOKEN(Kernel_SVC), impl->svc_ticks[CurrentPhysicalCoreIndex()]); | 
					
						
							| 
									
										
										
										
											2020-03-12 16:48:43 -04:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-05-04 21:35:42 -07:00
										 |  |  | Init::KSlabResourceCounts& KernelCore::SlabResourceCounts() { | 
					
						
							|  |  |  |     return impl->slab_resource_counts; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | const Init::KSlabResourceCounts& KernelCore::SlabResourceCounts() const { | 
					
						
							|  |  |  |     return impl->slab_resource_counts; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-01-14 16:25:37 -08:00
										 |  |  | KWorkerTaskManager& KernelCore::WorkerTaskManager() { | 
					
						
							|  |  |  |     return impl->worker_task_manager; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | const KWorkerTaskManager& KernelCore::WorkerTaskManager() const { | 
					
						
							|  |  |  |     return impl->worker_task_manager; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-02-26 10:46:31 -08:00
										 |  |  | const KMemoryLayout& KernelCore::MemoryLayout() const { | 
					
						
							| 
									
										
										
										
											2022-02-27 11:05:07 -08:00
										 |  |  |     return *impl->memory_layout; | 
					
						
							| 
									
										
										
										
											2022-02-26 10:46:31 -08:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-12-31 02:13:02 -08:00
										 |  |  | bool KernelCore::IsPhantomModeForSingleCore() const { | 
					
						
							|  |  |  |     return impl->IsPhantomModeForSingleCore(); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void KernelCore::SetIsPhantomModeForSingleCore(bool value) { | 
					
						
							|  |  |  |     impl->SetIsPhantomModeForSingleCore(value); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-04-03 19:11:46 -07:00
										 |  |  | Core::System& KernelCore::System() { | 
					
						
							|  |  |  |     return impl->system; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | const Core::System& KernelCore::System() const { | 
					
						
							|  |  |  |     return impl->system; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-02-23 15:49:42 -05:00
										 |  |  | struct KernelCore::SlabHeapContainer { | 
					
						
							|  |  |  |     KSlabHeap<KClientSession> client_session; | 
					
						
							|  |  |  |     KSlabHeap<KEvent> event; | 
					
						
							|  |  |  |     KSlabHeap<KPort> port; | 
					
						
							|  |  |  |     KSlabHeap<KProcess> process; | 
					
						
							|  |  |  |     KSlabHeap<KResourceLimit> resource_limit; | 
					
						
							|  |  |  |     KSlabHeap<KSession> session; | 
					
						
							| 
									
										
										
										
											2023-12-06 19:54:52 -05:00
										 |  |  |     KSlabHeap<KLightSession> light_session; | 
					
						
							| 
									
										
										
										
											2023-02-23 15:49:42 -05:00
										 |  |  |     KSlabHeap<KSharedMemory> shared_memory; | 
					
						
							|  |  |  |     KSlabHeap<KSharedMemoryInfo> shared_memory_info; | 
					
						
							|  |  |  |     KSlabHeap<KThread> thread; | 
					
						
							|  |  |  |     KSlabHeap<KTransferMemory> transfer_memory; | 
					
						
							|  |  |  |     KSlabHeap<KCodeMemory> code_memory; | 
					
						
							|  |  |  |     KSlabHeap<KDeviceAddressSpace> device_address_space; | 
					
						
							|  |  |  |     KSlabHeap<KPageBuffer> page_buffer; | 
					
						
							|  |  |  |     KSlabHeap<KThreadLocalPage> thread_local_page; | 
					
						
							|  |  |  |     KSlabHeap<KObjectName> object_name; | 
					
						
							|  |  |  |     KSlabHeap<KSessionRequest> session_request; | 
					
						
							|  |  |  |     KSlabHeap<KSecureSystemResource> secure_system_resource; | 
					
						
							|  |  |  |     KSlabHeap<KThread::LockWithPriorityInheritanceInfo> lock_info; | 
					
						
							|  |  |  |     KSlabHeap<KEventInfo> event_info; | 
					
						
							|  |  |  |     KSlabHeap<KDebug> debug; | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | template <typename T> | 
					
						
							|  |  |  | KSlabHeap<T>& KernelCore::SlabHeap() { | 
					
						
							|  |  |  |     if constexpr (std::is_same_v<T, KClientSession>) { | 
					
						
							|  |  |  |         return slab_heap_container->client_session; | 
					
						
							|  |  |  |     } else if constexpr (std::is_same_v<T, KEvent>) { | 
					
						
							|  |  |  |         return slab_heap_container->event; | 
					
						
							|  |  |  |     } else if constexpr (std::is_same_v<T, KPort>) { | 
					
						
							|  |  |  |         return slab_heap_container->port; | 
					
						
							|  |  |  |     } else if constexpr (std::is_same_v<T, KProcess>) { | 
					
						
							|  |  |  |         return slab_heap_container->process; | 
					
						
							|  |  |  |     } else if constexpr (std::is_same_v<T, KResourceLimit>) { | 
					
						
							|  |  |  |         return slab_heap_container->resource_limit; | 
					
						
							|  |  |  |     } else if constexpr (std::is_same_v<T, KSession>) { | 
					
						
							|  |  |  |         return slab_heap_container->session; | 
					
						
							| 
									
										
										
										
											2023-12-06 19:54:52 -05:00
										 |  |  |     } else if constexpr (std::is_same_v<T, KLightSession>) { | 
					
						
							|  |  |  |         return slab_heap_container->light_session; | 
					
						
							| 
									
										
										
										
											2023-02-23 15:49:42 -05:00
										 |  |  |     } else if constexpr (std::is_same_v<T, KSharedMemory>) { | 
					
						
							|  |  |  |         return slab_heap_container->shared_memory; | 
					
						
							|  |  |  |     } else if constexpr (std::is_same_v<T, KSharedMemoryInfo>) { | 
					
						
							|  |  |  |         return slab_heap_container->shared_memory_info; | 
					
						
							|  |  |  |     } else if constexpr (std::is_same_v<T, KThread>) { | 
					
						
							|  |  |  |         return slab_heap_container->thread; | 
					
						
							|  |  |  |     } else if constexpr (std::is_same_v<T, KTransferMemory>) { | 
					
						
							|  |  |  |         return slab_heap_container->transfer_memory; | 
					
						
							|  |  |  |     } else if constexpr (std::is_same_v<T, KCodeMemory>) { | 
					
						
							|  |  |  |         return slab_heap_container->code_memory; | 
					
						
							|  |  |  |     } else if constexpr (std::is_same_v<T, KDeviceAddressSpace>) { | 
					
						
							|  |  |  |         return slab_heap_container->device_address_space; | 
					
						
							|  |  |  |     } else if constexpr (std::is_same_v<T, KPageBuffer>) { | 
					
						
							|  |  |  |         return slab_heap_container->page_buffer; | 
					
						
							|  |  |  |     } else if constexpr (std::is_same_v<T, KThreadLocalPage>) { | 
					
						
							|  |  |  |         return slab_heap_container->thread_local_page; | 
					
						
							|  |  |  |     } else if constexpr (std::is_same_v<T, KObjectName>) { | 
					
						
							|  |  |  |         return slab_heap_container->object_name; | 
					
						
							|  |  |  |     } else if constexpr (std::is_same_v<T, KSessionRequest>) { | 
					
						
							|  |  |  |         return slab_heap_container->session_request; | 
					
						
							|  |  |  |     } else if constexpr (std::is_same_v<T, KSecureSystemResource>) { | 
					
						
							|  |  |  |         return slab_heap_container->secure_system_resource; | 
					
						
							|  |  |  |     } else if constexpr (std::is_same_v<T, KThread::LockWithPriorityInheritanceInfo>) { | 
					
						
							|  |  |  |         return slab_heap_container->lock_info; | 
					
						
							|  |  |  |     } else if constexpr (std::is_same_v<T, KEventInfo>) { | 
					
						
							|  |  |  |         return slab_heap_container->event_info; | 
					
						
							|  |  |  |     } else if constexpr (std::is_same_v<T, KDebug>) { | 
					
						
							|  |  |  |         return slab_heap_container->debug; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | template KSlabHeap<KClientSession>& KernelCore::SlabHeap(); | 
					
						
							|  |  |  | template KSlabHeap<KEvent>& KernelCore::SlabHeap(); | 
					
						
							|  |  |  | template KSlabHeap<KPort>& KernelCore::SlabHeap(); | 
					
						
							|  |  |  | template KSlabHeap<KProcess>& KernelCore::SlabHeap(); | 
					
						
							|  |  |  | template KSlabHeap<KResourceLimit>& KernelCore::SlabHeap(); | 
					
						
							|  |  |  | template KSlabHeap<KSession>& KernelCore::SlabHeap(); | 
					
						
							| 
									
										
										
										
											2023-12-06 19:54:52 -05:00
										 |  |  | template KSlabHeap<KLightSession>& KernelCore::SlabHeap(); | 
					
						
							| 
									
										
										
										
											2023-02-23 15:49:42 -05:00
										 |  |  | template KSlabHeap<KSharedMemory>& KernelCore::SlabHeap(); | 
					
						
							|  |  |  | template KSlabHeap<KSharedMemoryInfo>& KernelCore::SlabHeap(); | 
					
						
							|  |  |  | template KSlabHeap<KThread>& KernelCore::SlabHeap(); | 
					
						
							|  |  |  | template KSlabHeap<KTransferMemory>& KernelCore::SlabHeap(); | 
					
						
							|  |  |  | template KSlabHeap<KCodeMemory>& KernelCore::SlabHeap(); | 
					
						
							|  |  |  | template KSlabHeap<KDeviceAddressSpace>& KernelCore::SlabHeap(); | 
					
						
							|  |  |  | template KSlabHeap<KPageBuffer>& KernelCore::SlabHeap(); | 
					
						
							|  |  |  | template KSlabHeap<KThreadLocalPage>& KernelCore::SlabHeap(); | 
					
						
							|  |  |  | template KSlabHeap<KObjectName>& KernelCore::SlabHeap(); | 
					
						
							|  |  |  | template KSlabHeap<KSessionRequest>& KernelCore::SlabHeap(); | 
					
						
							|  |  |  | template KSlabHeap<KSecureSystemResource>& KernelCore::SlabHeap(); | 
					
						
							|  |  |  | template KSlabHeap<KThread::LockWithPriorityInheritanceInfo>& KernelCore::SlabHeap(); | 
					
						
							|  |  |  | template KSlabHeap<KEventInfo>& KernelCore::SlabHeap(); | 
					
						
							|  |  |  | template KSlabHeap<KDebug>& KernelCore::SlabHeap(); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-01-01 13:25:37 -05:00
										 |  |  | } // namespace Kernel
 |