Commit c044ac20 authored by Marton Balint's avatar Marton Balint

avfilter/vf_geq: use per-thread AVExpr for expression evaluation

There was no consensus about separating AVExprState from AVExpr so here is a
minimal patch using the existing AVExpr to fix ticket #7528.
Signed-off-by: 's avatarMarton Balint <cus@passwd.hu>
parent b82825eb
...@@ -11468,6 +11468,11 @@ Default is bilinear. ...@@ -11468,6 +11468,11 @@ Default is bilinear.
For functions, if @var{x} and @var{y} are outside the area, the value will be For functions, if @var{x} and @var{y} are outside the area, the value will be
automatically clipped to the closer edge. automatically clipped to the closer edge.
Please note that this filter can use multiple threads in which case each slice
will have its own expression state. If you want to use only a single expression
state because your expressions depend on previous state then you should limit
the number of filter threads to 1.
@subsection Examples @subsection Examples
@itemize @itemize
......
...@@ -33,6 +33,7 @@ ...@@ -33,6 +33,7 @@
#include "libavutil/pixdesc.h" #include "libavutil/pixdesc.h"
#include "internal.h" #include "internal.h"
#define MAX_NB_THREADS 32
#define NB_PLANES 4 #define NB_PLANES 4
enum InterpolationMethods { enum InterpolationMethods {
...@@ -46,7 +47,7 @@ enum { VAR_X, VAR_Y, VAR_W, VAR_H, VAR_N, VAR_ ...@@ -46,7 +47,7 @@ enum { VAR_X, VAR_Y, VAR_W, VAR_H, VAR_N, VAR_
typedef struct GEQContext { typedef struct GEQContext {
const AVClass *class; const AVClass *class;
AVExpr *e[NB_PLANES]; ///< expressions for each plane AVExpr *e[NB_PLANES][MAX_NB_THREADS]; ///< expressions for each plane and thread
char *expr_str[4+3]; ///< expression strings for each plane char *expr_str[4+3]; ///< expression strings for each plane
AVFrame *picref; ///< current input buffer AVFrame *picref; ///< current input buffer
uint8_t *dst; ///< reference pointer to the 8bits output uint8_t *dst; ///< reference pointer to the 8bits output
...@@ -288,12 +289,14 @@ static av_cold int geq_init(AVFilterContext *ctx) ...@@ -288,12 +289,14 @@ static av_cold int geq_init(AVFilterContext *ctx)
NULL }; NULL };
int counter[10] = {0}; int counter[10] = {0};
ret = av_expr_parse(&geq->e[plane], geq->expr_str[plane < 3 && geq->is_rgb ? plane+4 : plane], var_names, for (int i = 0; i < MAX_NB_THREADS; i++) {
NULL, NULL, func2_names, func2, 0, ctx); ret = av_expr_parse(&geq->e[plane][i], geq->expr_str[plane < 3 && geq->is_rgb ? plane+4 : plane], var_names,
if (ret < 0) NULL, NULL, func2_names, func2, 0, ctx);
break; if (ret < 0)
goto end;
}
av_expr_count_func(geq->e[plane], counter, FF_ARRAY_ELEMS(counter), 2); av_expr_count_func(geq->e[plane][0], counter, FF_ARRAY_ELEMS(counter), 2);
geq->needs_sum[plane] = counter[5] + counter[6] + counter[7] + counter[8] + counter[9]; geq->needs_sum[plane] = counter[5] + counter[6] + counter[7] + counter[8] + counter[9];
} }
...@@ -391,7 +394,7 @@ static int slice_geq_filter(AVFilterContext *ctx, void *arg, int jobnr, int nb_j ...@@ -391,7 +394,7 @@ static int slice_geq_filter(AVFilterContext *ctx, void *arg, int jobnr, int nb_j
for (x = 0; x < width; x++) { for (x = 0; x < width; x++) {
values[VAR_X] = x; values[VAR_X] = x;
ptr[x] = av_expr_eval(geq->e[plane], values, geq); ptr[x] = av_expr_eval(geq->e[plane][jobnr], values, geq);
} }
ptr += linesize; ptr += linesize;
} }
...@@ -401,7 +404,7 @@ static int slice_geq_filter(AVFilterContext *ctx, void *arg, int jobnr, int nb_j ...@@ -401,7 +404,7 @@ static int slice_geq_filter(AVFilterContext *ctx, void *arg, int jobnr, int nb_j
values[VAR_Y] = y; values[VAR_Y] = y;
for (x = 0; x < width; x++) { for (x = 0; x < width; x++) {
values[VAR_X] = x; values[VAR_X] = x;
ptr16[x] = av_expr_eval(geq->e[plane], values, geq); ptr16[x] = av_expr_eval(geq->e[plane][jobnr], values, geq);
} }
ptr16 += linesize/2; ptr16 += linesize/2;
} }
...@@ -414,7 +417,7 @@ static int geq_filter_frame(AVFilterLink *inlink, AVFrame *in) ...@@ -414,7 +417,7 @@ static int geq_filter_frame(AVFilterLink *inlink, AVFrame *in)
{ {
int plane; int plane;
AVFilterContext *ctx = inlink->dst; AVFilterContext *ctx = inlink->dst;
const int nb_threads = ff_filter_get_nb_threads(ctx); const int nb_threads = FFMIN(MAX_NB_THREADS, ff_filter_get_nb_threads(ctx));
GEQContext *geq = ctx->priv; GEQContext *geq = ctx->priv;
AVFilterLink *outlink = inlink->dst->outputs[0]; AVFilterLink *outlink = inlink->dst->outputs[0];
AVFrame *out; AVFrame *out;
...@@ -464,8 +467,9 @@ static av_cold void geq_uninit(AVFilterContext *ctx) ...@@ -464,8 +467,9 @@ static av_cold void geq_uninit(AVFilterContext *ctx)
int i; int i;
GEQContext *geq = ctx->priv; GEQContext *geq = ctx->priv;
for (i = 0; i < FF_ARRAY_ELEMS(geq->e); i++) for (i = 0; i < NB_PLANES; i++)
av_expr_free(geq->e[i]); for (int j = 0; j < MAX_NB_THREADS; j++)
av_expr_free(geq->e[i][j]);
for (i = 0; i < NB_PLANES; i++) for (i = 0; i < NB_PLANES; i++)
av_freep(&geq->pixel_sums); av_freep(&geq->pixel_sums);
} }
......
Markdown is supported
0% or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment