|
|
@ -23,55 +23,182 @@ |
|
|
|
#include "bsf.h" |
|
|
|
#include "bsf.h" |
|
|
|
#include "bsf_internal.h" |
|
|
|
#include "bsf_internal.h" |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
#include "libavutil/avstring.h" |
|
|
|
#include "libavutil/log.h" |
|
|
|
#include "libavutil/log.h" |
|
|
|
#include "libavutil/opt.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 { |
|
|
|
typedef struct NoiseContext { |
|
|
|
const AVClass *class; |
|
|
|
const AVClass *class; |
|
|
|
int amount; |
|
|
|
|
|
|
|
int dropamount; |
|
|
|
char *amount_str; |
|
|
|
|
|
|
|
char *drop_str; |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
AVExpr *amount_pexpr; |
|
|
|
|
|
|
|
AVExpr *drop_pexpr; |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
double var_values[VAR_VARS_NB]; |
|
|
|
|
|
|
|
|
|
|
|
unsigned int state; |
|
|
|
unsigned int state; |
|
|
|
|
|
|
|
unsigned int pkt_idx; |
|
|
|
} NoiseContext; |
|
|
|
} NoiseContext; |
|
|
|
|
|
|
|
|
|
|
|
static int noise(AVBSFContext *ctx, AVPacket *pkt) |
|
|
|
static int noise_init(AVBSFContext *ctx) |
|
|
|
{ |
|
|
|
{ |
|
|
|
NoiseContext *s = ctx->priv_data; |
|
|
|
NoiseContext *s = ctx->priv_data; |
|
|
|
int amount = s->amount > 0 ? s->amount : (s->state % 10001 + 1); |
|
|
|
int ret; |
|
|
|
int i, ret; |
|
|
|
|
|
|
|
|
|
|
|
if (!s->amount_str) { |
|
|
|
|
|
|
|
s->amount_str = !s->drop_str ? av_strdup("-1") : av_strdup("0"); |
|
|
|
|
|
|
|
if (!s->amount_str) |
|
|
|
|
|
|
|
return AVERROR(ENOMEM); |
|
|
|
|
|
|
|
} |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
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) { |
|
|
|
|
|
|
|
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; |
|
|
|
|
|
|
|
|
|
|
|
if (amount <= 0) |
|
|
|
return 0; |
|
|
|
return AVERROR(EINVAL); |
|
|
|
} |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
static int noise(AVBSFContext *ctx, AVPacket *pkt) |
|
|
|
|
|
|
|
{ |
|
|
|
|
|
|
|
NoiseContext *s = ctx->priv_data; |
|
|
|
|
|
|
|
int i, ret, amount, drop; |
|
|
|
|
|
|
|
double res; |
|
|
|
|
|
|
|
|
|
|
|
ret = ff_bsf_get_packet_ref(ctx, pkt); |
|
|
|
ret = ff_bsf_get_packet_ref(ctx, pkt); |
|
|
|
if (ret < 0) |
|
|
|
if (ret < 0) |
|
|
|
return ret; |
|
|
|
return ret; |
|
|
|
|
|
|
|
|
|
|
|
if (s->dropamount > 0 && s->state % s->dropamount == 0) { |
|
|
|
s->var_values[VAR_N] = s->pkt_idx++; |
|
|
|
s->state++; |
|
|
|
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; |
|
|
|
|
|
|
|
} |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
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 (s->drop_str && drop) { |
|
|
|
|
|
|
|
s->var_values[VAR_STATE] = ++s->state; |
|
|
|
av_packet_unref(pkt); |
|
|
|
av_packet_unref(pkt); |
|
|
|
return AVERROR(EAGAIN); |
|
|
|
return AVERROR(EAGAIN); |
|
|
|
} |
|
|
|
} |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
if (amount) { |
|
|
|
ret = av_packet_make_writable(pkt); |
|
|
|
ret = av_packet_make_writable(pkt); |
|
|
|
if (ret < 0) { |
|
|
|
if (ret < 0) { |
|
|
|
av_packet_unref(pkt); |
|
|
|
av_packet_unref(pkt); |
|
|
|
return ret; |
|
|
|
return ret; |
|
|
|
} |
|
|
|
} |
|
|
|
|
|
|
|
} |
|
|
|
|
|
|
|
|
|
|
|
for (i = 0; i < pkt->size; i++) { |
|
|
|
for (i = 0; i < pkt->size; i++) { |
|
|
|
s->state += pkt->data[i] + 1; |
|
|
|
s->state += pkt->data[i] + 1; |
|
|
|
if (s->state % amount == 0) |
|
|
|
if (amount && s->state % amount == 0) |
|
|
|
pkt->data[i] = s->state; |
|
|
|
pkt->data[i] = s->state; |
|
|
|
} |
|
|
|
} |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
s->var_values[VAR_STATE] = s->state; |
|
|
|
|
|
|
|
|
|
|
|
return 0; |
|
|
|
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 OFFSET(x) offsetof(NoiseContext, x) |
|
|
|
#define FLAGS (AV_OPT_FLAG_VIDEO_PARAM|AV_OPT_FLAG_AUDIO_PARAM|AV_OPT_FLAG_BSF_PARAM) |
|
|
|
#define FLAGS (AV_OPT_FLAG_VIDEO_PARAM|AV_OPT_FLAG_AUDIO_PARAM|AV_OPT_FLAG_BSF_PARAM) |
|
|
|
static const AVOption options[] = { |
|
|
|
static const AVOption options[] = { |
|
|
|
{ "amount", NULL, OFFSET(amount), AV_OPT_TYPE_INT, { .i64 = 0 }, 0, INT_MAX, FLAGS }, |
|
|
|
{ "amount", NULL, OFFSET(amount_str), AV_OPT_TYPE_STRING, { .str = NULL }, 0, 0, FLAGS }, |
|
|
|
{ "dropamount", NULL, OFFSET(dropamount), AV_OPT_TYPE_INT, { .i64 = 0 }, 0, INT_MAX, FLAGS }, |
|
|
|
{ "drop", NULL, OFFSET(drop_str), AV_OPT_TYPE_STRING, { .str = NULL }, 0, 0, FLAGS }, |
|
|
|
|
|
|
|
{ "dropamount", NULL, OFFSET(drop_str), AV_OPT_TYPE_STRING, { .str = NULL }, 0, 0, FLAGS }, |
|
|
|
{ NULL }, |
|
|
|
{ NULL }, |
|
|
|
}; |
|
|
|
}; |
|
|
|
|
|
|
|
|
|
|
@ -86,5 +213,7 @@ const AVBitStreamFilter ff_noise_bsf = { |
|
|
|
.name = "noise", |
|
|
|
.name = "noise", |
|
|
|
.priv_data_size = sizeof(NoiseContext), |
|
|
|
.priv_data_size = sizeof(NoiseContext), |
|
|
|
.priv_class = &noise_class, |
|
|
|
.priv_class = &noise_class, |
|
|
|
|
|
|
|
.init = noise_init, |
|
|
|
|
|
|
|
.close = noise_close, |
|
|
|
.filter = noise, |
|
|
|
.filter = noise, |
|
|
|
}; |
|
|
|
}; |
|
|
|