1 /*
2 * Copyright (c) 2013 Paul B Mahol
3 *
4 * This file is part of FFmpeg.
5 *
6 * FFmpeg is free software; you can redistribute it and/or
7 * modify it under the terms of the GNU Lesser General Public
8 * License as published by the Free Software Foundation; either
9 * version 2.1 of the License, or (at your option) any later version.
10 *
11 * FFmpeg is distributed in the hope that it will be useful,
12 * but WITHOUT ANY WARRANTY; without even the implied warranty of
13 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
14 * Lesser General Public License for more details.
15 *
16 * You should have received a copy of the GNU Lesser General Public
17 * License along with FFmpeg; if not, write to the Free Software
18 * Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA
19 */
20
28
40
45
46 #define OFFSET(x) offsetof(AudioEchoContext, x)
47 #define A AV_OPT_FLAG_AUDIO_PARAM|AV_OPT_FLAG_FILTERING_PARAM
48
55 };
56
58
60 {
61 char *p;
62
63 *nb_items = 1;
64 for (p = item_str; *p; p++) {
65 if (*p == '|')
66 (*nb_items)++;
67 }
68
69 }
70
71 static void fill_items(
char *item_str,
int *nb_items,
float *items)
72 {
73 char *p, *saveptr =
NULL;
74 int i, new_nb_items = 0;
75
76 p = item_str;
77 for (i = 0; i < *nb_items; i++) {
80 if (tstr)
81 new_nb_items += sscanf(tstr, "%f", &items[new_nb_items]) == 1;
82 }
83
84 *nb_items = new_nb_items;
85 }
86
88 {
90
94
98 }
99
101 {
103 int nb_delays, nb_decays, i;
104
108 }
109
112
117
120
121 if (nb_delays != nb_decays) {
122 av_log(ctx,
AV_LOG_ERROR,
"Number of delays %d differs from number of decays %d.\n", nb_delays, nb_decays);
124 }
125
130 }
131
135
136 for (i = 0; i < nb_delays; i++) {
137 if (s->
delay[i] <= 0 || s->
delay[i] > 90000) {
140 }
144 }
145 }
146
148
150 return 0;
151 }
152
154 {
161 };
162 int ret;
163
165 if (!layouts)
168 if (ret < 0)
169 return ret;
170
172 if (!formats)
175 if (ret < 0)
176 return ret;
177
179 if (!formats)
182 }
183
184 #define MOD(a, b) (((a) >= (b)) ? (a) - (b) : (a))
185
186 #define ECHO(name, type, min, max) \
187 static void echo_samples_## name ##p(AudioEchoContext *ctx, \
188 uint8_t **delayptrs, \
189 uint8_t * const *src, uint8_t **dst, \
190 int nb_samples, int channels) \
191 { \
192 const double out_gain = ctx->out_gain; \
193 const double in_gain = ctx->in_gain; \
194 const int nb_echoes = ctx->nb_echoes; \
195 const int max_samples = ctx->max_samples; \
196 int i, j, chan, av_uninit(index); \
197 \
198 av_assert1(channels > 0); /* would corrupt delay_index */ \
199 \
200 for (chan = 0; chan < channels; chan++) { \
201 const type *s = (type *)src[chan]; \
202 type *d = (type *)dst[chan]; \
203 type *dbuf = (type *)delayptrs[chan]; \
204 \
205 index = ctx->delay_index; \
206 for (i = 0; i < nb_samples; i++, s++, d++) { \
207 double out, in; \
208 \
209 in = *s; \
210 out = in * in_gain; \
211 for (j = 0; j < nb_echoes; j++) { \
212 int ix = index + max_samples - ctx->samples[j]; \
213 ix = MOD(ix, max_samples); \
214 out += dbuf[ix] * ctx->decay[j]; \
215 } \
216 out *= out_gain; \
217 \
218 *d = av_clipd(out, min, max); \
219 dbuf[index] = in; \
220 \
221 index = MOD(index + 1, max_samples); \
222 } \
223 } \
224 ctx->delay_index = index; \
225 }
226
227 ECHO(dbl,
double, -1.0, 1.0 )
228 ECHO(flt,
float, -1.0, 1.0 )
229 ECHO(s16, int16_t, INT16_MIN, INT16_MAX)
231
233 {
236 float volume = 1.0;
237 int i;
238
240 s->
samples[i] = s->
delay[i] * outlink->sample_rate / 1000.0;
242 volume += s->
decay[i];
243 }
244
248 }
250
253 "out_gain %f can cause saturation of output\n", s->
out_gain);
254
255 switch (outlink->format) {
260 }
261
262
266
268 outlink->channels,
270 outlink->format, 0);
271 }
272
274 {
278
281 } else {
283 if (!out_frame) {
286 }
288 }
289
292
294
295 if (
frame != out_frame)
297
299 }
300
302 {
305 int ret;
306
308
312
314 if (!frame)
317
322
325
329
331 }
332
333 return ret;
334 }
335
337 {
341 },
343 };
344
346 {
351 },
353 };
354
360 .priv_class = &aecho_class,
365 };
This structure describes decoded (raw) audio or video data.
#define av_realloc_f(p, o, n)
#define AV_LOG_WARNING
Something somehow does not look correct.
Main libavfilter public API header.
AVFILTER_DEFINE_CLASS(aecho)
static void count_items(char *item_str, int *nb_items)
static int filter_frame(AVFilterLink *inlink, AVFrame *frame)
int is_disabled
the enabled state from the last expression evaluation
int av_samples_alloc_array_and_samples(uint8_t ***audio_data, int *linesize, int nb_channels, int nb_samples, enum AVSampleFormat sample_fmt, int align)
Allocate a data pointers array, samples buffer for nb_samples samples, and fill data pointers and lin...
static void fill_items(char *item_str, int *nb_items, float *items)
void(* echo_samples)(struct AudioEchoContext *ctx, uint8_t **delayptrs, uint8_t *const *src, uint8_t **dst, int nb_samples, int channels)
const char * name
Pad name.
static av_cold void uninit(AVFilterContext *ctx)
AVFilterLink ** inputs
array of pointers to input links
int ff_filter_frame(AVFilterLink *link, AVFrame *frame)
Send a frame of data to the next filter.
static const AVOption aecho_options[]
#define ECHO(name, type, min, max)
int64_t pts
Presentation timestamp in time_base units (time when frame should be shown to user).
static int request_frame(AVFilterLink *outlink)
#define AVERROR_EOF
End of file.
A filter pad used for either input or output.
int64_t av_rescale_q(int64_t a, AVRational bq, AVRational cq)
Rescale a 64-bit integer by 2 rational numbers.
A link between two filters.
#define AV_LOG_ERROR
Something went wrong and cannot losslessly be recovered.
static int query_formats(AVFilterContext *ctx)
int av_samples_set_silence(uint8_t **audio_data, int offset, int nb_samples, int nb_channels, enum AVSampleFormat sample_fmt)
Fill an audio buffer with silence.
int sample_rate
samples per second
AVFrame * ff_get_audio_buffer(AVFilterLink *link, int nb_samples)
Request an audio samples buffer with a specific set of permissions.
void av_frame_free(AVFrame **frame)
Free the frame and any dynamically allocated objects in it, e.g.
#define NULL_IF_CONFIG_SMALL(x)
Return NULL if CONFIG_SMALL is true, otherwise the argument without modification. ...
void * priv
private data for use by the filter
#define AV_LOG_DEBUG
Stuff which is only useful for libav* developers.
static int config_output(AVFilterLink *outlink)
AVRational time_base
Define the time base used by the PTS of the frames/samples which will pass through this link...
simple assert() macros that are a bit more flexible than ISO C assert().
typedef void(APIENTRY *FF_PFNGLACTIVETEXTUREPROC)(GLenum texture)
AVFilterContext * src
source filter
static const AVFilterPad inputs[]
static const AVFilterPad outputs[]
A list of supported channel layouts.
int format
format of the frame, -1 if unknown or unset Values correspond to enum AVPixelFormat for video frames...
AVSampleFormat
Audio sample formats.
int av_frame_is_writable(AVFrame *frame)
Check if the frame data is writable.
static const AVFilterPad aecho_outputs[]
Describe the class of an AVClass context structure.
Rational number (pair of numerator and denominator).
const char * name
Filter name.
AVFilterLink ** outputs
array of pointers to output links
enum MovChannelLayoutTag * layouts
static av_cold int init(AVFilterContext *ctx)
char * av_strtok(char *s, const char *delim, char **saveptr)
Split the string into several tokens which can be accessed by successive calls to av_strtok()...
int channels
Number of channels.
AVFilterContext * dst
dest filter
static const AVFilterPad aecho_inputs[]
static enum AVSampleFormat sample_fmts[]
int ff_request_frame(AVFilterLink *link)
Request an input frame from the filter at the other end of the link.
uint8_t ** extended_data
pointers to the data planes/channels.
int nb_samples
number of audio samples (per channel) described by this frame
int av_frame_copy_props(AVFrame *dst, const AVFrame *src)
Copy only "metadata" fields from src to dst.
#define AV_NOPTS_VALUE
Undefined timestamp value.