| 
									
										
										
										
											2018-12-19 12:57:47 -05:00
										 |  |  | // Copyright 2018 yuzu emulator team
 | 
					
						
							|  |  |  | // Licensed under GPLv2 or any later version
 | 
					
						
							|  |  |  | // Refer to the license.txt file included.
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-01-15 02:02:57 -05:00
										 |  |  | #include <bit>
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-12-19 12:57:47 -05:00
										 |  |  | #include "common/bit_util.h"
 | 
					
						
							| 
									
										
										
										
											2020-04-29 14:53:53 +10:00
										 |  |  | #include "common/logging/log.h"
 | 
					
						
							| 
									
										
										
										
											2018-12-19 12:57:47 -05:00
										 |  |  | #include "core/hle/kernel/handle_table.h"
 | 
					
						
							| 
									
										
										
										
											2021-02-12 17:58:31 -08:00
										 |  |  | #include "core/hle/kernel/k_page_table.h"
 | 
					
						
							| 
									
										
										
										
											2018-12-19 12:57:47 -05:00
										 |  |  | #include "core/hle/kernel/process_capability.h"
 | 
					
						
							| 
									
										
										
										
											2021-02-13 10:43:01 +11:00
										 |  |  | #include "core/hle/kernel/svc_results.h"
 | 
					
						
							| 
									
										
										
										
											2018-12-19 12:57:47 -05:00
										 |  |  | 
 | 
					
						
							|  |  |  | namespace Kernel { | 
					
						
							|  |  |  | namespace { | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | // clang-format off
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | // Shift offsets for kernel capability types.
 | 
					
						
							|  |  |  | enum : u32 { | 
					
						
							|  |  |  |     CapabilityOffset_PriorityAndCoreNum = 3, | 
					
						
							|  |  |  |     CapabilityOffset_Syscall            = 4, | 
					
						
							|  |  |  |     CapabilityOffset_MapPhysical        = 6, | 
					
						
							|  |  |  |     CapabilityOffset_MapIO              = 7, | 
					
						
							|  |  |  |     CapabilityOffset_Interrupt          = 11, | 
					
						
							|  |  |  |     CapabilityOffset_ProgramType        = 13, | 
					
						
							|  |  |  |     CapabilityOffset_KernelVersion      = 14, | 
					
						
							|  |  |  |     CapabilityOffset_HandleTableSize    = 15, | 
					
						
							|  |  |  |     CapabilityOffset_Debug              = 16, | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | // Combined mask of all parameters that may be initialized only once.
 | 
					
						
							|  |  |  | constexpr u32 InitializeOnceMask = (1U << CapabilityOffset_PriorityAndCoreNum) | | 
					
						
							|  |  |  |                                    (1U << CapabilityOffset_ProgramType) | | 
					
						
							|  |  |  |                                    (1U << CapabilityOffset_KernelVersion) | | 
					
						
							|  |  |  |                                    (1U << CapabilityOffset_HandleTableSize) | | 
					
						
							|  |  |  |                                    (1U << CapabilityOffset_Debug); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | // Packed kernel version indicating 10.4.0
 | 
					
						
							|  |  |  | constexpr u32 PackedKernelVersion = 0x520000; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | // Indicates possible types of capabilities that can be specified.
 | 
					
						
							|  |  |  | enum class CapabilityType : u32 { | 
					
						
							|  |  |  |     Unset              = 0U, | 
					
						
							|  |  |  |     PriorityAndCoreNum = (1U << CapabilityOffset_PriorityAndCoreNum) - 1, | 
					
						
							|  |  |  |     Syscall            = (1U << CapabilityOffset_Syscall) - 1, | 
					
						
							|  |  |  |     MapPhysical        = (1U << CapabilityOffset_MapPhysical) - 1, | 
					
						
							|  |  |  |     MapIO              = (1U << CapabilityOffset_MapIO) - 1, | 
					
						
							|  |  |  |     Interrupt          = (1U << CapabilityOffset_Interrupt) - 1, | 
					
						
							|  |  |  |     ProgramType        = (1U << CapabilityOffset_ProgramType) - 1, | 
					
						
							|  |  |  |     KernelVersion      = (1U << CapabilityOffset_KernelVersion) - 1, | 
					
						
							|  |  |  |     HandleTableSize    = (1U << CapabilityOffset_HandleTableSize) - 1, | 
					
						
							|  |  |  |     Debug              = (1U << CapabilityOffset_Debug) - 1, | 
					
						
							|  |  |  |     Ignorable          = 0xFFFFFFFFU, | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | // clang-format on
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | constexpr CapabilityType GetCapabilityType(u32 value) { | 
					
						
							|  |  |  |     return static_cast<CapabilityType>((~value & (value + 1)) - 1); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | u32 GetFlagBitOffset(CapabilityType type) { | 
					
						
							|  |  |  |     const auto value = static_cast<u32>(type); | 
					
						
							| 
									
										
										
										
											2021-01-15 02:02:57 -05:00
										 |  |  |     return static_cast<u32>(Common::BitSize<u32>() - static_cast<u32>(std::countl_zero(value))); | 
					
						
							| 
									
										
										
										
											2018-12-19 12:57:47 -05:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | } // Anonymous namespace
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | ResultCode ProcessCapabilities::InitializeForKernelProcess(const u32* capabilities, | 
					
						
							|  |  |  |                                                            std::size_t num_capabilities, | 
					
						
							| 
									
										
										
										
											2021-02-12 17:58:31 -08:00
										 |  |  |                                                            KPageTable& page_table) { | 
					
						
							| 
									
										
										
										
											2018-12-19 12:57:47 -05:00
										 |  |  |     Clear(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     // Allow all cores and priorities.
 | 
					
						
							|  |  |  |     core_mask = 0xF; | 
					
						
							|  |  |  |     priority_mask = 0xFFFFFFFFFFFFFFFF; | 
					
						
							|  |  |  |     kernel_version = PackedKernelVersion; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-04-08 17:30:34 -04:00
										 |  |  |     return ParseCapabilities(capabilities, num_capabilities, page_table); | 
					
						
							| 
									
										
										
										
											2018-12-19 12:57:47 -05:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | ResultCode ProcessCapabilities::InitializeForUserProcess(const u32* capabilities, | 
					
						
							|  |  |  |                                                          std::size_t num_capabilities, | 
					
						
							| 
									
										
										
										
											2021-02-12 17:58:31 -08:00
										 |  |  |                                                          KPageTable& page_table) { | 
					
						
							| 
									
										
										
										
											2018-12-19 12:57:47 -05:00
										 |  |  |     Clear(); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-04-08 17:30:34 -04:00
										 |  |  |     return ParseCapabilities(capabilities, num_capabilities, page_table); | 
					
						
							| 
									
										
										
										
											2018-12-19 12:57:47 -05:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void ProcessCapabilities::InitializeForMetadatalessProcess() { | 
					
						
							|  |  |  |     // Allow all cores and priorities
 | 
					
						
							|  |  |  |     core_mask = 0xF; | 
					
						
							|  |  |  |     priority_mask = 0xFFFFFFFFFFFFFFFF; | 
					
						
							|  |  |  |     kernel_version = PackedKernelVersion; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     // Allow all system calls and interrupts.
 | 
					
						
							|  |  |  |     svc_capabilities.set(); | 
					
						
							|  |  |  |     interrupt_capabilities.set(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     // Allow using the maximum possible amount of handles
 | 
					
						
							| 
									
										
										
										
											2019-02-25 10:13:52 -05:00
										 |  |  |     handle_table_size = static_cast<s32>(HandleTable::MAX_COUNT); | 
					
						
							| 
									
										
										
										
											2018-12-19 12:57:47 -05:00
										 |  |  | 
 | 
					
						
							|  |  |  |     // Allow all debugging capabilities.
 | 
					
						
							|  |  |  |     is_debuggable = true; | 
					
						
							|  |  |  |     can_force_debug = true; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | ResultCode ProcessCapabilities::ParseCapabilities(const u32* capabilities, | 
					
						
							|  |  |  |                                                   std::size_t num_capabilities, | 
					
						
							| 
									
										
										
										
											2021-02-12 17:58:31 -08:00
										 |  |  |                                                   KPageTable& page_table) { | 
					
						
							| 
									
										
										
										
											2018-12-19 12:57:47 -05:00
										 |  |  |     u32 set_flags = 0; | 
					
						
							|  |  |  |     u32 set_svc_bits = 0; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     for (std::size_t i = 0; i < num_capabilities; ++i) { | 
					
						
							|  |  |  |         const u32 descriptor = capabilities[i]; | 
					
						
							|  |  |  |         const auto type = GetCapabilityType(descriptor); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         if (type == CapabilityType::MapPhysical) { | 
					
						
							|  |  |  |             i++; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             // The MapPhysical type uses two descriptor flags for its parameters.
 | 
					
						
							|  |  |  |             // If there's only one, then there's a problem.
 | 
					
						
							|  |  |  |             if (i >= num_capabilities) { | 
					
						
							| 
									
										
										
										
											2020-04-29 14:53:53 +10:00
										 |  |  |                 LOG_ERROR(Kernel, "Invalid combination! i={}", i); | 
					
						
							| 
									
										
										
										
											2021-02-13 10:43:01 +11:00
										 |  |  |                 return ResultInvalidCombination; | 
					
						
							| 
									
										
										
										
											2018-12-19 12:57:47 -05:00
										 |  |  |             } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             const auto size_flags = capabilities[i]; | 
					
						
							|  |  |  |             if (GetCapabilityType(size_flags) != CapabilityType::MapPhysical) { | 
					
						
							| 
									
										
										
										
											2020-04-29 14:53:53 +10:00
										 |  |  |                 LOG_ERROR(Kernel, "Invalid capability type! size_flags={}", size_flags); | 
					
						
							| 
									
										
										
										
											2021-02-13 10:43:01 +11:00
										 |  |  |                 return ResultInvalidCombination; | 
					
						
							| 
									
										
										
										
											2018-12-19 12:57:47 -05:00
										 |  |  |             } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-04-08 17:30:34 -04:00
										 |  |  |             const auto result = HandleMapPhysicalFlags(descriptor, size_flags, page_table); | 
					
						
							| 
									
										
										
										
											2018-12-19 12:57:47 -05:00
										 |  |  |             if (result.IsError()) { | 
					
						
							| 
									
										
										
										
											2020-04-29 14:53:53 +10:00
										 |  |  |                 LOG_ERROR(Kernel, "Failed to map physical flags! descriptor={}, size_flags={}", | 
					
						
							|  |  |  |                           descriptor, size_flags); | 
					
						
							| 
									
										
										
										
											2018-12-19 12:57:47 -05:00
										 |  |  |                 return result; | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |         } else { | 
					
						
							|  |  |  |             const auto result = | 
					
						
							| 
									
										
										
										
											2020-04-08 17:30:34 -04:00
										 |  |  |                 ParseSingleFlagCapability(set_flags, set_svc_bits, descriptor, page_table); | 
					
						
							| 
									
										
										
										
											2018-12-19 12:57:47 -05:00
										 |  |  |             if (result.IsError()) { | 
					
						
							| 
									
										
										
										
											2020-04-29 14:53:53 +10:00
										 |  |  |                 LOG_ERROR( | 
					
						
							|  |  |  |                     Kernel, | 
					
						
							|  |  |  |                     "Failed to parse capability flag! set_flags={}, set_svc_bits={}, descriptor={}", | 
					
						
							|  |  |  |                     set_flags, set_svc_bits, descriptor); | 
					
						
							| 
									
										
										
										
											2018-12-19 12:57:47 -05:00
										 |  |  |                 return result; | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     return RESULT_SUCCESS; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | ResultCode ProcessCapabilities::ParseSingleFlagCapability(u32& set_flags, u32& set_svc_bits, | 
					
						
							| 
									
										
										
										
											2021-02-12 17:58:31 -08:00
										 |  |  |                                                           u32 flag, KPageTable& page_table) { | 
					
						
							| 
									
										
										
										
											2018-12-19 12:57:47 -05:00
										 |  |  |     const auto type = GetCapabilityType(flag); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (type == CapabilityType::Unset) { | 
					
						
							| 
									
										
										
										
											2021-04-11 11:41:48 -07:00
										 |  |  |         return ResultInvalidArgument; | 
					
						
							| 
									
										
										
										
											2018-12-19 12:57:47 -05:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     // Bail early on ignorable entries, as one would expect,
 | 
					
						
							|  |  |  |     // ignorable descriptors can be ignored.
 | 
					
						
							|  |  |  |     if (type == CapabilityType::Ignorable) { | 
					
						
							|  |  |  |         return RESULT_SUCCESS; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     // Ensure that the give flag hasn't already been initialized before.
 | 
					
						
							|  |  |  |     // If it has been, then bail.
 | 
					
						
							|  |  |  |     const u32 flag_length = GetFlagBitOffset(type); | 
					
						
							|  |  |  |     const u32 set_flag = 1U << flag_length; | 
					
						
							|  |  |  |     if ((set_flag & set_flags & InitializeOnceMask) != 0) { | 
					
						
							| 
									
										
										
										
											2020-04-29 14:53:53 +10:00
										 |  |  |         LOG_ERROR(Kernel, | 
					
						
							|  |  |  |                   "Attempted to initialize flags that may only be initialized once. set_flags={}", | 
					
						
							|  |  |  |                   set_flags); | 
					
						
							| 
									
										
										
										
											2021-02-13 10:43:01 +11:00
										 |  |  |         return ResultInvalidCombination; | 
					
						
							| 
									
										
										
										
											2018-12-19 12:57:47 -05:00
										 |  |  |     } | 
					
						
							|  |  |  |     set_flags |= set_flag; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     switch (type) { | 
					
						
							|  |  |  |     case CapabilityType::PriorityAndCoreNum: | 
					
						
							|  |  |  |         return HandlePriorityCoreNumFlags(flag); | 
					
						
							|  |  |  |     case CapabilityType::Syscall: | 
					
						
							|  |  |  |         return HandleSyscallFlags(set_svc_bits, flag); | 
					
						
							|  |  |  |     case CapabilityType::MapIO: | 
					
						
							| 
									
										
										
										
											2020-04-08 17:30:34 -04:00
										 |  |  |         return HandleMapIOFlags(flag, page_table); | 
					
						
							| 
									
										
										
										
											2018-12-19 12:57:47 -05:00
										 |  |  |     case CapabilityType::Interrupt: | 
					
						
							|  |  |  |         return HandleInterruptFlags(flag); | 
					
						
							|  |  |  |     case CapabilityType::ProgramType: | 
					
						
							|  |  |  |         return HandleProgramTypeFlags(flag); | 
					
						
							|  |  |  |     case CapabilityType::KernelVersion: | 
					
						
							|  |  |  |         return HandleKernelVersionFlags(flag); | 
					
						
							|  |  |  |     case CapabilityType::HandleTableSize: | 
					
						
							|  |  |  |         return HandleHandleTableFlags(flag); | 
					
						
							|  |  |  |     case CapabilityType::Debug: | 
					
						
							|  |  |  |         return HandleDebugFlags(flag); | 
					
						
							|  |  |  |     default: | 
					
						
							|  |  |  |         break; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-12-07 22:00:34 -05:00
										 |  |  |     LOG_ERROR(Kernel, "Invalid capability type! type={}", type); | 
					
						
							| 
									
										
										
										
											2021-04-11 11:41:48 -07:00
										 |  |  |     return ResultInvalidArgument; | 
					
						
							| 
									
										
										
										
											2018-12-19 12:57:47 -05:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void ProcessCapabilities::Clear() { | 
					
						
							|  |  |  |     svc_capabilities.reset(); | 
					
						
							|  |  |  |     interrupt_capabilities.reset(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     core_mask = 0; | 
					
						
							|  |  |  |     priority_mask = 0; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     handle_table_size = 0; | 
					
						
							|  |  |  |     kernel_version = 0; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-12-19 21:14:47 -05:00
										 |  |  |     program_type = ProgramType::SysModule; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-12-19 12:57:47 -05:00
										 |  |  |     is_debuggable = false; | 
					
						
							|  |  |  |     can_force_debug = false; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | ResultCode ProcessCapabilities::HandlePriorityCoreNumFlags(u32 flags) { | 
					
						
							| 
									
										
										
										
											2018-12-19 19:09:18 -05:00
										 |  |  |     if (priority_mask != 0 || core_mask != 0) { | 
					
						
							| 
									
										
										
										
											2020-04-29 14:53:53 +10:00
										 |  |  |         LOG_ERROR(Kernel, "Core or priority mask are not zero! priority_mask={}, core_mask={}", | 
					
						
							|  |  |  |                   priority_mask, core_mask); | 
					
						
							| 
									
										
										
										
											2021-04-11 11:41:48 -07:00
										 |  |  |         return ResultInvalidArgument; | 
					
						
							| 
									
										
										
										
											2018-12-19 19:09:18 -05:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     const u32 core_num_min = (flags >> 16) & 0xFF; | 
					
						
							|  |  |  |     const u32 core_num_max = (flags >> 24) & 0xFF; | 
					
						
							|  |  |  |     if (core_num_min > core_num_max) { | 
					
						
							| 
									
										
										
										
											2020-04-29 14:53:53 +10:00
										 |  |  |         LOG_ERROR(Kernel, "Core min is greater than core max! core_num_min={}, core_num_max={}", | 
					
						
							|  |  |  |                   core_num_min, core_num_max); | 
					
						
							| 
									
										
										
										
											2021-02-13 10:43:01 +11:00
										 |  |  |         return ResultInvalidCombination; | 
					
						
							| 
									
										
										
										
											2018-12-19 19:09:18 -05:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     const u32 priority_min = (flags >> 10) & 0x3F; | 
					
						
							|  |  |  |     const u32 priority_max = (flags >> 4) & 0x3F; | 
					
						
							|  |  |  |     if (priority_min > priority_max) { | 
					
						
							| 
									
										
										
										
											2020-04-29 14:53:53 +10:00
										 |  |  |         LOG_ERROR(Kernel, | 
					
						
							|  |  |  |                   "Priority min is greater than priority max! priority_min={}, priority_max={}", | 
					
						
							|  |  |  |                   core_num_min, priority_max); | 
					
						
							| 
									
										
										
										
											2021-02-13 10:43:01 +11:00
										 |  |  |         return ResultInvalidCombination; | 
					
						
							| 
									
										
										
										
											2018-12-19 19:09:18 -05:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     // The switch only has 4 usable cores.
 | 
					
						
							|  |  |  |     if (core_num_max >= 4) { | 
					
						
							| 
									
										
										
										
											2020-04-29 14:53:53 +10:00
										 |  |  |         LOG_ERROR(Kernel, "Invalid max cores specified! core_num_max={}", core_num_max); | 
					
						
							| 
									
										
										
										
											2021-02-13 10:43:01 +11:00
										 |  |  |         return ResultInvalidCoreId; | 
					
						
							| 
									
										
										
										
											2018-12-19 19:09:18 -05:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     const auto make_mask = [](u64 min, u64 max) { | 
					
						
							|  |  |  |         const u64 range = max - min + 1; | 
					
						
							|  |  |  |         const u64 mask = (1ULL << range) - 1; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         return mask << min; | 
					
						
							|  |  |  |     }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     core_mask = make_mask(core_num_min, core_num_max); | 
					
						
							|  |  |  |     priority_mask = make_mask(priority_min, priority_max); | 
					
						
							| 
									
										
										
										
											2018-12-19 12:57:47 -05:00
										 |  |  |     return RESULT_SUCCESS; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | ResultCode ProcessCapabilities::HandleSyscallFlags(u32& set_svc_bits, u32 flags) { | 
					
						
							| 
									
										
										
										
											2018-12-20 22:54:05 -05:00
										 |  |  |     const u32 index = flags >> 29; | 
					
						
							|  |  |  |     const u32 svc_bit = 1U << index; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     // If we've already set this svc before, bail.
 | 
					
						
							|  |  |  |     if ((set_svc_bits & svc_bit) != 0) { | 
					
						
							| 
									
										
										
										
											2021-02-13 10:43:01 +11:00
										 |  |  |         return ResultInvalidCombination; | 
					
						
							| 
									
										
										
										
											2018-12-20 22:54:05 -05:00
										 |  |  |     } | 
					
						
							|  |  |  |     set_svc_bits |= svc_bit; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     const u32 svc_mask = (flags >> 5) & 0xFFFFFF; | 
					
						
							|  |  |  |     for (u32 i = 0; i < 24; ++i) { | 
					
						
							|  |  |  |         const u32 svc_number = index * 24 + i; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         if ((svc_mask & (1U << i)) == 0) { | 
					
						
							|  |  |  |             continue; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         svc_capabilities[svc_number] = true; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-12-19 12:57:47 -05:00
										 |  |  |     return RESULT_SUCCESS; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | ResultCode ProcessCapabilities::HandleMapPhysicalFlags(u32 flags, u32 size_flags, | 
					
						
							| 
									
										
										
										
											2021-02-12 17:58:31 -08:00
										 |  |  |                                                        KPageTable& page_table) { | 
					
						
							| 
									
										
										
										
											2018-12-19 12:57:47 -05:00
										 |  |  |     // TODO(Lioncache): Implement once the memory manager can handle this.
 | 
					
						
							|  |  |  |     return RESULT_SUCCESS; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-02-12 17:58:31 -08:00
										 |  |  | ResultCode ProcessCapabilities::HandleMapIOFlags(u32 flags, KPageTable& page_table) { | 
					
						
							| 
									
										
										
										
											2018-12-19 12:57:47 -05:00
										 |  |  |     // TODO(Lioncache): Implement once the memory manager can handle this.
 | 
					
						
							|  |  |  |     return RESULT_SUCCESS; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | ResultCode ProcessCapabilities::HandleInterruptFlags(u32 flags) { | 
					
						
							| 
									
										
										
										
											2018-12-19 20:38:29 -05:00
										 |  |  |     constexpr u32 interrupt_ignore_value = 0x3FF; | 
					
						
							|  |  |  |     const u32 interrupt0 = (flags >> 12) & 0x3FF; | 
					
						
							|  |  |  |     const u32 interrupt1 = (flags >> 22) & 0x3FF; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     for (u32 interrupt : {interrupt0, interrupt1}) { | 
					
						
							|  |  |  |         if (interrupt == interrupt_ignore_value) { | 
					
						
							|  |  |  |             continue; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         // NOTE:
 | 
					
						
							|  |  |  |         // This should be checking a generic interrupt controller value
 | 
					
						
							|  |  |  |         // as part of the calculation, however, given we don't currently
 | 
					
						
							|  |  |  |         // emulate that, it's sufficient to mark every interrupt as defined.
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         if (interrupt >= interrupt_capabilities.size()) { | 
					
						
							| 
									
										
										
										
											2020-04-29 14:53:53 +10:00
										 |  |  |             LOG_ERROR(Kernel, "Process interrupt capability is out of range! svc_number={}", | 
					
						
							|  |  |  |                       interrupt); | 
					
						
							| 
									
										
										
										
											2021-02-13 10:43:01 +11:00
										 |  |  |             return ResultOutOfRange; | 
					
						
							| 
									
										
										
										
											2018-12-19 20:38:29 -05:00
										 |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         interrupt_capabilities[interrupt] = true; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-12-19 12:57:47 -05:00
										 |  |  |     return RESULT_SUCCESS; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | ResultCode ProcessCapabilities::HandleProgramTypeFlags(u32 flags) { | 
					
						
							| 
									
										
										
										
											2018-12-19 21:14:47 -05:00
										 |  |  |     const u32 reserved = flags >> 17; | 
					
						
							|  |  |  |     if (reserved != 0) { | 
					
						
							| 
									
										
										
										
											2020-04-29 14:53:53 +10:00
										 |  |  |         LOG_ERROR(Kernel, "Reserved value is non-zero! reserved={}", reserved); | 
					
						
							| 
									
										
										
										
											2021-04-11 11:41:48 -07:00
										 |  |  |         return ResultReservedUsed; | 
					
						
							| 
									
										
										
										
											2018-12-19 21:14:47 -05:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     program_type = static_cast<ProgramType>((flags >> 14) & 0b111); | 
					
						
							| 
									
										
										
										
											2018-12-19 12:57:47 -05:00
										 |  |  |     return RESULT_SUCCESS; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | ResultCode ProcessCapabilities::HandleKernelVersionFlags(u32 flags) { | 
					
						
							| 
									
										
										
										
											2018-12-19 21:28:44 -05:00
										 |  |  |     // Yes, the internal member variable is checked in the actual kernel here.
 | 
					
						
							|  |  |  |     // This might look odd for options that are only allowed to be initialized
 | 
					
						
							|  |  |  |     // just once, however the kernel has a separate initialization function for
 | 
					
						
							|  |  |  |     // kernel processes and userland processes. The kernel variant sets this
 | 
					
						
							|  |  |  |     // member variable ahead of time.
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     const u32 major_version = kernel_version >> 19; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (major_version != 0 || flags < 0x80000) { | 
					
						
							| 
									
										
										
										
											2020-04-29 14:53:53 +10:00
										 |  |  |         LOG_ERROR(Kernel, | 
					
						
							|  |  |  |                   "Kernel version is non zero or flags are too small! major_version={}, flags={}", | 
					
						
							|  |  |  |                   major_version, flags); | 
					
						
							| 
									
										
										
										
											2021-04-11 11:41:48 -07:00
										 |  |  |         return ResultInvalidArgument; | 
					
						
							| 
									
										
										
										
											2018-12-19 21:28:44 -05:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     kernel_version = flags; | 
					
						
							| 
									
										
										
										
											2018-12-19 12:57:47 -05:00
										 |  |  |     return RESULT_SUCCESS; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | ResultCode ProcessCapabilities::HandleHandleTableFlags(u32 flags) { | 
					
						
							| 
									
										
										
										
											2018-12-19 21:43:10 -05:00
										 |  |  |     const u32 reserved = flags >> 26; | 
					
						
							|  |  |  |     if (reserved != 0) { | 
					
						
							| 
									
										
										
										
											2020-04-29 14:53:53 +10:00
										 |  |  |         LOG_ERROR(Kernel, "Reserved value is non-zero! reserved={}", reserved); | 
					
						
							| 
									
										
										
										
											2021-04-11 11:41:48 -07:00
										 |  |  |         return ResultReservedUsed; | 
					
						
							| 
									
										
										
										
											2018-12-19 21:43:10 -05:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-02-25 10:13:52 -05:00
										 |  |  |     handle_table_size = static_cast<s32>((flags >> 16) & 0x3FF); | 
					
						
							| 
									
										
										
										
											2018-12-19 12:57:47 -05:00
										 |  |  |     return RESULT_SUCCESS; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | ResultCode ProcessCapabilities::HandleDebugFlags(u32 flags) { | 
					
						
							| 
									
										
										
										
											2018-12-20 23:40:30 -05:00
										 |  |  |     const u32 reserved = flags >> 19; | 
					
						
							|  |  |  |     if (reserved != 0) { | 
					
						
							| 
									
										
										
										
											2020-04-29 14:53:53 +10:00
										 |  |  |         LOG_ERROR(Kernel, "Reserved value is non-zero! reserved={}", reserved); | 
					
						
							| 
									
										
										
										
											2021-04-11 11:41:48 -07:00
										 |  |  |         return ResultReservedUsed; | 
					
						
							| 
									
										
										
										
											2018-12-20 23:40:30 -05:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     is_debuggable = (flags & 0x20000) != 0; | 
					
						
							|  |  |  |     can_force_debug = (flags & 0x40000) != 0; | 
					
						
							| 
									
										
										
										
											2018-12-19 12:57:47 -05:00
										 |  |  |     return RESULT_SUCCESS; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | } // namespace Kernel
 |