This commit is contained in:
Tobias 2024-03-01 08:58:35 +01:00 committed by GitHub
commit 6d0b92ab0b
No known key found for this signature in database
GPG Key ID: B5690EEEBB952194
14 changed files with 365 additions and 259 deletions

View File

@ -930,8 +930,17 @@ add_library(core STATIC
hle/service/pctl/pctl_types.h
hle/service/pcv/pcv.cpp
hle/service/pcv/pcv.h
hle/service/pm/boot_mode_service.cpp
hle/service/pm/boot_mode_service.h
hle/service/pm/debug_monitor_service.cpp
hle/service/pm/debug_monitor_service.h
hle/service/pm/information_service.cpp
hle/service/pm/information_service.h
hle/service/pm/pm.cpp
hle/service/pm/pm.h
hle/service/pm/pm_types.h
hle/service/pm/shell_service.cpp
hle/service/pm/shell_service.h
hle/service/prepo/prepo.cpp
hle/service/prepo/prepo.h
hle/service/psc/ovln/ovln_types.h

View File

@ -154,9 +154,9 @@ Result ICommonStateGetter::GetPerformanceMode(Out<APM::PerformanceMode> out_perf
R_SUCCEED();
}
Result ICommonStateGetter::GetBootMode(Out<PM::SystemBootMode> out_boot_mode) {
Result ICommonStateGetter::GetBootMode(Out<PM::BootMode> out_boot_mode) {
LOG_DEBUG(Service_AM, "called");
*out_boot_mode = Service::PM::SystemBootMode::Normal;
*out_boot_mode = Service::PM::BootMode::Normal;
R_SUCCEED();
}

View File

@ -6,7 +6,7 @@
#include "core/hle/service/am/am_types.h"
#include "core/hle/service/apm/apm_controller.h"
#include "core/hle/service/cmif_types.h"
#include "core/hle/service/pm/pm.h"
#include "core/hle/service/pm/pm_types.h"
#include "core/hle/service/service.h"
#include "core/hle/service/set/settings_types.h"
@ -37,7 +37,7 @@ private:
Result GetDefaultDisplayResolutionChangeEvent(OutCopyHandle<Kernel::KReadableEvent> out_event);
Result GetOperationMode(Out<OperationMode> out_operation_mode);
Result GetPerformanceMode(Out<APM::PerformanceMode> out_performance_mode);
Result GetBootMode(Out<PM::SystemBootMode> out_boot_mode);
Result GetBootMode(Out<PM::BootMode> out_boot_mode);
Result IsVrModeEnabled(Out<bool> out_is_vr_mode_enabled);
Result SetVrModeEnabled(bool is_vr_mode_enabled);
Result SetLcdBacklighOffEnabled(bool is_lcd_backlight_off_enabled);

View File

@ -0,0 +1,33 @@
// SPDX-FileCopyrightText: Copyright 2024 yuzu Emulator Project
// SPDX-License-Identifier: GPL-2.0-or-later
#include "core/hle/service/cmif_serialization.h"
#include "core/hle/service/pm/boot_mode_service.h"
namespace Service::PM {
BootModeService::BootModeService(Core::System& system_) : ServiceFramework{system_, "pm:bm"} {
static const FunctionInfo functions[] = {
{0, C<&BootModeService::GetBootMode>, "GetBootMode"},
{1, C<&BootModeService::SetMaintenanceBoot>, "SetMaintenanceBoot"},
};
RegisterHandlers(functions);
}
Result BootModeService::GetBootMode(Out<u32> out_boot_mode) {
LOG_DEBUG(Service_PM, "called");
*out_boot_mode = static_cast<u32>(boot_mode);
R_SUCCEED();
}
Result BootModeService::SetMaintenanceBoot() {
LOG_DEBUG(Service_PM, "called");
boot_mode = BootMode::Maintenance;
R_SUCCEED();
}
} // namespace Service::PM

View File

@ -0,0 +1,23 @@
// SPDX-FileCopyrightText: Copyright 2024 yuzu Emulator Project
// SPDX-License-Identifier: GPL-2.0-or-later
#pragma once
#include "core/hle/service/cmif_types.h"
#include "core/hle/service/pm/pm_types.h"
#include "core/hle/service/service.h"
namespace Service::PM {
class BootModeService final : public ServiceFramework<BootModeService> {
public:
explicit BootModeService(Core::System& system_);
private:
Result GetBootMode(Out<u32> out_boot_mode);
Result SetMaintenanceBoot();
BootMode boot_mode = BootMode::Normal;
};
} // namespace Service::PM

View File

@ -0,0 +1,74 @@
// SPDX-FileCopyrightText: Copyright 2024 yuzu Emulator Project
// SPDX-License-Identifier: GPL-2.0-or-later
#include "core/hle/service/cmif_serialization.h"
#include "core/hle/service/pm/debug_monitor_service.h"
namespace Service::PM {
DebugMonitorService::DebugMonitorService(Core::System& system_)
: ServiceFramework{system_, "pm:dmnt"} {
// clang-format off
static const FunctionInfo functions[] = {
{0, nullptr, "GetExceptionProcessIdList"},
{1, nullptr, "StartProcess"},
{2, C<&DebugMonitorService::GetProcessId>, "GetProcessId"},
{3, nullptr, "HookToCreateProcess"},
{4, C<&DebugMonitorService::GetApplicationProcessId>, "GetApplicationProcessId"},
{5, nullptr, "HookToCreateApplicationProcess"},
{6, nullptr, "ClearHook"},
{65000, C<&DebugMonitorService::AtmosphereGetProcessInfo>, "AtmosphereGetProcessInfo"},
{65001, nullptr, "AtmosphereGetCurrentLimitInfo"},
};
// clang-format on
RegisterHandlers(functions);
}
Result DebugMonitorService::GetProcessId(Out<ProcessId> out_process_id, u64 program_id) {
LOG_DEBUG(Service_PM, "called, program_id={:016X}", program_id);
auto list = kernel.GetProcessList();
auto process =
SearchProcessList(list, [program_id](auto& p) { return p->GetProgramId() == program_id; });
R_UNLESS(!process.IsNull(), ResultProcessNotFound);
*out_process_id = ProcessId(process->GetProcessId());
R_SUCCEED();
}
Result DebugMonitorService::GetApplicationProcessId(Out<ProcessId> out_process_id) {
LOG_DEBUG(Service_PM, "called");
auto list = kernel.GetProcessList();
R_RETURN(GetApplicationPidGeneric(out_process_id, list));
}
Result DebugMonitorService::AtmosphereGetProcessInfo(
OutCopyHandle<Kernel::KProcess> out_process_handle, Out<ProgramLocation> out_location,
Out<OverrideStatus> out_status, ProcessId process_id) {
// https://github.com/Atmosphere-NX/Atmosphere/blob/master/stratosphere/pm/source/impl/pm_process_manager.cpp#L614
// This implementation is incomplete; only a handle to the process is returned.
const auto pid = process_id.pid;
LOG_WARNING(Service_PM, "(Partial Implementation) called, pid={:016X}", pid);
auto list = kernel.GetProcessList();
auto process = SearchProcessList(list, [pid](auto& p) { return p->GetProcessId() == pid; });
R_UNLESS(!process.IsNull(), ResultProcessNotFound);
OverrideStatus override_status{};
ProgramLocation program_location{
.program_id = process->GetProgramId(),
.storage_id = 0,
};
*out_process_handle = process.GetPointerUnsafe();
*out_location = program_location;
*out_status = override_status;
R_SUCCEED();
}
} // namespace Service::PM

View File

@ -0,0 +1,23 @@
// SPDX-FileCopyrightText: Copyright 2024 yuzu Emulator Project
// SPDX-License-Identifier: GPL-2.0-or-later
#pragma once
#include "core/hle/service/cmif_types.h"
#include "core/hle/service/pm/pm_types.h"
#include "core/hle/service/service.h"
namespace Service::PM {
class DebugMonitorService final : public ServiceFramework<DebugMonitorService> {
public:
explicit DebugMonitorService(Core::System& system_);
private:
Result GetProcessId(Out<ProcessId> out_process_id, u64 program_id);
Result GetApplicationProcessId(Out<ProcessId> out_process_id);
Result AtmosphereGetProcessInfo(OutCopyHandle<Kernel::KProcess> out_process_handle,
Out<ProgramLocation> out_location,
Out<OverrideStatus> out_status, ProcessId process_id);
};
} // namespace Service::PM

View File

@ -0,0 +1,49 @@
// SPDX-FileCopyrightText: Copyright 2024 yuzu Emulator Project
// SPDX-License-Identifier: GPL-2.0-or-later
#include "core/hle/service/cmif_serialization.h"
#include "core/hle/service/pm/information_service.h"
#include "core/hle/service/pm/pm_types.h"
namespace Service::PM {
InformationService::InformationService(Core::System& system_)
: ServiceFramework{system_, "pm:info"} {
static const FunctionInfo functions[] = {
{0, C<&InformationService::GetProgramId>, "GetProgramId"},
{65000, C<&InformationService::AtmosphereGetProcessId>, "AtmosphereGetProcessId"},
{65001, nullptr, "AtmosphereHasLaunchedProgram"},
{65002, nullptr, "AtmosphereGetProcessInfo"},
};
RegisterHandlers(functions);
}
Result InformationService::GetProgramId(Out<u64> out_program_id, u64 process_id) {
LOG_DEBUG(Service_PM, "called, process_id={:016X}", process_id);
auto list = kernel.GetProcessList();
auto process =
SearchProcessList(list, [process_id](auto& p) { return p->GetProcessId() == process_id; });
R_UNLESS(!process.IsNull(), ResultProcessNotFound);
*out_program_id = process->GetProgramId();
R_SUCCEED();
}
Result InformationService::AtmosphereGetProcessId(Out<ProcessId> out_process_id, u64 program_id) {
LOG_DEBUG(Service_PM, "called, program_id={:016X}", program_id);
auto list = system.Kernel().GetProcessList();
auto process =
SearchProcessList(list, [program_id](auto& p) { return p->GetProgramId() == program_id; });
R_UNLESS(!process.IsNull(), ResultProcessNotFound);
*out_process_id = ProcessId(process->GetProcessId());
R_SUCCEED();
}
} // namespace Service::PM

View File

@ -0,0 +1,20 @@
// SPDX-FileCopyrightText: Copyright 2024 yuzu Emulator Project
// SPDX-License-Identifier: GPL-2.0-or-later
#pragma once
#include "core/hle/service/cmif_types.h"
#include "core/hle/service/service.h"
namespace Service::PM {
class InformationService final : public ServiceFramework<InformationService> {
public:
explicit InformationService(Core::System& system_);
private:
Result GetProgramId(Out<u64> out_program_id, u64 process_id);
Result AtmosphereGetProcessId(Out<ProcessId> out_process_id, u64 program_id);
};
} // namespace Service::PM

View File

@ -2,264 +2,22 @@
// SPDX-License-Identifier: GPL-2.0-or-later
#include "core/core.h"
#include "core/hle/kernel/k_process.h"
#include "core/hle/kernel/kernel.h"
#include "core/hle/service/ipc_helpers.h"
#include "core/hle/service/pm/boot_mode_service.h"
#include "core/hle/service/pm/debug_monitor_service.h"
#include "core/hle/service/pm/information_service.h"
#include "core/hle/service/pm/pm.h"
#include "core/hle/service/pm/shell_service.h"
#include "core/hle/service/server_manager.h"
#include "core/hle/service/service.h"
namespace Service::PM {
namespace {
constexpr Result ResultProcessNotFound{ErrorModule::PM, 1};
[[maybe_unused]] constexpr Result ResultAlreadyStarted{ErrorModule::PM, 2};
[[maybe_unused]] constexpr Result ResultNotTerminated{ErrorModule::PM, 3};
[[maybe_unused]] constexpr Result ResultDebugHookInUse{ErrorModule::PM, 4};
[[maybe_unused]] constexpr Result ResultApplicationRunning{ErrorModule::PM, 5};
[[maybe_unused]] constexpr Result ResultInvalidSize{ErrorModule::PM, 6};
constexpr u64 NO_PROCESS_FOUND_PID{0};
using ProcessList = std::list<Kernel::KScopedAutoObject<Kernel::KProcess>>;
template <typename F>
Kernel::KScopedAutoObject<Kernel::KProcess> SearchProcessList(ProcessList& process_list,
F&& predicate) {
const auto iter = std::find_if(process_list.begin(), process_list.end(), predicate);
if (iter == process_list.end()) {
return nullptr;
}
return iter->GetPointerUnsafe();
}
void GetApplicationPidGeneric(HLERequestContext& ctx, ProcessList& process_list) {
auto process = SearchProcessList(process_list, [](auto& p) { return p->IsApplication(); });
IPC::ResponseBuilder rb{ctx, 4};
rb.Push(ResultSuccess);
rb.Push(process.IsNull() ? NO_PROCESS_FOUND_PID : process->GetProcessId());
}
} // Anonymous namespace
class BootMode final : public ServiceFramework<BootMode> {
public:
explicit BootMode(Core::System& system_) : ServiceFramework{system_, "pm:bm"} {
static const FunctionInfo functions[] = {
{0, &BootMode::GetBootMode, "GetBootMode"},
{1, &BootMode::SetMaintenanceBoot, "SetMaintenanceBoot"},
};
RegisterHandlers(functions);
}
private:
void GetBootMode(HLERequestContext& ctx) {
LOG_DEBUG(Service_PM, "called");
IPC::ResponseBuilder rb{ctx, 3};
rb.Push(ResultSuccess);
rb.PushEnum(boot_mode);
}
void SetMaintenanceBoot(HLERequestContext& ctx) {
LOG_DEBUG(Service_PM, "called");
boot_mode = SystemBootMode::Maintenance;
IPC::ResponseBuilder rb{ctx, 2};
rb.Push(ResultSuccess);
}
SystemBootMode boot_mode = SystemBootMode::Normal;
};
class DebugMonitor final : public ServiceFramework<DebugMonitor> {
public:
explicit DebugMonitor(Core::System& system_) : ServiceFramework{system_, "pm:dmnt"} {
// clang-format off
static const FunctionInfo functions[] = {
{0, nullptr, "GetJitDebugProcessIdList"},
{1, nullptr, "StartProcess"},
{2, &DebugMonitor::GetProcessId, "GetProcessId"},
{3, nullptr, "HookToCreateProcess"},
{4, &DebugMonitor::GetApplicationProcessId, "GetApplicationProcessId"},
{5, nullptr, "HookToCreateApplicationProgress"},
{6, nullptr, "ClearHook"},
{65000, &DebugMonitor::AtmosphereGetProcessInfo, "AtmosphereGetProcessInfo"},
{65001, nullptr, "AtmosphereGetCurrentLimitInfo"},
};
// clang-format on
RegisterHandlers(functions);
}
private:
void GetProcessId(HLERequestContext& ctx) {
IPC::RequestParser rp{ctx};
const auto program_id = rp.PopRaw<u64>();
LOG_DEBUG(Service_PM, "called, program_id={:016X}", program_id);
auto list = kernel.GetProcessList();
auto process = SearchProcessList(
list, [program_id](auto& p) { return p->GetProgramId() == program_id; });
if (process.IsNull()) {
IPC::ResponseBuilder rb{ctx, 2};
rb.Push(ResultProcessNotFound);
return;
}
IPC::ResponseBuilder rb{ctx, 4};
rb.Push(ResultSuccess);
rb.Push(process->GetProcessId());
}
void GetApplicationProcessId(HLERequestContext& ctx) {
LOG_DEBUG(Service_PM, "called");
auto list = kernel.GetProcessList();
GetApplicationPidGeneric(ctx, list);
}
void AtmosphereGetProcessInfo(HLERequestContext& ctx) {
// https://github.com/Atmosphere-NX/Atmosphere/blob/master/stratosphere/pm/source/impl/pm_process_manager.cpp#L614
// This implementation is incomplete; only a handle to the process is returned.
IPC::RequestParser rp{ctx};
const auto pid = rp.PopRaw<u64>();
LOG_WARNING(Service_PM, "(Partial Implementation) called, pid={:016X}", pid);
auto list = kernel.GetProcessList();
auto process = SearchProcessList(list, [pid](auto& p) { return p->GetProcessId() == pid; });
if (process.IsNull()) {
IPC::ResponseBuilder rb{ctx, 2};
rb.Push(ResultProcessNotFound);
return;
}
struct ProgramLocation {
u64 program_id;
u8 storage_id;
};
static_assert(sizeof(ProgramLocation) == 0x10, "ProgramLocation has an invalid size");
struct OverrideStatus {
u64 keys_held;
u64 flags;
};
static_assert(sizeof(OverrideStatus) == 0x10, "OverrideStatus has an invalid size");
OverrideStatus override_status{};
ProgramLocation program_location{
.program_id = process->GetProgramId(),
.storage_id = 0,
};
IPC::ResponseBuilder rb{ctx, 10, 1};
rb.Push(ResultSuccess);
rb.PushCopyObjects(*process);
rb.PushRaw(program_location);
rb.PushRaw(override_status);
}
};
class Info final : public ServiceFramework<Info> {
public:
explicit Info(Core::System& system_) : ServiceFramework{system_, "pm:info"} {
static const FunctionInfo functions[] = {
{0, &Info::GetProgramId, "GetProgramId"},
{65000, &Info::AtmosphereGetProcessId, "AtmosphereGetProcessId"},
{65001, nullptr, "AtmosphereHasLaunchedProgram"},
{65002, nullptr, "AtmosphereGetProcessInfo"},
};
RegisterHandlers(functions);
}
private:
void GetProgramId(HLERequestContext& ctx) {
IPC::RequestParser rp{ctx};
const auto process_id = rp.PopRaw<u64>();
LOG_DEBUG(Service_PM, "called, process_id={:016X}", process_id);
auto list = kernel.GetProcessList();
auto process = SearchProcessList(
list, [process_id](auto& p) { return p->GetProcessId() == process_id; });
if (process.IsNull()) {
IPC::ResponseBuilder rb{ctx, 2};
rb.Push(ResultProcessNotFound);
return;
}
IPC::ResponseBuilder rb{ctx, 4};
rb.Push(ResultSuccess);
rb.Push(process->GetProgramId());
}
void AtmosphereGetProcessId(HLERequestContext& ctx) {
IPC::RequestParser rp{ctx};
const auto program_id = rp.PopRaw<u64>();
LOG_DEBUG(Service_PM, "called, program_id={:016X}", program_id);
auto list = system.Kernel().GetProcessList();
auto process = SearchProcessList(
list, [program_id](auto& p) { return p->GetProgramId() == program_id; });
if (process.IsNull()) {
IPC::ResponseBuilder rb{ctx, 2};
rb.Push(ResultProcessNotFound);
return;
}
IPC::ResponseBuilder rb{ctx, 4};
rb.Push(ResultSuccess);
rb.Push(process->GetProcessId());
}
};
class Shell final : public ServiceFramework<Shell> {
public:
explicit Shell(Core::System& system_) : ServiceFramework{system_, "pm:shell"} {
// clang-format off
static const FunctionInfo functions[] = {
{0, nullptr, "LaunchProgram"},
{1, nullptr, "TerminateProcess"},
{2, nullptr, "TerminateProgram"},
{3, nullptr, "GetProcessEventHandle"},
{4, nullptr, "GetProcessEventInfo"},
{5, nullptr, "NotifyBootFinished"},
{6, &Shell::GetApplicationProcessIdForShell, "GetApplicationProcessIdForShell"},
{7, nullptr, "BoostSystemMemoryResourceLimit"},
{8, nullptr, "BoostApplicationThreadResourceLimit"},
{9, nullptr, "GetBootFinishedEventHandle"},
};
// clang-format on
RegisterHandlers(functions);
}
private:
void GetApplicationProcessIdForShell(HLERequestContext& ctx) {
LOG_DEBUG(Service_PM, "called");
auto list = kernel.GetProcessList();
GetApplicationPidGeneric(ctx, list);
}
};
void LoopProcess(Core::System& system) {
auto server_manager = std::make_unique<ServerManager>(system);
server_manager->RegisterNamedService("pm:bm", std::make_shared<BootMode>(system));
server_manager->RegisterNamedService("pm:dmnt", std::make_shared<DebugMonitor>(system));
server_manager->RegisterNamedService("pm:info", std::make_shared<Info>(system));
server_manager->RegisterNamedService("pm:shell", std::make_shared<Shell>(system));
server_manager->RegisterNamedService("pm:bm", std::make_shared<BootModeService>(system));
server_manager->RegisterNamedService("pm:dmnt", std::make_shared<DebugMonitorService>(system));
server_manager->RegisterNamedService("pm:info", std::make_shared<InformationService>(system));
server_manager->RegisterNamedService("pm:shell", std::make_shared<ShellService>(system));
ServerManager::RunServer(std::move(server_manager));
}

View File

@ -9,11 +9,6 @@ class System;
namespace Service::PM {
enum class SystemBootMode {
Normal,
Maintenance,
};
void LoopProcess(Core::System& system);
} // namespace Service::PM

View File

@ -0,0 +1,65 @@
// SPDX-FileCopyrightText: Copyright 2024 yuzu Emulator Project
// SPDX-License-Identifier: GPL-2.0-or-later
#pragma once
#include "common/common_types.h"
#include "core/hle/kernel/k_process.h"
#include "core/hle/kernel/kernel.h"
#include "core/hle/result.h"
#include "core/hle/service/cmif_types.h"
namespace Service::PM {
constexpr Result ResultProcessNotFound{ErrorModule::PM, 1};
constexpr Result ResultAlreadyStarted{ErrorModule::PM, 2};
constexpr Result ResultNotTerminated{ErrorModule::PM, 3};
constexpr Result ResultDebugHookInUse{ErrorModule::PM, 4};
constexpr Result ResultApplicationRunning{ErrorModule::PM, 5};
constexpr Result ResultInvalidSize{ErrorModule::PM, 6};
constexpr u64 NO_PROCESS_FOUND_PID{0};
enum class BootMode {
Normal = 0,
Maintenance = 1,
SafeMode = 2,
};
struct ProgramLocation {
u64 program_id;
u8 storage_id;
};
static_assert(sizeof(ProgramLocation) == 0x10, "ProgramLocation has an invalid size");
struct OverrideStatus {
u64 keys_held;
u64 flags;
};
static_assert(sizeof(OverrideStatus) == 0x10, "OverrideStatus has an invalid size");
using ProcessList = std::list<Kernel::KScopedAutoObject<Kernel::KProcess>>;
template <typename F>
static inline Kernel::KScopedAutoObject<Kernel::KProcess> SearchProcessList(
ProcessList& process_list, F&& predicate) {
const auto iter = std::find_if(process_list.begin(), process_list.end(), predicate);
if (iter == process_list.end()) {
return nullptr;
}
return iter->GetPointerUnsafe();
}
static inline Result GetApplicationPidGeneric(Out<ProcessId> out_process_id,
ProcessList& process_list) {
auto process = SearchProcessList(process_list, [](auto& p) { return p->IsApplication(); });
*out_process_id =
process.IsNull() ? ProcessId(NO_PROCESS_FOUND_PID) : ProcessId(process->GetProcessId());
R_SUCCEED();
}
} // namespace Service::PM

View File

@ -0,0 +1,37 @@
// SPDX-FileCopyrightText: Copyright 2024 yuzu Emulator Project
// SPDX-License-Identifier: GPL-2.0-or-later
#include "core/hle/service/cmif_serialization.h"
#include "core/hle/service/pm/pm_types.h"
#include "core/hle/service/pm/shell_service.h"
namespace Service::PM {
ShellService::ShellService(Core::System& system_) : ServiceFramework{system_, "pm:shell"} {
// clang-format off
static const FunctionInfo functions[] = {
{0, nullptr, "LaunchProgram"},
{1, nullptr, "TerminateProcess"},
{2, nullptr, "TerminateProgram"},
{3, nullptr, "GetProcessEventHandle"},
{4, nullptr, "GetProcessEventInfo"},
{5, nullptr, "NotifyBootFinished"},
{6, C<&ShellService::GetApplicationProcessIdForShell>, "GetApplicationProcessIdForShell"},
{7, nullptr, "BoostSystemMemoryResourceLimit"},
{8, nullptr, "BoostApplicationThreadResourceLimit"},
{9, nullptr, "GetBootFinishedEventHandle"},
};
// clang-format on
RegisterHandlers(functions);
}
Result ShellService::GetApplicationProcessIdForShell(Out<ProcessId> out_process_id) {
LOG_DEBUG(Service_PM, "called");
auto list = kernel.GetProcessList();
R_RETURN(GetApplicationPidGeneric(out_process_id, list));
}
} // namespace Service::PM

View File

@ -0,0 +1,20 @@
// SPDX-FileCopyrightText: Copyright 2024 yuzu Emulator Project
// SPDX-License-Identifier: GPL-2.0-or-later
#pragma once
#include "core/hle/service/cmif_types.h"
#include "core/hle/service/ipc_helpers.h"
#include "core/hle/service/service.h"
namespace Service::PM {
class ShellService final : public ServiceFramework<ShellService> {
public:
explicit ShellService(Core::System& system_);
private:
Result GetApplicationProcessIdForShell(Out<ProcessId> out_process_id);
};
} // namespace Service::PM