API: gst_parse_launch_full()
[platform/upstream/gstreamer.git] / gst / gst.c
1 /* GStreamer
2  * Copyright (C) 1999,2000 Erik Walthinsen <omega@cse.ogi.edu>
3  *                    2000 Wim Taymans <wtay@chello.be>
4  *
5  * gst.c: Initialization and non-pipeline operations
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 /**
24  * SECTION:gst
25  * @short_description: Media library supporting arbitrary formats and filter
26  *                     graphs.
27  * @see_also: Check out both <ulink url="http://www.cse.ogi.edu/sysl/">OGI's
28  *            pipeline</ulink> and Microsoft's DirectShow for some background.
29  *
30  * GStreamer is a framework for constructing graphs of various filters
31  * (termed elements here) that will handle streaming media.  Any discreet
32  * (packetizable) media type is supported, with provisions for automatically
33  * determining source type.  Formatting/framing information is provided with
34  * a powerful negotiation framework.  Plugins are heavily used to provide for
35  * all elements, allowing one to construct plugins outside of the GST
36  * library, even released binary-only if license require (please don't).
37  *
38  * GStreamer borrows heavily from both the <ulink
39  * url="http://www.cse.ogi.edu/sysl/">OGI media pipeline</ulink> and
40  * Microsoft's DirectShow, hopefully taking the best of both and leaving the
41  * cruft behind. Its interface is slowly getting stable.
42  *
43  * The <application>GStreamer</application> library should be initialized with
44  * gst_init() before it can be used. You should pass pointers to the main argc
45  * and argv variables so that GStreamer can process its own command line
46  * options, as shown in the following example.
47  *
48  * <example>
49  * <title>Initializing the gstreamer library</title>
50  * <programlisting language="c">
51  * int
52  * main (int argc, char *argv[])
53  * {
54  *   // initialize the GStreamer library
55  *   gst_init (&amp;argc, &amp;argv);
56  *   ...
57  * }
58  * </programlisting>
59  * </example>
60  *
61  * It's allowed to pass two NULL pointers to gst_init() in case you don't want
62  * to pass the command line args to GStreamer.
63  *
64  * You can also use GOption to initialize your own parameters as shown in
65  * the next code fragment:
66  * <example>
67  * <title>Initializing own parameters when initializing gstreamer</title>
68  * <programlisting>
69  * static gboolean stats = FALSE;
70  * ...
71  * int
72  * main (int argc, char *argv[])
73  * {
74  *  GOptionEntry options[] = {
75  *   {"tags", 't', 0, G_OPTION_ARG_NONE, &amp;tags,
76  *       N_("Output tags (also known as metadata)"), NULL},
77  *   {NULL}
78  *  };
79  *  // must initialise the threading system before using any other GLib funtion
80  *  if (!g_thread_supported ())
81  *    g_thread_init (NULL);
82  *  ctx = g_option_context_new ("[ADDITIONAL ARGUMENTS]");
83  *  g_option_context_add_main_entries (ctx, options, GETTEXT_PACKAGE);
84  *  g_option_context_add_group (ctx, gst_init_get_option_group ());
85  *  if (!g_option_context_parse (ctx, &amp;argc, &amp;argv, &amp;err)) {
86  *    g_print ("Error initializing: &percnt;s\n", GST_STR_NULL (err->message));
87  *    exit (1);
88  *  }
89  *  g_option_context_free (ctx);
90  * ...
91  * }
92  * </programlisting>
93  * </example>
94  *
95  * Use gst_version() to query the library version at runtime or use the
96  * GST_VERSION_* macros to find the version at compile time. Optionally
97  * gst_version_string() returns a printable string.
98  *
99  * The gst_deinit() call is used to clean up all internal resources used
100  * by <application>GStreamer</application>. It is mostly used in unit tests 
101  * to check for leaks.
102  *
103  * Last reviewed on 2006-08-11 (0.10.10)
104  */
105
106 #include "gst_private.h"
107 #include <stdlib.h>
108 #include <stdio.h>
109 #include <sys/types.h>
110 #ifdef HAVE_FORK
111 #include <sys/wait.h>
112 #endif /* HAVE_FORK */
113 #ifdef HAVE_SYS_UTSNAME_H
114 #include <sys/utsname.h>
115 #endif
116 #ifdef HAVE_UNISTD_H
117 #include <unistd.h>
118 #endif
119
120 #include "gst-i18n-lib.h"
121 #include <locale.h>             /* for LC_ALL */
122
123 #include "gst.h"
124
125 #define GST_CAT_DEFAULT GST_CAT_GST_INIT
126
127 #define MAX_PATH_SPLIT  16
128 #define GST_PLUGIN_SEPARATOR ","
129
130 static gboolean gst_initialized = FALSE;
131 static gboolean gst_deinitialized = FALSE;
132
133 #ifndef GST_DISABLE_REGISTRY
134 static GList *plugin_paths = NULL;      /* for delayed processing in post_init */
135 #endif
136
137 #ifndef GST_DISABLE_GST_DEBUG
138 extern const gchar *priv_gst_dump_dot_dir;
139 #endif
140
141 /* defaults */
142 #if defined(HAVE_FORK) && !defined(GST_HAVE_UNSAFE_FORK)
143 #define DEFAULT_FORK TRUE
144 #else
145 #define DEFAULT_FORK FALSE
146 #endif /* HAVE_FORK */
147
148 /* set to TRUE when segfaults need to be left as is */
149 static gboolean _gst_disable_segtrap = FALSE;
150
151 /* control the behaviour of registry rebuild */
152 static gboolean _gst_enable_registry_fork = DEFAULT_FORK;
153
154 /*set to TRUE when registry needn't to be updated */
155 static gboolean _gst_disable_registry_update = FALSE;
156
157 static void load_plugin_func (gpointer data, gpointer user_data);
158 static gboolean init_pre (GOptionContext * context, GOptionGroup * group,
159     gpointer data, GError ** error);
160 static gboolean init_post (GOptionContext * context, GOptionGroup * group,
161     gpointer data, GError ** error);
162 #ifndef GST_DISABLE_OPTION_PARSING
163 static gboolean parse_goption_arg (const gchar * s_opt,
164     const gchar * arg, gpointer data, GError ** err);
165 #endif
166
167 static GSList *preload_plugins = NULL;
168
169 const gchar g_log_domain_gstreamer[] = "GStreamer";
170
171 static void
172 debug_log_handler (const gchar * log_domain,
173     GLogLevelFlags log_level, const gchar * message, gpointer user_data)
174 {
175   g_log_default_handler (log_domain, log_level, message, user_data);
176   /* FIXME: do we still need this ? fatal errors these days are all
177    * other than core errors */
178   /* g_on_error_query (NULL); */
179 }
180
181 enum
182 {
183   ARG_VERSION = 1,
184   ARG_FATAL_WARNINGS,
185 #ifndef GST_DISABLE_GST_DEBUG
186   ARG_DEBUG_LEVEL,
187   ARG_DEBUG,
188   ARG_DEBUG_DISABLE,
189   ARG_DEBUG_NO_COLOR,
190   ARG_DEBUG_HELP,
191 #endif
192   ARG_PLUGIN_SPEW,
193   ARG_PLUGIN_PATH,
194   ARG_PLUGIN_LOAD,
195   ARG_SEGTRAP_DISABLE,
196   ARG_REGISTRY_UPDATE_DISABLE,
197   ARG_REGISTRY_FORK_DISABLE
198 };
199
200 /* debug-spec ::= category-spec [, category-spec]*
201  * category-spec ::= category:val | val
202  * category ::= [^:]+
203  * val ::= [0-5]
204  */
205
206 #ifndef NUL
207 #define NUL '\0'
208 #endif
209
210 #ifndef GST_DISABLE_GST_DEBUG
211 static gboolean
212 parse_debug_category (gchar * str, const gchar ** category)
213 {
214   if (!str)
215     return FALSE;
216
217   /* works in place */
218   g_strstrip (str);
219
220   if (str[0] != NUL) {
221     *category = str;
222     return TRUE;
223   }
224
225   return FALSE;
226 }
227
228 static gboolean
229 parse_debug_level (gchar * str, gint * level)
230 {
231   if (!str)
232     return FALSE;
233
234   /* works in place */
235   g_strstrip (str);
236
237   if (str[0] != NUL && str[1] == NUL
238       && str[0] >= '0' && str[0] < '0' + GST_LEVEL_COUNT) {
239     *level = str[0] - '0';
240     return TRUE;
241   }
242
243   return FALSE;
244 }
245
246 static void
247 parse_debug_list (const gchar * list)
248 {
249   gchar **split;
250   gchar **walk;
251
252   g_return_if_fail (list != NULL);
253
254   split = g_strsplit (list, ",", 0);
255
256   for (walk = split; *walk; walk++) {
257     if (strchr (*walk, ':')) {
258       gchar **values = g_strsplit (*walk, ":", 2);
259
260       if (values[0] && values[1]) {
261         gint level;
262         const gchar *category;
263
264         if (parse_debug_category (values[0], &category)
265             && parse_debug_level (values[1], &level))
266           gst_debug_set_threshold_for_name (category, level);
267       }
268
269       g_strfreev (values);
270     } else {
271       gint level;
272
273       if (parse_debug_level (*walk, &level))
274         gst_debug_set_default_threshold (level);
275     }
276   }
277
278   g_strfreev (split);
279 }
280 #endif
281
282 /**
283  * gst_init_get_option_group:
284  *
285  * Returns a #GOptionGroup with GStreamer's argument specifications. The
286  * group is set up to use standard GOption callbacks, so when using this
287  * group in combination with GOption parsing methods, all argument parsing
288  * and initialization is automated.
289  *
290  * This function is useful if you want to integrate GStreamer with other
291  * libraries that use GOption (see g_option_context_add_group() ).
292  *
293  * If you use this function, you should make sure you initialise the GLib
294  * threading system as one of the very first things in your program
295  * (see the example at the beginning of this section).
296  *
297  * Returns: a pointer to GStreamer's option group.
298  */
299
300 GOptionGroup *
301 gst_init_get_option_group (void)
302 {
303 #ifndef GST_DISABLE_OPTION_PARSING
304   GOptionGroup *group;
305   static const GOptionEntry gst_args[] = {
306     {"gst-version", 0, G_OPTION_FLAG_NO_ARG, G_OPTION_ARG_CALLBACK,
307         (gpointer) parse_goption_arg, N_("Print the GStreamer version"), NULL},
308     {"gst-fatal-warnings", 0, G_OPTION_FLAG_NO_ARG, G_OPTION_ARG_CALLBACK,
309         (gpointer) parse_goption_arg, N_("Make all warnings fatal"), NULL},
310 #ifndef GST_DISABLE_GST_DEBUG
311     {"gst-debug-help", 0, G_OPTION_FLAG_NO_ARG, G_OPTION_ARG_CALLBACK,
312           (gpointer) parse_goption_arg,
313           N_("Print available debug categories and exit"),
314         NULL},
315     {"gst-debug-level", 0, 0, G_OPTION_ARG_CALLBACK,
316           (gpointer) parse_goption_arg,
317           N_("Default debug level from 1 (only error) to 5 (anything) or "
318               "0 for no output"),
319         N_("LEVEL")},
320     {"gst-debug", 0, 0, G_OPTION_ARG_CALLBACK, (gpointer) parse_goption_arg,
321           N_("Comma-separated list of category_name:level pairs to set "
322               "specific levels for the individual categories. Example: "
323               "GST_AUTOPLUG:5,GST_ELEMENT_*:3"),
324         N_("LIST")},
325     {"gst-debug-no-color", 0, G_OPTION_FLAG_NO_ARG, G_OPTION_ARG_CALLBACK,
326           (gpointer) parse_goption_arg, N_("Disable colored debugging output"),
327         NULL},
328     {"gst-debug-disable", 0, G_OPTION_FLAG_NO_ARG, G_OPTION_ARG_CALLBACK,
329         (gpointer) parse_goption_arg, N_("Disable debugging"), NULL},
330 #endif
331     {"gst-plugin-spew", 0, G_OPTION_FLAG_NO_ARG, G_OPTION_ARG_CALLBACK,
332           (gpointer) parse_goption_arg,
333           N_("Enable verbose plugin loading diagnostics"),
334         NULL},
335     {"gst-plugin-path", 0, 0, G_OPTION_ARG_CALLBACK,
336           (gpointer) parse_goption_arg,
337         N_("Colon-separated paths containing plugins"), N_("PATHS")},
338     {"gst-plugin-load", 0, 0, G_OPTION_ARG_CALLBACK,
339           (gpointer) parse_goption_arg,
340           N_("Comma-separated list of plugins to preload in addition to the "
341               "list stored in environment variable GST_PLUGIN_PATH"),
342         N_("PLUGINS")},
343     {"gst-disable-segtrap", 0, G_OPTION_FLAG_NO_ARG, G_OPTION_ARG_CALLBACK,
344           (gpointer) parse_goption_arg,
345           N_("Disable trapping of segmentation faults during plugin loading"),
346         NULL},
347     {"gst-disable-registry-update", 0, G_OPTION_FLAG_NO_ARG,
348           G_OPTION_ARG_CALLBACK,
349           (gpointer) parse_goption_arg,
350           N_("Disable updating the registry"),
351         NULL},
352     {"gst-disable-registry-fork", 0, G_OPTION_FLAG_NO_ARG,
353           G_OPTION_ARG_CALLBACK,
354           (gpointer) parse_goption_arg,
355           N_("Disable the use of fork() while scanning the registry"),
356         NULL},
357     {NULL}
358   };
359
360   /* The GLib threading system must be initialised before calling any other
361    * GLib function according to the documentation; if the application hasn't
362    * called gst_init() yet or initialised the threading system otherwise, we
363    * better issue a warning here (since chances are high that the application
364    * has already called other GLib functions such as g_option_context_new() */
365   if (!g_thread_supported ()) {
366     g_warning ("The GStreamer function gst_init_get_option_group() was\n"
367         "\tcalled, but the GLib threading system has not been initialised\n"
368         "\tyet, something that must happen before any other GLib function\n"
369         "\tis called. The application needs to be fixed so that it calls\n"
370         "\t   if (!g_thread_supported ()) g_thread_init(NULL);\n"
371         "\tas very first thing in its main() function. Please file a bug\n"
372         "\tagainst this application.");
373     g_thread_init (NULL);
374   }
375
376   group = g_option_group_new ("gst", _("GStreamer Options"),
377       _("Show GStreamer Options"), NULL, NULL);
378   g_option_group_set_parse_hooks (group, (GOptionParseFunc) init_pre,
379       (GOptionParseFunc) init_post);
380
381   g_option_group_add_entries (group, gst_args);
382   g_option_group_set_translation_domain (group, GETTEXT_PACKAGE);
383
384   return group;
385 #else
386   return NULL;
387 #endif
388 }
389
390 /**
391  * gst_init_check:
392  * @argc: pointer to application's argc
393  * @argv: pointer to application's argv
394  * @err: pointer to a #GError to which a message will be posted on error
395  *
396  * Initializes the GStreamer library, setting up internal path lists,
397  * registering built-in elements, and loading standard plugins.
398  *
399  * This function will return %FALSE if GStreamer could not be initialized
400  * for some reason.  If you want your program to fail fatally,
401  * use gst_init() instead.
402  *
403  * This function should be called before calling any other GLib functions. If
404  * this is not an option, your program must initialise the GLib thread system
405  * using g_thread_init() before any other GLib functions are called.
406  *
407  * Returns: %TRUE if GStreamer could be initialized.
408  */
409 gboolean
410 gst_init_check (int *argc, char **argv[], GError ** err)
411 {
412 #ifndef GST_DISABLE_OPTION_PARSING
413   GOptionGroup *group;
414   GOptionContext *ctx;
415 #endif
416   gboolean res;
417
418   if (!g_thread_supported ())
419     g_thread_init (NULL);
420
421   if (gst_initialized) {
422     GST_DEBUG ("already initialized gst");
423     return TRUE;
424   }
425 #ifndef GST_DISABLE_OPTION_PARSING
426   ctx = g_option_context_new ("- GStreamer initialization");
427   g_option_context_set_ignore_unknown_options (ctx, TRUE);
428   group = gst_init_get_option_group ();
429   g_option_context_add_group (ctx, group);
430   res = g_option_context_parse (ctx, argc, argv, err);
431   g_option_context_free (ctx);
432 #else
433   init_pre (NULL, NULL, NULL, NULL);
434   init_post (NULL, NULL, NULL, NULL);
435   res = TRUE;
436 #endif
437
438   gst_initialized = res;
439
440   if (res) {
441     GST_INFO ("initialized GStreamer successfully");
442   } else {
443     GST_INFO ("failed to initialize GStreamer");
444   }
445
446   return res;
447 }
448
449 /**
450  * gst_init:
451  * @argc: pointer to application's argc
452  * @argv: pointer to application's argv
453  *
454  * Initializes the GStreamer library, setting up internal path lists,
455  * registering built-in elements, and loading standard plugins.
456  *
457  * This function should be called before calling any other GLib functions. If
458  * this is not an option, your program must initialise the GLib thread system
459  * using g_thread_init() before any other GLib functions are called.
460  *
461  * <note><para>
462  * This function will terminate your program if it was unable to initialize
463  * GStreamer for some reason.  If you want your program to fall back,
464  * use gst_init_check() instead.
465  * </para></note>
466  *
467  * WARNING: This function does not work in the same way as corresponding
468  * functions in other glib-style libraries, such as gtk_init().  In
469  * particular, unknown command line options cause this function to
470  * abort program execution.
471  */
472 void
473 gst_init (int *argc, char **argv[])
474 {
475   GError *err = NULL;
476
477   if (!gst_init_check (argc, argv, &err)) {
478     g_print ("Could not initialize GStreamer: %s\n",
479         err ? err->message : "unknown error occurred");
480     if (err) {
481       g_error_free (err);
482     }
483     exit (1);
484   }
485 }
486
487 #ifndef GST_DISABLE_REGISTRY
488 static void
489 add_path_func (gpointer data, gpointer user_data)
490 {
491   GST_INFO ("Adding plugin path: \"%s\", will scan later", (gchar *) data);
492   plugin_paths = g_list_append (plugin_paths, g_strdup (data));
493 }
494 #endif
495
496 #ifndef GST_DISABLE_OPTION_PARSING
497 static void
498 prepare_for_load_plugin_func (gpointer data, gpointer user_data)
499 {
500   preload_plugins = g_slist_prepend (preload_plugins, g_strdup (data));
501 }
502 #endif
503
504 static void
505 load_plugin_func (gpointer data, gpointer user_data)
506 {
507   GstPlugin *plugin;
508   const gchar *filename;
509   GError *err = NULL;
510
511   filename = (const gchar *) data;
512
513   plugin = gst_plugin_load_file (filename, &err);
514
515   if (plugin) {
516     GST_INFO ("Loaded plugin: \"%s\"", filename);
517
518     gst_default_registry_add_plugin (plugin);
519   } else {
520     if (err) {
521       /* Report error to user, and free error */
522       GST_ERROR ("Failed to load plugin: %s", err->message);
523       g_error_free (err);
524     } else {
525       GST_WARNING ("Failed to load plugin: \"%s\"", filename);
526     }
527   }
528 }
529
530 #ifndef GST_DISABLE_OPTION_PARSING
531 static void
532 split_and_iterate (const gchar * stringlist, gchar * separator, GFunc iterator,
533     gpointer user_data)
534 {
535   gchar **strings;
536   gint j = 0;
537   gchar *lastlist = g_strdup (stringlist);
538
539   while (lastlist) {
540     strings = g_strsplit (lastlist, separator, MAX_PATH_SPLIT);
541     g_free (lastlist);
542     lastlist = NULL;
543
544     while (strings[j]) {
545       iterator (strings[j], user_data);
546       if (++j == MAX_PATH_SPLIT) {
547         lastlist = g_strdup (strings[j]);
548         j = 0;
549         break;
550       }
551     }
552     g_strfreev (strings);
553   }
554 }
555 #endif
556
557 /* we have no fail cases yet, but maybe in the future */
558 static gboolean
559 init_pre (GOptionContext * context, GOptionGroup * group, gpointer data,
560     GError ** error)
561 {
562   if (gst_initialized) {
563     GST_DEBUG ("already initialized");
564     return TRUE;
565   }
566
567   /* GStreamer was built against a GLib >= 2.8 and is therefore not doing
568    * the refcount hack. Check that it isn't being run against an older GLib */
569   if (glib_major_version < 2 ||
570       (glib_major_version == 2 && glib_minor_version < 8)) {
571     g_warning ("GStreamer was compiled against GLib %d.%d.%d but is running"
572         " against %d.%d.%d. This will cause reference counting issues",
573         GLIB_MAJOR_VERSION, GLIB_MINOR_VERSION, GLIB_MICRO_VERSION,
574         glib_major_version, glib_minor_version, glib_micro_version);
575   }
576
577   g_type_init ();
578
579   /* we need threading to be enabled right here */
580   g_assert (g_thread_supported ());
581   _gst_debug_init ();
582
583 #ifdef ENABLE_NLS
584   setlocale (LC_ALL, "");
585   bindtextdomain (GETTEXT_PACKAGE, LOCALEDIR);
586 #endif /* ENABLE_NLS */
587
588 #ifndef GST_DISABLE_GST_DEBUG
589   {
590     const gchar *debug_list;
591
592     if (g_getenv ("GST_DEBUG_NO_COLOR") != NULL)
593       gst_debug_set_colored (FALSE);
594
595     debug_list = g_getenv ("GST_DEBUG");
596     if (debug_list) {
597       parse_debug_list (debug_list);
598     }
599   }
600
601   priv_gst_dump_dot_dir = g_getenv ("GST_DEBUG_DUMP_DOT_DIR");
602 #endif
603   /* This is the earliest we can make stuff show up in the logs.
604    * So give some useful info about GStreamer here */
605   GST_INFO ("Initializing GStreamer Core Library version %s", VERSION);
606   GST_INFO ("Using library installed in %s", LIBDIR);
607
608   /* Print some basic system details if possible (OS/architecture) */
609 #ifdef HAVE_SYS_UTSNAME_H
610   {
611     struct utsname sys_details;
612
613     if (uname (&sys_details) == 0) {
614       GST_INFO ("%s %s %s %s %s", sys_details.sysname,
615           sys_details.nodename, sys_details.release, sys_details.version,
616           sys_details.machine);
617     }
618   }
619 #endif
620
621   return TRUE;
622 }
623
624 static gboolean
625 gst_register_core_elements (GstPlugin * plugin)
626 {
627   /* register some standard builtin types */
628   if (!gst_element_register (plugin, "bin", GST_RANK_PRIMARY,
629           GST_TYPE_BIN) ||
630       !gst_element_register (plugin, "pipeline", GST_RANK_PRIMARY,
631           GST_TYPE_PIPELINE)
632       )
633     g_assert_not_reached ();
634
635   return TRUE;
636 }
637
638 #ifndef GST_DISABLE_REGISTRY
639
640 typedef enum
641 {
642   REGISTRY_SCAN_AND_UPDATE_FAILURE = 0,
643   REGISTRY_SCAN_AND_UPDATE_SUCCESS_NOT_CHANGED,
644   REGISTRY_SCAN_AND_UPDATE_SUCCESS_UPDATED
645 } GstRegistryScanAndUpdateResult;
646
647 /*
648  * scan_and_update_registry:
649  * @default_registry: the #GstRegistry
650  * @registry_file: registry filename
651  * @write_changes: write registry if it has changed?
652  *
653  * Scans for registry changes and eventually updates the registry cache. 
654  *
655  * Return: %REGISTRY_SCAN_AND_UPDATE_FAILURE if the registry could not scanned
656  *         or updated, %REGISTRY_SCAN_AND_UPDATE_SUCCESS_NOT_CHANGED if the
657  *         registry is clean and %REGISTRY_SCAN_AND_UPDATE_SUCCESS_UPDATED if
658  *         it has been updated and the cache needs to be re-read.
659  */
660 static GstRegistryScanAndUpdateResult
661 scan_and_update_registry (GstRegistry * default_registry,
662     const gchar * registry_file, gboolean write_changes, GError ** error)
663 {
664   const gchar *plugin_path;
665   gboolean changed = FALSE;
666   GList *l;
667
668   GST_INFO ("Validating registry cache: %s", registry_file);
669   /* It sounds tempting to just compare the mtime of directories with the mtime
670    * of the registry cache, but it does not work. It would not catch updated
671    * plugins, which might bring more or less features.
672    */
673
674   /* scan paths specified via --gst-plugin-path */
675   GST_DEBUG ("scanning paths added via --gst-plugin-path");
676   for (l = plugin_paths; l != NULL; l = l->next) {
677     GST_INFO ("Scanning plugin path: \"%s\"", (gchar *) l->data);
678     /* FIXME: add changed |= here as well? */
679     gst_registry_scan_path (default_registry, (gchar *) l->data);
680   }
681   /* keep plugin_paths around in case a re-scan is forced later on */
682
683   /* GST_PLUGIN_PATH specifies a list of directories to scan for
684    * additional plugins.  These take precedence over the system plugins */
685   plugin_path = g_getenv ("GST_PLUGIN_PATH");
686   if (plugin_path) {
687     char **list;
688     int i;
689
690     GST_DEBUG ("GST_PLUGIN_PATH set to %s", plugin_path);
691     list = g_strsplit (plugin_path, G_SEARCHPATH_SEPARATOR_S, 0);
692     for (i = 0; list[i]; i++) {
693       changed |= gst_registry_scan_path (default_registry, list[i]);
694     }
695     g_strfreev (list);
696   } else {
697     GST_DEBUG ("GST_PLUGIN_PATH not set");
698   }
699
700   /* GST_PLUGIN_SYSTEM_PATH specifies a list of plugins that are always
701    * loaded by default.  If not set, this defaults to the system-installed
702    * path, and the plugins installed in the user's home directory */
703   plugin_path = g_getenv ("GST_PLUGIN_SYSTEM_PATH");
704   if (plugin_path == NULL) {
705     char *home_plugins;
706
707     GST_DEBUG ("GST_PLUGIN_SYSTEM_PATH not set");
708
709     /* plugins in the user's home directory take precedence over
710      * system-installed ones */
711     home_plugins = g_build_filename (g_get_home_dir (),
712         ".gstreamer-" GST_MAJORMINOR, "plugins", NULL);
713     GST_DEBUG ("scanning home plugins %s", home_plugins);
714     changed |= gst_registry_scan_path (default_registry, home_plugins);
715     g_free (home_plugins);
716
717     /* add the main (installed) library path */
718     GST_DEBUG ("scanning main plugins %s", PLUGINDIR);
719     changed |= gst_registry_scan_path (default_registry, PLUGINDIR);
720   } else {
721     gchar **list;
722     gint i;
723
724     GST_DEBUG ("GST_PLUGIN_SYSTEM_PATH set to %s", plugin_path);
725     list = g_strsplit (plugin_path, G_SEARCHPATH_SEPARATOR_S, 0);
726     for (i = 0; list[i]; i++) {
727       changed |= gst_registry_scan_path (default_registry, list[i]);
728     }
729     g_strfreev (list);
730   }
731
732   /* Remove cached plugins so stale info is cleared. */
733   changed |= _priv_gst_registry_remove_cache_plugins (default_registry);
734
735   if (!changed) {
736     GST_INFO ("Registry cache has not changed");
737     return REGISTRY_SCAN_AND_UPDATE_SUCCESS_NOT_CHANGED;
738   }
739
740   if (!write_changes) {
741     GST_INFO ("Registry cache changed, but writing is disabled. Not writing.");
742     return REGISTRY_SCAN_AND_UPDATE_FAILURE;
743   }
744
745   GST_INFO ("Registry cache changed. Writing new registry cache");
746 #ifdef USE_BINARY_REGISTRY
747   if (!gst_registry_binary_write_cache (default_registry, registry_file)) {
748 #else
749   if (!gst_registry_xml_write_cache (default_registry, registry_file)) {
750 #endif
751     g_set_error (error, GST_CORE_ERROR, GST_CORE_ERROR_FAILED,
752         _("Error writing registry cache to %s: %s"),
753         registry_file, g_strerror (errno));
754     return REGISTRY_SCAN_AND_UPDATE_FAILURE;
755   }
756
757   GST_INFO ("Registry cache written successfully");
758   return REGISTRY_SCAN_AND_UPDATE_SUCCESS_UPDATED;
759 }
760
761 static gboolean
762 ensure_current_registry_nonforking (GstRegistry * default_registry,
763     const gchar * registry_file, GError ** error)
764 {
765   /* fork() not available */
766   GST_DEBUG ("Updating registry cache in-process");
767   scan_and_update_registry (default_registry, registry_file, TRUE, error);
768   return TRUE;
769 }
770
771 /* when forking is not available this function always does nothing but return
772  * TRUE immediatly */
773 static gboolean
774 ensure_current_registry_forking (GstRegistry * default_registry,
775     const gchar * registry_file, GError ** error)
776 {
777 #ifdef HAVE_FORK
778   pid_t pid;
779   int pfd[2];
780   int ret;
781
782   /* We fork here, and let the child read and possibly rebuild the registry.
783    * After that, the parent will re-read the freshly generated registry. */
784   GST_DEBUG ("forking to update registry");
785
786   if (pipe (pfd) == -1) {
787     g_set_error (error, GST_CORE_ERROR, GST_CORE_ERROR_FAILED,
788         _("Error re-scanning registry %s: %s"),
789         ", could not create pipes. Error", g_strerror (errno));
790     return FALSE;
791   }
792
793   pid = fork ();
794   if (pid == -1) {
795     GST_ERROR ("Failed to fork()");
796     g_set_error (error, GST_CORE_ERROR, GST_CORE_ERROR_FAILED,
797         _("Error re-scanning registry %s: %s"),
798         ", failed to fork. Error", g_strerror (errno));
799     return FALSE;
800   }
801
802   if (pid == 0) {
803     gint result_code;
804
805     /* this is the child. Close the read pipe */
806     (void) close (pfd[0]);
807
808     GST_DEBUG ("child reading registry cache");
809     result_code =
810         scan_and_update_registry (default_registry, registry_file, TRUE, NULL);
811
812     /* need to use _exit, so that any exit handlers registered don't
813      * bring down the main program */
814     GST_DEBUG ("child exiting: %d", result_code);
815
816     /* make valgrind happy (yes, you can call it insane) */
817     g_free ((char *) registry_file);
818
819     /* write a result byte to the pipe */
820     do {
821       ret = write (pfd[1], &result_code, sizeof (result_code));
822     } while (ret == -1 && errno == EINTR);
823     /* if ret == -1 now, we could not write to pipe, probably 
824      * means parent has exited before us */
825     (void) close (pfd[1]);
826
827     _exit (0);
828   } else {
829     gint result_code;
830
831     /* parent. Close write pipe */
832     (void) close (pfd[1]);
833
834     /* Wait for result from the pipe */
835     GST_DEBUG ("Waiting for data from child");
836     do {
837       ret = read (pfd[0], &result_code, sizeof (result_code));
838     } while (ret == -1 && errno == EINTR);
839
840     if (ret == -1) {
841       g_set_error (error, GST_CORE_ERROR, GST_CORE_ERROR_FAILED,
842           _("Error re-scanning registry %s: %s"),
843           ", read returned error", g_strerror (errno));
844       close (pfd[0]);
845       return FALSE;
846     }
847     (void) close (pfd[0]);
848
849     /* Wait to ensure the child is reaped, but ignore the result */
850     GST_DEBUG ("parent waiting on child");
851     waitpid (pid, NULL, 0);
852     GST_DEBUG ("parent done waiting on child");
853
854     if (ret == 0) {
855       GST_ERROR ("child did not exit normally, terminated by signal");
856       g_set_error (error, GST_CORE_ERROR, GST_CORE_ERROR_FAILED,
857           _("Error re-scanning registry %s"), ", child terminated by signal");
858       return FALSE;
859     }
860
861     if (result_code == REGISTRY_SCAN_AND_UPDATE_SUCCESS_UPDATED) {
862       GST_DEBUG ("Child succeeded. Parent reading registry cache");
863       _priv_gst_registry_remove_cache_plugins (default_registry);
864 #ifdef USE_BINARY_REGISTRY
865       gst_registry_binary_read_cache (default_registry, registry_file);
866 #else
867       gst_registry_xml_read_cache (default_registry, registry_file);
868 #endif
869     } else if (result_code == REGISTRY_SCAN_AND_UPDATE_FAILURE) {
870       GST_DEBUG ("Child failed. Parent re-scanning registry, ignoring errors.");
871       scan_and_update_registry (default_registry, registry_file, FALSE, NULL);
872     }
873   }
874 #endif /* HAVE_FORK */
875   return TRUE;
876 }
877
878 static gboolean
879 ensure_current_registry (GError ** error)
880 {
881   gchar *registry_file;
882   GstRegistry *default_registry;
883   gboolean ret = TRUE;
884   gboolean do_fork;
885   gboolean do_update;
886
887   default_registry = gst_registry_get_default ();
888   registry_file = g_strdup (g_getenv ("GST_REGISTRY"));
889   if (registry_file == NULL) {
890 #ifdef USE_BINARY_REGISTRY
891     registry_file = g_build_filename (g_get_home_dir (),
892         ".gstreamer-" GST_MAJORMINOR, "registry." HOST_CPU ".bin", NULL);
893 #else
894     registry_file = g_build_filename (g_get_home_dir (),
895         ".gstreamer-" GST_MAJORMINOR, "registry." HOST_CPU ".xml", NULL);
896 #endif
897   }
898
899   GST_INFO ("reading registry cache: %s", registry_file);
900 #ifdef USE_BINARY_REGISTRY
901   gst_registry_binary_read_cache (default_registry, registry_file);
902 #else
903   gst_registry_xml_read_cache (default_registry, registry_file);
904 #endif
905
906   do_update = !_gst_disable_registry_update;
907   if (do_update) {
908     const gchar *update_env;
909
910     if ((update_env = g_getenv ("GST_REGISTRY_UPDATE"))) {
911       /* do update for any value different from "no" */
912       do_update = (strcmp (update_env, "no") != 0);
913     }
914   }
915
916   if (do_update) {
917     /* first see if forking is enabled */
918     do_fork = _gst_enable_registry_fork;
919     if (do_fork) {
920       const gchar *fork_env;
921
922       /* forking enabled, see if it is disabled with an env var */
923       if ((fork_env = g_getenv ("GST_REGISTRY_FORK"))) {
924         /* fork enabled for any value different from "no" */
925         do_fork = strcmp (fork_env, "no") != 0;
926       }
927     }
928
929     /* now check registry with or without forking */
930     if (do_fork) {
931       GST_DEBUG ("forking for registry rebuild");
932       ret = ensure_current_registry_forking (default_registry, registry_file,
933           error);
934     } else {
935       GST_DEBUG ("requested not to fork for registry rebuild");
936       ret = ensure_current_registry_nonforking (default_registry, registry_file,
937           error);
938     }
939   }
940
941   g_free (registry_file);
942   GST_INFO ("registry reading and updating done, result = %d", ret);
943
944   return ret;
945 }
946 #endif /* GST_DISABLE_REGISTRY */
947
948 /*
949  * this bit handles:
950  * - initalization of threads if we use them
951  * - log handler
952  * - initial output
953  * - initializes gst_format
954  * - registers a bunch of types for gst_objects
955  *
956  * - we don't have cases yet where this fails, but in the future
957  *   we might and then it's nice to be able to return that
958  */
959 static gboolean
960 init_post (GOptionContext * context, GOptionGroup * group, gpointer data,
961     GError ** error)
962 {
963   GLogLevelFlags llf;
964
965 #ifndef GST_DISABLE_TRACE
966   GstTrace *gst_trace;
967 #endif /* GST_DISABLE_TRACE */
968
969   if (gst_initialized) {
970     GST_DEBUG ("already initialized");
971     return TRUE;
972   }
973
974   llf = G_LOG_LEVEL_CRITICAL | G_LOG_LEVEL_ERROR | G_LOG_FLAG_FATAL;
975   g_log_set_handler (g_log_domain_gstreamer, llf, debug_log_handler, NULL);
976
977   _priv_gst_quarks_initialize ();
978   _gst_format_initialize ();
979   _gst_query_initialize ();
980   g_type_class_ref (gst_object_get_type ());
981   g_type_class_ref (gst_pad_get_type ());
982   g_type_class_ref (gst_element_factory_get_type ());
983   g_type_class_ref (gst_element_get_type ());
984   g_type_class_ref (gst_type_find_factory_get_type ());
985   g_type_class_ref (gst_bin_get_type ());
986   g_type_class_ref (gst_bus_get_type ());
987   g_type_class_ref (gst_task_get_type ());
988   g_type_class_ref (gst_clock_get_type ());
989
990 #ifndef GST_DISABLE_INDEX
991   g_type_class_ref (gst_index_factory_get_type ());
992 #endif /* GST_DISABLE_INDEX */
993 #ifndef GST_DISABLE_URI
994   gst_uri_handler_get_type ();
995 #endif /* GST_DISABLE_URI */
996
997 #ifndef GST_DISABLE_ENUMTYPES
998   g_type_class_ref (gst_object_flags_get_type ());
999   g_type_class_ref (gst_bin_flags_get_type ());
1000   g_type_class_ref (gst_buffer_flag_get_type ());
1001   g_type_class_ref (gst_buffer_copy_flags_get_type ());
1002   g_type_class_ref (gst_bus_flags_get_type ());
1003   g_type_class_ref (gst_bus_sync_reply_get_type ());
1004   g_type_class_ref (gst_caps_flags_get_type ());
1005   g_type_class_ref (gst_clock_return_get_type ());
1006   g_type_class_ref (gst_clock_entry_type_get_type ());
1007   g_type_class_ref (gst_clock_flags_get_type ());
1008   g_type_class_ref (gst_debug_graph_details_get_type ());
1009   g_type_class_ref (gst_state_get_type ());
1010   g_type_class_ref (gst_state_change_return_get_type ());
1011   g_type_class_ref (gst_state_change_get_type ());
1012   g_type_class_ref (gst_element_flags_get_type ());
1013   g_type_class_ref (gst_core_error_get_type ());
1014   g_type_class_ref (gst_library_error_get_type ());
1015   g_type_class_ref (gst_resource_error_get_type ());
1016   g_type_class_ref (gst_stream_error_get_type ());
1017   g_type_class_ref (gst_event_type_flags_get_type ());
1018   g_type_class_ref (gst_event_type_get_type ());
1019   g_type_class_ref (gst_seek_type_get_type ());
1020   g_type_class_ref (gst_seek_flags_get_type ());
1021   g_type_class_ref (gst_format_get_type ());
1022   g_type_class_ref (gst_index_certainty_get_type ());
1023   g_type_class_ref (gst_index_entry_type_get_type ());
1024   g_type_class_ref (gst_index_lookup_method_get_type ());
1025   g_type_class_ref (gst_assoc_flags_get_type ());
1026   g_type_class_ref (gst_index_resolver_method_get_type ());
1027   g_type_class_ref (gst_index_flags_get_type ());
1028   g_type_class_ref (gst_debug_level_get_type ());
1029   g_type_class_ref (gst_debug_color_flags_get_type ());
1030   g_type_class_ref (gst_iterator_result_get_type ());
1031   g_type_class_ref (gst_iterator_item_get_type ());
1032   g_type_class_ref (gst_message_type_get_type ());
1033   g_type_class_ref (gst_mini_object_flags_get_type ());
1034   g_type_class_ref (gst_pad_link_return_get_type ());
1035   g_type_class_ref (gst_flow_return_get_type ());
1036   g_type_class_ref (gst_activate_mode_get_type ());
1037   g_type_class_ref (gst_pad_direction_get_type ());
1038   g_type_class_ref (gst_pad_flags_get_type ());
1039   g_type_class_ref (gst_pad_presence_get_type ());
1040   g_type_class_ref (gst_pad_template_flags_get_type ());
1041   g_type_class_ref (gst_pipeline_flags_get_type ());
1042   g_type_class_ref (gst_plugin_error_get_type ());
1043   g_type_class_ref (gst_plugin_flags_get_type ());
1044   g_type_class_ref (gst_rank_get_type ());
1045   g_type_class_ref (gst_query_type_get_type ());
1046   g_type_class_ref (gst_buffering_mode_get_type ());
1047   g_type_class_ref (gst_tag_merge_mode_get_type ());
1048   g_type_class_ref (gst_tag_flag_get_type ());
1049   g_type_class_ref (gst_task_state_get_type ());
1050   g_type_class_ref (gst_alloc_trace_flags_get_type ());
1051   g_type_class_ref (gst_type_find_probability_get_type ());
1052   g_type_class_ref (gst_uri_type_get_type ());
1053   g_type_class_ref (gst_parse_error_get_type ());
1054   g_type_class_ref (gst_parse_flags_get_type ());
1055 #endif
1056
1057   gst_structure_get_type ();
1058   _gst_value_initialize ();
1059   g_type_class_ref (gst_param_spec_fraction_get_type ());
1060   gst_caps_get_type ();
1061   _gst_event_initialize ();
1062   _gst_buffer_initialize ();
1063   _gst_message_initialize ();
1064   _gst_tag_initialize ();
1065
1066   _gst_plugin_initialize ();
1067
1068   gst_g_error_get_type ();
1069
1070   /* register core plugins */
1071   gst_plugin_register_static (GST_VERSION_MAJOR, GST_VERSION_MINOR,
1072       "staticelements", "core elements linked into the GStreamer library",
1073       gst_register_core_elements, VERSION, GST_LICENSE, PACKAGE,
1074       GST_PACKAGE_NAME, GST_PACKAGE_ORIGIN);
1075
1076   /*
1077    * Any errors happening below this point are non-fatal, we therefore mark
1078    * gstreamer as being initialized, since it is the case from a plugin point of
1079    * view.
1080    *
1081    * If anything fails, it will be put back to FALSE in gst_init_check().
1082    * This allows some special plugins that would call gst_init() to not cause a
1083    * looping effect (i.e. initializing GStreamer twice).
1084    */
1085   gst_initialized = TRUE;
1086
1087 #ifndef GST_DISABLE_REGISTRY
1088   if (!ensure_current_registry (error))
1089     return FALSE;
1090 #endif /* GST_DISABLE_REGISTRY */
1091
1092   /* if we need to preload plugins, do so now */
1093   g_slist_foreach (preload_plugins, load_plugin_func, NULL);
1094   /* keep preload_plugins around in case a re-scan is forced later on */
1095
1096 #ifndef GST_DISABLE_TRACE
1097   _gst_trace_on = 0;
1098   if (_gst_trace_on) {
1099     gst_trace = gst_trace_new ("gst.trace", 1024);
1100     gst_trace_set_default (gst_trace);
1101   }
1102 #endif /* GST_DISABLE_TRACE */
1103
1104   return TRUE;
1105 }
1106
1107 #ifndef GST_DISABLE_GST_DEBUG
1108 static gboolean
1109 select_all (GstPlugin * plugin, gpointer user_data)
1110 {
1111   return TRUE;
1112 }
1113
1114 static gint
1115 sort_by_category_name (gconstpointer a, gconstpointer b)
1116 {
1117   return strcmp (gst_debug_category_get_name ((GstDebugCategory *) a),
1118       gst_debug_category_get_name ((GstDebugCategory *) b));
1119 }
1120
1121 static void
1122 gst_debug_help (void)
1123 {
1124   GSList *list, *walk;
1125   GList *list2, *g;
1126
1127   /* Need to ensure the registry is loaded to get debug categories */
1128   if (!init_post (NULL, NULL, NULL, NULL))
1129     exit (1);
1130
1131   list2 = gst_registry_plugin_filter (gst_registry_get_default (),
1132       select_all, FALSE, NULL);
1133
1134   /* FIXME this is gross.  why don't debug have categories PluginFeatures? */
1135   for (g = list2; g; g = g_list_next (g)) {
1136     GstPlugin *plugin = GST_PLUGIN_CAST (g->data);
1137
1138     gst_plugin_load (plugin);
1139   }
1140   g_list_free (list2);
1141
1142   list = gst_debug_get_all_categories ();
1143   walk = list = g_slist_sort (list, sort_by_category_name);
1144
1145   g_print ("\n");
1146   g_print ("name                  level    description\n");
1147   g_print ("---------------------+--------+--------------------------------\n");
1148
1149   while (walk) {
1150     GstDebugCategory *cat = (GstDebugCategory *) walk->data;
1151
1152     if (gst_debug_is_colored ()) {
1153       gchar *color = gst_debug_construct_term_color (cat->color);
1154
1155       g_print ("%s%-20s\033[00m  %1d %s  %s%s\033[00m\n",
1156           color,
1157           gst_debug_category_get_name (cat),
1158           gst_debug_category_get_threshold (cat),
1159           gst_debug_level_get_name (gst_debug_category_get_threshold (cat)),
1160           color, gst_debug_category_get_description (cat));
1161       g_free (color);
1162     } else {
1163       g_print ("%-20s  %1d %s  %s\n", gst_debug_category_get_name (cat),
1164           gst_debug_category_get_threshold (cat),
1165           gst_debug_level_get_name (gst_debug_category_get_threshold (cat)),
1166           gst_debug_category_get_description (cat));
1167     }
1168     walk = g_slist_next (walk);
1169   }
1170   g_slist_free (list);
1171   g_print ("\n");
1172 }
1173 #endif
1174
1175 #ifndef GST_DISABLE_OPTION_PARSING
1176 static gboolean
1177 parse_one_option (gint opt, const gchar * arg, GError ** err)
1178 {
1179   switch (opt) {
1180     case ARG_VERSION:
1181       g_print ("GStreamer Core Library version %s\n", PACKAGE_VERSION);
1182       exit (0);
1183     case ARG_FATAL_WARNINGS:{
1184       GLogLevelFlags fatal_mask;
1185
1186       fatal_mask = g_log_set_always_fatal (G_LOG_FATAL_MASK);
1187       fatal_mask |= G_LOG_LEVEL_WARNING | G_LOG_LEVEL_CRITICAL;
1188       g_log_set_always_fatal (fatal_mask);
1189       break;
1190     }
1191 #ifndef GST_DISABLE_GST_DEBUG
1192     case ARG_DEBUG_LEVEL:{
1193       gint tmp = 0;
1194
1195       tmp = strtol (arg, NULL, 0);
1196       if (tmp >= 0 && tmp < GST_LEVEL_COUNT) {
1197         gst_debug_set_default_threshold (tmp);
1198       }
1199       break;
1200     }
1201     case ARG_DEBUG:
1202       parse_debug_list (arg);
1203       break;
1204     case ARG_DEBUG_NO_COLOR:
1205       gst_debug_set_colored (FALSE);
1206       break;
1207     case ARG_DEBUG_DISABLE:
1208       gst_debug_set_active (FALSE);
1209       break;
1210     case ARG_DEBUG_HELP:
1211       gst_debug_help ();
1212       exit (0);
1213 #endif
1214     case ARG_PLUGIN_SPEW:
1215       break;
1216     case ARG_PLUGIN_PATH:
1217 #ifndef GST_DISABLE_REGISTRY
1218       split_and_iterate (arg, G_SEARCHPATH_SEPARATOR_S, add_path_func, NULL);
1219 #endif /* GST_DISABLE_REGISTRY */
1220       break;
1221     case ARG_PLUGIN_LOAD:
1222       split_and_iterate (arg, ",", prepare_for_load_plugin_func, NULL);
1223       break;
1224     case ARG_SEGTRAP_DISABLE:
1225       _gst_disable_segtrap = TRUE;
1226       break;
1227     case ARG_REGISTRY_UPDATE_DISABLE:
1228       _gst_disable_registry_update = TRUE;
1229       break;
1230     case ARG_REGISTRY_FORK_DISABLE:
1231       _gst_enable_registry_fork = FALSE;
1232       break;
1233     default:
1234       g_set_error (err, G_OPTION_ERROR, G_OPTION_ERROR_UNKNOWN_OPTION,
1235           _("Unknown option"));
1236       return FALSE;
1237   }
1238
1239   return TRUE;
1240 }
1241
1242 static gboolean
1243 parse_goption_arg (const gchar * opt,
1244     const gchar * arg, gpointer data, GError ** err)
1245 {
1246   static const struct
1247   {
1248     gchar *opt;
1249     int val;
1250   } options[] = {
1251     {
1252     "--gst-version", ARG_VERSION}, {
1253     "--gst-fatal-warnings", ARG_FATAL_WARNINGS},
1254 #ifndef GST_DISABLE_GST_DEBUG
1255     {
1256     "--gst-debug-level", ARG_DEBUG_LEVEL}, {
1257     "--gst-debug", ARG_DEBUG}, {
1258     "--gst-debug-disable", ARG_DEBUG_DISABLE}, {
1259     "--gst-debug-no-color", ARG_DEBUG_NO_COLOR}, {
1260     "--gst-debug-help", ARG_DEBUG_HELP},
1261 #endif
1262     {
1263     "--gst-plugin-spew", ARG_PLUGIN_SPEW}, {
1264     "--gst-plugin-path", ARG_PLUGIN_PATH}, {
1265     "--gst-plugin-load", ARG_PLUGIN_LOAD}, {
1266     "--gst-disable-segtrap", ARG_SEGTRAP_DISABLE}, {
1267     "--gst-disable-registry-update", ARG_REGISTRY_UPDATE_DISABLE}, {
1268     "--gst-disable-registry-fork", ARG_REGISTRY_FORK_DISABLE}, {
1269     NULL}
1270   };
1271   gint val = 0, n;
1272
1273   for (n = 0; options[n].opt; n++) {
1274     if (!strcmp (opt, options[n].opt)) {
1275       val = options[n].val;
1276       break;
1277     }
1278   }
1279
1280   return parse_one_option (val, arg, err);
1281 }
1282 #endif
1283
1284 extern GstRegistry *_gst_registry_default;
1285
1286 /**
1287  * gst_deinit:
1288  *
1289  * Clean up any resources created by GStreamer in gst_init().
1290  *
1291  * It is normally not needed to call this function in a normal application
1292  * as the resources will automatically be freed when the program terminates.
1293  * This function is therefore mostly used by testsuites and other memory
1294  * profiling tools.
1295  *
1296  * After this call GStreamer (including this method) should not be used anymore. 
1297  */
1298 void
1299 gst_deinit (void)
1300 {
1301   GstClock *clock;
1302
1303   GST_INFO ("deinitializing GStreamer");
1304
1305   if (gst_deinitialized) {
1306     GST_DEBUG ("already deinitialized");
1307     return;
1308   }
1309
1310   g_slist_foreach (preload_plugins, (GFunc) g_free, NULL);
1311   g_slist_free (preload_plugins);
1312   preload_plugins = NULL;
1313
1314 #ifndef GST_DISABLE_REGISTRY
1315   g_list_foreach (plugin_paths, (GFunc) g_free, NULL);
1316   g_list_free (plugin_paths);
1317   plugin_paths = NULL;
1318 #endif
1319
1320   clock = gst_system_clock_obtain ();
1321   gst_object_unref (clock);
1322   gst_object_unref (clock);
1323
1324   _priv_gst_registry_cleanup ();
1325
1326   g_type_class_unref (g_type_class_peek (gst_object_get_type ()));
1327   g_type_class_unref (g_type_class_peek (gst_pad_get_type ()));
1328   g_type_class_unref (g_type_class_peek (gst_element_factory_get_type ()));
1329   g_type_class_unref (g_type_class_peek (gst_element_get_type ()));
1330   g_type_class_unref (g_type_class_peek (gst_type_find_factory_get_type ()));
1331   g_type_class_unref (g_type_class_peek (gst_bin_get_type ()));
1332   g_type_class_unref (g_type_class_peek (gst_bus_get_type ()));
1333   g_type_class_unref (g_type_class_peek (gst_task_get_type ()));
1334 #ifndef GST_DISABLE_INDEX
1335   g_type_class_unref (g_type_class_peek (gst_index_factory_get_type ()));
1336 #endif /* GST_DISABLE_INDEX */
1337 #ifndef GST_DISABLE_ENUMTYPES
1338   g_type_class_unref (g_type_class_peek (gst_object_flags_get_type ()));
1339   g_type_class_unref (g_type_class_peek (gst_bin_flags_get_type ()));
1340   g_type_class_unref (g_type_class_peek (gst_buffer_flag_get_type ()));
1341   g_type_class_unref (g_type_class_peek (gst_buffer_copy_flags_get_type ()));
1342   g_type_class_unref (g_type_class_peek (gst_bus_flags_get_type ()));
1343   g_type_class_unref (g_type_class_peek (gst_bus_sync_reply_get_type ()));
1344   g_type_class_unref (g_type_class_peek (gst_caps_flags_get_type ()));
1345   g_type_class_unref (g_type_class_peek (gst_clock_return_get_type ()));
1346   g_type_class_unref (g_type_class_peek (gst_clock_entry_type_get_type ()));
1347   g_type_class_unref (g_type_class_peek (gst_clock_flags_get_type ()));
1348   g_type_class_unref (g_type_class_peek (gst_debug_graph_details_get_type ()));
1349   g_type_class_unref (g_type_class_peek (gst_state_get_type ()));
1350   g_type_class_unref (g_type_class_peek (gst_state_change_return_get_type ()));
1351   g_type_class_unref (g_type_class_peek (gst_state_change_get_type ()));
1352   g_type_class_unref (g_type_class_peek (gst_element_flags_get_type ()));
1353   g_type_class_unref (g_type_class_peek (gst_core_error_get_type ()));
1354   g_type_class_unref (g_type_class_peek (gst_library_error_get_type ()));
1355   g_type_class_unref (g_type_class_peek (gst_resource_error_get_type ()));
1356   g_type_class_unref (g_type_class_peek (gst_stream_error_get_type ()));
1357   g_type_class_unref (g_type_class_peek (gst_event_type_flags_get_type ()));
1358   g_type_class_unref (g_type_class_peek (gst_event_type_get_type ()));
1359   g_type_class_unref (g_type_class_peek (gst_seek_type_get_type ()));
1360   g_type_class_unref (g_type_class_peek (gst_seek_flags_get_type ()));
1361   g_type_class_unref (g_type_class_peek (gst_format_get_type ()));
1362   g_type_class_unref (g_type_class_peek (gst_index_certainty_get_type ()));
1363   g_type_class_unref (g_type_class_peek (gst_index_entry_type_get_type ()));
1364   g_type_class_unref (g_type_class_peek (gst_index_lookup_method_get_type ()));
1365   g_type_class_unref (g_type_class_peek (gst_assoc_flags_get_type ()));
1366   g_type_class_unref (g_type_class_peek (gst_index_resolver_method_get_type
1367           ()));
1368   g_type_class_unref (g_type_class_peek (gst_index_flags_get_type ()));
1369   g_type_class_unref (g_type_class_peek (gst_debug_level_get_type ()));
1370   g_type_class_unref (g_type_class_peek (gst_debug_color_flags_get_type ()));
1371   g_type_class_unref (g_type_class_peek (gst_iterator_result_get_type ()));
1372   g_type_class_unref (g_type_class_peek (gst_iterator_item_get_type ()));
1373   g_type_class_unref (g_type_class_peek (gst_message_type_get_type ()));
1374   g_type_class_unref (g_type_class_peek (gst_mini_object_flags_get_type ()));
1375   g_type_class_unref (g_type_class_peek (gst_pad_link_return_get_type ()));
1376   g_type_class_unref (g_type_class_peek (gst_flow_return_get_type ()));
1377   g_type_class_unref (g_type_class_peek (gst_activate_mode_get_type ()));
1378   g_type_class_unref (g_type_class_peek (gst_pad_direction_get_type ()));
1379   g_type_class_unref (g_type_class_peek (gst_pad_flags_get_type ()));
1380   g_type_class_unref (g_type_class_peek (gst_pad_presence_get_type ()));
1381   g_type_class_unref (g_type_class_peek (gst_pad_template_flags_get_type ()));
1382   g_type_class_unref (g_type_class_peek (gst_pipeline_flags_get_type ()));
1383   g_type_class_unref (g_type_class_peek (gst_plugin_error_get_type ()));
1384   g_type_class_unref (g_type_class_peek (gst_plugin_flags_get_type ()));
1385   g_type_class_unref (g_type_class_peek (gst_rank_get_type ()));
1386   g_type_class_unref (g_type_class_peek (gst_query_type_get_type ()));
1387   g_type_class_unref (g_type_class_peek (gst_buffering_mode_get_type ()));
1388   g_type_class_unref (g_type_class_peek (gst_tag_merge_mode_get_type ()));
1389   g_type_class_unref (g_type_class_peek (gst_tag_flag_get_type ()));
1390   g_type_class_unref (g_type_class_peek (gst_task_state_get_type ()));
1391   g_type_class_unref (g_type_class_peek (gst_alloc_trace_flags_get_type ()));
1392   g_type_class_unref (g_type_class_peek (gst_type_find_probability_get_type
1393           ()));
1394   g_type_class_unref (g_type_class_peek (gst_uri_type_get_type ()));
1395   g_type_class_unref (g_type_class_peek (gst_parse_error_get_type ()));
1396 #endif
1397   g_type_class_unref (g_type_class_peek (gst_param_spec_fraction_get_type ()));
1398
1399   gst_deinitialized = TRUE;
1400   GST_INFO ("deinitialized GStreamer");
1401 }
1402
1403 /**
1404  * gst_version:
1405  * @major: pointer to a guint to store the major version number
1406  * @minor: pointer to a guint to store the minor version number
1407  * @micro: pointer to a guint to store the micro version number
1408  * @nano:  pointer to a guint to store the nano version number
1409  *
1410  * Gets the version number of the GStreamer library.
1411  */
1412 void
1413 gst_version (guint * major, guint * minor, guint * micro, guint * nano)
1414 {
1415   g_return_if_fail (major);
1416   g_return_if_fail (minor);
1417   g_return_if_fail (micro);
1418   g_return_if_fail (nano);
1419
1420   *major = GST_VERSION_MAJOR;
1421   *minor = GST_VERSION_MINOR;
1422   *micro = GST_VERSION_MICRO;
1423   *nano = GST_VERSION_NANO;
1424 }
1425
1426 /**
1427  * gst_version_string:
1428  *
1429  * This function returns a string that is useful for describing this version
1430  * of GStreamer to the outside world: user agent strings, logging, ...
1431  *
1432  * Returns: a newly allocated string describing this version of GStreamer.
1433  */
1434
1435 gchar *
1436 gst_version_string ()
1437 {
1438   guint major, minor, micro, nano;
1439
1440   gst_version (&major, &minor, &micro, &nano);
1441   if (nano == 0)
1442     return g_strdup_printf ("GStreamer %d.%d.%d", major, minor, micro);
1443   else if (nano == 1)
1444     return g_strdup_printf ("GStreamer %d.%d.%d (CVS)", major, minor, micro);
1445   else
1446     return g_strdup_printf ("GStreamer %d.%d.%d (prerelease)", major, minor,
1447         micro);
1448 }
1449
1450 /**
1451  * gst_segtrap_is_enabled:
1452  *
1453  * Some functions in the GStreamer core might install a custom SIGSEGV handler
1454  * to better catch and report errors to the application. Currently this feature
1455  * is enabled by default when loading plugins.
1456  *
1457  * Applications might want to disable this behaviour with the
1458  * gst_segtrap_set_enabled() function. This is typically done if the application
1459  * wants to install its own handler without GStreamer interfering.
1460  *
1461  * Returns: %TRUE if GStreamer is allowed to install a custom SIGSEGV handler.
1462  *
1463  * Since: 0.10.10
1464  */
1465 gboolean
1466 gst_segtrap_is_enabled (void)
1467 {
1468   /* yeps, it's enabled when it's not disabled */
1469   return !_gst_disable_segtrap;
1470 }
1471
1472 /**
1473  * gst_segtrap_set_enabled:
1474  * @enabled: whether a custom SIGSEGV handler should be installed.
1475  *
1476  * Applications might want to disable/enable the SIGSEGV handling of
1477  * the GStreamer core. See gst_segtrap_is_enabled() for more information.
1478  *
1479  * Since: 0.10.10
1480  */
1481 void
1482 gst_segtrap_set_enabled (gboolean enabled)
1483 {
1484   _gst_disable_segtrap = !enabled;
1485 }
1486
1487 /**
1488  * gst_registry_fork_is_enabled:
1489  *
1490  * By default GStreamer will perform a fork() when scanning and rebuilding the
1491  * registry file. 
1492  *
1493  * Applications might want to disable this behaviour with the
1494  * gst_registry_fork_set_enabled() function. 
1495  *
1496  * Returns: %TRUE if GStreamer will use fork() when rebuilding the registry. On
1497  * platforms without fork(), this function will always return %FALSE.
1498  *
1499  * Since: 0.10.10
1500  */
1501 gboolean
1502 gst_registry_fork_is_enabled (void)
1503 {
1504   return _gst_enable_registry_fork;
1505 }
1506
1507 /**
1508  * gst_registry_fork_set_enabled:
1509  * @enabled: whether rebuilding the registry may fork
1510  *
1511  * Applications might want to disable/enable the usage of fork() when rebuilding
1512  * the registry. See gst_registry_fork_is_enabled() for more information.
1513  *
1514  * On platforms without fork(), this function will have no effect on the return
1515  * value of gst_registry_fork_is_enabled().
1516  *
1517  * Since: 0.10.10
1518  */
1519 void
1520 gst_registry_fork_set_enabled (gboolean enabled)
1521 {
1522 #ifdef HAVE_FORK
1523   _gst_enable_registry_fork = enabled;
1524 #endif /* HAVE_FORK */
1525 }
1526
1527
1528 /**
1529  * gst_update_registry:
1530  *
1531  * Forces GStreamer to re-scan its plugin paths and update the default
1532  * plugin registry.
1533  *
1534  * Applications will almost never need to call this function, it is only
1535  * useful if the application knows new plugins have been installed (or old
1536  * ones removed) since the start of the application (or, to be precise, the
1537  * first call to gst_init()) and the application wants to make use of any
1538  * newly-installed plugins without restarting the application.
1539  *
1540  * Applications should assume that the registry update is neither atomic nor
1541  * thread-safe and should therefore not have any dynamic pipelines running
1542  * (including the playbin and decodebin elements) and should also not create
1543  * any elements or access the GStreamer registry while the update is in
1544  * progress.
1545  *
1546  * Note that this function may block for a significant amount of time.
1547  *
1548  * Returns: %TRUE if the registry has been updated successfully (does not
1549  *          imply that there were changes), otherwise %FALSE.
1550  *
1551  * Since: 0.10.12
1552  */
1553 gboolean
1554 gst_update_registry (void)
1555 {
1556   gboolean res = FALSE;
1557
1558 #ifndef GST_DISABLE_REGISTRY
1559   GError *err = NULL;
1560
1561   res = ensure_current_registry (&err);
1562   if (err) {
1563     GST_WARNING ("registry update failed: %s", err->message);
1564     g_error_free (err);
1565   } else {
1566     GST_LOG ("registry update succeeded");
1567   }
1568
1569   if (preload_plugins) {
1570     g_slist_foreach (preload_plugins, load_plugin_func, NULL);
1571   }
1572 #else
1573   GST_WARNING ("registry update failed: %s", "registry disabled");
1574 #endif /* GST_DISABLE_REGISTRY */
1575
1576   return res;
1577 }