2 * Copyright (C) 1999,2000 Erik Walthinsen <omega@cse.ogi.edu>
3 * 2000 Wim Taymans <wtay@chello.be>
5 * gstpad.c: Pads for connecting elements together
7 * This library is free software; you can redistribute it and/or
8 * modify it under the terms of the GNU Library General Public
9 * License as published by the Free Software Foundation; either
10 * version 2 of the License, or (at your option) any later version.
12 * This library is distributed in the hope that it will be useful,
13 * but WITHOUT ANY WARRANTY; without even the implied warranty of
14 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
15 * Library General Public License for more details.
17 * You should have received a copy of the GNU Library General Public
18 * License along with this library; if not, write to the
19 * Free Software Foundation, Inc., 59 Temple Place - Suite 330,
20 * Boston, MA 02111-1307, USA.
23 /* #define GST_DEBUG_ENABLED */
24 #include "gst_private.h"
28 #include "gstelement.h"
31 #include "gstscheduler.h"
34 GType _gst_pad_type = 0;
36 /***** Start with the base GstPad class *****/
37 static void gst_pad_class_init (GstPadClass *klass);
38 static void gst_pad_init (GstPad *pad);
40 static gboolean gst_pad_try_reconnect_filtered_func (GstRealPad *srcpad, GstRealPad *sinkpad,
41 GstCaps *caps, gboolean clear);
43 #ifndef GST_DISABLE_LOADSAVE
44 static xmlNodePtr gst_pad_save_thyself (GstObject *object, xmlNodePtr parent);
47 static GstObject *pad_parent_class = NULL;
50 gst_pad_get_type(void)
53 static const GTypeInfo pad_info = {
57 (GClassInitFunc)gst_pad_class_init,
62 (GInstanceInitFunc)gst_pad_init,
65 _gst_pad_type = g_type_register_static(GST_TYPE_OBJECT, "GstPad", &pad_info, 0);
71 gst_pad_class_init (GstPadClass *klass)
73 pad_parent_class = g_type_class_ref(GST_TYPE_OBJECT);
77 gst_pad_init (GstPad *pad)
79 pad->element_private = NULL;
81 pad->padtemplate = NULL;
86 /***** Then do the Real Pad *****/
87 /* Pad signals and args */
91 REAL_CAPS_NEGO_FAILED,
105 static void gst_real_pad_class_init (GstRealPadClass *klass);
106 static void gst_real_pad_init (GstRealPad *pad);
108 static void gst_real_pad_set_property (GObject *object, guint prop_id, const GValue *value, GParamSpec *pspec);
109 static void gst_real_pad_get_property (GObject *object, guint prop_id, GValue *value, GParamSpec *pspec);
111 static void gst_real_pad_dispose (GObject *object);
113 static void gst_pad_push_func (GstPad *pad, GstBuffer *buf);
115 GType _gst_real_pad_type = 0;
117 static GstPad *real_pad_parent_class = NULL;
118 static guint gst_real_pad_signals[REAL_LAST_SIGNAL] = { 0 };
121 gst_real_pad_get_type(void) {
122 if (!_gst_real_pad_type) {
123 static const GTypeInfo pad_info = {
124 sizeof(GstRealPadClass),
127 (GClassInitFunc)gst_real_pad_class_init,
132 (GInstanceInitFunc)gst_real_pad_init,
135 _gst_real_pad_type = g_type_register_static(GST_TYPE_PAD, "GstRealPad", &pad_info, 0);
137 return _gst_real_pad_type;
141 gst_real_pad_class_init (GstRealPadClass *klass)
143 GObjectClass *gobject_class;
144 GstObjectClass *gstobject_class;
146 gobject_class = (GObjectClass*) klass;
147 gstobject_class = (GstObjectClass*) klass;
149 real_pad_parent_class = g_type_class_ref (GST_TYPE_PAD);
151 gobject_class->dispose = GST_DEBUG_FUNCPTR (gst_real_pad_dispose);
152 gobject_class->set_property = GST_DEBUG_FUNCPTR (gst_real_pad_set_property);
153 gobject_class->get_property = GST_DEBUG_FUNCPTR (gst_real_pad_get_property);
155 gst_real_pad_signals[REAL_SET_ACTIVE] =
156 g_signal_new ("set_active", G_TYPE_FROM_CLASS (klass), G_SIGNAL_RUN_LAST,
157 G_STRUCT_OFFSET (GstRealPadClass, set_active), NULL, NULL,
158 gst_marshal_VOID__BOOLEAN, G_TYPE_NONE, 1,
160 gst_real_pad_signals[REAL_CAPS_CHANGED] =
161 g_signal_new ("caps_changed", G_TYPE_FROM_CLASS (klass), G_SIGNAL_RUN_LAST,
162 G_STRUCT_OFFSET (GstRealPadClass, caps_changed), NULL, NULL,
163 gst_marshal_VOID__POINTER, G_TYPE_NONE, 1,
165 gst_real_pad_signals[REAL_CAPS_NEGO_FAILED] =
166 g_signal_new ("caps_nego_failed", G_TYPE_FROM_CLASS (klass), G_SIGNAL_RUN_LAST,
167 G_STRUCT_OFFSET (GstRealPadClass, caps_nego_failed), NULL, NULL,
168 gst_marshal_VOID__POINTER, G_TYPE_NONE, 1,
170 gst_real_pad_signals[REAL_CONNECTED] =
171 g_signal_new ("connected", G_TYPE_FROM_CLASS (klass), G_SIGNAL_RUN_LAST,
172 G_STRUCT_OFFSET (GstRealPadClass, connected), NULL, NULL,
173 gst_marshal_VOID__POINTER, G_TYPE_NONE, 1,
175 gst_real_pad_signals[REAL_DISCONNECTED] =
176 g_signal_new ("disconnected", G_TYPE_FROM_CLASS (klass), G_SIGNAL_RUN_LAST,
177 G_STRUCT_OFFSET (GstRealPadClass, disconnected), NULL, NULL,
178 gst_marshal_VOID__POINTER, G_TYPE_NONE, 1,
180 gst_real_pad_signals[REAL_EVENT_RECEIVED] =
181 g_signal_new ("event_received", G_TYPE_FROM_CLASS (klass), G_SIGNAL_RUN_LAST,
182 G_STRUCT_OFFSET (GstRealPadClass, event_received), NULL, NULL,
183 gst_marshal_VOID__POINTER, G_TYPE_NONE, 1,
186 /* gtk_object_add_arg_type ("GstRealPad::active", G_TYPE_BOOLEAN, */
187 /* GTK_ARG_READWRITE, REAL_ARG_ACTIVE); */
188 g_object_class_install_property (G_OBJECT_CLASS (klass), REAL_ARG_ACTIVE,
189 g_param_spec_boolean ("active", "Active", "Whether the pad is active.",
190 TRUE,G_PARAM_READWRITE));
192 #ifndef GST_DISABLE_LOADSAVE
193 gstobject_class->save_thyself = GST_DEBUG_FUNCPTR (gst_pad_save_thyself);
195 gstobject_class->path_string_separator = ".";
199 gst_real_pad_init (GstRealPad *pad)
201 pad->direction = GST_PAD_UNKNOWN;
205 pad->sched_private = NULL;
207 pad->chainfunc = NULL;
209 pad->getregionfunc = NULL;
211 pad->chainhandler = GST_DEBUG_FUNCPTR (gst_pad_push_func);
212 pad->gethandler = NULL;
213 pad->pullregionfunc = NULL;
215 pad->bufferpoolfunc = NULL;
216 pad->ghostpads = NULL;
219 pad->connectfunc = NULL;
220 pad->getcapsfunc = NULL;
224 gst_real_pad_set_property (GObject *object, guint prop_id, const GValue *value, GParamSpec *pspec)
226 g_return_if_fail (GST_IS_PAD (object));
229 case REAL_ARG_ACTIVE:
230 if (g_value_get_boolean (value)) {
231 GST_DEBUG (GST_CAT_PADS, "activating pad %s:%s\n", GST_DEBUG_PAD_NAME (object));
232 GST_FLAG_UNSET (object, GST_PAD_DISABLED);
234 GST_DEBUG (GST_CAT_PADS, "de-activating pad %s:%s\n", GST_DEBUG_PAD_NAME (object));
235 GST_FLAG_SET (object, GST_PAD_DISABLED);
237 g_signal_emit (G_OBJECT (object), gst_real_pad_signals[REAL_SET_ACTIVE], 0,
238 !GST_FLAG_IS_SET (object, GST_PAD_DISABLED));
246 gst_real_pad_get_property (GObject *object, guint prop_id, GValue *value, GParamSpec *pspec)
248 /* it's not null if we got it, but it might not be ours */
249 g_return_if_fail (GST_IS_PAD (object));
252 case REAL_ARG_ACTIVE:
253 g_value_set_boolean (value, !GST_FLAG_IS_SET (object, GST_PAD_DISABLED));
263 * @name: name of new pad
264 * @direction: either GST_PAD_SRC or GST_PAD_SINK
266 * Create a new pad with given name.
271 gst_pad_new (gchar *name,
272 GstPadDirection direction)
276 g_return_val_if_fail (name != NULL, NULL);
277 g_return_val_if_fail (direction != GST_PAD_UNKNOWN, NULL);
279 pad = g_object_new (gst_real_pad_get_type (), NULL);
280 gst_object_set_name (GST_OBJECT (pad), name);
281 GST_RPAD_DIRECTION (pad) = direction;
283 return GST_PAD (pad);
287 * gst_pad_new_from_template:
288 * @templ: the pad template to use
289 * @name: the name of the element
291 * Create a new pad with given name from the given template.
296 gst_pad_new_from_template (GstPadTemplate *templ,
301 g_return_val_if_fail (name != NULL, NULL);
302 g_return_val_if_fail (templ != NULL, NULL);
304 pad = gst_pad_new (name, templ->direction);
306 gst_object_ref (GST_OBJECT (templ));
307 GST_PAD_PADTEMPLATE (pad) = templ;
313 * gst_pad_get_direction:
314 * @pad: the Pad to get the direction from
316 * Get the direction of the pad.
318 * Returns: the direction of the pad
321 gst_pad_get_direction (GstPad *pad)
323 g_return_val_if_fail (pad != NULL, GST_PAD_UNKNOWN);
324 g_return_val_if_fail (GST_IS_PAD (pad), GST_PAD_UNKNOWN);
326 return GST_PAD_DIRECTION (pad);
331 * @pad: the pad to set the name of
332 * @name: the name of the pad
334 * Set the name of a pad.
337 gst_pad_set_name (GstPad *pad,
340 g_return_if_fail (pad != NULL);
341 g_return_if_fail (GST_IS_PAD (pad));
343 gst_object_set_name (GST_OBJECT (pad), name);
348 * @pad: the pad to get the name of
350 * Get the name of a pad.
352 * Returns: the name of the pad, don't free.
355 gst_pad_get_name (GstPad *pad)
357 g_return_val_if_fail (pad != NULL, NULL);
358 g_return_val_if_fail (GST_IS_PAD (pad), NULL);
360 return GST_OBJECT_NAME (pad);
364 * gst_pad_set_chain_function:
365 * @pad: the pad to set the chain function for
366 * @chain: the chain function
368 * Set the given chain function for the pad.
371 gst_pad_set_chain_function (GstPad *pad,
372 GstPadChainFunction chain)
374 g_return_if_fail (pad != NULL);
375 g_return_if_fail (GST_IS_REAL_PAD (pad));
377 GST_RPAD_CHAINFUNC(pad) = chain;
378 GST_DEBUG (GST_CAT_PADS, "chainfunc for %s:%s set to %s\n",
379 GST_DEBUG_PAD_NAME (pad), GST_DEBUG_FUNCPTR_NAME (chain));
383 * gst_pad_set_get_function:
384 * @pad: the pad to set the get function for
385 * @get: the get function
387 * Set the given get function for the pad.
390 gst_pad_set_get_function (GstPad *pad,
391 GstPadGetFunction get)
393 g_return_if_fail (pad != NULL);
394 g_return_if_fail (GST_IS_REAL_PAD (pad));
396 GST_RPAD_GETFUNC(pad) = get;
397 GST_DEBUG (GST_CAT_PADS, "getfunc for %s:%s set to %s\n",
398 GST_DEBUG_PAD_NAME (pad), GST_DEBUG_FUNCPTR_NAME (get));
402 * gst_pad_set_event_function:
403 * @pad: the pad to set the event handler for
404 * @event: the event handler
406 * Set the given event handler for the pad.
409 gst_pad_set_event_function (GstPad *pad,
410 GstPadEventFunction event)
412 g_return_if_fail (pad != NULL);
413 g_return_if_fail (GST_IS_REAL_PAD (pad));
415 GST_RPAD_EVENTFUNC(pad) = event;
416 GST_DEBUG (GST_CAT_PADS, "eventfunc for %s:%s set to %s\n",
417 GST_DEBUG_PAD_NAME (pad), GST_DEBUG_FUNCPTR_NAME (event));
421 * gst_pad_set_getregion_function:
422 * @pad: the pad to set the getregion function for
423 * @getregion: the getregion function
425 * Set the given getregion function for the pad.
428 gst_pad_set_getregion_function (GstPad *pad,
429 GstPadGetRegionFunction getregion)
431 g_return_if_fail (pad != NULL);
432 g_return_if_fail (GST_IS_REAL_PAD (pad));
434 GST_RPAD_GETREGIONFUNC(pad) = getregion;
435 GST_DEBUG (GST_CAT_PADS, "getregionfunc for %s:%s set to %s\n",
436 GST_DEBUG_PAD_NAME (pad), GST_DEBUG_FUNCPTR_NAME (getregion));
440 * gst_pad_set_connect_function:
441 * @pad: the pad to set the connect function for
442 * @connect: the connect function
444 * Set the given connect function for the pad. It will be called
445 * when the pad is connected or reconnected with caps.
448 gst_pad_set_connect_function (GstPad *pad,
449 GstPadConnectFunction connect)
451 g_return_if_fail (pad != NULL);
452 g_return_if_fail (GST_IS_REAL_PAD (pad));
454 GST_RPAD_CONNECTFUNC (pad) = connect;
455 GST_DEBUG (GST_CAT_PADS, "connectfunc for %s:%s set to %s\n",
456 GST_DEBUG_PAD_NAME (pad), GST_DEBUG_FUNCPTR_NAME (connect));
460 * gst_pad_set_getcaps_function:
461 * @pad: the pad to set the getcaps function for
462 * @getcaps: the getcaps function
464 * Set the given getcaps function for the pad.
467 gst_pad_set_getcaps_function (GstPad *pad,
468 GstPadGetCapsFunction getcaps)
470 g_return_if_fail (pad != NULL);
471 g_return_if_fail (GST_IS_REAL_PAD (pad));
473 GST_RPAD_GETCAPSFUNC (pad) = getcaps;
474 GST_DEBUG (GST_CAT_PADS, "getcapsfunc for %s:%s set to %s\n",
475 GST_DEBUG_PAD_NAME (pad), GST_DEBUG_FUNCPTR_NAME (getcaps));
478 * gst_pad_set_bufferpool_function:
479 * @pad: the pad to set the bufferpool function for
480 * @bufpool: the bufferpool function
482 * Set the given bufferpool function for the pad.
485 gst_pad_set_bufferpool_function (GstPad *pad,
486 GstPadBufferPoolFunction bufpool)
488 g_return_if_fail (pad != NULL);
489 g_return_if_fail (GST_IS_REAL_PAD (pad));
491 GST_RPAD_BUFFERPOOLFUNC (pad) = bufpool;
492 GST_DEBUG (GST_CAT_PADS, "bufferpoolfunc for %s:%s set to %s\n",
493 GST_DEBUG_PAD_NAME (pad), GST_DEBUG_FUNCPTR_NAME (bufpool));
497 gst_pad_push_func(GstPad *pad, GstBuffer *buf)
499 if (GST_RPAD_CHAINFUNC (GST_RPAD_PEER (pad)) != NULL) {
500 GST_DEBUG (GST_CAT_DATAFLOW, "calling chain function %s\n",
501 GST_DEBUG_FUNCPTR_NAME (GST_RPAD_CHAINFUNC (GST_RPAD_PEER (pad))));
502 (GST_RPAD_CHAINFUNC (GST_RPAD_PEER (pad))) (pad, buf);
504 GST_DEBUG (GST_CAT_DATAFLOW, "default pad_push handler in place, no chain function\n");
505 g_warning ("(internal error) default pad_push in place for pad %s:%s but it has no chain function",
506 GST_DEBUG_PAD_NAME (pad));
512 * gst_pad_disconnect:
513 * @srcpad: the source pad to disconnect
514 * @sinkpad: the sink pad to disconnect
516 * Disconnects the source pad from the sink pad.
519 gst_pad_disconnect (GstPad *srcpad,
522 GstRealPad *realsrc, *realsink;
525 g_return_if_fail (srcpad != NULL);
526 g_return_if_fail (GST_IS_PAD (srcpad));
527 g_return_if_fail (sinkpad != NULL);
528 g_return_if_fail (GST_IS_PAD (sinkpad));
530 GST_INFO (GST_CAT_ELEMENT_PADS, "disconnecting %s:%s(%p) and %s:%s(%p)",
531 GST_DEBUG_PAD_NAME (srcpad), srcpad, GST_DEBUG_PAD_NAME (sinkpad), sinkpad);
533 /* now we need to deal with the real/ghost stuff */
534 realsrc = GST_PAD_REALIZE (srcpad);
535 realsink = GST_PAD_REALIZE (sinkpad);
537 g_return_if_fail (GST_RPAD_PEER (realsrc) != NULL);
538 g_return_if_fail (GST_RPAD_PEER (realsink) != NULL);
540 if ((GST_RPAD_DIRECTION (realsrc) == GST_PAD_SINK) &&
541 (GST_RPAD_DIRECTION (realsink) == GST_PAD_SRC)) {
548 g_return_if_fail ((GST_RPAD_DIRECTION (realsrc) == GST_PAD_SRC) &&
549 (GST_RPAD_DIRECTION (realsink) == GST_PAD_SINK));
551 /* first clear peers */
552 GST_RPAD_PEER (realsrc) = NULL;
553 GST_RPAD_PEER (realsink) = NULL;
555 /* reset the filters, both filters are refcounted once */
556 if (GST_RPAD_FILTER (realsrc)) {
557 gst_caps_unref (GST_RPAD_FILTER (realsrc));
558 GST_RPAD_FILTER (realsink) = NULL;
559 GST_RPAD_FILTER (realsrc) = NULL;
562 /* now tell the scheduler */
563 if (GST_PAD_PARENT (realsrc)->sched)
564 gst_scheduler_pad_disconnect (GST_PAD_PARENT (realsrc)->sched, (GstPad *)realsrc, (GstPad *)realsink);
565 else if (GST_PAD_PARENT (realsink)->sched)
566 gst_scheduler_pad_disconnect (GST_PAD_PARENT (realsink)->sched, (GstPad *)realsrc, (GstPad *)realsink);
568 /* hold a reference, as they can go away in the signal handlers */
569 gst_object_ref (GST_OBJECT (realsrc));
570 gst_object_ref (GST_OBJECT (realsink));
572 /* fire off a signal to each of the pads telling them that they've been disconnected */
573 g_signal_emit (G_OBJECT (realsrc), gst_real_pad_signals[REAL_DISCONNECTED], 0, realsink);
574 g_signal_emit (G_OBJECT (realsink), gst_real_pad_signals[REAL_DISCONNECTED], 0, realsrc);
576 GST_INFO (GST_CAT_ELEMENT_PADS, "disconnected %s:%s and %s:%s",
577 GST_DEBUG_PAD_NAME (srcpad), GST_DEBUG_PAD_NAME (sinkpad));
579 gst_object_unref (GST_OBJECT (realsrc));
580 gst_object_unref (GST_OBJECT (realsink));
584 * gst_pad_can_connect_filtered:
585 * @srcpad: the source pad to connect
586 * @sinkpad: the sink pad to connect
587 * @filtercaps: the filter caps.
589 * Checks if the source pad and the sink pad can be connected.
590 * The filter indicates the media type that should flow trought this connection.
592 * Returns: TRUE if the pad can be connected, FALSE otherwise
595 gst_pad_can_connect_filtered (GstPad *srcpad, GstPad *sinkpad, GstCaps *filtercaps)
597 gint num_decoupled = 0;
598 GstRealPad *realsrc, *realsink;
601 g_return_val_if_fail (srcpad != NULL, FALSE);
602 g_return_val_if_fail (GST_IS_PAD (srcpad), FALSE);
603 g_return_val_if_fail (sinkpad != NULL, FALSE);
604 g_return_val_if_fail (GST_IS_PAD (sinkpad), FALSE);
606 /* now we need to deal with the real/ghost stuff */
607 realsrc = GST_PAD_REALIZE (srcpad);
608 realsink = GST_PAD_REALIZE (sinkpad);
610 g_return_val_if_fail (GST_RPAD_PEER (realsrc) == NULL, FALSE);
611 g_return_val_if_fail (GST_RPAD_PEER (realsink) == NULL, FALSE);
612 g_return_val_if_fail (GST_PAD_PARENT (realsrc) != NULL, FALSE);
613 g_return_val_if_fail (GST_PAD_PARENT (realsink) != NULL, FALSE);
615 if (realsrc->sched && realsink->sched) {
616 if (GST_FLAG_IS_SET (GST_PAD_PARENT (realsrc), GST_ELEMENT_DECOUPLED))
618 if (GST_FLAG_IS_SET (GST_PAD_PARENT (realsink), GST_ELEMENT_DECOUPLED))
621 if (realsrc->sched != realsink->sched && num_decoupled != 1) {
622 g_warning ("connecting pads with different scheds requires exactly one decoupled element (queue)\n");
627 /* check if the directions are compatible */
628 if (!(((GST_RPAD_DIRECTION (realsrc) == GST_PAD_SINK) &&
629 (GST_RPAD_DIRECTION (realsink) == GST_PAD_SRC)) ||
630 ((GST_RPAD_DIRECTION (realsrc) == GST_PAD_SRC) &&
631 (GST_RPAD_DIRECTION (realsink) == GST_PAD_SINK))))
639 * gst_pad_can_connect:
640 * @srcpad: the source pad to connect
641 * @sinkpad: the sink pad to connect
643 * Checks if the source pad can be connected to the sink pad.
645 * Returns: TRUE if the pads can be connected, FALSE otherwise
648 gst_pad_can_connect (GstPad *srcpad, GstPad *sinkpad)
650 return gst_pad_can_connect_filtered (srcpad, sinkpad, NULL);
654 * gst_pad_connect_filtered:
655 * @srcpad: the source pad to connect
656 * @sinkpad: the sink pad to connect
657 * @filtercaps: the filter caps.
659 * Connects the source pad to the sink pad. The filter indicates the media type
660 * that should flow trought this connection.
662 * Returns: TRUE if the pad could be connected, FALSE otherwise
665 gst_pad_connect_filtered (GstPad *srcpad, GstPad *sinkpad, GstCaps *filtercaps)
667 GstRealPad *realsrc, *realsink;
668 gint num_decoupled = 0;
671 g_return_val_if_fail (srcpad != NULL, FALSE);
672 g_return_val_if_fail (GST_IS_PAD (srcpad), FALSE);
673 g_return_val_if_fail (sinkpad != NULL, FALSE);
674 g_return_val_if_fail (GST_IS_PAD (sinkpad), FALSE);
676 GST_INFO (GST_CAT_PADS, "connecting %s:%s and %s:%s",
677 GST_DEBUG_PAD_NAME (srcpad), GST_DEBUG_PAD_NAME (sinkpad));
679 /* now we need to deal with the real/ghost stuff */
680 realsrc = GST_PAD_REALIZE (srcpad);
681 realsink = GST_PAD_REALIZE (sinkpad);
683 if ((GST_PAD (realsrc) != srcpad) || (GST_PAD (realsink) != sinkpad)) {
684 GST_INFO (GST_CAT_PADS, "*actually* connecting %s:%s and %s:%s",
685 GST_DEBUG_PAD_NAME (realsrc), GST_DEBUG_PAD_NAME (realsink));
688 g_return_val_if_fail (GST_RPAD_PEER (realsrc) == NULL, FALSE);
689 g_return_val_if_fail (GST_RPAD_PEER (realsink) == NULL, FALSE);
690 g_return_val_if_fail (GST_PAD_PARENT (realsrc) != NULL, FALSE);
691 g_return_val_if_fail (GST_PAD_PARENT (realsink) != NULL, FALSE);
693 if (realsrc->sched && realsink->sched) {
694 if (GST_FLAG_IS_SET (GST_PAD_PARENT (realsrc), GST_ELEMENT_DECOUPLED))
696 if (GST_FLAG_IS_SET (GST_PAD_PARENT (realsink), GST_ELEMENT_DECOUPLED))
699 if (realsrc->sched != realsink->sched && num_decoupled != 1) {
700 g_warning ("connecting pads with different scheds requires exactly one decoupled element (queue)\n");
705 /* check for reversed directions and swap if necessary */
706 if ((GST_RPAD_DIRECTION (realsrc) == GST_PAD_SINK) &&
707 (GST_RPAD_DIRECTION (realsink) == GST_PAD_SRC)) {
714 g_return_val_if_fail ((GST_RPAD_DIRECTION (realsrc) == GST_PAD_SRC) &&
715 (GST_RPAD_DIRECTION (realsink) == GST_PAD_SINK), FALSE);
717 /* first set peers */
718 GST_RPAD_PEER (realsrc) = realsink;
719 GST_RPAD_PEER (realsink) = realsrc;
721 /* try to negotiate the pads, we don't need to clear the caps here */
722 if (!gst_pad_try_reconnect_filtered_func (realsrc, realsink, filtercaps, FALSE)) {
723 GST_DEBUG (GST_CAT_CAPS, "pads cannot connect\n");
725 GST_RPAD_PEER (realsrc) = NULL;
726 GST_RPAD_PEER (realsink) = NULL;
731 /* fire off a signal to each of the pads telling them that they've been connected */
732 g_signal_emit (G_OBJECT (realsrc), gst_real_pad_signals[REAL_CONNECTED], 0, realsink);
733 g_signal_emit (G_OBJECT (realsink), gst_real_pad_signals[REAL_CONNECTED], 0, realsrc);
735 /* now tell the scheduler(s) */
737 gst_scheduler_pad_connect (realsrc->sched, (GstPad *)realsrc, (GstPad *)realsink);
738 else if (realsink->sched)
739 gst_scheduler_pad_connect (realsink->sched, (GstPad *)realsrc, (GstPad *)realsink);
741 GST_INFO (GST_CAT_PADS, "connected %s:%s and %s:%s",
742 GST_DEBUG_PAD_NAME (srcpad), GST_DEBUG_PAD_NAME (sinkpad));
749 * @srcpad: the source pad to connect
750 * @sinkpad: the sink pad to connect
752 * Connects the source pad to the sink pad.
754 * Returns: TRUE if the pad could be connected, FALSE otherwise
757 gst_pad_connect (GstPad *srcpad, GstPad *sinkpad)
759 return gst_pad_connect_filtered (srcpad, sinkpad, NULL);
763 * gst_pad_set_parent:
764 * @pad: the pad to set the parent
765 * @parent: the object to set the parent to
767 * Sets the parent object of a pad.
770 gst_pad_set_parent (GstPad *pad,
773 g_return_if_fail (pad != NULL);
774 g_return_if_fail (GST_IS_PAD (pad));
775 g_return_if_fail (GST_PAD_PARENT (pad) == NULL);
776 g_return_if_fail (parent != NULL);
777 g_return_if_fail (GST_IS_OBJECT (parent));
778 g_return_if_fail ((gpointer)pad != (gpointer)parent);
780 gst_object_set_parent (GST_OBJECT (pad), parent);
784 * gst_pad_get_parent:
785 * @pad: the pad to get the parent from
787 * Get the parent object of this pad.
789 * Returns: the parent object
792 gst_pad_get_parent (GstPad *pad)
794 g_return_val_if_fail (pad != NULL, NULL);
795 g_return_val_if_fail (GST_IS_PAD (pad), NULL);
797 return GST_PAD_PARENT (pad);
801 * gst_pad_get_padtemplate:
802 * @pad: the pad to get the padtemplate from
804 * Get the padtemplate object of this pad.
806 * Returns: the padtemplate object
809 gst_pad_get_padtemplate (GstPad *pad)
811 g_return_val_if_fail (pad != NULL, NULL);
812 g_return_val_if_fail (GST_IS_PAD (pad), NULL);
814 return GST_PAD_PADTEMPLATE (pad);
820 * @pad: the pad to set the scheduler for
821 * @sched: The scheduler to set
823 * Set the scheduler for the pad
826 gst_pad_set_sched (GstPad *pad, GstScheduler *sched)
828 g_return_if_fail (pad != NULL);
829 g_return_if_fail (GST_IS_PAD (pad));
831 GST_RPAD_SCHED(pad) = sched;
836 * @pad: the pad to get the scheduler from
838 * Get the scheduler of the pad
840 * Returns: the scheduler of the pad.
843 gst_pad_get_sched (GstPad *pad)
845 g_return_val_if_fail (pad != NULL, NULL);
846 g_return_val_if_fail (GST_IS_PAD (pad), NULL);
848 return GST_RPAD_SCHED(pad);
852 * gst_pad_unset_sched:
853 * @pad: the pad to unset the scheduler for
855 * Unset the scheduler for the pad
858 gst_pad_unset_sched (GstPad *pad)
860 g_return_if_fail (pad != NULL);
861 g_return_if_fail (GST_IS_PAD (pad));
863 GST_RPAD_SCHED(pad) = NULL;
867 * gst_pad_get_real_parent:
868 * @pad: the pad to get the parent from
870 * Get the real parent object of this pad. If the pad
871 * is a ghostpad, the actual owner of the real pad is
872 * returned, as opposed to the gst_pad_get_parent().
874 * Returns: the parent object
877 gst_pad_get_real_parent (GstPad *pad)
879 g_return_val_if_fail (pad != NULL, NULL);
880 g_return_val_if_fail (GST_IS_PAD (pad), NULL);
882 return GST_PAD_PARENT (GST_PAD (GST_PAD_REALIZE (pad)));
886 * gst_pad_add_ghost_pad:
887 * @pad: the pad to set the ghost parent
888 * @ghostpad: the ghost pad to add
890 * Add a ghost pad to a pad.
893 gst_pad_add_ghost_pad (GstPad *pad,
898 g_return_if_fail (pad != NULL);
899 g_return_if_fail (GST_IS_PAD (pad));
900 g_return_if_fail (ghostpad != NULL);
901 g_return_if_fail (GST_IS_GHOST_PAD (ghostpad));
903 realpad = GST_PAD_REALIZE (pad);
905 realpad->ghostpads = g_list_prepend (realpad->ghostpads, ghostpad);
910 * gst_pad_remove_ghost_pad:
911 * @pad: the pad to remove the ghost parent
912 * @ghostpad: the ghost pad to remove from the pad
914 * Remove a ghost pad from a pad.
917 gst_pad_remove_ghost_pad (GstPad *pad,
922 g_return_if_fail (pad != NULL);
923 g_return_if_fail (GST_IS_PAD (pad));
924 g_return_if_fail (ghostpad != NULL);
925 g_return_if_fail (GST_IS_GHOST_PAD (ghostpad));
927 realpad = GST_PAD_REALIZE (pad);
929 realpad->ghostpads = g_list_remove (realpad->ghostpads, ghostpad);
933 * gst_pad_get_ghost_pad_list:
934 * @pad: the pad to get the ghost parents from
936 * Get the ghost parents of this pad.
938 * Returns: a GList of ghost pads
941 gst_pad_get_ghost_pad_list (GstPad *pad)
943 g_return_val_if_fail (pad != NULL, NULL);
944 g_return_val_if_fail (GST_IS_PAD (pad), NULL);
946 return GST_PAD_REALIZE(pad)->ghostpads;
949 /* an internal caps negotiation helper function does:
951 * 1. optinally calls the pad connect function with the provided caps
952 * 2. deal with the result code of the connect function
953 * 3. set fixed caps on the pad.
955 static GstPadConnectReturn
956 gst_pad_try_set_caps_func (GstRealPad *pad, GstCaps *caps, gboolean notify)
959 GstElement *parent = GST_PAD_PARENT (pad);
961 /* if this pad has a parent and the parent is not READY, delay the
963 if (parent && GST_STATE (parent) < GST_STATE_READY)
964 return GST_PAD_CONNECT_DELAYED;
966 GST_INFO (GST_CAT_CAPS, "trying to set caps %p on pad %s:%s",
967 caps, GST_DEBUG_PAD_NAME (pad));
969 /* we need to notify the connect function */
970 if (notify && GST_RPAD_CONNECTFUNC (pad)) {
971 GstPadConnectReturn res;
974 GST_INFO (GST_CAT_CAPS, "calling connect function on pad %s:%s",
975 GST_DEBUG_PAD_NAME (pad));
977 /* call the connect function */
978 res = GST_RPAD_CONNECTFUNC (pad) (GST_PAD (pad), caps);
981 case GST_PAD_CONNECT_REFUSED:
982 debug_string = "REFUSED";
984 case GST_PAD_CONNECT_OK:
987 case GST_PAD_CONNECT_DONE:
988 debug_string = "DONE";
990 case GST_PAD_CONNECT_DELAYED:
991 debug_string = "DELAYED";
994 g_warning ("unknown return code from connect function of pad %s:%s",
995 GST_DEBUG_PAD_NAME (pad));
996 return GST_PAD_CONNECT_REFUSED;
999 GST_INFO (GST_CAT_CAPS, "got reply %s (%d) from connect function on pad %s:%s",
1000 debug_string, res, GST_DEBUG_PAD_NAME (pad));
1002 /* done means the connect function called another caps negotiate function
1003 * on this pad that succeeded, we dont need to continue */
1004 if (res == GST_PAD_CONNECT_DONE) {
1005 GST_INFO (GST_CAT_CAPS, "pad %s:%s is done", GST_DEBUG_PAD_NAME (pad));
1006 return GST_PAD_CONNECT_DONE;
1008 if (res == GST_PAD_CONNECT_REFUSED) {
1009 GST_INFO (GST_CAT_CAPS, "pad %s:%s doesn't accept caps",
1010 GST_DEBUG_PAD_NAME (pad));
1011 return GST_PAD_CONNECT_REFUSED;
1014 /* we can only set caps on the pad if they are ficed */
1015 if (GST_CAPS_IS_FIXED (caps)) {
1017 GST_INFO (GST_CAT_CAPS, "setting caps on pad %s:%s",
1018 GST_DEBUG_PAD_NAME (pad));
1019 /* if we got this far all is ok, remove the old caps, set the new one */
1020 oldcaps = GST_PAD_CAPS (pad);
1021 if (caps) gst_caps_ref (caps);
1022 GST_PAD_CAPS (pad) = caps;
1023 if (oldcaps) gst_caps_unref (oldcaps);
1026 GST_INFO (GST_CAT_CAPS, "caps are not fixed on pad %s:%s, not setting them yet",
1027 GST_DEBUG_PAD_NAME (pad));
1030 return GST_PAD_CONNECT_OK;
1034 * gst_pad_try_set_caps:
1035 * @pad: the pad to try to set the caps on
1036 * @caps: the caps to set
1038 * Try to set the caps on the given pad.
1040 * Returns: TRUE if the caps could be set
1043 gst_pad_try_set_caps (GstPad *pad, GstCaps *caps)
1046 GstRealPad *peer, *realpad;
1048 realpad = GST_PAD_REALIZE (pad);
1049 peer = GST_RPAD_PEER (realpad);
1051 GST_INFO (GST_CAT_CAPS, "trying to set caps %p on pad %s:%s",
1052 caps, GST_DEBUG_PAD_NAME (realpad));
1054 /* setting non fixed caps on a pad is not allowed */
1055 if (!GST_CAPS_IS_FIXED (caps)) {
1056 g_warning ("trying to set non fixed caps on pad %s:%s",
1057 GST_DEBUG_PAD_NAME (realpad));
1058 gst_caps_debug (caps);
1061 /* if we have a peer try to set the caps, notifying the peerpad
1062 * if it has a connect function */
1063 if (peer && !gst_pad_try_set_caps_func (peer, caps, TRUE))
1065 /* then try to set our own caps, we don't need to be notified */
1066 if (!gst_pad_try_set_caps_func (realpad, caps, FALSE))
1072 /* this is a caps negotiation convenience routine, it performs:
1074 * 1. optionally clear any pad caps
1075 * 2. calculate the intersection between the two pad tamplate/getcaps caps
1076 * 3. calculate the intersection with the (optional) filtercaps.
1077 * 4. store the intersection in the pad filter
1078 * 5. store the app filtercaps in the pad appfilter.
1079 * 6. start the caps negotiation.
1082 gst_pad_try_reconnect_filtered_func (GstRealPad *srcpad, GstRealPad *sinkpad, GstCaps *filtercaps, gboolean clear)
1084 GstCaps *srccaps, *sinkcaps;
1085 GstCaps *intersection = NULL;
1086 GstRealPad *realsrc, *realsink;
1088 realsrc = GST_PAD_REALIZE (srcpad);
1089 realsink = GST_PAD_REALIZE (sinkpad);
1091 g_return_val_if_fail (GST_RPAD_PEER (realsrc) != NULL, FALSE);
1092 g_return_val_if_fail (GST_RPAD_PEER (realsink) == realsrc, FALSE);
1094 /* optinally clear the caps */
1096 GST_INFO (GST_CAT_PADS, "reconnect filtered %s:%s and %s:%s, clearing caps",
1097 GST_DEBUG_PAD_NAME (realsrc), GST_DEBUG_PAD_NAME (realsink));
1099 GST_PAD_CAPS (GST_PAD (realsrc)) = NULL;
1100 GST_PAD_CAPS (GST_PAD (realsink)) = NULL;
1103 GST_INFO (GST_CAT_PADS, "reconnect filtered %s:%s and %s:%s",
1104 GST_DEBUG_PAD_NAME (realsrc), GST_DEBUG_PAD_NAME (realsink));
1107 srccaps = gst_pad_get_caps (GST_PAD (realsrc));
1108 GST_INFO (GST_CAT_PADS, "dumping caps of pad %s:%s", GST_DEBUG_PAD_NAME (realsrc));
1109 gst_caps_debug (srccaps);
1110 sinkcaps = gst_pad_get_caps (GST_PAD (realsink));
1111 GST_INFO (GST_CAT_PADS, "dumping caps of pad %s:%s", GST_DEBUG_PAD_NAME (realsink));
1112 gst_caps_debug (sinkcaps);
1114 /* first take the intersection of the pad caps */
1115 intersection = gst_caps_intersect (srccaps, sinkcaps);
1117 /* if we have no intersection but one of the caps was not NULL.. */
1118 if (!intersection && (srccaps || sinkcaps )) {
1119 /* the intersection is NULL but the pad caps were not both NULL,
1120 * this means they have no common format */
1121 GST_INFO (GST_CAT_PADS, "pads %s:%s and %s:%s have no common type",
1122 GST_DEBUG_PAD_NAME (realsrc), GST_DEBUG_PAD_NAME (realsink));
1126 GST_INFO (GST_CAT_PADS, "pads %s:%s and %s:%s intersected to %s caps",
1127 GST_DEBUG_PAD_NAME (realsrc), GST_DEBUG_PAD_NAME (realsink),
1128 ((intersection && GST_CAPS_IS_FIXED (intersection)) ? "fixed" : "variable"));
1130 /* then filter this against the app filter */
1132 GstCaps *filtered_intersection = gst_caps_intersect (intersection, filtercaps);
1134 /* get rid of the old intersection here */
1135 gst_caps_unref (intersection);
1137 if (!filtered_intersection) {
1138 GST_INFO (GST_CAT_PADS, "filtered connection between pads %s:%s and %s:%s is empty",
1139 GST_DEBUG_PAD_NAME (realsrc), GST_DEBUG_PAD_NAME (realsink));
1142 intersection = filtered_intersection;
1144 /* keep a reference to the app caps */
1145 GST_RPAD_APPFILTER (realsink) = filtercaps;
1146 GST_RPAD_APPFILTER (realsrc) = filtercaps;
1149 GST_DEBUG (GST_CAT_CAPS, "setting filter for connection to:\n");
1150 gst_caps_debug (intersection);
1152 GST_RPAD_FILTER (realsrc) = intersection;
1153 GST_RPAD_FILTER (realsink) = intersection;
1155 return gst_pad_perform_negotiate (GST_PAD (realsrc), GST_PAD (realsink));
1159 * gst_pad_perform_negotiate:
1161 * @sinkpad: a sinkpad
1163 * Try to negotiate the pads.
1165 * Returns: a boolean indicating the pad succesfully negotiated.
1168 gst_pad_perform_negotiate (GstPad *srcpad, GstPad *sinkpad)
1170 GstCaps *intersection;
1171 GstRealPad *realsrc, *realsink;
1173 g_return_val_if_fail (srcpad != NULL, FALSE);
1174 g_return_val_if_fail (sinkpad != NULL, FALSE);
1176 realsrc = GST_PAD_REALIZE (srcpad);
1177 realsink = GST_PAD_REALIZE (sinkpad);
1179 g_return_val_if_fail (GST_RPAD_PEER (realsrc) != NULL, FALSE);
1180 g_return_val_if_fail (GST_RPAD_PEER (realsink) == realsrc, FALSE);
1182 /* it doesn't matter which filter we take */
1183 intersection = GST_RPAD_FILTER (realsrc);
1185 /* no negotiation is performed it the pads have filtercaps */
1187 GstPadConnectReturn res;
1189 res = gst_pad_try_set_caps_func (realsrc, intersection, TRUE);
1190 if (res == GST_PAD_CONNECT_REFUSED)
1192 if (res == GST_PAD_CONNECT_DONE)
1195 res = gst_pad_try_set_caps_func (realsink, intersection, TRUE);
1196 if (res == GST_PAD_CONNECT_REFUSED)
1198 if (res == GST_PAD_CONNECT_DONE)
1205 * gst_pad_try_reconnect_filtered:
1206 * @pad: the pad to reconnect
1207 * @caps: the capabilities to use in the reconnectiong
1209 * Try to reconnect this pad and its peer with the specified caps
1211 * Returns: a boolean indicating the peer pad could accept the caps.
1214 gst_pad_try_reconnect_filtered (GstPad *srcpad, GstPad *sinkpad, GstCaps *filtercaps)
1216 GstRealPad *realsrc, *realsink;
1218 g_return_val_if_fail (srcpad != NULL, FALSE);
1219 g_return_val_if_fail (sinkpad != NULL, FALSE);
1221 realsrc = GST_PAD_REALIZE (srcpad);
1222 realsink = GST_PAD_REALIZE (sinkpad);
1224 g_return_val_if_fail (GST_RPAD_PEER (realsrc) != NULL, FALSE);
1225 g_return_val_if_fail (GST_RPAD_PEER (realsink) == realsrc, FALSE);
1227 return gst_pad_try_reconnect_filtered_func (realsrc, realsink, filtercaps, TRUE);
1231 * gst_pad_reconnect_filtered:
1232 * @pad: the pad to reconnect
1233 * @caps: the capabilities to use in the reconnectiong
1235 * Try to reconnect this pad and its peer with the specified caps.
1237 * Returns: a boolean indicating the peer pad could accept the caps.
1238 * if FALSE is returned, the pads are disconnected.
1241 gst_pad_reconnect_filtered (GstPad *srcpad, GstPad *sinkpad, GstCaps *filtercaps)
1243 GstRealPad *realsrc, *realsink;
1245 g_return_val_if_fail (srcpad != NULL, FALSE);
1246 g_return_val_if_fail (sinkpad != NULL, FALSE);
1248 realsrc = GST_PAD_REALIZE (srcpad);
1249 realsink = GST_PAD_REALIZE (sinkpad);
1251 g_return_val_if_fail (GST_RPAD_PEER (realsrc) != NULL, FALSE);
1252 g_return_val_if_fail (GST_RPAD_PEER (realsink) == realsrc, FALSE);
1254 if (!gst_pad_try_reconnect_filtered_func (realsrc, realsink, filtercaps, TRUE)) {
1255 gst_pad_disconnect (srcpad, GST_PAD (GST_PAD_PEER (srcpad)));
1262 * gst_pad_proxy_connect:
1263 * @pad: the pad to proxy to
1264 * @caps: the capabilities to use in the proxying
1266 * Proxy the connect function to the specified pad.
1268 * Returns: a boolean indicating the peer pad could accept the caps.
1271 gst_pad_proxy_connect (GstPad *pad, GstCaps *caps)
1273 GstRealPad *peer, *realpad;
1275 realpad = GST_PAD_REALIZE (pad);
1277 peer = GST_RPAD_PEER (realpad);
1279 GST_INFO (GST_CAT_CAPS, "proxy connect to pad %s:%s",
1280 GST_DEBUG_PAD_NAME (realpad));
1282 if (peer && !gst_pad_try_set_caps_func (peer, caps, TRUE))
1283 return GST_PAD_CONNECT_REFUSED;
1284 if (!gst_pad_try_set_caps_func (realpad, caps, FALSE))
1285 return GST_PAD_CONNECT_REFUSED;
1287 return GST_PAD_CONNECT_OK;
1292 * @pad: the pad to get the capabilities from
1294 * Get the capabilities of this pad.
1296 * Returns: the capabilities of this pad
1299 gst_pad_get_caps (GstPad *pad)
1301 GstRealPad *realpad;
1303 g_return_val_if_fail (pad != NULL, NULL);
1304 g_return_val_if_fail (GST_IS_PAD (pad), NULL);
1306 realpad = GST_PAD_REALIZE (pad);
1308 GST_DEBUG (GST_CAT_CAPS, "get pad caps of %s:%s (%p)\n",
1309 GST_DEBUG_PAD_NAME (realpad), realpad);
1311 if (GST_PAD_CAPS (realpad)) {
1312 GST_DEBUG (GST_CAT_CAPS, "using pad real caps\n");
1313 return GST_PAD_CAPS (realpad);
1315 else if GST_RPAD_GETCAPSFUNC (realpad) {
1316 GST_DEBUG (GST_CAT_CAPS, "using pad get function\n");
1317 return GST_RPAD_GETCAPSFUNC (realpad) (GST_PAD_CAST (realpad), NULL);
1319 else if (GST_PAD_PADTEMPLATE (realpad)) {
1320 GST_DEBUG (GST_CAT_CAPS, "using pad template\n");
1321 return GST_PADTEMPLATE_CAPS (GST_PAD_PADTEMPLATE (realpad));
1323 GST_DEBUG (GST_CAT_CAPS, "pad has no caps\n");
1329 * gst_pad_get_padtemplate_caps:
1330 * @pad: the pad to get the capabilities from
1332 * Get the capabilities of this pad.
1334 * Returns: a list of the capabilities of this pad
1337 gst_pad_get_padtemplate_caps (GstPad *pad)
1339 g_return_val_if_fail (pad != NULL, NULL);
1340 g_return_val_if_fail (GST_IS_PAD (pad), NULL);
1342 if (GST_PAD_PADTEMPLATE (pad))
1343 return GST_PADTEMPLATE_CAPS (GST_PAD_PADTEMPLATE (pad));
1350 * gst_padtemplate_get_caps_by_name:
1351 * @templ: the padtemplate to get the capabilities from
1352 * @name: the name of the capability to get
1354 * Get the capability with the given name from this padtemplate.
1356 * Returns: a capability or NULL if not found
1359 gst_padtemplate_get_caps_by_name (GstPadTemplate *templ, const gchar *name)
1363 g_return_val_if_fail (templ != NULL, NULL);
1365 caps = GST_PADTEMPLATE_CAPS (templ);
1369 return gst_caps_get_by_name (caps, name);
1373 * gst_pad_check_compatibility:
1374 * @srcpad: the srcpad to check
1375 * @sinkpad: the sinkpad to check against
1377 * Check if two pads have compatible capabilities.
1379 * Returns: TRUE if they are compatible or the capabilities
1380 * could not be checked
1383 gst_pad_check_compatibility (GstPad *srcpad, GstPad *sinkpad)
1385 g_return_val_if_fail (srcpad != NULL, FALSE);
1386 g_return_val_if_fail (GST_IS_PAD (srcpad), FALSE);
1387 g_return_val_if_fail (sinkpad != NULL, FALSE);
1388 g_return_val_if_fail (GST_IS_PAD (sinkpad), FALSE);
1390 if (GST_PAD_CAPS (srcpad) && GST_PAD_CAPS (sinkpad)) {
1391 if (!gst_caps_check_compatibility (GST_PAD_CAPS (srcpad), GST_PAD_CAPS (sinkpad))) {
1399 GST_DEBUG (GST_CAT_PADS, "could not check capabilities of pads (%s:%s) and (%s:%s) %p %p\n",
1400 GST_DEBUG_PAD_NAME (srcpad), GST_DEBUG_PAD_NAME (sinkpad),
1401 GST_PAD_CAPS (srcpad), GST_PAD_CAPS (sinkpad));
1408 * @pad: the pad to get the peer from
1410 * Get the peer pad of this pad.
1412 * Returns: the peer pad
1415 gst_pad_get_peer (GstPad *pad)
1417 g_return_val_if_fail (pad != NULL, NULL);
1418 g_return_val_if_fail (GST_IS_PAD (pad), NULL);
1420 return GST_PAD (GST_PAD_PEER (pad));
1424 * gst_pad_get_allowed_caps:
1425 * @pad: the pad to get the allowed caps from
1427 * Gst the caps of the allowed media types that can
1428 * go through this pad.
1430 * Returns: the allowed caps
1433 gst_pad_get_allowed_caps (GstPad *pad)
1435 g_return_val_if_fail (pad != NULL, NULL);
1436 g_return_val_if_fail (GST_IS_PAD (pad), NULL);
1438 GST_DEBUG (GST_CAT_PROPERTIES, "get allowed caps of %s:%s\n", GST_DEBUG_PAD_NAME (pad));
1440 return gst_caps_copy (GST_RPAD_FILTER (pad));
1444 * gst_pad_get_allowed_caps:
1445 * @pad: the pad to get the allowed caps from
1447 * Gst the caps of the allowed media types that can
1448 * go through this pad.
1450 * Returns: the allowed caps
1453 gst_pad_recalc_allowed_caps (GstPad *pad)
1457 g_return_if_fail (pad != NULL);
1458 g_return_if_fail (GST_IS_PAD (pad));
1460 GST_DEBUG (GST_CAT_PROPERTIES, "set allowed caps of %s:%s\n", GST_DEBUG_PAD_NAME (pad));
1462 peer = GST_RPAD_PEER (pad);
1464 gst_pad_try_reconnect_filtered (pad, GST_PAD (peer), GST_RPAD_APPFILTER (pad));
1468 * gst_pad_get_bufferpool:
1469 * @pad: the pad to get the bufferpool from
1471 * Get the bufferpool of the peer pad of the given
1474 * Returns: The GstBufferPool or NULL.
1477 gst_pad_get_bufferpool (GstPad *pad)
1481 g_return_val_if_fail (pad != NULL, NULL);
1482 g_return_val_if_fail (GST_IS_PAD (pad), NULL);
1484 peer = GST_RPAD_PEER (pad);
1489 GST_DEBUG_ENTER ("(%s:%s)", GST_DEBUG_PAD_NAME (pad));
1491 if (peer->bufferpoolfunc) {
1492 GST_DEBUG (GST_CAT_PADS, "calling bufferpoolfunc &%s (@%p) of peer pad %s:%s\n",
1493 GST_DEBUG_FUNCPTR_NAME (peer->bufferpoolfunc), &peer->bufferpoolfunc, GST_DEBUG_PAD_NAME (((GstPad*) peer)));
1494 return (peer->bufferpoolfunc) (((GstPad*) peer));
1496 GST_DEBUG (GST_CAT_PADS, "no bufferpoolfunc for peer pad %s:%s at %p\n",
1497 GST_DEBUG_PAD_NAME (((GstPad*) peer)), &peer->bufferpoolfunc);
1503 gst_real_pad_dispose (GObject *object)
1505 GstPad *pad = GST_PAD (object);
1507 /* No connected pad can ever be disposed.
1508 * It has to have a parent to be connected and a parent would hold a reference */
1509 g_assert (GST_PAD_PEER (pad) == NULL);
1511 GST_DEBUG (GST_CAT_REFCOUNTING, "dispose %s:%s\n", GST_DEBUG_PAD_NAME(pad));
1513 if (GST_PAD_PADTEMPLATE (pad)){
1514 GST_DEBUG (GST_CAT_REFCOUNTING, "unreffing padtemplate'%s'\n", GST_OBJECT_NAME (GST_PAD_PADTEMPLATE (pad)));
1515 gst_object_unref (GST_OBJECT (GST_PAD_PADTEMPLATE (pad)));
1516 GST_PAD_PADTEMPLATE (pad) = NULL;
1519 /* we destroy the ghostpads, because they are nothing without the real pad */
1520 if (GST_REAL_PAD (pad)->ghostpads) {
1521 GList *orig, *ghostpads;
1523 orig = ghostpads = g_list_copy (GST_REAL_PAD (pad)->ghostpads);
1526 GstPad *ghostpad = GST_PAD (ghostpads->data);
1528 if (GST_IS_ELEMENT (GST_OBJECT_PARENT (ghostpad))){
1529 GST_DEBUG (GST_CAT_REFCOUNTING, "removing ghost pad from element '%s'\n",
1530 GST_OBJECT_NAME (GST_OBJECT_PARENT (ghostpad)));
1532 gst_element_remove_ghost_pad (GST_ELEMENT (GST_OBJECT_PARENT (ghostpad)), GST_PAD (ghostpad));
1534 ghostpads = g_list_next (ghostpads);
1537 g_list_free (GST_REAL_PAD(pad)->ghostpads);
1540 if (GST_IS_ELEMENT (GST_OBJECT_PARENT (pad))){
1541 GST_DEBUG (GST_CAT_REFCOUNTING, "removing pad from element '%s'\n",
1542 GST_OBJECT_NAME (GST_OBJECT (GST_ELEMENT (GST_OBJECT_PARENT (pad)))));
1544 gst_element_remove_pad (GST_ELEMENT (GST_OBJECT_PARENT (pad)), pad);
1547 G_OBJECT_CLASS (real_pad_parent_class)->dispose (object);
1551 #ifndef GST_DISABLE_LOADSAVE
1553 * gst_pad_load_and_connect:
1554 * @self: the XML node to read the description from
1555 * @parent: the element that has the pad
1557 * Read the pad definition from the XML node and connect the given pad
1558 * in element to a pad of an element up in the hierarchy.
1561 gst_pad_load_and_connect (xmlNodePtr self,
1564 xmlNodePtr field = self->xmlChildrenNode;
1565 GstPad *pad = NULL, *targetpad;
1566 guchar *peer = NULL;
1569 GstObject *grandparent;
1572 if (!strcmp (field->name, "name")) {
1573 pad = gst_element_get_pad (GST_ELEMENT (parent), xmlNodeGetContent (field));
1575 else if (!strcmp(field->name, "peer")) {
1576 peer = xmlNodeGetContent (field);
1578 field = field->next;
1580 g_return_if_fail (pad != NULL);
1582 if (peer == NULL) return;
1584 split = g_strsplit (peer, ".", 2);
1586 g_return_if_fail (split[0] != NULL);
1587 g_return_if_fail (split[1] != NULL);
1589 grandparent = gst_object_get_parent (parent);
1591 if (grandparent && GST_IS_BIN (grandparent)) {
1592 target = gst_bin_get_by_name_recurse_up (GST_BIN (grandparent), split[0]);
1597 if (target == NULL) goto cleanup;
1599 targetpad = gst_element_get_pad (target, split[1]);
1601 if (targetpad == NULL) goto cleanup;
1603 gst_pad_connect (pad, targetpad);
1610 * gst_pad_save_thyself:
1611 * @pad: the pad to save
1612 * @parent: the parent XML node to save the description in
1614 * Saves the pad into an xml representation
1616 * Returns: the xml representation of the pad
1619 gst_pad_save_thyself (GstObject *object,
1622 GstRealPad *realpad;
1625 g_return_val_if_fail (GST_IS_REAL_PAD (object), NULL);
1627 realpad = GST_REAL_PAD(object);
1629 xmlNewChild(parent,NULL,"name", GST_PAD_NAME (realpad));
1630 if (GST_RPAD_PEER(realpad) != NULL) {
1631 peer = GST_PAD(GST_RPAD_PEER(realpad));
1632 /* first check to see if the peer's parent's parent is the same */
1633 /* we just save it off */
1634 xmlNewChild(parent,NULL,"peer",g_strdup_printf("%s.%s",
1635 GST_OBJECT_NAME (GST_PAD_PARENT (peer)), GST_PAD_NAME (peer)));
1637 xmlNewChild(parent,NULL,"peer","");
1643 * gst_pad_ghost_save_thyself:
1644 * @pad: the pad to save
1646 * @parent: the parent XML node to save the description in
1648 * Saves the ghost pad into an xml representation.
1650 * Returns: the xml representation of the pad
1653 gst_pad_ghost_save_thyself (GstPad *pad,
1659 g_return_val_if_fail (GST_IS_GHOST_PAD (pad), NULL);
1661 self = xmlNewChild (parent, NULL, "ghostpad", NULL);
1662 xmlNewChild (self, NULL, "name", GST_PAD_NAME (pad));
1663 xmlNewChild (self, NULL, "parent", GST_OBJECT_NAME (GST_PAD_PARENT (pad)));
1665 /* FIXME FIXME FIXME! */
1669 #endif /* GST_DISABLE_LOADSAVE */
1671 #ifndef gst_pad_push
1674 * @pad: the pad to push
1675 * @buf: the buffer to push
1677 * Push a buffer to the peer of the pad.
1680 gst_pad_push (GstPad *pad, GstBuffer *buf)
1682 GstRealPad *peer = GST_RPAD_PEER (pad);
1684 GST_DEBUG_ENTER ("(%s:%s)", GST_DEBUG_PAD_NAME (pad));
1686 g_return_if_fail (GST_PAD_DIRECTION (pad) == GST_PAD_SRC);
1689 g_warning ("push on pad %s:%s but it is unconnected", GST_DEBUG_PAD_NAME (pad));
1692 if (peer->chainhandler) {
1694 GST_DEBUG (GST_CAT_DATAFLOW, "calling chainhandler &%s of peer pad %s:%s\n",
1695 GST_DEBUG_FUNCPTR_NAME (peer->chainhandler), GST_DEBUG_PAD_NAME (GST_PAD (peer)));
1696 (peer->chainhandler) (GST_PAD_CAST (peer), buf);
1700 g_warning ("trying to push a NULL buffer on pad %s:%s", GST_DEBUG_PAD_NAME (peer));
1705 g_warning ("(internal error) push on pad %s:%s but it has no chainhandler", GST_DEBUG_PAD_NAME (peer));
1708 /* clean up the mess here */
1710 if (GST_IS_BUFFER (buf))
1711 gst_buffer_unref (buf);
1713 gst_event_free (GST_EVENT (buf));
1718 #ifndef gst_pad_pull
1721 * @pad: the pad to pull
1723 * Pull a buffer from the peer pad.
1725 * Returns: a new buffer from the peer pad.
1728 gst_pad_pull (GstPad *pad)
1730 GstRealPad *peer = GST_RPAD_PEER(pad);
1732 GST_DEBUG_ENTER("(%s:%s)",GST_DEBUG_PAD_NAME(pad));
1734 g_return_val_if_fail (GST_PAD_DIRECTION (pad) == GST_PAD_SINK, NULL);
1737 gst_element_error (GST_PAD_PARENT (pad),
1738 "pull on pad %s:%s but it was unconnected",
1739 GST_ELEMENT_NAME (GST_PAD_PARENT (pad)), GST_PAD_NAME (pad),
1743 if (peer->gethandler) {
1746 GST_DEBUG (GST_CAT_DATAFLOW, "calling gethandler %s of peer pad %s:%s\n",
1747 GST_DEBUG_FUNCPTR_NAME (peer->gethandler), GST_DEBUG_PAD_NAME (peer));
1749 buf = (peer->gethandler) (GST_PAD_CAST (peer));
1752 /* no null buffers allowed */
1753 gst_element_error (GST_PAD_PARENT (pad),
1754 "NULL buffer during pull on %s:%s", GST_DEBUG_PAD_NAME (pad), NULL);
1757 gst_element_error (GST_PAD_PARENT (pad),
1758 "(internal error) pull on pad %s:%s but the peer pad %s:%s has no gethandler",
1759 GST_DEBUG_PAD_NAME (pad), GST_DEBUG_PAD_NAME (peer),
1767 #ifndef gst_pad_pullregion
1769 * gst_pad_pullregion:
1770 * @pad: the pad to pull the region from
1771 * @type: the regiontype
1772 * @offset: the offset/start of the buffer to pull
1773 * @len: the length of the buffer to pull
1775 * Pull a buffer region from the peer pad. The region to pull can be
1776 * specified with a offset/lenght pair or with a start/legnth time
1777 * indicator as specified by the type parameter.
1779 * Returns: a new buffer from the peer pad with data in the specified
1783 gst_pad_pullregion (GstPad *pad, GstRegionType type, guint64 offset, guint64 len)
1786 GstBuffer *result = NULL;
1788 g_return_val_if_fail (GST_PAD_DIRECTION (pad) == GST_PAD_SINK, NULL);
1791 peer = GST_RPAD_PEER(pad);
1792 g_return_val_if_fail (peer != NULL, NULL);
1795 gst_buffer_unref (result);
1797 GST_DEBUG_ENTER("(%s:%s,%d,%lld,%lld)",GST_DEBUG_PAD_NAME(pad),type,offset,len);
1799 if (peer->pullregionfunc) {
1800 GST_DEBUG (GST_CAT_DATAFLOW, "calling pullregionfunc &%s of peer pad %s:%s\n",
1801 GST_DEBUG_FUNCPTR_NAME (peer->pullregionfunc), GST_DEBUG_PAD_NAME(GST_PAD_CAST (peer)));
1802 result = (peer->pullregionfunc) (GST_PAD_CAST (peer), type, offset, len);
1804 GST_DEBUG (GST_CAT_DATAFLOW,"no pullregionfunc\n");
1810 while (result && !(GST_BUFFER_OFFSET (result) == offset &&
1811 GST_BUFFER_SIZE (result) == len));
1819 * @pad: the pad to peek
1821 * Peek for a buffer from the peer pad.
1823 * Returns: a from the peer pad or NULL if the peer has no buffer.
1826 gst_pad_peek (GstPad *pad)
1828 g_return_val_if_fail (GST_PAD_DIRECTION (pad) == GST_PAD_SINK, NULL);
1830 return GST_RPAD_BUFPEN (GST_RPAD_PEER (pad));
1835 * @padlist: A list of pads
1837 * Wait for a buffer on the list of pads.
1839 * Returns: The pad that has a buffer available, use
1840 * #gst_pad_pull to get the buffer.
1843 gst_pad_select (GList *padlist)
1847 pad = gst_scheduler_pad_select (GST_PAD_PARENT (padlist->data)->sched, padlist);
1854 * @pad: The first pad to perform the select on
1857 * Wait for a buffer on the given of pads.
1859 * Returns: The pad that has a buffer available, use
1860 * #gst_pad_pull to get the buffer.
1863 gst_pad_selectv (GstPad *pad, ...)
1866 GList *padlist = NULL;
1872 va_start (var_args, pad);
1875 padlist = g_list_prepend (padlist, pad);
1876 pad = va_arg (var_args, GstPad *);
1878 result = gst_pad_select (padlist);
1879 g_list_free (padlist);
1886 /************************************************************************
1891 static void gst_padtemplate_class_init (GstPadTemplateClass *klass);
1892 static void gst_padtemplate_init (GstPadTemplate *templ);
1900 static GstObject *padtemplate_parent_class = NULL;
1901 static guint gst_padtemplate_signals[TEMPL_LAST_SIGNAL] = { 0 };
1904 gst_padtemplate_get_type (void)
1906 static GType padtemplate_type = 0;
1908 if (!padtemplate_type) {
1909 static const GTypeInfo padtemplate_info = {
1910 sizeof(GstPadTemplateClass),
1913 (GClassInitFunc)gst_padtemplate_class_init,
1916 sizeof(GstPadTemplate),
1918 (GInstanceInitFunc)gst_padtemplate_init,
1921 padtemplate_type = g_type_register_static(GST_TYPE_OBJECT, "GstPadTemplate", &padtemplate_info, 0);
1923 return padtemplate_type;
1927 gst_padtemplate_class_init (GstPadTemplateClass *klass)
1929 GObjectClass *gobject_class;
1930 GstObjectClass *gstobject_class;
1932 gobject_class = (GObjectClass*)klass;
1933 gstobject_class = (GstObjectClass*)klass;
1935 padtemplate_parent_class = g_type_class_ref(GST_TYPE_OBJECT);
1937 gst_padtemplate_signals[TEMPL_PAD_CREATED] =
1938 g_signal_new ("pad_created", G_TYPE_FROM_CLASS(klass), G_SIGNAL_RUN_LAST,
1939 G_STRUCT_OFFSET (GstPadTemplateClass, pad_created), NULL, NULL,
1940 gst_marshal_VOID__POINTER, G_TYPE_NONE, 1,
1944 gstobject_class->path_string_separator = "*";
1948 gst_padtemplate_init (GstPadTemplate *templ)
1953 * gst_padtemplate_new:
1954 * @name_template: the name template
1955 * @direction: the direction for the template
1956 * @presence: the presence of the pad
1957 * @caps: a list of capabilities for the template
1958 * @...: more capabilities
1960 * Creates a new padtemplate from the given arguments.
1962 * Returns: the new padtemplate
1965 gst_padtemplate_new (gchar *name_template,
1966 GstPadDirection direction, GstPadPresence presence,
1969 GstPadTemplate *new;
1971 GstCaps *thecaps = NULL;
1973 g_return_val_if_fail (name_template != NULL, NULL);
1975 new = g_object_new(gst_padtemplate_get_type () ,NULL);
1977 GST_PADTEMPLATE_NAME_TEMPLATE (new) = name_template;
1978 GST_PADTEMPLATE_DIRECTION (new) = direction;
1979 GST_PADTEMPLATE_PRESENCE (new) = presence;
1981 va_start (var_args, caps);
1984 new->fixed &= caps->fixed;
1985 thecaps = gst_caps_append (thecaps, caps);
1986 caps = va_arg (var_args, GstCaps*);
1990 GST_PADTEMPLATE_CAPS (new) = thecaps;
1996 * gst_padtemplate_get_caps:
1997 * @templ: the padtemplate to use
1999 * Get the capabilities of the padtemplate
2001 * Returns: a GstCaps*
2004 gst_padtemplate_get_caps (GstPadTemplate *templ)
2006 g_return_val_if_fail (templ != NULL, NULL);
2008 return GST_PADTEMPLATE_CAPS (templ);
2011 #ifndef GST_DISABLE_LOADSAVE
2013 * gst_padtemplate_save_thyself:
2014 * @templ: the padtemplate to save
2015 * @parent: the parent XML tree
2017 * Saves the padtemplate into XML.
2019 * Returns: the new XML tree
2022 gst_padtemplate_save_thyself (GstPadTemplate *templ, xmlNodePtr parent)
2027 GST_DEBUG (GST_CAT_XML,"saving padtemplate %s\n", templ->name_template);
2029 xmlNewChild(parent,NULL,"nametemplate", templ->name_template);
2030 xmlNewChild(parent,NULL,"direction", (templ->direction == GST_PAD_SINK? "sink":"src"));
2032 switch (templ->presence) {
2033 case GST_PAD_ALWAYS:
2034 presence = "always";
2036 case GST_PAD_SOMETIMES:
2037 presence = "sometimes";
2039 case GST_PAD_REQUEST:
2040 presence = "request";
2043 presence = "unknown";
2046 xmlNewChild(parent,NULL,"presence", presence);
2048 if (GST_PADTEMPLATE_CAPS (templ)) {
2049 subtree = xmlNewChild (parent, NULL, "caps", NULL);
2050 gst_caps_save_thyself (GST_PADTEMPLATE_CAPS (templ), subtree);
2057 * gst_padtemplate_load_thyself:
2058 * @parent: the source XML tree
2060 * Loads a padtemplate from the XML tree.
2062 * Returns: the new padtemplate
2065 gst_padtemplate_load_thyself (xmlNodePtr parent)
2067 xmlNodePtr field = parent->xmlChildrenNode;
2068 GstPadTemplate *factory;
2069 gchar *name_template = NULL;
2070 GstPadDirection direction = GST_PAD_UNKNOWN;
2071 GstPadPresence presence = GST_PAD_ALWAYS;
2072 GstCaps *caps = NULL;
2075 if (!strcmp(field->name, "nametemplate")) {
2076 name_template = xmlNodeGetContent(field);
2078 if (!strcmp(field->name, "direction")) {
2079 gchar *value = xmlNodeGetContent(field);
2081 if (!strcmp(value, "sink")) {
2082 direction = GST_PAD_SINK;
2084 else if (!strcmp(value, "src")) {
2085 direction = GST_PAD_SRC;
2089 if (!strcmp(field->name, "presence")) {
2090 gchar *value = xmlNodeGetContent(field);
2092 if (!strcmp(value, "always")) {
2093 presence = GST_PAD_ALWAYS;
2095 else if (!strcmp(value, "sometimes")) {
2096 presence = GST_PAD_SOMETIMES;
2098 else if (!strcmp(value, "request")) {
2099 presence = GST_PAD_REQUEST;
2103 else if (!strcmp(field->name, "caps")) {
2104 caps = gst_caps_load_thyself (field);
2106 field = field->next;
2109 factory = gst_padtemplate_new (name_template, direction, presence, caps, NULL);
2113 #endif /* !GST_DISABLE_LOADSAVE */
2117 * gst_pad_set_element_private:
2118 * @pad: the pad to set the private data to
2119 * @priv: The private data to attach to the pad
2121 * Set the given private data pointer to the pad. This
2122 * function can only be used by the element that own the
2126 gst_pad_set_element_private (GstPad *pad, gpointer priv)
2128 pad->element_private = priv;
2132 * gst_pad_get_element_private:
2133 * @pad: the pad to get the private data of
2135 * Get the private data of a pad. The private data can
2136 * only be set by the parent element of this pad.
2138 * Returns: a pointer to the private data.
2141 gst_pad_get_element_private (GstPad *pad)
2143 return pad->element_private;
2147 /***** ghost pads *****/
2148 GType _gst_ghost_pad_type = 0;
2150 static void gst_ghost_pad_class_init (GstGhostPadClass *klass);
2151 static void gst_ghost_pad_init (GstGhostPad *pad);
2153 static GstPad *ghost_pad_parent_class = NULL;
2154 /* static guint gst_ghost_pad_signals[LAST_SIGNAL] = { 0 }; */
2157 gst_ghost_pad_get_type(void) {
2158 if (!_gst_ghost_pad_type) {
2159 static const GTypeInfo pad_info = {
2160 sizeof(GstGhostPadClass),
2163 (GClassInitFunc)gst_ghost_pad_class_init,
2166 sizeof(GstGhostPad),
2168 (GInstanceInitFunc)gst_ghost_pad_init,
2171 _gst_ghost_pad_type = g_type_register_static(GST_TYPE_PAD, "GstGhostPad", &pad_info, 0);
2173 return _gst_ghost_pad_type;
2177 gst_ghost_pad_class_init (GstGhostPadClass *klass)
2179 GObjectClass *gobject_class;
2181 gobject_class = (GObjectClass*)klass;
2183 ghost_pad_parent_class = g_type_class_ref(GST_TYPE_PAD);
2187 gst_ghost_pad_init (GstGhostPad *pad)
2189 pad->realpad = NULL;
2193 * gst_ghost_pad_new:
2194 * @name: name of the new ghost pad
2195 * @pad: the pad to create a ghost pad of
2197 * Create a new ghost pad associated with the given pad.
2199 * Returns: new ghost pad
2202 gst_ghost_pad_new (gchar *name,
2205 GstGhostPad *ghostpad;
2207 g_return_val_if_fail (name != NULL, NULL);
2208 g_return_val_if_fail (GST_IS_PAD (pad), NULL);
2210 ghostpad = g_object_new (gst_ghost_pad_get_type () ,NULL);
2211 gst_pad_set_name (GST_PAD (ghostpad), name);
2212 GST_GPAD_REALPAD (ghostpad) = GST_PAD_REALIZE (pad);
2213 GST_PAD_PADTEMPLATE (ghostpad) = GST_PAD_PADTEMPLATE (pad);
2215 /* add ourselves to the real pad's list of ghostpads */
2216 gst_pad_add_ghost_pad (pad, GST_PAD (ghostpad));
2218 /* FIXME need to ref the real pad here... ? */
2220 GST_DEBUG (GST_CAT_PADS, "created ghost pad \"%s\"\n", name);
2222 return GST_PAD (ghostpad);
2226 gst_pad_event_default_dispatch (GstPad *pad, GstElement *element, GstEvent *event)
2228 GList *pads = element->pads;
2231 GstPad *eventpad = GST_PAD (pads->data);
2232 pads = g_list_next (pads);
2234 /* for all pads in the opposite direction that are connected */
2235 if (GST_PAD_DIRECTION (eventpad) != GST_PAD_DIRECTION (pad) && GST_PAD_IS_CONNECTED (eventpad)) {
2236 if (GST_PAD_DIRECTION (eventpad) == GST_PAD_SRC) {
2237 gst_pad_push (eventpad, GST_BUFFER (gst_event_new (GST_EVENT_TYPE (event))));
2240 GstPad *peerpad = GST_PAD_CAST (GST_RPAD_PEER (eventpad));
2242 gst_pad_send_event (peerpad, gst_event_new (GST_EVENT_TYPE (event)));
2249 * gst_pad_event_default:
2250 * @pad: the pad to operate on
2251 * @event: the event to handle
2253 * Invoke the default event handler for the given pad.
2256 gst_pad_event_default (GstPad *pad, GstEvent *event)
2258 GstElement *element = GST_PAD_PARENT (pad);
2260 g_signal_emit (G_OBJECT (pad), gst_real_pad_signals[REAL_EVENT_RECEIVED], 0, event);
2262 switch (GST_EVENT_TYPE (event)) {
2264 gst_element_set_eos (element);
2265 gst_pad_event_default_dispatch (pad, element, event);
2266 gst_event_free (event);
2267 /* we have to try to schedule another element because this one is disabled */
2268 gst_element_yield (element);
2270 case GST_EVENT_FLUSH:
2272 gst_pad_event_default_dispatch (pad, element, event);
2273 gst_event_free (event);
2279 * gst_pad_send_event:
2280 * @pad: the pad to send the event to
2281 * @event: the event to send to the pad.
2283 * Send the event to the pad.
2285 * Returns: TRUE if the event was handled.
2288 gst_pad_send_event (GstPad *pad, GstEvent *event)
2290 gboolean handled = FALSE;
2292 g_return_val_if_fail (event, FALSE);
2294 if (GST_EVENT_SRC (event) == NULL)
2295 GST_EVENT_SRC (event) = gst_object_ref (GST_OBJECT (pad));
2297 GST_DEBUG (GST_CAT_EVENT, "have event %d on pad %s:%s\n",
2298 GST_EVENT_TYPE (event), GST_DEBUG_PAD_NAME (pad));
2300 if (GST_RPAD_EVENTFUNC (pad))
2301 handled = GST_RPAD_EVENTFUNC (pad) (pad, event);
2303 GST_DEBUG(GST_CAT_EVENT, "there's no event function for pad %s:%s\n", GST_DEBUG_PAD_NAME (pad));
2307 GST_DEBUG(GST_CAT_EVENT, "proceeding with default event behavior here\n");
2308 gst_pad_event_default (pad, event);