daemon initial implementation
[platform/core/appfw/message-port-dbus.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/log.h"
26 #include "dbus-manager.h"
27 #include "dbus-service.h"
28
29 G_DEFINE_TYPE (MsgPortManager, msgport_manager, G_TYPE_OBJECT)
30
31 #define MSGPORT_MANAGER_GET_PRIV(obj) \
32     G_TYPE_INSTANCE_GET_PRIVATE ((obj), MSGPORT_TYPE_MANAGER, MsgPortManagerPrivate)
33
34 struct _MsgPortManagerPrivate {
35     /*
36      * Key : const gchar * - object_path of the service
37      * Value : MsgPortDbusService * (transfe full)
38      */
39     GHashTable *path_service_map; /* {object_path,MsgPortDbusService} */
40
41     /*
42      * Holds services owned by a client 
43      * Key : MsgPortDbusManager *
44      * Value : GList<MsgPortDbusService *> (tranfer none)
45      */
46     GHashTable *owner_service_map; /* {app_id,GList[MsgPortDbusService]} */
47 };
48
49 static void
50 _manager_finalize (GObject *self)
51 {
52     //MsgPortManager *manager = MSGPORT_MANAGER (self);
53
54     G_OBJECT_CLASS (msgport_manager_parent_class)->finalize (self);
55 }
56
57 static void
58 _manager_dispose (GObject *self)
59 {
60     MsgPortManager *manager = MSGPORT_MANAGER (self);
61
62     g_hash_table_unref (manager->priv->owner_service_map);
63     manager->priv->owner_service_map = NULL;
64
65     g_hash_table_unref (manager->priv->path_service_map);
66     manager->priv->path_service_map = NULL;
67
68     G_OBJECT_CLASS (msgport_manager_parent_class)->dispose (self);
69 }
70
71 static void
72 msgport_manager_init (MsgPortManager *self)
73 {
74     MsgPortManagerPrivate *priv = MSGPORT_MANAGER_GET_PRIV (self);
75
76     priv->path_service_map = g_hash_table_new_full (
77                 g_str_hash, g_str_equal, NULL, g_object_unref);
78     priv->owner_service_map = g_hash_table_new_full (
79                 g_direct_hash, g_direct_equal, 
80                 NULL, (GDestroyNotify) g_list_free);
81
82     self->priv = priv;
83 }
84
85 static void
86 msgport_manager_class_init (MsgPortManagerClass *klass)
87 {
88     GObjectClass *gklass = G_OBJECT_CLASS(klass);
89
90     g_type_class_add_private (klass, sizeof(MsgPortManagerPrivate));
91
92     gklass->finalize = _manager_finalize;
93     gklass->dispose = _manager_dispose;
94 }
95
96 MsgPortManager *
97 msgport_manager_new ()
98 {
99     static GObject *manager = NULL;
100
101     if (!manager) {
102         manager =  g_object_new (MSGPORT_TYPE_MANAGER, NULL);
103         g_object_add_weak_pointer (manager, (gpointer *)&manager);
104
105         return MSGPORT_MANAGER (manager);
106     }
107
108     return MSGPORT_MANAGER (g_object_ref (manager));
109 }
110
111 MsgPortDbusService *
112 msgport_manager_register_service (
113     MsgPortManager     *manager,
114     MsgPortDbusManager *owner,
115     const gchar        *port_name,
116     gboolean            is_trusted,
117     GError            **error)
118 {
119     GList   *service_list  = NULL; /* services list by app_id */
120     gboolean was_empty = TRUE;
121     MsgPortDbusService *dbus_service = NULL;
122
123     g_return_val_if_fail (manager && MSGPORT_IS_MANAGER (manager), NULL);
124     g_return_val_if_fail (owner && MSGPORT_IS_DBUS_MANAGER (owner), NULL);
125
126     if ((service_list = (GList *)g_hash_table_lookup (manager->priv->owner_service_map, owner)) != NULL) {
127         GList *list = NULL;
128
129         for (list = service_list; list != NULL; list = list->next) {
130             MsgPortDbusService *dbus_service = (MsgPortDbusService *)list->data;
131
132             if ( !g_strcmp0 (port_name, msgport_dbus_service_get_port_name (dbus_service))) {
133                 /* FIXME: return EALREADY error */
134                 return NULL;
135             }
136         }
137     }
138
139     was_empty = (service_list == NULL);
140
141     dbus_service = msgport_dbus_service_new (owner, port_name, is_trusted);
142     /* cache newly created service */
143     g_hash_table_insert (manager->priv->path_service_map, 
144         (gpointer)msgport_dbus_service_get_object_path (dbus_service),
145         (gpointer)dbus_service);
146
147     /* append to list of services */
148     service_list = g_list_append (service_list, dbus_service);
149     if (was_empty) {
150         g_hash_table_insert (manager->priv->owner_service_map, owner, service_list);
151     }
152
153     return dbus_service;
154 }
155
156 MsgPortDbusService *
157 msgport_manager_get_service (
158     MsgPortManager      *manager,
159     MsgPortDbusManager  *owner,
160     const gchar         *remote_port_name,
161     gboolean             is_trusted,
162     GError             **error)
163 {
164     GList *service_list = NULL;
165     g_return_val_if_fail (manager && MSGPORT_IS_MANAGER (manager), NULL);
166
167     service_list = g_hash_table_lookup (manager->priv->owner_service_map, owner);
168     while (service_list != NULL) {
169         MsgPortDbusService *dbus_service = MSGPORT_DBUS_SERVICE (service_list->data);
170
171         if ( !g_strcmp0 (remote_port_name, msgport_dbus_service_get_port_name (dbus_service)) && 
172              is_trusted == msgport_dbus_service_get_is_trusted (dbus_service)) {
173             return dbus_service ;
174         }
175
176         service_list = service_list->next;
177     }
178
179     /* FIXME: return ENOTFOUND */
180     return NULL;
181 }
182
183 MsgPortDbusService *
184 msgport_manager_get_service_by_path (
185     MsgPortManager *manager,
186     const gchar    *service_object_path,
187     GError        **error)
188 {
189     MsgPortDbusService *dbus_service = NULL;
190     g_return_val_if_fail (manager && MSGPORT_IS_MANAGER (manager), NULL);
191
192     dbus_service = g_hash_table_lookup (manager->priv->path_service_map, service_object_path);
193
194     if (!dbus_service) {
195         /* FIXME: return ENOTFOUND error */
196     }
197
198     return dbus_service;
199 }
200
201 static void
202 _unref_dbus_manager_cb (gpointer data, gpointer user_data)
203 {
204     MsgPortDbusService *service = MSGPORT_DBUS_SERVICE (data);
205     MsgPortManager *manager = MSGPORT_MANAGER (user_data);
206     const gchar *object_path = NULL;
207     
208     g_assert (manager);
209     g_assert (service);
210     object_path = msgport_dbus_service_get_object_path (service);
211
212     DBG ("Unregistering service %s:%s(%s)", 
213         msgport_dbus_manager_get_app_id (msgport_dbus_service_get_owner (service)),
214         msgport_dbus_service_get_port_name (service),
215         object_path);
216     /* remove the service from object_path:service map,
217      * as its being unregisted */
218     g_hash_table_remove (manager->priv->path_service_map, object_path);
219 }
220
221 /*
222  * unregister all the services owned by a client
223  */
224 gboolean
225 msgport_manager_unregister_services (
226     MsgPortManager     *manager,
227     MsgPortDbusManager *owner)
228 {
229 DBG("{");
230     GList *service_list = NULL;
231     g_return_val_if_fail (manager && MSGPORT_IS_MANAGER (manager), FALSE);
232
233     /* fetch sevice list owned by the client */
234     service_list = g_hash_table_lookup (manager->priv->owner_service_map, owner);
235     if (!service_list) {
236         DBG("   No service found");
237         return FALSE;
238     }
239
240     /* remove all the service from the list */
241     g_list_foreach (service_list, _unref_dbus_manager_cb, manager);
242     g_hash_table_remove (manager->priv->owner_service_map, owner);
243 DBG("}");
244     return TRUE;
245 }
246
247 /*
248  * unregister a signle service for given object path
249  */
250 gboolean
251 msgport_manager_unregister_service (
252     MsgPortManager *manager,
253     const gchar *service_object_path)
254 {
255     MsgPortDbusService *service = NULL;
256     MsgPortDbusManager *owner = NULL;
257     GList *service_list = NULL, *new_service_list = NULL;
258     g_return_val_if_fail (manager && MSGPORT_IS_MANAGER (manager), FALSE);
259 DBG ("{");
260     service = g_hash_table_lookup (manager->priv->path_service_map, service_object_path);
261
262     if (!service) return FALSE;
263
264     owner = msgport_dbus_service_get_owner (service);
265
266     service_list = g_hash_table_lookup (manager->priv->owner_service_map, owner);
267
268     /* remove service from services list owned by the 'owner'*/
269     new_service_list = g_list_remove (service_list, service);
270     if (new_service_list != service_list) {
271         g_hash_table_steal (manager->priv->owner_service_map, owner);
272         /* update the new list on this owner */
273         g_hash_table_insert (manager->priv->owner_service_map, owner, new_service_list);
274     }
275
276     /* remove from the object_path:servcie table */
277     g_hash_table_remove (manager->priv->path_service_map, service_object_path);
278 DBG("}");
279     return TRUE;
280 }
281