g_thread_init() is deprecated in glib master
[platform/upstream/gstreamer.git] / gst / gst.c
index 7f71567..ff7f250 100644 (file)
--- a/gst/gst.c
+++ b/gst/gst.c
@@ -24,8 +24,6 @@
  * SECTION:gst
  * @short_description: Media library supporting arbitrary formats and filter
  *                     graphs.
- * @see_also: Check out both <ulink url="http://www.cse.ogi.edu/sysl/">OGI's
- *            pipeline</ulink> and Microsoft's DirectShow for some background.
  *
  * GStreamer is a framework for constructing graphs of various filters
  * (termed elements here) that will handle streaming media.  Any discreet
  * a powerful negotiation framework.  Plugins are heavily used to provide for
  * all elements, allowing one to construct plugins outside of the GST
  * library, even released binary-only if license require (please don't).
- *
- * GStreamer borrows heavily from both the <ulink
- * url="http://www.cse.ogi.edu/sysl/">OGI media pipeline</ulink> and
- * Microsoft's DirectShow, hopefully taking the best of both and leaving the
- * cruft behind.  Its interface is still very fluid and thus can be changed
- * to increase the sanity/noise ratio.
+ * GStreamer covers a wide range of use cases including: playback, recording,
+ * editing, serving streams, voice over ip and video calls.
  *
  * The <application>GStreamer</application> library should be initialized with
  * gst_init() before it can be used. You should pass pointers to the main argc
  *       N_("Output tags (also known as metadata)"), NULL},
  *   {NULL}
  *  };
