mirror of
https://gitlab.freedesktop.org/gstreamer/gstreamer.git
synced 2024-12-01 06:01:04 +00:00
a9fe79610a
Add helpers to import EGLImage objects into VA surfaces. There are two operational modes: (i) gst_vaapi_surface_new_from_egl_image(), which allows for implicit conversion from EGLImage to a VA surface in native video format, and (ii) gst_vaapi_surface_new_with_egl_image(), which exactly wraps the source EGLImage, typically in RGBA format with linear storage. Note: in case of (i), the EGLImage can be disposed right after the VA surface creation call, unlike in (ii) where the user shall ensure that the EGLImage is live until the associated VA surface is no longer needed. https://bugzilla.gnome.org/show_bug.cgi?id=743847
255 lines
8.7 KiB
C
255 lines
8.7 KiB
C
/*
|
|
* gstvaapisurface_egl.c - VA surface abstraction (EGL interop)
|
|
*
|
|
* Copyright (C) 2014 Intel Corporation
|
|
* Author: Gwenole Beauchesne <gwenole.beauchesne@intel.com>
|
|
*
|
|
* 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 "sysdeps.h"
|
|
#include "gstvaapisurface_egl.h"
|
|
#include "gstvaapisurface_drm.h"
|
|
#include "gstvaapisurface_priv.h"
|
|
#include "gstvaapiimage_priv.h"
|
|
#include "gstvaapibufferproxy_priv.h"
|
|
#include "gstvaapidisplay_egl_priv.h"
|
|
#include "gstvaapifilter.h"
|
|
|
|
typedef struct
|
|
{
|
|
GstVaapiDisplayEGL *display;
|
|
EGLImageKHR image;
|
|
GstVideoFormat format;
|
|
guint width;
|
|
guint height;
|
|
GstVaapiSurface *surface; /* result */
|
|
} CreateSurfaceWithEGLImageArgs;
|
|
|
|
static GstVaapiSurface *
|
|
do_create_surface_with_egl_image_unlocked (GstVaapiDisplayEGL * display,
|
|
EGLImageKHR image, GstVideoFormat format, guint width, guint height)
|
|
{
|
|
GstVaapiDisplay *const base_display = GST_VAAPI_DISPLAY_CAST (display);
|
|
EglContext *const ctx = GST_VAAPI_DISPLAY_EGL_CONTEXT (display);
|
|
EglVTable *vtable;
|
|
gsize size, offset[GST_VIDEO_MAX_PLANES];
|
|
gint name, stride[GST_VIDEO_MAX_PLANES];
|
|
|
|
if (!ctx || !(vtable = egl_context_get_vtable (ctx, FALSE)))
|
|
return NULL;
|
|
|
|
memset (offset, 0, sizeof (offset));
|
|
memset (stride, 0, sizeof (stride));
|
|
|
|
if (vtable->has_EGL_MESA_drm_image) {
|
|
/* EGL_MESA_drm_image extension */
|
|
if (!vtable->eglExportDRMImageMESA (ctx->display->base.handle.p, image,
|
|
&name, NULL, &stride[0]))
|
|
goto error_export_image_gem_buf;
|
|
|
|
size = height * stride[0];
|
|
return gst_vaapi_surface_new_with_gem_buf_handle (base_display, name, size,
|
|
format, width, height, offset, stride);
|
|
}
|
|
return NULL;
|
|
|
|
/* ERRORS */
|
|
error_export_image_gem_buf:
|
|
GST_ERROR ("failed to export EGL image to GEM buffer");
|
|
return NULL;
|
|
}
|
|
|
|
static void
|
|
do_create_surface_with_egl_image (CreateSurfaceWithEGLImageArgs * args)
|
|
{
|
|
GST_VAAPI_DISPLAY_LOCK (args->display);
|
|
args->surface = do_create_surface_with_egl_image_unlocked (args->display,
|
|
args->image, args->format, args->width, args->height);
|
|
GST_VAAPI_DISPLAY_UNLOCK (args->display);
|
|
}
|
|
|
|
// Creates VA surface with EGLImage buffer as backing storage (internal)
|
|
static inline GstVaapiSurface *
|
|
create_surface_with_egl_image (GstVaapiDisplayEGL * display, EGLImageKHR image,
|
|
GstVideoFormat format, guint width, guint height)
|
|
{
|
|
CreateSurfaceWithEGLImageArgs args =
|
|
{ display, image, format, width, height };
|
|
|
|
if (!egl_context_run (GST_VAAPI_DISPLAY_EGL_CONTEXT (display),
|
|
(EglContextRunFunc) do_create_surface_with_egl_image, &args))
|
|
return NULL;
|
|
return args.surface;
|
|
}
|
|
|
|
// Creates VA surface from an EGLImage buffer copy (internal)
|
|
static GstVaapiSurface *
|
|
create_surface_from_egl_image (GstVaapiDisplayEGL * display,
|
|
const GstVideoInfo * vip, EGLImageKHR image, GstVideoFormat format,
|
|
guint width, guint height, guint flags)
|
|
{
|
|
GstVaapiDisplay *const base_display = GST_VAAPI_DISPLAY_CAST (display);
|
|
GstVaapiSurface *img_surface = NULL, *out_surface = NULL;
|
|
gboolean use_native_format = TRUE;
|
|
GstVaapiFilter *filter = NULL;
|
|
GstVaapiFilterStatus filter_status;
|
|
|
|
img_surface = create_surface_with_egl_image (display, image, format,
|
|
width, height);
|
|
if (!img_surface)
|
|
return NULL;
|
|
|
|
if (vip) {
|
|
use_native_format =
|
|
GST_VIDEO_INFO_FORMAT (vip) == GST_VIDEO_FORMAT_ENCODED ||
|
|
GST_VIDEO_INFO_FORMAT (vip) == GST_VIDEO_FORMAT_UNKNOWN;
|
|
|
|
if (GST_VIDEO_INFO_WIDTH (vip) && GST_VIDEO_INFO_HEIGHT (vip)) {
|
|
width = GST_VIDEO_INFO_WIDTH (vip);
|
|
height = GST_VIDEO_INFO_HEIGHT (vip);
|
|
}
|
|
}
|
|
|
|
if (use_native_format) {
|
|
out_surface = gst_vaapi_surface_new (base_display,
|
|
GST_VAAPI_CHROMA_TYPE_YUV420, width, height);
|
|
} else {
|
|
out_surface = gst_vaapi_surface_new_with_format (base_display,
|
|
GST_VIDEO_INFO_FORMAT (vip), width, height);
|
|
}
|
|
if (!out_surface)
|
|
goto error_create_surface;
|
|
|
|
filter = gst_vaapi_filter_new (base_display);
|
|
if (!filter)
|
|
goto error_create_filter;
|
|
|
|
filter_status = gst_vaapi_filter_process (filter,
|
|
img_surface, out_surface, flags);
|
|
if (filter_status != GST_VAAPI_FILTER_STATUS_SUCCESS)
|
|
goto error_convert_surface;
|
|
|
|
gst_vaapi_object_unref (img_surface);
|
|
gst_vaapi_filter_unref (filter);
|
|
return out_surface;
|
|
|
|
/* ERRORS */
|
|
error_create_surface:
|
|
GST_ERROR ("failed to create output surface format:%s size:%dx%d",
|
|
gst_vaapi_video_format_to_string (vip ? GST_VIDEO_INFO_FORMAT (vip) :
|
|
GST_VIDEO_FORMAT_ENCODED), width, height);
|
|
goto error_cleanup;
|
|
error_convert_surface:
|
|
GST_ERROR ("failed to transfer EGL image to VA surface (status = %d)",
|
|
filter_status);
|
|
goto error_cleanup;
|
|
error_create_filter:
|
|
GST_ERROR ("failed to create video processing filter");
|
|
// fall-through
|
|
error_cleanup:
|
|
gst_vaapi_object_replace (&img_surface, NULL);
|
|
gst_vaapi_object_replace (&out_surface, NULL);
|
|
gst_vaapi_filter_replace (&filter, NULL);
|
|
return NULL;
|
|
}
|
|
|
|
/**
|
|
* gst_vaapi_surface_new_from_egl_image:
|
|
* @display: a #GstVaapiDisplay
|
|
* @vip: the desired (optional) #GstVideoInfo constraints
|
|
* @image: the EGL image to import
|
|
* @format: the EGL @image format
|
|
* @width: the EGL @image width in pixels
|
|
* @height: the EGL @image height in pixels
|
|
* @flags: postprocessing flags. See #GstVaapiSurfaceRenderFlags
|
|
*
|
|
* Creates a new #GstVaapiSurface with a copy of the EGL image
|
|
* contents. i.e. the input EGL @image can be disposed and the
|
|
* resulting VA surface would still be valid with the contents at the
|
|
* time this function was called.
|
|
*
|
|
* If @vip is %NULL, then the resulting VA surface will be created
|
|
* with the same video format and size as the original @image. If @vip
|
|
* is non-%NULL and the desired format is GST_VIDEO_FORMAT_ENCODED,
|
|
* then the resulting VA surface will have the best "native" HW
|
|
* format, usually NV12.
|
|
*
|
|
* Return value: the newly allocated #GstVaapiSurface object, or %NULL
|
|
* if creation from EGL image failed, or is not supported
|
|
*/
|
|
GstVaapiSurface *
|
|
gst_vaapi_surface_new_from_egl_image (GstVaapiDisplay * base_display,
|
|
const GstVideoInfo * vip, EGLImageKHR image, GstVideoFormat format,
|
|
guint width, guint height, guint flags)
|
|
{
|
|
GstVaapiDisplayEGL *display;
|
|
|
|
g_return_val_if_fail (GST_VAAPI_IS_DISPLAY_EGL (base_display), NULL);
|
|
g_return_val_if_fail (image != EGL_NO_IMAGE_KHR, NULL);
|
|
g_return_val_if_fail (width > 0, NULL);
|
|
g_return_val_if_fail (height > 0, NULL);
|
|
|
|
display = GST_VAAPI_DISPLAY_EGL (base_display);
|
|
if (!display || !GST_VAAPI_IS_DISPLAY_EGL (display))
|
|
goto error_invalid_display;
|
|
return create_surface_from_egl_image (display, vip, image, format,
|
|
width, height, flags);
|
|
|
|
/* ERRORS */
|
|
error_invalid_display:
|
|
GST_ERROR ("invalid display (NULL or not of EGL class");
|
|
return NULL;
|
|
}
|
|
|
|
/**
|
|
* gst_vaapi_surface_new_with_egl_image:
|
|
* @display: a #GstVaapiDisplay
|
|
* @image: the EGL image to import
|
|
* @format: the EGL @image format
|
|
* @width: the EGL @image width in pixels
|
|
* @height: the EGL @image height in pixels
|
|
*
|
|
* Creates a new #GstVaapiSurface bound to an external EGL image.
|
|
*
|
|
* The caller maintains the lifetime of the EGL image object. In
|
|
* particular, the EGL image shall not be destroyed before the last
|
|
* reference to the resulting VA surface is released.
|
|
*
|
|
* Return value: the newly allocated #GstVaapiSurface object, or %NULL
|
|
* if creation from EGL image failed, or is not supported
|
|
*/
|
|
GstVaapiSurface *
|
|
gst_vaapi_surface_new_with_egl_image (GstVaapiDisplay * base_display,
|
|
EGLImageKHR image, GstVideoFormat format, guint width, guint height)
|
|
{
|
|
GstVaapiDisplayEGL *display;
|
|
|
|
g_return_val_if_fail (GST_VAAPI_IS_DISPLAY_EGL (base_display), NULL);
|
|
g_return_val_if_fail (image != EGL_NO_IMAGE_KHR, NULL);
|
|
g_return_val_if_fail (width > 0, NULL);
|
|
g_return_val_if_fail (height > 0, NULL);
|
|
|
|
display = GST_VAAPI_DISPLAY_EGL (base_display);
|
|
if (!display || !GST_VAAPI_IS_DISPLAY_EGL (display))
|
|
goto error_invalid_display;
|
|
return create_surface_with_egl_image (display, image, format, width, height);
|
|
|
|
/* ERRORS */
|
|
error_invalid_display:
|
|
GST_ERROR ("invalid display (NULL or not of EGL class");
|
|
return NULL;
|
|
}
|