2018-09-07 16:00:13 +01:00
|
|
|
/*
|
2019-04-08 03:00:49 +01:00
|
|
|
* Copyright (c) 2018-2019 Atmosphère-NX
|
2018-09-07 16:00:13 +01:00
|
|
|
*
|
|
|
|
* This program is free software; you can redistribute it and/or modify it
|
|
|
|
* under the terms and conditions of the GNU General Public License,
|
|
|
|
* version 2, as published by the Free Software Foundation.
|
|
|
|
*
|
|
|
|
* This program is distributed in the hope it will be useful, but WITHOUT
|
|
|
|
* ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
|
|
|
|
* FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License for
|
|
|
|
* more details.
|
|
|
|
*
|
|
|
|
* You should have received a copy of the GNU General Public License
|
|
|
|
* along with this program. If not, see <http://www.gnu.org/licenses/>.
|
|
|
|
*/
|
2019-05-28 05:52:28 +01:00
|
|
|
|
2018-05-05 01:25:26 +01:00
|
|
|
#include <switch.h>
|
2018-06-15 00:50:01 +01:00
|
|
|
#include <stratosphere.hpp>
|
2018-05-05 01:25:26 +01:00
|
|
|
#include "pm_registration.hpp"
|
2018-05-07 10:50:33 +01:00
|
|
|
#include "pm_resource_limits.hpp"
|
2018-05-05 01:25:26 +01:00
|
|
|
#include "pm_shell.hpp"
|
2018-07-27 10:23:53 +01:00
|
|
|
#include "pm_boot2.hpp"
|
2018-05-05 01:25:26 +01:00
|
|
|
|
|
|
|
static bool g_has_boot_finished = false;
|
|
|
|
|
2018-10-30 13:29:30 +00:00
|
|
|
Result ShellService::LaunchProcess(Out<u64> pid, Registration::TidSid tid_sid, u32 launch_flags) {
|
2018-10-30 00:29:35 +00:00
|
|
|
return Registration::LaunchProcessByTidSid(tid_sid, launch_flags, pid.GetPointer());
|
2018-05-05 01:25:26 +01:00
|
|
|
}
|
|
|
|
|
2018-10-30 00:29:35 +00:00
|
|
|
Result ShellService::TerminateProcessId(u64 pid) {
|
2019-03-26 00:12:19 +00:00
|
|
|
std::scoped_lock<ProcessList &> lk(Registration::GetProcessList());
|
2019-05-28 05:52:28 +01:00
|
|
|
|
2018-10-30 00:29:35 +00:00
|
|
|
auto proc = Registration::GetProcess(pid);
|
|
|
|
if (proc != nullptr) {
|
|
|
|
return svcTerminateProcess(proc->handle);
|
2018-05-05 02:56:59 +01:00
|
|
|
} else {
|
2019-03-28 22:16:36 +00:00
|
|
|
return ResultPmProcessNotFound;
|
2018-05-05 02:56:59 +01:00
|
|
|
}
|
2018-05-05 01:25:26 +01:00
|
|
|
}
|
|
|
|
|
2018-10-30 00:29:35 +00:00
|
|
|
Result ShellService::TerminateTitleId(u64 tid) {
|
2019-03-26 00:12:19 +00:00
|
|
|
std::scoped_lock<ProcessList &> lk(Registration::GetProcessList());
|
2019-05-28 05:52:28 +01:00
|
|
|
|
2018-10-30 00:29:35 +00:00
|
|
|
auto proc = Registration::GetProcessByTitleId(tid);
|
2018-05-05 02:56:59 +01:00
|
|
|
if (proc != NULL) {
|
2018-10-30 00:29:35 +00:00
|
|
|
return svcTerminateProcess(proc->handle);
|
2018-05-05 02:56:59 +01:00
|
|
|
} else {
|
2019-03-28 22:16:36 +00:00
|
|
|
return ResultPmProcessNotFound;
|
2018-05-05 02:56:59 +01:00
|
|
|
}
|
2018-05-05 01:25:26 +01:00
|
|
|
}
|
|
|
|
|
2018-10-30 00:29:35 +00:00
|
|
|
void ShellService::GetProcessWaitEvent(Out<CopiedHandle> event) {
|
|
|
|
event.SetValue(Registration::GetProcessEventHandle());
|
2018-05-05 01:25:26 +01:00
|
|
|
}
|
|
|
|
|
2018-10-30 00:29:35 +00:00
|
|
|
void ShellService::GetProcessEventType(Out<u64> type, Out<u64> pid) {
|
|
|
|
Registration::GetProcessEventType(pid.GetPointer(), type.GetPointer());
|
2018-05-05 01:25:26 +01:00
|
|
|
}
|
|
|
|
|
2018-10-30 00:29:35 +00:00
|
|
|
Result ShellService::FinalizeExitedProcess(u64 pid) {
|
2019-03-26 00:12:19 +00:00
|
|
|
std::scoped_lock<ProcessList &> lk(Registration::GetProcessList());
|
2019-05-28 05:52:28 +01:00
|
|
|
|
2018-10-30 00:29:35 +00:00
|
|
|
auto proc = Registration::GetProcess(pid);
|
2018-05-05 02:56:59 +01:00
|
|
|
if (proc == NULL) {
|
2019-03-28 22:16:36 +00:00
|
|
|
return ResultPmProcessNotFound;
|
2018-05-05 02:56:59 +01:00
|
|
|
} else if (proc->state != ProcessState_Exited) {
|
2019-03-28 22:16:36 +00:00
|
|
|
return ResultPmNotExited;
|
2018-05-05 02:56:59 +01:00
|
|
|
} else {
|
|
|
|
Registration::FinalizeExitedProcess(proc);
|
2019-03-29 05:39:39 +00:00
|
|
|
return ResultSuccess;
|
2018-05-05 02:56:59 +01:00
|
|
|
}
|
2018-05-05 01:25:26 +01:00
|
|
|
}
|
|
|
|
|
2018-10-30 00:29:35 +00:00
|
|
|
Result ShellService::ClearProcessNotificationFlag(u64 pid) {
|
2019-03-26 00:12:19 +00:00
|
|
|
std::scoped_lock<ProcessList &> lk(Registration::GetProcessList());
|
2019-05-28 05:52:28 +01:00
|
|
|
|
2018-10-30 00:29:35 +00:00
|
|
|
auto proc = Registration::GetProcess(pid);
|
2018-05-05 02:56:59 +01:00
|
|
|
if (proc != NULL) {
|
2018-07-27 09:56:24 +01:00
|
|
|
proc->flags &= ~PROCESSFLAGS_CRASHED;
|
2019-03-29 05:39:39 +00:00
|
|
|
return ResultSuccess;
|
2018-05-05 02:56:59 +01:00
|
|
|
} else {
|
2019-03-28 22:16:36 +00:00
|
|
|
return ResultPmProcessNotFound;
|
2018-05-05 02:56:59 +01:00
|
|
|
}
|
2018-05-05 01:25:26 +01:00
|
|
|
}
|
|
|
|
|
2018-10-30 00:29:35 +00:00
|
|
|
void ShellService::NotifyBootFinished() {
|
2018-05-05 01:25:26 +01:00
|
|
|
if (!g_has_boot_finished) {
|
|
|
|
g_has_boot_finished = true;
|
2019-05-28 05:52:28 +01:00
|
|
|
Registration::SignalBootFinished();
|
2018-07-27 10:23:53 +01:00
|
|
|
EmbeddedBoot2::Main();
|
2018-05-05 01:25:26 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-10-30 00:29:35 +00:00
|
|
|
Result ShellService::GetApplicationProcessId(Out<u64> pid) {
|
2019-03-26 00:12:19 +00:00
|
|
|
std::scoped_lock<ProcessList &> lk(Registration::GetProcessList());
|
2019-05-28 05:52:28 +01:00
|
|
|
|
2018-06-15 07:47:07 +01:00
|
|
|
std::shared_ptr<Registration::Process> app_proc;
|
2018-05-05 02:56:59 +01:00
|
|
|
if (Registration::HasApplicationProcess(&app_proc)) {
|
2018-10-30 00:29:35 +00:00
|
|
|
pid.SetValue(app_proc->pid);
|
2019-03-29 05:39:39 +00:00
|
|
|
return ResultSuccess;
|
2018-05-05 02:56:59 +01:00
|
|
|
}
|
2019-03-28 22:16:36 +00:00
|
|
|
return ResultPmProcessNotFound;
|
2018-05-05 01:25:26 +01:00
|
|
|
}
|
|
|
|
|
2018-10-30 00:29:35 +00:00
|
|
|
Result ShellService::BoostSystemMemoryResourceLimit(u64 sysmem_size) {
|
|
|
|
return ResourceLimitUtils::BoostSystemMemoryResourceLimit(sysmem_size);
|
2018-05-05 01:25:26 +01:00
|
|
|
}
|
2019-01-31 11:32:47 +00:00
|
|
|
|
|
|
|
Result ShellService::BoostSystemThreadsResourceLimit() {
|
|
|
|
/* Starting in 7.0.0, Nintendo reduces the number of system threads from 0x260 to 0x60, */
|
|
|
|
/* Until this command is called to double that amount to 0xC0. */
|
|
|
|
/* We will simply not reduce the number of system threads available for no reason. */
|
2019-03-29 05:39:39 +00:00
|
|
|
return ResultSuccess;
|
2019-01-31 11:32:47 +00:00
|
|
|
}
|
2019-04-18 08:18:53 +01:00
|
|
|
|
|
|
|
|
2019-05-28 05:52:28 +01:00
|
|
|
void ShellService::GetBootFinishedEvent(Out<CopiedHandle> event) {
|
|
|
|
/* In 8.0.0, Nintendo added this command, which signals that the boot sysmodule has finished. */
|
|
|
|
/* Nintendo only signals it in safe mode FIRM, and this function aborts on normal FIRM. */
|
|
|
|
/* We will signal it always, but only allow this function to succeed on safe mode. */
|
|
|
|
{
|
|
|
|
u64 is_recovery_boot = 0;
|
2019-06-20 10:00:59 +01:00
|
|
|
R_ASSERT(SmcGetConfig(SplConfigItem_IsRecoveryBoot, &is_recovery_boot));
|
|
|
|
if (!is_recovery_boot) {
|
2019-05-28 05:52:28 +01:00
|
|
|
std::abort();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
event.SetValue(Registration::GetBootFinishedEventHandle());
|
2019-04-18 08:18:53 +01:00
|
|
|
}
|