- *  ctx = g_option_context_new ("gst-launch");
+ *  // must initialise the threading system before using any other GLib funtion
+ *  if (!g_thread_supported ())
+ *    g_thread_init (NULL);
+ *  ctx = g_option_context_new ("[ADDITIONAL ARGUMENTS]");
  *  g_option_context_add_main_entries (ctx, options, GETTEXT_PACKAGE);
  *  g_option_context_add_group (ctx, gst_init_get_option_group ());
  *  if (!g_option_context_parse (ctx, &amp;argc, &amp;argv, &amp;err)) {
  * by <application>GStreamer</application>. It is mostly used in unit tests 
  * to check for leaks.
  *
- * Last reviewed on 2005-11-23 (0.9.5)
+ * Last reviewed on 2006-08-11 (0.10.10)
  */
 
 #include "gst_private.h"
+#include "gstconfig.h"
 #include <stdlib.h>
 #include <stdio.h>
 #include <sys/types.h>
-#ifdef HAVE_FORK
-#include <sys/wait.h>
-#endif //HAVE_FORK
+#ifdef HAVE_SYS_UTSNAME_H
+#include <sys/utsname.h>
+#endif
+#ifdef HAVE_UNISTD_H
 #include <unistd.h>
+#endif
+#ifdef G_OS_WIN32
+#define WIN32_LEAN_AND_MEAN     /* prevents from including too many things */
+#include <windows.h>            /* GetStdHandle, windows console */
+#endif
 
 #include "gst-i18n-lib.h"
 #include <locale.h>             /* for LC_ALL */
 #define GST_PLUGIN_SEPARATOR ","
 
 static gboolean gst_initialized = FALSE;
+static gboolean gst_deinitialized = FALSE;
+
+#ifdef G_OS_WIN32
+HMODULE _priv_gst_dll_handle = NULL;
+#endif
 
 #ifndef GST_DISABLE_REGISTRY
-static GList *plugin_paths = NULL;      /* for delayed processing in post_init */
+GList *_priv_gst_plugin_paths = NULL;   /* for delayed processing in post_init */
+
+extern gboolean _priv_gst_disable_registry_update;
+#endif
+
+#ifndef GST_DISABLE_GST_DEBUG
+extern const gchar *priv_gst_dump_dot_dir;
 #endif
 
-extern gint _gst_trace_on;
+/* defaults */
 
 /* set to TRUE when segfaults need to be left as is */
-gboolean _gst_disable_segtrap = FALSE;
+static gboolean _gst_disable_segtrap = FALSE;
 
-static void load_plugin_func (gpointer data, gpointer user_data);
-static gboolean init_pre (void);
-static gboolean init_post (void);
+static gboolean init_pre (GOptionContext * context, GOptionGroup * group,
+    gpointer data, GError ** error);
+static gboolean init_post (GOptionContext * context, GOptionGroup * group,
+    gpointer data, GError ** error);
+#ifndef GST_DISABLE_OPTION_PARSING
 static gboolean parse_goption_arg (const gchar * s_opt,
     const gchar * arg, gpointer data, GError ** err);
+#endif
 
-static GSList *preload_plugins = NULL;
+GSList *_priv_gst_preload_plugins = NULL;
 
 const gchar g_log_domain_gstreamer[] = "GStreamer";
 
@@ -165,7 +183,9 @@ enum
   ARG_PLUGIN_SPEW,
   ARG_PLUGIN_PATH,
   ARG_PLUGIN_LOAD,
-  ARG_SEGTRAP_DISABLE
+  ARG_SEGTRAP_DISABLE,
+  ARG_REGISTRY_UPDATE_DISABLE,
+  ARG_REGISTRY_FORK_DISABLE
 };
 
 /* debug-spec ::= category-spec [, category-spec]*
@@ -197,7 +217,7 @@ parse_debug_category (gchar * str, const gchar ** category)
 }
 
 static gboolean
-parse_debug_level (gchar * str, gint * level)
+parse_debug_level (gchar * str, GstDebugLevel * level)
 {
   if (!str)
     return FALSE;
@@ -207,7 +227,7 @@ parse_debug_level (gchar * str, gint * level)
 
   if (str[0] != NUL && str[1] == NUL
       && str[0] >= '0' && str[0] < '0' + GST_LEVEL_COUNT) {
-    *level = str[0] - '0';
+    *level = (GstDebugLevel) (str[0] - '0');
     return TRUE;
   }
 
@@ -220,7 +240,7 @@ parse_debug_list (const gchar * list)
   gchar **split;
   gchar **walk;
 
-  g_return_if_fail (list != NULL);
+  g_assert (list);
 
   split = g_strsplit (list, ",", 0);
 
@@ -229,7 +249,7 @@ parse_debug_list (const gchar * list)
       gchar **values = g_strsplit (*walk, ":", 2);
 
       if (values[0] && values[1]) {
-        gint level;
+        GstDebugLevel level;
         const gchar *category;
 
         if (parse_debug_category (values[0], &category)
@@ -239,7 +259,7 @@ parse_debug_list (const gchar * list)
 
       g_strfreev (values);
     } else {
-      gint level;
+      GstDebugLevel level;
 
       if (parse_debug_level (*walk, &level))
         gst_debug_set_default_threshold (level);
@@ -250,6 +270,18 @@ parse_debug_list (const gchar * list)
 }
 #endif
 
+#ifdef G_OS_WIN32
+BOOL WINAPI DllMain (HINSTANCE hinstDLL, DWORD fdwReason, LPVOID lpvReserved);
+BOOL WINAPI
+DllMain (HINSTANCE hinstDLL, DWORD fdwReason, LPVOID lpvReserved)
+{
+  if (fdwReason == DLL_PROCESS_ATTACH)
+    _priv_gst_dll_handle = (HMODULE) hinstDLL;
+  return TRUE;
+}
+
+#endif
+
 /**
  * gst_init_get_option_group:
  *
@@ -261,53 +293,99 @@ parse_debug_list (const gchar * list)
  * This function is useful if you want to integrate GStreamer with other
  * libraries that use GOption (see g_option_context_add_group() ).
  *
- * Returns: a pointer to GStreamer's option group. Should be dereferenced
- * after use.
+ * If you use this function, you should make sure you initialise the GLib
+ * threading system as one of the very first things in your program
+ * (see the example at the beginning of this section).
+ *
+ * Returns: (transfer full): a pointer to GStreamer's option group.
  */
 
 GOptionGroup *
 gst_init_get_option_group (void)
 {
+#ifndef GST_DISABLE_OPTION_PARSING
   GOptionGroup *group;
-  static GOptionEntry gst_args[] = {
+  static const GOptionEntry gst_args[] = {
     {"gst-version", 0, G_OPTION_FLAG_NO_ARG, G_OPTION_ARG_CALLBACK,
-        parse_goption_arg, N_("Print the GStreamer version"), NULL},
+        (gpointer) parse_goption_arg, N_("Print the GStreamer version"), NULL},
     {"gst-fatal-warnings", 0, G_OPTION_FLAG_NO_ARG, G_OPTION_ARG_CALLBACK,
-        parse_goption_arg, N_("Make all warnings fatal"), NULL},
+        (gpointer) parse_goption_arg, N_("Make all warnings fatal"), NULL},
 #ifndef GST_DISABLE_GST_DEBUG
     {"gst-debug-help", 0, G_OPTION_FLAG_NO_ARG, G_OPTION_ARG_CALLBACK,
-          parse_goption_arg, N_("Print available debug categories and exit"),
+          (gpointer) parse_goption_arg,
+          N_("Print available debug categories and exit"),
         NULL},
-    {"gst-debug-level", 0, 0, G_OPTION_ARG_CALLBACK, parse_goption_arg,
+    {"gst-debug-level", 0, 0, G_OPTION_ARG_CALLBACK,
+          (gpointer) parse_goption_arg,
           N_("Default debug level from 1 (only error) to 5 (anything) or "
               "0 for no output"),
         N_("LEVEL")},
-    {"gst-debug", 0, 0, G_OPTION_ARG_CALLBACK, parse_goption_arg,
+    {"gst-debug", 0, 0, G_OPTION_ARG_CALLBACK, (gpointer) parse_goption_arg,
           N_("Comma-separated list of category_name:level pairs to set "
               "specific levels for the individual categories. Example: "
               "GST_AUTOPLUG:5,GST_ELEMENT_*:3"),
         N_("LIST")},
     {"gst-debug-no-color", 0, G_OPTION_FLAG_NO_ARG, G_OPTION_ARG_CALLBACK,
-        parse_goption_arg, N_("Disable colored debugging output"), NULL},
+          (gpointer) parse_goption_arg, N_("Disable colored debugging output"),
+        NULL},
     {"gst-debug-disable", 0, G_OPTION_FLAG_NO_ARG, G_OPTION_ARG_CALLBACK,
-        parse_goption_arg, N_("Disable debugging"), NULL},
+        (gpointer) parse_goption_arg, N_("Disable debugging"), NULL},
 #endif
     {"gst-plugin-spew", 0, G_OPTION_FLAG_NO_ARG, G_OPTION_ARG_CALLBACK,
-          parse_goption_arg, N_("Enable verbose plugin loading diagnostics"),
+          (gpointer) parse_goption_arg,
+          N_("Enable verbose plugin loading diagnostics"),
         NULL},
-    {"gst-plugin-path", 0, 0, G_OPTION_ARG_CALLBACK, parse_goption_arg,
+    {"gst-plugin-path", 0, 0, G_OPTION_ARG_CALLBACK,
+          (gpointer) parse_goption_arg,
         N_("Colon-separated paths containing plugins"), N_("PATHS")},
-    {"gst-plugin-load", 0, 0, G_OPTION_ARG_CALLBACK, parse_goption_arg,
+    {"gst-plugin-load", 0, 0, G_OPTION_ARG_CALLBACK,
+          (gpointer) parse_goption_arg,
           N_("Comma-separated list of plugins to preload in addition to the "
               "list stored in environment variable GST_PLUGIN_PATH"),
         N_("PLUGINS")},
     {"gst-disable-segtrap", 0, G_OPTION_FLAG_NO_ARG, G_OPTION_ARG_CALLBACK,
-          parse_goption_arg,
+          (gpointer) parse_goption_arg,
           N_("Disable trapping of segmentation faults during plugin loading"),
         NULL},
+    {"gst-disable-registry-update", 0, G_OPTION_FLAG_NO_ARG,
+          G_OPTION_ARG_CALLBACK,
+          (gpointer) parse_goption_arg,
+          N_("Disable updating the registry"),
+        NULL},
+    {"gst-disable-registry-fork", 0, G_OPTION_FLAG_NO_ARG,
+          G_OPTION_ARG_CALLBACK,
+          (gpointer) parse_goption_arg,
+          N_("Disable spawning a helper process while scanning the registry"),
+        NULL},
     {NULL}
   };
 
+/* Since GLib 2.31.0 threading is always enabled and g_thread_init()
+ * is not needed any longer and deprecated */
+#if !GLIB_CHECK_VERSION (2, 31, 0)
+  /* Since GLib 2.23.2 calling g_thread_init() 'late' is allowed and is
+   * automatically done as part of g_type_init() */
+  if (glib_check_version (2, 23, 3)) {
+    /* The GLib threading system must be initialised before calling any other
+     * GLib function according to the documentation; if the application hasn't
+     * called gst_init() yet or initialised the threading system otherwise, we
+     * better issue a warning here (since chances are high that the application
+     * has already called other GLib functions such as g_option_context_new() */
+    if (!g_thread_get_initialized ()) {
+      g_warning ("The GStreamer function gst_init_get_option_group() was\n"
+          "\tcalled, but the GLib threading system has not been initialised\n"
+          "\tyet, something that must happen before any other GLib function\n"
+          "\tis called. The application needs to be fixed so that it calls\n"
+          "\t   if (!g_thread_get_initialized ()) g_thread_init(NULL);\n"
+          "\tas very first thing in its main() function. Please file a bug\n"
+          "\tagainst this application.");
+      g_thread_init (NULL);
+    }
+  } else {
+    /* GLib >= 2.23.2 */
+  }
+#endif
+
   group = g_option_group_new ("gst", _("GStreamer Options"),
       _("Show GStreamer Options"), NULL, NULL);
   g_option_group_set_parse_hooks (group, (GOptionParseFunc) init_pre,
@@ -317,12 +395,15 @@ gst_init_get_option_group (void)
   g_option_group_set_translation_domain (group, GETTEXT_PACKAGE);
 
   return group;
+#else
+  return NULL;
+#endif
 }
 
 /**
  * gst_init_check:
- * @argc: pointer to application's argc
- * @argv: pointer to application's argv
+ * @argc: (inout) (allow-none): pointer to application's argc
+ * @argv: (inout) (array length=argc) (allow-none): pointer to application's argv
  * @err: pointer to a #GError to which a message will be posted on error
  *
  * Initializes the GStreamer library, setting up internal path lists,
@@ -332,29 +413,49 @@ gst_init_get_option_group (void)
  * for some reason.  If you want your program to fail fatally,
  * use gst_init() instead.
  *
+ * This function should be called before calling any other GLib functions. If
+ * this is not an option, your program must initialise the GLib thread system
+ * using g_thread_init() before any other GLib functions are called.
+ *
  * Returns: %TRUE if GStreamer could be initialized.
  */
 gboolean
 gst_init_check (int *argc, char **argv[], GError ** err)
 {
+#ifndef GST_DISABLE_OPTION_PARSING
   GOptionGroup *group;
   GOptionContext *ctx;
+#endif
   gboolean res;
 
+#if !GLIB_CHECK_VERSION (2, 31, 0)
+  if (!g_thread_get_initialized ())
+    g_thread_init (NULL);
+#endif
+
   if (gst_initialized) {
     GST_DEBUG ("already initialized gst");
     return TRUE;
   }
-
+#ifndef GST_DISABLE_OPTION_PARSING
   ctx = g_option_context_new ("- GStreamer initialization");
   g_option_context_set_ignore_unknown_options (ctx, TRUE);
   group = gst_init_get_option_group ();
   g_option_context_add_group (ctx, group);
   res = g_option_context_parse (ctx, argc, argv, err);
   g_option_context_free (ctx);
+#else
+  init_pre (NULL, NULL, NULL, NULL);
+  init_post (NULL, NULL, NULL, NULL);
+  res = TRUE;
+#endif
+
+  gst_initialized = res;
 
   if (res) {
-    gst_initialized = TRUE;
+    GST_INFO ("initialized GStreamer successfully");
+  } else {
+    GST_INFO ("failed to initialize GStreamer");
   }
 
   return res;
@@ -362,12 +463,23 @@ gst_init_check (int *argc, char **argv[], GError ** err)
 
 /**
  * gst_init:
- * @argc: pointer to application's argc
- * @argv: pointer to application's argv
+ * @argc: (inout) (allow-none): pointer to application's argc
+ * @argv: (inout) (array length=argc) (allow-none): pointer to application's argv
  *
  * Initializes the GStreamer library, setting up internal path lists,
  * registering built-in elements, and loading standard plugins.
  *
+ * Unless the plugin registry is disabled at compile time, the registry will be
+ * loaded. By default this will also check if the registry cache needs to be
+ * updated and rescan all plugins if needed. See gst_update_registry() for
+ * details and section
+ * <link linkend="gst-running">Running GStreamer Applications</link>
+ * for how to disable automatic registry updates.
+ *
+ * This function should be called before calling any other GLib functions. If
+ * this is not an option, your program must initialise the GLib thread system
+ * using g_thread_init() before any other GLib functions are called.
+ *
  * <note><para>
  * This function will terminate your program if it was unable to initialize
  * GStreamer for some reason.  If you want your program to fall back,
@@ -385,7 +497,7 @@ gst_init (int *argc, char **argv[])
   GError *err = NULL;
 
   if (!gst_init_check (argc, argv, &err)) {
-    g_print ("Could not initialized GStreamer: %s\n",
+    g_print ("Could not initialize GStreamer: %s\n",
         err ? err->message : "unknown error occurred");
     if (err) {
       g_error_free (err);
@@ -394,52 +506,45 @@ gst_init (int *argc, char **argv[])
   }
 }
 
+/**
+ * gst_is_initialized:
+ *
+ * Use this function to check if GStreamer has been initialized with gst_init()
+ * or gst_init_check().
+ *
+ * Returns: TRUE if initialization has been done, FALSE otherwise.
+ *
+ * Since: 0.10.31
+ */
+gboolean
+gst_is_initialized (void)
+{
+  return gst_initialized;
+}
+
 #ifndef GST_DISABLE_REGISTRY
 static void
 add_path_func (gpointer data, gpointer user_data)
 {
   GST_INFO ("Adding plugin path: \"%s\", will scan later", (gchar *) data);
-  plugin_paths = g_list_append (plugin_paths, g_strdup (data));
+  _priv_gst_plugin_paths =
+      g_list_append (_priv_gst_plugin_paths, g_strdup (data));
 }
 #endif
 
+#ifndef GST_DISABLE_OPTION_PARSING
 static void
 prepare_for_load_plugin_func (gpointer data, gpointer user_data)
 {
-  preload_plugins = g_slist_prepend (preload_plugins, g_strdup (data));
-}
-
-static void
-load_plugin_func (gpointer data, gpointer user_data)
-{
-  GstPlugin *plugin;
-  const gchar *filename;
-  GError *err = NULL;
-
-  filename = (const gchar *) data;
-
-  plugin = gst_plugin_load_file (filename, &err);
-
-  if (plugin) {
-    GST_INFO ("Loaded plugin: \"%s\"", filename);
-
-    gst_default_registry_add_plugin (plugin);
-  } else {
-    if (err) {
-      /* Report error to user, and free error */
-      GST_ERROR ("Failed to load plugin: %s\n", err->message);
-      g_error_free (err);
-    } else {
-      GST_WARNING ("Failed to load plugin: \"%s\"", filename);
-    }
-  }
-
-  g_free (data);
+  _priv_gst_preload_plugins =
+      g_slist_prepend (_priv_gst_preload_plugins, g_strdup (data));
 }
+#endif
 
+#ifndef GST_DISABLE_OPTION_PARSING
 static void
-split_and_iterate (const gchar * stringlist, gchar * separator, GFunc iterator,
-    gpointer user_data)
+split_and_iterate (const gchar * stringlist, const gchar * separator,
+    GFunc iterator, gpointer user_data)
 {
   gchar **strings;
   gint j = 0;
@@ -461,34 +566,31 @@ split_and_iterate (const gchar * stringlist, gchar * separator, GFunc iterator,
     g_strfreev (strings);
   }
 }
+#endif
 
 /* we have no fail cases yet, but maybe in the future */
 static gboolean
-init_pre (void)
+init_pre (GOptionContext * context, GOptionGroup * group, gpointer data,
+    GError ** error)
 {
-  /* GStreamer was built against a GLib >= 2.8 and is therefore not doing
-   * the refcount hack. Check that it isn't being run against an older GLib */
-  if (glib_major_version < 2 ||
-      (glib_major_version == 2 && glib_minor_version < 8)) {
-    g_warning ("GStreamer was compiled against GLib %d.%d.%d but is running"
-        " against %d.%d.%d. This will cause reference counting issues",
-        GLIB_MAJOR_VERSION, GLIB_MINOR_VERSION, GLIB_MICRO_VERSION,
-        glib_major_version, glib_minor_version, glib_micro_version);
+  if (gst_initialized) {
+    GST_DEBUG ("already initialized");
+    return TRUE;
   }
 
   g_type_init ();
 
-  if (g_thread_supported ()) {
-    /* somebody already initialized threading */
-  } else {
-    g_thread_init (NULL);
-  }
+#if !GLIB_CHECK_VERSION (2, 31, 0)
   /* we need threading to be enabled right here */
+  g_assert (g_thread_get_initialized ());
+#endif
+
   _gst_debug_init ();
 
 #ifdef ENABLE_NLS
   setlocale (LC_ALL, "");
   bindtextdomain (GETTEXT_PACKAGE, LOCALEDIR);
+  bind_textdomain_codeset (GETTEXT_PACKAGE, "UTF-8");
 #endif /* ENABLE_NLS */
 
 #ifndef GST_DISABLE_GST_DEBUG
@@ -503,12 +605,27 @@ init_pre (void)
       parse_debug_list (debug_list);
     }
   }
+
+  priv_gst_dump_dot_dir = g_getenv ("GST_DEBUG_DUMP_DOT_DIR");
 #endif
   /* This is the earliest we can make stuff show up in the logs.
    * So give some useful info about GStreamer here */
   GST_INFO ("Initializing GStreamer Core Library version %s", VERSION);
   GST_INFO ("Using library installed in %s", LIBDIR);
 
+  /* Print some basic system details if possible (OS/architecture) */
+#ifdef HAVE_SYS_UTSNAME_H
+  {
+    struct utsname sys_details;
+
+    if (uname (&sys_details) == 0) {
+      GST_INFO ("%s %s %s %s %s", sys_details.sysname,
+          sys_details.nodename, sys_details.release, sys_details.version,
+          sys_details.machine);
+    }
+  }
+#endif
+
   return TRUE;
 }
 
@@ -526,224 +643,6 @@ gst_register_core_elements (GstPlugin * plugin)
   return TRUE;
 }
 
-static GstPluginDesc plugin_desc = {
-  GST_VERSION_MAJOR,
-  GST_VERSION_MINOR,
-  "staticelements",
-  "core elements linked into the GStreamer library",
-  gst_register_core_elements,
-  VERSION,
-  GST_LICENSE,
-  PACKAGE,
-  GST_PACKAGE_NAME,
-  GST_PACKAGE_ORIGIN,
-
-  GST_PADDING_INIT
-};
-
-#ifndef GST_DISABLE_REGISTRY
-
-static gboolean
-scan_and_update_registry (GstRegistry * default_registry,
-    const gchar * registry_file, gboolean write_changes)
-{
-  const gchar *plugin_path;
-  gboolean changed = FALSE;
-  GList *l;
-
-  GST_DEBUG ("reading registry cache: %s", registry_file);
-  gst_registry_xml_read_cache (default_registry, registry_file);
-
-  /* scan paths specified via --gst-plugin-path */
-  GST_DEBUG ("scanning paths added via --gst-plugin-path");
-  for (l = plugin_paths; l != NULL; l = l->next) {
-    GST_INFO ("Scanning plugin path: \"%s\"", (gchar *) l->data);
-    /* CHECKME: add changed |= here as well? */
-    gst_registry_scan_path (default_registry, (gchar *) l->data);
-    g_free (l->data);
-  }
-  g_list_free (plugin_paths);
-  plugin_paths = NULL;
-
-  /* GST_PLUGIN_PATH specifies a list of directories to scan for
-   * additional plugins.  These take precedence over the system plugins */
-  plugin_path = g_getenv ("GST_PLUGIN_PATH");
-  if (plugin_path) {
-    char **list;
-    int i;
-
-    GST_DEBUG ("GST_PLUGIN_PATH set to %s", plugin_path);
-    list = g_strsplit (plugin_path, G_SEARCHPATH_SEPARATOR_S, 0);
-    for (i = 0; list[i]; i++) {
-      changed |= gst_registry_scan_path (default_registry, list[i]);
-    }
-    g_strfreev (list);
-  } else {
-    GST_DEBUG ("GST_PLUGIN_PATH not set");
-  }
-
-  /* GST_PLUGIN_SYSTEM_PATH specifies a list of plugins that are always
-   * loaded by default.  If not set, this defaults to the system-installed
-   * path, and the plugins installed in the user's home directory */
-  plugin_path = g_getenv ("GST_PLUGIN_SYSTEM_PATH");
-  if (plugin_path == NULL) {
-    char *home_plugins;
-
-    GST_DEBUG ("GST_PLUGIN_SYSTEM_PATH not set");
-
-    /* plugins in the user's home directory take precedence over
-     * system-installed ones */
-    home_plugins = g_build_filename (g_get_home_dir (),
-        ".gstreamer-" GST_MAJORMINOR, "plugins", NULL);
-    changed |= gst_registry_scan_path (default_registry, home_plugins);
-    g_free (home_plugins);
-
-    /* add the main (installed) library path */
-    changed |= gst_registry_scan_path (default_registry, PLUGINDIR);
-  } else {
-    gchar **list;
-    gint i;
-
-    GST_DEBUG ("GST_PLUGIN_SYSTEM_PATH set to %s", plugin_path);
-    list = g_strsplit (plugin_path, G_SEARCHPATH_SEPARATOR_S, 0);
-    for (i = 0; list[i]; i++) {
-      changed |= gst_registry_scan_path (default_registry, list[i]);
-    }
-    g_strfreev (list);
-  }
-
-  if (!changed) {
-    GST_DEBUG ("registry cache has not changed");
-    return TRUE;
-  }
-
-  if (!write_changes) {
-    GST_DEBUG ("not trying to write registry cache changes to file");
-    return TRUE;
-  }
-
-  GST_DEBUG ("writing registry cache");
-  if (!gst_registry_xml_write_cache (default_registry, registry_file)) {
-    g_warning ("Problem writing registry cache to %s: %s", registry_file,
-        g_strerror (errno));
-    return FALSE;
-  }
-
-  GST_DEBUG ("registry cache written successfully");
-  return TRUE;
-}
-
-static gboolean
-ensure_current_registry_nonforking (GstRegistry * default_registry,
-    const gchar * registry_file)
-{
-  /* fork() not available */
-  GST_DEBUG ("updating registry cache");
-  scan_and_update_registry (default_registry, registry_file, TRUE);
-  return TRUE;
-}
-
-#ifdef HAVE_FORK
-static gboolean
-ensure_current_registry_forking (GstRegistry * default_registry,
-    const gchar * registry_file)
-{
-  pid_t pid;
-
-  /* We fork here, and let the child read and possibly rebuild the registry.
-   * After that, the parent will re-read the freshly generated registry. */
-
-  GST_DEBUG ("forking");
-  pid = fork ();
-  if (pid == -1) {
-    GST_ERROR ("Failed to fork()");
-    return FALSE;
-  }
-
-  if (pid == 0) {
-    gboolean res;
-
-    /* this is the child */
-    GST_DEBUG ("child reading registry cache");
-    res = scan_and_update_registry (default_registry, registry_file, TRUE);
-    _gst_registry_remove_cache_plugins (default_registry);
-
-    /* need to use _exit, so that any exit handlers registered don't
-     * bring down the main program */
-    GST_DEBUG ("child exiting: %s", (res) ? "SUCCESS" : "FAILURE");
-
-    /* make valgrind happy (yes, you can call it insane) */
-    g_free ((char *) registry_file);
-
-    _exit ((res) ? EXIT_SUCCESS : EXIT_FAILURE);
-  } else {
-    /* parent */
-    int status;
-    pid_t ret;
-
-    GST_DEBUG ("parent waiting on child");
-    ret = waitpid (pid, &status, 0);
-    GST_DEBUG ("parent done waiting on child");
-    if (ret == -1) {
-      GST_ERROR ("error during waitpid: %s", g_strerror (errno));
-      return FALSE;
-    }
-
-    if (!WIFEXITED (status)) {
-      GST_ERROR ("child did not exit normally, status: %d", status);
-      return FALSE;
-    }
-
-    GST_DEBUG ("child exited normally with return value %d",
-        WEXITSTATUS (status));
-
-    if (WEXITSTATUS (status) == EXIT_SUCCESS) {
-      GST_DEBUG ("parent reading registry cache");
-      gst_registry_xml_read_cache (default_registry, registry_file);
-    } else {
-      GST_DEBUG ("parent re-scanning registry");
-      scan_and_update_registry (default_registry, registry_file, FALSE);
-    }
-  }
-
-  return TRUE;
-}
-#endif /* HAVE_FORK */
-
-static gboolean
-ensure_current_registry (void)
-{
-  char *registry_file;
-  GstRegistry *default_registry;
-  gboolean ret;
-
-  default_registry = gst_registry_get_default ();
-  registry_file = g_strdup (g_getenv ("GST_REGISTRY"));
-  if (registry_file == NULL) {
-    registry_file = g_build_filename (g_get_home_dir (),
-        ".gstreamer-" GST_MAJORMINOR, "registry." HOST_CPU ".xml", NULL);
-  }
-#ifdef HAVE_FORK
-  if (g_getenv ("GST_REGISTRY_FORK") == NULL
-      || strcmp (g_getenv ("GST_REGISTRY_FORK"), "no") != 0) {
-    ret = ensure_current_registry_forking (default_registry, registry_file);
-  } else {
-    GST_DEBUG ("requested not to fork");
-#endif /* HAVE_FORK */
-
-    ret = ensure_current_registry_nonforking (default_registry, registry_file);
-
-#ifdef HAVE_FORK
-  }
-#endif /* HAVE_FORK */
-
-  g_free (registry_file);
-
-  return ret;
-}
-
-#endif /* GST_DISABLE_REGISTRY */
-
 /*
  * this bit handles:
  * - initalization of threads if we use them
@@ -756,7 +655,8 @@ ensure_current_registry (void)
  *   we might and then it's nice to be able to return that
  */
 static gboolean
-init_post (void)
+init_post (GOptionContext * context, GOptionGroup * group, gpointer data,
+    GError ** error)
 {
   GLogLevelFlags llf;
 
@@ -764,49 +664,135 @@ init_post (void)
   GstTrace *gst_trace;
 #endif /* GST_DISABLE_TRACE */
 
+  if (gst_initialized) {
+    GST_DEBUG ("already initialized");
+    return TRUE;
+  }
+
   llf = G_LOG_LEVEL_CRITICAL | G_LOG_LEVEL_ERROR | G_LOG_FLAG_FATAL;
   g_log_set_handler (g_log_domain_gstreamer, llf, debug_log_handler, NULL);
 
+  _priv_gst_quarks_initialize ();
   _gst_format_initialize ();
   _gst_query_initialize ();
-  gst_object_get_type ();
-  gst_pad_get_type ();
-  gst_element_factory_get_type ();
-  gst_element_get_type ();
-  gst_type_find_factory_get_type ();
-  gst_bin_get_type ();
-
-#ifndef GST_DISABLE_INDEX
-  gst_index_factory_get_type ();
-#endif /* GST_DISABLE_INDEX */
-#ifndef GST_DISABLE_URI
+  g_type_class_ref (gst_object_get_type ());
+  g_type_class_ref (gst_pad_get_type ());
+  g_type_class_ref (gst_element_factory_get_type ());
+  g_type_class_ref (gst_element_get_type ());
+  g_type_class_ref (gst_type_find_factory_get_type ());
+  g_type_class_ref (gst_bin_get_type ());
+  g_type_class_ref (gst_bus_get_type ());
+  g_type_class_ref (gst_task_get_type ());
+  g_type_class_ref (gst_clock_get_type ());
+
+  g_type_class_ref (gst_index_factory_get_type ());
   gst_uri_handler_get_type ();
-#endif /* GST_DISABLE_URI */
+
+  g_type_class_ref (gst_object_flags_get_type ());
+  g_type_class_ref (gst_bin_flags_get_type ());
+  g_type_class_ref (gst_buffer_flag_get_type ());
+  g_type_class_ref (gst_buffer_copy_flags_get_type ());
+  g_type_class_ref (gst_buffer_list_item_get_type ());
+  g_type_class_ref (gst_bus_flags_get_type ());
+  g_type_class_ref (gst_bus_sync_reply_get_type ());
+  g_type_class_ref (gst_caps_flags_get_type ());
+  g_type_class_ref (gst_clock_return_get_type ());
+  g_type_class_ref (gst_clock_entry_type_get_type ());
+  g_type_class_ref (gst_clock_flags_get_type ());
+  g_type_class_ref (gst_clock_type_get_type ());
+  g_type_class_ref (gst_debug_graph_details_get_type ());
+  g_type_class_ref (gst_state_get_type ());
+  g_type_class_ref (gst_state_change_return_get_type ());
+  g_type_class_ref (gst_state_change_get_type ());
+  g_type_class_ref (gst_element_flags_get_type ());
+  g_type_class_ref (gst_core_error_get_type ());
+  g_type_class_ref (gst_library_error_get_type ());
+  g_type_class_ref (gst_resource_error_get_type ());
+  g_type_class_ref (gst_stream_error_get_type ());
+  g_type_class_ref (gst_event_type_flags_get_type ());
+  g_type_class_ref (gst_event_type_get_type ());
+  g_type_class_ref (gst_seek_type_get_type ());
+  g_type_class_ref (gst_seek_flags_get_type ());
+  g_type_class_ref (gst_qos_type_get_type ());
+  g_type_class_ref (gst_format_get_type ());
+  g_type_class_ref (gst_index_certainty_get_type ());
+  g_type_class_ref (gst_index_entry_type_get_type ());
+  g_type_class_ref (gst_index_lookup_method_get_type ());
+  g_type_class_ref (gst_assoc_flags_get_type ());
+  g_type_class_ref (gst_index_resolver_method_get_type ());
+  g_type_class_ref (gst_index_flags_get_type ());
+  g_type_class_ref (gst_debug_level_get_type ());
+  g_type_class_ref (gst_debug_color_flags_get_type ());
+  g_type_class_ref (gst_iterator_result_get_type ());
+  g_type_class_ref (gst_iterator_item_get_type ());
+  g_type_class_ref (gst_message_type_get_type ());
+  g_type_class_ref (gst_mini_object_flags_get_type ());
+  g_type_class_ref (gst_pad_link_return_get_type ());
+  g_type_class_ref (gst_pad_link_check_get_type ());
+  g_type_class_ref (gst_flow_return_get_type ());
+  g_type_class_ref (gst_activate_mode_get_type ());
+  g_type_class_ref (gst_pad_direction_get_type ());
+  g_type_class_ref (gst_pad_flags_get_type ());
+  g_type_class_ref (gst_pad_presence_get_type ());
+  g_type_class_ref (gst_pad_template_flags_get_type ());
+  g_type_class_ref (gst_pipeline_flags_get_type ());
+  g_type_class_ref (gst_plugin_error_get_type ());
+  g_type_class_ref (gst_plugin_flags_get_type ());
+  g_type_class_ref (gst_plugin_dependency_flags_get_type ());
+  g_type_class_ref (gst_rank_get_type ());
+  g_type_class_ref (gst_query_type_get_type ());
+  g_type_class_ref (gst_buffering_mode_get_type ());
+  g_type_class_ref (gst_stream_status_type_get_type ());
+  g_type_class_ref (gst_structure_change_type_get_type ());
+  g_type_class_ref (gst_tag_merge_mode_get_type ());
+  g_type_class_ref (gst_tag_flag_get_type ());
+  g_type_class_ref (gst_task_pool_get_type ());
+  g_type_class_ref (gst_task_state_get_type ());
+  g_type_class_ref (gst_alloc_trace_flags_get_type ());
+  g_type_class_ref (gst_type_find_probability_get_type ());
+  g_type_class_ref (gst_uri_type_get_type ());
+  g_type_class_ref (gst_parse_error_get_type ());
+  g_type_class_ref (gst_parse_flags_get_type ());
+  g_type_class_ref (gst_search_mode_get_type ());
+  g_type_class_ref (gst_progress_type_get_type ());
+  g_type_class_ref (gst_caps_intersect_mode_get_type ());
 
   gst_structure_get_type ();
   _gst_value_initialize ();
+  g_type_class_ref (gst_param_spec_fraction_get_type ());
   gst_caps_get_type ();
   _gst_event_initialize ();
   _gst_buffer_initialize ();
+  _gst_buffer_list_initialize ();
+  gst_buffer_list_iterator_get_type ();
   _gst_message_initialize ();
   _gst_tag_initialize ();
-
-  /* register core plugins */
-  _gst_plugin_register_static (&plugin_desc);
+  gst_parse_context_get_type ();
 
   _gst_plugin_initialize ();
 
-#ifndef GST_DISABLE_REGISTRY
-  if (!ensure_current_registry ())
+  gst_g_error_get_type ();
+
+  /* register core plugins */
+  gst_plugin_register_static (GST_VERSION_MAJOR, GST_VERSION_MINOR,
+      "staticelements", "core elements linked into the GStreamer library",
+      gst_register_core_elements, VERSION, GST_LICENSE, PACKAGE,
+      GST_PACKAGE_NAME, GST_PACKAGE_ORIGIN);
+
+  /*
+   * Any errors happening below this point are non-fatal, we therefore mark
+   * gstreamer as being initialized, since it is the case from a plugin point of
+   * view.
+   *
+   * If anything fails, it will be put back to FALSE in gst_init_check().
+   * This allows some special plugins that would call gst_init() to not cause a
+   * looping effect (i.e. initializing GStreamer twice).
+   */
+  gst_initialized = TRUE;
+
+  if (!gst_update_registry ())
     return FALSE;
-#endif /* GST_DISABLE_REGISTRY */
 
-  /* if we need to preload plugins */
-  if (preload_plugins) {
-    g_slist_foreach (preload_plugins, load_plugin_func, NULL);
-    g_slist_free (preload_plugins);
-    preload_plugins = NULL;
-  }
 #ifndef GST_DISABLE_TRACE
   _gst_trace_on = 0;
   if (_gst_trace_on) {
@@ -815,6 +801,11 @@ init_post (void)
   }
 #endif /* GST_DISABLE_TRACE */
 
+  GST_INFO ("GLib runtime version: %d.%d.%d", glib_major_version,
+      glib_minor_version, glib_micro_version);
+  GST_INFO ("GLib headers version: %d.%d.%d", GLIB_MAJOR_VERSION,
+      GLIB_MINOR_VERSION, GLIB_MICRO_VERSION);
+
   return TRUE;
 }
 
@@ -831,13 +822,15 @@ sort_by_category_name (gconstpointer a, gconstpointer b)
   return strcmp (gst_debug_category_get_name ((GstDebugCategory *) a),
       gst_debug_category_get_name ((GstDebugCategory *) b));
 }
