2020-04-03 02:10:02 +02:00
|
|
|
using System;
|
|
|
|
using System.IO;
|
|
|
|
using System.Runtime.CompilerServices;
|
|
|
|
using System.Runtime.InteropServices;
|
|
|
|
using Ryujinx.Common.Logging;
|
|
|
|
using Ryujinx.HLE.HOS.Services.Hid;
|
2021-05-02 22:01:30 +02:00
|
|
|
using Ryujinx.HLE.HOS.Services.Hid.Types;
|
2020-04-03 02:10:02 +02:00
|
|
|
using Ryujinx.HLE.HOS.Services.Am.AppletAE;
|
|
|
|
|
|
|
|
using static Ryujinx.HLE.HOS.Services.Hid.HidServer.HidUtils;
|
|
|
|
|
|
|
|
namespace Ryujinx.HLE.HOS.Applets
|
|
|
|
{
|
|
|
|
internal class ControllerApplet : IApplet
|
|
|
|
{
|
|
|
|
private Horizon _system;
|
|
|
|
|
|
|
|
private AppletSession _normalSession;
|
|
|
|
|
|
|
|
public event EventHandler AppletStateChanged;
|
|
|
|
|
|
|
|
public ControllerApplet(Horizon system)
|
|
|
|
{
|
|
|
|
_system = system;
|
|
|
|
}
|
|
|
|
|
2021-11-01 23:38:13 +01:00
|
|
|
public ResultCode Start(AppletSession normalSession, AppletSession interactiveSession)
|
2020-04-03 02:10:02 +02:00
|
|
|
{
|
|
|
|
_normalSession = normalSession;
|
|
|
|
|
|
|
|
byte[] launchParams = _normalSession.Pop();
|
|
|
|
byte[] controllerSupportArgPrivate = _normalSession.Pop();
|
|
|
|
ControllerSupportArgPrivate privateArg = IApplet.ReadStruct<ControllerSupportArgPrivate>(controllerSupportArgPrivate);
|
|
|
|
|
2020-08-23 22:54:11 +02:00
|
|
|
Logger.Stub?.PrintStub(LogClass.ServiceHid, $"ControllerApplet ArgPriv {privateArg.PrivateSize} {privateArg.ArgSize} {privateArg.Mode} " +
|
2020-04-03 02:10:02 +02:00
|
|
|
$"HoldType:{(NpadJoyHoldType)privateArg.NpadJoyHoldType} StyleSets:{(ControllerType)privateArg.NpadStyleSet}");
|
|
|
|
|
|
|
|
if (privateArg.Mode != ControllerSupportMode.ShowControllerSupport)
|
|
|
|
{
|
|
|
|
_normalSession.Push(BuildResponse()); // Dummy response for other modes
|
|
|
|
AppletStateChanged?.Invoke(this, null);
|
|
|
|
|
|
|
|
return ResultCode.Success;
|
|
|
|
}
|
|
|
|
|
|
|
|
byte[] controllerSupportArg = _normalSession.Pop();
|
|
|
|
|
|
|
|
ControllerSupportArgHeader argHeader;
|
|
|
|
|
2020-08-23 22:54:11 +02:00
|
|
|
if (privateArg.ArgSize == Marshal.SizeOf<ControllerSupportArgV7>())
|
2020-04-03 02:10:02 +02:00
|
|
|
{
|
2020-08-23 22:54:11 +02:00
|
|
|
ControllerSupportArgV7 arg = IApplet.ReadStruct<ControllerSupportArgV7>(controllerSupportArg);
|
2020-04-03 02:10:02 +02:00
|
|
|
argHeader = arg.Header;
|
2020-08-23 22:54:11 +02:00
|
|
|
|
|
|
|
Logger.Stub?.PrintStub(LogClass.ServiceHid, $"ControllerSupportArg Version 7 EnableExplainText={arg.EnableExplainText != 0}");
|
|
|
|
// Read enable text here?
|
|
|
|
}
|
|
|
|
else if (privateArg.ArgSize == Marshal.SizeOf<ControllerSupportArgVPre7>())
|
|
|
|
{
|
|
|
|
ControllerSupportArgVPre7 arg = IApplet.ReadStruct<ControllerSupportArgVPre7>(controllerSupportArg);
|
|
|
|
argHeader = arg.Header;
|
|
|
|
|
|
|
|
Logger.Stub?.PrintStub(LogClass.ServiceHid, $"ControllerSupportArg Version Pre-7 EnableExplainText={arg.EnableExplainText != 0}");
|
2020-04-03 02:10:02 +02:00
|
|
|
// Read enable text here?
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2020-08-23 22:54:11 +02:00
|
|
|
Logger.Stub?.PrintStub(LogClass.ServiceHid, $"ControllerSupportArg Version Unknown");
|
2020-04-03 02:10:02 +02:00
|
|
|
|
|
|
|
argHeader = IApplet.ReadStruct<ControllerSupportArgHeader>(controllerSupportArg); // Read just the header
|
|
|
|
}
|
|
|
|
|
2020-08-23 22:54:11 +02:00
|
|
|
int playerMin = argHeader.PlayerCountMin;
|
|
|
|
int playerMax = argHeader.PlayerCountMax;
|
2022-08-15 09:46:08 +02:00
|
|
|
bool singleMode = argHeader.EnableSingleMode != 0;
|
2020-08-23 22:54:11 +02:00
|
|
|
|
|
|
|
Logger.Stub?.PrintStub(LogClass.ServiceHid, $"ControllerApplet Arg {playerMin} {playerMax} {argHeader.EnableTakeOverConnection} {argHeader.EnableSingleMode}");
|
2020-04-03 02:10:02 +02:00
|
|
|
|
2022-08-15 09:46:08 +02:00
|
|
|
if (singleMode)
|
|
|
|
{
|
|
|
|
// Applications can set an arbitrary player range even with SingleMode, so clamp it
|
|
|
|
playerMin = playerMax = 1;
|
|
|
|
}
|
|
|
|
|
2020-08-23 22:54:11 +02:00
|
|
|
int configuredCount = 0;
|
|
|
|
PlayerIndex primaryIndex = PlayerIndex.Unknown;
|
|
|
|
while (!_system.Device.Hid.Npads.Validate(playerMin, playerMax, (ControllerType)privateArg.NpadStyleSet, out configuredCount, out primaryIndex))
|
2020-04-03 02:10:02 +02:00
|
|
|
{
|
2020-08-23 22:54:11 +02:00
|
|
|
ControllerAppletUiArgs uiArgs = new ControllerAppletUiArgs
|
|
|
|
{
|
|
|
|
PlayerCountMin = playerMin,
|
|
|
|
PlayerCountMax = playerMax,
|
|
|
|
SupportedStyles = (ControllerType)privateArg.NpadStyleSet,
|
|
|
|
SupportedPlayers = _system.Device.Hid.Npads.GetSupportedPlayers(),
|
|
|
|
IsDocked = _system.State.DockedMode
|
|
|
|
};
|
|
|
|
|
|
|
|
if (!_system.Device.UiHandler.DisplayMessageDialog(uiArgs))
|
|
|
|
{
|
|
|
|
break;
|
|
|
|
}
|
2020-04-03 02:10:02 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
ControllerSupportResultInfo result = new ControllerSupportResultInfo
|
|
|
|
{
|
2020-08-23 22:54:11 +02:00
|
|
|
PlayerCount = (sbyte)configuredCount,
|
|
|
|
SelectedId = (uint)GetNpadIdTypeFromIndex(primaryIndex)
|
2020-04-03 02:10:02 +02:00
|
|
|
};
|
|
|
|
|
2020-08-04 01:32:53 +02:00
|
|
|
Logger.Stub?.PrintStub(LogClass.ServiceHid, $"ControllerApplet ReturnResult {result.PlayerCount} {result.SelectedId}");
|
2020-04-03 02:10:02 +02:00
|
|
|
|
|
|
|
_normalSession.Push(BuildResponse(result));
|
|
|
|
AppletStateChanged?.Invoke(this, null);
|
|
|
|
|
2021-04-17 18:57:03 +02:00
|
|
|
_system.ReturnFocus();
|
|
|
|
|
2020-04-03 02:10:02 +02:00
|
|
|
return ResultCode.Success;
|
|
|
|
}
|
|
|
|
|
|
|
|
public ResultCode GetResult()
|
|
|
|
{
|
|
|
|
return ResultCode.Success;
|
|
|
|
}
|
|
|
|
|
|
|
|
private byte[] BuildResponse(ControllerSupportResultInfo result)
|
|
|
|
{
|
|
|
|
using (MemoryStream stream = new MemoryStream())
|
|
|
|
using (BinaryWriter writer = new BinaryWriter(stream))
|
|
|
|
{
|
|
|
|
writer.Write(MemoryMarshal.AsBytes(MemoryMarshal.CreateReadOnlySpan(ref result, Unsafe.SizeOf<ControllerSupportResultInfo>())));
|
|
|
|
|
|
|
|
return stream.ToArray();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
private byte[] BuildResponse()
|
|
|
|
{
|
|
|
|
using (MemoryStream stream = new MemoryStream())
|
|
|
|
using (BinaryWriter writer = new BinaryWriter(stream))
|
|
|
|
{
|
|
|
|
writer.Write((ulong)ResultCode.Success);
|
|
|
|
|
|
|
|
return stream.ToArray();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|