mirror of
https://gitlab.freedesktop.org/gstreamer/gstreamer.git
synced 2024-12-24 09:10:36 +00:00
gst/elements/gstcapsfilter.c: Reimplement using basetransform, fixes buffer_alloc proxying among other things.
Original commit message from CVS: 2005-08-04 Andy Wingo <wingo@pobox.com> * gst/elements/gstcapsfilter.c: Reimplement using basetransform, fixes buffer_alloc proxying among other things. * gst/base/gstbasetransform.c: * gst/base/gstbasetransform.h: Revert patch to gstbasetransform from 7-28 removing delay_configure. * gst/base/gstbasetransform.h (GstBaseTransformClass.get_size): * gst/base/gstbasetransform.c (gst_base_transform_get_size): Semantics changed, should return not the size of the output buffer but the byte size of a buffer with a given caps. * gst/base/gstbasetransform.c (gst_base_transform_getcaps): Better debug object. (gst_base_transform_configure_caps): Don't set out_size here: (in, out) are not the pad caps until setcaps finishes. (gst_base_transform_buffer_alloc): Proxy the buffer_alloc for the not-in-place case as well. Deal with changing from in-place to not-in-place within calling pad_alloc_buffer. Still a bit concerned about the overhead here...
This commit is contained in:
parent
8ff106a019
commit
e20ed97cb2
7 changed files with 353 additions and 258 deletions
29
ChangeLog
29
ChangeLog
|
@ -1,3 +1,32 @@
|
||||||
|
2005-08-04 Andy Wingo <wingo@pobox.com>
|
||||||
|
|
||||||
|
* gst/elements/gstcapsfilter.c: Reimplement using basetransform,
|
||||||
|
fixes buffer_alloc proxying among other things.
|
||||||
|
|
||||||
|
* gst/base/gstbasetransform.c:
|
||||||
|
* gst/base/gstbasetransform.h:
|
||||||
|
Revert patch to gstbasetransform from 7-28 removing
|
||||||
|
delay_configure.
|
||||||
|
|
||||||
|
* gst/base/gstbasetransform.h (GstBaseTransformClass.get_size):
|
||||||
|
* gst/base/gstbasetransform.c (gst_base_transform_get_size):
|
||||||
|
Semantics changed, should return not the size of the output buffer
|
||||||
|
but the byte size of a buffer with a given caps.
|
||||||
|
|
||||||
|
* gst/base/gstbasetransform.c (gst_base_transform_getcaps): Better
|
||||||
|
debug object.
|
||||||
|
(gst_base_transform_configure_caps): Don't set out_size here: (in,
|
||||||
|
out) are not the pad caps until setcaps finishes.
|
||||||
|
(gst_base_transform_buffer_alloc): Proxy the buffer_alloc for the
|
||||||
|
not-in-place case as well. Deal with changing from in-place to
|
||||||
|
not-in-place within calling pad_alloc_buffer. Still a bit
|
||||||
|
concerned about the overhead here...
|
||||||
|
|
||||||
|
2005-08-03 Andy Wingo <wingo@pobox.com>
|
||||||
|
|
||||||
|
* gst/base/gstbasetransform.c (gst_base_transform_setcaps): Not
|
||||||
|
fixating is an error.
|
||||||
|
|
||||||
2005-08-04 Edward Hervey <edward@fluendo.com>
|
2005-08-04 Edward Hervey <edward@fluendo.com>
|
||||||
|
|
||||||
* gst/base/gstadapter.h:
|
* gst/base/gstadapter.h:
|
||||||
|
|
|
@ -110,7 +110,8 @@ static gboolean gst_base_transform_src_activate_pull (GstPad * pad,
|
||||||
gboolean active);
|
gboolean active);
|
||||||
static gboolean gst_base_transform_sink_activate_push (GstPad * pad,
|
static gboolean gst_base_transform_sink_activate_push (GstPad * pad,
|
||||||
gboolean active);
|
gboolean active);
|
||||||
static guint gst_base_transform_get_size (GstBaseTransform * trans);
|
static guint gst_base_transform_get_size (GstBaseTransform * trans,
|
||||||
|
GstCaps * caps);
|
||||||
|
|
||||||
static GstElementStateReturn gst_base_transform_change_state (GstElement *
|
static GstElementStateReturn gst_base_transform_change_state (GstElement *
|
||||||
element);
|
element);
|
||||||
|
@ -264,26 +265,26 @@ gst_base_transform_getcaps (GstPad * pad)
|
||||||
GstCaps *temp;
|
GstCaps *temp;
|
||||||
const GstCaps *templ;
|
const GstCaps *templ;
|
||||||
|
|
||||||
GST_DEBUG_OBJECT (trans, "peer caps %" GST_PTR_FORMAT, caps);
|
GST_DEBUG_OBJECT (pad, "peer caps %" GST_PTR_FORMAT, caps);
|
||||||
|
|
||||||
/* filtered against our padtemplate */
|
/* filtered against our padtemplate */
|
||||||
templ = gst_pad_get_pad_template_caps (otherpad);
|
templ = gst_pad_get_pad_template_caps (otherpad);
|
||||||
GST_DEBUG_OBJECT (trans, "our template %" GST_PTR_FORMAT, templ);
|
GST_DEBUG_OBJECT (pad, "our template %" GST_PTR_FORMAT, templ);
|
||||||
temp = gst_caps_intersect (caps, templ);
|
temp = gst_caps_intersect (caps, templ);
|
||||||
GST_DEBUG_OBJECT (trans, "intersected %" GST_PTR_FORMAT, temp);
|
GST_DEBUG_OBJECT (pad, "intersected %" GST_PTR_FORMAT, temp);
|
||||||
gst_caps_unref (caps);
|
gst_caps_unref (caps);
|
||||||
/* then see what we can tranform this to */
|
/* then see what we can tranform this to */
|
||||||
caps = gst_base_transform_transform_caps (trans, otherpad, temp);
|
caps = gst_base_transform_transform_caps (trans, otherpad, temp);
|
||||||
GST_DEBUG_OBJECT (trans, "transformed %" GST_PTR_FORMAT, caps);
|
GST_DEBUG_OBJECT (pad, "transformed %" GST_PTR_FORMAT, caps);
|
||||||
gst_caps_unref (temp);
|
gst_caps_unref (temp);
|
||||||
if (caps == NULL)
|
if (caps == NULL)
|
||||||
goto done;
|
goto done;
|
||||||
|
|
||||||
/* and filter against the template again */
|
/* and filter against the template again */
|
||||||
templ = gst_pad_get_pad_template_caps (pad);
|
templ = gst_pad_get_pad_template_caps (pad);
|
||||||
GST_DEBUG_OBJECT (trans, "our template %" GST_PTR_FORMAT, templ);
|
GST_DEBUG_OBJECT (pad, "our template %" GST_PTR_FORMAT, templ);
|
||||||
temp = gst_caps_intersect (caps, templ);
|
temp = gst_caps_intersect (caps, templ);
|
||||||
GST_DEBUG_OBJECT (trans, "intersected %" GST_PTR_FORMAT, temp);
|
GST_DEBUG_OBJECT (pad, "intersected %" GST_PTR_FORMAT, temp);
|
||||||
gst_caps_unref (caps);
|
gst_caps_unref (caps);
|
||||||
/* this is what we can do */
|
/* this is what we can do */
|
||||||
caps = temp;
|
caps = temp;
|
||||||
|
@ -314,11 +315,6 @@ gst_base_transform_configure_caps (GstBaseTransform * trans, GstCaps * in,
|
||||||
ret = klass->set_caps (trans, in, out);
|
ret = klass->set_caps (trans, in, out);
|
||||||
}
|
}
|
||||||
|
|
||||||
/* if all goes well, get the size of the output buffer */
|
|
||||||
if (ret) {
|
|
||||||
trans->out_size = gst_base_transform_get_size (trans);
|
|
||||||
GST_DEBUG_OBJECT (trans, "output buffer size %d", trans->out_size);
|
|
||||||
}
|
|
||||||
return ret;
|
return ret;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -427,10 +423,12 @@ gst_base_transform_setcaps (GstPad * pad, GstCaps * caps)
|
||||||
GST_DEBUG_OBJECT (trans, "in_place: %d", trans->in_place);
|
GST_DEBUG_OBJECT (trans, "in_place: %d", trans->in_place);
|
||||||
|
|
||||||
/* see if we have to configure the element now */
|
/* see if we have to configure the element now */
|
||||||
if (pad == trans->sinkpad)
|
if (!trans->delay_configure) {
|
||||||
ret = gst_base_transform_configure_caps (trans, caps, othercaps);
|
if (pad == trans->sinkpad)
|
||||||
else
|
ret = gst_base_transform_configure_caps (trans, caps, othercaps);
|
||||||
ret = gst_base_transform_configure_caps (trans, othercaps, caps);
|
else
|
||||||
|
ret = gst_base_transform_configure_caps (trans, othercaps, caps);
|
||||||
|
}
|
||||||
|
|
||||||
done:
|
done:
|
||||||
if (otherpeer)
|
if (otherpeer)
|
||||||
|
@ -460,7 +458,7 @@ no_transform_possible:
|
||||||
}
|
}
|
||||||
could_not_fixate:
|
could_not_fixate:
|
||||||
{
|
{
|
||||||
GST_DEBUG_OBJECT (trans, "FAILED to fixate %" GST_PTR_FORMAT, othercaps);
|
GST_ERROR_OBJECT (trans, "FAILED to fixate %" GST_PTR_FORMAT, othercaps);
|
||||||
ret = FALSE;
|
ret = FALSE;
|
||||||
goto done;
|
goto done;
|
||||||
}
|
}
|
||||||
|
@ -474,15 +472,15 @@ peer_no_accept:
|
||||||
}
|
}
|
||||||
|
|
||||||
static guint
|
static guint
|
||||||
gst_base_transform_get_size (GstBaseTransform * trans)
|
gst_base_transform_get_size (GstBaseTransform * trans, GstCaps * caps)
|
||||||
{
|
{
|
||||||
guint res = -1;
|
guint res = -1;
|
||||||
GstBaseTransformClass *bclass;
|
GstBaseTransformClass *bclass;
|
||||||
|
|
||||||
bclass = GST_BASE_TRANSFORM_GET_CLASS (trans);
|
bclass = GST_BASE_TRANSFORM_GET_CLASS (trans);
|
||||||
if (bclass->get_size) {
|
if (bclass->get_size) {
|
||||||
res = bclass->get_size (trans);
|
res = bclass->get_size (trans, caps);
|
||||||
GST_DEBUG_OBJECT (trans, "get size function returned %d", res);
|
GST_DEBUG_OBJECT (trans, "get size(%p) returned %d", caps, res);
|
||||||
}
|
}
|
||||||
|
|
||||||
return res;
|
return res;
|
||||||
|
@ -494,21 +492,74 @@ gst_base_transform_buffer_alloc (GstPad * pad, guint64 offset, guint size,
|
||||||
{
|
{
|
||||||
GstBaseTransform *trans;
|
GstBaseTransform *trans;
|
||||||
GstFlowReturn res;
|
GstFlowReturn res;
|
||||||
|
guint got_size;
|
||||||
|
|
||||||
trans = GST_BASE_TRANSFORM (gst_pad_get_parent (pad));
|
trans = GST_BASE_TRANSFORM (gst_pad_get_parent (pad));
|
||||||
|
|
||||||
|
*buf = NULL;
|
||||||
|
|
||||||
if (trans->in_place) {
|
if (trans->in_place) {
|
||||||
/* we can only proxy the bufferpool if we do in_place transforms */
|
/* request a buffer with the same caps */
|
||||||
res = gst_pad_alloc_buffer (trans->srcpad, offset, size, caps, buf);
|
res = gst_pad_alloc_buffer (trans->srcpad, offset, size, caps, buf);
|
||||||
} else {
|
} else {
|
||||||
/* else let the default alloc function allocate a buffer */
|
/* if we are configured, request a buffer with the src caps */
|
||||||
*buf = NULL;
|
GstCaps *srccaps = gst_pad_get_negotiated_caps (trans->srcpad);
|
||||||
|
|
||||||
|
if (!srccaps)
|
||||||
|
goto not_configured;
|
||||||
|
|
||||||
|
got_size = gst_base_transform_get_size (trans, srccaps);
|
||||||
|
if (got_size == -1) {
|
||||||
|
gst_caps_unref (srccaps);
|
||||||
|
goto unknown_size;
|
||||||
|
}
|
||||||
|
|
||||||
|
res = gst_pad_alloc_buffer (trans->srcpad, offset, got_size, srccaps, buf);
|
||||||
|
gst_caps_unref (srccaps);
|
||||||
|
}
|
||||||
|
|
||||||
|
if (res == GST_FLOW_OK && !trans->in_place) {
|
||||||
|
/* note that we might have been in place before, but calling the
|
||||||
|
alloc_buffer caused setcaps to switch us out of in_place -- in any case
|
||||||
|
the alloc_buffer served to transmit caps information but we can't use the
|
||||||
|
buffer. fall through and allocate a buffer corresponding to our sink
|
||||||
|
caps, if any */
|
||||||
|
GstCaps *sinkcaps = gst_pad_get_negotiated_caps (trans->sinkpad);
|
||||||
|
|
||||||
|
if (!sinkcaps)
|
||||||
|
goto not_configured;
|
||||||
|
|
||||||
|
got_size = gst_base_transform_get_size (trans, sinkcaps);
|
||||||
|
if (got_size == -1) {
|
||||||
|
gst_caps_unref (sinkcaps);
|
||||||
|
goto unknown_size;
|
||||||
|
}
|
||||||
|
|
||||||
|
*buf = gst_buffer_new_and_alloc (got_size);
|
||||||
|
gst_buffer_set_caps (*buf, sinkcaps);
|
||||||
|
GST_BUFFER_OFFSET (*buf) = offset;
|
||||||
res = GST_FLOW_OK;
|
res = GST_FLOW_OK;
|
||||||
|
|
||||||
|
gst_caps_unref (sinkcaps);
|
||||||
}
|
}
|
||||||
|
|
||||||
gst_object_unref (trans);
|
gst_object_unref (trans);
|
||||||
|
|
||||||
return res;
|
return res;
|
||||||
|
|
||||||
|
not_configured:
|
||||||
|
{
|
||||||
|
/* let the default allocator handle it */
|
||||||
|
*buf = NULL;
|
||||||
|
gst_object_unref (trans);
|
||||||
|
return GST_FLOW_OK;
|
||||||
|
}
|
||||||
|
unknown_size:
|
||||||
|
{
|
||||||
|
/* let the default allocator handle it */
|
||||||
|
*buf = NULL;
|
||||||
|
gst_object_unref (trans);
|
||||||
|
return GST_FLOW_OK;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
|
@ -584,16 +635,25 @@ gst_base_transform_handle_buffer (GstBaseTransform * trans, GstBuffer * inbuf,
|
||||||
/* figure out the output size */
|
/* figure out the output size */
|
||||||
if (trans->out_size == -1) {
|
if (trans->out_size == -1) {
|
||||||
/* ask subclass */
|
/* ask subclass */
|
||||||
if ((trans->out_size = gst_base_transform_get_size (trans)) == -1)
|
trans->out_size = gst_base_transform_get_size (trans,
|
||||||
/* else we have an error */
|
GST_PAD_CAPS (trans->srcpad));
|
||||||
|
if (trans->out_size == -1)
|
||||||
|
/* we have an error */
|
||||||
goto no_size;
|
goto no_size;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/* we cannot reconfigure the element yet as we are still processing
|
||||||
|
* the old buffer. We will therefore delay the reconfiguration of the
|
||||||
|
* element until we have processed this last buffer. */
|
||||||
|
trans->delay_configure = TRUE;
|
||||||
|
|
||||||
/* no in place transform, get buffer, this might renegotiate. */
|
/* no in place transform, get buffer, this might renegotiate. */
|
||||||
ret = gst_pad_alloc_buffer (trans->srcpad,
|
ret = gst_pad_alloc_buffer (trans->srcpad,
|
||||||
GST_BUFFER_OFFSET (inbuf), trans->out_size,
|
GST_BUFFER_OFFSET (inbuf), trans->out_size,
|
||||||
GST_PAD_CAPS (trans->srcpad), outbuf);
|
GST_PAD_CAPS (trans->srcpad), outbuf);
|
||||||
|
|
||||||
|
trans->delay_configure = FALSE;
|
||||||
|
|
||||||
if (ret != GST_FLOW_OK)
|
if (ret != GST_FLOW_OK)
|
||||||
goto no_buffer;
|
goto no_buffer;
|
||||||
|
|
||||||
|
|
|
@ -54,6 +54,7 @@ struct _GstBaseTransform {
|
||||||
|
|
||||||
gboolean in_place;
|
gboolean in_place;
|
||||||
guint out_size;
|
guint out_size;
|
||||||
|
gboolean delay_configure;
|
||||||
};
|
};
|
||||||
|
|
||||||
struct _GstBaseTransformClass {
|
struct _GstBaseTransformClass {
|
||||||
|
@ -70,8 +71,8 @@ struct _GstBaseTransformClass {
|
||||||
gboolean (*set_caps) (GstBaseTransform *trans, GstCaps *incaps,
|
gboolean (*set_caps) (GstBaseTransform *trans, GstCaps *incaps,
|
||||||
GstCaps *outcaps);
|
GstCaps *outcaps);
|
||||||
|
|
||||||
/* get the size of the output buffer, -1 on error */
|
/* get the byte size of a given caps, -1 on error */
|
||||||
guint (*get_size) (GstBaseTransform *trans);
|
guint (*get_size) (GstBaseTransform *trans, GstCaps *caps);
|
||||||
|
|
||||||
/* start and stop processing, ideal for opening/closing the resource */
|
/* start and stop processing, ideal for opening/closing the resource */
|
||||||
gboolean (*start) (GstBaseTransform *trans);
|
gboolean (*start) (GstBaseTransform *trans);
|
||||||
|
|
|
@ -20,16 +20,19 @@
|
||||||
* Boston, MA 02111-1307, USA.
|
* Boston, MA 02111-1307, USA.
|
||||||
*/
|
*/
|
||||||
|
|
||||||
|
|
||||||
#include <stdlib.h>
|
|
||||||
|
|
||||||
#ifdef HAVE_CONFIG_H
|
#ifdef HAVE_CONFIG_H
|
||||||
# include "config.h"
|
#include "config.h"
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
#include "../gst-i18n-lib.h"
|
#include "../gst-i18n-lib.h"
|
||||||
#include <gst/gstmarshal.h>
|
|
||||||
#include <gst/gst.h>
|
#include <gst/gst.h>
|
||||||
|
#include <gst/base/gstbasetransform.h>
|
||||||
|
|
||||||
|
|
||||||
|
GstElementDetails gst_capsfilter_details = GST_ELEMENT_DETAILS ("CapsFilter",
|
||||||
|
"Generic",
|
||||||
|
"Pass data without modification, limiting formats",
|
||||||
|
"David Schleef <ds@schleef.org>");
|
||||||
|
|
||||||
|
|
||||||
#define GST_TYPE_CAPSFILTER \
|
#define GST_TYPE_CAPSFILTER \
|
||||||
|
@ -48,23 +51,25 @@ typedef struct _GstCapsFilterClass GstCapsFilterClass;
|
||||||
|
|
||||||
struct _GstCapsFilter
|
struct _GstCapsFilter
|
||||||
{
|
{
|
||||||
GstElement element;
|
GstBaseTransform trans;
|
||||||
|
|
||||||
GstPad *srcpad;
|
|
||||||
GstPad *sinkpad;
|
|
||||||
|
|
||||||
GstCaps *filter_caps;
|
GstCaps *filter_caps;
|
||||||
};
|
};
|
||||||
|
|
||||||
struct _GstCapsFilterClass
|
struct _GstCapsFilterClass
|
||||||
{
|
{
|
||||||
GstElementClass element_class;
|
GstBaseTransformClass trans_class;
|
||||||
|
|
||||||
};
|
};
|
||||||
|
|
||||||
GType gst_capsfilter_get_type (void);
|
GType gst_capsfilter_get_type (void);
|
||||||
|
|
||||||
|
|
||||||
|
enum
|
||||||
|
{
|
||||||
|
PROP_0,
|
||||||
|
PROP_FILTER_CAPS
|
||||||
|
};
|
||||||
|
|
||||||
|
|
||||||
static GstStaticPadTemplate sinktemplate = GST_STATIC_PAD_TEMPLATE ("sink",
|
static GstStaticPadTemplate sinktemplate = GST_STATIC_PAD_TEMPLATE ("sink",
|
||||||
GST_PAD_SINK,
|
GST_PAD_SINK,
|
||||||
|
@ -76,133 +81,74 @@ static GstStaticPadTemplate srctemplate = GST_STATIC_PAD_TEMPLATE ("src",
|
||||||
GST_PAD_ALWAYS,
|
GST_PAD_ALWAYS,
|
||||||
GST_STATIC_CAPS_ANY);
|
GST_STATIC_CAPS_ANY);
|
||||||
|
|
||||||
|
|
||||||
GST_DEBUG_CATEGORY_STATIC (gst_capsfilter_debug);
|
GST_DEBUG_CATEGORY_STATIC (gst_capsfilter_debug);
|
||||||
#define GST_CAT_DEFAULT gst_capsfilter_debug
|
#define GST_CAT_DEFAULT gst_capsfilter_debug
|
||||||
|
|
||||||
GstElementDetails gst_capsfilter_details = GST_ELEMENT_DETAILS ("CapsFilter",
|
|
||||||
"Generic",
|
|
||||||
"Pass data without modification, limiting formats",
|
|
||||||
"David Schleef <ds@schleef.org>");
|
|
||||||
|
|
||||||
enum
|
|
||||||
{
|
|
||||||
PROP_0,
|
|
||||||
PROP_FILTER_CAPS
|
|
||||||
};
|
|
||||||
|
|
||||||
|
|
||||||
#define _do_init(bla) \
|
#define _do_init(bla) \
|
||||||
GST_DEBUG_CATEGORY_INIT (gst_capsfilter_debug, "capsfilter", 0, "capsfilter element");
|
GST_DEBUG_CATEGORY_INIT (gst_capsfilter_debug, "capsfilter", 0, \
|
||||||
|
"capsfilter element");
|
||||||
|
|
||||||
|
GST_BOILERPLATE_FULL (GstCapsFilter, gst_capsfilter, GstBaseTransform,
|
||||||
|
GST_TYPE_BASE_TRANSFORM, _do_init);
|
||||||
|
|
||||||
GST_BOILERPLATE_FULL (GstCapsFilter, gst_capsfilter, GstElement,
|
|
||||||
GST_TYPE_ELEMENT, _do_init);
|
|
||||||
|
|
||||||
static void gst_capsfilter_finalize (GObject * object);
|
|
||||||
static void gst_capsfilter_set_property (GObject * object, guint prop_id,
|
static void gst_capsfilter_set_property (GObject * object, guint prop_id,
|
||||||
const GValue * value, GParamSpec * pspec);
|
const GValue * value, GParamSpec * pspec);
|
||||||
static void gst_capsfilter_get_property (GObject * object, guint prop_id,
|
static void gst_capsfilter_get_property (GObject * object, guint prop_id,
|
||||||
GValue * value, GParamSpec * pspec);
|
GValue * value, GParamSpec * pspec);
|
||||||
|
static void gst_capsfilter_dispose (GObject * object);
|
||||||
|
static GstCaps *gst_capsfilter_transform_caps (GstBaseTransform * base,
|
||||||
|
GstPad * pad, GstCaps * caps);
|
||||||
|
static GstFlowReturn gst_capsfilter_transform_ip (GstBaseTransform * base,
|
||||||
|
GstBuffer * buf);
|
||||||
|
|
||||||
static GstCaps *gst_capsfilter_getcaps (GstPad * pad);
|
|
||||||
static GstFlowReturn gst_capsfilter_chain (GstPad * pad, GstBuffer * buf);
|
|
||||||
|
|
||||||
static void
|
static void
|
||||||
gst_capsfilter_base_init (gpointer g_class)
|
gst_capsfilter_base_init (gpointer g_class)
|
||||||
{
|
{
|
||||||
GstElementClass *gstelement_class = GST_ELEMENT_CLASS (g_class);
|
GstElementClass *element_class = GST_ELEMENT_CLASS (g_class);
|
||||||
|
|
||||||
gst_element_class_add_pad_template (gstelement_class,
|
gst_element_class_add_pad_template (element_class,
|
||||||
gst_static_pad_template_get (&srctemplate));
|
gst_static_pad_template_get (&srctemplate));
|
||||||
gst_element_class_add_pad_template (gstelement_class,
|
gst_element_class_add_pad_template (element_class,
|
||||||
gst_static_pad_template_get (&sinktemplate));
|
gst_static_pad_template_get (&sinktemplate));
|
||||||
gst_element_class_set_details (gstelement_class, &gst_capsfilter_details);
|
gst_element_class_set_details (element_class, &gst_capsfilter_details);
|
||||||
}
|
|
||||||
|
|
||||||
static void
|
|
||||||
gst_capsfilter_finalize (GObject * object)
|
|
||||||
{
|
|
||||||
GstCapsFilter *capsfilter;
|
|
||||||
|
|
||||||
capsfilter = GST_CAPSFILTER (object);
|
|
||||||
|
|
||||||
gst_caps_unref (capsfilter->filter_caps);
|
|
||||||
|
|
||||||
G_OBJECT_CLASS (parent_class)->finalize (object);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
static void
|
static void
|
||||||
gst_capsfilter_class_init (GstCapsFilterClass * klass)
|
gst_capsfilter_class_init (GstCapsFilterClass * klass)
|
||||||
{
|
{
|
||||||
GObjectClass *gobject_class;
|
GObjectClass *gobject_class;
|
||||||
GstElementClass *gstelement_class;
|
GstBaseTransformClass *trans_class;
|
||||||
|
|
||||||
gobject_class = G_OBJECT_CLASS (klass);
|
gobject_class = (GObjectClass *) klass;
|
||||||
gstelement_class = GST_ELEMENT_CLASS (klass);
|
gobject_class->set_property = gst_capsfilter_set_property;
|
||||||
|
gobject_class->get_property = gst_capsfilter_get_property;
|
||||||
gobject_class->set_property = GST_DEBUG_FUNCPTR (gst_capsfilter_set_property);
|
gobject_class->dispose = gst_capsfilter_dispose;
|
||||||
gobject_class->get_property = GST_DEBUG_FUNCPTR (gst_capsfilter_get_property);
|
|
||||||
gobject_class->finalize = GST_DEBUG_FUNCPTR (gst_capsfilter_finalize);
|
|
||||||
|
|
||||||
g_object_class_install_property (G_OBJECT_CLASS (klass), PROP_FILTER_CAPS,
|
g_object_class_install_property (G_OBJECT_CLASS (klass), PROP_FILTER_CAPS,
|
||||||
g_param_spec_boxed ("filter_caps", _("Filter caps"),
|
g_param_spec_boxed ("filter_caps", _("Filter caps"),
|
||||||
_("Restrict the possible allowed formats"),
|
_("Restrict the possible allowed formats"),
|
||||||
GST_TYPE_CAPS, G_PARAM_READWRITE));
|
GST_TYPE_CAPS, G_PARAM_READWRITE));
|
||||||
|
|
||||||
|
trans_class = (GstBaseTransformClass *) klass;
|
||||||
|
trans_class->transform_caps = gst_capsfilter_transform_caps;
|
||||||
|
trans_class->transform_ip = gst_capsfilter_transform_ip;
|
||||||
}
|
}
|
||||||
|
|
||||||
static void
|
static void
|
||||||
gst_capsfilter_init (GstCapsFilter * capsfilter)
|
gst_capsfilter_init (GstCapsFilter * filter)
|
||||||
{
|
{
|
||||||
gst_element_create_all_pads (GST_ELEMENT (capsfilter));
|
gst_base_transform_set_passthrough (GST_BASE_TRANSFORM (filter), TRUE);
|
||||||
|
filter->filter_caps = gst_caps_new_any ();
|
||||||
capsfilter->srcpad = gst_element_get_pad (GST_ELEMENT (capsfilter), "src");
|
|
||||||
capsfilter->sinkpad = gst_element_get_pad (GST_ELEMENT (capsfilter), "sink");
|
|
||||||
|
|
||||||
gst_pad_set_getcaps_function (capsfilter->srcpad, gst_capsfilter_getcaps);
|
|
||||||
|
|
||||||
gst_pad_set_getcaps_function (capsfilter->sinkpad, gst_capsfilter_getcaps);
|
|
||||||
gst_pad_set_chain_function (capsfilter->sinkpad, gst_capsfilter_chain);
|
|
||||||
|
|
||||||
capsfilter->filter_caps = gst_caps_new_any ();
|
|
||||||
}
|
|
||||||
|
|
||||||
static GstCaps *
|
|
||||||
gst_capsfilter_getcaps (GstPad * pad)
|
|
||||||
{
|
|
||||||
GstPad *otherpad;
|
|
||||||
GstCapsFilter *capsfilter = GST_CAPSFILTER (GST_OBJECT_PARENT (pad));
|
|
||||||
GstCaps *caps;
|
|
||||||
GstCaps *icaps;
|
|
||||||
|
|
||||||
otherpad = (pad == capsfilter->srcpad) ? capsfilter->sinkpad :
|
|
||||||
capsfilter->srcpad;
|
|
||||||
|
|
||||||
caps = gst_pad_peer_get_caps (otherpad);
|
|
||||||
if (caps == NULL)
|
|
||||||
caps = gst_caps_new_any ();
|
|
||||||
|
|
||||||
icaps = gst_caps_intersect (caps, capsfilter->filter_caps);
|
|
||||||
gst_caps_unref (caps);
|
|
||||||
|
|
||||||
return icaps;
|
|
||||||
}
|
|
||||||
|
|
||||||
static GstFlowReturn
|
|
||||||
gst_capsfilter_chain (GstPad * pad, GstBuffer * buf)
|
|
||||||
{
|
|
||||||
GstCapsFilter *capsfilter = GST_CAPSFILTER (GST_PAD_PARENT (pad));
|
|
||||||
|
|
||||||
gst_pad_push (capsfilter->srcpad, buf);
|
|
||||||
|
|
||||||
return GST_FLOW_OK;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
static void
|
static void
|
||||||
gst_capsfilter_set_property (GObject * object, guint prop_id,
|
gst_capsfilter_set_property (GObject * object, guint prop_id,
|
||||||
const GValue * value, GParamSpec * pspec)
|
const GValue * value, GParamSpec * pspec)
|
||||||
{
|
{
|
||||||
GstCapsFilter *capsfilter;
|
GstCapsFilter *capsfilter = GST_CAPSFILTER (object);
|
||||||
|
|
||||||
capsfilter = GST_CAPSFILTER (object);
|
|
||||||
|
|
||||||
switch (prop_id) {
|
switch (prop_id) {
|
||||||
case PROP_FILTER_CAPS:{
|
case PROP_FILTER_CAPS:{
|
||||||
|
@ -230,9 +176,7 @@ static void
|
||||||
gst_capsfilter_get_property (GObject * object, guint prop_id, GValue * value,
|
gst_capsfilter_get_property (GObject * object, guint prop_id, GValue * value,
|
||||||
GParamSpec * pspec)
|
GParamSpec * pspec)
|
||||||
{
|
{
|
||||||
GstCapsFilter *capsfilter;
|
GstCapsFilter *capsfilter = GST_CAPSFILTER (object);
|
||||||
|
|
||||||
capsfilter = GST_CAPSFILTER (object);
|
|
||||||
|
|
||||||
switch (prop_id) {
|
switch (prop_id) {
|
||||||
case PROP_FILTER_CAPS:
|
case PROP_FILTER_CAPS:
|
||||||
|
@ -243,3 +187,31 @@ gst_capsfilter_get_property (GObject * object, guint prop_id, GValue * value,
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
static void
|
||||||
|
gst_capsfilter_dispose (GObject * object)
|
||||||
|
{
|
||||||
|
GstCapsFilter *filter = GST_CAPSFILTER (object);
|
||||||
|
|
||||||
|
gst_caps_replace (&filter->filter_caps, NULL);
|
||||||
|
|
||||||
|
G_OBJECT_CLASS (parent_class)->dispose (object);
|
||||||
|
}
|
||||||
|
|
||||||
|
static GstCaps *
|
||||||
|
gst_capsfilter_transform_caps (GstBaseTransform * base, GstPad * pad,
|
||||||
|
GstCaps * caps)
|
||||||
|
{
|
||||||
|
GstCapsFilter *capsfilter = GST_CAPSFILTER (base);
|
||||||
|
GstCaps *ret;
|
||||||
|
|
||||||
|
ret = gst_caps_intersect (caps, capsfilter->filter_caps);
|
||||||
|
|
||||||
|
return ret;
|
||||||
|
}
|
||||||
|
|
||||||
|
static GstFlowReturn
|
||||||
|
gst_capsfilter_transform_ip (GstBaseTransform * base, GstBuffer * buf)
|
||||||
|
{
|
||||||
|
return GST_FLOW_OK;
|
||||||
|
}
|
||||||
|
|
|
@ -110,7 +110,8 @@ static gboolean gst_base_transform_src_activate_pull (GstPad * pad,
|
||||||
gboolean active);
|
gboolean active);
|
||||||
static gboolean gst_base_transform_sink_activate_push (GstPad * pad,
|
static gboolean gst_base_transform_sink_activate_push (GstPad * pad,
|
||||||
gboolean active);
|
gboolean active);
|
||||||
static guint gst_base_transform_get_size (GstBaseTransform * trans);
|
static guint gst_base_transform_get_size (GstBaseTransform * trans,
|
||||||
|
GstCaps * caps);
|
||||||
|
|
||||||
static GstElementStateReturn gst_base_transform_change_state (GstElement *
|
static GstElementStateReturn gst_base_transform_change_state (GstElement *
|
||||||
element);
|
element);
|
||||||
|
@ -264,26 +265,26 @@ gst_base_transform_getcaps (GstPad * pad)
|
||||||
GstCaps *temp;
|
GstCaps *temp;
|
||||||
const GstCaps *templ;
|
const GstCaps *templ;
|
||||||
|
|
||||||
GST_DEBUG_OBJECT (trans, "peer caps %" GST_PTR_FORMAT, caps);
|
GST_DEBUG_OBJECT (pad, "peer caps %" GST_PTR_FORMAT, caps);
|
||||||
|
|
||||||
/* filtered against our padtemplate */
|
/* filtered against our padtemplate */
|
||||||
templ = gst_pad_get_pad_template_caps (otherpad);
|
templ = gst_pad_get_pad_template_caps (otherpad);
|
||||||
GST_DEBUG_OBJECT (trans, "our template %" GST_PTR_FORMAT, templ);
|
GST_DEBUG_OBJECT (pad, "our template %" GST_PTR_FORMAT, templ);
|
||||||
temp = gst_caps_intersect (caps, templ);
|
temp = gst_caps_intersect (caps, templ);
|
||||||
GST_DEBUG_OBJECT (trans, "intersected %" GST_PTR_FORMAT, temp);
|
GST_DEBUG_OBJECT (pad, "intersected %" GST_PTR_FORMAT, temp);
|
||||||
gst_caps_unref (caps);
|
gst_caps_unref (caps);
|
||||||
/* then see what we can tranform this to */
|
/* then see what we can tranform this to */
|
||||||
caps = gst_base_transform_transform_caps (trans, otherpad, temp);
|
caps = gst_base_transform_transform_caps (trans, otherpad, temp);
|
||||||
GST_DEBUG_OBJECT (trans, "transformed %" GST_PTR_FORMAT, caps);
|
GST_DEBUG_OBJECT (pad, "transformed %" GST_PTR_FORMAT, caps);
|
||||||
gst_caps_unref (temp);
|
gst_caps_unref (temp);
|
||||||
if (caps == NULL)
|
if (caps == NULL)
|
||||||
goto done;
|
goto done;
|
||||||
|
|
||||||
/* and filter against the template again */
|
/* and filter against the template again */
|
||||||
templ = gst_pad_get_pad_template_caps (pad);
|
templ = gst_pad_get_pad_template_caps (pad);
|
||||||
GST_DEBUG_OBJECT (trans, "our template %" GST_PTR_FORMAT, templ);
|
GST_DEBUG_OBJECT (pad, "our template %" GST_PTR_FORMAT, templ);
|
||||||
temp = gst_caps_intersect (caps, templ);
|
temp = gst_caps_intersect (caps, templ);
|
||||||
GST_DEBUG_OBJECT (trans, "intersected %" GST_PTR_FORMAT, temp);
|
GST_DEBUG_OBJECT (pad, "intersected %" GST_PTR_FORMAT, temp);
|
||||||
gst_caps_unref (caps);
|
gst_caps_unref (caps);
|
||||||
/* this is what we can do */
|
/* this is what we can do */
|
||||||
caps = temp;
|
caps = temp;
|
||||||
|
@ -314,11 +315,6 @@ gst_base_transform_configure_caps (GstBaseTransform * trans, GstCaps * in,
|
||||||
ret = klass->set_caps (trans, in, out);
|
ret = klass->set_caps (trans, in, out);
|
||||||
}
|
}
|
||||||
|
|
||||||
/* if all goes well, get the size of the output buffer */
|
|
||||||
if (ret) {
|
|
||||||
trans->out_size = gst_base_transform_get_size (trans);
|
|
||||||
GST_DEBUG_OBJECT (trans, "output buffer size %d", trans->out_size);
|
|
||||||
}
|
|
||||||
return ret;
|
return ret;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -427,10 +423,12 @@ gst_base_transform_setcaps (GstPad * pad, GstCaps * caps)
|
||||||
GST_DEBUG_OBJECT (trans, "in_place: %d", trans->in_place);
|
GST_DEBUG_OBJECT (trans, "in_place: %d", trans->in_place);
|
||||||
|
|
||||||
/* see if we have to configure the element now */
|
/* see if we have to configure the element now */
|
||||||
if (pad == trans->sinkpad)
|
if (!trans->delay_configure) {
|
||||||
ret = gst_base_transform_configure_caps (trans, caps, othercaps);
|
if (pad == trans->sinkpad)
|
||||||
else
|
ret = gst_base_transform_configure_caps (trans, caps, othercaps);
|
||||||
ret = gst_base_transform_configure_caps (trans, othercaps, caps);
|
else
|
||||||
|
ret = gst_base_transform_configure_caps (trans, othercaps, caps);
|
||||||
|
}
|
||||||
|
|
||||||
done:
|
done:
|
||||||
if (otherpeer)
|
if (otherpeer)
|
||||||
|
@ -460,7 +458,7 @@ no_transform_possible:
|
||||||
}
|
}
|
||||||
could_not_fixate:
|
could_not_fixate:
|
||||||
{
|
{
|
||||||
GST_DEBUG_OBJECT (trans, "FAILED to fixate %" GST_PTR_FORMAT, othercaps);
|
GST_ERROR_OBJECT (trans, "FAILED to fixate %" GST_PTR_FORMAT, othercaps);
|
||||||
ret = FALSE;
|
ret = FALSE;
|
||||||
goto done;
|
goto done;
|
||||||
}
|
}
|
||||||
|
@ -474,15 +472,15 @@ peer_no_accept:
|
||||||
}
|
}
|
||||||
|
|
||||||
static guint
|
static guint
|
||||||
gst_base_transform_get_size (GstBaseTransform * trans)
|
gst_base_transform_get_size (GstBaseTransform * trans, GstCaps * caps)
|
||||||
{
|
{
|
||||||
guint res = -1;
|
guint res = -1;
|
||||||
GstBaseTransformClass *bclass;
|
GstBaseTransformClass *bclass;
|
||||||
|
|
||||||
bclass = GST_BASE_TRANSFORM_GET_CLASS (trans);
|
bclass = GST_BASE_TRANSFORM_GET_CLASS (trans);
|
||||||
if (bclass->get_size) {
|
if (bclass->get_size) {
|
||||||
res = bclass->get_size (trans);
|
res = bclass->get_size (trans, caps);
|
||||||
GST_DEBUG_OBJECT (trans, "get size function returned %d", res);
|
GST_DEBUG_OBJECT (trans, "get size(%p) returned %d", caps, res);
|
||||||
}
|
}
|
||||||
|
|
||||||
return res;
|
return res;
|
||||||
|
@ -494,21 +492,74 @@ gst_base_transform_buffer_alloc (GstPad * pad, guint64 offset, guint size,
|
||||||
{
|
{
|
||||||
GstBaseTransform *trans;
|
GstBaseTransform *trans;
|
||||||
GstFlowReturn res;
|
GstFlowReturn res;
|
||||||
|
guint got_size;
|
||||||
|
|
||||||
trans = GST_BASE_TRANSFORM (gst_pad_get_parent (pad));
|
trans = GST_BASE_TRANSFORM (gst_pad_get_parent (pad));
|
||||||
|
|
||||||
|
*buf = NULL;
|
||||||
|
|
||||||
if (trans->in_place) {
|
if (trans->in_place) {
|
||||||
/* we can only proxy the bufferpool if we do in_place transforms */
|
/* request a buffer with the same caps */
|
||||||
res = gst_pad_alloc_buffer (trans->srcpad, offset, size, caps, buf);
|
res = gst_pad_alloc_buffer (trans->srcpad, offset, size, caps, buf);
|
||||||
} else {
|
} else {
|
||||||
/* else let the default alloc function allocate a buffer */
|
/* if we are configured, request a buffer with the src caps */
|
||||||
*buf = NULL;
|
GstCaps *srccaps = gst_pad_get_negotiated_caps (trans->srcpad);
|
||||||
|
|
||||||
|
if (!srccaps)
|
||||||
|
goto not_configured;
|
||||||
|
|
||||||
|
got_size = gst_base_transform_get_size (trans, srccaps);
|
||||||
|
if (got_size == -1) {
|
||||||
|
gst_caps_unref (srccaps);
|
||||||
|
goto unknown_size;
|
||||||
|
}
|
||||||
|
|
||||||
|
res = gst_pad_alloc_buffer (trans->srcpad, offset, got_size, srccaps, buf);
|
||||||
|
gst_caps_unref (srccaps);
|
||||||
|
}
|
||||||
|
|
||||||
|
if (res == GST_FLOW_OK && !trans->in_place) {
|
||||||
|
/* note that we might have been in place before, but calling the
|
||||||
|
alloc_buffer caused setcaps to switch us out of in_place -- in any case
|
||||||
|
the alloc_buffer served to transmit caps information but we can't use the
|
||||||
|
buffer. fall through and allocate a buffer corresponding to our sink
|
||||||
|
caps, if any */
|
||||||
|
GstCaps *sinkcaps = gst_pad_get_negotiated_caps (trans->sinkpad);
|
||||||
|
|
||||||
|
if (!sinkcaps)
|
||||||
|
goto not_configured;
|
||||||
|
|
||||||
|
got_size = gst_base_transform_get_size (trans, sinkcaps);
|
||||||
|
if (got_size == -1) {
|
||||||
|
gst_caps_unref (sinkcaps);
|
||||||
|
goto unknown_size;
|
||||||
|
}
|
||||||
|
|
||||||
|
*buf = gst_buffer_new_and_alloc (got_size);
|
||||||
|
gst_buffer_set_caps (*buf, sinkcaps);
|
||||||
|
GST_BUFFER_OFFSET (*buf) = offset;
|
||||||
res = GST_FLOW_OK;
|
res = GST_FLOW_OK;
|
||||||
|
|
||||||
|
gst_caps_unref (sinkcaps);
|
||||||
}
|
}
|
||||||
|
|
||||||
gst_object_unref (trans);
|
gst_object_unref (trans);
|
||||||
|
|
||||||
return res;
|
return res;
|
||||||
|
|
||||||
|
not_configured:
|
||||||
|
{
|
||||||
|
/* let the default allocator handle it */
|
||||||
|
*buf = NULL;
|
||||||
|
gst_object_unref (trans);
|
||||||
|
return GST_FLOW_OK;
|
||||||
|
}
|
||||||
|
unknown_size:
|
||||||
|
{
|
||||||
|
/* let the default allocator handle it */
|
||||||
|
*buf = NULL;
|
||||||
|
gst_object_unref (trans);
|
||||||
|
return GST_FLOW_OK;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
|
@ -584,16 +635,25 @@ gst_base_transform_handle_buffer (GstBaseTransform * trans, GstBuffer * inbuf,
|
||||||
/* figure out the output size */
|
/* figure out the output size */
|
||||||
if (trans->out_size == -1) {
|
if (trans->out_size == -1) {
|
||||||
/* ask subclass */
|
/* ask subclass */
|
||||||
if ((trans->out_size = gst_base_transform_get_size (trans)) == -1)
|
trans->out_size = gst_base_transform_get_size (trans,
|
||||||
/* else we have an error */
|
GST_PAD_CAPS (trans->srcpad));
|
||||||
|
if (trans->out_size == -1)
|
||||||
|
/* we have an error */
|
||||||
goto no_size;
|
goto no_size;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/* we cannot reconfigure the element yet as we are still processing
|
||||||
|
* the old buffer. We will therefore delay the reconfiguration of the
|
||||||
|
* element until we have processed this last buffer. */
|
||||||
|
trans->delay_configure = TRUE;
|
||||||
|
|
||||||
/* no in place transform, get buffer, this might renegotiate. */
|
/* no in place transform, get buffer, this might renegotiate. */
|
||||||
ret = gst_pad_alloc_buffer (trans->srcpad,
|
ret = gst_pad_alloc_buffer (trans->srcpad,
|
||||||
GST_BUFFER_OFFSET (inbuf), trans->out_size,
|
GST_BUFFER_OFFSET (inbuf), trans->out_size,
|
||||||
GST_PAD_CAPS (trans->srcpad), outbuf);
|
GST_PAD_CAPS (trans->srcpad), outbuf);
|
||||||
|
|
||||||
|
trans->delay_configure = FALSE;
|
||||||
|
|
||||||
if (ret != GST_FLOW_OK)
|
if (ret != GST_FLOW_OK)
|
||||||
goto no_buffer;
|
goto no_buffer;
|
||||||
|
|
||||||
|
|
|
@ -54,6 +54,7 @@ struct _GstBaseTransform {
|
||||||
|
|
||||||
gboolean in_place;
|
gboolean in_place;
|
||||||
guint out_size;
|
guint out_size;
|
||||||
|
gboolean delay_configure;
|
||||||
};
|
};
|
||||||
|
|
||||||
struct _GstBaseTransformClass {
|
struct _GstBaseTransformClass {
|
||||||
|
@ -70,8 +71,8 @@ struct _GstBaseTransformClass {
|
||||||
gboolean (*set_caps) (GstBaseTransform *trans, GstCaps *incaps,
|
gboolean (*set_caps) (GstBaseTransform *trans, GstCaps *incaps,
|
||||||
GstCaps *outcaps);
|
GstCaps *outcaps);
|
||||||
|
|
||||||
/* get the size of the output buffer, -1 on error */
|
/* get the byte size of a given caps, -1 on error */
|
||||||
guint (*get_size) (GstBaseTransform *trans);
|
guint (*get_size) (GstBaseTransform *trans, GstCaps *caps);
|
||||||
|
|
||||||
/* start and stop processing, ideal for opening/closing the resource */
|
/* start and stop processing, ideal for opening/closing the resource */
|
||||||
gboolean (*start) (GstBaseTransform *trans);
|
gboolean (*start) (GstBaseTransform *trans);
|
||||||
|
|
|
@ -20,16 +20,19 @@
|
||||||
* Boston, MA 02111-1307, USA.
|
* Boston, MA 02111-1307, USA.
|
||||||
*/
|
*/
|
||||||
|
|
||||||
|
|
||||||
#include <stdlib.h>
|
|
||||||
|
|
||||||
#ifdef HAVE_CONFIG_H
|
#ifdef HAVE_CONFIG_H
|
||||||
# include "config.h"
|
#include "config.h"
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
#include "../gst-i18n-lib.h"
|
#include "../gst-i18n-lib.h"
|
||||||
#include <gst/gstmarshal.h>
|
|
||||||
#include <gst/gst.h>
|
#include <gst/gst.h>
|
||||||
|
#include <gst/base/gstbasetransform.h>
|
||||||
|
|
||||||
|
|
||||||
|
GstElementDetails gst_capsfilter_details = GST_ELEMENT_DETAILS ("CapsFilter",
|
||||||
|
"Generic",
|
||||||
|
"Pass data without modification, limiting formats",
|
||||||
|
"David Schleef <ds@schleef.org>");
|
||||||
|
|
||||||
|
|
||||||
#define GST_TYPE_CAPSFILTER \
|
#define GST_TYPE_CAPSFILTER \
|
||||||
|
@ -48,23 +51,25 @@ typedef struct _GstCapsFilterClass GstCapsFilterClass;
|
||||||
|
|
||||||
struct _GstCapsFilter
|
struct _GstCapsFilter
|
||||||
{
|
{
|
||||||
GstElement element;
|
GstBaseTransform trans;
|
||||||
|
|
||||||
GstPad *srcpad;
|
|
||||||
GstPad *sinkpad;
|
|
||||||
|
|
||||||
GstCaps *filter_caps;
|
GstCaps *filter_caps;
|
||||||
};
|
};
|
||||||
|
|
||||||
struct _GstCapsFilterClass
|
struct _GstCapsFilterClass
|
||||||
{
|
{
|
||||||
GstElementClass element_class;
|
GstBaseTransformClass trans_class;
|
||||||
|
|
||||||
};
|
};
|
||||||
|
|
||||||
GType gst_capsfilter_get_type (void);
|
GType gst_capsfilter_get_type (void);
|
||||||
|
|
||||||
|
|
||||||
|
enum
|
||||||
|
{
|
||||||
|
PROP_0,
|
||||||
|
PROP_FILTER_CAPS
|
||||||
|
};
|
||||||
|
|
||||||
|
|
||||||
static GstStaticPadTemplate sinktemplate = GST_STATIC_PAD_TEMPLATE ("sink",
|
static GstStaticPadTemplate sinktemplate = GST_STATIC_PAD_TEMPLATE ("sink",
|
||||||
GST_PAD_SINK,
|
GST_PAD_SINK,
|
||||||
|
@ -76,133 +81,74 @@ static GstStaticPadTemplate srctemplate = GST_STATIC_PAD_TEMPLATE ("src",
|
||||||
GST_PAD_ALWAYS,
|
GST_PAD_ALWAYS,
|
||||||
GST_STATIC_CAPS_ANY);
|
GST_STATIC_CAPS_ANY);
|
||||||
|
|
||||||
|
|
||||||
GST_DEBUG_CATEGORY_STATIC (gst_capsfilter_debug);
|
GST_DEBUG_CATEGORY_STATIC (gst_capsfilter_debug);
|
||||||
#define GST_CAT_DEFAULT gst_capsfilter_debug
|
#define GST_CAT_DEFAULT gst_capsfilter_debug
|
||||||
|
|
||||||
GstElementDetails gst_capsfilter_details = GST_ELEMENT_DETAILS ("CapsFilter",
|
|
||||||
"Generic",
|
|
||||||
"Pass data without modification, limiting formats",
|
|
||||||
"David Schleef <ds@schleef.org>");
|
|
||||||
|
|
||||||
enum
|
|
||||||
{
|
|
||||||
PROP_0,
|
|
||||||
PROP_FILTER_CAPS
|
|
||||||
};
|
|
||||||
|
|
||||||
|
|
||||||
#define _do_init(bla) \
|
#define _do_init(bla) \
|
||||||
GST_DEBUG_CATEGORY_INIT (gst_capsfilter_debug, "capsfilter", 0, "capsfilter element");
|
GST_DEBUG_CATEGORY_INIT (gst_capsfilter_debug, "capsfilter", 0, \
|
||||||
|
"capsfilter element");
|
||||||
|
|
||||||
|
GST_BOILERPLATE_FULL (GstCapsFilter, gst_capsfilter, GstBaseTransform,
|
||||||
|
GST_TYPE_BASE_TRANSFORM, _do_init);
|
||||||
|
|
||||||
GST_BOILERPLATE_FULL (GstCapsFilter, gst_capsfilter, GstElement,
|
|
||||||
GST_TYPE_ELEMENT, _do_init);
|
|
||||||
|
|
||||||
static void gst_capsfilter_finalize (GObject * object);
|
|
||||||
static void gst_capsfilter_set_property (GObject * object, guint prop_id,
|
static void gst_capsfilter_set_property (GObject * object, guint prop_id,
|
||||||
const GValue * value, GParamSpec * pspec);
|
const GValue * value, GParamSpec * pspec);
|
||||||
static void gst_capsfilter_get_property (GObject * object, guint prop_id,
|
static void gst_capsfilter_get_property (GObject * object, guint prop_id,
|
||||||
GValue * value, GParamSpec * pspec);
|
GValue * value, GParamSpec * pspec);
|
||||||
|
static void gst_capsfilter_dispose (GObject * object);
|
||||||
|
static GstCaps *gst_capsfilter_transform_caps (GstBaseTransform * base,
|
||||||
|
GstPad * pad, GstCaps * caps);
|
||||||
|
static GstFlowReturn gst_capsfilter_transform_ip (GstBaseTransform * base,
|
||||||
|
GstBuffer * buf);
|
||||||
|
|
||||||
static GstCaps *gst_capsfilter_getcaps (GstPad * pad);
|
|
||||||
static GstFlowReturn gst_capsfilter_chain (GstPad * pad, GstBuffer * buf);
|
|
||||||
|
|
||||||
static void
|
static void
|
||||||
gst_capsfilter_base_init (gpointer g_class)
|
gst_capsfilter_base_init (gpointer g_class)
|
||||||
{
|
{
|
||||||
GstElementClass *gstelement_class = GST_ELEMENT_CLASS (g_class);
|
GstElementClass *element_class = GST_ELEMENT_CLASS (g_class);
|
||||||
|
|
||||||
gst_element_class_add_pad_template (gstelement_class,
|
gst_element_class_add_pad_template (element_class,
|
||||||
gst_static_pad_template_get (&srctemplate));
|
gst_static_pad_template_get (&srctemplate));
|
||||||
gst_element_class_add_pad_template (gstelement_class,
|
gst_element_class_add_pad_template (element_class,
|
||||||
gst_static_pad_template_get (&sinktemplate));
|
gst_static_pad_template_get (&sinktemplate));
|
||||||
gst_element_class_set_details (gstelement_class, &gst_capsfilter_details);
|
gst_element_class_set_details (element_class, &gst_capsfilter_details);
|
||||||
}
|
|
||||||
|
|
||||||
static void
|
|
||||||
gst_capsfilter_finalize (GObject * object)
|
|
||||||
{
|
|
||||||
GstCapsFilter *capsfilter;
|
|
||||||
|
|
||||||
capsfilter = GST_CAPSFILTER (object);
|
|
||||||
|
|
||||||
gst_caps_unref (capsfilter->filter_caps);
|
|
||||||
|
|
||||||
G_OBJECT_CLASS (parent_class)->finalize (object);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
static void
|
static void
|
||||||
gst_capsfilter_class_init (GstCapsFilterClass * klass)
|
gst_capsfilter_class_init (GstCapsFilterClass * klass)
|
||||||
{
|
{
|
||||||
GObjectClass *gobject_class;
|
GObjectClass *gobject_class;
|
||||||
GstElementClass *gstelement_class;
|
GstBaseTransformClass *trans_class;
|
||||||
|
|
||||||
gobject_class = G_OBJECT_CLASS (klass);
|
gobject_class = (GObjectClass *) klass;
|
||||||
gstelement_class = GST_ELEMENT_CLASS (klass);
|
gobject_class->set_property = gst_capsfilter_set_property;
|
||||||
|
gobject_class->get_property = gst_capsfilter_get_property;
|
||||||
gobject_class->set_property = GST_DEBUG_FUNCPTR (gst_capsfilter_set_property);
|
gobject_class->dispose = gst_capsfilter_dispose;
|
||||||
gobject_class->get_property = GST_DEBUG_FUNCPTR (gst_capsfilter_get_property);
|
|
||||||
gobject_class->finalize = GST_DEBUG_FUNCPTR (gst_capsfilter_finalize);
|
|
||||||
|
|
||||||
g_object_class_install_property (G_OBJECT_CLASS (klass), PROP_FILTER_CAPS,
|
g_object_class_install_property (G_OBJECT_CLASS (klass), PROP_FILTER_CAPS,
|
||||||
g_param_spec_boxed ("filter_caps", _("Filter caps"),
|
g_param_spec_boxed ("filter_caps", _("Filter caps"),
|
||||||
_("Restrict the possible allowed formats"),
|
_("Restrict the possible allowed formats"),
|
||||||
GST_TYPE_CAPS, G_PARAM_READWRITE));
|
GST_TYPE_CAPS, G_PARAM_READWRITE));
|
||||||
|
|
||||||
|
trans_class = (GstBaseTransformClass *) klass;
|
||||||
|
trans_class->transform_caps = gst_capsfilter_transform_caps;
|
||||||
|
trans_class->transform_ip = gst_capsfilter_transform_ip;
|
||||||
}
|
}
|
||||||
|
|
||||||
static void
|
static void
|
||||||
gst_capsfilter_init (GstCapsFilter * capsfilter)
|
gst_capsfilter_init (GstCapsFilter * filter)
|
||||||
{
|
{
|
||||||
gst_element_create_all_pads (GST_ELEMENT (capsfilter));
|
gst_base_transform_set_passthrough (GST_BASE_TRANSFORM (filter), TRUE);
|
||||||
|
filter->filter_caps = gst_caps_new_any ();
|
||||||
capsfilter->srcpad = gst_element_get_pad (GST_ELEMENT (capsfilter), "src");
|
|
||||||
capsfilter->sinkpad = gst_element_get_pad (GST_ELEMENT (capsfilter), "sink");
|
|
||||||
|
|
||||||
gst_pad_set_getcaps_function (capsfilter->srcpad, gst_capsfilter_getcaps);
|
|
||||||
|
|
||||||
gst_pad_set_getcaps_function (capsfilter->sinkpad, gst_capsfilter_getcaps);
|
|
||||||
gst_pad_set_chain_function (capsfilter->sinkpad, gst_capsfilter_chain);
|
|
||||||
|
|
||||||
capsfilter->filter_caps = gst_caps_new_any ();
|
|
||||||
}
|
|
||||||
|
|
||||||
static GstCaps *
|
|
||||||
gst_capsfilter_getcaps (GstPad * pad)
|
|
||||||
{
|
|
||||||
GstPad *otherpad;
|
|
||||||
GstCapsFilter *capsfilter = GST_CAPSFILTER (GST_OBJECT_PARENT (pad));
|
|
||||||
GstCaps *caps;
|
|
||||||
GstCaps *icaps;
|
|
||||||
|
|
||||||
otherpad = (pad == capsfilter->srcpad) ? capsfilter->sinkpad :
|
|
||||||
capsfilter->srcpad;
|
|
||||||
|
|
||||||
caps = gst_pad_peer_get_caps (otherpad);
|
|
||||||
if (caps == NULL)
|
|
||||||
caps = gst_caps_new_any ();
|
|
||||||
|
|
||||||
icaps = gst_caps_intersect (caps, capsfilter->filter_caps);
|
|
||||||
gst_caps_unref (caps);
|
|
||||||
|
|
||||||
return icaps;
|
|
||||||
}
|
|
||||||
|
|
||||||
static GstFlowReturn
|
|
||||||
gst_capsfilter_chain (GstPad * pad, GstBuffer * buf)
|
|
||||||
{
|
|
||||||
GstCapsFilter *capsfilter = GST_CAPSFILTER (GST_PAD_PARENT (pad));
|
|
||||||
|
|
||||||
gst_pad_push (capsfilter->srcpad, buf);
|
|
||||||
|
|
||||||
return GST_FLOW_OK;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
static void
|
static void
|
||||||
gst_capsfilter_set_property (GObject * object, guint prop_id,
|
gst_capsfilter_set_property (GObject * object, guint prop_id,
|
||||||
const GValue * value, GParamSpec * pspec)
|
const GValue * value, GParamSpec * pspec)
|
||||||
{
|
{
|
||||||
GstCapsFilter *capsfilter;
|
GstCapsFilter *capsfilter = GST_CAPSFILTER (object);
|
||||||
|
|
||||||
capsfilter = GST_CAPSFILTER (object);
|
|
||||||
|
|
||||||
switch (prop_id) {
|
switch (prop_id) {
|
||||||
case PROP_FILTER_CAPS:{
|
case PROP_FILTER_CAPS:{
|
||||||
|
@ -230,9 +176,7 @@ static void
|
||||||
gst_capsfilter_get_property (GObject * object, guint prop_id, GValue * value,
|
gst_capsfilter_get_property (GObject * object, guint prop_id, GValue * value,
|
||||||
GParamSpec * pspec)
|
GParamSpec * pspec)
|
||||||
{
|
{
|
||||||
GstCapsFilter *capsfilter;
|
GstCapsFilter *capsfilter = GST_CAPSFILTER (object);
|
||||||
|
|
||||||
capsfilter = GST_CAPSFILTER (object);
|
|
||||||
|
|
||||||
switch (prop_id) {
|
switch (prop_id) {
|
||||||
case PROP_FILTER_CAPS:
|
case PROP_FILTER_CAPS:
|
||||||
|
@ -243,3 +187,31 @@ gst_capsfilter_get_property (GObject * object, guint prop_id, GValue * value,
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
static void
|
||||||
|
gst_capsfilter_dispose (GObject * object)
|
||||||
|
{
|
||||||
|
GstCapsFilter *filter = GST_CAPSFILTER (object);
|
||||||
|
|
||||||
|
gst_caps_replace (&filter->filter_caps, NULL);
|
||||||
|
|
||||||
|
G_OBJECT_CLASS (parent_class)->dispose (object);
|
||||||
|
}
|
||||||
|
|
||||||
|
static GstCaps *
|
||||||
|
gst_capsfilter_transform_caps (GstBaseTransform * base, GstPad * pad,
|
||||||
|
GstCaps * caps)
|
||||||
|
{
|
||||||
|
GstCapsFilter *capsfilter = GST_CAPSFILTER (base);
|
||||||
|
GstCaps *ret;
|
||||||
|
|
||||||
|
ret = gst_caps_intersect (caps, capsfilter->filter_caps);
|
||||||
|
|
||||||
|
return ret;
|
||||||
|
}
|
||||||
|
|
||||||
|
static GstFlowReturn
|
||||||
|
gst_capsfilter_transform_ip (GstBaseTransform * base, GstBuffer * buf)
|
||||||
|
{
|
||||||
|
return GST_FLOW_OK;
|
||||||
|
}
|
||||||
|
|
Loading…
Reference in a new issue