gst/gstelementfactory.c (gst_element_factory_create): Avoid eating the caller's refcount.

Original commit message from CVS:
2005-09-20  Andy Wingo  <wingo@pobox.com>

* gst/gstelementfactory.c (gst_element_factory_create): Avoid
eating the caller's refcount.

* gst/gstobject.h (GST_OBJECT_REFCOUNT)
(GST_OBJECT_REFCOUNT_VALUE): Conditionally fondle the right
refcount.

* gst/gstconfig.h.in (GST_HAVE_GLIB_2_8):
* configure.ac (GST_HAVE_GLIB_2_8_DEFINE): Make the availability
of GLib 2.8 public, so we can know which refcount to check in
tests.

* gst/gstobject.c: Use the GST_HAVE_GLIB_2_8 define.
(gst_object_init): Only set the gst refcount if we're going ahead
with the refcount hack.
This commit is contained in:
Andy Wingo 2005-09-20 11:09:50 +00:00
parent 2ed824baac
commit b12471008b
6 changed files with 49 additions and 7 deletions

View file

@ -1,3 +1,21 @@
2005-09-20 Andy Wingo <wingo@pobox.com>
* gst/gstelementfactory.c (gst_element_factory_create): Avoid
eating the caller's refcount.
* gst/gstobject.h (GST_OBJECT_REFCOUNT)
(GST_OBJECT_REFCOUNT_VALUE): Conditionally fondle the right
refcount.
* gst/gstconfig.h.in (GST_HAVE_GLIB_2_8):
* configure.ac (GST_HAVE_GLIB_2_8_DEFINE): Make the availability
of GLib 2.8 public, so we can know which refcount to check in
tests.
* gst/gstobject.c: Use the GST_HAVE_GLIB_2_8 define.
(gst_object_init): Only set the gst refcount if we're going ahead
with the refcount hack.
2005-09-20 Stefan Kost <ensonic@users.sf.net> 2005-09-20 Stefan Kost <ensonic@users.sf.net>
* check/gst-libs/controller.c: (plugin_init), (GST_START_TEST): * check/gst-libs/controller.c: (plugin_init), (GST_START_TEST):

View file

@ -238,10 +238,12 @@ AC_SUBST(GLIB_CFLAGS)
AC_MSG_CHECKING([glib version >= 2.8]) AC_MSG_CHECKING([glib version >= 2.8])
if pkg-config --atleast-version=2.8 glib-2.0; then if pkg-config --atleast-version=2.8 glib-2.0; then
AC_MSG_RESULT(yes) AC_MSG_RESULT(yes)
AC_DEFINE(HAVE_GLIB_2_8, [1], [Define to 1 if we can use API new in glib-2.8]) GST_HAVE_GLIB_2_8_DEFINE="#define GST_HAVE_GLIB_2_8 1"
else else
AC_MSG_RESULT(no) AC_MSG_RESULT(no)
GST_HAVE_GLIB_2_8_DEFINE="/* #undef GST_HAVE_GLIB_2_8 */"
fi fi
AC_SUBST(GST_HAVE_GLIB_2_8_DEFINE)
if test "x$HAVE_GLIB2" = "xno"; then if test "x$HAVE_GLIB2" = "xno"; then
AC_MSG_ERROR([GStreamer requires GLib $GLIB2_REQ to compile.]) AC_MSG_ERROR([GStreamer requires GLib $GLIB2_REQ to compile.])

View file

@ -112,6 +112,10 @@
/* whether or not the CPU supports unaligned access */ /* whether or not the CPU supports unaligned access */
@GST_HAVE_UNALIGNED_ACCESS_DEFINE@ @GST_HAVE_UNALIGNED_ACCESS_DEFINE@
/* whether or not we are using glib 2.8 api, e.g. atomic gobject
refcounting */
@GST_HAVE_GLIB_2_8_DEFINE@
/***** Deal with XML stuff, we have to handle both loadsave and registry *****/ /***** Deal with XML stuff, we have to handle both loadsave and registry *****/
#if (! (defined(GST_DISABLE_LOADSAVE) && defined(GST_DISABLE_REGISTRY)) ) #if (! (defined(GST_DISABLE_LOADSAVE) && defined(GST_DISABLE_REGISTRY)) )

View file

