#include #include #ifdef G_OS_UNIX #include #endif #define G_TYPE_TEST (my_test_get_type ()) #define MY_TEST(test) (G_TYPE_CHECK_INSTANCE_CAST ((test), G_TYPE_TEST, GTest)) #define MY_IS_TEST(test) (G_TYPE_CHECK_INSTANCE_TYPE ((test), G_TYPE_TEST)) #define MY_TEST_CLASS(tclass) (G_TYPE_CHECK_CLASS_CAST ((tclass), G_TYPE_TEST, GTestClass)) #define MY_IS_TEST_CLASS(tclass) (G_TYPE_CHECK_CLASS_TYPE ((tclass), G_TYPE_TEST)) #define MY_TEST_GET_CLASS(test) (G_TYPE_INSTANCE_GET_CLASS ((test), G_TYPE_TEST, GTestClass)) typedef struct _GTest GTest; typedef struct _GTestClass GTestClass; struct _GTest { GObject object; gint value; }; struct _GTestClass { GObjectClass parent_class; void (*test_signal1) (GTest * test, gint an_int); void (*test_signal2) (GTest * test, gint an_int); gchar * (*test_signal3) (GTest * test, gint an_int); }; static GType my_test_get_type (void); static gboolean stopping; /* Element signals and args */ enum { TEST_SIGNAL1, TEST_SIGNAL2, TEST_SIGNAL3, /* add more above */ LAST_SIGNAL }; enum { ARG_0, ARG_TEST_PROP }; static void my_test_class_init (GTestClass * klass); static void my_test_init (GTest * test); static void my_test_dispose (GObject * object); static void signal2_handler (GTest * test, gint anint); static gchar * signal3_handler (GTest * test, gint anint); static void my_test_set_property (GObject * object, guint prop_id, const GValue * value, GParamSpec * pspec); static void my_test_get_property (GObject * object, guint prop_id, GValue * value, GParamSpec * pspec); static GObjectClass *parent_class = NULL; static guint my_test_signals[LAST_SIGNAL] = { 0 }; static GType my_test_get_type (void) { static GType test_type = 0; if (!test_type) { const GTypeInfo test_info = { sizeof (GTestClass), NULL, NULL, (GClassInitFunc) my_test_class_init, NULL, NULL, sizeof (GTest), 0, (GInstanceInitFunc) my_test_init, NULL }; test_type = g_type_register_static (G_TYPE_OBJECT, "GTest", &test_info, 0); } return test_type; } static void my_test_class_init (GTestClass * klass) { GObjectClass *gobject_class; gobject_class = (GObjectClass *) klass; parent_class = g_type_class_ref (G_TYPE_OBJECT); gobject_class->dispose = my_test_dispose; gobject_class->set_property = my_test_set_property; gobject_class->get_property = my_test_get_property; my_test_signals[TEST_SIGNAL1] = g_signal_new ("test-signal1", G_TYPE_FROM_CLASS (klass), G_SIGNAL_RUN_LAST, G_STRUCT_OFFSET (GTestClass, test_signal1), NULL, NULL, g_cclosure_marshal_VOID__INT, G_TYPE_NONE, 1, G_TYPE_INT); my_test_signals[TEST_SIGNAL2] = g_signal_new ("test-signal2", G_TYPE_FROM_CLASS (klass), G_SIGNAL_RUN_LAST, G_STRUCT_OFFSET (GTestClass, test_signal2), NULL, NULL, g_cclosure_marshal_VOID__INT, G_TYPE_NONE, 1, G_TYPE_INT); my_test_signals[TEST_SIGNAL3] = g_signal_new ("test-signal3", G_TYPE_FROM_CLASS (klass), G_SIGNAL_RUN_LAST, G_STRUCT_OFFSET (GTestClass, test_signal3), NULL, NULL, g_cclosure_marshal_generic, G_TYPE_STRING, 1, G_TYPE_INT); g_object_class_install_property (G_OBJECT_CLASS (klass), ARG_TEST_PROP, g_param_spec_int ("test-prop", "Test Prop", "Test property", 0, 1, 0, G_PARAM_READWRITE)); klass->test_signal2 = signal2_handler; klass->test_signal3 = signal3_handler; } static void my_test_init (GTest * test) { g_test_message ("init %p\n", test); test->value = 0; } static void my_test_dispose (GObject * object) { GTest *test; test = MY_TEST (object); g_test_message ("dispose %p!\n", test); G_OBJECT_CLASS (parent_class)->dispose (object); } static void my_test_set_property (GObject * object, guint prop_id, const GValue * value, GParamSpec * pspec) { GTest *test; test = MY_TEST (object); switch (prop_id) { case ARG_TEST_PROP: test->value = g_value_get_int (value); break; default: G_OBJECT_WARN_INVALID_PROPERTY_ID (object, prop_id, pspec); break; } } static void my_test_get_property (GObject * object, guint prop_id, GValue * value, GParamSpec * pspec) { GTest *test; test = MY_TEST (object); switch (prop_id) { case ARG_TEST_PROP: g_value_set_int (value, test->value); break; default: G_OBJECT_WARN_INVALID_PROPERTY_ID (object, prop_id, pspec); break; } } static void my_test_do_signal1 (GTest * test) { g_signal_emit (G_OBJECT (test), my_test_signals[TEST_SIGNAL1], 0, 0); } static void signal2_handler (GTest * test, gint anint) { } static void my_test_do_signal2 (GTest * test) { g_signal_emit (G_OBJECT (test), my_test_signals[TEST_SIGNAL2], 0, 0); } static gchar * signal3_handler (GTest * test, gint anint) { return g_strdup ("test"); } static void my_test_do_signal3 (GTest * test) { gchar *res; g_signal_emit (G_OBJECT (test), my_test_signals[TEST_SIGNAL3], 0, 0, &res); g_assert (res); g_free (res); } static void my_test_do_prop (GTest * test) { test->value = g_random_int (); g_object_notify (G_OBJECT (test), "test-prop"); } static gpointer run_thread (GTest * test) { gint i = 1; while (!g_atomic_int_get (&stopping)) { if (TESTNUM == 1) my_test_do_signal1 (test); if (TESTNUM == 2) my_test_do_signal2 (test); if (TESTNUM == 3) my_test_do_prop (test); if (TESTNUM == 4) my_test_do_signal3 (test); if ((i++ % 10000) == 0) { g_thread_yield (); /* force context switch */ } } return NULL; } static void notify (GObject *object, GParamSpec *spec, gpointer user_data) { gint value; g_object_get (object, "test-prop", &value, NULL); if (TESTNUM != 3) g_assert_cmpint (value, ==, 0); } static void test_refcount_signals (void) { gint i; GTest *test1, *test2; GArray *test_threads; const gint n_threads = 1; test1 = g_object_new (G_TYPE_TEST, NULL); test2 = g_object_new (G_TYPE_TEST, NULL); g_signal_connect (test1, "notify::test-prop", G_CALLBACK (notify), NULL); g_signal_connect (test1, "test-signal1", G_CALLBACK (notify), NULL); g_signal_connect (test1, "test-signal2", G_CALLBACK (notify), NULL); test_threads = g_array_new (FALSE, FALSE, sizeof (GThread *)); stopping = FALSE; for (i = 0; i < n_threads; i++) { GThread *thread; thread = g_thread_new (NULL, (GThreadFunc) run_thread, test1); g_array_append_val (test_threads, thread); thread = g_thread_new (NULL, (GThreadFunc) run_thread, test2); g_array_append_val (test_threads, thread); } g_usleep (5000000); g_atomic_int_set (&stopping, TRUE); /* Join all threads */ for (i = 0; i < 2 * n_threads; i++) { GThread *thread; thread = g_array_index (test_threads, GThread *, i); g_thread_join (thread); } g_array_free (test_threads, TRUE); g_object_unref (test1); g_object_unref (test2); } int main (int argc, gchar *argv[]) { g_log_set_always_fatal (G_LOG_LEVEL_WARNING | G_LOG_LEVEL_CRITICAL | g_log_set_always_fatal (G_LOG_FATAL_MASK)); g_test_init (&argc, &argv, NULL); g_test_add_func ("/gobject/refcount/signals", test_refcount_signals); return g_test_run (); }