Fix missing update of signal strength from scan results
[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_update_network(struct connman_network *network)
157 {
158         DBG("network %p", network);
159
160         __connman_service_update_from_network(network);
161
162         return 0;
163 }
164
165 int __connman_profile_remove_network(struct connman_network *network)
166 {
167         DBG("network %p", network);
168
169         __connman_service_remove_from_network(network);
170
171         return 0;
172 }
173
174 void __connman_profile_list(DBusMessageIter *iter)
175 {
176         const char *path = __connman_profile_active_path();
177
178         DBG("");
179
180         dbus_message_iter_append_basic(iter, DBUS_TYPE_OBJECT_PATH, &path);
181 }
182
183 static DBusMessage *get_properties(DBusConnection *conn,
184                                         DBusMessage *msg, void *data)
185 {
186         const char *name = "Default";
187         DBusMessage *reply;
188         DBusMessageIter array, dict, entry;
189
190         DBG("conn %p", conn);
191
192         reply = dbus_message_new_method_return(msg);
193         if (reply == NULL)
194                 return NULL;
195
196         dbus_message_iter_init_append(reply, &array);
197
198         dbus_message_iter_open_container(&array, DBUS_TYPE_ARRAY,
199                         DBUS_DICT_ENTRY_BEGIN_CHAR_AS_STRING
200                         DBUS_TYPE_STRING_AS_STRING DBUS_TYPE_VARIANT_AS_STRING
201                         DBUS_DICT_ENTRY_END_CHAR_AS_STRING, &dict);
202
203         connman_dbus_dict_append_variant(&dict, "Name",
204                                                 DBUS_TYPE_STRING, &name);
205
206         dbus_message_iter_open_container(&dict, DBUS_TYPE_DICT_ENTRY,
207                                                                 NULL, &entry);
208         append_services(&entry);
209         dbus_message_iter_close_container(&dict, &entry);
210
211         dbus_message_iter_close_container(&array, &dict);
212
213         return reply;
214 }
215
216 static GDBusMethodTable profile_methods[] = {
217         { "GetProperties", "", "a{sv}", get_properties },
218         { },
219 };
220
221 static GDBusSignalTable profile_signals[] = {
222         { "PropertyChanged", "sv" },
223         { },
224 };
225
226 int __connman_profile_init(DBusConnection *conn)
227 {
228         const char *path = __connman_profile_active_path();
229
230         DBG("conn %p", conn);
231
232         connection = dbus_connection_ref(conn);
233         if (connection == NULL)
234                 return -1;
235
236         g_dbus_register_interface(connection, path,
237                                         CONNMAN_PROFILE_INTERFACE,
238                                         profile_methods, profile_signals,
239                                                         NULL, NULL, NULL);
240
241         return 0;
242 }
243
244 void __connman_profile_cleanup(void)
245 {
246         const char *path = __connman_profile_active_path();
247
248         DBG("conn %p", connection);
249
250         g_dbus_unregister_interface(connection, path,
251                                                 CONNMAN_PROFILE_INTERFACE);
252
253         if (connection == NULL)
254                 return;
255
256         dbus_connection_unref(connection);
257 }