mirror of
				https://gitlab.gnome.org/GNOME/glib.git
				synced 2025-11-04 01:58:54 +01:00 
			
		
		
		
	As with the previous commit, allow them to be smoketested when run without `-m perf`. Signed-off-by: Philip Withnall <pwithnall@endlessos.org>
		
			
				
	
	
		
			313 lines
		
	
	
		
			7.2 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
			
		
		
	
	
			313 lines
		
	
	
		
			7.2 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
#include <glib-object.h>
 | 
						|
 | 
						|
typedef struct {
 | 
						|
  GObject instance;
 | 
						|
} MyObj;
 | 
						|
 | 
						|
typedef struct {
 | 
						|
  GObjectClass parent_class;
 | 
						|
} MyObjClass;
 | 
						|
 | 
						|
enum {
 | 
						|
  SIGNAL1,
 | 
						|
  SIGNAL2,
 | 
						|
  LAST_SIGNAL
 | 
						|
};
 | 
						|
 | 
						|
guint signals[LAST_SIGNAL];
 | 
						|
 | 
						|
GType my_obj_get_type (void);
 | 
						|
 | 
						|
G_DEFINE_TYPE (MyObj, my_obj, G_TYPE_OBJECT)
 | 
						|
 | 
						|
static void
 | 
						|
my_obj_init (MyObj *o)
 | 
						|
{
 | 
						|
}
 | 
						|
 | 
						|
static void
 | 
						|
my_obj_class_init (MyObjClass *class)
 | 
						|
{
 | 
						|
  signals[SIGNAL1] =
 | 
						|
    g_signal_new ("signal1",
 | 
						|
                  G_TYPE_FROM_CLASS (class),
 | 
						|
                  G_SIGNAL_RUN_LAST,
 | 
						|
                  0, NULL, NULL, NULL, G_TYPE_NONE, 0);
 | 
						|
  signals[SIGNAL2] =
 | 
						|
    g_signal_new ("signal2",
 | 
						|
                  G_TYPE_FROM_CLASS (class),
 | 
						|
                  G_SIGNAL_RUN_LAST,
 | 
						|
                  0, NULL, NULL, NULL, G_TYPE_NONE, 0);
 | 
						|
}
 | 
						|
 | 
						|
static void
 | 
						|
nop (void)
 | 
						|
{
 | 
						|
}
 | 
						|
 | 
						|
static guint n_handlers = 0;
 | 
						|
 | 
						|
static void
 | 
						|
test_connect_many (void)
 | 
						|
{
 | 
						|
  MyObj *o;
 | 
						|
  gdouble time_elapsed;
 | 
						|
  guint i;
 | 
						|
 | 
						|
  o = g_object_new (my_obj_get_type (), NULL);
 | 
						|
 | 
						|
  g_test_timer_start ();
 | 
						|
 | 
						|
  for (i = 0; i < n_handlers; i++)
 | 
						|
    g_signal_connect (o, "signal1", G_CALLBACK (nop), NULL); 
 | 
						|
 | 
						|
  time_elapsed = g_test_timer_elapsed ();
 | 
						|
 | 
						|
  g_object_unref (o);
 | 
						|
 | 
						|
  g_test_minimized_result (time_elapsed, "connected %u handlers in %6.3f seconds", n_handlers, time_elapsed);
 | 
						|
}
 | 
						|
 | 
						|
static void
 | 
						|
test_disconnect_many_ordered (void)
 | 
						|
{
 | 
						|
  MyObj *o;
 | 
						|
  gulong *handlers;
 | 
						|
  gdouble time_elapsed;
 | 
						|
  guint i;
 | 
						|
 | 
						|
  handlers = g_malloc_n (n_handlers, sizeof (*handlers));
 | 
						|
  o = g_object_new (my_obj_get_type (), NULL);
 | 
						|
 | 
						|
  for (i = 0; i < n_handlers; i++)
 | 
						|
    handlers[i] = g_signal_connect (o, "signal1", G_CALLBACK (nop), NULL); 
 | 
						|
 | 
						|
  g_test_timer_start ();
 | 
						|
 | 
						|
  for (i = 0; i < n_handlers; i++)
 | 
						|
    g_signal_handler_disconnect (o, handlers[i]); 
 | 
						|
 | 
						|
  time_elapsed = g_test_timer_elapsed ();
 | 
						|
 | 
						|
  g_object_unref (o);
 | 
						|
  g_free (handlers);
 | 
						|
 | 
						|
  g_test_minimized_result (time_elapsed, "disconnected %u handlers in %6.3f seconds", n_handlers, time_elapsed);
 | 
						|
}
 | 
						|
 | 
						|
static void
 | 
						|
