videoconvertscale: Expose converter config as new property

This allows the user to have full control on the conversion parameters. If set,
the property takes precedence over the other similar conversion tweaking properties.

Part-of: <https://gitlab.freedesktop.org/gstreamer/gstreamer/-/merge_requests/2263>
This commit is contained in:
Philippe Normand 2022-03-23 13:38:00 +00:00
parent 886bd7e4e0
commit 9dbe8a1e36
2 changed files with 88 additions and 1 deletions

View file

@ -14548,6 +14548,17 @@
"type": "GstVideoResamplerMethod", "type": "GstVideoResamplerMethod",
"writable": true "writable": true
}, },
"converter-config": {
"blurb": "A GstStructure describing the configuration that should be used. This configuration, if set, takes precedence over the other similar conversion properties.",
"conditionally-available": false,
"construct": false,
"construct-only": false,
"controllable": false,
"mutable": "null",
"readable": true,
"type": "GstStructure",
"writable": true
},
"dither": { "dither": {
"blurb": "Apply dithering while converting", "blurb": "Apply dithering while converting",
"conditionally-available": false, "conditionally-available": false,
@ -14802,6 +14813,17 @@
"type": "GstVideoResamplerMethod", "type": "GstVideoResamplerMethod",
"writable": true "writable": true
}, },
"converter-config": {
"blurb": "A GstStructure describing the configuration that should be used. This configuration, if set, takes precedence over the other similar conversion properties.",
"conditionally-available": false,
"construct": false,
"construct-only": false,
"controllable": false,
"mutable": "null",
"readable": true,
"type": "GstStructure",
"writable": true
},
"dither": { "dither": {
"blurb": "Apply dithering while converting", "blurb": "Apply dithering while converting",
"conditionally-available": false, "conditionally-available": false,

View file

@ -105,6 +105,9 @@ typedef struct
GstVideoConverter *convert; GstVideoConverter *convert;
GstStructure *converter_config;
gboolean converter_config_changed;
gint borders_h; gint borders_h;
gint borders_w; gint borders_w;
} GstVideoConvertScalePrivate; } GstVideoConvertScalePrivate;
@ -158,6 +161,7 @@ enum
PROP_MATRIX_MODE, PROP_MATRIX_MODE,
PROP_GAMMA_MODE, PROP_GAMMA_MODE,
PROP_PRIMARIES_MODE, PROP_PRIMARIES_MODE,
PROP_CONVERTER_CONFIG,
}; };
#undef GST_VIDEO_SIZE_RANGE #undef GST_VIDEO_SIZE_RANGE
@ -375,6 +379,23 @@ gst_video_convert_scale_class_init (GstVideoConvertScaleClass * klass)
DEFAULT_PROP_PRIMARIES_MODE, DEFAULT_PROP_PRIMARIES_MODE,
G_PARAM_READWRITE | G_PARAM_STATIC_STRINGS)); G_PARAM_READWRITE | G_PARAM_STATIC_STRINGS));
/**
* GstVideoConvertScale:converter-config:
*
* A #GstStructure describing the configuration that should be used. This
* configuration, if set, takes precedence over the other similar conversion
* properties.
*
* Since: 1.24
*/
g_object_class_install_property (gobject_class,
PROP_CONVERTER_CONFIG, g_param_spec_boxed ("converter-config",
"Converter configuration",
"A GstStructure describing the configuration that should be used."
" This configuration, if set, takes precedence over the other similar conversion properties.",
GST_TYPE_STRUCTURE, G_PARAM_READWRITE | G_PARAM_STATIC_STRINGS));
gst_element_class_set_static_metadata (element_class, gst_element_class_set_static_metadata (element_class,
"Video colorspace converter and scaler", "Video colorspace converter and scaler",
"Filter/Converter/Video/Scaler/Colorspace", "Filter/Converter/Video/Scaler/Colorspace",
@ -431,6 +452,9 @@ gst_video_convert_scale_init (GstVideoConvertScale * self)
priv->matrix_mode = DEFAULT_PROP_MATRIX_MODE; priv->matrix_mode = DEFAULT_PROP_MATRIX_MODE;
priv->gamma_mode = DEFAULT_PROP_GAMMA_MODE; priv->gamma_mode = DEFAULT_PROP_GAMMA_MODE;
priv->primaries_mode = DEFAULT_PROP_PRIMARIES_MODE; priv->primaries_mode = DEFAULT_PROP_PRIMARIES_MODE;
priv->converter_config = NULL;
priv->converter_config_changed = FALSE;
} }
static void static void
@ -441,6 +465,10 @@ gst_video_convert_scale_finalize (GstVideoConvertScale * self)
if (priv->convert) if (priv->convert)
gst_video_converter_free (priv->convert); gst_video_converter_free (priv->convert);
if (priv->converter_config)
gst_structure_free (priv->converter_config);
priv->converter_config = NULL;
G_OBJECT_CLASS (parent_class)->finalize (G_OBJECT (self)); G_OBJECT_CLASS (parent_class)->finalize (G_OBJECT (self));
} }
@ -503,6 +531,12 @@ gst_video_convert_scale_set_property (GObject * object, guint prop_id,
case PROP_DITHER_QUANTIZATION: case PROP_DITHER_QUANTIZATION:
priv->dither_quantization = g_value_get_uint (value); priv->dither_quantization = g_value_get_uint (value);
break; break;
case PROP_CONVERTER_CONFIG:
if (priv->converter_config)
gst_structure_free (priv->converter_config);
priv->converter_config = g_value_dup_boxed (value);
priv->converter_config_changed = TRUE;
break;
default: default:
G_OBJECT_WARN_INVALID_PROPERTY_ID (object, prop_id, pspec); G_OBJECT_WARN_INVALID_PROPERTY_ID (object, prop_id, pspec);
break; break;
@ -566,6 +600,9 @@ gst_video_convert_scale_get_property (GObject * object, guint prop_id,
case PROP_DITHER_QUANTIZATION: case PROP_DITHER_QUANTIZATION:
g_value_set_uint (value, priv->dither_quantization); g_value_set_uint (value, priv->dither_quantization);
break; break;
case PROP_CONVERTER_CONFIG:
g_value_set_boxed (value, priv->converter_config);
break;
default: default:
G_OBJECT_WARN_INVALID_PROPERTY_ID (object, prop_id, pspec); G_OBJECT_WARN_INVALID_PROPERTY_ID (object, prop_id, pspec);
break; break;
@ -709,6 +746,14 @@ gst_video_convert_scale_transform_meta (GstBaseTransform * trans,
return TRUE; return TRUE;
} }
static GstStructure *
gst_video_convert_scale_get_converter_config (GstVideoConvertScale * self,
GstVideoInfo * out_info)
{
GstVideoConvertScalePrivate *priv = PRIV (self);
return gst_structure_copy (priv->converter_config);
}
static gboolean static gboolean
gst_video_convert_scale_set_info (GstVideoFilter * filter, GstCaps * in, gst_video_convert_scale_set_info (GstVideoFilter * filter, GstCaps * in,
GstVideoInfo * in_info, GstCaps * out, GstVideoInfo * out_info) GstVideoInfo * in_info, GstCaps * out, GstVideoInfo * out_info)
@ -717,6 +762,7 @@ gst_video_convert_scale_set_info (GstVideoFilter * filter, GstCaps * in,
GstVideoConvertScalePrivate *priv = PRIV (self); GstVideoConvertScalePrivate *priv = PRIV (self);
gint from_dar_n, from_dar_d, to_dar_n, to_dar_d; gint from_dar_n, from_dar_d, to_dar_n, to_dar_d;
GstVideoInfo tmp_info; GstVideoInfo tmp_info;
GstStructure *options;
if (priv->convert) { if (priv->convert) {
gst_video_converter_free (priv->convert); gst_video_converter_free (priv->convert);
@ -765,6 +811,13 @@ gst_video_convert_scale_set_info (GstVideoFilter * filter, GstCaps * in,
if (in_info->interlace_mode != out_info->interlace_mode) if (in_info->interlace_mode != out_info->interlace_mode)
goto format_mismatch; goto format_mismatch;
if (priv->converter_config) {
options = gst_video_convert_scale_get_converter_config (self, out_info);
GST_DEBUG_OBJECT (self,
"Using user-provided converter-config: %" GST_PTR_FORMAT, options);
goto build_converter;
}
/* if the only thing different in the caps is the transfer function, and /* if the only thing different in the caps is the transfer function, and
* we're converting between equivalent transfer functions and not * we're converting between equivalent transfer functions and not
* quantizing/dithering or adjusting alpha, then do passthrough */ * quantizing/dithering or adjusting alpha, then do passthrough */
@ -784,7 +837,6 @@ gst_video_convert_scale_set_info (GstVideoFilter * filter, GstCaps * in,
out_info->finfo->bits) && !need_dither && !need_alpha) { out_info->finfo->bits) && !need_dither && !need_alpha) {
gst_base_transform_set_passthrough (GST_BASE_TRANSFORM (filter), TRUE); gst_base_transform_set_passthrough (GST_BASE_TRANSFORM (filter), TRUE);
} else { } else {
GstStructure *options;
GST_CAT_DEBUG_OBJECT (CAT_PERFORMANCE, filter, "setup videoscaling"); GST_CAT_DEBUG_OBJECT (CAT_PERFORMANCE, filter, "setup videoscaling");
gst_base_transform_set_passthrough (GST_BASE_TRANSFORM (filter), FALSE); gst_base_transform_set_passthrough (GST_BASE_TRANSFORM (filter), FALSE);
@ -885,6 +937,7 @@ gst_video_convert_scale_set_info (GstVideoFilter * filter, GstCaps * in,
priv->primaries_mode, GST_VIDEO_CONVERTER_OPT_THREADS, G_TYPE_UINT, priv->primaries_mode, GST_VIDEO_CONVERTER_OPT_THREADS, G_TYPE_UINT,
priv->n_threads, NULL); priv->n_threads, NULL);
build_converter:
priv->convert = gst_video_converter_new (in_info, out_info, options); priv->convert = gst_video_converter_new (in_info, out_info, options);
if (priv->convert == NULL) if (priv->convert == NULL)
goto no_convert; goto no_convert;
@ -1748,6 +1801,18 @@ gst_video_convert_scale_transform_frame (GstVideoFilter * filter,
GST_CAT_DEBUG_OBJECT (CAT_PERFORMANCE, filter, "doing video scaling"); GST_CAT_DEBUG_OBJECT (CAT_PERFORMANCE, filter, "doing video scaling");
if (priv->converter_config_changed) {
GstStructure *options =
gst_video_convert_scale_get_converter_config (GST_VIDEO_CONVERT_SCALE
(filter), &filter->out_info);
gst_video_converter_free (priv->convert);
priv->convert =
gst_video_converter_new (&filter->in_info, &filter->out_info, options);
priv->converter_config_changed = FALSE;
}
gst_video_converter_frame (priv->convert, in_frame, out_frame); gst_video_converter_frame (priv->convert, in_frame, out_frame);
return ret; return ret;