2677aa1dd201ef987710a3c586129de311c427a4
[platform/upstream/gstreamer.git] / gst / gstelement.c
1 /* GStreamer
2  * Copyright (C) 1999,2000 Erik Walthinsen <omega@cse.ogi.edu>
3  *                    2004 Wim Taymans <wim@fluendo.com>
4  *
5  * gstelement.c: The base element, all elements derive from this
6  *
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.
11  *
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.
16  *
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.
21  */
22 /**
23  * SECTION:gstelement
24  * @short_description: Abstract base class for all pipeline elements
25  * @see_also: #GstElementFactory, #GstPad
26  *
27  * GstElement is the base class needed to construct an element that can be
28  * used in a GStreamer pipeline.  As such, it is not a functional entity, and
29  * cannot do anything when placed in a pipeline.
30  * 
31  * The name of a GstElement can be get with gst_element_get_name() and set with
32  * gst_element_set_name().  For speed, GST_ELEMENT_NAME() can be used in the 
33  * core. Do not use this in plug-ins or applications in order to retain ABI 
34  * compatibility.
35  * 
36  * All elements have pads (of the type #GstPad).  These pads link to pads on
37  * other elements.  Buffers flow between these linked pads.
38  * A GstElement has a GList of #GstPad structures for all their input (or sink)
39  * and output (or source) pads.  
40  * Core and plug-in writers can add and remove pads with gst_element_add_pad()
41  * and gst_element_remove_pad().
42  *
43  * A pad of an element can be retrieved by name with gst_element_get_pad().
44  * An iterator of all pads can be retrieved with gst_element_iterate_pads().
45  * 
46  * Elements can be linked through their pads.
47  * If the link is straightforward, use the gst_element_link() 
48  * convenience function to link two elements, or gst_element_link_many() 
49  * for more elements in a row.
50  * Use gst_element_link_filtered() to link two elements constrained by
51  * a specified set of #GstCaps.
52  * For finer control, use gst_element_link_pads() and 
53  * gst_element_link_pads_filtered() to specify the pads to link on 
54  * each element by name.
55  * 
56  * Each element has a state (see #GstState).  You can get and set the state
57  * of an element with gst_element_get_state() and gst_element_set_state().  
58  * You can wait for an element to change it's state with gst_element_wait_state_change().
59  * To get a string representation of a #GstState, use 
60  * gst_element_state_get_name().
61  * 
62  * You can get and set a #GstClock on an element using gst_element_get_clock()
63  * and gst_element_set_clock().  
64  * Some elements can provide a clock for the pipeline if  
65  * gst_element_provides_clock() returns TRUE. With the gst_element_provide_clock()
66  * method one can retrieve the clock provided by such an element.
67  * Not all elements require a clock to operate correctly. If 
68  * gst_element_requires_clock() returns TRUE, a clock should be set on the element
69  * with gst_element_set_clock(). 
70  * Note that clock slection and distribution is normally handled by the toplevel 
71  * GstPipeline so the clock functions are only to be used in very specific situations.
72  */
73 #include "gst_private.h"
74 #include <glib.h>
75 #include <stdarg.h>
76 #include <gobject/gvaluecollector.h>
77
78 #include "gstelement.h"
79 #include "gstbus.h"
80 #include "gstmarshal.h"
81 #include "gsterror.h"
82 #include "gstevent.h"
83 #include "gstutils.h"
84 #include "gstinfo.h"
85 #include "gst-i18n-lib.h"
86
87 /* Element signals and args */
88 enum
89 {
90   STATE_CHANGE,
91   NEW_PAD,
92   PAD_REMOVED,
93   NO_MORE_PADS,
94   /* add more above */
95   LAST_SIGNAL
96 };
97
98 enum
99 {
100   ARG_0
101       /* FILL ME */
102 };
103
104 extern void __gst_element_details_clear (GstElementDetails * dp);
105 extern void __gst_element_details_copy (GstElementDetails * dest,
106     const GstElementDetails * src);
107
108 static void gst_element_class_init (GstElementClass * klass);
109 static void gst_element_init (GstElement * element);
110 static void gst_element_base_class_init (gpointer g_class);
111 static void gst_element_base_class_finalize (gpointer g_class);
112
113 static void gst_element_dispose (GObject * object);
114 static void gst_element_finalize (GObject * object);
115
116 static GstStateChangeReturn gst_element_change_state (GstElement * element,
117     GstStateChange transition);
118 static GstStateChangeReturn gst_element_get_state_func (GstElement * element,
119     GstState * state, GstState * pending, GTimeVal * timeout);
120 static void gst_element_set_bus_func (GstElement * element, GstBus * bus);
121
122 #ifndef GST_DISABLE_LOADSAVE
123 static xmlNodePtr gst_element_save_thyself (GstObject * object,
124     xmlNodePtr parent);
125 static void gst_element_restore_thyself (GstObject * parent, xmlNodePtr self);
126 #endif
127
128 GType _gst_element_type = 0;
129
130 static GstObjectClass *parent_class = NULL;
131 static guint gst_element_signals[LAST_SIGNAL] = { 0 };
132
133 GType
134 gst_element_get_type (void)
135 {
136   if (!_gst_element_type) {
137     static const GTypeInfo element_info = {
138       sizeof (GstElementClass),
139       gst_element_base_class_init,
140       gst_element_base_class_finalize,
141       (GClassInitFunc) gst_element_class_init,
142       NULL,
143       NULL,
144       sizeof (GstElement),
145       0,
146       (GInstanceInitFunc) gst_element_init,
147       NULL
148     };
149
150     _gst_element_type = g_type_register_static (GST_TYPE_OBJECT, "GstElement",
151         &element_info, G_TYPE_FLAG_ABSTRACT);
152   }
153   return _gst_element_type;
154 }
155
156 static void
157 gst_element_class_init (GstElementClass * klass)
158 {
159   GObjectClass *gobject_class;
160   GstObjectClass *gstobject_class;
161
162   gobject_class = (GObjectClass *) klass;
163   gstobject_class = (GstObjectClass *) klass;
164
165   parent_class = g_type_class_ref (GST_TYPE_OBJECT);
166
167   /**
168    * GstElement::pad-added:
169    * @gstelement: the object which received the signal
170    * @new_pad: the pad that has been added
171    *
172    * a new #GstPad has been added to the element.
173    */
174   gst_element_signals[NEW_PAD] =
175       g_signal_new ("pad-added", G_TYPE_FROM_CLASS (klass), G_SIGNAL_RUN_LAST,
176       G_STRUCT_OFFSET (GstElementClass, pad_added), NULL, NULL,
177       gst_marshal_VOID__OBJECT, G_TYPE_NONE, 1, G_TYPE_OBJECT);
178   /**
179    * GstElement::pad-removed:
180    * @gstelement: the object which received the signal
181    * @old_pad: the pad that has been removed
182    *
183    * a #GstPad has been removed from the element
184    */
185   gst_element_signals[PAD_REMOVED] =
186       g_signal_new ("pad-removed", G_TYPE_FROM_CLASS (klass), G_SIGNAL_RUN_LAST,
187       G_STRUCT_OFFSET (GstElementClass, pad_removed), NULL, NULL,
188       gst_marshal_VOID__OBJECT, G_TYPE_NONE, 1, G_TYPE_OBJECT);
189   /**
190    * GstElement::no-more-pads:
191    * @gstelement: the object which received the signal
192    *
193    * This signals that the element will not generate more dynamic pads.
194    */
195   gst_element_signals[NO_MORE_PADS] =
196       g_signal_new ("no-more-pads", G_TYPE_FROM_CLASS (klass),
197       G_SIGNAL_RUN_LAST, G_STRUCT_OFFSET (GstElementClass, no_more_pads), NULL,
198       NULL, gst_marshal_VOID__VOID, G_TYPE_NONE, 0);
199
200   gobject_class->dispose = GST_DEBUG_FUNCPTR (gst_element_dispose);
201   gobject_class->finalize = GST_DEBUG_FUNCPTR (gst_element_finalize);
202
203 #ifndef GST_DISABLE_LOADSAVE
204   gstobject_class->save_thyself = GST_DEBUG_FUNCPTR (gst_element_save_thyself);
205   gstobject_class->restore_thyself =
206       GST_DEBUG_FUNCPTR (gst_element_restore_thyself);
207 #endif
208
209   klass->change_state = GST_DEBUG_FUNCPTR (gst_element_change_state);
210   klass->get_state = GST_DEBUG_FUNCPTR (gst_element_get_state_func);
211   klass->set_bus = GST_DEBUG_FUNCPTR (gst_element_set_bus_func);
212   klass->numpadtemplates = 0;
213
214   klass->elementfactory = NULL;
215 }
216
217 static void
218 gst_element_base_class_init (gpointer g_class)
219 {
220   GstElementClass *element_class = GST_ELEMENT_CLASS (g_class);
221
222   memset (&element_class->details, 0, sizeof (GstElementDetails));
223   element_class->padtemplates = NULL;
224 }
225
226 static void
227 gst_element_base_class_finalize (gpointer g_class)
228 {
229   GstElementClass *klass = GST_ELEMENT_CLASS (g_class);
230
231   g_list_foreach (klass->padtemplates, (GFunc) gst_object_unref, NULL);
232   g_list_free (klass->padtemplates);
233   __gst_element_details_clear (&klass->details);
234 }
235
236 static void
237 gst_element_init (GstElement * element)
238 {
239   element->current_state = GST_STATE_NULL;
240   element->pending_state = GST_STATE_VOID_PENDING;
241   element->state_lock = g_mutex_new ();
242   element->state_cond = g_cond_new ();
243 }
244
245 /**
246  * gst_element_default_error:
247  * @object: a #GObject that signalled the error.
248  * @orig: the #GstObject that initiated the error.
249  * @error: the GError.
250  * @debug: an additional debug information string, or NULL.
251  *
252  * A default error signal callback to attach to an element.
253  * The user data passed to the g_signal_connect is ignored.
254  *
255  * The default handler will simply print the error string using g_print.
256  *
257  * MT safe.
258  */
259 void
260 gst_element_default_error (GObject * object, GstObject * source, GError * error,
261     gchar * debug)
262 {
263   gchar *name = gst_object_get_path_string (source);
264
265   g_print (_("ERROR: from element %s: %s\n"), name, error->message);
266   if (debug)
267     g_print (_("Additional debug info:\n%s\n"), debug);
268
269   g_free (name);
270 }
271
272 /**
273  * gst_element_release_request_pad:
274  * @element: a #GstElement to release the request pad of.
275  * @pad: the #GstPad to release.
276  *
277  * Makes the element free the previously requested pad as obtained
278  * with gst_element_get_request_pad().
279  *
280  * MT safe.
281  */
282 void
283 gst_element_release_request_pad (GstElement * element, GstPad * pad)
284 {
285   GstElementClass *oclass;
286
287   g_return_if_fail (GST_IS_ELEMENT (element));
288   g_return_if_fail (GST_IS_PAD (pad));
289
290   oclass = GST_ELEMENT_GET_CLASS (element);
291
292   if (oclass->release_pad)
293     (oclass->release_pad) (element, pad);
294   else
295     gst_element_remove_pad (element, pad);
296 }
297
298 /**
299  * gst_element_requires_clock:
300  * @element: a #GstElement to query
301  *
302  * Query if the element requires a clock.
303  *
304  * Returns: TRUE if the element requires a clock
305  *
306  * MT safe.
307  */
308 gboolean
309 gst_element_requires_clock (GstElement * element)
310 {
311   gboolean result = FALSE;
312
313   g_return_val_if_fail (GST_IS_ELEMENT (element), result);
314
315   result = (GST_ELEMENT_GET_CLASS (element)->set_clock != NULL);
316
317   return result;
318 }
319
320 /**
321  * gst_element_provides_clock:
322  * @element: a #GstElement to query
323  *
324  * Query if the element provides a clock.
325  *
326  * Returns: TRUE if the element provides a clock
327  *
328  * MT safe.
329  */
330 gboolean
331 gst_element_provides_clock (GstElement * element)
332 {
333   gboolean result = FALSE;
334
335   g_return_val_if_fail (GST_IS_ELEMENT (element), FALSE);
336
337   result = (GST_ELEMENT_GET_CLASS (element)->provide_clock != NULL);
338
339   return result;
340 }
341
342 /**
343  * gst_element_provide_clock:
344  * @element: a #GstElement to query
345  *
346  * Get the clock provided by the given element.
347  *
348  * Returns: the GstClock provided by the element or NULL
349  * if no clock could be provided.
350  *
351  * MT safe.
352  */
353 GstClock *
354 gst_element_provide_clock (GstElement * element)
355 {
356   GstClock *result = NULL;
357   GstElementClass *oclass;
358
359   g_return_val_if_fail (GST_IS_ELEMENT (element), NULL);
360
361   oclass = GST_ELEMENT_GET_CLASS (element);
362
363   if (oclass->provide_clock)
364     result = oclass->provide_clock (element);
365
366   return result;
367 }
368
369 /**
370  * gst_element_set_clock:
371  * @element: a #GstElement to set the clock for.
372  * @clock: the #GstClock to set for the element.
373  *
374  * Sets the clock for the element. This function increases the
375  * refcount on the clock. Any previously set clock on the object
376  * is unreffed.
377  *
378  * MT safe.
379  */
380 void
381 gst_element_set_clock (GstElement * element, GstClock * clock)
382 {
383   GstElementClass *oclass;
384
385   g_return_if_fail (GST_IS_ELEMENT (element));
386
387   oclass = GST_ELEMENT_GET_CLASS (element);
388
389   if (oclass->set_clock)
390     oclass->set_clock (element, clock);
391
392   GST_LOCK (element);
393   gst_object_replace ((GstObject **) & element->clock, (GstObject *) clock);
394   GST_UNLOCK (element);
395 }
396
397 /**
398  * gst_element_get_clock:
399  * @element: a #GstElement to get the clock of.
400  *
401  * Gets the currently configured clock of the element.
402  *
403  * Returns: the #GstClock of the element. unref after usage.
404  *
405  * MT safe.
406  */
407 GstClock *
408 gst_element_get_clock (GstElement * element)
409 {
410   GstClock *result;
411
412   g_return_val_if_fail (GST_IS_ELEMENT (element), NULL);
413
414   GST_LOCK (element);
415   if ((result = element->clock))
416     gst_object_ref (result);
417   GST_UNLOCK (element);
418
419   return result;
420 }
421
422 /**
423  * gst_element_set_base_time:
424  * @element: a #GstElement.
425  * @time: the base time to set.
426  *
427  * Set the base time of an element. See @gst_element_get_base_time().
428  *
429  * MT safe.
430  */
431 void
432 gst_element_set_base_time (GstElement * element, GstClockTime time)
433 {
434   g_return_if_fail (GST_IS_ELEMENT (element));
435
436   GST_LOCK (element);
437   element->base_time = time;
438   GST_UNLOCK (element);
439 }
440
441 /**
442  * gst_element_get_base_time:
443  * @element: a #GstElement.
444  *
445  * Returns the base time of the element. The base time is the
446  * absolute time of the clock when this element was last put to
447  * PLAYING. Substracting the base time from the clock time gives
448  * the stream time of the element.
449  *
450  * Returns: the base time of the element.
451  *
452  * MT safe.
453  */
454 GstClockTime
455 gst_element_get_base_time (GstElement * element)
456 {
457   GstClockTime result;
458
459   g_return_val_if_fail (GST_IS_ELEMENT (element), GST_CLOCK_TIME_NONE);
460
461   GST_LOCK (element);
462   result = element->base_time;
463   GST_UNLOCK (element);
464
465   return result;
466 }
467
468 #ifndef GST_DISABLE_INDEX
469 /**
470  * gst_element_is_indexable:
471  * @element: a #GstElement.
472  *
473  * Queries if the element can be indexed.
474  *
475  * Returns: TRUE if the element can be indexed.
476  *
477  * MT safe.
478  */
479 gboolean
480 gst_element_is_indexable (GstElement * element)
481 {
482   gboolean result = FALSE;
483
484   g_return_val_if_fail (GST_IS_ELEMENT (element), result);
485
486   result = (GST_ELEMENT_GET_CLASS (element)->set_index != NULL);
487
488   return result;
489 }
490
491 /**
492  * gst_element_set_index:
493  * @element: a #GstElement.
494  * @index: a #GstIndex.
495  *
496  * Set the specified GstIndex on the element.
497  *
498  * MT safe.
499  */
500 void
501 gst_element_set_index (GstElement * element, GstIndex * index)
502 {
503   GstElementClass *oclass;
504
505   g_return_if_fail (GST_IS_ELEMENT (element));
506   g_return_if_fail (GST_IS_INDEX (index));
507
508   oclass = GST_ELEMENT_GET_CLASS (element);
509
510   if (oclass->set_index)
511     oclass->set_index (element, index);
512 }
513
514 /**
515  * gst_element_get_index:
516  * @element: a #GstElement.
517  *
518  * Gets the index from the element.
519  *
520  * Returns: a #GstIndex or NULL when no index was set on the
521  * element. unref after usage.
522  *
523  * MT safe.
524  */
525 GstIndex *
526 gst_element_get_index (GstElement * element)
527 {
528   GstElementClass *oclass;
529
530   g_return_val_if_fail (GST_IS_ELEMENT (element), NULL);
531
532   oclass = GST_ELEMENT_GET_CLASS (element);
533
534   if (oclass->get_index)
535     return oclass->get_index (element);
536
537   return NULL;
538 }
539 #endif
540
541 /**
542  * gst_element_add_pad:
543  * @element: a #GstElement to add the pad to.
544  * @pad: the #GstPad to add to the element.
545  *
546  * Adds a pad (link point) to @element. @pad's parent will be set to @element;
547  * see gst_object_set_parent() for refcounting information.
548  *
549  * Pads are not automatically activated so elements should perform the needed
550  * steps to activate the pad. 
551  *
552  * The pad and the element should be unlocked when calling this function.
553  *
554  * Returns: TRUE if the pad could be added. This function can fail when
555  * passing bad arguments or when a pad with the same name already existed.
556  *
557  * MT safe.
558  */
559 gboolean
560 gst_element_add_pad (GstElement * element, GstPad * pad)
561 {
562   gchar *pad_name;
563
564   g_return_val_if_fail (GST_IS_ELEMENT (element), FALSE);
565   g_return_val_if_fail (GST_IS_PAD (pad), FALSE);
566
567   /* locking pad to look at the name */
568   GST_LOCK (pad);
569   pad_name = g_strdup (GST_PAD_NAME (pad));
570   GST_CAT_INFO_OBJECT (GST_CAT_ELEMENT_PADS, element, "adding pad '%s'",
571       GST_STR_NULL (pad_name));
572   GST_UNLOCK (pad);
573
574   /* then check to see if there's already a pad by that name here */
575   GST_LOCK (element);
576   if (G_UNLIKELY (!gst_object_check_uniqueness (element->pads, pad_name)))
577     goto name_exists;
578
579   /* try to set the pad's parent */
580   if (G_UNLIKELY (!gst_object_set_parent (GST_OBJECT_CAST (pad),
581               GST_OBJECT_CAST (element))))
582     goto had_parent;
583
584   g_free (pad_name);
585
586   /* add it to the list */
587   switch (gst_pad_get_direction (pad)) {
588     case GST_PAD_SRC:
589       element->srcpads = g_list_prepend (element->srcpads, pad);
590       element->numsrcpads++;
591       break;
592     case GST_PAD_SINK:
593       element->sinkpads = g_list_prepend (element->sinkpads, pad);
594       element->numsinkpads++;
595       break;
596     default:
597       goto no_direction;
598   }
599   element->pads = g_list_prepend (element->pads, pad);
600   element->numpads++;
601   element->pads_cookie++;
602   GST_UNLOCK (element);
603
604   /* emit the NEW_PAD signal */
605   g_signal_emit (G_OBJECT (element), gst_element_signals[NEW_PAD], 0, pad);
606
607   return TRUE;
608
609   /* ERROR cases */
610 name_exists:
611   {
612     g_critical ("Padname %s is not unique in element %s, not adding",
613         pad_name, GST_ELEMENT_NAME (element));
614     GST_UNLOCK (element);
615     g_free (pad_name);
616     return FALSE;
617   }
618 had_parent:
619   {
620     g_critical
621         ("Pad %s already has parent when trying to add to element %s",
622         pad_name, GST_ELEMENT_NAME (element));
623     GST_UNLOCK (element);
624     g_free (pad_name);
625     return FALSE;
626   }
627 no_direction:
628   {
629     GST_LOCK (pad);
630     g_critical
631         ("Trying to add pad %s to element %s, but it has no direction",
632         GST_OBJECT_NAME (pad), GST_ELEMENT_NAME (element));
633     GST_UNLOCK (pad);
634     GST_UNLOCK (element);
635     return FALSE;
636   }
637 }
638
639 /**
640  * gst_element_remove_pad:
641  * @element: a #GstElement to remove pad from.
642  * @pad: the #GstPad to remove from the element.
643  *
644  * Removes @pad from @element. @pad will be destroyed if it has not been
645  * referenced elsewhere.
646  *
647  * Returns: TRUE if the pad could be removed. Can return FALSE if the
648  * pad is not belonging to the provided element or when wrong parameters
649  * are passed to this function.
650  *
651  * MT safe.
652  */
653 gboolean
654 gst_element_remove_pad (GstElement * element, GstPad * pad)
655 {
656   GstPad *peer;
657
658   g_return_val_if_fail (GST_IS_ELEMENT (element), FALSE);
659   g_return_val_if_fail (GST_IS_PAD (pad), FALSE);
660
661   /* locking pad to look at the name and parent */
662   GST_LOCK (pad);
663   GST_CAT_INFO_OBJECT (GST_CAT_ELEMENT_PADS, element, "removing pad '%s'",
664       GST_STR_NULL (GST_PAD_NAME (pad)));
665
666   if (G_UNLIKELY (GST_PAD_PARENT (pad) != element))
667     goto not_our_pad;
668   GST_UNLOCK (pad);
669
670   /* unlink */
671   if ((peer = gst_pad_get_peer (pad))) {
672     /* window for MT unsafeness, someone else could unlink here
673      * and then we call unlink with wrong pads. The unlink
674      * function would catch this and safely return failed. */
675     if (GST_PAD_IS_SRC (pad))
676       gst_pad_unlink (pad, peer);
677     else
678       gst_pad_unlink (peer, pad);
679
680     gst_object_unref (peer);
681   }
682
683   GST_LOCK (element);
684   /* remove it from the list */
685   switch (gst_pad_get_direction (pad)) {
686     case GST_PAD_SRC:
687       element->srcpads = g_list_remove (element->srcpads, pad);
688       element->numsrcpads--;
689       break;
690     case GST_PAD_SINK:
691       element->sinkpads = g_list_remove (element->sinkpads, pad);
692       element->numsinkpads--;
693       break;
694     default:
695       g_critical ("Removing pad without direction???");
696       break;
697   }
698   element->pads = g_list_remove (element->pads, pad);
699   element->numpads--;
700   element->pads_cookie++;
701   GST_UNLOCK (element);
702
703   g_signal_emit (G_OBJECT (element), gst_element_signals[PAD_REMOVED], 0, pad);
704
705   gst_object_unparent (GST_OBJECT (pad));
706
707   return TRUE;
708
709 not_our_pad:
710   {
711     /* FIXME, locking order? */
712     GST_LOCK (element);
713     g_critical ("Padname %s:%s does not belong to element %s when removing",
714         GST_ELEMENT_NAME (GST_PAD_PARENT (pad)), GST_PAD_NAME (pad),
715         GST_ELEMENT_NAME (element));
716     GST_UNLOCK (element);
717     GST_UNLOCK (pad);
718     return FALSE;
719   }
720 }
721
722 /**
723  * gst_element_no_more_pads:
724  * @element: a #GstElement
725  *
726  * Use this function to signal that the element does not expect any more pads
727  * to show up in the current pipeline. This function should be called whenever
728  * pads have been added by the element itself. Elements with GST_PAD_SOMETIMES
729  * pad templates use this in combination with autopluggers to figure out that
730  * the element is done initializing its pads.
731  *
732  * MT safe.
733  */
734 void
735 gst_element_no_more_pads (GstElement * element)
736 {
737   g_return_if_fail (GST_IS_ELEMENT (element));
738
739   g_signal_emit (element, gst_element_signals[NO_MORE_PADS], 0);
740 }
741
742 static gint
743 pad_compare_name (GstPad * pad1, const gchar * name)
744 {
745   gint result;
746
747   GST_LOCK (pad1);
748   result = strcmp (GST_PAD_NAME (pad1), name);
749   GST_UNLOCK (pad1);
750
751   return result;
752 }
753
754 /**
755  * gst_element_get_static_pad:
756  * @element: a #GstElement to find a static pad of.
757  * @name: the name of the static #GstPad to retrieve.
758  *
759  * Retrieves a pad from @element by name. This version only retrieves
760  * already-existing (i.e. 'static') pads.
761  *
762  * Returns: the requested #GstPad if found, otherwise NULL. unref after
763  * usage.
764  *
765  * MT safe.
766  */
767 GstPad *
768 gst_element_get_static_pad (GstElement * element, const gchar * name)
769 {
770   GList *find;
771   GstPad *result = NULL;
772
773   g_return_val_if_fail (GST_IS_ELEMENT (element), NULL);
774   g_return_val_if_fail (name != NULL, NULL);
775
776   GST_LOCK (element);
777   find =
778       g_list_find_custom (element->pads, name, (GCompareFunc) pad_compare_name);
779   if (find) {
780     result = GST_PAD_CAST (find->data);
781     gst_object_ref (result);
782   }
783
784   if (result == NULL) {
785     GST_CAT_INFO (GST_CAT_ELEMENT_PADS, "no such pad '%s' in element \"%s\"",
786         name, GST_ELEMENT_NAME (element));
787   } else {
788     GST_CAT_INFO (GST_CAT_ELEMENT_PADS, "found pad %s:%s",
789         GST_ELEMENT_NAME (element), name);
790   }
791   GST_UNLOCK (element);
792
793   return result;
794 }
795
796 static GstPad *
797 gst_element_request_pad (GstElement * element, GstPadTemplate * templ,
798     const gchar * name)
799 {
800   GstPad *newpad = NULL;
801   GstElementClass *oclass;
802
803   oclass = GST_ELEMENT_GET_CLASS (element);
804
805   if (oclass->request_new_pad)
806     newpad = (oclass->request_new_pad) (element, templ, name);
807
808   if (newpad)
809     gst_object_ref (newpad);
810
811   return newpad;
812 }
813
814 /**
815  * gst_element_get_request_pad:
816  * @element: a #GstElement to find a request pad of.
817  * @name: the name of the request #GstPad to retrieve.
818  *
819  * Retrieves a pad from the element by name. This version only retrieves
820  * request pads.
821  *
822  * Returns: requested #GstPad if found, otherwise NULL. Unref after usage.
823  */
824 GstPad *
825 gst_element_get_request_pad (GstElement * element, const gchar * name)
826 {
827   GstPadTemplate *templ = NULL;
828   GstPad *pad;
829   const gchar *req_name = NULL;
830   gboolean templ_found = FALSE;
831   GList *list;
832   gint n;
833   const gchar *data;
834   gchar *str, *endptr = NULL;
835   GstElementClass *class;
836
837   g_return_val_if_fail (GST_IS_ELEMENT (element), NULL);
838   g_return_val_if_fail (name != NULL, NULL);
839
840   class = GST_ELEMENT_GET_CLASS (element);
841
842   if (strstr (name, "%")) {
843     templ = gst_element_class_get_pad_template (class, name);
844     req_name = NULL;
845     if (templ)
846       templ_found = TRUE;
847   } else {
848     list = gst_element_class_get_pad_template_list (class);
849     while (!templ_found && list) {
850       templ = (GstPadTemplate *) list->data;
851       if (templ->presence == GST_PAD_REQUEST) {
852         /* Because of sanity checks in gst_pad_template_new(), we know that %s
853            and %d, occurring at the end of the name_template, are the only
854            possibilities. */
855         GST_CAT_DEBUG (GST_CAT_PADS, "comparing %s to %s", name,
856             templ->name_template);
857         if ((str = strchr (templ->name_template, '%'))
858             && strncmp (templ->name_template, name,
859                 str - templ->name_template) == 0
860             && strlen (name) > str - templ->name_template) {
861           data = name + (str - templ->name_template);
862           if (*(str + 1) == 'd') {
863             /* it's an int */
864             n = (gint) strtol (data, &endptr, 10);
865             if (endptr && *endptr == '\0') {
866               templ_found = TRUE;
867               req_name = name;
868               break;
869             }
870           } else {
871             /* it's a string */
872             templ_found = TRUE;
873             req_name = name;
874             break;
875           }
876         }
877       }
878       list = list->next;
879     }
880   }
881
882   if (!templ_found)
883     return NULL;
884
885   pad = gst_element_request_pad (element, templ, req_name);
886
887   return pad;
888 }
889
890 /**
891  * gst_element_get_pad:
892  * @element: a #GstElement.
893  * @name: the name of the pad to retrieve.
894  *
895  * Retrieves a pad from @element by name. Tries gst_element_get_static_pad()
896  * first, then gst_element_get_request_pad().
897  *
898  * Returns: the #GstPad if found, otherwise %NULL. Unref after usage.
899  */
900 GstPad *
901 gst_element_get_pad (GstElement * element, const gchar * name)
902 {
903   GstPad *pad;
904
905   g_return_val_if_fail (GST_IS_ELEMENT (element), NULL);
906   g_return_val_if_fail (name != NULL, NULL);
907
908   pad = gst_element_get_static_pad (element, name);
909   if (!pad)
910     pad = gst_element_get_request_pad (element, name);
911
912   return pad;
913 }
914
915 GstIteratorItem
916 iterate_pad (GstIterator * it, GstPad * pad)
917 {
918   gst_object_ref (pad);
919   return GST_ITERATOR_ITEM_PASS;
920 }
921
922 /**
923  * gst_element_iterate_pads:
924  * @element: a #GstElement to iterate pads of.
925  *
926  * Retrieves an iterattor of @element's pads. 
927  *
928  * Returns: the #GstIterator of #GstPad. Unref each pad after use.
929  *
930  * MT safe.
931  */
932 GstIterator *
933 gst_element_iterate_pads (GstElement * element)
934 {
935   GstIterator *result;
936
937   g_return_val_if_fail (GST_IS_ELEMENT (element), NULL);
938
939   GST_LOCK (element);
940   gst_object_ref (element);
941   result = gst_iterator_new_list (GST_GET_LOCK (element),
942       &element->pads_cookie,
943       &element->pads,
944       element,
945       (GstIteratorItemFunction) iterate_pad,
946       (GstIteratorDisposeFunction) gst_object_unref);
947   GST_UNLOCK (element);
948
949   return result;
950 }
951
952 static gint
953 direction_filter (gconstpointer pad, gconstpointer direction)
954 {
955   if (GST_PAD_DIRECTION (pad) == GPOINTER_TO_INT (direction)) {
956     /* pass the ref through */
957     return 0;
958   } else {
959     /* unref */
960     /* FIXME: this is very stupid */
961     gst_object_unref (GST_OBJECT_CAST (pad));
962     return 1;
963   }
964 }
965
966 /**
967  * gst_element_iterate_src_pads:
968  * @element: a #GstElement.
969  *
970  * Retrieves an iterator of @element's source pads. 
971  *
972  * Returns: the #GstIterator of #GstPad. Unref each pad after use.
973  *
974  * MT safe.
975  */
976 GstIterator *
977 gst_element_iterate_src_pads (GstElement * element)
978 {
979   g_return_val_if_fail (GST_IS_ELEMENT (element), NULL);
980
981   return gst_iterator_filter (gst_element_iterate_pads (element),
982       direction_filter, GINT_TO_POINTER (GST_PAD_SRC));
983 }
984
985 /**
986  * gst_element_iterate_sink_pads:
987  * @element: a #GstElement.
988  *
989  * Retrieves an iterator of @element's sink pads. 
990  *
991  * Returns: the #GstIterator of #GstPad. Unref each pad after use.
992  *
993  * MT safe.
994  */
995 GstIterator *
996 gst_element_iterate_sink_pads (GstElement * element)
997 {
998   g_return_val_if_fail (GST_IS_ELEMENT (element), NULL);
999
1000   return gst_iterator_filter (gst_element_iterate_pads (element),
1001       direction_filter, GINT_TO_POINTER (GST_PAD_SINK));
1002 }
1003
1004 /**
1005  * gst_element_class_add_pad_template:
1006  * @klass: the #GstElementClass to add the pad template to.
1007  * @templ: a #GstPadTemplate to add to the element class.
1008  *
1009  * Adds a padtemplate to an element class. This is mainly used in the _base_init
1010  * functions of classes.
1011  */
1012 void
1013 gst_element_class_add_pad_template (GstElementClass * klass,
1014     GstPadTemplate * templ)
1015 {
1016   g_return_if_fail (GST_IS_ELEMENT_CLASS (klass));
1017   g_return_if_fail (GST_IS_PAD_TEMPLATE (templ));
1018
1019   /* avoid registering pad templates with the same name */
1020   g_return_if_fail (gst_element_class_get_pad_template (klass,
1021           templ->name_template) == NULL);
1022
1023   klass->padtemplates = g_list_append (klass->padtemplates,
1024       gst_object_ref (templ));
1025   klass->numpadtemplates++;
1026 }
1027
1028 /**
1029  * gst_element_class_set_details:
1030  * @klass: class to set details for
1031  * @details: details to set
1032  *
1033  * Sets the detailed information for a #GstElementClass.
1034  * <note>This function is for use in _base_init functions only.</note>
1035  */
1036 void
1037 gst_element_class_set_details (GstElementClass * klass,
1038     const GstElementDetails * details)
1039 {
1040   g_return_if_fail (GST_IS_ELEMENT_CLASS (klass));
1041   g_return_if_fail (GST_IS_ELEMENT_DETAILS (details));
1042
1043   __gst_element_details_copy (&klass->details, details);
1044 }
1045
1046 /**
1047  * gst_element_class_get_pad_template_list:
1048  * @element_class: a #GstElementClass to get pad templates of.
1049  *
1050  * Retrieves a list of the pad templates associated with @element_class. The
1051  * list must not be modified by the calling code.
1052  * <note>If you use this function in the #GInstanceInitFunc of an object class
1053  * that has subclasses, make sure to pass the g_class parameter of the
1054  * #GInstanceInitFunc here.</note>
1055  *
1056  * Returns: the #GList of padtemplates.
1057  */
1058 GList *
1059 gst_element_class_get_pad_template_list (GstElementClass * element_class)
1060 {
1061   g_return_val_if_fail (GST_IS_ELEMENT_CLASS (element_class), NULL);
1062
1063   return element_class->padtemplates;
1064 }
1065
1066 /**
1067  * gst_element_class_get_pad_template:
1068  * @element_class: a #GstElementClass to get the pad template of.
1069  * @name: the name of the #GstPadTemplate to get.
1070  *
1071  * Retrieves a padtemplate from @element_class with the given name.
1072  * <note>If you use this function in the #GInstanceInitFunc of an object class
1073  * that has subclasses, make sure to pass the g_class parameter of the
1074  * #GInstanceInitFunc here.</note>
1075  *
1076  * Returns: the #GstPadTemplate with the given name, or NULL if none was found.
1077  * No unreferencing is necessary.
1078  */
1079 GstPadTemplate *
1080 gst_element_class_get_pad_template (GstElementClass * element_class,
1081     const gchar * name)
1082 {
1083   GList *padlist;
1084
1085   g_return_val_if_fail (GST_IS_ELEMENT_CLASS (element_class), NULL);
1086   g_return_val_if_fail (name != NULL, NULL);
1087
1088   padlist = gst_element_class_get_pad_template_list (element_class);
1089
1090   while (padlist) {
1091     GstPadTemplate *padtempl = (GstPadTemplate *) padlist->data;
1092
1093     if (strcmp (padtempl->name_template, name) == 0)
1094       return padtempl;
1095
1096     padlist = g_list_next (padlist);
1097   }
1098
1099   return NULL;
1100 }
1101
1102 static GstPad *
1103 gst_element_get_random_pad (GstElement * element, GstPadDirection dir)
1104 {
1105   GstPad *result = NULL;
1106   GList *pads;
1107
1108   GST_CAT_DEBUG (GST_CAT_ELEMENT_PADS, "getting a random pad");
1109
1110   switch (dir) {
1111     case GST_PAD_SRC:
1112       GST_LOCK (element);
1113       pads = element->srcpads;
1114       break;
1115     case GST_PAD_SINK:
1116       GST_LOCK (element);
1117       pads = element->sinkpads;
1118       break;
1119     default:
1120       goto wrong_direction;
1121   }
1122   for (; pads; pads = g_list_next (pads)) {
1123     GstPad *pad = GST_PAD (pads->data);
1124
1125     GST_LOCK (pad);
1126     GST_CAT_DEBUG (GST_CAT_ELEMENT_PADS, "checking pad %s:%s",
1127         GST_DEBUG_PAD_NAME (pad));
1128
1129     if (GST_PAD_IS_LINKED (pad)) {
1130       GST_UNLOCK (pad);
1131       result = pad;
1132       break;
1133     } else {
1134       GST_CAT_DEBUG (GST_CAT_ELEMENT_PADS, "pad %s:%s is not linked",
1135           GST_DEBUG_PAD_NAME (pad));
1136     }
1137     GST_UNLOCK (pad);
1138   }
1139   if (result)
1140     gst_object_ref (result);
1141
1142   GST_UNLOCK (element);
1143
1144   return result;
1145
1146   /* ERROR handling */
1147 wrong_direction:
1148   {
1149     g_warning ("unknown pad direction %d", dir);
1150     return NULL;
1151   }
1152 }
1153
1154 /**
1155  * gst_element_send_event:
1156  * @element: a #GstElement to send the event to.
1157  * @event: the #GstEvent to send to the element.
1158  *
1159  * Sends an event to an element. If the element doesn't
1160  * implement an event handler, the event will be forwarded
1161  * to a random sink pad. This function takes owership of the
1162  * provided event so you should _ref it if you want to reuse
1163  * the event after this call.
1164  *
1165  * Returns: TRUE if the event was handled.
1166  *
1167  * MT safe.
1168  */
1169 gboolean
1170 gst_element_send_event (GstElement * element, GstEvent * event)
1171 {
1172   GstElementClass *oclass;
1173   gboolean result = FALSE;
1174
1175   g_return_val_if_fail (GST_IS_ELEMENT (element), FALSE);
1176   g_return_val_if_fail (event != NULL, FALSE);
1177
1178   oclass = GST_ELEMENT_GET_CLASS (element);
1179
1180   if (oclass->send_event) {
1181     result = oclass->send_event (element, event);
1182   } else {
1183     GstPad *pad = gst_element_get_random_pad (element, GST_PAD_SINK);
1184
1185     if (pad) {
1186       GST_CAT_DEBUG (GST_CAT_ELEMENT_PADS,
1187           "pushing event to random pad %s:%s", GST_DEBUG_PAD_NAME (pad));
1188
1189       result = gst_pad_push_event (pad, event);
1190       gst_object_unref (pad);
1191     } else {
1192       GST_CAT_DEBUG (GST_CAT_ELEMENT_PADS, "can't send event on element %s",
1193           GST_ELEMENT_NAME (element));
1194     }
1195   }
1196   return result;
1197 }
1198
1199 /**
1200  * gst_element_seek:
1201  * @element: a #GstElement to send the event to.
1202  * @rate: The new playback rate
1203  * @format: The format of the seek values
1204  * @flags: The optional seek flags.
1205  * @cur_type: The type and flags for the new current position
1206  * @cur: The value of the new current position
1207  * @stop_type: The type and flags for the new stop position
1208  * @stop: The value of the new stop position
1209  *
1210  * Sends a seek event to an element.
1211  *
1212  * Returns: %TRUE if the event was handled.
1213  *
1214  * MT safe.
1215  */
1216 gboolean
1217 gst_element_seek (GstElement * element, gdouble rate, GstFormat format,
1218     GstSeekFlags flags, GstSeekType cur_type, gint64 cur,
1219     GstSeekType stop_type, gint64 stop)
1220 {
1221   GstEvent *event;
1222   gboolean result;
1223
1224   g_return_val_if_fail (GST_IS_ELEMENT (element), FALSE);
1225
1226   event =
1227       gst_event_new_seek (rate, format, flags, cur_type, cur, stop_type, stop);
1228   result = gst_element_send_event (element, event);
1229
1230   return result;
1231 }
1232
1233 /**
1234  * gst_element_get_query_types:
1235  * @element: a #GstElement to query
1236  *
1237  * Get an array of query types from the element.
1238  * If the element doesn't implement a query types function,
1239  * the query will be forwarded to a random sink pad.
1240  *
1241  * Returns: An array of #GstQueryType elements that should not
1242  * be freed or modified.
1243  *
1244  * MT safe.
1245  */
1246 const GstQueryType *
1247 gst_element_get_query_types (GstElement * element)
1248 {
1249   GstElementClass *oclass;
1250   const GstQueryType *result = NULL;
1251
1252   g_return_val_if_fail (GST_IS_ELEMENT (element), NULL);
1253
1254   oclass = GST_ELEMENT_GET_CLASS (element);
1255
1256   if (oclass->get_query_types) {
1257     result = oclass->get_query_types (element);
1258   } else {
1259     GstPad *pad = gst_element_get_random_pad (element, GST_PAD_SINK);
1260
1261     if (pad) {
1262       GstPad *peer = gst_pad_get_peer (pad);
1263
1264       if (peer) {
1265         result = gst_pad_get_query_types (peer);
1266
1267         gst_object_unref (peer);
1268       }
1269       gst_object_unref (pad);
1270     }
1271   }
1272   return result;
1273 }
1274
1275 /**
1276  * gst_element_query:
1277  * @element: a #GstElement to perform the query on.
1278  * @query: the #GstQuery.
1279  *
1280  * Performs a query on the given element. If the format is set
1281  * to GST_FORMAT_DEFAULT and this function returns TRUE, the
1282  * format pointer will hold the default format.
1283  * For element that don't implement a query handler, this function
1284  * forwards the query to a random usable sinkpad of this element.
1285  *
1286  * Returns: TRUE if the query could be performed.
1287  *
1288  * MT safe.
1289  */
1290 gboolean
1291 gst_element_query (GstElement * element, GstQuery * query)
1292 {
1293   GstElementClass *oclass;
1294   gboolean result = FALSE;
1295
1296   g_return_val_if_fail (GST_IS_ELEMENT (element), FALSE);
1297   g_return_val_if_fail (query != NULL, FALSE);
1298
1299   oclass = GST_ELEMENT_GET_CLASS (element);
1300
1301   if (oclass->query) {
1302     result = oclass->query (element, query);
1303   } else {
1304     GstPad *pad = gst_element_get_random_pad (element, GST_PAD_SRC);
1305
1306     if (pad) {
1307       result = gst_pad_query (pad, query);
1308
1309       gst_object_unref (pad);
1310     } else {
1311       pad = gst_element_get_random_pad (element, GST_PAD_SINK);
1312       if (pad) {
1313         GstPad *peer = gst_pad_get_peer (pad);
1314
1315         if (peer) {
1316           result = gst_pad_query (peer, query);
1317
1318           gst_object_unref (peer);
1319         }
1320         gst_object_unref (pad);
1321       }
1322     }
1323   }
1324   return result;
1325 }
1326
1327 /**
1328  * gst_element_post_message:
1329  * @element: a #GstElement posting the message
1330  * @message: a #GstMessage to post
1331  *
1332  * Post a message on the element's #GstBus.
1333  *
1334  * Returns: TRUE if the message was successfully posted.
1335  *
1336  * MT safe.
1337  */
1338 gboolean
1339 gst_element_post_message (GstElement * element, GstMessage * message)
1340 {
1341   GstBus *bus;
1342   gboolean result = FALSE;
1343
1344   GST_DEBUG ("posting message %p ...", message);
1345
1346   g_return_val_if_fail (GST_IS_ELEMENT (element), FALSE);
1347   g_return_val_if_fail (message != NULL, FALSE);
1348
1349   GST_LOCK (element);
1350   bus = element->bus;
1351
1352   if (G_UNLIKELY (bus == NULL)) {
1353     GST_DEBUG ("... but I won't because I have no bus");
1354     GST_UNLOCK (element);
1355     gst_message_unref (message);
1356     return FALSE;
1357   }
1358   gst_object_ref (bus);
1359   GST_DEBUG ("... on bus %" GST_PTR_FORMAT, bus);
1360   GST_UNLOCK (element);
1361
1362   result = gst_bus_post (bus, message);
1363   gst_object_unref (bus);
1364
1365   return result;
1366 }
1367
1368 /**
1369  * _gst_element_error_printf:
1370  * @format: the printf-like format to use, or NULL
1371  *
1372  * This function is only used internally by the #gst_element_error macro.
1373  *
1374  * Returns: a newly allocated string, or NULL if the format was NULL or ""
1375  *
1376  * MT safe.
1377  */
1378 gchar *
1379 _gst_element_error_printf (const gchar * format, ...)
1380 {
1381   va_list args;
1382   gchar *buffer;
1383
1384   if (format == NULL)
1385     return NULL;
1386   if (format[0] == 0)
1387     return NULL;
1388
1389   va_start (args, format);
1390   buffer = g_strdup_vprintf (format, args);
1391   va_end (args);
1392   return buffer;
1393 }
1394
1395 /**
1396  * gst_element_message_full:
1397  * @element:  a #GstElement to send message from
1398  * @type:     the #GstMessageType
1399  * @domain:   the GStreamer GError domain this message belongs to
1400  * @code:     the GError code belonging to the domain
1401  * @text:     an allocated text string to be used as a replacement for the
1402  *            default message connected to code, or NULL
1403  * @debug:    an allocated debug message to be used as a replacement for the
1404  *            default debugging information, or NULL
1405  * @file:     the source code file where the error was generated
1406  * @function: the source code function where the error was generated
1407  * @line:     the source code line where the error was generated
1408  *
1409  * Post an error or warning message on the bus from inside an element.
1410  *
1411  * MT safe.
1412  */
1413 void gst_element_message_full
1414     (GstElement * element, GstMessageType type,
1415     GQuark domain, gint code, gchar * text,
1416     gchar * debug, const gchar * file, const gchar * function, gint line)
1417 {
1418   GError *gerror = NULL;
1419   gchar *name;
1420   gchar *sent_text;
1421   gchar *sent_debug;
1422   GstMessage *message = NULL;
1423
1424   /* checks */
1425   GST_DEBUG ("start");
1426   g_return_if_fail (GST_IS_ELEMENT (element));
1427   g_return_if_fail ((type == GST_MESSAGE_ERROR) ||
1428       (type == GST_MESSAGE_WARNING));
1429
1430   /* check if we send the given text or the default error text */
1431   if ((text == NULL) || (text[0] == 0)) {
1432     /* text could have come from g_strdup_printf (""); */
1433     g_free (text);
1434     sent_text = gst_error_get_message (domain, code);
1435   } else
1436     sent_text = text;
1437
1438   /* construct a sent_debug with extra information from source */
1439   if ((debug == NULL) || (debug[0] == 0)) {
1440     /* debug could have come from g_strdup_printf (""); */
1441     sent_debug = NULL;
1442   } else {
1443     name = gst_object_get_path_string (GST_OBJECT (element));
1444     sent_debug = g_strdup_printf ("%s(%d): %s: %s:\n%s",
1445         file, line, function, name, debug ? debug : "");
1446     g_free (name);
1447   }
1448   g_free (debug);
1449
1450   /* create gerror and post message */
1451   GST_CAT_INFO_OBJECT (GST_CAT_ERROR_SYSTEM, element, "posting message: %s",
1452       sent_text);
1453   gerror = g_error_new_literal (domain, code, sent_text);
1454
1455   if (type == GST_MESSAGE_ERROR) {
1456     message = gst_message_new_error (GST_OBJECT (element), gerror, sent_debug);
1457   } else if (type == GST_MESSAGE_WARNING) {
1458     message = gst_message_new_warning (GST_OBJECT (element), gerror,
1459         sent_debug);
1460   } else {
1461     g_assert_not_reached ();
1462   }
1463   gst_element_post_message (element, message);
1464
1465   GST_CAT_INFO_OBJECT (GST_CAT_ERROR_SYSTEM, element, "posted message: %s",
1466       sent_text);
1467
1468   /* cleanup */
1469   g_error_free (gerror);
1470   g_free (sent_debug);
1471   g_free (sent_text);
1472 }
1473
1474 /**
1475  * gst_element_is_locked_state:
1476  * @element: a #GstElement.
1477  *
1478  * Checks if the state of an element is locked.
1479  * If the state of an element is locked, state changes of the parent don't
1480  * affect the element.
1481  * This way you can leave currently unused elements inside bins. Just lock their
1482  * state before changing the state from #GST_STATE_NULL.
1483  *
1484  * Returns: TRUE, if the element's state is locked.
1485  *
1486  * MT safe.
1487  */
1488 gboolean
1489 gst_element_is_locked_state (GstElement * element)
1490 {
1491   gboolean result = FALSE;
1492
1493   g_return_val_if_fail (GST_IS_ELEMENT (element), FALSE);
1494
1495   GST_LOCK (element);
1496   result = GST_FLAG_IS_SET (element, GST_ELEMENT_LOCKED_STATE);
1497   GST_UNLOCK (element);
1498
1499   return result;
1500 }
1501
1502 /**
1503  * gst_element_set_locked_state:
1504  * @element: a #GstElement
1505  * @locked_state: TRUE to lock the element's state
1506  *
1507  * Locks the state of an element, so state changes of the parent don't affect
1508  * this element anymore.
1509  *
1510  * Returns: TRUE if the state was changed, FALSE if bad params were given or
1511  * the element was already in the correct state.
1512  *
1513  * MT safe.
1514  */
1515 gboolean
1516 gst_element_set_locked_state (GstElement * element, gboolean locked_state)
1517 {
1518   gboolean old;
1519
1520   g_return_val_if_fail (GST_IS_ELEMENT (element), FALSE);
1521
1522   GST_LOCK (element);
1523   old = GST_FLAG_IS_SET (element, GST_ELEMENT_LOCKED_STATE);
1524
1525   if (G_UNLIKELY (old == locked_state))
1526     goto was_ok;
1527
1528   if (locked_state) {
1529     GST_CAT_DEBUG (GST_CAT_STATES, "locking state of element %s",
1530         GST_ELEMENT_NAME (element));
1531     GST_FLAG_SET (element, GST_ELEMENT_LOCKED_STATE);
1532   } else {
1533     GST_CAT_DEBUG (GST_CAT_STATES, "unlocking state of element %s",
1534         GST_ELEMENT_NAME (element));
1535     GST_FLAG_UNSET (element, GST_ELEMENT_LOCKED_STATE);
1536   }
1537   GST_UNLOCK (element);
1538
1539   return TRUE;
1540
1541 was_ok:
1542   GST_UNLOCK (element);
1543
1544   return FALSE;
1545 }
1546
1547 /**
1548  * gst_element_sync_state_with_parent:
1549  * @element: a #GstElement.
1550  *
1551  * Tries to change the state of the element to the same as its parent.
1552  * If this function returns FALSE, the state of element is undefined.
1553  *
1554  * Returns: TRUE, if the element's state could be synced to the parent's state.
1555  */
1556 gboolean
1557 gst_element_sync_state_with_parent (GstElement * element)
1558 {
1559   GstElement *parent;
1560
1561   g_return_val_if_fail (GST_IS_ELEMENT (element), FALSE);
1562   parent = GST_ELEMENT (GST_ELEMENT_PARENT (element));
1563   g_return_val_if_fail (GST_IS_BIN (parent), FALSE);
1564
1565   GST_CAT_DEBUG (GST_CAT_STATES, "syncing state of element %s (%s) to %s (%s)",
1566       GST_ELEMENT_NAME (element),
1567       gst_element_state_get_name (GST_STATE (element)),
1568       GST_ELEMENT_NAME (parent),
1569       gst_element_state_get_name (GST_STATE (parent)));
1570
1571   if (gst_element_set_state (element,
1572           GST_STATE (parent)) == GST_STATE_CHANGE_FAILURE) {
1573     return FALSE;
1574   }
1575   return TRUE;
1576 }
1577
1578 /* MT safe */
1579 static GstStateChangeReturn
1580 gst_element_get_state_func (GstElement * element,
1581     GstState * state, GstState * pending, GTimeVal * timeout)
1582 {
1583   GstStateChangeReturn ret = GST_STATE_CHANGE_FAILURE;
1584   GstState old_pending;
1585
1586   GST_CAT_INFO_OBJECT (GST_CAT_STATES, element, "getting state");
1587
1588   GST_STATE_LOCK (element);
1589   /* we got an error, report immediatly */
1590   if (GST_STATE_NO_PREROLL (element)) {
1591     ret = GST_STATE_CHANGE_NO_PREROLL;
1592     goto done;
1593   }
1594
1595   /* we got an error, report immediatly */
1596   if (GST_STATE_ERROR (element)) {
1597     ret = GST_STATE_CHANGE_FAILURE;
1598     goto done;
1599   }
1600
1601   old_pending = GST_STATE_PENDING (element);
1602   if (old_pending != GST_STATE_VOID_PENDING) {
1603     GTimeVal *timeval, abstimeout;
1604
1605     GST_CAT_INFO_OBJECT (GST_CAT_STATES, element, "wait for pending");
1606     if (timeout) {
1607       /* make timeout absolute */
1608       g_get_current_time (&abstimeout);
1609       g_time_val_add (&abstimeout,
1610           timeout->tv_sec * G_USEC_PER_SEC + timeout->tv_usec);
1611       timeval = &abstimeout;
1612     } else {
1613       timeval = NULL;
1614     }
1615     /* we have a pending state change, wait for it to complete */
1616     if (!GST_STATE_TIMED_WAIT (element, timeval)) {
1617       GST_CAT_INFO_OBJECT (GST_CAT_STATES, element, "timeout");
1618       /* timeout triggered */
1619       ret = GST_STATE_CHANGE_ASYNC;
1620     } else {
1621       /* could be success or failure */
1622       if (old_pending == GST_STATE (element)) {
1623         GST_CAT_INFO_OBJECT (GST_CAT_STATES, element, "got success");
1624         ret = GST_STATE_CHANGE_SUCCESS;
1625       } else {
1626         GST_CAT_INFO_OBJECT (GST_CAT_STATES, element, "got failure");
1627         ret = GST_STATE_CHANGE_FAILURE;
1628       }
1629     }
1630   }
1631   /* if nothing is pending anymore we can return SUCCESS */
1632   if (GST_STATE_PENDING (element) == GST_STATE_VOID_PENDING) {
1633     GST_CAT_INFO_OBJECT (GST_CAT_STATES, element, "nothing pending");
1634     ret = GST_STATE_CHANGE_SUCCESS;
1635   }
1636
1637 done:
1638   if (state)
1639     *state = GST_STATE (element);
1640   if (pending)
1641     *pending = GST_STATE_PENDING (element);
1642
1643   GST_CAT_INFO_OBJECT (GST_CAT_STATES, element,
1644       "state current: %s, pending: %s, error: %d, no_preroll: %d, result: %d",
1645       gst_element_state_get_name (GST_STATE (element)),
1646       gst_element_state_get_name (GST_STATE_PENDING (element)),
1647       GST_STATE_ERROR (element), GST_STATE_NO_PREROLL (element), ret);
1648
1649   GST_STATE_UNLOCK (element);
1650
1651   return ret;
1652 }
1653
1654 /**
1655  * gst_element_get_state:
1656  * @element: a #GstElement to get the state of.
1657  * @state: a pointer to #GstState to hold the state. Can be NULL.
1658  * @pending: a pointer to #GstState to hold the pending state.
1659  *           Can be NULL.
1660  * @timeout: a #GTimeVal to specify the timeout for an async
1661  *           state change or NULL for infinite timeout.
1662  *
1663  * Gets the state of the element. 
1664  *
1665  * For elements that performed an ASYNC state change, as reported by 
1666  * #gst_element_set_state(), this function will block up to the 
1667  * specified timeout value for the state change to complete. 
1668  * If the element completes the state change or goes into
1669  * an error, this function returns immediatly with a return value of
1670  * GST_STATE_CHANGE_SUCCESS or GST_STATE_CHANGE_FAILURE respectively. 
1671  *
1672  * Returns: GST_STATE_CHANGE_SUCCESS if the element has no more pending state and
1673  *          the last state change succeeded, GST_STATE_CHANGE_ASYNC
1674  *          if the element is still performing a state change or 
1675  *          GST_STATE_CHANGE_FAILURE if the last state change failed.
1676  *
1677  * MT safe.
1678  */
1679 GstStateChangeReturn
1680 gst_element_get_state (GstElement * element,
1681     GstState * state, GstState * pending, GTimeVal * timeout)
1682 {
1683   GstElementClass *oclass;
1684   GstStateChangeReturn result = GST_STATE_CHANGE_FAILURE;
1685
1686   g_return_val_if_fail (GST_IS_ELEMENT (element), GST_STATE_CHANGE_FAILURE);
1687
1688   oclass = GST_ELEMENT_GET_CLASS (element);
1689
1690   if (oclass->get_state)
1691     result = (oclass->get_state) (element, state, pending, timeout);
1692
1693   return result;
1694 }
1695
1696 /**
1697  * gst_element_abort_state:
1698  * @element: a #GstElement to abort the state of.
1699  *
1700  * Abort the state change of the element. This function is used
1701  * by elements that do asynchronous state changes and find out 
1702  * something is wrong.
1703  *
1704  * This function should be called with the STATE_LOCK held.
1705  *
1706  * MT safe.
1707  */
1708 void
1709 gst_element_abort_state (GstElement * element)
1710 {
1711   GstState pending;
1712
1713   g_return_if_fail (GST_IS_ELEMENT (element));
1714
1715   pending = GST_STATE_PENDING (element);
1716
1717   if (pending != GST_STATE_VOID_PENDING && !GST_STATE_ERROR (element)) {
1718 #ifndef GST_DISABLE_GST_DEBUG
1719     GstState old_state = GST_STATE (element);
1720 #endif
1721
1722     GST_CAT_INFO_OBJECT (GST_CAT_STATES, element,
1723         "aborting state from %s to %s", gst_element_state_get_name (old_state),
1724         gst_element_state_get_name (pending));
1725
1726     /* flag error */
1727     GST_STATE_ERROR (element) = TRUE;
1728
1729     GST_STATE_BROADCAST (element);
1730   }
1731 }
1732
1733 /**
1734  * gst_element_commit_state:
1735  * @element: a #GstElement to commit the state of.
1736  *
1737  * Commit the state change of the element. This function is used
1738  * by elements that do asynchronous state changes.
1739  *
1740  * This function can only be called with the STATE_LOCK held.
1741  *
1742  * MT safe.
1743  */
1744 void
1745 gst_element_commit_state (GstElement * element)
1746 {
1747   GstState pending;
1748
1749   g_return_if_fail (GST_IS_ELEMENT (element));
1750
1751   pending = GST_STATE_PENDING (element);
1752
1753   if (pending != GST_STATE_VOID_PENDING) {
1754     GstState old_state = GST_STATE (element);
1755     GstMessage *message;
1756
1757     GST_CAT_INFO_OBJECT (GST_CAT_STATES, element,
1758         "committing state from %s to %s",
1759         gst_element_state_get_name (old_state),
1760         gst_element_state_get_name (pending));
1761
1762     GST_STATE (element) = pending;
1763     GST_STATE_PENDING (element) = GST_STATE_VOID_PENDING;
1764     GST_STATE_ERROR (element) = FALSE;
1765
1766     message = gst_message_new_state_changed (GST_OBJECT (element),
1767         old_state, pending);
1768     gst_element_post_message (element, message);
1769     GST_STATE_BROADCAST (element);
1770   }
1771 }
1772
1773 /**
1774  * gst_element_lost_state:
1775  * @element: a #GstElement the state is lost of
1776  *
1777  * Brings the element to the lost state. The current state of the
1778  * element is copied to the pending state so that any call to
1779  * #gst_element_get_state() will return ASYNC.
1780  * This is mostly used for elements that lost their preroll buffer
1781  * in the PAUSED state after a flush, they become PAUSED again
1782  * if a new preroll buffer is queued.
1783  * This function can only be called when the element is currently
1784  * not in error or an async state change.
1785  *
1786  * This function can only be called with the STATE_LOCK held.
1787  *
1788  * MT safe.
1789  */
1790 void
1791 gst_element_lost_state (GstElement * element)
1792 {
1793   g_return_if_fail (GST_IS_ELEMENT (element));
1794
1795   if (GST_STATE_PENDING (element) == GST_STATE_VOID_PENDING &&
1796       !GST_STATE_ERROR (element)) {
1797     GstState current_state = GST_STATE (element);
1798     GstMessage *message;
1799
1800     GST_CAT_INFO_OBJECT (GST_CAT_STATES, element,
1801         "lost state of %s", gst_element_state_get_name (current_state));
1802
1803     GST_STATE_PENDING (element) = current_state;
1804     GST_STATE_ERROR (element) = FALSE;
1805
1806     message = gst_message_new_state_changed (GST_OBJECT (element),
1807         current_state, current_state);
1808     gst_element_post_message (element, message);
1809   }
1810 }
1811
1812 /**
1813  * gst_element_set_state:
1814  * @element: a #GstElement to change state of.
1815  * @state: the element's new #GstState.
1816  *
1817  * Sets the state of the element. This function will try to set the
1818  * requested state by going through all the intermediary states and calling
1819  * the class's state change function for each.
1820  *
1821  * Returns: Result of the state change using #GstStateChangeReturn.
1822  *
1823  * MT safe.
1824  */
1825 GstStateChangeReturn
1826 gst_element_set_state (GstElement * element, GstState state)
1827 {
1828   GstElementClass *oclass;
1829   GstState current;
1830   GstStateChangeReturn return_val = GST_STATE_CHANGE_SUCCESS;
1831   GstStateChangeReturn ret;
1832   GstState pending;
1833   GTimeVal tv;
1834
1835   g_return_val_if_fail (GST_IS_ELEMENT (element), GST_STATE_CHANGE_FAILURE);
1836
1837   GST_DEBUG_OBJECT (element, "set_state to %s",
1838       gst_element_state_get_name (state));
1839   /* get current element state,  need to call the method so that
1840    * we call the virtual method and subclasses can implement their
1841    * own algorithms */
1842   GST_TIME_TO_TIMEVAL (0, tv);
1843   ret = gst_element_get_state (element, &current, &pending, &tv);
1844
1845   /* get the element state lock */
1846   GST_STATE_LOCK (element);
1847   /* this is the state we should go to */
1848   GST_STATE_FINAL (element) = state;
1849   if (ret == GST_STATE_CHANGE_ASYNC) {
1850     /* force next state keeping ASYNC, this is atomic as we hold
1851      * the STATE_LOCK */
1852     gst_element_commit_state (element);
1853     gst_element_lost_state (element);
1854   }
1855
1856   /* start with the current state */
1857   current = GST_STATE (element);
1858
1859   GST_CAT_INFO_OBJECT (GST_CAT_STATES, element, "setting state from %s to %s",
1860       gst_element_state_get_name (current), gst_element_state_get_name (state));
1861
1862   oclass = GST_ELEMENT_GET_CLASS (element);
1863
1864   /* We always perform at least one state change, even if the 
1865    * current state is equal to the required state. This is needed
1866    * for bins that sync their children. */
1867   do {
1868     GstState pending;
1869
1870     /* calculate the pending state */
1871     if (current < state)
1872       pending = current + 1;
1873     else if (current > state)
1874       pending = current - 1;
1875     else
1876       pending = current;
1877
1878     /* set the pending state variable */
1879     GST_STATE_PENDING (element) = pending;
1880
1881     GST_CAT_DEBUG_OBJECT (GST_CAT_STATES, element,
1882         "%s: setting state from %s to %s",
1883         (pending != state ? "intermediate" : "final"),
1884         gst_element_state_get_name (current),
1885         gst_element_state_get_name (pending));
1886
1887     /* call the state change function so it can set the state */
1888     if (oclass->change_state)
1889       return_val = (oclass->change_state) (element, GST_STATE_CHANGE (element));
1890     else
1891       return_val = GST_STATE_CHANGE_FAILURE;
1892
1893     /* clear the error and preroll flag, we need to do that after
1894      * calling the virtual change_state function so that it can use the
1895      * old previous value. */
1896     GST_STATE_ERROR (element) = FALSE;
1897     GST_STATE_NO_PREROLL (element) = FALSE;
1898
1899     switch (return_val) {
1900       case GST_STATE_CHANGE_FAILURE:
1901         GST_CAT_INFO_OBJECT (GST_CAT_STATES, element,
1902             "have failed change_state return");
1903         /* state change failure exits the loop */
1904         gst_element_abort_state (element);
1905         goto exit;
1906       case GST_STATE_CHANGE_ASYNC:
1907         GST_CAT_INFO_OBJECT (GST_CAT_STATES, element,
1908             "element will change state async");
1909         /* an async state change exits the loop, we can only
1910          * go to the next state change when this one completes. */
1911         goto exit;
1912       case GST_STATE_CHANGE_SUCCESS:
1913         GST_CAT_INFO_OBJECT (GST_CAT_STATES, element,
1914             "element changed state successfully");
1915         /* we can commit the state now and proceed to the next state */
1916         gst_element_commit_state (element);
1917         GST_CAT_INFO_OBJECT (GST_CAT_STATES, element, "committed state");
1918         break;
1919       case GST_STATE_CHANGE_NO_PREROLL:
1920         GST_CAT_INFO_OBJECT (GST_CAT_STATES, element,
1921             "element changed state successfully and can't preroll");
1922         /* we can commit the state now and proceed to the next state */
1923         gst_element_commit_state (element);
1924         GST_STATE_NO_PREROLL (element) = TRUE;
1925         GST_CAT_INFO_OBJECT (GST_CAT_STATES, element, "committed state");
1926         break;
1927       default:
1928         goto invalid_return;
1929     }
1930     /* get the current state of the element and see if we need to do more
1931      * state changes */
1932     current = GST_STATE (element);
1933   }
1934   while (current != state);
1935
1936 exit:
1937   GST_STATE_FINAL (element) = GST_STATE_VOID_PENDING;
1938   GST_STATE_UNLOCK (element);
1939
1940   GST_CAT_INFO_OBJECT (GST_CAT_STATES, element, "exit state change");
1941
1942   return return_val;
1943
1944   /* ERROR */
1945 invalid_return:
1946   {
1947     GST_STATE_FINAL (element) = GST_STATE_VOID_PENDING;
1948     GST_STATE_UNLOCK (element);
1949     /* somebody added a GST_STATE_ and forgot to do stuff here ! */
1950     g_critical ("unknown return value %d from a state change function",
1951         return_val);
1952     return GST_STATE_CHANGE_FAILURE;
1953   }
1954 }
1955
1956 /* gst_iterator_fold functions for pads_activate */
1957
1958 static gboolean
1959 activate_pads (GstPad * pad, GValue * ret, gboolean * active)
1960 {
1961   if (!gst_pad_set_active (pad, *active))
1962     g_value_set_boolean (ret, FALSE);
1963   else if (!*active)
1964     gst_pad_set_caps (pad, NULL);
1965
1966   gst_object_unref (pad);
1967   return TRUE;
1968 }
1969
1970 /* returns false on error or early cutout of the fold, true otherwise */
1971 static gboolean
1972 iterator_fold_with_resync (GstIterator * iter, GstIteratorFoldFunction func,
1973     GValue * ret, gpointer user_data)
1974 {
1975   GstIteratorResult ires;
1976   gboolean res = TRUE;
1977
1978   while (1) {
1979     ires = gst_iterator_fold (iter, func, ret, user_data);
1980
1981     switch (ires) {
1982       case GST_ITERATOR_RESYNC:
1983         gst_iterator_resync (iter);
1984         break;
1985       case GST_ITERATOR_DONE:
1986         res = TRUE;
1987         goto done;
1988       default:
1989         res = FALSE;
1990         goto done;
1991     }
1992   }
1993
1994 done:
1995   return res;
1996 }
1997
1998 /* is called with STATE_LOCK
1999  */
2000 static gboolean
2001 gst_element_pads_activate (GstElement * element, gboolean active)
2002 {
2003   GValue ret = { 0, };
2004   GstIterator *iter;
2005   gboolean fold_ok;
2006
2007   /* no need to unset this later, it's just a boolean */
2008   g_value_init (&ret, G_TYPE_BOOLEAN);
2009   g_value_set_boolean (&ret, TRUE);
2010
2011   iter = gst_element_iterate_src_pads (element);
2012   fold_ok = iterator_fold_with_resync
2013       (iter, (GstIteratorFoldFunction) activate_pads, &ret, &active);
2014   gst_iterator_free (iter);
2015   if (!fold_ok || !g_value_get_boolean (&ret))
2016     return FALSE;
2017
2018   iter = gst_element_iterate_sink_pads (element);
2019   fold_ok = iterator_fold_with_resync
2020       (iter, (GstIteratorFoldFunction) activate_pads, &ret, &active);
2021   gst_iterator_free (iter);
2022   if (!fold_ok || !g_value_get_boolean (&ret))
2023     return FALSE;
2024
2025   return TRUE;
2026 }
2027
2028 /* is called with STATE_LOCK */
2029 static GstStateChangeReturn
2030 gst_element_change_state (GstElement * element, GstStateChange transition)
2031 {
2032   GstState state, pending;
2033   GstStateChangeReturn result = GST_STATE_CHANGE_SUCCESS;
2034
2035   g_return_val_if_fail (GST_IS_ELEMENT (element), GST_STATE_CHANGE_FAILURE);
2036
2037   state = GST_STATE (element);
2038   pending = GST_STATE_PENDING (element);
2039
2040   /* if the element already is in the given state, we just return success */
2041   if (pending == GST_STATE_VOID_PENDING || state == GST_STATE_PENDING (element)) {
2042     GST_CAT_DEBUG_OBJECT (GST_CAT_STATES, element,
2043         "element is already in the %s state",
2044         gst_element_state_get_name (state));
2045     if (GST_STATE_NO_PREROLL (element))
2046       return GST_STATE_CHANGE_NO_PREROLL;
2047     else
2048       return GST_STATE_CHANGE_SUCCESS;
2049   }
2050
2051   GST_CAT_LOG_OBJECT (GST_CAT_STATES, element,
2052       "default handler tries setting state from %s to %s (%04x)",
2053       gst_element_state_get_name (state),
2054       gst_element_state_get_name (pending), transition);
2055
2056   switch (transition) {
2057     case GST_STATE_CHANGE_NULL_TO_READY:
2058       break;
2059     case GST_STATE_CHANGE_READY_TO_PAUSED:
2060       if (!gst_element_pads_activate (element, TRUE)) {
2061         result = GST_STATE_CHANGE_FAILURE;
2062       }
2063       break;
2064     case GST_STATE_CHANGE_PAUSED_TO_PLAYING:
2065       break;
2066     case GST_STATE_CHANGE_PLAYING_TO_PAUSED:
2067       break;
2068     case GST_STATE_CHANGE_PAUSED_TO_READY:
2069     case GST_STATE_CHANGE_READY_TO_NULL:
2070       /* deactivate pads in both cases, since they are activated on
2071          ready->paused but the element might not have made it to paused */
2072       if (!gst_element_pads_activate (element, FALSE)) {
2073         result = GST_STATE_CHANGE_FAILURE;
2074       } else {
2075         GST_LOCK (element);
2076         element->base_time = 0;
2077         GST_UNLOCK (element);
2078       }
2079       break;
2080     default:
2081       /* this will catch real but unhandled state changes;
2082        * can only be caused by:
2083        * - a new state was added
2084        * - somehow the element was asked to jump across an intermediate state
2085        */
2086       g_warning ("Unhandled state change from %s to %s",
2087           gst_element_state_get_name (state),
2088           gst_element_state_get_name (pending));
2089       break;
2090   }
2091
2092   return result;
2093 }
2094
2095 /**
2096  * gst_element_get_factory:
2097  * @element: a #GstElement to request the element factory of.
2098  *
2099  * Retrieves the factory that was used to create this element.
2100  *
2101  * Returns: the #GstElementFactory used for creating this element.
2102  */
2103 GstElementFactory *
2104 gst_element_get_factory (GstElement * element)
2105 {
2106   g_return_val_if_fail (GST_IS_ELEMENT (element), NULL);
2107
2108   return GST_ELEMENT_GET_CLASS (element)->elementfactory;
2109 }
2110
2111 static void
2112 gst_element_dispose (GObject * object)
2113 {
2114   GstElement *element = GST_ELEMENT (object);
2115
2116   GST_CAT_INFO_OBJECT (GST_CAT_REFCOUNTING, element, "dispose");
2117
2118   /* first we break all our links with the outside */
2119   while (element->pads) {
2120     gst_element_remove_pad (element, GST_PAD_CAST (element->pads->data));
2121   }
2122   if (G_UNLIKELY (element->pads != 0)) {
2123     g_critical ("could not remove pads from element %s",
2124         GST_STR_NULL (GST_OBJECT_NAME (object)));
2125   }
2126
2127   GST_LOCK (element);
2128   gst_object_replace ((GstObject **) & element->clock, NULL);
2129   gst_object_replace ((GstObject **) & element->bus, NULL);
2130   GST_UNLOCK (element);
2131
2132   GST_CAT_INFO_OBJECT (GST_CAT_REFCOUNTING, element, "dispose parent");
2133
2134   G_OBJECT_CLASS (parent_class)->dispose (object);
2135 }
2136
2137 static void
2138 gst_element_finalize (GObject * object)
2139 {
2140   GstElement *element = GST_ELEMENT (object);
2141
2142   GST_CAT_INFO_OBJECT (GST_CAT_REFCOUNTING, element, "finalize");
2143
2144   GST_STATE_LOCK (element);
2145   if (element->state_cond)
2146     g_cond_free (element->state_cond);
2147   element->state_cond = NULL;
2148   GST_STATE_UNLOCK (element);
2149   g_mutex_free (element->state_lock);
2150
2151   GST_CAT_INFO_OBJECT (GST_CAT_REFCOUNTING, element, "finalize parent");
2152
2153   G_OBJECT_CLASS (parent_class)->finalize (object);
2154 }
2155
2156 #ifndef GST_DISABLE_LOADSAVE
2157 /**
2158  * gst_element_save_thyself:
2159  * @element: a #GstElement to save.
2160  * @parent: the xml parent node.
2161  *
2162  * Saves the element as part of the given XML structure.
2163  *
2164  * Returns: the new #xmlNodePtr.
2165  */
2166 static xmlNodePtr
2167 gst_element_save_thyself (GstObject * object, xmlNodePtr parent)
2168 {
2169   GList *pads;
2170   GstElementClass *oclass;
2171   GParamSpec **specs, *spec;
2172   guint nspecs;
2173   gint i;
2174   GValue value = { 0, };
2175   GstElement *element;
2176
2177   g_return_val_if_fail (GST_IS_ELEMENT (object), parent);
2178
2179   element = GST_ELEMENT (object);
2180
2181   oclass = GST_ELEMENT_GET_CLASS (element);
2182
2183   xmlNewChild (parent, NULL, (xmlChar *) "name",
2184       (xmlChar *) GST_ELEMENT_NAME (element));
2185
2186   if (oclass->elementfactory != NULL) {
2187     GstElementFactory *factory = (GstElementFactory *) oclass->elementfactory;
2188
2189     xmlNewChild (parent, NULL, (xmlChar *) "type",
2190         (xmlChar *) GST_PLUGIN_FEATURE (factory)->name);
2191   }
2192
2193   /* params */
2194   specs = g_object_class_list_properties (G_OBJECT_GET_CLASS (object), &nspecs);
2195
2196   for (i = 0; i < nspecs; i++) {
2197     spec = specs[i];
2198     if (spec->flags & G_PARAM_READABLE) {
2199       xmlNodePtr param;
2200       char *contents;
2201
2202       g_value_init (&value, G_PARAM_SPEC_VALUE_TYPE (spec));
2203
2204       g_object_get_property (G_OBJECT (element), spec->name, &value);
2205       param = xmlNewChild (parent, NULL, (xmlChar *) "param", NULL);
2206       xmlNewChild (param, NULL, (xmlChar *) "name", (xmlChar *) spec->name);
2207
2208       if (G_IS_PARAM_SPEC_STRING (spec))
2209         contents = g_value_dup_string (&value);
2210       else if (G_IS_PARAM_SPEC_ENUM (spec))
2211         contents = g_strdup_printf ("%d", g_value_get_enum (&value));
2212       else if (G_IS_PARAM_SPEC_INT64 (spec))
2213         contents = g_strdup_printf ("%" G_GINT64_FORMAT,
2214             g_value_get_int64 (&value));
2215       else
2216         contents = g_strdup_value_contents (&value);
2217
2218       xmlNewChild (param, NULL, (xmlChar *) "value", (xmlChar *) contents);
2219       g_free (contents);
2220
2221       g_value_unset (&value);
2222     }
2223   }
2224
2225   g_free (specs);
2226
2227   pads = GST_ELEMENT_PADS (element);
2228
2229   while (pads) {
2230     GstPad *pad = GST_PAD (pads->data);
2231
2232     /* figure out if it's a direct pad or a ghostpad */
2233     if (GST_ELEMENT (GST_OBJECT_PARENT (pad)) == element) {
2234       xmlNodePtr padtag = xmlNewChild (parent, NULL, (xmlChar *) "pad", NULL);
2235
2236       gst_object_save_thyself (GST_OBJECT (pad), padtag);
2237     }
2238     pads = g_list_next (pads);
2239   }
2240
2241   return parent;
2242 }
2243
2244 static void
2245 gst_element_restore_thyself (GstObject * object, xmlNodePtr self)
2246 {
2247   xmlNodePtr children;
2248   GstElement *element;
2249   gchar *name = NULL;
2250   gchar *value = NULL;
2251
2252   element = GST_ELEMENT (object);
2253   g_return_if_fail (element != NULL);
2254
2255   /* parameters */
2256   children = self->xmlChildrenNode;
2257   while (children) {
2258     if (!strcmp ((char *) children->name, "param")) {
2259       xmlNodePtr child = children->xmlChildrenNode;
2260
2261       while (child) {
2262         if (!strcmp ((char *) child->name, "name")) {
2263           name = (gchar *) xmlNodeGetContent (child);
2264         } else if (!strcmp ((char *) child->name, "value")) {
2265           value = (gchar *) xmlNodeGetContent (child);
2266         }
2267         child = child->next;
2268       }
2269       /* FIXME: can this just be g_object_set ? */
2270       gst_util_set_object_arg (G_OBJECT (element), name, value);
2271       /* g_object_set (G_OBJECT (element), name, value, NULL); */
2272       g_free (name);
2273       g_free (value);
2274     }
2275     children = children->next;
2276   }
2277
2278   /* pads */
2279   children = self->xmlChildrenNode;
2280   while (children) {
2281     if (!strcmp ((char *) children->name, "pad")) {
2282       gst_pad_load_and_link (children, GST_OBJECT (element));
2283     }
2284     children = children->next;
2285   }
2286
2287   if (GST_OBJECT_CLASS (parent_class)->restore_thyself)
2288     (GST_OBJECT_CLASS (parent_class)->restore_thyself) (object, self);
2289 }
2290 #endif /* GST_DISABLE_LOADSAVE */
2291
2292 static void
2293 gst_element_set_bus_func (GstElement * element, GstBus * bus)
2294 {
2295   g_return_if_fail (GST_IS_ELEMENT (element));
2296
2297   GST_CAT_DEBUG_OBJECT (GST_CAT_PARENTAGE, element, "setting bus to %p", bus);
2298
2299   GST_LOCK (element);
2300   gst_object_replace ((GstObject **) & GST_ELEMENT_BUS (element),
2301       GST_OBJECT (bus));
2302   GST_UNLOCK (element);
2303 }
2304
2305 /**
2306  * gst_element_set_bus:
2307  * @element: a #GstElement to set the bus of.
2308  * @bus: the #GstBus to set.
2309  *
2310  * Sets the bus of the element.  For internal use only, unless you're
2311  * testing elements.
2312  *
2313  * MT safe.
2314  */
2315 void
2316 gst_element_set_bus (GstElement * element, GstBus * bus)
2317 {
2318   GstElementClass *oclass;
2319
2320   g_return_if_fail (GST_IS_ELEMENT (element));
2321
2322   oclass = GST_ELEMENT_GET_CLASS (element);
2323
2324   if (oclass->set_bus)
2325     oclass->set_bus (element, bus);
2326 }
2327
2328 /**
2329  * gst_element_get_bus:
2330  * @element: a #GstElement to get the bus of.
2331  *
2332  * Returns the bus of the element.
2333  *
2334  * Returns: the element's #GstBus. unref after usage.
2335  *
2336  * MT safe.
2337  */
2338 GstBus *
2339 gst_element_get_bus (GstElement * element)
2340 {
2341   GstBus *result = NULL;
2342
2343   g_return_val_if_fail (GST_IS_ELEMENT (element), result);
2344
2345   GST_LOCK (element);
2346   result = GST_ELEMENT_BUS (element);
2347   gst_object_ref (result);
2348   GST_UNLOCK (element);
2349
2350   GST_DEBUG_OBJECT (element, "got bus %" GST_PTR_FORMAT, result);
2351
2352   return result;
2353 }