summaryrefslogtreecommitdiffstats
path: root/src/audio_core/common
diff options
context:
space:
mode:
Diffstat (limited to '')
-rw-r--r--src/audio_core/common.h132
-rw-r--r--src/audio_core/common/audio_renderer_parameter.h60
-rw-r--r--src/audio_core/common/common.h138
-rw-r--r--src/audio_core/common/feature_support.h105
-rw-r--r--src/audio_core/common/wave_buffer.h35
-rw-r--r--src/audio_core/common/workbuffer_allocator.h100
6 files changed, 438 insertions, 132 deletions
diff --git a/src/audio_core/common.h b/src/audio_core/common.h
deleted file mode 100644
index 056a0ac70..000000000
--- a/src/audio_core/common.h
+++ /dev/null
@@ -1,132 +0,0 @@
-// SPDX-FileCopyrightText: Copyright 2020 yuzu Emulator Project
-// SPDX-License-Identifier: GPL-2.0-or-later
-
-#pragma once
-
-#include "common/common_funcs.h"
-#include "common/common_types.h"
-#include "common/swap.h"
-#include "core/hle/result.h"
-
-namespace AudioCommon {
-namespace Audren {
-constexpr Result ERR_INVALID_PARAMETERS{ErrorModule::Audio, 41};
-constexpr Result ERR_SPLITTER_SORT_FAILED{ErrorModule::Audio, 43};
-} // namespace Audren
-
-constexpr u8 BASE_REVISION = '0';
-constexpr u32_le CURRENT_PROCESS_REVISION =
- Common::MakeMagic('R', 'E', 'V', static_cast<u8>(BASE_REVISION + 0xA));
-constexpr std::size_t MAX_MIX_BUFFERS = 24;
-constexpr std::size_t MAX_BIQUAD_FILTERS = 2;
-constexpr std::size_t MAX_CHANNEL_COUNT = 6;
-constexpr std::size_t MAX_WAVE_BUFFERS = 4;
-constexpr std::size_t MAX_SAMPLE_HISTORY = 4;
-constexpr u32 STREAM_SAMPLE_RATE = 48000;
-constexpr u32 STREAM_NUM_CHANNELS = 2;
-constexpr s32 NO_SPLITTER = -1;
-constexpr s32 NO_MIX = 0x7fffffff;
-constexpr s32 NO_FINAL_MIX = std::numeric_limits<s32>::min();
-constexpr s32 FINAL_MIX = 0;
-constexpr s32 NO_EFFECT_ORDER = -1;
-constexpr std::size_t TEMP_MIX_BASE_SIZE = 0x3f00; // TODO(ogniK): Work out this constant
-// Any size checks seem to take the sample history into account
-// and our const ends up being 0x3f04, the 4 bytes are most
-// likely the sample history
-constexpr std::size_t TOTAL_TEMP_MIX_SIZE = TEMP_MIX_BASE_SIZE + AudioCommon::MAX_SAMPLE_HISTORY;
-constexpr f32 I3DL2REVERB_MAX_LEVEL = 5000.0f;
-constexpr f32 I3DL2REVERB_MIN_REFLECTION_DURATION = 0.02f;
-constexpr std::size_t I3DL2REVERB_TAPS = 20;
-constexpr std::size_t I3DL2REVERB_DELAY_LINE_COUNT = 4;
-using Fractional = s32;
-
-template <typename T>
-constexpr Fractional ToFractional(T x) {
- return static_cast<Fractional>(x * static_cast<T>(0x4000));
-}
-
-constexpr Fractional MultiplyFractional(Fractional lhs, Fractional rhs) {
- return static_cast<Fractional>(static_cast<s64>(lhs) * rhs >> 14);
-}
-
-constexpr s32 FractionalToFixed(Fractional x) {
- const auto s = x & (1 << 13);
- return static_cast<s32>(x >> 14) + s;
-}
-
-constexpr s32 CalculateDelaySamples(s32 sample_rate_khz, float time) {
- return FractionalToFixed(MultiplyFractional(ToFractional(sample_rate_khz), ToFractional(time)));
-}
-
-static constexpr u32 VersionFromRevision(u32_le rev) {
- // "REV7" -> 7
- return ((rev >> 24) & 0xff) - 0x30;
-}
-
-static constexpr bool IsRevisionSupported(u32 required, u32_le user_revision) {
- const auto base = VersionFromRevision(user_revision);
- return required <= base;
-}
-
-static constexpr bool IsValidRevision(u32_le revision) {
- const auto base = VersionFromRevision(revision);
- constexpr auto max_rev = VersionFromRevision(CURRENT_PROCESS_REVISION);
- return base <= max_rev;
-}
-
-static constexpr bool CanConsumeBuffer(std::size_t size, std::size_t offset, std::size_t required) {
- if (offset > size) {
- return false;
- }
- if (size < required) {
- return false;
- }
- if ((size - offset) < required) {
- return false;
- }
- return true;
-}
-
-struct UpdateDataSizes {
- u32_le behavior{};
- u32_le memory_pool{};
- u32_le voice{};
- u32_le voice_channel_resource{};
- u32_le effect{};
- u32_le mixer{};
- u32_le sink{};
- u32_le performance{};
- u32_le splitter{};
- u32_le render_info{};
- INSERT_PADDING_WORDS(4);
-};
-static_assert(sizeof(UpdateDataSizes) == 0x38, "UpdateDataSizes is an invalid size");
-
-struct UpdateDataHeader {
- u32_le revision{};
- UpdateDataSizes size{};
- u32_le total_size{};
-};
-static_assert(sizeof(UpdateDataHeader) == 0x40, "UpdateDataHeader is an invalid size");
-
-struct AudioRendererParameter {
- u32_le sample_rate;
- u32_le sample_count;
- u32_le mix_buffer_count;
- u32_le submix_count;
- u32_le voice_count;
- u32_le sink_count;
- u32_le effect_count;
- u32_le performance_frame_count;
- u8 is_voice_drop_enabled;
- u8 unknown_21;
- u8 unknown_22;
- u8 execution_mode;
- u32_le splitter_count;
- u32_le num_splitter_send_channels;
- u32_le unknown_30;
- u32_le revision;
-};
-static_assert(sizeof(AudioRendererParameter) == 52, "AudioRendererParameter is an invalid size");
-
-} // namespace AudioCommon
diff --git a/src/audio_core/common/audio_renderer_parameter.h b/src/audio_core/common/audio_renderer_parameter.h
new file mode 100644
index 000000000..2f62c383b
--- /dev/null
+++ b/src/audio_core/common/audio_renderer_parameter.h
@@ -0,0 +1,60 @@
+// SPDX-FileCopyrightText: Copyright 2022 yuzu Emulator Project
+// SPDX-License-Identifier: GPL-2.0-or-later
+
+#pragma once
+
+#include <span>
+
+#include "audio_core/renderer/behavior/behavior_info.h"
+#include "audio_core/renderer/memory/memory_pool_info.h"
+#include "audio_core/renderer/upsampler/upsampler_manager.h"
+#include "common/common_types.h"
+
+namespace AudioCore {
+/**
+ * Execution mode of the audio renderer.
+ * Only Auto is currently supported.
+ */
+enum class ExecutionMode : u8 {
+ Auto,
+ Manual,
+};
+
+/**
+ * Parameters from the game, passed to the audio renderer for initialisation.
+ */
+struct AudioRendererParameterInternal {
+ /* 0x00 */ u32 sample_rate;
+ /* 0x04 */ u32 sample_count;
+ /* 0x08 */ u32 mixes;
+ /* 0x0C */ u32 sub_mixes;
+ /* 0x10 */ u32 voices;
+ /* 0x14 */ u32 sinks;
+ /* 0x18 */ u32 effects;
+ /* 0x1C */ u32 perf_frames;
+ /* 0x20 */ u16 voice_drop_enabled;
+ /* 0x22 */ u8 rendering_device;
+ /* 0x23 */ ExecutionMode execution_mode;
+ /* 0x24 */ u32 splitter_infos;
+ /* 0x28 */ s32 splitter_destinations;
+ /* 0x2C */ u32 external_context_size;
+ /* 0x30 */ u32 revision;
+ /* 0x34 */ char unk34[0x4];
+};
+static_assert(sizeof(AudioRendererParameterInternal) == 0x38,
+ "AudioRendererParameterInternal has the wrong size!");
+
+/**
+ * Context for rendering, contains a bunch of useful fields for the command generator.
+ */
+struct AudioRendererSystemContext {
+ s32 session_id;
+ s8 channels;
+ s16 mix_buffer_count;
+ AudioRenderer::BehaviorInfo* behavior;
+ std::span<s32> depop_buffer;
+ AudioRenderer::UpsamplerManager* upsampler_manager;
+ AudioRenderer::MemoryPoolInfo* memory_pool_info;
+};
+
+} // namespace AudioCore
diff --git a/src/audio_core/common/common.h b/src/audio_core/common/common.h
new file mode 100644
index 000000000..6abd9be45
--- /dev/null
+++ b/src/audio_core/common/common.h
@@ -0,0 +1,138 @@
+// SPDX-FileCopyrightText: Copyright 2022 yuzu Emulator Project
+// SPDX-License-Identifier: GPL-2.0-or-later
+
+#pragma once
+
+#include <numeric>
+#include <span>
+
+#include "common/assert.h"
+#include "common/common_funcs.h"
+#include "common/common_types.h"
+
+namespace AudioCore {
+using CpuAddr = std::uintptr_t;
+
+enum class PlayState : u8 {
+ Started,
+ Stopped,
+ Paused,
+};
+
+enum class SrcQuality : u8 {
+ Medium,
+ High,
+ Low,
+};
+
+enum class SampleFormat : u8 {
+ Invalid,
+ PcmInt8,
+ PcmInt16,
+ PcmInt24,
+ PcmInt32,
+ PcmFloat,
+ Adpcm,
+};
+
+enum class SessionTypes {
+ AudioIn,
+ AudioOut,
+ FinalOutputRecorder,
+};
+
+enum class Channels : u32 {
+ FrontLeft,
+ FrontRight,
+ Center,
+ LFE,
+ BackLeft,
+ BackRight,
+};
+
+// These are used by Delay, Reverb and I3dl2Reverb prior to Revision 11.
+enum class OldChannels : u32 {
+ FrontLeft,
+ FrontRight,
+ BackLeft,
+ BackRight,
+ Center,
+ LFE,
+};
+
+constexpr u32 BufferCount = 32;
+
+constexpr u32 MaxRendererSessions = 2;
+constexpr u32 TargetSampleCount = 240;
+constexpr u32 TargetSampleRate = 48'000;
+constexpr u32 MaxChannels = 6;
+constexpr u32 MaxMixBuffers = 24;
+constexpr u32 MaxWaveBuffers = 4;
+constexpr s32 LowestVoicePriority = 0xFF;
+constexpr s32 HighestVoicePriority = 0;
+constexpr u32 BufferAlignment = 0x40;
+constexpr u32 WorkbufferAlignment = 0x1000;
+constexpr s32 FinalMixId = 0;
+constexpr s32 InvalidDistanceFromFinalMix = std::numeric_limits<s32>::min();
+constexpr s32 UnusedSplitterId = -1;
+constexpr s32 UnusedMixId = std::numeric_limits<s32>::max();
+constexpr u32 InvalidNodeId = 0xF0000000;
+constexpr s32 InvalidProcessOrder = -1;
+constexpr u32 MaxBiquadFilters = 2;
+constexpr u32 MaxEffects = 256;
+
+constexpr bool IsChannelCountValid(u16 channel_count) {
+ return channel_count <= 6 &&
+ (channel_count == 1 || channel_count == 2 || channel_count == 4 || channel_count == 6);
+}
+
+constexpr void UseOldChannelMapping(std::span<s16> inputs, std::span<s16> outputs) {
+ constexpr auto old_center{static_cast<u32>(OldChannels::Center)};
+ constexpr auto new_center{static_cast<u32>(Channels::Center)};
+ constexpr auto old_lfe{static_cast<u32>(OldChannels::LFE)};
+ constexpr auto new_lfe{static_cast<u32>(Channels::LFE)};
+
+ auto center{inputs[old_center]};
+ auto lfe{inputs[old_lfe]};
+ inputs[old_center] = inputs[new_center];
+ inputs[old_lfe] = inputs[new_lfe];
+ inputs[new_center] = center;
+ inputs[new_lfe] = lfe;
+
+ center = outputs[old_center];
+ lfe = outputs[old_lfe];
+ outputs[old_center] = outputs[new_center];
+ outputs[old_lfe] = outputs[new_lfe];
+ outputs[new_center] = center;
+ outputs[new_lfe] = lfe;
+}
+
+constexpr u32 GetSplitterInParamHeaderMagic() {
+ return Common::MakeMagic('S', 'N', 'D', 'H');
+}
+
+constexpr u32 GetSplitterInfoMagic() {
+ return Common::MakeMagic('S', 'N', 'D', 'I');
+}
+
+constexpr u32 GetSplitterSendDataMagic() {
+ return Common::MakeMagic('S', 'N', 'D', 'D');
+}
+
+constexpr size_t GetSampleFormatByteSize(SampleFormat format) {
+ switch (format) {
+ case SampleFormat::PcmInt8:
+ return 1;
+ case SampleFormat::PcmInt16:
+ return 2;
+ case SampleFormat::PcmInt24:
+ return 3;
+ case SampleFormat::PcmInt32:
+ case SampleFormat::PcmFloat:
+ return 4;
+ default:
+ return 2;
+ }
+}
+
+} // namespace AudioCore
diff --git a/src/audio_core/common/feature_support.h b/src/audio_core/common/feature_support.h
new file mode 100644
index 000000000..55c9e690d
--- /dev/null
+++ b/src/audio_core/common/feature_support.h
@@ -0,0 +1,105 @@
+// SPDX-FileCopyrightText: Copyright 2022 yuzu Emulator Project
+// SPDX-License-Identifier: GPL-2.0-or-later
+
+#pragma once
+
+#include <map>
+#include <ranges>
+#include <tuple>
+
+#include "common/assert.h"
+#include "common/common_funcs.h"
+#include "common/common_types.h"
+
+namespace AudioCore {
+constexpr u32 CurrentRevision = 11;
+
+enum class SupportTags {
+ CommandProcessingTimeEstimatorVersion4,
+ CommandProcessingTimeEstimatorVersion3,
+ CommandProcessingTimeEstimatorVersion2,
+ MultiTapBiquadFilterProcessing,
+ EffectInfoVer2,
+ WaveBufferVer2,
+ BiquadFilterFloatProcessing,
+ VolumeMixParameterPrecisionQ23,
+ MixInParameterDirtyOnlyUpdate,
+ BiquadFilterEffectStateClearBugFix,
+ VoicePlayedSampleCountResetAtLoopPoint,
+ VoicePitchAndSrcSkipped,
+ SplitterBugFix,
+ FlushVoiceWaveBuffers,
+ ElapsedFrameCount,
+ AudioRendererVariadicCommandBufferSize,
+ PerformanceMetricsDataFormatVersion2,
+ AudioRendererProcessingTimeLimit80Percent,
+ AudioRendererProcessingTimeLimit75Percent,
+ AudioRendererProcessingTimeLimit70Percent,
+ AdpcmLoopContextBugFix,
+ Splitter,
+ LongSizePreDelay,
+ AudioUsbDeviceOutput,
+ DeviceApiVersion2,
+ DelayChannelMappingChange,
+ ReverbChannelMappingChange,
+ I3dl2ReverbChannelMappingChange,
+
+ // Not a real tag, just here to get the count.
+ Size
+};
+
+constexpr u32 GetRevisionNum(u32 user_revision) {
+ if (user_revision >= 0x100) {
+ user_revision -= Common::MakeMagic('R', 'E', 'V', '0');
+ user_revision >>= 24;
+ }
+ return user_revision;
+};
+
+constexpr bool CheckFeatureSupported(SupportTags tag, u32 user_revision) {
+ constexpr std::array<std::pair<SupportTags, u32>, static_cast<u32>(SupportTags::Size)> features{
+ {
+ {SupportTags::AudioRendererProcessingTimeLimit70Percent, 1},
+ {SupportTags::Splitter, 2},
+ {SupportTags::AdpcmLoopContextBugFix, 2},
+ {SupportTags::LongSizePreDelay, 3},
+ {SupportTags::AudioUsbDeviceOutput, 4},
+ {SupportTags::AudioRendererProcessingTimeLimit75Percent, 4},
+ {SupportTags::VoicePlayedSampleCountResetAtLoopPoint, 5},
+ {SupportTags::VoicePitchAndSrcSkipped, 5},
+ {SupportTags::SplitterBugFix, 5},
+ {SupportTags::FlushVoiceWaveBuffers, 5},
+ {SupportTags::ElapsedFrameCount, 5},
+ {SupportTags::AudioRendererProcessingTimeLimit80Percent, 5},
+ {SupportTags::AudioRendererVariadicCommandBufferSize, 5},
+ {SupportTags::PerformanceMetricsDataFormatVersion2, 5},
+ {SupportTags::CommandProcessingTimeEstimatorVersion2, 5},
+ {SupportTags::BiquadFilterEffectStateClearBugFix, 6},
+ {SupportTags::BiquadFilterFloatProcessing, 7},
+ {SupportTags::VolumeMixParameterPrecisionQ23, 7},
+ {SupportTags::MixInParameterDirtyOnlyUpdate, 7},
+ {SupportTags::WaveBufferVer2, 8},
+ {SupportTags::CommandProcessingTimeEstimatorVersion3, 8},
+ {SupportTags::EffectInfoVer2, 9},
+ {SupportTags::CommandProcessingTimeEstimatorVersion4, 10},
+ {SupportTags::MultiTapBiquadFilterProcessing, 10},
+ {SupportTags::DelayChannelMappingChange, 11},
+ {SupportTags::ReverbChannelMappingChange, 11},
+ {SupportTags::I3dl2ReverbChannelMappingChange, 11},
+ }};
+
+ const auto& feature =
+ std::ranges::find_if(features, [tag](const auto& entry) { return entry.first == tag; });
+ if (feature == features.cend()) {
+ LOG_ERROR(Service_Audio, "Invalid SupportTag {}!", static_cast<u32>(tag));
+ return false;
+ }
+ user_revision = GetRevisionNum(user_revision);
+ return (*feature).second <= user_revision;
+}
+
+constexpr bool CheckValidRevision(u32 user_revision) {
+ return GetRevisionNum(user_revision) <= CurrentRevision;
+};
+
+} // namespace AudioCore
diff --git a/src/audio_core/common/wave_buffer.h b/src/audio_core/common/wave_buffer.h
new file mode 100644
index 000000000..fc478ef79
--- /dev/null
+++ b/src/audio_core/common/wave_buffer.h
@@ -0,0 +1,35 @@
+// SPDX-FileCopyrightText: Copyright 2022 yuzu Emulator Project
+// SPDX-License-Identifier: GPL-2.0-or-later
+
+#pragma once
+
+#include "common/common_types.h"
+
+namespace AudioCore {
+
+struct WaveBufferVersion1 {
+ CpuAddr buffer;
+ u64 buffer_size;
+ u32 start_offset;
+ u32 end_offset;
+ bool loop;
+ bool stream_ended;
+ CpuAddr context;
+ u64 context_size;
+};
+
+struct WaveBufferVersion2 {
+ CpuAddr buffer;
+ CpuAddr context;
+ u64 buffer_size;
+ u64 context_size;
+ u32 start_offset;
+ u32 end_offset;
+ u32 loop_start_offset;
+ u32 loop_end_offset;
+ s32 loop_count;
+ bool loop;
+ bool stream_ended;
+};
+
+} // namespace AudioCore
diff --git a/src/audio_core/common/workbuffer_allocator.h b/src/audio_core/common/workbuffer_allocator.h
new file mode 100644
index 000000000..fb89f97fe
--- /dev/null
+++ b/src/audio_core/common/workbuffer_allocator.h
@@ -0,0 +1,100 @@
+// SPDX-FileCopyrightText: Copyright 2022 yuzu Emulator Project
+// SPDX-License-Identifier: GPL-2.0-or-later
+
+#pragma once
+
+#include <span>
+
+#include "common/alignment.h"
+#include "common/assert.h"
+#include "common/common_types.h"
+
+namespace AudioCore {
+/**
+ * Responsible for allocating up a workbuffer into multiple pieces.
+ * Takes in a buffer and size (it does not own them), and allocates up the buffer via Allocate.
+ */
+class WorkbufferAllocator {
+public:
+ explicit WorkbufferAllocator(std::span<u8> buffer_, u64 size_)
+ : buffer{reinterpret_cast<u64>(buffer_.data())}, size{size_} {}
+
+ /**
+ * Allocate the given count of T elements, aligned to alignment.
+ *
+ * @param count - The number of elements to allocate.
+ * @param alignment - The required starting alignment.
+ * @return Non-owning container of allocated elements.
+ */
+ template <typename T>
+ std::span<T> Allocate(u64 count, u64 alignment) {
+ u64 out{0};
+ u64 byte_size{count * sizeof(T)};
+
+ if (byte_size > 0) {
+ auto current{buffer + offset};
+ auto aligned_buffer{Common::AlignUp(current, alignment)};
+ if (aligned_buffer + byte_size <= buffer + size) {
+ out = aligned_buffer;
+ offset = byte_size - buffer + aligned_buffer;
+ } else {
+ LOG_ERROR(
+ Service_Audio,
+ "Allocated buffer was too small to hold new alloc.\nAllocator size={:08X}, "
+ "offset={:08X}.\nAttempting to allocate {:08X} with alignment={:02X}",
+ size, offset, byte_size, alignment);
+ count = 0;
+ }
+ }
+
+ return std::span<T>(reinterpret_cast<T*>(out), count);
+ }
+
+ /**
+ * Align the current offset to the given alignment.
+ *
+ * @param alignment - The required starting alignment.
+ */
+ void Align(u64 alignment) {
+ auto current{buffer + offset};
+ auto aligned_buffer{Common::AlignUp(current, alignment)};
+ offset = 0 - buffer + aligned_buffer;
+ }
+
+ /**
+ * Get the current buffer offset.
+ *
+ * @return The current allocating offset.
+ */
+ u64 GetCurrentOffset() const {
+ return offset;
+ }
+
+ /**
+ * Get the current buffer size.
+ *
+ * @return The size of the current buffer.
+ */
+ u64 GetSize() const {
+ return size;
+ }
+
+ /**
+ * Get the remaining size that can be allocated.
+ *
+ * @return The remaining size left in the buffer.
+ */
+ u64 GetRemainingSize() const {
+ return size - offset;
+ }
+
+private:
+ /// The buffer into which we are allocating.
+ u64 buffer;
+ /// Size of the buffer we're allocating to.
+ u64 size;
+ /// Current offset into the buffer, an error will be thrown if it exceeds size.
+ u64 offset{};
+};
+
+} // namespace AudioCore