avfilter/af_replaygain: export results into read-only options

pull/389/head
Paul B Mahol 2 years ago
parent 782ea8b2e5
commit 7153642992
  1. 10
      doc/filters.texi
  2. 35
      libavfilter/af_replaygain.c

@ -6058,6 +6058,16 @@ ReplayGain scanner filter. This filter takes an audio stream as an input and
outputs it unchanged. outputs it unchanged.
At end of filtering it displays @code{track_gain} and @code{track_peak}. At end of filtering it displays @code{track_gain} and @code{track_peak}.
The filter accepts the following exported read-only options:
@table @option
@item track_gain
Exported track gain in dB at end of stream.
@item track_peak
Exported track peak at end of stream.
@end table
@section resample @section resample
Convert the audio sample format, sample rate and channel layout. It is Convert the audio sample format, sample rate and channel layout. It is

@ -23,8 +23,11 @@
* ReplayGain scanner * ReplayGain scanner
*/ */
#include <float.h>
#include "libavutil/avassert.h" #include "libavutil/avassert.h"
#include "libavutil/channel_layout.h" #include "libavutil/channel_layout.h"
#include "libavutil/opt.h"
#include "audio.h" #include "audio.h"
#include "avfilter.h" #include "avfilter.h"
#include "internal.h" #include "internal.h"
@ -306,8 +309,11 @@ static const ReplayGainFreqInfo freqinfos[] =
}; };
typedef struct ReplayGainContext { typedef struct ReplayGainContext {
const AVClass *class;
uint32_t histogram[HISTOGRAM_SLOTS]; uint32_t histogram[HISTOGRAM_SLOTS];
float peak; float peak;
float gain;
int yule_hist_i, butter_hist_i; int yule_hist_i, butter_hist_i;
const double *yule_coeff_a; const double *yule_coeff_a;
const double *yule_coeff_b; const double *yule_coeff_b;
@ -576,15 +582,24 @@ static int filter_frame(AVFilterLink *inlink, AVFrame *in)
return ff_filter_frame(outlink, in); return ff_filter_frame(outlink, in);
} }
static av_cold void uninit(AVFilterContext *ctx) static int request_frame(AVFilterLink *outlink)
{ {
AVFilterContext *ctx = outlink->src;
ReplayGainContext *s = ctx->priv; ReplayGainContext *s = ctx->priv;
float gain = calc_replaygain(s->histogram); int ret = 0;
av_log(ctx, AV_LOG_INFO, "track_gain = %+.2f dB\n", gain); ret = ff_request_frame(ctx->inputs[0]);
if (ret == AVERROR_EOF) {
s->gain = calc_replaygain(s->histogram);
av_log(ctx, AV_LOG_INFO, "track_gain = %+.2f dB\n", s->gain);
av_log(ctx, AV_LOG_INFO, "track_peak = %.6f\n", s->peak); av_log(ctx, AV_LOG_INFO, "track_peak = %.6f\n", s->peak);
} }
return ret;
}
static const AVFilterPad replaygain_inputs[] = { static const AVFilterPad replaygain_inputs[] = {
{ {
.name = "default", .name = "default",
@ -598,14 +613,26 @@ static const AVFilterPad replaygain_outputs[] = {
{ {
.name = "default", .name = "default",
.type = AVMEDIA_TYPE_AUDIO, .type = AVMEDIA_TYPE_AUDIO,
.request_frame = request_frame,
}, },
}; };
#define OFFSET(x) offsetof(ReplayGainContext, x)
#define FLAGS AV_OPT_FLAG_AUDIO_PARAM|AV_OPT_FLAG_FILTERING_PARAM|AV_OPT_FLAG_EXPORT|AV_OPT_FLAG_READONLY
static const AVOption replaygain_options[] = {
{ "track_gain", "track gain (dB)", OFFSET(gain), AV_OPT_TYPE_FLOAT,{.dbl=0}, -FLT_MAX, FLT_MAX, FLAGS },
{ "track_peak", "track peak", OFFSET(peak), AV_OPT_TYPE_FLOAT,{.dbl=0}, -FLT_MAX, FLT_MAX, FLAGS },
{ NULL }
};
AVFILTER_DEFINE_CLASS(replaygain);
const AVFilter ff_af_replaygain = { const AVFilter ff_af_replaygain = {
.name = "replaygain", .name = "replaygain",
.description = NULL_IF_CONFIG_SMALL("ReplayGain scanner."), .description = NULL_IF_CONFIG_SMALL("ReplayGain scanner."),
.uninit = uninit,
.priv_size = sizeof(ReplayGainContext), .priv_size = sizeof(ReplayGainContext),
.priv_class = &replaygain_class,
.flags = AVFILTER_FLAG_METADATA_ONLY, .flags = AVFILTER_FLAG_METADATA_ONLY,
FILTER_INPUTS(replaygain_inputs), FILTER_INPUTS(replaygain_inputs),
FILTER_OUTPUTS(replaygain_outputs), FILTER_OUTPUTS(replaygain_outputs),

Loading…
Cancel
Save