mirror of
https://gitlab.freedesktop.org/gstreamer/gstreamer.git
synced 2024-11-03 16:09:39 +00:00
gst/interleave/deinterleave.*: Queue events until src pads were added and they can be sent. Otherwise downstream will...
Original commit message from CVS: * gst/interleave/deinterleave.c: (gst_deinterleave_finalize), (gst_deinterleave_init), (gst_deinterleave_sink_event), (gst_deinterleave_process), (gst_deinterleave_sink_activate_push): * gst/interleave/deinterleave.h: Queue events until src pads were added and they can be sent. Otherwise downstream will never get the first newsegment event.
This commit is contained in:
parent
838e10c219
commit
6827494919
4 changed files with 104 additions and 1 deletions
|
@ -1,3 +1,12 @@
|
||||||
|
2008-05-17 Sebastian Dröge <slomo@circular-chaos.org>
|
||||||
|
|
||||||
|
* gst/interleave/deinterleave.c: (gst_deinterleave_finalize),
|
||||||
|
(gst_deinterleave_init), (gst_deinterleave_sink_event),
|
||||||
|
(gst_deinterleave_process), (gst_deinterleave_sink_activate_push):
|
||||||
|
* gst/interleave/deinterleave.h:
|
||||||
|
Queue events until src pads were added and they can be sent. Otherwise
|
||||||
|
downstream will never get the first newsegment event.
|
||||||
|
|
||||||
2008-05-17 Sebastian Dröge <slomo@circular-chaos.org>
|
2008-05-17 Sebastian Dröge <slomo@circular-chaos.org>
|
||||||
|
|
||||||
* gst/interleave/deinterleave.c: (gst_deinterleave_sink_setcaps),
|
* gst/interleave/deinterleave.c: (gst_deinterleave_sink_setcaps),
|
||||||
|
|
2
common
2
common
|
@ -1 +1 @@
|
||||||
Subproject commit 3b3631082d04b426f450810e8836de94e9c5d60a
|
Subproject commit e365978c480a8fffa4bdb61568fb2cd989d1b197
|
|
@ -30,6 +30,23 @@
|
||||||
* and passing downstream caps changes upstream there
|
* and passing downstream caps changes upstream there
|
||||||
*/
|
*/
|
||||||
|
|
||||||
|
/**
|
||||||
|
* SECTION:element-deinterleave
|
||||||
|
*
|
||||||
|
* <refsect2>
|
||||||
|
* <para>
|
||||||
|
* Splits one interleaved multichannel audio stream into many mono audio streams.
|
||||||
|
* </para>
|
||||||
|
* <title>Example launch line</title>
|
||||||
|
* <para>
|
||||||
|
* <programlisting>
|
||||||
|
* gst-launch-0.10 filesrc location=/path/to/file.mp3 ! decodebin ! audioconvert ! "audio/x-raw-int,channels=2 ! deinterleave name=d d.src0 ! queue ! audioconvert ! vorbisenc ! oggmux ! filesink location=channel1.ogg d.src1 ! queue ! audioconvert ! vorbisenc ! oggmux ! filesink location=channel2.ogg
|
||||||
|
* </programlisting>
|
||||||
|
* Decodes an MP3 file and encodes the left and right channel into a separate
|
||||||
|
* </para>
|
||||||
|
* </refsect2>
|
||||||
|
*/
|
||||||
|
|
||||||
#ifdef HAVE_CONFIG_H
|
#ifdef HAVE_CONFIG_H
|
||||||
# include "config.h"
|
# include "config.h"
|
||||||
#endif
|
#endif
|
||||||
|
@ -111,6 +128,7 @@ static gboolean gst_deinterleave_sink_setcaps (GstPad * pad, GstCaps * caps);
|
||||||
static GstCaps *gst_deinterleave_getcaps (GstPad * pad);
|
static GstCaps *gst_deinterleave_getcaps (GstPad * pad);
|
||||||
static gboolean gst_deinterleave_sink_activate_push (GstPad * pad,
|
static gboolean gst_deinterleave_sink_activate_push (GstPad * pad,
|
||||||
gboolean active);
|
gboolean active);
|
||||||
|
static gboolean gst_deinterleave_sink_event (GstPad * pad, GstEvent * event);
|
||||||
|
|
||||||
static void
|
static void
|
||||||
gst_deinterleave_finalize (GObject * obj)
|
gst_deinterleave_finalize (GObject * obj)
|
||||||
|
@ -122,6 +140,12 @@ gst_deinterleave_finalize (GObject * obj)
|
||||||
self->pos = NULL;
|
self->pos = NULL;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if (self->pending_events) {
|
||||||
|
g_list_foreach (self->pending_events, (GFunc) gst_mini_object_unref, NULL);
|
||||||
|
g_list_free (self->pending_events);
|
||||||
|
self->pending_events = NULL;
|
||||||
|
}
|
||||||
|
|
||||||
G_OBJECT_CLASS (parent_class)->finalize (obj);
|
G_OBJECT_CLASS (parent_class)->finalize (obj);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -169,6 +193,8 @@ gst_deinterleave_init (GstDeinterleave * self, GstDeinterleaveClass * klass)
|
||||||
gst_pad_set_getcaps_function (self->sink, gst_deinterleave_getcaps);
|
gst_pad_set_getcaps_function (self->sink, gst_deinterleave_getcaps);
|
||||||
gst_pad_set_activatepush_function (self->sink,
|
gst_pad_set_activatepush_function (self->sink,
|
||||||
GST_DEBUG_FUNCPTR (gst_deinterleave_sink_activate_push));
|
GST_DEBUG_FUNCPTR (gst_deinterleave_sink_activate_push));
|
||||||
|
gst_pad_set_event_function (self->sink,
|
||||||
|
GST_DEBUG_FUNCPTR (gst_deinterleave_sink_event));
|
||||||
gst_element_add_pad (GST_ELEMENT (self), self->sink);
|
gst_element_add_pad (GST_ELEMENT (self), self->sink);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -486,6 +512,42 @@ gst_deinterleave_getcaps (GstPad * pad)
|
||||||
return ret;
|
return ret;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
static gboolean
|
||||||
|
gst_deinterleave_sink_event (GstPad * pad, GstEvent * event)
|
||||||
|
{
|
||||||
|
GstDeinterleave *self = GST_DEINTERLEAVE (gst_pad_get_parent (pad));
|
||||||
|
gboolean ret;
|
||||||
|
|
||||||
|
GST_DEBUG ("Got %s event on pad %s:%s", GST_EVENT_TYPE_NAME (event),
|
||||||
|
GST_DEBUG_PAD_NAME (pad));
|
||||||
|
|
||||||
|
/* Send FLUSH_STOP, FLUSH_START and EOS immediately, no matter if
|
||||||
|
* we have src pads already or not. Queue all other events and
|
||||||
|
* push them after we have src pads
|
||||||
|
*/
|
||||||
|
switch (GST_EVENT_TYPE (event)) {
|
||||||
|
case GST_EVENT_FLUSH_STOP:
|
||||||
|
case GST_EVENT_FLUSH_START:
|
||||||
|
case GST_EVENT_EOS:
|
||||||
|
ret = gst_pad_event_default (pad, event);
|
||||||
|
break;
|
||||||
|
default:
|
||||||
|
if (self->srcpads) {
|
||||||
|
ret = gst_pad_event_default (pad, event);
|
||||||
|
} else {
|
||||||
|
GST_OBJECT_LOCK (self);
|
||||||
|
self->pending_events = g_list_append (self->pending_events, event);
|
||||||
|
GST_OBJECT_UNLOCK (self);
|
||||||
|
ret = TRUE;
|
||||||
|
}
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
|
||||||
|
gst_object_unref (self);
|
||||||
|
|
||||||
|
return ret;
|
||||||
|
}
|
||||||
|
|
||||||
static GstFlowReturn
|
static GstFlowReturn
|
||||||
gst_deinterleave_process (GstDeinterleave * self, GstBuffer * buf)
|
gst_deinterleave_process (GstDeinterleave * self, GstBuffer * buf)
|
||||||
{
|
{
|
||||||
|
@ -499,6 +561,27 @@ gst_deinterleave_process (GstDeinterleave * self, GstBuffer * buf)
|
||||||
GstBuffer **buffers_out = g_new0 (GstBuffer *, channels);
|
GstBuffer **buffers_out = g_new0 (GstBuffer *, channels);
|
||||||
guint8 *in, *out;
|
guint8 *in, *out;
|
||||||
|
|
||||||
|
/* Send any pending events to all src pads */
|
||||||
|
GST_OBJECT_LOCK (self);
|
||||||
|
if (self->pending_events) {
|
||||||
|
GList *events;
|
||||||
|
GstEvent *event;
|
||||||
|
|
||||||
|
GST_DEBUG_OBJECT (self, "Sending pending events to all src pads");
|
||||||
|
|
||||||
|
for (events = self->pending_events; events != NULL; events = events->next) {
|
||||||
|
event = GST_EVENT (events->data);
|
||||||
|
|
||||||
|
for (srcs = self->srcpads; srcs != NULL; srcs = srcs->next)
|
||||||
|
gst_pad_push_event (GST_PAD (srcs->data), gst_event_ref (event));
|
||||||
|
gst_event_unref (event);
|
||||||
|
}
|
||||||
|
|
||||||
|
g_list_free (self->pending_events);
|
||||||
|
self->pending_events = NULL;
|
||||||
|
}
|
||||||
|
GST_OBJECT_UNLOCK (self);
|
||||||
|
|
||||||
/* Allocate buffers */
|
/* Allocate buffers */
|
||||||
for (srcs = self->srcpads, i = 0; srcs; srcs = srcs->next, i++) {
|
for (srcs = self->srcpads, i = 0; srcs; srcs = srcs->next, i++) {
|
||||||
GstPad *pad = (GstPad *) srcs->data;
|
GstPad *pad = (GstPad *) srcs->data;
|
||||||
|
@ -528,6 +611,8 @@ gst_deinterleave_process (GstDeinterleave * self, GstBuffer * buf)
|
||||||
|
|
||||||
/* Return NOT_LINKED if no pad was linked */
|
/* Return NOT_LINKED if no pad was linked */
|
||||||
if (!buffers_allocated) {
|
if (!buffers_allocated) {
|
||||||
|
GST_WARNING_OBJECT (self,
|
||||||
|
"Couldn't allocate any buffers because no pad was linked");
|
||||||
ret = GST_FLOW_NOT_LINKED;
|
ret = GST_FLOW_NOT_LINKED;
|
||||||
goto done;
|
goto done;
|
||||||
}
|
}
|
||||||
|
@ -631,6 +716,13 @@ gst_deinterleave_sink_activate_push (GstPad * pad, gboolean active)
|
||||||
self->channels = 0;
|
self->channels = 0;
|
||||||
self->width = 0;
|
self->width = 0;
|
||||||
self->func = NULL;
|
self->func = NULL;
|
||||||
|
|
||||||
|
if (self->pending_events) {
|
||||||
|
g_list_foreach (self->pending_events, (GFunc) gst_mini_object_unref,
|
||||||
|
NULL);
|
||||||
|
g_list_free (self->pending_events);
|
||||||
|
self->pending_events = NULL;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
gst_object_unref (self);
|
gst_object_unref (self);
|
||||||
|
|
|
@ -58,6 +58,8 @@ struct _GstDeinterleave
|
||||||
|
|
||||||
gint width;
|
gint width;
|
||||||
GstDeinterleaveFunc func;
|
GstDeinterleaveFunc func;
|
||||||
|
|
||||||
|
GList *pending_events;
|
||||||
};
|
};
|
||||||
|
|
||||||
struct _GstDeinterleaveClass
|
struct _GstDeinterleaveClass
|
||||||
|
|
Loading…
Reference in a new issue