1 /* GLib testing framework examples and tests
3 * Copyright (C) 2008-2010 Red Hat, Inc.
5 * This library is free software; you can redistribute it and/or
6 * modify it under the terms of the GNU Lesser General Public
7 * License as published by the Free Software Foundation; either
8 * version 2 of the License, or (at your option) any later version.
10 * This library is distributed in the hope that it will be useful,
11 * but WITHOUT ANY WARRANTY; without even the implied warranty of
12 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
13 * Lesser General Public License for more details.
15 * You should have received a copy of the GNU Lesser General
16 * Public License along with this library; if not, write to the
17 * Free Software Foundation, Inc., 59 Temple Place, Suite 330,
18 * Boston, MA 02111-1307, USA.
20 * Author: David Zeuthen <davidz@redhat.com>
27 #include "gdbus-tests.h"
29 /* all tests rely on a global connection */
30 static GDBusConnection *c = NULL;
32 /* ---------------------------------------------------------------------------------------------------- */
33 /* Ensure that signal and method replies are delivered in the right thread */
34 /* ---------------------------------------------------------------------------------------------------- */
38 GMainLoop *thread_loop;
43 msg_cb_expect_success (GDBusConnection *connection,
47 DeliveryData *data = user_data;
52 result = g_dbus_connection_call_finish (connection,
55 g_assert_no_error (error);
56 g_assert (result != NULL);
57 g_variant_unref (result);
59 g_assert (g_thread_self () == data->thread);
61 g_main_loop_quit (data->thread_loop);
65 msg_cb_expect_error_cancelled (GDBusConnection *connection,
69 DeliveryData *data = user_data;
74 result = g_dbus_connection_call_finish (connection,
77 g_assert_error (error, G_IO_ERROR, G_IO_ERROR_CANCELLED);
78 g_assert (!g_dbus_error_is_remote_error (error));
80 g_assert (result == NULL);
82 g_assert (g_thread_self () == data->thread);
84 g_main_loop_quit (data->thread_loop);
88 signal_handler (GDBusConnection *connection,
89 const gchar *sender_name,
90 const gchar *object_path,
91 const gchar *interface_name,
92 const gchar *signal_name,
96 DeliveryData *data = user_data;
98 g_assert (g_thread_self () == data->thread);
100 data->signal_count++;
102 g_main_loop_quit (data->thread_loop);
106 test_delivery_in_thread_func (gpointer _data)
108 GMainLoop *thread_loop;
109 GMainContext *thread_context;
112 guint subscription_id;
113 GDBusConnection *priv_c;
118 thread_context = g_main_context_new ();
119 thread_loop = g_main_loop_new (thread_context, FALSE);
120 g_main_context_push_thread_default (thread_context);
122 data.thread = g_thread_self ();
123 data.thread_loop = thread_loop;
124 data.signal_count = 0;
126 /* ---------------------------------------------------------------------------------------------------- */
129 * Check that we get a reply to the GetId() method call.
131 g_dbus_connection_call (c,
132 "org.freedesktop.DBus", /* bus_name */
133 "/org/freedesktop/DBus", /* object path */
134 "org.freedesktop.DBus", /* interface name */
135 "GetId", /* method name */
137 G_DBUS_CALL_FLAGS_NONE,
140 (GAsyncReadyCallback) msg_cb_expect_success,
142 g_main_loop_run (thread_loop);
145 * Check that we never actually send a message if the GCancellable
146 * is already cancelled - i.e. we should get #G_IO_ERROR_CANCELLED
147 * when the actual connection is not up.
149 ca = g_cancellable_new ();
150 g_cancellable_cancel (ca);
151 g_dbus_connection_call (c,
152 "org.freedesktop.DBus", /* bus_name */
153 "/org/freedesktop/DBus", /* object path */
154 "org.freedesktop.DBus", /* interface name */
155 "GetId", /* method name */
157 G_DBUS_CALL_FLAGS_NONE,
160 (GAsyncReadyCallback) msg_cb_expect_error_cancelled,
162 g_main_loop_run (thread_loop);
166 * Check that cancellation works when the message is already in flight.
168 ca = g_cancellable_new ();
169 g_dbus_connection_call (c,
170 "org.freedesktop.DBus", /* bus_name */
171 "/org/freedesktop/DBus", /* object path */
172 "org.freedesktop.DBus", /* interface name */
173 "GetId", /* method name */
175 G_DBUS_CALL_FLAGS_NONE,
178 (GAsyncReadyCallback) msg_cb_expect_error_cancelled,
180 g_cancellable_cancel (ca);
181 g_main_loop_run (thread_loop);
185 * Check that signals are delivered to the correct thread.
187 * First we subscribe to the signal, then we create a a private
188 * connection. This should cause a NameOwnerChanged message from
191 subscription_id = g_dbus_connection_signal_subscribe (c,
192 "org.freedesktop.DBus", /* sender */
193 "org.freedesktop.DBus", /* interface */
194 "NameOwnerChanged", /* member */
195 "/org/freedesktop/DBus", /* path */
197 G_DBUS_SIGNAL_FLAGS_NONE,
201 g_assert (subscription_id != 0);
202 g_assert (data.signal_count == 0);
204 priv_c = _g_bus_get_priv (G_BUS_TYPE_SESSION, NULL, &error);
205 g_assert_no_error (error);
206 g_assert (priv_c != NULL);
208 g_main_loop_run (thread_loop);
209 g_assert (data.signal_count == 1);
211 g_object_unref (priv_c);
213 g_dbus_connection_signal_unsubscribe (c, subscription_id);
215 /* ---------------------------------------------------------------------------------------------------- */
217 g_main_context_pop_thread_default (thread_context);
218 g_main_loop_unref (thread_loop);
219 g_main_context_unref (thread_context);
225 test_delivery_in_thread (void)
229 thread = g_thread_new ("deliver",
230 test_delivery_in_thread_func,
233 g_thread_join (thread);
236 /* ---------------------------------------------------------------------------------------------------- */
244 GMainLoop *thread_loop;
249 sleep_cb (GDBusProxy *proxy,
253 SyncThreadData *data = user_data;
258 result = g_dbus_proxy_call_finish (proxy,
261 g_assert_no_error (error);
262 g_assert (result != NULL);
263 g_assert_cmpstr (g_variant_get_type_string (result), ==, "()");
264 g_variant_unref (result);
266 g_assert (data->thread == g_thread_self ());
268 g_main_loop_quit (data->thread_loop);
270 //g_debug ("async cb (%p)", g_thread_self ());
274 test_sleep_in_thread_func (gpointer _data)
276 SyncThreadData *data = _data;
277 GMainContext *thread_context;
280 thread_context = g_main_context_new ();
281 data->thread_loop = g_main_loop_new (thread_context, FALSE);
282 g_main_context_push_thread_default (thread_context);
284 data->thread = g_thread_self ();
286 for (n = 0; n < data->num; n++)
290 //g_debug ("invoking async (%p)", g_thread_self ());
291 g_dbus_proxy_call (data->proxy,
293 g_variant_new ("(i)", data->msec),
294 G_DBUS_CALL_FLAGS_NONE,
297 (GAsyncReadyCallback) sleep_cb,
299 g_main_loop_run (data->thread_loop);
300 if (!g_test_quiet ())
302 //g_debug ("done invoking async (%p)", g_thread_self ());
310 //g_debug ("invoking sync (%p)", g_thread_self ());
311 result = g_dbus_proxy_call_sync (data->proxy,
313 g_variant_new ("(i)", data->msec),
314 G_DBUS_CALL_FLAGS_NONE,
318 if (!g_test_quiet ())
320 //g_debug ("done invoking sync (%p)", g_thread_self ());
321 g_assert_no_error (error);
322 g_assert (result != NULL);
323 g_assert_cmpstr (g_variant_get_type_string (result), ==, "()");
324 g_variant_unref (result);
328 g_main_context_pop_thread_default (thread_context);
329 g_main_loop_unref (data->thread_loop);
330 g_main_context_unref (thread_context);
336 test_method_calls_on_proxy (GDBusProxy *proxy)
341 * Check that multiple threads can do calls without interferring with
342 * each other. We do this by creating three threads that call the
343 * Sleep() method on the server (which handles it asynchronously, e.g.
344 * it won't block other requests) with different sleep durations and
345 * a number of times. We do this so each set of calls add up to 4000
348 * The dbus test server that this code calls into uses glib timeouts
349 * to do the sleeping which have only a granularity of 1ms. It is
350 * therefore possible to lose as much as 40ms; the test could finish
351 * in slightly less than 4 seconds.
353 * We run this test twice - first with async calls in each thread, then
354 * again with sync calls
357 for (n = 0; n < 2; n++)
363 SyncThreadData data1;
364 SyncThreadData data2;
365 SyncThreadData data3;
372 g_get_current_time (&start_time);
377 data1.async = do_async;
378 thread1 = g_thread_new ("sleep",
379 test_sleep_in_thread_func,
385 data2.async = do_async;
386 thread2 = g_thread_new ("sleep2",
387 test_sleep_in_thread_func,
393 data3.async = do_async;
394 thread3 = g_thread_new ("sleep3",
395 test_sleep_in_thread_func,
398 g_thread_join (thread1);
399 g_thread_join (thread2);
400 g_thread_join (thread3);
402 g_get_current_time (&end_time);
404 elapsed_msec = ((end_time.tv_sec * G_USEC_PER_SEC + end_time.tv_usec) -
405 (start_time.tv_sec * G_USEC_PER_SEC + start_time.tv_usec)) / 1000;
407 //g_debug ("Elapsed time for %s = %d msec", n == 0 ? "async" : "sync", elapsed_msec);
409 /* elapsed_msec should be 4000 msec +/- change for overhead/inaccuracy */
410 g_assert_cmpint (elapsed_msec, >=, 3950);
411 g_assert_cmpint (elapsed_msec, <, 8000);
413 if (!g_test_quiet ())
419 test_method_calls_in_thread (void)
422 GDBusConnection *connection;
427 connection = g_bus_get_sync (G_BUS_TYPE_SESSION,
430 g_assert_no_error (error);
432 proxy = g_dbus_proxy_new_sync (connection,
433 G_DBUS_PROXY_FLAGS_NONE,
434 NULL, /* GDBusInterfaceInfo */
435 "com.example.TestService", /* name */
436 "/com/example/TestObject", /* object path */
437 "com.example.Frob", /* interface */
438 NULL, /* GCancellable */
440 g_assert_no_error (error);
442 name_owner = g_dbus_proxy_get_name_owner (proxy);
443 g_assert_cmpstr (name_owner, !=, NULL);
446 test_method_calls_on_proxy (proxy);
448 g_object_unref (proxy);
449 g_object_unref (connection);
451 if (!g_test_quiet ())
455 #define SLEEP_MIN_USEC 1
456 #define SLEEP_MAX_USEC 10
458 /* Can run in any thread */
460 ensure_connection_works (GDBusConnection *conn)
463 GError *error = NULL;
465 v = g_dbus_connection_call_sync (conn, "org.freedesktop.DBus",
466 "/org/freedesktop/DBus", "org.freedesktop.DBus", "GetId", NULL, NULL, 0, -1,
468 g_assert_no_error (error);
469 g_assert (v != NULL);
470 g_assert (g_variant_is_of_type (v, G_VARIANT_TYPE ("(s)")));
475 * get_sync_in_thread:
476 * @data: (type guint): delay in microseconds
478 * Sleep for a short time, then get a session bus connection and call
481 * Runs in a non-main thread.
483 * Returns: (transfer full): the connection
486 get_sync_in_thread (gpointer data)
488 guint delay = GPOINTER_TO_UINT (data);
489 GError *error = NULL;
490 GDBusConnection *conn;
494 conn = g_bus_get_sync (G_BUS_TYPE_SESSION, NULL, &error);
495 g_assert_no_error (error);
497 ensure_connection_works (conn);
503 test_threaded_singleton (void)
506 guint unref_wins = 0;
509 if (g_test_thorough ())
514 for (i = 0; i < n; i++)
518 guint unref_delay, get_delay;
519 GDBusConnection *new_conn;
521 /* We want to be the last ref, so let it finish setting up */
522 for (j = 0; j < 100; j++)
524 guint r = g_atomic_int_get (&G_OBJECT (c)->ref_count);
529 g_debug ("run %u: refcount is %u, sleeping", i, r);
534 g_error ("connection had too many refs");
536 if (g_test_verbose () && (i % (n/50)) == 0)
537 g_print ("%u%%\n", ((i * 100) / n));
539 /* Delay for a random time on each side of the race, to perturb the
540 * timing. Ideally, we want each side to win half the races; these
541 * timings are about right on smcv's laptop.
543 unref_delay = g_random_int_range (SLEEP_MIN_USEC, SLEEP_MAX_USEC);
544 get_delay = g_random_int_range (SLEEP_MIN_USEC / 2, SLEEP_MAX_USEC / 2);
546 /* One half of the race is to call g_bus_get_sync... */
547 thread = g_thread_new ("get_sync_in_thread", get_sync_in_thread,
548 GUINT_TO_POINTER (get_delay));
550 /* ... and the other half is to unref the shared connection, which must
551 * have exactly one ref at this point
553 g_usleep (unref_delay);
556 /* Wait for the thread to run; see what it got */
557 new_conn = g_thread_join (thread);
559 /* If the thread won the race, it will have kept the same connection,
560 * and it'll have one ref
569 /* c is invalid now, but new_conn is suitable for the
575 ensure_connection_works (c);
578 if (g_test_verbose ())
579 g_print ("Unref won %u races; Get won %u races\n", unref_wins, get_wins);
582 /* ---------------------------------------------------------------------------------------------------- */
592 g_test_init (&argc, &argv, NULL);
596 /* this is safe; testserver will exit once the bus goes away */
597 path = g_test_build_filename (G_TEST_BUILT, "gdbus-testserver", NULL);
598 g_assert (g_spawn_command_line_async (path, NULL));
601 /* wait for the service to come up */
604 /* Create the connection in the main thread */
606 c = g_bus_get_sync (G_BUS_TYPE_SESSION, NULL, &error);
607 g_assert_no_error (error);
608 g_assert (c != NULL);
610 g_test_add_func ("/gdbus/delivery-in-thread", test_delivery_in_thread);
611 g_test_add_func ("/gdbus/method-calls-in-thread", test_method_calls_in_thread);
612 g_test_add_func ("/gdbus/threaded-singleton", test_threaded_singleton);