ControllerApplet.cs 5.6 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146
  1. using Ryujinx.Common.Logging;
  2. using Ryujinx.HLE.HOS.Services.Am.AppletAE;
  3. using Ryujinx.HLE.HOS.Services.Hid;
  4. using Ryujinx.HLE.HOS.Services.Hid.Types;
  5. using System;
  6. using System.IO;
  7. using System.Runtime.CompilerServices;
  8. using System.Runtime.InteropServices;
  9. using static Ryujinx.HLE.HOS.Services.Hid.HidServer.HidUtils;
  10. namespace Ryujinx.HLE.HOS.Applets
  11. {
  12. internal class ControllerApplet : IApplet
  13. {
  14. private Horizon _system;
  15. private AppletSession _normalSession;
  16. public event EventHandler AppletStateChanged;
  17. public ControllerApplet(Horizon system)
  18. {
  19. _system = system;
  20. }
  21. public ResultCode Start(AppletSession normalSession, AppletSession interactiveSession)
  22. {
  23. _normalSession = normalSession;
  24. byte[] launchParams = _normalSession.Pop();
  25. byte[] controllerSupportArgPrivate = _normalSession.Pop();
  26. ControllerSupportArgPrivate privateArg = IApplet.ReadStruct<ControllerSupportArgPrivate>(controllerSupportArgPrivate);
  27. Logger.Stub?.PrintStub(LogClass.ServiceHid, $"ControllerApplet ArgPriv {privateArg.PrivateSize} {privateArg.ArgSize} {privateArg.Mode} " +
  28. $"HoldType:{(NpadJoyHoldType)privateArg.NpadJoyHoldType} StyleSets:{(ControllerType)privateArg.NpadStyleSet}");
  29. if (privateArg.Mode != ControllerSupportMode.ShowControllerSupport)
  30. {
  31. _normalSession.Push(BuildResponse()); // Dummy response for other modes
  32. AppletStateChanged?.Invoke(this, null);
  33. return ResultCode.Success;
  34. }
  35. byte[] controllerSupportArg = _normalSession.Pop();
  36. ControllerSupportArgHeader argHeader;
  37. if (privateArg.ArgSize == Marshal.SizeOf<ControllerSupportArgV7>())
  38. {
  39. ControllerSupportArgV7 arg = IApplet.ReadStruct<ControllerSupportArgV7>(controllerSupportArg);
  40. argHeader = arg.Header;
  41. Logger.Stub?.PrintStub(LogClass.ServiceHid, $"ControllerSupportArg Version 7 EnableExplainText={arg.EnableExplainText != 0}");
  42. // Read enable text here?
  43. }
  44. else if (privateArg.ArgSize == Marshal.SizeOf<ControllerSupportArgVPre7>())
  45. {
  46. ControllerSupportArgVPre7 arg = IApplet.ReadStruct<ControllerSupportArgVPre7>(controllerSupportArg);
  47. argHeader = arg.Header;
  48. Logger.Stub?.PrintStub(LogClass.ServiceHid, $"ControllerSupportArg Version Pre-7 EnableExplainText={arg.EnableExplainText != 0}");
  49. // Read enable text here?
  50. }
  51. else
  52. {
  53. Logger.Stub?.PrintStub(LogClass.ServiceHid, $"ControllerSupportArg Version Unknown");
  54. argHeader = IApplet.ReadStruct<ControllerSupportArgHeader>(controllerSupportArg); // Read just the header
  55. }
  56. int playerMin = argHeader.PlayerCountMin;
  57. int playerMax = argHeader.PlayerCountMax;
  58. bool singleMode = argHeader.EnableSingleMode != 0;
  59. Logger.Stub?.PrintStub(LogClass.ServiceHid, $"ControllerApplet Arg {playerMin} {playerMax} {argHeader.EnableTakeOverConnection} {argHeader.EnableSingleMode}");
  60. if (singleMode)
  61. {
  62. // Applications can set an arbitrary player range even with SingleMode, so clamp it
  63. playerMin = playerMax = 1;
  64. }
  65. int configuredCount = 0;
  66. PlayerIndex primaryIndex = PlayerIndex.Unknown;
  67. while (!_system.Device.Hid.Npads.Validate(playerMin, playerMax, (ControllerType)privateArg.NpadStyleSet, out configuredCount, out primaryIndex))
  68. {
  69. ControllerAppletUiArgs uiArgs = new ControllerAppletUiArgs
  70. {
  71. PlayerCountMin = playerMin,
  72. PlayerCountMax = playerMax,
  73. SupportedStyles = (ControllerType)privateArg.NpadStyleSet,
  74. SupportedPlayers = _system.Device.Hid.Npads.GetSupportedPlayers(),
  75. IsDocked = _system.State.DockedMode
  76. };
  77. if (!_system.Device.UiHandler.DisplayMessageDialog(uiArgs))
  78. {
  79. break;
  80. }
  81. }
  82. ControllerSupportResultInfo result = new ControllerSupportResultInfo
  83. {
  84. PlayerCount = (sbyte)configuredCount,
  85. SelectedId = (uint)GetNpadIdTypeFromIndex(primaryIndex)
  86. };
  87. Logger.Stub?.PrintStub(LogClass.ServiceHid, $"ControllerApplet ReturnResult {result.PlayerCount} {result.SelectedId}");
  88. _normalSession.Push(BuildResponse(result));
  89. AppletStateChanged?.Invoke(this, null);
  90. _system.ReturnFocus();
  91. return ResultCode.Success;
  92. }
  93. public ResultCode GetResult()
  94. {
  95. return ResultCode.Success;
  96. }
  97. private byte[] BuildResponse(ControllerSupportResultInfo result)
  98. {
  99. using (MemoryStream stream = new MemoryStream())
  100. using (BinaryWriter writer = new BinaryWriter(stream))
  101. {
  102. writer.Write(MemoryMarshal.AsBytes(MemoryMarshal.CreateReadOnlySpan(ref result, Unsafe.SizeOf<ControllerSupportResultInfo>())));
  103. return stream.ToArray();
  104. }
  105. }
  106. private byte[] BuildResponse()
  107. {
  108. using (MemoryStream stream = new MemoryStream())
  109. using (BinaryWriter writer = new BinaryWriter(stream))
  110. {
  111. writer.Write((ulong)ResultCode.Success);
  112. return stream.ToArray();
  113. }
  114. }
  115. }
  116. }