/* * Copyright 2003-2019 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. */ #ifndef MPD_FFMPEG_FILTER_HXX #define MPD_FFMPEG_FILTER_HXX #include "Error.hxx" extern "C" { #include <libavfilter/avfilter.h> } #include <utility> struct AudioFormat; namespace Ffmpeg { class FilterInOut { friend class FilterGraph; AVFilterInOut *io = nullptr; explicit FilterInOut(AVFilterInOut *_io) noexcept :io(_io) {} public: FilterInOut() = default; FilterInOut(const char *name, AVFilterContext &context) :io(avfilter_inout_alloc()) { if (io == nullptr) throw std::bad_alloc(); io->name = av_strdup(name); io->filter_ctx = &context; io->pad_idx = 0; io->next = nullptr; } FilterInOut(FilterInOut &&src) noexcept :io(std::exchange(src.io, nullptr)) {} ~FilterInOut() noexcept { if (io != nullptr) avfilter_inout_free(&io); } FilterInOut &operator=(FilterInOut &&src) noexcept { using std::swap; swap(io, src.io); return *this; } auto *get() noexcept { return io; } }; class FilterContext { AVFilterContext *context = nullptr; public: FilterContext() = default; FilterContext(const AVFilter &filt, const char *name, const char *args, void *opaque, AVFilterGraph &graph_ctx) { int err = avfilter_graph_create_filter(&context, &filt, name, args, opaque, &graph_ctx); if (err < 0) throw MakeFfmpegError(err, "avfilter_graph_create_filter() failed"); } FilterContext(const AVFilter &filt, const char *name, AVFilterGraph &graph_ctx) :FilterContext(filt, name, nullptr, nullptr, graph_ctx) {} FilterContext(FilterContext &&src) noexcept :context(std::exchange(src.context, nullptr)) {} ~FilterContext() noexcept { if (context != nullptr) avfilter_free(context); } FilterContext &operator=(FilterContext &&src) noexcept { using std::swap; swap(context, src.context); return *this; } /** * Create an "abuffer" filter. * * @param the input audio format; may be modified by the * function to ask the caller to do format conversion */ static FilterContext MakeAudioBufferSource(AudioFormat &audio_format, AVFilterGraph &graph_ctx); /** * Create an "abuffersink" filter. */ static FilterContext MakeAudioBufferSink(AVFilterGraph &graph_ctx); auto &operator*() noexcept { return *context; } auto *get() noexcept { return context; } }; class FilterGraph { AVFilterGraph *graph = nullptr; public: FilterGraph(std::nullptr_t) noexcept {} FilterGraph():graph(avfilter_graph_alloc()) { if (graph == nullptr) throw std::bad_alloc(); } FilterGraph(FilterGraph &&src) noexcept :graph(std::exchange(src.graph, nullptr)) {} ~FilterGraph() noexcept { if (graph != nullptr) avfilter_graph_free(&graph); } FilterGraph &operator=(FilterGraph &&src) noexcept { using std::swap; swap(graph, src.graph); return *this; } auto &operator*() noexcept { return *graph; } auto *operator->() noexcept { return graph; } std::pair<FilterInOut, FilterInOut> Parse(const char *filters, FilterInOut &&inputs, FilterInOut &&outputs) { int err = avfilter_graph_parse_ptr(graph, filters, &inputs.io, &outputs.io, nullptr); if (err < 0) throw MakeFfmpegError(err, "avfilter_graph_parse_ptr() failed"); return std::make_pair(std::move(inputs), std::move(outputs)); } std::pair<FilterInOut, FilterInOut> Parse(const char *filters) { AVFilterInOut *inputs, *outputs; int err = avfilter_graph_parse2(graph, filters, &inputs, &outputs); if (err < 0) throw MakeFfmpegError(err, "avfilter_graph_parse2() failed"); return std::make_pair(FilterInOut{inputs}, FilterInOut{outputs}); } void CheckAndConfigure() { int err = avfilter_graph_config(graph, nullptr); if (err < 0) throw MakeFfmpegError(err, "avfilter_graph_config() failed"); } }; } #endif