mirror of
https://gitlab.gnome.org/GNOME/glib.git
synced 2024-12-25 06:56:14 +01:00
edb40c7171
gjs has some situations where it's not always aware of the @data that was passed into g_object_add_toggle_ref, so allow passing %NULL to just match on @notify. Rebased and updated by Nitin Wartkar Closes #817
281 lines
7.4 KiB
C
281 lines
7.4 KiB
C
/* GObject - GLib Type, Object, Parameter and Signal Library
|
|
* Copyright (C) 2005 Red Hat, Inc.
|
|
*
|
|
* This library is free software; you can redistribute it and/or
|
|
* modify it under the terms of the GNU Lesser General Public
|
|
* License as published by the Free Software Foundation; either
|
|
* version 2.1 of the License, or (at your option) any later version.
|
|
*
|
|
* This library is distributed in the hope that it will be useful,
|
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
|
|
* Lesser General Public License for more details.
|
|
*
|
|
* You should have received a copy of the GNU Lesser General
|
|
* Public License along with this library; if not, see <http://www.gnu.org/licenses/>.
|
|
*/
|
|
|
|
#undef G_LOG_DOMAIN
|
|
#define G_LOG_DOMAIN "TestReferences"
|
|
|
|
#undef G_DISABLE_ASSERT
|
|
#undef G_DISABLE_CHECKS
|
|
#undef G_DISABLE_CAST_CHECKS
|
|
|
|
#include <glib-object.h>
|
|
|
|
/* This test tests weak and toggle references
|
|
*/
|
|
|
|
static GObject *global_object;
|
|
|
|
static gboolean object_destroyed;
|
|
static gboolean weak_ref1_notified;
|
|
static gboolean weak_ref2_notified;
|
|
static gboolean toggle_ref1_weakened;
|
|
static gboolean toggle_ref1_strengthened;
|
|
static gboolean toggle_ref2_weakened;
|
|
static gboolean toggle_ref2_strengthened;
|
|
static gboolean toggle_ref3_weakened;
|
|
static gboolean toggle_ref3_strengthened;
|
|
|
|
/*
|
|
* TestObject, a parent class for TestObject
|
|
*/
|
|
static GType test_object_get_type (void);
|
|
#define TEST_TYPE_OBJECT (test_object_get_type ())
|
|
typedef struct _TestObject TestObject;
|
|
typedef struct _TestObjectClass TestObjectClass;
|
|
|
|
struct _TestObject
|
|
{
|
|
GObject parent_instance;
|
|
};
|
|
struct _TestObjectClass
|
|
{
|
|
GObjectClass parent_class;
|
|
};
|
|
|
|
G_DEFINE_TYPE (TestObject, test_object, G_TYPE_OBJECT)
|
|
|
|
static void
|
|
test_object_finalize (GObject *object)
|
|
{
|
|
object_destroyed = TRUE;
|
|
|
|
G_OBJECT_CLASS (test_object_parent_class)->finalize (object);
|
|
}
|
|
|
|
static void
|
|
test_object_class_init (TestObjectClass *class)
|
|
{
|
|
GObjectClass *object_class = G_OBJECT_CLASS (class);
|
|
|
|
object_class->finalize = test_object_finalize;
|
|
}
|
|
|
|
static void
|
|
test_object_init (TestObject *test_object)
|
|
{
|
|
}
|
|
|
|
static void
|
|
clear_flags (void)
|
|
{
|
|
object_destroyed = FALSE;
|
|
weak_ref1_notified = FALSE;
|
|
weak_ref2_notified = FALSE;
|
|
toggle_ref1_weakened = FALSE;
|
|
toggle_ref1_strengthened = FALSE;
|
|
toggle_ref2_weakened = FALSE;
|
|
toggle_ref2_strengthened = FALSE;
|
|
toggle_ref3_weakened = FALSE;
|
|
toggle_ref3_strengthened = FALSE;
|
|
}
|
|
|
|
static void
|
|
weak_ref1 (gpointer data,
|
|
GObject *object)
|
|
{
|
|
g_assert (object == global_object);
|
|
g_assert (data == GUINT_TO_POINTER (42));
|
|
|
|
weak_ref1_notified = TRUE;
|
|
}
|
|
|
|
static void
|
|
weak_ref2 (gpointer data,
|
|
GObject *object)
|
|
{
|
|
g_assert (object == global_object);
|
|
g_assert (data == GUINT_TO_POINTER (24));
|
|
|
|
weak_ref2_notified = TRUE;
|
|
}
|
|
|
|
static void
|
|
toggle_ref1 (gpointer data,
|
|
GObject *object,
|
|
gboolean is_last_ref)
|
|
{
|
|
g_assert (object == global_object);
|
|
g_assert (data == GUINT_TO_POINTER (42));
|
|
|
|
if (is_last_ref)
|
|
toggle_ref1_weakened = TRUE;
|
|
else
|
|
toggle_ref1_strengthened = TRUE;
|
|
}
|
|
|
|
static void
|
|
toggle_ref2 (gpointer data,
|
|
GObject *object,
|
|
gboolean is_last_ref)
|
|
{
|
|
g_assert (object == global_object);
|
|
g_assert (data == GUINT_TO_POINTER (24));
|
|
|
|
if (is_last_ref)
|
|
toggle_ref2_weakened = TRUE;
|
|
else
|
|
toggle_ref2_strengthened = TRUE;
|
|
}
|
|
|
|
static void
|
|
toggle_ref3 (gpointer data,
|
|
GObject *object,
|
|
gboolean is_last_ref)
|
|
{
|
|
g_assert (object == global_object);
|
|
g_assert (data == GUINT_TO_POINTER (34));
|
|
|
|
if (is_last_ref)
|
|
{
|
|
toggle_ref3_weakened = TRUE;
|
|
g_object_remove_toggle_ref (object, toggle_ref3, GUINT_TO_POINTER (34));
|
|
}
|
|
else
|
|
toggle_ref3_strengthened = TRUE;
|
|
}
|
|
|
|
int
|
|
main (int argc,
|
|
char *argv[])
|
|
{
|
|
GObject *object;
|
|
|
|
g_log_set_always_fatal (g_log_set_always_fatal (G_LOG_FATAL_MASK) |
|
|
G_LOG_LEVEL_WARNING |
|
|
G_LOG_LEVEL_CRITICAL);
|
|
|
|
/* Test basic weak reference operation
|
|
*/
|
|
global_object = object = g_object_new (TEST_TYPE_OBJECT, NULL);
|
|
|
|
g_object_weak_ref (object, weak_ref1, GUINT_TO_POINTER (42));
|
|
|
|
clear_flags ();
|
|
g_object_unref (object);
|
|
g_assert (weak_ref1_notified == TRUE);
|
|
g_assert (object_destroyed == TRUE);
|
|
|
|
/* Test two weak references at once
|
|
*/
|
|
global_object = object = g_object_new (TEST_TYPE_OBJECT, NULL);
|
|
|
|
g_object_weak_ref (object, weak_ref1, GUINT_TO_POINTER (42));
|
|
g_object_weak_ref (object, weak_ref2, GUINT_TO_POINTER (24));
|
|
|
|
clear_flags ();
|
|
g_object_unref (object);
|
|
g_assert (weak_ref1_notified == TRUE);
|
|
g_assert (weak_ref2_notified == TRUE);
|
|
g_assert (object_destroyed == TRUE);
|
|
|
|
/* Test remove weak references
|
|
*/
|
|
global_object = object = g_object_new (TEST_TYPE_OBJECT, NULL);
|
|
|
|
g_object_weak_ref (object, weak_ref1, GUINT_TO_POINTER (42));
|
|
g_object_weak_ref (object, weak_ref2, GUINT_TO_POINTER (24));
|
|
g_object_weak_unref (object, weak_ref1, GUINT_TO_POINTER (42));
|
|
|
|
clear_flags ();
|
|
g_object_unref (object);
|
|
g_assert (weak_ref1_notified == FALSE);
|
|
g_assert (weak_ref2_notified == TRUE);
|
|
g_assert (object_destroyed == TRUE);
|
|
|
|
/* Test basic toggle reference operation
|
|
*/
|
|
global_object = object = g_object_new (TEST_TYPE_OBJECT, NULL);
|
|
|
|
g_object_add_toggle_ref (object, toggle_ref1, GUINT_TO_POINTER (42));
|
|
|
|
clear_flags ();
|
|
g_object_unref (object);
|
|
g_assert (toggle_ref1_weakened == TRUE);
|
|
g_assert (toggle_ref1_strengthened == FALSE);
|
|
g_assert (object_destroyed == FALSE);
|
|
|
|
clear_flags ();
|
|
g_object_ref (object);
|
|
g_assert (toggle_ref1_weakened == FALSE);
|
|
g_assert (toggle_ref1_strengthened == TRUE);
|
|
g_assert (object_destroyed == FALSE);
|
|
|
|
g_object_unref (object);
|
|
|
|
clear_flags ();
|
|
g_object_remove_toggle_ref (object, toggle_ref1, GUINT_TO_POINTER (42));
|
|
g_assert (toggle_ref1_weakened == FALSE);
|
|
g_assert (toggle_ref1_strengthened == FALSE);
|
|
g_assert (object_destroyed == TRUE);
|
|
|
|
global_object = object = g_object_new (TEST_TYPE_OBJECT, NULL);
|
|
|
|
/* Test two toggle references at once
|
|
*/
|
|
g_object_add_toggle_ref (object, toggle_ref1, GUINT_TO_POINTER (42));
|
|
g_object_add_toggle_ref (object, toggle_ref2, GUINT_TO_POINTER (24));
|
|
|
|
clear_flags ();
|
|
g_object_unref (object);
|
|
g_assert (toggle_ref1_weakened == FALSE);
|
|
g_assert (toggle_ref1_strengthened == FALSE);
|
|
g_assert (toggle_ref2_weakened == FALSE);
|
|
g_assert (toggle_ref2_strengthened == FALSE);
|
|
g_assert (object_destroyed == FALSE);
|
|
|
|
clear_flags ();
|
|
g_object_remove_toggle_ref (object, toggle_ref1, GUINT_TO_POINTER (42));
|
|
g_assert (toggle_ref1_weakened == FALSE);
|
|
g_assert (toggle_ref1_strengthened == FALSE);
|
|
g_assert (toggle_ref2_weakened == TRUE);
|
|
g_assert (toggle_ref2_strengthened == FALSE);
|
|
g_assert (object_destroyed == FALSE);
|
|
|
|
clear_flags ();
|
|
/* Check that removing a toggle ref with %NULL data works fine. */
|
|
g_object_remove_toggle_ref (object, toggle_ref2, NULL);
|
|
g_assert (toggle_ref1_weakened == FALSE);
|
|
g_assert (toggle_ref1_strengthened == FALSE);
|
|
g_assert (toggle_ref2_weakened == FALSE);
|
|
g_assert (toggle_ref2_strengthened == FALSE);
|
|
g_assert (object_destroyed == TRUE);
|
|
|
|
/* Test a toggle reference that removes itself
|
|
*/
|
|
global_object = object = g_object_new (TEST_TYPE_OBJECT, NULL);
|
|
|
|
g_object_add_toggle_ref (object, toggle_ref3, GUINT_TO_POINTER (34));
|
|
|
|
clear_flags ();
|
|
g_object_unref (object);
|
|
g_assert (toggle_ref3_weakened == TRUE);
|
|
g_assert (toggle_ref3_strengthened == FALSE);
|
|
g_assert (object_destroyed == TRUE);
|
|
|
|
return 0;
|
|
}
|