mirror of
				https://gitlab.gnome.org/GNOME/glib.git
				synced 2025-11-04 01:58:54 +01:00 
			
		
		
		
	
		
			
	
	
		
			335 lines
		
	
	
		
			9.8 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
		
		
			
		
	
	
			335 lines
		
	
	
		
			9.8 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
| 
								 | 
							
								/*
							 | 
						||
| 
								 | 
							
								 * Copyright 2019-2022 Collabora Ltd.
							 | 
						||
| 
								 | 
							
								 *
							 | 
						||
| 
								 | 
							
								 * SPDX-License-Identifier: LGPL-2.1-or-later
							 | 
						||
| 
								 | 
							
								 *
							 | 
						||
| 
								 | 
							
								 * 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/>.
							 | 
						||
| 
								 | 
							
								 */
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								#include "config.h"
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								#include <errno.h>
							 | 
						||
| 
								 | 
							
								#include <locale.h>
							 | 
						||
| 
								 | 
							
								#include <stdlib.h>
							 | 
						||
| 
								 | 
							
								#include <string.h>
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								#include <glib.h>
							 | 
						||
| 
								 | 
							
								#include <glib/gstdio.h>
							 | 
						||
| 
								 | 
							
								#include <gio/gio.h>
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								/* For G_CREDENTIALS_*_SUPPORTED */
							 | 
						||
| 
								 | 
							
								#include <gio/gcredentialsprivate.h>
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								static const char * const explicit_external_initial_response_fail[] =
							 | 
						||
| 
								 | 
							
								{
							 | 
						||
| 
								 | 
							
								  "EXTERNAL with incorrect initial response",
							 | 
						||
| 
								 | 
							
								  "C:AUTH EXTERNAL <wrong-uid>",
							 | 
						||
| 
								 | 
							
								  "S:REJECTED.*$",
							 | 
						||
| 
								 | 
							
								  NULL
							 | 
						||
| 
								 | 
							
								};
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								static const char * const explicit_external_fail[] =
							 | 
						||
| 
								 | 
							
								{
							 | 
						||
| 
								 | 
							
								  "EXTERNAL without initial response, failing to authenticate",
							 | 
						||
| 
								 | 
							
								  "C:AUTH EXTERNAL",
							 | 
						||
| 
								 | 
							
								  "S:DATA$",
							 | 
						||
| 
								 | 
							
								  "C:DATA <wrong-uid>",
							 | 
						||
| 
								 | 
							
								  "S:REJECTED.*$",
							 | 
						||
| 
								 | 
							
								  NULL
							 | 
						||
| 
								 | 
							
								};
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								#if defined(G_CREDENTIALS_SOCKET_GET_CREDENTIALS_SUPPORTED) || defined(G_CREDENTIALS_UNIX_CREDENTIALS_MESSAGE_SUPPORTED)
							 | 
						||
| 
								 | 
							
								static const char * const explicit_external_initial_response[] =
							 | 
						||
| 
								 | 
							
								{
							 | 
						||
| 
								 | 
							
								  "EXTERNAL with initial response",
							 | 
						||
| 
								 | 
							
								  /* This is what most older D-Bus libraries do. */
							 | 
						||
| 
								 | 
							
								  "C:AUTH EXTERNAL <uid>",          /* I claim to be <uid> */
							 | 
						||
| 
								 | 
							
								  "S:OK [0-9a-f]+$",
							 | 
						||
| 
								 | 
							
								  NULL
							 | 
						||
| 
								 | 
							
								};
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								static const char * const explicit_external[] =
							 | 
						||
| 
								 | 
							
								{
							 | 
						||
| 
								 | 
							
								  "EXTERNAL without initial response",
							 | 
						||
| 
								 | 
							
								  /* In theory this is equally valid, although many D-Bus libraries
							 | 
						||
| 
								 | 
							
								   * probably don't support it correctly. */
							 | 
						||
| 
								 | 
							
								  "C:AUTH EXTERNAL",                /* Start EXTERNAL, no initial response */
							 | 
						||
| 
								 | 
							
								  "S:DATA$",                        /* Who are you? */
							 | 
						||
| 
								 | 
							
								  "C:DATA <uid>",                   /* I claim to be <uid> */
							 | 
						||
| 
								 | 
							
								  "S:OK [0-9a-f]+$",
							 | 
						||
| 
								 | 
							
								  NULL
							 | 
						||
| 
								 | 
							
								};
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								static const char * const implicit_external[] =
							 | 
						||
