2018-10-17 17:15:50 +00:00
|
|
|
using Ryujinx.Common.Logging;
|
2020-04-19 01:25:57 +00:00
|
|
|
using Ryujinx.HLE.HOS.Services.Mm.Types;
|
|
|
|
using System.Collections.Generic;
|
2018-05-26 20:49:21 +00:00
|
|
|
|
2018-08-16 23:47:36 +00:00
|
|
|
namespace Ryujinx.HLE.HOS.Services.Mm
|
2018-05-26 20:49:21 +00:00
|
|
|
{
|
2019-07-10 15:59:54 +00:00
|
|
|
[Service("mm:u")]
|
2018-05-26 20:49:21 +00:00
|
|
|
class IRequest : IpcService
|
|
|
|
{
|
2020-04-19 01:25:57 +00:00
|
|
|
private static object _sessionListLock = new object();
|
|
|
|
private static List<MultiMediaSession> _sessionList = new List<MultiMediaSession>();
|
|
|
|
|
|
|
|
private static uint _uniqueId = 1;
|
|
|
|
|
2021-04-22 21:35:01 +00:00
|
|
|
public IRequest(ServiceCtx context) { }
|
2018-05-26 20:49:21 +00:00
|
|
|
|
2023-04-14 23:00:34 +00:00
|
|
|
[CommandCmif(0)]
|
2018-10-07 15:12:11 +00:00
|
|
|
// InitializeOld(u32, u32, u32)
|
2019-07-14 19:04:38 +00:00
|
|
|
public ResultCode InitializeOld(ServiceCtx context)
|
2018-10-07 15:12:11 +00:00
|
|
|
{
|
2020-04-19 01:25:57 +00:00
|
|
|
MultiMediaOperationType operationType = (MultiMediaOperationType)context.RequestData.ReadUInt32();
|
|
|
|
int fgmId = context.RequestData.ReadInt32();
|
|
|
|
bool isAutoClearEvent = context.RequestData.ReadInt32() != 0;
|
2018-10-07 15:12:11 +00:00
|
|
|
|
2020-08-03 23:32:53 +00:00
|
|
|
Logger.Stub?.PrintStub(LogClass.ServiceMm, new { operationType, fgmId, isAutoClearEvent });
|
2020-04-19 01:25:57 +00:00
|
|
|
|
|
|
|
Register(operationType, fgmId, isAutoClearEvent);
|
2018-12-07 15:19:10 +00:00
|
|
|
|
2019-07-14 19:04:38 +00:00
|
|
|
return ResultCode.Success;
|
2018-12-07 15:19:10 +00:00
|
|
|
}
|
|
|
|
|
2023-04-14 23:00:34 +00:00
|
|
|
[CommandCmif(1)]
|
2018-12-07 15:19:10 +00:00
|
|
|
// FinalizeOld(u32)
|
2019-07-14 19:04:38 +00:00
|
|
|
public ResultCode FinalizeOld(ServiceCtx context)
|
2018-12-07 15:19:10 +00:00
|
|
|
{
|
2020-04-19 01:25:57 +00:00
|
|
|
MultiMediaOperationType operationType = (MultiMediaOperationType)context.RequestData.ReadUInt32();
|
|
|
|
|
2020-08-03 23:32:53 +00:00
|
|
|
Logger.Stub?.PrintStub(LogClass.ServiceMm, new { operationType });
|
2020-04-19 01:25:57 +00:00
|
|
|
|
|
|
|
lock (_sessionListLock)
|
|
|
|
{
|
|
|
|
_sessionList.Remove(GetSessionByType(operationType));
|
|
|
|
}
|
2018-10-07 15:12:11 +00:00
|
|
|
|
2019-07-14 19:04:38 +00:00
|
|
|
return ResultCode.Success;
|
2018-10-07 15:12:11 +00:00
|
|
|
}
|
|
|
|
|
2023-04-14 23:00:34 +00:00
|
|
|
[CommandCmif(2)]
|
2018-12-07 15:19:10 +00:00
|
|
|
// SetAndWaitOld(u32, u32, u32)
|
2019-07-14 19:04:38 +00:00
|
|
|
public ResultCode SetAndWaitOld(ServiceCtx context)
|
2018-12-07 15:19:10 +00:00
|
|
|
{
|
2020-04-19 01:25:57 +00:00
|
|
|
MultiMediaOperationType operationType = (MultiMediaOperationType)context.RequestData.ReadUInt32();
|
2021-04-22 21:35:01 +00:00
|
|
|
uint frequenceHz = context.RequestData.ReadUInt32();
|
2020-04-19 01:25:57 +00:00
|
|
|
int timeout = context.RequestData.ReadInt32();
|
|
|
|
|
2021-04-22 21:35:01 +00:00
|
|
|
Logger.Stub?.PrintStub(LogClass.ServiceMm, new { operationType, frequenceHz, timeout });
|
2020-04-19 01:25:57 +00:00
|
|
|
|
|
|
|
lock (_sessionListLock)
|
|
|
|
{
|
2021-04-22 21:35:01 +00:00
|
|
|
GetSessionByType(operationType)?.SetAndWait(frequenceHz, timeout);
|
2020-04-19 01:25:57 +00:00
|
|
|
}
|
2018-12-07 15:19:10 +00:00
|
|
|
|
2019-07-14 19:04:38 +00:00
|
|
|
return ResultCode.Success;
|
2018-12-07 15:19:10 +00:00
|
|
|
}
|
|
|
|
|
2023-04-14 23:00:34 +00:00
|
|
|
[CommandCmif(3)]
|
2018-12-07 15:19:10 +00:00
|
|
|
// GetOld(u32) -> u32
|
2019-07-14 19:04:38 +00:00
|
|
|
public ResultCode GetOld(ServiceCtx context)
|
2018-12-07 15:19:10 +00:00
|
|
|
{
|
2020-04-19 01:25:57 +00:00
|
|
|
MultiMediaOperationType operationType = (MultiMediaOperationType)context.RequestData.ReadUInt32();
|
2018-12-07 15:19:10 +00:00
|
|
|
|
2020-08-03 23:32:53 +00:00
|
|
|
Logger.Stub?.PrintStub(LogClass.ServiceMm, new { operationType });
|
2018-12-07 15:19:10 +00:00
|
|
|
|
2020-04-19 01:25:57 +00:00
|
|
|
lock (_sessionListLock)
|
|
|
|
{
|
|
|
|
MultiMediaSession session = GetSessionByType(operationType);
|
|
|
|
|
|
|
|
uint currentValue = session == null ? 0 : session.CurrentValue;
|
|
|
|
|
|
|
|
context.ResponseData.Write(currentValue);
|
|
|
|
}
|
2018-12-07 15:19:10 +00:00
|
|
|
|
2019-07-14 19:04:38 +00:00
|
|
|
return ResultCode.Success;
|
2018-12-07 15:19:10 +00:00
|
|
|
}
|
|
|
|
|
2023-04-14 23:00:34 +00:00
|
|
|
[CommandCmif(4)]
|
2020-04-19 01:25:57 +00:00
|
|
|
// Initialize(u32, u32, u32) -> u32
|
2019-07-14 19:04:38 +00:00
|
|
|
public ResultCode Initialize(ServiceCtx context)
|
2018-05-26 20:49:21 +00:00
|
|
|
{
|
2020-04-19 01:25:57 +00:00
|
|
|
MultiMediaOperationType operationType = (MultiMediaOperationType)context.RequestData.ReadUInt32();
|
|
|
|
int fgmId = context.RequestData.ReadInt32();
|
|
|
|
bool isAutoClearEvent = context.RequestData.ReadInt32() != 0;
|
|
|
|
|
2020-08-03 23:32:53 +00:00
|
|
|
Logger.Stub?.PrintStub(LogClass.ServiceMm, new { operationType, fgmId, isAutoClearEvent });
|
2020-04-19 01:25:57 +00:00
|
|
|
|
|
|
|
uint id = Register(operationType, fgmId, isAutoClearEvent);
|
|
|
|
|
|
|
|
context.ResponseData.Write(id);
|
2018-05-26 20:49:21 +00:00
|
|
|
|
2019-07-14 19:04:38 +00:00
|
|
|
return ResultCode.Success;
|
2018-05-26 20:49:21 +00:00
|
|
|
}
|
|
|
|
|
2023-04-14 23:00:34 +00:00
|
|
|
[CommandCmif(5)]
|
2018-12-07 15:19:10 +00:00
|
|
|
// Finalize(u32)
|
2019-07-14 19:04:38 +00:00
|
|
|
public ResultCode Finalize(ServiceCtx context)
|
2018-12-03 02:38:47 +00:00
|
|
|
{
|
2020-04-19 01:25:57 +00:00
|
|
|
uint id = context.RequestData.ReadUInt32();
|
|
|
|
|
2020-08-03 23:32:53 +00:00
|
|
|
Logger.Stub?.PrintStub(LogClass.ServiceMm, new { id });
|
2020-04-19 01:25:57 +00:00
|
|
|
|
|
|
|
lock (_sessionListLock)
|
|
|
|
{
|
|
|
|
_sessionList.Remove(GetSessionById(id));
|
|
|
|
}
|
2018-12-03 02:38:47 +00:00
|
|
|
|
2019-07-14 19:04:38 +00:00
|
|
|
return ResultCode.Success;
|
2018-12-03 02:38:47 +00:00
|
|
|
}
|
|
|
|
|
2023-04-14 23:00:34 +00:00
|
|
|
[CommandCmif(6)]
|
2018-12-07 15:19:10 +00:00
|
|
|
// SetAndWait(u32, u32, u32)
|
2019-07-14 19:04:38 +00:00
|
|
|
public ResultCode SetAndWait(ServiceCtx context)
|
2018-05-26 20:49:21 +00:00
|
|
|
{
|
2021-04-22 21:35:01 +00:00
|
|
|
uint id = context.RequestData.ReadUInt32();
|
|
|
|
uint frequenceHz = context.RequestData.ReadUInt32();
|
|
|
|
int timeout = context.RequestData.ReadInt32();
|
2018-12-07 15:19:10 +00:00
|
|
|
|
2021-04-22 21:35:01 +00:00
|
|
|
Logger.Stub?.PrintStub(LogClass.ServiceMm, new { id, frequenceHz, timeout });
|
2020-04-19 01:25:57 +00:00
|
|
|
|
|
|
|
lock (_sessionListLock)
|
|
|
|
{
|
2021-04-22 21:35:01 +00:00
|
|
|
GetSessionById(id)?.SetAndWait(frequenceHz, timeout);
|
2020-04-19 01:25:57 +00:00
|
|
|
}
|
2018-05-26 20:49:21 +00:00
|
|
|
|
2019-07-14 19:04:38 +00:00
|
|
|
return ResultCode.Success;
|
2018-05-26 20:49:21 +00:00
|
|
|
}
|
|
|
|
|
2023-04-14 23:00:34 +00:00
|
|
|
[CommandCmif(7)]
|
2018-12-07 15:19:10 +00:00
|
|
|
// Get(u32) -> u32
|
2019-07-14 19:04:38 +00:00
|
|
|
public ResultCode Get(ServiceCtx context)
|
2018-05-26 20:49:21 +00:00
|
|
|
{
|
2020-04-19 01:25:57 +00:00
|
|
|
uint id = context.RequestData.ReadUInt32();
|
|
|
|
|
2020-08-03 23:32:53 +00:00
|
|
|
Logger.Stub?.PrintStub(LogClass.ServiceMm, new { id });
|
2018-05-26 20:49:21 +00:00
|
|
|
|
2020-04-19 01:25:57 +00:00
|
|
|
lock (_sessionListLock)
|
|
|
|
{
|
|
|
|
MultiMediaSession session = GetSessionById(id);
|
2018-12-07 15:19:10 +00:00
|
|
|
|
2020-04-19 01:25:57 +00:00
|
|
|
uint currentValue = session == null ? 0 : session.CurrentValue;
|
|
|
|
|
|
|
|
context.ResponseData.Write(currentValue);
|
|
|
|
}
|
2018-05-26 20:49:21 +00:00
|
|
|
|
2019-07-14 19:04:38 +00:00
|
|
|
return ResultCode.Success;
|
2018-05-26 20:49:21 +00:00
|
|
|
}
|
2020-04-19 01:25:57 +00:00
|
|
|
|
|
|
|
private MultiMediaSession GetSessionById(uint id)
|
|
|
|
{
|
|
|
|
foreach (MultiMediaSession session in _sessionList)
|
|
|
|
{
|
|
|
|
if (session.Id == id)
|
|
|
|
{
|
|
|
|
return session;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return null;
|
|
|
|
}
|
|
|
|
|
|
|
|
private MultiMediaSession GetSessionByType(MultiMediaOperationType type)
|
|
|
|
{
|
|
|
|
foreach (MultiMediaSession session in _sessionList)
|
|
|
|
{
|
|
|
|
if (session.Type == type)
|
|
|
|
{
|
|
|
|
return session;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return null;
|
|
|
|
}
|
|
|
|
|
|
|
|
private uint Register(MultiMediaOperationType type, int fgmId, bool isAutoClearEvent)
|
|
|
|
{
|
|
|
|
lock (_sessionListLock)
|
|
|
|
{
|
|
|
|
// Nintendo ignore the fgm id as the other interfaces were deprecated.
|
|
|
|
MultiMediaSession session = new MultiMediaSession(_uniqueId++, type, isAutoClearEvent);
|
|
|
|
|
|
|
|
_sessionList.Add(session);
|
|
|
|
|
|
|
|
return session.Id;
|
|
|
|
}
|
|
|
|
}
|
2018-05-26 20:49:21 +00:00
|
|
|
}
|
|
|
|
}
|