WifiServiceImpl.java revision e33b3346b262507ef2361d50a89d16bef69d9a57
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;
380469548a809654861f74dc42a3409f29ae750550Svetoslav Ganovimport android.content.pm.ApplicationInfo;
39155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpandeimport android.content.pm.PackageManager;
40155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpandeimport android.database.ContentObserver;
4134f173a54307c39ffe5dfab52e7ef8166a00539eRobert Greenwaltimport android.net.ConnectivityManager;
42155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpandeimport android.net.DhcpInfo;
43155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpandeimport android.net.DhcpResults;
449846078598c3468f8813dbfa58238a1846bd81efSanket Padaweimport android.net.Network;
45155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpandeimport android.net.NetworkUtils;
46e487a4648dd41881e754f1224aaedead78a0777dSky Faberimport android.net.Uri;
4784d962ec8f487f824214744498bba505a6db0c59Randy Panimport android.net.ip.IpManager;
483ecf5a032e94b6538a56f94a5b33e50cbc464007Jaewan Kimimport android.net.wifi.IWifiManager;
498be2ac8cec40a2000715849b627642bdc87c10b3Shirish Kaleleimport android.net.wifi.ScanResult;
508be2ac8cec40a2000715849b627642bdc87c10b3Shirish Kaleleimport android.net.wifi.ScanSettings;
518be2ac8cec40a2000715849b627642bdc87c10b3Shirish Kaleleimport android.net.wifi.WifiActivityEnergyInfo;
528be2ac8cec40a2000715849b627642bdc87c10b3Shirish Kaleleimport android.net.wifi.WifiConfiguration;
538be2ac8cec40a2000715849b627642bdc87c10b3Shirish Kaleleimport android.net.wifi.WifiConnectionStatistics;
548be2ac8cec40a2000715849b627642bdc87c10b3Shirish Kaleleimport android.net.wifi.WifiEnterpriseConfig;
558be2ac8cec40a2000715849b627642bdc87c10b3Shirish Kaleleimport android.net.wifi.WifiInfo;
568be2ac8cec40a2000715849b627642bdc87c10b3Shirish Kaleleimport android.net.wifi.WifiLinkLayerStats;
578be2ac8cec40a2000715849b627642bdc87c10b3Shirish Kaleleimport android.net.wifi.WifiManager;
586c11cd5d6ff04afdf1bbbc99a850025d46c7bdc9Rebecca Silbersteinimport android.net.wifi.WifiScanner;
593ebf0b17fd1cdd3a0ef22d9e199c1686c01dcc9fPeter Qiuimport android.net.wifi.hotspot2.PasspointConfiguration;
603ecf5a032e94b6538a56f94a5b33e50cbc464007Jaewan Kimimport android.os.AsyncTask;
61c48fd721a9a3273f0faa927d93caad4c9832af7dAdam Lesinskiimport android.os.BatteryStats;
62155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpandeimport android.os.Binder;
63e660aa2b9c9ca3a5c7903c37571ca7b91feb4ccfFyodor Kupolovimport android.os.Build;
64c48fd721a9a3273f0faa927d93caad4c9832af7dAdam Lesinskiimport android.os.Bundle;
65155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpandeimport android.os.Handler;
66155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpandeimport android.os.HandlerThread;
67155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpandeimport android.os.IBinder;
6840abf54c81c5624641543d86e1d7ab21ebe30175Paul Stewartimport android.os.Looper;
69155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpandeimport android.os.Message;
703ecf5a032e94b6538a56f94a5b33e50cbc464007Jaewan Kimimport android.os.Messenger;
7103b70b37a8a20cb92d860b5015f6762dcb5e75ccDianne Hackbornimport android.os.PowerManager;
720469548a809654861f74dc42a3409f29ae750550Svetoslav Ganovimport android.os.Process;
73155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpandeimport android.os.RemoteException;
74c48fd721a9a3273f0faa927d93caad4c9832af7dAdam Lesinskiimport android.os.ResultReceiver;
75186f7513a080eb1a419b6d83286d62961168a5feAdam Lesinskiimport android.os.SystemClock;
76155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpandeimport android.os.UserHandle;
77fd5470391e5342daa38d00b68ccbccfeacbe6d33Alexandra Gherghinaimport android.os.UserManager;
78155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpandeimport android.os.WorkSource;
79155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpandeimport android.provider.Settings;
804d381bc39f5263effdae73ec99065eb299b806caVinit Deshpandeimport android.text.TextUtils;
81992ae00f25a9cc22cf5db3261bd7e72927069cf7vandwalleimport android.util.Log;
82155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpandeimport android.util.Slog;
83155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
8498e43e5f48a4c87343fc311feda4fa5489948822Vinit Deshpandeimport com.android.internal.telephony.IccCardConstants;
85090aa74f2d36010abba6563a39c87ab7ba8a7473Vinit Deshpandeimport com.android.internal.telephony.PhoneConstants;
868be2ac8cec40a2000715849b627642bdc87c10b3Shirish Kaleleimport com.android.internal.telephony.TelephonyIntents;
878be2ac8cec40a2000715849b627642bdc87c10b3Shirish Kaleleimport com.android.internal.util.AsyncChannel;
88c04479601696f56c9cc240d4c7fc49fa99f51825Peter Qiuimport com.android.server.wifi.hotspot2.PasspointManager;
8992f4d52b14d8847f6f81712bf3bbfa0f1203eef5Sohani Raoimport com.android.server.wifi.util.WifiPermissionsUtil;
908be2ac8cec40a2000715849b627642bdc87c10b3Shirish Kalele
91155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpandeimport java.io.BufferedReader;
92155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpandeimport java.io.FileDescriptor;
938be2ac8cec40a2000715849b627642bdc87c10b3Shirish Kaleleimport java.io.FileNotFoundException;
94155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpandeimport java.io.FileReader;
95155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpandeimport java.io.IOException;
96155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpandeimport java.io.PrintWriter;
97155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpandeimport java.net.Inet4Address;
988be2ac8cec40a2000715849b627642bdc87c10b3Shirish Kaleleimport java.net.InetAddress;
99a1edc185d46d85e04930a5e12b465de9fea64afeJan Nordqvistimport java.security.GeneralSecurityException;
10031891afce40b903ada9b24ec12e3648ae6aa27b2Jan Nordqvistimport java.security.KeyStore;
10131891afce40b903ada9b24ec12e3648ae6aa27b2Jan Nordqvistimport java.security.cert.CertPath;
10231891afce40b903ada9b24ec12e3648ae6aa27b2Jan Nordqvistimport java.security.cert.CertPathValidator;
1039a069cba2e5ae31ee00d9ec9a3c25bdb7052aa78Jan Nordqvistimport java.security.cert.CertPathValidatorException;
10431891afce40b903ada9b24ec12e3648ae6aa27b2Jan Nordqvistimport java.security.cert.CertificateFactory;
10531891afce40b903ada9b24ec12e3648ae6aa27b2Jan Nordqvistimport java.security.cert.PKIXParameters;
10631891afce40b903ada9b24ec12e3648ae6aa27b2Jan Nordqvistimport java.security.cert.X509Certificate;
107155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpandeimport java.util.ArrayList;
10831891afce40b903ada9b24ec12e3648ae6aa27b2Jan Nordqvistimport java.util.Arrays;
109155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpandeimport java.util.List;
110155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
111155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande/**
112155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande * WifiService handles remote WiFi operation requests by implementing
113155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande * the IWifiManager interface.
114155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande *
115155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande * @hide
116155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande */
11779a4204d12f32d2f6a4dfc8500f5e74718cabb8dVinit Deshpandepublic class WifiServiceImpl extends IWifiManager.Stub {
118155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    private static final String TAG = "WifiService";
119f22d23092ab37286a5ef9d257d5bb32c421d2669vandwalle    private static final boolean DBG = true;
12070b75cda33a05e8b6e6724ed08e4e72f6fee0471Pierre Vandwalle    private static final boolean VDBG = false;
121155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
12284ee4726383fd968ab43caf120d15d9f47a51a54Paul Stewart    // Dumpsys argument to enable/disable disconnect on IP reachability failures.
12384ee4726383fd968ab43caf120d15d9f47a51a54Paul Stewart    private static final String DUMP_ARG_SET_IPREACH_DISCONNECT = "set-ipreach-disconnect";
12484ee4726383fd968ab43caf120d15d9f47a51a54Paul Stewart    private static final String DUMP_ARG_SET_IPREACH_DISCONNECT_ENABLED = "enabled";
12584ee4726383fd968ab43caf120d15d9f47a51a54Paul Stewart    private static final String DUMP_ARG_SET_IPREACH_DISCONNECT_DISABLED = "disabled";
12684ee4726383fd968ab43caf120d15d9f47a51a54Paul Stewart
127155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    final WifiStateMachine mWifiStateMachine;
128155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
129155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    private final Context mContext;
1300091305175e8c6fe7fc6d01efb9d405961db4ac7Ningyuan Wang    private final FrameworkFacade mFacade;
131155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
13203b70b37a8a20cb92d860b5015f6762dcb5e75ccDianne Hackborn    private final PowerManager mPowerManager;
133155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    private final AppOpsManager mAppOps;
134f9d866f1b4e93593b3411886fd71e729f4a5a205Stuart Scott    private final UserManager mUserManager;
135d02611ce4158fda6c2d14ee13ad7f9553f416d21Ningyuan Wang    private final WifiCountryCode mCountryCode;
136ebb7e0cb4572f73325a749539d589ebdefb8635cvandwalle    // Debug counter tracking scan requests sent by WifiManager
137ebb7e0cb4572f73325a749539d589ebdefb8635cvandwalle    private int scanRequestCounter = 0;
138ebb7e0cb4572f73325a749539d589ebdefb8635cvandwalle
139155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    /* Tracks the open wi-fi network notification */
140155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    private WifiNotificationController mNotificationController;
141f09b6de086d5a00417613886aa43402285d2a8abStephen Chen    private WifiWakeupController mWifiWakeupController;
142155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    /* Polls traffic stats and notifies clients */
143155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    private WifiTrafficPoller mTrafficPoller;
144155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    /* Tracks the persisted states for wi-fi & airplane mode */
145155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    final WifiSettingsStore mSettingsStore;
1461b067831bbff14f8e7a99b927b69f714d1b03448Glen Kuhne    /* Logs connection events and some general router and scan stats */
1471b067831bbff14f8e7a99b927b69f714d1b03448Glen Kuhne    private final WifiMetrics mWifiMetrics;
1482bfc67c9893c0a525b224d68dd73a74212b0c29fArthur Hsu    /* Manages affiliated certificates for current user */
1492bfc67c9893c0a525b224d68dd73a74212b0c29fArthur Hsu    private final WifiCertManager mCertManager;
1502bfc67c9893c0a525b224d68dd73a74212b0c29fArthur Hsu
151637a86ffb3a036a4f26a471378b57d8817f35c25Glen Kuhne    private final WifiInjector mWifiInjector;
152ff9686aec5b6ee21e01556899fb13ab55915f3c0Roshan Pius    /* Backup/Restore Module */
153ff9686aec5b6ee21e01556899fb13ab55915f3c0Roshan Pius    private final WifiBackupRestore mWifiBackupRestore;
154ff9686aec5b6ee21e01556899fb13ab55915f3c0Roshan Pius
1556c11cd5d6ff04afdf1bbbc99a850025d46c7bdc9Rebecca Silberstein    private WifiScanner mWifiScanner;
1566c11cd5d6ff04afdf1bbbc99a850025d46c7bdc9Rebecca Silberstein
157155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    /**
158155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande     * Asynchronous channel to WifiStateMachine
159155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande     */
160155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    private AsyncChannel mWifiStateMachineChannel;
161155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
16292f4d52b14d8847f6f81712bf3bbfa0f1203eef5Sohani Rao    private WifiPermissionsUtil mWifiPermissionsUtil;
16392f4d52b14d8847f6f81712bf3bbfa0f1203eef5Sohani Rao
1640469548a809654861f74dc42a3409f29ae750550Svetoslav Ganov    private final boolean mPermissionReviewRequired;
165c04479601696f56c9cc240d4c7fc49fa99f51825Peter Qiu    private final PasspointManager mPasspointManager;
1660469548a809654861f74dc42a3409f29ae750550Svetoslav Ganov
167155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    /**
168155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande     * Handles client connections
169155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande     */
170155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    private class ClientHandler extends Handler {
171155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
17240abf54c81c5624641543d86e1d7ab21ebe30175Paul Stewart        ClientHandler(Looper looper) {
173155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande            super(looper);
174155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        }
175155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
176155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        @Override
177155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        public void handleMessage(Message msg) {
178155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande            switch (msg.what) {
179155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande                case AsyncChannel.CMD_CHANNEL_HALF_CONNECTED: {
180155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande                    if (msg.arg1 == AsyncChannel.STATUS_SUCCESSFUL) {
181155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande                        if (DBG) Slog.d(TAG, "New client listening to asynchronous messages");
182155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande                        // We track the clients by the Messenger
183155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande                        // since it is expected to be always available
184155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande                        mTrafficPoller.addClient(msg.replyTo);
185155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande                    } else {
186155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande                        Slog.e(TAG, "Client connection failure, error=" + msg.arg1);
187155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande                    }
188155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande                    break;
189155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande                }
190155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande                case AsyncChannel.CMD_CHANNEL_DISCONNECTED: {
191155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande                    if (msg.arg1 == AsyncChannel.STATUS_SEND_UNSUCCESSFUL) {
192155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande                        if (DBG) Slog.d(TAG, "Send failed, client connection lost");
193155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande                    } else {
194155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande                        if (DBG) Slog.d(TAG, "Client connection lost with reason: " + msg.arg1);
195155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande                    }
196155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande                    mTrafficPoller.removeClient(msg.replyTo);
197155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande                    break;
198155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande                }
199155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande                case AsyncChannel.CMD_CHANNEL_FULL_CONNECTION: {
200155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande                    AsyncChannel ac = new AsyncChannel();
201155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande                    ac.connect(mContext, this, msg.replyTo);
202155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande                    break;
203155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande                }
204155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande                /* Client commands are forwarded to state machine */
205155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande                case WifiManager.CONNECT_NETWORK:
206155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande                case WifiManager.SAVE_NETWORK: {
207155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande                    WifiConfiguration config = (WifiConfiguration) msg.obj;
208155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande                    int networkId = msg.arg1;
209992ae00f25a9cc22cf5db3261bd7e72927069cf7vandwalle                    if (msg.what == WifiManager.SAVE_NETWORK) {
210f9946f5663f2d7a9a69cb8b7e82a4ef8b1825c48Mitchell Wills                        Slog.d("WiFiServiceImpl ", "SAVE"
211992ae00f25a9cc22cf5db3261bd7e72927069cf7vandwalle                                + " nid=" + Integer.toString(networkId)
212e487a4648dd41881e754f1224aaedead78a0777dSky Faber                                + " uid=" + msg.sendingUid
213e487a4648dd41881e754f1224aaedead78a0777dSky Faber                                + " name="
214e487a4648dd41881e754f1224aaedead78a0777dSky Faber                                + mContext.getPackageManager().getNameForUid(msg.sendingUid));
215992ae00f25a9cc22cf5db3261bd7e72927069cf7vandwalle                    }
216992ae00f25a9cc22cf5db3261bd7e72927069cf7vandwalle                    if (msg.what == WifiManager.CONNECT_NETWORK) {
217f9946f5663f2d7a9a69cb8b7e82a4ef8b1825c48Mitchell Wills                        Slog.d("WiFiServiceImpl ", "CONNECT "
218992ae00f25a9cc22cf5db3261bd7e72927069cf7vandwalle                                + " nid=" + Integer.toString(networkId)
219e487a4648dd41881e754f1224aaedead78a0777dSky Faber                                + " uid=" + msg.sendingUid
220e487a4648dd41881e754f1224aaedead78a0777dSky Faber                                + " name="
221e487a4648dd41881e754f1224aaedead78a0777dSky Faber                                + mContext.getPackageManager().getNameForUid(msg.sendingUid));
222992ae00f25a9cc22cf5db3261bd7e72927069cf7vandwalle                    }
223e487a4648dd41881e754f1224aaedead78a0777dSky Faber
224d03a9283285bb7adef6c687eb3a91fa4a8c4b502Vinit Deshpande                    if (config != null && isValid(config)) {
22564c98f4afda6c7970c5d563580165a308d5e7c15Jason Monk                        if (DBG) Slog.d(TAG, "Connect with config" + config);
22664c98f4afda6c7970c5d563580165a308d5e7c15Jason Monk                        mWifiStateMachine.sendMessage(Message.obtain(msg));
227155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande                    } else if (config == null
228155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande                            && networkId != WifiConfiguration.INVALID_NETWORK_ID) {
229155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande                        if (DBG) Slog.d(TAG, "Connect with networkId" + networkId);
230155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande                        mWifiStateMachine.sendMessage(Message.obtain(msg));
231155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande                    } else {
232155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande                        Slog.e(TAG, "ClientHandler.handleMessage ignoring invalid msg=" + msg);
233155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande                        if (msg.what == WifiManager.CONNECT_NETWORK) {
234c35d728a15e9270f5642ef79f5245c89d749285fSky Faber                            replyFailed(msg, WifiManager.CONNECT_NETWORK_FAILED,
235c35d728a15e9270f5642ef79f5245c89d749285fSky Faber                                    WifiManager.INVALID_ARGS);
236155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande                        } else {
237c35d728a15e9270f5642ef79f5245c89d749285fSky Faber                            replyFailed(msg, WifiManager.SAVE_NETWORK_FAILED,
238c35d728a15e9270f5642ef79f5245c89d749285fSky Faber                                    WifiManager.INVALID_ARGS);
239155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande                        }
240155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande                    }
241155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande                    break;
242155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande                }
243155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande                case WifiManager.FORGET_NETWORK:
244bace539f9cdc4ffb521c8251dd1c56073e805cd4Bartosz Fabianowski                    mWifiStateMachine.sendMessage(Message.obtain(msg));
245002ea67f42b0f69a8c9fa4719391f9476567b762Sky Faber                    break;
246155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande                case WifiManager.START_WPS:
247155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande                case WifiManager.CANCEL_WPS:
248155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande                case WifiManager.DISABLE_NETWORK:
249155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande                case WifiManager.RSSI_PKTCNT_FETCH: {
250155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande                    mWifiStateMachine.sendMessage(Message.obtain(msg));
251155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande                    break;
252155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande                }
253155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande                default: {
254155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande                    Slog.d(TAG, "ClientHandler.handleMessage ignoring msg=" + msg);
255155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande                    break;
256155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande                }
257155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande            }
258155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        }
259155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
260c35d728a15e9270f5642ef79f5245c89d749285fSky Faber        private void replyFailed(Message msg, int what, int why) {
2618fcd1fe86ddc3056e1a26e69bc68f148ae0e9ddcChristopher Wiley            Message reply = Message.obtain();
262155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande            reply.what = what;
263c35d728a15e9270f5642ef79f5245c89d749285fSky Faber            reply.arg1 = why;
264155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande            try {
265155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande                msg.replyTo.send(reply);
266155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande            } catch (RemoteException e) {
267155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande                // There's not much we can do if reply can't be sent!
268155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande            }
269155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        }
270155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    }
271155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    private ClientHandler mClientHandler;
272155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
273155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    /**
274155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande     * Handles interaction with WifiStateMachine
275155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande     */
276da918df16e03ee19be62343313d954027d3eb3abRebecca Silberstein    protected class WifiStateMachineHandler extends Handler {
277155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        private AsyncChannel mWsmChannel;
278155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
279da918df16e03ee19be62343313d954027d3eb3abRebecca Silberstein        public WifiStateMachineHandler(Looper looper, AsyncChannel asyncChannel) {
280155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande            super(looper);
281da918df16e03ee19be62343313d954027d3eb3abRebecca Silberstein            mWsmChannel = asyncChannel;
282155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande            mWsmChannel.connect(mContext, this, mWifiStateMachine.getHandler());
283155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        }
284155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
285155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        @Override
286155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        public void handleMessage(Message msg) {
287155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande            switch (msg.what) {
288155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande                case AsyncChannel.CMD_CHANNEL_HALF_CONNECTED: {
289155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande                    if (msg.arg1 == AsyncChannel.STATUS_SUCCESSFUL) {
290155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande                        mWifiStateMachineChannel = mWsmChannel;
291155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande                    } else {
292155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande                        Slog.e(TAG, "WifiStateMachine connection failure, error=" + msg.arg1);
293155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande                        mWifiStateMachineChannel = null;
294155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande                    }
295155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande                    break;
296155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande                }
297155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande                case AsyncChannel.CMD_CHANNEL_DISCONNECTED: {
298155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande                    Slog.e(TAG, "WifiStateMachine channel lost, msg.arg1 =" + msg.arg1);
299155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande                    mWifiStateMachineChannel = null;
300155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande                    //Re-establish connection to state machine
301155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande                    mWsmChannel.connect(mContext, this, mWifiStateMachine.getHandler());
302155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande                    break;
303155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande                }
304155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande                default: {
305155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande                    Slog.d(TAG, "WifiStateMachineHandler.handleMessage ignoring msg=" + msg);
306155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande                    break;
307155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande                }
308155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande            }
309155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        }
310155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    }
311155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
312155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    WifiStateMachineHandler mWifiStateMachineHandler;
313155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    private WifiController mWifiController;
3142a7c2bec5d307a9e52de429c6303b7df9de900dcRebecca Silberstein    private final WifiLockManager mWifiLockManager;
31561312e14a088a9487d4db64f08285162476e870fPaul Stewart    private final WifiMulticastLockManager mWifiMulticastLockManager;
316155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
317da918df16e03ee19be62343313d954027d3eb3abRebecca Silberstein    public WifiServiceImpl(Context context, WifiInjector wifiInjector, AsyncChannel asyncChannel) {
318155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        mContext = context;
319da918df16e03ee19be62343313d954027d3eb3abRebecca Silberstein        mWifiInjector = wifiInjector;
320fca64fc16ec43befde2e7ac7c3bfd84ced1f7778Rebecca Silberstein
321fca64fc16ec43befde2e7ac7c3bfd84ced1f7778Rebecca Silberstein        mFacade = mWifiInjector.getFrameworkFacade();
322637a86ffb3a036a4f26a471378b57d8817f35c25Glen Kuhne        mWifiMetrics = mWifiInjector.getWifiMetrics();
323fca64fc16ec43befde2e7ac7c3bfd84ced1f7778Rebecca Silberstein        mTrafficPoller = mWifiInjector.getWifiTrafficPoller();
3240fb9cf43830d67894b519c9a6271320c4842a5c0Bartosz Fabianowski        mUserManager = UserManager.get(mContext);
325fca64fc16ec43befde2e7ac7c3bfd84ced1f7778Rebecca Silberstein        mCountryCode = mWifiInjector.getWifiCountryCode();
326fca64fc16ec43befde2e7ac7c3bfd84ced1f7778Rebecca Silberstein        mWifiStateMachine = mWifiInjector.getWifiStateMachine();
327155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        mWifiStateMachine.enableRssiPolling(true);
328fca64fc16ec43befde2e7ac7c3bfd84ced1f7778Rebecca Silberstein        mSettingsStore = mWifiInjector.getWifiSettingsStore();
329fca64fc16ec43befde2e7ac7c3bfd84ced1f7778Rebecca Silberstein        mPowerManager = mContext.getSystemService(PowerManager.class);
330fca64fc16ec43befde2e7ac7c3bfd84ced1f7778Rebecca Silberstein        mAppOps = (AppOpsManager) mContext.getSystemService(Context.APP_OPS_SERVICE);
331fca64fc16ec43befde2e7ac7c3bfd84ced1f7778Rebecca Silberstein        mCertManager = mWifiInjector.getWifiCertManager();
332155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
333fca64fc16ec43befde2e7ac7c3bfd84ced1f7778Rebecca Silberstein        mNotificationController = mWifiInjector.getWifiNotificationController();
334f09b6de086d5a00417613886aa43402285d2a8abStephen Chen        mWifiWakeupController = mWifiInjector.getWifiWakeupController();
335155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
336fca64fc16ec43befde2e7ac7c3bfd84ced1f7778Rebecca Silberstein        mWifiLockManager = mWifiInjector.getWifiLockManager();
33761312e14a088a9487d4db64f08285162476e870fPaul Stewart        mWifiMulticastLockManager = mWifiInjector.getWifiMulticastLockManager();
338fca64fc16ec43befde2e7ac7c3bfd84ced1f7778Rebecca Silberstein        HandlerThread wifiServiceHandlerThread = mWifiInjector.getWifiServiceHandlerThread();
339fca64fc16ec43befde2e7ac7c3bfd84ced1f7778Rebecca Silberstein        mClientHandler = new ClientHandler(wifiServiceHandlerThread.getLooper());
340da918df16e03ee19be62343313d954027d3eb3abRebecca Silberstein        mWifiStateMachineHandler = new WifiStateMachineHandler(wifiServiceHandlerThread.getLooper(),
341da918df16e03ee19be62343313d954027d3eb3abRebecca Silberstein                                                                asyncChannel);
342fca64fc16ec43befde2e7ac7c3bfd84ced1f7778Rebecca Silberstein        mWifiController = mWifiInjector.getWifiController();
3433204fb9682242a7b5a749489076c66d448c42577Roshan Pius        mWifiBackupRestore = mWifiInjector.getWifiBackupRestore();
3440469548a809654861f74dc42a3409f29ae750550Svetoslav Ganov        mPermissionReviewRequired = Build.PERMISSIONS_REVIEW_REQUIRED
3450469548a809654861f74dc42a3409f29ae750550Svetoslav Ganov                || context.getResources().getBoolean(
3460469548a809654861f74dc42a3409f29ae750550Svetoslav Ganov                com.android.internal.R.bool.config_permissionReviewRequired);
347d098b17dcef5b03a0626f34cc72d5fbce1bacb8fSohani Rao        mWifiPermissionsUtil = mWifiInjector.getWifiPermissionsUtil();
348c04479601696f56c9cc240d4c7fc49fa99f51825Peter Qiu        mPasspointManager = mWifiInjector.getPasspointManager();
349c04479601696f56c9cc240d4c7fc49fa99f51825Peter Qiu
35000ec8bf178ba4abee4ee8bbc6eb09c9fcd986a89Rebecca Silberstein        enableVerboseLoggingInternal(getVerboseLoggingLevel());
3511c28bb29cb19367d5e3bdbe6d8558d0fd4b8f33eVinit Deshapnde    }
3521c28bb29cb19367d5e3bdbe6d8558d0fd4b8f33eVinit Deshapnde
3531c28bb29cb19367d5e3bdbe6d8558d0fd4b8f33eVinit Deshapnde    /**
3541c28bb29cb19367d5e3bdbe6d8558d0fd4b8f33eVinit Deshapnde     * Check if Wi-Fi needs to be enabled and start
3551c28bb29cb19367d5e3bdbe6d8558d0fd4b8f33eVinit Deshapnde     * if needed
3561c28bb29cb19367d5e3bdbe6d8558d0fd4b8f33eVinit Deshapnde     *
3571c28bb29cb19367d5e3bdbe6d8558d0fd4b8f33eVinit Deshapnde     * This function is used only at boot time
3581c28bb29cb19367d5e3bdbe6d8558d0fd4b8f33eVinit Deshapnde     */
3591c28bb29cb19367d5e3bdbe6d8558d0fd4b8f33eVinit Deshapnde    public void checkAndStartWifi() {
3601c28bb29cb19367d5e3bdbe6d8558d0fd4b8f33eVinit Deshapnde        /* Check if wi-fi needs to be enabled */
3611c28bb29cb19367d5e3bdbe6d8558d0fd4b8f33eVinit Deshapnde        boolean wifiEnabled = mSettingsStore.isWifiToggleEnabled();
3621c28bb29cb19367d5e3bdbe6d8558d0fd4b8f33eVinit Deshapnde        Slog.i(TAG, "WifiService starting up with Wi-Fi " +
3631c28bb29cb19367d5e3bdbe6d8558d0fd4b8f33eVinit Deshapnde                (wifiEnabled ? "enabled" : "disabled"));
364155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
365155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        registerForScanModeChange();
366155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        mContext.registerReceiver(
367155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande                new BroadcastReceiver() {
368155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande                    @Override
369155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande                    public void onReceive(Context context, Intent intent) {
370155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande                        if (mSettingsStore.handleAirplaneModeToggled()) {
371155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande                            mWifiController.sendMessage(CMD_AIRPLANE_TOGGLED);
372155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande                        }
37398e43e5f48a4c87343fc311feda4fa5489948822Vinit Deshpande                        if (mSettingsStore.isAirplaneModeOn()) {
37498e43e5f48a4c87343fc311feda4fa5489948822Vinit Deshpande                            Log.d(TAG, "resetting country code because Airplane mode is ON");
375d02611ce4158fda6c2d14ee13ad7f9553f416d21Ningyuan Wang                            mCountryCode.airplaneModeEnabled();
37698e43e5f48a4c87343fc311feda4fa5489948822Vinit Deshpande                        }
377155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande                    }
378155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande                },
379155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande                new IntentFilter(Intent.ACTION_AIRPLANE_MODE_CHANGED));
380155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
38198e43e5f48a4c87343fc311feda4fa5489948822Vinit Deshpande        mContext.registerReceiver(
38298e43e5f48a4c87343fc311feda4fa5489948822Vinit Deshpande                new BroadcastReceiver() {
38398e43e5f48a4c87343fc311feda4fa5489948822Vinit Deshpande                    @Override
38498e43e5f48a4c87343fc311feda4fa5489948822Vinit Deshpande                    public void onReceive(Context context, Intent intent) {
38598e43e5f48a4c87343fc311feda4fa5489948822Vinit Deshpande                        String state = intent.getStringExtra(IccCardConstants.INTENT_KEY_ICC_STATE);
386a7bf9d90d68a4833cc3c64da98024167931c4b1epkanwar                        if (IccCardConstants.INTENT_VALUE_ICC_ABSENT.equals(state)) {
387fdce524fd05844c996cf1c5c0c102a87fde8e32cVinit Deshpande                            Log.d(TAG, "resetting networks because SIM was removed");
3883c8094ab45f3320dbe45e6460c5d62dcc24ce7aeMitchell Wills                            mWifiStateMachine.resetSimAuthNetworks(false);
38998e43e5f48a4c87343fc311feda4fa5489948822Vinit Deshpande                            Log.d(TAG, "resetting country code because SIM is removed");
390d02611ce4158fda6c2d14ee13ad7f9553f416d21Ningyuan Wang                            mCountryCode.simCardRemoved();
3913c8094ab45f3320dbe45e6460c5d62dcc24ce7aeMitchell Wills                        } else if (IccCardConstants.INTENT_VALUE_ICC_LOADED.equals(state)) {
3923c8094ab45f3320dbe45e6460c5d62dcc24ce7aeMitchell Wills                            Log.d(TAG, "resetting networks because SIM was loaded");
3933c8094ab45f3320dbe45e6460c5d62dcc24ce7aeMitchell Wills                            mWifiStateMachine.resetSimAuthNetworks(true);
39498e43e5f48a4c87343fc311feda4fa5489948822Vinit Deshpande                        }
39598e43e5f48a4c87343fc311feda4fa5489948822Vinit Deshpande                    }
39698e43e5f48a4c87343fc311feda4fa5489948822Vinit Deshpande                },
39798e43e5f48a4c87343fc311feda4fa5489948822Vinit Deshpande                new IntentFilter(TelephonyIntents.ACTION_SIM_STATE_CHANGED));
39898e43e5f48a4c87343fc311feda4fa5489948822Vinit Deshpande
399155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        // Adding optimizations of only receiving broadcasts when wifi is enabled
400155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        // can result in race conditions when apps toggle wifi in the background
401155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        // without active user involvement. Always receive broadcasts.
402155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        registerForBroadcasts();
403faa2ce7f93eef2c436796fb1ea3574e2d1412491Robin Lee        registerForPackageOrUserRemoval();
40403b70b37a8a20cb92d860b5015f6762dcb5e75ccDianne Hackborn        mInIdleMode = mPowerManager.isDeviceIdleMode();
405155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
4061c28bb29cb19367d5e3bdbe6d8558d0fd4b8f33eVinit Deshapnde        mWifiController.start();
407155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
408155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        // If we are already disabled (could be due to airplane mode), avoid changing persist
409155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        // state here
4100469548a809654861f74dc42a3409f29ae750550Svetoslav Ganov        if (wifiEnabled) {
4110469548a809654861f74dc42a3409f29ae750550Svetoslav Ganov            try {
4120469548a809654861f74dc42a3409f29ae750550Svetoslav Ganov                setWifiEnabled(mContext.getPackageName(), wifiEnabled);
4130469548a809654861f74dc42a3409f29ae750550Svetoslav Ganov            } catch (RemoteException e) {
4140469548a809654861f74dc42a3409f29ae750550Svetoslav Ganov                /* ignore - local call */
4150469548a809654861f74dc42a3409f29ae750550Svetoslav Ganov            }
4160469548a809654861f74dc42a3409f29ae750550Svetoslav Ganov        }
417155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    }
418155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
419bcdabb1fa1894fcca610692ec94459fe623afa74Bartosz Fabianowski    public void handleUserSwitch(int userId) {
420bcdabb1fa1894fcca610692ec94459fe623afa74Bartosz Fabianowski        mWifiStateMachine.handleUserSwitch(userId);
421bcdabb1fa1894fcca610692ec94459fe623afa74Bartosz Fabianowski    }
4223bc487aa49deecbc358ee819e0dd4b2534412281Roshan Pius
4233bc487aa49deecbc358ee819e0dd4b2534412281Roshan Pius    public void handleUserUnlock(int userId) {
4243bc487aa49deecbc358ee819e0dd4b2534412281Roshan Pius        mWifiStateMachine.handleUserUnlock(userId);
4253bc487aa49deecbc358ee819e0dd4b2534412281Roshan Pius    }
4263bc487aa49deecbc358ee819e0dd4b2534412281Roshan Pius
4273bc487aa49deecbc358ee819e0dd4b2534412281Roshan Pius    public void handleUserStop(int userId) {
4283bc487aa49deecbc358ee819e0dd4b2534412281Roshan Pius        mWifiStateMachine.handleUserStop(userId);
4293bc487aa49deecbc358ee819e0dd4b2534412281Roshan Pius    }
430bcdabb1fa1894fcca610692ec94459fe623afa74Bartosz Fabianowski
431155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    /**
432155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande     * see {@link android.net.wifi.WifiManager#pingSupplicant()}
433155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande     * @return {@code true} if the operation succeeds, {@code false} otherwise
434155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande     */
4358fcd1fe86ddc3056e1a26e69bc68f148ae0e9ddcChristopher Wiley    @Override
436155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    public boolean pingSupplicant() {
437155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        enforceAccessPermission();
438155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        if (mWifiStateMachineChannel != null) {
439155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande            return mWifiStateMachine.syncPingSupplicant(mWifiStateMachineChannel);
440155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        } else {
441155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande            Slog.e(TAG, "mWifiStateMachineChannel is not initialized");
442155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande            return false;
443155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        }
444155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    }
445155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
446155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    /**
447a0009d14075b5345b8f916c3fb3f2260c938cb9dYuhao Zheng     * see {@link android.net.wifi.WifiManager#startScan}
448a0009d14075b5345b8f916c3fb3f2260c938cb9dYuhao Zheng     * and {@link android.net.wifi.WifiManager#startCustomizedScan}
449155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande     *
450a0009d14075b5345b8f916c3fb3f2260c938cb9dYuhao Zheng     * @param settings If null, use default parameter, i.e. full scan.
451a0009d14075b5345b8f916c3fb3f2260c938cb9dYuhao Zheng     * @param workSource If null, all blame is given to the calling uid.
452155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande     */
4538fcd1fe86ddc3056e1a26e69bc68f148ae0e9ddcChristopher Wiley    @Override
454a0009d14075b5345b8f916c3fb3f2260c938cb9dYuhao Zheng    public void startScan(ScanSettings settings, WorkSource workSource) {
455155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        enforceChangePermission();
4561227b49a94f33844ad0606b48b591bea4d27b08eVinit Deshpande        synchronized (this) {
4576c11cd5d6ff04afdf1bbbc99a850025d46c7bdc9Rebecca Silberstein            if (mWifiScanner == null) {
4586c11cd5d6ff04afdf1bbbc99a850025d46c7bdc9Rebecca Silberstein                mWifiScanner = mWifiInjector.getWifiScanner();
4596c11cd5d6ff04afdf1bbbc99a850025d46c7bdc9Rebecca Silberstein            }
46003b70b37a8a20cb92d860b5015f6762dcb5e75ccDianne Hackborn            if (mInIdleMode) {
46149dfc6768889a33eb67102b16faf21667af2a60dVinit Deshpande                // Need to send an immediate scan result broadcast in case the
46249dfc6768889a33eb67102b16faf21667af2a60dVinit Deshpande                // caller is waiting for a result ..
463691fb51e7ac5e4d0b91a4719cd0ed10a8c212e50Vinit Deshpande
464691fb51e7ac5e4d0b91a4719cd0ed10a8c212e50Vinit Deshpande                // clear calling identity to send broadcast
465691fb51e7ac5e4d0b91a4719cd0ed10a8c212e50Vinit Deshpande                long callingIdentity = Binder.clearCallingIdentity();
466691fb51e7ac5e4d0b91a4719cd0ed10a8c212e50Vinit Deshpande                try {
4676c11cd5d6ff04afdf1bbbc99a850025d46c7bdc9Rebecca Silberstein                    // TODO: investigate if the logic to cancel scans when idle can move to
4686c11cd5d6ff04afdf1bbbc99a850025d46c7bdc9Rebecca Silberstein                    // WifiScanningServiceImpl.  This will 1 - clean up WifiServiceImpl and 2 -
4696c11cd5d6ff04afdf1bbbc99a850025d46c7bdc9Rebecca Silberstein                    // avoid plumbing an awkward path to report a cancelled/failed scan.  This will
4706c11cd5d6ff04afdf1bbbc99a850025d46c7bdc9Rebecca Silberstein                    // be sent directly until b/31398592 is fixed.
4716c11cd5d6ff04afdf1bbbc99a850025d46c7bdc9Rebecca Silberstein                    Intent intent = new Intent(WifiManager.SCAN_RESULTS_AVAILABLE_ACTION);
4726c11cd5d6ff04afdf1bbbc99a850025d46c7bdc9Rebecca Silberstein                    intent.addFlags(Intent.FLAG_RECEIVER_REGISTERED_ONLY_BEFORE_BOOT);
4736c11cd5d6ff04afdf1bbbc99a850025d46c7bdc9Rebecca Silberstein                    intent.putExtra(WifiManager.EXTRA_RESULTS_UPDATED, false);
4746c11cd5d6ff04afdf1bbbc99a850025d46c7bdc9Rebecca Silberstein                    mContext.sendBroadcastAsUser(intent, UserHandle.ALL);
475691fb51e7ac5e4d0b91a4719cd0ed10a8c212e50Vinit Deshpande                } finally {
476691fb51e7ac5e4d0b91a4719cd0ed10a8c212e50Vinit Deshpande                    // restore calling identity
477691fb51e7ac5e4d0b91a4719cd0ed10a8c212e50Vinit Deshpande                    Binder.restoreCallingIdentity(callingIdentity);
478691fb51e7ac5e4d0b91a4719cd0ed10a8c212e50Vinit Deshpande                }
47903b70b37a8a20cb92d860b5015f6762dcb5e75ccDianne Hackborn                mScanPending = true;
48003b70b37a8a20cb92d860b5015f6762dcb5e75ccDianne Hackborn                return;
48103b70b37a8a20cb92d860b5015f6762dcb5e75ccDianne Hackborn            }
48203b70b37a8a20cb92d860b5015f6762dcb5e75ccDianne Hackborn        }
483a0009d14075b5345b8f916c3fb3f2260c938cb9dYuhao Zheng        if (settings != null) {
484a0009d14075b5345b8f916c3fb3f2260c938cb9dYuhao Zheng            settings = new ScanSettings(settings);
485a0009d14075b5345b8f916c3fb3f2260c938cb9dYuhao Zheng            if (!settings.isValid()) {
486a0009d14075b5345b8f916c3fb3f2260c938cb9dYuhao Zheng                Slog.e(TAG, "invalid scan setting");
487a0009d14075b5345b8f916c3fb3f2260c938cb9dYuhao Zheng                return;
488a0009d14075b5345b8f916c3fb3f2260c938cb9dYuhao Zheng            }
489a0009d14075b5345b8f916c3fb3f2260c938cb9dYuhao Zheng        }
490155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        if (workSource != null) {
491155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande            enforceWorkSourcePermission();
492155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande            // WifiManager currently doesn't use names, so need to clear names out of the
493155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande            // supplied WorkSource to allow future WorkSource combining.
494155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande            workSource.clearNames();
495155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        }
49648444cb4214a48a3a0bf4bbb93945c2aa68c9980Mitchell Wills        if (workSource == null && Binder.getCallingUid() >= 0) {
49748444cb4214a48a3a0bf4bbb93945c2aa68c9980Mitchell Wills            workSource = new WorkSource(Binder.getCallingUid());
49848444cb4214a48a3a0bf4bbb93945c2aa68c9980Mitchell Wills        }
499ebb7e0cb4572f73325a749539d589ebdefb8635cvandwalle        mWifiStateMachine.startScan(Binder.getCallingUid(), scanRequestCounter++,
500ebb7e0cb4572f73325a749539d589ebdefb8635cvandwalle                settings, workSource);
501155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    }
502155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
5038fcd1fe86ddc3056e1a26e69bc68f148ae0e9ddcChristopher Wiley    @Override
5043f7ef65ab71619040032aee96b5599849881d6fdAndres Morales    public String getWpsNfcConfigurationToken(int netId) {
505f3f4029b3ac41da4cc6a1bc22bae47c750a47048Andres Morales        enforceConnectivityInternalPermission();
5063f7ef65ab71619040032aee96b5599849881d6fdAndres Morales        return mWifiStateMachine.syncGetWpsNfcConfigurationToken(netId);
5073f7ef65ab71619040032aee96b5599849881d6fdAndres Morales    }
5083f7ef65ab71619040032aee96b5599849881d6fdAndres Morales
5091227b49a94f33844ad0606b48b591bea4d27b08eVinit Deshpande    boolean mInIdleMode;
5101227b49a94f33844ad0606b48b591bea4d27b08eVinit Deshpande    boolean mScanPending;
5111227b49a94f33844ad0606b48b591bea4d27b08eVinit Deshpande
51203b70b37a8a20cb92d860b5015f6762dcb5e75ccDianne Hackborn    void handleIdleModeChanged() {
51303b70b37a8a20cb92d860b5015f6762dcb5e75ccDianne Hackborn        boolean doScan = false;
5141227b49a94f33844ad0606b48b591bea4d27b08eVinit Deshpande        synchronized (this) {
51503b70b37a8a20cb92d860b5015f6762dcb5e75ccDianne Hackborn            boolean idle = mPowerManager.isDeviceIdleMode();
51603b70b37a8a20cb92d860b5015f6762dcb5e75ccDianne Hackborn            if (mInIdleMode != idle) {
51703b70b37a8a20cb92d860b5015f6762dcb5e75ccDianne Hackborn                mInIdleMode = idle;
51803b70b37a8a20cb92d860b5015f6762dcb5e75ccDianne Hackborn                if (!idle) {
51903b70b37a8a20cb92d860b5015f6762dcb5e75ccDianne Hackborn                    if (mScanPending) {
52003b70b37a8a20cb92d860b5015f6762dcb5e75ccDianne Hackborn                        mScanPending = false;
52103b70b37a8a20cb92d860b5015f6762dcb5e75ccDianne Hackborn                        doScan = true;
52203b70b37a8a20cb92d860b5015f6762dcb5e75ccDianne Hackborn                    }
52303b70b37a8a20cb92d860b5015f6762dcb5e75ccDianne Hackborn                }
52403b70b37a8a20cb92d860b5015f6762dcb5e75ccDianne Hackborn            }
52503b70b37a8a20cb92d860b5015f6762dcb5e75ccDianne Hackborn        }
52603b70b37a8a20cb92d860b5015f6762dcb5e75ccDianne Hackborn        if (doScan) {
52703b70b37a8a20cb92d860b5015f6762dcb5e75ccDianne Hackborn            // Someone requested a scan while we were idle; do a full scan now.
52803b70b37a8a20cb92d860b5015f6762dcb5e75ccDianne Hackborn            startScan(null, null);
52903b70b37a8a20cb92d860b5015f6762dcb5e75ccDianne Hackborn        }
53003b70b37a8a20cb92d860b5015f6762dcb5e75ccDianne Hackborn    }
53103b70b37a8a20cb92d860b5015f6762dcb5e75ccDianne Hackborn
532155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    private void enforceAccessPermission() {
533155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        mContext.enforceCallingOrSelfPermission(android.Manifest.permission.ACCESS_WIFI_STATE,
534a632d8a6edd350c7644b593b18eceaa5b368505bVinit Deshpande                "WifiService");
535155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    }
536155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
537155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    private void enforceChangePermission() {
538155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        mContext.enforceCallingOrSelfPermission(android.Manifest.permission.CHANGE_WIFI_STATE,
53931891afce40b903ada9b24ec12e3648ae6aa27b2Jan Nordqvist                "WifiService");
5409086afccf6938a49eb9a2cd248917c1cb0943942vandwalle    }
541155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
5429086afccf6938a49eb9a2cd248917c1cb0943942vandwalle    private void enforceLocationHardwarePermission() {
5439086afccf6938a49eb9a2cd248917c1cb0943942vandwalle        mContext.enforceCallingOrSelfPermission(Manifest.permission.LOCATION_HARDWARE,
5449086afccf6938a49eb9a2cd248917c1cb0943942vandwalle                "LocationHardware");
545155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    }
546155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
5479878c61bbd81176561991be025af44efc67332feWenchao Tong    private void enforceReadCredentialPermission() {
5489878c61bbd81176561991be025af44efc67332feWenchao Tong        mContext.enforceCallingOrSelfPermission(android.Manifest.permission.READ_WIFI_CREDENTIAL,
5499878c61bbd81176561991be025af44efc67332feWenchao Tong                                                "WifiService");
5509878c61bbd81176561991be025af44efc67332feWenchao Tong    }
5519878c61bbd81176561991be025af44efc67332feWenchao Tong
552155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    private void enforceWorkSourcePermission() {
553155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        mContext.enforceCallingPermission(android.Manifest.permission.UPDATE_DEVICE_STATS,
554a632d8a6edd350c7644b593b18eceaa5b368505bVinit Deshpande                "WifiService");
555155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
556155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    }
557155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
558155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    private void enforceMulticastChangePermission() {
559155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        mContext.enforceCallingOrSelfPermission(
560155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande                android.Manifest.permission.CHANGE_WIFI_MULTICAST_STATE,
561155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande                "WifiService");
562155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    }
563155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
564155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    private void enforceConnectivityInternalPermission() {
565155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        mContext.enforceCallingOrSelfPermission(
566155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande                android.Manifest.permission.CONNECTIVITY_INTERNAL,
567155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande                "ConnectivityService");
568155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    }
569155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
570155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    /**
571155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande     * see {@link android.net.wifi.WifiManager#setWifiEnabled(boolean)}
572155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande     * @param enable {@code true} to enable, {@code false} to disable.
573155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande     * @return {@code true} if the enable/disable operation was
574155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande     *         started or is already in the queue.
575155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande     */
5768fcd1fe86ddc3056e1a26e69bc68f148ae0e9ddcChristopher Wiley    @Override
5770469548a809654861f74dc42a3409f29ae750550Svetoslav Ganov    public synchronized boolean setWifiEnabled(String packageName, boolean enable)
5780469548a809654861f74dc42a3409f29ae750550Svetoslav Ganov            throws RemoteException {
579155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        enforceChangePermission();
580155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        Slog.d(TAG, "setWifiEnabled: " + enable + " pid=" + Binder.getCallingPid()
581155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande                    + ", uid=" + Binder.getCallingUid());
582155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
583155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        /*
584155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        * Caller might not have WRITE_SECURE_SETTINGS,
585155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        * only CHANGE_WIFI_STATE is enforced
586155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        */
587155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        long ident = Binder.clearCallingIdentity();
588155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        try {
589155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande            if (! mSettingsStore.handleWifiToggled(enable)) {
590155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande                // Nothing to do if wifi cannot be toggled
591155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande                return true;
592155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande            }
593155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        } finally {
594155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande            Binder.restoreCallingIdentity(ident);
595155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        }
596155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
5970469548a809654861f74dc42a3409f29ae750550Svetoslav Ganov
5980469548a809654861f74dc42a3409f29ae750550Svetoslav Ganov        if (mPermissionReviewRequired) {
5990469548a809654861f74dc42a3409f29ae750550Svetoslav Ganov            final int wiFiEnabledState = getWifiEnabledState();
6000469548a809654861f74dc42a3409f29ae750550Svetoslav Ganov            if (enable) {
6010469548a809654861f74dc42a3409f29ae750550Svetoslav Ganov                if (wiFiEnabledState == WifiManager.WIFI_STATE_DISABLING
6020469548a809654861f74dc42a3409f29ae750550Svetoslav Ganov                        || wiFiEnabledState == WifiManager.WIFI_STATE_DISABLED) {
6030469548a809654861f74dc42a3409f29ae750550Svetoslav Ganov                    if (startConsentUiIfNeeded(packageName, Binder.getCallingUid(),
6040469548a809654861f74dc42a3409f29ae750550Svetoslav Ganov                            WifiManager.ACTION_REQUEST_ENABLE)) {
6050469548a809654861f74dc42a3409f29ae750550Svetoslav Ganov                        return true;
6060469548a809654861f74dc42a3409f29ae750550Svetoslav Ganov                    }
6070469548a809654861f74dc42a3409f29ae750550Svetoslav Ganov                }
6080469548a809654861f74dc42a3409f29ae750550Svetoslav Ganov            } else if (wiFiEnabledState == WifiManager.WIFI_STATE_ENABLING
6090469548a809654861f74dc42a3409f29ae750550Svetoslav Ganov                    || wiFiEnabledState == WifiManager.WIFI_STATE_ENABLED) {
6100469548a809654861f74dc42a3409f29ae750550Svetoslav Ganov                if (startConsentUiIfNeeded(packageName, Binder.getCallingUid(),
6110469548a809654861f74dc42a3409f29ae750550Svetoslav Ganov                        WifiManager.ACTION_REQUEST_DISABLE)) {
6120469548a809654861f74dc42a3409f29ae750550Svetoslav Ganov                    return true;
6130469548a809654861f74dc42a3409f29ae750550Svetoslav Ganov                }
6140469548a809654861f74dc42a3409f29ae750550Svetoslav Ganov            }
6150469548a809654861f74dc42a3409f29ae750550Svetoslav Ganov        }
6160469548a809654861f74dc42a3409f29ae750550Svetoslav Ganov
617155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        mWifiController.sendMessage(CMD_WIFI_TOGGLED);
618155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        return true;
619155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    }
620155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
621155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    /**
622155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande     * see {@link WifiManager#getWifiState()}
623155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande     * @return One of {@link WifiManager#WIFI_STATE_DISABLED},
624155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande     *         {@link WifiManager#WIFI_STATE_DISABLING},
625155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande     *         {@link WifiManager#WIFI_STATE_ENABLED},
626155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande     *         {@link WifiManager#WIFI_STATE_ENABLING},
627155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande     *         {@link WifiManager#WIFI_STATE_UNKNOWN}
628155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande     */
6298fcd1fe86ddc3056e1a26e69bc68f148ae0e9ddcChristopher Wiley    @Override
630155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    public int getWifiEnabledState() {
631155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        enforceAccessPermission();
632155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        return mWifiStateMachine.syncGetWifiState();
633155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    }
634155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
635155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    /**
636155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande     * see {@link android.net.wifi.WifiManager#setWifiApEnabled(WifiConfiguration, boolean)}
637155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande     * @param wifiConfig SSID, security and channel details as
638155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande     *        part of WifiConfiguration
639155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande     * @param enabled true to enable and false to disable
640155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande     */
6418fcd1fe86ddc3056e1a26e69bc68f148ae0e9ddcChristopher Wiley    @Override
642155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    public void setWifiApEnabled(WifiConfiguration wifiConfig, boolean enabled) {
643328fc5407927799843c11f2b767a8cf47b89f366Robert Greenwalt        enforceChangePermission();
64434f173a54307c39ffe5dfab52e7ef8166a00539eRobert Greenwalt        ConnectivityManager.enforceTetherChangePermission(mContext);
645f9d866f1b4e93593b3411886fd71e729f4a5a205Stuart Scott        if (mUserManager.hasUserRestriction(UserManager.DISALLOW_CONFIG_TETHERING)) {
64613cddb5337418359eb5f9bebd0504fbc5c47fc41Julia Reynolds            throw new SecurityException("DISALLOW_CONFIG_TETHERING is enabled for this user.");
64713cddb5337418359eb5f9bebd0504fbc5c47fc41Julia Reynolds        }
648155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        // null wifiConfig is a meaningful input for CMD_SET_AP
649d03a9283285bb7adef6c687eb3a91fa4a8c4b502Vinit Deshpande        if (wifiConfig == null || isValid(wifiConfig)) {
650155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande            mWifiController.obtainMessage(CMD_SET_AP, enabled ? 1 : 0, 0, wifiConfig).sendToTarget();
651155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        } else {
652155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande            Slog.e(TAG, "Invalid WifiConfiguration");
653155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        }
654155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    }
655155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
656155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    /**
657155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande     * see {@link WifiManager#getWifiApState()}
658155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande     * @return One of {@link WifiManager#WIFI_AP_STATE_DISABLED},
659155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande     *         {@link WifiManager#WIFI_AP_STATE_DISABLING},
660155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande     *         {@link WifiManager#WIFI_AP_STATE_ENABLED},
661155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande     *         {@link WifiManager#WIFI_AP_STATE_ENABLING},
662155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande     *         {@link WifiManager#WIFI_AP_STATE_FAILED}
663155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande     */
6648fcd1fe86ddc3056e1a26e69bc68f148ae0e9ddcChristopher Wiley    @Override
665155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    public int getWifiApEnabledState() {
666155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        enforceAccessPermission();
667155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        return mWifiStateMachine.syncGetWifiApState();
668155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    }
669155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
670155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    /**
671155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande     * see {@link WifiManager#getWifiApConfiguration()}
672155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande     * @return soft access point configuration
673155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande     */
6748fcd1fe86ddc3056e1a26e69bc68f148ae0e9ddcChristopher Wiley    @Override
675155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    public WifiConfiguration getWifiApConfiguration() {
676155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        enforceAccessPermission();
677155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        return mWifiStateMachine.syncGetWifiApConfiguration();
678155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    }
679155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
680155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    /**
681155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande     * see {@link WifiManager#setWifiApConfiguration(WifiConfiguration)}
682155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande     * @param wifiConfig WifiConfiguration details for soft access point
683155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande     */
6848fcd1fe86ddc3056e1a26e69bc68f148ae0e9ddcChristopher Wiley    @Override
685155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    public void setWifiApConfiguration(WifiConfiguration wifiConfig) {
686155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        enforceChangePermission();
687155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        if (wifiConfig == null)
688155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande            return;
689d03a9283285bb7adef6c687eb3a91fa4a8c4b502Vinit Deshpande        if (isValid(wifiConfig)) {
690155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande            mWifiStateMachine.setWifiApConfiguration(wifiConfig);
691155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        } else {
692155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande            Slog.e(TAG, "Invalid WifiConfiguration");
693155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        }
694155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    }
695155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
696155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    /**
6978fcd1fe86ddc3056e1a26e69bc68f148ae0e9ddcChristopher Wiley     * see {@link android.net.wifi.WifiManager#isScanAlwaysAvailable()}
698155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande     */
6998fcd1fe86ddc3056e1a26e69bc68f148ae0e9ddcChristopher Wiley    @Override
700155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    public boolean isScanAlwaysAvailable() {
701155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        enforceAccessPermission();
702155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        return mSettingsStore.isScanAlwaysAvailable();
703155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    }
704155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
705155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    /**
706155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande     * see {@link android.net.wifi.WifiManager#disconnect()}
707155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande     */
7088fcd1fe86ddc3056e1a26e69bc68f148ae0e9ddcChristopher Wiley    @Override
709155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    public void disconnect() {
710155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        enforceChangePermission();
711155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        mWifiStateMachine.disconnectCommand();
712155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    }
713155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
714155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    /**
715155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande     * see {@link android.net.wifi.WifiManager#reconnect()}
716155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande     */
7178fcd1fe86ddc3056e1a26e69bc68f148ae0e9ddcChristopher Wiley    @Override
718155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    public void reconnect() {
719155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        enforceChangePermission();
720155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        mWifiStateMachine.reconnectCommand();
721155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    }
722155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
723155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    /**
724155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande     * see {@link android.net.wifi.WifiManager#reassociate()}
725155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande     */
7268fcd1fe86ddc3056e1a26e69bc68f148ae0e9ddcChristopher Wiley    @Override
727155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    public void reassociate() {
728155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        enforceChangePermission();
729155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        mWifiStateMachine.reassociateCommand();
730155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    }
731155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
732155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    /**
733048dc8df2bd9394979dbb090b0cfe8d6627d23e3Vinit Deshpande     * see {@link android.net.wifi.WifiManager#getSupportedFeatures}
734a632d8a6edd350c7644b593b18eceaa5b368505bVinit Deshpande     */
7358fcd1fe86ddc3056e1a26e69bc68f148ae0e9ddcChristopher Wiley    @Override
736048dc8df2bd9394979dbb090b0cfe8d6627d23e3Vinit Deshpande    public int getSupportedFeatures() {
737a632d8a6edd350c7644b593b18eceaa5b368505bVinit Deshpande        enforceAccessPermission();
738a632d8a6edd350c7644b593b18eceaa5b368505bVinit Deshpande        if (mWifiStateMachineChannel != null) {
739048dc8df2bd9394979dbb090b0cfe8d6627d23e3Vinit Deshpande            return mWifiStateMachine.syncGetSupportedFeatures(mWifiStateMachineChannel);
740a632d8a6edd350c7644b593b18eceaa5b368505bVinit Deshpande        } else {
741a632d8a6edd350c7644b593b18eceaa5b368505bVinit Deshpande            Slog.e(TAG, "mWifiStateMachineChannel is not initialized");
742048dc8df2bd9394979dbb090b0cfe8d6627d23e3Vinit Deshpande            return 0;
743a632d8a6edd350c7644b593b18eceaa5b368505bVinit Deshpande        }
744a632d8a6edd350c7644b593b18eceaa5b368505bVinit Deshpande    }
745a632d8a6edd350c7644b593b18eceaa5b368505bVinit Deshpande
746c48fd721a9a3273f0faa927d93caad4c9832af7dAdam Lesinski    @Override
747c48fd721a9a3273f0faa927d93caad4c9832af7dAdam Lesinski    public void requestActivityInfo(ResultReceiver result) {
748c48fd721a9a3273f0faa927d93caad4c9832af7dAdam Lesinski        Bundle bundle = new Bundle();
749c48fd721a9a3273f0faa927d93caad4c9832af7dAdam Lesinski        bundle.putParcelable(BatteryStats.RESULT_RECEIVER_CONTROLLER_KEY, reportActivityInfo());
750c48fd721a9a3273f0faa927d93caad4c9832af7dAdam Lesinski        result.send(0, bundle);
751c48fd721a9a3273f0faa927d93caad4c9832af7dAdam Lesinski    }
752c48fd721a9a3273f0faa927d93caad4c9832af7dAdam Lesinski
753a632d8a6edd350c7644b593b18eceaa5b368505bVinit Deshpande    /**
75494a2dd090b7d3823058b7a69cf4653d5663708cdAdam Lesinski     * see {@link android.net.wifi.WifiManager#getControllerActivityEnergyInfo(int)}
755200e8ee5097134010a6edee8d031bb02ff7eeb5avandwalle     */
7568fcd1fe86ddc3056e1a26e69bc68f148ae0e9ddcChristopher Wiley    @Override
757048dc8df2bd9394979dbb090b0cfe8d6627d23e3Vinit Deshpande    public WifiActivityEnergyInfo reportActivityInfo() {
758200e8ee5097134010a6edee8d031bb02ff7eeb5avandwalle        enforceAccessPermission();
7591f8f2971309aee4de9c35723f6fbca6fcab9a3beMitchell Wills        if ((getSupportedFeatures() & WifiManager.WIFI_FEATURE_LINK_LAYER_STATS) == 0) {
7601f8f2971309aee4de9c35723f6fbca6fcab9a3beMitchell Wills            return null;
7611f8f2971309aee4de9c35723f6fbca6fcab9a3beMitchell Wills        }
762200e8ee5097134010a6edee8d031bb02ff7eeb5avandwalle        WifiLinkLayerStats stats;
763200e8ee5097134010a6edee8d031bb02ff7eeb5avandwalle        WifiActivityEnergyInfo energyInfo = null;
764200e8ee5097134010a6edee8d031bb02ff7eeb5avandwalle        if (mWifiStateMachineChannel != null) {
765048dc8df2bd9394979dbb090b0cfe8d6627d23e3Vinit Deshpande            stats = mWifiStateMachine.syncGetLinkLayerStats(mWifiStateMachineChannel);
766200e8ee5097134010a6edee8d031bb02ff7eeb5avandwalle            if (stats != null) {
76794a2dd090b7d3823058b7a69cf4653d5663708cdAdam Lesinski                final long rxIdleCurrent = mContext.getResources().getInteger(
76894a2dd090b7d3823058b7a69cf4653d5663708cdAdam Lesinski                        com.android.internal.R.integer.config_wifi_idle_receive_cur_ma);
76994a2dd090b7d3823058b7a69cf4653d5663708cdAdam Lesinski                final long rxCurrent = mContext.getResources().getInteger(
77094a2dd090b7d3823058b7a69cf4653d5663708cdAdam Lesinski                        com.android.internal.R.integer.config_wifi_active_rx_cur_ma);
77194a2dd090b7d3823058b7a69cf4653d5663708cdAdam Lesinski                final long txCurrent = mContext.getResources().getInteger(
77294a2dd090b7d3823058b7a69cf4653d5663708cdAdam Lesinski                        com.android.internal.R.integer.config_wifi_tx_cur_ma);
77394a2dd090b7d3823058b7a69cf4653d5663708cdAdam Lesinski                final double voltage = mContext.getResources().getInteger(
77494a2dd090b7d3823058b7a69cf4653d5663708cdAdam Lesinski                        com.android.internal.R.integer.config_wifi_operating_voltage_mv)
77594a2dd090b7d3823058b7a69cf4653d5663708cdAdam Lesinski                        / 1000.0;
77694a2dd090b7d3823058b7a69cf4653d5663708cdAdam Lesinski
77794a2dd090b7d3823058b7a69cf4653d5663708cdAdam Lesinski                final long rxIdleTime = stats.on_time - stats.tx_time - stats.rx_time;
7789c8892ebd2ec5340785f6b787805fdc6088069d7Roshan Pius                final long[] txTimePerLevel;
7799c8892ebd2ec5340785f6b787805fdc6088069d7Roshan Pius                if (stats.tx_time_per_level != null) {
7809c8892ebd2ec5340785f6b787805fdc6088069d7Roshan Pius                    txTimePerLevel = new long[stats.tx_time_per_level.length];
7819c8892ebd2ec5340785f6b787805fdc6088069d7Roshan Pius                    for (int i = 0; i < txTimePerLevel.length; i++) {
7829c8892ebd2ec5340785f6b787805fdc6088069d7Roshan Pius                        txTimePerLevel[i] = stats.tx_time_per_level[i];
7839c8892ebd2ec5340785f6b787805fdc6088069d7Roshan Pius                        // TODO(b/27227497): Need to read the power consumed per level from config
7849c8892ebd2ec5340785f6b787805fdc6088069d7Roshan Pius                    }
7859c8892ebd2ec5340785f6b787805fdc6088069d7Roshan Pius                } else {
7869c8892ebd2ec5340785f6b787805fdc6088069d7Roshan Pius                    // This will happen if the HAL get link layer API returned null.
7879c8892ebd2ec5340785f6b787805fdc6088069d7Roshan Pius                    txTimePerLevel = new long[0];
788a1514cb347fccf11566e5172ed71030c695d2abdRoshan Pius                }
78994a2dd090b7d3823058b7a69cf4653d5663708cdAdam Lesinski                final long energyUsed = (long)((stats.tx_time * txCurrent +
79094a2dd090b7d3823058b7a69cf4653d5663708cdAdam Lesinski                        stats.rx_time * rxCurrent +
79194a2dd090b7d3823058b7a69cf4653d5663708cdAdam Lesinski                        rxIdleTime * rxIdleCurrent) * voltage);
792d76cdd8e08476ad5025f07a5d77a8d8f920a0721Adam Lesinski                if (VDBG || rxIdleTime < 0 || stats.on_time < 0 || stats.tx_time < 0 ||
793d76cdd8e08476ad5025f07a5d77a8d8f920a0721Adam Lesinski                        stats.rx_time < 0 || energyUsed < 0) {
79470b75cda33a05e8b6e6724ed08e4e72f6fee0471Pierre Vandwalle                    StringBuilder sb = new StringBuilder();
79570b75cda33a05e8b6e6724ed08e4e72f6fee0471Pierre Vandwalle                    sb.append(" rxIdleCur=" + rxIdleCurrent);
79670b75cda33a05e8b6e6724ed08e4e72f6fee0471Pierre Vandwalle                    sb.append(" rxCur=" + rxCurrent);
79770b75cda33a05e8b6e6724ed08e4e72f6fee0471Pierre Vandwalle                    sb.append(" txCur=" + txCurrent);
79870b75cda33a05e8b6e6724ed08e4e72f6fee0471Pierre Vandwalle                    sb.append(" voltage=" + voltage);
79970b75cda33a05e8b6e6724ed08e4e72f6fee0471Pierre Vandwalle                    sb.append(" on_time=" + stats.on_time);
80070b75cda33a05e8b6e6724ed08e4e72f6fee0471Pierre Vandwalle                    sb.append(" tx_time=" + stats.tx_time);
8019c8892ebd2ec5340785f6b787805fdc6088069d7Roshan Pius                    sb.append(" tx_time_per_level=" + Arrays.toString(txTimePerLevel));
80270b75cda33a05e8b6e6724ed08e4e72f6fee0471Pierre Vandwalle                    sb.append(" rx_time=" + stats.rx_time);
80370b75cda33a05e8b6e6724ed08e4e72f6fee0471Pierre Vandwalle                    sb.append(" rxIdleTime=" + rxIdleTime);
80470b75cda33a05e8b6e6724ed08e4e72f6fee0471Pierre Vandwalle                    sb.append(" energy=" + energyUsed);
805f9946f5663f2d7a9a69cb8b7e82a4ef8b1825c48Mitchell Wills                    Log.d(TAG, " reportActivityInfo: " + sb.toString());
80670b75cda33a05e8b6e6724ed08e4e72f6fee0471Pierre Vandwalle                }
80784a11a3d54b7c6f043dcf7a874b7c0d45709677aPierre Vandwalle
808200e8ee5097134010a6edee8d031bb02ff7eeb5avandwalle                // Convert the LinkLayerStats into EnergyActivity
809186f7513a080eb1a419b6d83286d62961168a5feAdam Lesinski                energyInfo = new WifiActivityEnergyInfo(SystemClock.elapsedRealtime(),
810200e8ee5097134010a6edee8d031bb02ff7eeb5avandwalle                        WifiActivityEnergyInfo.STACK_STATE_STATE_IDLE, stats.tx_time,
811a1514cb347fccf11566e5172ed71030c695d2abdRoshan Pius                        txTimePerLevel, stats.rx_time, rxIdleTime, energyUsed);
812200e8ee5097134010a6edee8d031bb02ff7eeb5avandwalle            }
8131f8f2971309aee4de9c35723f6fbca6fcab9a3beMitchell Wills            if (energyInfo != null && energyInfo.isValid()) {
8141f8f2971309aee4de9c35723f6fbca6fcab9a3beMitchell Wills                return energyInfo;
8151f8f2971309aee4de9c35723f6fbca6fcab9a3beMitchell Wills            } else {
8161f8f2971309aee4de9c35723f6fbca6fcab9a3beMitchell Wills                return null;
8171f8f2971309aee4de9c35723f6fbca6fcab9a3beMitchell Wills            }
818200e8ee5097134010a6edee8d031bb02ff7eeb5avandwalle        } else {
819200e8ee5097134010a6edee8d031bb02ff7eeb5avandwalle            Slog.e(TAG, "mWifiStateMachineChannel is not initialized");
820200e8ee5097134010a6edee8d031bb02ff7eeb5avandwalle            return null;
821200e8ee5097134010a6edee8d031bb02ff7eeb5avandwalle        }
822200e8ee5097134010a6edee8d031bb02ff7eeb5avandwalle    }
823200e8ee5097134010a6edee8d031bb02ff7eeb5avandwalle
824200e8ee5097134010a6edee8d031bb02ff7eeb5avandwalle    /**
825155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande     * see {@link android.net.wifi.WifiManager#getConfiguredNetworks()}
826155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande     * @return the list of configured networks
827155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande     */
8288fcd1fe86ddc3056e1a26e69bc68f148ae0e9ddcChristopher Wiley    @Override
829155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    public List<WifiConfiguration> getConfiguredNetworks() {
830155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        enforceAccessPermission();
831155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        if (mWifiStateMachineChannel != null) {
83282199a285f4a45a46b44eb8253999aa918534753vandwalle            return mWifiStateMachine.syncGetConfiguredNetworks(Binder.getCallingUid(),
83382199a285f4a45a46b44eb8253999aa918534753vandwalle                    mWifiStateMachineChannel);
834155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        } else {
835155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande            Slog.e(TAG, "mWifiStateMachineChannel is not initialized");
836155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande            return null;
837155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        }
838155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    }
839155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
840155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    /**
8419878c61bbd81176561991be025af44efc67332feWenchao Tong     * see {@link android.net.wifi.WifiManager#getPrivilegedConfiguredNetworks()}
8429878c61bbd81176561991be025af44efc67332feWenchao Tong     * @return the list of configured networks with real preSharedKey
8439878c61bbd81176561991be025af44efc67332feWenchao Tong     */
8448fcd1fe86ddc3056e1a26e69bc68f148ae0e9ddcChristopher Wiley    @Override
8459878c61bbd81176561991be025af44efc67332feWenchao Tong    public List<WifiConfiguration> getPrivilegedConfiguredNetworks() {
8469878c61bbd81176561991be025af44efc67332feWenchao Tong        enforceReadCredentialPermission();
8479878c61bbd81176561991be025af44efc67332feWenchao Tong        enforceAccessPermission();
8489878c61bbd81176561991be025af44efc67332feWenchao Tong        if (mWifiStateMachineChannel != null) {
8499878c61bbd81176561991be025af44efc67332feWenchao Tong            return mWifiStateMachine.syncGetPrivilegedConfiguredNetwork(mWifiStateMachineChannel);
8509878c61bbd81176561991be025af44efc67332feWenchao Tong        } else {
8519878c61bbd81176561991be025af44efc67332feWenchao Tong            Slog.e(TAG, "mWifiStateMachineChannel is not initialized");
8529878c61bbd81176561991be025af44efc67332feWenchao Tong            return null;
8539878c61bbd81176561991be025af44efc67332feWenchao Tong        }
8549878c61bbd81176561991be025af44efc67332feWenchao Tong    }
8559878c61bbd81176561991be025af44efc67332feWenchao Tong
8569878c61bbd81176561991be025af44efc67332feWenchao Tong    /**
857ccbe820337a230f0827f734aca2c51a244a0837fVinit Deshpande     * Returns a WifiConfiguration matching this ScanResult
858ccbe820337a230f0827f734aca2c51a244a0837fVinit Deshpande     * @param scanResult scanResult that represents the BSSID
859ccbe820337a230f0827f734aca2c51a244a0837fVinit Deshpande     * @return {@link WifiConfiguration} that matches this BSSID or null
860ccbe820337a230f0827f734aca2c51a244a0837fVinit Deshpande     */
8618fcd1fe86ddc3056e1a26e69bc68f148ae0e9ddcChristopher Wiley    @Override
862ccbe820337a230f0827f734aca2c51a244a0837fVinit Deshpande    public WifiConfiguration getMatchingWifiConfig(ScanResult scanResult) {
863ccbe820337a230f0827f734aca2c51a244a0837fVinit Deshpande        enforceAccessPermission();
864ccbe820337a230f0827f734aca2c51a244a0837fVinit Deshpande        return mWifiStateMachine.syncGetMatchingWifiConfig(scanResult, mWifiStateMachineChannel);
865ccbe820337a230f0827f734aca2c51a244a0837fVinit Deshpande    }
866ccbe820337a230f0827f734aca2c51a244a0837fVinit Deshpande
867ccbe820337a230f0827f734aca2c51a244a0837fVinit Deshpande    /**
868155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande     * see {@link android.net.wifi.WifiManager#addOrUpdateNetwork(WifiConfiguration)}
869155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande     * @return the supplicant-assigned identifier for the new or updated
870155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande     * network if the operation succeeds, or {@code -1} if it fails
871155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande     */
8728fcd1fe86ddc3056e1a26e69bc68f148ae0e9ddcChristopher Wiley    @Override
873155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    public int addOrUpdateNetwork(WifiConfiguration config) {
874155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        enforceChangePermission();
8754aef3283bf78e233a11fa34af2c7c38362879625Jan Nordqvist        if (isValid(config) && isValidPasspoint(config)) {
8768be2ac8cec40a2000715849b627642bdc87c10b3Shirish Kalele
87731891afce40b903ada9b24ec12e3648ae6aa27b2Jan Nordqvist            WifiEnterpriseConfig enterpriseConfig = config.enterpriseConfig;
87831891afce40b903ada9b24ec12e3648ae6aa27b2Jan Nordqvist
8792e814680f4dd27a5f825afab189843582235cedcJan Nordqvist            if (config.isPasspoint() &&
8802e814680f4dd27a5f825afab189843582235cedcJan Nordqvist                    (enterpriseConfig.getEapMethod() == WifiEnterpriseConfig.Eap.TLS ||
8812e814680f4dd27a5f825afab189843582235cedcJan Nordqvist                            enterpriseConfig.getEapMethod() == WifiEnterpriseConfig.Eap.TTLS)) {
8822e814680f4dd27a5f825afab189843582235cedcJan Nordqvist                if (config.updateIdentifier != null) {
8832e814680f4dd27a5f825afab189843582235cedcJan Nordqvist                    enforceAccessPermission();
8842e814680f4dd27a5f825afab189843582235cedcJan Nordqvist                }
8852e814680f4dd27a5f825afab189843582235cedcJan Nordqvist                else {
886e511ec7a4f53f7204a85075b23318f9f11c46d94Mitchell Wills                    try {
887e511ec7a4f53f7204a85075b23318f9f11c46d94Mitchell Wills                        verifyCert(enterpriseConfig.getCaCertificate());
888e511ec7a4f53f7204a85075b23318f9f11c46d94Mitchell Wills                    } catch (CertPathValidatorException cpve) {
889e511ec7a4f53f7204a85075b23318f9f11c46d94Mitchell Wills                        Slog.e(TAG, "CA Cert " +
890e511ec7a4f53f7204a85075b23318f9f11c46d94Mitchell Wills                                enterpriseConfig.getCaCertificate().getSubjectX500Principal() +
891e511ec7a4f53f7204a85075b23318f9f11c46d94Mitchell Wills                                " untrusted: " + cpve.getMessage());
892e511ec7a4f53f7204a85075b23318f9f11c46d94Mitchell Wills                        return -1;
893e511ec7a4f53f7204a85075b23318f9f11c46d94Mitchell Wills                    } catch (GeneralSecurityException | IOException e) {
894e511ec7a4f53f7204a85075b23318f9f11c46d94Mitchell Wills                        Slog.e(TAG, "Failed to verify certificate" +
895e511ec7a4f53f7204a85075b23318f9f11c46d94Mitchell Wills                                enterpriseConfig.getCaCertificate().getSubjectX500Principal() +
896e511ec7a4f53f7204a85075b23318f9f11c46d94Mitchell Wills                                ": " + e);
897e511ec7a4f53f7204a85075b23318f9f11c46d94Mitchell Wills                        return -1;
898e511ec7a4f53f7204a85075b23318f9f11c46d94Mitchell Wills                    }
89931891afce40b903ada9b24ec12e3648ae6aa27b2Jan Nordqvist                }
90031891afce40b903ada9b24ec12e3648ae6aa27b2Jan Nordqvist            }
90131891afce40b903ada9b24ec12e3648ae6aa27b2Jan Nordqvist
902992ae00f25a9cc22cf5db3261bd7e72927069cf7vandwalle            //TODO: pass the Uid the WifiStateMachine as a message parameter
90331891afce40b903ada9b24ec12e3648ae6aa27b2Jan Nordqvist            Slog.i("addOrUpdateNetwork", " uid = " + Integer.toString(Binder.getCallingUid())
904992ae00f25a9cc22cf5db3261bd7e72927069cf7vandwalle                    + " SSID " + config.SSID
905992ae00f25a9cc22cf5db3261bd7e72927069cf7vandwalle                    + " nid=" + Integer.toString(config.networkId));
906992ae00f25a9cc22cf5db3261bd7e72927069cf7vandwalle            if (config.networkId == WifiConfiguration.INVALID_NETWORK_ID) {
907992ae00f25a9cc22cf5db3261bd7e72927069cf7vandwalle                config.creatorUid = Binder.getCallingUid();
908992ae00f25a9cc22cf5db3261bd7e72927069cf7vandwalle            } else {
909992ae00f25a9cc22cf5db3261bd7e72927069cf7vandwalle                config.lastUpdateUid = Binder.getCallingUid();
910992ae00f25a9cc22cf5db3261bd7e72927069cf7vandwalle            }
911155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande            if (mWifiStateMachineChannel != null) {
912155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande                return mWifiStateMachine.syncAddOrUpdateNetwork(mWifiStateMachineChannel, config);
913155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande            } else {
914155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande                Slog.e(TAG, "mWifiStateMachineChannel is not initialized");
915155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande                return -1;
916155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande            }
917155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        } else {
918155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande            Slog.e(TAG, "bad network configuration");
919155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande            return -1;
920155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        }
921155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    }
922155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
92331891afce40b903ada9b24ec12e3648ae6aa27b2Jan Nordqvist    public static void verifyCert(X509Certificate caCert)
92431891afce40b903ada9b24ec12e3648ae6aa27b2Jan Nordqvist            throws GeneralSecurityException, IOException {
92531891afce40b903ada9b24ec12e3648ae6aa27b2Jan Nordqvist        CertificateFactory factory = CertificateFactory.getInstance("X.509");
92631891afce40b903ada9b24ec12e3648ae6aa27b2Jan Nordqvist        CertPathValidator validator =
92731891afce40b903ada9b24ec12e3648ae6aa27b2Jan Nordqvist                CertPathValidator.getInstance(CertPathValidator.getDefaultType());
92831891afce40b903ada9b24ec12e3648ae6aa27b2Jan Nordqvist        CertPath path = factory.generateCertPath(
92931891afce40b903ada9b24ec12e3648ae6aa27b2Jan Nordqvist                Arrays.asList(caCert));
93031891afce40b903ada9b24ec12e3648ae6aa27b2Jan Nordqvist        KeyStore ks = KeyStore.getInstance("AndroidCAStore");
93131891afce40b903ada9b24ec12e3648ae6aa27b2Jan Nordqvist        ks.load(null, null);
93231891afce40b903ada9b24ec12e3648ae6aa27b2Jan Nordqvist        PKIXParameters params = new PKIXParameters(ks);
93331891afce40b903ada9b24ec12e3648ae6aa27b2Jan Nordqvist        params.setRevocationEnabled(false);
93431891afce40b903ada9b24ec12e3648ae6aa27b2Jan Nordqvist        validator.validate(path, params);
93531891afce40b903ada9b24ec12e3648ae6aa27b2Jan Nordqvist    }
93631891afce40b903ada9b24ec12e3648ae6aa27b2Jan Nordqvist
93731891afce40b903ada9b24ec12e3648ae6aa27b2Jan Nordqvist    /**
938155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande     * See {@link android.net.wifi.WifiManager#removeNetwork(int)}
939155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande     * @param netId the integer that identifies the network configuration
940155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande     * to the supplicant
941155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande     * @return {@code true} if the operation succeeded
942155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande     */
9438fcd1fe86ddc3056e1a26e69bc68f148ae0e9ddcChristopher Wiley    @Override
944155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    public boolean removeNetwork(int netId) {
945155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        enforceChangePermission();
946c35d728a15e9270f5642ef79f5245c89d749285fSky Faber
947155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        if (mWifiStateMachineChannel != null) {
948155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande            return mWifiStateMachine.syncRemoveNetwork(mWifiStateMachineChannel, netId);
949155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        } else {
950155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande            Slog.e(TAG, "mWifiStateMachineChannel is not initialized");
951155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande            return false;
952155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        }
953155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    }
954155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
955155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    /**
956155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande     * See {@link android.net.wifi.WifiManager#enableNetwork(int, boolean)}
957155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande     * @param netId the integer that identifies the network configuration
958155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande     * to the supplicant
959155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande     * @param disableOthers if true, disable all other networks.
960155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande     * @return {@code true} if the operation succeeded
961155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande     */
9628fcd1fe86ddc3056e1a26e69bc68f148ae0e9ddcChristopher Wiley    @Override
963155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    public boolean enableNetwork(int netId, boolean disableOthers) {
964155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        enforceChangePermission();
965155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        if (mWifiStateMachineChannel != null) {
966155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande            return mWifiStateMachine.syncEnableNetwork(mWifiStateMachineChannel, netId,
967155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande                    disableOthers);
968155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        } else {
969155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande            Slog.e(TAG, "mWifiStateMachineChannel is not initialized");
970155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande            return false;
971155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        }
972155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    }
973155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
974155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    /**
975155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande     * See {@link android.net.wifi.WifiManager#disableNetwork(int)}
976155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande     * @param netId the integer that identifies the network configuration
977155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande     * to the supplicant
978155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande     * @return {@code true} if the operation succeeded
979155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande     */
9808fcd1fe86ddc3056e1a26e69bc68f148ae0e9ddcChristopher Wiley    @Override
981155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    public boolean disableNetwork(int netId) {
982155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        enforceChangePermission();
983155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        if (mWifiStateMachineChannel != null) {
984155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande            return mWifiStateMachine.syncDisableNetwork(mWifiStateMachineChannel, netId);
985155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        } else {
986155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande            Slog.e(TAG, "mWifiStateMachineChannel is not initialized");
987155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande            return false;
988155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        }
989155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    }
990155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
991155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    /**
992155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande     * See {@link android.net.wifi.WifiManager#getConnectionInfo()}
993155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande     * @return the Wi-Fi information, contained in {@link WifiInfo}.
994155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande     */
9958fcd1fe86ddc3056e1a26e69bc68f148ae0e9ddcChristopher Wiley    @Override
996155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    public WifiInfo getConnectionInfo() {
997155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        enforceAccessPermission();
998155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        /*
999155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande         * Make sure we have the latest information, by sending
1000155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande         * a status request to the supplicant.
1001155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande         */
1002155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        return mWifiStateMachine.syncRequestConnectionInfo();
1003155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    }
1004155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
1005155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    /**
1006155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande     * Return the results of the most recent access point scan, in the form of
1007155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande     * a list of {@link ScanResult} objects.
1008155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande     * @return the list of results
1009155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande     */
10108fcd1fe86ddc3056e1a26e69bc68f148ae0e9ddcChristopher Wiley    @Override
1011155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    public List<ScanResult> getScanResults(String callingPackage) {
1012155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        enforceAccessPermission();
1013155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        int uid = Binder.getCallingUid();
1014155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        long ident = Binder.clearCallingIdentity();
1015155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        try {
101692f4d52b14d8847f6f81712bf3bbfa0f1203eef5Sohani Rao            if (!mWifiPermissionsUtil.canAccessScanResults(callingPackage,
101792f4d52b14d8847f6f81712bf3bbfa0f1203eef5Sohani Rao                      uid, Build.VERSION_CODES.M)) {
10181ecb0083490436303cdb89bc4c46b6743ea0afa8Sky Faber                return new ArrayList<ScanResult>();
10191ecb0083490436303cdb89bc4c46b6743ea0afa8Sky Faber            }
1020d94a62ffdf0e7f282948109d05d96e682eb32eefRebecca Silberstein            if (mWifiScanner == null) {
1021d94a62ffdf0e7f282948109d05d96e682eb32eefRebecca Silberstein                mWifiScanner = mWifiInjector.getWifiScanner();
1022d94a62ffdf0e7f282948109d05d96e682eb32eefRebecca Silberstein            }
1023d94a62ffdf0e7f282948109d05d96e682eb32eefRebecca Silberstein            return mWifiScanner.getSingleScanResults();
1024155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        } finally {
1025155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande            Binder.restoreCallingIdentity(ident);
1026155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        }
1027155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    }
1028155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
10291d5cd3938f9191184cd9aea3059a3b62bf3a0372Jan Nordqvist    /**
10303ebf0b17fd1cdd3a0ef22d9e199c1686c01dcc9fPeter Qiu     * Add a Passpoint configuration.
10313ebf0b17fd1cdd3a0ef22d9e199c1686c01dcc9fPeter Qiu     *
10323ebf0b17fd1cdd3a0ef22d9e199c1686c01dcc9fPeter Qiu     * @param config The Passpoint configuration to be added
10333ebf0b17fd1cdd3a0ef22d9e199c1686c01dcc9fPeter Qiu     * @return true on success or false on failure
10343ebf0b17fd1cdd3a0ef22d9e199c1686c01dcc9fPeter Qiu     */
10353ebf0b17fd1cdd3a0ef22d9e199c1686c01dcc9fPeter Qiu    @Override
10363ebf0b17fd1cdd3a0ef22d9e199c1686c01dcc9fPeter Qiu    public boolean addPasspointConfiguration(PasspointConfiguration config) {
1037c04479601696f56c9cc240d4c7fc49fa99f51825Peter Qiu        enforceChangePermission();
1038c04479601696f56c9cc240d4c7fc49fa99f51825Peter Qiu        return mPasspointManager.addProvider(config);
10393ebf0b17fd1cdd3a0ef22d9e199c1686c01dcc9fPeter Qiu    }
10403ebf0b17fd1cdd3a0ef22d9e199c1686c01dcc9fPeter Qiu
10413ebf0b17fd1cdd3a0ef22d9e199c1686c01dcc9fPeter Qiu    /**
10423ebf0b17fd1cdd3a0ef22d9e199c1686c01dcc9fPeter Qiu     * Remove the Passpoint configuration identified by its FQDN (Fully Qualified Domain Name).
10433ebf0b17fd1cdd3a0ef22d9e199c1686c01dcc9fPeter Qiu     *
10443ebf0b17fd1cdd3a0ef22d9e199c1686c01dcc9fPeter Qiu     * @param fqdn The FQDN of the Passpoint configuration to be removed
10453ebf0b17fd1cdd3a0ef22d9e199c1686c01dcc9fPeter Qiu     * @return true on success or false on failure
10463ebf0b17fd1cdd3a0ef22d9e199c1686c01dcc9fPeter Qiu     */
10473ebf0b17fd1cdd3a0ef22d9e199c1686c01dcc9fPeter Qiu    @Override
10483ebf0b17fd1cdd3a0ef22d9e199c1686c01dcc9fPeter Qiu    public boolean removePasspointConfiguration(String fqdn) {
1049c04479601696f56c9cc240d4c7fc49fa99f51825Peter Qiu        enforceChangePermission();
1050c04479601696f56c9cc240d4c7fc49fa99f51825Peter Qiu        return mPasspointManager.removeProvider(fqdn);
10513ebf0b17fd1cdd3a0ef22d9e199c1686c01dcc9fPeter Qiu    }
10523ebf0b17fd1cdd3a0ef22d9e199c1686c01dcc9fPeter Qiu
10533ebf0b17fd1cdd3a0ef22d9e199c1686c01dcc9fPeter Qiu    /**
10543ebf0b17fd1cdd3a0ef22d9e199c1686c01dcc9fPeter Qiu     * Return the list of the installed Passpoint configurations.
10553ebf0b17fd1cdd3a0ef22d9e199c1686c01dcc9fPeter Qiu     *
10563ebf0b17fd1cdd3a0ef22d9e199c1686c01dcc9fPeter Qiu     * @return A list of {@link PasspointConfiguration} or null
10573ebf0b17fd1cdd3a0ef22d9e199c1686c01dcc9fPeter Qiu     */
10583ebf0b17fd1cdd3a0ef22d9e199c1686c01dcc9fPeter Qiu    @Override
10593ebf0b17fd1cdd3a0ef22d9e199c1686c01dcc9fPeter Qiu    public List<PasspointConfiguration> getPasspointConfigurations() {
1060c04479601696f56c9cc240d4c7fc49fa99f51825Peter Qiu        enforceAccessPermission();
1061c04479601696f56c9cc240d4c7fc49fa99f51825Peter Qiu        return mPasspointManager.getProviderConfigs();
10623ebf0b17fd1cdd3a0ef22d9e199c1686c01dcc9fPeter Qiu    }
10633ebf0b17fd1cdd3a0ef22d9e199c1686c01dcc9fPeter Qiu
10643ebf0b17fd1cdd3a0ef22d9e199c1686c01dcc9fPeter Qiu    /**
10652e814680f4dd27a5f825afab189843582235cedcJan Nordqvist     * Query for a Hotspot 2.0 release 2 OSU icon
10662e814680f4dd27a5f825afab189843582235cedcJan Nordqvist     * @param bssid The BSSID of the AP
10672e814680f4dd27a5f825afab189843582235cedcJan Nordqvist     * @param fileName Icon file name
10682e814680f4dd27a5f825afab189843582235cedcJan Nordqvist     */
10698fcd1fe86ddc3056e1a26e69bc68f148ae0e9ddcChristopher Wiley    @Override
10702e814680f4dd27a5f825afab189843582235cedcJan Nordqvist    public void queryPasspointIcon(long bssid, String fileName) {
10712e814680f4dd27a5f825afab189843582235cedcJan Nordqvist        mWifiStateMachine.syncQueryPasspointIcon(mWifiStateMachineChannel, bssid, fileName);
10722e814680f4dd27a5f825afab189843582235cedcJan Nordqvist    }
10731d5cd3938f9191184cd9aea3059a3b62bf3a0372Jan Nordqvist
10742e814680f4dd27a5f825afab189843582235cedcJan Nordqvist    /**
10752e814680f4dd27a5f825afab189843582235cedcJan Nordqvist     * Match the currently associated network against the SP matching the given FQDN
10762e814680f4dd27a5f825afab189843582235cedcJan Nordqvist     * @param fqdn FQDN of the SP
10772e814680f4dd27a5f825afab189843582235cedcJan Nordqvist     * @return ordinal [HomeProvider, RoamingProvider, Incomplete, None, Declined]
10782e814680f4dd27a5f825afab189843582235cedcJan Nordqvist     */
10798fcd1fe86ddc3056e1a26e69bc68f148ae0e9ddcChristopher Wiley    @Override
10802e814680f4dd27a5f825afab189843582235cedcJan Nordqvist    public int matchProviderWithCurrentNetwork(String fqdn) {
10812e814680f4dd27a5f825afab189843582235cedcJan Nordqvist        return mWifiStateMachine.matchProviderWithCurrentNetwork(mWifiStateMachineChannel, fqdn);
10821d5cd3938f9191184cd9aea3059a3b62bf3a0372Jan Nordqvist    }
10831d5cd3938f9191184cd9aea3059a3b62bf3a0372Jan Nordqvist
10842e814680f4dd27a5f825afab189843582235cedcJan Nordqvist    /**
10852e814680f4dd27a5f825afab189843582235cedcJan Nordqvist     * Deauthenticate and set the re-authentication hold off time for the current network
10862e814680f4dd27a5f825afab189843582235cedcJan Nordqvist     * @param holdoff hold off time in milliseconds
10872e814680f4dd27a5f825afab189843582235cedcJan Nordqvist     * @param ess set if the hold off pertains to an ESS rather than a BSS
10882e814680f4dd27a5f825afab189843582235cedcJan Nordqvist     */
10898fcd1fe86ddc3056e1a26e69bc68f148ae0e9ddcChristopher Wiley    @Override
10902e814680f4dd27a5f825afab189843582235cedcJan Nordqvist    public void deauthenticateNetwork(long holdoff, boolean ess) {
10912e814680f4dd27a5f825afab189843582235cedcJan Nordqvist        mWifiStateMachine.deauthenticateNetwork(mWifiStateMachineChannel, holdoff, ess);
10921d5cd3938f9191184cd9aea3059a3b62bf3a0372Jan Nordqvist    }
10931d5cd3938f9191184cd9aea3059a3b62bf3a0372Jan Nordqvist
1094fd5470391e5342daa38d00b68ccbccfeacbe6d33Alexandra Gherghina    /**
1095155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande     * Tell the supplicant to persist the current list of configured networks.
1096155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande     * @return {@code true} if the operation succeeded
1097155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande     *
1098155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande     * TODO: deprecate this
1099155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande     */
11008fcd1fe86ddc3056e1a26e69bc68f148ae0e9ddcChristopher Wiley    @Override
1101155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    public boolean saveConfiguration() {
1102155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        enforceChangePermission();
1103155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        if (mWifiStateMachineChannel != null) {
1104155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande            return mWifiStateMachine.syncSaveConfig(mWifiStateMachineChannel);
1105155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        } else {
1106155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande            Slog.e(TAG, "mWifiStateMachineChannel is not initialized");
1107155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande            return false;
1108155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        }
1109155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    }
1110155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
1111155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    /**
1112155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande     * Set the country code
1113155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande     * @param countryCode ISO 3166 country code.
1114155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande     * @param persist {@code true} if the setting should be remembered.
1115155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande     *
1116155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande     * The persist behavior exists so that wifi can fall back to the last
1117155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande     * persisted country code on a restart, when the locale information is
1118155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande     * not available from telephony.
1119155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande     */
11208fcd1fe86ddc3056e1a26e69bc68f148ae0e9ddcChristopher Wiley    @Override
1121155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    public void setCountryCode(String countryCode, boolean persist) {
1122155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        Slog.i(TAG, "WifiService trying to set country code to " + countryCode +
1123155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande                " with persist set to " + persist);
11249c22c8ea193360f50ad490368ca6a11af0684362Robert Greenwalt        enforceConnectivityInternalPermission();
1125155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        final long token = Binder.clearCallingIdentity();
1126155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        try {
11270091305175e8c6fe7fc6d01efb9d405961db4ac7Ningyuan Wang            if (mCountryCode.setCountryCode(countryCode, persist) && persist) {
11280091305175e8c6fe7fc6d01efb9d405961db4ac7Ningyuan Wang                // Save this country code to persistent storage
11290091305175e8c6fe7fc6d01efb9d405961db4ac7Ningyuan Wang                mFacade.setStringSetting(mContext,
11300091305175e8c6fe7fc6d01efb9d405961db4ac7Ningyuan Wang                        Settings.Global.WIFI_COUNTRY_CODE,
11310091305175e8c6fe7fc6d01efb9d405961db4ac7Ningyuan Wang                        countryCode);
11320091305175e8c6fe7fc6d01efb9d405961db4ac7Ningyuan Wang            }
1133155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        } finally {
1134155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande            Binder.restoreCallingIdentity(token);
1135155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        }
1136155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    }
1137155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
113880d14d67a956e6c69433470aaa73100898166efaxinhe     /**
113980d14d67a956e6c69433470aaa73100898166efaxinhe     * Get the country code
114037b06cd7aae7fe27cfaf1d95cc9901548765406bNingyuan Wang     * @return Get the best choice country code for wifi, regardless of if it was set or
114137b06cd7aae7fe27cfaf1d95cc9901548765406bNingyuan Wang     * not.
114237b06cd7aae7fe27cfaf1d95cc9901548765406bNingyuan Wang     * Returns null when there is no country code available.
114380d14d67a956e6c69433470aaa73100898166efaxinhe     */
11448fcd1fe86ddc3056e1a26e69bc68f148ae0e9ddcChristopher Wiley    @Override
114580d14d67a956e6c69433470aaa73100898166efaxinhe    public String getCountryCode() {
114680d14d67a956e6c69433470aaa73100898166efaxinhe        enforceConnectivityInternalPermission();
114737b06cd7aae7fe27cfaf1d95cc9901548765406bNingyuan Wang        String country = mCountryCode.getCountryCode();
114880d14d67a956e6c69433470aaa73100898166efaxinhe        return country;
114980d14d67a956e6c69433470aaa73100898166efaxinhe    }
1150155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
11518fcd1fe86ddc3056e1a26e69bc68f148ae0e9ddcChristopher Wiley    @Override
1152155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    public boolean isDualBandSupported() {
1153155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        //TODO: Should move towards adding a driver API that checks at runtime
1154155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        return mContext.getResources().getBoolean(
1155155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande                com.android.internal.R.bool.config_wifi_dual_band_support);
1156155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    }
1157155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
1158155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    /**
1159155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande     * Return the DHCP-assigned addresses from the last successful DHCP request,
1160155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande     * if any.
1161155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande     * @return the DHCP information
1162155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande     * @deprecated
1163155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande     */
11648fcd1fe86ddc3056e1a26e69bc68f148ae0e9ddcChristopher Wiley    @Override
11658fcd1fe86ddc3056e1a26e69bc68f148ae0e9ddcChristopher Wiley    @Deprecated
1166155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    public DhcpInfo getDhcpInfo() {
1167155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        enforceAccessPermission();
1168155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        DhcpResults dhcpResults = mWifiStateMachine.syncGetDhcpResults();
1169155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
1170155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        DhcpInfo info = new DhcpInfo();
11713b26801d62a06475b722bbf29cba7f48f376654eLorenzo Colitti
11723b26801d62a06475b722bbf29cba7f48f376654eLorenzo Colitti        if (dhcpResults.ipAddress != null &&
11733b26801d62a06475b722bbf29cba7f48f376654eLorenzo Colitti                dhcpResults.ipAddress.getAddress() instanceof Inet4Address) {
11743b26801d62a06475b722bbf29cba7f48f376654eLorenzo Colitti            info.ipAddress = NetworkUtils.inetAddressToInt((Inet4Address) dhcpResults.ipAddress.getAddress());
1175155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        }
11763b26801d62a06475b722bbf29cba7f48f376654eLorenzo Colitti
11773b26801d62a06475b722bbf29cba7f48f376654eLorenzo Colitti        if (dhcpResults.gateway != null) {
11783b26801d62a06475b722bbf29cba7f48f376654eLorenzo Colitti            info.gateway = NetworkUtils.inetAddressToInt((Inet4Address) dhcpResults.gateway);
1179155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        }
11803b26801d62a06475b722bbf29cba7f48f376654eLorenzo Colitti
1181155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        int dnsFound = 0;
11823b26801d62a06475b722bbf29cba7f48f376654eLorenzo Colitti        for (InetAddress dns : dhcpResults.dnsServers) {
1183155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande            if (dns instanceof Inet4Address) {
1184155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande                if (dnsFound == 0) {
1185155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande                    info.dns1 = NetworkUtils.inetAddressToInt((Inet4Address)dns);
1186155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande                } else {
1187155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande                    info.dns2 = NetworkUtils.inetAddressToInt((Inet4Address)dns);
1188155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande                }
1189155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande                if (++dnsFound > 1) break;
1190155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande            }
1191155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        }
11922af03130d7f85823223b8591dc52858d851b301dMitchell Wills        Inet4Address serverAddress = dhcpResults.serverAddress;
11932af03130d7f85823223b8591dc52858d851b301dMitchell Wills        if (serverAddress != null) {
11942af03130d7f85823223b8591dc52858d851b301dMitchell Wills            info.serverAddress = NetworkUtils.inetAddressToInt(serverAddress);
1195155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        }
1196155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        info.leaseDuration = dhcpResults.leaseDuration;
1197155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
1198155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        return info;
1199155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    }
1200155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
1201155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    /**
1202155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande     * enable TDLS for the local NIC to remote NIC
1203155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande     * The APPs don't know the remote MAC address to identify NIC though,
1204155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande     * so we need to do additional work to find it from remote IP address
1205155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande     */
1206155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
1207155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    class TdlsTaskParams {
1208155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        public String remoteIpAddress;
1209155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        public boolean enable;
1210155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    }
1211155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
1212155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    class TdlsTask extends AsyncTask<TdlsTaskParams, Integer, Integer> {
1213155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        @Override
1214155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        protected Integer doInBackground(TdlsTaskParams... params) {
1215155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
1216155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande            // Retrieve parameters for the call
1217155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande            TdlsTaskParams param = params[0];
1218155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande            String remoteIpAddress = param.remoteIpAddress.trim();
1219155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande            boolean enable = param.enable;
1220155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
1221155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande            // Get MAC address of Remote IP
1222155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande            String macAddress = null;
1223155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
1224155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande            BufferedReader reader = null;
1225155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
1226155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande            try {
1227155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande                reader = new BufferedReader(new FileReader("/proc/net/arp"));
1228155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
1229155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande                // Skip over the line bearing colum titles
1230155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande                String line = reader.readLine();
1231155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
1232155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande                while ((line = reader.readLine()) != null) {
1233155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande                    String[] tokens = line.split("[ ]+");
1234155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande                    if (tokens.length < 6) {
1235155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande                        continue;
1236155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande                    }
1237155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
1238155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande                    // ARP column format is
1239155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande                    // Address HWType HWAddress Flags Mask IFace
1240155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande                    String ip = tokens[0];
1241155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande                    String mac = tokens[3];
1242155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
1243155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande                    if (remoteIpAddress.equals(ip)) {
1244155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande                        macAddress = mac;
1245155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande                        break;
1246155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande                    }
1247155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande                }
1248155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
1249155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande                if (macAddress == null) {
1250155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande                    Slog.w(TAG, "Did not find remoteAddress {" + remoteIpAddress + "} in " +
1251155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande                            "/proc/net/arp");
1252155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande                } else {
1253155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande                    enableTdlsWithMacAddress(macAddress, enable);
1254155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande                }
1255155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
1256155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande            } catch (FileNotFoundException e) {
1257155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande                Slog.e(TAG, "Could not open /proc/net/arp to lookup mac address");
1258155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande            } catch (IOException e) {
1259155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande                Slog.e(TAG, "Could not read /proc/net/arp to lookup mac address");
1260155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande            } finally {
1261155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande                try {
1262155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande                    if (reader != null) {
1263155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande                        reader.close();
1264155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande                    }
1265155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande                }
1266155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande                catch (IOException e) {
1267155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande                    // Do nothing
1268155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande                }
1269155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande            }
1270155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
1271155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande            return 0;
1272155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        }
1273155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    }
1274155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
12758fcd1fe86ddc3056e1a26e69bc68f148ae0e9ddcChristopher Wiley    @Override
1276155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    public void enableTdls(String remoteAddress, boolean enable) {
12778e6144b4059fc2117be23d7222893bc57a33f64dJon Larimer        if (remoteAddress == null) {
12788e6144b4059fc2117be23d7222893bc57a33f64dJon Larimer          throw new IllegalArgumentException("remoteAddress cannot be null");
12798e6144b4059fc2117be23d7222893bc57a33f64dJon Larimer        }
12808e6144b4059fc2117be23d7222893bc57a33f64dJon Larimer
1281155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        TdlsTaskParams params = new TdlsTaskParams();
1282155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        params.remoteIpAddress = remoteAddress;
1283155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        params.enable = enable;
1284155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        new TdlsTask().execute(params);
1285155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    }
1286155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
1287155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
12888fcd1fe86ddc3056e1a26e69bc68f148ae0e9ddcChristopher Wiley    @Override
1289155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    public void enableTdlsWithMacAddress(String remoteMacAddress, boolean enable) {
12908e6144b4059fc2117be23d7222893bc57a33f64dJon Larimer        if (remoteMacAddress == null) {
12918e6144b4059fc2117be23d7222893bc57a33f64dJon Larimer          throw new IllegalArgumentException("remoteMacAddress cannot be null");
12928e6144b4059fc2117be23d7222893bc57a33f64dJon Larimer        }
12938e6144b4059fc2117be23d7222893bc57a33f64dJon Larimer
1294155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        mWifiStateMachine.enableTdls(remoteMacAddress, enable);
1295155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    }
1296155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
1297155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    /**
1298155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande     * Get a reference to handler. This is used by a client to establish
1299155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande     * an AsyncChannel communication with WifiService
1300155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande     */
13018fcd1fe86ddc3056e1a26e69bc68f148ae0e9ddcChristopher Wiley    @Override
1302155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    public Messenger getWifiServiceMessenger() {
1303155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        enforceAccessPermission();
1304155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        enforceChangePermission();
1305155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        return new Messenger(mClientHandler);
1306155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    }
1307155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
13082ce99b40c36ed0352b31aa85d5f9383d5f0506f5vandwalle    /**
13092ce99b40c36ed0352b31aa85d5f9383d5f0506f5vandwalle     * Disable an ephemeral network, i.e. network that is created thru a WiFi Scorer
13102ce99b40c36ed0352b31aa85d5f9383d5f0506f5vandwalle     */
13118fcd1fe86ddc3056e1a26e69bc68f148ae0e9ddcChristopher Wiley    @Override
13122ce99b40c36ed0352b31aa85d5f9383d5f0506f5vandwalle    public void disableEphemeralNetwork(String SSID) {
13132ce99b40c36ed0352b31aa85d5f9383d5f0506f5vandwalle        enforceAccessPermission();
13142ce99b40c36ed0352b31aa85d5f9383d5f0506f5vandwalle        enforceChangePermission();
13152ce99b40c36ed0352b31aa85d5f9383d5f0506f5vandwalle        mWifiStateMachine.disableEphemeralNetwork(SSID);
13162ce99b40c36ed0352b31aa85d5f9383d5f0506f5vandwalle    }
1317155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
1318155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    private final BroadcastReceiver mReceiver = new BroadcastReceiver() {
1319155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        @Override
1320155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        public void onReceive(Context context, Intent intent) {
1321155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande            String action = intent.getAction();
1322155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande            if (action.equals(Intent.ACTION_SCREEN_ON)) {
1323155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande                mWifiController.sendMessage(CMD_SCREEN_ON);
1324155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande            } else if (action.equals(Intent.ACTION_USER_PRESENT)) {
1325155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande                mWifiController.sendMessage(CMD_USER_PRESENT);
1326155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande            } else if (action.equals(Intent.ACTION_SCREEN_OFF)) {
1327155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande                mWifiController.sendMessage(CMD_SCREEN_OFF);
1328155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande            } else if (action.equals(Intent.ACTION_BATTERY_CHANGED)) {
1329155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande                int pluggedType = intent.getIntExtra("plugged", 0);
1330155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande                mWifiController.sendMessage(CMD_BATTERY_CHANGED, pluggedType, 0, null);
1331155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande            } else if (action.equals(BluetoothAdapter.ACTION_CONNECTION_STATE_CHANGED)) {
1332155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande                int state = intent.getIntExtra(BluetoothAdapter.EXTRA_CONNECTION_STATE,
1333155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande                        BluetoothAdapter.STATE_DISCONNECTED);
1334155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande                mWifiStateMachine.sendBluetoothAdapterStateChange(state);
1335155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande            } else if (action.equals(TelephonyIntents.ACTION_EMERGENCY_CALLBACK_MODE_CHANGED)) {
1336155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande                boolean emergencyMode = intent.getBooleanExtra("phoneinECMState", false);
1337155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande                mWifiController.sendMessage(CMD_EMERGENCY_MODE_CHANGED, emergencyMode ? 1 : 0, 0);
1338090aa74f2d36010abba6563a39c87ab7ba8a7473Vinit Deshpande            } else if (action.equals(TelephonyIntents.ACTION_EMERGENCY_CALL_STATE_CHANGED)) {
1339090aa74f2d36010abba6563a39c87ab7ba8a7473Vinit Deshpande                boolean inCall = intent.getBooleanExtra(PhoneConstants.PHONE_IN_EMERGENCY_CALL, false);
1340090aa74f2d36010abba6563a39c87ab7ba8a7473Vinit Deshpande                mWifiController.sendMessage(CMD_EMERGENCY_CALL_STATE_CHANGED, inCall ? 1 : 0, 0);
134103b70b37a8a20cb92d860b5015f6762dcb5e75ccDianne Hackborn            } else if (action.equals(PowerManager.ACTION_DEVICE_IDLE_MODE_CHANGED)) {
134203b70b37a8a20cb92d860b5015f6762dcb5e75ccDianne Hackborn                handleIdleModeChanged();
1343155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande            }
1344155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        }
1345155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    };
1346155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
13470469548a809654861f74dc42a3409f29ae750550Svetoslav Ganov    private boolean startConsentUiIfNeeded(String packageName,
13480469548a809654861f74dc42a3409f29ae750550Svetoslav Ganov            int callingUid, String intentAction) throws RemoteException {
13490469548a809654861f74dc42a3409f29ae750550Svetoslav Ganov        if (UserHandle.getAppId(callingUid) == Process.SYSTEM_UID) {
13500469548a809654861f74dc42a3409f29ae750550Svetoslav Ganov            return false;
13510469548a809654861f74dc42a3409f29ae750550Svetoslav Ganov        }
13520469548a809654861f74dc42a3409f29ae750550Svetoslav Ganov        try {
13530469548a809654861f74dc42a3409f29ae750550Svetoslav Ganov            // Validate the package only if we are going to use it
13540469548a809654861f74dc42a3409f29ae750550Svetoslav Ganov            ApplicationInfo applicationInfo = mContext.getPackageManager()
13550469548a809654861f74dc42a3409f29ae750550Svetoslav Ganov                    .getApplicationInfoAsUser(packageName,
13560469548a809654861f74dc42a3409f29ae750550Svetoslav Ganov                            PackageManager.MATCH_DEBUG_TRIAGED_MISSING,
13570469548a809654861f74dc42a3409f29ae750550Svetoslav Ganov                            UserHandle.getUserId(callingUid));
13580469548a809654861f74dc42a3409f29ae750550Svetoslav Ganov            if (applicationInfo.uid != callingUid) {
13590469548a809654861f74dc42a3409f29ae750550Svetoslav Ganov                throw new SecurityException("Package " + callingUid
13600469548a809654861f74dc42a3409f29ae750550Svetoslav Ganov                        + " not in uid " + callingUid);
13610469548a809654861f74dc42a3409f29ae750550Svetoslav Ganov            }
13620469548a809654861f74dc42a3409f29ae750550Svetoslav Ganov
13630469548a809654861f74dc42a3409f29ae750550Svetoslav Ganov            // Legacy apps in permission review mode trigger a user prompt
13640469548a809654861f74dc42a3409f29ae750550Svetoslav Ganov            if (applicationInfo.targetSdkVersion < Build.VERSION_CODES.M) {
13650469548a809654861f74dc42a3409f29ae750550Svetoslav Ganov                Intent intent = new Intent(intentAction);
13660469548a809654861f74dc42a3409f29ae750550Svetoslav Ganov                intent.addFlags(Intent.FLAG_ACTIVITY_NEW_TASK
13670469548a809654861f74dc42a3409f29ae750550Svetoslav Ganov                        | Intent.FLAG_ACTIVITY_EXCLUDE_FROM_RECENTS);
13680469548a809654861f74dc42a3409f29ae750550Svetoslav Ganov                intent.putExtra(Intent.EXTRA_PACKAGE_NAME, packageName);
13690469548a809654861f74dc42a3409f29ae750550Svetoslav Ganov                mContext.startActivity(intent);
13700469548a809654861f74dc42a3409f29ae750550Svetoslav Ganov                return true;
13710469548a809654861f74dc42a3409f29ae750550Svetoslav Ganov            }
13720469548a809654861f74dc42a3409f29ae750550Svetoslav Ganov        } catch (PackageManager.NameNotFoundException e) {
13730469548a809654861f74dc42a3409f29ae750550Svetoslav Ganov            throw new RemoteException(e.getMessage());
13740469548a809654861f74dc42a3409f29ae750550Svetoslav Ganov        }
13750469548a809654861f74dc42a3409f29ae750550Svetoslav Ganov        return false;
13760469548a809654861f74dc42a3409f29ae750550Svetoslav Ganov    }
13770469548a809654861f74dc42a3409f29ae750550Svetoslav Ganov
1378155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    /**
1379155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande     * Observes settings changes to scan always mode.
1380155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande     */
1381155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    private void registerForScanModeChange() {
1382155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        ContentObserver contentObserver = new ContentObserver(null) {
1383155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande            @Override
1384155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande            public void onChange(boolean selfChange) {
1385155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande                mSettingsStore.handleWifiScanAlwaysAvailableToggled();
1386155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande                mWifiController.sendMessage(CMD_SCAN_ALWAYS_MODE_CHANGED);
1387155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande            }
1388155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        };
1389155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
1390155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        mContext.getContentResolver().registerContentObserver(
1391155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande                Settings.Global.getUriFor(Settings.Global.WIFI_SCAN_ALWAYS_AVAILABLE),
1392155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande                false, contentObserver);
1393155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    }
1394155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
1395155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    private void registerForBroadcasts() {
1396155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        IntentFilter intentFilter = new IntentFilter();
1397155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        intentFilter.addAction(Intent.ACTION_SCREEN_ON);
1398155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        intentFilter.addAction(Intent.ACTION_USER_PRESENT);
1399155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        intentFilter.addAction(Intent.ACTION_SCREEN_OFF);
1400155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        intentFilter.addAction(Intent.ACTION_BATTERY_CHANGED);
1401155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        intentFilter.addAction(WifiManager.NETWORK_STATE_CHANGED_ACTION);
1402155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        intentFilter.addAction(BluetoothAdapter.ACTION_CONNECTION_STATE_CHANGED);
1403155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        intentFilter.addAction(TelephonyIntents.ACTION_EMERGENCY_CALLBACK_MODE_CHANGED);
140403b70b37a8a20cb92d860b5015f6762dcb5e75ccDianne Hackborn        intentFilter.addAction(PowerManager.ACTION_DEVICE_IDLE_MODE_CHANGED);
1405090aa74f2d36010abba6563a39c87ab7ba8a7473Vinit Deshpande
1406090aa74f2d36010abba6563a39c87ab7ba8a7473Vinit Deshpande        boolean trackEmergencyCallState = mContext.getResources().getBoolean(
1407090aa74f2d36010abba6563a39c87ab7ba8a7473Vinit Deshpande                com.android.internal.R.bool.config_wifi_turn_off_during_emergency_call);
1408090aa74f2d36010abba6563a39c87ab7ba8a7473Vinit Deshpande        if (trackEmergencyCallState) {
1409090aa74f2d36010abba6563a39c87ab7ba8a7473Vinit Deshpande            intentFilter.addAction(TelephonyIntents.ACTION_EMERGENCY_CALL_STATE_CHANGED);
1410090aa74f2d36010abba6563a39c87ab7ba8a7473Vinit Deshpande        }
1411090aa74f2d36010abba6563a39c87ab7ba8a7473Vinit Deshpande
1412155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        mContext.registerReceiver(mReceiver, intentFilter);
1413155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    }
1414155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
1415faa2ce7f93eef2c436796fb1ea3574e2d1412491Robin Lee    private void registerForPackageOrUserRemoval() {
1416e487a4648dd41881e754f1224aaedead78a0777dSky Faber        IntentFilter intentFilter = new IntentFilter();
1417e487a4648dd41881e754f1224aaedead78a0777dSky Faber        intentFilter.addAction(Intent.ACTION_PACKAGE_REMOVED);
1418faa2ce7f93eef2c436796fb1ea3574e2d1412491Robin Lee        intentFilter.addAction(Intent.ACTION_USER_REMOVED);
1419faa2ce7f93eef2c436796fb1ea3574e2d1412491Robin Lee        mContext.registerReceiverAsUser(new BroadcastReceiver() {
1420e487a4648dd41881e754f1224aaedead78a0777dSky Faber            @Override
1421e487a4648dd41881e754f1224aaedead78a0777dSky Faber            public void onReceive(Context context, Intent intent) {
1422faa2ce7f93eef2c436796fb1ea3574e2d1412491Robin Lee                switch (intent.getAction()) {
1423faa2ce7f93eef2c436796fb1ea3574e2d1412491Robin Lee                    case Intent.ACTION_PACKAGE_REMOVED: {
1424faa2ce7f93eef2c436796fb1ea3574e2d1412491Robin Lee                        if (intent.getBooleanExtra(Intent.EXTRA_REPLACING, false)) {
1425faa2ce7f93eef2c436796fb1ea3574e2d1412491Robin Lee                            return;
1426faa2ce7f93eef2c436796fb1ea3574e2d1412491Robin Lee                        }
1427faa2ce7f93eef2c436796fb1ea3574e2d1412491Robin Lee                        int uid = intent.getIntExtra(Intent.EXTRA_UID, -1);
1428faa2ce7f93eef2c436796fb1ea3574e2d1412491Robin Lee                        Uri uri = intent.getData();
1429faa2ce7f93eef2c436796fb1ea3574e2d1412491Robin Lee                        if (uid == -1 || uri == null) {
1430faa2ce7f93eef2c436796fb1ea3574e2d1412491Robin Lee                            return;
1431faa2ce7f93eef2c436796fb1ea3574e2d1412491Robin Lee                        }
1432faa2ce7f93eef2c436796fb1ea3574e2d1412491Robin Lee                        String pkgName = uri.getSchemeSpecificPart();
1433faa2ce7f93eef2c436796fb1ea3574e2d1412491Robin Lee                        mWifiStateMachine.removeAppConfigs(pkgName, uid);
1434faa2ce7f93eef2c436796fb1ea3574e2d1412491Robin Lee                        break;
1435faa2ce7f93eef2c436796fb1ea3574e2d1412491Robin Lee                    }
1436faa2ce7f93eef2c436796fb1ea3574e2d1412491Robin Lee                    case Intent.ACTION_USER_REMOVED: {
1437faa2ce7f93eef2c436796fb1ea3574e2d1412491Robin Lee                        int userHandle = intent.getIntExtra(Intent.EXTRA_USER_HANDLE, 0);
1438faa2ce7f93eef2c436796fb1ea3574e2d1412491Robin Lee                        mWifiStateMachine.removeUserConfigs(userHandle);
1439faa2ce7f93eef2c436796fb1ea3574e2d1412491Robin Lee                        break;
1440e487a4648dd41881e754f1224aaedead78a0777dSky Faber                    }
1441e487a4648dd41881e754f1224aaedead78a0777dSky Faber                }
1442e487a4648dd41881e754f1224aaedead78a0777dSky Faber            }
1443faa2ce7f93eef2c436796fb1ea3574e2d1412491Robin Lee        }, UserHandle.ALL, intentFilter, null, null);
1444e487a4648dd41881e754f1224aaedead78a0777dSky Faber    }
1445e487a4648dd41881e754f1224aaedead78a0777dSky Faber
1446155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    @Override
1447155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    protected void dump(FileDescriptor fd, PrintWriter pw, String[] args) {
1448155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        if (mContext.checkCallingOrSelfPermission(android.Manifest.permission.DUMP)
1449155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande                != PackageManager.PERMISSION_GRANTED) {
1450155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande            pw.println("Permission Denial: can't dump WifiService from from pid="
1451155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande                    + Binder.getCallingPid()
1452155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande                    + ", uid=" + Binder.getCallingUid());
1453155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande            return;
1454155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        }
145511638f348ba45f9f417928e79b81186cef76c561Glen Kuhne        if (args.length > 0 && WifiMetrics.PROTO_DUMP_ARG.equals(args[0])) {
145611638f348ba45f9f417928e79b81186cef76c561Glen Kuhne            // WifiMetrics proto bytes were requested. Dump only these.
145711638f348ba45f9f417928e79b81186cef76c561Glen Kuhne            mWifiStateMachine.updateWifiMetrics();
145811638f348ba45f9f417928e79b81186cef76c561Glen Kuhne            mWifiMetrics.dump(fd, pw, args);
1459675de6070a3c1980fcc99bc6863b2689bfeb0271Lorenzo Colitti        } else if (args.length > 0 && IpManager.DUMP_ARG.equals(args[0])) {
1460675de6070a3c1980fcc99bc6863b2689bfeb0271Lorenzo Colitti            // IpManager dump was requested. Pass it along and take no further action.
1461675de6070a3c1980fcc99bc6863b2689bfeb0271Lorenzo Colitti            String[] ipManagerArgs = new String[args.length - 1];
1462675de6070a3c1980fcc99bc6863b2689bfeb0271Lorenzo Colitti            System.arraycopy(args, 1, ipManagerArgs, 0, ipManagerArgs.length);
1463675de6070a3c1980fcc99bc6863b2689bfeb0271Lorenzo Colitti            mWifiStateMachine.dumpIpManager(fd, pw, ipManagerArgs);
146484ee4726383fd968ab43caf120d15d9f47a51a54Paul Stewart        } else if (args.length > 0 && DUMP_ARG_SET_IPREACH_DISCONNECT.equals(args[0])) {
146584ee4726383fd968ab43caf120d15d9f47a51a54Paul Stewart            if (args.length > 1) {
146684ee4726383fd968ab43caf120d15d9f47a51a54Paul Stewart                if (DUMP_ARG_SET_IPREACH_DISCONNECT_ENABLED.equals(args[1])) {
146784ee4726383fd968ab43caf120d15d9f47a51a54Paul Stewart                    mWifiStateMachine.setIpReachabilityDisconnectEnabled(true);
146884ee4726383fd968ab43caf120d15d9f47a51a54Paul Stewart                } else if (DUMP_ARG_SET_IPREACH_DISCONNECT_DISABLED.equals(args[1])) {
146984ee4726383fd968ab43caf120d15d9f47a51a54Paul Stewart                    mWifiStateMachine.setIpReachabilityDisconnectEnabled(false);
147084ee4726383fd968ab43caf120d15d9f47a51a54Paul Stewart                }
147184ee4726383fd968ab43caf120d15d9f47a51a54Paul Stewart            }
147284ee4726383fd968ab43caf120d15d9f47a51a54Paul Stewart            pw.println("IPREACH_DISCONNECT state is "
147384ee4726383fd968ab43caf120d15d9f47a51a54Paul Stewart                    + mWifiStateMachine.getIpReachabilityDisconnectEnabled());
147484ee4726383fd968ab43caf120d15d9f47a51a54Paul Stewart            return;
147511638f348ba45f9f417928e79b81186cef76c561Glen Kuhne        } else {
147611638f348ba45f9f417928e79b81186cef76c561Glen Kuhne            pw.println("Wi-Fi is " + mWifiStateMachine.syncGetWifiStateByName());
147711638f348ba45f9f417928e79b81186cef76c561Glen Kuhne            pw.println("Stay-awake conditions: " +
147811638f348ba45f9f417928e79b81186cef76c561Glen Kuhne                    Settings.Global.getInt(mContext.getContentResolver(),
147911638f348ba45f9f417928e79b81186cef76c561Glen Kuhne                                           Settings.Global.STAY_ON_WHILE_PLUGGED_IN, 0));
148011638f348ba45f9f417928e79b81186cef76c561Glen Kuhne            pw.println("mInIdleMode " + mInIdleMode);
148111638f348ba45f9f417928e79b81186cef76c561Glen Kuhne            pw.println("mScanPending " + mScanPending);
148211638f348ba45f9f417928e79b81186cef76c561Glen Kuhne            mWifiController.dump(fd, pw, args);
148311638f348ba45f9f417928e79b81186cef76c561Glen Kuhne            mSettingsStore.dump(fd, pw, args);
148411638f348ba45f9f417928e79b81186cef76c561Glen Kuhne            mNotificationController.dump(fd, pw, args);
1485f09b6de086d5a00417613886aa43402285d2a8abStephen Chen            mWifiWakeupController.dump(fd, pw, args);
148611638f348ba45f9f417928e79b81186cef76c561Glen Kuhne            mTrafficPoller.dump(fd, pw, args);
148711638f348ba45f9f417928e79b81186cef76c561Glen Kuhne            pw.println();
148811638f348ba45f9f417928e79b81186cef76c561Glen Kuhne            pw.println("Locks held:");
14892a7c2bec5d307a9e52de429c6303b7df9de900dcRebecca Silberstein            mWifiLockManager.dump(pw);
14902a7c2bec5d307a9e52de429c6303b7df9de900dcRebecca Silberstein            pw.println();
149161312e14a088a9487d4db64f08285162476e870fPaul Stewart            mWifiMulticastLockManager.dump(pw);
149211638f348ba45f9f417928e79b81186cef76c561Glen Kuhne            pw.println();
149311638f348ba45f9f417928e79b81186cef76c561Glen Kuhne            mWifiStateMachine.dump(fd, pw, args);
149411638f348ba45f9f417928e79b81186cef76c561Glen Kuhne            pw.println();
14953204fb9682242a7b5a749489076c66d448c42577Roshan Pius            mWifiBackupRestore.dump(fd, pw, args);
14963204fb9682242a7b5a749489076c66d448c42577Roshan Pius            pw.println();
149711638f348ba45f9f417928e79b81186cef76c561Glen Kuhne        }
1498155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    }
1499155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
15008fcd1fe86ddc3056e1a26e69bc68f148ae0e9ddcChristopher Wiley    @Override
1501155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    public boolean acquireWifiLock(IBinder binder, int lockMode, String tag, WorkSource ws) {
15022a7c2bec5d307a9e52de429c6303b7df9de900dcRebecca Silberstein        if (mWifiLockManager.acquireWifiLock(lockMode, tag, binder, ws)) {
1503155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande            mWifiController.sendMessage(CMD_LOCKS_CHANGED);
1504155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande            return true;
1505155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        }
15062a7c2bec5d307a9e52de429c6303b7df9de900dcRebecca Silberstein        return false;
1507155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    }
1508155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
15098fcd1fe86ddc3056e1a26e69bc68f148ae0e9ddcChristopher Wiley    @Override
15102a7c2bec5d307a9e52de429c6303b7df9de900dcRebecca Silberstein    public void updateWifiLockWorkSource(IBinder binder, WorkSource ws) {
15112a7c2bec5d307a9e52de429c6303b7df9de900dcRebecca Silberstein        mWifiLockManager.updateWifiLockWorkSource(binder, ws);
1512155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    }
1513155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
15148fcd1fe86ddc3056e1a26e69bc68f148ae0e9ddcChristopher Wiley    @Override
15152a7c2bec5d307a9e52de429c6303b7df9de900dcRebecca Silberstein    public boolean releaseWifiLock(IBinder binder) {
15162a7c2bec5d307a9e52de429c6303b7df9de900dcRebecca Silberstein        if (mWifiLockManager.releaseWifiLock(binder)) {
15172a7c2bec5d307a9e52de429c6303b7df9de900dcRebecca Silberstein            mWifiController.sendMessage(CMD_LOCKS_CHANGED);
15182a7c2bec5d307a9e52de429c6303b7df9de900dcRebecca Silberstein            return true;
1519155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        }
15202a7c2bec5d307a9e52de429c6303b7df9de900dcRebecca Silberstein        return false;
1521155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    }
1522155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
15238fcd1fe86ddc3056e1a26e69bc68f148ae0e9ddcChristopher Wiley    @Override
1524155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    public void initializeMulticastFiltering() {
1525155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        enforceMulticastChangePermission();
152661312e14a088a9487d4db64f08285162476e870fPaul Stewart        mWifiMulticastLockManager.initializeFiltering();
1527155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    }
1528155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
15298fcd1fe86ddc3056e1a26e69bc68f148ae0e9ddcChristopher Wiley    @Override
1530155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    public void acquireMulticastLock(IBinder binder, String tag) {
1531155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        enforceMulticastChangePermission();
153261312e14a088a9487d4db64f08285162476e870fPaul Stewart        mWifiMulticastLockManager.acquireLock(binder, tag);
1533155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    }
1534155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
15358fcd1fe86ddc3056e1a26e69bc68f148ae0e9ddcChristopher Wiley    @Override
1536155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    public void releaseMulticastLock() {
1537155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        enforceMulticastChangePermission();
153861312e14a088a9487d4db64f08285162476e870fPaul Stewart        mWifiMulticastLockManager.releaseLock();
1539155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    }
1540155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
15418fcd1fe86ddc3056e1a26e69bc68f148ae0e9ddcChristopher Wiley    @Override
1542155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    public boolean isMulticastEnabled() {
1543155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        enforceAccessPermission();
154461312e14a088a9487d4db64f08285162476e870fPaul Stewart        return mWifiMulticastLockManager.isMulticastEnabled();
1545155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    }
15469ae6b45c038fa74a9e7285ef4834551dd93da332Yuhao Zheng
15478fcd1fe86ddc3056e1a26e69bc68f148ae0e9ddcChristopher Wiley    @Override
1548ed9938883ae2dade81c8be6cd6ceaef3febd5239vandwalle    public void enableVerboseLogging(int verbose) {
1549ed9938883ae2dade81c8be6cd6ceaef3febd5239vandwalle        enforceAccessPermission();
155000ec8bf178ba4abee4ee8bbc6eb09c9fcd986a89Rebecca Silberstein        mFacade.setIntegerSetting(
155100ec8bf178ba4abee4ee8bbc6eb09c9fcd986a89Rebecca Silberstein                mContext, Settings.Global.WIFI_VERBOSE_LOGGING_ENABLED, verbose);
155200ec8bf178ba4abee4ee8bbc6eb09c9fcd986a89Rebecca Silberstein        enableVerboseLoggingInternal(verbose);
155300ec8bf178ba4abee4ee8bbc6eb09c9fcd986a89Rebecca Silberstein    }
155400ec8bf178ba4abee4ee8bbc6eb09c9fcd986a89Rebecca Silberstein
1555da918df16e03ee19be62343313d954027d3eb3abRebecca Silberstein    void enableVerboseLoggingInternal(int verbose) {
1556ed9938883ae2dade81c8be6cd6ceaef3febd5239vandwalle        mWifiStateMachine.enableVerboseLogging(verbose);
15572a7c2bec5d307a9e52de429c6303b7df9de900dcRebecca Silberstein        mWifiLockManager.enableVerboseLogging(verbose);
155861312e14a088a9487d4db64f08285162476e870fPaul Stewart        mWifiMulticastLockManager.enableVerboseLogging(verbose);
155975727af748e2b53baf365139ecfa7bf87a449d04Rebecca Silberstein        mWifiInjector.getWifiLastResortWatchdog().enableVerboseLogging(verbose);
156000ec8bf178ba4abee4ee8bbc6eb09c9fcd986a89Rebecca Silberstein        mWifiInjector.getWifiBackupRestore().enableVerboseLogging(verbose);
1561e33b3346b262507ef2361d50a89d16bef69d9a57Sohani Rao        LogcatLog.enableVerboseLogging(verbose);
1562ed9938883ae2dade81c8be6cd6ceaef3febd5239vandwalle    }
1563ed9938883ae2dade81c8be6cd6ceaef3febd5239vandwalle
15648fcd1fe86ddc3056e1a26e69bc68f148ae0e9ddcChristopher Wiley    @Override
1565ed9938883ae2dade81c8be6cd6ceaef3febd5239vandwalle    public int getVerboseLoggingLevel() {
1566ed9938883ae2dade81c8be6cd6ceaef3febd5239vandwalle        enforceAccessPermission();
156700ec8bf178ba4abee4ee8bbc6eb09c9fcd986a89Rebecca Silberstein        return mFacade.getIntegerSetting(
156800ec8bf178ba4abee4ee8bbc6eb09c9fcd986a89Rebecca Silberstein                mContext, Settings.Global.WIFI_VERBOSE_LOGGING_ENABLED, 0);
1569ed9938883ae2dade81c8be6cd6ceaef3febd5239vandwalle    }
1570c6f06c628ee3583b60ff31a7da442e0ac7b26d97vandwalle
15718fcd1fe86ddc3056e1a26e69bc68f148ae0e9ddcChristopher Wiley    @Override
1572c6f06c628ee3583b60ff31a7da442e0ac7b26d97vandwalle    public void enableAggressiveHandover(int enabled) {
1573c6f06c628ee3583b60ff31a7da442e0ac7b26d97vandwalle        enforceAccessPermission();
1574c6f06c628ee3583b60ff31a7da442e0ac7b26d97vandwalle        mWifiStateMachine.enableAggressiveHandover(enabled);
1575c6f06c628ee3583b60ff31a7da442e0ac7b26d97vandwalle    }
1576c6f06c628ee3583b60ff31a7da442e0ac7b26d97vandwalle
15778fcd1fe86ddc3056e1a26e69bc68f148ae0e9ddcChristopher Wiley    @Override
1578c6f06c628ee3583b60ff31a7da442e0ac7b26d97vandwalle    public int getAggressiveHandover() {
1579c6f06c628ee3583b60ff31a7da442e0ac7b26d97vandwalle        enforceAccessPermission();
1580c6f06c628ee3583b60ff31a7da442e0ac7b26d97vandwalle        return mWifiStateMachine.getAggressiveHandover();
1581c6f06c628ee3583b60ff31a7da442e0ac7b26d97vandwalle    }
1582c6f06c628ee3583b60ff31a7da442e0ac7b26d97vandwalle
15838fcd1fe86ddc3056e1a26e69bc68f148ae0e9ddcChristopher Wiley    @Override
1584c6f06c628ee3583b60ff31a7da442e0ac7b26d97vandwalle    public void setAllowScansWithTraffic(int enabled) {
1585c6f06c628ee3583b60ff31a7da442e0ac7b26d97vandwalle        enforceAccessPermission();
1586c6f06c628ee3583b60ff31a7da442e0ac7b26d97vandwalle        mWifiStateMachine.setAllowScansWithTraffic(enabled);
1587c6f06c628ee3583b60ff31a7da442e0ac7b26d97vandwalle    }
1588c6f06c628ee3583b60ff31a7da442e0ac7b26d97vandwalle
15898fcd1fe86ddc3056e1a26e69bc68f148ae0e9ddcChristopher Wiley    @Override
1590c6f06c628ee3583b60ff31a7da442e0ac7b26d97vandwalle    public int getAllowScansWithTraffic() {
1591c6f06c628ee3583b60ff31a7da442e0ac7b26d97vandwalle        enforceAccessPermission();
1592c6f06c628ee3583b60ff31a7da442e0ac7b26d97vandwalle        return mWifiStateMachine.getAllowScansWithTraffic();
1593c6f06c628ee3583b60ff31a7da442e0ac7b26d97vandwalle    }
1594c298087de50ea56c31a4ade7ee1e83b313bb63c7vandwalle
15958fcd1fe86ddc3056e1a26e69bc68f148ae0e9ddcChristopher Wiley    @Override
159643eba5ba17a9af5d9a050a4ba5e12e93c92f722dSamuel Tan    public boolean setEnableAutoJoinWhenAssociated(boolean enabled) {
1597a8647b8cb29de22765062714cb265247234c3d32xinhe        enforceChangePermission();
159843eba5ba17a9af5d9a050a4ba5e12e93c92f722dSamuel Tan        return mWifiStateMachine.setEnableAutoJoinWhenAssociated(enabled);
1599e2614ba1b774529ee68cc4ac189f7492cff89db3Pierre Vandwalle    }
1600e2614ba1b774529ee68cc4ac189f7492cff89db3Pierre Vandwalle
16018fcd1fe86ddc3056e1a26e69bc68f148ae0e9ddcChristopher Wiley    @Override
1602a8647b8cb29de22765062714cb265247234c3d32xinhe    public boolean getEnableAutoJoinWhenAssociated() {
1603e2614ba1b774529ee68cc4ac189f7492cff89db3Pierre Vandwalle        enforceAccessPermission();
1604a8647b8cb29de22765062714cb265247234c3d32xinhe        return mWifiStateMachine.getEnableAutoJoinWhenAssociated();
1605e2614ba1b774529ee68cc4ac189f7492cff89db3Pierre Vandwalle    }
1606b0b0cc202b7d7aaad7b3f69d73e9b58ea2968b05Pierre Vandwalle
1607c298087de50ea56c31a4ade7ee1e83b313bb63c7vandwalle    /* Return the Wifi Connection statistics object */
16088fcd1fe86ddc3056e1a26e69bc68f148ae0e9ddcChristopher Wiley    @Override
1609c298087de50ea56c31a4ade7ee1e83b313bb63c7vandwalle    public WifiConnectionStatistics getConnectionStatistics() {
1610c298087de50ea56c31a4ade7ee1e83b313bb63c7vandwalle        enforceAccessPermission();
1611c298087de50ea56c31a4ade7ee1e83b313bb63c7vandwalle        enforceReadCredentialPermission();
1612c298087de50ea56c31a4ade7ee1e83b313bb63c7vandwalle        if (mWifiStateMachineChannel != null) {
1613c298087de50ea56c31a4ade7ee1e83b313bb63c7vandwalle            return mWifiStateMachine.syncGetConnectionStatistics(mWifiStateMachineChannel);
1614c298087de50ea56c31a4ade7ee1e83b313bb63c7vandwalle        } else {
1615c298087de50ea56c31a4ade7ee1e83b313bb63c7vandwalle            Slog.e(TAG, "mWifiStateMachineChannel is not initialized");
1616c298087de50ea56c31a4ade7ee1e83b313bb63c7vandwalle            return null;
1617c298087de50ea56c31a4ade7ee1e83b313bb63c7vandwalle        }
1618c298087de50ea56c31a4ade7ee1e83b313bb63c7vandwalle    }
1619b8671cfafc2830ebddeafcfb2d91f2b39b19019bStuart Scott
16208fcd1fe86ddc3056e1a26e69bc68f148ae0e9ddcChristopher Wiley    @Override
1621b8671cfafc2830ebddeafcfb2d91f2b39b19019bStuart Scott    public void factoryReset() {
1622b8671cfafc2830ebddeafcfb2d91f2b39b19019bStuart Scott        enforceConnectivityInternalPermission();
1623b8671cfafc2830ebddeafcfb2d91f2b39b19019bStuart Scott
1624f9d866f1b4e93593b3411886fd71e729f4a5a205Stuart Scott        if (mUserManager.hasUserRestriction(UserManager.DISALLOW_NETWORK_RESET)) {
1625f9d866f1b4e93593b3411886fd71e729f4a5a205Stuart Scott            return;
1626f9d866f1b4e93593b3411886fd71e729f4a5a205Stuart Scott        }
1627b8671cfafc2830ebddeafcfb2d91f2b39b19019bStuart Scott
1628f9d866f1b4e93593b3411886fd71e729f4a5a205Stuart Scott        if (!mUserManager.hasUserRestriction(UserManager.DISALLOW_CONFIG_TETHERING)) {
1629f9d866f1b4e93593b3411886fd71e729f4a5a205Stuart Scott            // Turn mobile hotspot off
1630f9d866f1b4e93593b3411886fd71e729f4a5a205Stuart Scott            setWifiApEnabled(null, false);
1631f9d866f1b4e93593b3411886fd71e729f4a5a205Stuart Scott        }
1632f9d866f1b4e93593b3411886fd71e729f4a5a205Stuart Scott
1633f9d866f1b4e93593b3411886fd71e729f4a5a205Stuart Scott        if (!mUserManager.hasUserRestriction(UserManager.DISALLOW_CONFIG_WIFI)) {
1634f9d866f1b4e93593b3411886fd71e729f4a5a205Stuart Scott            // Enable wifi
16350469548a809654861f74dc42a3409f29ae750550Svetoslav Ganov            try {
16360469548a809654861f74dc42a3409f29ae750550Svetoslav Ganov                setWifiEnabled(mContext.getOpPackageName(), true);
16370469548a809654861f74dc42a3409f29ae750550Svetoslav Ganov            } catch (RemoteException e) {
16380469548a809654861f74dc42a3409f29ae750550Svetoslav Ganov                /* ignore - local call */
16390469548a809654861f74dc42a3409f29ae750550Svetoslav Ganov            }
1640f9d866f1b4e93593b3411886fd71e729f4a5a205Stuart Scott            // Delete all Wifi SSIDs
1641f9d866f1b4e93593b3411886fd71e729f4a5a205Stuart Scott            List<WifiConfiguration> networks = getConfiguredNetworks();
1642f9d866f1b4e93593b3411886fd71e729f4a5a205Stuart Scott            if (networks != null) {
1643f9d866f1b4e93593b3411886fd71e729f4a5a205Stuart Scott                for (WifiConfiguration config : networks) {
1644f9d866f1b4e93593b3411886fd71e729f4a5a205Stuart Scott                    removeNetwork(config.networkId);
1645f9d866f1b4e93593b3411886fd71e729f4a5a205Stuart Scott                }
1646f9d866f1b4e93593b3411886fd71e729f4a5a205Stuart Scott                saveConfiguration();
1647b8671cfafc2830ebddeafcfb2d91f2b39b19019bStuart Scott            }
1648b8671cfafc2830ebddeafcfb2d91f2b39b19019bStuart Scott        }
1649b8671cfafc2830ebddeafcfb2d91f2b39b19019bStuart Scott    }
16508be2ac8cec40a2000715849b627642bdc87c10b3Shirish Kalele
16514d381bc39f5263effdae73ec99065eb299b806caVinit Deshpande    /* private methods */
165231891afce40b903ada9b24ec12e3648ae6aa27b2Jan Nordqvist    static boolean logAndReturnFalse(String s) {
16534d381bc39f5263effdae73ec99065eb299b806caVinit Deshpande        Log.d(TAG, s);
16544d381bc39f5263effdae73ec99065eb299b806caVinit Deshpande        return false;
16554d381bc39f5263effdae73ec99065eb299b806caVinit Deshpande    }
16564d381bc39f5263effdae73ec99065eb299b806caVinit Deshpande
165731891afce40b903ada9b24ec12e3648ae6aa27b2Jan Nordqvist    public static boolean isValid(WifiConfiguration config) {
165831891afce40b903ada9b24ec12e3648ae6aa27b2Jan Nordqvist        String validity = checkValidity(config);
165931891afce40b903ada9b24ec12e3648ae6aa27b2Jan Nordqvist        return validity == null || logAndReturnFalse(validity);
166031891afce40b903ada9b24ec12e3648ae6aa27b2Jan Nordqvist    }
16614d381bc39f5263effdae73ec99065eb299b806caVinit Deshpande
16624aef3283bf78e233a11fa34af2c7c38362879625Jan Nordqvist    public static boolean isValidPasspoint(WifiConfiguration config) {
16634aef3283bf78e233a11fa34af2c7c38362879625Jan Nordqvist        String validity = checkPasspointValidity(config);
16644aef3283bf78e233a11fa34af2c7c38362879625Jan Nordqvist        return validity == null || logAndReturnFalse(validity);
16654aef3283bf78e233a11fa34af2c7c38362879625Jan Nordqvist    }
16664aef3283bf78e233a11fa34af2c7c38362879625Jan Nordqvist
166731891afce40b903ada9b24ec12e3648ae6aa27b2Jan Nordqvist    public static String checkValidity(WifiConfiguration config) {
166831891afce40b903ada9b24ec12e3648ae6aa27b2Jan Nordqvist        if (config.allowedKeyManagement == null)
166931891afce40b903ada9b24ec12e3648ae6aa27b2Jan Nordqvist            return "allowed kmgmt";
16704d381bc39f5263effdae73ec99065eb299b806caVinit Deshpande
16714d381bc39f5263effdae73ec99065eb299b806caVinit Deshpande        if (config.allowedKeyManagement.cardinality() > 1) {
16724d381bc39f5263effdae73ec99065eb299b806caVinit Deshpande            if (config.allowedKeyManagement.cardinality() != 2) {
167331891afce40b903ada9b24ec12e3648ae6aa27b2Jan Nordqvist                return "cardinality != 2";
16744d381bc39f5263effdae73ec99065eb299b806caVinit Deshpande            }
16754d381bc39f5263effdae73ec99065eb299b806caVinit Deshpande            if (!config.allowedKeyManagement.get(WifiConfiguration.KeyMgmt.WPA_EAP)) {
167631891afce40b903ada9b24ec12e3648ae6aa27b2Jan Nordqvist                return "not WPA_EAP";
16774d381bc39f5263effdae73ec99065eb299b806caVinit Deshpande            }
16784d381bc39f5263effdae73ec99065eb299b806caVinit Deshpande            if ((!config.allowedKeyManagement.get(WifiConfiguration.KeyMgmt.IEEE8021X))
16794d381bc39f5263effdae73ec99065eb299b806caVinit Deshpande                    && (!config.allowedKeyManagement.get(WifiConfiguration.KeyMgmt.WPA_PSK))) {
168031891afce40b903ada9b24ec12e3648ae6aa27b2Jan Nordqvist                return "not PSK or 8021X";
16814d381bc39f5263effdae73ec99065eb299b806caVinit Deshpande            }
16824d381bc39f5263effdae73ec99065eb299b806caVinit Deshpande        }
16834aef3283bf78e233a11fa34af2c7c38362879625Jan Nordqvist        return null;
16844aef3283bf78e233a11fa34af2c7c38362879625Jan Nordqvist    }
16854d381bc39f5263effdae73ec99065eb299b806caVinit Deshpande
16864aef3283bf78e233a11fa34af2c7c38362879625Jan Nordqvist    public static String checkPasspointValidity(WifiConfiguration config) {
16874d381bc39f5263effdae73ec99065eb299b806caVinit Deshpande        if (!TextUtils.isEmpty(config.FQDN)) {
16884d381bc39f5263effdae73ec99065eb299b806caVinit Deshpande            /* this is passpoint configuration; it must not have an SSID */
16894d381bc39f5263effdae73ec99065eb299b806caVinit Deshpande            if (!TextUtils.isEmpty(config.SSID)) {
169031891afce40b903ada9b24ec12e3648ae6aa27b2Jan Nordqvist                return "SSID not expected for Passpoint: '" + config.SSID +
169131891afce40b903ada9b24ec12e3648ae6aa27b2Jan Nordqvist                        "' FQDN " + toHexString(config.FQDN);
16924d381bc39f5263effdae73ec99065eb299b806caVinit Deshpande            }
16934d381bc39f5263effdae73ec99065eb299b806caVinit Deshpande            /* this is passpoint configuration; it must have a providerFriendlyName */
16944d381bc39f5263effdae73ec99065eb299b806caVinit Deshpande            if (TextUtils.isEmpty(config.providerFriendlyName)) {
169531891afce40b903ada9b24ec12e3648ae6aa27b2Jan Nordqvist                return "no provider friendly name";
16964d381bc39f5263effdae73ec99065eb299b806caVinit Deshpande            }
169707f11f6f2ee7ec17cb08180035dfb5002aaaf5dfJan Nordqvist            WifiEnterpriseConfig enterpriseConfig = config.enterpriseConfig;
16984d381bc39f5263effdae73ec99065eb299b806caVinit Deshpande            /* this is passpoint configuration; it must have enterprise config */
169907f11f6f2ee7ec17cb08180035dfb5002aaaf5dfJan Nordqvist            if (enterpriseConfig == null
170007f11f6f2ee7ec17cb08180035dfb5002aaaf5dfJan Nordqvist                    || enterpriseConfig.getEapMethod() == WifiEnterpriseConfig.Eap.NONE ) {
170131891afce40b903ada9b24ec12e3648ae6aa27b2Jan Nordqvist                return "no enterprise config";
17024d381bc39f5263effdae73ec99065eb299b806caVinit Deshpande            }
170307f11f6f2ee7ec17cb08180035dfb5002aaaf5dfJan Nordqvist            if ((enterpriseConfig.getEapMethod() == WifiEnterpriseConfig.Eap.TLS ||
170407f11f6f2ee7ec17cb08180035dfb5002aaaf5dfJan Nordqvist                    enterpriseConfig.getEapMethod() == WifiEnterpriseConfig.Eap.TTLS ||
170507f11f6f2ee7ec17cb08180035dfb5002aaaf5dfJan Nordqvist                    enterpriseConfig.getEapMethod() == WifiEnterpriseConfig.Eap.PEAP) &&
170607f11f6f2ee7ec17cb08180035dfb5002aaaf5dfJan Nordqvist                    enterpriseConfig.getCaCertificate() == null) {
170731891afce40b903ada9b24ec12e3648ae6aa27b2Jan Nordqvist                return "no CA certificate";
17084d381bc39f5263effdae73ec99065eb299b806caVinit Deshpande            }
17094d381bc39f5263effdae73ec99065eb299b806caVinit Deshpande        }
171031891afce40b903ada9b24ec12e3648ae6aa27b2Jan Nordqvist        return null;
1711cd0af1c9ddcac881f49c815fb9da45433eb50736Vinit Deshpande    }
1712b8671cfafc2830ebddeafcfb2d91f2b39b19019bStuart Scott
17138fcd1fe86ddc3056e1a26e69bc68f148ae0e9ddcChristopher Wiley    @Override
17149846078598c3468f8813dbfa58238a1846bd81efSanket Padawe    public Network getCurrentNetwork() {
17159846078598c3468f8813dbfa58238a1846bd81efSanket Padawe        enforceAccessPermission();
17169846078598c3468f8813dbfa58238a1846bd81efSanket Padawe        return mWifiStateMachine.getCurrentNetwork();
17179846078598c3468f8813dbfa58238a1846bd81efSanket Padawe    }
17189846078598c3468f8813dbfa58238a1846bd81efSanket Padawe
171931891afce40b903ada9b24ec12e3648ae6aa27b2Jan Nordqvist    public static String toHexString(String s) {
172031891afce40b903ada9b24ec12e3648ae6aa27b2Jan Nordqvist        if (s == null) {
172131891afce40b903ada9b24ec12e3648ae6aa27b2Jan Nordqvist            return "null";
172231891afce40b903ada9b24ec12e3648ae6aa27b2Jan Nordqvist        }
172331891afce40b903ada9b24ec12e3648ae6aa27b2Jan Nordqvist        StringBuilder sb = new StringBuilder();
172431891afce40b903ada9b24ec12e3648ae6aa27b2Jan Nordqvist        sb.append('\'').append(s).append('\'');
172531891afce40b903ada9b24ec12e3648ae6aa27b2Jan Nordqvist        for (int n = 0; n < s.length(); n++) {
172631891afce40b903ada9b24ec12e3648ae6aa27b2Jan Nordqvist            sb.append(String.format(" %02x", s.charAt(n) & 0xffff));
172731891afce40b903ada9b24ec12e3648ae6aa27b2Jan Nordqvist        }
172831891afce40b903ada9b24ec12e3648ae6aa27b2Jan Nordqvist        return sb.toString();
17294d381bc39f5263effdae73ec99065eb299b806caVinit Deshpande    }
1730cd0af1c9ddcac881f49c815fb9da45433eb50736Vinit Deshpande
17312bfc67c9893c0a525b224d68dd73a74212b0c29fArthur Hsu    public void hideCertFromUnaffiliatedUsers(String alias) {
17322bfc67c9893c0a525b224d68dd73a74212b0c29fArthur Hsu        mCertManager.hideCertFromUnaffiliatedUsers(alias);
17332bfc67c9893c0a525b224d68dd73a74212b0c29fArthur Hsu    }
17342bfc67c9893c0a525b224d68dd73a74212b0c29fArthur Hsu
17352bfc67c9893c0a525b224d68dd73a74212b0c29fArthur Hsu    public String[] listClientCertsForCurrentUser() {
17362bfc67c9893c0a525b224d68dd73a74212b0c29fArthur Hsu        return mCertManager.listClientCertsForCurrentUser();
17372bfc67c9893c0a525b224d68dd73a74212b0c29fArthur Hsu    }
17382bfc67c9893c0a525b224d68dd73a74212b0c29fArthur Hsu
1739c065315c23f78ec462a56aec1aaa4e2a34549b8dFyodor Kupolov    /**
1740466158a6669d51541ce6c5c4e04a71dad36cdb4eRandy Pan     * Enable/disable WifiConnectivityManager at runtime
1741466158a6669d51541ce6c5c4e04a71dad36cdb4eRandy Pan     *
1742466158a6669d51541ce6c5c4e04a71dad36cdb4eRandy Pan     * @param enabled true-enable; false-disable
1743466158a6669d51541ce6c5c4e04a71dad36cdb4eRandy Pan     */
17448fcd1fe86ddc3056e1a26e69bc68f148ae0e9ddcChristopher Wiley    @Override
1745466158a6669d51541ce6c5c4e04a71dad36cdb4eRandy Pan    public void enableWifiConnectivityManager(boolean enabled) {
1746466158a6669d51541ce6c5c4e04a71dad36cdb4eRandy Pan        enforceConnectivityInternalPermission();
1747466158a6669d51541ce6c5c4e04a71dad36cdb4eRandy Pan        mWifiStateMachine.enableWifiConnectivityManager(enabled);
1748466158a6669d51541ce6c5c4e04a71dad36cdb4eRandy Pan    }
1749ff9686aec5b6ee21e01556899fb13ab55915f3c0Roshan Pius
1750ff9686aec5b6ee21e01556899fb13ab55915f3c0Roshan Pius    /**
1751ff9686aec5b6ee21e01556899fb13ab55915f3c0Roshan Pius     * Retrieve the data to be backed to save the current state.
17523031903d4a68659350994571525fc86a47c02dd0Roshan Pius     *
1753ff9686aec5b6ee21e01556899fb13ab55915f3c0Roshan Pius     * @return  Raw byte stream of the data to be backed up.
1754ff9686aec5b6ee21e01556899fb13ab55915f3c0Roshan Pius     */
1755ff9686aec5b6ee21e01556899fb13ab55915f3c0Roshan Pius    @Override
1756ff9686aec5b6ee21e01556899fb13ab55915f3c0Roshan Pius    public byte[] retrieveBackupData() {
1757ff9686aec5b6ee21e01556899fb13ab55915f3c0Roshan Pius        enforceReadCredentialPermission();
1758ff9686aec5b6ee21e01556899fb13ab55915f3c0Roshan Pius        enforceAccessPermission();
1759ff9686aec5b6ee21e01556899fb13ab55915f3c0Roshan Pius        if (mWifiStateMachineChannel == null) {
1760ff9686aec5b6ee21e01556899fb13ab55915f3c0Roshan Pius            Slog.e(TAG, "mWifiStateMachineChannel is not initialized");
1761ff9686aec5b6ee21e01556899fb13ab55915f3c0Roshan Pius            return null;
1762ff9686aec5b6ee21e01556899fb13ab55915f3c0Roshan Pius        }
1763ff9686aec5b6ee21e01556899fb13ab55915f3c0Roshan Pius
17643204fb9682242a7b5a749489076c66d448c42577Roshan Pius        Slog.d(TAG, "Retrieving backup data");
1765ff9686aec5b6ee21e01556899fb13ab55915f3c0Roshan Pius        List<WifiConfiguration> wifiConfigurations =
1766ff9686aec5b6ee21e01556899fb13ab55915f3c0Roshan Pius                mWifiStateMachine.syncGetPrivilegedConfiguredNetwork(mWifiStateMachineChannel);
17673204fb9682242a7b5a749489076c66d448c42577Roshan Pius        byte[] backupData =
17683204fb9682242a7b5a749489076c66d448c42577Roshan Pius                mWifiBackupRestore.retrieveBackupDataFromConfigurations(wifiConfigurations);
17693204fb9682242a7b5a749489076c66d448c42577Roshan Pius        Slog.d(TAG, "Retrieved backup data");
17703204fb9682242a7b5a749489076c66d448c42577Roshan Pius        return backupData;
1771ff9686aec5b6ee21e01556899fb13ab55915f3c0Roshan Pius    }
1772ff9686aec5b6ee21e01556899fb13ab55915f3c0Roshan Pius
1773ff9686aec5b6ee21e01556899fb13ab55915f3c0Roshan Pius    /**
17743031903d4a68659350994571525fc86a47c02dd0Roshan Pius     * Helper method to restore networks retrieved from backup data.
17753031903d4a68659350994571525fc86a47c02dd0Roshan Pius     *
17763031903d4a68659350994571525fc86a47c02dd0Roshan Pius     * @param configurations list of WifiConfiguration objects parsed from the backup data.
17773031903d4a68659350994571525fc86a47c02dd0Roshan Pius     */
17783031903d4a68659350994571525fc86a47c02dd0Roshan Pius    private void restoreNetworks(List<WifiConfiguration> configurations) {
17793031903d4a68659350994571525fc86a47c02dd0Roshan Pius        if (configurations == null) {
17803031903d4a68659350994571525fc86a47c02dd0Roshan Pius            Slog.e(TAG, "Backup data parse failed");
17813031903d4a68659350994571525fc86a47c02dd0Roshan Pius            return;
17823031903d4a68659350994571525fc86a47c02dd0Roshan Pius        }
17833031903d4a68659350994571525fc86a47c02dd0Roshan Pius        for (WifiConfiguration configuration : configurations) {
17843031903d4a68659350994571525fc86a47c02dd0Roshan Pius            int networkId = mWifiStateMachine.syncAddOrUpdateNetwork(
17853031903d4a68659350994571525fc86a47c02dd0Roshan Pius                    mWifiStateMachineChannel, configuration);
17863031903d4a68659350994571525fc86a47c02dd0Roshan Pius            if (networkId == WifiConfiguration.INVALID_NETWORK_ID) {
17873031903d4a68659350994571525fc86a47c02dd0Roshan Pius                Slog.e(TAG, "Restore network failed: " + configuration.configKey());
17883031903d4a68659350994571525fc86a47c02dd0Roshan Pius                continue;
17893031903d4a68659350994571525fc86a47c02dd0Roshan Pius            }
17903031903d4a68659350994571525fc86a47c02dd0Roshan Pius            // Enable all networks restored.
17913031903d4a68659350994571525fc86a47c02dd0Roshan Pius            mWifiStateMachine.syncEnableNetwork(mWifiStateMachineChannel, networkId, false);
17923031903d4a68659350994571525fc86a47c02dd0Roshan Pius        }
17933031903d4a68659350994571525fc86a47c02dd0Roshan Pius    }
17943031903d4a68659350994571525fc86a47c02dd0Roshan Pius
17953031903d4a68659350994571525fc86a47c02dd0Roshan Pius    /**
1796ff9686aec5b6ee21e01556899fb13ab55915f3c0Roshan Pius     * Restore state from the backed up data.
1797e3831b70d4a8a967fe8df5496d542a432692c434Roshan Pius     *
1798ff9686aec5b6ee21e01556899fb13ab55915f3c0Roshan Pius     * @param data Raw byte stream of the backed up data.
1799ff9686aec5b6ee21e01556899fb13ab55915f3c0Roshan Pius     */
1800ff9686aec5b6ee21e01556899fb13ab55915f3c0Roshan Pius    @Override
1801ff9686aec5b6ee21e01556899fb13ab55915f3c0Roshan Pius    public void restoreBackupData(byte[] data) {
1802ff9686aec5b6ee21e01556899fb13ab55915f3c0Roshan Pius        enforceChangePermission();
1803ff9686aec5b6ee21e01556899fb13ab55915f3c0Roshan Pius        if (mWifiStateMachineChannel == null) {
1804ff9686aec5b6ee21e01556899fb13ab55915f3c0Roshan Pius            Slog.e(TAG, "mWifiStateMachineChannel is not initialized");
1805ff9686aec5b6ee21e01556899fb13ab55915f3c0Roshan Pius            return;
1806ff9686aec5b6ee21e01556899fb13ab55915f3c0Roshan Pius        }
1807ff9686aec5b6ee21e01556899fb13ab55915f3c0Roshan Pius
18083204fb9682242a7b5a749489076c66d448c42577Roshan Pius        Slog.d(TAG, "Restoring backup data");
1809ff9686aec5b6ee21e01556899fb13ab55915f3c0Roshan Pius        List<WifiConfiguration> wifiConfigurations =
1810ff9686aec5b6ee21e01556899fb13ab55915f3c0Roshan Pius                mWifiBackupRestore.retrieveConfigurationsFromBackupData(data);
18113031903d4a68659350994571525fc86a47c02dd0Roshan Pius        restoreNetworks(wifiConfigurations);
18123204fb9682242a7b5a749489076c66d448c42577Roshan Pius        Slog.d(TAG, "Restored backup data");
1813ff9686aec5b6ee21e01556899fb13ab55915f3c0Roshan Pius    }
1814e3831b70d4a8a967fe8df5496d542a432692c434Roshan Pius
1815e3831b70d4a8a967fe8df5496d542a432692c434Roshan Pius    /**
1816e3831b70d4a8a967fe8df5496d542a432692c434Roshan Pius     * Restore state from the older supplicant back up data.
1817e3831b70d4a8a967fe8df5496d542a432692c434Roshan Pius     * The old backup data was essentially a backup of wpa_supplicant.conf & ipconfig.txt file.
1818e3831b70d4a8a967fe8df5496d542a432692c434Roshan Pius     *
1819e3831b70d4a8a967fe8df5496d542a432692c434Roshan Pius     * @param supplicantData Raw byte stream of wpa_supplicant.conf
1820e3831b70d4a8a967fe8df5496d542a432692c434Roshan Pius     * @param ipConfigData Raw byte stream of ipconfig.txt
1821e3831b70d4a8a967fe8df5496d542a432692c434Roshan Pius     */
1822e3831b70d4a8a967fe8df5496d542a432692c434Roshan Pius    public void restoreSupplicantBackupData(byte[] supplicantData, byte[] ipConfigData) {
1823e3831b70d4a8a967fe8df5496d542a432692c434Roshan Pius        enforceChangePermission();
1824e3831b70d4a8a967fe8df5496d542a432692c434Roshan Pius        if (mWifiStateMachineChannel == null) {
1825e3831b70d4a8a967fe8df5496d542a432692c434Roshan Pius            Slog.e(TAG, "mWifiStateMachineChannel is not initialized");
1826e3831b70d4a8a967fe8df5496d542a432692c434Roshan Pius            return;
1827e3831b70d4a8a967fe8df5496d542a432692c434Roshan Pius        }
1828e3831b70d4a8a967fe8df5496d542a432692c434Roshan Pius
18293204fb9682242a7b5a749489076c66d448c42577Roshan Pius        Slog.d(TAG, "Restoring supplicant backup data");
1830e3831b70d4a8a967fe8df5496d542a432692c434Roshan Pius        List<WifiConfiguration> wifiConfigurations =
1831e3831b70d4a8a967fe8df5496d542a432692c434Roshan Pius                mWifiBackupRestore.retrieveConfigurationsFromSupplicantBackupData(
1832e3831b70d4a8a967fe8df5496d542a432692c434Roshan Pius                        supplicantData, ipConfigData);
18333031903d4a68659350994571525fc86a47c02dd0Roshan Pius        restoreNetworks(wifiConfigurations);
18343204fb9682242a7b5a749489076c66d448c42577Roshan Pius        Slog.d(TAG, "Restored supplicant backup data");
1835e3831b70d4a8a967fe8df5496d542a432692c434Roshan Pius    }
1836155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande}
1837