Merge "Unify code paths for collapsing action views in action bars" into lmp-dev
diff --git a/api/current.txt b/api/current.txt
index 5159a3a..5b8ba4d 100644
--- a/api/current.txt
+++ b/api/current.txt
@@ -28490,12 +28490,12 @@
method public int describeContents();
method public void writeToParcel(android.os.Parcel, int);
field public static final android.os.Parcelable.Creator CREATOR;
- field public static int ROUTE_ALL;
- field public static int ROUTE_BLUETOOTH;
- field public static int ROUTE_EARPIECE;
- field public static int ROUTE_SPEAKER;
- field public static int ROUTE_WIRED_HEADSET;
- field public static int ROUTE_WIRED_OR_EARPIECE;
+ field public static final int ROUTE_ALL = 15; // 0xf
+ field public static final int ROUTE_BLUETOOTH = 2; // 0x2
+ field public static final int ROUTE_EARPIECE = 1; // 0x1
+ field public static final int ROUTE_SPEAKER = 8; // 0x8
+ field public static final int ROUTE_WIRED_HEADSET = 4; // 0x4
+ field public static final int ROUTE_WIRED_OR_EARPIECE = 5; // 0x5
field public final boolean isMuted;
field public final int route;
field public final int supportedRouteMask;
diff --git a/core/java/android/app/AppOpsManager.java b/core/java/android/app/AppOpsManager.java
index 0d2ad08..66928ca 100644
--- a/core/java/android/app/AppOpsManager.java
+++ b/core/java/android/app/AppOpsManager.java
@@ -16,25 +16,25 @@
package android.app;
-import android.Manifest;
+import android.annotation.SystemApi;
+import android.app.usage.UsageStatsManager;
+import android.content.Context;
import android.media.AudioAttributes.AttributeUsage;
import android.os.Binder;
import android.os.IBinder;
-import android.os.UserManager;
-import android.util.ArrayMap;
-
-import com.android.internal.app.IAppOpsService;
-import com.android.internal.app.IAppOpsCallback;
-
-import java.util.ArrayList;
-import java.util.HashMap;
-import java.util.List;
-
-import android.content.Context;
import android.os.Parcel;
import android.os.Parcelable;
import android.os.Process;
import android.os.RemoteException;
+import android.os.UserManager;
+import android.util.ArrayMap;
+
+import com.android.internal.app.IAppOpsCallback;
+import com.android.internal.app.IAppOpsService;
+
+import java.util.ArrayList;
+import java.util.HashMap;
+import java.util.List;
/**
* API for interacting with "application operation" tracking.
@@ -203,8 +203,10 @@
public static final int OP_TOAST_WINDOW = 45;
/** @hide Capture the device's display contents and/or audio */
public static final int OP_PROJECT_MEDIA = 46;
+ /** @hide Activate a VPN connection without user intervention. */
+ public static final int OP_ACTIVATE_VPN = 47;
/** @hide */
- public static final int _NUM_OP = 47;
+ public static final int _NUM_OP = 48;
/** Access to coarse location information. */
public static final String OPSTR_COARSE_LOCATION =
@@ -218,6 +220,9 @@
/** Continually monitoring location data with a relatively high power request. */
public static final String OPSTR_MONITOR_HIGH_POWER_LOCATION
= "android:monitor_location_high_power";
+ /** Activate a VPN connection without user intervention. @hide */
+ @SystemApi
+ public static final String OPSTR_ACTIVATE_VPN = "android:activate_vpn";
/**
* This maps each operation to the operation that serves as the
@@ -275,6 +280,7 @@
OP_MUTE_MICROPHONE,
OP_TOAST_WINDOW,
OP_PROJECT_MEDIA,
+ OP_ACTIVATE_VPN,
};
/**
@@ -329,6 +335,7 @@
null,
null,
null,
+ OPSTR_ACTIVATE_VPN,
};
/**
@@ -383,6 +390,7 @@
"MUTE_MICROPHONE",
"TOAST_WINDOW",
"PROJECT_MEDIA",
+ "ACTIVATE_VPN",
};
/**
@@ -437,6 +445,7 @@
null, // no permission for muting/unmuting microphone
null, // no permission for displaying toasts
null, // no permission for projecting media
+ null, // no permission for activating vpn
};
/**
@@ -492,6 +501,7 @@
UserManager.DISALLOW_UNMUTE_MICROPHONE, // MUTE_MICROPHONE
UserManager.DISALLOW_CREATE_WINDOWS, // TOAST_WINDOW
null, //PROJECT_MEDIA
+ UserManager.DISALLOW_CONFIG_VPN, // ACTIVATE_VPN
};
/**
@@ -546,6 +556,7 @@
false, //MUTE_MICROPHONE
true, //TOAST_WINDOW
false, //PROJECT_MEDIA
+ false, //ACTIVATE_VPN
};
/**
@@ -599,6 +610,7 @@
AppOpsManager.MODE_ALLOWED,
AppOpsManager.MODE_ALLOWED,
AppOpsManager.MODE_IGNORED, // OP_PROJECT_MEDIA
+ AppOpsManager.MODE_IGNORED, // OP_ACTIVATE_VPN
};
/**
@@ -656,6 +668,7 @@
false,
false,
false,
+ false,
};
private static HashMap<String, Integer> sOpStrToOp = new HashMap<String, Integer>();
diff --git a/core/java/android/net/IConnectivityManager.aidl b/core/java/android/net/IConnectivityManager.aidl
index 0814e0f..b2fc3be 100644
--- a/core/java/android/net/IConnectivityManager.aidl
+++ b/core/java/android/net/IConnectivityManager.aidl
@@ -108,6 +108,8 @@
boolean prepareVpn(String oldPackage, String newPackage);
+ void setVpnPackageAuthorization(boolean authorized);
+
ParcelFileDescriptor establishVpn(in VpnConfig config);
VpnConfig getVpnConfig();
diff --git a/core/java/com/android/internal/net/LegacyVpnInfo.java b/core/java/com/android/internal/net/LegacyVpnInfo.java
index d6f6d0b..f812ad6 100644
--- a/core/java/com/android/internal/net/LegacyVpnInfo.java
+++ b/core/java/com/android/internal/net/LegacyVpnInfo.java
@@ -40,7 +40,6 @@
public String key;
public int state = -1;
- public PendingIntent intent;
@Override
public int describeContents() {
@@ -51,7 +50,6 @@
public void writeToParcel(Parcel out, int flags) {
out.writeString(key);
out.writeInt(state);
- out.writeParcelable(intent, flags);
}
public static final Parcelable.Creator<LegacyVpnInfo> CREATOR =
@@ -61,7 +59,6 @@
LegacyVpnInfo info = new LegacyVpnInfo();
info.key = in.readString();
info.state = in.readInt();
- info.intent = in.readParcelable(null);
return info;
}
diff --git a/core/java/com/android/internal/net/VpnConfig.java b/core/java/com/android/internal/net/VpnConfig.java
index 0099269..aa66d7d 100644
--- a/core/java/com/android/internal/net/VpnConfig.java
+++ b/core/java/com/android/internal/net/VpnConfig.java
@@ -20,17 +20,19 @@
import android.content.ComponentName;
import android.content.Context;
import android.content.Intent;
+import android.content.pm.PackageManager;
+import android.content.pm.PackageManager.NameNotFoundException;
+import android.content.pm.ResolveInfo;
import android.content.res.Resources;
+import android.net.LinkAddress;
+import android.net.RouteInfo;
import android.os.Parcel;
import android.os.Parcelable;
-import android.os.UserHandle;
-import android.net.RouteInfo;
-import android.net.LinkAddress;
import java.net.Inet4Address;
import java.net.InetAddress;
-import java.util.List;
import java.util.ArrayList;
+import java.util.List;
/**
* A simple container used to carry information in VpnBuilder, VpnDialogs,
@@ -55,12 +57,19 @@
return intent;
}
- public static PendingIntent getIntentForStatusPanel(Context context) {
- Intent intent = new Intent();
- intent.setClassName(DIALOGS_PACKAGE, DIALOGS_PACKAGE + ".ManageDialog");
- intent.addFlags(Intent.FLAG_ACTIVITY_NEW_TASK | Intent.FLAG_ACTIVITY_NO_HISTORY |
- Intent.FLAG_ACTIVITY_EXCLUDE_FROM_RECENTS);
- return PendingIntent.getActivityAsUser(context, 0, intent, 0, null, UserHandle.CURRENT);
+ public static CharSequence getVpnLabel(Context context, String packageName)
+ throws NameNotFoundException {
+ PackageManager pm = context.getPackageManager();
+ Intent intent = new Intent(SERVICE_INTERFACE);
+ intent.setPackage(packageName);
+ List<ResolveInfo> services = pm.queryIntentServices(intent, 0 /* flags */);
+ if (services != null && services.size() == 1) {
+ // This app contains exactly one VPN service. Call loadLabel, which will attempt to
+ // load the service's label, and fall back to the app label if none is present.
+ return services.get(0).loadLabel(pm);
+ } else {
+ return pm.getApplicationInfo(packageName, 0).loadLabel(pm);
+ }
}
public String user;
diff --git a/core/jni/android/graphics/BitmapFactory.cpp b/core/jni/android/graphics/BitmapFactory.cpp
index 2ce1b15..62ea351 100644
--- a/core/jni/android/graphics/BitmapFactory.cpp
+++ b/core/jni/android/graphics/BitmapFactory.cpp
@@ -18,6 +18,7 @@
#include <android_runtime/AndroidRuntime.h>
#include <androidfw/Asset.h>
#include <androidfw/ResourceTypes.h>
+#include <cutils/compiler.h>
#include <netinet/in.h>
#include <stdio.h>
#include <sys/mman.h>
@@ -87,27 +88,39 @@
return options != NULL && env->GetBooleanField(options, gOptions_justBoundsFieldID);
}
-static void scaleNinePatchChunk(android::Res_png_9patch* chunk, float scale) {
+static void scaleDivRange(int32_t* divs, int count, float scale, int maxValue) {
+ for (int i = 0; i < count; i++) {
+ divs[i] = int32_t(divs[i] * scale + 0.5f);
+ if (i > 0 && divs[i] == divs[i - 1]) {
+ divs[i]++; // avoid collisions
+ }
+ }
+
+ if (CC_UNLIKELY(divs[count - 1] > maxValue)) {
+ // if the collision avoidance above put some divs outside the bounds of the bitmap,
+ // slide outer stretchable divs inward to stay within bounds
+ int highestAvailable = maxValue;
+ for (int i = count - 1; i >= 0; i--) {
+ divs[i] = highestAvailable;
+ if (i > 0 && divs[i] <= divs[i-1]){
+ // keep shifting
+ highestAvailable = divs[i] - 1;
+ } else {
+ break;
+ }
+ }
+ }
+}
+
+static void scaleNinePatchChunk(android::Res_png_9patch* chunk, float scale,
+ int scaledWidth, int scaledHeight) {
chunk->paddingLeft = int(chunk->paddingLeft * scale + 0.5f);
chunk->paddingTop = int(chunk->paddingTop * scale + 0.5f);
chunk->paddingRight = int(chunk->paddingRight * scale + 0.5f);
chunk->paddingBottom = int(chunk->paddingBottom * scale + 0.5f);
- int32_t* xDivs = chunk->getXDivs();
- for (int i = 0; i < chunk->numXDivs; i++) {
- xDivs[i] = int32_t(xDivs[i] * scale + 0.5f);
- if (i > 0 && xDivs[i] == xDivs[i - 1]) {
- xDivs[i]++;
- }
- }
-
- int32_t* yDivs = chunk->getYDivs();
- for (int i = 0; i < chunk->numYDivs; i++) {
- yDivs[i] = int32_t(yDivs[i] * scale + 0.5f);
- if (i > 0 && yDivs[i] == yDivs[i - 1]) {
- yDivs[i]++;
- }
- }
+ scaleDivRange(chunk->getXDivs(), chunk->numXDivs, scale, scaledWidth);
+ scaleDivRange(chunk->getYDivs(), chunk->numYDivs, scale, scaledHeight);
}
static SkColorType colorTypeForScaledOutput(SkColorType colorType) {
@@ -330,7 +343,7 @@
jbyteArray ninePatchChunk = NULL;
if (peeker.mPatch != NULL) {
if (willScale) {
- scaleNinePatchChunk(peeker.mPatch, scale);
+ scaleNinePatchChunk(peeker.mPatch, scale, scaledWidth, scaledHeight);
}
size_t ninePatchArraySize = peeker.mPatch->serializedSize();
diff --git a/core/tests/ConnectivityManagerTest/src/com/android/connectivitymanagertest/AccessPointParserHelper.java b/core/tests/ConnectivityManagerTest/src/com/android/connectivitymanagertest/AccessPointParserHelper.java
index 93b16e7..1222c8b 100644
--- a/core/tests/ConnectivityManagerTest/src/com/android/connectivitymanagertest/AccessPointParserHelper.java
+++ b/core/tests/ConnectivityManagerTest/src/com/android/connectivitymanagertest/AccessPointParserHelper.java
@@ -66,7 +66,6 @@
* networkprefixlength.
*/
public class AccessPointParserHelper {
- private static final String TAG = "AccessPointParserHelper";
static final int NONE = 0;
static final int WEP = 1;
static final int PSK = 2;
diff --git a/core/tests/ConnectivityManagerTest/src/com/android/connectivitymanagertest/ConnectivityManagerStressTestRunner.java b/core/tests/ConnectivityManagerTest/src/com/android/connectivitymanagertest/ConnectivityManagerStressTestRunner.java
index 3ec9031..fbaf0f3 100644
--- a/core/tests/ConnectivityManagerTest/src/com/android/connectivitymanagertest/ConnectivityManagerStressTestRunner.java
+++ b/core/tests/ConnectivityManagerTest/src/com/android/connectivitymanagertest/ConnectivityManagerStressTestRunner.java
@@ -35,14 +35,14 @@
*/
public class ConnectivityManagerStressTestRunner extends InstrumentationTestRunner {
- public int mSoftapIterations = 100;
- public int mScanIterations = 100;
- public int mReconnectIterations = 100;
+ private int mSoftApIterations = 100;
+ private int mScanIterations = 100;
+ private int mReconnectIterations = 100;
// sleep time before restart wifi, default is set to 2 minutes
- public int mSleepTime = 2 * 60 * 1000;
- public String mReconnectSsid = "securenetdhcp";
- public String mReconnectPassword = "androidwifi";
- public boolean mWifiOnlyFlag = false;
+ private long mSleepTime = 2 * 60 * 1000;
+ private String mReconnectSsid = null;
+ private String mReconnectPassword = null;
+ private boolean mWifiOnlyFlag = false;
@Override
public TestSuite getAllTests() {
@@ -60,15 +60,15 @@
@Override
public void onCreate(Bundle icicle) {
super.onCreate(icicle);
- String valueStr = (String) icicle.get("softap_iterations");
+ String valueStr = icicle.getString("softap_iterations");
if (valueStr != null) {
int iteration = Integer.parseInt(valueStr);
if (iteration > 0) {
- mSoftapIterations = iteration;
+ mSoftApIterations = iteration;
}
}
- String scanIterationStr = (String) icicle.get("scan_iterations");
+ String scanIterationStr = icicle.getString("scan_iterations");
if (scanIterationStr != null) {
int scanIteration = Integer.parseInt(scanIterationStr);
if (scanIteration > 0) {
@@ -76,17 +76,17 @@
}
}
- String ssidStr= (String) icicle.get("reconnect_ssid");
+ String ssidStr= icicle.getString("reconnect_ssid");
if (ssidStr != null) {
mReconnectSsid = ssidStr;
}
- String passwordStr = (String) icicle.get("reconnect_password");
+ String passwordStr = icicle.getString("reconnect_password");
if (passwordStr != null) {
mReconnectPassword = passwordStr;
}
- String reconnectStr = (String) icicle.get("reconnect_iterations");
+ String reconnectStr = icicle.getString("reconnect_iterations");
if (reconnectStr != null) {
int iteration = Integer.parseInt(reconnectStr);
if (iteration > 0) {
@@ -94,7 +94,7 @@
}
}
- String sleepTimeStr = (String) icicle.get("sleep_time");
+ String sleepTimeStr = icicle.getString("sleep_time");
if (sleepTimeStr != null) {
int sleepTime = Integer.parseInt(sleepTimeStr);
if (sleepTime > 0) {
@@ -102,9 +102,37 @@
}
}
- String wifiOnlyFlag = (String) icicle.get("wifi-only");
+ String wifiOnlyFlag = icicle.getString("wifi-only");
if (wifiOnlyFlag != null) {
mWifiOnlyFlag = true;
}
}
+
+ public int getSoftApInterations() {
+ return mSoftApIterations;
+ }
+
+ public int getScanIterations() {
+ return mScanIterations;
+ }
+
+ public int getReconnectIterations() {
+ return mReconnectIterations;
+ }
+
+ public boolean isWifiOnly() {
+ return mWifiOnlyFlag;
+ }
+
+ public long getSleepTime() {
+ return mSleepTime;
+ }
+
+ public String getReconnectSsid() {
+ return mReconnectSsid;
+ }
+
+ public String getReconnectPassword() {
+ return mReconnectPassword;
+ }
}
diff --git a/core/tests/ConnectivityManagerTest/src/com/android/connectivitymanagertest/ConnectivityManagerTestBase.java b/core/tests/ConnectivityManagerTest/src/com/android/connectivitymanagertest/ConnectivityManagerTestBase.java
index 30eda75..a35a8ee 100644
--- a/core/tests/ConnectivityManagerTest/src/com/android/connectivitymanagertest/ConnectivityManagerTestBase.java
+++ b/core/tests/ConnectivityManagerTest/src/com/android/connectivitymanagertest/ConnectivityManagerTestBase.java
@@ -17,31 +17,28 @@
package com.android.connectivitymanagertest;
import android.app.KeyguardManager;
-import android.content.Context;
import android.content.BroadcastReceiver;
+import android.content.Context;
import android.content.Intent;
import android.content.IntentFilter;
import android.net.ConnectivityManager;
import android.net.NetworkInfo;
import android.net.NetworkInfo.State;
+import android.net.wifi.ScanResult;
import android.net.wifi.WifiConfiguration;
-import android.net.wifi.WifiManager;
import android.net.wifi.WifiConfiguration.KeyMgmt;
-import android.os.Handler;
-import android.os.Message;
+import android.net.wifi.WifiManager;
import android.os.PowerManager;
import android.os.SystemClock;
import android.test.InstrumentationTestCase;
import android.util.Log;
import android.view.KeyEvent;
-import com.android.internal.util.AsyncChannel;
-
import java.io.IOException;
import java.io.InputStream;
import java.net.UnknownHostException;
-import java.util.ArrayList;
import java.util.List;
+import java.util.regex.Pattern;
/**
@@ -55,51 +52,39 @@
*/
public class ConnectivityManagerTestBase extends InstrumentationTestCase {
- public static final String LOG_TAG = "ConnectivityManagerTestBase";
- public static final int WAIT_FOR_SCAN_RESULT = 10 * 1000; //10 seconds
- public static final int WIFI_SCAN_TIMEOUT = 50 * 1000; // 50 seconds
- public static final int SHORT_TIMEOUT = 5 * 1000; // 5 seconds
- public static final long LONG_TIMEOUT = 50 * 1000; // 50 seconds
- public static final long WIFI_CONNECTION_TIMEOUT = 5 * 60 * 1000; // 5 minutes
- // 2 minutes timer between wifi stop and start
- public static final long WIFI_STOP_START_INTERVAL = 2 * 60 * 1000; // 2 minutes
- // Set ping test timer to be 3 minutes
- public static final long PING_TIMER = 3 * 60 *1000; // 3 minutes
- public static final int SUCCESS = 0; // for Wifi tethering state change
- public static final int FAILURE = 1;
- public static final int INIT = -1;
+ private static final String LOG_TAG = "ConnectivityManagerTestBase";
private static final String ACCESS_POINT_FILE = "accesspoints.xml";
- public ConnectivityReceiver mConnectivityReceiver = null;
- public WifiReceiver mWifiReceiver = null;
+ private static final String PING_IP_ADDR = "8.8.8.8";
+
+ protected static final int WAIT_FOR_SCAN_RESULT = 10 * 1000; //10 seconds
+ protected static final int WIFI_SCAN_TIMEOUT = 50 * 1000; // 50 seconds
+ protected static final int SHORT_TIMEOUT = 5 * 1000; // 5 seconds
+ protected static final long LONG_TIMEOUT = 50 * 1000; // 50 seconds
+ protected static final long WIFI_CONNECTION_TIMEOUT = 5 * 60 * 1000; // 5 minutes
+ // 2 minutes timer between wifi stop and start
+ protected static final long WIFI_STOP_START_INTERVAL = 2 * 60 * 1000; // 2 minutes
+ // Set ping test timer to be 3 minutes
+ protected static final long PING_TIMER = 3 * 60 *1000; // 3 minutes
+ protected static final int SUCCESS = 0; // for Wifi tethering state change
+ protected static final int FAILURE = 1;
+ protected static final int INIT = -1;
+
+ private ConnectivityReceiver mConnectivityReceiver = null;
+ private WifiReceiver mWifiReceiver = null;
private AccessPointParserHelper mParseHelper = null;
- /*
- * Track network connectivity information
- */
- public State mState;
- public NetworkInfo mNetworkInfo;
- public NetworkInfo mOtherNetworkInfo;
- public boolean mIsFailOver;
- public String mReason;
- public boolean mScanResultIsAvailable = false;
- public ConnectivityManager mCM;
- public Object wifiObject = new Object();
- public Object connectivityObject = new Object();
- public int mWifiState;
- public NetworkInfo mWifiNetworkInfo;
- public String mBssid;
- public String mPowerSsid = "opennet"; //Default power SSID
+
+ private long mLastConnectivityChangeTime = -1;
+ protected ConnectivityManager mCm;
private Context mContext;
- public boolean scanResultAvailable = false;
+ protected List<ScanResult> mLastScanResult;
+ protected Object mWifiScanResultLock = new Object();
/* Control Wifi States */
public WifiManager mWifiManager;
- /* Verify connectivity state */
- public static final int NUM_NETWORK_TYPES = ConnectivityManager.MAX_NETWORK_TYPE + 1;
- NetworkState[] connectivityState = new NetworkState[NUM_NETWORK_TYPES];
- // For wifi tethering tests
- private String[] mWifiRegexs;
- public int mWifiTetherResult = INIT; // -1 is initialization state
+ protected long getLastConnectivityChangeTime() {
+ return mLastConnectivityChangeTime;
+ }
/**
* A wrapper of a broadcast receiver which provides network connectivity information
@@ -108,40 +93,12 @@
private class ConnectivityReceiver extends BroadcastReceiver {
@Override
public void onReceive(Context context, Intent intent) {
- log("ConnectivityReceiver: onReceive() is called with " + intent);
+ mLastConnectivityChangeTime = SystemClock.uptimeMillis();
+ log("ConnectivityReceiver: " + intent);
String action = intent.getAction();
if (!action.equals(ConnectivityManager.CONNECTIVITY_ACTION)) {
Log.v("ConnectivityReceiver", "onReceive() called with " + intent);
- return;
}
-
- boolean noConnectivity =
- intent.getBooleanExtra(ConnectivityManager.EXTRA_NO_CONNECTIVITY, false);
-
- if (noConnectivity) {
- mState = State.DISCONNECTED;
- } else {
- mState = State.CONNECTED;
- }
-
- mNetworkInfo = (NetworkInfo)
- intent.getParcelableExtra(ConnectivityManager.EXTRA_NETWORK_INFO);
-
- mOtherNetworkInfo = (NetworkInfo)
- intent.getParcelableExtra(ConnectivityManager.EXTRA_OTHER_NETWORK_INFO);
-
- mReason = intent.getStringExtra(ConnectivityManager.EXTRA_REASON);
- mIsFailOver = intent.getBooleanExtra(ConnectivityManager.EXTRA_IS_FAILOVER, false);
-
- log("mNetworkInfo: " + mNetworkInfo.toString());
- if (mOtherNetworkInfo != null) {
- log("mOtherNetworkInfo: " + mOtherNetworkInfo.toString());
- }
- recordNetworkState(mNetworkInfo.getType(), mNetworkInfo.getState());
- if (mOtherNetworkInfo != null) {
- recordNetworkState(mOtherNetworkInfo.getType(), mOtherNetworkInfo.getState());
- }
- notifyNetworkConnectivityChange();
}
}
@@ -152,62 +109,21 @@
Log.v("WifiReceiver", "onReceive() is calleld with " + intent);
if (action.equals(WifiManager.SCAN_RESULTS_AVAILABLE_ACTION)) {
log("scan results are available");
- notifyScanResult();
- } else if (action.equals(WifiManager.NETWORK_STATE_CHANGED_ACTION)) {
- mWifiNetworkInfo =
- (NetworkInfo) intent.getParcelableExtra(WifiManager.EXTRA_NETWORK_INFO);
- log("mWifiNetworkInfo: " + mWifiNetworkInfo.toString());
- if (mWifiNetworkInfo.getState() == State.CONNECTED) {
- mBssid = intent.getStringExtra(WifiManager.EXTRA_BSSID);
+ synchronized (mWifiScanResultLock) {
+ mLastScanResult = mWifiManager.getScanResults();
+ mWifiScanResultLock.notifyAll();
}
- notifyWifiState();
- } else if (action.equals(WifiManager.WIFI_STATE_CHANGED_ACTION)) {
- mWifiState = intent.getIntExtra(WifiManager.EXTRA_WIFI_STATE,
- WifiManager.WIFI_STATE_UNKNOWN);
- notifyWifiState();
- } else if (action.equals(WifiManager.WIFI_AP_STATE_CHANGED_ACTION)) {
- notifyWifiAPState();
- } else if (action.equals(ConnectivityManager.ACTION_TETHER_STATE_CHANGED)) {
- ArrayList<String> available = intent.getStringArrayListExtra(
- ConnectivityManager.EXTRA_AVAILABLE_TETHER);
- ArrayList<String> active = intent.getStringArrayListExtra(
- ConnectivityManager.EXTRA_ACTIVE_TETHER);
- ArrayList<String> errored = intent.getStringArrayListExtra(
- ConnectivityManager.EXTRA_ERRORED_TETHER);
- updateTetherState(available.toArray(), active.toArray(), errored.toArray());
- }
- else {
- return;
- }
- }
- }
-
- private class WifiServiceHandler extends Handler {
- @Override
- public void handleMessage(Message msg) {
- switch (msg.what) {
- case AsyncChannel.CMD_CHANNEL_HALF_CONNECTED:
- if (msg.arg1 == AsyncChannel.STATUS_SUCCESSFUL) {
- //AsyncChannel in msg.obj
- } else {
- log("Failed to establish AsyncChannel connection");
- }
- break;
- default:
- //Ignore
- break;
}
}
}
@Override
- public void setUp() throws Exception {
- mState = State.UNKNOWN;
- scanResultAvailable = false;
+ protected void setUp() throws Exception {
+ mLastScanResult = null;
mContext = getInstrumentation().getContext();
// Get an instance of ConnectivityManager
- mCM = (ConnectivityManager)mContext.getSystemService(Context.CONNECTIVITY_SERVICE);
+ mCm = (ConnectivityManager)mContext.getSystemService(Context.CONNECTIVITY_SERVICE);
// Get an instance of WifiManager
mWifiManager =(WifiManager)mContext.getSystemService(Context.WIFI_SERVICE);
@@ -217,8 +133,6 @@
log("Disable soft ap");
}
- initializeNetworkStates();
-
// register a connectivity receiver for CONNECTIVITY_ACTION;
mConnectivityReceiver = new ConnectivityReceiver();
mContext.registerReceiver(mConnectivityReceiver,
@@ -236,211 +150,73 @@
log("Clear Wifi before we start the test.");
removeConfiguredNetworksAndDisableWifi();
- mWifiRegexs = mCM.getTetherableWifiRegexs();
}
- public List<WifiConfiguration> loadNetworkConfigurations() throws Exception {
+ protected List<WifiConfiguration> loadNetworkConfigurations() throws Exception {
InputStream in = mContext.getAssets().open(ACCESS_POINT_FILE);
mParseHelper = new AccessPointParserHelper(in);
return mParseHelper.getNetworkConfigurations();
}
- // for each network type, initialize network states to UNKNOWN, and no verification flag is set
- public void initializeNetworkStates() {
- for (int networkType = NUM_NETWORK_TYPES - 1; networkType >=0; networkType--) {
- connectivityState[networkType] = new NetworkState();
- log("Initialize network state for " + networkType + ": " +
- connectivityState[networkType].toString());
- }
- }
-
- // deposit a network state
- public void recordNetworkState(int networkType, State networkState) {
- log("record network state for network " + networkType +
- ", state is " + networkState);
- if (connectivityState == null) {
- log("ConnectivityState is null");
- }
- if (connectivityState[networkType] == null) {
- log("connectivityState[networkType] is null");
- }
- connectivityState[networkType].recordState(networkState);
- }
-
- // set the state transition criteria
- public void setStateTransitionCriteria(int networkType, State initState,
- int transitionDir, State targetState) {
- connectivityState[networkType].setStateTransitionCriteria(
- initState, transitionDir, targetState);
- }
-
- // Validate the states recorded
- public boolean validateNetworkStates(int networkType) {
- log("validate network state for " + networkType + ": ");
- return connectivityState[networkType].validateStateTransition();
- }
-
- // return result from network state validation
- public String getTransitionFailureReason(int networkType) {
- log("get network state transition failure reason for " + networkType + ": " +
- connectivityState[networkType].toString());
- return connectivityState[networkType].getReason();
- }
-
- private void notifyNetworkConnectivityChange() {
- synchronized(connectivityObject) {
- log("notify network connectivity changed");
- connectivityObject.notifyAll();
- }
- }
- private void notifyScanResult() {
- synchronized (this) {
- log("notify that scan results are available");
- scanResultAvailable = true;
- this.notify();
- }
- }
-
- private void notifyWifiState() {
- synchronized (wifiObject) {
- log("notify wifi state changed");
- wifiObject.notify();
- }
- }
-
- private void notifyWifiAPState() {
- synchronized (this) {
- log("notify wifi AP state changed");
- this.notify();
- }
- }
-
- // Update wifi tethering state
- private void updateTetherState(Object[] available, Object[] tethered, Object[] errored) {
- boolean wifiTethered = false;
- boolean wifiErrored = false;
-
- synchronized (this) {
- for (Object obj: tethered) {
- String str = (String)obj;
- for (String tethRex: mWifiRegexs) {
- log("str: " + str +"tethRex: " + tethRex);
- if (str.matches(tethRex)) {
- wifiTethered = true;
- }
- }
- }
-
- for (Object obj: errored) {
- String str = (String)obj;
- for (String tethRex: mWifiRegexs) {
- log("error: str: " + str +"tethRex: " + tethRex);
- if (str.matches(tethRex)) {
- wifiErrored = true;
- }
- }
- }
-
- if (wifiTethered) {
- mWifiTetherResult = SUCCESS; // wifi tethering is successful
- } else if (wifiErrored) {
- mWifiTetherResult = FAILURE; // wifi tethering failed
- }
- log("mWifiTetherResult: " + mWifiTetherResult);
- this.notify();
- }
- }
-
-
- // Wait for network connectivity state: CONNECTING, CONNECTED, SUSPENDED,
- // DISCONNECTING, DISCONNECTED, UNKNOWN
- public boolean waitForNetworkState(int networkType, State expectedState, long timeout) {
- long startTime = System.currentTimeMillis();
+ // wait for network connectivity state: CONNECTING, CONNECTED, SUSPENDED, DISCONNECTING,
+ // DISCONNECTED, UNKNOWN
+ protected boolean waitForNetworkState(int networkType, State expectedState, long timeout) {
+ long startTime = SystemClock.uptimeMillis();
while (true) {
- if ((System.currentTimeMillis() - startTime) > timeout) {
- log("waitForNetworkState time out, the state of network type " + networkType +
- " is: " + mCM.getNetworkInfo(networkType).getState());
- if (mCM.getNetworkInfo(networkType).getState() != expectedState) {
- return false;
- } else {
- // the broadcast has been sent out. the state has been changed.
- log("networktype: " + networkType + " state: " +
- mCM.getNetworkInfo(networkType));
- return true;
- }
- }
- log("Wait for the connectivity state for network: " + networkType +
- " to be " + expectedState.toString());
- synchronized (connectivityObject) {
- try {
- connectivityObject.wait(SHORT_TIMEOUT);
- } catch (InterruptedException e) {
- e.printStackTrace();
- }
- if ((mNetworkInfo.getType() != networkType) ||
- (mNetworkInfo.getState() != expectedState)) {
- log("network state for " + mNetworkInfo.getType() +
- "is: " + mNetworkInfo.getState());
- continue;
- }
+ NetworkInfo ni = mCm.getNetworkInfo(networkType);
+ String niString = ni == null ? "null" : ni.toString();
+ if (ni != null && expectedState.equals(ni.getState())) {
+ log("waitForNetworkState success: " + niString);
return true;
}
+ if ((SystemClock.uptimeMillis() - startTime) > timeout) {
+ log("waitForNetworkState timeout: " + niString);
+ return false;
+ }
+ log("waitForNetworkState interim: " + niString);
+ SystemClock.sleep(SHORT_TIMEOUT);
}
}
- // Wait for Wifi state: WIFI_STATE_DISABLED, WIFI_STATE_DISABLING, WIFI_STATE_ENABLED,
+ // wait for Wifi state: WIFI_STATE_DISABLED, WIFI_STATE_DISABLING, WIFI_STATE_ENABLED,
// WIFI_STATE_ENALBING, WIFI_STATE_UNKNOWN
- public boolean waitForWifiState(int expectedState, long timeout) {
- long startTime = System.currentTimeMillis();
+ protected boolean waitForWifiState(int expectedState, long timeout) {
+ long startTime = SystemClock.uptimeMillis();
while (true) {
- if ((System.currentTimeMillis() - startTime) > timeout) {
- if (mWifiState != expectedState) {
- return false;
- } else {
- return true;
- }
- }
- log("Wait for wifi state to be: " + expectedState);
- synchronized (wifiObject) {
- try {
- wifiObject.wait(SHORT_TIMEOUT);
- } catch (InterruptedException e) {
- e.printStackTrace();
- }
- if (mWifiState != expectedState) {
- log("Wifi state is: " + mWifiState);
- continue;
- }
+ int state = mWifiManager.getWifiState();
+ if (state == expectedState) {
+ log("waitForWifiState success: state=" + state);
return true;
}
+ if ((SystemClock.uptimeMillis() - startTime) > timeout) {
+ log(String.format("waitForWifiState timeout: expected=%d, actual=%d",
+ expectedState, state));
+ return false;
+ }
+ log(String.format("waitForWifiState interim: expected=%d, actual=%d",
+ expectedState, state));
+ SystemClock.sleep(SHORT_TIMEOUT);
}
}
// Wait for Wifi AP state: WIFI_AP_STATE_DISABLED, WIFI_AP_STATE_DISABLING,
// WIFI_AP_STATE_ENABLED, WIFI_STATE_ENALBING, WIFI_STATE_UNKNOWN
- public boolean waitForWifiAPState(int expectedState, long timeout) {
- long startTime = System.currentTimeMillis();
+ protected boolean waitForWifiApState(int expectedState, long timeout) {
+ long startTime = SystemClock.uptimeMillis();
while (true) {
- if ((System.currentTimeMillis() - startTime) > timeout) {
- if (mWifiManager.getWifiApState() != expectedState) {
- return false;
- } else {
- return true;
- }
- }
- log("Wait for wifi AP state to be: " + expectedState);
- synchronized (wifiObject) {
- try {
- wifiObject.wait(SHORT_TIMEOUT);
- } catch (InterruptedException e) {
- e.printStackTrace();
- }
- if (mWifiManager.getWifiApState() != expectedState) {
- log("Wifi state is: " + mWifiManager.getWifiApState());
- continue;
- }
+ int state = mWifiManager.getWifiApState();
+ if (state == expectedState) {
+ log("waitForWifiAPState success: state=" + state);
return true;
}
+ if ((SystemClock.uptimeMillis() - startTime) > timeout) {
+ log(String.format("waitForWifiAPState timeout: expected=%d, actual=%d",
+ expectedState, state));
+ return false;
+ }
+ log(String.format("waitForWifiAPState interim: expected=%d, actual=%d",
+ expectedState, state));
+ SystemClock.sleep(SHORT_TIMEOUT);
}
}
@@ -450,44 +226,49 @@
* @return SUCCESS if tethering result is successful
* FAILURE if tethering result returns error.
*/
- public int waitForTetherStateChange(long timeout) {
- long startTime = System.currentTimeMillis();
+ protected boolean waitForTetherStateChange(long timeout) {
+ long startTime = SystemClock.uptimeMillis();
+ String[] wifiRegexes = mCm.getTetherableWifiRegexs();
while (true) {
- if ((System.currentTimeMillis() - startTime) > timeout) {
- return mWifiTetherResult;
+ if ((SystemClock.uptimeMillis() - startTime) > timeout) {
+ return false;
}
- log("Wait for wifi tethering result.");
- synchronized (this) {
- try {
- this.wait(SHORT_TIMEOUT);
- } catch (InterruptedException e) {
- e.printStackTrace();
- }
- if (mWifiTetherResult == INIT ) {
- continue;
- } else {
- return mWifiTetherResult;
+ String[] active = mCm.getTetheredIfaces();
+ String[] error = mCm.getTetheringErroredIfaces();
+ for (String iface: active) {
+ for (String regex: wifiRegexes) {
+ if (iface.matches(regex)) {
+ return true;
+ }
}
}
+ for (String iface: error) {
+ for (String regex: wifiRegexes) {
+ if (iface.matches(regex)) {
+ return false;
+ }
+ }
+ }
+ SystemClock.sleep(SHORT_TIMEOUT);
}
}
// Return true if device is currently connected to mobile network
- public boolean isConnectedToMobile() {
- return (mNetworkInfo.getType() == ConnectivityManager.TYPE_MOBILE);
+ protected boolean isConnectedToMobile() {
+ return (mCm.getActiveNetworkInfo().getType() == ConnectivityManager.TYPE_MOBILE);
}
// Return true if device is currently connected to Wifi
- public boolean isConnectedToWifi() {
- return (mNetworkInfo.getType() == ConnectivityManager.TYPE_WIFI);
+ protected boolean isConnectedToWifi() {
+ return (mCm.getActiveNetworkInfo().getType() == ConnectivityManager.TYPE_WIFI);
}
- public boolean enableWifi() {
+ protected boolean enableWifi() {
return mWifiManager.setWifiEnabled(true);
}
// Turn screen off
- public void turnScreenOff() {
+ protected void turnScreenOff() {
log("Turn screen off");
PowerManager pm =
(PowerManager) mContext.getSystemService(Context.POWER_SERVICE);
@@ -495,7 +276,7 @@
}
// Turn screen on
- public void turnScreenOn() {
+ protected void turnScreenOn() {
log("Turn screen on");
PowerManager pm =
(PowerManager) mContext.getSystemService(Context.POWER_SERVICE);
@@ -511,7 +292,7 @@
* @param pingServerList a list of servers that can be used for ping test, can be null
* @return true if the ping test is successful, false otherwise.
*/
- public boolean pingTest(String[] pingServerList) {
+ protected boolean pingTest(String[] pingServerList) {
String[] hostList = {"www.google.com", "www.yahoo.com",
"www.bing.com", "www.facebook.com", "www.ask.com"};
if (pingServerList != null) {
@@ -549,7 +330,7 @@
* If the device is already associated with a WiFi, disconnect and forget it,
* We don't verify whether the connection is successful or not, leave this to the test
*/
- public boolean connectToWifi(String knownSSID) {
+ protected boolean connectToWifi(String knownSSID) {
WifiConfiguration config = new WifiConfiguration();
config.SSID = knownSSID;
config.allowedKeyManagement.set(KeyMgmt.NONE);
@@ -562,7 +343,7 @@
* @param config
* @return
*/
- public boolean connectToWifiWithConfiguration(WifiConfiguration config) {
+ protected boolean connectToWifiWithConfiguration(WifiConfiguration config) {
String ssid = config.SSID;
config.SSID = convertToQuotedString(ssid);
@@ -571,7 +352,7 @@
log("Wifi is not enabled, enable it");
mWifiManager.setWifiEnabled(true);
// wait for the wifi state change before start scanning.
- if (!waitForWifiState(WifiManager.WIFI_STATE_ENABLED, 2*SHORT_TIMEOUT)) {
+ if (!waitForWifiState(WifiManager.WIFI_STATE_ENABLED, LONG_TIMEOUT)) {
log("wait for WIFI_STATE_ENABLED failed");
return false;
}
@@ -592,12 +373,12 @@
/*
* Disconnect from the current AP and remove configured networks.
*/
- public boolean disconnectAP() {
+ protected boolean disconnectAP() {
// remove saved networks
if (!mWifiManager.isWifiEnabled()) {
log("Enabled wifi before remove configured networks");
mWifiManager.setWifiEnabled(true);
- sleep(SHORT_TIMEOUT);
+ SystemClock.sleep(SHORT_TIMEOUT);
}
List<WifiConfiguration> wifiConfigList = mWifiManager.getConfiguredNetworks();
@@ -623,37 +404,81 @@
* Disable Wifi
* @return true if Wifi is disabled successfully
*/
- public boolean disableWifi() {
+ protected boolean disableWifi() {
return mWifiManager.setWifiEnabled(false);
}
/**
* Remove configured networks and disable wifi
*/
- public boolean removeConfiguredNetworksAndDisableWifi() {
+ protected boolean removeConfiguredNetworksAndDisableWifi() {
if (!disconnectAP()) {
return false;
}
- sleep(SHORT_TIMEOUT);
+ SystemClock.sleep(SHORT_TIMEOUT);
if (!mWifiManager.setWifiEnabled(false)) {
return false;
}
- sleep(SHORT_TIMEOUT);
+ SystemClock.sleep(SHORT_TIMEOUT);
return true;
}
- private void sleep(long sleeptime) {
- try {
- Thread.sleep(sleeptime);
- } catch (InterruptedException e) {}
- }
-
protected static String convertToQuotedString(String string) {
return "\"" + string + "\"";
}
+ protected boolean waitForActiveNetworkConnection(long timeout) {
+ long startTime = SystemClock.uptimeMillis();
+ while (true) {
+ NetworkInfo ni = mCm.getActiveNetworkInfo();
+ String niString = ni == null ? "null" : ni.toString();
+ if (ni != null && ni.isConnected()) {
+ return true;
+ }
+ if ((SystemClock.uptimeMillis() - startTime) > timeout) {
+ log("waitForActiveNetworkConnection timeout: " + niString);
+ return false;
+ }
+ log("waitForActiveNetworkConnection interim: " + niString);
+ SystemClock.sleep(SHORT_TIMEOUT);
+ }
+ }
+
+ protected boolean waitUntilNoActiveNetworkConnection(long timeout) {
+ long startTime = SystemClock.uptimeMillis();
+ while (true) {
+ NetworkInfo ni = mCm.getActiveNetworkInfo();
+ if (ni == null) {
+ return true;
+ }
+ String niString = ni.toString();
+ if ((SystemClock.uptimeMillis() - startTime) > timeout) {
+ log("waitForActiveNetworkConnection timeout: " + niString);
+ return false;
+ }
+ log("waitForActiveNetworkConnection interim: " + niString);
+ SystemClock.sleep(SHORT_TIMEOUT);
+ }
+ }
+
+ // use ping request against Google public DNS to verify connectivity
+ protected boolean checkNetworkConnectivity() {
+ assertTrue("no active network connection", waitForActiveNetworkConnection(LONG_TIMEOUT));
+ try {
+ Process proc = Runtime.getRuntime().exec(new String[]{
+ "/system/bin/ping", "-W", "30", "-c", "1", PING_IP_ADDR});
+ int exitCode = proc.waitFor();
+ return exitCode == 0;
+ } catch (InterruptedException ie) {
+ Log.e(LOG_TAG, "InterruptedException while waiting for ping");
+ } catch (IOException ioe) {
+ Log.e(LOG_TAG, "IOException during ping", ioe);
+ }
+ return false;
+ }
+
@Override
- public void tearDown() throws Exception{
+ protected void tearDown() throws Exception{
//Unregister receiver
if (mConnectivityReceiver != null) {
mContext.unregisterReceiver(mConnectivityReceiver);
@@ -667,4 +492,36 @@
private void log(String message) {
Log.v(LOG_TAG, message);
}
+
+ /**
+ * Connect to the provided Wi-Fi network
+ * @param config is the network configuration
+ * @throws AssertionError if fails to associate and connect to wifi ap
+ */
+ protected void connectToWifi(WifiConfiguration config) {
+ // step 1: connect to the test access point
+ assertTrue("failed to associate with " + config.SSID,
+ connectToWifiWithConfiguration(config));
+
+ // step 2: verify Wifi state and network state;
+ assertTrue("wifi state not connected with " + config.SSID,
+ waitForNetworkState(ConnectivityManager.TYPE_WIFI,
+ State.CONNECTED, LONG_TIMEOUT));
+
+ // step 3: verify the current connected network is the given SSID
+ assertNotNull("no active wifi info", mWifiManager.getConnectionInfo());
+ assertEquals("SSID mismatch", config.SSID, mWifiManager.getConnectionInfo().getSSID());
+ }
+
+ /**
+ * checks if the input is a hexadecimal string of given length
+ *
+ * @param input string to be checked
+ * @param length required length of the string
+ * @return
+ */
+ protected boolean isHex(String input, int length) {
+ Pattern p = Pattern.compile(String.format("[0-9A-Fa-f]{%d}", length));
+ return p.matcher(input).matches();
+ }
}
diff --git a/core/tests/ConnectivityManagerTest/src/com/android/connectivitymanagertest/NetworkState.java b/core/tests/ConnectivityManagerTest/src/com/android/connectivitymanagertest/NetworkState.java
deleted file mode 100644
index 9d97ac5..0000000
--- a/core/tests/ConnectivityManagerTest/src/com/android/connectivitymanagertest/NetworkState.java
+++ /dev/null
@@ -1,203 +0,0 @@
-/*
- * Copyright (C) 2010 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 com.android.connectivitymanagertest;
-
-import android.net.NetworkInfo.State;
-import android.util.Log;
-
-import java.util.List;
-import java.util.ArrayList;
-
-public class NetworkState {
- public static final int TO_DISCONNECTION = 0; // transition to disconnection
- public static final int TO_CONNECTION = 1; // transition to connection
- public static final int DO_NOTHING = -1; // no state change
- private final String LOG_TAG = "NetworkState";
- private List<State> mStateDepository;
- private State mTransitionTarget;
- private int mTransitionDirection;
- private String mReason = null; // record mReason of state transition failure
-
- public NetworkState() {
- mStateDepository = new ArrayList<State>();
- mTransitionDirection = DO_NOTHING;
- mTransitionTarget = State.UNKNOWN;
- }
-
- public NetworkState(State currentState) {
- mStateDepository = new ArrayList<State>();
- mStateDepository.add(currentState);
- mTransitionDirection = DO_NOTHING;
- mTransitionTarget = State.UNKNOWN;
- }
-
- // Reinitialize the network state
- public void resetNetworkState() {
- mStateDepository.clear();
- mTransitionDirection = DO_NOTHING;
- mTransitionTarget = State.UNKNOWN;
- }
-
- // set the transition criteria, transitionDir could be:
- // DO_NOTHING, TO_CONNECTION, TO_DISCONNECTION
- public void setStateTransitionCriteria(State initState, int transitionDir, State targetState) {
- if (!mStateDepository.isEmpty()) {
- mStateDepository.clear();
- }
- mStateDepository.add(initState);
- mTransitionDirection = transitionDir;
- mTransitionTarget = targetState;
- Log.v(LOG_TAG, "setStateTransitionCriteria: " + printStates());
- }
-
- public void recordState(State currentState) {
- mStateDepository.add(currentState);
- }
-
- // Verify state transition
- public boolean validateStateTransition() {
- Log.v(LOG_TAG, "print state depository: " + printStates());
- if (mTransitionDirection == DO_NOTHING) {
- if (mStateDepository.isEmpty()) {
- Log.v(LOG_TAG, "no state is recorded");
- mReason = "no state is recorded.";
- return false;
- } else if (mStateDepository.size() > 1) {
- for (int i = 0; i < mStateDepository.size(); i++) {
- if (mStateDepository.get(i) != mTransitionTarget) {
- Log.v(LOG_TAG, "state changed.");
- mReason = "Unexpected state change";
- return false;
- }
- }
- } else if (mStateDepository.get(0) != mTransitionTarget) {
- Log.v(LOG_TAG, mTransitionTarget + " is expected, but it is " +
- mStateDepository.get(0));
- mReason = mTransitionTarget + " is expected, but it is " + mStateDepository.get(0);
- return false;
- }
- return true;
- } else if (mTransitionDirection == TO_CONNECTION) {
- Log.v(LOG_TAG, "transition to CONNECTED");
- return transitToConnection();
- } else {
- Log.v(LOG_TAG, "transition to DISCONNECTED");
- return transitToDisconnection();
- }
- }
-
- /*
- * Verifies state transition from CONNECTED->...-> DISCONNECTED.
- *
- * returns false if initial state or target state is not correct, or if there is
- * any transition from DISCONNECTING/DISCONNECTED -> CONNECTED.
- */
- public boolean transitToDisconnection () {
- mReason = "states: " + printStates();
- if (mStateDepository.get(0) != State.CONNECTED) {
- mReason += " initial state should be CONNECTED, but it is " +
- mStateDepository.get(0) + ".";
- return false;
- }
- State lastState = mStateDepository.get(mStateDepository.size() - 1);
- if ( lastState != mTransitionTarget) {
- mReason += " the last state should be DISCONNECTED, but it is " + lastState;
- return false;
- }
- for (int i = 1; i < mStateDepository.size() - 1; i++) {
- State preState = mStateDepository.get(i-1);
- State curState = mStateDepository.get(i);
- if (preState == curState) {
- continue;
- } else if ((preState == State.CONNECTED) && ((curState == State.DISCONNECTING) ||
- (curState == State.DISCONNECTED))) {
- continue;
- } else if ((preState == State.DISCONNECTING) && (curState == State.DISCONNECTED)) {
- continue;
- } else {
- mReason += " Transition state from " + preState.toString() + " to " +
- curState.toString() + " is not valid.";
- return false;
- }
- }
- return true;
- }
-
- /*
- * Verifies state transition from DISCONNECTED->...-> CONNECTED.
- *
- * returns false if initial state or target state is not correct, or if there is
- * any transition from CONNECED -> DISCONNECTED.
- */
- public boolean transitToConnection() {
- mReason = "states: " + printStates();
- if (mStateDepository.get(0) != State.DISCONNECTED) {
- mReason += " initial state should be DISCONNECTED, but it is " +
- mStateDepository.get(0) + ".";
- return false;
- }
- State lastState = mStateDepository.get(mStateDepository.size() - 1);
- if ( lastState != mTransitionTarget) {
- mReason += "The last state should be " + mTransitionTarget + ", but it is " + lastState;
- return false;
- }
- for (int i = 1; i < mStateDepository.size(); i++) {
- State preState = mStateDepository.get(i-1);
- State curState = mStateDepository.get(i);
- if (preState == curState) {
- continue;
- }
- if ((preState == State.DISCONNECTED) && ((curState == State.CONNECTING) ||
- (curState == State.CONNECTED))) {
- continue;
- } else if ((preState == State.CONNECTING) && (curState == State.CONNECTED)) {
- continue;
- } else {
- mReason += " Transition state from " + preState.toString() + " to " +
- curState.toString() + " is not valid.";
- return false;
- }
- }
- return true;
- }
-
- public List<State> getTransitionStates() {
- return mStateDepository;
- }
-
- // return state failure mReason
- public String getReason() {
- return mReason;
- }
-
- public String printStates() {
- StringBuilder stateBuilder = new StringBuilder("");
- for (int i = 0; i < mStateDepository.size(); i++) {
- stateBuilder.append(" ").append(mStateDepository.get(i).toString()).append("->");
- }
- return stateBuilder.toString();
- }
-
- @Override
- public String toString() {
- StringBuilder builder = new StringBuilder(" ");
- builder.append("mTransitionDirection: ").append(Integer.toString(mTransitionDirection)).
- append("; ").append("states:").
- append(printStates()).append("; ");
- return builder.toString();
- }
-}
diff --git a/core/tests/ConnectivityManagerTest/src/com/android/connectivitymanagertest/WifiAssociationTestRunner.java b/core/tests/ConnectivityManagerTest/src/com/android/connectivitymanagertest/WifiAssociationTestRunner.java
index 722df2e..2354484 100644
--- a/core/tests/ConnectivityManagerTest/src/com/android/connectivitymanagertest/WifiAssociationTestRunner.java
+++ b/core/tests/ConnectivityManagerTest/src/com/android/connectivitymanagertest/WifiAssociationTestRunner.java
@@ -57,8 +57,7 @@
@Override
public void onCreate(Bundle icicle) {
super.onCreate(icicle);
- Bundle arguments = icicle;
- String mFrequencyBand = arguments.getString("frequency-band");
+ String mFrequencyBand = icicle.getString("frequency-band");
if (mFrequencyBand != null) {
setFrequencyBand(mFrequencyBand);
}
diff --git a/core/tests/ConnectivityManagerTest/src/com/android/connectivitymanagertest/functional/ConnectivityManagerMobileTest.java b/core/tests/ConnectivityManagerTest/src/com/android/connectivitymanagertest/functional/ConnectivityManagerMobileTest.java
index 05462b4..b4b0e53 100644
--- a/core/tests/ConnectivityManagerTest/src/com/android/connectivitymanagertest/functional/ConnectivityManagerMobileTest.java
+++ b/core/tests/ConnectivityManagerTest/src/com/android/connectivitymanagertest/functional/ConnectivityManagerMobileTest.java
@@ -16,27 +16,23 @@
package com.android.connectivitymanagertest.functional;
-import android.content.Context;
import android.net.ConnectivityManager;
import android.net.NetworkInfo;
import android.net.NetworkInfo.State;
import android.net.wifi.WifiManager;
-import android.os.PowerManager;
-import android.os.PowerManager.WakeLock;
+import android.os.SystemClock;
import android.provider.Settings;
import android.test.suitebuilder.annotation.LargeTest;
import android.util.Log;
import com.android.connectivitymanagertest.ConnectivityManagerTestBase;
import com.android.connectivitymanagertest.ConnectivityManagerTestRunner;
-import com.android.connectivitymanagertest.NetworkState;
public class ConnectivityManagerMobileTest extends
ConnectivityManagerTestBase {
private static final String TAG = "ConnectivityManagerMobileTest";
private String mTestAccessPoint;
- private WakeLock wl;
private boolean mWifiOnlyFlag;
@Override
@@ -47,15 +43,11 @@
mTestAccessPoint = mRunner.mTestSsid;
mWifiOnlyFlag = mRunner.mWifiOnlyFlag;
- PowerManager pm = (PowerManager)getInstrumentation().
- getContext().getSystemService(Context.POWER_SERVICE);
- wl = pm.newWakeLock(PowerManager.SCREEN_BRIGHT_WAKE_LOCK, "CMWakeLock");
- wl.acquire();
// Each test case will start with cellular connection
if (Settings.Global.getInt(getInstrumentation().getContext().getContentResolver(),
Settings.Global.AIRPLANE_MODE_ON) == 1) {
log("airplane is not disabled, disable it.");
- mCM.setAirplaneMode(false);
+ mCm.setAirplaneMode(false);
}
if (!mWifiOnlyFlag) {
@@ -72,15 +64,14 @@
@Override
public void tearDown() throws Exception {
- wl.release();
removeConfiguredNetworksAndDisableWifi();
- mCM.setAirplaneMode(false);
+ mCm.setAirplaneMode(false);
super.tearDown();
}
// help function to verify 3G connection
public void verifyCellularConnection() {
- NetworkInfo extraNetInfo = mCM.getActiveNetworkInfo();
+ NetworkInfo extraNetInfo = mCm.getActiveNetworkInfo();
assertEquals("network type is not MOBILE", ConnectivityManager.TYPE_MOBILE,
extraNetInfo.getType());
assertTrue("not connected to cellular network", extraNetInfo.isConnected());
@@ -90,431 +81,262 @@
Log.v(TAG, message);
}
- private void sleep(long sleeptime) {
- try {
- Thread.sleep(sleeptime);
- } catch (InterruptedException e) {}
- }
-
// Test case 1: Test enabling Wifi without associating with any AP, no broadcast on network
// event should be expected.
@LargeTest
public void test3GToWifiNotification() {
if (mWifiOnlyFlag) {
- Log.v(TAG, this.getName() + " is excluded for wifi-only test");
+ Log.v(TAG, getName() + " is excluded for wifi-only test");
return;
}
- // Enable Wi-Fi to avoid initial UNKNOWN state
- enableWifi();
- sleep(2 * SHORT_TIMEOUT);
- // Wi-Fi is disabled
- disableWifi();
+ // disable WiFi
+ assertTrue("failed to disable WiFi", disableWifi());
- assertTrue(waitForNetworkState(ConnectivityManager.TYPE_WIFI,
- State.DISCONNECTED, LONG_TIMEOUT));
- assertTrue(waitForNetworkState(ConnectivityManager.TYPE_MOBILE,
- State.CONNECTED, LONG_TIMEOUT));
- // Wait for 10 seconds for broadcasts to be sent out
- sleep(10 * 1000);
+ // wait for mobile
+ assertTrue("failed to wait for mobile connection", waitForNetworkState(
+ ConnectivityManager.TYPE_MOBILE, State.CONNECTED, LONG_TIMEOUT));
- // As Wifi stays in DISCONNETED, Mobile statys in CONNECTED,
- // the connectivity manager will not broadcast any network connectivity event for Wifi
- NetworkInfo networkInfo = mCM.getNetworkInfo(ConnectivityManager.TYPE_MOBILE);
- setStateTransitionCriteria(ConnectivityManager.TYPE_MOBILE,
- networkInfo.getState(), NetworkState.DO_NOTHING, State.CONNECTED);
- networkInfo = mCM.getNetworkInfo(ConnectivityManager.TYPE_WIFI);
- setStateTransitionCriteria(ConnectivityManager.TYPE_WIFI, networkInfo.getState(),
- NetworkState.DO_NOTHING, State.DISCONNECTED);
- // Eanble Wifi without associating with any AP
- enableWifi();
- sleep(2 * SHORT_TIMEOUT);
+ // assert that we are indeed using mobile
+ NetworkInfo ni = mCm.getActiveNetworkInfo();
+ assertEquals("active network is not mobile", ConnectivityManager.TYPE_MOBILE, ni.getType());
- // validate state and broadcast
- if (!validateNetworkStates(ConnectivityManager.TYPE_WIFI)) {
- log("the state for WIFI is changed");
- log("reason: " +
- getTransitionFailureReason(ConnectivityManager.TYPE_WIFI));
- assertTrue("state validation fail", false);
- }
- if (!validateNetworkStates(ConnectivityManager.TYPE_MOBILE)) {
- log("the state for MOBILE is changed");
- log("reason: " +
- getTransitionFailureReason(ConnectivityManager.TYPE_MOBILE));
- assertTrue("state validation fail", false);
- }
- // Verify that the device is still connected to MOBILE
+ long timestamp = SystemClock.uptimeMillis();
+ // now enable WiFi
+ assertTrue("failed to enable WiFi", enableWifi());
+ // assert that WiFi state settles at disconnected since no AP should be configured
+ assertTrue("WiFi state is not DISCONNECTED after enabling", waitForWifiState(
+ WifiManager.WIFI_STATE_DISABLED, LONG_TIMEOUT));
+
+ // assert that no connectivity change broadcast was sent since we enable wifi
+ assertTrue("connectivity has changed since wifi enable",
+ timestamp > getLastConnectivityChangeTime());
+
+ // verify that the device is still connected to MOBILE
verifyCellularConnection();
+ // verify that connection actually works
+ assertTrue("no network connectivity at end of test", checkNetworkConnectivity());
}
// Test case 2: test connection to a given AP
@LargeTest
public void testConnectToWifi() {
assertNotNull("SSID is null", mTestAccessPoint);
- NetworkInfo networkInfo;
- if (!mWifiOnlyFlag) {
- //Prepare for connectivity verification
- networkInfo = mCM.getNetworkInfo(ConnectivityManager.TYPE_MOBILE);
- setStateTransitionCriteria(ConnectivityManager.TYPE_MOBILE,
- networkInfo.getState(), NetworkState.TO_DISCONNECTION, State.DISCONNECTED);
- }
- networkInfo = mCM.getNetworkInfo(ConnectivityManager.TYPE_WIFI);
- setStateTransitionCriteria(ConnectivityManager.TYPE_WIFI, networkInfo.getState(),
- NetworkState.TO_CONNECTION, State.CONNECTED);
- // Enable Wifi and connect to a test access point
+ // assert that we are able to connect to the ap
assertTrue("failed to connect to " + mTestAccessPoint,
connectToWifi(mTestAccessPoint));
-
+ // assert that WifiManager reports correct state
assertTrue(waitForWifiState(WifiManager.WIFI_STATE_ENABLED, LONG_TIMEOUT));
- log("wifi state is enabled");
+ // assert that ConnectivityManager reports correct state for Wifi
assertTrue(waitForNetworkState(ConnectivityManager.TYPE_WIFI, State.CONNECTED,
WIFI_CONNECTION_TIMEOUT));
- if (!mWifiOnlyFlag) {
- assertTrue(waitForNetworkState(ConnectivityManager.TYPE_MOBILE,
- State.DISCONNECTED, LONG_TIMEOUT));
- }
-
- // validate states
- if (!validateNetworkStates(ConnectivityManager.TYPE_WIFI)) {
- log("Wifi state transition validation failed.");
- log("reason: " +
- getTransitionFailureReason(ConnectivityManager.TYPE_WIFI));
- assertTrue(false);
- }
- if (!mWifiOnlyFlag) {
- if (!validateNetworkStates(ConnectivityManager.TYPE_MOBILE)) {
- log("Mobile state transition validation failed.");
- log("reason: " +
- getTransitionFailureReason(ConnectivityManager.TYPE_MOBILE));
- assertTrue(false);
- }
- }
+ // below check disbabled since we have bug in what ConnectivityManager returns
+// if (!mWifiOnlyFlag) {
+// // assert that ConnectivityManager reports correct state for mobile
+// assertTrue("mobile not disconnected", waitForNetworkState(
+// ConnectivityManager.TYPE_MOBILE, State.DISCONNECTED, LONG_TIMEOUT));
+// }
+ // verify that connection actually works
+ assertTrue("no network connectivity at end of test", checkNetworkConnectivity());
}
- // Test case 3: connect to Wifi with known AP
+ // Test case 3: connect & reconnect to Wifi with known AP
@LargeTest
public void testConnectToWifWithKnownAP() {
assertNotNull("SSID is null", mTestAccessPoint);
- // Connect to mTestAccessPoint
- assertTrue("failed to connect to " + mTestAccessPoint,
- connectToWifi(mTestAccessPoint));
- assertTrue(waitForWifiState(WifiManager.WIFI_STATE_ENABLED, LONG_TIMEOUT));
- assertTrue(waitForNetworkState(ConnectivityManager.TYPE_WIFI, State.CONNECTED,
- WIFI_CONNECTION_TIMEOUT));
+ // enable WiFi
+ assertTrue("failed to enable wifi", enableWifi());
+ // wait for wifi enable
+ assertTrue("wifi not enabled", waitForWifiState(
+ WifiManager.WIFI_STATE_ENABLED, LONG_TIMEOUT));
+ // Connect to AP
+ assertTrue("failed to connect to " + mTestAccessPoint, connectToWifi(mTestAccessPoint));
+ // verify wifi connected as reported by ConnectivityManager
+ assertTrue("wifi not connected", waitForNetworkState(
+ ConnectivityManager.TYPE_WIFI, State.CONNECTED, WIFI_CONNECTION_TIMEOUT));
- sleep(SHORT_TIMEOUT);
- // Disable Wifi
- log("Disable Wifi");
- if (!disableWifi()) {
- log("disable Wifi failed");
- return;
- }
+ assertTrue("failed to disable wifi", disableWifi());
// Wait for the Wifi state to be DISABLED
- assertTrue(waitForWifiState(WifiManager.WIFI_STATE_DISABLED, LONG_TIMEOUT));
- assertTrue(waitForNetworkState(ConnectivityManager.TYPE_WIFI,
- State.DISCONNECTED, LONG_TIMEOUT));
+ assertTrue("wifi state not disabled", waitForWifiState(
+ WifiManager.WIFI_STATE_DISABLED, LONG_TIMEOUT));
+ // below check disbabled since we have bug in what ConnectivityManager returns
+// assertTrue("wifi not disconnected", waitForNetworkState(ConnectivityManager.TYPE_WIFI,
+// State.DISCONNECTED, LONG_TIMEOUT));
if (!mWifiOnlyFlag) {
- assertTrue(waitForNetworkState(ConnectivityManager.TYPE_MOBILE,
- State.CONNECTED, LONG_TIMEOUT));
+ assertTrue("mobile not connected after wifi disable", waitForNetworkState(
+ ConnectivityManager.TYPE_MOBILE, State.CONNECTED, LONG_TIMEOUT));
}
- NetworkInfo networkInfo;
- if (!mWifiOnlyFlag) {
- //Prepare for connectivity state verification
- networkInfo = mCM.getNetworkInfo(ConnectivityManager.TYPE_MOBILE);
- setStateTransitionCriteria(ConnectivityManager.TYPE_MOBILE,
- networkInfo.getState(), NetworkState.DO_NOTHING,
- State.DISCONNECTED);
- }
- networkInfo = mCM.getNetworkInfo(ConnectivityManager.TYPE_WIFI);
- setStateTransitionCriteria(ConnectivityManager.TYPE_WIFI, networkInfo.getState(),
- NetworkState.TO_CONNECTION, State.CONNECTED);
+ // wait for 30s before restart wifi
+ SystemClock.sleep(LONG_TIMEOUT);
+ assertTrue("failed to enable wifi after disable", enableWifi());
- // wait for 2 minutes before restart wifi
- sleep(WIFI_STOP_START_INTERVAL);
- // Enable Wifi again
- log("Enable Wifi again");
- enableWifi();
-
+ // wait for wifi enable
+ assertTrue("wifi not enabled after toggle", waitForWifiState(
+ WifiManager.WIFI_STATE_ENABLED, LONG_TIMEOUT));
// Wait for Wifi to be connected and mobile to be disconnected
- assertTrue(waitForNetworkState(ConnectivityManager.TYPE_WIFI, State.CONNECTED,
- WIFI_CONNECTION_TIMEOUT));
- if (!mWifiOnlyFlag) {
- assertTrue(waitForNetworkState(ConnectivityManager.TYPE_MOBILE,
- State.DISCONNECTED, LONG_TIMEOUT));
- }
-
- // validate wifi states
- if (!validateNetworkStates(ConnectivityManager.TYPE_WIFI)) {
- log("Wifi state transition validation failed.");
- log("reason: " +
- getTransitionFailureReason(ConnectivityManager.TYPE_WIFI));
- assertTrue(false);
- }
+ assertTrue("wifi not connected after toggle", waitForNetworkState(
+ ConnectivityManager.TYPE_WIFI, State.CONNECTED, WIFI_CONNECTION_TIMEOUT));
+ // below check disbabled since we have bug in what ConnectivityManager returns
+// if (!mWifiOnlyFlag) {
+// assertTrue("mobile not disconnected after wifi toggle", waitForNetworkState(
+// ConnectivityManager.TYPE_MOBILE, State.DISCONNECTED, LONG_TIMEOUT));
+// }
+ // verify that connection actually works
+ assertTrue("no network connectivity at end of test", checkNetworkConnectivity());
}
- // Test case 4: test disconnect Wifi
+ // Test case 4: test disconnect and clear wifi settings
@LargeTest
public void testDisconnectWifi() {
assertNotNull("SSID is null", mTestAccessPoint);
+ // enable WiFi
+ assertTrue("failed to enable wifi", enableWifi());
+ // wait for wifi enable
+ assertTrue("wifi not enabled", waitForWifiState(
+ WifiManager.WIFI_STATE_ENABLED, LONG_TIMEOUT));
// connect to Wifi
assertTrue("failed to connect to " + mTestAccessPoint,
connectToWifi(mTestAccessPoint));
- assertTrue(waitForNetworkState(ConnectivityManager.TYPE_WIFI, State.CONNECTED,
- WIFI_CONNECTION_TIMEOUT));
-
- // Wait for a few seconds to avoid the state that both Mobile and Wifi is connected
- sleep(SHORT_TIMEOUT);
-
- NetworkInfo networkInfo;
- if (!mWifiOnlyFlag) {
- networkInfo = mCM.getNetworkInfo(ConnectivityManager.TYPE_MOBILE);
- setStateTransitionCriteria(ConnectivityManager.TYPE_MOBILE,
- networkInfo.getState(),
- NetworkState.TO_CONNECTION,
- State.CONNECTED);
- }
- networkInfo = mCM.getNetworkInfo(ConnectivityManager.TYPE_WIFI);
- setStateTransitionCriteria(ConnectivityManager.TYPE_WIFI, networkInfo.getState(),
- NetworkState.TO_DISCONNECTION, State.DISCONNECTED);
+ assertTrue("wifi not connected", waitForNetworkState(
+ ConnectivityManager.TYPE_WIFI, State.CONNECTED, WIFI_CONNECTION_TIMEOUT));
// clear Wifi
removeConfiguredNetworksAndDisableWifi();
- assertTrue(waitForNetworkState(ConnectivityManager.TYPE_WIFI,
- State.DISCONNECTED, LONG_TIMEOUT));
+ // assert that wifi has been disabled
+ assertTrue("wifi state not disabled", waitForWifiState(
+ WifiManager.WIFI_STATE_DISABLED, LONG_TIMEOUT));
if (!mWifiOnlyFlag) {
- assertTrue(waitForNetworkState(ConnectivityManager.TYPE_MOBILE,
+ // assert that mobile is now connected
+ assertTrue("mobile not enabled", waitForNetworkState(ConnectivityManager.TYPE_MOBILE,
State.CONNECTED, LONG_TIMEOUT));
- }
-
- // validate states
- if (!validateNetworkStates(ConnectivityManager.TYPE_WIFI)) {
- log("Wifi state transition validation failed.");
- log("reason: " +
- getTransitionFailureReason(ConnectivityManager.TYPE_WIFI));
- assertTrue(false);
- }
- if (!mWifiOnlyFlag) {
- if (!validateNetworkStates(ConnectivityManager.TYPE_MOBILE)) {
- log("Mobile state transition validation failed.");
- log("reason: " +
- getTransitionFailureReason(ConnectivityManager.TYPE_MOBILE));
- assertTrue(false);
- }
+ // verify that connection actually works
+ assertTrue("no network connectivity at end of test", checkNetworkConnectivity());
}
}
- // Test case 5: test connectivity from 3G to airplane mode, then to 3G again
+ // Test case 5: test connectivity with mobile->airplane mode->mobile
@LargeTest
public void testDataConnectionWith3GToAmTo3G() {
if (mWifiOnlyFlag) {
- Log.v(TAG, this.getName() + " is excluded for wifi-only test");
+ Log.v(TAG, getName() + " is excluded for wifi-only test");
return;
}
- //Prepare for state verification
- NetworkInfo networkInfo = mCM.getNetworkInfo(ConnectivityManager.TYPE_MOBILE);
- setStateTransitionCriteria(ConnectivityManager.TYPE_MOBILE,
- networkInfo.getState(),
- NetworkState.TO_DISCONNECTION,
- State.DISCONNECTED);
- networkInfo = mCM.getNetworkInfo(ConnectivityManager.TYPE_WIFI);
- assertEquals(State.DISCONNECTED, networkInfo.getState());
+ // disable wifi
+ assertTrue("failed to disable wifi", disableWifi());
+ assertTrue("wifi state not disabled", waitForWifiState(
+ WifiManager.WIFI_STATE_DISABLED, LONG_TIMEOUT));
+ // assert that we have mobile connection
+ assertTrue("no mobile connection", waitForNetworkState(
+ ConnectivityManager.TYPE_MOBILE, State.CONNECTED, LONG_TIMEOUT));
- // Enable airplane mode
- log("Enable airplane mode");
- mCM.setAirplaneMode(true);
- sleep(SHORT_TIMEOUT);
-
- networkInfo = mCM.getNetworkInfo(ConnectivityManager.TYPE_WIFI);
- assertEquals(State.DISCONNECTED, networkInfo.getState());
- // wait until mobile is turn off
- assertTrue(waitForNetworkState(ConnectivityManager.TYPE_MOBILE,
- State.DISCONNECTED, LONG_TIMEOUT));
- if (!validateNetworkStates(ConnectivityManager.TYPE_MOBILE)) {
- log("Mobile state transition validation failed.");
- log("reason: " +
- getTransitionFailureReason(ConnectivityManager.TYPE_MOBILE));
- assertTrue(false);
- }
-
- // reset state recorder
- networkInfo = mCM.getNetworkInfo(ConnectivityManager.TYPE_MOBILE);
- setStateTransitionCriteria(ConnectivityManager.TYPE_MOBILE,
- networkInfo.getState(),
- NetworkState.TO_CONNECTION,
- State.CONNECTED);
- networkInfo = mCM.getNetworkInfo(ConnectivityManager.TYPE_WIFI);
- setStateTransitionCriteria(ConnectivityManager.TYPE_WIFI, networkInfo.getState(),
- NetworkState.DO_NOTHING, State.DISCONNECTED);
+ // enable airplane mode
+ mCm.setAirplaneMode(true);
+ // assert no active network connection after airplane mode enabled
+ assertTrue("still has active network connection",
+ waitUntilNoActiveNetworkConnection(LONG_TIMEOUT));
// disable airplane mode
- mCM.setAirplaneMode(false);
+ mCm.setAirplaneMode(false);
+ // assert there is active network connection after airplane mode disabled
+ assertTrue("no active network connection after airplane mode disable",
+ waitForActiveNetworkConnection(LONG_TIMEOUT));
- assertTrue(waitForNetworkState(ConnectivityManager.TYPE_MOBILE,
- State.CONNECTED, LONG_TIMEOUT));
-
- // Validate the state transition
- if (!validateNetworkStates(ConnectivityManager.TYPE_MOBILE)) {
- log("Mobile state transition validation failed.");
- log("reason: " +
- getTransitionFailureReason(ConnectivityManager.TYPE_MOBILE));
- assertTrue(false);
- }
- if (!validateNetworkStates(ConnectivityManager.TYPE_WIFI)) {
- log("Wifi state transition validation failed.");
- log("reason: " +
- getTransitionFailureReason(ConnectivityManager.TYPE_WIFI));
- assertTrue(false);
- }
+ // assert that we have mobile connection
+ assertTrue("no mobile connection", waitForNetworkState(
+ ConnectivityManager.TYPE_MOBILE, State.CONNECTED, LONG_TIMEOUT));
+ // verify that connection actually works
+ assertTrue("no network connectivity at end of test", checkNetworkConnectivity());
}
- // Test case 6: test connectivity with airplane mode Wifi connected
+ // Test case 6: test connectivity with airplane mode on but wifi enabled
@LargeTest
public void testDataConnectionOverAMWithWifi() {
- if (mWifiOnlyFlag) {
- Log.v(TAG, this.getName() + " is excluded for wifi-only test");
- return;
- }
assertNotNull("SSID is null", mTestAccessPoint);
- // Eanble airplane mode
- log("Enable airplane mode");
- mCM.setAirplaneMode(true);
+ // enable airplane mode
+ mCm.setAirplaneMode(true);
+ // assert there is active network connection after airplane mode disabled
+ assertTrue("still has active network connection",
+ waitUntilNoActiveNetworkConnection(LONG_TIMEOUT));
- NetworkInfo networkInfo;
- if (!mWifiOnlyFlag) {
- assertTrue(waitForNetworkState(ConnectivityManager.TYPE_MOBILE,
- State.DISCONNECTED, LONG_TIMEOUT));
- networkInfo = mCM.getNetworkInfo(ConnectivityManager.TYPE_MOBILE);
- setStateTransitionCriteria(ConnectivityManager.TYPE_MOBILE,
- networkInfo.getState(),
- NetworkState.DO_NOTHING,
- State.DISCONNECTED);
- }
- networkInfo = mCM.getNetworkInfo(ConnectivityManager.TYPE_WIFI);
- setStateTransitionCriteria(ConnectivityManager.TYPE_WIFI, networkInfo.getState(),
- NetworkState.TO_CONNECTION, State.CONNECTED);
-
- // Connect to Wifi
+ // connect to Wifi
assertTrue("failed to connect to " + mTestAccessPoint,
connectToWifi(mTestAccessPoint));
assertTrue(waitForNetworkState(ConnectivityManager.TYPE_WIFI, State.CONNECTED,
WIFI_CONNECTION_TIMEOUT));
+ // verify that connection actually works
+ assertTrue("no network connectivity after wifi enable", checkNetworkConnectivity());
- // validate state and broadcast
- if (!validateNetworkStates(ConnectivityManager.TYPE_WIFI)) {
- log("state validate for Wifi failed");
- log("reason: " +
- getTransitionFailureReason(ConnectivityManager.TYPE_WIFI));
- assertTrue("State validation failed", false);
- }
- if (!mWifiOnlyFlag) {
- if (!validateNetworkStates(ConnectivityManager.TYPE_MOBILE)) {
- log("state validation for Mobile failed");
- log("reason: " +
- getTransitionFailureReason(ConnectivityManager.TYPE_MOBILE));
- assertTrue("state validation failed", false);
- }
- }
- mCM.setAirplaneMode(false);
+ // disable airplane mode
+ mCm.setAirplaneMode(false);
}
// Test case 7: test connectivity while transit from Wifi->AM->Wifi
@LargeTest
public void testDataConnectionWithWifiToAMToWifi () {
- if (mWifiOnlyFlag) {
- Log.v(TAG, this.getName() + " is excluded for wifi-only test");
- return;
- }
- // Connect to mTestAccessPoint
+ // connect to mTestAccessPoint
assertNotNull("SSID is null", mTestAccessPoint);
- // Connect to Wifi
+ // enable WiFi
+ assertTrue("failed to enable wifi", enableWifi());
+ // wait for wifi enable
+ assertTrue("wifi not enabled", waitForWifiState(
+ WifiManager.WIFI_STATE_ENABLED, LONG_TIMEOUT));
+ // connect to Wifi
assertTrue("failed to connect to " + mTestAccessPoint,
connectToWifi(mTestAccessPoint));
-
assertTrue(waitForNetworkState(ConnectivityManager.TYPE_WIFI, State.CONNECTED,
WIFI_CONNECTION_TIMEOUT));
- try {
- Thread.sleep(SHORT_TIMEOUT);
- } catch (Exception e) {
- log("exception: " + e.toString());
- }
+ // enable airplane mode without clearing Wifi
+ mCm.setAirplaneMode(true);
+ // assert there is active network connection after airplane mode disabled
+ assertTrue("still has active network connection",
+ waitUntilNoActiveNetworkConnection(LONG_TIMEOUT));
- // Enable airplane mode without clearing Wifi
- mCM.setAirplaneMode(true);
-
- assertTrue(waitForNetworkState(ConnectivityManager.TYPE_WIFI,
- State.DISCONNECTED, LONG_TIMEOUT));
-
- try {
- Thread.sleep(SHORT_TIMEOUT);
- } catch (Exception e) {
- log("exception: " + e.toString());
- }
-
- // Prepare for state validation
- NetworkInfo networkInfo = mCM.getNetworkInfo(ConnectivityManager.TYPE_WIFI);
- assertEquals(State.DISCONNECTED, networkInfo.getState());
- setStateTransitionCriteria(ConnectivityManager.TYPE_WIFI,
- networkInfo.getState(), NetworkState.TO_CONNECTION, State.CONNECTED);
-
- // Disable airplane mode
- mCM.setAirplaneMode(false);
-
- assertTrue(waitForNetworkState(ConnectivityManager.TYPE_WIFI, State.CONNECTED,
- WIFI_CONNECTION_TIMEOUT));
- if (!mWifiOnlyFlag) {
- assertTrue(waitForNetworkState(ConnectivityManager.TYPE_MOBILE,
- State.DISCONNECTED, LONG_TIMEOUT));
- }
-
- // validate the state transition
- if (!validateNetworkStates(ConnectivityManager.TYPE_WIFI)) {
- log("Wifi state transition validation failed.");
- log("reason: " +
- getTransitionFailureReason(ConnectivityManager.TYPE_WIFI));
- assertTrue(false);
- }
+ // disable airplane mode
+ mCm.setAirplaneMode(false);
+ // assert there is active network connection after airplane mode disabled
+ assertTrue("no active network connection after airplane mode disable",
+ waitForActiveNetworkConnection(LONG_TIMEOUT));
+ // assert that we have a Wifi connection
+ assertTrue("wifi not connected after airplane mode disable", waitForNetworkState(
+ ConnectivityManager.TYPE_WIFI, State.CONNECTED, WIFI_CONNECTION_TIMEOUT));
+ // verify that connection actually works
+ assertTrue("no network connectivity at end of test", checkNetworkConnectivity());
}
// Test case 8: test wifi state change while connecting/disconnecting to/from an AP
@LargeTest
public void testWifiStateChange () {
assertNotNull("SSID is null", mTestAccessPoint);
- //Connect to mTestAccessPoint
+ // enable WiFi
+ assertTrue("failed to enable wifi", enableWifi());
+ // wait for wifi enable
+ assertTrue("wifi not enabled", waitForWifiState(
+ WifiManager.WIFI_STATE_ENABLED, LONG_TIMEOUT));
+ // connect to Wifi
assertTrue("failed to connect to " + mTestAccessPoint,
connectToWifi(mTestAccessPoint));
- assertTrue(waitForWifiState(WifiManager.WIFI_STATE_ENABLED, LONG_TIMEOUT));
assertTrue(waitForNetworkState(ConnectivityManager.TYPE_WIFI, State.CONNECTED,
WIFI_CONNECTION_TIMEOUT));
- assertNotNull("Not associated with any AP",
- mWifiManager.getConnectionInfo().getBSSID());
+ assertNotNull("not associated with any AP", mWifiManager.getConnectionInfo().getBSSID());
- try {
- Thread.sleep(SHORT_TIMEOUT);
- } catch (Exception e) {
- log("exception: " + e.toString());
- }
+ // disconnect from the current AP
+ assertTrue("failed to disconnect from AP", disconnectAP());
- // Disconnect from the current AP
- log("disconnect from the AP");
- if (!disconnectAP()) {
- log("failed to disconnect from " + mTestAccessPoint);
- }
-
+ // below check disbabled since we have bug in what ConnectivityManager returns
// Verify the connectivity state for Wifi is DISCONNECTED
- assertTrue(waitForNetworkState(ConnectivityManager.TYPE_WIFI,
- State.DISCONNECTED, LONG_TIMEOUT));
+// assertTrue(waitForNetworkState(ConnectivityManager.TYPE_WIFI,
+// State.DISCONNECTED, LONG_TIMEOUT));
- if (!disableWifi()) {
- log("disable Wifi failed");
- return;
- }
- assertTrue(waitForWifiState(WifiManager.WIFI_STATE_DISABLED, LONG_TIMEOUT));
+ // disable WiFi
+ assertTrue("failed to disable wifi", disableWifi());
+ assertTrue("wifi state not disabled", waitForWifiState(
+ WifiManager.WIFI_STATE_DISABLED, LONG_TIMEOUT));
}
}
diff --git a/core/tests/ConnectivityManagerTest/src/com/android/connectivitymanagertest/functional/WifiAssociationTest.java b/core/tests/ConnectivityManagerTest/src/com/android/connectivitymanagertest/functional/WifiAssociationTest.java
index 183f2a9..afaf801 100644
--- a/core/tests/ConnectivityManagerTest/src/com/android/connectivitymanagertest/functional/WifiAssociationTest.java
+++ b/core/tests/ConnectivityManagerTest/src/com/android/connectivitymanagertest/functional/WifiAssociationTest.java
@@ -16,24 +16,22 @@
package com.android.connectivitymanagertest.functional;
-import com.android.connectivitymanagertest.ConnectivityManagerTestBase;
-import com.android.connectivitymanagertest.WifiAssociationTestRunner;
-
-import android.content.Context;
-import android.os.Bundle;
-import android.net.wifi.WifiInfo;
-import android.net.wifi.WifiConfiguration;
-import android.net.wifi.WifiConfiguration.KeyMgmt;
-import android.net.wifi.WifiConfiguration.AuthAlgorithm;
-import android.net.wifi.WifiConfiguration.GroupCipher;
-import android.net.wifi.WifiConfiguration.PairwiseCipher;
-import android.net.wifi.WifiConfiguration.Protocol;
-import android.net.wifi.WifiManager;
import android.net.ConnectivityManager;
import android.net.NetworkInfo.State;
+import android.net.wifi.WifiConfiguration;
+import android.net.wifi.WifiConfiguration.AuthAlgorithm;
+import android.net.wifi.WifiConfiguration.GroupCipher;
+import android.net.wifi.WifiConfiguration.KeyMgmt;
+import android.net.wifi.WifiConfiguration.PairwiseCipher;
+import android.net.wifi.WifiConfiguration.Protocol;
+import android.net.wifi.WifiInfo;
+import android.os.Bundle;
import android.test.suitebuilder.annotation.LargeTest;
import android.util.Log;
+import com.android.connectivitymanagertest.ConnectivityManagerTestBase;
+import com.android.connectivitymanagertest.WifiAssociationTestRunner;
+
/**
* Test Wi-Fi connection with different configuration
* To run this tests:
@@ -41,8 +39,7 @@
* -e security-type [OPEN|WEP64|WEP128|WPA_TKIP|WPA2_AES] -e frequency-band [2.4|5.0|auto]
* -w com.android.connectivitymanagertest/.WifiAssociationTestRunner"
*/
-public class WifiAssociationTest
- extends ConnectivityManagerTestBase {
+public class WifiAssociationTest extends ConnectivityManagerTestBase {
private static final String TAG = "WifiAssociationTest";
private String mSsid = null;
private String mPassword = null;
@@ -55,67 +52,37 @@
}
@Override
- public void setUp() throws Exception {
+ protected void setUp() throws Exception {
super.setUp();
- WifiAssociationTestRunner mRunner = (WifiAssociationTestRunner)getInstrumentation();
- Bundle arguments = mRunner.getArguments();
+ WifiAssociationTestRunner runner = (WifiAssociationTestRunner)getInstrumentation();
+ Bundle arguments = runner.getArguments();
mSecurityType = arguments.getString("security-type");
mSsid = arguments.getString("ssid");
mPassword = arguments.getString("password");
mFrequencyBand = arguments.getString("frequency-band");
- mBand = mRunner.mBand;
- assertNotNull("Security type is empty", mSecurityType);
- assertNotNull("Ssid is empty", mSsid);
+ mBand = runner.mBand;
+ assertNotNull("security type is empty", mSecurityType);
+ assertNotNull("ssid is empty", mSsid);
validateFrequencyBand();
- // enable Wifi and verify wpa_supplicant is started
- assertTrue("enable Wifi failed", enableWifi());
- sleep(2 * SHORT_TIMEOUT, "interrupted while waiting for WPA_SUPPLICANT to start");
- WifiInfo mConnection = mWifiManager.getConnectionInfo();
- assertNotNull(mConnection);
- assertTrue("wpa_supplicant is not started ", mWifiManager.pingSupplicant());
- }
- @Override
- public void tearDown() throws Exception {
- super.tearDown();
+ // enable wifi and verify wpa_supplicant is started
+ assertTrue("enable Wifi failed", enableWifi());
+ assertTrue("wifi not connected", waitForNetworkState(
+ ConnectivityManager.TYPE_WIFI, State.CONNECTED, LONG_TIMEOUT));
+ WifiInfo wi = mWifiManager.getConnectionInfo();
+ assertNotNull("no active wifi info", wi);
+ assertTrue("failed to ping wpa_supplicant ", mWifiManager.pingSupplicant());
}
private void validateFrequencyBand() {
if (mFrequencyBand != null) {
int currentFreq = mWifiManager.getFrequencyBand();
Log.v(TAG, "read frequency band: " + currentFreq);
- assertTrue("device frequency band is not set successfully", (mBand == currentFreq));
+ assertEquals("specified frequency band does not match operational band of WifiManager",
+ currentFreq, mBand);
}
}
- /**
- * Connect to the provided Wi-Fi network
- * @param config is the network configuration
- * @return true if the connection is successful.
- */
- private void connectToWifi(WifiConfiguration config) {
- // step 1: connect to the test access point
- assertTrue("failed to associate with " + config.SSID,
- connectToWifiWithConfiguration(config));
-
- // step 2: verify Wifi state and network state;
- assertTrue("failed to connect with " + config.SSID,
- waitForNetworkState(ConnectivityManager.TYPE_WIFI,
- State.CONNECTED, WIFI_CONNECTION_TIMEOUT));
-
- // step 3: verify the current connected network is the given SSID
- assertNotNull("Wifi connection returns null", mWifiManager.getConnectionInfo());
- assertTrue(config.SSID.contains(mWifiManager.getConnectionInfo().getSSID()));
- }
-
- private void sleep(long sometime, String errorMsg) {
- try {
- Thread.sleep(sometime);
- } catch (InterruptedException e) {
- fail(errorMsg);
- }
- }
-
private void log(String message) {
Log.v(TAG, message);
}
@@ -133,42 +100,37 @@
config.allowedKeyManagement.set(KeyMgmt.NONE);
break;
case WEP64:
- // always use hex pair for WEP-40
- assertTrue("not a WEP64 security type?", mPassword.length() == 10);
+ assertNotNull("password is empty", mPassword);
config.allowedKeyManagement.set(KeyMgmt.NONE);
config.allowedAuthAlgorithms.set(AuthAlgorithm.OPEN);
config.allowedAuthAlgorithms.set(AuthAlgorithm.SHARED);
config.allowedGroupCiphers.set(GroupCipher.WEP40);
if (mPassword != null) {
- int length = mPassword.length();
- // WEP-40
- if (mPassword.matches("[0-9A-Fa-f]*")) {
+ // always use hex pair for WEP-40
+ if (isHex(mPassword, 10)) {
config.wepKeys[0] = mPassword;
} else {
- fail("Please type hex pair for the password");
+ fail("password should be 10-character hex");
}
}
break;
case WEP128:
assertNotNull("password is empty", mPassword);
- // always use hex pair for WEP-104
- assertTrue("not a WEP128 security type?", mPassword.length() == 26);
config.allowedKeyManagement.set(KeyMgmt.NONE);
config.allowedAuthAlgorithms.set(AuthAlgorithm.OPEN);
config.allowedAuthAlgorithms.set(AuthAlgorithm.SHARED);
config.allowedGroupCiphers.set(GroupCipher.WEP104);
if (mPassword != null) {
- int length = mPassword.length();
- // WEP-40
- if (mPassword.matches("[0-9A-Fa-f]*")) {
+ // always use hex pair for WEP-104
+ if (isHex(mPassword, 26)) {
config.wepKeys[0] = mPassword;
} else {
- fail("Please type hex pair for the password");
+ fail("password should be 26-character hex");
}
}
break;
case WPA_TKIP:
- assertNotNull("missing password", mPassword);
+ assertNotNull("password is empty", mPassword);
config.allowedKeyManagement.set(KeyMgmt.WPA_PSK);
config.allowedAuthAlgorithms.set(AuthAlgorithm.OPEN);
config.allowedProtocols.set(Protocol.WPA);
@@ -181,7 +143,7 @@
}
break;
case WPA2_AES:
- assertNotNull("missing password", mPassword);
+ assertNotNull("password is empty", mPassword);
config.allowedKeyManagement.set(KeyMgmt.WPA_PSK);
config.allowedAuthAlgorithms.set(AuthAlgorithm.OPEN);
config.allowedProtocols.set(Protocol.RSN);
@@ -200,5 +162,7 @@
}
Log.v(TAG, "network config: " + config.toString());
connectToWifi(config);
+ // verify that connection actually works
+ assertTrue("no network connectivity at end of test", checkNetworkConnectivity());
}
}
diff --git a/core/tests/ConnectivityManagerTest/src/com/android/connectivitymanagertest/functional/WifiConnectionTest.java b/core/tests/ConnectivityManagerTest/src/com/android/connectivitymanagertest/functional/WifiConnectionTest.java
index ad73ee1..740ffb8 100644
--- a/core/tests/ConnectivityManagerTest/src/com/android/connectivitymanagertest/functional/WifiConnectionTest.java
+++ b/core/tests/ConnectivityManagerTest/src/com/android/connectivitymanagertest/functional/WifiConnectionTest.java
@@ -16,21 +16,17 @@
package com.android.connectivitymanagertest.functional;
-import com.android.connectivitymanagertest.ConnectivityManagerTestBase;
-import com.android.connectivitymanagertest.ConnectivityManagerTestRunner;
-
-import android.content.Context;
-import android.net.wifi.WifiConfiguration;
-import android.net.wifi.WifiInfo;
-import android.net.wifi.WifiManager;
import android.net.ConnectivityManager;
import android.net.NetworkInfo.State;
+import android.net.wifi.WifiConfiguration;
+import android.net.wifi.WifiInfo;
import android.test.suitebuilder.annotation.LargeTest;
import android.util.Log;
+import com.android.connectivitymanagertest.ConnectivityManagerTestBase;
+
import java.util.ArrayList;
import java.util.List;
-import java.util.Set;
/**
* Test Wi-Fi connection with different configuration
@@ -43,28 +39,29 @@
extends ConnectivityManagerTestBase {
private static final String TAG = "WifiConnectionTest";
private static final boolean DEBUG = false;
- private List<WifiConfiguration> networks = new ArrayList<WifiConfiguration>();
+ private List<WifiConfiguration> mNetworks = new ArrayList<WifiConfiguration>();
@Override
public void setUp() throws Exception {
super.setUp();
- networks = loadNetworkConfigurations();
+ mNetworks = loadNetworkConfigurations();
if (DEBUG) {
printNetworkConfigurations();
}
- // enable Wifi and verify wpa_supplicant is started
+ // enable wifi and verify wpa_supplicant is started
assertTrue("enable Wifi failed", enableWifi());
- sleep(2 * SHORT_TIMEOUT, "interrupted while waiting for WPA_SUPPLICANT to start");
- WifiInfo mConnection = mWifiManager.getConnectionInfo();
- assertNotNull(mConnection);
- assertTrue("wpa_supplicant is not started ", mWifiManager.pingSupplicant());
+ assertTrue("wifi not connected", waitForNetworkState(
+ ConnectivityManager.TYPE_WIFI, State.CONNECTED, LONG_TIMEOUT));
+ WifiInfo wi = mWifiManager.getConnectionInfo();
+ assertNotNull("no active wifi info", wi);
+ assertTrue("failed to ping wpa_supplicant ", mWifiManager.pingSupplicant());
}
private void printNetworkConfigurations() {
log("==== print network configurations parsed from XML file ====");
- log("number of access points: " + networks.size());
- for (WifiConfiguration config : networks) {
+ log("number of access points: " + mNetworks.size());
+ for (WifiConfiguration config : mNetworks) {
log(config.toString());
}
}
@@ -75,51 +72,25 @@
super.tearDown();
}
- /**
- * Connect to the provided Wi-Fi network
- * @param config is the network configuration
- * @return true if the connection is successful.
- */
- private void connectToWifi(WifiConfiguration config) {
- // step 1: connect to the test access point
- assertTrue("failed to connect to " + config.SSID,
- connectToWifiWithConfiguration(config));
-
- // step 2: verify Wifi state and network state;
- assertTrue(waitForNetworkState(ConnectivityManager.TYPE_WIFI,
- State.CONNECTED, WIFI_CONNECTION_TIMEOUT));
-
- // step 3: verify the current connected network is the given SSID
- assertNotNull("Wifi connection returns null", mWifiManager.getConnectionInfo());
- if (DEBUG) {
- log("config.SSID = " + config.SSID);
- log("mWifiManager.getConnectionInfo.getSSID()" +
- mWifiManager.getConnectionInfo().getSSID());
- }
- assertTrue(config.SSID.contains(mWifiManager.getConnectionInfo().getSSID()));
- }
-
- private void sleep(long sometime, String errorMsg) {
- try {
- Thread.sleep(sometime);
- } catch (InterruptedException e) {
- fail(errorMsg);
- }
- }
-
private void log(String message) {
Log.v(TAG, message);
}
@LargeTest
public void testWifiConnections() {
- for (int i = 0; i < networks.size(); i++) {
- String ssid = networks.get(i).SSID;
+ for (int i = 0; i < mNetworks.size(); i++) {
+ String ssid = mNetworks.get(i).SSID;
log("-- START Wi-Fi connection test to : " + ssid + " --");
- connectToWifi(networks.get(i));
- // wait for 2 minutes between wifi stop and start
- sleep(WIFI_STOP_START_INTERVAL, "interruped while connected to wifi");
+ connectToWifi(mNetworks.get(i));
+ // verify that connection actually works
+ assertTrue("no network connectivity at end of test", checkNetworkConnectivity());
log("-- END Wi-Fi connection test to " + ssid + " -- ");
+ log("pausing for 1 minute");
+ try {
+ Thread.sleep(60 * 1000);
+ } catch (InterruptedException e) {
+ // ignore
+ }
}
}
}
diff --git a/core/tests/ConnectivityManagerTest/src/com/android/connectivitymanagertest/stress/WifiApStress.java b/core/tests/ConnectivityManagerTest/src/com/android/connectivitymanagertest/stress/WifiApStress.java
index 790ca38..aead65b 100644
--- a/core/tests/ConnectivityManagerTest/src/com/android/connectivitymanagertest/stress/WifiApStress.java
+++ b/core/tests/ConnectivityManagerTest/src/com/android/connectivitymanagertest/stress/WifiApStress.java
@@ -33,7 +33,7 @@
import java.io.FileWriter;
/**
- * Stress the wifi driver as access point.
+ * Stress test setting up device as wifi hotspot
*/
public class WifiApStress
extends ConnectivityManagerTestBase {
@@ -41,27 +41,28 @@
private static String NETWORK_ID = "AndroidAPTest";
private static String PASSWD = "androidwifi";
private final static String OUTPUT_FILE = "WifiStressTestOutput.txt";
- private int iterations;
+ private int mTotalIterations;
private BufferedWriter mOutputWriter = null;
private int mLastIteration = 0;
private boolean mWifiOnlyFlag;
@Override
- public void setUp() throws Exception {
+ protected void setUp() throws Exception {
super.setUp();
ConnectivityManagerStressTestRunner mRunner =
(ConnectivityManagerStressTestRunner)getInstrumentation();
- iterations = mRunner.mSoftapIterations;
- mWifiOnlyFlag = mRunner.mWifiOnlyFlag;
+ mTotalIterations = mRunner.getSoftApInterations();
+ mWifiOnlyFlag = mRunner.isWifiOnly();
turnScreenOn();
}
@Override
- public void tearDown() throws Exception {
+ protected void tearDown() throws Exception {
// write the total number of iterations into output file
mOutputWriter = new BufferedWriter(new FileWriter(new File(
Environment.getExternalStorageDirectory(), OUTPUT_FILE)));
- mOutputWriter.write(String.format("iteration %d out of %d\n", mLastIteration, iterations));
+ mOutputWriter.write(String.format("iteration %d out of %d\n",
+ mLastIteration + 1, mTotalIterations));
mOutputWriter.flush();
mOutputWriter.close();
super.tearDown();
@@ -79,40 +80,44 @@
config.allowedAuthAlgorithms.set(AuthAlgorithm.OPEN);
config.preSharedKey = PASSWD;
- // If Wifi is enabled, disable it
+ // if wifiap enabled, disable it
+ assertTrue("failed to disable wifi hotspot",
+ mWifiManager.setWifiApEnabled(config, false));
+ assertTrue("wifi hotspot not enabled", waitForWifiApState(
+ WifiManager.WIFI_AP_STATE_DISABLED, 2 * LONG_TIMEOUT));
+
+ // if Wifi is enabled, disable it
if (mWifiManager.isWifiEnabled()) {
- disableWifi();
+ assertTrue("failed to disable wifi", disableWifi());
+ // wait for the wifi state to be DISABLED
+ assertTrue("wifi state not disabled", waitForWifiState(
+ WifiManager.WIFI_STATE_DISABLED, LONG_TIMEOUT));
}
int i;
- for (i = 0; i < iterations; i++) {
+ for (i = 0; i < mTotalIterations; i++) {
Log.v(TAG, "iteration: " + i);
mLastIteration = i;
// enable Wifi tethering
- assertTrue(mWifiManager.setWifiApEnabled(config, true));
- // Wait for wifi ap state to be ENABLED
- assertTrue(waitForWifiAPState(WifiManager.WIFI_AP_STATE_ENABLED, 2 * LONG_TIMEOUT));
- // Wait for wifi tethering result
- assertEquals(SUCCESS, waitForTetherStateChange(2 * SHORT_TIMEOUT));
- // Allow the wifi tethering to be enabled for 10 seconds
+ assertTrue("failed to enable wifi hotspot",
+ mWifiManager.setWifiApEnabled(config, true));
+ // wait for wifi ap state to be ENABLED
+ assertTrue("wifi hotspot not enabled", waitForWifiApState(
+ WifiManager.WIFI_AP_STATE_ENABLED, 2 * LONG_TIMEOUT));
+ // wait for wifi tethering result
+ assertTrue("tether state not changed", waitForTetherStateChange(LONG_TIMEOUT));
+ // allow the wifi tethering to be enabled for 10 seconds
try {
Thread.sleep(2 * SHORT_TIMEOUT);
} catch (Exception e) {
- fail("thread in sleep is interrupted");
+ // ignore
}
assertTrue("no uplink data connection after Wi-Fi tethering", pingTest(null));
- // Disable soft AP
- assertTrue(mWifiManager.setWifiApEnabled(config, false));
- // Wait for 30 seconds until Wi-Fi tethering is stopped
- try {
- Thread.sleep(30 * 1000);
- Log.v(TAG, "wait for Wi-Fi tethering to be disabled.");
- } catch (Exception e) {
- fail("thread in sleep is interrupted");
- }
- assertFalse("Wi-Fi AP disable failed", mWifiManager.isWifiApEnabled());
- }
- if (i == iterations) {
- mLastIteration = iterations;
+ // disable wifi hotspot
+ assertTrue("failed to disable wifi hotspot",
+ mWifiManager.setWifiApEnabled(config, false));
+ assertTrue("wifi hotspot not enabled", waitForWifiApState(
+ WifiManager.WIFI_AP_STATE_DISABLED, 2 * LONG_TIMEOUT));
+ assertFalse("wifi hotspot still enabled", mWifiManager.isWifiApEnabled());
}
}
diff --git a/core/tests/ConnectivityManagerTest/src/com/android/connectivitymanagertest/stress/WifiStressTest.java b/core/tests/ConnectivityManagerTest/src/com/android/connectivitymanagertest/stress/WifiStressTest.java
index 08618d6..859c30c 100644
--- a/core/tests/ConnectivityManagerTest/src/com/android/connectivitymanagertest/stress/WifiStressTest.java
+++ b/core/tests/ConnectivityManagerTest/src/com/android/connectivitymanagertest/stress/WifiStressTest.java
@@ -16,20 +16,22 @@
package com.android.connectivitymanagertest.stress;
+import android.app.Activity;
import android.content.Context;
import android.net.ConnectivityManager;
import android.net.IpConfiguration.IpAssignment;
import android.net.IpConfiguration.ProxySettings;
+import android.net.NetworkInfo;
import android.net.NetworkInfo.State;
import android.net.wifi.ScanResult;
-import android.net.wifi.WifiConfiguration.KeyMgmt;
import android.net.wifi.WifiConfiguration;
+import android.net.wifi.WifiConfiguration.KeyMgmt;
import android.net.wifi.WifiManager;
+import android.os.Bundle;
import android.os.Environment;
import android.os.PowerManager;
import android.os.SystemClock;
import android.provider.Settings;
-import android.view.KeyEvent;
import android.test.suitebuilder.annotation.LargeTest;
import android.util.Log;
@@ -49,15 +51,14 @@
* adb shell am instrument -e class com.android.connectivitymanagertest.stress.WifiStressTest
* -w com.android.connectivitymanagertest/.ConnectivityManagerStressTestRunner
*/
-public class WifiStressTest
- extends ConnectivityManagerTestBase {
+public class WifiStressTest extends ConnectivityManagerTestBase {
private final static String TAG = "WifiStressTest";
private final static long SCREEN_OFF_TIMER = 500; //500ms
/**
* Wi-Fi idle time for default sleep policy
*/
- private final static long WIFI_IDLE_MS = 60 * 1000;
+ private final static long WIFI_IDLE_MS = 15 * 1000;
/**
* Delay after issuing wifi shutdown.
@@ -69,7 +70,7 @@
private final static String OUTPUT_FILE = "WifiStressTestOutput.txt";
private int mReconnectIterations;
- private int mWifiSleepTime;
+ private long mWifiSleepTime;
private int mScanIterations;
private String mSsid;
private String mPassword;
@@ -78,16 +79,16 @@
private boolean mWifiOnlyFlag;
@Override
- public void setUp() throws Exception {
+ protected void setUp() throws Exception {
super.setUp();
mRunner = (ConnectivityManagerStressTestRunner) getInstrumentation();
- mReconnectIterations = mRunner.mReconnectIterations;
- mSsid = mRunner.mReconnectSsid;
- mPassword = mRunner.mReconnectPassword;
- mScanIterations = mRunner.mScanIterations;
- mWifiSleepTime = mRunner.mSleepTime;
- mWifiOnlyFlag = mRunner.mWifiOnlyFlag;
+ mReconnectIterations = mRunner.getReconnectIterations();
+ mSsid = mRunner.getReconnectSsid();
+ mPassword = mRunner.getReconnectPassword();
+ mScanIterations = mRunner.getScanIterations();
+ mWifiSleepTime = mRunner.getSleepTime();
+ mWifiOnlyFlag = mRunner.isWifiOnly();
log(String.format("mReconnectIterations(%d), mSsid(%s), mPassword(%s),"
+ "mScanIterations(%d), mWifiSleepTime(%d)", mReconnectIterations, mSsid,
mPassword, mScanIterations, mWifiSleepTime));
@@ -105,7 +106,7 @@
}
@Override
- public void tearDown() throws Exception {
+ protected void tearDown() throws Exception {
log("tearDown()");
if (mOutputWriter != null) {
mOutputWriter.close();
@@ -145,109 +146,90 @@
*/
@LargeTest
public void testWifiScanning() {
- int scanTimeSum = 0;
- int i;
+ long scanTimeSum = 0, i, averageScanTime = -1;
int ssidAppearInScanResultsCount = 0; // count times of given ssid appear in scan results.
- for (i = 0; i < mScanIterations; i++) {
+ for (i = 1; i <= mScanIterations; i++) {
log("testWifiScanning: iteration: " + i);
- int averageScanTime = 0;
- if (i > 0) {
- averageScanTime = scanTimeSum/i;
- }
- writeOutput(String.format("iteration %d out of %d",
- i, mScanIterations));
+ averageScanTime = scanTimeSum / i;
+ writeOutput(String.format("iteration %d out of %d", i, mScanIterations));
writeOutput(String.format("average scanning time is %d", averageScanTime));
writeOutput(String.format("ssid appear %d out of %d scan iterations",
ssidAppearInScanResultsCount, i));
- long startTime = SystemClock.uptimeMillis();
- scanResultAvailable = false;
- assertTrue("start scan failed", mWifiManager.startScan());
- while (true) {
- if ((SystemClock.uptimeMillis() - startTime) >
- WIFI_SCAN_TIMEOUT) {
- fail("Wifi scanning takes more than " + WIFI_SCAN_TIMEOUT + " ms");
+ List<ScanResult> scanResultLocal = null;
+ // wait for a scan result
+ long start = 0;
+ synchronized (mWifiScanResultLock) {
+ start = SystemClock.uptimeMillis();
+ assertTrue("start scan failed", mWifiManager.startScan());
+ try {
+ mWifiScanResultLock.wait(WAIT_FOR_SCAN_RESULT);
+ } catch (InterruptedException e) {
+ // ignore
}
- synchronized(this) {
- try {
- wait(WAIT_FOR_SCAN_RESULT);
- } catch (InterruptedException e) {
- e.printStackTrace();
- }
- if (scanResultAvailable) {
- long scanTime = (SystemClock.uptimeMillis() - startTime);
- scanTimeSum += scanTime;
- break;
- }
- }
+ scanTimeSum += SystemClock.uptimeMillis() - start;
+ // save the scan result while in lock
+ scanResultLocal = mLastScanResult;
}
- if ((mWifiManager.getScanResults() == null) ||
- (mWifiManager.getScanResults().size() <= 0)) {
+ if (scanResultLocal == null || scanResultLocal.isEmpty()) {
fail("Scan results are empty ");
}
-
- List<ScanResult> netList = mWifiManager.getScanResults();
- if (netList != null) {
- log("size of scan result list: " + netList.size());
- for (int s = 0; s < netList.size(); s++) {
- ScanResult sr= netList.get(s);
- log(String.format("scan result for %s is: %s", sr.SSID, sr.toString()));
- log(String.format("signal level for %s is %d ", sr.SSID, sr.level));
- if (sr.SSID.equals(mSsid)) {
- ssidAppearInScanResultsCount += 1;
- log("Number of times " + mSsid + " appear in the scan list: " +
- ssidAppearInScanResultsCount);
- break;
- }
+ log("size of scan result list: " + scanResultLocal.size());
+ for (ScanResult sr : scanResultLocal) {
+ log(String.format("scan result: " + sr.toString()));
+ if (mSsid.equals(sr.SSID)) {
+ ssidAppearInScanResultsCount += 1;
+ break;
}
}
}
- if (i == mScanIterations) {
- writeOutput(String.format("iteration %d out of %d",
- i, mScanIterations));
- writeOutput(String.format("average scanning time is %d", scanTimeSum/mScanIterations));
+ Bundle result = new Bundle();
+ result.putLong("actual-iterations", i - 1);
+ result.putLong("avg-scan-time", averageScanTime);
+ result.putInt("ap-discovered", ssidAppearInScanResultsCount);
+ getInstrumentation().sendStatus(Activity.RESULT_FIRST_USER, result);
+ if (i == mScanIterations + 1) {
+ writeOutput(String.format("iteration %d out of %d", i, mScanIterations));
+ writeOutput(String.format("average scanning time is %d", scanTimeSum / (i - 1)));
writeOutput(String.format("ssid appear %d out of %d scan iterations",
- ssidAppearInScanResultsCount, mScanIterations));
+ ssidAppearInScanResultsCount, i));
}
}
// Stress Wifi reconnection to secure net after sleep
@LargeTest
public void testWifiReconnectionAfterSleep() {
- int value = Settings.Global.getInt(mRunner.getContext().getContentResolver(),
- Settings.Global.WIFI_SLEEP_POLICY, -1);
- log("wifi sleep policy is: " + value);
- if (value != Settings.Global.WIFI_SLEEP_POLICY_DEFAULT) {
- Settings.Global.putInt(mRunner.getContext().getContentResolver(),
- Settings.Global.WIFI_SLEEP_POLICY, Settings.Global.WIFI_SLEEP_POLICY_DEFAULT);
- log("set wifi sleep policy to default value");
- }
+ // set wifi sleep policy to never on while in sleep
+ Settings.Global.putInt(mRunner.getContext().getContentResolver(),
+ Settings.Global.WIFI_SLEEP_POLICY, Settings.Global.WIFI_SLEEP_POLICY_NEVER);
+ // set idle timeout for wifi to 15s
Settings.Global.putLong(mRunner.getContext().getContentResolver(),
Settings.Global.WIFI_IDLE_MS, WIFI_IDLE_MS);
// Connect to a Wi-Fi network
WifiConfiguration config = new WifiConfiguration();
config.SSID = mSsid;
- config.allowedKeyManagement.set(KeyMgmt.WPA_PSK);
- if (mPassword.matches("[0-9A-Fa-f]{64}")) {
- config.preSharedKey = mPassword;
+ if (mPassword != null) {
+ config.allowedKeyManagement.set(KeyMgmt.WPA_PSK);
+ if (isHex(mPassword, 64)) {
+ config.preSharedKey = mPassword;
+ } else {
+ config.preSharedKey = '"' + mPassword + '"';
+ }
} else {
- config.preSharedKey = '"' + mPassword + '"';
+ config.allowedKeyManagement.set(KeyMgmt.NONE);
}
config.setIpAssignment(IpAssignment.DHCP);
config.setProxySettings(ProxySettings.NONE);
assertTrue("Failed to connect to Wi-Fi network: " + mSsid,
connectToWifiWithConfiguration(config));
- assertTrue(waitForWifiState(WifiManager.WIFI_STATE_ENABLED,
- SHORT_TIMEOUT));
- assertTrue(waitForNetworkState(ConnectivityManager.TYPE_WIFI, State.CONNECTED,
- WIFI_CONNECTION_TIMEOUT));
+ assertTrue("wifi not connected", waitForNetworkState(ConnectivityManager.TYPE_WIFI,
+ State.CONNECTED, WIFI_CONNECTION_TIMEOUT));
// Run ping test to verify the data connection
assertTrue("Wi-Fi is connected, but no data connection.", pingTest(null));
- int i;
- long sum = 0;
- for (i = 0; i < mReconnectIterations; i++) {
+ long i, sum = 0, avgReconnectTime = 0;
+ for (i = 1; i <= mReconnectIterations; i++) {
// 1. Put device into sleep mode
// 2. Wait for the device to sleep for sometime, verify wi-fi is off and mobile is on.
// 3. Maintain the sleep mode for some time,
@@ -261,53 +243,61 @@
long start = SystemClock.uptimeMillis();
PowerManager pm =
(PowerManager)mRunner.getContext().getSystemService(Context.POWER_SERVICE);
- while (pm.isScreenOn() && ((SystemClock.uptimeMillis() - start) < SCREEN_OFF_TIMER)) {
- sleep(100, "wait for screen off");
+ while (pm.isInteractive() &&
+ ((SystemClock.uptimeMillis() - start) < SCREEN_OFF_TIMER)) {
+ SystemClock.sleep(100);
}
- assertFalse(pm.isScreenOn());
- sleep(WIFI_IDLE_MS + WIFI_SHUTDOWN_DELAY, "Interruped while wait for wifi to be idle");
- assertTrue("Wait for Wi-Fi to idle timeout",
- waitForNetworkState(ConnectivityManager.TYPE_WIFI, State.DISCONNECTED,
- 6 * SHORT_TIMEOUT));
- if (!mWifiOnlyFlag) {
+ assertFalse("screen still on", pm.isInteractive());
+ // wait for WiFi timeout
+ SystemClock.sleep(WIFI_IDLE_MS + WIFI_SHUTDOWN_DELAY);
+ // below check temporarily disabled due to bug in ConnectivityManager return
+// assertTrue("Wait for Wi-Fi to idle timeout",
+// waitForNetworkState(ConnectivityManager.TYPE_WIFI, State.DISCONNECTED,
+// 6 * SHORT_TIMEOUT));
+ if (mWifiOnlyFlag) {
+ assertTrue("expected wifi disconnect, still has active connection",
+ waitUntilNoActiveNetworkConnection(2 * LONG_TIMEOUT));
+ } else {
// use long timeout as the pppd startup may take several retries.
- assertTrue("Wait for cellular connection timeout",
+ assertTrue("no fallback on mobile or wifi didn't disconnect",
waitForNetworkState(ConnectivityManager.TYPE_MOBILE, State.CONNECTED,
2 * LONG_TIMEOUT));
}
- sleep(mWifiSleepTime, "Interrupted while device is in sleep mode");
- // Verify the wi-fi is still off and data connection is on
- assertEquals("Wi-Fi is reconnected", State.DISCONNECTED,
- mCM.getNetworkInfo(ConnectivityManager.TYPE_WIFI).getState());
-
- if (!mWifiOnlyFlag) {
- assertEquals("Cellular connection is down", State.CONNECTED,
- mCM.getNetworkInfo(ConnectivityManager.TYPE_MOBILE).getState());
- assertTrue("Mobile is connected, but no data connection.", pingTest(null));
+ SystemClock.sleep(mWifiSleepTime);
+ // verify the wi-fi is still off and either we have no connectivity or fallback on mobile
+ if (mWifiOnlyFlag) {
+ NetworkInfo ni = mCm.getActiveNetworkInfo();
+ if (ni != null) {
+ Log.e(TAG, "has active network while in wifi sleep: " + ni.toString());
+ fail("active network detected");
+ }
+ } else {
+ assertEquals("mobile not connected", State.CONNECTED,
+ mCm.getNetworkInfo(ConnectivityManager.TYPE_MOBILE).getState());
+ assertTrue("no connectivity over mobile", pingTest(null));
}
// Turn screen on again
turnScreenOn();
- // Wait for 2 seconds for the lock screen
- sleep(2 * 1000, "wait 2 seconds for lock screen");
- // Disable lock screen by inject menu key event
- mRunner.sendKeyDownUpSync(KeyEvent.KEYCODE_MENU);
-
// Measure the time for Wi-Fi to get connected
long startTime = SystemClock.uptimeMillis();
- assertTrue("Wait for Wi-Fi enable timeout after wake up",
- waitForWifiState(WifiManager.WIFI_STATE_ENABLED,
- SHORT_TIMEOUT));
- assertTrue("Wait for Wi-Fi connection timeout after wake up",
+ assertTrue("screen on: wifi not enabled before timeout",
+ waitForWifiState(WifiManager.WIFI_STATE_ENABLED, SHORT_TIMEOUT));
+ assertTrue("screen on: wifi not connected before timeout",
waitForNetworkState(ConnectivityManager.TYPE_WIFI, State.CONNECTED,
- WIFI_CONNECTION_TIMEOUT));
+ LONG_TIMEOUT));
long connectionTime = SystemClock.uptimeMillis() - startTime;
sum += connectionTime;
- log("average reconnection time is: " + sum/(i+1));
+ avgReconnectTime = sum / i;
+ log("average reconnection time is: " + avgReconnectTime);
assertTrue("Reconnect to Wi-Fi network, but no data connection.", pingTest(null));
}
- if (i == mReconnectIterations) {
+ Bundle result = new Bundle();
+ result.putLong("actual-iterations", i - 1);
+ result.putLong("avg-reconnect-time", avgReconnectTime);
+ getInstrumentation().sendStatus(Activity.RESULT_FIRST_USER, result);
+ if (i == mReconnectIterations + 1) {
writeOutput(String.format("iteration %d out of %d",
i, mReconnectIterations));
}
diff --git a/core/tests/ConnectivityManagerTest/src/com/android/connectivitymanagertest/unit/WifiClientTest.java b/core/tests/ConnectivityManagerTest/src/com/android/connectivitymanagertest/unit/WifiClientTest.java
index 7a9bc78..5c2f388 100644
--- a/core/tests/ConnectivityManagerTest/src/com/android/connectivitymanagertest/unit/WifiClientTest.java
+++ b/core/tests/ConnectivityManagerTest/src/com/android/connectivitymanagertest/unit/WifiClientTest.java
@@ -38,7 +38,6 @@
public class WifiClientTest extends AndroidTestCase {
private WifiManager mWifiManager;
- private final String TAG = "WifiClientTest";
//10s delay for turning on wifi
private static final int DELAY = 10000;
diff --git a/packages/SystemUI/res/values/strings.xml b/packages/SystemUI/res/values/strings.xml
index ed3fa58..8f05f7b 100644
--- a/packages/SystemUI/res/values/strings.xml
+++ b/packages/SystemUI/res/values/strings.xml
@@ -834,8 +834,8 @@
<!-- Monitoring dialog title for normal devices [CHAR LIMIT=35]-->
<string name="monitoring_title">Network monitoring</string>
- <!-- Monitoring dialog open app button [CHAR LIMIT=30] -->
- <string name="open_app">Open app</string>
+ <!-- Monitoring dialog disable vpn button [CHAR LIMIT=30] -->
+ <string name="disable_vpn">Disable VPN</string>
<!-- Monitoring dialog disconnect vpn button [CHAR LIMIT=30] -->
<string name="disconnect_vpn">Disconnect VPN</string>
diff --git a/packages/SystemUI/src/com/android/systemui/qs/QSFooter.java b/packages/SystemUI/src/com/android/systemui/qs/QSFooter.java
index bfbc56c..a8199fa 100644
--- a/packages/SystemUI/src/com/android/systemui/qs/QSFooter.java
+++ b/packages/SystemUI/src/com/android/systemui/qs/QSFooter.java
@@ -119,11 +119,7 @@
@Override
public void onClick(DialogInterface dialog, int which) {
if (which == DialogInterface.BUTTON_NEGATIVE) {
- if (mSecurityController.isLegacyVpn()) {
- mSecurityController.disconnectFromLegacyVpn();
- } else {
- mSecurityController.openVpnApp();
- }
+ mSecurityController.disconnectFromVpn();
}
}
@@ -142,7 +138,7 @@
if (mSecurityController.isLegacyVpn()) {
return mContext.getString(R.string.disconnect_vpn);
} else {
- return mContext.getString(R.string.open_app);
+ return mContext.getString(R.string.disable_vpn);
}
}
diff --git a/packages/SystemUI/src/com/android/systemui/statusbar/policy/SecurityController.java b/packages/SystemUI/src/com/android/systemui/statusbar/policy/SecurityController.java
index ede8129..3a5a53b 100644
--- a/packages/SystemUI/src/com/android/systemui/statusbar/policy/SecurityController.java
+++ b/packages/SystemUI/src/com/android/systemui/statusbar/policy/SecurityController.java
@@ -23,8 +23,7 @@
String getVpnApp();
boolean isLegacyVpn();
String getLegacyVpnName();
- void openVpnApp();
- void disconnectFromLegacyVpn();
+ void disconnectFromVpn();
void addCallback(VpnCallback callback);
void removeCallback(VpnCallback callback);
diff --git a/packages/SystemUI/src/com/android/systemui/statusbar/policy/SecurityControllerImpl.java b/packages/SystemUI/src/com/android/systemui/statusbar/policy/SecurityControllerImpl.java
index 8e04e5e..ae0291b 100644
--- a/packages/SystemUI/src/com/android/systemui/statusbar/policy/SecurityControllerImpl.java
+++ b/packages/SystemUI/src/com/android/systemui/statusbar/policy/SecurityControllerImpl.java
@@ -18,7 +18,6 @@
import android.app.admin.DevicePolicyManager;
import android.content.Context;
import android.content.Intent;
-import android.content.pm.ApplicationInfo;
import android.content.pm.PackageManager.NameNotFoundException;
import android.net.ConnectivityManager;
import android.net.ConnectivityManager.NetworkCallback;
@@ -109,18 +108,17 @@
}
@Override
- public void openVpnApp() {
- Intent i = mContext.getPackageManager().getLaunchIntentForPackage(mVpnConfig.user);
- if (i != null) {
- i.addFlags(Intent.FLAG_ACTIVITY_NEW_TASK);
- mContext.startActivity(i);
- }
- }
-
- @Override
- public void disconnectFromLegacyVpn() {
+ public void disconnectFromVpn() {
try {
- mConnectivityService.prepareVpn(VpnConfig.LEGACY_VPN, VpnConfig.LEGACY_VPN);
+ if (isLegacyVpn()) {
+ mConnectivityService.prepareVpn(VpnConfig.LEGACY_VPN, VpnConfig.LEGACY_VPN);
+ } else {
+ // Prevent this app from initiating VPN connections in the future without user
+ // intervention.
+ mConnectivityService.setVpnPackageAuthorization(false);
+
+ mConnectivityService.prepareVpn(mVpnConfig.user, VpnConfig.LEGACY_VPN);
+ }
} catch (Exception e) {
Log.e(TAG, "Unable to disconnect from VPN", e);
}
@@ -154,9 +152,7 @@
mIsVpnEnabled = mVpnConfig != null;
if (mVpnConfig != null && !mVpnConfig.legacy) {
- ApplicationInfo info =
- mContext.getPackageManager().getApplicationInfo(mVpnConfig.user, 0);
- mVpnName = mContext.getPackageManager().getApplicationLabel(info).toString();
+ mVpnName = VpnConfig.getVpnLabel(mContext, mVpnConfig.user).toString();
}
} catch (RemoteException | NameNotFoundException e) {
Log.w(TAG, "Unable to get current VPN", e);
diff --git a/packages/VpnDialogs/AndroidManifest.xml b/packages/VpnDialogs/AndroidManifest.xml
index 03d920a..1768400 100644
--- a/packages/VpnDialogs/AndroidManifest.xml
+++ b/packages/VpnDialogs/AndroidManifest.xml
@@ -28,14 +28,5 @@
<category android:name="android.intent.category.DEFAULT"/>
</intent-filter>
</activity>
-
- <activity android:name=".ManageDialog"
- android:theme="@*android:style/Theme.DeviceDefault.Light.Dialog.Alert"
- android:noHistory="true">
- <intent-filter>
- <action android:name="android.intent.action.MAIN"/>
- <category android:name="android.intent.category.DEFAULT"/>
- </intent-filter>
- </activity>
</application>
</manifest>
diff --git a/packages/VpnDialogs/res/drawable-hdpi/ic_vpn_dialog.png b/packages/VpnDialogs/res/drawable-hdpi/ic_vpn_dialog.png
new file mode 100644
index 0000000..a0b4b61
--- /dev/null
+++ b/packages/VpnDialogs/res/drawable-hdpi/ic_vpn_dialog.png
Binary files differ
diff --git a/packages/VpnDialogs/res/drawable-mdpi/ic_vpn_dialog.png b/packages/VpnDialogs/res/drawable-mdpi/ic_vpn_dialog.png
new file mode 100644
index 0000000..df5dfe8
--- /dev/null
+++ b/packages/VpnDialogs/res/drawable-mdpi/ic_vpn_dialog.png
Binary files differ
diff --git a/packages/VpnDialogs/res/drawable-xhdpi/ic_vpn_dialog.png b/packages/VpnDialogs/res/drawable-xhdpi/ic_vpn_dialog.png
new file mode 100644
index 0000000..18d5a3a
--- /dev/null
+++ b/packages/VpnDialogs/res/drawable-xhdpi/ic_vpn_dialog.png
Binary files differ
diff --git a/packages/VpnDialogs/res/drawable-xxhdpi/ic_vpn_dialog.png b/packages/VpnDialogs/res/drawable-xxhdpi/ic_vpn_dialog.png
new file mode 100644
index 0000000..4d475dc
--- /dev/null
+++ b/packages/VpnDialogs/res/drawable-xxhdpi/ic_vpn_dialog.png
Binary files differ
diff --git a/packages/VpnDialogs/res/drawable-xxxhdpi/ic_vpn_dialog.png b/packages/VpnDialogs/res/drawable-xxxhdpi/ic_vpn_dialog.png
new file mode 100644
index 0000000..9d458b4
--- /dev/null
+++ b/packages/VpnDialogs/res/drawable-xxxhdpi/ic_vpn_dialog.png
Binary files differ
diff --git a/packages/VpnDialogs/res/layout/confirm.xml b/packages/VpnDialogs/res/layout/confirm.xml
index ee7f4b8..66fec59 100644
--- a/packages/VpnDialogs/res/layout/confirm.xml
+++ b/packages/VpnDialogs/res/layout/confirm.xml
@@ -18,41 +18,12 @@
<ScrollView xmlns:android="http://schemas.android.com/apk/res/android"
android:layout_width="match_parent"
android:layout_height="wrap_content">
- <LinearLayout android:layout_width="match_parent"
+ <TextView android:id="@+id/warning"
+ android:layout_width="fill_parent"
android:layout_height="wrap_content"
- android:orientation="vertical"
- android:padding="3mm">
-
- <LinearLayout android:layout_width="match_parent"
- android:layout_height="wrap_content"
- android:orientation="horizontal"
- android:gravity="center_vertical">
-
- <ImageView android:id="@+id/icon"
- android:layout_width="@android:dimen/app_icon_size"
- android:layout_height="@android:dimen/app_icon_size"
- android:paddingRight="1mm"/>
-
- <TextView android:id="@+id/prompt"
- android:layout_width="fill_parent"
- android:layout_height="wrap_content"
- android:textSize="18sp"/>
- </LinearLayout>
-
- <TextView android:id="@+id/warning"
- android:layout_width="fill_parent"
- android:layout_height="wrap_content"
- android:paddingTop="1mm"
- android:paddingBottom="1mm"
- android:text="@string/warning"
- android:textSize="18sp"/>
-
- <CheckBox android:id="@+id/check"
- android:layout_width="fill_parent"
- android:layout_height="wrap_content"
- android:text="@string/accept"
- android:textSize="20sp"
- android:filterTouchesWhenObscured="true"
- android:checked="false"/>
- </LinearLayout>
+ android:textSize="18sp"
+ android:paddingTop="4mm"
+ android:paddingLeft="3mm"
+ android:paddingRight="3mm"
+ android:paddingBottom="4mm"/>
</ScrollView>
diff --git a/packages/VpnDialogs/res/layout/manage.xml b/packages/VpnDialogs/res/layout/manage.xml
deleted file mode 100644
index 56332c3..0000000
--- a/packages/VpnDialogs/res/layout/manage.xml
+++ /dev/null
@@ -1,45 +0,0 @@
-<?xml version="1.0" encoding="utf-8"?>
-<!-- Copyright (C) 2011 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.
--->
-
-<TableLayout xmlns:android="http://schemas.android.com/apk/res/android"
- android:layout_width="wrap_content"
- android:layout_height="wrap_content"
- android:padding="3mm"
- android:stretchColumns="0,1"
- android:shrinkColumns="1">
-
- <TableRow>
- <TextView android:text="@string/session" style="@style/label"/>
- <TextView android:id="@+id/session" style="@style/value"/>
- </TableRow>
-
- <TableRow>
- <TextView android:text="@string/duration" style="@style/label"/>
- <TextView android:id="@+id/duration" style="@style/value"/>
- </TableRow>
-
- <TableRow android:id="@+id/data_transmitted_row" android:visibility="gone">
- <TextView android:text="@string/data_transmitted" style="@style/label"/>
- <TextView android:id="@+id/data_transmitted" style="@style/value"/>
- </TableRow>
-
- <TableRow android:id="@+id/data_received_row" android:visibility="gone">
- <TextView android:text="@string/data_received" style="@style/label"/>
- <TextView android:id="@+id/data_received" style="@style/value"/>
- </TableRow>
-
-</TableLayout>
diff --git a/packages/VpnDialogs/res/values/strings.xml b/packages/VpnDialogs/res/values/strings.xml
index 3ff767a..84206a1 100644
--- a/packages/VpnDialogs/res/values/strings.xml
+++ b/packages/VpnDialogs/res/values/strings.xml
@@ -17,40 +17,15 @@
<resources xmlns:xliff="urn:oasis:names:tc:xliff:document:1.2">
<!-- Dialog title to identify the request from a VPN application. [CHAR LIMIT=60] -->
- <string name="prompt"><xliff:g id="app">%s</xliff:g>
- attempts to create a VPN connection.
- </string>
+ <string name="prompt">Connection request</string>
<!-- Dialog message to warn about the risk of using a VPN application. [CHAR LIMIT=NONE] -->
- <string name="warning">By proceeding, you are giving the application
- permission to intercept all network traffic.
- <b>Do NOT accept unless you trust the application.</b> Otherwise,
- you run the risk of having your data compromised by a malicious
- software.
- </string>
-
- <!-- Checkbox label to accept the request from a VPN application. [CHAR LIMIT=60] -->
- <string name="accept">I trust this application.</string>
-
- <!-- Dialog title for built-in VPN. [CHAR LIMIT=40] -->
- <string name="legacy_title">VPN is connected</string>
- <!-- Button label to configure the current VPN session. [CHAR LIMIT=20] -->
- <string name="configure">Configure</string>
- <!-- Button label to disconnect the current VPN session. [CHAR LIMIT=20] -->
- <string name="disconnect">Disconnect</string>
-
- <!-- Label for the name of the current VPN session. [CHAR LIMIT=20] -->
- <string name="session">Session:</string>
- <!-- Label for the duration of the current VPN session. [CHAR LIMIT=20] -->
- <string name="duration">Duration:</string>
- <!-- Label for the network usage of data transmitted over VPN. [CHAR LIMIT=20] -->
- <string name="data_transmitted">Sent:</string>
- <!-- Label for the network usage of data received over VPN. [CHAR LIMIT=20] -->
- <string name="data_received">Received:</string>
-
- <!-- Formatted string for the network usage over VPN. [CHAR LIMIT=40] -->
- <string name="data_value_format">
- <xliff:g id="number">%1$s</xliff:g> bytes /
- <xliff:g id="number">%2$s</xliff:g> packets
+ <string name="warning"><xliff:g id="app">%s</xliff:g> wants to set up a VPN connection
+ that allows it to monitor network traffic. Only accept if you trust the source.
+ <![CDATA[
+ <br />
+ <br />
+ <img src="vpn_icon" />
+ ]]> appears at the top of your screen when VPN is active.
</string>
</resources>
diff --git a/packages/VpnDialogs/src/com/android/vpndialogs/ConfirmDialog.java b/packages/VpnDialogs/src/com/android/vpndialogs/ConfirmDialog.java
index ddafc66..897c96cf 100644
--- a/packages/VpnDialogs/src/com/android/vpndialogs/ConfirmDialog.java
+++ b/packages/VpnDialogs/src/com/android/vpndialogs/ConfirmDialog.java
@@ -18,21 +18,28 @@
import android.content.Context;
import android.content.DialogInterface;
+import android.content.Intent;
import android.content.pm.ApplicationInfo;
import android.content.pm.PackageManager;
+import android.content.pm.ResolveInfo;
+import android.graphics.drawable.Drawable;
import android.net.IConnectivityManager;
+import android.net.VpnService;
import android.os.ServiceManager;
+import android.text.Html;
+import android.text.Html.ImageGetter;
import android.util.Log;
import android.view.View;
import android.widget.Button;
-import android.widget.CompoundButton;
-import android.widget.ImageView;
import android.widget.TextView;
import com.android.internal.app.AlertActivity;
+import com.android.internal.net.VpnConfig;
-public class ConfirmDialog extends AlertActivity implements
- CompoundButton.OnCheckedChangeListener, DialogInterface.OnClickListener {
+import java.util.List;
+
+public class ConfirmDialog extends AlertActivity
+ implements DialogInterface.OnClickListener, ImageGetter {
private static final String TAG = "VpnConfirm";
private String mPackage;
@@ -56,27 +63,22 @@
return;
}
- PackageManager pm = getPackageManager();
- ApplicationInfo app = pm.getApplicationInfo(mPackage, 0);
-
View view = View.inflate(this, R.layout.confirm, null);
- ((ImageView) view.findViewById(R.id.icon)).setImageDrawable(app.loadIcon(pm));
- ((TextView) view.findViewById(R.id.prompt)).setText(
- getString(R.string.prompt, app.loadLabel(pm)));
- ((CompoundButton) view.findViewById(R.id.check)).setOnCheckedChangeListener(this);
- mAlertParams.mIconAttrId = android.R.attr.alertDialogIcon;
- mAlertParams.mTitle = getText(android.R.string.dialog_alert_title);
+ ((TextView) view.findViewById(R.id.warning)).setText(
+ Html.fromHtml(
+ getString(R.string.warning, VpnConfig.getVpnLabel(this, mPackage)),
+ this, null /* tagHandler */));
+
+ mAlertParams.mTitle = getText(R.string.prompt);
mAlertParams.mPositiveButtonText = getText(android.R.string.ok);
mAlertParams.mPositiveButtonListener = this;
mAlertParams.mNegativeButtonText = getText(android.R.string.cancel);
- mAlertParams.mNegativeButtonListener = this;
mAlertParams.mView = view;
setupAlert();
getWindow().setCloseOnTouchOutside(false);
mButton = mAlert.getButton(DialogInterface.BUTTON_POSITIVE);
- mButton.setEnabled(false);
mButton.setFilterTouchesWhenObscured(true);
} catch (Exception e) {
Log.e(TAG, "onResume", e);
@@ -85,18 +87,24 @@
}
@Override
- public void onBackPressed() {
+ public Drawable getDrawable(String source) {
+ // Should only reach this when fetching the VPN icon for the warning string.
+ Drawable icon = getDrawable(R.drawable.ic_vpn_dialog);
+ icon.setBounds(0, 0, icon.getIntrinsicWidth(), icon.getIntrinsicHeight());
+ return icon;
}
@Override
- public void onCheckedChanged(CompoundButton button, boolean checked) {
- mButton.setEnabled(checked);
+ public void onBackPressed() {
}
@Override
public void onClick(DialogInterface dialog, int which) {
try {
- if (which == DialogInterface.BUTTON_POSITIVE && mService.prepareVpn(null, mPackage)) {
+ if (mService.prepareVpn(null, mPackage)) {
+ // Authorize this app to initiate VPN connections in the future without user
+ // intervention.
+ mService.setVpnPackageAuthorization(true);
setResult(RESULT_OK);
}
} catch (Exception e) {
diff --git a/packages/VpnDialogs/src/com/android/vpndialogs/ManageDialog.java b/packages/VpnDialogs/src/com/android/vpndialogs/ManageDialog.java
deleted file mode 100644
index eb20995..0000000
--- a/packages/VpnDialogs/src/com/android/vpndialogs/ManageDialog.java
+++ /dev/null
@@ -1,204 +0,0 @@
-/*
- * Copyright (C) 2011 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 com.android.vpndialogs;
-
-import android.content.Context;
-import android.content.DialogInterface;
-import android.content.pm.ApplicationInfo;
-import android.content.pm.PackageManager;
-import android.net.IConnectivityManager;
-import android.os.Handler;
-import android.os.Message;
-import android.os.ServiceManager;
-import android.os.SystemClock;
-import android.util.Log;
-import android.view.View;
-import android.widget.TextView;
-
-import com.android.internal.app.AlertActivity;
-import com.android.internal.net.VpnConfig;
-
-import java.io.DataInputStream;
-import java.io.FileInputStream;
-
-public class ManageDialog extends AlertActivity implements
- DialogInterface.OnClickListener, Handler.Callback {
- private static final String TAG = "VpnManage";
-
- private VpnConfig mConfig;
-
- private IConnectivityManager mService;
-
- private TextView mDuration;
- private TextView mDataTransmitted;
- private TextView mDataReceived;
- private boolean mDataRowsHidden;
-
- private Handler mHandler;
-
- @Override
- protected void onResume() {
- super.onResume();
-
- if (getCallingPackage() != null) {
- Log.e(TAG, getCallingPackage() + " cannot start this activity");
- finish();
- return;
- }
-
- try {
-
- mService = IConnectivityManager.Stub.asInterface(
- ServiceManager.getService(Context.CONNECTIVITY_SERVICE));
-
- mConfig = mService.getVpnConfig();
-
- // mConfig can be null if we are a restricted user, in that case don't show this dialog
- if (mConfig == null) {
- finish();
- return;
- }
-
- View view = View.inflate(this, R.layout.manage, null);
- if (mConfig.session != null) {
- ((TextView) view.findViewById(R.id.session)).setText(mConfig.session);
- }
- mDuration = (TextView) view.findViewById(R.id.duration);
- mDataTransmitted = (TextView) view.findViewById(R.id.data_transmitted);
- mDataReceived = (TextView) view.findViewById(R.id.data_received);
- mDataRowsHidden = true;
-
- if (mConfig.legacy) {
- mAlertParams.mIconId = android.R.drawable.ic_dialog_info;
- mAlertParams.mTitle = getText(R.string.legacy_title);
- } else {
- PackageManager pm = getPackageManager();
- ApplicationInfo app = pm.getApplicationInfo(mConfig.user, 0);
- mAlertParams.mIcon = app.loadIcon(pm);
- mAlertParams.mTitle = app.loadLabel(pm);
- }
- if (mConfig.configureIntent != null) {
- mAlertParams.mPositiveButtonText = getText(R.string.configure);
- mAlertParams.mPositiveButtonListener = this;
- }
- mAlertParams.mNeutralButtonText = getText(R.string.disconnect);
- mAlertParams.mNeutralButtonListener = this;
- mAlertParams.mNegativeButtonText = getText(android.R.string.cancel);
- mAlertParams.mNegativeButtonListener = this;
- mAlertParams.mView = view;
- setupAlert();
-
- if (mHandler == null) {
- mHandler = new Handler(this);
- }
- mHandler.sendEmptyMessage(0);
- } catch (Exception e) {
- Log.e(TAG, "onResume", e);
- finish();
- }
- }
-
- @Override
- protected void onPause() {
- super.onPause();
- if (!isFinishing()) {
- finish();
- }
- }
-
- @Override
- public void onClick(DialogInterface dialog, int which) {
- try {
- if (which == DialogInterface.BUTTON_POSITIVE) {
- mConfig.configureIntent.send();
- } else if (which == DialogInterface.BUTTON_NEUTRAL) {
- if (mConfig.legacy) {
- mService.prepareVpn(VpnConfig.LEGACY_VPN, VpnConfig.LEGACY_VPN);
- } else {
- mService.prepareVpn(mConfig.user, VpnConfig.LEGACY_VPN);
- }
- }
- } catch (Exception e) {
- Log.e(TAG, "onClick", e);
- finish();
- }
- }
-
- @Override
- public boolean handleMessage(Message message) {
- mHandler.removeMessages(0);
-
- if (!isFinishing()) {
- if (mConfig.startTime != -1) {
- long seconds = (SystemClock.elapsedRealtime() - mConfig.startTime) / 1000;
- mDuration.setText(String.format("%02d:%02d:%02d",
- seconds / 3600, seconds / 60 % 60, seconds % 60));
- }
-
- String[] numbers = getNumbers();
- if (numbers != null) {
- // First unhide the related data rows.
- if (mDataRowsHidden) {
- findViewById(R.id.data_transmitted_row).setVisibility(View.VISIBLE);
- findViewById(R.id.data_received_row).setVisibility(View.VISIBLE);
- mDataRowsHidden = false;
- }
-
- // [1] and [2] are received data in bytes and packets.
- mDataReceived.setText(getString(R.string.data_value_format,
- numbers[1], numbers[2]));
-
- // [9] and [10] are transmitted data in bytes and packets.
- mDataTransmitted.setText(getString(R.string.data_value_format,
- numbers[9], numbers[10]));
- }
- mHandler.sendEmptyMessageDelayed(0, 1000);
- }
- return true;
- }
-
- private String[] getNumbers() {
- DataInputStream in = null;
- try {
- // See dev_seq_printf_stats() in net/core/dev.c.
- in = new DataInputStream(new FileInputStream("/proc/net/dev"));
- String prefix = mConfig.interfaze + ':';
-
- while (true) {
- String line = in.readLine().trim();
- if (line.startsWith(prefix)) {
- String[] numbers = line.substring(prefix.length()).split(" +");
- for (int i = 1; i < 17; ++i) {
- if (!numbers[i].equals("0")) {
- return numbers;
- }
- }
- break;
- }
- }
- } catch (Exception e) {
- // ignore
- } finally {
- try {
- in.close();
- } catch (Exception e) {
- // ignore
- }
- }
- return null;
- }
-}
diff --git a/services/core/java/com/android/server/ConnectivityService.java b/services/core/java/com/android/server/ConnectivityService.java
index 5b216ed..491f5a6 100644
--- a/services/core/java/com/android/server/ConnectivityService.java
+++ b/services/core/java/com/android/server/ConnectivityService.java
@@ -2663,6 +2663,20 @@
}
/**
+ * Set whether the current VPN package has the ability to launch VPNs without
+ * user intervention. This method is used by system UIs and not available
+ * in ConnectivityManager. Permissions are checked in Vpn class.
+ * @hide
+ */
+ @Override
+ public void setVpnPackageAuthorization(boolean authorized) {
+ int user = UserHandle.getUserId(Binder.getCallingUid());
+ synchronized(mVpns) {
+ mVpns.get(user).setPackageAuthorization(authorized);
+ }
+ }
+
+ /**
* Configure a TUN interface and return its file descriptor. Parameters
* are encoded and opaque to this class. This method is used by VpnBuilder
* and not available in ConnectivityManager. Permissions are checked in
diff --git a/services/core/java/com/android/server/connectivity/Vpn.java b/services/core/java/com/android/server/connectivity/Vpn.java
index 0f6b3ad..69caab9 100644
--- a/services/core/java/com/android/server/connectivity/Vpn.java
+++ b/services/core/java/com/android/server/connectivity/Vpn.java
@@ -21,9 +21,7 @@
import static android.system.OsConstants.AF_INET6;
import android.app.AppGlobals;
-import android.app.Notification;
-import android.app.NotificationManager;
-import android.app.PendingIntent;
+import android.app.AppOpsManager;
import android.content.BroadcastReceiver;
import android.content.ComponentName;
import android.content.Context;
@@ -35,10 +33,6 @@
import android.content.pm.PackageManager.NameNotFoundException;
import android.content.pm.ResolveInfo;
import android.content.pm.UserInfo;
-import android.graphics.Bitmap;
-import android.graphics.Canvas;
-import android.graphics.drawable.Drawable;
-import android.net.BaseNetworkStateTracker;
import android.net.ConnectivityManager;
import android.net.IConnectivityManager;
import android.net.INetworkManagementEventObserver;
@@ -51,7 +45,6 @@
import android.net.NetworkInfo;
import android.net.NetworkInfo.DetailedState;
import android.net.NetworkMisc;
-import android.net.NetworkUtils;
import android.net.RouteInfo;
import android.net.UidRange;
import android.os.Binder;
@@ -70,15 +63,15 @@
import android.security.Credentials;
import android.security.KeyStore;
import android.util.Log;
-import android.util.SparseBooleanArray;
import com.android.internal.annotations.GuardedBy;
-import com.android.internal.R;
import com.android.internal.net.LegacyVpnInfo;
import com.android.internal.net.VpnConfig;
import com.android.internal.net.VpnProfile;
import com.android.server.net.BaseNetworkObserver;
+import libcore.io.IoUtils;
+
import java.io.File;
import java.io.IOException;
import java.io.InputStream;
@@ -92,8 +85,6 @@
import java.util.List;
import java.util.concurrent.atomic.AtomicInteger;
-import libcore.io.IoUtils;
-
/**
* @hide
*/
@@ -114,8 +105,6 @@
private boolean mAllowIPv6;
private Connection mConnection;
private LegacyVpnRunner mLegacyVpnRunner;
- private PendingIntent mStatusIntent;
- private volatile boolean mEnableNotif = true;
private volatile boolean mEnableTeardown = true;
private final IConnectivityManager mConnService;
private final INetworkManagementService mNetd;
@@ -180,14 +169,6 @@
}
/**
- * Set if this object is responsible for showing its own notifications. When
- * {@code false}, notifications are handled externally by someone else.
- */
- public void setEnableNotifications(boolean enableNotif) {
- mEnableNotif = enableNotif;
- }
-
- /**
* Set if this object is responsible for watching for {@link NetworkInfo}
* teardown. When {@code false}, teardown is handled externally by someone
* else.
@@ -228,6 +209,20 @@
public synchronized boolean prepare(String oldPackage, String newPackage) {
// Return false if the package does not match.
if (oldPackage != null && !oldPackage.equals(mPackage)) {
+ // The package doesn't match. If this VPN was not previously authorized, return false
+ // to force user authorization. Otherwise, revoke the VPN anyway.
+ if (!oldPackage.equals(VpnConfig.LEGACY_VPN) && isVpnUserPreConsented(oldPackage)) {
+ long token = Binder.clearCallingIdentity();
+ try {
+ // This looks bizarre, but it is what ConfirmDialog in VpnDialogs is doing when
+ // the user clicks through to allow the VPN to consent. So we are emulating the
+ // action of the dialog without actually showing it.
+ prepare(null, oldPackage);
+ } finally {
+ Binder.restoreCallingIdentity(token);
+ }
+ return true;
+ }
return false;
}
@@ -240,11 +235,8 @@
// Check if the caller is authorized.
enforceControlPermission();
- // Reset the interface and hide the notification.
+ // Reset the interface.
if (mInterface != null) {
- for (UidRange uidRange : mVpnUsers) {
- hideNotification(uidRange.getStartUser());
- }
agentDisconnect();
jniReset(mInterface);
mInterface = null;
@@ -287,12 +279,46 @@
Binder.restoreCallingIdentity(token);
}
mConfig = null;
+
updateState(DetailedState.IDLE, "prepare");
return true;
}
+ /**
+ * Set whether the current package has the ability to launch VPNs without user intervention.
+ */
+ public void setPackageAuthorization(boolean authorized) {
+ // Check if the caller is authorized.
+ enforceControlPermission();
+
+ if (mPackage == null || VpnConfig.LEGACY_VPN.equals(mPackage)) {
+ return;
+ }
+
+ long token = Binder.clearCallingIdentity();
+ try {
+ AppOpsManager appOps =
+ (AppOpsManager) mContext.getSystemService(Context.APP_OPS_SERVICE);
+ appOps.setMode(AppOpsManager.OP_ACTIVATE_VPN, mOwnerUID, mPackage,
+ authorized ? AppOpsManager.MODE_ALLOWED : AppOpsManager.MODE_IGNORED);
+ } catch (Exception e) {
+ Log.wtf(TAG, "Failed to set app ops for package " + mPackage, e);
+ } finally {
+ Binder.restoreCallingIdentity(token);
+ }
+ }
+
+ private boolean isVpnUserPreConsented(String packageName) {
+ AppOpsManager appOps =
+ (AppOpsManager) mContext.getSystemService(Context.APP_OPS_SERVICE);
+
+ // Verify that the caller matches the given package and has permission to activate VPNs.
+ return appOps.noteOpNoThrow(AppOpsManager.OP_ACTIVATE_VPN, Binder.getCallingUid(),
+ packageName) == AppOpsManager.MODE_ALLOWED;
+ }
+
private int getAppUid(String app) {
- if (app == VpnConfig.LEGACY_VPN) {
+ if (VpnConfig.LEGACY_VPN.equals(app)) {
return Process.myUid();
}
PackageManager pm = mContext.getPackageManager();
@@ -355,9 +381,10 @@
try {
mNetworkAgent = new NetworkAgent(mLooper, mContext, NETWORKTYPE,
mNetworkInfo, mNetworkCapabilities, lp, 0, networkMisc) {
+ @Override
public void unwanted() {
// We are user controlled, not driven by NetworkRequest.
- };
+ }
};
} finally {
Binder.restoreCallingIdentity(token);
@@ -540,39 +567,6 @@
// add the user
mVpnUsers.add(UidRange.createForUser(user));
-
- // show the notification
- if (!mPackage.equals(VpnConfig.LEGACY_VPN)) {
- // Load everything for the user's notification
- PackageManager pm = mContext.getPackageManager();
- ApplicationInfo app = null;
- final long token = Binder.clearCallingIdentity();
- try {
- app = AppGlobals.getPackageManager().getApplicationInfo(mPackage, 0, mUserId);
- } catch (RemoteException e) {
- throw new IllegalStateException("Invalid application");
- } finally {
- Binder.restoreCallingIdentity(token);
- }
- String label = app.loadLabel(pm).toString();
- // Load the icon and convert it into a bitmap.
- Drawable icon = app.loadIcon(pm);
- Bitmap bitmap = null;
- if (icon.getIntrinsicWidth() > 0 && icon.getIntrinsicHeight() > 0) {
- int width = mContext.getResources().getDimensionPixelSize(
- android.R.dimen.notification_large_icon_width);
- int height = mContext.getResources().getDimensionPixelSize(
- android.R.dimen.notification_large_icon_height);
- icon.setBounds(0, 0, width, height);
- bitmap = Bitmap.createBitmap(width, height, Bitmap.Config.ARGB_8888);
- Canvas c = new Canvas(bitmap);
- icon.draw(c);
- c.setBitmap(null);
- }
- showNotification(label, bitmap, user);
- } else {
- showNotification(null, null, user);
- }
}
private void removeVpnUserLocked(int user) {
@@ -584,7 +578,6 @@
mNetworkAgent.removeUidRanges(new UidRange[] { uidRange });
}
mVpnUsers.remove(uidRange);
- hideNotification(user);
}
private void onUserAdded(int userId) {
@@ -652,9 +645,6 @@
public void interfaceRemoved(String interfaze) {
synchronized (Vpn.this) {
if (interfaze.equals(mInterface) && jniCheck(interfaze) == 0) {
- for (UidRange uidRange : mVpnUsers) {
- hideNotification(uidRange.getStartUser());
- }
mVpnUsers = null;
mInterface = null;
if (mConnection != null) {
@@ -712,56 +702,6 @@
}
}
- private void showNotification(String label, Bitmap icon, int user) {
- if (!mEnableNotif) return;
- final long token = Binder.clearCallingIdentity();
- try {
- mStatusIntent = VpnConfig.getIntentForStatusPanel(mContext);
-
- NotificationManager nm = (NotificationManager)
- mContext.getSystemService(Context.NOTIFICATION_SERVICE);
-
- if (nm != null) {
- String title = (label == null) ? mContext.getString(R.string.vpn_title) :
- mContext.getString(R.string.vpn_title_long, label);
- String text = (mConfig.session == null) ? mContext.getString(R.string.vpn_text) :
- mContext.getString(R.string.vpn_text_long, mConfig.session);
-
- Notification notification = new Notification.Builder(mContext)
- .setSmallIcon(R.drawable.vpn_connected)
- .setLargeIcon(icon)
- .setContentTitle(title)
- .setContentText(text)
- .setContentIntent(mStatusIntent)
- .setDefaults(0)
- .setOngoing(true)
- .setColor(mContext.getResources().getColor(
- com.android.internal.R.color.system_notification_accent_color))
- .build();
- nm.notifyAsUser(null, R.drawable.vpn_connected, notification, new UserHandle(user));
- }
- } finally {
- Binder.restoreCallingIdentity(token);
- }
- }
-
- private void hideNotification(int user) {
- if (!mEnableNotif) return;
- mStatusIntent = null;
-
- NotificationManager nm = (NotificationManager)
- mContext.getSystemService(Context.NOTIFICATION_SERVICE);
-
- if (nm != null) {
- final long token = Binder.clearCallingIdentity();
- try {
- nm.cancelAsUser(null, R.drawable.vpn_connected, new UserHandle(user));
- } finally {
- Binder.restoreCallingIdentity(token);
- }
- }
- }
-
public synchronized boolean addAddress(String address, int prefixLength) {
if (Binder.getCallingUid() != mOwnerUID || mInterface == null || mNetworkAgent == null) {
return false;
@@ -971,9 +911,6 @@
final LegacyVpnInfo info = new LegacyVpnInfo();
info.key = mConfig.user;
info.state = LegacyVpnInfo.stateFromNetworkInfo(mNetworkInfo);
- if (mNetworkInfo.isConnected()) {
- info.intent = mStatusIntent;
- }
return info;
}
@@ -1263,6 +1200,7 @@
}
} catch (Exception e) {
Log.i(TAG, "Aborting", e);
+ updateState(DetailedState.FAILED, e.getMessage());
exit();
} finally {
// Kill the daemons if they fail to stop.
diff --git a/services/core/java/com/android/server/net/LockdownVpnTracker.java b/services/core/java/com/android/server/net/LockdownVpnTracker.java
index 52e741b..04df3e7 100644
--- a/services/core/java/com/android/server/net/LockdownVpnTracker.java
+++ b/services/core/java/com/android/server/net/LockdownVpnTracker.java
@@ -205,7 +205,6 @@
private void initLocked() {
Slog.d(TAG, "initLocked()");
- mVpn.setEnableNotifications(false);
mVpn.setEnableTeardown(false);
final IntentFilter resetFilter = new IntentFilter(ACTION_LOCKDOWN_RESET);
@@ -249,7 +248,6 @@
hideNotification();
mContext.unregisterReceiver(mResetReceiver);
- mVpn.setEnableNotifications(true);
mVpn.setEnableTeardown(true);
}
diff --git a/telecomm/java/android/telecomm/AudioState.java b/telecomm/java/android/telecomm/AudioState.java
index 491af14..a5fda79 100644
--- a/telecomm/java/android/telecomm/AudioState.java
+++ b/telecomm/java/android/telecomm/AudioState.java
@@ -26,25 +26,25 @@
*/
public final class AudioState implements Parcelable {
/** Direct the audio stream through the device's earpiece. */
- public static int ROUTE_EARPIECE = 0x00000001;
+ public static final int ROUTE_EARPIECE = 0x00000001;
/** Direct the audio stream through Bluetooth. */
- public static int ROUTE_BLUETOOTH = 0x00000002;
+ public static final int ROUTE_BLUETOOTH = 0x00000002;
/** Direct the audio stream through a wired headset. */
- public static int ROUTE_WIRED_HEADSET = 0x00000004;
+ public static final int ROUTE_WIRED_HEADSET = 0x00000004;
/** Direct the audio stream through the device's spakerphone. */
- public static int ROUTE_SPEAKER = 0x00000008;
+ public static final int ROUTE_SPEAKER = 0x00000008;
/**
* Direct the audio stream through the device's earpiece or wired headset if one is
* connected.
*/
- public static int ROUTE_WIRED_OR_EARPIECE = ROUTE_EARPIECE | ROUTE_WIRED_HEADSET;
+ public static final int ROUTE_WIRED_OR_EARPIECE = ROUTE_EARPIECE | ROUTE_WIRED_HEADSET;
/** Bit mask of all possible audio routes. */
- public static int ROUTE_ALL = ROUTE_EARPIECE | ROUTE_BLUETOOTH | ROUTE_WIRED_HEADSET |
+ public static final int ROUTE_ALL = ROUTE_EARPIECE | ROUTE_BLUETOOTH | ROUTE_WIRED_HEADSET |
ROUTE_SPEAKER;
/** True if the call is muted, false otherwise. */
diff --git a/telephony/java/android/telephony/TelephonyManager.java b/telephony/java/android/telephony/TelephonyManager.java
index d4f8362..0eb170c 100644
--- a/telephony/java/android/telephony/TelephonyManager.java
+++ b/telephony/java/android/telephony/TelephonyManager.java
@@ -617,6 +617,37 @@
}
/**
+ * Returns the IMEI. Return null if IMEI is not available.
+ *
+ * <p>Requires Permission:
+ * {@link android.Manifest.permission#READ_PHONE_STATE READ_PHONE_STATE}
+ */
+ /** {@hide} */
+ public String getImei() {
+ return getImei(getDefaultSim());
+ }
+
+ /**
+ * Returns the IMEI. Return null if IMEI is not available.
+ *
+ * <p>Requires Permission:
+ * {@link android.Manifest.permission#READ_PHONE_STATE READ_PHONE_STATE}
+ *
+ * @param slotId of which deviceID is returned
+ */
+ /** {@hide} */
+ public String getImei(int slotId) {
+ long[] subId = SubscriptionManager.getSubId(slotId);
+ try {
+ return getSubscriberInfo().getImeiUsingSubId(subId[0]);
+ } catch (RemoteException ex) {
+ return null;
+ } catch (NullPointerException ex) {
+ return null;
+ }
+ }
+
+ /**
* Returns the current location of the device.
*<p>
* If there is only one radio in the device and that radio has an LTE connection,
diff --git a/telephony/java/com/android/internal/telephony/IPhoneSubInfo.aidl b/telephony/java/com/android/internal/telephony/IPhoneSubInfo.aidl
index 4734965..552abaf 100644
--- a/telephony/java/com/android/internal/telephony/IPhoneSubInfo.aidl
+++ b/telephony/java/com/android/internal/telephony/IPhoneSubInfo.aidl
@@ -33,6 +33,10 @@
*/
String getDeviceIdUsingSubId(long subId);
+ /**
+ * Retrieves the IMEI.
+ */
+ String getImeiUsingSubId(long subId);
/**
* Retrieves the software version number for the device, e.g., IMEI/SV