mirror of
https://gitlab.freedesktop.org/gstreamer/gstreamer.git
synced 2025-01-06 07:28:48 +00:00
9bf1ae04cc
Fix ensure_image() to only zero-initialize the first line of each plane. Properly initializing each plane to their full vertical resolution would require to actually compute it based on the image format. In particular, for NV12 images, the UV plane has half vertical resolution vs. the Y plane. So using the full image height to initialize the UV plane will obviously lead to a buffer overflow. Likewise for other YUV format. Since ensure_image() is only a helper function to initialize something, and not necessarily the whole thing, it is fine to initializ the first line only. Besides, the target surface is not rendered either. Signed-off-by: Gwenole Beauchesne <gwenole.beauchesne@intel.com>
482 lines
14 KiB
C
482 lines
14 KiB
C
/*
|
|
* gstvaapiuploader.c - VA-API video upload helper
|
|
*
|
|
* 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
|
|
*/
|
|
|
|
#include "gst/vaapi/sysdeps.h"
|
|
#include <string.h>
|
|
#include <gst/video/video.h>
|
|
#include <gst/vaapi/gstvaapisurface.h>
|
|
#include <gst/vaapi/gstvaapiimagepool.h>
|
|
#include <gst/vaapi/gstvaapisurfacepool.h>
|
|
#include <gst/vaapi/gstvaapivideometa.h>
|
|
|
|
#include "gstvaapiuploader.h"
|
|
#include "gstvaapipluginbuffer.h"
|
|
|
|
#define GST_HELPER_NAME "vaapiupload"
|
|
#define GST_HELPER_DESC "VA-API video uploader"
|
|
|
|
GST_DEBUG_CATEGORY_STATIC(gst_debug_vaapi_uploader);
|
|
#define GST_CAT_DEFAULT gst_debug_vaapi_uploader
|
|
|
|
G_DEFINE_TYPE(GstVaapiUploader, gst_vaapi_uploader, G_TYPE_OBJECT)
|
|
|
|
#define GST_VAAPI_UPLOADER_CAST(obj) \
|
|
((GstVaapiUploader *)(obj))
|
|
|
|
#define GST_VAAPI_UPLOADER_GET_PRIVATE(obj) \
|
|
(G_TYPE_INSTANCE_GET_PRIVATE((obj), \
|
|
GST_VAAPI_TYPE_UPLOADER, \
|
|
GstVaapiUploaderPrivate))
|
|
|
|
struct _GstVaapiUploaderPrivate {
|
|
GstVaapiDisplay *display;
|
|
GstCaps *allowed_caps;
|
|
GstVaapiVideoPool *images;
|
|
GstCaps *image_caps;
|
|
guint image_width;
|
|
guint image_height;
|
|
GstVaapiVideoPool *surfaces;
|
|
guint surface_width;
|
|
guint surface_height;
|
|
guint direct_rendering;
|
|
};
|
|
|
|
enum {
|
|
PROP_0,
|
|
|
|
PROP_DISPLAY,
|
|
};
|
|
|
|
static void
|
|
gst_vaapi_uploader_destroy(GstVaapiUploader *uploader)
|
|
{
|
|
GstVaapiUploaderPrivate * const priv = uploader->priv;
|
|
|
|
gst_caps_replace(&priv->image_caps, NULL);
|
|
gst_caps_replace(&priv->allowed_caps, NULL);
|
|
|
|
g_clear_object(&priv->images);
|
|
g_clear_object(&priv->surfaces);
|
|
g_clear_object(&priv->display);
|
|
}
|
|
|
|
static gboolean
|
|
ensure_display(GstVaapiUploader *uploader, GstVaapiDisplay *display)
|
|
{
|
|
GstVaapiUploaderPrivate * const priv = uploader->priv;
|
|
|
|
if (priv->display == display)
|
|
return TRUE;
|
|
|
|
g_clear_object(&priv->display);
|
|
if (display)
|
|
priv->display = g_object_ref(display);
|
|
return TRUE;
|
|
}
|
|
|
|
static gboolean
|
|
ensure_image(GstVaapiImage *image)
|
|
{
|
|
guint i, num_planes, width, height;
|
|
|
|
/* Make the image fully dirty */
|
|
if (!gst_vaapi_image_map(image))
|
|
return FALSE;
|
|
|
|
gst_vaapi_image_get_size(image, &width, &height);
|
|
|
|
num_planes = gst_vaapi_image_get_plane_count(image);
|
|
for (i = 0; i < num_planes; i++) {
|
|
guchar * const plane = gst_vaapi_image_get_plane(image, i);
|
|
if (plane)
|
|
memset(plane, 0, gst_vaapi_image_get_pitch(image, i));
|
|
}
|
|
|
|
if (!gst_vaapi_image_unmap(image))
|
|
gst_vaapi_image_unmap(image);
|
|
return TRUE;
|
|
}
|
|
|
|
static gboolean
|
|
ensure_allowed_caps(GstVaapiUploader *uploader)
|
|
{
|
|
GstVaapiUploaderPrivate * const priv = uploader->priv;
|
|
GstVaapiSurface *surface = NULL;
|
|
GstCaps *out_caps, *image_caps = NULL;
|
|
guint i, n_structures;
|
|
gboolean success = FALSE;
|
|
|
|
enum { WIDTH = 64, HEIGHT = 64 };
|
|
|
|
if (priv->allowed_caps)
|
|
return TRUE;
|
|
|
|
out_caps = gst_caps_new_empty();
|
|
if (!out_caps)
|
|
return FALSE;
|
|
|
|
image_caps = gst_vaapi_display_get_image_caps(priv->display);
|
|
if (!image_caps)
|
|
goto end;
|
|
|
|
surface = gst_vaapi_surface_new(priv->display,
|
|
GST_VAAPI_CHROMA_TYPE_YUV420, WIDTH, HEIGHT);
|
|
if (!surface)
|
|
goto end;
|
|
|
|
n_structures = gst_caps_get_size(image_caps);
|
|
for (i = 0; i < n_structures; i++) {
|
|
GstStructure * const structure = gst_caps_get_structure(image_caps, i);
|
|
GstVaapiImage *image;
|
|
GstVaapiImageFormat format;
|
|
guint32 fourcc;
|
|
|
|
if (!gst_structure_get_fourcc(structure, "format", &fourcc))
|
|
continue;
|
|
format = gst_vaapi_image_format_from_fourcc(fourcc);
|
|
if (!format)
|
|
continue;
|
|
image = gst_vaapi_image_new(priv->display, format, WIDTH, HEIGHT);
|
|
if (!image)
|
|
continue;
|
|
if (ensure_image(image) && gst_vaapi_surface_put_image(surface, image))
|
|
gst_caps_append_structure(out_caps, gst_structure_copy(structure));
|
|
gst_object_unref(image);
|
|
}
|
|
|
|
gst_caps_replace(&priv->allowed_caps, out_caps);
|
|
success = TRUE;
|
|
|
|
end:
|
|
gst_caps_unref(out_caps);
|
|
if (image_caps)
|
|
gst_caps_unref(image_caps);
|
|
if (surface)
|
|
gst_object_unref(surface);
|
|
return success;
|
|
}
|
|
|
|
static gboolean
|
|
ensure_image_pool(GstVaapiUploader *uploader, GstCaps *caps)
|
|
{
|
|
GstVaapiUploaderPrivate * const priv = uploader->priv;
|
|
GstStructure * const structure = gst_caps_get_structure(caps, 0);
|
|
gint width, height;
|
|
|
|
gst_structure_get_int(structure, "width", &width);
|
|
gst_structure_get_int(structure, "height", &height);
|
|
|
|
if (width != priv->image_width || height != priv->image_height) {
|
|
priv->image_width = width;
|
|
priv->image_height = height;
|
|
g_clear_object(&priv->images);
|
|
priv->images = gst_vaapi_image_pool_new(priv->display, caps);
|
|
if (!priv->images)
|
|
return FALSE;
|
|
gst_caps_replace(&priv->image_caps, caps);
|
|
}
|
|
return TRUE;
|
|
}
|
|
|
|
static gboolean
|
|
ensure_surface_pool(GstVaapiUploader *uploader, GstCaps *caps)
|
|
{
|
|
GstVaapiUploaderPrivate * const priv = uploader->priv;
|
|
GstStructure * const structure = gst_caps_get_structure(caps, 0);
|
|
gint width, height;
|
|
|
|
gst_structure_get_int(structure, "width", &width);
|
|
gst_structure_get_int(structure, "height", &height);
|
|
|
|
if (width != priv->surface_width || height != priv->surface_height) {
|
|
priv->surface_width = width;
|
|
priv->surface_height = height;
|
|
g_clear_object(&priv->surfaces);
|
|
priv->surfaces = gst_vaapi_surface_pool_new(priv->display, caps);
|
|
if (!priv->surfaces)
|
|
return FALSE;
|
|
}
|
|
return TRUE;
|
|
}
|
|
|
|
static void
|
|
gst_vaapi_uploader_finalize(GObject *object)
|
|
{
|
|
gst_vaapi_uploader_destroy(GST_VAAPI_UPLOADER_CAST(object));
|
|
|
|
G_OBJECT_CLASS(gst_vaapi_uploader_parent_class)->finalize(object);
|
|
}
|
|
|
|
static void
|
|
gst_vaapi_uploader_set_property(GObject *object, guint prop_id,
|
|
const GValue *value, GParamSpec *pspec)
|
|
{
|
|
GstVaapiUploader * const uploader = GST_VAAPI_UPLOADER_CAST(object);
|
|
|
|
switch (prop_id) {
|
|
case PROP_DISPLAY:
|
|
ensure_display(uploader, g_value_get_object(value));
|
|
break;
|
|
default:
|
|
G_OBJECT_WARN_INVALID_PROPERTY_ID(object, prop_id, pspec);
|
|
break;
|
|
}
|
|
}
|
|
|
|
static void
|
|
gst_vaapi_uploader_get_property(GObject *object, guint prop_id,
|
|
GValue *value, GParamSpec *pspec)
|
|
{
|
|
GstVaapiUploader * const uploader = GST_VAAPI_UPLOADER_CAST(object);
|
|
|
|
switch (prop_id) {
|
|
case PROP_DISPLAY:
|
|
g_value_set_object(value, uploader->priv->display);
|
|
break;
|
|
default:
|
|
G_OBJECT_WARN_INVALID_PROPERTY_ID(object, prop_id, pspec);
|
|
break;
|
|
}
|
|
}
|
|
|
|
static void
|
|
gst_vaapi_uploader_class_init(GstVaapiUploaderClass *klass)
|
|
{
|
|
GObjectClass * const object_class = G_OBJECT_CLASS(klass);
|
|
|
|
GST_DEBUG_CATEGORY_INIT(gst_debug_vaapi_uploader,
|
|
GST_HELPER_NAME, 0, GST_HELPER_DESC);
|
|
|
|
g_type_class_add_private(klass, sizeof(GstVaapiUploaderPrivate));
|
|
|
|
object_class->finalize = gst_vaapi_uploader_finalize;
|
|
object_class->set_property = gst_vaapi_uploader_set_property;
|
|
object_class->get_property = gst_vaapi_uploader_get_property;
|
|
|
|
g_object_class_install_property(
|
|
object_class,
|
|
PROP_DISPLAY,
|
|
g_param_spec_object(
|
|
"display",
|
|
"Display",
|
|
"The GstVaapiDisplay this object is bound to",
|
|
GST_VAAPI_TYPE_DISPLAY,
|
|
G_PARAM_READWRITE));
|
|
}
|
|
|
|
static void
|
|
gst_vaapi_uploader_init(GstVaapiUploader *uploader)
|
|
{
|
|
GstVaapiUploaderPrivate *priv;
|
|
|
|
priv = GST_VAAPI_UPLOADER_GET_PRIVATE(uploader);
|
|
uploader->priv = priv;
|
|
}
|
|
|
|
GstVaapiUploader *
|
|
gst_vaapi_uploader_new(GstVaapiDisplay *display)
|
|
{
|
|
return g_object_new(GST_VAAPI_TYPE_UPLOADER, "display", display, NULL);
|
|
}
|
|
|
|
gboolean
|
|
gst_vaapi_uploader_ensure_display(
|
|
GstVaapiUploader *uploader,
|
|
GstVaapiDisplay *display
|
|
)
|
|
{
|
|
g_return_val_if_fail(GST_VAAPI_IS_UPLOADER(uploader), FALSE);
|
|
g_return_val_if_fail(GST_VAAPI_IS_DISPLAY(display), FALSE);
|
|
|
|
return ensure_display(uploader,display);
|
|
}
|
|
|
|
gboolean
|
|
gst_vaapi_uploader_ensure_caps(
|
|
GstVaapiUploader *uploader,
|
|
GstCaps *src_caps,
|
|
GstCaps *out_caps
|
|
)
|
|
{
|
|
GstVaapiUploaderPrivate *priv;
|
|
GstVaapiImage *image;
|
|
GstVaapiImageFormat vaformat;
|
|
GstVideoFormat vformat;
|
|
GstStructure *structure;
|
|
gint width, height;
|
|
|
|
g_return_val_if_fail(GST_VAAPI_IS_UPLOADER(uploader), FALSE);
|
|
g_return_val_if_fail(src_caps != NULL, FALSE);
|
|
|
|
if (!ensure_image_pool(uploader, src_caps))
|
|
return FALSE;
|
|
if (!ensure_surface_pool(uploader, out_caps ? out_caps : src_caps))
|
|
return FALSE;
|
|
|
|
priv = uploader->priv;
|
|
priv->direct_rendering = 0;
|
|
|
|
structure = gst_caps_get_structure(src_caps, 0);
|
|
if (!structure)
|
|
return FALSE;
|
|
gst_structure_get_int(structure, "width", &width);
|
|
gst_structure_get_int(structure, "height", &height);
|
|
|
|
/* Translate from Gst video format to VA image format */
|
|
if (!gst_video_format_parse_caps(src_caps, &vformat, NULL, NULL))
|
|
return FALSE;
|
|
if (!gst_video_format_is_yuv(vformat))
|
|
return FALSE;
|
|
vaformat = gst_vaapi_image_format_from_video(vformat);
|
|
if (!vaformat)
|
|
return FALSE;
|
|
|
|
/* Check if we can alias source and output buffers (same data_size) */
|
|
image = gst_vaapi_video_pool_get_object(priv->images);
|
|
if (image) {
|
|
if (gst_vaapi_image_get_format(image) == vaformat &&
|
|
gst_vaapi_image_is_linear(image) &&
|
|
(gst_vaapi_image_get_data_size(image) ==
|
|
gst_video_format_get_size(vformat, width, height)))
|
|
priv->direct_rendering = 1;
|
|
gst_vaapi_video_pool_put_object(priv->images, image);
|
|
}
|
|
return TRUE;
|
|
}
|
|
|
|
gboolean
|
|
gst_vaapi_uploader_process(
|
|
GstVaapiUploader *uploader,
|
|
GstBuffer *src_buffer,
|
|
GstBuffer *out_buffer
|
|
)
|
|
{
|
|
GstVaapiVideoMeta *src_meta, *out_meta;
|
|
GstVaapiSurface *surface;
|
|
GstVaapiImage *image;
|
|
|
|
g_return_val_if_fail(GST_VAAPI_IS_UPLOADER(uploader), FALSE);
|
|
|
|
out_meta = gst_buffer_get_vaapi_video_meta(out_buffer);
|
|
if (!out_meta) {
|
|
GST_WARNING("expected an output video buffer");
|
|
return FALSE;
|
|
}
|
|
|
|
surface = gst_vaapi_video_meta_get_surface(out_meta);
|
|
g_return_val_if_fail(surface != NULL, FALSE);
|
|
|
|
src_meta = gst_buffer_get_vaapi_video_meta(src_buffer);
|
|
if (src_meta) {
|
|
/* GstVaapiVideoBuffer with mapped VA image */
|
|
image = gst_vaapi_video_meta_get_image(src_meta);
|
|
if (!image || !gst_vaapi_image_unmap(image))
|
|
return FALSE;
|
|
}
|
|
else {
|
|
/* Regular GstBuffer that needs to be uploaded to a VA image */
|
|
image = gst_vaapi_video_meta_get_image(out_meta);
|
|
if (!image) {
|
|
image = gst_vaapi_video_pool_get_object(uploader->priv->images);
|
|
if (!image)
|
|
return FALSE;
|
|
gst_vaapi_video_meta_set_image(out_meta, image);
|
|
}
|
|
if (!gst_vaapi_image_update_from_buffer(image, src_buffer, NULL))
|
|
return FALSE;
|
|
}
|
|
g_return_val_if_fail(image != NULL, FALSE);
|
|
|
|
if (!gst_vaapi_surface_put_image(surface, image)) {
|
|
GST_WARNING("failed to upload YUV buffer to VA surface");
|
|
return FALSE;
|
|
}
|
|
|
|
/* Map again for next uploads */
|
|
if (!gst_vaapi_image_map(image))
|
|
return FALSE;
|
|
return TRUE;
|
|
}
|
|
|
|
GstCaps *
|
|
gst_vaapi_uploader_get_caps(GstVaapiUploader *uploader)
|
|
{
|
|
g_return_val_if_fail(GST_VAAPI_IS_UPLOADER(uploader), NULL);
|
|
|
|
if (!ensure_allowed_caps(uploader))
|
|
return NULL;
|
|
return uploader->priv->allowed_caps;
|
|
}
|
|
|
|
GstBuffer *
|
|
gst_vaapi_uploader_get_buffer(GstVaapiUploader *uploader)
|
|
{
|
|
GstVaapiUploaderPrivate *priv;
|
|
GstVaapiSurface *surface;
|
|
GstVaapiImage *image;
|
|
GstVaapiVideoMeta *meta;
|
|
GstBuffer *buffer;
|
|
|
|
g_return_val_if_fail(GST_VAAPI_IS_UPLOADER(uploader), NULL);
|
|
|
|
priv = uploader->priv;
|
|
|
|
buffer = gst_vaapi_video_buffer_new_from_pool(priv->images);
|
|
if (!buffer) {
|
|
GST_WARNING("failed to allocate video buffer");
|
|
goto error;
|
|
}
|
|
|
|
meta = gst_buffer_get_vaapi_video_meta(buffer);
|
|
|
|
surface = gst_vaapi_video_pool_get_object(priv->surfaces);
|
|
if (!surface) {
|
|
GST_WARNING("failed to allocate VA surface");
|
|
goto error;
|
|
}
|
|
|
|
gst_vaapi_video_meta_set_surface(meta, surface);
|
|
|
|
image = gst_vaapi_video_meta_get_image(meta);
|
|
if (!gst_vaapi_image_map(image)) {
|
|
GST_WARNING("failed to map VA image");
|
|
goto error;
|
|
}
|
|
|
|
GST_BUFFER_DATA(buffer) = gst_vaapi_image_get_plane(image, 0);
|
|
GST_BUFFER_SIZE(buffer) = gst_vaapi_image_get_data_size(image);
|
|
|
|
gst_buffer_set_caps(buffer, priv->image_caps);
|
|
return buffer;
|
|
|
|
error:
|
|
gst_buffer_unref(buffer);
|
|
return buffer;
|
|
}
|
|
|
|
gboolean
|
|
gst_vaapi_uploader_has_direct_rendering(GstVaapiUploader *uploader)
|
|
{
|
|
g_return_val_if_fail(GST_VAAPI_IS_UPLOADER(uploader), FALSE);
|
|
|
|
return uploader->priv->direct_rendering;
|
|
}
|