3 #include <glib-object.h>
6 GTypeInterface g_iface;
9 GType foo_get_type (void);
11 G_DEFINE_INTERFACE (Foo, foo, G_TYPE_OBJECT)
14 foo_default_init (FooInterface *iface)
23 GObjectClass parent_class;
27 baa_init_foo (FooInterface *iface)
31 GType baa_get_type (void);
33 G_DEFINE_TYPE_WITH_CODE (Baa, baa, G_TYPE_OBJECT,
34 G_IMPLEMENT_INTERFACE (foo_get_type (), baa_init_foo))
42 baa_class_init (BaaClass *class)
46 typedef struct _BindingSource
48 GObject parent_instance;
57 typedef struct _BindingSourceClass
59 GObjectClass parent_class;
68 PROP_SOURCE_DOUBLE_VALUE,
73 static GType binding_source_get_type (void);
74 G_DEFINE_TYPE (BindingSource, binding_source, G_TYPE_OBJECT)
77 binding_source_set_property (GObject *gobject,
82 BindingSource *source = (BindingSource *) gobject;
87 source->foo = g_value_get_int (value);
91 source->bar = g_value_get_int (value);
94 case PROP_SOURCE_DOUBLE_VALUE:
95 source->double_value = g_value_get_double (value);
98 case PROP_SOURCE_TOGGLE:
99 source->toggle = g_value_get_boolean (value);
102 case PROP_SOURCE_OBJECT:
103 source->item = g_value_get_object (value);
107 G_OBJECT_WARN_INVALID_PROPERTY_ID (gobject, prop_id, pspec);
112 binding_source_get_property (GObject *gobject,
117 BindingSource *source = (BindingSource *) gobject;
121 case PROP_SOURCE_FOO:
122 g_value_set_int (value, source->foo);
125 case PROP_SOURCE_BAR:
126 g_value_set_int (value, source->bar);
129 case PROP_SOURCE_DOUBLE_VALUE:
130 g_value_set_double (value, source->double_value);
133 case PROP_SOURCE_TOGGLE:
134 g_value_set_boolean (value, source->toggle);
137 case PROP_SOURCE_OBJECT:
138 g_value_set_object (value, source->item);
142 G_OBJECT_WARN_INVALID_PROPERTY_ID (gobject, prop_id, pspec);
147 binding_source_class_init (BindingSourceClass *klass)
149 GObjectClass *gobject_class = G_OBJECT_CLASS (klass);
151 gobject_class->set_property = binding_source_set_property;
152 gobject_class->get_property = binding_source_get_property;
154 g_object_class_install_property (gobject_class, PROP_SOURCE_FOO,
155 g_param_spec_int ("foo", "Foo", "Foo",
159 g_object_class_install_property (gobject_class, PROP_SOURCE_BAR,
160 g_param_spec_int ("bar", "Bar", "Bar",
164 g_object_class_install_property (gobject_class, PROP_SOURCE_DOUBLE_VALUE,
165 g_param_spec_double ("double-value", "Value", "Value",
169 g_object_class_install_property (gobject_class, PROP_SOURCE_TOGGLE,
170 g_param_spec_boolean ("toggle", "Toggle", "Toggle",
173 g_object_class_install_property (gobject_class, PROP_SOURCE_OBJECT,
174 g_param_spec_object ("object", "Object", "Object",
180 binding_source_init (BindingSource *self)
184 typedef struct _BindingTarget
186 GObject parent_instance;
189 gdouble double_value;
194 typedef struct _BindingTargetClass
196 GObjectClass parent_class;
197 } BindingTargetClass;
204 PROP_TARGET_DOUBLE_VALUE,
209 static GType binding_target_get_type (void);
210 G_DEFINE_TYPE (BindingTarget, binding_target, G_TYPE_OBJECT)
213 binding_target_set_property (GObject *gobject,
218 BindingTarget *target = (BindingTarget *) gobject;
222 case PROP_TARGET_BAR:
223 target->bar = g_value_get_int (value);
226 case PROP_TARGET_DOUBLE_VALUE:
227 target->double_value = g_value_get_double (value);
230 case PROP_TARGET_TOGGLE:
231 target->toggle = g_value_get_boolean (value);
234 case PROP_TARGET_FOO:
235 target->foo = g_value_get_object (value);
239 G_OBJECT_WARN_INVALID_PROPERTY_ID (gobject, prop_id, pspec);
244 binding_target_get_property (GObject *gobject,
249 BindingTarget *target = (BindingTarget *) gobject;
253 case PROP_TARGET_BAR:
254 g_value_set_int (value, target->bar);
257 case PROP_TARGET_DOUBLE_VALUE:
258 g_value_set_double (value, target->double_value);
261 case PROP_TARGET_TOGGLE:
262 g_value_set_boolean (value, target->toggle);
265 case PROP_TARGET_FOO:
266 g_value_set_object (value, target->foo);
270 G_OBJECT_WARN_INVALID_PROPERTY_ID (gobject, prop_id, pspec);
275 binding_target_class_init (BindingTargetClass *klass)
277 GObjectClass *gobject_class = G_OBJECT_CLASS (klass);
279 gobject_class->set_property = binding_target_set_property;
280 gobject_class->get_property = binding_target_get_property;
282 g_object_class_install_property (gobject_class, PROP_TARGET_BAR,
283 g_param_spec_int ("bar", "Bar", "Bar",
287 g_object_class_install_property (gobject_class, PROP_TARGET_DOUBLE_VALUE,
288 g_param_spec_double ("double-value", "Value", "Value",
292 g_object_class_install_property (gobject_class, PROP_TARGET_TOGGLE,
293 g_param_spec_boolean ("toggle", "Toggle", "Toggle",
296 g_object_class_install_property (gobject_class, PROP_TARGET_FOO,
297 g_param_spec_object ("foo", "Foo", "Foo",
303 binding_target_init (BindingTarget *self)
308 celsius_to_fahrenheit (GBinding *binding,
309 const GValue *from_value,
311 gpointer user_data G_GNUC_UNUSED)
313 gdouble celsius, fahrenheit;
315 g_assert_true (G_VALUE_HOLDS (from_value, G_TYPE_DOUBLE));
316 g_assert_true (G_VALUE_HOLDS (to_value, G_TYPE_DOUBLE));
318 celsius = g_value_get_double (from_value);
319 fahrenheit = (9 * celsius / 5) + 32.0;
321 if (g_test_verbose ())
322 g_printerr ("Converting %.2fC to %.2fF\n", celsius, fahrenheit);
324 g_value_set_double (to_value, fahrenheit);
330 fahrenheit_to_celsius (GBinding *binding,
331 const GValue *from_value,
333 gpointer user_data G_GNUC_UNUSED)
335 gdouble celsius, fahrenheit;
337 g_assert_true (G_VALUE_HOLDS (from_value, G_TYPE_DOUBLE));
338 g_assert_true (G_VALUE_HOLDS (to_value, G_TYPE_DOUBLE));
340 fahrenheit = g_value_get_double (from_value);
341 celsius = 5 * (fahrenheit - 32.0) / 9;
343 if (g_test_verbose ())
344 g_printerr ("Converting %.2fF to %.2fC\n", fahrenheit, celsius);
346 g_value_set_double (to_value, celsius);
352 binding_default (void)
354 BindingSource *source = g_object_new (binding_source_get_type (), NULL);
355 BindingTarget *target = g_object_new (binding_target_get_type (), NULL);
359 binding = g_object_bind_property (source, "foo",
363 g_object_add_weak_pointer (G_OBJECT (binding), (gpointer *) &binding);
364 tmp = g_binding_dup_source (binding);
365 g_assert_nonnull (tmp);
366 g_assert_true ((BindingSource *) tmp == source);
367 g_object_unref (tmp);
368 tmp = g_binding_dup_target (binding);
369 g_assert_nonnull (tmp);
370 g_assert_true ((BindingTarget *) tmp == target);
371 g_object_unref (tmp);
372 g_assert_cmpstr (g_binding_get_source_property (binding), ==, "foo");
373 g_assert_cmpstr (g_binding_get_target_property (binding), ==, "bar");
374 g_assert_cmpint (g_binding_get_flags (binding), ==, G_BINDING_DEFAULT);
376 g_object_set (source, "foo", 42, NULL);
377 g_assert_cmpint (source->foo, ==, target->bar);
379 g_object_set (target, "bar", 47, NULL);
380 g_assert_cmpint (source->foo, !=, target->bar);
382 g_object_unref (binding);
384 g_object_set (source, "foo", 0, NULL);
385 g_assert_cmpint (source->foo, !=, target->bar);
387 g_object_unref (source);
388 g_object_unref (target);
389 g_assert_null (binding);
393 binding_canonicalisation (void)
395 BindingSource *source = g_object_new (binding_source_get_type (), NULL);
396 BindingTarget *target = g_object_new (binding_target_get_type (), NULL);
400 g_test_summary ("Test that bindings set up with non-canonical property names work");
402 binding = g_object_bind_property (source, "double_value",
403 target, "double_value",
406 g_object_add_weak_pointer (G_OBJECT (binding), (gpointer *) &binding);
407 tmp = g_binding_dup_source (binding);
408 g_assert_nonnull (tmp);
409 g_assert_true ((BindingSource *) tmp == source);
410 g_object_unref (tmp);
411 tmp = g_binding_dup_target (binding);
412 g_assert_nonnull (tmp);
413 g_assert_true ((BindingTarget *) tmp == target);
414 g_object_unref (tmp);
415 g_assert_cmpstr (g_binding_get_source_property (binding), ==, "double-value");
416 g_assert_cmpstr (g_binding_get_target_property (binding), ==, "double-value");
417 g_assert_cmpint (g_binding_get_flags (binding), ==, G_BINDING_DEFAULT);
419 g_object_set (source, "double-value", 24.0, NULL);
420 g_assert_cmpfloat (target->double_value, ==, source->double_value);
422 g_object_set (target, "double-value", 69.0, NULL);
423 g_assert_cmpfloat (source->double_value, !=, target->double_value);
425 g_object_unref (target);
426 g_object_unref (source);
427 g_assert_null (binding);
431 binding_bidirectional (void)
433 BindingSource *source = g_object_new (binding_source_get_type (), NULL);
434 BindingTarget *target = g_object_new (binding_target_get_type (), NULL);
437 binding = g_object_bind_property (source, "foo",
439 G_BINDING_BIDIRECTIONAL);
440 g_object_add_weak_pointer (G_OBJECT (binding), (gpointer *) &binding);
442 g_object_set (source, "foo", 42, NULL);
443 g_assert_cmpint (source->foo, ==, target->bar);
445 g_object_set (target, "bar", 47, NULL);
446 g_assert_cmpint (source->foo, ==, target->bar);
448 g_object_unref (binding);
450 g_object_set (source, "foo", 0, NULL);
451 g_assert_cmpint (source->foo, !=, target->bar);
453 g_object_unref (source);
454 g_object_unref (target);
455 g_assert_null (binding);
459 data_free (gpointer data)
467 binding_transform_default (void)
469 BindingSource *source = g_object_new (binding_source_get_type (), NULL);
470 BindingTarget *target = g_object_new (binding_target_get_type (), NULL);
473 gchar *src_prop, *trg_prop;
476 binding = g_object_bind_property (source, "foo",
477 target, "double-value",
478 G_BINDING_BIDIRECTIONAL);
480 g_object_add_weak_pointer (G_OBJECT (binding), (gpointer *) &binding);
482 g_object_get (binding,
484 "source-property", &src_prop,
486 "target-property", &trg_prop,
489 g_assert_true (src == source);
490 g_assert_true (trg == target);
491 g_assert_cmpstr (src_prop, ==, "foo");
492 g_assert_cmpstr (trg_prop, ==, "double-value");
493 g_assert_cmpint (flags, ==, G_BINDING_BIDIRECTIONAL);
494 g_object_unref (src);
495 g_object_unref (trg);
499 g_object_set (source, "foo", 24, NULL);
500 g_assert_cmpfloat (target->double_value, ==, 24.0);
502 g_object_set (target, "double-value", 69.0, NULL);
503 g_assert_cmpint (source->foo, ==, 69);
505 g_object_unref (target);
506 g_object_unref (source);
507 g_assert_null (binding);
511 binding_transform (void)
513 BindingSource *source = g_object_new (binding_source_get_type (), NULL);
514 BindingTarget *target = g_object_new (binding_target_get_type (), NULL);
515 GBinding *binding G_GNUC_UNUSED;
516 gboolean unused_data = FALSE;
518 binding = g_object_bind_property_full (source, "double-value",
519 target, "double-value",
520 G_BINDING_BIDIRECTIONAL,
521 celsius_to_fahrenheit,
522 fahrenheit_to_celsius,
523 &unused_data, data_free);
525 g_object_set (source, "double-value", 24.0, NULL);
526 g_assert_cmpfloat (target->double_value, ==, ((9 * 24.0 / 5) + 32.0));
528 g_object_set (target, "double-value", 69.0, NULL);
529 g_assert_cmpfloat (source->double_value, ==, (5 * (69.0 - 32.0) / 9));
531 g_object_unref (source);
532 g_object_unref (target);
534 g_assert_true (unused_data);
538 binding_transform_closure (void)
540 BindingSource *source = g_object_new (binding_source_get_type (), NULL);
541 BindingTarget *target = g_object_new (binding_target_get_type (), NULL);
542 GBinding *binding G_GNUC_UNUSED;
543 gboolean unused_data_1 = FALSE, unused_data_2 = FALSE;
544 GClosure *c2f_clos, *f2c_clos;
546 c2f_clos = g_cclosure_new (G_CALLBACK (celsius_to_fahrenheit), &unused_data_1, (GClosureNotify) data_free);
548 f2c_clos = g_cclosure_new (G_CALLBACK (fahrenheit_to_celsius), &unused_data_2, (GClosureNotify) data_free);
550 binding = g_object_bind_property_with_closures (source, "double-value",
551 target, "double-value",
552 G_BINDING_BIDIRECTIONAL,
556 g_object_set (source, "double-value", 24.0, NULL);
557 g_assert_cmpfloat (target->double_value, ==, ((9 * 24.0 / 5) + 32.0));
559 g_object_set (target, "double-value", 69.0, NULL);
560 g_assert_cmpfloat (source->double_value, ==, (5 * (69.0 - 32.0) / 9));
562 g_object_unref (source);
563 g_object_unref (target);
565 g_assert_true (unused_data_1);
566 g_assert_true (unused_data_2);
572 BindingSource *a = g_object_new (binding_source_get_type (), NULL);
573 BindingSource *b = g_object_new (binding_source_get_type (), NULL);
574 BindingSource *c = g_object_new (binding_source_get_type (), NULL);
575 GBinding *binding_1, *binding_2;
577 g_test_bug_base ("http://bugzilla.gnome.org/");
578 g_test_bug ("621782");
581 binding_1 = g_object_bind_property (a, "foo", b, "foo", G_BINDING_BIDIRECTIONAL);
582 g_object_add_weak_pointer (G_OBJECT (binding_1), (gpointer *) &binding_1);
584 binding_2 = g_object_bind_property (b, "foo", c, "foo", G_BINDING_BIDIRECTIONAL);
585 g_object_add_weak_pointer (G_OBJECT (binding_2), (gpointer *) &binding_2);
587 /* verify the chain */
588 g_object_set (a, "foo", 42, NULL);
589 g_assert_cmpint (a->foo, ==, b->foo);
590 g_assert_cmpint (b->foo, ==, c->foo);
592 /* unbind A -> B and B -> C */
593 g_object_unref (binding_1);
594 g_assert_null (binding_1);
595 g_object_unref (binding_2);
596 g_assert_null (binding_2);
598 /* bind A -> C directly */
599 binding_2 = g_object_bind_property (a, "foo", c, "foo", G_BINDING_BIDIRECTIONAL);
601 /* verify the chain is broken */
602 g_object_set (a, "foo", 47, NULL);
603 g_assert_cmpint (a->foo, !=, b->foo);
604 g_assert_cmpint (a->foo, ==, c->foo);
612 binding_sync_create (void)
614 BindingSource *source = g_object_new (binding_source_get_type (),
617 BindingTarget *target = g_object_new (binding_target_get_type (),
622 binding = g_object_bind_property (source, "foo",
624 G_BINDING_DEFAULT | G_BINDING_SYNC_CREATE);
626 g_assert_cmpint (source->foo, ==, 42);
627 g_assert_cmpint (target->bar, ==, 42);
629 g_object_set (source, "foo", 47, NULL);
630 g_assert_cmpint (source->foo, ==, target->bar);
632 g_object_unref (binding);
634 g_object_set (target, "bar", 49, NULL);
636 binding = g_object_bind_property (source, "foo",
638 G_BINDING_BIDIRECTIONAL | G_BINDING_SYNC_CREATE);
639 g_assert_cmpint (source->foo, ==, 47);
640 g_assert_cmpint (target->bar, ==, 47);
642 g_object_unref (source);
643 g_object_unref (target);
647 binding_invert_boolean (void)
649 BindingSource *source = g_object_new (binding_source_get_type (),
652 BindingTarget *target = g_object_new (binding_target_get_type (),
657 binding = g_object_bind_property (source, "toggle",
659 G_BINDING_BIDIRECTIONAL | G_BINDING_INVERT_BOOLEAN);
661 g_assert_true (source->toggle);
662 g_assert_false (target->toggle);
664 g_object_set (source, "toggle", FALSE, NULL);
665 g_assert_false (source->toggle);
666 g_assert_true (target->toggle);
668 g_object_set (target, "toggle", FALSE, NULL);
669 g_assert_true (source->toggle);
670 g_assert_false (target->toggle);
672 g_object_unref (binding);
673 g_object_unref (source);
674 g_object_unref (target);
678 binding_same_object (void)
680 BindingSource *source = g_object_new (binding_source_get_type (),
686 binding = g_object_bind_property (source, "foo",
688 G_BINDING_BIDIRECTIONAL);
689 g_object_add_weak_pointer (G_OBJECT (binding), (gpointer *) &binding);
691 g_object_set (source, "foo", 10, NULL);
692 g_assert_cmpint (source->foo, ==, 10);
693 g_assert_cmpint (source->bar, ==, 10);
694 g_object_set (source, "bar", 30, NULL);
695 g_assert_cmpint (source->foo, ==, 30);
696 g_assert_cmpint (source->bar, ==, 30);
698 g_object_unref (source);
699 g_assert_null (binding);
703 binding_unbind (void)
705 BindingSource *source = g_object_new (binding_source_get_type (), NULL);
706 BindingTarget *target = g_object_new (binding_target_get_type (), NULL);
709 binding = g_object_bind_property (source, "foo",
712 g_object_add_weak_pointer (G_OBJECT (binding), (gpointer *) &binding);
714 g_object_set (source, "foo", 42, NULL);
715 g_assert_cmpint (source->foo, ==, target->bar);
717 g_object_set (target, "bar", 47, NULL);
718 g_assert_cmpint (source->foo, !=, target->bar);
720 g_binding_unbind (binding);
721 g_assert_null (binding);
723 g_object_set (source, "foo", 0, NULL);
724 g_assert_cmpint (source->foo, !=, target->bar);
726 g_object_unref (source);
727 g_object_unref (target);
730 /* g_binding_unbind() has a special case for this */
731 source = g_object_new (binding_source_get_type (), NULL);
732 binding = g_object_bind_property (source, "foo",
735 g_object_add_weak_pointer (G_OBJECT (binding), (gpointer *) &binding);
737 g_binding_unbind (binding);
738 g_assert_null (binding);
740 g_object_unref (source);
743 /* When source or target die, so does the binding if there is no other ref */
745 binding_unbind_weak (void)
748 BindingSource *source;
749 BindingTarget *target;
751 /* first source, then target */
752 source = g_object_new (binding_source_get_type (), NULL);
753 target = g_object_new (binding_target_get_type (), NULL);
754 binding = g_object_bind_property (source, "foo",
757 g_object_add_weak_pointer (G_OBJECT (binding), (gpointer *) &binding);
758 g_assert_nonnull (binding);
759 g_object_unref (source);
760 g_assert_null (binding);
761 g_object_unref (target);
762 g_assert_null (binding);
764 /* first target, then source */
765 source = g_object_new (binding_source_get_type (), NULL);
766 target = g_object_new (binding_target_get_type (), NULL);
767 binding = g_object_bind_property (source, "foo",
770 g_object_add_weak_pointer (G_OBJECT (binding), (gpointer *) &binding);
771 g_assert_nonnull (binding);
772 g_object_unref (target);
773 g_assert_null (binding);
774 g_object_unref (source);
775 g_assert_null (binding);
777 /* target and source are the same */
778 source = g_object_new (binding_source_get_type (), NULL);
779 binding = g_object_bind_property (source, "foo",
782 g_object_add_weak_pointer (G_OBJECT (binding), (gpointer *) &binding);
783 g_assert_nonnull (binding);
784 g_object_unref (source);
785 g_assert_null (binding);
788 /* Test that every call to unbind() after the first is a noop */
790 binding_unbind_multiple (void)
792 BindingSource *source = g_object_new (binding_source_get_type (), NULL);
793 BindingTarget *target = g_object_new (binding_target_get_type (), NULL);
799 binding = g_object_bind_property (source, "foo",
802 g_object_ref (binding);
803 g_object_add_weak_pointer (G_OBJECT (binding), (gpointer *) &binding);
804 g_assert_nonnull (binding);
806 /* this shouldn't crash */
807 for (i = 0; i < 50; i++)
809 g_binding_unbind (binding);
810 g_assert_nonnull (binding);
813 g_object_unref (binding);
814 g_assert_null (binding);
816 g_object_unref (source);
817 g_object_unref (target);
823 BindingSource *source = g_object_new (binding_source_get_type (), NULL);
824 BindingTarget *target = g_object_new (binding_target_get_type (), NULL);
827 /* double -> boolean is not supported */
828 binding = g_object_bind_property (source, "double-value",
831 g_object_add_weak_pointer (G_OBJECT (binding), (gpointer *) &binding);
833 g_test_expect_message ("GLib-GObject", G_LOG_LEVEL_WARNING,
834 "*Unable to convert*double*boolean*");
835 g_object_set (source, "double-value", 1.0, NULL);
836 g_test_assert_expected_messages ();
838 g_object_unref (source);
839 g_object_unref (target);
840 g_assert_null (binding);
844 transform_to_func (GBinding *binding,
845 const GValue *value_a,
849 if (g_value_type_compatible (G_VALUE_TYPE (value_a), G_VALUE_TYPE (value_b)))
851 g_value_copy (value_a, value_b);
855 if (g_value_type_transformable (G_VALUE_TYPE (value_a), G_VALUE_TYPE (value_b)))
857 if (g_value_transform (value_a, value_b))
865 binding_interface (void)
867 BindingSource *source = g_object_new (binding_source_get_type (), NULL);
868 BindingTarget *target = g_object_new (binding_target_get_type (), NULL);
871 GClosure *transform_to;
873 /* binding a generic object property to an interface-valued one */
874 binding = g_object_bind_property (source, "object",
878 baa = g_object_new (baa_get_type (), NULL);
879 g_object_set (source, "object", baa, NULL);
880 g_object_unref (baa);
882 g_binding_unbind (binding);
884 /* the same, with a generic marshaller */
885 transform_to = g_cclosure_new (G_CALLBACK (transform_to_func), NULL, NULL);
886 g_closure_set_marshal (transform_to, g_cclosure_marshal_generic);
887 binding = g_object_bind_property_with_closures (source, "object",
893 baa = g_object_new (baa_get_type (), NULL);
894 g_object_set (source, "object", baa, NULL);
895 g_object_unref (baa);
897 g_binding_unbind (binding);
899 g_object_unref (source);
900 g_object_unref (target);
909 gint *count; /* (atomic) */
910 } ConcurrentUnbindData;
913 concurrent_unbind_func (gpointer data)
915 ConcurrentUnbindData *unbind_data = data;
917 g_mutex_lock (unbind_data->lock);
918 g_atomic_int_inc (unbind_data->count);
919 while (*unbind_data->wait)
920 g_cond_wait (unbind_data->cond, unbind_data->lock);
921 g_mutex_unlock (unbind_data->lock);
922 g_binding_unbind (unbind_data->binding);
923 g_object_unref (unbind_data->binding);
929 binding_concurrent_unbind (void)
933 g_test_summary ("Test that unbinding from multiple threads concurrently works correctly");
935 for (i = 0; i < 50; i++)
937 BindingSource *source = g_object_new (binding_source_get_type (), NULL);
938 BindingTarget *target = g_object_new (binding_target_get_type (), NULL);
940 GQueue threads = G_QUEUE_INIT;
943 gboolean wait = TRUE;
944 gint count = 0; /* (atomic) */
945 ConcurrentUnbindData *data;
947 g_mutex_init (&lock);
950 binding = g_object_bind_property (source, "foo",
952 G_BINDING_BIDIRECTIONAL);
953 g_object_ref (binding);
955 for (j = 0; j < 10; j++)
957 data = g_new0 (ConcurrentUnbindData, 1);
959 data->binding = g_object_ref (binding);
963 data->count = &count;
965 data->thread = g_thread_new ("binding-concurrent", concurrent_unbind_func, data);
966 g_queue_push_tail (&threads, data);
969 /* wait until all threads are started */
970 while (g_atomic_int_get (&count) < 10)
973 g_mutex_lock (&lock);
975 g_cond_broadcast (&cond);
976 g_mutex_unlock (&lock);
978 while ((data = g_queue_pop_head (&threads)))
980 g_thread_join (data->thread);
984 g_mutex_clear (&lock);
985 g_cond_clear (&cond);
987 g_object_unref (binding);
988 g_object_unref (source);
989 g_object_unref (target);
997 gint *count; /* (atomic) */
999 } ConcurrentFinalizeData;
1002 concurrent_finalize_func (gpointer data)
1004 ConcurrentFinalizeData *finalize_data = data;
1006 g_mutex_lock (finalize_data->lock);
1007 g_atomic_int_inc (finalize_data->count);
1008 while (*finalize_data->wait)
1009 g_cond_wait (finalize_data->cond, finalize_data->lock);
1010 g_mutex_unlock (finalize_data->lock);
1011 g_object_unref (finalize_data->object);
1012 g_free (finalize_data);
1018 binding_concurrent_finalizing (void)
1022 g_test_summary ("Test that finalizing source/target from multiple threads concurrently works correctly");
1024 for (i = 0; i < 50; i++)
1026 BindingSource *source = g_object_new (binding_source_get_type (), NULL);
1027 BindingTarget *target = g_object_new (binding_target_get_type (), NULL);
1031 gboolean wait = TRUE;
1032 ConcurrentFinalizeData *data;
1033 GThread *source_thread, *target_thread;
1034 gint count = 0; /* (atomic) */
1036 g_mutex_init (&lock);
1037 g_cond_init (&cond);
1039 binding = g_object_bind_property (source, "foo",
1041 G_BINDING_BIDIRECTIONAL);
1042 g_object_ref (binding);
1044 data = g_new0 (ConcurrentFinalizeData, 1);
1045 data->object = (GObject *) source;
1049 data->count = &count;
1050 source_thread = g_thread_new ("binding-concurrent", concurrent_finalize_func, data);
1052 data = g_new0 (ConcurrentFinalizeData, 1);
1053 data->object = (GObject *) target;
1057 data->count = &count;
1058 target_thread = g_thread_new ("binding-concurrent", concurrent_finalize_func, data);
1060 /* wait until all threads are started */
1061 while (g_atomic_int_get (&count) < 2)
1064 g_mutex_lock (&lock);
1066 g_cond_broadcast (&cond);
1067 g_mutex_unlock (&lock);
1069 g_thread_join (source_thread);
1070 g_thread_join (target_thread);
1072 g_mutex_clear (&lock);
1073 g_cond_clear (&cond);
1075 g_object_unref (binding);
1080 main (int argc, char *argv[])
1082 g_test_init (&argc, &argv, NULL);
1084 g_test_bug_base ("https://gitlab.gnome.org/GNOME/glib/issues/");
1086 g_test_add_func ("/binding/default", binding_default);
1087 g_test_add_func ("/binding/canonicalisation", binding_canonicalisation);
1088 g_test_add_func ("/binding/bidirectional", binding_bidirectional);
1089 g_test_add_func ("/binding/transform", binding_transform);
1090 g_test_add_func ("/binding/transform-default", binding_transform_default);
1091 g_test_add_func ("/binding/transform-closure", binding_transform_closure);
1092 g_test_add_func ("/binding/chain", binding_chain);
1093 g_test_add_func ("/binding/sync-create", binding_sync_create);
1094 g_test_add_func ("/binding/invert-boolean", binding_invert_boolean);
1095 g_test_add_func ("/binding/same-object", binding_same_object);
1096 g_test_add_func ("/binding/unbind", binding_unbind);
1097 g_test_add_func ("/binding/unbind-weak", binding_unbind_weak);
1098 g_test_add_func ("/binding/unbind-multiple", binding_unbind_multiple);
1099 g_test_add_func ("/binding/fail", binding_fail);
1100 g_test_add_func ("/binding/interface", binding_interface);
1101 g_test_add_func ("/binding/concurrent-unbind", binding_concurrent_unbind);
1102 g_test_add_func ("/binding/concurrent-finalizing", binding_concurrent_finalizing);
1104 return g_test_run ();