@ -349,15 +349,27 @@ gst_element_factory_create (GstElementFactory * factory, const gchar * name)
{ {
GstElement *element; GstElement *element;
GstElementClass *oclass; GstElementClass *oclass;
GstElementFactory *newfactory;
g_return_val_if_fail (factory != NULL, NULL); g_return_val_if_fail (factory != NULL, NULL);
factory = gst_object_ref (factory);
newfactory =
GST_ELEMENT_FACTORY (gst_plugin_feature_load (GST_PLUGIN_FEATURE GST_ELEMENT_FACTORY (gst_plugin_feature_load (GST_PLUGIN_FEATURE
(factory))); (factory)));
if (factory == NULL) { if (newfactory == NULL) {
GST_DEBUG ("warning: loading the plugin for this factory returned NULL"); /* it could be factory is invalid. with ref-eating functions nothing is
certain! */
GST_WARNING ("loading the plugin for factory %p returned NULL", factory);
return NULL; return NULL;
} else if (newfactory != factory) {
/* gst_plugin_feature_load ate the ref we added to the factory */
factory = newfactory;
} else {
/* strip off our extra ref */
gst_object_unref (factory);
factory = newfactory;
} }
if (name) if (name)
@ -392,7 +404,6 @@ gst_element_factory_create (GstElementFactory * factory, const gchar * name)
gst_object_set_name (GST_OBJECT (element), name); gst_object_set_name (GST_OBJECT (element), name);
GST_DEBUG ("created \"%s\"", GST_PLUGIN_FEATURE_NAME (factory)); GST_DEBUG ("created \"%s\"", GST_PLUGIN_FEATURE_NAME (factory));
gst_object_unref (factory);
return element; return element;
} }

View file

@ -33,7 +33,7 @@
#endif #endif
#define DEBUG_REFCOUNT #define DEBUG_REFCOUNT
#ifndef HAVE_GLIB_2_8 #ifndef GST_HAVE_GLIB_2_8
#define REFCOUNT_HACK #define REFCOUNT_HACK
#endif #endif
@ -218,7 +218,9 @@ gst_object_init (GTypeInstance * instance, gpointer g_class)
object->parent = NULL; object->parent = NULL;
object->name = NULL; object->name = NULL;
GST_CAT_LOG_OBJECT (GST_CAT_REFCOUNTING, object, "%p new", object); GST_CAT_LOG_OBJECT (GST_CAT_REFCOUNTING, object, "%p new", object);
#ifdef REFCOUNT_HACK
gst_atomic_int_set (&object->refcount, 1); gst_atomic_int_set (&object->refcount, 1);
#endif
PATCH_REFCOUNT (object); PATCH_REFCOUNT (object);
object->flags = 0; object->flags = 0;

View file

@ -55,8 +55,13 @@ typedef enum
GST_OBJECT_FLAG_LAST = 4 GST_OBJECT_FLAG_LAST = 4
} GstObjectFlags; } GstObjectFlags;
#ifdef GST_HAVE_GLIB_2_8
#define GST_OBJECT_REFCOUNT(obj) (((GObject*)(obj))->ref_count)
#define GST_OBJECT_REFCOUNT_VALUE(obj) (g_atomic_int_get (&((GObject*)(obj))->ref_count))
#else
#define GST_OBJECT_REFCOUNT(obj) ((GST_OBJECT_CAST(obj))->refcount) #define GST_OBJECT_REFCOUNT(obj) ((GST_OBJECT_CAST(obj))->refcount)
#define GST_OBJECT_REFCOUNT_VALUE(obj) (g_atomic_int_get (&(GST_OBJECT_CAST(obj))->refcount)) #define GST_OBJECT_REFCOUNT_VALUE(obj) (g_atomic_int_get (&(GST_OBJECT_CAST(obj))->refcount))
#endif /* GST_HAVE_GLIB_2_8 */
/* we do a GST_OBJECT_CAST to avoid type checking, better call these /* we do a GST_OBJECT_CAST to avoid type checking, better call these
* function with a valid object! */ * function with a valid object! */
@ -84,7 +89,7 @@ struct _GstObject {
GObject object; GObject object;
/*< public >*/ /*< public >*/
gint refcount; gint refcount; /* only used ifndef GST_HAVE_GLIB_0_8 */
/*< public >*/ /* with LOCK */ /*< public >*/ /* with LOCK */
GMutex *lock; /* object LOCK */ GMutex *lock; /* object LOCK */