mirror of
				https://git.suyu.dev/suyu/suyu
				synced 2025-11-04 00:49:02 -06:00 
			
		
		
		
	Merge pull request #9742 from liamwhite/svc-wrap-only
kernel/svc: switch to generated wrappers
This commit is contained in:
		@@ -310,6 +310,7 @@ add_library(core STATIC
 | 
			
		||||
    hle/kernel/svc/svc_event.cpp
 | 
			
		||||
    hle/kernel/svc/svc_exception.cpp
 | 
			
		||||
    hle/kernel/svc/svc_info.cpp
 | 
			
		||||
    hle/kernel/svc/svc_insecure_memory.cpp
 | 
			
		||||
    hle/kernel/svc/svc_interrupt_event.cpp
 | 
			
		||||
    hle/kernel/svc/svc_io_pool.cpp
 | 
			
		||||
    hle/kernel/svc/svc_ipc.cpp
 | 
			
		||||
 
 | 
			
		||||
@@ -5,6 +5,7 @@
 | 
			
		||||
 | 
			
		||||
#include <array>
 | 
			
		||||
#include <span>
 | 
			
		||||
#include <string>
 | 
			
		||||
#include <vector>
 | 
			
		||||
 | 
			
		||||
#include <dynarmic/interface/halt_reason.h>
 | 
			
		||||
 
 | 
			
		||||
										
											
												File diff suppressed because it is too large
												Load Diff
											
										
									
								
							@@ -1,172 +1,536 @@
 | 
			
		||||
// SPDX-FileCopyrightText: Copyright 2018 yuzu Emulator Project
 | 
			
		||||
// SPDX-FileCopyrightText: Copyright 2023 yuzu Emulator Project
 | 
			
		||||
// SPDX-License-Identifier: GPL-2.0-or-later
 | 
			
		||||
 | 
			
		||||
#pragma once
 | 
			
		||||
// This file is automatically generated using svc_generator.py.
 | 
			
		||||
 | 
			
		||||
#include "common/common_types.h"
 | 
			
		||||
#include "core/hle/kernel/svc_types.h"
 | 
			
		||||
#include "core/hle/result.h"
 | 
			
		||||
#pragma once
 | 
			
		||||
 | 
			
		||||
namespace Core {
 | 
			
		||||
class System;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
#include "common/common_types.h"
 | 
			
		||||
#include "core/hle/kernel/svc_types.h"
 | 
			
		||||
#include "core/hle/result.h"
 | 
			
		||||
 | 
			
		||||
namespace Kernel::Svc {
 | 
			
		||||
 | 
			
		||||
void Call(Core::System& system, u32 immediate);
 | 
			
		||||
 | 
			
		||||
Result SetHeapSize(Core::System& system, VAddr* out_address, u64 size);
 | 
			
		||||
Result SetMemoryPermission(Core::System& system, VAddr address, u64 size, MemoryPermission perm);
 | 
			
		||||
Result SetMemoryAttribute(Core::System& system, VAddr address, u64 size, u32 mask, u32 attr);
 | 
			
		||||
Result MapMemory(Core::System& system, VAddr dst_addr, VAddr src_addr, u64 size);
 | 
			
		||||
Result UnmapMemory(Core::System& system, VAddr dst_addr, VAddr src_addr, u64 size);
 | 
			
		||||
Result QueryMemory(Core::System& system, VAddr memory_info_address, VAddr page_info_address,
 | 
			
		||||
                   VAddr query_address);
 | 
			
		||||
// clang-format off
 | 
			
		||||
Result SetHeapSize(Core::System& system, uintptr_t* out_address, uint64_t size);
 | 
			
		||||
Result SetMemoryPermission(Core::System& system, uint64_t address, uint64_t size, MemoryPermission perm);
 | 
			
		||||
Result SetMemoryAttribute(Core::System& system, uint64_t address, uint64_t size, uint32_t mask, uint32_t attr);
 | 
			
		||||
Result MapMemory(Core::System& system, uint64_t dst_address, uint64_t src_address, uint64_t size);
 | 
			
		||||
Result UnmapMemory(Core::System& system, uint64_t dst_address, uint64_t src_address, uint64_t size);
 | 
			
		||||
Result QueryMemory(Core::System& system, uint64_t out_memory_info, PageInfo* out_page_info, uint64_t address);
 | 
			
		||||
void ExitProcess(Core::System& system);
 | 
			
		||||
Result CreateThread(Core::System& system, Handle* out_handle, VAddr entry_point, u64 arg,
 | 
			
		||||
                    VAddr stack_bottom, u32 priority, s32 core_id);
 | 
			
		||||
Result CreateThread(Core::System& system, Handle* out_handle, uint64_t func, uint64_t arg, uint64_t stack_bottom, int32_t priority, int32_t core_id);
 | 
			
		||||
Result StartThread(Core::System& system, Handle thread_handle);
 | 
			
		||||
void ExitThread(Core::System& system);
 | 
			
		||||
void SleepThread(Core::System& system, s64 nanoseconds);
 | 
			
		||||
Result GetThreadPriority(Core::System& system, u32* out_priority, Handle handle);
 | 
			
		||||
Result SetThreadPriority(Core::System& system, Handle thread_handle, u32 priority);
 | 
			
		||||
Result GetThreadCoreMask(Core::System& system, Handle thread_handle, s32* out_core_id,
 | 
			
		||||
                         u64* out_affinity_mask);
 | 
			
		||||
Result SetThreadCoreMask(Core::System& system, Handle thread_handle, s32 core_id,
 | 
			
		||||
                         u64 affinity_mask);
 | 
			
		||||
u32 GetCurrentProcessorNumber(Core::System& system);
 | 
			
		||||
void SleepThread(Core::System& system, int64_t ns);
 | 
			
		||||
Result GetThreadPriority(Core::System& system, int32_t* out_priority, Handle thread_handle);
 | 
			
		||||
Result SetThreadPriority(Core::System& system, Handle thread_handle, int32_t priority);
 | 
			
		||||
Result GetThreadCoreMask(Core::System& system, int32_t* out_core_id, uint64_t* out_affinity_mask, Handle thread_handle);
 | 
			
		||||
Result SetThreadCoreMask(Core::System& system, Handle thread_handle, int32_t core_id, uint64_t affinity_mask);
 | 
			
		||||
int32_t GetCurrentProcessorNumber(Core::System& system);
 | 
			
		||||
Result SignalEvent(Core::System& system, Handle event_handle);
 | 
			
		||||
Result ClearEvent(Core::System& system, Handle event_handle);
 | 
			
		||||
Result MapSharedMemory(Core::System& system, Handle shmem_handle, VAddr address, u64 size,
 | 
			
		||||
                       MemoryPermission map_perm);
 | 
			
		||||
Result UnmapSharedMemory(Core::System& system, Handle shmem_handle, VAddr address, u64 size);
 | 
			
		||||
Result CreateTransferMemory(Core::System& system, Handle* out, VAddr address, u64 size,
 | 
			
		||||
                            MemoryPermission map_perm);
 | 
			
		||||
Result MapSharedMemory(Core::System& system, Handle shmem_handle, uint64_t address, uint64_t size, MemoryPermission map_perm);
 | 
			
		||||
Result UnmapSharedMemory(Core::System& system, Handle shmem_handle, uint64_t address, uint64_t size);
 | 
			
		||||
Result CreateTransferMemory(Core::System& system, Handle* out_handle, uint64_t address, uint64_t size, MemoryPermission map_perm);
 | 
			
		||||
Result CloseHandle(Core::System& system, Handle handle);
 | 
			
		||||
Result ResetSignal(Core::System& system, Handle handle);
 | 
			
		||||
Result WaitSynchronization(Core::System& system, s32* index, VAddr handles_address, s32 num_handles,
 | 
			
		||||
                           s64 nano_seconds);
 | 
			
		||||
Result WaitSynchronization(Core::System& system, int32_t* out_index, uint64_t handles, int32_t num_handles, int64_t timeout_ns);
 | 
			
		||||
Result CancelSynchronization(Core::System& system, Handle handle);
 | 
			
		||||
Result ArbitrateLock(Core::System& system, Handle thread_handle, VAddr address, u32 tag);
 | 
			
		||||
Result ArbitrateUnlock(Core::System& system, VAddr address);
 | 
			
		||||
Result WaitProcessWideKeyAtomic(Core::System& system, VAddr address, VAddr cv_key, u32 tag,
 | 
			
		||||
                                s64 timeout_ns);
 | 
			
		||||
void SignalProcessWideKey(Core::System& system, VAddr cv_key, s32 count);
 | 
			
		||||
u64 GetSystemTick(Core::System& system);
 | 
			
		||||
Result ConnectToNamedPort(Core::System& system, Handle* out, VAddr port_name_address);
 | 
			
		||||
Result SendSyncRequest(Core::System& system, Handle handle);
 | 
			
		||||
Result GetProcessId(Core::System& system, u64* out_process_id, Handle handle);
 | 
			
		||||
Result GetThreadId(Core::System& system, u64* out_thread_id, Handle thread_handle);
 | 
			
		||||
void Break(Core::System& system, u32 reason, u64 info1, u64 info2);
 | 
			
		||||
void OutputDebugString(Core::System& system, VAddr address, u64 len);
 | 
			
		||||
Result GetInfo(Core::System& system, u64* result, u64 info_id, Handle handle, u64 info_sub_id);
 | 
			
		||||
Result MapPhysicalMemory(Core::System& system, VAddr addr, u64 size);
 | 
			
		||||
Result UnmapPhysicalMemory(Core::System& system, VAddr addr, u64 size);
 | 
			
		||||
Result GetResourceLimitLimitValue(Core::System& system, u64* out_limit_value,
 | 
			
		||||
                                  Handle resource_limit_handle, LimitableResource which);
 | 
			
		||||
Result GetResourceLimitCurrentValue(Core::System& system, u64* out_current_value,
 | 
			
		||||
                                    Handle resource_limit_handle, LimitableResource which);
 | 
			
		||||
Result SetThreadActivity(Core::System& system, Handle thread_handle,
 | 
			
		||||
                         ThreadActivity thread_activity);
 | 
			
		||||
Result GetThreadContext(Core::System& system, VAddr out_context, Handle thread_handle);
 | 
			
		||||
Result WaitForAddress(Core::System& system, VAddr address, ArbitrationType arb_type, s32 value,
 | 
			
		||||
                      s64 timeout_ns);
 | 
			
		||||
Result SignalToAddress(Core::System& system, VAddr address, SignalType signal_type, s32 value,
 | 
			
		||||
                       s32 count);
 | 
			
		||||
Result ArbitrateLock(Core::System& system, Handle thread_handle, uint64_t address, uint32_t tag);
 | 
			
		||||
Result ArbitrateUnlock(Core::System& system, uint64_t address);
 | 
			
		||||
Result WaitProcessWideKeyAtomic(Core::System& system, uint64_t address, uint64_t cv_key, uint32_t tag, int64_t timeout_ns);
 | 
			
		||||
void SignalProcessWideKey(Core::System& system, uint64_t cv_key, int32_t count);
 | 
			
		||||
int64_t GetSystemTick(Core::System& system);
 | 
			
		||||
Result ConnectToNamedPort(Core::System& system, Handle* out_handle, uint64_t name);
 | 
			
		||||
Result SendSyncRequest(Core::System& system, Handle session_handle);
 | 
			
		||||
Result SendSyncRequestWithUserBuffer(Core::System& system, uint64_t message_buffer, uint64_t message_buffer_size, Handle session_handle);
 | 
			
		||||
Result SendAsyncRequestWithUserBuffer(Core::System& system, Handle* out_event_handle, uint64_t message_buffer, uint64_t message_buffer_size, Handle session_handle);
 | 
			
		||||
Result GetProcessId(Core::System& system, uint64_t* out_process_id, Handle process_handle);
 | 
			
		||||
Result GetThreadId(Core::System& system, uint64_t* out_thread_id, Handle thread_handle);
 | 
			
		||||
void Break(Core::System& system, BreakReason break_reason, uint64_t arg, uint64_t size);
 | 
			
		||||
Result OutputDebugString(Core::System& system, uint64_t debug_str, uint64_t len);
 | 
			
		||||
void ReturnFromException(Core::System& system, Result result);
 | 
			
		||||
Result GetInfo(Core::System& system, uint64_t* out, InfoType info_type, Handle handle, uint64_t info_subtype);
 | 
			
		||||
void FlushEntireDataCache(Core::System& system);
 | 
			
		||||
Result FlushDataCache(Core::System& system, uint64_t address, uint64_t size);
 | 
			
		||||
Result MapPhysicalMemory(Core::System& system, uint64_t address, uint64_t size);
 | 
			
		||||
Result UnmapPhysicalMemory(Core::System& system, uint64_t address, uint64_t size);
 | 
			
		||||
Result GetDebugFutureThreadInfo(Core::System& system, lp64::LastThreadContext* out_context, uint64_t* out_thread_id, Handle debug_handle, int64_t ns);
 | 
			
		||||
Result GetLastThreadInfo(Core::System& system, lp64::LastThreadContext* out_context, uintptr_t* out_tls_address, uint32_t* out_flags);
 | 
			
		||||
Result GetResourceLimitLimitValue(Core::System& system, int64_t* out_limit_value, Handle resource_limit_handle, LimitableResource which);
 | 
			
		||||
Result GetResourceLimitCurrentValue(Core::System& system, int64_t* out_current_value, Handle resource_limit_handle, LimitableResource which);
 | 
			
		||||
Result SetThreadActivity(Core::System& system, Handle thread_handle, ThreadActivity thread_activity);
 | 
			
		||||
Result GetThreadContext3(Core::System& system, uint64_t out_context, Handle thread_handle);
 | 
			
		||||
Result WaitForAddress(Core::System& system, uint64_t address, ArbitrationType arb_type, int32_t value, int64_t timeout_ns);
 | 
			
		||||
Result SignalToAddress(Core::System& system, uint64_t address, SignalType signal_type, int32_t value, int32_t count);
 | 
			
		||||
void SynchronizePreemptionState(Core::System& system);
 | 
			
		||||
void KernelDebug(Core::System& system, u32 kernel_debug_type, u64 param1, u64 param2, u64 param3);
 | 
			
		||||
void ChangeKernelTraceState(Core::System& system, u32 trace_state);
 | 
			
		||||
Result CreateSession(Core::System& system, Handle* out_server, Handle* out_client, u32 is_light,
 | 
			
		||||
                     u64 name);
 | 
			
		||||
Result ReplyAndReceive(Core::System& system, s32* out_index, Handle* handles, s32 num_handles,
 | 
			
		||||
                       Handle reply_target, s64 timeout_ns);
 | 
			
		||||
Result CreateEvent(Core::System& system, Handle* out_write, Handle* out_read);
 | 
			
		||||
Result CreateCodeMemory(Core::System& system, Handle* out, VAddr address, size_t size);
 | 
			
		||||
Result ControlCodeMemory(Core::System& system, Handle code_memory_handle, u32 operation,
 | 
			
		||||
                         VAddr address, size_t size, MemoryPermission perm);
 | 
			
		||||
Result GetProcessList(Core::System& system, u32* out_num_processes, VAddr out_process_ids,
 | 
			
		||||
                      u32 out_process_ids_size);
 | 
			
		||||
Result GetThreadList(Core::System& system, u32* out_num_threads, VAddr out_thread_ids,
 | 
			
		||||
                     u32 out_thread_ids_size, Handle debug_handle);
 | 
			
		||||
Result SetProcessMemoryPermission(Core::System& system, Handle process_handle, VAddr address,
 | 
			
		||||
                                  u64 size, MemoryPermission perm);
 | 
			
		||||
Result MapProcessMemory(Core::System& system, VAddr dst_address, Handle process_handle,
 | 
			
		||||
                        VAddr src_address, u64 size);
 | 
			
		||||
Result UnmapProcessMemory(Core::System& system, VAddr dst_address, Handle process_handle,
 | 
			
		||||
                          VAddr src_address, u64 size);
 | 
			
		||||
Result QueryProcessMemory(Core::System& system, VAddr memory_info_address, VAddr page_info_address,
 | 
			
		||||
                          Handle process_handle, VAddr address);
 | 
			
		||||
Result MapProcessCodeMemory(Core::System& system, Handle process_handle, u64 dst_address,
 | 
			
		||||
                            u64 src_address, u64 size);
 | 
			
		||||
Result UnmapProcessCodeMemory(Core::System& system, Handle process_handle, u64 dst_address,
 | 
			
		||||
                              u64 src_address, u64 size);
 | 
			
		||||
Result GetProcessInfo(Core::System& system, u64* out, Handle process_handle, u32 type);
 | 
			
		||||
Result GetResourceLimitPeakValue(Core::System& system, int64_t* out_peak_value, Handle resource_limit_handle, LimitableResource which);
 | 
			
		||||
Result CreateIoPool(Core::System& system, Handle* out_handle, IoPoolType which);
 | 
			
		||||
Result CreateIoRegion(Core::System& system, Handle* out_handle, Handle io_pool, uint64_t physical_address, uint64_t size, MemoryMapping mapping, MemoryPermission perm);
 | 
			
		||||
void KernelDebug(Core::System& system, KernelDebugType kern_debug_type, uint64_t arg0, uint64_t arg1, uint64_t arg2);
 | 
			
		||||
void ChangeKernelTraceState(Core::System& system, KernelTraceState kern_trace_state);
 | 
			
		||||
Result CreateSession(Core::System& system, Handle* out_server_session_handle, Handle* out_client_session_handle, bool is_light, uint64_t name);
 | 
			
		||||
Result AcceptSession(Core::System& system, Handle* out_handle, Handle port);
 | 
			
		||||
Result ReplyAndReceive(Core::System& system, int32_t* out_index, uint64_t handles, int32_t num_handles, Handle reply_target, int64_t timeout_ns);
 | 
			
		||||
Result ReplyAndReceiveWithUserBuffer(Core::System& system, int32_t* out_index, uint64_t message_buffer, uint64_t message_buffer_size, uint64_t handles, int32_t num_handles, Handle reply_target, int64_t timeout_ns);
 | 
			
		||||
Result CreateEvent(Core::System& system, Handle* out_write_handle, Handle* out_read_handle);
 | 
			
		||||
Result MapIoRegion(Core::System& system, Handle io_region, uint64_t address, uint64_t size, MemoryPermission perm);
 | 
			
		||||
Result UnmapIoRegion(Core::System& system, Handle io_region, uint64_t address, uint64_t size);
 | 
			
		||||
Result MapPhysicalMemoryUnsafe(Core::System& system, uint64_t address, uint64_t size);
 | 
			
		||||
Result UnmapPhysicalMemoryUnsafe(Core::System& system, uint64_t address, uint64_t size);
 | 
			
		||||
Result SetUnsafeLimit(Core::System& system, uint64_t limit);
 | 
			
		||||
Result CreateCodeMemory(Core::System& system, Handle* out_handle, uint64_t address, uint64_t size);
 | 
			
		||||
Result ControlCodeMemory(Core::System& system, Handle code_memory_handle, CodeMemoryOperation operation, uint64_t address, uint64_t size, MemoryPermission perm);
 | 
			
		||||
void SleepSystem(Core::System& system);
 | 
			
		||||
Result ReadWriteRegister(Core::System& system, uint32_t* out_value, uint64_t address, uint32_t mask, uint32_t value);
 | 
			
		||||
Result SetProcessActivity(Core::System& system, Handle process_handle, ProcessActivity process_activity);
 | 
			
		||||
Result CreateSharedMemory(Core::System& system, Handle* out_handle, uint64_t size, MemoryPermission owner_perm, MemoryPermission remote_perm);
 | 
			
		||||
Result MapTransferMemory(Core::System& system, Handle trmem_handle, uint64_t address, uint64_t size, MemoryPermission owner_perm);
 | 
			
		||||
Result UnmapTransferMemory(Core::System& system, Handle trmem_handle, uint64_t address, uint64_t size);
 | 
			
		||||
Result CreateInterruptEvent(Core::System& system, Handle* out_read_handle, int32_t interrupt_id, InterruptType interrupt_type);
 | 
			
		||||
Result QueryPhysicalAddress(Core::System& system, lp64::PhysicalMemoryInfo* out_info, uint64_t address);
 | 
			
		||||
Result QueryIoMapping(Core::System& system, uintptr_t* out_address, uintptr_t* out_size, uint64_t physical_address, uint64_t size);
 | 
			
		||||
Result CreateDeviceAddressSpace(Core::System& system, Handle* out_handle, uint64_t das_address, uint64_t das_size);
 | 
			
		||||
Result AttachDeviceAddressSpace(Core::System& system, DeviceName device_name, Handle das_handle);
 | 
			
		||||
Result DetachDeviceAddressSpace(Core::System& system, DeviceName device_name, Handle das_handle);
 | 
			
		||||
Result MapDeviceAddressSpaceByForce(Core::System& system, Handle das_handle, Handle process_handle, uint64_t process_address, uint64_t size, uint64_t device_address, uint32_t option);
 | 
			
		||||
Result MapDeviceAddressSpaceAligned(Core::System& system, Handle das_handle, Handle process_handle, uint64_t process_address, uint64_t size, uint64_t device_address, uint32_t option);
 | 
			
		||||
Result UnmapDeviceAddressSpace(Core::System& system, Handle das_handle, Handle process_handle, uint64_t process_address, uint64_t size, uint64_t device_address);
 | 
			
		||||
Result InvalidateProcessDataCache(Core::System& system, Handle process_handle, uint64_t address, uint64_t size);
 | 
			
		||||
Result StoreProcessDataCache(Core::System& system, Handle process_handle, uint64_t address, uint64_t size);
 | 
			
		||||
Result FlushProcessDataCache(Core::System& system, Handle process_handle, uint64_t address, uint64_t size);
 | 
			
		||||
Result DebugActiveProcess(Core::System& system, Handle* out_handle, uint64_t process_id);
 | 
			
		||||
Result BreakDebugProcess(Core::System& system, Handle debug_handle);
 | 
			
		||||
Result TerminateDebugProcess(Core::System& system, Handle debug_handle);
 | 
			
		||||
Result GetDebugEvent(Core::System& system, uint64_t out_info, Handle debug_handle);
 | 
			
		||||
Result ContinueDebugEvent(Core::System& system, Handle debug_handle, uint32_t flags, uint64_t thread_ids, int32_t num_thread_ids);
 | 
			
		||||
Result GetProcessList(Core::System& system, int32_t* out_num_processes, uint64_t out_process_ids, int32_t max_out_count);
 | 
			
		||||
Result GetThreadList(Core::System& system, int32_t* out_num_threads, uint64_t out_thread_ids, int32_t max_out_count, Handle debug_handle);
 | 
			
		||||
Result GetDebugThreadContext(Core::System& system, uint64_t out_context, Handle debug_handle, uint64_t thread_id, uint32_t context_flags);
 | 
			
		||||
Result SetDebugThreadContext(Core::System& system, Handle debug_handle, uint64_t thread_id, uint64_t context, uint32_t context_flags);
 | 
			
		||||
Result QueryDebugProcessMemory(Core::System& system, uint64_t out_memory_info, PageInfo* out_page_info, Handle process_handle, uint64_t address);
 | 
			
		||||
Result ReadDebugProcessMemory(Core::System& system, uint64_t buffer, Handle debug_handle, uint64_t address, uint64_t size);
 | 
			
		||||
Result WriteDebugProcessMemory(Core::System& system, Handle debug_handle, uint64_t buffer, uint64_t address, uint64_t size);
 | 
			
		||||
Result SetHardwareBreakPoint(Core::System& system, HardwareBreakPointRegisterName name, uint64_t flags, uint64_t value);
 | 
			
		||||
Result GetDebugThreadParam(Core::System& system, uint64_t* out_64, uint32_t* out_32, Handle debug_handle, uint64_t thread_id, DebugThreadParam param);
 | 
			
		||||
Result GetSystemInfo(Core::System& system, uint64_t* out, SystemInfoType info_type, Handle handle, uint64_t info_subtype);
 | 
			
		||||
Result CreatePort(Core::System& system, Handle* out_server_handle, Handle* out_client_handle, int32_t max_sessions, bool is_light, uint64_t name);
 | 
			
		||||
Result ManageNamedPort(Core::System& system, Handle* out_server_handle, uint64_t name, int32_t max_sessions);
 | 
			
		||||
Result ConnectToPort(Core::System& system, Handle* out_handle, Handle port);
 | 
			
		||||
Result SetProcessMemoryPermission(Core::System& system, Handle process_handle, uint64_t address, uint64_t size, MemoryPermission perm);
 | 
			
		||||
Result MapProcessMemory(Core::System& system, uint64_t dst_address, Handle process_handle, uint64_t src_address, uint64_t size);
 | 
			
		||||
Result UnmapProcessMemory(Core::System& system, uint64_t dst_address, Handle process_handle, uint64_t src_address, uint64_t size);
 | 
			
		||||
Result QueryProcessMemory(Core::System& system, uint64_t out_memory_info, PageInfo* out_page_info, Handle process_handle, uint64_t address);
 | 
			
		||||
Result MapProcessCodeMemory(Core::System& system, Handle process_handle, uint64_t dst_address, uint64_t src_address, uint64_t size);
 | 
			
		||||
Result UnmapProcessCodeMemory(Core::System& system, Handle process_handle, uint64_t dst_address, uint64_t src_address, uint64_t size);
 | 
			
		||||
Result CreateProcess(Core::System& system, Handle* out_handle, uint64_t parameters, uint64_t caps, int32_t num_caps);
 | 
			
		||||
Result StartProcess(Core::System& system, Handle process_handle, int32_t priority, int32_t core_id, uint64_t main_thread_stack_size);
 | 
			
		||||
Result TerminateProcess(Core::System& system, Handle process_handle);
 | 
			
		||||
Result GetProcessInfo(Core::System& system, int64_t* out_info, Handle process_handle, ProcessInfoType info_type);
 | 
			
		||||
Result CreateResourceLimit(Core::System& system, Handle* out_handle);
 | 
			
		||||
Result SetResourceLimitLimitValue(Core::System& system, Handle resource_limit_handle,
 | 
			
		||||
                                  LimitableResource which, u64 limit_value);
 | 
			
		||||
Result SetResourceLimitLimitValue(Core::System& system, Handle resource_limit_handle, LimitableResource which, int64_t limit_value);
 | 
			
		||||
Result MapInsecureMemory(Core::System& system, uint64_t address, uint64_t size);
 | 
			
		||||
Result UnmapInsecureMemory(Core::System& system, uint64_t address, uint64_t size);
 | 
			
		||||
 | 
			
		||||
//
 | 
			
		||||
Result SetHeapSize64From32(Core::System& system, uintptr_t* out_address, uint32_t size);
 | 
			
		||||
Result SetMemoryPermission64From32(Core::System& system, uint32_t address, uint32_t size, MemoryPermission perm);
 | 
			
		||||
Result SetMemoryAttribute64From32(Core::System& system, uint32_t address, uint32_t size, uint32_t mask, uint32_t attr);
 | 
			
		||||
Result MapMemory64From32(Core::System& system, uint32_t dst_address, uint32_t src_address, uint32_t size);
 | 
			
		||||
Result UnmapMemory64From32(Core::System& system, uint32_t dst_address, uint32_t src_address, uint32_t size);
 | 
			
		||||
Result QueryMemory64From32(Core::System& system, uint32_t out_memory_info, PageInfo* out_page_info, uint32_t address);
 | 
			
		||||
void ExitProcess64From32(Core::System& system);
 | 
			
		||||
Result CreateThread64From32(Core::System& system, Handle* out_handle, uint32_t func, uint32_t arg, uint32_t stack_bottom, int32_t priority, int32_t core_id);
 | 
			
		||||
Result StartThread64From32(Core::System& system, Handle thread_handle);
 | 
			
		||||
void ExitThread64From32(Core::System& system);
 | 
			
		||||
void SleepThread64From32(Core::System& system, int64_t ns);
 | 
			
		||||
Result GetThreadPriority64From32(Core::System& system, int32_t* out_priority, Handle thread_handle);
 | 
			
		||||
Result SetThreadPriority64From32(Core::System& system, Handle thread_handle, int32_t priority);
 | 
			
		||||
Result GetThreadCoreMask64From32(Core::System& system, int32_t* out_core_id, uint64_t* out_affinity_mask, Handle thread_handle);
 | 
			
		||||
Result SetThreadCoreMask64From32(Core::System& system, Handle thread_handle, int32_t core_id, uint64_t affinity_mask);
 | 
			
		||||
int32_t GetCurrentProcessorNumber64From32(Core::System& system);
 | 
			
		||||
Result SignalEvent64From32(Core::System& system, Handle event_handle);
 | 
			
		||||
Result ClearEvent64From32(Core::System& system, Handle event_handle);
 | 
			
		||||
Result MapSharedMemory64From32(Core::System& system, Handle shmem_handle, uint32_t address, uint32_t size, MemoryPermission map_perm);
 | 
			
		||||
Result UnmapSharedMemory64From32(Core::System& system, Handle shmem_handle, uint32_t address, uint32_t size);
 | 
			
		||||
Result CreateTransferMemory64From32(Core::System& system, Handle* out_handle, uint32_t address, uint32_t size, MemoryPermission map_perm);
 | 
			
		||||
Result CloseHandle64From32(Core::System& system, Handle handle);
 | 
			
		||||
Result ResetSignal64From32(Core::System& system, Handle handle);
 | 
			
		||||
Result WaitSynchronization64From32(Core::System& system, int32_t* out_index, uint32_t handles, int32_t num_handles, int64_t timeout_ns);
 | 
			
		||||
Result CancelSynchronization64From32(Core::System& system, Handle handle);
 | 
			
		||||
Result ArbitrateLock64From32(Core::System& system, Handle thread_handle, uint32_t address, uint32_t tag);
 | 
			
		||||
Result ArbitrateUnlock64From32(Core::System& system, uint32_t address);
 | 
			
		||||
Result WaitProcessWideKeyAtomic64From32(Core::System& system, uint32_t address, uint32_t cv_key, uint32_t tag, int64_t timeout_ns);
 | 
			
		||||
void SignalProcessWideKey64From32(Core::System& system, uint32_t cv_key, int32_t count);
 | 
			
		||||
int64_t GetSystemTick64From32(Core::System& system);
 | 
			
		||||
Result ConnectToNamedPort64From32(Core::System& system, Handle* out_handle, uint32_t name);
 | 
			
		||||
Result SendSyncRequest64From32(Core::System& system, Handle session_handle);
 | 
			
		||||
Result SendSyncRequestWithUserBuffer64From32(Core::System& system, uint32_t message_buffer, uint32_t message_buffer_size, Handle session_handle);
 | 
			
		||||
Result SendAsyncRequestWithUserBuffer64From32(Core::System& system, Handle* out_event_handle, uint32_t message_buffer, uint32_t message_buffer_size, Handle session_handle);
 | 
			
		||||
Result GetProcessId64From32(Core::System& system, uint64_t* out_process_id, Handle process_handle);
 | 
			
		||||
Result GetThreadId64From32(Core::System& system, uint64_t* out_thread_id, Handle thread_handle);
 | 
			
		||||
void Break64From32(Core::System& system, BreakReason break_reason, uint32_t arg, uint32_t size);
 | 
			
		||||
Result OutputDebugString64From32(Core::System& system, uint32_t debug_str, uint32_t len);
 | 
			
		||||
void ReturnFromException64From32(Core::System& system, Result result);
 | 
			
		||||
Result GetInfo64From32(Core::System& system, uint64_t* out, InfoType info_type, Handle handle, uint64_t info_subtype);
 | 
			
		||||
void FlushEntireDataCache64From32(Core::System& system);
 | 
			
		||||
Result FlushDataCache64From32(Core::System& system, uint32_t address, uint32_t size);
 | 
			
		||||
Result MapPhysicalMemory64From32(Core::System& system, uint32_t address, uint32_t size);
 | 
			
		||||
Result UnmapPhysicalMemory64From32(Core::System& system, uint32_t address, uint32_t size);
 | 
			
		||||
Result GetDebugFutureThreadInfo64From32(Core::System& system, ilp32::LastThreadContext* out_context, uint64_t* out_thread_id, Handle debug_handle, int64_t ns);
 | 
			
		||||
Result GetLastThreadInfo64From32(Core::System& system, ilp32::LastThreadContext* out_context, uintptr_t* out_tls_address, uint32_t* out_flags);
 | 
			
		||||
Result GetResourceLimitLimitValue64From32(Core::System& system, int64_t* out_limit_value, Handle resource_limit_handle, LimitableResource which);
 | 
			
		||||
Result GetResourceLimitCurrentValue64From32(Core::System& system, int64_t* out_current_value, Handle resource_limit_handle, LimitableResource which);
 | 
			
		||||
Result SetThreadActivity64From32(Core::System& system, Handle thread_handle, ThreadActivity thread_activity);
 | 
			
		||||
Result GetThreadContext364From32(Core::System& system, uint32_t out_context, Handle thread_handle);
 | 
			
		||||
Result WaitForAddress64From32(Core::System& system, uint32_t address, ArbitrationType arb_type, int32_t value, int64_t timeout_ns);
 | 
			
		||||
Result SignalToAddress64From32(Core::System& system, uint32_t address, SignalType signal_type, int32_t value, int32_t count);
 | 
			
		||||
void SynchronizePreemptionState64From32(Core::System& system);
 | 
			
		||||
Result GetResourceLimitPeakValue64From32(Core::System& system, int64_t* out_peak_value, Handle resource_limit_handle, LimitableResource which);
 | 
			
		||||
Result CreateIoPool64From32(Core::System& system, Handle* out_handle, IoPoolType which);
 | 
			
		||||
Result CreateIoRegion64From32(Core::System& system, Handle* out_handle, Handle io_pool, uint64_t physical_address, uint32_t size, MemoryMapping mapping, MemoryPermission perm);
 | 
			
		||||
void KernelDebug64From32(Core::System& system, KernelDebugType kern_debug_type, uint64_t arg0, uint64_t arg1, uint64_t arg2);
 | 
			
		||||
void ChangeKernelTraceState64From32(Core::System& system, KernelTraceState kern_trace_state);
 | 
			
		||||
Result CreateSession64From32(Core::System& system, Handle* out_server_session_handle, Handle* out_client_session_handle, bool is_light, uint32_t name);
 | 
			
		||||
Result AcceptSession64From32(Core::System& system, Handle* out_handle, Handle port);
 | 
			
		||||
Result ReplyAndReceive64From32(Core::System& system, int32_t* out_index, uint32_t handles, int32_t num_handles, Handle reply_target, int64_t timeout_ns);
 | 
			
		||||
Result ReplyAndReceiveWithUserBuffer64From32(Core::System& system, int32_t* out_index, uint32_t message_buffer, uint32_t message_buffer_size, uint32_t handles, int32_t num_handles, Handle reply_target, int64_t timeout_ns);
 | 
			
		||||
Result CreateEvent64From32(Core::System& system, Handle* out_write_handle, Handle* out_read_handle);
 | 
			
		||||
Result MapIoRegion64From32(Core::System& system, Handle io_region, uint32_t address, uint32_t size, MemoryPermission perm);
 | 
			
		||||
Result UnmapIoRegion64From32(Core::System& system, Handle io_region, uint32_t address, uint32_t size);
 | 
			
		||||
Result MapPhysicalMemoryUnsafe64From32(Core::System& system, uint32_t address, uint32_t size);
 | 
			
		||||
Result UnmapPhysicalMemoryUnsafe64From32(Core::System& system, uint32_t address, uint32_t size);
 | 
			
		||||
Result SetUnsafeLimit64From32(Core::System& system, uint32_t limit);
 | 
			
		||||
Result CreateCodeMemory64From32(Core::System& system, Handle* out_handle, uint32_t address, uint32_t size);
 | 
			
		||||
Result ControlCodeMemory64From32(Core::System& system, Handle code_memory_handle, CodeMemoryOperation operation, uint64_t address, uint64_t size, MemoryPermission perm);
 | 
			
		||||
void SleepSystem64From32(Core::System& system);
 | 
			
		||||
Result ReadWriteRegister64From32(Core::System& system, uint32_t* out_value, uint64_t address, uint32_t mask, uint32_t value);
 | 
			
		||||
Result SetProcessActivity64From32(Core::System& system, Handle process_handle, ProcessActivity process_activity);
 | 
			
		||||
Result CreateSharedMemory64From32(Core::System& system, Handle* out_handle, uint32_t size, MemoryPermission owner_perm, MemoryPermission remote_perm);
 | 
			
		||||
Result MapTransferMemory64From32(Core::System& system, Handle trmem_handle, uint32_t address, uint32_t size, MemoryPermission owner_perm);
 | 
			
		||||
Result UnmapTransferMemory64From32(Core::System& system, Handle trmem_handle, uint32_t address, uint32_t size);
 | 
			
		||||
Result CreateInterruptEvent64From32(Core::System& system, Handle* out_read_handle, int32_t interrupt_id, InterruptType interrupt_type);
 | 
			
		||||
Result QueryPhysicalAddress64From32(Core::System& system, ilp32::PhysicalMemoryInfo* out_info, uint32_t address);
 | 
			
		||||
Result QueryIoMapping64From32(Core::System& system, uintptr_t* out_address, uintptr_t* out_size, uint64_t physical_address, uint32_t size);
 | 
			
		||||
Result CreateDeviceAddressSpace64From32(Core::System& system, Handle* out_handle, uint64_t das_address, uint64_t das_size);
 | 
			
		||||
Result AttachDeviceAddressSpace64From32(Core::System& system, DeviceName device_name, Handle das_handle);
 | 
			
		||||
Result DetachDeviceAddressSpace64From32(Core::System& system, DeviceName device_name, Handle das_handle);
 | 
			
		||||
Result MapDeviceAddressSpaceByForce64From32(Core::System& system, Handle das_handle, Handle process_handle, uint64_t process_address, uint32_t size, uint64_t device_address, uint32_t option);
 | 
			
		||||
Result MapDeviceAddressSpaceAligned64From32(Core::System& system, Handle das_handle, Handle process_handle, uint64_t process_address, uint32_t size, uint64_t device_address, uint32_t option);
 | 
			
		||||
Result UnmapDeviceAddressSpace64From32(Core::System& system, Handle das_handle, Handle process_handle, uint64_t process_address, uint32_t size, uint64_t device_address);
 | 
			
		||||
Result InvalidateProcessDataCache64From32(Core::System& system, Handle process_handle, uint64_t address, uint64_t size);
 | 
			
		||||
Result StoreProcessDataCache64From32(Core::System& system, Handle process_handle, uint64_t address, uint64_t size);
 | 
			
		||||
Result FlushProcessDataCache64From32(Core::System& system, Handle process_handle, uint64_t address, uint64_t size);
 | 
			
		||||
Result DebugActiveProcess64From32(Core::System& system, Handle* out_handle, uint64_t process_id);
 | 
			
		||||
Result BreakDebugProcess64From32(Core::System& system, Handle debug_handle);
 | 
			
		||||
Result TerminateDebugProcess64From32(Core::System& system, Handle debug_handle);
 | 
			
		||||
Result GetDebugEvent64From32(Core::System& system, uint32_t out_info, Handle debug_handle);
 | 
			
		||||
Result ContinueDebugEvent64From32(Core::System& system, Handle debug_handle, uint32_t flags, uint32_t thread_ids, int32_t num_thread_ids);
 | 
			
		||||
Result GetProcessList64From32(Core::System& system, int32_t* out_num_processes, uint32_t out_process_ids, int32_t max_out_count);
 | 
			
		||||
Result GetThreadList64From32(Core::System& system, int32_t* out_num_threads, uint32_t out_thread_ids, int32_t max_out_count, Handle debug_handle);
 | 
			
		||||
Result GetDebugThreadContext64From32(Core::System& system, uint32_t out_context, Handle debug_handle, uint64_t thread_id, uint32_t context_flags);
 | 
			
		||||
Result SetDebugThreadContext64From32(Core::System& system, Handle debug_handle, uint64_t thread_id, uint32_t context, uint32_t context_flags);
 | 
			
		||||
Result QueryDebugProcessMemory64From32(Core::System& system, uint32_t out_memory_info, PageInfo* out_page_info, Handle process_handle, uint32_t address);
 | 
			
		||||
Result ReadDebugProcessMemory64From32(Core::System& system, uint32_t buffer, Handle debug_handle, uint32_t address, uint32_t size);
 | 
			
		||||
Result WriteDebugProcessMemory64From32(Core::System& system, Handle debug_handle, uint32_t buffer, uint32_t address, uint32_t size);
 | 
			
		||||
Result SetHardwareBreakPoint64From32(Core::System& system, HardwareBreakPointRegisterName name, uint64_t flags, uint64_t value);
 | 
			
		||||
Result GetDebugThreadParam64From32(Core::System& system, uint64_t* out_64, uint32_t* out_32, Handle debug_handle, uint64_t thread_id, DebugThreadParam param);
 | 
			
		||||
Result GetSystemInfo64From32(Core::System& system, uint64_t* out, SystemInfoType info_type, Handle handle, uint64_t info_subtype);
 | 
			
		||||
Result CreatePort64From32(Core::System& system, Handle* out_server_handle, Handle* out_client_handle, int32_t max_sessions, bool is_light, uint32_t name);
 | 
			
		||||
Result ManageNamedPort64From32(Core::System& system, Handle* out_server_handle, uint32_t name, int32_t max_sessions);
 | 
			
		||||
Result ConnectToPort64From32(Core::System& system, Handle* out_handle, Handle port);
 | 
			
		||||
Result SetProcessMemoryPermission64From32(Core::System& system, Handle process_handle, uint64_t address, uint64_t size, MemoryPermission perm);
 | 
			
		||||
Result MapProcessMemory64From32(Core::System& system, uint32_t dst_address, Handle process_handle, uint64_t src_address, uint32_t size);
 | 
			
		||||
Result UnmapProcessMemory64From32(Core::System& system, uint32_t dst_address, Handle process_handle, uint64_t src_address, uint32_t size);
 | 
			
		||||
Result QueryProcessMemory64From32(Core::System& system, uint32_t out_memory_info, PageInfo* out_page_info, Handle process_handle, uint64_t address);
 | 
			
		||||
Result MapProcessCodeMemory64From32(Core::System& system, Handle process_handle, uint64_t dst_address, uint64_t src_address, uint64_t size);
 | 
			
		||||
Result UnmapProcessCodeMemory64From32(Core::System& system, Handle process_handle, uint64_t dst_address, uint64_t src_address, uint64_t size);
 | 
			
		||||
Result CreateProcess64From32(Core::System& system, Handle* out_handle, uint32_t parameters, uint32_t caps, int32_t num_caps);
 | 
			
		||||
Result StartProcess64From32(Core::System& system, Handle process_handle, int32_t priority, int32_t core_id, uint64_t main_thread_stack_size);
 | 
			
		||||
Result TerminateProcess64From32(Core::System& system, Handle process_handle);
 | 
			
		||||
Result GetProcessInfo64From32(Core::System& system, int64_t* out_info, Handle process_handle, ProcessInfoType info_type);
 | 
			
		||||
Result CreateResourceLimit64From32(Core::System& system, Handle* out_handle);
 | 
			
		||||
Result SetResourceLimitLimitValue64From32(Core::System& system, Handle resource_limit_handle, LimitableResource which, int64_t limit_value);
 | 
			
		||||
Result MapInsecureMemory64From32(Core::System& system, uint32_t address, uint32_t size);
 | 
			
		||||
Result UnmapInsecureMemory64From32(Core::System& system, uint32_t address, uint32_t size);
 | 
			
		||||
 | 
			
		||||
Result SetHeapSize32(Core::System& system, u32* heap_addr, u32 heap_size);
 | 
			
		||||
Result SetMemoryAttribute32(Core::System& system, u32 address, u32 size, u32 mask, u32 attr);
 | 
			
		||||
Result MapMemory32(Core::System& system, u32 dst_addr, u32 src_addr, u32 size);
 | 
			
		||||
Result UnmapMemory32(Core::System& system, u32 dst_addr, u32 src_addr, u32 size);
 | 
			
		||||
Result QueryMemory32(Core::System& system, u32 memory_info_address, u32 page_info_address,
 | 
			
		||||
                     u32 query_address);
 | 
			
		||||
void ExitProcess32(Core::System& system);
 | 
			
		||||
Result CreateThread32(Core::System& system, Handle* out_handle, u32 priority, u32 entry_point,
 | 
			
		||||
                      u32 arg, u32 stack_top, s32 processor_id);
 | 
			
		||||
Result StartThread32(Core::System& system, Handle thread_handle);
 | 
			
		||||
void ExitThread32(Core::System& system);
 | 
			
		||||
void SleepThread32(Core::System& system, u32 nanoseconds_low, u32 nanoseconds_high);
 | 
			
		||||
Result GetThreadPriority32(Core::System& system, u32* out_priority, Handle handle);
 | 
			
		||||
Result SetThreadPriority32(Core::System& system, Handle thread_handle, u32 priority);
 | 
			
		||||
Result GetThreadCoreMask32(Core::System& system, Handle thread_handle, s32* out_core_id,
 | 
			
		||||
                           u32* out_affinity_mask_low, u32* out_affinity_mask_high);
 | 
			
		||||
Result SetThreadCoreMask32(Core::System& system, Handle thread_handle, s32 core_id,
 | 
			
		||||
                           u32 affinity_mask_low, u32 affinity_mask_high);
 | 
			
		||||
u32 GetCurrentProcessorNumber32(Core::System& system);
 | 
			
		||||
Result SignalEvent32(Core::System& system, Handle event_handle);
 | 
			
		||||
Result ClearEvent32(Core::System& system, Handle event_handle);
 | 
			
		||||
Result MapSharedMemory32(Core::System& system, Handle shmem_handle, u32 address, u32 size,
 | 
			
		||||
                         MemoryPermission map_perm);
 | 
			
		||||
Result UnmapSharedMemory32(Core::System& system, Handle shmem_handle, u32 address, u32 size);
 | 
			
		||||
Result CreateTransferMemory32(Core::System& system, Handle* out, u32 address, u32 size,
 | 
			
		||||
                              MemoryPermission map_perm);
 | 
			
		||||
Result CloseHandle32(Core::System& system, Handle handle);
 | 
			
		||||
Result ResetSignal32(Core::System& system, Handle handle);
 | 
			
		||||
Result WaitSynchronization32(Core::System& system, u32 timeout_low, u32 handles_address,
 | 
			
		||||
                             s32 num_handles, u32 timeout_high, s32* index);
 | 
			
		||||
Result CancelSynchronization32(Core::System& system, Handle handle);
 | 
			
		||||
Result ArbitrateLock32(Core::System& system, Handle thread_handle, u32 address, u32 tag);
 | 
			
		||||
Result ArbitrateUnlock32(Core::System& system, u32 address);
 | 
			
		||||
Result WaitProcessWideKeyAtomic32(Core::System& system, u32 address, u32 cv_key, u32 tag,
 | 
			
		||||
                                  u32 timeout_ns_low, u32 timeout_ns_high);
 | 
			
		||||
void SignalProcessWideKey32(Core::System& system, u32 cv_key, s32 count);
 | 
			
		||||
void GetSystemTick32(Core::System& system, u32* time_low, u32* time_high);
 | 
			
		||||
Result ConnectToNamedPort32(Core::System& system, Handle* out_handle, u32 port_name_address);
 | 
			
		||||
Result SendSyncRequest32(Core::System& system, Handle handle);
 | 
			
		||||
Result GetProcessId32(Core::System& system, u32* out_process_id_low, u32* out_process_id_high,
 | 
			
		||||
                      Handle handle);
 | 
			
		||||
Result GetThreadId32(Core::System& system, u32* out_thread_id_low, u32* out_thread_id_high,
 | 
			
		||||
                     Handle thread_handle);
 | 
			
		||||
void Break32(Core::System& system, u32 reason, u32 info1, u32 info2);
 | 
			
		||||
void OutputDebugString32(Core::System& system, u32 address, u32 len);
 | 
			
		||||
Result GetInfo32(Core::System& system, u32* result_low, u32* result_high, u32 sub_id_low,
 | 
			
		||||
                 u32 info_id, u32 handle, u32 sub_id_high);
 | 
			
		||||
Result MapPhysicalMemory32(Core::System& system, u32 addr, u32 size);
 | 
			
		||||
Result UnmapPhysicalMemory32(Core::System& system, u32 addr, u32 size);
 | 
			
		||||
Result SetThreadActivity32(Core::System& system, Handle thread_handle,
 | 
			
		||||
                           ThreadActivity thread_activity);
 | 
			
		||||
Result GetThreadContext32(Core::System& system, u32 out_context, Handle thread_handle);
 | 
			
		||||
Result WaitForAddress32(Core::System& system, u32 address, ArbitrationType arb_type, s32 value,
 | 
			
		||||
                        u32 timeout_ns_low, u32 timeout_ns_high);
 | 
			
		||||
Result SignalToAddress32(Core::System& system, u32 address, SignalType signal_type, s32 value,
 | 
			
		||||
                         s32 count);
 | 
			
		||||
Result CreateEvent32(Core::System& system, Handle* out_write, Handle* out_read);
 | 
			
		||||
Result CreateCodeMemory32(Core::System& system, Handle* out, u32 address, u32 size);
 | 
			
		||||
Result ControlCodeMemory32(Core::System& system, Handle code_memory_handle, u32 operation,
 | 
			
		||||
                           u64 address, u64 size, MemoryPermission perm);
 | 
			
		||||
Result FlushProcessDataCache32(Core::System& system, Handle process_handle, u64 address, u64 size);
 | 
			
		||||
Result SetHeapSize64(Core::System& system, uintptr_t* out_address, uint64_t size);
 | 
			
		||||
Result SetMemoryPermission64(Core::System& system, uint64_t address, uint64_t size, MemoryPermission perm);
 | 
			
		||||
Result SetMemoryAttribute64(Core::System& system, uint64_t address, uint64_t size, uint32_t mask, uint32_t attr);
 | 
			
		||||
Result MapMemory64(Core::System& system, uint64_t dst_address, uint64_t src_address, uint64_t size);
 | 
			
		||||
Result UnmapMemory64(Core::System& system, uint64_t dst_address, uint64_t src_address, uint64_t size);
 | 
			
		||||
Result QueryMemory64(Core::System& system, uint64_t out_memory_info, PageInfo* out_page_info, uint64_t address);
 | 
			
		||||
void ExitProcess64(Core::System& system);
 | 
			
		||||
Result CreateThread64(Core::System& system, Handle* out_handle, uint64_t func, uint64_t arg, uint64_t stack_bottom, int32_t priority, int32_t core_id);
 | 
			
		||||
Result StartThread64(Core::System& system, Handle thread_handle);
 | 
			
		||||
void ExitThread64(Core::System& system);
 | 
			
		||||
void SleepThread64(Core::System& system, int64_t ns);
 | 
			
		||||
Result GetThreadPriority64(Core::System& system, int32_t* out_priority, Handle thread_handle);
 | 
			
		||||
Result SetThreadPriority64(Core::System& system, Handle thread_handle, int32_t priority);
 | 
			
		||||
Result GetThreadCoreMask64(Core::System& system, int32_t* out_core_id, uint64_t* out_affinity_mask, Handle thread_handle);
 | 
			
		||||
Result SetThreadCoreMask64(Core::System& system, Handle thread_handle, int32_t core_id, uint64_t affinity_mask);
 | 
			
		||||
int32_t GetCurrentProcessorNumber64(Core::System& system);
 | 
			
		||||
Result SignalEvent64(Core::System& system, Handle event_handle);
 | 
			
		||||
Result ClearEvent64(Core::System& system, Handle event_handle);
 | 
			
		||||
Result MapSharedMemory64(Core::System& system, Handle shmem_handle, uint64_t address, uint64_t size, MemoryPermission map_perm);
 | 
			
		||||
Result UnmapSharedMemory64(Core::System& system, Handle shmem_handle, uint64_t address, uint64_t size);
 | 
			
		||||
Result CreateTransferMemory64(Core::System& system, Handle* out_handle, uint64_t address, uint64_t size, MemoryPermission map_perm);
 | 
			
		||||
Result CloseHandle64(Core::System& system, Handle handle);
 | 
			
		||||
Result ResetSignal64(Core::System& system, Handle handle);
 | 
			
		||||
Result WaitSynchronization64(Core::System& system, int32_t* out_index, uint64_t handles, int32_t num_handles, int64_t timeout_ns);
 | 
			
		||||
Result CancelSynchronization64(Core::System& system, Handle handle);
 | 
			
		||||
Result ArbitrateLock64(Core::System& system, Handle thread_handle, uint64_t address, uint32_t tag);
 | 
			
		||||
Result ArbitrateUnlock64(Core::System& system, uint64_t address);
 | 
			
		||||
Result WaitProcessWideKeyAtomic64(Core::System& system, uint64_t address, uint64_t cv_key, uint32_t tag, int64_t timeout_ns);
 | 
			
		||||
void SignalProcessWideKey64(Core::System& system, uint64_t cv_key, int32_t count);
 | 
			
		||||
int64_t GetSystemTick64(Core::System& system);
 | 
			
		||||
Result ConnectToNamedPort64(Core::System& system, Handle* out_handle, uint64_t name);
 | 
			
		||||
Result SendSyncRequest64(Core::System& system, Handle session_handle);
 | 
			
		||||
Result SendSyncRequestWithUserBuffer64(Core::System& system, uint64_t message_buffer, uint64_t message_buffer_size, Handle session_handle);
 | 
			
		||||
Result SendAsyncRequestWithUserBuffer64(Core::System& system, Handle* out_event_handle, uint64_t message_buffer, uint64_t message_buffer_size, Handle session_handle);
 | 
			
		||||
Result GetProcessId64(Core::System& system, uint64_t* out_process_id, Handle process_handle);
 | 
			
		||||
Result GetThreadId64(Core::System& system, uint64_t* out_thread_id, Handle thread_handle);
 | 
			
		||||
void Break64(Core::System& system, BreakReason break_reason, uint64_t arg, uint64_t size);
 | 
			
		||||
Result OutputDebugString64(Core::System& system, uint64_t debug_str, uint64_t len);
 | 
			
		||||
void ReturnFromException64(Core::System& system, Result result);
 | 
			
		||||
Result GetInfo64(Core::System& system, uint64_t* out, InfoType info_type, Handle handle, uint64_t info_subtype);
 | 
			
		||||
void FlushEntireDataCache64(Core::System& system);
 | 
			
		||||
Result FlushDataCache64(Core::System& system, uint64_t address, uint64_t size);
 | 
			
		||||
Result MapPhysicalMemory64(Core::System& system, uint64_t address, uint64_t size);
 | 
			
		||||
Result UnmapPhysicalMemory64(Core::System& system, uint64_t address, uint64_t size);
 | 
			
		||||
Result GetDebugFutureThreadInfo64(Core::System& system, lp64::LastThreadContext* out_context, uint64_t* out_thread_id, Handle debug_handle, int64_t ns);
 | 
			
		||||
Result GetLastThreadInfo64(Core::System& system, lp64::LastThreadContext* out_context, uintptr_t* out_tls_address, uint32_t* out_flags);
 | 
			
		||||
Result GetResourceLimitLimitValue64(Core::System& system, int64_t* out_limit_value, Handle resource_limit_handle, LimitableResource which);
 | 
			
		||||
Result GetResourceLimitCurrentValue64(Core::System& system, int64_t* out_current_value, Handle resource_limit_handle, LimitableResource which);
 | 
			
		||||
Result SetThreadActivity64(Core::System& system, Handle thread_handle, ThreadActivity thread_activity);
 | 
			
		||||
Result GetThreadContext364(Core::System& system, uint64_t out_context, Handle thread_handle);
 | 
			
		||||
Result WaitForAddress64(Core::System& system, uint64_t address, ArbitrationType arb_type, int32_t value, int64_t timeout_ns);
 | 
			
		||||
Result SignalToAddress64(Core::System& system, uint64_t address, SignalType signal_type, int32_t value, int32_t count);
 | 
			
		||||
void SynchronizePreemptionState64(Core::System& system);
 | 
			
		||||
Result GetResourceLimitPeakValue64(Core::System& system, int64_t* out_peak_value, Handle resource_limit_handle, LimitableResource which);
 | 
			
		||||
Result CreateIoPool64(Core::System& system, Handle* out_handle, IoPoolType which);
 | 
			
		||||
Result CreateIoRegion64(Core::System& system, Handle* out_handle, Handle io_pool, uint64_t physical_address, uint64_t size, MemoryMapping mapping, MemoryPermission perm);
 | 
			
		||||
void KernelDebug64(Core::System& system, KernelDebugType kern_debug_type, uint64_t arg0, uint64_t arg1, uint64_t arg2);
 | 
			
		||||
void ChangeKernelTraceState64(Core::System& system, KernelTraceState kern_trace_state);
 | 
			
		||||
Result CreateSession64(Core::System& system, Handle* out_server_session_handle, Handle* out_client_session_handle, bool is_light, uint64_t name);
 | 
			
		||||
Result AcceptSession64(Core::System& system, Handle* out_handle, Handle port);
 | 
			
		||||
Result ReplyAndReceive64(Core::System& system, int32_t* out_index, uint64_t handles, int32_t num_handles, Handle reply_target, int64_t timeout_ns);
 | 
			
		||||
Result ReplyAndReceiveWithUserBuffer64(Core::System& system, int32_t* out_index, uint64_t message_buffer, uint64_t message_buffer_size, uint64_t handles, int32_t num_handles, Handle reply_target, int64_t timeout_ns);
 | 
			
		||||
Result CreateEvent64(Core::System& system, Handle* out_write_handle, Handle* out_read_handle);
 | 
			
		||||
Result MapIoRegion64(Core::System& system, Handle io_region, uint64_t address, uint64_t size, MemoryPermission perm);
 | 
			
		||||
Result UnmapIoRegion64(Core::System& system, Handle io_region, uint64_t address, uint64_t size);
 | 
			
		||||
Result MapPhysicalMemoryUnsafe64(Core::System& system, uint64_t address, uint64_t size);
 | 
			
		||||
Result UnmapPhysicalMemoryUnsafe64(Core::System& system, uint64_t address, uint64_t size);
 | 
			
		||||
Result SetUnsafeLimit64(Core::System& system, uint64_t limit);
 | 
			
		||||
Result CreateCodeMemory64(Core::System& system, Handle* out_handle, uint64_t address, uint64_t size);
 | 
			
		||||
Result ControlCodeMemory64(Core::System& system, Handle code_memory_handle, CodeMemoryOperation operation, uint64_t address, uint64_t size, MemoryPermission perm);
 | 
			
		||||
void SleepSystem64(Core::System& system);
 | 
			
		||||
Result ReadWriteRegister64(Core::System& system, uint32_t* out_value, uint64_t address, uint32_t mask, uint32_t value);
 | 
			
		||||
Result SetProcessActivity64(Core::System& system, Handle process_handle, ProcessActivity process_activity);
 | 
			
		||||
Result CreateSharedMemory64(Core::System& system, Handle* out_handle, uint64_t size, MemoryPermission owner_perm, MemoryPermission remote_perm);
 | 
			
		||||
Result MapTransferMemory64(Core::System& system, Handle trmem_handle, uint64_t address, uint64_t size, MemoryPermission owner_perm);
 | 
			
		||||
Result UnmapTransferMemory64(Core::System& system, Handle trmem_handle, uint64_t address, uint64_t size);
 | 
			
		||||
Result CreateInterruptEvent64(Core::System& system, Handle* out_read_handle, int32_t interrupt_id, InterruptType interrupt_type);
 | 
			
		||||
Result QueryPhysicalAddress64(Core::System& system, lp64::PhysicalMemoryInfo* out_info, uint64_t address);
 | 
			
		||||
Result QueryIoMapping64(Core::System& system, uintptr_t* out_address, uintptr_t* out_size, uint64_t physical_address, uint64_t size);
 | 
			
		||||
Result CreateDeviceAddressSpace64(Core::System& system, Handle* out_handle, uint64_t das_address, uint64_t das_size);
 | 
			
		||||
Result AttachDeviceAddressSpace64(Core::System& system, DeviceName device_name, Handle das_handle);
 | 
			
		||||
Result DetachDeviceAddressSpace64(Core::System& system, DeviceName device_name, Handle das_handle);
 | 
			
		||||
Result MapDeviceAddressSpaceByForce64(Core::System& system, Handle das_handle, Handle process_handle, uint64_t process_address, uint64_t size, uint64_t device_address, uint32_t option);
 | 
			
		||||
Result MapDeviceAddressSpaceAligned64(Core::System& system, Handle das_handle, Handle process_handle, uint64_t process_address, uint64_t size, uint64_t device_address, uint32_t option);
 | 
			
		||||
Result UnmapDeviceAddressSpace64(Core::System& system, Handle das_handle, Handle process_handle, uint64_t process_address, uint64_t size, uint64_t device_address);
 | 
			
		||||
Result InvalidateProcessDataCache64(Core::System& system, Handle process_handle, uint64_t address, uint64_t size);
 | 
			
		||||
Result StoreProcessDataCache64(Core::System& system, Handle process_handle, uint64_t address, uint64_t size);
 | 
			
		||||
Result FlushProcessDataCache64(Core::System& system, Handle process_handle, uint64_t address, uint64_t size);
 | 
			
		||||
Result DebugActiveProcess64(Core::System& system, Handle* out_handle, uint64_t process_id);
 | 
			
		||||
Result BreakDebugProcess64(Core::System& system, Handle debug_handle);
 | 
			
		||||
Result TerminateDebugProcess64(Core::System& system, Handle debug_handle);
 | 
			
		||||
Result GetDebugEvent64(Core::System& system, uint64_t out_info, Handle debug_handle);
 | 
			
		||||
Result ContinueDebugEvent64(Core::System& system, Handle debug_handle, uint32_t flags, uint64_t thread_ids, int32_t num_thread_ids);
 | 
			
		||||
Result GetProcessList64(Core::System& system, int32_t* out_num_processes, uint64_t out_process_ids, int32_t max_out_count);
 | 
			
		||||
Result GetThreadList64(Core::System& system, int32_t* out_num_threads, uint64_t out_thread_ids, int32_t max_out_count, Handle debug_handle);
 | 
			
		||||
Result GetDebugThreadContext64(Core::System& system, uint64_t out_context, Handle debug_handle, uint64_t thread_id, uint32_t context_flags);
 | 
			
		||||
Result SetDebugThreadContext64(Core::System& system, Handle debug_handle, uint64_t thread_id, uint64_t context, uint32_t context_flags);
 | 
			
		||||
Result QueryDebugProcessMemory64(Core::System& system, uint64_t out_memory_info, PageInfo* out_page_info, Handle process_handle, uint64_t address);
 | 
			
		||||
Result ReadDebugProcessMemory64(Core::System& system, uint64_t buffer, Handle debug_handle, uint64_t address, uint64_t size);
 | 
			
		||||
Result WriteDebugProcessMemory64(Core::System& system, Handle debug_handle, uint64_t buffer, uint64_t address, uint64_t size);
 | 
			
		||||
Result SetHardwareBreakPoint64(Core::System& system, HardwareBreakPointRegisterName name, uint64_t flags, uint64_t value);
 | 
			
		||||
Result GetDebugThreadParam64(Core::System& system, uint64_t* out_64, uint32_t* out_32, Handle debug_handle, uint64_t thread_id, DebugThreadParam param);
 | 
			
		||||
Result GetSystemInfo64(Core::System& system, uint64_t* out, SystemInfoType info_type, Handle handle, uint64_t info_subtype);
 | 
			
		||||
Result CreatePort64(Core::System& system, Handle* out_server_handle, Handle* out_client_handle, int32_t max_sessions, bool is_light, uint64_t name);
 | 
			
		||||
Result ManageNamedPort64(Core::System& system, Handle* out_server_handle, uint64_t name, int32_t max_sessions);
 | 
			
		||||
Result ConnectToPort64(Core::System& system, Handle* out_handle, Handle port);
 | 
			
		||||
Result SetProcessMemoryPermission64(Core::System& system, Handle process_handle, uint64_t address, uint64_t size, MemoryPermission perm);
 | 
			
		||||
Result MapProcessMemory64(Core::System& system, uint64_t dst_address, Handle process_handle, uint64_t src_address, uint64_t size);
 | 
			
		||||
Result UnmapProcessMemory64(Core::System& system, uint64_t dst_address, Handle process_handle, uint64_t src_address, uint64_t size);
 | 
			
		||||
Result QueryProcessMemory64(Core::System& system, uint64_t out_memory_info, PageInfo* out_page_info, Handle process_handle, uint64_t address);
 | 
			
		||||
Result MapProcessCodeMemory64(Core::System& system, Handle process_handle, uint64_t dst_address, uint64_t src_address, uint64_t size);
 | 
			
		||||
Result UnmapProcessCodeMemory64(Core::System& system, Handle process_handle, uint64_t dst_address, uint64_t src_address, uint64_t size);
 | 
			
		||||
Result CreateProcess64(Core::System& system, Handle* out_handle, uint64_t parameters, uint64_t caps, int32_t num_caps);
 | 
			
		||||
Result StartProcess64(Core::System& system, Handle process_handle, int32_t priority, int32_t core_id, uint64_t main_thread_stack_size);
 | 
			
		||||
Result TerminateProcess64(Core::System& system, Handle process_handle);
 | 
			
		||||
Result GetProcessInfo64(Core::System& system, int64_t* out_info, Handle process_handle, ProcessInfoType info_type);
 | 
			
		||||
Result CreateResourceLimit64(Core::System& system, Handle* out_handle);
 | 
			
		||||
Result SetResourceLimitLimitValue64(Core::System& system, Handle resource_limit_handle, LimitableResource which, int64_t limit_value);
 | 
			
		||||
Result MapInsecureMemory64(Core::System& system, uint64_t address, uint64_t size);
 | 
			
		||||
Result UnmapInsecureMemory64(Core::System& system, uint64_t address, uint64_t size);
 | 
			
		||||
 | 
			
		||||
enum class SvcId : u32 {
 | 
			
		||||
    SetHeapSize = 0x1,
 | 
			
		||||
    SetMemoryPermission = 0x2,
 | 
			
		||||
    SetMemoryAttribute = 0x3,
 | 
			
		||||
    MapMemory = 0x4,
 | 
			
		||||
    UnmapMemory = 0x5,
 | 
			
		||||
    QueryMemory = 0x6,
 | 
			
		||||
    ExitProcess = 0x7,
 | 
			
		||||
    CreateThread = 0x8,
 | 
			
		||||
    StartThread = 0x9,
 | 
			
		||||
    ExitThread = 0xa,
 | 
			
		||||
    SleepThread = 0xb,
 | 
			
		||||
    GetThreadPriority = 0xc,
 | 
			
		||||
    SetThreadPriority = 0xd,
 | 
			
		||||
    GetThreadCoreMask = 0xe,
 | 
			
		||||
    SetThreadCoreMask = 0xf,
 | 
			
		||||
    GetCurrentProcessorNumber = 0x10,
 | 
			
		||||
    SignalEvent = 0x11,
 | 
			
		||||
    ClearEvent = 0x12,
 | 
			
		||||
    MapSharedMemory = 0x13,
 | 
			
		||||
    UnmapSharedMemory = 0x14,
 | 
			
		||||
    CreateTransferMemory = 0x15,
 | 
			
		||||
    CloseHandle = 0x16,
 | 
			
		||||
    ResetSignal = 0x17,
 | 
			
		||||
    WaitSynchronization = 0x18,
 | 
			
		||||
    CancelSynchronization = 0x19,
 | 
			
		||||
    ArbitrateLock = 0x1a,
 | 
			
		||||
    ArbitrateUnlock = 0x1b,
 | 
			
		||||
    WaitProcessWideKeyAtomic = 0x1c,
 | 
			
		||||
    SignalProcessWideKey = 0x1d,
 | 
			
		||||
    GetSystemTick = 0x1e,
 | 
			
		||||
    ConnectToNamedPort = 0x1f,
 | 
			
		||||
    SendSyncRequestLight = 0x20,
 | 
			
		||||
    SendSyncRequest = 0x21,
 | 
			
		||||
    SendSyncRequestWithUserBuffer = 0x22,
 | 
			
		||||
    SendAsyncRequestWithUserBuffer = 0x23,
 | 
			
		||||
    GetProcessId = 0x24,
 | 
			
		||||
    GetThreadId = 0x25,
 | 
			
		||||
    Break = 0x26,
 | 
			
		||||
    OutputDebugString = 0x27,
 | 
			
		||||
    ReturnFromException = 0x28,
 | 
			
		||||
    GetInfo = 0x29,
 | 
			
		||||
    FlushEntireDataCache = 0x2a,
 | 
			
		||||
    FlushDataCache = 0x2b,
 | 
			
		||||
    MapPhysicalMemory = 0x2c,
 | 
			
		||||
    UnmapPhysicalMemory = 0x2d,
 | 
			
		||||
    GetDebugFutureThreadInfo = 0x2e,
 | 
			
		||||
    GetLastThreadInfo = 0x2f,
 | 
			
		||||
    GetResourceLimitLimitValue = 0x30,
 | 
			
		||||
    GetResourceLimitCurrentValue = 0x31,
 | 
			
		||||
    SetThreadActivity = 0x32,
 | 
			
		||||
    GetThreadContext3 = 0x33,
 | 
			
		||||
    WaitForAddress = 0x34,
 | 
			
		||||
    SignalToAddress = 0x35,
 | 
			
		||||
    SynchronizePreemptionState = 0x36,
 | 
			
		||||
    GetResourceLimitPeakValue = 0x37,
 | 
			
		||||
    CreateIoPool = 0x39,
 | 
			
		||||
    CreateIoRegion = 0x3a,
 | 
			
		||||
    KernelDebug = 0x3c,
 | 
			
		||||
    ChangeKernelTraceState = 0x3d,
 | 
			
		||||
    CreateSession = 0x40,
 | 
			
		||||
    AcceptSession = 0x41,
 | 
			
		||||
    ReplyAndReceiveLight = 0x42,
 | 
			
		||||
    ReplyAndReceive = 0x43,
 | 
			
		||||
    ReplyAndReceiveWithUserBuffer = 0x44,
 | 
			
		||||
    CreateEvent = 0x45,
 | 
			
		||||
    MapIoRegion = 0x46,
 | 
			
		||||
    UnmapIoRegion = 0x47,
 | 
			
		||||
    MapPhysicalMemoryUnsafe = 0x48,
 | 
			
		||||
    UnmapPhysicalMemoryUnsafe = 0x49,
 | 
			
		||||
    SetUnsafeLimit = 0x4a,
 | 
			
		||||
    CreateCodeMemory = 0x4b,
 | 
			
		||||
    ControlCodeMemory = 0x4c,
 | 
			
		||||
    SleepSystem = 0x4d,
 | 
			
		||||
    ReadWriteRegister = 0x4e,
 | 
			
		||||
    SetProcessActivity = 0x4f,
 | 
			
		||||
    CreateSharedMemory = 0x50,
 | 
			
		||||
    MapTransferMemory = 0x51,
 | 
			
		||||
    UnmapTransferMemory = 0x52,
 | 
			
		||||
    CreateInterruptEvent = 0x53,
 | 
			
		||||
    QueryPhysicalAddress = 0x54,
 | 
			
		||||
    QueryIoMapping = 0x55,
 | 
			
		||||
    CreateDeviceAddressSpace = 0x56,
 | 
			
		||||
    AttachDeviceAddressSpace = 0x57,
 | 
			
		||||
    DetachDeviceAddressSpace = 0x58,
 | 
			
		||||
    MapDeviceAddressSpaceByForce = 0x59,
 | 
			
		||||
    MapDeviceAddressSpaceAligned = 0x5a,
 | 
			
		||||
    UnmapDeviceAddressSpace = 0x5c,
 | 
			
		||||
    InvalidateProcessDataCache = 0x5d,
 | 
			
		||||
    StoreProcessDataCache = 0x5e,
 | 
			
		||||
    FlushProcessDataCache = 0x5f,
 | 
			
		||||
    DebugActiveProcess = 0x60,
 | 
			
		||||
    BreakDebugProcess = 0x61,
 | 
			
		||||
    TerminateDebugProcess = 0x62,
 | 
			
		||||
    GetDebugEvent = 0x63,
 | 
			
		||||
    ContinueDebugEvent = 0x64,
 | 
			
		||||
    GetProcessList = 0x65,
 | 
			
		||||
    GetThreadList = 0x66,
 | 
			
		||||
    GetDebugThreadContext = 0x67,
 | 
			
		||||
    SetDebugThreadContext = 0x68,
 | 
			
		||||
    QueryDebugProcessMemory = 0x69,
 | 
			
		||||
    ReadDebugProcessMemory = 0x6a,
 | 
			
		||||
    WriteDebugProcessMemory = 0x6b,
 | 
			
		||||
    SetHardwareBreakPoint = 0x6c,
 | 
			
		||||
    GetDebugThreadParam = 0x6d,
 | 
			
		||||
    GetSystemInfo = 0x6f,
 | 
			
		||||
    CreatePort = 0x70,
 | 
			
		||||
    ManageNamedPort = 0x71,
 | 
			
		||||
    ConnectToPort = 0x72,
 | 
			
		||||
    SetProcessMemoryPermission = 0x73,
 | 
			
		||||
    MapProcessMemory = 0x74,
 | 
			
		||||
    UnmapProcessMemory = 0x75,
 | 
			
		||||
    QueryProcessMemory = 0x76,
 | 
			
		||||
    MapProcessCodeMemory = 0x77,
 | 
			
		||||
    UnmapProcessCodeMemory = 0x78,
 | 
			
		||||
    CreateProcess = 0x79,
 | 
			
		||||
    StartProcess = 0x7a,
 | 
			
		||||
    TerminateProcess = 0x7b,
 | 
			
		||||
    GetProcessInfo = 0x7c,
 | 
			
		||||
    CreateResourceLimit = 0x7d,
 | 
			
		||||
    SetResourceLimitLimitValue = 0x7e,
 | 
			
		||||
    CallSecureMonitor = 0x7f,
 | 
			
		||||
    MapInsecureMemory = 0x90,
 | 
			
		||||
    UnmapInsecureMemory = 0x91,
 | 
			
		||||
};
 | 
			
		||||
// clang-format on
 | 
			
		||||
 | 
			
		||||
// Custom ABI.
 | 
			
		||||
Result ReplyAndReceiveLight(Core::System& system, Handle handle, uint32_t* args);
 | 
			
		||||
Result ReplyAndReceiveLight64From32(Core::System& system, Handle handle, uint32_t* args);
 | 
			
		||||
Result ReplyAndReceiveLight64(Core::System& system, Handle handle, uint32_t* args);
 | 
			
		||||
 | 
			
		||||
Result SendSyncRequestLight(Core::System& system, Handle session_handle, uint32_t* args);
 | 
			
		||||
Result SendSyncRequestLight64From32(Core::System& system, Handle session_handle, uint32_t* args);
 | 
			
		||||
Result SendSyncRequestLight64(Core::System& system, Handle session_handle, uint32_t* args);
 | 
			
		||||
 | 
			
		||||
void CallSecureMonitor(Core::System& system, lp64::SecureMonitorArguments* args);
 | 
			
		||||
void CallSecureMonitor64From32(Core::System& system, ilp32::SecureMonitorArguments* args);
 | 
			
		||||
void CallSecureMonitor64(Core::System& system, lp64::SecureMonitorArguments* args);
 | 
			
		||||
 | 
			
		||||
// Defined in svc_light_ipc.cpp.
 | 
			
		||||
void SvcWrap_ReplyAndReceiveLight64From32(Core::System& system);
 | 
			
		||||
void SvcWrap_ReplyAndReceiveLight64(Core::System& system);
 | 
			
		||||
 | 
			
		||||
void SvcWrap_SendSyncRequestLight64From32(Core::System& system);
 | 
			
		||||
void SvcWrap_SendSyncRequestLight64(Core::System& system);
 | 
			
		||||
 | 
			
		||||
// Defined in svc_secure_monitor_call.cpp.
 | 
			
		||||
void SvcWrap_CallSecureMonitor64From32(Core::System& system);
 | 
			
		||||
void SvcWrap_CallSecureMonitor64(Core::System& system);
 | 
			
		||||
 | 
			
		||||
// Perform a supervisor call by index.
 | 
			
		||||
void Call(Core::System& system, u32 imm);
 | 
			
		||||
 | 
			
		||||
} // namespace Kernel::Svc
 | 
			
		||||
 
 | 
			
		||||
@@ -36,9 +36,30 @@ Result SetThreadActivity(Core::System& system, Handle thread_handle,
 | 
			
		||||
    return ResultSuccess;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
Result SetThreadActivity32(Core::System& system, Handle thread_handle,
 | 
			
		||||
Result SetProcessActivity(Core::System& system, Handle process_handle,
 | 
			
		||||
                          ProcessActivity process_activity) {
 | 
			
		||||
    UNIMPLEMENTED();
 | 
			
		||||
    R_THROW(ResultNotImplemented);
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
Result SetThreadActivity64(Core::System& system, Handle thread_handle,
 | 
			
		||||
                           ThreadActivity thread_activity) {
 | 
			
		||||
    return SetThreadActivity(system, thread_handle, thread_activity);
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
Result SetProcessActivity64(Core::System& system, Handle process_handle,
 | 
			
		||||
                            ProcessActivity process_activity) {
 | 
			
		||||
    return SetProcessActivity(system, process_handle, process_activity);
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
Result SetThreadActivity64From32(Core::System& system, Handle thread_handle,
 | 
			
		||||
                                 ThreadActivity thread_activity) {
 | 
			
		||||
    return SetThreadActivity(system, thread_handle, thread_activity);
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
Result SetProcessActivity64From32(Core::System& system, Handle process_handle,
 | 
			
		||||
                                  ProcessActivity process_activity) {
 | 
			
		||||
    return SetProcessActivity(system, process_handle, process_activity);
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
} // namespace Kernel::Svc
 | 
			
		||||
 
 | 
			
		||||
@@ -75,12 +75,6 @@ Result WaitForAddress(Core::System& system, VAddr address, ArbitrationType arb_t
 | 
			
		||||
    return system.Kernel().CurrentProcess()->WaitAddressArbiter(address, arb_type, value, timeout);
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
Result WaitForAddress32(Core::System& system, u32 address, ArbitrationType arb_type, s32 value,
 | 
			
		||||
                        u32 timeout_ns_low, u32 timeout_ns_high) {
 | 
			
		||||
    const auto timeout = static_cast<s64>(timeout_ns_low | (u64{timeout_ns_high} << 32));
 | 
			
		||||
    return WaitForAddress(system, address, arb_type, value, timeout);
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
// Signals to an address (via Address Arbiter)
 | 
			
		||||
Result SignalToAddress(Core::System& system, VAddr address, SignalType signal_type, s32 value,
 | 
			
		||||
                       s32 count) {
 | 
			
		||||
@@ -105,9 +99,24 @@ Result SignalToAddress(Core::System& system, VAddr address, SignalType signal_ty
 | 
			
		||||
                                                                  count);
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
Result SignalToAddress32(Core::System& system, u32 address, SignalType signal_type, s32 value,
 | 
			
		||||
Result WaitForAddress64(Core::System& system, VAddr address, ArbitrationType arb_type, s32 value,
 | 
			
		||||
                        s64 timeout_ns) {
 | 
			
		||||
    return WaitForAddress(system, address, arb_type, value, timeout_ns);
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
Result SignalToAddress64(Core::System& system, VAddr address, SignalType signal_type, s32 value,
 | 
			
		||||
                         s32 count) {
 | 
			
		||||
    return SignalToAddress(system, address, signal_type, value, count);
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
Result WaitForAddress64From32(Core::System& system, u32 address, ArbitrationType arb_type,
 | 
			
		||||
                              s32 value, s64 timeout_ns) {
 | 
			
		||||
    return WaitForAddress(system, address, arb_type, value, timeout_ns);
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
Result SignalToAddress64From32(Core::System& system, u32 address, SignalType signal_type, s32 value,
 | 
			
		||||
                               s32 count) {
 | 
			
		||||
    return SignalToAddress(system, address, signal_type, value, count);
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
} // namespace Kernel::Svc
 | 
			
		||||
 
 | 
			
		||||
@@ -2,5 +2,49 @@
 | 
			
		||||
// SPDX-License-Identifier: GPL-2.0-or-later
 | 
			
		||||
 | 
			
		||||
#include "core/hle/kernel/svc.h"
 | 
			
		||||
#include "core/hle/kernel/svc_results.h"
 | 
			
		||||
 | 
			
		||||
namespace Kernel::Svc {} // namespace Kernel::Svc
 | 
			
		||||
namespace Kernel::Svc {
 | 
			
		||||
 | 
			
		||||
Result QueryPhysicalAddress(Core::System& system, lp64::PhysicalMemoryInfo* out_info,
 | 
			
		||||
                            uint64_t address) {
 | 
			
		||||
    UNIMPLEMENTED();
 | 
			
		||||
    R_THROW(ResultNotImplemented);
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
Result QueryIoMapping(Core::System& system, uintptr_t* out_address, uintptr_t* out_size,
 | 
			
		||||
                      uint64_t physical_address, uint64_t size) {
 | 
			
		||||
    UNIMPLEMENTED();
 | 
			
		||||
    R_THROW(ResultNotImplemented);
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
Result QueryPhysicalAddress64(Core::System& system, lp64::PhysicalMemoryInfo* out_info,
 | 
			
		||||
                              uint64_t address) {
 | 
			
		||||
    R_RETURN(QueryPhysicalAddress(system, out_info, address));
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
Result QueryIoMapping64(Core::System& system, uintptr_t* out_address, uintptr_t* out_size,
 | 
			
		||||
                        uint64_t physical_address, uint64_t size) {
 | 
			
		||||
    R_RETURN(QueryIoMapping(system, out_address, out_size, physical_address, size));
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
Result QueryPhysicalAddress64From32(Core::System& system, ilp32::PhysicalMemoryInfo* out_info,
 | 
			
		||||
                                    uint32_t address) {
 | 
			
		||||
    lp64::PhysicalMemoryInfo info{};
 | 
			
		||||
    R_TRY(QueryPhysicalAddress(system, std::addressof(info), address));
 | 
			
		||||
 | 
			
		||||
    *out_info = {
 | 
			
		||||
        .physical_address = info.physical_address,
 | 
			
		||||
        .virtual_address = static_cast<u32>(info.virtual_address),
 | 
			
		||||
        .size = static_cast<u32>(info.size),
 | 
			
		||||
    };
 | 
			
		||||
    R_SUCCEED();
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
Result QueryIoMapping64From32(Core::System& system, uintptr_t* out_address, uintptr_t* out_size,
 | 
			
		||||
                              uint64_t physical_address, uint32_t size) {
 | 
			
		||||
    R_RETURN(QueryIoMapping(system, reinterpret_cast<uintptr_t*>(out_address),
 | 
			
		||||
                            reinterpret_cast<uintptr_t*>(out_size), physical_address, size));
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
} // namespace Kernel::Svc
 | 
			
		||||
 
 | 
			
		||||
@@ -9,7 +9,28 @@
 | 
			
		||||
 | 
			
		||||
namespace Kernel::Svc {
 | 
			
		||||
 | 
			
		||||
Result FlushProcessDataCache32(Core::System& system, Handle process_handle, u64 address, u64 size) {
 | 
			
		||||
void FlushEntireDataCache(Core::System& system) {
 | 
			
		||||
    UNIMPLEMENTED();
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
Result FlushDataCache(Core::System& system, VAddr address, size_t size) {
 | 
			
		||||
    UNIMPLEMENTED();
 | 
			
		||||
    R_THROW(ResultNotImplemented);
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
Result InvalidateProcessDataCache(Core::System& system, Handle process_handle, uint64_t address,
 | 
			
		||||
                                  uint64_t size) {
 | 
			
		||||
    UNIMPLEMENTED();
 | 
			
		||||
    R_THROW(ResultNotImplemented);
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
Result StoreProcessDataCache(Core::System& system, Handle process_handle, uint64_t address,
 | 
			
		||||
                             uint64_t size) {
 | 
			
		||||
    UNIMPLEMENTED();
 | 
			
		||||
    R_THROW(ResultNotImplemented);
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
Result FlushProcessDataCache(Core::System& system, Handle process_handle, u64 address, u64 size) {
 | 
			
		||||
    // Validate address/size.
 | 
			
		||||
    R_UNLESS(size > 0, ResultInvalidSize);
 | 
			
		||||
    R_UNLESS(address == static_cast<uintptr_t>(address), ResultInvalidCurrentMemory);
 | 
			
		||||
@@ -28,4 +49,50 @@ Result FlushProcessDataCache32(Core::System& system, Handle process_handle, u64
 | 
			
		||||
    R_RETURN(system.Memory().FlushDataCache(*process, address, size));
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
void FlushEntireDataCache64(Core::System& system) {
 | 
			
		||||
    FlushEntireDataCache(system);
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
Result FlushDataCache64(Core::System& system, VAddr address, size_t size) {
 | 
			
		||||
    R_RETURN(FlushDataCache(system, address, size));
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
Result InvalidateProcessDataCache64(Core::System& system, Handle process_handle, uint64_t address,
 | 
			
		||||
                                    uint64_t size) {
 | 
			
		||||
    R_RETURN(InvalidateProcessDataCache(system, process_handle, address, size));
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
Result StoreProcessDataCache64(Core::System& system, Handle process_handle, uint64_t address,
 | 
			
		||||
                               uint64_t size) {
 | 
			
		||||
    R_RETURN(StoreProcessDataCache(system, process_handle, address, size));
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
Result FlushProcessDataCache64(Core::System& system, Handle process_handle, uint64_t address,
 | 
			
		||||
                               uint64_t size) {
 | 
			
		||||
    R_RETURN(FlushProcessDataCache(system, process_handle, address, size));
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
void FlushEntireDataCache64From32(Core::System& system) {
 | 
			
		||||
    return FlushEntireDataCache(system);
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
Result FlushDataCache64From32(Core::System& system, uint32_t address, uint32_t size) {
 | 
			
		||||
    R_RETURN(FlushDataCache(system, address, size));
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
Result InvalidateProcessDataCache64From32(Core::System& system, Handle process_handle,
 | 
			
		||||
                                          uint64_t address, uint64_t size) {
 | 
			
		||||
    R_RETURN(InvalidateProcessDataCache(system, process_handle, address, size));
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
Result StoreProcessDataCache64From32(Core::System& system, Handle process_handle, uint64_t address,
 | 
			
		||||
                                     uint64_t size) {
 | 
			
		||||
    R_RETURN(StoreProcessDataCache(system, process_handle, address, size));
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
Result FlushProcessDataCache64From32(Core::System& system, Handle process_handle, uint64_t address,
 | 
			
		||||
                                     uint64_t size) {
 | 
			
		||||
    R_RETURN(FlushProcessDataCache(system, process_handle, address, size));
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
} // namespace Kernel::Svc
 | 
			
		||||
 
 | 
			
		||||
@@ -63,12 +63,9 @@ Result CreateCodeMemory(Core::System& system, Handle* out, VAddr address, size_t
 | 
			
		||||
    return ResultSuccess;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
Result CreateCodeMemory32(Core::System& system, Handle* out, u32 address, u32 size) {
 | 
			
		||||
    return CreateCodeMemory(system, out, address, size);
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
Result ControlCodeMemory(Core::System& system, Handle code_memory_handle, u32 operation,
 | 
			
		||||
                         VAddr address, size_t size, MemoryPermission perm) {
 | 
			
		||||
Result ControlCodeMemory(Core::System& system, Handle code_memory_handle,
 | 
			
		||||
                         CodeMemoryOperation operation, VAddr address, size_t size,
 | 
			
		||||
                         MemoryPermission perm) {
 | 
			
		||||
 | 
			
		||||
    LOG_TRACE(Kernel_SVC,
 | 
			
		||||
              "called, code_memory_handle=0x{:X}, operation=0x{:X}, address=0x{:X}, size=0x{:X}, "
 | 
			
		||||
@@ -90,7 +87,7 @@ Result ControlCodeMemory(Core::System& system, Handle code_memory_handle, u32 op
 | 
			
		||||
    // This enables homebrew usage of these SVCs for JIT.
 | 
			
		||||
 | 
			
		||||
    // Perform the operation.
 | 
			
		||||
    switch (static_cast<CodeMemoryOperation>(operation)) {
 | 
			
		||||
    switch (operation) {
 | 
			
		||||
    case CodeMemoryOperation::Map: {
 | 
			
		||||
        // Check that the region is in range.
 | 
			
		||||
        R_UNLESS(
 | 
			
		||||
@@ -146,9 +143,26 @@ Result ControlCodeMemory(Core::System& system, Handle code_memory_handle, u32 op
 | 
			
		||||
    return ResultSuccess;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
Result ControlCodeMemory32(Core::System& system, Handle code_memory_handle, u32 operation,
 | 
			
		||||
                           u64 address, u64 size, MemoryPermission perm) {
 | 
			
		||||
    return ControlCodeMemory(system, code_memory_handle, operation, address, size, perm);
 | 
			
		||||
Result CreateCodeMemory64(Core::System& system, Handle* out_handle, uint64_t address,
 | 
			
		||||
                          uint64_t size) {
 | 
			
		||||
    R_RETURN(CreateCodeMemory(system, out_handle, address, size));
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
Result ControlCodeMemory64(Core::System& system, Handle code_memory_handle,
 | 
			
		||||
                           CodeMemoryOperation operation, uint64_t address, uint64_t size,
 | 
			
		||||
                           MemoryPermission perm) {
 | 
			
		||||
    R_RETURN(ControlCodeMemory(system, code_memory_handle, operation, address, size, perm));
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
Result CreateCodeMemory64From32(Core::System& system, Handle* out_handle, uint32_t address,
 | 
			
		||||
                                uint32_t size) {
 | 
			
		||||
    R_RETURN(CreateCodeMemory(system, out_handle, address, size));
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
Result ControlCodeMemory64From32(Core::System& system, Handle code_memory_handle,
 | 
			
		||||
                                 CodeMemoryOperation operation, uint64_t address, uint64_t size,
 | 
			
		||||
                                 MemoryPermission perm) {
 | 
			
		||||
    R_RETURN(ControlCodeMemory(system, code_memory_handle, operation, address, size, perm));
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
} // namespace Kernel::Svc
 | 
			
		||||
 
 | 
			
		||||
@@ -47,12 +47,6 @@ Result WaitProcessWideKeyAtomic(Core::System& system, VAddr address, VAddr cv_ke
 | 
			
		||||
        address, Common::AlignDown(cv_key, sizeof(u32)), tag, timeout);
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
Result WaitProcessWideKeyAtomic32(Core::System& system, u32 address, u32 cv_key, u32 tag,
 | 
			
		||||
                                  u32 timeout_ns_low, u32 timeout_ns_high) {
 | 
			
		||||
    const auto timeout_ns = static_cast<s64>(timeout_ns_low | (u64{timeout_ns_high} << 32));
 | 
			
		||||
    return WaitProcessWideKeyAtomic(system, address, cv_key, tag, timeout_ns);
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
/// Signal process wide key
 | 
			
		||||
void SignalProcessWideKey(Core::System& system, VAddr cv_key, s32 count) {
 | 
			
		||||
    LOG_TRACE(Kernel_SVC, "called, cv_key=0x{:X}, count=0x{:08X}", cv_key, count);
 | 
			
		||||
@@ -62,7 +56,21 @@ void SignalProcessWideKey(Core::System& system, VAddr cv_key, s32 count) {
 | 
			
		||||
        Common::AlignDown(cv_key, sizeof(u32)), count);
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
void SignalProcessWideKey32(Core::System& system, u32 cv_key, s32 count) {
 | 
			
		||||
Result WaitProcessWideKeyAtomic64(Core::System& system, uint64_t address, uint64_t cv_key,
 | 
			
		||||
                                  uint32_t tag, int64_t timeout_ns) {
 | 
			
		||||
    R_RETURN(WaitProcessWideKeyAtomic(system, address, cv_key, tag, timeout_ns));
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
void SignalProcessWideKey64(Core::System& system, uint64_t cv_key, int32_t count) {
 | 
			
		||||
    SignalProcessWideKey(system, cv_key, count);
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
Result WaitProcessWideKeyAtomic64From32(Core::System& system, uint32_t address, uint32_t cv_key,
 | 
			
		||||
                                        uint32_t tag, int64_t timeout_ns) {
 | 
			
		||||
    R_RETURN(WaitProcessWideKeyAtomic(system, address, cv_key, tag, timeout_ns));
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
void SignalProcessWideKey64From32(Core::System& system, uint32_t cv_key, int32_t count) {
 | 
			
		||||
    SignalProcessWideKey(system, cv_key, count);
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
 
 | 
			
		||||
@@ -2,5 +2,193 @@
 | 
			
		||||
// SPDX-License-Identifier: GPL-2.0-or-later
 | 
			
		||||
 | 
			
		||||
#include "core/hle/kernel/svc.h"
 | 
			
		||||
#include "core/hle/kernel/svc_results.h"
 | 
			
		||||
 | 
			
		||||
namespace Kernel::Svc {} // namespace Kernel::Svc
 | 
			
		||||
namespace Kernel::Svc {
 | 
			
		||||
 | 
			
		||||
Result DebugActiveProcess(Core::System& system, Handle* out_handle, uint64_t process_id) {
 | 
			
		||||
    UNIMPLEMENTED();
 | 
			
		||||
    R_THROW(ResultNotImplemented);
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
Result BreakDebugProcess(Core::System& system, Handle debug_handle) {
 | 
			
		||||
    UNIMPLEMENTED();
 | 
			
		||||
    R_THROW(ResultNotImplemented);
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
Result TerminateDebugProcess(Core::System& system, Handle debug_handle) {
 | 
			
		||||
    UNIMPLEMENTED();
 | 
			
		||||
    R_THROW(ResultNotImplemented);
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
Result GetDebugEvent(Core::System& system, uint64_t out_info, Handle debug_handle) {
 | 
			
		||||
    UNIMPLEMENTED();
 | 
			
		||||
    R_THROW(ResultNotImplemented);
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
Result ContinueDebugEvent(Core::System& system, Handle debug_handle, uint32_t flags,
 | 
			
		||||
                          uint64_t user_thread_ids, int32_t num_thread_ids) {
 | 
			
		||||
    UNIMPLEMENTED();
 | 
			
		||||
    R_THROW(ResultNotImplemented);
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
Result GetDebugThreadContext(Core::System& system, uint64_t out_context, Handle debug_handle,
 | 
			
		||||
                             uint64_t thread_id, uint32_t context_flags) {
 | 
			
		||||
    UNIMPLEMENTED();
 | 
			
		||||
    R_THROW(ResultNotImplemented);
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
Result SetDebugThreadContext(Core::System& system, Handle debug_handle, uint64_t thread_id,
 | 
			
		||||
                             uint64_t user_context, uint32_t context_flags) {
 | 
			
		||||
    UNIMPLEMENTED();
 | 
			
		||||
    R_THROW(ResultNotImplemented);
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
Result QueryDebugProcessMemory(Core::System& system, uint64_t out_memory_info,
 | 
			
		||||
                               PageInfo* out_page_info, Handle debug_handle, uintptr_t address) {
 | 
			
		||||
    UNIMPLEMENTED();
 | 
			
		||||
    R_THROW(ResultNotImplemented);
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
Result ReadDebugProcessMemory(Core::System& system, uintptr_t buffer, Handle debug_handle,
 | 
			
		||||
                              uintptr_t address, size_t size) {
 | 
			
		||||
    UNIMPLEMENTED();
 | 
			
		||||
    R_THROW(ResultNotImplemented);
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
Result WriteDebugProcessMemory(Core::System& system, Handle debug_handle, uintptr_t buffer,
 | 
			
		||||
                               uintptr_t address, size_t size) {
 | 
			
		||||
    UNIMPLEMENTED();
 | 
			
		||||
    R_THROW(ResultNotImplemented);
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
Result SetHardwareBreakPoint(Core::System& system, HardwareBreakPointRegisterName name,
 | 
			
		||||
                             uint64_t flags, uint64_t value) {
 | 
			
		||||
    UNIMPLEMENTED();
 | 
			
		||||
    R_THROW(ResultNotImplemented);
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
Result GetDebugThreadParam(Core::System& system, uint64_t* out_64, uint32_t* out_32,
 | 
			
		||||
                           Handle debug_handle, uint64_t thread_id, DebugThreadParam param) {
 | 
			
		||||
    UNIMPLEMENTED();
 | 
			
		||||
    R_THROW(ResultNotImplemented);
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
Result DebugActiveProcess64(Core::System& system, Handle* out_handle, uint64_t process_id) {
 | 
			
		||||
    R_RETURN(DebugActiveProcess(system, out_handle, process_id));
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
Result BreakDebugProcess64(Core::System& system, Handle debug_handle) {
 | 
			
		||||
    R_RETURN(BreakDebugProcess(system, debug_handle));
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
Result TerminateDebugProcess64(Core::System& system, Handle debug_handle) {
 | 
			
		||||
    R_RETURN(TerminateDebugProcess(system, debug_handle));
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
Result GetDebugEvent64(Core::System& system, uint64_t out_info, Handle debug_handle) {
 | 
			
		||||
    R_RETURN(GetDebugEvent(system, out_info, debug_handle));
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
Result ContinueDebugEvent64(Core::System& system, Handle debug_handle, uint32_t flags,
 | 
			
		||||
                            uint64_t thread_ids, int32_t num_thread_ids) {
 | 
			
		||||
    R_RETURN(ContinueDebugEvent(system, debug_handle, flags, thread_ids, num_thread_ids));
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
Result GetDebugThreadContext64(Core::System& system, uint64_t out_context, Handle debug_handle,
 | 
			
		||||
                               uint64_t thread_id, uint32_t context_flags) {
 | 
			
		||||
    R_RETURN(GetDebugThreadContext(system, out_context, debug_handle, thread_id, context_flags));
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
Result SetDebugThreadContext64(Core::System& system, Handle debug_handle, uint64_t thread_id,
 | 
			
		||||
                               uint64_t context, uint32_t context_flags) {
 | 
			
		||||
    R_RETURN(SetDebugThreadContext(system, debug_handle, thread_id, context, context_flags));
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
Result QueryDebugProcessMemory64(Core::System& system, uint64_t out_memory_info,
 | 
			
		||||
                                 PageInfo* out_page_info, Handle debug_handle, uint64_t address) {
 | 
			
		||||
    R_RETURN(
 | 
			
		||||
        QueryDebugProcessMemory(system, out_memory_info, out_page_info, debug_handle, address));
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
Result ReadDebugProcessMemory64(Core::System& system, uint64_t buffer, Handle debug_handle,
 | 
			
		||||
                                uint64_t address, uint64_t size) {
 | 
			
		||||
    R_RETURN(ReadDebugProcessMemory(system, buffer, debug_handle, address, size));
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
Result WriteDebugProcessMemory64(Core::System& system, Handle debug_handle, uint64_t buffer,
 | 
			
		||||
                                 uint64_t address, uint64_t size) {
 | 
			
		||||
    R_RETURN(WriteDebugProcessMemory(system, debug_handle, buffer, address, size));
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
Result SetHardwareBreakPoint64(Core::System& system, HardwareBreakPointRegisterName name,
 | 
			
		||||
                               uint64_t flags, uint64_t value) {
 | 
			
		||||
    R_RETURN(SetHardwareBreakPoint(system, name, flags, value));
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
Result GetDebugThreadParam64(Core::System& system, uint64_t* out_64, uint32_t* out_32,
 | 
			
		||||
                             Handle debug_handle, uint64_t thread_id, DebugThreadParam param) {
 | 
			
		||||
    R_RETURN(GetDebugThreadParam(system, out_64, out_32, debug_handle, thread_id, param));
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
Result DebugActiveProcess64From32(Core::System& system, Handle* out_handle, uint64_t process_id) {
 | 
			
		||||
    R_RETURN(DebugActiveProcess(system, out_handle, process_id));
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
Result BreakDebugProcess64From32(Core::System& system, Handle debug_handle) {
 | 
			
		||||
    R_RETURN(BreakDebugProcess(system, debug_handle));
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
Result TerminateDebugProcess64From32(Core::System& system, Handle debug_handle) {
 | 
			
		||||
    R_RETURN(TerminateDebugProcess(system, debug_handle));
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
Result GetDebugEvent64From32(Core::System& system, uint32_t out_info, Handle debug_handle) {
 | 
			
		||||
    R_RETURN(GetDebugEvent(system, out_info, debug_handle));
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
Result ContinueDebugEvent64From32(Core::System& system, Handle debug_handle, uint32_t flags,
 | 
			
		||||
                                  uint32_t thread_ids, int32_t num_thread_ids) {
 | 
			
		||||
    R_RETURN(ContinueDebugEvent(system, debug_handle, flags, thread_ids, num_thread_ids));
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
Result GetDebugThreadContext64From32(Core::System& system, uint32_t out_context,
 | 
			
		||||
                                     Handle debug_handle, uint64_t thread_id,
 | 
			
		||||
                                     uint32_t context_flags) {
 | 
			
		||||
    R_RETURN(GetDebugThreadContext(system, out_context, debug_handle, thread_id, context_flags));
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
Result SetDebugThreadContext64From32(Core::System& system, Handle debug_handle, uint64_t thread_id,
 | 
			
		||||
                                     uint32_t context, uint32_t context_flags) {
 | 
			
		||||
    R_RETURN(SetDebugThreadContext(system, debug_handle, thread_id, context, context_flags));
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
Result QueryDebugProcessMemory64From32(Core::System& system, uint32_t out_memory_info,
 | 
			
		||||
                                       PageInfo* out_page_info, Handle debug_handle,
 | 
			
		||||
                                       uint32_t address) {
 | 
			
		||||
    R_RETURN(
 | 
			
		||||
        QueryDebugProcessMemory(system, out_memory_info, out_page_info, debug_handle, address));
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
Result ReadDebugProcessMemory64From32(Core::System& system, uint32_t buffer, Handle debug_handle,
 | 
			
		||||
                                      uint32_t address, uint32_t size) {
 | 
			
		||||
    R_RETURN(ReadDebugProcessMemory(system, buffer, debug_handle, address, size));
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
Result WriteDebugProcessMemory64From32(Core::System& system, Handle debug_handle, uint32_t buffer,
 | 
			
		||||
                                       uint32_t address, uint32_t size) {
 | 
			
		||||
    R_RETURN(WriteDebugProcessMemory(system, debug_handle, buffer, address, size));
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
Result SetHardwareBreakPoint64From32(Core::System& system, HardwareBreakPointRegisterName name,
 | 
			
		||||
                                     uint64_t flags, uint64_t value) {
 | 
			
		||||
    R_RETURN(SetHardwareBreakPoint(system, name, flags, value));
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
Result GetDebugThreadParam64From32(Core::System& system, uint64_t* out_64, uint32_t* out_32,
 | 
			
		||||
                                   Handle debug_handle, uint64_t thread_id,
 | 
			
		||||
                                   DebugThreadParam param) {
 | 
			
		||||
    R_RETURN(GetDebugThreadParam(system, out_64, out_32, debug_handle, thread_id, param));
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
} // namespace Kernel::Svc
 | 
			
		||||
 
 | 
			
		||||
@@ -8,18 +8,22 @@
 | 
			
		||||
namespace Kernel::Svc {
 | 
			
		||||
 | 
			
		||||
/// Used to output a message on a debug hardware unit - does nothing on a retail unit
 | 
			
		||||
void OutputDebugString(Core::System& system, VAddr address, u64 len) {
 | 
			
		||||
    if (len == 0) {
 | 
			
		||||
        return;
 | 
			
		||||
    }
 | 
			
		||||
Result OutputDebugString(Core::System& system, VAddr address, u64 len) {
 | 
			
		||||
    R_SUCCEED_IF(len == 0);
 | 
			
		||||
 | 
			
		||||
    std::string str(len, '\0');
 | 
			
		||||
    system.Memory().ReadBlock(address, str.data(), str.size());
 | 
			
		||||
    LOG_DEBUG(Debug_Emulated, "{}", str);
 | 
			
		||||
 | 
			
		||||
    R_SUCCEED();
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
void OutputDebugString32(Core::System& system, u32 address, u32 len) {
 | 
			
		||||
    OutputDebugString(system, address, len);
 | 
			
		||||
Result OutputDebugString64(Core::System& system, uint64_t debug_str, uint64_t len) {
 | 
			
		||||
    R_RETURN(OutputDebugString(system, debug_str, len));
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
Result OutputDebugString64From32(Core::System& system, uint32_t debug_str, uint32_t len) {
 | 
			
		||||
    R_RETURN(OutputDebugString(system, debug_str, len));
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
} // namespace Kernel::Svc
 | 
			
		||||
 
 | 
			
		||||
@@ -1,6 +1,253 @@
 | 
			
		||||
// SPDX-FileCopyrightText: Copyright 2023 yuzu Emulator Project
 | 
			
		||||
// SPDX-License-Identifier: GPL-2.0-or-later
 | 
			
		||||
 | 
			
		||||
#include "common/alignment.h"
 | 
			
		||||
#include "common/scope_exit.h"
 | 
			
		||||
#include "core/core.h"
 | 
			
		||||
#include "core/hle/kernel/k_device_address_space.h"
 | 
			
		||||
#include "core/hle/kernel/k_process.h"
 | 
			
		||||
#include "core/hle/kernel/svc.h"
 | 
			
		||||
 | 
			
		||||
namespace Kernel::Svc {} // namespace Kernel::Svc
 | 
			
		||||
namespace Kernel::Svc {
 | 
			
		||||
 | 
			
		||||
constexpr inline u64 DeviceAddressSpaceAlignMask = (1ULL << 22) - 1;
 | 
			
		||||
 | 
			
		||||
constexpr bool IsProcessAndDeviceAligned(uint64_t process_address, uint64_t device_address) {
 | 
			
		||||
    return (process_address & DeviceAddressSpaceAlignMask) ==
 | 
			
		||||
           (device_address & DeviceAddressSpaceAlignMask);
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
Result CreateDeviceAddressSpace(Core::System& system, Handle* out, uint64_t das_address,
 | 
			
		||||
                                uint64_t das_size) {
 | 
			
		||||
    // Validate input.
 | 
			
		||||
    R_UNLESS(Common::IsAligned(das_address, PageSize), ResultInvalidMemoryRegion);
 | 
			
		||||
    R_UNLESS(Common::IsAligned(das_size, PageSize), ResultInvalidMemoryRegion);
 | 
			
		||||
    R_UNLESS(das_size > 0, ResultInvalidMemoryRegion);
 | 
			
		||||
    R_UNLESS((das_address < das_address + das_size), ResultInvalidMemoryRegion);
 | 
			
		||||
 | 
			
		||||
    // Create the device address space.
 | 
			
		||||
    KDeviceAddressSpace* das = KDeviceAddressSpace::Create(system.Kernel());
 | 
			
		||||
    R_UNLESS(das != nullptr, ResultOutOfResource);
 | 
			
		||||
    SCOPE_EXIT({ das->Close(); });
 | 
			
		||||
 | 
			
		||||
    // Initialize the device address space.
 | 
			
		||||
    R_TRY(das->Initialize(das_address, das_size));
 | 
			
		||||
 | 
			
		||||
    // Register the device address space.
 | 
			
		||||
    KDeviceAddressSpace::Register(system.Kernel(), das);
 | 
			
		||||
 | 
			
		||||
    // Add to the handle table.
 | 
			
		||||
    R_TRY(system.CurrentProcess()->GetHandleTable().Add(out, das));
 | 
			
		||||
 | 
			
		||||
    R_SUCCEED();
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
Result AttachDeviceAddressSpace(Core::System& system, DeviceName device_name, Handle das_handle) {
 | 
			
		||||
    // Get the device address space.
 | 
			
		||||
    KScopedAutoObject das =
 | 
			
		||||
        system.CurrentProcess()->GetHandleTable().GetObject<KDeviceAddressSpace>(das_handle);
 | 
			
		||||
    R_UNLESS(das.IsNotNull(), ResultInvalidHandle);
 | 
			
		||||
 | 
			
		||||
    // Attach.
 | 
			
		||||
    R_RETURN(das->Attach(device_name));
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
Result DetachDeviceAddressSpace(Core::System& system, DeviceName device_name, Handle das_handle) {
 | 
			
		||||
    // Get the device address space.
 | 
			
		||||
    KScopedAutoObject das =
 | 
			
		||||
        system.CurrentProcess()->GetHandleTable().GetObject<KDeviceAddressSpace>(das_handle);
 | 
			
		||||
    R_UNLESS(das.IsNotNull(), ResultInvalidHandle);
 | 
			
		||||
 | 
			
		||||
    // Detach.
 | 
			
		||||
    R_RETURN(das->Detach(device_name));
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
constexpr bool IsValidDeviceMemoryPermission(MemoryPermission device_perm) {
 | 
			
		||||
    switch (device_perm) {
 | 
			
		||||
    case MemoryPermission::Read:
 | 
			
		||||
    case MemoryPermission::Write:
 | 
			
		||||
    case MemoryPermission::ReadWrite:
 | 
			
		||||
        return true;
 | 
			
		||||
    default:
 | 
			
		||||
        return false;
 | 
			
		||||
    }
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
Result MapDeviceAddressSpaceByForce(Core::System& system, Handle das_handle, Handle process_handle,
 | 
			
		||||
                                    uint64_t process_address, size_t size, uint64_t device_address,
 | 
			
		||||
                                    u32 option) {
 | 
			
		||||
    // Decode the option.
 | 
			
		||||
    const MapDeviceAddressSpaceOption option_pack{option};
 | 
			
		||||
    const auto device_perm = option_pack.permission;
 | 
			
		||||
    const auto reserved = option_pack.reserved;
 | 
			
		||||
 | 
			
		||||
    // Validate input.
 | 
			
		||||
    R_UNLESS(Common::IsAligned(process_address, PageSize), ResultInvalidAddress);
 | 
			
		||||
    R_UNLESS(Common::IsAligned(device_address, PageSize), ResultInvalidAddress);
 | 
			
		||||
    R_UNLESS(Common::IsAligned(size, PageSize), ResultInvalidSize);
 | 
			
		||||
    R_UNLESS(size > 0, ResultInvalidSize);
 | 
			
		||||
    R_UNLESS((process_address < process_address + size), ResultInvalidCurrentMemory);
 | 
			
		||||
    R_UNLESS((device_address < device_address + size), ResultInvalidMemoryRegion);
 | 
			
		||||
    R_UNLESS((process_address == static_cast<uintptr_t>(process_address)),
 | 
			
		||||
             ResultInvalidCurrentMemory);
 | 
			
		||||
    R_UNLESS(IsValidDeviceMemoryPermission(device_perm), ResultInvalidNewMemoryPermission);
 | 
			
		||||
    R_UNLESS(reserved == 0, ResultInvalidEnumValue);
 | 
			
		||||
 | 
			
		||||
    // Get the device address space.
 | 
			
		||||
    KScopedAutoObject das =
 | 
			
		||||
        system.CurrentProcess()->GetHandleTable().GetObject<KDeviceAddressSpace>(das_handle);
 | 
			
		||||
    R_UNLESS(das.IsNotNull(), ResultInvalidHandle);
 | 
			
		||||
 | 
			
		||||
    // Get the process.
 | 
			
		||||
    KScopedAutoObject process =
 | 
			
		||||
        system.CurrentProcess()->GetHandleTable().GetObject<KProcess>(process_handle);
 | 
			
		||||
    R_UNLESS(process.IsNotNull(), ResultInvalidHandle);
 | 
			
		||||
 | 
			
		||||
    // Validate that the process address is within range.
 | 
			
		||||
    auto& page_table = process->PageTable();
 | 
			
		||||
    R_UNLESS(page_table.Contains(process_address, size), ResultInvalidCurrentMemory);
 | 
			
		||||
 | 
			
		||||
    // Map.
 | 
			
		||||
    R_RETURN(
 | 
			
		||||
        das->MapByForce(std::addressof(page_table), process_address, size, device_address, option));
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
Result MapDeviceAddressSpaceAligned(Core::System& system, Handle das_handle, Handle process_handle,
 | 
			
		||||
                                    uint64_t process_address, size_t size, uint64_t device_address,
 | 
			
		||||
                                    u32 option) {
 | 
			
		||||
    // Decode the option.
 | 
			
		||||
    const MapDeviceAddressSpaceOption option_pack{option};
 | 
			
		||||
    const auto device_perm = option_pack.permission;
 | 
			
		||||
    const auto reserved = option_pack.reserved;
 | 
			
		||||
 | 
			
		||||
    // Validate input.
 | 
			
		||||
    R_UNLESS(Common::IsAligned(process_address, PageSize), ResultInvalidAddress);
 | 
			
		||||
    R_UNLESS(Common::IsAligned(device_address, PageSize), ResultInvalidAddress);
 | 
			
		||||
    R_UNLESS(IsProcessAndDeviceAligned(process_address, device_address), ResultInvalidAddress);
 | 
			
		||||
    R_UNLESS(Common::IsAligned(size, PageSize), ResultInvalidSize);
 | 
			
		||||
    R_UNLESS(size > 0, ResultInvalidSize);
 | 
			
		||||
    R_UNLESS((process_address < process_address + size), ResultInvalidCurrentMemory);
 | 
			
		||||
    R_UNLESS((device_address < device_address + size), ResultInvalidMemoryRegion);
 | 
			
		||||
    R_UNLESS((process_address == static_cast<uintptr_t>(process_address)),
 | 
			
		||||
             ResultInvalidCurrentMemory);
 | 
			
		||||
    R_UNLESS(IsValidDeviceMemoryPermission(device_perm), ResultInvalidNewMemoryPermission);
 | 
			
		||||
    R_UNLESS(reserved == 0, ResultInvalidEnumValue);
 | 
			
		||||
 | 
			
		||||
    // Get the device address space.
 | 
			
		||||
    KScopedAutoObject das =
 | 
			
		||||
        system.CurrentProcess()->GetHandleTable().GetObject<KDeviceAddressSpace>(das_handle);
 | 
			
		||||
    R_UNLESS(das.IsNotNull(), ResultInvalidHandle);
 | 
			
		||||
 | 
			
		||||
    // Get the process.
 | 
			
		||||
    KScopedAutoObject process =
 | 
			
		||||
        system.CurrentProcess()->GetHandleTable().GetObject<KProcess>(process_handle);
 | 
			
		||||
    R_UNLESS(process.IsNotNull(), ResultInvalidHandle);
 | 
			
		||||
 | 
			
		||||
    // Validate that the process address is within range.
 | 
			
		||||
    auto& page_table = process->PageTable();
 | 
			
		||||
    R_UNLESS(page_table.Contains(process_address, size), ResultInvalidCurrentMemory);
 | 
			
		||||
 | 
			
		||||
    // Map.
 | 
			
		||||
    R_RETURN(
 | 
			
		||||
        das->MapAligned(std::addressof(page_table), process_address, size, device_address, option));
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
Result UnmapDeviceAddressSpace(Core::System& system, Handle das_handle, Handle process_handle,
 | 
			
		||||
                               uint64_t process_address, size_t size, uint64_t device_address) {
 | 
			
		||||
    // Validate input.
 | 
			
		||||
    R_UNLESS(Common::IsAligned(process_address, PageSize), ResultInvalidAddress);
 | 
			
		||||
    R_UNLESS(Common::IsAligned(device_address, PageSize), ResultInvalidAddress);
 | 
			
		||||
    R_UNLESS(Common::IsAligned(size, PageSize), ResultInvalidSize);
 | 
			
		||||
    R_UNLESS(size > 0, ResultInvalidSize);
 | 
			
		||||
    R_UNLESS((process_address < process_address + size), ResultInvalidCurrentMemory);
 | 
			
		||||
    R_UNLESS((device_address < device_address + size), ResultInvalidMemoryRegion);
 | 
			
		||||
    R_UNLESS((process_address == static_cast<uintptr_t>(process_address)),
 | 
			
		||||
             ResultInvalidCurrentMemory);
 | 
			
		||||
 | 
			
		||||
    // Get the device address space.
 | 
			
		||||
    KScopedAutoObject das =
 | 
			
		||||
        system.CurrentProcess()->GetHandleTable().GetObject<KDeviceAddressSpace>(das_handle);
 | 
			
		||||
    R_UNLESS(das.IsNotNull(), ResultInvalidHandle);
 | 
			
		||||
 | 
			
		||||
    // Get the process.
 | 
			
		||||
    KScopedAutoObject process =
 | 
			
		||||
        system.CurrentProcess()->GetHandleTable().GetObject<KProcess>(process_handle);
 | 
			
		||||
    R_UNLESS(process.IsNotNull(), ResultInvalidHandle);
 | 
			
		||||
 | 
			
		||||
    // Validate that the process address is within range.
 | 
			
		||||
    auto& page_table = process->PageTable();
 | 
			
		||||
    R_UNLESS(page_table.Contains(process_address, size), ResultInvalidCurrentMemory);
 | 
			
		||||
 | 
			
		||||
    R_RETURN(das->Unmap(std::addressof(page_table), process_address, size, device_address));
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
Result CreateDeviceAddressSpace64(Core::System& system, Handle* out_handle, uint64_t das_address,
 | 
			
		||||
                                  uint64_t das_size) {
 | 
			
		||||
    R_RETURN(CreateDeviceAddressSpace(system, out_handle, das_address, das_size));
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
Result AttachDeviceAddressSpace64(Core::System& system, DeviceName device_name, Handle das_handle) {
 | 
			
		||||
    R_RETURN(AttachDeviceAddressSpace(system, device_name, das_handle));
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
Result DetachDeviceAddressSpace64(Core::System& system, DeviceName device_name, Handle das_handle) {
 | 
			
		||||
    R_RETURN(DetachDeviceAddressSpace(system, device_name, das_handle));
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
Result MapDeviceAddressSpaceByForce64(Core::System& system, Handle das_handle,
 | 
			
		||||
                                      Handle process_handle, uint64_t process_address,
 | 
			
		||||
                                      uint64_t size, uint64_t device_address, u32 option) {
 | 
			
		||||
    R_RETURN(MapDeviceAddressSpaceByForce(system, das_handle, process_handle, process_address, size,
 | 
			
		||||
                                          device_address, option));
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
Result MapDeviceAddressSpaceAligned64(Core::System& system, Handle das_handle,
 | 
			
		||||
                                      Handle process_handle, uint64_t process_address,
 | 
			
		||||
                                      uint64_t size, uint64_t device_address, u32 option) {
 | 
			
		||||
    R_RETURN(MapDeviceAddressSpaceAligned(system, das_handle, process_handle, process_address, size,
 | 
			
		||||
                                          device_address, option));
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
Result UnmapDeviceAddressSpace64(Core::System& system, Handle das_handle, Handle process_handle,
 | 
			
		||||
                                 uint64_t process_address, uint64_t size, uint64_t device_address) {
 | 
			
		||||
    R_RETURN(UnmapDeviceAddressSpace(system, das_handle, process_handle, process_address, size,
 | 
			
		||||
                                     device_address));
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
Result CreateDeviceAddressSpace64From32(Core::System& system, Handle* out_handle,
 | 
			
		||||
                                        uint64_t das_address, uint64_t das_size) {
 | 
			
		||||
    R_RETURN(CreateDeviceAddressSpace(system, out_handle, das_address, das_size));
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
Result AttachDeviceAddressSpace64From32(Core::System& system, DeviceName device_name,
 | 
			
		||||
                                        Handle das_handle) {
 | 
			
		||||
    R_RETURN(AttachDeviceAddressSpace(system, device_name, das_handle));
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
Result DetachDeviceAddressSpace64From32(Core::System& system, DeviceName device_name,
 | 
			
		||||
                                        Handle das_handle) {
 | 
			
		||||
    R_RETURN(DetachDeviceAddressSpace(system, device_name, das_handle));
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
Result MapDeviceAddressSpaceByForce64From32(Core::System& system, Handle das_handle,
 | 
			
		||||
                                            Handle process_handle, uint64_t process_address,
 | 
			
		||||
                                            uint32_t size, uint64_t device_address, u32 option) {
 | 
			
		||||
    R_RETURN(MapDeviceAddressSpaceByForce(system, das_handle, process_handle, process_address, size,
 | 
			
		||||
                                          device_address, option));
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
Result MapDeviceAddressSpaceAligned64From32(Core::System& system, Handle das_handle,
 | 
			
		||||
                                            Handle process_handle, uint64_t process_address,
 | 
			
		||||
                                            uint32_t size, uint64_t device_address, u32 option) {
 | 
			
		||||
    R_RETURN(MapDeviceAddressSpaceAligned(system, das_handle, process_handle, process_address, size,
 | 
			
		||||
                                          device_address, option));
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
Result UnmapDeviceAddressSpace64From32(Core::System& system, Handle das_handle,
 | 
			
		||||
                                       Handle process_handle, uint64_t process_address,
 | 
			
		||||
                                       uint32_t size, uint64_t device_address) {
 | 
			
		||||
    R_RETURN(UnmapDeviceAddressSpace(system, das_handle, process_handle, process_address, size,
 | 
			
		||||
                                     device_address));
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
} // namespace Kernel::Svc
 | 
			
		||||
 
 | 
			
		||||
@@ -24,10 +24,6 @@ Result SignalEvent(Core::System& system, Handle event_handle) {
 | 
			
		||||
    return event->Signal();
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
Result SignalEvent32(Core::System& system, Handle event_handle) {
 | 
			
		||||
    return SignalEvent(system, event_handle);
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
Result ClearEvent(Core::System& system, Handle event_handle) {
 | 
			
		||||
    LOG_TRACE(Kernel_SVC, "called, event_handle=0x{:08X}", event_handle);
 | 
			
		||||
 | 
			
		||||
@@ -55,10 +51,6 @@ Result ClearEvent(Core::System& system, Handle event_handle) {
 | 
			
		||||
    return ResultInvalidHandle;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
Result ClearEvent32(Core::System& system, Handle event_handle) {
 | 
			
		||||
    return ClearEvent(system, event_handle);
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
Result CreateEvent(Core::System& system, Handle* out_write, Handle* out_read) {
 | 
			
		||||
    LOG_DEBUG(Kernel_SVC, "called");
 | 
			
		||||
 | 
			
		||||
@@ -104,8 +96,29 @@ Result CreateEvent(Core::System& system, Handle* out_write, Handle* out_read) {
 | 
			
		||||
    return ResultSuccess;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
Result CreateEvent32(Core::System& system, Handle* out_write, Handle* out_read) {
 | 
			
		||||
    return CreateEvent(system, out_write, out_read);
 | 
			
		||||
Result SignalEvent64(Core::System& system, Handle event_handle) {
 | 
			
		||||
    R_RETURN(SignalEvent(system, event_handle));
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
Result ClearEvent64(Core::System& system, Handle event_handle) {
 | 
			
		||||
    R_RETURN(ClearEvent(system, event_handle));
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
Result CreateEvent64(Core::System& system, Handle* out_write_handle, Handle* out_read_handle) {
 | 
			
		||||
    R_RETURN(CreateEvent(system, out_write_handle, out_read_handle));
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
Result SignalEvent64From32(Core::System& system, Handle event_handle) {
 | 
			
		||||
    R_RETURN(SignalEvent(system, event_handle));
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
Result ClearEvent64From32(Core::System& system, Handle event_handle) {
 | 
			
		||||
    R_RETURN(ClearEvent(system, event_handle));
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
Result CreateEvent64From32(Core::System& system, Handle* out_write_handle,
 | 
			
		||||
                           Handle* out_read_handle) {
 | 
			
		||||
    R_RETURN(CreateEvent(system, out_write_handle, out_read_handle));
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
} // namespace Kernel::Svc
 | 
			
		||||
 
 | 
			
		||||
@@ -12,10 +12,10 @@
 | 
			
		||||
namespace Kernel::Svc {
 | 
			
		||||
 | 
			
		||||
/// Break program execution
 | 
			
		||||
void Break(Core::System& system, u32 reason, u64 info1, u64 info2) {
 | 
			
		||||
void Break(Core::System& system, BreakReason reason, u64 info1, u64 info2) {
 | 
			
		||||
    BreakReason break_reason =
 | 
			
		||||
        static_cast<BreakReason>(reason & ~static_cast<u32>(BreakReason::NotificationOnlyFlag));
 | 
			
		||||
    bool notification_only = (reason & static_cast<u32>(BreakReason::NotificationOnlyFlag)) != 0;
 | 
			
		||||
        reason & static_cast<BreakReason>(~BreakReason::NotificationOnlyFlag);
 | 
			
		||||
    bool notification_only = True(reason & BreakReason::NotificationOnlyFlag);
 | 
			
		||||
 | 
			
		||||
    bool has_dumped_buffer{};
 | 
			
		||||
    std::vector<u8> debug_buffer;
 | 
			
		||||
@@ -90,9 +90,9 @@ void Break(Core::System& system, u32 reason, u64 info1, u64 info2) {
 | 
			
		||||
        break;
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    system.GetReporter().SaveSvcBreakReport(reason, notification_only, info1, info2,
 | 
			
		||||
                                            has_dumped_buffer ? std::make_optional(debug_buffer)
 | 
			
		||||
                                                              : std::nullopt);
 | 
			
		||||
    system.GetReporter().SaveSvcBreakReport(
 | 
			
		||||
        static_cast<u32>(reason), notification_only, info1, info2,
 | 
			
		||||
        has_dumped_buffer ? std::make_optional(debug_buffer) : std::nullopt);
 | 
			
		||||
 | 
			
		||||
    if (!notification_only) {
 | 
			
		||||
        LOG_CRITICAL(
 | 
			
		||||
@@ -114,8 +114,24 @@ void Break(Core::System& system, u32 reason, u64 info1, u64 info2) {
 | 
			
		||||
    }
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
void Break32(Core::System& system, u32 reason, u32 info1, u32 info2) {
 | 
			
		||||
    Break(system, reason, info1, info2);
 | 
			
		||||
void ReturnFromException(Core::System& system, Result result) {
 | 
			
		||||
    UNIMPLEMENTED();
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
void Break64(Core::System& system, BreakReason break_reason, uint64_t arg, uint64_t size) {
 | 
			
		||||
    Break(system, break_reason, arg, size);
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
void Break64From32(Core::System& system, BreakReason break_reason, uint32_t arg, uint32_t size) {
 | 
			
		||||
    Break(system, break_reason, arg, size);
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
void ReturnFromException64(Core::System& system, Result result) {
 | 
			
		||||
    ReturnFromException(system, result);
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
void ReturnFromException64From32(Core::System& system, Result result) {
 | 
			
		||||
    ReturnFromException(system, result);
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
} // namespace Kernel::Svc
 | 
			
		||||
 
 | 
			
		||||
@@ -10,11 +10,12 @@
 | 
			
		||||
namespace Kernel::Svc {
 | 
			
		||||
 | 
			
		||||
/// Gets system/memory information for the current process
 | 
			
		||||
Result GetInfo(Core::System& system, u64* result, u64 info_id, Handle handle, u64 info_sub_id) {
 | 
			
		||||
Result GetInfo(Core::System& system, u64* result, InfoType info_id_type, Handle handle,
 | 
			
		||||
               u64 info_sub_id) {
 | 
			
		||||
    LOG_TRACE(Kernel_SVC, "called info_id=0x{:X}, info_sub_id=0x{:X}, handle=0x{:08X}", info_id,
 | 
			
		||||
              info_sub_id, handle);
 | 
			
		||||
 | 
			
		||||
    const auto info_id_type = static_cast<InfoType>(info_id);
 | 
			
		||||
    u32 info_id = static_cast<u32>(info_id_type);
 | 
			
		||||
 | 
			
		||||
    switch (info_id_type) {
 | 
			
		||||
    case InfoType::CoreMask:
 | 
			
		||||
@@ -267,16 +268,30 @@ Result GetInfo(Core::System& system, u64* result, u64 info_id, Handle handle, u6
 | 
			
		||||
    }
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
Result GetInfo32(Core::System& system, u32* result_low, u32* result_high, u32 sub_id_low,
 | 
			
		||||
                 u32 info_id, u32 handle, u32 sub_id_high) {
 | 
			
		||||
    const u64 sub_id{u64{sub_id_low} | (u64{sub_id_high} << 32)};
 | 
			
		||||
    u64 res_value{};
 | 
			
		||||
Result GetSystemInfo(Core::System& system, uint64_t* out, SystemInfoType info_type, Handle handle,
 | 
			
		||||
                     uint64_t info_subtype) {
 | 
			
		||||
    UNIMPLEMENTED();
 | 
			
		||||
    R_THROW(ResultNotImplemented);
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
    const Result result{GetInfo(system, &res_value, info_id, handle, sub_id)};
 | 
			
		||||
    *result_high = static_cast<u32>(res_value >> 32);
 | 
			
		||||
    *result_low = static_cast<u32>(res_value & std::numeric_limits<u32>::max());
 | 
			
		||||
Result GetInfo64(Core::System& system, uint64_t* out, InfoType info_type, Handle handle,
 | 
			
		||||
                 uint64_t info_subtype) {
 | 
			
		||||
    R_RETURN(GetInfo(system, out, info_type, handle, info_subtype));
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
    return result;
 | 
			
		||||
Result GetSystemInfo64(Core::System& system, uint64_t* out, SystemInfoType info_type, Handle handle,
 | 
			
		||||
                       uint64_t info_subtype) {
 | 
			
		||||
    R_RETURN(GetSystemInfo(system, out, info_type, handle, info_subtype));
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
Result GetInfo64From32(Core::System& system, uint64_t* out, InfoType info_type, Handle handle,
 | 
			
		||||
                       uint64_t info_subtype) {
 | 
			
		||||
    R_RETURN(GetInfo(system, out, info_type, handle, info_subtype));
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
Result GetSystemInfo64From32(Core::System& system, uint64_t* out, SystemInfoType info_type,
 | 
			
		||||
                             Handle handle, uint64_t info_subtype) {
 | 
			
		||||
    R_RETURN(GetSystemInfo(system, out, info_type, handle, info_subtype));
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
} // namespace Kernel::Svc
 | 
			
		||||
 
 | 
			
		||||
							
								
								
									
										35
									
								
								src/core/hle/kernel/svc/svc_insecure_memory.cpp
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										35
									
								
								src/core/hle/kernel/svc/svc_insecure_memory.cpp
									
									
									
									
									
										Normal file
									
								
							@@ -0,0 +1,35 @@
 | 
			
		||||
// SPDX-FileCopyrightText: Copyright 2023 yuzu Emulator Project
 | 
			
		||||
// SPDX-License-Identifier: GPL-2.0-or-later
 | 
			
		||||
 | 
			
		||||
#include "core/hle/kernel/svc.h"
 | 
			
		||||
#include "core/hle/kernel/svc_results.h"
 | 
			
		||||
 | 
			
		||||
namespace Kernel::Svc {
 | 
			
		||||
 | 
			
		||||
Result MapInsecureMemory(Core::System& system, uintptr_t address, size_t size) {
 | 
			
		||||
    UNIMPLEMENTED();
 | 
			
		||||
    R_THROW(ResultNotImplemented);
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
Result UnmapInsecureMemory(Core::System& system, uintptr_t address, size_t size) {
 | 
			
		||||
    UNIMPLEMENTED();
 | 
			
		||||
    R_THROW(ResultNotImplemented);
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
Result MapInsecureMemory64(Core::System& system, uint64_t address, uint64_t size) {
 | 
			
		||||
    R_RETURN(MapInsecureMemory(system, address, size));
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
Result UnmapInsecureMemory64(Core::System& system, uint64_t address, uint64_t size) {
 | 
			
		||||
    R_RETURN(UnmapInsecureMemory(system, address, size));
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
Result MapInsecureMemory64From32(Core::System& system, uint32_t address, uint32_t size) {
 | 
			
		||||
    R_RETURN(MapInsecureMemory(system, address, size));
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
Result UnmapInsecureMemory64From32(Core::System& system, uint32_t address, uint32_t size) {
 | 
			
		||||
    R_RETURN(UnmapInsecureMemory(system, address, size));
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
} // namespace Kernel::Svc
 | 
			
		||||
@@ -2,5 +2,24 @@
 | 
			
		||||
// SPDX-License-Identifier: GPL-2.0-or-later
 | 
			
		||||
 | 
			
		||||
#include "core/hle/kernel/svc.h"
 | 
			
		||||
#include "core/hle/kernel/svc_results.h"
 | 
			
		||||
 | 
			
		||||
namespace Kernel::Svc {} // namespace Kernel::Svc
 | 
			
		||||
namespace Kernel::Svc {
 | 
			
		||||
 | 
			
		||||
Result CreateInterruptEvent(Core::System& system, Handle* out, int32_t interrupt_id,
 | 
			
		||||
                            InterruptType type) {
 | 
			
		||||
    UNIMPLEMENTED();
 | 
			
		||||
    R_THROW(ResultNotImplemented);
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
Result CreateInterruptEvent64(Core::System& system, Handle* out_read_handle, int32_t interrupt_id,
 | 
			
		||||
                              InterruptType interrupt_type) {
 | 
			
		||||
    R_RETURN(CreateInterruptEvent(system, out_read_handle, interrupt_id, interrupt_type));
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
Result CreateInterruptEvent64From32(Core::System& system, Handle* out_read_handle,
 | 
			
		||||
                                    int32_t interrupt_id, InterruptType interrupt_type) {
 | 
			
		||||
    R_RETURN(CreateInterruptEvent(system, out_read_handle, interrupt_id, interrupt_type));
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
} // namespace Kernel::Svc
 | 
			
		||||
 
 | 
			
		||||
@@ -2,5 +2,70 @@
 | 
			
		||||
// SPDX-License-Identifier: GPL-2.0-or-later
 | 
			
		||||
 | 
			
		||||
#include "core/hle/kernel/svc.h"
 | 
			
		||||
#include "core/hle/kernel/svc_results.h"
 | 
			
		||||
 | 
			
		||||
namespace Kernel::Svc {} // namespace Kernel::Svc
 | 
			
		||||
namespace Kernel::Svc {
 | 
			
		||||
 | 
			
		||||
Result CreateIoPool(Core::System& system, Handle* out, IoPoolType pool_type) {
 | 
			
		||||
    UNIMPLEMENTED();
 | 
			
		||||
    R_THROW(ResultNotImplemented);
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
Result CreateIoRegion(Core::System& system, Handle* out, Handle io_pool_handle, uint64_t phys_addr,
 | 
			
		||||
                      size_t size, MemoryMapping mapping, MemoryPermission perm) {
 | 
			
		||||
    UNIMPLEMENTED();
 | 
			
		||||
    R_THROW(ResultNotImplemented);
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
Result MapIoRegion(Core::System& system, Handle io_region_handle, uintptr_t address, size_t size,
 | 
			
		||||
                   MemoryPermission map_perm) {
 | 
			
		||||
    UNIMPLEMENTED();
 | 
			
		||||
    R_THROW(ResultNotImplemented);
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
Result UnmapIoRegion(Core::System& system, Handle io_region_handle, uintptr_t address,
 | 
			
		||||
                     size_t size) {
 | 
			
		||||
    UNIMPLEMENTED();
 | 
			
		||||
    R_THROW(ResultNotImplemented);
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
Result CreateIoPool64(Core::System& system, Handle* out_handle, IoPoolType pool_type) {
 | 
			
		||||
    R_RETURN(CreateIoPool(system, out_handle, pool_type));
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
Result CreateIoRegion64(Core::System& system, Handle* out_handle, Handle io_pool,
 | 
			
		||||
                        uint64_t physical_address, uint64_t size, MemoryMapping mapping,
 | 
			
		||||
                        MemoryPermission perm) {
 | 
			
		||||
    R_RETURN(CreateIoRegion(system, out_handle, io_pool, physical_address, size, mapping, perm));
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
Result MapIoRegion64(Core::System& system, Handle io_region, uint64_t address, uint64_t size,
 | 
			
		||||
                     MemoryPermission perm) {
 | 
			
		||||
    R_RETURN(MapIoRegion(system, io_region, address, size, perm));
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
Result UnmapIoRegion64(Core::System& system, Handle io_region, uint64_t address, uint64_t size) {
 | 
			
		||||
    R_RETURN(UnmapIoRegion(system, io_region, address, size));
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
Result CreateIoPool64From32(Core::System& system, Handle* out_handle, IoPoolType pool_type) {
 | 
			
		||||
    R_RETURN(CreateIoPool(system, out_handle, pool_type));
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
Result CreateIoRegion64From32(Core::System& system, Handle* out_handle, Handle io_pool,
 | 
			
		||||
                              uint64_t physical_address, uint32_t size, MemoryMapping mapping,
 | 
			
		||||
                              MemoryPermission perm) {
 | 
			
		||||
    R_RETURN(CreateIoRegion(system, out_handle, io_pool, physical_address, size, mapping, perm));
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
Result MapIoRegion64From32(Core::System& system, Handle io_region, uint32_t address, uint32_t size,
 | 
			
		||||
                           MemoryPermission perm) {
 | 
			
		||||
    R_RETURN(MapIoRegion(system, io_region, address, size, perm));
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
Result UnmapIoRegion64From32(Core::System& system, Handle io_region, uint32_t address,
 | 
			
		||||
                             uint32_t size) {
 | 
			
		||||
    R_RETURN(UnmapIoRegion(system, io_region, address, size));
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
} // namespace Kernel::Svc
 | 
			
		||||
 
 | 
			
		||||
@@ -24,20 +24,37 @@ Result SendSyncRequest(Core::System& system, Handle handle) {
 | 
			
		||||
    return session->SendSyncRequest();
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
Result SendSyncRequest32(Core::System& system, Handle handle) {
 | 
			
		||||
    return SendSyncRequest(system, handle);
 | 
			
		||||
Result SendSyncRequestWithUserBuffer(Core::System& system, uint64_t message_buffer,
 | 
			
		||||
                                     uint64_t message_buffer_size, Handle session_handle) {
 | 
			
		||||
    UNIMPLEMENTED();
 | 
			
		||||
    R_THROW(ResultNotImplemented);
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
Result ReplyAndReceive(Core::System& system, s32* out_index, Handle* handles, s32 num_handles,
 | 
			
		||||
Result SendAsyncRequestWithUserBuffer(Core::System& system, Handle* out_event_handle,
 | 
			
		||||
                                      uint64_t message_buffer, uint64_t message_buffer_size,
 | 
			
		||||
                                      Handle session_handle) {
 | 
			
		||||
    UNIMPLEMENTED();
 | 
			
		||||
    R_THROW(ResultNotImplemented);
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
Result ReplyAndReceive(Core::System& system, s32* out_index, uint64_t handles_addr, s32 num_handles,
 | 
			
		||||
                       Handle reply_target, s64 timeout_ns) {
 | 
			
		||||
    auto& kernel = system.Kernel();
 | 
			
		||||
    auto& handle_table = GetCurrentThread(kernel).GetOwnerProcess()->GetHandleTable();
 | 
			
		||||
 | 
			
		||||
    R_UNLESS(0 <= num_handles && num_handles <= ArgumentHandleCountMax, ResultOutOfRange);
 | 
			
		||||
    R_UNLESS(system.Memory().IsValidVirtualAddressRange(
 | 
			
		||||
                 handles_addr, static_cast<u64>(sizeof(Handle) * num_handles)),
 | 
			
		||||
             ResultInvalidPointer);
 | 
			
		||||
 | 
			
		||||
    std::vector<Handle> handles(num_handles);
 | 
			
		||||
    system.Memory().ReadBlock(handles_addr, handles.data(), sizeof(Handle) * num_handles);
 | 
			
		||||
 | 
			
		||||
    // Convert handle list to object table.
 | 
			
		||||
    std::vector<KSynchronizationObject*> objs(num_handles);
 | 
			
		||||
    R_UNLESS(
 | 
			
		||||
        handle_table.GetMultipleObjects<KSynchronizationObject>(objs.data(), handles, num_handles),
 | 
			
		||||
        ResultInvalidHandle);
 | 
			
		||||
    R_UNLESS(handle_table.GetMultipleObjects<KSynchronizationObject>(objs.data(), handles.data(),
 | 
			
		||||
                                                                     num_handles),
 | 
			
		||||
             ResultInvalidHandle);
 | 
			
		||||
 | 
			
		||||
    // Ensure handles are closed when we're done.
 | 
			
		||||
    SCOPE_EXIT({
 | 
			
		||||
@@ -86,4 +103,72 @@ Result ReplyAndReceive(Core::System& system, s32* out_index, Handle* handles, s3
 | 
			
		||||
    }
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
Result ReplyAndReceiveWithUserBuffer(Core::System& system, int32_t* out_index,
 | 
			
		||||
                                     uint64_t message_buffer, uint64_t message_buffer_size,
 | 
			
		||||
                                     uint64_t handles, int32_t num_handles, Handle reply_target,
 | 
			
		||||
                                     int64_t timeout_ns) {
 | 
			
		||||
    UNIMPLEMENTED();
 | 
			
		||||
    R_THROW(ResultNotImplemented);
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
Result SendSyncRequest64(Core::System& system, Handle session_handle) {
 | 
			
		||||
    R_RETURN(SendSyncRequest(system, session_handle));
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
Result SendSyncRequestWithUserBuffer64(Core::System& system, uint64_t message_buffer,
 | 
			
		||||
                                       uint64_t message_buffer_size, Handle session_handle) {
 | 
			
		||||
    R_RETURN(
 | 
			
		||||
        SendSyncRequestWithUserBuffer(system, message_buffer, message_buffer_size, session_handle));
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
Result SendAsyncRequestWithUserBuffer64(Core::System& system, Handle* out_event_handle,
 | 
			
		||||
                                        uint64_t message_buffer, uint64_t message_buffer_size,
 | 
			
		||||
                                        Handle session_handle) {
 | 
			
		||||
    R_RETURN(SendAsyncRequestWithUserBuffer(system, out_event_handle, message_buffer,
 | 
			
		||||
                                            message_buffer_size, session_handle));
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
Result ReplyAndReceive64(Core::System& system, int32_t* out_index, uint64_t handles,
 | 
			
		||||
                         int32_t num_handles, Handle reply_target, int64_t timeout_ns) {
 | 
			
		||||
    R_RETURN(ReplyAndReceive(system, out_index, handles, num_handles, reply_target, timeout_ns));
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
Result ReplyAndReceiveWithUserBuffer64(Core::System& system, int32_t* out_index,
 | 
			
		||||
                                       uint64_t message_buffer, uint64_t message_buffer_size,
 | 
			
		||||
                                       uint64_t handles, int32_t num_handles, Handle reply_target,
 | 
			
		||||
                                       int64_t timeout_ns) {
 | 
			
		||||
    R_RETURN(ReplyAndReceiveWithUserBuffer(system, out_index, message_buffer, message_buffer_size,
 | 
			
		||||
                                           handles, num_handles, reply_target, timeout_ns));
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
Result SendSyncRequest64From32(Core::System& system, Handle session_handle) {
 | 
			
		||||
    R_RETURN(SendSyncRequest(system, session_handle));
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
Result SendSyncRequestWithUserBuffer64From32(Core::System& system, uint32_t message_buffer,
 | 
			
		||||
                                             uint32_t message_buffer_size, Handle session_handle) {
 | 
			
		||||
    R_RETURN(
 | 
			
		||||
        SendSyncRequestWithUserBuffer(system, message_buffer, message_buffer_size, session_handle));
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
Result SendAsyncRequestWithUserBuffer64From32(Core::System& system, Handle* out_event_handle,
 | 
			
		||||
                                              uint32_t message_buffer, uint32_t message_buffer_size,
 | 
			
		||||
                                              Handle session_handle) {
 | 
			
		||||
    R_RETURN(SendAsyncRequestWithUserBuffer(system, out_event_handle, message_buffer,
 | 
			
		||||
                                            message_buffer_size, session_handle));
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
Result ReplyAndReceive64From32(Core::System& system, int32_t* out_index, uint32_t handles,
 | 
			
		||||
                               int32_t num_handles, Handle reply_target, int64_t timeout_ns) {
 | 
			
		||||
    R_RETURN(ReplyAndReceive(system, out_index, handles, num_handles, reply_target, timeout_ns));
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
Result ReplyAndReceiveWithUserBuffer64From32(Core::System& system, int32_t* out_index,
 | 
			
		||||
                                             uint32_t message_buffer, uint32_t message_buffer_size,
 | 
			
		||||
                                             uint32_t handles, int32_t num_handles,
 | 
			
		||||
                                             Handle reply_target, int64_t timeout_ns) {
 | 
			
		||||
    R_RETURN(ReplyAndReceiveWithUserBuffer(system, out_index, message_buffer, message_buffer_size,
 | 
			
		||||
                                           handles, num_handles, reply_target, timeout_ns));
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
} // namespace Kernel::Svc
 | 
			
		||||
 
 | 
			
		||||
@@ -5,15 +5,31 @@
 | 
			
		||||
 | 
			
		||||
namespace Kernel::Svc {
 | 
			
		||||
 | 
			
		||||
void KernelDebug([[maybe_unused]] Core::System& system, [[maybe_unused]] u32 kernel_debug_type,
 | 
			
		||||
                 [[maybe_unused]] u64 param1, [[maybe_unused]] u64 param2,
 | 
			
		||||
                 [[maybe_unused]] u64 param3) {
 | 
			
		||||
void KernelDebug(Core::System& system, KernelDebugType kernel_debug_type, u64 arg0, u64 arg1,
 | 
			
		||||
                 u64 arg2) {
 | 
			
		||||
    // Intentionally do nothing, as this does nothing in released kernel binaries.
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
void ChangeKernelTraceState([[maybe_unused]] Core::System& system,
 | 
			
		||||
                            [[maybe_unused]] u32 trace_state) {
 | 
			
		||||
void ChangeKernelTraceState(Core::System& system, KernelTraceState trace_state) {
 | 
			
		||||
    // Intentionally do nothing, as this does nothing in released kernel binaries.
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
void KernelDebug64(Core::System& system, KernelDebugType kern_debug_type, uint64_t arg0,
 | 
			
		||||
                   uint64_t arg1, uint64_t arg2) {
 | 
			
		||||
    KernelDebug(system, kern_debug_type, arg0, arg1, arg2);
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
void ChangeKernelTraceState64(Core::System& system, KernelTraceState kern_trace_state) {
 | 
			
		||||
    ChangeKernelTraceState(system, kern_trace_state);
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
void KernelDebug64From32(Core::System& system, KernelDebugType kern_debug_type, uint64_t arg0,
 | 
			
		||||
                         uint64_t arg1, uint64_t arg2) {
 | 
			
		||||
    KernelDebug(system, kern_debug_type, arg0, arg1, arg2);
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
void ChangeKernelTraceState64From32(Core::System& system, KernelTraceState kern_trace_state) {
 | 
			
		||||
    ChangeKernelTraceState(system, kern_trace_state);
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
} // namespace Kernel::Svc
 | 
			
		||||
 
 | 
			
		||||
@@ -1,6 +1,73 @@
 | 
			
		||||
// SPDX-FileCopyrightText: Copyright 2023 yuzu Emulator Project
 | 
			
		||||
// SPDX-License-Identifier: GPL-2.0-or-later
 | 
			
		||||
 | 
			
		||||
#include "core/arm/arm_interface.h"
 | 
			
		||||
#include "core/core.h"
 | 
			
		||||
#include "core/hle/kernel/svc.h"
 | 
			
		||||
#include "core/hle/kernel/svc_results.h"
 | 
			
		||||
 | 
			
		||||
namespace Kernel::Svc {} // namespace Kernel::Svc
 | 
			
		||||
namespace Kernel::Svc {
 | 
			
		||||
 | 
			
		||||
Result SendSyncRequestLight(Core::System& system, Handle session_handle, u32* args) {
 | 
			
		||||
    UNIMPLEMENTED();
 | 
			
		||||
    R_THROW(ResultNotImplemented);
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
Result ReplyAndReceiveLight(Core::System& system, Handle session_handle, u32* args) {
 | 
			
		||||
    UNIMPLEMENTED();
 | 
			
		||||
    R_THROW(ResultNotImplemented);
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
Result SendSyncRequestLight64(Core::System& system, Handle session_handle, u32* args) {
 | 
			
		||||
    R_RETURN(SendSyncRequestLight(system, session_handle, args));
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
Result ReplyAndReceiveLight64(Core::System& system, Handle session_handle, u32* args) {
 | 
			
		||||
    R_RETURN(ReplyAndReceiveLight(system, session_handle, args));
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
Result SendSyncRequestLight64From32(Core::System& system, Handle session_handle, u32* args) {
 | 
			
		||||
    R_RETURN(SendSyncRequestLight(system, session_handle, args));
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
Result ReplyAndReceiveLight64From32(Core::System& system, Handle session_handle, u32* args) {
 | 
			
		||||
    R_RETURN(ReplyAndReceiveLight(system, session_handle, args));
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
// Custom ABI implementation for light IPC.
 | 
			
		||||
 | 
			
		||||
template <typename F>
 | 
			
		||||
static void SvcWrap_LightIpc(Core::System& system, F&& cb) {
 | 
			
		||||
    auto& core = system.CurrentArmInterface();
 | 
			
		||||
    std::array<u32, 7> arguments{};
 | 
			
		||||
 | 
			
		||||
    Handle session_handle = static_cast<Handle>(core.GetReg(0));
 | 
			
		||||
    for (int i = 0; i < 7; i++) {
 | 
			
		||||
        arguments[i] = static_cast<u32>(core.GetReg(i + 1));
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    Result ret = cb(system, session_handle, arguments.data());
 | 
			
		||||
 | 
			
		||||
    core.SetReg(0, ret.raw);
 | 
			
		||||
    for (int i = 0; i < 7; i++) {
 | 
			
		||||
        core.SetReg(i + 1, arguments[i]);
 | 
			
		||||
    }
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
void SvcWrap_SendSyncRequestLight64(Core::System& system) {
 | 
			
		||||
    SvcWrap_LightIpc(system, SendSyncRequestLight64);
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
void SvcWrap_ReplyAndReceiveLight64(Core::System& system) {
 | 
			
		||||
    SvcWrap_LightIpc(system, ReplyAndReceiveLight64);
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
void SvcWrap_SendSyncRequestLight64From32(Core::System& system) {
 | 
			
		||||
    SvcWrap_LightIpc(system, SendSyncRequestLight64From32);
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
void SvcWrap_ReplyAndReceiveLight64From32(Core::System& system) {
 | 
			
		||||
    SvcWrap_LightIpc(system, ReplyAndReceiveLight64From32);
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
} // namespace Kernel::Svc
 | 
			
		||||
 
 | 
			
		||||
@@ -27,10 +27,6 @@ Result ArbitrateLock(Core::System& system, Handle thread_handle, VAddr address,
 | 
			
		||||
    return system.Kernel().CurrentProcess()->WaitForAddress(thread_handle, address, tag);
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
Result ArbitrateLock32(Core::System& system, Handle thread_handle, u32 address, u32 tag) {
 | 
			
		||||
    return ArbitrateLock(system, thread_handle, address, tag);
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
/// Unlock a mutex
 | 
			
		||||
Result ArbitrateUnlock(Core::System& system, VAddr address) {
 | 
			
		||||
    LOG_TRACE(Kernel_SVC, "called address=0x{:X}", address);
 | 
			
		||||
@@ -50,8 +46,21 @@ Result ArbitrateUnlock(Core::System& system, VAddr address) {
 | 
			
		||||
    return system.Kernel().CurrentProcess()->SignalToAddress(address);
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
Result ArbitrateUnlock32(Core::System& system, u32 address) {
 | 
			
		||||
    return ArbitrateUnlock(system, address);
 | 
			
		||||
Result ArbitrateLock64(Core::System& system, Handle thread_handle, uint64_t address, uint32_t tag) {
 | 
			
		||||
    R_RETURN(ArbitrateLock(system, thread_handle, address, tag));
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
Result ArbitrateUnlock64(Core::System& system, uint64_t address) {
 | 
			
		||||
    R_RETURN(ArbitrateUnlock(system, address));
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
Result ArbitrateLock64From32(Core::System& system, Handle thread_handle, uint32_t address,
 | 
			
		||||
                             uint32_t tag) {
 | 
			
		||||
    R_RETURN(ArbitrateLock(system, thread_handle, address, tag));
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
Result ArbitrateUnlock64From32(Core::System& system, uint32_t address) {
 | 
			
		||||
    R_RETURN(ArbitrateUnlock(system, address));
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
} // namespace Kernel::Svc
 | 
			
		||||
 
 | 
			
		||||
@@ -144,10 +144,6 @@ Result SetMemoryAttribute(Core::System& system, VAddr address, u64 size, u32 mas
 | 
			
		||||
    return page_table.SetMemoryAttribute(address, size, mask, attr);
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
Result SetMemoryAttribute32(Core::System& system, u32 address, u32 size, u32 mask, u32 attr) {
 | 
			
		||||
    return SetMemoryAttribute(system, address, size, mask, attr);
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
/// Maps a memory range into a different range.
 | 
			
		||||
Result MapMemory(Core::System& system, VAddr dst_addr, VAddr src_addr, u64 size) {
 | 
			
		||||
    LOG_TRACE(Kernel_SVC, "called, dst_addr=0x{:X}, src_addr=0x{:X}, size=0x{:X}", dst_addr,
 | 
			
		||||
@@ -163,10 +159,6 @@ Result MapMemory(Core::System& system, VAddr dst_addr, VAddr src_addr, u64 size)
 | 
			
		||||
    return page_table.MapMemory(dst_addr, src_addr, size);
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
Result MapMemory32(Core::System& system, u32 dst_addr, u32 src_addr, u32 size) {
 | 
			
		||||
    return MapMemory(system, dst_addr, src_addr, size);
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
/// Unmaps a region that was previously mapped with svcMapMemory
 | 
			
		||||
Result UnmapMemory(Core::System& system, VAddr dst_addr, VAddr src_addr, u64 size) {
 | 
			
		||||
    LOG_TRACE(Kernel_SVC, "called, dst_addr=0x{:X}, src_addr=0x{:X}, size=0x{:X}", dst_addr,
 | 
			
		||||
@@ -182,8 +174,44 @@ Result UnmapMemory(Core::System& system, VAddr dst_addr, VAddr src_addr, u64 siz
 | 
			
		||||
    return page_table.UnmapMemory(dst_addr, src_addr, size);
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
Result UnmapMemory32(Core::System& system, u32 dst_addr, u32 src_addr, u32 size) {
 | 
			
		||||
    return UnmapMemory(system, dst_addr, src_addr, size);
 | 
			
		||||
Result SetMemoryPermission64(Core::System& system, uint64_t address, uint64_t size,
 | 
			
		||||
                             MemoryPermission perm) {
 | 
			
		||||
    R_RETURN(SetMemoryPermission(system, address, size, perm));
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
Result SetMemoryAttribute64(Core::System& system, uint64_t address, uint64_t size, uint32_t mask,
 | 
			
		||||
                            uint32_t attr) {
 | 
			
		||||
    R_RETURN(SetMemoryAttribute(system, address, size, mask, attr));
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
Result MapMemory64(Core::System& system, uint64_t dst_address, uint64_t src_address,
 | 
			
		||||
                   uint64_t size) {
 | 
			
		||||
    R_RETURN(MapMemory(system, dst_address, src_address, size));
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
Result UnmapMemory64(Core::System& system, uint64_t dst_address, uint64_t src_address,
 | 
			
		||||
                     uint64_t size) {
 | 
			
		||||
    R_RETURN(UnmapMemory(system, dst_address, src_address, size));
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
Result SetMemoryPermission64From32(Core::System& system, uint32_t address, uint32_t size,
 | 
			
		||||
                                   MemoryPermission perm) {
 | 
			
		||||
    R_RETURN(SetMemoryPermission(system, address, size, perm));
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
Result SetMemoryAttribute64From32(Core::System& system, uint32_t address, uint32_t size,
 | 
			
		||||
                                  uint32_t mask, uint32_t attr) {
 | 
			
		||||
    R_RETURN(SetMemoryAttribute(system, address, size, mask, attr));
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
Result MapMemory64From32(Core::System& system, uint32_t dst_address, uint32_t src_address,
 | 
			
		||||
                         uint32_t size) {
 | 
			
		||||
    R_RETURN(MapMemory(system, dst_address, src_address, size));
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
Result UnmapMemory64From32(Core::System& system, uint32_t dst_address, uint32_t src_address,
 | 
			
		||||
                           uint32_t size) {
 | 
			
		||||
    R_RETURN(UnmapMemory(system, dst_address, src_address, size));
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
} // namespace Kernel::Svc
 | 
			
		||||
 
 | 
			
		||||
@@ -21,13 +21,6 @@ Result SetHeapSize(Core::System& system, VAddr* out_address, u64 size) {
 | 
			
		||||
    return ResultSuccess;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
Result SetHeapSize32(Core::System& system, u32* heap_addr, u32 heap_size) {
 | 
			
		||||
    VAddr temp_heap_addr{};
 | 
			
		||||
    const Result result{SetHeapSize(system, &temp_heap_addr, heap_size)};
 | 
			
		||||
    *heap_addr = static_cast<u32>(temp_heap_addr);
 | 
			
		||||
    return result;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
/// Maps memory at a desired address
 | 
			
		||||
Result MapPhysicalMemory(Core::System& system, VAddr addr, u64 size) {
 | 
			
		||||
    LOG_DEBUG(Kernel_SVC, "called, addr=0x{:016X}, size=0x{:X}", addr, size);
 | 
			
		||||
@@ -77,10 +70,6 @@ Result MapPhysicalMemory(Core::System& system, VAddr addr, u64 size) {
 | 
			
		||||
    return page_table.MapPhysicalMemory(addr, size);
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
Result MapPhysicalMemory32(Core::System& system, u32 addr, u32 size) {
 | 
			
		||||
    return MapPhysicalMemory(system, addr, size);
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
/// Unmaps memory previously mapped via MapPhysicalMemory
 | 
			
		||||
Result UnmapPhysicalMemory(Core::System& system, VAddr addr, u64 size) {
 | 
			
		||||
    LOG_DEBUG(Kernel_SVC, "called, addr=0x{:016X}, size=0x{:X}", addr, size);
 | 
			
		||||
@@ -130,8 +119,67 @@ Result UnmapPhysicalMemory(Core::System& system, VAddr addr, u64 size) {
 | 
			
		||||
    return page_table.UnmapPhysicalMemory(addr, size);
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
Result UnmapPhysicalMemory32(Core::System& system, u32 addr, u32 size) {
 | 
			
		||||
    return UnmapPhysicalMemory(system, addr, size);
 | 
			
		||||
Result MapPhysicalMemoryUnsafe(Core::System& system, uint64_t address, uint64_t size) {
 | 
			
		||||
    UNIMPLEMENTED();
 | 
			
		||||
    R_THROW(ResultNotImplemented);
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
Result UnmapPhysicalMemoryUnsafe(Core::System& system, uint64_t address, uint64_t size) {
 | 
			
		||||
    UNIMPLEMENTED();
 | 
			
		||||
    R_THROW(ResultNotImplemented);
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
Result SetUnsafeLimit(Core::System& system, uint64_t limit) {
 | 
			
		||||
    UNIMPLEMENTED();
 | 
			
		||||
    R_THROW(ResultNotImplemented);
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
Result SetHeapSize64(Core::System& system, uint64_t* out_address, uint64_t size) {
 | 
			
		||||
    R_RETURN(SetHeapSize(system, out_address, size));
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
Result MapPhysicalMemory64(Core::System& system, uint64_t address, uint64_t size) {
 | 
			
		||||
    R_RETURN(MapPhysicalMemory(system, address, size));
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
Result UnmapPhysicalMemory64(Core::System& system, uint64_t address, uint64_t size) {
 | 
			
		||||
    R_RETURN(UnmapPhysicalMemory(system, address, size));
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
Result MapPhysicalMemoryUnsafe64(Core::System& system, uint64_t address, uint64_t size) {
 | 
			
		||||
    R_RETURN(MapPhysicalMemoryUnsafe(system, address, size));
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
Result UnmapPhysicalMemoryUnsafe64(Core::System& system, uint64_t address, uint64_t size) {
 | 
			
		||||
    R_RETURN(UnmapPhysicalMemoryUnsafe(system, address, size));
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
Result SetUnsafeLimit64(Core::System& system, uint64_t limit) {
 | 
			
		||||
    R_RETURN(SetUnsafeLimit(system, limit));
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
Result SetHeapSize64From32(Core::System& system, uintptr_t* out_address, uint32_t size) {
 | 
			
		||||
    R_RETURN(SetHeapSize(system, out_address, size));
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
Result MapPhysicalMemory64From32(Core::System& system, uint32_t address, uint32_t size) {
 | 
			
		||||
    R_RETURN(MapPhysicalMemory(system, address, size));
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
Result UnmapPhysicalMemory64From32(Core::System& system, uint32_t address, uint32_t size) {
 | 
			
		||||
    R_RETURN(UnmapPhysicalMemory(system, address, size));
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
Result MapPhysicalMemoryUnsafe64From32(Core::System& system, uint32_t address, uint32_t size) {
 | 
			
		||||
    R_RETURN(MapPhysicalMemoryUnsafe(system, address, size));
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
Result UnmapPhysicalMemoryUnsafe64From32(Core::System& system, uint32_t address, uint32_t size) {
 | 
			
		||||
    R_RETURN(UnmapPhysicalMemoryUnsafe(system, address, size));
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
Result SetUnsafeLimit64From32(Core::System& system, uint32_t limit) {
 | 
			
		||||
    R_RETURN(SetUnsafeLimit(system, limit));
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
} // namespace Kernel::Svc
 | 
			
		||||
 
 | 
			
		||||
@@ -63,9 +63,60 @@ Result ConnectToNamedPort(Core::System& system, Handle* out, VAddr port_name_add
 | 
			
		||||
    return ResultSuccess;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
Result ConnectToNamedPort32(Core::System& system, Handle* out_handle, u32 port_name_address) {
 | 
			
		||||
Result CreatePort(Core::System& system, Handle* out_server, Handle* out_client,
 | 
			
		||||
                  int32_t max_sessions, bool is_light, uintptr_t name) {
 | 
			
		||||
    UNIMPLEMENTED();
 | 
			
		||||
    R_THROW(ResultNotImplemented);
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
    return ConnectToNamedPort(system, out_handle, port_name_address);
 | 
			
		||||
Result ConnectToPort(Core::System& system, Handle* out_handle, Handle port) {
 | 
			
		||||
    UNIMPLEMENTED();
 | 
			
		||||
    R_THROW(ResultNotImplemented);
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
Result ManageNamedPort(Core::System& system, Handle* out_server_handle, uint64_t name,
 | 
			
		||||
                       int32_t max_sessions) {
 | 
			
		||||
    UNIMPLEMENTED();
 | 
			
		||||
    R_THROW(ResultNotImplemented);
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
Result ConnectToNamedPort64(Core::System& system, Handle* out_handle, uint64_t name) {
 | 
			
		||||
    R_RETURN(ConnectToNamedPort(system, out_handle, name));
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
Result CreatePort64(Core::System& system, Handle* out_server_handle, Handle* out_client_handle,
 | 
			
		||||
                    int32_t max_sessions, bool is_light, uint64_t name) {
 | 
			
		||||
    R_RETURN(
 | 
			
		||||
        CreatePort(system, out_server_handle, out_client_handle, max_sessions, is_light, name));
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
Result ManageNamedPort64(Core::System& system, Handle* out_server_handle, uint64_t name,
 | 
			
		||||
                         int32_t max_sessions) {
 | 
			
		||||
    R_RETURN(ManageNamedPort(system, out_server_handle, name, max_sessions));
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
Result ConnectToPort64(Core::System& system, Handle* out_handle, Handle port) {
 | 
			
		||||
    R_RETURN(ConnectToPort(system, out_handle, port));
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
Result ConnectToNamedPort64From32(Core::System& system, Handle* out_handle, uint32_t name) {
 | 
			
		||||
    R_RETURN(ConnectToNamedPort(system, out_handle, name));
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
Result CreatePort64From32(Core::System& system, Handle* out_server_handle,
 | 
			
		||||
                          Handle* out_client_handle, int32_t max_sessions, bool is_light,
 | 
			
		||||
                          uint32_t name) {
 | 
			
		||||
    R_RETURN(
 | 
			
		||||
        CreatePort(system, out_server_handle, out_client_handle, max_sessions, is_light, name));
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
Result ManageNamedPort64From32(Core::System& system, Handle* out_server_handle, uint32_t name,
 | 
			
		||||
                               int32_t max_sessions) {
 | 
			
		||||
    R_RETURN(ManageNamedPort(system, out_server_handle, name, max_sessions));
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
Result ConnectToPort64From32(Core::System& system, Handle* out_handle, Handle port) {
 | 
			
		||||
    R_RETURN(ConnectToPort(system, out_handle, port));
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
} // namespace Kernel::Svc
 | 
			
		||||
 
 | 
			
		||||
@@ -2,5 +2,20 @@
 | 
			
		||||
// SPDX-License-Identifier: GPL-2.0-or-later
 | 
			
		||||
 | 
			
		||||
#include "core/hle/kernel/svc.h"
 | 
			
		||||
#include "core/hle/kernel/svc_results.h"
 | 
			
		||||
 | 
			
		||||
namespace Kernel::Svc {} // namespace Kernel::Svc
 | 
			
		||||
namespace Kernel::Svc {
 | 
			
		||||
 | 
			
		||||
void SleepSystem(Core::System& system) {
 | 
			
		||||
    UNIMPLEMENTED();
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
void SleepSystem64(Core::System& system) {
 | 
			
		||||
    return SleepSystem(system);
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
void SleepSystem64From32(Core::System& system) {
 | 
			
		||||
    return SleepSystem(system);
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
} // namespace Kernel::Svc
 | 
			
		||||
 
 | 
			
		||||
@@ -18,10 +18,6 @@ void ExitProcess(Core::System& system) {
 | 
			
		||||
    system.Exit();
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
void ExitProcess32(Core::System& system) {
 | 
			
		||||
    ExitProcess(system);
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
/// Gets the ID of the specified process or a specified thread's owning process.
 | 
			
		||||
Result GetProcessId(Core::System& system, u64* out_process_id, Handle handle) {
 | 
			
		||||
    LOG_DEBUG(Kernel_SVC, "called handle=0x{:08X}", handle);
 | 
			
		||||
@@ -54,17 +50,8 @@ Result GetProcessId(Core::System& system, u64* out_process_id, Handle handle) {
 | 
			
		||||
    return ResultSuccess;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
Result GetProcessId32(Core::System& system, u32* out_process_id_low, u32* out_process_id_high,
 | 
			
		||||
                      Handle handle) {
 | 
			
		||||
    u64 out_process_id{};
 | 
			
		||||
    const auto result = GetProcessId(system, &out_process_id, handle);
 | 
			
		||||
    *out_process_id_low = static_cast<u32>(out_process_id);
 | 
			
		||||
    *out_process_id_high = static_cast<u32>(out_process_id >> 32);
 | 
			
		||||
    return result;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
Result GetProcessList(Core::System& system, u32* out_num_processes, VAddr out_process_ids,
 | 
			
		||||
                      u32 out_process_ids_size) {
 | 
			
		||||
Result GetProcessList(Core::System& system, s32* out_num_processes, VAddr out_process_ids,
 | 
			
		||||
                      int32_t out_process_ids_size) {
 | 
			
		||||
    LOG_DEBUG(Kernel_SVC, "called. out_process_ids=0x{:016X}, out_process_ids_size={}",
 | 
			
		||||
              out_process_ids, out_process_ids_size);
 | 
			
		||||
 | 
			
		||||
@@ -89,7 +76,8 @@ Result GetProcessList(Core::System& system, u32* out_num_processes, VAddr out_pr
 | 
			
		||||
    auto& memory = system.Memory();
 | 
			
		||||
    const auto& process_list = kernel.GetProcessList();
 | 
			
		||||
    const auto num_processes = process_list.size();
 | 
			
		||||
    const auto copy_amount = std::min(std::size_t{out_process_ids_size}, num_processes);
 | 
			
		||||
    const auto copy_amount =
 | 
			
		||||
        std::min(static_cast<std::size_t>(out_process_ids_size), num_processes);
 | 
			
		||||
 | 
			
		||||
    for (std::size_t i = 0; i < copy_amount; ++i) {
 | 
			
		||||
        memory.Write64(out_process_ids, process_list[i]->GetProcessID());
 | 
			
		||||
@@ -100,8 +88,9 @@ Result GetProcessList(Core::System& system, u32* out_num_processes, VAddr out_pr
 | 
			
		||||
    return ResultSuccess;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
Result GetProcessInfo(Core::System& system, u64* out, Handle process_handle, u32 type) {
 | 
			
		||||
    LOG_DEBUG(Kernel_SVC, "called, handle=0x{:08X}, type=0x{:X}", process_handle, type);
 | 
			
		||||
Result GetProcessInfo(Core::System& system, s64* out, Handle process_handle,
 | 
			
		||||
                      ProcessInfoType info_type) {
 | 
			
		||||
    LOG_DEBUG(Kernel_SVC, "called, handle=0x{:08X}, type=0x{:X}", process_handle, info_type);
 | 
			
		||||
 | 
			
		||||
    const auto& handle_table = system.Kernel().CurrentProcess()->GetHandleTable();
 | 
			
		||||
    KScopedAutoObject process = handle_table.GetObject<KProcess>(process_handle);
 | 
			
		||||
@@ -111,14 +100,95 @@ Result GetProcessInfo(Core::System& system, u64* out, Handle process_handle, u32
 | 
			
		||||
        return ResultInvalidHandle;
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    const auto info_type = static_cast<ProcessInfoType>(type);
 | 
			
		||||
    if (info_type != ProcessInfoType::ProcessState) {
 | 
			
		||||
        LOG_ERROR(Kernel_SVC, "Expected info_type to be ProcessState but got {} instead", type);
 | 
			
		||||
        LOG_ERROR(Kernel_SVC, "Expected info_type to be ProcessState but got {} instead",
 | 
			
		||||
                  info_type);
 | 
			
		||||
        return ResultInvalidEnumValue;
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    *out = static_cast<u64>(process->GetState());
 | 
			
		||||
    *out = static_cast<s64>(process->GetState());
 | 
			
		||||
    return ResultSuccess;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
Result CreateProcess(Core::System& system, Handle* out_handle, uint64_t parameters, uint64_t caps,
 | 
			
		||||
                     int32_t num_caps) {
 | 
			
		||||
    UNIMPLEMENTED();
 | 
			
		||||
    R_THROW(ResultNotImplemented);
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
Result StartProcess(Core::System& system, Handle process_handle, int32_t priority, int32_t core_id,
 | 
			
		||||
                    uint64_t main_thread_stack_size) {
 | 
			
		||||
    UNIMPLEMENTED();
 | 
			
		||||
    R_THROW(ResultNotImplemented);
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
Result TerminateProcess(Core::System& system, Handle process_handle) {
 | 
			
		||||
    UNIMPLEMENTED();
 | 
			
		||||
    R_THROW(ResultNotImplemented);
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
void ExitProcess64(Core::System& system) {
 | 
			
		||||
    ExitProcess(system);
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
Result GetProcessId64(Core::System& system, uint64_t* out_process_id, Handle process_handle) {
 | 
			
		||||
    R_RETURN(GetProcessId(system, out_process_id, process_handle));
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
Result GetProcessList64(Core::System& system, int32_t* out_num_processes, uint64_t out_process_ids,
 | 
			
		||||
                        int32_t max_out_count) {
 | 
			
		||||
    R_RETURN(GetProcessList(system, out_num_processes, out_process_ids, max_out_count));
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
Result CreateProcess64(Core::System& system, Handle* out_handle, uint64_t parameters, uint64_t caps,
 | 
			
		||||
                       int32_t num_caps) {
 | 
			
		||||
    R_RETURN(CreateProcess(system, out_handle, parameters, caps, num_caps));
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
Result StartProcess64(Core::System& system, Handle process_handle, int32_t priority,
 | 
			
		||||
                      int32_t core_id, uint64_t main_thread_stack_size) {
 | 
			
		||||
    R_RETURN(StartProcess(system, process_handle, priority, core_id, main_thread_stack_size));
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
Result TerminateProcess64(Core::System& system, Handle process_handle) {
 | 
			
		||||
    R_RETURN(TerminateProcess(system, process_handle));
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
Result GetProcessInfo64(Core::System& system, int64_t* out_info, Handle process_handle,
 | 
			
		||||
                        ProcessInfoType info_type) {
 | 
			
		||||
    R_RETURN(GetProcessInfo(system, out_info, process_handle, info_type));
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
void ExitProcess64From32(Core::System& system) {
 | 
			
		||||
    ExitProcess(system);
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
Result GetProcessId64From32(Core::System& system, uint64_t* out_process_id, Handle process_handle) {
 | 
			
		||||
    R_RETURN(GetProcessId(system, out_process_id, process_handle));
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
Result GetProcessList64From32(Core::System& system, int32_t* out_num_processes,
 | 
			
		||||
                              uint32_t out_process_ids, int32_t max_out_count) {
 | 
			
		||||
    R_RETURN(GetProcessList(system, out_num_processes, out_process_ids, max_out_count));
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
Result CreateProcess64From32(Core::System& system, Handle* out_handle, uint32_t parameters,
 | 
			
		||||
                             uint32_t caps, int32_t num_caps) {
 | 
			
		||||
    R_RETURN(CreateProcess(system, out_handle, parameters, caps, num_caps));
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
Result StartProcess64From32(Core::System& system, Handle process_handle, int32_t priority,
 | 
			
		||||
                            int32_t core_id, uint64_t main_thread_stack_size) {
 | 
			
		||||
    R_RETURN(StartProcess(system, process_handle, priority, core_id, main_thread_stack_size));
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
Result TerminateProcess64From32(Core::System& system, Handle process_handle) {
 | 
			
		||||
    R_RETURN(TerminateProcess(system, process_handle));
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
Result GetProcessInfo64From32(Core::System& system, int64_t* out_info, Handle process_handle,
 | 
			
		||||
                              ProcessInfoType info_type) {
 | 
			
		||||
    R_RETURN(GetProcessInfo(system, out_info, process_handle, info_type));
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
} // namespace Kernel::Svc
 | 
			
		||||
 
 | 
			
		||||
@@ -271,4 +271,54 @@ Result UnmapProcessCodeMemory(Core::System& system, Handle process_handle, u64 d
 | 
			
		||||
                                      KPageTable::ICacheInvalidationStrategy::InvalidateAll);
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
Result SetProcessMemoryPermission64(Core::System& system, Handle process_handle, uint64_t address,
 | 
			
		||||
                                    uint64_t size, MemoryPermission perm) {
 | 
			
		||||
    R_RETURN(SetProcessMemoryPermission(system, process_handle, address, size, perm));
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
Result MapProcessMemory64(Core::System& system, uint64_t dst_address, Handle process_handle,
 | 
			
		||||
                          uint64_t src_address, uint64_t size) {
 | 
			
		||||
    R_RETURN(MapProcessMemory(system, dst_address, process_handle, src_address, size));
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
Result UnmapProcessMemory64(Core::System& system, uint64_t dst_address, Handle process_handle,
 | 
			
		||||
                            uint64_t src_address, uint64_t size) {
 | 
			
		||||
    R_RETURN(UnmapProcessMemory(system, dst_address, process_handle, src_address, size));
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
Result MapProcessCodeMemory64(Core::System& system, Handle process_handle, uint64_t dst_address,
 | 
			
		||||
                              uint64_t src_address, uint64_t size) {
 | 
			
		||||
    R_RETURN(MapProcessCodeMemory(system, process_handle, dst_address, src_address, size));
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
Result UnmapProcessCodeMemory64(Core::System& system, Handle process_handle, uint64_t dst_address,
 | 
			
		||||
                                uint64_t src_address, uint64_t size) {
 | 
			
		||||
    R_RETURN(UnmapProcessCodeMemory(system, process_handle, dst_address, src_address, size));
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
Result SetProcessMemoryPermission64From32(Core::System& system, Handle process_handle,
 | 
			
		||||
                                          uint64_t address, uint64_t size, MemoryPermission perm) {
 | 
			
		||||
    R_RETURN(SetProcessMemoryPermission(system, process_handle, address, size, perm));
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
Result MapProcessMemory64From32(Core::System& system, uint32_t dst_address, Handle process_handle,
 | 
			
		||||
                                uint64_t src_address, uint32_t size) {
 | 
			
		||||
    R_RETURN(MapProcessMemory(system, dst_address, process_handle, src_address, size));
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
Result UnmapProcessMemory64From32(Core::System& system, uint32_t dst_address, Handle process_handle,
 | 
			
		||||
                                  uint64_t src_address, uint32_t size) {
 | 
			
		||||
    R_RETURN(UnmapProcessMemory(system, dst_address, process_handle, src_address, size));
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
Result MapProcessCodeMemory64From32(Core::System& system, Handle process_handle,
 | 
			
		||||
                                    uint64_t dst_address, uint64_t src_address, uint64_t size) {
 | 
			
		||||
    R_RETURN(MapProcessCodeMemory(system, process_handle, dst_address, src_address, size));
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
Result UnmapProcessCodeMemory64From32(Core::System& system, Handle process_handle,
 | 
			
		||||
                                      uint64_t dst_address, uint64_t src_address, uint64_t size) {
 | 
			
		||||
    R_RETURN(UnmapProcessCodeMemory(system, process_handle, dst_address, src_address, size));
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
} // namespace Kernel::Svc
 | 
			
		||||
 
 | 
			
		||||
@@ -9,12 +9,16 @@
 | 
			
		||||
namespace Kernel::Svc {
 | 
			
		||||
 | 
			
		||||
/// Get which CPU core is executing the current thread
 | 
			
		||||
u32 GetCurrentProcessorNumber(Core::System& system) {
 | 
			
		||||
int32_t GetCurrentProcessorNumber(Core::System& system) {
 | 
			
		||||
    LOG_TRACE(Kernel_SVC, "called");
 | 
			
		||||
    return static_cast<u32>(system.CurrentPhysicalCore().CoreIndex());
 | 
			
		||||
    return static_cast<int32_t>(system.CurrentPhysicalCore().CoreIndex());
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
u32 GetCurrentProcessorNumber32(Core::System& system) {
 | 
			
		||||
int32_t GetCurrentProcessorNumber64(Core::System& system) {
 | 
			
		||||
    return GetCurrentProcessorNumber(system);
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
int32_t GetCurrentProcessorNumber64From32(Core::System& system) {
 | 
			
		||||
    return GetCurrentProcessorNumber(system);
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
 
 | 
			
		||||
@@ -7,24 +7,20 @@
 | 
			
		||||
 | 
			
		||||
namespace Kernel::Svc {
 | 
			
		||||
 | 
			
		||||
Result QueryMemory(Core::System& system, VAddr memory_info_address, VAddr page_info_address,
 | 
			
		||||
Result QueryMemory(Core::System& system, uint64_t out_memory_info, PageInfo* out_page_info,
 | 
			
		||||
                   VAddr query_address) {
 | 
			
		||||
    LOG_TRACE(Kernel_SVC,
 | 
			
		||||
              "called, memory_info_address=0x{:016X}, page_info_address=0x{:016X}, "
 | 
			
		||||
              "called, out_memory_info=0x{:016X}, "
 | 
			
		||||
              "query_address=0x{:016X}",
 | 
			
		||||
              memory_info_address, page_info_address, query_address);
 | 
			
		||||
              out_memory_info, query_address);
 | 
			
		||||
 | 
			
		||||
    return QueryProcessMemory(system, memory_info_address, page_info_address, CurrentProcess,
 | 
			
		||||
    // Query memory is just QueryProcessMemory on the current process.
 | 
			
		||||
    return QueryProcessMemory(system, out_memory_info, out_page_info, CurrentProcess,
 | 
			
		||||
                              query_address);
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
Result QueryMemory32(Core::System& system, u32 memory_info_address, u32 page_info_address,
 | 
			
		||||
                     u32 query_address) {
 | 
			
		||||
    return QueryMemory(system, memory_info_address, page_info_address, query_address);
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
Result QueryProcessMemory(Core::System& system, VAddr memory_info_address, VAddr page_info_address,
 | 
			
		||||
                          Handle process_handle, VAddr address) {
 | 
			
		||||
Result QueryProcessMemory(Core::System& system, uint64_t out_memory_info, PageInfo* out_page_info,
 | 
			
		||||
                          Handle process_handle, uint64_t address) {
 | 
			
		||||
    LOG_TRACE(Kernel_SVC, "called process=0x{:08X} address={:X}", process_handle, address);
 | 
			
		||||
    const auto& handle_table = system.Kernel().CurrentProcess()->GetHandleTable();
 | 
			
		||||
    KScopedAutoObject process = handle_table.GetObject<KProcess>(process_handle);
 | 
			
		||||
@@ -37,19 +33,33 @@ Result QueryProcessMemory(Core::System& system, VAddr memory_info_address, VAddr
 | 
			
		||||
    auto& memory{system.Memory()};
 | 
			
		||||
    const auto memory_info{process->PageTable().QueryInfo(address).GetSvcMemoryInfo()};
 | 
			
		||||
 | 
			
		||||
    memory.Write64(memory_info_address + 0x00, memory_info.base_address);
 | 
			
		||||
    memory.Write64(memory_info_address + 0x08, memory_info.size);
 | 
			
		||||
    memory.Write32(memory_info_address + 0x10, static_cast<u32>(memory_info.state) & 0xff);
 | 
			
		||||
    memory.Write32(memory_info_address + 0x14, static_cast<u32>(memory_info.attribute));
 | 
			
		||||
    memory.Write32(memory_info_address + 0x18, static_cast<u32>(memory_info.permission));
 | 
			
		||||
    memory.Write32(memory_info_address + 0x1c, memory_info.ipc_count);
 | 
			
		||||
    memory.Write32(memory_info_address + 0x20, memory_info.device_count);
 | 
			
		||||
    memory.Write32(memory_info_address + 0x24, 0);
 | 
			
		||||
    memory.WriteBlock(out_memory_info, &memory_info, sizeof(memory_info));
 | 
			
		||||
 | 
			
		||||
    // Page info appears to be currently unused by the kernel and is always set to zero.
 | 
			
		||||
    memory.Write32(page_info_address, 0);
 | 
			
		||||
    //! This is supposed to be part of the QueryInfo call.
 | 
			
		||||
    *out_page_info = {};
 | 
			
		||||
 | 
			
		||||
    return ResultSuccess;
 | 
			
		||||
    R_SUCCEED();
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
Result QueryMemory64(Core::System& system, uint64_t out_memory_info, PageInfo* out_page_info,
 | 
			
		||||
                     uint64_t address) {
 | 
			
		||||
    R_RETURN(QueryMemory(system, out_memory_info, out_page_info, address));
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
Result QueryProcessMemory64(Core::System& system, uint64_t out_memory_info, PageInfo* out_page_info,
 | 
			
		||||
                            Handle process_handle, uint64_t address) {
 | 
			
		||||
    R_RETURN(QueryProcessMemory(system, out_memory_info, out_page_info, process_handle, address));
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
Result QueryMemory64From32(Core::System& system, uint32_t out_memory_info, PageInfo* out_page_info,
 | 
			
		||||
                           uint32_t address) {
 | 
			
		||||
    R_RETURN(QueryMemory(system, out_memory_info, out_page_info, address));
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
Result QueryProcessMemory64From32(Core::System& system, uint32_t out_memory_info,
 | 
			
		||||
                                  PageInfo* out_page_info, Handle process_handle,
 | 
			
		||||
                                  uint64_t address) {
 | 
			
		||||
    R_RETURN(QueryProcessMemory(system, out_memory_info, out_page_info, process_handle, address));
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
} // namespace Kernel::Svc
 | 
			
		||||
 
 | 
			
		||||
@@ -2,5 +2,26 @@
 | 
			
		||||
// SPDX-License-Identifier: GPL-2.0-or-later
 | 
			
		||||
 | 
			
		||||
#include "core/hle/kernel/svc.h"
 | 
			
		||||
#include "core/hle/kernel/svc_results.h"
 | 
			
		||||
 | 
			
		||||
namespace Kernel::Svc {} // namespace Kernel::Svc
 | 
			
		||||
namespace Kernel::Svc {
 | 
			
		||||
 | 
			
		||||
Result ReadWriteRegister(Core::System& system, uint32_t* out, uint64_t address, uint32_t mask,
 | 
			
		||||
                         uint32_t value) {
 | 
			
		||||
    *out = 0;
 | 
			
		||||
 | 
			
		||||
    UNIMPLEMENTED();
 | 
			
		||||
    R_THROW(ResultNotImplemented);
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
Result ReadWriteRegister64(Core::System& system, uint32_t* out_value, uint64_t address,
 | 
			
		||||
                           uint32_t mask, uint32_t value) {
 | 
			
		||||
    R_RETURN(ReadWriteRegister(system, out_value, address, mask, value));
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
Result ReadWriteRegister64From32(Core::System& system, uint32_t* out_value, uint64_t address,
 | 
			
		||||
                                 uint32_t mask, uint32_t value) {
 | 
			
		||||
    R_RETURN(ReadWriteRegister(system, out_value, address, mask, value));
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
} // namespace Kernel::Svc
 | 
			
		||||
 
 | 
			
		||||
@@ -32,7 +32,7 @@ Result CreateResourceLimit(Core::System& system, Handle* out_handle) {
 | 
			
		||||
    return ResultSuccess;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
Result GetResourceLimitLimitValue(Core::System& system, u64* out_limit_value,
 | 
			
		||||
Result GetResourceLimitLimitValue(Core::System& system, s64* out_limit_value,
 | 
			
		||||
                                  Handle resource_limit_handle, LimitableResource which) {
 | 
			
		||||
    LOG_DEBUG(Kernel_SVC, "called, resource_limit_handle={:08X}, which={}", resource_limit_handle,
 | 
			
		||||
              which);
 | 
			
		||||
@@ -52,7 +52,7 @@ Result GetResourceLimitLimitValue(Core::System& system, u64* out_limit_value,
 | 
			
		||||
    return ResultSuccess;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
Result GetResourceLimitCurrentValue(Core::System& system, u64* out_current_value,
 | 
			
		||||
Result GetResourceLimitCurrentValue(Core::System& system, s64* out_current_value,
 | 
			
		||||
                                    Handle resource_limit_handle, LimitableResource which) {
 | 
			
		||||
    LOG_DEBUG(Kernel_SVC, "called, resource_limit_handle={:08X}, which={}", resource_limit_handle,
 | 
			
		||||
              which);
 | 
			
		||||
@@ -73,7 +73,7 @@ Result GetResourceLimitCurrentValue(Core::System& system, u64* out_current_value
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
Result SetResourceLimitLimitValue(Core::System& system, Handle resource_limit_handle,
 | 
			
		||||
                                  LimitableResource which, u64 limit_value) {
 | 
			
		||||
                                  LimitableResource which, s64 limit_value) {
 | 
			
		||||
    LOG_DEBUG(Kernel_SVC, "called, resource_limit_handle={:08X}, which={}, limit_value={}",
 | 
			
		||||
              resource_limit_handle, which, limit_value);
 | 
			
		||||
 | 
			
		||||
@@ -92,4 +92,58 @@ Result SetResourceLimitLimitValue(Core::System& system, Handle resource_limit_ha
 | 
			
		||||
    return ResultSuccess;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
Result GetResourceLimitPeakValue(Core::System& system, int64_t* out_peak_value,
 | 
			
		||||
                                 Handle resource_limit_handle, LimitableResource which) {
 | 
			
		||||
    UNIMPLEMENTED();
 | 
			
		||||
    R_THROW(ResultNotImplemented);
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
Result GetResourceLimitLimitValue64(Core::System& system, int64_t* out_limit_value,
 | 
			
		||||
                                    Handle resource_limit_handle, LimitableResource which) {
 | 
			
		||||
    R_RETURN(GetResourceLimitLimitValue(system, out_limit_value, resource_limit_handle, which));
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
Result GetResourceLimitCurrentValue64(Core::System& system, int64_t* out_current_value,
 | 
			
		||||
                                      Handle resource_limit_handle, LimitableResource which) {
 | 
			
		||||
    R_RETURN(GetResourceLimitCurrentValue(system, out_current_value, resource_limit_handle, which));
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
Result GetResourceLimitPeakValue64(Core::System& system, int64_t* out_peak_value,
 | 
			
		||||
                                   Handle resource_limit_handle, LimitableResource which) {
 | 
			
		||||
    R_RETURN(GetResourceLimitPeakValue(system, out_peak_value, resource_limit_handle, which));
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
Result CreateResourceLimit64(Core::System& system, Handle* out_handle) {
 | 
			
		||||
    R_RETURN(CreateResourceLimit(system, out_handle));
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
Result SetResourceLimitLimitValue64(Core::System& system, Handle resource_limit_handle,
 | 
			
		||||
                                    LimitableResource which, int64_t limit_value) {
 | 
			
		||||
    R_RETURN(SetResourceLimitLimitValue(system, resource_limit_handle, which, limit_value));
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
Result GetResourceLimitLimitValue64From32(Core::System& system, int64_t* out_limit_value,
 | 
			
		||||
                                          Handle resource_limit_handle, LimitableResource which) {
 | 
			
		||||
    R_RETURN(GetResourceLimitLimitValue(system, out_limit_value, resource_limit_handle, which));
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
Result GetResourceLimitCurrentValue64From32(Core::System& system, int64_t* out_current_value,
 | 
			
		||||
                                            Handle resource_limit_handle, LimitableResource which) {
 | 
			
		||||
    R_RETURN(GetResourceLimitCurrentValue(system, out_current_value, resource_limit_handle, which));
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
Result GetResourceLimitPeakValue64From32(Core::System& system, int64_t* out_peak_value,
 | 
			
		||||
                                         Handle resource_limit_handle, LimitableResource which) {
 | 
			
		||||
    R_RETURN(GetResourceLimitPeakValue(system, out_peak_value, resource_limit_handle, which));
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
Result CreateResourceLimit64From32(Core::System& system, Handle* out_handle) {
 | 
			
		||||
    R_RETURN(CreateResourceLimit(system, out_handle));
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
Result SetResourceLimitLimitValue64From32(Core::System& system, Handle resource_limit_handle,
 | 
			
		||||
                                          LimitableResource which, int64_t limit_value) {
 | 
			
		||||
    R_RETURN(SetResourceLimitLimitValue(system, resource_limit_handle, which, limit_value));
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
} // namespace Kernel::Svc
 | 
			
		||||
 
 | 
			
		||||
@@ -1,6 +1,53 @@
 | 
			
		||||
// SPDX-FileCopyrightText: Copyright 2023 yuzu Emulator Project
 | 
			
		||||
// SPDX-License-Identifier: GPL-2.0-or-later
 | 
			
		||||
 | 
			
		||||
#include "core/core.h"
 | 
			
		||||
#include "core/hle/kernel/physical_core.h"
 | 
			
		||||
#include "core/hle/kernel/svc.h"
 | 
			
		||||
 | 
			
		||||
namespace Kernel::Svc {} // namespace Kernel::Svc
 | 
			
		||||
namespace Kernel::Svc {
 | 
			
		||||
 | 
			
		||||
void CallSecureMonitor(Core::System& system, lp64::SecureMonitorArguments* args) {
 | 
			
		||||
    UNIMPLEMENTED();
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
void CallSecureMonitor64(Core::System& system, lp64::SecureMonitorArguments* args) {
 | 
			
		||||
    CallSecureMonitor(system, args);
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
void CallSecureMonitor64From32(Core::System& system, ilp32::SecureMonitorArguments* args) {
 | 
			
		||||
    // CallSecureMonitor64From32 is not supported.
 | 
			
		||||
    UNIMPLEMENTED_MSG("CallSecureMonitor64From32");
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
// Custom ABI for CallSecureMonitor.
 | 
			
		||||
 | 
			
		||||
void SvcWrap_CallSecureMonitor64(Core::System& system) {
 | 
			
		||||
    auto& core = system.CurrentPhysicalCore().ArmInterface();
 | 
			
		||||
    lp64::SecureMonitorArguments args{};
 | 
			
		||||
    for (int i = 0; i < 8; i++) {
 | 
			
		||||
        args.r[i] = core.GetReg(i);
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    CallSecureMonitor64(system, &args);
 | 
			
		||||
 | 
			
		||||
    for (int i = 0; i < 8; i++) {
 | 
			
		||||
        core.SetReg(i, args.r[i]);
 | 
			
		||||
    }
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
void SvcWrap_CallSecureMonitor64From32(Core::System& system) {
 | 
			
		||||
    auto& core = system.CurrentPhysicalCore().ArmInterface();
 | 
			
		||||
    ilp32::SecureMonitorArguments args{};
 | 
			
		||||
    for (int i = 0; i < 8; i++) {
 | 
			
		||||
        args.r[i] = static_cast<u32>(core.GetReg(i));
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    CallSecureMonitor64From32(system, &args);
 | 
			
		||||
 | 
			
		||||
    for (int i = 0; i < 8; i++) {
 | 
			
		||||
        core.SetReg(i, args.r[i]);
 | 
			
		||||
    }
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
} // namespace Kernel::Svc
 | 
			
		||||
 
 | 
			
		||||
@@ -90,14 +90,39 @@ Result CreateSession(Core::System& system, Handle* out_server, Handle* out_clien
 | 
			
		||||
 | 
			
		||||
} // namespace
 | 
			
		||||
 | 
			
		||||
Result CreateSession(Core::System& system, Handle* out_server, Handle* out_client, u32 is_light,
 | 
			
		||||
Result CreateSession(Core::System& system, Handle* out_server, Handle* out_client, bool is_light,
 | 
			
		||||
                     u64 name) {
 | 
			
		||||
    if (is_light) {
 | 
			
		||||
        // return CreateSession<KLightSession>(system, out_server, out_client, name);
 | 
			
		||||
        return ResultUnknown;
 | 
			
		||||
        return ResultNotImplemented;
 | 
			
		||||
    } else {
 | 
			
		||||
        return CreateSession<KSession>(system, out_server, out_client, name);
 | 
			
		||||
    }
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
Result AcceptSession(Core::System& system, Handle* out_handle, Handle port_handle) {
 | 
			
		||||
    UNIMPLEMENTED();
 | 
			
		||||
    R_THROW(ResultNotImplemented);
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
Result CreateSession64(Core::System& system, Handle* out_server_session_handle,
 | 
			
		||||
                       Handle* out_client_session_handle, bool is_light, uint64_t name) {
 | 
			
		||||
    R_RETURN(CreateSession(system, out_server_session_handle, out_client_session_handle, is_light,
 | 
			
		||||
                           name));
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
Result AcceptSession64(Core::System& system, Handle* out_handle, Handle port) {
 | 
			
		||||
    R_RETURN(AcceptSession(system, out_handle, port));
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
Result CreateSession64From32(Core::System& system, Handle* out_server_session_handle,
 | 
			
		||||
                             Handle* out_client_session_handle, bool is_light, uint32_t name) {
 | 
			
		||||
    R_RETURN(CreateSession(system, out_server_session_handle, out_client_session_handle, is_light,
 | 
			
		||||
                           name));
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
Result AcceptSession64From32(Core::System& system, Handle* out_handle, Handle port) {
 | 
			
		||||
    R_RETURN(AcceptSession(system, out_handle, port));
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
} // namespace Kernel::Svc
 | 
			
		||||
 
 | 
			
		||||
@@ -67,11 +67,6 @@ Result MapSharedMemory(Core::System& system, Handle shmem_handle, VAddr address,
 | 
			
		||||
    return ResultSuccess;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
Result MapSharedMemory32(Core::System& system, Handle shmem_handle, u32 address, u32 size,
 | 
			
		||||
                         Svc::MemoryPermission map_perm) {
 | 
			
		||||
    return MapSharedMemory(system, shmem_handle, address, size, map_perm);
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
Result UnmapSharedMemory(Core::System& system, Handle shmem_handle, VAddr address, u64 size) {
 | 
			
		||||
    // Validate the address/size.
 | 
			
		||||
    R_UNLESS(Common::IsAligned(address, PageSize), ResultInvalidAddress);
 | 
			
		||||
@@ -99,8 +94,40 @@ Result UnmapSharedMemory(Core::System& system, Handle shmem_handle, VAddr addres
 | 
			
		||||
    return ResultSuccess;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
Result UnmapSharedMemory32(Core::System& system, Handle shmem_handle, u32 address, u32 size) {
 | 
			
		||||
    return UnmapSharedMemory(system, shmem_handle, address, size);
 | 
			
		||||
Result CreateSharedMemory(Core::System& system, Handle* out_handle, uint64_t size,
 | 
			
		||||
                          MemoryPermission owner_perm, MemoryPermission remote_perm) {
 | 
			
		||||
    UNIMPLEMENTED();
 | 
			
		||||
    R_THROW(ResultNotImplemented);
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
Result MapSharedMemory64(Core::System& system, Handle shmem_handle, uint64_t address, uint64_t size,
 | 
			
		||||
                         MemoryPermission map_perm) {
 | 
			
		||||
    R_RETURN(MapSharedMemory(system, shmem_handle, address, size, map_perm));
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
Result UnmapSharedMemory64(Core::System& system, Handle shmem_handle, uint64_t address,
 | 
			
		||||
                           uint64_t size) {
 | 
			
		||||
    R_RETURN(UnmapSharedMemory(system, shmem_handle, address, size));
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
Result CreateSharedMemory64(Core::System& system, Handle* out_handle, uint64_t size,
 | 
			
		||||
                            MemoryPermission owner_perm, MemoryPermission remote_perm) {
 | 
			
		||||
    R_RETURN(CreateSharedMemory(system, out_handle, size, owner_perm, remote_perm));
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
Result MapSharedMemory64From32(Core::System& system, Handle shmem_handle, uint32_t address,
 | 
			
		||||
                               uint32_t size, MemoryPermission map_perm) {
 | 
			
		||||
    R_RETURN(MapSharedMemory(system, shmem_handle, address, size, map_perm));
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
Result UnmapSharedMemory64From32(Core::System& system, Handle shmem_handle, uint32_t address,
 | 
			
		||||
                                 uint32_t size) {
 | 
			
		||||
    R_RETURN(UnmapSharedMemory(system, shmem_handle, address, size));
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
Result CreateSharedMemory64From32(Core::System& system, Handle* out_handle, uint32_t size,
 | 
			
		||||
                                  MemoryPermission owner_perm, MemoryPermission remote_perm) {
 | 
			
		||||
    R_RETURN(CreateSharedMemory(system, out_handle, size, owner_perm, remote_perm));
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
} // namespace Kernel::Svc
 | 
			
		||||
 
 | 
			
		||||
@@ -20,10 +20,6 @@ Result CloseHandle(Core::System& system, Handle handle) {
 | 
			
		||||
    return ResultSuccess;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
Result CloseHandle32(Core::System& system, Handle handle) {
 | 
			
		||||
    return CloseHandle(system, handle);
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
/// Clears the signaled state of an event or process.
 | 
			
		||||
Result ResetSignal(Core::System& system, Handle handle) {
 | 
			
		||||
    LOG_DEBUG(Kernel_SVC, "called handle 0x{:08X}", handle);
 | 
			
		||||
@@ -52,10 +48,6 @@ Result ResetSignal(Core::System& system, Handle handle) {
 | 
			
		||||
    return ResultInvalidHandle;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
Result ResetSignal32(Core::System& system, Handle handle) {
 | 
			
		||||
    return ResetSignal(system, handle);
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
/// Wait for the given handles to synchronize, timeout after the specified nanoseconds
 | 
			
		||||
Result WaitSynchronization(Core::System& system, s32* index, VAddr handles_address, s32 num_handles,
 | 
			
		||||
                           s64 nano_seconds) {
 | 
			
		||||
@@ -93,12 +85,6 @@ Result WaitSynchronization(Core::System& system, s32* index, VAddr handles_addre
 | 
			
		||||
                                        nano_seconds);
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
Result WaitSynchronization32(Core::System& system, u32 timeout_low, u32 handles_address,
 | 
			
		||||
                             s32 num_handles, u32 timeout_high, s32* index) {
 | 
			
		||||
    const s64 nano_seconds{(static_cast<s64>(timeout_high) << 32) | static_cast<s64>(timeout_low)};
 | 
			
		||||
    return WaitSynchronization(system, index, handles_address, num_handles, nano_seconds);
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
/// Resumes a thread waiting on WaitSynchronization
 | 
			
		||||
Result CancelSynchronization(Core::System& system, Handle handle) {
 | 
			
		||||
    LOG_TRACE(Kernel_SVC, "called handle=0x{:X}", handle);
 | 
			
		||||
@@ -113,10 +99,6 @@ Result CancelSynchronization(Core::System& system, Handle handle) {
 | 
			
		||||
    return ResultSuccess;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
Result CancelSynchronization32(Core::System& system, Handle handle) {
 | 
			
		||||
    return CancelSynchronization(system, handle);
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
void SynchronizePreemptionState(Core::System& system) {
 | 
			
		||||
    auto& kernel = system.Kernel();
 | 
			
		||||
 | 
			
		||||
@@ -136,4 +118,46 @@ void SynchronizePreemptionState(Core::System& system) {
 | 
			
		||||
    }
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
Result CloseHandle64(Core::System& system, Handle handle) {
 | 
			
		||||
    R_RETURN(CloseHandle(system, handle));
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
Result ResetSignal64(Core::System& system, Handle handle) {
 | 
			
		||||
    R_RETURN(ResetSignal(system, handle));
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
Result WaitSynchronization64(Core::System& system, int32_t* out_index, uint64_t handles,
 | 
			
		||||
                             int32_t num_handles, int64_t timeout_ns) {
 | 
			
		||||
    R_RETURN(WaitSynchronization(system, out_index, handles, num_handles, timeout_ns));
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
Result CancelSynchronization64(Core::System& system, Handle handle) {
 | 
			
		||||
    R_RETURN(CancelSynchronization(system, handle));
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
void SynchronizePreemptionState64(Core::System& system) {
 | 
			
		||||
    SynchronizePreemptionState(system);
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
Result CloseHandle64From32(Core::System& system, Handle handle) {
 | 
			
		||||
    R_RETURN(CloseHandle(system, handle));
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
Result ResetSignal64From32(Core::System& system, Handle handle) {
 | 
			
		||||
    R_RETURN(ResetSignal(system, handle));
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
Result WaitSynchronization64From32(Core::System& system, int32_t* out_index, uint32_t handles,
 | 
			
		||||
                                   int32_t num_handles, int64_t timeout_ns) {
 | 
			
		||||
    R_RETURN(WaitSynchronization(system, out_index, handles, num_handles, timeout_ns));
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
Result CancelSynchronization64From32(Core::System& system, Handle handle) {
 | 
			
		||||
    R_RETURN(CancelSynchronization(system, handle));
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
void SynchronizePreemptionState64From32(Core::System& system) {
 | 
			
		||||
    SynchronizePreemptionState(system);
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
} // namespace Kernel::Svc
 | 
			
		||||
 
 | 
			
		||||
@@ -20,7 +20,7 @@ constexpr bool IsValidVirtualCoreId(int32_t core_id) {
 | 
			
		||||
 | 
			
		||||
/// Creates a new thread
 | 
			
		||||
Result CreateThread(Core::System& system, Handle* out_handle, VAddr entry_point, u64 arg,
 | 
			
		||||
                    VAddr stack_bottom, u32 priority, s32 core_id) {
 | 
			
		||||
                    VAddr stack_bottom, s32 priority, s32 core_id) {
 | 
			
		||||
    LOG_DEBUG(Kernel_SVC,
 | 
			
		||||
              "called entry_point=0x{:08X}, arg=0x{:08X}, stack_bottom=0x{:08X}, "
 | 
			
		||||
              "priority=0x{:08X}, core_id=0x{:08X}",
 | 
			
		||||
@@ -91,11 +91,6 @@ Result CreateThread(Core::System& system, Handle* out_handle, VAddr entry_point,
 | 
			
		||||
    return ResultSuccess;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
Result CreateThread32(Core::System& system, Handle* out_handle, u32 priority, u32 entry_point,
 | 
			
		||||
                      u32 arg, u32 stack_top, s32 processor_id) {
 | 
			
		||||
    return CreateThread(system, out_handle, entry_point, arg, stack_top, priority, processor_id);
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
/// Starts the thread for the provided handle
 | 
			
		||||
Result StartThread(Core::System& system, Handle thread_handle) {
 | 
			
		||||
    LOG_DEBUG(Kernel_SVC, "called thread=0x{:08X}", thread_handle);
 | 
			
		||||
@@ -115,10 +110,6 @@ Result StartThread(Core::System& system, Handle thread_handle) {
 | 
			
		||||
    return ResultSuccess;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
Result StartThread32(Core::System& system, Handle thread_handle) {
 | 
			
		||||
    return StartThread(system, thread_handle);
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
/// Called when a thread exits
 | 
			
		||||
void ExitThread(Core::System& system) {
 | 
			
		||||
    LOG_DEBUG(Kernel_SVC, "called, pc=0x{:08X}", system.CurrentArmInterface().GetPC());
 | 
			
		||||
@@ -129,10 +120,6 @@ void ExitThread(Core::System& system) {
 | 
			
		||||
    system.Kernel().UnregisterInUseObject(current_thread);
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
void ExitThread32(Core::System& system) {
 | 
			
		||||
    ExitThread(system);
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
/// Sleep the current thread
 | 
			
		||||
void SleepThread(Core::System& system, s64 nanoseconds) {
 | 
			
		||||
    auto& kernel = system.Kernel();
 | 
			
		||||
@@ -160,13 +147,8 @@ void SleepThread(Core::System& system, s64 nanoseconds) {
 | 
			
		||||
    }
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
void SleepThread32(Core::System& system, u32 nanoseconds_low, u32 nanoseconds_high) {
 | 
			
		||||
    const auto nanoseconds = static_cast<s64>(u64{nanoseconds_low} | (u64{nanoseconds_high} << 32));
 | 
			
		||||
    SleepThread(system, nanoseconds);
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
/// Gets the thread context
 | 
			
		||||
Result GetThreadContext(Core::System& system, VAddr out_context, Handle thread_handle) {
 | 
			
		||||
Result GetThreadContext3(Core::System& system, VAddr out_context, Handle thread_handle) {
 | 
			
		||||
    LOG_DEBUG(Kernel_SVC, "called, out_context=0x{:08X}, thread_handle=0x{:X}", out_context,
 | 
			
		||||
              thread_handle);
 | 
			
		||||
 | 
			
		||||
@@ -223,12 +205,8 @@ Result GetThreadContext(Core::System& system, VAddr out_context, Handle thread_h
 | 
			
		||||
    return ResultSuccess;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
Result GetThreadContext32(Core::System& system, u32 out_context, Handle thread_handle) {
 | 
			
		||||
    return GetThreadContext(system, out_context, thread_handle);
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
/// Gets the priority for the specified thread
 | 
			
		||||
Result GetThreadPriority(Core::System& system, u32* out_priority, Handle handle) {
 | 
			
		||||
Result GetThreadPriority(Core::System& system, s32* out_priority, Handle handle) {
 | 
			
		||||
    LOG_TRACE(Kernel_SVC, "called");
 | 
			
		||||
 | 
			
		||||
    // Get the thread from its handle.
 | 
			
		||||
@@ -241,12 +219,8 @@ Result GetThreadPriority(Core::System& system, u32* out_priority, Handle handle)
 | 
			
		||||
    return ResultSuccess;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
Result GetThreadPriority32(Core::System& system, u32* out_priority, Handle handle) {
 | 
			
		||||
    return GetThreadPriority(system, out_priority, handle);
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
/// Sets the priority for the specified thread
 | 
			
		||||
Result SetThreadPriority(Core::System& system, Handle thread_handle, u32 priority) {
 | 
			
		||||
Result SetThreadPriority(Core::System& system, Handle thread_handle, s32 priority) {
 | 
			
		||||
    // Get the current process.
 | 
			
		||||
    KProcess& process = *system.Kernel().CurrentProcess();
 | 
			
		||||
 | 
			
		||||
@@ -264,12 +238,8 @@ Result SetThreadPriority(Core::System& system, Handle thread_handle, u32 priorit
 | 
			
		||||
    return ResultSuccess;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
Result SetThreadPriority32(Core::System& system, Handle thread_handle, u32 priority) {
 | 
			
		||||
    return SetThreadPriority(system, thread_handle, priority);
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
Result GetThreadList(Core::System& system, u32* out_num_threads, VAddr out_thread_ids,
 | 
			
		||||
                     u32 out_thread_ids_size, Handle debug_handle) {
 | 
			
		||||
Result GetThreadList(Core::System& system, s32* out_num_threads, VAddr out_thread_ids,
 | 
			
		||||
                     s32 out_thread_ids_size, Handle debug_handle) {
 | 
			
		||||
    // TODO: Handle this case when debug events are supported.
 | 
			
		||||
    UNIMPLEMENTED_IF(debug_handle != InvalidHandle);
 | 
			
		||||
 | 
			
		||||
@@ -296,7 +266,7 @@ Result GetThreadList(Core::System& system, u32* out_num_threads, VAddr out_threa
 | 
			
		||||
    auto& memory = system.Memory();
 | 
			
		||||
    const auto& thread_list = current_process->GetThreadList();
 | 
			
		||||
    const auto num_threads = thread_list.size();
 | 
			
		||||
    const auto copy_amount = std::min(std::size_t{out_thread_ids_size}, num_threads);
 | 
			
		||||
    const auto copy_amount = std::min(static_cast<std::size_t>(out_thread_ids_size), num_threads);
 | 
			
		||||
 | 
			
		||||
    auto list_iter = thread_list.cbegin();
 | 
			
		||||
    for (std::size_t i = 0; i < copy_amount; ++i, ++list_iter) {
 | 
			
		||||
@@ -308,8 +278,8 @@ Result GetThreadList(Core::System& system, u32* out_num_threads, VAddr out_threa
 | 
			
		||||
    return ResultSuccess;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
Result GetThreadCoreMask(Core::System& system, Handle thread_handle, s32* out_core_id,
 | 
			
		||||
                         u64* out_affinity_mask) {
 | 
			
		||||
Result GetThreadCoreMask(Core::System& system, s32* out_core_id, u64* out_affinity_mask,
 | 
			
		||||
                         Handle thread_handle) {
 | 
			
		||||
    LOG_TRACE(Kernel_SVC, "called, handle=0x{:08X}", thread_handle);
 | 
			
		||||
 | 
			
		||||
    // Get the thread from its handle.
 | 
			
		||||
@@ -323,15 +293,6 @@ Result GetThreadCoreMask(Core::System& system, Handle thread_handle, s32* out_co
 | 
			
		||||
    return ResultSuccess;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
Result GetThreadCoreMask32(Core::System& system, Handle thread_handle, s32* out_core_id,
 | 
			
		||||
                           u32* out_affinity_mask_low, u32* out_affinity_mask_high) {
 | 
			
		||||
    u64 out_affinity_mask{};
 | 
			
		||||
    const auto result = GetThreadCoreMask(system, thread_handle, out_core_id, &out_affinity_mask);
 | 
			
		||||
    *out_affinity_mask_high = static_cast<u32>(out_affinity_mask >> 32);
 | 
			
		||||
    *out_affinity_mask_low = static_cast<u32>(out_affinity_mask);
 | 
			
		||||
    return result;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
Result SetThreadCoreMask(Core::System& system, Handle thread_handle, s32 core_id,
 | 
			
		||||
                         u64 affinity_mask) {
 | 
			
		||||
    // Determine the core id/affinity mask.
 | 
			
		||||
@@ -364,12 +325,6 @@ Result SetThreadCoreMask(Core::System& system, Handle thread_handle, s32 core_id
 | 
			
		||||
    return ResultSuccess;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
Result SetThreadCoreMask32(Core::System& system, Handle thread_handle, s32 core_id,
 | 
			
		||||
                           u32 affinity_mask_low, u32 affinity_mask_high) {
 | 
			
		||||
    const auto affinity_mask = u64{affinity_mask_low} | (u64{affinity_mask_high} << 32);
 | 
			
		||||
    return SetThreadCoreMask(system, thread_handle, core_id, affinity_mask);
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
/// Get the ID for the specified thread.
 | 
			
		||||
Result GetThreadId(Core::System& system, u64* out_thread_id, Handle thread_handle) {
 | 
			
		||||
    // Get the thread from its handle.
 | 
			
		||||
@@ -382,15 +337,101 @@ Result GetThreadId(Core::System& system, u64* out_thread_id, Handle thread_handl
 | 
			
		||||
    return ResultSuccess;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
Result GetThreadId32(Core::System& system, u32* out_thread_id_low, u32* out_thread_id_high,
 | 
			
		||||
                     Handle thread_handle) {
 | 
			
		||||
    u64 out_thread_id{};
 | 
			
		||||
    const Result result{GetThreadId(system, &out_thread_id, thread_handle)};
 | 
			
		||||
Result CreateThread64(Core::System& system, Handle* out_handle, uint64_t func, uint64_t arg,
 | 
			
		||||
                      uint64_t stack_bottom, int32_t priority, int32_t core_id) {
 | 
			
		||||
    R_RETURN(CreateThread(system, out_handle, func, arg, stack_bottom, priority, core_id));
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
    *out_thread_id_low = static_cast<u32>(out_thread_id >> 32);
 | 
			
		||||
    *out_thread_id_high = static_cast<u32>(out_thread_id & std::numeric_limits<u32>::max());
 | 
			
		||||
Result StartThread64(Core::System& system, Handle thread_handle) {
 | 
			
		||||
    R_RETURN(StartThread(system, thread_handle));
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
    return result;
 | 
			
		||||
void ExitThread64(Core::System& system) {
 | 
			
		||||
    return ExitThread(system);
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
void SleepThread64(Core::System& system, int64_t ns) {
 | 
			
		||||
    return SleepThread(system, ns);
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
Result GetThreadPriority64(Core::System& system, int32_t* out_priority, Handle thread_handle) {
 | 
			
		||||
    R_RETURN(GetThreadPriority(system, out_priority, thread_handle));
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
Result SetThreadPriority64(Core::System& system, Handle thread_handle, int32_t priority) {
 | 
			
		||||
    R_RETURN(SetThreadPriority(system, thread_handle, priority));
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
Result GetThreadCoreMask64(Core::System& system, int32_t* out_core_id, uint64_t* out_affinity_mask,
 | 
			
		||||
                           Handle thread_handle) {
 | 
			
		||||
    R_RETURN(GetThreadCoreMask(system, out_core_id, out_affinity_mask, thread_handle));
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
Result SetThreadCoreMask64(Core::System& system, Handle thread_handle, int32_t core_id,
 | 
			
		||||
                           uint64_t affinity_mask) {
 | 
			
		||||
    R_RETURN(SetThreadCoreMask(system, thread_handle, core_id, affinity_mask));
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
Result GetThreadId64(Core::System& system, uint64_t* out_thread_id, Handle thread_handle) {
 | 
			
		||||
    R_RETURN(GetThreadId(system, out_thread_id, thread_handle));
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
Result GetThreadContext364(Core::System& system, uint64_t out_context, Handle thread_handle) {
 | 
			
		||||
    R_RETURN(GetThreadContext3(system, out_context, thread_handle));
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
Result GetThreadList64(Core::System& system, int32_t* out_num_threads, uint64_t out_thread_ids,
 | 
			
		||||
                       int32_t max_out_count, Handle debug_handle) {
 | 
			
		||||
    R_RETURN(GetThreadList(system, out_num_threads, out_thread_ids, max_out_count, debug_handle));
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
Result CreateThread64From32(Core::System& system, Handle* out_handle, uint32_t func, uint32_t arg,
 | 
			
		||||
                            uint32_t stack_bottom, int32_t priority, int32_t core_id) {
 | 
			
		||||
    R_RETURN(CreateThread(system, out_handle, func, arg, stack_bottom, priority, core_id));
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
Result StartThread64From32(Core::System& system, Handle thread_handle) {
 | 
			
		||||
    R_RETURN(StartThread(system, thread_handle));
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
void ExitThread64From32(Core::System& system) {
 | 
			
		||||
    return ExitThread(system);
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
void SleepThread64From32(Core::System& system, int64_t ns) {
 | 
			
		||||
    return SleepThread(system, ns);
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
Result GetThreadPriority64From32(Core::System& system, int32_t* out_priority,
 | 
			
		||||
                                 Handle thread_handle) {
 | 
			
		||||
    R_RETURN(GetThreadPriority(system, out_priority, thread_handle));
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
Result SetThreadPriority64From32(Core::System& system, Handle thread_handle, int32_t priority) {
 | 
			
		||||
    R_RETURN(SetThreadPriority(system, thread_handle, priority));
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
Result GetThreadCoreMask64From32(Core::System& system, int32_t* out_core_id,
 | 
			
		||||
                                 uint64_t* out_affinity_mask, Handle thread_handle) {
 | 
			
		||||
    R_RETURN(GetThreadCoreMask(system, out_core_id, out_affinity_mask, thread_handle));
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
Result SetThreadCoreMask64From32(Core::System& system, Handle thread_handle, int32_t core_id,
 | 
			
		||||
                                 uint64_t affinity_mask) {
 | 
			
		||||
    R_RETURN(SetThreadCoreMask(system, thread_handle, core_id, affinity_mask));
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
Result GetThreadId64From32(Core::System& system, uint64_t* out_thread_id, Handle thread_handle) {
 | 
			
		||||
    R_RETURN(GetThreadId(system, out_thread_id, thread_handle));
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
Result GetThreadContext364From32(Core::System& system, uint32_t out_context, Handle thread_handle) {
 | 
			
		||||
    R_RETURN(GetThreadContext3(system, out_context, thread_handle));
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
Result GetThreadList64From32(Core::System& system, int32_t* out_num_threads,
 | 
			
		||||
                             uint32_t out_thread_ids, int32_t max_out_count, Handle debug_handle) {
 | 
			
		||||
    R_RETURN(GetThreadList(system, out_num_threads, out_thread_ids, max_out_count, debug_handle));
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
} // namespace Kernel::Svc
 | 
			
		||||
 
 | 
			
		||||
@@ -2,5 +2,59 @@
 | 
			
		||||
// SPDX-License-Identifier: GPL-2.0-or-later
 | 
			
		||||
 | 
			
		||||
#include "core/hle/kernel/svc.h"
 | 
			
		||||
#include "core/hle/kernel/svc_results.h"
 | 
			
		||||
 | 
			
		||||
namespace Kernel::Svc {} // namespace Kernel::Svc
 | 
			
		||||
namespace Kernel::Svc {
 | 
			
		||||
 | 
			
		||||
Result GetDebugFutureThreadInfo(Core::System& system, lp64::LastThreadContext* out_context,
 | 
			
		||||
                                uint64_t* out_thread_id, Handle debug_handle, int64_t ns) {
 | 
			
		||||
    UNIMPLEMENTED();
 | 
			
		||||
    R_THROW(ResultNotImplemented);
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
Result GetLastThreadInfo(Core::System& system, lp64::LastThreadContext* out_context,
 | 
			
		||||
                         uint64_t* out_tls_address, uint32_t* out_flags) {
 | 
			
		||||
    UNIMPLEMENTED();
 | 
			
		||||
    R_THROW(ResultNotImplemented);
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
Result GetDebugFutureThreadInfo64(Core::System& system, lp64::LastThreadContext* out_context,
 | 
			
		||||
                                  uint64_t* out_thread_id, Handle debug_handle, int64_t ns) {
 | 
			
		||||
    R_RETURN(GetDebugFutureThreadInfo(system, out_context, out_thread_id, debug_handle, ns));
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
Result GetLastThreadInfo64(Core::System& system, lp64::LastThreadContext* out_context,
 | 
			
		||||
                           uint64_t* out_tls_address, uint32_t* out_flags) {
 | 
			
		||||
    R_RETURN(GetLastThreadInfo(system, out_context, out_tls_address, out_flags));
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
Result GetDebugFutureThreadInfo64From32(Core::System& system, ilp32::LastThreadContext* out_context,
 | 
			
		||||
                                        uint64_t* out_thread_id, Handle debug_handle, int64_t ns) {
 | 
			
		||||
    lp64::LastThreadContext context{};
 | 
			
		||||
    R_TRY(
 | 
			
		||||
        GetDebugFutureThreadInfo(system, std::addressof(context), out_thread_id, debug_handle, ns));
 | 
			
		||||
 | 
			
		||||
    *out_context = {
 | 
			
		||||
        .fp = static_cast<u32>(context.fp),
 | 
			
		||||
        .sp = static_cast<u32>(context.sp),
 | 
			
		||||
        .lr = static_cast<u32>(context.lr),
 | 
			
		||||
        .pc = static_cast<u32>(context.pc),
 | 
			
		||||
    };
 | 
			
		||||
    R_SUCCEED();
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
Result GetLastThreadInfo64From32(Core::System& system, ilp32::LastThreadContext* out_context,
 | 
			
		||||
                                 uint64_t* out_tls_address, uint32_t* out_flags) {
 | 
			
		||||
    lp64::LastThreadContext context{};
 | 
			
		||||
    R_TRY(GetLastThreadInfo(system, std::addressof(context), out_tls_address, out_flags));
 | 
			
		||||
 | 
			
		||||
    *out_context = {
 | 
			
		||||
        .fp = static_cast<u32>(context.fp),
 | 
			
		||||
        .sp = static_cast<u32>(context.sp),
 | 
			
		||||
        .lr = static_cast<u32>(context.lr),
 | 
			
		||||
        .pc = static_cast<u32>(context.pc),
 | 
			
		||||
    };
 | 
			
		||||
    R_SUCCEED();
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
} // namespace Kernel::Svc
 | 
			
		||||
 
 | 
			
		||||
@@ -9,7 +9,7 @@
 | 
			
		||||
namespace Kernel::Svc {
 | 
			
		||||
 | 
			
		||||
/// This returns the total CPU ticks elapsed since the CPU was powered-on
 | 
			
		||||
u64 GetSystemTick(Core::System& system) {
 | 
			
		||||
int64_t GetSystemTick(Core::System& system) {
 | 
			
		||||
    LOG_TRACE(Kernel_SVC, "called");
 | 
			
		||||
 | 
			
		||||
    auto& core_timing = system.CoreTiming();
 | 
			
		||||
@@ -21,13 +21,15 @@ u64 GetSystemTick(Core::System& system) {
 | 
			
		||||
        core_timing.AddTicks(400U);
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    return result;
 | 
			
		||||
    return static_cast<int64_t>(result);
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
void GetSystemTick32(Core::System& system, u32* time_low, u32* time_high) {
 | 
			
		||||
    const auto time = GetSystemTick(system);
 | 
			
		||||
    *time_low = static_cast<u32>(time);
 | 
			
		||||
    *time_high = static_cast<u32>(time >> 32);
 | 
			
		||||
int64_t GetSystemTick64(Core::System& system) {
 | 
			
		||||
    return GetSystemTick(system);
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
int64_t GetSystemTick64From32(Core::System& system) {
 | 
			
		||||
    return GetSystemTick(system);
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
} // namespace Kernel::Svc
 | 
			
		||||
 
 | 
			
		||||
@@ -72,8 +72,46 @@ Result CreateTransferMemory(Core::System& system, Handle* out, VAddr address, u6
 | 
			
		||||
    return ResultSuccess;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
Result CreateTransferMemory32(Core::System& system, Handle* out, u32 address, u32 size,
 | 
			
		||||
                              MemoryPermission map_perm) {
 | 
			
		||||
    return CreateTransferMemory(system, out, address, size, map_perm);
 | 
			
		||||
Result MapTransferMemory(Core::System& system, Handle trmem_handle, uint64_t address, uint64_t size,
 | 
			
		||||
                         MemoryPermission owner_perm) {
 | 
			
		||||
    UNIMPLEMENTED();
 | 
			
		||||
    R_THROW(ResultNotImplemented);
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
Result UnmapTransferMemory(Core::System& system, Handle trmem_handle, uint64_t address,
 | 
			
		||||
                           uint64_t size) {
 | 
			
		||||
    UNIMPLEMENTED();
 | 
			
		||||
    R_THROW(ResultNotImplemented);
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
Result MapTransferMemory64(Core::System& system, Handle trmem_handle, uint64_t address,
 | 
			
		||||
                           uint64_t size, MemoryPermission owner_perm) {
 | 
			
		||||
    R_RETURN(MapTransferMemory(system, trmem_handle, address, size, owner_perm));
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
Result UnmapTransferMemory64(Core::System& system, Handle trmem_handle, uint64_t address,
 | 
			
		||||
                             uint64_t size) {
 | 
			
		||||
    R_RETURN(UnmapTransferMemory(system, trmem_handle, address, size));
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
Result CreateTransferMemory64(Core::System& system, Handle* out_handle, uint64_t address,
 | 
			
		||||
                              uint64_t size, MemoryPermission map_perm) {
 | 
			
		||||
    R_RETURN(CreateTransferMemory(system, out_handle, address, size, map_perm));
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
Result MapTransferMemory64From32(Core::System& system, Handle trmem_handle, uint32_t address,
 | 
			
		||||
                                 uint32_t size, MemoryPermission owner_perm) {
 | 
			
		||||
    R_RETURN(MapTransferMemory(system, trmem_handle, address, size, owner_perm));
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
Result UnmapTransferMemory64From32(Core::System& system, Handle trmem_handle, uint32_t address,
 | 
			
		||||
                                   uint32_t size) {
 | 
			
		||||
    R_RETURN(UnmapTransferMemory(system, trmem_handle, address, size));
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
Result CreateTransferMemory64From32(Core::System& system, Handle* out_handle, uint32_t address,
 | 
			
		||||
                                    uint32_t size, MemoryPermission map_perm) {
 | 
			
		||||
    R_RETURN(CreateTransferMemory(system, out_handle, address, size, map_perm));
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
} // namespace Kernel::Svc
 | 
			
		||||
 
 | 
			
		||||
							
								
								
									
										716
									
								
								src/core/hle/kernel/svc_generator.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										716
									
								
								src/core/hle/kernel/svc_generator.py
									
									
									
									
									
										Normal file
									
								
							@@ -0,0 +1,716 @@
 | 
			
		||||
# SPDX-FileCopyrightText: Copyright 2023 yuzu Emulator Project
 | 
			
		||||
# SPDX-License-Identifier: GPL-2.0-or-later
 | 
			
		||||
 | 
			
		||||
# Raw SVC definitions from the kernel.
 | 
			
		||||
#
 | 
			
		||||
# Avoid modifying the prototypes; see below for how to customize generation
 | 
			
		||||
# for a given typename.
 | 
			
		||||
SVCS = [
 | 
			
		||||
     [0x01, "Result SetHeapSize(Address* out_address, Size size);"],
 | 
			
		||||
     [0x02, "Result SetMemoryPermission(Address address, Size size, MemoryPermission perm);"],
 | 
			
		||||
     [0x03, "Result SetMemoryAttribute(Address address, Size size, uint32_t mask, uint32_t attr);"],
 | 
			
		||||
     [0x04, "Result MapMemory(Address dst_address, Address src_address, Size size);"],
 | 
			
		||||
     [0x05, "Result UnmapMemory(Address dst_address, Address src_address, Size size);"],
 | 
			
		||||
     [0x06, "Result QueryMemory(Address out_memory_info, PageInfo* out_page_info, Address address);"],
 | 
			
		||||
     [0x07, "void ExitProcess();"],
 | 
			
		||||
     [0x08, "Result CreateThread(Handle* out_handle, ThreadFunc func, Address arg, Address stack_bottom, int32_t priority, int32_t core_id);"],
 | 
			
		||||
     [0x09, "Result StartThread(Handle thread_handle);"],
 | 
			
		||||
     [0x0A, "void ExitThread();"],
 | 
			
		||||
     [0x0B, "void SleepThread(int64_t ns);"],
 | 
			
		||||
     [0x0C, "Result GetThreadPriority(int32_t* out_priority, Handle thread_handle);"],
 | 
			
		||||
     [0x0D, "Result SetThreadPriority(Handle thread_handle, int32_t priority);"],
 | 
			
		||||
     [0x0E, "Result GetThreadCoreMask(int32_t* out_core_id, uint64_t* out_affinity_mask, Handle thread_handle);"],
 | 
			
		||||
     [0x0F, "Result SetThreadCoreMask(Handle thread_handle, int32_t core_id, uint64_t affinity_mask);"],
 | 
			
		||||
     [0x10, "int32_t GetCurrentProcessorNumber();"],
 | 
			
		||||
     [0x11, "Result SignalEvent(Handle event_handle);"],
 | 
			
		||||
     [0x12, "Result ClearEvent(Handle event_handle);"],
 | 
			
		||||
     [0x13, "Result MapSharedMemory(Handle shmem_handle, Address address, Size size, MemoryPermission map_perm);"],
 | 
			
		||||
     [0x14, "Result UnmapSharedMemory(Handle shmem_handle, Address address, Size size);"],
 | 
			
		||||
     [0x15, "Result CreateTransferMemory(Handle* out_handle, Address address, Size size, MemoryPermission map_perm);"],
 | 
			
		||||
     [0x16, "Result CloseHandle(Handle handle);"],
 | 
			
		||||
     [0x17, "Result ResetSignal(Handle handle);"],
 | 
			
		||||
     [0x18, "Result WaitSynchronization(int32_t* out_index, Address handles, int32_t num_handles, int64_t timeout_ns);"],
 | 
			
		||||
     [0x19, "Result CancelSynchronization(Handle handle);"],
 | 
			
		||||
     [0x1A, "Result ArbitrateLock(Handle thread_handle, Address address, uint32_t tag);"],
 | 
			
		||||
     [0x1B, "Result ArbitrateUnlock(Address address);"],
 | 
			
		||||
     [0x1C, "Result WaitProcessWideKeyAtomic(Address address, Address cv_key, uint32_t tag, int64_t timeout_ns);"],
 | 
			
		||||
     [0x1D, "void SignalProcessWideKey(Address cv_key, int32_t count);"],
 | 
			
		||||
     [0x1E, "int64_t GetSystemTick();"],
 | 
			
		||||
     [0x1F, "Result ConnectToNamedPort(Handle* out_handle, Address name);"],
 | 
			
		||||
     [0x20, "Result SendSyncRequestLight(Handle session_handle);"],
 | 
			
		||||
     [0x21, "Result SendSyncRequest(Handle session_handle);"],
 | 
			
		||||
     [0x22, "Result SendSyncRequestWithUserBuffer(Address message_buffer, Size message_buffer_size, Handle session_handle);"],
 | 
			
		||||
     [0x23, "Result SendAsyncRequestWithUserBuffer(Handle* out_event_handle, Address message_buffer, Size message_buffer_size, Handle session_handle);"],
 | 
			
		||||
     [0x24, "Result GetProcessId(uint64_t* out_process_id, Handle process_handle);"],
 | 
			
		||||
     [0x25, "Result GetThreadId(uint64_t* out_thread_id, Handle thread_handle);"],
 | 
			
		||||
     [0x26, "void Break(BreakReason break_reason, Address arg, Size size);"],
 | 
			
		||||
     [0x27, "Result OutputDebugString(Address debug_str, Size len);"],
 | 
			
		||||
     [0x28, "void ReturnFromException(Result result);"],
 | 
			
		||||
     [0x29, "Result GetInfo(uint64_t* out, InfoType info_type, Handle handle, uint64_t info_subtype);"],
 | 
			
		||||
     [0x2A, "void FlushEntireDataCache();"],
 | 
			
		||||
     [0x2B, "Result FlushDataCache(Address address, Size size);"],
 | 
			
		||||
     [0x2C, "Result MapPhysicalMemory(Address address, Size size);"],
 | 
			
		||||
     [0x2D, "Result UnmapPhysicalMemory(Address address, Size size);"],
 | 
			
		||||
     [0x2E, "Result GetDebugFutureThreadInfo(LastThreadContext* out_context, uint64_t* out_thread_id, Handle debug_handle, int64_t ns);"],
 | 
			
		||||
     [0x2F, "Result GetLastThreadInfo(LastThreadContext* out_context, Address* out_tls_address, uint32_t* out_flags);"],
 | 
			
		||||
     [0x30, "Result GetResourceLimitLimitValue(int64_t* out_limit_value, Handle resource_limit_handle, LimitableResource which);"],
 | 
			
		||||
     [0x31, "Result GetResourceLimitCurrentValue(int64_t* out_current_value, Handle resource_limit_handle, LimitableResource which);"],
 | 
			
		||||
     [0x32, "Result SetThreadActivity(Handle thread_handle, ThreadActivity thread_activity);"],
 | 
			
		||||
     [0x33, "Result GetThreadContext3(Address out_context, Handle thread_handle);"],
 | 
			
		||||
     [0x34, "Result WaitForAddress(Address address, ArbitrationType arb_type, int32_t value, int64_t timeout_ns);"],
 | 
			
		||||
     [0x35, "Result SignalToAddress(Address address, SignalType signal_type, int32_t value, int32_t count);"],
 | 
			
		||||
     [0x36, "void SynchronizePreemptionState();"],
 | 
			
		||||
     [0x37, "Result GetResourceLimitPeakValue(int64_t* out_peak_value, Handle resource_limit_handle, LimitableResource which);"],
 | 
			
		||||
 | 
			
		||||
     [0x39, "Result CreateIoPool(Handle* out_handle, IoPoolType which);"],
 | 
			
		||||
     [0x3A, "Result CreateIoRegion(Handle* out_handle, Handle io_pool, PhysicalAddress physical_address, Size size, MemoryMapping mapping, MemoryPermission perm);"],
 | 
			
		||||
 | 
			
		||||
     [0x3C, "void KernelDebug(KernelDebugType kern_debug_type, uint64_t arg0, uint64_t arg1, uint64_t arg2);"],
 | 
			
		||||
     [0x3D, "void ChangeKernelTraceState(KernelTraceState kern_trace_state);"],
 | 
			
		||||
 | 
			
		||||
     [0x40, "Result CreateSession(Handle* out_server_session_handle, Handle* out_client_session_handle, bool is_light, Address name);"],
 | 
			
		||||
     [0x41, "Result AcceptSession(Handle* out_handle, Handle port);"],
 | 
			
		||||
     [0x42, "Result ReplyAndReceiveLight(Handle handle);"],
 | 
			
		||||
     [0x43, "Result ReplyAndReceive(int32_t* out_index, Address handles, int32_t num_handles, Handle reply_target, int64_t timeout_ns);"],
 | 
			
		||||
     [0x44, "Result ReplyAndReceiveWithUserBuffer(int32_t* out_index, Address message_buffer, Size message_buffer_size, Address handles, int32_t num_handles, Handle reply_target, int64_t timeout_ns);"],
 | 
			
		||||
     [0x45, "Result CreateEvent(Handle* out_write_handle, Handle* out_read_handle);"],
 | 
			
		||||
     [0x46, "Result MapIoRegion(Handle io_region, Address address, Size size, MemoryPermission perm);"],
 | 
			
		||||
     [0x47, "Result UnmapIoRegion(Handle io_region, Address address, Size size);"],
 | 
			
		||||
     [0x48, "Result MapPhysicalMemoryUnsafe(Address address, Size size);"],
 | 
			
		||||
     [0x49, "Result UnmapPhysicalMemoryUnsafe(Address address, Size size);"],
 | 
			
		||||
     [0x4A, "Result SetUnsafeLimit(Size limit);"],
 | 
			
		||||
     [0x4B, "Result CreateCodeMemory(Handle* out_handle, Address address, Size size);"],
 | 
			
		||||
     [0x4C, "Result ControlCodeMemory(Handle code_memory_handle, CodeMemoryOperation operation, uint64_t address, uint64_t size, MemoryPermission perm);"],
 | 
			
		||||
     [0x4D, "void SleepSystem();"],
 | 
			
		||||
     [0x4E, "Result ReadWriteRegister(uint32_t* out_value, PhysicalAddress address, uint32_t mask, uint32_t value);"],
 | 
			
		||||
     [0x4F, "Result SetProcessActivity(Handle process_handle, ProcessActivity process_activity);"],
 | 
			
		||||
     [0x50, "Result CreateSharedMemory(Handle* out_handle, Size size, MemoryPermission owner_perm, MemoryPermission remote_perm);"],
 | 
			
		||||
     [0x51, "Result MapTransferMemory(Handle trmem_handle, Address address, Size size, MemoryPermission owner_perm);"],
 | 
			
		||||
     [0x52, "Result UnmapTransferMemory(Handle trmem_handle, Address address, Size size);"],
 | 
			
		||||
     [0x53, "Result CreateInterruptEvent(Handle* out_read_handle, int32_t interrupt_id, InterruptType interrupt_type);"],
 | 
			
		||||
     [0x54, "Result QueryPhysicalAddress(PhysicalMemoryInfo* out_info, Address address);"],
 | 
			
		||||
     [0x55, "Result QueryIoMapping(Address* out_address, Size* out_size, PhysicalAddress physical_address, Size size);"],
 | 
			
		||||
     [0x56, "Result CreateDeviceAddressSpace(Handle* out_handle, uint64_t das_address, uint64_t das_size);"],
 | 
			
		||||
     [0x57, "Result AttachDeviceAddressSpace(DeviceName device_name, Handle das_handle);"],
 | 
			
		||||
     [0x58, "Result DetachDeviceAddressSpace(DeviceName device_name, Handle das_handle);"],
 | 
			
		||||
     [0x59, "Result MapDeviceAddressSpaceByForce(Handle das_handle, Handle process_handle, uint64_t process_address, Size size, uint64_t device_address, uint32_t option);"],
 | 
			
		||||
     [0x5A, "Result MapDeviceAddressSpaceAligned(Handle das_handle, Handle process_handle, uint64_t process_address, Size size, uint64_t device_address, uint32_t option);"],
 | 
			
		||||
     [0x5C, "Result UnmapDeviceAddressSpace(Handle das_handle, Handle process_handle, uint64_t process_address, Size size, uint64_t device_address);"],
 | 
			
		||||
     [0x5D, "Result InvalidateProcessDataCache(Handle process_handle, uint64_t address, uint64_t size);"],
 | 
			
		||||
     [0x5E, "Result StoreProcessDataCache(Handle process_handle, uint64_t address, uint64_t size);"],
 | 
			
		||||
     [0x5F, "Result FlushProcessDataCache(Handle process_handle, uint64_t address, uint64_t size);"],
 | 
			
		||||
     [0x60, "Result DebugActiveProcess(Handle* out_handle, uint64_t process_id);"],
 | 
			
		||||
     [0x61, "Result BreakDebugProcess(Handle debug_handle);"],
 | 
			
		||||
     [0x62, "Result TerminateDebugProcess(Handle debug_handle);"],
 | 
			
		||||
     [0x63, "Result GetDebugEvent(Address out_info, Handle debug_handle);"],
 | 
			
		||||
     [0x64, "Result ContinueDebugEvent(Handle debug_handle, uint32_t flags, Address thread_ids, int32_t num_thread_ids);"],
 | 
			
		||||
     [0x65, "Result GetProcessList(int32_t* out_num_processes, Address out_process_ids, int32_t max_out_count);"],
 | 
			
		||||
     [0x66, "Result GetThreadList(int32_t* out_num_threads, Address out_thread_ids, int32_t max_out_count, Handle debug_handle);"],
 | 
			
		||||
     [0x67, "Result GetDebugThreadContext(Address out_context, Handle debug_handle, uint64_t thread_id, uint32_t context_flags);"],
 | 
			
		||||
     [0x68, "Result SetDebugThreadContext(Handle debug_handle, uint64_t thread_id, Address context, uint32_t context_flags);"],
 | 
			
		||||
     [0x69, "Result QueryDebugProcessMemory(Address out_memory_info, PageInfo* out_page_info, Handle process_handle, Address address);"],
 | 
			
		||||
     [0x6A, "Result ReadDebugProcessMemory(Address buffer, Handle debug_handle, Address address, Size size);"],
 | 
			
		||||
     [0x6B, "Result WriteDebugProcessMemory(Handle debug_handle, Address buffer, Address address, Size size);"],
 | 
			
		||||
     [0x6C, "Result SetHardwareBreakPoint(HardwareBreakPointRegisterName name, uint64_t flags, uint64_t value);"],
 | 
			
		||||
     [0x6D, "Result GetDebugThreadParam(uint64_t* out_64, uint32_t* out_32, Handle debug_handle, uint64_t thread_id, DebugThreadParam param);"],
 | 
			
		||||
 | 
			
		||||
     [0x6F, "Result GetSystemInfo(uint64_t* out, SystemInfoType info_type, Handle handle, uint64_t info_subtype);"],
 | 
			
		||||
     [0x70, "Result CreatePort(Handle* out_server_handle, Handle* out_client_handle, int32_t max_sessions, bool is_light, Address name);"],
 | 
			
		||||
     [0x71, "Result ManageNamedPort(Handle* out_server_handle, Address name, int32_t max_sessions);"],
 | 
			
		||||
     [0x72, "Result ConnectToPort(Handle* out_handle, Handle port);"],
 | 
			
		||||
     [0x73, "Result SetProcessMemoryPermission(Handle process_handle, uint64_t address, uint64_t size, MemoryPermission perm);"],
 | 
			
		||||
     [0x74, "Result MapProcessMemory(Address dst_address, Handle process_handle, uint64_t src_address, Size size);"],
 | 
			
		||||
     [0x75, "Result UnmapProcessMemory(Address dst_address, Handle process_handle, uint64_t src_address, Size size);"],
 | 
			
		||||
     [0x76, "Result QueryProcessMemory(Address out_memory_info, PageInfo* out_page_info, Handle process_handle, uint64_t address);"],
 | 
			
		||||
     [0x77, "Result MapProcessCodeMemory(Handle process_handle, uint64_t dst_address, uint64_t src_address, uint64_t size);"],
 | 
			
		||||
     [0x78, "Result UnmapProcessCodeMemory(Handle process_handle, uint64_t dst_address, uint64_t src_address, uint64_t size);"],
 | 
			
		||||
     [0x79, "Result CreateProcess(Handle* out_handle, Address parameters, Address caps, int32_t num_caps);"],
 | 
			
		||||
     [0x7A, "Result StartProcess(Handle process_handle, int32_t priority, int32_t core_id, uint64_t main_thread_stack_size);"],
 | 
			
		||||
     [0x7B, "Result TerminateProcess(Handle process_handle);"],
 | 
			
		||||
     [0x7C, "Result GetProcessInfo(int64_t* out_info, Handle process_handle, ProcessInfoType info_type);"],
 | 
			
		||||
     [0x7D, "Result CreateResourceLimit(Handle* out_handle);"],
 | 
			
		||||
     [0x7E, "Result SetResourceLimitLimitValue(Handle resource_limit_handle, LimitableResource which, int64_t limit_value);"],
 | 
			
		||||
     [0x7F, "void CallSecureMonitor(SecureMonitorArguments args);"],
 | 
			
		||||
 | 
			
		||||
     [0x90, "Result MapInsecureMemory(Address address, Size size);"],
 | 
			
		||||
     [0x91, "Result UnmapInsecureMemory(Address address, Size size);"],
 | 
			
		||||
]
 | 
			
		||||
 | 
			
		||||
# These use a custom ABI, and therefore require custom wrappers
 | 
			
		||||
SKIP_WRAPPERS = {
 | 
			
		||||
    0x20: "SendSyncRequestLight",
 | 
			
		||||
    0x42: "ReplyAndReceiveLight",
 | 
			
		||||
    0x7F: "CallSecureMonitor",
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
BIT_32 = 0
 | 
			
		||||
BIT_64 = 1
 | 
			
		||||
 | 
			
		||||
REG_SIZES = [4, 8]
 | 
			
		||||
SUFFIX_NAMES = ["64From32", "64"]
 | 
			
		||||
TYPE_SIZES = {
 | 
			
		||||
    # SVC types
 | 
			
		||||
    "ArbitrationType": 4,
 | 
			
		||||
    "BreakReason": 4,
 | 
			
		||||
    "CodeMemoryOperation": 4,
 | 
			
		||||
    "DebugThreadParam": 4,
 | 
			
		||||
    "DeviceName": 4,
 | 
			
		||||
    "HardwareBreakPointRegisterName": 4,
 | 
			
		||||
    "Handle": 4,
 | 
			
		||||
    "InfoType": 4,
 | 
			
		||||
    "InterruptType": 4,
 | 
			
		||||
    "IoPoolType": 4,
 | 
			
		||||
    "KernelDebugType": 4,
 | 
			
		||||
    "KernelTraceState": 4,
 | 
			
		||||
    "LimitableResource": 4,
 | 
			
		||||
    "MemoryMapping": 4,
 | 
			
		||||
    "MemoryPermission": 4,
 | 
			
		||||
    "PageInfo": 4,
 | 
			
		||||
    "ProcessActivity": 4,
 | 
			
		||||
    "ProcessInfoType": 4,
 | 
			
		||||
    "Result": 4,
 | 
			
		||||
    "SignalType": 4,
 | 
			
		||||
    "SystemInfoType": 4,
 | 
			
		||||
    "ThreadActivity": 4,
 | 
			
		||||
 | 
			
		||||
    # Arch-specific types
 | 
			
		||||
    "ilp32::LastThreadContext": 16,
 | 
			
		||||
    "ilp32::PhysicalMemoryInfo": 16,
 | 
			
		||||
    "ilp32::SecureMonitorArguments": 32,
 | 
			
		||||
    "lp64::LastThreadContext": 32,
 | 
			
		||||
    "lp64::PhysicalMemoryInfo": 24,
 | 
			
		||||
    "lp64::SecureMonitorArguments": 64,
 | 
			
		||||
 | 
			
		||||
    # Generic types
 | 
			
		||||
    "bool": 1,
 | 
			
		||||
    "int32_t": 4,
 | 
			
		||||
    "int64_t": 8,
 | 
			
		||||
    "uint32_t": 4,
 | 
			
		||||
    "uint64_t": 8,
 | 
			
		||||
    "void": 0,
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
TYPE_REPLACEMENTS = {
 | 
			
		||||
    "Address": ["uint32_t", "uint64_t"],
 | 
			
		||||
    "LastThreadContext": ["ilp32::LastThreadContext", "lp64::LastThreadContext"],
 | 
			
		||||
    "PhysicalAddress": ["uint64_t", "uint64_t"],
 | 
			
		||||
    "PhysicalMemoryInfo": ["ilp32::PhysicalMemoryInfo", "lp64::PhysicalMemoryInfo"],
 | 
			
		||||
    "SecureMonitorArguments": ["ilp32::SecureMonitorArguments", "lp64::SecureMonitorArguments"],
 | 
			
		||||
    "Size": ["uint32_t", "uint64_t"],
 | 
			
		||||
    "ThreadFunc": ["uint32_t", "uint64_t"],
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
# Statically verify that the hardcoded sizes match the intended
 | 
			
		||||
# sizes in C++.
 | 
			
		||||
def emit_size_check():
 | 
			
		||||
    lines = []
 | 
			
		||||
 | 
			
		||||
    for type, size in TYPE_SIZES.items():
 | 
			
		||||
        if type != "void":
 | 
			
		||||
            lines.append(f"static_assert(sizeof({type}) == {size});")
 | 
			
		||||
 | 
			
		||||
    return "\n".join(lines)
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
# Replaces a type with an arch-specific one, if it exists.
 | 
			
		||||
def substitute_type(name, bitness):
 | 
			
		||||
    if name in TYPE_REPLACEMENTS:
 | 
			
		||||
        return TYPE_REPLACEMENTS[name][bitness]
 | 
			
		||||
    else:
 | 
			
		||||
        return name
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
class Argument:
 | 
			
		||||
    def __init__(self, type_name, var_name, is_output, is_outptr, is_address):
 | 
			
		||||
        self.type_name = type_name
 | 
			
		||||
        self.var_name = var_name
 | 
			
		||||
        self.is_output = is_output
 | 
			
		||||
        self.is_outptr = is_outptr
 | 
			
		||||
        self.is_address = is_address
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
# Parses C-style string declarations for SVCs.
 | 
			
		||||
def parse_declaration(declaration, bitness):
 | 
			
		||||
    return_type, rest = declaration.split(" ", 1)
 | 
			
		||||
    func_name, rest = rest.split("(", 1)
 | 
			
		||||
    arg_names, rest = rest.split(")", 1)
 | 
			
		||||
    argument_types = []
 | 
			
		||||
 | 
			
		||||
    return_type = substitute_type(return_type, bitness)
 | 
			
		||||
    assert return_type in TYPE_SIZES, f"Unknown type '{return_type}'"
 | 
			
		||||
 | 
			
		||||
    if arg_names:
 | 
			
		||||
        for arg_name in arg_names.split(", "):
 | 
			
		||||
            type_name, var_name = arg_name.replace("*", "").split(" ", 1)
 | 
			
		||||
 | 
			
		||||
            # All outputs must contain out_ in the name.
 | 
			
		||||
            is_output = var_name == "out" or var_name.find("out_") != -1
 | 
			
		||||
 | 
			
		||||
            # User-pointer outputs are not written to registers.
 | 
			
		||||
            is_outptr = is_output and arg_name.find("*") == -1
 | 
			
		||||
 | 
			
		||||
            # Special handling is performed for output addresses to avoid awkwardness
 | 
			
		||||
            # in conversion for the 32-bit equivalents.
 | 
			
		||||
            is_address = is_output and not is_outptr and \
 | 
			
		||||
                type_name in ["Address", "Size"]
 | 
			
		||||
            type_name = substitute_type(type_name, bitness)
 | 
			
		||||
 | 
			
		||||
            assert type_name in TYPE_SIZES, f"Unknown type '{type_name}'"
 | 
			
		||||
 | 
			
		||||
            argument_types.append(
 | 
			
		||||
                Argument(type_name, var_name, is_output, is_outptr, is_address))
 | 
			
		||||
 | 
			
		||||
    return (return_type, func_name, argument_types)
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
class RegisterAllocator:
 | 
			
		||||
    def __init__(self, num_regs, byte_size, parameter_count):
 | 
			
		||||
        self.registers = {}
 | 
			
		||||
        self.num_regs = num_regs
 | 
			
		||||
        self.byte_size = byte_size
 | 
			
		||||
        self.parameter_count = parameter_count
 | 
			
		||||
 | 
			
		||||
    # Mark the given register as allocated, for use in layout
 | 
			
		||||
    # calculation if the NGRN exceeds the ABI parameter count.
 | 
			
		||||
    def allocate(self, i):
 | 
			
		||||
        assert i not in self.registers, f"Register R{i} already allocated"
 | 
			
		||||
        self.registers[i] = True
 | 
			
		||||
        return i
 | 
			
		||||
 | 
			
		||||
    # Calculate the next available location for a register;
 | 
			
		||||
    # the NGRN has exceeded the ABI parameter count.
 | 
			
		||||
    def allocate_first_free(self):
 | 
			
		||||
        for i in range(0, self.num_regs):
 | 
			
		||||
            if i in self.registers:
 | 
			
		||||
                continue
 | 
			
		||||
 | 
			
		||||
            self.allocate(i)
 | 
			
		||||
            return i
 | 
			
		||||
 | 
			
		||||
        assert False, "No registers available"
 | 
			
		||||
 | 
			
		||||
    # Add a single register at the given NGRN.
 | 
			
		||||
    # If the index exceeds the ABI parameter count, try to find a
 | 
			
		||||
    # location to add it. Returns the output location and increment.
 | 
			
		||||
    def add_single(self, ngrn):
 | 
			
		||||
        if ngrn >= self.parameter_count:
 | 
			
		||||
            return (self.allocate_first_free(), 0)
 | 
			
		||||
        else:
 | 
			
		||||
            return (self.allocate(ngrn), 1)
 | 
			
		||||
 | 
			
		||||
    # Add registers at the given NGRN for a data type of
 | 
			
		||||
    # the given size. Returns the output locations and increment.
 | 
			
		||||
    def add(self, ngrn, data_size, align=True):
 | 
			
		||||
        if data_size <= self.byte_size:
 | 
			
		||||
            r, i = self.add_single(ngrn)
 | 
			
		||||
            return ([r], i)
 | 
			
		||||
 | 
			
		||||
        regs = []
 | 
			
		||||
        inc = ngrn % 2 if align else 0
 | 
			
		||||
        remaining_size = data_size
 | 
			
		||||
        while remaining_size > 0:
 | 
			
		||||
            r, i = self.add_single(ngrn + inc)
 | 
			
		||||
            regs.append(r)
 | 
			
		||||
            inc += i
 | 
			
		||||
            remaining_size -= self.byte_size
 | 
			
		||||
 | 
			
		||||
        return (regs, inc)
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
def reg_alloc(bitness):
 | 
			
		||||
    if bitness == 0:
 | 
			
		||||
        # aapcs32: 4 4-byte registers
 | 
			
		||||
        return RegisterAllocator(8, 4, 4)
 | 
			
		||||
    elif bitness == 1:
 | 
			
		||||
        # aapcs64: 8 8-byte registers
 | 
			
		||||
        return RegisterAllocator(8, 8, 8)
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
# Converts a parsed SVC declaration into register lists for
 | 
			
		||||
# the return value, outputs, and inputs.
 | 
			
		||||
def get_registers(parse_result, bitness):
 | 
			
		||||
    output_alloc = reg_alloc(bitness)
 | 
			
		||||
    input_alloc = reg_alloc(bitness)
 | 
			
		||||
    return_type, _, arguments = parse_result
 | 
			
		||||
 | 
			
		||||
    return_write = []
 | 
			
		||||
    output_writes = []
 | 
			
		||||
    input_reads = []
 | 
			
		||||
 | 
			
		||||
    input_ngrn = 0
 | 
			
		||||
    output_ngrn = 0
 | 
			
		||||
 | 
			
		||||
    # Run the input calculation.
 | 
			
		||||
    for arg in arguments:
 | 
			
		||||
        if arg.is_output and not arg.is_outptr:
 | 
			
		||||
            input_ngrn += 1
 | 
			
		||||
            continue
 | 
			
		||||
 | 
			
		||||
        regs, increment = input_alloc.add(
 | 
			
		||||
            input_ngrn, TYPE_SIZES[arg.type_name], align=True)
 | 
			
		||||
        input_reads.append([arg.type_name, arg.var_name, regs])
 | 
			
		||||
        input_ngrn += increment
 | 
			
		||||
 | 
			
		||||
    # Include the return value if this SVC returns a value.
 | 
			
		||||
    if return_type != "void":
 | 
			
		||||
        regs, increment = output_alloc.add(
 | 
			
		||||
            output_ngrn, TYPE_SIZES[return_type], align=False)
 | 
			
		||||
        return_write.append([return_type, regs])
 | 
			
		||||
        output_ngrn += increment
 | 
			
		||||
 | 
			
		||||
    # Run the output calculation.
 | 
			
		||||
    for arg in arguments:
 | 
			
		||||
        if not arg.is_output or arg.is_outptr:
 | 
			
		||||
            continue
 | 
			
		||||
 | 
			
		||||
        regs, increment = output_alloc.add(
 | 
			
		||||
            output_ngrn, TYPE_SIZES[arg.type_name], align=False)
 | 
			
		||||
        output_writes.append(
 | 
			
		||||
            [arg.type_name, arg.var_name, regs, arg.is_address])
 | 
			
		||||
        output_ngrn += increment
 | 
			
		||||
 | 
			
		||||
    return (return_write, output_writes, input_reads)
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
# Collects possibly multiple source registers into the named C++ value.
 | 
			
		||||
def emit_gather(sources, name, type_name, reg_size):
 | 
			
		||||
    get_fn = f"GetReg{reg_size*8}"
 | 
			
		||||
 | 
			
		||||
    if len(sources) == 1:
 | 
			
		||||
        s, = sources
 | 
			
		||||
        line = f"{name} = Convert<{type_name}>({get_fn}(system, {s}));"
 | 
			
		||||
        return [line]
 | 
			
		||||
 | 
			
		||||
    var_type = f"std::array<uint{reg_size*8}_t, {len(sources)}>"
 | 
			
		||||
    lines = [
 | 
			
		||||
        f"{var_type} {name}_gather{{}};"
 | 
			
		||||
    ]
 | 
			
		||||
    for i in range(0, len(sources)):
 | 
			
		||||
        lines.append(
 | 
			
		||||
            f"{name}_gather[{i}] = {get_fn}(system, {sources[i]});")
 | 
			
		||||
 | 
			
		||||
    lines.append(f"{name} = Convert<{type_name}>({name}_gather);")
 | 
			
		||||
    return lines
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
# Produces one or more statements which assign the named C++ value
 | 
			
		||||
# into possibly multiple registers.
 | 
			
		||||
def emit_scatter(destinations, name, reg_size):
 | 
			
		||||
    set_fn = f"SetReg{reg_size*8}"
 | 
			
		||||
    reg_type = f"uint{reg_size*8}_t"
 | 
			
		||||
 | 
			
		||||
    if len(destinations) == 1:
 | 
			
		||||
        d, = destinations
 | 
			
		||||
        line = f"{set_fn}(system, {d}, Convert<{reg_type}>({name}));"
 | 
			
		||||
        return [line]
 | 
			
		||||
 | 
			
		||||
    var_type = f"std::array<{reg_type}, {len(destinations)}>"
 | 
			
		||||
    lines = [
 | 
			
		||||
        f"auto {name}_scatter = Convert<{var_type}>({name});"
 | 
			
		||||
    ]
 | 
			
		||||
 | 
			
		||||
    for i in range(0, len(destinations)):
 | 
			
		||||
        lines.append(
 | 
			
		||||
            f"{set_fn}(system, {destinations[i]}, {name}_scatter[{i}]);")
 | 
			
		||||
 | 
			
		||||
    return lines
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
def emit_lines(lines, indent='    '):
 | 
			
		||||
    output_lines = []
 | 
			
		||||
    first = True
 | 
			
		||||
    for line in lines:
 | 
			
		||||
        if line and not first:
 | 
			
		||||
            output_lines.append(indent + line)
 | 
			
		||||
        else:
 | 
			
		||||
            output_lines.append(line)
 | 
			
		||||
        first = False
 | 
			
		||||
 | 
			
		||||
    return "\n".join(output_lines)
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
# Emit a C++ function to wrap a guest SVC.
 | 
			
		||||
def emit_wrapper(wrapped_fn, suffix, register_info, arguments, byte_size):
 | 
			
		||||
    return_write, output_writes, input_reads = register_info
 | 
			
		||||
    lines = [
 | 
			
		||||
        f"static void SvcWrap_{wrapped_fn}{suffix}(Core::System& system) {{"
 | 
			
		||||
    ]
 | 
			
		||||
 | 
			
		||||
    # Get everything ready.
 | 
			
		||||
    for return_type, _ in return_write:
 | 
			
		||||
        lines.append(f"{return_type} ret{{}};")
 | 
			
		||||
    if return_write:
 | 
			
		||||
        lines.append("")
 | 
			
		||||
 | 
			
		||||
    for output_type, var_name, _, is_address in output_writes:
 | 
			
		||||
        output_type = "uintptr_t" if is_address else output_type
 | 
			
		||||
        lines.append(f"{output_type} {var_name}{{}};")
 | 
			
		||||
    for input_type, var_name, _ in input_reads:
 | 
			
		||||
        lines.append(f"{input_type} {var_name}{{}};")
 | 
			
		||||
 | 
			
		||||
    if output_writes or input_reads:
 | 
			
		||||
        lines.append("")
 | 
			
		||||
 | 
			
		||||
    for input_type, var_name, sources in input_reads:
 | 
			
		||||
        lines += emit_gather(sources, var_name, input_type, byte_size)
 | 
			
		||||
    if input_reads:
 | 
			
		||||
        lines.append("")
 | 
			
		||||
 | 
			
		||||
    # Build the call.
 | 
			
		||||
    call_arguments = ["system"]
 | 
			
		||||
    for arg in arguments:
 | 
			
		||||
        if arg.is_output and not arg.is_outptr:
 | 
			
		||||
            call_arguments.append(f"&{arg.var_name}")
 | 
			
		||||
        else:
 | 
			
		||||
            call_arguments.append(arg.var_name)
 | 
			
		||||
 | 
			
		||||
    line = ""
 | 
			
		||||
    if return_write:
 | 
			
		||||
        line += "ret = "
 | 
			
		||||
 | 
			
		||||
    line += f"{wrapped_fn}{suffix}({', '.join(call_arguments)});"
 | 
			
		||||
    lines.append(line)
 | 
			
		||||
 | 
			
		||||
    if return_write or output_writes:
 | 
			
		||||
        lines.append("")
 | 
			
		||||
 | 
			
		||||
    # Write back the return value and outputs.
 | 
			
		||||
    for _, destinations in return_write:
 | 
			
		||||
        lines += emit_scatter(destinations, "ret", byte_size)
 | 
			
		||||
    for _, var_name, destinations, _ in output_writes:
 | 
			
		||||
        lines += emit_scatter(destinations, var_name, byte_size)
 | 
			
		||||
 | 
			
		||||
    # Finish.
 | 
			
		||||
    return emit_lines(lines) + "\n}"
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
COPYRIGHT = """\
 | 
			
		||||
// SPDX-FileCopyrightText: Copyright 2023 yuzu Emulator Project
 | 
			
		||||
// SPDX-License-Identifier: GPL-2.0-or-later
 | 
			
		||||
 | 
			
		||||
// This file is automatically generated using svc_generator.py.
 | 
			
		||||
"""
 | 
			
		||||
 | 
			
		||||
PROLOGUE_H = """
 | 
			
		||||
#pragma once
 | 
			
		||||
 | 
			
		||||
namespace Core {
 | 
			
		||||
class System;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
#include "common/common_types.h"
 | 
			
		||||
#include "core/hle/kernel/svc_types.h"
 | 
			
		||||
#include "core/hle/result.h"
 | 
			
		||||
 | 
			
		||||
namespace Kernel::Svc {
 | 
			
		||||
 | 
			
		||||
// clang-format off
 | 
			
		||||
"""
 | 
			
		||||
 | 
			
		||||
EPILOGUE_H = """
 | 
			
		||||
// clang-format on
 | 
			
		||||
 | 
			
		||||
// Custom ABI.
 | 
			
		||||
Result ReplyAndReceiveLight(Core::System& system, Handle handle, uint32_t* args);
 | 
			
		||||
Result ReplyAndReceiveLight64From32(Core::System& system, Handle handle, uint32_t* args);
 | 
			
		||||
Result ReplyAndReceiveLight64(Core::System& system, Handle handle, uint32_t* args);
 | 
			
		||||
 | 
			
		||||
Result SendSyncRequestLight(Core::System& system, Handle session_handle, uint32_t* args);
 | 
			
		||||
Result SendSyncRequestLight64From32(Core::System& system, Handle session_handle, uint32_t* args);
 | 
			
		||||
Result SendSyncRequestLight64(Core::System& system, Handle session_handle, uint32_t* args);
 | 
			
		||||
 | 
			
		||||
void CallSecureMonitor(Core::System& system, lp64::SecureMonitorArguments* args);
 | 
			
		||||
void CallSecureMonitor64From32(Core::System& system, ilp32::SecureMonitorArguments* args);
 | 
			
		||||
void CallSecureMonitor64(Core::System& system, lp64::SecureMonitorArguments* args);
 | 
			
		||||
 | 
			
		||||
// Defined in svc_light_ipc.cpp.
 | 
			
		||||
void SvcWrap_ReplyAndReceiveLight64From32(Core::System& system);
 | 
			
		||||
void SvcWrap_ReplyAndReceiveLight64(Core::System& system);
 | 
			
		||||
 | 
			
		||||
void SvcWrap_SendSyncRequestLight64From32(Core::System& system);
 | 
			
		||||
void SvcWrap_SendSyncRequestLight64(Core::System& system);
 | 
			
		||||
 | 
			
		||||
// Defined in svc_secure_monitor_call.cpp.
 | 
			
		||||
void SvcWrap_CallSecureMonitor64From32(Core::System& system);
 | 
			
		||||
void SvcWrap_CallSecureMonitor64(Core::System& system);
 | 
			
		||||
 | 
			
		||||
// Perform a supervisor call by index.
 | 
			
		||||
void Call(Core::System& system, u32 imm);
 | 
			
		||||
 | 
			
		||||
} // namespace Kernel::Svc
 | 
			
		||||
"""
 | 
			
		||||
 | 
			
		||||
PROLOGUE_CPP = """
 | 
			
		||||
#include <type_traits>
 | 
			
		||||
 | 
			
		||||
#include "core/arm/arm_interface.h"
 | 
			
		||||
#include "core/core.h"
 | 
			
		||||
#include "core/hle/kernel/k_process.h"
 | 
			
		||||
#include "core/hle/kernel/svc.h"
 | 
			
		||||
 | 
			
		||||
namespace Kernel::Svc {
 | 
			
		||||
 | 
			
		||||
static uint32_t GetReg32(Core::System& system, int n) {
 | 
			
		||||
    return static_cast<uint32_t>(system.CurrentArmInterface().GetReg(n));
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
static void SetReg32(Core::System& system, int n, uint32_t result) {
 | 
			
		||||
    system.CurrentArmInterface().SetReg(n, static_cast<uint64_t>(result));
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
static uint64_t GetReg64(Core::System& system, int n) {
 | 
			
		||||
    return system.CurrentArmInterface().GetReg(n);
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
static void SetReg64(Core::System& system, int n, uint64_t result) {
 | 
			
		||||
    system.CurrentArmInterface().SetReg(n, result);
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
// Like bit_cast, but handles the case when the source and dest
 | 
			
		||||
// are differently-sized.
 | 
			
		||||
template <typename To, typename From>
 | 
			
		||||
    requires(std::is_trivial_v<To> && std::is_trivially_copyable_v<From>)
 | 
			
		||||
static To Convert(const From& from) {
 | 
			
		||||
    To to{};
 | 
			
		||||
 | 
			
		||||
    if constexpr (sizeof(To) >= sizeof(From)) {
 | 
			
		||||
        std::memcpy(&to, &from, sizeof(From));
 | 
			
		||||
    } else {
 | 
			
		||||
        std::memcpy(&to, &from, sizeof(To));
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    return to;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
// clang-format off
 | 
			
		||||
"""
 | 
			
		||||
 | 
			
		||||
EPILOGUE_CPP = """
 | 
			
		||||
// clang-format on
 | 
			
		||||
 | 
			
		||||
void Call(Core::System& system, u32 imm) {
 | 
			
		||||
    auto& kernel = system.Kernel();
 | 
			
		||||
    kernel.EnterSVCProfile();
 | 
			
		||||
 | 
			
		||||
    if (system.CurrentProcess()->Is64BitProcess()) {
 | 
			
		||||
        Call64(system, imm);
 | 
			
		||||
    } else {
 | 
			
		||||
        Call32(system, imm);
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    kernel.ExitSVCProfile();
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
} // namespace Kernel::Svc
 | 
			
		||||
"""
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
def emit_call(bitness, names, suffix):
 | 
			
		||||
    bit_size = REG_SIZES[bitness]*8
 | 
			
		||||
    indent = "    "
 | 
			
		||||
    lines = [
 | 
			
		||||
        f"static void Call{bit_size}(Core::System& system, u32 imm) {{",
 | 
			
		||||
        f"{indent}switch (static_cast<SvcId>(imm)) {{"
 | 
			
		||||
    ]
 | 
			
		||||
 | 
			
		||||
    for _, name in names:
 | 
			
		||||
        lines.append(f"{indent}case SvcId::{name}:")
 | 
			
		||||
        lines.append(f"{indent*2}return SvcWrap_{name}{suffix}(system);")
 | 
			
		||||
 | 
			
		||||
    lines.append(f"{indent}default:")
 | 
			
		||||
    lines.append(
 | 
			
		||||
        f"{indent*2}LOG_CRITICAL(Kernel_SVC, \"Unknown SVC {{:x}}!\", imm);")
 | 
			
		||||
    lines.append(f"{indent*2}break;")
 | 
			
		||||
    lines.append(f"{indent}}}")
 | 
			
		||||
    lines.append("}")
 | 
			
		||||
 | 
			
		||||
    return "\n".join(lines)
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
def build_fn_declaration(return_type, name, arguments):
 | 
			
		||||
    arg_list = ["Core::System& system"]
 | 
			
		||||
    for arg in arguments:
 | 
			
		||||
        type_name = "uintptr_t" if arg.is_address else arg.type_name
 | 
			
		||||
        pointer = "*" if arg.is_output and not arg.is_outptr else ""
 | 
			
		||||
        arg_list.append(f"{type_name}{pointer} {arg.var_name}")
 | 
			
		||||
 | 
			
		||||
    return f"{return_type} {name}({', '.join(arg_list)});"
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
def build_enum_declarations():
 | 
			
		||||
    lines = ["enum class SvcId : u32 {"]
 | 
			
		||||
    indent = "    "
 | 
			
		||||
 | 
			
		||||
    for imm, decl in SVCS:
 | 
			
		||||
        _, name, _ = parse_declaration(decl, BIT_64)
 | 
			
		||||
        lines.append(f"{indent}{name} = {hex(imm)},")
 | 
			
		||||
 | 
			
		||||
    lines.append("};")
 | 
			
		||||
    return "\n".join(lines)
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
def main():
 | 
			
		||||
    arch_fw_declarations = [[], []]
 | 
			
		||||
    svc_fw_declarations = []
 | 
			
		||||
    wrapper_fns = []
 | 
			
		||||
    names = []
 | 
			
		||||
 | 
			
		||||
    for imm, decl in SVCS:
 | 
			
		||||
        return_type, name, arguments = parse_declaration(decl, BIT_64)
 | 
			
		||||
 | 
			
		||||
        if imm not in SKIP_WRAPPERS:
 | 
			
		||||
            svc_fw_declarations.append(
 | 
			
		||||
                build_fn_declaration(return_type, name, arguments))
 | 
			
		||||
 | 
			
		||||
        names.append([imm, name])
 | 
			
		||||
 | 
			
		||||
    for bitness in range(2):
 | 
			
		||||
        byte_size = REG_SIZES[bitness]
 | 
			
		||||
        suffix = SUFFIX_NAMES[bitness]
 | 
			
		||||
 | 
			
		||||
        for imm, decl in SVCS:
 | 
			
		||||
            if imm in SKIP_WRAPPERS:
 | 
			
		||||
                continue
 | 
			
		||||
 | 
			
		||||
            parse_result = parse_declaration(decl, bitness)
 | 
			
		||||
            return_type, name, arguments = parse_result
 | 
			
		||||
 | 
			
		||||
            register_info = get_registers(parse_result, bitness)
 | 
			
		||||
            wrapper_fns.append(
 | 
			
		||||
                emit_wrapper(name, suffix, register_info, arguments, byte_size))
 | 
			
		||||
            arch_fw_declarations[bitness].append(
 | 
			
		||||
                build_fn_declaration(return_type, name + suffix, arguments))
 | 
			
		||||
 | 
			
		||||
    call_32 = emit_call(BIT_32, names, SUFFIX_NAMES[BIT_32])
 | 
			
		||||
    call_64 = emit_call(BIT_64, names, SUFFIX_NAMES[BIT_64])
 | 
			
		||||
    enum_decls = build_enum_declarations()
 | 
			
		||||
 | 
			
		||||
    with open("svc.h", "w") as f:
 | 
			
		||||
        f.write(COPYRIGHT)
 | 
			
		||||
        f.write(PROLOGUE_H)
 | 
			
		||||
        f.write("\n".join(svc_fw_declarations))
 | 
			
		||||
        f.write("\n\n")
 | 
			
		||||
        f.write("\n".join(arch_fw_declarations[BIT_32]))
 | 
			
		||||
        f.write("\n\n")
 | 
			
		||||
        f.write("\n".join(arch_fw_declarations[BIT_64]))
 | 
			
		||||
        f.write("\n\n")
 | 
			
		||||
        f.write(enum_decls)
 | 
			
		||||
        f.write(EPILOGUE_H)
 | 
			
		||||
 | 
			
		||||
    with open("svc.cpp", "w") as f:
 | 
			
		||||
        f.write(COPYRIGHT)
 | 
			
		||||
        f.write(PROLOGUE_CPP)
 | 
			
		||||
        f.write(emit_size_check())
 | 
			
		||||
        f.write("\n\n")
 | 
			
		||||
        f.write("\n\n".join(wrapper_fns))
 | 
			
		||||
        f.write("\n\n")
 | 
			
		||||
        f.write(call_32)
 | 
			
		||||
        f.write("\n\n")
 | 
			
		||||
        f.write(call_64)
 | 
			
		||||
        f.write(EPILOGUE_CPP)
 | 
			
		||||
 | 
			
		||||
    print(f"Done (emitted {len(names)} definitions)")
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
if __name__ == "__main__":
 | 
			
		||||
    main()
 | 
			
		||||
@@ -11,6 +11,7 @@ namespace Kernel {
 | 
			
		||||
 | 
			
		||||
constexpr Result ResultOutOfSessions{ErrorModule::Kernel, 7};
 | 
			
		||||
constexpr Result ResultInvalidArgument{ErrorModule::Kernel, 14};
 | 
			
		||||
constexpr Result ResultNotImplemented{ErrorModule::Kernel, 33};
 | 
			
		||||
constexpr Result ResultNoSynchronizationObject{ErrorModule::Kernel, 57};
 | 
			
		||||
constexpr Result ResultTerminationRequested{ErrorModule::Kernel, 59};
 | 
			
		||||
constexpr Result ResultInvalidSize{ErrorModule::Kernel, 101};
 | 
			
		||||
 
 | 
			
		||||
@@ -168,6 +168,7 @@ enum class BreakReason : u32 {
 | 
			
		||||
 | 
			
		||||
    NotificationOnlyFlag = 0x80000000,
 | 
			
		||||
};
 | 
			
		||||
DECLARE_ENUM_FLAG_OPERATORS(BreakReason);
 | 
			
		||||
 | 
			
		||||
enum class DebugEvent : u32 {
 | 
			
		||||
    CreateProcess = 0,
 | 
			
		||||
@@ -596,6 +597,11 @@ enum class ProcessInfoType : u32 {
 | 
			
		||||
    ProcessState = 0,
 | 
			
		||||
};
 | 
			
		||||
 | 
			
		||||
enum class ProcessActivity : u32 {
 | 
			
		||||
    Runnable,
 | 
			
		||||
    Paused,
 | 
			
		||||
};
 | 
			
		||||
 | 
			
		||||
struct CreateProcessParameter {
 | 
			
		||||
    std::array<char, 12> name;
 | 
			
		||||
    u32 version;
 | 
			
		||||
@@ -611,4 +617,9 @@ static_assert(sizeof(CreateProcessParameter) == 0x30);
 | 
			
		||||
constexpr size_t NumSupervisorCalls = 0xC0;
 | 
			
		||||
using SvcAccessFlagSet = std::bitset<NumSupervisorCalls>;
 | 
			
		||||
 | 
			
		||||
enum class InitialProcessIdRangeInfo : u64 {
 | 
			
		||||
    Minimum = 0,
 | 
			
		||||
    Maximum = 1,
 | 
			
		||||
};
 | 
			
		||||
 | 
			
		||||
} // namespace Kernel::Svc
 | 
			
		||||
 
 | 
			
		||||
@@ -1,733 +0,0 @@
 | 
			
		||||
// SPDX-FileCopyrightText: Copyright 2018 yuzu Emulator Project
 | 
			
		||||
// SPDX-License-Identifier: GPL-2.0-or-later
 | 
			
		||||
 | 
			
		||||
#pragma once
 | 
			
		||||
 | 
			
		||||
#include "common/common_types.h"
 | 
			
		||||
#include "core/arm/arm_interface.h"
 | 
			
		||||
#include "core/core.h"
 | 
			
		||||
#include "core/hle/kernel/svc_types.h"
 | 
			
		||||
#include "core/hle/result.h"
 | 
			
		||||
#include "core/memory.h"
 | 
			
		||||
 | 
			
		||||
namespace Kernel {
 | 
			
		||||
 | 
			
		||||
static inline u64 Param(const Core::System& system, int n) {
 | 
			
		||||
    return system.CurrentArmInterface().GetReg(n);
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
static inline u32 Param32(const Core::System& system, int n) {
 | 
			
		||||
    return static_cast<u32>(system.CurrentArmInterface().GetReg(n));
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
/**
 | 
			
		||||
 * HLE a function return from the current ARM userland process
 | 
			
		||||
 * @param system System context
 | 
			
		||||
 * @param result Result to return
 | 
			
		||||
 */
 | 
			
		||||
static inline void FuncReturn(Core::System& system, u64 result) {
 | 
			
		||||
    system.CurrentArmInterface().SetReg(0, result);
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
static inline void FuncReturn32(Core::System& system, u32 result) {
 | 
			
		||||
    system.CurrentArmInterface().SetReg(0, (u64)result);
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
////////////////////////////////////////////////////////////////////////////////////////////////////
 | 
			
		||||
// Function wrappers that return type Result
 | 
			
		||||
 | 
			
		||||
template <Result func(Core::System&, u64)>
 | 
			
		||||
void SvcWrap64(Core::System& system) {
 | 
			
		||||
    FuncReturn(system, func(system, Param(system, 0)).raw);
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
template <Result func(Core::System&, u64, u64)>
 | 
			
		||||
void SvcWrap64(Core::System& system) {
 | 
			
		||||
    FuncReturn(system, func(system, Param(system, 0), Param(system, 1)).raw);
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
template <Result func(Core::System&, u32)>
 | 
			
		||||
void SvcWrap64(Core::System& system) {
 | 
			
		||||
    FuncReturn(system, func(system, static_cast<u32>(Param(system, 0))).raw);
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
template <Result func(Core::System&, u32, u32)>
 | 
			
		||||
void SvcWrap64(Core::System& system) {
 | 
			
		||||
    FuncReturn(
 | 
			
		||||
        system,
 | 
			
		||||
        func(system, static_cast<u32>(Param(system, 0)), static_cast<u32>(Param(system, 1))).raw);
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
// Used by SetThreadActivity
 | 
			
		||||
template <Result func(Core::System&, Handle, Svc::ThreadActivity)>
 | 
			
		||||
void SvcWrap64(Core::System& system) {
 | 
			
		||||
    FuncReturn(system, func(system, static_cast<u32>(Param(system, 0)),
 | 
			
		||||
                            static_cast<Svc::ThreadActivity>(Param(system, 1)))
 | 
			
		||||
                           .raw);
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
template <Result func(Core::System&, u32, u64, u64, u64)>
 | 
			
		||||
void SvcWrap64(Core::System& system) {
 | 
			
		||||
    FuncReturn(system, func(system, static_cast<u32>(Param(system, 0)), Param(system, 1),
 | 
			
		||||
                            Param(system, 2), Param(system, 3))
 | 
			
		||||
                           .raw);
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
// Used by MapProcessMemory and UnmapProcessMemory
 | 
			
		||||
template <Result func(Core::System&, u64, u32, u64, u64)>
 | 
			
		||||
void SvcWrap64(Core::System& system) {
 | 
			
		||||
    FuncReturn(system, func(system, Param(system, 0), static_cast<u32>(Param(system, 1)),
 | 
			
		||||
                            Param(system, 2), Param(system, 3))
 | 
			
		||||
                           .raw);
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
// Used by ControlCodeMemory
 | 
			
		||||
template <Result func(Core::System&, Handle, u32, VAddr, size_t, Svc::MemoryPermission)>
 | 
			
		||||
void SvcWrap64(Core::System& system) {
 | 
			
		||||
    FuncReturn(system, func(system, static_cast<Handle>(Param(system, 0)),
 | 
			
		||||
                            static_cast<u32>(Param(system, 1)), Param(system, 2), Param(system, 3),
 | 
			
		||||
                            static_cast<Svc::MemoryPermission>(Param(system, 4)))
 | 
			
		||||
                           .raw);
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
template <Result func(Core::System&, u32*)>
 | 
			
		||||
void SvcWrap64(Core::System& system) {
 | 
			
		||||
    u32 param = 0;
 | 
			
		||||
    const u32 retval = func(system, ¶m).raw;
 | 
			
		||||
    system.CurrentArmInterface().SetReg(1, param);
 | 
			
		||||
    FuncReturn(system, retval);
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
template <Result func(Core::System&, u32*, u32)>
 | 
			
		||||
void SvcWrap64(Core::System& system) {
 | 
			
		||||
    u32 param_1 = 0;
 | 
			
		||||
    const u32 retval = func(system, ¶m_1, static_cast<u32>(Param(system, 1))).raw;
 | 
			
		||||
    system.CurrentArmInterface().SetReg(1, param_1);
 | 
			
		||||
    FuncReturn(system, retval);
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
template <Result func(Core::System&, u32*, u32*)>
 | 
			
		||||
void SvcWrap64(Core::System& system) {
 | 
			
		||||
    u32 param_1 = 0;
 | 
			
		||||
    u32 param_2 = 0;
 | 
			
		||||
    const u32 retval = func(system, ¶m_1, ¶m_2).raw;
 | 
			
		||||
 | 
			
		||||
    auto& arm_interface = system.CurrentArmInterface();
 | 
			
		||||
    arm_interface.SetReg(1, param_1);
 | 
			
		||||
    arm_interface.SetReg(2, param_2);
 | 
			
		||||
 | 
			
		||||
    FuncReturn(system, retval);
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
template <Result func(Core::System&, u32*, u64)>
 | 
			
		||||
void SvcWrap64(Core::System& system) {
 | 
			
		||||
    u32 param_1 = 0;
 | 
			
		||||
    const u32 retval = func(system, ¶m_1, Param(system, 1)).raw;
 | 
			
		||||
    system.CurrentArmInterface().SetReg(1, param_1);
 | 
			
		||||
    FuncReturn(system, retval);
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
template <Result func(Core::System&, u32*, u64, u32)>
 | 
			
		||||
void SvcWrap64(Core::System& system) {
 | 
			
		||||
    u32 param_1 = 0;
 | 
			
		||||
    const u32 retval =
 | 
			
		||||
        func(system, ¶m_1, Param(system, 1), static_cast<u32>(Param(system, 2))).raw;
 | 
			
		||||
 | 
			
		||||
    system.CurrentArmInterface().SetReg(1, param_1);
 | 
			
		||||
    FuncReturn(system, retval);
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
template <Result func(Core::System&, u64*, u32)>
 | 
			
		||||
void SvcWrap64(Core::System& system) {
 | 
			
		||||
    u64 param_1 = 0;
 | 
			
		||||
    const u32 retval = func(system, ¶m_1, static_cast<u32>(Param(system, 1))).raw;
 | 
			
		||||
 | 
			
		||||
    system.CurrentArmInterface().SetReg(1, param_1);
 | 
			
		||||
    FuncReturn(system, retval);
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
template <Result func(Core::System&, u64, u32)>
 | 
			
		||||
void SvcWrap64(Core::System& system) {
 | 
			
		||||
    FuncReturn(system, func(system, Param(system, 0), static_cast<u32>(Param(system, 1))).raw);
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
template <Result func(Core::System&, u64*, u64)>
 | 
			
		||||
void SvcWrap64(Core::System& system) {
 | 
			
		||||
    u64 param_1 = 0;
 | 
			
		||||
    const u32 retval = func(system, ¶m_1, Param(system, 1)).raw;
 | 
			
		||||
 | 
			
		||||
    system.CurrentArmInterface().SetReg(1, param_1);
 | 
			
		||||
    FuncReturn(system, retval);
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
template <Result func(Core::System&, u64*, u32, u32)>
 | 
			
		||||
void SvcWrap64(Core::System& system) {
 | 
			
		||||
    u64 param_1 = 0;
 | 
			
		||||
    const u32 retval = func(system, ¶m_1, static_cast<u32>(Param(system, 1)),
 | 
			
		||||
                            static_cast<u32>(Param(system, 2)))
 | 
			
		||||
                           .raw;
 | 
			
		||||
 | 
			
		||||
    system.CurrentArmInterface().SetReg(1, param_1);
 | 
			
		||||
    FuncReturn(system, retval);
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
// Used by GetResourceLimitLimitValue.
 | 
			
		||||
template <Result func(Core::System&, u64*, Handle, Svc::LimitableResource)>
 | 
			
		||||
void SvcWrap64(Core::System& system) {
 | 
			
		||||
    u64 param_1 = 0;
 | 
			
		||||
    const u32 retval = func(system, ¶m_1, static_cast<Handle>(Param(system, 1)),
 | 
			
		||||
                            static_cast<Svc::LimitableResource>(Param(system, 2)))
 | 
			
		||||
                           .raw;
 | 
			
		||||
 | 
			
		||||
    system.CurrentArmInterface().SetReg(1, param_1);
 | 
			
		||||
    FuncReturn(system, retval);
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
template <Result func(Core::System&, u32, u64)>
 | 
			
		||||
void SvcWrap64(Core::System& system) {
 | 
			
		||||
    FuncReturn(system, func(system, static_cast<u32>(Param(system, 0)), Param(system, 1)).raw);
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
// Used by SetResourceLimitLimitValue
 | 
			
		||||
template <Result func(Core::System&, Handle, Svc::LimitableResource, u64)>
 | 
			
		||||
void SvcWrap64(Core::System& system) {
 | 
			
		||||
    FuncReturn(system, func(system, static_cast<Handle>(Param(system, 0)),
 | 
			
		||||
                            static_cast<Svc::LimitableResource>(Param(system, 1)), Param(system, 2))
 | 
			
		||||
                           .raw);
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
// Used by SetThreadCoreMask
 | 
			
		||||
template <Result func(Core::System&, Handle, s32, u64)>
 | 
			
		||||
void SvcWrap64(Core::System& system) {
 | 
			
		||||
    FuncReturn(system, func(system, static_cast<u32>(Param(system, 0)),
 | 
			
		||||
                            static_cast<s32>(Param(system, 1)), Param(system, 2))
 | 
			
		||||
                           .raw);
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
// Used by GetThreadCoreMask
 | 
			
		||||
template <Result func(Core::System&, Handle, s32*, u64*)>
 | 
			
		||||
void SvcWrap64(Core::System& system) {
 | 
			
		||||
    s32 param_1 = 0;
 | 
			
		||||
    u64 param_2 = 0;
 | 
			
		||||
    const Result retval = func(system, static_cast<u32>(Param(system, 2)), ¶m_1, ¶m_2);
 | 
			
		||||
 | 
			
		||||
    system.CurrentArmInterface().SetReg(1, param_1);
 | 
			
		||||
    system.CurrentArmInterface().SetReg(2, param_2);
 | 
			
		||||
    FuncReturn(system, retval.raw);
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
template <Result func(Core::System&, u64, u64, u32, u32)>
 | 
			
		||||
void SvcWrap64(Core::System& system) {
 | 
			
		||||
    FuncReturn(system, func(system, Param(system, 0), Param(system, 1),
 | 
			
		||||
                            static_cast<u32>(Param(system, 2)), static_cast<u32>(Param(system, 3)))
 | 
			
		||||
                           .raw);
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
template <Result func(Core::System&, u64, u64, u32, u64)>
 | 
			
		||||
void SvcWrap64(Core::System& system) {
 | 
			
		||||
    FuncReturn(system, func(system, Param(system, 0), Param(system, 1),
 | 
			
		||||
                            static_cast<u32>(Param(system, 2)), Param(system, 3))
 | 
			
		||||
                           .raw);
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
template <Result func(Core::System&, u32, u64, u32)>
 | 
			
		||||
void SvcWrap64(Core::System& system) {
 | 
			
		||||
    FuncReturn(system, func(system, static_cast<u32>(Param(system, 0)), Param(system, 1),
 | 
			
		||||
                            static_cast<u32>(Param(system, 2)))
 | 
			
		||||
                           .raw);
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
template <Result func(Core::System&, u64, u64, u64)>
 | 
			
		||||
void SvcWrap64(Core::System& system) {
 | 
			
		||||
    FuncReturn(system, func(system, Param(system, 0), Param(system, 1), Param(system, 2)).raw);
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
template <Result func(Core::System&, u64, u64, u32)>
 | 
			
		||||
void SvcWrap64(Core::System& system) {
 | 
			
		||||
    FuncReturn(
 | 
			
		||||
        system,
 | 
			
		||||
        func(system, Param(system, 0), Param(system, 1), static_cast<u32>(Param(system, 2))).raw);
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
// Used by SetMemoryPermission
 | 
			
		||||
template <Result func(Core::System&, u64, u64, Svc::MemoryPermission)>
 | 
			
		||||
void SvcWrap64(Core::System& system) {
 | 
			
		||||
    FuncReturn(system, func(system, Param(system, 0), Param(system, 1),
 | 
			
		||||
                            static_cast<Svc::MemoryPermission>(Param(system, 2)))
 | 
			
		||||
                           .raw);
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
// Used by MapSharedMemory
 | 
			
		||||
template <Result func(Core::System&, Handle, u64, u64, Svc::MemoryPermission)>
 | 
			
		||||
void SvcWrap64(Core::System& system) {
 | 
			
		||||
    FuncReturn(system, func(system, static_cast<Handle>(Param(system, 0)), Param(system, 1),
 | 
			
		||||
                            Param(system, 2), static_cast<Svc::MemoryPermission>(Param(system, 3)))
 | 
			
		||||
                           .raw);
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
template <Result func(Core::System&, u32, u64, u64)>
 | 
			
		||||
void SvcWrap64(Core::System& system) {
 | 
			
		||||
    FuncReturn(
 | 
			
		||||
        system,
 | 
			
		||||
        func(system, static_cast<u32>(Param(system, 0)), Param(system, 1), Param(system, 2)).raw);
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
// Used by WaitSynchronization
 | 
			
		||||
template <Result func(Core::System&, s32*, u64, s32, s64)>
 | 
			
		||||
void SvcWrap64(Core::System& system) {
 | 
			
		||||
    s32 param_1 = 0;
 | 
			
		||||
    const u32 retval = func(system, ¶m_1, Param(system, 1), static_cast<s32>(Param(system, 2)),
 | 
			
		||||
                            static_cast<s64>(Param(system, 3)))
 | 
			
		||||
                           .raw;
 | 
			
		||||
 | 
			
		||||
    system.CurrentArmInterface().SetReg(1, param_1);
 | 
			
		||||
    FuncReturn(system, retval);
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
template <Result func(Core::System&, u64, u64, u32, s64)>
 | 
			
		||||
void SvcWrap64(Core::System& system) {
 | 
			
		||||
    FuncReturn(system, func(system, Param(system, 0), Param(system, 1),
 | 
			
		||||
                            static_cast<u32>(Param(system, 2)), static_cast<s64>(Param(system, 3)))
 | 
			
		||||
                           .raw);
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
// Used by GetInfo
 | 
			
		||||
template <Result func(Core::System&, u64*, u64, Handle, u64)>
 | 
			
		||||
void SvcWrap64(Core::System& system) {
 | 
			
		||||
    u64 param_1 = 0;
 | 
			
		||||
    const u32 retval = func(system, ¶m_1, Param(system, 1),
 | 
			
		||||
                            static_cast<Handle>(Param(system, 2)), Param(system, 3))
 | 
			
		||||
                           .raw;
 | 
			
		||||
 | 
			
		||||
    system.CurrentArmInterface().SetReg(1, param_1);
 | 
			
		||||
    FuncReturn(system, retval);
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
template <Result func(Core::System&, u32*, u64, u64, u64, u32, s32)>
 | 
			
		||||
void SvcWrap64(Core::System& system) {
 | 
			
		||||
    u32 param_1 = 0;
 | 
			
		||||
    const u32 retval = func(system, ¶m_1, Param(system, 1), Param(system, 2), Param(system, 3),
 | 
			
		||||
                            static_cast<u32>(Param(system, 4)), static_cast<s32>(Param(system, 5)))
 | 
			
		||||
                           .raw;
 | 
			
		||||
 | 
			
		||||
    system.CurrentArmInterface().SetReg(1, param_1);
 | 
			
		||||
    FuncReturn(system, retval);
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
// Used by CreateTransferMemory
 | 
			
		||||
template <Result func(Core::System&, Handle*, u64, u64, Svc::MemoryPermission)>
 | 
			
		||||
void SvcWrap64(Core::System& system) {
 | 
			
		||||
    u32 param_1 = 0;
 | 
			
		||||
    const u32 retval = func(system, ¶m_1, Param(system, 1), Param(system, 2),
 | 
			
		||||
                            static_cast<Svc::MemoryPermission>(Param(system, 3)))
 | 
			
		||||
                           .raw;
 | 
			
		||||
 | 
			
		||||
    system.CurrentArmInterface().SetReg(1, param_1);
 | 
			
		||||
    FuncReturn(system, retval);
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
// Used by CreateCodeMemory
 | 
			
		||||
template <Result func(Core::System&, Handle*, VAddr, size_t)>
 | 
			
		||||
void SvcWrap64(Core::System& system) {
 | 
			
		||||
    u32 param_1 = 0;
 | 
			
		||||
    const u32 retval = func(system, ¶m_1, Param(system, 1), Param(system, 2)).raw;
 | 
			
		||||
 | 
			
		||||
    system.CurrentArmInterface().SetReg(1, param_1);
 | 
			
		||||
    FuncReturn(system, retval);
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
template <Result func(Core::System&, Handle*, u64, u32, u32)>
 | 
			
		||||
void SvcWrap64(Core::System& system) {
 | 
			
		||||
    u32 param_1 = 0;
 | 
			
		||||
    const u32 retval = func(system, ¶m_1, Param(system, 1), static_cast<u32>(Param(system, 2)),
 | 
			
		||||
                            static_cast<u32>(Param(system, 3)))
 | 
			
		||||
                           .raw;
 | 
			
		||||
 | 
			
		||||
    system.CurrentArmInterface().SetReg(1, param_1);
 | 
			
		||||
    FuncReturn(system, retval);
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
// Used by CreateSession
 | 
			
		||||
template <Result func(Core::System&, Handle*, Handle*, u32, u64)>
 | 
			
		||||
void SvcWrap64(Core::System& system) {
 | 
			
		||||
    Handle param_1 = 0;
 | 
			
		||||
    Handle param_2 = 0;
 | 
			
		||||
    const u32 retval = func(system, ¶m_1, ¶m_2, static_cast<u32>(Param(system, 2)),
 | 
			
		||||
                            static_cast<u32>(Param(system, 3)))
 | 
			
		||||
                           .raw;
 | 
			
		||||
 | 
			
		||||
    system.CurrentArmInterface().SetReg(1, param_1);
 | 
			
		||||
    system.CurrentArmInterface().SetReg(2, param_2);
 | 
			
		||||
    FuncReturn(system, retval);
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
// Used by ReplyAndReceive
 | 
			
		||||
template <Result func(Core::System&, s32*, Handle*, s32, Handle, s64)>
 | 
			
		||||
void SvcWrap64(Core::System& system) {
 | 
			
		||||
    s32 param_1 = 0;
 | 
			
		||||
    s32 num_handles = static_cast<s32>(Param(system, 2));
 | 
			
		||||
 | 
			
		||||
    std::vector<Handle> handles(num_handles);
 | 
			
		||||
    system.Memory().ReadBlock(Param(system, 1), handles.data(), num_handles * sizeof(Handle));
 | 
			
		||||
 | 
			
		||||
    const u32 retval = func(system, ¶m_1, handles.data(), num_handles,
 | 
			
		||||
                            static_cast<s32>(Param(system, 3)), static_cast<s64>(Param(system, 4)))
 | 
			
		||||
                           .raw;
 | 
			
		||||
 | 
			
		||||
    system.CurrentArmInterface().SetReg(1, param_1);
 | 
			
		||||
    FuncReturn(system, retval);
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
// Used by WaitForAddress
 | 
			
		||||
template <Result func(Core::System&, u64, Svc::ArbitrationType, s32, s64)>
 | 
			
		||||
void SvcWrap64(Core::System& system) {
 | 
			
		||||
    FuncReturn(system,
 | 
			
		||||
               func(system, Param(system, 0), static_cast<Svc::ArbitrationType>(Param(system, 1)),
 | 
			
		||||
                    static_cast<s32>(Param(system, 2)), static_cast<s64>(Param(system, 3)))
 | 
			
		||||
                   .raw);
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
// Used by SignalToAddress
 | 
			
		||||
template <Result func(Core::System&, u64, Svc::SignalType, s32, s32)>
 | 
			
		||||
void SvcWrap64(Core::System& system) {
 | 
			
		||||
    FuncReturn(system,
 | 
			
		||||
               func(system, Param(system, 0), static_cast<Svc::SignalType>(Param(system, 1)),
 | 
			
		||||
                    static_cast<s32>(Param(system, 2)), static_cast<s32>(Param(system, 3)))
 | 
			
		||||
                   .raw);
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
////////////////////////////////////////////////////////////////////////////////////////////////////
 | 
			
		||||
// Function wrappers that return type u32
 | 
			
		||||
 | 
			
		||||
template <u32 func(Core::System&)>
 | 
			
		||||
void SvcWrap64(Core::System& system) {
 | 
			
		||||
    FuncReturn(system, func(system));
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
////////////////////////////////////////////////////////////////////////////////////////////////////
 | 
			
		||||
// Function wrappers that return type u64
 | 
			
		||||
 | 
			
		||||
template <u64 func(Core::System&)>
 | 
			
		||||
void SvcWrap64(Core::System& system) {
 | 
			
		||||
    FuncReturn(system, func(system));
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
////////////////////////////////////////////////////////////////////////////////////////////////////
 | 
			
		||||
/// Function wrappers that return type void
 | 
			
		||||
 | 
			
		||||
template <void func(Core::System&)>
 | 
			
		||||
void SvcWrap64(Core::System& system) {
 | 
			
		||||
    func(system);
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
template <void func(Core::System&, u32)>
 | 
			
		||||
void SvcWrap64(Core::System& system) {
 | 
			
		||||
    func(system, static_cast<u32>(Param(system, 0)));
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
template <void func(Core::System&, u32, u64, u64, u64)>
 | 
			
		||||
void SvcWrap64(Core::System& system) {
 | 
			
		||||
    func(system, static_cast<u32>(Param(system, 0)), Param(system, 1), Param(system, 2),
 | 
			
		||||
         Param(system, 3));
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
template <void func(Core::System&, s64)>
 | 
			
		||||
void SvcWrap64(Core::System& system) {
 | 
			
		||||
    func(system, static_cast<s64>(Param(system, 0)));
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
template <void func(Core::System&, u64, s32)>
 | 
			
		||||
void SvcWrap64(Core::System& system) {
 | 
			
		||||
    func(system, Param(system, 0), static_cast<s32>(Param(system, 1)));
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
template <void func(Core::System&, u64, u64)>
 | 
			
		||||
void SvcWrap64(Core::System& system) {
 | 
			
		||||
    func(system, Param(system, 0), Param(system, 1));
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
template <void func(Core::System&, u64, u64, u64)>
 | 
			
		||||
void SvcWrap64(Core::System& system) {
 | 
			
		||||
    func(system, Param(system, 0), Param(system, 1), Param(system, 2));
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
template <void func(Core::System&, u32, u64, u64)>
 | 
			
		||||
void SvcWrap64(Core::System& system) {
 | 
			
		||||
    func(system, static_cast<u32>(Param(system, 0)), Param(system, 1), Param(system, 2));
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
// Used by QueryMemory32, ArbitrateLock32
 | 
			
		||||
template <Result func(Core::System&, u32, u32, u32)>
 | 
			
		||||
void SvcWrap32(Core::System& system) {
 | 
			
		||||
    FuncReturn32(system,
 | 
			
		||||
                 func(system, Param32(system, 0), Param32(system, 1), Param32(system, 2)).raw);
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
// Used by Break32
 | 
			
		||||
template <void func(Core::System&, u32, u32, u32)>
 | 
			
		||||
void SvcWrap32(Core::System& system) {
 | 
			
		||||
    func(system, Param32(system, 0), Param32(system, 1), Param32(system, 2));
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
// Used by ExitProcess32, ExitThread32
 | 
			
		||||
template <void func(Core::System&)>
 | 
			
		||||
void SvcWrap32(Core::System& system) {
 | 
			
		||||
    func(system);
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
// Used by GetCurrentProcessorNumber32
 | 
			
		||||
template <u32 func(Core::System&)>
 | 
			
		||||
void SvcWrap32(Core::System& system) {
 | 
			
		||||
    FuncReturn32(system, func(system));
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
// Used by SleepThread32
 | 
			
		||||
template <void func(Core::System&, u32, u32)>
 | 
			
		||||
void SvcWrap32(Core::System& system) {
 | 
			
		||||
    func(system, Param32(system, 0), Param32(system, 1));
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
// Used by CreateThread32
 | 
			
		||||
template <Result func(Core::System&, Handle*, u32, u32, u32, u32, s32)>
 | 
			
		||||
void SvcWrap32(Core::System& system) {
 | 
			
		||||
    Handle param_1 = 0;
 | 
			
		||||
 | 
			
		||||
    const u32 retval = func(system, ¶m_1, Param32(system, 0), Param32(system, 1),
 | 
			
		||||
                            Param32(system, 2), Param32(system, 3), Param32(system, 4))
 | 
			
		||||
                           .raw;
 | 
			
		||||
 | 
			
		||||
    system.CurrentArmInterface().SetReg(1, param_1);
 | 
			
		||||
    FuncReturn(system, retval);
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
// Used by GetInfo32
 | 
			
		||||
template <Result func(Core::System&, u32*, u32*, u32, u32, u32, u32)>
 | 
			
		||||
void SvcWrap32(Core::System& system) {
 | 
			
		||||
    u32 param_1 = 0;
 | 
			
		||||
    u32 param_2 = 0;
 | 
			
		||||
 | 
			
		||||
    const u32 retval = func(system, ¶m_1, ¶m_2, Param32(system, 0), Param32(system, 1),
 | 
			
		||||
                            Param32(system, 2), Param32(system, 3))
 | 
			
		||||
                           .raw;
 | 
			
		||||
 | 
			
		||||
    system.CurrentArmInterface().SetReg(1, param_1);
 | 
			
		||||
    system.CurrentArmInterface().SetReg(2, param_2);
 | 
			
		||||
    FuncReturn(system, retval);
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
// Used by GetThreadPriority32, ConnectToNamedPort32
 | 
			
		||||
template <Result func(Core::System&, u32*, u32)>
 | 
			
		||||
void SvcWrap32(Core::System& system) {
 | 
			
		||||
    u32 param_1 = 0;
 | 
			
		||||
    const u32 retval = func(system, ¶m_1, Param32(system, 1)).raw;
 | 
			
		||||
    system.CurrentArmInterface().SetReg(1, param_1);
 | 
			
		||||
    FuncReturn(system, retval);
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
// Used by GetThreadId32
 | 
			
		||||
template <Result func(Core::System&, u32*, u32*, u32)>
 | 
			
		||||
void SvcWrap32(Core::System& system) {
 | 
			
		||||
    u32 param_1 = 0;
 | 
			
		||||
    u32 param_2 = 0;
 | 
			
		||||
 | 
			
		||||
    const u32 retval = func(system, ¶m_1, ¶m_2, Param32(system, 1)).raw;
 | 
			
		||||
    system.CurrentArmInterface().SetReg(1, param_1);
 | 
			
		||||
    system.CurrentArmInterface().SetReg(2, param_2);
 | 
			
		||||
    FuncReturn(system, retval);
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
// Used by GetSystemTick32
 | 
			
		||||
template <void func(Core::System&, u32*, u32*)>
 | 
			
		||||
void SvcWrap32(Core::System& system) {
 | 
			
		||||
    u32 param_1 = 0;
 | 
			
		||||
    u32 param_2 = 0;
 | 
			
		||||
 | 
			
		||||
    func(system, ¶m_1, ¶m_2);
 | 
			
		||||
    system.CurrentArmInterface().SetReg(0, param_1);
 | 
			
		||||
    system.CurrentArmInterface().SetReg(1, param_2);
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
// Used by CreateEvent32
 | 
			
		||||
template <Result func(Core::System&, Handle*, Handle*)>
 | 
			
		||||
void SvcWrap32(Core::System& system) {
 | 
			
		||||
    Handle param_1 = 0;
 | 
			
		||||
    Handle param_2 = 0;
 | 
			
		||||
 | 
			
		||||
    const u32 retval = func(system, ¶m_1, ¶m_2).raw;
 | 
			
		||||
    system.CurrentArmInterface().SetReg(1, param_1);
 | 
			
		||||
    system.CurrentArmInterface().SetReg(2, param_2);
 | 
			
		||||
    FuncReturn(system, retval);
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
// Used by GetThreadId32
 | 
			
		||||
template <Result func(Core::System&, Handle, u32*, u32*, u32*)>
 | 
			
		||||
void SvcWrap32(Core::System& system) {
 | 
			
		||||
    u32 param_1 = 0;
 | 
			
		||||
    u32 param_2 = 0;
 | 
			
		||||
    u32 param_3 = 0;
 | 
			
		||||
 | 
			
		||||
    const u32 retval = func(system, Param32(system, 2), ¶m_1, ¶m_2, ¶m_3).raw;
 | 
			
		||||
    system.CurrentArmInterface().SetReg(1, param_1);
 | 
			
		||||
    system.CurrentArmInterface().SetReg(2, param_2);
 | 
			
		||||
    system.CurrentArmInterface().SetReg(3, param_3);
 | 
			
		||||
    FuncReturn(system, retval);
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
// Used by GetThreadCoreMask32
 | 
			
		||||
template <Result func(Core::System&, Handle, s32*, u32*, u32*)>
 | 
			
		||||
void SvcWrap32(Core::System& system) {
 | 
			
		||||
    s32 param_1 = 0;
 | 
			
		||||
    u32 param_2 = 0;
 | 
			
		||||
    u32 param_3 = 0;
 | 
			
		||||
 | 
			
		||||
    const u32 retval = func(system, Param32(system, 2), ¶m_1, ¶m_2, ¶m_3).raw;
 | 
			
		||||
    system.CurrentArmInterface().SetReg(1, param_1);
 | 
			
		||||
    system.CurrentArmInterface().SetReg(2, param_2);
 | 
			
		||||
    system.CurrentArmInterface().SetReg(3, param_3);
 | 
			
		||||
    FuncReturn(system, retval);
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
// Used by SignalProcessWideKey32
 | 
			
		||||
template <void func(Core::System&, u32, s32)>
 | 
			
		||||
void SvcWrap32(Core::System& system) {
 | 
			
		||||
    func(system, static_cast<u32>(Param(system, 0)), static_cast<s32>(Param(system, 1)));
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
// Used by SetThreadActivity32
 | 
			
		||||
template <Result func(Core::System&, Handle, Svc::ThreadActivity)>
 | 
			
		||||
void SvcWrap32(Core::System& system) {
 | 
			
		||||
    const u32 retval = func(system, static_cast<Handle>(Param(system, 0)),
 | 
			
		||||
                            static_cast<Svc::ThreadActivity>(Param(system, 1)))
 | 
			
		||||
                           .raw;
 | 
			
		||||
    FuncReturn(system, retval);
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
// Used by SetThreadPriority32
 | 
			
		||||
template <Result func(Core::System&, Handle, u32)>
 | 
			
		||||
void SvcWrap32(Core::System& system) {
 | 
			
		||||
    const u32 retval =
 | 
			
		||||
        func(system, static_cast<Handle>(Param(system, 0)), static_cast<u32>(Param(system, 1))).raw;
 | 
			
		||||
    FuncReturn(system, retval);
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
// Used by SetMemoryAttribute32
 | 
			
		||||
template <Result func(Core::System&, Handle, u32, u32, u32)>
 | 
			
		||||
void SvcWrap32(Core::System& system) {
 | 
			
		||||
    const u32 retval =
 | 
			
		||||
        func(system, static_cast<Handle>(Param(system, 0)), static_cast<u32>(Param(system, 1)),
 | 
			
		||||
             static_cast<u32>(Param(system, 2)), static_cast<u32>(Param(system, 3)))
 | 
			
		||||
            .raw;
 | 
			
		||||
    FuncReturn(system, retval);
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
// Used by MapSharedMemory32
 | 
			
		||||
template <Result func(Core::System&, Handle, u32, u32, Svc::MemoryPermission)>
 | 
			
		||||
void SvcWrap32(Core::System& system) {
 | 
			
		||||
    const u32 retval = func(system, static_cast<Handle>(Param(system, 0)),
 | 
			
		||||
                            static_cast<u32>(Param(system, 1)), static_cast<u32>(Param(system, 2)),
 | 
			
		||||
                            static_cast<Svc::MemoryPermission>(Param(system, 3)))
 | 
			
		||||
                           .raw;
 | 
			
		||||
    FuncReturn(system, retval);
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
// Used by SetThreadCoreMask32
 | 
			
		||||
template <Result func(Core::System&, Handle, s32, u32, u32)>
 | 
			
		||||
void SvcWrap32(Core::System& system) {
 | 
			
		||||
    const u32 retval =
 | 
			
		||||
        func(system, static_cast<Handle>(Param(system, 0)), static_cast<s32>(Param(system, 1)),
 | 
			
		||||
             static_cast<u32>(Param(system, 2)), static_cast<u32>(Param(system, 3)))
 | 
			
		||||
            .raw;
 | 
			
		||||
    FuncReturn(system, retval);
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
// Used by WaitProcessWideKeyAtomic32
 | 
			
		||||
template <Result func(Core::System&, u32, u32, Handle, u32, u32)>
 | 
			
		||||
void SvcWrap32(Core::System& system) {
 | 
			
		||||
    const u32 retval =
 | 
			
		||||
        func(system, static_cast<u32>(Param(system, 0)), static_cast<u32>(Param(system, 1)),
 | 
			
		||||
             static_cast<Handle>(Param(system, 2)), static_cast<u32>(Param(system, 3)),
 | 
			
		||||
             static_cast<u32>(Param(system, 4)))
 | 
			
		||||
            .raw;
 | 
			
		||||
    FuncReturn(system, retval);
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
// Used by WaitForAddress32
 | 
			
		||||
template <Result func(Core::System&, u32, Svc::ArbitrationType, s32, u32, u32)>
 | 
			
		||||
void SvcWrap32(Core::System& system) {
 | 
			
		||||
    const u32 retval = func(system, static_cast<u32>(Param(system, 0)),
 | 
			
		||||
                            static_cast<Svc::ArbitrationType>(Param(system, 1)),
 | 
			
		||||
                            static_cast<s32>(Param(system, 2)), static_cast<u32>(Param(system, 3)),
 | 
			
		||||
                            static_cast<u32>(Param(system, 4)))
 | 
			
		||||
                           .raw;
 | 
			
		||||
    FuncReturn(system, retval);
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
// Used by SignalToAddress32
 | 
			
		||||
template <Result func(Core::System&, u32, Svc::SignalType, s32, s32)>
 | 
			
		||||
void SvcWrap32(Core::System& system) {
 | 
			
		||||
    const u32 retval = func(system, static_cast<u32>(Param(system, 0)),
 | 
			
		||||
                            static_cast<Svc::SignalType>(Param(system, 1)),
 | 
			
		||||
                            static_cast<s32>(Param(system, 2)), static_cast<s32>(Param(system, 3)))
 | 
			
		||||
                           .raw;
 | 
			
		||||
    FuncReturn(system, retval);
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
// Used by SendSyncRequest32, ArbitrateUnlock32
 | 
			
		||||
template <Result func(Core::System&, u32)>
 | 
			
		||||
void SvcWrap32(Core::System& system) {
 | 
			
		||||
    FuncReturn(system, func(system, static_cast<u32>(Param(system, 0))).raw);
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
// Used by CreateTransferMemory32
 | 
			
		||||
template <Result func(Core::System&, Handle*, u32, u32, Svc::MemoryPermission)>
 | 
			
		||||
void SvcWrap32(Core::System& system) {
 | 
			
		||||
    Handle handle = 0;
 | 
			
		||||
    const u32 retval = func(system, &handle, Param32(system, 1), Param32(system, 2),
 | 
			
		||||
                            static_cast<Svc::MemoryPermission>(Param32(system, 3)))
 | 
			
		||||
                           .raw;
 | 
			
		||||
    system.CurrentArmInterface().SetReg(1, handle);
 | 
			
		||||
    FuncReturn(system, retval);
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
// Used by WaitSynchronization32
 | 
			
		||||
template <Result func(Core::System&, u32, u32, s32, u32, s32*)>
 | 
			
		||||
void SvcWrap32(Core::System& system) {
 | 
			
		||||
    s32 param_1 = 0;
 | 
			
		||||
    const u32 retval = func(system, Param32(system, 0), Param32(system, 1), Param32(system, 2),
 | 
			
		||||
                            Param32(system, 3), ¶m_1)
 | 
			
		||||
                           .raw;
 | 
			
		||||
    system.CurrentArmInterface().SetReg(1, param_1);
 | 
			
		||||
    FuncReturn(system, retval);
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
// Used by CreateCodeMemory32
 | 
			
		||||
template <Result func(Core::System&, Handle*, u32, u32)>
 | 
			
		||||
void SvcWrap32(Core::System& system) {
 | 
			
		||||
    Handle handle = 0;
 | 
			
		||||
 | 
			
		||||
    const u32 retval = func(system, &handle, Param32(system, 1), Param32(system, 2)).raw;
 | 
			
		||||
 | 
			
		||||
    system.CurrentArmInterface().SetReg(1, handle);
 | 
			
		||||
    FuncReturn(system, retval);
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
// Used by ControlCodeMemory32
 | 
			
		||||
template <Result func(Core::System&, Handle, u32, u64, u64, Svc::MemoryPermission)>
 | 
			
		||||
void SvcWrap32(Core::System& system) {
 | 
			
		||||
    const u32 retval =
 | 
			
		||||
        func(system, Param32(system, 0), Param32(system, 1), Param(system, 2), Param(system, 4),
 | 
			
		||||
             static_cast<Svc::MemoryPermission>(Param32(system, 6)))
 | 
			
		||||
            .raw;
 | 
			
		||||
 | 
			
		||||
    FuncReturn(system, retval);
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
// Used by Invalidate/Store/FlushProcessDataCache32
 | 
			
		||||
template <Result func(Core::System&, Handle, u64, u64)>
 | 
			
		||||
void SvcWrap32(Core::System& system) {
 | 
			
		||||
    const u64 address = (Param(system, 3) << 32) | Param(system, 2);
 | 
			
		||||
    const u64 size = (Param(system, 4) << 32) | Param(system, 1);
 | 
			
		||||
    FuncReturn32(system, func(system, Param32(system, 0), address, size).raw);
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
} // namespace Kernel
 | 
			
		||||
		Reference in New Issue
	
	Block a user