Add support for handling mobile provisioning networks.

When a sim is new or it has expired it needs to be provisioned
with the carrier. Basically provisioning is associating a sim with
a user account. When a sim isn't provisioned then operators will
restrict access to the network and only allow certain addresses
or services to be used.

This set of changes allows two types of provisioning networks to be
recognized. The first is a network that causes all DNS lookups to be
redirected to a different address than was intended. This is exemplified
by how T-Mobile works.

The second technique uses a special apn for provisioning. An example is
AT&T where lwaactivate is the provisioning apn and broadband is the
normal apn. We first try broadband and if we are unable to connect we
try lwaactivate. When we see the activate we identify it as special and
the ApnContext.isProvisioningApn will return true.

In the future our plan is to create a new network type that can be added
to the apn list, but for now it identified by name.

Here is a list of significant changes:

 - CaptivePortalTracker now only test WiFi networks instead of all networks
 - checkMobileProvisioning checks for provisioning networks and doesn't
   try to ping.
 - IConnectivityManager.aidl changes:
   * getProvisioningOrActiveNetworkInfo was added to and used by Manage
     mobile plan in WirelessSettings so even when there is no active
     network it will still allow provisioning. Otherwise it would report
     no internet connection.
   * setSignInErrorNotificationVisible is used by both
     CaptiviePortalTracker and checkMobileProvisioning so they use the
     same code for the notifications.
   * checkMobileProvisioning was simplified to have only a timeout as
     returning the result is now harder as we abort simultaneous call
     otherwise we'd could get into loops because we now check every time
     we connect to mobile.
 - Enhanced MDST to handle the provisioning network.
 - Added CONNECTED_TO_PROVISIONING_NETWORK to NetworkInfo to make a new
   state so we don't announce to the world we're connected.
 - TelephonyIntents.ACTION_DATA_CONNECTION_CONNECTED_TO_PROVISIONING_APN
   is sent by the low level data connection code to notify Connectivity
   Service that a provisioning apn has connected. This allows CS to
   handle the connection differently than a normal connection.

Bug: 10328264
Change-Id: I3925004011bb1243793c4c1b963d923dc2b00cb5
This commit is contained in:
Wink Saville
2013-08-29 08:55:16 -07:00
parent d23fa090e9
commit 9a1a7ef57c
4 changed files with 410 additions and 143 deletions

View File

@@ -582,6 +582,29 @@ public class ConnectivityManager {
} }
} }
/**
* Returns details about the Provisioning or currently active default data network. When
* connected, this network is the default route for outgoing connections.
* You should always check {@link NetworkInfo#isConnected()} before initiating
* network traffic. This may return {@code null} when there is no default
* network.
*
* @return a {@link NetworkInfo} object for the current default network
* or {@code null} if no network default network is currently active
*
* <p>This method requires the call to hold the permission
* {@link android.Manifest.permission#ACCESS_NETWORK_STATE}.
*
* {@hide}
*/
public NetworkInfo getProvisioningOrActiveNetworkInfo() {
try {
return mService.getProvisioningOrActiveNetworkInfo();
} catch (RemoteException e) {
return null;
}
}
/** /**
* Returns the IP information for the current default network. * Returns the IP information for the current default network.
* *
@@ -1316,63 +1339,19 @@ public class ConnectivityManager {
} }
/** /**
* The ResultReceiver resultCode for checkMobileProvisioning (CMP_RESULT_CODE) * Check mobile provisioning.
*/
/**
* No connection was possible to the network.
* {@hide}
*/
public static final int CMP_RESULT_CODE_NO_CONNECTION = 0;
/**
* A connection was made to the internet, all is well.
* {@hide}
*/
public static final int CMP_RESULT_CODE_CONNECTABLE = 1;
/**
* A connection was made but there was a redirection, we appear to be in walled garden.
* This is an indication of a warm sim on a mobile network.
* {@hide}
*/
public static final int CMP_RESULT_CODE_REDIRECTED = 2;
/**
* A connection was made but no dns server was available to resolve a name to address.
* This is an indication of a warm sim on a mobile network.
* *
* {@hide}
*/
public static final int CMP_RESULT_CODE_NO_DNS = 3;
/**
* A connection was made but could not open a TCP connection.
* This is an indication of a warm sim on a mobile network.
* {@hide}
*/
public static final int CMP_RESULT_CODE_NO_TCP_CONNECTION = 4;
/**
* Check mobile provisioning. The resultCode passed to
* onReceiveResult will be one of the CMP_RESULT_CODE_xxxx values above.
* This may take a minute or more to complete.
*
* @param sendNotificaiton, when true a notification will be sent to user.
* @param suggestedTimeOutMs, timeout in milliseconds * @param suggestedTimeOutMs, timeout in milliseconds
* @param resultReceiver needs to be supplied to receive the result
* *
* @return time out that will be used, maybe less that suggestedTimeOutMs * @return time out that will be used, maybe less that suggestedTimeOutMs
* -1 if an error. * -1 if an error.
* *
* {@hide} * {@hide}
*/ */
public int checkMobileProvisioning(boolean sendNotification, int suggestedTimeOutMs, public int checkMobileProvisioning(int suggestedTimeOutMs) {
ResultReceiver resultReceiver) {
int timeOutMs = -1; int timeOutMs = -1;
try { try {
timeOutMs = mService.checkMobileProvisioning(sendNotification, suggestedTimeOutMs, timeOutMs = mService.checkMobileProvisioning(suggestedTimeOutMs);
resultReceiver);
} catch (RemoteException e) { } catch (RemoteException e) {
} }
return timeOutMs; return timeOutMs;
@@ -1401,4 +1380,20 @@ public class ConnectivityManager {
} }
return null; return null;
} }
/**
* Set sign in error notification to visible or in visible
*
* @param visible
* @param networkType
*
* {@hide}
*/
public void setProvisioningNotificationVisible(boolean visible, int networkType,
String extraInfo, String url) {
try {
mService.setProvisioningNotificationVisible(visible, networkType, extraInfo, url);
} catch (RemoteException e) {
}
}
} }

View File

