// Copyright 2017 plutoo #include "types.h" #include "result.h" #include "arm/atomics.h" #include "kernel/ipc.h" #include "services/pm.h" #include "services/sm.h" static Service g_pmdmntSrv, g_pmshellSrv; static u64 g_pmdmntRefCnt, g_pmshellRefCnt; Result pmdmntInitialize(void) { atomicIncrement64(&g_pmdmntRefCnt); if (serviceIsActive(&g_pmdmntSrv)) return 0; return smGetService(&g_pmdmntSrv, "pm:dmnt"); } void pmdmntExit(void) { if (atomicDecrement64(&g_pmdmntRefCnt) == 0) { serviceClose(&g_pmdmntSrv); } } Result pmshellInitialize(void) { atomicIncrement64(&g_pmshellRefCnt); if (serviceIsActive(&g_pmshellSrv)) return 0; return smGetService(&g_pmshellSrv, "pm:shell"); } void pmshellExit(void) { if (atomicDecrement64(&g_pmshellRefCnt) == 0) { serviceClose(&g_pmshellSrv); } } Result pmdmntStartProcess(u64 pid) { IpcCommand c; ipcInitialize(&c); struct { u64 magic; u64 cmd_id; u64 pid; } *raw; raw = ipcPrepareHeader(&c, sizeof(*raw)); raw->magic = SFCI_MAGIC; raw->cmd_id = 2; raw->pid = pid; Result rc = serviceIpcDispatch(&g_pmdmntSrv); if (R_SUCCEEDED(rc)) { IpcParsedCommand r; ipcParse(&r); struct { u64 magic; u64 result; } *resp = r.Raw; rc = resp->result; } return rc; } Result pmdmntGetTitlePid(u64* pid_out, u64 title_id) { IpcCommand c; ipcInitialize(&c); struct { u64 magic; u64 cmd_id; u64 title_id; } *raw; raw = ipcPrepareHeader(&c, sizeof(*raw)); raw->magic = SFCI_MAGIC; raw->cmd_id = 3; raw->title_id = title_id; Result rc = serviceIpcDispatch(&g_pmdmntSrv); if (R_SUCCEEDED(rc)) { IpcParsedCommand r; ipcParse(&r); struct { u64 magic; u64 result; u64 pid; } *resp = r.Raw; rc = resp->result; if (R_SUCCEEDED(rc)) { *pid_out = resp->pid; } } return rc; } Result pmdmntEnableDebugForTitleId(Handle* handle_out, u64 title_id) { IpcCommand c; ipcInitialize(&c); struct { u64 magic; u64 cmd_id; u64 title_id; } *raw; raw = ipcPrepareHeader(&c, sizeof(*raw)); raw->magic = SFCI_MAGIC; raw->cmd_id = 4; raw->title_id = title_id; Result rc = serviceIpcDispatch(&g_pmdmntSrv); if (R_SUCCEEDED(rc)) { IpcParsedCommand r; ipcParse(&r); struct { u64 magic; u64 result; } *resp = r.Raw; rc = resp->result; if (R_SUCCEEDED(rc)) { *handle_out = r.Handles[0]; } } return rc; } Result pmdmntGetApplicationPid(u64* pid_out) { IpcCommand c; ipcInitialize(&c); struct { u64 magic; u64 cmd_id; } *raw; raw = ipcPrepareHeader(&c, sizeof(*raw)); raw->magic = SFCI_MAGIC; raw->cmd_id = 5; Result rc = serviceIpcDispatch(&g_pmdmntSrv); if (R_SUCCEEDED(rc)) { IpcParsedCommand r; ipcParse(&r); struct { u64 magic; u64 result; u64 pid; } *resp = r.Raw; rc = resp->result; if (R_SUCCEEDED(rc)) { *pid_out = resp->pid; } } return rc; } Result pmdmntEnableDebugForApplication(Handle* handle_out) { IpcCommand c; ipcInitialize(&c); struct { u64 magic; u64 cmd_id; } *raw; raw = ipcPrepareHeader(&c, sizeof(*raw)); raw->magic = SFCI_MAGIC; raw->cmd_id = 6; Result rc = serviceIpcDispatch(&g_pmdmntSrv); if (R_SUCCEEDED(rc)) { IpcParsedCommand r; ipcParse(&r); struct { u64 magic; u64 result; } *resp = r.Raw; rc = resp->result; if (R_SUCCEEDED(rc)) { *handle_out = r.Handles[0]; } } return rc; } Result pmshellLaunchProcess(u32 launch_flags, u64 titleID, u64 storageID, u64 *pid) { IpcCommand c; ipcInitialize(&c); struct { u64 magic; u64 cmd_id; u32 launch_flags; u64 titleID; u64 storageID; } *raw; raw = ipcPrepareHeader(&c, sizeof(*raw)); raw->magic = SFCI_MAGIC; raw->cmd_id = 0; raw->launch_flags = launch_flags; raw->titleID = titleID; raw->storageID = storageID; Result rc = serviceIpcDispatch(&g_pmshellSrv); if (R_SUCCEEDED(rc)) { IpcParsedCommand r; ipcParse(&r); struct { u64 magic; u64 result; u64 pid; } *resp = r.Raw; rc = resp->result; if (R_SUCCEEDED(rc) && pid) *pid = resp->pid; } return rc; }