test_disconnect_many_inverse (void)
 | 
						|
{
 | 
						|
  MyObj *o;
 | 
						|
  gulong *handlers;
 | 
						|
  gdouble time_elapsed;
 | 
						|
  guint i;
 | 
						|
 | 
						|
  handlers = g_malloc_n (n_handlers, sizeof (*handlers));
 | 
						|
  o = g_object_new (my_obj_get_type (), NULL);
 | 
						|
 | 
						|
  for (i = 0; i < n_handlers; i++)
 | 
						|
    handlers[i] = g_signal_connect (o, "signal1", G_CALLBACK (nop), NULL); 
 | 
						|
 | 
						|
  g_test_timer_start ();
 | 
						|
 | 
						|
  for (i = n_handlers; i > 0; i--)
 | 
						|
    g_signal_handler_disconnect (o, handlers[i - 1]);
 | 
						|
 | 
						|
  time_elapsed = g_test_timer_elapsed ();
 | 
						|
 | 
						|
  g_object_unref (o);
 | 
						|
  g_free (handlers);
 | 
						|
 | 
						|
  g_test_minimized_result (time_elapsed, "disconnected %u handlers in %6.3f seconds", n_handlers, time_elapsed);
 | 
						|
}
 | 
						|
 | 
						|
static void
 | 
						|
test_disconnect_many_random (void)
 | 
						|
{
 | 
						|
  MyObj *o;
 | 
						|
  gulong *handlers;
 | 
						|
  gulong id;
 | 
						|
  gdouble time_elapsed;
 | 
						|
  guint i, j;
 | 
						|
 | 
						|
  handlers = g_malloc_n (n_handlers, sizeof (*handlers));
 | 
						|
  o = g_object_new (my_obj_get_type (), NULL);
 | 
						|
 | 
						|
  for (i = 0; i < n_handlers; i++)
 | 
						|
    handlers[i] = g_signal_connect (o, "signal1", G_CALLBACK (nop), NULL); 
 | 
						|
 | 
						|
  for (i = 0; i < n_handlers; i++)
 | 
						|
    {
 | 
						|
      j = g_test_rand_int_range (0, n_handlers);
 | 
						|
      id = handlers[i];
 | 
						|
      handlers[i] = handlers[j];
 | 
						|
      handlers[j] = id;
 | 
						|
    }
 | 
						|
 | 
						|
  g_test_timer_start ();
 | 
						|
 | 
						|
  for (i = 0; i < n_handlers; i++)
 | 
						|
    g_signal_handler_disconnect (o, handlers[i]); 
 | 
						|
 | 
						|
  time_elapsed = g_test_timer_elapsed ();
 | 
						|
 | 
						|
  g_object_unref (o);
 | 
						|
  g_free (handlers);
 | 
						|
 | 
						|
  g_test_minimized_result (time_elapsed, "disconnected %u handlers in %6.3f seconds", n_handlers, time_elapsed);
 | 
						|
}
 | 
						|
 | 
						|
static void
 | 
						|
test_disconnect_2_signals (void)
 | 
						|
{
 | 
						|
  MyObj *o;
 | 
						|
  gulong *handlers;
 | 
						|
  gulong id;
 | 
						|
  gdouble time_elapsed;
 | 
						|
  guint i, j;
 | 
						|
 | 
						|
  handlers = g_malloc_n (n_handlers, sizeof (*handlers));
 | 
						|
  o = g_object_new (my_obj_get_type (), NULL);
 | 
						|
 | 
						|
  for (i = 0; i < n_handlers; i++)
 | 
						|
    {
 | 
						|
      if (i % 2 == 0)
 | 
						|
        handlers[i] = g_signal_connect (o, "signal1", G_CALLBACK (nop), NULL); 
 | 
						|
      else
 | 
						|
        handlers[i] = g_signal_connect (o, "signal2", G_CALLBACK (nop), NULL); 
 | 
						|
    }
 | 
						|
 | 
						|
  for (i = 0; i < n_handlers; i++)
 | 
						|
    {
 | 
						|
      j = g_test_rand_int_range (0, n_handlers);
 | 
						|
      id = handlers[i];
 | 
						|
      handlers[i] = handlers[j];
 | 
						|
      handlers[j] = id;
 | 
						|
    }
 | 
						|
 | 
						|
  g_test_timer_start ();
 | 
						|
 | 
						|
  for (i = 0; i < n_handlers; i++)
 | 
						|
    g_signal_handler_disconnect (o, handlers[i]); 
 | 
						|
 | 
						|
  time_elapsed = g_test_timer_elapsed ();
 | 
						|
 | 
						|
  g_object_unref (o);
 | 
						|
  g_free (handlers);
 | 
						|
 | 
						|
  g_test_minimized_result (time_elapsed, "disconnected %u handlers in %6.3f seconds", n_handlers, time_elapsed);
 | 
						|
}
 | 
						|
 | 
						|
static void
 | 
						|
