| /* GLib testing framework examples and tests |
| * |
| * Copyright (C) 2008-2010 Red Hat, Inc. |
| * |
| * This library is free software; you can redistribute it and/or |
| * modify it under the terms of the GNU Lesser General Public |
| * License as published by the Free Software Foundation; either |
| * version 2.1 of the License, or (at your option) any later version. |
| * |
| * This library is distributed in the hope that it will be useful, |
| * but WITHOUT ANY WARRANTY; without even the implied warranty of |
| * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU |
| * Lesser General Public License for more details. |
| * |
| * You should have received a copy of the GNU Lesser General |
| * Public License along with this library; if not, see <http://www.gnu.org/licenses/>. |
| * |
| * Author: David Zeuthen <davidz@redhat.com> |
| */ |
| |
| #include <gio/gio.h> |
| #include <unistd.h> |
| #include <string.h> |
| |
| #include "gdbus-tests.h" |
| |
| /* all tests rely on a global connection */ |
| static GDBusConnection *c = NULL; |
| |
| /* ---------------------------------------------------------------------------------------------------- */ |
| /* Ensure that signal and method replies are delivered in the right thread */ |
| /* ---------------------------------------------------------------------------------------------------- */ |
| |
| typedef struct { |
| GThread *thread; |
| GMainLoop *thread_loop; |
| guint signal_count; |
| } DeliveryData; |
| |
| static void |
| msg_cb_expect_success (GDBusConnection *connection, |
| GAsyncResult *res, |
| gpointer user_data) |
| { |
| DeliveryData *data = user_data; |
| GError *error; |
| GVariant *result; |
| |
| error = NULL; |
| result = g_dbus_connection_call_finish (connection, |
| res, |
| &error); |
| g_assert_no_error (error); |
| g_assert (result != NULL); |
| g_variant_unref (result); |
| |
| g_assert (g_thread_self () == data->thread); |
| |
| g_main_loop_quit (data->thread_loop); |
| } |
| |
| static void |
| msg_cb_expect_error_cancelled (GDBusConnection *connection, |
| GAsyncResult *res, |
| gpointer user_data) |
| { |
| DeliveryData *data = user_data; |
| GError *error; |
| GVariant *result; |
| |
| error = NULL; |
| result = g_dbus_connection_call_finish (connection, |
| res, |
| &error); |
| g_assert_error (error, G_IO_ERROR, G_IO_ERROR_CANCELLED); |
| g_assert (!g_dbus_error_is_remote_error (error)); |
| g_error_free (error); |
| g_assert (result == NULL); |
| |
| g_assert (g_thread_self () == data->thread); |
| |
| g_main_loop_quit (data->thread_loop); |
| } |
| |
| static void |
| signal_handler (GDBusConnection *connection, |
| const gchar *sender_name, |
| const gchar *object_path, |
| const gchar *interface_name, |
| const gchar *signal_name, |
| GVariant *parameters, |
| gpointer user_data) |
| { |
| DeliveryData *data = user_data; |
| |
| g_assert (g_thread_self () == data->thread); |
| |
| data->signal_count++; |
| |
| g_main_loop_quit (data->thread_loop); |
| } |
| |
| static gpointer |
| test_delivery_in_thread_func (gpointer _data) |
| { |
| GMainLoop *thread_loop; |
| GMainContext *thread_context; |
| DeliveryData data; |
| GCancellable *ca; |
| guint subscription_id; |
| GDBusConnection *priv_c; |
| GError *error; |
| |
| error = NULL; |
| |
| thread_context = g_main_context_new (); |
| thread_loop = g_main_loop_new (thread_context, FALSE); |
| g_main_context_push_thread_default (thread_context); |
| |
| data.thread = g_thread_self (); |
| data.thread_loop = thread_loop; |
| data.signal_count = 0; |
| |
| /* ---------------------------------------------------------------------------------------------------- */ |
| |
| /* |
| * Check that we get a reply to the GetId() method call. |
| */ |
| g_dbus_connection_call (c, |
| "org.freedesktop.DBus", /* bus_name */ |
| "/org/freedesktop/DBus", /* object path */ |
| "org.freedesktop.DBus", /* interface name */ |
| "GetId", /* method name */ |
| NULL, NULL, |
| G_DBUS_CALL_FLAGS_NONE, |
| -1, |
| NULL, |
| (GAsyncReadyCallback) msg_cb_expect_success, |
| &data); |
| g_main_loop_run (thread_loop); |
| |
| /* |
| * Check that we never actually send a message if the GCancellable |
| * is already cancelled - i.e. we should get #G_IO_ERROR_CANCELLED |
| * when the actual connection is not up. |
| */ |
| ca = g_cancellable_new (); |
| g_cancellable_cancel (ca); |
| g_dbus_connection_call (c, |
| "org.freedesktop.DBus", /* bus_name */ |
| "/org/freedesktop/DBus", /* object path */ |
| "org.freedesktop.DBus", /* interface name */ |
| "GetId", /* method name */ |
| NULL, NULL, |
| G_DBUS_CALL_FLAGS_NONE, |
| -1, |
| ca, |
| (GAsyncReadyCallback) msg_cb_expect_error_cancelled, |
| &data); |
| g_main_loop_run (thread_loop); |
| g_object_unref (ca); |
| |
| /* |
| * Check that cancellation works when the message is already in flight. |
| */ |
| ca = g_cancellable_new (); |
| g_dbus_connection_call (c, |
| "org.freedesktop.DBus", /* bus_name */ |
| "/org/freedesktop/DBus", /* object path */ |
| "org.freedesktop.DBus", /* interface name */ |
| "GetId", /* method name */ |
| NULL, NULL, |
| G_DBUS_CALL_FLAGS_NONE, |
| -1, |
| ca, |
| (GAsyncReadyCallback) msg_cb_expect_error_cancelled, |
| &data); |
| g_cancellable_cancel (ca); |
| g_main_loop_run (thread_loop); |
| g_object_unref (ca); |
| |
| /* |
| * Check that signals are delivered to the correct thread. |
| * |
| * First we subscribe to the signal, then we create a a private |
| * connection. This should cause a NameOwnerChanged message from |
| * the message bus. |
| */ |
| subscription_id = g_dbus_connection_signal_subscribe (c, |
| "org.freedesktop.DBus", /* sender */ |
| "org.freedesktop.DBus", /* interface */ |
| "NameOwnerChanged", /* member */ |
| "/org/freedesktop/DBus", /* path */ |
| NULL, |
| G_DBUS_SIGNAL_FLAGS_NONE, |
| signal_handler, |
| &data, |
| NULL); |
| g_assert (subscription_id != 0); |
| g_assert (data.signal_count == 0); |
| |
| priv_c = _g_bus_get_priv (G_BUS_TYPE_SESSION, NULL, &error); |
| g_assert_no_error (error); |
| g_assert (priv_c != NULL); |
| |
| g_main_loop_run (thread_loop); |
| g_assert (data.signal_count == 1); |
| |
| g_object_unref (priv_c); |
| |
| g_dbus_connection_signal_unsubscribe (c, subscription_id); |
| |
| /* ---------------------------------------------------------------------------------------------------- */ |
| |
| g_main_context_pop_thread_default (thread_context); |
| g_main_loop_unref (thread_loop); |
| g_main_context_unref (thread_context); |
| |
| return NULL; |
| } |
| |
| static void |
| test_delivery_in_thread (void) |
| { |
| GThread *thread; |
| |
| thread = g_thread_new ("deliver", |
| test_delivery_in_thread_func, |
| NULL); |
| |
| g_thread_join (thread); |
| } |
| |
| /* ---------------------------------------------------------------------------------------------------- */ |
| |
| typedef struct { |
| GDBusProxy *proxy; |
| gint msec; |
| guint num; |
| gboolean async; |
| |
| GMainLoop *thread_loop; |
| GThread *thread; |
| } SyncThreadData; |
| |
| static void |
| sleep_cb (GDBusProxy *proxy, |
| GAsyncResult *res, |
| gpointer user_data) |
| { |
| SyncThreadData *data = user_data; |
| GError *error; |
| GVariant *result; |
| |
| error = NULL; |
| result = g_dbus_proxy_call_finish (proxy, |
| res, |
| &error); |
| g_assert_no_error (error); |
| g_assert (result != NULL); |
| g_assert_cmpstr (g_variant_get_type_string (result), ==, "()"); |
| g_variant_unref (result); |
| |
| g_assert (data->thread == g_thread_self ()); |
| |
| g_main_loop_quit (data->thread_loop); |
| |
| //g_debug ("async cb (%p)", g_thread_self ()); |
| } |
| |
| static gpointer |
| test_sleep_in_thread_func (gpointer _data) |
| { |
| SyncThreadData *data = _data; |
| GMainContext *thread_context; |
| guint n; |
| |
| thread_context = g_main_context_new (); |
| data->thread_loop = g_main_loop_new (thread_context, FALSE); |
| g_main_context_push_thread_default (thread_context); |
| |
| data->thread = g_thread_self (); |
| |
| for (n = 0; n < data->num; n++) |
| { |
| if (data->async) |
| { |
| //g_debug ("invoking async (%p)", g_thread_self ()); |
| g_dbus_proxy_call (data->proxy, |
| "Sleep", |
| g_variant_new ("(i)", data->msec), |
| G_DBUS_CALL_FLAGS_NONE, |
| -1, |
| NULL, |
| (GAsyncReadyCallback) sleep_cb, |
| data); |
| g_main_loop_run (data->thread_loop); |
| if (g_test_verbose ()) |
| g_printerr ("A"); |
| //g_debug ("done invoking async (%p)", g_thread_self ()); |
| } |
| else |
| { |
| GError *error; |
| GVariant *result; |
| |
| error = NULL; |
| //g_debug ("invoking sync (%p)", g_thread_self ()); |
| result = g_dbus_proxy_call_sync (data->proxy, |
| "Sleep", |
| g_variant_new ("(i)", data->msec), |
| G_DBUS_CALL_FLAGS_NONE, |
| -1, |
| NULL, |
| &error); |
| if (g_test_verbose ()) |
| g_printerr ("S"); |
| //g_debug ("done invoking sync (%p)", g_thread_self ()); |
| g_assert_no_error (error); |
| g_assert (result != NULL); |
| g_assert_cmpstr (g_variant_get_type_string (result), ==, "()"); |
| g_variant_unref (result); |
| } |
| } |
| |
| g_main_context_pop_thread_default (thread_context); |
| g_main_loop_unref (data->thread_loop); |
| g_main_context_unref (thread_context); |
| |
| return NULL; |
| } |
| |
| static void |
| test_method_calls_on_proxy (GDBusProxy *proxy) |
| { |
| guint n; |
| |
| /* |
| * Check that multiple threads can do calls without interferring with |
| * each other. We do this by creating three threads that call the |
| * Sleep() method on the server (which handles it asynchronously, e.g. |
| * it won't block other requests) with different sleep durations and |
| * a number of times. We do this so each set of calls add up to 4000 |
| * milliseconds. |
| * |
| * The dbus test server that this code calls into uses glib timeouts |
| * to do the sleeping which have only a granularity of 1ms. It is |
| * therefore possible to lose as much as 40ms; the test could finish |
| * in slightly less than 4 seconds. |
| * |
| * We run this test twice - first with async calls in each thread, then |
| * again with sync calls |
| */ |
| |
| for (n = 0; n < 2; n++) |
| { |
| gboolean do_async; |
| GThread *thread1; |
| GThread *thread2; |
| GThread *thread3; |
| SyncThreadData data1; |
| SyncThreadData data2; |
| SyncThreadData data3; |
| GTimeVal start_time; |
| GTimeVal end_time; |
| guint elapsed_msec; |
| |
| do_async = (n == 0); |
| |
| g_get_current_time (&start_time); |
| |
| data1.proxy = proxy; |
| data1.msec = 40; |
| data1.num = 100; |
| data1.async = do_async; |
| thread1 = g_thread_new ("sleep", |
| test_sleep_in_thread_func, |
| &data1); |
| |
| data2.proxy = proxy; |
| data2.msec = 20; |
| data2.num = 200; |
| data2.async = do_async; |
| thread2 = g_thread_new ("sleep2", |
| test_sleep_in_thread_func, |
| &data2); |
| |
| data3.proxy = proxy; |
| data3.msec = 100; |
| data3.num = 40; |
| data3.async = do_async; |
| thread3 = g_thread_new ("sleep3", |
| test_sleep_in_thread_func, |
| &data3); |
| |
| g_thread_join (thread1); |
| g_thread_join (thread2); |
| g_thread_join (thread3); |
| |
| g_get_current_time (&end_time); |
| |
| elapsed_msec = ((end_time.tv_sec * G_USEC_PER_SEC + end_time.tv_usec) - |
| (start_time.tv_sec * G_USEC_PER_SEC + start_time.tv_usec)) / 1000; |
| |
| //g_debug ("Elapsed time for %s = %d msec", n == 0 ? "async" : "sync", elapsed_msec); |
| |
| /* elapsed_msec should be 4000 msec +/- change for overhead/inaccuracy */ |
| g_assert_cmpint (elapsed_msec, >=, 3950); |
| g_assert_cmpint (elapsed_msec, <, 30000); |
| |
| if (g_test_verbose ()) |
| g_printerr (" "); |
| } |
| } |
| |
| static void |
| test_method_calls_in_thread (void) |
| { |
| GDBusProxy *proxy; |
| GDBusConnection *connection; |
| GError *error; |
| |
| error = NULL; |
| connection = g_bus_get_sync (G_BUS_TYPE_SESSION, |
| NULL, |
| &error); |
| g_assert_no_error (error); |
| error = NULL; |
| proxy = g_dbus_proxy_new_sync (connection, |
| G_DBUS_PROXY_FLAGS_NONE, |
| NULL, /* GDBusInterfaceInfo */ |
| "com.example.TestService", /* name */ |
| "/com/example/TestObject", /* object path */ |
| "com.example.Frob", /* interface */ |
| NULL, /* GCancellable */ |
| &error); |
| g_assert_no_error (error); |
| |
| test_method_calls_on_proxy (proxy); |
| |
| g_object_unref (proxy); |
| g_object_unref (connection); |
| |
| if (g_test_verbose ()) |
| g_printerr ("\n"); |
| } |
| |
| #define SLEEP_MIN_USEC 1 |
| #define SLEEP_MAX_USEC 10 |
| |
| /* Can run in any thread */ |
| static void |
| ensure_connection_works (GDBusConnection *conn) |
| { |
| GVariant *v; |
| GError *error = NULL; |
| |
| v = g_dbus_connection_call_sync (conn, "org.freedesktop.DBus", |
| "/org/freedesktop/DBus", "org.freedesktop.DBus", "GetId", NULL, NULL, 0, -1, |
| NULL, &error); |
| g_assert_no_error (error); |
| g_assert (v != NULL); |
| g_assert (g_variant_is_of_type (v, G_VARIANT_TYPE ("(s)"))); |
| g_variant_unref (v); |
| } |
| |
| /** |
| * get_sync_in_thread: |
| * @data: (type guint): delay in microseconds |
| * |
| * Sleep for a short time, then get a session bus connection and call |
| * a method on it. |
| * |
| * Runs in a non-main thread. |
| * |
| * Returns: (transfer full): the connection |
| */ |
| static gpointer |
| get_sync_in_thread (gpointer data) |
| { |
| guint delay = GPOINTER_TO_UINT (data); |
| GError *error = NULL; |
| GDBusConnection *conn; |
| |
| g_usleep (delay); |
| |
| conn = g_bus_get_sync (G_BUS_TYPE_SESSION, NULL, &error); |
| g_assert_no_error (error); |
| |
| ensure_connection_works (conn); |
| |
| return conn; |
| } |
| |
| static void |
| test_threaded_singleton (void) |
| { |
| guint i, n; |
| guint unref_wins = 0; |
| guint get_wins = 0; |
| |
| if (g_test_thorough ()) |
| n = 100000; |
| else |
| n = 5000; |
| |
| for (i = 0; i < n; i++) |
| { |
| GThread *thread; |
| guint j; |
| guint unref_delay, get_delay; |
| GDBusConnection *new_conn; |
| |
| /* We want to be the last ref, so let it finish setting up */ |
| for (j = 0; j < 100; j++) |
| { |
| guint r = g_atomic_int_get (&G_OBJECT (c)->ref_count); |
| |
| if (r == 1) |
| break; |
| |
| g_debug ("run %u: refcount is %u, sleeping", i, r); |
| g_usleep (1000); |
| } |
| |
| if (j == 100) |
| g_error ("connection had too many refs"); |
| |
| if (g_test_verbose () && (i % (n/50)) == 0) |
| g_printerr ("%u%%\n", ((i * 100) / n)); |
| |
| /* Delay for a random time on each side of the race, to perturb the |
| * timing. Ideally, we want each side to win half the races; these |
| * timings are about right on smcv's laptop. |
| */ |
| unref_delay = g_random_int_range (SLEEP_MIN_USEC, SLEEP_MAX_USEC); |
| get_delay = g_random_int_range (SLEEP_MIN_USEC / 2, SLEEP_MAX_USEC / 2); |
| |
| /* One half of the race is to call g_bus_get_sync... */ |
| thread = g_thread_new ("get_sync_in_thread", get_sync_in_thread, |
| GUINT_TO_POINTER (get_delay)); |
| |
| /* ... and the other half is to unref the shared connection, which must |
| * have exactly one ref at this point |
| */ |
| g_usleep (unref_delay); |
| g_object_unref (c); |
| |
| /* Wait for the thread to run; see what it got */ |
| new_conn = g_thread_join (thread); |
| |
| /* If the thread won the race, it will have kept the same connection, |
| * and it'll have one ref |
| */ |
| if (new_conn == c) |
| { |
| get_wins++; |
| } |
| else |
| { |
| unref_wins++; |
| /* c is invalid now, but new_conn is suitable for the |
| * next round |
| */ |
| c = new_conn; |
| } |
| |
| ensure_connection_works (c); |
| } |
| |
| if (g_test_verbose ()) |
| g_printerr ("Unref won %u races; Get won %u races\n", unref_wins, get_wins); |
| } |
| |
| /* ---------------------------------------------------------------------------------------------------- */ |
| |
| int |
| main (int argc, |
| char *argv[]) |
| { |
| GError *error; |
| gint ret; |
| gchar *path; |
| |
| g_test_init (&argc, &argv, NULL); |
| |
| session_bus_up (); |
| |
| /* this is safe; testserver will exit once the bus goes away */ |
| path = g_test_build_filename (G_TEST_BUILT, "gdbus-testserver", NULL); |
| g_assert (g_spawn_command_line_async (path, NULL)); |
| g_free (path); |
| |
| ensure_gdbus_testserver_up (); |
| |
| /* Create the connection in the main thread */ |
| error = NULL; |
| c = g_bus_get_sync (G_BUS_TYPE_SESSION, NULL, &error); |
| g_assert_no_error (error); |
| g_assert (c != NULL); |
| |
| g_test_add_func ("/gdbus/delivery-in-thread", test_delivery_in_thread); |
| g_test_add_func ("/gdbus/method-calls-in-thread", test_method_calls_in_thread); |
| g_test_add_func ("/gdbus/threaded-singleton", test_threaded_singleton); |
| |
| ret = g_test_run(); |
| |
| g_object_unref (c); |
| |
| /* tear down bus */ |
| session_bus_down (); |
| |
| return ret; |
| } |