|
|
|
/*
|
|
|
|
* copyright (c) 2006 Michael Niedermayer <michaelni@gmx.at>
|
|
|
|
*
|
|
|
|
* 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
|
|
|
|
*/
|
|
|
|
|
|
|
|
#include <stdlib.h>
|
|
|
|
|
|
|
|
#include "bsf.h"
|
|
|
|
#include "bsf_internal.h"
|
|
|
|
|
|
|
|
#include "libavutil/log.h"
|
|
|
|
#include "libavutil/opt.h"
|
|
|
|
#include "libavutil/eval.h"
|
|
|
|
|
|
|
|
static const char *const var_names[] = {
|
|
|
|
"n", /// packet index, starting from zero
|
|
|
|
"tb", /// timebase
|
|
|
|
"pts", /// packet presentation timestamp
|
|
|
|
"dts", /// packet decoding timestamp
|
|
|
|
"nopts", /// AV_NOPTS_VALUE
|
|
|
|
"startpts", /// first seen non-AV_NOPTS_VALUE packet timestamp
|
|
|
|
"startdts", /// first seen non-AV_NOPTS_VALUE packet timestamp
|
|
|
|
"duration", "d", /// packet duration
|
|
|
|
"pos", /// original position of packet in its source
|
|
|
|
"size", /// packet size
|
|
|
|
"key" , /// packet keyframe flag
|
|
|
|
"state", /// random-ish state
|
|
|
|
NULL
|
|
|
|
};
|
|
|
|
|
|
|
|
enum var_name {
|
|
|
|
VAR_N,
|
|
|
|
VAR_TB,
|
|
|
|
VAR_PTS,
|
|
|
|
VAR_DTS,
|
|
|
|
VAR_NOPTS,
|
|
|
|
VAR_STARTPTS,
|
|
|
|
VAR_STARTDTS,
|
|
|
|
VAR_DURATION, VAR_D,
|
|
|
|
VAR_POS,
|
|
|
|
VAR_SIZE,
|
|
|
|
VAR_KEY,
|
|
|
|
VAR_STATE,
|
|
|
|
VAR_VARS_NB
|
|
|
|
};
|
|
|
|
|
|
|
|
typedef struct NoiseContext {
|
|
|
|
const AVClass *class;
|
|
|
|
|
|
|
|
char *amount_str;
|
|
|
|
char *drop_str;
|
|
|
|
int dropamount;
|
|
|
|
|
|
|
|
AVExpr *amount_pexpr;
|
|
|
|
AVExpr *drop_pexpr;
|
|
|
|
|
|
|
|
double var_values[VAR_VARS_NB];
|
|
|
|
|
|
|
|
unsigned int state;
|
|
|
|
unsigned int pkt_idx;
|
|
|
|
} NoiseContext;
|
|
|
|
|
|
|
|
static int noise_init(AVBSFContext *ctx)
|
|
|
|
{
|
|
|
|
NoiseContext *s = ctx->priv_data;
|
|
|
|
int ret;
|
|
|
|
|
|
|
|
if (!s->amount_str) {
|
|
|
|
s->amount_str = (!s->drop_str && !s->dropamount) ? av_strdup("-1") : av_strdup("0");
|
|
|
|
if (!s->amount_str)
|
|
|
|
return AVERROR(ENOMEM);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (ctx->par_in->codec_id == AV_CODEC_ID_WRAPPED_AVFRAME &&
|
|
|
|
strcmp(s->amount_str, "0")) {
|
|
|
|
av_log(ctx, AV_LOG_ERROR, "Wrapped AVFrame noising is unsupported\n");
|
|
|
|
return AVERROR_PATCHWELCOME;
|
|
|
|
}
|
|
|
|
|
|
|
|
ret = av_expr_parse(&s->amount_pexpr, s->amount_str,
|
|
|
|
var_names, NULL, NULL, NULL, NULL, 0, ctx);
|
|
|
|
if (ret < 0) {
|
|
|
|
av_log(ctx, AV_LOG_ERROR, "Error in parsing expr for amount: %s\n", s->amount_str);
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (s->drop_str && s->dropamount) {
|
|
|
|
av_log(ctx, AV_LOG_WARNING, "Both drop '%s' and dropamount=%d set. Ignoring dropamount.\n",
|
|
|
|
s->drop_str, s->dropamount);
|
|
|
|
s->dropamount = 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (s->drop_str) {
|
|
|
|
ret = av_expr_parse(&s->drop_pexpr, s->drop_str,
|
|
|
|
var_names, NULL, NULL, NULL, NULL, 0, ctx);
|
|
|
|
if (ret < 0) {
|
|
|
|
av_log(ctx, AV_LOG_ERROR, "Error in parsing expr for drop: %s\n", s->drop_str);
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
s->var_values[VAR_TB] = ctx->time_base_out.den ? av_q2d(ctx->time_base_out) : 0;
|
|
|
|
s->var_values[VAR_NOPTS] = AV_NOPTS_VALUE;
|
|
|
|
s->var_values[VAR_STARTPTS] = AV_NOPTS_VALUE;
|
|
|
|
s->var_values[VAR_STARTDTS] = AV_NOPTS_VALUE;
|
|
|
|
s->var_values[VAR_STATE] = 0;
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
static int noise(AVBSFContext *ctx, AVPacket *pkt)
|
|
|
|
{
|
|
|
|
NoiseContext *s = ctx->priv_data;
|
|
|
|
int i, ret, amount, drop = 0;
|
|
|
|
double res;
|
|
|
|
|
|
|
|
ret = ff_bsf_get_packet_ref(ctx, pkt);
|
|
|
|
if (ret < 0)
|
|
|
|
return ret;
|
|
|
|
|
|
|
|
s->var_values[VAR_N] = s->pkt_idx++;
|
|
|
|
s->var_values[VAR_PTS] = pkt->pts;
|
|
|
|
s->var_values[VAR_DTS] = pkt->dts;
|
|
|
|
s->var_values[VAR_DURATION] =
|
|
|
|
s->var_values[VAR_D] = pkt->duration;
|
|
|
|
s->var_values[VAR_SIZE] = pkt->size;
|
|
|
|
s->var_values[VAR_KEY] = !!(pkt->flags & AV_PKT_FLAG_KEY);
|
|
|
|
s->var_values[VAR_POS] = pkt->pos;
|
|
|
|
|
|
|
|
if (s->var_values[VAR_STARTPTS] == AV_NOPTS_VALUE)
|
|
|
|
s->var_values[VAR_STARTPTS] = pkt->pts;
|
|
|
|
|
|
|
|
if (s->var_values[VAR_STARTDTS] == AV_NOPTS_VALUE)
|
|
|
|
s->var_values[VAR_STARTDTS] = pkt->dts;
|
|
|
|
|
|
|
|
res = av_expr_eval(s->amount_pexpr, s->var_values, NULL);
|
|
|
|
|
|
|
|
if (isnan(res))
|
|
|
|
amount = 0;
|
|
|
|
else if (res < 0)
|
|
|
|
amount = (s->state % 10001 + 1);
|
|
|
|
else
|
|
|
|
amount = (int)res;
|
|
|
|
|
|
|
|
if (s->drop_str) {
|
|
|
|
res = av_expr_eval(s->drop_pexpr, s->var_values, NULL);
|
|
|
|
|
|
|
|
if (isnan(res))
|
|
|
|
drop = 0;
|
|
|
|
else if (res < 0)
|
|
|
|
drop = !(s->state % FFABS((int)res));
|
|
|
|
else
|
|
|
|
drop = !!res;
|
|
|
|
}
|
|
|
|
|
|
|
|
if(s->dropamount) {
|
|
|
|
drop = !(s->state % s->dropamount);
|
|
|
|
}
|
|
|
|
|
|
|
|
av_log(ctx, AV_LOG_VERBOSE, "Stream #%d packet %d pts %"PRId64" - amount %d drop %d\n",
|
|
|
|
pkt->stream_index, (unsigned int)s->var_values[VAR_N], pkt->pts, amount, drop);
|
|
|
|
|
|
|
|
if (drop) {
|
|
|
|
s->var_values[VAR_STATE] = ++s->state;
|
|
|
|
av_packet_unref(pkt);
|
|
|
|
return AVERROR(EAGAIN);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (amount) {
|
|
|
|
ret = av_packet_make_writable(pkt);
|
|
|
|
if (ret < 0) {
|
|
|
|
av_packet_unref(pkt);
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
for (i = 0; i < pkt->size; i++) {
|
|
|
|
s->state += pkt->data[i] + 1;
|
|
|
|
if (amount && s->state % amount == 0)
|
|
|
|
pkt->data[i] = s->state;
|
|
|
|
}
|
|
|
|
|
|
|
|
s->var_values[VAR_STATE] = s->state;
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void noise_close(AVBSFContext *bsf)
|
|
|
|
{
|
|
|
|
NoiseContext *s = bsf->priv_data;
|
|
|
|
|
|
|
|
av_expr_free(s->amount_pexpr);
|
|
|
|
av_expr_free(s->drop_pexpr);
|
|
|
|
s->amount_pexpr = s->drop_pexpr = NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
#define OFFSET(x) offsetof(NoiseContext, x)
|
|
|
|
#define FLAGS (AV_OPT_FLAG_VIDEO_PARAM|AV_OPT_FLAG_AUDIO_PARAM|AV_OPT_FLAG_BSF_PARAM)
|
|
|
|
static const AVOption options[] = {
|
|
|
|
{ "amount", NULL, OFFSET(amount_str), AV_OPT_TYPE_STRING, { .str = NULL }, 0, 0, FLAGS },
|
|
|
|
{ "drop", NULL, OFFSET(drop_str), AV_OPT_TYPE_STRING, { .str = NULL }, 0, 0, FLAGS },
|
|
|
|
{ "dropamount", NULL, OFFSET(dropamount), AV_OPT_TYPE_INT, { .i64 = 0 }, 0, INT_MAX, FLAGS },
|
|
|
|
{ NULL },
|
|
|
|
};
|
|
|
|
|
|
|
|
static const AVClass noise_class = {
|
|
|
|
.class_name = "noise",
|
|
|
|
.option = options,
|
|
|
|
.version = LIBAVUTIL_VERSION_INT,
|
|
|
|
};
|
|
|
|
|
|
|
|
const FFBitStreamFilter ff_noise_bsf = {
|
|
|
|
.p.name = "noise",
|
|
|
|
.p.priv_class = &noise_class,
|
|
|
|
.priv_data_size = sizeof(NoiseContext),
|
|
|
|
.init = noise_init,
|
|
|
|
.close = noise_close,
|
|
|
|
.filter = noise,
|
|
|
|
};
|