// SPDX-License-Identifier: GPL-2.0-or-later // Copyright The Music Player Daemon Project #ifndef AUDIO_OUTPUT_SOURCE_HXX #define AUDIO_OUTPUT_SOURCE_HXX #include "SharedPipeConsumer.hxx" #include "ReplayGainMode.hxx" #include "pcm/AudioFormat.hxx" #include "pcm/Buffer.hxx" #include "pcm/Dither.hxx" #include "thread/Mutex.hxx" #include #include #include #include #include struct MusicChunk; struct Tag; class Filter; class PreparedFilter; /** * Source of audio data to be played by an #AudioOutput. It receives * #MusicChunk instances from a #MusicPipe (via #SharedPipeConsumer). * It applies configured filters, ReplayGain and returns plain PCM * data. */ class AudioOutputSource { /** * The audio_format in which audio data is received from the * player thread (which in turn receives it from the decoder). */ AudioFormat in_audio_format = AudioFormat::Undefined(); ReplayGainMode replay_gain_mode = ReplayGainMode::OFF; /** * A reference to the #MusicPipe and the current position. */ SharedPipeConsumer pipe; /** * The serial number of the last replay gain info. 0 means no * replay gain info was available. */ unsigned replay_gain_serial; /** * The serial number of the last replay gain info by the * "other" chunk during cross-fading. */ unsigned other_replay_gain_serial; /** * The replay_gain_filter_plugin instance of this audio * output. */ std::unique_ptr replay_gain_filter; /** * The replay_gain_filter_plugin instance of this audio * output, to be applied to the second chunk during * cross-fading. */ std::unique_ptr other_replay_gain_filter; /** * The buffer used to allocate the cross-fading result. */ PcmBuffer cross_fade_buffer; /** * The dithering state for cross-fading two streams. */ PcmDither cross_fade_dither; /** * The filter object of this audio output. This is an * instance of chain_filter_plugin. */ std::unique_ptr filter; /** * The #MusicChunk currently being processed (see * #pending_tag, #pending_data). */ const MusicChunk *current_chunk = nullptr; /** * The #Tag to be processed by the #AudioOutput. It is owned * by #current_chunk (MusicChunk::tag). */ const Tag *pending_tag; /** * Filtered #MusicChunk PCM data to be processed by the * #AudioOutput. */ std::span pending_data; public: AudioOutputSource() noexcept; ~AudioOutputSource() noexcept; void SetReplayGainMode(ReplayGainMode _mode) noexcept { replay_gain_mode = _mode; } bool IsOpen() const { return in_audio_format.IsDefined(); } const AudioFormat &GetInputAudioFormat() const { assert(IsOpen()); return in_audio_format; } AudioFormat Open(AudioFormat audio_format, const MusicPipe &_pipe, PreparedFilter *prepared_replay_gain_filter, PreparedFilter *prepared_other_replay_gain_filter, PreparedFilter &prepared_filter); void Close() noexcept; void Cancel() noexcept; /** * Ensure that ReadTag() or PeekData() return any input. * * Throws on error * * @param mutex the #Mutex which protects the * #SharedPipeConsumer; it is locked by the caller, and may be * unlocked temporarily by this method * @return true if any input is available, false if the source * has (temporarily?) run empty */ bool Fill(Mutex &mutex); /** * Reads the #Tag to be processed. Be sure to call Fill() * successfully before calling this metohd. */ const Tag *ReadTag() noexcept { assert(current_chunk != nullptr); return std::exchange(pending_tag, nullptr); } /** * Returns the remaining filtered PCM data be played. The * caller shall use ConsumeData() to mark portions of the * return value as "consumed". * * Be sure to call Fill() successfully before calling this * metohd. */ std::span PeekData() const noexcept { return pending_data; } /** * Mark portions of the PeekData() return value as "consumed". */ void ConsumeData(size_t nbytes) noexcept; bool IsChunkConsumed(const MusicChunk &chunk) const noexcept { assert(IsOpen()); return pipe.IsConsumed(chunk); } void ClearTailChunk(const MusicChunk &chunk) noexcept { pipe.ClearTail(chunk); } /** * Wrapper for Filter::Flush(). */ std::span Flush(); private: void OpenFilter(AudioFormat audio_format, PreparedFilter *prepared_replay_gain_filter, PreparedFilter *prepared_other_replay_gain_filter, PreparedFilter &prepared_filter); void CloseFilter() noexcept; std::span GetChunkData(const MusicChunk &chunk, Filter *replay_gain_filter, unsigned *replay_gain_serial_p); std::span FilterChunk(const MusicChunk &chunk); void DropCurrentChunk() noexcept { assert(current_chunk != nullptr); pipe.Consume(*std::exchange(current_chunk, nullptr)); } }; #endif