2 * Network Configuration Module
4 * Copyright (c) 2012-2013 Samsung Electronics Co., Ltd. All rights reserved.
6 * Licensed under the Apache License, Version 2.0 (the "License");
7 * you may not use this file except in compliance with the License.
8 * You may obtain a copy of the License at
10 * http://www.apache.org/licenses/LICENSE-2.0
12 * Unless required by applicable law or agreed to in writing, software
13 * distributed under the License is distributed on an "AS IS" BASIS,
14 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
15 * See the License for the specific language governing permissions and
16 * limitations under the License.
23 #include <vconf-keys.h>
24 #include <wifi-direct.h>
32 #include "netconfig.h"
34 #include "network-statistics.h"
35 #include "wifi-background-scan.h"
36 #include "wifi-power.h"
37 #include "wifi-state.h"
38 #include "wifi-agent.h"
39 #include "wifi-eap-config.h"
42 #define WLAN_DRIVER_SCRIPT "/usr/bin/wlan.sh"
44 static gboolean power_in_progress = FALSE;
45 static gboolean fm_waiting = FALSE;
47 static void __netconfig_wifi_technology_reply(DBusPendingCall *call, void *data)
51 message = dbus_pending_call_steal_reply(call);
53 if (dbus_message_get_type(message) == DBUS_MESSAGE_TYPE_ERROR) {
54 ERR("%s", dbus_message_get_error_name(message));
56 if (dbus_message_is_error(message,
57 CONNMAN_ERROR_INTERFACE ".AlreadyEnabled") == TRUE) {
58 netconfig_wifi_update_power_state(TRUE);
59 power_in_progress = FALSE;
60 } else if (dbus_message_is_error(message,
61 CONNMAN_ERROR_INTERFACE ".AlreadyDisabled") == TRUE) {
62 netconfig_wifi_update_power_state(FALSE);
63 power_in_progress = FALSE;
66 DBG("Successfully requested");
69 dbus_message_unref(message);
70 dbus_pending_call_unref(call);
73 static gboolean __netconfig_wifi_enable_technology(void)
75 gboolean reply = FALSE;
76 char param0[] = "string:Powered";
77 char param1[] = "variant:boolean:true";
78 char *param_array[] = { NULL, NULL, NULL };
80 param_array[0] = param0;
81 param_array[1] = param1;
83 reply = netconfig_invoke_dbus_method_nonblock(CONNMAN_SERVICE,
84 CONNMAN_WIFI_TECHNOLOGY_PREFIX, CONNMAN_TECHNOLOGY_INTERFACE,
85 "SetProperty", param_array, __netconfig_wifi_technology_reply);
88 ERR("Fail to enable Wi-Fi");
93 static gboolean __netconfig_wifi_disable_technology(void)
95 gboolean reply = FALSE;
96 char param0[] = "string:Powered";
97 char param1[] = "variant:boolean:false";
98 char *param_array[] = { NULL, NULL, NULL };
100 param_array[0] = param0;
101 param_array[1] = param1;
103 reply = netconfig_invoke_dbus_method_nonblock(CONNMAN_SERVICE,
104 CONNMAN_WIFI_TECHNOLOGY_PREFIX, CONNMAN_TECHNOLOGY_INTERFACE,
105 "SetProperty", param_array, __netconfig_wifi_technology_reply);
108 ERR("Fail to disable Wi-Fi");
113 static gboolean __netconfig_wifi_load_driver(void)
116 const char *path = WLAN_DRIVER_SCRIPT;
117 char *const args[] = { "wlan.sh", "start", NULL };
118 char *const envs[] = { NULL };
120 if (netconfig_emulator_is_emulated() == TRUE)
123 rv = netconfig_execute_file(path, args, envs);
125 DBG("Failed to load wireless device driver");
129 DBG("Successfully loaded wireless device driver");
133 gboolean netconfig_wifi_remove_driver(void)
136 const char *path = WLAN_DRIVER_SCRIPT;
137 char *const args[] = { "wlan.sh", "stop", NULL };
138 char *const env[] = { NULL };
140 if (netconfig_emulator_is_emulated() == TRUE)
143 rv = netconfig_execute_file(path, args, env);
145 DBG("Failed to remove wireless device driver");
149 DBG("Successfully removed wireless device driver");
153 static int __netconfig_wifi_try_to_load_driver(void);
154 static gboolean __netconfig_wifi_try_to_remove_driver(void);
156 void netconfig_wifi_notify_power_completed(gboolean power_on)
159 DBusConnection *connection;
164 sig_name = "PowerOnCompleted";
166 sig_name = "PowerOffCompleted";
168 dbus_error_init(&error);
170 connection = dbus_bus_get(DBUS_BUS_SYSTEM, &error);
171 if (connection == NULL) {
172 ERR("Error!!! Failed to get system DBus, error [%s]", error.message);
173 dbus_error_free(&error);
177 signal = dbus_message_new_signal(NETCONFIG_WIFI_PATH,
178 NETCONFIG_WIFI_INTERFACE, sig_name);
182 dbus_connection_send(connection, signal, NULL);
184 dbus_message_unref(signal);
185 dbus_connection_unref(connection);
187 INFO("(%s)", sig_name);
190 static void __netconfig_wifi_direct_state_cb(int error_code,
191 wifi_direct_device_state_e device_state, void *user_data)
193 wifi_direct_unset_device_state_changed_cb();
194 wifi_direct_deinitialize();
196 if (device_state == WIFI_DIRECT_DEVICE_STATE_DEACTIVATED) {
197 if (__netconfig_wifi_try_to_load_driver() < 0) {
198 power_in_progress = FALSE;
200 /* TODO: error report */
207 static gboolean __netconfig_wifi_direct_power_off(void)
209 DBG("Wi-Fi direct is turning off");
211 if (wifi_direct_initialize() < 0)
214 if (wifi_direct_set_device_state_changed_cb(
215 __netconfig_wifi_direct_state_cb, NULL) < 0)
218 if (wifi_direct_deactivate() < 0)
224 gboolean netconfig_is_wifi_allowed(void)
229 static int __netconfig_wifi_try_to_load_driver(void)
231 if (netconfig_is_wifi_allowed() != TRUE)
234 if (netconfig_is_wifi_tethering_on() == TRUE) {
235 /* TODO: Wi-Fi tethering turns off here */
240 if (netconfig_is_wifi_direct_on() == TRUE) {
241 if (__netconfig_wifi_direct_power_off() == TRUE) {
242 power_in_progress = TRUE;
248 if (__netconfig_wifi_load_driver() != TRUE) {
249 netconfig_wifi_remove_driver();
254 if (__netconfig_wifi_enable_technology() != TRUE) {
255 netconfig_wifi_remove_driver();
259 power_in_progress = TRUE;
264 static gboolean __netconfig_wifi_try_to_remove_driver(void)
266 netconfig_wifi_device_picker_service_stop();
268 netconfig_wifi_statistics_update_powered_off();
270 if (__netconfig_wifi_disable_technology() != TRUE)
273 power_in_progress = TRUE;
278 static void __netconfig_wifi_airplane_mode(keynode_t* node,
283 static gboolean powered_off_by_flightmode = FALSE;
285 if (power_in_progress) {
292 vconf_get_bool(VCONFKEY_TELEPHONY_FLIGHT_MODE, &value);
293 vconf_get_int(VCONFKEY_WIFI_STATE, &wifi_state);
295 DBG("flight mode %s", value > 0 ? "ON" : "OFF");
296 DBG("Wi-Fi state %d, Wi-Fi was off by flight mode %s",
297 wifi_state, powered_off_by_flightmode == TRUE ? "Yes" : "No");
300 /* flight mode enabled */
301 if (wifi_state == VCONFKEY_WIFI_OFF)
304 DBG("Turning Wi-Fi off");
306 __netconfig_wifi_try_to_remove_driver();
308 powered_off_by_flightmode = TRUE;
309 } else if (value == 0) {
310 /* flight mode disabled */
311 if (wifi_state > VCONFKEY_WIFI_OFF)
314 if (powered_off_by_flightmode != TRUE)
317 __netconfig_wifi_try_to_load_driver();
319 powered_off_by_flightmode = FALSE;
321 DBG("Invalid value (%d)", value);
324 static void __netconfig_wifi_pm_state_mode(keynode_t* node,
329 static int prev_state = VCONFKEY_PM_STATE_NORMAL;
331 /*** vconf-keys.h ***
332 * VCONFKEY_PM_STATE_NORMAL = 1,
333 * VCONFKEY_PM_STATE_LCDDIM,
334 * VCONFKEY_PM_STATE_LCDOFF,
335 * VCONFKEY_PM_STATE_SLEEP
338 if(vconf_get_int(VCONFKEY_WIFI_STATE, &wifi_state) == 0) {
339 DBG("wifi state : %d (0 off / 1 on / 2 connected)", wifi_state);
340 if(wifi_state <= VCONFKEY_WIFI_OFF)
344 if(vconf_get_int(VCONFKEY_PM_STATE, &value) < 0) {
345 ERR("VCONFKEY_PM_STATE get failed");
349 DBG("Old state: %d, current: %d", prev_state, value);
351 if((value == VCONFKEY_PM_STATE_NORMAL) && (prev_state >= VCONFKEY_PM_STATE_LCDOFF)) {
352 DBG("PM state : Wake UP!");
354 netconfig_wifi_bgscan_stop();
355 netconfig_wifi_bgscan_start();
361 void netconfig_set_power_in_progress(gboolean in_progress)
363 power_in_progress = in_progress;
366 void netconfig_check_fm_waiting(void)
369 __netconfig_wifi_airplane_mode(NULL, NULL);
372 void netconfig_wifi_power_configuration(void)
374 int wifi_last_power_state = 0;
376 vconf_notify_key_changed(VCONFKEY_TELEPHONY_FLIGHT_MODE,
377 __netconfig_wifi_airplane_mode, NULL);
379 vconf_notify_key_changed(VCONFKEY_PM_STATE,
380 __netconfig_wifi_pm_state_mode, NULL);
382 vconf_get_int(VCONF_WIFI_LAST_POWER_STATE, &wifi_last_power_state);
384 if (wifi_last_power_state == WIFI_POWER_ON) {
385 DBG("Turn Wi-Fi on automatically");
387 __netconfig_wifi_try_to_load_driver();
391 gboolean netconfig_iface_wifi_load_driver(NetconfigWifi *wifi, GError **error)
393 DBG("Wi-Fi power on requested");
395 g_return_val_if_fail(wifi != NULL, FALSE);
399 if (netconfig_is_wifi_allowed() != TRUE) {
400 netconfig_error_security_restricted(error);
405 if (power_in_progress) {
406 netconfig_error_wifi_driver_failed(error);
410 err = __netconfig_wifi_try_to_load_driver();
412 if (err == -EINPROGRESS)
413 netconfig_error_wifi_load_inprogress(error);
415 netconfig_error_wifi_driver_failed(error);
423 gboolean netconfig_iface_wifi_remove_driver(NetconfigWifi *wifi, GError **error)
425 DBG("Wi-Fi power off requested");
427 g_return_val_if_fail(wifi != NULL, FALSE);
429 if (power_in_progress) {
430 netconfig_error_wifi_driver_failed(error);
434 if (__netconfig_wifi_try_to_remove_driver() != TRUE) {
435 netconfig_error_wifi_driver_failed(error);