blob: c91e1a12078e90f4fb3213e366a4e4a1b565212a [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 Klinec438e302017-07-04 22:02:49 +090019import static android.hardware.usb.UsbManager.USB_CONFIGURED;
Erik Kline2e88b5e2017-01-18 11:57:45 +090020import static android.hardware.usb.UsbManager.USB_CONNECTED;
21import static android.hardware.usb.UsbManager.USB_FUNCTION_RNDIS;
Erik Kline7a26ba32018-03-09 14:18:02 +090022import static android.net.ConnectivityManager.ACTION_TETHER_STATE_CHANGED;
23import static android.net.ConnectivityManager.CONNECTIVITY_ACTION;
24import static android.net.ConnectivityManager.EXTRA_ACTIVE_LOCAL_ONLY;
25import static android.net.ConnectivityManager.EXTRA_ACTIVE_TETHER;
Erik Kline7a26ba32018-03-09 14:18:02 +090026import static android.net.ConnectivityManager.EXTRA_AVAILABLE_TETHER;
27import static android.net.ConnectivityManager.EXTRA_ERRORED_TETHER;
28import static android.net.ConnectivityManager.EXTRA_NETWORK_INFO;
Erik Kline7a26ba32018-03-09 14:18:02 +090029import static android.net.ConnectivityManager.TETHERING_BLUETOOTH;
30import static android.net.ConnectivityManager.TETHERING_INVALID;
31import static android.net.ConnectivityManager.TETHERING_USB;
32import static android.net.ConnectivityManager.TETHERING_WIFI;
markchienb6eb2c22018-07-18 14:29:20 +080033import static android.net.ConnectivityManager.TETHER_ERROR_MASTER_ERROR;
34import static android.net.ConnectivityManager.TETHER_ERROR_NO_ERROR;
35import static android.net.ConnectivityManager.TETHER_ERROR_SERVICE_UNAVAIL;
36import static android.net.ConnectivityManager.TETHER_ERROR_UNAVAIL_IFACE;
37import static android.net.ConnectivityManager.TETHER_ERROR_UNKNOWN_IFACE;
Erik Kline2efb8272017-05-31 15:53:53 +090038import static android.net.wifi.WifiManager.EXTRA_WIFI_AP_INTERFACE_NAME;
39import static android.net.wifi.WifiManager.EXTRA_WIFI_AP_MODE;
Erik Kline2e88b5e2017-01-18 11:57:45 +090040import static android.net.wifi.WifiManager.EXTRA_WIFI_AP_STATE;
Erik Kline2efb8272017-05-31 15:53:53 +090041import static android.net.wifi.WifiManager.IFACE_IP_MODE_CONFIGURATION_ERROR;
42import static android.net.wifi.WifiManager.IFACE_IP_MODE_LOCAL_ONLY;
43import static android.net.wifi.WifiManager.IFACE_IP_MODE_TETHERED;
44import static android.net.wifi.WifiManager.IFACE_IP_MODE_UNSPECIFIED;
Erik Kline2e88b5e2017-01-18 11:57:45 +090045import static android.net.wifi.WifiManager.WIFI_AP_STATE_DISABLED;
Erik Klinede637722017-10-12 22:16:01 +090046import static android.telephony.CarrierConfigManager.ACTION_CARRIER_CONFIG_CHANGED;
markchienb6eb2c22018-07-18 14:29:20 +080047
Erik Kline7747fd42017-05-12 16:52:48 +090048import static com.android.server.ConnectivityService.SHORT_ARG;
Erik Kline2e88b5e2017-01-18 11:57:45 +090049
Robert Greenwaltd0e18ff2010-01-26 11:40:34 -080050import android.app.Notification;
51import android.app.NotificationManager;
52import android.app.PendingIntent;
Jeremy Klein36c7aa02016-01-22 14:11:45 -080053import android.bluetooth.BluetoothAdapter;
54import android.bluetooth.BluetoothPan;
55import android.bluetooth.BluetoothProfile;
56import android.bluetooth.BluetoothProfile.ServiceListener;
Robert Greenwaltd0e18ff2010-01-26 11:40:34 -080057import android.content.BroadcastReceiver;
Robert Greenwaltd0e18ff2010-01-26 11:40:34 -080058import android.content.Context;
59import android.content.Intent;
60import android.content.IntentFilter;
61import android.content.res.Resources;
Mike Lockwoodc4308f02011-03-01 08:04:54 -080062import android.hardware.usb.UsbManager;
Christopher Wileydeebfec2016-09-16 11:14:36 -070063import android.net.INetworkPolicyManager;
Jeff Sharkey367d15a2011-09-22 14:59:51 -070064import android.net.INetworkStatsService;
Erik Kline3a5278f2017-06-24 19:29:10 +090065import android.net.IpPrefix;
Erik Klineed962a82017-07-06 19:49:35 +090066import android.net.LinkAddress;
Robert Greenwalt05d89362011-01-23 16:04:05 -080067import android.net.LinkProperties;
Lorenzo Colittib57edc52014-08-22 17:10:50 -070068import android.net.Network;
Robert Greenwalt2a091d72010-02-11 18:18:40 -080069import android.net.NetworkInfo;
Erik Kline6ff17f72015-12-10 20:42:12 +090070import android.net.NetworkState;
Robert Greenwalte5903732011-02-22 16:00:42 -080071import android.net.NetworkUtils;
markchienb6eb2c22018-07-18 14:29:20 +080072import android.net.ip.IpServer;
Remi NGUYEN VAN6c02f992018-03-09 14:07:18 +090073import android.net.util.InterfaceSet;
Erik Kline32179ff2017-07-04 18:28:11 +090074import android.net.util.PrefixUtils;
Erik Kline7747fd42017-05-12 16:52:48 +090075import android.net.util.SharedLog;
Erik Klinede637722017-10-12 22:16:01 +090076import android.net.util.VersionedBroadcastListener;
Jeremy Klein36c7aa02016-01-22 14:11:45 -080077import android.net.wifi.WifiManager;
Robert Greenwalt2a091d72010-02-11 18:18:40 -080078import android.os.Binder;
Jeremy Klein36c7aa02016-01-22 14:11:45 -080079import android.os.Bundle;
Erik Klinef4b6e342017-04-25 19:19:59 +090080import android.os.Handler;
Robert Greenwaltd0e18ff2010-01-26 11:40:34 -080081import android.os.INetworkManagementService;
Robert Greenwaltdfadaea2010-03-11 15:03:08 -080082import android.os.Looper;
Robert Greenwalt2a091d72010-02-11 18:18:40 -080083import android.os.Message;
Jeremy Klein36c7aa02016-01-22 14:11:45 -080084import android.os.Parcel;
Christopher Wileydeebfec2016-09-16 11:14:36 -070085import android.os.RemoteException;
Jeremy Klein36c7aa02016-01-22 14:11:45 -080086import android.os.ResultReceiver;
Dianne Hackborn5ac72a22012-08-29 18:32:08 -070087import android.os.UserHandle;
Alexandru-Andrei Rotarufa6d5c52017-07-18 16:49:22 +010088import android.os.UserManager;
89import android.os.UserManagerInternal;
90import android.os.UserManagerInternal.UserRestrictionsListener;
Lorenzo Colitticd63d242016-04-10 15:39:53 +090091import android.text.TextUtils;
Christopher Wileye9490392016-05-26 15:57:29 -070092import android.util.ArrayMap;
Robert Greenwaltd0e18ff2010-01-26 11:40:34 -080093import android.util.Log;
Lorenzo Colitticd63d242016-04-10 15:39:53 +090094import android.util.SparseArray;
Robert Greenwaltd0e18ff2010-01-26 11:40:34 -080095
Christopher Wiley497c1472016-10-11 13:26:03 -070096import com.android.internal.annotations.VisibleForTesting;
Chris Wren282cfef2017-03-27 15:01:44 -040097import com.android.internal.messages.nano.SystemMessageProto.SystemMessage;
Geoffrey Pitschaf759c52017-02-15 09:35:38 -050098import com.android.internal.notification.SystemNotificationChannels;
markchien293422f2019-01-08 23:52:21 +080099import com.android.internal.telephony.TelephonyIntents;
Jeff Sharkeyfe9a53b2017-03-31 14:08:23 -0600100import com.android.internal.util.DumpUtils;
Jeremy Klein36c7aa02016-01-22 14:11:45 -0800101import com.android.internal.util.IndentingPrintWriter;
Lorenzo Colitticd63d242016-04-10 15:39:53 +0900102import com.android.internal.util.MessageUtils;
103import com.android.internal.util.Protocol;
Wink Saville64c42ca2011-04-18 14:55:10 -0700104import com.android.internal.util.State;
105import com.android.internal.util.StateMachine;
Alexandru-Andrei Rotarufa6d5c52017-07-18 16:49:22 +0100106import com.android.server.LocalServices;
markchienb6eb2c22018-07-18 14:29:20 +0800107import com.android.server.connectivity.tethering.EntitlementManager;
Erik Kline1eb8c692016-07-08 17:21:26 +0900108import com.android.server.connectivity.tethering.IPv6TetheringCoordinator;
Erik Klinee0cce212017-03-06 14:05:23 +0900109import com.android.server.connectivity.tethering.OffloadController;
Hugo Benichib55fb222017-03-10 14:20:57 +0900110import com.android.server.connectivity.tethering.TetheringConfiguration;
Erik Kline47222fc2017-04-30 19:36:15 +0900111import com.android.server.connectivity.tethering.TetheringDependencies;
Remi NGUYEN VAN6c02f992018-03-09 14:07:18 +0900112import com.android.server.connectivity.tethering.TetheringInterfaceUtils;
Erik Kline9bba3402017-01-13 16:46:52 +0900113import com.android.server.connectivity.tethering.UpstreamNetworkMonitor;
Lorenzo Colittif4e90ea2013-10-31 23:30:47 +0900114import com.android.server.net.BaseNetworkObserver;
Robert Greenwalt2a091d72010-02-11 18:18:40 -0800115
116import java.io.FileDescriptor;
117import java.io.PrintWriter;
Jeremy Klein36c7aa02016-01-22 14:11:45 -0800118import java.net.InetAddress;
Robert Greenwaltd0e18ff2010-01-26 11:40:34 -0800119import java.util.ArrayList;
Lorenzo Colittib57edc52014-08-22 17:10:50 -0700120import java.util.Arrays;
Robert Greenwaltccf83af12011-06-02 17:30:47 -0700121import java.util.Collection;
Erik Klineea9cc482017-03-10 19:35:34 +0900122import java.util.HashSet;
Erik Kline3a5278f2017-06-24 19:29:10 +0900123import java.util.Set;
Robert Greenwalt2ffe4122014-12-12 12:22:31 -0800124
Robert Greenwaltd0e18ff2010-01-26 11:40:34 -0800125/**
126 * @hide
Robert Greenwalt2a091d72010-02-11 18:18:40 -0800127 *
Christopher Wiley3b1d9222016-05-20 16:44:04 -0700128 * This class holds much of the business logic to allow Android devices
129 * to act as IP gateways via USB, BT, and WiFi interfaces.
Robert Greenwaltd0e18ff2010-01-26 11:40:34 -0800130 */
Erik Kline6e9a1012017-06-06 19:24:21 +0900131public class Tethering extends BaseNetworkObserver {
Robert Greenwaltd0e18ff2010-01-26 11:40:34 -0800132
Hugo Benichib55fb222017-03-10 14:20:57 +0900133 private final static String TAG = Tethering.class.getSimpleName();
Joe Onorato12acbd72016-02-01 17:49:31 -0800134 private final static boolean DBG = false;
Robert Greenwaltfd1be2b2011-11-11 12:30:19 -0800135 private final static boolean VDBG = false;
Robert Greenwaltd0e18ff2010-01-26 11:40:34 -0800136
Lorenzo Colitticd63d242016-04-10 15:39:53 +0900137 private static final Class[] messageClasses = {
Erik Kline7a4ccc62018-08-27 17:26:47 +0900138 Tethering.class, TetherMasterSM.class, IpServer.class
Lorenzo Colitticd63d242016-04-10 15:39:53 +0900139 };
140 private static final SparseArray<String> sMagicDecoderRing =
141 MessageUtils.findMessageNames(messageClasses);
142
Hugo Benichib55fb222017-03-10 14:20:57 +0900143 private static class TetherState {
Erik Kline7a4ccc62018-08-27 17:26:47 +0900144 public final IpServer ipServer;
Hugo Benichib55fb222017-03-10 14:20:57 +0900145 public int lastState;
146 public int lastError;
Erik Klineea9cc482017-03-10 19:35:34 +0900147
Erik Kline7a4ccc62018-08-27 17:26:47 +0900148 public TetherState(IpServer ipServer) {
149 this.ipServer = ipServer;
Hugo Benichib55fb222017-03-10 14:20:57 +0900150 // Assume all state machines start out available and with no errors.
Erik Kline7a4ccc62018-08-27 17:26:47 +0900151 lastState = IpServer.STATE_AVAILABLE;
Erik Kline7a26ba32018-03-09 14:18:02 +0900152 lastError = TETHER_ERROR_NO_ERROR;
Hugo Benichib55fb222017-03-10 14:20:57 +0900153 }
Erik Klineea9cc482017-03-10 19:35:34 +0900154
155 public boolean isCurrentlyServing() {
156 switch (lastState) {
Erik Kline7a4ccc62018-08-27 17:26:47 +0900157 case IpServer.STATE_TETHERED:
158 case IpServer.STATE_LOCAL_ONLY:
Erik Klineea9cc482017-03-10 19:35:34 +0900159 return true;
160 default:
161 return false;
162 }
163 }
Hugo Benichib55fb222017-03-10 14:20:57 +0900164 }
Robert Greenwaltccf83af12011-06-02 17:30:47 -0700165
Erik Kline7747fd42017-05-12 16:52:48 +0900166 private final SharedLog mLog = new SharedLog(TAG);
Erik Kline1fdc2e22017-05-08 17:56:35 +0900167
Robert Greenwaltb4453622011-11-03 16:01:40 -0700168 // used to synchronize public access to members
Erik Kline6ff17f72015-12-10 20:42:12 +0900169 private final Object mPublicSync;
Hugo Benichib55fb222017-03-10 14:20:57 +0900170 private final Context mContext;
171 private final ArrayMap<String, TetherState> mTetherStates;
172 private final BroadcastReceiver mStateReceiver;
Jeff Sharkey367d15a2011-09-22 14:59:51 -0700173 private final INetworkManagementService mNMService;
174 private final INetworkStatsService mStatsService;
Christopher Wileydeebfec2016-09-16 11:14:36 -0700175 private final INetworkPolicyManager mPolicyManager;
Erik Kline6ff17f72015-12-10 20:42:12 +0900176 private final Looper mLooper;
Erik Kline6ff17f72015-12-10 20:42:12 +0900177 private final StateMachine mTetherMasterSM;
Erik Klinee0cce212017-03-06 14:05:23 +0900178 private final OffloadController mOffloadController;
Erik Kline6ff17f72015-12-10 20:42:12 +0900179 private final UpstreamNetworkMonitor mUpstreamNetworkMonitor;
Erik Kline6e9a1012017-06-06 19:24:21 +0900180 // TODO: Figure out how to merge this and other downstream-tracking objects
181 // into a single coherent structure.
Erik Kline7a4ccc62018-08-27 17:26:47 +0900182 private final HashSet<IpServer> mForwardedDownstreams;
Erik Klinede637722017-10-12 22:16:01 +0900183 private final VersionedBroadcastListener mCarrierConfigChange;
markchien293422f2019-01-08 23:52:21 +0800184 private final VersionedBroadcastListener mDefaultSubscriptionChange;
Remi NGUYEN VAN5d0dc452018-03-15 11:57:14 +0900185 private final TetheringDependencies mDeps;
markchienb6eb2c22018-07-18 14:29:20 +0800186 private final EntitlementManager mEntitlementMgr;
Robert Greenwalta599fe7c2010-03-08 18:30:14 -0800187
Hugo Benichib55fb222017-03-10 14:20:57 +0900188 private volatile TetheringConfiguration mConfig;
Remi NGUYEN VAN6c02f992018-03-09 14:07:18 +0900189 private InterfaceSet mCurrentUpstreamIfaceSet;
Chris Wren1ce4b6d2015-06-11 10:19:43 -0400190 private Notification.Builder mTetheredNotificationBuilder;
191 private int mLastNotificationId;
Erik Klineea9cc482017-03-10 19:35:34 +0900192
Mike Lockwood6c2260b2011-07-19 13:04:47 -0700193 private boolean mRndisEnabled; // track the RNDIS function enabled state
Erik Kline6e9a1012017-06-06 19:24:21 +0900194 // True iff. WiFi tethering should be started when soft AP is ready.
Christopher Wiley5c0b10a2016-05-31 14:43:08 -0700195 private boolean mWifiTetherRequested;
196
Jeff Sharkey367d15a2011-09-22 14:59:51 -0700197 public Tethering(Context context, INetworkManagementService nmService,
Christopher Wiley497c1472016-10-11 13:26:03 -0700198 INetworkStatsService statsService, INetworkPolicyManager policyManager,
Erik Kline47222fc2017-04-30 19:36:15 +0900199 Looper looper, MockableSystemProperties systemProperties,
200 TetheringDependencies deps) {
Erik Kline7747fd42017-05-12 16:52:48 +0900201 mLog.mark("constructed");
Robert Greenwaltd0e18ff2010-01-26 11:40:34 -0800202 mContext = context;
Chia-chi Yehc9338302011-05-11 16:35:13 -0700203 mNMService = nmService;
Jeff Sharkey367d15a2011-09-22 14:59:51 -0700204 mStatsService = statsService;
Christopher Wileydeebfec2016-09-16 11:14:36 -0700205 mPolicyManager = policyManager;
Christopher Wiley497c1472016-10-11 13:26:03 -0700206 mLooper = looper;
Remi NGUYEN VAN5d0dc452018-03-15 11:57:14 +0900207 mDeps = deps;
Robert Greenwaltd0e18ff2010-01-26 11:40:34 -0800208
Robert Greenwaltb4453622011-11-03 16:01:40 -0700209 mPublicSync = new Object();
210
Christopher Wileyd985dde2016-05-31 10:44:35 -0700211 mTetherStates = new ArrayMap<>();
Robert Greenwaltd0e18ff2010-01-26 11:40:34 -0800212
Remi NGUYEN VAN5d0dc452018-03-15 11:57:14 +0900213 mTetherMasterSM = new TetherMasterSM("TetherMaster", mLooper, deps);
Robert Greenwalt2a091d72010-02-11 18:18:40 -0800214 mTetherMasterSM.start();
215
Erik Klinef4b6e342017-04-25 19:19:59 +0900216 final Handler smHandler = mTetherMasterSM.getHandler();
217 mOffloadController = new OffloadController(smHandler,
Erik Kline7a26ba32018-03-09 14:18:02 +0900218 mDeps.getOffloadHardwareInterface(smHandler, mLog),
Lorenzo Colitti5a7dea12017-07-12 15:48:07 +0900219 mContext.getContentResolver(), mNMService,
Erik Klinef4b6e342017-04-25 19:19:59 +0900220 mLog);
Remi NGUYEN VAN5d0dc452018-03-15 11:57:14 +0900221 mUpstreamNetworkMonitor = deps.getUpstreamNetworkMonitor(mContext, mTetherMasterSM, mLog,
222 TetherMasterSM.EVENT_UPSTREAM_CALLBACK);
Erik Klineea9cc482017-03-10 19:35:34 +0900223 mForwardedDownstreams = new HashSet<>();
Erik Klinede637722017-10-12 22:16:01 +0900224
225 IntentFilter filter = new IntentFilter();
226 filter.addAction(ACTION_CARRIER_CONFIG_CHANGED);
markchienf2731272019-01-16 17:44:13 +0800227 mEntitlementMgr = mDeps.getEntitlementManager(mContext, mTetherMasterSM,
228 mLog, systemProperties);
Erik Klinede637722017-10-12 22:16:01 +0900229 mCarrierConfigChange = new VersionedBroadcastListener(
230 "CarrierConfigChangeListener", mContext, smHandler, filter,
231 (Intent ignored) -> {
232 mLog.log("OBSERVED carrier config change");
Erik Kline80b7a9f2018-02-28 15:01:35 +0900233 updateConfiguration();
markchienb6eb2c22018-07-18 14:29:20 +0800234 mEntitlementMgr.reevaluateSimCardProvisioning();
Erik Klinede637722017-10-12 22:16:01 +0900235 });
Erik Kline6ff17f72015-12-10 20:42:12 +0900236
markchien293422f2019-01-08 23:52:21 +0800237 filter = new IntentFilter();
238 filter.addAction(TelephonyIntents.ACTION_DEFAULT_DATA_SUBSCRIPTION_CHANGED);
239 mDefaultSubscriptionChange = new VersionedBroadcastListener(
240 "DefaultSubscriptionChangeListener", mContext, smHandler, filter,
241 (Intent ignored) -> {
242 mLog.log("OBSERVED default data subscription change");
243 updateConfiguration();
244 mEntitlementMgr.reevaluateSimCardProvisioning();
245 });
Robert Greenwaltbb51d9f2010-03-12 10:23:35 -0800246 mStateReceiver = new StateReceiver();
Erik Kline80b7a9f2018-02-28 15:01:35 +0900247
248 // Load tethering configuration.
249 updateConfiguration();
250
251 startStateMachineUpdaters();
252 }
253
254 private void startStateMachineUpdaters() {
255 mCarrierConfigChange.startListening();
markchien293422f2019-01-08 23:52:21 +0800256 mDefaultSubscriptionChange.startListening();
Erik Kline80b7a9f2018-02-28 15:01:35 +0900257
258 final Handler handler = mTetherMasterSM.getHandler();
259 IntentFilter filter = new IntentFilter();
Mike Lockwood770126a2010-12-09 22:30:37 -0800260 filter.addAction(UsbManager.ACTION_USB_STATE);
Erik Kline7a26ba32018-03-09 14:18:02 +0900261 filter.addAction(CONNECTIVITY_ACTION);
Christopher Wiley5c0b10a2016-05-31 14:43:08 -0700262 filter.addAction(WifiManager.WIFI_AP_STATE_CHANGED_ACTION);
Robert Greenwaltc13368b2013-07-18 14:24:42 -0700263 filter.addAction(Intent.ACTION_CONFIGURATION_CHANGED);
Erik Kline80b7a9f2018-02-28 15:01:35 +0900264 mContext.registerReceiver(mStateReceiver, filter, null, handler);
Robert Greenwaltbb51d9f2010-03-12 10:23:35 -0800265
266 filter = new IntentFilter();
Robert Greenwalt030f5e12010-03-10 16:41:03 -0800267 filter.addAction(Intent.ACTION_MEDIA_SHARED);
268 filter.addAction(Intent.ACTION_MEDIA_UNSHARED);
Robert Greenwaltbb51d9f2010-03-12 10:23:35 -0800269 filter.addDataScheme("file");
Erik Kline80b7a9f2018-02-28 15:01:35 +0900270 mContext.registerReceiver(mStateReceiver, filter, null, handler);
Robert Greenwaltbb51d9f2010-03-12 10:23:35 -0800271
Erik Kline80b7a9f2018-02-28 15:01:35 +0900272 final UserManagerInternal umi = LocalServices.getService(UserManagerInternal.class);
273 // This check is useful only for some unit tests; example: ConnectivityServiceTest.
274 if (umi != null) {
275 umi.addUserRestrictionsListener(new TetheringUserRestrictionListener(this));
Alexandru-Andrei Rotarufa6d5c52017-07-18 16:49:22 +0100276 }
Robert Greenwalt49348e72011-10-21 16:54:26 -0700277 }
278
Erik Kline216af6d2017-04-27 20:57:23 +0900279 private WifiManager getWifiManager() {
280 return (WifiManager) mContext.getSystemService(Context.WIFI_SERVICE);
281 }
282
Erik Kline93c4afa2017-06-04 11:36:01 +0900283 // NOTE: This is always invoked on the mLooper thread.
Erik Klined781fba2017-01-23 13:01:58 +0900284 private void updateConfiguration() {
markchien293422f2019-01-08 23:52:21 +0800285 final int subId = mDeps.getDefaultDataSubscriptionId();
286 mConfig = new TetheringConfiguration(mContext, mLog, subId);
Erik Kline93c4afa2017-06-04 11:36:01 +0900287 mUpstreamNetworkMonitor.updateMobileRequiresDun(mConfig.isDunRequired);
markchienb6eb2c22018-07-18 14:29:20 +0800288 mEntitlementMgr.updateConfiguration(mConfig);
Robert Greenwaltd0e18ff2010-01-26 11:40:34 -0800289 }
290
Erik Kline6ee73da2017-07-08 20:36:37 +0900291 private void maybeUpdateConfiguration() {
292 final int dunCheck = TetheringConfiguration.checkDunRequired(mContext);
293 if (dunCheck == mConfig.dunCheck) return;
294 updateConfiguration();
295 }
296
Erik Kline0c9cb2b2015-11-20 17:47:08 +0900297 @Override
Mike J. Chen6143f5f2011-06-23 15:17:51 -0700298 public void interfaceStatusChanged(String iface, boolean up) {
Lorenzo Colitticd63d242016-04-10 15:39:53 +0900299 // Never called directly: only called from interfaceLinkStateChanged.
300 // See NetlinkHandler.cpp:71.
Wink Savillec9acde92011-09-21 11:05:43 -0700301 if (VDBG) Log.d(TAG, "interfaceStatusChanged " + iface + ", " + up);
Robert Greenwaltb4453622011-11-03 16:01:40 -0700302 synchronized (mPublicSync) {
Mike J. Chen6143f5f2011-06-23 15:17:51 -0700303 if (up) {
Erik Kline4dd9bb82017-04-26 11:11:07 +0900304 maybeTrackNewInterfaceLocked(iface);
Robert Greenwalt2a091d72010-02-11 18:18:40 -0800305 } else {
Erik Kline7a26ba32018-03-09 14:18:02 +0900306 if (ifaceNameToType(iface) == TETHERING_BLUETOOTH) {
Erik Kline4dd9bb82017-04-26 11:11:07 +0900307 stopTrackingInterfaceLocked(iface);
Christopher Wileyd30aaeb2016-07-08 09:33:50 -0700308 } else {
309 // Ignore usb0 down after enabling RNDIS.
310 // We will handle disconnect in interfaceRemoved.
311 // Similarly, ignore interface down for WiFi. We monitor WiFi AP status
312 // through the WifiManager.WIFI_AP_STATE_CHANGED_ACTION intent.
313 if (VDBG) Log.d(TAG, "ignore interface down for " + iface);
Robert Greenwalt2a091d72010-02-11 18:18:40 -0800314 }
315 }
316 }
Robert Greenwaltd0e18ff2010-01-26 11:40:34 -0800317 }
318
Erik Kline0c9cb2b2015-11-20 17:47:08 +0900319 @Override
Mike J. Chenf59c7d02011-06-23 15:33:15 -0700320 public void interfaceLinkStateChanged(String iface, boolean up) {
Irfan Sheriff23eb2972011-07-22 15:21:10 -0700321 interfaceStatusChanged(iface, up);
Mike J. Chenf59c7d02011-06-23 15:33:15 -0700322 }
323
Christopher Wiley5c0b10a2016-05-31 14:43:08 -0700324 private int ifaceNameToType(String iface) {
Erik Kline3e756652017-01-17 13:42:19 +0900325 final TetheringConfiguration cfg = mConfig;
326
327 if (cfg.isWifi(iface)) {
Erik Kline7a26ba32018-03-09 14:18:02 +0900328 return TETHERING_WIFI;
Erik Kline3e756652017-01-17 13:42:19 +0900329 } else if (cfg.isUsb(iface)) {
Erik Kline7a26ba32018-03-09 14:18:02 +0900330 return TETHERING_USB;
Erik Kline3e756652017-01-17 13:42:19 +0900331 } else if (cfg.isBluetooth(iface)) {
Erik Kline7a26ba32018-03-09 14:18:02 +0900332 return TETHERING_BLUETOOTH;
Christopher Wiley5c0b10a2016-05-31 14:43:08 -0700333 }
Erik Kline7a26ba32018-03-09 14:18:02 +0900334 return TETHERING_INVALID;
Christopher Wiley5c0b10a2016-05-31 14:43:08 -0700335 }
336
Erik Kline0c9cb2b2015-11-20 17:47:08 +0900337 @Override
Robert Greenwalt2a091d72010-02-11 18:18:40 -0800338 public void interfaceAdded(String iface) {
Wink Savillec9acde92011-09-21 11:05:43 -0700339 if (VDBG) Log.d(TAG, "interfaceAdded " + iface);
Robert Greenwaltb4453622011-11-03 16:01:40 -0700340 synchronized (mPublicSync) {
Erik Kline4dd9bb82017-04-26 11:11:07 +0900341 maybeTrackNewInterfaceLocked(iface);
Robert Greenwaltd0e18ff2010-01-26 11:40:34 -0800342 }
Robert Greenwaltd0e18ff2010-01-26 11:40:34 -0800343 }
344
Erik Kline0c9cb2b2015-11-20 17:47:08 +0900345 @Override
Robert Greenwalt2a091d72010-02-11 18:18:40 -0800346 public void interfaceRemoved(String iface) {
Wink Savillec9acde92011-09-21 11:05:43 -0700347 if (VDBG) Log.d(TAG, "interfaceRemoved " + iface);
Robert Greenwaltb4453622011-11-03 16:01:40 -0700348 synchronized (mPublicSync) {
Erik Kline4dd9bb82017-04-26 11:11:07 +0900349 stopTrackingInterfaceLocked(iface);
Robert Greenwaltd0e18ff2010-01-26 11:40:34 -0800350 }
351 }
352
Hugo Benichib55fb222017-03-10 14:20:57 +0900353 public void startTethering(int type, ResultReceiver receiver, boolean showProvisioningUi) {
markchienb6eb2c22018-07-18 14:29:20 +0800354 mEntitlementMgr.startTethering(type);
355 if (!mEntitlementMgr.isTetherProvisioningRequired()) {
Jeremy Klein36c7aa02016-01-22 14:11:45 -0800356 enableTetheringInternal(type, true, receiver);
357 return;
358 }
359
markchienb6eb2c22018-07-18 14:29:20 +0800360 final ResultReceiver proxyReceiver = getProxyReceiver(type, receiver);
Jeremy Klein36c7aa02016-01-22 14:11:45 -0800361 if (showProvisioningUi) {
markchienb6eb2c22018-07-18 14:29:20 +0800362 mEntitlementMgr.runUiTetherProvisioningAndEnable(type, proxyReceiver);
Jeremy Klein36c7aa02016-01-22 14:11:45 -0800363 } else {
markchienb6eb2c22018-07-18 14:29:20 +0800364 mEntitlementMgr.runSilentTetherProvisioningAndEnable(type, proxyReceiver);
Jeremy Klein36c7aa02016-01-22 14:11:45 -0800365 }
366 }
367
368 public void stopTethering(int type) {
369 enableTetheringInternal(type, false, null);
markchienb6eb2c22018-07-18 14:29:20 +0800370 mEntitlementMgr.stopTethering(type);
371 if (mEntitlementMgr.isTetherProvisioningRequired()) {
372 // There are lurking bugs where the notion of "provisioning required" or
373 // "tethering supported" may change without notifying tethering properly, then
374 // tethering can't shutdown correctly.
375 // TODO: cancel re-check all the time
376 if (mDeps.isTetheringSupported()) {
377 mEntitlementMgr.cancelTetherProvisioningRechecks(type);
378 }
Jeremy Klein36c7aa02016-01-22 14:11:45 -0800379 }
380 }
381
382 /**
Jeremy Klein36c7aa02016-01-22 14:11:45 -0800383 * Enables or disables tethering for the given type. This should only be called once
384 * provisioning has succeeded or is not necessary. It will also schedule provisioning rechecks
385 * for the specified interface.
386 */
387 private void enableTetheringInternal(int type, boolean enable, ResultReceiver receiver) {
markchienb6eb2c22018-07-18 14:29:20 +0800388 boolean isProvisioningRequired = enable && mEntitlementMgr.isTetherProvisioningRequired();
Christopher Wiley5c0b10a2016-05-31 14:43:08 -0700389 int result;
Jeremy Klein36c7aa02016-01-22 14:11:45 -0800390 switch (type) {
Erik Kline7a26ba32018-03-09 14:18:02 +0900391 case TETHERING_WIFI:
Christopher Wiley5c0b10a2016-05-31 14:43:08 -0700392 result = setWifiTethering(enable);
Erik Kline7a26ba32018-03-09 14:18:02 +0900393 if (isProvisioningRequired && result == TETHER_ERROR_NO_ERROR) {
markchienb6eb2c22018-07-18 14:29:20 +0800394 mEntitlementMgr.scheduleProvisioningRechecks(type);
Jeremy Klein36c7aa02016-01-22 14:11:45 -0800395 }
Christopher Wiley5c0b10a2016-05-31 14:43:08 -0700396 sendTetherResult(receiver, result);
Jeremy Klein36c7aa02016-01-22 14:11:45 -0800397 break;
Erik Kline7a26ba32018-03-09 14:18:02 +0900398 case TETHERING_USB:
Christopher Wiley5c0b10a2016-05-31 14:43:08 -0700399 result = setUsbTethering(enable);
Erik Kline7a26ba32018-03-09 14:18:02 +0900400 if (isProvisioningRequired && result == TETHER_ERROR_NO_ERROR) {
markchienb6eb2c22018-07-18 14:29:20 +0800401 mEntitlementMgr.scheduleProvisioningRechecks(type);
Jeremy Klein36c7aa02016-01-22 14:11:45 -0800402 }
403 sendTetherResult(receiver, result);
404 break;
Erik Kline7a26ba32018-03-09 14:18:02 +0900405 case TETHERING_BLUETOOTH:
Jeremy Klein36c7aa02016-01-22 14:11:45 -0800406 setBluetoothTethering(enable, receiver);
407 break;
408 default:
409 Log.w(TAG, "Invalid tether type.");
Erik Kline7a26ba32018-03-09 14:18:02 +0900410 sendTetherResult(receiver, TETHER_ERROR_UNKNOWN_IFACE);
Jeremy Klein36c7aa02016-01-22 14:11:45 -0800411 }
412 }
413
414 private void sendTetherResult(ResultReceiver receiver, int result) {
415 if (receiver != null) {
416 receiver.send(result, null);
417 }
418 }
419
Christopher Wiley5c0b10a2016-05-31 14:43:08 -0700420 private int setWifiTethering(final boolean enable) {
Erik Kline7a26ba32018-03-09 14:18:02 +0900421 int rval = TETHER_ERROR_MASTER_ERROR;
Erik Kline256be782017-04-29 13:20:04 +0900422 final long ident = Binder.clearCallingIdentity();
423 try {
424 synchronized (mPublicSync) {
425 mWifiTetherRequested = enable;
426 final WifiManager mgr = getWifiManager();
427 if ((enable && mgr.startSoftAp(null /* use existing wifi config */)) ||
428 (!enable && mgr.stopSoftAp())) {
Erik Kline7a26ba32018-03-09 14:18:02 +0900429 rval = TETHER_ERROR_NO_ERROR;
Erik Kline256be782017-04-29 13:20:04 +0900430 }
Christopher Wiley5c0b10a2016-05-31 14:43:08 -0700431 }
Erik Kline256be782017-04-29 13:20:04 +0900432 } finally {
433 Binder.restoreCallingIdentity(ident);
Christopher Wiley5c0b10a2016-05-31 14:43:08 -0700434 }
Erik Kline256be782017-04-29 13:20:04 +0900435 return rval;
Christopher Wiley5c0b10a2016-05-31 14:43:08 -0700436 }
437
Jeremy Klein36c7aa02016-01-22 14:11:45 -0800438 private void setBluetoothTethering(final boolean enable, final ResultReceiver receiver) {
439 final BluetoothAdapter adapter = BluetoothAdapter.getDefaultAdapter();
440 if (adapter == null || !adapter.isEnabled()) {
441 Log.w(TAG, "Tried to enable bluetooth tethering with null or disabled adapter. null: " +
442 (adapter == null));
Erik Kline7a26ba32018-03-09 14:18:02 +0900443 sendTetherResult(receiver, TETHER_ERROR_SERVICE_UNAVAIL);
Jeremy Klein36c7aa02016-01-22 14:11:45 -0800444 return;
445 }
446
447 adapter.getProfileProxy(mContext, new ServiceListener() {
448 @Override
449 public void onServiceDisconnected(int profile) { }
450
451 @Override
452 public void onServiceConnected(int profile, BluetoothProfile proxy) {
453 ((BluetoothPan) proxy).setBluetoothTethering(enable);
454 // TODO: Enabling bluetooth tethering can fail asynchronously here.
455 // We should figure out a way to bubble up that failure instead of sending success.
Erik Kline7a26ba32018-03-09 14:18:02 +0900456 final int result = (((BluetoothPan) proxy).isTetheringOn() == enable)
457 ? TETHER_ERROR_NO_ERROR
458 : TETHER_ERROR_MASTER_ERROR;
Jeremy Klein36c7aa02016-01-22 14:11:45 -0800459 sendTetherResult(receiver, result);
markchienb6eb2c22018-07-18 14:29:20 +0800460 if (enable && mEntitlementMgr.isTetherProvisioningRequired()) {
461 mEntitlementMgr.scheduleProvisioningRechecks(TETHERING_BLUETOOTH);
Jeremy Klein36c7aa02016-01-22 14:11:45 -0800462 }
463 adapter.closeProfileProxy(BluetoothProfile.PAN, proxy);
464 }
465 }, BluetoothProfile.PAN);
466 }
467
Jeremy Klein36c7aa02016-01-22 14:11:45 -0800468 /**
Erik Klinefdd41112017-01-12 17:02:51 +0900469 * Creates a proxy {@link ResultReceiver} which enables tethering if the provisioning result
470 * is successful before firing back up to the wrapped receiver.
Jeremy Klein36c7aa02016-01-22 14:11:45 -0800471 *
472 * @param type The type of tethering being enabled.
473 * @param receiver A ResultReceiver which will be called back with an int resultCode.
474 * @return The proxy receiver.
475 */
476 private ResultReceiver getProxyReceiver(final int type, final ResultReceiver receiver) {
477 ResultReceiver rr = new ResultReceiver(null) {
478 @Override
479 protected void onReceiveResult(int resultCode, Bundle resultData) {
480 // If provisioning is successful, enable tethering, otherwise just send the error.
Erik Kline7a26ba32018-03-09 14:18:02 +0900481 if (resultCode == TETHER_ERROR_NO_ERROR) {
Jeremy Klein36c7aa02016-01-22 14:11:45 -0800482 enableTetheringInternal(type, true, receiver);
483 } else {
484 sendTetherResult(receiver, resultCode);
485 }
markchienf2731272019-01-16 17:44:13 +0800486 mEntitlementMgr.updateEntitlementCacheValue(type, resultCode);
Jeremy Klein36c7aa02016-01-22 14:11:45 -0800487 }
488 };
489
490 // The following is necessary to avoid unmarshalling issues when sending the receiver
Erik Kline6ff17f72015-12-10 20:42:12 +0900491 // across processes.
Jeremy Klein36c7aa02016-01-22 14:11:45 -0800492 Parcel parcel = Parcel.obtain();
493 rr.writeToParcel(parcel,0);
494 parcel.setDataPosition(0);
495 ResultReceiver receiverForSending = ResultReceiver.CREATOR.createFromParcel(parcel);
496 parcel.recycle();
497 return receiverForSending;
498 }
499
Robert Greenwalt5a735062010-03-02 17:25:02 -0800500 public int tether(String iface) {
Erik Kline7a4ccc62018-08-27 17:26:47 +0900501 return tether(iface, IpServer.STATE_TETHERED);
Erik Klineea9cc482017-03-10 19:35:34 +0900502 }
503
504 private int tether(String iface, int requestedState) {
Wink Savillec9acde92011-09-21 11:05:43 -0700505 if (DBG) Log.d(TAG, "Tethering " + iface);
Robert Greenwaltb4453622011-11-03 16:01:40 -0700506 synchronized (mPublicSync) {
Christopher Wileyd985dde2016-05-31 10:44:35 -0700507 TetherState tetherState = mTetherStates.get(iface);
508 if (tetherState == null) {
Erik Kline00019f42016-06-30 19:31:46 +0900509 Log.e(TAG, "Tried to Tether an unknown iface: " + iface + ", ignoring");
Erik Kline7a26ba32018-03-09 14:18:02 +0900510 return TETHER_ERROR_UNKNOWN_IFACE;
Christopher Wiley4312a4c2016-05-23 13:58:00 -0700511 }
512 // Ignore the error status of the interface. If the interface is available,
513 // the errors are referring to past tethering attempts anyway.
Erik Kline7a4ccc62018-08-27 17:26:47 +0900514 if (tetherState.lastState != IpServer.STATE_AVAILABLE) {
Erik Kline00019f42016-06-30 19:31:46 +0900515 Log.e(TAG, "Tried to Tether an unavailable iface: " + iface + ", ignoring");
Erik Kline7a26ba32018-03-09 14:18:02 +0900516 return TETHER_ERROR_UNAVAIL_IFACE;
Christopher Wiley4312a4c2016-05-23 13:58:00 -0700517 }
Erik Klineea9cc482017-03-10 19:35:34 +0900518 // NOTE: If a CMD_TETHER_REQUESTED message is already in the TISM's
519 // queue but not yet processed, this will be a no-op and it will not
520 // return an error.
521 //
522 // TODO: reexamine the threading and messaging model.
Erik Kline7a4ccc62018-08-27 17:26:47 +0900523 tetherState.ipServer.sendMessage(IpServer.CMD_TETHER_REQUESTED, requestedState);
Erik Kline7a26ba32018-03-09 14:18:02 +0900524 return TETHER_ERROR_NO_ERROR;
Robert Greenwalt2a091d72010-02-11 18:18:40 -0800525 }
Robert Greenwaltd0e18ff2010-01-26 11:40:34 -0800526 }
527
Robert Greenwalt5a735062010-03-02 17:25:02 -0800528 public int untether(String iface) {
Wink Savillec9acde92011-09-21 11:05:43 -0700529 if (DBG) Log.d(TAG, "Untethering " + iface);
Robert Greenwaltb4453622011-11-03 16:01:40 -0700530 synchronized (mPublicSync) {
Christopher Wileyd985dde2016-05-31 10:44:35 -0700531 TetherState tetherState = mTetherStates.get(iface);
532 if (tetherState == null) {
Christopher Wiley4312a4c2016-05-23 13:58:00 -0700533 Log.e(TAG, "Tried to Untether an unknown iface :" + iface + ", ignoring");
Erik Kline7a26ba32018-03-09 14:18:02 +0900534 return TETHER_ERROR_UNKNOWN_IFACE;
Christopher Wiley4312a4c2016-05-23 13:58:00 -0700535 }
Erik Klineea9cc482017-03-10 19:35:34 +0900536 if (!tetherState.isCurrentlyServing()) {
537 Log.e(TAG, "Tried to untether an inactive iface :" + iface + ", ignoring");
Erik Kline7a26ba32018-03-09 14:18:02 +0900538 return TETHER_ERROR_UNAVAIL_IFACE;
Christopher Wiley4312a4c2016-05-23 13:58:00 -0700539 }
Erik Kline7a4ccc62018-08-27 17:26:47 +0900540 tetherState.ipServer.sendMessage(IpServer.CMD_TETHER_UNREQUESTED);
Erik Kline7a26ba32018-03-09 14:18:02 +0900541 return TETHER_ERROR_NO_ERROR;
Robert Greenwalt2a091d72010-02-11 18:18:40 -0800542 }
Robert Greenwalt5a735062010-03-02 17:25:02 -0800543 }
544
Felipe Leme70c8b9b2016-04-25 14:41:31 -0700545 public void untetherAll() {
Erik Kline7a26ba32018-03-09 14:18:02 +0900546 stopTethering(TETHERING_WIFI);
547 stopTethering(TETHERING_USB);
548 stopTethering(TETHERING_BLUETOOTH);
Felipe Leme70c8b9b2016-04-25 14:41:31 -0700549 }
550
Robert Greenwalt5a735062010-03-02 17:25:02 -0800551 public int getLastTetherError(String iface) {
Robert Greenwaltb4453622011-11-03 16:01:40 -0700552 synchronized (mPublicSync) {
Christopher Wileyd985dde2016-05-31 10:44:35 -0700553 TetherState tetherState = mTetherStates.get(iface);
554 if (tetherState == null) {
Robert Greenwaltb4453622011-11-03 16:01:40 -0700555 Log.e(TAG, "Tried to getLastTetherError on an unknown iface :" + iface +
556 ", ignoring");
Erik Kline7a26ba32018-03-09 14:18:02 +0900557 return TETHER_ERROR_UNKNOWN_IFACE;
Robert Greenwaltb4453622011-11-03 16:01:40 -0700558 }
Hugo Benichib55fb222017-03-10 14:20:57 +0900559 return tetherState.lastError;
Robert Greenwalt5a735062010-03-02 17:25:02 -0800560 }
Robert Greenwaltd0e18ff2010-01-26 11:40:34 -0800561 }
562
Erik Klineea9cc482017-03-10 19:35:34 +0900563 // TODO: Figure out how to update for local hotspot mode interfaces.
Christopher Wileyd985dde2016-05-31 10:44:35 -0700564 private void sendTetherStateChangedBroadcast() {
Erik Kline7a26ba32018-03-09 14:18:02 +0900565 if (!mDeps.isTetheringSupported()) return;
Robert Greenwalt2a091d72010-02-11 18:18:40 -0800566
Erik Kline8351faa2017-04-17 16:47:23 +0900567 final ArrayList<String> availableList = new ArrayList<>();
568 final ArrayList<String> tetherList = new ArrayList<>();
569 final ArrayList<String> localOnlyList = new ArrayList<>();
570 final ArrayList<String> erroredList = new ArrayList<>();
Robert Greenwalt2a091d72010-02-11 18:18:40 -0800571
Robert Greenwalta599fe7c2010-03-08 18:30:14 -0800572 boolean wifiTethered = false;
573 boolean usbTethered = false;
Danica Chang6fdd0c62010-08-11 14:54:43 -0700574 boolean bluetoothTethered = false;
Robert Greenwalta599fe7c2010-03-08 18:30:14 -0800575
Erik Kline3e756652017-01-17 13:42:19 +0900576 final TetheringConfiguration cfg = mConfig;
577
Robert Greenwaltb4453622011-11-03 16:01:40 -0700578 synchronized (mPublicSync) {
Christopher Wileyd985dde2016-05-31 10:44:35 -0700579 for (int i = 0; i < mTetherStates.size(); i++) {
580 TetherState tetherState = mTetherStates.valueAt(i);
581 String iface = mTetherStates.keyAt(i);
Erik Kline7a26ba32018-03-09 14:18:02 +0900582 if (tetherState.lastError != TETHER_ERROR_NO_ERROR) {
Christopher Wileyd985dde2016-05-31 10:44:35 -0700583 erroredList.add(iface);
Erik Kline7a4ccc62018-08-27 17:26:47 +0900584 } else if (tetherState.lastState == IpServer.STATE_AVAILABLE) {
Christopher Wileyd985dde2016-05-31 10:44:35 -0700585 availableList.add(iface);
Erik Kline7a4ccc62018-08-27 17:26:47 +0900586 } else if (tetherState.lastState == IpServer.STATE_LOCAL_ONLY) {
Erik Kline8351faa2017-04-17 16:47:23 +0900587 localOnlyList.add(iface);
Erik Kline7a4ccc62018-08-27 17:26:47 +0900588 } else if (tetherState.lastState == IpServer.STATE_TETHERED) {
Erik Kline3e756652017-01-17 13:42:19 +0900589 if (cfg.isUsb(iface)) {
Christopher Wileyd985dde2016-05-31 10:44:35 -0700590 usbTethered = true;
Erik Kline3e756652017-01-17 13:42:19 +0900591 } else if (cfg.isWifi(iface)) {
Christopher Wileyd985dde2016-05-31 10:44:35 -0700592 wifiTethered = true;
Erik Kline3e756652017-01-17 13:42:19 +0900593 } else if (cfg.isBluetooth(iface)) {
Christopher Wileyd985dde2016-05-31 10:44:35 -0700594 bluetoothTethered = true;
Robert Greenwalt2a091d72010-02-11 18:18:40 -0800595 }
Erik Kline8351faa2017-04-17 16:47:23 +0900596 tetherList.add(iface);
Robert Greenwalt2a091d72010-02-11 18:18:40 -0800597 }
598 }
599 }
Erik Kline7a26ba32018-03-09 14:18:02 +0900600 final Intent bcast = new Intent(ACTION_TETHER_STATE_CHANGED);
Erik Kline8351faa2017-04-17 16:47:23 +0900601 bcast.addFlags(Intent.FLAG_RECEIVER_REPLACE_PENDING |
Robert Greenwalt98c79e52011-07-28 11:51:11 -0700602 Intent.FLAG_RECEIVER_REGISTERED_ONLY_BEFORE_BOOT);
Erik Kline7a26ba32018-03-09 14:18:02 +0900603 bcast.putStringArrayListExtra(EXTRA_AVAILABLE_TETHER, availableList);
604 bcast.putStringArrayListExtra(EXTRA_ACTIVE_LOCAL_ONLY, localOnlyList);
605 bcast.putStringArrayListExtra(EXTRA_ACTIVE_TETHER, tetherList);
606 bcast.putStringArrayListExtra(EXTRA_ERRORED_TETHER, erroredList);
Erik Kline8351faa2017-04-17 16:47:23 +0900607 mContext.sendStickyBroadcastAsUser(bcast, UserHandle.ALL);
Robert Greenwaltfd1be2b2011-11-11 12:30:19 -0800608 if (DBG) {
Lorenzo Colitticd63d242016-04-10 15:39:53 +0900609 Log.d(TAG, String.format(
Erik Kline8351faa2017-04-17 16:47:23 +0900610 "sendTetherStateChangedBroadcast %s=[%s] %s=[%s] %s=[%s] %s=[%s]",
611 "avail", TextUtils.join(",", availableList),
612 "local_only", TextUtils.join(",", localOnlyList),
613 "tether", TextUtils.join(",", tetherList),
614 "error", TextUtils.join(",", erroredList)));
Robert Greenwalt924cc942010-06-28 10:26:19 -0700615 }
Robert Greenwalta599fe7c2010-03-08 18:30:14 -0800616
617 if (usbTethered) {
Danica Chang6fdd0c62010-08-11 14:54:43 -0700618 if (wifiTethered || bluetoothTethered) {
Chris Wren282cfef2017-03-27 15:01:44 -0400619 showTetheredNotification(SystemMessage.NOTE_TETHER_GENERAL);
Robert Greenwalta599fe7c2010-03-08 18:30:14 -0800620 } else {
Chris Wren282cfef2017-03-27 15:01:44 -0400621 showTetheredNotification(SystemMessage.NOTE_TETHER_USB);
Robert Greenwalta599fe7c2010-03-08 18:30:14 -0800622 }
623 } else if (wifiTethered) {
Danica Chang6fdd0c62010-08-11 14:54:43 -0700624 if (bluetoothTethered) {
Chris Wren282cfef2017-03-27 15:01:44 -0400625 showTetheredNotification(SystemMessage.NOTE_TETHER_GENERAL);
Danica Chang6fdd0c62010-08-11 14:54:43 -0700626 } else {
Vinit Deshpande2576a802014-11-18 13:56:15 -0800627 /* We now have a status bar icon for WifiTethering, so drop the notification */
628 clearTetheredNotification();
Danica Chang6fdd0c62010-08-11 14:54:43 -0700629 }
630 } else if (bluetoothTethered) {
Chris Wren282cfef2017-03-27 15:01:44 -0400631 showTetheredNotification(SystemMessage.NOTE_TETHER_BLUETOOTH);
Robert Greenwalta599fe7c2010-03-08 18:30:14 -0800632 } else {
633 clearTetheredNotification();
634 }
635 }
636
Chris Wren282cfef2017-03-27 15:01:44 -0400637 private void showTetheredNotification(int id) {
Alexandru-Andrei Rotarufa6d5c52017-07-18 16:49:22 +0100638 showTetheredNotification(id, true);
639 }
640
641 @VisibleForTesting
642 protected void showTetheredNotification(int id, boolean tetheringOn) {
Robert Greenwalta599fe7c2010-03-08 18:30:14 -0800643 NotificationManager notificationManager =
Erik Klinec438e302017-07-04 22:02:49 +0900644 (NotificationManager) mContext.getSystemService(Context.NOTIFICATION_SERVICE);
Robert Greenwalta599fe7c2010-03-08 18:30:14 -0800645 if (notificationManager == null) {
646 return;
647 }
Chris Wren282cfef2017-03-27 15:01:44 -0400648 int icon = 0;
649 switch(id) {
650 case SystemMessage.NOTE_TETHER_USB:
651 icon = com.android.internal.R.drawable.stat_sys_tether_usb;
652 break;
653 case SystemMessage.NOTE_TETHER_BLUETOOTH:
654 icon = com.android.internal.R.drawable.stat_sys_tether_bluetooth;
655 break;
656 case SystemMessage.NOTE_TETHER_GENERAL:
657 default:
658 icon = com.android.internal.R.drawable.stat_sys_tether_general;
659 break;
660 }
Robert Greenwalta599fe7c2010-03-08 18:30:14 -0800661
Chris Wren1ce4b6d2015-06-11 10:19:43 -0400662 if (mLastNotificationId != 0) {
663 if (mLastNotificationId == icon) {
Robert Greenwaltdb3fe9e2010-03-18 16:28:30 -0700664 return;
665 }
Chris Wren1ce4b6d2015-06-11 10:19:43 -0400666 notificationManager.cancelAsUser(null, mLastNotificationId,
Robert Greenwalt3cab6b02012-10-04 16:44:26 -0700667 UserHandle.ALL);
Chris Wren1ce4b6d2015-06-11 10:19:43 -0400668 mLastNotificationId = 0;
Robert Greenwaltdb3fe9e2010-03-18 16:28:30 -0700669 }
670
Robert Greenwalta599fe7c2010-03-08 18:30:14 -0800671 Intent intent = new Intent();
672 intent.setClassName("com.android.settings", "com.android.settings.TetherSettings");
673 intent.setFlags(Intent.FLAG_ACTIVITY_NO_HISTORY);
674
Dianne Hackborn50cdf7c32012-09-23 17:08:57 -0700675 PendingIntent pi = PendingIntent.getActivityAsUser(mContext, 0, intent, 0,
676 null, UserHandle.CURRENT);
Robert Greenwalta599fe7c2010-03-08 18:30:14 -0800677
678 Resources r = Resources.getSystem();
Alexandru-Andrei Rotarufa6d5c52017-07-18 16:49:22 +0100679 final CharSequence title;
680 final CharSequence message;
681
682 if (tetheringOn) {
683 title = r.getText(com.android.internal.R.string.tethered_notification_title);
684 message = r.getText(com.android.internal.R.string.tethered_notification_message);
685 } else {
686 title = r.getText(com.android.internal.R.string.disable_tether_notification_title);
687 message = r.getText(com.android.internal.R.string.disable_tether_notification_message);
688 }
Robert Greenwalta599fe7c2010-03-08 18:30:14 -0800689
Chris Wren1ce4b6d2015-06-11 10:19:43 -0400690 if (mTetheredNotificationBuilder == null) {
Geoffrey Pitschaf759c52017-02-15 09:35:38 -0500691 mTetheredNotificationBuilder =
692 new Notification.Builder(mContext, SystemNotificationChannels.NETWORK_STATUS);
Chris Wren1ce4b6d2015-06-11 10:19:43 -0400693 mTetheredNotificationBuilder.setWhen(0)
694 .setOngoing(true)
695 .setColor(mContext.getColor(
696 com.android.internal.R.color.system_notification_accent_color))
697 .setVisibility(Notification.VISIBILITY_PUBLIC)
698 .setCategory(Notification.CATEGORY_STATUS);
Robert Greenwalta599fe7c2010-03-08 18:30:14 -0800699 }
Chris Wren1ce4b6d2015-06-11 10:19:43 -0400700 mTetheredNotificationBuilder.setSmallIcon(icon)
701 .setContentTitle(title)
702 .setContentText(message)
703 .setContentIntent(pi);
Chris Wren282cfef2017-03-27 15:01:44 -0400704 mLastNotificationId = id;
Robert Greenwalta599fe7c2010-03-08 18:30:14 -0800705
Chris Wren1ce4b6d2015-06-11 10:19:43 -0400706 notificationManager.notifyAsUser(null, mLastNotificationId,
zhouzhijie130d4592017-05-18 10:02:59 +0800707 mTetheredNotificationBuilder.buildInto(new Notification()), UserHandle.ALL);
Robert Greenwalta599fe7c2010-03-08 18:30:14 -0800708 }
709
Alexandru-Andrei Rotarufa6d5c52017-07-18 16:49:22 +0100710 @VisibleForTesting
711 protected void clearTetheredNotification() {
Robert Greenwalta599fe7c2010-03-08 18:30:14 -0800712 NotificationManager notificationManager =
Erik Klinec438e302017-07-04 22:02:49 +0900713 (NotificationManager) mContext.getSystemService(Context.NOTIFICATION_SERVICE);
Chris Wren1ce4b6d2015-06-11 10:19:43 -0400714 if (notificationManager != null && mLastNotificationId != 0) {
715 notificationManager.cancelAsUser(null, mLastNotificationId,
Dianne Hackborn50cdf7c32012-09-23 17:08:57 -0700716 UserHandle.ALL);
Chris Wren1ce4b6d2015-06-11 10:19:43 -0400717 mLastNotificationId = 0;
Robert Greenwalta599fe7c2010-03-08 18:30:14 -0800718 }
Robert Greenwalt5a735062010-03-02 17:25:02 -0800719 }
720
Robert Greenwalt2a091d72010-02-11 18:18:40 -0800721 private class StateReceiver extends BroadcastReceiver {
Nick Kralevich70c117a2014-05-27 15:30:02 -0700722 @Override
Robert Greenwaltd0e18ff2010-01-26 11:40:34 -0800723 public void onReceive(Context content, Intent intent) {
Erik Kline2e88b5e2017-01-18 11:57:45 +0900724 final String action = intent.getAction();
725 if (action == null) return;
726
Mike Lockwood770126a2010-12-09 22:30:37 -0800727 if (action.equals(UsbManager.ACTION_USB_STATE)) {
Erik Kline2e88b5e2017-01-18 11:57:45 +0900728 handleUsbAction(intent);
Erik Kline7a26ba32018-03-09 14:18:02 +0900729 } else if (action.equals(CONNECTIVITY_ACTION)) {
Erik Kline2e88b5e2017-01-18 11:57:45 +0900730 handleConnectivityAction(intent);
Christopher Wiley5c0b10a2016-05-31 14:43:08 -0700731 } else if (action.equals(WifiManager.WIFI_AP_STATE_CHANGED_ACTION)) {
Erik Kline2e88b5e2017-01-18 11:57:45 +0900732 handleWifiApAction(intent);
Robert Greenwaltc13368b2013-07-18 14:24:42 -0700733 } else if (action.equals(Intent.ACTION_CONFIGURATION_CHANGED)) {
Erik Klinede637722017-10-12 22:16:01 +0900734 mLog.log("OBSERVED configuration changed");
Robert Greenwaltc13368b2013-07-18 14:24:42 -0700735 updateConfiguration();
Robert Greenwaltd0e18ff2010-01-26 11:40:34 -0800736 }
737 }
Erik Kline2e88b5e2017-01-18 11:57:45 +0900738
739 private void handleConnectivityAction(Intent intent) {
Erik Kline7a26ba32018-03-09 14:18:02 +0900740 final NetworkInfo networkInfo =
741 (NetworkInfo) intent.getParcelableExtra(EXTRA_NETWORK_INFO);
Erik Kline2e88b5e2017-01-18 11:57:45 +0900742 if (networkInfo == null ||
743 networkInfo.getDetailedState() == NetworkInfo.DetailedState.FAILED) {
744 return;
745 }
746
747 if (VDBG) Log.d(TAG, "Tethering got CONNECTIVITY_ACTION: " + networkInfo.toString());
748 mTetherMasterSM.sendMessage(TetherMasterSM.CMD_UPSTREAM_CHANGED);
749 }
750
751 private void handleUsbAction(Intent intent) {
752 final boolean usbConnected = intent.getBooleanExtra(USB_CONNECTED, false);
Erik Klinec438e302017-07-04 22:02:49 +0900753 final boolean usbConfigured = intent.getBooleanExtra(USB_CONFIGURED, false);
Erik Kline2e88b5e2017-01-18 11:57:45 +0900754 final boolean rndisEnabled = intent.getBooleanExtra(USB_FUNCTION_RNDIS, false);
Erik Klinec438e302017-07-04 22:02:49 +0900755
756 mLog.log(String.format("USB bcast connected:%s configured:%s rndis:%s",
757 usbConnected, usbConfigured, rndisEnabled));
758
759 // There are three types of ACTION_USB_STATE:
760 //
761 // - DISCONNECTED (USB_CONNECTED and USB_CONFIGURED are 0)
762 // Meaning: USB connection has ended either because of
763 // software reset or hard unplug.
764 //
765 // - CONNECTED (USB_CONNECTED is 1, USB_CONFIGURED is 0)
766 // Meaning: the first stage of USB protocol handshake has
767 // occurred but it is not complete.
768 //
769 // - CONFIGURED (USB_CONNECTED and USB_CONFIGURED are 1)
770 // Meaning: the USB handshake is completely done and all the
771 // functions are ready to use.
772 //
773 // For more explanation, see b/62552150 .
Erik Kline2e88b5e2017-01-18 11:57:45 +0900774 synchronized (Tethering.this.mPublicSync) {
Jerry Zhang656a7bc2017-12-20 14:26:39 -0800775 if (!usbConnected && mRndisEnabled) {
776 // Turn off tethering if it was enabled and there is a disconnect.
Erik Kline7a4ccc62018-08-27 17:26:47 +0900777 tetherMatchingInterfaces(IpServer.STATE_AVAILABLE, TETHERING_USB);
Jerry Zhang656a7bc2017-12-20 14:26:39 -0800778 } else if (usbConfigured && rndisEnabled) {
779 // Tether if rndis is enabled and usb is configured.
Erik Kline7a4ccc62018-08-27 17:26:47 +0900780 tetherMatchingInterfaces(IpServer.STATE_TETHERED, TETHERING_USB);
Erik Kline2e88b5e2017-01-18 11:57:45 +0900781 }
Jerry Zhang656a7bc2017-12-20 14:26:39 -0800782 mRndisEnabled = usbConfigured && rndisEnabled;
Erik Kline2e88b5e2017-01-18 11:57:45 +0900783 }
784 }
785
786 private void handleWifiApAction(Intent intent) {
Erik Kline2efb8272017-05-31 15:53:53 +0900787 final int curState = intent.getIntExtra(EXTRA_WIFI_AP_STATE, WIFI_AP_STATE_DISABLED);
788 final String ifname = intent.getStringExtra(EXTRA_WIFI_AP_INTERFACE_NAME);
789 final int ipmode = intent.getIntExtra(EXTRA_WIFI_AP_MODE, IFACE_IP_MODE_UNSPECIFIED);
790
Erik Kline2e88b5e2017-01-18 11:57:45 +0900791 synchronized (Tethering.this.mPublicSync) {
792 switch (curState) {
793 case WifiManager.WIFI_AP_STATE_ENABLING:
794 // We can see this state on the way to both enabled and failure states.
795 break;
796 case WifiManager.WIFI_AP_STATE_ENABLED:
Erik Kline2efb8272017-05-31 15:53:53 +0900797 enableWifiIpServingLocked(ifname, ipmode);
Erik Kline2e88b5e2017-01-18 11:57:45 +0900798 break;
799 case WifiManager.WIFI_AP_STATE_DISABLED:
800 case WifiManager.WIFI_AP_STATE_DISABLING:
801 case WifiManager.WIFI_AP_STATE_FAILED:
802 default:
Erik Kline562e0c12017-06-09 16:36:29 +0900803 disableWifiIpServingLocked(ifname, curState);
Erik Kline2efb8272017-05-31 15:53:53 +0900804 break;
Erik Kline2e88b5e2017-01-18 11:57:45 +0900805 }
806 }
807 }
Robert Greenwaltd0e18ff2010-01-26 11:40:34 -0800808 }
809
Alexandru-Andrei Rotarufa6d5c52017-07-18 16:49:22 +0100810 @VisibleForTesting
811 protected static class TetheringUserRestrictionListener implements UserRestrictionsListener {
812 private final Tethering mWrapper;
813
814 public TetheringUserRestrictionListener(Tethering wrapper) {
815 mWrapper = wrapper;
816 }
817
818 public void onUserRestrictionsChanged(int userId,
819 Bundle newRestrictions,
820 Bundle prevRestrictions) {
821 final boolean newlyDisallowed =
822 newRestrictions.getBoolean(UserManager.DISALLOW_CONFIG_TETHERING);
823 final boolean previouslyDisallowed =
824 prevRestrictions.getBoolean(UserManager.DISALLOW_CONFIG_TETHERING);
825 final boolean tetheringDisallowedChanged = (newlyDisallowed != previouslyDisallowed);
826
827 if (!tetheringDisallowedChanged) {
828 return;
829 }
830
831 mWrapper.clearTetheredNotification();
832 final boolean isTetheringActiveOnDevice = (mWrapper.getTetheredIfaces().length != 0);
833
834 if (newlyDisallowed && isTetheringActiveOnDevice) {
835 mWrapper.showTetheredNotification(
836 com.android.internal.R.drawable.stat_sys_tether_general, false);
837 mWrapper.untetherAll();
838 }
839 }
840 }
841
Erik Kline562e0c12017-06-09 16:36:29 +0900842 private void disableWifiIpServingLocked(String ifname, int apState) {
843 mLog.log("Canceling WiFi tethering request - AP_STATE=" + apState);
Erik Kline2efb8272017-05-31 15:53:53 +0900844
Erik Kline562e0c12017-06-09 16:36:29 +0900845 // 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 // TODO: Remove this altogether, once Wi-Fi reliably gives us an
848 // interface name with every broadcast.
849 mWifiTetherRequested = false;
850
851 if (!TextUtils.isEmpty(ifname)) {
852 final TetherState ts = mTetherStates.get(ifname);
853 if (ts != null) {
Erik Kline7a4ccc62018-08-27 17:26:47 +0900854 ts.ipServer.unwanted();
Erik Kline562e0c12017-06-09 16:36:29 +0900855 return;
856 }
857 }
858
Erik Kline2efb8272017-05-31 15:53:53 +0900859 for (int i = 0; i < mTetherStates.size(); i++) {
Erik Kline7a4ccc62018-08-27 17:26:47 +0900860 final IpServer ipServer = mTetherStates.valueAt(i).ipServer;
861 if (ipServer.interfaceType() == TETHERING_WIFI) {
862 ipServer.unwanted();
Erik Kline562e0c12017-06-09 16:36:29 +0900863 return;
Erik Kline2efb8272017-05-31 15:53:53 +0900864 }
865 }
Erik Kline562e0c12017-06-09 16:36:29 +0900866
867 mLog.log("Error disabling Wi-Fi IP serving; " +
868 (TextUtils.isEmpty(ifname) ? "no interface name specified"
869 : "specified interface: " + ifname));
Erik Kline2efb8272017-05-31 15:53:53 +0900870 }
871
872 private void enableWifiIpServingLocked(String ifname, int wifiIpMode) {
Erik Kline7a4ccc62018-08-27 17:26:47 +0900873 // Map wifiIpMode values to IpServer.Callback serving states, inferring
Erik Kline2efb8272017-05-31 15:53:53 +0900874 // from mWifiTetherRequested as a final "best guess".
875 final int ipServingMode;
876 switch (wifiIpMode) {
877 case IFACE_IP_MODE_TETHERED:
Erik Kline7a4ccc62018-08-27 17:26:47 +0900878 ipServingMode = IpServer.STATE_TETHERED;
Erik Kline2efb8272017-05-31 15:53:53 +0900879 break;
880 case IFACE_IP_MODE_LOCAL_ONLY:
Erik Kline7a4ccc62018-08-27 17:26:47 +0900881 ipServingMode = IpServer.STATE_LOCAL_ONLY;
Erik Kline2efb8272017-05-31 15:53:53 +0900882 break;
883 default:
Erik Kline9e225542017-06-08 17:48:48 +0900884 mLog.e("Cannot enable IP serving in unknown WiFi mode: " + wifiIpMode);
885 return;
Erik Kline2efb8272017-05-31 15:53:53 +0900886 }
887
888 if (!TextUtils.isEmpty(ifname)) {
Erik Kline7a26ba32018-03-09 14:18:02 +0900889 maybeTrackNewInterfaceLocked(ifname, TETHERING_WIFI);
Erik Kline2efb8272017-05-31 15:53:53 +0900890 changeInterfaceState(ifname, ipServingMode);
891 } else {
Erik Kline9e225542017-06-08 17:48:48 +0900892 mLog.e(String.format(
893 "Cannot enable IP serving in mode %s on missing interface name",
894 ipServingMode));
Erik Kline2efb8272017-05-31 15:53:53 +0900895 }
896 }
897
Erik Klineea9cc482017-03-10 19:35:34 +0900898 // TODO: Consider renaming to something more accurate in its description.
899 // This method:
900 // - allows requesting either tethering or local hotspot serving states
901 // - handles both enabling and disabling serving states
902 // - only tethers the first matching interface in listInterfaces()
903 // order of a given type
904 private void tetherMatchingInterfaces(int requestedState, int interfaceType) {
905 if (VDBG) {
906 Log.d(TAG, "tetherMatchingInterfaces(" + requestedState + ", " + interfaceType + ")");
907 }
Mike Lockwood3c2a2f62011-06-08 15:10:26 -0700908
Christopher Wiley5c0b10a2016-05-31 14:43:08 -0700909 String[] ifaces = null;
Robert Greenwalt65ae29b2010-02-18 11:25:54 -0800910 try {
Chia-chi Yehc9338302011-05-11 16:35:13 -0700911 ifaces = mNMService.listInterfaces();
Robert Greenwalt65ae29b2010-02-18 11:25:54 -0800912 } catch (Exception e) {
Mike Lockwood3c2a2f62011-06-08 15:10:26 -0700913 Log.e(TAG, "Error listing Interfaces", e);
Robert Greenwalt65ae29b2010-02-18 11:25:54 -0800914 return;
915 }
Christopher Wiley5c0b10a2016-05-31 14:43:08 -0700916 String chosenIface = null;
917 if (ifaces != null) {
918 for (String iface : ifaces) {
919 if (ifaceNameToType(iface) == interfaceType) {
920 chosenIface = iface;
921 break;
Robert Greenwalt65ae29b2010-02-18 11:25:54 -0800922 }
923 }
924 }
Christopher Wiley5c0b10a2016-05-31 14:43:08 -0700925 if (chosenIface == null) {
926 Log.e(TAG, "could not find iface of type " + interfaceType);
927 return;
928 }
929
Erik Kline2efb8272017-05-31 15:53:53 +0900930 changeInterfaceState(chosenIface, requestedState);
931 }
932
933 private void changeInterfaceState(String ifname, int requestedState) {
Erik Klineea9cc482017-03-10 19:35:34 +0900934 final int result;
935 switch (requestedState) {
Erik Kline7a4ccc62018-08-27 17:26:47 +0900936 case IpServer.STATE_UNAVAILABLE:
937 case IpServer.STATE_AVAILABLE:
Erik Kline2efb8272017-05-31 15:53:53 +0900938 result = untether(ifname);
Erik Klineea9cc482017-03-10 19:35:34 +0900939 break;
Erik Kline7a4ccc62018-08-27 17:26:47 +0900940 case IpServer.STATE_TETHERED:
941 case IpServer.STATE_LOCAL_ONLY:
Erik Kline2efb8272017-05-31 15:53:53 +0900942 result = tether(ifname, requestedState);
Erik Klineea9cc482017-03-10 19:35:34 +0900943 break;
944 default:
945 Log.wtf(TAG, "Unknown interface state: " + requestedState);
946 return;
947 }
Erik Kline7a26ba32018-03-09 14:18:02 +0900948 if (result != TETHER_ERROR_NO_ERROR) {
Erik Kline2efb8272017-05-31 15:53:53 +0900949 Log.e(TAG, "unable start or stop tethering on iface " + ifname);
Christopher Wiley5c0b10a2016-05-31 14:43:08 -0700950 return;
951 }
Robert Greenwalt65ae29b2010-02-18 11:25:54 -0800952 }
953
Erik Kline3e756652017-01-17 13:42:19 +0900954 public TetheringConfiguration getTetheringConfiguration() {
955 return mConfig;
956 }
957
Erik Klined781fba2017-01-23 13:01:58 +0900958 public boolean hasTetherableConfiguration() {
959 final TetheringConfiguration cfg = mConfig;
960 final boolean hasDownstreamConfiguration =
markchien1be8d8f2018-12-05 21:20:01 +0800961 (cfg.tetherableUsbRegexs.length != 0)
962 || (cfg.tetherableWifiRegexs.length != 0)
963 || (cfg.tetherableBluetoothRegexs.length != 0);
964 final boolean hasUpstreamConfiguration = !cfg.preferredUpstreamIfaceTypes.isEmpty()
965 || cfg.chooseUpstreamAutomatically;
Erik Klined781fba2017-01-23 13:01:58 +0900966
967 return hasDownstreamConfiguration && hasUpstreamConfiguration;
968 }
969
Erik Kline3e756652017-01-17 13:42:19 +0900970 // TODO - update callers to use getTetheringConfiguration(),
971 // which has only final members.
Robert Greenwalt2a091d72010-02-11 18:18:40 -0800972 public String[] getTetherableUsbRegexs() {
Erik Kline3e756652017-01-17 13:42:19 +0900973 return copy(mConfig.tetherableUsbRegexs);
Robert Greenwaltd0e18ff2010-01-26 11:40:34 -0800974 }
975
Robert Greenwalt2a091d72010-02-11 18:18:40 -0800976 public String[] getTetherableWifiRegexs() {
Erik Kline3e756652017-01-17 13:42:19 +0900977 return copy(mConfig.tetherableWifiRegexs);
Robert Greenwalt2a091d72010-02-11 18:18:40 -0800978 }
979
Danica Chang6fdd0c62010-08-11 14:54:43 -0700980 public String[] getTetherableBluetoothRegexs() {
Erik Kline3e756652017-01-17 13:42:19 +0900981 return copy(mConfig.tetherableBluetoothRegexs);
Danica Chang6fdd0c62010-08-11 14:54:43 -0700982 }
983
Mike Lockwood6c2260b2011-07-19 13:04:47 -0700984 public int setUsbTethering(boolean enable) {
Wink Savillec9acde92011-09-21 11:05:43 -0700985 if (VDBG) Log.d(TAG, "setUsbTethering(" + enable + ")");
Erik Klinec438e302017-07-04 22:02:49 +0900986 UsbManager usbManager = (UsbManager) mContext.getSystemService(Context.USB_SERVICE);
Robert Greenwaltb4453622011-11-03 16:01:40 -0700987 synchronized (mPublicSync) {
Jerry Zhang327b8092018-01-09 17:53:04 -0800988 usbManager.setCurrentFunctions(enable ? UsbManager.FUNCTION_RNDIS
989 : UsbManager.FUNCTION_NONE);
Mike Lockwood6c2260b2011-07-19 13:04:47 -0700990 }
Erik Kline7a26ba32018-03-09 14:18:02 +0900991 return TETHER_ERROR_NO_ERROR;
Mike Lockwood6c2260b2011-07-19 13:04:47 -0700992 }
993
Erik Kline1fdc2e22017-05-08 17:56:35 +0900994 // TODO review API - figure out how to delete these entirely.
Robert Greenwalt2a091d72010-02-11 18:18:40 -0800995 public String[] getTetheredIfaces() {
996 ArrayList<String> list = new ArrayList<String>();
Robert Greenwaltb4453622011-11-03 16:01:40 -0700997 synchronized (mPublicSync) {
Christopher Wileyd985dde2016-05-31 10:44:35 -0700998 for (int i = 0; i < mTetherStates.size(); i++) {
999 TetherState tetherState = mTetherStates.valueAt(i);
Erik Kline7a4ccc62018-08-27 17:26:47 +09001000 if (tetherState.lastState == IpServer.STATE_TETHERED) {
Christopher Wileyd985dde2016-05-31 10:44:35 -07001001 list.add(mTetherStates.keyAt(i));
Robert Greenwalt2a091d72010-02-11 18:18:40 -08001002 }
1003 }
Robert Greenwaltd0e18ff2010-01-26 11:40:34 -08001004 }
Christopher Wileyd985dde2016-05-31 10:44:35 -07001005 return list.toArray(new String[list.size()]);
Robert Greenwalt2a091d72010-02-11 18:18:40 -08001006 }
1007
1008 public String[] getTetherableIfaces() {
1009 ArrayList<String> list = new ArrayList<String>();
Robert Greenwaltb4453622011-11-03 16:01:40 -07001010 synchronized (mPublicSync) {
Christopher Wileyd985dde2016-05-31 10:44:35 -07001011 for (int i = 0; i < mTetherStates.size(); i++) {
1012 TetherState tetherState = mTetherStates.valueAt(i);
Erik Kline7a4ccc62018-08-27 17:26:47 +09001013 if (tetherState.lastState == IpServer.STATE_AVAILABLE) {
Christopher Wileyd985dde2016-05-31 10:44:35 -07001014 list.add(mTetherStates.keyAt(i));
Robert Greenwalt2a091d72010-02-11 18:18:40 -08001015 }
1016 }
1017 }
Christopher Wileyd985dde2016-05-31 10:44:35 -07001018 return list.toArray(new String[list.size()]);
Robert Greenwalt2a091d72010-02-11 18:18:40 -08001019 }
1020
Robert Greenwalt9c7e2c22014-06-23 14:53:42 -07001021 public String[] getTetheredDhcpRanges() {
Remi NGUYEN VANe3bb5c52018-06-12 15:57:04 +09001022 // TODO: this is only valid for the old DHCP server. Latest search suggests it is only used
1023 // by WifiP2pServiceImpl to start dnsmasq: remove/deprecate after migrating callers.
1024 return mConfig.legacyDhcpRanges;
Robert Greenwalt9c7e2c22014-06-23 14:53:42 -07001025 }
1026
Robert Greenwalt5a735062010-03-02 17:25:02 -08001027 public String[] getErroredIfaces() {
1028 ArrayList<String> list = new ArrayList<String>();
Robert Greenwaltb4453622011-11-03 16:01:40 -07001029 synchronized (mPublicSync) {
Christopher Wileyd985dde2016-05-31 10:44:35 -07001030 for (int i = 0; i < mTetherStates.size(); i++) {
1031 TetherState tetherState = mTetherStates.valueAt(i);
Erik Kline7a26ba32018-03-09 14:18:02 +09001032 if (tetherState.lastError != TETHER_ERROR_NO_ERROR) {
Christopher Wileyd985dde2016-05-31 10:44:35 -07001033 list.add(mTetherStates.keyAt(i));
Robert Greenwalt5a735062010-03-02 17:25:02 -08001034 }
1035 }
1036 }
Christopher Wileyd985dde2016-05-31 10:44:35 -07001037 return list.toArray(new String[list.size()]);
Robert Greenwalt5a735062010-03-02 17:25:02 -08001038 }
1039
Erik Kline22108902017-07-06 16:40:06 +09001040 private void logMessage(State state, int what) {
1041 mLog.log(state.getName() + " got " + sMagicDecoderRing.get(what, Integer.toString(what)));
Lorenzo Colitticd63d242016-04-10 15:39:53 +09001042 }
1043
Erik Klineea9cc482017-03-10 19:35:34 +09001044 private boolean upstreamWanted() {
1045 if (!mForwardedDownstreams.isEmpty()) return true;
1046
1047 synchronized (mPublicSync) {
Jerry Zhang656a7bc2017-12-20 14:26:39 -08001048 return mWifiTetherRequested;
Erik Klineea9cc482017-03-10 19:35:34 +09001049 }
1050 }
1051
Erik Kline00019f42016-06-30 19:31:46 +09001052 // Needed because the canonical source of upstream truth is just the
Remi NGUYEN VAN6c02f992018-03-09 14:07:18 +09001053 // upstream interface set, |mCurrentUpstreamIfaceSet|.
Erik Kline3a5278f2017-06-24 19:29:10 +09001054 private boolean pertainsToCurrentUpstream(NetworkState ns) {
Remi NGUYEN VAN6c02f992018-03-09 14:07:18 +09001055 if (ns != null && ns.linkProperties != null && mCurrentUpstreamIfaceSet != null) {
Erik Kline00019f42016-06-30 19:31:46 +09001056 for (String ifname : ns.linkProperties.getAllInterfaceNames()) {
Remi NGUYEN VAN6c02f992018-03-09 14:07:18 +09001057 if (mCurrentUpstreamIfaceSet.ifnames.contains(ifname)) {
Erik Kline00019f42016-06-30 19:31:46 +09001058 return true;
Erik Kline6ff17f72015-12-10 20:42:12 +09001059 }
1060 }
Erik Kline6ff17f72015-12-10 20:42:12 +09001061 }
Erik Kline00019f42016-06-30 19:31:46 +09001062 return false;
Erik Kline6ff17f72015-12-10 20:42:12 +09001063 }
1064
Wink Saville64c42ca2011-04-18 14:55:10 -07001065 class TetherMasterSM extends StateMachine {
Lorenzo Colitticd63d242016-04-10 15:39:53 +09001066 private static final int BASE_MASTER = Protocol.BASE_TETHERING;
Erik Klineea9cc482017-03-10 19:35:34 +09001067 // an interface SM has requested Tethering/Local Hotspot
1068 static final int EVENT_IFACE_SERVING_STATE_ACTIVE = BASE_MASTER + 1;
1069 // an interface SM has unrequested Tethering/Local Hotspot
1070 static final int EVENT_IFACE_SERVING_STATE_INACTIVE = BASE_MASTER + 2;
Robert Greenwalt6a1967c2010-03-17 11:19:57 -07001071 // upstream connection change - do the right thing
Lorenzo Colitticd63d242016-04-10 15:39:53 +09001072 static final int CMD_UPSTREAM_CHANGED = BASE_MASTER + 3;
Robert Greenwalt6a1967c2010-03-17 11:19:57 -07001073 // we don't have a valid upstream conn, check again after a delay
Lorenzo Colitticd63d242016-04-10 15:39:53 +09001074 static final int CMD_RETRY_UPSTREAM = BASE_MASTER + 4;
Erik Kline6ff17f72015-12-10 20:42:12 +09001075 // Events from NetworkCallbacks that we process on the master state
1076 // machine thread on behalf of the UpstreamNetworkMonitor.
Erik Kline00019f42016-06-30 19:31:46 +09001077 static final int EVENT_UPSTREAM_CALLBACK = BASE_MASTER + 5;
Yohei, Oshima977aad52016-12-08 13:39:20 +09001078 // we treated the error and want now to clear it
1079 static final int CMD_CLEAR_ERROR = BASE_MASTER + 6;
Erik Kline6e9a1012017-06-06 19:24:21 +09001080 static final int EVENT_IFACE_UPDATE_LINKPROPERTIES = BASE_MASTER + 7;
Robert Greenwalt2a091d72010-02-11 18:18:40 -08001081
Erik Kline32179ff2017-07-04 18:28:11 +09001082 private final State mInitialState;
1083 private final State mTetherModeAliveState;
Robert Greenwalt2a091d72010-02-11 18:18:40 -08001084
Erik Kline32179ff2017-07-04 18:28:11 +09001085 private final State mSetIpForwardingEnabledErrorState;
1086 private final State mSetIpForwardingDisabledErrorState;
1087 private final State mStartTetheringErrorState;
1088 private final State mStopTetheringErrorState;
1089 private final 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 Kline7a4ccc62018-08-27 17:26:47 +09001103 private final ArrayList<IpServer> mNotifyList;
Erik Kline1eb8c692016-07-08 17:21:26 +09001104 private final IPv6TetheringCoordinator mIPv6TetheringCoordinator;
Erik Klineed962a82017-07-06 19:49:35 +09001105 private final OffloadWrapper mOffload;
Robert Greenwalt2a091d72010-02-11 18:18:40 -08001106
Robert Greenwalt6a1967c2010-03-17 11:19:57 -07001107 private static final int UPSTREAM_SETTLE_TIME_MS = 10000;
Robert Greenwalt2a091d72010-02-11 18:18:40 -08001108
Remi NGUYEN VAN5d0dc452018-03-15 11:57:14 +09001109 TetherMasterSM(String name, Looper looper, TetheringDependencies deps) {
Robert Greenwaltdfadaea2010-03-11 15:03:08 -08001110 super(name, looper);
Robert Greenwalt2a091d72010-02-11 18:18:40 -08001111
Robert Greenwalt2a091d72010-02-11 18:18:40 -08001112 mInitialState = new InitialState();
Robert Greenwalt2a091d72010-02-11 18:18:40 -08001113 mTetherModeAliveState = new TetherModeAliveState();
Robert Greenwalt2a091d72010-02-11 18:18:40 -08001114 mSetIpForwardingEnabledErrorState = new SetIpForwardingEnabledErrorState();
Robert Greenwalt2a091d72010-02-11 18:18:40 -08001115 mSetIpForwardingDisabledErrorState = new SetIpForwardingDisabledErrorState();
Robert Greenwalt2a091d72010-02-11 18:18:40 -08001116 mStartTetheringErrorState = new StartTetheringErrorState();
Robert Greenwalt2a091d72010-02-11 18:18:40 -08001117 mStopTetheringErrorState = new StopTetheringErrorState();
Robert Greenwalt2a091d72010-02-11 18:18:40 -08001118 mSetDnsForwardersErrorState = new SetDnsForwardersErrorState();
Erik Kline1fdc2e22017-05-08 17:56:35 +09001119
1120 addState(mInitialState);
1121 addState(mTetherModeAliveState);
1122 addState(mSetIpForwardingEnabledErrorState);
1123 addState(mSetIpForwardingDisabledErrorState);
1124 addState(mStartTetheringErrorState);
1125 addState(mStopTetheringErrorState);
Robert Greenwalt2a091d72010-02-11 18:18:40 -08001126 addState(mSetDnsForwardersErrorState);
1127
Mitchell Wills7040b4e2016-05-23 16:40:10 -07001128 mNotifyList = new ArrayList<>();
Remi NGUYEN VAN5d0dc452018-03-15 11:57:14 +09001129 mIPv6TetheringCoordinator = deps.getIPv6TetheringCoordinator(mNotifyList, mLog);
Erik Klineed962a82017-07-06 19:49:35 +09001130 mOffload = new OffloadWrapper();
Erik Kline32179ff2017-07-04 18:28:11 +09001131
Robert Greenwalt2a091d72010-02-11 18:18:40 -08001132 setInitialState(mInitialState);
1133 }
1134
Erik Kline1fdc2e22017-05-08 17:56:35 +09001135 class InitialState extends State {
1136 @Override
1137 public boolean processMessage(Message message) {
Erik Kline22108902017-07-06 16:40:06 +09001138 logMessage(this, message.what);
Erik Kline1fdc2e22017-05-08 17:56:35 +09001139 switch (message.what) {
Erik Kline7a4ccc62018-08-27 17:26:47 +09001140 case EVENT_IFACE_SERVING_STATE_ACTIVE: {
1141 final IpServer who = (IpServer) message.obj;
Erik Kline1fdc2e22017-05-08 17:56:35 +09001142 if (VDBG) Log.d(TAG, "Tether Mode requested by " + who);
1143 handleInterfaceServingStateActive(message.arg1, who);
1144 transitionTo(mTetherModeAliveState);
1145 break;
Erik Kline7a4ccc62018-08-27 17:26:47 +09001146 }
1147 case EVENT_IFACE_SERVING_STATE_INACTIVE: {
1148 final IpServer who = (IpServer) message.obj;
Erik Kline1fdc2e22017-05-08 17:56:35 +09001149 if (VDBG) Log.d(TAG, "Tether Mode unrequested by " + who);
1150 handleInterfaceServingStateInactive(who);
1151 break;
Erik Kline7a4ccc62018-08-27 17:26:47 +09001152 }
Erik Kline6e9a1012017-06-06 19:24:21 +09001153 case EVENT_IFACE_UPDATE_LINKPROPERTIES:
1154 // Silently ignore these for now.
1155 break;
Erik Kline1fdc2e22017-05-08 17:56:35 +09001156 default:
1157 return NOT_HANDLED;
1158 }
1159 return HANDLED;
1160 }
1161 }
1162
Erik Kline3a5278f2017-06-24 19:29:10 +09001163 protected boolean turnOnMasterTetherSettings() {
1164 final TetheringConfiguration cfg = mConfig;
1165 try {
1166 mNMService.setIpForwardingEnabled(true);
1167 } catch (Exception e) {
1168 mLog.e(e);
1169 transitionTo(mSetIpForwardingEnabledErrorState);
Robert Greenwaltd70a3d42010-02-23 18:15:29 -08001170 return false;
1171 }
Erik Kline3a5278f2017-06-24 19:29:10 +09001172 // TODO: Randomize DHCPv4 ranges, especially in hotspot mode.
Remi NGUYEN VANe3bb5c52018-06-12 15:57:04 +09001173 // Legacy DHCP server is disabled if passed an empty ranges array
1174 final String[] dhcpRanges = cfg.enableLegacyDhcpServer
1175 ? cfg.legacyDhcpRanges
1176 : new String[0];
Erik Kline3a5278f2017-06-24 19:29:10 +09001177 try {
1178 // TODO: Find a more accurate method name (startDHCPv4()?).
Remi NGUYEN VANe3bb5c52018-06-12 15:57:04 +09001179 mNMService.startTethering(dhcpRanges);
Erik Kline3a5278f2017-06-24 19:29:10 +09001180 } catch (Exception e) {
Robert Greenwaltd70a3d42010-02-23 18:15:29 -08001181 try {
Chia-chi Yehc9338302011-05-11 16:35:13 -07001182 mNMService.stopTethering();
Remi NGUYEN VANe3bb5c52018-06-12 15:57:04 +09001183 mNMService.startTethering(dhcpRanges);
Erik Kline3a5278f2017-06-24 19:29:10 +09001184 } catch (Exception ee) {
1185 mLog.e(ee);
1186 transitionTo(mStartTetheringErrorState);
Robert Greenwaltd70a3d42010-02-23 18:15:29 -08001187 return false;
1188 }
Robert Greenwaltd70a3d42010-02-23 18:15:29 -08001189 }
Erik Kline3a5278f2017-06-24 19:29:10 +09001190 mLog.log("SET master tether settings: ON");
1191 return true;
1192 }
Robert Greenwalt10398722010-12-17 15:20:36 -08001193
Erik Kline3a5278f2017-06-24 19:29:10 +09001194 protected boolean turnOffMasterTetherSettings() {
1195 try {
1196 mNMService.stopTethering();
1197 } catch (Exception e) {
1198 mLog.e(e);
1199 transitionTo(mStopTetheringErrorState);
1200 return false;
Erik Kline14f7faf2017-02-14 19:03:09 +09001201 }
Erik Kline3a5278f2017-06-24 19:29:10 +09001202 try {
1203 mNMService.setIpForwardingEnabled(false);
1204 } catch (Exception e) {
1205 mLog.e(e);
1206 transitionTo(mSetIpForwardingDisabledErrorState);
1207 return false;
1208 }
1209 transitionTo(mInitialState);
1210 mLog.log("SET master tether settings: OFF");
1211 return true;
1212 }
Erik Kline14f7faf2017-02-14 19:03:09 +09001213
Erik Kline3a5278f2017-06-24 19:29:10 +09001214 protected void chooseUpstreamType(boolean tryCell) {
Erik Kline6ee73da2017-07-08 20:36:37 +09001215 // We rebuild configuration on ACTION_CONFIGURATION_CHANGED, but we
1216 // do not currently know how to watch for changes in DUN settings.
1217 maybeUpdateConfiguration();
Erik Kline1e2897d2017-06-09 17:08:52 +09001218
Erik Kline72302902018-06-14 17:36:40 +09001219 final TetheringConfiguration config = mConfig;
1220 final NetworkState ns = (config.chooseUpstreamAutomatically)
1221 ? mUpstreamNetworkMonitor.getCurrentPreferredUpstream()
1222 : mUpstreamNetworkMonitor.selectPreferredUpstreamType(
1223 config.preferredUpstreamIfaceTypes);
Erik Kline3a5278f2017-06-24 19:29:10 +09001224 if (ns == null) {
1225 if (tryCell) {
1226 mUpstreamNetworkMonitor.registerMobileNetworkRequest();
1227 // We think mobile should be coming up; don't set a retry.
1228 } else {
1229 sendMessageDelayed(CMD_RETRY_UPSTREAM, UPSTREAM_SETTLE_TIME_MS);
Erik Kline1e2897d2017-06-09 17:08:52 +09001230 }
Erik Kline3a5278f2017-06-24 19:29:10 +09001231 }
Erik Kline60caf7d2017-10-10 11:54:08 +09001232 mUpstreamNetworkMonitor.setCurrentUpstream((ns != null) ? ns.network : null);
Erik Kline3a5278f2017-06-24 19:29:10 +09001233 setUpstreamNetwork(ns);
1234 }
1235
1236 protected void setUpstreamNetwork(NetworkState ns) {
Remi NGUYEN VAN6c02f992018-03-09 14:07:18 +09001237 InterfaceSet ifaces = null;
Erik Klinee8bdb402018-02-23 14:16:06 -08001238 if (ns != null) {
Erik Kline3a5278f2017-06-24 19:29:10 +09001239 // Find the interface with the default IPv4 route. It may be the
1240 // interface described by linkProperties, or one of the interfaces
1241 // stacked on top of it.
Erik Klinee8bdb402018-02-23 14:16:06 -08001242 mLog.i("Looking for default routes on: " + ns.linkProperties);
Remi NGUYEN VAN6c02f992018-03-09 14:07:18 +09001243 ifaces = TetheringInterfaceUtils.getTetheringInterfaces(ns);
1244 mLog.i("Found upstream interface(s): " + ifaces);
Robert Greenwalt6a1967c2010-03-17 11:19:57 -07001245 }
Robert Greenwaltccf83af12011-06-02 17:30:47 -07001246
Remi NGUYEN VAN6c02f992018-03-09 14:07:18 +09001247 if (ifaces != null) {
Erik Kline3a5278f2017-06-24 19:29:10 +09001248 setDnsForwarders(ns.network, ns.linkProperties);
Erik Kline6ff17f72015-12-10 20:42:12 +09001249 }
Remi NGUYEN VAN6c02f992018-03-09 14:07:18 +09001250 notifyDownstreamsOfNewUpstreamIface(ifaces);
Erik Kline3a5278f2017-06-24 19:29:10 +09001251 if (ns != null && pertainsToCurrentUpstream(ns)) {
Erik Kline72302902018-06-14 17:36:40 +09001252 // If we already have NetworkState for this network update it immediately.
Erik Kline3a5278f2017-06-24 19:29:10 +09001253 handleNewUpstreamNetworkState(ns);
Remi NGUYEN VAN6c02f992018-03-09 14:07:18 +09001254 } else if (mCurrentUpstreamIfaceSet == null) {
1255 // There are no available upstream networks.
Erik Kline3a5278f2017-06-24 19:29:10 +09001256 handleNewUpstreamNetworkState(null);
1257 }
1258 }
Erik Kline6ff17f72015-12-10 20:42:12 +09001259
Erik Kline3a5278f2017-06-24 19:29:10 +09001260 protected void setDnsForwarders(final Network network, final LinkProperties lp) {
1261 // TODO: Set v4 and/or v6 DNS per available connectivity.
1262 String[] dnsServers = mConfig.defaultIPv4DNS;
1263 final Collection<InetAddress> dnses = lp.getDnsServers();
1264 // TODO: Properly support the absence of DNS servers.
1265 if (dnses != null && !dnses.isEmpty()) {
1266 // TODO: remove this invocation of NetworkUtils.makeStrings().
1267 dnsServers = NetworkUtils.makeStrings(dnses);
Robert Greenwalt6a1967c2010-03-17 11:19:57 -07001268 }
Erik Kline3a5278f2017-06-24 19:29:10 +09001269 try {
1270 mNMService.setDnsForwarders(network, dnsServers);
1271 mLog.log(String.format(
1272 "SET DNS forwarders: network=%s dnsServers=%s",
1273 network, Arrays.toString(dnsServers)));
1274 } catch (Exception e) {
1275 // TODO: Investigate how this can fail and what exactly
1276 // happens if/when such failures occur.
1277 mLog.e("setting DNS forwarders failed, " + e);
1278 transitionTo(mSetDnsForwardersErrorState);
1279 }
1280 }
Erik Kline00019f42016-06-30 19:31:46 +09001281
Remi NGUYEN VAN6c02f992018-03-09 14:07:18 +09001282 protected void notifyDownstreamsOfNewUpstreamIface(InterfaceSet ifaces) {
1283 mCurrentUpstreamIfaceSet = ifaces;
Erik Kline7a4ccc62018-08-27 17:26:47 +09001284 for (IpServer ipServer : mNotifyList) {
1285 ipServer.sendMessage(IpServer.CMD_TETHER_CONNECTION_CHANGED, ifaces);
Erik Kline00019f42016-06-30 19:31:46 +09001286 }
Robert Greenwaltd70a3d42010-02-23 18:15:29 -08001287 }
Robert Greenwalt2a091d72010-02-11 18:18:40 -08001288
Erik Kline3a5278f2017-06-24 19:29:10 +09001289 protected void handleNewUpstreamNetworkState(NetworkState ns) {
1290 mIPv6TetheringCoordinator.updateUpstreamNetworkState(ns);
Erik Klineed962a82017-07-06 19:49:35 +09001291 mOffload.updateUpstreamNetworkState(ns);
Erik Kline3a5278f2017-06-24 19:29:10 +09001292 }
1293
Erik Kline7a4ccc62018-08-27 17:26:47 +09001294 private void handleInterfaceServingStateActive(int mode, IpServer who) {
Erik Klineea9cc482017-03-10 19:35:34 +09001295 if (mNotifyList.indexOf(who) < 0) {
1296 mNotifyList.add(who);
1297 mIPv6TetheringCoordinator.addActiveDownstream(who, mode);
1298 }
1299
Erik Kline7a4ccc62018-08-27 17:26:47 +09001300 if (mode == IpServer.STATE_TETHERED) {
Erik Klineed962a82017-07-06 19:49:35 +09001301 // No need to notify OffloadController just yet as there are no
1302 // "offload-able" prefixes to pass along. This will handled
1303 // when the TISM informs Tethering of its LinkProperties.
Erik Klineea9cc482017-03-10 19:35:34 +09001304 mForwardedDownstreams.add(who);
1305 } else {
Erik Klineed962a82017-07-06 19:49:35 +09001306 mOffload.excludeDownstreamInterface(who.interfaceName());
Erik Klineea9cc482017-03-10 19:35:34 +09001307 mForwardedDownstreams.remove(who);
1308 }
Erik Kline216af6d2017-04-27 20:57:23 +09001309
1310 // If this is a Wi-Fi interface, notify WifiManager of the active serving state.
Erik Kline7a26ba32018-03-09 14:18:02 +09001311 if (who.interfaceType() == TETHERING_WIFI) {
Erik Kline216af6d2017-04-27 20:57:23 +09001312 final WifiManager mgr = getWifiManager();
1313 final String iface = who.interfaceName();
1314 switch (mode) {
Erik Kline7a4ccc62018-08-27 17:26:47 +09001315 case IpServer.STATE_TETHERED:
Erik Kline2efb8272017-05-31 15:53:53 +09001316 mgr.updateInterfaceIpState(iface, IFACE_IP_MODE_TETHERED);
Erik Kline216af6d2017-04-27 20:57:23 +09001317 break;
Erik Kline7a4ccc62018-08-27 17:26:47 +09001318 case IpServer.STATE_LOCAL_ONLY:
Erik Kline2efb8272017-05-31 15:53:53 +09001319 mgr.updateInterfaceIpState(iface, IFACE_IP_MODE_LOCAL_ONLY);
Erik Kline216af6d2017-04-27 20:57:23 +09001320 break;
1321 default:
1322 Log.wtf(TAG, "Unknown active serving mode: " + mode);
1323 break;
1324 }
1325 }
Erik Klineea9cc482017-03-10 19:35:34 +09001326 }
1327
Erik Kline7a4ccc62018-08-27 17:26:47 +09001328 private void handleInterfaceServingStateInactive(IpServer who) {
Erik Klineea9cc482017-03-10 19:35:34 +09001329 mNotifyList.remove(who);
1330 mIPv6TetheringCoordinator.removeActiveDownstream(who);
Erik Klineed962a82017-07-06 19:49:35 +09001331 mOffload.excludeDownstreamInterface(who.interfaceName());
Erik Klineea9cc482017-03-10 19:35:34 +09001332 mForwardedDownstreams.remove(who);
Erik Kline216af6d2017-04-27 20:57:23 +09001333
1334 // If this is a Wi-Fi interface, tell WifiManager of any errors.
Erik Kline7a26ba32018-03-09 14:18:02 +09001335 if (who.interfaceType() == TETHERING_WIFI) {
1336 if (who.lastError() != TETHER_ERROR_NO_ERROR) {
Erik Kline216af6d2017-04-27 20:57:23 +09001337 getWifiManager().updateInterfaceIpState(
Erik Kline2efb8272017-05-31 15:53:53 +09001338 who.interfaceName(), IFACE_IP_MODE_CONFIGURATION_ERROR);
Erik Kline216af6d2017-04-27 20:57:23 +09001339 }
1340 }
Erik Klineea9cc482017-03-10 19:35:34 +09001341 }
1342
Erik Kline3a5278f2017-06-24 19:29:10 +09001343 private void handleUpstreamNetworkMonitorCallback(int arg1, Object o) {
Erik Kline32179ff2017-07-04 18:28:11 +09001344 if (arg1 == UpstreamNetworkMonitor.NOTIFY_LOCAL_PREFIXES) {
Erik Klineed962a82017-07-06 19:49:35 +09001345 mOffload.sendOffloadExemptPrefixes((Set<IpPrefix>) o);
Erik Kline3a5278f2017-06-24 19:29:10 +09001346 return;
1347 }
1348
1349 final NetworkState ns = (NetworkState) o;
1350
1351 if (ns == null || !pertainsToCurrentUpstream(ns)) {
1352 // TODO: In future, this is where upstream evaluation and selection
1353 // could be handled for notifications which include sufficient data.
1354 // For example, after CONNECTIVITY_ACTION listening is removed, here
1355 // is where we could observe a Wi-Fi network becoming available and
1356 // passing validation.
Remi NGUYEN VAN6c02f992018-03-09 14:07:18 +09001357 if (mCurrentUpstreamIfaceSet == null) {
Erik Kline3a5278f2017-06-24 19:29:10 +09001358 // If we have no upstream interface, try to run through upstream
1359 // selection again. If, for example, IPv4 connectivity has shown up
1360 // after IPv6 (e.g., 464xlat became available) we want the chance to
1361 // notice and act accordingly.
1362 chooseUpstreamType(false);
1363 }
1364 return;
1365 }
1366
1367 switch (arg1) {
Erik Kline3a5278f2017-06-24 19:29:10 +09001368 case UpstreamNetworkMonitor.EVENT_ON_CAPABILITIES:
1369 handleNewUpstreamNetworkState(ns);
1370 break;
1371 case UpstreamNetworkMonitor.EVENT_ON_LINKPROPERTIES:
Remi NGUYEN VAN6c02f992018-03-09 14:07:18 +09001372 chooseUpstreamType(false);
Erik Kline3a5278f2017-06-24 19:29:10 +09001373 break;
1374 case UpstreamNetworkMonitor.EVENT_ON_LOST:
1375 // TODO: Re-evaluate possible upstreams. Currently upstream
1376 // reevaluation is triggered via received CONNECTIVITY_ACTION
1377 // broadcasts that result in being passed a
1378 // TetherMasterSM.CMD_UPSTREAM_CHANGED.
1379 handleNewUpstreamNetworkState(null);
1380 break;
1381 default:
1382 mLog.e("Unknown arg1 value: " + arg1);
1383 break;
1384 }
1385 }
1386
1387 class TetherModeAliveState extends State {
Erik Klineea9cc482017-03-10 19:35:34 +09001388 boolean mUpstreamWanted = false;
Erik Kline6ff17f72015-12-10 20:42:12 +09001389 boolean mTryCell = true;
Erik Klinee0cce212017-03-06 14:05:23 +09001390
Robert Greenwalt2a091d72010-02-11 18:18:40 -08001391 @Override
1392 public void enter() {
Erik Kline1fdc2e22017-05-08 17:56:35 +09001393 // If turning on master tether settings fails, we have already
1394 // transitioned to an error state; exit early.
1395 if (!turnOnMasterTetherSettings()) {
1396 return;
1397 }
1398
markchiena6c72872018-11-13 18:34:56 +09001399 mUpstreamNetworkMonitor.startObserveAllNetworks();
Robert Greenwalt4f74d552011-12-19 16:59:31 -08001400
Erik Klinef4b6e342017-04-25 19:19:59 +09001401 // TODO: De-duplicate with updateUpstreamWanted() below.
Erik Klineea9cc482017-03-10 19:35:34 +09001402 if (upstreamWanted()) {
1403 mUpstreamWanted = true;
Erik Klineed962a82017-07-06 19:49:35 +09001404 mOffload.start();
Erik Klineea9cc482017-03-10 19:35:34 +09001405 chooseUpstreamType(true);
1406 mTryCell = false;
1407 }
Robert Greenwalt6a1967c2010-03-17 11:19:57 -07001408 }
Erik Klinefa37b2f2016-08-02 18:27:03 +09001409
Robert Greenwalt6a1967c2010-03-17 11:19:57 -07001410 @Override
1411 public void exit() {
Erik Klineed962a82017-07-06 19:49:35 +09001412 mOffload.stop();
Erik Kline6ff17f72015-12-10 20:42:12 +09001413 mUpstreamNetworkMonitor.stop();
Erik Kline22108902017-07-06 16:40:06 +09001414 notifyDownstreamsOfNewUpstreamIface(null);
Erik Klinefa37b2f2016-08-02 18:27:03 +09001415 handleNewUpstreamNetworkState(null);
Robert Greenwalt2a091d72010-02-11 18:18:40 -08001416 }
Erik Klinefa37b2f2016-08-02 18:27:03 +09001417
Erik Klineea9cc482017-03-10 19:35:34 +09001418 private boolean updateUpstreamWanted() {
1419 final boolean previousUpstreamWanted = mUpstreamWanted;
1420 mUpstreamWanted = upstreamWanted();
Erik Klinef4b6e342017-04-25 19:19:59 +09001421 if (mUpstreamWanted != previousUpstreamWanted) {
1422 if (mUpstreamWanted) {
Erik Klineed962a82017-07-06 19:49:35 +09001423 mOffload.start();
Erik Klinef4b6e342017-04-25 19:19:59 +09001424 } else {
Erik Klineed962a82017-07-06 19:49:35 +09001425 mOffload.stop();
Erik Klinef4b6e342017-04-25 19:19:59 +09001426 }
1427 }
Erik Klineea9cc482017-03-10 19:35:34 +09001428 return previousUpstreamWanted;
1429 }
1430
Robert Greenwalt2a091d72010-02-11 18:18:40 -08001431 @Override
1432 public boolean processMessage(Message message) {
Erik Kline22108902017-07-06 16:40:06 +09001433 logMessage(this, message.what);
Robert Greenwalt2a091d72010-02-11 18:18:40 -08001434 boolean retValue = true;
1435 switch (message.what) {
Erik Klineea9cc482017-03-10 19:35:34 +09001436 case EVENT_IFACE_SERVING_STATE_ACTIVE: {
Erik Kline7a4ccc62018-08-27 17:26:47 +09001437 IpServer who = (IpServer) message.obj;
Robert Greenwalt68ea9b02012-05-10 16:58:16 -07001438 if (VDBG) Log.d(TAG, "Tether Mode requested by " + who);
Erik Klineea9cc482017-03-10 19:35:34 +09001439 handleInterfaceServingStateActive(message.arg1, who);
Erik Kline7a4ccc62018-08-27 17:26:47 +09001440 who.sendMessage(IpServer.CMD_TETHER_CONNECTION_CHANGED,
Remi NGUYEN VAN6c02f992018-03-09 14:07:18 +09001441 mCurrentUpstreamIfaceSet);
Erik Klineea9cc482017-03-10 19:35:34 +09001442 // If there has been a change and an upstream is now
1443 // desired, kick off the selection process.
1444 final boolean previousUpstreamWanted = updateUpstreamWanted();
1445 if (!previousUpstreamWanted && mUpstreamWanted) {
1446 chooseUpstreamType(true);
1447 }
Robert Greenwalt2a091d72010-02-11 18:18:40 -08001448 break;
Erik Kline6e29bf02016-08-15 16:16:18 +09001449 }
Erik Klineea9cc482017-03-10 19:35:34 +09001450 case EVENT_IFACE_SERVING_STATE_INACTIVE: {
Erik Kline7a4ccc62018-08-27 17:26:47 +09001451 IpServer who = (IpServer) message.obj;
Robert Greenwalt68ea9b02012-05-10 16:58:16 -07001452 if (VDBG) Log.d(TAG, "Tether Mode unrequested by " + who);
Erik Klineea9cc482017-03-10 19:35:34 +09001453 handleInterfaceServingStateInactive(who);
1454
1455 if (mNotifyList.isEmpty()) {
Erik Kline47222fc2017-04-30 19:36:15 +09001456 // This transitions us out of TetherModeAliveState,
1457 // either to InitialState or an error state.
Erik Kline1fdc2e22017-05-08 17:56:35 +09001458 turnOffMasterTetherSettings();
1459 break;
1460 }
1461
1462 if (DBG) {
1463 Log.d(TAG, "TetherModeAlive still has " + mNotifyList.size() +
1464 " live requests:");
Erik Kline7a4ccc62018-08-27 17:26:47 +09001465 for (IpServer o : mNotifyList) {
Erik Kline1fdc2e22017-05-08 17:56:35 +09001466 Log.d(TAG, " " + o);
Robert Greenwalt2a091d72010-02-11 18:18:40 -08001467 }
1468 }
Erik Klineea9cc482017-03-10 19:35:34 +09001469 // If there has been a change and an upstream is no
1470 // longer desired, release any mobile requests.
1471 final boolean previousUpstreamWanted = updateUpstreamWanted();
1472 if (previousUpstreamWanted && !mUpstreamWanted) {
1473 mUpstreamNetworkMonitor.releaseMobileNetworkRequest();
1474 }
Robert Greenwalt2a091d72010-02-11 18:18:40 -08001475 break;
Erik Kline6e29bf02016-08-15 16:16:18 +09001476 }
Erik Kline6e9a1012017-06-06 19:24:21 +09001477 case EVENT_IFACE_UPDATE_LINKPROPERTIES: {
1478 final LinkProperties newLp = (LinkProperties) message.obj;
Erik Kline7a4ccc62018-08-27 17:26:47 +09001479 if (message.arg1 == IpServer.STATE_TETHERED) {
Erik Klineed962a82017-07-06 19:49:35 +09001480 mOffload.updateDownstreamLinkProperties(newLp);
Erik Kline6e9a1012017-06-06 19:24:21 +09001481 } else {
Erik Klineed962a82017-07-06 19:49:35 +09001482 mOffload.excludeDownstreamInterface(newLp.getInterfaceName());
Erik Kline6e9a1012017-06-06 19:24:21 +09001483 }
1484 break;
1485 }
Robert Greenwalt6a1967c2010-03-17 11:19:57 -07001486 case CMD_UPSTREAM_CHANGED:
Erik Klineea9cc482017-03-10 19:35:34 +09001487 updateUpstreamWanted();
1488 if (!mUpstreamWanted) break;
1489
Erik Klinefb413432017-02-14 18:26:04 +09001490 // Need to try DUN immediately if Wi-Fi goes down.
1491 chooseUpstreamType(true);
1492 mTryCell = false;
Robert Greenwalt6a1967c2010-03-17 11:19:57 -07001493 break;
Kazuhiro Ondo01758e82011-04-30 20:10:57 -05001494 case CMD_RETRY_UPSTREAM:
Erik Klineea9cc482017-03-10 19:35:34 +09001495 updateUpstreamWanted();
1496 if (!mUpstreamWanted) break;
1497
Kazuhiro Ondo01758e82011-04-30 20:10:57 -05001498 chooseUpstreamType(mTryCell);
1499 mTryCell = !mTryCell;
1500 break;
Erik Kline00019f42016-06-30 19:31:46 +09001501 case EVENT_UPSTREAM_CALLBACK: {
Erik Klineea9cc482017-03-10 19:35:34 +09001502 updateUpstreamWanted();
Erik Kline3a5278f2017-06-24 19:29:10 +09001503 if (mUpstreamWanted) {
1504 handleUpstreamNetworkMonitorCallback(message.arg1, message.obj);
Erik Kline6ff17f72015-12-10 20:42:12 +09001505 }
1506 break;
Erik Kline00019f42016-06-30 19:31:46 +09001507 }
Kazuhiro Ondo01758e82011-04-30 20:10:57 -05001508 default:
1509 retValue = false;
1510 break;
Robert Greenwalt2a091d72010-02-11 18:18:40 -08001511 }
1512 return retValue;
1513 }
1514 }
1515
Wink Saville64c42ca2011-04-18 14:55:10 -07001516 class ErrorState extends State {
Erik Kline8351faa2017-04-17 16:47:23 +09001517 private int mErrorNotification;
1518
Robert Greenwalt2a091d72010-02-11 18:18:40 -08001519 @Override
1520 public boolean processMessage(Message message) {
1521 boolean retValue = true;
1522 switch (message.what) {
Erik Klineea9cc482017-03-10 19:35:34 +09001523 case EVENT_IFACE_SERVING_STATE_ACTIVE:
Erik Kline7a4ccc62018-08-27 17:26:47 +09001524 IpServer who = (IpServer) message.obj;
Robert Greenwalt6a1967c2010-03-17 11:19:57 -07001525 who.sendMessage(mErrorNotification);
Robert Greenwalt2a091d72010-02-11 18:18:40 -08001526 break;
Yohei, Oshima977aad52016-12-08 13:39:20 +09001527 case CMD_CLEAR_ERROR:
Erik Kline7a26ba32018-03-09 14:18:02 +09001528 mErrorNotification = TETHER_ERROR_NO_ERROR;
Yohei, Oshima977aad52016-12-08 13:39:20 +09001529 transitionTo(mInitialState);
1530 break;
Robert Greenwalt2a091d72010-02-11 18:18:40 -08001531 default:
1532 retValue = false;
1533 }
1534 return retValue;
1535 }
Erik Kline8351faa2017-04-17 16:47:23 +09001536
Robert Greenwalt2a091d72010-02-11 18:18:40 -08001537 void notify(int msgType) {
1538 mErrorNotification = msgType;
Erik Kline7a4ccc62018-08-27 17:26:47 +09001539 for (IpServer ipServer : mNotifyList) {
1540 ipServer.sendMessage(msgType);
Robert Greenwalt2a091d72010-02-11 18:18:40 -08001541 }
1542 }
1543
1544 }
Erik Kline8351faa2017-04-17 16:47:23 +09001545
Robert Greenwalt2a091d72010-02-11 18:18:40 -08001546 class SetIpForwardingEnabledErrorState extends ErrorState {
1547 @Override
1548 public void enter() {
1549 Log.e(TAG, "Error in setIpForwardingEnabled");
Erik Kline7a4ccc62018-08-27 17:26:47 +09001550 notify(IpServer.CMD_IP_FORWARDING_ENABLE_ERROR);
Robert Greenwalt2a091d72010-02-11 18:18:40 -08001551 }
1552 }
1553
1554 class SetIpForwardingDisabledErrorState extends ErrorState {
1555 @Override
1556 public void enter() {
1557 Log.e(TAG, "Error in setIpForwardingDisabled");
Erik Kline7a4ccc62018-08-27 17:26:47 +09001558 notify(IpServer.CMD_IP_FORWARDING_DISABLE_ERROR);
Robert Greenwalt2a091d72010-02-11 18:18:40 -08001559 }
1560 }
1561
1562 class StartTetheringErrorState extends ErrorState {
1563 @Override
1564 public void enter() {
1565 Log.e(TAG, "Error in startTethering");
Erik Kline7a4ccc62018-08-27 17:26:47 +09001566 notify(IpServer.CMD_START_TETHERING_ERROR);
Robert Greenwalt2a091d72010-02-11 18:18:40 -08001567 try {
Chia-chi Yehc9338302011-05-11 16:35:13 -07001568 mNMService.setIpForwardingEnabled(false);
Robert Greenwalt2a091d72010-02-11 18:18:40 -08001569 } catch (Exception e) {}
1570 }
1571 }
1572
1573 class StopTetheringErrorState extends ErrorState {
1574 @Override
1575 public void enter() {
1576 Log.e(TAG, "Error in stopTethering");
Erik Kline7a4ccc62018-08-27 17:26:47 +09001577 notify(IpServer.CMD_STOP_TETHERING_ERROR);
Robert Greenwalt2a091d72010-02-11 18:18:40 -08001578 try {
Chia-chi Yehc9338302011-05-11 16:35:13 -07001579 mNMService.setIpForwardingEnabled(false);
Robert Greenwalt2a091d72010-02-11 18:18:40 -08001580 } catch (Exception e) {}
1581 }
1582 }
1583
1584 class SetDnsForwardersErrorState extends ErrorState {
1585 @Override
1586 public void enter() {
1587 Log.e(TAG, "Error in setDnsForwarders");
Erik Kline7a4ccc62018-08-27 17:26:47 +09001588 notify(IpServer.CMD_SET_DNS_FORWARDERS_ERROR);
Robert Greenwalt2a091d72010-02-11 18:18:40 -08001589 try {
Chia-chi Yehc9338302011-05-11 16:35:13 -07001590 mNMService.stopTethering();
Robert Greenwalt2a091d72010-02-11 18:18:40 -08001591 } catch (Exception e) {}
1592 try {
Chia-chi Yehc9338302011-05-11 16:35:13 -07001593 mNMService.setIpForwardingEnabled(false);
Robert Greenwalt2a091d72010-02-11 18:18:40 -08001594 } catch (Exception e) {}
1595 }
1596 }
Erik Klineed962a82017-07-06 19:49:35 +09001597
1598 // A wrapper class to handle multiple situations where several calls to
1599 // the OffloadController need to happen together.
1600 //
1601 // TODO: This suggests that the interface between OffloadController and
1602 // Tethering is in need of improvement. Refactor these calls into the
1603 // OffloadController implementation.
1604 class OffloadWrapper {
1605 public void start() {
1606 mOffloadController.start();
1607 sendOffloadExemptPrefixes();
1608 }
1609
1610 public void stop() {
1611 mOffloadController.stop();
1612 }
1613
1614 public void updateUpstreamNetworkState(NetworkState ns) {
1615 mOffloadController.setUpstreamLinkProperties(
1616 (ns != null) ? ns.linkProperties : null);
1617 }
1618
1619 public void updateDownstreamLinkProperties(LinkProperties newLp) {
1620 // Update the list of offload-exempt prefixes before adding
1621 // new prefixes on downstream interfaces to the offload HAL.
1622 sendOffloadExemptPrefixes();
1623 mOffloadController.notifyDownstreamLinkProperties(newLp);
1624 }
1625
1626 public void excludeDownstreamInterface(String ifname) {
1627 // This and other interfaces may be in local-only hotspot mode;
1628 // resend all local prefixes to the OffloadController.
1629 sendOffloadExemptPrefixes();
1630 mOffloadController.removeDownstreamInterface(ifname);
1631 }
1632
1633 public void sendOffloadExemptPrefixes() {
1634 sendOffloadExemptPrefixes(mUpstreamNetworkMonitor.getLocalPrefixes());
1635 }
1636
1637 public void sendOffloadExemptPrefixes(final Set<IpPrefix> localPrefixes) {
1638 // Add in well-known minimum set.
1639 PrefixUtils.addNonForwardablePrefixes(localPrefixes);
1640 // Add tragically hardcoded prefixes.
1641 localPrefixes.add(PrefixUtils.DEFAULT_WIFI_P2P_PREFIX);
1642
1643 // Maybe add prefixes or addresses for downstreams, depending on
1644 // the IP serving mode of each.
Erik Kline7a4ccc62018-08-27 17:26:47 +09001645 for (IpServer ipServer : mNotifyList) {
1646 final LinkProperties lp = ipServer.linkProperties();
Erik Klineed962a82017-07-06 19:49:35 +09001647
Erik Kline7a4ccc62018-08-27 17:26:47 +09001648 switch (ipServer.servingMode()) {
1649 case IpServer.STATE_UNAVAILABLE:
1650 case IpServer.STATE_AVAILABLE:
Erik Klineed962a82017-07-06 19:49:35 +09001651 // No usable LinkProperties in these states.
1652 continue;
Erik Kline7a4ccc62018-08-27 17:26:47 +09001653 case IpServer.STATE_TETHERED:
Erik Klineed962a82017-07-06 19:49:35 +09001654 // Only add IPv4 /32 and IPv6 /128 prefixes. The
1655 // directly-connected prefixes will be sent as
1656 // downstream "offload-able" prefixes.
1657 for (LinkAddress addr : lp.getAllLinkAddresses()) {
1658 final InetAddress ip = addr.getAddress();
1659 if (ip.isLinkLocalAddress()) continue;
1660 localPrefixes.add(PrefixUtils.ipAddressAsPrefix(ip));
1661 }
1662 break;
Erik Kline7a4ccc62018-08-27 17:26:47 +09001663 case IpServer.STATE_LOCAL_ONLY:
Erik Klineed962a82017-07-06 19:49:35 +09001664 // Add prefixes covering all local IPs.
1665 localPrefixes.addAll(PrefixUtils.localPrefixesFrom(lp));
1666 break;
1667 }
1668 }
1669
1670 mOffloadController.setLocalPrefixes(localPrefixes);
1671 }
1672 }
Robert Greenwalt2a091d72010-02-11 18:18:40 -08001673 }
1674
markchiena6c72872018-11-13 18:34:56 +09001675 public void systemReady() {
1676 mUpstreamNetworkMonitor.startTrackDefaultNetwork(mDeps.getDefaultNetworkRequest());
1677 }
1678
markchienf2731272019-01-16 17:44:13 +08001679 /** Get the latest value of the tethering entitlement check. */
1680 public void getLatestTetheringEntitlementValue(int type, ResultReceiver receiver,
1681 boolean showEntitlementUi) {
1682 if (receiver != null) {
1683 mEntitlementMgr.getLatestTetheringEntitlementValue(type, receiver, showEntitlementUi);
1684 }
1685 }
1686
Christopher Wiley499a57a2016-05-16 16:19:07 -07001687 @Override
Lorenzo Colittie3805462015-06-03 11:18:24 +09001688 public void dump(FileDescriptor fd, PrintWriter writer, String[] args) {
Christopher Wiley499a57a2016-05-16 16:19:07 -07001689 // Binder.java closes the resource for us.
1690 @SuppressWarnings("resource")
Lorenzo Colittie3805462015-06-03 11:18:24 +09001691 final IndentingPrintWriter pw = new IndentingPrintWriter(writer, " ");
Jeff Sharkeyfe9a53b2017-03-31 14:08:23 -06001692 if (!DumpUtils.checkDumpPermission(mContext, TAG, pw)) return;
Robert Greenwalt2a091d72010-02-11 18:18:40 -08001693
Lorenzo Colittie3805462015-06-03 11:18:24 +09001694 pw.println("Tethering:");
1695 pw.increaseIndent();
Lorenzo Colittie3805462015-06-03 11:18:24 +09001696
Erik Kline9db1b542017-03-16 14:10:27 +09001697 pw.println("Configuration:");
1698 pw.increaseIndent();
1699 final TetheringConfiguration cfg = mConfig;
1700 cfg.dump(pw);
1701 pw.decreaseIndent();
1702
1703 synchronized (mPublicSync) {
Robert Greenwaltb4453622011-11-03 16:01:40 -07001704 pw.println("Tether state:");
Lorenzo Colittie3805462015-06-03 11:18:24 +09001705 pw.increaseIndent();
Christopher Wileyd985dde2016-05-31 10:44:35 -07001706 for (int i = 0; i < mTetherStates.size(); i++) {
1707 final String iface = mTetherStates.keyAt(i);
1708 final TetherState tetherState = mTetherStates.valueAt(i);
1709 pw.print(iface + " - ");
1710
Hugo Benichib55fb222017-03-10 14:20:57 +09001711 switch (tetherState.lastState) {
Erik Kline7a4ccc62018-08-27 17:26:47 +09001712 case IpServer.STATE_UNAVAILABLE:
Christopher Wileyd985dde2016-05-31 10:44:35 -07001713 pw.print("UnavailableState");
1714 break;
Erik Kline7a4ccc62018-08-27 17:26:47 +09001715 case IpServer.STATE_AVAILABLE:
Christopher Wileyd985dde2016-05-31 10:44:35 -07001716 pw.print("AvailableState");
1717 break;
Erik Kline7a4ccc62018-08-27 17:26:47 +09001718 case IpServer.STATE_TETHERED:
Christopher Wileyd985dde2016-05-31 10:44:35 -07001719 pw.print("TetheredState");
1720 break;
Erik Kline7a4ccc62018-08-27 17:26:47 +09001721 case IpServer.STATE_LOCAL_ONLY:
Erik Klineea9cc482017-03-10 19:35:34 +09001722 pw.print("LocalHotspotState");
1723 break;
Christopher Wileyd985dde2016-05-31 10:44:35 -07001724 default:
1725 pw.print("UnknownState");
1726 break;
1727 }
Hugo Benichib55fb222017-03-10 14:20:57 +09001728 pw.println(" - lastError = " + tetherState.lastError);
Robert Greenwalt2a091d72010-02-11 18:18:40 -08001729 }
Erik Klineea9cc482017-03-10 19:35:34 +09001730 pw.println("Upstream wanted: " + upstreamWanted());
Remi NGUYEN VAN6c02f992018-03-09 14:07:18 +09001731 pw.println("Current upstream interface(s): " + mCurrentUpstreamIfaceSet);
Lorenzo Colittie3805462015-06-03 11:18:24 +09001732 pw.decreaseIndent();
Robert Greenwalt2a091d72010-02-11 18:18:40 -08001733 }
Erik Kline1fdc2e22017-05-08 17:56:35 +09001734
Lorenzo Colittic2519c52017-07-13 01:48:26 +09001735 pw.println("Hardware offload:");
1736 pw.increaseIndent();
1737 mOffloadController.dump(pw);
1738 pw.decreaseIndent();
1739
Erik Kline1fdc2e22017-05-08 17:56:35 +09001740 pw.println("Log:");
1741 pw.increaseIndent();
Erik Kline7747fd42017-05-12 16:52:48 +09001742 if (argsContain(args, SHORT_ARG)) {
1743 pw.println("<log removed for brevity>");
1744 } else {
1745 mLog.dump(fd, pw, args);
1746 }
Erik Kline1fdc2e22017-05-08 17:56:35 +09001747 pw.decreaseIndent();
1748
Lorenzo Colittie3805462015-06-03 11:18:24 +09001749 pw.decreaseIndent();
Robert Greenwaltd0e18ff2010-01-26 11:40:34 -08001750 }
Christopher Wileye03fb442016-05-18 13:45:20 -07001751
Erik Kline7747fd42017-05-12 16:52:48 +09001752 private static boolean argsContain(String[] args, String target) {
1753 for (String arg : args) {
Erik Klineee363c42017-05-29 09:11:03 +09001754 if (target.equals(arg)) return true;
Erik Kline7747fd42017-05-12 16:52:48 +09001755 }
1756 return false;
1757 }
1758
Erik Kline7a4ccc62018-08-27 17:26:47 +09001759 private IpServer.Callback makeControlCallback() {
1760 return new IpServer.Callback() {
Erik Kline6e9a1012017-06-06 19:24:21 +09001761 @Override
Erik Kline7a4ccc62018-08-27 17:26:47 +09001762 public void updateInterfaceState(IpServer who, int state, int lastError) {
1763 notifyInterfaceStateChange(who, state, lastError);
Erik Kline6e9a1012017-06-06 19:24:21 +09001764 }
1765
1766 @Override
Erik Kline7a4ccc62018-08-27 17:26:47 +09001767 public void updateLinkProperties(IpServer who, LinkProperties newLp) {
1768 notifyLinkPropertiesChanged(who, newLp);
Erik Kline6e9a1012017-06-06 19:24:21 +09001769 }
1770 };
1771 }
1772
1773 // TODO: Move into TetherMasterSM.
Erik Kline7a4ccc62018-08-27 17:26:47 +09001774 private void notifyInterfaceStateChange(IpServer who, int state, int error) {
1775 final String iface = who.interfaceName();
Christopher Wileyd985dde2016-05-31 10:44:35 -07001776 synchronized (mPublicSync) {
Erik Kline216af6d2017-04-27 20:57:23 +09001777 final TetherState tetherState = mTetherStates.get(iface);
Erik Kline7a4ccc62018-08-27 17:26:47 +09001778 if (tetherState != null && tetherState.ipServer.equals(who)) {
Hugo Benichib55fb222017-03-10 14:20:57 +09001779 tetherState.lastState = state;
1780 tetherState.lastError = error;
Christopher Wileyd985dde2016-05-31 10:44:35 -07001781 } else {
1782 if (DBG) Log.d(TAG, "got notification from stale iface " + iface);
1783 }
1784 }
1785
Erik Kline7747fd42017-05-12 16:52:48 +09001786 mLog.log(String.format("OBSERVED iface=%s state=%s error=%s", iface, state, error));
Christopher Wileyd985dde2016-05-31 10:44:35 -07001787
Christopher Wileydeebfec2016-09-16 11:14:36 -07001788 try {
1789 // Notify that we're tethering (or not) this interface.
1790 // This is how data saver for instance knows if the user explicitly
1791 // turned on tethering (thus keeping us from being in data saver mode).
Erik Kline7a4ccc62018-08-27 17:26:47 +09001792 mPolicyManager.onTetheringChanged(iface, state == IpServer.STATE_TETHERED);
Christopher Wileydeebfec2016-09-16 11:14:36 -07001793 } catch (RemoteException e) {
1794 // Not really very much we can do here.
1795 }
1796
Yohei, Oshima977aad52016-12-08 13:39:20 +09001797 // If TetherMasterSM is in ErrorState, TetherMasterSM stays there.
1798 // Thus we give a chance for TetherMasterSM to recover to InitialState
1799 // by sending CMD_CLEAR_ERROR
Erik Kline7a26ba32018-03-09 14:18:02 +09001800 if (error == TETHER_ERROR_MASTER_ERROR) {
Yohei, Oshima977aad52016-12-08 13:39:20 +09001801 mTetherMasterSM.sendMessage(TetherMasterSM.CMD_CLEAR_ERROR, who);
1802 }
Erik Klineea9cc482017-03-10 19:35:34 +09001803 int which;
Christopher Wileyd985dde2016-05-31 10:44:35 -07001804 switch (state) {
Erik Kline7a4ccc62018-08-27 17:26:47 +09001805 case IpServer.STATE_UNAVAILABLE:
1806 case IpServer.STATE_AVAILABLE:
Erik Klineea9cc482017-03-10 19:35:34 +09001807 which = TetherMasterSM.EVENT_IFACE_SERVING_STATE_INACTIVE;
Christopher Wileyd985dde2016-05-31 10:44:35 -07001808 break;
Erik Kline7a4ccc62018-08-27 17:26:47 +09001809 case IpServer.STATE_TETHERED:
1810 case IpServer.STATE_LOCAL_ONLY:
Erik Klineea9cc482017-03-10 19:35:34 +09001811 which = TetherMasterSM.EVENT_IFACE_SERVING_STATE_ACTIVE;
Christopher Wileyd985dde2016-05-31 10:44:35 -07001812 break;
Erik Klineea9cc482017-03-10 19:35:34 +09001813 default:
1814 Log.wtf(TAG, "Unknown interface state: " + state);
1815 return;
Christopher Wileyd985dde2016-05-31 10:44:35 -07001816 }
Erik Klineea9cc482017-03-10 19:35:34 +09001817 mTetherMasterSM.sendMessage(which, state, 0, who);
Christopher Wileyd985dde2016-05-31 10:44:35 -07001818 sendTetherStateChangedBroadcast();
1819 }
1820
Erik Kline7a4ccc62018-08-27 17:26:47 +09001821 private void notifyLinkPropertiesChanged(IpServer who, LinkProperties newLp) {
1822 final String iface = who.interfaceName();
Erik Kline6e9a1012017-06-06 19:24:21 +09001823 final int state;
1824 synchronized (mPublicSync) {
1825 final TetherState tetherState = mTetherStates.get(iface);
Erik Kline7a4ccc62018-08-27 17:26:47 +09001826 if (tetherState != null && tetherState.ipServer.equals(who)) {
Erik Kline6e9a1012017-06-06 19:24:21 +09001827 state = tetherState.lastState;
1828 } else {
1829 mLog.log("got notification from stale iface " + iface);
1830 return;
1831 }
1832 }
1833
Erik Kline7fd696c2017-06-12 18:20:08 +09001834 mLog.log(String.format(
1835 "OBSERVED LinkProperties update iface=%s state=%s lp=%s",
Erik Kline7a4ccc62018-08-27 17:26:47 +09001836 iface, IpServer.getStateString(state), newLp));
Erik Kline6e9a1012017-06-06 19:24:21 +09001837 final int which = TetherMasterSM.EVENT_IFACE_UPDATE_LINKPROPERTIES;
1838 mTetherMasterSM.sendMessage(which, state, 0, newLp);
1839 }
1840
Erik Kline4dd9bb82017-04-26 11:11:07 +09001841 private void maybeTrackNewInterfaceLocked(final String iface) {
1842 // If we don't care about this type of interface, ignore.
1843 final int interfaceType = ifaceNameToType(iface);
Erik Kline7a26ba32018-03-09 14:18:02 +09001844 if (interfaceType == TETHERING_INVALID) {
Erik Kline4dd9bb82017-04-26 11:11:07 +09001845 mLog.log(iface + " is not a tetherable iface, ignoring");
1846 return;
1847 }
Erik Klinea9cde8b2017-06-20 21:18:31 +09001848 maybeTrackNewInterfaceLocked(iface, interfaceType);
1849 }
Erik Kline4dd9bb82017-04-26 11:11:07 +09001850
Erik Klinea9cde8b2017-06-20 21:18:31 +09001851 private void maybeTrackNewInterfaceLocked(final String iface, int interfaceType) {
Erik Kline4dd9bb82017-04-26 11:11:07 +09001852 // If we have already started a TISM for this interface, skip.
1853 if (mTetherStates.containsKey(iface)) {
1854 mLog.log("active iface (" + iface + ") reported as added, ignoring");
1855 return;
1856 }
1857
1858 mLog.log("adding TetheringInterfaceStateMachine for: " + iface);
1859 final TetherState tetherState = new TetherState(
Erik Kline7a4ccc62018-08-27 17:26:47 +09001860 new IpServer(iface, mLooper, interfaceType, mLog, mNMService, mStatsService,
1861 makeControlCallback(), mConfig.enableLegacyDhcpServer,
Remi NGUYEN VAN310da6f2019-02-14 18:04:20 +09001862 mDeps.getIpServerDependencies()));
Christopher Wileyd985dde2016-05-31 10:44:35 -07001863 mTetherStates.put(iface, tetherState);
Erik Kline7a4ccc62018-08-27 17:26:47 +09001864 tetherState.ipServer.start();
Christopher Wileye03fb442016-05-18 13:45:20 -07001865 }
Erik Kline3e756652017-01-17 13:42:19 +09001866
Erik Kline4dd9bb82017-04-26 11:11:07 +09001867 private void stopTrackingInterfaceLocked(final String iface) {
1868 final TetherState tetherState = mTetherStates.get(iface);
1869 if (tetherState == null) {
1870 mLog.log("attempting to remove unknown iface (" + iface + "), ignoring");
1871 return;
1872 }
Erik Kline7a4ccc62018-08-27 17:26:47 +09001873 tetherState.ipServer.stop();
Erik Kline4dd9bb82017-04-26 11:11:07 +09001874 mLog.log("removing TetheringInterfaceStateMachine for: " + iface);
1875 mTetherStates.remove(iface);
1876 }
1877
Erik Kline3e756652017-01-17 13:42:19 +09001878 private static String[] copy(String[] strarray) {
1879 return Arrays.copyOf(strarray, strarray.length);
1880 }
Robert Greenwaltd0e18ff2010-01-26 11:40:34 -08001881}