/* * Copyright 2003-2021 The Music Player Daemon Project * http://www.musicpd.org * * This program is free software; you can redistribute it and/or modify * it under the terms of the GNU General Public License as published by * the Free Software Foundation; either version 2 of the License, or * (at your option) any later version. * * This program is distributed in the hope that it will be useful, * but WITHOUT ANY WARRANTY; without even the implied warranty of * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the * GNU General Public License for more details. * * You should have received a copy of the GNU General Public License along * with this program; if not, write to the Free Software Foundation, Inc., * 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA. */ #include "Filtered.hxx" #include "Interface.hxx" #include "Domain.hxx" #include "lib/fmt/AudioFormatFormatter.hxx" #include "lib/fmt/ExceptionFormatter.hxx" #include "mixer/MixerInternal.hxx" #include "mixer/plugins/SoftwareMixerPlugin.hxx" #include "filter/plugins/ConvertFilterPlugin.hxx" #include "util/RuntimeError.hxx" #include "util/StringBuffer.hxx" #include "Log.hxx" bool FilteredAudioOutput::SupportsEnableDisable() const noexcept { return output->SupportsEnableDisable(); } bool FilteredAudioOutput::SupportsPause() const noexcept { return output->SupportsPause(); } std::map FilteredAudioOutput::GetAttributes() const noexcept { return output->GetAttributes(); } void FilteredAudioOutput::SetAttribute(std::string &&_name, std::string &&_value) { output->SetAttribute(std::move(_name), std::move(_value)); } void FilteredAudioOutput::Enable() { try { output->Enable(); } catch (...) { std::throw_with_nested(FormatRuntimeError("Failed to enable output %s", GetLogName())); } } void FilteredAudioOutput::Disable() noexcept { output->Disable(); } void FilteredAudioOutput::ConfigureConvertFilter() { try { convert_filter_set(convert_filter.Get(), out_audio_format); } catch (...) { std::throw_with_nested(FormatRuntimeError("Failed to convert for %s", GetLogName())); } } void FilteredAudioOutput::OpenOutputAndConvert(AudioFormat desired_audio_format) { out_audio_format = desired_audio_format; try { output->Open(out_audio_format); } catch (...) { std::throw_with_nested(FormatRuntimeError("Failed to open %s", GetLogName())); } FmtDebug(output_domain, "opened {} audio_format={}", GetLogName(), out_audio_format); try { ConfigureConvertFilter(); } catch (...) { output->Close(); if (out_audio_format.format == SampleFormat::DSD) { /* if the audio output supports DSD, but not the given sample rate, it asks MPD to resample; resampling DSD however is not implemented; our last resort is to give up DSD and fall back to PCM */ LogError(std::current_exception()); LogError(output_domain, "Retrying without DSD"); desired_audio_format.format = SampleFormat::FLOAT; OpenOutputAndConvert(desired_audio_format); return; } throw; } } void FilteredAudioOutput::CloseOutput(bool drain) noexcept { if (drain) { try { Drain(); } catch (...) { FmtError(output_domain, "Failed to drain {}: {}", GetLogName(), std::current_exception()); } } else Cancel(); output->Close(); } void FilteredAudioOutput::OpenSoftwareMixer() noexcept { if (mixer != nullptr && mixer->IsPlugin(software_mixer_plugin)) software_mixer_set_filter(*mixer, volume_filter.Get()); } void FilteredAudioOutput::CloseSoftwareMixer() noexcept { if (mixer != nullptr && mixer->IsPlugin(software_mixer_plugin)) software_mixer_set_filter(*mixer, nullptr); } void FilteredAudioOutput::Close(bool drain) noexcept { CloseOutput(drain); CloseSoftwareMixer(); FmtDebug(output_domain, "closed {}", GetLogName()); } std::chrono::steady_clock::duration FilteredAudioOutput::Delay() noexcept { return output->Delay(); } void FilteredAudioOutput::SendTag(const Tag &tag) { output->SendTag(tag); } std::size_t FilteredAudioOutput::Play(std::span src) { return output->Play(src); } void FilteredAudioOutput::Drain() { output->Drain(); } void FilteredAudioOutput::Interrupt() noexcept { output->Interrupt(); } void FilteredAudioOutput::Cancel() noexcept { output->Cancel(); } void FilteredAudioOutput::BeginPause() noexcept { Cancel(); } bool FilteredAudioOutput::IteratePause() { return output->Pause(); }