blob: afbb19b302b5e54be9406a0decc11f5fff4670dc [file] [log] [blame]
/*
* Copyright (C) 2008 The Android Open Source Project
*
* Licensed under the Apache License, Version 2.0 (the "License");
* you may not use this file except in compliance with the License.
* You may obtain a copy of the License at
*
* http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and
* limitations under the License.
*/
package android.net.wifi;
import android.app.ActivityManagerNative;
import android.net.NetworkInfo;
import android.net.NetworkStateTracker;
import android.net.DhcpInfo;
import android.net.NetworkUtils;
import android.net.ConnectivityManager;
import android.net.NetworkInfo.DetailedState;
import android.net.NetworkInfo.State;
import android.os.Message;
import android.os.Parcelable;
import android.os.Handler;
import android.os.HandlerThread;
import android.os.SystemProperties;
import android.os.Looper;
import android.os.RemoteException;
import android.os.ServiceManager;
import android.provider.Settings;
import android.text.TextUtils;
import android.util.EventLog;
import android.util.Log;
import android.util.Config;
import android.app.Notification;
import android.app.PendingIntent;
import android.bluetooth.BluetoothHeadset;
import android.bluetooth.BluetoothA2dp;
import android.content.ContentResolver;
import android.content.Intent;
import android.content.Context;
import android.database.ContentObserver;
import com.android.internal.app.IBatteryStats;
import java.util.List;
import java.util.ArrayList;
import java.net.UnknownHostException;
/**
* Track the state of Wifi connectivity. All event handling is done here,
* and all changes in connectivity state are initiated here.
*
* @hide
*/
public class WifiStateTracker extends NetworkStateTracker {
private static final boolean LOCAL_LOGD = Config.LOGD || false;
private static final String TAG = "WifiStateTracker";
// Event log tags (must be in sync with event-log-tags)
private static final int EVENTLOG_NETWORK_STATE_CHANGED = 50021;
private static final int EVENTLOG_SUPPLICANT_STATE_CHANGED = 50022;
private static final int EVENTLOG_DRIVER_STATE_CHANGED = 50023;
private static final int EVENTLOG_INTERFACE_CONFIGURATION_STATE_CHANGED = 50024;
private static final int EVENTLOG_SUPPLICANT_CONNECTION_STATE_CHANGED = 50025;
// Event codes
private static final int EVENT_SUPPLICANT_CONNECTION = 1;
private static final int EVENT_SUPPLICANT_DISCONNECT = 2;
private static final int EVENT_SUPPLICANT_STATE_CHANGED = 3;
private static final int EVENT_NETWORK_STATE_CHANGED = 4;
private static final int EVENT_SCAN_RESULTS_AVAILABLE = 5;
private static final int EVENT_INTERFACE_CONFIGURATION_SUCCEEDED = 6;
private static final int EVENT_INTERFACE_CONFIGURATION_FAILED = 7;
private static final int EVENT_POLL_INTERVAL = 8;
private static final int EVENT_DHCP_START = 9;
private static final int EVENT_DEFERRED_DISCONNECT = 10;
private static final int EVENT_DEFERRED_RECONNECT = 11;
/**
* The driver is started or stopped. The object will be the state: true for
* started, false for stopped.
*/
private static final int EVENT_DRIVER_STATE_CHANGED = 12;
private static final int EVENT_PASSWORD_KEY_MAY_BE_INCORRECT = 13;
/**
* Interval in milliseconds between polling for connection
* status items that are not sent via asynchronous events.
* An example is RSSI (signal strength).
*/
private static final int POLL_STATUS_INTERVAL_MSECS = 3000;
/**
* The max number of the WPA supplicant loop iterations before we
* decide that the loop should be terminated:
*/
private static final int MAX_SUPPLICANT_LOOP_ITERATIONS = 4;
/**
* When a DISCONNECT event is received, we defer handling it to
* allow for the possibility that the DISCONNECT is about to
* be followed shortly by a CONNECT to the same network we were
* just connected to. In such a case, we don't want to report
* the network as down, nor do we want to reconfigure the network
* interface, etc. If we get a CONNECT event for another network
* within the delay window, we immediately handle the pending
* disconnect before processing the CONNECT.<p/>
* The five second delay is chosen somewhat arbitrarily, but is
* meant to cover most of the cases where a DISCONNECT/CONNECT
* happens to a network.
*/
private static final int DISCONNECT_DELAY_MSECS = 5000;
/**
* When the supplicant goes idle after we do an explicit disconnect
* following a DHCP failure, we need to kick the supplicant into
* trying to associate with access points.
*/
private static final int RECONNECT_DELAY_MSECS = 2000;
/**
* The maximum number of times we will retry a connection to an access point
* for which we have failed in acquiring an IP address from DHCP. A value of
* N means that we will make N+1 connection attempts in all.
* <p>
* See {@link Settings.Secure#WIFI_MAX_DHCP_RETRY_COUNT}. This is the default
* value if a Settings value is not present.
*/
private static final int DEFAULT_MAX_DHCP_RETRIES = 2;
private static final int DRIVER_POWER_MODE_AUTO = 0;
private static final int DRIVER_POWER_MODE_ACTIVE = 1;
/**
* The current WPA supplicant loop state (used to detect looping behavior):
*/
private SupplicantState mSupplicantLoopState = SupplicantState.DISCONNECTED;
/**
* The current number of WPA supplicant loop iterations:
*/
private int mNumSupplicantLoopIterations = 0;
/**
* True if we received an event that that a password-key may be incorrect.
* If the next incoming supplicant state change event is DISCONNECT,
* broadcast a message that we have a possible password error and disable
* the network.
*/
private boolean mPasswordKeyMayBeIncorrect = false;
public static final int SUPPL_SCAN_HANDLING_NORMAL = 1;
public static final int SUPPL_SCAN_HANDLING_LIST_ONLY = 2;
private WifiMonitor mWifiMonitor;
private WifiInfo mWifiInfo;
private List<ScanResult> mScanResults;
private WifiManager mWM;
private boolean mHaveIpAddress;
private boolean mObtainingIpAddress;
private boolean mTornDownByConnMgr;
/**
* A DISCONNECT event has been received, but processing it
* is being deferred.
*/
private boolean mDisconnectPending;
/**
* An operation has been performed as a result of which we expect the next event
* will be a DISCONNECT.
*/
private boolean mDisconnectExpected;
private DhcpHandler mDhcpTarget;
private DhcpInfo mDhcpInfo;
private int mLastSignalLevel = -1;
private String mLastBssid;
private String mLastSsid;
private int mLastNetworkId = -1;
private boolean mUseStaticIp = false;
private int mReconnectCount;
// used to store the (non-persisted) num determined during device boot
// (from mcc or other phone info) before the driver is started.
private int mNumAllowedChannels = 0;
// Variables relating to the 'available networks' notification
/**
* The icon to show in the 'available networks' notification. This will also
* be the ID of the Notification given to the NotificationManager.
*/
private static final int ICON_NETWORKS_AVAILABLE =
com.android.internal.R.drawable.stat_notify_wifi_in_range;
/**
* When a notification is shown, we wait this amount before possibly showing it again.
*/
private final long NOTIFICATION_REPEAT_DELAY_MS;
/**
* Whether the user has set the setting to show the 'available networks' notification.
*/
private boolean mNotificationEnabled;
/**
* Observes the user setting to keep {@link #mNotificationEnabled} in sync.
*/
private NotificationEnabledSettingObserver mNotificationEnabledSettingObserver;
/**
* The {@link System#currentTimeMillis()} must be at least this value for us
* to show the notification again.
*/
private long mNotificationRepeatTime;
/**
* The Notification object given to the NotificationManager.
*/
private Notification mNotification;
/**
* Whether the notification is being shown, as set by us. That is, if the
* user cancels the notification, we will not receive the callback so this
* will still be true. We only guarantee if this is false, then the
* notification is not showing.
*/
private boolean mNotificationShown;
/**
* The number of continuous scans that must occur before consider the
* supplicant in a scanning state. This allows supplicant to associate with
* remembered networks that are in the scan results.
*/
private static final int NUM_SCANS_BEFORE_ACTUALLY_SCANNING = 3;
/**
* The number of scans since the last network state change. When this
* exceeds {@link #NUM_SCANS_BEFORE_ACTUALLY_SCANNING}, we consider the
* supplicant to actually be scanning. When the network state changes to
* something other than scanning, we reset this to 0.
*/
private int mNumScansSinceNetworkStateChange;
/**
* Observes the static IP address settings.
*/
private SettingsObserver mSettingsObserver;
private boolean mIsScanModeActive;
private boolean mIsScanModeSetDueToAHiddenNetwork;
// Wi-Fi run states:
private static final int RUN_STATE_STARTING = 1;
private static final int RUN_STATE_RUNNING = 2;
private static final int RUN_STATE_STOPPING = 3;
private static final int RUN_STATE_STOPPED = 4;
private static final String mRunStateNames[] = {
"Starting",
"Running",
"Stopping",
"Stopped"
};
private int mRunState;
private final IBatteryStats mBatteryStats;
private boolean mIsScanOnly;
private BluetoothA2dp mBluetoothA2dp;
private String mInterfaceName;
private static String LS = System.getProperty("line.separator");
private Runnable mReleaseWakeLockCallback;
private static String[] sDnsPropNames;
/**
* A structure for supplying information about a supplicant state
* change in the STATE_CHANGE event message that comes from the
* WifiMonitor
* thread.
*/
private static class SupplicantStateChangeResult {
SupplicantStateChangeResult(int networkId, SupplicantState state) {
this.state = state;
this.networkId = networkId;
}
int networkId;
SupplicantState state;
}
/**
* A structure for supplying information about a connection in
* the CONNECTED event message that comes from the WifiMonitor
* thread.
*/
private static class NetworkStateChangeResult {
NetworkStateChangeResult(DetailedState state, String BSSID, int networkId) {
this.state = state;
this.BSSID = BSSID;
this.networkId = networkId;
}
DetailedState state;
String BSSID;
int networkId;
}
public WifiStateTracker(Context context, Handler target) {
super(context, target, ConnectivityManager.TYPE_WIFI, 0, "WIFI", "");
mWifiInfo = new WifiInfo();
mWifiMonitor = new WifiMonitor(this);
mHaveIpAddress = false;
mObtainingIpAddress = false;
setTornDownByConnMgr(false);
mDisconnectPending = false;
mScanResults = new ArrayList<ScanResult>();
// Allocate DHCP info object once, and fill it in on each request
mDhcpInfo = new DhcpInfo();
mIsScanModeSetDueToAHiddenNetwork = false;
mRunState = RUN_STATE_STARTING;
// Setting is in seconds
NOTIFICATION_REPEAT_DELAY_MS = Settings.Secure.getInt(context.getContentResolver(),
Settings.Secure.WIFI_NETWORKS_AVAILABLE_REPEAT_DELAY, 900) * 1000l;
mNotificationEnabledSettingObserver = new NotificationEnabledSettingObserver(new Handler());
mNotificationEnabledSettingObserver.register();
mSettingsObserver = new SettingsObserver(new Handler());
mInterfaceName = SystemProperties.get("wifi.interface", "tiwlan0");
sDnsPropNames = new String[] {
"dhcp." + mInterfaceName + ".dns1",
"dhcp." + mInterfaceName + ".dns2"
};
mBatteryStats = IBatteryStats.Stub.asInterface(ServiceManager.getService("batteryinfo"));
}
/**
* Helper method: sets the supplicant state and keeps the network
* info updated.
* @param state the new state
*/
private void setSupplicantState(SupplicantState state) {
mWifiInfo.setSupplicantState(state);
updateNetworkInfo();
}
public SupplicantState getSupplicantState() {
return mWifiInfo.getSupplicantState();
}
/**
* Helper method: sets the supplicant state and keeps the network
* info updated (string version).
* @param stateName the string name of the new state
*/
private void setSupplicantState(String stateName) {
mWifiInfo.setSupplicantState(stateName);
updateNetworkInfo();
}
/**
* Helper method: sets the boolean indicating that the connection
* manager asked the network to be torn down (and so only the connection
* manager can set it up again).
* network info updated.
* @param flag {@code true} if explicitly disabled.
*/
private void setTornDownByConnMgr(boolean flag) {
mTornDownByConnMgr = flag;
updateNetworkInfo();
}
/**
* Return the IP addresses of the DNS servers available for the WLAN
* network interface.
* @return a list of DNS addresses, with no holes.
*/
public String[] getNameServers() {
return getNameServerList(sDnsPropNames);
}
/**
* Return the system properties name associated with the tcp buffer sizes
* for this network.
*/
public String getTcpBufferSizesPropName() {
return "net.tcp.buffersize.wifi";
}
public void startMonitoring() {
/*
* Get a handle on the WifiManager. This cannot be done in our
* constructor, because the Wifi service is not yet registered.
*/
mWM = (WifiManager)mContext.getSystemService(Context.WIFI_SERVICE);
}
public void startEventLoop() {
mWifiMonitor.startMonitoring();
}
/**
* Wi-Fi is considered available as long as we have a connection to the
* supplicant daemon and there is at least one enabled network. If a teardown
* was explicitly requested, then Wi-Fi can be restarted with a reconnect
* request, so it is considered available. If the driver has been stopped
* for any reason other than a teardown request, Wi-Fi is considered
* unavailable.
* @return {@code true} if Wi-Fi connections are possible
*/
public synchronized boolean isAvailable() {
/*
* TODO: Need to also look at scan results to see whether we're
* in range of any access points. If we have scan results that
* are no more than N seconds old, use those, otherwise, initiate
* a scan and wait for the results. This only matters if we
* allow mobile to be the preferred network.
*/
SupplicantState suppState = mWifiInfo.getSupplicantState();
return suppState != SupplicantState.UNINITIALIZED &&
suppState != SupplicantState.INACTIVE &&
(mTornDownByConnMgr || !isDriverStopped());
}
/**
* {@inheritDoc}
* There are currently no defined Wi-Fi subtypes.
*/
public int getNetworkSubtype() {
return 0;
}
/**
* Helper method: updates the network info object to keep it in sync with
* the Wi-Fi state tracker.
*/
private void updateNetworkInfo() {
mNetworkInfo.setIsAvailable(isAvailable());
}
/**
* Report whether the Wi-Fi connection is fully configured for data.
* @return {@code true} if the {@link SupplicantState} is
* {@link android.net.wifi.SupplicantState#COMPLETED COMPLETED}.
*/
public boolean isConnectionCompleted() {
return mWifiInfo.getSupplicantState() == SupplicantState.COMPLETED;
}
/**
* Report whether the Wi-Fi connection has successfully acquired an IP address.
* @return {@code true} if the Wi-Fi connection has been assigned an IP address.
*/
public boolean hasIpAddress() {
return mHaveIpAddress;
}
/**
* Send the tracker a notification that a user-entered password key
* may be incorrect (i.e., caused authentication to fail).
*/
void notifyPasswordKeyMayBeIncorrect() {
sendEmptyMessage(EVENT_PASSWORD_KEY_MAY_BE_INCORRECT);
}
/**
* Send the tracker a notification that a connection to the supplicant
* daemon has been established.
*/
void notifySupplicantConnection() {
sendEmptyMessage(EVENT_SUPPLICANT_CONNECTION);
}
/**
* Send the tracker a notification that the state of the supplicant
* has changed.
* @param networkId the configured network on which the state change occurred
* @param newState the new {@code SupplicantState}
*/
void notifyStateChange(int networkId, SupplicantState newState) {
Message msg = Message.obtain(
this, EVENT_SUPPLICANT_STATE_CHANGED,
new SupplicantStateChangeResult(networkId, newState));
msg.sendToTarget();
}
/**
* Send the tracker a notification that the state of Wifi connectivity
* has changed.
* @param networkId the configured network on which the state change occurred
* @param newState the new network state
* @param BSSID when the new state is {@link DetailedState#CONNECTED
* NetworkInfo.DetailedState.CONNECTED},
* this is the MAC address of the access point. Otherwise, it
* is {@code null}.
*/
void notifyStateChange(DetailedState newState, String BSSID, int networkId) {
Message msg = Message.obtain(
this, EVENT_NETWORK_STATE_CHANGED,
new NetworkStateChangeResult(newState, BSSID, networkId));
msg.sendToTarget();
}
/**
* Send the tracker a notification that a scan has completed, and results
* are available.
*/
void notifyScanResultsAvailable() {
// reset the supplicant's handling of scan results to "normal" mode
synchronized (this) {
WifiNative.setScanResultHandlingCommand(SUPPL_SCAN_HANDLING_NORMAL);
}
sendEmptyMessage(EVENT_SCAN_RESULTS_AVAILABLE);
}
/**
* Send the tracker a notification that we can no longer communicate with
* the supplicant daemon.
*/
void notifySupplicantLost() {
sendEmptyMessage(EVENT_SUPPLICANT_DISCONNECT);
}
/**
* Send the tracker a notification that the Wi-Fi driver has been stopped.
*/
void notifyDriverStopped() {
mRunState = RUN_STATE_STOPPED;
// Send a driver stopped message to our handler
Message.obtain(this, EVENT_DRIVER_STATE_CHANGED, 0, 0).sendToTarget();
}
/**
* Send the tracker a notification that the Wi-Fi driver has been restarted after
* having been stopped.
*/
void notifyDriverStarted() {
// Send a driver started message to our handler
Message.obtain(this, EVENT_DRIVER_STATE_CHANGED, 1, 0).sendToTarget();
}
/**
* Set the interval timer for polling connection information
* that is not delivered asynchronously.
*/
private synchronized void setPollTimer () {
if (!hasMessages(EVENT_POLL_INTERVAL)) {
sendEmptyMessageDelayed(EVENT_POLL_INTERVAL, POLL_STATUS_INTERVAL_MSECS);
}
}
private synchronized boolean isDriverStopped() {
return mRunState == RUN_STATE_STOPPED || mRunState == RUN_STATE_STOPPING;
}
private void noteRunState() {
try {
if (mRunState == RUN_STATE_RUNNING) {
mBatteryStats.noteWifiRunning();
} else if (mRunState == RUN_STATE_STOPPED) {
mBatteryStats.noteWifiStopped();
}
} catch (RemoteException ignore) {
}
}
/**
* Set the number of allowed radio frequency channels from the system
* setting value, if any.
* @return {@code true} if the operation succeeds, {@code false} otherwise, e.g.,
* the number of channels is invalid.
*/
public boolean setNumAllowedChannels() {
try {
return setNumAllowedChannels(
Settings.Secure.getInt(mContext.getContentResolver(),
Settings.Secure.WIFI_NUM_ALLOWED_CHANNELS));
} catch (Settings.SettingNotFoundException e) {
if (mNumAllowedChannels != 0) {
WifiNative.setNumAllowedChannelsCommand(mNumAllowedChannels);
}
// otherwise, use the driver default
}
return true;
}
/**
* Set the number of radio frequency channels that are allowed to be used
* in the current regulatory domain.
* @param numChannels the number of allowed channels. Must be greater than 0
* and less than or equal to 16.
* @return {@code true} if the operation succeeds, {@code false} otherwise, e.g.,
* {@code numChannels} is outside the valid range.
*/
public synchronized boolean setNumAllowedChannels(int numChannels) {
mNumAllowedChannels = numChannels;
return WifiNative.setNumAllowedChannelsCommand(numChannels);
}
/**
* Set the run state to either "normal" or "scan-only".
* @param scanOnlyMode true if the new mode should be scan-only.
*/
public synchronized void setScanOnlyMode(boolean scanOnlyMode) {
// do nothing unless scan-only mode is changing
if (mIsScanOnly != scanOnlyMode) {
int scanType = (scanOnlyMode ?
SUPPL_SCAN_HANDLING_LIST_ONLY : SUPPL_SCAN_HANDLING_NORMAL);
if (LOCAL_LOGD) Log.v(TAG, "Scan-only mode changing to " + scanOnlyMode + " scanType=" + scanType);
if (WifiNative.setScanResultHandlingCommand(scanType)) {
mIsScanOnly = scanOnlyMode;
if (!isDriverStopped()) {
if (scanOnlyMode) {
WifiNative.disconnectCommand();
} else {
WifiNative.reconnectCommand();
}
}
}
}
}
/**
* Enable or disable Bluetooth coexistence scan mode. When this mode is on,
* some of the low-level scan parameters used by the driver are changed to
* reduce interference with A2DP streaming.
*
* @param isBluetoothPlaying whether to enable or disable this mode
*/
public synchronized void setBluetoothScanMode(boolean isBluetoothPlaying) {
WifiNative.setBluetoothCoexistenceScanModeCommand(isBluetoothPlaying);
}
private void checkIsBluetoothPlaying() {
boolean isBluetoothPlaying = false;
List<String> connected = mBluetoothA2dp.listConnectedSinks();
for (String address : connected) {
if (mBluetoothA2dp.getSinkState(address) == BluetoothA2dp.STATE_PLAYING) {
isBluetoothPlaying = true;
break;
}
}
setBluetoothScanMode(isBluetoothPlaying);
}
@Override
public void releaseWakeLock() {
if (mReleaseWakeLockCallback != null) {
mReleaseWakeLockCallback.run();
}
}
public void setReleaseWakeLockCallback(Runnable callback) {
mReleaseWakeLockCallback = callback;
}
/**
* Tracks the WPA supplicant states to detect "loop" situations.
* @param newSupplicantState The new WPA supplicant state.
* @return {@code true} if the supplicant loop should be stopped
* and {@code false} if it should continue.
*/
private boolean isSupplicantLooping(SupplicantState newSupplicantState) {
if (SupplicantState.ASSOCIATING.ordinal() <= newSupplicantState.ordinal()
&& newSupplicantState.ordinal() < SupplicantState.COMPLETED.ordinal()) {
if (mSupplicantLoopState != newSupplicantState) {
if (newSupplicantState.ordinal() < mSupplicantLoopState.ordinal()) {
++mNumSupplicantLoopIterations;
}
mSupplicantLoopState = newSupplicantState;
}
} else if (newSupplicantState == SupplicantState.COMPLETED) {
resetSupplicantLoopState();
}
return mNumSupplicantLoopIterations >= MAX_SUPPLICANT_LOOP_ITERATIONS;
}
/**
* Resets the WPA supplicant loop state.
*/
private void resetSupplicantLoopState() {
mNumSupplicantLoopIterations = 0;
}
@Override
public void handleMessage(Message msg) {
Intent intent;
switch (msg.what) {
case EVENT_SUPPLICANT_CONNECTION:
mRunState = RUN_STATE_RUNNING;
noteRunState();
checkUseStaticIp();
/*
* DHCP requests are blocking, so run them in a separate thread.
*/
HandlerThread dhcpThread = new HandlerThread("DHCP Handler Thread");
dhcpThread.start();
mDhcpTarget = new DhcpHandler(dhcpThread.getLooper(), this);
mIsScanModeActive = true;
mTornDownByConnMgr = false;
mLastBssid = null;
mLastSsid = null;
requestConnectionInfo();
SupplicantState supplState = mWifiInfo.getSupplicantState();
/**
* The MAC address isn't going to change, so just request it
* once here.
*/
String macaddr;
synchronized (this) {
macaddr = WifiNative.getMacAddressCommand();
}
if (macaddr != null) {
mWifiInfo.setMacAddress(macaddr);
}
if (LOCAL_LOGD) Log.v(TAG, "Connection to supplicant established, state=" +
supplState);
// Wi-Fi supplicant connection state changed:
// [31- 2] Reserved for future use
// [ 1- 0] Connected to supplicant (1), disconnected from supplicant (0) ,
// or supplicant died (2)
EventLog.writeEvent(EVENTLOG_SUPPLICANT_CONNECTION_STATE_CHANGED, 1);
/*
* The COMPLETED state change from the supplicant may have occurred
* in between polling for supplicant availability, in which case
* we didn't perform a DHCP request to get an IP address.
*/
if (supplState == SupplicantState.COMPLETED) {
mLastBssid = mWifiInfo.getBSSID();
mLastSsid = mWifiInfo.getSSID();
configureInterface();
}
if (ActivityManagerNative.isSystemReady()) {
intent = new Intent(WifiManager.SUPPLICANT_CONNECTION_CHANGE_ACTION);
intent.putExtra(WifiManager.EXTRA_SUPPLICANT_CONNECTED, true);
mContext.sendBroadcast(intent);
}
if (supplState == SupplicantState.COMPLETED && mHaveIpAddress) {
setDetailedState(DetailedState.CONNECTED);
} else {
setDetailedState(WifiInfo.getDetailedStateOf(supplState));
}
/*
* Filter out multicast packets. This saves battery power, since
* the CPU doesn't have to spend time processing packets that
* are going to end up being thrown away.
*
* Note that rather than turn this off directly, we use the
* public api - this keeps us all in sync - turn multicast on
* first and then off.. if nobody else wants it on it'll be
* off then and it's all synchronized within the API.
*/
WifiManager.MulticastLock l =
mWM.createMulticastLock("WifiStateTracker");
l.acquire();
l.release();
if (mBluetoothA2dp == null) {
mBluetoothA2dp = new BluetoothA2dp(mContext);
}
checkIsBluetoothPlaying();
break;
case EVENT_SUPPLICANT_DISCONNECT:
mRunState = RUN_STATE_STOPPED;
noteRunState();
int wifiState = mWM.getWifiState();
boolean died = wifiState != WifiManager.WIFI_STATE_DISABLED &&
wifiState != WifiManager.WIFI_STATE_DISABLING;
if (died) {
if (LOCAL_LOGD) Log.v(TAG, "Supplicant died unexpectedly");
} else {
if (LOCAL_LOGD) Log.v(TAG, "Connection to supplicant lost");
}
// Wi-Fi supplicant connection state changed:
// [31- 2] Reserved for future use
// [ 1- 0] Connected to supplicant (1), disconnected from supplicant (0) ,
// or supplicant died (2)
EventLog.writeEvent(EVENTLOG_SUPPLICANT_CONNECTION_STATE_CHANGED, died ? 2 : 0);
synchronized (this) {
WifiNative.closeSupplicantConnection();
}
if (died) {
resetInterface();
}
// When supplicant dies, kill the DHCP thread
if (mDhcpTarget != null) {
mDhcpTarget.getLooper().quit();
mDhcpTarget = null;
}
mContext.removeStickyBroadcast(new Intent(WifiManager.NETWORK_STATE_CHANGED_ACTION));
if (ActivityManagerNative.isSystemReady()) {
intent = new Intent(WifiManager.SUPPLICANT_CONNECTION_CHANGE_ACTION);
intent.putExtra(WifiManager.EXTRA_SUPPLICANT_CONNECTED, false);
mContext.sendBroadcast(intent);
}
setDetailedState(DetailedState.DISCONNECTED);
setSupplicantState(SupplicantState.UNINITIALIZED);
mHaveIpAddress = false;
mObtainingIpAddress = false;
if (died) {
mWM.setWifiEnabled(false);
}
break;
case EVENT_SUPPLICANT_STATE_CHANGED:
SupplicantStateChangeResult supplicantStateResult =
(SupplicantStateChangeResult) msg.obj;
SupplicantState newState = supplicantStateResult.state;
SupplicantState currentState = mWifiInfo.getSupplicantState();
// Wi-Fi supplicant state changed:
// [31- 6] Reserved for future use
// [ 5- 0] Supplicant state ordinal (as defined by SupplicantState)
int eventLogParam = (newState.ordinal() & 0x3f);
EventLog.writeEvent(EVENTLOG_SUPPLICANT_STATE_CHANGED, eventLogParam);
if (LOCAL_LOGD) Log.v(TAG, "Changing supplicant state: "
+ currentState +
" ==> " + newState);
int networkId = supplicantStateResult.networkId;
/*
* Did we get to DISCONNECTED state due to an
* authentication (password) failure?
*/
boolean failedToAuthenticate = false;
if (newState == SupplicantState.DISCONNECTED) {
failedToAuthenticate = mPasswordKeyMayBeIncorrect;
}
mPasswordKeyMayBeIncorrect = false;
/*
* Keep track of the supplicant state and check if we should
* disable the network
*/
boolean disabledNetwork = false;
if (isSupplicantLooping(newState)) {
if (LOCAL_LOGD) {
Log.v(TAG,
"Stop WPA supplicant loop and disable network");
}
disabledNetwork = wifiManagerDisableNetwork(networkId);
}
if (disabledNetwork) {
/*
* Reset the loop state if we disabled the network
*/
resetSupplicantLoopState();
} else if (newState != currentState ||
(newState == SupplicantState.DISCONNECTED && isDriverStopped())) {
setSupplicantState(newState);
if (newState == SupplicantState.DORMANT) {
DetailedState newDetailedState;
if (mIsScanOnly || mRunState == RUN_STATE_STOPPING) {
newDetailedState = DetailedState.IDLE;
} else {
newDetailedState = DetailedState.FAILED;
}
handleDisconnectedState(newDetailedState);
/**
* If we were associated with a network (networkId != -1),
* assume we reached this state because of a failed attempt
* to acquire an IP address, and attempt another connection
* and IP address acquisition in RECONNECT_DELAY_MSECS
* milliseconds.
*/
if (mRunState == RUN_STATE_RUNNING && !mIsScanOnly && networkId != -1) {
sendEmptyMessageDelayed(EVENT_DEFERRED_RECONNECT, RECONNECT_DELAY_MSECS);
} else if (mRunState == RUN_STATE_STOPPING) {
synchronized (this) {
WifiNative.stopDriverCommand();
}
} else if (mRunState == RUN_STATE_STARTING && !mIsScanOnly) {
synchronized (this) {
WifiNative.reconnectCommand();
}
}
} else if (newState == SupplicantState.DISCONNECTED) {
if (isDriverStopped() || mDisconnectExpected) {
handleDisconnectedState(DetailedState.DISCONNECTED);
} else {
scheduleDisconnect();
}
} else if (newState != SupplicantState.COMPLETED && !mDisconnectPending) {
/**
* Ignore events that don't change the connectivity state,
* such as WPA rekeying operations.
*/
if (!(currentState == SupplicantState.COMPLETED &&
(newState == SupplicantState.ASSOCIATING ||
newState == SupplicantState.ASSOCIATED ||
newState == SupplicantState.FOUR_WAY_HANDSHAKE ||
newState == SupplicantState.GROUP_HANDSHAKE))) {
setDetailedState(WifiInfo.getDetailedStateOf(newState));
}
}
mDisconnectExpected = false;
intent = new Intent(WifiManager.SUPPLICANT_STATE_CHANGED_ACTION);
intent.addFlags(Intent.FLAG_RECEIVER_REGISTERED_ONLY_BEFORE_BOOT);
intent.putExtra(WifiManager.EXTRA_NEW_STATE, (Parcelable)newState);
if (failedToAuthenticate) {
if (LOCAL_LOGD) Log.d(TAG, "Failed to authenticate, disabling network " + networkId);
wifiManagerDisableNetwork(networkId);
intent.putExtra(
WifiManager.EXTRA_SUPPLICANT_ERROR,
WifiManager.ERROR_AUTHENTICATING);
}
mContext.sendStickyBroadcast(intent);
}
break;
case EVENT_NETWORK_STATE_CHANGED:
/*
* Each CONNECT or DISCONNECT generates a pair of events.
* One is a supplicant state change event, and the other
* is a network state change event. For connects, the
* supplicant event always arrives first, followed by
* the network state change event. Only the latter event
* has the BSSID, which we are interested in capturing.
* For disconnects, the order is the opposite -- the
* network state change event comes first, followed by
* the supplicant state change event.
*/
NetworkStateChangeResult result =
(NetworkStateChangeResult) msg.obj;
// Wi-Fi network state changed:
// [31- 6] Reserved for future use
// [ 5- 0] Detailed state ordinal (as defined by NetworkInfo.DetailedState)
eventLogParam = (result.state.ordinal() & 0x3f);
EventLog.writeEvent(EVENTLOG_NETWORK_STATE_CHANGED, eventLogParam);
if (LOCAL_LOGD) Log.v(TAG, "New network state is " + result.state);
/*
* If we're in scan-only mode, don't advance the state machine, and
* don't report the state change to clients.
*/
if (mIsScanOnly) {
if (LOCAL_LOGD) Log.v(TAG, "Dropping event in scan-only mode");
break;
}
if (result.state != DetailedState.SCANNING) {
/*
* Reset the scan count since there was a network state
* change. This could be from supplicant trying to associate
* with a network.
*/
mNumScansSinceNetworkStateChange = 0;
}
/*
* If the supplicant sent us a CONNECTED event, we don't
* want to send out an indication of overall network
* connectivity until we have our IP address. If the
* supplicant sent us a DISCONNECTED event, we delay
* sending a notification in case a reconnection to
* the same access point occurs within a short time.
*/
if (result.state == DetailedState.DISCONNECTED) {
if (mWifiInfo.getSupplicantState() != SupplicantState.DORMANT) {
scheduleDisconnect();
}
break;
}
requestConnectionStatus(mWifiInfo);
if (!(result.state == DetailedState.CONNECTED &&
(!mHaveIpAddress || mDisconnectPending))) {
setDetailedState(result.state);
}
if (result.state == DetailedState.CONNECTED) {
/*
* Remove the 'available networks' notification when we
* successfully connect to a network.
*/
setNotificationVisible(false, 0, false, 0);
boolean wasDisconnectPending = mDisconnectPending;
cancelDisconnect();
if (!TextUtils.equals(mWifiInfo.getSSID(), mLastSsid)) {
/*
* The connection is fully configured as far as link-level
* connectivity is concerned, but we may still need to obtain
* an IP address. But do this only if we are connecting to
* a different network than we were connected to previously.
*/
if (wasDisconnectPending) {
DetailedState saveState = getNetworkInfo().getDetailedState();
handleDisconnectedState(DetailedState.DISCONNECTED);
setDetailedStateInternal(saveState);
}
configureInterface();
}
mLastBssid = result.BSSID;
mLastSsid = mWifiInfo.getSSID();
mLastNetworkId = result.networkId;
if (mHaveIpAddress) {
setDetailedState(DetailedState.CONNECTED);
} else {
setDetailedState(DetailedState.OBTAINING_IPADDR);
}
}
sendNetworkStateChangeBroadcast(mWifiInfo.getBSSID());
break;
case EVENT_SCAN_RESULTS_AVAILABLE:
if (ActivityManagerNative.isSystemReady()) {
mContext.sendBroadcast(new Intent(WifiManager.SCAN_RESULTS_AVAILABLE_ACTION));
}
sendScanResultsAvailable();
/**
* On receiving the first scan results after connecting to
* the supplicant, switch scan mode over to passive.
*/
if (!mIsScanModeSetDueToAHiddenNetwork) {
// This is the only place at the moment where we set
// the scan mode NOT due to a hidden network. This is
// what the second parameter value (false) stands for.
setScanMode(false, false);
}
break;
case EVENT_POLL_INTERVAL:
if (mWifiInfo.getSupplicantState() != SupplicantState.UNINITIALIZED) {
requestPolledInfo(mWifiInfo, true);
if (mWifiInfo.getSupplicantState() == SupplicantState.COMPLETED) {
setPollTimer();
}
}
break;
case EVENT_DEFERRED_DISCONNECT:
if (mWifiInfo.getSupplicantState() != SupplicantState.UNINITIALIZED) {
handleDisconnectedState(DetailedState.DISCONNECTED);
}
break;
case EVENT_DEFERRED_RECONNECT:
/*
* If we've exceeded the maximum number of retries for reconnecting
* to a given network, disable the network so that the supplicant
* will try some other network, if any is available.
* TODO: network ID may have changed since we stored it.
*/
if (mWifiInfo.getSupplicantState() != SupplicantState.UNINITIALIZED) {
if (++mReconnectCount > getMaxDhcpRetries()) {
mWM.disableNetwork(mLastNetworkId);
}
synchronized(this) {
WifiNative.reconnectCommand();
}
}
break;
case EVENT_INTERFACE_CONFIGURATION_SUCCEEDED:
/**
* Since this event is sent from another thread, it might have been
* sent after we closed our connection to the supplicant in the course
* of disabling Wi-Fi. In that case, we should just ignore the event.
*/
if (mWifiInfo.getSupplicantState() == SupplicantState.UNINITIALIZED) {
break;
}
mReconnectCount = 0;
mHaveIpAddress = true;
mObtainingIpAddress = false;
mWifiInfo.setIpAddress(mDhcpInfo.ipAddress);
mLastSignalLevel = -1; // force update of signal strength
if (mNetworkInfo.getDetailedState() != DetailedState.CONNECTED) {
setDetailedState(DetailedState.CONNECTED);
sendNetworkStateChangeBroadcast(mWifiInfo.getBSSID());
} else {
mTarget.sendEmptyMessage(EVENT_CONFIGURATION_CHANGED);
}
if (LOCAL_LOGD) Log.v(TAG, "IP configuration: " + mDhcpInfo);
// Wi-Fi interface configuration state changed:
// [31- 1] Reserved for future use
// [ 0- 0] Interface configuration succeeded (1) or failed (0)
EventLog.writeEvent(EVENTLOG_INTERFACE_CONFIGURATION_STATE_CHANGED, 1);
// We've connected successfully, so allow the notification again in the future
resetNotificationTimer();
break;
case EVENT_INTERFACE_CONFIGURATION_FAILED:
if (mWifiInfo.getSupplicantState() != SupplicantState.UNINITIALIZED) {
// Wi-Fi interface configuration state changed:
// [31- 1] Reserved for future use
// [ 0- 0] Interface configuration succeeded (1) or failed (0)
EventLog.writeEvent(EVENTLOG_INTERFACE_CONFIGURATION_STATE_CHANGED, 0);
mHaveIpAddress = false;
mWifiInfo.setIpAddress(0);
mObtainingIpAddress = false;
synchronized(this) {
WifiNative.disconnectCommand();
}
}
break;
case EVENT_DRIVER_STATE_CHANGED:
boolean driverStarted = msg.arg1 != 0;
// Wi-Fi driver state changed:
// [31- 1] Reserved for future use
// [ 0- 0] Driver start (1) or stopped (0)
eventLogParam = driverStarted ? 1 : 0;
EventLog.writeEvent(EVENTLOG_DRIVER_STATE_CHANGED, eventLogParam);
if (driverStarted) {
/**
* Set the number of allowed radio channels according
* to the system setting, since it gets reset by the
* driver upon changing to the STARTED state.
*/
setNumAllowedChannels();
synchronized (this) {
if (mRunState == RUN_STATE_STARTING) {
mRunState = RUN_STATE_RUNNING;
if (!mIsScanOnly) {
WifiNative.reconnectCommand();
} else {
// In some situations, supplicant needs to be kickstarted to
// start the background scanning
WifiNative.scanCommand(true);
}
}
}
}
noteRunState();
break;
case EVENT_PASSWORD_KEY_MAY_BE_INCORRECT:
mPasswordKeyMayBeIncorrect = true;
break;
}
}
private boolean wifiManagerDisableNetwork(int networkId) {
boolean disabledNetwork = false;
if (0 <= networkId) {
disabledNetwork = mWM.disableNetwork(networkId);
if (LOCAL_LOGD) {
if (disabledNetwork) {
Log.v(TAG, "Disabled network: " + networkId);
}
}
}
if (LOCAL_LOGD) {
if (!disabledNetwork) {
Log.e(TAG, "Failed to disable network:" +
" invalid network id: " + networkId);
}
}
return disabledNetwork;
}
public synchronized void setScanMode(
boolean isScanModeActive, boolean setDueToAHiddenNetwork) {
mIsScanModeSetDueToAHiddenNetwork = setDueToAHiddenNetwork;
if (mIsScanModeActive != isScanModeActive) {
WifiNative.setScanModeCommand(mIsScanModeActive = isScanModeActive);
}
}
private void configureInterface() {
setPollTimer();
mLastSignalLevel = -1;
if (!mUseStaticIp) {
if (!mHaveIpAddress && !mObtainingIpAddress) {
mObtainingIpAddress = true;
mDhcpTarget.sendEmptyMessage(EVENT_DHCP_START);
}
} else {
int event;
if (NetworkUtils.configureInterface(mInterfaceName, mDhcpInfo)) {
mHaveIpAddress = true;
event = EVENT_INTERFACE_CONFIGURATION_SUCCEEDED;
if (LOCAL_LOGD) Log.v(TAG, "Static IP configuration succeeded");
} else {
mHaveIpAddress = false;
event = EVENT_INTERFACE_CONFIGURATION_FAILED;
if (LOCAL_LOGD) Log.v(TAG, "Static IP configuration failed");
}
sendEmptyMessage(event);
}
}
/**
* Reset our IP state and send out broadcasts following a disconnect.
* @param newState the {@code DetailedState} to set. Should be either
* {@code DISCONNECTED} or {@code FAILED}.
*/
private void handleDisconnectedState(DetailedState newState) {
if (LOCAL_LOGD) Log.d(TAG, "Deconfiguring interface and stopping DHCP");
if (mDisconnectPending) {
cancelDisconnect();
}
mDisconnectExpected = false;
resetInterface();
setDetailedState(newState);
sendNetworkStateChangeBroadcast(mLastBssid);
mWifiInfo.setBSSID(null);
mLastBssid = null;
mLastSsid = null;
mDisconnectPending = false;
}
/**
* Resets the Wi-Fi interface by clearing any state, resetting any sockets
* using the interface, stopping DHCP, and disabling the interface.
*/
public void resetInterface() {
mHaveIpAddress = false;
mObtainingIpAddress = false;
mWifiInfo.setIpAddress(0);
/*
* Reset connection depends on both the interface and the IP assigned,
* so it should be done before any chance of the IP being lost.
*/
NetworkUtils.resetConnections(mInterfaceName);
// Stop DHCP
if (mDhcpTarget != null) {
mDhcpTarget.setCancelCallback(true);
mDhcpTarget.removeMessages(EVENT_DHCP_START);
}
if (!NetworkUtils.stopDhcp(mInterfaceName)) {
Log.e(TAG, "Could not stop DHCP");
}
NetworkUtils.disableInterface(mInterfaceName);
}
/**
* The supplicant is reporting that we are disconnected from the current
* access point. Often, however, a disconnect will be followed very shortly
* by a reconnect to the same access point. Therefore, we delay resetting
* the connection's IP state for a bit.
*/
private void scheduleDisconnect() {
mDisconnectPending = true;
if (!hasMessages(EVENT_DEFERRED_DISCONNECT)) {
sendEmptyMessageDelayed(EVENT_DEFERRED_DISCONNECT, DISCONNECT_DELAY_MSECS);
}
}
private void cancelDisconnect() {
mDisconnectPending = false;
removeMessages(EVENT_DEFERRED_DISCONNECT);
}
public DhcpInfo getDhcpInfo() {
return mDhcpInfo;
}
public synchronized List<ScanResult> getScanResultsList() {
return mScanResults;
}
public synchronized void setScanResultsList(List<ScanResult> scanList) {
mScanResults = scanList;
}
/**
* Get status information for the current connection, if any.
* @return a {@link WifiInfo} object containing information about the current connection
*/
public WifiInfo requestConnectionInfo() {
requestConnectionStatus(mWifiInfo);
requestPolledInfo(mWifiInfo, false);
return mWifiInfo;
}
private void requestConnectionStatus(WifiInfo info) {
String reply;
synchronized (this) {
reply = WifiNative.statusCommand();
}
if (reply == null) {
return;
}
/*
* Parse the reply from the supplicant to the status command, and update
* local state accordingly. The reply is a series of lines of the form
* "name=value".
*/
String SSID = null;
String BSSID = null;
String suppState = null;
int netId = -1;
String[] lines = reply.split("\n");
for (String line : lines) {
String[] prop = line.split(" *= *");
if (prop.length < 2)
continue;
String name = prop[0];
String value = prop[1];
if (name.equalsIgnoreCase("id"))
netId = Integer.parseInt(value);
else if (name.equalsIgnoreCase("ssid"))
SSID = value;
else if (name.equalsIgnoreCase("bssid"))
BSSID = value;
else if (name.equalsIgnoreCase("wpa_state"))
suppState = value;
}
info.setNetworkId(netId);
info.setSSID(SSID);
info.setBSSID(BSSID);
/*
* We only set the supplicant state if the previous state was
* UNINITIALIZED. This should only happen when we first connect to
* the supplicant. Once we're connected, we should always receive
* an event upon any state change, but in this case, we want to
* make sure any listeners are made aware of the state change.
*/
if (mWifiInfo.getSupplicantState() == SupplicantState.UNINITIALIZED && suppState != null)
setSupplicantState(suppState);
}
/**
* Get the dynamic information that is not reported via events.
* @param info the object into which the information should be captured.
*/
private synchronized void requestPolledInfo(WifiInfo info, boolean polling)
{
int newRssi = WifiNative.getRssiCommand();
if (newRssi != -1 && -200 < newRssi && newRssi < 256) { // screen out invalid values
/* some implementations avoid negative values by adding 256
* so we need to adjust for that here.
*/
if (newRssi > 0) newRssi -= 256;
info.setRssi(newRssi);
/*
* Rather then sending the raw RSSI out every time it
* changes, we precalculate the signal level that would
* be displayed in the status bar, and only send the
* broadcast if that much more coarse-grained number
* changes. This cuts down greatly on the number of
* broadcasts, at the cost of not informing others
* interested in RSSI of all the changes in signal
* level.
*/
// TODO: The second arg to the call below needs to be a symbol somewhere, but
// it's actually the size of an array of icons that's private
// to StatusBar Policy.
int newSignalLevel = WifiManager.calculateSignalLevel(newRssi, 4);
if (newSignalLevel != mLastSignalLevel) {
sendRssiChangeBroadcast(newRssi);
}
mLastSignalLevel = newSignalLevel;
} else {
info.setRssi(-200);
}
int newLinkSpeed = WifiNative.getLinkSpeedCommand();
if (newLinkSpeed != -1) {
info.setLinkSpeed(newLinkSpeed);
}
}
private void sendRssiChangeBroadcast(final int newRssi) {
if (ActivityManagerNative.isSystemReady()) {
Intent intent = new Intent(WifiManager.RSSI_CHANGED_ACTION);
intent.putExtra(WifiManager.EXTRA_NEW_RSSI, newRssi);
mContext.sendBroadcast(intent);
}
}
private void sendNetworkStateChangeBroadcast(String bssid) {
Intent intent = new Intent(WifiManager.NETWORK_STATE_CHANGED_ACTION);
intent.addFlags(Intent.FLAG_RECEIVER_REGISTERED_ONLY_BEFORE_BOOT);
intent.putExtra(WifiManager.EXTRA_NETWORK_INFO, mNetworkInfo);
if (bssid != null)
intent.putExtra(WifiManager.EXTRA_BSSID, bssid);
mContext.sendStickyBroadcast(intent);
}
/**
* Disable Wi-Fi connectivity by stopping the driver.
*/
public boolean teardown() {
if (!mTornDownByConnMgr) {
if (disconnectAndStop()) {
setTornDownByConnMgr(true);
return true;
} else {
return false;
}
} else {
return true;
}
}
/**
* Reenable Wi-Fi connectivity by restarting the driver.
*/
public boolean reconnect() {
if (mTornDownByConnMgr) {
if (restart()) {
setTornDownByConnMgr(false);
return true;
} else {
return false;
}
} else {
return true;
}
}
/**
* We want to stop the driver, but if we're connected to a network,
* we first want to disconnect, so that the supplicant is always in
* a known state (DISCONNECTED) when the driver is stopped.
* @return {@code true} if the operation succeeds, which means that the
* disconnect or stop command was initiated.
*/
public synchronized boolean disconnectAndStop() {
if (mRunState != RUN_STATE_STOPPING && mRunState != RUN_STATE_STOPPED) {
// Take down any open network notifications
setNotificationVisible(false, 0, false, 0);
mRunState = RUN_STATE_STOPPING;
if (mWifiInfo.getSupplicantState() == SupplicantState.DORMANT) {
return WifiNative.stopDriverCommand();
} else {
return WifiNative.disconnectCommand();
}
} else {
/*
* The "driver-stop" wake lock normally is released from the
* connectivity manager after the mobile data connection has
* been established, or after a timeout period, if that never
* happens. Because WifiService.updateWifiState() can get called
* multiple times, we can end up acquiring the wake lock and calling
* disconnectAndStop() even when a disconnect or stop operation
* is already in progress. In that case, we want to ignore the
* disconnectAndStop request and release the (ref-counted) wake
* lock, so that eventually, when the mobile data connection is
* established, the ref count will drop to zero.
*/
releaseWakeLock();
}
return true;
}
public synchronized boolean restart() {
if (mRunState == RUN_STATE_STOPPED) {
mRunState = RUN_STATE_STARTING;
return WifiNative.startDriverCommand();
} else if (mRunState == RUN_STATE_STOPPING) {
mRunState = RUN_STATE_STARTING;
}
return true;
}
public synchronized boolean removeNetwork(int networkId) {
return mDisconnectExpected = WifiNative.removeNetworkCommand(networkId);
}
public boolean setRadio(boolean turnOn) {
return mWM.setWifiEnabled(turnOn);
}
/**
* {@inheritDoc}
* There are currently no Wi-Fi-specific features supported.
* @param feature the name of the feature
* @return {@code -1} indicating failure, always
*/
public int startUsingNetworkFeature(String feature, int callingPid, int callingUid) {
return -1;
}
/**
* {@inheritDoc}
* There are currently no Wi-Fi-specific features supported.
* @param feature the name of the feature
* @return {@code -1} indicating failure, always
*/
public int stopUsingNetworkFeature(String feature, int callingPid, int callingUid) {
return -1;
}
@Override
public void interpretScanResultsAvailable() {
// If we shouldn't place a notification on available networks, then
// don't bother doing any of the following
if (!mNotificationEnabled) return;
NetworkInfo networkInfo = getNetworkInfo();
State state = networkInfo.getState();
if ((state == NetworkInfo.State.DISCONNECTED)
|| (state == NetworkInfo.State.UNKNOWN)) {
// Look for an open network
List<ScanResult> scanResults = getScanResultsList();
if (scanResults != null) {
int numOpenNetworks = 0;
for (int i = scanResults.size() - 1; i >= 0; i--) {
ScanResult scanResult = scanResults.get(i);
if (TextUtils.isEmpty(scanResult.capabilities)) {
numOpenNetworks++;
}
}
if (numOpenNetworks > 0) {
if (++mNumScansSinceNetworkStateChange >= NUM_SCANS_BEFORE_ACTUALLY_SCANNING) {
/*
* We've scanned continuously at least
* NUM_SCANS_BEFORE_NOTIFICATION times. The user
* probably does not have a remembered network in range,
* since otherwise supplicant would have tried to
* associate and thus resetting this counter.
*/
setNotificationVisible(true, numOpenNetworks, false, 0);
}
return;
}
}
}
// No open networks in range, remove the notification
setNotificationVisible(false, 0, false, 0);
}
/**
* Display or don't display a notification that there are open Wi-Fi networks.
* @param visible {@code true} if notification should be visible, {@code false} otherwise
* @param numNetworks the number networks seen
* @param force {@code true} to force notification to be shown/not-shown,
* even if it is already shown/not-shown.
* @param delay time in milliseconds after which the notification should be made
* visible or invisible.
*/
public void setNotificationVisible(boolean visible, int numNetworks, boolean force, int delay) {
// Since we use auto cancel on the notification, when the
// mNetworksAvailableNotificationShown is true, the notification may
// have actually been canceled. However, when it is false we know
// for sure that it is not being shown (it will not be shown any other
// place than here)
// If it should be hidden and it is already hidden, then noop
if (!visible && !mNotificationShown && !force) {
return;
}
Message message;
if (visible) {
// Not enough time has passed to show the notification again
if (System.currentTimeMillis() < mNotificationRepeatTime) {
return;
}
if (mNotification == null) {
// Cache the Notification mainly so we can remove the
// EVENT_NOTIFICATION_CHANGED message with this Notification from
// the queue later
mNotification = new Notification();
mNotification.when = 0;
mNotification.icon = ICON_NETWORKS_AVAILABLE;
mNotification.flags = Notification.FLAG_AUTO_CANCEL;
mNotification.contentIntent = PendingIntent.getActivity(mContext, 0,
new Intent(WifiManager.ACTION_PICK_WIFI_NETWORK), 0);
}
CharSequence title = mContext.getResources().getQuantityText(
com.android.internal.R.plurals.wifi_available, numNetworks);
CharSequence details = mContext.getResources().getQuantityText(
com.android.internal.R.plurals.wifi_available_detailed, numNetworks);
mNotification.tickerText = title;
mNotification.setLatestEventInfo(mContext, title, details, mNotification.contentIntent);
mNotificationRepeatTime = System.currentTimeMillis() + NOTIFICATION_REPEAT_DELAY_MS;
message = mTarget.obtainMessage(EVENT_NOTIFICATION_CHANGED, 1,
ICON_NETWORKS_AVAILABLE, mNotification);
} else {
// Remove any pending messages to show the notification
mTarget.removeMessages(EVENT_NOTIFICATION_CHANGED, mNotification);
message = mTarget.obtainMessage(EVENT_NOTIFICATION_CHANGED, 0, ICON_NETWORKS_AVAILABLE);
}
mTarget.sendMessageDelayed(message, delay);
mNotificationShown = visible;
}
/**
* Clears variables related to tracking whether a notification has been
* shown recently.
* <p>
* After calling this method, the timer that prevents notifications from
* being shown too often will be cleared.
*/
private void resetNotificationTimer() {
mNotificationRepeatTime = 0;
mNumScansSinceNetworkStateChange = 0;
}
public synchronized boolean addToBlacklist(String bssid) {
return WifiNative.addToBlacklistCommand(bssid);
}
public synchronized boolean clearBlacklist() {
return WifiNative.clearBlacklistCommand();
}
@Override
public String toString() {
StringBuffer sb = new StringBuffer();
sb.append("interface ").append(mInterfaceName);
sb.append(" runState=");
if (mRunState >= 1 && mRunState <= mRunStateNames.length) {
sb.append(mRunStateNames[mRunState-1]);
} else {
sb.append(mRunState);
}
sb.append(LS).append(mWifiInfo).append(LS);
sb.append(mDhcpInfo).append(LS);
sb.append("haveIpAddress=").append(mHaveIpAddress).
append(", obtainingIpAddress=").append(mObtainingIpAddress).
append(", scanModeActive=").append(mIsScanModeActive).append(LS).
append("lastSignalLevel=").append(mLastSignalLevel).
append(", explicitlyDisabled=").append(mTornDownByConnMgr);
return sb.toString();
}
private class DhcpHandler extends Handler {
private Handler mTarget;
/**
* Whether to skip the DHCP result callback to the target. For example,
* this could be set if the network we were requesting an IP for has
* since been disconnected.
* <p>
* Note: There is still a chance where the client's intended DHCP
* request not being canceled. For example, we are request for IP on
* A, and he queues request for IP on B, and then cancels the request on
* B while we're still requesting from A.
*/
private boolean mCancelCallback;
/**
* Instance of the bluetooth headset helper. This needs to be created
* early because there is a delay before it actually 'connects', as
* noted by its javadoc. If we check before it is connected, it will be
* in an error state and we will not disable coexistence.
*/
private BluetoothHeadset mBluetoothHeadset;
public DhcpHandler(Looper looper, Handler target) {
super(looper);
mTarget = target;
mBluetoothHeadset = new BluetoothHeadset(mContext, null);
}
public void handleMessage(Message msg) {
int event;
switch (msg.what) {
case EVENT_DHCP_START:
boolean modifiedBluetoothCoexistenceMode = false;
if (shouldDisableCoexistenceMode()) {
/*
* There are problems setting the Wi-Fi driver's power
* mode to active when bluetooth coexistence mode is
* enabled or sense.
* <p>
* We set Wi-Fi to active mode when
* obtaining an IP address because we've found
* compatibility issues with some routers with low power
* mode.
* <p>
* In order for this active power mode to properly be set,
* we disable coexistence mode until we're done with
* obtaining an IP address. One exception is if we
* are currently connected to a headset, since disabling
* coexistence would interrupt that connection.
*/
modifiedBluetoothCoexistenceMode = true;
// Disable the coexistence mode
synchronized (WifiStateTracker.this) {
WifiNative.setBluetoothCoexistenceModeCommand(
WifiNative.BLUETOOTH_COEXISTENCE_MODE_DISABLED);
}
}
synchronized (WifiStateTracker.this) {
WifiNative.setPowerModeCommand(DRIVER_POWER_MODE_ACTIVE);
}
synchronized (this) {
// A new request is being made, so assume we will callback
mCancelCallback = false;
}
Log.d(TAG, "DhcpHandler: DHCP request started");
if (NetworkUtils.runDhcp(mInterfaceName, mDhcpInfo)) {
event = EVENT_INTERFACE_CONFIGURATION_SUCCEEDED;
if (LOCAL_LOGD) Log.v(TAG, "DhcpHandler: DHCP request succeeded");
} else {
event = EVENT_INTERFACE_CONFIGURATION_FAILED;
Log.i(TAG, "DhcpHandler: DHCP request failed: " +
NetworkUtils.getDhcpError());
}
synchronized (WifiStateTracker.this) {
WifiNative.setPowerModeCommand(DRIVER_POWER_MODE_AUTO);
}
if (modifiedBluetoothCoexistenceMode) {
// Set the coexistence mode back to its default value
synchronized (WifiStateTracker.this) {
WifiNative.setBluetoothCoexistenceModeCommand(
WifiNative.BLUETOOTH_COEXISTENCE_MODE_SENSE);
}
}
synchronized (this) {
if (!mCancelCallback) {
mTarget.sendEmptyMessage(event);
}
}
break;
}
}
public synchronized void setCancelCallback(boolean cancelCallback) {
mCancelCallback = cancelCallback;
}
/**
* Whether to disable coexistence mode while obtaining IP address. This
* logic will return true only if the current bluetooth
* headset/handsfree state is disconnected. This means if it is in an
* error state, we will NOT disable coexistence mode to err on the side
* of safety.
*
* @return Whether to disable coexistence mode.
*/
private boolean shouldDisableCoexistenceMode() {
int state = mBluetoothHeadset.getState();
return state == BluetoothHeadset.STATE_DISCONNECTED;
}
}
private void checkUseStaticIp() {
mUseStaticIp = false;
final ContentResolver cr = mContext.getContentResolver();
try {
if (Settings.System.getInt(cr, Settings.System.WIFI_USE_STATIC_IP) == 0) {
return;
}
} catch (Settings.SettingNotFoundException e) {
return;
}
try {
String addr = Settings.System.getString(cr, Settings.System.WIFI_STATIC_IP);
if (addr != null) {
mDhcpInfo.ipAddress = stringToIpAddr(addr);
} else {
return;
}
addr = Settings.System.getString(cr, Settings.System.WIFI_STATIC_GATEWAY);
if (addr != null) {
mDhcpInfo.gateway = stringToIpAddr(addr);
} else {
return;
}
addr = Settings.System.getString(cr, Settings.System.WIFI_STATIC_NETMASK);
if (addr != null) {
mDhcpInfo.netmask = stringToIpAddr(addr);
} else {
return;
}
addr = Settings.System.getString(cr, Settings.System.WIFI_STATIC_DNS1);
if (addr != null) {
mDhcpInfo.dns1 = stringToIpAddr(addr);
} else {
return;
}
addr = Settings.System.getString(cr, Settings.System.WIFI_STATIC_DNS2);
if (addr != null) {
mDhcpInfo.dns2 = stringToIpAddr(addr);
} else {
mDhcpInfo.dns2 = 0;
}
} catch (UnknownHostException e) {
return;
}
mUseStaticIp = true;
}
private static int stringToIpAddr(String addrString) throws UnknownHostException {
try {
String[] parts = addrString.split("\\.");
if (parts.length != 4) {
throw new UnknownHostException(addrString);
}
int a = Integer.parseInt(parts[0]) ;
int b = Integer.parseInt(parts[1]) << 8;
int c = Integer.parseInt(parts[2]) << 16;
int d = Integer.parseInt(parts[3]) << 24;
return a | b | c | d;
} catch (NumberFormatException ex) {
throw new UnknownHostException(addrString);
}
}
private int getMaxDhcpRetries() {
return Settings.Secure.getInt(mContext.getContentResolver(),
Settings.Secure.WIFI_MAX_DHCP_RETRY_COUNT,
DEFAULT_MAX_DHCP_RETRIES);
}
private class SettingsObserver extends ContentObserver {
public SettingsObserver(Handler handler) {
super(handler);
ContentResolver cr = mContext.getContentResolver();
cr.registerContentObserver(Settings.System.getUriFor(
Settings.System.WIFI_USE_STATIC_IP), false, this);
cr.registerContentObserver(Settings.System.getUriFor(
Settings.System.WIFI_STATIC_IP), false, this);
cr.registerContentObserver(Settings.System.getUriFor(
Settings.System.WIFI_STATIC_GATEWAY), false, this);
cr.registerContentObserver(Settings.System.getUriFor(
Settings.System.WIFI_STATIC_NETMASK), false, this);
cr.registerContentObserver(Settings.System.getUriFor(
Settings.System.WIFI_STATIC_DNS1), false, this);
cr.registerContentObserver(Settings.System.getUriFor(
Settings.System.WIFI_STATIC_DNS2), false, this);
}
public void onChange(boolean selfChange) {
super.onChange(selfChange);
boolean wasStaticIp = mUseStaticIp;
int oIp, oGw, oMsk, oDns1, oDns2;
oIp = oGw = oMsk = oDns1 = oDns2 = 0;
if (wasStaticIp) {
oIp = mDhcpInfo.ipAddress;
oGw = mDhcpInfo.gateway;
oMsk = mDhcpInfo.netmask;
oDns1 = mDhcpInfo.dns1;
oDns2 = mDhcpInfo.dns2;
}
checkUseStaticIp();
if (mWifiInfo.getSupplicantState() == SupplicantState.UNINITIALIZED) {
return;
}
boolean changed =
(wasStaticIp != mUseStaticIp) ||
(wasStaticIp && (
oIp != mDhcpInfo.ipAddress ||
oGw != mDhcpInfo.gateway ||
oMsk != mDhcpInfo.netmask ||
oDns1 != mDhcpInfo.dns1 ||
oDns2 != mDhcpInfo.dns2));
if (changed) {
resetInterface();
configureInterface();
if (mUseStaticIp) {
mTarget.sendEmptyMessage(EVENT_CONFIGURATION_CHANGED);
}
}
}
}
private class NotificationEnabledSettingObserver extends ContentObserver {
public NotificationEnabledSettingObserver(Handler handler) {
super(handler);
}
public void register() {
ContentResolver cr = mContext.getContentResolver();
cr.registerContentObserver(Settings.Secure.getUriFor(
Settings.Secure.WIFI_NETWORKS_AVAILABLE_NOTIFICATION_ON), true, this);
mNotificationEnabled = getValue();
}
@Override
public void onChange(boolean selfChange) {
super.onChange(selfChange);
mNotificationEnabled = getValue();
if (!mNotificationEnabled) {
// Remove any notification that may be showing
setNotificationVisible(false, 0, true, 0);
}
resetNotificationTimer();
}
private boolean getValue() {
return Settings.Secure.getInt(mContext.getContentResolver(),
Settings.Secure.WIFI_NETWORKS_AVAILABLE_NOTIFICATION_ON, 1) == 1;
}
}
}