Files
ryujinx/src/Ryujinx.HLE/HOS/Services/Pctl/ParentalControlServiceFactory/IParentalControlService.cs
Babib3l b62c58c2fe River : HLE: Make process identity explicit for service metadata resolution (#100)
This PR is the first in a batch of structural changes to Ryujinx.

**Changes**

- Added `ProcessIdentity` and `ProcessKind` to describe loaded programs by:
  - PID, program ID, application ID, program index, display version, process kind
- Stored identity metadata on `ProcessResult`.
- Added PID-based process lookup helpers to `ProcessLoader`.
- Updated HLE services to resolve application metadata through the caller PID instead of `Processes.ActiveApplication`.
- Added PTC/JIT disk cache initialization logging with PID, title ID, display version, selector, and enabled state.
- Added `ClientProcessId` property to ServiceCtx (/src/Ryujinx.HLE/HOS/ServiceCtx.cs) that uses the handle descriptor PId when available, falling back to `Process.Pid`.
- Updated 15 HLE service files to use `context.ClientProcessId` instead of `context.Process.Pid` for client process access, ensuring services correctly identify the calling process even when invoked via IPC with handle descriptors.

These changes make service metadata resolution more explicit and prepare the emulator for other structural changes later on.

Reviewed-on: https://git.ryujinx.app/projects/Ryubing/pulls/100
2026-05-25 12:08:31 +00:00

267 lines
8.7 KiB
C#

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 readonly ulong _pid;
private readonly int _permissionFlag;
private ulong _titleId;
private ParentalControlFlagValue _parentalControlFlag;
#pragma warning disable IDE0052, CS0414 // Remove unread private member
private int[] _ratingAge;
// TODO: Find where they are set.
private readonly bool _restrictionEnabled = false;
private readonly bool _featuresRestriction = false;
private bool _freeCommunicationEnabled = false;
private readonly bool _stereoVisionRestrictionConfigurable = true;
private bool _stereoVisionRestriction = false;
#pragma warning restore IDE0052, 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, _pid).TitleId;
if (titleId != 0)
{
_titleId = titleId;
var process = context.Device.Processes.GetProcess(_pid);
// TODO: Call nn::arp::GetApplicationControlProperty here when implemented, if it return ResultCode.Success we assign fields.
_ratingAge = new int[process.ApplicationControlProperties.RatingAge.Length];
for (int i = 0; i < _ratingAge.Length; i++)
{
_ratingAge[i] = Convert.ToInt32(process.ApplicationControlProperties.RatingAge[i]);
}
_parentalControlFlag = process.ApplicationControlProperties.ParentalControlFlag;
}
}
if (_titleId != 0)
{
// TODO: Service store some private fields in another object.
if ((_permissionFlag & 0x8040) == 0)
{
// TODO: Service store TitleId and FreeCommunicationEnabled in another object.
// When it's done it signal an event in this 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;
}
#pragma warning disable // Remove unnecessary value assignment
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;
}
}
}
}