headers inclusion is a real mess...
[framework/uifw/edbus.git] / src / lib / connman / e_connman.c
1 #include "e_connman_private.h"
2 #include <stdlib.h>
3 #include <string.h>
4
5 static E_DBus_Signal_Handler *cb_name_owner_changed = NULL;
6 static DBusPendingCall *pending_get_name_owner = NULL;
7 static unsigned int init_count = 0;
8 static char *unique_name = NULL;
9
10 static const char bus_name[] = "org.moblin.connman";
11 static const char fdo_bus_name[] = "org.freedesktop.DBus";
12 static const char fdo_interface[] = "org.freedesktop.DBus";
13 static const char fdo_path[] = "/org/freedesktop/DBus";
14
15 E_DBus_Connection *e_connman_conn = NULL;
16
17 EAPI int E_CONNMAN_EVENT_MANAGER_IN = 0;
18 EAPI int E_CONNMAN_EVENT_MANAGER_OUT = 0;
19 EAPI int E_CONNMAN_EVENT_ELEMENT_ADD = 0;
20 EAPI int E_CONNMAN_EVENT_ELEMENT_DEL = 0;
21 EAPI int E_CONNMAN_EVENT_ELEMENT_UPDATED = 0;
22
23 const char *e_connman_iface_manager = NULL;
24 const char *e_connman_iface_network = NULL;
25 const char *e_connman_iface_profile = NULL;
26 const char *e_connman_iface_service = NULL;
27 const char *e_connman_iface_device = NULL;
28 const char *e_connman_iface_connection = NULL;
29
30 const char *e_connman_prop_available = NULL;
31 const char *e_connman_prop_connected = NULL;
32 const char *e_connman_prop_connections = NULL;
33 const char *e_connman_prop_default = NULL;
34 const char *e_connman_prop_device = NULL;
35 const char *e_connman_prop_devices = NULL;
36 const char *e_connman_prop_interface = NULL;
37 const char *e_connman_prop_ipv4_address = NULL;
38 const char *e_connman_prop_ipv4_method = NULL;
39 const char *e_connman_prop_name = NULL;
40 const char *e_connman_prop_network = NULL;
41 const char *e_connman_prop_networks = NULL;
42 const char *e_connman_prop_offline_mode = NULL;
43 const char *e_connman_prop_policy = NULL;
44 const char *e_connman_prop_powered = NULL;
45 const char *e_connman_prop_priority = NULL;
46 const char *e_connman_prop_profiles = NULL;
47 const char *e_connman_prop_remember = NULL;
48 const char *e_connman_prop_scan_interval = NULL;
49 const char *e_connman_prop_scanning = NULL;
50 const char *e_connman_prop_state = NULL;
51 const char *e_connman_prop_strengh = NULL;
52 const char *e_connman_prop_type = NULL;
53 const char *e_connman_prop_wifi_mode = NULL;
54 const char *e_connman_prop_wifi_passphrase = NULL;
55 const char *e_connman_prop_wifi_security = NULL;
56 const char *e_connman_prop_wifi_ssid = NULL;
57
58 int _e_dbus_connman_log_dom = -1;
59
60 const char *
61 e_connman_system_bus_name_get(void)
62 {
63    return unique_name ? unique_name : bus_name;
64 }
65
66
67 /***********************************************************************
68  * Manager
69  ***********************************************************************/
70
71 /**
72  * Synchronize elements with server.
73  *
74  * This will call Manager.GetProperties() on server, retrieve properties
75  * and some element paths and then request their properties.
76  *
77  * This call will add events E_CONNMAN_EVENT_ELEMENT_ADD and
78  * E_CONNMAN_EVENT_ELEMENT_UPDATED to the main loop.
79  *
80  * This will not remove stale elements.
81  *
82  * @return 1 on success, 0 otherwise.
83  */
84 bool
85 e_connman_manager_sync_elements(void)
86 {
87    E_Connman_Element *manager;
88
89    if (!unique_name)
90      return FALSE;
91    manager = e_connman_element_register(manager_path, e_connman_iface_manager);
92    if (manager)
93      e_connman_element_properties_sync(manager);
94    else
95      return FALSE;
96
97    DBG("sync_manager: %s (%s)", unique_name, bus_name);
98
99    return TRUE;
100 }
101
102 static void
103 _e_connman_system_name_owner_exit(void)
104 {
105    e_connman_manager_clear_elements();
106    ecore_event_add(E_CONNMAN_EVENT_MANAGER_OUT, NULL, NULL, NULL);
107
108    free(unique_name);
109    unique_name = NULL;
110 }
111
112 static void
113 _e_connman_system_name_owner_enter(const char *uid)
114 {
115    DBG("E-Dbus connman: enter connman at %s (old was %s)", uid, unique_name);
116    if (unique_name && strcmp(unique_name, uid) == 0)
117      {
118         DBG("E-Dbus connman: same unique_name for connman, ignore.");
119         return;
120      }
121
122    if (unique_name)
123      _e_connman_system_name_owner_exit();
124
125    unique_name = strdup(uid);
126
127    ecore_event_add(E_CONNMAN_EVENT_MANAGER_IN, NULL, NULL, NULL);
128    e_connman_manager_sync_elements();
129 }
130
131 static void
132 _e_connman_system_name_owner_changed(void *data, DBusMessage *msg)
133 {
134    DBusError err;
135    const char *name, *from, *to;
136
137    dbus_error_init(&err);
138    if (!dbus_message_get_args(msg, &err,
139                               DBUS_TYPE_STRING, &name,
140                               DBUS_TYPE_STRING, &from,
141                               DBUS_TYPE_STRING, &to,
142                               DBUS_TYPE_INVALID))
143      {
144         ERR("E-Dbus connman: could not get NameOwnerChanged arguments: %s: %s",
145             err.name, err.message);
146         dbus_error_free(&err);
147         return;
148      }
149
150    if (strcmp(name, bus_name) != 0)
151      return;
152
153    DBG("E-Dbus connman: NameOwnerChanged from=[%s] to=[%s]", from, to);
154
155    if (from[0] == '\0' && to[0] != '\0')
156      _e_connman_system_name_owner_enter(to);
157    else if (from[0] != '\0' && to[0] == '\0')
158      {
159         DBG("E-Dbus connman: exit connman at %s", from);
160         if (strcmp(unique_name, from) != 0)
161           DBG("E-Dbus connman: %s was not the known name %s, ignored.", from, unique_name);
162         else
163           _e_connman_system_name_owner_exit();
164      }
165    else
166      DBG("E-Dbus connman: unknow change from %s to %s", from, to);
167 }
168
169 static void
170 _e_connman_get_name_owner(void *data, DBusMessage *msg, DBusError *err)
171 {
172    DBusMessageIter itr;
173    int t;
174    const char *uid;
175
176    pending_get_name_owner = NULL;
177
178    if (!_dbus_callback_check_and_init(msg, &itr, err))
179      return;
180
181    t = dbus_message_iter_get_arg_type(&itr);
182    if (!_dbus_iter_type_check(t, DBUS_TYPE_STRING))
183      return;
184
185    dbus_message_iter_get_basic(&itr, &uid);
186    if (!uid)
187      {
188         ERR("E-Dbus connman: no name owner!");
189         return;
190      }
191
192    _e_connman_system_name_owner_enter(uid);
193    return;
194 }
195
196 /**
197  * Initialize E Connection Manager (E_Connman) system.
198  *
199  * This will connect and watch org.moblin.connman.Manager and Element
200  * events and translate to Ecore main loop events, also provide a
201  * proxy for method invocation on server.
202  *
203  * Interesting events are:
204  *   - E_CONNMAN_EVENT_MANAGER_IN: issued when connman is avaiable.
205  *   - E_CONNMAN_EVENT_MANAGER_OUT: issued when connman connection is lost.
206  *   - E_CONNMAN_EVENT_ELEMENT_ADD: element was added.
207  *   - E_CONNMAN_EVENT_ELEMENT_DEL: element was deleted.
208  *   - E_CONNMAN_EVENT_ELEMENT_UPDATED: element was updated (properties
209  *     or state changed).
210  *
211  * Manager IN/OUT events do not provide any event information, just
212  * tells you that system is usable or not. After manager is out, all
213  * elements will be removed, so after this event do not use the system anymore.
214  *
215  * Element events will give you an element object. After DEL event callback
216  * returns, that element will not be valid anymore.
217  */
218 unsigned int
219 e_connman_system_init(E_DBus_Connection *edbus_conn)
220 {
221    init_count++;
222
223    if (init_count > 1)
224      return init_count;
225    
226    _e_dbus_connman_log_dom = eina_log_domain_register("e_dbus_connman",EINA_LOG_DEFAULT_COLOR);
227
228    if(_e_dbus_connman_log_dom < 0) 
229      {
230        ERR("E-Dbus connman error : impossible to create a log domain for edbus_connman module");
231        return -1;
232      }
233
234    if (E_CONNMAN_EVENT_MANAGER_IN == 0)
235      E_CONNMAN_EVENT_MANAGER_IN = ecore_event_type_new();
236    if (E_CONNMAN_EVENT_MANAGER_OUT == 0)
237      E_CONNMAN_EVENT_MANAGER_OUT = ecore_event_type_new();
238    if (E_CONNMAN_EVENT_ELEMENT_ADD == 0)
239      E_CONNMAN_EVENT_ELEMENT_ADD = ecore_event_type_new();
240    if (E_CONNMAN_EVENT_ELEMENT_DEL == 0)
241      E_CONNMAN_EVENT_ELEMENT_DEL = ecore_event_type_new();
242    if (E_CONNMAN_EVENT_ELEMENT_UPDATED == 0)
243      E_CONNMAN_EVENT_ELEMENT_UPDATED = ecore_event_type_new();
244
245    if (e_connman_iface_manager == NULL)
246      e_connman_iface_manager = eina_stringshare_add("org.moblin.connman.Manager");
247    if (e_connman_iface_network == NULL)
248      e_connman_iface_network = eina_stringshare_add("org.moblin.connman.Network");
249    if (e_connman_iface_profile == NULL)
250      e_connman_iface_profile = eina_stringshare_add("org.moblin.connman.Profile");
251    if (e_connman_iface_service == NULL)
252      e_connman_iface_service = eina_stringshare_add("org.moblin.connman.Service");
253    if (e_connman_iface_device == NULL)
254      e_connman_iface_device = eina_stringshare_add("org.moblin.connman.Device");
255    if (e_connman_iface_connection == NULL)
256      e_connman_iface_connection = eina_stringshare_add("org.moblin.connman.Connection");
257
258    if (e_connman_prop_available == NULL)
259      e_connman_prop_available = eina_stringshare_add("Available");
260    if (e_connman_prop_connected == NULL)
261      e_connman_prop_connected = eina_stringshare_add("Connected");
262    if (e_connman_prop_connections == NULL)
263      e_connman_prop_connections = eina_stringshare_add("Connections");
264    if (e_connman_prop_default == NULL)
265      e_connman_prop_default = eina_stringshare_add("Default");
266    if (e_connman_prop_device == NULL)
267      e_connman_prop_device = eina_stringshare_add("Device");
268    if (e_connman_prop_devices == NULL)
269      e_connman_prop_devices = eina_stringshare_add("Devices");
270    if (e_connman_prop_interface == NULL)
271      e_connman_prop_interface = eina_stringshare_add("Interface");
272    if (e_connman_prop_ipv4_address == NULL)
273      e_connman_prop_ipv4_address = eina_stringshare_add("IPv4.Address");
274    if (e_connman_prop_ipv4_method == NULL)
275      e_connman_prop_ipv4_method = eina_stringshare_add("IPv4.Method");
276    if (e_connman_prop_name == NULL)
277      e_connman_prop_name = eina_stringshare_add("Name");
278    if (e_connman_prop_network == NULL)
279      e_connman_prop_network = eina_stringshare_add("Network");
280    if (e_connman_prop_networks == NULL)
281      e_connman_prop_networks = eina_stringshare_add("Networks");
282    if (e_connman_prop_offline_mode == NULL)
283      e_connman_prop_offline_mode = eina_stringshare_add("OfflineMode");
284    if (e_connman_prop_policy == NULL)
285      e_connman_prop_policy = eina_stringshare_add("Policy");
286    if (e_connman_prop_powered == NULL)
287      e_connman_prop_powered = eina_stringshare_add("Powered");
288    if (e_connman_prop_priority == NULL)
289      e_connman_prop_priority = eina_stringshare_add("Priority");
290    if (e_connman_prop_profiles == NULL)
291      e_connman_prop_profiles = eina_stringshare_add("Profiles");
292    if (e_connman_prop_remember == NULL)
293      e_connman_prop_remember = eina_stringshare_add("Remember");
294    if (e_connman_prop_scan_interval == NULL)
295      e_connman_prop_scan_interval = eina_stringshare_add("ScanInterval");
296    if (e_connman_prop_scanning == NULL)
297      e_connman_prop_scanning = eina_stringshare_add("Scanning");
298    if (e_connman_prop_state == NULL)
299      e_connman_prop_state = eina_stringshare_add("State");
300    if (e_connman_prop_strengh == NULL)
301      e_connman_prop_strengh = eina_stringshare_add("Strength");
302    if (e_connman_prop_type == NULL)
303      e_connman_prop_type = eina_stringshare_add("Type");
304    if (e_connman_prop_wifi_mode == NULL)
305      e_connman_prop_wifi_mode = eina_stringshare_add("WiFi.Mode");
306    if (e_connman_prop_wifi_passphrase == NULL)
307      e_connman_prop_wifi_passphrase = eina_stringshare_add("WiFi.Passphrase");
308    if (e_connman_prop_wifi_security == NULL)
309      e_connman_prop_wifi_security = eina_stringshare_add("WiFi.Security");
310    if (e_connman_prop_wifi_ssid == NULL)
311      e_connman_prop_wifi_ssid = eina_stringshare_add("WiFi.SSID");
312
313    e_connman_conn = edbus_conn;
314    cb_name_owner_changed = e_dbus_signal_handler_add
315      (e_connman_conn, fdo_bus_name, fdo_path, fdo_interface, "NameOwnerChanged",
316       _e_connman_system_name_owner_changed, NULL);
317
318    if (pending_get_name_owner)
319      dbus_pending_call_cancel(pending_get_name_owner);
320
321    pending_get_name_owner = e_dbus_get_name_owner
322      (e_connman_conn, bus_name, _e_connman_get_name_owner, NULL);
323
324    e_connman_elements_init();
325
326    return init_count;
327 }
328
329 static inline void
330 _stringshare_del(const char **str)
331 {
332    if (!*str)
333      return;
334    eina_stringshare_del(*str);
335    *str = NULL;
336 }
337
338 /**
339  * Shutdown connman system.
340  *
341  * When count drops to 0 resources will be released and no calls should be
342  * made anymore.
343  */
344 unsigned int
345 e_connman_system_shutdown(void)
346 {
347    if (init_count == 0)
348      {
349         ERR("E-Dbus connman Error: connman system already shut down.");
350         return 0;
351      }
352    init_count--;
353    if (init_count > 0)
354      return init_count;
355
356    _stringshare_del(&e_connman_iface_manager);
357    _stringshare_del(&e_connman_iface_network);
358    _stringshare_del(&e_connman_iface_profile);
359    _stringshare_del(&e_connman_iface_service);
360    _stringshare_del(&e_connman_iface_device);
361    _stringshare_del(&e_connman_iface_connection);
362
363    _stringshare_del(&e_connman_prop_available);
364    _stringshare_del(&e_connman_prop_connected);
365    _stringshare_del(&e_connman_prop_connections);
366    _stringshare_del(&e_connman_prop_default);
367    _stringshare_del(&e_connman_prop_device);
368    _stringshare_del(&e_connman_prop_devices);
369    _stringshare_del(&e_connman_prop_interface);
370    _stringshare_del(&e_connman_prop_ipv4_address);
371    _stringshare_del(&e_connman_prop_ipv4_method);
372    _stringshare_del(&e_connman_prop_name);
373    _stringshare_del(&e_connman_prop_network);
374    _stringshare_del(&e_connman_prop_networks);
375    _stringshare_del(&e_connman_prop_offline_mode);
376    _stringshare_del(&e_connman_prop_policy);
377    _stringshare_del(&e_connman_prop_powered);
378    _stringshare_del(&e_connman_prop_priority);
379    _stringshare_del(&e_connman_prop_profiles);
380    _stringshare_del(&e_connman_prop_remember);
381    _stringshare_del(&e_connman_prop_scan_interval);
382    _stringshare_del(&e_connman_prop_scanning);
383    _stringshare_del(&e_connman_prop_state);
384    _stringshare_del(&e_connman_prop_strengh);
385    _stringshare_del(&e_connman_prop_type);
386    _stringshare_del(&e_connman_prop_wifi_mode);
387    _stringshare_del(&e_connman_prop_wifi_passphrase);
388    _stringshare_del(&e_connman_prop_wifi_security);
389    _stringshare_del(&e_connman_prop_wifi_ssid);
390
391    if (pending_get_name_owner)
392      {
393         dbus_pending_call_cancel(pending_get_name_owner);
394         pending_get_name_owner = NULL;
395      }
396
397    if (cb_name_owner_changed)
398      {
399         e_dbus_signal_handler_del(e_connman_conn, cb_name_owner_changed);
400         cb_name_owner_changed = NULL;
401      }
402
403    if (unique_name)
404      _e_connman_system_name_owner_exit();
405
406    e_connman_elements_shutdown();
407    eina_log_domain_unregister(_e_dbus_connman_log_dom);
408    e_connman_conn = NULL;
409
410    return init_count;
411 }