gstreamer/gst/vaapi/gstvaapidownload.c

622 lines
18 KiB
C
Raw Normal View History

/*
* gstvaapidownload.c - VA-API video downloader
*
* Copyright (C) 2010-2011 Splitted-Desktop Systems
* Copyright (C) 2011-2012 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
*/
/**
* SECTION:gstvaapidownload
* @short_description: A VA to video flow filter
*
* vaapidownload converts from VA surfaces to raw YUV pixels.
*/
#include "config.h"
#include <gst/gst.h>
#include <gst/video/video.h>
#include <gst/video/videocontext.h>
#include <gst/vaapi/gstvaapivideosink.h>
#include <gst/vaapi/gstvaapivideobuffer.h>
#include "gstvaapidownload.h"
#include "gstvaapipluginutil.h"
#include "gstvaapipluginbuffer.h"
#define GST_PLUGIN_NAME "vaapidownload"
#define GST_PLUGIN_DESC "A VA to video flow filter"
GST_DEBUG_CATEGORY_STATIC(gst_debug_vaapidownload);
#define GST_CAT_DEFAULT gst_debug_vaapidownload
/* ElementFactory information */
static const GstElementDetails gst_vaapidownload_details =
GST_ELEMENT_DETAILS(
"VA-API colorspace converter",
"Filter/Converter/Video",
GST_PLUGIN_DESC,
"Gwenole Beauchesne <gwenole.beauchesne@intel.com>");
/* Default templates */
static const char gst_vaapidownload_yuv_caps_str[] =
"video/x-raw-yuv, "
"width = (int) [ 1, MAX ], "
"height = (int) [ 1, MAX ]; ";
static const char gst_vaapidownload_vaapi_caps_str[] =
GST_VAAPI_SURFACE_CAPS;
static GstStaticPadTemplate gst_vaapidownload_sink_factory =
GST_STATIC_PAD_TEMPLATE(
"sink",
GST_PAD_SINK,
GST_PAD_ALWAYS,
GST_STATIC_CAPS(gst_vaapidownload_vaapi_caps_str));
static GstStaticPadTemplate gst_vaapidownload_src_factory =
GST_STATIC_PAD_TEMPLATE(
"src",
GST_PAD_SRC,
GST_PAD_ALWAYS,
GST_STATIC_CAPS(gst_vaapidownload_yuv_caps_str));
typedef struct _TransformSizeCache TransformSizeCache;
struct _TransformSizeCache {
GstCaps *caps;
guint size;
};
struct _GstVaapiDownload {
/*< private >*/
GstBaseTransform parent_instance;
GstVaapiDisplay *display;
GstCaps *allowed_caps;
TransformSizeCache transform_size_cache[2];
GstVaapiVideoPool *images;
GstVaapiImageFormat image_format;
guint image_width;
guint image_height;
unsigned int images_reset : 1;
};
struct _GstVaapiDownloadClass {
/*< private >*/
GstBaseTransformClass parent_class;
};
static void
gst_vaapidownload_implements_iface_init(GstImplementsInterfaceClass *iface);
static void
gst_video_context_interface_init(GstVideoContextInterface *iface);
#define GstVideoContextClass GstVideoContextInterface
G_DEFINE_TYPE_WITH_CODE(
GstVaapiDownload,
gst_vaapidownload,
GST_TYPE_BASE_TRANSFORM,
G_IMPLEMENT_INTERFACE(GST_TYPE_IMPLEMENTS_INTERFACE,
gst_vaapidownload_implements_iface_init);
G_IMPLEMENT_INTERFACE(GST_TYPE_VIDEO_CONTEXT,
gst_video_context_interface_init));
static gboolean
gst_vaapidownload_start(GstBaseTransform *trans);
static gboolean
gst_vaapidownload_stop(GstBaseTransform *trans);
static void
gst_vaapidownload_before_transform(GstBaseTransform *trans, GstBuffer *buffer);
static GstFlowReturn
gst_vaapidownload_transform(
GstBaseTransform *trans,
GstBuffer *inbuf,
GstBuffer *outbuf
);
static GstCaps *
gst_vaapidownload_transform_caps(
GstBaseTransform *trans,
GstPadDirection direction,
GstCaps *caps
);
static gboolean
gst_vaapidownload_transform_size(
GstBaseTransform *trans,
GstPadDirection direction,
GstCaps *caps,
guint size,
GstCaps *othercaps,
guint *othersize
);
static gboolean
gst_vaapidownload_set_caps(
GstBaseTransform *trans,
GstCaps *incaps,
GstCaps *outcaps
);
static gboolean
gst_vaapidownload_query(
GstPad *pad,
GstQuery *query
);
/* GstImplementsInterface interface */
static gboolean
gst_vaapidownload_implements_interface_supported(
GstImplementsInterface *iface,
GType type
)
{
return (type == GST_TYPE_VIDEO_CONTEXT);
}
static void
gst_vaapidownload_implements_iface_init(GstImplementsInterfaceClass *iface)
{
iface->supported = gst_vaapidownload_implements_interface_supported;
}
/* GstVideoContext interface */
static void
gst_vaapidownload_set_video_context(GstVideoContext *context, const gchar *type,
const GValue *value)
{
GstVaapiDownload *download = GST_VAAPIDOWNLOAD (context);
gst_vaapi_set_display (type, value, &download->display);
}
static void
gst_video_context_interface_init(GstVideoContextInterface *iface)
{
iface->set_context = gst_vaapidownload_set_video_context;
}
static void
gst_vaapidownload_destroy(GstVaapiDownload *download)
{
guint i;
for (i = 0; i < G_N_ELEMENTS(download->transform_size_cache); i++) {
TransformSizeCache * const tsc = &download->transform_size_cache[i];
if (tsc->caps) {
gst_caps_unref(tsc->caps);
tsc->caps = NULL;
tsc->size = 0;
}
}
if (download->allowed_caps) {
gst_caps_unref(download->allowed_caps);
download->allowed_caps = NULL;
}
g_clear_object(&download->images);
g_clear_object(&download->display);
}
static void
gst_vaapidownload_finalize(GObject *object)
{
gst_vaapidownload_destroy(GST_VAAPIDOWNLOAD(object));
G_OBJECT_CLASS(gst_vaapidownload_parent_class)->finalize(object);
}
static void
gst_vaapidownload_class_init(GstVaapiDownloadClass *klass)
{
GObjectClass * const object_class = G_OBJECT_CLASS(klass);
GstBaseTransformClass * const trans_class = GST_BASE_TRANSFORM_CLASS(klass);
GstElementClass * const element_class = GST_ELEMENT_CLASS(klass);
GstPadTemplate *pad_template;
GST_DEBUG_CATEGORY_INIT(gst_debug_vaapidownload,
GST_PLUGIN_NAME, 0, GST_PLUGIN_DESC);
object_class->finalize = gst_vaapidownload_finalize;
trans_class->start = gst_vaapidownload_start;
trans_class->stop = gst_vaapidownload_stop;
trans_class->before_transform = gst_vaapidownload_before_transform;
trans_class->transform = gst_vaapidownload_transform;
trans_class->transform_caps = gst_vaapidownload_transform_caps;
trans_class->transform_size = gst_vaapidownload_transform_size;
trans_class->set_caps = gst_vaapidownload_set_caps;
gst_element_class_set_details_simple(
element_class,
gst_vaapidownload_details.longname,
gst_vaapidownload_details.klass,
gst_vaapidownload_details.description,
gst_vaapidownload_details.author
);
/* sink pad */
pad_template = gst_static_pad_template_get(&gst_vaapidownload_sink_factory);
gst_element_class_add_pad_template(element_class, pad_template);
gst_object_unref(pad_template);
/* src pad */
pad_template = gst_static_pad_template_get(&gst_vaapidownload_src_factory);
gst_element_class_add_pad_template(element_class, pad_template);
gst_object_unref(pad_template);
}
static void
gst_vaapidownload_init(GstVaapiDownload *download)
{
GstPad *sinkpad, *srcpad;
download->display = NULL;
download->allowed_caps = NULL;
download->images = NULL;
download->images_reset = FALSE;
download->image_format = (GstVaapiImageFormat)0;
download->image_width = 0;
download->image_height = 0;
/* Override buffer allocator on sink pad */
sinkpad = gst_element_get_static_pad(GST_ELEMENT(download), "sink");
gst_pad_set_query_function(sinkpad, gst_vaapidownload_query);
gst_object_unref(sinkpad);
/* Override query on src pad */
srcpad = gst_element_get_static_pad(GST_ELEMENT(download), "src");
gst_pad_set_query_function(srcpad, gst_vaapidownload_query);
gst_object_unref(srcpad);
}
static inline gboolean
gst_vaapidownload_ensure_display(GstVaapiDownload *download)
{
return gst_vaapi_ensure_display(download, GST_VAAPI_DISPLAY_TYPE_ANY,
&download->display);
}
static gboolean
gst_vaapidownload_start(GstBaseTransform *trans)
{
GstVaapiDownload * const download = GST_VAAPIDOWNLOAD(trans);
if (!gst_vaapidownload_ensure_display(download))
return FALSE;
return TRUE;
}
static gboolean
gst_vaapidownload_stop(GstBaseTransform *trans)
{
GstVaapiDownload * const download = GST_VAAPIDOWNLOAD(trans);
g_clear_object(&download->display);
return TRUE;
}
static GstVaapiImageFormat
get_surface_format(GstVaapiSurface *surface)
{
GstVaapiImage *image;
GstVaapiImageFormat format = GST_VAAPI_IMAGE_NV12;
/* XXX: NV12 is assumed by default */
image = gst_vaapi_surface_derive_image(surface);
if (image) {
format = gst_vaapi_image_get_format(image);
g_object_unref(image);
}
return format;
}
static gboolean
gst_vaapidownload_update_src_caps(GstVaapiDownload *download, GstBuffer *buffer)
{
GstVaapiVideoBuffer *vbuffer;
GstVaapiSurface *surface;
GstVaapiImageFormat format;
GstPad *srcpad;
GstCaps *in_caps, *out_caps;
vbuffer = GST_VAAPI_VIDEO_BUFFER(buffer);
surface = gst_vaapi_video_buffer_get_surface(vbuffer);
if (!surface) {
GST_WARNING("failed to retrieve VA surface from buffer");
return FALSE;
}
format = get_surface_format(surface);
if (format == download->image_format)
return TRUE;
in_caps = GST_BUFFER_CAPS(buffer);
if (!in_caps) {
GST_WARNING("failed to retrieve caps from buffer");
return FALSE;
}
out_caps = gst_vaapi_image_format_get_caps(format);
if (!out_caps) {
GST_WARNING("failed to create caps from format %" GST_FOURCC_FORMAT,
GST_FOURCC_ARGS(format));
return FALSE;
}
if (!gst_vaapi_append_surface_caps(out_caps, in_caps)) {
gst_caps_unref(out_caps);
return FALSE;
}
/* Try to renegotiate downstream caps */
srcpad = gst_element_get_static_pad(GST_ELEMENT(download), "src");
gst_pad_set_caps(srcpad, out_caps);
gst_object_unref(srcpad);
gst_vaapidownload_set_caps(GST_BASE_TRANSFORM(download), in_caps, out_caps);
gst_caps_replace(&download->allowed_caps, out_caps);
gst_caps_unref(out_caps);
return TRUE;
}
static void
gst_vaapidownload_before_transform(GstBaseTransform *trans, GstBuffer *buffer)
{
GstVaapiDownload * const download = GST_VAAPIDOWNLOAD(trans);
gst_vaapidownload_update_src_caps(download, buffer);
}
static GstFlowReturn
gst_vaapidownload_transform(
GstBaseTransform *trans,
GstBuffer *inbuf,
GstBuffer *outbuf
)
{
GstVaapiDownload * const download = GST_VAAPIDOWNLOAD(trans);
GstVaapiVideoBuffer *vbuffer;
GstVaapiSurface *surface;
GstVaapiImage *image = NULL;
gboolean success;
vbuffer = GST_VAAPI_VIDEO_BUFFER(inbuf);
surface = gst_vaapi_video_buffer_get_surface(vbuffer);
if (!surface)
return GST_FLOW_UNEXPECTED;
image = gst_vaapi_video_pool_get_object(download->images);
if (!image)
return GST_FLOW_UNEXPECTED;
if (!gst_vaapi_surface_get_image(surface, image))
goto error_get_image;
success = gst_vaapi_image_get_buffer(image, outbuf, NULL);
gst_vaapi_video_pool_put_object(download->images, image);
if (!success)
goto error_get_buffer;
return GST_FLOW_OK;
error_get_image:
{
GST_WARNING("failed to download %" GST_FOURCC_FORMAT " image "
"from surface 0x%08x",
GST_FOURCC_ARGS(gst_vaapi_image_get_format(image)),
gst_vaapi_surface_get_id(surface));
gst_vaapi_video_pool_put_object(download->images, image);
return GST_FLOW_UNEXPECTED;
}
error_get_buffer:
{
GST_WARNING("failed to transfer image to output video buffer");
return GST_FLOW_UNEXPECTED;
}
}
static GstCaps *
gst_vaapidownload_transform_caps(
GstBaseTransform *trans,
GstPadDirection direction,
GstCaps *caps
)
{
GstVaapiDownload * const download = GST_VAAPIDOWNLOAD(trans);
GstPad *srcpad;
GstCaps *allowed_caps, *inter_caps, *out_caps = NULL;
GstStructure *structure;
g_return_val_if_fail(GST_IS_CAPS(caps), NULL);
structure = gst_caps_get_structure(caps, 0);
if (direction == GST_PAD_SINK) {
if (!gst_structure_has_name(structure, GST_VAAPI_SURFACE_CAPS_NAME))
return NULL;
if (!gst_vaapidownload_ensure_display(download))
return NULL;
out_caps = gst_caps_from_string(gst_vaapidownload_yuv_caps_str);
/* Build up allowed caps */
/* XXX: we don't know the decoded surface format yet so we
expose whatever VA images we support */
if (download->allowed_caps)
allowed_caps = gst_caps_ref(download->allowed_caps);
else {
allowed_caps = gst_vaapi_display_get_image_caps(download->display);
if (!allowed_caps)
return NULL;
}
inter_caps = gst_caps_intersect(out_caps, allowed_caps);
gst_caps_unref(allowed_caps);
gst_caps_unref(out_caps);
out_caps = inter_caps;
/* Intersect with allowed caps from the peer, if any */
srcpad = gst_element_get_static_pad(GST_ELEMENT(download), "src");
allowed_caps = gst_pad_peer_get_caps(srcpad);
if (allowed_caps) {
inter_caps = gst_caps_intersect(out_caps, allowed_caps);
gst_caps_unref(allowed_caps);
gst_caps_unref(out_caps);
out_caps = inter_caps;
}
}
else {
if (!gst_structure_has_name(structure, "video/x-raw-yuv"))
return NULL;
out_caps = gst_caps_from_string(gst_vaapidownload_vaapi_caps_str);
structure = gst_caps_get_structure(out_caps, 0);
gst_structure_set(
structure,
"type", G_TYPE_STRING, "vaapi",
"opengl", G_TYPE_BOOLEAN, USE_GLX,
NULL
);
}
if (!gst_vaapi_append_surface_caps(out_caps, caps)) {
gst_caps_unref(out_caps);
return NULL;
}
return out_caps;
}
static gboolean
gst_vaapidownload_ensure_image_pool(GstVaapiDownload *download, GstCaps *caps)
{
GstStructure * const structure = gst_caps_get_structure(caps, 0);
GstVaapiImageFormat format;
gint width, height;
format = gst_vaapi_image_format_from_caps(caps);
gst_structure_get_int(structure, "width", &width);
gst_structure_get_int(structure, "height", &height);
if (format != download->image_format ||
width != download->image_width ||
height != download->image_height) {
download->image_format = format;
download->image_width = width;
download->image_height = height;
g_clear_object(&download->images);
download->images = gst_vaapi_image_pool_new(download->display, caps);
if (!download->images)
return FALSE;
download->images_reset = TRUE;
}
return TRUE;
}
static inline gboolean
gst_vaapidownload_negotiate_buffers(
GstVaapiDownload *download,
GstCaps *incaps,
GstCaps *outcaps
)
{
if (!gst_vaapidownload_ensure_image_pool(download, outcaps))
return FALSE;
return TRUE;
}
static gboolean
gst_vaapidownload_set_caps(
GstBaseTransform *trans,
GstCaps *incaps,
GstCaps *outcaps
)
{
GstVaapiDownload * const download = GST_VAAPIDOWNLOAD(trans);
if (!gst_vaapidownload_negotiate_buffers(download, incaps, outcaps))
return FALSE;
return TRUE;
}
static gboolean
gst_vaapidownload_transform_size(
GstBaseTransform *trans,
GstPadDirection direction,
GstCaps *caps,
guint size,
GstCaps *othercaps,
guint *othersize
)
{
GstVaapiDownload * const download = GST_VAAPIDOWNLOAD(trans);
GstStructure * const structure = gst_caps_get_structure(othercaps, 0);
GstVideoFormat format;
gint width, height;
guint i;
/* Lookup in cache */
for (i = 0; i < G_N_ELEMENTS(download->transform_size_cache); i++) {
TransformSizeCache * const tsc = &download->transform_size_cache[i];
if (tsc->caps && tsc->caps == othercaps) {
*othersize = tsc->size;
return TRUE;
}
}
/* Compute requested buffer size */
if (gst_structure_has_name(structure, GST_VAAPI_SURFACE_CAPS_NAME))
*othersize = 0;
else {
if (!gst_video_format_parse_caps(othercaps, &format, &width, &height))
return FALSE;
*othersize = gst_video_format_get_size(format, width, height);
}
/* Update cache */
for (i = 0; i < G_N_ELEMENTS(download->transform_size_cache); i++) {
TransformSizeCache * const tsc = &download->transform_size_cache[i];
if (!tsc->caps) {
gst_caps_replace(&tsc->caps, othercaps);
tsc->size = *othersize;
}
}
return TRUE;
}
static gboolean
gst_vaapidownload_query(GstPad *pad, GstQuery *query)
{
GstVaapiDownload * const download = GST_VAAPIDOWNLOAD(gst_pad_get_parent_element(pad));
gboolean res;
GST_DEBUG("sharing display %p", download->display);
if (gst_vaapi_reply_to_query(query, download->display))
res = TRUE;
else
res = gst_pad_query_default(pad, query);
g_object_unref(download);
return res;
}