IAudioRendererManager.cs 7.7 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201
  1. using Ryujinx.Audio;
  2. using Ryujinx.Common.Logging;
  3. using Ryujinx.HLE.HOS.Ipc;
  4. using Ryujinx.HLE.HOS.Services.Aud.AudioRenderer;
  5. using Ryujinx.HLE.Utilities;
  6. using System.Collections.Generic;
  7. using static Ryujinx.HLE.HOS.ErrorCode;
  8. namespace Ryujinx.HLE.HOS.Services.Aud
  9. {
  10. class IAudioRendererManager : IpcService
  11. {
  12. private const int Rev0Magic = ('R' << 0) |
  13. ('E' << 8) |
  14. ('V' << 16) |
  15. ('0' << 24);
  16. private const int Rev = 5;
  17. public const int RevMagic = Rev0Magic + (Rev << 24);
  18. private Dictionary<int, ServiceProcessRequest> _commands;
  19. public override IReadOnlyDictionary<int, ServiceProcessRequest> Commands => _commands;
  20. public IAudioRendererManager()
  21. {
  22. _commands = new Dictionary<int, ServiceProcessRequest>
  23. {
  24. { 0, OpenAudioRenderer },
  25. { 1, GetAudioRendererWorkBufferSize },
  26. { 2, GetAudioDeviceService },
  27. { 4, GetAudioDeviceServiceWithRevisionInfo }
  28. };
  29. }
  30. public long OpenAudioRenderer(ServiceCtx context)
  31. {
  32. IAalOutput audioOut = context.Device.AudioOut;
  33. AudioRendererParameter Params = GetAudioRendererParameter(context);
  34. MakeObject(context, new IAudioRenderer(
  35. context.Device.System,
  36. context.Memory,
  37. audioOut,
  38. Params));
  39. return 0;
  40. }
  41. public long GetAudioRendererWorkBufferSize(ServiceCtx context)
  42. {
  43. AudioRendererParameter Params = GetAudioRendererParameter(context);
  44. int revision = (Params.Revision - Rev0Magic) >> 24;
  45. if (revision <= Rev)
  46. {
  47. bool isSplitterSupported = revision >= 3;
  48. bool isVariadicCommandBufferSizeSupported = revision >= 5;
  49. long size;
  50. size = IntUtils.AlignUp(Params.Unknown8 * 4, 64);
  51. size += Params.MixCount * 0x400;
  52. size += (Params.MixCount + 1) * 0x940;
  53. size += Params.VoiceCount * 0x3F0;
  54. size += IntUtils.AlignUp((Params.MixCount + 1) * 8, 16);
  55. size += IntUtils.AlignUp(Params.VoiceCount * 8, 16);
  56. size += IntUtils.AlignUp(
  57. ((Params.SinkCount + Params.MixCount) * 0x3C0 + Params.SampleCount * 4) *
  58. (Params.Unknown8 + 6), 64);
  59. size += (Params.SinkCount + Params.MixCount) * 0x2C0;
  60. size += (Params.EffectCount + Params.VoiceCount * 4) * 0x30 + 0x50;
  61. if (isSplitterSupported)
  62. {
  63. size += IntUtils.AlignUp((
  64. NodeStatesGetWorkBufferSize(Params.MixCount + 1) +
  65. EdgeMatrixGetWorkBufferSize(Params.MixCount + 1)), 16);
  66. size += Params.SplitterDestinationDataCount * 0xE0;
  67. size += Params.SplitterCount * 0x20;
  68. size += IntUtils.AlignUp(Params.SplitterDestinationDataCount * 4, 16);
  69. }
  70. size = Params.EffectCount * 0x4C0 +
  71. Params.SinkCount * 0x170 +
  72. Params.VoiceCount * 0x100 +
  73. IntUtils.AlignUp(size, 64) + 0x40;
  74. if (Params.PerformanceManagerCount >= 1)
  75. {
  76. size += (((Params.EffectCount +
  77. Params.SinkCount +
  78. Params.VoiceCount +
  79. Params.MixCount + 1) * 16 + 0x658) *
  80. (Params.PerformanceManagerCount + 1) + 0x13F) & ~0x3FL;
  81. }
  82. if (isVariadicCommandBufferSizeSupported)
  83. {
  84. size += Params.EffectCount * 0x840 +
  85. Params.MixCount * 0x5A38 +
  86. Params.SinkCount * 0x148 +
  87. Params.SplitterDestinationDataCount * 0x540 +
  88. Params.VoiceCount * (Params.SplitterCount * 0x68 + 0x2E0) +
  89. ((Params.VoiceCount + Params.MixCount + Params.EffectCount + Params.SinkCount + 0x65) << 6) + 0x3F8 + 0x7E;
  90. }
  91. else
  92. {
  93. size += 0x1807E;
  94. }
  95. size = size & ~0xFFFL;
  96. context.ResponseData.Write(size);
  97. Logger.PrintDebug(LogClass.ServiceAudio, $"WorkBufferSize is 0x{size:x16}.");
  98. return 0;
  99. }
  100. else
  101. {
  102. context.ResponseData.Write(0L);
  103. Logger.PrintWarning(LogClass.ServiceAudio, $"Library Revision 0x{Params.Revision:x8} is not supported!");
  104. return MakeError(ErrorModule.Audio, AudErr.UnsupportedRevision);
  105. }
  106. }
  107. private AudioRendererParameter GetAudioRendererParameter(ServiceCtx context)
  108. {
  109. AudioRendererParameter Params = new AudioRendererParameter();
  110. Params.SampleRate = context.RequestData.ReadInt32();
  111. Params.SampleCount = context.RequestData.ReadInt32();
  112. Params.Unknown8 = context.RequestData.ReadInt32();
  113. Params.MixCount = context.RequestData.ReadInt32();
  114. Params.VoiceCount = context.RequestData.ReadInt32();
  115. Params.SinkCount = context.RequestData.ReadInt32();
  116. Params.EffectCount = context.RequestData.ReadInt32();
  117. Params.PerformanceManagerCount = context.RequestData.ReadInt32();
  118. Params.VoiceDropEnable = context.RequestData.ReadInt32();
  119. Params.SplitterCount = context.RequestData.ReadInt32();
  120. Params.SplitterDestinationDataCount = context.RequestData.ReadInt32();
  121. Params.Unknown2C = context.RequestData.ReadInt32();
  122. Params.Revision = context.RequestData.ReadInt32();
  123. return Params;
  124. }
  125. private static int NodeStatesGetWorkBufferSize(int value)
  126. {
  127. int result = IntUtils.AlignUp(value, 64);
  128. if (result < 0)
  129. {
  130. result |= 7;
  131. }
  132. return 4 * (value * value) + 0x12 * value + 2 * (result / 8);
  133. }
  134. private static int EdgeMatrixGetWorkBufferSize(int value)
  135. {
  136. int result = IntUtils.AlignUp(value * value, 64);
  137. if (result < 0)
  138. {
  139. result |= 7;
  140. }
  141. return result / 8;
  142. }
  143. // GetAudioDeviceService(nn::applet::AppletResourceUserId) -> object<nn::audio::detail::IAudioDevice>
  144. public long GetAudioDeviceService(ServiceCtx context)
  145. {
  146. long appletResourceUserId = context.RequestData.ReadInt64();
  147. MakeObject(context, new IAudioDevice(context.Device.System));
  148. return 0;
  149. }
  150. // GetAudioDeviceServiceWithRevisionInfo(nn::applet::AppletResourceUserId, u32) -> object<nn::audio::detail::IAudioDevice>
  151. private long GetAudioDeviceServiceWithRevisionInfo(ServiceCtx context)
  152. {
  153. long appletResourceUserId = context.RequestData.ReadInt64();
  154. int revisionInfo = context.RequestData.ReadInt32();
  155. Logger.PrintStub(LogClass.ServiceAudio, new { appletResourceUserId, revisionInfo });
  156. return GetAudioDeviceService(context);
  157. }
  158. }
  159. }