mirror of
https://github.com/ryujinx-mirror/ryujinx.git
synced 2024-12-11 07:56:02 +01:00
7de7b559ad
* Improve kernel events implementation * Some cleanup * Address PR feedback
186 lines
5.4 KiB
C#
186 lines
5.4 KiB
C#
using Ryujinx.HLE.HOS.Ipc;
|
|
using Ryujinx.HLE.HOS.Kernel;
|
|
using Ryujinx.HLE.Logging;
|
|
using System;
|
|
using System.Collections.Generic;
|
|
using System.IO;
|
|
|
|
namespace Ryujinx.HLE.HOS.Services
|
|
{
|
|
abstract class IpcService : IIpcService
|
|
{
|
|
public abstract IReadOnlyDictionary<int, ServiceProcessRequest> Commands { get; }
|
|
|
|
private IdDictionary DomainObjects;
|
|
|
|
private int SelfId;
|
|
|
|
private bool IsDomain;
|
|
|
|
public IpcService()
|
|
{
|
|
DomainObjects = new IdDictionary();
|
|
|
|
SelfId = -1;
|
|
}
|
|
|
|
public int ConvertToDomain()
|
|
{
|
|
if (SelfId == -1)
|
|
{
|
|
SelfId = DomainObjects.Add(this);
|
|
}
|
|
|
|
IsDomain = true;
|
|
|
|
return SelfId;
|
|
}
|
|
|
|
public void ConvertToSession()
|
|
{
|
|
IsDomain = false;
|
|
}
|
|
|
|
public void CallMethod(ServiceCtx Context)
|
|
{
|
|
IIpcService Service = this;
|
|
|
|
if (IsDomain)
|
|
{
|
|
int DomainWord0 = Context.RequestData.ReadInt32();
|
|
int DomainObjId = Context.RequestData.ReadInt32();
|
|
|
|
int DomainCmd = (DomainWord0 >> 0) & 0xff;
|
|
int InputObjCount = (DomainWord0 >> 8) & 0xff;
|
|
int DataPayloadSize = (DomainWord0 >> 16) & 0xffff;
|
|
|
|
Context.RequestData.BaseStream.Seek(0x10 + DataPayloadSize, SeekOrigin.Begin);
|
|
|
|
for (int Index = 0; Index < InputObjCount; Index++)
|
|
{
|
|
Context.Request.ObjectIds.Add(Context.RequestData.ReadInt32());
|
|
}
|
|
|
|
Context.RequestData.BaseStream.Seek(0x10, SeekOrigin.Begin);
|
|
|
|
if (DomainCmd == 1)
|
|
{
|
|
Service = GetObject(DomainObjId);
|
|
|
|
Context.ResponseData.Write(0L);
|
|
Context.ResponseData.Write(0L);
|
|
}
|
|
else if (DomainCmd == 2)
|
|
{
|
|
Delete(DomainObjId);
|
|
|
|
Context.ResponseData.Write(0L);
|
|
|
|
return;
|
|
}
|
|
else
|
|
{
|
|
throw new NotImplementedException($"Domain command: {DomainCmd}");
|
|
}
|
|
}
|
|
|
|
long SfciMagic = Context.RequestData.ReadInt64();
|
|
int CommandId = (int)Context.RequestData.ReadInt64();
|
|
|
|
if (Service.Commands.TryGetValue(CommandId, out ServiceProcessRequest ProcessRequest))
|
|
{
|
|
Context.ResponseData.BaseStream.Seek(IsDomain ? 0x20 : 0x10, SeekOrigin.Begin);
|
|
|
|
Context.Device.Log.PrintDebug(LogClass.KernelIpc, $"{Service.GetType().Name}: {ProcessRequest.Method.Name}");
|
|
|
|
long Result = ProcessRequest(Context);
|
|
|
|
if (IsDomain)
|
|
{
|
|
foreach (int Id in Context.Response.ObjectIds)
|
|
{
|
|
Context.ResponseData.Write(Id);
|
|
}
|
|
|
|
Context.ResponseData.BaseStream.Seek(0, SeekOrigin.Begin);
|
|
|
|
Context.ResponseData.Write(Context.Response.ObjectIds.Count);
|
|
}
|
|
|
|
Context.ResponseData.BaseStream.Seek(IsDomain ? 0x10 : 0, SeekOrigin.Begin);
|
|
|
|
Context.ResponseData.Write(IpcMagic.Sfco);
|
|
Context.ResponseData.Write(Result);
|
|
}
|
|
else
|
|
{
|
|
string DbgMessage = $"{Context.Session.ServiceName} {Service.GetType().Name}: {CommandId}";
|
|
|
|
throw new NotImplementedException(DbgMessage);
|
|
}
|
|
}
|
|
|
|
protected static void MakeObject(ServiceCtx Context, IpcService Obj)
|
|
{
|
|
IpcService Service = Context.Session.Service;
|
|
|
|
if (Service.IsDomain)
|
|
{
|
|
Context.Response.ObjectIds.Add(Service.Add(Obj));
|
|
}
|
|
else
|
|
{
|
|
KSession Session = new KSession(Obj, Context.Session.ServiceName);
|
|
|
|
if (Context.Process.HandleTable.GenerateHandle(Session, out int Handle) != KernelResult.Success)
|
|
{
|
|
throw new InvalidOperationException("Out of handles!");
|
|
}
|
|
|
|
Context.Response.HandleDesc = IpcHandleDesc.MakeMove(Handle);
|
|
}
|
|
}
|
|
|
|
protected static T GetObject<T>(ServiceCtx Context, int Index) where T : IpcService
|
|
{
|
|
IpcService Service = Context.Session.Service;
|
|
|
|
if (!Service.IsDomain)
|
|
{
|
|
int Handle = Context.Request.HandleDesc.ToMove[Index];
|
|
|
|
KSession Session = Context.Process.HandleTable.GetObject<KSession>(Handle);
|
|
|
|
return Session?.Service is T ? (T)Session.Service : null;
|
|
}
|
|
|
|
int ObjId = Context.Request.ObjectIds[Index];
|
|
|
|
IIpcService Obj = Service.GetObject(ObjId);
|
|
|
|
return Obj is T ? (T)Obj : null;
|
|
}
|
|
|
|
private int Add(IIpcService Obj)
|
|
{
|
|
return DomainObjects.Add(Obj);
|
|
}
|
|
|
|
private bool Delete(int Id)
|
|
{
|
|
object Obj = DomainObjects.Delete(Id);
|
|
|
|
if (Obj is IDisposable DisposableObj)
|
|
{
|
|
DisposableObj.Dispose();
|
|
}
|
|
|
|
return Obj != null;
|
|
}
|
|
|
|
private IIpcService GetObject(int Id)
|
|
{
|
|
return DomainObjects.GetData<IIpcService>(Id);
|
|
}
|
|
}
|
|
} |