mirror of
https://git.suyu.dev/suyu/suyu
synced 2025-01-17 05:10:13 -06:00
service/audren_u: Handle variadic command buffers in GetWorkBufferSize()
Also introduced in REV5 was a variable-size audio command buffer. This also affects how the size of the work buffer should be determined, so we can add handling for this as well. Thankfully, no other alterations were made to how the work buffer size is calculated in 7.0.0-8.0.0. There were indeed changes made to to how some of the actual audio commands are generated though (particularly in REV7), however they don't apply here.
This commit is contained in:
parent
03746be097
commit
2bcb8a20b4
@ -263,12 +263,15 @@ void AudRenU::OpenAudioRenderer(Kernel::HLERequestContext& ctx) {
|
|||||||
OpenAudioRendererImpl(ctx);
|
OpenAudioRendererImpl(ctx);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
static u64 CalculateNumPerformanceEntries(const AudioCore::AudioRendererParameter& params) {
|
||||||
|
// +1 represents the final mix.
|
||||||
|
return u64{params.effect_count} + params.submix_count + params.sink_count + params.voice_count +
|
||||||
|
1;
|
||||||
|
}
|
||||||
|
|
||||||
void AudRenU::GetAudioRendererWorkBufferSize(Kernel::HLERequestContext& ctx) {
|
void AudRenU::GetAudioRendererWorkBufferSize(Kernel::HLERequestContext& ctx) {
|
||||||
LOG_DEBUG(Service_Audio, "called");
|
LOG_DEBUG(Service_Audio, "called");
|
||||||
|
|
||||||
IPC::RequestParser rp{ctx};
|
|
||||||
const auto params = rp.PopRaw<AudioCore::AudioRendererParameter>();
|
|
||||||
|
|
||||||
// Several calculations below align the sizes being calculated
|
// Several calculations below align the sizes being calculated
|
||||||
// onto a 64 byte boundary.
|
// onto a 64 byte boundary.
|
||||||
static constexpr u64 buffer_alignment_size = 64;
|
static constexpr u64 buffer_alignment_size = 64;
|
||||||
@ -278,6 +281,10 @@ void AudRenU::GetAudioRendererWorkBufferSize(Kernel::HLERequestContext& ctx) {
|
|||||||
// the result of some of their calcularions on a 16 byte boundary.
|
// the result of some of their calcularions on a 16 byte boundary.
|
||||||
static constexpr u64 info_field_alignment_size = 16;
|
static constexpr u64 info_field_alignment_size = 16;
|
||||||
|
|
||||||
|
// Maximum detail entries that may exist at one time for performance
|
||||||
|
// frame statistics.
|
||||||
|
static constexpr u64 max_perf_detail_entries = 100;
|
||||||
|
|
||||||
// Size of the data structure representing the bulk of the voice-related state.
|
// Size of the data structure representing the bulk of the voice-related state.
|
||||||
static constexpr u64 voice_state_size = 0x100;
|
static constexpr u64 voice_state_size = 0x100;
|
||||||
|
|
||||||
@ -440,14 +447,9 @@ void AudRenU::GetAudioRendererWorkBufferSize(Kernel::HLERequestContext& ctx) {
|
|||||||
const u64 entry_size = is_v2 ? 0x18 : 0x10;
|
const u64 entry_size = is_v2 ? 0x18 : 0x10;
|
||||||
const u64 detail_size = is_v2 ? 0x18 : 0x10;
|
const u64 detail_size = is_v2 ? 0x18 : 0x10;
|
||||||
|
|
||||||
constexpr u64 max_detail_entries = 100;
|
const u64 entry_count = CalculateNumPerformanceEntries(params);
|
||||||
|
|
||||||
// + 1 to include the final mix, similar to calculating mix info.
|
|
||||||
const u64 entry_count = u64{params.effect_count} + params.submix_count + params.sink_count +
|
|
||||||
params.voice_count + 1;
|
|
||||||
|
|
||||||
const u64 size_per_frame =
|
const u64 size_per_frame =
|
||||||
header_size + (entry_size * entry_count) + (detail_size * max_detail_entries);
|
header_size + (entry_size * entry_count) + (detail_size * max_perf_detail_entries);
|
||||||
|
|
||||||
u64 size = 0;
|
u64 size = 0;
|
||||||
size += Common::AlignUp(size_per_frame * params.performance_frame_count + 1,
|
size += Common::AlignUp(size_per_frame * params.performance_frame_count + 1,
|
||||||
@ -458,11 +460,81 @@ void AudRenU::GetAudioRendererWorkBufferSize(Kernel::HLERequestContext& ctx) {
|
|||||||
};
|
};
|
||||||
|
|
||||||
// Calculates the part of the size that relates to the audio command buffer.
|
// Calculates the part of the size that relates to the audio command buffer.
|
||||||
const auto calculate_command_buffer_size = [] {
|
const auto calculate_command_buffer_size =
|
||||||
constexpr u64 command_buffer_size = 0x18000;
|
[this](const AudioCore::AudioRendererParameter& params) {
|
||||||
constexpr u64 alignment = (buffer_alignment_size - 1) * 2;
|
constexpr u64 alignment = (buffer_alignment_size - 1) * 2;
|
||||||
return command_buffer_size + alignment;
|
|
||||||
};
|
if (!IsFeatureSupported(AudioFeatures::VariadicCommandBuffer, params.revision)) {
|
||||||
|
constexpr u64 command_buffer_size = 0x18000;
|
||||||
|
|
||||||
|
return command_buffer_size + alignment;
|
||||||
|
}
|
||||||
|
|
||||||
|
// When the variadic command buffer is supported, this means
|
||||||
|
// the command generator for the audio renderer can issue commands
|
||||||
|
// that are (as one would expect), variable in size. So what we need to do
|
||||||
|
// is determine the maximum possible size for a few command data structures
|
||||||
|
// then multiply them by the amount of present commands indicated by the given
|
||||||
|
// respective audio parameters.
|
||||||
|
|
||||||
|
constexpr u64 max_biquad_filters = 2;
|
||||||
|
constexpr u64 max_mix_buffers = 24;
|
||||||
|
|
||||||
|
constexpr u64 biquad_filter_command_size = 0x2C;
|
||||||
|
|
||||||
|
constexpr u64 depop_mix_command_size = 0x24;
|
||||||
|
constexpr u64 depop_setup_command_size = 0x50;
|
||||||
|
|
||||||
|
constexpr u64 effect_command_max_size = 0x540;
|
||||||
|
|
||||||
|
constexpr u64 mix_command_size = 0x1C;
|
||||||
|
constexpr u64 mix_ramp_command_size = 0x24;
|
||||||
|
constexpr u64 mix_ramp_grouped_command_size = 0x13C;
|
||||||
|
|
||||||
|
constexpr u64 perf_command_size = 0x28;
|
||||||
|
|
||||||
|
constexpr u64 sink_command_size = 0x130;
|
||||||
|
|
||||||
|
constexpr u64 submix_command_max_size =
|
||||||
|
depop_mix_command_size + (mix_command_size * max_mix_buffers) * max_mix_buffers;
|
||||||
|
|
||||||
|
constexpr u64 volume_command_size = 0x1C;
|
||||||
|
constexpr u64 volume_ramp_command_size = 0x20;
|
||||||
|
|
||||||
|
constexpr u64 voice_biquad_filter_command_size =
|
||||||
|
biquad_filter_command_size * max_biquad_filters;
|
||||||
|
constexpr u64 voice_data_command_size = 0x9C;
|
||||||
|
const u64 voice_command_max_size =
|
||||||
|
(params.splitter_count * depop_setup_command_size) +
|
||||||
|
(voice_data_command_size + voice_biquad_filter_command_size +
|
||||||
|
volume_ramp_command_size + mix_ramp_grouped_command_size);
|
||||||
|
|
||||||
|
// Now calculate the individual elements that comprise the size and add them together.
|
||||||
|
const u64 effect_commands_size = params.effect_count * effect_command_max_size;
|
||||||
|
|
||||||
|
const u64 final_mix_commands_size =
|
||||||
|
depop_mix_command_size + volume_command_size * max_mix_buffers;
|
||||||
|
|
||||||
|
const u64 perf_commands_size =
|
||||||
|
perf_command_size *
|
||||||
|
(CalculateNumPerformanceEntries(params) + max_perf_detail_entries);
|
||||||
|
|
||||||
|
const u64 sink_commands_size = params.sink_count * sink_command_size;
|
||||||
|
|
||||||
|
const u64 splitter_commands_size =
|
||||||
|
params.num_splitter_send_channels * max_mix_buffers * mix_ramp_command_size;
|
||||||
|
|
||||||
|
const u64 submix_commands_size = params.submix_count * submix_command_max_size;
|
||||||
|
|
||||||
|
const u64 voice_commands_size = params.voice_count * voice_command_max_size;
|
||||||
|
|
||||||
|
return effect_commands_size + final_mix_commands_size + perf_commands_size +
|
||||||
|
sink_commands_size + splitter_commands_size + submix_commands_size +
|
||||||
|
voice_commands_size + alignment;
|
||||||
|
};
|
||||||
|
|
||||||
|
IPC::RequestParser rp{ctx};
|
||||||
|
const auto params = rp.PopRaw<AudioCore::AudioRendererParameter>();
|
||||||
|
|
||||||
u64 size = 0;
|
u64 size = 0;
|
||||||
size += calculate_mix_buffer_sizes(params);
|
size += calculate_mix_buffer_sizes(params);
|
||||||
@ -479,7 +551,7 @@ void AudRenU::GetAudioRendererWorkBufferSize(Kernel::HLERequestContext& ctx) {
|
|||||||
size += calculate_sink_info_size(params);
|
size += calculate_sink_info_size(params);
|
||||||
size += calculate_voice_state_size(params);
|
size += calculate_voice_state_size(params);
|
||||||
size += calculate_perf_size(params);
|
size += calculate_perf_size(params);
|
||||||
size += calculate_command_buffer_size();
|
size += calculate_command_buffer_size(params);
|
||||||
|
|
||||||
// finally, 4KB page align the size, and we're done.
|
// finally, 4KB page align the size, and we're done.
|
||||||
size = Common::AlignUp(size, 4096);
|
size = Common::AlignUp(size, 4096);
|
||||||
@ -526,11 +598,14 @@ void AudRenU::OpenAudioRendererImpl(Kernel::HLERequestContext& ctx) {
|
|||||||
}
|
}
|
||||||
|
|
||||||
bool AudRenU::IsFeatureSupported(AudioFeatures feature, u32_le revision) const {
|
bool AudRenU::IsFeatureSupported(AudioFeatures feature, u32_le revision) const {
|
||||||
u32_be version_num = (revision - Common::MakeMagic('R', 'E', 'V', '0')); // Byte swap
|
// Byte swap
|
||||||
|
const u32_be version_num = revision - Common::MakeMagic('R', 'E', 'V', '0');
|
||||||
|
|
||||||
switch (feature) {
|
switch (feature) {
|
||||||
case AudioFeatures::Splitter:
|
case AudioFeatures::Splitter:
|
||||||
return version_num >= 2U;
|
return version_num >= 2U;
|
||||||
case AudioFeatures::PerformanceMetricsVersion2:
|
case AudioFeatures::PerformanceMetricsVersion2:
|
||||||
|
case AudioFeatures::VariadicCommandBuffer:
|
||||||
return version_num >= 5U;
|
return version_num >= 5U;
|
||||||
default:
|
default:
|
||||||
return false;
|
return false;
|
||||||
|
@ -29,6 +29,7 @@ private:
|
|||||||
enum class AudioFeatures : u32 {
|
enum class AudioFeatures : u32 {
|
||||||
Splitter,
|
Splitter,
|
||||||
PerformanceMetricsVersion2,
|
PerformanceMetricsVersion2,
|
||||||
|
VariadicCommandBuffer,
|
||||||
};
|
};
|
||||||
|
|
||||||
bool IsFeatureSupported(AudioFeatures feature, u32_le revision) const;
|
bool IsFeatureSupported(AudioFeatures feature, u32_le revision) const;
|
||||||
|
Loading…
Reference in New Issue
Block a user