/* * gstvaapivideoconverter_x11.h - VA video converter to X11 pixmap * * Copyright (C) 2013 Intel Corporation * Author: Gwenole Beauchesne * * This library is free software; you can redistribute it and/or * modify it under the terms of the GNU Lesser General Public License * as published by the Free Software Foundation; either version 2.1 * 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 * Lesser General Public License for more details. * * You should have received a copy of the GNU Lesser General Public * License along with this library; if not, write to the Free * Software Foundation, Inc., 51 Franklin Street, Fifth Floor, * Boston, MA 02110-1301 USA */ #include "gst/vaapi/sysdeps.h" #include #include "gstvaapivideoconverter_x11.h" #include "gstvaapipluginutil.h" #include "gstvaapivideometa.h" #if GST_CHECK_VERSION(1,0,0) typedef gboolean (*GstSurfaceUploadFunction) (GstSurfaceConverter *, GstBuffer *); #else typedef gboolean (*GstSurfaceUploadFunction) (GstSurfaceConverter *, GstSurfaceBuffer *); #endif static void gst_vaapi_video_converter_x11_iface_init (GstSurfaceConverterInterface * iface); G_DEFINE_TYPE_WITH_CODE (GstVaapiVideoConverterX11, gst_vaapi_video_converter_x11, G_TYPE_OBJECT, G_IMPLEMENT_INTERFACE (GST_TYPE_SURFACE_CONVERTER, gst_vaapi_video_converter_x11_iface_init)); #define GST_VAAPI_VIDEO_CONVERTER_X11_GET_PRIVATE(obj) \ (G_TYPE_INSTANCE_GET_PRIVATE ((obj), GST_VAAPI_TYPE_VIDEO_CONVERTER_X11, \ GstVaapiVideoConverterX11Private)) struct _GstVaapiVideoConverterX11Private { GstVaapiPixmap *pixmap; XID pixmap_id; }; static gboolean gst_vaapi_video_converter_x11_upload (GstSurfaceConverter * self, GstBuffer * buffer); static void gst_vaapi_video_converter_x11_dispose (GObject * object) { GstVaapiVideoConverterX11Private *const priv = GST_VAAPI_VIDEO_CONVERTER_X11 (object)->priv; gst_vaapi_pixmap_replace (&priv->pixmap, NULL); G_OBJECT_CLASS (gst_vaapi_video_converter_x11_parent_class)->dispose (object); } static void gst_vaapi_video_converter_x11_class_init (GstVaapiVideoConverterX11Class * klass) { GObjectClass *const object_class = G_OBJECT_CLASS (klass); g_type_class_add_private (klass, sizeof (GstVaapiVideoConverterX11Private)); object_class->dispose = gst_vaapi_video_converter_x11_dispose; } static void gst_vaapi_video_converter_x11_init (GstVaapiVideoConverterX11 * buffer) { buffer->priv = GST_VAAPI_VIDEO_CONVERTER_X11_GET_PRIVATE (buffer); } static void gst_vaapi_video_converter_x11_iface_init (GstSurfaceConverterInterface * iface) { iface->upload = (GstSurfaceUploadFunction) gst_vaapi_video_converter_x11_upload; } static gboolean set_pixmap (GstVaapiVideoConverterX11 * converter, GstVaapiDisplay * display, XID pixmap_id) { GstVaapiVideoConverterX11Private *const priv = converter->priv; GstVaapiPixmap *pixmap; pixmap = gst_vaapi_pixmap_x11_new_with_xid (display, pixmap_id); if (!pixmap) return FALSE; gst_vaapi_pixmap_replace (&priv->pixmap, pixmap); gst_vaapi_pixmap_unref (pixmap); priv->pixmap_id = pixmap_id; return TRUE; } /** * gst_vaapi_video_converter_x11_new: * @surface: the #GstSurfaceBuffer * @type: type of the target buffer (must be "x11-pixmap") * @dest: target of the conversion (must be an X11 pixmap id) * * Creates an empty #GstBuffer. The caller is responsible for * completing the initialization of the buffer with the * gst_vaapi_video_converter_x11_set_*() functions. * * Return value: the newly allocated #GstBuffer, or %NULL on error */ GstSurfaceConverter * gst_vaapi_video_converter_x11_new (GstBuffer * buffer, const gchar * type, GValue * dest) { GstVaapiVideoMeta *const meta = gst_buffer_get_vaapi_video_meta (buffer); GstVaapiVideoConverterX11 *converter; /* We only support X11 pixmap conversion */ if (strcmp (type, "x11-pixmap") != 0 || !G_VALUE_HOLDS_UINT (dest)) return NULL; converter = g_object_new (GST_VAAPI_TYPE_VIDEO_CONVERTER_X11, NULL); if (!converter) return NULL; if (!set_pixmap (converter, gst_vaapi_video_meta_get_display (meta), g_value_get_uint (dest))) goto error; return GST_SURFACE_CONVERTER (converter); error: g_object_unref (converter); return NULL; } gboolean gst_vaapi_video_converter_x11_upload (GstSurfaceConverter * self, GstBuffer * buffer) { GstVaapiVideoConverterX11 *const converter = GST_VAAPI_VIDEO_CONVERTER_X11 (self); GstVaapiVideoConverterX11Private *const priv = converter->priv; GstVaapiVideoMeta *const meta = gst_buffer_get_vaapi_video_meta (buffer); const GstVaapiRectangle *crop_rect = NULL; GstVaapiSurface *surface; GstVaapiDisplay *old_display, *new_display; g_return_val_if_fail (meta != NULL, FALSE); surface = gst_vaapi_video_meta_get_surface (meta); if (!surface) return FALSE; old_display = gst_vaapi_object_get_display (GST_VAAPI_OBJECT (priv->pixmap)); new_display = gst_vaapi_object_get_display (GST_VAAPI_OBJECT (surface)); if (old_display != new_display) { if (!set_pixmap (converter, new_display, priv->pixmap_id)) return FALSE; } if (!gst_vaapi_apply_composition (surface, buffer)) GST_WARNING ("could not update subtitles"); #if GST_CHECK_VERSION(1,0,0) GstVideoCropMeta *const crop_meta = gst_buffer_get_video_crop_meta (buffer); if (crop_meta) { GstVaapiRectangle crop_rect_tmp; crop_rect = &crop_rect_tmp; crop_rect_tmp.x = crop_meta->x; crop_rect_tmp.y = crop_meta->y; crop_rect_tmp.width = crop_meta->width; crop_rect_tmp.height = crop_meta->height; } #endif if (!crop_rect) crop_rect = gst_vaapi_video_meta_get_render_rect (meta); return gst_vaapi_pixmap_put_surface (priv->pixmap, surface, crop_rect, gst_vaapi_video_meta_get_render_flags (meta)); }