SVC: hide details in pimpl
This commit is contained in:
parent
aec8b1e375
commit
c57ee36222
4 changed files with 143 additions and 129 deletions
|
@ -158,7 +158,7 @@ public:
|
||||||
|
|
||||||
ARM_Dynarmic& parent;
|
ARM_Dynarmic& parent;
|
||||||
Core::Timing& timing;
|
Core::Timing& timing;
|
||||||
Kernel::SVC svc_context;
|
Kernel::SVCContext svc_context;
|
||||||
};
|
};
|
||||||
|
|
||||||
ARM_Dynarmic::ARM_Dynarmic(Core::System& system, PrivilegeMode initial_mode)
|
ARM_Dynarmic::ARM_Dynarmic(Core::System& system, PrivilegeMode initial_mode)
|
||||||
|
|
|
@ -3864,7 +3864,7 @@ SWI_INST : {
|
||||||
cpu->NumInstrsToExecute =
|
cpu->NumInstrsToExecute =
|
||||||
num_instrs >= cpu->NumInstrsToExecute ? 0 : cpu->NumInstrsToExecute - num_instrs;
|
num_instrs >= cpu->NumInstrsToExecute ? 0 : cpu->NumInstrsToExecute - num_instrs;
|
||||||
num_instrs = 0;
|
num_instrs = 0;
|
||||||
Kernel::SVC{Core::System::GetInstance()}.CallSVC(inst_cream->num & 0xFFFF);
|
Kernel::SVCContext{Core::System::GetInstance()}.CallSVC(inst_cream->num & 0xFFFF);
|
||||||
// The kernel would call ERET to get here, which clears exclusive memory state.
|
// The kernel would call ERET to get here, which clears exclusive memory state.
|
||||||
cpu->UnsetExclusiveMemoryAddress();
|
cpu->UnsetExclusiveMemoryAddress();
|
||||||
}
|
}
|
||||||
|
|
|
@ -29,6 +29,7 @@
|
||||||
#include "core/hle/kernel/session.h"
|
#include "core/hle/kernel/session.h"
|
||||||
#include "core/hle/kernel/shared_memory.h"
|
#include "core/hle/kernel/shared_memory.h"
|
||||||
#include "core/hle/kernel/svc.h"
|
#include "core/hle/kernel/svc.h"
|
||||||
|
#include "core/hle/kernel/svc_wrapper.h"
|
||||||
#include "core/hle/kernel/thread.h"
|
#include "core/hle/kernel/thread.h"
|
||||||
#include "core/hle/kernel/timer.h"
|
#include "core/hle/kernel/timer.h"
|
||||||
#include "core/hle/kernel/vm_manager.h"
|
#include "core/hle/kernel/vm_manager.h"
|
||||||
|
@ -56,6 +57,133 @@ enum ControlMemoryOperation {
|
||||||
MEMOP_LINEAR = 0x10000,
|
MEMOP_LINEAR = 0x10000,
|
||||||
};
|
};
|
||||||
|
|
||||||
|
struct MemoryInfo {
|
||||||
|
u32 base_address;
|
||||||
|
u32 size;
|
||||||
|
u32 permission;
|
||||||
|
u32 state;
|
||||||
|
};
|
||||||
|
|
||||||
|
struct PageInfo {
|
||||||
|
u32 flags;
|
||||||
|
};
|
||||||
|
|
||||||
|
/// Values accepted by svcGetSystemInfo's type parameter.
|
||||||
|
enum class SystemInfoType {
|
||||||
|
/**
|
||||||
|
* Reports total used memory for all regions or a specific one, according to the extra
|
||||||
|
* parameter. See `SystemInfoMemUsageRegion`.
|
||||||
|
*/
|
||||||
|
REGION_MEMORY_USAGE = 0,
|
||||||
|
/**
|
||||||
|
* Returns the memory usage for certain allocations done internally by the kernel.
|
||||||
|
*/
|
||||||
|
KERNEL_ALLOCATED_PAGES = 2,
|
||||||
|
/**
|
||||||
|
* "This returns the total number of processes which were launched directly by the kernel.
|
||||||
|
* For the ARM11 NATIVE_FIRM kernel, this is 5, for processes sm, fs, pm, loader, and pxi."
|
||||||
|
*/
|
||||||
|
KERNEL_SPAWNED_PIDS = 26,
|
||||||
|
};
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Accepted by svcGetSystemInfo param with REGION_MEMORY_USAGE type. Selects a region to query
|
||||||
|
* memory usage of.
|
||||||
|
*/
|
||||||
|
enum class SystemInfoMemUsageRegion {
|
||||||
|
ALL = 0,
|
||||||
|
APPLICATION = 1,
|
||||||
|
SYSTEM = 2,
|
||||||
|
BASE = 3,
|
||||||
|
};
|
||||||
|
|
||||||
|
class SVC : public SVCWrapper<SVC> {
|
||||||
|
public:
|
||||||
|
SVC(Core::System& system);
|
||||||
|
void CallSVC(u32 immediate);
|
||||||
|
|
||||||
|
private:
|
||||||
|
Core::System& system;
|
||||||
|
Kernel::KernelSystem& kernel;
|
||||||
|
|
||||||
|
friend class SVCWrapper<SVC>;
|
||||||
|
|
||||||
|
// ARM interfaces
|
||||||
|
|
||||||
|
u32 GetReg(std::size_t n);
|
||||||
|
void SetReg(std::size_t n, u32 value);
|
||||||
|
|
||||||
|
// SVC interfaces
|
||||||
|
|
||||||
|
ResultCode ControlMemory(u32* out_addr, u32 addr0, u32 addr1, u32 size, u32 operation,
|
||||||
|
u32 permissions);
|
||||||
|
void ExitProcess();
|
||||||
|
ResultCode MapMemoryBlock(Handle handle, u32 addr, u32 permissions, u32 other_permissions);
|
||||||
|
ResultCode UnmapMemoryBlock(Handle handle, u32 addr);
|
||||||
|
ResultCode ConnectToPort(Handle* out_handle, VAddr port_name_address);
|
||||||
|
ResultCode SendSyncRequest(Handle handle);
|
||||||
|
ResultCode CloseHandle(Handle handle);
|
||||||
|
ResultCode WaitSynchronization1(Handle handle, s64 nano_seconds);
|
||||||
|
ResultCode WaitSynchronizationN(s32* out, VAddr handles_address, s32 handle_count,
|
||||||
|
bool wait_all, s64 nano_seconds);
|
||||||
|
ResultCode ReplyAndReceive(s32* index, VAddr handles_address, s32 handle_count,
|
||||||
|
Handle reply_target);
|
||||||
|
ResultCode CreateAddressArbiter(Handle* out_handle);
|
||||||
|
ResultCode ArbitrateAddress(Handle handle, u32 address, u32 type, u32 value, s64 nanoseconds);
|
||||||
|
void Break(u8 break_reason);
|
||||||
|
void OutputDebugString(VAddr address, s32 len);
|
||||||
|
ResultCode GetResourceLimit(Handle* resource_limit, Handle process_handle);
|
||||||
|
ResultCode GetResourceLimitCurrentValues(VAddr values, Handle resource_limit_handle,
|
||||||
|
VAddr names, u32 name_count);
|
||||||
|
ResultCode GetResourceLimitLimitValues(VAddr values, Handle resource_limit_handle, VAddr names,
|
||||||
|
u32 name_count);
|
||||||
|
ResultCode CreateThread(Handle* out_handle, u32 entry_point, u32 arg, VAddr stack_top,
|
||||||
|
u32 priority, s32 processor_id);
|
||||||
|
void ExitThread();
|
||||||
|
ResultCode GetThreadPriority(u32* priority, Handle handle);
|
||||||
|
ResultCode SetThreadPriority(Handle handle, u32 priority);
|
||||||
|
ResultCode CreateMutex(Handle* out_handle, u32 initial_locked);
|
||||||
|
ResultCode ReleaseMutex(Handle handle);
|
||||||
|
ResultCode GetProcessId(u32* process_id, Handle process_handle);
|
||||||
|
ResultCode GetProcessIdOfThread(u32* process_id, Handle thread_handle);
|
||||||
|
ResultCode GetThreadId(u32* thread_id, Handle handle);
|
||||||
|
ResultCode CreateSemaphore(Handle* out_handle, s32 initial_count, s32 max_count);
|
||||||
|
ResultCode ReleaseSemaphore(s32* count, Handle handle, s32 release_count);
|
||||||
|
ResultCode QueryProcessMemory(MemoryInfo* memory_info, PageInfo* page_info,
|
||||||
|
Handle process_handle, u32 addr);
|
||||||
|
ResultCode QueryMemory(MemoryInfo* memory_info, PageInfo* page_info, u32 addr);
|
||||||
|
ResultCode CreateEvent(Handle* out_handle, u32 reset_type);
|
||||||
|
ResultCode DuplicateHandle(Handle* out, Handle handle);
|
||||||
|
ResultCode SignalEvent(Handle handle);
|
||||||
|
ResultCode ClearEvent(Handle handle);
|
||||||
|
ResultCode CreateTimer(Handle* out_handle, u32 reset_type);
|
||||||
|
ResultCode ClearTimer(Handle handle);
|
||||||
|
ResultCode SetTimer(Handle handle, s64 initial, s64 interval);
|
||||||
|
ResultCode CancelTimer(Handle handle);
|
||||||
|
void SleepThread(s64 nanoseconds);
|
||||||
|
s64 GetSystemTick();
|
||||||
|
ResultCode CreateMemoryBlock(Handle* out_handle, u32 addr, u32 size, u32 my_permission,
|
||||||
|
u32 other_permission);
|
||||||
|
ResultCode CreatePort(Handle* server_port, Handle* client_port, VAddr name_address,
|
||||||
|
u32 max_sessions);
|
||||||
|
ResultCode CreateSessionToPort(Handle* out_client_session, Handle client_port_handle);
|
||||||
|
ResultCode CreateSession(Handle* server_session, Handle* client_session);
|
||||||
|
ResultCode AcceptSession(Handle* out_server_session, Handle server_port_handle);
|
||||||
|
ResultCode GetSystemInfo(s64* out, u32 type, s32 param);
|
||||||
|
ResultCode GetProcessInfo(s64* out, Handle process_handle, u32 type);
|
||||||
|
|
||||||
|
struct FunctionDef {
|
||||||
|
using Func = void (SVC::*)();
|
||||||
|
|
||||||
|
u32 id;
|
||||||
|
Func func;
|
||||||
|
const char* name;
|
||||||
|
};
|
||||||
|
|
||||||
|
static const FunctionDef SVC_Table[];
|
||||||
|
static const FunctionDef* GetSVCInfo(u32 func_num);
|
||||||
|
};
|
||||||
|
|
||||||
/// Map application or GSP heap memory
|
/// Map application or GSP heap memory
|
||||||
ResultCode SVC::ControlMemory(u32* out_addr, u32 addr0, u32 addr1, u32 size, u32 operation,
|
ResultCode SVC::ControlMemory(u32* out_addr, u32 addr0, u32 addr1, u32 size, u32 operation,
|
||||||
u32 permissions) {
|
u32 permissions) {
|
||||||
|
@ -1446,4 +1574,11 @@ void SVC::SetReg(std::size_t n, u32 value) {
|
||||||
system.CPU().SetReg(static_cast<int>(n), value);
|
system.CPU().SetReg(static_cast<int>(n), value);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
SVCContext::SVCContext(Core::System& system) : impl(std::make_unique<SVC>(system)) {}
|
||||||
|
SVCContext::~SVCContext() = default;
|
||||||
|
|
||||||
|
void SVCContext::CallSVC(u32 immediate) {
|
||||||
|
impl->CallSVC(immediate);
|
||||||
|
}
|
||||||
|
|
||||||
} // namespace Kernel
|
} // namespace Kernel
|
||||||
|
|
|
@ -4,12 +4,8 @@
|
||||||
|
|
||||||
#pragma once
|
#pragma once
|
||||||
|
|
||||||
|
#include <memory>
|
||||||
#include "common/common_types.h"
|
#include "common/common_types.h"
|
||||||
#include "core/hle/kernel/handle_table.h"
|
|
||||||
#include "core/hle/kernel/svc_wrapper.h"
|
|
||||||
#include "core/hle/result.h"
|
|
||||||
|
|
||||||
class ARM_Interface;
|
|
||||||
|
|
||||||
namespace Core {
|
namespace Core {
|
||||||
class System;
|
class System;
|
||||||
|
@ -17,133 +13,16 @@ class System;
|
||||||
|
|
||||||
namespace Kernel {
|
namespace Kernel {
|
||||||
|
|
||||||
class KernelSystem;
|
class SVC;
|
||||||
|
|
||||||
struct MemoryInfo {
|
class SVCContext {
|
||||||
u32 base_address;
|
|
||||||
u32 size;
|
|
||||||
u32 permission;
|
|
||||||
u32 state;
|
|
||||||
};
|
|
||||||
|
|
||||||
struct PageInfo {
|
|
||||||
u32 flags;
|
|
||||||
};
|
|
||||||
|
|
||||||
/// Values accepted by svcGetSystemInfo's type parameter.
|
|
||||||
enum class SystemInfoType {
|
|
||||||
/**
|
|
||||||
* Reports total used memory for all regions or a specific one, according to the extra
|
|
||||||
* parameter. See `SystemInfoMemUsageRegion`.
|
|
||||||
*/
|
|
||||||
REGION_MEMORY_USAGE = 0,
|
|
||||||
/**
|
|
||||||
* Returns the memory usage for certain allocations done internally by the kernel.
|
|
||||||
*/
|
|
||||||
KERNEL_ALLOCATED_PAGES = 2,
|
|
||||||
/**
|
|
||||||
* "This returns the total number of processes which were launched directly by the kernel.
|
|
||||||
* For the ARM11 NATIVE_FIRM kernel, this is 5, for processes sm, fs, pm, loader, and pxi."
|
|
||||||
*/
|
|
||||||
KERNEL_SPAWNED_PIDS = 26,
|
|
||||||
};
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Accepted by svcGetSystemInfo param with REGION_MEMORY_USAGE type. Selects a region to query
|
|
||||||
* memory usage of.
|
|
||||||
*/
|
|
||||||
enum class SystemInfoMemUsageRegion {
|
|
||||||
ALL = 0,
|
|
||||||
APPLICATION = 1,
|
|
||||||
SYSTEM = 2,
|
|
||||||
BASE = 3,
|
|
||||||
};
|
|
||||||
|
|
||||||
class SVC : public SVCWrapper<SVC> {
|
|
||||||
public:
|
public:
|
||||||
SVC(Core::System& system);
|
SVCContext(Core::System& system);
|
||||||
|
~SVCContext();
|
||||||
void CallSVC(u32 immediate);
|
void CallSVC(u32 immediate);
|
||||||
|
|
||||||
private:
|
private:
|
||||||
Core::System& system;
|
std::unique_ptr<SVC> impl;
|
||||||
Kernel::KernelSystem& kernel;
|
|
||||||
|
|
||||||
friend class SVCWrapper<SVC>;
|
|
||||||
|
|
||||||
// ARM interfaces
|
|
||||||
|
|
||||||
u32 GetReg(std::size_t n);
|
|
||||||
void SetReg(std::size_t n, u32 value);
|
|
||||||
|
|
||||||
// SVC interfaces
|
|
||||||
|
|
||||||
ResultCode ControlMemory(u32* out_addr, u32 addr0, u32 addr1, u32 size, u32 operation,
|
|
||||||
u32 permissions);
|
|
||||||
void ExitProcess();
|
|
||||||
ResultCode MapMemoryBlock(Handle handle, u32 addr, u32 permissions, u32 other_permissions);
|
|
||||||
ResultCode UnmapMemoryBlock(Handle handle, u32 addr);
|
|
||||||
ResultCode ConnectToPort(Handle* out_handle, VAddr port_name_address);
|
|
||||||
ResultCode SendSyncRequest(Handle handle);
|
|
||||||
ResultCode CloseHandle(Handle handle);
|
|
||||||
ResultCode WaitSynchronization1(Handle handle, s64 nano_seconds);
|
|
||||||
ResultCode WaitSynchronizationN(s32* out, VAddr handles_address, s32 handle_count,
|
|
||||||
bool wait_all, s64 nano_seconds);
|
|
||||||
ResultCode ReplyAndReceive(s32* index, VAddr handles_address, s32 handle_count,
|
|
||||||
Handle reply_target);
|
|
||||||
ResultCode CreateAddressArbiter(Handle* out_handle);
|
|
||||||
ResultCode ArbitrateAddress(Handle handle, u32 address, u32 type, u32 value, s64 nanoseconds);
|
|
||||||
void Break(u8 break_reason);
|
|
||||||
void OutputDebugString(VAddr address, s32 len);
|
|
||||||
ResultCode GetResourceLimit(Handle* resource_limit, Handle process_handle);
|
|
||||||
ResultCode GetResourceLimitCurrentValues(VAddr values, Handle resource_limit_handle,
|
|
||||||
VAddr names, u32 name_count);
|
|
||||||
ResultCode GetResourceLimitLimitValues(VAddr values, Handle resource_limit_handle, VAddr names,
|
|
||||||
u32 name_count);
|
|
||||||
ResultCode CreateThread(Handle* out_handle, u32 entry_point, u32 arg, VAddr stack_top,
|
|
||||||
u32 priority, s32 processor_id);
|
|
||||||
void ExitThread();
|
|
||||||
ResultCode GetThreadPriority(u32* priority, Handle handle);
|
|
||||||
ResultCode SetThreadPriority(Handle handle, u32 priority);
|
|
||||||
ResultCode CreateMutex(Handle* out_handle, u32 initial_locked);
|
|
||||||
ResultCode ReleaseMutex(Handle handle);
|
|
||||||
ResultCode GetProcessId(u32* process_id, Handle process_handle);
|
|
||||||
ResultCode GetProcessIdOfThread(u32* process_id, Handle thread_handle);
|
|
||||||
ResultCode GetThreadId(u32* thread_id, Handle handle);
|
|
||||||
ResultCode CreateSemaphore(Handle* out_handle, s32 initial_count, s32 max_count);
|
|
||||||
ResultCode ReleaseSemaphore(s32* count, Handle handle, s32 release_count);
|
|
||||||
ResultCode QueryProcessMemory(MemoryInfo* memory_info, PageInfo* page_info,
|
|
||||||
Handle process_handle, u32 addr);
|
|
||||||
ResultCode QueryMemory(MemoryInfo* memory_info, PageInfo* page_info, u32 addr);
|
|
||||||
ResultCode CreateEvent(Handle* out_handle, u32 reset_type);
|
|
||||||
ResultCode DuplicateHandle(Handle* out, Handle handle);
|
|
||||||
ResultCode SignalEvent(Handle handle);
|
|
||||||
ResultCode ClearEvent(Handle handle);
|
|
||||||
ResultCode CreateTimer(Handle* out_handle, u32 reset_type);
|
|
||||||
ResultCode ClearTimer(Handle handle);
|
|
||||||
ResultCode SetTimer(Handle handle, s64 initial, s64 interval);
|
|
||||||
ResultCode CancelTimer(Handle handle);
|
|
||||||
void SleepThread(s64 nanoseconds);
|
|
||||||
s64 GetSystemTick();
|
|
||||||
ResultCode CreateMemoryBlock(Handle* out_handle, u32 addr, u32 size, u32 my_permission,
|
|
||||||
u32 other_permission);
|
|
||||||
ResultCode CreatePort(Handle* server_port, Handle* client_port, VAddr name_address,
|
|
||||||
u32 max_sessions);
|
|
||||||
ResultCode CreateSessionToPort(Handle* out_client_session, Handle client_port_handle);
|
|
||||||
ResultCode CreateSession(Handle* server_session, Handle* client_session);
|
|
||||||
ResultCode AcceptSession(Handle* out_server_session, Handle server_port_handle);
|
|
||||||
ResultCode GetSystemInfo(s64* out, u32 type, s32 param);
|
|
||||||
ResultCode GetProcessInfo(s64* out, Handle process_handle, u32 type);
|
|
||||||
|
|
||||||
struct FunctionDef {
|
|
||||||
using Func = void (SVC::*)();
|
|
||||||
|
|
||||||
u32 id;
|
|
||||||
Func func;
|
|
||||||
const char* name;
|
|
||||||
};
|
|
||||||
|
|
||||||
static const FunctionDef SVC_Table[];
|
|
||||||
static const FunctionDef* GetSVCInfo(u32 func_num);
|
|
||||||
};
|
};
|
||||||
|
|
||||||
} // namespace Kernel
|
} // namespace Kernel
|
||||||
|
|
Loading…
Reference in a new issue