2010-05-06 20:13:59 +02:00
|
|
|
/* GLib testing framework examples and tests
|
|
|
|
*
|
2010-05-10 14:07:28 +02:00
|
|
|
* Copyright (C) 2008-2010 Red Hat, Inc.
|
2010-05-06 20:13:59 +02:00
|
|
|
*
|
|
|
|
* 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
|
2017-05-27 17:19:21 +02:00
|
|
|
* version 2.1 of the License, or (at your option) any later version.
|
2010-05-06 20:13:59 +02:00
|
|
|
*
|
|
|
|
* 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
|
2014-01-23 12:58:29 +01:00
|
|
|
* Public License along with this library; if not, see <http://www.gnu.org/licenses/>.
|
2010-05-06 20:13:59 +02:00
|
|
|
*
|
|
|
|
* Author: David Zeuthen <davidz@redhat.com>
|
|
|
|
*/
|
|
|
|
|
|
|
|
#include <gio/gio.h>
|
2018-09-09 14:46:33 +02:00
|
|
|
#ifndef _MSC_VER
|
2010-05-06 20:13:59 +02:00
|
|
|
#include <unistd.h>
|
2018-09-09 14:46:33 +02:00
|
|
|
#endif
|
2010-05-06 20:13:59 +02:00
|
|
|
|
|
|
|
#include "gdbus-tests.h"
|
|
|
|
|
|
|
|
/* ---------------------------------------------------------------------------------------------------- */
|
|
|
|
|
|
|
|
typedef struct
|
|
|
|
{
|
|
|
|
GMainLoop *loop;
|
|
|
|
gboolean timed_out;
|
|
|
|
} PropertyNotifyData;
|
|
|
|
|
|
|
|
static void
|
|
|
|
on_property_notify (GObject *object,
|
|
|
|
GParamSpec *pspec,
|
|
|
|
gpointer user_data)
|
|
|
|
{
|
|
|
|
PropertyNotifyData *data = user_data;
|
|
|
|
g_main_loop_quit (data->loop);
|
|
|
|
}
|
|
|
|
|
|
|
|
static gboolean
|
|
|
|
on_property_notify_timeout (gpointer user_data)
|
|
|
|
{
|
|
|
|
PropertyNotifyData *data = user_data;
|
|
|
|
data->timed_out = TRUE;
|
|
|
|
g_main_loop_quit (data->loop);
|
2020-08-03 16:53:12 +02:00
|
|
|
return G_SOURCE_CONTINUE;
|
2010-05-06 20:13:59 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
gboolean
|
|
|
|
_g_assert_property_notify_run (gpointer object,
|
|
|
|
const gchar *property_name)
|
|
|
|
{
|
|
|
|
gchar *s;
|
|
|
|
gulong handler_id;
|
|
|
|
guint timeout_id;
|
|
|
|
PropertyNotifyData data;
|
|
|
|
|
2011-04-11 16:51:37 +02:00
|
|
|
data.loop = g_main_loop_new (g_main_context_get_thread_default (), FALSE);
|
2010-05-06 20:13:59 +02:00
|
|
|
data.timed_out = FALSE;
|
|
|
|
s = g_strdup_printf ("notify::%s", property_name);
|
|
|
|
handler_id = g_signal_connect (object,
|
|
|
|
s,
|
|
|
|
G_CALLBACK (on_property_notify),
|
|
|
|
&data);
|
|
|
|
g_free (s);
|
2013-02-03 16:21:20 +01:00
|
|
|
timeout_id = g_timeout_add_seconds (30,
|
|
|
|
on_property_notify_timeout,
|
|
|
|
&data);
|
2010-05-06 20:13:59 +02:00
|
|
|
g_main_loop_run (data.loop);
|
|
|
|
g_signal_handler_disconnect (object, handler_id);
|
|
|
|
g_source_remove (timeout_id);
|
|
|
|
g_main_loop_unref (data.loop);
|
|
|
|
|
|
|
|
return data.timed_out;
|
|
|
|
}
|
|
|
|
|
2014-03-31 12:06:05 +02:00
|
|
|
static gboolean
|
|
|
|
_give_up (gpointer data)
|
|
|
|
{
|
|
|
|
g_error ("%s", (const gchar *) data);
|
2020-08-03 16:53:12 +02:00
|
|
|
g_return_val_if_reached (G_SOURCE_CONTINUE);
|
2014-03-31 12:06:05 +02:00
|
|
|
}
|
|
|
|
|
tests: Wait until unwatching the gdbus-testserver name has completed
Previously, the code in `ensure_gdbus_testserver_up()` created a proxy
object and watched its `name-owner` to see when the
`com.example.TestService` name appeared.
This ended up subscribing to three signals (one of them for name
ownership, and two unused for properties of the proxy), and was racy. In
particular, the `name-owner` property could be set before all D-Bus
messages had been processed — it could have been derived from getting
the owner of the name, for example.
This left unprocessed messages hanging around in the `context`, but that
context was never iterated again, which essentially leaked the
references held by those messages. That included a reference to the
`GDBusConnection`.
The first part of the fix is to simplify the code to use
`g_bus_watch_name_on_connection()`, so there’s only one signal
subscription to worry about.
The second part of the fix is to use the `GDestroyNotify` callback for
the watch data to be notified of when all D-Bus traffic has been
processed and the signal unsubscription is complete. At this point, it’s
guaranteed that there are no idle callbacks pending in the
`GMainContext`, since the `GDestroyNotify` callback is the last one
invoked on the `GMainContext`.
Essentially, this commit uses the `GDestroyNotify` callback as a
synchronisation message between the D-Bus worker thread and the thread
calling `ensure_gdbus_testserver_up()`.
Signed-off-by: Philip Withnall <withnall@endlessm.com>
Fixes: #1515
2020-02-21 13:07:53 +01:00
|
|
|
typedef struct
|
|
|
|
{
|
|
|
|
GMainContext *context;
|
|
|
|
gboolean name_appeared;
|
|
|
|
gboolean unwatch_complete;
|
|
|
|
} WatchData;
|
|
|
|
|
|
|
|
static void
|
|
|
|
name_appeared_cb (GDBusConnection *connection,
|
|
|
|
const gchar *name,
|
|
|
|
const gchar *name_owner,
|
|
|
|
gpointer user_data)
|
|
|
|
{
|
|
|
|
WatchData *data = user_data;
|
|
|
|
|
|
|
|
g_assert (name_owner != NULL);
|
|
|
|
data->name_appeared = TRUE;
|
|
|
|
g_main_context_wakeup (data->context);
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
watch_free_cb (gpointer user_data)
|
|
|
|
{
|
|
|
|
WatchData *data = user_data;
|
|
|
|
|
|
|
|
data->unwatch_complete = TRUE;
|
|
|
|
g_main_context_wakeup (data->context);
|
|
|
|
}
|
|
|
|
|
2014-03-31 12:06:05 +02:00
|
|
|
void
|
2020-02-20 12:49:12 +01:00
|
|
|
ensure_gdbus_testserver_up (GDBusConnection *connection,
|
|
|
|
GMainContext *context)
|
2014-03-31 12:06:05 +02:00
|
|
|
{
|
2020-02-20 12:49:12 +01:00
|
|
|
GSource *timeout_source = NULL;
|
tests: Wait until unwatching the gdbus-testserver name has completed
Previously, the code in `ensure_gdbus_testserver_up()` created a proxy
object and watched its `name-owner` to see when the
`com.example.TestService` name appeared.
This ended up subscribing to three signals (one of them for name
ownership, and two unused for properties of the proxy), and was racy. In
particular, the `name-owner` property could be set before all D-Bus
messages had been processed — it could have been derived from getting
the owner of the name, for example.
This left unprocessed messages hanging around in the `context`, but that
context was never iterated again, which essentially leaked the
references held by those messages. That included a reference to the
`GDBusConnection`.
The first part of the fix is to simplify the code to use
`g_bus_watch_name_on_connection()`, so there’s only one signal
subscription to worry about.
The second part of the fix is to use the `GDestroyNotify` callback for
the watch data to be notified of when all D-Bus traffic has been
processed and the signal unsubscription is complete. At this point, it’s
guaranteed that there are no idle callbacks pending in the
`GMainContext`, since the `GDestroyNotify` callback is the last one
invoked on the `GMainContext`.
Essentially, this commit uses the `GDestroyNotify` callback as a
synchronisation message between the D-Bus worker thread and the thread
calling `ensure_gdbus_testserver_up()`.
Signed-off-by: Philip Withnall <withnall@endlessm.com>
Fixes: #1515
2020-02-21 13:07:53 +01:00
|
|
|
guint watch_id;
|
|
|
|
WatchData data = { context, FALSE, FALSE };
|
2014-03-31 12:06:05 +02:00
|
|
|
|
2020-02-20 12:49:12 +01:00
|
|
|
g_main_context_push_thread_default (context);
|
2014-03-31 12:06:05 +02:00
|
|
|
|
tests: Wait until unwatching the gdbus-testserver name has completed
Previously, the code in `ensure_gdbus_testserver_up()` created a proxy
object and watched its `name-owner` to see when the
`com.example.TestService` name appeared.
This ended up subscribing to three signals (one of them for name
ownership, and two unused for properties of the proxy), and was racy. In
particular, the `name-owner` property could be set before all D-Bus
messages had been processed — it could have been derived from getting
the owner of the name, for example.
This left unprocessed messages hanging around in the `context`, but that
context was never iterated again, which essentially leaked the
references held by those messages. That included a reference to the
`GDBusConnection`.
The first part of the fix is to simplify the code to use
`g_bus_watch_name_on_connection()`, so there’s only one signal
subscription to worry about.
The second part of the fix is to use the `GDestroyNotify` callback for
the watch data to be notified of when all D-Bus traffic has been
processed and the signal unsubscription is complete. At this point, it’s
guaranteed that there are no idle callbacks pending in the
`GMainContext`, since the `GDestroyNotify` callback is the last one
invoked on the `GMainContext`.
Essentially, this commit uses the `GDestroyNotify` callback as a
synchronisation message between the D-Bus worker thread and the thread
calling `ensure_gdbus_testserver_up()`.
Signed-off-by: Philip Withnall <withnall@endlessm.com>
Fixes: #1515
2020-02-21 13:07:53 +01:00
|
|
|
watch_id = g_bus_watch_name_on_connection (connection,
|
|
|
|
"com.example.TestService",
|
|
|
|
G_BUS_NAME_WATCHER_FLAGS_NONE,
|
|
|
|
name_appeared_cb,
|
|
|
|
NULL,
|
|
|
|
&data,
|
|
|
|
watch_free_cb);
|
2014-03-31 12:06:05 +02:00
|
|
|
|
2020-02-20 12:49:12 +01:00
|
|
|
timeout_source = g_timeout_source_new_seconds (60);
|
|
|
|
g_source_set_callback (timeout_source, _give_up,
|
|
|
|
"waited more than ~ 60s for gdbus-testserver to take its bus name",
|
|
|
|
NULL);
|
|
|
|
g_source_attach (timeout_source, context);
|
2014-03-31 12:06:05 +02:00
|
|
|
|
tests: Wait until unwatching the gdbus-testserver name has completed
Previously, the code in `ensure_gdbus_testserver_up()` created a proxy
object and watched its `name-owner` to see when the
`com.example.TestService` name appeared.
This ended up subscribing to three signals (one of them for name
ownership, and two unused for properties of the proxy), and was racy. In
particular, the `name-owner` property could be set before all D-Bus
messages had been processed — it could have been derived from getting
the owner of the name, for example.
This left unprocessed messages hanging around in the `context`, but that
context was never iterated again, which essentially leaked the
references held by those messages. That included a reference to the
`GDBusConnection`.
The first part of the fix is to simplify the code to use
`g_bus_watch_name_on_connection()`, so there’s only one signal
subscription to worry about.
The second part of the fix is to use the `GDestroyNotify` callback for
the watch data to be notified of when all D-Bus traffic has been
processed and the signal unsubscription is complete. At this point, it’s
guaranteed that there are no idle callbacks pending in the
`GMainContext`, since the `GDestroyNotify` callback is the last one
invoked on the `GMainContext`.
Essentially, this commit uses the `GDestroyNotify` callback as a
synchronisation message between the D-Bus worker thread and the thread
calling `ensure_gdbus_testserver_up()`.
Signed-off-by: Philip Withnall <withnall@endlessm.com>
Fixes: #1515
2020-02-21 13:07:53 +01:00
|
|
|
while (!data.name_appeared)
|
|
|
|
g_main_context_iteration (context, TRUE);
|
2014-03-31 12:06:05 +02:00
|
|
|
|
tests: Wait until unwatching the gdbus-testserver name has completed
Previously, the code in `ensure_gdbus_testserver_up()` created a proxy
object and watched its `name-owner` to see when the
`com.example.TestService` name appeared.
This ended up subscribing to three signals (one of them for name
ownership, and two unused for properties of the proxy), and was racy. In
particular, the `name-owner` property could be set before all D-Bus
messages had been processed — it could have been derived from getting
the owner of the name, for example.
This left unprocessed messages hanging around in the `context`, but that
context was never iterated again, which essentially leaked the
references held by those messages. That included a reference to the
`GDBusConnection`.
The first part of the fix is to simplify the code to use
`g_bus_watch_name_on_connection()`, so there’s only one signal
subscription to worry about.
The second part of the fix is to use the `GDestroyNotify` callback for
the watch data to be notified of when all D-Bus traffic has been
processed and the signal unsubscription is complete. At this point, it’s
guaranteed that there are no idle callbacks pending in the
`GMainContext`, since the `GDestroyNotify` callback is the last one
invoked on the `GMainContext`.
Essentially, this commit uses the `GDestroyNotify` callback as a
synchronisation message between the D-Bus worker thread and the thread
calling `ensure_gdbus_testserver_up()`.
Signed-off-by: Philip Withnall <withnall@endlessm.com>
Fixes: #1515
2020-02-21 13:07:53 +01:00
|
|
|
g_bus_unwatch_name (watch_id);
|
2014-03-31 12:06:05 +02:00
|
|
|
|
tests: Wait until unwatching the gdbus-testserver name has completed
Previously, the code in `ensure_gdbus_testserver_up()` created a proxy
object and watched its `name-owner` to see when the
`com.example.TestService` name appeared.
This ended up subscribing to three signals (one of them for name
ownership, and two unused for properties of the proxy), and was racy. In
particular, the `name-owner` property could be set before all D-Bus
messages had been processed — it could have been derived from getting
the owner of the name, for example.
This left unprocessed messages hanging around in the `context`, but that
context was never iterated again, which essentially leaked the
references held by those messages. That included a reference to the
`GDBusConnection`.
The first part of the fix is to simplify the code to use
`g_bus_watch_name_on_connection()`, so there’s only one signal
subscription to worry about.
The second part of the fix is to use the `GDestroyNotify` callback for
the watch data to be notified of when all D-Bus traffic has been
processed and the signal unsubscription is complete. At this point, it’s
guaranteed that there are no idle callbacks pending in the
`GMainContext`, since the `GDestroyNotify` callback is the last one
invoked on the `GMainContext`.
Essentially, this commit uses the `GDestroyNotify` callback as a
synchronisation message between the D-Bus worker thread and the thread
calling `ensure_gdbus_testserver_up()`.
Signed-off-by: Philip Withnall <withnall@endlessm.com>
Fixes: #1515
2020-02-21 13:07:53 +01:00
|
|
|
while (!data.unwatch_complete)
|
|
|
|
g_main_context_iteration (context, TRUE);
|
2014-03-31 12:06:05 +02:00
|
|
|
|
2020-02-20 12:49:12 +01:00
|
|
|
g_source_destroy (timeout_source);
|
|
|
|
g_source_unref (timeout_source);
|
|
|
|
|
|
|
|
g_main_context_pop_thread_default (context);
|
2014-03-31 12:06:05 +02:00
|
|
|
}
|
|
|
|
|
2010-05-06 20:13:59 +02:00
|
|
|
/* ---------------------------------------------------------------------------------------------------- */
|
|
|
|
|
|
|
|
typedef struct
|
|
|
|
{
|
|
|
|
GMainLoop *loop;
|
|
|
|
gboolean timed_out;
|
|
|
|
} SignalReceivedData;
|
|
|
|
|
|
|
|
static void
|
|
|
|
on_signal_received (gpointer user_data)
|
|
|
|
{
|
|
|
|
SignalReceivedData *data = user_data;
|
|
|
|
g_main_loop_quit (data->loop);
|
|
|
|
}
|
|
|
|
|
|
|
|
static gboolean
|
|
|
|
on_signal_received_timeout (gpointer user_data)
|
|
|
|
{
|
|
|
|
SignalReceivedData *data = user_data;
|
|
|
|
data->timed_out = TRUE;
|
|
|
|
g_main_loop_quit (data->loop);
|
2020-08-03 16:53:12 +02:00
|
|
|
return G_SOURCE_CONTINUE;
|
2010-05-06 20:13:59 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
gboolean
|
|
|
|
_g_assert_signal_received_run (gpointer object,
|
|
|
|
const gchar *signal_name)
|
|
|
|
{
|
|
|
|
gulong handler_id;
|
|
|
|
guint timeout_id;
|
|
|
|
SignalReceivedData data;
|
|
|
|
|
2011-04-11 16:51:37 +02:00
|
|
|
data.loop = g_main_loop_new (g_main_context_get_thread_default (), FALSE);
|
2010-05-06 20:13:59 +02:00
|
|
|
data.timed_out = FALSE;
|
|
|
|
handler_id = g_signal_connect_swapped (object,
|
|
|
|
signal_name,
|
|
|
|
G_CALLBACK (on_signal_received),
|
|
|
|
&data);
|
2013-02-03 16:21:20 +01:00
|
|
|
timeout_id = g_timeout_add_seconds (30,
|
|
|
|
on_signal_received_timeout,
|
|
|
|
&data);
|
2010-05-06 20:13:59 +02:00
|
|
|
g_main_loop_run (data.loop);
|
|
|
|
g_signal_handler_disconnect (object, handler_id);
|
|
|
|
g_source_remove (timeout_id);
|
|
|
|
g_main_loop_unref (data.loop);
|
|
|
|
|
|
|
|
return data.timed_out;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* ---------------------------------------------------------------------------------------------------- */
|
|
|
|
|
|
|
|
GDBusConnection *
|
|
|
|
_g_bus_get_priv (GBusType bus_type,
|
|
|
|
GCancellable *cancellable,
|
|
|
|
GError **error)
|
|
|
|
{
|
|
|
|
gchar *address;
|
|
|
|
GDBusConnection *ret;
|
|
|
|
|
|
|
|
ret = NULL;
|
|
|
|
|
|
|
|
address = g_dbus_address_get_for_bus_sync (bus_type, cancellable, error);
|
|
|
|
if (address == NULL)
|
|
|
|
goto out;
|
|
|
|
|
|
|
|
ret = g_dbus_connection_new_for_address_sync (address,
|
|
|
|
G_DBUS_CONNECTION_FLAGS_AUTHENTICATION_CLIENT |
|
|
|
|
G_DBUS_CONNECTION_FLAGS_MESSAGE_BUS_CONNECTION,
|
2010-05-13 22:32:11 +02:00
|
|
|
NULL, /* GDBusAuthObserver */
|
2010-05-06 20:13:59 +02:00
|
|
|
cancellable,
|
|
|
|
error);
|
|
|
|
g_free (address);
|
|
|
|
|
|
|
|
out:
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* ---------------------------------------------------------------------------------------------------- */
|