mirror of
https://gitlab.freedesktop.org/gstreamer/gstreamer.git
synced 2024-11-11 11:51:34 +00:00
293 lines
7.8 KiB
C
293 lines
7.8 KiB
C
|
/* GStreamer
|
||
|
* Copyright (C) 2021 Igalia, S.L.
|
||
|
* Author: Víctor Jáquez <vjaquez@igalia.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 the
|
||
|
* Free Software Foundation, Inc., 51 Franklin St, Fifth Floor,
|
||
|
* Boston, MA 02110-1301, USA.
|
||
|
*/
|
||
|
|
||
|
#include "vasurfaceimage.h"
|
||
|
|
||
|
#include "gstvavideoformat.h"
|
||
|
|
||
|
gboolean
|
||
|
va_destroy_surfaces (GstVaDisplay * display, VASurfaceID * surfaces,
|
||
|
gint num_surfaces)
|
||
|
{
|
||
|
VADisplay dpy = gst_va_display_get_va_dpy (display);
|
||
|
VAStatus status;
|
||
|
|
||
|
g_return_val_if_fail (num_surfaces > 0, FALSE);
|
||
|
|
||
|
gst_va_display_lock (display);
|
||
|
status = vaDestroySurfaces (dpy, surfaces, num_surfaces);
|
||
|
gst_va_display_unlock (display);
|
||
|
if (status != VA_STATUS_SUCCESS) {
|
||
|
GST_ERROR ("vaDestroySurfaces: %s", vaErrorStr (status));
|
||
|
return FALSE;
|
||
|
}
|
||
|
|
||
|
return TRUE;
|
||
|
|
||
|
}
|
||
|
|
||
|
gboolean
|
||
|
va_create_surfaces (GstVaDisplay * display, guint rt_format, guint fourcc,
|
||
|
guint width, guint height, gint usage_hint,
|
||
|
VASurfaceAttribExternalBuffers * ext_buf, VASurfaceID * surfaces,
|
||
|
guint num_surfaces)
|
||
|
{
|
||
|
VADisplay dpy = gst_va_display_get_va_dpy (display);
|
||
|
/* *INDENT-OFF* */
|
||
|
VASurfaceAttrib attrs[5] = {
|
||
|
{
|
||
|
.type = VASurfaceAttribUsageHint,
|
||
|
.flags = VA_SURFACE_ATTRIB_SETTABLE,
|
||
|
.value.type = VAGenericValueTypeInteger,
|
||
|
.value.value.i = usage_hint,
|
||
|
},
|
||
|
{
|
||
|
.type = VASurfaceAttribMemoryType,
|
||
|
.flags = VA_SURFACE_ATTRIB_SETTABLE,
|
||
|
.value.type = VAGenericValueTypeInteger,
|
||
|
.value.value.i = (ext_buf && ext_buf->num_buffers > 0)
|
||
|
? VA_SURFACE_ATTRIB_MEM_TYPE_DRM_PRIME
|
||
|
: VA_SURFACE_ATTRIB_MEM_TYPE_VA,
|
||
|
},
|
||
|
};
|
||
|
/* *INDENT-ON* */
|
||
|
VAStatus status;
|
||
|
guint num_attrs = 2;
|
||
|
|
||
|
g_return_val_if_fail (num_surfaces > 0, FALSE);
|
||
|
|
||
|
if (fourcc > 0) {
|
||
|
/* *INDENT-OFF* */
|
||
|
attrs[num_attrs++] = (VASurfaceAttrib) {
|
||
|
.type = VASurfaceAttribPixelFormat,
|
||
|
.flags = VA_SURFACE_ATTRIB_SETTABLE,
|
||
|
.value.type = VAGenericValueTypeInteger,
|
||
|
.value.value.i = fourcc,
|
||
|
};
|
||
|
/* *INDENT-ON* */
|
||
|
}
|
||
|
|
||
|
if (ext_buf) {
|
||
|
/* *INDENT-OFF* */
|
||
|
attrs[num_attrs++] = (VASurfaceAttrib) {
|
||
|
.type = VASurfaceAttribExternalBufferDescriptor,
|
||
|
.flags = VA_SURFACE_ATTRIB_SETTABLE,
|
||
|
.value.type = VAGenericValueTypePointer,
|
||
|
.value.value.p = ext_buf,
|
||
|
};
|
||
|
/* *INDENT-ON* */
|
||
|
}
|
||
|
|
||
|
gst_va_display_lock (display);
|
||
|
status = vaCreateSurfaces (dpy, rt_format, width, height, surfaces,
|
||
|
num_surfaces, attrs, num_attrs);
|
||
|
gst_va_display_unlock (display);
|
||
|
if (status != VA_STATUS_SUCCESS) {
|
||
|
GST_ERROR ("vaCreateSurfaces: %s", vaErrorStr (status));
|
||
|
return FALSE;
|
||
|
}
|
||
|
|
||
|
return TRUE;
|
||
|
}
|
||
|
|
||
|
gboolean
|
||
|
va_export_surface_to_dmabuf (GstVaDisplay * display, VASurfaceID surface,
|
||
|
guint32 flags, VADRMPRIMESurfaceDescriptor * desc)
|
||
|
{
|
||
|
VADisplay dpy = gst_va_display_get_va_dpy (display);
|
||
|
VAStatus status;
|
||
|
|
||
|
gst_va_display_lock (display);
|
||
|
status = vaExportSurfaceHandle (dpy, surface,
|
||
|
VA_SURFACE_ATTRIB_MEM_TYPE_DRM_PRIME_2, flags, desc);
|
||
|
gst_va_display_unlock (display);
|
||
|
if (status != VA_STATUS_SUCCESS) {
|
||
|
GST_ERROR ("vaExportSurfaceHandle: %s", vaErrorStr (status));
|
||
|
return FALSE;
|
||
|
}
|
||
|
|
||
|
return TRUE;
|
||
|
}
|
||
|
|
||
|
gboolean
|
||
|
va_destroy_image (GstVaDisplay * display, VAImageID image_id)
|
||
|
{
|
||
|
VADisplay dpy = gst_va_display_get_va_dpy (display);
|
||
|
VAStatus status;
|
||
|
|
||
|
gst_va_display_lock (display);
|
||
|
status = vaDestroyImage (dpy, image_id);
|
||
|
gst_va_display_unlock (display);
|
||
|
if (status != VA_STATUS_SUCCESS) {
|
||
|
GST_ERROR ("vaDestroyImage: %s", vaErrorStr (status));
|
||
|
return FALSE;
|
||
|
}
|
||
|
return TRUE;
|
||
|
}
|
||
|
|
||
|
gboolean
|
||
|
va_get_derive_image (GstVaDisplay * display, VASurfaceID surface,
|
||
|
VAImage * image)
|
||
|
{
|
||
|
VADisplay dpy = gst_va_display_get_va_dpy (display);
|
||
|
VAStatus status;
|
||
|
|
||
|
gst_va_display_lock (display);
|
||
|
status = vaDeriveImage (dpy, surface, image);
|
||
|
gst_va_display_unlock (display);
|
||
|
if (status != VA_STATUS_SUCCESS) {
|
||
|
GST_WARNING ("vaDeriveImage: %s", vaErrorStr (status));
|
||
|
return FALSE;
|
||
|
}
|
||
|
|
||
|
return TRUE;
|
||
|
}
|
||
|
|
||
|
gboolean
|
||
|
va_create_image (GstVaDisplay * display, GstVideoFormat format, gint width,
|
||
|
gint height, VAImage * image)
|
||
|
{
|
||
|
VADisplay dpy = gst_va_display_get_va_dpy (display);
|
||
|
const VAImageFormat *va_format;
|
||
|
VAStatus status;
|
||
|
|
||
|
va_format = gst_va_image_format_from_video_format (format);
|
||
|
if (!va_format)
|
||
|
return FALSE;
|
||
|
|
||
|
gst_va_display_lock (display);
|
||
|
status =
|
||
|
vaCreateImage (dpy, (VAImageFormat *) va_format, width, height, image);
|
||
|
gst_va_display_unlock (display);
|
||
|
if (status != VA_STATUS_SUCCESS) {
|
||
|
GST_ERROR ("vaCreateImage: %s", vaErrorStr (status));
|
||
|
return FALSE;
|
||
|
}
|
||
|
return TRUE;
|
||
|
}
|
||
|
|
||
|
gboolean
|
||
|
va_get_image (GstVaDisplay * display, VASurfaceID surface, VAImage * image)
|
||
|
{
|
||
|
VADisplay dpy = gst_va_display_get_va_dpy (display);
|
||
|
VAStatus status;
|
||
|
|
||
|
gst_va_display_lock (display);
|
||
|
status = vaGetImage (dpy, surface, 0, 0, image->width, image->height,
|
||
|
image->image_id);
|
||
|
gst_va_display_unlock (display);
|
||
|
if (status != VA_STATUS_SUCCESS) {
|
||
|
GST_ERROR ("vaGetImage: %s", vaErrorStr (status));
|
||
|
return FALSE;
|
||
|
}
|
||
|
|
||
|
return TRUE;
|
||
|
}
|
||
|
|
||
|
gboolean
|
||
|
va_sync_surface (GstVaDisplay * display, VASurfaceID surface)
|
||
|
{
|
||
|
VADisplay dpy = gst_va_display_get_va_dpy (display);
|
||
|
VAStatus status;
|
||
|
|
||
|
gst_va_display_lock (display);
|
||
|
status = vaSyncSurface (dpy, surface);
|
||
|
gst_va_display_unlock (display);
|
||
|
if (status != VA_STATUS_SUCCESS) {
|
||
|
GST_WARNING ("vaSyncSurface: %s", vaErrorStr (status));
|
||
|
return FALSE;
|
||
|
}
|
||
|
return TRUE;
|
||
|
}
|
||
|
|
||
|
gboolean
|
||
|
va_map_buffer (GstVaDisplay * display, VABufferID buffer, gpointer * data)
|
||
|
{
|
||
|
VADisplay dpy = gst_va_display_get_va_dpy (display);
|
||
|
VAStatus status;
|
||
|
|
||
|
gst_va_display_lock (display);
|
||
|
status = vaMapBuffer (dpy, buffer, data);
|
||
|
gst_va_display_unlock (display);
|
||
|
if (status != VA_STATUS_SUCCESS) {
|
||
|
GST_WARNING ("vaMapBuffer: %s", vaErrorStr (status));
|
||
|
return FALSE;
|
||
|
}
|
||
|
return TRUE;
|
||
|
}
|
||
|
|
||
|
gboolean
|
||
|
va_unmap_buffer (GstVaDisplay * display, VABufferID buffer)
|
||
|
{
|
||
|
VADisplay dpy = gst_va_display_get_va_dpy (display);
|
||
|
VAStatus status;
|
||
|
|
||
|
gst_va_display_lock (display);
|
||
|
status = vaUnmapBuffer (dpy, buffer);
|
||
|
gst_va_display_unlock (display);
|
||
|
if (status != VA_STATUS_SUCCESS) {
|
||
|
GST_WARNING ("vaUnmapBuffer: %s", vaErrorStr (status));
|
||
|
return FALSE;
|
||
|
}
|
||
|
return TRUE;
|
||
|
}
|
||
|
|
||
|
gboolean
|
||
|
va_put_image (GstVaDisplay * display, VASurfaceID surface, VAImage * image)
|
||
|
{
|
||
|
VADisplay dpy = gst_va_display_get_va_dpy (display);
|
||
|
VAStatus status;
|
||
|
|
||
|
if (!va_sync_surface (display, surface))
|
||
|
return FALSE;
|
||
|
|
||
|
gst_va_display_lock (display);
|
||
|
status = vaPutImage (dpy, surface, image->image_id, 0, 0, image->width,
|
||
|
image->height, 0, 0, image->width, image->height);
|
||
|
gst_va_display_unlock (display);
|
||
|
if (status != VA_STATUS_SUCCESS) {
|
||
|
GST_ERROR ("vaPutImage: %s", vaErrorStr (status));
|
||
|
return FALSE;
|
||
|
}
|
||
|
return TRUE;
|
||
|
}
|
||
|
|
||
|
gboolean
|
||
|
va_ensure_image (GstVaDisplay * display, VASurfaceID surface,
|
||
|
GstVideoInfo * info, VAImage * image, gboolean derived)
|
||
|
{
|
||
|
gboolean ret = TRUE;
|
||
|
|
||
|
if (image->image_id != VA_INVALID_ID)
|
||
|
return TRUE;
|
||
|
|
||
|
if (!va_sync_surface (display, surface))
|
||
|
return FALSE;
|
||
|
|
||
|
if (derived) {
|
||
|
ret = va_get_derive_image (display, surface, image);
|
||
|
} else {
|
||
|
ret = va_create_image (display, GST_VIDEO_INFO_FORMAT (info),
|
||
|
GST_VIDEO_INFO_WIDTH (info), GST_VIDEO_INFO_HEIGHT (info), image);
|
||
|
}
|
||
|
|
||
|
return ret;
|
||
|
}
|