blob: da9c5474b66bf9279919c88d5c3d04d7260d53dc [file] [log] [blame]
Robert Greenwaltd0e18ff2010-01-26 11:40:34 -08001/*
2 * Copyright (C) 2010 The Android Open Source Project
3 *
4 * Licensed under the Apache License, Version 2.0 (the "License");
5 * you may not use this file except in compliance with the License.
6 * You may obtain a copy of the License at
7 *
8 * http://www.apache.org/licenses/LICENSE-2.0
9 *
10 * Unless required by applicable law or agreed to in writing, software
11 * distributed under the License is distributed on an "AS IS" BASIS,
12 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
13 * See the License for the specific language governing permissions and
14 * limitations under the License.
15 */
16
17package com.android.server.connectivity;
18
19import android.app.Notification;
20import android.app.NotificationManager;
21import android.app.PendingIntent;
Jeremy Klein36c7aa02016-01-22 14:11:45 -080022import android.bluetooth.BluetoothAdapter;
23import android.bluetooth.BluetoothPan;
24import android.bluetooth.BluetoothProfile;
25import android.bluetooth.BluetoothProfile.ServiceListener;
Robert Greenwaltd0e18ff2010-01-26 11:40:34 -080026import android.content.BroadcastReceiver;
Robert Greenwalt2ffe4122014-12-12 12:22:31 -080027import android.content.ComponentName;
Robert Greenwaltd0e18ff2010-01-26 11:40:34 -080028import android.content.Context;
29import android.content.Intent;
30import android.content.IntentFilter;
Robert Greenwalt2a091d72010-02-11 18:18:40 -080031import android.content.pm.PackageManager;
Robert Greenwaltd0e18ff2010-01-26 11:40:34 -080032import android.content.res.Resources;
Mike Lockwoodc4308f02011-03-01 08:04:54 -080033import android.hardware.usb.UsbManager;
Robert Greenwaltd0e18ff2010-01-26 11:40:34 -080034import android.net.ConnectivityManager;
Erik Kline6ff17f72015-12-10 20:42:12 +090035import android.net.ConnectivityManager.NetworkCallback;
Christopher Wileydeebfec2016-09-16 11:14:36 -070036import android.net.INetworkPolicyManager;
Jeff Sharkey367d15a2011-09-22 14:59:51 -070037import android.net.INetworkStatsService;
Robert Greenwalt05d89362011-01-23 16:04:05 -080038import android.net.LinkProperties;
Lorenzo Colittib57edc52014-08-22 17:10:50 -070039import android.net.Network;
Erik Kline0c9cb2b2015-11-20 17:47:08 +090040import android.net.NetworkCapabilities;
Robert Greenwalt2a091d72010-02-11 18:18:40 -080041import android.net.NetworkInfo;
Erik Kline0c9cb2b2015-11-20 17:47:08 +090042import android.net.NetworkRequest;
Erik Kline6ff17f72015-12-10 20:42:12 +090043import android.net.NetworkState;
Robert Greenwalte5903732011-02-22 16:00:42 -080044import android.net.NetworkUtils;
Lorenzo Colitti13c9fde2013-03-15 04:22:37 +090045import android.net.RouteInfo;
Jeremy Klein36c7aa02016-01-22 14:11:45 -080046import android.net.wifi.WifiManager;
Robert Greenwalt2a091d72010-02-11 18:18:40 -080047import android.os.Binder;
Jeremy Klein36c7aa02016-01-22 14:11:45 -080048import android.os.Bundle;
Robert Greenwaltd0e18ff2010-01-26 11:40:34 -080049import android.os.INetworkManagementService;
Robert Greenwaltdfadaea2010-03-11 15:03:08 -080050import android.os.Looper;
Robert Greenwalt2a091d72010-02-11 18:18:40 -080051import android.os.Message;
Jeremy Klein36c7aa02016-01-22 14:11:45 -080052import android.os.Parcel;
Christopher Wileydeebfec2016-09-16 11:14:36 -070053import android.os.RemoteException;
Jeremy Klein36c7aa02016-01-22 14:11:45 -080054import android.os.ResultReceiver;
55import android.os.SystemProperties;
Dianne Hackborn5ac72a22012-08-29 18:32:08 -070056import android.os.UserHandle;
Jeremy Klein15f3d212016-01-24 17:01:09 -080057import android.provider.Settings;
Jeremy Klein36c7aa02016-01-22 14:11:45 -080058import android.telephony.CarrierConfigManager;
Junda Liu38531582014-10-28 13:52:51 -070059import android.telephony.TelephonyManager;
Lorenzo Colitticd63d242016-04-10 15:39:53 +090060import android.text.TextUtils;
Christopher Wileye9490392016-05-26 15:57:29 -070061import android.util.ArrayMap;
Robert Greenwaltd0e18ff2010-01-26 11:40:34 -080062import android.util.Log;
Lorenzo Colitticd63d242016-04-10 15:39:53 +090063import android.util.SparseArray;
Robert Greenwaltd0e18ff2010-01-26 11:40:34 -080064
Robert Greenwalt2ffe4122014-12-12 12:22:31 -080065import com.android.internal.telephony.IccCardConstants;
Robert Greenwalt2ffe4122014-12-12 12:22:31 -080066import com.android.internal.telephony.TelephonyIntents;
Jeremy Klein36c7aa02016-01-22 14:11:45 -080067import com.android.internal.util.IndentingPrintWriter;
Lorenzo Colitticd63d242016-04-10 15:39:53 +090068import com.android.internal.util.MessageUtils;
69import com.android.internal.util.Protocol;
Wink Saville64c42ca2011-04-18 14:55:10 -070070import com.android.internal.util.State;
71import com.android.internal.util.StateMachine;
Dianne Hackborn8d044e82013-04-30 17:24:15 -070072import com.android.server.IoThread;
Christopher Wileye03fb442016-05-18 13:45:20 -070073import com.android.server.connectivity.tethering.IControlsTethering;
Erik Kline1eb8c692016-07-08 17:21:26 +090074import com.android.server.connectivity.tethering.IPv6TetheringCoordinator;
Mitchell Wills7040b4e2016-05-23 16:40:10 -070075import com.android.server.connectivity.tethering.TetherInterfaceStateMachine;
Lorenzo Colittif4e90ea2013-10-31 23:30:47 +090076import com.android.server.net.BaseNetworkObserver;
Robert Greenwalt2a091d72010-02-11 18:18:40 -080077
78import java.io.FileDescriptor;
79import java.io.PrintWriter;
Robert Greenwalt4f74d552011-12-19 16:59:31 -080080import java.net.Inet4Address;
Jeremy Klein36c7aa02016-01-22 14:11:45 -080081import java.net.InetAddress;
Robert Greenwaltd0e18ff2010-01-26 11:40:34 -080082import java.util.ArrayList;
Lorenzo Colittib57edc52014-08-22 17:10:50 -070083import java.util.Arrays;
Robert Greenwaltccf83af12011-06-02 17:30:47 -070084import java.util.Collection;
Robert Greenwalt2a091d72010-02-11 18:18:40 -080085import java.util.HashMap;
Robert Greenwaltccf83af12011-06-02 17:30:47 -070086import java.util.Iterator;
Robert Greenwalt2ffe4122014-12-12 12:22:31 -080087import java.util.concurrent.atomic.AtomicInteger;
88
Jeff Sharkeycdd02c5d2011-09-16 01:52:49 -070089
Robert Greenwaltd0e18ff2010-01-26 11:40:34 -080090/**
91 * @hide
Robert Greenwalt2a091d72010-02-11 18:18:40 -080092 *
Christopher Wiley3b1d9222016-05-20 16:44:04 -070093 * This class holds much of the business logic to allow Android devices
94 * to act as IP gateways via USB, BT, and WiFi interfaces.
Robert Greenwaltd0e18ff2010-01-26 11:40:34 -080095 */
Christopher Wileye03fb442016-05-18 13:45:20 -070096public class Tethering extends BaseNetworkObserver implements IControlsTethering {
Robert Greenwaltd0e18ff2010-01-26 11:40:34 -080097
Erik Kline6ff17f72015-12-10 20:42:12 +090098 private final Context mContext;
Robert Greenwalt924cc942010-06-28 10:26:19 -070099 private final static String TAG = "Tethering";
Joe Onorato12acbd72016-02-01 17:49:31 -0800100 private final static boolean DBG = false;
Robert Greenwaltfd1be2b2011-11-11 12:30:19 -0800101 private final static boolean VDBG = false;
Robert Greenwaltd0e18ff2010-01-26 11:40:34 -0800102
Lorenzo Colitticd63d242016-04-10 15:39:53 +0900103 private static final Class[] messageClasses = {
Mitchell Wills7040b4e2016-05-23 16:40:10 -0700104 Tethering.class, TetherMasterSM.class, TetherInterfaceStateMachine.class
Lorenzo Colitticd63d242016-04-10 15:39:53 +0900105 };
106 private static final SparseArray<String> sMagicDecoderRing =
107 MessageUtils.findMessageNames(messageClasses);
108
Robert Greenwalt2a091d72010-02-11 18:18:40 -0800109 // TODO - remove both of these - should be part of interface inspection/selection stuff
110 private String[] mTetherableUsbRegexs;
111 private String[] mTetherableWifiRegexs;
Danica Chang6fdd0c62010-08-11 14:54:43 -0700112 private String[] mTetherableBluetoothRegexs;
Robert Greenwaltccf83af12011-06-02 17:30:47 -0700113 private Collection<Integer> mUpstreamIfaceTypes;
114
Robert Greenwaltb4453622011-11-03 16:01:40 -0700115 // used to synchronize public access to members
Erik Kline6ff17f72015-12-10 20:42:12 +0900116 private final Object mPublicSync;
Robert Greenwaltb4453622011-11-03 16:01:40 -0700117
Robert Greenwaltccf83af12011-06-02 17:30:47 -0700118 private static final Integer MOBILE_TYPE = new Integer(ConnectivityManager.TYPE_MOBILE);
119 private static final Integer HIPRI_TYPE = new Integer(ConnectivityManager.TYPE_MOBILE_HIPRI);
120 private static final Integer DUN_TYPE = new Integer(ConnectivityManager.TYPE_MOBILE_DUN);
121
122 // if we have to connect to mobile, what APN type should we use? Calculated by examining the
123 // upstream type list and the DUN_REQUIRED secure-setting
124 private int mPreferredUpstreamMobileApn = ConnectivityManager.TYPE_NONE;
Robert Greenwalt2a091d72010-02-11 18:18:40 -0800125
Jeff Sharkey367d15a2011-09-22 14:59:51 -0700126 private final INetworkManagementService mNMService;
127 private final INetworkStatsService mStatsService;
Christopher Wileydeebfec2016-09-16 11:14:36 -0700128 private final INetworkPolicyManager mPolicyManager;
Erik Kline6ff17f72015-12-10 20:42:12 +0900129 private final Looper mLooper;
Robert Greenwaltdfadaea2010-03-11 15:03:08 -0800130
Christopher Wileyd985dde2016-05-31 10:44:35 -0700131 private static class TetherState {
132 public final TetherInterfaceStateMachine mStateMachine;
133 public int mLastState;
134 public int mLastError;
135 public TetherState(TetherInterfaceStateMachine sm) {
136 mStateMachine = sm;
137 // Assume all state machines start out available and with no errors.
138 mLastState = IControlsTethering.STATE_AVAILABLE;
139 mLastError = ConnectivityManager.TETHER_ERROR_NO_ERROR;
140 }
141 }
142 private final ArrayMap<String, TetherState> mTetherStates;
Robert Greenwaltd0e18ff2010-01-26 11:40:34 -0800143
Christopher Wiley5c0b10a2016-05-31 14:43:08 -0700144 private final BroadcastReceiver mStateReceiver;
Robert Greenwaltd0e18ff2010-01-26 11:40:34 -0800145
Jeremy Klein36c7aa02016-01-22 14:11:45 -0800146 // {@link ComponentName} of the Service used to run tether provisioning.
147 private static final ComponentName TETHER_SERVICE = ComponentName.unflattenFromString(Resources
148 .getSystem().getString(com.android.internal.R.string.config_wifi_tether_enable));
149
Jaikumar Ganesh707952e2010-09-13 19:04:54 -0700150 // USB is 192.168.42.1 and 255.255.255.0
151 // Wifi is 192.168.43.1 and 255.255.255.0
152 // BT is limited to max default of 5 connections. 192.168.44.1 to 192.168.48.1
153 // with 255.255.255.0
Robert Greenwalt9c7e2c22014-06-23 14:53:42 -0700154 // P2P is 192.168.49.1 and 255.255.255.0
Robert Greenwaltd70a3d42010-02-23 18:15:29 -0800155
Robert Greenwalt2a091d72010-02-11 18:18:40 -0800156 private String[] mDhcpRange;
Chia-chi Yehc9338302011-05-11 16:35:13 -0700157 private static final String[] DHCP_DEFAULT_RANGE = {
158 "192.168.42.2", "192.168.42.254", "192.168.43.2", "192.168.43.254",
159 "192.168.44.2", "192.168.44.254", "192.168.45.2", "192.168.45.254",
160 "192.168.46.2", "192.168.46.254", "192.168.47.2", "192.168.47.254",
Robert Greenwalt9c7e2c22014-06-23 14:53:42 -0700161 "192.168.48.2", "192.168.48.254", "192.168.49.2", "192.168.49.254",
Chia-chi Yehc9338302011-05-11 16:35:13 -0700162 };
Robert Greenwalt2a091d72010-02-11 18:18:40 -0800163
Robert Greenwaltbb4fd0b2011-12-09 15:36:50 -0800164 private String[] mDefaultDnsServers;
Robert Greenwaltd70a3d42010-02-23 18:15:29 -0800165 private static final String DNS_DEFAULT_SERVER1 = "8.8.8.8";
Robert Greenwaltdf255c22010-10-07 15:32:08 -0700166 private static final String DNS_DEFAULT_SERVER2 = "8.8.4.4";
Robert Greenwalt2a091d72010-02-11 18:18:40 -0800167
Erik Kline6ff17f72015-12-10 20:42:12 +0900168 private final StateMachine mTetherMasterSM;
169 private final UpstreamNetworkMonitor mUpstreamNetworkMonitor;
170 private String mCurrentUpstreamIface;
Robert Greenwalta599fe7c2010-03-08 18:30:14 -0800171
Chris Wren1ce4b6d2015-06-11 10:19:43 -0400172 private Notification.Builder mTetheredNotificationBuilder;
173 private int mLastNotificationId;
Robert Greenwalt2a091d72010-02-11 18:18:40 -0800174
Mike Lockwood6c2260b2011-07-19 13:04:47 -0700175 private boolean mRndisEnabled; // track the RNDIS function enabled state
176 private boolean mUsbTetherRequested; // true if USB tethering should be started
177 // when RNDIS is enabled
Robert Greenwalt030f5e12010-03-10 16:41:03 -0800178
Christopher Wiley5c0b10a2016-05-31 14:43:08 -0700179 // True iff WiFi tethering should be started when soft AP is ready.
180 private boolean mWifiTetherRequested;
181
Jeff Sharkey367d15a2011-09-22 14:59:51 -0700182 public Tethering(Context context, INetworkManagementService nmService,
Christopher Wileydeebfec2016-09-16 11:14:36 -0700183 INetworkStatsService statsService, INetworkPolicyManager policyManager) {
Robert Greenwaltd0e18ff2010-01-26 11:40:34 -0800184 mContext = context;
Chia-chi Yehc9338302011-05-11 16:35:13 -0700185 mNMService = nmService;
Jeff Sharkey367d15a2011-09-22 14:59:51 -0700186 mStatsService = statsService;
Christopher Wileydeebfec2016-09-16 11:14:36 -0700187 mPolicyManager = policyManager;
Robert Greenwaltd0e18ff2010-01-26 11:40:34 -0800188
Robert Greenwaltb4453622011-11-03 16:01:40 -0700189 mPublicSync = new Object();
190
Christopher Wileyd985dde2016-05-31 10:44:35 -0700191 mTetherStates = new ArrayMap<>();
Robert Greenwaltd0e18ff2010-01-26 11:40:34 -0800192
Robert Greenwalt7eae2502010-04-20 15:51:20 -0700193 // make our own thread so we don't anr the system
Dianne Hackborn8d044e82013-04-30 17:24:15 -0700194 mLooper = IoThread.get().getLooper();
Robert Greenwaltdfadaea2010-03-11 15:03:08 -0800195 mTetherMasterSM = new TetherMasterSM("TetherMaster", mLooper);
Robert Greenwalt2a091d72010-02-11 18:18:40 -0800196 mTetherMasterSM.start();
197
Erik Kline6ff17f72015-12-10 20:42:12 +0900198 mUpstreamNetworkMonitor = new UpstreamNetworkMonitor();
199
Robert Greenwaltbb51d9f2010-03-12 10:23:35 -0800200 mStateReceiver = new StateReceiver();
Robert Greenwaltd0e18ff2010-01-26 11:40:34 -0800201 IntentFilter filter = new IntentFilter();
Mike Lockwood770126a2010-12-09 22:30:37 -0800202 filter.addAction(UsbManager.ACTION_USB_STATE);
Robert Greenwalt2a091d72010-02-11 18:18:40 -0800203 filter.addAction(ConnectivityManager.CONNECTIVITY_ACTION);
Christopher Wiley5c0b10a2016-05-31 14:43:08 -0700204 filter.addAction(WifiManager.WIFI_AP_STATE_CHANGED_ACTION);
Robert Greenwaltc13368b2013-07-18 14:24:42 -0700205 filter.addAction(Intent.ACTION_CONFIGURATION_CHANGED);
Robert Greenwaltbb51d9f2010-03-12 10:23:35 -0800206 mContext.registerReceiver(mStateReceiver, filter);
207
208 filter = new IntentFilter();
Robert Greenwalt030f5e12010-03-10 16:41:03 -0800209 filter.addAction(Intent.ACTION_MEDIA_SHARED);
210 filter.addAction(Intent.ACTION_MEDIA_UNSHARED);
Robert Greenwaltbb51d9f2010-03-12 10:23:35 -0800211 filter.addDataScheme("file");
212 mContext.registerReceiver(mStateReceiver, filter);
213
Robert Greenwalt2a091d72010-02-11 18:18:40 -0800214 mDhcpRange = context.getResources().getStringArray(
215 com.android.internal.R.array.config_tether_dhcp_range);
Robert Greenwaltbfb7bfa2010-03-24 16:03:21 -0700216 if ((mDhcpRange.length == 0) || (mDhcpRange.length % 2 ==1)) {
Chia-chi Yehc9338302011-05-11 16:35:13 -0700217 mDhcpRange = DHCP_DEFAULT_RANGE;
Robert Greenwalt2a091d72010-02-11 18:18:40 -0800218 }
219
Robert Greenwalt49348e72011-10-21 16:54:26 -0700220 // load device config info
221 updateConfiguration();
222
223 // TODO - remove and rely on real notifications of the current iface
Robert Greenwaltbb4fd0b2011-12-09 15:36:50 -0800224 mDefaultDnsServers = new String[2];
225 mDefaultDnsServers[0] = DNS_DEFAULT_SERVER1;
226 mDefaultDnsServers[1] = DNS_DEFAULT_SERVER2;
Robert Greenwalt49348e72011-10-21 16:54:26 -0700227 }
228
Sreeram Ramachandran62a61312014-07-24 16:05:17 -0700229 // We can't do this once in the Tethering() constructor and cache the value, because the
230 // CONNECTIVITY_SERVICE is registered only after the Tethering() constructor has completed.
231 private ConnectivityManager getConnectivityManager() {
232 return (ConnectivityManager) mContext.getSystemService(Context.CONNECTIVITY_SERVICE);
233 }
234
Robert Greenwalt49348e72011-10-21 16:54:26 -0700235 void updateConfiguration() {
Robert Greenwaltb4453622011-11-03 16:01:40 -0700236 String[] tetherableUsbRegexs = mContext.getResources().getStringArray(
Robert Greenwalt2a091d72010-02-11 18:18:40 -0800237 com.android.internal.R.array.config_tether_usb_regexs);
Robert Greenwaltb4453622011-11-03 16:01:40 -0700238 String[] tetherableWifiRegexs = mContext.getResources().getStringArray(
Robert Greenwalt2a091d72010-02-11 18:18:40 -0800239 com.android.internal.R.array.config_tether_wifi_regexs);
Robert Greenwaltb4453622011-11-03 16:01:40 -0700240 String[] tetherableBluetoothRegexs = mContext.getResources().getStringArray(
Danica Chang6fdd0c62010-08-11 14:54:43 -0700241 com.android.internal.R.array.config_tether_bluetooth_regexs);
Robert Greenwalt49348e72011-10-21 16:54:26 -0700242
243 int ifaceTypes[] = mContext.getResources().getIntArray(
Robert Greenwaltccf83af12011-06-02 17:30:47 -0700244 com.android.internal.R.array.config_tether_upstream_types);
Christopher Wiley499a57a2016-05-16 16:19:07 -0700245 Collection<Integer> upstreamIfaceTypes = new ArrayList<>();
Robert Greenwaltccf83af12011-06-02 17:30:47 -0700246 for (int i : ifaceTypes) {
Robert Greenwaltb4453622011-11-03 16:01:40 -0700247 upstreamIfaceTypes.add(new Integer(i));
248 }
249
250 synchronized (mPublicSync) {
251 mTetherableUsbRegexs = tetherableUsbRegexs;
252 mTetherableWifiRegexs = tetherableWifiRegexs;
253 mTetherableBluetoothRegexs = tetherableBluetoothRegexs;
254 mUpstreamIfaceTypes = upstreamIfaceTypes;
Robert Greenwaltccf83af12011-06-02 17:30:47 -0700255 }
256
257 // check if the upstream type list needs to be modified due to secure-settings
258 checkDunRequired();
Robert Greenwaltd0e18ff2010-01-26 11:40:34 -0800259 }
260
Erik Kline0c9cb2b2015-11-20 17:47:08 +0900261 @Override
Mike J. Chen6143f5f2011-06-23 15:17:51 -0700262 public void interfaceStatusChanged(String iface, boolean up) {
Lorenzo Colitticd63d242016-04-10 15:39:53 +0900263 // Never called directly: only called from interfaceLinkStateChanged.
264 // See NetlinkHandler.cpp:71.
Wink Savillec9acde92011-09-21 11:05:43 -0700265 if (VDBG) Log.d(TAG, "interfaceStatusChanged " + iface + ", " + up);
Robert Greenwaltb4453622011-11-03 16:01:40 -0700266 synchronized (mPublicSync) {
Christopher Wiley5c0b10a2016-05-31 14:43:08 -0700267 int interfaceType = ifaceNameToType(iface);
268 if (interfaceType == ConnectivityManager.TETHERING_INVALID) {
269 return;
Robert Greenwaltb4453622011-11-03 16:01:40 -0700270 }
Robert Greenwalt2a091d72010-02-11 18:18:40 -0800271
Christopher Wileyd985dde2016-05-31 10:44:35 -0700272 TetherState tetherState = mTetherStates.get(iface);
Mike J. Chen6143f5f2011-06-23 15:17:51 -0700273 if (up) {
Christopher Wileyd985dde2016-05-31 10:44:35 -0700274 if (tetherState == null) {
275 trackNewTetherableInterface(iface, interfaceType);
Robert Greenwalt2a091d72010-02-11 18:18:40 -0800276 }
277 } else {
Christopher Wileyd30aaeb2016-07-08 09:33:50 -0700278 if (interfaceType == ConnectivityManager.TETHERING_BLUETOOTH) {
Christopher Wileyd985dde2016-05-31 10:44:35 -0700279 tetherState.mStateMachine.sendMessage(
280 TetherInterfaceStateMachine.CMD_INTERFACE_DOWN);
281 mTetherStates.remove(iface);
Christopher Wileyd30aaeb2016-07-08 09:33:50 -0700282 } else {
283 // Ignore usb0 down after enabling RNDIS.
284 // We will handle disconnect in interfaceRemoved.
285 // Similarly, ignore interface down for WiFi. We monitor WiFi AP status
286 // through the WifiManager.WIFI_AP_STATE_CHANGED_ACTION intent.
287 if (VDBG) Log.d(TAG, "ignore interface down for " + iface);
Robert Greenwalt2a091d72010-02-11 18:18:40 -0800288 }
289 }
290 }
Robert Greenwaltd0e18ff2010-01-26 11:40:34 -0800291 }
292
Erik Kline0c9cb2b2015-11-20 17:47:08 +0900293 @Override
Mike J. Chenf59c7d02011-06-23 15:33:15 -0700294 public void interfaceLinkStateChanged(String iface, boolean up) {
Irfan Sheriff23eb2972011-07-22 15:21:10 -0700295 interfaceStatusChanged(iface, up);
Mike J. Chenf59c7d02011-06-23 15:33:15 -0700296 }
297
Robert Greenwalta599fe7c2010-03-08 18:30:14 -0800298 private boolean isUsb(String iface) {
Robert Greenwaltb4453622011-11-03 16:01:40 -0700299 synchronized (mPublicSync) {
300 for (String regex : mTetherableUsbRegexs) {
301 if (iface.matches(regex)) return true;
302 }
303 return false;
Robert Greenwalta599fe7c2010-03-08 18:30:14 -0800304 }
Robert Greenwalta599fe7c2010-03-08 18:30:14 -0800305 }
306
Christopher Wiley5c0b10a2016-05-31 14:43:08 -0700307 private boolean isWifi(String iface) {
Robert Greenwaltb4453622011-11-03 16:01:40 -0700308 synchronized (mPublicSync) {
309 for (String regex : mTetherableWifiRegexs) {
310 if (iface.matches(regex)) return true;
311 }
312 return false;
Robert Greenwalta599fe7c2010-03-08 18:30:14 -0800313 }
Robert Greenwalta599fe7c2010-03-08 18:30:14 -0800314 }
315
Christopher Wiley5c0b10a2016-05-31 14:43:08 -0700316 private boolean isBluetooth(String iface) {
Robert Greenwaltb4453622011-11-03 16:01:40 -0700317 synchronized (mPublicSync) {
318 for (String regex : mTetherableBluetoothRegexs) {
319 if (iface.matches(regex)) return true;
320 }
321 return false;
Danica Chang6fdd0c62010-08-11 14:54:43 -0700322 }
Danica Chang6fdd0c62010-08-11 14:54:43 -0700323 }
Mike Lockwood6c2260b2011-07-19 13:04:47 -0700324
Christopher Wiley5c0b10a2016-05-31 14:43:08 -0700325 private int ifaceNameToType(String iface) {
326 if (isWifi(iface)) {
327 return ConnectivityManager.TETHERING_WIFI;
328 } else if (isUsb(iface)) {
329 return ConnectivityManager.TETHERING_USB;
330 } else if (isBluetooth(iface)) {
331 return ConnectivityManager.TETHERING_BLUETOOTH;
332 }
333 return ConnectivityManager.TETHERING_INVALID;
334 }
335
Erik Kline0c9cb2b2015-11-20 17:47:08 +0900336 @Override
Robert Greenwalt2a091d72010-02-11 18:18:40 -0800337 public void interfaceAdded(String iface) {
Wink Savillec9acde92011-09-21 11:05:43 -0700338 if (VDBG) Log.d(TAG, "interfaceAdded " + iface);
Robert Greenwaltb4453622011-11-03 16:01:40 -0700339 synchronized (mPublicSync) {
Christopher Wiley5c0b10a2016-05-31 14:43:08 -0700340 int interfaceType = ifaceNameToType(iface);
341 if (interfaceType == ConnectivityManager.TETHERING_INVALID) {
Robert Greenwaltb4453622011-11-03 16:01:40 -0700342 if (VDBG) Log.d(TAG, iface + " is not a tetherable iface, ignoring");
343 return;
344 }
Robert Greenwalt65ae29b2010-02-18 11:25:54 -0800345
Christopher Wileyd985dde2016-05-31 10:44:35 -0700346 TetherState tetherState = mTetherStates.get(iface);
347 if (tetherState == null) {
348 trackNewTetherableInterface(iface, interfaceType);
349 } else {
Wink Savillec9acde92011-09-21 11:05:43 -0700350 if (VDBG) Log.d(TAG, "active iface (" + iface + ") reported as added, ignoring");
Robert Greenwalt2a091d72010-02-11 18:18:40 -0800351 }
Robert Greenwaltd0e18ff2010-01-26 11:40:34 -0800352 }
Robert Greenwaltd0e18ff2010-01-26 11:40:34 -0800353 }
354
Erik Kline0c9cb2b2015-11-20 17:47:08 +0900355 @Override
Robert Greenwalt2a091d72010-02-11 18:18:40 -0800356 public void interfaceRemoved(String iface) {
Wink Savillec9acde92011-09-21 11:05:43 -0700357 if (VDBG) Log.d(TAG, "interfaceRemoved " + iface);
Robert Greenwaltb4453622011-11-03 16:01:40 -0700358 synchronized (mPublicSync) {
Christopher Wileyd985dde2016-05-31 10:44:35 -0700359 TetherState tetherState = mTetherStates.get(iface);
360 if (tetherState == null) {
Wink Savillec9acde92011-09-21 11:05:43 -0700361 if (VDBG) {
Robert Greenwalt924cc942010-06-28 10:26:19 -0700362 Log.e(TAG, "attempting to remove unknown iface (" + iface + "), ignoring");
363 }
Robert Greenwalt2a091d72010-02-11 18:18:40 -0800364 return;
365 }
Christopher Wileyd985dde2016-05-31 10:44:35 -0700366 tetherState.mStateMachine.sendMessage(TetherInterfaceStateMachine.CMD_INTERFACE_DOWN);
367 mTetherStates.remove(iface);
Robert Greenwaltd0e18ff2010-01-26 11:40:34 -0800368 }
369 }
370
Jeremy Klein36c7aa02016-01-22 14:11:45 -0800371 public void startTethering(int type, ResultReceiver receiver,
372 boolean showProvisioningUi) {
373 if (!isTetherProvisioningRequired()) {
374 enableTetheringInternal(type, true, receiver);
375 return;
376 }
377
378 if (showProvisioningUi) {
379 runUiTetherProvisioningAndEnable(type, receiver);
380 } else {
381 runSilentTetherProvisioningAndEnable(type, receiver);
382 }
383 }
384
385 public void stopTethering(int type) {
386 enableTetheringInternal(type, false, null);
387 if (isTetherProvisioningRequired()) {
388 cancelTetherProvisioningRechecks(type);
389 }
390 }
391
392 /**
393 * Check if the device requires a provisioning check in order to enable tethering.
394 *
395 * @return a boolean - {@code true} indicating tether provisioning is required by the carrier.
396 */
397 private boolean isTetherProvisioningRequired() {
398 String[] provisionApp = mContext.getResources().getStringArray(
399 com.android.internal.R.array.config_mobile_hotspot_provision_app);
400 if (SystemProperties.getBoolean("net.tethering.noprovisioning", false)
401 || provisionApp == null) {
402 return false;
403 }
404
405 // Check carrier config for entitlement checks
406 final CarrierConfigManager configManager = (CarrierConfigManager) mContext
407 .getSystemService(Context.CARRIER_CONFIG_SERVICE);
Rebecca Silbersteinb07f1402016-10-04 13:53:33 -0700408 if (configManager != null && configManager.getConfig() != null) {
409 // we do have a CarrierConfigManager and it has a config.
410 boolean isEntitlementCheckRequired = configManager.getConfig().getBoolean(
411 CarrierConfigManager.KEY_REQUIRE_ENTITLEMENT_CHECKS_BOOL);
412 if (!isEntitlementCheckRequired) {
413 return false;
414 }
Jeremy Klein36c7aa02016-01-22 14:11:45 -0800415 }
416 return (provisionApp.length == 2);
417 }
418
419 /**
420 * Enables or disables tethering for the given type. This should only be called once
421 * provisioning has succeeded or is not necessary. It will also schedule provisioning rechecks
422 * for the specified interface.
423 */
424 private void enableTetheringInternal(int type, boolean enable, ResultReceiver receiver) {
Christopher Wiley5c0b10a2016-05-31 14:43:08 -0700425 boolean isProvisioningRequired = enable && isTetherProvisioningRequired();
426 int result;
Jeremy Klein36c7aa02016-01-22 14:11:45 -0800427 switch (type) {
428 case ConnectivityManager.TETHERING_WIFI:
Christopher Wiley5c0b10a2016-05-31 14:43:08 -0700429 result = setWifiTethering(enable);
430 if (isProvisioningRequired && result == ConnectivityManager.TETHER_ERROR_NO_ERROR) {
431 scheduleProvisioningRechecks(type);
Jeremy Klein36c7aa02016-01-22 14:11:45 -0800432 }
Christopher Wiley5c0b10a2016-05-31 14:43:08 -0700433 sendTetherResult(receiver, result);
Jeremy Klein36c7aa02016-01-22 14:11:45 -0800434 break;
435 case ConnectivityManager.TETHERING_USB:
Christopher Wiley5c0b10a2016-05-31 14:43:08 -0700436 result = setUsbTethering(enable);
437 if (isProvisioningRequired && result == ConnectivityManager.TETHER_ERROR_NO_ERROR) {
Jeremy Klein36c7aa02016-01-22 14:11:45 -0800438 scheduleProvisioningRechecks(type);
439 }
440 sendTetherResult(receiver, result);
441 break;
442 case ConnectivityManager.TETHERING_BLUETOOTH:
443 setBluetoothTethering(enable, receiver);
444 break;
445 default:
446 Log.w(TAG, "Invalid tether type.");
447 sendTetherResult(receiver, ConnectivityManager.TETHER_ERROR_UNKNOWN_IFACE);
448 }
449 }
450
451 private void sendTetherResult(ResultReceiver receiver, int result) {
452 if (receiver != null) {
453 receiver.send(result, null);
454 }
455 }
456
Christopher Wiley5c0b10a2016-05-31 14:43:08 -0700457 private int setWifiTethering(final boolean enable) {
458 synchronized (mPublicSync) {
Christopher Wiley5c0b10a2016-05-31 14:43:08 -0700459 mWifiTetherRequested = enable;
460 final WifiManager wifiManager =
461 (WifiManager) mContext.getSystemService(Context.WIFI_SERVICE);
462 if (wifiManager.setWifiApEnabled(null /* use existing wifi config */, enable)) {
463 return ConnectivityManager.TETHER_ERROR_NO_ERROR;
464 }
465 return ConnectivityManager.TETHER_ERROR_MASTER_ERROR;
466 }
467 }
468
Jeremy Klein36c7aa02016-01-22 14:11:45 -0800469 private void setBluetoothTethering(final boolean enable, final ResultReceiver receiver) {
470 final BluetoothAdapter adapter = BluetoothAdapter.getDefaultAdapter();
471 if (adapter == null || !adapter.isEnabled()) {
472 Log.w(TAG, "Tried to enable bluetooth tethering with null or disabled adapter. null: " +
473 (adapter == null));
474 sendTetherResult(receiver, ConnectivityManager.TETHER_ERROR_SERVICE_UNAVAIL);
475 return;
476 }
477
478 adapter.getProfileProxy(mContext, new ServiceListener() {
479 @Override
480 public void onServiceDisconnected(int profile) { }
481
482 @Override
483 public void onServiceConnected(int profile, BluetoothProfile proxy) {
484 ((BluetoothPan) proxy).setBluetoothTethering(enable);
485 // TODO: Enabling bluetooth tethering can fail asynchronously here.
486 // We should figure out a way to bubble up that failure instead of sending success.
Jeremy Klein45ae04a2016-02-05 10:37:32 -0800487 int result = ((BluetoothPan) proxy).isTetheringOn() == enable ?
Jeremy Klein36c7aa02016-01-22 14:11:45 -0800488 ConnectivityManager.TETHER_ERROR_NO_ERROR :
489 ConnectivityManager.TETHER_ERROR_MASTER_ERROR;
490 sendTetherResult(receiver, result);
491 if (enable && isTetherProvisioningRequired()) {
492 scheduleProvisioningRechecks(ConnectivityManager.TETHERING_BLUETOOTH);
493 }
494 adapter.closeProfileProxy(BluetoothProfile.PAN, proxy);
495 }
496 }, BluetoothProfile.PAN);
497 }
498
499 private void runUiTetherProvisioningAndEnable(int type, ResultReceiver receiver) {
Jeremy Klein15f3d212016-01-24 17:01:09 -0800500 ResultReceiver proxyReceiver = getProxyReceiver(type, receiver);
501 sendUiTetherProvisionIntent(type, proxyReceiver);
502 }
503
504 private void sendUiTetherProvisionIntent(int type, ResultReceiver receiver) {
505 Intent intent = new Intent(Settings.ACTION_TETHER_PROVISIONING);
506 intent.putExtra(ConnectivityManager.EXTRA_ADD_TETHER_TYPE, type);
507 intent.putExtra(ConnectivityManager.EXTRA_PROVISION_CALLBACK, receiver);
508 final long ident = Binder.clearCallingIdentity();
509 try {
510 mContext.startActivityAsUser(intent, UserHandle.CURRENT);
511 } finally {
512 Binder.restoreCallingIdentity(ident);
513 }
Jeremy Klein36c7aa02016-01-22 14:11:45 -0800514 }
515
516 /**
517 * Creates a proxy {@link ResultReceiver} which enables tethering if the provsioning result is
518 * successful before firing back up to the wrapped receiver.
519 *
520 * @param type The type of tethering being enabled.
521 * @param receiver A ResultReceiver which will be called back with an int resultCode.
522 * @return The proxy receiver.
523 */
524 private ResultReceiver getProxyReceiver(final int type, final ResultReceiver receiver) {
525 ResultReceiver rr = new ResultReceiver(null) {
526 @Override
527 protected void onReceiveResult(int resultCode, Bundle resultData) {
528 // If provisioning is successful, enable tethering, otherwise just send the error.
529 if (resultCode == ConnectivityManager.TETHER_ERROR_NO_ERROR) {
530 enableTetheringInternal(type, true, receiver);
531 } else {
532 sendTetherResult(receiver, resultCode);
533 }
534 }
535 };
536
537 // The following is necessary to avoid unmarshalling issues when sending the receiver
Erik Kline6ff17f72015-12-10 20:42:12 +0900538 // across processes.
Jeremy Klein36c7aa02016-01-22 14:11:45 -0800539 Parcel parcel = Parcel.obtain();
540 rr.writeToParcel(parcel,0);
541 parcel.setDataPosition(0);
542 ResultReceiver receiverForSending = ResultReceiver.CREATOR.createFromParcel(parcel);
543 parcel.recycle();
544 return receiverForSending;
545 }
546
547 private void scheduleProvisioningRechecks(int type) {
548 Intent intent = new Intent();
549 intent.putExtra(ConnectivityManager.EXTRA_ADD_TETHER_TYPE, type);
550 intent.putExtra(ConnectivityManager.EXTRA_SET_ALARM, true);
551 intent.setComponent(TETHER_SERVICE);
552 final long ident = Binder.clearCallingIdentity();
553 try {
554 mContext.startServiceAsUser(intent, UserHandle.CURRENT);
555 } finally {
556 Binder.restoreCallingIdentity(ident);
557 }
558 }
559
560 private void runSilentTetherProvisioningAndEnable(int type, ResultReceiver receiver) {
561 ResultReceiver proxyReceiver = getProxyReceiver(type, receiver);
562 sendSilentTetherProvisionIntent(type, proxyReceiver);
563 }
564
565 private void sendSilentTetherProvisionIntent(int type, ResultReceiver receiver) {
566 Intent intent = new Intent();
567 intent.putExtra(ConnectivityManager.EXTRA_ADD_TETHER_TYPE, type);
568 intent.putExtra(ConnectivityManager.EXTRA_RUN_PROVISION, true);
569 intent.putExtra(ConnectivityManager.EXTRA_PROVISION_CALLBACK, receiver);
570 intent.setComponent(TETHER_SERVICE);
571 final long ident = Binder.clearCallingIdentity();
572 try {
573 mContext.startServiceAsUser(intent, UserHandle.CURRENT);
574 } finally {
575 Binder.restoreCallingIdentity(ident);
576 }
577 }
578
579 private void cancelTetherProvisioningRechecks(int type) {
580 if (getConnectivityManager().isTetheringSupported()) {
581 Intent intent = new Intent();
582 intent.putExtra(ConnectivityManager.EXTRA_REM_TETHER_TYPE, type);
583 intent.setComponent(TETHER_SERVICE);
584 final long ident = Binder.clearCallingIdentity();
585 try {
586 mContext.startServiceAsUser(intent, UserHandle.CURRENT);
587 } finally {
588 Binder.restoreCallingIdentity(ident);
589 }
590 }
591 }
Erik Kline6ff17f72015-12-10 20:42:12 +0900592
Robert Greenwalt5a735062010-03-02 17:25:02 -0800593 public int tether(String iface) {
Wink Savillec9acde92011-09-21 11:05:43 -0700594 if (DBG) Log.d(TAG, "Tethering " + iface);
Robert Greenwaltb4453622011-11-03 16:01:40 -0700595 synchronized (mPublicSync) {
Christopher Wileyd985dde2016-05-31 10:44:35 -0700596 TetherState tetherState = mTetherStates.get(iface);
597 if (tetherState == null) {
Erik Kline00019f42016-06-30 19:31:46 +0900598 Log.e(TAG, "Tried to Tether an unknown iface: " + iface + ", ignoring");
Christopher Wiley4312a4c2016-05-23 13:58:00 -0700599 return ConnectivityManager.TETHER_ERROR_UNKNOWN_IFACE;
600 }
601 // Ignore the error status of the interface. If the interface is available,
602 // the errors are referring to past tethering attempts anyway.
Christopher Wileyd985dde2016-05-31 10:44:35 -0700603 if (tetherState.mLastState != IControlsTethering.STATE_AVAILABLE) {
Erik Kline00019f42016-06-30 19:31:46 +0900604 Log.e(TAG, "Tried to Tether an unavailable iface: " + iface + ", ignoring");
Christopher Wiley4312a4c2016-05-23 13:58:00 -0700605 return ConnectivityManager.TETHER_ERROR_UNAVAIL_IFACE;
Christopher Wiley4312a4c2016-05-23 13:58:00 -0700606 }
Christopher Wileyd985dde2016-05-31 10:44:35 -0700607 tetherState.mStateMachine.sendMessage(TetherInterfaceStateMachine.CMD_TETHER_REQUESTED);
Christopher Wiley4312a4c2016-05-23 13:58:00 -0700608 return ConnectivityManager.TETHER_ERROR_NO_ERROR;
Robert Greenwalt2a091d72010-02-11 18:18:40 -0800609 }
Robert Greenwaltd0e18ff2010-01-26 11:40:34 -0800610 }
611
Robert Greenwalt5a735062010-03-02 17:25:02 -0800612 public int untether(String iface) {
Wink Savillec9acde92011-09-21 11:05:43 -0700613 if (DBG) Log.d(TAG, "Untethering " + iface);
Robert Greenwaltb4453622011-11-03 16:01:40 -0700614 synchronized (mPublicSync) {
Christopher Wileyd985dde2016-05-31 10:44:35 -0700615 TetherState tetherState = mTetherStates.get(iface);
616 if (tetherState == null) {
Christopher Wiley4312a4c2016-05-23 13:58:00 -0700617 Log.e(TAG, "Tried to Untether an unknown iface :" + iface + ", ignoring");
618 return ConnectivityManager.TETHER_ERROR_UNKNOWN_IFACE;
619 }
Christopher Wileyd985dde2016-05-31 10:44:35 -0700620 if (tetherState.mLastState != IControlsTethering.STATE_TETHERED) {
621 Log.e(TAG, "Tried to untether an untethered iface :" + iface + ", ignoring");
Christopher Wiley4312a4c2016-05-23 13:58:00 -0700622 return ConnectivityManager.TETHER_ERROR_UNAVAIL_IFACE;
623 }
Christopher Wileyd985dde2016-05-31 10:44:35 -0700624 tetherState.mStateMachine.sendMessage(
625 TetherInterfaceStateMachine.CMD_TETHER_UNREQUESTED);
Christopher Wiley4312a4c2016-05-23 13:58:00 -0700626 return ConnectivityManager.TETHER_ERROR_NO_ERROR;
Robert Greenwalt2a091d72010-02-11 18:18:40 -0800627 }
Robert Greenwalt5a735062010-03-02 17:25:02 -0800628 }
629
Felipe Leme70c8b9b2016-04-25 14:41:31 -0700630 public void untetherAll() {
Christopher Wileyfa6a4862016-09-19 08:58:06 -0700631 stopTethering(ConnectivityManager.TETHERING_WIFI);
632 stopTethering(ConnectivityManager.TETHERING_USB);
633 stopTethering(ConnectivityManager.TETHERING_BLUETOOTH);
Felipe Leme70c8b9b2016-04-25 14:41:31 -0700634 }
635
Robert Greenwalt5a735062010-03-02 17:25:02 -0800636 public int getLastTetherError(String iface) {
Robert Greenwaltb4453622011-11-03 16:01:40 -0700637 synchronized (mPublicSync) {
Christopher Wileyd985dde2016-05-31 10:44:35 -0700638 TetherState tetherState = mTetherStates.get(iface);
639 if (tetherState == null) {
Robert Greenwaltb4453622011-11-03 16:01:40 -0700640 Log.e(TAG, "Tried to getLastTetherError on an unknown iface :" + iface +
641 ", ignoring");
642 return ConnectivityManager.TETHER_ERROR_UNKNOWN_IFACE;
643 }
Christopher Wileyd985dde2016-05-31 10:44:35 -0700644 return tetherState.mLastError;
Robert Greenwalt5a735062010-03-02 17:25:02 -0800645 }
Robert Greenwaltd0e18ff2010-01-26 11:40:34 -0800646 }
647
Christopher Wileyd985dde2016-05-31 10:44:35 -0700648 private void sendTetherStateChangedBroadcast() {
Sreeram Ramachandran62a61312014-07-24 16:05:17 -0700649 if (!getConnectivityManager().isTetheringSupported()) return;
Robert Greenwalt2a091d72010-02-11 18:18:40 -0800650
651 ArrayList<String> availableList = new ArrayList<String>();
652 ArrayList<String> activeList = new ArrayList<String>();
653 ArrayList<String> erroredList = new ArrayList<String>();
654
Robert Greenwalta599fe7c2010-03-08 18:30:14 -0800655 boolean wifiTethered = false;
656 boolean usbTethered = false;
Danica Chang6fdd0c62010-08-11 14:54:43 -0700657 boolean bluetoothTethered = false;
Robert Greenwalta599fe7c2010-03-08 18:30:14 -0800658
Robert Greenwaltb4453622011-11-03 16:01:40 -0700659 synchronized (mPublicSync) {
Christopher Wileyd985dde2016-05-31 10:44:35 -0700660 for (int i = 0; i < mTetherStates.size(); i++) {
661 TetherState tetherState = mTetherStates.valueAt(i);
662 String iface = mTetherStates.keyAt(i);
663 if (tetherState.mLastError != ConnectivityManager.TETHER_ERROR_NO_ERROR) {
664 erroredList.add(iface);
665 } else if (tetherState.mLastState == IControlsTethering.STATE_AVAILABLE) {
666 availableList.add(iface);
667 } else if (tetherState.mLastState == IControlsTethering.STATE_TETHERED) {
668 if (isUsb(iface)) {
669 usbTethered = true;
670 } else if (isWifi(iface)) {
671 wifiTethered = true;
672 } else if (isBluetooth(iface)) {
673 bluetoothTethered = true;
Robert Greenwalt2a091d72010-02-11 18:18:40 -0800674 }
Christopher Wileyd985dde2016-05-31 10:44:35 -0700675 activeList.add(iface);
Robert Greenwalt2a091d72010-02-11 18:18:40 -0800676 }
677 }
678 }
Robert Greenwaltd0e18ff2010-01-26 11:40:34 -0800679 Intent broadcast = new Intent(ConnectivityManager.ACTION_TETHER_STATE_CHANGED);
Robert Greenwalt98c79e52011-07-28 11:51:11 -0700680 broadcast.addFlags(Intent.FLAG_RECEIVER_REPLACE_PENDING |
681 Intent.FLAG_RECEIVER_REGISTERED_ONLY_BEFORE_BOOT);
Robert Greenwalt2a091d72010-02-11 18:18:40 -0800682 broadcast.putStringArrayListExtra(ConnectivityManager.EXTRA_AVAILABLE_TETHER,
683 availableList);
684 broadcast.putStringArrayListExtra(ConnectivityManager.EXTRA_ACTIVE_TETHER, activeList);
685 broadcast.putStringArrayListExtra(ConnectivityManager.EXTRA_ERRORED_TETHER,
686 erroredList);
Dianne Hackborn5ac72a22012-08-29 18:32:08 -0700687 mContext.sendStickyBroadcastAsUser(broadcast, UserHandle.ALL);
Robert Greenwaltfd1be2b2011-11-11 12:30:19 -0800688 if (DBG) {
Lorenzo Colitticd63d242016-04-10 15:39:53 +0900689 Log.d(TAG, String.format(
690 "sendTetherStateChangedBroadcast avail=[%s] active=[%s] error=[%s]",
691 TextUtils.join(",", availableList),
692 TextUtils.join(",", activeList),
693 TextUtils.join(",", erroredList)));
Robert Greenwalt924cc942010-06-28 10:26:19 -0700694 }
Robert Greenwalta599fe7c2010-03-08 18:30:14 -0800695
696 if (usbTethered) {
Danica Chang6fdd0c62010-08-11 14:54:43 -0700697 if (wifiTethered || bluetoothTethered) {
Robert Greenwalta599fe7c2010-03-08 18:30:14 -0800698 showTetheredNotification(com.android.internal.R.drawable.stat_sys_tether_general);
699 } else {
700 showTetheredNotification(com.android.internal.R.drawable.stat_sys_tether_usb);
701 }
702 } else if (wifiTethered) {
Danica Chang6fdd0c62010-08-11 14:54:43 -0700703 if (bluetoothTethered) {
704 showTetheredNotification(com.android.internal.R.drawable.stat_sys_tether_general);
705 } else {
Vinit Deshpande2576a802014-11-18 13:56:15 -0800706 /* We now have a status bar icon for WifiTethering, so drop the notification */
707 clearTetheredNotification();
Danica Chang6fdd0c62010-08-11 14:54:43 -0700708 }
709 } else if (bluetoothTethered) {
710 showTetheredNotification(com.android.internal.R.drawable.stat_sys_tether_bluetooth);
Robert Greenwalta599fe7c2010-03-08 18:30:14 -0800711 } else {
712 clearTetheredNotification();
713 }
714 }
715
716 private void showTetheredNotification(int icon) {
717 NotificationManager notificationManager =
718 (NotificationManager)mContext.getSystemService(Context.NOTIFICATION_SERVICE);
719 if (notificationManager == null) {
720 return;
721 }
722
Chris Wren1ce4b6d2015-06-11 10:19:43 -0400723 if (mLastNotificationId != 0) {
724 if (mLastNotificationId == icon) {
Robert Greenwaltdb3fe9e2010-03-18 16:28:30 -0700725 return;
726 }
Chris Wren1ce4b6d2015-06-11 10:19:43 -0400727 notificationManager.cancelAsUser(null, mLastNotificationId,
Robert Greenwalt3cab6b02012-10-04 16:44:26 -0700728 UserHandle.ALL);
Chris Wren1ce4b6d2015-06-11 10:19:43 -0400729 mLastNotificationId = 0;
Robert Greenwaltdb3fe9e2010-03-18 16:28:30 -0700730 }
731
Robert Greenwalta599fe7c2010-03-08 18:30:14 -0800732 Intent intent = new Intent();
733 intent.setClassName("com.android.settings", "com.android.settings.TetherSettings");
734 intent.setFlags(Intent.FLAG_ACTIVITY_NO_HISTORY);
735
Dianne Hackborn50cdf7c32012-09-23 17:08:57 -0700736 PendingIntent pi = PendingIntent.getActivityAsUser(mContext, 0, intent, 0,
737 null, UserHandle.CURRENT);
Robert Greenwalta599fe7c2010-03-08 18:30:14 -0800738
739 Resources r = Resources.getSystem();
740 CharSequence title = r.getText(com.android.internal.R.string.tethered_notification_title);
741 CharSequence message = r.getText(com.android.internal.R.string.
742 tethered_notification_message);
743
Chris Wren1ce4b6d2015-06-11 10:19:43 -0400744 if (mTetheredNotificationBuilder == null) {
745 mTetheredNotificationBuilder = new Notification.Builder(mContext);
746 mTetheredNotificationBuilder.setWhen(0)
747 .setOngoing(true)
748 .setColor(mContext.getColor(
749 com.android.internal.R.color.system_notification_accent_color))
750 .setVisibility(Notification.VISIBILITY_PUBLIC)
751 .setCategory(Notification.CATEGORY_STATUS);
Robert Greenwalta599fe7c2010-03-08 18:30:14 -0800752 }
Chris Wren1ce4b6d2015-06-11 10:19:43 -0400753 mTetheredNotificationBuilder.setSmallIcon(icon)
754 .setContentTitle(title)
755 .setContentText(message)
756 .setContentIntent(pi);
757 mLastNotificationId = icon;
Robert Greenwalta599fe7c2010-03-08 18:30:14 -0800758
Chris Wren1ce4b6d2015-06-11 10:19:43 -0400759 notificationManager.notifyAsUser(null, mLastNotificationId,
760 mTetheredNotificationBuilder.build(), UserHandle.ALL);
Robert Greenwalta599fe7c2010-03-08 18:30:14 -0800761 }
762
763 private void clearTetheredNotification() {
764 NotificationManager notificationManager =
765 (NotificationManager)mContext.getSystemService(Context.NOTIFICATION_SERVICE);
Chris Wren1ce4b6d2015-06-11 10:19:43 -0400766 if (notificationManager != null && mLastNotificationId != 0) {
767 notificationManager.cancelAsUser(null, mLastNotificationId,
Dianne Hackborn50cdf7c32012-09-23 17:08:57 -0700768 UserHandle.ALL);
Chris Wren1ce4b6d2015-06-11 10:19:43 -0400769 mLastNotificationId = 0;
Robert Greenwalta599fe7c2010-03-08 18:30:14 -0800770 }
Robert Greenwalt5a735062010-03-02 17:25:02 -0800771 }
772
Robert Greenwalt2a091d72010-02-11 18:18:40 -0800773 private class StateReceiver extends BroadcastReceiver {
Nick Kralevich70c117a2014-05-27 15:30:02 -0700774 @Override
Robert Greenwaltd0e18ff2010-01-26 11:40:34 -0800775 public void onReceive(Context content, Intent intent) {
Robert Greenwalt2a091d72010-02-11 18:18:40 -0800776 String action = intent.getAction();
Nick Kralevich70c117a2014-05-27 15:30:02 -0700777 if (action == null) { return; }
Mike Lockwood770126a2010-12-09 22:30:37 -0800778 if (action.equals(UsbManager.ACTION_USB_STATE)) {
Robert Greenwaltb4453622011-11-03 16:01:40 -0700779 synchronized (Tethering.this.mPublicSync) {
Mike Lockwood6c2260b2011-07-19 13:04:47 -0700780 boolean usbConnected = intent.getBooleanExtra(UsbManager.USB_CONNECTED, false);
781 mRndisEnabled = intent.getBooleanExtra(UsbManager.USB_FUNCTION_RNDIS, false);
782 // start tethering if we have a request pending
783 if (usbConnected && mRndisEnabled && mUsbTetherRequested) {
Christopher Wiley5c0b10a2016-05-31 14:43:08 -0700784 tetherMatchingInterfaces(true, ConnectivityManager.TETHERING_USB);
Mike Lockwood6c2260b2011-07-19 13:04:47 -0700785 }
786 mUsbTetherRequested = false;
787 }
Robert Greenwalt2a091d72010-02-11 18:18:40 -0800788 } else if (action.equals(ConnectivityManager.CONNECTIVITY_ACTION)) {
Robert Greenwaltadfb0d02012-06-14 15:50:32 -0700789 NetworkInfo networkInfo = (NetworkInfo)intent.getParcelableExtra(
790 ConnectivityManager.EXTRA_NETWORK_INFO);
791 if (networkInfo != null &&
792 networkInfo.getDetailedState() != NetworkInfo.DetailedState.FAILED) {
793 if (VDBG) Log.d(TAG, "Tethering got CONNECTIVITY_ACTION");
794 mTetherMasterSM.sendMessage(TetherMasterSM.CMD_UPSTREAM_CHANGED);
795 }
Christopher Wiley5c0b10a2016-05-31 14:43:08 -0700796 } else if (action.equals(WifiManager.WIFI_AP_STATE_CHANGED_ACTION)) {
797 synchronized (Tethering.this.mPublicSync) {
Christopher Wiley5c0b10a2016-05-31 14:43:08 -0700798 int curState = intent.getIntExtra(WifiManager.EXTRA_WIFI_AP_STATE,
799 WifiManager.WIFI_AP_STATE_DISABLED);
800 switch (curState) {
801 case WifiManager.WIFI_AP_STATE_ENABLING:
802 // We can see this state on the way to both enabled and failure states.
803 break;
804 case WifiManager.WIFI_AP_STATE_ENABLED:
Christopher Wileye52b24a2016-07-13 18:57:34 -0700805 // When the AP comes up and we've been requested to tether it, do so.
806 if (mWifiTetherRequested) {
807 tetherMatchingInterfaces(true, ConnectivityManager.TETHERING_WIFI);
808 }
Christopher Wiley5c0b10a2016-05-31 14:43:08 -0700809 break;
810 case WifiManager.WIFI_AP_STATE_DISABLED:
811 case WifiManager.WIFI_AP_STATE_DISABLING:
812 case WifiManager.WIFI_AP_STATE_FAILED:
813 default:
814 if (DBG) {
815 Log.d(TAG, "Canceling WiFi tethering request - AP_STATE=" +
816 curState);
817 }
Christopher Wileye52b24a2016-07-13 18:57:34 -0700818 // Tell appropriate interface state machines that they should tear
819 // themselves down.
820 for (int i = 0; i < mTetherStates.size(); i++) {
821 TetherInterfaceStateMachine tism =
822 mTetherStates.valueAt(i).mStateMachine;
823 if (tism.interfaceType() == ConnectivityManager.TETHERING_WIFI) {
824 tism.sendMessage(
825 TetherInterfaceStateMachine.CMD_TETHER_UNREQUESTED);
826 break; // There should be at most one of these.
827 }
828 }
829 // Regardless of whether we requested this transition, the AP has gone
830 // down. Don't try to tether again unless we're requested to do so.
831 mWifiTetherRequested = false;
Christopher Wiley5c0b10a2016-05-31 14:43:08 -0700832 break;
833 }
834 }
Robert Greenwaltc13368b2013-07-18 14:24:42 -0700835 } else if (action.equals(Intent.ACTION_CONFIGURATION_CHANGED)) {
836 updateConfiguration();
Robert Greenwaltd0e18ff2010-01-26 11:40:34 -0800837 }
838 }
839 }
840
Christopher Wiley5c0b10a2016-05-31 14:43:08 -0700841 private void tetherMatchingInterfaces(boolean enable, int interfaceType) {
842 if (VDBG) Log.d(TAG, "tetherMatchingInterfaces(" + enable + ", " + interfaceType + ")");
Mike Lockwood3c2a2f62011-06-08 15:10:26 -0700843
Christopher Wiley5c0b10a2016-05-31 14:43:08 -0700844 String[] ifaces = null;
Robert Greenwalt65ae29b2010-02-18 11:25:54 -0800845 try {
Chia-chi Yehc9338302011-05-11 16:35:13 -0700846 ifaces = mNMService.listInterfaces();
Robert Greenwalt65ae29b2010-02-18 11:25:54 -0800847 } catch (Exception e) {
Mike Lockwood3c2a2f62011-06-08 15:10:26 -0700848 Log.e(TAG, "Error listing Interfaces", e);
Robert Greenwalt65ae29b2010-02-18 11:25:54 -0800849 return;
850 }
Christopher Wiley5c0b10a2016-05-31 14:43:08 -0700851 String chosenIface = null;
852 if (ifaces != null) {
853 for (String iface : ifaces) {
854 if (ifaceNameToType(iface) == interfaceType) {
855 chosenIface = iface;
856 break;
Robert Greenwalt65ae29b2010-02-18 11:25:54 -0800857 }
858 }
859 }
Christopher Wiley5c0b10a2016-05-31 14:43:08 -0700860 if (chosenIface == null) {
861 Log.e(TAG, "could not find iface of type " + interfaceType);
862 return;
863 }
864
865 int result = (enable ? tether(chosenIface) : untether(chosenIface));
866 if (result != ConnectivityManager.TETHER_ERROR_NO_ERROR) {
867 Log.e(TAG, "unable start or stop tethering on iface " + chosenIface);
868 return;
869 }
Robert Greenwalt65ae29b2010-02-18 11:25:54 -0800870 }
871
Robert Greenwaltb4453622011-11-03 16:01:40 -0700872 // TODO - return copies so people can't tamper
Robert Greenwalt2a091d72010-02-11 18:18:40 -0800873 public String[] getTetherableUsbRegexs() {
874 return mTetherableUsbRegexs;
Robert Greenwaltd0e18ff2010-01-26 11:40:34 -0800875 }
876
Robert Greenwalt2a091d72010-02-11 18:18:40 -0800877 public String[] getTetherableWifiRegexs() {
878 return mTetherableWifiRegexs;
879 }
880
Danica Chang6fdd0c62010-08-11 14:54:43 -0700881 public String[] getTetherableBluetoothRegexs() {
882 return mTetherableBluetoothRegexs;
883 }
884
Mike Lockwood6c2260b2011-07-19 13:04:47 -0700885 public int setUsbTethering(boolean enable) {
Wink Savillec9acde92011-09-21 11:05:43 -0700886 if (VDBG) Log.d(TAG, "setUsbTethering(" + enable + ")");
Mike Lockwood6c2260b2011-07-19 13:04:47 -0700887 UsbManager usbManager = (UsbManager)mContext.getSystemService(Context.USB_SERVICE);
888
Robert Greenwaltb4453622011-11-03 16:01:40 -0700889 synchronized (mPublicSync) {
Mike Lockwood6c2260b2011-07-19 13:04:47 -0700890 if (enable) {
891 if (mRndisEnabled) {
Jeremy Klein36c7aa02016-01-22 14:11:45 -0800892 final long ident = Binder.clearCallingIdentity();
893 try {
Christopher Wiley5c0b10a2016-05-31 14:43:08 -0700894 tetherMatchingInterfaces(true, ConnectivityManager.TETHERING_USB);
Jeremy Klein36c7aa02016-01-22 14:11:45 -0800895 } finally {
896 Binder.restoreCallingIdentity(ident);
897 }
Mike Lockwood6c2260b2011-07-19 13:04:47 -0700898 } else {
899 mUsbTetherRequested = true;
Nick Kralevichfcf10f72015-05-13 11:54:03 -0700900 usbManager.setCurrentFunction(UsbManager.USB_FUNCTION_RNDIS);
Mike Lockwood6c2260b2011-07-19 13:04:47 -0700901 }
902 } else {
Jeremy Klein36c7aa02016-01-22 14:11:45 -0800903 final long ident = Binder.clearCallingIdentity();
904 try {
Christopher Wiley5c0b10a2016-05-31 14:43:08 -0700905 tetherMatchingInterfaces(false, ConnectivityManager.TETHERING_USB);
Jeremy Klein36c7aa02016-01-22 14:11:45 -0800906 } finally {
907 Binder.restoreCallingIdentity(ident);
908 }
Mike Lockwood6c2260b2011-07-19 13:04:47 -0700909 if (mRndisEnabled) {
Nick Kralevichfcf10f72015-05-13 11:54:03 -0700910 usbManager.setCurrentFunction(null);
Mike Lockwood6c2260b2011-07-19 13:04:47 -0700911 }
912 mUsbTetherRequested = false;
913 }
914 }
915 return ConnectivityManager.TETHER_ERROR_NO_ERROR;
916 }
917
Robert Greenwaltccf83af12011-06-02 17:30:47 -0700918 public int[] getUpstreamIfaceTypes() {
Robert Greenwaltb4453622011-11-03 16:01:40 -0700919 int values[];
920 synchronized (mPublicSync) {
Robert Greenwaltc13368b2013-07-18 14:24:42 -0700921 updateConfiguration(); // TODO - remove?
Robert Greenwaltb4453622011-11-03 16:01:40 -0700922 values = new int[mUpstreamIfaceTypes.size()];
923 Iterator<Integer> iterator = mUpstreamIfaceTypes.iterator();
924 for (int i=0; i < mUpstreamIfaceTypes.size(); i++) {
925 values[i] = iterator.next();
926 }
Robert Greenwaltccf83af12011-06-02 17:30:47 -0700927 }
928 return values;
Robert Greenwaltc9d5fb72010-02-25 12:29:30 -0800929 }
930
Erik Kline0c9cb2b2015-11-20 17:47:08 +0900931 private void checkDunRequired() {
Junda Liu38531582014-10-28 13:52:51 -0700932 int secureSetting = 2;
933 TelephonyManager tm = (TelephonyManager) mContext.getSystemService(Context.TELEPHONY_SERVICE);
934 if (tm != null) {
935 secureSetting = tm.getTetherApnRequired();
936 }
Robert Greenwaltb4453622011-11-03 16:01:40 -0700937 synchronized (mPublicSync) {
938 // 2 = not set, 0 = DUN not required, 1 = DUN required
939 if (secureSetting != 2) {
940 int requiredApn = (secureSetting == 1 ?
941 ConnectivityManager.TYPE_MOBILE_DUN :
942 ConnectivityManager.TYPE_MOBILE_HIPRI);
943 if (requiredApn == ConnectivityManager.TYPE_MOBILE_DUN) {
944 while (mUpstreamIfaceTypes.contains(MOBILE_TYPE)) {
945 mUpstreamIfaceTypes.remove(MOBILE_TYPE);
946 }
947 while (mUpstreamIfaceTypes.contains(HIPRI_TYPE)) {
948 mUpstreamIfaceTypes.remove(HIPRI_TYPE);
949 }
950 if (mUpstreamIfaceTypes.contains(DUN_TYPE) == false) {
951 mUpstreamIfaceTypes.add(DUN_TYPE);
952 }
953 } else {
954 while (mUpstreamIfaceTypes.contains(DUN_TYPE)) {
955 mUpstreamIfaceTypes.remove(DUN_TYPE);
956 }
957 if (mUpstreamIfaceTypes.contains(MOBILE_TYPE) == false) {
958 mUpstreamIfaceTypes.add(MOBILE_TYPE);
959 }
960 if (mUpstreamIfaceTypes.contains(HIPRI_TYPE) == false) {
961 mUpstreamIfaceTypes.add(HIPRI_TYPE);
962 }
Robert Greenwaltccf83af12011-06-02 17:30:47 -0700963 }
964 }
Robert Greenwaltb4453622011-11-03 16:01:40 -0700965 if (mUpstreamIfaceTypes.contains(DUN_TYPE)) {
966 mPreferredUpstreamMobileApn = ConnectivityManager.TYPE_MOBILE_DUN;
967 } else {
968 mPreferredUpstreamMobileApn = ConnectivityManager.TYPE_MOBILE_HIPRI;
969 }
Robert Greenwaltccf83af12011-06-02 17:30:47 -0700970 }
Robert Greenwaltc9d5fb72010-02-25 12:29:30 -0800971 }
972
Robert Greenwaltb4453622011-11-03 16:01:40 -0700973 // TODO review API - maybe return ArrayList<String> here and below?
Robert Greenwalt2a091d72010-02-11 18:18:40 -0800974 public String[] getTetheredIfaces() {
975 ArrayList<String> list = new ArrayList<String>();
Robert Greenwaltb4453622011-11-03 16:01:40 -0700976 synchronized (mPublicSync) {
Christopher Wileyd985dde2016-05-31 10:44:35 -0700977 for (int i = 0; i < mTetherStates.size(); i++) {
978 TetherState tetherState = mTetherStates.valueAt(i);
979 if (tetherState.mLastState == IControlsTethering.STATE_TETHERED) {
980 list.add(mTetherStates.keyAt(i));
Robert Greenwalt2a091d72010-02-11 18:18:40 -0800981 }
982 }
Robert Greenwaltd0e18ff2010-01-26 11:40:34 -0800983 }
Christopher Wileyd985dde2016-05-31 10:44:35 -0700984 return list.toArray(new String[list.size()]);
Robert Greenwalt2a091d72010-02-11 18:18:40 -0800985 }
986
987 public String[] getTetherableIfaces() {
988 ArrayList<String> list = new ArrayList<String>();
Robert Greenwaltb4453622011-11-03 16:01:40 -0700989 synchronized (mPublicSync) {
Christopher Wileyd985dde2016-05-31 10:44:35 -0700990 for (int i = 0; i < mTetherStates.size(); i++) {
991 TetherState tetherState = mTetherStates.valueAt(i);
992 if (tetherState.mLastState == IControlsTethering.STATE_AVAILABLE) {
993 list.add(mTetherStates.keyAt(i));
Robert Greenwalt2a091d72010-02-11 18:18:40 -0800994 }
995 }
996 }
Christopher Wileyd985dde2016-05-31 10:44:35 -0700997 return list.toArray(new String[list.size()]);
Robert Greenwalt2a091d72010-02-11 18:18:40 -0800998 }
999
Robert Greenwalt9c7e2c22014-06-23 14:53:42 -07001000 public String[] getTetheredDhcpRanges() {
1001 return mDhcpRange;
1002 }
1003
Robert Greenwalt5a735062010-03-02 17:25:02 -08001004 public String[] getErroredIfaces() {
1005 ArrayList<String> list = new ArrayList<String>();
Robert Greenwaltb4453622011-11-03 16:01:40 -07001006 synchronized (mPublicSync) {
Christopher Wileyd985dde2016-05-31 10:44:35 -07001007 for (int i = 0; i < mTetherStates.size(); i++) {
1008 TetherState tetherState = mTetherStates.valueAt(i);
1009 if (tetherState.mLastError != ConnectivityManager.TETHER_ERROR_NO_ERROR) {
1010 list.add(mTetherStates.keyAt(i));
Robert Greenwalt5a735062010-03-02 17:25:02 -08001011 }
1012 }
1013 }
Christopher Wileyd985dde2016-05-31 10:44:35 -07001014 return list.toArray(new String[list.size()]);
Robert Greenwalt5a735062010-03-02 17:25:02 -08001015 }
1016
Lorenzo Colitticd63d242016-04-10 15:39:53 +09001017 private void maybeLogMessage(State state, int what) {
1018 if (DBG) {
1019 Log.d(TAG, state.getName() + " got " +
1020 sMagicDecoderRing.get(what, Integer.toString(what)));
1021 }
1022 }
1023
Christopher Wileye03fb442016-05-18 13:45:20 -07001024 /**
Erik Kline6ff17f72015-12-10 20:42:12 +09001025 * A NetworkCallback class that relays information of interest to the
1026 * tethering master state machine thread for subsequent processing.
1027 */
1028 class UpstreamNetworkCallback extends NetworkCallback {
1029 @Override
Erik Kline00019f42016-06-30 19:31:46 +09001030 public void onAvailable(Network network) {
1031 mTetherMasterSM.sendMessage(TetherMasterSM.EVENT_UPSTREAM_CALLBACK,
1032 UpstreamNetworkMonitor.EVENT_ON_AVAILABLE, 0, network);
1033 }
1034
1035 @Override
1036 public void onCapabilitiesChanged(Network network, NetworkCapabilities newNc) {
1037 mTetherMasterSM.sendMessage(TetherMasterSM.EVENT_UPSTREAM_CALLBACK,
1038 UpstreamNetworkMonitor.EVENT_ON_CAPABILITIES, 0,
1039 new NetworkState(null, null, newNc, network, null, null));
1040 }
1041
1042 @Override
Erik Kline6ff17f72015-12-10 20:42:12 +09001043 public void onLinkPropertiesChanged(Network network, LinkProperties newLp) {
Erik Kline00019f42016-06-30 19:31:46 +09001044 mTetherMasterSM.sendMessage(TetherMasterSM.EVENT_UPSTREAM_CALLBACK,
1045 UpstreamNetworkMonitor.EVENT_ON_LINKPROPERTIES, 0,
Erik Kline6ff17f72015-12-10 20:42:12 +09001046 new NetworkState(null, newLp, null, network, null, null));
1047 }
1048
1049 @Override
1050 public void onLost(Network network) {
Erik Kline00019f42016-06-30 19:31:46 +09001051 mTetherMasterSM.sendMessage(TetherMasterSM.EVENT_UPSTREAM_CALLBACK,
1052 UpstreamNetworkMonitor.EVENT_ON_LOST, 0, network);
Erik Kline6ff17f72015-12-10 20:42:12 +09001053 }
1054 }
1055
1056 /**
1057 * A class to centralize all the network and link properties information
1058 * pertaining to the current and any potential upstream network.
1059 *
1060 * Calling #start() registers two callbacks: one to track the system default
1061 * network and a second to specifically observe TYPE_MOBILE_DUN networks.
1062 *
1063 * The methods and data members of this class are only to be accessed and
1064 * modified from the tethering master state machine thread. Any other
1065 * access semantics would necessitate the addition of locking.
1066 *
1067 * TODO: Investigate whether more "upstream-specific" logic/functionality
1068 * could/should be moved here.
1069 */
1070 class UpstreamNetworkMonitor {
Erik Kline00019f42016-06-30 19:31:46 +09001071 static final int EVENT_ON_AVAILABLE = 1;
1072 static final int EVENT_ON_CAPABILITIES = 2;
1073 static final int EVENT_ON_LINKPROPERTIES = 3;
1074 static final int EVENT_ON_LOST = 4;
1075
Christopher Wiley499a57a2016-05-16 16:19:07 -07001076 final HashMap<Network, NetworkState> mNetworkMap = new HashMap<>();
Erik Kline6ff17f72015-12-10 20:42:12 +09001077 NetworkCallback mDefaultNetworkCallback;
1078 NetworkCallback mDunTetheringCallback;
1079
1080 void start() {
1081 stop();
1082
1083 mDefaultNetworkCallback = new UpstreamNetworkCallback();
1084 getConnectivityManager().registerDefaultNetworkCallback(mDefaultNetworkCallback);
1085
1086 final NetworkRequest dunTetheringRequest = new NetworkRequest.Builder()
1087 .addTransportType(NetworkCapabilities.TRANSPORT_CELLULAR)
1088 .removeCapability(NetworkCapabilities.NET_CAPABILITY_NOT_RESTRICTED)
1089 .addCapability(NetworkCapabilities.NET_CAPABILITY_DUN)
1090 .build();
1091 mDunTetheringCallback = new UpstreamNetworkCallback();
1092 getConnectivityManager().registerNetworkCallback(
1093 dunTetheringRequest, mDunTetheringCallback);
1094 }
1095
1096 void stop() {
1097 if (mDefaultNetworkCallback != null) {
1098 getConnectivityManager().unregisterNetworkCallback(mDefaultNetworkCallback);
1099 mDefaultNetworkCallback = null;
1100 }
1101
1102 if (mDunTetheringCallback != null) {
1103 getConnectivityManager().unregisterNetworkCallback(mDunTetheringCallback);
1104 mDunTetheringCallback = null;
1105 }
1106
1107 mNetworkMap.clear();
1108 }
1109
Erik Kline00019f42016-06-30 19:31:46 +09001110 NetworkState lookup(Network network) {
1111 return (network != null) ? mNetworkMap.get(network) : null;
1112 }
Erik Kline6ff17f72015-12-10 20:42:12 +09001113
Erik Kline00019f42016-06-30 19:31:46 +09001114 NetworkState processCallback(int arg1, Object obj) {
1115 switch (arg1) {
1116 case EVENT_ON_AVAILABLE: {
1117 final Network network = (Network) obj;
1118 if (VDBG) {
1119 Log.d(TAG, "EVENT_ON_AVAILABLE for " + network);
Erik Kline6ff17f72015-12-10 20:42:12 +09001120 }
Erik Kline00019f42016-06-30 19:31:46 +09001121 if (!mNetworkMap.containsKey(network)) {
1122 mNetworkMap.put(network,
1123 new NetworkState(null, null, null, network, null, null));
1124 }
1125
1126 final ConnectivityManager cm = getConnectivityManager();
1127
1128 if (mDefaultNetworkCallback != null) {
1129 cm.requestNetworkCapabilities(mDefaultNetworkCallback);
1130 cm.requestLinkProperties(mDefaultNetworkCallback);
1131 }
1132
1133 // Requesting updates for mDunTetheringCallback is not
1134 // necessary. Because it's a listen, it will already have
1135 // heard all NetworkCapabilities and LinkProperties updates
1136 // since UpstreamNetworkMonitor was started. Because we
1137 // start UpstreamNetworkMonitor before chooseUpstreamType()
1138 // is ever invoked (it can register a DUN request) this is
1139 // mostly safe. However, if a DUN network is already up for
1140 // some reason (unlikely, because DUN is restricted and,
1141 // unless the DUN network is shared with another APN, only
1142 // the system can request it and this is the only part of
1143 // the system that requests it) we won't know its
1144 // LinkProperties or NetworkCapabilities.
1145
1146 return mNetworkMap.get(network);
1147 }
1148 case EVENT_ON_CAPABILITIES: {
1149 final NetworkState ns = (NetworkState) obj;
1150 if (!mNetworkMap.containsKey(ns.network)) {
1151 // Ignore updates for networks for which we have not yet
1152 // received onAvailable() - which should never happen -
1153 // or for which we have already received onLost().
1154 return null;
1155 }
1156 if (VDBG) {
1157 Log.d(TAG, String.format("EVENT_ON_CAPABILITIES for %s: %s",
1158 ns.network, ns.networkCapabilities));
1159 }
1160
1161 final NetworkState prev = mNetworkMap.get(ns.network);
1162 mNetworkMap.put(ns.network,
1163 new NetworkState(null, prev.linkProperties, ns.networkCapabilities,
1164 ns.network, null, null));
1165 return mNetworkMap.get(ns.network);
1166 }
1167 case EVENT_ON_LINKPROPERTIES: {
1168 final NetworkState ns = (NetworkState) obj;
1169 if (!mNetworkMap.containsKey(ns.network)) {
1170 // Ignore updates for networks for which we have not yet
1171 // received onAvailable() - which should never happen -
1172 // or for which we have already received onLost().
1173 return null;
1174 }
1175 if (VDBG) {
1176 Log.d(TAG, String.format("EVENT_ON_LINKPROPERTIES for %s: %s",
1177 ns.network, ns.linkProperties));
1178 }
1179
1180 final NetworkState prev = mNetworkMap.get(ns.network);
1181 mNetworkMap.put(ns.network,
1182 new NetworkState(null, ns.linkProperties, prev.networkCapabilities,
1183 ns.network, null, null));
1184 return mNetworkMap.get(ns.network);
1185 }
1186 case EVENT_ON_LOST: {
1187 final Network network = (Network) obj;
1188 if (VDBG) {
1189 Log.d(TAG, "EVENT_ON_LOST for " + network);
1190 }
1191 return mNetworkMap.remove(network);
1192 }
1193 default:
1194 return null;
1195 }
1196 }
1197 }
1198
1199 // Needed because the canonical source of upstream truth is just the
1200 // upstream interface name, |mCurrentUpstreamIface|. This is ripe for
1201 // future simplification, once the upstream Network is canonical.
1202 boolean pertainsToCurrentUpstream(NetworkState ns) {
1203 if (ns != null && ns.linkProperties != null && mCurrentUpstreamIface != null) {
1204 for (String ifname : ns.linkProperties.getAllInterfaceNames()) {
1205 if (mCurrentUpstreamIface.equals(ifname)) {
1206 return true;
Erik Kline6ff17f72015-12-10 20:42:12 +09001207 }
1208 }
Erik Kline6ff17f72015-12-10 20:42:12 +09001209 }
Erik Kline00019f42016-06-30 19:31:46 +09001210 return false;
Erik Kline6ff17f72015-12-10 20:42:12 +09001211 }
1212
Wink Saville64c42ca2011-04-18 14:55:10 -07001213 class TetherMasterSM extends StateMachine {
Lorenzo Colitticd63d242016-04-10 15:39:53 +09001214 private static final int BASE_MASTER = Protocol.BASE_TETHERING;
Robert Greenwalt2a091d72010-02-11 18:18:40 -08001215 // an interface SM has requested Tethering
Lorenzo Colitticd63d242016-04-10 15:39:53 +09001216 static final int CMD_TETHER_MODE_REQUESTED = BASE_MASTER + 1;
Robert Greenwalt2a091d72010-02-11 18:18:40 -08001217 // an interface SM has unrequested Tethering
Lorenzo Colitticd63d242016-04-10 15:39:53 +09001218 static final int CMD_TETHER_MODE_UNREQUESTED = BASE_MASTER + 2;
Robert Greenwalt6a1967c2010-03-17 11:19:57 -07001219 // upstream connection change - do the right thing
Lorenzo Colitticd63d242016-04-10 15:39:53 +09001220 static final int CMD_UPSTREAM_CHANGED = BASE_MASTER + 3;
Robert Greenwalt6a1967c2010-03-17 11:19:57 -07001221 // we don't have a valid upstream conn, check again after a delay
Lorenzo Colitticd63d242016-04-10 15:39:53 +09001222 static final int CMD_RETRY_UPSTREAM = BASE_MASTER + 4;
Erik Kline6ff17f72015-12-10 20:42:12 +09001223 // Events from NetworkCallbacks that we process on the master state
1224 // machine thread on behalf of the UpstreamNetworkMonitor.
Erik Kline00019f42016-06-30 19:31:46 +09001225 static final int EVENT_UPSTREAM_CALLBACK = BASE_MASTER + 5;
Robert Greenwalt2a091d72010-02-11 18:18:40 -08001226
Wink Saville64c42ca2011-04-18 14:55:10 -07001227 private State mInitialState;
1228 private State mTetherModeAliveState;
Robert Greenwalt2a091d72010-02-11 18:18:40 -08001229
Wink Saville64c42ca2011-04-18 14:55:10 -07001230 private State mSetIpForwardingEnabledErrorState;
1231 private State mSetIpForwardingDisabledErrorState;
1232 private State mStartTetheringErrorState;
1233 private State mStopTetheringErrorState;
1234 private State mSetDnsForwardersErrorState;
Robert Greenwalt2a091d72010-02-11 18:18:40 -08001235
Christopher Wileyd985dde2016-05-31 10:44:35 -07001236 // This list is a little subtle. It contains all the interfaces that currently are
1237 // requesting tethering, regardless of whether these interfaces are still members of
1238 // mTetherStates. This allows us to maintain the following predicates:
1239 //
1240 // 1) mTetherStates contains the set of all currently existing, tetherable, link state up
1241 // interfaces.
1242 // 2) mNotifyList contains all state machines that may have outstanding tethering state
1243 // that needs to be torn down.
1244 //
1245 // Because we excise interfaces immediately from mTetherStates, we must maintain mNotifyList
1246 // so that the garbage collector does not clean up the state machine before it has a chance
1247 // to tear itself down.
Erik Kline1eb8c692016-07-08 17:21:26 +09001248 private final ArrayList<TetherInterfaceStateMachine> mNotifyList;
1249 private final IPv6TetheringCoordinator mIPv6TetheringCoordinator;
Robert Greenwalt2a091d72010-02-11 18:18:40 -08001250
Robert Greenwaltccf83af12011-06-02 17:30:47 -07001251 private int mMobileApnReserved = ConnectivityManager.TYPE_NONE;
Erik Kline6ff17f72015-12-10 20:42:12 +09001252 private NetworkCallback mMobileUpstreamCallback;
Robert Greenwaltb37f0c62010-03-25 14:50:34 -07001253
Robert Greenwalt6a1967c2010-03-17 11:19:57 -07001254 private static final int UPSTREAM_SETTLE_TIME_MS = 10000;
Robert Greenwalt2a091d72010-02-11 18:18:40 -08001255
Robert Greenwaltdfadaea2010-03-11 15:03:08 -08001256 TetherMasterSM(String name, Looper looper) {
1257 super(name, looper);
Robert Greenwalt2a091d72010-02-11 18:18:40 -08001258
1259 //Add states
1260 mInitialState = new InitialState();
1261 addState(mInitialState);
Robert Greenwalt2a091d72010-02-11 18:18:40 -08001262 mTetherModeAliveState = new TetherModeAliveState();
1263 addState(mTetherModeAliveState);
Robert Greenwalt2a091d72010-02-11 18:18:40 -08001264
Robert Greenwalt2a091d72010-02-11 18:18:40 -08001265 mSetIpForwardingEnabledErrorState = new SetIpForwardingEnabledErrorState();
1266 addState(mSetIpForwardingEnabledErrorState);
1267 mSetIpForwardingDisabledErrorState = new SetIpForwardingDisabledErrorState();
1268 addState(mSetIpForwardingDisabledErrorState);
1269 mStartTetheringErrorState = new StartTetheringErrorState();
1270 addState(mStartTetheringErrorState);
1271 mStopTetheringErrorState = new StopTetheringErrorState();
1272 addState(mStopTetheringErrorState);
1273 mSetDnsForwardersErrorState = new SetDnsForwardersErrorState();
1274 addState(mSetDnsForwardersErrorState);
1275
Mitchell Wills7040b4e2016-05-23 16:40:10 -07001276 mNotifyList = new ArrayList<>();
Erik Kline1eb8c692016-07-08 17:21:26 +09001277 mIPv6TetheringCoordinator = new IPv6TetheringCoordinator(mNotifyList);
Robert Greenwalt2a091d72010-02-11 18:18:40 -08001278 setInitialState(mInitialState);
1279 }
1280
Wink Saville64c42ca2011-04-18 14:55:10 -07001281 class TetherMasterUtilState extends State {
Robert Greenwaltd70a3d42010-02-23 18:15:29 -08001282 @Override
1283 public boolean processMessage(Message m) {
1284 return false;
1285 }
Erik Kline0c9cb2b2015-11-20 17:47:08 +09001286
Robert Greenwaltccf83af12011-06-02 17:30:47 -07001287 protected boolean turnOnUpstreamMobileConnection(int apnType) {
Erik Kline0c9cb2b2015-11-20 17:47:08 +09001288 if (apnType == ConnectivityManager.TYPE_NONE) { return false; }
1289
1290 if (apnType != mMobileApnReserved) {
1291 // Unregister any previous mobile upstream callback because
1292 // this request, if any, will be different.
1293 turnOffUpstreamMobileConnection();
Robert Greenwalt6a1967c2010-03-17 11:19:57 -07001294 }
1295
Erik Kline0c9cb2b2015-11-20 17:47:08 +09001296 if (mMobileUpstreamCallback != null) {
1297 // Looks like we already filed a request for this apnType.
1298 return true;
Robert Greenwaltd70a3d42010-02-23 18:15:29 -08001299 }
Erik Kline0c9cb2b2015-11-20 17:47:08 +09001300
1301 switch (apnType) {
1302 case ConnectivityManager.TYPE_MOBILE_DUN:
1303 case ConnectivityManager.TYPE_MOBILE:
1304 case ConnectivityManager.TYPE_MOBILE_HIPRI:
1305 mMobileApnReserved = apnType;
1306 break;
1307 default:
1308 return false;
1309 }
1310
Erik Kline6ff17f72015-12-10 20:42:12 +09001311 final NetworkRequest.Builder builder = new NetworkRequest.Builder()
Erik Kline0c9cb2b2015-11-20 17:47:08 +09001312 .addTransportType(NetworkCapabilities.TRANSPORT_CELLULAR);
1313 if (apnType == ConnectivityManager.TYPE_MOBILE_DUN) {
Erik Kline6ff17f72015-12-10 20:42:12 +09001314 builder.removeCapability(NetworkCapabilities.NET_CAPABILITY_NOT_RESTRICTED)
1315 .addCapability(NetworkCapabilities.NET_CAPABILITY_DUN);
Erik Kline0c9cb2b2015-11-20 17:47:08 +09001316 } else {
1317 builder.addCapability(NetworkCapabilities.NET_CAPABILITY_INTERNET);
1318 }
Erik Kline6ff17f72015-12-10 20:42:12 +09001319 final NetworkRequest mobileUpstreamRequest = builder.build();
1320
1321 // The UpstreamNetworkMonitor's callback will be notified.
1322 // Therefore, to avoid duplicate notifications, we only register a no-op.
1323 mMobileUpstreamCallback = new NetworkCallback();
1324
Erik Kline0c9cb2b2015-11-20 17:47:08 +09001325 // TODO: Change the timeout from 0 (no onUnavailable callback) to use some
1326 // moderate callback time (once timeout callbacks are implemented). This might
1327 // be useful for updating some UI. Additionally, we should definitely log a
Erik Kline6ff17f72015-12-10 20:42:12 +09001328 // message to aid in any subsequent debugging
Erik Kline0c9cb2b2015-11-20 17:47:08 +09001329 if (DBG) Log.d(TAG, "requesting mobile upstream network: " + mobileUpstreamRequest);
1330 getConnectivityManager().requestNetwork(
1331 mobileUpstreamRequest, mMobileUpstreamCallback, 0, apnType);
Robert Greenwaltd70a3d42010-02-23 18:15:29 -08001332 return true;
1333 }
Erik Kline0c9cb2b2015-11-20 17:47:08 +09001334
1335 protected void turnOffUpstreamMobileConnection() {
1336 if (mMobileUpstreamCallback != null) {
1337 getConnectivityManager().unregisterNetworkCallback(mMobileUpstreamCallback);
1338 mMobileUpstreamCallback = null;
1339 }
1340 mMobileApnReserved = ConnectivityManager.TYPE_NONE;
1341 }
1342
Robert Greenwalt6a1967c2010-03-17 11:19:57 -07001343 protected boolean turnOnMasterTetherSettings() {
Robert Greenwaltd70a3d42010-02-23 18:15:29 -08001344 try {
Chia-chi Yehc9338302011-05-11 16:35:13 -07001345 mNMService.setIpForwardingEnabled(true);
Robert Greenwaltd70a3d42010-02-23 18:15:29 -08001346 } catch (Exception e) {
Wink Saville951749b2010-03-19 09:03:13 -07001347 transitionTo(mSetIpForwardingEnabledErrorState);
Robert Greenwaltd70a3d42010-02-23 18:15:29 -08001348 return false;
1349 }
1350 try {
Chia-chi Yehc9338302011-05-11 16:35:13 -07001351 mNMService.startTethering(mDhcpRange);
Robert Greenwaltd70a3d42010-02-23 18:15:29 -08001352 } catch (Exception e) {
Robert Greenwalt547e4e72011-01-14 17:24:03 -08001353 try {
Chia-chi Yehc9338302011-05-11 16:35:13 -07001354 mNMService.stopTethering();
1355 mNMService.startTethering(mDhcpRange);
Robert Greenwalt547e4e72011-01-14 17:24:03 -08001356 } catch (Exception ee) {
1357 transitionTo(mStartTetheringErrorState);
1358 return false;
1359 }
Robert Greenwaltd70a3d42010-02-23 18:15:29 -08001360 }
Robert Greenwaltd70a3d42010-02-23 18:15:29 -08001361 return true;
1362 }
Erik Kline6ff17f72015-12-10 20:42:12 +09001363
Robert Greenwalt6a1967c2010-03-17 11:19:57 -07001364 protected boolean turnOffMasterTetherSettings() {
Robert Greenwaltd70a3d42010-02-23 18:15:29 -08001365 try {
Chia-chi Yehc9338302011-05-11 16:35:13 -07001366 mNMService.stopTethering();
Robert Greenwaltd70a3d42010-02-23 18:15:29 -08001367 } catch (Exception e) {
1368 transitionTo(mStopTetheringErrorState);
1369 return false;
1370 }
1371 try {
Chia-chi Yehc9338302011-05-11 16:35:13 -07001372 mNMService.setIpForwardingEnabled(false);
Robert Greenwaltd70a3d42010-02-23 18:15:29 -08001373 } catch (Exception e) {
1374 transitionTo(mSetIpForwardingDisabledErrorState);
1375 return false;
1376 }
1377 transitionTo(mInitialState);
1378 return true;
1379 }
Robert Greenwalt10398722010-12-17 15:20:36 -08001380
Robert Greenwalt6a1967c2010-03-17 11:19:57 -07001381 protected void chooseUpstreamType(boolean tryCell) {
Erik Kline00019f42016-06-30 19:31:46 +09001382 final ConnectivityManager cm = getConnectivityManager();
Robert Greenwaltccf83af12011-06-02 17:30:47 -07001383 int upType = ConnectivityManager.TYPE_NONE;
1384 String iface = null;
1385
Robert Greenwaltc13368b2013-07-18 14:24:42 -07001386 updateConfiguration(); // TODO - remove?
Robert Greenwalt49348e72011-10-21 16:54:26 -07001387
Robert Greenwaltb4453622011-11-03 16:01:40 -07001388 synchronized (mPublicSync) {
1389 if (VDBG) {
1390 Log.d(TAG, "chooseUpstreamType has upstream iface types:");
1391 for (Integer netType : mUpstreamIfaceTypes) {
1392 Log.d(TAG, " " + netType);
1393 }
Robert Greenwalt49348e72011-10-21 16:54:26 -07001394 }
Robert Greenwalt49348e72011-10-21 16:54:26 -07001395
Robert Greenwaltb4453622011-11-03 16:01:40 -07001396 for (Integer netType : mUpstreamIfaceTypes) {
Erik Kline00019f42016-06-30 19:31:46 +09001397 NetworkInfo info = cm.getNetworkInfo(netType.intValue());
Erik Klinefa37b2f2016-08-02 18:27:03 +09001398 // TODO: if the network is suspended we should consider
1399 // that to be the same as connected here.
Robert Greenwaltb4453622011-11-03 16:01:40 -07001400 if ((info != null) && info.isConnected()) {
1401 upType = netType.intValue();
1402 break;
1403 }
Robert Greenwalt6a1967c2010-03-17 11:19:57 -07001404 }
1405 }
Robert Greenwaltccf83af12011-06-02 17:30:47 -07001406
Robert Greenwaltfd1be2b2011-11-11 12:30:19 -08001407 if (DBG) {
Erik Kline0c9cb2b2015-11-20 17:47:08 +09001408 Log.d(TAG, "chooseUpstreamType(" + tryCell + "),"
1409 + " preferredApn="
1410 + ConnectivityManager.getNetworkTypeName(mPreferredUpstreamMobileApn)
1411 + ", got type="
1412 + ConnectivityManager.getNetworkTypeName(upType));
Robert Greenwaltccf83af12011-06-02 17:30:47 -07001413 }
1414
Erik Kline0c9cb2b2015-11-20 17:47:08 +09001415 switch (upType) {
1416 case ConnectivityManager.TYPE_MOBILE_DUN:
1417 case ConnectivityManager.TYPE_MOBILE_HIPRI:
1418 // If we're on DUN, put our own grab on it.
1419 turnOnUpstreamMobileConnection(upType);
1420 break;
1421 case ConnectivityManager.TYPE_NONE:
1422 if (tryCell &&
1423 turnOnUpstreamMobileConnection(mPreferredUpstreamMobileApn)) {
1424 // We think mobile should be coming up; don't set a retry.
1425 } else {
1426 sendMessageDelayed(CMD_RETRY_UPSTREAM, UPSTREAM_SETTLE_TIME_MS);
1427 }
1428 break;
1429 default:
1430 /* If we've found an active upstream connection that's not DUN/HIPRI
1431 * we should stop any outstanding DUN/HIPRI start requests.
1432 *
1433 * If we found NONE we don't want to do this as we want any previous
1434 * requests to keep trying to bring up something we can use.
1435 */
1436 turnOffUpstreamMobileConnection();
1437 break;
Robert Greenwaltccf83af12011-06-02 17:30:47 -07001438 }
1439
Erik Kline00019f42016-06-30 19:31:46 +09001440 Network network = null;
Erik Kline0c9cb2b2015-11-20 17:47:08 +09001441 if (upType != ConnectivityManager.TYPE_NONE) {
Erik Kline00019f42016-06-30 19:31:46 +09001442 LinkProperties linkProperties = cm.getLinkProperties(upType);
Robert Greenwaltbb4fd0b2011-12-09 15:36:50 -08001443 if (linkProperties != null) {
Lorenzo Colitti13c9fde2013-03-15 04:22:37 +09001444 // Find the interface with the default IPv4 route. It may be the
1445 // interface described by linkProperties, or one of the interfaces
1446 // stacked on top of it.
1447 Log.i(TAG, "Finding IPv4 upstream interface on: " + linkProperties);
1448 RouteInfo ipv4Default = RouteInfo.selectBestRoute(
1449 linkProperties.getAllRoutes(), Inet4Address.ANY);
1450 if (ipv4Default != null) {
1451 iface = ipv4Default.getInterface();
1452 Log.i(TAG, "Found interface " + ipv4Default.getInterface());
1453 } else {
1454 Log.i(TAG, "No IPv4 upstream interface, giving up.");
1455 }
1456 }
1457
1458 if (iface != null) {
Erik Kline00019f42016-06-30 19:31:46 +09001459 network = cm.getNetworkForType(upType);
Erik Kline6ff17f72015-12-10 20:42:12 +09001460 if (network == null) {
1461 Log.e(TAG, "No Network for upstream type " + upType + "!");
Robert Greenwaltbb4fd0b2011-12-09 15:36:50 -08001462 }
Erik Kline6ff17f72015-12-10 20:42:12 +09001463 setDnsForwarders(network, linkProperties);
Robert Greenwaltbb4fd0b2011-12-09 15:36:50 -08001464 }
Robert Greenwalt6a1967c2010-03-17 11:19:57 -07001465 }
Robert Greenwaltb37f0c62010-03-25 14:50:34 -07001466 notifyTetheredOfNewUpstreamIface(iface);
Erik Kline00019f42016-06-30 19:31:46 +09001467 NetworkState ns = mUpstreamNetworkMonitor.lookup(network);
1468 if (ns != null && pertainsToCurrentUpstream(ns)) {
1469 // If we already have NetworkState for this network examine
1470 // it immediately, because there likely will be no second
1471 // EVENT_ON_AVAILABLE (it was already received).
1472 handleNewUpstreamNetworkState(ns);
Erik Klinefa37b2f2016-08-02 18:27:03 +09001473 } else if (mCurrentUpstreamIface == null) {
1474 // There are no available upstream networks, or none that
1475 // have an IPv4 default route (current metric for success).
1476 handleNewUpstreamNetworkState(null);
Erik Kline00019f42016-06-30 19:31:46 +09001477 }
Robert Greenwalt6a1967c2010-03-17 11:19:57 -07001478 }
Robert Greenwaltccf83af12011-06-02 17:30:47 -07001479
Erik Kline6ff17f72015-12-10 20:42:12 +09001480 protected void setDnsForwarders(final Network network, final LinkProperties lp) {
1481 String[] dnsServers = mDefaultDnsServers;
1482 final Collection<InetAddress> dnses = lp.getDnsServers();
1483 // TODO: Properly support the absence of DNS servers.
1484 if (dnses != null && !dnses.isEmpty()) {
1485 // TODO: remove this invocation of NetworkUtils.makeStrings().
1486 dnsServers = NetworkUtils.makeStrings(dnses);
1487 }
1488 if (VDBG) {
1489 Log.d(TAG, "Setting DNS forwarders: Network=" + network +
1490 ", dnsServers=" + Arrays.toString(dnsServers));
1491 }
1492 try {
1493 mNMService.setDnsForwarders(network, dnsServers);
1494 } catch (Exception e) {
1495 // TODO: Investigate how this can fail and what exactly
1496 // happens if/when such failures occur.
1497 Log.e(TAG, "Setting DNS forwarders failed!");
1498 transitionTo(mSetDnsForwardersErrorState);
1499 }
1500 }
1501
Robert Greenwaltb37f0c62010-03-25 14:50:34 -07001502 protected void notifyTetheredOfNewUpstreamIface(String ifaceName) {
Lorenzo Colitticd63d242016-04-10 15:39:53 +09001503 if (DBG) Log.d(TAG, "Notifying tethered with upstream=" + ifaceName);
Erik Kline6ff17f72015-12-10 20:42:12 +09001504 mCurrentUpstreamIface = ifaceName;
Mitchell Wills7040b4e2016-05-23 16:40:10 -07001505 for (TetherInterfaceStateMachine sm : mNotifyList) {
1506 sm.sendMessage(TetherInterfaceStateMachine.CMD_TETHER_CONNECTION_CHANGED,
Robert Greenwalt6a1967c2010-03-17 11:19:57 -07001507 ifaceName);
1508 }
1509 }
Erik Kline00019f42016-06-30 19:31:46 +09001510
1511 protected void handleNewUpstreamNetworkState(NetworkState ns) {
1512 mIPv6TetheringCoordinator.updateUpstreamNetworkState(ns);
1513 }
Robert Greenwaltd70a3d42010-02-23 18:15:29 -08001514 }
Robert Greenwalt2a091d72010-02-11 18:18:40 -08001515
Robert Greenwalt2ffe4122014-12-12 12:22:31 -08001516 private final AtomicInteger mSimBcastGenerationNumber = new AtomicInteger(0);
1517 private SimChangeBroadcastReceiver mBroadcastReceiver = null;
1518
Robert Greenwalt2ffe4122014-12-12 12:22:31 -08001519 private void startListeningForSimChanges() {
1520 if (DBG) Log.d(TAG, "startListeningForSimChanges");
1521 if (mBroadcastReceiver == null) {
1522 mBroadcastReceiver = new SimChangeBroadcastReceiver(
1523 mSimBcastGenerationNumber.incrementAndGet());
1524 final IntentFilter filter = new IntentFilter();
1525 filter.addAction(TelephonyIntents.ACTION_SIM_STATE_CHANGED);
1526
1527 mContext.registerReceiver(mBroadcastReceiver, filter);
1528 }
1529 }
1530
1531 private void stopListeningForSimChanges() {
1532 if (DBG) Log.d(TAG, "stopListeningForSimChanges");
1533 if (mBroadcastReceiver != null) {
1534 mSimBcastGenerationNumber.incrementAndGet();
1535 mContext.unregisterReceiver(mBroadcastReceiver);
1536 mBroadcastReceiver = null;
1537 }
1538 }
1539
1540 class SimChangeBroadcastReceiver extends BroadcastReceiver {
1541 // used to verify this receiver is still current
1542 final private int mGenerationNumber;
1543
1544 // we're interested in edge-triggered LOADED notifications, so
1545 // ignore LOADED unless we saw an ABSENT state first
1546 private boolean mSimAbsentSeen = false;
1547
1548 public SimChangeBroadcastReceiver(int generationNumber) {
1549 super();
1550 mGenerationNumber = generationNumber;
1551 }
1552
1553 @Override
1554 public void onReceive(Context context, Intent intent) {
1555 if (DBG) {
1556 Log.d(TAG, "simchange mGenerationNumber=" + mGenerationNumber +
1557 ", current generationNumber=" + mSimBcastGenerationNumber.get());
1558 }
1559 if (mGenerationNumber != mSimBcastGenerationNumber.get()) return;
1560
1561 final String state =
1562 intent.getStringExtra(IccCardConstants.INTENT_KEY_ICC_STATE);
1563
1564 Log.d(TAG, "got Sim changed to state " + state + ", mSimAbsentSeen=" +
1565 mSimAbsentSeen);
1566 if (!mSimAbsentSeen && IccCardConstants.INTENT_VALUE_ICC_ABSENT.equals(state)) {
1567 mSimAbsentSeen = true;
1568 }
1569
1570 if (mSimAbsentSeen && IccCardConstants.INTENT_VALUE_ICC_LOADED.equals(state)) {
1571 mSimAbsentSeen = false;
1572 try {
1573 if (mContext.getResources().getString(com.android.internal.R.string.
1574 config_mobile_hotspot_provision_app_no_ui).isEmpty() == false) {
Robert Greenwalt2ffe4122014-12-12 12:22:31 -08001575 ArrayList<Integer> tethered = new ArrayList<Integer>();
1576 synchronized (mPublicSync) {
Christopher Wileyd985dde2016-05-31 10:44:35 -07001577 for (int i = 0; i < mTetherStates.size(); i++) {
1578 TetherState tetherState = mTetherStates.valueAt(i);
1579 if (tetherState.mLastState !=
1580 IControlsTethering.STATE_TETHERED) {
1581 continue; // Skip interfaces that aren't tethered.
1582 }
1583 String iface = mTetherStates.keyAt(i);
1584 int interfaceType = ifaceNameToType(iface);
1585 if (interfaceType != ConnectivityManager.TETHERING_INVALID) {
1586 tethered.add(new Integer(interfaceType));
Robert Greenwalt2ffe4122014-12-12 12:22:31 -08001587 }
1588 }
1589 }
1590 for (int tetherType : tethered) {
1591 Intent startProvIntent = new Intent();
Jeremy Klein36c7aa02016-01-22 14:11:45 -08001592 startProvIntent.putExtra(
1593 ConnectivityManager.EXTRA_ADD_TETHER_TYPE, tetherType);
1594 startProvIntent.putExtra(
1595 ConnectivityManager.EXTRA_RUN_PROVISION, true);
1596 startProvIntent.setComponent(TETHER_SERVICE);
Robert Greenwalt2ffe4122014-12-12 12:22:31 -08001597 mContext.startServiceAsUser(startProvIntent, UserHandle.CURRENT);
1598 }
1599 Log.d(TAG, "re-evaluate provisioning");
1600 } else {
1601 Log.d(TAG, "no prov-check needed for new SIM");
1602 }
1603 } catch (Resources.NotFoundException e) {
1604 Log.d(TAG, "no prov-check needed for new SIM");
1605 // not defined, do nothing
1606 }
1607 }
1608 }
1609 }
1610
Robert Greenwaltd70a3d42010-02-23 18:15:29 -08001611 class InitialState extends TetherMasterUtilState {
Robert Greenwalt2a091d72010-02-11 18:18:40 -08001612 @Override
1613 public boolean processMessage(Message message) {
Lorenzo Colitticd63d242016-04-10 15:39:53 +09001614 maybeLogMessage(this, message.what);
Robert Greenwalt2a091d72010-02-11 18:18:40 -08001615 boolean retValue = true;
1616 switch (message.what) {
1617 case CMD_TETHER_MODE_REQUESTED:
Mitchell Wills7040b4e2016-05-23 16:40:10 -07001618 TetherInterfaceStateMachine who = (TetherInterfaceStateMachine)message.obj;
Robert Greenwalt68ea9b02012-05-10 16:58:16 -07001619 if (VDBG) Log.d(TAG, "Tether Mode requested by " + who);
Christopher Wileyd985dde2016-05-31 10:44:35 -07001620 if (mNotifyList.indexOf(who) < 0) {
1621 mNotifyList.add(who);
Erik Kline6e29bf02016-08-15 16:16:18 +09001622 mIPv6TetheringCoordinator.addActiveDownstream(who);
Christopher Wileyd985dde2016-05-31 10:44:35 -07001623 }
Robert Greenwalt6a1967c2010-03-17 11:19:57 -07001624 transitionTo(mTetherModeAliveState);
Robert Greenwalt2a091d72010-02-11 18:18:40 -08001625 break;
1626 case CMD_TETHER_MODE_UNREQUESTED:
Mitchell Wills7040b4e2016-05-23 16:40:10 -07001627 who = (TetherInterfaceStateMachine)message.obj;
Robert Greenwalt68ea9b02012-05-10 16:58:16 -07001628 if (VDBG) Log.d(TAG, "Tether Mode unrequested by " + who);
Christopher Wileyd985dde2016-05-31 10:44:35 -07001629 mNotifyList.remove(who);
Erik Kline6e29bf02016-08-15 16:16:18 +09001630 mIPv6TetheringCoordinator.removeActiveDownstream(who);
Robert Greenwalt2a091d72010-02-11 18:18:40 -08001631 break;
Robert Greenwalt2a091d72010-02-11 18:18:40 -08001632 default:
1633 retValue = false;
1634 break;
1635 }
1636 return retValue;
1637 }
1638 }
1639
Robert Greenwaltd70a3d42010-02-23 18:15:29 -08001640 class TetherModeAliveState extends TetherMasterUtilState {
Erik Kline6ff17f72015-12-10 20:42:12 +09001641 boolean mTryCell = true;
Robert Greenwalt2a091d72010-02-11 18:18:40 -08001642 @Override
1643 public void enter() {
Erik Kline6ff17f72015-12-10 20:42:12 +09001644 // TODO: examine if we should check the return value.
Robert Greenwalt4f74d552011-12-19 16:59:31 -08001645 turnOnMasterTetherSettings(); // may transition us out
Robert Greenwalt2ffe4122014-12-12 12:22:31 -08001646 startListeningForSimChanges();
Erik Kline6ff17f72015-12-10 20:42:12 +09001647 mUpstreamNetworkMonitor.start();
Robert Greenwalt4f74d552011-12-19 16:59:31 -08001648
Erik Kline6ff17f72015-12-10 20:42:12 +09001649 mTryCell = true; // better try something first pass or crazy tests cases will fail
Robert Greenwalt6a1967c2010-03-17 11:19:57 -07001650 chooseUpstreamType(mTryCell);
1651 mTryCell = !mTryCell;
Robert Greenwalt6a1967c2010-03-17 11:19:57 -07001652 }
Erik Klinefa37b2f2016-08-02 18:27:03 +09001653
Robert Greenwalt6a1967c2010-03-17 11:19:57 -07001654 @Override
1655 public void exit() {
Erik Kline6ff17f72015-12-10 20:42:12 +09001656 // TODO: examine if we should check the return value.
Robert Greenwaltccf83af12011-06-02 17:30:47 -07001657 turnOffUpstreamMobileConnection();
Erik Kline6ff17f72015-12-10 20:42:12 +09001658 mUpstreamNetworkMonitor.stop();
Robert Greenwalt2ffe4122014-12-12 12:22:31 -08001659 stopListeningForSimChanges();
Robert Greenwaltb37f0c62010-03-25 14:50:34 -07001660 notifyTetheredOfNewUpstreamIface(null);
Erik Klinefa37b2f2016-08-02 18:27:03 +09001661 handleNewUpstreamNetworkState(null);
Robert Greenwalt2a091d72010-02-11 18:18:40 -08001662 }
Erik Klinefa37b2f2016-08-02 18:27:03 +09001663
Robert Greenwalt2a091d72010-02-11 18:18:40 -08001664 @Override
1665 public boolean processMessage(Message message) {
Lorenzo Colitticd63d242016-04-10 15:39:53 +09001666 maybeLogMessage(this, message.what);
Robert Greenwalt2a091d72010-02-11 18:18:40 -08001667 boolean retValue = true;
1668 switch (message.what) {
Erik Kline6e29bf02016-08-15 16:16:18 +09001669 case CMD_TETHER_MODE_REQUESTED: {
Mitchell Wills7040b4e2016-05-23 16:40:10 -07001670 TetherInterfaceStateMachine who = (TetherInterfaceStateMachine)message.obj;
Robert Greenwalt68ea9b02012-05-10 16:58:16 -07001671 if (VDBG) Log.d(TAG, "Tether Mode requested by " + who);
Christopher Wileyd985dde2016-05-31 10:44:35 -07001672 if (mNotifyList.indexOf(who) < 0) {
1673 mNotifyList.add(who);
Erik Kline6e29bf02016-08-15 16:16:18 +09001674 mIPv6TetheringCoordinator.addActiveDownstream(who);
Christopher Wileyd985dde2016-05-31 10:44:35 -07001675 }
Mitchell Wills7040b4e2016-05-23 16:40:10 -07001676 who.sendMessage(TetherInterfaceStateMachine.CMD_TETHER_CONNECTION_CHANGED,
Erik Kline6ff17f72015-12-10 20:42:12 +09001677 mCurrentUpstreamIface);
Robert Greenwalt2a091d72010-02-11 18:18:40 -08001678 break;
Erik Kline6e29bf02016-08-15 16:16:18 +09001679 }
1680 case CMD_TETHER_MODE_UNREQUESTED: {
1681 TetherInterfaceStateMachine who = (TetherInterfaceStateMachine)message.obj;
Robert Greenwalt68ea9b02012-05-10 16:58:16 -07001682 if (VDBG) Log.d(TAG, "Tether Mode unrequested by " + who);
Christopher Wileyd985dde2016-05-31 10:44:35 -07001683 if (mNotifyList.remove(who)) {
Robert Greenwalt68ea9b02012-05-10 16:58:16 -07001684 if (DBG) Log.d(TAG, "TetherModeAlive removing notifyee " + who);
Robert Greenwalt2a091d72010-02-11 18:18:40 -08001685 if (mNotifyList.isEmpty()) {
Robert Greenwaltd70a3d42010-02-23 18:15:29 -08001686 turnOffMasterTetherSettings(); // transitions appropriately
Robert Greenwalt68ea9b02012-05-10 16:58:16 -07001687 } else {
1688 if (DBG) {
1689 Log.d(TAG, "TetherModeAlive still has " + mNotifyList.size() +
1690 " live requests:");
Christopher Wileyd985dde2016-05-31 10:44:35 -07001691 for (TetherInterfaceStateMachine o : mNotifyList) {
1692 Log.d(TAG, " " + o);
1693 }
Robert Greenwalt68ea9b02012-05-10 16:58:16 -07001694 }
Robert Greenwalt2a091d72010-02-11 18:18:40 -08001695 }
Robert Greenwalt68ea9b02012-05-10 16:58:16 -07001696 } else {
1697 Log.e(TAG, "TetherModeAliveState UNREQUESTED has unknown who: " + who);
Robert Greenwalt2a091d72010-02-11 18:18:40 -08001698 }
Erik Kline6e29bf02016-08-15 16:16:18 +09001699 mIPv6TetheringCoordinator.removeActiveDownstream(who);
Robert Greenwalt2a091d72010-02-11 18:18:40 -08001700 break;
Erik Kline6e29bf02016-08-15 16:16:18 +09001701 }
Robert Greenwalt6a1967c2010-03-17 11:19:57 -07001702 case CMD_UPSTREAM_CHANGED:
Robert Greenwalt05d89362011-01-23 16:04:05 -08001703 // need to try DUN immediately if Wifi goes down
Erik Kline6ff17f72015-12-10 20:42:12 +09001704 mTryCell = true;
Robert Greenwalt6a1967c2010-03-17 11:19:57 -07001705 chooseUpstreamType(mTryCell);
1706 mTryCell = !mTryCell;
1707 break;
Kazuhiro Ondo01758e82011-04-30 20:10:57 -05001708 case CMD_RETRY_UPSTREAM:
1709 chooseUpstreamType(mTryCell);
1710 mTryCell = !mTryCell;
1711 break;
Erik Kline00019f42016-06-30 19:31:46 +09001712 case EVENT_UPSTREAM_CALLBACK: {
1713 // First: always update local state about every network.
1714 final NetworkState ns = mUpstreamNetworkMonitor.processCallback(
1715 message.arg1, message.obj);
1716
1717 if (ns == null || !pertainsToCurrentUpstream(ns)) {
1718 // TODO: In future, this is where upstream evaluation and selection
1719 // could be handled for notifications which include sufficient data.
1720 // For example, after CONNECTIVITY_ACTION listening is removed, here
1721 // is where we could observe a Wi-Fi network becoming available and
1722 // passing validation.
1723 if (mCurrentUpstreamIface == null) {
1724 // If we have no upstream interface, try to run through upstream
1725 // selection again. If, for example, IPv4 connectivity has shown up
1726 // after IPv6 (e.g., 464xlat became available) we want the chance to
1727 // notice and act accordingly.
1728 chooseUpstreamType(false);
1729 }
1730 break;
1731 }
1732
1733 switch (message.arg1) {
1734 case UpstreamNetworkMonitor.EVENT_ON_AVAILABLE:
1735 // The default network changed, or DUN connected
1736 // before this callback was processed. Updates
1737 // for the current NetworkCapabilities and
1738 // LinkProperties have been requested (default
1739 // request) or are being sent shortly (DUN). Do
1740 // nothing until they arrive; if no updates
1741 // arrive there's nothing to do.
1742 break;
1743 case UpstreamNetworkMonitor.EVENT_ON_CAPABILITIES:
1744 handleNewUpstreamNetworkState(ns);
1745 break;
1746 case UpstreamNetworkMonitor.EVENT_ON_LINKPROPERTIES:
1747 setDnsForwarders(ns.network, ns.linkProperties);
1748 handleNewUpstreamNetworkState(ns);
1749 break;
1750 case UpstreamNetworkMonitor.EVENT_ON_LOST:
1751 // TODO: Re-evaluate possible upstreams. Currently upstream
1752 // reevaluation is triggered via received CONNECTIVITY_ACTION
1753 // broadcasts that result in being passed a
1754 // TetherMasterSM.CMD_UPSTREAM_CHANGED.
Erik Klinefa37b2f2016-08-02 18:27:03 +09001755 handleNewUpstreamNetworkState(null);
Erik Kline00019f42016-06-30 19:31:46 +09001756 break;
1757 default:
1758 break;
Erik Kline6ff17f72015-12-10 20:42:12 +09001759 }
1760 break;
Erik Kline00019f42016-06-30 19:31:46 +09001761 }
Kazuhiro Ondo01758e82011-04-30 20:10:57 -05001762 default:
1763 retValue = false;
1764 break;
Robert Greenwalt2a091d72010-02-11 18:18:40 -08001765 }
1766 return retValue;
1767 }
1768 }
1769
Wink Saville64c42ca2011-04-18 14:55:10 -07001770 class ErrorState extends State {
Robert Greenwalt2a091d72010-02-11 18:18:40 -08001771 int mErrorNotification;
1772 @Override
1773 public boolean processMessage(Message message) {
1774 boolean retValue = true;
1775 switch (message.what) {
1776 case CMD_TETHER_MODE_REQUESTED:
Mitchell Wills7040b4e2016-05-23 16:40:10 -07001777 TetherInterfaceStateMachine who = (TetherInterfaceStateMachine)message.obj;
Robert Greenwalt6a1967c2010-03-17 11:19:57 -07001778 who.sendMessage(mErrorNotification);
Robert Greenwalt2a091d72010-02-11 18:18:40 -08001779 break;
1780 default:
1781 retValue = false;
1782 }
1783 return retValue;
1784 }
1785 void notify(int msgType) {
1786 mErrorNotification = msgType;
Christopher Wileyd985dde2016-05-31 10:44:35 -07001787 for (TetherInterfaceStateMachine sm : mNotifyList) {
Robert Greenwalt6a1967c2010-03-17 11:19:57 -07001788 sm.sendMessage(msgType);
Robert Greenwalt2a091d72010-02-11 18:18:40 -08001789 }
1790 }
1791
1792 }
Robert Greenwalt2a091d72010-02-11 18:18:40 -08001793 class SetIpForwardingEnabledErrorState extends ErrorState {
1794 @Override
1795 public void enter() {
1796 Log.e(TAG, "Error in setIpForwardingEnabled");
Mitchell Wills7040b4e2016-05-23 16:40:10 -07001797 notify(TetherInterfaceStateMachine.CMD_IP_FORWARDING_ENABLE_ERROR);
Robert Greenwalt2a091d72010-02-11 18:18:40 -08001798 }
1799 }
1800
1801 class SetIpForwardingDisabledErrorState extends ErrorState {
1802 @Override
1803 public void enter() {
1804 Log.e(TAG, "Error in setIpForwardingDisabled");
Mitchell Wills7040b4e2016-05-23 16:40:10 -07001805 notify(TetherInterfaceStateMachine.CMD_IP_FORWARDING_DISABLE_ERROR);
Robert Greenwalt2a091d72010-02-11 18:18:40 -08001806 }
1807 }
1808
1809 class StartTetheringErrorState extends ErrorState {
1810 @Override
1811 public void enter() {
1812 Log.e(TAG, "Error in startTethering");
Mitchell Wills7040b4e2016-05-23 16:40:10 -07001813 notify(TetherInterfaceStateMachine.CMD_START_TETHERING_ERROR);
Robert Greenwalt2a091d72010-02-11 18:18:40 -08001814 try {
Chia-chi Yehc9338302011-05-11 16:35:13 -07001815 mNMService.setIpForwardingEnabled(false);
Robert Greenwalt2a091d72010-02-11 18:18:40 -08001816 } catch (Exception e) {}
1817 }
1818 }
1819
1820 class StopTetheringErrorState extends ErrorState {
1821 @Override
1822 public void enter() {
1823 Log.e(TAG, "Error in stopTethering");
Mitchell Wills7040b4e2016-05-23 16:40:10 -07001824 notify(TetherInterfaceStateMachine.CMD_STOP_TETHERING_ERROR);
Robert Greenwalt2a091d72010-02-11 18:18:40 -08001825 try {
Chia-chi Yehc9338302011-05-11 16:35:13 -07001826 mNMService.setIpForwardingEnabled(false);
Robert Greenwalt2a091d72010-02-11 18:18:40 -08001827 } catch (Exception e) {}
1828 }
1829 }
1830
1831 class SetDnsForwardersErrorState extends ErrorState {
1832 @Override
1833 public void enter() {
1834 Log.e(TAG, "Error in setDnsForwarders");
Mitchell Wills7040b4e2016-05-23 16:40:10 -07001835 notify(TetherInterfaceStateMachine.CMD_SET_DNS_FORWARDERS_ERROR);
Robert Greenwalt2a091d72010-02-11 18:18:40 -08001836 try {
Chia-chi Yehc9338302011-05-11 16:35:13 -07001837 mNMService.stopTethering();
Robert Greenwalt2a091d72010-02-11 18:18:40 -08001838 } catch (Exception e) {}
1839 try {
Chia-chi Yehc9338302011-05-11 16:35:13 -07001840 mNMService.setIpForwardingEnabled(false);
Robert Greenwalt2a091d72010-02-11 18:18:40 -08001841 } catch (Exception e) {}
1842 }
1843 }
1844 }
1845
Christopher Wiley499a57a2016-05-16 16:19:07 -07001846 @Override
Lorenzo Colittie3805462015-06-03 11:18:24 +09001847 public void dump(FileDescriptor fd, PrintWriter writer, String[] args) {
Christopher Wiley499a57a2016-05-16 16:19:07 -07001848 // Binder.java closes the resource for us.
1849 @SuppressWarnings("resource")
Lorenzo Colittie3805462015-06-03 11:18:24 +09001850 final IndentingPrintWriter pw = new IndentingPrintWriter(writer, " ");
Robert Greenwalt2a091d72010-02-11 18:18:40 -08001851 if (mContext.checkCallingOrSelfPermission(
1852 android.Manifest.permission.DUMP) != PackageManager.PERMISSION_GRANTED) {
1853 pw.println("Permission Denial: can't dump ConnectivityService.Tether " +
1854 "from from pid=" + Binder.getCallingPid() + ", uid=" +
1855 Binder.getCallingUid());
1856 return;
1857 }
1858
Lorenzo Colittie3805462015-06-03 11:18:24 +09001859 pw.println("Tethering:");
1860 pw.increaseIndent();
1861 pw.print("mUpstreamIfaceTypes:");
Robert Greenwaltb4453622011-11-03 16:01:40 -07001862 synchronized (mPublicSync) {
Robert Greenwaltb4453622011-11-03 16:01:40 -07001863 for (Integer netType : mUpstreamIfaceTypes) {
Lorenzo Colittie3805462015-06-03 11:18:24 +09001864 pw.print(" " + ConnectivityManager.getNetworkTypeName(netType));
Robert Greenwaltb4453622011-11-03 16:01:40 -07001865 }
Robert Greenwaltb4453622011-11-03 16:01:40 -07001866 pw.println();
Lorenzo Colittie3805462015-06-03 11:18:24 +09001867
Robert Greenwaltb4453622011-11-03 16:01:40 -07001868 pw.println("Tether state:");
Lorenzo Colittie3805462015-06-03 11:18:24 +09001869 pw.increaseIndent();
Christopher Wileyd985dde2016-05-31 10:44:35 -07001870 for (int i = 0; i < mTetherStates.size(); i++) {
1871 final String iface = mTetherStates.keyAt(i);
1872 final TetherState tetherState = mTetherStates.valueAt(i);
1873 pw.print(iface + " - ");
1874
1875 switch (tetherState.mLastState) {
1876 case IControlsTethering.STATE_UNAVAILABLE:
1877 pw.print("UnavailableState");
1878 break;
1879 case IControlsTethering.STATE_AVAILABLE:
1880 pw.print("AvailableState");
1881 break;
1882 case IControlsTethering.STATE_TETHERED:
1883 pw.print("TetheredState");
1884 break;
1885 default:
1886 pw.print("UnknownState");
1887 break;
1888 }
1889 pw.println(" - lastError = " + tetherState.mLastError);
Robert Greenwalt2a091d72010-02-11 18:18:40 -08001890 }
Lorenzo Colittie3805462015-06-03 11:18:24 +09001891 pw.decreaseIndent();
Robert Greenwalt2a091d72010-02-11 18:18:40 -08001892 }
Lorenzo Colittie3805462015-06-03 11:18:24 +09001893 pw.decreaseIndent();
Robert Greenwaltd0e18ff2010-01-26 11:40:34 -08001894 }
Christopher Wileye03fb442016-05-18 13:45:20 -07001895
1896 @Override
Christopher Wileyd985dde2016-05-31 10:44:35 -07001897 public void notifyInterfaceStateChange(String iface, TetherInterfaceStateMachine who,
1898 int state, int error) {
1899 synchronized (mPublicSync) {
1900 TetherState tetherState = mTetherStates.get(iface);
1901 if (tetherState != null && tetherState.mStateMachine.equals(who)) {
1902 tetherState.mLastState = state;
1903 tetherState.mLastError = error;
1904 } else {
1905 if (DBG) Log.d(TAG, "got notification from stale iface " + iface);
1906 }
1907 }
1908
1909 if (DBG) {
1910 Log.d(TAG, "iface " + iface + " notified that it was in state " + state +
1911 " with error " + error);
1912 }
1913
Christopher Wileydeebfec2016-09-16 11:14:36 -07001914 try {
1915 // Notify that we're tethering (or not) this interface.
1916 // This is how data saver for instance knows if the user explicitly
1917 // turned on tethering (thus keeping us from being in data saver mode).
1918 mPolicyManager.onTetheringChanged(iface, state == IControlsTethering.STATE_TETHERED);
1919 } catch (RemoteException e) {
1920 // Not really very much we can do here.
1921 }
1922
Christopher Wileyd985dde2016-05-31 10:44:35 -07001923 switch (state) {
1924 case IControlsTethering.STATE_UNAVAILABLE:
1925 case IControlsTethering.STATE_AVAILABLE:
1926 mTetherMasterSM.sendMessage(TetherMasterSM.CMD_TETHER_MODE_UNREQUESTED, who);
1927 break;
1928 case IControlsTethering.STATE_TETHERED:
1929 mTetherMasterSM.sendMessage(TetherMasterSM.CMD_TETHER_MODE_REQUESTED, who);
1930 break;
1931 }
1932 sendTetherStateChangedBroadcast();
1933 }
1934
1935 private void trackNewTetherableInterface(String iface, int interfaceType) {
1936 TetherState tetherState;
1937 tetherState = new TetherState(new TetherInterfaceStateMachine(iface, mLooper,
1938 interfaceType, mNMService, mStatsService, this));
1939 mTetherStates.put(iface, tetherState);
1940 tetherState.mStateMachine.start();
Christopher Wileye03fb442016-05-18 13:45:20 -07001941 }
Robert Greenwaltd0e18ff2010-01-26 11:40:34 -08001942}