mirror of
				https://gitlab.gnome.org/GNOME/glib.git
				synced 2025-10-31 16:32:18 +01:00 
			
		
		
		
	There’s a painful inconsistency in the types of the
`g_{test_rand,random,rand}_int{,_range}()` functions, which vary
arbitrarily between `gint32` and `guint32`.
Unfortunately since those functions mention `int` explicitly in the name
(and then some of them return an `unsigned` integer), I don’t see a way
to make the APIs consistent without significant deprecations or
additions.
So, for the moment, to fix various `-Wsign-conversion` warnings, plaster
the tests with casts.
Signed-off-by: Philip Withnall <pwithnall@gnome.org>
Helps: #3405
		
	
		
			
				
	
	
		
			314 lines
		
	
	
		
			7.2 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
			
		
		
	
	
			314 lines
		
	
	
		
			7.2 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
| #include <glib.h>
 | |
| #include <glib-object.h>
 | |
| 
 | |
| #ifdef G_OS_UNIX
 | |
| #include <unistd.h>
 | |
| #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 = (int) 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 ();
 | |
| }
 |