mirror of
https://gitlab.freedesktop.org/gstreamer/gstreamer.git
synced 2025-01-10 09:25:42 +00:00
474 lines
13 KiB
C
474 lines
13 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.
|
|
*/
|
|
|
|
#ifdef HAVE_CONFIG_H
|
|
#include "config.h"
|
|
#endif
|
|
|
|
#include "vasurfaceimage.h"
|
|
#include "gstvavideoformat.h"
|
|
#include <va/va.h>
|
|
|
|
/* XXX: find a better log category */
|
|
#define GST_CAT_DEFAULT gst_va_display_debug
|
|
GST_DEBUG_CATEGORY_EXTERN (gst_va_display_debug);
|
|
|
|
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);
|
|
|
|
status = vaDestroySurfaces (dpy, surfaces, num_surfaces);
|
|
if (status != VA_STATUS_SUCCESS) {
|
|
GST_ERROR ("vaDestroySurfaces: %s", vaErrorStr (status));
|
|
return FALSE;
|
|
}
|
|
|
|
return TRUE;
|
|
}
|
|
|
|
static gboolean
|
|
_rt_format_is_rgb (guint rt_format)
|
|
{
|
|
switch (rt_format) {
|
|
case VA_RT_FORMAT_RGB16:
|
|
case VA_RT_FORMAT_RGB32:
|
|
case VA_RT_FORMAT_RGB32_10:
|
|
return TRUE;
|
|
default:
|
|
break;
|
|
}
|
|
|
|
return FALSE;
|
|
}
|
|
|
|
gboolean
|
|
va_create_surfaces (GstVaDisplay * display, guint rt_format, guint fourcc,
|
|
guint width, guint height, gint usage_hint, guint64 * modifiers,
|
|
guint num_modifiers, VADRMPRIMESurfaceDescriptor * desc,
|
|
VASurfaceID * surfaces, guint num_surfaces)
|
|
{
|
|
VADisplay dpy = gst_va_display_get_va_dpy (display);
|
|
/* *INDENT-OFF* */
|
|
VASurfaceAttrib attrs[6] = {
|
|
{
|
|
.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 = (desc && desc->num_objects > 0)
|
|
? VA_SURFACE_ATTRIB_MEM_TYPE_DRM_PRIME_2
|
|
: VA_SURFACE_ATTRIB_MEM_TYPE_VA,
|
|
},
|
|
};
|
|
VADRMFormatModifierList modifier_list = {
|
|
.num_modifiers = num_modifiers,
|
|
.modifiers = modifiers,
|
|
};
|
|
VASurfaceAttribExternalBuffers extbuf = {
|
|
.width = width,
|
|
.height = height,
|
|
.num_planes = 1,
|
|
.pixel_format = fourcc,
|
|
};
|
|
/* *INDENT-ON* */
|
|
VAStatus status;
|
|
guint num_attrs = 2;
|
|
|
|
g_return_val_if_fail (num_surfaces > 0, FALSE);
|
|
/* must have modifiers when num_modifiers > 0 */
|
|
g_return_val_if_fail (num_modifiers == 0 || modifiers, 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 (desc && desc->num_objects > 0) {
|
|
/* *INDENT-OFF* */
|
|
attrs[num_attrs++] = (VASurfaceAttrib) {
|
|
.type = VASurfaceAttribExternalBufferDescriptor,
|
|
.flags = VA_SURFACE_ATTRIB_SETTABLE,
|
|
.value.type = VAGenericValueTypePointer,
|
|
.value.value.p = desc,
|
|
};
|
|
/* *INDENT-ON* */
|
|
} else if (GST_VA_DISPLAY_IS_IMPLEMENTATION (display, INTEL_I965)
|
|
&& _rt_format_is_rgb (rt_format)) {
|
|
/* HACK(victor): disable tiling for i965 driver for RGB formats */
|
|
/* *INDENT-OFF* */
|
|
attrs[num_attrs++] = (VASurfaceAttrib) {
|
|
.type = VASurfaceAttribExternalBufferDescriptor,
|
|
.flags = VA_SURFACE_ATTRIB_SETTABLE,
|
|
.value.type = VAGenericValueTypePointer,
|
|
.value.value.p = &extbuf,
|
|
};
|
|
/* *INDENT-ON* */
|
|
}
|
|
|
|
if (num_modifiers > 0 && modifiers) {
|
|
/* *INDENT-OFF* */
|
|
attrs[num_attrs++] = (VASurfaceAttrib) {
|
|
.type = VASurfaceAttribDRMFormatModifiers,
|
|
.flags = VA_SURFACE_ATTRIB_SETTABLE,
|
|
.value.type = VAGenericValueTypePointer,
|
|
.value.value.p = &modifier_list,
|
|
};
|
|
/* *INDENT-ON* */
|
|
}
|
|
|
|
retry:
|
|
status = vaCreateSurfaces (dpy, rt_format, width, height, surfaces,
|
|
num_surfaces, attrs, num_attrs);
|
|
|
|
if (status == VA_STATUS_ERROR_ATTR_NOT_SUPPORTED
|
|
&& attrs[num_attrs - 1].type == VASurfaceAttribDRMFormatModifiers) {
|
|
int i;
|
|
|
|
/* if requested modifiers contain linear, let's remove the attribute and
|
|
* "hope" the driver will create linear dmabufs */
|
|
for (i = 0; i < num_modifiers; ++i) {
|
|
if (modifiers[i] == DRM_FORMAT_MOD_LINEAR) {
|
|
num_attrs--;
|
|
goto retry;
|
|
}
|
|
}
|
|
}
|
|
|
|
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;
|
|
|
|
status = vaExportSurfaceHandle (dpy, surface,
|
|
VA_SURFACE_ATTRIB_MEM_TYPE_DRM_PRIME_2, flags, desc);
|
|
if (status != VA_STATUS_SUCCESS) {
|
|
GST_INFO ("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;
|
|
|
|
status = vaDestroyImage (dpy, image_id);
|
|
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;
|
|
VASurfaceStatus state;
|
|
|
|
/* When directly accessing a surface special care must be taken to insure sync
|
|
* proper synchronization with the graphics hardware. Clients should call
|
|
* vaQuerySurfaceStatus to insure that a surface is not the target of
|
|
* concurrent rendering or currently being displayed by an overlay. */
|
|
status = vaQuerySurfaceStatus (dpy, surface, &state);
|
|
if (status != VA_STATUS_SUCCESS) {
|
|
GST_WARNING ("vaQuerySurfaceStatus: %s", vaErrorStr (status));
|
|
return FALSE;
|
|
}
|
|
if (state != VASurfaceReady) {
|
|
GST_INFO ("Surface not ready");
|
|
return FALSE;
|
|
}
|
|
|
|
status = vaDeriveImage (dpy, surface, image);
|
|
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;
|
|
|
|
status =
|
|
vaCreateImage (dpy, (VAImageFormat *) va_format, width, height, image);
|
|
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;
|
|
|
|
status = vaGetImage (dpy, surface, 0, 0, image->width, image->height,
|
|
image->image_id);
|
|
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;
|
|
|
|
status = vaSyncSurface (dpy, surface);
|
|
if (status != VA_STATUS_SUCCESS) {
|
|
GST_WARNING ("vaSyncSurface: %s", vaErrorStr (status));
|
|
return FALSE;
|
|
}
|
|
return TRUE;
|
|
}
|
|
|
|
gboolean
|
|
va_map_buffer (GstVaDisplay * display, VABufferID buffer, GstMapFlags flags,
|
|
gpointer * data)
|
|
{
|
|
VADisplay dpy = gst_va_display_get_va_dpy (display);
|
|
VAStatus status;
|
|
|
|
#if VA_CHECK_VERSION(1, 21, 0)
|
|
uint32_t vaflags = 0;
|
|
if (flags & GST_MAP_READ)
|
|
vaflags |= VA_MAPBUFFER_FLAG_READ;
|
|
if (flags & GST_MAP_WRITE)
|
|
vaflags |= VA_MAPBUFFER_FLAG_WRITE;
|
|
status = vaMapBuffer2 (dpy, buffer, data, vaflags);
|
|
#else
|
|
status = vaMapBuffer (dpy, buffer, data);
|
|
#endif
|
|
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;
|
|
|
|
status = vaUnmapBuffer (dpy, buffer);
|
|
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;
|
|
|
|
status = vaPutImage (dpy, surface, image->image_id, 0, 0, image->width,
|
|
image->height, 0, 0, image->width, image->height);
|
|
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;
|
|
}
|
|
|
|
gboolean
|
|
va_check_surface (GstVaDisplay * display, VASurfaceID surface)
|
|
{
|
|
return va_check_surface_has_status (display, surface, 0);
|
|
}
|
|
|
|
#ifndef GST_DISABLE_GST_DEBUG
|
|
static const char *surface_status_str_map[] = {
|
|
[VASurfaceRendering] = "rendering",
|
|
[VASurfaceDisplaying] = "displaying",
|
|
[VASurfaceReady] = "ready",
|
|
[VASurfaceSkipped] = "skipped"
|
|
};
|
|
#endif
|
|
|
|
gboolean
|
|
va_check_surface_has_status (GstVaDisplay * display, VASurfaceID surface,
|
|
VASurfaceStatus surface_status)
|
|
{
|
|
VADisplay dpy = gst_va_display_get_va_dpy (display);
|
|
VAStatus status;
|
|
VASurfaceStatus state;
|
|
|
|
status = vaQuerySurfaceStatus (dpy, surface, &state);
|
|
|
|
if (status != VA_STATUS_SUCCESS) {
|
|
GST_ERROR ("vaQuerySurfaceStatus: %s", vaErrorStr (status));
|
|
return FALSE;
|
|
}
|
|
|
|
GST_LOG ("surface %#x status: %s", surface, surface_status_str_map[state]);
|
|
|
|
/* Just query the surface, no flag to compare, we succeed. */
|
|
if (!surface_status)
|
|
return TRUE;
|
|
|
|
return ((state & surface_status) == surface_status);
|
|
}
|
|
|
|
gboolean
|
|
va_copy_surface (GstVaDisplay * display, VASurfaceID dst, VASurfaceID src)
|
|
{
|
|
VADisplay dpy = gst_va_display_get_va_dpy (display);
|
|
/* *INDENT-OFF* */
|
|
VACopyObject obj_src = {
|
|
.obj_type = VACopyObjectSurface,
|
|
.object = {
|
|
.surface_id = src,
|
|
},
|
|
};
|
|
VACopyObject obj_dst = {
|
|
.obj_type = VACopyObjectSurface,
|
|
.object = {
|
|
.surface_id = dst,
|
|
},
|
|
};
|
|
VACopyOption option = {
|
|
.bits = {
|
|
.va_copy_sync = VA_EXEC_SYNC,
|
|
.va_copy_mode = VA_EXEC_MODE_DEFAULT,
|
|
},
|
|
};
|
|
/* *INDENT-ON* */
|
|
VAStatus status;
|
|
|
|
status = vaCopy (dpy, &obj_dst, &obj_src, option);
|
|
if (status != VA_STATUS_SUCCESS) {
|
|
GST_INFO ("vaCopy: %s", vaErrorStr (status));
|
|
return FALSE;
|
|
}
|
|
return TRUE;
|
|
}
|
|
|
|
guint
|
|
va_get_surface_usage_hint (GstVaDisplay * display, VAEntrypoint entrypoint,
|
|
GstPadDirection dir, gboolean is_dma)
|
|
{
|
|
switch (entrypoint) {
|
|
case VAEntrypointVideoProc:{
|
|
/* For DMA kind caps, we use VA_SURFACE_ATTRIB_USAGE_HINT_VPP_READ |
|
|
VA_SURFACE_ATTRIB_USAGE_HINT_VPP_WRITE to detect the modifiers.
|
|
And in runtime, we should use the same flags in order to keep
|
|
the same modifiers. */
|
|
if (is_dma)
|
|
return VA_SURFACE_ATTRIB_USAGE_HINT_VPP_READ |
|
|
VA_SURFACE_ATTRIB_USAGE_HINT_VPP_WRITE;
|
|
|
|
if (dir == GST_PAD_SINK)
|
|
return VA_SURFACE_ATTRIB_USAGE_HINT_VPP_READ;
|
|
else if (dir == GST_PAD_SRC)
|
|
return VA_SURFACE_ATTRIB_USAGE_HINT_VPP_WRITE;
|
|
|
|
break;
|
|
}
|
|
case VAEntrypointVLD:
|
|
return VA_SURFACE_ATTRIB_USAGE_HINT_DECODER;
|
|
case VAEntrypointEncSlice:
|
|
case VAEntrypointEncSliceLP:
|
|
case VAEntrypointEncPicture:
|
|
return VA_SURFACE_ATTRIB_USAGE_HINT_ENCODER;
|
|
default:
|
|
break;
|
|
}
|
|
|
|
return VA_SURFACE_ATTRIB_USAGE_HINT_GENERIC;
|
|
}
|