mirror of
https://gitlab.freedesktop.org/gstreamer/gstreamer.git
synced 2025-01-15 11:55:32 +00:00
de6db03b53
This will prevent the converter to be picked automatically in case someone implement dynamic converter selection support. I'd like this to be ranked only for known device, as it's hard to be sure a device is a converter suited for general purpose. Re-negotiation is also needed before we can rank it. https://bugzilla.gnome.org/show_bug.cgi?id=733607
757 lines
21 KiB
C
757 lines
21 KiB
C
/*
|
|
* Copyright (C) 2014 Collabora Ltd.
|
|
* Author: Nicolas Dufresne <nicolas.dufresne@collabora.co.uk>
|
|
*
|
|
* 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 <sys/stat.h>
|
|
#include <fcntl.h>
|
|
#include <errno.h>
|
|
#include <unistd.h>
|
|
#include <string.h>
|
|
|
|
#include "gstv4l2transform.h"
|
|
#include "v4l2_calls.h"
|
|
|
|
#include <string.h>
|
|
#include <gst/gst-i18n-plugin.h>
|
|
|
|
#define DEFAULT_PROP_DEVICE "/dev/video10"
|
|
|
|
#define V4L2_TRANSFORM_QUARK \
|
|
g_quark_from_static_string("gst-v4l2-transform-info")
|
|
|
|
GST_DEBUG_CATEGORY_STATIC (gst_v4l2_transform_debug);
|
|
#define GST_CAT_DEFAULT gst_v4l2_transform_debug
|
|
|
|
|
|
enum
|
|
{
|
|
PROP_0,
|
|
V4L2_STD_OBJECT_PROPS
|
|
};
|
|
|
|
typedef struct
|
|
{
|
|
gchar *device;
|
|
GstCaps *sink_caps;
|
|
GstCaps *src_caps;
|
|
} GstV4l2TransformCData;
|
|
|
|
#define gst_v4l2_transform_parent_class parent_class
|
|
G_DEFINE_ABSTRACT_TYPE (GstV4l2Transform, gst_v4l2_transform,
|
|
GST_TYPE_BASE_TRANSFORM);
|
|
|
|
static void
|
|
gst_v4l2_transform_set_property (GObject * object,
|
|
guint prop_id, const GValue * value, GParamSpec * pspec)
|
|
{
|
|
GstV4l2Transform *self = GST_V4L2_TRANSFORM (object);
|
|
|
|
switch (prop_id) {
|
|
case PROP_OUTPUT_IO_MODE:
|
|
gst_v4l2_object_set_property_helper (self->v4l2output, prop_id, value,
|
|
pspec);
|
|
break;
|
|
case PROP_CAPTURE_IO_MODE:
|
|
gst_v4l2_object_set_property_helper (self->v4l2capture, PROP_IO_MODE,
|
|
value, pspec);
|
|
break;
|
|
|
|
/* By default, only set on output */
|
|
default:
|
|
if (!gst_v4l2_object_set_property_helper (self->v4l2output,
|
|
prop_id, value, pspec)) {
|
|
G_OBJECT_WARN_INVALID_PROPERTY_ID (object, prop_id, pspec);
|
|
}
|
|
break;
|
|
}
|
|
}
|
|
|
|
static void
|
|
gst_v4l2_transform_get_property (GObject * object,
|
|
guint prop_id, GValue * value, GParamSpec * pspec)
|
|
{
|
|
GstV4l2Transform *self = GST_V4L2_TRANSFORM (object);
|
|
|
|
switch (prop_id) {
|
|
case PROP_OUTPUT_IO_MODE:
|
|
gst_v4l2_object_get_property_helper (self->v4l2output, prop_id, value,
|
|
pspec);
|
|
break;
|
|
case PROP_CAPTURE_IO_MODE:
|
|
gst_v4l2_object_get_property_helper (self->v4l2capture, prop_id, value,
|
|
pspec);
|
|
break;
|
|
|
|
/* By default read from output */
|
|
default:
|
|
if (!gst_v4l2_object_get_property_helper (self->v4l2output,
|
|
prop_id, value, pspec)) {
|
|
G_OBJECT_WARN_INVALID_PROPERTY_ID (object, prop_id, pspec);
|
|
}
|
|
break;
|
|
}
|
|
}
|
|
|
|
static gboolean
|
|
gst_v4l2_transform_open (GstV4l2Transform * self)
|
|
{
|
|
GST_DEBUG_OBJECT (self, "Opening");
|
|
|
|
if (!gst_v4l2_object_open (self->v4l2output))
|
|
goto failure;
|
|
|
|
if (!gst_v4l2_object_open_shared (self->v4l2capture, self->v4l2output))
|
|
goto failure;
|
|
|
|
self->probed_sinkcaps = gst_v4l2_object_get_caps (self->v4l2output,
|
|
gst_v4l2_object_get_raw_caps ());
|
|
|
|
if (gst_caps_is_empty (self->probed_sinkcaps))
|
|
goto no_input_format;
|
|
|
|
self->probed_srccaps = gst_v4l2_object_get_caps (self->v4l2capture,
|
|
gst_v4l2_object_get_raw_caps ());
|
|
|
|
if (gst_caps_is_empty (self->probed_srccaps))
|
|
goto no_output_format;
|
|
|
|
return TRUE;
|
|
|
|
no_input_format:
|
|
GST_ELEMENT_ERROR (self, RESOURCE, SETTINGS,
|
|
(_("Converter on device %s has no supported input format"),
|
|
self->v4l2output->videodev), (NULL));
|
|
goto failure;
|
|
|
|
|
|
no_output_format:
|
|
GST_ELEMENT_ERROR (self, RESOURCE, SETTINGS,
|
|
(_("Converter on device %s has no supported output format"),
|
|
self->v4l2output->videodev), (NULL));
|
|
goto failure;
|
|
|
|
failure:
|
|
if (GST_V4L2_IS_OPEN (self->v4l2output))
|
|
gst_v4l2_object_close (self->v4l2output);
|
|
|
|
if (GST_V4L2_IS_OPEN (self->v4l2capture))
|
|
gst_v4l2_object_close (self->v4l2capture);
|
|
|
|
gst_caps_replace (&self->probed_srccaps, NULL);
|
|
gst_caps_replace (&self->probed_sinkcaps, NULL);
|
|
|
|
return FALSE;
|
|
}
|
|
|
|
static void
|
|
gst_v4l2_transform_close (GstV4l2Transform * self)
|
|
{
|
|
GST_DEBUG_OBJECT (self, "Closing");
|
|
|
|
gst_v4l2_object_close (self->v4l2output);
|
|
gst_v4l2_object_close (self->v4l2capture);
|
|
|
|
gst_caps_replace (&self->probed_srccaps, NULL);
|
|
gst_caps_replace (&self->probed_srccaps, NULL);
|
|
}
|
|
|
|
static gboolean
|
|
gst_v4l2_transform_stop (GstBaseTransform * trans)
|
|
{
|
|
GstV4l2Transform *self = GST_V4L2_TRANSFORM (trans);
|
|
|
|
GST_DEBUG_OBJECT (self, "Stop");
|
|
|
|
gst_v4l2_object_stop (self->v4l2output);
|
|
gst_v4l2_object_stop (self->v4l2capture);
|
|
gst_caps_replace (&self->incaps, NULL);
|
|
gst_caps_replace (&self->outcaps, NULL);
|
|
|
|
return TRUE;
|
|
}
|
|
|
|
static gboolean
|
|
gst_v4l2_transform_set_caps (GstBaseTransform * trans, GstCaps * incaps,
|
|
GstCaps * outcaps)
|
|
{
|
|
GstV4l2Transform *self = GST_V4L2_TRANSFORM (trans);
|
|
|
|
if (self->incaps && self->outcaps) {
|
|
if (gst_caps_is_equal (incaps, self->incaps) &&
|
|
gst_caps_is_equal (outcaps, self->outcaps)) {
|
|
GST_DEBUG_OBJECT (trans, "Caps did not changed");
|
|
return TRUE;
|
|
}
|
|
}
|
|
|
|
/* TODO Add renegotiation support */
|
|
g_return_val_if_fail (!GST_V4L2_IS_ACTIVE (self->v4l2output), FALSE);
|
|
g_return_val_if_fail (!GST_V4L2_IS_ACTIVE (self->v4l2capture), FALSE);
|
|
|
|
gst_caps_replace (&self->incaps, incaps);
|
|
gst_caps_replace (&self->outcaps, outcaps);
|
|
|
|
if (!gst_v4l2_object_set_format (self->v4l2output, incaps))
|
|
goto incaps_failed;
|
|
|
|
if (!gst_v4l2_object_set_format (self->v4l2capture, outcaps))
|
|
goto outcaps_failed;
|
|
|
|
/* FIXME implement fallback if crop not supported */
|
|
if (!gst_v4l2_object_set_crop (self->v4l2output))
|
|
goto failed;
|
|
|
|
if (!gst_v4l2_object_set_crop (self->v4l2capture))
|
|
goto failed;
|
|
|
|
return TRUE;
|
|
|
|
incaps_failed:
|
|
{
|
|
GST_ERROR_OBJECT (self, "failed to set input caps: %" GST_PTR_FORMAT,
|
|
incaps);
|
|
goto failed;
|
|
}
|
|
outcaps_failed:
|
|
{
|
|
gst_v4l2_object_stop (self->v4l2output);
|
|
GST_ERROR_OBJECT (self, "failed to set output caps: %" GST_PTR_FORMAT,
|
|
outcaps);
|
|
goto failed;
|
|
}
|
|
failed:
|
|
return FALSE;
|
|
}
|
|
|
|
static gboolean
|
|
gst_v4l2_transform_query (GstBaseTransform * trans, GstPadDirection direction,
|
|
GstQuery * query)
|
|
{
|
|
GstV4l2Transform *self = GST_V4L2_TRANSFORM (trans);
|
|
gboolean ret = TRUE;
|
|
|
|
switch (GST_QUERY_TYPE (query)) {
|
|
case GST_QUERY_CAPS:{
|
|
GstCaps *filter, *caps = NULL, *result = NULL;
|
|
GstPad *pad, *otherpad;
|
|
|
|
gst_query_parse_caps (query, &filter);
|
|
|
|
if (direction == GST_PAD_SRC) {
|
|
pad = GST_BASE_TRANSFORM_SRC_PAD (trans);
|
|
otherpad = GST_BASE_TRANSFORM_SINK_PAD (trans);
|
|
if (self->probed_srccaps)
|
|
caps = gst_caps_ref (self->probed_srccaps);
|
|
} else {
|
|
pad = GST_BASE_TRANSFORM_SINK_PAD (trans);
|
|
otherpad = GST_BASE_TRANSFORM_SRC_PAD (trans);
|
|
if (self->probed_sinkcaps)
|
|
caps = gst_caps_ref (self->probed_sinkcaps);
|
|
}
|
|
|
|
if (!caps)
|
|
caps = gst_pad_get_pad_template_caps (pad);
|
|
|
|
if (filter) {
|
|
GstCaps *tmp = caps;
|
|
caps = gst_caps_intersect_full (filter, tmp, GST_CAPS_INTERSECT_FIRST);
|
|
gst_caps_unref (tmp);
|
|
}
|
|
|
|
result = gst_pad_peer_query_caps (otherpad, caps);
|
|
result = gst_caps_make_writable (result);
|
|
gst_caps_append (result, caps);
|
|
|
|
GST_DEBUG_OBJECT (self, "Returning %s caps %" GST_PTR_FORMAT,
|
|
GST_PAD_NAME (pad), result);
|
|
|
|
gst_query_set_caps_result (query, result);
|
|
gst_caps_unref (result);
|
|
break;
|
|
}
|
|
|
|
default:
|
|
ret = GST_BASE_TRANSFORM_CLASS (parent_class)->query (trans, direction,
|
|
query);
|
|
break;
|
|
}
|
|
|
|
return ret;
|
|
}
|
|
|
|
static gboolean
|
|
gst_v4l2_transform_decide_allocation (GstBaseTransform * trans,
|
|
GstQuery * query)
|
|
{
|
|
GstV4l2Transform *self = GST_V4L2_TRANSFORM (trans);
|
|
gboolean ret = FALSE;
|
|
|
|
GST_DEBUG_OBJECT (self, "called");
|
|
|
|
if (gst_v4l2_object_decide_allocation (self->v4l2capture, query)) {
|
|
GstBufferPool *pool = GST_BUFFER_POOL (self->v4l2capture->pool);
|
|
|
|
ret = GST_BASE_TRANSFORM_CLASS (parent_class)->decide_allocation (trans,
|
|
query);
|
|
|
|
if (!gst_buffer_pool_set_active (pool, TRUE))
|
|
goto activate_failed;
|
|
}
|
|
|
|
return ret;
|
|
|
|
activate_failed:
|
|
GST_ELEMENT_ERROR (self, RESOURCE, SETTINGS,
|
|
("failed to activate bufferpool"), ("failed to activate bufferpool"));
|
|
return TRUE;
|
|
}
|
|
|
|
static gboolean
|
|
gst_v4l2_transform_propose_allocation (GstBaseTransform * trans,
|
|
GstQuery * decide_query, GstQuery * query)
|
|
{
|
|
GstV4l2Transform *self = GST_V4L2_TRANSFORM (trans);
|
|
gboolean ret = FALSE;
|
|
|
|
GST_DEBUG_OBJECT (self, "called");
|
|
|
|
if (decide_query == NULL)
|
|
ret = TRUE;
|
|
else
|
|
ret = gst_v4l2_object_propose_allocation (self->v4l2output, query);
|
|
|
|
if (ret)
|
|
ret = GST_BASE_TRANSFORM_CLASS (parent_class)->propose_allocation (trans,
|
|
decide_query, query);
|
|
|
|
return ret;
|
|
}
|
|
|
|
/* copies the given caps */
|
|
static GstCaps *
|
|
gst_v4l2_transform_caps_remove_format_info (GstCaps * caps)
|
|
{
|
|
GstStructure *st;
|
|
GstCapsFeatures *f;
|
|
gint i, n;
|
|
GstCaps *res;
|
|
|
|
res = gst_caps_new_empty ();
|
|
|
|
n = gst_caps_get_size (caps);
|
|
for (i = 0; i < n; i++) {
|
|
st = gst_caps_get_structure (caps, i);
|
|
f = gst_caps_get_features (caps, i);
|
|
|
|
/* If this is already expressed by the existing caps
|
|
* skip this structure */
|
|
if (i > 0 && gst_caps_is_subset_structure_full (res, st, f))
|
|
continue;
|
|
|
|
st = gst_structure_copy (st);
|
|
/* Only remove format info for the cases when we can actually convert */
|
|
if (!gst_caps_features_is_any (f)
|
|
&& gst_caps_features_is_equal (f,
|
|
GST_CAPS_FEATURES_MEMORY_SYSTEM_MEMORY))
|
|
gst_structure_remove_fields (st, "format", "colorimetry", "chroma-site",
|
|
NULL);
|
|
|
|
gst_caps_append_structure_full (res, st, gst_caps_features_copy (f));
|
|
}
|
|
|
|
return res;
|
|
}
|
|
|
|
/* The caps can be transformed into any other caps with format info removed.
|
|
* However, we should prefer passthrough, so if passthrough is possible,
|
|
* put it first in the list. */
|
|
static GstCaps *
|
|
gst_v4l2_transform_transform_caps (GstBaseTransform * btrans,
|
|
GstPadDirection direction, GstCaps * caps, GstCaps * filter)
|
|
{
|
|
GstCaps *tmp, *tmp2;
|
|
GstCaps *result;
|
|
|
|
/* Get all possible caps that we can transform to */
|
|
tmp = gst_v4l2_transform_caps_remove_format_info (caps);
|
|
|
|
if (filter) {
|
|
tmp2 = gst_caps_intersect_full (filter, tmp, GST_CAPS_INTERSECT_FIRST);
|
|
gst_caps_unref (tmp);
|
|
tmp = tmp2;
|
|
}
|
|
|
|
result = tmp;
|
|
|
|
GST_DEBUG_OBJECT (btrans, "transformed %" GST_PTR_FORMAT " into %"
|
|
GST_PTR_FORMAT, caps, result);
|
|
|
|
return result;
|
|
}
|
|
|
|
static GstCaps *
|
|
gst_v4l2_transform_fixate_caps (GstBaseTransform * trans,
|
|
GstPadDirection direction, GstCaps * caps, GstCaps * othercaps)
|
|
{
|
|
GstCaps *result;
|
|
|
|
GST_DEBUG_OBJECT (trans, "trying to fixate othercaps %" GST_PTR_FORMAT
|
|
" based on caps %" GST_PTR_FORMAT, othercaps, caps);
|
|
|
|
result = gst_caps_intersect (othercaps, caps);
|
|
if (gst_caps_is_empty (result)) {
|
|
gst_caps_unref (result);
|
|
result = othercaps;
|
|
} else {
|
|
gst_caps_unref (othercaps);
|
|
}
|
|
|
|
GST_DEBUG_OBJECT (trans, "now fixating %" GST_PTR_FORMAT, result);
|
|
|
|
result = gst_caps_fixate (result);
|
|
|
|
return result;
|
|
}
|
|
|
|
static GstFlowReturn
|
|
gst_v4l2_transform_prepare_output_buffer (GstBaseTransform * trans,
|
|
GstBuffer * inbuf, GstBuffer ** outbuf)
|
|
{
|
|
GstV4l2Transform *self = GST_V4L2_TRANSFORM (trans);
|
|
GstBufferPool *pool = GST_BUFFER_POOL (self->v4l2output->pool);
|
|
GstFlowReturn ret = GST_FLOW_OK;
|
|
GstBaseTransformClass *bclass = GST_BASE_TRANSFORM_CLASS (parent_class);
|
|
|
|
if (gst_base_transform_is_passthrough (trans)) {
|
|
GST_DEBUG_OBJECT (self, "Passthrough, no need to do anything");
|
|
*outbuf = inbuf;
|
|
goto beach;
|
|
}
|
|
|
|
/* Ensure input internal pool is active */
|
|
if (!gst_buffer_pool_is_active (pool)) {
|
|
GstStructure *config = gst_buffer_pool_get_config (pool);
|
|
gst_buffer_pool_config_set_params (config, self->incaps,
|
|
self->v4l2output->info.size, 2, 2);
|
|
|
|
/* There is no reason to refuse this config */
|
|
if (!gst_buffer_pool_set_config (pool, config))
|
|
goto activate_failed;
|
|
|
|
if (!gst_buffer_pool_set_active (pool, TRUE))
|
|
goto activate_failed;
|
|
}
|
|
|
|
GST_DEBUG_OBJECT (self, "Queue input buffer");
|
|
ret = gst_v4l2_buffer_pool_process (GST_V4L2_BUFFER_POOL (pool), &inbuf);
|
|
if (G_UNLIKELY (ret != GST_FLOW_OK))
|
|
goto beach;
|
|
|
|
pool = gst_base_transform_get_buffer_pool (trans);
|
|
|
|
if (!gst_buffer_pool_set_active (pool, TRUE))
|
|
goto activate_failed;
|
|
|
|
GST_DEBUG_OBJECT (self, "Dequeue output buffer");
|
|
ret = gst_buffer_pool_acquire_buffer (pool, outbuf, NULL);
|
|
g_object_unref (pool);
|
|
|
|
if (ret != GST_FLOW_OK)
|
|
goto alloc_failed;
|
|
|
|
pool = self->v4l2capture->pool;
|
|
ret = gst_v4l2_buffer_pool_process (GST_V4L2_BUFFER_POOL (pool), outbuf);
|
|
|
|
if (ret != GST_FLOW_OK) {
|
|
gst_buffer_unref (*outbuf);
|
|
*outbuf = NULL;
|
|
}
|
|
|
|
if (bclass->copy_metadata)
|
|
if (!bclass->copy_metadata (trans, inbuf, *outbuf)) {
|
|
/* something failed, post a warning */
|
|
GST_ELEMENT_WARNING (self, STREAM, NOT_IMPLEMENTED,
|
|
("could not copy metadata"), (NULL));
|
|
}
|
|
|
|
beach:
|
|
return ret;
|
|
|
|
activate_failed:
|
|
GST_ELEMENT_ERROR (self, RESOURCE, SETTINGS,
|
|
("failed to activate bufferpool"), ("failed to activate bufferpool"));
|
|
g_object_unref (pool);
|
|
return GST_FLOW_ERROR;
|
|
|
|
alloc_failed:
|
|
GST_DEBUG_OBJECT (self, "could not allocate buffer from pool");
|
|
return ret;
|
|
}
|
|
|
|
static GstFlowReturn
|
|
gst_v4l2_transform_transform (GstBaseTransform * trans, GstBuffer * inbuf,
|
|
GstBuffer * outbuf)
|
|
{
|
|
/* Nothing to do */
|
|
return GST_FLOW_OK;
|
|
}
|
|
|
|
static gboolean
|
|
gst_v4l2_transform_sink_event (GstBaseTransform * trans, GstEvent * event)
|
|
{
|
|
GstV4l2Transform *self = GST_V4L2_TRANSFORM (trans);
|
|
gboolean ret;
|
|
|
|
/* Nothing to flush in passthrough */
|
|
if (gst_base_transform_is_passthrough (trans))
|
|
return GST_BASE_TRANSFORM_CLASS (parent_class)->sink_event (trans, event);
|
|
|
|
switch (GST_EVENT_TYPE (event)) {
|
|
case GST_EVENT_FLUSH_START:
|
|
GST_DEBUG_OBJECT (self, "flush start");
|
|
gst_v4l2_object_unlock (self->v4l2output);
|
|
gst_v4l2_object_unlock (self->v4l2capture);
|
|
break;
|
|
default:
|
|
break;
|
|
}
|
|
|
|
ret = GST_BASE_TRANSFORM_CLASS (parent_class)->sink_event (trans, event);
|
|
|
|
switch (GST_EVENT_TYPE (event)) {
|
|
case GST_EVENT_FLUSH_STOP:
|
|
/* Buffer should be back now */
|
|
GST_DEBUG_OBJECT (self, "flush stop");
|
|
gst_v4l2_object_unlock_stop (self->v4l2capture);
|
|
gst_v4l2_object_unlock_stop (self->v4l2output);
|
|
break;
|
|
default:
|
|
break;
|
|
}
|
|
|
|
return ret;
|
|
}
|
|
|
|
static GstStateChangeReturn
|
|
gst_v4l2_transform_change_state (GstElement * element,
|
|
GstStateChange transition)
|
|
{
|
|
GstV4l2Transform *self = GST_V4L2_TRANSFORM (element);
|
|
GstStateChangeReturn ret;
|
|
|
|
switch (transition) {
|
|
case GST_STATE_CHANGE_NULL_TO_READY:
|
|
if (!gst_v4l2_transform_open (self))
|
|
return GST_STATE_CHANGE_FAILURE;
|
|
break;
|
|
case GST_STATE_CHANGE_PAUSED_TO_READY:
|
|
gst_v4l2_object_unlock (self->v4l2output);
|
|
gst_v4l2_object_unlock (self->v4l2capture);
|
|
break;
|
|
default:
|
|
break;
|
|
}
|
|
|
|
ret = GST_ELEMENT_CLASS (parent_class)->change_state (element, transition);
|
|
|
|
switch (transition) {
|
|
case GST_STATE_CHANGE_READY_TO_NULL:
|
|
gst_v4l2_transform_close (self);
|
|
break;
|
|
default:
|
|
break;
|
|
}
|
|
|
|
return ret;
|
|
}
|
|
|
|
static void
|
|
gst_v4l2_transform_dispose (GObject * object)
|
|
{
|
|
GstV4l2Transform *self = GST_V4L2_TRANSFORM (object);
|
|
|
|
gst_caps_replace (&self->probed_sinkcaps, NULL);
|
|
gst_caps_replace (&self->probed_srccaps, NULL);
|
|
|
|
G_OBJECT_CLASS (parent_class)->dispose (object);
|
|
}
|
|
|
|
static void
|
|
gst_v4l2_transform_finalize (GObject * object)
|
|
{
|
|
GstV4l2Transform *self = GST_V4L2_TRANSFORM (object);
|
|
|
|
gst_v4l2_object_destroy (self->v4l2capture);
|
|
gst_v4l2_object_destroy (self->v4l2output);
|
|
|
|
G_OBJECT_CLASS (parent_class)->finalize (object);
|
|
}
|
|
|
|
static void
|
|
gst_v4l2_transform_init (GstV4l2Transform * self)
|
|
{
|
|
/* V4L2 object are created in subinstance_init */
|
|
/* enable QoS */
|
|
gst_base_transform_set_qos_enabled (GST_BASE_TRANSFORM (self), TRUE);
|
|
}
|
|
|
|
static void
|
|
gst_v4l2_transform_subinstance_init (GTypeInstance * instance, gpointer g_class)
|
|
{
|
|
GstV4l2TransformClass *klass = GST_V4L2_TRANSFORM_CLASS (g_class);
|
|
GstV4l2Transform *self = GST_V4L2_TRANSFORM (instance);
|
|
|
|
self->v4l2output = gst_v4l2_object_new (GST_ELEMENT (self),
|
|
V4L2_BUF_TYPE_VIDEO_OUTPUT, klass->default_device,
|
|
gst_v4l2_get_output, gst_v4l2_set_output, NULL);
|
|
self->v4l2output->no_initial_format = TRUE;
|
|
self->v4l2output->keep_aspect = FALSE;
|
|
|
|
self->v4l2capture = gst_v4l2_object_new (GST_ELEMENT (self),
|
|
V4L2_BUF_TYPE_VIDEO_CAPTURE, klass->default_device,
|
|
gst_v4l2_get_input, gst_v4l2_set_input, NULL);
|
|
self->v4l2capture->no_initial_format = TRUE;
|
|
self->v4l2output->keep_aspect = FALSE;
|
|
}
|
|
|
|
static void
|
|
gst_v4l2_transform_class_init (GstV4l2TransformClass * klass)
|
|
{
|
|
GstElementClass *element_class;
|
|
GObjectClass *gobject_class;
|
|
GstBaseTransformClass *base_transform_class;
|
|
|
|
element_class = (GstElementClass *) klass;
|
|
gobject_class = (GObjectClass *) klass;
|
|
base_transform_class = (GstBaseTransformClass *) klass;
|
|
|
|
GST_DEBUG_CATEGORY_INIT (gst_v4l2_transform_debug, "v4l2transform", 0,
|
|
"V4L2 Converter");
|
|
|
|
gst_element_class_set_static_metadata (element_class,
|
|
"V4L2 Video Converter",
|
|
"Filter/Converter/Video",
|
|
"Transform streams via V4L2 API",
|
|
"Nicolas Dufresne <nicolas.dufresne@collabora.com>");
|
|
|
|
gobject_class->dispose = GST_DEBUG_FUNCPTR (gst_v4l2_transform_dispose);
|
|
gobject_class->finalize = GST_DEBUG_FUNCPTR (gst_v4l2_transform_finalize);
|
|
gobject_class->set_property =
|
|
GST_DEBUG_FUNCPTR (gst_v4l2_transform_set_property);
|
|
gobject_class->get_property =
|
|
GST_DEBUG_FUNCPTR (gst_v4l2_transform_get_property);
|
|
|
|
base_transform_class->stop = GST_DEBUG_FUNCPTR (gst_v4l2_transform_stop);
|
|
base_transform_class->set_caps =
|
|
GST_DEBUG_FUNCPTR (gst_v4l2_transform_set_caps);
|
|
base_transform_class->query = GST_DEBUG_FUNCPTR (gst_v4l2_transform_query);
|
|
base_transform_class->sink_event =
|
|
GST_DEBUG_FUNCPTR (gst_v4l2_transform_sink_event);
|
|
base_transform_class->decide_allocation =
|
|
GST_DEBUG_FUNCPTR (gst_v4l2_transform_decide_allocation);
|
|
base_transform_class->propose_allocation =
|
|
GST_DEBUG_FUNCPTR (gst_v4l2_transform_propose_allocation);
|
|
base_transform_class->transform_caps =
|
|
GST_DEBUG_FUNCPTR (gst_v4l2_transform_transform_caps);
|
|
base_transform_class->fixate_caps =
|
|
GST_DEBUG_FUNCPTR (gst_v4l2_transform_fixate_caps);
|
|
base_transform_class->prepare_output_buffer =
|
|
GST_DEBUG_FUNCPTR (gst_v4l2_transform_prepare_output_buffer);
|
|
base_transform_class->transform =
|
|
GST_DEBUG_FUNCPTR (gst_v4l2_transform_transform);
|
|
|
|
base_transform_class->passthrough_on_same_caps = TRUE;
|
|
|
|
element_class->change_state =
|
|
GST_DEBUG_FUNCPTR (gst_v4l2_transform_change_state);
|
|
|
|
gst_v4l2_object_install_m2m_properties_helper (gobject_class);
|
|
}
|
|
|
|
static void
|
|
gst_v4l2_transform_subclass_init (gpointer g_class, gpointer data)
|
|
{
|
|
GstV4l2TransformClass *klass = GST_V4L2_TRANSFORM_CLASS (g_class);
|
|
GstElementClass *element_class = GST_ELEMENT_CLASS (g_class);
|
|
GstV4l2TransformCData *cdata = data;
|
|
|
|
klass->default_device = cdata->device;
|
|
|
|
/* Note: gst_pad_template_new() take the floating ref from the caps */
|
|
gst_element_class_add_pad_template (element_class,
|
|
gst_pad_template_new ("sink", GST_PAD_SINK, GST_PAD_ALWAYS,
|
|
cdata->sink_caps));
|
|
gst_element_class_add_pad_template (element_class,
|
|
gst_pad_template_new ("src", GST_PAD_SRC, GST_PAD_ALWAYS,
|
|
cdata->src_caps));
|
|
|
|
g_free (cdata);
|
|
}
|
|
|
|
/* Probing functions */
|
|
gboolean
|
|
gst_v4l2_is_transform (GstCaps * sink_caps, GstCaps * src_caps)
|
|
{
|
|
gboolean ret = FALSE;
|
|
|
|
if (gst_caps_is_subset (sink_caps, gst_v4l2_object_get_raw_caps ())
|
|
&& gst_caps_is_subset (src_caps, gst_v4l2_object_get_raw_caps ()))
|
|
ret = TRUE;
|
|
|
|
return ret;
|
|
}
|
|
|
|
gboolean
|
|
gst_v4l2_transform_register (GstPlugin * plugin, const gchar * basename,
|
|
const gchar * device_path, GstCaps * sink_caps, GstCaps * src_caps)
|
|
{
|
|
GTypeQuery type_query;
|
|
GTypeInfo type_info = { 0, };
|
|
GType type, subtype;
|
|
gchar *type_name;
|
|
GstV4l2TransformCData *cdata;
|
|
|
|
cdata = g_new0 (GstV4l2TransformCData, 1);
|
|
cdata->device = g_strdup (device_path);
|
|
cdata->sink_caps = gst_caps_ref (sink_caps);
|
|
cdata->src_caps = gst_caps_ref (src_caps);
|
|
|
|
type = gst_v4l2_transform_get_type ();
|
|
g_type_query (type, &type_query);
|
|
memset (&type_info, 0, sizeof (type_info));
|
|
type_info.class_size = type_query.class_size;
|
|
type_info.instance_size = type_query.instance_size;
|
|
type_info.class_init = gst_v4l2_transform_subclass_init;
|
|
type_info.class_data = cdata;
|
|
type_info.instance_init = gst_v4l2_transform_subinstance_init;
|
|
|
|
type_name = g_strdup_printf ("v4l2%sconvert", basename);
|
|
subtype = g_type_register_static (type, type_name, &type_info, 0);
|
|
|
|
gst_element_register (plugin, type_name, GST_RANK_NONE, subtype);
|
|
|
|
g_free (type_name);
|
|
|
|
return TRUE;
|
|
}
|