IAudioRendererManager.cs 6.2 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173
  1. using Ryujinx.Audio;
  2. using Ryujinx.HLE.HOS.Ipc;
  3. using Ryujinx.HLE.HOS.Services.Aud.AudioRenderer;
  4. using Ryujinx.HLE.Logging;
  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 = 4;
  17. public const int RevMagic = Rev0Magic + (Rev << 24);
  18. private Dictionary<int, ServiceProcessRequest> m_Commands;
  19. public override IReadOnlyDictionary<int, ServiceProcessRequest> Commands => m_Commands;
  20. public IAudioRendererManager()
  21. {
  22. m_Commands = new Dictionary<int, ServiceProcessRequest>()
  23. {
  24. { 0, OpenAudioRenderer },
  25. { 1, GetAudioRendererWorkBufferSize },
  26. { 2, GetAudioDevice }
  27. };
  28. }
  29. public long OpenAudioRenderer(ServiceCtx Context)
  30. {
  31. IAalOutput AudioOut = Context.Device.AudioOut;
  32. AudioRendererParameter Params = GetAudioRendererParameter(Context);
  33. MakeObject(Context, new IAudioRenderer(
  34. Context.Device.System,
  35. Context.Memory,
  36. AudioOut,
  37. Params));
  38. return 0;
  39. }
  40. public long GetAudioRendererWorkBufferSize(ServiceCtx Context)
  41. {
  42. AudioRendererParameter Params = GetAudioRendererParameter(Context);
  43. int Revision = (Params.Revision - Rev0Magic) >> 24;
  44. if (Revision <= Rev)
  45. {
  46. bool IsSplitterSupported = Revision >= 3;
  47. long Size;
  48. Size = IntUtils.AlignUp(Params.Unknown8 * 4, 64);
  49. Size += Params.MixCount * 0x400;
  50. Size += (Params.MixCount + 1) * 0x940;
  51. Size += Params.VoiceCount * 0x3F0;
  52. Size += IntUtils.AlignUp((Params.MixCount + 1) * 8, 16);
  53. Size += IntUtils.AlignUp(Params.VoiceCount * 8, 16);
  54. Size += IntUtils.AlignUp(
  55. ((Params.SinkCount + Params.MixCount) * 0x3C0 + Params.SampleCount * 4) *
  56. (Params.Unknown8 + 6), 64);
  57. Size += (Params.SinkCount + Params.MixCount) * 0x2C0;
  58. Size += (Params.EffectCount + Params.VoiceCount * 4) * 0x30 + 0x50;
  59. if (IsSplitterSupported)
  60. {
  61. Size += IntUtils.AlignUp((
  62. NodeStatesGetWorkBufferSize(Params.MixCount + 1) +
  63. EdgeMatrixGetWorkBufferSize(Params.MixCount + 1)), 16);
  64. Size += Params.SplitterDestinationDataCount * 0xE0;
  65. Size += Params.SplitterCount * 0x20;
  66. Size += IntUtils.AlignUp(Params.SplitterDestinationDataCount * 4, 16);
  67. }
  68. Size = Params.EffectCount * 0x4C0 +
  69. Params.SinkCount * 0x170 +
  70. Params.VoiceCount * 0x100 +
  71. IntUtils.AlignUp(Size, 64) + 0x40;
  72. if (Params.PerformanceManagerCount >= 1)
  73. {
  74. Size += (((Params.EffectCount +
  75. Params.SinkCount +
  76. Params.VoiceCount +
  77. Params.MixCount + 1) * 16 + 0x658) *
  78. (Params.PerformanceManagerCount + 1) + 0x13F) & ~0x3FL;
  79. }
  80. Size = (Size + 0x1907D) & ~0xFFFL;
  81. Context.ResponseData.Write(Size);
  82. Context.Device.Log.PrintDebug(LogClass.ServiceAudio, $"WorkBufferSize is 0x{Size:x16}.");
  83. return 0;
  84. }
  85. else
  86. {
  87. Context.ResponseData.Write(0L);
  88. Context.Device.Log.PrintWarning(LogClass.ServiceAudio, $"Library Revision 0x{Params.Revision:x8} is not supported!");
  89. return MakeError(ErrorModule.Audio, AudErr.UnsupportedRevision);
  90. }
  91. }
  92. private AudioRendererParameter GetAudioRendererParameter(ServiceCtx Context)
  93. {
  94. AudioRendererParameter Params = new AudioRendererParameter();
  95. Params.SampleRate = Context.RequestData.ReadInt32();
  96. Params.SampleCount = Context.RequestData.ReadInt32();
  97. Params.Unknown8 = Context.RequestData.ReadInt32();
  98. Params.MixCount = Context.RequestData.ReadInt32();
  99. Params.VoiceCount = Context.RequestData.ReadInt32();
  100. Params.SinkCount = Context.RequestData.ReadInt32();
  101. Params.EffectCount = Context.RequestData.ReadInt32();
  102. Params.PerformanceManagerCount = Context.RequestData.ReadInt32();
  103. Params.VoiceDropEnable = Context.RequestData.ReadInt32();
  104. Params.SplitterCount = Context.RequestData.ReadInt32();
  105. Params.SplitterDestinationDataCount = Context.RequestData.ReadInt32();
  106. Params.Unknown2C = Context.RequestData.ReadInt32();
  107. Params.Revision = Context.RequestData.ReadInt32();
  108. return Params;
  109. }
  110. private static int NodeStatesGetWorkBufferSize(int Value)
  111. {
  112. int Result = IntUtils.AlignUp(Value, 64);
  113. if (Result < 0)
  114. {
  115. Result |= 7;
  116. }
  117. return 4 * (Value * Value) + 0x12 * Value + 2 * (Result / 8);
  118. }
  119. private static int EdgeMatrixGetWorkBufferSize(int Value)
  120. {
  121. int Result = IntUtils.AlignUp(Value * Value, 64);
  122. if (Result < 0)
  123. {
  124. Result |= 7;
  125. }
  126. return Result / 8;
  127. }
  128. public long GetAudioDevice(ServiceCtx Context)
  129. {
  130. long UserId = Context.RequestData.ReadInt64();
  131. MakeObject(Context, new IAudioDevice(Context.Device.System));
  132. return 0;
  133. }
  134. }
  135. }