186 lines
		
	
	
		
			No EOL
		
	
	
		
			5.4 KiB
		
	
	
	
		
			C#
		
	
	
	
	
	
			
		
		
	
	
			186 lines
		
	
	
		
			No EOL
		
	
	
		
			5.4 KiB
		
	
	
	
		
			C#
		
	
	
	
	
	
| using Ryujinx.Common.Logging;
 | |
| using Ryujinx.HLE.HOS.Ipc;
 | |
| using Ryujinx.HLE.HOS.Kernel;
 | |
| 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);
 | |
| 
 | |
|                 Logger.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);
 | |
|         }
 | |
|     }
 | |
| } | 
