2018-12-18 03:33:36 -02:00
|
|
|
using Ryujinx.HLE.HOS.Kernel.Common;
|
2019-01-18 20:26:39 -02:00
|
|
|
using Ryujinx.HLE.HOS.Kernel.Process;
|
2023-01-04 19:15:45 -03:00
|
|
|
using Ryujinx.Horizon.Common;
|
2020-07-17 01:19:07 -03:00
|
|
|
using System.Threading;
|
2018-12-18 03:33:36 -02:00
|
|
|
|
|
|
|
namespace Ryujinx.HLE.HOS.Kernel.Ipc
|
2018-11-28 20:18:09 -02:00
|
|
|
{
|
|
|
|
class KClientPort : KSynchronizationObject
|
|
|
|
{
|
2018-12-06 05:16:24 -06:00
|
|
|
private int _sessionsCount;
|
2020-05-04 00:41:29 -03:00
|
|
|
private readonly int _maxSessions;
|
2018-11-28 20:18:09 -02:00
|
|
|
|
2020-05-04 00:41:29 -03:00
|
|
|
private readonly KPort _parent;
|
2018-11-28 20:18:09 -02:00
|
|
|
|
2019-01-18 20:26:39 -02:00
|
|
|
public bool IsLight => _parent.IsLight;
|
2018-11-28 20:18:09 -02:00
|
|
|
|
2020-05-04 00:41:29 -03:00
|
|
|
public KClientPort(KernelContext context, KPort parent, int maxSessions) : base(context)
|
2018-11-28 20:18:09 -02:00
|
|
|
{
|
2018-12-06 05:16:24 -06:00
|
|
|
_maxSessions = maxSessions;
|
|
|
|
_parent = parent;
|
2019-01-18 20:26:39 -02:00
|
|
|
}
|
|
|
|
|
2023-01-04 19:15:45 -03:00
|
|
|
public Result Connect(out KClientSession clientSession)
|
2019-01-18 20:26:39 -02:00
|
|
|
{
|
|
|
|
clientSession = null;
|
|
|
|
|
2020-12-09 19:20:05 -03:00
|
|
|
KProcess currentProcess = KernelStatic.GetCurrentProcess();
|
2019-01-18 20:26:39 -02:00
|
|
|
|
|
|
|
if (currentProcess.ResourceLimit != null &&
|
|
|
|
!currentProcess.ResourceLimit.Reserve(LimitableResource.Session, 1))
|
|
|
|
{
|
|
|
|
return KernelResult.ResLimitExceeded;
|
|
|
|
}
|
|
|
|
|
2020-07-17 01:19:07 -03:00
|
|
|
if (!IncrementSessionsCount())
|
2019-01-18 20:26:39 -02:00
|
|
|
{
|
2020-07-17 01:19:07 -03:00
|
|
|
currentProcess.ResourceLimit?.Release(LimitableResource.Session, 1);
|
2019-01-18 20:26:39 -02:00
|
|
|
|
2020-07-17 01:19:07 -03:00
|
|
|
return KernelResult.SessionCountExceeded;
|
2019-01-18 20:26:39 -02:00
|
|
|
}
|
|
|
|
|
2020-07-17 01:19:07 -03:00
|
|
|
KSession session = new KSession(KernelContext, this);
|
2019-01-18 20:26:39 -02:00
|
|
|
|
2023-01-04 19:15:45 -03:00
|
|
|
Result result = _parent.EnqueueIncomingSession(session.ServerSession);
|
2019-01-18 20:26:39 -02:00
|
|
|
|
2023-01-04 19:15:45 -03:00
|
|
|
if (result != Result.Success)
|
2019-01-18 20:26:39 -02:00
|
|
|
{
|
|
|
|
session.ClientSession.DecrementReferenceCount();
|
|
|
|
session.ServerSession.DecrementReferenceCount();
|
|
|
|
|
|
|
|
return result;
|
|
|
|
}
|
|
|
|
|
|
|
|
clientSession = session.ClientSession;
|
|
|
|
|
|
|
|
return result;
|
|
|
|
}
|
|
|
|
|
2023-01-04 19:15:45 -03:00
|
|
|
public Result ConnectLight(out KLightClientSession clientSession)
|
2019-01-18 20:26:39 -02:00
|
|
|
{
|
|
|
|
clientSession = null;
|
|
|
|
|
2020-12-09 19:20:05 -03:00
|
|
|
KProcess currentProcess = KernelStatic.GetCurrentProcess();
|
2019-01-18 20:26:39 -02:00
|
|
|
|
|
|
|
if (currentProcess.ResourceLimit != null &&
|
|
|
|
!currentProcess.ResourceLimit.Reserve(LimitableResource.Session, 1))
|
|
|
|
{
|
|
|
|
return KernelResult.ResLimitExceeded;
|
|
|
|
}
|
|
|
|
|
2020-07-17 01:19:07 -03:00
|
|
|
if (!IncrementSessionsCount())
|
2019-01-18 20:26:39 -02:00
|
|
|
{
|
2020-07-17 01:19:07 -03:00
|
|
|
currentProcess.ResourceLimit?.Release(LimitableResource.Session, 1);
|
2019-01-18 20:26:39 -02:00
|
|
|
|
2020-07-17 01:19:07 -03:00
|
|
|
return KernelResult.SessionCountExceeded;
|
2019-01-18 20:26:39 -02:00
|
|
|
}
|
|
|
|
|
2020-05-04 00:41:29 -03:00
|
|
|
KLightSession session = new KLightSession(KernelContext);
|
2019-01-18 20:26:39 -02:00
|
|
|
|
2023-01-04 19:15:45 -03:00
|
|
|
Result result = _parent.EnqueueIncomingLightSession(session.ServerSession);
|
2019-01-18 20:26:39 -02:00
|
|
|
|
2023-01-04 19:15:45 -03:00
|
|
|
if (result != Result.Success)
|
2019-01-18 20:26:39 -02:00
|
|
|
{
|
|
|
|
session.ClientSession.DecrementReferenceCount();
|
|
|
|
session.ServerSession.DecrementReferenceCount();
|
|
|
|
|
|
|
|
return result;
|
|
|
|
}
|
|
|
|
|
|
|
|
clientSession = session.ClientSession;
|
|
|
|
|
|
|
|
return result;
|
2018-11-28 20:18:09 -02:00
|
|
|
}
|
|
|
|
|
2020-07-17 01:19:07 -03:00
|
|
|
private bool IncrementSessionsCount()
|
|
|
|
{
|
|
|
|
while (true)
|
|
|
|
{
|
|
|
|
int currentCount = _sessionsCount;
|
|
|
|
|
|
|
|
if (currentCount < _maxSessions)
|
|
|
|
{
|
|
|
|
if (Interlocked.CompareExchange(ref _sessionsCount, currentCount + 1, currentCount) == currentCount)
|
|
|
|
{
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
public void Disconnect()
|
|
|
|
{
|
|
|
|
KernelContext.CriticalSection.Enter();
|
|
|
|
|
|
|
|
SignalIfMaximumReached(Interlocked.Decrement(ref _sessionsCount));
|
|
|
|
|
|
|
|
KernelContext.CriticalSection.Leave();
|
|
|
|
}
|
|
|
|
|
|
|
|
private void SignalIfMaximumReached(int value)
|
|
|
|
{
|
|
|
|
if (value == _maxSessions)
|
|
|
|
{
|
|
|
|
Signal();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2023-01-04 19:15:45 -03:00
|
|
|
public new static Result RemoveName(KernelContext context, string name)
|
2018-11-28 20:18:09 -02:00
|
|
|
{
|
2020-05-04 00:41:29 -03:00
|
|
|
KAutoObject foundObj = FindNamedObject(context, name);
|
2018-11-28 20:18:09 -02:00
|
|
|
|
2018-12-06 05:16:24 -06:00
|
|
|
if (!(foundObj is KClientPort))
|
2018-11-28 20:18:09 -02:00
|
|
|
{
|
|
|
|
return KernelResult.NotFound;
|
|
|
|
}
|
|
|
|
|
2020-05-04 00:41:29 -03:00
|
|
|
return KAutoObject.RemoveName(context, name);
|
2018-11-28 20:18:09 -02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|