One second timeout for service list update delays is enough
[framework/connectivity/connman.git] / src / profile.c
1 /*
2  *
3  *  Connection Manager
4  *
5  *  Copyright (C) 2007-2009  Intel Corporation. All rights reserved.
6  *
7  *  This program is free software; you can redistribute it and/or modify
8  *  it under the terms of the GNU General Public License version 2 as
9  *  published by the Free Software Foundation.
10  *
11  *  This program is distributed in the hope that it will be useful,
12  *  but WITHOUT ANY WARRANTY; without even the implied warranty of
13  *  MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
14  *  GNU General Public License for more details.
15  *
16  *  You should have received a copy of the GNU General Public License
17  *  along with this program; if not, write to the Free Software
18  *  Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA  02110-1301  USA
19  *
20  */
21
22 #ifdef HAVE_CONFIG_H
23 #include <config.h>
24 #endif
25
26 #include <glib.h>
27 #include <gdbus.h>
28
29 #include "connman.h"
30
31 #define PROFILE_DEFAULT_IDENT  "default"
32
33 static DBusConnection *connection = NULL;
34
35 const char *__connman_profile_active_ident(void)
36 {
37         DBG("");
38
39         return PROFILE_DEFAULT_IDENT;
40 }
41
42 const char *__connman_profile_active_path(void)
43 {
44         DBG("");
45
46         return "/profile/" PROFILE_DEFAULT_IDENT;
47 }
48
49 static void append_services(DBusMessageIter *entry)
50 {
51         DBusMessageIter value, iter;
52         const char *key = "Services";
53
54         dbus_message_iter_append_basic(entry, DBUS_TYPE_STRING, &key);
55
56         dbus_message_iter_open_container(entry, DBUS_TYPE_VARIANT,
57                 DBUS_TYPE_ARRAY_AS_STRING DBUS_TYPE_OBJECT_PATH_AS_STRING,
58                                                                 &value);
59
60         dbus_message_iter_open_container(&value, DBUS_TYPE_ARRAY,
61                                 DBUS_TYPE_OBJECT_PATH_AS_STRING, &iter);
62         __connman_service_list(&iter);
63         dbus_message_iter_close_container(&value, &iter);
64
65         dbus_message_iter_close_container(entry, &value);
66 }
67
68 static guint changed_timeout = 0;
69
70 static gboolean services_changed(gpointer user_data)
71 {
72         const char *path = __connman_profile_active_path();
73         DBusMessage *signal;
74         DBusMessageIter entry;
75
76         changed_timeout = 0;
77
78         signal = dbus_message_new_signal(path,
79                                 CONNMAN_PROFILE_INTERFACE, "PropertyChanged");
80         if (signal == NULL)
81                 return FALSE;
82
83         dbus_message_iter_init_append(signal, &entry);
84         append_services(&entry);
85         g_dbus_send_message(connection, signal);
86
87         signal = dbus_message_new_signal(CONNMAN_MANAGER_PATH,
88                                 CONNMAN_MANAGER_INTERFACE, "PropertyChanged");
89         if (signal == NULL)
90                 return FALSE;
91
92         dbus_message_iter_init_append(signal, &entry);
93         append_services(&entry);
94         g_dbus_send_message(connection, signal);
95
96         return FALSE;
97 }
98
99 void __connman_profile_changed(gboolean delayed)
100 {
101         DBG("");
102
103         if (changed_timeout > 0) {
104                 g_source_remove(changed_timeout);
105                 changed_timeout = 0;
106         }
107
108         if (__connman_connection_update_gateway() == TRUE) {
109                 services_changed(NULL);
110                 return;
111         }
112
113         if (delayed == FALSE) {
114                 services_changed(NULL);
115                 return;
116         }
117
118         changed_timeout = g_timeout_add_seconds(1, services_changed, NULL);
119 }
120
121 int __connman_profile_add_device(struct connman_device *device)
122 {
123         struct connman_service *service;
124
125         DBG("device %p", device);
126
127         service = __connman_service_create_from_device(device);
128         if (service == NULL)
129                 return -EINVAL;
130
131         return 0;
132 }
133
134 int __connman_profile_remove_device(struct connman_device *device)
135 {
136         DBG("device %p", device);
137
138         __connman_service_remove_from_device(device);
139
140         return 0;
141 }
142
143 int __connman_profile_add_network(struct connman_network *network)
144 {
145         struct connman_service *service;
146
147         DBG("network %p", network);
148
149         service = __connman_service_create_from_network(network);
150         if (service == NULL)
151                 return -EINVAL;
152
153         return 0;
154 }
155
156 int __connman_profile_remove_network(struct connman_network *network)
157 {
158         DBG("network %p", network);
159
160         __connman_service_remove_from_network(network);
161
162         return 0;
163 }
164
165 void __connman_profile_list(DBusMessageIter *iter)
166 {
167         const char *path = __connman_profile_active_path();
168
169         DBG("");
170
171         dbus_message_iter_append_basic(iter, DBUS_TYPE_OBJECT_PATH, &path);
172 }
173
174 static DBusMessage *get_properties(DBusConnection *conn,
175                                         DBusMessage *msg, void *data)
176 {
177         const char *name = "Default";
178         DBusMessage *reply;
179         DBusMessageIter array, dict, entry;
180
181         DBG("conn %p", conn);
182
183         reply = dbus_message_new_method_return(msg);
184         if (reply == NULL)
185                 return NULL;
186
187         dbus_message_iter_init_append(reply, &array);
188
189         dbus_message_iter_open_container(&array, DBUS_TYPE_ARRAY,
190                         DBUS_DICT_ENTRY_BEGIN_CHAR_AS_STRING
191                         DBUS_TYPE_STRING_AS_STRING DBUS_TYPE_VARIANT_AS_STRING
192                         DBUS_DICT_ENTRY_END_CHAR_AS_STRING, &dict);
193
194         connman_dbus_dict_append_variant(&dict, "Name",
195                                                 DBUS_TYPE_STRING, &name);
196
197         dbus_message_iter_open_container(&dict, DBUS_TYPE_DICT_ENTRY,
198                                                                 NULL, &entry);
199         append_services(&entry);
200         dbus_message_iter_close_container(&dict, &entry);
201
202         dbus_message_iter_close_container(&array, &dict);
203
204         return reply;
205 }
206
207 static GDBusMethodTable profile_methods[] = {
208         { "GetProperties", "", "a{sv}", get_properties },
209         { },
210 };
211
212 static GDBusSignalTable profile_signals[] = {
213         { "PropertyChanged", "sv" },
214         { },
215 };
216
217 int __connman_profile_init(DBusConnection *conn)
218 {
219         const char *path = __connman_profile_active_path();
220
221         DBG("conn %p", conn);
222
223         connection = dbus_connection_ref(conn);
224         if (connection == NULL)
225                 return -1;
226
227         g_dbus_register_interface(connection, path,
228                                         CONNMAN_PROFILE_INTERFACE,
229                                         profile_methods, profile_signals,
230                                                         NULL, NULL, NULL);
231
232         return 0;
233 }
234
235 void __connman_profile_cleanup(void)
236 {
237         const char *path = __connman_profile_active_path();
238
239         DBG("conn %p", connection);
240
241         g_dbus_unregister_interface(connection, path,
242                                                 CONNMAN_PROFILE_INTERFACE);
243
244         if (connection == NULL)
245                 return;
246
247         dbus_connection_unref(connection);
248 }