| 
								 | 
							
								{
							 | 
						||
| 
								 | 
							
								  "EXTERNAL with empty authorization identity",
							 | 
						||
| 
								 | 
							
								  /* This is what sd-bus does. */
							 | 
						||
| 
								 | 
							
								  "C:AUTH EXTERNAL",                /* Start EXTERNAL, no initial response */
							 | 
						||
| 
								 | 
							
								  "S:DATA$",                        /* Who are you? */
							 | 
						||
| 
								 | 
							
								  "C:DATA",                         /* I'm whoever the kernel says I am */
							 | 
						||
| 
								 | 
							
								  "S:OK [0-9a-f]+$",
							 | 
						||
| 
								 | 
							
								  NULL
							 | 
						||
| 
								 | 
							
								};
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								static const char * const implicit_external_space[] =
							 | 
						||
| 
								 | 
							
								{
							 | 
						||
| 
								 | 
							
								  "EXTERNAL with empty authorization identity and whitespace",
							 | 
						||
| 
								 | 
							
								  /* GDBus used to represent empty data blocks like this, although it
							 | 
						||
| 
								 | 
							
								   * isn't interoperable to do so (in particular sd-bus would reject this). */
							 | 
						||
| 
								 | 
							
								  "C:AUTH EXTERNAL",                /* Start EXTERNAL, no initial response */
							 | 
						||
| 
								 | 
							
								  "S:DATA$",                        /* Who are you? */
							 | 
						||
| 
								 | 
							
								  "C:DATA ",                        /* I'm whoever the kernel says I am */
							 | 
						||
| 
								 | 
							
								  "S:OK [0-9a-f]+$",
							 | 
						||
| 
								 | 
							
								  NULL
							 | 
						||
| 
								 | 
							
								};
							 | 
						||
| 
								 | 
							
								#endif
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								static const char * const * const handshakes[] =
							 | 
						||
| 
								 | 
							
								{
							 | 
						||
| 
								 | 
							
								  explicit_external_initial_response_fail,
							 | 
						||
| 
								 | 
							
								  explicit_external_fail,
							 | 
						||
| 
								 | 
							
								#if defined(G_CREDENTIALS_SOCKET_GET_CREDENTIALS_SUPPORTED) || defined(G_CREDENTIALS_UNIX_CREDENTIALS_MESSAGE_SUPPORTED)
							 | 
						||
| 
								 | 
							
								  explicit_external_initial_response,
							 | 
						||
| 
								 | 
							
								  explicit_external,
							 | 
						||
| 
								 | 
							
								  implicit_external,
							 | 
						||
| 
								 | 
							
								  implicit_external_space,
							 | 
						||
| 
								 | 
							
								#endif
							 | 
						||
| 
								 | 
							
								};
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								static void
							 | 
						||
| 
								 | 
							
								encode_uid (guint uid,
							 | 
						||
| 
								 | 
							
								            GString *dest)
							 | 
						||
| 
								 | 
							
								{
							 | 
						||
| 
								 | 
							
								  gchar *str = g_strdup_printf ("%u", uid);
							 | 
						||
| 
								 | 
							
								  gchar *p;
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								  g_string_assign (dest, "");
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								  for (p = str; *p != '\0'; p++)
							 | 
						||
| 
								 | 
							
								    g_string_append_printf (dest, "%02x", (unsigned char) *p);
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								  g_free (str);
							 | 
						||
| 
								 | 
							
								}
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								typedef struct
							 | 
						||
| 
								 | 
							
								{
							 | 
						||
| 
								 | 
							
								  GCond cond;
							 | 
						||
| 
								 | 
							
								  GMutex mutex;
							 | 
						||
| 
								 | 
							
								  GDBusServerFlags server_flags;
							 | 
						||
| 
								 | 
							
								  GMainContext *ctx;
							 | 
						||
| 
								 | 
							
								  GMainLoop *loop;
							 | 
						||
| 
								 | 
							
								  gchar *guid;
							 | 
						||
| 
								 | 
							
								  gchar *listenable_address;
							 | 
						||
| 
								 | 
							
								  gboolean ready;
							 | 
						||
| 
								 | 
							
								} ServerInfo;
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								static gboolean
							 | 
						||
| 
								 | 
							
								idle_in_server_thread_cb (gpointer user_data)
							 | 
						||
