| #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 (); |
| } |