summaryrefslogtreecommitdiff
diff options
context:
space:
mode:
authorNiklas Haas <git@haasn.dev>2023-05-01 16:41:51 +0200
committerNiklas Haas <git@haasn.dev>2023-05-03 23:40:20 +0200
commit4b11a07550363e1f1d6f8e43923d9b6b327737b0 (patch)
tree29a734a1b93e07f7215300392ef5e288d8a09c3a
parentd94c6df975e277b6f64db865074a4b64cbf0ae7a (diff)
avfilter/vf_libplacebo: add fillcolor option
In some circumstances, libplacebo will clear the background as a result of cropping/padding. Currently, this uses the hard-coded default fill color of black. This option makes this behavior configurable.
-rw-r--r--doc/filters.texi6
-rw-r--r--libavfilter/vf_libplacebo.c22
2 files changed, 28 insertions, 0 deletions
diff --git a/doc/filters.texi b/doc/filters.texi
index a63304ccd9..34212b513d 100644
--- a/doc/filters.texi
+++ b/doc/filters.texi
@@ -16024,6 +16024,12 @@ content with black borders, while a value of @code{1.0} always crops off parts
of the content. Intermediate values are possible, leading to a mix of the two
approaches.
+@item fillcolor
+Set the color used to fill the output area not covered by the output image, for
+example as a result of @ref{normalize_sar}. For the general syntax of this
+option, check the @ref{color syntax,,"Color" section in the ffmpeg-utils
+manual,ffmpeg-utils}. Defaults to @code{black}.
+
@item colorspace
@item color_primaries
@item color_trc
diff --git a/libavfilter/vf_libplacebo.c b/libavfilter/vf_libplacebo.c
index 66929223dd..fcdc97e48e 100644
--- a/libavfilter/vf_libplacebo.c
+++ b/libavfilter/vf_libplacebo.c
@@ -18,6 +18,7 @@
#include "libavutil/file.h"
#include "libavutil/opt.h"
+#include "libavutil/parseutils.h"
#include "internal.h"
#include "vulkan_filter.h"
#include "scale_eval.h"
@@ -73,6 +74,7 @@ typedef struct LibplaceboContext {
/* settings */
char *out_format_string;
enum AVPixelFormat out_format;
+ char *fillcolor;
char *w_expr;
char *h_expr;
AVRational target_sar;
@@ -225,6 +227,24 @@ static int find_scaler(AVFilterContext *avctx,
return AVERROR(EINVAL);
}
+static int parse_fillcolor(AVFilterContext *avctx,
+ struct pl_render_params *params,
+ const char *color_str)
+{
+ int err = 0;
+ uint8_t color_rgba[4];
+
+ RET(av_parse_color(color_rgba, color_str, -1, avctx));
+ params->background_color[0] = (float) color_rgba[0] / UINT8_MAX;
+ params->background_color[1] = (float) color_rgba[1] / UINT8_MAX;
+ params->background_color[2] = (float) color_rgba[2] / UINT8_MAX;
+ params->background_transparency = 1.0f - (float) color_rgba[3] / UINT8_MAX;
+ return 0;
+
+fail:
+ return err;
+}
+
static void libplacebo_uninit(AVFilterContext *avctx);
static int libplacebo_init(AVFilterContext *avctx)
@@ -469,6 +489,7 @@ static int process_frames(AVFilterContext *avctx, AVFrame *out, AVFrame *in)
RET(find_scaler(avctx, &params.upscaler, s->upscaler));
RET(find_scaler(avctx, &params.downscaler, s->downscaler));
+ RET(parse_fillcolor(avctx, &params, s->fillcolor));
pl_render_image(s->renderer, &image, &target, &params);
pl_unmap_avframe(s->gpu, &image);
@@ -703,6 +724,7 @@ static const AVOption libplacebo_options[] = {
{ "force_divisible_by", "enforce that the output resolution is divisible by a defined integer when force_original_aspect_ratio is used", OFFSET(force_divisible_by), AV_OPT_TYPE_INT, { .i64 = 1 }, 1, 256, STATIC },
{ "normalize_sar", "force SAR normalization to 1:1", OFFSET(normalize_sar), AV_OPT_TYPE_BOOL, {.i64 = 0}, 0, 1, STATIC },
{ "pad_crop_ratio", "ratio between padding and cropping when normalizing SAR (0=pad, 1=crop)", OFFSET(pad_crop_ratio), AV_OPT_TYPE_FLOAT, {.dbl=0.0}, 0.0, 1.0, DYNAMIC },
+ { "fillcolor", "Background fill color", OFFSET(fillcolor), AV_OPT_TYPE_STRING, {.str = "black"}, .flags = DYNAMIC },
{"colorspace", "select colorspace", OFFSET(colorspace), AV_OPT_TYPE_INT, {.i64=-1}, -1, AVCOL_SPC_NB-1, DYNAMIC, "colorspace"},
{"auto", "keep the same colorspace", 0, AV_OPT_TYPE_CONST, {.i64=-1}, INT_MIN, INT_MAX, STATIC, "colorspace"},