+
 static void
 gst_debug_help (void)
 {
   GSList *list, *walk;
   GList *list2, *g;
 
-  if (!init_post ())
+  /* Need to ensure the registry is loaded to get debug categories */
+  if (!init_post (NULL, NULL, NULL, NULL))
     exit (1);
 
   list2 = gst_registry_plugin_filter (gst_registry_get_default (),
@@ -862,6 +855,20 @@ gst_debug_help (void)
     GstDebugCategory *cat = (GstDebugCategory *) walk->data;
 
     if (gst_debug_is_colored ()) {
+#ifdef G_OS_WIN32
+      gint color = gst_debug_construct_win_color (cat->color);
+      const gint clear = FOREGROUND_RED | FOREGROUND_GREEN | FOREGROUND_BLUE;
+
+      SetConsoleTextAttribute (GetStdHandle (STD_OUTPUT_HANDLE), color);
+      g_print ("%-20s", gst_debug_category_get_name (cat));
+      SetConsoleTextAttribute (GetStdHandle (STD_OUTPUT_HANDLE), clear);
+      g_print (" %1d %s ", gst_debug_category_get_threshold (cat),
+          gst_debug_level_get_name (gst_debug_category_get_threshold (cat)));
+      SetConsoleTextAttribute (GetStdHandle (STD_OUTPUT_HANDLE), color);
+      g_print ("%s", gst_debug_category_get_description (cat));
+      SetConsoleTextAttribute (GetStdHandle (STD_OUTPUT_HANDLE), clear);
+      g_print ("\n");
+#else /* G_OS_WIN32 */
       gchar *color = gst_debug_construct_term_color (cat->color);
 
       g_print ("%s%-20s\033[00m  %1d %s  %s%s\033[00m\n",
@@ -871,6 +878,7 @@ gst_debug_help (void)
           gst_debug_level_get_name (gst_debug_category_get_threshold (cat)),
           color, gst_debug_category_get_description (cat));
       g_free (color);
+#endif /* G_OS_WIN32 */
     } else {
       g_print ("%-20s  %1d %s  %s\n", gst_debug_category_get_name (cat),
           gst_debug_category_get_threshold (cat),
@@ -884,6 +892,7 @@ gst_debug_help (void)
 }
 #endif
 
+#ifndef GST_DISABLE_OPTION_PARSING
 static gboolean
 parse_one_option (gint opt, const gchar * arg, GError ** err)
 {
@@ -901,9 +910,9 @@ parse_one_option (gint opt, const gchar * arg, GError ** err)
     }
 #ifndef GST_DISABLE_GST_DEBUG
     case ARG_DEBUG_LEVEL:{
-      gint tmp = 0;
+      GstDebugLevel tmp = GST_LEVEL_NONE;
 
-      tmp = strtol (arg, NULL, 0);
+      tmp = (GstDebugLevel) strtol (arg, NULL, 0);
       if (tmp >= 0 && tmp < GST_LEVEL_COUNT) {
         gst_debug_set_default_threshold (tmp);
       }
@@ -935,6 +944,14 @@ parse_one_option (gint opt, const gchar * arg, GError ** err)
     case ARG_SEGTRAP_DISABLE:
       _gst_disable_segtrap = TRUE;
       break;
+    case ARG_REGISTRY_UPDATE_DISABLE:
+#ifndef GST_DISABLE_REGISTRY
+      _priv_gst_disable_registry_update = TRUE;
+#endif
+      break;
+    case ARG_REGISTRY_FORK_DISABLE:
+      gst_registry_fork_set_enabled (FALSE);
+      break;
     default:
       g_set_error (err, G_OPTION_ERROR, G_OPTION_ERROR_UNKNOWN_OPTION,
           _("Unknown option"));
@@ -948,9 +965,9 @@ static gboolean
 parse_goption_arg (const gchar * opt,
     const gchar * arg, gpointer data, GError ** err)
 {
-  const struct
+  static const struct
   {
-    gchar *opt;
+    const gchar *opt;
     int val;
   } options[] = {
     {
@@ -969,6 +986,8 @@ parse_goption_arg (const gchar * opt,
     "--gst-plugin-path", ARG_PLUGIN_PATH}, {
     "--gst-plugin-load", ARG_PLUGIN_LOAD}, {
     "--gst-disable-segtrap", ARG_SEGTRAP_DISABLE}, {
+    "--gst-disable-registry-update", ARG_REGISTRY_UPDATE_DISABLE}, {
+    "--gst-disable-registry-fork", ARG_REGISTRY_FORK_DISABLE}, {
     NULL}
   };
   gint val = 0, n;
@@ -982,38 +1001,140 @@ parse_goption_arg (const gchar * opt,
 
   return parse_one_option (val, arg, err);
 }
+#endif
 
 /**
  * gst_deinit:
  *
- * Clean up.
- * Call only once, before exiting.
- * After this call GStreamer should not be used anymore.
+ * Clean up any resources created by GStreamer in gst_init().
+ *
+ * It is normally not needed to call this function in a normal application
+ * as the resources will automatically be freed when the program terminates.
+ * This function is therefore mostly used by testsuites and other memory
+ * profiling tools.
+ *
+ * After this call GStreamer (including this method) should not be used anymore. 
  */
-
-extern GstRegistry *_gst_registry_default;
 void
 gst_deinit (void)
 {
   GstClock *clock;
 
   GST_INFO ("deinitializing GStreamer");
+
+  if (gst_deinitialized) {
+    GST_DEBUG ("already deinitialized");
+    return;
+  }
+
+  g_slist_foreach (_priv_gst_preload_plugins, (GFunc) g_free, NULL);
+  g_slist_free (_priv_gst_preload_plugins);
+  _priv_gst_preload_plugins = NULL;
+
+#ifndef GST_DISABLE_REGISTRY
+  g_list_foreach (_priv_gst_plugin_paths, (GFunc) g_free, NULL);
+  g_list_free (_priv_gst_plugin_paths);
+  _priv_gst_plugin_paths = NULL;
+#endif
+
   clock = gst_system_clock_obtain ();
   gst_object_unref (clock);
   gst_object_unref (clock);
 
-  _gst_registry_cleanup ();
-
-  gst_initialized = FALSE;
+  _priv_gst_registry_cleanup ();
+
+  g_type_class_unref (g_type_class_peek (gst_object_get_type ()));
+  g_type_class_unref (g_type_class_peek (gst_pad_get_type ()));
+  g_type_class_unref (g_type_class_peek (gst_element_factory_get_type ()));
+  g_type_class_unref (g_type_class_peek (gst_element_get_type ()));
+  g_type_class_unref (g_type_class_peek (gst_type_find_factory_get_type ()));
+  g_type_class_unref (g_type_class_peek (gst_bin_get_type ()));
+  g_type_class_unref (g_type_class_peek (gst_bus_get_type ()));
+  g_type_class_unref (g_type_class_peek (gst_task_get_type ()));
+  g_type_class_unref (g_type_class_peek (gst_index_factory_get_type ()));
+  g_type_class_unref (g_type_class_peek (gst_object_flags_get_type ()));
+  g_type_class_unref (g_type_class_peek (gst_bin_flags_get_type ()));
+  g_type_class_unref (g_type_class_peek (gst_buffer_flag_get_type ()));
+  g_type_class_unref (g_type_class_peek (gst_buffer_copy_flags_get_type ()));
+  g_type_class_unref (g_type_class_peek (gst_buffer_list_item_get_type ()));
+  g_type_class_unref (g_type_class_peek (gst_bus_flags_get_type ()));
+  g_type_class_unref (g_type_class_peek (gst_bus_sync_reply_get_type ()));
+  g_type_class_unref (g_type_class_peek (gst_caps_flags_get_type ()));
+  g_type_class_unref (g_type_class_peek (gst_clock_type_get_type ()));
+  g_type_class_unref (g_type_class_peek (gst_clock_return_get_type ()));
+  g_type_class_unref (g_type_class_peek (gst_clock_entry_type_get_type ()));
+  g_type_class_unref (g_type_class_peek (gst_clock_flags_get_type ()));
+  g_type_class_unref (g_type_class_peek (gst_debug_graph_details_get_type ()));
+  g_type_class_unref (g_type_class_peek (gst_state_get_type ()));
+  g_type_class_unref (g_type_class_peek (gst_state_change_return_get_type ()));
+  g_type_class_unref (g_type_class_peek (gst_state_change_get_type ()));
+  g_type_class_unref (g_type_class_peek (gst_element_flags_get_type ()));
+  g_type_class_unref (g_type_class_peek (gst_core_error_get_type ()));
+  g_type_class_unref (g_type_class_peek (gst_library_error_get_type ()));
+  g_type_class_unref (g_type_class_peek (gst_plugin_dependency_flags_get_type
+          ()));
+  g_type_class_unref (g_type_class_peek (gst_parse_flags_get_type ()));
+  g_type_class_unref (g_type_class_peek (gst_resource_error_get_type ()));
+  g_type_class_unref (g_type_class_peek (gst_search_mode_get_type ()));
+  g_type_class_unref (g_type_class_peek (gst_stream_error_get_type ()));
+  g_type_class_unref (g_type_class_peek (gst_stream_status_type_get_type ()));
+  g_type_class_unref (g_type_class_peek (gst_structure_change_type_get_type
+          ()));
+  g_type_class_unref (g_type_class_peek (gst_event_type_flags_get_type ()));
+  g_type_class_unref (g_type_class_peek (gst_event_type_get_type ()));
+  g_type_class_unref (g_type_class_peek (gst_seek_type_get_type ()));
+  g_type_class_unref (g_type_class_peek (gst_seek_flags_get_type ()));
+  g_type_class_unref (g_type_class_peek (gst_qos_type_get_type ()));
+  g_type_class_unref (g_type_class_peek (gst_format_get_type ()));
+  g_type_class_unref (g_type_class_peek (gst_index_certainty_get_type ()));
+  g_type_class_unref (g_type_class_peek (gst_index_entry_type_get_type ()));
+  g_type_class_unref (g_type_class_peek (gst_index_lookup_method_get_type ()));
+  g_type_class_unref (g_type_class_peek (gst_assoc_flags_get_type ()));
+  g_type_class_unref (g_type_class_peek (gst_index_resolver_method_get_type
+          ()));
+  g_type_class_unref (g_type_class_peek (gst_index_flags_get_type ()));
+  g_type_class_unref (g_type_class_peek (gst_debug_level_get_type ()));
+  g_type_class_unref (g_type_class_peek (gst_debug_color_flags_get_type ()));
+  g_type_class_unref (g_type_class_peek (gst_iterator_result_get_type ()));
+  g_type_class_unref (g_type_class_peek (gst_iterator_item_get_type ()));
+  g_type_class_unref (g_type_class_peek (gst_message_type_get_type ()));
+  g_type_class_unref (g_type_class_peek (gst_mini_object_flags_get_type ()));
+  g_type_class_unref (g_type_class_peek (gst_pad_link_return_get_type ()));
+  g_type_class_unref (g_type_class_peek (gst_pad_link_check_get_type ()));
+  g_type_class_unref (g_type_class_peek (gst_flow_return_get_type ()));
+  g_type_class_unref (g_type_class_peek (gst_activate_mode_get_type ()));
+  g_type_class_unref (g_type_class_peek (gst_pad_direction_get_type ()));
+  g_type_class_unref (g_type_class_peek (gst_pad_flags_get_type ()));
+  g_type_class_unref (g_type_class_peek (gst_pad_presence_get_type ()));
+  g_type_class_unref (g_type_class_peek (gst_pad_template_flags_get_type ()));
+  g_type_class_unref (g_type_class_peek (gst_pipeline_flags_get_type ()));
+  g_type_class_unref (g_type_class_peek (gst_plugin_error_get_type ()));
+  g_type_class_unref (g_type_class_peek (gst_plugin_flags_get_type ()));
+  g_type_class_unref (g_type_class_peek (gst_rank_get_type ()));
+  g_type_class_unref (g_type_class_peek (gst_query_type_get_type ()));
+  g_type_class_unref (g_type_class_peek (gst_buffering_mode_get_type ()));
+  g_type_class_unref (g_type_class_peek (gst_tag_merge_mode_get_type ()));
+  g_type_class_unref (g_type_class_peek (gst_tag_flag_get_type ()));
+  g_type_class_unref (g_type_class_peek (gst_task_state_get_type ()));
+  g_type_class_unref (g_type_class_peek (gst_alloc_trace_flags_get_type ()));
+  g_type_class_unref (g_type_class_peek (gst_type_find_probability_get_type
+          ()));
+  g_type_class_unref (g_type_class_peek (gst_uri_type_get_type ()));
+  g_type_class_unref (g_type_class_peek (gst_parse_error_get_type ()));
+  g_type_class_unref (g_type_class_peek (gst_param_spec_fraction_get_type ()));
+  g_type_class_unref (g_type_class_peek (gst_progress_type_get_type ()));
+  g_type_class_unref (g_type_class_peek (gst_caps_intersect_mode_get_type ()));
+
+  gst_deinitialized = TRUE;
   GST_INFO ("deinitialized GStreamer");
 }
 
 /**
  * gst_version:
- * @major: pointer to a guint to store the major version number
- * @minor: pointer to a guint to store the minor version number
- * @micro: pointer to a guint to store the micro version number
- * @nano:  pointer to a guint to store the nano version number
+ * @major: (out): pointer to a guint to store the major version number
+ * @minor: (out): pointer to a guint to store the minor version number
+ * @micro: (out): pointer to a guint to store the micro version number
+ * @nano:  (out): pointer to a guint to store the nano version number
  *
  * Gets the version number of the GStreamer library.
  */
@@ -1037,11 +1158,12 @@ gst_version (guint * major, guint * minor, guint * micro, guint * nano)
  * This function returns a string that is useful for describing this version
  * of GStreamer to the outside world: user agent strings, logging, ...
  *
- * Returns: a newly allocated string describing this version of GStreamer.
+ * Returns: (transfer full): a newly allocated string describing this version
+ *     of GStreamer.
  */
 
 gchar *
-gst_version_string ()
+gst_version_string (void)
 {
   guint major, minor, micro, nano;
 
@@ -1049,7 +1171,7 @@ gst_version_string ()
   if (nano == 0)
     return g_strdup_printf ("GStreamer %d.%d.%d", major, minor, micro);
   else if (nano == 1)
-    return g_strdup_printf ("GStreamer %d.%d.%d (CVS)", major, minor, micro);
+    return g_strdup_printf ("GStreamer %d.%d.%d (GIT)", major, minor, micro);
   else
     return g_strdup_printf ("GStreamer %d.%d.%d (prerelease)", major, minor,
         micro);
@@ -1058,9 +1180,9 @@ gst_version_string ()
 /**
  * gst_segtrap_is_enabled:
  *
- * Some functions in the GStreamer core might install a custom SIGSEGV handler to
- * better catch and report errors to the application. Currently this feature is 
- * enabled by default when loading plugins.
+ * Some functions in the GStreamer core might install a custom SIGSEGV handler
+ * to better catch and report errors to the application. Currently this feature
+ * is enabled by default when loading plugins.
  *
  * Applications might want to disable this behaviour with the
  * gst_segtrap_set_enabled() function. This is typically done if the application