WifiServiceImpl.java revision 3871ff67f4a6970f1831fc8951392746c9e2bfa2
1155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande/*
2155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande * Copyright (C) 2010 The Android Open Source Project
3155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande *
4155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande * Licensed under the Apache License, Version 2.0 (the "License");
5155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande * you may not use this file except in compliance with the License.
6155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande * You may obtain a copy of the License at
7155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande *
8155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande *      http://www.apache.org/licenses/LICENSE-2.0
9155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande *
10155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande * Unless required by applicable law or agreed to in writing, software
11155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande * distributed under the License is distributed on an "AS IS" BASIS,
12155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
13155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande * See the License for the specific language governing permissions and
14155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande * limitations under the License.
15155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande */
16155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
17155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpandepackage com.android.server.wifi;
18155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
19eee1d479d8d402a2e78e2f143e957030cfc77749Vinit Deshpandeimport static com.android.server.wifi.WifiController.CMD_AIRPLANE_TOGGLED;
20eee1d479d8d402a2e78e2f143e957030cfc77749Vinit Deshpandeimport static com.android.server.wifi.WifiController.CMD_BATTERY_CHANGED;
21090aa74f2d36010abba6563a39c87ab7ba8a7473Vinit Deshpandeimport static com.android.server.wifi.WifiController.CMD_EMERGENCY_CALL_STATE_CHANGED;
22eee1d479d8d402a2e78e2f143e957030cfc77749Vinit Deshpandeimport static com.android.server.wifi.WifiController.CMD_EMERGENCY_MODE_CHANGED;
23eee1d479d8d402a2e78e2f143e957030cfc77749Vinit Deshpandeimport static com.android.server.wifi.WifiController.CMD_LOCKS_CHANGED;
24eee1d479d8d402a2e78e2f143e957030cfc77749Vinit Deshpandeimport static com.android.server.wifi.WifiController.CMD_SCAN_ALWAYS_MODE_CHANGED;
25eee1d479d8d402a2e78e2f143e957030cfc77749Vinit Deshpandeimport static com.android.server.wifi.WifiController.CMD_SCREEN_OFF;
26eee1d479d8d402a2e78e2f143e957030cfc77749Vinit Deshpandeimport static com.android.server.wifi.WifiController.CMD_SCREEN_ON;
27eee1d479d8d402a2e78e2f143e957030cfc77749Vinit Deshpandeimport static com.android.server.wifi.WifiController.CMD_SET_AP;
28eee1d479d8d402a2e78e2f143e957030cfc77749Vinit Deshpandeimport static com.android.server.wifi.WifiController.CMD_USER_PRESENT;
29eee1d479d8d402a2e78e2f143e957030cfc77749Vinit Deshpandeimport static com.android.server.wifi.WifiController.CMD_WIFI_TOGGLED;
30eee1d479d8d402a2e78e2f143e957030cfc77749Vinit Deshpande
319086afccf6938a49eb9a2cd248917c1cb0943942vandwalleimport android.Manifest;
32155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpandeimport android.app.AppOpsManager;
33155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpandeimport android.bluetooth.BluetoothAdapter;
34155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpandeimport android.content.BroadcastReceiver;
35155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpandeimport android.content.Context;
36155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpandeimport android.content.Intent;
37155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpandeimport android.content.IntentFilter;
38dcf967aa402a4ab1a79c727aea934b8013c1fa6aSvetoslav Ganovimport android.content.pm.ApplicationInfo;
39155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpandeimport android.content.pm.PackageManager;
40f384821f8979152ba83ec70614a59094c838c8c6zhangshuxiaoimport android.content.pm.ParceledListSlice;
41155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpandeimport android.database.ContentObserver;
42155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpandeimport android.net.DhcpInfo;
43155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpandeimport android.net.DhcpResults;
44653cd53f0906a90fbf5b1d9d0bd30917043d1bfcGlen Kuhneimport android.net.IpConfiguration;
459846078598c3468f8813dbfa58238a1846bd81efSanket Padaweimport android.net.Network;
46155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpandeimport android.net.NetworkUtils;
47653cd53f0906a90fbf5b1d9d0bd30917043d1bfcGlen Kuhneimport android.net.StaticIpConfiguration;
48e487a4648dd41881e754f1224aaedead78a0777dSky Faberimport android.net.Uri;
4984d962ec8f487f824214744498bba505a6db0c59Randy Panimport android.net.ip.IpManager;
503ecf5a032e94b6538a56f94a5b33e50cbc464007Jaewan Kimimport android.net.wifi.IWifiManager;
518be2ac8cec40a2000715849b627642bdc87c10b3Shirish Kaleleimport android.net.wifi.ScanResult;
528be2ac8cec40a2000715849b627642bdc87c10b3Shirish Kaleleimport android.net.wifi.ScanSettings;
538be2ac8cec40a2000715849b627642bdc87c10b3Shirish Kaleleimport android.net.wifi.WifiActivityEnergyInfo;
548be2ac8cec40a2000715849b627642bdc87c10b3Shirish Kaleleimport android.net.wifi.WifiConfiguration;
558be2ac8cec40a2000715849b627642bdc87c10b3Shirish Kaleleimport android.net.wifi.WifiConnectionStatistics;
568be2ac8cec40a2000715849b627642bdc87c10b3Shirish Kaleleimport android.net.wifi.WifiEnterpriseConfig;
578be2ac8cec40a2000715849b627642bdc87c10b3Shirish Kaleleimport android.net.wifi.WifiInfo;
588be2ac8cec40a2000715849b627642bdc87c10b3Shirish Kaleleimport android.net.wifi.WifiLinkLayerStats;
598be2ac8cec40a2000715849b627642bdc87c10b3Shirish Kaleleimport android.net.wifi.WifiManager;
606c11cd5d6ff04afdf1bbbc99a850025d46c7bdc9Rebecca Silbersteinimport android.net.wifi.WifiScanner;
613ebf0b17fd1cdd3a0ef22d9e199c1686c01dcc9fPeter Qiuimport android.net.wifi.hotspot2.PasspointConfiguration;
623ecf5a032e94b6538a56f94a5b33e50cbc464007Jaewan Kimimport android.os.AsyncTask;
63c48fd721a9a3273f0faa927d93caad4c9832af7dAdam Lesinskiimport android.os.BatteryStats;
64155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpandeimport android.os.Binder;
65e660aa2b9c9ca3a5c7903c37571ca7b91feb4ccfFyodor Kupolovimport android.os.Build;
66c48fd721a9a3273f0faa927d93caad4c9832af7dAdam Lesinskiimport android.os.Bundle;
67155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpandeimport android.os.HandlerThread;
68155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpandeimport android.os.IBinder;
6940abf54c81c5624641543d86e1d7ab21ebe30175Paul Stewartimport android.os.Looper;
70155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpandeimport android.os.Message;
713ecf5a032e94b6538a56f94a5b33e50cbc464007Jaewan Kimimport android.os.Messenger;
7203b70b37a8a20cb92d860b5015f6762dcb5e75ccDianne Hackbornimport android.os.PowerManager;
73dcf967aa402a4ab1a79c727aea934b8013c1fa6aSvetoslav Ganovimport android.os.Process;
74155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpandeimport android.os.RemoteException;
75c48fd721a9a3273f0faa927d93caad4c9832af7dAdam Lesinskiimport android.os.ResultReceiver;
76186f7513a080eb1a419b6d83286d62961168a5feAdam Lesinskiimport android.os.SystemClock;
77155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpandeimport android.os.UserHandle;
78fd5470391e5342daa38d00b68ccbccfeacbe6d33Alexandra Gherghinaimport android.os.UserManager;
79155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpandeimport android.os.WorkSource;
80155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpandeimport android.provider.Settings;
814d381bc39f5263effdae73ec99065eb299b806caVinit Deshpandeimport android.text.TextUtils;
82992ae00f25a9cc22cf5db3261bd7e72927069cf7vandwalleimport android.util.Log;
83155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpandeimport android.util.Slog;
84155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
85f2fdf411925ad172b5e0b25b0c6df880256691d4Sohani Raoimport com.android.internal.annotations.VisibleForTesting;
8698e43e5f48a4c87343fc311feda4fa5489948822Vinit Deshpandeimport com.android.internal.telephony.IccCardConstants;
87090aa74f2d36010abba6563a39c87ab7ba8a7473Vinit Deshpandeimport com.android.internal.telephony.PhoneConstants;
888be2ac8cec40a2000715849b627642bdc87c10b3Shirish Kaleleimport com.android.internal.telephony.TelephonyIntents;
898be2ac8cec40a2000715849b627642bdc87c10b3Shirish Kaleleimport com.android.internal.util.AsyncChannel;
90f2fdf411925ad172b5e0b25b0c6df880256691d4Sohani Raoimport com.android.server.wifi.util.WifiHandler;
91868b692e6faa9ec3c8dd0cd42d4302082e28b992Sohani Raoimport com.android.server.wifi.util.WifiPermissionsUtil;
928be2ac8cec40a2000715849b627642bdc87c10b3Shirish Kalele
93155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpandeimport java.io.BufferedReader;
94155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpandeimport java.io.FileDescriptor;
958be2ac8cec40a2000715849b627642bdc87c10b3Shirish Kaleleimport java.io.FileNotFoundException;
96155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpandeimport java.io.FileReader;
97155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpandeimport java.io.IOException;
98155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpandeimport java.io.PrintWriter;
99155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpandeimport java.net.Inet4Address;
1008be2ac8cec40a2000715849b627642bdc87c10b3Shirish Kaleleimport java.net.InetAddress;
101a1edc185d46d85e04930a5e12b465de9fea64afeJan Nordqvistimport java.security.GeneralSecurityException;
10231891afce40b903ada9b24ec12e3648ae6aa27b2Jan Nordqvistimport java.security.KeyStore;
10331891afce40b903ada9b24ec12e3648ae6aa27b2Jan Nordqvistimport java.security.cert.CertPath;
10431891afce40b903ada9b24ec12e3648ae6aa27b2Jan Nordqvistimport java.security.cert.CertPathValidator;
1059a069cba2e5ae31ee00d9ec9a3c25bdb7052aa78Jan Nordqvistimport java.security.cert.CertPathValidatorException;
10631891afce40b903ada9b24ec12e3648ae6aa27b2Jan Nordqvistimport java.security.cert.CertificateFactory;
10731891afce40b903ada9b24ec12e3648ae6aa27b2Jan Nordqvistimport java.security.cert.PKIXParameters;
10831891afce40b903ada9b24ec12e3648ae6aa27b2Jan Nordqvistimport java.security.cert.X509Certificate;
109155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpandeimport java.util.ArrayList;
11031891afce40b903ada9b24ec12e3648ae6aa27b2Jan Nordqvistimport java.util.Arrays;
111155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpandeimport java.util.List;
112155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
113155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande/**
114155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande * WifiService handles remote WiFi operation requests by implementing
115155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande * the IWifiManager interface.
116155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande *
117155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande * @hide
118155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande */
11979a4204d12f32d2f6a4dfc8500f5e74718cabb8dVinit Deshpandepublic class WifiServiceImpl extends IWifiManager.Stub {
120155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    private static final String TAG = "WifiService";
121f22d23092ab37286a5ef9d257d5bb32c421d2669vandwalle    private static final boolean DBG = true;
12270b75cda33a05e8b6e6724ed08e4e72f6fee0471Pierre Vandwalle    private static final boolean VDBG = false;
123155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
12405d6ff7612ee795daf3a6fd0ee5a99e01c7a8685Rebecca Silberstein    // Package names for Settings, QuickSettings and QuickQuickSettings
12505d6ff7612ee795daf3a6fd0ee5a99e01c7a8685Rebecca Silberstein    private static final String SYSUI_PACKAGE_NAME = "com.android.systemui";
12605d6ff7612ee795daf3a6fd0ee5a99e01c7a8685Rebecca Silberstein    private static final String SETTINGS_PACKAGE_NAME = "com.android.settings";
12705d6ff7612ee795daf3a6fd0ee5a99e01c7a8685Rebecca Silberstein
12884ee4726383fd968ab43caf120d15d9f47a51a54Paul Stewart    // Dumpsys argument to enable/disable disconnect on IP reachability failures.
12984ee4726383fd968ab43caf120d15d9f47a51a54Paul Stewart    private static final String DUMP_ARG_SET_IPREACH_DISCONNECT = "set-ipreach-disconnect";
13084ee4726383fd968ab43caf120d15d9f47a51a54Paul Stewart    private static final String DUMP_ARG_SET_IPREACH_DISCONNECT_ENABLED = "enabled";
13184ee4726383fd968ab43caf120d15d9f47a51a54Paul Stewart    private static final String DUMP_ARG_SET_IPREACH_DISCONNECT_DISABLED = "disabled";
13284ee4726383fd968ab43caf120d15d9f47a51a54Paul Stewart
133155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    final WifiStateMachine mWifiStateMachine;
134155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
135155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    private final Context mContext;
1360091305175e8c6fe7fc6d01efb9d405961db4ac7Ningyuan Wang    private final FrameworkFacade mFacade;
137155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
13803b70b37a8a20cb92d860b5015f6762dcb5e75ccDianne Hackborn    private final PowerManager mPowerManager;
139155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    private final AppOpsManager mAppOps;
140f9d866f1b4e93593b3411886fd71e729f4a5a205Stuart Scott    private final UserManager mUserManager;
141d02611ce4158fda6c2d14ee13ad7f9553f416d21Ningyuan Wang    private final WifiCountryCode mCountryCode;
142ebb7e0cb4572f73325a749539d589ebdefb8635cvandwalle    // Debug counter tracking scan requests sent by WifiManager
143ebb7e0cb4572f73325a749539d589ebdefb8635cvandwalle    private int scanRequestCounter = 0;
144ebb7e0cb4572f73325a749539d589ebdefb8635cvandwalle
145155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    /* Polls traffic stats and notifies clients */
146155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    private WifiTrafficPoller mTrafficPoller;
147155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    /* Tracks the persisted states for wi-fi & airplane mode */
148155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    final WifiSettingsStore mSettingsStore;
1491b067831bbff14f8e7a99b927b69f714d1b03448Glen Kuhne    /* Logs connection events and some general router and scan stats */
1501b067831bbff14f8e7a99b927b69f714d1b03448Glen Kuhne    private final WifiMetrics mWifiMetrics;
1512bfc67c9893c0a525b224d68dd73a74212b0c29fArthur Hsu    /* Manages affiliated certificates for current user */
1522bfc67c9893c0a525b224d68dd73a74212b0c29fArthur Hsu    private final WifiCertManager mCertManager;
1532bfc67c9893c0a525b224d68dd73a74212b0c29fArthur Hsu
154637a86ffb3a036a4f26a471378b57d8817f35c25Glen Kuhne    private final WifiInjector mWifiInjector;
155ff9686aec5b6ee21e01556899fb13ab55915f3c0Roshan Pius    /* Backup/Restore Module */
156ff9686aec5b6ee21e01556899fb13ab55915f3c0Roshan Pius    private final WifiBackupRestore mWifiBackupRestore;
157ff9686aec5b6ee21e01556899fb13ab55915f3c0Roshan Pius
1586c11cd5d6ff04afdf1bbbc99a850025d46c7bdc9Rebecca Silberstein    private WifiScanner mWifiScanner;
1596c11cd5d6ff04afdf1bbbc99a850025d46c7bdc9Rebecca Silberstein
160f2fdf411925ad172b5e0b25b0c6df880256691d4Sohani Rao    private WifiLog mLog;
161155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    /**
162155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande     * Asynchronous channel to WifiStateMachine
163155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande     */
164155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    private AsyncChannel mWifiStateMachineChannel;
165155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
166dcf967aa402a4ab1a79c727aea934b8013c1fa6aSvetoslav Ganov    private final boolean mPermissionReviewRequired;
167f2fdf411925ad172b5e0b25b0c6df880256691d4Sohani Rao    private final FrameworkFacade mFrameworkFacade;
168dcf967aa402a4ab1a79c727aea934b8013c1fa6aSvetoslav Ganov
169868b692e6faa9ec3c8dd0cd42d4302082e28b992Sohani Rao    private WifiPermissionsUtil mWifiPermissionsUtil;
170868b692e6faa9ec3c8dd0cd42d4302082e28b992Sohani Rao
171155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    /**
172155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande     * Handles client connections
173155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande     */
174f2fdf411925ad172b5e0b25b0c6df880256691d4Sohani Rao    private class ClientHandler extends WifiHandler {
175155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
176f2fdf411925ad172b5e0b25b0c6df880256691d4Sohani Rao        ClientHandler(String tag, Looper looper) {
177f2fdf411925ad172b5e0b25b0c6df880256691d4Sohani Rao            super(tag, looper);
178155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        }
179155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
180155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        @Override
181155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        public void handleMessage(Message msg) {
182f2fdf411925ad172b5e0b25b0c6df880256691d4Sohani Rao            super.handleMessage(msg);
183155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande            switch (msg.what) {
184155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande                case AsyncChannel.CMD_CHANNEL_HALF_CONNECTED: {
185155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande                    if (msg.arg1 == AsyncChannel.STATUS_SUCCESSFUL) {
186155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande                        if (DBG) Slog.d(TAG, "New client listening to asynchronous messages");
187155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande                        // We track the clients by the Messenger
188155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande                        // since it is expected to be always available
189155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande                        mTrafficPoller.addClient(msg.replyTo);
190155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande                    } else {
191155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande                        Slog.e(TAG, "Client connection failure, error=" + msg.arg1);
192155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande                    }
193155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande                    break;
194155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande                }
195155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande                case AsyncChannel.CMD_CHANNEL_DISCONNECTED: {
196155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande                    if (msg.arg1 == AsyncChannel.STATUS_SEND_UNSUCCESSFUL) {
197155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande                        if (DBG) Slog.d(TAG, "Send failed, client connection lost");
198155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande                    } else {
199155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande                        if (DBG) Slog.d(TAG, "Client connection lost with reason: " + msg.arg1);
200155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande                    }
201155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande                    mTrafficPoller.removeClient(msg.replyTo);
202155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande                    break;
203155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande                }
204155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande                case AsyncChannel.CMD_CHANNEL_FULL_CONNECTION: {
205f2fdf411925ad172b5e0b25b0c6df880256691d4Sohani Rao                    AsyncChannel ac = mFrameworkFacade.makeWifiAsyncChannel(TAG);
206155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande                    ac.connect(mContext, this, msg.replyTo);
207155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande                    break;
208155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande                }
2091316bfba023da3371bbc590ab2edee19c9475c03Ningyuan Wang                case WifiManager.CONNECT_NETWORK: {
210155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande                    WifiConfiguration config = (WifiConfiguration) msg.obj;
211155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande                    int networkId = msg.arg1;
2121316bfba023da3371bbc590ab2edee19c9475c03Ningyuan Wang                    Slog.d("WiFiServiceImpl ", "CONNECT "
2131316bfba023da3371bbc590ab2edee19c9475c03Ningyuan Wang                            + " nid=" + Integer.toString(networkId)
2141316bfba023da3371bbc590ab2edee19c9475c03Ningyuan Wang                            + " uid=" + msg.sendingUid
2151316bfba023da3371bbc590ab2edee19c9475c03Ningyuan Wang                            + " name="
2161316bfba023da3371bbc590ab2edee19c9475c03Ningyuan Wang                            + mContext.getPackageManager().getNameForUid(msg.sendingUid));
217d03a9283285bb7adef6c687eb3a91fa4a8c4b502Vinit Deshpande                    if (config != null && isValid(config)) {
2181316bfba023da3371bbc590ab2edee19c9475c03Ningyuan Wang                        if (DBG) Slog.d(TAG, "Connect with config " + config);
2191316bfba023da3371bbc590ab2edee19c9475c03Ningyuan Wang                        /* Command is forwarded to state machine */
22064c98f4afda6c7970c5d563580165a308d5e7c15Jason Monk                        mWifiStateMachine.sendMessage(Message.obtain(msg));
221155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande                    } else if (config == null
222155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande                            && networkId != WifiConfiguration.INVALID_NETWORK_ID) {
2231316bfba023da3371bbc590ab2edee19c9475c03Ningyuan Wang                        if (DBG) Slog.d(TAG, "Connect with networkId " + networkId);
2241316bfba023da3371bbc590ab2edee19c9475c03Ningyuan Wang                        mWifiStateMachine.sendMessage(Message.obtain(msg));
2251316bfba023da3371bbc590ab2edee19c9475c03Ningyuan Wang                    } else {
2261316bfba023da3371bbc590ab2edee19c9475c03Ningyuan Wang                        Slog.e(TAG, "ClientHandler.handleMessage ignoring invalid msg=" + msg);
2271316bfba023da3371bbc590ab2edee19c9475c03Ningyuan Wang                        replyFailed(msg, WifiManager.CONNECT_NETWORK_FAILED,
2281316bfba023da3371bbc590ab2edee19c9475c03Ningyuan Wang                                WifiManager.INVALID_ARGS);
2291316bfba023da3371bbc590ab2edee19c9475c03Ningyuan Wang                    }
2301316bfba023da3371bbc590ab2edee19c9475c03Ningyuan Wang                    break;
2311316bfba023da3371bbc590ab2edee19c9475c03Ningyuan Wang                }
2321316bfba023da3371bbc590ab2edee19c9475c03Ningyuan Wang                case WifiManager.SAVE_NETWORK: {
2331316bfba023da3371bbc590ab2edee19c9475c03Ningyuan Wang                    WifiConfiguration config = (WifiConfiguration) msg.obj;
2341316bfba023da3371bbc590ab2edee19c9475c03Ningyuan Wang                    int networkId = msg.arg1;
2351316bfba023da3371bbc590ab2edee19c9475c03Ningyuan Wang                    Slog.d("WiFiServiceImpl ", "SAVE"
2361316bfba023da3371bbc590ab2edee19c9475c03Ningyuan Wang                            + " nid=" + Integer.toString(networkId)
2371316bfba023da3371bbc590ab2edee19c9475c03Ningyuan Wang                            + " uid=" + msg.sendingUid
2381316bfba023da3371bbc590ab2edee19c9475c03Ningyuan Wang                            + " name="
2391316bfba023da3371bbc590ab2edee19c9475c03Ningyuan Wang                            + mContext.getPackageManager().getNameForUid(msg.sendingUid));
2401316bfba023da3371bbc590ab2edee19c9475c03Ningyuan Wang                    if (config != null && isValid(config)) {
2411316bfba023da3371bbc590ab2edee19c9475c03Ningyuan Wang                        if (DBG) Slog.d(TAG, "Save network with config " + config);
2421316bfba023da3371bbc590ab2edee19c9475c03Ningyuan Wang                        /* Command is forwarded to state machine */
243155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande                        mWifiStateMachine.sendMessage(Message.obtain(msg));
244155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande                    } else {
245155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande                        Slog.e(TAG, "ClientHandler.handleMessage ignoring invalid msg=" + msg);
2461316bfba023da3371bbc590ab2edee19c9475c03Ningyuan Wang                        replyFailed(msg, WifiManager.SAVE_NETWORK_FAILED,
2471316bfba023da3371bbc590ab2edee19c9475c03Ningyuan Wang                                WifiManager.INVALID_ARGS);
248155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande                    }
249155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande                    break;
250155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande                }
251155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande                case WifiManager.FORGET_NETWORK:
252bace539f9cdc4ffb521c8251dd1c56073e805cd4Bartosz Fabianowski                    mWifiStateMachine.sendMessage(Message.obtain(msg));
253002ea67f42b0f69a8c9fa4719391f9476567b762Sky Faber                    break;
254155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande                case WifiManager.START_WPS:
255155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande                case WifiManager.CANCEL_WPS:
256155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande                case WifiManager.DISABLE_NETWORK:
257155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande                case WifiManager.RSSI_PKTCNT_FETCH: {
258155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande                    mWifiStateMachine.sendMessage(Message.obtain(msg));
259155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande                    break;
260155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande                }
261155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande                default: {
262155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande                    Slog.d(TAG, "ClientHandler.handleMessage ignoring msg=" + msg);
263155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande                    break;
264155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande                }
265155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande            }
266155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        }
267155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
268c35d728a15e9270f5642ef79f5245c89d749285fSky Faber        private void replyFailed(Message msg, int what, int why) {
269e1d14389cd77b0757ba3533b122cf00ad65e2099Paul Stewart            if (msg.replyTo == null) return;
2708fcd1fe86ddc3056e1a26e69bc68f148ae0e9ddcChristopher Wiley            Message reply = Message.obtain();
271155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande            reply.what = what;
272c35d728a15e9270f5642ef79f5245c89d749285fSky Faber            reply.arg1 = why;
273155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande            try {
274155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande                msg.replyTo.send(reply);
275155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande            } catch (RemoteException e) {
276155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande                // There's not much we can do if reply can't be sent!
277155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande            }
278155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        }
279155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    }
280155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    private ClientHandler mClientHandler;
281155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
282155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    /**
283155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande     * Handles interaction with WifiStateMachine
284155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande     */
285f2fdf411925ad172b5e0b25b0c6df880256691d4Sohani Rao    private class WifiStateMachineHandler extends WifiHandler {
286155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        private AsyncChannel mWsmChannel;
287155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
288f2fdf411925ad172b5e0b25b0c6df880256691d4Sohani Rao        WifiStateMachineHandler(String tag, Looper looper, AsyncChannel asyncChannel) {
289f2fdf411925ad172b5e0b25b0c6df880256691d4Sohani Rao            super(tag, looper);
290da918df16e03ee19be62343313d954027d3eb3abRebecca Silberstein            mWsmChannel = asyncChannel;
291155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande            mWsmChannel.connect(mContext, this, mWifiStateMachine.getHandler());
292155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        }
293155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
294155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        @Override
295155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        public void handleMessage(Message msg) {
296f2fdf411925ad172b5e0b25b0c6df880256691d4Sohani Rao            super.handleMessage(msg);
297155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande            switch (msg.what) {
298155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande                case AsyncChannel.CMD_CHANNEL_HALF_CONNECTED: {
299155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande                    if (msg.arg1 == AsyncChannel.STATUS_SUCCESSFUL) {
300155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande                        mWifiStateMachineChannel = mWsmChannel;
301155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande                    } else {
302155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande                        Slog.e(TAG, "WifiStateMachine connection failure, error=" + msg.arg1);
303155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande                        mWifiStateMachineChannel = null;
304155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande                    }
305155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande                    break;
306155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande                }
307155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande                case AsyncChannel.CMD_CHANNEL_DISCONNECTED: {
308155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande                    Slog.e(TAG, "WifiStateMachine channel lost, msg.arg1 =" + msg.arg1);
309155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande                    mWifiStateMachineChannel = null;
310155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande                    //Re-establish connection to state machine
311155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande                    mWsmChannel.connect(mContext, this, mWifiStateMachine.getHandler());
312155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande                    break;
313155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande                }
314155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande                default: {
315155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande                    Slog.d(TAG, "WifiStateMachineHandler.handleMessage ignoring msg=" + msg);
316155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande                    break;
317155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande                }
318155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande            }
319155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        }
320155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    }
321155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
322155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    WifiStateMachineHandler mWifiStateMachineHandler;
323155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    private WifiController mWifiController;
3242a7c2bec5d307a9e52de429c6303b7df9de900dcRebecca Silberstein    private final WifiLockManager mWifiLockManager;
32561312e14a088a9487d4db64f08285162476e870fPaul Stewart    private final WifiMulticastLockManager mWifiMulticastLockManager;
326155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
327da918df16e03ee19be62343313d954027d3eb3abRebecca Silberstein    public WifiServiceImpl(Context context, WifiInjector wifiInjector, AsyncChannel asyncChannel) {
328155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        mContext = context;
329da918df16e03ee19be62343313d954027d3eb3abRebecca Silberstein        mWifiInjector = wifiInjector;
330fca64fc16ec43befde2e7ac7c3bfd84ced1f7778Rebecca Silberstein
331fca64fc16ec43befde2e7ac7c3bfd84ced1f7778Rebecca Silberstein        mFacade = mWifiInjector.getFrameworkFacade();
332637a86ffb3a036a4f26a471378b57d8817f35c25Glen Kuhne        mWifiMetrics = mWifiInjector.getWifiMetrics();
333fca64fc16ec43befde2e7ac7c3bfd84ced1f7778Rebecca Silberstein        mTrafficPoller = mWifiInjector.getWifiTrafficPoller();
3343871ff67f4a6970f1831fc8951392746c9e2bfa2Rebecca Silberstein        mUserManager = mWifiInjector.getUserManager();
335fca64fc16ec43befde2e7ac7c3bfd84ced1f7778Rebecca Silberstein        mCountryCode = mWifiInjector.getWifiCountryCode();
336fca64fc16ec43befde2e7ac7c3bfd84ced1f7778Rebecca Silberstein        mWifiStateMachine = mWifiInjector.getWifiStateMachine();
337155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        mWifiStateMachine.enableRssiPolling(true);
338fca64fc16ec43befde2e7ac7c3bfd84ced1f7778Rebecca Silberstein        mSettingsStore = mWifiInjector.getWifiSettingsStore();
339fca64fc16ec43befde2e7ac7c3bfd84ced1f7778Rebecca Silberstein        mPowerManager = mContext.getSystemService(PowerManager.class);
340fca64fc16ec43befde2e7ac7c3bfd84ced1f7778Rebecca Silberstein        mAppOps = (AppOpsManager) mContext.getSystemService(Context.APP_OPS_SERVICE);
341fca64fc16ec43befde2e7ac7c3bfd84ced1f7778Rebecca Silberstein        mCertManager = mWifiInjector.getWifiCertManager();
342fca64fc16ec43befde2e7ac7c3bfd84ced1f7778Rebecca Silberstein        mWifiLockManager = mWifiInjector.getWifiLockManager();
34361312e14a088a9487d4db64f08285162476e870fPaul Stewart        mWifiMulticastLockManager = mWifiInjector.getWifiMulticastLockManager();
344fca64fc16ec43befde2e7ac7c3bfd84ced1f7778Rebecca Silberstein        HandlerThread wifiServiceHandlerThread = mWifiInjector.getWifiServiceHandlerThread();
345f2fdf411925ad172b5e0b25b0c6df880256691d4Sohani Rao        mClientHandler = new ClientHandler(TAG, wifiServiceHandlerThread.getLooper());
346f2fdf411925ad172b5e0b25b0c6df880256691d4Sohani Rao        mWifiStateMachineHandler = new WifiStateMachineHandler(TAG,
347f2fdf411925ad172b5e0b25b0c6df880256691d4Sohani Rao                wifiServiceHandlerThread.getLooper(), asyncChannel);
348fca64fc16ec43befde2e7ac7c3bfd84ced1f7778Rebecca Silberstein        mWifiController = mWifiInjector.getWifiController();
3493204fb9682242a7b5a749489076c66d448c42577Roshan Pius        mWifiBackupRestore = mWifiInjector.getWifiBackupRestore();
350dcf967aa402a4ab1a79c727aea934b8013c1fa6aSvetoslav Ganov        mPermissionReviewRequired = Build.PERMISSIONS_REVIEW_REQUIRED
351dcf967aa402a4ab1a79c727aea934b8013c1fa6aSvetoslav Ganov                || context.getResources().getBoolean(
352dcf967aa402a4ab1a79c727aea934b8013c1fa6aSvetoslav Ganov                com.android.internal.R.bool.config_permissionReviewRequired);
353868b692e6faa9ec3c8dd0cd42d4302082e28b992Sohani Rao        mWifiPermissionsUtil = mWifiInjector.getWifiPermissionsUtil();
354f2fdf411925ad172b5e0b25b0c6df880256691d4Sohani Rao        mLog = mWifiInjector.makeLog(TAG);
355f2fdf411925ad172b5e0b25b0c6df880256691d4Sohani Rao        mFrameworkFacade = wifiInjector.getFrameworkFacade();
35600ec8bf178ba4abee4ee8bbc6eb09c9fcd986a89Rebecca Silberstein        enableVerboseLoggingInternal(getVerboseLoggingLevel());
3571c28bb29cb19367d5e3bdbe6d8558d0fd4b8f33eVinit Deshapnde    }
3581c28bb29cb19367d5e3bdbe6d8558d0fd4b8f33eVinit Deshapnde
3591c28bb29cb19367d5e3bdbe6d8558d0fd4b8f33eVinit Deshapnde    /**
360f2fdf411925ad172b5e0b25b0c6df880256691d4Sohani Rao     * Provide a way for unit tests to set valid log object in the WifiHandler
361f2fdf411925ad172b5e0b25b0c6df880256691d4Sohani Rao     * @param log WifiLog object to assign to the clientHandler
362f2fdf411925ad172b5e0b25b0c6df880256691d4Sohani Rao     */
363f2fdf411925ad172b5e0b25b0c6df880256691d4Sohani Rao    @VisibleForTesting
364f2fdf411925ad172b5e0b25b0c6df880256691d4Sohani Rao    public void setWifiHandlerLogForTest(WifiLog log) {
365f2fdf411925ad172b5e0b25b0c6df880256691d4Sohani Rao        mClientHandler.setWifiLog(log);
366f2fdf411925ad172b5e0b25b0c6df880256691d4Sohani Rao    }
36736f52ffb5fd16c89f022dead27d8df9e51874160Rebecca Silberstein
368f2fdf411925ad172b5e0b25b0c6df880256691d4Sohani Rao    /**
36936f52ffb5fd16c89f022dead27d8df9e51874160Rebecca Silberstein     * Check if we are ready to start wifi.
37036f52ffb5fd16c89f022dead27d8df9e51874160Rebecca Silberstein     *
37136f52ffb5fd16c89f022dead27d8df9e51874160Rebecca Silberstein     * First check if we will be restarting system services to decrypt the device. If the device is
37236f52ffb5fd16c89f022dead27d8df9e51874160Rebecca Silberstein     * not encrypted, check if Wi-Fi needs to be enabled and start if needed
3731c28bb29cb19367d5e3bdbe6d8558d0fd4b8f33eVinit Deshapnde     *
37436f52ffb5fd16c89f022dead27d8df9e51874160Rebecca Silberstein     * This function is used only at boot time.
3751c28bb29cb19367d5e3bdbe6d8558d0fd4b8f33eVinit Deshapnde     */
3761c28bb29cb19367d5e3bdbe6d8558d0fd4b8f33eVinit Deshapnde    public void checkAndStartWifi() {
37736f52ffb5fd16c89f022dead27d8df9e51874160Rebecca Silberstein        // First check if we will end up restarting WifiService
37836f52ffb5fd16c89f022dead27d8df9e51874160Rebecca Silberstein        if (mFrameworkFacade.inStorageManagerCryptKeeperBounce()) {
37936f52ffb5fd16c89f022dead27d8df9e51874160Rebecca Silberstein            Log.d(TAG, "Device still encrypted. Need to restart SystemServer.  Do not start wifi.");
38036f52ffb5fd16c89f022dead27d8df9e51874160Rebecca Silberstein            return;
38136f52ffb5fd16c89f022dead27d8df9e51874160Rebecca Silberstein        }
38236f52ffb5fd16c89f022dead27d8df9e51874160Rebecca Silberstein
38336f52ffb5fd16c89f022dead27d8df9e51874160Rebecca Silberstein        // Check if wi-fi needs to be enabled
3841c28bb29cb19367d5e3bdbe6d8558d0fd4b8f33eVinit Deshapnde        boolean wifiEnabled = mSettingsStore.isWifiToggleEnabled();
3851c28bb29cb19367d5e3bdbe6d8558d0fd4b8f33eVinit Deshapnde        Slog.i(TAG, "WifiService starting up with Wi-Fi " +
3861c28bb29cb19367d5e3bdbe6d8558d0fd4b8f33eVinit Deshapnde                (wifiEnabled ? "enabled" : "disabled"));
387155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
388155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        registerForScanModeChange();
389155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        mContext.registerReceiver(
390155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande                new BroadcastReceiver() {
391155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande                    @Override
392155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande                    public void onReceive(Context context, Intent intent) {
393155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande                        if (mSettingsStore.handleAirplaneModeToggled()) {
394155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande                            mWifiController.sendMessage(CMD_AIRPLANE_TOGGLED);
395155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande                        }
39698e43e5f48a4c87343fc311feda4fa5489948822Vinit Deshpande                        if (mSettingsStore.isAirplaneModeOn()) {
39798e43e5f48a4c87343fc311feda4fa5489948822Vinit Deshpande                            Log.d(TAG, "resetting country code because Airplane mode is ON");
398d02611ce4158fda6c2d14ee13ad7f9553f416d21Ningyuan Wang                            mCountryCode.airplaneModeEnabled();
39998e43e5f48a4c87343fc311feda4fa5489948822Vinit Deshpande                        }
400155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande                    }
401155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande                },
402155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande                new IntentFilter(Intent.ACTION_AIRPLANE_MODE_CHANGED));
403155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
40498e43e5f48a4c87343fc311feda4fa5489948822Vinit Deshpande        mContext.registerReceiver(
40598e43e5f48a4c87343fc311feda4fa5489948822Vinit Deshpande                new BroadcastReceiver() {
40698e43e5f48a4c87343fc311feda4fa5489948822Vinit Deshpande                    @Override
40798e43e5f48a4c87343fc311feda4fa5489948822Vinit Deshpande                    public void onReceive(Context context, Intent intent) {
40898e43e5f48a4c87343fc311feda4fa5489948822Vinit Deshpande                        String state = intent.getStringExtra(IccCardConstants.INTENT_KEY_ICC_STATE);
409a7bf9d90d68a4833cc3c64da98024167931c4b1epkanwar                        if (IccCardConstants.INTENT_VALUE_ICC_ABSENT.equals(state)) {
410fdce524fd05844c996cf1c5c0c102a87fde8e32cVinit Deshpande                            Log.d(TAG, "resetting networks because SIM was removed");
4113c8094ab45f3320dbe45e6460c5d62dcc24ce7aeMitchell Wills                            mWifiStateMachine.resetSimAuthNetworks(false);
41298e43e5f48a4c87343fc311feda4fa5489948822Vinit Deshpande                            Log.d(TAG, "resetting country code because SIM is removed");
413d02611ce4158fda6c2d14ee13ad7f9553f416d21Ningyuan Wang                            mCountryCode.simCardRemoved();
4143c8094ab45f3320dbe45e6460c5d62dcc24ce7aeMitchell Wills                        } else if (IccCardConstants.INTENT_VALUE_ICC_LOADED.equals(state)) {
4153c8094ab45f3320dbe45e6460c5d62dcc24ce7aeMitchell Wills                            Log.d(TAG, "resetting networks because SIM was loaded");
4163c8094ab45f3320dbe45e6460c5d62dcc24ce7aeMitchell Wills                            mWifiStateMachine.resetSimAuthNetworks(true);
41798e43e5f48a4c87343fc311feda4fa5489948822Vinit Deshpande                        }
41898e43e5f48a4c87343fc311feda4fa5489948822Vinit Deshpande                    }
41998e43e5f48a4c87343fc311feda4fa5489948822Vinit Deshpande                },
42098e43e5f48a4c87343fc311feda4fa5489948822Vinit Deshpande                new IntentFilter(TelephonyIntents.ACTION_SIM_STATE_CHANGED));
42198e43e5f48a4c87343fc311feda4fa5489948822Vinit Deshpande
422155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        // Adding optimizations of only receiving broadcasts when wifi is enabled
423155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        // can result in race conditions when apps toggle wifi in the background
424155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        // without active user involvement. Always receive broadcasts.
425155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        registerForBroadcasts();
426faa2ce7f93eef2c436796fb1ea3574e2d1412491Robin Lee        registerForPackageOrUserRemoval();
42703b70b37a8a20cb92d860b5015f6762dcb5e75ccDianne Hackborn        mInIdleMode = mPowerManager.isDeviceIdleMode();
428155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
429d84fd37259c6e956d0f00c261f573dfa319acb91Roshan Pius        if (!mWifiStateMachine.syncInitialize(mWifiStateMachineChannel)) {
430d84fd37259c6e956d0f00c261f573dfa319acb91Roshan Pius            Log.wtf(TAG, "Failed to initialize WifiStateMachine");
431d84fd37259c6e956d0f00c261f573dfa319acb91Roshan Pius        }
4321c28bb29cb19367d5e3bdbe6d8558d0fd4b8f33eVinit Deshapnde        mWifiController.start();
433155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
434155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        // If we are already disabled (could be due to airplane mode), avoid changing persist
435155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        // state here
436dcf967aa402a4ab1a79c727aea934b8013c1fa6aSvetoslav Ganov        if (wifiEnabled) {
437dcf967aa402a4ab1a79c727aea934b8013c1fa6aSvetoslav Ganov            try {
438dcf967aa402a4ab1a79c727aea934b8013c1fa6aSvetoslav Ganov                setWifiEnabled(mContext.getPackageName(), wifiEnabled);
439dcf967aa402a4ab1a79c727aea934b8013c1fa6aSvetoslav Ganov            } catch (RemoteException e) {
440dcf967aa402a4ab1a79c727aea934b8013c1fa6aSvetoslav Ganov                /* ignore - local call */
441dcf967aa402a4ab1a79c727aea934b8013c1fa6aSvetoslav Ganov            }
442dcf967aa402a4ab1a79c727aea934b8013c1fa6aSvetoslav Ganov        }
443155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    }
444155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
445bcdabb1fa1894fcca610692ec94459fe623afa74Bartosz Fabianowski    public void handleUserSwitch(int userId) {
446bcdabb1fa1894fcca610692ec94459fe623afa74Bartosz Fabianowski        mWifiStateMachine.handleUserSwitch(userId);
447bcdabb1fa1894fcca610692ec94459fe623afa74Bartosz Fabianowski    }
448bcdabb1fa1894fcca610692ec94459fe623afa74Bartosz Fabianowski
4493bc487aa49deecbc358ee819e0dd4b2534412281Roshan Pius    public void handleUserUnlock(int userId) {
4503bc487aa49deecbc358ee819e0dd4b2534412281Roshan Pius        mWifiStateMachine.handleUserUnlock(userId);
4513bc487aa49deecbc358ee819e0dd4b2534412281Roshan Pius    }
4523bc487aa49deecbc358ee819e0dd4b2534412281Roshan Pius
4533bc487aa49deecbc358ee819e0dd4b2534412281Roshan Pius    public void handleUserStop(int userId) {
4543bc487aa49deecbc358ee819e0dd4b2534412281Roshan Pius        mWifiStateMachine.handleUserStop(userId);
4553bc487aa49deecbc358ee819e0dd4b2534412281Roshan Pius    }
456bcdabb1fa1894fcca610692ec94459fe623afa74Bartosz Fabianowski
457155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    /**
458a0009d14075b5345b8f916c3fb3f2260c938cb9dYuhao Zheng     * see {@link android.net.wifi.WifiManager#startScan}
459a0009d14075b5345b8f916c3fb3f2260c938cb9dYuhao Zheng     * and {@link android.net.wifi.WifiManager#startCustomizedScan}
460155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande     *
461a0009d14075b5345b8f916c3fb3f2260c938cb9dYuhao Zheng     * @param settings If null, use default parameter, i.e. full scan.
462a0009d14075b5345b8f916c3fb3f2260c938cb9dYuhao Zheng     * @param workSource If null, all blame is given to the calling uid.
463155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande     */
4648fcd1fe86ddc3056e1a26e69bc68f148ae0e9ddcChristopher Wiley    @Override
465a0009d14075b5345b8f916c3fb3f2260c938cb9dYuhao Zheng    public void startScan(ScanSettings settings, WorkSource workSource) {
466155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        enforceChangePermission();
467f2fdf411925ad172b5e0b25b0c6df880256691d4Sohani Rao        mLog.trace("startScan uid=%").c(Binder.getCallingUid()).flush();
4681227b49a94f33844ad0606b48b591bea4d27b08eVinit Deshpande        synchronized (this) {
4696c11cd5d6ff04afdf1bbbc99a850025d46c7bdc9Rebecca Silberstein            if (mWifiScanner == null) {
4706c11cd5d6ff04afdf1bbbc99a850025d46c7bdc9Rebecca Silberstein                mWifiScanner = mWifiInjector.getWifiScanner();
4716c11cd5d6ff04afdf1bbbc99a850025d46c7bdc9Rebecca Silberstein            }
47203b70b37a8a20cb92d860b5015f6762dcb5e75ccDianne Hackborn            if (mInIdleMode) {
47349dfc6768889a33eb67102b16faf21667af2a60dVinit Deshpande                // Need to send an immediate scan result broadcast in case the
47449dfc6768889a33eb67102b16faf21667af2a60dVinit Deshpande                // caller is waiting for a result ..
475691fb51e7ac5e4d0b91a4719cd0ed10a8c212e50Vinit Deshpande
476691fb51e7ac5e4d0b91a4719cd0ed10a8c212e50Vinit Deshpande                // clear calling identity to send broadcast
477691fb51e7ac5e4d0b91a4719cd0ed10a8c212e50Vinit Deshpande                long callingIdentity = Binder.clearCallingIdentity();
478691fb51e7ac5e4d0b91a4719cd0ed10a8c212e50Vinit Deshpande                try {
4796c11cd5d6ff04afdf1bbbc99a850025d46c7bdc9Rebecca Silberstein                    // TODO: investigate if the logic to cancel scans when idle can move to
4806c11cd5d6ff04afdf1bbbc99a850025d46c7bdc9Rebecca Silberstein                    // WifiScanningServiceImpl.  This will 1 - clean up WifiServiceImpl and 2 -
4816c11cd5d6ff04afdf1bbbc99a850025d46c7bdc9Rebecca Silberstein                    // avoid plumbing an awkward path to report a cancelled/failed scan.  This will
4826c11cd5d6ff04afdf1bbbc99a850025d46c7bdc9Rebecca Silberstein                    // be sent directly until b/31398592 is fixed.
4836c11cd5d6ff04afdf1bbbc99a850025d46c7bdc9Rebecca Silberstein                    Intent intent = new Intent(WifiManager.SCAN_RESULTS_AVAILABLE_ACTION);
4846c11cd5d6ff04afdf1bbbc99a850025d46c7bdc9Rebecca Silberstein                    intent.addFlags(Intent.FLAG_RECEIVER_REGISTERED_ONLY_BEFORE_BOOT);
4856c11cd5d6ff04afdf1bbbc99a850025d46c7bdc9Rebecca Silberstein                    intent.putExtra(WifiManager.EXTRA_RESULTS_UPDATED, false);
4866c11cd5d6ff04afdf1bbbc99a850025d46c7bdc9Rebecca Silberstein                    mContext.sendBroadcastAsUser(intent, UserHandle.ALL);
487691fb51e7ac5e4d0b91a4719cd0ed10a8c212e50Vinit Deshpande                } finally {
488691fb51e7ac5e4d0b91a4719cd0ed10a8c212e50Vinit Deshpande                    // restore calling identity
489691fb51e7ac5e4d0b91a4719cd0ed10a8c212e50Vinit Deshpande                    Binder.restoreCallingIdentity(callingIdentity);
490691fb51e7ac5e4d0b91a4719cd0ed10a8c212e50Vinit Deshpande                }
49103b70b37a8a20cb92d860b5015f6762dcb5e75ccDianne Hackborn                mScanPending = true;
49203b70b37a8a20cb92d860b5015f6762dcb5e75ccDianne Hackborn                return;
49303b70b37a8a20cb92d860b5015f6762dcb5e75ccDianne Hackborn            }
49403b70b37a8a20cb92d860b5015f6762dcb5e75ccDianne Hackborn        }
495a0009d14075b5345b8f916c3fb3f2260c938cb9dYuhao Zheng        if (settings != null) {
496a0009d14075b5345b8f916c3fb3f2260c938cb9dYuhao Zheng            settings = new ScanSettings(settings);
497a0009d14075b5345b8f916c3fb3f2260c938cb9dYuhao Zheng            if (!settings.isValid()) {
498a0009d14075b5345b8f916c3fb3f2260c938cb9dYuhao Zheng                Slog.e(TAG, "invalid scan setting");
499a0009d14075b5345b8f916c3fb3f2260c938cb9dYuhao Zheng                return;
500a0009d14075b5345b8f916c3fb3f2260c938cb9dYuhao Zheng            }
501a0009d14075b5345b8f916c3fb3f2260c938cb9dYuhao Zheng        }
502155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        if (workSource != null) {
503155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande            enforceWorkSourcePermission();
504155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande            // WifiManager currently doesn't use names, so need to clear names out of the
505155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande            // supplied WorkSource to allow future WorkSource combining.
506155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande            workSource.clearNames();
507155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        }
50848444cb4214a48a3a0bf4bbb93945c2aa68c9980Mitchell Wills        if (workSource == null && Binder.getCallingUid() >= 0) {
50948444cb4214a48a3a0bf4bbb93945c2aa68c9980Mitchell Wills            workSource = new WorkSource(Binder.getCallingUid());
51048444cb4214a48a3a0bf4bbb93945c2aa68c9980Mitchell Wills        }
511ebb7e0cb4572f73325a749539d589ebdefb8635cvandwalle        mWifiStateMachine.startScan(Binder.getCallingUid(), scanRequestCounter++,
512ebb7e0cb4572f73325a749539d589ebdefb8635cvandwalle                settings, workSource);
513155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    }
514155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
5158fcd1fe86ddc3056e1a26e69bc68f148ae0e9ddcChristopher Wiley    @Override
516ba3f5bc64ef27f2ec0d3eae3f53c633ea9e66268Amin Shaikh    public String getCurrentNetworkWpsNfcConfigurationToken() {
517f3f4029b3ac41da4cc6a1bc22bae47c750a47048Andres Morales        enforceConnectivityInternalPermission();
518ba3f5bc64ef27f2ec0d3eae3f53c633ea9e66268Amin Shaikh        mLog.trace("getCurrentNetworkWpsNfcConfigurationToken uid=%")
519ba3f5bc64ef27f2ec0d3eae3f53c633ea9e66268Amin Shaikh                .c(Binder.getCallingUid()).flush();
520f2fdf411925ad172b5e0b25b0c6df880256691d4Sohani Rao        // TODO Add private logging for netId b/33807876
521ba3f5bc64ef27f2ec0d3eae3f53c633ea9e66268Amin Shaikh        return mWifiStateMachine.syncGetCurrentNetworkWpsNfcConfigurationToken();
5223f7ef65ab71619040032aee96b5599849881d6fdAndres Morales    }
5233f7ef65ab71619040032aee96b5599849881d6fdAndres Morales
5241227b49a94f33844ad0606b48b591bea4d27b08eVinit Deshpande    boolean mInIdleMode;
5251227b49a94f33844ad0606b48b591bea4d27b08eVinit Deshpande    boolean mScanPending;
5261227b49a94f33844ad0606b48b591bea4d27b08eVinit Deshpande
52703b70b37a8a20cb92d860b5015f6762dcb5e75ccDianne Hackborn    void handleIdleModeChanged() {
52803b70b37a8a20cb92d860b5015f6762dcb5e75ccDianne Hackborn        boolean doScan = false;
5291227b49a94f33844ad0606b48b591bea4d27b08eVinit Deshpande        synchronized (this) {
53003b70b37a8a20cb92d860b5015f6762dcb5e75ccDianne Hackborn            boolean idle = mPowerManager.isDeviceIdleMode();
53103b70b37a8a20cb92d860b5015f6762dcb5e75ccDianne Hackborn            if (mInIdleMode != idle) {
53203b70b37a8a20cb92d860b5015f6762dcb5e75ccDianne Hackborn                mInIdleMode = idle;
53303b70b37a8a20cb92d860b5015f6762dcb5e75ccDianne Hackborn                if (!idle) {
53403b70b37a8a20cb92d860b5015f6762dcb5e75ccDianne Hackborn                    if (mScanPending) {
53503b70b37a8a20cb92d860b5015f6762dcb5e75ccDianne Hackborn                        mScanPending = false;
53603b70b37a8a20cb92d860b5015f6762dcb5e75ccDianne Hackborn                        doScan = true;
53703b70b37a8a20cb92d860b5015f6762dcb5e75ccDianne Hackborn                    }
53803b70b37a8a20cb92d860b5015f6762dcb5e75ccDianne Hackborn                }
53903b70b37a8a20cb92d860b5015f6762dcb5e75ccDianne Hackborn            }
54003b70b37a8a20cb92d860b5015f6762dcb5e75ccDianne Hackborn        }
54103b70b37a8a20cb92d860b5015f6762dcb5e75ccDianne Hackborn        if (doScan) {
54203b70b37a8a20cb92d860b5015f6762dcb5e75ccDianne Hackborn            // Someone requested a scan while we were idle; do a full scan now.
54303b70b37a8a20cb92d860b5015f6762dcb5e75ccDianne Hackborn            startScan(null, null);
54403b70b37a8a20cb92d860b5015f6762dcb5e75ccDianne Hackborn        }
54503b70b37a8a20cb92d860b5015f6762dcb5e75ccDianne Hackborn    }
54603b70b37a8a20cb92d860b5015f6762dcb5e75ccDianne Hackborn
547155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    private void enforceAccessPermission() {
548155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        mContext.enforceCallingOrSelfPermission(android.Manifest.permission.ACCESS_WIFI_STATE,
549a632d8a6edd350c7644b593b18eceaa5b368505bVinit Deshpande                "WifiService");
550155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    }
551155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
552155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    private void enforceChangePermission() {
553155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        mContext.enforceCallingOrSelfPermission(android.Manifest.permission.CHANGE_WIFI_STATE,
55431891afce40b903ada9b24ec12e3648ae6aa27b2Jan Nordqvist                "WifiService");
5559086afccf6938a49eb9a2cd248917c1cb0943942vandwalle    }
556155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
5579086afccf6938a49eb9a2cd248917c1cb0943942vandwalle    private void enforceLocationHardwarePermission() {
5589086afccf6938a49eb9a2cd248917c1cb0943942vandwalle        mContext.enforceCallingOrSelfPermission(Manifest.permission.LOCATION_HARDWARE,
5599086afccf6938a49eb9a2cd248917c1cb0943942vandwalle                "LocationHardware");
560155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    }
561155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
5629878c61bbd81176561991be025af44efc67332feWenchao Tong    private void enforceReadCredentialPermission() {
5639878c61bbd81176561991be025af44efc67332feWenchao Tong        mContext.enforceCallingOrSelfPermission(android.Manifest.permission.READ_WIFI_CREDENTIAL,
5649878c61bbd81176561991be025af44efc67332feWenchao Tong                                                "WifiService");
5659878c61bbd81176561991be025af44efc67332feWenchao Tong    }
5669878c61bbd81176561991be025af44efc67332feWenchao Tong
567155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    private void enforceWorkSourcePermission() {
568155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        mContext.enforceCallingPermission(android.Manifest.permission.UPDATE_DEVICE_STATS,
569a632d8a6edd350c7644b593b18eceaa5b368505bVinit Deshpande                "WifiService");
570155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
571155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    }
572155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
573155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    private void enforceMulticastChangePermission() {
574155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        mContext.enforceCallingOrSelfPermission(
575155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande                android.Manifest.permission.CHANGE_WIFI_MULTICAST_STATE,
576155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande                "WifiService");
577155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    }
578155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
579155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    private void enforceConnectivityInternalPermission() {
580155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        mContext.enforceCallingOrSelfPermission(
581155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande                android.Manifest.permission.CONNECTIVITY_INTERNAL,
582155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande                "ConnectivityService");
583155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    }
584155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
585155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    /**
586155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande     * see {@link android.net.wifi.WifiManager#setWifiEnabled(boolean)}
587155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande     * @param enable {@code true} to enable, {@code false} to disable.
588155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande     * @return {@code true} if the enable/disable operation was
589155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande     *         started or is already in the queue.
590155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande     */
5918fcd1fe86ddc3056e1a26e69bc68f148ae0e9ddcChristopher Wiley    @Override
592dcf967aa402a4ab1a79c727aea934b8013c1fa6aSvetoslav Ganov    public synchronized boolean setWifiEnabled(String packageName, boolean enable)
593dcf967aa402a4ab1a79c727aea934b8013c1fa6aSvetoslav Ganov            throws RemoteException {
594155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        enforceChangePermission();
595155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        Slog.d(TAG, "setWifiEnabled: " + enable + " pid=" + Binder.getCallingPid()
59605d6ff7612ee795daf3a6fd0ee5a99e01c7a8685Rebecca Silberstein                    + ", uid=" + Binder.getCallingUid() + ", package=" + packageName);
59705d6ff7612ee795daf3a6fd0ee5a99e01c7a8685Rebecca Silberstein        mLog.trace("setWifiEnabled package=% uid=% enable=%").c(packageName)
59805d6ff7612ee795daf3a6fd0ee5a99e01c7a8685Rebecca Silberstein                .c(Binder.getCallingUid()).c(enable).flush();
59905d6ff7612ee795daf3a6fd0ee5a99e01c7a8685Rebecca Silberstein
60005d6ff7612ee795daf3a6fd0ee5a99e01c7a8685Rebecca Silberstein        // If SoftAp is enabled, only Settings is allowed to toggle wifi
60105d6ff7612ee795daf3a6fd0ee5a99e01c7a8685Rebecca Silberstein        boolean apEnabled =
60205d6ff7612ee795daf3a6fd0ee5a99e01c7a8685Rebecca Silberstein                mWifiStateMachine.syncGetWifiApState() != WifiManager.WIFI_AP_STATE_DISABLED;
60305d6ff7612ee795daf3a6fd0ee5a99e01c7a8685Rebecca Silberstein        boolean isFromSettings =
60405d6ff7612ee795daf3a6fd0ee5a99e01c7a8685Rebecca Silberstein                packageName.equals(SYSUI_PACKAGE_NAME) || packageName.equals(SETTINGS_PACKAGE_NAME);
60505d6ff7612ee795daf3a6fd0ee5a99e01c7a8685Rebecca Silberstein        if (apEnabled && !isFromSettings) {
60605d6ff7612ee795daf3a6fd0ee5a99e01c7a8685Rebecca Silberstein            mLog.trace("setWifiEnabled SoftAp not disabled: only Settings can enable wifi").flush();
60705d6ff7612ee795daf3a6fd0ee5a99e01c7a8685Rebecca Silberstein            return false;
60805d6ff7612ee795daf3a6fd0ee5a99e01c7a8685Rebecca Silberstein        }
60905d6ff7612ee795daf3a6fd0ee5a99e01c7a8685Rebecca Silberstein
610155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        /*
611155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        * Caller might not have WRITE_SECURE_SETTINGS,
612155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        * only CHANGE_WIFI_STATE is enforced
613155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        */
614155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        long ident = Binder.clearCallingIdentity();
615155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        try {
616155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande            if (! mSettingsStore.handleWifiToggled(enable)) {
617155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande                // Nothing to do if wifi cannot be toggled
618155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande                return true;
619155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande            }
620155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        } finally {
621155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande            Binder.restoreCallingIdentity(ident);
622155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        }
623155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
624dcf967aa402a4ab1a79c727aea934b8013c1fa6aSvetoslav Ganov
625dcf967aa402a4ab1a79c727aea934b8013c1fa6aSvetoslav Ganov        if (mPermissionReviewRequired) {
626dcf967aa402a4ab1a79c727aea934b8013c1fa6aSvetoslav Ganov            final int wiFiEnabledState = getWifiEnabledState();
627dcf967aa402a4ab1a79c727aea934b8013c1fa6aSvetoslav Ganov            if (enable) {
628dcf967aa402a4ab1a79c727aea934b8013c1fa6aSvetoslav Ganov                if (wiFiEnabledState == WifiManager.WIFI_STATE_DISABLING
629dcf967aa402a4ab1a79c727aea934b8013c1fa6aSvetoslav Ganov                        || wiFiEnabledState == WifiManager.WIFI_STATE_DISABLED) {
63002938a0a735da7fafaaed84e31e1aa93cdf80a56Ivan Podogov                    if (startConsentUi(packageName, Binder.getCallingUid(),
631dcf967aa402a4ab1a79c727aea934b8013c1fa6aSvetoslav Ganov                            WifiManager.ACTION_REQUEST_ENABLE)) {
632dcf967aa402a4ab1a79c727aea934b8013c1fa6aSvetoslav Ganov                        return true;
633dcf967aa402a4ab1a79c727aea934b8013c1fa6aSvetoslav Ganov                    }
634dcf967aa402a4ab1a79c727aea934b8013c1fa6aSvetoslav Ganov                }
635dcf967aa402a4ab1a79c727aea934b8013c1fa6aSvetoslav Ganov            } else if (wiFiEnabledState == WifiManager.WIFI_STATE_ENABLING
636dcf967aa402a4ab1a79c727aea934b8013c1fa6aSvetoslav Ganov                    || wiFiEnabledState == WifiManager.WIFI_STATE_ENABLED) {
63702938a0a735da7fafaaed84e31e1aa93cdf80a56Ivan Podogov                if (startConsentUi(packageName, Binder.getCallingUid(),
638dcf967aa402a4ab1a79c727aea934b8013c1fa6aSvetoslav Ganov                        WifiManager.ACTION_REQUEST_DISABLE)) {
639dcf967aa402a4ab1a79c727aea934b8013c1fa6aSvetoslav Ganov                    return true;
640dcf967aa402a4ab1a79c727aea934b8013c1fa6aSvetoslav Ganov                }
641dcf967aa402a4ab1a79c727aea934b8013c1fa6aSvetoslav Ganov            }
642dcf967aa402a4ab1a79c727aea934b8013c1fa6aSvetoslav Ganov        }
643dcf967aa402a4ab1a79c727aea934b8013c1fa6aSvetoslav Ganov
644155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        mWifiController.sendMessage(CMD_WIFI_TOGGLED);
645155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        return true;
646155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    }
647155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
648155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    /**
649155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande     * see {@link WifiManager#getWifiState()}
650155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande     * @return One of {@link WifiManager#WIFI_STATE_DISABLED},
651155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande     *         {@link WifiManager#WIFI_STATE_DISABLING},
652155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande     *         {@link WifiManager#WIFI_STATE_ENABLED},
653155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande     *         {@link WifiManager#WIFI_STATE_ENABLING},
654155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande     *         {@link WifiManager#WIFI_STATE_UNKNOWN}
655155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande     */
6568fcd1fe86ddc3056e1a26e69bc68f148ae0e9ddcChristopher Wiley    @Override
657155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    public int getWifiEnabledState() {
658155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        enforceAccessPermission();
659f2fdf411925ad172b5e0b25b0c6df880256691d4Sohani Rao        mLog.trace("getWifiEnabledState uid=%").c(Binder.getCallingUid()).flush();
660155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        return mWifiStateMachine.syncGetWifiState();
661155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    }
662155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
663155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    /**
664155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande     * see {@link android.net.wifi.WifiManager#setWifiApEnabled(WifiConfiguration, boolean)}
665155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande     * @param wifiConfig SSID, security and channel details as
666155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande     *        part of WifiConfiguration
667155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande     * @param enabled true to enable and false to disable
668155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande     */
6698fcd1fe86ddc3056e1a26e69bc68f148ae0e9ddcChristopher Wiley    @Override
670155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    public void setWifiApEnabled(WifiConfiguration wifiConfig, boolean enabled) {
671328fc5407927799843c11f2b767a8cf47b89f366Robert Greenwalt        enforceChangePermission();
6723871ff67f4a6970f1831fc8951392746c9e2bfa2Rebecca Silberstein        mWifiPermissionsUtil.enforceTetherChangePermission(mContext);
673f2fdf411925ad172b5e0b25b0c6df880256691d4Sohani Rao
674f2fdf411925ad172b5e0b25b0c6df880256691d4Sohani Rao        mLog.trace("setWifiApEnabled uid=% enable=%").c(Binder.getCallingUid()).c(enabled).flush();
675f2fdf411925ad172b5e0b25b0c6df880256691d4Sohani Rao
676f9d866f1b4e93593b3411886fd71e729f4a5a205Stuart Scott        if (mUserManager.hasUserRestriction(UserManager.DISALLOW_CONFIG_TETHERING)) {
67713cddb5337418359eb5f9bebd0504fbc5c47fc41Julia Reynolds            throw new SecurityException("DISALLOW_CONFIG_TETHERING is enabled for this user.");
67813cddb5337418359eb5f9bebd0504fbc5c47fc41Julia Reynolds        }
679155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        // null wifiConfig is a meaningful input for CMD_SET_AP
680d03a9283285bb7adef6c687eb3a91fa4a8c4b502Vinit Deshpande        if (wifiConfig == null || isValid(wifiConfig)) {
681184a4207f60c2255486c28b45724c71738d087b7Rebecca Silberstein            mWifiController.sendMessage(CMD_SET_AP, enabled ? 1 : 0, 0, wifiConfig);
682155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        } else {
683155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande            Slog.e(TAG, "Invalid WifiConfiguration");
684155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        }
685155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    }
686155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
687155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    /**
688155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande     * see {@link WifiManager#getWifiApState()}
689155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande     * @return One of {@link WifiManager#WIFI_AP_STATE_DISABLED},
690155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande     *         {@link WifiManager#WIFI_AP_STATE_DISABLING},
691155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande     *         {@link WifiManager#WIFI_AP_STATE_ENABLED},
692155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande     *         {@link WifiManager#WIFI_AP_STATE_ENABLING},
693155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande     *         {@link WifiManager#WIFI_AP_STATE_FAILED}
694155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande     */
6958fcd1fe86ddc3056e1a26e69bc68f148ae0e9ddcChristopher Wiley    @Override
696155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    public int getWifiApEnabledState() {
697155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        enforceAccessPermission();
698f2fdf411925ad172b5e0b25b0c6df880256691d4Sohani Rao        mLog.trace("getWifiApEnabledState uid=%").c(Binder.getCallingUid()).flush();
699155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        return mWifiStateMachine.syncGetWifiApState();
700155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    }
701155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
702155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    /**
703155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande     * see {@link WifiManager#getWifiApConfiguration()}
704155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande     * @return soft access point configuration
7055c5e8b347d32f7bf5bb1e6f40b1e350f1e9fb0f3Rebecca Silberstein     * @throws SecurityException if the caller does not have permission to retrieve the softap
7065c5e8b347d32f7bf5bb1e6f40b1e350f1e9fb0f3Rebecca Silberstein     * config
707155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande     */
7088fcd1fe86ddc3056e1a26e69bc68f148ae0e9ddcChristopher Wiley    @Override
709155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    public WifiConfiguration getWifiApConfiguration() {
710155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        enforceAccessPermission();
7115c5e8b347d32f7bf5bb1e6f40b1e350f1e9fb0f3Rebecca Silberstein        int uid = Binder.getCallingUid();
7125c5e8b347d32f7bf5bb1e6f40b1e350f1e9fb0f3Rebecca Silberstein        // only allow Settings UI to get the saved SoftApConfig
7135c5e8b347d32f7bf5bb1e6f40b1e350f1e9fb0f3Rebecca Silberstein        if (!mWifiPermissionsUtil.checkConfigOverridePermission(uid)) {
7145c5e8b347d32f7bf5bb1e6f40b1e350f1e9fb0f3Rebecca Silberstein            // random apps should not be allowed to read the user specified config
7155c5e8b347d32f7bf5bb1e6f40b1e350f1e9fb0f3Rebecca Silberstein            throw new SecurityException("App not allowed to read or update stored WiFi Ap config "
7165c5e8b347d32f7bf5bb1e6f40b1e350f1e9fb0f3Rebecca Silberstein                    + "(uid = " + uid + ")");
7175c5e8b347d32f7bf5bb1e6f40b1e350f1e9fb0f3Rebecca Silberstein        }
7185c5e8b347d32f7bf5bb1e6f40b1e350f1e9fb0f3Rebecca Silberstein        mLog.trace("getWifiApConfiguration uid=%").c(uid).flush();
719155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        return mWifiStateMachine.syncGetWifiApConfiguration();
720155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    }
721155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
722155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    /**
723155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande     * see {@link WifiManager#setWifiApConfiguration(WifiConfiguration)}
724155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande     * @param wifiConfig WifiConfiguration details for soft access point
7255c5e8b347d32f7bf5bb1e6f40b1e350f1e9fb0f3Rebecca Silberstein     * @throws SecurityException if the caller does not have permission to write the sotap config
726155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande     */
7278fcd1fe86ddc3056e1a26e69bc68f148ae0e9ddcChristopher Wiley    @Override
728155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    public void setWifiApConfiguration(WifiConfiguration wifiConfig) {
729155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        enforceChangePermission();
7305c5e8b347d32f7bf5bb1e6f40b1e350f1e9fb0f3Rebecca Silberstein        int uid = Binder.getCallingUid();
7315c5e8b347d32f7bf5bb1e6f40b1e350f1e9fb0f3Rebecca Silberstein        // only allow Settings UI to write the stored SoftApConfig
7325c5e8b347d32f7bf5bb1e6f40b1e350f1e9fb0f3Rebecca Silberstein        if (!mWifiPermissionsUtil.checkConfigOverridePermission(uid)) {
7335c5e8b347d32f7bf5bb1e6f40b1e350f1e9fb0f3Rebecca Silberstein            // random apps should not be allowed to read the user specified config
7345c5e8b347d32f7bf5bb1e6f40b1e350f1e9fb0f3Rebecca Silberstein            throw new SecurityException("App not allowed to read or update stored WiFi AP config "
7355c5e8b347d32f7bf5bb1e6f40b1e350f1e9fb0f3Rebecca Silberstein                    + "(uid = " + uid + ")");
7365c5e8b347d32f7bf5bb1e6f40b1e350f1e9fb0f3Rebecca Silberstein        }
7375c5e8b347d32f7bf5bb1e6f40b1e350f1e9fb0f3Rebecca Silberstein        mLog.trace("setWifiApConfiguration uid=%").c(uid).flush();
738155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        if (wifiConfig == null)
739155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande            return;
740d03a9283285bb7adef6c687eb3a91fa4a8c4b502Vinit Deshpande        if (isValid(wifiConfig)) {
741155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande            mWifiStateMachine.setWifiApConfiguration(wifiConfig);
742155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        } else {
743155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande            Slog.e(TAG, "Invalid WifiConfiguration");
744155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        }
745155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    }
746155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
747155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    /**
7488fcd1fe86ddc3056e1a26e69bc68f148ae0e9ddcChristopher Wiley     * see {@link android.net.wifi.WifiManager#isScanAlwaysAvailable()}
749155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande     */
7508fcd1fe86ddc3056e1a26e69bc68f148ae0e9ddcChristopher Wiley    @Override
751155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    public boolean isScanAlwaysAvailable() {
752155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        enforceAccessPermission();
753f2fdf411925ad172b5e0b25b0c6df880256691d4Sohani Rao        mLog.trace("isScanAlwaysAvailable uid=%").c(Binder.getCallingUid()).flush();
754155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        return mSettingsStore.isScanAlwaysAvailable();
755155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    }
756155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
757155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    /**
758155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande     * see {@link android.net.wifi.WifiManager#disconnect()}
759155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande     */
7608fcd1fe86ddc3056e1a26e69bc68f148ae0e9ddcChristopher Wiley    @Override
761155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    public void disconnect() {
762155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        enforceChangePermission();
763f2fdf411925ad172b5e0b25b0c6df880256691d4Sohani Rao        mLog.trace("disconnect uid=%").c(Binder.getCallingUid()).flush();
764155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        mWifiStateMachine.disconnectCommand();
765155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    }
766155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
767155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    /**
768155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande     * see {@link android.net.wifi.WifiManager#reconnect()}
769155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande     */
7708fcd1fe86ddc3056e1a26e69bc68f148ae0e9ddcChristopher Wiley    @Override
771155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    public void reconnect() {
772155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        enforceChangePermission();
773f2fdf411925ad172b5e0b25b0c6df880256691d4Sohani Rao        mLog.trace("reconnect uid=%").c(Binder.getCallingUid()).flush();
774155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        mWifiStateMachine.reconnectCommand();
775155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    }
776155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
777155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    /**
778155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande     * see {@link android.net.wifi.WifiManager#reassociate()}
779155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande     */
7808fcd1fe86ddc3056e1a26e69bc68f148ae0e9ddcChristopher Wiley    @Override
781155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    public void reassociate() {
782155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        enforceChangePermission();
783f2fdf411925ad172b5e0b25b0c6df880256691d4Sohani Rao        mLog.trace("reassociate uid=%").c(Binder.getCallingUid()).flush();
784155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        mWifiStateMachine.reassociateCommand();
785155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    }
786155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
787155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    /**
788048dc8df2bd9394979dbb090b0cfe8d6627d23e3Vinit Deshpande     * see {@link android.net.wifi.WifiManager#getSupportedFeatures}
789a632d8a6edd350c7644b593b18eceaa5b368505bVinit Deshpande     */
7908fcd1fe86ddc3056e1a26e69bc68f148ae0e9ddcChristopher Wiley    @Override
791048dc8df2bd9394979dbb090b0cfe8d6627d23e3Vinit Deshpande    public int getSupportedFeatures() {
792a632d8a6edd350c7644b593b18eceaa5b368505bVinit Deshpande        enforceAccessPermission();
793f2fdf411925ad172b5e0b25b0c6df880256691d4Sohani Rao        mLog.trace("getSupportedFeatures uid=%").c(Binder.getCallingUid()).flush();
794a632d8a6edd350c7644b593b18eceaa5b368505bVinit Deshpande        if (mWifiStateMachineChannel != null) {
795048dc8df2bd9394979dbb090b0cfe8d6627d23e3Vinit Deshpande            return mWifiStateMachine.syncGetSupportedFeatures(mWifiStateMachineChannel);
796a632d8a6edd350c7644b593b18eceaa5b368505bVinit Deshpande        } else {
797a632d8a6edd350c7644b593b18eceaa5b368505bVinit Deshpande            Slog.e(TAG, "mWifiStateMachineChannel is not initialized");
798048dc8df2bd9394979dbb090b0cfe8d6627d23e3Vinit Deshpande            return 0;
799a632d8a6edd350c7644b593b18eceaa5b368505bVinit Deshpande        }
800a632d8a6edd350c7644b593b18eceaa5b368505bVinit Deshpande    }
801a632d8a6edd350c7644b593b18eceaa5b368505bVinit Deshpande
802c48fd721a9a3273f0faa927d93caad4c9832af7dAdam Lesinski    @Override
803c48fd721a9a3273f0faa927d93caad4c9832af7dAdam Lesinski    public void requestActivityInfo(ResultReceiver result) {
804c48fd721a9a3273f0faa927d93caad4c9832af7dAdam Lesinski        Bundle bundle = new Bundle();
805f2fdf411925ad172b5e0b25b0c6df880256691d4Sohani Rao        mLog.trace("requestActivityInfo uid=%").c(Binder.getCallingUid()).flush();
806c48fd721a9a3273f0faa927d93caad4c9832af7dAdam Lesinski        bundle.putParcelable(BatteryStats.RESULT_RECEIVER_CONTROLLER_KEY, reportActivityInfo());
807c48fd721a9a3273f0faa927d93caad4c9832af7dAdam Lesinski        result.send(0, bundle);
808c48fd721a9a3273f0faa927d93caad4c9832af7dAdam Lesinski    }
809c48fd721a9a3273f0faa927d93caad4c9832af7dAdam Lesinski
810a632d8a6edd350c7644b593b18eceaa5b368505bVinit Deshpande    /**
81194a2dd090b7d3823058b7a69cf4653d5663708cdAdam Lesinski     * see {@link android.net.wifi.WifiManager#getControllerActivityEnergyInfo(int)}
812200e8ee5097134010a6edee8d031bb02ff7eeb5avandwalle     */
8138fcd1fe86ddc3056e1a26e69bc68f148ae0e9ddcChristopher Wiley    @Override
814048dc8df2bd9394979dbb090b0cfe8d6627d23e3Vinit Deshpande    public WifiActivityEnergyInfo reportActivityInfo() {
815200e8ee5097134010a6edee8d031bb02ff7eeb5avandwalle        enforceAccessPermission();
816f2fdf411925ad172b5e0b25b0c6df880256691d4Sohani Rao        mLog.trace("reportActivityInfo uid=%").c(Binder.getCallingUid()).flush();
8171f8f2971309aee4de9c35723f6fbca6fcab9a3beMitchell Wills        if ((getSupportedFeatures() & WifiManager.WIFI_FEATURE_LINK_LAYER_STATS) == 0) {
8181f8f2971309aee4de9c35723f6fbca6fcab9a3beMitchell Wills            return null;
8191f8f2971309aee4de9c35723f6fbca6fcab9a3beMitchell Wills        }
820200e8ee5097134010a6edee8d031bb02ff7eeb5avandwalle        WifiLinkLayerStats stats;
821200e8ee5097134010a6edee8d031bb02ff7eeb5avandwalle        WifiActivityEnergyInfo energyInfo = null;
822200e8ee5097134010a6edee8d031bb02ff7eeb5avandwalle        if (mWifiStateMachineChannel != null) {
823048dc8df2bd9394979dbb090b0cfe8d6627d23e3Vinit Deshpande            stats = mWifiStateMachine.syncGetLinkLayerStats(mWifiStateMachineChannel);
824200e8ee5097134010a6edee8d031bb02ff7eeb5avandwalle            if (stats != null) {
82594a2dd090b7d3823058b7a69cf4653d5663708cdAdam Lesinski                final long rxIdleCurrent = mContext.getResources().getInteger(
82694a2dd090b7d3823058b7a69cf4653d5663708cdAdam Lesinski                        com.android.internal.R.integer.config_wifi_idle_receive_cur_ma);
82794a2dd090b7d3823058b7a69cf4653d5663708cdAdam Lesinski                final long rxCurrent = mContext.getResources().getInteger(
82894a2dd090b7d3823058b7a69cf4653d5663708cdAdam Lesinski                        com.android.internal.R.integer.config_wifi_active_rx_cur_ma);
82994a2dd090b7d3823058b7a69cf4653d5663708cdAdam Lesinski                final long txCurrent = mContext.getResources().getInteger(
83094a2dd090b7d3823058b7a69cf4653d5663708cdAdam Lesinski                        com.android.internal.R.integer.config_wifi_tx_cur_ma);
83194a2dd090b7d3823058b7a69cf4653d5663708cdAdam Lesinski                final double voltage = mContext.getResources().getInteger(
83294a2dd090b7d3823058b7a69cf4653d5663708cdAdam Lesinski                        com.android.internal.R.integer.config_wifi_operating_voltage_mv)
83394a2dd090b7d3823058b7a69cf4653d5663708cdAdam Lesinski                        / 1000.0;
83494a2dd090b7d3823058b7a69cf4653d5663708cdAdam Lesinski
83594a2dd090b7d3823058b7a69cf4653d5663708cdAdam Lesinski                final long rxIdleTime = stats.on_time - stats.tx_time - stats.rx_time;
8369c8892ebd2ec5340785f6b787805fdc6088069d7Roshan Pius                final long[] txTimePerLevel;
8379c8892ebd2ec5340785f6b787805fdc6088069d7Roshan Pius                if (stats.tx_time_per_level != null) {
8389c8892ebd2ec5340785f6b787805fdc6088069d7Roshan Pius                    txTimePerLevel = new long[stats.tx_time_per_level.length];
8399c8892ebd2ec5340785f6b787805fdc6088069d7Roshan Pius                    for (int i = 0; i < txTimePerLevel.length; i++) {
8409c8892ebd2ec5340785f6b787805fdc6088069d7Roshan Pius                        txTimePerLevel[i] = stats.tx_time_per_level[i];
8419c8892ebd2ec5340785f6b787805fdc6088069d7Roshan Pius                        // TODO(b/27227497): Need to read the power consumed per level from config
8429c8892ebd2ec5340785f6b787805fdc6088069d7Roshan Pius                    }
8439c8892ebd2ec5340785f6b787805fdc6088069d7Roshan Pius                } else {
8449c8892ebd2ec5340785f6b787805fdc6088069d7Roshan Pius                    // This will happen if the HAL get link layer API returned null.
8459c8892ebd2ec5340785f6b787805fdc6088069d7Roshan Pius                    txTimePerLevel = new long[0];
846a1514cb347fccf11566e5172ed71030c695d2abdRoshan Pius                }
84794a2dd090b7d3823058b7a69cf4653d5663708cdAdam Lesinski                final long energyUsed = (long)((stats.tx_time * txCurrent +
84894a2dd090b7d3823058b7a69cf4653d5663708cdAdam Lesinski                        stats.rx_time * rxCurrent +
84994a2dd090b7d3823058b7a69cf4653d5663708cdAdam Lesinski                        rxIdleTime * rxIdleCurrent) * voltage);
850d76cdd8e08476ad5025f07a5d77a8d8f920a0721Adam Lesinski                if (VDBG || rxIdleTime < 0 || stats.on_time < 0 || stats.tx_time < 0 ||
851d76cdd8e08476ad5025f07a5d77a8d8f920a0721Adam Lesinski                        stats.rx_time < 0 || energyUsed < 0) {
85270b75cda33a05e8b6e6724ed08e4e72f6fee0471Pierre Vandwalle                    StringBuilder sb = new StringBuilder();
85370b75cda33a05e8b6e6724ed08e4e72f6fee0471Pierre Vandwalle                    sb.append(" rxIdleCur=" + rxIdleCurrent);
85470b75cda33a05e8b6e6724ed08e4e72f6fee0471Pierre Vandwalle                    sb.append(" rxCur=" + rxCurrent);
85570b75cda33a05e8b6e6724ed08e4e72f6fee0471Pierre Vandwalle                    sb.append(" txCur=" + txCurrent);
85670b75cda33a05e8b6e6724ed08e4e72f6fee0471Pierre Vandwalle                    sb.append(" voltage=" + voltage);
85770b75cda33a05e8b6e6724ed08e4e72f6fee0471Pierre Vandwalle                    sb.append(" on_time=" + stats.on_time);
85870b75cda33a05e8b6e6724ed08e4e72f6fee0471Pierre Vandwalle                    sb.append(" tx_time=" + stats.tx_time);
8599c8892ebd2ec5340785f6b787805fdc6088069d7Roshan Pius                    sb.append(" tx_time_per_level=" + Arrays.toString(txTimePerLevel));
86070b75cda33a05e8b6e6724ed08e4e72f6fee0471Pierre Vandwalle                    sb.append(" rx_time=" + stats.rx_time);
86170b75cda33a05e8b6e6724ed08e4e72f6fee0471Pierre Vandwalle                    sb.append(" rxIdleTime=" + rxIdleTime);
86270b75cda33a05e8b6e6724ed08e4e72f6fee0471Pierre Vandwalle                    sb.append(" energy=" + energyUsed);
863f9946f5663f2d7a9a69cb8b7e82a4ef8b1825c48Mitchell Wills                    Log.d(TAG, " reportActivityInfo: " + sb.toString());
86470b75cda33a05e8b6e6724ed08e4e72f6fee0471Pierre Vandwalle                }
86584a11a3d54b7c6f043dcf7a874b7c0d45709677aPierre Vandwalle
866200e8ee5097134010a6edee8d031bb02ff7eeb5avandwalle                // Convert the LinkLayerStats into EnergyActivity
867186f7513a080eb1a419b6d83286d62961168a5feAdam Lesinski                energyInfo = new WifiActivityEnergyInfo(SystemClock.elapsedRealtime(),
868200e8ee5097134010a6edee8d031bb02ff7eeb5avandwalle                        WifiActivityEnergyInfo.STACK_STATE_STATE_IDLE, stats.tx_time,
869a1514cb347fccf11566e5172ed71030c695d2abdRoshan Pius                        txTimePerLevel, stats.rx_time, rxIdleTime, energyUsed);
870200e8ee5097134010a6edee8d031bb02ff7eeb5avandwalle            }
8711f8f2971309aee4de9c35723f6fbca6fcab9a3beMitchell Wills            if (energyInfo != null && energyInfo.isValid()) {
8721f8f2971309aee4de9c35723f6fbca6fcab9a3beMitchell Wills                return energyInfo;
8731f8f2971309aee4de9c35723f6fbca6fcab9a3beMitchell Wills            } else {
8741f8f2971309aee4de9c35723f6fbca6fcab9a3beMitchell Wills                return null;
8751f8f2971309aee4de9c35723f6fbca6fcab9a3beMitchell Wills            }
876200e8ee5097134010a6edee8d031bb02ff7eeb5avandwalle        } else {
877200e8ee5097134010a6edee8d031bb02ff7eeb5avandwalle            Slog.e(TAG, "mWifiStateMachineChannel is not initialized");
878200e8ee5097134010a6edee8d031bb02ff7eeb5avandwalle            return null;
879200e8ee5097134010a6edee8d031bb02ff7eeb5avandwalle        }
880200e8ee5097134010a6edee8d031bb02ff7eeb5avandwalle    }
881200e8ee5097134010a6edee8d031bb02ff7eeb5avandwalle
882200e8ee5097134010a6edee8d031bb02ff7eeb5avandwalle    /**
883155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande     * see {@link android.net.wifi.WifiManager#getConfiguredNetworks()}
884155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande     * @return the list of configured networks
885155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande     */
8868fcd1fe86ddc3056e1a26e69bc68f148ae0e9ddcChristopher Wiley    @Override
887f384821f8979152ba83ec70614a59094c838c8c6zhangshuxiao    public ParceledListSlice<WifiConfiguration> getConfiguredNetworks() {
888155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        enforceAccessPermission();
889f2fdf411925ad172b5e0b25b0c6df880256691d4Sohani Rao        mLog.trace("getConfiguredNetworks uid=%").c(Binder.getCallingUid()).flush();
890155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        if (mWifiStateMachineChannel != null) {
891f384821f8979152ba83ec70614a59094c838c8c6zhangshuxiao            List<WifiConfiguration> configs = mWifiStateMachine.syncGetConfiguredNetworks(
892f384821f8979152ba83ec70614a59094c838c8c6zhangshuxiao                    Binder.getCallingUid(), mWifiStateMachineChannel);
893f384821f8979152ba83ec70614a59094c838c8c6zhangshuxiao            if (configs != null) {
894f384821f8979152ba83ec70614a59094c838c8c6zhangshuxiao                return new ParceledListSlice<WifiConfiguration>(configs);
895f384821f8979152ba83ec70614a59094c838c8c6zhangshuxiao            }
896155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        } else {
897155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande            Slog.e(TAG, "mWifiStateMachineChannel is not initialized");
898155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        }
899f384821f8979152ba83ec70614a59094c838c8c6zhangshuxiao        return null;
900155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    }
901155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
902155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    /**
9039878c61bbd81176561991be025af44efc67332feWenchao Tong     * see {@link android.net.wifi.WifiManager#getPrivilegedConfiguredNetworks()}
9049878c61bbd81176561991be025af44efc67332feWenchao Tong     * @return the list of configured networks with real preSharedKey
9059878c61bbd81176561991be025af44efc67332feWenchao Tong     */
9068fcd1fe86ddc3056e1a26e69bc68f148ae0e9ddcChristopher Wiley    @Override
907f384821f8979152ba83ec70614a59094c838c8c6zhangshuxiao    public ParceledListSlice<WifiConfiguration> getPrivilegedConfiguredNetworks() {
9089878c61bbd81176561991be025af44efc67332feWenchao Tong        enforceReadCredentialPermission();
9099878c61bbd81176561991be025af44efc67332feWenchao Tong        enforceAccessPermission();
910f2fdf411925ad172b5e0b25b0c6df880256691d4Sohani Rao        mLog.trace("getPrivilegedConfiguredNetworks uid=%").c(Binder.getCallingUid()).flush();
9119878c61bbd81176561991be025af44efc67332feWenchao Tong        if (mWifiStateMachineChannel != null) {
912f384821f8979152ba83ec70614a59094c838c8c6zhangshuxiao            List<WifiConfiguration> configs =
913f384821f8979152ba83ec70614a59094c838c8c6zhangshuxiao                    mWifiStateMachine.syncGetPrivilegedConfiguredNetwork(mWifiStateMachineChannel);
914f384821f8979152ba83ec70614a59094c838c8c6zhangshuxiao            if (configs != null) {
915f384821f8979152ba83ec70614a59094c838c8c6zhangshuxiao                return new ParceledListSlice<WifiConfiguration>(configs);
916f384821f8979152ba83ec70614a59094c838c8c6zhangshuxiao            }
9179878c61bbd81176561991be025af44efc67332feWenchao Tong        } else {
9189878c61bbd81176561991be025af44efc67332feWenchao Tong            Slog.e(TAG, "mWifiStateMachineChannel is not initialized");
9199878c61bbd81176561991be025af44efc67332feWenchao Tong        }
920f384821f8979152ba83ec70614a59094c838c8c6zhangshuxiao        return null;
9219878c61bbd81176561991be025af44efc67332feWenchao Tong    }
9229878c61bbd81176561991be025af44efc67332feWenchao Tong
9239878c61bbd81176561991be025af44efc67332feWenchao Tong    /**
9249af7a553d86c910a14410ddecea6fb9422c41cdbPeter Qiu     * Returns a WifiConfiguration for a Passpoint network matching this ScanResult.
9259af7a553d86c910a14410ddecea6fb9422c41cdbPeter Qiu     *
926ccbe820337a230f0827f734aca2c51a244a0837fVinit Deshpande     * @param scanResult scanResult that represents the BSSID
927ccbe820337a230f0827f734aca2c51a244a0837fVinit Deshpande     * @return {@link WifiConfiguration} that matches this BSSID or null
928ccbe820337a230f0827f734aca2c51a244a0837fVinit Deshpande     */
9298fcd1fe86ddc3056e1a26e69bc68f148ae0e9ddcChristopher Wiley    @Override
930ccbe820337a230f0827f734aca2c51a244a0837fVinit Deshpande    public WifiConfiguration getMatchingWifiConfig(ScanResult scanResult) {
931ccbe820337a230f0827f734aca2c51a244a0837fVinit Deshpande        enforceAccessPermission();
932f2fdf411925ad172b5e0b25b0c6df880256691d4Sohani Rao        mLog.trace("getMatchingWifiConfig uid=%").c(Binder.getCallingUid()).flush();
9339af7a553d86c910a14410ddecea6fb9422c41cdbPeter Qiu        if (!mContext.getResources().getBoolean(
9349af7a553d86c910a14410ddecea6fb9422c41cdbPeter Qiu                com.android.internal.R.bool.config_wifi_hotspot2_enabled)) {
9359af7a553d86c910a14410ddecea6fb9422c41cdbPeter Qiu            throw new UnsupportedOperationException("Passpoint not enabled");
9369af7a553d86c910a14410ddecea6fb9422c41cdbPeter Qiu        }
937ccbe820337a230f0827f734aca2c51a244a0837fVinit Deshpande        return mWifiStateMachine.syncGetMatchingWifiConfig(scanResult, mWifiStateMachineChannel);
938ccbe820337a230f0827f734aca2c51a244a0837fVinit Deshpande    }
939ccbe820337a230f0827f734aca2c51a244a0837fVinit Deshpande
940ccbe820337a230f0827f734aca2c51a244a0837fVinit Deshpande    /**
941155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande     * see {@link android.net.wifi.WifiManager#addOrUpdateNetwork(WifiConfiguration)}
942155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande     * @return the supplicant-assigned identifier for the new or updated
943155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande     * network if the operation succeeds, or {@code -1} if it fails
944155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande     */
9458fcd1fe86ddc3056e1a26e69bc68f148ae0e9ddcChristopher Wiley    @Override
946155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    public int addOrUpdateNetwork(WifiConfiguration config) {
947155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        enforceChangePermission();
948f2fdf411925ad172b5e0b25b0c6df880256691d4Sohani Rao        mLog.trace("addOrUpdateNetwork uid=%").c(Binder.getCallingUid()).flush();
9494aef3283bf78e233a11fa34af2c7c38362879625Jan Nordqvist        if (isValid(config) && isValidPasspoint(config)) {
9508be2ac8cec40a2000715849b627642bdc87c10b3Shirish Kalele
95131891afce40b903ada9b24ec12e3648ae6aa27b2Jan Nordqvist            WifiEnterpriseConfig enterpriseConfig = config.enterpriseConfig;
95231891afce40b903ada9b24ec12e3648ae6aa27b2Jan Nordqvist
9532e814680f4dd27a5f825afab189843582235cedcJan Nordqvist            if (config.isPasspoint() &&
9542e814680f4dd27a5f825afab189843582235cedcJan Nordqvist                    (enterpriseConfig.getEapMethod() == WifiEnterpriseConfig.Eap.TLS ||
9552e814680f4dd27a5f825afab189843582235cedcJan Nordqvist                            enterpriseConfig.getEapMethod() == WifiEnterpriseConfig.Eap.TTLS)) {
9562e814680f4dd27a5f825afab189843582235cedcJan Nordqvist                if (config.updateIdentifier != null) {
9572e814680f4dd27a5f825afab189843582235cedcJan Nordqvist                    enforceAccessPermission();
9582e814680f4dd27a5f825afab189843582235cedcJan Nordqvist                }
9592e814680f4dd27a5f825afab189843582235cedcJan Nordqvist                else {
960e511ec7a4f53f7204a85075b23318f9f11c46d94Mitchell Wills                    try {
961e511ec7a4f53f7204a85075b23318f9f11c46d94Mitchell Wills                        verifyCert(enterpriseConfig.getCaCertificate());
962e511ec7a4f53f7204a85075b23318f9f11c46d94Mitchell Wills                    } catch (CertPathValidatorException cpve) {
963e511ec7a4f53f7204a85075b23318f9f11c46d94Mitchell Wills                        Slog.e(TAG, "CA Cert " +
964e511ec7a4f53f7204a85075b23318f9f11c46d94Mitchell Wills                                enterpriseConfig.getCaCertificate().getSubjectX500Principal() +
965e511ec7a4f53f7204a85075b23318f9f11c46d94Mitchell Wills                                " untrusted: " + cpve.getMessage());
966e511ec7a4f53f7204a85075b23318f9f11c46d94Mitchell Wills                        return -1;
967e511ec7a4f53f7204a85075b23318f9f11c46d94Mitchell Wills                    } catch (GeneralSecurityException | IOException e) {
968e511ec7a4f53f7204a85075b23318f9f11c46d94Mitchell Wills                        Slog.e(TAG, "Failed to verify certificate" +
969e511ec7a4f53f7204a85075b23318f9f11c46d94Mitchell Wills                                enterpriseConfig.getCaCertificate().getSubjectX500Principal() +
970e511ec7a4f53f7204a85075b23318f9f11c46d94Mitchell Wills                                ": " + e);
971e511ec7a4f53f7204a85075b23318f9f11c46d94Mitchell Wills                        return -1;
972e511ec7a4f53f7204a85075b23318f9f11c46d94Mitchell Wills                    }
97331891afce40b903ada9b24ec12e3648ae6aa27b2Jan Nordqvist                }
97431891afce40b903ada9b24ec12e3648ae6aa27b2Jan Nordqvist            }
97531891afce40b903ada9b24ec12e3648ae6aa27b2Jan Nordqvist
976992ae00f25a9cc22cf5db3261bd7e72927069cf7vandwalle            //TODO: pass the Uid the WifiStateMachine as a message parameter
97731891afce40b903ada9b24ec12e3648ae6aa27b2Jan Nordqvist            Slog.i("addOrUpdateNetwork", " uid = " + Integer.toString(Binder.getCallingUid())
978992ae00f25a9cc22cf5db3261bd7e72927069cf7vandwalle                    + " SSID " + config.SSID
979992ae00f25a9cc22cf5db3261bd7e72927069cf7vandwalle                    + " nid=" + Integer.toString(config.networkId));
980992ae00f25a9cc22cf5db3261bd7e72927069cf7vandwalle            if (config.networkId == WifiConfiguration.INVALID_NETWORK_ID) {
981992ae00f25a9cc22cf5db3261bd7e72927069cf7vandwalle                config.creatorUid = Binder.getCallingUid();
982992ae00f25a9cc22cf5db3261bd7e72927069cf7vandwalle            } else {
983992ae00f25a9cc22cf5db3261bd7e72927069cf7vandwalle                config.lastUpdateUid = Binder.getCallingUid();
984992ae00f25a9cc22cf5db3261bd7e72927069cf7vandwalle            }
985155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande            if (mWifiStateMachineChannel != null) {
986155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande                return mWifiStateMachine.syncAddOrUpdateNetwork(mWifiStateMachineChannel, config);
987155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande            } else {
988155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande                Slog.e(TAG, "mWifiStateMachineChannel is not initialized");
989155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande                return -1;
990155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande            }
991155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        } else {
992155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande            Slog.e(TAG, "bad network configuration");
993155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande            return -1;
994155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        }
995155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    }
996155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
99731891afce40b903ada9b24ec12e3648ae6aa27b2Jan Nordqvist    public static void verifyCert(X509Certificate caCert)
99831891afce40b903ada9b24ec12e3648ae6aa27b2Jan Nordqvist            throws GeneralSecurityException, IOException {
99931891afce40b903ada9b24ec12e3648ae6aa27b2Jan Nordqvist        CertificateFactory factory = CertificateFactory.getInstance("X.509");
100031891afce40b903ada9b24ec12e3648ae6aa27b2Jan Nordqvist        CertPathValidator validator =
100131891afce40b903ada9b24ec12e3648ae6aa27b2Jan Nordqvist                CertPathValidator.getInstance(CertPathValidator.getDefaultType());
100231891afce40b903ada9b24ec12e3648ae6aa27b2Jan Nordqvist        CertPath path = factory.generateCertPath(
100331891afce40b903ada9b24ec12e3648ae6aa27b2Jan Nordqvist                Arrays.asList(caCert));
100431891afce40b903ada9b24ec12e3648ae6aa27b2Jan Nordqvist        KeyStore ks = KeyStore.getInstance("AndroidCAStore");
100531891afce40b903ada9b24ec12e3648ae6aa27b2Jan Nordqvist        ks.load(null, null);
100631891afce40b903ada9b24ec12e3648ae6aa27b2Jan Nordqvist        PKIXParameters params = new PKIXParameters(ks);
100731891afce40b903ada9b24ec12e3648ae6aa27b2Jan Nordqvist        params.setRevocationEnabled(false);
100831891afce40b903ada9b24ec12e3648ae6aa27b2Jan Nordqvist        validator.validate(path, params);
100931891afce40b903ada9b24ec12e3648ae6aa27b2Jan Nordqvist    }
101031891afce40b903ada9b24ec12e3648ae6aa27b2Jan Nordqvist
101131891afce40b903ada9b24ec12e3648ae6aa27b2Jan Nordqvist    /**
1012155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande     * See {@link android.net.wifi.WifiManager#removeNetwork(int)}
1013155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande     * @param netId the integer that identifies the network configuration
1014155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande     * to the supplicant
1015155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande     * @return {@code true} if the operation succeeded
1016155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande     */
10178fcd1fe86ddc3056e1a26e69bc68f148ae0e9ddcChristopher Wiley    @Override
1018155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    public boolean removeNetwork(int netId) {
1019155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        enforceChangePermission();
1020f2fdf411925ad172b5e0b25b0c6df880256691d4Sohani Rao        mLog.trace("removeNetwork uid=%").c(Binder.getCallingUid()).flush();
1021f2fdf411925ad172b5e0b25b0c6df880256691d4Sohani Rao        // TODO Add private logging for netId b/33807876
1022155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        if (mWifiStateMachineChannel != null) {
1023155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande            return mWifiStateMachine.syncRemoveNetwork(mWifiStateMachineChannel, netId);
1024155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        } else {
1025155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande            Slog.e(TAG, "mWifiStateMachineChannel is not initialized");
1026155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande            return false;
1027155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        }
1028155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    }
1029155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
1030155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    /**
1031155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande     * See {@link android.net.wifi.WifiManager#enableNetwork(int, boolean)}
1032155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande     * @param netId the integer that identifies the network configuration
1033155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande     * to the supplicant
1034155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande     * @param disableOthers if true, disable all other networks.
1035155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande     * @return {@code true} if the operation succeeded
1036155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande     */
10378fcd1fe86ddc3056e1a26e69bc68f148ae0e9ddcChristopher Wiley    @Override
1038155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    public boolean enableNetwork(int netId, boolean disableOthers) {
1039155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        enforceChangePermission();
1040f2fdf411925ad172b5e0b25b0c6df880256691d4Sohani Rao        // TODO b/33807876 Log netId
1041f2fdf411925ad172b5e0b25b0c6df880256691d4Sohani Rao        mLog.trace("enableNetwork uid=% disableOthers=%")
1042f2fdf411925ad172b5e0b25b0c6df880256691d4Sohani Rao                .c(Binder.getCallingUid())
1043f2fdf411925ad172b5e0b25b0c6df880256691d4Sohani Rao                .c(disableOthers).flush();
1044f2fdf411925ad172b5e0b25b0c6df880256691d4Sohani Rao
1045155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        if (mWifiStateMachineChannel != null) {
1046155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande            return mWifiStateMachine.syncEnableNetwork(mWifiStateMachineChannel, netId,
1047155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande                    disableOthers);
1048155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        } else {
1049155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande            Slog.e(TAG, "mWifiStateMachineChannel is not initialized");
1050155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande            return false;
1051155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        }
1052155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    }
1053155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
1054155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    /**
1055155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande     * See {@link android.net.wifi.WifiManager#disableNetwork(int)}
1056155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande     * @param netId the integer that identifies the network configuration
1057155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande     * to the supplicant
1058155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande     * @return {@code true} if the operation succeeded
1059155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande     */
10608fcd1fe86ddc3056e1a26e69bc68f148ae0e9ddcChristopher Wiley    @Override
1061155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    public boolean disableNetwork(int netId) {
1062155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        enforceChangePermission();
1063f2fdf411925ad172b5e0b25b0c6df880256691d4Sohani Rao        // TODO b/33807876 Log netId
1064f2fdf411925ad172b5e0b25b0c6df880256691d4Sohani Rao        mLog.trace("disableNetwork uid=%").c(Binder.getCallingUid()).flush();
1065f2fdf411925ad172b5e0b25b0c6df880256691d4Sohani Rao
1066155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        if (mWifiStateMachineChannel != null) {
1067155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande            return mWifiStateMachine.syncDisableNetwork(mWifiStateMachineChannel, netId);
1068155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        } else {
1069155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande            Slog.e(TAG, "mWifiStateMachineChannel is not initialized");
1070155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande            return false;
1071155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        }
1072155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    }
1073155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
1074155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    /**
1075155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande     * See {@link android.net.wifi.WifiManager#getConnectionInfo()}
1076155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande     * @return the Wi-Fi information, contained in {@link WifiInfo}.
1077155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande     */
10788fcd1fe86ddc3056e1a26e69bc68f148ae0e9ddcChristopher Wiley    @Override
1079155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    public WifiInfo getConnectionInfo() {
1080155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        enforceAccessPermission();
1081f2fdf411925ad172b5e0b25b0c6df880256691d4Sohani Rao        mLog.trace("getConnectionInfo uid=%").c(Binder.getCallingUid()).flush();
1082155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        /*
1083155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande         * Make sure we have the latest information, by sending
1084155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande         * a status request to the supplicant.
1085155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande         */
1086155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        return mWifiStateMachine.syncRequestConnectionInfo();
1087155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    }
1088155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
1089155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    /**
1090155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande     * Return the results of the most recent access point scan, in the form of
1091155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande     * a list of {@link ScanResult} objects.
1092155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande     * @return the list of results
1093155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande     */
10948fcd1fe86ddc3056e1a26e69bc68f148ae0e9ddcChristopher Wiley    @Override
1095155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    public List<ScanResult> getScanResults(String callingPackage) {
1096155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        enforceAccessPermission();
1097155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        int uid = Binder.getCallingUid();
1098155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        long ident = Binder.clearCallingIdentity();
1099155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        try {
1100868b692e6faa9ec3c8dd0cd42d4302082e28b992Sohani Rao            if (!mWifiPermissionsUtil.canAccessScanResults(callingPackage,
1101868b692e6faa9ec3c8dd0cd42d4302082e28b992Sohani Rao                      uid, Build.VERSION_CODES.M)) {
11021ecb0083490436303cdb89bc4c46b6743ea0afa8Sky Faber                return new ArrayList<ScanResult>();
11031ecb0083490436303cdb89bc4c46b6743ea0afa8Sky Faber            }
1104d94a62ffdf0e7f282948109d05d96e682eb32eefRebecca Silberstein            if (mWifiScanner == null) {
1105d94a62ffdf0e7f282948109d05d96e682eb32eefRebecca Silberstein                mWifiScanner = mWifiInjector.getWifiScanner();
1106d94a62ffdf0e7f282948109d05d96e682eb32eefRebecca Silberstein            }
1107d94a62ffdf0e7f282948109d05d96e682eb32eefRebecca Silberstein            return mWifiScanner.getSingleScanResults();
1108155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        } finally {
1109155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande            Binder.restoreCallingIdentity(ident);
1110155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        }
1111155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    }
1112155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
11131d5cd3938f9191184cd9aea3059a3b62bf3a0372Jan Nordqvist    /**
1114d28cfdde236d3d7c72f0c57ca7f18622b16d421aPeter Qiu     * Add or update a Passpoint configuration.
11153ebf0b17fd1cdd3a0ef22d9e199c1686c01dcc9fPeter Qiu     *
11163ebf0b17fd1cdd3a0ef22d9e199c1686c01dcc9fPeter Qiu     * @param config The Passpoint configuration to be added
11173ebf0b17fd1cdd3a0ef22d9e199c1686c01dcc9fPeter Qiu     * @return true on success or false on failure
11183ebf0b17fd1cdd3a0ef22d9e199c1686c01dcc9fPeter Qiu     */
11193ebf0b17fd1cdd3a0ef22d9e199c1686c01dcc9fPeter Qiu    @Override
1120d28cfdde236d3d7c72f0c57ca7f18622b16d421aPeter Qiu    public boolean addOrUpdatePasspointConfiguration(PasspointConfiguration config) {
11214781e9e2904824ef1fbf8a0cf75e89fa957d6a92Peter Qiu        enforceChangePermission();
1122f2fdf411925ad172b5e0b25b0c6df880256691d4Sohani Rao        mLog.trace("addorUpdatePasspointConfiguration uid=%").c(Binder.getCallingUid()).flush();
11239af7a553d86c910a14410ddecea6fb9422c41cdbPeter Qiu        if (!mContext.getResources().getBoolean(
11249af7a553d86c910a14410ddecea6fb9422c41cdbPeter Qiu                com.android.internal.R.bool.config_wifi_hotspot2_enabled)) {
11259af7a553d86c910a14410ddecea6fb9422c41cdbPeter Qiu            throw new UnsupportedOperationException("Passpoint not enabled");
11269af7a553d86c910a14410ddecea6fb9422c41cdbPeter Qiu        }
112707816a4745b8030911869ceb58fa735e47834fe4Peter Qiu        return mWifiStateMachine.syncAddOrUpdatePasspointConfig(mWifiStateMachineChannel, config,
112807816a4745b8030911869ceb58fa735e47834fe4Peter Qiu                Binder.getCallingUid());
11293ebf0b17fd1cdd3a0ef22d9e199c1686c01dcc9fPeter Qiu    }
11303ebf0b17fd1cdd3a0ef22d9e199c1686c01dcc9fPeter Qiu
11313ebf0b17fd1cdd3a0ef22d9e199c1686c01dcc9fPeter Qiu    /**
11323ebf0b17fd1cdd3a0ef22d9e199c1686c01dcc9fPeter Qiu     * Remove the Passpoint configuration identified by its FQDN (Fully Qualified Domain Name).
11333ebf0b17fd1cdd3a0ef22d9e199c1686c01dcc9fPeter Qiu     *
11343ebf0b17fd1cdd3a0ef22d9e199c1686c01dcc9fPeter Qiu     * @param fqdn The FQDN of the Passpoint configuration to be removed
11353ebf0b17fd1cdd3a0ef22d9e199c1686c01dcc9fPeter Qiu     * @return true on success or false on failure
11363ebf0b17fd1cdd3a0ef22d9e199c1686c01dcc9fPeter Qiu     */
11373ebf0b17fd1cdd3a0ef22d9e199c1686c01dcc9fPeter Qiu    @Override
11383ebf0b17fd1cdd3a0ef22d9e199c1686c01dcc9fPeter Qiu    public boolean removePasspointConfiguration(String fqdn) {
11394781e9e2904824ef1fbf8a0cf75e89fa957d6a92Peter Qiu        enforceChangePermission();
1140f2fdf411925ad172b5e0b25b0c6df880256691d4Sohani Rao        mLog.trace("removePasspointConfiguration uid=%").c(Binder.getCallingUid()).flush();
11419af7a553d86c910a14410ddecea6fb9422c41cdbPeter Qiu        if (!mContext.getResources().getBoolean(
11429af7a553d86c910a14410ddecea6fb9422c41cdbPeter Qiu                com.android.internal.R.bool.config_wifi_hotspot2_enabled)) {
11439af7a553d86c910a14410ddecea6fb9422c41cdbPeter Qiu            throw new UnsupportedOperationException("Passpoint not enabled");
11449af7a553d86c910a14410ddecea6fb9422c41cdbPeter Qiu        }
1145dd97694345f47ba6a952c1162e7dcdd66fb72060Peter Qiu        return mWifiStateMachine.syncRemovePasspointConfig(mWifiStateMachineChannel, fqdn);
11463ebf0b17fd1cdd3a0ef22d9e199c1686c01dcc9fPeter Qiu    }
11473ebf0b17fd1cdd3a0ef22d9e199c1686c01dcc9fPeter Qiu
11483ebf0b17fd1cdd3a0ef22d9e199c1686c01dcc9fPeter Qiu    /**
11493ebf0b17fd1cdd3a0ef22d9e199c1686c01dcc9fPeter Qiu     * Return the list of the installed Passpoint configurations.
11503ebf0b17fd1cdd3a0ef22d9e199c1686c01dcc9fPeter Qiu     *
1151d28cfdde236d3d7c72f0c57ca7f18622b16d421aPeter Qiu     * An empty list will be returned when no configuration is installed.
1152d28cfdde236d3d7c72f0c57ca7f18622b16d421aPeter Qiu     *
1153d28cfdde236d3d7c72f0c57ca7f18622b16d421aPeter Qiu     * @return A list of {@link PasspointConfiguration}
11543ebf0b17fd1cdd3a0ef22d9e199c1686c01dcc9fPeter Qiu     */
11553ebf0b17fd1cdd3a0ef22d9e199c1686c01dcc9fPeter Qiu    @Override
11563ebf0b17fd1cdd3a0ef22d9e199c1686c01dcc9fPeter Qiu    public List<PasspointConfiguration> getPasspointConfigurations() {
11574781e9e2904824ef1fbf8a0cf75e89fa957d6a92Peter Qiu        enforceAccessPermission();
1158f2fdf411925ad172b5e0b25b0c6df880256691d4Sohani Rao        mLog.trace("getPasspointConfigurations uid=%").c(Binder.getCallingUid()).flush();
11599af7a553d86c910a14410ddecea6fb9422c41cdbPeter Qiu        if (!mContext.getResources().getBoolean(
11609af7a553d86c910a14410ddecea6fb9422c41cdbPeter Qiu                com.android.internal.R.bool.config_wifi_hotspot2_enabled)) {
11619af7a553d86c910a14410ddecea6fb9422c41cdbPeter Qiu            throw new UnsupportedOperationException("Passpoint not enabled");
11629af7a553d86c910a14410ddecea6fb9422c41cdbPeter Qiu        }
1163dd97694345f47ba6a952c1162e7dcdd66fb72060Peter Qiu        return mWifiStateMachine.syncGetPasspointConfigs(mWifiStateMachineChannel);
11643ebf0b17fd1cdd3a0ef22d9e199c1686c01dcc9fPeter Qiu    }
11653ebf0b17fd1cdd3a0ef22d9e199c1686c01dcc9fPeter Qiu
11663ebf0b17fd1cdd3a0ef22d9e199c1686c01dcc9fPeter Qiu    /**
11672e814680f4dd27a5f825afab189843582235cedcJan Nordqvist     * Query for a Hotspot 2.0 release 2 OSU icon
11682e814680f4dd27a5f825afab189843582235cedcJan Nordqvist     * @param bssid The BSSID of the AP
11692e814680f4dd27a5f825afab189843582235cedcJan Nordqvist     * @param fileName Icon file name
11702e814680f4dd27a5f825afab189843582235cedcJan Nordqvist     */
11718fcd1fe86ddc3056e1a26e69bc68f148ae0e9ddcChristopher Wiley    @Override
11722e814680f4dd27a5f825afab189843582235cedcJan Nordqvist    public void queryPasspointIcon(long bssid, String fileName) {
117383b8a3d25eb4991277d8ec3720bc1f3d821166b6Peter Qiu        enforceAccessPermission();
1174f2fdf411925ad172b5e0b25b0c6df880256691d4Sohani Rao        mLog.trace("queryPasspointIcon uid=%").c(Binder.getCallingUid()).flush();
11759af7a553d86c910a14410ddecea6fb9422c41cdbPeter Qiu        if (!mContext.getResources().getBoolean(
11769af7a553d86c910a14410ddecea6fb9422c41cdbPeter Qiu                com.android.internal.R.bool.config_wifi_hotspot2_enabled)) {
11779af7a553d86c910a14410ddecea6fb9422c41cdbPeter Qiu            throw new UnsupportedOperationException("Passpoint not enabled");
11789af7a553d86c910a14410ddecea6fb9422c41cdbPeter Qiu        }
11792e814680f4dd27a5f825afab189843582235cedcJan Nordqvist        mWifiStateMachine.syncQueryPasspointIcon(mWifiStateMachineChannel, bssid, fileName);
11802e814680f4dd27a5f825afab189843582235cedcJan Nordqvist    }
11811d5cd3938f9191184cd9aea3059a3b62bf3a0372Jan Nordqvist
11822e814680f4dd27a5f825afab189843582235cedcJan Nordqvist    /**
11832e814680f4dd27a5f825afab189843582235cedcJan Nordqvist     * Match the currently associated network against the SP matching the given FQDN
11842e814680f4dd27a5f825afab189843582235cedcJan Nordqvist     * @param fqdn FQDN of the SP
11852e814680f4dd27a5f825afab189843582235cedcJan Nordqvist     * @return ordinal [HomeProvider, RoamingProvider, Incomplete, None, Declined]
11862e814680f4dd27a5f825afab189843582235cedcJan Nordqvist     */
11878fcd1fe86ddc3056e1a26e69bc68f148ae0e9ddcChristopher Wiley    @Override
11882e814680f4dd27a5f825afab189843582235cedcJan Nordqvist    public int matchProviderWithCurrentNetwork(String fqdn) {
1189f2fdf411925ad172b5e0b25b0c6df880256691d4Sohani Rao        mLog.trace("matchProviderWithCurrentNetwork uid=%").c(Binder.getCallingUid()).flush();
11902e814680f4dd27a5f825afab189843582235cedcJan Nordqvist        return mWifiStateMachine.matchProviderWithCurrentNetwork(mWifiStateMachineChannel, fqdn);
11911d5cd3938f9191184cd9aea3059a3b62bf3a0372Jan Nordqvist    }
11921d5cd3938f9191184cd9aea3059a3b62bf3a0372Jan Nordqvist
11932e814680f4dd27a5f825afab189843582235cedcJan Nordqvist    /**
11942e814680f4dd27a5f825afab189843582235cedcJan Nordqvist     * Deauthenticate and set the re-authentication hold off time for the current network
11952e814680f4dd27a5f825afab189843582235cedcJan Nordqvist     * @param holdoff hold off time in milliseconds
11962e814680f4dd27a5f825afab189843582235cedcJan Nordqvist     * @param ess set if the hold off pertains to an ESS rather than a BSS
11972e814680f4dd27a5f825afab189843582235cedcJan Nordqvist     */
11988fcd1fe86ddc3056e1a26e69bc68f148ae0e9ddcChristopher Wiley    @Override
11992e814680f4dd27a5f825afab189843582235cedcJan Nordqvist    public void deauthenticateNetwork(long holdoff, boolean ess) {
1200f2fdf411925ad172b5e0b25b0c6df880256691d4Sohani Rao        mLog.trace("deauthenticateNetwork uid=%").c(Binder.getCallingUid()).flush();
12012e814680f4dd27a5f825afab189843582235cedcJan Nordqvist        mWifiStateMachine.deauthenticateNetwork(mWifiStateMachineChannel, holdoff, ess);
12021d5cd3938f9191184cd9aea3059a3b62bf3a0372Jan Nordqvist    }
12031d5cd3938f9191184cd9aea3059a3b62bf3a0372Jan Nordqvist
1204fd5470391e5342daa38d00b68ccbccfeacbe6d33Alexandra Gherghina    /**
1205155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande     * Tell the supplicant to persist the current list of configured networks.
1206155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande     * @return {@code true} if the operation succeeded
1207155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande     *
1208155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande     * TODO: deprecate this
1209155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande     */
12108fcd1fe86ddc3056e1a26e69bc68f148ae0e9ddcChristopher Wiley    @Override
1211155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    public boolean saveConfiguration() {
1212155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        enforceChangePermission();
1213f2fdf411925ad172b5e0b25b0c6df880256691d4Sohani Rao        mLog.trace("saveConfiguration uid=%").c(Binder.getCallingUid()).flush();
1214155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        if (mWifiStateMachineChannel != null) {
1215155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande            return mWifiStateMachine.syncSaveConfig(mWifiStateMachineChannel);
1216155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        } else {
1217155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande            Slog.e(TAG, "mWifiStateMachineChannel is not initialized");
1218155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande            return false;
1219155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        }
1220155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    }
1221155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
1222155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    /**
1223155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande     * Set the country code
1224155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande     * @param countryCode ISO 3166 country code.
1225155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande     * @param persist {@code true} if the setting should be remembered.
1226155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande     *
1227155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande     * The persist behavior exists so that wifi can fall back to the last
1228155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande     * persisted country code on a restart, when the locale information is
1229155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande     * not available from telephony.
1230155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande     */
12318fcd1fe86ddc3056e1a26e69bc68f148ae0e9ddcChristopher Wiley    @Override
1232155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    public void setCountryCode(String countryCode, boolean persist) {
1233155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        Slog.i(TAG, "WifiService trying to set country code to " + countryCode +
1234155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande                " with persist set to " + persist);
12359c22c8ea193360f50ad490368ca6a11af0684362Robert Greenwalt        enforceConnectivityInternalPermission();
1236f2fdf411925ad172b5e0b25b0c6df880256691d4Sohani Rao        mLog.trace("setCountryCode uid=%").c(Binder.getCallingUid()).flush();
1237155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        final long token = Binder.clearCallingIdentity();
1238586197ab0ed6d1c9d76c7b461a19ab8733ce9b3aNingyuan Wang        mCountryCode.setCountryCode(countryCode);
1239586197ab0ed6d1c9d76c7b461a19ab8733ce9b3aNingyuan Wang        Binder.restoreCallingIdentity(token);
1240155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    }
1241155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
124280d14d67a956e6c69433470aaa73100898166efaxinhe     /**
124380d14d67a956e6c69433470aaa73100898166efaxinhe     * Get the country code
124437b06cd7aae7fe27cfaf1d95cc9901548765406bNingyuan Wang     * @return Get the best choice country code for wifi, regardless of if it was set or
124537b06cd7aae7fe27cfaf1d95cc9901548765406bNingyuan Wang     * not.
124637b06cd7aae7fe27cfaf1d95cc9901548765406bNingyuan Wang     * Returns null when there is no country code available.
124780d14d67a956e6c69433470aaa73100898166efaxinhe     */
12488fcd1fe86ddc3056e1a26e69bc68f148ae0e9ddcChristopher Wiley    @Override
124980d14d67a956e6c69433470aaa73100898166efaxinhe    public String getCountryCode() {
125080d14d67a956e6c69433470aaa73100898166efaxinhe        enforceConnectivityInternalPermission();
1251f2fdf411925ad172b5e0b25b0c6df880256691d4Sohani Rao        mLog.trace("getCountryCode uid=%").c(Binder.getCallingUid()).flush();
125237b06cd7aae7fe27cfaf1d95cc9901548765406bNingyuan Wang        String country = mCountryCode.getCountryCode();
125380d14d67a956e6c69433470aaa73100898166efaxinhe        return country;
125480d14d67a956e6c69433470aaa73100898166efaxinhe    }
1255155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
12568fcd1fe86ddc3056e1a26e69bc68f148ae0e9ddcChristopher Wiley    @Override
1257155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    public boolean isDualBandSupported() {
1258155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        //TODO: Should move towards adding a driver API that checks at runtime
1259f2fdf411925ad172b5e0b25b0c6df880256691d4Sohani Rao        mLog.trace("isDualBandSupported uid=%").c(Binder.getCallingUid()).flush();
1260155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        return mContext.getResources().getBoolean(
1261155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande                com.android.internal.R.bool.config_wifi_dual_band_support);
1262155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    }
1263155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
1264155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    /**
1265155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande     * Return the DHCP-assigned addresses from the last successful DHCP request,
1266155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande     * if any.
1267155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande     * @return the DHCP information
1268155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande     * @deprecated
1269155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande     */
12708fcd1fe86ddc3056e1a26e69bc68f148ae0e9ddcChristopher Wiley    @Override
12718fcd1fe86ddc3056e1a26e69bc68f148ae0e9ddcChristopher Wiley    @Deprecated
1272155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    public DhcpInfo getDhcpInfo() {
1273155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        enforceAccessPermission();
1274f2fdf411925ad172b5e0b25b0c6df880256691d4Sohani Rao        mLog.trace("getDhcpInfo uid=%").c(Binder.getCallingUid()).flush();
1275155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        DhcpResults dhcpResults = mWifiStateMachine.syncGetDhcpResults();
1276155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
1277155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        DhcpInfo info = new DhcpInfo();
12783b26801d62a06475b722bbf29cba7f48f376654eLorenzo Colitti
12793b26801d62a06475b722bbf29cba7f48f376654eLorenzo Colitti        if (dhcpResults.ipAddress != null &&
12803b26801d62a06475b722bbf29cba7f48f376654eLorenzo Colitti                dhcpResults.ipAddress.getAddress() instanceof Inet4Address) {
12813b26801d62a06475b722bbf29cba7f48f376654eLorenzo Colitti            info.ipAddress = NetworkUtils.inetAddressToInt((Inet4Address) dhcpResults.ipAddress.getAddress());
1282155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        }
12833b26801d62a06475b722bbf29cba7f48f376654eLorenzo Colitti
12843b26801d62a06475b722bbf29cba7f48f376654eLorenzo Colitti        if (dhcpResults.gateway != null) {
12853b26801d62a06475b722bbf29cba7f48f376654eLorenzo Colitti            info.gateway = NetworkUtils.inetAddressToInt((Inet4Address) dhcpResults.gateway);
1286155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        }
12873b26801d62a06475b722bbf29cba7f48f376654eLorenzo Colitti
1288155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        int dnsFound = 0;
12893b26801d62a06475b722bbf29cba7f48f376654eLorenzo Colitti        for (InetAddress dns : dhcpResults.dnsServers) {
1290155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande            if (dns instanceof Inet4Address) {
1291155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande                if (dnsFound == 0) {
1292155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande                    info.dns1 = NetworkUtils.inetAddressToInt((Inet4Address)dns);
1293155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande                } else {
1294155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande                    info.dns2 = NetworkUtils.inetAddressToInt((Inet4Address)dns);
1295155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande                }
1296155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande                if (++dnsFound > 1) break;
1297155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande            }
1298155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        }
12992af03130d7f85823223b8591dc52858d851b301dMitchell Wills        Inet4Address serverAddress = dhcpResults.serverAddress;
13002af03130d7f85823223b8591dc52858d851b301dMitchell Wills        if (serverAddress != null) {
13012af03130d7f85823223b8591dc52858d851b301dMitchell Wills            info.serverAddress = NetworkUtils.inetAddressToInt(serverAddress);
1302155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        }
1303155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        info.leaseDuration = dhcpResults.leaseDuration;
1304155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
1305155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        return info;
1306155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    }
1307155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
1308155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    /**
1309155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande     * enable TDLS for the local NIC to remote NIC
1310155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande     * The APPs don't know the remote MAC address to identify NIC though,
1311155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande     * so we need to do additional work to find it from remote IP address
1312155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande     */
1313155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
1314155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    class TdlsTaskParams {
1315155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        public String remoteIpAddress;
1316155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        public boolean enable;
1317155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    }
1318155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
1319155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    class TdlsTask extends AsyncTask<TdlsTaskParams, Integer, Integer> {
1320155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        @Override
1321155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        protected Integer doInBackground(TdlsTaskParams... params) {
1322155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
1323155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande            // Retrieve parameters for the call
1324155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande            TdlsTaskParams param = params[0];
1325155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande            String remoteIpAddress = param.remoteIpAddress.trim();
1326155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande            boolean enable = param.enable;
1327155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
1328155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande            // Get MAC address of Remote IP
1329155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande            String macAddress = null;
1330155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
1331155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande            BufferedReader reader = null;
1332155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
1333155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande            try {
1334155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande                reader = new BufferedReader(new FileReader("/proc/net/arp"));
1335155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
1336155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande                // Skip over the line bearing colum titles
1337155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande                String line = reader.readLine();
1338155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
1339155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande                while ((line = reader.readLine()) != null) {
1340155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande                    String[] tokens = line.split("[ ]+");
1341155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande                    if (tokens.length < 6) {
1342155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande                        continue;
1343155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande                    }
1344155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
1345155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande                    // ARP column format is
1346155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande                    // Address HWType HWAddress Flags Mask IFace
1347155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande                    String ip = tokens[0];
1348155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande                    String mac = tokens[3];
1349155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
1350155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande                    if (remoteIpAddress.equals(ip)) {
1351155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande                        macAddress = mac;
1352155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande                        break;
1353155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande                    }
1354155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande                }
1355155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
1356155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande                if (macAddress == null) {
1357155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande                    Slog.w(TAG, "Did not find remoteAddress {" + remoteIpAddress + "} in " +
1358155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande                            "/proc/net/arp");
1359155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande                } else {
1360155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande                    enableTdlsWithMacAddress(macAddress, enable);
1361155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande                }
1362155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
1363155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande            } catch (FileNotFoundException e) {
1364155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande                Slog.e(TAG, "Could not open /proc/net/arp to lookup mac address");
1365155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande            } catch (IOException e) {
1366155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande                Slog.e(TAG, "Could not read /proc/net/arp to lookup mac address");
1367155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande            } finally {
1368155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande                try {
1369155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande                    if (reader != null) {
1370155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande                        reader.close();
1371155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande                    }
1372155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande                }
1373155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande                catch (IOException e) {
1374155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande                    // Do nothing
1375155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande                }
1376155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande            }
1377155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
1378155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande            return 0;
1379155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        }
1380155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    }
1381155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
13828fcd1fe86ddc3056e1a26e69bc68f148ae0e9ddcChristopher Wiley    @Override
1383155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    public void enableTdls(String remoteAddress, boolean enable) {
13848e6144b4059fc2117be23d7222893bc57a33f64dJon Larimer        if (remoteAddress == null) {
13858e6144b4059fc2117be23d7222893bc57a33f64dJon Larimer          throw new IllegalArgumentException("remoteAddress cannot be null");
13868e6144b4059fc2117be23d7222893bc57a33f64dJon Larimer        }
1387f2fdf411925ad172b5e0b25b0c6df880256691d4Sohani Rao        mLog.trace("enableTdls uid=% enable=%").c(Binder.getCallingUid()).c(enable).flush();
1388155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        TdlsTaskParams params = new TdlsTaskParams();
1389155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        params.remoteIpAddress = remoteAddress;
1390155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        params.enable = enable;
1391155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        new TdlsTask().execute(params);
1392155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    }
1393155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
1394155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
13958fcd1fe86ddc3056e1a26e69bc68f148ae0e9ddcChristopher Wiley    @Override
1396155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    public void enableTdlsWithMacAddress(String remoteMacAddress, boolean enable) {
1397f2fdf411925ad172b5e0b25b0c6df880256691d4Sohani Rao        mLog.trace("enableTdlsWithMacAddress uid=% enable=%")
1398f2fdf411925ad172b5e0b25b0c6df880256691d4Sohani Rao                .c(Binder.getCallingUid())
1399f2fdf411925ad172b5e0b25b0c6df880256691d4Sohani Rao                .c(enable)
1400f2fdf411925ad172b5e0b25b0c6df880256691d4Sohani Rao                .flush();
14018e6144b4059fc2117be23d7222893bc57a33f64dJon Larimer        if (remoteMacAddress == null) {
14028e6144b4059fc2117be23d7222893bc57a33f64dJon Larimer          throw new IllegalArgumentException("remoteMacAddress cannot be null");
14038e6144b4059fc2117be23d7222893bc57a33f64dJon Larimer        }
14048e6144b4059fc2117be23d7222893bc57a33f64dJon Larimer
1405155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        mWifiStateMachine.enableTdls(remoteMacAddress, enable);
1406155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    }
1407155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
1408155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    /**
1409155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande     * Get a reference to handler. This is used by a client to establish
1410155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande     * an AsyncChannel communication with WifiService
1411155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande     */
14128fcd1fe86ddc3056e1a26e69bc68f148ae0e9ddcChristopher Wiley    @Override
1413155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    public Messenger getWifiServiceMessenger() {
1414155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        enforceAccessPermission();
1415155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        enforceChangePermission();
1416f2fdf411925ad172b5e0b25b0c6df880256691d4Sohani Rao        mLog.trace("getWifiServiceMessenger uid=%").c(Binder.getCallingUid()).flush();
1417155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        return new Messenger(mClientHandler);
1418155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    }
1419155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
14202ce99b40c36ed0352b31aa85d5f9383d5f0506f5vandwalle    /**
14212ce99b40c36ed0352b31aa85d5f9383d5f0506f5vandwalle     * Disable an ephemeral network, i.e. network that is created thru a WiFi Scorer
14222ce99b40c36ed0352b31aa85d5f9383d5f0506f5vandwalle     */
14238fcd1fe86ddc3056e1a26e69bc68f148ae0e9ddcChristopher Wiley    @Override
14242ce99b40c36ed0352b31aa85d5f9383d5f0506f5vandwalle    public void disableEphemeralNetwork(String SSID) {
14252ce99b40c36ed0352b31aa85d5f9383d5f0506f5vandwalle        enforceAccessPermission();
14262ce99b40c36ed0352b31aa85d5f9383d5f0506f5vandwalle        enforceChangePermission();
1427f2fdf411925ad172b5e0b25b0c6df880256691d4Sohani Rao        mLog.trace("disableEphemeralNetwork uid=%").c(Binder.getCallingUid()).flush();
14282ce99b40c36ed0352b31aa85d5f9383d5f0506f5vandwalle        mWifiStateMachine.disableEphemeralNetwork(SSID);
14292ce99b40c36ed0352b31aa85d5f9383d5f0506f5vandwalle    }
1430155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
1431155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    private final BroadcastReceiver mReceiver = new BroadcastReceiver() {
1432155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        @Override
1433155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        public void onReceive(Context context, Intent intent) {
1434155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande            String action = intent.getAction();
1435155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande            if (action.equals(Intent.ACTION_SCREEN_ON)) {
1436155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande                mWifiController.sendMessage(CMD_SCREEN_ON);
1437155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande            } else if (action.equals(Intent.ACTION_USER_PRESENT)) {
1438155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande                mWifiController.sendMessage(CMD_USER_PRESENT);
1439155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande            } else if (action.equals(Intent.ACTION_SCREEN_OFF)) {
1440155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande                mWifiController.sendMessage(CMD_SCREEN_OFF);
1441155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande            } else if (action.equals(Intent.ACTION_BATTERY_CHANGED)) {
1442155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande                int pluggedType = intent.getIntExtra("plugged", 0);
1443155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande                mWifiController.sendMessage(CMD_BATTERY_CHANGED, pluggedType, 0, null);
1444155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande            } else if (action.equals(BluetoothAdapter.ACTION_CONNECTION_STATE_CHANGED)) {
1445155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande                int state = intent.getIntExtra(BluetoothAdapter.EXTRA_CONNECTION_STATE,
1446155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande                        BluetoothAdapter.STATE_DISCONNECTED);
1447155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande                mWifiStateMachine.sendBluetoothAdapterStateChange(state);
1448155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande            } else if (action.equals(TelephonyIntents.ACTION_EMERGENCY_CALLBACK_MODE_CHANGED)) {
1449155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande                boolean emergencyMode = intent.getBooleanExtra("phoneinECMState", false);
1450155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande                mWifiController.sendMessage(CMD_EMERGENCY_MODE_CHANGED, emergencyMode ? 1 : 0, 0);
1451090aa74f2d36010abba6563a39c87ab7ba8a7473Vinit Deshpande            } else if (action.equals(TelephonyIntents.ACTION_EMERGENCY_CALL_STATE_CHANGED)) {
1452090aa74f2d36010abba6563a39c87ab7ba8a7473Vinit Deshpande                boolean inCall = intent.getBooleanExtra(PhoneConstants.PHONE_IN_EMERGENCY_CALL, false);
1453090aa74f2d36010abba6563a39c87ab7ba8a7473Vinit Deshpande                mWifiController.sendMessage(CMD_EMERGENCY_CALL_STATE_CHANGED, inCall ? 1 : 0, 0);
145403b70b37a8a20cb92d860b5015f6762dcb5e75ccDianne Hackborn            } else if (action.equals(PowerManager.ACTION_DEVICE_IDLE_MODE_CHANGED)) {
145503b70b37a8a20cb92d860b5015f6762dcb5e75ccDianne Hackborn                handleIdleModeChanged();
1456155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande            }
1457155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        }
1458155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    };
1459155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
146002938a0a735da7fafaaed84e31e1aa93cdf80a56Ivan Podogov    private boolean startConsentUi(String packageName,
1461dcf967aa402a4ab1a79c727aea934b8013c1fa6aSvetoslav Ganov            int callingUid, String intentAction) throws RemoteException {
1462dcf967aa402a4ab1a79c727aea934b8013c1fa6aSvetoslav Ganov        if (UserHandle.getAppId(callingUid) == Process.SYSTEM_UID) {
1463dcf967aa402a4ab1a79c727aea934b8013c1fa6aSvetoslav Ganov            return false;
1464dcf967aa402a4ab1a79c727aea934b8013c1fa6aSvetoslav Ganov        }
1465dcf967aa402a4ab1a79c727aea934b8013c1fa6aSvetoslav Ganov        try {
1466dcf967aa402a4ab1a79c727aea934b8013c1fa6aSvetoslav Ganov            // Validate the package only if we are going to use it
1467dcf967aa402a4ab1a79c727aea934b8013c1fa6aSvetoslav Ganov            ApplicationInfo applicationInfo = mContext.getPackageManager()
1468dcf967aa402a4ab1a79c727aea934b8013c1fa6aSvetoslav Ganov                    .getApplicationInfoAsUser(packageName,
1469dcf967aa402a4ab1a79c727aea934b8013c1fa6aSvetoslav Ganov                            PackageManager.MATCH_DEBUG_TRIAGED_MISSING,
1470dcf967aa402a4ab1a79c727aea934b8013c1fa6aSvetoslav Ganov                            UserHandle.getUserId(callingUid));
1471dcf967aa402a4ab1a79c727aea934b8013c1fa6aSvetoslav Ganov            if (applicationInfo.uid != callingUid) {
1472dcf967aa402a4ab1a79c727aea934b8013c1fa6aSvetoslav Ganov                throw new SecurityException("Package " + callingUid
1473dcf967aa402a4ab1a79c727aea934b8013c1fa6aSvetoslav Ganov                        + " not in uid " + callingUid);
1474dcf967aa402a4ab1a79c727aea934b8013c1fa6aSvetoslav Ganov            }
1475dcf967aa402a4ab1a79c727aea934b8013c1fa6aSvetoslav Ganov
147602938a0a735da7fafaaed84e31e1aa93cdf80a56Ivan Podogov            // Permission review mode, trigger a user prompt
147702938a0a735da7fafaaed84e31e1aa93cdf80a56Ivan Podogov            Intent intent = new Intent(intentAction);
147802938a0a735da7fafaaed84e31e1aa93cdf80a56Ivan Podogov            intent.addFlags(Intent.FLAG_ACTIVITY_NEW_TASK
147902938a0a735da7fafaaed84e31e1aa93cdf80a56Ivan Podogov                    | Intent.FLAG_ACTIVITY_EXCLUDE_FROM_RECENTS);
148002938a0a735da7fafaaed84e31e1aa93cdf80a56Ivan Podogov            intent.putExtra(Intent.EXTRA_PACKAGE_NAME, packageName);
148102938a0a735da7fafaaed84e31e1aa93cdf80a56Ivan Podogov            mContext.startActivity(intent);
148202938a0a735da7fafaaed84e31e1aa93cdf80a56Ivan Podogov            return true;
1483dcf967aa402a4ab1a79c727aea934b8013c1fa6aSvetoslav Ganov        } catch (PackageManager.NameNotFoundException e) {
1484dcf967aa402a4ab1a79c727aea934b8013c1fa6aSvetoslav Ganov            throw new RemoteException(e.getMessage());
1485dcf967aa402a4ab1a79c727aea934b8013c1fa6aSvetoslav Ganov        }
1486dcf967aa402a4ab1a79c727aea934b8013c1fa6aSvetoslav Ganov    }
1487dcf967aa402a4ab1a79c727aea934b8013c1fa6aSvetoslav Ganov
1488155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    /**
1489155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande     * Observes settings changes to scan always mode.
1490155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande     */
1491155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    private void registerForScanModeChange() {
1492155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        ContentObserver contentObserver = new ContentObserver(null) {
1493155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande            @Override
1494155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande            public void onChange(boolean selfChange) {
1495155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande                mSettingsStore.handleWifiScanAlwaysAvailableToggled();
1496155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande                mWifiController.sendMessage(CMD_SCAN_ALWAYS_MODE_CHANGED);
1497155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande            }
1498155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        };
14996232408a5fc660b2d8642d56747e05c15c6255b8Rebecca Silberstein        mFrameworkFacade.registerContentObserver(mContext,
1500155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande                Settings.Global.getUriFor(Settings.Global.WIFI_SCAN_ALWAYS_AVAILABLE),
1501155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande                false, contentObserver);
1502155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    }
1503155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
1504155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    private void registerForBroadcasts() {
1505155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        IntentFilter intentFilter = new IntentFilter();
1506155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        intentFilter.addAction(Intent.ACTION_SCREEN_ON);
1507155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        intentFilter.addAction(Intent.ACTION_USER_PRESENT);
1508155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        intentFilter.addAction(Intent.ACTION_SCREEN_OFF);
1509155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        intentFilter.addAction(Intent.ACTION_BATTERY_CHANGED);
1510155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        intentFilter.addAction(WifiManager.NETWORK_STATE_CHANGED_ACTION);
1511155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        intentFilter.addAction(BluetoothAdapter.ACTION_CONNECTION_STATE_CHANGED);
1512155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        intentFilter.addAction(TelephonyIntents.ACTION_EMERGENCY_CALLBACK_MODE_CHANGED);
151303b70b37a8a20cb92d860b5015f6762dcb5e75ccDianne Hackborn        intentFilter.addAction(PowerManager.ACTION_DEVICE_IDLE_MODE_CHANGED);
1514090aa74f2d36010abba6563a39c87ab7ba8a7473Vinit Deshpande
1515090aa74f2d36010abba6563a39c87ab7ba8a7473Vinit Deshpande        boolean trackEmergencyCallState = mContext.getResources().getBoolean(
1516090aa74f2d36010abba6563a39c87ab7ba8a7473Vinit Deshpande                com.android.internal.R.bool.config_wifi_turn_off_during_emergency_call);
1517090aa74f2d36010abba6563a39c87ab7ba8a7473Vinit Deshpande        if (trackEmergencyCallState) {
1518090aa74f2d36010abba6563a39c87ab7ba8a7473Vinit Deshpande            intentFilter.addAction(TelephonyIntents.ACTION_EMERGENCY_CALL_STATE_CHANGED);
1519090aa74f2d36010abba6563a39c87ab7ba8a7473Vinit Deshpande        }
1520090aa74f2d36010abba6563a39c87ab7ba8a7473Vinit Deshpande
1521155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        mContext.registerReceiver(mReceiver, intentFilter);
1522155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    }
1523155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
1524faa2ce7f93eef2c436796fb1ea3574e2d1412491Robin Lee    private void registerForPackageOrUserRemoval() {
1525e487a4648dd41881e754f1224aaedead78a0777dSky Faber        IntentFilter intentFilter = new IntentFilter();
1526e487a4648dd41881e754f1224aaedead78a0777dSky Faber        intentFilter.addAction(Intent.ACTION_PACKAGE_REMOVED);
1527faa2ce7f93eef2c436796fb1ea3574e2d1412491Robin Lee        intentFilter.addAction(Intent.ACTION_USER_REMOVED);
1528faa2ce7f93eef2c436796fb1ea3574e2d1412491Robin Lee        mContext.registerReceiverAsUser(new BroadcastReceiver() {
1529e487a4648dd41881e754f1224aaedead78a0777dSky Faber            @Override
1530e487a4648dd41881e754f1224aaedead78a0777dSky Faber            public void onReceive(Context context, Intent intent) {
1531faa2ce7f93eef2c436796fb1ea3574e2d1412491Robin Lee                switch (intent.getAction()) {
1532faa2ce7f93eef2c436796fb1ea3574e2d1412491Robin Lee                    case Intent.ACTION_PACKAGE_REMOVED: {
1533faa2ce7f93eef2c436796fb1ea3574e2d1412491Robin Lee                        if (intent.getBooleanExtra(Intent.EXTRA_REPLACING, false)) {
1534faa2ce7f93eef2c436796fb1ea3574e2d1412491Robin Lee                            return;
1535faa2ce7f93eef2c436796fb1ea3574e2d1412491Robin Lee                        }
1536faa2ce7f93eef2c436796fb1ea3574e2d1412491Robin Lee                        int uid = intent.getIntExtra(Intent.EXTRA_UID, -1);
1537faa2ce7f93eef2c436796fb1ea3574e2d1412491Robin Lee                        Uri uri = intent.getData();
1538faa2ce7f93eef2c436796fb1ea3574e2d1412491Robin Lee                        if (uid == -1 || uri == null) {
1539faa2ce7f93eef2c436796fb1ea3574e2d1412491Robin Lee                            return;
1540faa2ce7f93eef2c436796fb1ea3574e2d1412491Robin Lee                        }
1541faa2ce7f93eef2c436796fb1ea3574e2d1412491Robin Lee                        String pkgName = uri.getSchemeSpecificPart();
1542faa2ce7f93eef2c436796fb1ea3574e2d1412491Robin Lee                        mWifiStateMachine.removeAppConfigs(pkgName, uid);
1543faa2ce7f93eef2c436796fb1ea3574e2d1412491Robin Lee                        break;
1544faa2ce7f93eef2c436796fb1ea3574e2d1412491Robin Lee                    }
1545faa2ce7f93eef2c436796fb1ea3574e2d1412491Robin Lee                    case Intent.ACTION_USER_REMOVED: {
1546faa2ce7f93eef2c436796fb1ea3574e2d1412491Robin Lee                        int userHandle = intent.getIntExtra(Intent.EXTRA_USER_HANDLE, 0);
1547faa2ce7f93eef2c436796fb1ea3574e2d1412491Robin Lee                        mWifiStateMachine.removeUserConfigs(userHandle);
1548faa2ce7f93eef2c436796fb1ea3574e2d1412491Robin Lee                        break;
1549e487a4648dd41881e754f1224aaedead78a0777dSky Faber                    }
1550e487a4648dd41881e754f1224aaedead78a0777dSky Faber                }
1551e487a4648dd41881e754f1224aaedead78a0777dSky Faber            }
1552faa2ce7f93eef2c436796fb1ea3574e2d1412491Robin Lee        }, UserHandle.ALL, intentFilter, null, null);
1553e487a4648dd41881e754f1224aaedead78a0777dSky Faber    }
1554e487a4648dd41881e754f1224aaedead78a0777dSky Faber
1555155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    @Override
1556155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    protected void dump(FileDescriptor fd, PrintWriter pw, String[] args) {
1557155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        if (mContext.checkCallingOrSelfPermission(android.Manifest.permission.DUMP)
1558155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande                != PackageManager.PERMISSION_GRANTED) {
1559155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande            pw.println("Permission Denial: can't dump WifiService from from pid="
1560155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande                    + Binder.getCallingPid()
1561155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande                    + ", uid=" + Binder.getCallingUid());
1562155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande            return;
1563155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        }
156411638f348ba45f9f417928e79b81186cef76c561Glen Kuhne        if (args.length > 0 && WifiMetrics.PROTO_DUMP_ARG.equals(args[0])) {
156511638f348ba45f9f417928e79b81186cef76c561Glen Kuhne            // WifiMetrics proto bytes were requested. Dump only these.
156611638f348ba45f9f417928e79b81186cef76c561Glen Kuhne            mWifiStateMachine.updateWifiMetrics();
156711638f348ba45f9f417928e79b81186cef76c561Glen Kuhne            mWifiMetrics.dump(fd, pw, args);
1568675de6070a3c1980fcc99bc6863b2689bfeb0271Lorenzo Colitti        } else if (args.length > 0 && IpManager.DUMP_ARG.equals(args[0])) {
1569675de6070a3c1980fcc99bc6863b2689bfeb0271Lorenzo Colitti            // IpManager dump was requested. Pass it along and take no further action.
1570675de6070a3c1980fcc99bc6863b2689bfeb0271Lorenzo Colitti            String[] ipManagerArgs = new String[args.length - 1];
1571675de6070a3c1980fcc99bc6863b2689bfeb0271Lorenzo Colitti            System.arraycopy(args, 1, ipManagerArgs, 0, ipManagerArgs.length);
1572675de6070a3c1980fcc99bc6863b2689bfeb0271Lorenzo Colitti            mWifiStateMachine.dumpIpManager(fd, pw, ipManagerArgs);
157384ee4726383fd968ab43caf120d15d9f47a51a54Paul Stewart        } else if (args.length > 0 && DUMP_ARG_SET_IPREACH_DISCONNECT.equals(args[0])) {
157484ee4726383fd968ab43caf120d15d9f47a51a54Paul Stewart            if (args.length > 1) {
157584ee4726383fd968ab43caf120d15d9f47a51a54Paul Stewart                if (DUMP_ARG_SET_IPREACH_DISCONNECT_ENABLED.equals(args[1])) {
157684ee4726383fd968ab43caf120d15d9f47a51a54Paul Stewart                    mWifiStateMachine.setIpReachabilityDisconnectEnabled(true);
157784ee4726383fd968ab43caf120d15d9f47a51a54Paul Stewart                } else if (DUMP_ARG_SET_IPREACH_DISCONNECT_DISABLED.equals(args[1])) {
157884ee4726383fd968ab43caf120d15d9f47a51a54Paul Stewart                    mWifiStateMachine.setIpReachabilityDisconnectEnabled(false);
157984ee4726383fd968ab43caf120d15d9f47a51a54Paul Stewart                }
158084ee4726383fd968ab43caf120d15d9f47a51a54Paul Stewart            }
158184ee4726383fd968ab43caf120d15d9f47a51a54Paul Stewart            pw.println("IPREACH_DISCONNECT state is "
158284ee4726383fd968ab43caf120d15d9f47a51a54Paul Stewart                    + mWifiStateMachine.getIpReachabilityDisconnectEnabled());
158384ee4726383fd968ab43caf120d15d9f47a51a54Paul Stewart            return;
158411638f348ba45f9f417928e79b81186cef76c561Glen Kuhne        } else {
158511638f348ba45f9f417928e79b81186cef76c561Glen Kuhne            pw.println("Wi-Fi is " + mWifiStateMachine.syncGetWifiStateByName());
158611638f348ba45f9f417928e79b81186cef76c561Glen Kuhne            pw.println("Stay-awake conditions: " +
158711638f348ba45f9f417928e79b81186cef76c561Glen Kuhne                    Settings.Global.getInt(mContext.getContentResolver(),
158811638f348ba45f9f417928e79b81186cef76c561Glen Kuhne                                           Settings.Global.STAY_ON_WHILE_PLUGGED_IN, 0));
158911638f348ba45f9f417928e79b81186cef76c561Glen Kuhne            pw.println("mInIdleMode " + mInIdleMode);
159011638f348ba45f9f417928e79b81186cef76c561Glen Kuhne            pw.println("mScanPending " + mScanPending);
159111638f348ba45f9f417928e79b81186cef76c561Glen Kuhne            mWifiController.dump(fd, pw, args);
159211638f348ba45f9f417928e79b81186cef76c561Glen Kuhne            mSettingsStore.dump(fd, pw, args);
159311638f348ba45f9f417928e79b81186cef76c561Glen Kuhne            mTrafficPoller.dump(fd, pw, args);
159411638f348ba45f9f417928e79b81186cef76c561Glen Kuhne            pw.println();
159511638f348ba45f9f417928e79b81186cef76c561Glen Kuhne            pw.println("Locks held:");
15962a7c2bec5d307a9e52de429c6303b7df9de900dcRebecca Silberstein            mWifiLockManager.dump(pw);
15972a7c2bec5d307a9e52de429c6303b7df9de900dcRebecca Silberstein            pw.println();
159861312e14a088a9487d4db64f08285162476e870fPaul Stewart            mWifiMulticastLockManager.dump(pw);
159911638f348ba45f9f417928e79b81186cef76c561Glen Kuhne            pw.println();
160011638f348ba45f9f417928e79b81186cef76c561Glen Kuhne            mWifiStateMachine.dump(fd, pw, args);
16013c6e92f51984ba8999d5efe622b4a52dd18bdaffGlen Kuhne            pw.println();
16023c6e92f51984ba8999d5efe622b4a52dd18bdaffGlen Kuhne            mWifiStateMachine.updateWifiMetrics();
16033c6e92f51984ba8999d5efe622b4a52dd18bdaffGlen Kuhne            mWifiMetrics.dump(fd, pw, args);
160411638f348ba45f9f417928e79b81186cef76c561Glen Kuhne            pw.println();
16053204fb9682242a7b5a749489076c66d448c42577Roshan Pius            mWifiBackupRestore.dump(fd, pw, args);
16063204fb9682242a7b5a749489076c66d448c42577Roshan Pius            pw.println();
160711638f348ba45f9f417928e79b81186cef76c561Glen Kuhne        }
1608155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    }
1609155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
16108fcd1fe86ddc3056e1a26e69bc68f148ae0e9ddcChristopher Wiley    @Override
1611155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    public boolean acquireWifiLock(IBinder binder, int lockMode, String tag, WorkSource ws) {
1612f2fdf411925ad172b5e0b25b0c6df880256691d4Sohani Rao        mLog.trace("acquireWifiLock uid=% lockMode=%")
1613f2fdf411925ad172b5e0b25b0c6df880256691d4Sohani Rao                .c(Binder.getCallingUid())
1614f2fdf411925ad172b5e0b25b0c6df880256691d4Sohani Rao                .c(lockMode).flush();
16152a7c2bec5d307a9e52de429c6303b7df9de900dcRebecca Silberstein        if (mWifiLockManager.acquireWifiLock(lockMode, tag, binder, ws)) {
1616155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande            mWifiController.sendMessage(CMD_LOCKS_CHANGED);
1617155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande            return true;
1618155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        }
16192a7c2bec5d307a9e52de429c6303b7df9de900dcRebecca Silberstein        return false;
1620155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    }
1621155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
16228fcd1fe86ddc3056e1a26e69bc68f148ae0e9ddcChristopher Wiley    @Override
16232a7c2bec5d307a9e52de429c6303b7df9de900dcRebecca Silberstein    public void updateWifiLockWorkSource(IBinder binder, WorkSource ws) {
1624f2fdf411925ad172b5e0b25b0c6df880256691d4Sohani Rao        mLog.trace("updateWifiLockWorkSource uid=%").c(Binder.getCallingUid()).flush();
16252a7c2bec5d307a9e52de429c6303b7df9de900dcRebecca Silberstein        mWifiLockManager.updateWifiLockWorkSource(binder, ws);
1626155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    }
1627155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
16288fcd1fe86ddc3056e1a26e69bc68f148ae0e9ddcChristopher Wiley    @Override
16292a7c2bec5d307a9e52de429c6303b7df9de900dcRebecca Silberstein    public boolean releaseWifiLock(IBinder binder) {
1630f2fdf411925ad172b5e0b25b0c6df880256691d4Sohani Rao        mLog.trace("releaseWifiLock uid=%").c(Binder.getCallingUid()).flush();
16312a7c2bec5d307a9e52de429c6303b7df9de900dcRebecca Silberstein        if (mWifiLockManager.releaseWifiLock(binder)) {
16322a7c2bec5d307a9e52de429c6303b7df9de900dcRebecca Silberstein            mWifiController.sendMessage(CMD_LOCKS_CHANGED);
16332a7c2bec5d307a9e52de429c6303b7df9de900dcRebecca Silberstein            return true;
1634155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        }
16352a7c2bec5d307a9e52de429c6303b7df9de900dcRebecca Silberstein        return false;
1636155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    }
1637155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
16388fcd1fe86ddc3056e1a26e69bc68f148ae0e9ddcChristopher Wiley    @Override
1639155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    public void initializeMulticastFiltering() {
1640155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        enforceMulticastChangePermission();
1641f2fdf411925ad172b5e0b25b0c6df880256691d4Sohani Rao        mLog.trace("initializeMulticastFiltering uid=%").c(Binder.getCallingUid()).flush();
164261312e14a088a9487d4db64f08285162476e870fPaul Stewart        mWifiMulticastLockManager.initializeFiltering();
1643155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    }
1644155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
16458fcd1fe86ddc3056e1a26e69bc68f148ae0e9ddcChristopher Wiley    @Override
1646155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    public void acquireMulticastLock(IBinder binder, String tag) {
1647155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        enforceMulticastChangePermission();
1648f2fdf411925ad172b5e0b25b0c6df880256691d4Sohani Rao        mLog.trace("acquireMulticastLock uid=%").c(Binder.getCallingUid()).flush();
164961312e14a088a9487d4db64f08285162476e870fPaul Stewart        mWifiMulticastLockManager.acquireLock(binder, tag);
1650155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    }
1651155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
16528fcd1fe86ddc3056e1a26e69bc68f148ae0e9ddcChristopher Wiley    @Override
1653155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    public void releaseMulticastLock() {
1654155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        enforceMulticastChangePermission();
1655f2fdf411925ad172b5e0b25b0c6df880256691d4Sohani Rao        mLog.trace("releaseMulticastLock uid=%").c(Binder.getCallingUid()).flush();
165661312e14a088a9487d4db64f08285162476e870fPaul Stewart        mWifiMulticastLockManager.releaseLock();
1657155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    }
1658155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
16598fcd1fe86ddc3056e1a26e69bc68f148ae0e9ddcChristopher Wiley    @Override
1660155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    public boolean isMulticastEnabled() {
1661155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        enforceAccessPermission();
1662f2fdf411925ad172b5e0b25b0c6df880256691d4Sohani Rao        mLog.trace("isMulticastEnabled uid=%").c(Binder.getCallingUid()).flush();
166361312e14a088a9487d4db64f08285162476e870fPaul Stewart        return mWifiMulticastLockManager.isMulticastEnabled();
1664155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    }
16659ae6b45c038fa74a9e7285ef4834551dd93da332Yuhao Zheng
16668fcd1fe86ddc3056e1a26e69bc68f148ae0e9ddcChristopher Wiley    @Override
1667ed9938883ae2dade81c8be6cd6ceaef3febd5239vandwalle    public void enableVerboseLogging(int verbose) {
1668ed9938883ae2dade81c8be6cd6ceaef3febd5239vandwalle        enforceAccessPermission();
1669f2fdf411925ad172b5e0b25b0c6df880256691d4Sohani Rao        mLog.trace("enableVerboseLogging uid=% verbose=%")
1670f2fdf411925ad172b5e0b25b0c6df880256691d4Sohani Rao                .c(Binder.getCallingUid())
1671f2fdf411925ad172b5e0b25b0c6df880256691d4Sohani Rao                .c(verbose).flush();
167200ec8bf178ba4abee4ee8bbc6eb09c9fcd986a89Rebecca Silberstein        mFacade.setIntegerSetting(
167300ec8bf178ba4abee4ee8bbc6eb09c9fcd986a89Rebecca Silberstein                mContext, Settings.Global.WIFI_VERBOSE_LOGGING_ENABLED, verbose);
167400ec8bf178ba4abee4ee8bbc6eb09c9fcd986a89Rebecca Silberstein        enableVerboseLoggingInternal(verbose);
167500ec8bf178ba4abee4ee8bbc6eb09c9fcd986a89Rebecca Silberstein    }
167600ec8bf178ba4abee4ee8bbc6eb09c9fcd986a89Rebecca Silberstein
1677da918df16e03ee19be62343313d954027d3eb3abRebecca Silberstein    void enableVerboseLoggingInternal(int verbose) {
1678ed9938883ae2dade81c8be6cd6ceaef3febd5239vandwalle        mWifiStateMachine.enableVerboseLogging(verbose);
16792a7c2bec5d307a9e52de429c6303b7df9de900dcRebecca Silberstein        mWifiLockManager.enableVerboseLogging(verbose);
168061312e14a088a9487d4db64f08285162476e870fPaul Stewart        mWifiMulticastLockManager.enableVerboseLogging(verbose);
168175727af748e2b53baf365139ecfa7bf87a449d04Rebecca Silberstein        mWifiInjector.getWifiLastResortWatchdog().enableVerboseLogging(verbose);
168200ec8bf178ba4abee4ee8bbc6eb09c9fcd986a89Rebecca Silberstein        mWifiInjector.getWifiBackupRestore().enableVerboseLogging(verbose);
1683e33b3346b262507ef2361d50a89d16bef69d9a57Sohani Rao        LogcatLog.enableVerboseLogging(verbose);
1684ed9938883ae2dade81c8be6cd6ceaef3febd5239vandwalle    }
1685ed9938883ae2dade81c8be6cd6ceaef3febd5239vandwalle
16868fcd1fe86ddc3056e1a26e69bc68f148ae0e9ddcChristopher Wiley    @Override
1687ed9938883ae2dade81c8be6cd6ceaef3febd5239vandwalle    public int getVerboseLoggingLevel() {
1688ed9938883ae2dade81c8be6cd6ceaef3febd5239vandwalle        enforceAccessPermission();
1689f2fdf411925ad172b5e0b25b0c6df880256691d4Sohani Rao        mLog.trace("getVerboseLoggingLevel uid=%").c(Binder.getCallingUid()).flush();
169000ec8bf178ba4abee4ee8bbc6eb09c9fcd986a89Rebecca Silberstein        return mFacade.getIntegerSetting(
169100ec8bf178ba4abee4ee8bbc6eb09c9fcd986a89Rebecca Silberstein                mContext, Settings.Global.WIFI_VERBOSE_LOGGING_ENABLED, 0);
1692ed9938883ae2dade81c8be6cd6ceaef3febd5239vandwalle    }
1693c6f06c628ee3583b60ff31a7da442e0ac7b26d97vandwalle
16948fcd1fe86ddc3056e1a26e69bc68f148ae0e9ddcChristopher Wiley    @Override
1695c6f06c628ee3583b60ff31a7da442e0ac7b26d97vandwalle    public void enableAggressiveHandover(int enabled) {
1696c6f06c628ee3583b60ff31a7da442e0ac7b26d97vandwalle        enforceAccessPermission();
1697f2fdf411925ad172b5e0b25b0c6df880256691d4Sohani Rao        mLog.trace("enableAggressiveHandover uid=% enabled=%")
1698f2fdf411925ad172b5e0b25b0c6df880256691d4Sohani Rao            .c(Binder.getCallingUid())
1699f2fdf411925ad172b5e0b25b0c6df880256691d4Sohani Rao            .c(enabled)
1700f2fdf411925ad172b5e0b25b0c6df880256691d4Sohani Rao            .flush();
1701c6f06c628ee3583b60ff31a7da442e0ac7b26d97vandwalle        mWifiStateMachine.enableAggressiveHandover(enabled);
1702c6f06c628ee3583b60ff31a7da442e0ac7b26d97vandwalle    }
1703c6f06c628ee3583b60ff31a7da442e0ac7b26d97vandwalle
17048fcd1fe86ddc3056e1a26e69bc68f148ae0e9ddcChristopher Wiley    @Override
1705c6f06c628ee3583b60ff31a7da442e0ac7b26d97vandwalle    public int getAggressiveHandover() {
1706c6f06c628ee3583b60ff31a7da442e0ac7b26d97vandwalle        enforceAccessPermission();
1707f2fdf411925ad172b5e0b25b0c6df880256691d4Sohani Rao        mLog.trace("getAggressiveHandover uid=%").c(Binder.getCallingUid()).flush();
1708c6f06c628ee3583b60ff31a7da442e0ac7b26d97vandwalle        return mWifiStateMachine.getAggressiveHandover();
1709c6f06c628ee3583b60ff31a7da442e0ac7b26d97vandwalle    }
1710c6f06c628ee3583b60ff31a7da442e0ac7b26d97vandwalle
17118fcd1fe86ddc3056e1a26e69bc68f148ae0e9ddcChristopher Wiley    @Override
1712c6f06c628ee3583b60ff31a7da442e0ac7b26d97vandwalle    public void setAllowScansWithTraffic(int enabled) {
1713c6f06c628ee3583b60ff31a7da442e0ac7b26d97vandwalle        enforceAccessPermission();
1714f2fdf411925ad172b5e0b25b0c6df880256691d4Sohani Rao        mLog.trace("setAllowScansWithTraffic uid=% enabled=%")
1715f2fdf411925ad172b5e0b25b0c6df880256691d4Sohani Rao                .c(Binder.getCallingUid())
1716f2fdf411925ad172b5e0b25b0c6df880256691d4Sohani Rao                .c(enabled).flush();
1717c6f06c628ee3583b60ff31a7da442e0ac7b26d97vandwalle        mWifiStateMachine.setAllowScansWithTraffic(enabled);
1718c6f06c628ee3583b60ff31a7da442e0ac7b26d97vandwalle    }
1719c6f06c628ee3583b60ff31a7da442e0ac7b26d97vandwalle
17208fcd1fe86ddc3056e1a26e69bc68f148ae0e9ddcChristopher Wiley    @Override
1721c6f06c628ee3583b60ff31a7da442e0ac7b26d97vandwalle    public int getAllowScansWithTraffic() {
1722c6f06c628ee3583b60ff31a7da442e0ac7b26d97vandwalle        enforceAccessPermission();
1723f2fdf411925ad172b5e0b25b0c6df880256691d4Sohani Rao        mLog.trace("getAllowScansWithTraffic uid=%").c(Binder.getCallingUid()).flush();
1724c6f06c628ee3583b60ff31a7da442e0ac7b26d97vandwalle        return mWifiStateMachine.getAllowScansWithTraffic();
1725c6f06c628ee3583b60ff31a7da442e0ac7b26d97vandwalle    }
1726c298087de50ea56c31a4ade7ee1e83b313bb63c7vandwalle
17278fcd1fe86ddc3056e1a26e69bc68f148ae0e9ddcChristopher Wiley    @Override
172843eba5ba17a9af5d9a050a4ba5e12e93c92f722dSamuel Tan    public boolean setEnableAutoJoinWhenAssociated(boolean enabled) {
1729a8647b8cb29de22765062714cb265247234c3d32xinhe        enforceChangePermission();
1730f2fdf411925ad172b5e0b25b0c6df880256691d4Sohani Rao        mLog.trace("setEnableAutoJoinWhenAssociated uid=% enabled=%")
1731f2fdf411925ad172b5e0b25b0c6df880256691d4Sohani Rao                .c(Binder.getCallingUid())
1732f2fdf411925ad172b5e0b25b0c6df880256691d4Sohani Rao                .c(enabled).flush();
173343eba5ba17a9af5d9a050a4ba5e12e93c92f722dSamuel Tan        return mWifiStateMachine.setEnableAutoJoinWhenAssociated(enabled);
1734e2614ba1b774529ee68cc4ac189f7492cff89db3Pierre Vandwalle    }
1735e2614ba1b774529ee68cc4ac189f7492cff89db3Pierre Vandwalle
17368fcd1fe86ddc3056e1a26e69bc68f148ae0e9ddcChristopher Wiley    @Override
1737a8647b8cb29de22765062714cb265247234c3d32xinhe    public boolean getEnableAutoJoinWhenAssociated() {
1738e2614ba1b774529ee68cc4ac189f7492cff89db3Pierre Vandwalle        enforceAccessPermission();
1739f2fdf411925ad172b5e0b25b0c6df880256691d4Sohani Rao        mLog.trace("getEnableAutoJoinWhenAssociated uid=%").c(Binder.getCallingUid()).flush();
1740a8647b8cb29de22765062714cb265247234c3d32xinhe        return mWifiStateMachine.getEnableAutoJoinWhenAssociated();
1741e2614ba1b774529ee68cc4ac189f7492cff89db3Pierre Vandwalle    }
1742b0b0cc202b7d7aaad7b3f69d73e9b58ea2968b05Pierre Vandwalle
1743c298087de50ea56c31a4ade7ee1e83b313bb63c7vandwalle    /* Return the Wifi Connection statistics object */
17448fcd1fe86ddc3056e1a26e69bc68f148ae0e9ddcChristopher Wiley    @Override
1745c298087de50ea56c31a4ade7ee1e83b313bb63c7vandwalle    public WifiConnectionStatistics getConnectionStatistics() {
1746c298087de50ea56c31a4ade7ee1e83b313bb63c7vandwalle        enforceAccessPermission();
1747c298087de50ea56c31a4ade7ee1e83b313bb63c7vandwalle        enforceReadCredentialPermission();
1748f2fdf411925ad172b5e0b25b0c6df880256691d4Sohani Rao        mLog.trace("getConnectionStatistics uid=%").c(Binder.getCallingUid()).flush();
1749c298087de50ea56c31a4ade7ee1e83b313bb63c7vandwalle        if (mWifiStateMachineChannel != null) {
1750c298087de50ea56c31a4ade7ee1e83b313bb63c7vandwalle            return mWifiStateMachine.syncGetConnectionStatistics(mWifiStateMachineChannel);
1751c298087de50ea56c31a4ade7ee1e83b313bb63c7vandwalle        } else {
1752c298087de50ea56c31a4ade7ee1e83b313bb63c7vandwalle            Slog.e(TAG, "mWifiStateMachineChannel is not initialized");
1753c298087de50ea56c31a4ade7ee1e83b313bb63c7vandwalle            return null;
1754c298087de50ea56c31a4ade7ee1e83b313bb63c7vandwalle        }
1755c298087de50ea56c31a4ade7ee1e83b313bb63c7vandwalle    }
1756b8671cfafc2830ebddeafcfb2d91f2b39b19019bStuart Scott
17578fcd1fe86ddc3056e1a26e69bc68f148ae0e9ddcChristopher Wiley    @Override
1758b8671cfafc2830ebddeafcfb2d91f2b39b19019bStuart Scott    public void factoryReset() {
1759b8671cfafc2830ebddeafcfb2d91f2b39b19019bStuart Scott        enforceConnectivityInternalPermission();
1760f2fdf411925ad172b5e0b25b0c6df880256691d4Sohani Rao        mLog.trace("factoryReset uid=%").c(Binder.getCallingUid()).flush();
1761f9d866f1b4e93593b3411886fd71e729f4a5a205Stuart Scott        if (mUserManager.hasUserRestriction(UserManager.DISALLOW_NETWORK_RESET)) {
1762f9d866f1b4e93593b3411886fd71e729f4a5a205Stuart Scott            return;
1763f9d866f1b4e93593b3411886fd71e729f4a5a205Stuart Scott        }
1764b8671cfafc2830ebddeafcfb2d91f2b39b19019bStuart Scott
1765f9d866f1b4e93593b3411886fd71e729f4a5a205Stuart Scott        if (!mUserManager.hasUserRestriction(UserManager.DISALLOW_CONFIG_TETHERING)) {
1766f9d866f1b4e93593b3411886fd71e729f4a5a205Stuart Scott            // Turn mobile hotspot off
1767f9d866f1b4e93593b3411886fd71e729f4a5a205Stuart Scott            setWifiApEnabled(null, false);
1768f9d866f1b4e93593b3411886fd71e729f4a5a205Stuart Scott        }
1769f9d866f1b4e93593b3411886fd71e729f4a5a205Stuart Scott
1770f9d866f1b4e93593b3411886fd71e729f4a5a205Stuart Scott        if (!mUserManager.hasUserRestriction(UserManager.DISALLOW_CONFIG_WIFI)) {
1771f9d866f1b4e93593b3411886fd71e729f4a5a205Stuart Scott            // Enable wifi
1772dcf967aa402a4ab1a79c727aea934b8013c1fa6aSvetoslav Ganov            try {
1773dcf967aa402a4ab1a79c727aea934b8013c1fa6aSvetoslav Ganov                setWifiEnabled(mContext.getOpPackageName(), true);
1774dcf967aa402a4ab1a79c727aea934b8013c1fa6aSvetoslav Ganov            } catch (RemoteException e) {
1775dcf967aa402a4ab1a79c727aea934b8013c1fa6aSvetoslav Ganov                /* ignore - local call */
1776dcf967aa402a4ab1a79c727aea934b8013c1fa6aSvetoslav Ganov            }
1777f9d866f1b4e93593b3411886fd71e729f4a5a205Stuart Scott            // Delete all Wifi SSIDs
1778f384821f8979152ba83ec70614a59094c838c8c6zhangshuxiao            if (mWifiStateMachineChannel != null) {
1779f384821f8979152ba83ec70614a59094c838c8c6zhangshuxiao                List<WifiConfiguration> networks = mWifiStateMachine.syncGetConfiguredNetworks(
1780f384821f8979152ba83ec70614a59094c838c8c6zhangshuxiao                        Binder.getCallingUid(), mWifiStateMachineChannel);
1781f384821f8979152ba83ec70614a59094c838c8c6zhangshuxiao                if (networks != null) {
1782f384821f8979152ba83ec70614a59094c838c8c6zhangshuxiao                    for (WifiConfiguration config : networks) {
1783f384821f8979152ba83ec70614a59094c838c8c6zhangshuxiao                        removeNetwork(config.networkId);
1784f384821f8979152ba83ec70614a59094c838c8c6zhangshuxiao                    }
1785f384821f8979152ba83ec70614a59094c838c8c6zhangshuxiao                    saveConfiguration();
1786f9d866f1b4e93593b3411886fd71e729f4a5a205Stuart Scott                }
1787b8671cfafc2830ebddeafcfb2d91f2b39b19019bStuart Scott            }
1788b8671cfafc2830ebddeafcfb2d91f2b39b19019bStuart Scott        }
1789b8671cfafc2830ebddeafcfb2d91f2b39b19019bStuart Scott    }
17908be2ac8cec40a2000715849b627642bdc87c10b3Shirish Kalele
17914d381bc39f5263effdae73ec99065eb299b806caVinit Deshpande    /* private methods */
179231891afce40b903ada9b24ec12e3648ae6aa27b2Jan Nordqvist    static boolean logAndReturnFalse(String s) {
17934d381bc39f5263effdae73ec99065eb299b806caVinit Deshpande        Log.d(TAG, s);
17944d381bc39f5263effdae73ec99065eb299b806caVinit Deshpande        return false;
17954d381bc39f5263effdae73ec99065eb299b806caVinit Deshpande    }
17964d381bc39f5263effdae73ec99065eb299b806caVinit Deshpande
179731891afce40b903ada9b24ec12e3648ae6aa27b2Jan Nordqvist    public static boolean isValid(WifiConfiguration config) {
179831891afce40b903ada9b24ec12e3648ae6aa27b2Jan Nordqvist        String validity = checkValidity(config);
179931891afce40b903ada9b24ec12e3648ae6aa27b2Jan Nordqvist        return validity == null || logAndReturnFalse(validity);
180031891afce40b903ada9b24ec12e3648ae6aa27b2Jan Nordqvist    }
18014d381bc39f5263effdae73ec99065eb299b806caVinit Deshpande
18024aef3283bf78e233a11fa34af2c7c38362879625Jan Nordqvist    public static boolean isValidPasspoint(WifiConfiguration config) {
18034aef3283bf78e233a11fa34af2c7c38362879625Jan Nordqvist        String validity = checkPasspointValidity(config);
18044aef3283bf78e233a11fa34af2c7c38362879625Jan Nordqvist        return validity == null || logAndReturnFalse(validity);
18054aef3283bf78e233a11fa34af2c7c38362879625Jan Nordqvist    }
18064aef3283bf78e233a11fa34af2c7c38362879625Jan Nordqvist
180731891afce40b903ada9b24ec12e3648ae6aa27b2Jan Nordqvist    public static String checkValidity(WifiConfiguration config) {
180831891afce40b903ada9b24ec12e3648ae6aa27b2Jan Nordqvist        if (config.allowedKeyManagement == null)
180931891afce40b903ada9b24ec12e3648ae6aa27b2Jan Nordqvist            return "allowed kmgmt";
18104d381bc39f5263effdae73ec99065eb299b806caVinit Deshpande
18114d381bc39f5263effdae73ec99065eb299b806caVinit Deshpande        if (config.allowedKeyManagement.cardinality() > 1) {
18124d381bc39f5263effdae73ec99065eb299b806caVinit Deshpande            if (config.allowedKeyManagement.cardinality() != 2) {
181331891afce40b903ada9b24ec12e3648ae6aa27b2Jan Nordqvist                return "cardinality != 2";
18144d381bc39f5263effdae73ec99065eb299b806caVinit Deshpande            }
18154d381bc39f5263effdae73ec99065eb299b806caVinit Deshpande            if (!config.allowedKeyManagement.get(WifiConfiguration.KeyMgmt.WPA_EAP)) {
181631891afce40b903ada9b24ec12e3648ae6aa27b2Jan Nordqvist                return "not WPA_EAP";
18174d381bc39f5263effdae73ec99065eb299b806caVinit Deshpande            }
18184d381bc39f5263effdae73ec99065eb299b806caVinit Deshpande            if ((!config.allowedKeyManagement.get(WifiConfiguration.KeyMgmt.IEEE8021X))
18194d381bc39f5263effdae73ec99065eb299b806caVinit Deshpande                    && (!config.allowedKeyManagement.get(WifiConfiguration.KeyMgmt.WPA_PSK))) {
182031891afce40b903ada9b24ec12e3648ae6aa27b2Jan Nordqvist                return "not PSK or 8021X";
18214d381bc39f5263effdae73ec99065eb299b806caVinit Deshpande            }
18224d381bc39f5263effdae73ec99065eb299b806caVinit Deshpande        }
1823653cd53f0906a90fbf5b1d9d0bd30917043d1bfcGlen Kuhne        if (config.getIpAssignment() == IpConfiguration.IpAssignment.STATIC) {
1824653cd53f0906a90fbf5b1d9d0bd30917043d1bfcGlen Kuhne            StaticIpConfiguration staticIpConf = config.getStaticIpConfiguration();
1825653cd53f0906a90fbf5b1d9d0bd30917043d1bfcGlen Kuhne            if (staticIpConf == null) {
1826653cd53f0906a90fbf5b1d9d0bd30917043d1bfcGlen Kuhne                return "null StaticIpConfiguration";
1827653cd53f0906a90fbf5b1d9d0bd30917043d1bfcGlen Kuhne            }
1828653cd53f0906a90fbf5b1d9d0bd30917043d1bfcGlen Kuhne            if (staticIpConf.ipAddress == null) {
1829653cd53f0906a90fbf5b1d9d0bd30917043d1bfcGlen Kuhne                return "null static ip Address";
1830653cd53f0906a90fbf5b1d9d0bd30917043d1bfcGlen Kuhne            }
1831653cd53f0906a90fbf5b1d9d0bd30917043d1bfcGlen Kuhne        }
18324aef3283bf78e233a11fa34af2c7c38362879625Jan Nordqvist        return null;
18334aef3283bf78e233a11fa34af2c7c38362879625Jan Nordqvist    }
18344d381bc39f5263effdae73ec99065eb299b806caVinit Deshpande
18354aef3283bf78e233a11fa34af2c7c38362879625Jan Nordqvist    public static String checkPasspointValidity(WifiConfiguration config) {
18364d381bc39f5263effdae73ec99065eb299b806caVinit Deshpande        if (!TextUtils.isEmpty(config.FQDN)) {
18374d381bc39f5263effdae73ec99065eb299b806caVinit Deshpande            /* this is passpoint configuration; it must not have an SSID */
18384d381bc39f5263effdae73ec99065eb299b806caVinit Deshpande            if (!TextUtils.isEmpty(config.SSID)) {
183931891afce40b903ada9b24ec12e3648ae6aa27b2Jan Nordqvist                return "SSID not expected for Passpoint: '" + config.SSID +
184031891afce40b903ada9b24ec12e3648ae6aa27b2Jan Nordqvist                        "' FQDN " + toHexString(config.FQDN);
18414d381bc39f5263effdae73ec99065eb299b806caVinit Deshpande            }
18424d381bc39f5263effdae73ec99065eb299b806caVinit Deshpande            /* this is passpoint configuration; it must have a providerFriendlyName */
18434d381bc39f5263effdae73ec99065eb299b806caVinit Deshpande            if (TextUtils.isEmpty(config.providerFriendlyName)) {
184431891afce40b903ada9b24ec12e3648ae6aa27b2Jan Nordqvist                return "no provider friendly name";
18454d381bc39f5263effdae73ec99065eb299b806caVinit Deshpande            }
184607f11f6f2ee7ec17cb08180035dfb5002aaaf5dfJan Nordqvist            WifiEnterpriseConfig enterpriseConfig = config.enterpriseConfig;
18474d381bc39f5263effdae73ec99065eb299b806caVinit Deshpande            /* this is passpoint configuration; it must have enterprise config */
184807f11f6f2ee7ec17cb08180035dfb5002aaaf5dfJan Nordqvist            if (enterpriseConfig == null
184907f11f6f2ee7ec17cb08180035dfb5002aaaf5dfJan Nordqvist                    || enterpriseConfig.getEapMethod() == WifiEnterpriseConfig.Eap.NONE ) {
185031891afce40b903ada9b24ec12e3648ae6aa27b2Jan Nordqvist                return "no enterprise config";
18514d381bc39f5263effdae73ec99065eb299b806caVinit Deshpande            }
185207f11f6f2ee7ec17cb08180035dfb5002aaaf5dfJan Nordqvist            if ((enterpriseConfig.getEapMethod() == WifiEnterpriseConfig.Eap.TLS ||
185307f11f6f2ee7ec17cb08180035dfb5002aaaf5dfJan Nordqvist                    enterpriseConfig.getEapMethod() == WifiEnterpriseConfig.Eap.TTLS ||
185407f11f6f2ee7ec17cb08180035dfb5002aaaf5dfJan Nordqvist                    enterpriseConfig.getEapMethod() == WifiEnterpriseConfig.Eap.PEAP) &&
185507f11f6f2ee7ec17cb08180035dfb5002aaaf5dfJan Nordqvist                    enterpriseConfig.getCaCertificate() == null) {
185631891afce40b903ada9b24ec12e3648ae6aa27b2Jan Nordqvist                return "no CA certificate";
18574d381bc39f5263effdae73ec99065eb299b806caVinit Deshpande            }
18584d381bc39f5263effdae73ec99065eb299b806caVinit Deshpande        }
185931891afce40b903ada9b24ec12e3648ae6aa27b2Jan Nordqvist        return null;
1860cd0af1c9ddcac881f49c815fb9da45433eb50736Vinit Deshpande    }
1861b8671cfafc2830ebddeafcfb2d91f2b39b19019bStuart Scott
18628fcd1fe86ddc3056e1a26e69bc68f148ae0e9ddcChristopher Wiley    @Override
18639846078598c3468f8813dbfa58238a1846bd81efSanket Padawe    public Network getCurrentNetwork() {
18649846078598c3468f8813dbfa58238a1846bd81efSanket Padawe        enforceAccessPermission();
1865f2fdf411925ad172b5e0b25b0c6df880256691d4Sohani Rao        mLog.trace("getCurrentNetwork uid=%").c(Binder.getCallingUid()).flush();
18669846078598c3468f8813dbfa58238a1846bd81efSanket Padawe        return mWifiStateMachine.getCurrentNetwork();
18679846078598c3468f8813dbfa58238a1846bd81efSanket Padawe    }
18689846078598c3468f8813dbfa58238a1846bd81efSanket Padawe
186931891afce40b903ada9b24ec12e3648ae6aa27b2Jan Nordqvist    public static String toHexString(String s) {
187031891afce40b903ada9b24ec12e3648ae6aa27b2Jan Nordqvist        if (s == null) {
187131891afce40b903ada9b24ec12e3648ae6aa27b2Jan Nordqvist            return "null";
187231891afce40b903ada9b24ec12e3648ae6aa27b2Jan Nordqvist        }
187331891afce40b903ada9b24ec12e3648ae6aa27b2Jan Nordqvist        StringBuilder sb = new StringBuilder();
187431891afce40b903ada9b24ec12e3648ae6aa27b2Jan Nordqvist        sb.append('\'').append(s).append('\'');
187531891afce40b903ada9b24ec12e3648ae6aa27b2Jan Nordqvist        for (int n = 0; n < s.length(); n++) {
187631891afce40b903ada9b24ec12e3648ae6aa27b2Jan Nordqvist            sb.append(String.format(" %02x", s.charAt(n) & 0xffff));
187731891afce40b903ada9b24ec12e3648ae6aa27b2Jan Nordqvist        }
187831891afce40b903ada9b24ec12e3648ae6aa27b2Jan Nordqvist        return sb.toString();
18794d381bc39f5263effdae73ec99065eb299b806caVinit Deshpande    }
1880cd0af1c9ddcac881f49c815fb9da45433eb50736Vinit Deshpande
18812bfc67c9893c0a525b224d68dd73a74212b0c29fArthur Hsu    public void hideCertFromUnaffiliatedUsers(String alias) {
18822bfc67c9893c0a525b224d68dd73a74212b0c29fArthur Hsu        mCertManager.hideCertFromUnaffiliatedUsers(alias);
18832bfc67c9893c0a525b224d68dd73a74212b0c29fArthur Hsu    }
18842bfc67c9893c0a525b224d68dd73a74212b0c29fArthur Hsu
18852bfc67c9893c0a525b224d68dd73a74212b0c29fArthur Hsu    public String[] listClientCertsForCurrentUser() {
18862bfc67c9893c0a525b224d68dd73a74212b0c29fArthur Hsu        return mCertManager.listClientCertsForCurrentUser();
18872bfc67c9893c0a525b224d68dd73a74212b0c29fArthur Hsu    }
18882bfc67c9893c0a525b224d68dd73a74212b0c29fArthur Hsu
1889c065315c23f78ec462a56aec1aaa4e2a34549b8dFyodor Kupolov    /**
1890466158a6669d51541ce6c5c4e04a71dad36cdb4eRandy Pan     * Enable/disable WifiConnectivityManager at runtime
1891466158a6669d51541ce6c5c4e04a71dad36cdb4eRandy Pan     *
1892466158a6669d51541ce6c5c4e04a71dad36cdb4eRandy Pan     * @param enabled true-enable; false-disable
1893466158a6669d51541ce6c5c4e04a71dad36cdb4eRandy Pan     */
18948fcd1fe86ddc3056e1a26e69bc68f148ae0e9ddcChristopher Wiley    @Override
1895466158a6669d51541ce6c5c4e04a71dad36cdb4eRandy Pan    public void enableWifiConnectivityManager(boolean enabled) {
1896466158a6669d51541ce6c5c4e04a71dad36cdb4eRandy Pan        enforceConnectivityInternalPermission();
1897f2fdf411925ad172b5e0b25b0c6df880256691d4Sohani Rao        mLog.trace("enableWifiConnectivityManager uid=% enabled=%")
1898f2fdf411925ad172b5e0b25b0c6df880256691d4Sohani Rao            .c(Binder.getCallingUid())
1899f2fdf411925ad172b5e0b25b0c6df880256691d4Sohani Rao            .c(enabled).flush();
1900466158a6669d51541ce6c5c4e04a71dad36cdb4eRandy Pan        mWifiStateMachine.enableWifiConnectivityManager(enabled);
1901466158a6669d51541ce6c5c4e04a71dad36cdb4eRandy Pan    }
1902ff9686aec5b6ee21e01556899fb13ab55915f3c0Roshan Pius
1903ff9686aec5b6ee21e01556899fb13ab55915f3c0Roshan Pius    /**
1904ff9686aec5b6ee21e01556899fb13ab55915f3c0Roshan Pius     * Retrieve the data to be backed to save the current state.
19053031903d4a68659350994571525fc86a47c02dd0Roshan Pius     *
1906ff9686aec5b6ee21e01556899fb13ab55915f3c0Roshan Pius     * @return  Raw byte stream of the data to be backed up.
1907ff9686aec5b6ee21e01556899fb13ab55915f3c0Roshan Pius     */
1908ff9686aec5b6ee21e01556899fb13ab55915f3c0Roshan Pius    @Override
1909ff9686aec5b6ee21e01556899fb13ab55915f3c0Roshan Pius    public byte[] retrieveBackupData() {
1910ff9686aec5b6ee21e01556899fb13ab55915f3c0Roshan Pius        enforceReadCredentialPermission();
1911ff9686aec5b6ee21e01556899fb13ab55915f3c0Roshan Pius        enforceAccessPermission();
1912f2fdf411925ad172b5e0b25b0c6df880256691d4Sohani Rao        mLog.trace("retrieveBackupData uid=%").c(Binder.getCallingUid()).flush();
1913ff9686aec5b6ee21e01556899fb13ab55915f3c0Roshan Pius        if (mWifiStateMachineChannel == null) {
1914ff9686aec5b6ee21e01556899fb13ab55915f3c0Roshan Pius            Slog.e(TAG, "mWifiStateMachineChannel is not initialized");
1915ff9686aec5b6ee21e01556899fb13ab55915f3c0Roshan Pius            return null;
1916ff9686aec5b6ee21e01556899fb13ab55915f3c0Roshan Pius        }
1917ff9686aec5b6ee21e01556899fb13ab55915f3c0Roshan Pius
19183204fb9682242a7b5a749489076c66d448c42577Roshan Pius        Slog.d(TAG, "Retrieving backup data");
1919ff9686aec5b6ee21e01556899fb13ab55915f3c0Roshan Pius        List<WifiConfiguration> wifiConfigurations =
1920ff9686aec5b6ee21e01556899fb13ab55915f3c0Roshan Pius                mWifiStateMachine.syncGetPrivilegedConfiguredNetwork(mWifiStateMachineChannel);
19213204fb9682242a7b5a749489076c66d448c42577Roshan Pius        byte[] backupData =
19223204fb9682242a7b5a749489076c66d448c42577Roshan Pius                mWifiBackupRestore.retrieveBackupDataFromConfigurations(wifiConfigurations);
19233204fb9682242a7b5a749489076c66d448c42577Roshan Pius        Slog.d(TAG, "Retrieved backup data");
19243204fb9682242a7b5a749489076c66d448c42577Roshan Pius        return backupData;
1925ff9686aec5b6ee21e01556899fb13ab55915f3c0Roshan Pius    }
1926ff9686aec5b6ee21e01556899fb13ab55915f3c0Roshan Pius
1927ff9686aec5b6ee21e01556899fb13ab55915f3c0Roshan Pius    /**
19283031903d4a68659350994571525fc86a47c02dd0Roshan Pius     * Helper method to restore networks retrieved from backup data.
19293031903d4a68659350994571525fc86a47c02dd0Roshan Pius     *
19303031903d4a68659350994571525fc86a47c02dd0Roshan Pius     * @param configurations list of WifiConfiguration objects parsed from the backup data.
19313031903d4a68659350994571525fc86a47c02dd0Roshan Pius     */
19323031903d4a68659350994571525fc86a47c02dd0Roshan Pius    private void restoreNetworks(List<WifiConfiguration> configurations) {
19333031903d4a68659350994571525fc86a47c02dd0Roshan Pius        if (configurations == null) {
19343031903d4a68659350994571525fc86a47c02dd0Roshan Pius            Slog.e(TAG, "Backup data parse failed");
19353031903d4a68659350994571525fc86a47c02dd0Roshan Pius            return;
19363031903d4a68659350994571525fc86a47c02dd0Roshan Pius        }
19373031903d4a68659350994571525fc86a47c02dd0Roshan Pius        for (WifiConfiguration configuration : configurations) {
19383031903d4a68659350994571525fc86a47c02dd0Roshan Pius            int networkId = mWifiStateMachine.syncAddOrUpdateNetwork(
19393031903d4a68659350994571525fc86a47c02dd0Roshan Pius                    mWifiStateMachineChannel, configuration);
19403031903d4a68659350994571525fc86a47c02dd0Roshan Pius            if (networkId == WifiConfiguration.INVALID_NETWORK_ID) {
19413031903d4a68659350994571525fc86a47c02dd0Roshan Pius                Slog.e(TAG, "Restore network failed: " + configuration.configKey());
19423031903d4a68659350994571525fc86a47c02dd0Roshan Pius                continue;
19433031903d4a68659350994571525fc86a47c02dd0Roshan Pius            }
19443031903d4a68659350994571525fc86a47c02dd0Roshan Pius            // Enable all networks restored.
19453031903d4a68659350994571525fc86a47c02dd0Roshan Pius            mWifiStateMachine.syncEnableNetwork(mWifiStateMachineChannel, networkId, false);
19463031903d4a68659350994571525fc86a47c02dd0Roshan Pius        }
19473031903d4a68659350994571525fc86a47c02dd0Roshan Pius    }
19483031903d4a68659350994571525fc86a47c02dd0Roshan Pius
19493031903d4a68659350994571525fc86a47c02dd0Roshan Pius    /**
1950ff9686aec5b6ee21e01556899fb13ab55915f3c0Roshan Pius     * Restore state from the backed up data.
1951e3831b70d4a8a967fe8df5496d542a432692c434Roshan Pius     *
1952ff9686aec5b6ee21e01556899fb13ab55915f3c0Roshan Pius     * @param data Raw byte stream of the backed up data.
1953ff9686aec5b6ee21e01556899fb13ab55915f3c0Roshan Pius     */
1954ff9686aec5b6ee21e01556899fb13ab55915f3c0Roshan Pius    @Override
1955ff9686aec5b6ee21e01556899fb13ab55915f3c0Roshan Pius    public void restoreBackupData(byte[] data) {
1956ff9686aec5b6ee21e01556899fb13ab55915f3c0Roshan Pius        enforceChangePermission();
1957f2fdf411925ad172b5e0b25b0c6df880256691d4Sohani Rao        mLog.trace("restoreBackupData uid=%").c(Binder.getCallingUid()).flush();
1958ff9686aec5b6ee21e01556899fb13ab55915f3c0Roshan Pius        if (mWifiStateMachineChannel == null) {
1959ff9686aec5b6ee21e01556899fb13ab55915f3c0Roshan Pius            Slog.e(TAG, "mWifiStateMachineChannel is not initialized");
1960ff9686aec5b6ee21e01556899fb13ab55915f3c0Roshan Pius            return;
1961ff9686aec5b6ee21e01556899fb13ab55915f3c0Roshan Pius        }
1962ff9686aec5b6ee21e01556899fb13ab55915f3c0Roshan Pius
19633204fb9682242a7b5a749489076c66d448c42577Roshan Pius        Slog.d(TAG, "Restoring backup data");
1964ff9686aec5b6ee21e01556899fb13ab55915f3c0Roshan Pius        List<WifiConfiguration> wifiConfigurations =
1965ff9686aec5b6ee21e01556899fb13ab55915f3c0Roshan Pius                mWifiBackupRestore.retrieveConfigurationsFromBackupData(data);
19663031903d4a68659350994571525fc86a47c02dd0Roshan Pius        restoreNetworks(wifiConfigurations);
19673204fb9682242a7b5a749489076c66d448c42577Roshan Pius        Slog.d(TAG, "Restored backup data");
1968ff9686aec5b6ee21e01556899fb13ab55915f3c0Roshan Pius    }
1969e3831b70d4a8a967fe8df5496d542a432692c434Roshan Pius
1970e3831b70d4a8a967fe8df5496d542a432692c434Roshan Pius    /**
1971e3831b70d4a8a967fe8df5496d542a432692c434Roshan Pius     * Restore state from the older supplicant back up data.
1972e3831b70d4a8a967fe8df5496d542a432692c434Roshan Pius     * The old backup data was essentially a backup of wpa_supplicant.conf & ipconfig.txt file.
1973e3831b70d4a8a967fe8df5496d542a432692c434Roshan Pius     *
1974e3831b70d4a8a967fe8df5496d542a432692c434Roshan Pius     * @param supplicantData Raw byte stream of wpa_supplicant.conf
1975e3831b70d4a8a967fe8df5496d542a432692c434Roshan Pius     * @param ipConfigData Raw byte stream of ipconfig.txt
1976e3831b70d4a8a967fe8df5496d542a432692c434Roshan Pius     */
1977e3831b70d4a8a967fe8df5496d542a432692c434Roshan Pius    public void restoreSupplicantBackupData(byte[] supplicantData, byte[] ipConfigData) {
1978e3831b70d4a8a967fe8df5496d542a432692c434Roshan Pius        enforceChangePermission();
1979f2fdf411925ad172b5e0b25b0c6df880256691d4Sohani Rao        mLog.trace("restoreSupplicantBackupData uid=%").c(Binder.getCallingUid()).flush();
1980e3831b70d4a8a967fe8df5496d542a432692c434Roshan Pius        if (mWifiStateMachineChannel == null) {
1981e3831b70d4a8a967fe8df5496d542a432692c434Roshan Pius            Slog.e(TAG, "mWifiStateMachineChannel is not initialized");
1982e3831b70d4a8a967fe8df5496d542a432692c434Roshan Pius            return;
1983e3831b70d4a8a967fe8df5496d542a432692c434Roshan Pius        }
1984e3831b70d4a8a967fe8df5496d542a432692c434Roshan Pius
19853204fb9682242a7b5a749489076c66d448c42577Roshan Pius        Slog.d(TAG, "Restoring supplicant backup data");
1986e3831b70d4a8a967fe8df5496d542a432692c434Roshan Pius        List<WifiConfiguration> wifiConfigurations =
1987e3831b70d4a8a967fe8df5496d542a432692c434Roshan Pius                mWifiBackupRestore.retrieveConfigurationsFromSupplicantBackupData(
1988e3831b70d4a8a967fe8df5496d542a432692c434Roshan Pius                        supplicantData, ipConfigData);
19893031903d4a68659350994571525fc86a47c02dd0Roshan Pius        restoreNetworks(wifiConfigurations);
19903204fb9682242a7b5a749489076c66d448c42577Roshan Pius        Slog.d(TAG, "Restored supplicant backup data");
1991e3831b70d4a8a967fe8df5496d542a432692c434Roshan Pius    }
1992155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande}
1993