mirror of
https://github.com/FFmpeg/FFmpeg.git
synced 2024-11-26 19:01:44 +02:00
avfilter/overlay_vaapi: enable expressions for overlay parameters
Signed-off-by: softworkz <softworkz@hotmail.com>
This commit is contained in:
parent
ff075f5abb
commit
c8050d82a9
@ -27,19 +27,106 @@
|
|||||||
#include "formats.h"
|
#include "formats.h"
|
||||||
#include "internal.h"
|
#include "internal.h"
|
||||||
#include "vaapi_vpp.h"
|
#include "vaapi_vpp.h"
|
||||||
|
#include "libavutil/eval.h"
|
||||||
|
|
||||||
|
enum var_name {
|
||||||
|
VAR_MAIN_IW, VAR_MW,
|
||||||
|
VAR_MAIN_IH, VAR_MH,
|
||||||
|
VAR_OVERLAY_IW,
|
||||||
|
VAR_OVERLAY_IH,
|
||||||
|
VAR_OVERLAY_X, VAR_OX,
|
||||||
|
VAR_OVERLAY_Y, VAR_OY,
|
||||||
|
VAR_OVERLAY_W, VAR_OW,
|
||||||
|
VAR_OVERLAY_H, VAR_OH,
|
||||||
|
VAR_VARS_NB
|
||||||
|
};
|
||||||
|
|
||||||
typedef struct OverlayVAAPIContext {
|
typedef struct OverlayVAAPIContext {
|
||||||
VAAPIVPPContext vpp_ctx; /**< must be the first field */
|
VAAPIVPPContext vpp_ctx; /**< must be the first field */
|
||||||
FFFrameSync fs;
|
FFFrameSync fs;
|
||||||
int overlay_ox;
|
|
||||||
int overlay_oy;
|
double var_values[VAR_VARS_NB];
|
||||||
int overlay_ow;
|
char *overlay_ox;
|
||||||
int overlay_oh;
|
char *overlay_oy;
|
||||||
|
char *overlay_ow;
|
||||||
|
char *overlay_oh;
|
||||||
|
int ox;
|
||||||
|
int oy;
|
||||||
|
int ow;
|
||||||
|
int oh;
|
||||||
float alpha;
|
float alpha;
|
||||||
unsigned int blend_flags;
|
unsigned int blend_flags;
|
||||||
float blend_alpha;
|
float blend_alpha;
|
||||||
} OverlayVAAPIContext;
|
} OverlayVAAPIContext;
|
||||||
|
|
||||||
|
static const char *const var_names[] = {
|
||||||
|
"main_w", "W", /* input width of the main layer */
|
||||||
|
"main_h", "H", /* input height of the main layer */
|
||||||
|
"overlay_iw", /* input width of the overlay layer */
|
||||||
|
"overlay_ih", /* input height of the overlay layer */
|
||||||
|
"overlay_x", "x", /* x position of the overlay layer inside of main */
|
||||||
|
"overlay_y", "y", /* y position of the overlay layer inside of main */
|
||||||
|
"overlay_w", "w", /* output width of overlay layer */
|
||||||
|
"overlay_h", "h", /* output height of overlay layer */
|
||||||
|
NULL
|
||||||
|
};
|
||||||
|
|
||||||
|
static int eval_expr(AVFilterContext *avctx)
|
||||||
|
{
|
||||||
|
OverlayVAAPIContext *ctx = avctx->priv;
|
||||||
|
double *var_values = ctx->var_values;
|
||||||
|
int ret = 0;
|
||||||
|
AVExpr *ox_expr = NULL, *oy_expr = NULL;
|
||||||
|
AVExpr *ow_expr = NULL, *oh_expr = NULL;
|
||||||
|
|
||||||
|
#define PARSE_EXPR(e, s) {\
|
||||||
|
ret = av_expr_parse(&(e), s, var_names, NULL, NULL, NULL, NULL, 0, ctx); \
|
||||||
|
if (ret < 0) {\
|
||||||
|
av_log(ctx, AV_LOG_ERROR, "Error when parsing '%s'.\n", s);\
|
||||||
|
goto release;\
|
||||||
|
}\
|
||||||
|
}
|
||||||
|
PARSE_EXPR(ox_expr, ctx->overlay_ox)
|
||||||
|
PARSE_EXPR(oy_expr, ctx->overlay_oy)
|
||||||
|
PARSE_EXPR(ow_expr, ctx->overlay_ow)
|
||||||
|
PARSE_EXPR(oh_expr, ctx->overlay_oh)
|
||||||
|
#undef PASS_EXPR
|
||||||
|
|
||||||
|
var_values[VAR_OVERLAY_W] =
|
||||||
|
var_values[VAR_OW] = av_expr_eval(ow_expr, var_values, NULL);
|
||||||
|
var_values[VAR_OVERLAY_H] =
|
||||||
|
var_values[VAR_OH] = av_expr_eval(oh_expr, var_values, NULL);
|
||||||
|
|
||||||
|
/* calc again in case ow is relative to oh */
|
||||||
|
var_values[VAR_OVERLAY_W] =
|
||||||
|
var_values[VAR_OW] = av_expr_eval(ow_expr, var_values, NULL);
|
||||||
|
|
||||||
|
var_values[VAR_OVERLAY_X] =
|
||||||
|
var_values[VAR_OX] = av_expr_eval(ox_expr, var_values, NULL);
|
||||||
|
var_values[VAR_OVERLAY_Y] =
|
||||||
|
var_values[VAR_OY] = av_expr_eval(oy_expr, var_values, NULL);
|
||||||
|
|
||||||
|
/* calc again in case ox is relative to oy */
|
||||||
|
var_values[VAR_OVERLAY_X] =
|
||||||
|
var_values[VAR_OX] = av_expr_eval(ox_expr, var_values, NULL);
|
||||||
|
|
||||||
|
/* calc overlay_w and overlay_h again incase relative to ox,oy */
|
||||||
|
var_values[VAR_OVERLAY_W] =
|
||||||
|
var_values[VAR_OW] = av_expr_eval(ow_expr, var_values, NULL);
|
||||||
|
var_values[VAR_OVERLAY_H] =
|
||||||
|
var_values[VAR_OH] = av_expr_eval(oh_expr, var_values, NULL);
|
||||||
|
var_values[VAR_OVERLAY_W] =
|
||||||
|
var_values[VAR_OW] = av_expr_eval(ow_expr, var_values, NULL);
|
||||||
|
|
||||||
|
release:
|
||||||
|
av_expr_free(ox_expr);
|
||||||
|
av_expr_free(oy_expr);
|
||||||
|
av_expr_free(ow_expr);
|
||||||
|
av_expr_free(oh_expr);
|
||||||
|
|
||||||
|
return ret;
|
||||||
|
}
|
||||||
|
|
||||||
static int overlay_vaapi_build_filter_params(AVFilterContext *avctx)
|
static int overlay_vaapi_build_filter_params(AVFilterContext *avctx)
|
||||||
{
|
{
|
||||||
VAAPIVPPContext *vpp_ctx = avctx->priv;
|
VAAPIVPPContext *vpp_ctx = avctx->priv;
|
||||||
@ -233,10 +320,10 @@ static int overlay_vaapi_blend(FFFrameSync *fs)
|
|||||||
input_overlay->width, input_overlay->height, input_overlay->pts);
|
input_overlay->width, input_overlay->height, input_overlay->pts);
|
||||||
|
|
||||||
overlay_region = (VARectangle) {
|
overlay_region = (VARectangle) {
|
||||||
.x = ctx->overlay_ox,
|
.x = ctx->ox,
|
||||||
.y = ctx->overlay_oy,
|
.y = ctx->oy,
|
||||||
.width = ctx->overlay_ow ? ctx->overlay_ow : input_overlay->width,
|
.width = ctx->ow ? ctx->ow : input_overlay->width,
|
||||||
.height = ctx->overlay_oh ? ctx->overlay_oh : input_overlay->height,
|
.height = ctx->oh ? ctx->oh : input_overlay->height,
|
||||||
};
|
};
|
||||||
|
|
||||||
if (overlay_region.x + overlay_region.width > input_main->width ||
|
if (overlay_region.x + overlay_region.width > input_main->width ||
|
||||||
@ -289,10 +376,36 @@ static int have_alpha_planar(AVFilterLink *link)
|
|||||||
return !!(desc->flags & AV_PIX_FMT_FLAG_ALPHA);
|
return !!(desc->flags & AV_PIX_FMT_FLAG_ALPHA);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
static int overlay_vaapi_config_input_main(AVFilterLink *inlink)
|
||||||
|
{
|
||||||
|
AVFilterContext *avctx = inlink->dst;
|
||||||
|
OverlayVAAPIContext *ctx = avctx->priv;
|
||||||
|
|
||||||
|
ctx->var_values[VAR_MAIN_IW] =
|
||||||
|
ctx->var_values[VAR_MW] = inlink->w;
|
||||||
|
ctx->var_values[VAR_MAIN_IH] =
|
||||||
|
ctx->var_values[VAR_MH] = inlink->h;
|
||||||
|
|
||||||
|
return ff_vaapi_vpp_config_input(inlink);
|
||||||
|
}
|
||||||
|
|
||||||
static int overlay_vaapi_config_input_overlay(AVFilterLink *inlink)
|
static int overlay_vaapi_config_input_overlay(AVFilterLink *inlink)
|
||||||
{
|
{
|
||||||
AVFilterContext *avctx = inlink->dst;
|
AVFilterContext *avctx = inlink->dst;
|
||||||
OverlayVAAPIContext *ctx = avctx->priv;
|
OverlayVAAPIContext *ctx = avctx->priv;
|
||||||
|
int ret;
|
||||||
|
|
||||||
|
ctx->var_values[VAR_OVERLAY_IW] = inlink->w;
|
||||||
|
ctx->var_values[VAR_OVERLAY_IH] = inlink->h;
|
||||||
|
|
||||||
|
ret = eval_expr(avctx);
|
||||||
|
if (ret < 0)
|
||||||
|
return ret;
|
||||||
|
|
||||||
|
ctx->ox = (int)ctx->var_values[VAR_OX];
|
||||||
|
ctx->oy = (int)ctx->var_values[VAR_OY];
|
||||||
|
ctx->ow = (int)ctx->var_values[VAR_OW];
|
||||||
|
ctx->oh = (int)ctx->var_values[VAR_OH];
|
||||||
|
|
||||||
ctx->blend_flags = 0;
|
ctx->blend_flags = 0;
|
||||||
ctx->blend_alpha = 1.0f;
|
ctx->blend_alpha = 1.0f;
|
||||||
@ -365,11 +478,11 @@ static av_cold void overlay_vaapi_uninit(AVFilterContext *avctx)
|
|||||||
#define OFFSET(x) offsetof(OverlayVAAPIContext, x)
|
#define OFFSET(x) offsetof(OverlayVAAPIContext, x)
|
||||||
#define FLAGS (AV_OPT_FLAG_FILTERING_PARAM | AV_OPT_FLAG_VIDEO_PARAM)
|
#define FLAGS (AV_OPT_FLAG_FILTERING_PARAM | AV_OPT_FLAG_VIDEO_PARAM)
|
||||||
static const AVOption overlay_vaapi_options[] = {
|
static const AVOption overlay_vaapi_options[] = {
|
||||||
{ "x", "Overlay x position", OFFSET(overlay_ox), AV_OPT_TYPE_INT, { .i64 = 0 }, 0, INT_MAX, .flags = FLAGS },
|
{ "x", "Overlay x position", OFFSET(overlay_ox), AV_OPT_TYPE_STRING, { .str="0"}, 0, 255, .flags = FLAGS},
|
||||||
{ "y", "Overlay y position", OFFSET(overlay_oy), AV_OPT_TYPE_INT, { .i64 = 0 }, 0, INT_MAX, .flags = FLAGS },
|
{ "y", "Overlay y position", OFFSET(overlay_oy), AV_OPT_TYPE_STRING, { .str="0"}, 0, 255, .flags = FLAGS},
|
||||||
{ "w", "Overlay width", OFFSET(overlay_ow), AV_OPT_TYPE_INT, { .i64 = 0 }, 0, INT_MAX, .flags = FLAGS },
|
{ "w", "Overlay width", OFFSET(overlay_ow), AV_OPT_TYPE_STRING, { .str="overlay_iw"}, 0, 255, .flags = FLAGS},
|
||||||
{ "h", "Overlay height", OFFSET(overlay_oh), AV_OPT_TYPE_INT, { .i64 = 0 }, 0, INT_MAX, .flags = FLAGS },
|
{ "h", "Overlay height", OFFSET(overlay_oh), AV_OPT_TYPE_STRING, { .str="overlay_ih*w/overlay_iw"}, 0, 255, .flags = FLAGS},
|
||||||
{ "alpha", "Overlay global alpha", OFFSET(alpha), AV_OPT_TYPE_FLOAT, { .dbl = 1.0 }, 0.0, 1.0, .flags = FLAGS },
|
{ "alpha", "Overlay global alpha", OFFSET(alpha), AV_OPT_TYPE_FLOAT, { .dbl = 1.0 }, 0.0, 1.0, .flags = FLAGS },
|
||||||
{ "eof_action", "Action to take when encountering EOF from secondary input ",
|
{ "eof_action", "Action to take when encountering EOF from secondary input ",
|
||||||
OFFSET(fs.opt_eof_action), AV_OPT_TYPE_INT, { .i64 = EOF_ACTION_REPEAT },
|
OFFSET(fs.opt_eof_action), AV_OPT_TYPE_INT, { .i64 = EOF_ACTION_REPEAT },
|
||||||
EOF_ACTION_REPEAT, EOF_ACTION_PASS, .flags = FLAGS, "eof_action" },
|
EOF_ACTION_REPEAT, EOF_ACTION_PASS, .flags = FLAGS, "eof_action" },
|
||||||
@ -387,7 +500,7 @@ static const AVFilterPad overlay_vaapi_inputs[] = {
|
|||||||
{
|
{
|
||||||
.name = "main",
|
.name = "main",
|
||||||
.type = AVMEDIA_TYPE_VIDEO,
|
.type = AVMEDIA_TYPE_VIDEO,
|
||||||
.config_props = &ff_vaapi_vpp_config_input,
|
.config_props = overlay_vaapi_config_input_main,
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
.name = "overlay",
|
.name = "overlay",
|
||||||
|
Loading…
Reference in New Issue
Block a user