2019-01-18 22:26:39 +00:00
|
|
|
using Ryujinx.HLE.HOS.Kernel.Common;
|
|
|
|
using Ryujinx.HLE.HOS.Kernel.Process;
|
|
|
|
using Ryujinx.HLE.HOS.Kernel.Threading;
|
|
|
|
using Ryujinx.HLE.HOS.Services;
|
2020-07-19 18:24:18 +00:00
|
|
|
using System;
|
2019-01-18 22:26:39 +00:00
|
|
|
|
|
|
|
namespace Ryujinx.HLE.HOS.Kernel.Ipc
|
|
|
|
{
|
|
|
|
class KClientSession : KSynchronizationObject
|
|
|
|
{
|
|
|
|
public KProcess CreatorProcess { get; }
|
|
|
|
|
|
|
|
private KSession _parent;
|
|
|
|
|
|
|
|
public ChannelState State { get; set; }
|
|
|
|
|
2020-07-17 04:19:07 +00:00
|
|
|
public KClientPort ParentPort { get; }
|
|
|
|
|
2019-07-02 02:39:22 +00:00
|
|
|
// TODO: Remove that, we need it for now to allow HLE
|
|
|
|
// services implementation to work with the new IPC system.
|
2019-01-18 22:26:39 +00:00
|
|
|
public IpcService Service { get; set; }
|
|
|
|
|
2020-07-17 04:19:07 +00:00
|
|
|
public KClientSession(KernelContext context, KSession parent, KClientPort parentPort) : base(context)
|
2019-01-18 22:26:39 +00:00
|
|
|
{
|
2020-07-17 04:19:07 +00:00
|
|
|
_parent = parent;
|
|
|
|
ParentPort = parentPort;
|
|
|
|
|
|
|
|
parentPort?.IncrementReferenceCount();
|
2019-01-18 22:26:39 +00:00
|
|
|
|
|
|
|
State = ChannelState.Open;
|
|
|
|
|
2020-05-04 03:41:29 +00:00
|
|
|
CreatorProcess = context.Scheduler.GetCurrentProcess();
|
2019-01-18 22:26:39 +00:00
|
|
|
CreatorProcess.IncrementReferenceCount();
|
|
|
|
}
|
|
|
|
|
|
|
|
public KernelResult SendSyncRequest(ulong customCmdBuffAddr = 0, ulong customCmdBuffSize = 0)
|
|
|
|
{
|
2020-05-04 03:41:29 +00:00
|
|
|
KThread currentThread = KernelContext.Scheduler.GetCurrentThread();
|
2019-01-18 22:26:39 +00:00
|
|
|
|
|
|
|
KSessionRequest request = new KSessionRequest(currentThread, customCmdBuffAddr, customCmdBuffSize);
|
|
|
|
|
2020-05-04 03:41:29 +00:00
|
|
|
KernelContext.CriticalSection.Enter();
|
2019-01-18 22:26:39 +00:00
|
|
|
|
|
|
|
currentThread.SignaledObj = null;
|
|
|
|
currentThread.ObjSyncResult = KernelResult.Success;
|
|
|
|
|
|
|
|
KernelResult result = _parent.ServerSession.EnqueueRequest(request);
|
|
|
|
|
2020-05-04 03:41:29 +00:00
|
|
|
KernelContext.CriticalSection.Leave();
|
2019-01-18 22:26:39 +00:00
|
|
|
|
|
|
|
if (result == KernelResult.Success)
|
|
|
|
{
|
|
|
|
result = currentThread.ObjSyncResult;
|
|
|
|
}
|
|
|
|
|
|
|
|
return result;
|
|
|
|
}
|
|
|
|
|
2020-07-17 04:19:07 +00:00
|
|
|
public KernelResult SendAsyncRequest(KWritableEvent asyncEvent, ulong customCmdBuffAddr = 0, ulong customCmdBuffSize = 0)
|
|
|
|
{
|
|
|
|
KThread currentThread = KernelContext.Scheduler.GetCurrentThread();
|
|
|
|
|
|
|
|
KSessionRequest request = new KSessionRequest(currentThread, customCmdBuffAddr, customCmdBuffSize, asyncEvent);
|
|
|
|
|
|
|
|
KernelContext.CriticalSection.Enter();
|
|
|
|
|
|
|
|
KernelResult result = _parent.ServerSession.EnqueueRequest(request);
|
|
|
|
|
|
|
|
KernelContext.CriticalSection.Leave();
|
|
|
|
|
|
|
|
return result;
|
|
|
|
}
|
|
|
|
|
|
|
|
public void DisconnectFromPort()
|
|
|
|
{
|
|
|
|
if (ParentPort != null)
|
|
|
|
{
|
|
|
|
ParentPort.Disconnect();
|
|
|
|
ParentPort.DecrementReferenceCount();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-01-18 22:26:39 +00:00
|
|
|
protected override void Destroy()
|
|
|
|
{
|
|
|
|
_parent.DisconnectClient();
|
|
|
|
_parent.DecrementReferenceCount();
|
2020-07-19 18:24:18 +00:00
|
|
|
|
|
|
|
if (Service is IDisposable disposableObj)
|
|
|
|
{
|
|
|
|
disposableObj.Dispose();
|
|
|
|
}
|
2019-01-18 22:26:39 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|