mirror of
https://gitlab.freedesktop.org/gstreamer/gstreamer.git
synced 2024-11-09 19:09:41 +00:00
fbbf9c629a
Since allocators keep an available memory queue to reuse, video format and usage hint are now persistant while allocator's memories are around. This patch adds _set_format() and _get_format() for both VA allocators. _set_format() validates if given format can be used or reused. If no allocated surface previously it creates a dummy one to fetch its offsets and strides. Updated info is returned to callee. GstVaPool uses _set_format() at config to verify the allocator capacity and to get the surfaces offsets and strides, which are going to be used by the video meta. Allocator extracted caps are compared with caps from config and if they have different strides or offsets, force_videometa is set. A new bufferpool method gst_va_pool_requires_video_meta() is added return the value of force_videometa. This value is checked in order to know if decoders need to copy the surface if downstream doesn't announce video meta support. Part-of: <https://gitlab.freedesktop.org/gstreamer/gst-plugins-bad/-/merge_requests/1667>
374 lines
10 KiB
C
374 lines
10 KiB
C
/* GStreamer
|
|
* Copyright (C) 2020 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 "gstvapool.h"
|
|
|
|
#include "gstvaallocator.h"
|
|
#include "gstvacaps.h"
|
|
|
|
GST_DEBUG_CATEGORY_STATIC (gst_va_pool_debug);
|
|
#define GST_CAT_DEFAULT gst_va_pool_debug
|
|
|
|
struct _GstVaPool
|
|
{
|
|
GstBufferPool parent;
|
|
|
|
GstVideoInfo alloc_info;
|
|
GstVideoInfo caps_info;
|
|
GstAllocator *allocator;
|
|
gboolean force_videometa;
|
|
gboolean add_videometa;
|
|
gboolean need_alignment;
|
|
GstVideoAlignment video_align;
|
|
|
|
gboolean starting;
|
|
};
|
|
|
|
#define gst_va_pool_parent_class parent_class
|
|
G_DEFINE_TYPE_WITH_CODE (GstVaPool, gst_va_pool, GST_TYPE_BUFFER_POOL,
|
|
GST_DEBUG_CATEGORY_INIT (gst_va_pool_debug, "vapool", 0, "VA Pool"));
|
|
|
|
static const gchar **
|
|
gst_va_pool_get_options (GstBufferPool * pool)
|
|
{
|
|
static const gchar *options[] = { GST_BUFFER_POOL_OPTION_VIDEO_META,
|
|
GST_BUFFER_POOL_OPTION_VIDEO_ALIGNMENT, NULL
|
|
};
|
|
return options;
|
|
}
|
|
|
|
static inline gboolean
|
|
gst_buffer_pool_config_get_va_allocation_params (GstStructure * config,
|
|
guint32 * usage_hint)
|
|
{
|
|
if (!gst_structure_get (config, "usage-hint", G_TYPE_UINT, usage_hint, NULL))
|
|
*usage_hint = VA_SURFACE_ATTRIB_USAGE_HINT_GENERIC;
|
|
|
|
return TRUE;
|
|
}
|
|
|
|
static gboolean
|
|
gst_va_pool_set_config (GstBufferPool * pool, GstStructure * config)
|
|
{
|
|
GstAllocator *allocator;
|
|
GstCaps *caps;
|
|
GstVaPool *vpool = GST_VA_POOL (pool);
|
|
GstVideoAlignment video_align = { 0, };
|
|
GstVideoInfo caps_info, alloc_info, orig_info;
|
|
gint width, height;
|
|
guint i, min_buffers, max_buffers;
|
|
guint32 usage_hint;
|
|
|
|
if (!gst_buffer_pool_config_get_params (config, &caps, NULL, &min_buffers,
|
|
&max_buffers))
|
|
goto wrong_config;
|
|
|
|
if (!caps)
|
|
goto no_caps;
|
|
|
|
if (!gst_video_info_from_caps (&caps_info, caps))
|
|
goto wrong_caps;
|
|
|
|
if (!gst_buffer_pool_config_get_allocator (config, &allocator, NULL))
|
|
goto wrong_config;
|
|
|
|
if (!(allocator && (GST_IS_VA_DMABUF_ALLOCATOR (allocator)
|
|
|| GST_IS_VA_ALLOCATOR (allocator))))
|
|
goto wrong_config;
|
|
|
|
if (!gst_buffer_pool_config_get_va_allocation_params (config, &usage_hint))
|
|
goto wrong_config;
|
|
|
|
orig_info = caps_info;
|
|
width = GST_VIDEO_INFO_WIDTH (&caps_info);
|
|
height = GST_VIDEO_INFO_HEIGHT (&caps_info);
|
|
|
|
GST_LOG_OBJECT (vpool, "%dx%d | %" GST_PTR_FORMAT, width, height, caps);
|
|
|
|
/* enable metadata based on config of the pool */
|
|
vpool->add_videometa = gst_buffer_pool_config_has_option (config,
|
|
GST_BUFFER_POOL_OPTION_VIDEO_META);
|
|
|
|
/* parse extra alignment info */
|
|
vpool->need_alignment = gst_buffer_pool_config_has_option (config,
|
|
GST_BUFFER_POOL_OPTION_VIDEO_ALIGNMENT);
|
|
|
|
if (vpool->need_alignment && vpool->add_videometa) {
|
|
gst_buffer_pool_config_get_video_alignment (config, &video_align);
|
|
|
|
width += video_align.padding_left + video_align.padding_right;
|
|
height += video_align.padding_bottom + video_align.padding_top;
|
|
|
|
/* apply the alignment to the info */
|
|
if (!gst_video_info_align (&caps_info, &video_align))
|
|
goto failed_to_align;
|
|
|
|
gst_buffer_pool_config_set_video_alignment (config, &video_align);
|
|
}
|
|
|
|
/* update allocation info with aligned size */
|
|
alloc_info = caps_info;
|
|
GST_VIDEO_INFO_WIDTH (&alloc_info) = width;
|
|
GST_VIDEO_INFO_HEIGHT (&alloc_info) = height;
|
|
|
|
if (GST_IS_VA_DMABUF_ALLOCATOR (allocator)) {
|
|
if (!gst_va_dmabuf_allocator_set_format (allocator, &alloc_info,
|
|
usage_hint))
|
|
goto failed_allocator;
|
|
} else if (GST_IS_VA_ALLOCATOR (allocator)) {
|
|
if (!gst_va_allocator_set_format (allocator, &alloc_info, usage_hint))
|
|
goto failed_allocator;
|
|
}
|
|
|
|
gst_object_replace ((GstObject **) & vpool->allocator,
|
|
GST_OBJECT (allocator));
|
|
|
|
vpool->caps_info = caps_info;
|
|
vpool->alloc_info = alloc_info;
|
|
vpool->video_align = video_align;
|
|
|
|
if (gst_caps_is_raw (caps)) {
|
|
for (i = 0; i < GST_VIDEO_INFO_N_PLANES (&caps_info); i++) {
|
|
if (GST_VIDEO_INFO_PLANE_STRIDE (&orig_info, i) !=
|
|
GST_VIDEO_INFO_PLANE_STRIDE (&alloc_info, i) ||
|
|
GST_VIDEO_INFO_PLANE_OFFSET (&orig_info, i) !=
|
|
GST_VIDEO_INFO_PLANE_OFFSET (&alloc_info, i)) {
|
|
GST_INFO_OBJECT (vpool, "Video meta is required in buffer.");
|
|
vpool->force_videometa = TRUE;
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
|
|
/* with pooled allocators bufferpool->release_buffer() is cheated
|
|
* because the memories are removed from the buffer at
|
|
* reset_buffer(), then buffer is an empty holder with size 0 while
|
|
* releasing. */
|
|
gst_buffer_pool_config_set_params (config, caps, 0, min_buffers, max_buffers);
|
|
|
|
return GST_BUFFER_POOL_CLASS (parent_class)->set_config (pool, config);
|
|
|
|
/* ERRORS */
|
|
wrong_config:
|
|
{
|
|
GST_WARNING_OBJECT (vpool, "invalid config");
|
|
return FALSE;
|
|
}
|
|
no_caps:
|
|
{
|
|
GST_WARNING_OBJECT (vpool, "no caps in config");
|
|
return FALSE;
|
|
}
|
|
wrong_caps:
|
|
{
|
|
GST_WARNING_OBJECT (vpool,
|
|
"failed getting geometry from caps %" GST_PTR_FORMAT, caps);
|
|
return FALSE;
|
|
}
|
|
failed_to_align:
|
|
{
|
|
GST_WARNING_OBJECT (vpool, "Failed to align");
|
|
return FALSE;
|
|
}
|
|
failed_allocator:
|
|
{
|
|
GST_WARNING_OBJECT (vpool, "Failed to set format to allocator");
|
|
return FALSE;
|
|
}
|
|
}
|
|
|
|
static GstFlowReturn
|
|
gst_va_pool_alloc (GstBufferPool * pool, GstBuffer ** buffer,
|
|
GstBufferPoolAcquireParams * params)
|
|
{
|
|
GstBuffer *buf;
|
|
GstVideoMeta *vmeta;
|
|
GstVaPool *vpool = GST_VA_POOL (pool);
|
|
|
|
buf = gst_buffer_new ();
|
|
|
|
if (GST_IS_VA_DMABUF_ALLOCATOR (vpool->allocator)) {
|
|
if (!gst_va_dmabuf_allocator_setup_buffer (vpool->allocator, buf))
|
|
goto no_memory;
|
|
} else if (GST_IS_VA_ALLOCATOR (vpool->allocator)) {
|
|
GstMemory *mem = gst_va_allocator_alloc (vpool->allocator);
|
|
if (!mem)
|
|
goto no_memory;
|
|
gst_buffer_append_memory (buf, mem);
|
|
} else
|
|
goto no_memory;
|
|
|
|
if (vpool->add_videometa) {
|
|
/* GstVaAllocator may update offset/stride given the physical
|
|
* memory */
|
|
vmeta = gst_buffer_add_video_meta_full (buf, GST_VIDEO_FRAME_FLAG_NONE,
|
|
GST_VIDEO_INFO_FORMAT (&vpool->caps_info),
|
|
GST_VIDEO_INFO_WIDTH (&vpool->caps_info),
|
|
GST_VIDEO_INFO_HEIGHT (&vpool->caps_info),
|
|
GST_VIDEO_INFO_N_PLANES (&vpool->caps_info),
|
|
vpool->alloc_info.offset, vpool->alloc_info.stride);
|
|
|
|
if (vpool->need_alignment)
|
|
gst_video_meta_set_alignment (vmeta, vpool->video_align);
|
|
}
|
|
|
|
*buffer = buf;
|
|
|
|
return GST_FLOW_OK;
|
|
|
|
/* ERROR */
|
|
no_memory:
|
|
{
|
|
gst_buffer_unref (buf);
|
|
GST_WARNING_OBJECT (vpool, "can't create memory");
|
|
return GST_FLOW_ERROR;
|
|
}
|
|
}
|
|
|
|
static void
|
|
gst_va_pool_reset_buffer (GstBufferPool * pool, GstBuffer * buffer)
|
|
{
|
|
GstVaPool *vpool = GST_VA_POOL (pool);
|
|
|
|
/* Clears all the memories and only pool the GstBuffer objects */
|
|
if (G_LIKELY (!vpool->starting))
|
|
gst_buffer_remove_all_memory (buffer);
|
|
|
|
GST_BUFFER_POOL_CLASS (parent_class)->reset_buffer (pool, buffer);
|
|
|
|
if (G_LIKELY (!vpool->starting))
|
|
GST_BUFFER_FLAGS (buffer) = 0;
|
|
}
|
|
|
|
static GstFlowReturn
|
|
gst_va_pool_acquire_buffer (GstBufferPool * pool, GstBuffer ** buffer,
|
|
GstBufferPoolAcquireParams * params)
|
|
{
|
|
GstFlowReturn ret;
|
|
GstVaPool *vpool = GST_VA_POOL (pool);
|
|
|
|
ret = GST_BUFFER_POOL_CLASS (parent_class)->acquire_buffer (pool, buffer,
|
|
params);
|
|
if (ret != GST_FLOW_OK)
|
|
return ret;
|
|
|
|
/* if buffer is new, return it */
|
|
if (gst_buffer_n_memory (*buffer) > 0)
|
|
return GST_FLOW_OK;
|
|
|
|
if (GST_IS_VA_DMABUF_ALLOCATOR (vpool->allocator)) {
|
|
if (gst_va_dmabuf_allocator_prepare_buffer (vpool->allocator, *buffer))
|
|
return GST_FLOW_OK;
|
|
} else if (GST_IS_VA_ALLOCATOR (vpool->allocator)) {
|
|
if (gst_va_allocator_prepare_buffer (vpool->allocator, *buffer))
|
|
return GST_FLOW_OK;
|
|
}
|
|
|
|
gst_buffer_replace (buffer, NULL);
|
|
return GST_FLOW_ERROR;
|
|
}
|
|
|
|
static void
|
|
gst_va_pool_flush_start (GstBufferPool * pool)
|
|
{
|
|
GstVaPool *vpool = GST_VA_POOL (pool);
|
|
|
|
if (GST_IS_VA_DMABUF_ALLOCATOR (vpool->allocator))
|
|
gst_va_dmabuf_allocator_flush (vpool->allocator);
|
|
else if (GST_IS_VA_ALLOCATOR (vpool->allocator))
|
|
gst_va_allocator_flush (vpool->allocator);
|
|
}
|
|
|
|
static gboolean
|
|
gst_va_pool_start (GstBufferPool * pool)
|
|
{
|
|
GstVaPool *vpool = GST_VA_POOL (pool);
|
|
gboolean ret;
|
|
|
|
vpool->starting = TRUE;
|
|
ret = GST_BUFFER_POOL_CLASS (parent_class)->start (pool);
|
|
vpool->starting = FALSE;
|
|
|
|
return ret;
|
|
}
|
|
|
|
static void
|
|
gst_va_pool_dispose (GObject * object)
|
|
{
|
|
GstVaPool *pool = GST_VA_POOL (object);
|
|
|
|
GST_LOG_OBJECT (pool, "finalize video buffer pool %p", pool);
|
|
|
|
gst_clear_object (&pool->allocator);
|
|
|
|
G_OBJECT_CLASS (parent_class)->dispose (object);
|
|
}
|
|
|
|
static void
|
|
gst_va_pool_class_init (GstVaPoolClass * klass)
|
|
{
|
|
GObjectClass *gobject_class = G_OBJECT_CLASS (klass);
|
|
GstBufferPoolClass *gstbufferpool_class = GST_BUFFER_POOL_CLASS (klass);
|
|
|
|
gobject_class->dispose = gst_va_pool_dispose;
|
|
|
|
gstbufferpool_class->get_options = gst_va_pool_get_options;
|
|
gstbufferpool_class->set_config = gst_va_pool_set_config;
|
|
gstbufferpool_class->alloc_buffer = gst_va_pool_alloc;
|
|
gstbufferpool_class->reset_buffer = gst_va_pool_reset_buffer;
|
|
gstbufferpool_class->acquire_buffer = gst_va_pool_acquire_buffer;
|
|
gstbufferpool_class->flush_start = gst_va_pool_flush_start;
|
|
gstbufferpool_class->start = gst_va_pool_start;
|
|
}
|
|
|
|
static void
|
|
gst_va_pool_init (GstVaPool * self)
|
|
{
|
|
}
|
|
|
|
GstBufferPool *
|
|
gst_va_pool_new (void)
|
|
{
|
|
GstVaPool *pool;
|
|
|
|
pool = g_object_new (GST_TYPE_VA_POOL, NULL);
|
|
gst_object_ref_sink (pool);
|
|
|
|
GST_LOG_OBJECT (pool, "new va video buffer pool %p", pool);
|
|
|
|
return GST_BUFFER_POOL_CAST (pool);
|
|
}
|
|
|
|
void
|
|
gst_buffer_pool_config_set_va_allocation_params (GstStructure * config,
|
|
guint usage_hint)
|
|
{
|
|
gst_structure_set (config, "usage-hint", G_TYPE_UINT, usage_hint, NULL);
|
|
}
|
|
|
|
gboolean
|
|
gst_va_pool_requires_video_meta (GstBufferPool * pool)
|
|
{
|
|
return GST_VA_POOL (pool)->force_videometa;
|
|
}
|