@@ -46,6 +46,8 @@ interface IConnectivityManager
NetworkInfo getNetworkInfo(int networkType); NetworkInfo getNetworkInfo(int networkType);
NetworkInfo[] getAllNetworkInfo(); NetworkInfo[] getAllNetworkInfo();
NetworkInfo getProvisioningOrActiveNetworkInfo();
boolean isNetworkSupported(int networkType); boolean isNetworkSupported(int networkType);
LinkProperties getActiveLinkProperties(); LinkProperties getActiveLinkProperties();
@@ -135,9 +137,11 @@ interface IConnectivityManager
int findConnectionTypeForIface(in String iface); int findConnectionTypeForIface(in String iface);
int checkMobileProvisioning(boolean sendNotification, int suggestedTimeOutMs, in ResultReceiver resultReceiver); int checkMobileProvisioning(int suggestedTimeOutMs);
String getMobileProvisioningUrl(); String getMobileProvisioningUrl();
String getMobileRedirectedProvisioningUrl(); String getMobileRedirectedProvisioningUrl();
void setProvisioningNotificationVisible(boolean visible, int networkType, in String extraInfo, in String url);
} }

View File

@@ -84,6 +84,12 @@ public class NetworkInfo implements Parcelable {
VERIFYING_POOR_LINK, VERIFYING_POOR_LINK,
/** Checking if network is a captive portal */ /** Checking if network is a captive portal */
CAPTIVE_PORTAL_CHECK, CAPTIVE_PORTAL_CHECK,
/**
* Network is connected to provisioning network
* TODO: Probably not needed when we add TYPE_PROVISIONING_NETWORK
* @hide
*/
CONNECTED_TO_PROVISIONING_NETWORK
} }
/** /**
@@ -108,6 +114,7 @@ public class NetworkInfo implements Parcelable {
stateMap.put(DetailedState.DISCONNECTED, State.DISCONNECTED); stateMap.put(DetailedState.DISCONNECTED, State.DISCONNECTED);
stateMap.put(DetailedState.FAILED, State.DISCONNECTED); stateMap.put(DetailedState.FAILED, State.DISCONNECTED);
stateMap.put(DetailedState.BLOCKED, State.DISCONNECTED); stateMap.put(DetailedState.BLOCKED, State.DISCONNECTED);
stateMap.put(DetailedState.CONNECTED_TO_PROVISIONING_NETWORK, State.CONNECTED);
} }
private int mNetworkType; private int mNetworkType;

View File

@@ -35,6 +35,7 @@ import android.app.Notification;
import android.app.NotificationManager; import android.app.NotificationManager;
import android.app.PendingIntent; import android.app.PendingIntent;
import android.bluetooth.BluetoothTetheringDataTracker; import android.bluetooth.BluetoothTetheringDataTracker;
import android.content.ActivityNotFoundException;
import android.content.BroadcastReceiver; import android.content.BroadcastReceiver;
import android.content.ContentResolver; import android.content.ContentResolver;
import android.content.Context; import android.content.Context;
@@ -143,6 +144,7 @@ import java.util.GregorianCalendar;
import java.util.HashSet; import java.util.HashSet;
import java.util.List; import java.util.List;
import java.util.Random; import java.util.Random;
import java.util.concurrent.atomic.AtomicBoolean;
import java.util.concurrent.atomic.AtomicInteger; import java.util.concurrent.atomic.AtomicInteger;
/** /**
@@ -383,9 +385,6 @@ public class ConnectivityService extends IConnectivityManager.Stub {
TelephonyManager mTelephonyManager; TelephonyManager mTelephonyManager;
// We only want one checkMobileProvisioning after booting.
volatile boolean mFirstProvisioningCheckStarted = false;
public ConnectivityService(Context context, INetworkManagementService netd, public ConnectivityService(Context context, INetworkManagementService netd,
INetworkStatsService statsService, INetworkPolicyManager policyManager) { INetworkStatsService statsService, INetworkPolicyManager policyManager) {
// Currently, omitting a NetworkFactory will create one internally // Currently, omitting a NetworkFactory will create one internally
@@ -609,8 +608,9 @@ public class ConnectivityService extends IConnectivityManager.Stub {
mSettingsObserver = new SettingsObserver(mHandler, EVENT_APPLY_GLOBAL_HTTP_PROXY); mSettingsObserver = new SettingsObserver(mHandler, EVENT_APPLY_GLOBAL_HTTP_PROXY);
mSettingsObserver.observe(mContext); mSettingsObserver.observe(mContext);
mCaptivePortalTracker = CaptivePortalTracker.makeCaptivePortalTracker(mContext, this); IntentFilter filter = new IntentFilter();
loadGlobalProxy(); filter.addAction(CONNECTED_TO_PROVISIONING_NETWORK_ACTION);
mContext.registerReceiver(mProvisioningReceiver, filter);
} }
/** /**
@@ -879,6 +879,46 @@ public class ConnectivityService extends IConnectivityManager.Stub {
return getNetworkInfo(mActiveDefaultNetwork, uid); return getNetworkInfo(mActiveDefaultNetwork, uid);
} }
/**
* Find the first Provisioning network.
*
* @return NetworkInfo or null if none.
*/
private NetworkInfo getProvisioningNetworkInfo() {
enforceAccessPermission();
// Find the first Provisioning Network
NetworkInfo provNi = null;
for (NetworkInfo ni : getAllNetworkInfo()) {
if (ni.getDetailedState()
== NetworkInfo.DetailedState.CONNECTED_TO_PROVISIONING_NETWORK) {
provNi = ni;
break;
}
}
if (DBG) log("getProvisioningNetworkInfo: X provNi=" + provNi);
return provNi;
}
/**
* Find the first Provisioning network or the ActiveDefaultNetwork
* if there is no Provisioning network
*
* @return NetworkInfo or null if none.
*/
@Override
public NetworkInfo getProvisioningOrActiveNetworkInfo() {
enforceAccessPermission();
NetworkInfo provNi = getProvisioningNetworkInfo();
if (provNi == null) {
final int uid = Binder.getCallingUid();
provNi = getNetworkInfo(mActiveDefaultNetwork, uid);
}
if (DBG) log("getProvisioningOrActiveNetworkInfo: X provNi=" + provNi);
return provNi;
}
public NetworkInfo getActiveNetworkInfoUnfiltered() { public NetworkInfo getActiveNetworkInfoUnfiltered() {
enforceAccessPermission(); enforceAccessPermission();
if (isNetworkTypeValid(mActiveDefaultNetwork)) { if (isNetworkTypeValid(mActiveDefaultNetwork)) {
@@ -1241,8 +1281,10 @@ public class ConnectivityService extends IConnectivityManager.Stub {
feature); feature);
} }
if (network.reconnect()) { if (network.reconnect()) {
if (DBG) log("startUsingNetworkFeature X: return APN_REQUEST_STARTED");
return PhoneConstants.APN_REQUEST_STARTED; return PhoneConstants.APN_REQUEST_STARTED;
} else { } else {
if (DBG) log("startUsingNetworkFeature X: return APN_REQUEST_FAILED");
return PhoneConstants.APN_REQUEST_FAILED; return PhoneConstants.APN_REQUEST_FAILED;
} }
} else { } else {
@@ -1254,9 +1296,11 @@ public class ConnectivityService extends IConnectivityManager.Stub {
mNetRequestersPids[usedNetworkType].add(currentPid); mNetRequestersPids[usedNetworkType].add(currentPid);
} }
} }
if (DBG) log("startUsingNetworkFeature X: return -1 unsupported feature.");
return -1; return -1;
} }
} }
if (DBG) log("startUsingNetworkFeature X: return APN_TYPE_NOT_AVAILABLE");
return PhoneConstants.APN_TYPE_NOT_AVAILABLE; return PhoneConstants.APN_TYPE_NOT_AVAILABLE;
} finally { } finally {
if (DBG) { if (DBG) {
@@ -1290,11 +1334,12 @@ public class ConnectivityService extends IConnectivityManager.Stub {
} }
} }
if (found && u != null) { if (found && u != null) {
if (VDBG) log("stopUsingNetworkFeature: X");
// stop regardless of how many other time this proc had called start // stop regardless of how many other time this proc had called start
return stopUsingNetworkFeature(u, true); return stopUsingNetworkFeature(u, true);
} else { } else {
// none found! // none found!
if (VDBG) log("stopUsingNetworkFeature - not a live request, ignoring"); if (VDBG) log("stopUsingNetworkFeature: X not a live request, ignoring");
return 1; return 1;
} }
} }
@@ -1849,6 +1894,9 @@ public class ConnectivityService extends IConnectivityManager.Stub {
*/ */
if (mNetConfigs[prevNetType].isDefault()) { if (mNetConfigs[prevNetType].isDefault()) {
if (mActiveDefaultNetwork == prevNetType) { if (mActiveDefaultNetwork == prevNetType) {
if (DBG) {
log("tryFailover: set mActiveDefaultNetwork=-1, prevNetType=" + prevNetType);
}
mActiveDefaultNetwork = -1; mActiveDefaultNetwork = -1;
} }
@@ -2041,6 +2089,9 @@ public class ConnectivityService extends IConnectivityManager.Stub {
} }
void systemReady() { void systemReady() {
mCaptivePortalTracker = CaptivePortalTracker.makeCaptivePortalTracker(mContext, this);
loadGlobalProxy();
synchronized(this) { synchronized(this) {
mSystemReady = true; mSystemReady = true;
if (mInitialBroadcast != null) { if (mInitialBroadcast != null) {
@@ -2071,10 +2122,11 @@ public class ConnectivityService extends IConnectivityManager.Stub {
}; };
private boolean isNewNetTypePreferredOverCurrentNetType(int type) { private boolean isNewNetTypePreferredOverCurrentNetType(int type) {
if ((type != mNetworkPreference && if (((type != mNetworkPreference)
mNetConfigs[mActiveDefaultNetwork].priority > && (mNetConfigs[mActiveDefaultNetwork].priority > mNetConfigs[type].priority))
mNetConfigs[type].priority) || || (mNetworkPreference == mActiveDefaultNetwork)) {
mNetworkPreference == mActiveDefaultNetwork) return false; return false;
}
return true; return true;
} }
@@ -2088,6 +2140,11 @@ public class ConnectivityService extends IConnectivityManager.Stub {
final NetworkStateTracker thisNet = mNetTrackers[newNetType]; final NetworkStateTracker thisNet = mNetTrackers[newNetType];
final String thisIface = thisNet.getLinkProperties().getInterfaceName(); final String thisIface = thisNet.getLinkProperties().getInterfaceName();
if (VDBG) {
log("handleConnect: E newNetType=" + newNetType + " thisIface=" + thisIface
+ " isFailover" + isFailover);
}
// if this is a default net and other default is running // if this is a default net and other default is running
// kill the one not preferred // kill the one not preferred
if (mNetConfigs[newNetType].isDefault()) { if (mNetConfigs[newNetType].isDefault()) {
@@ -2249,6 +2306,10 @@ public class ConnectivityService extends IConnectivityManager.Stub {
*/ */
private void handleConnectivityChange(int netType, boolean doReset) { private void handleConnectivityChange(int netType, boolean doReset) {
int resetMask = doReset ? NetworkUtils.RESET_ALL_ADDRESSES : 0; int resetMask = doReset ? NetworkUtils.RESET_ALL_ADDRESSES : 0;
if (VDBG) {
log("handleConnectivityChange: netType=" + netType + " doReset=" + doReset
+ " resetMask=" + resetMask);
}
/* /*
* If a non-default network is enabled, add the host routes that * If a non-default network is enabled, add the host routes that
@@ -2316,7 +2377,9 @@ public class ConnectivityService extends IConnectivityManager.Stub {
boolean resetDns = updateRoutes(newLp, curLp, mNetConfigs[netType].isDefault()); boolean resetDns = updateRoutes(newLp, curLp, mNetConfigs[netType].isDefault());
if (resetMask != 0 || resetDns) { if (resetMask != 0 || resetDns) {
if (VDBG) log("handleConnectivityChange: resetting");
if (curLp != null) { if (curLp != null) {
if (VDBG) log("handleConnectivityChange: resetting curLp=" + curLp);
for (String iface : curLp.getAllInterfaceNames()) { for (String iface : curLp.getAllInterfaceNames()) {
if (TextUtils.isEmpty(iface) == false) { if (TextUtils.isEmpty(iface) == false) {
if (resetMask != 0) { if (resetMask != 0) {
@@ -2349,6 +2412,7 @@ public class ConnectivityService extends IConnectivityManager.Stub {
// Update 464xlat state. // Update 464xlat state.
NetworkStateTracker tracker = mNetTrackers[netType]; NetworkStateTracker tracker = mNetTrackers[netType];
if (mClat.requiresClat(netType, tracker)) { if (mClat.requiresClat(netType, tracker)) {
// If the connection was previously using clat, but is not using it now, stop the clat // If the connection was previously using clat, but is not using it now, stop the clat
// daemon. Normally, this happens automatically when the connection disconnects, but if // daemon. Normally, this happens automatically when the connection disconnects, but if
// the disconnect is not reported, or if the connection's LinkProperties changed for // the disconnect is not reported, or if the connection's LinkProperties changed for
@@ -2402,6 +2466,7 @@ public class ConnectivityService extends IConnectivityManager.Stub {
for (RouteInfo r : routeDiff.removed) { for (RouteInfo r : routeDiff.removed) {
if (isLinkDefault || ! r.isDefaultRoute()) { if (isLinkDefault || ! r.isDefaultRoute()) {
if (VDBG) log("updateRoutes: default remove route r=" + r);
removeRoute(curLp, r, TO_DEFAULT_TABLE); removeRoute(curLp, r, TO_DEFAULT_TABLE);
} }
if (isLinkDefault == false) { if (isLinkDefault == false) {
@@ -2446,7 +2511,6 @@ public class ConnectivityService extends IConnectivityManager.Stub {
// remove the default route unless somebody else has asked for it // remove the default route unless somebody else has asked for it
String ifaceName = newLp.getInterfaceName(); String ifaceName = newLp.getInterfaceName();
if (TextUtils.isEmpty(ifaceName) == false && mAddedRoutes.contains(r) == false) { if (TextUtils.isEmpty(ifaceName) == false && mAddedRoutes.contains(r) == false) {
if (VDBG) log("Removing " + r + " for interface " + ifaceName);
try { try {
mNetd.removeRoute(ifaceName, r); mNetd.removeRoute(ifaceName, r);
} catch (Exception e) { } catch (Exception e) {
@@ -2736,9 +2800,8 @@ public class ConnectivityService extends IConnectivityManager.Stub {
public void handleMessage(Message msg) { public void handleMessage(Message msg) {
NetworkInfo info; NetworkInfo info;
switch (msg.what) { switch (msg.what) {
case NetworkStateTracker.EVENT_STATE_CHANGED: case NetworkStateTracker.EVENT_STATE_CHANGED: {
info = (NetworkInfo) msg.obj; info = (NetworkInfo) msg.obj;
int type = info.getType();
NetworkInfo.State state = info.getState(); NetworkInfo.State state = info.getState();
if (VDBG || (state == NetworkInfo.State.CONNECTED) || if (VDBG || (state == NetworkInfo.State.CONNECTED) ||
@@ -2748,15 +2811,17 @@ public class ConnectivityService extends IConnectivityManager.Stub {
state + "/" + info.getDetailedState()); state + "/" + info.getDetailedState());
} }
// After booting we'll check once for mobile provisioning // Since mobile has the notion of a network/apn that can be used for
// if we've provisioned by and connected. // provisioning we need to check every time we're connected as
if (!mFirstProvisioningCheckStarted // CaptiveProtalTracker won't detected it because DCT doesn't report it
// as connected as ACTION_ANY_DATA_CONNECTION_STATE_CHANGED instead its
// reported as ACTION_DATA_CONNECTION_CONNECTED_TO_PROVISIONING_APN. Which
// is received by MDST and sent here as EVENT_STATE_CHANGED.
if (ConnectivityManager.isNetworkTypeMobile(info.getType())
&& (0 != Settings.Global.getInt(mContext.getContentResolver(), && (0 != Settings.Global.getInt(mContext.getContentResolver(),
Settings.Global.DEVICE_PROVISIONED, 0)) Settings.Global.DEVICE_PROVISIONED, 0))
&& (state == NetworkInfo.State.CONNECTED)) { && (state == NetworkInfo.State.CONNECTED)) {
log("check provisioning after booting"); checkMobileProvisioning(CheckMp.MAX_TIMEOUT_MS);
mFirstProvisioningCheckStarted = true;
checkMobileProvisioning(true, CheckMp.MAX_TIMEOUT_MS, null);
} }
EventLogTags.writeConnectivityStateChanged( EventLogTags.writeConnectivityStateChanged(
@@ -2768,6 +2833,30 @@ public class ConnectivityService extends IConnectivityManager.Stub {
} else if (info.getDetailedState() == } else if (info.getDetailedState() ==
DetailedState.CAPTIVE_PORTAL_CHECK) { DetailedState.CAPTIVE_PORTAL_CHECK) {
handleCaptivePortalTrackerCheck(info); handleCaptivePortalTrackerCheck(info);
} else if (info.getDetailedState() ==
DetailedState.CONNECTED_TO_PROVISIONING_NETWORK) {
/**
* TODO: Create ConnectivityManager.TYPE_MOBILE_PROVISIONING
* for now its an in between network, its a network that
* is actually a default network but we don't want it to be
* announced as such to keep background applications from
* trying to use it. It turns out that some still try so we
* take the additional step of clearing any default routes
* to the link that may have incorrectly setup by the lower
* levels.
*/
LinkProperties lp = getLinkProperties(info.getType());
if (DBG) {
log("EVENT_STATE_CHANGED: connected to provisioning network, lp=" + lp);
}
// Clear any default routes setup by the radio so
// any activity by applications trying to use this
// connection will fail until the provisioning network
// is enabled.
for (RouteInfo r : lp.getRoutes()) {
removeRoute(lp, r, TO_DEFAULT_TABLE);
}
} else if (state == NetworkInfo.State.DISCONNECTED) { } else if (state == NetworkInfo.State.DISCONNECTED) {
handleDisconnect(info); handleDisconnect(info);
} else if (state == NetworkInfo.State.SUSPENDED) { } else if (state == NetworkInfo.State.SUSPENDED) {
@@ -2786,18 +2875,21 @@ public class ConnectivityService extends IConnectivityManager.Stub {
mLockdownTracker.onNetworkInfoChanged(info); mLockdownTracker.onNetworkInfoChanged(info);
} }
break; break;
case NetworkStateTracker.EVENT_CONFIGURATION_CHANGED: }
case NetworkStateTracker.EVENT_CONFIGURATION_CHANGED: {
info = (NetworkInfo) msg.obj; info = (NetworkInfo) msg.obj;
// TODO: Temporary allowing network configuration // TODO: Temporary allowing network configuration
// change not resetting sockets. // change not resetting sockets.
// @see bug/4455071 // @see bug/4455071
handleConnectivityChange(info.getType(), false); handleConnectivityChange(info.getType(), false);
break; break;
case NetworkStateTracker.EVENT_NETWORK_SUBTYPE_CHANGED: }
case NetworkStateTracker.EVENT_NETWORK_SUBTYPE_CHANGED: {
info = (NetworkInfo) msg.obj; info = (NetworkInfo) msg.obj;
type = info.getType(); int type = info.getType();
updateNetworkSettings(mNetTrackers[type]); updateNetworkSettings(mNetTrackers[type]);
break; break;
}
} }
} }
} }
@@ -3574,76 +3666,151 @@ public class ConnectivityService extends IConnectivityManager.Stub {
private boolean isMobileDataStateTrackerReady() { private boolean isMobileDataStateTrackerReady() {
MobileDataStateTracker mdst = MobileDataStateTracker mdst =
(MobileDataStateTracker) mNetTrackers[ConnectivityManager.TYPE_MOBILE]; (MobileDataStateTracker) mNetTrackers[ConnectivityManager.TYPE_MOBILE_HIPRI];
return (mdst != null) && (mdst.isReady()); return (mdst != null) && (mdst.isReady());
} }
/**
* The ResultReceiver resultCode for checkMobileProvisioning (CMP_RESULT_CODE)
*/
/**
* No connection was possible to the network.
*/
public static final int CMP_RESULT_CODE_NO_CONNECTION = 0;
/**
* A connection was made to the internet, all is well.
*/
public static final int CMP_RESULT_CODE_CONNECTABLE = 1;
/**
* A connection was made but there was a redirection, we appear to be in walled garden.
* This is an indication of a warm sim on a mobile network.
*/
public static final int CMP_RESULT_CODE_REDIRECTED = 2;
/**
* A connection was made but no dns server was available to resolve a name to address.
* This is an indication of a warm sim on a mobile network.
*/
public static final int CMP_RESULT_CODE_NO_DNS = 3;
/**
* A connection was made but could not open a TCP connection.
* This is an indication of a warm sim on a mobile network.
*/
public static final int CMP_RESULT_CODE_NO_TCP_CONNECTION = 4;
/**
* The mobile network is a provisioning network.
* This is an indication of a warm sim on a mobile network.
*/
public static final int CMP_RESULT_CODE_PROVISIONING_NETWORK = 5;
AtomicBoolean mIsCheckingMobileProvisioning = new AtomicBoolean(false);
@Override @Override
public int checkMobileProvisioning(boolean sendNotification, int suggestedTimeOutMs, public int checkMobileProvisioning(int suggestedTimeOutMs) {
final ResultReceiver resultReceiver) { int timeOutMs = -1;
log("checkMobileProvisioning: E sendNotification=" + sendNotification if (DBG) log("checkMobileProvisioning: E suggestedTimeOutMs=" + suggestedTimeOutMs);
+ " suggestedTimeOutMs=" + suggestedTimeOutMs enforceConnectivityInternalPermission();
+ " resultReceiver=" + resultReceiver);
enforceChangePermission();
mFirstProvisioningCheckStarted = true;
int timeOutMs = suggestedTimeOutMs;
if (suggestedTimeOutMs > CheckMp.MAX_TIMEOUT_MS) {
timeOutMs = CheckMp.MAX_TIMEOUT_MS;
}
// Check that mobile networks are supported
if (!isNetworkSupported(ConnectivityManager.TYPE_MOBILE)
|| !isNetworkSupported(ConnectivityManager.TYPE_MOBILE_HIPRI)) {
log("checkMobileProvisioning: X no mobile network");
if (resultReceiver != null) {
resultReceiver.send(ConnectivityManager.CMP_RESULT_CODE_NO_CONNECTION, null);
}
return timeOutMs;
}
final long token = Binder.clearCallingIdentity(); final long token = Binder.clearCallingIdentity();
try { try {
timeOutMs = suggestedTimeOutMs;
if (suggestedTimeOutMs > CheckMp.MAX_TIMEOUT_MS) {
timeOutMs = CheckMp.MAX_TIMEOUT_MS;
}
// Check that mobile networks are supported
if (!isNetworkSupported(ConnectivityManager.TYPE_MOBILE)
|| !isNetworkSupported(ConnectivityManager.TYPE_MOBILE_HIPRI)) {
if (DBG) log("checkMobileProvisioning: X no mobile network");
return timeOutMs;
}
// If we're already checking don't do it again
// TODO: Add a queue of results...
if (mIsCheckingMobileProvisioning.getAndSet(true)) {
if (DBG) log("checkMobileProvisioning: X already checking ignore for the moment");
return timeOutMs;
}
// Start off with notification off
setProvNotificationVisible(false, ConnectivityManager.TYPE_NONE, null, null);
// See if we've alreadying determined if we've got a provsioning connection
// if so we don't need to do anything active
MobileDataStateTracker mdstDefault = (MobileDataStateTracker)
mNetTrackers[ConnectivityManager.TYPE_MOBILE];
boolean isDefaultProvisioning = mdstDefault.isProvisioningNetwork();
MobileDataStateTracker mdstHipri = (MobileDataStateTracker)
mNetTrackers[ConnectivityManager.TYPE_MOBILE_HIPRI];
boolean isHipriProvisioning = mdstHipri.isProvisioningNetwork();
if (isDefaultProvisioning || isHipriProvisioning) {
if (mIsNotificationVisible) {
if (DBG) {
log("checkMobileProvisioning: provisioning-ignore notification is visible");
}
} else {
NetworkInfo ni = null;
if (isDefaultProvisioning) {
ni = mdstDefault.getNetworkInfo();
}
if (isHipriProvisioning) {
ni = mdstHipri.getNetworkInfo();
}
String url = getMobileProvisioningUrl();
if ((ni != null) && (!TextUtils.isEmpty(url))) {
setProvNotificationVisible(true, ni.getType(), ni.getExtraInfo(), url);
} else {
if (DBG) log("checkMobileProvisioning: provisioning but no url, ignore");
}
}
mIsCheckingMobileProvisioning.set(false);
return timeOutMs;
}
CheckMp checkMp = new CheckMp(mContext, this); CheckMp checkMp = new CheckMp(mContext, this);
CheckMp.CallBack cb = new CheckMp.CallBack() { CheckMp.CallBack cb = new CheckMp.CallBack() {
@Override @Override
void onComplete(Integer result) { void onComplete(Integer result) {
log("CheckMp.onComplete: result=" + result); if (DBG) log("CheckMp.onComplete: result=" + result);
if (resultReceiver != null) {
log("CheckMp.onComplete: send result");
resultReceiver.send(result, null);
}
NetworkInfo ni = NetworkInfo ni =
mNetTrackers[ConnectivityManager.TYPE_MOBILE_HIPRI].getNetworkInfo(); mNetTrackers[ConnectivityManager.TYPE_MOBILE_HIPRI].getNetworkInfo();
switch(result) { switch(result) {
case ConnectivityManager.CMP_RESULT_CODE_CONNECTABLE: case CMP_RESULT_CODE_CONNECTABLE:
case ConnectivityManager.CMP_RESULT_CODE_NO_CONNECTION: { case CMP_RESULT_CODE_NO_CONNECTION: {
log("CheckMp.onComplete: ignore, connected or no connection"); if (DBG) log("CheckMp.onComplete: ignore, connected or no connection");
break; break;
} }
case ConnectivityManager.CMP_RESULT_CODE_REDIRECTED: { case CMP_RESULT_CODE_REDIRECTED: {
log("CheckMp.onComplete: warm sim"); if (DBG) log("CheckMp.onComplete: warm sim");
String url = getMobileProvisioningUrl(); String url = getMobileProvisioningUrl();
if (TextUtils.isEmpty(url)) { if (TextUtils.isEmpty(url)) {
url = getMobileRedirectedProvisioningUrl(); url = getMobileRedirectedProvisioningUrl();
} }
if (TextUtils.isEmpty(url) == false) { if (TextUtils.isEmpty(url) == false) {
log("CheckMp.onComplete: warm sim (redirected), url=" + url); if (DBG) log("CheckMp.onComplete: warm (redirected), url=" + url);
setNotificationVisible(true, ni, url); setProvNotificationVisible(true, ni.getType(), ni.getExtraInfo(),
url);
} else { } else {
log("CheckMp.onComplete: warm sim (redirected), no url"); if (DBG) log("CheckMp.onComplete: warm (redirected), no url");
} }
break; break;
} }
case ConnectivityManager.CMP_RESULT_CODE_NO_DNS: case CMP_RESULT_CODE_NO_DNS:
case ConnectivityManager.CMP_RESULT_CODE_NO_TCP_CONNECTION: { case CMP_RESULT_CODE_NO_TCP_CONNECTION: {
String url = getMobileProvisioningUrl(); String url = getMobileProvisioningUrl();
if (TextUtils.isEmpty(url) == false) { if (TextUtils.isEmpty(url) == false) {
log("CheckMp.onComplete: warm sim (no dns/tcp), url=" + url); if (DBG) log("CheckMp.onComplete: warm (no dns/tcp), url=" + url);
setNotificationVisible(true, ni, url); setProvNotificationVisible(true, ni.getType(), ni.getExtraInfo(),
url);
} else { } else {
log("CheckMp.onComplete: warm sim (no dns/tcp), no url"); if (DBG) log("CheckMp.onComplete: warm (no dns/tcp), no url");
} }
break; break;
} }
@@ -3652,16 +3819,16 @@ public class ConnectivityService extends IConnectivityManager.Stub {
break; break;
} }
} }
mIsCheckingMobileProvisioning.set(false);
} }
}; };
CheckMp.Params params = CheckMp.Params params =
new CheckMp.Params(checkMp.getDefaultUrl(), timeOutMs, cb); new CheckMp.Params(checkMp.getDefaultUrl(), timeOutMs, cb);
log("checkMobileProvisioning: params=" + params); if (DBG) log("checkMobileProvisioning: params=" + params);
setNotificationVisible(false, null, null);
checkMp.execute(params); checkMp.execute(params);
} finally { } finally {
Binder.restoreCallingIdentity(token); Binder.restoreCallingIdentity(token);
log("checkMobileProvisioning: X"); if (DBG) log("checkMobileProvisioning: X");
} }
return timeOutMs; return timeOutMs;
} }
@@ -3733,14 +3900,14 @@ public class ConnectivityService extends IConnectivityManager.Stub {
* a known address that fetches the data we expect. * a known address that fetches the data we expect.
*/ */
private synchronized Integer isMobileOk(Params params) { private synchronized Integer isMobileOk(Params params) {
Integer result = ConnectivityManager.CMP_RESULT_CODE_NO_CONNECTION; Integer result = CMP_RESULT_CODE_NO_CONNECTION;
Uri orgUri = Uri.parse(params.mUrl); Uri orgUri = Uri.parse(params.mUrl);
Random rand = new Random(); Random rand = new Random();
mParams = params; mParams = params;
if (mCs.isNetworkSupported(ConnectivityManager.TYPE_MOBILE) == false) { if (mCs.isNetworkSupported(ConnectivityManager.TYPE_MOBILE) == false) {
log("isMobileOk: not mobile capable"); log("isMobileOk: not mobile capable");
result = ConnectivityManager.CMP_RESULT_CODE_NO_CONNECTION; result = CMP_RESULT_CODE_NO_CONNECTION;
return result; return result;
} }
@@ -3776,7 +3943,7 @@ public class ConnectivityService extends IConnectivityManager.Stub {
break; break;
} }
if (VDBG) log("isMobileOk: hipri not started yet"); if (VDBG) log("isMobileOk: hipri not started yet");
result = ConnectivityManager.CMP_RESULT_CODE_NO_CONNECTION; result = CMP_RESULT_CODE_NO_CONNECTION;
sleep(1); sleep(1);
} }
@@ -3789,15 +3956,26 @@ public class ConnectivityService extends IConnectivityManager.Stub {
NetworkInfo.State state = mCs NetworkInfo.State state = mCs
.getNetworkInfo(ConnectivityManager.TYPE_MOBILE_HIPRI).getState(); .getNetworkInfo(ConnectivityManager.TYPE_MOBILE_HIPRI).getState();
if (state != NetworkInfo.State.CONNECTED) { if (state != NetworkInfo.State.CONNECTED) {
if (VDBG) { if (true/*VDBG*/) {
log("isMobileOk: not connected ni=" + log("isMobileOk: not connected ni=" +
mCs.getNetworkInfo(ConnectivityManager.TYPE_MOBILE_HIPRI)); mCs.getNetworkInfo(ConnectivityManager.TYPE_MOBILE_HIPRI));
} }
sleep(1); sleep(1);
result = ConnectivityManager.CMP_RESULT_CODE_NO_CONNECTION; result = CMP_RESULT_CODE_NO_CONNECTION;
continue; continue;
} }
// Hipri has started check if this is a provisioning url
MobileDataStateTracker mdst = (MobileDataStateTracker)
mCs.mNetTrackers[ConnectivityManager.TYPE_MOBILE_HIPRI];
if (mdst.isProvisioningNetwork()) {
if (DBG) log("isMobileOk: isProvisioningNetwork is true, no TCP conn");
result = CMP_RESULT_CODE_NO_TCP_CONNECTION;
return result;
} else {
if (DBG) log("isMobileOk: isProvisioningNetwork is false, continue");
}
// Get of the addresses associated with the url host. We need to use the // Get of the addresses associated with the url host. We need to use the
// address otherwise HttpURLConnection object will use the name to get // address otherwise HttpURLConnection object will use the name to get
// the addresses and is will try every address but that will bypass the // the addresses and is will try every address but that will bypass the
@@ -3808,7 +3986,7 @@ public class ConnectivityService extends IConnectivityManager.Stub {
addresses = InetAddress.getAllByName(orgUri.getHost()); addresses = InetAddress.getAllByName(orgUri.getHost());
} catch (UnknownHostException e) { } catch (UnknownHostException e) {
log("isMobileOk: UnknownHostException"); log("isMobileOk: UnknownHostException");
result = ConnectivityManager.CMP_RESULT_CODE_NO_DNS; result = CMP_RESULT_CODE_NO_DNS;
return result; return result;
} }
log("isMobileOk: addresses=" + inetAddressesToString(addresses)); log("isMobileOk: addresses=" + inetAddressesToString(addresses));
@@ -3873,9 +4051,9 @@ public class ConnectivityService extends IConnectivityManager.Stub {
urlConn.setRequestProperty("Connection", "close"); urlConn.setRequestProperty("Connection", "close");
int responseCode = urlConn.getResponseCode(); int responseCode = urlConn.getResponseCode();
if (responseCode == 204) { if (responseCode == 204) {
result = ConnectivityManager.CMP_RESULT_CODE_CONNECTABLE; result = CMP_RESULT_CODE_CONNECTABLE;
} else { } else {
result = ConnectivityManager.CMP_RESULT_CODE_REDIRECTED; result = CMP_RESULT_CODE_REDIRECTED;
} }
log("isMobileOk: connected responseCode=" + responseCode); log("isMobileOk: connected responseCode=" + responseCode);
urlConn.disconnect(); urlConn.disconnect();
@@ -3889,7 +4067,7 @@ public class ConnectivityService extends IConnectivityManager.Stub {
} }
} }
} }
result = ConnectivityManager.CMP_RESULT_CODE_NO_TCP_CONNECTION; result = CMP_RESULT_CODE_NO_TCP_CONNECTION;
log("isMobileOk: loops|timed out"); log("isMobileOk: loops|timed out");
return result; return result;
} catch (Exception e) { } catch (Exception e) {
@@ -3903,6 +4081,23 @@ public class ConnectivityService extends IConnectivityManager.Stub {
mCs.setEnableFailFastMobileData(DctConstants.DISABLED); mCs.setEnableFailFastMobileData(DctConstants.DISABLED);
mCs.stopUsingNetworkFeature(ConnectivityManager.TYPE_MOBILE, mCs.stopUsingNetworkFeature(ConnectivityManager.TYPE_MOBILE,
Phone.FEATURE_ENABLE_HIPRI); Phone.FEATURE_ENABLE_HIPRI);
// Wait for hipri to disconnect.
long endTime = SystemClock.elapsedRealtime() + 5000;
while(SystemClock.elapsedRealtime() < endTime) {
NetworkInfo.State state = mCs
.getNetworkInfo(ConnectivityManager.TYPE_MOBILE_HIPRI).getState();
if (state != NetworkInfo.State.DISCONNECTED) {
if (VDBG) {
log("isMobileOk: connected ni=" +
mCs.getNetworkInfo(ConnectivityManager.TYPE_MOBILE_HIPRI));
}
sleep(1);
continue;
}
}
log("isMobileOk: X result=" + result); log("isMobileOk: X result=" + result);
} }
return result; return result;
@@ -3982,10 +4177,55 @@ public class ConnectivityService extends IConnectivityManager.Stub {
} }
} }
private static final String NOTIFICATION_ID = "CaptivePortal.Notification"; // TODO: Move to ConnectivityManager and make public?
private static final String CONNECTED_TO_PROVISIONING_NETWORK_ACTION =
"com.android.server.connectivityservice.CONNECTED_TO_PROVISIONING_NETWORK_ACTION";
private void setNotificationVisible(boolean visible, NetworkInfo networkInfo, String url) { private BroadcastReceiver mProvisioningReceiver = new BroadcastReceiver() {
log("setNotificationVisible: E visible=" + visible + " ni=" + networkInfo + " url=" + url); @Override
public void onReceive(Context context, Intent intent) {
if (intent.getAction().equals(CONNECTED_TO_PROVISIONING_NETWORK_ACTION)) {
handleMobileProvisioningAction(intent.getStringExtra("EXTRA_URL"));
}
}
};
private void handleMobileProvisioningAction(String url) {
// Notication mark notification as not visible
setProvNotificationVisible(false, ConnectivityManager.TYPE_NONE, null, null);
// If provisioning network handle as a special case,
// otherwise launch browser with the intent directly.
NetworkInfo ni = getProvisioningNetworkInfo();
if ((ni != null) && ni.getDetailedState() ==
NetworkInfo.DetailedState.CONNECTED_TO_PROVISIONING_NETWORK) {
if (DBG) log("handleMobileProvisioningAction: on provisioning network");
MobileDataStateTracker mdst = (MobileDataStateTracker)
mNetTrackers[ConnectivityManager.TYPE_MOBILE];
mdst.enableMobileProvisioning(url);
} else {
if (DBG) log("handleMobileProvisioningAction: on default network");
Intent newIntent =
new Intent(Intent.ACTION_VIEW, Uri.parse(url));
newIntent.setFlags(Intent.FLAG_ACTIVITY_BROUGHT_TO_FRONT |
Intent.FLAG_ACTIVITY_NEW_TASK);
try {
mContext.startActivity(newIntent);
} catch (ActivityNotFoundException e) {
loge("handleMobileProvisioningAction: startActivity failed" + e);
}
}
}
private static final String NOTIFICATION_ID = "CaptivePortal.Notification";
private volatile boolean mIsNotificationVisible = false;
private void setProvNotificationVisible(boolean visible, int networkType, String extraInfo,
String url) {
if (DBG) {
log("setProvNotificationVisible: E visible=" + visible + " networkType=" + networkType
+ " extraInfo=" + extraInfo + " url=" + url);
}
Resources r = Resources.getSystem(); Resources r = Resources.getSystem();
NotificationManager notificationManager = (NotificationManager) mContext NotificationManager notificationManager = (NotificationManager) mContext
@@ -3995,50 +4235,64 @@ public class ConnectivityService extends IConnectivityManager.Stub {
CharSequence title; CharSequence title;
CharSequence details; CharSequence details;
int icon; int icon;
switch (networkInfo.getType()) { Intent intent;
Notification notification = new Notification();
switch (networkType) {
case ConnectivityManager.TYPE_WIFI: case ConnectivityManager.TYPE_WIFI:
log("setNotificationVisible: TYPE_WIFI");
title = r.getString(R.string.wifi_available_sign_in, 0); title = r.getString(R.string.wifi_available_sign_in, 0);
details = r.getString(R.string.network_available_sign_in_detailed, details = r.getString(R.string.network_available_sign_in_detailed,
networkInfo.getExtraInfo()); extraInfo);
icon = R.drawable.stat_notify_wifi_in_range; icon = R.drawable.stat_notify_wifi_in_range;
intent = new Intent(Intent.ACTION_VIEW, Uri.parse(url));
intent.setFlags(Intent.FLAG_ACTIVITY_BROUGHT_TO_FRONT |
Intent.FLAG_ACTIVITY_NEW_TASK);
notification.contentIntent = PendingIntent.getActivity(mContext, 0, intent, 0);
break; break;
case ConnectivityManager.TYPE_MOBILE: case ConnectivityManager.TYPE_MOBILE:
case ConnectivityManager.TYPE_MOBILE_HIPRI: case ConnectivityManager.TYPE_MOBILE_HIPRI:
log("setNotificationVisible: TYPE_MOBILE|HIPRI");
title = r.getString(R.string.network_available_sign_in, 0); title = r.getString(R.string.network_available_sign_in, 0);
// TODO: Change this to pull from NetworkInfo once a printable // TODO: Change this to pull from NetworkInfo once a printable
// name has been added to it // name has been added to it
details = mTelephonyManager.getNetworkOperatorName(); details = mTelephonyManager.getNetworkOperatorName();
icon = R.drawable.stat_notify_rssi_in_range; icon = R.drawable.stat_notify_rssi_in_range;
intent = new Intent(CONNECTED_TO_PROVISIONING_NETWORK_ACTION);
intent.putExtra("EXTRA_URL", url);
intent.setFlags(0);
notification.contentIntent = PendingIntent.getBroadcast(mContext, 0, intent, 0);
break; break;
default: default:
log("setNotificationVisible: other type=" + networkInfo.getType());
title = r.getString(R.string.network_available_sign_in, 0); title = r.getString(R.string.network_available_sign_in, 0);
details = r.getString(R.string.network_available_sign_in_detailed, details = r.getString(R.string.network_available_sign_in_detailed,
networkInfo.getExtraInfo()); extraInfo);
icon = R.drawable.stat_notify_rssi_in_range; icon = R.drawable.stat_notify_rssi_in_range;
intent = new Intent(Intent.ACTION_VIEW, Uri.parse(url));
intent.setFlags(Intent.FLAG_ACTIVITY_BROUGHT_TO_FRONT |
Intent.FLAG_ACTIVITY_NEW_TASK);
notification.contentIntent = PendingIntent.getActivity(mContext, 0, intent, 0);
break; break;
} }
Notification notification = new Notification();
notification.when = 0; notification.when = 0;
notification.icon = icon; notification.icon = icon;
notification.flags = Notification.FLAG_AUTO_CANCEL; notification.flags = Notification.FLAG_AUTO_CANCEL;
Intent intent = new Intent(Intent.ACTION_VIEW, Uri.parse(url));
intent.setFlags(Intent.FLAG_ACTIVITY_BROUGHT_TO_FRONT |
Intent.FLAG_ACTIVITY_NEW_TASK);
notification.contentIntent = PendingIntent.getActivity(mContext, 0, intent, 0);
notification.tickerText = title; notification.tickerText = title;
notification.setLatestEventInfo(mContext, title, details, notification.contentIntent); notification.setLatestEventInfo(mContext, title, details, notification.contentIntent);
log("setNotificaitionVisible: notify notificaiton=" + notification); try {
notificationManager.notify(NOTIFICATION_ID, 1, notification); notificationManager.notify(NOTIFICATION_ID, 1, notification);
} catch (NullPointerException npe) {
loge("setNotificaitionVisible: visible notificationManager npe=" + npe);
npe.printStackTrace();
}
} else { } else {
log("setNotificaitionVisible: cancel"); try {
notificationManager.cancel(NOTIFICATION_ID, 1); notificationManager.cancel(NOTIFICATION_ID, 1);
} catch (NullPointerException npe) {
loge("setNotificaitionVisible: cancel notificationManager npe=" + npe);
npe.printStackTrace();
}
} }
log("setNotificationVisible: X visible=" + visible + " ni=" + networkInfo + " url=" + url); mIsNotificationVisible = visible;
} }
/** Location to an updatable file listing carrier provisioning urls. /** Location to an updatable file listing carrier provisioning urls.
@@ -4166,4 +4420,11 @@ public class ConnectivityService extends IConnectivityManager.Stub {
return url; return url;
} }
@Override
public void setProvisioningNotificationVisible(boolean visible, int networkType,
String extraInfo, String url) {
enforceConnectivityInternalPermission();
setProvNotificationVisible(visible, networkType, extraInfo, url);
}
} }