/* * Copyright (c) 2018-2020 Atmosphère-NX * * 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/>. */ #include <mesosphere.hpp> namespace ams::kern::svc { /* ============================= Common ============================= */ namespace { constexpr bool IsValidThreadActivity(ams::svc::ThreadActivity thread_activity) { switch (thread_activity) { case ams::svc::ThreadActivity_Runnable: case ams::svc::ThreadActivity_Paused: return true; default: return false; } } constexpr bool IsValidProcessActivity(ams::svc::ProcessActivity process_activity) { switch (process_activity) { case ams::svc::ProcessActivity_Runnable: case ams::svc::ProcessActivity_Paused: return true; default: return false; } } Result SetThreadActivity(ams::svc::Handle thread_handle, ams::svc::ThreadActivity thread_activity) { /* Validate the activity. */ R_UNLESS(IsValidThreadActivity(thread_activity), svc::ResultInvalidEnumValue()); /* Get the thread from its handle. */ KScopedAutoObject thread = GetCurrentProcess().GetHandleTable().GetObject<KThread>(thread_handle); R_UNLESS(thread.IsNotNull(), svc::ResultInvalidHandle()); /* Check that the activity is being set on a non-current thread for the current process. */ R_UNLESS(thread->GetOwnerProcess() == GetCurrentProcessPointer(), svc::ResultInvalidHandle()); R_UNLESS(thread.GetPointerUnsafe() != GetCurrentThreadPointer(), svc::ResultBusy()); /* Set the activity. */ R_TRY(thread->SetActivity(thread_activity)); return ResultSuccess(); } Result SetProcessActivity(ams::svc::Handle process_handle, ams::svc::ProcessActivity process_activity) { /* Validate the activity. */ R_UNLESS(IsValidProcessActivity(process_activity), svc::ResultInvalidEnumValue()); /* Get the process from its handle. */ KScopedAutoObject process = GetCurrentProcess().GetHandleTable().GetObject<KProcess>(process_handle); R_UNLESS(process.IsNotNull(), svc::ResultInvalidHandle()); /* Check that the activity isn't being set on the current process. */ R_UNLESS(process.GetPointerUnsafe() != GetCurrentProcessPointer(), svc::ResultBusy()); /* Set the activity. */ R_TRY(process->SetActivity(process_activity)); return ResultSuccess(); } } /* ============================= 64 ABI ============================= */ Result SetThreadActivity64(ams::svc::Handle thread_handle, ams::svc::ThreadActivity thread_activity) { return SetThreadActivity(thread_handle, thread_activity); } Result SetProcessActivity64(ams::svc::Handle process_handle, ams::svc::ProcessActivity process_activity) { return SetProcessActivity(process_handle, process_activity); } /* ============================= 64From32 ABI ============================= */ Result SetThreadActivity64From32(ams::svc::Handle thread_handle, ams::svc::ThreadActivity thread_activity) { return SetThreadActivity(thread_handle, thread_activity); } Result SetProcessActivity64From32(ams::svc::Handle process_handle, ams::svc::ProcessActivity process_activity) { return SetProcessActivity(process_handle, process_activity); } }