/* * gstvaapivideobuffer.c - Gstreamer/VA video buffer * * Copyright (C) 2010-2011 Splitted-Desktop Systems * Copyright (C) 2011-2013 Intel Corporation * * 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 */ /** * SECTION:gstvaapivideobuffer * @short_description: VA video buffer for GStreamer */ #include "gst/vaapi/sysdeps.h" #include "gstvaapivideobuffer.h" #if USE_X11 && !GST_CHECK_VERSION(1,1,0) # include "gstvaapivideoconverter_x11.h" #endif #if USE_GLX && !GST_CHECK_VERSION(1,1,0) # include "gstvaapivideoconverter_glx.h" #endif #if GST_CHECK_VERSION(1,1,0) static inline GstBuffer * gst_surface_buffer_new(void) { return gst_buffer_new(); } #elif GST_CHECK_VERSION(1,0,0) #include <gst/video/gstsurfacemeta.h> #define GST_VAAPI_SURFACE_META_CAST(obj) \ ((GstVaapiSurfaceMeta *)(obj)) typedef struct _GstVaapiSurfaceMeta GstVaapiSurfaceMeta; struct _GstVaapiSurfaceMeta { GstSurfaceMeta base; GstBuffer *buffer; }; #define GST_VAAPI_SURFACE_META_INFO gst_vaapi_surface_meta_get_info() static const GstMetaInfo * gst_vaapi_surface_meta_get_info(void); typedef GstSurfaceConverter *(*GstSurfaceConverterCreateFunc)( GstSurfaceMeta *meta, const gchar *type, GValue *dest); #if USE_X11 static GstSurfaceConverter * gst_vaapi_surface_create_converter_x11(GstSurfaceMeta *base_meta, const gchar *type, GValue *dest) { GstVaapiSurfaceMeta * const meta = GST_VAAPI_SURFACE_META_CAST(base_meta); return gst_vaapi_video_converter_x11_new(meta->buffer, type, dest); } #undef gst_vaapi_video_converter_x11_new #define gst_vaapi_video_converter_x11_new \ gst_vaapi_surface_create_converter_x11 #endif #if USE_GLX static GstSurfaceConverter * gst_vaapi_surface_create_converter_glx(GstSurfaceMeta *base_meta, const gchar *type, GValue *dest) { GstVaapiSurfaceMeta * const meta = GST_VAAPI_SURFACE_META_CAST(base_meta); return gst_vaapi_video_converter_glx_new(meta->buffer, type, dest); } #undef gst_vaapi_video_converter_glx_new #define gst_vaapi_video_converter_glx_new \ gst_vaapi_surface_create_converter_glx #endif static GstSurfaceConverter * gst_vaapi_surface_create_converter(GstSurfaceMeta *base_meta, const gchar *type, GValue *dest) { GstVaapiSurfaceMeta * const meta = GST_VAAPI_SURFACE_META_CAST(base_meta); GstVaapiVideoMeta * const vmeta = gst_buffer_get_vaapi_video_meta(meta->buffer); GstSurfaceConverterCreateFunc func; if (G_UNLIKELY(!vmeta)) return NULL; func = (GstSurfaceConverterCreateFunc) gst_vaapi_video_meta_get_surface_converter(vmeta); return func ? func(base_meta, type, dest) : NULL; } static gboolean gst_vaapi_surface_meta_init(GstVaapiSurfaceMeta *meta, gpointer params, GstBuffer *buffer) { meta->base.create_converter = gst_vaapi_surface_create_converter; meta->buffer = buffer; return TRUE; } static void gst_vaapi_surface_meta_free(GstVaapiSurfaceMeta *meta, GstBuffer *buffer) { } static gboolean gst_vaapi_surface_meta_transform(GstBuffer *dst_buffer, GstMeta *meta, GstBuffer *src_buffer, GQuark type, gpointer data) { GstVaapiVideoMeta * const src_vmeta = gst_buffer_get_vaapi_video_meta(src_buffer); if (GST_META_TRANSFORM_IS_COPY(type)) { GstVaapiSurfaceMeta * const dst_smeta = GST_VAAPI_SURFACE_META_CAST( gst_buffer_add_meta(dst_buffer, GST_VAAPI_SURFACE_META_INFO, NULL)); /* Note: avoid meta lookups in gst_vaapi_surface_create_converter() by directly calling the GstVaapiVideoMeta::surface_converter hook */ dst_smeta->base.create_converter = (GstSurfaceConverterCreateFunc) gst_vaapi_video_meta_get_surface_converter(src_vmeta); return TRUE; } return FALSE; } const GstMetaInfo * gst_vaapi_surface_meta_get_info(void) { static gsize g_meta_info; if (g_once_init_enter(&g_meta_info)) { gsize meta_info = GPOINTER_TO_SIZE(gst_meta_register( GST_SURFACE_META_API_TYPE, "GstVaapiSurfaceMeta", sizeof(GstVaapiSurfaceMeta), (GstMetaInitFunction)gst_vaapi_surface_meta_init, (GstMetaFreeFunction)gst_vaapi_surface_meta_free, (GstMetaTransformFunction)gst_vaapi_surface_meta_transform)); g_once_init_leave(&g_meta_info, meta_info); } return GSIZE_TO_POINTER(g_meta_info); } static GstBuffer * gst_surface_buffer_new(void) { GstBuffer * const buffer = gst_buffer_new(); if (buffer) gst_buffer_add_meta(buffer, GST_VAAPI_SURFACE_META_INFO, NULL); return buffer; } #else #include <gst/video/gstsurfacebuffer.h> #define GST_VAAPI_TYPE_VIDEO_BUFFER \ (gst_vaapi_video_buffer_get_type()) #define GST_VAAPI_VIDEO_BUFFER(obj) \ (G_TYPE_CHECK_INSTANCE_CAST((obj), \ GST_VAAPI_TYPE_VIDEO_BUFFER, \ GstVaapiVideoBuffer)) #define GST_VAAPI_VIDEO_BUFFER_CLASS(klass) \ (G_TYPE_CHECK_CLASS_CAST((klass), \ GST_VAAPI_TYPE_VIDEO_BUFFER, \ GstVaapiVideoBufferClass)) #define GST_VAAPI_IS_VIDEO_BUFFER(obj) \ (G_TYPE_CHECK_INSTANCE_TYPE((obj), GST_VAAPI_TYPE_VIDEO_BUFFER)) #define GST_VAAPI_IS_VIDEO_BUFFER_CLASS(klass) \ (G_TYPE_CHECK_CLASS_TYPE((klass), GST_VAAPI_TYPE_VIDEO_BUFFER)) #define GST_VAAPI_VIDEO_BUFFER_GET_CLASS(obj) \ (G_TYPE_INSTANCE_GET_CLASS((obj), \ GST_VAAPI_TYPE_VIDEO_BUFFER, \ GstVaapiVideoBufferClass)) typedef struct _GstVaapiVideoBufferClass GstVaapiVideoBufferClass; /** * GstVaapiVideoBuffer: * * A #GstBuffer holding video objects (#GstVaapiSurface and #GstVaapiImage). */ struct _GstVaapiVideoBuffer { /*< private >*/ GstSurfaceBuffer parent_instance; }; /** * GstVaapiVideoBufferClass: * * A #GstBuffer holding video objects */ struct _GstVaapiVideoBufferClass { /*< private >*/ GstSurfaceBufferClass parent_class; }; GType gst_vaapi_video_buffer_get_type(void) G_GNUC_CONST; G_DEFINE_TYPE(GstVaapiVideoBuffer, gst_vaapi_video_buffer, GST_TYPE_SURFACE_BUFFER) typedef GstSurfaceConverter *(*GstSurfaceConverterCreateFunc)( GstSurfaceBuffer *surface, const gchar *type, GValue *dest); static GstSurfaceConverter * gst_vaapi_video_buffer_create_converter(GstSurfaceBuffer *surface, const gchar *type, GValue *dest) { GstVaapiVideoMeta * const meta = gst_buffer_get_vaapi_video_meta(GST_BUFFER(surface)); GstSurfaceConverterCreateFunc func; g_return_val_if_fail(meta != NULL, NULL); func = (GstSurfaceConverterCreateFunc) gst_vaapi_video_meta_get_surface_converter(meta); return func ? func(surface, type, dest) : NULL; } static void gst_vaapi_video_buffer_class_init(GstVaapiVideoBufferClass *klass) { GstSurfaceBufferClass * const surface_class = GST_SURFACE_BUFFER_CLASS(klass); surface_class->create_converter = gst_vaapi_video_buffer_create_converter; } static void gst_vaapi_video_buffer_init(GstVaapiVideoBuffer *buffer) { } static inline GstBuffer * gst_surface_buffer_new(void) { return GST_BUFFER_CAST(gst_mini_object_new(GST_VAAPI_TYPE_VIDEO_BUFFER)); } #endif static GFunc get_surface_converter(GstVaapiDisplay *display) { GFunc func; switch (gst_vaapi_display_get_display_type(display)) { #if USE_X11 && !GST_CHECK_VERSION(1,1,0) case GST_VAAPI_DISPLAY_TYPE_X11: func = (GFunc)gst_vaapi_video_converter_x11_new; break; #endif #if USE_GLX && !GST_CHECK_VERSION(1,1,0) case GST_VAAPI_DISPLAY_TYPE_GLX: func = (GFunc)gst_vaapi_video_converter_glx_new; break; #endif default: func = NULL; break; } return func; } static GstBuffer * new_vbuffer(GstVaapiVideoMeta *meta) { GstBuffer *buffer; g_return_val_if_fail(meta != NULL, NULL); gst_vaapi_video_meta_set_surface_converter(meta, get_surface_converter(gst_vaapi_video_meta_get_display(meta))); buffer = gst_surface_buffer_new(); if (buffer) gst_buffer_set_vaapi_video_meta(buffer, meta); gst_vaapi_video_meta_unref(meta); return buffer; } GstBuffer * gst_vaapi_video_buffer_new(GstVaapiVideoMeta *meta) { g_return_val_if_fail(meta != NULL, NULL); return new_vbuffer(gst_vaapi_video_meta_ref(meta)); } GstBuffer * gst_vaapi_video_buffer_new_empty(void) { return gst_surface_buffer_new(); } GstBuffer * gst_vaapi_video_buffer_new_from_pool(GstVaapiVideoPool *pool) { return new_vbuffer(gst_vaapi_video_meta_new_from_pool(pool)); } GstBuffer * gst_vaapi_video_buffer_new_from_buffer(GstBuffer *buffer) { GstVaapiVideoMeta * const meta = gst_buffer_get_vaapi_video_meta(buffer); return meta ? new_vbuffer(gst_vaapi_video_meta_ref(meta)) : NULL; } GstBuffer * gst_vaapi_video_buffer_new_with_image(GstVaapiImage *image) { return new_vbuffer(gst_vaapi_video_meta_new_with_image(image)); } GstBuffer * gst_vaapi_video_buffer_new_with_surface(GstVaapiSurface *surface) { return new_vbuffer(gst_vaapi_video_meta_new_with_surface(surface)); } GstBuffer * gst_vaapi_video_buffer_new_with_surface_proxy(GstVaapiSurfaceProxy *proxy) { return new_vbuffer(gst_vaapi_video_meta_new_with_surface_proxy(proxy)); }