diff options
Diffstat (limited to 'src/audio_core/hle/source.h')
| -rw-r--r-- | src/audio_core/hle/source.h | 144 |
1 files changed, 144 insertions, 0 deletions
diff --git a/src/audio_core/hle/source.h b/src/audio_core/hle/source.h new file mode 100644 index 000000000..7ee08d424 --- /dev/null +++ b/src/audio_core/hle/source.h | |||
| @@ -0,0 +1,144 @@ | |||
| 1 | // Copyright 2016 Citra Emulator Project | ||
| 2 | // Licensed under GPLv2 or any later version | ||
| 3 | // Refer to the license.txt file included. | ||
| 4 | |||
| 5 | #pragma once | ||
| 6 | |||
| 7 | #include <array> | ||
| 8 | #include <queue> | ||
| 9 | #include <vector> | ||
| 10 | |||
| 11 | #include "audio_core/codec.h" | ||
| 12 | #include "audio_core/hle/common.h" | ||
| 13 | #include "audio_core/hle/dsp.h" | ||
| 14 | #include "audio_core/hle/filter.h" | ||
| 15 | #include "audio_core/interpolate.h" | ||
| 16 | |||
| 17 | #include "common/common_types.h" | ||
| 18 | |||
| 19 | namespace DSP { | ||
| 20 | namespace HLE { | ||
| 21 | |||
| 22 | /** | ||
| 23 | * This module performs: | ||
| 24 | * - Buffer management | ||
| 25 | * - Decoding of buffers | ||
| 26 | * - Buffer resampling and interpolation | ||
| 27 | * - Per-source filtering (SimpleFilter, BiquadFilter) | ||
| 28 | * - Per-source gain | ||
| 29 | * - Other per-source processing | ||
| 30 | */ | ||
| 31 | class Source final { | ||
| 32 | public: | ||
| 33 | explicit Source(size_t source_id_) : source_id(source_id_) { | ||
| 34 | Reset(); | ||
| 35 | } | ||
| 36 | |||
| 37 | /// Resets internal state. | ||
| 38 | void Reset(); | ||
| 39 | |||
| 40 | /** | ||
| 41 | * This is called once every audio frame. This performs per-source processing every frame. | ||
| 42 | * @param config The new configuration we've got for this Source from the application. | ||
| 43 | * @param adpcm_coeffs ADPCM coefficients to use if config tells us to use them (may contain invalid values otherwise). | ||
| 44 | * @return The current status of this Source. This is given back to the emulated application via SharedMemory. | ||
| 45 | */ | ||
| 46 | SourceStatus::Status Tick(SourceConfiguration::Configuration& config, const s16_le (&adpcm_coeffs)[16]); | ||
| 47 | |||
| 48 | /** | ||
| 49 | * Mix this source's output into dest, using the gains for the `intermediate_mix_id`-th intermediate mixer. | ||
| 50 | * @param dest The QuadFrame32 to mix into. | ||
| 51 | * @param intermediate_mix_id The id of the intermediate mix whose gains we are using. | ||
| 52 | */ | ||
| 53 | void MixInto(QuadFrame32& dest, size_t intermediate_mix_id) const; | ||
| 54 | |||
| 55 | private: | ||
| 56 | const size_t source_id; | ||
| 57 | StereoFrame16 current_frame; | ||
| 58 | |||
| 59 | using Format = SourceConfiguration::Configuration::Format; | ||
| 60 | using InterpolationMode = SourceConfiguration::Configuration::InterpolationMode; | ||
| 61 | using MonoOrStereo = SourceConfiguration::Configuration::MonoOrStereo; | ||
| 62 | |||
| 63 | /// Internal representation of a buffer for our buffer queue | ||
| 64 | struct Buffer { | ||
| 65 | PAddr physical_address; | ||
| 66 | u32 length; | ||
| 67 | u8 adpcm_ps; | ||
| 68 | std::array<u16, 2> adpcm_yn; | ||
| 69 | bool adpcm_dirty; | ||
| 70 | bool is_looping; | ||
| 71 | u16 buffer_id; | ||
| 72 | |||
| 73 | MonoOrStereo mono_or_stereo; | ||
| 74 | Format format; | ||
| 75 | |||
| 76 | bool from_queue; | ||
| 77 | }; | ||
| 78 | |||
| 79 | struct BufferOrder { | ||
| 80 | bool operator() (const Buffer& a, const Buffer& b) const { | ||
| 81 | // Lower buffer_id comes first. | ||
| 82 | return a.buffer_id > b.buffer_id; | ||
| 83 | } | ||
| 84 | }; | ||
| 85 | |||
| 86 | struct { | ||
| 87 | |||
| 88 | // State variables | ||
| 89 | |||
| 90 | bool enabled = false; | ||
| 91 | u16 sync = 0; | ||
| 92 | |||
| 93 | // Mixing | ||
| 94 | |||
| 95 | std::array<std::array<float, 4>, 3> gain = {}; | ||
| 96 | |||
| 97 | // Buffer queue | ||
| 98 | |||
| 99 | std::priority_queue<Buffer, std::vector<Buffer>, BufferOrder> input_queue; | ||
| 100 | MonoOrStereo mono_or_stereo = MonoOrStereo::Mono; | ||
| 101 | Format format = Format::ADPCM; | ||
| 102 | |||
| 103 | // Current buffer | ||
| 104 | |||
| 105 | u32 current_sample_number = 0; | ||
| 106 | u32 next_sample_number = 0; | ||
| 107 | std::vector<std::array<s16, 2>> current_buffer; | ||
| 108 | |||
| 109 | // buffer_id state | ||
| 110 | |||
| 111 | bool buffer_update = false; | ||
| 112 | u32 current_buffer_id = 0; | ||
| 113 | |||
| 114 | // Decoding state | ||
| 115 | |||
| 116 | std::array<s16, 16> adpcm_coeffs = {}; | ||
| 117 | Codec::ADPCMState adpcm_state = {}; | ||
| 118 | |||
| 119 | // Resampling state | ||
| 120 | |||
| 121 | float rate_multiplier = 1.0; | ||
| 122 | InterpolationMode interpolation_mode = InterpolationMode::Polyphase; | ||
| 123 | AudioInterp::State interp_state = {}; | ||
| 124 | |||
| 125 | // Filter state | ||
| 126 | |||
| 127 | SourceFilters filters; | ||
| 128 | |||
| 129 | } state; | ||
| 130 | |||
| 131 | // Internal functions | ||
| 132 | |||
| 133 | /// INTERNAL: Update our internal state based on the current config. | ||
| 134 | void ParseConfig(SourceConfiguration::Configuration& config, const s16_le (&adpcm_coeffs)[16]); | ||
| 135 | /// INTERNAL: Generate the current audio output for this frame based on our internal state. | ||
| 136 | void GenerateFrame(); | ||
| 137 | /// INTERNAL: Dequeues a buffer and does preprocessing on it (decoding, resampling). Puts it into current_buffer. | ||
| 138 | bool DequeueBuffer(); | ||
| 139 | /// INTERNAL: Generates a SourceStatus::Status based on our internal state. | ||
| 140 | SourceStatus::Status GetCurrentStatus(); | ||
| 141 | }; | ||
| 142 | |||
| 143 | } // namespace HLE | ||
| 144 | } // namespace DSP | ||