forked from Mirror/Ryujinx
audren: Fix AudioRenderer implementation (#773)
* Fix AudioRenderer implementation According to RE: - `GetAudioRendererWorkBufferSize` is updated and improved to support `REV7` - `RequestUpdateAudioRenderer` is updated to `REV7` too Should improve results on recent game and close #718 and #707 * Fix NodeStates.GetWorkBufferSize * Use BitUtils instead of IntUtils * Nits
This commit is contained in:
parent
a0720b5681
commit
f17b772c56
15 changed files with 244 additions and 110 deletions
|
@ -0,0 +1,9 @@
|
||||||
|
namespace Ryujinx.HLE.HOS.Services.Audio.AudioRendererManager
|
||||||
|
{
|
||||||
|
static class AudioRendererCommon
|
||||||
|
{
|
||||||
|
public static bool CheckValidRevision(AudioRendererParameter parameters) => GetRevisionVersion(parameters.Revision) <= AudioRendererConsts.Revision;
|
||||||
|
public static bool CheckFeatureSupported(int revision, int supportedRevision) => revision >= supportedRevision;
|
||||||
|
public static int GetRevisionVersion(int revision) => (revision - AudioRendererConsts.Rev0Magic) >> 24;
|
||||||
|
}
|
||||||
|
}
|
|
@ -0,0 +1,30 @@
|
||||||
|
namespace Ryujinx.HLE.HOS.Services.Audio.AudioRendererManager
|
||||||
|
{
|
||||||
|
class BehaviorInfo
|
||||||
|
{
|
||||||
|
private const int _revision = AudioRendererConsts.Revision;
|
||||||
|
|
||||||
|
private int _userRevision = 0;
|
||||||
|
|
||||||
|
public BehaviorInfo()
|
||||||
|
{
|
||||||
|
/* TODO: this class got a size of 0xC0
|
||||||
|
0x00 - uint - Internal Revision
|
||||||
|
0x04 - uint - User Revision
|
||||||
|
0x08 - ... unknown ...
|
||||||
|
*/
|
||||||
|
}
|
||||||
|
|
||||||
|
public bool IsSplitterSupported() => AudioRendererCommon.CheckFeatureSupported(_userRevision, SupportTags.Splitter);
|
||||||
|
public bool IsSplitterBugFixed() => AudioRendererCommon.CheckFeatureSupported(_userRevision, SupportTags.SplitterBugFix);
|
||||||
|
public bool IsVariadicCommandBufferSizeSupported() => AudioRendererCommon.CheckFeatureSupported(_userRevision, SupportTags.VariadicCommandBufferSize);
|
||||||
|
public bool IsElapsedFrameCountSupported() => AudioRendererCommon.CheckFeatureSupported(_userRevision, SupportTags.ElapsedFrameCount);
|
||||||
|
|
||||||
|
public int GetPerformanceMetricsDataFormat() => AudioRendererCommon.CheckFeatureSupported(_userRevision, SupportTags.PerformanceMetricsDataFormatVersion2) ? 2 : 1;
|
||||||
|
|
||||||
|
public void SetUserLibRevision(int revision)
|
||||||
|
{
|
||||||
|
_userRevision = AudioRendererCommon.GetRevisionVersion(revision);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
|
@ -0,0 +1,16 @@
|
||||||
|
namespace Ryujinx.HLE.HOS.Services.Audio.AudioRendererManager
|
||||||
|
{
|
||||||
|
static class CommandGenerator
|
||||||
|
{
|
||||||
|
public static long CalculateCommandBufferSize(AudioRendererParameter parameters)
|
||||||
|
{
|
||||||
|
return parameters.EffectCount * 0x840 +
|
||||||
|
parameters.SubMixCount * 0x5A38 +
|
||||||
|
parameters.SinkCount * 0x148 +
|
||||||
|
parameters.SplitterDestinationDataCount * 0x540 +
|
||||||
|
(parameters.SplitterCount * 0x68 + 0x2E0) * parameters.VoiceCount +
|
||||||
|
((parameters.VoiceCount + parameters.SubMixCount + parameters.EffectCount + parameters.SinkCount + 0x65) << 6) +
|
||||||
|
0x3F8;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
|
@ -0,0 +1,19 @@
|
||||||
|
using Ryujinx.Common;
|
||||||
|
|
||||||
|
namespace Ryujinx.HLE.HOS.Services.Audio.AudioRendererManager
|
||||||
|
{
|
||||||
|
static class EdgeMatrix
|
||||||
|
{
|
||||||
|
public static int GetWorkBufferSize(int totalMixCount)
|
||||||
|
{
|
||||||
|
int size = BitUtils.AlignUp(totalMixCount * totalMixCount, AudioRendererConsts.BufferAlignment);
|
||||||
|
|
||||||
|
if (size < 0)
|
||||||
|
{
|
||||||
|
size |= 7;
|
||||||
|
}
|
||||||
|
|
||||||
|
return size / 8;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
|
@ -51,8 +51,8 @@ namespace Ryujinx.HLE.HOS.Services.Audio.AudioRendererManager
|
||||||
_params = Params;
|
_params = Params;
|
||||||
|
|
||||||
_track = audioOut.OpenTrack(
|
_track = audioOut.OpenTrack(
|
||||||
AudioConsts.HostSampleRate,
|
AudioRendererConsts.HostSampleRate,
|
||||||
AudioConsts.HostChannelsCount,
|
AudioRendererConsts.HostChannelsCount,
|
||||||
AudioCallback);
|
AudioCallback);
|
||||||
|
|
||||||
_memoryPools = CreateArray<MemoryPoolContext>(Params.EffectCount + Params.VoiceCount * 4);
|
_memoryPools = CreateArray<MemoryPoolContext>(Params.EffectCount + Params.VoiceCount * 4);
|
||||||
|
@ -86,7 +86,7 @@ namespace Ryujinx.HLE.HOS.Services.Audio.AudioRendererManager
|
||||||
// GetMixBufferCount() -> u32
|
// GetMixBufferCount() -> u32
|
||||||
public ResultCode GetMixBufferCount(ServiceCtx context)
|
public ResultCode GetMixBufferCount(ServiceCtx context)
|
||||||
{
|
{
|
||||||
context.ResponseData.Write(_params.MixCount);
|
context.ResponseData.Write(_params.SubMixCount);
|
||||||
|
|
||||||
return ResultCode.Success;
|
return ResultCode.Success;
|
||||||
}
|
}
|
||||||
|
@ -145,6 +145,10 @@ namespace Ryujinx.HLE.HOS.Services.Audio.AudioRendererManager
|
||||||
|
|
||||||
UpdateDataHeader inputHeader = reader.Read<UpdateDataHeader>();
|
UpdateDataHeader inputHeader = reader.Read<UpdateDataHeader>();
|
||||||
|
|
||||||
|
BehaviorInfo behaviorInfo = new BehaviorInfo();
|
||||||
|
|
||||||
|
behaviorInfo.SetUserLibRevision(inputHeader.Revision);
|
||||||
|
|
||||||
reader.Read<BehaviorIn>(inputHeader.BehaviorSize);
|
reader.Read<BehaviorIn>(inputHeader.BehaviorSize);
|
||||||
|
|
||||||
MemoryPoolIn[] memoryPoolsIn = reader.Read<MemoryPoolIn>(inputHeader.MemoryPoolSize);
|
MemoryPoolIn[] memoryPoolsIn = reader.Read<MemoryPoolIn>(inputHeader.MemoryPoolSize);
|
||||||
|
@ -207,20 +211,27 @@ namespace Ryujinx.HLE.HOS.Services.Audio.AudioRendererManager
|
||||||
|
|
||||||
int updateHeaderSize = Marshal.SizeOf<UpdateDataHeader>();
|
int updateHeaderSize = Marshal.SizeOf<UpdateDataHeader>();
|
||||||
|
|
||||||
outputHeader.Revision = IAudioRendererManager.RevMagic;
|
outputHeader.Revision = AudioRendererConsts.RevMagic;
|
||||||
outputHeader.BehaviorSize = 0xb0;
|
outputHeader.BehaviorSize = 0xb0;
|
||||||
outputHeader.MemoryPoolSize = (_params.EffectCount + _params.VoiceCount * 4) * 0x10;
|
outputHeader.MemoryPoolSize = (_params.EffectCount + _params.VoiceCount * 4) * 0x10;
|
||||||
outputHeader.VoiceSize = _params.VoiceCount * 0x10;
|
outputHeader.VoiceSize = _params.VoiceCount * 0x10;
|
||||||
outputHeader.EffectSize = _params.EffectCount * 0x10;
|
outputHeader.EffectSize = _params.EffectCount * 0x10;
|
||||||
outputHeader.SinkSize = _params.SinkCount * 0x20;
|
outputHeader.SinkSize = _params.SinkCount * 0x20;
|
||||||
outputHeader.PerformanceManagerSize = 0x10;
|
outputHeader.PerformanceManagerSize = 0x10;
|
||||||
|
|
||||||
|
if (behaviorInfo.IsElapsedFrameCountSupported())
|
||||||
|
{
|
||||||
|
outputHeader.ElapsedFrameCountInfoSize = 0x10;
|
||||||
|
}
|
||||||
|
|
||||||
outputHeader.TotalSize = updateHeaderSize +
|
outputHeader.TotalSize = updateHeaderSize +
|
||||||
outputHeader.BehaviorSize +
|
outputHeader.BehaviorSize +
|
||||||
outputHeader.MemoryPoolSize +
|
outputHeader.MemoryPoolSize +
|
||||||
outputHeader.VoiceSize +
|
outputHeader.VoiceSize +
|
||||||
outputHeader.EffectSize +
|
outputHeader.EffectSize +
|
||||||
outputHeader.SinkSize +
|
outputHeader.SinkSize +
|
||||||
outputHeader.PerformanceManagerSize;
|
outputHeader.PerformanceManagerSize +
|
||||||
|
outputHeader.ElapsedFrameCountInfoSize;
|
||||||
|
|
||||||
writer.Write(outputHeader);
|
writer.Write(outputHeader);
|
||||||
|
|
||||||
|
@ -305,7 +316,7 @@ namespace Ryujinx.HLE.HOS.Services.Audio.AudioRendererManager
|
||||||
|
|
||||||
private void AppendMixedBuffer(long tag)
|
private void AppendMixedBuffer(long tag)
|
||||||
{
|
{
|
||||||
int[] mixBuffer = new int[MixBufferSamplesCount * AudioConsts.HostChannelsCount];
|
int[] mixBuffer = new int[MixBufferSamplesCount * AudioRendererConsts.HostChannelsCount];
|
||||||
|
|
||||||
foreach (VoiceContext voice in _voices)
|
foreach (VoiceContext voice in _voices)
|
||||||
{
|
{
|
||||||
|
|
|
@ -0,0 +1,19 @@
|
||||||
|
using Ryujinx.Common;
|
||||||
|
|
||||||
|
namespace Ryujinx.HLE.HOS.Services.Audio.AudioRendererManager
|
||||||
|
{
|
||||||
|
static class NodeStates
|
||||||
|
{
|
||||||
|
public static long GetWorkBufferSize(int totalMixCount)
|
||||||
|
{
|
||||||
|
int size = BitUtils.AlignUp(totalMixCount, AudioRendererConsts.BufferAlignment);
|
||||||
|
|
||||||
|
if (size < 0)
|
||||||
|
{
|
||||||
|
size |= 7;
|
||||||
|
}
|
||||||
|
|
||||||
|
return 4 * (totalMixCount * totalMixCount) + 12 * totalMixCount + 2 * (size / 8);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
|
@ -0,0 +1,30 @@
|
||||||
|
using Ryujinx.Common.Logging;
|
||||||
|
|
||||||
|
namespace Ryujinx.HLE.HOS.Services.Audio.AudioRendererManager
|
||||||
|
{
|
||||||
|
static class PerformanceManager
|
||||||
|
{
|
||||||
|
public static long GetRequiredBufferSizeForPerformanceMetricsPerFrame(BehaviorInfo behaviorInfo, AudioRendererParameter parameters)
|
||||||
|
{
|
||||||
|
int performanceMetricsDataFormat = behaviorInfo.GetPerformanceMetricsDataFormat();
|
||||||
|
|
||||||
|
if (performanceMetricsDataFormat == 2)
|
||||||
|
{
|
||||||
|
return 24 * (parameters.VoiceCount +
|
||||||
|
parameters.EffectCount +
|
||||||
|
parameters.SubMixCount +
|
||||||
|
parameters.SinkCount + 1) + 0x990;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (performanceMetricsDataFormat != 1)
|
||||||
|
{
|
||||||
|
Logger.PrintWarning(LogClass.ServiceAudio, $"PerformanceMetricsDataFormat: {performanceMetricsDataFormat} is not supported!");
|
||||||
|
}
|
||||||
|
|
||||||
|
return (((parameters.VoiceCount +
|
||||||
|
parameters.EffectCount +
|
||||||
|
parameters.SubMixCount +
|
||||||
|
parameters.SinkCount + 1) << 32) >> 0x1C) + 0x658;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
|
@ -0,0 +1,25 @@
|
||||||
|
using Ryujinx.Common;
|
||||||
|
|
||||||
|
namespace Ryujinx.HLE.HOS.Services.Audio.AudioRendererManager
|
||||||
|
{
|
||||||
|
class SplitterContext
|
||||||
|
{
|
||||||
|
public static long CalcWorkBufferSize(BehaviorInfo behaviorInfo, AudioRendererParameter parameters)
|
||||||
|
{
|
||||||
|
if (!behaviorInfo.IsSplitterSupported())
|
||||||
|
{
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
long size = parameters.SplitterDestinationDataCount * 0xE0 +
|
||||||
|
parameters.SplitterCount * 0x20;
|
||||||
|
|
||||||
|
if (!behaviorInfo.IsSplitterBugFixed())
|
||||||
|
{
|
||||||
|
size += BitUtils.AlignUp(4 * parameters.SplitterDestinationDataCount, 16);
|
||||||
|
}
|
||||||
|
|
||||||
|
return size;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
|
@ -1,8 +0,0 @@
|
||||||
namespace Ryujinx.HLE.HOS.Services.Audio.AudioRendererManager
|
|
||||||
{
|
|
||||||
static class AudioConsts
|
|
||||||
{
|
|
||||||
public const int HostSampleRate = 48000;
|
|
||||||
public const int HostChannelsCount = 2;
|
|
||||||
}
|
|
||||||
}
|
|
|
@ -0,0 +1,17 @@
|
||||||
|
namespace Ryujinx.HLE.HOS.Services.Audio.AudioRendererManager
|
||||||
|
{
|
||||||
|
static class AudioRendererConsts
|
||||||
|
{
|
||||||
|
// Revision Consts
|
||||||
|
public const int Revision = 7;
|
||||||
|
public const int Rev0Magic = ('R' << 0) | ('E' << 8) | ('V' << 16) | ('0' << 24);
|
||||||
|
public const int RevMagic = Rev0Magic + (Revision << 24);
|
||||||
|
|
||||||
|
// Misc Consts
|
||||||
|
public const int BufferAlignment = 0x40;
|
||||||
|
|
||||||
|
// Host Consts
|
||||||
|
public const int HostSampleRate = 48000;
|
||||||
|
public const int HostChannelsCount = 2;
|
||||||
|
}
|
||||||
|
}
|
|
@ -7,8 +7,8 @@ namespace Ryujinx.HLE.HOS.Services.Audio.AudioRendererManager
|
||||||
{
|
{
|
||||||
public int SampleRate;
|
public int SampleRate;
|
||||||
public int SampleCount;
|
public int SampleCount;
|
||||||
public int Unknown8;
|
public int MixBufferCount;
|
||||||
public int MixCount;
|
public int SubMixCount;
|
||||||
public int VoiceCount;
|
public int VoiceCount;
|
||||||
public int SinkCount;
|
public int SinkCount;
|
||||||
public int EffectCount;
|
public int EffectCount;
|
||||||
|
|
|
@ -0,0 +1,11 @@
|
||||||
|
namespace Ryujinx.HLE.HOS.Services.Audio.AudioRendererManager
|
||||||
|
{
|
||||||
|
static class SupportTags
|
||||||
|
{
|
||||||
|
public const int Splitter = 2;
|
||||||
|
public const int SplitterBugFix = 5;
|
||||||
|
public const int PerformanceMetricsDataFormatVersion2 = 5;
|
||||||
|
public const int VariadicCommandBufferSize = 5;
|
||||||
|
public const int ElapsedFrameCount = 5;
|
||||||
|
}
|
||||||
|
}
|
|
@ -12,7 +12,7 @@
|
||||||
public int SinkSize;
|
public int SinkSize;
|
||||||
public int PerformanceManagerSize;
|
public int PerformanceManagerSize;
|
||||||
public int Unknown24;
|
public int Unknown24;
|
||||||
public int Unknown28;
|
public int ElapsedFrameCountInfoSize;
|
||||||
public int Unknown2C;
|
public int Unknown2C;
|
||||||
public int Unknown30;
|
public int Unknown30;
|
||||||
public int Unknown34;
|
public int Unknown34;
|
||||||
|
|
|
@ -85,7 +85,7 @@ namespace Ryujinx.HLE.HOS.Services.Audio.AudioRendererManager
|
||||||
|
|
||||||
int maxSize = _samples.Length - _offset;
|
int maxSize = _samples.Length - _offset;
|
||||||
|
|
||||||
int size = maxSamples * AudioConsts.HostChannelsCount;
|
int size = maxSamples * AudioRendererConsts.HostChannelsCount;
|
||||||
|
|
||||||
if (size > maxSize)
|
if (size > maxSize)
|
||||||
{
|
{
|
||||||
|
@ -96,7 +96,7 @@ namespace Ryujinx.HLE.HOS.Services.Audio.AudioRendererManager
|
||||||
|
|
||||||
Array.Copy(_samples, _offset, output, 0, size);
|
Array.Copy(_samples, _offset, output, 0, size);
|
||||||
|
|
||||||
samplesCount = size / AudioConsts.HostChannelsCount;
|
samplesCount = size / AudioRendererConsts.HostChannelsCount;
|
||||||
|
|
||||||
_outStatus.PlayedSamplesCount += samplesCount;
|
_outStatus.PlayedSamplesCount += samplesCount;
|
||||||
|
|
||||||
|
@ -140,7 +140,7 @@ namespace Ryujinx.HLE.HOS.Services.Audio.AudioRendererManager
|
||||||
{
|
{
|
||||||
int samplesCount = (int)(wb.Size / (sizeof(short) * ChannelsCount));
|
int samplesCount = (int)(wb.Size / (sizeof(short) * ChannelsCount));
|
||||||
|
|
||||||
_samples = new int[samplesCount * AudioConsts.HostChannelsCount];
|
_samples = new int[samplesCount * AudioRendererConsts.HostChannelsCount];
|
||||||
|
|
||||||
if (ChannelsCount == 1)
|
if (ChannelsCount == 1)
|
||||||
{
|
{
|
||||||
|
@ -171,19 +171,19 @@ namespace Ryujinx.HLE.HOS.Services.Audio.AudioRendererManager
|
||||||
throw new InvalidOperationException();
|
throw new InvalidOperationException();
|
||||||
}
|
}
|
||||||
|
|
||||||
if (SampleRate != AudioConsts.HostSampleRate)
|
if (SampleRate != AudioRendererConsts.HostSampleRate)
|
||||||
{
|
{
|
||||||
// TODO: We should keep the frames being discarded (see the 4 below)
|
// TODO: We should keep the frames being discarded (see the 4 below)
|
||||||
// on a buffer and include it on the next samples buffer, to allow
|
// on a buffer and include it on the next samples buffer, to allow
|
||||||
// the resampler to do seamless interpolation between wave buffers.
|
// the resampler to do seamless interpolation between wave buffers.
|
||||||
int samplesCount = _samples.Length / AudioConsts.HostChannelsCount;
|
int samplesCount = _samples.Length / AudioRendererConsts.HostChannelsCount;
|
||||||
|
|
||||||
samplesCount = Math.Max(samplesCount - 4, 0);
|
samplesCount = Math.Max(samplesCount - 4, 0);
|
||||||
|
|
||||||
_samples = Resampler.Resample2Ch(
|
_samples = Resampler.Resample2Ch(
|
||||||
_samples,
|
_samples,
|
||||||
SampleRate,
|
SampleRate,
|
||||||
AudioConsts.HostSampleRate,
|
AudioRendererConsts.HostSampleRate,
|
||||||
samplesCount,
|
samplesCount,
|
||||||
ref _resamplerFracPart);
|
ref _resamplerFracPart);
|
||||||
}
|
}
|
||||||
|
|
|
@ -1,22 +1,13 @@
|
||||||
using Ryujinx.Audio;
|
using Ryujinx.Audio;
|
||||||
|
using Ryujinx.Common;
|
||||||
using Ryujinx.Common.Logging;
|
using Ryujinx.Common.Logging;
|
||||||
using Ryujinx.HLE.HOS.Services.Audio.AudioRendererManager;
|
using Ryujinx.HLE.HOS.Services.Audio.AudioRendererManager;
|
||||||
using Ryujinx.HLE.Utilities;
|
|
||||||
|
|
||||||
namespace Ryujinx.HLE.HOS.Services.Audio
|
namespace Ryujinx.HLE.HOS.Services.Audio
|
||||||
{
|
{
|
||||||
[Service("audren:u")]
|
[Service("audren:u")]
|
||||||
class IAudioRendererManager : IpcService
|
class IAudioRendererManager : IpcService
|
||||||
{
|
{
|
||||||
private const int Rev0Magic = ('R' << 0) |
|
|
||||||
('E' << 8) |
|
|
||||||
('V' << 16) |
|
|
||||||
('0' << 24);
|
|
||||||
|
|
||||||
private const int Rev = 5;
|
|
||||||
|
|
||||||
public const int RevMagic = Rev0Magic + (Rev << 24);
|
|
||||||
|
|
||||||
public IAudioRendererManager(ServiceCtx context) { }
|
public IAudioRendererManager(ServiceCtx context) { }
|
||||||
|
|
||||||
[Command(0)]
|
[Command(0)]
|
||||||
|
@ -41,69 +32,57 @@ namespace Ryujinx.HLE.HOS.Services.Audio
|
||||||
// GetWorkBufferSize(nn::audio::detail::AudioRendererParameterInternal) -> u64
|
// GetWorkBufferSize(nn::audio::detail::AudioRendererParameterInternal) -> u64
|
||||||
public ResultCode GetAudioRendererWorkBufferSize(ServiceCtx context)
|
public ResultCode GetAudioRendererWorkBufferSize(ServiceCtx context)
|
||||||
{
|
{
|
||||||
AudioRendererParameter Params = GetAudioRendererParameter(context);
|
AudioRendererParameter parameters = GetAudioRendererParameter(context);
|
||||||
|
|
||||||
int revision = (Params.Revision - Rev0Magic) >> 24;
|
if (AudioRendererCommon.CheckValidRevision(parameters))
|
||||||
|
|
||||||
if (revision <= Rev)
|
|
||||||
{
|
{
|
||||||
bool isSplitterSupported = revision >= 3;
|
BehaviorInfo behaviorInfo = new BehaviorInfo();
|
||||||
bool isVariadicCommandBufferSizeSupported = revision >= 5;
|
|
||||||
|
behaviorInfo.SetUserLibRevision(parameters.Revision);
|
||||||
|
|
||||||
long size;
|
long size;
|
||||||
|
|
||||||
size = IntUtils.AlignUp(Params.Unknown8 * 4, 64);
|
int totalMixCount = parameters.SubMixCount + 1;
|
||||||
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(
|
|
||||||
((Params.SinkCount + Params.MixCount) * 0x3C0 + Params.SampleCount * 4) *
|
|
||||||
(Params.Unknown8 + 6), 64);
|
|
||||||
size += (Params.SinkCount + Params.MixCount) * 0x2C0;
|
|
||||||
size += (Params.EffectCount + Params.VoiceCount * 4) * 0x30 + 0x50;
|
|
||||||
|
|
||||||
if (isSplitterSupported)
|
size = BitUtils.AlignUp(parameters.MixBufferCount * 4, AudioRendererConsts.BufferAlignment) +
|
||||||
|
parameters.SubMixCount * 0x400 +
|
||||||
|
totalMixCount * 0x940 +
|
||||||
|
parameters.VoiceCount * 0x3F0 +
|
||||||
|
BitUtils.AlignUp(totalMixCount * 8, 16) +
|
||||||
|
BitUtils.AlignUp(parameters.VoiceCount * 8, 16) +
|
||||||
|
BitUtils.AlignUp(((parameters.SinkCount + parameters.SubMixCount) * 0x3C0 + parameters.SampleCount * 4) *
|
||||||
|
(parameters.MixBufferCount + 6), AudioRendererConsts.BufferAlignment) +
|
||||||
|
(parameters.SinkCount + parameters.SubMixCount) * 0x2C0 +
|
||||||
|
(parameters.EffectCount + parameters.VoiceCount * 4) * 0x30 +
|
||||||
|
0x50;
|
||||||
|
|
||||||
|
if (behaviorInfo.IsSplitterSupported())
|
||||||
{
|
{
|
||||||
size += IntUtils.AlignUp((
|
size += BitUtils.AlignUp(NodeStates.GetWorkBufferSize(totalMixCount) + EdgeMatrix.GetWorkBufferSize(totalMixCount), 16);
|
||||||
NodeStatesGetWorkBufferSize(Params.MixCount + 1) +
|
|
||||||
EdgeMatrixGetWorkBufferSize(Params.MixCount + 1)), 16);
|
|
||||||
|
|
||||||
size += Params.SplitterDestinationDataCount * 0xE0;
|
|
||||||
size += Params.SplitterCount * 0x20;
|
|
||||||
size += IntUtils.AlignUp(Params.SplitterDestinationDataCount * 4, 16);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
size = Params.EffectCount * 0x4C0 +
|
size = parameters.SinkCount * 0x170 +
|
||||||
Params.SinkCount * 0x170 +
|
(parameters.SinkCount + parameters.SubMixCount) * 0x280 +
|
||||||
Params.VoiceCount * 0x100 +
|
parameters.EffectCount * 0x4C0 +
|
||||||
IntUtils.AlignUp(size, 64) + 0x40;
|
((size + SplitterContext.CalcWorkBufferSize(behaviorInfo, parameters) + 0x30 * parameters.EffectCount + (4 * parameters.VoiceCount) + 0x8F) & ~0x3FL) +
|
||||||
|
((parameters.VoiceCount << 8) | 0x40);
|
||||||
|
|
||||||
if (Params.PerformanceManagerCount >= 1)
|
if (parameters.PerformanceManagerCount >= 1)
|
||||||
{
|
{
|
||||||
size += (((Params.EffectCount +
|
size += (PerformanceManager.GetRequiredBufferSizeForPerformanceMetricsPerFrame(behaviorInfo, parameters) *
|
||||||
Params.SinkCount +
|
(parameters.PerformanceManagerCount + 1) + 0xFF) & ~0x3FL;
|
||||||
Params.VoiceCount +
|
|
||||||
Params.MixCount + 1) * 16 + 0x658) *
|
|
||||||
(Params.PerformanceManagerCount + 1) + 0x13F) & ~0x3FL;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
if (isVariadicCommandBufferSizeSupported)
|
if (behaviorInfo.IsVariadicCommandBufferSizeSupported())
|
||||||
{
|
{
|
||||||
size += Params.EffectCount * 0x840 +
|
size += CommandGenerator.CalculateCommandBufferSize(parameters) + 0x7E;
|
||||||
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
|
else
|
||||||
{
|
{
|
||||||
size += 0x1807E;
|
size += 0x1807E;
|
||||||
}
|
}
|
||||||
|
|
||||||
size = size & ~0xFFFL;
|
size = BitUtils.AlignUp(size, 0x1000);
|
||||||
|
|
||||||
context.ResponseData.Write(size);
|
context.ResponseData.Write(size);
|
||||||
|
|
||||||
|
@ -115,7 +94,7 @@ namespace Ryujinx.HLE.HOS.Services.Audio
|
||||||
{
|
{
|
||||||
context.ResponseData.Write(0L);
|
context.ResponseData.Write(0L);
|
||||||
|
|
||||||
Logger.PrintWarning(LogClass.ServiceAudio, $"Library Revision 0x{Params.Revision:x8} is not supported!");
|
Logger.PrintWarning(LogClass.ServiceAudio, $"Library Revision REV{AudioRendererCommon.GetRevisionVersion(parameters.Revision)} is not supported!");
|
||||||
|
|
||||||
return ResultCode.UnsupportedRevision;
|
return ResultCode.UnsupportedRevision;
|
||||||
}
|
}
|
||||||
|
@ -127,8 +106,8 @@ namespace Ryujinx.HLE.HOS.Services.Audio
|
||||||
{
|
{
|
||||||
SampleRate = context.RequestData.ReadInt32(),
|
SampleRate = context.RequestData.ReadInt32(),
|
||||||
SampleCount = context.RequestData.ReadInt32(),
|
SampleCount = context.RequestData.ReadInt32(),
|
||||||
Unknown8 = context.RequestData.ReadInt32(),
|
MixBufferCount = context.RequestData.ReadInt32(),
|
||||||
MixCount = context.RequestData.ReadInt32(),
|
SubMixCount = context.RequestData.ReadInt32(),
|
||||||
VoiceCount = context.RequestData.ReadInt32(),
|
VoiceCount = context.RequestData.ReadInt32(),
|
||||||
SinkCount = context.RequestData.ReadInt32(),
|
SinkCount = context.RequestData.ReadInt32(),
|
||||||
EffectCount = context.RequestData.ReadInt32(),
|
EffectCount = context.RequestData.ReadInt32(),
|
||||||
|
@ -143,30 +122,6 @@ namespace Ryujinx.HLE.HOS.Services.Audio
|
||||||
return Params;
|
return Params;
|
||||||
}
|
}
|
||||||
|
|
||||||
private static int NodeStatesGetWorkBufferSize(int value)
|
|
||||||
{
|
|
||||||
int result = IntUtils.AlignUp(value, 64);
|
|
||||||
|
|
||||||
if (result < 0)
|
|
||||||
{
|
|
||||||
result |= 7;
|
|
||||||
}
|
|
||||||
|
|
||||||
return 4 * (value * value) + 0x12 * value + 2 * (result / 8);
|
|
||||||
}
|
|
||||||
|
|
||||||
private static int EdgeMatrixGetWorkBufferSize(int value)
|
|
||||||
{
|
|
||||||
int result = IntUtils.AlignUp(value * value, 64);
|
|
||||||
|
|
||||||
if (result < 0)
|
|
||||||
{
|
|
||||||
result |= 7;
|
|
||||||
}
|
|
||||||
|
|
||||||
return result / 8;
|
|
||||||
}
|
|
||||||
|
|
||||||
[Command(2)]
|
[Command(2)]
|
||||||
// GetAudioDeviceService(nn::applet::AppletResourceUserId) -> object<nn::audio::detail::IAudioDevice>
|
// GetAudioDeviceService(nn::applet::AppletResourceUserId) -> object<nn::audio::detail::IAudioDevice>
|
||||||
public ResultCode GetAudioDeviceService(ServiceCtx context)
|
public ResultCode GetAudioDeviceService(ServiceCtx context)
|
||||||
|
|
Reference in a new issue