sl@0: /* -*- mode: C; c-file-style: "gnu" -*- */ sl@0: /* dbus-gproxy.c Proxy for remote objects sl@0: * sl@0: * Copyright (C) 2003, 2004, 2005 Red Hat, Inc. sl@0: * Copyright (C) 2005 Nokia sl@0: * Portion Copyright © 2008 Nokia Corporation and/or its subsidiary(-ies). All rights reserved. sl@0: * Licensed under the Academic Free License version 2.1 sl@0: * sl@0: * This program is free software; you can redistribute it and/or modify sl@0: * it under the terms of the GNU General Public License as published by sl@0: * the Free Software Foundation; either version 2 of the License, or sl@0: * (at your option) any later version. sl@0: * sl@0: * This program is distributed in the hope that it will be useful, sl@0: * but WITHOUT ANY WARRANTY; without even the implied warranty of sl@0: * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the sl@0: * GNU General Public License for more details. sl@0: * sl@0: * You should have received a copy of the GNU General Public License sl@0: * along with this program; if not, write to the Free Software sl@0: * Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA sl@0: * sl@0: */ sl@0: #include sl@0: #include sl@0: #include sl@0: #include "dbus-gutils.h" sl@0: #include "dbus-gsignature.h" sl@0: #include "dbus-gvalue.h" sl@0: #include "dbus-gvalue-utils.h" sl@0: #include "dbus-gobject.h" sl@0: #include sl@0: sl@0: sl@0: #ifdef __SYMBIAN32__ sl@0: #include sl@0: #include sl@0: #endif sl@0: sl@0: #ifndef __SYMBIAN32__ sl@0: #include sl@0: #include sl@0: #define _(x) dgettext (GETTEXT_PACKAGE, x) sl@0: #define N_(x) x sl@0: #else sl@0: sl@0: #define _(x) x sl@0: #define N_(x) x sl@0: #endif sl@0: sl@0: #include sl@0: sl@0: #ifdef __SYMBIAN32__ sl@0: #include "libdbus_glib_wsd_solution.h" sl@0: #endif sl@0: sl@0: #define DBUS_G_PROXY_CALL_TO_ID(x) (GPOINTER_TO_UINT(x)) sl@0: #define DBUS_G_PROXY_ID_TO_CALL(x) (GUINT_TO_POINTER(x)) sl@0: #define DBUS_G_PROXY_GET_PRIVATE(o) \ sl@0: (G_TYPE_INSTANCE_GET_PRIVATE ((o), DBUS_TYPE_G_PROXY, DBusGProxyPrivate)) sl@0: sl@0: sl@0: /** sl@0: * @addtogroup DBusGLibInternals sl@0: * sl@0: * @{ sl@0: */ sl@0: sl@0: /** sl@0: * DBusGProxyManager typedef sl@0: */ sl@0: sl@0: typedef struct _DBusGProxyManager DBusGProxyManager; sl@0: sl@0: typedef struct _DBusGProxyPrivate DBusGProxyPrivate; sl@0: sl@0: /** sl@0: * Internals of DBusGProxy sl@0: */ sl@0: struct _DBusGProxyPrivate sl@0: { sl@0: DBusGProxyManager *manager; /**< Proxy manager */ sl@0: char *name; /**< Name messages go to or NULL */ sl@0: char *path; /**< Path messages go to or NULL */ sl@0: char *interface; /**< Interface messages go to or NULL */ sl@0: sl@0: DBusGProxyCall *name_call; /**< Pending call id to retrieve name owner */ sl@0: guint for_owner : 1; /**< Whether or not this proxy is for a name owner */ sl@0: guint associated : 1; /**< Whether or not this proxy is associated (for name proxies) */ sl@0: sl@0: /* FIXME: make threadsafe? */ sl@0: guint call_id_counter; /**< Integer counter for pending calls */ sl@0: sl@0: GData *signal_signatures; /**< D-BUS signatures for each signal */ sl@0: sl@0: GHashTable *pending_calls; /**< Calls made on this proxy which have not yet returned */ sl@0: }; sl@0: sl@0: static void dbus_g_proxy_init (DBusGProxy *proxy); sl@0: static void dbus_g_proxy_class_init (DBusGProxyClass *klass); sl@0: static GObject *dbus_g_proxy_constructor (GType type, sl@0: guint n_construct_properties, sl@0: GObjectConstructParam *construct_properties); sl@0: static void dbus_g_proxy_set_property (GObject *object, sl@0: guint prop_id, sl@0: const GValue *value, sl@0: GParamSpec *pspec); sl@0: static void dbus_g_proxy_get_property (GObject *object, sl@0: guint prop_id, sl@0: GValue *value, sl@0: GParamSpec *pspec); sl@0: sl@0: static void dbus_g_proxy_finalize (GObject *object); sl@0: static void dbus_g_proxy_dispose (GObject *object); sl@0: static void dbus_g_proxy_destroy (DBusGProxy *proxy); sl@0: static void dbus_g_proxy_emit_remote_signal (DBusGProxy *proxy, sl@0: DBusMessage *message); sl@0: sl@0: static DBusGProxyCall *manager_begin_bus_call (DBusGProxyManager *manager, sl@0: const char *method, sl@0: DBusGProxyCallNotify notify, sl@0: gpointer data, sl@0: GDestroyNotify destroy, sl@0: GType first_arg_type, sl@0: ...); sl@0: static guint dbus_g_proxy_begin_call_internal (DBusGProxy *proxy, sl@0: const char *method, sl@0: DBusGProxyCallNotify notify, sl@0: gpointer data, sl@0: GDestroyNotify destroy, sl@0: GValueArray *args, sl@0: int timeout ); sl@0: static gboolean dbus_g_proxy_end_call_internal (DBusGProxy *proxy, sl@0: guint call_id, sl@0: GError **error, sl@0: GType first_arg_type, sl@0: va_list args); sl@0: sl@0: /** sl@0: * A list of proxies with a given name+path+interface, used to sl@0: * route incoming signals. sl@0: */ sl@0: typedef struct sl@0: { sl@0: GSList *proxies; /**< The list of proxies */ sl@0: sl@0: char name[4]; /**< name (empty string for none), nul byte, sl@0: * path, nul byte, sl@0: * interface, nul byte sl@0: */ sl@0: sl@0: } DBusGProxyList; sl@0: sl@0: /** sl@0: * DBusGProxyManager's primary task is to route signals to the proxies sl@0: * those signals are emitted on. In order to do this it also has to sl@0: * track the owners of the names proxies are bound to. sl@0: */ sl@0: struct _DBusGProxyManager sl@0: { sl@0: GStaticMutex lock; /**< Thread lock */ sl@0: int refcount; /**< Reference count */ sl@0: DBusConnection *connection; /**< Connection we're associated with. */ sl@0: sl@0: DBusGProxy *bus_proxy; /**< Special internal proxy used to talk to the bus */ sl@0: sl@0: GHashTable *proxy_lists; /**< Hash used to route incoming signals sl@0: * and iterate over proxies sl@0: */ sl@0: GHashTable *owner_names; /**< Hash to keep track of mapping from sl@0: * base name -> [name,name,...] for proxies which sl@0: * are for names. sl@0: */ sl@0: GSList *unassociated_proxies; /**< List of name proxies for which sl@0: * there was no result for sl@0: * GetNameOwner sl@0: */ sl@0: }; sl@0: sl@0: static DBusGProxyManager *dbus_g_proxy_manager_ref (DBusGProxyManager *manager); sl@0: static DBusHandlerResult dbus_g_proxy_manager_filter (DBusConnection *connection, sl@0: DBusMessage *message, sl@0: void *user_data); sl@0: sl@0: sl@0: /** Lock the DBusGProxyManager */ sl@0: #define LOCK_MANAGER(mgr) (g_static_mutex_lock (&(mgr)->lock)) sl@0: /** Unlock the DBusGProxyManager */ sl@0: #define UNLOCK_MANAGER(mgr) (g_static_mutex_unlock (&(mgr)->lock)) sl@0: sl@0: sl@0: #if EMULATOR sl@0: GET_STATIC_VAR_FROM_TLS(g_proxy_manager_slot,dbus_gproxy,int ) sl@0: #define g_proxy_manager_slot (*GET_DBUS_WSD_VAR_NAME(g_proxy_manager_slot,dbus_gproxy,s)()) sl@0: GET_STATIC_VAR_FROM_TLS(connection_g_proxy_lock,dbus_gproxy,GStaticMutex ) sl@0: #define connection_g_proxy_lock (*GET_DBUS_WSD_VAR_NAME(connection_g_proxy_lock,dbus_gproxy,s)()) sl@0: sl@0: #else sl@0: static int g_proxy_manager_slot = -1; sl@0: /* Lock controlling get/set manager as data on each connection */ sl@0: static GStaticMutex connection_g_proxy_lock = G_STATIC_MUTEX_INIT; sl@0: sl@0: #endif sl@0: sl@0: sl@0: sl@0: sl@0: static DBusGProxyManager* sl@0: dbus_g_proxy_manager_get (DBusConnection *connection) sl@0: { sl@0: DBusGProxyManager *manager; sl@0: sl@0: dbus_connection_allocate_data_slot (&g_proxy_manager_slot); sl@0: if (g_proxy_manager_slot < 0) sl@0: g_error ("out of memory"); sl@0: sl@0: g_static_mutex_lock (&connection_g_proxy_lock); sl@0: sl@0: manager = dbus_connection_get_data (connection, g_proxy_manager_slot); sl@0: if (manager != NULL) sl@0: { sl@0: dbus_connection_free_data_slot (&g_proxy_manager_slot); sl@0: dbus_g_proxy_manager_ref (manager); sl@0: g_static_mutex_unlock (&connection_g_proxy_lock); sl@0: return manager; sl@0: } sl@0: sl@0: manager = g_new0 (DBusGProxyManager, 1); sl@0: sl@0: manager->refcount = 1; sl@0: manager->connection = connection; sl@0: sl@0: g_static_mutex_init (&manager->lock); sl@0: sl@0: /* Proxy managers keep the connection alive, which means that sl@0: * DBusGProxy indirectly does. To free a connection you have to free sl@0: * all the proxies referring to it. sl@0: */ sl@0: dbus_connection_ref (manager->connection); sl@0: sl@0: dbus_connection_set_data (connection, g_proxy_manager_slot, sl@0: manager, NULL); sl@0: sl@0: dbus_connection_add_filter (connection, dbus_g_proxy_manager_filter, sl@0: manager, NULL); sl@0: sl@0: g_static_mutex_unlock (&connection_g_proxy_lock); sl@0: sl@0: return manager; sl@0: } sl@0: sl@0: static DBusGProxyManager * sl@0: dbus_g_proxy_manager_ref (DBusGProxyManager *manager) sl@0: { sl@0: g_assert (manager != NULL); sl@0: g_assert (manager->refcount > 0); sl@0: sl@0: LOCK_MANAGER (manager); sl@0: sl@0: manager->refcount += 1; sl@0: sl@0: UNLOCK_MANAGER (manager); sl@0: sl@0: return manager; sl@0: } sl@0: sl@0: static void sl@0: dbus_g_proxy_manager_unref (DBusGProxyManager *manager) sl@0: { sl@0: g_assert (manager != NULL); sl@0: g_assert (manager->refcount > 0); sl@0: sl@0: LOCK_MANAGER (manager); sl@0: manager->refcount -= 1; sl@0: if (manager->refcount == 0) sl@0: { sl@0: UNLOCK_MANAGER (manager); sl@0: sl@0: if (manager->bus_proxy) sl@0: g_object_unref (manager->bus_proxy); sl@0: sl@0: if (manager->proxy_lists) sl@0: { sl@0: /* can't have any proxies left since they hold sl@0: * a reference to the proxy manager. sl@0: */ sl@0: g_assert (g_hash_table_size (manager->proxy_lists) == 0); sl@0: sl@0: g_hash_table_destroy (manager->proxy_lists); sl@0: manager->proxy_lists = NULL; sl@0: sl@0: } sl@0: sl@0: if (manager->owner_names) sl@0: { sl@0: /* Since we destroyed all proxies, none can be tracking sl@0: * name owners sl@0: */ sl@0: g_assert (g_hash_table_size (manager->owner_names) == 0); sl@0: sl@0: g_hash_table_destroy (manager->owner_names); sl@0: manager->owner_names = NULL; sl@0: } sl@0: sl@0: g_assert (manager->unassociated_proxies == NULL); sl@0: sl@0: g_static_mutex_free (&manager->lock); sl@0: sl@0: g_static_mutex_lock (&connection_g_proxy_lock); sl@0: sl@0: dbus_connection_remove_filter (manager->connection, dbus_g_proxy_manager_filter, sl@0: manager); sl@0: sl@0: dbus_connection_set_data (manager->connection, sl@0: g_proxy_manager_slot, sl@0: NULL, NULL); sl@0: sl@0: g_static_mutex_unlock (&connection_g_proxy_lock); sl@0: sl@0: dbus_connection_unref (manager->connection); sl@0: g_free (manager); sl@0: sl@0: dbus_connection_free_data_slot (&g_proxy_manager_slot); sl@0: } sl@0: else sl@0: { sl@0: UNLOCK_MANAGER (manager); sl@0: } sl@0: } sl@0: sl@0: static guint sl@0: tristring_hash (gconstpointer key) sl@0: { sl@0: const char *p = key; sl@0: guint h = *p; sl@0: sl@0: if (h) sl@0: { sl@0: for (p += 1; *p != '\0'; p++) sl@0: h = (h << 5) - h + *p; sl@0: } sl@0: sl@0: /* skip nul and do the next substring */ sl@0: for (p += 1; *p != '\0'; p++) sl@0: h = (h << 5) - h + *p; sl@0: sl@0: /* skip nul again and another substring */ sl@0: for (p += 1; *p != '\0'; p++) sl@0: h = (h << 5) - h + *p; sl@0: sl@0: return h; sl@0: } sl@0: sl@0: static gboolean sl@0: strequal_len (const char *a, sl@0: const char *b, sl@0: size_t *lenp) sl@0: { sl@0: size_t a_len; sl@0: size_t b_len; sl@0: sl@0: a_len = strlen (a); sl@0: b_len = strlen (b); sl@0: sl@0: if (a_len != b_len) sl@0: return FALSE; sl@0: sl@0: if (memcmp (a, b, a_len) != 0) sl@0: return FALSE; sl@0: sl@0: *lenp = a_len; sl@0: sl@0: return TRUE; sl@0: } sl@0: sl@0: static gboolean sl@0: tristring_equal (gconstpointer a, sl@0: gconstpointer b) sl@0: { sl@0: const char *ap = a; sl@0: const char *bp = b; sl@0: size_t len; sl@0: sl@0: if (!strequal_len (ap, bp, &len)) sl@0: return FALSE; sl@0: sl@0: ap += len + 1; sl@0: bp += len + 1; sl@0: sl@0: if (!strequal_len (ap, bp, &len)) sl@0: return FALSE; sl@0: sl@0: ap += len + 1; sl@0: bp += len + 1; sl@0: sl@0: if (strcmp (ap, bp) != 0) sl@0: return FALSE; sl@0: sl@0: return TRUE; sl@0: } sl@0: sl@0: static char* sl@0: tristring_alloc_from_strings (size_t padding_before, sl@0: const char *name, sl@0: const char *path, sl@0: const char *interface) sl@0: { sl@0: size_t name_len, iface_len, path_len, len; sl@0: char *tri; sl@0: sl@0: if (name) sl@0: name_len = strlen (name); sl@0: else sl@0: name_len = 0; sl@0: sl@0: path_len = strlen (path); sl@0: sl@0: iface_len = strlen (interface); sl@0: sl@0: tri = g_malloc (padding_before + name_len + path_len + iface_len + 3); sl@0: sl@0: len = padding_before; sl@0: sl@0: if (name) sl@0: memcpy (&tri[len], name, name_len); sl@0: sl@0: len += name_len; sl@0: tri[len] = '\0'; sl@0: len += 1; sl@0: sl@0: g_assert (len == (padding_before + name_len + 1)); sl@0: sl@0: memcpy (&tri[len], path, path_len); sl@0: len += path_len; sl@0: tri[len] = '\0'; sl@0: len += 1; sl@0: sl@0: g_assert (len == (padding_before + name_len + path_len + 2)); sl@0: sl@0: memcpy (&tri[len], interface, iface_len); sl@0: len += iface_len; sl@0: tri[len] = '\0'; sl@0: len += 1; sl@0: sl@0: g_assert (len == (padding_before + name_len + path_len + iface_len + 3)); sl@0: sl@0: return tri; sl@0: } sl@0: sl@0: static char* sl@0: tristring_from_proxy (DBusGProxy *proxy) sl@0: { sl@0: DBusGProxyPrivate *priv = DBUS_G_PROXY_GET_PRIVATE(proxy); sl@0: sl@0: return tristring_alloc_from_strings (0, sl@0: priv->name, sl@0: priv->path, sl@0: priv->interface); sl@0: } sl@0: sl@0: static char* sl@0: tristring_from_message (DBusMessage *message) sl@0: { sl@0: const char *path; sl@0: const char *interface; sl@0: sl@0: path = dbus_message_get_path (message); sl@0: interface = dbus_message_get_interface (message); sl@0: sl@0: g_assert (path); sl@0: g_assert (interface); sl@0: sl@0: return tristring_alloc_from_strings (0, sl@0: dbus_message_get_sender (message), sl@0: path, interface); sl@0: } sl@0: sl@0: static DBusGProxyList* sl@0: g_proxy_list_new (DBusGProxy *first_proxy) sl@0: { sl@0: DBusGProxyList *list; sl@0: DBusGProxyPrivate *priv = DBUS_G_PROXY_GET_PRIVATE(first_proxy); sl@0: sl@0: list = (void*) tristring_alloc_from_strings (G_STRUCT_OFFSET (DBusGProxyList, name), sl@0: priv->name, sl@0: priv->path, sl@0: priv->interface); sl@0: list->proxies = NULL; sl@0: sl@0: return list; sl@0: } sl@0: sl@0: static void sl@0: g_proxy_list_free (DBusGProxyList *list) sl@0: { sl@0: /* we don't hold a reference to the proxies in the list, sl@0: * as they ref the GProxyManager sl@0: */ sl@0: g_slist_free (list->proxies); sl@0: sl@0: g_free (list); sl@0: } sl@0: sl@0: static char* sl@0: g_proxy_get_match_rule (DBusGProxy *proxy) sl@0: { sl@0: DBusGProxyPrivate *priv = DBUS_G_PROXY_GET_PRIVATE(proxy); sl@0: /* FIXME Escaping is required here */ sl@0: sl@0: if (priv->name) sl@0: return g_strdup_printf ("type='signal',sender='%s',path='%s',interface='%s'", sl@0: priv->name, priv->path, priv->interface); sl@0: else sl@0: return g_strdup_printf ("type='signal',path='%s',interface='%s'", sl@0: priv->path, priv->interface); sl@0: } sl@0: sl@0: typedef struct sl@0: { sl@0: char *name; sl@0: guint refcount; sl@0: } DBusGProxyNameOwnerInfo; sl@0: sl@0: static gint sl@0: find_name_in_info (gconstpointer a, gconstpointer b) sl@0: { sl@0: const DBusGProxyNameOwnerInfo *info = a; sl@0: const char *name = b; sl@0: sl@0: return strcmp (info->name, name); sl@0: } sl@0: sl@0: typedef struct sl@0: { sl@0: const char *name; sl@0: const char *owner; sl@0: DBusGProxyNameOwnerInfo *info; sl@0: } DBusGProxyNameOwnerForeachData; sl@0: sl@0: static void sl@0: name_owner_foreach (gpointer key, gpointer val, gpointer data) sl@0: { sl@0: const char *owner; sl@0: DBusGProxyNameOwnerForeachData *foreach_data; sl@0: GSList *names; sl@0: GSList *link; sl@0: sl@0: owner = key; sl@0: names = val; sl@0: foreach_data = data; sl@0: sl@0: if (foreach_data->owner != NULL) sl@0: return; sl@0: sl@0: g_assert (foreach_data->info == NULL); sl@0: sl@0: link = g_slist_find_custom (names, foreach_data->name, find_name_in_info); sl@0: if (link) sl@0: { sl@0: foreach_data->owner = owner; sl@0: foreach_data->info = link->data; sl@0: } sl@0: } sl@0: sl@0: static gboolean sl@0: dbus_g_proxy_manager_lookup_name_owner (DBusGProxyManager *manager, sl@0: const char *name, sl@0: DBusGProxyNameOwnerInfo **info, sl@0: const char **owner) sl@0: { sl@0: DBusGProxyNameOwnerForeachData foreach_data; sl@0: sl@0: foreach_data.name = name; sl@0: foreach_data.owner = NULL; sl@0: foreach_data.info = NULL; sl@0: sl@0: g_hash_table_foreach (manager->owner_names, name_owner_foreach, &foreach_data); sl@0: sl@0: *info = foreach_data.info; sl@0: *owner = foreach_data.owner; sl@0: return *info != NULL; sl@0: } sl@0: sl@0: static void sl@0: insert_nameinfo (DBusGProxyManager *manager, sl@0: const char *owner, sl@0: DBusGProxyNameOwnerInfo *info) sl@0: { sl@0: GSList *names; sl@0: gboolean insert; sl@0: sl@0: names = g_hash_table_lookup (manager->owner_names, owner); sl@0: sl@0: /* Only need to g_hash_table_insert the first time */ sl@0: insert = (names == NULL); sl@0: sl@0: names = g_slist_append (names, info); sl@0: sl@0: if (insert) sl@0: g_hash_table_insert (manager->owner_names, g_strdup (owner), names); sl@0: } sl@0: sl@0: static void sl@0: dbus_g_proxy_manager_monitor_name_owner (DBusGProxyManager *manager, sl@0: const char *owner, sl@0: const char *name) sl@0: { sl@0: GSList *names; sl@0: GSList *link; sl@0: DBusGProxyNameOwnerInfo *nameinfo; sl@0: sl@0: names = g_hash_table_lookup (manager->owner_names, owner); sl@0: link = g_slist_find_custom (names, name, find_name_in_info); sl@0: sl@0: if (!link) sl@0: { sl@0: nameinfo = g_new0 (DBusGProxyNameOwnerInfo, 1); sl@0: nameinfo->name = g_strdup (name); sl@0: nameinfo->refcount = 1; sl@0: sl@0: insert_nameinfo (manager, owner, nameinfo); sl@0: } sl@0: else sl@0: { sl@0: nameinfo = link->data; sl@0: nameinfo->refcount++; sl@0: } sl@0: } sl@0: sl@0: static void sl@0: dbus_g_proxy_manager_unmonitor_name_owner (DBusGProxyManager *manager, sl@0: const char *name) sl@0: { sl@0: DBusGProxyNameOwnerInfo *info; sl@0: const char *owner; sl@0: gboolean ret; sl@0: sl@0: ret = dbus_g_proxy_manager_lookup_name_owner (manager, name, &info, &owner); sl@0: g_assert (ret); sl@0: g_assert (info != NULL); sl@0: g_assert (owner != NULL); sl@0: sl@0: info->refcount--; sl@0: if (info->refcount == 0) sl@0: { sl@0: GSList *names; sl@0: GSList *link; sl@0: sl@0: names = g_hash_table_lookup (manager->owner_names, owner); sl@0: link = g_slist_find_custom (names, name, find_name_in_info); sl@0: names = g_slist_delete_link (names, link); sl@0: if (names != NULL) sl@0: g_hash_table_insert (manager->owner_names, g_strdup (owner), names); sl@0: else sl@0: g_hash_table_remove (manager->owner_names, owner); sl@0: sl@0: g_free (info->name); sl@0: g_free (info); sl@0: } sl@0: } sl@0: sl@0: typedef struct sl@0: { sl@0: const char *name; sl@0: GSList *destroyed; sl@0: } DBusGProxyUnassociateData; sl@0: sl@0: static void sl@0: unassociate_proxies (gpointer key, gpointer val, gpointer user_data) sl@0: { sl@0: DBusGProxyList *list; sl@0: const char *name; sl@0: GSList *tmp; sl@0: DBusGProxyUnassociateData *data; sl@0: sl@0: list = val; sl@0: data = user_data; sl@0: name = data->name; sl@0: sl@0: for (tmp = list->proxies; tmp; tmp = tmp->next) sl@0: { sl@0: DBusGProxy *proxy = DBUS_G_PROXY (tmp->data); sl@0: DBusGProxyPrivate *priv = DBUS_G_PROXY_GET_PRIVATE(proxy); sl@0: DBusGProxyManager *manager; sl@0: sl@0: manager = priv->manager; sl@0: sl@0: if (!strcmp (priv->name, name)) sl@0: { sl@0: if (!priv->for_owner) sl@0: { sl@0: g_assert (priv->associated); sl@0: g_assert (priv->name_call == NULL); sl@0: sl@0: priv->associated = FALSE; sl@0: manager->unassociated_proxies = g_slist_prepend (manager->unassociated_proxies, proxy); sl@0: } sl@0: else sl@0: { sl@0: data->destroyed = g_slist_prepend (data->destroyed, proxy); sl@0: /* make contents of list into weak pointers in case the objects sl@0: * unref each other when disposing */ sl@0: g_object_add_weak_pointer (G_OBJECT (proxy), sl@0: &(data->destroyed->data)); sl@0: } sl@0: } sl@0: } sl@0: } sl@0: sl@0: static void sl@0: dbus_g_proxy_manager_replace_name_owner (DBusGProxyManager *manager, sl@0: const char *name, sl@0: const char *prev_owner, sl@0: const char *new_owner) sl@0: { sl@0: GSList *names; sl@0: sl@0: if (prev_owner[0] == '\0') sl@0: { sl@0: GSList *tmp; sl@0: GSList *removed; sl@0: sl@0: /* We have a new service, look at unassociated proxies */ sl@0: sl@0: removed = NULL; sl@0: sl@0: for (tmp = manager->unassociated_proxies; tmp ; tmp = tmp->next) sl@0: { sl@0: DBusGProxy *proxy = tmp->data; sl@0: DBusGProxyPrivate *priv = DBUS_G_PROXY_GET_PRIVATE(proxy); sl@0: sl@0: if (!strcmp (priv->name, name)) sl@0: { sl@0: removed = g_slist_prepend (removed, tmp); sl@0: sl@0: dbus_g_proxy_manager_monitor_name_owner (manager, new_owner, name); sl@0: priv->associated = TRUE; sl@0: } sl@0: } sl@0: sl@0: for (tmp = removed; tmp; tmp = tmp->next) sl@0: manager->unassociated_proxies = g_slist_delete_link (manager->unassociated_proxies, tmp->data); sl@0: g_slist_free (removed); sl@0: } sl@0: else sl@0: { sl@0: DBusGProxyNameOwnerInfo *info; sl@0: GSList *link; sl@0: sl@0: /* Name owner changed or deleted */ sl@0: sl@0: names = g_hash_table_lookup (manager->owner_names, prev_owner); sl@0: sl@0: info = NULL; sl@0: if (names != NULL) sl@0: { sl@0: link = g_slist_find_custom (names, name, find_name_in_info); sl@0: sl@0: if (link != NULL) sl@0: { sl@0: info = link->data; sl@0: sl@0: names = g_slist_delete_link (names, link); sl@0: sl@0: if (names == NULL) sl@0: g_hash_table_remove (manager->owner_names, prev_owner); sl@0: } sl@0: } sl@0: sl@0: if (new_owner[0] == '\0') sl@0: { sl@0: DBusGProxyUnassociateData data; sl@0: GSList *tmp; sl@0: sl@0: data.name = name; sl@0: data.destroyed = NULL; sl@0: sl@0: /* A service went away, we need to unassociate proxies */ sl@0: g_hash_table_foreach (manager->proxy_lists, sl@0: unassociate_proxies, &data); sl@0: sl@0: UNLOCK_MANAGER (manager); sl@0: sl@0: /* the destroyed list's data pointers are weak pointers, so that we sl@0: * don't end up calling destroy on proxies which have already been sl@0: * freed up as a result of other ones being destroyed */ sl@0: for (tmp = data.destroyed; tmp; tmp = tmp->next) sl@0: if (tmp->data != NULL) sl@0: { sl@0: g_object_remove_weak_pointer (G_OBJECT (tmp->data), sl@0: &(tmp->data)); sl@0: dbus_g_proxy_destroy (tmp->data); sl@0: } sl@0: g_slist_free (data.destroyed); sl@0: sl@0: LOCK_MANAGER (manager); sl@0: sl@0: if (info) sl@0: { sl@0: g_free (info->name); sl@0: g_free (info); sl@0: } sl@0: } sl@0: else if (info) sl@0: { sl@0: insert_nameinfo (manager, new_owner, info); sl@0: } sl@0: } sl@0: } sl@0: sl@0: static void sl@0: got_name_owner_cb (DBusGProxy *bus_proxy, sl@0: DBusGProxyCall *call, sl@0: void *user_data) sl@0: { sl@0: DBusGProxy *proxy = user_data; sl@0: DBusGProxyPrivate *priv = DBUS_G_PROXY_GET_PRIVATE(proxy); sl@0: GError *error; sl@0: char *owner; sl@0: sl@0: error = NULL; sl@0: owner = NULL; sl@0: sl@0: LOCK_MANAGER (priv->manager); sl@0: sl@0: if (!dbus_g_proxy_end_call (bus_proxy, call, &error, sl@0: G_TYPE_STRING, &owner, sl@0: G_TYPE_INVALID)) sl@0: { sl@0: if (error->domain == DBUS_GERROR && error->code == DBUS_GERROR_NAME_HAS_NO_OWNER) sl@0: { sl@0: priv->manager->unassociated_proxies = g_slist_prepend (priv->manager->unassociated_proxies, proxy); sl@0: } sl@0: else if (error->domain == DBUS_GERROR && error->code == DBUS_GERROR_REMOTE_EXCEPTION) sl@0: g_warning ("Couldn't get name owner (%s): %s", sl@0: dbus_g_error_get_name (error), sl@0: error->message); sl@0: else sl@0: g_warning ("Couldn't get name owner (code %d): %s", sl@0: error->code, error->message); sl@0: g_clear_error (&error); sl@0: goto out; sl@0: } sl@0: else sl@0: { sl@0: dbus_g_proxy_manager_monitor_name_owner (priv->manager, owner, priv->name); sl@0: priv->associated = TRUE; sl@0: } sl@0: sl@0: out: sl@0: priv->name_call = NULL; sl@0: UNLOCK_MANAGER (priv->manager); sl@0: g_free (owner); sl@0: } sl@0: sl@0: static char * sl@0: get_name_owner (DBusConnection *connection, sl@0: const char *name, sl@0: GError **error) sl@0: { sl@0: DBusError derror; sl@0: DBusMessage *request, *reply; sl@0: char *base_name; sl@0: sl@0: dbus_error_init (&derror); sl@0: sl@0: base_name = NULL; sl@0: reply = NULL; sl@0: sl@0: request = dbus_message_new_method_call (DBUS_SERVICE_DBUS, sl@0: DBUS_PATH_DBUS, sl@0: DBUS_INTERFACE_DBUS, sl@0: "GetNameOwner"); sl@0: if (request == NULL) sl@0: g_error ("Out of memory"); sl@0: sl@0: if (!dbus_message_append_args (request, sl@0: DBUS_TYPE_STRING, &name, sl@0: DBUS_TYPE_INVALID)) sl@0: g_error ("Out of memory"); sl@0: sl@0: reply = sl@0: dbus_connection_send_with_reply_and_block (connection, sl@0: request, sl@0: 2000, &derror); sl@0: if (reply == NULL) sl@0: goto error; sl@0: sl@0: if (dbus_set_error_from_message (&derror, reply)) sl@0: goto error; sl@0: sl@0: if (!dbus_message_get_args (reply, &derror, sl@0: DBUS_TYPE_STRING, &base_name, sl@0: DBUS_TYPE_INVALID)) sl@0: goto error; sl@0: sl@0: base_name = g_strdup (base_name); sl@0: goto out; sl@0: sl@0: error: sl@0: g_assert (dbus_error_is_set (&derror)); sl@0: dbus_set_g_error (error, &derror); sl@0: dbus_error_free (&derror); sl@0: sl@0: out: sl@0: if (request) sl@0: dbus_message_unref (request); sl@0: if (reply) sl@0: dbus_message_unref (reply); sl@0: sl@0: return base_name; sl@0: } sl@0: sl@0: sl@0: static void sl@0: dbus_g_proxy_manager_register (DBusGProxyManager *manager, sl@0: DBusGProxy *proxy) sl@0: { sl@0: DBusGProxyList *list; sl@0: DBusGProxyPrivate *priv = DBUS_G_PROXY_GET_PRIVATE(proxy); sl@0: sl@0: LOCK_MANAGER (manager); sl@0: sl@0: if (manager->proxy_lists == NULL) sl@0: { sl@0: g_assert (manager->owner_names == NULL); sl@0: sl@0: list = NULL; sl@0: manager->proxy_lists = g_hash_table_new_full (tristring_hash, sl@0: tristring_equal, sl@0: NULL, sl@0: (GFreeFunc) g_proxy_list_free); sl@0: manager->owner_names = g_hash_table_new_full (g_str_hash, sl@0: g_str_equal, sl@0: g_free, sl@0: NULL); sl@0: /* FIXME - for now we listen for all NameOwnerChanged; once sl@0: * Anders' detail patch lands we should add individual rules sl@0: */ sl@0: dbus_bus_add_match (manager->connection, sl@0: "type='signal',sender='" DBUS_SERVICE_DBUS sl@0: "',path='" DBUS_PATH_DBUS sl@0: "',interface='" DBUS_INTERFACE_DBUS sl@0: "',member='NameOwnerChanged'", sl@0: NULL); sl@0: } sl@0: else sl@0: { sl@0: char *tri; sl@0: sl@0: tri = tristring_from_proxy (proxy); sl@0: sl@0: list = g_hash_table_lookup (manager->proxy_lists, tri); sl@0: sl@0: g_free (tri); sl@0: } sl@0: sl@0: if (list == NULL) sl@0: { sl@0: list = g_proxy_list_new (proxy); sl@0: sl@0: g_hash_table_replace (manager->proxy_lists, sl@0: list->name, list); sl@0: } sl@0: sl@0: if (list->proxies == NULL && priv->name) sl@0: { sl@0: /* We have to add the match rule to the server, sl@0: * but only if the server is a message bus, sl@0: * not if it's a peer. sl@0: */ sl@0: char *rule; sl@0: sl@0: rule = g_proxy_get_match_rule (proxy); sl@0: sl@0: /* We don't check for errors; it's not like anyone would handle them, and sl@0: * we don't want a round trip here. sl@0: */ sl@0: dbus_bus_add_match (manager->connection, sl@0: rule, NULL); sl@0: sl@0: g_free (rule); sl@0: } sl@0: sl@0: g_assert (g_slist_find (list->proxies, proxy) == NULL); sl@0: sl@0: list->proxies = g_slist_prepend (list->proxies, proxy); sl@0: sl@0: if (!priv->for_owner) sl@0: { sl@0: const char *owner; sl@0: DBusGProxyNameOwnerInfo *info; sl@0: sl@0: if (!dbus_g_proxy_manager_lookup_name_owner (manager, priv->name, &info, &owner)) sl@0: { sl@0: priv->name_call = manager_begin_bus_call (manager, "GetNameOwner", sl@0: got_name_owner_cb, sl@0: proxy, NULL, sl@0: G_TYPE_STRING, sl@0: priv->name, sl@0: G_TYPE_INVALID); sl@0: sl@0: priv->associated = FALSE; sl@0: } sl@0: else sl@0: { sl@0: info->refcount++; sl@0: priv->associated = TRUE; sl@0: } sl@0: } sl@0: sl@0: UNLOCK_MANAGER (manager); sl@0: } sl@0: sl@0: static void sl@0: dbus_g_proxy_manager_unregister (DBusGProxyManager *manager, sl@0: DBusGProxy *proxy) sl@0: { sl@0: DBusGProxyList *list; sl@0: DBusGProxyPrivate *priv = DBUS_G_PROXY_GET_PRIVATE(proxy); sl@0: char *tri; sl@0: sl@0: LOCK_MANAGER (manager); sl@0: sl@0: #ifndef G_DISABLE_CHECKS sl@0: if (manager->proxy_lists == NULL) sl@0: { sl@0: g_warning ("Trying to unregister a proxy but there aren't any registered"); sl@0: return; sl@0: } sl@0: #endif sl@0: sl@0: tri = tristring_from_proxy (proxy); sl@0: sl@0: list = g_hash_table_lookup (manager->proxy_lists, tri); sl@0: sl@0: #ifndef G_DISABLE_CHECKS sl@0: if (list == NULL) sl@0: { sl@0: g_warning ("Trying to unregister a proxy but it isn't registered"); sl@0: return; sl@0: } sl@0: #endif sl@0: sl@0: g_assert (g_slist_find (list->proxies, proxy) != NULL); sl@0: sl@0: list->proxies = g_slist_remove (list->proxies, proxy); sl@0: sl@0: g_assert (g_slist_find (list->proxies, proxy) == NULL); sl@0: sl@0: if (!priv->for_owner) sl@0: { sl@0: if (!priv->associated) sl@0: { sl@0: GSList *link; sl@0: sl@0: if (priv->name_call != 0) sl@0: { sl@0: dbus_g_proxy_cancel_call (manager->bus_proxy, priv->name_call); sl@0: priv->name_call = 0; sl@0: } sl@0: else sl@0: { sl@0: link = g_slist_find (manager->unassociated_proxies, proxy); sl@0: g_assert (link != NULL); sl@0: sl@0: manager->unassociated_proxies = g_slist_delete_link (manager->unassociated_proxies, link); sl@0: } sl@0: } sl@0: else sl@0: { sl@0: g_assert (priv->name_call == 0); sl@0: sl@0: dbus_g_proxy_manager_unmonitor_name_owner (manager, priv->name); sl@0: } sl@0: } sl@0: sl@0: if (list->proxies == NULL) sl@0: { sl@0: char *rule; sl@0: g_hash_table_remove (manager->proxy_lists, sl@0: tri); sl@0: list = NULL; sl@0: sl@0: rule = g_proxy_get_match_rule (proxy); sl@0: dbus_bus_remove_match (manager->connection, sl@0: rule, NULL); sl@0: g_free (rule); sl@0: } sl@0: sl@0: if (g_hash_table_size (manager->proxy_lists) == 0) sl@0: { sl@0: g_hash_table_destroy (manager->proxy_lists); sl@0: manager->proxy_lists = NULL; sl@0: } sl@0: sl@0: g_free (tri); sl@0: sl@0: UNLOCK_MANAGER (manager); sl@0: } sl@0: sl@0: static void sl@0: list_proxies_foreach (gpointer key, sl@0: gpointer value, sl@0: gpointer user_data) sl@0: { sl@0: DBusGProxyList *list; sl@0: GSList **ret; sl@0: GSList *tmp; sl@0: sl@0: list = value; sl@0: ret = user_data; sl@0: sl@0: tmp = list->proxies; sl@0: while (tmp != NULL) sl@0: { sl@0: DBusGProxy *proxy = DBUS_G_PROXY (tmp->data); sl@0: sl@0: g_object_ref (proxy); sl@0: *ret = g_slist_prepend (*ret, proxy); sl@0: sl@0: tmp = tmp->next; sl@0: } sl@0: } sl@0: sl@0: static GSList* sl@0: dbus_g_proxy_manager_list_all (DBusGProxyManager *manager) sl@0: { sl@0: GSList *ret; sl@0: sl@0: ret = NULL; sl@0: sl@0: if (manager->proxy_lists) sl@0: { sl@0: g_hash_table_foreach (manager->proxy_lists, sl@0: list_proxies_foreach, sl@0: &ret); sl@0: } sl@0: sl@0: return ret; sl@0: } sl@0: sl@0: static DBusHandlerResult sl@0: dbus_g_proxy_manager_filter (DBusConnection *connection, sl@0: DBusMessage *message, sl@0: void *user_data) sl@0: { sl@0: DBusGProxyManager *manager; sl@0: sl@0: if (dbus_message_get_type (message) != DBUS_MESSAGE_TYPE_SIGNAL) sl@0: return DBUS_HANDLER_RESULT_NOT_YET_HANDLED; sl@0: sl@0: manager = user_data; sl@0: sl@0: dbus_g_proxy_manager_ref (manager); sl@0: sl@0: LOCK_MANAGER (manager); sl@0: sl@0: if (dbus_message_is_signal (message, sl@0: DBUS_INTERFACE_LOCAL, sl@0: "Disconnected")) sl@0: { sl@0: /* Destroy all the proxies, quite possibly resulting in unreferencing sl@0: * the proxy manager and the connection as well. sl@0: */ sl@0: GSList *all; sl@0: GSList *tmp; sl@0: sl@0: all = dbus_g_proxy_manager_list_all (manager); sl@0: sl@0: tmp = all; sl@0: while (tmp != NULL) sl@0: { sl@0: DBusGProxy *proxy; sl@0: sl@0: proxy = DBUS_G_PROXY (tmp->data); sl@0: sl@0: UNLOCK_MANAGER (manager); sl@0: dbus_g_proxy_destroy (proxy); sl@0: g_object_unref (G_OBJECT (proxy)); sl@0: LOCK_MANAGER (manager); sl@0: sl@0: tmp = tmp->next; sl@0: } sl@0: sl@0: g_slist_free (all); sl@0: sl@0: #ifndef G_DISABLE_CHECKS sl@0: if (manager->proxy_lists != NULL) sl@0: g_warning ("Disconnection emitted \"destroy\" on all DBusGProxy, but somehow new proxies were created in response to one of those destroy signals. This will cause a memory leak."); sl@0: #endif sl@0: } sl@0: else sl@0: { sl@0: char *tri; sl@0: GSList *full_list; sl@0: GSList *owned_names; sl@0: GSList *tmp; sl@0: const char *sender; sl@0: sl@0: /* First we handle NameOwnerChanged internally */ sl@0: if (dbus_message_is_signal (message, sl@0: DBUS_INTERFACE_DBUS, sl@0: "NameOwnerChanged")) sl@0: { sl@0: const char *name; sl@0: const char *prev_owner; sl@0: const char *new_owner; sl@0: DBusError derr; sl@0: sl@0: dbus_error_init (&derr); sl@0: if (!dbus_message_get_args (message, sl@0: &derr, sl@0: DBUS_TYPE_STRING, sl@0: &name, sl@0: DBUS_TYPE_STRING, sl@0: &prev_owner, sl@0: DBUS_TYPE_STRING, sl@0: &new_owner, sl@0: DBUS_TYPE_INVALID)) sl@0: { sl@0: /* Ignore this error */ sl@0: dbus_error_free (&derr); sl@0: } sl@0: else if (manager->owner_names != NULL) sl@0: { sl@0: dbus_g_proxy_manager_replace_name_owner (manager, name, prev_owner, new_owner); sl@0: } sl@0: } sl@0: sl@0: sender = dbus_message_get_sender (message); sl@0: sl@0: /* dbus spec requires these, libdbus validates */ sl@0: g_assert (dbus_message_get_path (message) != NULL); sl@0: g_assert (dbus_message_get_interface (message) != NULL); sl@0: g_assert (dbus_message_get_member (message) != NULL); sl@0: sl@0: tri = tristring_from_message (message); sl@0: sl@0: if (manager->proxy_lists) sl@0: { sl@0: DBusGProxyList *owner_list; sl@0: owner_list = g_hash_table_lookup (manager->proxy_lists, tri); sl@0: if (owner_list) sl@0: full_list = g_slist_copy (owner_list->proxies); sl@0: else sl@0: full_list = NULL; sl@0: } sl@0: else sl@0: full_list = NULL; sl@0: sl@0: g_free (tri); sl@0: sl@0: if (manager->owner_names && sender) sl@0: { sl@0: owned_names = g_hash_table_lookup (manager->owner_names, sender); sl@0: for (tmp = owned_names; tmp; tmp = tmp->next) sl@0: { sl@0: DBusGProxyList *owner_list; sl@0: DBusGProxyNameOwnerInfo *nameinfo; sl@0: sl@0: nameinfo = tmp->data; sl@0: g_assert (nameinfo->refcount > 0); sl@0: tri = tristring_alloc_from_strings (0, nameinfo->name, sl@0: dbus_message_get_path (message), sl@0: dbus_message_get_interface (message)); sl@0: sl@0: owner_list = g_hash_table_lookup (manager->proxy_lists, tri); sl@0: if (owner_list != NULL) sl@0: full_list = g_slist_concat (full_list, g_slist_copy (owner_list->proxies)); sl@0: g_free (tri); sl@0: } sl@0: } sl@0: sl@0: #if 0 sl@0: g_print ("proxy got %s,%s,%s = list %p\n", sl@0: tri, sl@0: tri + strlen (tri) + 1, sl@0: tri + strlen (tri) + 1 + strlen (tri + strlen (tri) + 1) + 1, sl@0: list); sl@0: #endif sl@0: sl@0: /* Emit the signal */ sl@0: sl@0: g_slist_foreach (full_list, (GFunc) g_object_ref, NULL); sl@0: sl@0: for (tmp = full_list; tmp; tmp = tmp->next) sl@0: { sl@0: DBusGProxy *proxy; sl@0: sl@0: proxy = DBUS_G_PROXY (tmp->data); sl@0: sl@0: UNLOCK_MANAGER (manager); sl@0: dbus_g_proxy_emit_remote_signal (proxy, message); sl@0: g_object_unref (G_OBJECT (proxy)); sl@0: LOCK_MANAGER (manager); sl@0: } sl@0: g_slist_free (full_list); sl@0: } sl@0: sl@0: UNLOCK_MANAGER (manager); sl@0: dbus_g_proxy_manager_unref (manager); sl@0: sl@0: /* "Handling" signals doesn't make sense, they are for everyone sl@0: * who cares sl@0: */ sl@0: return DBUS_HANDLER_RESULT_NOT_YET_HANDLED; sl@0: } sl@0: sl@0: sl@0: sl@0: /* ---------- DBusGProxy -------------- */ sl@0: #define DBUS_G_PROXY_DESTROYED(proxy) (DBUS_G_PROXY_GET_PRIVATE(proxy)->manager == NULL) sl@0: sl@0: static void sl@0: marshal_dbus_message_to_g_marshaller (GClosure *closure, sl@0: GValue *return_value, sl@0: guint n_param_values, sl@0: const GValue *param_values, sl@0: gpointer invocation_hint, sl@0: gpointer marshal_data); sl@0: enum sl@0: { sl@0: PROP_0, sl@0: PROP_NAME, sl@0: PROP_PATH, sl@0: PROP_INTERFACE, sl@0: PROP_CONNECTION sl@0: }; sl@0: sl@0: enum sl@0: { sl@0: DESTROY, sl@0: RECEIVED, sl@0: LAST_SIGNAL sl@0: }; sl@0: sl@0: #if EMULATOR sl@0: GET_STATIC_VAR_FROM_TLS(parent_class,dbus_gproxy,void* ) sl@0: #define parent_class (*GET_DBUS_WSD_VAR_NAME(parent_class,dbus_gproxy,s)()) sl@0: sl@0: GET_STATIC_ARRAY_FROM_TLS(signals,dbus_gproxy,guint ) sl@0: #define signals (GET_DBUS_WSD_VAR_NAME(signals,dbus_gproxy,s)()) sl@0: sl@0: #else sl@0: static void *parent_class; sl@0: static guint signals[LAST_SIGNAL] = { 0 }; sl@0: sl@0: #endif sl@0: sl@0: sl@0: static void sl@0: dbus_g_proxy_init (DBusGProxy *proxy) sl@0: { sl@0: DBusGProxyPrivate *priv = DBUS_G_PROXY_GET_PRIVATE(proxy); sl@0: sl@0: g_datalist_init (&priv->signal_signatures); sl@0: priv->pending_calls = g_hash_table_new_full (NULL, NULL, NULL, sl@0: (GDestroyNotify) dbus_pending_call_unref); sl@0: priv->name_call = 0; sl@0: priv->associated = FALSE; sl@0: } sl@0: sl@0: static GObject * sl@0: dbus_g_proxy_constructor (GType type, sl@0: guint n_construct_properties, sl@0: GObjectConstructParam *construct_properties) sl@0: { sl@0: DBusGProxy *proxy; sl@0: DBusGProxyClass *klass; sl@0: GObjectClass *parent_class; sl@0: DBusGProxyPrivate *priv; sl@0: sl@0: klass = DBUS_G_PROXY_CLASS (g_type_class_peek (DBUS_TYPE_G_PROXY)); sl@0: sl@0: parent_class = G_OBJECT_CLASS (g_type_class_peek_parent (klass)); sl@0: sl@0: proxy = DBUS_G_PROXY (parent_class->constructor (type, n_construct_properties, sl@0: construct_properties)); sl@0: sl@0: priv = DBUS_G_PROXY_GET_PRIVATE (proxy); sl@0: sl@0: /* if these assertions fail, a deriving class has not set our required sl@0: * parameters - our own public constructors do return_if_fail checks sl@0: * on these parameters being provided. unfortunately we can't assert sl@0: * for manager because it's allowed to be NULL when tha mangager is sl@0: * setting up a bus proxy for its own calls */ sl@0: g_assert (priv->path != NULL); sl@0: g_assert (priv->interface != NULL); sl@0: sl@0: if (priv->manager != NULL) sl@0: { sl@0: dbus_g_proxy_manager_register (priv->manager, proxy); sl@0: } sl@0: sl@0: return G_OBJECT (proxy); sl@0: } sl@0: sl@0: static void sl@0: dbus_g_proxy_class_init (DBusGProxyClass *klass) sl@0: { sl@0: GObjectClass *object_class = G_OBJECT_CLASS (klass); sl@0: sl@0: parent_class = g_type_class_peek_parent (klass); sl@0: sl@0: g_type_class_add_private (klass, sizeof (DBusGProxyPrivate)); sl@0: sl@0: object_class->set_property = dbus_g_proxy_set_property; sl@0: object_class->get_property = dbus_g_proxy_get_property; sl@0: sl@0: g_object_class_install_property (object_class, sl@0: PROP_NAME, sl@0: g_param_spec_string ("name", sl@0: "name", sl@0: "name", sl@0: NULL, sl@0: // #ifdef WINSCW sl@0: G_PARAM_READWRITE | G_PARAM_CONSTRUCT_ONLY)); sl@0: // #else sl@0: // (GParamFlags)11)); // ARM complaining about enums mixing sl@0: // #endif sl@0: sl@0: g_object_class_install_property (object_class, sl@0: PROP_PATH, sl@0: g_param_spec_string ("path", sl@0: "path", sl@0: "path", sl@0: NULL, sl@0: // #ifdef WINSCW sl@0: G_PARAM_READWRITE | G_PARAM_CONSTRUCT_ONLY)); sl@0: // #else sl@0: // (GParamFlags)11)); // ARM complaining about enums mixing sl@0: // #endif sl@0: sl@0: g_object_class_install_property (object_class, sl@0: PROP_INTERFACE, sl@0: g_param_spec_string ("interface", sl@0: "interface", sl@0: "interface", sl@0: NULL, sl@0: // #ifdef WINSCW sl@0: G_PARAM_READWRITE | G_PARAM_CONSTRUCT_ONLY)); sl@0: // #else sl@0: // (GParamFlags)11)); // ARM complaining about enums mixing sl@0: // #endif sl@0: sl@0: g_object_class_install_property (object_class, sl@0: PROP_CONNECTION, sl@0: g_param_spec_boxed ("connection", sl@0: "connection", sl@0: "connection", sl@0: DBUS_TYPE_G_CONNECTION, sl@0: // #ifdef WINSCW sl@0: G_PARAM_READWRITE | G_PARAM_CONSTRUCT_ONLY)); sl@0: // #else sl@0: // (GParamFlags)11)); // ARM complaining about enums mixing sl@0: // #endif sl@0: sl@0: object_class->finalize = dbus_g_proxy_finalize; sl@0: object_class->dispose = dbus_g_proxy_dispose; sl@0: object_class->constructor = dbus_g_proxy_constructor; sl@0: sl@0: signals[DESTROY] = sl@0: g_signal_new ("destroy", sl@0: G_OBJECT_CLASS_TYPE (object_class), sl@0: G_SIGNAL_RUN_CLEANUP | G_SIGNAL_NO_RECURSE | G_SIGNAL_NO_HOOKS, sl@0: 0, sl@0: NULL, NULL, sl@0: g_cclosure_marshal_VOID__VOID, sl@0: G_TYPE_NONE, 0); sl@0: sl@0: signals[RECEIVED] = sl@0: g_signal_new ("received", sl@0: G_OBJECT_CLASS_TYPE (object_class), sl@0: G_SIGNAL_RUN_LAST | G_SIGNAL_DETAILED, sl@0: 0, sl@0: NULL, NULL, sl@0: marshal_dbus_message_to_g_marshaller, sl@0: G_TYPE_NONE, 2, DBUS_TYPE_MESSAGE, G_TYPE_POINTER); sl@0: } sl@0: sl@0: static void sl@0: cancel_pending_call (gpointer key, gpointer val, gpointer data) sl@0: { sl@0: DBusGProxyCall *call = key; sl@0: DBusGProxy *proxy = data; sl@0: sl@0: dbus_g_proxy_cancel_call (proxy, call); sl@0: } sl@0: sl@0: static void sl@0: dbus_g_proxy_dispose (GObject *object) sl@0: { sl@0: DBusGProxy *proxy = DBUS_G_PROXY (object); sl@0: DBusGProxyPrivate *priv = DBUS_G_PROXY_GET_PRIVATE(proxy); sl@0: sl@0: if (priv->pending_calls == NULL) sl@0: { sl@0: return; sl@0: } sl@0: sl@0: /* Cancel outgoing pending calls */ sl@0: g_hash_table_foreach (priv->pending_calls, cancel_pending_call, proxy); sl@0: g_hash_table_destroy (priv->pending_calls); sl@0: priv->pending_calls = NULL; sl@0: sl@0: if (priv->manager && proxy != priv->manager->bus_proxy) sl@0: { sl@0: dbus_g_proxy_manager_unregister (priv->manager, proxy); sl@0: dbus_g_proxy_manager_unref (priv->manager); sl@0: } sl@0: priv->manager = NULL; sl@0: sl@0: g_datalist_clear (&priv->signal_signatures); sl@0: sl@0: g_signal_emit (object, signals[DESTROY], 0); sl@0: sl@0: G_OBJECT_CLASS (parent_class)->dispose (object); sl@0: } sl@0: sl@0: static void sl@0: dbus_g_proxy_finalize (GObject *object) sl@0: { sl@0: DBusGProxy *proxy = DBUS_G_PROXY (object); sl@0: DBusGProxyPrivate *priv = DBUS_G_PROXY_GET_PRIVATE(proxy); sl@0: sl@0: g_return_if_fail (DBUS_G_PROXY_DESTROYED (proxy)); sl@0: sl@0: g_free (priv->name); sl@0: g_free (priv->path); sl@0: g_free (priv->interface); sl@0: sl@0: G_OBJECT_CLASS (parent_class)->finalize (object); sl@0: } sl@0: sl@0: static void sl@0: dbus_g_proxy_destroy (DBusGProxy *proxy) sl@0: { sl@0: /* FIXME do we need the GTK_IN_DESTRUCTION style flag sl@0: * from GtkObject? sl@0: */ sl@0: g_object_run_dispose (G_OBJECT (proxy)); sl@0: } sl@0: sl@0: static void sl@0: dbus_g_proxy_set_property (GObject *object, sl@0: guint prop_id, sl@0: const GValue *value, sl@0: GParamSpec *pspec) sl@0: { sl@0: DBusGProxy *proxy = DBUS_G_PROXY (object); sl@0: DBusGProxyPrivate *priv = DBUS_G_PROXY_GET_PRIVATE(proxy); sl@0: DBusGConnection *connection; sl@0: sl@0: switch (prop_id) sl@0: { sl@0: case PROP_NAME: sl@0: priv->name = g_strdup (g_value_get_string (value)); sl@0: if (priv->name) sl@0: priv->for_owner = (priv->name[0] == ':'); sl@0: else sl@0: priv->for_owner = TRUE; sl@0: break; sl@0: case PROP_PATH: sl@0: priv->path = g_strdup (g_value_get_string (value)); sl@0: break; sl@0: case PROP_INTERFACE: sl@0: priv->interface = g_strdup (g_value_get_string (value)); sl@0: break; sl@0: case PROP_CONNECTION: sl@0: connection = g_value_get_boxed (value); sl@0: if (connection != NULL) sl@0: { sl@0: priv->manager = dbus_g_proxy_manager_get (DBUS_CONNECTION_FROM_G_CONNECTION (connection)); sl@0: } sl@0: break; sl@0: default: sl@0: G_OBJECT_WARN_INVALID_PROPERTY_ID (object, prop_id, pspec); sl@0: break; sl@0: } sl@0: } sl@0: sl@0: static void sl@0: dbus_g_proxy_get_property (GObject *object, sl@0: guint prop_id, sl@0: GValue *value, sl@0: GParamSpec *pspec) sl@0: { sl@0: DBusGProxy *proxy = DBUS_G_PROXY (object); sl@0: DBusGProxyPrivate *priv = DBUS_G_PROXY_GET_PRIVATE(proxy); sl@0: sl@0: switch (prop_id) sl@0: { sl@0: case PROP_NAME: sl@0: g_value_set_string (value, priv->name); sl@0: break; sl@0: case PROP_PATH: sl@0: g_value_set_string (value, priv->path); sl@0: break; sl@0: case PROP_INTERFACE: sl@0: g_value_set_string (value, priv->interface); sl@0: break; sl@0: case PROP_CONNECTION: sl@0: g_value_set_boxed (value, DBUS_G_CONNECTION_FROM_CONNECTION(priv->manager->connection)); sl@0: break; sl@0: default: sl@0: G_OBJECT_WARN_INVALID_PROPERTY_ID (object, prop_id, pspec); sl@0: break; sl@0: } sl@0: } sl@0: sl@0: /* this is to avoid people using g_signal_connect() directly, sl@0: * to avoid confusion with local signal names, and because sl@0: * of the horribly broken current setup (signals are added sl@0: * globally to all proxies) sl@0: */ sl@0: static char* sl@0: create_signal_name (const char *interface, sl@0: const char *signal) sl@0: { sl@0: GString *str; sl@0: char *p; sl@0: sl@0: str = g_string_new (interface); sl@0: sl@0: g_string_append (str, "-"); sl@0: sl@0: g_string_append (str, signal); sl@0: sl@0: /* GLib will silently barf on '.' in signal names */ sl@0: p = str->str; sl@0: while (*p) sl@0: { sl@0: if (*p == '.') sl@0: *p = '-'; sl@0: ++p; sl@0: } sl@0: sl@0: return g_string_free (str, FALSE); sl@0: } sl@0: sl@0: static void sl@0: marshal_dbus_message_to_g_marshaller (GClosure *closure, sl@0: GValue *return_value, sl@0: guint n_param_values, sl@0: const GValue *param_values, sl@0: gpointer invocation_hint, sl@0: gpointer marshal_data) sl@0: { sl@0: /* Incoming here we have three params, the instance (Proxy), the sl@0: * DBusMessage, the signature. We want to convert that to an sl@0: * expanded GValue array, then call an appropriate normal GLib sl@0: * marshaller. sl@0: */ sl@0: #define MAX_SIGNATURE_ARGS 20 sl@0: GValueArray *value_array; sl@0: GSignalCMarshaller c_marshaller; sl@0: DBusGProxy *proxy; sl@0: DBusMessage *message; sl@0: GArray *gsignature; sl@0: const GType *types; sl@0: DBusGProxyPrivate *priv; sl@0: sl@0: g_assert (n_param_values == 3); sl@0: sl@0: proxy = g_value_get_object (¶m_values[0]); sl@0: message = g_value_get_boxed (¶m_values[1]); sl@0: gsignature = g_value_get_pointer (¶m_values[2]); sl@0: sl@0: g_return_if_fail (DBUS_IS_G_PROXY (proxy)); sl@0: g_return_if_fail (message != NULL); sl@0: g_return_if_fail (gsignature != NULL); sl@0: sl@0: priv = DBUS_G_PROXY_GET_PRIVATE(proxy); sl@0: sl@0: c_marshaller = _dbus_gobject_lookup_marshaller (G_TYPE_NONE, gsignature->len, sl@0: (GType*) gsignature->data); sl@0: sl@0: g_return_if_fail (c_marshaller != NULL); sl@0: sl@0: { sl@0: DBusGValueMarshalCtx context; sl@0: context.gconnection = DBUS_G_CONNECTION_FROM_CONNECTION (priv->manager->connection); sl@0: context.proxy = proxy; sl@0: sl@0: types = (const GType*) gsignature->data; sl@0: value_array = _dbus_gvalue_demarshal_message (&context, message, sl@0: gsignature->len, types, NULL); sl@0: } sl@0: sl@0: if (value_array == NULL) sl@0: return; sl@0: sl@0: g_value_array_prepend (value_array, NULL); sl@0: g_value_init (g_value_array_get_nth (value_array, 0), G_TYPE_FROM_INSTANCE (proxy)); sl@0: g_value_set_instance (g_value_array_get_nth (value_array, 0), proxy); sl@0: sl@0: (* c_marshaller) (closure, return_value, value_array->n_values, sl@0: value_array->values, invocation_hint, marshal_data); sl@0: sl@0: g_value_array_free (value_array); sl@0: } sl@0: sl@0: static void sl@0: dbus_g_proxy_emit_remote_signal (DBusGProxy *proxy, sl@0: DBusMessage *message) sl@0: { sl@0: const char *interface; sl@0: const char *signal; sl@0: char *name; sl@0: GQuark q; sl@0: DBusGProxyPrivate *priv = DBUS_G_PROXY_GET_PRIVATE(proxy); sl@0: GArray *msg_gsignature = NULL; sl@0: sl@0: g_return_if_fail (!DBUS_G_PROXY_DESTROYED (proxy)); sl@0: sl@0: interface = dbus_message_get_interface (message); sl@0: signal = dbus_message_get_member (message); sl@0: sl@0: g_assert (interface != NULL); sl@0: g_assert (signal != NULL); sl@0: sl@0: name = create_signal_name (interface, signal); sl@0: sl@0: /* If the quark isn't preexisting, there's no way there sl@0: * are any handlers connected. We don't want to create sl@0: * extra quarks for every possible signal. sl@0: */ sl@0: q = g_quark_try_string (name); sl@0: sl@0: if (q != 0) sl@0: { sl@0: GArray *gsignature; sl@0: guint i; sl@0: sl@0: gsignature = g_datalist_id_get_data (&priv->signal_signatures, q); sl@0: if (gsignature == NULL) sl@0: goto out; sl@0: sl@0: msg_gsignature = _dbus_gtypes_from_arg_signature (dbus_message_get_signature (message), sl@0: TRUE); sl@0: for (i = 0; i < gsignature->len; i++) sl@0: { sl@0: if (msg_gsignature->len == i sl@0: || g_array_index (gsignature, GType, i) != g_array_index (msg_gsignature, GType, i)) sl@0: goto mismatch; sl@0: } sl@0: if (msg_gsignature->len != i) sl@0: goto mismatch; sl@0: sl@0: g_signal_emit (proxy, sl@0: signals[RECEIVED], sl@0: q, sl@0: message, sl@0: msg_gsignature); sl@0: } sl@0: sl@0: out: sl@0: g_free (name); sl@0: if (msg_gsignature) sl@0: g_array_free (msg_gsignature, TRUE); sl@0: return; sl@0: mismatch: sl@0: #if 0 sl@0: /* Don't spew on remote errors */ sl@0: g_warning ("Unexpected message signature '%s' for signal '%s'\n", sl@0: dbus_message_get_signature (message), sl@0: name); sl@0: #endif sl@0: goto out; sl@0: } sl@0: sl@0: typedef struct sl@0: { sl@0: DBusGProxy *proxy; sl@0: guint call_id; sl@0: DBusGProxyCallNotify func; sl@0: void *data; sl@0: GDestroyNotify free_data_func; sl@0: } GPendingNotifyClosure; sl@0: sl@0: static void sl@0: d_pending_call_notify (DBusPendingCall *dcall, sl@0: void *data) sl@0: { sl@0: GPendingNotifyClosure *closure = data; sl@0: sl@0: (* closure->func) (closure->proxy, DBUS_G_PROXY_ID_TO_CALL (closure->call_id), closure->data); sl@0: } sl@0: sl@0: static void sl@0: d_pending_call_free (void *data) sl@0: { sl@0: GPendingNotifyClosure *closure = data; sl@0: sl@0: if (closure->free_data_func) sl@0: (* closure->free_data_func) (closure->data); sl@0: sl@0: g_free (closure); sl@0: } sl@0: sl@0: #define DBUS_G_VALUE_ARRAY_COLLECT_ALL(VALARRAY, FIRST_ARG_TYPE, ARGS) \ sl@0: do { \ sl@0: GType valtype; \ sl@0: int i = 0; \ sl@0: VALARRAY = g_value_array_new (6); \ sl@0: valtype = FIRST_ARG_TYPE; \ sl@0: while (valtype != G_TYPE_INVALID) \ sl@0: { \ sl@0: const char *collect_err; \ sl@0: GValue *val; \ sl@0: g_value_array_append (VALARRAY, NULL); \ sl@0: val = g_value_array_get_nth (VALARRAY, i); \ sl@0: g_value_init (val, valtype); \ sl@0: collect_err = NULL; \ sl@0: G_VALUE_COLLECT (val, ARGS, G_VALUE_NOCOPY_CONTENTS, &collect_err); \ sl@0: valtype = va_arg (ARGS, GType); \ sl@0: i++; \ sl@0: } \ sl@0: } while (0) sl@0: sl@0: DBusGProxyCall * sl@0: manager_begin_bus_call (DBusGProxyManager *manager, sl@0: const char *method, sl@0: DBusGProxyCallNotify notify, sl@0: gpointer user_data, sl@0: GDestroyNotify destroy, sl@0: GType first_arg_type, sl@0: ...) sl@0: { sl@0: DBusGProxyCall *call; sl@0: DBusGProxyPrivate *priv; sl@0: va_list args; sl@0: GValueArray *arg_values; sl@0: sl@0: va_start (args, first_arg_type); sl@0: sl@0: if (!manager->bus_proxy) sl@0: { sl@0: manager->bus_proxy = g_object_new (DBUS_TYPE_G_PROXY, sl@0: "name", DBUS_SERVICE_DBUS, sl@0: "path", DBUS_PATH_DBUS, sl@0: "interface", DBUS_INTERFACE_DBUS, sl@0: NULL); sl@0: priv = DBUS_G_PROXY_GET_PRIVATE(manager->bus_proxy); sl@0: priv->manager = manager; sl@0: } sl@0: sl@0: DBUS_G_VALUE_ARRAY_COLLECT_ALL (arg_values, first_arg_type, args); sl@0: sl@0: call = DBUS_G_PROXY_ID_TO_CALL (dbus_g_proxy_begin_call_internal (manager->bus_proxy, method, notify, user_data, destroy, arg_values,-1)); sl@0: sl@0: g_value_array_free (arg_values); sl@0: sl@0: va_end (args); sl@0: sl@0: return call; sl@0: } sl@0: sl@0: /** @} End of DBusGLibInternals */ sl@0: sl@0: /** @addtogroup DBusGLib sl@0: * @{ sl@0: */ sl@0: sl@0: /** sl@0: * SECTION:dbus-gproxy sl@0: * @short_description: DBus Proxy sl@0: * @see_also: #DBusProxy sl@0: * @stability: Stable sl@0: * sl@0: * A #DBusGProxy is a boxed type abstracting a #DBusProxy. sl@0: */ sl@0: sl@0: /** sl@0: * dbus_g_proxy_get_type: sl@0: * Standard GObject get_type() function for DBusGProxy. sl@0: * sl@0: * Returns: type ID for DBusGProxy class sl@0: */ sl@0: #if EMULATOR sl@0: GET_STATIC_VAR_FROM_TLS(object_type,dbus_gproxy,GType) sl@0: #define object_type (*GET_DBUS_WSD_VAR_NAME(object_type,dbus_gproxy,s)()) sl@0: #endif sl@0: #ifdef __SYMBIAN32__ sl@0: EXPORT_C sl@0: #endif sl@0: GType sl@0: dbus_g_proxy_get_type (void) sl@0: { sl@0: #ifndef EMULATOR sl@0: static GType object_type = 0; sl@0: #endif sl@0: sl@0: sl@0: if (!object_type) sl@0: { sl@0: static const GTypeInfo object_info = sl@0: { sl@0: sizeof (DBusGProxyClass), sl@0: (GBaseInitFunc) NULL, sl@0: (GBaseFinalizeFunc) NULL, sl@0: (GClassInitFunc) dbus_g_proxy_class_init, sl@0: NULL, /* class_finalize */ sl@0: NULL, /* class_data */ sl@0: sizeof (DBusGProxy), sl@0: 0, /* n_preallocs */ sl@0: (GInstanceInitFunc) dbus_g_proxy_init, sl@0: }; sl@0: sl@0: object_type = g_type_register_static (G_TYPE_OBJECT, sl@0: "DBusGProxy", sl@0: &object_info, 0); sl@0: } sl@0: sl@0: return object_type; sl@0: } sl@0: sl@0: static DBusGProxy* sl@0: dbus_g_proxy_new (DBusGConnection *connection, sl@0: const char *name, sl@0: const char *path_name, sl@0: const char *interface_name) sl@0: { sl@0: DBusGProxy *proxy; sl@0: sl@0: g_assert (connection != NULL); sl@0: sl@0: proxy = g_object_new (DBUS_TYPE_G_PROXY, sl@0: "name", name, sl@0: "path", path_name, sl@0: "interface", interface_name, sl@0: "connection", connection, NULL); sl@0: sl@0: return proxy; sl@0: } sl@0: sl@0: /** sl@0: * dbus_g_proxy_new_for_name: sl@0: * @connection: the connection to the remote bus sl@0: * @name: any name on the message bus sl@0: * @path_name: name of the object instance to call methods on sl@0: * @interface_name: name of the interface to call methods on sl@0: * sl@0: * Creates a new proxy for a remote interface exported by a connection sl@0: * on a message bus. Method calls and signal connections over this sl@0: * proxy will go to the name owner; the name's owner is expected to sl@0: * support the given interface name. THE NAME OWNER MAY CHANGE OVER sl@0: * TIME, for example between two different method calls, unless the sl@0: * name is a unique name. If you need a fixed owner, you need to sl@0: * request the current owner and bind a proxy to its unique name sl@0: * rather than to the generic name; see sl@0: * dbus_g_proxy_new_for_name_owner(). sl@0: * sl@0: * A name-associated proxy only makes sense with a message bus, not sl@0: * for app-to-app direct dbus connections. sl@0: * sl@0: * This proxy will only emit the "destroy" signal if the sl@0: * #DBusConnection is disconnected, the proxy has no remaining sl@0: * references, or the name is a unique name and its owner sl@0: * disappears. If a well-known name changes owner, the proxy will sl@0: * still be alive. sl@0: * sl@0: * Returns: new proxy object sl@0: */ sl@0: #ifdef __SYMBIAN32__ sl@0: EXPORT_C sl@0: #endif sl@0: DBusGProxy* sl@0: dbus_g_proxy_new_for_name (DBusGConnection *connection, sl@0: const char *name, sl@0: const char *path_name, sl@0: const char *interface_name) sl@0: { sl@0: g_return_val_if_fail (connection != NULL, NULL); sl@0: g_return_val_if_fail (name != NULL, NULL); sl@0: g_return_val_if_fail (path_name != NULL, NULL); sl@0: g_return_val_if_fail (interface_name != NULL, NULL); sl@0: sl@0: return dbus_g_proxy_new (connection, name, sl@0: path_name, interface_name); sl@0: } sl@0: sl@0: /** sl@0: * dbus_g_proxy_new_for_name_owner: sl@0: * @connection: the connection to the remote bus sl@0: * @name: any name on the message bus sl@0: * @path_name: name of the object inside the service to call methods on sl@0: * @interface_name: name of the interface to call methods on sl@0: * @error: return location for an error sl@0: * sl@0: * Similar to dbus_g_proxy_new_for_name(), but makes a round-trip sl@0: * request to the message bus to get the current name owner, then sl@0: * binds the proxy to the unique name of the current owner, rather sl@0: * than to the well-known name. As a result, the name owner will sl@0: * not change over time, and the proxy will emit the "destroy" signal sl@0: * when the owner disappears from the message bus. sl@0: * sl@0: * An example of the difference between dbus_g_proxy_new_for_name() sl@0: * and dbus_g_proxy_new_for_name_owner(): if you provide the well-known name sl@0: * "org.freedesktop.Database" dbus_g_proxy_new_for_name() remains bound sl@0: * to that name as it changes owner. dbus_g_proxy_new_for_name_owner() sl@0: * will fail if the name has no owner. If the name has an owner, sl@0: * dbus_g_proxy_new_for_name_owner() will bind to the unique name sl@0: * of that owner rather than the generic name. sl@0: * sl@0: * Returns: new proxy object, or #NULL on error sl@0: */ sl@0: #ifdef __SYMBIAN32__ sl@0: EXPORT_C sl@0: #endif sl@0: DBusGProxy* sl@0: dbus_g_proxy_new_for_name_owner (DBusGConnection *connection, sl@0: const char *name, sl@0: const char *path_name, sl@0: const char *interface_name, sl@0: GError **error) sl@0: { sl@0: DBusGProxy *proxy; sl@0: char *unique_name; sl@0: sl@0: g_return_val_if_fail (connection != NULL, NULL); sl@0: g_return_val_if_fail (name != NULL, NULL); sl@0: g_return_val_if_fail (path_name != NULL, NULL); sl@0: g_return_val_if_fail (interface_name != NULL, NULL); sl@0: sl@0: if (!(unique_name = get_name_owner (DBUS_CONNECTION_FROM_G_CONNECTION (connection), name, error))) sl@0: return NULL; sl@0: sl@0: proxy = dbus_g_proxy_new (connection, unique_name, sl@0: path_name, interface_name); sl@0: g_free (unique_name); sl@0: return proxy; sl@0: } sl@0: sl@0: /** sl@0: * dbus_g_proxy_new_from_proxy: sl@0: * @proxy: the proxy to use as a template sl@0: * @path: of the object inside the peer to call methods on sl@0: * @interface: name of the interface to call methods on sl@0: * sl@0: * Creates a proxy using an existing proxy as a template, substituting sl@0: * the specified interface and path. Either or both may be NULL. sl@0: * sl@0: * Returns: new proxy object sl@0: */ sl@0: #ifdef __SYMBIAN32__ sl@0: EXPORT_C sl@0: #endif sl@0: DBusGProxy* sl@0: dbus_g_proxy_new_from_proxy (DBusGProxy *proxy, sl@0: const char *interface, sl@0: const char *path) sl@0: { sl@0: DBusGProxyPrivate *priv; sl@0: sl@0: g_return_val_if_fail (proxy != NULL, NULL); sl@0: sl@0: priv = DBUS_G_PROXY_GET_PRIVATE(proxy); sl@0: sl@0: if (interface == NULL) sl@0: interface = priv->interface; sl@0: if (path == NULL) sl@0: path = priv->path; sl@0: sl@0: return dbus_g_proxy_new (DBUS_G_CONNECTION_FROM_CONNECTION (priv->manager->connection), sl@0: priv->name, sl@0: path, interface); sl@0: } sl@0: sl@0: /** sl@0: * dbus_g_proxy_new_for_peer: sl@0: * @connection: the connection to the peer sl@0: * @path_name: name of the object inside the peer to call methods on sl@0: * @interface_name: name of the interface to call methods on sl@0: * sl@0: * Creates a proxy for an object in peer application (one sl@0: * we're directly connected to). That is, this function is sl@0: * intended for use when there's no message bus involved, sl@0: * we're doing a simple 1-to-1 communication between two sl@0: * applications. sl@0: * sl@0: * Returns: new proxy object sl@0: */ sl@0: #ifdef __SYMBIAN32__ sl@0: EXPORT_C sl@0: #endif sl@0: DBusGProxy* sl@0: dbus_g_proxy_new_for_peer (DBusGConnection *connection, sl@0: const char *path_name, sl@0: const char *interface_name) sl@0: { sl@0: DBusGProxy *proxy; sl@0: sl@0: g_return_val_if_fail (connection != NULL, NULL); sl@0: g_return_val_if_fail (path_name != NULL, NULL); sl@0: g_return_val_if_fail (interface_name != NULL, NULL); sl@0: sl@0: proxy = dbus_g_proxy_new (connection, NULL, sl@0: path_name, interface_name); sl@0: sl@0: return proxy; sl@0: } sl@0: sl@0: /** sl@0: * dbus_g_proxy_get_bus_name: sl@0: * @proxy: the proxy sl@0: * sl@0: * Gets the bus name a proxy is bound to (may be #NULL in some cases). sl@0: * If you created the proxy with dbus_g_proxy_new_for_name(), then sl@0: * the name you passed to that will be returned. sl@0: * If you created it with dbus_g_proxy_new_for_name_owner(), then the sl@0: * unique connection name will be returned. If you created it sl@0: * with dbus_g_proxy_new_for_peer() then #NULL will be returned. sl@0: * sl@0: * Returns: the bus name the proxy sends messages to sl@0: */ sl@0: #ifdef __SYMBIAN32__ sl@0: EXPORT_C sl@0: #endif sl@0: const char* sl@0: dbus_g_proxy_get_bus_name (DBusGProxy *proxy) sl@0: { sl@0: DBusGProxyPrivate *priv; sl@0: sl@0: g_return_val_if_fail (DBUS_IS_G_PROXY (proxy), NULL); sl@0: g_return_val_if_fail (!DBUS_G_PROXY_DESTROYED (proxy), NULL); sl@0: sl@0: priv = DBUS_G_PROXY_GET_PRIVATE(proxy); sl@0: sl@0: return priv->name; sl@0: } sl@0: sl@0: /** sl@0: * dbus_g_proxy_get_interface: sl@0: * @proxy: the proxy sl@0: * sl@0: * Gets the object interface proxy is bound to (may be #NULL in some cases). sl@0: * sl@0: * Returns: an object interface sl@0: */ sl@0: #ifdef __SYMBIAN32__ sl@0: EXPORT_C sl@0: #endif sl@0: const char* sl@0: dbus_g_proxy_get_interface (DBusGProxy *proxy) sl@0: { sl@0: DBusGProxyPrivate *priv; sl@0: sl@0: g_return_val_if_fail (DBUS_IS_G_PROXY (proxy), NULL); sl@0: g_return_val_if_fail (!DBUS_G_PROXY_DESTROYED (proxy), NULL); sl@0: sl@0: priv = DBUS_G_PROXY_GET_PRIVATE(proxy); sl@0: sl@0: return priv->interface; sl@0: } sl@0: sl@0: /** sl@0: * dbus_g_proxy_set_interface: sl@0: * @proxy: the proxy sl@0: * @interface_name: an object interface sl@0: * sl@0: * Sets the object interface proxy is bound to sl@0: */ sl@0: #ifdef __SYMBIAN32__ sl@0: EXPORT_C sl@0: #endif sl@0: void sl@0: dbus_g_proxy_set_interface (DBusGProxy *proxy, sl@0: const char *interface_name) sl@0: { sl@0: DBusGProxyPrivate *priv = DBUS_G_PROXY_GET_PRIVATE(proxy); sl@0: /* FIXME - need to unregister when we switch interface for now sl@0: * later should support idea of unset interface sl@0: */ sl@0: dbus_g_proxy_manager_unregister (priv->manager, proxy); sl@0: g_free (priv->interface); sl@0: priv->interface = g_strdup (interface_name); sl@0: dbus_g_proxy_manager_register (priv->manager, proxy); sl@0: } sl@0: sl@0: /** sl@0: * dbus_g_proxy_get_path: sl@0: * Gets the path this proxy is bound to sl@0: * @proxy: the proxy sl@0: * sl@0: * Returns: an object path sl@0: */ sl@0: #ifdef __SYMBIAN32__ sl@0: EXPORT_C sl@0: #endif sl@0: const char* sl@0: dbus_g_proxy_get_path (DBusGProxy *proxy) sl@0: { sl@0: DBusGProxyPrivate *priv; sl@0: sl@0: g_return_val_if_fail (DBUS_IS_G_PROXY (proxy), NULL); sl@0: g_return_val_if_fail (!DBUS_G_PROXY_DESTROYED (proxy), NULL); sl@0: sl@0: priv = DBUS_G_PROXY_GET_PRIVATE(proxy); sl@0: sl@0: return priv->path; sl@0: } sl@0: sl@0: static DBusMessage * sl@0: dbus_g_proxy_marshal_args_to_message (DBusGProxy *proxy, sl@0: const char *method, sl@0: GValueArray *args) sl@0: { sl@0: DBusMessage *message; sl@0: DBusMessageIter msgiter; sl@0: guint i; sl@0: DBusGProxyPrivate *priv = DBUS_G_PROXY_GET_PRIVATE(proxy); sl@0: sl@0: message = dbus_message_new_method_call (priv->name, sl@0: priv->path, sl@0: priv->interface, sl@0: method); sl@0: if (message == NULL) sl@0: goto oom; sl@0: sl@0: dbus_message_iter_init_append (message, &msgiter); sl@0: for (i = 0; i < args->n_values; i++) sl@0: { sl@0: GValue *gvalue; sl@0: sl@0: gvalue = g_value_array_get_nth (args, i); sl@0: sl@0: if (!_dbus_gvalue_marshal (&msgiter, gvalue)) sl@0: g_assert_not_reached (); sl@0: } sl@0: return message; sl@0: oom: sl@0: return NULL; sl@0: } sl@0: sl@0: static guint sl@0: dbus_g_proxy_begin_call_internal (DBusGProxy *proxy, sl@0: const char *method, sl@0: DBusGProxyCallNotify notify, sl@0: gpointer user_data, sl@0: GDestroyNotify destroy, sl@0: GValueArray *args, sl@0: int timeout) sl@0: { sl@0: DBusMessage *message; sl@0: DBusPendingCall *pending; sl@0: GPendingNotifyClosure *closure; sl@0: guint call_id; sl@0: DBusGProxyPrivate *priv = DBUS_G_PROXY_GET_PRIVATE(proxy); sl@0: sl@0: pending = NULL; sl@0: sl@0: message = dbus_g_proxy_marshal_args_to_message (proxy, method, args); sl@0: if (!message) sl@0: goto oom; sl@0: sl@0: if (!dbus_connection_send_with_reply (priv->manager->connection, sl@0: message, sl@0: &pending, sl@0: timeout)) sl@0: goto oom; sl@0: dbus_message_unref (message); sl@0: g_assert (pending != NULL); sl@0: sl@0: call_id = ++priv->call_id_counter; sl@0: sl@0: if (notify != NULL) sl@0: { sl@0: closure = g_new (GPendingNotifyClosure, 1); sl@0: closure->proxy = proxy; /* No need to ref as the lifecycle is tied to proxy */ sl@0: closure->call_id = call_id; sl@0: closure->func = notify; sl@0: closure->data = user_data; sl@0: closure->free_data_func = destroy; sl@0: dbus_pending_call_set_notify (pending, d_pending_call_notify, sl@0: closure, sl@0: d_pending_call_free); sl@0: } sl@0: sl@0: g_hash_table_insert (priv->pending_calls, GUINT_TO_POINTER (call_id), pending); sl@0: sl@0: return call_id; sl@0: oom: sl@0: g_error ("Out of memory"); sl@0: return 0; sl@0: } sl@0: sl@0: static gboolean sl@0: dbus_g_proxy_end_call_internal (DBusGProxy *proxy, sl@0: guint call_id, sl@0: GError **error, sl@0: GType first_arg_type, sl@0: va_list args) sl@0: { sl@0: DBusMessage *reply; sl@0: DBusMessageIter msgiter; sl@0: DBusError derror; sl@0: va_list args_unwind; sl@0: guint over; sl@0: int n_retvals_processed; sl@0: gboolean ret; sl@0: GType valtype; sl@0: DBusPendingCall *pending; sl@0: DBusGProxyPrivate *priv = DBUS_G_PROXY_GET_PRIVATE(proxy); sl@0: sl@0: reply = NULL; sl@0: ret = FALSE; sl@0: n_retvals_processed = 0; sl@0: over = 0; sl@0: sl@0: pending = g_hash_table_lookup (priv->pending_calls, GUINT_TO_POINTER (call_id)); sl@0: sl@0: dbus_pending_call_block (pending); sl@0: reply = dbus_pending_call_steal_reply (pending); sl@0: sl@0: g_assert (reply != NULL); sl@0: sl@0: dbus_error_init (&derror); sl@0: sl@0: switch (dbus_message_get_type (reply)) sl@0: { sl@0: case DBUS_MESSAGE_TYPE_METHOD_RETURN: sl@0: dbus_message_iter_init (reply, &msgiter); sl@0: valtype = first_arg_type; sl@0: while (valtype != G_TYPE_INVALID) sl@0: { sl@0: int arg_type; sl@0: gpointer return_storage; sl@0: GValue gvalue = { 0, }; sl@0: DBusGValueMarshalCtx context; sl@0: sl@0: context.gconnection = DBUS_G_CONNECTION_FROM_CONNECTION (priv->manager->connection); sl@0: context.proxy = proxy; sl@0: sl@0: arg_type = dbus_message_iter_get_arg_type (&msgiter); sl@0: if (arg_type == DBUS_TYPE_INVALID) sl@0: { sl@0: g_set_error (error, DBUS_GERROR, sl@0: DBUS_GERROR_INVALID_ARGS, sl@0: _("Too few arguments in reply")); sl@0: goto out; sl@0: } sl@0: sl@0: return_storage = va_arg (args, gpointer); sl@0: if (return_storage == NULL) sl@0: goto next; sl@0: sl@0: /* We handle variants specially; the caller is expected sl@0: * to have already allocated storage for them. sl@0: */ sl@0: if (arg_type == DBUS_TYPE_VARIANT sl@0: && g_type_is_a (valtype, G_TYPE_VALUE)) sl@0: { sl@0: if (!_dbus_gvalue_demarshal_variant (&context, &msgiter, (GValue*) return_storage, NULL)) sl@0: { sl@0: g_set_error (error, sl@0: DBUS_GERROR, sl@0: DBUS_GERROR_INVALID_ARGS, sl@0: _("Couldn't convert argument, expected \"%s\""), sl@0: g_type_name (valtype)); sl@0: goto out; sl@0: } sl@0: } sl@0: else sl@0: { sl@0: g_value_init (&gvalue, valtype); sl@0: sl@0: if (!_dbus_gvalue_demarshal (&context, &msgiter, &gvalue, error)) sl@0: goto out; sl@0: sl@0: /* Anything that can be demarshaled must be storable */ sl@0: if (!_dbus_gvalue_store (&gvalue, (gpointer*) return_storage)) sl@0: g_assert_not_reached (); sl@0: /* Ownership of the value passes to the client, don't unset */ sl@0: } sl@0: sl@0: next: sl@0: n_retvals_processed++; sl@0: dbus_message_iter_next (&msgiter); sl@0: valtype = va_arg (args, GType); sl@0: } sl@0: sl@0: while (dbus_message_iter_get_arg_type (&msgiter) != DBUS_TYPE_INVALID) sl@0: { sl@0: over++; sl@0: dbus_message_iter_next (&msgiter); sl@0: } sl@0: sl@0: if (over > 0) sl@0: { sl@0: g_set_error (error, DBUS_GERROR, sl@0: DBUS_GERROR_INVALID_ARGS, sl@0: _("Too many arguments in reply; expected %d, got %d"), sl@0: n_retvals_processed, over); sl@0: goto out; sl@0: } sl@0: break; sl@0: case DBUS_MESSAGE_TYPE_ERROR: sl@0: dbus_set_error_from_message (&derror, reply); sl@0: dbus_set_g_error (error, &derror); sl@0: dbus_error_free (&derror); sl@0: goto out; sl@0: break; sl@0: default: sl@0: dbus_set_error (&derror, DBUS_ERROR_FAILED, sl@0: "Reply was neither a method return nor an exception"); sl@0: dbus_set_g_error (error, &derror); sl@0: dbus_error_free (&derror); sl@0: goto out; sl@0: break; sl@0: } sl@0: sl@0: ret = TRUE; sl@0: out: sl@0: va_end (args); sl@0: sl@0: if (ret == FALSE) sl@0: { sl@0: int i; sl@0: for (i = 0; i < n_retvals_processed; i++) sl@0: { sl@0: gpointer retval; sl@0: sl@0: retval = va_arg (args_unwind, gpointer); sl@0: sl@0: g_free (retval); sl@0: } sl@0: } sl@0: va_end (args_unwind); sl@0: sl@0: g_hash_table_remove (priv->pending_calls, GUINT_TO_POINTER (call_id)); sl@0: sl@0: if (reply) sl@0: dbus_message_unref (reply); sl@0: return ret; sl@0: } sl@0: sl@0: /** sl@0: * dbus_g_proxy_begin_call: sl@0: * @proxy: a proxy for a remote interface sl@0: * @method: the name of the method to invoke sl@0: * @notify: callback to be invoked when method returns sl@0: * @user_data: user data passed to callback sl@0: * @destroy: function called to destroy user_data sl@0: * @first_arg_type: type of the first argument sl@0: * sl@0: * Asynchronously invokes a method on a remote interface. The method sl@0: * call will not be sent over the wire until the application returns sl@0: * to the main loop, or blocks in dbus_connection_flush() to write out sl@0: * pending data. The call will be completed after a timeout, or when sl@0: * a reply is received. When the call returns, the callback specified sl@0: * will be invoked; you can then collect the results of the call sl@0: * (which may be an error, or a reply), use dbus_g_proxy_end_call(). sl@0: * sl@0: * TODO this particular function shouldn't die on out of memory, sl@0: * since you should be able to do a call with large arguments. sl@0: * sl@0: * Returns: call identifier. sl@0: */ sl@0: #ifdef __SYMBIAN32__ sl@0: EXPORT_C sl@0: #endif sl@0: DBusGProxyCall * sl@0: dbus_g_proxy_begin_call (DBusGProxy *proxy, sl@0: const char *method, sl@0: DBusGProxyCallNotify notify, sl@0: gpointer user_data, sl@0: GDestroyNotify destroy, sl@0: GType first_arg_type, sl@0: ...) sl@0: { sl@0: guint call_id; sl@0: va_list args; sl@0: GValueArray *arg_values; sl@0: sl@0: g_return_val_if_fail (DBUS_IS_G_PROXY (proxy), NULL); sl@0: g_return_val_if_fail (!DBUS_G_PROXY_DESTROYED (proxy), NULL); sl@0: sl@0: va_start (args, first_arg_type); sl@0: sl@0: DBUS_G_VALUE_ARRAY_COLLECT_ALL (arg_values, first_arg_type, args); sl@0: sl@0: call_id = dbus_g_proxy_begin_call_internal (proxy, method, notify, user_data, destroy, arg_values,-1); sl@0: sl@0: g_value_array_free (arg_values); sl@0: sl@0: va_end (args); sl@0: sl@0: return DBUS_G_PROXY_ID_TO_CALL (call_id); sl@0: } sl@0: sl@0: /** sl@0: * dbus_g_proxy_begin_call_with_timeout: sl@0: * @proxy: a proxy for a remote interface sl@0: * @method: the name of the method to invoke sl@0: * @notify: callback to be invoked when method returns sl@0: * @user_data: user data passed to callback sl@0: * @destroy: function called to destroy user_data sl@0: * @timeout: specify the timeout in milliseconds sl@0: * @first_arg_type: type of the first argument sl@0: * sl@0: * Asynchronously invokes a method on a remote interface. The method sl@0: * call will not be sent over the wire until the application returns sl@0: * to the main loop, or blocks in dbus_connection_flush() to write out sl@0: * pending data. The call will be completed after a timeout, or when sl@0: * a reply is received. When the call returns, the callback specified sl@0: * will be invoked; you can then collect the results of the call sl@0: * (which may be an error, or a reply), use dbus_g_proxy_end_call(). sl@0: * sl@0: * TODO this particular function shouldn't die on out of memory, sl@0: * since you should be able to do a call with large arguments. sl@0: * sl@0: * Returns: call identifier. sl@0: */ sl@0: #ifdef __SYMBIAN32__ sl@0: EXPORT_C sl@0: #endif sl@0: DBusGProxyCall * sl@0: dbus_g_proxy_begin_call_with_timeout (DBusGProxy *proxy, sl@0: const char *method, sl@0: DBusGProxyCallNotify notify, sl@0: gpointer user_data, sl@0: GDestroyNotify destroy, sl@0: int timeout, sl@0: GType first_arg_type, sl@0: ...) sl@0: { sl@0: guint call_id; sl@0: va_list args; sl@0: GValueArray *arg_values; sl@0: sl@0: g_return_val_if_fail (DBUS_IS_G_PROXY (proxy), NULL); sl@0: g_return_val_if_fail (!DBUS_G_PROXY_DESTROYED (proxy), NULL); sl@0: sl@0: va_start (args, first_arg_type); sl@0: sl@0: DBUS_G_VALUE_ARRAY_COLLECT_ALL (arg_values, first_arg_type, args); sl@0: sl@0: call_id = dbus_g_proxy_begin_call_internal (proxy, method, notify, user_data, destroy, arg_values,timeout); sl@0: sl@0: g_value_array_free (arg_values); sl@0: sl@0: va_end (args); sl@0: sl@0: return DBUS_G_PROXY_ID_TO_CALL (call_id); sl@0: } sl@0: sl@0: /** sl@0: * dbus_g_proxy_end_call: sl@0: * @proxy: a proxy for a remote interface sl@0: * @call: the pending call ID from dbus_g_proxy_begin_call() sl@0: * @error: return location for an error sl@0: * @first_arg_type: type of first "out" argument sl@0: * sl@0: * Collects the results of a method call. The method call was normally sl@0: * initiated with dbus_g_proxy_end_call(). You may use this function sl@0: * outside of the callback given to dbus_g_proxy_begin_call; in that sl@0: * case this function will block if the results haven't yet been sl@0: * received. sl@0: * sl@0: * If the call results in an error, the error is set as normal for sl@0: * GError and the function returns #FALSE. sl@0: * sl@0: * Otherwise, the "out" parameters and return value of the sl@0: * method are stored in the provided varargs list. sl@0: * The list should be terminated with G_TYPE_INVALID. sl@0: * sl@0: * Returns: #FALSE if an error is set. sl@0: */ sl@0: #ifdef __SYMBIAN32__ sl@0: EXPORT_C sl@0: #endif sl@0: gboolean sl@0: dbus_g_proxy_end_call (DBusGProxy *proxy, sl@0: DBusGProxyCall *call, sl@0: GError **error, sl@0: GType first_arg_type, sl@0: ...) sl@0: { sl@0: gboolean ret; sl@0: va_list args; sl@0: sl@0: va_start (args, first_arg_type); sl@0: sl@0: ret = dbus_g_proxy_end_call_internal (proxy, GPOINTER_TO_UINT (call), error, first_arg_type, args); sl@0: sl@0: va_end (args); sl@0: sl@0: return ret; sl@0: } sl@0: sl@0: /** sl@0: * dbus_g_proxy_call: sl@0: * @proxy: a proxy for a remote interface sl@0: * @method: method to invoke sl@0: * @error: return location for an error sl@0: * @first_arg_type: type of first "in" argument sl@0: * sl@0: * Function for synchronously invoking a method and receiving reply sl@0: * values. This function is equivalent to dbus_g_proxy_begin_call sl@0: * followed by dbus_g_proxy_end_call. All of the input arguments are sl@0: * specified first, followed by G_TYPE_INVALID, followed by all of the sl@0: * output values, followed by a second G_TYPE_INVALID. Note that sl@0: * this means you must always specify G_TYPE_INVALID twice. sl@0: * sl@0: * Returns: #FALSE if an error is set, #TRUE otherwise. sl@0: */ sl@0: #ifdef __SYMBIAN32__ sl@0: EXPORT_C sl@0: #endif sl@0: gboolean sl@0: dbus_g_proxy_call (DBusGProxy *proxy, sl@0: const char *method, sl@0: GError **error, sl@0: GType first_arg_type, sl@0: ...) sl@0: { sl@0: gboolean ret; sl@0: guint call_id; sl@0: va_list args; sl@0: GValueArray *in_args; sl@0: sl@0: g_return_val_if_fail (DBUS_IS_G_PROXY (proxy), FALSE); sl@0: g_return_val_if_fail (!DBUS_G_PROXY_DESTROYED (proxy), FALSE); sl@0: sl@0: va_start (args, first_arg_type); sl@0: sl@0: DBUS_G_VALUE_ARRAY_COLLECT_ALL (in_args, first_arg_type, args); sl@0: sl@0: call_id = dbus_g_proxy_begin_call_internal (proxy, method, NULL, NULL, NULL, in_args,-1); sl@0: sl@0: g_value_array_free (in_args); sl@0: sl@0: first_arg_type = va_arg (args, GType); sl@0: ret = dbus_g_proxy_end_call_internal (proxy, call_id, error, first_arg_type, args); sl@0: sl@0: va_end (args); sl@0: sl@0: return ret; sl@0: } sl@0: sl@0: /** sl@0: * dbus_g_proxy_call_with_timeout: sl@0: * @proxy: a proxy for a remote interface sl@0: * @method: method to invoke sl@0: * @timeout: specify the timeout in milliseconds sl@0: * @error: return location for an error sl@0: * @first_arg_type: type of first "in" argument sl@0: * sl@0: * Function for synchronously invoking a method and receiving reply sl@0: * values. This function is equivalent to dbus_g_proxy_begin_call sl@0: * followed by dbus_g_proxy_end_call. All of the input arguments are sl@0: * specified first, followed by G_TYPE_INVALID, followed by all of the sl@0: * output values, followed by a second G_TYPE_INVALID. Note that sl@0: * this means you must always specify G_TYPE_INVALID twice. sl@0: * sl@0: * Returns: #FALSE if an error is set, #TRUE otherwise. sl@0: */ sl@0: #ifdef __SYMBIAN32__ sl@0: EXPORT_C sl@0: #endif sl@0: gboolean sl@0: dbus_g_proxy_call_with_timeout (DBusGProxy *proxy, sl@0: const char *method, sl@0: int timeout, sl@0: GError **error, sl@0: GType first_arg_type, sl@0: ...) sl@0: { sl@0: gboolean ret; sl@0: guint call_id; sl@0: va_list args; sl@0: GValueArray *in_args; sl@0: sl@0: g_return_val_if_fail (DBUS_IS_G_PROXY (proxy), FALSE); sl@0: g_return_val_if_fail (!DBUS_G_PROXY_DESTROYED (proxy), FALSE); sl@0: sl@0: va_start (args, first_arg_type); sl@0: sl@0: DBUS_G_VALUE_ARRAY_COLLECT_ALL (in_args, first_arg_type, args); sl@0: sl@0: call_id = dbus_g_proxy_begin_call_internal (proxy, method, NULL, NULL, NULL, in_args,timeout); sl@0: sl@0: g_value_array_free (in_args); sl@0: sl@0: first_arg_type = va_arg (args, GType); sl@0: ret = dbus_g_proxy_end_call_internal (proxy, call_id, error, first_arg_type, args); sl@0: sl@0: va_end (args); sl@0: sl@0: return ret; sl@0: } sl@0: sl@0: /** sl@0: * dbus_g_proxy_call_no_reply: sl@0: * @proxy: a proxy for a remote interface sl@0: * @method: the name of the method to invoke sl@0: * @first_arg_type: type of the first argument sl@0: * sl@0: * Sends a method call message as with dbus_g_proxy_begin_call(), but sl@0: * does not ask for a reply or allow you to receive one. sl@0: * sl@0: * TODO: this particular function shouldn't die on out of memory, sl@0: * since you should be able to do a call with large arguments. sl@0: */ sl@0: #ifdef __SYMBIAN32__ sl@0: EXPORT_C sl@0: #endif sl@0: void sl@0: dbus_g_proxy_call_no_reply (DBusGProxy *proxy, sl@0: const char *method, sl@0: GType first_arg_type, sl@0: ...) sl@0: { sl@0: DBusMessage *message; sl@0: va_list args; sl@0: GValueArray *in_args; sl@0: DBusGProxyPrivate *priv; sl@0: sl@0: g_return_if_fail (DBUS_IS_G_PROXY (proxy)); sl@0: g_return_if_fail (!DBUS_G_PROXY_DESTROYED (proxy)); sl@0: sl@0: priv = DBUS_G_PROXY_GET_PRIVATE(proxy); sl@0: sl@0: va_start (args, first_arg_type); sl@0: DBUS_G_VALUE_ARRAY_COLLECT_ALL (in_args, first_arg_type, args); sl@0: sl@0: message = dbus_g_proxy_marshal_args_to_message (proxy, method, in_args); sl@0: sl@0: g_value_array_free (in_args); sl@0: va_end (args); sl@0: sl@0: if (!message) sl@0: goto oom; sl@0: sl@0: dbus_message_set_no_reply (message, TRUE); sl@0: sl@0: if (!dbus_connection_send (priv->manager->connection, sl@0: message, sl@0: NULL)) sl@0: goto oom; sl@0: dbus_message_unref (message); sl@0: return; sl@0: sl@0: oom: sl@0: g_error ("Out of memory"); sl@0: } sl@0: sl@0: /** sl@0: * dbus_g_proxy_cancel_call sl@0: * @proxy: a proxy for a remote interface sl@0: * @call: the pending call ID from dbus_g_proxy_begin_call() sl@0: * sl@0: * Cancels a pending method call. The method call was normally sl@0: * initiated with dbus_g_proxy_begin_call(). This function sl@0: * may not be used on pending calls that have already been sl@0: * ended with dbus_g_proxy_end_call. sl@0: */ sl@0: #ifdef __SYMBIAN32__ sl@0: EXPORT_C sl@0: #endif sl@0: void sl@0: dbus_g_proxy_cancel_call (DBusGProxy *proxy, sl@0: DBusGProxyCall *call) sl@0: { sl@0: guint call_id; sl@0: DBusPendingCall *pending; sl@0: DBusGProxyPrivate *priv; sl@0: sl@0: g_return_if_fail (DBUS_IS_G_PROXY (proxy)); sl@0: g_return_if_fail (!DBUS_G_PROXY_DESTROYED (proxy)); sl@0: sl@0: priv = DBUS_G_PROXY_GET_PRIVATE(proxy); sl@0: sl@0: call_id = DBUS_G_PROXY_CALL_TO_ID (call); sl@0: sl@0: pending = g_hash_table_lookup (priv->pending_calls, GUINT_TO_POINTER (call_id)); sl@0: g_return_if_fail (pending != NULL); sl@0: sl@0: dbus_pending_call_cancel (pending); sl@0: sl@0: g_hash_table_remove (priv->pending_calls, GUINT_TO_POINTER (call_id)); sl@0: } sl@0: sl@0: /** sl@0: * dbus_g_proxy_send: sl@0: * @proxy: a proxy for a remote interface sl@0: * @message: the message to address and send sl@0: * @client_serial: return location for message's serial, or #NULL sl@0: * sl@0: * Sends a message to the interface we're proxying for. Does not sl@0: * block or wait for a reply. The message is only actually written out sl@0: * when you return to the main loop or block in sl@0: * dbus_connection_flush(). sl@0: * sl@0: * The message is modified to be addressed to the target interface. sl@0: * That is, a destination name field or whatever is needed will be sl@0: * added to the message. The basic point of this function is to add sl@0: * the necessary header fields, otherwise it's equivalent to sl@0: * dbus_connection_send(). sl@0: * sl@0: * This function adds a reference to the message, so the caller sl@0: * still owns its original reference. sl@0: */ sl@0: #ifdef __SYMBIAN32__ sl@0: EXPORT_C sl@0: #endif sl@0: void sl@0: dbus_g_proxy_send (DBusGProxy *proxy, sl@0: DBusMessage *message, sl@0: dbus_uint32_t *client_serial) sl@0: { sl@0: DBusGProxyPrivate *priv; sl@0: sl@0: g_return_if_fail (DBUS_IS_G_PROXY (proxy)); sl@0: g_return_if_fail (!DBUS_G_PROXY_DESTROYED (proxy)); sl@0: sl@0: priv = DBUS_G_PROXY_GET_PRIVATE(proxy); sl@0: sl@0: if (priv->name) sl@0: { sl@0: if (!dbus_message_set_destination (message, priv->name)) sl@0: g_error ("Out of memory"); sl@0: } sl@0: if (priv->path) sl@0: { sl@0: if (!dbus_message_set_path (message, priv->path)) sl@0: g_error ("Out of memory"); sl@0: } sl@0: if (priv->interface) sl@0: { sl@0: if (!dbus_message_set_interface (message, priv->interface)) sl@0: g_error ("Out of memory"); sl@0: } sl@0: sl@0: if (!dbus_connection_send (priv->manager->connection, message, client_serial)) sl@0: g_error ("Out of memory\n"); sl@0: } sl@0: sl@0: static void sl@0: array_free_all (gpointer array) sl@0: { sl@0: g_array_free (array, TRUE); sl@0: } sl@0: sl@0: /** sl@0: * dbus_g_proxy_add_signal: sl@0: * @proxy: the proxy for a remote interface sl@0: * @signal_name: the name of the signal sl@0: * @first_type: the first argument type, or G_TYPE_INVALID if none sl@0: * sl@0: * Specifies the argument signature of a signal;.only necessary sl@0: * if the remote object does not support introspection. The arguments sl@0: * specified are the GLib types expected. sl@0: */ sl@0: #ifdef __SYMBIAN32__ sl@0: EXPORT_C sl@0: #endif sl@0: void sl@0: dbus_g_proxy_add_signal (DBusGProxy *proxy, sl@0: const char *signal_name, sl@0: GType first_type, sl@0: ...) sl@0: { sl@0: GQuark q; sl@0: char *name; sl@0: GArray *gtypesig; sl@0: GType gtype; sl@0: va_list args; sl@0: DBusGProxyPrivate *priv; sl@0: sl@0: g_return_if_fail (DBUS_IS_G_PROXY (proxy)); sl@0: g_return_if_fail (!DBUS_G_PROXY_DESTROYED (proxy)); sl@0: g_return_if_fail (signal_name != NULL); sl@0: sl@0: priv = DBUS_G_PROXY_GET_PRIVATE(proxy); sl@0: sl@0: name = create_signal_name (priv->interface, signal_name); sl@0: sl@0: q = g_quark_from_string (name); sl@0: sl@0: g_return_if_fail (g_datalist_id_get_data (&priv->signal_signatures, q) == NULL); sl@0: sl@0: gtypesig = g_array_new (FALSE, TRUE, sizeof (GType)); sl@0: sl@0: va_start (args, first_type); sl@0: gtype = first_type; sl@0: while (gtype != G_TYPE_INVALID) sl@0: { sl@0: g_array_append_val (gtypesig, gtype); sl@0: gtype = va_arg (args, GType); sl@0: } sl@0: va_end (args); sl@0: sl@0: #ifndef G_DISABLE_CHECKS sl@0: if (_dbus_gobject_lookup_marshaller (G_TYPE_NONE, gtypesig->len, (const GType*) gtypesig->data) == NULL) sl@0: g_warning ("No marshaller for signature of signal '%s'", signal_name); sl@0: #endif sl@0: sl@0: sl@0: g_datalist_id_set_data_full (&priv->signal_signatures, sl@0: q, gtypesig, sl@0: array_free_all); sl@0: sl@0: g_free (name); sl@0: } sl@0: sl@0: /** sl@0: * dbus_g_proxy_connect_signal: sl@0: * @proxy: a proxy for a remote interface sl@0: * @signal_name: the DBus signal name to listen for sl@0: * @handler: the handler to connect sl@0: * @data: data to pass to handler sl@0: * @free_data_func: callback function to destroy data sl@0: * sl@0: * Connect a signal handler to a proxy for a remote interface. When sl@0: * the remote interface emits the specified signal, the proxy will sl@0: * emit a corresponding GLib signal. sl@0: */ sl@0: #ifdef __SYMBIAN32__ sl@0: EXPORT_C sl@0: #endif sl@0: void sl@0: dbus_g_proxy_connect_signal (DBusGProxy *proxy, sl@0: const char *signal_name, sl@0: GCallback handler, sl@0: void *data, sl@0: GClosureNotify free_data_func) sl@0: { sl@0: char *name; sl@0: GClosure *closure; sl@0: GQuark q; sl@0: DBusGProxyPrivate *priv; sl@0: sl@0: g_return_if_fail (DBUS_IS_G_PROXY (proxy)); sl@0: g_return_if_fail (!DBUS_G_PROXY_DESTROYED (proxy)); sl@0: g_return_if_fail (signal_name != NULL); sl@0: g_return_if_fail (handler != NULL); sl@0: sl@0: priv = DBUS_G_PROXY_GET_PRIVATE(proxy); sl@0: name = create_signal_name (priv->interface, signal_name); sl@0: sl@0: q = g_quark_try_string (name); sl@0: sl@0: #ifndef G_DISABLE_CHECKS sl@0: if (q == 0 || g_datalist_id_get_data (&priv->signal_signatures, q) == NULL) sl@0: { sl@0: g_warning ("Must add the signal '%s' with dbus_g_proxy_add_signal() prior to connecting to it\n", name); sl@0: g_free (name); sl@0: return; sl@0: } sl@0: #endif sl@0: sl@0: closure = g_cclosure_new (G_CALLBACK (handler), data, free_data_func); sl@0: sl@0: g_signal_connect_closure_by_id (G_OBJECT (proxy), sl@0: signals[RECEIVED], sl@0: q, sl@0: closure, FALSE); sl@0: sl@0: g_free (name); sl@0: } sl@0: sl@0: /** sl@0: * dbus_g_proxy_disconnect_signal: sl@0: * @proxy: a proxy for a remote interface sl@0: * @signal_name: the DBus signal name to disconnect sl@0: * @handler: the handler to disconnect sl@0: * @data: the data that was registered with handler sl@0: * sl@0: * Disconnect all signal handlers from a proxy that match the given sl@0: * criteria. sl@0: */ sl@0: #ifdef __SYMBIAN32__ sl@0: EXPORT_C sl@0: #endif sl@0: void sl@0: dbus_g_proxy_disconnect_signal (DBusGProxy *proxy, sl@0: const char *signal_name, sl@0: GCallback handler, sl@0: void *data) sl@0: { sl@0: char *name; sl@0: GQuark q; sl@0: DBusGProxyPrivate *priv; sl@0: sl@0: g_return_if_fail (DBUS_IS_G_PROXY (proxy)); sl@0: g_return_if_fail (!DBUS_G_PROXY_DESTROYED (proxy)); sl@0: g_return_if_fail (signal_name != NULL); sl@0: g_return_if_fail (handler != NULL); sl@0: sl@0: priv = DBUS_G_PROXY_GET_PRIVATE(proxy); sl@0: name = create_signal_name (priv->interface, signal_name); sl@0: sl@0: q = g_quark_try_string (name); sl@0: sl@0: if (q != 0) sl@0: { sl@0: g_signal_handlers_disconnect_matched (G_OBJECT (proxy), sl@0: G_SIGNAL_MATCH_DETAIL | sl@0: G_SIGNAL_MATCH_FUNC | sl@0: G_SIGNAL_MATCH_DATA, sl@0: signals[RECEIVED], sl@0: q, sl@0: NULL, sl@0: #ifdef WINSCW sl@0: G_CALLBACK (handler), data); sl@0: #else sl@0: (gpointer)(handler), data); sl@0: #endif sl@0: } sl@0: else sl@0: { sl@0: g_warning ("Attempt to disconnect from signal '%s' which is not registered\n", sl@0: name); sl@0: } sl@0: sl@0: g_free (name); sl@0: } sl@0: sl@0: /** @} End of DBusGLib public */ sl@0: sl@0: #ifdef DBUS_BUILD_TESTS sl@0: sl@0: /** sl@0: * @ingroup DBusGLibInternals sl@0: * Unit test for GLib proxy functions sl@0: * Returns: #TRUE on success. sl@0: */ sl@0: gboolean sl@0: _dbus_g_proxy_test (void) sl@0: { sl@0: sl@0: sl@0: return TRUE; sl@0: } sl@0: sl@0: #endif /* DBUS_BUILD_TESTS */