| 
								 | 
							
								{
							 | 
						||
| 
								 | 
							
								  ServerInfo *info = user_data;
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								  g_mutex_lock (&info->mutex);
							 | 
						||
| 
								 | 
							
								  info->ready = TRUE;
							 | 
						||
| 
								 | 
							
								  g_cond_broadcast (&info->cond);
							 | 
						||
| 
								 | 
							
								  g_mutex_unlock (&info->mutex);
							 | 
						||
| 
								 | 
							
								  return G_SOURCE_REMOVE;
							 | 
						||
| 
								 | 
							
								}
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								static gpointer
							 | 
						||
| 
								 | 
							
								server_thread_cb (gpointer user_data)
							 | 
						||
| 
								 | 
							
								{
							 | 
						||
| 
								 | 
							
								  GDBusServer *server = NULL;
							 | 
						||
| 
								 | 
							
								  GError *error = NULL;
							 | 
						||
| 
								 | 
							
								  GSource *source;
							 | 
						||
| 
								 | 
							
								  ServerInfo *info = user_data;
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								  g_main_context_push_thread_default (info->ctx);
							 | 
						||
| 
								 | 
							
								  server = g_dbus_server_new_sync (info->listenable_address,
							 | 
						||
| 
								 | 
							
								                                   info->server_flags,
							 | 
						||
| 
								 | 
							
								                                   info->guid,
							 | 
						||
| 
								 | 
							
								                                   NULL,
							 | 
						||
| 
								 | 
							
								                                   NULL,
							 | 
						||
| 
								 | 
							
								                                   &error);
							 | 
						||
| 
								 | 
							
								  g_assert_no_error (error);
							 | 
						||
| 
								 | 
							
								  g_assert_nonnull (server);
							 | 
						||
| 
								 | 
							
								  g_dbus_server_start (server);
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								  /* Tell the main thread when the server is ready to accept connections */
							 | 
						||
| 
								 | 
							
								  source = g_idle_source_new ();
							 | 
						||
| 
								 | 
							
								  g_source_set_callback (source, idle_in_server_thread_cb, info, NULL);
							 | 
						||
| 
								 | 
							
								  g_source_attach (source, info->ctx);
							 | 
						||
| 
								 | 
							
								  g_source_unref (source);
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								  g_main_loop_run (info->loop);
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								  g_main_context_pop_thread_default (info->ctx);
							 | 
						||
| 
								 | 
							
								  g_dbus_server_stop (server);
							 | 
						||
| 
								 | 
							
								  g_clear_object (&server);
							 | 
						||
| 
								 | 
							
								  return NULL;
							 | 
						||
| 
								 | 
							
								}
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								static void
							 | 
						||
| 
								 | 
							
								test_sasl_server (void)
							 | 
						||