test_disconnect_2_objects (void)
 | 
						|
{
 | 
						|
  MyObj *o1, *o2, *o;
 | 
						|
  gulong *handlers;
 | 
						|
  MyObj **objects;
 | 
						|
  gulong id;
 | 
						|
  gdouble time_elapsed;
 | 
						|
  guint i, j;
 | 
						|
 | 
						|
  handlers = g_malloc_n (n_handlers, sizeof (*handlers));
 | 
						|
  objects = g_malloc_n (n_handlers, sizeof (*objects));
 | 
						|
  o1 = g_object_new (my_obj_get_type (), NULL);
 | 
						|
  o2 = g_object_new (my_obj_get_type (), NULL);
 | 
						|
 | 
						|
  for (i = 0; i < n_handlers; i++)
 | 
						|
    {
 | 
						|
      if (i % 2 == 0)
 | 
						|
        {
 | 
						|
          handlers[i] = g_signal_connect (o1, "signal1", G_CALLBACK (nop), NULL); 
 | 
						|
          objects[i] = o1;
 | 
						|
        }
 | 
						|
      else
 | 
						|
        {
 | 
						|
          handlers[i] = g_signal_connect (o2, "signal1", G_CALLBACK (nop), NULL); 
 | 
						|
          objects[i] = o2;
 | 
						|
        }
 | 
						|
    }
 | 
						|
 | 
						|
  for (i = 0; i < n_handlers; i++)
 | 
						|
    {
 | 
						|
      j = g_test_rand_int_range (0, n_handlers);
 | 
						|
      id = handlers[i];
 | 
						|
      handlers[i] = handlers[j];
 | 
						|
      handlers[j] = id;
 | 
						|
      o = objects[i];
 | 
						|
      objects[i] = objects[j];
 | 
						|
      objects[j] = o;
 | 
						|
    }
 | 
						|
 | 
						|
  g_test_timer_start ();
 | 
						|
 | 
						|
  for (i = 0; i < n_handlers; i++)
 | 
						|
    g_signal_handler_disconnect (objects[i], handlers[i]); 
 | 
						|
 | 
						|
  time_elapsed = g_test_timer_elapsed ();
 | 
						|
 | 
						|
  g_object_unref (o1);
 | 
						|
  g_object_unref (o2);
 | 
						|
  g_free (objects);
 | 
						|
  g_free (handlers);
 | 
						|
 | 
						|
  g_test_minimized_result (time_elapsed, "disconnected %u handlers in %6.3f seconds", n_handlers, time_elapsed);
 | 
						|
}
 | 
						|
 | 
						|
static void
 | 
						|
test_block_many (void)
 | 
						|
{
 | 
						|
  MyObj *o;
 | 
						|
  gulong *handlers;
 | 
						|
  gulong id;
 | 
						|
  gdouble time_elapsed;
 | 
						|
  guint i, j;
 | 
						|
 | 
						|
  handlers = g_malloc_n (n_handlers, sizeof (*handlers));
 | 
						|
  o = g_object_new (my_obj_get_type (), NULL);
 | 
						|
 | 
						|
  for (i = 0; i < n_handlers; i++)
 | 
						|
    handlers[i] = g_signal_connect (o, "signal1", G_CALLBACK (nop), NULL); 
 | 
						|
 | 
						|
  for (i = 0; i < n_handlers; i++)
 | 
						|
    {
 | 
						|
      j = g_test_rand_int_range (0, n_handlers);
 | 
						|
      id = handlers[i];
 | 
						|
      handlers[i] = handlers[j];
 | 
						|
      handlers[j] = id;
 | 
						|
    }
 | 
						|
 | 
						|
  g_test_timer_start ();
 | 
						|
 | 
						|
  for (i = 0; i < n_handlers; i++)
 | 
						|
    g_signal_handler_block (o, handlers[i]); 
 | 
						|
 | 
						|
  for (i = n_handlers; i > 0; i--)
 | 
						|
    g_signal_handler_unblock (o, handlers[i - 1]);
 | 
						|
 | 
						|
  time_elapsed = g_test_timer_elapsed ();
 | 
						|
 | 
						|
  g_object_unref (o);
 | 
						|
  g_free (handlers);
 | 
						|
 | 
						|
  g_test_minimized_result (time_elapsed, "blocked and unblocked %u handlers in %6.3f seconds", n_handlers, time_elapsed);
 | 
						|
}
 | 
						|
 | 
						|
int
 | 
						|
main (int argc, char *argv[])
 | 
						|
{
 | 
						|
  g_test_init (&argc, &argv, NULL);
 | 
						|
 | 
						|
  n_handlers = g_test_perf () ? 500000 : 1;
 | 
						|
 | 
						|
  g_test_add_func ("/signal/handler/connect-many", test_connect_many);
 | 
						|
  g_test_add_func ("/signal/handler/disconnect-many-ordered", test_disconnect_many_ordered);
 | 
						|
  g_test_add_func ("/signal/handler/disconnect-many-inverse", test_disconnect_many_inverse);
 | 
						|
  g_test_add_func ("/signal/handler/disconnect-many-random", test_disconnect_many_random);
 | 
						|
  g_test_add_func ("/signal/handler/disconnect-2-signals", test_disconnect_2_signals);
 | 
						|
  g_test_add_func ("/signal/handler/disconnect-2-objects", test_disconnect_2_objects);
 | 
						|
  g_test_add_func ("/signal/handler/block-many", test_block_many);
 | 
						|
 | 
						|
  return g_test_run ();
 | 
						|
}
 |