1 /* 2 * Copyright (C) 2013 The Android Open Source Project 3 * 4 * Licensed under the Apache License, Version 2.0 (the "License"); 5 * you may not use this file except in compliance with the License. 6 * You may obtain a copy of the License at 7 * 8 * http://www.apache.org/licenses/LICENSE-2.0 9 * 10 * Unless required by applicable law or agreed to in writing, software 11 * distributed under the License is distributed on an "AS IS" BASIS, 12 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. 13 * See the License for the specific language governing permissions and 14 * limitations under the License. 15 */ 16 17 package com.android.server.wifi; 18 19 import android.content.ContentResolver; 20 import android.content.Context; 21 import android.provider.Settings; 22 23 import java.io.FileDescriptor; 24 import java.io.PrintWriter; 25 26 /* Tracks persisted settings for Wi-Fi and airplane mode interaction */ 27 public class WifiSettingsStore { 28 /* Values tracked in Settings.Global.WIFI_ON */ 29 static final int WIFI_DISABLED = 0; 30 static final int WIFI_ENABLED = 1; 31 32 /* Wifi enabled while in airplane mode */ 33 private static final int WIFI_ENABLED_AIRPLANE_OVERRIDE = 2; 34 /* Wifi disabled due to airplane mode on */ 35 private static final int WIFI_DISABLED_AIRPLANE_ON = 3; 36 37 /* Persisted state that tracks the wifi & airplane interaction from settings */ 38 private int mPersistWifiState = WIFI_DISABLED; 39 /* Tracks current airplane mode state */ 40 private boolean mAirplaneModeOn = false; 41 42 /* Tracks the setting of scan being available even when wi-fi is turned off 43 */ 44 private boolean mScanAlwaysAvailable; 45 46 private final Context mContext; 47 48 /* Tracks if we have checked the saved wi-fi state after boot */ 49 private boolean mCheckSavedStateAtBoot = false; 50 51 WifiSettingsStore(Context context) { 52 mContext = context; 53 mAirplaneModeOn = getPersistedAirplaneModeOn(); 54 mPersistWifiState = getPersistedWifiState(); 55 mScanAlwaysAvailable = getPersistedScanAlwaysAvailable(); 56 } 57 58 public synchronized boolean isWifiToggleEnabled() { 59 if (!mCheckSavedStateAtBoot) { 60 mCheckSavedStateAtBoot = true; 61 if (testAndClearWifiSavedState()) return true; 62 } 63 64 if (mAirplaneModeOn) { 65 return mPersistWifiState == WIFI_ENABLED_AIRPLANE_OVERRIDE; 66 } else { 67 return mPersistWifiState != WIFI_DISABLED; 68 } 69 } 70 71 /** 72 * Returns true if airplane mode is currently on. 73 * @return {@code true} if airplane mode is on. 74 */ 75 public synchronized boolean isAirplaneModeOn() { 76 return mAirplaneModeOn; 77 } 78 79 public synchronized boolean isScanAlwaysAvailable() { 80 return !mAirplaneModeOn && mScanAlwaysAvailable; 81 } 82 83 public synchronized boolean handleWifiToggled(boolean wifiEnabled) { 84 // Can Wi-Fi be toggled in airplane mode ? 85 if (mAirplaneModeOn && !isAirplaneToggleable()) { 86 return false; 87 } 88 89 if (wifiEnabled) { 90 if (mAirplaneModeOn) { 91 persistWifiState(WIFI_ENABLED_AIRPLANE_OVERRIDE); 92 } else { 93 persistWifiState(WIFI_ENABLED); 94 } 95 } else { 96 // When wifi state is disabled, we do not care 97 // if airplane mode is on or not. The scenario of 98 // wifi being disabled due to airplane mode being turned on 99 // is handled handleAirplaneModeToggled() 100 persistWifiState(WIFI_DISABLED); 101 } 102 return true; 103 } 104 105 synchronized boolean handleAirplaneModeToggled() { 106 // Is Wi-Fi sensitive to airplane mode changes ? 107 if (!isAirplaneSensitive()) { 108 return false; 109 } 110 111 mAirplaneModeOn = getPersistedAirplaneModeOn(); 112 if (mAirplaneModeOn) { 113 // Wifi disabled due to airplane on 114 if (mPersistWifiState == WIFI_ENABLED) { 115 persistWifiState(WIFI_DISABLED_AIRPLANE_ON); 116 } 117 } else { 118 /* On airplane mode disable, restore wifi state if necessary */ 119 if (testAndClearWifiSavedState() || 120 mPersistWifiState == WIFI_ENABLED_AIRPLANE_OVERRIDE) { 121 persistWifiState(WIFI_ENABLED); 122 } 123 } 124 return true; 125 } 126 127 synchronized void handleWifiScanAlwaysAvailableToggled() { 128 mScanAlwaysAvailable = getPersistedScanAlwaysAvailable(); 129 } 130 131 void dump(FileDescriptor fd, PrintWriter pw, String[] args) { 132 pw.println("mPersistWifiState " + mPersistWifiState); 133 pw.println("mAirplaneModeOn " + mAirplaneModeOn); 134 } 135 136 private void persistWifiState(int state) { 137 final ContentResolver cr = mContext.getContentResolver(); 138 mPersistWifiState = state; 139 Settings.Global.putInt(cr, Settings.Global.WIFI_ON, state); 140 } 141 142 /* Does Wi-Fi need to be disabled when airplane mode is on ? */ 143 private boolean isAirplaneSensitive() { 144 String airplaneModeRadios = Settings.Global.getString(mContext.getContentResolver(), 145 Settings.Global.AIRPLANE_MODE_RADIOS); 146 return airplaneModeRadios == null 147 || airplaneModeRadios.contains(Settings.Global.RADIO_WIFI); 148 } 149 150 /* Is Wi-Fi allowed to be re-enabled while airplane mode is on ? */ 151 private boolean isAirplaneToggleable() { 152 String toggleableRadios = Settings.Global.getString(mContext.getContentResolver(), 153 Settings.Global.AIRPLANE_MODE_TOGGLEABLE_RADIOS); 154 return toggleableRadios != null 155 && toggleableRadios.contains(Settings.Global.RADIO_WIFI); 156 } 157 158 /** 159 * After a reboot, we restore wi-fi to be on if it was turned off temporarily for tethering. 160 * The settings app tracks the saved state, but the framework has to check it at boot to 161 * make sure the wi-fi is turned on in case it was turned off for the purpose of tethering. 162 * 163 * Note that this is not part of the regular WIFI_ON setting because this only needs to 164 * be controlled through the settings app and not the Wi-Fi public API. 165 */ 166 private boolean testAndClearWifiSavedState() { 167 int wifiSavedState = getWifiSavedState(); 168 if (wifiSavedState == WIFI_ENABLED) { 169 setWifiSavedState(WIFI_DISABLED); 170 } 171 return (wifiSavedState == WIFI_ENABLED); 172 } 173 174 /** 175 * Allow callers to set the Settings.Global.WIFI_SAVED_STATE property. 176 * 177 * When changing states, we need to remember what the wifi state was before switching. An 178 * example of this is when WiFiController switches to APEnabledState. Before swtiching to the 179 * new state, WifiController sets the current WiFi enabled/disabled state. When the AP is 180 * turned off, the WIFI_SAVED_STATE setting is used to restore the previous wifi state. 181 * 182 * @param state WiFi state to store with the Settings.Global.WIFI_SAVED_STATE property. 183 */ 184 public void setWifiSavedState(int state) { 185 Settings.Global.putInt(mContext.getContentResolver(), 186 Settings.Global.WIFI_SAVED_STATE, state); 187 } 188 189 /** 190 * Allow callers to get the Settings.Global.WIFI_SAVED_STATE property. 191 * 192 * When changing states we remember what the wifi state was before switching. This function is 193 * used to get the saved state. 194 * 195 * @return int Value for the previously saved state. 196 */ 197 public int getWifiSavedState() { 198 try { 199 return Settings.Global.getInt(mContext.getContentResolver(), 200 Settings.Global.WIFI_SAVED_STATE); 201 } catch (Settings.SettingNotFoundException e) { 202 // If we have an error, return wifiSavedState off. 203 return WIFI_DISABLED; 204 } 205 } 206 207 private int getPersistedWifiState() { 208 final ContentResolver cr = mContext.getContentResolver(); 209 try { 210 return Settings.Global.getInt(cr, Settings.Global.WIFI_ON); 211 } catch (Settings.SettingNotFoundException e) { 212 Settings.Global.putInt(cr, Settings.Global.WIFI_ON, WIFI_DISABLED); 213 return WIFI_DISABLED; 214 } 215 } 216 217 private boolean getPersistedAirplaneModeOn() { 218 return Settings.Global.getInt(mContext.getContentResolver(), 219 Settings.Global.AIRPLANE_MODE_ON, 0) == 1; 220 } 221 222 private boolean getPersistedScanAlwaysAvailable() { 223 return Settings.Global.getInt(mContext.getContentResolver(), 224 Settings.Global.WIFI_SCAN_ALWAYS_AVAILABLE, 225 0) == 1; 226 } 227 } 228