blob: 4bc61899a4900491803e7d36274bb8fed76008c2 [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
Erik Kline2e88b5e2017-01-18 11:57:45 +090019import static android.hardware.usb.UsbManager.USB_CONNECTED;
20import static android.hardware.usb.UsbManager.USB_FUNCTION_RNDIS;
21import static android.net.wifi.WifiManager.EXTRA_WIFI_AP_STATE;
22import static android.net.wifi.WifiManager.WIFI_AP_STATE_DISABLED;
23
Robert Greenwaltd0e18ff2010-01-26 11:40:34 -080024import android.app.Notification;
25import android.app.NotificationManager;
26import android.app.PendingIntent;
Jeremy Klein36c7aa02016-01-22 14:11:45 -080027import android.bluetooth.BluetoothAdapter;
28import android.bluetooth.BluetoothPan;
29import android.bluetooth.BluetoothProfile;
30import android.bluetooth.BluetoothProfile.ServiceListener;
Robert Greenwaltd0e18ff2010-01-26 11:40:34 -080031import android.content.BroadcastReceiver;
Robert Greenwalt2ffe4122014-12-12 12:22:31 -080032import android.content.ComponentName;
Robert Greenwaltd0e18ff2010-01-26 11:40:34 -080033import android.content.Context;
34import android.content.Intent;
35import android.content.IntentFilter;
Robert Greenwalt2a091d72010-02-11 18:18:40 -080036import android.content.pm.PackageManager;
Robert Greenwaltd0e18ff2010-01-26 11:40:34 -080037import android.content.res.Resources;
Mike Lockwoodc4308f02011-03-01 08:04:54 -080038import android.hardware.usb.UsbManager;
Robert Greenwaltd0e18ff2010-01-26 11:40:34 -080039import android.net.ConnectivityManager;
Christopher Wileydeebfec2016-09-16 11:14:36 -070040import android.net.INetworkPolicyManager;
Jeff Sharkey367d15a2011-09-22 14:59:51 -070041import android.net.INetworkStatsService;
Robert Greenwalt05d89362011-01-23 16:04:05 -080042import android.net.LinkProperties;
Lorenzo Colittib57edc52014-08-22 17:10:50 -070043import android.net.Network;
Erik Kline0c9cb2b2015-11-20 17:47:08 +090044import android.net.NetworkCapabilities;
Robert Greenwalt2a091d72010-02-11 18:18:40 -080045import android.net.NetworkInfo;
Erik Kline0c9cb2b2015-11-20 17:47:08 +090046import android.net.NetworkRequest;
Erik Kline6ff17f72015-12-10 20:42:12 +090047import android.net.NetworkState;
Robert Greenwalte5903732011-02-22 16:00:42 -080048import android.net.NetworkUtils;
Lorenzo Colitti13c9fde2013-03-15 04:22:37 +090049import android.net.RouteInfo;
Jeremy Klein36c7aa02016-01-22 14:11:45 -080050import android.net.wifi.WifiManager;
Robert Greenwalt2a091d72010-02-11 18:18:40 -080051import android.os.Binder;
Jeremy Klein36c7aa02016-01-22 14:11:45 -080052import android.os.Bundle;
Robert Greenwaltd0e18ff2010-01-26 11:40:34 -080053import android.os.INetworkManagementService;
Robert Greenwaltdfadaea2010-03-11 15:03:08 -080054import android.os.Looper;
Robert Greenwalt2a091d72010-02-11 18:18:40 -080055import android.os.Message;
Jeremy Klein36c7aa02016-01-22 14:11:45 -080056import android.os.Parcel;
Christopher Wileydeebfec2016-09-16 11:14:36 -070057import android.os.RemoteException;
Jeremy Klein36c7aa02016-01-22 14:11:45 -080058import android.os.ResultReceiver;
Dianne Hackborn5ac72a22012-08-29 18:32:08 -070059import android.os.UserHandle;
Jeremy Klein15f3d212016-01-24 17:01:09 -080060import android.provider.Settings;
Jeremy Klein36c7aa02016-01-22 14:11:45 -080061import android.telephony.CarrierConfigManager;
Junda Liu38531582014-10-28 13:52:51 -070062import android.telephony.TelephonyManager;
Lorenzo Colitticd63d242016-04-10 15:39:53 +090063import android.text.TextUtils;
Christopher Wileye9490392016-05-26 15:57:29 -070064import android.util.ArrayMap;
Robert Greenwaltd0e18ff2010-01-26 11:40:34 -080065import android.util.Log;
Lorenzo Colitticd63d242016-04-10 15:39:53 +090066import android.util.SparseArray;
Robert Greenwaltd0e18ff2010-01-26 11:40:34 -080067
Christopher Wiley497c1472016-10-11 13:26:03 -070068import com.android.internal.annotations.VisibleForTesting;
Chris Wren282cfef2017-03-27 15:01:44 -040069import com.android.internal.messages.nano.SystemMessageProto.SystemMessage;
Geoffrey Pitschaf759c52017-02-15 09:35:38 -050070import com.android.internal.notification.SystemNotificationChannels;
Robert Greenwalt2ffe4122014-12-12 12:22:31 -080071import com.android.internal.telephony.IccCardConstants;
Robert Greenwalt2ffe4122014-12-12 12:22:31 -080072import com.android.internal.telephony.TelephonyIntents;
Jeff Sharkeyfe9a53b2017-03-31 14:08:23 -060073import com.android.internal.util.DumpUtils;
Jeremy Klein36c7aa02016-01-22 14:11:45 -080074import com.android.internal.util.IndentingPrintWriter;
Lorenzo Colitticd63d242016-04-10 15:39:53 +090075import com.android.internal.util.MessageUtils;
76import com.android.internal.util.Protocol;
Wink Saville64c42ca2011-04-18 14:55:10 -070077import com.android.internal.util.State;
78import com.android.internal.util.StateMachine;
Christopher Wileye03fb442016-05-18 13:45:20 -070079import com.android.server.connectivity.tethering.IControlsTethering;
Erik Kline1eb8c692016-07-08 17:21:26 +090080import com.android.server.connectivity.tethering.IPv6TetheringCoordinator;
Lorenzo Colittif537c5e2016-10-28 17:45:55 +090081import com.android.server.connectivity.tethering.IPv6TetheringInterfaceServices;
Erik Klinee0cce212017-03-06 14:05:23 +090082import com.android.server.connectivity.tethering.OffloadController;
Erik Kline227648f2017-01-20 20:26:10 +090083import com.android.server.connectivity.tethering.SimChangeListener;
Mitchell Wills7040b4e2016-05-23 16:40:10 -070084import com.android.server.connectivity.tethering.TetherInterfaceStateMachine;
Hugo Benichib55fb222017-03-10 14:20:57 +090085import com.android.server.connectivity.tethering.TetheringConfiguration;
Erik Kline9bba3402017-01-13 16:46:52 +090086import com.android.server.connectivity.tethering.UpstreamNetworkMonitor;
Lorenzo Colittif4e90ea2013-10-31 23:30:47 +090087import com.android.server.net.BaseNetworkObserver;
Robert Greenwalt2a091d72010-02-11 18:18:40 -080088
89import java.io.FileDescriptor;
90import java.io.PrintWriter;
Robert Greenwalt4f74d552011-12-19 16:59:31 -080091import java.net.Inet4Address;
Jeremy Klein36c7aa02016-01-22 14:11:45 -080092import java.net.InetAddress;
Robert Greenwaltd0e18ff2010-01-26 11:40:34 -080093import java.util.ArrayList;
Lorenzo Colittib57edc52014-08-22 17:10:50 -070094import java.util.Arrays;
Robert Greenwaltccf83af12011-06-02 17:30:47 -070095import java.util.Collection;
Robert Greenwalt2a091d72010-02-11 18:18:40 -080096import java.util.HashMap;
Erik Klineea9cc482017-03-10 19:35:34 +090097import java.util.HashSet;
Robert Greenwaltccf83af12011-06-02 17:30:47 -070098import java.util.Iterator;
Erik Klineea9cc482017-03-10 19:35:34 +090099import java.util.Map;
Robert Greenwalt2ffe4122014-12-12 12:22:31 -0800100import java.util.concurrent.atomic.AtomicInteger;
101
Jeff Sharkeycdd02c5d2011-09-16 01:52:49 -0700102
Robert Greenwaltd0e18ff2010-01-26 11:40:34 -0800103/**
104 * @hide
Robert Greenwalt2a091d72010-02-11 18:18:40 -0800105 *
Christopher Wiley3b1d9222016-05-20 16:44:04 -0700106 * This class holds much of the business logic to allow Android devices
107 * to act as IP gateways via USB, BT, and WiFi interfaces.
Robert Greenwaltd0e18ff2010-01-26 11:40:34 -0800108 */
Christopher Wileye03fb442016-05-18 13:45:20 -0700109public class Tethering extends BaseNetworkObserver implements IControlsTethering {
Robert Greenwaltd0e18ff2010-01-26 11:40:34 -0800110
Hugo Benichib55fb222017-03-10 14:20:57 +0900111 private final static String TAG = Tethering.class.getSimpleName();
Joe Onorato12acbd72016-02-01 17:49:31 -0800112 private final static boolean DBG = false;
Robert Greenwaltfd1be2b2011-11-11 12:30:19 -0800113 private final static boolean VDBG = false;
Robert Greenwaltd0e18ff2010-01-26 11:40:34 -0800114
Christopher Wiley497c1472016-10-11 13:26:03 -0700115 protected static final String DISABLE_PROVISIONING_SYSPROP_KEY = "net.tethering.noprovisioning";
116
Lorenzo Colitticd63d242016-04-10 15:39:53 +0900117 private static final Class[] messageClasses = {
Mitchell Wills7040b4e2016-05-23 16:40:10 -0700118 Tethering.class, TetherMasterSM.class, TetherInterfaceStateMachine.class
Lorenzo Colitticd63d242016-04-10 15:39:53 +0900119 };
120 private static final SparseArray<String> sMagicDecoderRing =
121 MessageUtils.findMessageNames(messageClasses);
122
Hugo Benichib55fb222017-03-10 14:20:57 +0900123 // {@link ComponentName} of the Service used to run tether provisioning.
124 private static final ComponentName TETHER_SERVICE = ComponentName.unflattenFromString(Resources
125 .getSystem().getString(com.android.internal.R.string.config_wifi_tether_enable));
126
127 private static class TetherState {
128 public final TetherInterfaceStateMachine stateMachine;
129 public int lastState;
130 public int lastError;
Erik Klineea9cc482017-03-10 19:35:34 +0900131
Hugo Benichib55fb222017-03-10 14:20:57 +0900132 public TetherState(TetherInterfaceStateMachine sm) {
133 stateMachine = sm;
134 // Assume all state machines start out available and with no errors.
135 lastState = IControlsTethering.STATE_AVAILABLE;
136 lastError = ConnectivityManager.TETHER_ERROR_NO_ERROR;
137 }
Erik Klineea9cc482017-03-10 19:35:34 +0900138
139 public boolean isCurrentlyServing() {
140 switch (lastState) {
141 case IControlsTethering.STATE_TETHERED:
142 case IControlsTethering.STATE_LOCAL_HOTSPOT:
143 return true;
144 default:
145 return false;
146 }
147 }
Hugo Benichib55fb222017-03-10 14:20:57 +0900148 }
Robert Greenwaltccf83af12011-06-02 17:30:47 -0700149
Robert Greenwaltb4453622011-11-03 16:01:40 -0700150 // used to synchronize public access to members
Erik Kline6ff17f72015-12-10 20:42:12 +0900151 private final Object mPublicSync;
Hugo Benichib55fb222017-03-10 14:20:57 +0900152 private final Context mContext;
153 private final ArrayMap<String, TetherState> mTetherStates;
154 private final BroadcastReceiver mStateReceiver;
Jeff Sharkey367d15a2011-09-22 14:59:51 -0700155 private final INetworkManagementService mNMService;
156 private final INetworkStatsService mStatsService;
Christopher Wileydeebfec2016-09-16 11:14:36 -0700157 private final INetworkPolicyManager mPolicyManager;
Erik Kline6ff17f72015-12-10 20:42:12 +0900158 private final Looper mLooper;
Christopher Wiley497c1472016-10-11 13:26:03 -0700159 private final MockableSystemProperties mSystemProperties;
Erik Kline6ff17f72015-12-10 20:42:12 +0900160 private final StateMachine mTetherMasterSM;
Erik Klinee0cce212017-03-06 14:05:23 +0900161 private final OffloadController mOffloadController;
Erik Kline6ff17f72015-12-10 20:42:12 +0900162 private final UpstreamNetworkMonitor mUpstreamNetworkMonitor;
Erik Klineea9cc482017-03-10 19:35:34 +0900163 private final HashSet<TetherInterfaceStateMachine> mForwardedDownstreams;
Erik Kline227648f2017-01-20 20:26:10 +0900164 private final SimChangeListener mSimChange;
Robert Greenwalta599fe7c2010-03-08 18:30:14 -0800165
Hugo Benichib55fb222017-03-10 14:20:57 +0900166 private volatile TetheringConfiguration mConfig;
167 private String mCurrentUpstreamIface;
Chris Wren1ce4b6d2015-06-11 10:19:43 -0400168 private Notification.Builder mTetheredNotificationBuilder;
169 private int mLastNotificationId;
Erik Klineea9cc482017-03-10 19:35:34 +0900170
Mike Lockwood6c2260b2011-07-19 13:04:47 -0700171 private boolean mRndisEnabled; // track the RNDIS function enabled state
172 private boolean mUsbTetherRequested; // true if USB tethering should be started
173 // when RNDIS is enabled
Christopher Wiley5c0b10a2016-05-31 14:43:08 -0700174 // True iff WiFi tethering should be started when soft AP is ready.
175 private boolean mWifiTetherRequested;
176
Jeff Sharkey367d15a2011-09-22 14:59:51 -0700177 public Tethering(Context context, INetworkManagementService nmService,
Christopher Wiley497c1472016-10-11 13:26:03 -0700178 INetworkStatsService statsService, INetworkPolicyManager policyManager,
179 Looper looper, MockableSystemProperties systemProperties) {
Robert Greenwaltd0e18ff2010-01-26 11:40:34 -0800180 mContext = context;
Chia-chi Yehc9338302011-05-11 16:35:13 -0700181 mNMService = nmService;
Jeff Sharkey367d15a2011-09-22 14:59:51 -0700182 mStatsService = statsService;
Christopher Wileydeebfec2016-09-16 11:14:36 -0700183 mPolicyManager = policyManager;
Christopher Wiley497c1472016-10-11 13:26:03 -0700184 mLooper = looper;
185 mSystemProperties = systemProperties;
Robert Greenwaltd0e18ff2010-01-26 11:40:34 -0800186
Robert Greenwaltb4453622011-11-03 16:01:40 -0700187 mPublicSync = new Object();
188
Christopher Wileyd985dde2016-05-31 10:44:35 -0700189 mTetherStates = new ArrayMap<>();
Robert Greenwaltd0e18ff2010-01-26 11:40:34 -0800190
Robert Greenwaltdfadaea2010-03-11 15:03:08 -0800191 mTetherMasterSM = new TetherMasterSM("TetherMaster", mLooper);
Robert Greenwalt2a091d72010-02-11 18:18:40 -0800192 mTetherMasterSM.start();
193
Erik Klinee0cce212017-03-06 14:05:23 +0900194 mOffloadController = new OffloadController(mTetherMasterSM.getHandler());
Erik Klineebdb8c82017-01-10 17:37:54 +0900195 mUpstreamNetworkMonitor = new UpstreamNetworkMonitor(
196 mContext, mTetherMasterSM, TetherMasterSM.EVENT_UPSTREAM_CALLBACK);
Erik Klineea9cc482017-03-10 19:35:34 +0900197 mForwardedDownstreams = new HashSet<>();
Erik Kline227648f2017-01-20 20:26:10 +0900198 mSimChange = new SimChangeListener(
199 mContext, mTetherMasterSM.getHandler(), () -> reevaluateSimCardProvisioning());
Erik Kline6ff17f72015-12-10 20:42:12 +0900200
Robert Greenwaltbb51d9f2010-03-12 10:23:35 -0800201 mStateReceiver = new StateReceiver();
Robert Greenwaltd0e18ff2010-01-26 11:40:34 -0800202 IntentFilter filter = new IntentFilter();
Mike Lockwood770126a2010-12-09 22:30:37 -0800203 filter.addAction(UsbManager.ACTION_USB_STATE);
Robert Greenwalt2a091d72010-02-11 18:18:40 -0800204 filter.addAction(ConnectivityManager.CONNECTIVITY_ACTION);
Christopher Wiley5c0b10a2016-05-31 14:43:08 -0700205 filter.addAction(WifiManager.WIFI_AP_STATE_CHANGED_ACTION);
Robert Greenwaltc13368b2013-07-18 14:24:42 -0700206 filter.addAction(Intent.ACTION_CONFIGURATION_CHANGED);
Erik Kline2e88b5e2017-01-18 11:57:45 +0900207 mContext.registerReceiver(mStateReceiver, filter, null, mTetherMasterSM.getHandler());
Robert Greenwaltbb51d9f2010-03-12 10:23:35 -0800208
209 filter = new IntentFilter();
Robert Greenwalt030f5e12010-03-10 16:41:03 -0800210 filter.addAction(Intent.ACTION_MEDIA_SHARED);
211 filter.addAction(Intent.ACTION_MEDIA_UNSHARED);
Robert Greenwaltbb51d9f2010-03-12 10:23:35 -0800212 filter.addDataScheme("file");
Erik Kline2e88b5e2017-01-18 11:57:45 +0900213 mContext.registerReceiver(mStateReceiver, filter, null, mTetherMasterSM.getHandler());
Robert Greenwaltbb51d9f2010-03-12 10:23:35 -0800214
Robert Greenwalt49348e72011-10-21 16:54:26 -0700215 // load device config info
216 updateConfiguration();
Robert Greenwalt49348e72011-10-21 16:54:26 -0700217 }
218
Sreeram Ramachandran62a61312014-07-24 16:05:17 -0700219 // We can't do this once in the Tethering() constructor and cache the value, because the
220 // CONNECTIVITY_SERVICE is registered only after the Tethering() constructor has completed.
221 private ConnectivityManager getConnectivityManager() {
222 return (ConnectivityManager) mContext.getSystemService(Context.CONNECTIVITY_SERVICE);
223 }
224
Erik Klined781fba2017-01-23 13:01:58 +0900225 private void updateConfiguration() {
Erik Kline3e756652017-01-17 13:42:19 +0900226 mConfig = new TetheringConfiguration(mContext);
Robert Greenwaltd0e18ff2010-01-26 11:40:34 -0800227 }
228
Erik Kline0c9cb2b2015-11-20 17:47:08 +0900229 @Override
Mike J. Chen6143f5f2011-06-23 15:17:51 -0700230 public void interfaceStatusChanged(String iface, boolean up) {
Lorenzo Colitticd63d242016-04-10 15:39:53 +0900231 // Never called directly: only called from interfaceLinkStateChanged.
232 // See NetlinkHandler.cpp:71.
Wink Savillec9acde92011-09-21 11:05:43 -0700233 if (VDBG) Log.d(TAG, "interfaceStatusChanged " + iface + ", " + up);
Robert Greenwaltb4453622011-11-03 16:01:40 -0700234 synchronized (mPublicSync) {
Christopher Wiley5c0b10a2016-05-31 14:43:08 -0700235 int interfaceType = ifaceNameToType(iface);
236 if (interfaceType == ConnectivityManager.TETHERING_INVALID) {
237 return;
Robert Greenwaltb4453622011-11-03 16:01:40 -0700238 }
Robert Greenwalt2a091d72010-02-11 18:18:40 -0800239
Christopher Wileyd985dde2016-05-31 10:44:35 -0700240 TetherState tetherState = mTetherStates.get(iface);
Mike J. Chen6143f5f2011-06-23 15:17:51 -0700241 if (up) {
Christopher Wileyd985dde2016-05-31 10:44:35 -0700242 if (tetherState == null) {
243 trackNewTetherableInterface(iface, interfaceType);
Robert Greenwalt2a091d72010-02-11 18:18:40 -0800244 }
245 } else {
Christopher Wileyd30aaeb2016-07-08 09:33:50 -0700246 if (interfaceType == ConnectivityManager.TETHERING_BLUETOOTH) {
Hugo Benichib55fb222017-03-10 14:20:57 +0900247 tetherState.stateMachine.sendMessage(
Christopher Wileyd985dde2016-05-31 10:44:35 -0700248 TetherInterfaceStateMachine.CMD_INTERFACE_DOWN);
249 mTetherStates.remove(iface);
Christopher Wileyd30aaeb2016-07-08 09:33:50 -0700250 } else {
251 // Ignore usb0 down after enabling RNDIS.
252 // We will handle disconnect in interfaceRemoved.
253 // Similarly, ignore interface down for WiFi. We monitor WiFi AP status
254 // through the WifiManager.WIFI_AP_STATE_CHANGED_ACTION intent.
255 if (VDBG) Log.d(TAG, "ignore interface down for " + iface);
Robert Greenwalt2a091d72010-02-11 18:18:40 -0800256 }
257 }
258 }
Robert Greenwaltd0e18ff2010-01-26 11:40:34 -0800259 }
260
Erik Kline0c9cb2b2015-11-20 17:47:08 +0900261 @Override
Mike J. Chenf59c7d02011-06-23 15:33:15 -0700262 public void interfaceLinkStateChanged(String iface, boolean up) {
Irfan Sheriff23eb2972011-07-22 15:21:10 -0700263 interfaceStatusChanged(iface, up);
Mike J. Chenf59c7d02011-06-23 15:33:15 -0700264 }
265
Christopher Wiley5c0b10a2016-05-31 14:43:08 -0700266 private int ifaceNameToType(String iface) {
Erik Kline3e756652017-01-17 13:42:19 +0900267 final TetheringConfiguration cfg = mConfig;
268
269 if (cfg.isWifi(iface)) {
Christopher Wiley5c0b10a2016-05-31 14:43:08 -0700270 return ConnectivityManager.TETHERING_WIFI;
Erik Kline3e756652017-01-17 13:42:19 +0900271 } else if (cfg.isUsb(iface)) {
Christopher Wiley5c0b10a2016-05-31 14:43:08 -0700272 return ConnectivityManager.TETHERING_USB;
Erik Kline3e756652017-01-17 13:42:19 +0900273 } else if (cfg.isBluetooth(iface)) {
Christopher Wiley5c0b10a2016-05-31 14:43:08 -0700274 return ConnectivityManager.TETHERING_BLUETOOTH;
275 }
276 return ConnectivityManager.TETHERING_INVALID;
277 }
278
Erik Kline0c9cb2b2015-11-20 17:47:08 +0900279 @Override
Robert Greenwalt2a091d72010-02-11 18:18:40 -0800280 public void interfaceAdded(String iface) {
Wink Savillec9acde92011-09-21 11:05:43 -0700281 if (VDBG) Log.d(TAG, "interfaceAdded " + iface);
Robert Greenwaltb4453622011-11-03 16:01:40 -0700282 synchronized (mPublicSync) {
Christopher Wiley5c0b10a2016-05-31 14:43:08 -0700283 int interfaceType = ifaceNameToType(iface);
284 if (interfaceType == ConnectivityManager.TETHERING_INVALID) {
Robert Greenwaltb4453622011-11-03 16:01:40 -0700285 if (VDBG) Log.d(TAG, iface + " is not a tetherable iface, ignoring");
286 return;
287 }
Robert Greenwalt65ae29b2010-02-18 11:25:54 -0800288
Christopher Wileyd985dde2016-05-31 10:44:35 -0700289 TetherState tetherState = mTetherStates.get(iface);
290 if (tetherState == null) {
291 trackNewTetherableInterface(iface, interfaceType);
292 } else {
Wink Savillec9acde92011-09-21 11:05:43 -0700293 if (VDBG) Log.d(TAG, "active iface (" + iface + ") reported as added, ignoring");
Robert Greenwalt2a091d72010-02-11 18:18:40 -0800294 }
Robert Greenwaltd0e18ff2010-01-26 11:40:34 -0800295 }
Robert Greenwaltd0e18ff2010-01-26 11:40:34 -0800296 }
297
Erik Kline0c9cb2b2015-11-20 17:47:08 +0900298 @Override
Robert Greenwalt2a091d72010-02-11 18:18:40 -0800299 public void interfaceRemoved(String iface) {
Wink Savillec9acde92011-09-21 11:05:43 -0700300 if (VDBG) Log.d(TAG, "interfaceRemoved " + iface);
Robert Greenwaltb4453622011-11-03 16:01:40 -0700301 synchronized (mPublicSync) {
Christopher Wileyd985dde2016-05-31 10:44:35 -0700302 TetherState tetherState = mTetherStates.get(iface);
303 if (tetherState == null) {
Wink Savillec9acde92011-09-21 11:05:43 -0700304 if (VDBG) {
Robert Greenwalt924cc942010-06-28 10:26:19 -0700305 Log.e(TAG, "attempting to remove unknown iface (" + iface + "), ignoring");
306 }
Robert Greenwalt2a091d72010-02-11 18:18:40 -0800307 return;
308 }
Hugo Benichib55fb222017-03-10 14:20:57 +0900309 tetherState.stateMachine.sendMessage(TetherInterfaceStateMachine.CMD_INTERFACE_DOWN);
Christopher Wileyd985dde2016-05-31 10:44:35 -0700310 mTetherStates.remove(iface);
Robert Greenwaltd0e18ff2010-01-26 11:40:34 -0800311 }
312 }
313
Hugo Benichib55fb222017-03-10 14:20:57 +0900314 public void startTethering(int type, ResultReceiver receiver, boolean showProvisioningUi) {
Jeremy Klein36c7aa02016-01-22 14:11:45 -0800315 if (!isTetherProvisioningRequired()) {
316 enableTetheringInternal(type, true, receiver);
317 return;
318 }
319
320 if (showProvisioningUi) {
321 runUiTetherProvisioningAndEnable(type, receiver);
322 } else {
323 runSilentTetherProvisioningAndEnable(type, receiver);
324 }
325 }
326
327 public void stopTethering(int type) {
328 enableTetheringInternal(type, false, null);
329 if (isTetherProvisioningRequired()) {
330 cancelTetherProvisioningRechecks(type);
331 }
332 }
333
334 /**
335 * Check if the device requires a provisioning check in order to enable tethering.
336 *
337 * @return a boolean - {@code true} indicating tether provisioning is required by the carrier.
338 */
Christopher Wiley497c1472016-10-11 13:26:03 -0700339 @VisibleForTesting
340 protected boolean isTetherProvisioningRequired() {
Jeremy Klein36c7aa02016-01-22 14:11:45 -0800341 String[] provisionApp = mContext.getResources().getStringArray(
342 com.android.internal.R.array.config_mobile_hotspot_provision_app);
Christopher Wiley497c1472016-10-11 13:26:03 -0700343 if (mSystemProperties.getBoolean(DISABLE_PROVISIONING_SYSPROP_KEY, false)
Jeremy Klein36c7aa02016-01-22 14:11:45 -0800344 || provisionApp == null) {
345 return false;
346 }
347
348 // Check carrier config for entitlement checks
349 final CarrierConfigManager configManager = (CarrierConfigManager) mContext
350 .getSystemService(Context.CARRIER_CONFIG_SERVICE);
Rebecca Silbersteinb07f1402016-10-04 13:53:33 -0700351 if (configManager != null && configManager.getConfig() != null) {
352 // we do have a CarrierConfigManager and it has a config.
353 boolean isEntitlementCheckRequired = configManager.getConfig().getBoolean(
354 CarrierConfigManager.KEY_REQUIRE_ENTITLEMENT_CHECKS_BOOL);
355 if (!isEntitlementCheckRequired) {
356 return false;
357 }
Jeremy Klein36c7aa02016-01-22 14:11:45 -0800358 }
359 return (provisionApp.length == 2);
360 }
361
Erik Kline227648f2017-01-20 20:26:10 +0900362 // Used by the SIM card change observation code.
363 // TODO: De-duplicate above code.
364 private boolean hasMobileHotspotProvisionApp() {
365 try {
366 if (!mContext.getResources().getString(com.android.internal.R.string.
367 config_mobile_hotspot_provision_app_no_ui).isEmpty()) {
368 Log.d(TAG, "re-evaluate provisioning");
369 return true;
370 }
371 } catch (Resources.NotFoundException e) {}
372 Log.d(TAG, "no prov-check needed for new SIM");
373 return false;
374 }
375
Jeremy Klein36c7aa02016-01-22 14:11:45 -0800376 /**
377 * Enables or disables tethering for the given type. This should only be called once
378 * provisioning has succeeded or is not necessary. It will also schedule provisioning rechecks
379 * for the specified interface.
380 */
381 private void enableTetheringInternal(int type, boolean enable, ResultReceiver receiver) {
Christopher Wiley5c0b10a2016-05-31 14:43:08 -0700382 boolean isProvisioningRequired = enable && isTetherProvisioningRequired();
383 int result;
Jeremy Klein36c7aa02016-01-22 14:11:45 -0800384 switch (type) {
385 case ConnectivityManager.TETHERING_WIFI:
Christopher Wiley5c0b10a2016-05-31 14:43:08 -0700386 result = setWifiTethering(enable);
387 if (isProvisioningRequired && result == ConnectivityManager.TETHER_ERROR_NO_ERROR) {
388 scheduleProvisioningRechecks(type);
Jeremy Klein36c7aa02016-01-22 14:11:45 -0800389 }
Christopher Wiley5c0b10a2016-05-31 14:43:08 -0700390 sendTetherResult(receiver, result);
Jeremy Klein36c7aa02016-01-22 14:11:45 -0800391 break;
392 case ConnectivityManager.TETHERING_USB:
Christopher Wiley5c0b10a2016-05-31 14:43:08 -0700393 result = setUsbTethering(enable);
394 if (isProvisioningRequired && result == ConnectivityManager.TETHER_ERROR_NO_ERROR) {
Jeremy Klein36c7aa02016-01-22 14:11:45 -0800395 scheduleProvisioningRechecks(type);
396 }
397 sendTetherResult(receiver, result);
398 break;
399 case ConnectivityManager.TETHERING_BLUETOOTH:
400 setBluetoothTethering(enable, receiver);
401 break;
402 default:
403 Log.w(TAG, "Invalid tether type.");
404 sendTetherResult(receiver, ConnectivityManager.TETHER_ERROR_UNKNOWN_IFACE);
405 }
406 }
407
408 private void sendTetherResult(ResultReceiver receiver, int result) {
409 if (receiver != null) {
410 receiver.send(result, null);
411 }
412 }
413
Christopher Wiley5c0b10a2016-05-31 14:43:08 -0700414 private int setWifiTethering(final boolean enable) {
415 synchronized (mPublicSync) {
Christopher Wiley5c0b10a2016-05-31 14:43:08 -0700416 mWifiTetherRequested = enable;
417 final WifiManager wifiManager =
418 (WifiManager) mContext.getSystemService(Context.WIFI_SERVICE);
419 if (wifiManager.setWifiApEnabled(null /* use existing wifi config */, enable)) {
420 return ConnectivityManager.TETHER_ERROR_NO_ERROR;
421 }
422 return ConnectivityManager.TETHER_ERROR_MASTER_ERROR;
423 }
424 }
425
Jeremy Klein36c7aa02016-01-22 14:11:45 -0800426 private void setBluetoothTethering(final boolean enable, final ResultReceiver receiver) {
427 final BluetoothAdapter adapter = BluetoothAdapter.getDefaultAdapter();
428 if (adapter == null || !adapter.isEnabled()) {
429 Log.w(TAG, "Tried to enable bluetooth tethering with null or disabled adapter. null: " +
430 (adapter == null));
431 sendTetherResult(receiver, ConnectivityManager.TETHER_ERROR_SERVICE_UNAVAIL);
432 return;
433 }
434
435 adapter.getProfileProxy(mContext, new ServiceListener() {
436 @Override
437 public void onServiceDisconnected(int profile) { }
438
439 @Override
440 public void onServiceConnected(int profile, BluetoothProfile proxy) {
441 ((BluetoothPan) proxy).setBluetoothTethering(enable);
442 // TODO: Enabling bluetooth tethering can fail asynchronously here.
443 // We should figure out a way to bubble up that failure instead of sending success.
Jeremy Klein45ae04a2016-02-05 10:37:32 -0800444 int result = ((BluetoothPan) proxy).isTetheringOn() == enable ?
Jeremy Klein36c7aa02016-01-22 14:11:45 -0800445 ConnectivityManager.TETHER_ERROR_NO_ERROR :
446 ConnectivityManager.TETHER_ERROR_MASTER_ERROR;
447 sendTetherResult(receiver, result);
448 if (enable && isTetherProvisioningRequired()) {
449 scheduleProvisioningRechecks(ConnectivityManager.TETHERING_BLUETOOTH);
450 }
451 adapter.closeProfileProxy(BluetoothProfile.PAN, proxy);
452 }
453 }, BluetoothProfile.PAN);
454 }
455
456 private void runUiTetherProvisioningAndEnable(int type, ResultReceiver receiver) {
Jeremy Klein15f3d212016-01-24 17:01:09 -0800457 ResultReceiver proxyReceiver = getProxyReceiver(type, receiver);
458 sendUiTetherProvisionIntent(type, proxyReceiver);
459 }
460
461 private void sendUiTetherProvisionIntent(int type, ResultReceiver receiver) {
462 Intent intent = new Intent(Settings.ACTION_TETHER_PROVISIONING);
463 intent.putExtra(ConnectivityManager.EXTRA_ADD_TETHER_TYPE, type);
464 intent.putExtra(ConnectivityManager.EXTRA_PROVISION_CALLBACK, receiver);
465 final long ident = Binder.clearCallingIdentity();
466 try {
467 mContext.startActivityAsUser(intent, UserHandle.CURRENT);
468 } finally {
469 Binder.restoreCallingIdentity(ident);
470 }
Jeremy Klein36c7aa02016-01-22 14:11:45 -0800471 }
472
473 /**
Erik Klinefdd41112017-01-12 17:02:51 +0900474 * Creates a proxy {@link ResultReceiver} which enables tethering if the provisioning result
475 * is successful before firing back up to the wrapped receiver.
Jeremy Klein36c7aa02016-01-22 14:11:45 -0800476 *
477 * @param type The type of tethering being enabled.
478 * @param receiver A ResultReceiver which will be called back with an int resultCode.
479 * @return The proxy receiver.
480 */
481 private ResultReceiver getProxyReceiver(final int type, final ResultReceiver receiver) {
482 ResultReceiver rr = new ResultReceiver(null) {
483 @Override
484 protected void onReceiveResult(int resultCode, Bundle resultData) {
485 // If provisioning is successful, enable tethering, otherwise just send the error.
486 if (resultCode == ConnectivityManager.TETHER_ERROR_NO_ERROR) {
487 enableTetheringInternal(type, true, receiver);
488 } else {
489 sendTetherResult(receiver, resultCode);
490 }
491 }
492 };
493
494 // The following is necessary to avoid unmarshalling issues when sending the receiver
Erik Kline6ff17f72015-12-10 20:42:12 +0900495 // across processes.
Jeremy Klein36c7aa02016-01-22 14:11:45 -0800496 Parcel parcel = Parcel.obtain();
497 rr.writeToParcel(parcel,0);
498 parcel.setDataPosition(0);
499 ResultReceiver receiverForSending = ResultReceiver.CREATOR.createFromParcel(parcel);
500 parcel.recycle();
501 return receiverForSending;
502 }
503
504 private void scheduleProvisioningRechecks(int type) {
505 Intent intent = new Intent();
506 intent.putExtra(ConnectivityManager.EXTRA_ADD_TETHER_TYPE, type);
507 intent.putExtra(ConnectivityManager.EXTRA_SET_ALARM, true);
508 intent.setComponent(TETHER_SERVICE);
509 final long ident = Binder.clearCallingIdentity();
510 try {
511 mContext.startServiceAsUser(intent, UserHandle.CURRENT);
512 } finally {
513 Binder.restoreCallingIdentity(ident);
514 }
515 }
516
517 private void runSilentTetherProvisioningAndEnable(int type, ResultReceiver receiver) {
518 ResultReceiver proxyReceiver = getProxyReceiver(type, receiver);
519 sendSilentTetherProvisionIntent(type, proxyReceiver);
520 }
521
522 private void sendSilentTetherProvisionIntent(int type, ResultReceiver receiver) {
523 Intent intent = new Intent();
524 intent.putExtra(ConnectivityManager.EXTRA_ADD_TETHER_TYPE, type);
525 intent.putExtra(ConnectivityManager.EXTRA_RUN_PROVISION, true);
526 intent.putExtra(ConnectivityManager.EXTRA_PROVISION_CALLBACK, receiver);
527 intent.setComponent(TETHER_SERVICE);
528 final long ident = Binder.clearCallingIdentity();
529 try {
530 mContext.startServiceAsUser(intent, UserHandle.CURRENT);
531 } finally {
532 Binder.restoreCallingIdentity(ident);
533 }
534 }
535
536 private void cancelTetherProvisioningRechecks(int type) {
537 if (getConnectivityManager().isTetheringSupported()) {
538 Intent intent = new Intent();
539 intent.putExtra(ConnectivityManager.EXTRA_REM_TETHER_TYPE, type);
540 intent.setComponent(TETHER_SERVICE);
541 final long ident = Binder.clearCallingIdentity();
542 try {
543 mContext.startServiceAsUser(intent, UserHandle.CURRENT);
544 } finally {
545 Binder.restoreCallingIdentity(ident);
546 }
547 }
548 }
Erik Kline6ff17f72015-12-10 20:42:12 +0900549
Erik Kline227648f2017-01-20 20:26:10 +0900550 // Used by the SIM card change observation code.
551 // TODO: De-duplicate with above code, where possible.
552 private void startProvisionIntent(int tetherType) {
553 final Intent startProvIntent = new Intent();
554 startProvIntent.putExtra(ConnectivityManager.EXTRA_ADD_TETHER_TYPE, tetherType);
555 startProvIntent.putExtra(ConnectivityManager.EXTRA_RUN_PROVISION, true);
556 startProvIntent.setComponent(TETHER_SERVICE);
557 mContext.startServiceAsUser(startProvIntent, UserHandle.CURRENT);
558 }
559
Robert Greenwalt5a735062010-03-02 17:25:02 -0800560 public int tether(String iface) {
Erik Klineea9cc482017-03-10 19:35:34 +0900561 return tether(iface, IControlsTethering.STATE_TETHERED);
562 }
563
564 private int tether(String iface, int requestedState) {
Wink Savillec9acde92011-09-21 11:05:43 -0700565 if (DBG) Log.d(TAG, "Tethering " + iface);
Robert Greenwaltb4453622011-11-03 16:01:40 -0700566 synchronized (mPublicSync) {
Christopher Wileyd985dde2016-05-31 10:44:35 -0700567 TetherState tetherState = mTetherStates.get(iface);
568 if (tetherState == null) {
Erik Kline00019f42016-06-30 19:31:46 +0900569 Log.e(TAG, "Tried to Tether an unknown iface: " + iface + ", ignoring");
Christopher Wiley4312a4c2016-05-23 13:58:00 -0700570 return ConnectivityManager.TETHER_ERROR_UNKNOWN_IFACE;
571 }
572 // Ignore the error status of the interface. If the interface is available,
573 // the errors are referring to past tethering attempts anyway.
Hugo Benichib55fb222017-03-10 14:20:57 +0900574 if (tetherState.lastState != IControlsTethering.STATE_AVAILABLE) {
Erik Kline00019f42016-06-30 19:31:46 +0900575 Log.e(TAG, "Tried to Tether an unavailable iface: " + iface + ", ignoring");
Christopher Wiley4312a4c2016-05-23 13:58:00 -0700576 return ConnectivityManager.TETHER_ERROR_UNAVAIL_IFACE;
Christopher Wiley4312a4c2016-05-23 13:58:00 -0700577 }
Erik Klineea9cc482017-03-10 19:35:34 +0900578 // NOTE: If a CMD_TETHER_REQUESTED message is already in the TISM's
579 // queue but not yet processed, this will be a no-op and it will not
580 // return an error.
581 //
582 // TODO: reexamine the threading and messaging model.
583 tetherState.stateMachine.sendMessage(
584 TetherInterfaceStateMachine.CMD_TETHER_REQUESTED, requestedState);
Christopher Wiley4312a4c2016-05-23 13:58:00 -0700585 return ConnectivityManager.TETHER_ERROR_NO_ERROR;
Robert Greenwalt2a091d72010-02-11 18:18:40 -0800586 }
Robert Greenwaltd0e18ff2010-01-26 11:40:34 -0800587 }
588
Robert Greenwalt5a735062010-03-02 17:25:02 -0800589 public int untether(String iface) {
Wink Savillec9acde92011-09-21 11:05:43 -0700590 if (DBG) Log.d(TAG, "Untethering " + iface);
Robert Greenwaltb4453622011-11-03 16:01:40 -0700591 synchronized (mPublicSync) {
Christopher Wileyd985dde2016-05-31 10:44:35 -0700592 TetherState tetherState = mTetherStates.get(iface);
593 if (tetherState == null) {
Christopher Wiley4312a4c2016-05-23 13:58:00 -0700594 Log.e(TAG, "Tried to Untether an unknown iface :" + iface + ", ignoring");
595 return ConnectivityManager.TETHER_ERROR_UNKNOWN_IFACE;
596 }
Erik Klineea9cc482017-03-10 19:35:34 +0900597 if (!tetherState.isCurrentlyServing()) {
598 Log.e(TAG, "Tried to untether an inactive iface :" + iface + ", ignoring");
Christopher Wiley4312a4c2016-05-23 13:58:00 -0700599 return ConnectivityManager.TETHER_ERROR_UNAVAIL_IFACE;
600 }
Hugo Benichib55fb222017-03-10 14:20:57 +0900601 tetherState.stateMachine.sendMessage(
Christopher Wileyd985dde2016-05-31 10:44:35 -0700602 TetherInterfaceStateMachine.CMD_TETHER_UNREQUESTED);
Christopher Wiley4312a4c2016-05-23 13:58:00 -0700603 return ConnectivityManager.TETHER_ERROR_NO_ERROR;
Robert Greenwalt2a091d72010-02-11 18:18:40 -0800604 }
Robert Greenwalt5a735062010-03-02 17:25:02 -0800605 }
606
Felipe Leme70c8b9b2016-04-25 14:41:31 -0700607 public void untetherAll() {
Christopher Wileyfa6a4862016-09-19 08:58:06 -0700608 stopTethering(ConnectivityManager.TETHERING_WIFI);
609 stopTethering(ConnectivityManager.TETHERING_USB);
610 stopTethering(ConnectivityManager.TETHERING_BLUETOOTH);
Felipe Leme70c8b9b2016-04-25 14:41:31 -0700611 }
612
Robert Greenwalt5a735062010-03-02 17:25:02 -0800613 public int getLastTetherError(String 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) {
Robert Greenwaltb4453622011-11-03 16:01:40 -0700617 Log.e(TAG, "Tried to getLastTetherError on an unknown iface :" + iface +
618 ", ignoring");
619 return ConnectivityManager.TETHER_ERROR_UNKNOWN_IFACE;
620 }
Hugo Benichib55fb222017-03-10 14:20:57 +0900621 return tetherState.lastError;
Robert Greenwalt5a735062010-03-02 17:25:02 -0800622 }
Robert Greenwaltd0e18ff2010-01-26 11:40:34 -0800623 }
624
Erik Klineea9cc482017-03-10 19:35:34 +0900625 // TODO: Figure out how to update for local hotspot mode interfaces.
Christopher Wileyd985dde2016-05-31 10:44:35 -0700626 private void sendTetherStateChangedBroadcast() {
Sreeram Ramachandran62a61312014-07-24 16:05:17 -0700627 if (!getConnectivityManager().isTetheringSupported()) return;
Robert Greenwalt2a091d72010-02-11 18:18:40 -0800628
629 ArrayList<String> availableList = new ArrayList<String>();
630 ArrayList<String> activeList = new ArrayList<String>();
631 ArrayList<String> erroredList = new ArrayList<String>();
632
Robert Greenwalta599fe7c2010-03-08 18:30:14 -0800633 boolean wifiTethered = false;
634 boolean usbTethered = false;
Danica Chang6fdd0c62010-08-11 14:54:43 -0700635 boolean bluetoothTethered = false;
Robert Greenwalta599fe7c2010-03-08 18:30:14 -0800636
Erik Kline3e756652017-01-17 13:42:19 +0900637 final TetheringConfiguration cfg = mConfig;
638
Robert Greenwaltb4453622011-11-03 16:01:40 -0700639 synchronized (mPublicSync) {
Christopher Wileyd985dde2016-05-31 10:44:35 -0700640 for (int i = 0; i < mTetherStates.size(); i++) {
641 TetherState tetherState = mTetherStates.valueAt(i);
642 String iface = mTetherStates.keyAt(i);
Hugo Benichib55fb222017-03-10 14:20:57 +0900643 if (tetherState.lastError != ConnectivityManager.TETHER_ERROR_NO_ERROR) {
Christopher Wileyd985dde2016-05-31 10:44:35 -0700644 erroredList.add(iface);
Hugo Benichib55fb222017-03-10 14:20:57 +0900645 } else if (tetherState.lastState == IControlsTethering.STATE_AVAILABLE) {
Christopher Wileyd985dde2016-05-31 10:44:35 -0700646 availableList.add(iface);
Hugo Benichib55fb222017-03-10 14:20:57 +0900647 } else if (tetherState.lastState == IControlsTethering.STATE_TETHERED) {
Erik Kline3e756652017-01-17 13:42:19 +0900648 if (cfg.isUsb(iface)) {
Christopher Wileyd985dde2016-05-31 10:44:35 -0700649 usbTethered = true;
Erik Kline3e756652017-01-17 13:42:19 +0900650 } else if (cfg.isWifi(iface)) {
Christopher Wileyd985dde2016-05-31 10:44:35 -0700651 wifiTethered = true;
Erik Kline3e756652017-01-17 13:42:19 +0900652 } else if (cfg.isBluetooth(iface)) {
Christopher Wileyd985dde2016-05-31 10:44:35 -0700653 bluetoothTethered = true;
Robert Greenwalt2a091d72010-02-11 18:18:40 -0800654 }
Christopher Wileyd985dde2016-05-31 10:44:35 -0700655 activeList.add(iface);
Robert Greenwalt2a091d72010-02-11 18:18:40 -0800656 }
657 }
658 }
Robert Greenwaltd0e18ff2010-01-26 11:40:34 -0800659 Intent broadcast = new Intent(ConnectivityManager.ACTION_TETHER_STATE_CHANGED);
Robert Greenwalt98c79e52011-07-28 11:51:11 -0700660 broadcast.addFlags(Intent.FLAG_RECEIVER_REPLACE_PENDING |
661 Intent.FLAG_RECEIVER_REGISTERED_ONLY_BEFORE_BOOT);
Robert Greenwalt2a091d72010-02-11 18:18:40 -0800662 broadcast.putStringArrayListExtra(ConnectivityManager.EXTRA_AVAILABLE_TETHER,
663 availableList);
664 broadcast.putStringArrayListExtra(ConnectivityManager.EXTRA_ACTIVE_TETHER, activeList);
665 broadcast.putStringArrayListExtra(ConnectivityManager.EXTRA_ERRORED_TETHER,
666 erroredList);
Dianne Hackborn5ac72a22012-08-29 18:32:08 -0700667 mContext.sendStickyBroadcastAsUser(broadcast, UserHandle.ALL);
Robert Greenwaltfd1be2b2011-11-11 12:30:19 -0800668 if (DBG) {
Lorenzo Colitticd63d242016-04-10 15:39:53 +0900669 Log.d(TAG, String.format(
670 "sendTetherStateChangedBroadcast avail=[%s] active=[%s] error=[%s]",
671 TextUtils.join(",", availableList),
672 TextUtils.join(",", activeList),
673 TextUtils.join(",", erroredList)));
Robert Greenwalt924cc942010-06-28 10:26:19 -0700674 }
Robert Greenwalta599fe7c2010-03-08 18:30:14 -0800675
676 if (usbTethered) {
Danica Chang6fdd0c62010-08-11 14:54:43 -0700677 if (wifiTethered || bluetoothTethered) {
Chris Wren282cfef2017-03-27 15:01:44 -0400678 showTetheredNotification(SystemMessage.NOTE_TETHER_GENERAL);
Robert Greenwalta599fe7c2010-03-08 18:30:14 -0800679 } else {
Chris Wren282cfef2017-03-27 15:01:44 -0400680 showTetheredNotification(SystemMessage.NOTE_TETHER_USB);
Robert Greenwalta599fe7c2010-03-08 18:30:14 -0800681 }
682 } else if (wifiTethered) {
Danica Chang6fdd0c62010-08-11 14:54:43 -0700683 if (bluetoothTethered) {
Chris Wren282cfef2017-03-27 15:01:44 -0400684 showTetheredNotification(SystemMessage.NOTE_TETHER_GENERAL);
Danica Chang6fdd0c62010-08-11 14:54:43 -0700685 } else {
Vinit Deshpande2576a802014-11-18 13:56:15 -0800686 /* We now have a status bar icon for WifiTethering, so drop the notification */
687 clearTetheredNotification();
Danica Chang6fdd0c62010-08-11 14:54:43 -0700688 }
689 } else if (bluetoothTethered) {
Chris Wren282cfef2017-03-27 15:01:44 -0400690 showTetheredNotification(SystemMessage.NOTE_TETHER_BLUETOOTH);
Robert Greenwalta599fe7c2010-03-08 18:30:14 -0800691 } else {
692 clearTetheredNotification();
693 }
694 }
695
Chris Wren282cfef2017-03-27 15:01:44 -0400696 private void showTetheredNotification(int id) {
Robert Greenwalta599fe7c2010-03-08 18:30:14 -0800697 NotificationManager notificationManager =
698 (NotificationManager)mContext.getSystemService(Context.NOTIFICATION_SERVICE);
699 if (notificationManager == null) {
700 return;
701 }
Chris Wren282cfef2017-03-27 15:01:44 -0400702 int icon = 0;
703 switch(id) {
704 case SystemMessage.NOTE_TETHER_USB:
705 icon = com.android.internal.R.drawable.stat_sys_tether_usb;
706 break;
707 case SystemMessage.NOTE_TETHER_BLUETOOTH:
708 icon = com.android.internal.R.drawable.stat_sys_tether_bluetooth;
709 break;
710 case SystemMessage.NOTE_TETHER_GENERAL:
711 default:
712 icon = com.android.internal.R.drawable.stat_sys_tether_general;
713 break;
714 }
Robert Greenwalta599fe7c2010-03-08 18:30:14 -0800715
Chris Wren1ce4b6d2015-06-11 10:19:43 -0400716 if (mLastNotificationId != 0) {
717 if (mLastNotificationId == icon) {
Robert Greenwaltdb3fe9e2010-03-18 16:28:30 -0700718 return;
719 }
Chris Wren1ce4b6d2015-06-11 10:19:43 -0400720 notificationManager.cancelAsUser(null, mLastNotificationId,
Robert Greenwalt3cab6b02012-10-04 16:44:26 -0700721 UserHandle.ALL);
Chris Wren1ce4b6d2015-06-11 10:19:43 -0400722 mLastNotificationId = 0;
Robert Greenwaltdb3fe9e2010-03-18 16:28:30 -0700723 }
724
Robert Greenwalta599fe7c2010-03-08 18:30:14 -0800725 Intent intent = new Intent();
726 intent.setClassName("com.android.settings", "com.android.settings.TetherSettings");
727 intent.setFlags(Intent.FLAG_ACTIVITY_NO_HISTORY);
728
Dianne Hackborn50cdf7c32012-09-23 17:08:57 -0700729 PendingIntent pi = PendingIntent.getActivityAsUser(mContext, 0, intent, 0,
730 null, UserHandle.CURRENT);
Robert Greenwalta599fe7c2010-03-08 18:30:14 -0800731
732 Resources r = Resources.getSystem();
733 CharSequence title = r.getText(com.android.internal.R.string.tethered_notification_title);
734 CharSequence message = r.getText(com.android.internal.R.string.
735 tethered_notification_message);
736
Chris Wren1ce4b6d2015-06-11 10:19:43 -0400737 if (mTetheredNotificationBuilder == null) {
Geoffrey Pitschaf759c52017-02-15 09:35:38 -0500738 mTetheredNotificationBuilder =
739 new Notification.Builder(mContext, SystemNotificationChannels.NETWORK_STATUS);
Chris Wren1ce4b6d2015-06-11 10:19:43 -0400740 mTetheredNotificationBuilder.setWhen(0)
741 .setOngoing(true)
742 .setColor(mContext.getColor(
743 com.android.internal.R.color.system_notification_accent_color))
744 .setVisibility(Notification.VISIBILITY_PUBLIC)
745 .setCategory(Notification.CATEGORY_STATUS);
Robert Greenwalta599fe7c2010-03-08 18:30:14 -0800746 }
Chris Wren1ce4b6d2015-06-11 10:19:43 -0400747 mTetheredNotificationBuilder.setSmallIcon(icon)
748 .setContentTitle(title)
749 .setContentText(message)
750 .setContentIntent(pi);
Chris Wren282cfef2017-03-27 15:01:44 -0400751 mLastNotificationId = id;
Robert Greenwalta599fe7c2010-03-08 18:30:14 -0800752
Chris Wren1ce4b6d2015-06-11 10:19:43 -0400753 notificationManager.notifyAsUser(null, mLastNotificationId,
754 mTetheredNotificationBuilder.build(), UserHandle.ALL);
Robert Greenwalta599fe7c2010-03-08 18:30:14 -0800755 }
756
757 private void clearTetheredNotification() {
758 NotificationManager notificationManager =
759 (NotificationManager)mContext.getSystemService(Context.NOTIFICATION_SERVICE);
Chris Wren1ce4b6d2015-06-11 10:19:43 -0400760 if (notificationManager != null && mLastNotificationId != 0) {
761 notificationManager.cancelAsUser(null, mLastNotificationId,
Dianne Hackborn50cdf7c32012-09-23 17:08:57 -0700762 UserHandle.ALL);
Chris Wren1ce4b6d2015-06-11 10:19:43 -0400763 mLastNotificationId = 0;
Robert Greenwalta599fe7c2010-03-08 18:30:14 -0800764 }
Robert Greenwalt5a735062010-03-02 17:25:02 -0800765 }
766
Robert Greenwalt2a091d72010-02-11 18:18:40 -0800767 private class StateReceiver extends BroadcastReceiver {
Nick Kralevich70c117a2014-05-27 15:30:02 -0700768 @Override
Robert Greenwaltd0e18ff2010-01-26 11:40:34 -0800769 public void onReceive(Context content, Intent intent) {
Erik Kline2e88b5e2017-01-18 11:57:45 +0900770 final String action = intent.getAction();
771 if (action == null) return;
772
Mike Lockwood770126a2010-12-09 22:30:37 -0800773 if (action.equals(UsbManager.ACTION_USB_STATE)) {
Erik Kline2e88b5e2017-01-18 11:57:45 +0900774 handleUsbAction(intent);
Robert Greenwalt2a091d72010-02-11 18:18:40 -0800775 } else if (action.equals(ConnectivityManager.CONNECTIVITY_ACTION)) {
Erik Kline2e88b5e2017-01-18 11:57:45 +0900776 handleConnectivityAction(intent);
Christopher Wiley5c0b10a2016-05-31 14:43:08 -0700777 } else if (action.equals(WifiManager.WIFI_AP_STATE_CHANGED_ACTION)) {
Erik Kline2e88b5e2017-01-18 11:57:45 +0900778 handleWifiApAction(intent);
Robert Greenwaltc13368b2013-07-18 14:24:42 -0700779 } else if (action.equals(Intent.ACTION_CONFIGURATION_CHANGED)) {
780 updateConfiguration();
Robert Greenwaltd0e18ff2010-01-26 11:40:34 -0800781 }
782 }
Erik Kline2e88b5e2017-01-18 11:57:45 +0900783
784 private void handleConnectivityAction(Intent intent) {
785 final NetworkInfo networkInfo = (NetworkInfo)intent.getParcelableExtra(
786 ConnectivityManager.EXTRA_NETWORK_INFO);
787 if (networkInfo == null ||
788 networkInfo.getDetailedState() == NetworkInfo.DetailedState.FAILED) {
789 return;
790 }
791
792 if (VDBG) Log.d(TAG, "Tethering got CONNECTIVITY_ACTION: " + networkInfo.toString());
793 mTetherMasterSM.sendMessage(TetherMasterSM.CMD_UPSTREAM_CHANGED);
794 }
795
796 private void handleUsbAction(Intent intent) {
797 final boolean usbConnected = intent.getBooleanExtra(USB_CONNECTED, false);
798 final boolean rndisEnabled = intent.getBooleanExtra(USB_FUNCTION_RNDIS, false);
799 synchronized (Tethering.this.mPublicSync) {
800 mRndisEnabled = rndisEnabled;
801 // start tethering if we have a request pending
802 if (usbConnected && mRndisEnabled && mUsbTetherRequested) {
Erik Klineea9cc482017-03-10 19:35:34 +0900803 tetherMatchingInterfaces(
804 IControlsTethering.STATE_TETHERED,
805 ConnectivityManager.TETHERING_USB);
Erik Kline2e88b5e2017-01-18 11:57:45 +0900806 }
807 mUsbTetherRequested = false;
808 }
809 }
810
811 private void handleWifiApAction(Intent intent) {
812 final int curState = intent.getIntExtra(EXTRA_WIFI_AP_STATE, WIFI_AP_STATE_DISABLED);
813 synchronized (Tethering.this.mPublicSync) {
814 switch (curState) {
815 case WifiManager.WIFI_AP_STATE_ENABLING:
816 // We can see this state on the way to both enabled and failure states.
817 break;
818 case WifiManager.WIFI_AP_STATE_ENABLED:
819 // When the AP comes up and we've been requested to tether it, do so.
Erik Klineea9cc482017-03-10 19:35:34 +0900820 // Otherwise, assume it's a local-only hotspot request.
821 final int state = mWifiTetherRequested
822 ? IControlsTethering.STATE_TETHERED
823 : IControlsTethering.STATE_LOCAL_HOTSPOT;
824 tetherMatchingInterfaces(state, ConnectivityManager.TETHERING_WIFI);
Erik Kline2e88b5e2017-01-18 11:57:45 +0900825 break;
826 case WifiManager.WIFI_AP_STATE_DISABLED:
827 case WifiManager.WIFI_AP_STATE_DISABLING:
828 case WifiManager.WIFI_AP_STATE_FAILED:
829 default:
830 if (DBG) {
831 Log.d(TAG, "Canceling WiFi tethering request - AP_STATE=" +
832 curState);
833 }
834 // Tell appropriate interface state machines that they should tear
835 // themselves down.
836 for (int i = 0; i < mTetherStates.size(); i++) {
837 TetherInterfaceStateMachine tism =
Hugo Benichib55fb222017-03-10 14:20:57 +0900838 mTetherStates.valueAt(i).stateMachine;
Erik Kline2e88b5e2017-01-18 11:57:45 +0900839 if (tism.interfaceType() == ConnectivityManager.TETHERING_WIFI) {
840 tism.sendMessage(
841 TetherInterfaceStateMachine.CMD_TETHER_UNREQUESTED);
842 break; // There should be at most one of these.
843 }
844 }
845 // Regardless of whether we requested this transition, the AP has gone
846 // down. Don't try to tether again unless we're requested to do so.
847 mWifiTetherRequested = false;
848 break;
849 }
850 }
851 }
Robert Greenwaltd0e18ff2010-01-26 11:40:34 -0800852 }
853
Erik Klineea9cc482017-03-10 19:35:34 +0900854 // TODO: Consider renaming to something more accurate in its description.
855 // This method:
856 // - allows requesting either tethering or local hotspot serving states
857 // - handles both enabling and disabling serving states
858 // - only tethers the first matching interface in listInterfaces()
859 // order of a given type
860 private void tetherMatchingInterfaces(int requestedState, int interfaceType) {
861 if (VDBG) {
862 Log.d(TAG, "tetherMatchingInterfaces(" + requestedState + ", " + interfaceType + ")");
863 }
Mike Lockwood3c2a2f62011-06-08 15:10:26 -0700864
Christopher Wiley5c0b10a2016-05-31 14:43:08 -0700865 String[] ifaces = null;
Robert Greenwalt65ae29b2010-02-18 11:25:54 -0800866 try {
Chia-chi Yehc9338302011-05-11 16:35:13 -0700867 ifaces = mNMService.listInterfaces();
Robert Greenwalt65ae29b2010-02-18 11:25:54 -0800868 } catch (Exception e) {
Mike Lockwood3c2a2f62011-06-08 15:10:26 -0700869 Log.e(TAG, "Error listing Interfaces", e);
Robert Greenwalt65ae29b2010-02-18 11:25:54 -0800870 return;
871 }
Christopher Wiley5c0b10a2016-05-31 14:43:08 -0700872 String chosenIface = null;
873 if (ifaces != null) {
874 for (String iface : ifaces) {
875 if (ifaceNameToType(iface) == interfaceType) {
876 chosenIface = iface;
877 break;
Robert Greenwalt65ae29b2010-02-18 11:25:54 -0800878 }
879 }
880 }
Christopher Wiley5c0b10a2016-05-31 14:43:08 -0700881 if (chosenIface == null) {
882 Log.e(TAG, "could not find iface of type " + interfaceType);
883 return;
884 }
885
Erik Klineea9cc482017-03-10 19:35:34 +0900886 final int result;
887 switch (requestedState) {
888 case IControlsTethering.STATE_UNAVAILABLE:
889 case IControlsTethering.STATE_AVAILABLE:
890 result = untether(chosenIface);
891 break;
892 case IControlsTethering.STATE_TETHERED:
893 case IControlsTethering.STATE_LOCAL_HOTSPOT:
894 result = tether(chosenIface, requestedState);
895 break;
896 default:
897 Log.wtf(TAG, "Unknown interface state: " + requestedState);
898 return;
899 }
Christopher Wiley5c0b10a2016-05-31 14:43:08 -0700900 if (result != ConnectivityManager.TETHER_ERROR_NO_ERROR) {
901 Log.e(TAG, "unable start or stop tethering on iface " + chosenIface);
902 return;
903 }
Robert Greenwalt65ae29b2010-02-18 11:25:54 -0800904 }
905
Erik Kline3e756652017-01-17 13:42:19 +0900906 public TetheringConfiguration getTetheringConfiguration() {
907 return mConfig;
908 }
909
Erik Klined781fba2017-01-23 13:01:58 +0900910 public boolean hasTetherableConfiguration() {
911 final TetheringConfiguration cfg = mConfig;
912 final boolean hasDownstreamConfiguration =
913 (cfg.tetherableUsbRegexs.length != 0) ||
914 (cfg.tetherableWifiRegexs.length != 0) ||
915 (cfg.tetherableBluetoothRegexs.length != 0);
916 final boolean hasUpstreamConfiguration = !cfg.preferredUpstreamIfaceTypes.isEmpty();
917
918 return hasDownstreamConfiguration && hasUpstreamConfiguration;
919 }
920
Erik Kline3e756652017-01-17 13:42:19 +0900921 // TODO - update callers to use getTetheringConfiguration(),
922 // which has only final members.
Robert Greenwalt2a091d72010-02-11 18:18:40 -0800923 public String[] getTetherableUsbRegexs() {
Erik Kline3e756652017-01-17 13:42:19 +0900924 return copy(mConfig.tetherableUsbRegexs);
Robert Greenwaltd0e18ff2010-01-26 11:40:34 -0800925 }
926
Robert Greenwalt2a091d72010-02-11 18:18:40 -0800927 public String[] getTetherableWifiRegexs() {
Erik Kline3e756652017-01-17 13:42:19 +0900928 return copy(mConfig.tetherableWifiRegexs);
Robert Greenwalt2a091d72010-02-11 18:18:40 -0800929 }
930
Danica Chang6fdd0c62010-08-11 14:54:43 -0700931 public String[] getTetherableBluetoothRegexs() {
Erik Kline3e756652017-01-17 13:42:19 +0900932 return copy(mConfig.tetherableBluetoothRegexs);
Danica Chang6fdd0c62010-08-11 14:54:43 -0700933 }
934
Mike Lockwood6c2260b2011-07-19 13:04:47 -0700935 public int setUsbTethering(boolean enable) {
Wink Savillec9acde92011-09-21 11:05:43 -0700936 if (VDBG) Log.d(TAG, "setUsbTethering(" + enable + ")");
Erik Klinefdd41112017-01-12 17:02:51 +0900937 UsbManager usbManager = mContext.getSystemService(UsbManager.class);
Mike Lockwood6c2260b2011-07-19 13:04:47 -0700938
Robert Greenwaltb4453622011-11-03 16:01:40 -0700939 synchronized (mPublicSync) {
Mike Lockwood6c2260b2011-07-19 13:04:47 -0700940 if (enable) {
941 if (mRndisEnabled) {
Jeremy Klein36c7aa02016-01-22 14:11:45 -0800942 final long ident = Binder.clearCallingIdentity();
943 try {
Erik Klineea9cc482017-03-10 19:35:34 +0900944 tetherMatchingInterfaces(IControlsTethering.STATE_TETHERED,
945 ConnectivityManager.TETHERING_USB);
Jeremy Klein36c7aa02016-01-22 14:11:45 -0800946 } finally {
947 Binder.restoreCallingIdentity(ident);
948 }
Mike Lockwood6c2260b2011-07-19 13:04:47 -0700949 } else {
950 mUsbTetherRequested = true;
Jerry Zhang935372b2016-10-12 15:49:32 -0700951 usbManager.setCurrentFunction(UsbManager.USB_FUNCTION_RNDIS, false);
Mike Lockwood6c2260b2011-07-19 13:04:47 -0700952 }
953 } else {
Jeremy Klein36c7aa02016-01-22 14:11:45 -0800954 final long ident = Binder.clearCallingIdentity();
955 try {
Erik Klineea9cc482017-03-10 19:35:34 +0900956 tetherMatchingInterfaces(IControlsTethering.STATE_AVAILABLE,
957 ConnectivityManager.TETHERING_USB);
Jeremy Klein36c7aa02016-01-22 14:11:45 -0800958 } finally {
959 Binder.restoreCallingIdentity(ident);
960 }
Mike Lockwood6c2260b2011-07-19 13:04:47 -0700961 if (mRndisEnabled) {
Jerry Zhang935372b2016-10-12 15:49:32 -0700962 usbManager.setCurrentFunction(null, false);
Mike Lockwood6c2260b2011-07-19 13:04:47 -0700963 }
964 mUsbTetherRequested = false;
965 }
966 }
967 return ConnectivityManager.TETHER_ERROR_NO_ERROR;
968 }
969
Robert Greenwaltb4453622011-11-03 16:01:40 -0700970 // TODO review API - maybe return ArrayList<String> here and below?
Robert Greenwalt2a091d72010-02-11 18:18:40 -0800971 public String[] getTetheredIfaces() {
972 ArrayList<String> list = new ArrayList<String>();
Robert Greenwaltb4453622011-11-03 16:01:40 -0700973 synchronized (mPublicSync) {
Christopher Wileyd985dde2016-05-31 10:44:35 -0700974 for (int i = 0; i < mTetherStates.size(); i++) {
975 TetherState tetherState = mTetherStates.valueAt(i);
Hugo Benichib55fb222017-03-10 14:20:57 +0900976 if (tetherState.lastState == IControlsTethering.STATE_TETHERED) {
Christopher Wileyd985dde2016-05-31 10:44:35 -0700977 list.add(mTetherStates.keyAt(i));
Robert Greenwalt2a091d72010-02-11 18:18:40 -0800978 }
979 }
Robert Greenwaltd0e18ff2010-01-26 11:40:34 -0800980 }
Christopher Wileyd985dde2016-05-31 10:44:35 -0700981 return list.toArray(new String[list.size()]);
Robert Greenwalt2a091d72010-02-11 18:18:40 -0800982 }
983
984 public String[] getTetherableIfaces() {
985 ArrayList<String> list = new ArrayList<String>();
Robert Greenwaltb4453622011-11-03 16:01:40 -0700986 synchronized (mPublicSync) {
Christopher Wileyd985dde2016-05-31 10:44:35 -0700987 for (int i = 0; i < mTetherStates.size(); i++) {
988 TetherState tetherState = mTetherStates.valueAt(i);
Hugo Benichib55fb222017-03-10 14:20:57 +0900989 if (tetherState.lastState == IControlsTethering.STATE_AVAILABLE) {
Christopher Wileyd985dde2016-05-31 10:44:35 -0700990 list.add(mTetherStates.keyAt(i));
Robert Greenwalt2a091d72010-02-11 18:18:40 -0800991 }
992 }
993 }
Christopher Wileyd985dde2016-05-31 10:44:35 -0700994 return list.toArray(new String[list.size()]);
Robert Greenwalt2a091d72010-02-11 18:18:40 -0800995 }
996
Robert Greenwalt9c7e2c22014-06-23 14:53:42 -0700997 public String[] getTetheredDhcpRanges() {
Erik Kline3e756652017-01-17 13:42:19 +0900998 return mConfig.dhcpRanges;
Robert Greenwalt9c7e2c22014-06-23 14:53:42 -0700999 }
1000
Robert Greenwalt5a735062010-03-02 17:25:02 -08001001 public String[] getErroredIfaces() {
1002 ArrayList<String> list = new ArrayList<String>();
Robert Greenwaltb4453622011-11-03 16:01:40 -07001003 synchronized (mPublicSync) {
Christopher Wileyd985dde2016-05-31 10:44:35 -07001004 for (int i = 0; i < mTetherStates.size(); i++) {
1005 TetherState tetherState = mTetherStates.valueAt(i);
Hugo Benichib55fb222017-03-10 14:20:57 +09001006 if (tetherState.lastError != ConnectivityManager.TETHER_ERROR_NO_ERROR) {
Christopher Wileyd985dde2016-05-31 10:44:35 -07001007 list.add(mTetherStates.keyAt(i));
Robert Greenwalt5a735062010-03-02 17:25:02 -08001008 }
1009 }
1010 }
Christopher Wileyd985dde2016-05-31 10:44:35 -07001011 return list.toArray(new String[list.size()]);
Robert Greenwalt5a735062010-03-02 17:25:02 -08001012 }
1013
Lorenzo Colitticd63d242016-04-10 15:39:53 +09001014 private void maybeLogMessage(State state, int what) {
1015 if (DBG) {
1016 Log.d(TAG, state.getName() + " got " +
1017 sMagicDecoderRing.get(what, Integer.toString(what)));
1018 }
1019 }
1020
Erik Klineea9cc482017-03-10 19:35:34 +09001021 private boolean upstreamWanted() {
1022 if (!mForwardedDownstreams.isEmpty()) return true;
1023
1024 synchronized (mPublicSync) {
1025 return mUsbTetherRequested || mWifiTetherRequested;
1026 }
1027 }
1028
Erik Kline00019f42016-06-30 19:31:46 +09001029 // Needed because the canonical source of upstream truth is just the
1030 // upstream interface name, |mCurrentUpstreamIface|. This is ripe for
1031 // future simplification, once the upstream Network is canonical.
1032 boolean pertainsToCurrentUpstream(NetworkState ns) {
1033 if (ns != null && ns.linkProperties != null && mCurrentUpstreamIface != null) {
1034 for (String ifname : ns.linkProperties.getAllInterfaceNames()) {
1035 if (mCurrentUpstreamIface.equals(ifname)) {
1036 return true;
Erik Kline6ff17f72015-12-10 20:42:12 +09001037 }
1038 }
Erik Kline6ff17f72015-12-10 20:42:12 +09001039 }
Erik Kline00019f42016-06-30 19:31:46 +09001040 return false;
Erik Kline6ff17f72015-12-10 20:42:12 +09001041 }
1042
Erik Kline227648f2017-01-20 20:26:10 +09001043 private void reevaluateSimCardProvisioning() {
1044 if (!hasMobileHotspotProvisionApp()) return;
1045
1046 ArrayList<Integer> tethered = new ArrayList<>();
1047 synchronized (mPublicSync) {
1048 for (int i = 0; i < mTetherStates.size(); i++) {
1049 TetherState tetherState = mTetherStates.valueAt(i);
1050 if (tetherState.lastState != IControlsTethering.STATE_TETHERED) {
1051 continue; // Skip interfaces that aren't tethered.
1052 }
1053 String iface = mTetherStates.keyAt(i);
1054 int interfaceType = ifaceNameToType(iface);
1055 if (interfaceType != ConnectivityManager.TETHERING_INVALID) {
1056 tethered.add(interfaceType);
1057 }
1058 }
1059 }
1060
1061 for (int tetherType : tethered) {
1062 startProvisionIntent(tetherType);
1063 }
1064 }
1065
Wink Saville64c42ca2011-04-18 14:55:10 -07001066 class TetherMasterSM extends StateMachine {
Lorenzo Colitticd63d242016-04-10 15:39:53 +09001067 private static final int BASE_MASTER = Protocol.BASE_TETHERING;
Erik Klineea9cc482017-03-10 19:35:34 +09001068 // an interface SM has requested Tethering/Local Hotspot
1069 static final int EVENT_IFACE_SERVING_STATE_ACTIVE = BASE_MASTER + 1;
1070 // an interface SM has unrequested Tethering/Local Hotspot
1071 static final int EVENT_IFACE_SERVING_STATE_INACTIVE = BASE_MASTER + 2;
Robert Greenwalt6a1967c2010-03-17 11:19:57 -07001072 // upstream connection change - do the right thing
Lorenzo Colitticd63d242016-04-10 15:39:53 +09001073 static final int CMD_UPSTREAM_CHANGED = BASE_MASTER + 3;
Robert Greenwalt6a1967c2010-03-17 11:19:57 -07001074 // we don't have a valid upstream conn, check again after a delay
Lorenzo Colitticd63d242016-04-10 15:39:53 +09001075 static final int CMD_RETRY_UPSTREAM = BASE_MASTER + 4;
Erik Kline6ff17f72015-12-10 20:42:12 +09001076 // Events from NetworkCallbacks that we process on the master state
1077 // machine thread on behalf of the UpstreamNetworkMonitor.
Erik Kline00019f42016-06-30 19:31:46 +09001078 static final int EVENT_UPSTREAM_CALLBACK = BASE_MASTER + 5;
Yohei, Oshima977aad52016-12-08 13:39:20 +09001079 // we treated the error and want now to clear it
1080 static final int CMD_CLEAR_ERROR = BASE_MASTER + 6;
Robert Greenwalt2a091d72010-02-11 18:18:40 -08001081
Wink Saville64c42ca2011-04-18 14:55:10 -07001082 private State mInitialState;
1083 private State mTetherModeAliveState;
Robert Greenwalt2a091d72010-02-11 18:18:40 -08001084
Wink Saville64c42ca2011-04-18 14:55:10 -07001085 private State mSetIpForwardingEnabledErrorState;
1086 private State mSetIpForwardingDisabledErrorState;
1087 private State mStartTetheringErrorState;
1088 private State mStopTetheringErrorState;
1089 private State mSetDnsForwardersErrorState;
Robert Greenwalt2a091d72010-02-11 18:18:40 -08001090
Christopher Wileyd985dde2016-05-31 10:44:35 -07001091 // This list is a little subtle. It contains all the interfaces that currently are
1092 // requesting tethering, regardless of whether these interfaces are still members of
1093 // mTetherStates. This allows us to maintain the following predicates:
1094 //
1095 // 1) mTetherStates contains the set of all currently existing, tetherable, link state up
1096 // interfaces.
1097 // 2) mNotifyList contains all state machines that may have outstanding tethering state
1098 // that needs to be torn down.
1099 //
1100 // Because we excise interfaces immediately from mTetherStates, we must maintain mNotifyList
1101 // so that the garbage collector does not clean up the state machine before it has a chance
1102 // to tear itself down.
Erik Kline1eb8c692016-07-08 17:21:26 +09001103 private final ArrayList<TetherInterfaceStateMachine> mNotifyList;
1104 private final IPv6TetheringCoordinator mIPv6TetheringCoordinator;
Robert Greenwalt2a091d72010-02-11 18:18:40 -08001105
Robert Greenwalt6a1967c2010-03-17 11:19:57 -07001106 private static final int UPSTREAM_SETTLE_TIME_MS = 10000;
Robert Greenwalt2a091d72010-02-11 18:18:40 -08001107
Robert Greenwaltdfadaea2010-03-11 15:03:08 -08001108 TetherMasterSM(String name, Looper looper) {
1109 super(name, looper);
Robert Greenwalt2a091d72010-02-11 18:18:40 -08001110
1111 //Add states
1112 mInitialState = new InitialState();
1113 addState(mInitialState);
Robert Greenwalt2a091d72010-02-11 18:18:40 -08001114 mTetherModeAliveState = new TetherModeAliveState();
1115 addState(mTetherModeAliveState);
Robert Greenwalt2a091d72010-02-11 18:18:40 -08001116
Robert Greenwalt2a091d72010-02-11 18:18:40 -08001117 mSetIpForwardingEnabledErrorState = new SetIpForwardingEnabledErrorState();
1118 addState(mSetIpForwardingEnabledErrorState);
1119 mSetIpForwardingDisabledErrorState = new SetIpForwardingDisabledErrorState();
1120 addState(mSetIpForwardingDisabledErrorState);
1121 mStartTetheringErrorState = new StartTetheringErrorState();
1122 addState(mStartTetheringErrorState);
1123 mStopTetheringErrorState = new StopTetheringErrorState();
1124 addState(mStopTetheringErrorState);
1125 mSetDnsForwardersErrorState = new SetDnsForwardersErrorState();
1126 addState(mSetDnsForwardersErrorState);
1127
Mitchell Wills7040b4e2016-05-23 16:40:10 -07001128 mNotifyList = new ArrayList<>();
Erik Kline1eb8c692016-07-08 17:21:26 +09001129 mIPv6TetheringCoordinator = new IPv6TetheringCoordinator(mNotifyList);
Robert Greenwalt2a091d72010-02-11 18:18:40 -08001130 setInitialState(mInitialState);
1131 }
1132
Wink Saville64c42ca2011-04-18 14:55:10 -07001133 class TetherMasterUtilState extends State {
Robert Greenwaltd70a3d42010-02-23 18:15:29 -08001134 @Override
1135 public boolean processMessage(Message m) {
1136 return false;
1137 }
Erik Kline0c9cb2b2015-11-20 17:47:08 +09001138
Erik Klinefb413432017-02-14 18:26:04 +09001139 protected void requestUpstreamMobileConnection() {
Erik Kline5645af42017-01-23 16:42:01 +09001140 mUpstreamNetworkMonitor.updateMobileRequiresDun(mConfig.isDunRequired);
Erik Klinefdd41112017-01-12 17:02:51 +09001141 mUpstreamNetworkMonitor.registerMobileNetworkRequest();
Robert Greenwaltd70a3d42010-02-23 18:15:29 -08001142 }
Erik Kline0c9cb2b2015-11-20 17:47:08 +09001143
Erik Klineefdd3f4c2017-01-20 16:31:29 +09001144 protected void unrequestUpstreamMobileConnection() {
Erik Klinefdd41112017-01-12 17:02:51 +09001145 mUpstreamNetworkMonitor.releaseMobileNetworkRequest();
Erik Kline0c9cb2b2015-11-20 17:47:08 +09001146 }
1147
Robert Greenwalt6a1967c2010-03-17 11:19:57 -07001148 protected boolean turnOnMasterTetherSettings() {
Erik Kline3e756652017-01-17 13:42:19 +09001149 final TetheringConfiguration cfg = mConfig;
Robert Greenwaltd70a3d42010-02-23 18:15:29 -08001150 try {
Chia-chi Yehc9338302011-05-11 16:35:13 -07001151 mNMService.setIpForwardingEnabled(true);
Robert Greenwaltd70a3d42010-02-23 18:15:29 -08001152 } catch (Exception e) {
Wink Saville951749b2010-03-19 09:03:13 -07001153 transitionTo(mSetIpForwardingEnabledErrorState);
Robert Greenwaltd70a3d42010-02-23 18:15:29 -08001154 return false;
1155 }
Erik Klineea9cc482017-03-10 19:35:34 +09001156 // TODO: Randomize DHCPv4 ranges, especially in hotspot mode.
Robert Greenwaltd70a3d42010-02-23 18:15:29 -08001157 try {
Erik Klineea9cc482017-03-10 19:35:34 +09001158 // TODO: Find a more accurate method name (startDHCPv4()?).
Erik Kline3e756652017-01-17 13:42:19 +09001159 mNMService.startTethering(cfg.dhcpRanges);
Robert Greenwaltd70a3d42010-02-23 18:15:29 -08001160 } catch (Exception e) {
Robert Greenwalt547e4e72011-01-14 17:24:03 -08001161 try {
Chia-chi Yehc9338302011-05-11 16:35:13 -07001162 mNMService.stopTethering();
Erik Kline3e756652017-01-17 13:42:19 +09001163 mNMService.startTethering(cfg.dhcpRanges);
Robert Greenwalt547e4e72011-01-14 17:24:03 -08001164 } catch (Exception ee) {
1165 transitionTo(mStartTetheringErrorState);
1166 return false;
1167 }
Robert Greenwaltd70a3d42010-02-23 18:15:29 -08001168 }
Robert Greenwaltd70a3d42010-02-23 18:15:29 -08001169 return true;
1170 }
Erik Kline6ff17f72015-12-10 20:42:12 +09001171
Robert Greenwalt6a1967c2010-03-17 11:19:57 -07001172 protected boolean turnOffMasterTetherSettings() {
Robert Greenwaltd70a3d42010-02-23 18:15:29 -08001173 try {
Chia-chi Yehc9338302011-05-11 16:35:13 -07001174 mNMService.stopTethering();
Robert Greenwaltd70a3d42010-02-23 18:15:29 -08001175 } catch (Exception e) {
1176 transitionTo(mStopTetheringErrorState);
1177 return false;
1178 }
1179 try {
Chia-chi Yehc9338302011-05-11 16:35:13 -07001180 mNMService.setIpForwardingEnabled(false);
Robert Greenwaltd70a3d42010-02-23 18:15:29 -08001181 } catch (Exception e) {
1182 transitionTo(mSetIpForwardingDisabledErrorState);
1183 return false;
1184 }
1185 transitionTo(mInitialState);
1186 return true;
1187 }
Robert Greenwalt10398722010-12-17 15:20:36 -08001188
Robert Greenwalt6a1967c2010-03-17 11:19:57 -07001189 protected void chooseUpstreamType(boolean tryCell) {
Erik Kline14f7faf2017-02-14 19:03:09 +09001190 final int upstreamType = findPreferredUpstreamType(tryCell);
1191 setUpstreamByType(upstreamType);
1192 }
1193
1194 protected int findPreferredUpstreamType(boolean tryCell) {
Erik Kline00019f42016-06-30 19:31:46 +09001195 final ConnectivityManager cm = getConnectivityManager();
Robert Greenwaltccf83af12011-06-02 17:30:47 -07001196 int upType = ConnectivityManager.TYPE_NONE;
Robert Greenwaltccf83af12011-06-02 17:30:47 -07001197
Robert Greenwaltc13368b2013-07-18 14:24:42 -07001198 updateConfiguration(); // TODO - remove?
Robert Greenwalt49348e72011-10-21 16:54:26 -07001199
Erik Kline3e756652017-01-17 13:42:19 +09001200 final TetheringConfiguration cfg = mConfig;
1201 if (VDBG) {
1202 Log.d(TAG, "chooseUpstreamType has upstream iface types:");
1203 for (Integer netType : cfg.preferredUpstreamIfaceTypes) {
1204 Log.d(TAG, " " + netType);
Robert Greenwalt6a1967c2010-03-17 11:19:57 -07001205 }
1206 }
Robert Greenwaltccf83af12011-06-02 17:30:47 -07001207
Erik Kline3e756652017-01-17 13:42:19 +09001208 for (Integer netType : cfg.preferredUpstreamIfaceTypes) {
1209 NetworkInfo info = cm.getNetworkInfo(netType.intValue());
1210 // TODO: if the network is suspended we should consider
1211 // that to be the same as connected here.
1212 if ((info != null) && info.isConnected()) {
1213 upType = netType.intValue();
1214 break;
1215 }
1216 }
1217
1218 final int preferredUpstreamMobileApn = cfg.isDunRequired
1219 ? ConnectivityManager.TYPE_MOBILE_DUN
1220 : ConnectivityManager.TYPE_MOBILE_HIPRI;
Robert Greenwaltfd1be2b2011-11-11 12:30:19 -08001221 if (DBG) {
Erik Kline0c9cb2b2015-11-20 17:47:08 +09001222 Log.d(TAG, "chooseUpstreamType(" + tryCell + "),"
1223 + " preferredApn="
Erik Kline3e756652017-01-17 13:42:19 +09001224 + ConnectivityManager.getNetworkTypeName(preferredUpstreamMobileApn)
Erik Kline0c9cb2b2015-11-20 17:47:08 +09001225 + ", got type="
1226 + ConnectivityManager.getNetworkTypeName(upType));
Robert Greenwaltccf83af12011-06-02 17:30:47 -07001227 }
1228
Erik Kline0c9cb2b2015-11-20 17:47:08 +09001229 switch (upType) {
1230 case ConnectivityManager.TYPE_MOBILE_DUN:
1231 case ConnectivityManager.TYPE_MOBILE_HIPRI:
1232 // If we're on DUN, put our own grab on it.
Erik Kline5645af42017-01-23 16:42:01 +09001233 requestUpstreamMobileConnection();
Erik Kline0c9cb2b2015-11-20 17:47:08 +09001234 break;
1235 case ConnectivityManager.TYPE_NONE:
Erik Klinefb413432017-02-14 18:26:04 +09001236 if (tryCell) {
1237 requestUpstreamMobileConnection();
Erik Kline0c9cb2b2015-11-20 17:47:08 +09001238 // We think mobile should be coming up; don't set a retry.
1239 } else {
1240 sendMessageDelayed(CMD_RETRY_UPSTREAM, UPSTREAM_SETTLE_TIME_MS);
1241 }
1242 break;
1243 default:
1244 /* If we've found an active upstream connection that's not DUN/HIPRI
1245 * we should stop any outstanding DUN/HIPRI start requests.
1246 *
1247 * If we found NONE we don't want to do this as we want any previous
1248 * requests to keep trying to bring up something we can use.
1249 */
Erik Klineefdd3f4c2017-01-20 16:31:29 +09001250 unrequestUpstreamMobileConnection();
Erik Kline0c9cb2b2015-11-20 17:47:08 +09001251 break;
Robert Greenwaltccf83af12011-06-02 17:30:47 -07001252 }
1253
Erik Kline14f7faf2017-02-14 19:03:09 +09001254 return upType;
1255 }
1256
1257 protected void setUpstreamByType(int upType) {
1258 final ConnectivityManager cm = getConnectivityManager();
Erik Kline00019f42016-06-30 19:31:46 +09001259 Network network = null;
Erik Kline14f7faf2017-02-14 19:03:09 +09001260 String iface = null;
Erik Kline0c9cb2b2015-11-20 17:47:08 +09001261 if (upType != ConnectivityManager.TYPE_NONE) {
Erik Kline00019f42016-06-30 19:31:46 +09001262 LinkProperties linkProperties = cm.getLinkProperties(upType);
Robert Greenwaltbb4fd0b2011-12-09 15:36:50 -08001263 if (linkProperties != null) {
Lorenzo Colitti13c9fde2013-03-15 04:22:37 +09001264 // Find the interface with the default IPv4 route. It may be the
1265 // interface described by linkProperties, or one of the interfaces
1266 // stacked on top of it.
1267 Log.i(TAG, "Finding IPv4 upstream interface on: " + linkProperties);
1268 RouteInfo ipv4Default = RouteInfo.selectBestRoute(
1269 linkProperties.getAllRoutes(), Inet4Address.ANY);
1270 if (ipv4Default != null) {
1271 iface = ipv4Default.getInterface();
1272 Log.i(TAG, "Found interface " + ipv4Default.getInterface());
1273 } else {
1274 Log.i(TAG, "No IPv4 upstream interface, giving up.");
1275 }
1276 }
1277
1278 if (iface != null) {
Erik Kline00019f42016-06-30 19:31:46 +09001279 network = cm.getNetworkForType(upType);
Erik Kline6ff17f72015-12-10 20:42:12 +09001280 if (network == null) {
1281 Log.e(TAG, "No Network for upstream type " + upType + "!");
Robert Greenwaltbb4fd0b2011-12-09 15:36:50 -08001282 }
Erik Kline6ff17f72015-12-10 20:42:12 +09001283 setDnsForwarders(network, linkProperties);
Robert Greenwaltbb4fd0b2011-12-09 15:36:50 -08001284 }
Robert Greenwalt6a1967c2010-03-17 11:19:57 -07001285 }
Robert Greenwaltb37f0c62010-03-25 14:50:34 -07001286 notifyTetheredOfNewUpstreamIface(iface);
Erik Kline00019f42016-06-30 19:31:46 +09001287 NetworkState ns = mUpstreamNetworkMonitor.lookup(network);
1288 if (ns != null && pertainsToCurrentUpstream(ns)) {
1289 // If we already have NetworkState for this network examine
1290 // it immediately, because there likely will be no second
1291 // EVENT_ON_AVAILABLE (it was already received).
1292 handleNewUpstreamNetworkState(ns);
Erik Klinefa37b2f2016-08-02 18:27:03 +09001293 } else if (mCurrentUpstreamIface == null) {
1294 // There are no available upstream networks, or none that
1295 // have an IPv4 default route (current metric for success).
1296 handleNewUpstreamNetworkState(null);
Erik Kline00019f42016-06-30 19:31:46 +09001297 }
Robert Greenwalt6a1967c2010-03-17 11:19:57 -07001298 }
Robert Greenwaltccf83af12011-06-02 17:30:47 -07001299
Erik Kline6ff17f72015-12-10 20:42:12 +09001300 protected void setDnsForwarders(final Network network, final LinkProperties lp) {
Erik Kline3e756652017-01-17 13:42:19 +09001301 // TODO: Set v4 and/or v6 DNS per available connectivity.
1302 String[] dnsServers = mConfig.defaultIPv4DNS;
Erik Kline6ff17f72015-12-10 20:42:12 +09001303 final Collection<InetAddress> dnses = lp.getDnsServers();
1304 // TODO: Properly support the absence of DNS servers.
1305 if (dnses != null && !dnses.isEmpty()) {
1306 // TODO: remove this invocation of NetworkUtils.makeStrings().
1307 dnsServers = NetworkUtils.makeStrings(dnses);
1308 }
1309 if (VDBG) {
1310 Log.d(TAG, "Setting DNS forwarders: Network=" + network +
1311 ", dnsServers=" + Arrays.toString(dnsServers));
1312 }
1313 try {
1314 mNMService.setDnsForwarders(network, dnsServers);
1315 } catch (Exception e) {
1316 // TODO: Investigate how this can fail and what exactly
1317 // happens if/when such failures occur.
1318 Log.e(TAG, "Setting DNS forwarders failed!");
1319 transitionTo(mSetDnsForwardersErrorState);
1320 }
1321 }
1322
Robert Greenwaltb37f0c62010-03-25 14:50:34 -07001323 protected void notifyTetheredOfNewUpstreamIface(String ifaceName) {
Lorenzo Colitticd63d242016-04-10 15:39:53 +09001324 if (DBG) Log.d(TAG, "Notifying tethered with upstream=" + ifaceName);
Erik Kline6ff17f72015-12-10 20:42:12 +09001325 mCurrentUpstreamIface = ifaceName;
Mitchell Wills7040b4e2016-05-23 16:40:10 -07001326 for (TetherInterfaceStateMachine sm : mNotifyList) {
1327 sm.sendMessage(TetherInterfaceStateMachine.CMD_TETHER_CONNECTION_CHANGED,
Robert Greenwalt6a1967c2010-03-17 11:19:57 -07001328 ifaceName);
1329 }
1330 }
Erik Kline00019f42016-06-30 19:31:46 +09001331
1332 protected void handleNewUpstreamNetworkState(NetworkState ns) {
1333 mIPv6TetheringCoordinator.updateUpstreamNetworkState(ns);
Erik Kline34baa3d2017-03-07 13:17:39 +09001334 mOffloadController.setUpstreamLinkProperties(
1335 (ns != null) ? ns.linkProperties : null);
Erik Kline00019f42016-06-30 19:31:46 +09001336 }
Robert Greenwaltd70a3d42010-02-23 18:15:29 -08001337 }
Robert Greenwalt2a091d72010-02-11 18:18:40 -08001338
Erik Klineea9cc482017-03-10 19:35:34 +09001339 private void handleInterfaceServingStateActive(int mode, TetherInterfaceStateMachine who) {
1340 if (mNotifyList.indexOf(who) < 0) {
1341 mNotifyList.add(who);
1342 mIPv6TetheringCoordinator.addActiveDownstream(who, mode);
1343 }
1344
1345 if (mode == IControlsTethering.STATE_TETHERED) {
1346 mForwardedDownstreams.add(who);
1347 } else {
1348 mForwardedDownstreams.remove(who);
1349 }
1350 }
1351
1352 private void handleInterfaceServingStateInactive(TetherInterfaceStateMachine who) {
1353 mNotifyList.remove(who);
1354 mIPv6TetheringCoordinator.removeActiveDownstream(who);
1355 mForwardedDownstreams.remove(who);
1356 }
1357
Robert Greenwaltd70a3d42010-02-23 18:15:29 -08001358 class InitialState extends TetherMasterUtilState {
Robert Greenwalt2a091d72010-02-11 18:18:40 -08001359 @Override
1360 public boolean processMessage(Message message) {
Lorenzo Colitticd63d242016-04-10 15:39:53 +09001361 maybeLogMessage(this, message.what);
Robert Greenwalt2a091d72010-02-11 18:18:40 -08001362 boolean retValue = true;
1363 switch (message.what) {
Erik Klineea9cc482017-03-10 19:35:34 +09001364 case EVENT_IFACE_SERVING_STATE_ACTIVE:
Mitchell Wills7040b4e2016-05-23 16:40:10 -07001365 TetherInterfaceStateMachine who = (TetherInterfaceStateMachine)message.obj;
Robert Greenwalt68ea9b02012-05-10 16:58:16 -07001366 if (VDBG) Log.d(TAG, "Tether Mode requested by " + who);
Erik Klineea9cc482017-03-10 19:35:34 +09001367 handleInterfaceServingStateActive(message.arg1, who);
Robert Greenwalt6a1967c2010-03-17 11:19:57 -07001368 transitionTo(mTetherModeAliveState);
Robert Greenwalt2a091d72010-02-11 18:18:40 -08001369 break;
Erik Klineea9cc482017-03-10 19:35:34 +09001370 case EVENT_IFACE_SERVING_STATE_INACTIVE:
Mitchell Wills7040b4e2016-05-23 16:40:10 -07001371 who = (TetherInterfaceStateMachine)message.obj;
Robert Greenwalt68ea9b02012-05-10 16:58:16 -07001372 if (VDBG) Log.d(TAG, "Tether Mode unrequested by " + who);
Erik Klineea9cc482017-03-10 19:35:34 +09001373 handleInterfaceServingStateInactive(who);
Robert Greenwalt2a091d72010-02-11 18:18:40 -08001374 break;
Robert Greenwalt2a091d72010-02-11 18:18:40 -08001375 default:
1376 retValue = false;
1377 break;
1378 }
1379 return retValue;
1380 }
1381 }
1382
Robert Greenwaltd70a3d42010-02-23 18:15:29 -08001383 class TetherModeAliveState extends TetherMasterUtilState {
Erik Klineea9cc482017-03-10 19:35:34 +09001384 boolean mUpstreamWanted = false;
Erik Kline6ff17f72015-12-10 20:42:12 +09001385 boolean mTryCell = true;
Erik Klinee0cce212017-03-06 14:05:23 +09001386
Robert Greenwalt2a091d72010-02-11 18:18:40 -08001387 @Override
1388 public void enter() {
Erik Kline6ff17f72015-12-10 20:42:12 +09001389 // TODO: examine if we should check the return value.
Robert Greenwalt4f74d552011-12-19 16:59:31 -08001390 turnOnMasterTetherSettings(); // may transition us out
Erik Kline227648f2017-01-20 20:26:10 +09001391 mSimChange.startListening();
Erik Kline6ff17f72015-12-10 20:42:12 +09001392 mUpstreamNetworkMonitor.start();
Erik Klinee0cce212017-03-06 14:05:23 +09001393 mOffloadController.start();
Robert Greenwalt4f74d552011-12-19 16:59:31 -08001394
Erik Klineea9cc482017-03-10 19:35:34 +09001395 if (upstreamWanted()) {
1396 mUpstreamWanted = true;
1397 chooseUpstreamType(true);
1398 mTryCell = false;
1399 }
Robert Greenwalt6a1967c2010-03-17 11:19:57 -07001400 }
Erik Klinefa37b2f2016-08-02 18:27:03 +09001401
Robert Greenwalt6a1967c2010-03-17 11:19:57 -07001402 @Override
1403 public void exit() {
Erik Klinee0cce212017-03-06 14:05:23 +09001404 mOffloadController.stop();
Erik Klineefdd3f4c2017-01-20 16:31:29 +09001405 unrequestUpstreamMobileConnection();
Erik Kline6ff17f72015-12-10 20:42:12 +09001406 mUpstreamNetworkMonitor.stop();
Erik Kline227648f2017-01-20 20:26:10 +09001407 mSimChange.stopListening();
Robert Greenwaltb37f0c62010-03-25 14:50:34 -07001408 notifyTetheredOfNewUpstreamIface(null);
Erik Klinefa37b2f2016-08-02 18:27:03 +09001409 handleNewUpstreamNetworkState(null);
Robert Greenwalt2a091d72010-02-11 18:18:40 -08001410 }
Erik Klinefa37b2f2016-08-02 18:27:03 +09001411
Erik Klineea9cc482017-03-10 19:35:34 +09001412 private boolean updateUpstreamWanted() {
1413 final boolean previousUpstreamWanted = mUpstreamWanted;
1414 mUpstreamWanted = upstreamWanted();
1415 return previousUpstreamWanted;
1416 }
1417
Robert Greenwalt2a091d72010-02-11 18:18:40 -08001418 @Override
1419 public boolean processMessage(Message message) {
Lorenzo Colitticd63d242016-04-10 15:39:53 +09001420 maybeLogMessage(this, message.what);
Robert Greenwalt2a091d72010-02-11 18:18:40 -08001421 boolean retValue = true;
1422 switch (message.what) {
Erik Klineea9cc482017-03-10 19:35:34 +09001423 case EVENT_IFACE_SERVING_STATE_ACTIVE: {
Mitchell Wills7040b4e2016-05-23 16:40:10 -07001424 TetherInterfaceStateMachine who = (TetherInterfaceStateMachine)message.obj;
Robert Greenwalt68ea9b02012-05-10 16:58:16 -07001425 if (VDBG) Log.d(TAG, "Tether Mode requested by " + who);
Erik Klineea9cc482017-03-10 19:35:34 +09001426 handleInterfaceServingStateActive(message.arg1, who);
Mitchell Wills7040b4e2016-05-23 16:40:10 -07001427 who.sendMessage(TetherInterfaceStateMachine.CMD_TETHER_CONNECTION_CHANGED,
Erik Kline6ff17f72015-12-10 20:42:12 +09001428 mCurrentUpstreamIface);
Erik Klineea9cc482017-03-10 19:35:34 +09001429 // If there has been a change and an upstream is now
1430 // desired, kick off the selection process.
1431 final boolean previousUpstreamWanted = updateUpstreamWanted();
1432 if (!previousUpstreamWanted && mUpstreamWanted) {
1433 chooseUpstreamType(true);
1434 }
Robert Greenwalt2a091d72010-02-11 18:18:40 -08001435 break;
Erik Kline6e29bf02016-08-15 16:16:18 +09001436 }
Erik Klineea9cc482017-03-10 19:35:34 +09001437 case EVENT_IFACE_SERVING_STATE_INACTIVE: {
Erik Kline6e29bf02016-08-15 16:16:18 +09001438 TetherInterfaceStateMachine who = (TetherInterfaceStateMachine)message.obj;
Robert Greenwalt68ea9b02012-05-10 16:58:16 -07001439 if (VDBG) Log.d(TAG, "Tether Mode unrequested by " + who);
Erik Klineea9cc482017-03-10 19:35:34 +09001440 handleInterfaceServingStateInactive(who);
1441
1442 if (mNotifyList.isEmpty()) {
1443 turnOffMasterTetherSettings(); // transitions appropriately
1444 } else {
1445 if (DBG) {
1446 Log.d(TAG, "TetherModeAlive still has " + mNotifyList.size() +
1447 " live requests:");
1448 for (TetherInterfaceStateMachine o : mNotifyList) {
1449 Log.d(TAG, " " + o);
Robert Greenwalt68ea9b02012-05-10 16:58:16 -07001450 }
Robert Greenwalt2a091d72010-02-11 18:18:40 -08001451 }
1452 }
Erik Klineea9cc482017-03-10 19:35:34 +09001453 // If there has been a change and an upstream is no
1454 // longer desired, release any mobile requests.
1455 final boolean previousUpstreamWanted = updateUpstreamWanted();
1456 if (previousUpstreamWanted && !mUpstreamWanted) {
1457 mUpstreamNetworkMonitor.releaseMobileNetworkRequest();
1458 }
Robert Greenwalt2a091d72010-02-11 18:18:40 -08001459 break;
Erik Kline6e29bf02016-08-15 16:16:18 +09001460 }
Robert Greenwalt6a1967c2010-03-17 11:19:57 -07001461 case CMD_UPSTREAM_CHANGED:
Erik Klineea9cc482017-03-10 19:35:34 +09001462 updateUpstreamWanted();
1463 if (!mUpstreamWanted) break;
1464
Erik Klinefb413432017-02-14 18:26:04 +09001465 // Need to try DUN immediately if Wi-Fi goes down.
1466 chooseUpstreamType(true);
1467 mTryCell = false;
Robert Greenwalt6a1967c2010-03-17 11:19:57 -07001468 break;
Kazuhiro Ondo01758e82011-04-30 20:10:57 -05001469 case CMD_RETRY_UPSTREAM:
Erik Klineea9cc482017-03-10 19:35:34 +09001470 updateUpstreamWanted();
1471 if (!mUpstreamWanted) break;
1472
Kazuhiro Ondo01758e82011-04-30 20:10:57 -05001473 chooseUpstreamType(mTryCell);
1474 mTryCell = !mTryCell;
1475 break;
Erik Kline00019f42016-06-30 19:31:46 +09001476 case EVENT_UPSTREAM_CALLBACK: {
Erik Klineea9cc482017-03-10 19:35:34 +09001477 updateUpstreamWanted();
1478 if (!mUpstreamWanted) break;
1479
Erik Kline452b4b72017-01-11 17:34:50 +09001480 final NetworkState ns = (NetworkState) message.obj;
Erik Kline00019f42016-06-30 19:31:46 +09001481
1482 if (ns == null || !pertainsToCurrentUpstream(ns)) {
1483 // TODO: In future, this is where upstream evaluation and selection
1484 // could be handled for notifications which include sufficient data.
1485 // For example, after CONNECTIVITY_ACTION listening is removed, here
1486 // is where we could observe a Wi-Fi network becoming available and
1487 // passing validation.
1488 if (mCurrentUpstreamIface == null) {
1489 // If we have no upstream interface, try to run through upstream
1490 // selection again. If, for example, IPv4 connectivity has shown up
1491 // after IPv6 (e.g., 464xlat became available) we want the chance to
1492 // notice and act accordingly.
1493 chooseUpstreamType(false);
1494 }
1495 break;
1496 }
1497
1498 switch (message.arg1) {
1499 case UpstreamNetworkMonitor.EVENT_ON_AVAILABLE:
1500 // The default network changed, or DUN connected
1501 // before this callback was processed. Updates
1502 // for the current NetworkCapabilities and
1503 // LinkProperties have been requested (default
1504 // request) or are being sent shortly (DUN). Do
1505 // nothing until they arrive; if no updates
1506 // arrive there's nothing to do.
1507 break;
1508 case UpstreamNetworkMonitor.EVENT_ON_CAPABILITIES:
1509 handleNewUpstreamNetworkState(ns);
1510 break;
1511 case UpstreamNetworkMonitor.EVENT_ON_LINKPROPERTIES:
1512 setDnsForwarders(ns.network, ns.linkProperties);
1513 handleNewUpstreamNetworkState(ns);
1514 break;
1515 case UpstreamNetworkMonitor.EVENT_ON_LOST:
1516 // TODO: Re-evaluate possible upstreams. Currently upstream
1517 // reevaluation is triggered via received CONNECTIVITY_ACTION
1518 // broadcasts that result in being passed a
1519 // TetherMasterSM.CMD_UPSTREAM_CHANGED.
Erik Klinefa37b2f2016-08-02 18:27:03 +09001520 handleNewUpstreamNetworkState(null);
Erik Kline00019f42016-06-30 19:31:46 +09001521 break;
1522 default:
1523 break;
Erik Kline6ff17f72015-12-10 20:42:12 +09001524 }
1525 break;
Erik Kline00019f42016-06-30 19:31:46 +09001526 }
Kazuhiro Ondo01758e82011-04-30 20:10:57 -05001527 default:
1528 retValue = false;
1529 break;
Robert Greenwalt2a091d72010-02-11 18:18:40 -08001530 }
1531 return retValue;
1532 }
1533 }
1534
Wink Saville64c42ca2011-04-18 14:55:10 -07001535 class ErrorState extends State {
Robert Greenwalt2a091d72010-02-11 18:18:40 -08001536 int mErrorNotification;
1537 @Override
1538 public boolean processMessage(Message message) {
1539 boolean retValue = true;
1540 switch (message.what) {
Erik Klineea9cc482017-03-10 19:35:34 +09001541 case EVENT_IFACE_SERVING_STATE_ACTIVE:
Mitchell Wills7040b4e2016-05-23 16:40:10 -07001542 TetherInterfaceStateMachine who = (TetherInterfaceStateMachine)message.obj;
Robert Greenwalt6a1967c2010-03-17 11:19:57 -07001543 who.sendMessage(mErrorNotification);
Robert Greenwalt2a091d72010-02-11 18:18:40 -08001544 break;
Yohei, Oshima977aad52016-12-08 13:39:20 +09001545 case CMD_CLEAR_ERROR:
1546 mErrorNotification = ConnectivityManager.TETHER_ERROR_NO_ERROR;
1547 transitionTo(mInitialState);
1548 break;
Robert Greenwalt2a091d72010-02-11 18:18:40 -08001549 default:
1550 retValue = false;
1551 }
1552 return retValue;
1553 }
1554 void notify(int msgType) {
1555 mErrorNotification = msgType;
Christopher Wileyd985dde2016-05-31 10:44:35 -07001556 for (TetherInterfaceStateMachine sm : mNotifyList) {
Robert Greenwalt6a1967c2010-03-17 11:19:57 -07001557 sm.sendMessage(msgType);
Robert Greenwalt2a091d72010-02-11 18:18:40 -08001558 }
1559 }
1560
1561 }
Robert Greenwalt2a091d72010-02-11 18:18:40 -08001562 class SetIpForwardingEnabledErrorState extends ErrorState {
1563 @Override
1564 public void enter() {
1565 Log.e(TAG, "Error in setIpForwardingEnabled");
Mitchell Wills7040b4e2016-05-23 16:40:10 -07001566 notify(TetherInterfaceStateMachine.CMD_IP_FORWARDING_ENABLE_ERROR);
Robert Greenwalt2a091d72010-02-11 18:18:40 -08001567 }
1568 }
1569
1570 class SetIpForwardingDisabledErrorState extends ErrorState {
1571 @Override
1572 public void enter() {
1573 Log.e(TAG, "Error in setIpForwardingDisabled");
Mitchell Wills7040b4e2016-05-23 16:40:10 -07001574 notify(TetherInterfaceStateMachine.CMD_IP_FORWARDING_DISABLE_ERROR);
Robert Greenwalt2a091d72010-02-11 18:18:40 -08001575 }
1576 }
1577
1578 class StartTetheringErrorState extends ErrorState {
1579 @Override
1580 public void enter() {
1581 Log.e(TAG, "Error in startTethering");
Mitchell Wills7040b4e2016-05-23 16:40:10 -07001582 notify(TetherInterfaceStateMachine.CMD_START_TETHERING_ERROR);
Robert Greenwalt2a091d72010-02-11 18:18:40 -08001583 try {
Chia-chi Yehc9338302011-05-11 16:35:13 -07001584 mNMService.setIpForwardingEnabled(false);
Robert Greenwalt2a091d72010-02-11 18:18:40 -08001585 } catch (Exception e) {}
1586 }
1587 }
1588
1589 class StopTetheringErrorState extends ErrorState {
1590 @Override
1591 public void enter() {
1592 Log.e(TAG, "Error in stopTethering");
Mitchell Wills7040b4e2016-05-23 16:40:10 -07001593 notify(TetherInterfaceStateMachine.CMD_STOP_TETHERING_ERROR);
Robert Greenwalt2a091d72010-02-11 18:18:40 -08001594 try {
Chia-chi Yehc9338302011-05-11 16:35:13 -07001595 mNMService.setIpForwardingEnabled(false);
Robert Greenwalt2a091d72010-02-11 18:18:40 -08001596 } catch (Exception e) {}
1597 }
1598 }
1599
1600 class SetDnsForwardersErrorState extends ErrorState {
1601 @Override
1602 public void enter() {
1603 Log.e(TAG, "Error in setDnsForwarders");
Mitchell Wills7040b4e2016-05-23 16:40:10 -07001604 notify(TetherInterfaceStateMachine.CMD_SET_DNS_FORWARDERS_ERROR);
Robert Greenwalt2a091d72010-02-11 18:18:40 -08001605 try {
Chia-chi Yehc9338302011-05-11 16:35:13 -07001606 mNMService.stopTethering();
Robert Greenwalt2a091d72010-02-11 18:18:40 -08001607 } catch (Exception e) {}
1608 try {
Chia-chi Yehc9338302011-05-11 16:35:13 -07001609 mNMService.setIpForwardingEnabled(false);
Robert Greenwalt2a091d72010-02-11 18:18:40 -08001610 } catch (Exception e) {}
1611 }
1612 }
1613 }
1614
Christopher Wiley499a57a2016-05-16 16:19:07 -07001615 @Override
Lorenzo Colittie3805462015-06-03 11:18:24 +09001616 public void dump(FileDescriptor fd, PrintWriter writer, String[] args) {
Christopher Wiley499a57a2016-05-16 16:19:07 -07001617 // Binder.java closes the resource for us.
1618 @SuppressWarnings("resource")
Lorenzo Colittie3805462015-06-03 11:18:24 +09001619 final IndentingPrintWriter pw = new IndentingPrintWriter(writer, " ");
Jeff Sharkeyfe9a53b2017-03-31 14:08:23 -06001620 if (!DumpUtils.checkDumpPermission(mContext, TAG, pw)) return;
Robert Greenwalt2a091d72010-02-11 18:18:40 -08001621
Lorenzo Colittie3805462015-06-03 11:18:24 +09001622 pw.println("Tethering:");
1623 pw.increaseIndent();
Lorenzo Colittie3805462015-06-03 11:18:24 +09001624
Erik Kline9db1b542017-03-16 14:10:27 +09001625 pw.println("Configuration:");
1626 pw.increaseIndent();
1627 final TetheringConfiguration cfg = mConfig;
1628 cfg.dump(pw);
1629 pw.decreaseIndent();
1630
1631 synchronized (mPublicSync) {
Robert Greenwaltb4453622011-11-03 16:01:40 -07001632 pw.println("Tether state:");
Lorenzo Colittie3805462015-06-03 11:18:24 +09001633 pw.increaseIndent();
Christopher Wileyd985dde2016-05-31 10:44:35 -07001634 for (int i = 0; i < mTetherStates.size(); i++) {
1635 final String iface = mTetherStates.keyAt(i);
1636 final TetherState tetherState = mTetherStates.valueAt(i);
1637 pw.print(iface + " - ");
1638
Hugo Benichib55fb222017-03-10 14:20:57 +09001639 switch (tetherState.lastState) {
Christopher Wileyd985dde2016-05-31 10:44:35 -07001640 case IControlsTethering.STATE_UNAVAILABLE:
1641 pw.print("UnavailableState");
1642 break;
1643 case IControlsTethering.STATE_AVAILABLE:
1644 pw.print("AvailableState");
1645 break;
1646 case IControlsTethering.STATE_TETHERED:
1647 pw.print("TetheredState");
1648 break;
Erik Klineea9cc482017-03-10 19:35:34 +09001649 case IControlsTethering.STATE_LOCAL_HOTSPOT:
1650 pw.print("LocalHotspotState");
1651 break;
Christopher Wileyd985dde2016-05-31 10:44:35 -07001652 default:
1653 pw.print("UnknownState");
1654 break;
1655 }
Hugo Benichib55fb222017-03-10 14:20:57 +09001656 pw.println(" - lastError = " + tetherState.lastError);
Robert Greenwalt2a091d72010-02-11 18:18:40 -08001657 }
Erik Klineea9cc482017-03-10 19:35:34 +09001658 pw.println("Upstream wanted: " + upstreamWanted());
Lorenzo Colittie3805462015-06-03 11:18:24 +09001659 pw.decreaseIndent();
Robert Greenwalt2a091d72010-02-11 18:18:40 -08001660 }
Lorenzo Colittie3805462015-06-03 11:18:24 +09001661 pw.decreaseIndent();
Robert Greenwaltd0e18ff2010-01-26 11:40:34 -08001662 }
Christopher Wileye03fb442016-05-18 13:45:20 -07001663
1664 @Override
Christopher Wileyd985dde2016-05-31 10:44:35 -07001665 public void notifyInterfaceStateChange(String iface, TetherInterfaceStateMachine who,
1666 int state, int error) {
1667 synchronized (mPublicSync) {
1668 TetherState tetherState = mTetherStates.get(iface);
Hugo Benichib55fb222017-03-10 14:20:57 +09001669 if (tetherState != null && tetherState.stateMachine.equals(who)) {
1670 tetherState.lastState = state;
1671 tetherState.lastError = error;
Christopher Wileyd985dde2016-05-31 10:44:35 -07001672 } else {
1673 if (DBG) Log.d(TAG, "got notification from stale iface " + iface);
1674 }
1675 }
1676
1677 if (DBG) {
1678 Log.d(TAG, "iface " + iface + " notified that it was in state " + state +
1679 " with error " + error);
1680 }
1681
Christopher Wileydeebfec2016-09-16 11:14:36 -07001682 try {
1683 // Notify that we're tethering (or not) this interface.
1684 // This is how data saver for instance knows if the user explicitly
1685 // turned on tethering (thus keeping us from being in data saver mode).
1686 mPolicyManager.onTetheringChanged(iface, state == IControlsTethering.STATE_TETHERED);
1687 } catch (RemoteException e) {
1688 // Not really very much we can do here.
1689 }
1690
Yohei, Oshima977aad52016-12-08 13:39:20 +09001691 // If TetherMasterSM is in ErrorState, TetherMasterSM stays there.
1692 // Thus we give a chance for TetherMasterSM to recover to InitialState
1693 // by sending CMD_CLEAR_ERROR
1694 if (error == ConnectivityManager.TETHER_ERROR_MASTER_ERROR) {
1695 mTetherMasterSM.sendMessage(TetherMasterSM.CMD_CLEAR_ERROR, who);
1696 }
Erik Klineea9cc482017-03-10 19:35:34 +09001697 int which;
Christopher Wileyd985dde2016-05-31 10:44:35 -07001698 switch (state) {
1699 case IControlsTethering.STATE_UNAVAILABLE:
1700 case IControlsTethering.STATE_AVAILABLE:
Erik Klineea9cc482017-03-10 19:35:34 +09001701 which = TetherMasterSM.EVENT_IFACE_SERVING_STATE_INACTIVE;
Christopher Wileyd985dde2016-05-31 10:44:35 -07001702 break;
1703 case IControlsTethering.STATE_TETHERED:
Erik Klineea9cc482017-03-10 19:35:34 +09001704 case IControlsTethering.STATE_LOCAL_HOTSPOT:
1705 which = TetherMasterSM.EVENT_IFACE_SERVING_STATE_ACTIVE;
Christopher Wileyd985dde2016-05-31 10:44:35 -07001706 break;
Erik Klineea9cc482017-03-10 19:35:34 +09001707 default:
1708 Log.wtf(TAG, "Unknown interface state: " + state);
1709 return;
Christopher Wileyd985dde2016-05-31 10:44:35 -07001710 }
Erik Klineea9cc482017-03-10 19:35:34 +09001711 mTetherMasterSM.sendMessage(which, state, 0, who);
Christopher Wileyd985dde2016-05-31 10:44:35 -07001712 sendTetherStateChangedBroadcast();
1713 }
1714
1715 private void trackNewTetherableInterface(String iface, int interfaceType) {
1716 TetherState tetherState;
1717 tetherState = new TetherState(new TetherInterfaceStateMachine(iface, mLooper,
Lorenzo Colittif537c5e2016-10-28 17:45:55 +09001718 interfaceType, mNMService, mStatsService, this,
1719 new IPv6TetheringInterfaceServices(iface, mNMService)));
Christopher Wileyd985dde2016-05-31 10:44:35 -07001720 mTetherStates.put(iface, tetherState);
Hugo Benichib55fb222017-03-10 14:20:57 +09001721 tetherState.stateMachine.start();
Christopher Wileye03fb442016-05-18 13:45:20 -07001722 }
Erik Kline3e756652017-01-17 13:42:19 +09001723
1724 private static String[] copy(String[] strarray) {
1725 return Arrays.copyOf(strarray, strarray.length);
1726 }
Robert Greenwaltd0e18ff2010-01-26 11:40:34 -08001727}