aboutsummaryrefslogtreecommitdiff
path: root/src/Ryujinx.HLE/HOS/Services/Pctl
diff options
context:
space:
mode:
authorTSR Berry <20988865+TSRBerry@users.noreply.github.com>2023-04-08 01:22:00 +0200
committerMary <thog@protonmail.com>2023-04-27 23:51:14 +0200
commitcee712105850ac3385cd0091a923438167433f9f (patch)
tree4a5274b21d8b7f938c0d0ce18736d3f2993b11b1 /src/Ryujinx.HLE/HOS/Services/Pctl
parentcd124bda587ef09668a971fa1cac1c3f0cfc9f21 (diff)
Move solution and projects to src
Diffstat (limited to 'src/Ryujinx.HLE/HOS/Services/Pctl')
-rw-r--r--src/Ryujinx.HLE/HOS/Services/Pctl/IParentalControlServiceFactory.cs40
-rw-r--r--src/Ryujinx.HLE/HOS/Services/Pctl/ParentalControlServiceFactory/IParentalControlService.cs259
-rw-r--r--src/Ryujinx.HLE/HOS/Services/Pctl/ResultCode.cs16
3 files changed, 315 insertions, 0 deletions
diff --git a/src/Ryujinx.HLE/HOS/Services/Pctl/IParentalControlServiceFactory.cs b/src/Ryujinx.HLE/HOS/Services/Pctl/IParentalControlServiceFactory.cs
new file mode 100644
index 00000000..990aef09
--- /dev/null
+++ b/src/Ryujinx.HLE/HOS/Services/Pctl/IParentalControlServiceFactory.cs
@@ -0,0 +1,40 @@
+using Ryujinx.HLE.HOS.Services.Pctl.ParentalControlServiceFactory;
+
+namespace Ryujinx.HLE.HOS.Services.Pctl
+{
+ [Service("pctl", 0x303)]
+ [Service("pctl:a", 0x83BE)]
+ [Service("pctl:r", 0x8040)]
+ [Service("pctl:s", 0x838E)]
+ class IParentalControlServiceFactory : IpcService
+ {
+ private int _permissionFlag;
+
+ public IParentalControlServiceFactory(ServiceCtx context, int permissionFlag)
+ {
+ _permissionFlag = permissionFlag;
+ }
+
+ [CommandCmif(0)]
+ // CreateService(u64, pid) -> object<nn::pctl::detail::ipc::IParentalControlService>
+ public ResultCode CreateService(ServiceCtx context)
+ {
+ ulong pid = context.Request.HandleDesc.PId;
+
+ MakeObject(context, new IParentalControlService(context, pid, true, _permissionFlag));
+
+ return ResultCode.Success;
+ }
+
+ [CommandCmif(1)] // 4.0.0+
+ // CreateServiceWithoutInitialize(u64, pid) -> object<nn::pctl::detail::ipc::IParentalControlService>
+ public ResultCode CreateServiceWithoutInitialize(ServiceCtx context)
+ {
+ ulong pid = context.Request.HandleDesc.PId;
+
+ MakeObject(context, new IParentalControlService(context, pid, false, _permissionFlag));
+
+ return ResultCode.Success;
+ }
+ }
+} \ No newline at end of file
diff --git a/src/Ryujinx.HLE/HOS/Services/Pctl/ParentalControlServiceFactory/IParentalControlService.cs b/src/Ryujinx.HLE/HOS/Services/Pctl/ParentalControlServiceFactory/IParentalControlService.cs
new file mode 100644
index 00000000..594ee4e0
--- /dev/null
+++ b/src/Ryujinx.HLE/HOS/Services/Pctl/ParentalControlServiceFactory/IParentalControlService.cs
@@ -0,0 +1,259 @@
+using Ryujinx.Common.Logging;
+using Ryujinx.HLE.HOS.Services.Arp;
+using System;
+
+using static LibHac.Ns.ApplicationControlProperty;
+
+namespace Ryujinx.HLE.HOS.Services.Pctl.ParentalControlServiceFactory
+{
+ class IParentalControlService : IpcService
+ {
+ private ulong _pid;
+ private int _permissionFlag;
+ private ulong _titleId;
+ private ParentalControlFlagValue _parentalControlFlag;
+ private int[] _ratingAge;
+
+#pragma warning disable CS0414
+ // TODO: Find where they are set.
+ private bool _restrictionEnabled = false;
+ private bool _featuresRestriction = false;
+ private bool _freeCommunicationEnabled = false;
+ private bool _stereoVisionRestrictionConfigurable = true;
+ private bool _stereoVisionRestriction = false;
+#pragma warning restore CS0414
+
+ public IParentalControlService(ServiceCtx context, ulong pid, bool withInitialize, int permissionFlag)
+ {
+ _pid = pid;
+ _permissionFlag = permissionFlag;
+
+ if (withInitialize)
+ {
+ Initialize(context);
+ }
+ }
+
+ [CommandCmif(1)] // 4.0.0+
+ // Initialize()
+ public ResultCode Initialize(ServiceCtx context)
+ {
+ if ((_permissionFlag & 0x8001) == 0)
+ {
+ return ResultCode.PermissionDenied;
+ }
+
+ ResultCode resultCode = ResultCode.InvalidPid;
+
+ if (_pid != 0)
+ {
+ if ((_permissionFlag & 0x40) == 0)
+ {
+ ulong titleId = ApplicationLaunchProperty.GetByPid(context).TitleId;
+
+ if (titleId != 0)
+ {
+ _titleId = titleId;
+
+ // TODO: Call nn::arp::GetApplicationControlProperty here when implemented, if it return ResultCode.Success we assign fields.
+ _ratingAge = Array.ConvertAll(context.Device.Processes.ActiveApplication.ApplicationControlProperties.RatingAge.ItemsRo.ToArray(), Convert.ToInt32);
+ _parentalControlFlag = context.Device.Processes.ActiveApplication.ApplicationControlProperties.ParentalControlFlag;
+ }
+ }
+
+ if (_titleId != 0)
+ {
+ // TODO: Service store some private fields in another static object.
+
+ if ((_permissionFlag & 0x8040) == 0)
+ {
+ // TODO: Service store TitleId and FreeCommunicationEnabled in another static object.
+ // When it's done it signal an event in this static object.
+ Logger.Stub?.PrintStub(LogClass.ServicePctl);
+ }
+ }
+
+ resultCode = ResultCode.Success;
+ }
+
+ return resultCode;
+ }
+
+ [CommandCmif(1001)]
+ // CheckFreeCommunicationPermission()
+ public ResultCode CheckFreeCommunicationPermission(ServiceCtx context)
+ {
+ if (_parentalControlFlag == ParentalControlFlagValue.FreeCommunication && _restrictionEnabled)
+ {
+ // TODO: It seems to checks if an entry exists in the FreeCommunicationApplicationList using the TitleId.
+ // Then it returns FreeCommunicationDisabled if the entry doesn't exist.
+
+ return ResultCode.FreeCommunicationDisabled;
+ }
+
+ _freeCommunicationEnabled = true;
+
+ Logger.Stub?.PrintStub(LogClass.ServicePctl);
+
+ return ResultCode.Success;
+ }
+
+ [CommandCmif(1017)] // 10.0.0+
+ // EndFreeCommunication()
+ public ResultCode EndFreeCommunication(ServiceCtx context)
+ {
+ _freeCommunicationEnabled = false;
+
+ return ResultCode.Success;
+ }
+
+ [CommandCmif(1013)] // 4.0.0+
+ // ConfirmStereoVisionPermission()
+ public ResultCode ConfirmStereoVisionPermission(ServiceCtx context)
+ {
+ return IsStereoVisionPermittedImpl();
+ }
+
+ [CommandCmif(1018)]
+ // IsFreeCommunicationAvailable()
+ public ResultCode IsFreeCommunicationAvailable(ServiceCtx context)
+ {
+ if (_parentalControlFlag == ParentalControlFlagValue.FreeCommunication && _restrictionEnabled)
+ {
+ // TODO: It seems to checks if an entry exists in the FreeCommunicationApplicationList using the TitleId.
+ // Then it returns FreeCommunicationDisabled if the entry doesn't exist.
+
+ return ResultCode.FreeCommunicationDisabled;
+ }
+
+ Logger.Stub?.PrintStub(LogClass.ServicePctl);
+
+ return ResultCode.Success;
+ }
+
+ [CommandCmif(1031)]
+ // IsRestrictionEnabled() -> b8
+ public ResultCode IsRestrictionEnabled(ServiceCtx context)
+ {
+ if ((_permissionFlag & 0x140) == 0)
+ {
+ return ResultCode.PermissionDenied;
+ }
+
+ context.ResponseData.Write(_restrictionEnabled);
+
+ return ResultCode.Success;
+ }
+
+ [CommandCmif(1061)] // 4.0.0+
+ // ConfirmStereoVisionRestrictionConfigurable()
+ public ResultCode ConfirmStereoVisionRestrictionConfigurable(ServiceCtx context)
+ {
+ if ((_permissionFlag & 2) == 0)
+ {
+ return ResultCode.PermissionDenied;
+ }
+
+ if (_stereoVisionRestrictionConfigurable)
+ {
+ return ResultCode.Success;
+ }
+ else
+ {
+ return ResultCode.StereoVisionRestrictionConfigurableDisabled;
+ }
+ }
+
+ [CommandCmif(1062)] // 4.0.0+
+ // GetStereoVisionRestriction() -> bool
+ public ResultCode GetStereoVisionRestriction(ServiceCtx context)
+ {
+ if ((_permissionFlag & 0x200) == 0)
+ {
+ return ResultCode.PermissionDenied;
+ }
+
+ bool stereoVisionRestriction = false;
+
+ if (_stereoVisionRestrictionConfigurable)
+ {
+ stereoVisionRestriction = _stereoVisionRestriction;
+ }
+
+ context.ResponseData.Write(stereoVisionRestriction);
+
+ return ResultCode.Success;
+ }
+
+ [CommandCmif(1063)] // 4.0.0+
+ // SetStereoVisionRestriction(bool)
+ public ResultCode SetStereoVisionRestriction(ServiceCtx context)
+ {
+ if ((_permissionFlag & 0x200) == 0)
+ {
+ return ResultCode.PermissionDenied;
+ }
+
+ bool stereoVisionRestriction = context.RequestData.ReadBoolean();
+
+ if (!_featuresRestriction)
+ {
+ if (_stereoVisionRestrictionConfigurable)
+ {
+ _stereoVisionRestriction = stereoVisionRestriction;
+
+ // TODO: It signals an internal event of service. We have to determine where this event is used.
+ }
+ }
+
+ return ResultCode.Success;
+ }
+
+ [CommandCmif(1064)] // 5.0.0+
+ // ResetConfirmedStereoVisionPermission()
+ public ResultCode ResetConfirmedStereoVisionPermission(ServiceCtx context)
+ {
+ return ResultCode.Success;
+ }
+
+ [CommandCmif(1065)] // 5.0.0+
+ // IsStereoVisionPermitted() -> bool
+ public ResultCode IsStereoVisionPermitted(ServiceCtx context)
+ {
+ bool isStereoVisionPermitted = false;
+
+ ResultCode resultCode = IsStereoVisionPermittedImpl();
+
+ if (resultCode == ResultCode.Success)
+ {
+ isStereoVisionPermitted = true;
+ }
+
+ context.ResponseData.Write(isStereoVisionPermitted);
+
+ return resultCode;
+ }
+
+ private ResultCode IsStereoVisionPermittedImpl()
+ {
+ /*
+ // TODO: Application Exemptions are read from file "appExemptions.dat" in the service savedata.
+ // Since we don't support the pctl savedata for now, this can be implemented later.
+
+ if (appExemption)
+ {
+ return ResultCode.Success;
+ }
+ */
+
+ if (_stereoVisionRestrictionConfigurable && _stereoVisionRestriction)
+ {
+ return ResultCode.StereoVisionDenied;
+ }
+ else
+ {
+ return ResultCode.Success;
+ }
+ }
+ }
+} \ No newline at end of file
diff --git a/src/Ryujinx.HLE/HOS/Services/Pctl/ResultCode.cs b/src/Ryujinx.HLE/HOS/Services/Pctl/ResultCode.cs
new file mode 100644
index 00000000..fcf06ee9
--- /dev/null
+++ b/src/Ryujinx.HLE/HOS/Services/Pctl/ResultCode.cs
@@ -0,0 +1,16 @@
+namespace Ryujinx.HLE.HOS.Services.Pctl
+{
+ enum ResultCode
+ {
+ ModuleId = 142,
+ ErrorCodeShift = 9,
+
+ Success = 0,
+
+ FreeCommunicationDisabled = (101 << ErrorCodeShift) | ModuleId,
+ StereoVisionDenied = (104 << ErrorCodeShift) | ModuleId,
+ InvalidPid = (131 << ErrorCodeShift) | ModuleId,
+ PermissionDenied = (133 << ErrorCodeShift) | ModuleId,
+ StereoVisionRestrictionConfigurableDisabled = (181 << ErrorCodeShift) | ModuleId,
+ }
+}