| 
								 | 
							
								{
							 | 
						||
| 
								 | 
							
								  GError *error = NULL;
							 | 
						||
| 
								 | 
							
								  GSocketAddress *addr = NULL;
							 | 
						||
| 
								 | 
							
								  GString *buf = g_string_new ("");
							 | 
						||
| 
								 | 
							
								  GString *encoded_uid = g_string_new ("");
							 | 
						||
| 
								 | 
							
								  GString *encoded_wrong_uid = g_string_new ("");
							 | 
						||
| 
								 | 
							
								  GThread *server_thread = NULL;
							 | 
						||
| 
								 | 
							
								  ServerInfo info =
							 | 
						||
| 
								 | 
							
								  {
							 | 
						||
| 
								 | 
							
								    .server_flags = G_DBUS_SERVER_FLAGS_RUN_IN_THREAD,
							 | 
						||
| 
								 | 
							
								  };
							 | 
						||
| 
								 | 
							
								  gchar *escaped = NULL;
							 | 
						||
| 
								 | 
							
								  gchar *path = NULL;
							 | 
						||
| 
								 | 
							
								  gchar *tmpdir = NULL;
							 | 
						||
| 
								 | 
							
								  gsize i;
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								  tmpdir = g_dir_make_tmp ("gdbus-server-auth-XXXXXX", &error);
							 | 
						||
| 
								 | 
							
								  g_assert_no_error (error);
							 | 
						||
| 
								 | 
							
								  escaped = g_dbus_address_escape_value (tmpdir);
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								  path = g_build_filename (tmpdir, "socket", NULL);
							 | 
						||
| 
								 | 
							
								  g_cond_init (&info.cond);
							 | 
						||
| 
								 | 
							
								  g_mutex_init (&info.mutex);
							 | 
						||
| 
								 | 
							
								  info.ctx = g_main_context_new ();
							 | 
						||
| 
								 | 
							
								  info.guid = g_dbus_generate_guid ();
							 | 
						||
| 
								 | 
							
								  info.listenable_address = g_strdup_printf ("unix:path=%s/socket", escaped);
							 | 
						||
| 
								 | 
							
								  info.loop = g_main_loop_new (info.ctx, FALSE);
							 | 
						||
| 
								 | 
							
								  info.ready = FALSE;
							 | 
						||
| 
								 | 
							
								  server_thread = g_thread_new ("GDBusServer", server_thread_cb, &info);
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								  g_mutex_lock (&info.mutex);
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								  while (!info.ready)
							 | 
						||
| 
								 | 
							
								    g_cond_wait (&info.cond, &info.mutex);
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								  g_mutex_unlock (&info.mutex);
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								  addr = g_unix_socket_address_new (path);
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								  encode_uid (geteuid (), encoded_uid);
							 | 
						||
| 
								 | 
							
								  encode_uid (geteuid () == 0 ? 65534 : 0, encoded_wrong_uid);
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								  for (i = 0; i < G_N_ELEMENTS (handshakes); i++)
							 | 
						||
| 
								 | 
							
								    {
							 | 
						||
| 
								 | 
							
								      const char * const *handshake = handshakes[i];
							 | 
						||
| 
								 | 
							
								      GSocketClient *client;
							 | 
						||
| 
								 | 
							
								      GSocketConnection *conn;
							 | 
						||
| 
								 | 
							
								      GUnixConnection *conn_unix;       /* unowned */
							 | 
						||
| 
								 | 
							
								      GInputStream *istream;            /* unowned */
							 | 
						||
| 
								 | 
							
								      GDataInputStream *istream_data;
							 | 
						||
| 
								 | 
							
								      GOutputStream *ostream;           /* unowned */
							 | 
						||
| 
								 | 
							
								      GError *error = NULL;
							 | 
						||
| 
								 | 
							
								      gsize j;
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								      g_test_message ("New handshake: %s", handshake[0]);
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								      client = g_socket_client_new ();
							 | 
						||
| 
								 | 
							
								      conn = g_socket_client_connect (client, G_SOCKET_CONNECTABLE (addr),
							 | 
						||
| 
								 | 
							
								                                      NULL, &error);
							 | 
						||
| 
								 | 
							
								      g_assert_no_error (error);
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								      g_assert_true (G_IS_UNIX_CONNECTION (conn));
							 | 
						||
| 
								 | 
							
								      conn_unix = G_UNIX_CONNECTION (conn);
							 | 
						||
| 
								 | 
							
								      istream = g_io_stream_get_input_stream (G_IO_STREAM (conn));
							 | 
						||
| 
								 | 
							
								      ostream = g_io_stream_get_output_stream (G_IO_STREAM (conn));
							 | 
						||
| 
								 | 
							
								      istream_data = g_data_input_stream_new (istream);
							 | 
						||
| 
								 | 
							
								      g_data_input_stream_set_newline_type (istream_data, G_DATA_STREAM_NEWLINE_TYPE_CR_LF);
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								      g_unix_connection_send_credentials (conn_unix, NULL, &error);
							 | 
						||
| 
								 | 
							
								      g_assert_no_error (error);
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								      for (j = 1; handshake[j] != NULL; j++)
							 | 
						||
| 
								 | 
							
								        {
							 | 
						||
| 
								 | 
							
								          if (j % 2 == 1)
							 | 
						||
| 
								 | 
							
								            {
							 | 
						||
| 
								 | 
							
								              /* client to server */
							 | 
						||
| 
								 | 
							
								              const char *line = handshake[j];
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								              g_assert_cmpint (line[0], ==, 'C');
							 | 
						||
| 
								 | 
							
								              g_assert_cmpint (line[1], ==, ':');
							 | 
						||
| 
								 | 
							
								              g_string_assign (buf, line + 2);
							 | 
						||
| 
								 | 
							
								              g_string_replace (buf, "<uid>", encoded_uid->str, 0);
							 | 
						||
| 
								 | 
							
								              g_string_replace (buf, "<wrong-uid>", encoded_wrong_uid->str, 0);
							 | 
						||
| 
								 | 
							
								              g_test_message ("C:“%s”", buf->str);
							 | 
						||
| 
								 | 
							
								              g_string_append (buf, "\r\n");
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								              g_output_stream_write_all (ostream, buf->str, buf->len, NULL, NULL, &error);
							 | 
						||
| 
								 | 
							
								              g_assert_no_error (error);
							 | 
						||
| 
								 | 
							
								            }
							 | 
						||
| 
								 | 
							
								          else
							 | 
						||
| 
								 | 
							
								            {
							 | 
						||
| 
								 | 
							
								              /* server to client */
							 | 
						||
| 
								 | 
							
								              const char *pattern = handshake[j];
							 | 
						||
| 
								 | 
							
								              char *line;
							 | 
						||
| 
								 | 
							
								              gsize len;
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								              g_assert_cmpint (pattern[0], ==, 'S');
							 | 
						||
| 
								 | 
							
								              g_assert_cmpint (pattern[1], ==, ':');
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								              g_test_message ("Expect: /^%s/", pattern + 2);
							 | 
						||
| 
								 | 
							
								              line = g_data_input_stream_read_line (istream_data, &len, NULL, &error);
							 | 
						||
| 
								 | 
							
								              g_assert_no_error (error);
							 | 
						||
| 
								 | 
							
								              g_test_message ("S:“%s”", line);
							 | 
						||
| 
								 | 
							
								              g_assert_cmpuint (len, ==, strlen (line));
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								              if (!g_regex_match_simple (pattern + 2, line,
							 | 
						||
| 
								 | 
							
								                                         G_REGEX_ANCHORED,
							 | 
						||
| 
								 | 
							
								                                         G_REGEX_MATCH_ANCHORED))
							 | 
						||
| 
								 | 
							
								                g_error ("Expected /^%s/, got “%s”", pattern + 2, line);
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								              g_free (line);
							 | 
						||
| 
								 | 
							
								            }
							 | 
						||
| 
								 | 
							
								        }
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								      g_object_unref (istream_data);
							 | 
						||
| 
								 | 
							
								      g_object_unref (conn);
							 | 
						||
| 
								 | 
							
								      g_object_unref (client);
							 | 
						||
| 
								 | 
							
								    }
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								  g_main_loop_quit (info.loop);
							 | 
						||
| 
								 | 
							
								  g_thread_join (server_thread);
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								  if (tmpdir != NULL)
							 | 
						||
| 
								 | 
							
								    g_assert_no_errno (g_rmdir (tmpdir));
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								  g_clear_pointer (&info.ctx, g_main_context_unref);
							 | 
						||
| 
								 | 
							
								  g_clear_pointer (&info.loop, g_main_loop_unref);
							 | 
						||
| 
								 | 
							
								  g_clear_object (&addr);
							 | 
						||
| 
								 | 
							
								  g_string_free (buf, TRUE);
							 | 
						||
| 
								 | 
							
								  g_string_free (encoded_uid, TRUE);
							 | 
						||
| 
								 | 
							
								  g_string_free (encoded_wrong_uid, TRUE);
							 | 
						||
| 
								 | 
							
								  g_free (escaped);
							 | 
						||
| 
								 | 
							
								  g_free (info.guid);
							 | 
						||
| 
								 | 
							
								  g_free (info.listenable_address);
							 | 
						||
| 
								 | 
							
								  g_free (path);
							 | 
						||
| 
								 | 
							
								  g_free (tmpdir);
							 | 
						||
| 
								 | 
							
								  g_cond_clear (&info.cond);
							 | 
						||
| 
								 | 
							
								  g_mutex_clear (&info.mutex);
							 | 
						||
| 
								 | 
							
								}
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								int
							 | 
						||
| 
								 | 
							
								main (int   argc,
							 | 
						||
| 
								 | 
							
								      char *argv[])
							 | 
						||
| 
								 | 
							
								{
							 | 
						||
| 
								 | 
							
								  setlocale (LC_ALL, "");
							 | 
						||
| 
								 | 
							
								  g_test_init (&argc, &argv, G_TEST_OPTION_ISOLATE_DIRS, NULL);
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								  g_test_add_func ("/gdbus/sasl/server", test_sasl_server);
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								  return g_test_run();
							 | 
						||
| 
								 | 
							
								}
							 |