mirror of
https://gitlab.freedesktop.org/gstreamer/gstreamer.git
synced 2024-12-14 20:36:32 +00:00
990fbb3b52
In order to other plugins use gstva objects, such as allocators and buffer pools, this merge request move them from the va plugin to the gstva library. This objects are not exposed in <gst/va/gstva.h> since they are not expected to be used by users, only by plugin implementators. Because of the surface copy design, which is used to implement allocator's mem_copy() virtual function, depends on the vafilter, which is kept inside the plugin, memory copy through VAPosproc is disabled and removed temporarly. Also added some missing parameter validation. Part-of: <https://gitlab.freedesktop.org/gstreamer/gstreamer/-/merge_requests/2048>
120 lines
3 KiB
C
120 lines
3 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 "gstvasurfacecopy.h"
|
|
#include "vasurfaceimage.h"
|
|
|
|
#define GST_CAT_DEFAULT gst_va_memory_debug
|
|
GST_DEBUG_CATEGORY_EXTERN (gst_va_memory_debug);
|
|
|
|
struct _GstVaSurfaceCopy
|
|
{
|
|
GstVaDisplay *display;
|
|
|
|
GstVideoInfo info;
|
|
gboolean has_copy;
|
|
|
|
GRecMutex lock;
|
|
};
|
|
|
|
static gboolean
|
|
_has_copy (GstVaDisplay * display)
|
|
{
|
|
#if VA_CHECK_VERSION (1, 12, 0)
|
|
VADisplay dpy;
|
|
VADisplayAttribute attr = {
|
|
.type = VADisplayAttribCopy,
|
|
.flags = VA_DISPLAY_ATTRIB_GETTABLE,
|
|
};
|
|
VAStatus status;
|
|
|
|
dpy = gst_va_display_get_va_dpy (display);
|
|
|
|
status = vaGetDisplayAttributes (dpy, &attr, 1);
|
|
if (status != VA_STATUS_SUCCESS) {
|
|
GST_INFO ("vaGetDisplayAttribures: %s", vaErrorStr (status));
|
|
return FALSE;
|
|
}
|
|
|
|
return TRUE;
|
|
#else
|
|
return FALSE;
|
|
#endif
|
|
}
|
|
|
|
GstVaSurfaceCopy *
|
|
gst_va_surface_copy_new (GstVaDisplay * display, GstVideoInfo * vinfo)
|
|
{
|
|
GstVaSurfaceCopy *self;
|
|
|
|
g_return_val_if_fail (GST_IS_VA_DISPLAY (display), NULL);
|
|
g_return_val_if_fail (vinfo != NULL, NULL);
|
|
|
|
self = g_slice_new (GstVaSurfaceCopy);
|
|
self->display = gst_object_ref (display);
|
|
self->has_copy = _has_copy (display);
|
|
self->info = *vinfo;
|
|
g_rec_mutex_init (&self->lock);
|
|
|
|
return self;
|
|
}
|
|
|
|
void
|
|
gst_va_surface_copy_free (GstVaSurfaceCopy * self)
|
|
{
|
|
g_return_if_fail (self && GST_IS_VA_DISPLAY (self->display));
|
|
|
|
gst_clear_object (&self->display);
|
|
|
|
g_rec_mutex_clear (&self->lock);
|
|
|
|
g_slice_free (GstVaSurfaceCopy, self);
|
|
}
|
|
|
|
gboolean
|
|
gst_va_surface_copy (GstVaSurfaceCopy * self, VASurfaceID dst, VASurfaceID src)
|
|
{
|
|
VAImage image = {.image_id = VA_INVALID_ID, };
|
|
gboolean ret;
|
|
|
|
g_return_val_if_fail (self && GST_IS_VA_DISPLAY (self->display), FALSE);
|
|
|
|
if (self->has_copy && va_copy_surface (self->display, dst, src)) {
|
|
GST_LOG ("GPU copy of %#x to %#x", src, dst);
|
|
return TRUE;
|
|
}
|
|
|
|
/* TODO: Add the VPP copy. */
|
|
|
|
if (!va_ensure_image (self->display, src, &self->info, &image, FALSE))
|
|
return FALSE;
|
|
|
|
if ((ret = va_put_image (self->display, dst, &image)))
|
|
GST_LOG ("shallow copy of %#x to %#x", src, dst);
|
|
|
|
va_unmap_buffer (self->display, image.buf);
|
|
va_destroy_image (self->display, image.image_id);
|
|
|
|
return ret;
|
|
}
|