2018-05-05 02:25:26 +02:00
|
|
|
#include <switch.h>
|
|
|
|
#include "pm_registration.hpp"
|
2018-05-07 11:50:33 +02:00
|
|
|
#include "pm_resource_limits.hpp"
|
2018-05-05 02:25:26 +02:00
|
|
|
#include "pm_shell.hpp"
|
2018-05-06 10:23:47 +02:00
|
|
|
#include "pm_debug.hpp"
|
2018-05-05 02:25:26 +02:00
|
|
|
|
|
|
|
static bool g_has_boot_finished = false;
|
|
|
|
|
|
|
|
Result ShellService::dispatch(IpcParsedCommand &r, IpcCommand &out_c, u64 cmd_id, u8 *pointer_buffer, size_t pointer_buffer_size) {
|
|
|
|
Result rc = 0xF601;
|
2018-05-06 10:23:47 +02:00
|
|
|
LogForService("SHELLSRV", 8);
|
|
|
|
LogForService(&cmd_id, 8);
|
|
|
|
LogForService(armGetTls(), 0x100);
|
2018-05-05 02:25:26 +02:00
|
|
|
if (kernelAbove500()) {
|
|
|
|
switch ((ShellCmd_5X)cmd_id) {
|
|
|
|
case Shell_Cmd_5X_LaunchProcess:
|
|
|
|
rc = WrapIpcCommandImpl<&ShellService::launch_process>(this, r, out_c, pointer_buffer, pointer_buffer_size);
|
|
|
|
break;
|
|
|
|
case Shell_Cmd_5X_TerminateProcessId:
|
|
|
|
rc = WrapIpcCommandImpl<&ShellService::terminate_process_id>(this, r, out_c, pointer_buffer, pointer_buffer_size);
|
|
|
|
break;
|
|
|
|
case Shell_Cmd_5X_TerminateTitleId:
|
|
|
|
rc = WrapIpcCommandImpl<&ShellService::terminate_title_id>(this, r, out_c, pointer_buffer, pointer_buffer_size);
|
|
|
|
break;
|
|
|
|
case Shell_Cmd_5X_GetProcessWaitEvent:
|
|
|
|
rc = WrapIpcCommandImpl<&ShellService::get_process_wait_event>(this, r, out_c, pointer_buffer, pointer_buffer_size);
|
|
|
|
break;
|
|
|
|
case Shell_Cmd_5X_GetProcessEventType:
|
|
|
|
rc = WrapIpcCommandImpl<&ShellService::get_process_event_type>(this, r, out_c, pointer_buffer, pointer_buffer_size);
|
|
|
|
break;
|
|
|
|
case Shell_Cmd_5X_NotifyBootFinished:
|
|
|
|
rc = WrapIpcCommandImpl<&ShellService::notify_boot_finished>(this, r, out_c, pointer_buffer, pointer_buffer_size);
|
|
|
|
break;
|
|
|
|
case Shell_Cmd_5X_GetApplicationProcessId:
|
|
|
|
rc = WrapIpcCommandImpl<&ShellService::get_application_process_id>(this, r, out_c, pointer_buffer, pointer_buffer_size);
|
|
|
|
break;
|
|
|
|
case Shell_Cmd_5X_BoostSystemMemoryResourceLimit:
|
|
|
|
rc = WrapIpcCommandImpl<&ShellService::boost_system_memory_resource_limit>(this, r, out_c, pointer_buffer, pointer_buffer_size);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
switch ((ShellCmd)cmd_id) {
|
|
|
|
case Shell_Cmd_LaunchProcess:
|
|
|
|
rc = WrapIpcCommandImpl<&ShellService::launch_process>(this, r, out_c, pointer_buffer, pointer_buffer_size);
|
|
|
|
break;
|
|
|
|
case Shell_Cmd_TerminateProcessId:
|
|
|
|
rc = WrapIpcCommandImpl<&ShellService::terminate_process_id>(this, r, out_c, pointer_buffer, pointer_buffer_size);
|
|
|
|
break;
|
|
|
|
case Shell_Cmd_TerminateTitleId:
|
|
|
|
rc = WrapIpcCommandImpl<&ShellService::terminate_title_id>(this, r, out_c, pointer_buffer, pointer_buffer_size);
|
|
|
|
break;
|
|
|
|
case Shell_Cmd_GetProcessWaitEvent:
|
|
|
|
rc = WrapIpcCommandImpl<&ShellService::get_process_wait_event>(this, r, out_c, pointer_buffer, pointer_buffer_size);
|
|
|
|
break;
|
|
|
|
case Shell_Cmd_GetProcessEventType:
|
|
|
|
rc = WrapIpcCommandImpl<&ShellService::get_process_event_type>(this, r, out_c, pointer_buffer, pointer_buffer_size);
|
|
|
|
break;
|
2018-05-05 03:56:59 +02:00
|
|
|
case Shell_Cmd_FinalizeExitedProcess:
|
2018-05-05 04:16:40 +02:00
|
|
|
rc = WrapIpcCommandImpl<&ShellService::finalize_exited_process>(this, r, out_c, pointer_buffer, pointer_buffer_size);
|
2018-05-05 02:25:26 +02:00
|
|
|
break;
|
|
|
|
case Shell_Cmd_ClearProcessNotificationFlag:
|
|
|
|
rc = WrapIpcCommandImpl<&ShellService::clear_process_notification_flag>(this, r, out_c, pointer_buffer, pointer_buffer_size);
|
|
|
|
break;
|
|
|
|
case Shell_Cmd_NotifyBootFinished:
|
|
|
|
rc = WrapIpcCommandImpl<&ShellService::notify_boot_finished>(this, r, out_c, pointer_buffer, pointer_buffer_size);
|
|
|
|
break;
|
|
|
|
case Shell_Cmd_GetApplicationProcessId:
|
|
|
|
rc = WrapIpcCommandImpl<&ShellService::get_application_process_id>(this, r, out_c, pointer_buffer, pointer_buffer_size);
|
|
|
|
break;
|
|
|
|
case Shell_Cmd_BoostSystemMemoryResourceLimit:
|
|
|
|
rc = WrapIpcCommandImpl<&ShellService::boost_system_memory_resource_limit>(this, r, out_c, pointer_buffer, pointer_buffer_size);
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
2018-05-06 10:23:47 +02:00
|
|
|
//Log(armGetTls(), 0x100);
|
|
|
|
LogForService(armGetTls(), 0x100);
|
|
|
|
if (R_FAILED(rc)) {
|
|
|
|
|
|
|
|
Reboot();
|
|
|
|
}
|
2018-05-05 02:25:26 +02:00
|
|
|
return rc;
|
|
|
|
}
|
|
|
|
|
|
|
|
Result ShellService::handle_deferred() {
|
|
|
|
/* This service is never deferrable. */
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2018-05-05 04:16:40 +02:00
|
|
|
std::tuple<Result, u64> ShellService::launch_process(u64 launch_flags, Registration::TidSid tid_sid) {
|
2018-05-05 03:56:59 +02:00
|
|
|
u64 pid = 0;
|
|
|
|
Result rc = Registration::LaunchProcessByTidSid(tid_sid, launch_flags, &pid);
|
|
|
|
return {rc, pid};
|
2018-05-05 02:25:26 +02:00
|
|
|
}
|
|
|
|
|
2018-05-05 04:16:40 +02:00
|
|
|
std::tuple<Result> ShellService::terminate_process_id(u64 pid) {
|
2018-05-05 03:56:59 +02:00
|
|
|
Registration::AutoProcessListLock auto_lock;
|
|
|
|
|
|
|
|
Registration::Process *proc = Registration::GetProcess(pid);
|
|
|
|
if (proc != NULL) {
|
2018-05-06 10:23:47 +02:00
|
|
|
return {svcTerminateProcess(proc->handle)};
|
2018-05-05 03:56:59 +02:00
|
|
|
} else {
|
|
|
|
return {0x20F};
|
|
|
|
}
|
2018-05-05 02:25:26 +02:00
|
|
|
}
|
|
|
|
|
2018-05-05 04:16:40 +02:00
|
|
|
std::tuple<Result> ShellService::terminate_title_id(u64 tid) {
|
2018-05-05 03:56:59 +02:00
|
|
|
Registration::AutoProcessListLock auto_lock;
|
|
|
|
|
|
|
|
Registration::Process *proc = Registration::GetProcessByTitleId(tid);
|
|
|
|
if (proc != NULL) {
|
2018-05-06 10:23:47 +02:00
|
|
|
return {svcTerminateProcess(proc->handle)};
|
2018-05-05 03:56:59 +02:00
|
|
|
} else {
|
|
|
|
return {0x20F};
|
|
|
|
}
|
2018-05-05 02:25:26 +02:00
|
|
|
}
|
|
|
|
|
2018-05-05 04:16:40 +02:00
|
|
|
std::tuple<Result, CopiedHandle> ShellService::get_process_wait_event() {
|
2018-05-05 03:56:59 +02:00
|
|
|
return {0x0, Registration::GetProcessEventHandle()};
|
2018-05-05 02:25:26 +02:00
|
|
|
}
|
|
|
|
|
2018-05-05 04:16:40 +02:00
|
|
|
std::tuple<Result, u64, u64> ShellService::get_process_event_type() {
|
2018-05-05 03:56:59 +02:00
|
|
|
u64 type, pid;
|
|
|
|
Registration::GetProcessEventType(&pid, &type);
|
|
|
|
return {0x0, type, pid};
|
2018-05-05 02:25:26 +02:00
|
|
|
}
|
|
|
|
|
2018-05-05 04:16:40 +02:00
|
|
|
std::tuple<Result> ShellService::finalize_exited_process(u64 pid) {
|
2018-05-05 03:56:59 +02:00
|
|
|
Registration::AutoProcessListLock auto_lock;
|
|
|
|
|
|
|
|
Registration::Process *proc = Registration::GetProcess(pid);
|
|
|
|
if (proc == NULL) {
|
|
|
|
return {0x20F};
|
|
|
|
} else if (proc->state != ProcessState_Exited) {
|
|
|
|
return {0x60F};
|
|
|
|
} else {
|
|
|
|
Registration::FinalizeExitedProcess(proc);
|
|
|
|
return {0x0};
|
|
|
|
}
|
2018-05-05 02:25:26 +02:00
|
|
|
}
|
|
|
|
|
2018-05-05 04:16:40 +02:00
|
|
|
std::tuple<Result> ShellService::clear_process_notification_flag(u64 pid) {
|
2018-05-05 03:56:59 +02:00
|
|
|
Registration::AutoProcessListLock auto_lock;
|
|
|
|
|
|
|
|
Registration::Process *proc = Registration::GetProcess(pid);
|
|
|
|
if (proc != NULL) {
|
|
|
|
proc->flags &= ~2;
|
|
|
|
return {0x0};
|
|
|
|
} else {
|
|
|
|
return {0x20F};
|
|
|
|
}
|
2018-05-05 02:25:26 +02:00
|
|
|
}
|
|
|
|
|
2018-05-05 04:16:40 +02:00
|
|
|
std::tuple<Result> ShellService::notify_boot_finished() {
|
2018-05-05 03:56:59 +02:00
|
|
|
u64 boot2_pid;
|
2018-05-05 02:25:26 +02:00
|
|
|
if (!g_has_boot_finished) {
|
|
|
|
g_has_boot_finished = true;
|
2018-05-05 03:56:59 +02:00
|
|
|
return {Registration::LaunchProcess(BOOT2_TITLE_ID, FsStorageId_NandSystem, 0, &boot2_pid)};
|
2018-05-05 02:25:26 +02:00
|
|
|
}
|
|
|
|
return {0};
|
|
|
|
}
|
|
|
|
|
2018-05-05 04:16:40 +02:00
|
|
|
std::tuple<Result, u64> ShellService::get_application_process_id() {
|
2018-05-05 03:56:59 +02:00
|
|
|
Registration::AutoProcessListLock auto_lock;
|
|
|
|
|
|
|
|
Registration::Process *app_proc;
|
|
|
|
if (Registration::HasApplicationProcess(&app_proc)) {
|
|
|
|
return {0, app_proc->pid};
|
|
|
|
}
|
|
|
|
return {0x20F, 0};
|
2018-05-05 02:25:26 +02:00
|
|
|
}
|
|
|
|
|
2018-05-07 11:50:33 +02:00
|
|
|
std::tuple<Result> ShellService::boost_system_memory_resource_limit(u64 sysmem_size) {
|
2018-05-05 03:56:59 +02:00
|
|
|
if (!kernelAbove400()) {
|
|
|
|
return {0xF601};
|
|
|
|
}
|
|
|
|
|
2018-05-05 02:25:26 +02:00
|
|
|
/* TODO */
|
2018-05-07 11:50:33 +02:00
|
|
|
return {ResourceLimitUtils::BoostSystemMemoryResourceLimit(sysmem_size)};
|
2018-05-05 02:25:26 +02:00
|
|
|
}
|