|
|
|
@ -3,16 +3,30 @@
|
|
|
|
|
// Refer to the license.txt file included.
|
|
|
|
|
|
|
|
|
|
#include "common/logging/log.h"
|
|
|
|
|
#include "core/core.h"
|
|
|
|
|
#include "core/file_sys/control_metadata.h"
|
|
|
|
|
#include "core/file_sys/patch_manager.h"
|
|
|
|
|
#include "core/hle/ipc_helpers.h"
|
|
|
|
|
#include "core/hle/kernel/process.h"
|
|
|
|
|
#include "core/hle/service/pctl/module.h"
|
|
|
|
|
#include "core/hle/service/pctl/pctl.h"
|
|
|
|
|
|
|
|
|
|
namespace Service::PCTL {
|
|
|
|
|
|
|
|
|
|
namespace Error {
|
|
|
|
|
|
|
|
|
|
constexpr ResultCode ResultNoFreeCommunication{ErrorModule::PCTL, 101};
|
|
|
|
|
constexpr ResultCode ResultStereoVisionRestricted{ErrorModule::PCTL, 104};
|
|
|
|
|
constexpr ResultCode ResultNoCapability{ErrorModule::PCTL, 131};
|
|
|
|
|
constexpr ResultCode ResultNoRestrictionEnabled{ErrorModule::PCTL, 181};
|
|
|
|
|
|
|
|
|
|
} // namespace Error
|
|
|
|
|
|
|
|
|
|
class IParentalControlService final : public ServiceFramework<IParentalControlService> {
|
|
|
|
|
public:
|
|
|
|
|
explicit IParentalControlService(Core::System& system_)
|
|
|
|
|
: ServiceFramework{system_, "IParentalControlService"} {
|
|
|
|
|
explicit IParentalControlService(Core::System& system_, Capability capability)
|
|
|
|
|
: ServiceFramework{system_, "IParentalControlService"}, system(system_),
|
|
|
|
|
capability(capability) {
|
|
|
|
|
// clang-format off
|
|
|
|
|
static const FunctionInfo functions[] = {
|
|
|
|
|
{1, &IParentalControlService::Initialize, "Initialize"},
|
|
|
|
@ -28,13 +42,13 @@ public:
|
|
|
|
|
{1010, nullptr, "IsRestrictedSystemSettingsEntered"},
|
|
|
|
|
{1011, nullptr, "RevertRestrictedSystemSettingsEntered"},
|
|
|
|
|
{1012, nullptr, "GetRestrictedFeatures"},
|
|
|
|
|
{1013, nullptr, "ConfirmStereoVisionPermission"},
|
|
|
|
|
{1013, &IParentalControlService::ConfirmStereoVisionPermission, "ConfirmStereoVisionPermission"},
|
|
|
|
|
{1014, nullptr, "ConfirmPlayableApplicationVideoOld"},
|
|
|
|
|
{1015, nullptr, "ConfirmPlayableApplicationVideo"},
|
|
|
|
|
{1016, nullptr, "ConfirmShowNewsPermission"},
|
|
|
|
|
{1017, nullptr, "EndFreeCommunication"},
|
|
|
|
|
{1018, nullptr, "IsFreeCommunicationAvailable"},
|
|
|
|
|
{1031, nullptr, "IsRestrictionEnabled"},
|
|
|
|
|
{1018, &IParentalControlService::IsFreeCommunicationAvailable, "IsFreeCommunicationAvailable"},
|
|
|
|
|
{1031, &IParentalControlService::IsRestrictionEnabled, "IsRestrictionEnabled"},
|
|
|
|
|
{1032, nullptr, "GetSafetyLevel"},
|
|
|
|
|
{1033, nullptr, "SetSafetyLevel"},
|
|
|
|
|
{1034, nullptr, "GetSafetyLevelSettings"},
|
|
|
|
@ -122,62 +136,235 @@ public:
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
private:
|
|
|
|
|
void Initialize(Kernel::HLERequestContext& ctx) {
|
|
|
|
|
LOG_WARNING(Service_PCTL, "(STUBBED) called");
|
|
|
|
|
bool CheckFreeCommunicationPermissionImpl() const {
|
|
|
|
|
if (states.temporary_unlocked) {
|
|
|
|
|
return true;
|
|
|
|
|
}
|
|
|
|
|
if ((states.application_info.parental_control_flag & 1) == 0) {
|
|
|
|
|
return true;
|
|
|
|
|
}
|
|
|
|
|
if (pin_code[0] == '\0') {
|
|
|
|
|
return true;
|
|
|
|
|
}
|
|
|
|
|
if (!settings.is_free_communication_default_on) {
|
|
|
|
|
return true;
|
|
|
|
|
}
|
|
|
|
|
// TODO(ogniK): Check for blacklisted/exempted applications. Return false can happen here
|
|
|
|
|
// but as we don't have multiproceses support yet, we can just assume our application is
|
|
|
|
|
// valid for the time being
|
|
|
|
|
return true;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
bool ConfirmStereoVisionPermissionImpl() const {
|
|
|
|
|
if (states.temporary_unlocked) {
|
|
|
|
|
return true;
|
|
|
|
|
}
|
|
|
|
|
if (pin_code[0] == '\0') {
|
|
|
|
|
return true;
|
|
|
|
|
}
|
|
|
|
|
if (!settings.is_stero_vision_restricted) {
|
|
|
|
|
return false;
|
|
|
|
|
}
|
|
|
|
|
return true;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void SetStereoVisionRestrictionImpl(bool is_restricted) {
|
|
|
|
|
if (settings.disabled) {
|
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (pin_code[0] == '\0') {
|
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
settings.is_stero_vision_restricted = is_restricted;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void Initialize(Kernel::HLERequestContext& ctx) {
|
|
|
|
|
LOG_DEBUG(Service_PCTL, "called");
|
|
|
|
|
IPC::ResponseBuilder rb{ctx, 2};
|
|
|
|
|
|
|
|
|
|
if (False(capability & (Capability::Application | Capability::System))) {
|
|
|
|
|
LOG_ERROR(Service_PCTL, "Invalid capability! capability={:X}", capability);
|
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// TODO(ogniK): Recovery flag initialization for pctl:r
|
|
|
|
|
|
|
|
|
|
const auto tid = system.CurrentProcess()->GetTitleID();
|
|
|
|
|
if (tid != 0) {
|
|
|
|
|
const FileSys::PatchManager pm{tid, system.GetFileSystemController(),
|
|
|
|
|
system.GetContentProvider()};
|
|
|
|
|
const auto control = pm.GetControlMetadata();
|
|
|
|
|
if (control.first) {
|
|
|
|
|
states.tid_from_event = 0;
|
|
|
|
|
states.launch_time_valid = false;
|
|
|
|
|
states.is_suspended = false;
|
|
|
|
|
states.free_communication = false;
|
|
|
|
|
states.stereo_vision = false;
|
|
|
|
|
states.application_info = ApplicationInfo{
|
|
|
|
|
.tid = tid,
|
|
|
|
|
.age_rating = control.first->GetRatingAge(),
|
|
|
|
|
.parental_control_flag = control.first->GetParentalControlFlag(),
|
|
|
|
|
.capability = capability,
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
if (False(capability & (Capability::System | Capability::Recovery))) {
|
|
|
|
|
// TODO(ogniK): Signal application launch event
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
IPC::ResponseBuilder rb{ctx, 2, 0, 0};
|
|
|
|
|
rb.Push(RESULT_SUCCESS);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void CheckFreeCommunicationPermission(Kernel::HLERequestContext& ctx) {
|
|
|
|
|
LOG_WARNING(Service_PCTL, "(STUBBED) called");
|
|
|
|
|
LOG_DEBUG(Service_PCTL, "called");
|
|
|
|
|
|
|
|
|
|
IPC::ResponseBuilder rb{ctx, 2};
|
|
|
|
|
if (!CheckFreeCommunicationPermissionImpl()) {
|
|
|
|
|
rb.Push(Error::ResultNoFreeCommunication);
|
|
|
|
|
} else {
|
|
|
|
|
rb.Push(RESULT_SUCCESS);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
states.free_communication = true;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void ConfirmStereoVisionPermission(Kernel::HLERequestContext& ctx) {
|
|
|
|
|
LOG_DEBUG(Service_PCTL, "called");
|
|
|
|
|
states.stereo_vision = true;
|
|
|
|
|
|
|
|
|
|
IPC::ResponseBuilder rb{ctx, 2};
|
|
|
|
|
rb.Push(RESULT_SUCCESS);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void ConfirmStereoVisionRestrictionConfigurable(Kernel::HLERequestContext& ctx) {
|
|
|
|
|
void IsFreeCommunicationAvailable(Kernel::HLERequestContext& ctx) {
|
|
|
|
|
LOG_WARNING(Service_PCTL, "(STUBBED) called");
|
|
|
|
|
|
|
|
|
|
IPC::ResponseBuilder rb{ctx, 2};
|
|
|
|
|
if (!CheckFreeCommunicationPermissionImpl()) {
|
|
|
|
|
rb.Push(Error::ResultNoFreeCommunication);
|
|
|
|
|
} else {
|
|
|
|
|
rb.Push(RESULT_SUCCESS);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void IsRestrictionEnabled(Kernel::HLERequestContext& ctx) {
|
|
|
|
|
LOG_DEBUG(Service_PCTL, "called");
|
|
|
|
|
|
|
|
|
|
IPC::ResponseBuilder rb{ctx, 3};
|
|
|
|
|
if (False(capability & (Capability::Status | Capability::Recovery))) {
|
|
|
|
|
LOG_ERROR(Service_PCTL, "Application does not have Status or Recovery capabilities!");
|
|
|
|
|
rb.Push(Error::ResultNoCapability);
|
|
|
|
|
rb.Push(false);
|
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
rb.Push(pin_code[0] != '\0');
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void ConfirmStereoVisionRestrictionConfigurable(Kernel::HLERequestContext& ctx) {
|
|
|
|
|
LOG_DEBUG(Service_PCTL, "called");
|
|
|
|
|
|
|
|
|
|
IPC::ResponseBuilder rb{ctx, 2};
|
|
|
|
|
|
|
|
|
|
if (False(capability & Capability::StereoVision)) {
|
|
|
|
|
LOG_ERROR(Service_PCTL, "Application does not have StereoVision capability!");
|
|
|
|
|
rb.Push(Error::ResultNoCapability);
|
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (pin_code[0] == '\0') {
|
|
|
|
|
rb.Push(Error::ResultNoRestrictionEnabled);
|
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
rb.Push(RESULT_SUCCESS);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void IsStereoVisionPermitted(Kernel::HLERequestContext& ctx) {
|
|
|
|
|
LOG_WARNING(Service_PCTL, "(STUBBED) called");
|
|
|
|
|
LOG_DEBUG(Service_PCTL, "called");
|
|
|
|
|
|
|
|
|
|
IPC::ResponseBuilder rb{ctx, 3};
|
|
|
|
|
rb.Push(RESULT_SUCCESS);
|
|
|
|
|
rb.Push(true);
|
|
|
|
|
if (!ConfirmStereoVisionPermissionImpl()) {
|
|
|
|
|
rb.Push(Error::ResultStereoVisionRestricted);
|
|
|
|
|
rb.Push(false);
|
|
|
|
|
} else {
|
|
|
|
|
rb.Push(RESULT_SUCCESS);
|
|
|
|
|
rb.Push(true);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void SetStereoVisionRestriction(Kernel::HLERequestContext& ctx) {
|
|
|
|
|
IPC::RequestParser rp{ctx};
|
|
|
|
|
const auto can_use = rp.Pop<bool>();
|
|
|
|
|
LOG_WARNING(Service_PCTL, "(STUBBED) called, can_use={}", can_use);
|
|
|
|
|
|
|
|
|
|
can_use_stereo_vision = can_use;
|
|
|
|
|
LOG_DEBUG(Service_PCTL, "called, can_use={}", can_use);
|
|
|
|
|
|
|
|
|
|
IPC::ResponseBuilder rb{ctx, 2};
|
|
|
|
|
if (False(capability & Capability::StereoVision)) {
|
|
|
|
|
LOG_ERROR(Service_PCTL, "Application does not have StereoVision capability!");
|
|
|
|
|
rb.Push(Error::ResultNoCapability);
|
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
SetStereoVisionRestrictionImpl(can_use);
|
|
|
|
|
rb.Push(RESULT_SUCCESS);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void GetStereoVisionRestriction(Kernel::HLERequestContext& ctx) {
|
|
|
|
|
LOG_WARNING(Service_PCTL, "(STUBBED) called");
|
|
|
|
|
LOG_DEBUG(Service_PCTL, "called");
|
|
|
|
|
|
|
|
|
|
IPC::ResponseBuilder rb{ctx, 3};
|
|
|
|
|
if (False(capability & Capability::StereoVision)) {
|
|
|
|
|
LOG_ERROR(Service_PCTL, "Application does not have StereoVision capability!");
|
|
|
|
|
rb.Push(Error::ResultNoCapability);
|
|
|
|
|
rb.Push(false);
|
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
rb.Push(RESULT_SUCCESS);
|
|
|
|
|
rb.Push(can_use_stereo_vision);
|
|
|
|
|
rb.Push(settings.is_stero_vision_restricted);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void ResetConfirmedStereoVisionPermission(Kernel::HLERequestContext& ctx) {
|
|
|
|
|
LOG_WARNING(Service_PCTL, "(STUBBED) called");
|
|
|
|
|
LOG_DEBUG(Service_PCTL, "called");
|
|
|
|
|
|
|
|
|
|
states.stereo_vision = false;
|
|
|
|
|
|
|
|
|
|
IPC::ResponseBuilder rb{ctx, 2};
|
|
|
|
|
rb.Push(RESULT_SUCCESS);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
struct ApplicationInfo {
|
|
|
|
|
u64 tid{};
|
|
|
|
|
std::array<u8, 32> age_rating{};
|
|
|
|
|
u32 parental_control_flag{};
|
|
|
|
|
Capability capability{};
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
struct States {
|
|
|
|
|
u64 current_tid{};
|
|
|
|
|
ApplicationInfo application_info{};
|
|
|
|
|
u64 tid_from_event{};
|
|
|
|
|
bool launch_time_valid{};
|
|
|
|
|
bool is_suspended{};
|
|
|
|
|
bool temporary_unlocked{};
|
|
|
|
|
bool free_communication{};
|
|
|
|
|
bool stereo_vision{};
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
struct ParentalControlSettings {
|
|
|
|
|
bool is_stero_vision_restricted{};
|
|
|
|
|
bool is_free_communication_default_on{};
|
|
|
|
|
bool disabled{};
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
States states{};
|
|
|
|
|
ParentalControlSettings settings{};
|
|
|
|
|
std::array<char, 8> pin_code{};
|
|
|
|
|
bool can_use_stereo_vision = true;
|
|
|
|
|
Core::System& system;
|
|
|
|
|
Capability capability{};
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
void Module::Interface::CreateService(Kernel::HLERequestContext& ctx) {
|
|
|
|
@ -185,7 +372,9 @@ void Module::Interface::CreateService(Kernel::HLERequestContext& ctx) {
|
|
|
|
|
|
|
|
|
|
IPC::ResponseBuilder rb{ctx, 2, 0, 1};
|
|
|
|
|
rb.Push(RESULT_SUCCESS);
|
|
|
|
|
rb.PushIpcInterface<IParentalControlService>(system);
|
|
|
|
|
// TODO(ogniK): Get TID from process
|
|
|
|
|
|
|
|
|
|
rb.PushIpcInterface<IParentalControlService>(system, capability);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void Module::Interface::CreateServiceWithoutInitialize(Kernel::HLERequestContext& ctx) {
|
|
|
|
@ -193,21 +382,28 @@ void Module::Interface::CreateServiceWithoutInitialize(Kernel::HLERequestContext
|
|
|
|
|
|
|
|
|
|
IPC::ResponseBuilder rb{ctx, 2, 0, 1};
|
|
|
|
|
rb.Push(RESULT_SUCCESS);
|
|
|
|
|
rb.PushIpcInterface<IParentalControlService>(system);
|
|
|
|
|
rb.PushIpcInterface<IParentalControlService>(system, capability);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
Module::Interface::Interface(Core::System& system_, std::shared_ptr<Module> module_,
|
|
|
|
|
const char* name)
|
|
|
|
|
: ServiceFramework{system_, name}, module{std::move(module_)} {}
|
|
|
|
|
const char* name, Capability capability)
|
|
|
|
|
: ServiceFramework{system_, name}, module{std::move(module_)}, capability(capability) {}
|
|
|
|
|
|
|
|
|
|
Module::Interface::~Interface() = default;
|
|
|
|
|
|
|
|
|
|
void InstallInterfaces(SM::ServiceManager& service_manager, Core::System& system) {
|
|
|
|
|
auto module = std::make_shared<Module>();
|
|
|
|
|
std::make_shared<PCTL>(system, module, "pctl")->InstallAsService(service_manager);
|
|
|
|
|
std::make_shared<PCTL>(system, module, "pctl:a")->InstallAsService(service_manager);
|
|
|
|
|
std::make_shared<PCTL>(system, module, "pctl:r")->InstallAsService(service_manager);
|
|
|
|
|
std::make_shared<PCTL>(system, module, "pctl:s")->InstallAsService(service_manager);
|
|
|
|
|
std::make_shared<PCTL>(system, module, "pctl",
|
|
|
|
|
Capability::Application | Capability::SnsPost | Capability::Status |
|
|
|
|
|
Capability::StereoVision)
|
|
|
|
|
->InstallAsService(service_manager);
|
|
|
|
|
// TODO(ogniK): Implement remaining capabilities
|
|
|
|
|
std::make_shared<PCTL>(system, module, "pctl:a", Capability::None)
|
|
|
|
|
->InstallAsService(service_manager);
|
|
|
|
|
std::make_shared<PCTL>(system, module, "pctl:r", Capability::None)
|
|
|
|
|
->InstallAsService(service_manager);
|
|
|
|
|
std::make_shared<PCTL>(system, module, "pctl:s", Capability::None)
|
|
|
|
|
->InstallAsService(service_manager);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
} // namespace Service::PCTL
|
|
|
|
|