/*
* handle-repo-dynamic.c - mechanism to store and retrieve handles on a
* connection (general implementation with dynamic handle allocation and
* recycling)
*
* Copyright (C) 2007 Collabora Ltd.
* Copyright (C) 2007 Nokia Corporation
*
* This library is free software; you can redistribute it and/or
* modify it under the terms of the GNU Lesser General Public
* License as published by the Free Software Foundation; either
* version 2.1 of the License, or (at your option) any later version.
*
* This library is distributed in the hope that it will be useful,
* but WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
* Lesser General Public License for more details.
*
* You should have received a copy of the GNU Lesser General Public
* License along with this library; if not, write to the Free Software
* Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA
*/
/**
* SECTION:handle-repo-dynamic
* @title: TpDynamicHandleRepo
* @short_description: general handle repository implementation, with dynamic
* handle allocation and recycling
* @see_also: TpHandleRepoIface, TpStaticHandleRepo
*
* A dynamic handle repository will accept arbitrary handles, which can
* be created and destroyed at runtime.
*
* The #TpHandleRepoIface:handle-type property must be set at construction
* time; the #TpDynamicHandleRepo:normalize-function property may be set to
* perform validation and normalization on handle ID strings.
*
* Most connection managers will use this for all supported handle types
* except %TP_HANDLE_TYPE_CONTACT_LIST.
*/
#include
#include
#include
#include
#include
#include
/**
* TpDynamicHandleRepoNormalizeFunc:
* @repo: The repository on which tp_handle_lookup() or tp_handle_ensure()
* was called
* @id: The name to be normalized
* @context: Arbitrary context passed to tp_handle_lookup() or
* tp_handle_ensure()
* @error: Used to raise the Telepathy error InvalidHandle with an appropriate
* message if NULL is returned
*
* Signature of the normalization function optionally used by
* #TpDynamicHandleRepo instances.
*
* Returns: a normalized version of @id (to be freed with g_free by the
* caller), or NULL if @id is not valid for this repository
*/
/**
* tp_dynamic_handle_repo_new:
* @handle_type: The handle type
* @normalize_func: The function to be used to normalize and validate handles,
* or %NULL to accept all handles as-is
* @default_normalize_context: The context pointer to be passed to the
* @normalize_func if a %NULL context is passed to tp_handle_lookup() and
* tp_handle_ensure(); this may itself be %NULL
*
*
*
* Returns: a new dynamic handle repository
*/
/* Handle leak tracing */
#ifdef ENABLE_HANDLE_LEAK_DEBUG
#include
#include
#ifdef HAVE_EXECINFO_H
#include
#else
#error "Handle leak debug requires execinfo.h"
#endif
typedef enum {
HL_REFFED,
HL_CREATED,
HL_UNREFFED,
HL_CREATED_FLOATING
} HandleLeakEvent;
typedef struct _HandleLeakTrace HandleLeakTrace;
struct _HandleLeakTrace
{
char **trace;
int len;
HandleLeakEvent event;
};
static void
handle_leak_trace_free (HandleLeakTrace *hltrace)
{
free (hltrace->trace);
g_slice_free (HandleLeakTrace, hltrace);
}
static void
handle_leak_trace_free_gfunc (gpointer data, gpointer user_data)
{
return handle_leak_trace_free ((HandleLeakTrace *) data);
}
#endif /* ENABLE_HANDLE_LEAK_DEBUG */
/* Handle private data structure */
typedef struct _TpHandlePriv TpHandlePriv;
struct _TpHandlePriv
{
/* Reference count */
guint refcount;
/* Unique ID */
gchar *string;
#ifdef ENABLE_HANDLE_LEAK_DEBUG
GSList *traces;
#endif /* ENABLE_HANDLE_LEAK_DEBUG */
GData *datalist;
};
static TpHandlePriv *
handle_priv_new (void)
{
TpHandlePriv *priv;
priv = g_slice_new0 (TpHandlePriv);
priv->refcount = 1;
g_datalist_init (&(priv->datalist));
return priv;
}
/* Dynamic handle repo */
static void
handle_priv_free (TpHandlePriv *priv)
{
g_return_if_fail (priv != NULL);
g_free (priv->string);
g_datalist_clear (&(priv->datalist));
#ifdef ENABLE_HANDLE_LEAK_DEBUG
g_slist_foreach (priv->traces, handle_leak_trace_free_gfunc, NULL);
g_slist_free (priv->traces);
#endif /* ENABLE_HANDLE_LEAK_DEBUG */
g_slice_free (TpHandlePriv, priv);
}
enum
{
PROP_HANDLE_TYPE = 1,
PROP_NORMALIZE_FUNCTION,
PROP_DEFAULT_NORMALIZE_CONTEXT,
};
/**
* TpDynamicHandleRepoClass:
*
* The class of a dynamic handle repository. The contents of the struct
* are private.
*/
struct _TpDynamicHandleRepoClass {
GObjectClass parent_class;
};
/**
* TpDynamicHandleRepo:
*
* A dynamic handle repository. The contents of the struct are private.
*/
struct _TpDynamicHandleRepo {
GObject parent;
TpHandleType handle_type;
/* Map GUINT_TO_POINTER(handle) -> (TpHandlePriv *) */
GHashTable *handle_to_priv;
/* Map contact unique ID -> GUINT_TO_POINTER(handle) */
GHashTable *string_to_handle;
/* Heap-queue of GUINT_TO_POINTER(handle): previously used handles */
TpHeap *free_handles;
/* Smallest handle which has never been allocated */
guint next_handle;
/* Map (client name) -> (TpHandleSet *): handles being held by that client */
GData *holder_to_handle_set;
/* Normalization function */
TpDynamicHandleRepoNormalizeFunc normalize_function;
/* Context for normalization function if NULL is passed to _ensure or
* _lookup
*/
gpointer default_normalize_context;
TpDBusDaemon *bus_daemon;
};
static void dynamic_repo_iface_init (gpointer g_iface,
gpointer iface_data);
G_DEFINE_TYPE_WITH_CODE (TpDynamicHandleRepo, tp_dynamic_handle_repo,
G_TYPE_OBJECT, G_IMPLEMENT_INTERFACE (TP_TYPE_HANDLE_REPO_IFACE,
dynamic_repo_iface_init));
static inline TpHandlePriv *
handle_priv_lookup (TpDynamicHandleRepo *repo,
TpHandle handle)
{
return g_hash_table_lookup (repo->handle_to_priv, GINT_TO_POINTER (handle));
}
static TpHandle
handle_alloc (TpDynamicHandleRepo *repo)
{
g_assert (repo != NULL);
if (tp_heap_size (repo->free_handles))
return GPOINTER_TO_UINT (tp_heap_extract_first (repo->free_handles));
else
return repo->next_handle++;
}
static gint
handle_compare_func (gconstpointer a, gconstpointer b)
{
TpHandle first = GPOINTER_TO_UINT (a);
TpHandle second = GPOINTER_TO_UINT (b);
return (first == second) ? 0 : ((first < second) ? -1 : 1);
}
static void
handle_priv_remove (TpDynamicHandleRepo *repo,
TpHandle handle)
{
TpHandlePriv *priv;
const gchar *string;
g_return_if_fail (handle != 0);
g_return_if_fail (repo != NULL);
priv = handle_priv_lookup (repo, handle);
g_assert (priv != NULL);
string = priv->string;
g_hash_table_remove (repo->string_to_handle, string);
g_hash_table_remove (repo->handle_to_priv, GUINT_TO_POINTER (handle));
if (handle == repo->next_handle - 1)
repo->next_handle--;
else
tp_heap_add (repo->free_handles, GUINT_TO_POINTER (handle));
}
static void
handles_name_owner_changed_cb (TpDBusDaemon *dbus_daemon,
const gchar *name,
const gchar *new_owner,
gpointer user_data)
{
TpDynamicHandleRepo *repo = user_data;
if (new_owner == NULL || new_owner[0] == '\0')
{
tp_dbus_daemon_cancel_name_owner_watch (dbus_daemon, name,
handles_name_owner_changed_cb, repo);
g_datalist_remove_data (&repo->holder_to_handle_set, name);
}
}
static void
tp_dynamic_handle_repo_init (TpDynamicHandleRepo *self)
{
self->handle_to_priv = g_hash_table_new_full (g_direct_hash,
g_direct_equal, NULL, (GDestroyNotify) handle_priv_free);
self->string_to_handle = g_hash_table_new (g_str_hash, g_str_equal);
self->free_handles = tp_heap_new (handle_compare_func, NULL);
self->next_handle = 1;
g_datalist_init (&self->holder_to_handle_set);
self->bus_daemon = tp_dbus_daemon_dup (NULL);
if (self->bus_daemon == NULL)
g_error ("Unable to connect to starter bus");
return;
}
#ifdef ENABLE_HANDLE_LEAK_DEBUG
static const char *
handle_leak_describe_event (HandleLeakEvent event)
{
switch (event)
{
case HL_REFFED:
return "reffed";
case HL_UNREFFED:
return "unreffed";
case HL_CREATED:
return "created with 1 ref";
case HL_CREATED_FLOATING:
return "created with 0 refs";
}
g_assert_not_reached ();
return NULL;
}
static void
handle_leak_debug_printbt_foreach (gpointer data, gpointer user_data)
{
HandleLeakTrace *hltrace = (HandleLeakTrace *) data;
int i;
printf ("\t %s at:\n", handle_leak_describe_event (hltrace->event));
for (i = 1; i < hltrace->len; i++)
{
printf ("\t\t%s\n", hltrace->trace[i]);
}
printf ("\n");
}
static void
handle_leak_debug_printhandles_foreach (gpointer key,
gpointer value,
gpointer ignore)
{
TpHandle handle = GPOINTER_TO_UINT (key);
TpHandlePriv *priv = (TpHandlePriv *) value;
printf ("\t%05u: %s (%u refs), traces:\n", handle, priv->string,
priv->refcount);
g_slist_foreach (priv->traces, handle_leak_debug_printbt_foreach, NULL);
}
static void
handle_leak_debug_print_report (TpDynamicHandleRepo *self)
{
g_assert (self != NULL);
if (g_hash_table_size (self->handle_to_priv) == 0)
{
printf ("No handles were leaked\n");
return;
}
printf ("HANDLE LEAK: The following handles were not freed from repo %p:\n",
self);
g_hash_table_foreach (self->handle_to_priv,
handle_leak_debug_printhandles_foreach, NULL);
}
static HandleLeakTrace *
handle_leak_debug_bt (HandleLeakEvent event)
{
void *bt_addresses[16];
HandleLeakTrace *ret = g_slice_new0 (HandleLeakTrace);
ret->event = event;
ret->len = backtrace (bt_addresses, 16);
ret->trace = backtrace_symbols (bt_addresses, ret->len);
return ret;
}
#define HANDLE_LEAK_DEBUG_DO(traces_slist, self, handle, event) \
{ (traces_slist) = g_slist_append ((traces_slist), \
handle_leak_debug_bt (event)); \
g_debug ("%p: handle %u %s", self, handle, \
handle_leak_describe_event (event)); \
}
#else /* !ENABLE_HANDLE_LEAK_DEBUG */
#define HANDLE_LEAK_DEBUG_DO(traces_slist, self, handle, event) {}
#endif /* ENABLE_HANDLE_LEAK_DEBUG */
static void
foreach_cancel_watch (GQuark key_id,
gpointer handle_set,
gpointer user_data)
{
TpDynamicHandleRepo *self = user_data;
tp_dbus_daemon_cancel_name_owner_watch (self->bus_daemon,
g_quark_to_string (key_id), handles_name_owner_changed_cb, self);
}
static void
dynamic_dispose (GObject *obj)
{
TpDynamicHandleRepo *self = TP_DYNAMIC_HANDLE_REPO (obj);
if (self->bus_daemon != NULL)
{
g_datalist_foreach (&self->holder_to_handle_set, foreach_cancel_watch,
self);
g_object_unref (self->bus_daemon);
self->bus_daemon = NULL;
}
G_OBJECT_CLASS (tp_dynamic_handle_repo_parent_class)->dispose (obj);
}
static void
dynamic_finalize (GObject *obj)
{
TpDynamicHandleRepo *self = TP_DYNAMIC_HANDLE_REPO (obj);
GObjectClass *parent = G_OBJECT_CLASS (tp_dynamic_handle_repo_parent_class);
g_assert (self->handle_to_priv);
g_assert (self->string_to_handle);
g_datalist_clear (&self->holder_to_handle_set);
#ifdef ENABLE_HANDLE_LEAK_DEBUG
handle_leak_debug_print_report (self);
#endif
g_hash_table_destroy (self->handle_to_priv);
g_hash_table_destroy (self->string_to_handle);
tp_heap_destroy (self->free_handles);
if (parent->finalize)
parent->finalize (obj);
}
static void
dynamic_get_property (GObject *object,
guint property_id,
GValue *value,
GParamSpec *pspec)
{
TpDynamicHandleRepo *self = TP_DYNAMIC_HANDLE_REPO (object);
switch (property_id)
{
case PROP_HANDLE_TYPE:
g_value_set_uint (value, self->handle_type);
break;
case PROP_NORMALIZE_FUNCTION:
g_value_set_pointer (value, self->normalize_function);
break;
case PROP_DEFAULT_NORMALIZE_CONTEXT:
g_value_set_pointer (value, self->default_normalize_context);
break;
default:
G_OBJECT_WARN_INVALID_PROPERTY_ID (object, property_id, pspec);
break;
}
}
static void
dynamic_set_property (GObject *object,
guint property_id,
const GValue *value,
GParamSpec *pspec)
{
TpDynamicHandleRepo *self = TP_DYNAMIC_HANDLE_REPO (object);
switch (property_id)
{
case PROP_HANDLE_TYPE:
self->handle_type = g_value_get_uint (value);
break;
case PROP_NORMALIZE_FUNCTION:
self->normalize_function = g_value_get_pointer (value);
break;
case PROP_DEFAULT_NORMALIZE_CONTEXT:
self->default_normalize_context = g_value_get_pointer (value);
break;
default:
G_OBJECT_WARN_INVALID_PROPERTY_ID (object, property_id, pspec);
break;
}
}
static void
tp_dynamic_handle_repo_class_init (TpDynamicHandleRepoClass *klass)
{
GObjectClass *object_class = G_OBJECT_CLASS (klass);
GParamSpec *param_spec;
object_class->dispose = dynamic_dispose;
object_class->finalize = dynamic_finalize;
object_class->get_property = dynamic_get_property;
object_class->set_property = dynamic_set_property;
g_object_class_override_property (object_class, PROP_HANDLE_TYPE,
"handle-type");
/**
* TpDynamicHandleRepo:normalize-function:
*
* An optional #TpDynamicHandleRepoNormalizeFunc used to validate and
* normalize handle IDs. If %NULL (which is the default), any handle ID is
* accepted as-is (equivalent to supplying a pointer to a function that just
* calls g_strdup).
*/
param_spec = g_param_spec_pointer ("normalize-function",
"Normalization function",
"A TpDynamicHandleRepoNormalizeFunc used to normalize handle IDs.",
G_PARAM_CONSTRUCT_ONLY | G_PARAM_READWRITE |
G_PARAM_STATIC_NAME | G_PARAM_STATIC_BLURB | G_PARAM_STATIC_NICK);
g_object_class_install_property (object_class, PROP_NORMALIZE_FUNCTION,
param_spec);
/**
* TpDynamicHandleRepo:default-normalize-context:
*
* An optional default context given to the
* #TpDynamicHandleRepo:normalize-function if %NULL is passed as context to
* the ensure or lookup functions, e.g. when RequestHandle is called via
* D-Bus. The default is %NULL.
*/
param_spec = g_param_spec_pointer ("default-normalize-context",
"Default normalization context",
"The default context given to the normalize-function if NULL is passed "
"as context to the ensure or lookup function, e.g. when RequestHandle"
"is called via D-Bus. The default is NULL.",
G_PARAM_CONSTRUCT_ONLY | G_PARAM_READWRITE |
G_PARAM_STATIC_NAME | G_PARAM_STATIC_BLURB | G_PARAM_STATIC_NICK);
g_object_class_install_property (object_class,
PROP_DEFAULT_NORMALIZE_CONTEXT, param_spec);
}
static gboolean
dynamic_handle_is_valid (TpHandleRepoIface *irepo, TpHandle handle,
GError **error)
{
TpDynamicHandleRepo *self = TP_DYNAMIC_HANDLE_REPO (irepo);
if (handle_priv_lookup (self, handle) == NULL)
{
g_set_error (error, TP_ERRORS, TP_ERROR_INVALID_HANDLE,
"handle %u is not currently a valid %s handle (type %u)",
handle, tp_handle_type_to_string (self->handle_type),
self->handle_type);
return FALSE;
}
else
{
return TRUE;
}
}
static gboolean
dynamic_handles_are_valid (TpHandleRepoIface *irepo, const GArray *handles,
gboolean allow_zero, GError **error)
{
guint i;
g_return_val_if_fail (handles != NULL, FALSE);
for (i = 0; i < handles->len; i++)
{
TpHandle handle = g_array_index (handles, TpHandle, i);
if (handle == 0 && allow_zero)
continue;
if (!dynamic_handle_is_valid (irepo, handle, error))
return FALSE;
}
return TRUE;
}
static void
dynamic_unref_handle (TpHandleRepoIface *repo, TpHandle handle)
{
TpDynamicHandleRepo *self = TP_DYNAMIC_HANDLE_REPO (repo);
TpHandlePriv *priv = handle_priv_lookup (self, handle);
g_return_if_fail (priv != NULL);
HANDLE_LEAK_DEBUG_DO (priv->traces, repo, handle, HL_UNREFFED)
g_assert (priv->refcount > 0);
priv->refcount--;
if (priv->refcount == 0)
handle_priv_remove (self, handle);
}
static void
dynamic_ref_handle (TpHandleRepoIface *repo, TpHandle handle)
{
TpHandlePriv *priv = handle_priv_lookup (TP_DYNAMIC_HANDLE_REPO (repo),
handle);
g_return_if_fail (priv != NULL);
priv->refcount++;
HANDLE_LEAK_DEBUG_DO (priv->traces, repo, handle, HL_REFFED)
}
static gboolean
dynamic_client_hold_handle (TpHandleRepoIface *repo,
const gchar *client_name,
TpHandle handle,
GError **error)
{
TpDynamicHandleRepo *self;
TpHandleSet *handle_set;
g_return_val_if_fail (handle != 0, FALSE);
g_return_val_if_fail (repo != NULL, FALSE);
self = TP_DYNAMIC_HANDLE_REPO (repo);
if (!client_name || *client_name == '\0')
{
g_critical ("%s: called with invalid client name", G_STRFUNC);
g_set_error (error, TP_ERRORS, TP_ERROR_INVALID_ARGUMENT,
"invalid client name");
return FALSE;
}
handle_set = (TpHandleSet *) g_datalist_get_data (
&(self->holder_to_handle_set), client_name);
if (!handle_set)
{
handle_set = tp_handle_set_new (repo);
g_datalist_set_data_full (&(self->holder_to_handle_set),
client_name,
handle_set,
(GDestroyNotify) tp_handle_set_destroy);
tp_dbus_daemon_watch_name_owner (self->bus_daemon, client_name,
handles_name_owner_changed_cb, self, NULL);
}
tp_handle_set_add (handle_set, handle);
return TRUE;
}
static gboolean
dynamic_client_release_handle (TpHandleRepoIface *repo,
const gchar *client_name,
TpHandle handle,
GError **error)
{
TpDynamicHandleRepo *self;
TpHandleSet *handle_set;
g_return_val_if_fail (handle != 0, FALSE);
g_return_val_if_fail (repo != NULL, FALSE);
self = TP_DYNAMIC_HANDLE_REPO (repo);
if (!client_name || *client_name == '\0')
{
g_critical ("%s: called with invalid client name", G_STRFUNC);
g_set_error (error, TP_ERRORS, TP_ERROR_INVALID_ARGUMENT,
"invalid client name");
return FALSE;
}
handle_set = (TpHandleSet *) g_datalist_get_data (
&(self->holder_to_handle_set), client_name);
if (!handle_set)
{
g_debug ("%s: no handle set found for the given client %s",
G_STRFUNC, client_name);
g_set_error (error, TP_ERRORS, TP_ERROR_NOT_AVAILABLE,
"the given client %s wasn't holding any handles", client_name);
return FALSE;
}
if (!tp_handle_set_remove (handle_set, handle))
{
g_debug ("%s: the client %s wasn't holding the handle %u", G_STRFUNC,
client_name, handle);
g_set_error (error, TP_ERRORS, TP_ERROR_NOT_AVAILABLE,
"the given client %s wasn't holding the handle %u", client_name,
handle);
return FALSE;
}
return TRUE;
}
static const char *
dynamic_inspect_handle (TpHandleRepoIface *irepo, TpHandle handle)
{
TpDynamicHandleRepo *self = TP_DYNAMIC_HANDLE_REPO (irepo);
TpHandlePriv *priv = handle_priv_lookup (self, handle);
if (priv == NULL)
return NULL;
else
return priv->string;
}
/**
* tp_dynamic_handle_repo_lookup_exact:
* @irepo: The handle repository
* @id: The name to be looked up
*
* Look up a name in the repository, returning the corresponding handle if
* it is present in the repository, without creating a new reference.
*
* Unlike tp_handle_lookup() this function does not perform any normalization;
* it just looks for the literal string you requested. This can be useful to
* call from normalization callbacks (for instance, Gabble's contacts
* repository uses it to see whether we already know that a JID belongs
* to a multi-user chat room member).
*
* Returns: the handle corresponding to the given ID, or 0 if not present
*/
TpHandle
tp_dynamic_handle_repo_lookup_exact (TpHandleRepoIface *irepo,
const char *id)
{
TpDynamicHandleRepo *self = TP_DYNAMIC_HANDLE_REPO (irepo);
return GPOINTER_TO_UINT (g_hash_table_lookup (self->string_to_handle,
id));
}
static TpHandle
dynamic_lookup_handle (TpHandleRepoIface *irepo,
const char *id,
gpointer context,
GError **error)
{
TpDynamicHandleRepo *self = TP_DYNAMIC_HANDLE_REPO (irepo);
TpHandle handle;
gchar *normal_id = NULL;
if (context == NULL)
context = self->default_normalize_context;
if (self->normalize_function)
{
normal_id = (self->normalize_function) (irepo, id, context, error);
if (normal_id == NULL)
return 0;
id = normal_id;
}
handle = GPOINTER_TO_UINT (g_hash_table_lookup (self->string_to_handle,
id));
if (handle == 0)
{
g_set_error (error, TP_ERRORS, TP_ERROR_NOT_AVAILABLE,
"no %s handle (type %u) currently exists for ID \"%s\"",
tp_handle_type_to_string (self->handle_type),
self->handle_type, id);
}
g_free (normal_id);
return handle;
}
static TpHandle
dynamic_ensure_handle (TpHandleRepoIface *irepo,
const char *id,
gpointer context,
GError **error)
{
TpDynamicHandleRepo *self = TP_DYNAMIC_HANDLE_REPO (irepo);
TpHandle handle;
TpHandlePriv *priv;
gchar *normal_id = NULL;
if (context == NULL)
context = self->default_normalize_context;
if (self->normalize_function)
{
normal_id = (self->normalize_function) (irepo, id, context, error);
if (normal_id == NULL)
return 0;
id = normal_id;
}
handle = GPOINTER_TO_UINT (g_hash_table_lookup (self->string_to_handle,
id));
if (handle)
{
dynamic_ref_handle (irepo, handle);
g_free (normal_id);
return handle;
}
handle = handle_alloc (self);
priv = handle_priv_new ();
if (self->normalize_function)
priv->string = normal_id;
else
priv->string = g_strdup (id);
g_hash_table_insert (self->handle_to_priv, GUINT_TO_POINTER (handle), priv);
g_hash_table_insert (self->string_to_handle, priv->string,
GUINT_TO_POINTER (handle));
HANDLE_LEAK_DEBUG_DO (priv->traces, irepo, handle, HL_CREATED)
return handle;
}
static void
dynamic_set_qdata (TpHandleRepoIface *repo, TpHandle handle,
GQuark key_id, gpointer data, GDestroyNotify destroy)
{
TpDynamicHandleRepo *self = TP_DYNAMIC_HANDLE_REPO (repo);
TpHandlePriv *priv = handle_priv_lookup (self, handle);
g_return_if_fail (((void)"invalid handle", priv != NULL));
g_datalist_id_set_data_full (&priv->datalist, key_id, data, destroy);
}
static gpointer
dynamic_get_qdata (TpHandleRepoIface *repo, TpHandle handle,
GQuark key_id)
{
TpDynamicHandleRepo *self = TP_DYNAMIC_HANDLE_REPO (repo);
TpHandlePriv *priv = handle_priv_lookup (self, handle);
g_return_val_if_fail (((void)"invalid handle", priv != NULL), NULL);
return g_datalist_id_get_data (&priv->datalist, key_id);
}
static void
dynamic_repo_iface_init (gpointer g_iface,
gpointer iface_data)
{
TpHandleRepoIfaceClass *klass = (TpHandleRepoIfaceClass *) g_iface;
klass->handle_is_valid = dynamic_handle_is_valid;
klass->handles_are_valid = dynamic_handles_are_valid;
klass->ref_handle = dynamic_ref_handle;
klass->unref_handle = dynamic_unref_handle;
klass->client_hold_handle = dynamic_client_hold_handle;
klass->client_release_handle = dynamic_client_release_handle;
klass->inspect_handle = dynamic_inspect_handle;
klass->lookup_handle = dynamic_lookup_handle;
klass->ensure_handle = dynamic_ensure_handle;
klass->set_qdata = dynamic_set_qdata;
klass->get_qdata = dynamic_get_qdata;
}