2018-07-15 04:57:41 +02:00
|
|
|
using Ryujinx.Audio;
|
2018-10-17 19:15:50 +02:00
|
|
|
using Ryujinx.Common.Logging;
|
2018-08-17 01:47:36 +02:00
|
|
|
using Ryujinx.HLE.HOS.Services.Aud.AudioRenderer;
|
|
|
|
using Ryujinx.HLE.Utilities;
|
2018-07-15 04:57:41 +02:00
|
|
|
|
2018-08-17 01:47:36 +02:00
|
|
|
namespace Ryujinx.HLE.HOS.Services.Aud
|
2018-02-25 05:34:16 +01:00
|
|
|
{
|
2019-07-10 17:59:54 +02:00
|
|
|
[Service("audren:u")]
|
2018-03-22 00:30:10 +01:00
|
|
|
class IAudioRendererManager : IpcService
|
2018-02-25 05:34:16 +01:00
|
|
|
{
|
2018-05-07 21:03:30 +02:00
|
|
|
private const int Rev0Magic = ('R' << 0) |
|
|
|
|
('E' << 8) |
|
|
|
|
('V' << 16) |
|
|
|
|
('0' << 24);
|
|
|
|
|
2019-04-24 16:22:06 +02:00
|
|
|
private const int Rev = 5;
|
2018-07-15 04:57:41 +02:00
|
|
|
|
2018-07-15 05:42:59 +02:00
|
|
|
public const int RevMagic = Rev0Magic + (Rev << 24);
|
2018-07-15 04:57:41 +02:00
|
|
|
|
2019-07-12 03:13:43 +02:00
|
|
|
public IAudioRendererManager(ServiceCtx context) { }
|
2018-02-25 05:34:16 +01:00
|
|
|
|
2019-07-12 03:13:43 +02:00
|
|
|
[Command(0)]
|
|
|
|
// OpenAudioRenderer(nn::audio::detail::AudioRendererParameterInternal, u64, nn::applet::AppletResourceUserId, pid, handle<copy>, handle<copy>)
|
|
|
|
// -> object<nn::audio::detail::IAudioRenderer>
|
2019-07-14 21:04:38 +02:00
|
|
|
public ResultCode OpenAudioRenderer(ServiceCtx context)
|
2018-02-25 05:34:16 +01:00
|
|
|
{
|
2018-12-06 12:16:24 +01:00
|
|
|
IAalOutput audioOut = context.Device.AudioOut;
|
2018-05-04 16:52:07 +02:00
|
|
|
|
2018-12-06 12:16:24 +01:00
|
|
|
AudioRendererParameter Params = GetAudioRendererParameter(context);
|
2018-06-23 07:00:14 +02:00
|
|
|
|
2018-12-06 12:16:24 +01:00
|
|
|
MakeObject(context, new IAudioRenderer(
|
|
|
|
context.Device.System,
|
|
|
|
context.Memory,
|
|
|
|
audioOut,
|
2018-09-19 01:36:43 +02:00
|
|
|
Params));
|
2018-02-25 05:34:16 +01:00
|
|
|
|
2019-07-14 21:04:38 +02:00
|
|
|
return ResultCode.Success;
|
2018-02-25 05:34:16 +01:00
|
|
|
}
|
|
|
|
|
2019-07-12 03:13:43 +02:00
|
|
|
[Command(1)]
|
|
|
|
// GetWorkBufferSize(nn::audio::detail::AudioRendererParameterInternal) -> u64
|
2019-07-14 21:04:38 +02:00
|
|
|
public ResultCode GetAudioRendererWorkBufferSize(ServiceCtx context)
|
2018-02-25 05:34:16 +01:00
|
|
|
{
|
2018-12-06 12:16:24 +01:00
|
|
|
AudioRendererParameter Params = GetAudioRendererParameter(context);
|
2018-07-15 04:57:41 +02:00
|
|
|
|
2018-12-06 12:16:24 +01:00
|
|
|
int revision = (Params.Revision - Rev0Magic) >> 24;
|
2018-07-15 04:57:41 +02:00
|
|
|
|
2018-12-06 12:16:24 +01:00
|
|
|
if (revision <= Rev)
|
2018-05-02 02:02:28 +02:00
|
|
|
{
|
2019-04-24 16:22:06 +02:00
|
|
|
bool isSplitterSupported = revision >= 3;
|
|
|
|
bool isVariadicCommandBufferSizeSupported = revision >= 5;
|
|
|
|
|
2018-12-06 12:16:24 +01:00
|
|
|
long size;
|
2018-07-15 04:57:41 +02:00
|
|
|
|
2018-12-06 12:16:24 +01:00
|
|
|
size = IntUtils.AlignUp(Params.Unknown8 * 4, 64);
|
|
|
|
size += Params.MixCount * 0x400;
|
|
|
|
size += (Params.MixCount + 1) * 0x940;
|
|
|
|
size += Params.VoiceCount * 0x3F0;
|
|
|
|
size += IntUtils.AlignUp((Params.MixCount + 1) * 8, 16);
|
|
|
|
size += IntUtils.AlignUp(Params.VoiceCount * 8, 16);
|
|
|
|
size += IntUtils.AlignUp(
|
2018-07-15 04:57:41 +02:00
|
|
|
((Params.SinkCount + Params.MixCount) * 0x3C0 + Params.SampleCount * 4) *
|
|
|
|
(Params.Unknown8 + 6), 64);
|
2018-12-06 12:16:24 +01:00
|
|
|
size += (Params.SinkCount + Params.MixCount) * 0x2C0;
|
|
|
|
size += (Params.EffectCount + Params.VoiceCount * 4) * 0x30 + 0x50;
|
2018-07-15 04:57:41 +02:00
|
|
|
|
2018-12-06 12:16:24 +01:00
|
|
|
if (isSplitterSupported)
|
2018-05-04 16:52:07 +02:00
|
|
|
{
|
2018-12-06 12:16:24 +01:00
|
|
|
size += IntUtils.AlignUp((
|
2018-07-15 04:57:41 +02:00
|
|
|
NodeStatesGetWorkBufferSize(Params.MixCount + 1) +
|
|
|
|
EdgeMatrixGetWorkBufferSize(Params.MixCount + 1)), 16);
|
|
|
|
|
2018-12-06 12:16:24 +01:00
|
|
|
size += Params.SplitterDestinationDataCount * 0xE0;
|
|
|
|
size += Params.SplitterCount * 0x20;
|
|
|
|
size += IntUtils.AlignUp(Params.SplitterDestinationDataCount * 4, 16);
|
2018-05-04 16:52:07 +02:00
|
|
|
}
|
|
|
|
|
2018-12-06 12:16:24 +01:00
|
|
|
size = Params.EffectCount * 0x4C0 +
|
2018-07-15 04:57:41 +02:00
|
|
|
Params.SinkCount * 0x170 +
|
|
|
|
Params.VoiceCount * 0x100 +
|
2018-12-06 12:16:24 +01:00
|
|
|
IntUtils.AlignUp(size, 64) + 0x40;
|
2018-05-04 16:52:07 +02:00
|
|
|
|
2018-07-15 04:57:41 +02:00
|
|
|
if (Params.PerformanceManagerCount >= 1)
|
2018-05-02 02:02:28 +02:00
|
|
|
{
|
2018-12-06 12:16:24 +01:00
|
|
|
size += (((Params.EffectCount +
|
2018-07-15 04:57:41 +02:00
|
|
|
Params.SinkCount +
|
|
|
|
Params.VoiceCount +
|
|
|
|
Params.MixCount + 1) * 16 + 0x658) *
|
|
|
|
(Params.PerformanceManagerCount + 1) + 0x13F) & ~0x3FL;
|
2018-05-02 02:02:28 +02:00
|
|
|
}
|
|
|
|
|
2019-04-24 16:22:06 +02:00
|
|
|
if (isVariadicCommandBufferSizeSupported)
|
|
|
|
{
|
|
|
|
size += Params.EffectCount * 0x840 +
|
|
|
|
Params.MixCount * 0x5A38 +
|
|
|
|
Params.SinkCount * 0x148 +
|
|
|
|
Params.SplitterDestinationDataCount * 0x540 +
|
|
|
|
Params.VoiceCount * (Params.SplitterCount * 0x68 + 0x2E0) +
|
|
|
|
((Params.VoiceCount + Params.MixCount + Params.EffectCount + Params.SinkCount + 0x65) << 6) + 0x3F8 + 0x7E;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
size += 0x1807E;
|
|
|
|
}
|
|
|
|
|
|
|
|
size = size & ~0xFFFL;
|
2018-05-02 02:02:28 +02:00
|
|
|
|
2018-12-06 12:16:24 +01:00
|
|
|
context.ResponseData.Write(size);
|
2018-05-02 02:02:28 +02:00
|
|
|
|
2018-12-06 12:16:24 +01:00
|
|
|
Logger.PrintDebug(LogClass.ServiceAudio, $"WorkBufferSize is 0x{size:x16}.");
|
2018-05-02 02:02:28 +02:00
|
|
|
|
2019-07-14 21:04:38 +02:00
|
|
|
return ResultCode.Success;
|
2018-05-02 02:02:28 +02:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2018-12-06 12:16:24 +01:00
|
|
|
context.ResponseData.Write(0L);
|
2018-05-02 02:02:28 +02:00
|
|
|
|
2018-10-17 19:15:50 +02:00
|
|
|
Logger.PrintWarning(LogClass.ServiceAudio, $"Library Revision 0x{Params.Revision:x8} is not supported!");
|
2018-05-02 02:02:28 +02:00
|
|
|
|
2019-07-14 21:04:38 +02:00
|
|
|
return ResultCode.UnsupportedRevision;
|
2018-05-02 02:02:28 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-12-06 12:16:24 +01:00
|
|
|
private AudioRendererParameter GetAudioRendererParameter(ServiceCtx context)
|
2018-05-02 02:02:28 +02:00
|
|
|
{
|
2018-07-15 04:57:41 +02:00
|
|
|
AudioRendererParameter Params = new AudioRendererParameter();
|
|
|
|
|
2018-12-06 12:16:24 +01:00
|
|
|
Params.SampleRate = context.RequestData.ReadInt32();
|
|
|
|
Params.SampleCount = context.RequestData.ReadInt32();
|
|
|
|
Params.Unknown8 = context.RequestData.ReadInt32();
|
|
|
|
Params.MixCount = context.RequestData.ReadInt32();
|
|
|
|
Params.VoiceCount = context.RequestData.ReadInt32();
|
|
|
|
Params.SinkCount = context.RequestData.ReadInt32();
|
|
|
|
Params.EffectCount = context.RequestData.ReadInt32();
|
|
|
|
Params.PerformanceManagerCount = context.RequestData.ReadInt32();
|
|
|
|
Params.VoiceDropEnable = context.RequestData.ReadInt32();
|
|
|
|
Params.SplitterCount = context.RequestData.ReadInt32();
|
|
|
|
Params.SplitterDestinationDataCount = context.RequestData.ReadInt32();
|
|
|
|
Params.Unknown2C = context.RequestData.ReadInt32();
|
|
|
|
Params.Revision = context.RequestData.ReadInt32();
|
2018-07-15 04:57:41 +02:00
|
|
|
|
|
|
|
return Params;
|
2018-02-25 05:34:16 +01:00
|
|
|
}
|
2018-03-12 06:07:48 +01:00
|
|
|
|
2018-12-06 12:16:24 +01:00
|
|
|
private static int NodeStatesGetWorkBufferSize(int value)
|
2018-05-04 16:52:07 +02:00
|
|
|
{
|
2018-12-06 12:16:24 +01:00
|
|
|
int result = IntUtils.AlignUp(value, 64);
|
2018-05-04 16:52:07 +02:00
|
|
|
|
2018-12-06 12:16:24 +01:00
|
|
|
if (result < 0)
|
2018-05-04 16:52:07 +02:00
|
|
|
{
|
2018-12-06 12:16:24 +01:00
|
|
|
result |= 7;
|
2018-05-04 16:52:07 +02:00
|
|
|
}
|
|
|
|
|
2018-12-06 12:16:24 +01:00
|
|
|
return 4 * (value * value) + 0x12 * value + 2 * (result / 8);
|
2018-05-04 16:52:07 +02:00
|
|
|
}
|
|
|
|
|
2018-12-06 12:16:24 +01:00
|
|
|
private static int EdgeMatrixGetWorkBufferSize(int value)
|
2018-05-04 16:52:07 +02:00
|
|
|
{
|
2018-12-06 12:16:24 +01:00
|
|
|
int result = IntUtils.AlignUp(value * value, 64);
|
2018-05-04 16:52:07 +02:00
|
|
|
|
2018-12-06 12:16:24 +01:00
|
|
|
if (result < 0)
|
2018-05-04 16:52:07 +02:00
|
|
|
{
|
2018-12-06 12:16:24 +01:00
|
|
|
result |= 7;
|
2018-05-04 16:52:07 +02:00
|
|
|
}
|
|
|
|
|
2018-12-06 12:16:24 +01:00
|
|
|
return result / 8;
|
2018-05-04 16:52:07 +02:00
|
|
|
}
|
|
|
|
|
2019-07-12 03:13:43 +02:00
|
|
|
[Command(2)]
|
2018-10-07 17:12:11 +02:00
|
|
|
// GetAudioDeviceService(nn::applet::AppletResourceUserId) -> object<nn::audio::detail::IAudioDevice>
|
2019-07-14 21:04:38 +02:00
|
|
|
public ResultCode GetAudioDeviceService(ServiceCtx context)
|
2018-03-12 06:07:48 +01:00
|
|
|
{
|
2018-12-06 12:16:24 +01:00
|
|
|
long appletResourceUserId = context.RequestData.ReadInt64();
|
2018-03-12 20:29:06 +01:00
|
|
|
|
2018-12-06 12:16:24 +01:00
|
|
|
MakeObject(context, new IAudioDevice(context.Device.System));
|
2018-03-12 06:07:48 +01:00
|
|
|
|
2019-07-14 21:04:38 +02:00
|
|
|
return ResultCode.Success;
|
2018-03-12 06:07:48 +01:00
|
|
|
}
|
2018-10-07 17:12:11 +02:00
|
|
|
|
2019-07-12 03:13:43 +02:00
|
|
|
[Command(4)] // 4.0.0+
|
2018-10-07 17:12:11 +02:00
|
|
|
// GetAudioDeviceServiceWithRevisionInfo(nn::applet::AppletResourceUserId, u32) -> object<nn::audio::detail::IAudioDevice>
|
2019-07-14 21:04:38 +02:00
|
|
|
private ResultCode GetAudioDeviceServiceWithRevisionInfo(ServiceCtx context)
|
2018-10-07 17:12:11 +02:00
|
|
|
{
|
2018-12-06 12:16:24 +01:00
|
|
|
long appletResourceUserId = context.RequestData.ReadInt64();
|
|
|
|
int revisionInfo = context.RequestData.ReadInt32();
|
2018-10-07 17:12:11 +02:00
|
|
|
|
2019-01-11 01:11:46 +01:00
|
|
|
Logger.PrintStub(LogClass.ServiceAudio, new { appletResourceUserId, revisionInfo });
|
2018-10-07 17:12:11 +02:00
|
|
|
|
2018-12-06 12:16:24 +01:00
|
|
|
return GetAudioDeviceService(context);
|
2018-10-07 17:12:11 +02:00
|
|
|
}
|
2018-02-25 05:34:16 +01:00
|
|
|
}
|
2018-05-02 02:02:28 +02:00
|
|
|
}
|