mirror of
https://gitlab.freedesktop.org/dbus/dbus.git
synced 2026-05-31 18:18:18 +02:00
* doc/TODO: remove the "guid" item * test/glib/test-profile.c (no_bus_thread_func): use open_private (with_bus_thread_func): use open_private * dbus/dbus-connection.c (dbus_connection_open_private): new function that works like the old dbus_connection_open() (dbus_connection_open): now returns an existing connection if possible * dbus/dbus-server-unix.c (handle_new_client_fd_and_unlock): pass through the GUID to the transport * dbus/dbus-server.c (_dbus_server_init_base): keep around the GUID in hex-encoded form. * dbus/dbus-server-debug-pipe.c (_dbus_transport_debug_pipe_new): pass GUID argument in to the transport * dbus/dbus-transport-unix.c (_dbus_transport_new_for_fd): add guid argument * dbus/dbus-transport.c (_dbus_transport_init_base): add guid argument * dbus/dbus-auth.c (_dbus_auth_server_new): add guid argument
522 lines
14 KiB
C
522 lines
14 KiB
C
/* -*- mode: C; c-file-style: "gnu" -*- */
|
|
/* dbus-threads.h D-BUS threads handling
|
|
*
|
|
* Copyright (C) 2002, 2003 Red Hat Inc.
|
|
*
|
|
* Licensed under the Academic Free License version 2.1
|
|
*
|
|
* This program is free software; you can redistribute it and/or modify
|
|
* it under the terms of the GNU General Public License as published by
|
|
* the Free Software Foundation; either version 2 of the License, or
|
|
* (at your option) any later version.
|
|
*
|
|
* This program 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 General Public License for more details.
|
|
*
|
|
* You should have received a copy of the GNU General Public License
|
|
* along with this program; if not, write to the Free Software
|
|
* Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
|
|
*
|
|
*/
|
|
#include "dbus-threads.h"
|
|
#include "dbus-internals.h"
|
|
#include "dbus-threads-internal.h"
|
|
|
|
static DBusThreadFunctions thread_functions =
|
|
{
|
|
0,
|
|
NULL, NULL, NULL, NULL,
|
|
NULL, NULL, NULL, NULL, NULL,
|
|
|
|
NULL, NULL, NULL, NULL,
|
|
NULL, NULL, NULL, NULL
|
|
};
|
|
static int thread_init_generation = 0;
|
|
|
|
/** This is used for the no-op default mutex pointer, just to be distinct from #NULL */
|
|
#define _DBUS_DUMMY_MUTEX ((DBusMutex*)0xABCDEF)
|
|
|
|
/** This is used for the no-op default mutex pointer, just to be distinct from #NULL */
|
|
#define _DBUS_DUMMY_CONDVAR ((DBusCondVar*)0xABCDEF2)
|
|
|
|
/**
|
|
* @defgroup DBusThreadsInternals Thread functions
|
|
* @ingroup DBusInternals
|
|
* @brief _dbus_mutex_lock(), etc.
|
|
*
|
|
* Functions and macros related to threads and thread locks.
|
|
*
|
|
* @{
|
|
*/
|
|
|
|
/**
|
|
* Creates a new mutex using the function supplied to dbus_threads_init(),
|
|
* or creates a no-op mutex if threads are not initialized.
|
|
* May return #NULL even if threads are initialized, indicating
|
|
* out-of-memory.
|
|
*
|
|
* @returns new mutex or #NULL
|
|
*/
|
|
DBusMutex*
|
|
_dbus_mutex_new (void)
|
|
{
|
|
if (thread_functions.mutex_new)
|
|
return (* thread_functions.mutex_new) ();
|
|
else
|
|
return _DBUS_DUMMY_MUTEX;
|
|
}
|
|
|
|
/**
|
|
* Frees a mutex created with dbus_mutex_new(); does
|
|
* nothing if passed a #NULL pointer.
|
|
*/
|
|
void
|
|
_dbus_mutex_free (DBusMutex *mutex)
|
|
{
|
|
if (mutex && thread_functions.mutex_free)
|
|
(* thread_functions.mutex_free) (mutex);
|
|
}
|
|
|
|
/**
|
|
* Locks a mutex. Does nothing if passed a #NULL pointer.
|
|
* Locks are not recursive.
|
|
*
|
|
* @returns #TRUE on success
|
|
*/
|
|
dbus_bool_t
|
|
_dbus_mutex_lock (DBusMutex *mutex)
|
|
{
|
|
if (mutex && thread_functions.mutex_lock)
|
|
return (* thread_functions.mutex_lock) (mutex);
|
|
else
|
|
return TRUE;
|
|
}
|
|
|
|
/**
|
|
* Unlocks a mutex. Does nothing if passed a #NULL pointer.
|
|
*
|
|
* @returns #TRUE on success
|
|
*/
|
|
dbus_bool_t
|
|
_dbus_mutex_unlock (DBusMutex *mutex)
|
|
{
|
|
if (mutex && thread_functions.mutex_unlock)
|
|
return (* thread_functions.mutex_unlock) (mutex);
|
|
else
|
|
return TRUE;
|
|
}
|
|
|
|
/**
|
|
* Creates a new condition variable using the function supplied
|
|
* to dbus_threads_init(), or creates a no-op condition variable
|
|
* if threads are not initialized. May return #NULL even if
|
|
* threads are initialized, indicating out-of-memory.
|
|
*
|
|
* @returns new mutex or #NULL
|
|
*/
|
|
DBusCondVar *
|
|
_dbus_condvar_new (void)
|
|
{
|
|
if (thread_functions.condvar_new)
|
|
return (* thread_functions.condvar_new) ();
|
|
else
|
|
return _DBUS_DUMMY_CONDVAR;
|
|
}
|
|
|
|
/**
|
|
* Frees a conditional variable created with dbus_condvar_new(); does
|
|
* nothing if passed a #NULL pointer.
|
|
*/
|
|
void
|
|
_dbus_condvar_free (DBusCondVar *cond)
|
|
{
|
|
if (cond && thread_functions.condvar_free)
|
|
(* thread_functions.condvar_free) (cond);
|
|
}
|
|
|
|
/**
|
|
* Atomically unlocks the mutex and waits for the conditions
|
|
* variable to be signalled. Locks the mutex again before
|
|
* returning.
|
|
* Does nothing if passed a #NULL pointer.
|
|
*/
|
|
void
|
|
_dbus_condvar_wait (DBusCondVar *cond,
|
|
DBusMutex *mutex)
|
|
{
|
|
if (cond && mutex && thread_functions.condvar_wait)
|
|
(* thread_functions.condvar_wait) (cond, mutex);
|
|
}
|
|
|
|
/**
|
|
* Atomically unlocks the mutex and waits for the conditions
|
|
* variable to be signalled, or for a timeout. Locks the
|
|
* mutex again before returning.
|
|
* Does nothing if passed a #NULL pointer.
|
|
*
|
|
* @param cond the condition variable
|
|
* @param mutex the mutex
|
|
* @param timeout_milliseconds the maximum time to wait
|
|
* @returns TRUE if the condition was reached, or FALSE if the
|
|
* timeout was reached.
|
|
*/
|
|
dbus_bool_t
|
|
_dbus_condvar_wait_timeout (DBusCondVar *cond,
|
|
DBusMutex *mutex,
|
|
int timeout_milliseconds)
|
|
{
|
|
if (cond && mutex && thread_functions.condvar_wait)
|
|
return (* thread_functions.condvar_wait_timeout) (cond, mutex, timeout_milliseconds);
|
|
else
|
|
return TRUE;
|
|
}
|
|
|
|
/**
|
|
* If there are threads waiting on the condition variable, wake
|
|
* up exactly one.
|
|
* Does nothing if passed a #NULL pointer.
|
|
*/
|
|
void
|
|
_dbus_condvar_wake_one (DBusCondVar *cond)
|
|
{
|
|
if (cond && thread_functions.condvar_wake_one)
|
|
(* thread_functions.condvar_wake_one) (cond);
|
|
}
|
|
|
|
/**
|
|
* If there are threads waiting on the condition variable, wake
|
|
* up all of them.
|
|
* Does nothing if passed a #NULL pointer.
|
|
*/
|
|
void
|
|
_dbus_condvar_wake_all (DBusCondVar *cond)
|
|
{
|
|
if (cond && thread_functions.condvar_wake_all)
|
|
(* thread_functions.condvar_wake_all) (cond);
|
|
}
|
|
|
|
static void
|
|
shutdown_global_locks (void *data)
|
|
{
|
|
DBusMutex ***locks = data;
|
|
int i;
|
|
|
|
i = 0;
|
|
while (i < _DBUS_N_GLOBAL_LOCKS)
|
|
{
|
|
_dbus_mutex_free (*(locks[i]));
|
|
*(locks[i]) = NULL;
|
|
++i;
|
|
}
|
|
|
|
dbus_free (locks);
|
|
}
|
|
|
|
static dbus_bool_t
|
|
init_global_locks (void)
|
|
{
|
|
int i;
|
|
DBusMutex ***dynamic_global_locks;
|
|
|
|
DBusMutex **global_locks[] = {
|
|
#define LOCK_ADDR(name) (& _dbus_lock_##name)
|
|
LOCK_ADDR (list),
|
|
LOCK_ADDR (connection_slots),
|
|
LOCK_ADDR (pending_call_slots),
|
|
LOCK_ADDR (server_slots),
|
|
LOCK_ADDR (message_slots),
|
|
LOCK_ADDR (atomic),
|
|
LOCK_ADDR (bus),
|
|
LOCK_ADDR (shutdown_funcs),
|
|
LOCK_ADDR (system_users),
|
|
LOCK_ADDR (message_cache),
|
|
LOCK_ADDR (shared_connections)
|
|
#undef LOCK_ADDR
|
|
};
|
|
|
|
_dbus_assert (_DBUS_N_ELEMENTS (global_locks) ==
|
|
_DBUS_N_GLOBAL_LOCKS);
|
|
|
|
i = 0;
|
|
|
|
dynamic_global_locks = dbus_new (DBusMutex**, _DBUS_N_GLOBAL_LOCKS);
|
|
if (dynamic_global_locks == NULL)
|
|
goto failed;
|
|
|
|
while (i < _DBUS_N_ELEMENTS (global_locks))
|
|
{
|
|
*global_locks[i] = _dbus_mutex_new ();
|
|
|
|
if (*global_locks[i] == NULL)
|
|
goto failed;
|
|
|
|
dynamic_global_locks[i] = global_locks[i];
|
|
|
|
++i;
|
|
}
|
|
|
|
if (!_dbus_register_shutdown_func (shutdown_global_locks,
|
|
dynamic_global_locks))
|
|
goto failed;
|
|
|
|
return TRUE;
|
|
|
|
failed:
|
|
dbus_free (dynamic_global_locks);
|
|
|
|
for (i = i - 1; i >= 0; i--)
|
|
{
|
|
_dbus_mutex_free (*global_locks[i]);
|
|
*global_locks[i] = NULL;
|
|
}
|
|
return FALSE;
|
|
}
|
|
|
|
/** @} */ /* end of internals */
|
|
|
|
/**
|
|
* @defgroup DBusThreads Thread functions
|
|
* @ingroup DBus
|
|
* @brief dbus_threads_init()
|
|
*
|
|
* Functions and macros related to threads and thread locks.
|
|
*
|
|
* @{
|
|
*/
|
|
|
|
/**
|
|
*
|
|
* Initializes threads. If this function is not called,
|
|
* the D-BUS library will not lock any data structures.
|
|
* If it is called, D-BUS will do locking, at some cost
|
|
* in efficiency. Note that this function must be called
|
|
* BEFORE using any other D-BUS functions.
|
|
*
|
|
* This function may be called more than once, as long
|
|
* as you pass in the same functions each time. If it's
|
|
* called multiple times with different functions, then
|
|
* a warning is printed, because someone is confused.
|
|
*
|
|
* @param functions functions for using threads
|
|
* @returns #TRUE on success, #FALSE if no memory
|
|
*/
|
|
dbus_bool_t
|
|
dbus_threads_init (const DBusThreadFunctions *functions)
|
|
{
|
|
_dbus_assert (functions != NULL);
|
|
|
|
/* these base functions are required. Future additions to
|
|
* DBusThreadFunctions may be optional.
|
|
*/
|
|
_dbus_assert (functions->mask & DBUS_THREAD_FUNCTIONS_MUTEX_NEW_MASK);
|
|
_dbus_assert (functions->mask & DBUS_THREAD_FUNCTIONS_MUTEX_FREE_MASK);
|
|
_dbus_assert (functions->mask & DBUS_THREAD_FUNCTIONS_MUTEX_LOCK_MASK);
|
|
_dbus_assert (functions->mask & DBUS_THREAD_FUNCTIONS_MUTEX_UNLOCK_MASK);
|
|
_dbus_assert (functions->mask & DBUS_THREAD_FUNCTIONS_CONDVAR_NEW_MASK);
|
|
_dbus_assert (functions->mask & DBUS_THREAD_FUNCTIONS_CONDVAR_FREE_MASK);
|
|
_dbus_assert (functions->mask & DBUS_THREAD_FUNCTIONS_CONDVAR_WAIT_MASK);
|
|
_dbus_assert (functions->mask & DBUS_THREAD_FUNCTIONS_CONDVAR_WAIT_TIMEOUT_MASK);
|
|
_dbus_assert (functions->mask & DBUS_THREAD_FUNCTIONS_CONDVAR_WAKE_ONE_MASK);
|
|
_dbus_assert (functions->mask & DBUS_THREAD_FUNCTIONS_CONDVAR_WAKE_ALL_MASK);
|
|
_dbus_assert (functions->mutex_new != NULL);
|
|
_dbus_assert (functions->mutex_free != NULL);
|
|
_dbus_assert (functions->mutex_lock != NULL);
|
|
_dbus_assert (functions->mutex_unlock != NULL);
|
|
_dbus_assert (functions->condvar_new != NULL);
|
|
_dbus_assert (functions->condvar_free != NULL);
|
|
_dbus_assert (functions->condvar_wait != NULL);
|
|
_dbus_assert (functions->condvar_wait_timeout != NULL);
|
|
_dbus_assert (functions->condvar_wake_one != NULL);
|
|
_dbus_assert (functions->condvar_wake_all != NULL);
|
|
|
|
/* Check that all bits in the mask actually are valid mask bits.
|
|
* ensures people won't write code that breaks when we add
|
|
* new bits.
|
|
*/
|
|
_dbus_assert ((functions->mask & ~DBUS_THREAD_FUNCTIONS_ALL_MASK) == 0);
|
|
|
|
if (thread_init_generation != _dbus_current_generation)
|
|
thread_functions.mask = 0; /* allow re-init in new generation */
|
|
|
|
if (thread_functions.mask != 0)
|
|
{
|
|
/* Silently allow multiple init if the functions are the same ones.
|
|
* Well, we only bother checking two of them, just out of laziness.
|
|
*/
|
|
if (thread_functions.mask == functions->mask &&
|
|
thread_functions.mutex_new == functions->mutex_new &&
|
|
thread_functions.condvar_new == functions->condvar_new)
|
|
{
|
|
return TRUE;
|
|
}
|
|
else
|
|
{
|
|
_dbus_warn ("dbus_threads_init() called twice with two different sets of functions\n");
|
|
return FALSE;
|
|
}
|
|
}
|
|
|
|
thread_functions.mutex_new = functions->mutex_new;
|
|
thread_functions.mutex_free = functions->mutex_free;
|
|
thread_functions.mutex_lock = functions->mutex_lock;
|
|
thread_functions.mutex_unlock = functions->mutex_unlock;
|
|
|
|
thread_functions.condvar_new = functions->condvar_new;
|
|
thread_functions.condvar_free = functions->condvar_free;
|
|
thread_functions.condvar_wait = functions->condvar_wait;
|
|
thread_functions.condvar_wait_timeout = functions->condvar_wait_timeout;
|
|
thread_functions.condvar_wake_one = functions->condvar_wake_one;
|
|
thread_functions.condvar_wake_all = functions->condvar_wake_all;
|
|
|
|
thread_functions.mask = functions->mask;
|
|
|
|
if (!init_global_locks ())
|
|
return FALSE;
|
|
|
|
thread_init_generation = _dbus_current_generation;
|
|
|
|
return TRUE;
|
|
}
|
|
|
|
/** @} */
|
|
|
|
#ifdef DBUS_BUILD_TESTS
|
|
/** Fake mutex used for debugging */
|
|
typedef struct DBusFakeMutex DBusFakeMutex;
|
|
/** Fake mutex used for debugging */
|
|
struct DBusFakeMutex
|
|
{
|
|
dbus_bool_t locked; /**< Mutex is "locked" */
|
|
};
|
|
|
|
static DBusMutex * dbus_fake_mutex_new (void);
|
|
static void dbus_fake_mutex_free (DBusMutex *mutex);
|
|
static dbus_bool_t dbus_fake_mutex_lock (DBusMutex *mutex);
|
|
static dbus_bool_t dbus_fake_mutex_unlock (DBusMutex *mutex);
|
|
static DBusCondVar* dbus_fake_condvar_new (void);
|
|
static void dbus_fake_condvar_free (DBusCondVar *cond);
|
|
static void dbus_fake_condvar_wait (DBusCondVar *cond,
|
|
DBusMutex *mutex);
|
|
static dbus_bool_t dbus_fake_condvar_wait_timeout (DBusCondVar *cond,
|
|
DBusMutex *mutex,
|
|
int timeout_msec);
|
|
static void dbus_fake_condvar_wake_one (DBusCondVar *cond);
|
|
static void dbus_fake_condvar_wake_all (DBusCondVar *cond);
|
|
|
|
|
|
static const DBusThreadFunctions fake_functions =
|
|
{
|
|
DBUS_THREAD_FUNCTIONS_MUTEX_NEW_MASK |
|
|
DBUS_THREAD_FUNCTIONS_MUTEX_FREE_MASK |
|
|
DBUS_THREAD_FUNCTIONS_MUTEX_LOCK_MASK |
|
|
DBUS_THREAD_FUNCTIONS_MUTEX_UNLOCK_MASK |
|
|
DBUS_THREAD_FUNCTIONS_CONDVAR_NEW_MASK |
|
|
DBUS_THREAD_FUNCTIONS_CONDVAR_FREE_MASK |
|
|
DBUS_THREAD_FUNCTIONS_CONDVAR_WAIT_MASK |
|
|
DBUS_THREAD_FUNCTIONS_CONDVAR_WAIT_TIMEOUT_MASK |
|
|
DBUS_THREAD_FUNCTIONS_CONDVAR_WAKE_ONE_MASK|
|
|
DBUS_THREAD_FUNCTIONS_CONDVAR_WAKE_ALL_MASK,
|
|
dbus_fake_mutex_new,
|
|
dbus_fake_mutex_free,
|
|
dbus_fake_mutex_lock,
|
|
dbus_fake_mutex_unlock,
|
|
dbus_fake_condvar_new,
|
|
dbus_fake_condvar_free,
|
|
dbus_fake_condvar_wait,
|
|
dbus_fake_condvar_wait_timeout,
|
|
dbus_fake_condvar_wake_one,
|
|
dbus_fake_condvar_wake_all
|
|
};
|
|
|
|
static DBusMutex *
|
|
dbus_fake_mutex_new (void)
|
|
{
|
|
DBusFakeMutex *mutex;
|
|
|
|
mutex = dbus_new0 (DBusFakeMutex, 1);
|
|
|
|
return (DBusMutex *)mutex;
|
|
}
|
|
|
|
static void
|
|
dbus_fake_mutex_free (DBusMutex *mutex)
|
|
{
|
|
DBusFakeMutex *fake = (DBusFakeMutex*) mutex;
|
|
|
|
_dbus_assert (!fake->locked);
|
|
|
|
dbus_free (fake);
|
|
}
|
|
|
|
static dbus_bool_t
|
|
dbus_fake_mutex_lock (DBusMutex *mutex)
|
|
{
|
|
DBusFakeMutex *fake = (DBusFakeMutex*) mutex;
|
|
|
|
_dbus_assert (!fake->locked);
|
|
|
|
fake->locked = TRUE;
|
|
|
|
return TRUE;
|
|
}
|
|
|
|
static dbus_bool_t
|
|
dbus_fake_mutex_unlock (DBusMutex *mutex)
|
|
{
|
|
DBusFakeMutex *fake = (DBusFakeMutex*) mutex;
|
|
|
|
_dbus_assert (fake->locked);
|
|
|
|
fake->locked = FALSE;
|
|
|
|
return TRUE;
|
|
}
|
|
|
|
static DBusCondVar*
|
|
dbus_fake_condvar_new (void)
|
|
{
|
|
return (DBusCondVar*) _dbus_strdup ("FakeCondvar");
|
|
}
|
|
|
|
static void
|
|
dbus_fake_condvar_free (DBusCondVar *cond)
|
|
{
|
|
dbus_free (cond);
|
|
}
|
|
|
|
static void
|
|
dbus_fake_condvar_wait (DBusCondVar *cond,
|
|
DBusMutex *mutex)
|
|
{
|
|
|
|
}
|
|
|
|
static dbus_bool_t
|
|
dbus_fake_condvar_wait_timeout (DBusCondVar *cond,
|
|
DBusMutex *mutex,
|
|
int timeout_msec)
|
|
{
|
|
return TRUE;
|
|
}
|
|
|
|
static void
|
|
dbus_fake_condvar_wake_one (DBusCondVar *cond)
|
|
{
|
|
|
|
}
|
|
|
|
static void
|
|
dbus_fake_condvar_wake_all (DBusCondVar *cond)
|
|
{
|
|
|
|
}
|
|
|
|
dbus_bool_t
|
|
_dbus_threads_init_debug (void)
|
|
{
|
|
return dbus_threads_init (&fake_functions);
|
|
}
|
|
|
|
#endif /* DBUS_BUILD_TESTS */
|