Trigger auto-connect from service list changed signals
[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         __connman_service_auto_connect();
97
98         return FALSE;
99 }
100
101 void __connman_profile_changed(gboolean delayed)
102 {
103         DBG("");
104
105         if (changed_timeout > 0) {
106                 g_source_remove(changed_timeout);
107                 changed_timeout = 0;
108         }
109
110         if (__connman_connection_update_gateway() == TRUE) {
111                 services_changed(NULL);
112                 return;
113         }
114
115         if (delayed == FALSE) {
116                 services_changed(NULL);
117                 return;
118         }
119
120         changed_timeout = g_timeout_add_seconds(2, services_changed, NULL);
121 }
122
123 int __connman_profile_add_device(struct connman_device *device)
124 {
125         struct connman_service *service;
126
127         DBG("device %p", device);
128
129         service = __connman_service_create_from_device(device);
130         if (service == NULL)
131                 return -EINVAL;
132
133         return 0;
134 }
135
136 int __connman_profile_remove_device(struct connman_device *device)
137 {
138         struct connman_service *service;
139
140         DBG("device %p", device);
141
142         service = __connman_service_lookup_from_device(device);
143         if (service == NULL)
144                 return -EINVAL;
145
146         __connman_service_put(service);
147
148         return 0;
149 }
150
151 int __connman_profile_add_network(struct connman_network *network)
152 {
153         struct connman_service *service;
154
155         DBG("network %p", network);
156
157         service = __connman_service_create_from_network(network);
158         if (service == NULL)
159                 return -EINVAL;
160
161         return 0;
162 }
163
164 int __connman_profile_remove_network(struct connman_network *network)
165 {
166         struct connman_service *service;
167
168         DBG("network %p", network);
169
170         service = __connman_service_lookup_from_network(network);
171         if (service == NULL)
172                 return -EINVAL;
173
174         __connman_service_put(service);
175
176         return 0;
177 }
178
179 void __connman_profile_list(DBusMessageIter *iter)
180 {
181         const char *path = __connman_profile_active_path();
182
183         DBG("");
184
185         dbus_message_iter_append_basic(iter, DBUS_TYPE_OBJECT_PATH, &path);
186 }
187
188 static DBusMessage *get_properties(DBusConnection *conn,
189                                         DBusMessage *msg, void *data)
190 {
191         const char *name = "Default";
192         DBusMessage *reply;
193         DBusMessageIter array, dict, entry;
194
195         DBG("conn %p", conn);
196
197         reply = dbus_message_new_method_return(msg);
198         if (reply == NULL)
199                 return NULL;
200
201         dbus_message_iter_init_append(reply, &array);
202
203         dbus_message_iter_open_container(&array, DBUS_TYPE_ARRAY,
204                         DBUS_DICT_ENTRY_BEGIN_CHAR_AS_STRING
205                         DBUS_TYPE_STRING_AS_STRING DBUS_TYPE_VARIANT_AS_STRING
206                         DBUS_DICT_ENTRY_END_CHAR_AS_STRING, &dict);
207
208         connman_dbus_dict_append_variant(&dict, "Name",
209                                                 DBUS_TYPE_STRING, &name);
210
211         dbus_message_iter_open_container(&dict, DBUS_TYPE_DICT_ENTRY,
212                                                                 NULL, &entry);
213         append_services(&entry);
214         dbus_message_iter_close_container(&dict, &entry);
215
216         dbus_message_iter_close_container(&array, &dict);
217
218         return reply;
219 }
220
221 static GDBusMethodTable profile_methods[] = {
222         { "GetProperties", "", "a{sv}", get_properties },
223         { },
224 };
225
226 static GDBusSignalTable profile_signals[] = {
227         { "PropertyChanged", "sv" },
228         { },
229 };
230
231 int __connman_profile_init(DBusConnection *conn)
232 {
233         const char *path = __connman_profile_active_path();
234
235         DBG("conn %p", conn);
236
237         connection = dbus_connection_ref(conn);
238         if (connection == NULL)
239                 return -1;
240
241         g_dbus_register_interface(connection, path,
242                                         CONNMAN_PROFILE_INTERFACE,
243                                         profile_methods, profile_signals,
244                                                         NULL, NULL, NULL);
245
246         return 0;
247 }
248
249 void __connman_profile_cleanup(void)
250 {
251         const char *path = __connman_profile_active_path();
252
253         DBG("conn %p", connection);
254
255         g_dbus_unregister_interface(connection, path,
256                                                 CONNMAN_PROFILE_INTERFACE);
257
258         if (connection == NULL)
259                 return;
260
261         dbus_connection_unref(connection);
262 }