mirror of https://github.com/FFmpeg/FFmpeg.git
parent
b874e2d0a0
commit
36c06b0913
4 changed files with 308 additions and 0 deletions
@ -0,0 +1,226 @@ |
|||||||
|
/*
|
||||||
|
* Copyright (c) 2011 Stefano Sabatini |
||||||
|
* |
||||||
|
* This file is part of FFmpeg. |
||||||
|
* |
||||||
|
* FFmpeg is free software; you can redistribute it and/or |
||||||
|
* modify it under the terms of the GNU Lesser General Public |
||||||
|
* License as published by the Free Software Foundation; either |
||||||
|
* version 2.1 of the License, or (at your option) any later version. |
||||||
|
* |
||||||
|
* FFmpeg 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 |
||||||
|
* Lesser General Public License for more details. |
||||||
|
* |
||||||
|
* You should have received a copy of the GNU Lesser General Public |
||||||
|
* License along with FFmpeg; if not, write to the Free Software |
||||||
|
* Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA |
||||||
|
*/ |
||||||
|
|
||||||
|
/**
|
||||||
|
* @file |
||||||
|
* eval audio source |
||||||
|
*/ |
||||||
|
|
||||||
|
#include "libavutil/audioconvert.h" |
||||||
|
#include "libavutil/avassert.h" |
||||||
|
#include "libavutil/avstring.h" |
||||||
|
#include "libavutil/eval.h" |
||||||
|
#include "libavutil/opt.h" |
||||||
|
#include "avfilter.h" |
||||||
|
#include "internal.h" |
||||||
|
|
||||||
|
static const char *var_names[] = { |
||||||
|
"n", ///< number of frame
|
||||||
|
"t", ///< timestamp expressed in seconds
|
||||||
|
"s", ///< sample rate
|
||||||
|
NULL |
||||||
|
}; |
||||||
|
|
||||||
|
enum var_name { |
||||||
|
VAR_N, |
||||||
|
VAR_T, |
||||||
|
VAR_S, |
||||||
|
VAR_VARS_NB |
||||||
|
}; |
||||||
|
|
||||||
|
typedef struct { |
||||||
|
const AVClass *class; |
||||||
|
char *sample_rate_str; |
||||||
|
int sample_rate; |
||||||
|
int64_t chlayout; |
||||||
|
int nb_channels; |
||||||
|
int64_t pts; |
||||||
|
AVExpr *expr[8]; |
||||||
|
char *expr_str[8]; |
||||||
|
int nb_samples; ///< number of samples per requested frame
|
||||||
|
uint64_t n; |
||||||
|
double var_values[VAR_VARS_NB]; |
||||||
|
} EvalContext; |
||||||
|
|
||||||
|
#define OFFSET(x) offsetof(EvalContext, x) |
||||||
|
|
||||||
|
static const AVOption eval_options[]= { |
||||||
|
{ "nb_samples", "set the number of samples per requested frame", OFFSET(nb_samples), FF_OPT_TYPE_INT, {.dbl = 1024}, 0, INT_MAX }, |
||||||
|
{ "n", "set the number of samples per requested frame", OFFSET(nb_samples), FF_OPT_TYPE_INT, {.dbl = 1024}, 0, INT_MAX }, |
||||||
|
{ "sample_rate", "set the sample rate", OFFSET(sample_rate_str), FF_OPT_TYPE_STRING, {.str = "44100"}, 0, INT_MAX }, |
||||||
|
{ "s", "set the sample rate", OFFSET(sample_rate_str), FF_OPT_TYPE_STRING, {.str = "44100"}, 0, INT_MAX }, |
||||||
|
{NULL}, |
||||||
|
}; |
||||||
|
|
||||||
|
static const char *eval_get_name(void *ctx) |
||||||
|
{ |
||||||
|
return "aevalsrc"; |
||||||
|
} |
||||||
|
|
||||||
|
static const AVClass eval_class = { |
||||||
|
"AEvalSrcContext", |
||||||
|
eval_get_name, |
||||||
|
eval_options |
||||||
|
}; |
||||||
|
|
||||||
|
static int init(AVFilterContext *ctx, const char *args, void *opaque) |
||||||
|
{ |
||||||
|
EvalContext *eval = ctx->priv; |
||||||
|
char *args1 = av_strdup(args); |
||||||
|
char *expr, *buf, *bufptr; |
||||||
|
int ret, i; |
||||||
|
|
||||||
|
eval->class = &eval_class; |
||||||
|
av_opt_set_defaults(eval); |
||||||
|
|
||||||
|
/* parse expressions */ |
||||||
|
buf = args1; |
||||||
|
i = 0; |
||||||
|
while (expr = strtok_r(buf, ":", &bufptr)) { |
||||||
|
if (i >= 8) { |
||||||
|
av_log(ctx, AV_LOG_ERROR, |
||||||
|
"More than 8 expressions provided, unsupported.\n"); |
||||||
|
ret = AVERROR(EINVAL); |
||||||
|
return ret; |
||||||
|
} |
||||||
|
ret = av_expr_parse(&eval->expr[i], expr, var_names, |
||||||
|
NULL, NULL, NULL, NULL, 0, ctx); |
||||||
|
if (ret < 0) |
||||||
|
goto end; |
||||||
|
i++; |
||||||
|
if (bufptr && *bufptr == ':') { /* found last expression */ |
||||||
|
bufptr++; |
||||||
|
break; |
||||||
|
} |
||||||
|
buf = NULL; |
||||||
|
} |
||||||
|
|
||||||
|
/* guess channel layout from nb expressions/channels */ |
||||||
|
eval->nb_channels = i; |
||||||
|
eval->chlayout = av_get_default_channel_layout(eval->nb_channels); |
||||||
|
if (!eval->chlayout) { |
||||||
|
av_log(ctx, AV_LOG_ERROR, "Invalid number of channels '%d' provided\n", |
||||||
|
eval->nb_channels); |
||||||
|
ret = AVERROR(EINVAL); |
||||||
|
goto end; |
||||||
|
} |
||||||
|
|
||||||
|
if (bufptr && (ret = av_set_options_string(eval, bufptr, "=", ":")) < 0) |
||||||
|
goto end; |
||||||
|
|
||||||
|
if ((ret = ff_parse_sample_rate(&eval->sample_rate, eval->sample_rate_str, ctx))) |
||||||
|
goto end; |
||||||
|
eval->n = 0; |
||||||
|
|
||||||
|
end: |
||||||
|
av_free(args1); |
||||||
|
return ret; |
||||||
|
} |
||||||
|
|
||||||
|
static void uninit(AVFilterContext *ctx) |
||||||
|
{ |
||||||
|
EvalContext *eval = ctx->priv; |
||||||
|
int i; |
||||||
|
|
||||||
|
for (i = 0; i < 8; i++) { |
||||||
|
av_expr_free(eval->expr[i]); |
||||||
|
eval->expr[i] = NULL; |
||||||
|
} |
||||||
|
av_freep(&eval->sample_rate_str); |
||||||
|
} |
||||||
|
|
||||||
|
static int config_props(AVFilterLink *outlink) |
||||||
|
{ |
||||||
|
EvalContext *eval = outlink->src->priv; |
||||||
|
char buf[128]; |
||||||
|
|
||||||
|
outlink->time_base = (AVRational){1, eval->sample_rate}; |
||||||
|
outlink->sample_rate = eval->sample_rate; |
||||||
|
|
||||||
|
eval->var_values[VAR_S] = eval->sample_rate; |
||||||
|
|
||||||
|
av_get_channel_layout_string(buf, sizeof(buf), 0, eval->chlayout); |
||||||
|
|
||||||
|
av_log(outlink->src, AV_LOG_INFO, |
||||||
|
"sample_rate:%d chlayout:%s\n", eval->sample_rate, buf); |
||||||
|
|
||||||
|
return 0; |
||||||
|
} |
||||||
|
|
||||||
|
static int query_formats(AVFilterContext *ctx) |
||||||
|
{ |
||||||
|
EvalContext *eval = ctx->priv; |
||||||
|
enum AVSampleFormat sample_fmts[] = { AV_SAMPLE_FMT_DBL, AV_SAMPLE_FMT_NONE }; |
||||||
|
int64_t chlayouts[] = { eval->chlayout, -1 }; |
||||||
|
int packing_fmts[] = { AVFILTER_PLANAR, -1 }; |
||||||
|
|
||||||
|
avfilter_set_common_sample_formats (ctx, avfilter_make_format_list(sample_fmts)); |
||||||
|
avfilter_set_common_channel_layouts(ctx, avfilter_make_format64_list(chlayouts)); |
||||||
|
avfilter_set_common_packing_formats(ctx, avfilter_make_format_list(packing_fmts)); |
||||||
|
|
||||||
|
return 0; |
||||||
|
} |
||||||
|
|
||||||
|
static int request_frame(AVFilterLink *outlink) |
||||||
|
{ |
||||||
|
EvalContext *eval = outlink->src->priv; |
||||||
|
AVFilterBufferRef *samplesref; |
||||||
|
int i, j; |
||||||
|
|
||||||
|
samplesref = avfilter_get_audio_buffer(outlink, AV_PERM_WRITE, eval->nb_samples); |
||||||
|
|
||||||
|
/* evaluate expression for each single sample and for each channel */ |
||||||
|
for (i = 0; i < eval->nb_samples; i++, eval->n++) { |
||||||
|
eval->var_values[VAR_N] = eval->n; |
||||||
|
eval->var_values[VAR_T] = eval->var_values[VAR_N] * (double)1/eval->sample_rate; |
||||||
|
|
||||||
|
for (j = 0; j < eval->nb_channels; j++) { |
||||||
|
*((double *) samplesref->data[j] + i) = |
||||||
|
av_expr_eval(eval->expr[j], eval->var_values, NULL); |
||||||
|
} |
||||||
|
} |
||||||
|
|
||||||
|
samplesref->pts = eval->pts; |
||||||
|
samplesref->pos = -1; |
||||||
|
samplesref->audio->sample_rate = eval->sample_rate; |
||||||
|
eval->pts += eval->nb_samples; |
||||||
|
|
||||||
|
avfilter_filter_samples(outlink, samplesref); |
||||||
|
|
||||||
|
return 0; |
||||||
|
} |
||||||
|
|
||||||
|
AVFilter avfilter_asrc_aevalsrc = { |
||||||
|
.name = "aevalsrc", |
||||||
|
.description = NULL_IF_CONFIG_SMALL("Generate an audio signal generated by an expression."), |
||||||
|
|
||||||
|
.query_formats = query_formats, |
||||||
|
.init = init, |
||||||
|
.uninit = uninit, |
||||||
|
.priv_size = sizeof(EvalContext), |
||||||
|
|
||||||
|
.inputs = (AVFilterPad[]) {{ .name = NULL}}, |
||||||
|
|
||||||
|
.outputs = (AVFilterPad[]) {{ .name = "default", |
||||||
|
.type = AVMEDIA_TYPE_AUDIO, |
||||||
|
.config_props = config_props, |
||||||
|
.request_frame = request_frame, }, |
||||||
|
{ .name = NULL}}, |
||||||
|
}; |
Loading…
Reference in new issue