2018-02-04 20:08:20 -03:00
|
|
|
using ChocolArm64.Memory;
|
2018-08-16 20:47:36 -03:00
|
|
|
using Ryujinx.HLE.HOS.Kernel;
|
2018-02-04 20:08:20 -03:00
|
|
|
using System;
|
|
|
|
using System.IO;
|
|
|
|
|
2018-08-16 20:47:36 -03:00
|
|
|
namespace Ryujinx.HLE.HOS.Ipc
|
2018-02-04 20:08:20 -03:00
|
|
|
{
|
|
|
|
static class IpcHandler
|
|
|
|
{
|
NvServices refactoring (#120)
* Initial implementation of NvMap/NvHostCtrl
* More work on NvHostCtrl
* Refactoring of nvservices, move GPU Vmm, make Vmm per-process, refactor most gpu devices, move Gpu to Core, fix CbBind
* Implement GetGpuTime, support CancelSynchronization, fix issue on InsertWaitingMutex, proper double buffering support (again, not working properly for commercial games, only hb)
* Try to fix perf regression reading/writing textures, moved syncpts and events to a UserCtx class, delete global state when the process exits, other minor tweaks
* Remove now unused code, add comment about probably wrong result codes
2018-05-07 15:53:23 -03:00
|
|
|
public static long IpcCall(
|
2018-10-30 19:43:02 -06:00
|
|
|
Switch Ns,
|
|
|
|
Process Process,
|
|
|
|
MemoryManager Memory,
|
|
|
|
KSession Session,
|
|
|
|
IpcMessage Request,
|
|
|
|
long CmdPtr)
|
2018-02-04 20:08:20 -03:00
|
|
|
{
|
2018-03-19 15:58:46 -03:00
|
|
|
IpcMessage Response = new IpcMessage();
|
2018-02-04 20:08:20 -03:00
|
|
|
|
|
|
|
using (MemoryStream Raw = new MemoryStream(Request.RawData))
|
|
|
|
{
|
|
|
|
BinaryReader ReqReader = new BinaryReader(Raw);
|
|
|
|
|
2018-07-29 00:40:02 -04:00
|
|
|
if (Request.Type == IpcMessageType.Request ||
|
|
|
|
Request.Type == IpcMessageType.RequestWithContext)
|
2018-02-04 20:08:20 -03:00
|
|
|
{
|
2018-03-19 15:58:46 -03:00
|
|
|
Response.Type = IpcMessageType.Response;
|
2018-02-04 20:08:20 -03:00
|
|
|
|
2018-03-19 15:58:46 -03:00
|
|
|
using (MemoryStream ResMS = new MemoryStream())
|
2018-02-04 20:08:20 -03:00
|
|
|
{
|
2018-03-19 15:58:46 -03:00
|
|
|
BinaryWriter ResWriter = new BinaryWriter(ResMS);
|
2018-02-04 20:08:20 -03:00
|
|
|
|
2018-03-19 15:58:46 -03:00
|
|
|
ServiceCtx Context = new ServiceCtx(
|
|
|
|
Ns,
|
|
|
|
Process,
|
|
|
|
Memory,
|
|
|
|
Session,
|
|
|
|
Request,
|
|
|
|
Response,
|
|
|
|
ReqReader,
|
|
|
|
ResWriter);
|
2018-02-04 20:08:20 -03:00
|
|
|
|
2018-03-19 15:58:46 -03:00
|
|
|
Session.Service.CallMethod(Context);
|
2018-02-04 20:08:20 -03:00
|
|
|
|
2018-03-19 15:58:46 -03:00
|
|
|
Response.RawData = ResMS.ToArray();
|
2018-02-04 20:08:20 -03:00
|
|
|
}
|
|
|
|
}
|
2018-07-29 00:40:02 -04:00
|
|
|
else if (Request.Type == IpcMessageType.Control ||
|
|
|
|
Request.Type == IpcMessageType.ControlWithContext)
|
2018-02-04 20:08:20 -03:00
|
|
|
{
|
|
|
|
long Magic = ReqReader.ReadInt64();
|
|
|
|
long CmdId = ReqReader.ReadInt64();
|
|
|
|
|
|
|
|
switch (CmdId)
|
|
|
|
{
|
2018-03-12 01:04:52 -03:00
|
|
|
case 0:
|
|
|
|
{
|
2018-03-19 15:58:46 -03:00
|
|
|
Request = FillResponse(Response, 0, Session.Service.ConvertToDomain());
|
2018-03-12 01:04:52 -03:00
|
|
|
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
case 3:
|
|
|
|
{
|
|
|
|
Request = FillResponse(Response, 0, 0x500);
|
2018-04-04 21:01:36 -03:00
|
|
|
|
2018-03-12 01:04:52 -03:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
2018-04-04 21:01:36 -03:00
|
|
|
//TODO: Whats the difference between IpcDuplicateSession/Ex?
|
|
|
|
case 2:
|
2018-03-12 01:04:52 -03:00
|
|
|
case 4:
|
|
|
|
{
|
|
|
|
int Unknown = ReqReader.ReadInt32();
|
|
|
|
|
2018-09-23 15:11:46 -03:00
|
|
|
if (Process.HandleTable.GenerateHandle(Session, out int Handle) != KernelResult.Success)
|
|
|
|
{
|
|
|
|
throw new InvalidOperationException("Out of handles!");
|
|
|
|
}
|
2018-03-12 01:04:52 -03:00
|
|
|
|
|
|
|
Response.HandleDesc = IpcHandleDesc.MakeMove(Handle);
|
|
|
|
|
|
|
|
Request = FillResponse(Response, 0);
|
|
|
|
|
|
|
|
break;
|
|
|
|
}
|
2018-02-04 20:08:20 -03:00
|
|
|
|
|
|
|
default: throw new NotImplementedException(CmdId.ToString());
|
|
|
|
}
|
|
|
|
}
|
2018-03-12 01:04:52 -03:00
|
|
|
else if (Request.Type == IpcMessageType.CloseSession)
|
2018-02-04 20:08:20 -03:00
|
|
|
{
|
|
|
|
//TODO
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
throw new NotImplementedException(Request.Type.ToString());
|
|
|
|
}
|
|
|
|
|
2018-06-09 13:05:41 -03:00
|
|
|
Memory.WriteBytes(CmdPtr, Response.GetBytes(CmdPtr));
|
2018-02-04 20:08:20 -03:00
|
|
|
}
|
NvServices refactoring (#120)
* Initial implementation of NvMap/NvHostCtrl
* More work on NvHostCtrl
* Refactoring of nvservices, move GPU Vmm, make Vmm per-process, refactor most gpu devices, move Gpu to Core, fix CbBind
* Implement GetGpuTime, support CancelSynchronization, fix issue on InsertWaitingMutex, proper double buffering support (again, not working properly for commercial games, only hb)
* Try to fix perf regression reading/writing textures, moved syncpts and events to a UserCtx class, delete global state when the process exits, other minor tweaks
* Remove now unused code, add comment about probably wrong result codes
2018-05-07 15:53:23 -03:00
|
|
|
|
|
|
|
return 0;
|
2018-02-04 20:08:20 -03:00
|
|
|
}
|
|
|
|
|
|
|
|
private static IpcMessage FillResponse(IpcMessage Response, long Result, params int[] Values)
|
|
|
|
{
|
|
|
|
using (MemoryStream MS = new MemoryStream())
|
|
|
|
{
|
|
|
|
BinaryWriter Writer = new BinaryWriter(MS);
|
|
|
|
|
|
|
|
foreach (int Value in Values)
|
|
|
|
{
|
|
|
|
Writer.Write(Value);
|
|
|
|
}
|
|
|
|
|
|
|
|
return FillResponse(Response, Result, MS.ToArray());
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
private static IpcMessage FillResponse(IpcMessage Response, long Result, byte[] Data = null)
|
|
|
|
{
|
|
|
|
Response.Type = IpcMessageType.Response;
|
|
|
|
|
|
|
|
using (MemoryStream MS = new MemoryStream())
|
|
|
|
{
|
|
|
|
BinaryWriter Writer = new BinaryWriter(MS);
|
|
|
|
|
2018-03-19 15:58:46 -03:00
|
|
|
Writer.Write(IpcMagic.Sfco);
|
2018-02-04 20:08:20 -03:00
|
|
|
Writer.Write(Result);
|
|
|
|
|
|
|
|
if (Data != null)
|
|
|
|
{
|
|
|
|
Writer.Write(Data);
|
|
|
|
}
|
|
|
|
|
|
|
|
Response.RawData = MS.ToArray();
|
|
|
|
}
|
|
|
|
|
|
|
|
return Response;
|
|
|
|
}
|
|
|
|
}
|
2018-02-25 00:08:29 +01:00
|
|
|
}
|