That subclass was not needed and was causing issues wit doc generation. The only down side of removing it is that the decoder cast macro is no longer type safe. Part-of: <https://gitlab.freedesktop.org/gstreamer/gstreamer/-/merge_requests/5555>
247 lines
8.2 KiB
C
247 lines
8.2 KiB
C
/* GStreamer
|
|
* Copyright (C) <2021> Collabora Ltd.
|
|
* Author: Daniel Almeida <daniel.almeida@collabora.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 <gst/pbutils/pbutils.h>
|
|
|
|
#include "gstv4l2codecalphadecodebin.h"
|
|
#include "gstv4l2decoder.h"
|
|
|
|
/* When wrapping, use the original rank plus this offset. The ad-hoc rules is
|
|
* that hardware implementation will use PRIMARY+1 or +2 to override the
|
|
* software decoder, so the offset must be large enough to jump over those.
|
|
* This should also be small enough so that a marginal (64) or secondary
|
|
* wrapper does not cross the PRIMARY line.
|
|
*/
|
|
#define GST_V4L2_CODEC_ALPHA_DECODE_BIN_RANK_OFFSET 10
|
|
|
|
GST_DEBUG_CATEGORY_STATIC (v4l2_codecalphadecodebin_debug);
|
|
#define GST_CAT_DEFAULT (v4l2_codecalphadecodebin_debug)
|
|
|
|
typedef struct
|
|
{
|
|
GstBin parent;
|
|
|
|
gboolean constructed;
|
|
const gchar *missing_element;
|
|
} GstV4l2CodecAlphaDecodeBinPrivate;
|
|
|
|
#define gst_v4l2_codec_alpha_decode_bin_parent_class parent_class
|
|
G_DEFINE_ABSTRACT_TYPE_WITH_CODE (GstV4l2CodecAlphaDecodeBin,
|
|
gst_v4l2_codec_alpha_decode_bin, GST_TYPE_BIN,
|
|
G_ADD_PRIVATE (GstV4l2CodecAlphaDecodeBin);
|
|
GST_DEBUG_CATEGORY_INIT (v4l2_codecalphadecodebin_debug,
|
|
"v4l2codecs-alphadecodebin", 0, "V4L2 stateless alpha decode bin"));
|
|
|
|
|
|
static GstStaticPadTemplate gst_alpha_decode_bin_src_template =
|
|
GST_STATIC_PAD_TEMPLATE ("src",
|
|
GST_PAD_SRC,
|
|
GST_PAD_ALWAYS,
|
|
GST_STATIC_CAPS ("ANY")
|
|
);
|
|
|
|
static gboolean
|
|
gst_v4l2_codec_alpha_decode_bin_open (GstV4l2CodecAlphaDecodeBin * self)
|
|
{
|
|
GstV4l2CodecAlphaDecodeBinPrivate *priv =
|
|
gst_v4l2_codec_alpha_decode_bin_get_instance_private (self);
|
|
|
|
if (priv->missing_element) {
|
|
gst_element_post_message (GST_ELEMENT (self),
|
|
gst_missing_element_message_new (GST_ELEMENT (self),
|
|
priv->missing_element));
|
|
} else if (!priv->constructed) {
|
|
GST_ELEMENT_ERROR (self, CORE, FAILED,
|
|
("Failed to construct alpha decoder pipeline."), (NULL));
|
|
}
|
|
|
|
return priv->constructed;
|
|
}
|
|
|
|
static GstStateChangeReturn
|
|
gst_v4l2_codec_alpha_decode_bin_change_state (GstElement * element,
|
|
GstStateChange transition)
|
|
{
|
|
GstV4l2CodecAlphaDecodeBin *self = GST_V4L2_CODEC_ALPHA_DECODE_BIN (element);
|
|
|
|
switch (transition) {
|
|
case GST_STATE_CHANGE_NULL_TO_READY:
|
|
if (!gst_v4l2_codec_alpha_decode_bin_open (self))
|
|
return GST_STATE_CHANGE_FAILURE;
|
|
break;
|
|
default:
|
|
break;
|
|
}
|
|
|
|
return GST_ELEMENT_CLASS (parent_class)->change_state (element, transition);
|
|
}
|
|
|
|
static void
|
|
gst_v4l2_codec_alpha_decode_bin_constructed (GObject * obj)
|
|
{
|
|
GstV4l2CodecAlphaDecodeBin *self = GST_V4L2_CODEC_ALPHA_DECODE_BIN (obj);
|
|
GstV4l2CodecAlphaDecodeBinPrivate *priv =
|
|
gst_v4l2_codec_alpha_decode_bin_get_instance_private (self);
|
|
GstV4l2CodecAlphaDecodeBinClass *klass =
|
|
GST_V4L2_CODEC_ALPHA_DECODE_BIN_GET_CLASS (self);
|
|
GstPad *src_gpad, *sink_gpad;
|
|
GstPad *src_pad = NULL, *sink_pad = NULL;
|
|
GstElement *alphademux = NULL;
|
|
GstElement *queue = NULL;
|
|
GstElement *alpha_queue = NULL;
|
|
GstElement *decoder = NULL;
|
|
GstElement *alpha_decoder = NULL;
|
|
GstElement *alphacombine = NULL;
|
|
|
|
/* setup ghost pads */
|
|
sink_gpad = gst_ghost_pad_new_no_target_from_template ("sink",
|
|
gst_element_class_get_pad_template (GST_ELEMENT_CLASS (klass), "sink"));
|
|
gst_element_add_pad (GST_ELEMENT (self), sink_gpad);
|
|
|
|
src_gpad = gst_ghost_pad_new_no_target_from_template ("src",
|
|
gst_element_class_get_pad_template (GST_ELEMENT_CLASS (klass), "src"));
|
|
gst_element_add_pad (GST_ELEMENT (self), src_gpad);
|
|
|
|
/* create elements */
|
|
alphademux = gst_element_factory_make ("codecalphademux", NULL);
|
|
if (!alphademux) {
|
|
priv->missing_element = "codecalphademux";
|
|
goto cleanup;
|
|
}
|
|
|
|
queue = gst_element_factory_make ("queue", NULL);
|
|
alpha_queue = gst_element_factory_make ("queue", NULL);
|
|
if (!queue || !alpha_queue) {
|
|
priv->missing_element = "queue";
|
|
goto cleanup;
|
|
}
|
|
|
|
decoder = gst_element_factory_make (klass->decoder_name, "maindec");
|
|
if (!decoder) {
|
|
priv->missing_element = klass->decoder_name;
|
|
goto cleanup;
|
|
}
|
|
|
|
alpha_decoder = gst_element_factory_make (klass->decoder_name, "alphadec");
|
|
if (!alpha_decoder) {
|
|
priv->missing_element = klass->decoder_name;
|
|
goto cleanup;
|
|
}
|
|
|
|
/* We disable QoS on decoders because we need to maintain frame pairing in
|
|
* order for alphacombine to work. */
|
|
g_object_set (decoder, "qos", FALSE, NULL);
|
|
g_object_set (alpha_decoder, "qos", FALSE, NULL);
|
|
|
|
alphacombine = gst_element_factory_make ("alphacombine", NULL);
|
|
if (!alphacombine) {
|
|
priv->missing_element = "alphacombine";
|
|
goto cleanup;
|
|
}
|
|
|
|
gst_bin_add_many (GST_BIN (self), alphademux, queue, alpha_queue, decoder,
|
|
alpha_decoder, alphacombine, NULL);
|
|
|
|
/* link elements */
|
|
sink_pad = gst_element_get_static_pad (alphademux, "sink");
|
|
gst_ghost_pad_set_target (GST_GHOST_PAD (sink_gpad), sink_pad);
|
|
gst_clear_object (&sink_pad);
|
|
|
|
gst_element_link_pads (alphademux, "src", queue, "sink");
|
|
gst_element_link_pads (queue, "src", decoder, "sink");
|
|
gst_element_link_pads (decoder, "src", alphacombine, "sink");
|
|
|
|
gst_element_link_pads (alphademux, "alpha", alpha_queue, "sink");
|
|
gst_element_link_pads (alpha_queue, "src", alpha_decoder, "sink");
|
|
gst_element_link_pads (alpha_decoder, "src", alphacombine, "alpha");
|
|
|
|
src_pad = gst_element_get_static_pad (alphacombine, "src");
|
|
gst_ghost_pad_set_target (GST_GHOST_PAD (src_gpad), src_pad);
|
|
gst_object_unref (src_pad);
|
|
|
|
g_object_set (queue, "max-size-bytes", 0, "max-size-time",
|
|
G_GUINT64_CONSTANT (0), "max-size-buffers", 1, NULL);
|
|
g_object_set (alpha_queue, "max-size-bytes", 0, "max-size-time",
|
|
G_GUINT64_CONSTANT (0), "max-size-buffers", 1, NULL);
|
|
|
|
/* signal success, we will handle this in NULL->READY transition */
|
|
priv->constructed = TRUE;
|
|
return;
|
|
|
|
cleanup:
|
|
gst_clear_object (&alphademux);
|
|
gst_clear_object (&queue);
|
|
gst_clear_object (&alpha_queue);
|
|
gst_clear_object (&decoder);
|
|
gst_clear_object (&alpha_decoder);
|
|
gst_clear_object (&alphacombine);
|
|
|
|
G_OBJECT_CLASS (parent_class)->constructed (obj);
|
|
}
|
|
|
|
static void
|
|
gst_v4l2_codec_alpha_decode_bin_class_init (GstV4l2CodecAlphaDecodeBinClass *
|
|
klass)
|
|
{
|
|
GstElementClass *element_class = (GstElementClass *) klass;
|
|
GObjectClass *obj_class = (GObjectClass *) klass;
|
|
|
|
/* This is needed to access the subclass class instance, otherwise we cannot
|
|
* read the class parameters */
|
|
obj_class->constructed = gst_v4l2_codec_alpha_decode_bin_constructed;
|
|
|
|
gst_element_class_add_static_pad_template (element_class,
|
|
&gst_alpha_decode_bin_src_template);
|
|
element_class->change_state =
|
|
GST_DEBUG_FUNCPTR (gst_v4l2_codec_alpha_decode_bin_change_state);
|
|
|
|
/* let's make the doc generator happy */
|
|
gst_type_mark_as_plugin_api (GST_TYPE_V4L2_CODEC_ALPHA_DECODE_BIN, 0);
|
|
}
|
|
|
|
static void
|
|
gst_v4l2_codec_alpha_decode_bin_init (GstV4l2CodecAlphaDecodeBin * self)
|
|
{
|
|
}
|
|
|
|
void
|
|
gst_v4l2_codec_alpha_decode_bin_register (GstPlugin * plugin,
|
|
GClassInitFunc class_init, gconstpointer class_data,
|
|
const gchar * element_name_tmpl, GstV4l2CodecDevice * device, guint rank)
|
|
{
|
|
/* TODO check that we have compatible src format */
|
|
|
|
GTypeInfo type_info = {
|
|
.class_size = sizeof (GstV4l2CodecAlphaDecodeBinClass),
|
|
.class_init = class_init,
|
|
.class_data = class_data,
|
|
.instance_size = sizeof (GstV4l2CodecAlphaDecodeBin),
|
|
.instance_init = NULL,
|
|
};
|
|
|
|
gst_v4l2_decoder_register (plugin, GST_TYPE_V4L2_CODEC_ALPHA_DECODE_BIN,
|
|
&type_info, element_name_tmpl, device,
|
|
rank + GST_V4L2_CODEC_ALPHA_DECODE_BIN_RANK_OFFSET, NULL);
|
|
}
|