daemon: fixed issue in registring more than one ports by a client.
[profile/ivi/message-port.git] / daemon / manager.c
1 /* vi: set et sw=4 ts=4 cino=t0,(0: */
2 /* -*- Mode: C; indent-tabs-mode: nil; c-basic-offset: 4 -*- */
3 /*
4  * Copyright (C) 2013 Intel Corporation.
5  *
6  * Contact: Amarnath Valluri <amarnath.valluri@linux.intel.com>
7  *
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.
12  *
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.
17  *
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
21  * 02110-1301 USA
22  */
23
24 #include "manager.h"
25 #include "common/dbus-error.h"
26 #include "common/log.h"
27 #include "dbus-manager.h"
28 #include "dbus-service.h"
29 #include "utils.h"
30
31 G_DEFINE_TYPE (MsgPortManager, msgport_manager, G_TYPE_OBJECT)
32
33 #define MSGPORT_MANAGER_GET_PRIV(obj) \
34     G_TYPE_INSTANCE_GET_PRIVATE ((obj), MSGPORT_TYPE_MANAGER, MsgPortManagerPrivate)
35
36 struct _MsgPortManagerPrivate {
37     /*
38      * Key :   guint - Id of the service
39      * Value : MsgPortDbusService * (transfe full)
40      */
41     GHashTable *service_cache; /* {service_id,MsgPortDbusService} */
42
43     /*
44      * Holds services owned by a client 
45      * Key : MsgPortDbusManager *
46      * Value : GList<MsgPortDbusService *> (tranfer none)
47      */
48     GHashTable *owner_service_map; /* {MsgPortDbusManager*,GList[MsgPortDbusService]} */
49 };
50
51 static void
52 _manager_finalize (GObject *self)
53 {
54     //MsgPortManager *manager = MSGPORT_MANAGER (self);
55
56     G_OBJECT_CLASS (msgport_manager_parent_class)->finalize (self);
57 }
58
59 static void
60 _manager_dispose (GObject *self)
61 {
62     MsgPortManager *manager = MSGPORT_MANAGER (self);
63
64     g_hash_table_unref (manager->priv->owner_service_map);
65     manager->priv->owner_service_map = NULL;
66
67     g_hash_table_unref (manager->priv->service_cache);
68     manager->priv->service_cache = NULL;
69
70     G_OBJECT_CLASS (msgport_manager_parent_class)->dispose (self);
71 }
72
73 static void
74 msgport_manager_init (MsgPortManager *self)
75 {
76     MsgPortManagerPrivate *priv = MSGPORT_MANAGER_GET_PRIV (self);
77
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);
83
84     self->priv = priv;
85 }
86
87 static void
88 msgport_manager_class_init (MsgPortManagerClass *klass)
89 {
90     GObjectClass *gklass = G_OBJECT_CLASS(klass);
91
92     g_type_class_add_private (klass, sizeof(MsgPortManagerPrivate));
93
94     gklass->finalize = _manager_finalize;
95     gklass->dispose = _manager_dispose;
96 }
97
98 MsgPortManager *
99 msgport_manager_new ()
100 {
101     static GObject *manager = NULL;
102
103     if (!manager) {
104         manager =  g_object_new (MSGPORT_TYPE_MANAGER, NULL);
105         g_object_add_weak_pointer (manager, (gpointer *)&manager);
106
107         return MSGPORT_MANAGER (manager);
108     }
109
110     return MSGPORT_MANAGER (g_object_ref (manager));
111 }
112
113 /*
114  * It returns the serice pointer, if found with given owner, port_name and is_trusted 
115  * It assues the given arguments are valid.
116  */
117 MsgPortDbusService *
118 _manager_get_service_internal (
119     MsgPortManager     *manager,
120     MsgPortDbusManager *owner,
121     const gchar        *port_name,
122     gboolean            is_trusted)
123 {
124     GList *service_list = g_hash_table_lookup (manager->priv->owner_service_map, owner);
125     while (service_list != NULL) {
126         MsgPortDbusService *dbus_service = MSGPORT_DBUS_SERVICE (service_list->data);
127
128         DBG ("Owner : %p - Port : %s, Is_trusted : %d", owner,
129             msgport_dbus_service_get_port_name (dbus_service),
130             msgport_dbus_service_get_is_trusted (dbus_service));
131
132         if ( !g_strcmp0 (port_name, msgport_dbus_service_get_port_name (dbus_service)) && 
133              is_trusted == msgport_dbus_service_get_is_trusted (dbus_service)) {
134             return dbus_service ;
135         }
136
137         service_list = service_list->next;
138     }
139
140     return NULL;
141 }
142
143 MsgPortDbusService *
144 msgport_manager_register_service (
145     MsgPortManager     *manager,
146     MsgPortDbusManager *owner,
147     const gchar        *port_name,
148     gboolean            is_trusted,
149     GError            **error)
150 {
151     GList   *service_list  = NULL; /* services list owned by a client */
152     gboolean list_was_empty = TRUE;
153     MsgPortDbusService *dbus_service = NULL;
154
155     msgport_return_val_if_fail_with_error (manager && MSGPORT_IS_MANAGER (manager), NULL, error);
156     msgport_return_val_if_fail_with_error (owner && MSGPORT_IS_DBUS_MANAGER (owner), NULL, error);
157     msgport_return_val_if_fail_with_error (port_name && port_name[0], NULL, error);
158
159     /* check if port already existing with given params */
160     dbus_service = _manager_get_service_internal (manager, owner, port_name, is_trusted);
161     if (dbus_service != NULL)
162         return dbus_service;
163
164     /* create  new port/service */
165     dbus_service = msgport_dbus_service_new (owner, port_name, is_trusted, error);
166     if (!dbus_service) {
167         return NULL;
168     }
169     /* cache newly created service */
170     g_hash_table_insert (manager->priv->service_cache, 
171         GINT_TO_POINTER (msgport_dbus_service_get_id (dbus_service)),
172         (gpointer)dbus_service);
173
174     service_list = g_hash_table_lookup (manager->priv->owner_service_map, owner);
175     list_was_empty = (service_list == NULL);
176
177    /* append to list of services */
178     service_list = g_list_append (service_list, dbus_service);
179     if (list_was_empty) {
180         g_hash_table_insert (manager->priv->owner_service_map, owner, service_list);
181     }
182
183     return dbus_service;
184 }
185
186
187 MsgPortDbusService *
188 msgport_manager_get_service (
189     MsgPortManager      *manager,
190     MsgPortDbusManager  *owner,
191     const gchar         *port_name,
192     gboolean             is_trusted,
193     GError             **error)
194 {
195     MsgPortDbusService *service = NULL;
196
197     msgport_return_val_if_fail_with_error (manager && MSGPORT_IS_MANAGER (manager), NULL, error);
198     msgport_return_val_if_fail_with_error (owner && MSGPORT_IS_DBUS_MANAGER (owner), NULL, error);
199     msgport_return_val_if_fail_with_error (port_name && port_name[0], NULL, error);
200
201     service = _manager_get_service_internal (manager, owner, port_name, is_trusted);
202
203     if (!service && error) 
204         *error = msgport_error_port_not_found (msgport_dbus_manager_get_app_id (owner), port_name);
205
206     return service;
207 }
208
209 MsgPortDbusService *
210 msgport_manager_get_service_by_id (
211     MsgPortManager *manager,
212     guint           service_id,
213     GError        **error)
214 {
215     MsgPortDbusService *dbus_service = NULL;
216
217     msgport_return_val_if_fail_with_error (manager && MSGPORT_IS_MANAGER (manager), NULL, error);
218     msgport_return_val_if_fail_with_error (service_id != 0, NULL, error);
219
220     dbus_service = MSGPORT_DBUS_SERVICE (g_hash_table_lookup (
221             manager->priv->service_cache, GINT_TO_POINTER(service_id)));
222
223     return dbus_service;
224 }
225
226 static void
227 _manager_unref_dbus_manager_cb (gpointer data, gpointer user_data)
228 {
229     MsgPortManager *manager = MSGPORT_MANAGER (user_data);
230     MsgPortDbusService *service = MSGPORT_DBUS_SERVICE (data);
231     guint id = msgport_dbus_service_get_id (service);
232
233 #ifdef ENABLE_DEBUG
234     DBG ("Unregistering service %s:%s(%d)", 
235         msgport_dbus_manager_get_app_id (msgport_dbus_service_get_owner (service)),
236         msgport_dbus_service_get_port_name (service), id);
237 #endif
238     /* remove the service from id:service map,
239      * as its being unregisted */
240     g_hash_table_remove (manager->priv->service_cache, GINT_TO_POINTER(id));
241 }
242
243 /*
244  * unregister a signle service for given service id
245  */
246 gboolean
247 msgport_manager_unregister_service (
248     MsgPortManager *manager,
249     gint            service_id,
250     GError        **error)
251 {
252     MsgPortDbusService *service = NULL;
253     MsgPortDbusManager *owner = NULL;
254     GList *service_list = NULL, *new_service_list = NULL;
255
256     msgport_return_val_if_fail_with_error (manager && MSGPORT_IS_MANAGER (manager), FALSE, error);
257
258     service = g_hash_table_lookup (manager->priv->service_cache, GINT_TO_POINTER (service_id));
259
260     if (!service) {
261         if (error) *error = msgport_error_port_id_not_found_new (service_id);
262         return FALSE;
263     }
264
265     owner = msgport_dbus_service_get_owner (service);
266
267     service_list = g_hash_table_lookup (manager->priv->owner_service_map, owner);
268
269     /* remove service from services list owned by the 'owner'*/
270     new_service_list = g_list_remove (service_list, service);
271     if (new_service_list != service_list) {
272         g_hash_table_steal (manager->priv->owner_service_map, owner);
273         /* update the new list on this owner */
274         g_hash_table_insert (manager->priv->owner_service_map, owner, new_service_list);
275     }
276
277     /* remove from the service_id:servcie table */
278     g_hash_table_remove (manager->priv->service_cache, GINT_TO_POINTER(service_id));
279
280     return TRUE;
281 }
282
283 /*
284  * unregister all the services owned by a client
285  */
286 gboolean
287 msgport_manager_unregister_services (
288     MsgPortManager     *manager,
289     MsgPortDbusManager *owner,
290     GError            **error)
291 {
292
293     GList *service_list = NULL;
294
295     msgport_return_val_if_fail_with_error (manager && MSGPORT_IS_MANAGER (manager), FALSE, error);
296
297     /* fetch sevice list owned by the client */
298     service_list = g_hash_table_lookup (manager->priv->owner_service_map, owner);
299     if (!service_list) {
300         DBG("no services found on client '%p'", owner);
301         return TRUE;
302     }
303
304     /* remove all the service from the list */
305     g_list_foreach (service_list, _manager_unref_dbus_manager_cb, manager);
306     g_hash_table_remove (manager->priv->owner_service_map, owner);
307
308     return TRUE;
309 }
310