blob: da21729d72e7ab399eb36071ec79a6c79486a138 [file] [log] [blame]
Robert Greenwalt3901edb2010-01-26 10:22:37 -08001/*
2 * Copyright (C) 2008 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.settings;
18
Fan Zhangdefb1182017-07-25 14:18:47 -070019import static android.net.ConnectivityManager.TETHERING_BLUETOOTH;
20import static android.net.ConnectivityManager.TETHERING_USB;
Yuuki Habu557e4572020-02-05 16:24:52 +090021import static android.net.TetheringManager.TETHERING_ETHERNET;
Fan Zhangdefb1182017-07-25 14:18:47 -070022
Alex Johnston7d4db752021-01-15 20:46:05 +000023import static com.android.settingslib.RestrictedLockUtilsInternal.checkIfUsbDataSignalingIsDisabled;
24
Daisuke Miyakawa9c602c42010-09-10 12:04:37 -070025import android.app.Activity;
Fan Zhang31b21002019-01-16 13:49:47 -080026import android.app.settings.SettingsEnums;
Danica Chang32711b62010-08-10 18:41:29 -070027import android.bluetooth.BluetoothAdapter;
28import android.bluetooth.BluetoothPan;
Jaikumar Ganesh9ad703c2011-02-23 13:08:09 -080029import android.bluetooth.BluetoothProfile;
Robert Greenwalt3901edb2010-01-26 10:22:37 -080030import android.content.BroadcastReceiver;
31import android.content.Context;
32import android.content.Intent;
33import android.content.IntentFilter;
Mike Lockwood69a09572011-07-19 13:30:03 -070034import android.hardware.usb.UsbManager;
Robert Greenwalt3901edb2010-01-26 10:22:37 -080035import android.net.ConnectivityManager;
Yuuki Habu557e4572020-02-05 16:24:52 +090036import android.net.EthernetManager;
Xiao Madb7d7de2022-01-30 11:18:29 +000037import android.net.IpConfiguration;
Yuuki Habu557e4572020-02-05 16:24:52 +090038import android.net.TetheringManager;
Arc Wang005fe692020-03-04 21:03:54 +080039import android.net.wifi.WifiManager;
Daisuke Miyakawa9c602c42010-09-10 12:04:37 -070040import android.os.Bundle;
Mike Lockwood26dad3e2010-03-03 06:19:55 -050041import android.os.Environment;
Jeremy Kleine3e7b952016-01-25 14:43:49 -080042import android.os.Handler;
Alex Johnston7d4db752021-01-15 20:46:05 +000043import android.os.UserHandle;
Amith Yamasani394eaa22013-06-11 11:04:44 -070044import android.os.UserManager;
Chilund3358d82018-03-26 18:28:56 +080045import android.provider.SearchIndexableResource;
Zhen Zhangd2a7f9a2020-01-30 15:12:50 -080046import android.util.FeatureFlagUtils;
Hugh Chena3a06612021-07-13 10:42:00 +080047import android.util.Log;
Felipe Leme5c091842016-04-21 17:17:28 -070048
Xiao Madb7d7de2022-01-30 11:18:29 +000049import androidx.annotation.NonNull;
Fan Zhang23f8d592018-08-28 15:11:40 -070050import androidx.annotation.VisibleForTesting;
51import androidx.preference.Preference;
52import androidx.preference.SwitchPreference;
53
Zhen Zhangd2a7f9a2020-01-30 15:12:50 -080054import com.android.settings.core.FeatureFlags;
Felipe Leme5c091842016-04-21 17:17:28 -070055import com.android.settings.datausage.DataSaverBackend;
Chilund3358d82018-03-26 18:28:56 +080056import com.android.settings.search.BaseSearchIndexProvider;
Fan Zhang78d5efd2017-06-06 16:55:53 -070057import com.android.settings.wifi.tether.WifiTetherPreferenceController;
Alex Johnston685cacb2021-08-02 14:16:49 +010058import com.android.settingslib.RestrictedLockUtils;
Alex Johnston7d4db752021-01-15 20:46:05 +000059import com.android.settingslib.RestrictedSwitchPreference;
Jason Monk4896c012015-02-11 13:30:41 -050060import com.android.settingslib.TetherUtil;
Chilund3358d82018-03-26 18:28:56 +080061import com.android.settingslib.search.SearchIndexable;
Jeff Sharkeye16e44f2014-11-13 18:02:31 -080062
Jeremy Kleine3e7b952016-01-25 14:43:49 -080063import java.lang.ref.WeakReference;
Robert Greenwaltc4764d22010-02-12 14:21:37 -080064import java.util.ArrayList;
Chilund3358d82018-03-26 18:28:56 +080065import java.util.Arrays;
Xiao Madb7d7de2022-01-30 11:18:29 +000066import java.util.HashSet;
Chilund3358d82018-03-26 18:28:56 +080067import java.util.List;
Robert Greenwaltf60b92b2012-09-13 15:02:00 -070068import java.util.concurrent.atomic.AtomicReference;
Matthew Xieb5f144a2012-06-28 18:41:54 -070069
Robert Greenwalt3901edb2010-01-26 10:22:37 -080070/*
71 * Displays preferences for Tethering.
72 */
Chilund3358d82018-03-26 18:28:56 +080073@SearchIndexable
Sudheer Shanka7dbbe132016-02-16 14:19:32 +000074public class TetherSettings extends RestrictedSettingsFragment
Fan Zhangdefb1182017-07-25 14:18:47 -070075 implements DataSaverBackend.Listener {
Danica Chang32711b62010-08-10 18:41:29 -070076
Chilund3358d82018-03-26 18:28:56 +080077 @VisibleForTesting
78 static final String KEY_TETHER_PREFS_SCREEN = "tether_prefs_screen";
79 @VisibleForTesting
80 static final String KEY_WIFI_TETHER = "wifi_tether";
81 @VisibleForTesting
82 static final String KEY_USB_TETHER_SETTINGS = "usb_tether_settings";
83 @VisibleForTesting
84 static final String KEY_ENABLE_BLUETOOTH_TETHERING = "enable_bluetooth_tethering";
Yuuki Habu557e4572020-02-05 16:24:52 +090085 private static final String KEY_ENABLE_ETHERNET_TETHERING = "enable_ethernet_tethering";
Chilund3358d82018-03-26 18:28:56 +080086 private static final String KEY_DATA_SAVER_FOOTER = "disabled_on_data_saver";
Arc Wang005fe692020-03-04 21:03:54 +080087 @VisibleForTesting
Tsung-Mao Fang00956fb2020-11-16 18:13:36 +080088 static final String KEY_TETHER_PREFS_TOP_INTRO = "tether_prefs_top_intro";
Robert Greenwalt3901edb2010-01-26 10:22:37 -080089
Rebecca Silbersteinffb4fbb2016-03-04 15:47:43 -080090 private static final String TAG = "TetheringSettings";
Hugh Chena3a06612021-07-13 10:42:00 +080091 private static final boolean DEBUG = Log.isLoggable(TAG, Log.DEBUG);
Rebecca Silbersteinffb4fbb2016-03-04 15:47:43 -080092
Alex Johnston7d4db752021-01-15 20:46:05 +000093 private RestrictedSwitchPreference mUsbTether;
Robert Greenwalt3901edb2010-01-26 10:22:37 -080094
PauloftheWest656c88c2014-08-24 11:59:35 -070095 private SwitchPreference mBluetoothTether;
Danica Chang32711b62010-08-10 18:41:29 -070096
Yuuki Habu557e4572020-02-05 16:24:52 +090097 private SwitchPreference mEthernetTether;
98
Robert Greenwalt3901edb2010-01-26 10:22:37 -080099 private BroadcastReceiver mTetherChangeReceiver;
100
Danica Chang32711b62010-08-10 18:41:29 -0700101 private String[] mBluetoothRegexs;
Fan Zhangdefb1182017-07-25 14:18:47 -0700102 private AtomicReference<BluetoothPan> mBluetoothPan = new AtomicReference<>();
Danica Chang32711b62010-08-10 18:41:29 -0700103
Jeremy Kleine3e7b952016-01-25 14:43:49 -0800104 private Handler mHandler = new Handler();
105 private OnStartTetheringCallback mStartTetheringCallback;
Jeremy Kleine3e7b952016-01-25 14:43:49 -0800106 private ConnectivityManager mCm;
Yuuki Habu557e4572020-02-05 16:24:52 +0900107 private EthernetManager mEm;
Yuuki Habu557e4572020-02-05 16:24:52 +0900108 private TetheringEventCallback mTetheringEventCallback;
109 private EthernetListener mEthernetListener;
Xiao Madb7d7de2022-01-30 11:18:29 +0000110 private final HashSet<String> mAvailableInterfaces = new HashSet<>();
Amith Yamasani84a042c2011-03-02 11:25:04 -0800111
Fan Zhang78d5efd2017-06-06 16:55:53 -0700112 private WifiTetherPreferenceController mWifiTetherPreferenceController;
113
Mike Lockwood69a09572011-07-19 13:30:03 -0700114 private boolean mUsbConnected;
115 private boolean mMassStorageActive;
116
Jake Hambyc777ee22011-03-04 15:37:39 -0800117 private boolean mBluetoothEnableForTether;
Julia Reynoldsee27b9d2014-05-09 13:36:20 -0400118 private boolean mUnavailable;
119
Felipe Leme5c091842016-04-21 17:17:28 -0700120 private DataSaverBackend mDataSaverBackend;
121 private boolean mDataSaverEnabled;
122 private Preference mDataSaverFooter;
123
Hugh Chena3a06612021-07-13 10:42:00 +0800124 @VisibleForTesting
125 String[] mUsbRegexs;
126 @VisibleForTesting
127 Context mContext;
128 @VisibleForTesting
129 TetheringManager mTm;
130
Robert Greenwalt3901edb2010-01-26 10:22:37 -0800131 @Override
Fan Zhang65076132016-08-08 10:25:13 -0700132 public int getMetricsCategory() {
Fan Zhang31b21002019-01-16 13:49:47 -0800133 return SettingsEnums.TETHER;
Chris Wren8a963ba2015-03-20 10:29:14 -0400134 }
135
Sudheer Shanka7dbbe132016-02-16 14:19:32 +0000136 public TetherSettings() {
137 super(UserManager.DISALLOW_CONFIG_TETHERING);
138 }
139
Chris Wren8a963ba2015-03-20 10:29:14 -0400140 @Override
Fan Zhang78d5efd2017-06-06 16:55:53 -0700141 public void onAttach(Context context) {
142 super.onAttach(context);
143 mWifiTetherPreferenceController =
tmfang27c84de2018-06-28 11:39:05 +0800144 new WifiTetherPreferenceController(context, getSettingsLifecycle());
Fan Zhang78d5efd2017-06-06 16:55:53 -0700145 }
146
147 @Override
Daisuke Miyakawa9c602c42010-09-10 12:04:37 -0700148 public void onCreate(Bundle icicle) {
Robert Greenwalt3901edb2010-01-26 10:22:37 -0800149 super.onCreate(icicle);
Rohit Sisodia37155e92014-09-22 16:34:56 -0500150
Robert Greenwalt3901edb2010-01-26 10:22:37 -0800151 addPreferencesFromResource(R.xml.tether_prefs);
Hugh Chena3a06612021-07-13 10:42:00 +0800152 mContext = getContext();
153 mDataSaverBackend = new DataSaverBackend(mContext);
Felipe Leme5c091842016-04-21 17:17:28 -0700154 mDataSaverEnabled = mDataSaverBackend.isDataSaverEnabled();
Chilund3358d82018-03-26 18:28:56 +0800155 mDataSaverFooter = findPreference(KEY_DATA_SAVER_FOOTER);
Felipe Leme5c091842016-04-21 17:17:28 -0700156
Sudheer Shanka7dbbe132016-02-16 14:19:32 +0000157 setIfOnlyAvailableForAdmins(true);
158 if (isUiRestricted()) {
Julia Reynoldsee27b9d2014-05-09 13:36:20 -0400159 mUnavailable = true;
Doris Lingdd5693c2017-03-24 13:10:09 -0700160 getPreferenceScreen().removeAll();
Julia Reynoldsee27b9d2014-05-09 13:36:20 -0400161 return;
162 }
163
Daisuke Miyakawa9c602c42010-09-10 12:04:37 -0700164 final Activity activity = getActivity();
Jaikumar Ganesh9ad703c2011-02-23 13:08:09 -0800165 BluetoothAdapter adapter = BluetoothAdapter.getDefaultAdapter();
166 if (adapter != null) {
167 adapter.getProfileProxy(activity.getApplicationContext(), mProfileServiceListener,
168 BluetoothProfile.PAN);
169 }
170
Ted Wang47f2ebb2020-09-01 14:45:46 +0800171 setupTetherPreference();
Tsung-Mao Fang00956fb2020-11-16 18:13:36 +0800172 setTopIntroPreferenceTitle();
Robert Greenwaltc4764d22010-02-12 14:21:37 -0800173
Felipe Leme5c091842016-04-21 17:17:28 -0700174 mDataSaverBackend.addListener(this);
175
Rebecca Silbersteinffb4fbb2016-03-04 15:47:43 -0800176 mCm = (ConnectivityManager) getSystemService(Context.CONNECTIVITY_SERVICE);
Yuuki Habu557e4572020-02-05 16:24:52 +0900177 mTm = (TetheringManager) getSystemService(Context.TETHERING_SERVICE);
Xiao Madb7d7de2022-01-30 11:18:29 +0000178 // Some devices do not have available EthernetManager. In that case getSystemService will
179 // return null.
180 mEm = mContext.getSystemService(EthernetManager.class);
Irfan Sheriff47ebb782010-03-10 08:27:15 -0800181
paulhu6151c222021-02-20 11:21:53 +0800182 mUsbRegexs = mTm.getTetherableUsbRegexs();
183 mBluetoothRegexs = mTm.getTetherableBluetoothRegexs();
Danica Chang32711b62010-08-10 18:41:29 -0700184
Amith Yamasanie419bc12011-02-14 14:19:08 -0800185 final boolean usbAvailable = mUsbRegexs.length != 0;
Weilun Duf8757b72020-03-30 13:27:13 -0700186 final boolean bluetoothAvailable = adapter != null && mBluetoothRegexs.length != 0;
Xiao Madb7d7de2022-01-30 11:18:29 +0000187 final boolean ethernetAvailable = (mEm != null);
Danica Chang32711b62010-08-10 18:41:29 -0700188
Danica Chang32711b62010-08-10 18:41:29 -0700189 if (!usbAvailable || Utils.isMonkeyRunning()) {
190 getPreferenceScreen().removePreference(mUsbTether);
191 }
Amith Yamasanie419bc12011-02-14 14:19:08 -0800192
Fan Zhang78d5efd2017-06-06 16:55:53 -0700193 mWifiTetherPreferenceController.displayPreference(getPreferenceScreen());
Amith Yamasanie419bc12011-02-14 14:19:08 -0800194
Danica Chang32711b62010-08-10 18:41:29 -0700195 if (!bluetoothAvailable) {
196 getPreferenceScreen().removePreference(mBluetoothTether);
Danica Chang32711b62010-08-10 18:41:29 -0700197 } else {
Robert Greenwaltf60b92b2012-09-13 15:02:00 -0700198 BluetoothPan pan = mBluetoothPan.get();
199 if (pan != null && pan.isTetheringOn()) {
Danica Chang32711b62010-08-10 18:41:29 -0700200 mBluetoothTether.setChecked(true);
Danica Chang32711b62010-08-10 18:41:29 -0700201 } else {
202 mBluetoothTether.setChecked(false);
Danica Chang32711b62010-08-10 18:41:29 -0700203 }
204 }
Yuuki Habu557e4572020-02-05 16:24:52 +0900205 if (!ethernetAvailable) getPreferenceScreen().removePreference(mEthernetTether);
Felipe Leme5c091842016-04-21 17:17:28 -0700206 // Set initial state based on Data Saver mode.
207 onDataSaverChanged(mDataSaverBackend.isDataSaverEnabled());
208 }
209
210 @Override
211 public void onDestroy() {
212 mDataSaverBackend.remListener(this);
Robert Greenwalt89b8bab2016-06-13 13:01:36 -0700213
宋凯伦0fb2f2b2016-05-25 10:15:03 +0800214 BluetoothAdapter adapter = BluetoothAdapter.getDefaultAdapter();
215 BluetoothProfile profile = mBluetoothPan.getAndSet(null);
216 if (profile != null && adapter != null) {
217 adapter.closeProfileProxy(BluetoothProfile.PAN, profile);
218 }
Robert Greenwalt89b8bab2016-06-13 13:01:36 -0700219
Felipe Leme5c091842016-04-21 17:17:28 -0700220 super.onDestroy();
221 }
222
Ted Wang47f2ebb2020-09-01 14:45:46 +0800223 @VisibleForTesting
224 void setupTetherPreference() {
Alex Johnston7d4db752021-01-15 20:46:05 +0000225 mUsbTether = (RestrictedSwitchPreference) findPreference(KEY_USB_TETHER_SETTINGS);
Ted Wang47f2ebb2020-09-01 14:45:46 +0800226 mBluetoothTether = (SwitchPreference) findPreference(KEY_ENABLE_BLUETOOTH_TETHERING);
227 mEthernetTether = (SwitchPreference) findPreference(KEY_ENABLE_ETHERNET_TETHERING);
228 }
229
Felipe Leme5c091842016-04-21 17:17:28 -0700230 @Override
231 public void onDataSaverChanged(boolean isDataSaving) {
232 mDataSaverEnabled = isDataSaving;
Felipe Leme5c091842016-04-21 17:17:28 -0700233 mUsbTether.setEnabled(!mDataSaverEnabled);
234 mBluetoothTether.setEnabled(!mDataSaverEnabled);
Yuuki Habu557e4572020-02-05 16:24:52 +0900235 mEthernetTether.setEnabled(!mDataSaverEnabled);
Felipe Leme5c091842016-04-21 17:17:28 -0700236 mDataSaverFooter.setVisible(mDataSaverEnabled);
237 }
238
239 @Override
Bonian Chena5f08852020-07-08 02:47:18 +0800240 public void onAllowlistStatusChanged(int uid, boolean isAllowlisted) {
Felipe Leme5c091842016-04-21 17:17:28 -0700241 }
242
243 @Override
Bonian Chena5f08852020-07-08 02:47:18 +0800244 public void onDenylistStatusChanged(int uid, boolean isDenylisted) {
Rohit Sisodia37155e92014-09-22 16:34:56 -0500245 }
246
Arc Wang005fe692020-03-04 21:03:54 +0800247 @VisibleForTesting
Tsung-Mao Fang00956fb2020-11-16 18:13:36 +0800248 void setTopIntroPreferenceTitle() {
249 final Preference topIntroPreference = findPreference(KEY_TETHER_PREFS_TOP_INTRO);
Hugh Chena3a06612021-07-13 10:42:00 +0800250 final WifiManager wifiManager = mContext.getSystemService(WifiManager.class);
Arc Wang005fe692020-03-04 21:03:54 +0800251 if (wifiManager.isStaApConcurrencySupported()) {
Tsung-Mao Fang00956fb2020-11-16 18:13:36 +0800252 topIntroPreference.setTitle(R.string.tethering_footer_info_sta_ap_concurrency);
Arc Wang005fe692020-03-04 21:03:54 +0800253 } else {
Tsung-Mao Fang00956fb2020-11-16 18:13:36 +0800254 topIntroPreference.setTitle(R.string.tethering_footer_info);
Arc Wang005fe692020-03-04 21:03:54 +0800255 }
256 }
257
Robert Greenwalt3901edb2010-01-26 10:22:37 -0800258 private class TetherChangeReceiver extends BroadcastReceiver {
Danica Chang32711b62010-08-10 18:41:29 -0700259 @Override
Robert Greenwalt3901edb2010-01-26 10:22:37 -0800260 public void onReceive(Context content, Intent intent) {
Jake Hambyc777ee22011-03-04 15:37:39 -0800261 String action = intent.getAction();
Hugh Chena3a06612021-07-13 10:42:00 +0800262 if (DEBUG) {
263 Log.d(TAG, "onReceive() action : " + action);
264 }
265 // TODO(b/194961339): Stop using ACTION_TETHER_STATE_CHANGED and use
266 // mTetheringEventCallback instead.
paulhu6151c222021-02-20 11:21:53 +0800267 if (action.equals(TetheringManager.ACTION_TETHER_STATE_CHANGED)) {
Robert Greenwalta2488762010-03-10 16:57:08 -0800268 // TODO - this should understand the interface types
269 ArrayList<String> available = intent.getStringArrayListExtra(
paulhu6151c222021-02-20 11:21:53 +0800270 TetheringManager.EXTRA_AVAILABLE_TETHER);
Robert Greenwalta2488762010-03-10 16:57:08 -0800271 ArrayList<String> active = intent.getStringArrayListExtra(
paulhu6151c222021-02-20 11:21:53 +0800272 TetheringManager.EXTRA_ACTIVE_TETHER);
Hugh Chena3a06612021-07-13 10:42:00 +0800273 updateBluetoothState();
274 updateEthernetState(available.toArray(new String[available.size()]),
275 active.toArray(new String[active.size()]));
Mike Lockwood69a09572011-07-19 13:30:03 -0700276 } else if (action.equals(Intent.ACTION_MEDIA_SHARED)) {
277 mMassStorageActive = true;
Hugh Chena3a06612021-07-13 10:42:00 +0800278 updateBluetoothAndEthernetState();
279 updateUsbPreference();
Mike Lockwood69a09572011-07-19 13:30:03 -0700280 } else if (action.equals(Intent.ACTION_MEDIA_UNSHARED)) {
281 mMassStorageActive = false;
Hugh Chena3a06612021-07-13 10:42:00 +0800282 updateBluetoothAndEthernetState();
283 updateUsbPreference();
Mike Lockwood69a09572011-07-19 13:30:03 -0700284 } else if (action.equals(UsbManager.ACTION_USB_STATE)) {
285 mUsbConnected = intent.getBooleanExtra(UsbManager.USB_CONNECTED, false);
Hugh Chena3a06612021-07-13 10:42:00 +0800286 updateBluetoothAndEthernetState();
287 updateUsbPreference();
Jake Hambyc777ee22011-03-04 15:37:39 -0800288 } else if (action.equals(BluetoothAdapter.ACTION_STATE_CHANGED)) {
289 if (mBluetoothEnableForTether) {
290 switch (intent
291 .getIntExtra(BluetoothAdapter.EXTRA_STATE, BluetoothAdapter.ERROR)) {
292 case BluetoothAdapter.STATE_ON:
Jeremy Kleine3e7b952016-01-25 14:43:49 -0800293 startTethering(TETHERING_BLUETOOTH);
294 mBluetoothEnableForTether = false;
Jake Hambyc777ee22011-03-04 15:37:39 -0800295 break;
296
297 case BluetoothAdapter.STATE_OFF:
298 case BluetoothAdapter.ERROR:
299 mBluetoothEnableForTether = false;
300 break;
301
302 default:
303 // ignore transition states
304 }
305 }
Hugh Chena3a06612021-07-13 10:42:00 +0800306 updateBluetoothAndEthernetState();
Ted Wang47f2ebb2020-09-01 14:45:46 +0800307 } else if (action.equals(BluetoothPan.ACTION_TETHERING_STATE_CHANGED)) {
Hugh Chena3a06612021-07-13 10:42:00 +0800308 updateBluetoothAndEthernetState();
Robert Greenwalta2488762010-03-10 16:57:08 -0800309 }
Robert Greenwalt3901edb2010-01-26 10:22:37 -0800310 }
311 }
312
313 @Override
Amith Yamasani02cf71a2010-09-21 15:48:52 -0700314 public void onStart() {
315 super.onStart();
Robert Greenwalt3901edb2010-01-26 10:22:37 -0800316
Julia Reynoldsee27b9d2014-05-09 13:36:20 -0400317 if (mUnavailable) {
Sudheer Shanka7dbbe132016-02-16 14:19:32 +0000318 if (!isUiRestrictedByOnlyAdmin()) {
319 getEmptyTextView().setText(R.string.tethering_settings_not_available);
Julia Reynoldsee27b9d2014-05-09 13:36:20 -0400320 }
Sudheer Shanka7dbbe132016-02-16 14:19:32 +0000321 getPreferenceScreen().removeAll();
Julia Reynoldsee27b9d2014-05-09 13:36:20 -0400322 return;
323 }
324
Daisuke Miyakawa9c602c42010-09-10 12:04:37 -0700325
Jeremy Kleine3e7b952016-01-25 14:43:49 -0800326 mStartTetheringCallback = new OnStartTetheringCallback(this);
Yuuki Habu557e4572020-02-05 16:24:52 +0900327 mTetheringEventCallback = new TetheringEventCallback();
Xiao Maf621c4b2022-01-18 06:44:52 +0000328 mTm.registerTetheringEventCallback(r -> mHandler.post(r), mTetheringEventCallback);
Jeremy Kleine3e7b952016-01-25 14:43:49 -0800329
Mike Lockwood69a09572011-07-19 13:30:03 -0700330 mMassStorageActive = Environment.MEDIA_SHARED.equals(Environment.getExternalStorageState());
Ted Wang47f2ebb2020-09-01 14:45:46 +0800331 registerReceiver();
Jake Hambyc777ee22011-03-04 15:37:39 -0800332
Yuuki Habu557e4572020-02-05 16:24:52 +0900333 mEthernetListener = new EthernetListener();
Weilun Duf8757b72020-03-30 13:27:13 -0700334 if (mEm != null)
Xiao Madb7d7de2022-01-30 11:18:29 +0000335 mEm.addInterfaceStateListener(r -> mHandler.post(r), mEthernetListener);
Yuuki Habu557e4572020-02-05 16:24:52 +0900336
Hugh Chena3a06612021-07-13 10:42:00 +0800337 updateUsbState();
338 updateBluetoothAndEthernetState();
Robert Greenwalt3901edb2010-01-26 10:22:37 -0800339 }
340
341 @Override
Amith Yamasani02cf71a2010-09-21 15:48:52 -0700342 public void onStop() {
343 super.onStop();
Julia Reynoldsee27b9d2014-05-09 13:36:20 -0400344
345 if (mUnavailable) {
346 return;
347 }
Daisuke Miyakawa9c602c42010-09-10 12:04:37 -0700348 getActivity().unregisterReceiver(mTetherChangeReceiver);
Yuuki Habu557e4572020-02-05 16:24:52 +0900349 mTm.unregisterTetheringEventCallback(mTetheringEventCallback);
Weilun Duf8757b72020-03-30 13:27:13 -0700350 if (mEm != null)
Xiao Madb7d7de2022-01-30 11:18:29 +0000351 mEm.removeInterfaceStateListener(mEthernetListener);
Robert Greenwalt3901edb2010-01-26 10:22:37 -0800352 mTetherChangeReceiver = null;
Jeremy Kleine3e7b952016-01-25 14:43:49 -0800353 mStartTetheringCallback = null;
Yuuki Habu557e4572020-02-05 16:24:52 +0900354 mTetheringEventCallback = null;
Robert Greenwalt3901edb2010-01-26 10:22:37 -0800355 }
356
Ted Wang47f2ebb2020-09-01 14:45:46 +0800357 @VisibleForTesting
358 void registerReceiver() {
359 final Activity activity = getActivity();
360
361 mTetherChangeReceiver = new TetherChangeReceiver();
paulhu6151c222021-02-20 11:21:53 +0800362 IntentFilter filter = new IntentFilter(TetheringManager.ACTION_TETHER_STATE_CHANGED);
Ted Wang47f2ebb2020-09-01 14:45:46 +0800363 final Intent intent = activity.registerReceiver(mTetherChangeReceiver, filter);
364
365 filter = new IntentFilter();
366 filter.addAction(UsbManager.ACTION_USB_STATE);
367 activity.registerReceiver(mTetherChangeReceiver, filter);
368
369 filter = new IntentFilter();
370 filter.addAction(Intent.ACTION_MEDIA_SHARED);
371 filter.addAction(Intent.ACTION_MEDIA_UNSHARED);
372 filter.addDataScheme("file");
373 activity.registerReceiver(mTetherChangeReceiver, filter);
374
375 filter = new IntentFilter();
376 filter.addAction(BluetoothAdapter.ACTION_STATE_CHANGED);
377 filter.addAction(BluetoothPan.ACTION_TETHERING_STATE_CHANGED);
378 activity.registerReceiver(mTetherChangeReceiver, filter);
379
380 if (intent != null) mTetherChangeReceiver.onReceive(activity, intent);
381 }
382
Hugh Chena3a06612021-07-13 10:42:00 +0800383 // TODO(b/194961339): Separate the updateBluetoothAndEthernetState() to two methods,
384 // updateBluetoothAndEthernetState() and updateBluetoothAndEthernetPreference().
385 // Because we should update the state when only receiving tethering
386 // state changes and update preference when usb or media share changed.
387 private void updateBluetoothAndEthernetState() {
388 String[] tethered = mTm.getTetheredIfaces();
389 updateBluetoothAndEthernetState(tethered);
Robert Greenwalt209177a2010-03-04 13:29:02 -0800390 }
391
Hugh Chena3a06612021-07-13 10:42:00 +0800392 private void updateBluetoothAndEthernetState(String[] tethered) {
393 String[] available = mTm.getTetherableIfaces();
Doris Ling1e914eb2017-05-30 17:28:06 -0700394 updateBluetoothState();
Yuuki Habu557e4572020-02-05 16:24:52 +0900395 updateEthernetState(available, tethered);
Danica Chang32711b62010-08-10 18:41:29 -0700396 }
397
Hugh Chena3a06612021-07-13 10:42:00 +0800398 private void updateUsbState() {
399 String[] tethered = mTm.getTetheredIfaces();
400 updateUsbState(tethered);
401 }
402
Ted Wang47f2ebb2020-09-01 14:45:46 +0800403 @VisibleForTesting
Hugh Chena3a06612021-07-13 10:42:00 +0800404 void updateUsbState(String[] tethered) {
Jake Hamby436b29e2011-02-07 18:21:25 -0800405 boolean usbTethered = false;
Ben Clark0008d212010-07-27 16:20:59 +0100406 for (String s : tethered) {
Robert Greenwaltc4764d22010-02-12 14:21:37 -0800407 for (String regex : mUsbRegexs) {
408 if (s.matches(regex)) usbTethered = true;
409 }
Robert Greenwaltc4764d22010-02-12 14:21:37 -0800410 }
Hugh Chena3a06612021-07-13 10:42:00 +0800411 if (DEBUG) {
412 Log.d(TAG, "updateUsbState() mUsbConnected : " + mUsbConnected
413 + ", mMassStorageActive : " + mMassStorageActive
414 + ", usbTethered : " + usbTethered);
Robert Greenwaltd5f121c2010-03-02 17:33:11 -0800415 }
Robert Greenwaltc4764d22010-02-12 14:21:37 -0800416 if (usbTethered) {
Felipe Leme5c091842016-04-21 17:17:28 -0700417 mUsbTether.setEnabled(!mDataSaverEnabled);
Robert Greenwalt204e7c12010-03-05 19:00:30 -0800418 mUsbTether.setChecked(true);
Hugh Chena3a06612021-07-13 10:42:00 +0800419 mUsbTether.setDisabledByAdmin(
420 checkIfUsbDataSignalingIsDisabled(mContext, UserHandle.myUserId()));
421 } else {
Robert Greenwalt204e7c12010-03-05 19:00:30 -0800422 mUsbTether.setChecked(false);
Hugh Chena3a06612021-07-13 10:42:00 +0800423 updateUsbPreference();
424 }
425 }
426
427 private void updateUsbPreference() {
428 boolean usbAvailable = mUsbConnected && !mMassStorageActive;
Alex Johnston685cacb2021-08-02 14:16:49 +0100429 final RestrictedLockUtils.EnforcedAdmin enforcedAdmin =
430 checkIfUsbDataSignalingIsDisabled(mContext, UserHandle.myUserId());
Hugh Chena3a06612021-07-13 10:42:00 +0800431
Alex Johnston685cacb2021-08-02 14:16:49 +0100432 if (enforcedAdmin != null) {
433 mUsbTether.setDisabledByAdmin(enforcedAdmin);
434 } else if (usbAvailable) {
Hugh Chena3a06612021-07-13 10:42:00 +0800435 mUsbTether.setEnabled(!mDataSaverEnabled);
Robert Greenwalt3901edb2010-01-26 10:22:37 -0800436 } else {
Robert Greenwalt3901edb2010-01-26 10:22:37 -0800437 mUsbTether.setEnabled(false);
438 }
439 }
Robert Greenwalt209177a2010-03-04 13:29:02 -0800440
Ted Wang47f2ebb2020-09-01 14:45:46 +0800441 @VisibleForTesting
442 int getBluetoothState() {
Danica Chang32711b62010-08-10 18:41:29 -0700443 BluetoothAdapter adapter = BluetoothAdapter.getDefaultAdapter();
Rebecca Silbersteinffb4fbb2016-03-04 15:47:43 -0800444 if (adapter == null) {
Ted Wang47f2ebb2020-09-01 14:45:46 +0800445 return BluetoothAdapter.ERROR;
446 }
447 return adapter.getState();
448 }
449
450 @VisibleForTesting
451 boolean isBluetoothTetheringOn() {
452 final BluetoothPan bluetoothPan = mBluetoothPan.get();
453 return bluetoothPan != null && bluetoothPan.isTetheringOn();
454 }
455
456 private void updateBluetoothState() {
457 final int btState = getBluetoothState();
Hugh Chena3a06612021-07-13 10:42:00 +0800458 if (DEBUG) {
459 Log.d(TAG, "updateBluetoothState() btState : " + btState);
460 }
Ted Wang47f2ebb2020-09-01 14:45:46 +0800461 if (btState == BluetoothAdapter.ERROR) {
Hugh Chena3a06612021-07-13 10:42:00 +0800462 Log.w(TAG, "updateBluetoothState() Bluetooth state is error!");
Jianzheng Zhou508c7c12013-08-15 14:34:52 +0800463 return;
Rebecca Silbersteinffb4fbb2016-03-04 15:47:43 -0800464 }
Ted Wang47f2ebb2020-09-01 14:45:46 +0800465
Danica Chang32711b62010-08-10 18:41:29 -0700466 if (btState == BluetoothAdapter.STATE_TURNING_OFF) {
467 mBluetoothTether.setEnabled(false);
Danica Chang32711b62010-08-10 18:41:29 -0700468 } else if (btState == BluetoothAdapter.STATE_TURNING_ON) {
469 mBluetoothTether.setEnabled(false);
Danica Chang32711b62010-08-10 18:41:29 -0700470 } else {
Ted Wang47f2ebb2020-09-01 14:45:46 +0800471 if (btState == BluetoothAdapter.STATE_ON && isBluetoothTetheringOn()) {
Robert Greenwaltf60b92b2012-09-13 15:02:00 -0700472 mBluetoothTether.setChecked(true);
Felipe Leme5c091842016-04-21 17:17:28 -0700473 mBluetoothTether.setEnabled(!mDataSaverEnabled);
Robert Greenwaltf60b92b2012-09-13 15:02:00 -0700474 } else {
Felipe Leme5c091842016-04-21 17:17:28 -0700475 mBluetoothTether.setEnabled(!mDataSaverEnabled);
Robert Greenwaltf60b92b2012-09-13 15:02:00 -0700476 mBluetoothTether.setChecked(false);
Robert Greenwaltf60b92b2012-09-13 15:02:00 -0700477 }
Danica Chang32711b62010-08-10 18:41:29 -0700478 }
479 }
480
Ted Wang47f2ebb2020-09-01 14:45:46 +0800481 @VisibleForTesting
482 void updateEthernetState(String[] available, String[] tethered) {
Yuuki Habu557e4572020-02-05 16:24:52 +0900483 boolean isAvailable = false;
484 boolean isTethered = false;
485
486 for (String s : available) {
Xiao Madb7d7de2022-01-30 11:18:29 +0000487 if (mAvailableInterfaces.contains(s)) isAvailable = true;
Yuuki Habu557e4572020-02-05 16:24:52 +0900488 }
489
490 for (String s : tethered) {
Xiao Madb7d7de2022-01-30 11:18:29 +0000491 if (mAvailableInterfaces.contains(s)) isTethered = true;
Yuuki Habu557e4572020-02-05 16:24:52 +0900492 }
493
Hugh Chena3a06612021-07-13 10:42:00 +0800494 if (DEBUG) {
495 Log.d(TAG, "updateEthernetState() isAvailable : " + isAvailable
496 + ", isTethered : " + isTethered);
497 }
498
Yuuki Habu557e4572020-02-05 16:24:52 +0900499 if (isTethered) {
500 mEthernetTether.setEnabled(!mDataSaverEnabled);
501 mEthernetTether.setChecked(true);
Xiao Madb7d7de2022-01-30 11:18:29 +0000502 } else if (mAvailableInterfaces.size() > 0) {
Yuuki Habu557e4572020-02-05 16:24:52 +0900503 mEthernetTether.setEnabled(!mDataSaverEnabled);
504 mEthernetTether.setChecked(false);
505 } else {
506 mEthernetTether.setEnabled(false);
507 mEthernetTether.setChecked(false);
508 }
509 }
510
Jeremy Kleine3e7b952016-01-25 14:43:49 -0800511 private void startTethering(int choice) {
512 if (choice == TETHERING_BLUETOOTH) {
513 // Turn on Bluetooth first.
514 BluetoothAdapter adapter = BluetoothAdapter.getDefaultAdapter();
515 if (adapter.getState() == BluetoothAdapter.STATE_OFF) {
516 mBluetoothEnableForTether = true;
517 adapter.enable();
Jeremy Kleine3e7b952016-01-25 14:43:49 -0800518 mBluetoothTether.setEnabled(false);
519 return;
Irfan Sheriffaa3d2c42011-10-06 11:45:45 -0700520 }
521 }
Irfan Sheriffaa3d2c42011-10-06 11:45:45 -0700522
Jeremy Kleine3e7b952016-01-25 14:43:49 -0800523 mCm.startTethering(choice, true, mStartTetheringCallback, mHandler);
Irfan Sheriff01b32362011-11-04 12:08:56 -0700524 }
525
526 @Override
Jason Monk39b46742015-09-10 15:52:51 -0400527 public boolean onPreferenceTreeClick(Preference preference) {
Irfan Sheriff01b32362011-11-04 12:08:56 -0700528 if (preference == mUsbTether) {
Jeremy Kleine3e7b952016-01-25 14:43:49 -0800529 if (mUsbTether.isChecked()) {
530 startTethering(TETHERING_USB);
Irfan Sheriff01b32362011-11-04 12:08:56 -0700531 } else {
Jeremy Kleine3e7b952016-01-25 14:43:49 -0800532 mCm.stopTethering(TETHERING_USB);
Irfan Sheriff01b32362011-11-04 12:08:56 -0700533 }
534 } else if (preference == mBluetoothTether) {
Jeremy Kleine3e7b952016-01-25 14:43:49 -0800535 if (mBluetoothTether.isChecked()) {
536 startTethering(TETHERING_BLUETOOTH);
Danica Chang32711b62010-08-10 18:41:29 -0700537 } else {
Jeremy Kleine3e7b952016-01-25 14:43:49 -0800538 mCm.stopTethering(TETHERING_BLUETOOTH);
Danica Chang32711b62010-08-10 18:41:29 -0700539 }
Yuuki Habu557e4572020-02-05 16:24:52 +0900540 } else if (preference == mEthernetTether) {
541 if (mEthernetTether.isChecked()) {
542 startTethering(TETHERING_ETHERNET);
543 } else {
544 mCm.stopTethering(TETHERING_ETHERNET);
545 }
Robert Greenwalt209177a2010-03-04 13:29:02 -0800546 }
Daisuke Miyakawa9c602c42010-09-10 12:04:37 -0700547
Jason Monk39b46742015-09-10 15:52:51 -0400548 return super.onPreferenceTreeClick(preference);
Robert Greenwalt209177a2010-03-04 13:29:02 -0800549 }
550
Amith Yamasanid3fed682012-04-27 17:38:40 -0700551 @Override
552 public int getHelpResource() {
553 return R.string.help_url_tether;
554 }
Jeremy Kleine3e7b952016-01-25 14:43:49 -0800555
556 private BluetoothProfile.ServiceListener mProfileServiceListener =
557 new BluetoothProfile.ServiceListener() {
558 public void onServiceConnected(int profile, BluetoothProfile proxy) {
559 mBluetoothPan.set((BluetoothPan) proxy);
560 }
561 public void onServiceDisconnected(int profile) {
562 mBluetoothPan.set(null);
563 }
564 };
565
Raff Tsaiac3e0d02019-09-19 17:06:45 +0800566 public static final BaseSearchIndexProvider SEARCH_INDEX_DATA_PROVIDER =
Chilund3358d82018-03-26 18:28:56 +0800567 new BaseSearchIndexProvider() {
568 @Override
569 public List<SearchIndexableResource> getXmlResourcesToIndex(
570 Context context, boolean enabled) {
571 final SearchIndexableResource sir = new SearchIndexableResource(context);
572 sir.xmlResId = R.xml.tether_prefs;
573 return Arrays.asList(sir);
574 }
575
576 @Override
Zhen Zhang0ccc8492020-01-17 15:23:40 -0800577 protected boolean isPageSearchEnabled(Context context) {
Zhen Zhangd2a7f9a2020-01-30 15:12:50 -0800578 return !FeatureFlagUtils.isEnabled(context, FeatureFlags.TETHER_ALL_IN_ONE);
Zhen Zhang0ccc8492020-01-17 15:23:40 -0800579 }
580
581 @Override
Chilund3358d82018-03-26 18:28:56 +0800582 public List<String> getNonIndexableKeys(Context context) {
583 final List<String> keys = super.getNonIndexableKeys(context);
paulhu6151c222021-02-20 11:21:53 +0800584 final TetheringManager tm =
585 context.getSystemService(TetheringManager.class);
Chilund3358d82018-03-26 18:28:56 +0800586
587 if (!TetherUtil.isTetherAvailable(context)) {
588 keys.add(KEY_TETHER_PREFS_SCREEN);
589 keys.add(KEY_WIFI_TETHER);
590 }
591
592 final boolean usbAvailable =
paulhu6151c222021-02-20 11:21:53 +0800593 tm.getTetherableUsbRegexs().length != 0;
Chilund3358d82018-03-26 18:28:56 +0800594 if (!usbAvailable || Utils.isMonkeyRunning()) {
595 keys.add(KEY_USB_TETHER_SETTINGS);
596 }
597
598 final boolean bluetoothAvailable =
paulhu6151c222021-02-20 11:21:53 +0800599 tm.getTetherableBluetoothRegexs().length != 0;
Chilund3358d82018-03-26 18:28:56 +0800600 if (!bluetoothAvailable) {
601 keys.add(KEY_ENABLE_BLUETOOTH_TETHERING);
602 }
Yuuki Habu557e4572020-02-05 16:24:52 +0900603
Xiao Madb7d7de2022-01-30 11:18:29 +0000604 final EthernetManager em =
605 context.getSystemService(EthernetManager.class);
606 final boolean ethernetAvailable = (em != null);
Yuuki Habu557e4572020-02-05 16:24:52 +0900607 if (!ethernetAvailable) {
608 keys.add(KEY_ENABLE_ETHERNET_TETHERING);
609 }
Chilund3358d82018-03-26 18:28:56 +0800610 return keys;
611 }
612 };
613
Jeremy Kleine3e7b952016-01-25 14:43:49 -0800614 private static final class OnStartTetheringCallback extends
615 ConnectivityManager.OnStartTetheringCallback {
616 final WeakReference<TetherSettings> mTetherSettings;
617
618 OnStartTetheringCallback(TetherSettings settings) {
Fan Zhangdefb1182017-07-25 14:18:47 -0700619 mTetherSettings = new WeakReference<>(settings);
Jeremy Kleine3e7b952016-01-25 14:43:49 -0800620 }
621
622 @Override
623 public void onTetheringStarted() {
624 update();
625 }
626
627 @Override
628 public void onTetheringFailed() {
629 update();
630 }
631
632 private void update() {
633 TetherSettings settings = mTetherSettings.get();
634 if (settings != null) {
Hugh Chena3a06612021-07-13 10:42:00 +0800635 settings.updateBluetoothAndEthernetState();
Jeremy Kleine3e7b952016-01-25 14:43:49 -0800636 }
637 }
638 }
Yuuki Habu557e4572020-02-05 16:24:52 +0900639
640 private final class TetheringEventCallback implements TetheringManager.TetheringEventCallback {
641 @Override
642 public void onTetheredInterfacesChanged(List<String> interfaces) {
Hugh Chena3a06612021-07-13 10:42:00 +0800643 Log.d(TAG, "onTetheredInterfacesChanged() interfaces : " + interfaces.toString());
644 String[] tethered = interfaces.toArray(new String[interfaces.size()]);
645 updateUsbState(tethered);
646 updateBluetoothAndEthernetState(tethered);
Yuuki Habu557e4572020-02-05 16:24:52 +0900647 }
648 }
649
Xiao Madb7d7de2022-01-30 11:18:29 +0000650 private final class EthernetListener implements EthernetManager.InterfaceStateListener {
Xiao Mabb0a8f22022-03-04 06:09:33 +0000651 public void onInterfaceStateChanged(@NonNull String iface, int state, int role,
652 @NonNull IpConfiguration configuration) {
Xiao Madb7d7de2022-01-30 11:18:29 +0000653 if (state == EthernetManager.STATE_LINK_UP) {
654 mAvailableInterfaces.add(iface);
655 } else {
656 mAvailableInterfaces.remove(iface);
657 }
658 updateBluetoothAndEthernetState();
Yuuki Habu557e4572020-02-05 16:24:52 +0900659 }
660 }
Robert Greenwalt3901edb2010-01-26 10:22:37 -0800661}