ControllerApplet.cs 5.7 KB

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