1 /* vi: set et sw=4 ts=4 cino=t0,(0: */
2 /* -*- Mode: C; indent-tabs-mode: nil; c-basic-offset: 4 -*- */
4 * Copyright (C) 2013 Intel Corporation.
6 * Contact: Amarnath Valluri <amarnath.valluri@linux.intel.com>
8 * This library is free software; you can redistribute it and/or
9 * modify it under the terms of the GNU Lesser General Public
10 * License as published by the Free Software Foundation; either
11 * version 2.1 of the License, or (at your option) any later version.
13 * This library is distributed in the hope that it will be useful, but
14 * WITHOUT ANY WARRANTY; without even the implied warranty of
15 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
16 * Lesser General Public License for more details.
18 * You should have received a copy of the GNU Lesser General Public
19 * License along with this library; if not, write to the Free Software
20 * Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA
25 #include "common/dbus-error.h"
26 #include "common/log.h"
27 #include "dbus-manager.h"
28 #include "dbus-service.h"
31 G_DEFINE_TYPE (MsgPortManager, msgport_manager, G_TYPE_OBJECT)
33 #define MSGPORT_MANAGER_GET_PRIV(obj) \
34 G_TYPE_INSTANCE_GET_PRIVATE ((obj), MSGPORT_TYPE_MANAGER, MsgPortManagerPrivate)
36 struct _MsgPortManagerPrivate {
38 * Key : guint - Id of the service
39 * Value : MsgPortDbusService * (transfe full)
41 GHashTable *service_cache; /* {service_id,MsgPortDbusService} */
44 * Holds services owned by a client
45 * Key : MsgPortDbusManager *
46 * Value : GList<MsgPortDbusService *> (tranfer none)
48 GHashTable *owner_service_map; /* {MsgPortDbusManager*,GList[MsgPortDbusService]} */
52 _manager_finalize (GObject *self)
54 //MsgPortManager *manager = MSGPORT_MANAGER (self);
56 G_OBJECT_CLASS (msgport_manager_parent_class)->finalize (self);
60 _manager_dispose (GObject *self)
62 MsgPortManager *manager = MSGPORT_MANAGER (self);
64 g_hash_table_unref (manager->priv->owner_service_map);
65 manager->priv->owner_service_map = NULL;
67 g_hash_table_unref (manager->priv->service_cache);
68 manager->priv->service_cache = NULL;
70 G_OBJECT_CLASS (msgport_manager_parent_class)->dispose (self);
74 msgport_manager_init (MsgPortManager *self)
76 MsgPortManagerPrivate *priv = MSGPORT_MANAGER_GET_PRIV (self);
78 priv->service_cache = g_hash_table_new_full (
79 g_direct_hash, g_direct_equal, NULL, g_object_unref);
80 priv->owner_service_map = g_hash_table_new_full (
81 g_direct_hash, g_direct_equal,
82 NULL, (GDestroyNotify) g_list_free);
88 msgport_manager_class_init (MsgPortManagerClass *klass)
90 GObjectClass *gklass = G_OBJECT_CLASS(klass);
92 g_type_class_add_private (klass, sizeof(MsgPortManagerPrivate));
94 gklass->finalize = _manager_finalize;
95 gklass->dispose = _manager_dispose;
99 msgport_manager_new ()
101 static GObject *manager = NULL;
104 manager = g_object_new (MSGPORT_TYPE_MANAGER, NULL);
105 g_object_add_weak_pointer (manager, (gpointer *)&manager);
107 return MSGPORT_MANAGER (manager);
110 return MSGPORT_MANAGER (g_object_ref (manager));
114 msgport_manager_register_service (
115 MsgPortManager *manager,
116 MsgPortDbusManager *owner,
117 const gchar *port_name,
121 GList *service_list = NULL; /* services list owned by a client */
122 gboolean list_was_empty = TRUE;
123 MsgPortDbusService *dbus_service = NULL;
125 msgport_return_val_if_fail_with_error (manager && MSGPORT_IS_MANAGER (manager), NULL, error);
126 msgport_return_val_if_fail_with_error (owner && MSGPORT_IS_DBUS_MANAGER (owner), NULL, error);
127 msgport_return_val_if_fail_with_error (port_name && port_name[0], NULL, error);
129 if ((service_list = (GList *)g_hash_table_lookup (manager->priv->owner_service_map, owner)) != NULL) {
132 for (list = service_list; list != NULL; list = list->next) {
133 MsgPortDbusService *dbus_service = (MsgPortDbusService *)list->data;
135 if ( !g_strcmp0 (port_name, msgport_dbus_service_get_port_name (dbus_service)) &&
136 msgport_dbus_service_get_is_trusted (dbus_service) == is_trusted ) {
138 const gchar *app_id = msgport_dbus_service_get_app_id (dbus_service);
139 *error = msgport_error_port_existing_new (app_id, port_name);
146 dbus_service = msgport_dbus_service_new (owner, port_name, is_trusted, error);
150 /* cache newly created service */
151 g_hash_table_insert (manager->priv->service_cache,
152 GINT_TO_POINTER (msgport_dbus_service_get_id (dbus_service)),
153 (gpointer)dbus_service);
155 list_was_empty = (service_list == NULL);
157 /* append to list of services */
158 service_list = g_list_append (service_list, dbus_service);
159 if (list_was_empty) {
160 g_hash_table_insert (manager->priv->owner_service_map, owner, service_list);
167 msgport_manager_get_service (
168 MsgPortManager *manager,
169 MsgPortDbusManager *owner,
170 const gchar *remote_port_name,
174 GList *service_list = NULL;
176 msgport_return_val_if_fail_with_error (manager && MSGPORT_IS_MANAGER (manager), NULL, error);
177 msgport_return_val_if_fail_with_error (owner && MSGPORT_IS_DBUS_MANAGER (owner), NULL, error);
178 msgport_return_val_if_fail_with_error (remote_port_name && remote_port_name[0], NULL, error);
180 service_list = g_hash_table_lookup (manager->priv->owner_service_map, owner);
181 while (service_list != NULL) {
182 MsgPortDbusService *dbus_service = MSGPORT_DBUS_SERVICE (service_list->data);
184 DBG ("Owner : %p - Port : %s, Is_trusted : %d", owner,
185 msgport_dbus_service_get_port_name (dbus_service),
186 msgport_dbus_service_get_is_trusted (dbus_service));
188 if ( !g_strcmp0 (remote_port_name, msgport_dbus_service_get_port_name (dbus_service)) &&
189 is_trusted == msgport_dbus_service_get_is_trusted (dbus_service)) {
190 return dbus_service ;
193 service_list = service_list->next;
196 if (error) *error = msgport_error_port_not_found (msgport_dbus_manager_get_app_id (owner), remote_port_name);
202 msgport_manager_get_service_by_id (
203 MsgPortManager *manager,
207 MsgPortDbusService *dbus_service = NULL;
209 msgport_return_val_if_fail_with_error (manager && MSGPORT_IS_MANAGER (manager), NULL, error);
210 msgport_return_val_if_fail_with_error (service_id != 0, NULL, error);
212 dbus_service = MSGPORT_DBUS_SERVICE (g_hash_table_lookup (
213 manager->priv->service_cache, GINT_TO_POINTER(service_id)));
219 _unref_dbus_manager_cb (gpointer data, gpointer user_data)
221 MsgPortDbusService *service = MSGPORT_DBUS_SERVICE (data);
222 MsgPortManager *manager = MSGPORT_MANAGER (user_data);
223 guint id = msgport_dbus_service_get_id (service);
225 DBG ("Unregistering service %s:%s(%d)",
226 msgport_dbus_manager_get_app_id (msgport_dbus_service_get_owner (service)),
227 msgport_dbus_service_get_port_name (service), id);
228 /* remove the service from id:service map,
229 * as its being unregisted */
230 g_hash_table_remove (manager->priv->service_cache, GINT_TO_POINTER(id));
234 * unregister a signle service for given service id
237 msgport_manager_unregister_service (
238 MsgPortManager *manager,
242 MsgPortDbusService *service = NULL;
243 MsgPortDbusManager *owner = NULL;
244 GList *service_list = NULL, *new_service_list = NULL;
246 msgport_return_val_if_fail_with_error (manager && MSGPORT_IS_MANAGER (manager), FALSE, error);
248 service = g_hash_table_lookup (manager->priv->service_cache, GINT_TO_POINTER (service_id));
251 if (error) *error = msgport_error_port_id_not_found_new (service_id);
255 owner = msgport_dbus_service_get_owner (service);
257 service_list = g_hash_table_lookup (manager->priv->owner_service_map, owner);
259 /* remove service from services list owned by the 'owner'*/
260 new_service_list = g_list_remove (service_list, service);
261 if (new_service_list != service_list) {
262 g_hash_table_steal (manager->priv->owner_service_map, owner);
263 /* update the new list on this owner */
264 g_hash_table_insert (manager->priv->owner_service_map, owner, new_service_list);
267 /* remove from the service_id:servcie table */
268 g_hash_table_remove (manager->priv->service_cache, GINT_TO_POINTER(service_id));
274 * unregister all the services owned by a client
277 msgport_manager_unregister_services (
278 MsgPortManager *manager,
279 MsgPortDbusManager *owner,
283 GList *service_list = NULL;
285 msgport_return_val_if_fail_with_error (manager && MSGPORT_IS_MANAGER (manager), FALSE, error);
287 /* fetch sevice list owned by the client */
288 service_list = g_hash_table_lookup (manager->priv->owner_service_map, owner);
290 DBG("no services found on client '%p'", owner);
294 /* remove all the service from the list */
295 g_list_foreach (service_list, _unref_dbus_manager_cb, manager);
296 g_hash_table_remove (manager->priv->owner_service_map, owner);