--- /dev/null
+/* GStreamer
+ * Copyright (C) 2022 Intel Corporation
+ * Author: U. Artie Eoff <ullysses.a.eoff@intel.com>
+ *
+ * This library is free software; you can redistribute it and/or
+ * modify it under the terms of the GNU Library General Public
+ * License as published by the Free Software Foundation; either
+ * version 2 of the License, or (at your option) any later version.
+ *
+ * This library is distributed in the hope that it will be useful,
+ * but WITHOUT ANY WARRANTY; without even the implied warranty of
+ * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
+ * Library General Public License for more details.
+ *
+ * You should have received a copy of the GNU Library General Public
+ * License along with this library; if not, write to the0
+ * Free Software Foundation, Inc., 51 Franklin St, Fifth Floor,
+ * Boston, MA 02110-1301, USA.
+ */
+
+/**
+ * SECTION:element-vacompositor
+ * @title: vacompositor
+ * @short_description: A VA-API based video compositing element
+ *
+ * A video compositing element that uses VA-API VPP to accelerate the compose,
+ * blending, and scaling of multiple inputs into one output.
+ *
+ * ## Example launch line
+ * ```
+ * gst-launch-1.0 videotestsrc \
+ * ! "video/x-raw,format=(string)NV12,width=640,height=480" \
+ * ! tee name=testsrc ! queue ! vacompositor name=comp \
+ * sink_1::width=160 sink_1::height=120 sink_1::xpos=480 \
+ * sink_1::ypos=360 sink_1::alpha=0.75 \
+ * ! autovideosink testsrc. ! queue ! comp.
+ * ```
+ *
+ * Since: 1.22
+ *
+ */
+
+#ifdef HAVE_CONFIG_H
+#include "config.h"
+#endif
+
+#include "gstvacompositor.h"
+
+#include <gst/va/gstva.h>
+#include <gst/video/video.h>
+#include <va/va_drmcommon.h>
+
+#include "gstvacaps.h"
+#include "gstvadisplay_priv.h"
+#include "gstvafilter.h"
+
+GST_DEBUG_CATEGORY_STATIC (gst_va_compositor_debug);
+#define GST_CAT_DEFAULT gst_va_compositor_debug
+
+struct _GstVaCompositorPad
+{
+ GstVideoAggregatorPad parent;
+
+ GstBufferPool *pool;
+
+ gint xpos;
+ gint ypos;
+ gint width;
+ gint height;
+ gdouble alpha;
+};
+
+enum
+{
+ PROP_PAD_0,
+ PROP_PAD_XPOS,
+ PROP_PAD_YPOS,
+ PROP_PAD_WIDTH,
+ PROP_PAD_HEIGHT,
+ PROP_PAD_ALPHA,
+};
+
+#define DEFAULT_PAD_XPOS 0
+#define DEFAULT_PAD_YPOS 0
+#define DEFAULT_PAD_WIDTH 0
+#define DEFAULT_PAD_HEIGHT 0
+#define DEFAULT_PAD_ALPHA 1.0
+
+G_DEFINE_TYPE (GstVaCompositorPad, gst_va_compositor_pad,
+ GST_TYPE_VIDEO_AGGREGATOR_PAD);
+
+static void
+gst_va_compositor_pad_get_property (GObject * object, guint prop_id,
+ GValue * value, GParamSpec * pspec)
+{
+ GstVaCompositorPad *const self = GST_VA_COMPOSITOR_PAD (object);
+
+ switch (prop_id) {
+ case PROP_PAD_XPOS:
+ g_value_set_int (value, self->xpos);
+ break;
+ case PROP_PAD_YPOS:
+ g_value_set_int (value, self->ypos);
+ break;
+ case PROP_PAD_WIDTH:
+ g_value_set_int (value, self->width);
+ break;
+ case PROP_PAD_HEIGHT:
+ g_value_set_int (value, self->height);
+ break;
+ case PROP_PAD_ALPHA:
+ g_value_set_double (value, self->alpha);
+ break;
+ default:
+ G_OBJECT_WARN_INVALID_PROPERTY_ID (object, prop_id, pspec);
+ break;
+ }
+}
+
+static void
+gst_va_compositor_pad_set_property (GObject * object, guint prop_id,
+ const GValue * value, GParamSpec * pspec)
+{
+ GstVaCompositorPad *const self = GST_VA_COMPOSITOR_PAD (object);
+
+ GST_OBJECT_LOCK (object);
+ switch (prop_id) {
+ case PROP_PAD_XPOS:
+ self->xpos = g_value_get_int (value);
+ break;
+ case PROP_PAD_YPOS:
+ self->ypos = g_value_get_int (value);
+ break;
+ case PROP_PAD_WIDTH:
+ self->width = g_value_get_int (value);
+ break;
+ case PROP_PAD_HEIGHT:
+ self->height = g_value_get_int (value);
+ break;
+ case PROP_PAD_ALPHA:
+ self->alpha = g_value_get_double (value);
+ break;
+ default:
+ G_OBJECT_WARN_INVALID_PROPERTY_ID (object, prop_id, pspec);
+ break;
+ }
+ GST_OBJECT_UNLOCK (object);
+}
+
+static void
+gst_va_compositor_pad_finalize (GObject * object)
+{
+ GstVaCompositorPad *const self = GST_VA_COMPOSITOR_PAD (object);
+
+ if (self->pool) {
+ gst_buffer_pool_set_active (self->pool, FALSE);
+ gst_clear_object (&self->pool);
+ }
+
+ G_OBJECT_CLASS (gst_va_compositor_pad_parent_class)->finalize (object);
+}
+
+static void
+gst_va_compositor_pad_init (GstVaCompositorPad * self)
+{
+ self->pool = NULL;
+ self->xpos = DEFAULT_PAD_XPOS;
+ self->ypos = DEFAULT_PAD_YPOS;
+ self->width = DEFAULT_PAD_WIDTH;
+ self->height = DEFAULT_PAD_HEIGHT;
+ self->alpha = DEFAULT_PAD_ALPHA;
+}
+
+static void
+gst_va_compositor_pad_class_init (GstVaCompositorPadClass * klass)
+{
+ GObjectClass *const gobject_class = G_OBJECT_CLASS (klass);
+ GstVideoAggregatorPadClass *const vaggpad_class =
+ GST_VIDEO_AGGREGATOR_PAD_CLASS (klass);
+
+ gobject_class->finalize = gst_va_compositor_pad_finalize;
+ gobject_class->get_property = gst_va_compositor_pad_get_property;
+ gobject_class->set_property = gst_va_compositor_pad_set_property;
+
+ g_object_class_install_property (gobject_class, PROP_PAD_XPOS,
+ g_param_spec_int ("xpos", "X Position", "X Position of the picture",
+ G_MININT, G_MAXINT, DEFAULT_PAD_XPOS,
+ G_PARAM_READWRITE | GST_PARAM_CONTROLLABLE | G_PARAM_STATIC_STRINGS));
+ g_object_class_install_property (gobject_class, PROP_PAD_YPOS,
+ g_param_spec_int ("ypos", "Y Position", "Y Position of the picture",
+ G_MININT, G_MAXINT, DEFAULT_PAD_YPOS,
+ G_PARAM_READWRITE | GST_PARAM_CONTROLLABLE | G_PARAM_STATIC_STRINGS));
+ g_object_class_install_property (gobject_class, PROP_PAD_WIDTH,
+ g_param_spec_int ("width", "Width",
+ "Width of the picture (0, to use the width of the input frame)",
+ 0, G_MAXINT, DEFAULT_PAD_WIDTH,
+ G_PARAM_READWRITE | GST_PARAM_CONTROLLABLE | G_PARAM_STATIC_STRINGS));
+ g_object_class_install_property (gobject_class, PROP_PAD_HEIGHT,
+ g_param_spec_int ("height", "Height",
+ "Height of the picture (0, to use the height of the input frame)",
+ 0, G_MAXINT, DEFAULT_PAD_HEIGHT,
+ G_PARAM_READWRITE | GST_PARAM_CONTROLLABLE | G_PARAM_STATIC_STRINGS));
+ g_object_class_install_property (gobject_class, PROP_PAD_ALPHA,
+ g_param_spec_double ("alpha", "Alpha", "Alpha of the picture", 0.0, 1.0,
+ DEFAULT_PAD_ALPHA,
+ G_PARAM_READWRITE | GST_PARAM_CONTROLLABLE | G_PARAM_STATIC_STRINGS));
+
+ /* Don't use mapped video frames. Handle video buffers directly */
+ vaggpad_class->prepare_frame = NULL;
+ vaggpad_class->clean_frame = NULL;
+}
+
+#define GST_VA_COMPOSITOR(obj) ((GstVaCompositor *) obj)
+#define GST_VA_COMPOSITOR_CLASS(klass) ((GstVaCompositorClass *) klass)
+#define GST_VA_COMPOSITOR_GET_CLASS(obj) \
+ (G_TYPE_INSTANCE_GET_CLASS ((obj), G_TYPE_FROM_INSTANCE (obj), GstVaCompositorClass))
+
+typedef struct _GstVaCompositor GstVaCompositor;
+typedef struct _GstVaCompositorClass GstVaCompositorClass;
+
+struct _GstVaCompositorClass
+{
+ GstVideoAggregatorClass parent_class;
+
+ /*< private > */
+ gchar *render_device_path;
+};
+
+struct _GstVaCompositor
+{
+ GstVideoAggregator parent;
+
+ GstVaDisplay *display;
+ GstVaFilter *filter;
+
+ GstVideoInfo other_info; /* downstream info */
+ GstBufferPool *other_pool; /* downstream pool */
+
+ guint32 scale_method;
+};
+
+struct CData
+{
+ gchar *render_device_path;
+ gchar *description;
+};
+
+enum
+{
+ PROP_DEVICE_PATH = 1,
+ PROP_SCALE_METHOD,
+ N_PROPERTIES
+};
+
+static GParamSpec *properties[N_PROPERTIES];
+static GstElementClass *parent_class = NULL;
+
+static void
+gst_va_compositor_set_property (GObject * object, guint prop_id,
+ const GValue * value, GParamSpec * pspec)
+{
+ GstVaCompositor *const self = GST_VA_COMPOSITOR (object);
+
+ switch (prop_id) {
+ case PROP_SCALE_METHOD:
+ {
+ GST_OBJECT_LOCK (object);
+ self->scale_method = g_value_get_enum (value);
+ GST_OBJECT_UNLOCK (object);
+ break;
+ }
+ default:
+ G_OBJECT_WARN_INVALID_PROPERTY_ID (object, prop_id, pspec);
+ }
+}
+
+static void
+gst_va_compositor_get_property (GObject * object, guint prop_id,
+ GValue * value, GParamSpec * pspec)
+{
+ GstVaCompositor *const self = GST_VA_COMPOSITOR (object);
+
+ switch (prop_id) {
+ case PROP_DEVICE_PATH:
+ {
+ if (!(self->display && GST_IS_VA_DISPLAY_DRM (self->display))) {
+ g_value_set_string (value, NULL);
+ return;
+ }
+ g_object_get_property (G_OBJECT (self->display), "path", value);
+ break;
+ }
+ case PROP_SCALE_METHOD:
+ {
+ GST_OBJECT_LOCK (object);
+ g_value_set_enum (value, self->scale_method);
+ GST_OBJECT_UNLOCK (object);
+ break;
+ }
+ default:
+ G_OBJECT_WARN_INVALID_PROPERTY_ID (object, prop_id, pspec);
+ }
+}
+
+static gboolean
+gst_va_compositor_start (GstAggregator * agg)
+{
+ GstElement *const element = GST_ELEMENT (agg);
+ GstVaCompositor *const self = GST_VA_COMPOSITOR (agg);
+ GstVaCompositorClass *const klass = GST_VA_COMPOSITOR_GET_CLASS (agg);
+
+ if (!gst_va_ensure_element_data (element, klass->render_device_path,
+ &self->display))
+ return FALSE;
+ g_object_notify_by_pspec (G_OBJECT (self), properties[PROP_DEVICE_PATH]);
+
+ self->filter = gst_va_filter_new (self->display);
+ if (!gst_va_filter_open (self->filter))
+ return FALSE;
+
+ return GST_AGGREGATOR_CLASS (parent_class)->start (agg);
+}
+
+static gboolean
+gst_va_compositor_stop (GstAggregator * agg)
+{
+ GstVaCompositor *const self = GST_VA_COMPOSITOR (agg);
+
+ gst_va_filter_close (self->filter);
+ gst_clear_object (&self->filter);
+ gst_clear_object (&self->display);
+ g_object_notify_by_pspec (G_OBJECT (self), properties[PROP_DEVICE_PATH]);
+
+ return GST_AGGREGATOR_CLASS (parent_class)->stop (agg);
+}
+
+static void
+gst_va_compositor_dispose (GObject * object)
+{
+ GstVaCompositor *const self = GST_VA_COMPOSITOR (object);
+
+ if (self->other_pool) {
+ gst_buffer_pool_set_active (self->other_pool, FALSE);
+ gst_clear_object (&self->other_pool);
+ }
+
+ gst_clear_object (&self->display);
+
+ G_OBJECT_CLASS (parent_class)->dispose (object);
+}
+
+static GstPad *
+gst_va_compositor_request_new_pad (GstElement * element, GstPadTemplate * templ,
+ const gchar * req_name, const GstCaps * caps)
+{
+ GstPad *newpad = GST_PAD (GST_ELEMENT_CLASS
+ (parent_class)->request_new_pad (element, templ, req_name, caps));
+
+ if (!newpad)
+ GST_DEBUG_OBJECT (element, "could not create/add pad");
+ else
+ gst_child_proxy_child_added (GST_CHILD_PROXY (element), G_OBJECT (newpad),
+ GST_OBJECT_NAME (newpad));
+
+ return newpad;
+}
+
+static void
+gst_va_compositor_release_pad (GstElement * element, GstPad * pad)
+{
+ GstVaCompositor *const self = GST_VA_COMPOSITOR (element);
+
+ gst_child_proxy_child_removed (GST_CHILD_PROXY (self), G_OBJECT (pad),
+ GST_OBJECT_NAME (pad));
+
+ GST_ELEMENT_CLASS (parent_class)->release_pad (element, pad);
+}
+
+static void
+gst_va_compositor_set_context (GstElement * element, GstContext * context)
+{
+ GstVaDisplay *old_display, *new_display;
+ GstVaCompositor *const self = GST_VA_COMPOSITOR (element);
+ GstVaCompositorClass *const klass = GST_VA_COMPOSITOR_GET_CLASS (self);
+ gboolean ret;
+
+ old_display = self->display ? gst_object_ref (self->display) : NULL;
+ ret = gst_va_handle_set_context (element, context, klass->render_device_path,
+ &self->display);
+ new_display = self->display ? gst_object_ref (self->display) : NULL;
+
+ if (!ret
+ || (old_display && new_display && old_display != new_display
+ && self->filter)) {
+ GST_ELEMENT_WARNING (element, RESOURCE, BUSY,
+ ("Can't replace VA display while operating"), (NULL));
+ }
+
+ gst_clear_object (&old_display);
+ gst_clear_object (&new_display);
+
+ GST_ELEMENT_CLASS (parent_class)->set_context (element, context);
+}
+
+static gboolean
+_handle_context_query (GstVaCompositor * const self, GstQuery * query)
+{
+ GstVaDisplay *display = NULL;
+ gboolean ret = FALSE;
+
+ gst_object_replace ((GstObject **) & display, (GstObject *) self->display);
+ ret = gst_va_handle_context_query (GST_ELEMENT_CAST (self), query, display);
+ gst_clear_object (&display);
+
+ return ret;
+}
+
+static GstCaps *
+gst_va_compositor_sink_getcaps (GstPad * pad, GstCaps * filter)
+{
+ GstCaps *sinkcaps;
+ GstCaps *template_caps;
+ GstCaps *filtered_caps;
+ GstCaps *returned_caps;
+
+ template_caps = gst_pad_get_pad_template_caps (pad);
+
+ sinkcaps = gst_pad_get_current_caps (pad);
+ if (!sinkcaps) {
+ sinkcaps = gst_caps_ref (template_caps);
+ } else {
+ sinkcaps = gst_caps_merge (sinkcaps, gst_caps_ref (template_caps));
+ }
+
+ if (filter) {
+ filtered_caps = gst_caps_intersect (sinkcaps, filter);
+ gst_caps_unref (sinkcaps);
+ } else {
+ filtered_caps = sinkcaps;
+ }
+
+ returned_caps = gst_caps_intersect (filtered_caps, template_caps);
+
+ gst_caps_unref (template_caps);
+ gst_caps_unref (filtered_caps);
+
+ GST_DEBUG_OBJECT (pad, "returning %" GST_PTR_FORMAT, returned_caps);
+
+ return returned_caps;
+}
+
+static gboolean
+gst_va_compositor_sink_acceptcaps (GstPad * pad, GstCaps * caps)
+{
+ gboolean ret;
+ GstCaps *template_caps;
+
+ template_caps = gst_pad_get_pad_template_caps (pad);
+ template_caps = gst_caps_make_writable (template_caps);
+
+ ret = gst_caps_can_intersect (caps, template_caps);
+ GST_DEBUG_OBJECT (pad, "%saccepted caps %" GST_PTR_FORMAT,
+ (ret ? "" : "not "), caps);
+ gst_caps_unref (template_caps);
+
+ return ret;
+}
+
+static gboolean
+gst_va_compositor_sink_query (GstAggregator * agg, GstAggregatorPad * pad,
+ GstQuery * query)
+{
+ GstVaCompositor *const self = GST_VA_COMPOSITOR (agg);
+
+ switch (GST_QUERY_TYPE (query)) {
+ case GST_QUERY_CONTEXT:
+ {
+ if (_handle_context_query (self, query))
+ return TRUE;
+ break;
+ }
+ case GST_QUERY_CAPS:
+ {
+ GstCaps *filter, *caps;
+
+ gst_query_parse_caps (query, &filter);
+ caps = gst_va_compositor_sink_getcaps (GST_PAD (pad), filter);
+ gst_query_set_caps_result (query, caps);
+ gst_caps_unref (caps);
+ return TRUE;
+ }
+ case GST_QUERY_ACCEPT_CAPS:
+ {
+ GstCaps *caps;
+ gboolean ret;
+
+ gst_query_parse_accept_caps (query, &caps);
+ ret = gst_va_compositor_sink_acceptcaps (GST_PAD (pad), caps);
+ gst_query_set_accept_caps_result (query, ret);
+ return TRUE;
+ }
+ default:
+ break;
+ }
+
+ return GST_AGGREGATOR_CLASS (parent_class)->sink_query (agg, pad, query);
+}
+
+static gboolean
+gst_va_compositor_src_query (GstAggregator * agg, GstQuery * query)
+{
+ GstVaCompositor *const self = GST_VA_COMPOSITOR (agg);
+
+ switch (GST_QUERY_TYPE (query)) {
+ case GST_QUERY_CONTEXT:
+ if (_handle_context_query (self, query))
+ return TRUE;
+ break;
+ default:
+ break;
+ }
+
+ return GST_AGGREGATOR_CLASS (parent_class)->src_query (agg, query);
+}
+
+static GstAllocator *
+gst_va_compositor_allocator_from_caps (GstVaCompositor * const self,
+ GstCaps * caps)
+{
+ GstAllocator *allocator = NULL;
+
+ if (gst_caps_is_dmabuf (caps)) {
+ allocator = gst_va_dmabuf_allocator_new (self->display);
+ } else {
+ GArray *surface_formats = gst_va_filter_get_surface_formats (self->filter);
+ allocator = gst_va_allocator_new (self->display, surface_formats);
+ }
+
+ return allocator;
+}
+
+/* Answer upstream allocation query. */
+static gboolean
+gst_va_compositor_propose_allocation (GstAggregator * agg,
+ GstAggregatorPad * aggpad, GstQuery * decide_query, GstQuery * query)
+{
+ GstVaCompositor *const self = GST_VA_COMPOSITOR (agg);
+ GstAllocator *allocator = NULL;
+ GstAllocationParams params = { 0, };
+ GstBufferPool *pool;
+ GstCaps *caps;
+ GstVideoInfo info;
+ gboolean update_allocator = FALSE;
+ guint size, usage_hint = VA_SURFACE_ATTRIB_USAGE_HINT_GENERIC;
+
+ gst_query_parse_allocation (query, &caps, NULL);
+
+ if (!caps)
+ return FALSE;
+
+ if (!gst_video_info_from_caps (&info, caps))
+ return FALSE;
+
+ if (gst_query_get_n_allocation_pools (query) > 0)
+ return TRUE;
+
+ size = GST_VIDEO_INFO_SIZE (&info);
+
+ if (gst_query_get_n_allocation_params (query) > 0) {
+ gst_query_parse_nth_allocation_param (query, 0, &allocator, ¶ms);
+ if (!GST_IS_VA_DMABUF_ALLOCATOR (allocator)
+ && !GST_IS_VA_ALLOCATOR (allocator))
+ gst_clear_object (&allocator);
+ update_allocator = TRUE;
+ } else {
+ gst_allocation_params_init (¶ms);
+ }
+
+ if (!allocator) {
+ if (!(allocator = gst_va_compositor_allocator_from_caps (self, caps)))
+ return FALSE;
+ }
+
+ /* Now we have a VA-based allocator */
+
+ pool = gst_va_pool_new_with_config (caps, size, 1, 0, usage_hint,
+ GST_VA_FEATURE_AUTO, allocator, ¶ms);
+ if (!pool) {
+ gst_object_unref (allocator);
+ goto config_failed;
+ }
+
+ if (update_allocator)
+ gst_query_set_nth_allocation_param (query, 0, allocator, ¶ms);
+ else
+ gst_query_add_allocation_param (query, allocator, ¶ms);
+
+ gst_query_add_allocation_pool (query, pool, size, 1, 0);
+
+ GST_DEBUG_OBJECT (self,
+ "proposing %" GST_PTR_FORMAT " with allocator %" GST_PTR_FORMAT,
+ pool, allocator);
+
+ gst_object_unref (allocator);
+ gst_object_unref (pool);
+
+ gst_query_add_allocation_meta (query, GST_VIDEO_META_API_TYPE, NULL);
+
+ return TRUE;
+
+config_failed:
+ {
+ GST_ERROR_OBJECT (self, "failed to set config");
+ return FALSE;
+ }
+}
+
+static GstBufferPool *
+_create_other_pool (GstAllocator * allocator, GstAllocationParams * params,
+ GstCaps * caps, guint size)
+{
+ GstBufferPool *pool = NULL;
+ GstStructure *config;
+
+ pool = gst_video_buffer_pool_new ();
+ config = gst_buffer_pool_get_config (pool);
+
+ gst_buffer_pool_config_set_params (config, caps, size, 0, 0);
+ gst_buffer_pool_config_set_allocator (config, allocator, params);
+ if (!gst_buffer_pool_set_config (pool, config)) {
+ gst_clear_object (&pool);
+ }
+
+ return pool;
+}
+
+/* configure the allocation query that was answered downstream */
+static gboolean
+gst_va_compositor_decide_allocation (GstAggregator * agg, GstQuery * query)
+{
+ GstVaCompositor *const self = GST_VA_COMPOSITOR (agg);
+ GstVideoAggregator *const vagg = GST_VIDEO_AGGREGATOR (agg);
+
+ GstAllocator *allocator = NULL, *other_allocator = NULL;
+ GstAllocationParams params, other_params;
+ GstBufferPool *pool = NULL, *other_pool = NULL;
+ GstCaps *caps = NULL;
+ GstStructure *config;
+ GstVideoInfo info;
+ guint min, max, size = 0, usage_hint = VA_SURFACE_ATTRIB_USAGE_HINT_VPP_WRITE;
+ gboolean update_pool, update_allocator, has_videometa, copy_frames;
+
+ gst_query_parse_allocation (query, &caps, NULL);
+
+ gst_allocation_params_init (&other_params);
+ gst_allocation_params_init (¶ms);
+
+ if (!gst_video_info_from_caps (&info, caps)) {
+ GST_ERROR_OBJECT (self, "Cannot parse caps %" GST_PTR_FORMAT, caps);
+ return FALSE;
+ }
+
+ if (gst_query_get_n_allocation_params (query) > 0) {
+ gst_query_parse_nth_allocation_param (query, 0, &allocator, &other_params);
+ if (allocator && !(GST_IS_VA_DMABUF_ALLOCATOR (allocator)
+ || GST_IS_VA_ALLOCATOR (allocator))) {
+ /* save the allocator for the other pool */
+ other_allocator = allocator;
+ allocator = NULL;
+ }
+ update_allocator = TRUE;
+ } else {
+ update_allocator = FALSE;
+ }
+
+ if (gst_query_get_n_allocation_pools (query) > 0) {
+ gst_query_parse_nth_allocation_pool (query, 0, &pool, &size, &min, &max);
+
+ if (pool) {
+ if (!GST_IS_VA_POOL (pool)) {
+ GST_DEBUG_OBJECT (self,
+ "may need other pool for copy frames %" GST_PTR_FORMAT, pool);
+ other_pool = pool;
+ pool = NULL;
+ }
+ }
+
+ update_pool = TRUE;
+ } else {
+ size = GST_VIDEO_INFO_SIZE (&info);
+ min = 1;
+ max = 0;
+ update_pool = FALSE;
+ }
+
+ if (!allocator) {
+ if (gst_caps_is_dmabuf (caps) && GST_VIDEO_INFO_IS_RGB (&info))
+ usage_hint = VA_SURFACE_ATTRIB_USAGE_HINT_GENERIC;
+ if (!(allocator = gst_va_compositor_allocator_from_caps (self, caps)))
+ return FALSE;
+ }
+
+ if (!pool)
+ pool = gst_va_pool_new ();
+
+ config = gst_buffer_pool_get_config (pool);
+ gst_buffer_pool_config_set_allocator (config, allocator, ¶ms);
+ gst_buffer_pool_config_add_option (config, GST_BUFFER_POOL_OPTION_VIDEO_META);
+ gst_buffer_pool_config_set_params (config, caps, size, min, max);
+ gst_buffer_pool_config_set_va_allocation_params (config, usage_hint,
+ GST_VA_FEATURE_AUTO);
+ if (!gst_buffer_pool_set_config (pool, config)) {
+ gst_object_unref (allocator);
+ gst_object_unref (pool);
+ return FALSE;
+ }
+
+ if (GST_IS_VA_DMABUF_ALLOCATOR (allocator)) {
+ gst_va_dmabuf_allocator_get_format (allocator, &vagg->info, NULL);
+ } else if (GST_IS_VA_ALLOCATOR (allocator)) {
+ gst_va_allocator_get_format (allocator, &vagg->info, NULL, NULL);
+ }
+
+ if (update_allocator)
+ gst_query_set_nth_allocation_param (query, 0, allocator, ¶ms);
+ else
+ gst_query_add_allocation_param (query, allocator, ¶ms);
+
+ if (update_pool)
+ gst_query_set_nth_allocation_pool (query, 0, pool, size, min, max);
+ else
+ gst_query_add_allocation_pool (query, pool, size, min, max);
+
+ has_videometa = gst_query_find_allocation_meta (query,
+ GST_VIDEO_META_API_TYPE, NULL);
+
+ copy_frames = (!has_videometa && gst_va_pool_requires_video_meta (pool)
+ && gst_caps_is_raw (caps));
+ if (copy_frames) {
+ if (other_pool) {
+ gst_object_replace ((GstObject **) & self->other_pool,
+ (GstObject *) other_pool);
+ } else {
+ self->other_pool =
+ _create_other_pool (other_allocator, &other_params, caps, size);
+ }
+ GST_DEBUG_OBJECT (self, "Use the other pool for copy %" GST_PTR_FORMAT,
+ self->other_pool);
+ } else {
+ gst_clear_object (&self->other_pool);
+ }
+
+ GST_DEBUG_OBJECT (self,
+ "decided pool %" GST_PTR_FORMAT " with allocator %" GST_PTR_FORMAT,
+ pool, allocator);
+
+ gst_object_unref (allocator);
+ gst_object_unref (pool);
+ gst_clear_object (&other_allocator);
+ gst_clear_object (&other_pool);
+
+ return TRUE;
+}
+
+static GstBufferPool *
+_get_sinkpad_pool (GstVaCompositor * const self, GstVaCompositorPad * const pad)
+{
+ GstAllocator *allocator;
+ GstAllocationParams params = { 0, };
+ GstCaps *caps;
+ GstVideoInfo info;
+ guint size, usage_hint = VA_SURFACE_ATTRIB_USAGE_HINT_VPP_READ;
+
+ if (pad->pool)
+ return pad->pool;
+
+ gst_allocation_params_init (¶ms);
+
+ caps = gst_pad_get_current_caps (GST_PAD (pad));
+ gst_video_info_from_caps (&info, caps);
+
+ size = GST_VIDEO_INFO_SIZE (&info);
+
+ allocator = gst_va_compositor_allocator_from_caps (self, caps);
+ pad->pool = gst_va_pool_new_with_config (caps, size, 1, 0, usage_hint,
+ GST_VA_FEATURE_AUTO, allocator, ¶ms);
+ gst_caps_unref (caps);
+
+ if (!pad->pool) {
+ gst_object_unref (allocator);
+ return NULL;
+ }
+
+ if (GST_IS_VA_DMABUF_ALLOCATOR (allocator)) {
+ gst_va_dmabuf_allocator_get_format (allocator, &info, NULL);
+ } else if (GST_IS_VA_ALLOCATOR (allocator)) {
+ gst_va_allocator_get_format (allocator, &info, NULL, NULL);
+ }
+
+ gst_object_unref (allocator);
+
+ if (!gst_buffer_pool_set_active (pad->pool, TRUE)) {
+ GST_WARNING_OBJECT (self, "failed to active the sinkpad pool %"
+ GST_PTR_FORMAT, pad->pool);
+ return NULL;
+ }
+
+ return pad->pool;
+}
+
+static inline gsize
+_get_plane_data_size (GstVideoInfo * info, guint plane)
+{
+ gint comp[GST_VIDEO_MAX_COMPONENTS];
+ gint height, padded_height;
+
+ gst_video_format_info_component (info->finfo, plane, comp);
+
+ height = GST_VIDEO_INFO_HEIGHT (info);
+ padded_height =
+ GST_VIDEO_FORMAT_INFO_SCALE_HEIGHT (info->finfo, comp[0], height);
+
+ return GST_VIDEO_INFO_PLANE_STRIDE (info, plane) * padded_height;
+}
+
+static gboolean
+_try_import_dmabuf_unlocked (GstVaCompositor * const self, GstVideoInfo * info,
+ GstBuffer * inbuf)
+{
+ GstVideoMeta *meta;
+ GstMemory *mems[GST_VIDEO_MAX_PLANES];
+ guint i, n_mem, n_planes;
+ gsize offset[GST_VIDEO_MAX_PLANES];
+ uintptr_t fd[GST_VIDEO_MAX_PLANES];
+
+ n_planes = GST_VIDEO_INFO_N_PLANES (info);
+ n_mem = gst_buffer_n_memory (inbuf);
+ meta = gst_buffer_get_video_meta (inbuf);
+
+ /* This will eliminate most non-dmabuf out there */
+ if (!gst_is_dmabuf_memory (gst_buffer_peek_memory (inbuf, 0)))
+ return FALSE;
+
+ /* We cannot have multiple dmabuf per plane */
+ if (n_mem > n_planes)
+ return FALSE;
+
+ /* Update video info based on video meta */
+ if (meta) {
+ GST_VIDEO_INFO_WIDTH (info) = meta->width;
+ GST_VIDEO_INFO_HEIGHT (info) = meta->height;
+
+ for (i = 0; i < meta->n_planes; i++) {
+ GST_VIDEO_INFO_PLANE_OFFSET (info, i) = meta->offset[i];
+ GST_VIDEO_INFO_PLANE_STRIDE (info, i) = meta->stride[i];
+ }
+ }
+
+ /* Find and validate all memories */
+ for (i = 0; i < n_planes; i++) {
+ guint plane_size;
+ guint length;
+ guint mem_idx;
+ gsize mem_skip;
+
+ plane_size = _get_plane_data_size (info, i);
+
+ if (!gst_buffer_find_memory (inbuf, info->offset[i], plane_size,
+ &mem_idx, &length, &mem_skip))
+ return FALSE;
+
+ /* We can't have more then one dmabuf per plane */
+ if (length != 1)
+ return FALSE;
+
+ mems[i] = gst_buffer_peek_memory (inbuf, mem_idx);
+
+ /* And all memory found must be dmabuf */
+ if (!gst_is_dmabuf_memory (mems[i]))
+ return FALSE;
+
+ offset[i] = mems[i]->offset + mem_skip;
+ fd[i] = gst_dmabuf_memory_get_fd (mems[i]);
+ }
+
+ /* Now create a VASurfaceID for the buffer */
+ return gst_va_dmabuf_memories_setup (self->display, info, n_planes,
+ mems, fd, offset, VA_SURFACE_ATTRIB_USAGE_HINT_VPP_READ);
+}
+
+extern GRecMutex GST_VA_SHARED_LOCK;
+
+static gboolean
+_try_import_buffer (GstVaCompositor * const self,
+ GstVaCompositorPad * const pad, GstBuffer * inbuf)
+{
+ VASurfaceID surface;
+ GstCaps *caps;
+ GstVideoInfo info;
+ gboolean ret;
+
+ surface = gst_va_buffer_get_surface (inbuf);
+ if (surface != VA_INVALID_ID)
+ return TRUE;
+
+ caps = gst_pad_get_current_caps (GST_PAD (pad));
+ gst_video_info_from_caps (&info, caps);
+ gst_caps_unref (caps);
+
+ g_rec_mutex_lock (&GST_VA_SHARED_LOCK);
+ ret = _try_import_dmabuf_unlocked (self, &info, inbuf);
+ g_rec_mutex_unlock (&GST_VA_SHARED_LOCK);
+
+ return ret;
+}
+
+static GstFlowReturn
+gst_va_compositor_import_buffer (GstVaCompositor * const self,
+ GstVaCompositorPad * const pad, GstBuffer * inbuf, GstBuffer ** buf)
+{
+ GstBuffer *buffer = NULL;
+ GstBufferPool *pool;
+ GstFlowReturn ret;
+ GstCaps *caps;
+ GstVideoInfo info;
+ GstVideoFrame in_frame, out_frame;
+ gboolean imported, copied;
+
+ imported = _try_import_buffer (self, pad, inbuf);
+ if (imported) {
+ *buf = gst_buffer_ref (inbuf);
+ return GST_FLOW_OK;
+ }
+
+ /* input buffer doesn't come from a vapool, thus it is required to
+ * have a pool, grab from it a new buffer and copy the input
+ * buffer to the new one */
+ if (!(pool = _get_sinkpad_pool (self, pad)))
+ return GST_FLOW_ERROR;
+
+ ret = gst_buffer_pool_acquire_buffer (pool, &buffer, NULL);
+ if (ret != GST_FLOW_OK)
+ return ret;
+
+ GST_LOG_OBJECT (self, "copying input frame");
+
+ caps = gst_pad_get_current_caps (GST_PAD (pad));
+ gst_video_info_from_caps (&info, caps);
+ gst_caps_unref (caps);
+
+ if (!gst_video_frame_map (&in_frame, &info, inbuf, GST_MAP_READ))
+ goto invalid_buffer;
+
+ if (!gst_video_frame_map (&out_frame, &info, buffer, GST_MAP_WRITE)) {
+ gst_video_frame_unmap (&in_frame);
+ goto invalid_buffer;
+ }
+
+ copied = gst_video_frame_copy (&out_frame, &in_frame);
+
+ gst_video_frame_unmap (&out_frame);
+ gst_video_frame_unmap (&in_frame);
+
+ if (!copied)
+ goto invalid_buffer;
+
+ *buf = buffer;
+
+ return GST_FLOW_OK;
+
+invalid_buffer:
+ {
+ GST_ELEMENT_WARNING (self, CORE, NOT_IMPLEMENTED, (NULL),
+ ("invalid video buffer received"));
+ if (buffer)
+ gst_buffer_unref (buffer);
+ return GST_FLOW_OK;
+ }
+}
+
+typedef struct _GstVaCompositorSampleGenerator GstVaCompositorSampleGenerator;
+struct _GstVaCompositorSampleGenerator
+{
+ GstVaCompositor *comp;
+ GList *current;
+ GstVaComposeSample sample;
+};
+
+static GstVaComposeSample *
+gst_va_compositor_sample_next (gpointer data)
+{
+ GstVaCompositorSampleGenerator *generator;
+ GstVideoAggregatorPad *vaggpad;
+ GstVaCompositorPad *pad;
+ GstBuffer *inbuf;
+ GstBuffer *buf;
+ GstFlowReturn res;
+ GstVideoCropMeta *crop = NULL;
+
+ generator = (GstVaCompositorSampleGenerator *) data;
+
+ /* at the end of the generator? */
+ while (generator->current) {
+ /* get the current sinkpad for processing */
+ vaggpad = GST_VIDEO_AGGREGATOR_PAD (generator->current->data);
+
+ /* increment to next sinkpad */
+ generator->current = generator->current->next;
+
+ /* reset sample */
+ /* *INDENT-OFF* */
+ generator->sample = (GstVaComposeSample) { 0, };
+ /* *INDENT-ON* */
+
+ /* current sinkpad may not be queueing buffers yet (e.g. timestamp-offset)
+ * or it may have reached EOS */
+ if (!gst_video_aggregator_pad_has_current_buffer (vaggpad))
+ continue;
+
+ inbuf = gst_video_aggregator_pad_get_current_buffer (vaggpad);
+ pad = GST_VA_COMPOSITOR_PAD (vaggpad);
+
+ res = gst_va_compositor_import_buffer (generator->comp, pad, inbuf, &buf);
+ if (res != GST_FLOW_OK)
+ return &generator->sample;
+
+ crop = gst_buffer_get_video_crop_meta (buf);
+
+ GST_OBJECT_LOCK (vaggpad);
+ /* *INDENT-OFF* */
+ generator->sample = (GstVaComposeSample) {
+ .buffer = buf,
+ .input_region = (VARectangle) {
+ .x = crop ? crop->x : 0,
+ .y = crop ? crop->y : 0,
+ .width = crop ? crop->width : GST_VIDEO_INFO_WIDTH (&vaggpad->info),
+ .height = crop ? crop->height : GST_VIDEO_INFO_HEIGHT (&vaggpad->info),
+ },
+ .output_region = (VARectangle) {
+ .x = pad->xpos,
+ .y = pad->ypos,
+ .width = (pad->width == DEFAULT_PAD_WIDTH)
+ ? GST_VIDEO_INFO_WIDTH (&vaggpad->info) : pad->width,
+ .height = (pad->height == DEFAULT_PAD_HEIGHT)
+ ? GST_VIDEO_INFO_HEIGHT (&vaggpad->info) : pad->height,
+ },
+ .flags = generator->comp->scale_method,
+ .alpha = pad->alpha,
+ };
+ /* *INDENT-ON* */
+ GST_OBJECT_UNLOCK (vaggpad);
+
+ return &generator->sample;
+ }
+
+ return NULL;
+}
+
+static gboolean
+gst_va_compositor_copy_output_buffer (GstVaCompositor * const self,
+ GstBuffer * src_buf, GstBuffer * dst_buf)
+{
+ GstVideoAggregator *const vagg = GST_VIDEO_AGGREGATOR (self);
+ GstVideoFrame src_frame, dst_frame;
+
+ GST_LOG_OBJECT (self, "copying output buffer");
+
+ if (!gst_video_frame_map (&src_frame, &vagg->info, src_buf, GST_MAP_READ)) {
+ GST_ERROR_OBJECT (self, "couldn't map source buffer");
+ return FALSE;
+ }
+
+ if (!gst_video_frame_map (&dst_frame, &self->other_info, dst_buf,
+ GST_MAP_WRITE)) {
+ GST_ERROR_OBJECT (self, "couldn't map output buffer");
+ gst_video_frame_unmap (&src_frame);
+ return FALSE;
+ }
+
+ if (!gst_video_frame_copy (&dst_frame, &src_frame)) {
+ GST_ERROR_OBJECT (self, "couldn't copy output buffer");
+ gst_video_frame_unmap (&src_frame);
+ gst_video_frame_unmap (&dst_frame);
+ return FALSE;
+ }
+
+ gst_video_frame_unmap (&src_frame);
+ gst_video_frame_unmap (&dst_frame);
+
+ return TRUE;
+}
+
+static GstFlowReturn
+gst_va_compositor_aggregate_frames (GstVideoAggregator * vagg,
+ GstBuffer * outbuf)
+{
+ GstVaCompositor *const self = GST_VA_COMPOSITOR (vagg);
+ GstVaCompositorSampleGenerator generator;
+ GstVaComposeTransaction tx;
+ GstBuffer *vabuffer;
+ gboolean need_copy = FALSE;
+ GstFlowReturn ret = GST_FLOW_OK;
+
+ if (self->other_pool) {
+ /* create a va buffer for filter */
+ ret = GST_VIDEO_AGGREGATOR_CLASS (parent_class)->create_output_buffer
+ (vagg, &vabuffer);
+ if (ret != GST_FLOW_OK)
+ return ret;
+
+ need_copy = TRUE;
+ } else {
+ /* already a va buffer */
+ vabuffer = gst_buffer_ref (outbuf);
+ }
+
+ /* *INDENT-OFF* */
+ generator = (GstVaCompositorSampleGenerator) {
+ .comp = self,
+ .current = GST_ELEMENT (self)->sinkpads,
+ };
+ tx = (GstVaComposeTransaction) {
+ .next = gst_va_compositor_sample_next,
+ .output = vabuffer,
+ .user_data = (gpointer) &generator,
+ };
+ /* *INDENT-ON* */
+
+ GST_OBJECT_LOCK (self);
+
+ if (!gst_va_filter_compose (self->filter, &tx)) {
+ GST_ERROR_OBJECT (self, "couldn't apply filter");
+ ret = GST_FLOW_ERROR;
+ }
+
+ GST_OBJECT_UNLOCK (self);
+
+ if (ret != GST_FLOW_OK)
+ goto done;
+
+ if (need_copy && !gst_va_compositor_copy_output_buffer (self, vabuffer,
+ outbuf)) {
+ GST_ERROR_OBJECT (self, "couldn't copy va buffer to output buffer");
+ ret = GST_FLOW_ERROR;
+ }
+
+done:
+ gst_buffer_unref (vabuffer);
+ return ret;
+}
+
+static GstFlowReturn
+gst_va_compositor_create_output_buffer (GstVideoAggregator * vagg,
+ GstBuffer ** outbuf)
+{
+ GstVaCompositor *const self = GST_VA_COMPOSITOR (vagg);
+ GstFlowReturn ret;
+
+ *outbuf = NULL;
+
+ if (!self->other_pool)
+ /* no copy necessary, so use a va buffer directly */
+ return GST_VIDEO_AGGREGATOR_CLASS (parent_class)->create_output_buffer
+ (vagg, outbuf);
+
+ /* use output buffers from downstream pool for copy */
+ if (!gst_buffer_pool_is_active (self->other_pool) &&
+ !gst_buffer_pool_set_active (self->other_pool, TRUE)) {
+ GST_ERROR_OBJECT (self, "failed to activate other pool %"
+ GST_PTR_FORMAT, self->other_pool);
+ return GST_FLOW_ERROR;
+ }
+
+ /* acquire a buffer from downstream pool for copy */
+ ret = gst_buffer_pool_acquire_buffer (self->other_pool, outbuf, NULL);
+ if (ret != GST_FLOW_OK || !*outbuf) {
+ GST_ERROR_OBJECT (self, "failed to acquire output buffer");
+ return GST_FLOW_ERROR;
+ }
+
+ return GST_FLOW_OK;
+}
+
+static gboolean
+gst_va_compositor_negotiated_src_caps (GstAggregator * agg, GstCaps * caps)
+{
+ GstVaCompositor *const self = GST_VA_COMPOSITOR (agg);
+
+ if (!gst_video_info_from_caps (&self->other_info, caps)) {
+ GST_ERROR_OBJECT (self, "invalid caps");
+ return FALSE;
+ }
+
+ if (self->other_pool) {
+ gst_buffer_pool_set_active (self->other_pool, FALSE);
+ gst_clear_object (&self->other_pool);
+ }
+
+ return GST_AGGREGATOR_CLASS (parent_class)->negotiated_src_caps (agg, caps);
+}
+
+static void
+gst_va_compositor_pad_get_output_size (GstVaCompositorPad * const pad,
+ gint * width, gint * height)
+{
+ GstVideoAggregatorPad *vaggpad = GST_VIDEO_AGGREGATOR_PAD (pad);
+ *width = (pad->width == DEFAULT_PAD_WIDTH)
+ ? GST_VIDEO_INFO_WIDTH (&vaggpad->info) : pad->width;
+ *height = (pad->height == DEFAULT_PAD_HEIGHT)
+ ? GST_VIDEO_INFO_HEIGHT (&vaggpad->info) : pad->height;
+
+ *width += MAX (pad->xpos, 0);
+ *height += MAX (pad->ypos, 0);
+}
+
+static GstCaps *
+gst_va_compositor_fixate_src_caps (GstAggregator * agg, GstCaps * caps)
+{
+ GstVideoAggregator *const vagg = GST_VIDEO_AGGREGATOR (agg);
+ GList *l;
+ gint best_width = -1, best_height = -1;
+ gint best_fps_n = -1, best_fps_d = -1;
+ gdouble best_fps = 0.;
+ GstCaps *ret = NULL;
+ GstStructure *s;
+
+ ret = gst_caps_make_writable (caps);
+
+ GST_OBJECT_LOCK (vagg);
+ for (l = GST_ELEMENT (vagg)->sinkpads; l; l = l->next) {
+ GstVideoAggregatorPad *const vaggpad = l->data;
+ GstVaCompositorPad *const pad = GST_VA_COMPOSITOR_PAD (vaggpad);
+ gint this_width, this_height;
+ gint fps_n, fps_d;
+ gdouble cur_fps;
+
+ fps_n = GST_VIDEO_INFO_FPS_N (&vaggpad->info);
+ fps_d = GST_VIDEO_INFO_FPS_D (&vaggpad->info);
+
+ gst_va_compositor_pad_get_output_size (pad, &this_width, &this_height);
+
+ if (best_width < this_width)
+ best_width = this_width;
+ if (best_height < this_height)
+ best_height = this_height;
+
+ if (fps_d == 0)
+ cur_fps = 0.0;
+ else
+ gst_util_fraction_to_double (fps_n, fps_d, &cur_fps);
+
+ if (best_fps < cur_fps) {
+ best_fps = cur_fps;
+ best_fps_n = fps_n;
+ best_fps_d = fps_d;
+ }
+ }
+ GST_OBJECT_UNLOCK (vagg);
+
+ if (best_fps_n <= 0 || best_fps_d <= 0 || best_fps == 0.0) {
+ best_fps_n = 25;
+ best_fps_d = 1;
+ best_fps = 25.0;
+ }
+
+ s = gst_caps_get_structure (ret, 0);
+ gst_structure_fixate_field_nearest_int (s, "width", best_width);
+ gst_structure_fixate_field_nearest_int (s, "height", best_height);
+ if (gst_structure_has_field (s, "framerate")) {
+ gst_structure_fixate_field_nearest_fraction (s, "framerate", best_fps_n,
+ best_fps_d);
+ } else {
+ gst_structure_set (s, "framerate", GST_TYPE_FRACTION, best_fps_n,
+ best_fps_d, NULL);
+ }
+
+ return gst_caps_fixate (ret);
+}
+
+/* *INDENT-OFF* */
+static const gchar *caps_str =
+ GST_VIDEO_CAPS_MAKE_WITH_FEATURES (GST_CAPS_FEATURE_MEMORY_VA,
+ "{ NV12, I420, YV12, YUY2, RGBA, BGRA, P010_10LE, ARGB, ABGR }") " ;"
+ GST_VIDEO_CAPS_MAKE ("{ VUYA, GRAY8, NV12, NV21, YUY2, UYVY, YV12, "
+ "I420, P010_10LE, RGBA, BGRA, ARGB, ABGR }");
+/* *INDENT-ON* */
+
+static void
+gst_va_compositor_class_init (gpointer g_class, gpointer class_data)
+{
+ GstCaps *doc_caps, *caps = NULL;
+ GstPadTemplate *sink_pad_templ, *src_pad_templ;
+ GObjectClass *const object_class = G_OBJECT_CLASS (g_class);
+ GstElementClass *const element_class = GST_ELEMENT_CLASS (g_class);
+ GstAggregatorClass *const agg_class = GST_AGGREGATOR_CLASS (g_class);
+ GstVideoAggregatorClass *const vagg_class =
+ GST_VIDEO_AGGREGATOR_CLASS (g_class);
+ GstVaCompositorClass *const klass = GST_VA_COMPOSITOR_CLASS (g_class);
+ GstVaDisplay *display;
+ GstVaFilter *filter;
+ struct CData *cdata = class_data;
+ gchar *long_name;
+
+ parent_class = g_type_class_peek_parent (g_class);
+
+ klass->render_device_path = g_strdup (cdata->render_device_path);
+
+ if (cdata->description) {
+ long_name = g_strdup_printf ("VA-API Video Compositor in %s",
+ cdata->description);
+ } else {
+ long_name = g_strdup ("VA-API Video Compositor");
+ }
+
+ display = gst_va_display_drm_new_from_path (klass->render_device_path);
+ filter = gst_va_filter_new (display);
+
+ if (gst_va_filter_open (filter)) {
+ caps = gst_va_filter_get_caps (filter);
+ } else {
+ caps = gst_caps_from_string (caps_str);
+ }
+
+ object_class->dispose = GST_DEBUG_FUNCPTR (gst_va_compositor_dispose);
+ object_class->get_property =
+ GST_DEBUG_FUNCPTR (gst_va_compositor_get_property);
+ object_class->set_property =
+ GST_DEBUG_FUNCPTR (gst_va_compositor_set_property);
+
+ gst_element_class_set_static_metadata (element_class, long_name,
+ "Filter/Editor/Video/Compositor/Hardware",
+ "VA-API based video compositor",
+ "U. Artie Eoff <ullysses.a.eoff@intel.com>");
+
+ element_class->request_new_pad =
+ GST_DEBUG_FUNCPTR (gst_va_compositor_request_new_pad);
+ element_class->release_pad =
+ GST_DEBUG_FUNCPTR (gst_va_compositor_release_pad);
+ element_class->set_context =
+ GST_DEBUG_FUNCPTR (gst_va_compositor_set_context);
+
+ doc_caps = gst_caps_from_string (caps_str);
+
+ sink_pad_templ = gst_pad_template_new_with_gtype ("sink_%u", GST_PAD_SINK,
+ GST_PAD_REQUEST, caps, GST_TYPE_VA_COMPOSITOR_PAD);
+ gst_element_class_add_pad_template (element_class, sink_pad_templ);
+ gst_pad_template_set_documentation_caps (sink_pad_templ,
+ gst_caps_ref (doc_caps));
+
+ src_pad_templ = gst_pad_template_new_with_gtype ("src", GST_PAD_SRC,
+ GST_PAD_ALWAYS, caps, GST_TYPE_AGGREGATOR_PAD);
+ gst_element_class_add_pad_template (element_class, src_pad_templ);
+ gst_pad_template_set_documentation_caps (src_pad_templ,
+ gst_caps_ref (doc_caps));
+
+ gst_caps_unref (doc_caps);
+ gst_caps_unref (caps);
+
+ agg_class->sink_query = GST_DEBUG_FUNCPTR (gst_va_compositor_sink_query);
+ agg_class->src_query = GST_DEBUG_FUNCPTR (gst_va_compositor_src_query);
+ agg_class->start = GST_DEBUG_FUNCPTR (gst_va_compositor_start);
+ agg_class->stop = GST_DEBUG_FUNCPTR (gst_va_compositor_stop);
+ agg_class->propose_allocation =
+ GST_DEBUG_FUNCPTR (gst_va_compositor_propose_allocation);
+ agg_class->fixate_src_caps =
+ GST_DEBUG_FUNCPTR (gst_va_compositor_fixate_src_caps);
+ agg_class->negotiated_src_caps =
+ GST_DEBUG_FUNCPTR (gst_va_compositor_negotiated_src_caps);
+ agg_class->decide_allocation =
+ GST_DEBUG_FUNCPTR (gst_va_compositor_decide_allocation);
+
+ vagg_class->aggregate_frames =
+ GST_DEBUG_FUNCPTR (gst_va_compositor_aggregate_frames);
+ vagg_class->create_output_buffer =
+ GST_DEBUG_FUNCPTR (gst_va_compositor_create_output_buffer);
+
+ properties[PROP_DEVICE_PATH] = g_param_spec_string ("device-path",
+ "Device Path", "DRM device path", NULL,
+ G_PARAM_READABLE | G_PARAM_STATIC_STRINGS);
+
+ properties[PROP_SCALE_METHOD] = g_param_spec_enum ("scale-method",
+ "Scale Method", "Scale method to use", GST_TYPE_VA_SCALE_METHOD,
+ VA_FILTER_SCALING_DEFAULT, G_PARAM_READWRITE | G_PARAM_STATIC_STRINGS);
+
+ g_object_class_install_properties (object_class, N_PROPERTIES, properties);
+
+ g_free (long_name);
+ g_free (cdata->description);
+ g_free (cdata->render_device_path);
+ g_free (cdata);
+ gst_object_unref (filter);
+ gst_object_unref (display);
+}
+
+static GObject *
+gst_va_compositor_child_proxy_get_child_by_index (GstChildProxy * proxy,
+ guint index)
+{
+ GstVaCompositor *self = GST_VA_COMPOSITOR (proxy);
+ GObject *obj = NULL;
+
+ GST_OBJECT_LOCK (self);
+ obj = g_list_nth_data (GST_ELEMENT_CAST (self)->sinkpads, index);
+ if (obj)
+ gst_object_ref (obj);
+ GST_OBJECT_UNLOCK (self);
+
+ return obj;
+}
+
+static guint
+gst_va_compositor_child_proxy_get_children_count (GstChildProxy * proxy)
+{
+ GstVaCompositor *self = GST_VA_COMPOSITOR (proxy);
+ guint count = 0;
+
+ GST_OBJECT_LOCK (self);
+ count = GST_ELEMENT_CAST (self)->numsinkpads;
+ GST_OBJECT_UNLOCK (self);
+ GST_INFO_OBJECT (self, "Children Count: %d", count);
+
+ return count;
+}
+
+static void
+gst_va_compositor_child_proxy_init (gpointer g_iface, gpointer iface_data)
+{
+ GstChildProxyInterface *iface = (GstChildProxyInterface *) g_iface;
+
+ iface->get_child_by_index = gst_va_compositor_child_proxy_get_child_by_index;
+ iface->get_children_count = gst_va_compositor_child_proxy_get_children_count;
+}
+
+static void
+gst_va_compositor_init (GTypeInstance * instance, gpointer g_class)
+{
+ GstVaCompositor *const self = GST_VA_COMPOSITOR (instance);
+
+ self->other_pool = NULL;
+}
+
+static gpointer
+_register_debug_category (gpointer data)
+{
+ GST_DEBUG_CATEGORY_INIT (gst_va_compositor_debug, "vacompositor", 0,
+ "VA Video Compositor");
+
+ return NULL;
+}
+
+gboolean
+gst_va_compositor_register (GstPlugin * plugin, GstVaDevice * device,
+ guint rank)
+{
+ static GOnce debug_once = G_ONCE_INIT;
+ GType type;
+ GTypeInfo type_info = {
+ .class_size = sizeof (GstVaCompositorClass),
+ .class_init = gst_va_compositor_class_init,
+ .instance_size = sizeof (GstVaCompositor),
+ .instance_init = gst_va_compositor_init,
+ };
+ GInterfaceInfo interface_info = {
+ (GInterfaceInitFunc) gst_va_compositor_child_proxy_init,
+ };
+ struct CData *cdata;
+ gboolean ret;
+ gchar *type_name, *feature_name;
+
+ g_return_val_if_fail (GST_IS_PLUGIN (plugin), FALSE);
+ g_return_val_if_fail (GST_IS_VA_DEVICE (device), FALSE);
+
+ cdata = g_new (struct CData, 1);
+ cdata->description = NULL;
+ cdata->render_device_path = g_strdup (device->render_device_path);
+
+ type_info.class_data = cdata;
+
+ type_name = g_strdup ("GstVaCompositor");
+ feature_name = g_strdup ("vacompositor");
+
+ /* The first compositor to be registered should use a constant
+ * name, like vacompositor, for any additional compositors, we
+ * create unique names, using the render device name. */
+ if (g_type_from_name (type_name)) {
+ gchar *basename = g_path_get_basename (device->render_device_path);
+ g_free (type_name);
+ g_free (feature_name);
+ type_name = g_strdup_printf ("GstVa%sCompositor", basename);
+ feature_name = g_strdup_printf ("va%scompositor", basename);
+ cdata->description = basename;
+
+ /* lower rank for non-first device */
+ if (rank > 0)
+ rank--;
+ }
+
+ g_once (&debug_once, _register_debug_category, NULL);
+
+ type = g_type_register_static (GST_TYPE_VIDEO_AGGREGATOR, type_name,
+ &type_info, 0);
+ g_type_add_interface_static (type, GST_TYPE_CHILD_PROXY, &interface_info);
+
+ ret = gst_element_register (plugin, feature_name, rank, type);
+
+ g_free (type_name);
+ g_free (feature_name);
+
+ return ret;
+}