WifiNative.java revision 590f3fc2045389d5ef274c4b3bd6162d93b1a0ac
1155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande/*
2155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande * Copyright (C) 2008 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
1968cb8c02b4ad079c54a2ffd4407da921d18c7af9Wei Wangimport android.annotation.Nullable;
20fbb081b7c28c18f0644701061a1ab38a4627db27Bartosz Fabianowskiimport android.app.AlarmManager;
21fbb081b7c28c18f0644701061a1ab38a4627db27Bartosz Fabianowskiimport android.app.PendingIntent;
22fbb081b7c28c18f0644701061a1ab38a4627db27Bartosz Fabianowskiimport android.content.BroadcastReceiver;
232a6d76f0899289cd3b96e3428f02076fdbc0363eMitchell Willsimport android.content.Context;
242a6d76f0899289cd3b96e3428f02076fdbc0363eMitchell Willsimport android.content.Intent;
252a6d76f0899289cd3b96e3428f02076fdbc0363eMitchell Willsimport android.content.IntentFilter;
26e1dab7a2e3ab5911f812a302b4beed1f6eb5aba7Paul Jensenimport android.net.apf.ApfCapabilities;
27143657392bf0702a155fe688171a5abbf4c86570Vinit Deshpandeimport android.net.wifi.RttManager;
2868cb8c02b4ad079c54a2ffd4407da921d18c7af9Wei Wangimport android.net.wifi.RttManager.ResponderConfig;
29e4e3750390bec0a849a9153348b7c21b2cc8b843Vinit Deshapndeimport android.net.wifi.ScanResult;
30dd490cf1df37b70a8625dc4ec8e712f740dd1e4aPierre Vandwalleimport android.net.wifi.WifiConfiguration;
3118786eca942042388748b0d98979f21c9dff4a89Mitchell Willsimport android.net.wifi.WifiEnterpriseConfig;
32aabe7a9f2f32915fd1f25416a6d2034a844005d6vandwalleimport android.net.wifi.WifiLinkLayerStats;
3303cdd624f5da6d62cb731304aa7505921566f69dDmitry Shmidtimport android.net.wifi.WifiManager;
34e4e3750390bec0a849a9153348b7c21b2cc8b843Vinit Deshapndeimport android.net.wifi.WifiScanner;
35dd490cf1df37b70a8625dc4ec8e712f740dd1e4aPierre Vandwalleimport android.net.wifi.WifiSsid;
362a6d76f0899289cd3b96e3428f02076fdbc0363eMitchell Willsimport android.net.wifi.WifiWakeReasonAndCounts;
37155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpandeimport android.net.wifi.WpsInfo;
38155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpandeimport android.net.wifi.p2p.WifiP2pConfig;
39155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpandeimport android.net.wifi.p2p.WifiP2pGroup;
4003cdd624f5da6d62cb731304aa7505921566f69dDmitry Shmidtimport android.net.wifi.p2p.nsd.WifiP2pServiceInfo;
41f22d23092ab37286a5ef9d257d5bb32c421d2669vandwalleimport android.os.SystemClock;
4218786eca942042388748b0d98979f21c9dff4a89Mitchell Willsimport android.os.SystemProperties;
43155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpandeimport android.text.TextUtils;
44155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpandeimport android.util.LocalLog;
45155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpandeimport android.util.Log;
46fbb081b7c28c18f0644701061a1ab38a4627db27Bartosz Fabianowski
4709b1d0786d05436d524d7556c269e665a0962ee6mukesh agrawalimport com.android.internal.annotations.Immutable;
480fcb06473cbe2824e401a80c0520bb1c14ed8f41mukesh agrawalimport com.android.internal.util.HexDump;
49c55e88163b223db0ca6a99ed6ffe91845c30a576Prerepa Viswanadhamimport com.android.server.connectivity.KeepalivePacketData;
502afa54e3c8fa1153302a0d57b0e9b7bee35406ffMitchell Willsimport com.android.server.wifi.hotspot2.NetworkDetail;
512afa54e3c8fa1153302a0d57b0e9b7bee35406ffMitchell Willsimport com.android.server.wifi.hotspot2.SupplicantBridge;
522afa54e3c8fa1153302a0d57b0e9b7bee35406ffMitchell Willsimport com.android.server.wifi.hotspot2.Utils;
53590f3fc2045389d5ef274c4b3bd6162d93b1a0acSamuel Tanimport com.android.server.wifi.util.FrameParser;
545d31cedf4024e0f038b4dfc2081016c8631ee8feMitchell Willsimport com.android.server.wifi.util.InformationElementUtil;
55c55e88163b223db0ca6a99ed6ffe91845c30a576Prerepa Viswanadham
5618786eca942042388748b0d98979f21c9dff4a89Mitchell Willsimport libcore.util.HexEncoding;
5718786eca942042388748b0d98979f21c9dff4a89Mitchell Wills
58fbb081b7c28c18f0644701061a1ab38a4627db27Bartosz Fabianowskiimport org.json.JSONException;
59fbb081b7c28c18f0644701061a1ab38a4627db27Bartosz Fabianowskiimport org.json.JSONObject;
60fbb081b7c28c18f0644701061a1ab38a4627db27Bartosz Fabianowski
610fcb06473cbe2824e401a80c0520bb1c14ed8f41mukesh agrawalimport java.io.PrintWriter;
620fcb06473cbe2824e401a80c0520bb1c14ed8f41mukesh agrawalimport java.io.StringWriter;
63fbb081b7c28c18f0644701061a1ab38a4627db27Bartosz Fabianowskiimport java.io.UnsupportedEncodingException;
64fbb081b7c28c18f0644701061a1ab38a4627db27Bartosz Fabianowskiimport java.net.URLDecoder;
65fbb081b7c28c18f0644701061a1ab38a4627db27Bartosz Fabianowskiimport java.net.URLEncoder;
665cfd8d8b9f241dcad874125a1b5538ee0d6860fexinheimport java.nio.ByteBuffer;
675cfd8d8b9f241dcad874125a1b5538ee0d6860fexinheimport java.nio.CharBuffer;
685cfd8d8b9f241dcad874125a1b5538ee0d6860fexinheimport java.nio.charset.CharacterCodingException;
695cfd8d8b9f241dcad874125a1b5538ee0d6860fexinheimport java.nio.charset.CharsetDecoder;
705cfd8d8b9f241dcad874125a1b5538ee0d6860fexinheimport java.nio.charset.StandardCharsets;
71155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpandeimport java.util.ArrayList;
723571366ac36c70746b9f013ec2b54482861c9292Randy Panimport java.util.BitSet;
73fbb081b7c28c18f0644701061a1ab38a4627db27Bartosz Fabianowskiimport java.util.HashMap;
74fbb081b7c28c18f0644701061a1ab38a4627db27Bartosz Fabianowskiimport java.util.Iterator;
75155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpandeimport java.util.List;
76155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpandeimport java.util.Locale;
77fbb081b7c28c18f0644701061a1ab38a4627db27Bartosz Fabianowskiimport java.util.Map;
789ec71f6499e0e3d6f52310a41ff4a59d2fa4f8b2Mitchell Willsimport java.util.Set;
7918786eca942042388748b0d98979f21c9dff4a89Mitchell Wills
80fbb081b7c28c18f0644701061a1ab38a4627db27Bartosz Fabianowski
81155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande/**
82155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande * Native calls for bring up/shut down of the supplicant daemon and for
83155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande * sending requests to the supplicant daemon
84155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande *
85155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande * waitForEvent() is called on the monitor thread for events. All other methods
86155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande * must be serialized from the framework.
87155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande *
88155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande * {@hide}
89155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande */
90155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpandepublic class WifiNative {
91b66b29a00da970ee75052e24f592c8d6c16bd0edxinhe    private static boolean DBG = false;
92155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
930fcb06473cbe2824e401a80c0520bb1c14ed8f41mukesh agrawal    // Must match wifi_hal.h
940fcb06473cbe2824e401a80c0520bb1c14ed8f41mukesh agrawal    public static final int WIFI_SUCCESS = 0;
950fcb06473cbe2824e401a80c0520bb1c14ed8f41mukesh agrawal
9618786eca942042388748b0d98979f21c9dff4a89Mitchell Wills    /**
9718786eca942042388748b0d98979f21c9dff4a89Mitchell Wills     * Hold this lock before calling supplicant or HAL methods
9818786eca942042388748b0d98979f21c9dff4a89Mitchell Wills     * it is required to mutually exclude access to the driver
9918786eca942042388748b0d98979f21c9dff4a89Mitchell Wills     */
100956f54b391677d78379729dd14518edddf3c7660Etan Cohen    public static final Object sLock = new Object();
101155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
102956f54b391677d78379729dd14518edddf3c7660Etan Cohen    private static final LocalLog sLocalLog = new LocalLog(16384);
103155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
10418786eca942042388748b0d98979f21c9dff4a89Mitchell Wills    public static LocalLog getLocalLog() {
105956f54b391677d78379729dd14518edddf3c7660Etan Cohen        return sLocalLog;
10618786eca942042388748b0d98979f21c9dff4a89Mitchell Wills    }
107155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
108155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    /* Register native functions */
109155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    static {
110155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        /* Native functions are defined in libwifi-service.so */
111155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        System.loadLibrary("wifi-service");
112155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        registerNatives();
113155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    }
114155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
115155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    private static native int registerNatives();
116155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
11718786eca942042388748b0d98979f21c9dff4a89Mitchell Wills    /*
11818786eca942042388748b0d98979f21c9dff4a89Mitchell Wills     * Singleton WifiNative instances
119155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande     */
12018786eca942042388748b0d98979f21c9dff4a89Mitchell Wills    private static WifiNative wlanNativeInterface =
121c6ccad1ec19b0a53266962237774422b156ea726Ningyuan Wang            new WifiNative(SystemProperties.get("wifi.interface", "wlan0"), true);
12218786eca942042388748b0d98979f21c9dff4a89Mitchell Wills    public static WifiNative getWlanNativeInterface() {
12318786eca942042388748b0d98979f21c9dff4a89Mitchell Wills        return wlanNativeInterface;
12418786eca942042388748b0d98979f21c9dff4a89Mitchell Wills    }
125155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
126c6ccad1ec19b0a53266962237774422b156ea726Ningyuan Wang    private static WifiNative p2pNativeInterface =
127c6ccad1ec19b0a53266962237774422b156ea726Ningyuan Wang            // commands for p2p0 interface don't need prefix
128c6ccad1ec19b0a53266962237774422b156ea726Ningyuan Wang            new WifiNative(SystemProperties.get("wifi.direct.interface", "p2p0"), false);
12918786eca942042388748b0d98979f21c9dff4a89Mitchell Wills    public static WifiNative getP2pNativeInterface() {
13018786eca942042388748b0d98979f21c9dff4a89Mitchell Wills        return p2pNativeInterface;
13118786eca942042388748b0d98979f21c9dff4a89Mitchell Wills    }
132155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
133155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
13418786eca942042388748b0d98979f21c9dff4a89Mitchell Wills    private final String mTAG;
13518786eca942042388748b0d98979f21c9dff4a89Mitchell Wills    private final String mInterfaceName;
13618786eca942042388748b0d98979f21c9dff4a89Mitchell Wills    private final String mInterfacePrefix;
13718786eca942042388748b0d98979f21c9dff4a89Mitchell Wills
13818786eca942042388748b0d98979f21c9dff4a89Mitchell Wills    private Context mContext = null;
13918786eca942042388748b0d98979f21c9dff4a89Mitchell Wills    public void initContext(Context context) {
14018786eca942042388748b0d98979f21c9dff4a89Mitchell Wills        if (mContext == null && context != null) {
14118786eca942042388748b0d98979f21c9dff4a89Mitchell Wills            mContext = context;
14218786eca942042388748b0d98979f21c9dff4a89Mitchell Wills        }
14318786eca942042388748b0d98979f21c9dff4a89Mitchell Wills    }
144155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
145c6ccad1ec19b0a53266962237774422b156ea726Ningyuan Wang    private WifiNative(String interfaceName,
146c6ccad1ec19b0a53266962237774422b156ea726Ningyuan Wang                       boolean requiresPrefix) {
147155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        mInterfaceName = interfaceName;
148155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        mTAG = "WifiNative-" + interfaceName;
149e26ad459b63271548abbdeba4f8d77fcca9f88bdxinhe
150c6ccad1ec19b0a53266962237774422b156ea726Ningyuan Wang        if (requiresPrefix) {
151155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande            mInterfacePrefix = "IFNAME=" + interfaceName + " ";
152155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        } else {
153155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande            mInterfacePrefix = "";
154155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        }
155155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    }
156155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
15718786eca942042388748b0d98979f21c9dff4a89Mitchell Wills    public String getInterfaceName() {
15818786eca942042388748b0d98979f21c9dff4a89Mitchell Wills        return mInterfaceName;
159e26ad459b63271548abbdeba4f8d77fcca9f88bdxinhe    }
160e26ad459b63271548abbdeba4f8d77fcca9f88bdxinhe
16118786eca942042388748b0d98979f21c9dff4a89Mitchell Wills    // Note this affects logging on for all interfaces
162ed9938883ae2dade81c8be6cd6ceaef3febd5239vandwalle    void enableVerboseLogging(int verbose) {
163ed9938883ae2dade81c8be6cd6ceaef3febd5239vandwalle        if (verbose > 0) {
164ed9938883ae2dade81c8be6cd6ceaef3febd5239vandwalle            DBG = true;
165ed9938883ae2dade81c8be6cd6ceaef3febd5239vandwalle        } else {
166ed9938883ae2dade81c8be6cd6ceaef3febd5239vandwalle            DBG = false;
167ed9938883ae2dade81c8be6cd6ceaef3febd5239vandwalle        }
168ed9938883ae2dade81c8be6cd6ceaef3febd5239vandwalle    }
169ed9938883ae2dade81c8be6cd6ceaef3febd5239vandwalle
17018786eca942042388748b0d98979f21c9dff4a89Mitchell Wills    private void localLog(String s) {
171956f54b391677d78379729dd14518edddf3c7660Etan Cohen        if (sLocalLog != null) sLocalLog.log(mInterfaceName + ": " + s);
17218786eca942042388748b0d98979f21c9dff4a89Mitchell Wills    }
173155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
174155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
17518786eca942042388748b0d98979f21c9dff4a89Mitchell Wills
17618786eca942042388748b0d98979f21c9dff4a89Mitchell Wills    /*
17718786eca942042388748b0d98979f21c9dff4a89Mitchell Wills     * Driver and Supplicant management
17818786eca942042388748b0d98979f21c9dff4a89Mitchell Wills     */
17918786eca942042388748b0d98979f21c9dff4a89Mitchell Wills    private native static boolean loadDriverNative();
18018786eca942042388748b0d98979f21c9dff4a89Mitchell Wills    public boolean loadDriver() {
181956f54b391677d78379729dd14518edddf3c7660Etan Cohen        synchronized (sLock) {
18218786eca942042388748b0d98979f21c9dff4a89Mitchell Wills            return loadDriverNative();
18318786eca942042388748b0d98979f21c9dff4a89Mitchell Wills        }
184155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    }
185155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
18618786eca942042388748b0d98979f21c9dff4a89Mitchell Wills    private native static boolean isDriverLoadedNative();
18718786eca942042388748b0d98979f21c9dff4a89Mitchell Wills    public boolean isDriverLoaded() {
188956f54b391677d78379729dd14518edddf3c7660Etan Cohen        synchronized (sLock) {
18918786eca942042388748b0d98979f21c9dff4a89Mitchell Wills            return isDriverLoadedNative();
19018786eca942042388748b0d98979f21c9dff4a89Mitchell Wills        }
191155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    }
192155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
19318786eca942042388748b0d98979f21c9dff4a89Mitchell Wills    private native static boolean unloadDriverNative();
19418786eca942042388748b0d98979f21c9dff4a89Mitchell Wills    public boolean unloadDriver() {
195956f54b391677d78379729dd14518edddf3c7660Etan Cohen        synchronized (sLock) {
19618786eca942042388748b0d98979f21c9dff4a89Mitchell Wills            return unloadDriverNative();
19718786eca942042388748b0d98979f21c9dff4a89Mitchell Wills        }
19818786eca942042388748b0d98979f21c9dff4a89Mitchell Wills    }
19918786eca942042388748b0d98979f21c9dff4a89Mitchell Wills
20018786eca942042388748b0d98979f21c9dff4a89Mitchell Wills    private native static boolean startSupplicantNative(boolean p2pSupported);
20118786eca942042388748b0d98979f21c9dff4a89Mitchell Wills    public boolean startSupplicant(boolean p2pSupported) {
202956f54b391677d78379729dd14518edddf3c7660Etan Cohen        synchronized (sLock) {
20318786eca942042388748b0d98979f21c9dff4a89Mitchell Wills            return startSupplicantNative(p2pSupported);
20418786eca942042388748b0d98979f21c9dff4a89Mitchell Wills        }
20518786eca942042388748b0d98979f21c9dff4a89Mitchell Wills    }
20618786eca942042388748b0d98979f21c9dff4a89Mitchell Wills
20718786eca942042388748b0d98979f21c9dff4a89Mitchell Wills    /* Sends a kill signal to supplicant. To be used when we have lost connection
20818786eca942042388748b0d98979f21c9dff4a89Mitchell Wills       or when the supplicant is hung */
20918786eca942042388748b0d98979f21c9dff4a89Mitchell Wills    private native static boolean killSupplicantNative(boolean p2pSupported);
21018786eca942042388748b0d98979f21c9dff4a89Mitchell Wills    public boolean killSupplicant(boolean p2pSupported) {
211956f54b391677d78379729dd14518edddf3c7660Etan Cohen        synchronized (sLock) {
21218786eca942042388748b0d98979f21c9dff4a89Mitchell Wills            return killSupplicantNative(p2pSupported);
21318786eca942042388748b0d98979f21c9dff4a89Mitchell Wills        }
214155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    }
215155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
21618786eca942042388748b0d98979f21c9dff4a89Mitchell Wills    private native static boolean connectToSupplicantNative();
217155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    public boolean connectToSupplicant() {
218956f54b391677d78379729dd14518edddf3c7660Etan Cohen        synchronized (sLock) {
219b577f391af2c484e443c19b3df1d62cc0924692aVinit Deshpande            localLog(mInterfacePrefix + "connectToSupplicant");
220b577f391af2c484e443c19b3df1d62cc0924692aVinit Deshpande            return connectToSupplicantNative();
221b577f391af2c484e443c19b3df1d62cc0924692aVinit Deshpande        }
222155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    }
223155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
22418786eca942042388748b0d98979f21c9dff4a89Mitchell Wills    private native static void closeSupplicantConnectionNative();
225155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    public void closeSupplicantConnection() {
226956f54b391677d78379729dd14518edddf3c7660Etan Cohen        synchronized (sLock) {
227b577f391af2c484e443c19b3df1d62cc0924692aVinit Deshpande            localLog(mInterfacePrefix + "closeSupplicantConnection");
228b577f391af2c484e443c19b3df1d62cc0924692aVinit Deshpande            closeSupplicantConnectionNative();
229b577f391af2c484e443c19b3df1d62cc0924692aVinit Deshpande        }
230155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    }
231155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
23218786eca942042388748b0d98979f21c9dff4a89Mitchell Wills    /**
23318786eca942042388748b0d98979f21c9dff4a89Mitchell Wills     * Wait for the supplicant to send an event, returning the event string.
23418786eca942042388748b0d98979f21c9dff4a89Mitchell Wills     * @return the event string sent by the supplicant.
23518786eca942042388748b0d98979f21c9dff4a89Mitchell Wills     */
23618786eca942042388748b0d98979f21c9dff4a89Mitchell Wills    private native static String waitForEventNative();
237155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    public String waitForEvent() {
238155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        // No synchronization necessary .. it is implemented in WifiMonitor
239155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        return waitForEventNative();
240155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    }
241155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
24218786eca942042388748b0d98979f21c9dff4a89Mitchell Wills
24318786eca942042388748b0d98979f21c9dff4a89Mitchell Wills    /*
24418786eca942042388748b0d98979f21c9dff4a89Mitchell Wills     * Supplicant Command Primitives
24518786eca942042388748b0d98979f21c9dff4a89Mitchell Wills     */
24618786eca942042388748b0d98979f21c9dff4a89Mitchell Wills    private native boolean doBooleanCommandNative(String command);
24718786eca942042388748b0d98979f21c9dff4a89Mitchell Wills
24818786eca942042388748b0d98979f21c9dff4a89Mitchell Wills    private native int doIntCommandNative(String command);
24918786eca942042388748b0d98979f21c9dff4a89Mitchell Wills
25018786eca942042388748b0d98979f21c9dff4a89Mitchell Wills    private native String doStringCommandNative(String command);
25118786eca942042388748b0d98979f21c9dff4a89Mitchell Wills
252155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    private boolean doBooleanCommand(String command) {
253155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        if (DBG) Log.d(mTAG, "doBoolean: " + command);
254956f54b391677d78379729dd14518edddf3c7660Etan Cohen        synchronized (sLock) {
25518786eca942042388748b0d98979f21c9dff4a89Mitchell Wills            String toLog = mInterfacePrefix + command;
256155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande            boolean result = doBooleanCommandNative(mInterfacePrefix + command);
2577b581f46f6c9bc6edf0edd287d47106712fb2144vandwalle            localLog(toLog + " -> " + result);
2580888ce6f90bdaeee799dd8361ea4781e23a33b87vandwalle            if (DBG) Log.d(mTAG, command + ": returned " + result);
259155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande            return result;
260155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        }
261155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    }
262155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
263a6777abfc90496801e9942f57fbfa091ba85ae82Prerepa Viswanadham    private boolean doBooleanCommandWithoutLogging(String command) {
264a6777abfc90496801e9942f57fbfa091ba85ae82Prerepa Viswanadham        if (DBG) Log.d(mTAG, "doBooleanCommandWithoutLogging: " + command);
265956f54b391677d78379729dd14518edddf3c7660Etan Cohen        synchronized (sLock) {
266a6777abfc90496801e9942f57fbfa091ba85ae82Prerepa Viswanadham            boolean result = doBooleanCommandNative(mInterfacePrefix + command);
267a6777abfc90496801e9942f57fbfa091ba85ae82Prerepa Viswanadham            if (DBG) Log.d(mTAG, command + ": returned " + result);
268a6777abfc90496801e9942f57fbfa091ba85ae82Prerepa Viswanadham            return result;
269a6777abfc90496801e9942f57fbfa091ba85ae82Prerepa Viswanadham        }
270a6777abfc90496801e9942f57fbfa091ba85ae82Prerepa Viswanadham    }
271a6777abfc90496801e9942f57fbfa091ba85ae82Prerepa Viswanadham
272155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    private int doIntCommand(String command) {
273155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        if (DBG) Log.d(mTAG, "doInt: " + command);
274956f54b391677d78379729dd14518edddf3c7660Etan Cohen        synchronized (sLock) {
27518786eca942042388748b0d98979f21c9dff4a89Mitchell Wills            String toLog = mInterfacePrefix + command;
276155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande            int result = doIntCommandNative(mInterfacePrefix + command);
2777b581f46f6c9bc6edf0edd287d47106712fb2144vandwalle            localLog(toLog + " -> " + result);
278155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande            if (DBG) Log.d(mTAG, "   returned " + result);
279155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande            return result;
280155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        }
281155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    }
282155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
283155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    private String doStringCommand(String command) {
2840888ce6f90bdaeee799dd8361ea4781e23a33b87vandwalle        if (DBG) {
2850888ce6f90bdaeee799dd8361ea4781e23a33b87vandwalle            //GET_NETWORK commands flood the logs
2860888ce6f90bdaeee799dd8361ea4781e23a33b87vandwalle            if (!command.startsWith("GET_NETWORK")) {
2870888ce6f90bdaeee799dd8361ea4781e23a33b87vandwalle                Log.d(mTAG, "doString: [" + command + "]");
2880888ce6f90bdaeee799dd8361ea4781e23a33b87vandwalle            }
2890888ce6f90bdaeee799dd8361ea4781e23a33b87vandwalle        }
290956f54b391677d78379729dd14518edddf3c7660Etan Cohen        synchronized (sLock) {
29118786eca942042388748b0d98979f21c9dff4a89Mitchell Wills            String toLog = mInterfacePrefix + command;
292155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande            String result = doStringCommandNative(mInterfacePrefix + command);
29340ff222cec1bd05879edb53abc75c6deead734cavandwalle            if (result == null) {
29440ff222cec1bd05879edb53abc75c6deead734cavandwalle                if (DBG) Log.d(mTAG, "doStringCommandNative no result");
29540ff222cec1bd05879edb53abc75c6deead734cavandwalle            } else {
2967b581f46f6c9bc6edf0edd287d47106712fb2144vandwalle                if (!command.startsWith("STATUS-")) {
2977b581f46f6c9bc6edf0edd287d47106712fb2144vandwalle                    localLog(toLog + " -> " + result);
2987b581f46f6c9bc6edf0edd287d47106712fb2144vandwalle                }
29940ff222cec1bd05879edb53abc75c6deead734cavandwalle                if (DBG) Log.d(mTAG, "   returned " + result.replace("\n", " "));
30040ff222cec1bd05879edb53abc75c6deead734cavandwalle            }
301155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande            return result;
302155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        }
303155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    }
304155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
305155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    private String doStringCommandWithoutLogging(String command) {
3060888ce6f90bdaeee799dd8361ea4781e23a33b87vandwalle        if (DBG) {
3070888ce6f90bdaeee799dd8361ea4781e23a33b87vandwalle            //GET_NETWORK commands flood the logs
3080888ce6f90bdaeee799dd8361ea4781e23a33b87vandwalle            if (!command.startsWith("GET_NETWORK")) {
3090888ce6f90bdaeee799dd8361ea4781e23a33b87vandwalle                Log.d(mTAG, "doString: [" + command + "]");
3100888ce6f90bdaeee799dd8361ea4781e23a33b87vandwalle            }
31127355a942653264388e909a4276196ee63e57811vandwalle        }
312956f54b391677d78379729dd14518edddf3c7660Etan Cohen        synchronized (sLock) {
313155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande            return doStringCommandNative(mInterfacePrefix + command);
314155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        }
315155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    }
316155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
31718786eca942042388748b0d98979f21c9dff4a89Mitchell Wills    public String doCustomSupplicantCommand(String command) {
31818786eca942042388748b0d98979f21c9dff4a89Mitchell Wills        return doStringCommand(command);
31918786eca942042388748b0d98979f21c9dff4a89Mitchell Wills    }
32018786eca942042388748b0d98979f21c9dff4a89Mitchell Wills
32118786eca942042388748b0d98979f21c9dff4a89Mitchell Wills    /*
32218786eca942042388748b0d98979f21c9dff4a89Mitchell Wills     * Wrappers for supplicant commands
32318786eca942042388748b0d98979f21c9dff4a89Mitchell Wills     */
324155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    public boolean ping() {
325155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        String pong = doStringCommand("PING");
326155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        return (pong != null && pong.equals("PONG"));
327155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    }
328155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
329ad7319939c424d42fa6a3791c47f613db8ef3cd8vandwalle    public void setSupplicantLogLevel(String level) {
330ad7319939c424d42fa6a3791c47f613db8ef3cd8vandwalle        doStringCommand("LOG_LEVEL " + level);
331ad7319939c424d42fa6a3791c47f613db8ef3cd8vandwalle    }
332ad7319939c424d42fa6a3791c47f613db8ef3cd8vandwalle
333a0009d14075b5345b8f916c3fb3f2260c938cb9dYuhao Zheng    public String getFreqCapability() {
334a0009d14075b5345b8f916c3fb3f2260c938cb9dYuhao Zheng        return doStringCommand("GET_CAPABILITY freq");
335a0009d14075b5345b8f916c3fb3f2260c938cb9dYuhao Zheng    }
336a0009d14075b5345b8f916c3fb3f2260c938cb9dYuhao Zheng
3373d995b8a3a508e957e3dac2e508956020cd9abfeRoshan Pius    /**
3383d995b8a3a508e957e3dac2e508956020cd9abfeRoshan Pius     * Create a comma separate string from integer set.
3393d995b8a3a508e957e3dac2e508956020cd9abfeRoshan Pius     * @param values List of integers.
3403d995b8a3a508e957e3dac2e508956020cd9abfeRoshan Pius     * @return comma separated string.
3413d995b8a3a508e957e3dac2e508956020cd9abfeRoshan Pius     */
3423d995b8a3a508e957e3dac2e508956020cd9abfeRoshan Pius    private static String createCSVStringFromIntegerSet(Set<Integer> values) {
3433d995b8a3a508e957e3dac2e508956020cd9abfeRoshan Pius        StringBuilder list = new StringBuilder();
3443d995b8a3a508e957e3dac2e508956020cd9abfeRoshan Pius        boolean first = true;
3453d995b8a3a508e957e3dac2e508956020cd9abfeRoshan Pius        for (Integer value : values) {
3463d995b8a3a508e957e3dac2e508956020cd9abfeRoshan Pius            if (!first) {
3473d995b8a3a508e957e3dac2e508956020cd9abfeRoshan Pius                list.append(",");
3483d995b8a3a508e957e3dac2e508956020cd9abfeRoshan Pius            }
3493d995b8a3a508e957e3dac2e508956020cd9abfeRoshan Pius            list.append(value);
3503d995b8a3a508e957e3dac2e508956020cd9abfeRoshan Pius            first = false;
3513d995b8a3a508e957e3dac2e508956020cd9abfeRoshan Pius        }
3523d995b8a3a508e957e3dac2e508956020cd9abfeRoshan Pius        return list.toString();
3533d995b8a3a508e957e3dac2e508956020cd9abfeRoshan Pius    }
35418786eca942042388748b0d98979f21c9dff4a89Mitchell Wills
355ed6a985c7b63e295248fa7e8292c99b48b7a4283Mitchell Wills    /**
356ed6a985c7b63e295248fa7e8292c99b48b7a4283Mitchell Wills     * Start a scan using wpa_supplicant for the given frequencies.
3573d995b8a3a508e957e3dac2e508956020cd9abfeRoshan Pius     * @param freqs list of frequencies to scan for, if null scan all supported channels.
3583d995b8a3a508e957e3dac2e508956020cd9abfeRoshan Pius     * @param hiddenNetworkIds List of hidden networks to be scanned for.
359ed6a985c7b63e295248fa7e8292c99b48b7a4283Mitchell Wills     */
3603d995b8a3a508e957e3dac2e508956020cd9abfeRoshan Pius    public boolean scan(Set<Integer> freqs, Set<Integer> hiddenNetworkIds) {
3613d995b8a3a508e957e3dac2e508956020cd9abfeRoshan Pius        String freqList = null;
3623d995b8a3a508e957e3dac2e508956020cd9abfeRoshan Pius        String hiddenNetworkIdList = null;
3633d995b8a3a508e957e3dac2e508956020cd9abfeRoshan Pius        if (freqs != null && freqs.size() != 0) {
3643d995b8a3a508e957e3dac2e508956020cd9abfeRoshan Pius            freqList = createCSVStringFromIntegerSet(freqs);
3653d995b8a3a508e957e3dac2e508956020cd9abfeRoshan Pius        }
3663d995b8a3a508e957e3dac2e508956020cd9abfeRoshan Pius        if (hiddenNetworkIds != null && hiddenNetworkIds.size() != 0) {
3673d995b8a3a508e957e3dac2e508956020cd9abfeRoshan Pius            hiddenNetworkIdList = createCSVStringFromIntegerSet(hiddenNetworkIds);
3689ec71f6499e0e3d6f52310a41ff4a59d2fa4f8b2Mitchell Wills        }
3693d995b8a3a508e957e3dac2e508956020cd9abfeRoshan Pius        return scanWithParams(freqList, hiddenNetworkIdList);
3709ec71f6499e0e3d6f52310a41ff4a59d2fa4f8b2Mitchell Wills    }
3719ec71f6499e0e3d6f52310a41ff4a59d2fa4f8b2Mitchell Wills
3723d995b8a3a508e957e3dac2e508956020cd9abfeRoshan Pius    private boolean scanWithParams(String freqList, String hiddenNetworkIdList) {
3733d995b8a3a508e957e3dac2e508956020cd9abfeRoshan Pius        StringBuilder scanCommand = new StringBuilder();
3743d995b8a3a508e957e3dac2e508956020cd9abfeRoshan Pius        scanCommand.append("SCAN TYPE=ONLY");
3753d995b8a3a508e957e3dac2e508956020cd9abfeRoshan Pius        if (freqList != null) {
3763d995b8a3a508e957e3dac2e508956020cd9abfeRoshan Pius            scanCommand.append(" freq=" + freqList);
3773d995b8a3a508e957e3dac2e508956020cd9abfeRoshan Pius        }
3783d995b8a3a508e957e3dac2e508956020cd9abfeRoshan Pius        if (hiddenNetworkIdList != null) {
3793d995b8a3a508e957e3dac2e508956020cd9abfeRoshan Pius            scanCommand.append(" scan_id=" + hiddenNetworkIdList);
380155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        }
3813d995b8a3a508e957e3dac2e508956020cd9abfeRoshan Pius        return doBooleanCommand(scanCommand.toString());
382155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    }
383155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
384155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    /* Does a graceful shutdown of supplicant. Is a common stop function for both p2p and sta.
385155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande     *
386155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande     * Note that underneath we use a harsh-sounding "terminate" supplicant command
387155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande     * for a graceful stop and a mild-sounding "stop" interface
388155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande     * to kill the process
389155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande     */
390155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    public boolean stopSupplicant() {
391155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        return doBooleanCommand("TERMINATE");
392155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    }
393155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
394155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    public String listNetworks() {
395155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        return doStringCommand("LIST_NETWORKS");
396155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    }
397155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
398e3939cb40d9ba3842be105a6e85172dc06e14758Vinit Deshpande    public String listNetworks(int last_id) {
399e3939cb40d9ba3842be105a6e85172dc06e14758Vinit Deshpande        return doStringCommand("LIST_NETWORKS LAST_ID=" + last_id);
400e3939cb40d9ba3842be105a6e85172dc06e14758Vinit Deshpande    }
401e3939cb40d9ba3842be105a6e85172dc06e14758Vinit Deshpande
402155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    public int addNetwork() {
403155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        return doIntCommand("ADD_NETWORK");
404155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    }
405155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
406fbb081b7c28c18f0644701061a1ab38a4627db27Bartosz Fabianowski    public boolean setNetworkExtra(int netId, String name, Map<String, String> values) {
407fbb081b7c28c18f0644701061a1ab38a4627db27Bartosz Fabianowski        final String encoded;
408fbb081b7c28c18f0644701061a1ab38a4627db27Bartosz Fabianowski        try {
409fbb081b7c28c18f0644701061a1ab38a4627db27Bartosz Fabianowski            encoded = URLEncoder.encode(new JSONObject(values).toString(), "UTF-8");
410fbb081b7c28c18f0644701061a1ab38a4627db27Bartosz Fabianowski        } catch (NullPointerException e) {
411fbb081b7c28c18f0644701061a1ab38a4627db27Bartosz Fabianowski            Log.e(TAG, "Unable to serialize networkExtra: " + e.toString());
412fbb081b7c28c18f0644701061a1ab38a4627db27Bartosz Fabianowski            return false;
413fbb081b7c28c18f0644701061a1ab38a4627db27Bartosz Fabianowski        } catch (UnsupportedEncodingException e) {
414fbb081b7c28c18f0644701061a1ab38a4627db27Bartosz Fabianowski            Log.e(TAG, "Unable to serialize networkExtra: " + e.toString());
415fbb081b7c28c18f0644701061a1ab38a4627db27Bartosz Fabianowski            return false;
416fbb081b7c28c18f0644701061a1ab38a4627db27Bartosz Fabianowski        }
417fbb081b7c28c18f0644701061a1ab38a4627db27Bartosz Fabianowski        return setNetworkVariable(netId, name, "\"" + encoded + "\"");
418fbb081b7c28c18f0644701061a1ab38a4627db27Bartosz Fabianowski    }
419fbb081b7c28c18f0644701061a1ab38a4627db27Bartosz Fabianowski
420155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    public boolean setNetworkVariable(int netId, String name, String value) {
421155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        if (TextUtils.isEmpty(name) || TextUtils.isEmpty(value)) return false;
422a6777abfc90496801e9942f57fbfa091ba85ae82Prerepa Viswanadham        if (name.equals(WifiConfiguration.pskVarName)
423a6777abfc90496801e9942f57fbfa091ba85ae82Prerepa Viswanadham                || name.equals(WifiEnterpriseConfig.PASSWORD_KEY)) {
424a6777abfc90496801e9942f57fbfa091ba85ae82Prerepa Viswanadham            return doBooleanCommandWithoutLogging("SET_NETWORK " + netId + " " + name + " " + value);
425a6777abfc90496801e9942f57fbfa091ba85ae82Prerepa Viswanadham        } else {
426a6777abfc90496801e9942f57fbfa091ba85ae82Prerepa Viswanadham            return doBooleanCommand("SET_NETWORK " + netId + " " + name + " " + value);
427a6777abfc90496801e9942f57fbfa091ba85ae82Prerepa Viswanadham        }
428155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    }
429155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
430fbb081b7c28c18f0644701061a1ab38a4627db27Bartosz Fabianowski    public Map<String, String> getNetworkExtra(int netId, String name) {
431fbb081b7c28c18f0644701061a1ab38a4627db27Bartosz Fabianowski        final String wrapped = getNetworkVariable(netId, name);
432fbb081b7c28c18f0644701061a1ab38a4627db27Bartosz Fabianowski        if (wrapped == null || !wrapped.startsWith("\"") || !wrapped.endsWith("\"")) {
433fbb081b7c28c18f0644701061a1ab38a4627db27Bartosz Fabianowski            return null;
434fbb081b7c28c18f0644701061a1ab38a4627db27Bartosz Fabianowski        }
435fbb081b7c28c18f0644701061a1ab38a4627db27Bartosz Fabianowski        try {
436fbb081b7c28c18f0644701061a1ab38a4627db27Bartosz Fabianowski            final String encoded = wrapped.substring(1, wrapped.length() - 1);
437fbb081b7c28c18f0644701061a1ab38a4627db27Bartosz Fabianowski            // This method reads a JSON dictionary that was written by setNetworkExtra(). However,
438fbb081b7c28c18f0644701061a1ab38a4627db27Bartosz Fabianowski            // on devices that upgraded from Marshmallow, it may encounter a legacy value instead -
439fbb081b7c28c18f0644701061a1ab38a4627db27Bartosz Fabianowski            // an FQDN stored as a plain string. If such a value is encountered, the JSONObject
440fbb081b7c28c18f0644701061a1ab38a4627db27Bartosz Fabianowski            // constructor will thrown a JSONException and the method will return null.
441fbb081b7c28c18f0644701061a1ab38a4627db27Bartosz Fabianowski            final JSONObject json = new JSONObject(URLDecoder.decode(encoded, "UTF-8"));
442fbb081b7c28c18f0644701061a1ab38a4627db27Bartosz Fabianowski            final Map<String, String> values = new HashMap<String, String>();
443fbb081b7c28c18f0644701061a1ab38a4627db27Bartosz Fabianowski            final Iterator<?> it = json.keys();
444fbb081b7c28c18f0644701061a1ab38a4627db27Bartosz Fabianowski            while (it.hasNext()) {
445fbb081b7c28c18f0644701061a1ab38a4627db27Bartosz Fabianowski                final String key = (String) it.next();
446fbb081b7c28c18f0644701061a1ab38a4627db27Bartosz Fabianowski                final Object value = json.get(key);
447fbb081b7c28c18f0644701061a1ab38a4627db27Bartosz Fabianowski                if (value instanceof String) {
448fbb081b7c28c18f0644701061a1ab38a4627db27Bartosz Fabianowski                    values.put(key, (String) value);
449fbb081b7c28c18f0644701061a1ab38a4627db27Bartosz Fabianowski                }
450fbb081b7c28c18f0644701061a1ab38a4627db27Bartosz Fabianowski            }
451fbb081b7c28c18f0644701061a1ab38a4627db27Bartosz Fabianowski            return values;
452fbb081b7c28c18f0644701061a1ab38a4627db27Bartosz Fabianowski        } catch (UnsupportedEncodingException e) {
45364697f7a5b9c1c39f8c9d9c225b8ca4c798422c2Samuel Tan            Log.e(TAG, "Unable to deserialize networkExtra: " + e.toString());
454fbb081b7c28c18f0644701061a1ab38a4627db27Bartosz Fabianowski            return null;
455fbb081b7c28c18f0644701061a1ab38a4627db27Bartosz Fabianowski        } catch (JSONException e) {
456fbb081b7c28c18f0644701061a1ab38a4627db27Bartosz Fabianowski            // This is not necessarily an error. This exception will also occur if we encounter a
457fbb081b7c28c18f0644701061a1ab38a4627db27Bartosz Fabianowski            // legacy FQDN stored as a plain string. We want to return null in this case as no JSON
458fbb081b7c28c18f0644701061a1ab38a4627db27Bartosz Fabianowski            // dictionary of extras was found.
459fbb081b7c28c18f0644701061a1ab38a4627db27Bartosz Fabianowski            return null;
460fbb081b7c28c18f0644701061a1ab38a4627db27Bartosz Fabianowski        }
461fbb081b7c28c18f0644701061a1ab38a4627db27Bartosz Fabianowski    }
462fbb081b7c28c18f0644701061a1ab38a4627db27Bartosz Fabianowski
463155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    public String getNetworkVariable(int netId, String name) {
464155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        if (TextUtils.isEmpty(name)) return null;
465155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
466155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        // GET_NETWORK will likely flood the logs ...
467155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        return doStringCommandWithoutLogging("GET_NETWORK " + netId + " " + name);
468155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    }
469155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
470155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    public boolean removeNetwork(int netId) {
471155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        return doBooleanCommand("REMOVE_NETWORK " + netId);
472155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    }
473155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
474f22d23092ab37286a5ef9d257d5bb32c421d2669vandwalle
475f22d23092ab37286a5ef9d257d5bb32c421d2669vandwalle    private void logDbg(String debug) {
476f22d23092ab37286a5ef9d257d5bb32c421d2669vandwalle        long now = SystemClock.elapsedRealtimeNanos();
477f22d23092ab37286a5ef9d257d5bb32c421d2669vandwalle        String ts = String.format("[%,d us] ", now/1000);
478ecd2b88214b5d214fd1f63a9560caff9058912ddvandwalle        Log.e("WifiNative: ", ts+debug+ " stack:"
479ecd2b88214b5d214fd1f63a9560caff9058912ddvandwalle                + Thread.currentThread().getStackTrace()[2].getMethodName() +" - "
480ecd2b88214b5d214fd1f63a9560caff9058912ddvandwalle                + Thread.currentThread().getStackTrace()[3].getMethodName() +" - "
481ecd2b88214b5d214fd1f63a9560caff9058912ddvandwalle                + Thread.currentThread().getStackTrace()[4].getMethodName() +" - "
482ecd2b88214b5d214fd1f63a9560caff9058912ddvandwalle                + Thread.currentThread().getStackTrace()[5].getMethodName()+" - "
483ecd2b88214b5d214fd1f63a9560caff9058912ddvandwalle                + Thread.currentThread().getStackTrace()[6].getMethodName());
484f22d23092ab37286a5ef9d257d5bb32c421d2669vandwalle
485f22d23092ab37286a5ef9d257d5bb32c421d2669vandwalle    }
4869d7489491984e86915b2cf4fac38a882de1c8cdbRoshan Pius
4879d7489491984e86915b2cf4fac38a882de1c8cdbRoshan Pius    /**
4889d7489491984e86915b2cf4fac38a882de1c8cdbRoshan Pius     * Enables a network in wpa_supplicant.
4899d7489491984e86915b2cf4fac38a882de1c8cdbRoshan Pius     * @param netId - Network ID of the network to be enabled.
4909d7489491984e86915b2cf4fac38a882de1c8cdbRoshan Pius     * @return true if command succeeded, false otherwise.
4919d7489491984e86915b2cf4fac38a882de1c8cdbRoshan Pius     */
4929d7489491984e86915b2cf4fac38a882de1c8cdbRoshan Pius    public boolean enableNetwork(int netId) {
4939d7489491984e86915b2cf4fac38a882de1c8cdbRoshan Pius        if (DBG) logDbg("enableNetwork nid=" + Integer.toString(netId));
4949d7489491984e86915b2cf4fac38a882de1c8cdbRoshan Pius        return doBooleanCommand("ENABLE_NETWORK " + netId);
495155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    }
496155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
4979d7489491984e86915b2cf4fac38a882de1c8cdbRoshan Pius    /**
498163d510aac068682c390664e51e83bbf870fe5f5Sunil Dutt     * Enable a network in wpa_supplicant, do not connect.
499163d510aac068682c390664e51e83bbf870fe5f5Sunil Dutt     * @param netId - Network ID of the network to be enabled.
500163d510aac068682c390664e51e83bbf870fe5f5Sunil Dutt     * @return true if command succeeded, false otherwise.
501163d510aac068682c390664e51e83bbf870fe5f5Sunil Dutt     */
502163d510aac068682c390664e51e83bbf870fe5f5Sunil Dutt    public boolean enableNetworkWithoutConnect(int netId) {
503163d510aac068682c390664e51e83bbf870fe5f5Sunil Dutt        if (DBG) logDbg("enableNetworkWithoutConnect nid=" + Integer.toString(netId));
504163d510aac068682c390664e51e83bbf870fe5f5Sunil Dutt        return doBooleanCommand("ENABLE_NETWORK " + netId + " " + "no-connect");
505163d510aac068682c390664e51e83bbf870fe5f5Sunil Dutt    }
506163d510aac068682c390664e51e83bbf870fe5f5Sunil Dutt
507163d510aac068682c390664e51e83bbf870fe5f5Sunil Dutt    /**
5089d7489491984e86915b2cf4fac38a882de1c8cdbRoshan Pius     * Disables a network in wpa_supplicant.
5099d7489491984e86915b2cf4fac38a882de1c8cdbRoshan Pius     * @param netId - Network ID of the network to be disabled.
5109d7489491984e86915b2cf4fac38a882de1c8cdbRoshan Pius     * @return true if command succeeded, false otherwise.
5119d7489491984e86915b2cf4fac38a882de1c8cdbRoshan Pius     */
512155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    public boolean disableNetwork(int netId) {
513f22d23092ab37286a5ef9d257d5bb32c421d2669vandwalle        if (DBG) logDbg("disableNetwork nid=" + Integer.toString(netId));
514155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        return doBooleanCommand("DISABLE_NETWORK " + netId);
515155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    }
516155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
5179d7489491984e86915b2cf4fac38a882de1c8cdbRoshan Pius    /**
5189d7489491984e86915b2cf4fac38a882de1c8cdbRoshan Pius     * Select a network in wpa_supplicant (Disables all others).
5199d7489491984e86915b2cf4fac38a882de1c8cdbRoshan Pius     * @param netId - Network ID of the network to be selected.
5209d7489491984e86915b2cf4fac38a882de1c8cdbRoshan Pius     * @return true if command succeeded, false otherwise.
5219d7489491984e86915b2cf4fac38a882de1c8cdbRoshan Pius     */
5220047ccf563baa288777e06c6fe95d3681fcf5ccdVinit Deshpande    public boolean selectNetwork(int netId) {
5230047ccf563baa288777e06c6fe95d3681fcf5ccdVinit Deshpande        if (DBG) logDbg("selectNetwork nid=" + Integer.toString(netId));
5240047ccf563baa288777e06c6fe95d3681fcf5ccdVinit Deshpande        return doBooleanCommand("SELECT_NETWORK " + netId);
5250047ccf563baa288777e06c6fe95d3681fcf5ccdVinit Deshpande    }
5260047ccf563baa288777e06c6fe95d3681fcf5ccdVinit Deshpande
527155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    public boolean reconnect() {
528f22d23092ab37286a5ef9d257d5bb32c421d2669vandwalle        if (DBG) logDbg("RECONNECT ");
529155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        return doBooleanCommand("RECONNECT");
530155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    }
531155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
532155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    public boolean reassociate() {
533f22d23092ab37286a5ef9d257d5bb32c421d2669vandwalle        if (DBG) logDbg("REASSOCIATE ");
534155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        return doBooleanCommand("REASSOCIATE");
535155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    }
536155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
537155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    public boolean disconnect() {
53821bc54cb37a0085b1c909cb4d55ebb12a2facefbvandwalle        if (DBG) logDbg("DISCONNECT ");
539155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        return doBooleanCommand("DISCONNECT");
540155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    }
541155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
542155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    public String status() {
54399d385e3b4d34841d6efcfd7cc9bf1d5ae25de14vandwalle        return status(false);
544155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    }
545155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
54699d385e3b4d34841d6efcfd7cc9bf1d5ae25de14vandwalle    public String status(boolean noEvents) {
54799d385e3b4d34841d6efcfd7cc9bf1d5ae25de14vandwalle        if (noEvents) {
54899d385e3b4d34841d6efcfd7cc9bf1d5ae25de14vandwalle            return doStringCommand("STATUS-NO_EVENTS");
54999d385e3b4d34841d6efcfd7cc9bf1d5ae25de14vandwalle        } else {
55099d385e3b4d34841d6efcfd7cc9bf1d5ae25de14vandwalle            return doStringCommand("STATUS");
55199d385e3b4d34841d6efcfd7cc9bf1d5ae25de14vandwalle        }
55299d385e3b4d34841d6efcfd7cc9bf1d5ae25de14vandwalle    }
55399d385e3b4d34841d6efcfd7cc9bf1d5ae25de14vandwalle
554155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    public String getMacAddress() {
555155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        //Macaddr = XX.XX.XX.XX.XX.XX
556155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        String ret = doStringCommand("DRIVER MACADDR");
557155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        if (!TextUtils.isEmpty(ret)) {
558155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande            String[] tokens = ret.split(" = ");
559155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande            if (tokens.length == 2) return tokens[1];
560155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        }
561155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        return null;
562155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    }
563155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
564a0d34d3626bd9631039a485d2d428b1b7de390e2Pierre Vandwalle
565a0d34d3626bd9631039a485d2d428b1b7de390e2Pierre Vandwalle
566155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    /**
567155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande     * Format of results:
568155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande     * =================
569155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande     * id=1
5702afa54e3c8fa1153302a0d57b0e9b7bee35406ffMitchell Wills     * bssid=68:7f:76:d7:1a:6e
571155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande     * freq=2412
5722afa54e3c8fa1153302a0d57b0e9b7bee35406ffMitchell Wills     * level=-44
5732afa54e3c8fa1153302a0d57b0e9b7bee35406ffMitchell Wills     * tsf=1344626243700342
574155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande     * flags=[WPA2-PSK-CCMP][WPS][ESS]
5752afa54e3c8fa1153302a0d57b0e9b7bee35406ffMitchell Wills     * ssid=zfdy
5762afa54e3c8fa1153302a0d57b0e9b7bee35406ffMitchell Wills     * ====
5772afa54e3c8fa1153302a0d57b0e9b7bee35406ffMitchell Wills     * id=2
5782afa54e3c8fa1153302a0d57b0e9b7bee35406ffMitchell Wills     * bssid=68:5f:74:d7:1a:6f
5792afa54e3c8fa1153302a0d57b0e9b7bee35406ffMitchell Wills     * freq=5180
5802afa54e3c8fa1153302a0d57b0e9b7bee35406ffMitchell Wills     * level=-73
5812afa54e3c8fa1153302a0d57b0e9b7bee35406ffMitchell Wills     * tsf=1344626243700373
5822afa54e3c8fa1153302a0d57b0e9b7bee35406ffMitchell Wills     * flags=[WPA2-PSK-CCMP][WPS][ESS]
5832afa54e3c8fa1153302a0d57b0e9b7bee35406ffMitchell Wills     * ssid=zuby
584155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande     * ====
585155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande     *
586155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande     * RANGE=ALL gets all scan results
587155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande     * RANGE=ID- gets results from ID
588eb5731beb47704699dd8866b1a1dc59f3764c981Samuel Tan     * MASK=<N> BSS command information mask.
58977f2b82a2e80af8da52c22d69a76def6d4209757Jan Nordqvist     *
590eb5731beb47704699dd8866b1a1dc59f3764c981Samuel Tan     * The mask used in this method, 0x29d87, gets the following fields:
59177f2b82a2e80af8da52c22d69a76def6d4209757Jan Nordqvist     *
592eb5731beb47704699dd8866b1a1dc59f3764c981Samuel Tan     *     WPA_BSS_MASK_ID         (Bit 0)
593eb5731beb47704699dd8866b1a1dc59f3764c981Samuel Tan     *     WPA_BSS_MASK_BSSID      (Bit 1)
594eb5731beb47704699dd8866b1a1dc59f3764c981Samuel Tan     *     WPA_BSS_MASK_FREQ       (Bit 2)
595eb5731beb47704699dd8866b1a1dc59f3764c981Samuel Tan     *     WPA_BSS_MASK_LEVEL      (Bit 7)
596eb5731beb47704699dd8866b1a1dc59f3764c981Samuel Tan     *     WPA_BSS_MASK_TSF        (Bit 8)
597eb5731beb47704699dd8866b1a1dc59f3764c981Samuel Tan     *     WPA_BSS_MASK_IE         (Bit 10)
598eb5731beb47704699dd8866b1a1dc59f3764c981Samuel Tan     *     WPA_BSS_MASK_FLAGS      (Bit 11)
599eb5731beb47704699dd8866b1a1dc59f3764c981Samuel Tan     *     WPA_BSS_MASK_SSID       (Bit 12)
600eb5731beb47704699dd8866b1a1dc59f3764c981Samuel Tan     *     WPA_BSS_MASK_INTERNETW  (Bit 15) (adds ANQP info)
601eb5731beb47704699dd8866b1a1dc59f3764c981Samuel Tan     *     WPA_BSS_MASK_DELIM      (Bit 17)
602eb5731beb47704699dd8866b1a1dc59f3764c981Samuel Tan     *
603eb5731beb47704699dd8866b1a1dc59f3764c981Samuel Tan     * See wpa_supplicant/src/common/wpa_ctrl.h for details.
604155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande     */
6052afa54e3c8fa1153302a0d57b0e9b7bee35406ffMitchell Wills    private String getRawScanResults(String range) {
6062afa54e3c8fa1153302a0d57b0e9b7bee35406ffMitchell Wills        return doStringCommandWithoutLogging("BSS RANGE=" + range + " MASK=0x29d87");
6072afa54e3c8fa1153302a0d57b0e9b7bee35406ffMitchell Wills    }
6082afa54e3c8fa1153302a0d57b0e9b7bee35406ffMitchell Wills
6092afa54e3c8fa1153302a0d57b0e9b7bee35406ffMitchell Wills    private static final String BSS_IE_STR = "ie=";
6102afa54e3c8fa1153302a0d57b0e9b7bee35406ffMitchell Wills    private static final String BSS_ID_STR = "id=";
6112afa54e3c8fa1153302a0d57b0e9b7bee35406ffMitchell Wills    private static final String BSS_BSSID_STR = "bssid=";
6122afa54e3c8fa1153302a0d57b0e9b7bee35406ffMitchell Wills    private static final String BSS_FREQ_STR = "freq=";
6132afa54e3c8fa1153302a0d57b0e9b7bee35406ffMitchell Wills    private static final String BSS_LEVEL_STR = "level=";
6142afa54e3c8fa1153302a0d57b0e9b7bee35406ffMitchell Wills    private static final String BSS_TSF_STR = "tsf=";
6152afa54e3c8fa1153302a0d57b0e9b7bee35406ffMitchell Wills    private static final String BSS_FLAGS_STR = "flags=";
6162afa54e3c8fa1153302a0d57b0e9b7bee35406ffMitchell Wills    private static final String BSS_SSID_STR = "ssid=";
6172afa54e3c8fa1153302a0d57b0e9b7bee35406ffMitchell Wills    private static final String BSS_DELIMITER_STR = "====";
6182afa54e3c8fa1153302a0d57b0e9b7bee35406ffMitchell Wills    private static final String BSS_END_STR = "####";
6192afa54e3c8fa1153302a0d57b0e9b7bee35406ffMitchell Wills
6202afa54e3c8fa1153302a0d57b0e9b7bee35406ffMitchell Wills    public ArrayList<ScanDetail> getScanResults() {
6212afa54e3c8fa1153302a0d57b0e9b7bee35406ffMitchell Wills        int next_sid = 0;
6222afa54e3c8fa1153302a0d57b0e9b7bee35406ffMitchell Wills        ArrayList<ScanDetail> results = new ArrayList<>();
6232afa54e3c8fa1153302a0d57b0e9b7bee35406ffMitchell Wills        while(next_sid >= 0) {
6242afa54e3c8fa1153302a0d57b0e9b7bee35406ffMitchell Wills            String rawResult = getRawScanResults(next_sid+"-");
6252afa54e3c8fa1153302a0d57b0e9b7bee35406ffMitchell Wills            next_sid = -1;
6262afa54e3c8fa1153302a0d57b0e9b7bee35406ffMitchell Wills
6272afa54e3c8fa1153302a0d57b0e9b7bee35406ffMitchell Wills            if (TextUtils.isEmpty(rawResult))
6282afa54e3c8fa1153302a0d57b0e9b7bee35406ffMitchell Wills                break;
6292afa54e3c8fa1153302a0d57b0e9b7bee35406ffMitchell Wills
6302afa54e3c8fa1153302a0d57b0e9b7bee35406ffMitchell Wills            String[] lines = rawResult.split("\n");
6312afa54e3c8fa1153302a0d57b0e9b7bee35406ffMitchell Wills
6322afa54e3c8fa1153302a0d57b0e9b7bee35406ffMitchell Wills
6332afa54e3c8fa1153302a0d57b0e9b7bee35406ffMitchell Wills            // note that all these splits and substrings keep references to the original
6342afa54e3c8fa1153302a0d57b0e9b7bee35406ffMitchell Wills            // huge string buffer while the amount we really want is generally pretty small
6352afa54e3c8fa1153302a0d57b0e9b7bee35406ffMitchell Wills            // so make copies instead (one example b/11087956 wasted 400k of heap here).
6362afa54e3c8fa1153302a0d57b0e9b7bee35406ffMitchell Wills            final int bssidStrLen = BSS_BSSID_STR.length();
6372afa54e3c8fa1153302a0d57b0e9b7bee35406ffMitchell Wills            final int flagLen = BSS_FLAGS_STR.length();
6382afa54e3c8fa1153302a0d57b0e9b7bee35406ffMitchell Wills
6392afa54e3c8fa1153302a0d57b0e9b7bee35406ffMitchell Wills            String bssid = "";
6402afa54e3c8fa1153302a0d57b0e9b7bee35406ffMitchell Wills            int level = 0;
6412afa54e3c8fa1153302a0d57b0e9b7bee35406ffMitchell Wills            int freq = 0;
6422afa54e3c8fa1153302a0d57b0e9b7bee35406ffMitchell Wills            long tsf = 0;
6432afa54e3c8fa1153302a0d57b0e9b7bee35406ffMitchell Wills            String flags = "";
6442afa54e3c8fa1153302a0d57b0e9b7bee35406ffMitchell Wills            WifiSsid wifiSsid = null;
6455d31cedf4024e0f038b4dfc2081016c8631ee8feMitchell Wills            String infoElementsStr = null;
6462afa54e3c8fa1153302a0d57b0e9b7bee35406ffMitchell Wills            List<String> anqpLines = null;
6472afa54e3c8fa1153302a0d57b0e9b7bee35406ffMitchell Wills
6482afa54e3c8fa1153302a0d57b0e9b7bee35406ffMitchell Wills            for (String line : lines) {
6492afa54e3c8fa1153302a0d57b0e9b7bee35406ffMitchell Wills                if (line.startsWith(BSS_ID_STR)) { // Will find the last id line
6502afa54e3c8fa1153302a0d57b0e9b7bee35406ffMitchell Wills                    try {
6512afa54e3c8fa1153302a0d57b0e9b7bee35406ffMitchell Wills                        next_sid = Integer.parseInt(line.substring(BSS_ID_STR.length())) + 1;
6522afa54e3c8fa1153302a0d57b0e9b7bee35406ffMitchell Wills                    } catch (NumberFormatException e) {
6532afa54e3c8fa1153302a0d57b0e9b7bee35406ffMitchell Wills                        // Nothing to do
6542afa54e3c8fa1153302a0d57b0e9b7bee35406ffMitchell Wills                    }
6552afa54e3c8fa1153302a0d57b0e9b7bee35406ffMitchell Wills                } else if (line.startsWith(BSS_BSSID_STR)) {
6562afa54e3c8fa1153302a0d57b0e9b7bee35406ffMitchell Wills                    bssid = new String(line.getBytes(), bssidStrLen, line.length() - bssidStrLen);
6572afa54e3c8fa1153302a0d57b0e9b7bee35406ffMitchell Wills                } else if (line.startsWith(BSS_FREQ_STR)) {
6582afa54e3c8fa1153302a0d57b0e9b7bee35406ffMitchell Wills                    try {
6592afa54e3c8fa1153302a0d57b0e9b7bee35406ffMitchell Wills                        freq = Integer.parseInt(line.substring(BSS_FREQ_STR.length()));
6602afa54e3c8fa1153302a0d57b0e9b7bee35406ffMitchell Wills                    } catch (NumberFormatException e) {
6612afa54e3c8fa1153302a0d57b0e9b7bee35406ffMitchell Wills                        freq = 0;
6622afa54e3c8fa1153302a0d57b0e9b7bee35406ffMitchell Wills                    }
6632afa54e3c8fa1153302a0d57b0e9b7bee35406ffMitchell Wills                } else if (line.startsWith(BSS_LEVEL_STR)) {
6642afa54e3c8fa1153302a0d57b0e9b7bee35406ffMitchell Wills                    try {
6652afa54e3c8fa1153302a0d57b0e9b7bee35406ffMitchell Wills                        level = Integer.parseInt(line.substring(BSS_LEVEL_STR.length()));
6662afa54e3c8fa1153302a0d57b0e9b7bee35406ffMitchell Wills                        /* some implementations avoid negative values by adding 256
6672afa54e3c8fa1153302a0d57b0e9b7bee35406ffMitchell Wills                         * so we need to adjust for that here.
6682afa54e3c8fa1153302a0d57b0e9b7bee35406ffMitchell Wills                         */
6692afa54e3c8fa1153302a0d57b0e9b7bee35406ffMitchell Wills                        if (level > 0) level -= 256;
6702afa54e3c8fa1153302a0d57b0e9b7bee35406ffMitchell Wills                    } catch (NumberFormatException e) {
6712afa54e3c8fa1153302a0d57b0e9b7bee35406ffMitchell Wills                        level = 0;
6722afa54e3c8fa1153302a0d57b0e9b7bee35406ffMitchell Wills                    }
6732afa54e3c8fa1153302a0d57b0e9b7bee35406ffMitchell Wills                } else if (line.startsWith(BSS_TSF_STR)) {
6742afa54e3c8fa1153302a0d57b0e9b7bee35406ffMitchell Wills                    try {
6752afa54e3c8fa1153302a0d57b0e9b7bee35406ffMitchell Wills                        tsf = Long.parseLong(line.substring(BSS_TSF_STR.length()));
6762afa54e3c8fa1153302a0d57b0e9b7bee35406ffMitchell Wills                    } catch (NumberFormatException e) {
6772afa54e3c8fa1153302a0d57b0e9b7bee35406ffMitchell Wills                        tsf = 0;
6782afa54e3c8fa1153302a0d57b0e9b7bee35406ffMitchell Wills                    }
6792afa54e3c8fa1153302a0d57b0e9b7bee35406ffMitchell Wills                } else if (line.startsWith(BSS_FLAGS_STR)) {
6802afa54e3c8fa1153302a0d57b0e9b7bee35406ffMitchell Wills                    flags = new String(line.getBytes(), flagLen, line.length() - flagLen);
6812afa54e3c8fa1153302a0d57b0e9b7bee35406ffMitchell Wills                } else if (line.startsWith(BSS_SSID_STR)) {
6822afa54e3c8fa1153302a0d57b0e9b7bee35406ffMitchell Wills                    wifiSsid = WifiSsid.createFromAsciiEncoded(
6832afa54e3c8fa1153302a0d57b0e9b7bee35406ffMitchell Wills                            line.substring(BSS_SSID_STR.length()));
6842afa54e3c8fa1153302a0d57b0e9b7bee35406ffMitchell Wills                } else if (line.startsWith(BSS_IE_STR)) {
6855d31cedf4024e0f038b4dfc2081016c8631ee8feMitchell Wills                    infoElementsStr = line;
6862afa54e3c8fa1153302a0d57b0e9b7bee35406ffMitchell Wills                } else if (SupplicantBridge.isAnqpAttribute(line)) {
6872afa54e3c8fa1153302a0d57b0e9b7bee35406ffMitchell Wills                    if (anqpLines == null) {
6882afa54e3c8fa1153302a0d57b0e9b7bee35406ffMitchell Wills                        anqpLines = new ArrayList<>();
6892afa54e3c8fa1153302a0d57b0e9b7bee35406ffMitchell Wills                    }
6902afa54e3c8fa1153302a0d57b0e9b7bee35406ffMitchell Wills                    anqpLines.add(line);
6912afa54e3c8fa1153302a0d57b0e9b7bee35406ffMitchell Wills                } else if (line.startsWith(BSS_DELIMITER_STR) || line.startsWith(BSS_END_STR)) {
6922afa54e3c8fa1153302a0d57b0e9b7bee35406ffMitchell Wills                    if (bssid != null) {
6932afa54e3c8fa1153302a0d57b0e9b7bee35406ffMitchell Wills                        try {
6945d31cedf4024e0f038b4dfc2081016c8631ee8feMitchell Wills                            if (infoElementsStr == null) {
6955d31cedf4024e0f038b4dfc2081016c8631ee8feMitchell Wills                                throw new IllegalArgumentException("Null information element data");
6965d31cedf4024e0f038b4dfc2081016c8631ee8feMitchell Wills                            }
6975d31cedf4024e0f038b4dfc2081016c8631ee8feMitchell Wills                            int seperator = infoElementsStr.indexOf('=');
6985d31cedf4024e0f038b4dfc2081016c8631ee8feMitchell Wills                            if (seperator < 0) {
6995d31cedf4024e0f038b4dfc2081016c8631ee8feMitchell Wills                                throw new IllegalArgumentException("No element separator");
7005d31cedf4024e0f038b4dfc2081016c8631ee8feMitchell Wills                            }
7015d31cedf4024e0f038b4dfc2081016c8631ee8feMitchell Wills
7025d31cedf4024e0f038b4dfc2081016c8631ee8feMitchell Wills                            ScanResult.InformationElement[] infoElements =
7035d31cedf4024e0f038b4dfc2081016c8631ee8feMitchell Wills                                        InformationElementUtil.parseInformationElements(
7045d31cedf4024e0f038b4dfc2081016c8631ee8feMitchell Wills                                        Utils.hexToBytes(infoElementsStr.substring(seperator + 1)));
7055d31cedf4024e0f038b4dfc2081016c8631ee8feMitchell Wills
7065d31cedf4024e0f038b4dfc2081016c8631ee8feMitchell Wills                            NetworkDetail networkDetail = new NetworkDetail(bssid,
7075d31cedf4024e0f038b4dfc2081016c8631ee8feMitchell Wills                                    infoElements, anqpLines, freq);
708947e55415eab3989f2f5cede0c03745cf9268309Glen Kuhne                            if (DBG) {
709947e55415eab3989f2f5cede0c03745cf9268309Glen Kuhne                                Log.v(TAG + ":DTIM", "SSID" + networkDetail.getSSID()
710947e55415eab3989f2f5cede0c03745cf9268309Glen Kuhne                                        + ", DTIM=" + networkDetail.getDtimInterval() + ", "
711947e55415eab3989f2f5cede0c03745cf9268309Glen Kuhne                                        + " IEstr:" + infoElementsStr);
712947e55415eab3989f2f5cede0c03745cf9268309Glen Kuhne                            }
7132afa54e3c8fa1153302a0d57b0e9b7bee35406ffMitchell Wills                            String xssid = (wifiSsid != null) ? wifiSsid.toString() : WifiSsid.NONE;
7142afa54e3c8fa1153302a0d57b0e9b7bee35406ffMitchell Wills                            if (!xssid.equals(networkDetail.getTrimmedSSID())) {
7152afa54e3c8fa1153302a0d57b0e9b7bee35406ffMitchell Wills                                Log.d(TAG, String.format(
7162afa54e3c8fa1153302a0d57b0e9b7bee35406ffMitchell Wills                                        "Inconsistent SSID on BSSID '%s': '%s' vs '%s': %s",
7175d31cedf4024e0f038b4dfc2081016c8631ee8feMitchell Wills                                        bssid, xssid, networkDetail.getSSID(), infoElementsStr));
7182afa54e3c8fa1153302a0d57b0e9b7bee35406ffMitchell Wills                            }
7192afa54e3c8fa1153302a0d57b0e9b7bee35406ffMitchell Wills
7202afa54e3c8fa1153302a0d57b0e9b7bee35406ffMitchell Wills                            if (networkDetail.hasInterworking()) {
721af2b79e4cfc1da793b3d8fb4a96c144deefc7d58Joe Onorato                                if (DBG) Log.d(TAG, "HSNwk: '" + networkDetail);
7222afa54e3c8fa1153302a0d57b0e9b7bee35406ffMitchell Wills                            }
7239ec71f6499e0e3d6f52310a41ff4a59d2fa4f8b2Mitchell Wills                            ScanDetail scan = new ScanDetail(networkDetail, wifiSsid, bssid, flags,
72463539f1283899fbbf83ab90757961b4be51d5034Mitchell Wills                                    level, freq, tsf, infoElements, anqpLines);
7259ec71f6499e0e3d6f52310a41ff4a59d2fa4f8b2Mitchell Wills                            results.add(scan);
7262afa54e3c8fa1153302a0d57b0e9b7bee35406ffMitchell Wills                        } catch (IllegalArgumentException iae) {
7272afa54e3c8fa1153302a0d57b0e9b7bee35406ffMitchell Wills                            Log.d(TAG, "Failed to parse information elements: " + iae);
7282afa54e3c8fa1153302a0d57b0e9b7bee35406ffMitchell Wills                        }
7292afa54e3c8fa1153302a0d57b0e9b7bee35406ffMitchell Wills                    }
7302afa54e3c8fa1153302a0d57b0e9b7bee35406ffMitchell Wills                    bssid = null;
7312afa54e3c8fa1153302a0d57b0e9b7bee35406ffMitchell Wills                    level = 0;
7322afa54e3c8fa1153302a0d57b0e9b7bee35406ffMitchell Wills                    freq = 0;
7332afa54e3c8fa1153302a0d57b0e9b7bee35406ffMitchell Wills                    tsf = 0;
7342afa54e3c8fa1153302a0d57b0e9b7bee35406ffMitchell Wills                    flags = "";
7352afa54e3c8fa1153302a0d57b0e9b7bee35406ffMitchell Wills                    wifiSsid = null;
7365d31cedf4024e0f038b4dfc2081016c8631ee8feMitchell Wills                    infoElementsStr = null;
7372afa54e3c8fa1153302a0d57b0e9b7bee35406ffMitchell Wills                    anqpLines = null;
7382afa54e3c8fa1153302a0d57b0e9b7bee35406ffMitchell Wills                }
7392afa54e3c8fa1153302a0d57b0e9b7bee35406ffMitchell Wills            }
7402afa54e3c8fa1153302a0d57b0e9b7bee35406ffMitchell Wills        }
7412afa54e3c8fa1153302a0d57b0e9b7bee35406ffMitchell Wills        return results;
74277f2b82a2e80af8da52c22d69a76def6d4209757Jan Nordqvist    }
74377f2b82a2e80af8da52c22d69a76def6d4209757Jan Nordqvist
744155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    /**
745446db2d5457456743e4476029e14d7c3bb9f5bccYuhao Zheng     * Format of result:
746446db2d5457456743e4476029e14d7c3bb9f5bccYuhao Zheng     * id=1016
747446db2d5457456743e4476029e14d7c3bb9f5bccYuhao Zheng     * bssid=00:03:7f:40:84:10
748446db2d5457456743e4476029e14d7c3bb9f5bccYuhao Zheng     * freq=2462
749446db2d5457456743e4476029e14d7c3bb9f5bccYuhao Zheng     * beacon_int=200
750446db2d5457456743e4476029e14d7c3bb9f5bccYuhao Zheng     * capabilities=0x0431
751446db2d5457456743e4476029e14d7c3bb9f5bccYuhao Zheng     * qual=0
752446db2d5457456743e4476029e14d7c3bb9f5bccYuhao Zheng     * noise=0
753446db2d5457456743e4476029e14d7c3bb9f5bccYuhao Zheng     * level=-46
754446db2d5457456743e4476029e14d7c3bb9f5bccYuhao Zheng     * tsf=0000002669008476
755446db2d5457456743e4476029e14d7c3bb9f5bccYuhao Zheng     * age=5
756446db2d5457456743e4476029e14d7c3bb9f5bccYuhao Zheng     * ie=00105143412d485332302d52322d54455354010882848b960c12182403010b0706555...
757446db2d5457456743e4476029e14d7c3bb9f5bccYuhao Zheng     * flags=[WPA2-EAP-CCMP][ESS][P2P][HS20]
758446db2d5457456743e4476029e14d7c3bb9f5bccYuhao Zheng     * ssid=QCA-HS20-R2-TEST
759446db2d5457456743e4476029e14d7c3bb9f5bccYuhao Zheng     * p2p_device_name=
76056d0178183460eed9afbd85e5c0d215e27d5f5bcvandwalle     * p2p_config_methods=0x0SET_NE
761446db2d5457456743e4476029e14d7c3bb9f5bccYuhao Zheng     * anqp_venue_name=02083d656e6757692d466920416c6c69616e63650a3239383920436f...
762446db2d5457456743e4476029e14d7c3bb9f5bccYuhao Zheng     * anqp_network_auth_type=010000
763446db2d5457456743e4476029e14d7c3bb9f5bccYuhao Zheng     * anqp_roaming_consortium=03506f9a05001bc504bd
764446db2d5457456743e4476029e14d7c3bb9f5bccYuhao Zheng     * anqp_ip_addr_type_availability=0c
765446db2d5457456743e4476029e14d7c3bb9f5bccYuhao Zheng     * anqp_nai_realm=0200300000246d61696c2e6578616d706c652e636f6d3b636973636f2...
766446db2d5457456743e4476029e14d7c3bb9f5bccYuhao Zheng     * anqp_3gpp=000600040132f465
767446db2d5457456743e4476029e14d7c3bb9f5bccYuhao Zheng     * anqp_domain_name=0b65786d61706c652e636f6d
768446db2d5457456743e4476029e14d7c3bb9f5bccYuhao Zheng     * hs20_operator_friendly_name=11656e6757692d466920416c6c69616e63650e636869...
769446db2d5457456743e4476029e14d7c3bb9f5bccYuhao Zheng     * hs20_wan_metrics=01c40900008001000000000a00
770446db2d5457456743e4476029e14d7c3bb9f5bccYuhao Zheng     * hs20_connection_capability=0100000006140001061600000650000106bb010106bb0...
771446db2d5457456743e4476029e14d7c3bb9f5bccYuhao Zheng     * hs20_osu_providers_list=0b5143412d4f53552d425353010901310015656e6757692d...
772446db2d5457456743e4476029e14d7c3bb9f5bccYuhao Zheng     */
773446db2d5457456743e4476029e14d7c3bb9f5bccYuhao Zheng    public String scanResult(String bssid) {
774446db2d5457456743e4476029e14d7c3bb9f5bccYuhao Zheng        return doStringCommand("BSS " + bssid);
775446db2d5457456743e4476029e14d7c3bb9f5bccYuhao Zheng    }
776446db2d5457456743e4476029e14d7c3bb9f5bccYuhao Zheng
777155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    public boolean startDriver() {
778155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        return doBooleanCommand("DRIVER START");
779155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    }
780155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
781155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    public boolean stopDriver() {
782155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        return doBooleanCommand("DRIVER STOP");
783155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    }
784155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
785155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
786155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    /**
787155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande     * Start filtering out Multicast V4 packets
788155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande     * @return {@code true} if the operation succeeded, {@code false} otherwise
789155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande     *
790155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande     * Multicast filtering rules work as follows:
791155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande     *
792155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande     * The driver can filter multicast (v4 and/or v6) and broadcast packets when in
793155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande     * a power optimized mode (typically when screen goes off).
794155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande     *
795155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande     * In order to prevent the driver from filtering the multicast/broadcast packets, we have to
796155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande     * add a DRIVER RXFILTER-ADD rule followed by DRIVER RXFILTER-START to make the rule effective
797155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande     *
798155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande     * DRIVER RXFILTER-ADD Num
799155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande     *   where Num = 0 - Unicast, 1 - Broadcast, 2 - Mutil4 or 3 - Multi6
800155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande     *
801155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande     * and DRIVER RXFILTER-START
802155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande     * In order to stop the usage of these rules, we do
803155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande     *
804155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande     * DRIVER RXFILTER-STOP
805155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande     * DRIVER RXFILTER-REMOVE Num
806155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande     *   where Num is as described for RXFILTER-ADD
807155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande     *
808155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande     * The  SETSUSPENDOPT driver command overrides the filtering rules
809155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande     */
810155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    public boolean startFilteringMulticastV4Packets() {
811155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        return doBooleanCommand("DRIVER RXFILTER-STOP")
812155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande            && doBooleanCommand("DRIVER RXFILTER-REMOVE 2")
813155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande            && doBooleanCommand("DRIVER RXFILTER-START");
814155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    }
815155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
816155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    /**
817155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande     * Stop filtering out Multicast V4 packets.
818155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande     * @return {@code true} if the operation succeeded, {@code false} otherwise
819155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande     */
820155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    public boolean stopFilteringMulticastV4Packets() {
821155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        return doBooleanCommand("DRIVER RXFILTER-STOP")
822155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande            && doBooleanCommand("DRIVER RXFILTER-ADD 2")
823155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande            && doBooleanCommand("DRIVER RXFILTER-START");
824155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    }
825155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
826155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    /**
827155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande     * Start filtering out Multicast V6 packets
828155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande     * @return {@code true} if the operation succeeded, {@code false} otherwise
829155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande     */
830155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    public boolean startFilteringMulticastV6Packets() {
831155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        return doBooleanCommand("DRIVER RXFILTER-STOP")
832155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande            && doBooleanCommand("DRIVER RXFILTER-REMOVE 3")
833155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande            && doBooleanCommand("DRIVER RXFILTER-START");
834155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    }
835155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
836155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    /**
837155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande     * Stop filtering out Multicast V6 packets.
838155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande     * @return {@code true} if the operation succeeded, {@code false} otherwise
839155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande     */
840155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    public boolean stopFilteringMulticastV6Packets() {
841155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        return doBooleanCommand("DRIVER RXFILTER-STOP")
842155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande            && doBooleanCommand("DRIVER RXFILTER-ADD 3")
843155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande            && doBooleanCommand("DRIVER RXFILTER-START");
844155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    }
845155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
84603cdd624f5da6d62cb731304aa7505921566f69dDmitry Shmidt    /**
84703cdd624f5da6d62cb731304aa7505921566f69dDmitry Shmidt     * Set the operational frequency band
84803cdd624f5da6d62cb731304aa7505921566f69dDmitry Shmidt     * @param band One of
84903cdd624f5da6d62cb731304aa7505921566f69dDmitry Shmidt     *     {@link WifiManager#WIFI_FREQUENCY_BAND_AUTO},
85003cdd624f5da6d62cb731304aa7505921566f69dDmitry Shmidt     *     {@link WifiManager#WIFI_FREQUENCY_BAND_5GHZ},
85103cdd624f5da6d62cb731304aa7505921566f69dDmitry Shmidt     *     {@link WifiManager#WIFI_FREQUENCY_BAND_2GHZ},
85203cdd624f5da6d62cb731304aa7505921566f69dDmitry Shmidt     * @return {@code true} if the operation succeeded, {@code false} otherwise
85303cdd624f5da6d62cb731304aa7505921566f69dDmitry Shmidt     */
854155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    public boolean setBand(int band) {
85503cdd624f5da6d62cb731304aa7505921566f69dDmitry Shmidt        String bandstr;
85603cdd624f5da6d62cb731304aa7505921566f69dDmitry Shmidt
85703cdd624f5da6d62cb731304aa7505921566f69dDmitry Shmidt        if (band == WifiManager.WIFI_FREQUENCY_BAND_5GHZ)
85803cdd624f5da6d62cb731304aa7505921566f69dDmitry Shmidt            bandstr = "5G";
85903cdd624f5da6d62cb731304aa7505921566f69dDmitry Shmidt        else if (band == WifiManager.WIFI_FREQUENCY_BAND_2GHZ)
86003cdd624f5da6d62cb731304aa7505921566f69dDmitry Shmidt            bandstr = "2G";
86103cdd624f5da6d62cb731304aa7505921566f69dDmitry Shmidt        else
86203cdd624f5da6d62cb731304aa7505921566f69dDmitry Shmidt            bandstr = "AUTO";
86303cdd624f5da6d62cb731304aa7505921566f69dDmitry Shmidt        return doBooleanCommand("SET SETBAND " + bandstr);
864155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    }
865155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
86618786eca942042388748b0d98979f21c9dff4a89Mitchell Wills    public static final int BLUETOOTH_COEXISTENCE_MODE_ENABLED     = 0;
86718786eca942042388748b0d98979f21c9dff4a89Mitchell Wills    public static final int BLUETOOTH_COEXISTENCE_MODE_DISABLED    = 1;
86818786eca942042388748b0d98979f21c9dff4a89Mitchell Wills    public static final int BLUETOOTH_COEXISTENCE_MODE_SENSE       = 2;
8697ef73dd1b6e43c72b3841723504cd86dc402a134Vinit Deshapnde    /**
8707ef73dd1b6e43c72b3841723504cd86dc402a134Vinit Deshapnde      * Sets the bluetooth coexistence mode.
8717ef73dd1b6e43c72b3841723504cd86dc402a134Vinit Deshapnde      *
8727ef73dd1b6e43c72b3841723504cd86dc402a134Vinit Deshapnde      * @param mode One of {@link #BLUETOOTH_COEXISTENCE_MODE_DISABLED},
8737ef73dd1b6e43c72b3841723504cd86dc402a134Vinit Deshapnde      *            {@link #BLUETOOTH_COEXISTENCE_MODE_ENABLED}, or
8747ef73dd1b6e43c72b3841723504cd86dc402a134Vinit Deshapnde      *            {@link #BLUETOOTH_COEXISTENCE_MODE_SENSE}.
8757ef73dd1b6e43c72b3841723504cd86dc402a134Vinit Deshapnde      * @return Whether the mode was successfully set.
8767ef73dd1b6e43c72b3841723504cd86dc402a134Vinit Deshapnde      */
877155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    public boolean setBluetoothCoexistenceMode(int mode) {
878155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        return doBooleanCommand("DRIVER BTCOEXMODE " + mode);
879155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    }
880155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
881155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    /**
882155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande     * Enable or disable Bluetooth coexistence scan mode. When this mode is on,
883155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande     * some of the low-level scan parameters used by the driver are changed to
884155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande     * reduce interference with A2DP streaming.
885155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande     *
886155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande     * @param isSet whether to enable or disable this mode
887155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande     * @return {@code true} if the command succeeded, {@code false} otherwise.
888155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande     */
889155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    public boolean setBluetoothCoexistenceScanMode(boolean setCoexScanMode) {
890155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        if (setCoexScanMode) {
891155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande            return doBooleanCommand("DRIVER BTCOEXSCAN-START");
892155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        } else {
893155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande            return doBooleanCommand("DRIVER BTCOEXSCAN-STOP");
894155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        }
895155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    }
896155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
8970a696d168d7ad98ab5084d2a16e3d02c545a85aaVinit Deshapnde    public void enableSaveConfig() {
8980a696d168d7ad98ab5084d2a16e3d02c545a85aaVinit Deshapnde        doBooleanCommand("SET update_config 1");
8990a696d168d7ad98ab5084d2a16e3d02c545a85aaVinit Deshapnde    }
9000a696d168d7ad98ab5084d2a16e3d02c545a85aaVinit Deshapnde
901155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    public boolean saveConfig() {
902155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        return doBooleanCommand("SAVE_CONFIG");
903155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    }
904155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
905155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    public boolean addToBlacklist(String bssid) {
906155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        if (TextUtils.isEmpty(bssid)) return false;
907155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        return doBooleanCommand("BLACKLIST " + bssid);
908155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    }
909155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
910155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    public boolean clearBlacklist() {
911155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        return doBooleanCommand("BLACKLIST clear");
912155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    }
913155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
914155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    public boolean setSuspendOptimizations(boolean enabled) {
915155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        if (enabled) {
916155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande            return doBooleanCommand("DRIVER SETSUSPENDMODE 1");
917155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        } else {
918155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande            return doBooleanCommand("DRIVER SETSUSPENDMODE 0");
919155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        }
920155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    }
921155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
922155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    public boolean setCountryCode(String countryCode) {
9230465ff513cdccf7e883a505e14c5b78758e7e458Vinit Deshpande        if (countryCode != null)
9240465ff513cdccf7e883a505e14c5b78758e7e458Vinit Deshpande            return doBooleanCommand("DRIVER COUNTRY " + countryCode.toUpperCase(Locale.ROOT));
9250465ff513cdccf7e883a505e14c5b78758e7e458Vinit Deshpande        else
9260465ff513cdccf7e883a505e14c5b78758e7e458Vinit Deshpande            return doBooleanCommand("DRIVER COUNTRY");
927155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    }
928155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
9299153bd67d51b305ffdd61355e0748e3c332c2cafRoshan Pius    /**
9309153bd67d51b305ffdd61355e0748e3c332c2cafRoshan Pius     * Start/Stop PNO scan.
9319153bd67d51b305ffdd61355e0748e3c332c2cafRoshan Pius     * @param enable boolean indicating whether PNO is being enabled or disabled.
9329153bd67d51b305ffdd61355e0748e3c332c2cafRoshan Pius     */
9339153bd67d51b305ffdd61355e0748e3c332c2cafRoshan Pius    public boolean setPnoScan(boolean enable) {
9349153bd67d51b305ffdd61355e0748e3c332c2cafRoshan Pius        String cmd = enable ? "SET pno 1" : "SET pno 0";
9359153bd67d51b305ffdd61355e0748e3c332c2cafRoshan Pius        return doBooleanCommand(cmd);
9369153bd67d51b305ffdd61355e0748e3c332c2cafRoshan Pius    }
937ac9ad3283508db15b65b1cbb89b841278973276bRoshan Pius
938f22d23092ab37286a5ef9d257d5bb32c421d2669vandwalle    public void enableAutoConnect(boolean enable) {
939f22d23092ab37286a5ef9d257d5bb32c421d2669vandwalle        if (enable) {
940f22d23092ab37286a5ef9d257d5bb32c421d2669vandwalle            doBooleanCommand("STA_AUTOCONNECT 1");
941f22d23092ab37286a5ef9d257d5bb32c421d2669vandwalle        } else {
942f22d23092ab37286a5ef9d257d5bb32c421d2669vandwalle            doBooleanCommand("STA_AUTOCONNECT 0");
943f22d23092ab37286a5ef9d257d5bb32c421d2669vandwalle        }
944f22d23092ab37286a5ef9d257d5bb32c421d2669vandwalle    }
945f22d23092ab37286a5ef9d257d5bb32c421d2669vandwalle
946155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    public void setScanInterval(int scanInterval) {
947155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        doBooleanCommand("SCAN_INTERVAL " + scanInterval);
948155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    }
949155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
95056845c30ef2bc7ccfd92764e4c7352aee03ec3a8Vinit Deshpande    public void setHs20(boolean hs20) {
95156845c30ef2bc7ccfd92764e4c7352aee03ec3a8Vinit Deshpande        if (hs20) {
95256845c30ef2bc7ccfd92764e4c7352aee03ec3a8Vinit Deshpande            doBooleanCommand("SET HS20 1");
95356845c30ef2bc7ccfd92764e4c7352aee03ec3a8Vinit Deshpande        } else {
95456845c30ef2bc7ccfd92764e4c7352aee03ec3a8Vinit Deshpande            doBooleanCommand("SET HS20 0");
95556845c30ef2bc7ccfd92764e4c7352aee03ec3a8Vinit Deshpande        }
95656845c30ef2bc7ccfd92764e4c7352aee03ec3a8Vinit Deshpande    }
95756845c30ef2bc7ccfd92764e4c7352aee03ec3a8Vinit Deshpande
958155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    public void startTdls(String macAddr, boolean enable) {
959155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        if (enable) {
960956f54b391677d78379729dd14518edddf3c7660Etan Cohen            synchronized (sLock) {
96118786eca942042388748b0d98979f21c9dff4a89Mitchell Wills                doBooleanCommand("TDLS_DISCOVER " + macAddr);
96218786eca942042388748b0d98979f21c9dff4a89Mitchell Wills                doBooleanCommand("TDLS_SETUP " + macAddr);
96318786eca942042388748b0d98979f21c9dff4a89Mitchell Wills            }
964155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        } else {
965155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande            doBooleanCommand("TDLS_TEARDOWN " + macAddr);
966155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        }
967155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    }
968155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
969155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    /** Example output:
970155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande     * RSSI=-65
971155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande     * LINKSPEED=48
972155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande     * NOISE=9999
973155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande     * FREQUENCY=0
974155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande     */
975155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    public String signalPoll() {
976155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        return doStringCommandWithoutLogging("SIGNAL_POLL");
977155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    }
978155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
979155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    /** Example outout:
980155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande     * TXGOOD=396
981155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande     * TXBAD=1
982155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande     */
983155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    public String pktcntPoll() {
984155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        return doStringCommand("PKTCNT_POLL");
985155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    }
986155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
987155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    public void bssFlush() {
988155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        doBooleanCommand("BSS_FLUSH 0");
989155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    }
990155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
991155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    public boolean startWpsPbc(String bssid) {
992155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        if (TextUtils.isEmpty(bssid)) {
993155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande            return doBooleanCommand("WPS_PBC");
994155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        } else {
995155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande            return doBooleanCommand("WPS_PBC " + bssid);
996155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        }
997155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    }
998155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
999155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    public boolean startWpsPbc(String iface, String bssid) {
1000956f54b391677d78379729dd14518edddf3c7660Etan Cohen        synchronized (sLock) {
1001155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande            if (TextUtils.isEmpty(bssid)) {
1002155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande                return doBooleanCommandNative("IFNAME=" + iface + " WPS_PBC");
1003155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande            } else {
1004155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande                return doBooleanCommandNative("IFNAME=" + iface + " WPS_PBC " + bssid);
1005155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande            }
1006155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        }
1007155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    }
1008155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
1009155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    public boolean startWpsPinKeypad(String pin) {
1010155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        if (TextUtils.isEmpty(pin)) return false;
1011155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        return doBooleanCommand("WPS_PIN any " + pin);
1012155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    }
1013155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
1014155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    public boolean startWpsPinKeypad(String iface, String pin) {
1015155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        if (TextUtils.isEmpty(pin)) return false;
1016956f54b391677d78379729dd14518edddf3c7660Etan Cohen        synchronized (sLock) {
1017155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande            return doBooleanCommandNative("IFNAME=" + iface + " WPS_PIN any " + pin);
1018155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        }
1019155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    }
1020155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
1021155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
1022155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    public String startWpsPinDisplay(String bssid) {
1023155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        if (TextUtils.isEmpty(bssid)) {
1024155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande            return doStringCommand("WPS_PIN any");
1025155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        } else {
1026155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande            return doStringCommand("WPS_PIN " + bssid);
1027155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        }
1028155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    }
1029155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
1030155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    public String startWpsPinDisplay(String iface, String bssid) {
1031956f54b391677d78379729dd14518edddf3c7660Etan Cohen        synchronized (sLock) {
1032155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande            if (TextUtils.isEmpty(bssid)) {
1033155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande                return doStringCommandNative("IFNAME=" + iface + " WPS_PIN any");
1034155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande            } else {
1035155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande                return doStringCommandNative("IFNAME=" + iface + " WPS_PIN " + bssid);
1036155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande            }
1037155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        }
1038155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    }
1039155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
104033b575ca6bee66183929f9474b5a161432918604Vinit Deshpande    public boolean setExternalSim(boolean external) {
104118786eca942042388748b0d98979f21c9dff4a89Mitchell Wills        String value = external ? "1" : "0";
104218786eca942042388748b0d98979f21c9dff4a89Mitchell Wills        Log.d(TAG, "Setting external_sim to " + value);
104318786eca942042388748b0d98979f21c9dff4a89Mitchell Wills        return doBooleanCommand("SET external_sim " + value);
104433b575ca6bee66183929f9474b5a161432918604Vinit Deshpande    }
104533b575ca6bee66183929f9474b5a161432918604Vinit Deshpande
1046f97140d51d14ce0659d381f443c08dbd94dfea28Honore Tricot    public boolean simAuthResponse(int id, String type, String response) {
1047f97140d51d14ce0659d381f443c08dbd94dfea28Honore Tricot        // with type = GSM-AUTH, UMTS-AUTH or UMTS-AUTS
104818786eca942042388748b0d98979f21c9dff4a89Mitchell Wills        return doBooleanCommand("CTRL-RSP-SIM-" + id + ":" + type + response);
104933b575ca6bee66183929f9474b5a161432918604Vinit Deshpande    }
105033b575ca6bee66183929f9474b5a161432918604Vinit Deshpande
105126eebecc04b55129a902d1e7b670fe05668c14faVinit Deshpande    public boolean simAuthFailedResponse(int id) {
105226eebecc04b55129a902d1e7b670fe05668c14faVinit Deshpande        // should be used with type GSM-AUTH
105326eebecc04b55129a902d1e7b670fe05668c14faVinit Deshpande        return doBooleanCommand("CTRL-RSP-SIM-" + id + ":GSM-FAIL");
105426eebecc04b55129a902d1e7b670fe05668c14faVinit Deshpande    }
105526eebecc04b55129a902d1e7b670fe05668c14faVinit Deshpande
105626eebecc04b55129a902d1e7b670fe05668c14faVinit Deshpande    public boolean umtsAuthFailedResponse(int id) {
105726eebecc04b55129a902d1e7b670fe05668c14faVinit Deshpande        // should be used with type UMTS-AUTH
105826eebecc04b55129a902d1e7b670fe05668c14faVinit Deshpande        return doBooleanCommand("CTRL-RSP-SIM-" + id + ":UMTS-FAIL");
105926eebecc04b55129a902d1e7b670fe05668c14faVinit Deshpande    }
106026eebecc04b55129a902d1e7b670fe05668c14faVinit Deshpande
1061ad607d99c372160c7d4b38e755e1b47d6419856eHonore Tricot    public boolean simIdentityResponse(int id, String response) {
106218786eca942042388748b0d98979f21c9dff4a89Mitchell Wills        return doBooleanCommand("CTRL-RSP-IDENTITY-" + id + ":" + response);
1063ad607d99c372160c7d4b38e755e1b47d6419856eHonore Tricot    }
1064ad607d99c372160c7d4b38e755e1b47d6419856eHonore Tricot
1065155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    /* Configures an access point connection */
1066155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    public boolean startWpsRegistrar(String bssid, String pin) {
1067155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        if (TextUtils.isEmpty(bssid) || TextUtils.isEmpty(pin)) return false;
1068155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        return doBooleanCommand("WPS_REG " + bssid + " " + pin);
1069155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    }
1070155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
1071155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    public boolean cancelWps() {
1072155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        return doBooleanCommand("WPS_CANCEL");
1073155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    }
1074155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
1075155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    public boolean setPersistentReconnect(boolean enabled) {
1076155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        int value = (enabled == true) ? 1 : 0;
1077155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        return doBooleanCommand("SET persistent_reconnect " + value);
1078155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    }
1079155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
1080155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    public boolean setDeviceName(String name) {
1081155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        return doBooleanCommand("SET device_name " + name);
1082155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    }
1083155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
1084155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    public boolean setDeviceType(String type) {
1085155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        return doBooleanCommand("SET device_type " + type);
1086155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    }
1087155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
1088155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    public boolean setConfigMethods(String cfg) {
1089155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        return doBooleanCommand("SET config_methods " + cfg);
1090155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    }
1091155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
1092155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    public boolean setManufacturer(String value) {
1093155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        return doBooleanCommand("SET manufacturer " + value);
1094155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    }
1095155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
1096155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    public boolean setModelName(String value) {
1097155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        return doBooleanCommand("SET model_name " + value);
1098155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    }
1099155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
1100155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    public boolean setModelNumber(String value) {
1101155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        return doBooleanCommand("SET model_number " + value);
1102155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    }
1103155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
1104155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    public boolean setSerialNumber(String value) {
1105155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        return doBooleanCommand("SET serial_number " + value);
1106155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    }
1107155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
1108155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    public boolean setP2pSsidPostfix(String postfix) {
1109155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        return doBooleanCommand("SET p2p_ssid_postfix " + postfix);
1110155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    }
1111155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
1112155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    public boolean setP2pGroupIdle(String iface, int time) {
1113956f54b391677d78379729dd14518edddf3c7660Etan Cohen        synchronized (sLock) {
1114155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande            return doBooleanCommandNative("IFNAME=" + iface + " SET p2p_group_idle " + time);
1115155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        }
1116155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    }
1117155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
1118155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    public void setPowerSave(boolean enabled) {
1119155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        if (enabled) {
1120155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande            doBooleanCommand("SET ps 1");
1121155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        } else {
1122155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande            doBooleanCommand("SET ps 0");
1123155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        }
1124155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    }
1125155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
1126155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    public boolean setP2pPowerSave(String iface, boolean enabled) {
1127956f54b391677d78379729dd14518edddf3c7660Etan Cohen        synchronized (sLock) {
1128155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande            if (enabled) {
1129155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande                return doBooleanCommandNative("IFNAME=" + iface + " P2P_SET ps 1");
1130155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande            } else {
1131155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande                return doBooleanCommandNative("IFNAME=" + iface + " P2P_SET ps 0");
1132155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande            }
1133155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        }
1134155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    }
1135155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
1136155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    public boolean setWfdEnable(boolean enable) {
1137155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        return doBooleanCommand("SET wifi_display " + (enable ? "1" : "0"));
1138155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    }
1139155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
1140155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    public boolean setWfdDeviceInfo(String hex) {
1141155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        return doBooleanCommand("WFD_SUBELEM_SET 0 " + hex);
1142155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    }
1143155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
1144155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    /**
1145155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande     * "sta" prioritizes STA connection over P2P and "p2p" prioritizes
1146155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande     * P2P connection over STA
1147155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande     */
1148155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    public boolean setConcurrencyPriority(String s) {
1149155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        return doBooleanCommand("P2P_SET conc_pref " + s);
1150155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    }
1151155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
1152155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    public boolean p2pFind() {
1153155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        return doBooleanCommand("P2P_FIND");
1154155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    }
1155155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
1156155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    public boolean p2pFind(int timeout) {
1157155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        if (timeout <= 0) {
1158155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande            return p2pFind();
1159155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        }
1160155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        return doBooleanCommand("P2P_FIND " + timeout);
1161155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    }
1162155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
1163155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    public boolean p2pStopFind() {
1164155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande       return doBooleanCommand("P2P_STOP_FIND");
1165155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    }
1166155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
1167155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    public boolean p2pListen() {
1168155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        return doBooleanCommand("P2P_LISTEN");
1169155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    }
1170155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
1171155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    public boolean p2pListen(int timeout) {
1172155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        if (timeout <= 0) {
1173155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande            return p2pListen();
1174155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        }
1175155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        return doBooleanCommand("P2P_LISTEN " + timeout);
1176155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    }
1177155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
1178155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    public boolean p2pExtListen(boolean enable, int period, int interval) {
1179155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        if (enable && interval < period) {
1180155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande            return false;
1181155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        }
1182155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        return doBooleanCommand("P2P_EXT_LISTEN"
1183155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande                    + (enable ? (" " + period + " " + interval) : ""));
1184155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    }
1185155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
1186155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    public boolean p2pSetChannel(int lc, int oc) {
1187155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        if (DBG) Log.d(mTAG, "p2pSetChannel: lc="+lc+", oc="+oc);
1188155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
1189956f54b391677d78379729dd14518edddf3c7660Etan Cohen        synchronized (sLock) {
119018786eca942042388748b0d98979f21c9dff4a89Mitchell Wills            if (lc >=1 && lc <= 11) {
119118786eca942042388748b0d98979f21c9dff4a89Mitchell Wills                if (!doBooleanCommand("P2P_SET listen_channel " + lc)) {
119218786eca942042388748b0d98979f21c9dff4a89Mitchell Wills                    return false;
119318786eca942042388748b0d98979f21c9dff4a89Mitchell Wills                }
119418786eca942042388748b0d98979f21c9dff4a89Mitchell Wills            } else if (lc != 0) {
1195155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande                return false;
1196155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande            }
1197155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
119818786eca942042388748b0d98979f21c9dff4a89Mitchell Wills            if (oc >= 1 && oc <= 165 ) {
119918786eca942042388748b0d98979f21c9dff4a89Mitchell Wills                int freq = (oc <= 14 ? 2407 : 5000) + oc * 5;
120018786eca942042388748b0d98979f21c9dff4a89Mitchell Wills                return doBooleanCommand("P2P_SET disallow_freq 1000-"
120118786eca942042388748b0d98979f21c9dff4a89Mitchell Wills                        + (freq - 5) + "," + (freq + 5) + "-6000");
120218786eca942042388748b0d98979f21c9dff4a89Mitchell Wills            } else if (oc == 0) {
120318786eca942042388748b0d98979f21c9dff4a89Mitchell Wills                /* oc==0 disables "P2P_SET disallow_freq" (enables all freqs) */
120418786eca942042388748b0d98979f21c9dff4a89Mitchell Wills                return doBooleanCommand("P2P_SET disallow_freq \"\"");
120518786eca942042388748b0d98979f21c9dff4a89Mitchell Wills            }
1206155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        }
1207155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        return false;
1208155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    }
1209155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
1210155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    public boolean p2pFlush() {
1211155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        return doBooleanCommand("P2P_FLUSH");
1212155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    }
1213155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
121418786eca942042388748b0d98979f21c9dff4a89Mitchell Wills    private static final int DEFAULT_GROUP_OWNER_INTENT     = 6;
1215155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    /* p2p_connect <peer device address> <pbc|pin|PIN#> [label|display|keypad]
1216155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        [persistent] [join|auth] [go_intent=<0..15>] [freq=<in MHz>] */
1217155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    public String p2pConnect(WifiP2pConfig config, boolean joinExistingGroup) {
1218155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        if (config == null) return null;
1219155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        List<String> args = new ArrayList<String>();
1220155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        WpsInfo wps = config.wps;
1221155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        args.add(config.deviceAddress);
1222155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
1223155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        switch (wps.setup) {
1224155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande            case WpsInfo.PBC:
1225155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande                args.add("pbc");
1226155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande                break;
1227155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande            case WpsInfo.DISPLAY:
1228155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande                if (TextUtils.isEmpty(wps.pin)) {
1229155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande                    args.add("pin");
1230155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande                } else {
1231155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande                    args.add(wps.pin);
1232155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande                }
1233155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande                args.add("display");
1234155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande                break;
1235155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande            case WpsInfo.KEYPAD:
1236155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande                args.add(wps.pin);
1237155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande                args.add("keypad");
1238155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande                break;
1239155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande            case WpsInfo.LABEL:
1240155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande                args.add(wps.pin);
1241155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande                args.add("label");
1242155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande            default:
1243155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande                break;
1244155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        }
1245155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
1246155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        if (config.netId == WifiP2pGroup.PERSISTENT_NET_ID) {
1247155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande            args.add("persistent");
1248155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        }
1249155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
1250155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        if (joinExistingGroup) {
1251155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande            args.add("join");
1252155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        } else {
1253155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande            //TODO: This can be adapted based on device plugged in state and
1254155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande            //device battery state
1255155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande            int groupOwnerIntent = config.groupOwnerIntent;
1256155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande            if (groupOwnerIntent < 0 || groupOwnerIntent > 15) {
1257155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande                groupOwnerIntent = DEFAULT_GROUP_OWNER_INTENT;
1258155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande            }
1259155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande            args.add("go_intent=" + groupOwnerIntent);
1260155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        }
1261155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
1262155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        String command = "P2P_CONNECT ";
1263155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        for (String s : args) command += s + " ";
1264155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
1265155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        return doStringCommand(command);
1266155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    }
1267155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
1268155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    public boolean p2pCancelConnect() {
1269155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        return doBooleanCommand("P2P_CANCEL");
1270155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    }
1271155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
1272155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    public boolean p2pProvisionDiscovery(WifiP2pConfig config) {
1273155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        if (config == null) return false;
1274155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
1275155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        switch (config.wps.setup) {
1276155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande            case WpsInfo.PBC:
1277155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande                return doBooleanCommand("P2P_PROV_DISC " + config.deviceAddress + " pbc");
1278155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande            case WpsInfo.DISPLAY:
1279155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande                //We are doing display, so provision discovery is keypad
1280155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande                return doBooleanCommand("P2P_PROV_DISC " + config.deviceAddress + " keypad");
1281155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande            case WpsInfo.KEYPAD:
1282155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande                //We are doing keypad, so provision discovery is display
1283155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande                return doBooleanCommand("P2P_PROV_DISC " + config.deviceAddress + " display");
1284155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande            default:
1285155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande                break;
1286155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        }
1287155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        return false;
1288155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    }
1289155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
1290155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    public boolean p2pGroupAdd(boolean persistent) {
1291155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        if (persistent) {
1292155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande            return doBooleanCommand("P2P_GROUP_ADD persistent");
1293155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        }
1294155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        return doBooleanCommand("P2P_GROUP_ADD");
1295155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    }
1296155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
1297155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    public boolean p2pGroupAdd(int netId) {
1298155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        return doBooleanCommand("P2P_GROUP_ADD persistent=" + netId);
1299155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    }
1300155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
1301155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    public boolean p2pGroupRemove(String iface) {
1302155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        if (TextUtils.isEmpty(iface)) return false;
1303956f54b391677d78379729dd14518edddf3c7660Etan Cohen        synchronized (sLock) {
1304155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande            return doBooleanCommandNative("IFNAME=" + iface + " P2P_GROUP_REMOVE " + iface);
1305155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        }
1306155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    }
1307155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
1308155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    public boolean p2pReject(String deviceAddress) {
1309155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        return doBooleanCommand("P2P_REJECT " + deviceAddress);
1310155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    }
1311155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
1312155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    /* Invite a peer to a group */
1313155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    public boolean p2pInvite(WifiP2pGroup group, String deviceAddress) {
1314155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        if (TextUtils.isEmpty(deviceAddress)) return false;
1315155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
1316155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        if (group == null) {
1317155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande            return doBooleanCommand("P2P_INVITE peer=" + deviceAddress);
1318155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        } else {
1319155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande            return doBooleanCommand("P2P_INVITE group=" + group.getInterface()
1320155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande                    + " peer=" + deviceAddress + " go_dev_addr=" + group.getOwner().deviceAddress);
1321155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        }
1322155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    }
1323155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
1324155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    /* Reinvoke a persistent connection */
1325155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    public boolean p2pReinvoke(int netId, String deviceAddress) {
1326155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        if (TextUtils.isEmpty(deviceAddress) || netId < 0) return false;
1327155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
1328155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        return doBooleanCommand("P2P_INVITE persistent=" + netId + " peer=" + deviceAddress);
1329155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    }
1330155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
1331155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    public String p2pGetSsid(String deviceAddress) {
1332155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        return p2pGetParam(deviceAddress, "oper_ssid");
1333155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    }
1334155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
1335155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    public String p2pGetDeviceAddress() {
133636286b23c4e30f042ed753a670c2b462ebf13a48Vinit Deshpande        Log.d(TAG, "p2pGetDeviceAddress");
133736286b23c4e30f042ed753a670c2b462ebf13a48Vinit Deshpande
133827f4b0c6ea9533e91863da48cefc80f8b5a88d1eVinit Deshpande        String status = null;
133927f4b0c6ea9533e91863da48cefc80f8b5a88d1eVinit Deshpande
134036286b23c4e30f042ed753a670c2b462ebf13a48Vinit Deshpande        /* Explicitly calling the API without IFNAME= prefix to take care of the devices that
134136286b23c4e30f042ed753a670c2b462ebf13a48Vinit Deshpande        don't have p2p0 interface. Supplicant seems to be returning the correct address anyway. */
134236286b23c4e30f042ed753a670c2b462ebf13a48Vinit Deshpande
1343956f54b391677d78379729dd14518edddf3c7660Etan Cohen        synchronized (sLock) {
134427f4b0c6ea9533e91863da48cefc80f8b5a88d1eVinit Deshpande            status = doStringCommandNative("STATUS");
134527f4b0c6ea9533e91863da48cefc80f8b5a88d1eVinit Deshpande        }
134627f4b0c6ea9533e91863da48cefc80f8b5a88d1eVinit Deshpande
134727f4b0c6ea9533e91863da48cefc80f8b5a88d1eVinit Deshpande        String result = "";
134836286b23c4e30f042ed753a670c2b462ebf13a48Vinit Deshpande        if (status != null) {
134936286b23c4e30f042ed753a670c2b462ebf13a48Vinit Deshpande            String[] tokens = status.split("\n");
135036286b23c4e30f042ed753a670c2b462ebf13a48Vinit Deshpande            for (String token : tokens) {
135136286b23c4e30f042ed753a670c2b462ebf13a48Vinit Deshpande                if (token.startsWith("p2p_device_address=")) {
135236286b23c4e30f042ed753a670c2b462ebf13a48Vinit Deshpande                    String[] nameValue = token.split("=");
135336286b23c4e30f042ed753a670c2b462ebf13a48Vinit Deshpande                    if (nameValue.length != 2)
135436286b23c4e30f042ed753a670c2b462ebf13a48Vinit Deshpande                        break;
135536286b23c4e30f042ed753a670c2b462ebf13a48Vinit Deshpande                    result = nameValue[1];
135636286b23c4e30f042ed753a670c2b462ebf13a48Vinit Deshpande                }
1357155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande            }
1358155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        }
135936286b23c4e30f042ed753a670c2b462ebf13a48Vinit Deshpande
136036286b23c4e30f042ed753a670c2b462ebf13a48Vinit Deshpande        Log.d(TAG, "p2pGetDeviceAddress returning " + result);
136136286b23c4e30f042ed753a670c2b462ebf13a48Vinit Deshpande        return result;
1362155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    }
1363155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
1364155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    public int getGroupCapability(String deviceAddress) {
1365155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        int gc = 0;
1366155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        if (TextUtils.isEmpty(deviceAddress)) return gc;
1367155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        String peerInfo = p2pPeer(deviceAddress);
1368155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        if (TextUtils.isEmpty(peerInfo)) return gc;
1369155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
1370155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        String[] tokens = peerInfo.split("\n");
1371155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        for (String token : tokens) {
1372155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande            if (token.startsWith("group_capab=")) {
1373155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande                String[] nameValue = token.split("=");
1374155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande                if (nameValue.length != 2) break;
1375155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande                try {
1376155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande                    return Integer.decode(nameValue[1]);
1377155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande                } catch(NumberFormatException e) {
1378155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande                    return gc;
1379155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande                }
1380155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande            }
1381155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        }
1382155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        return gc;
1383155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    }
1384155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
1385155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    public String p2pPeer(String deviceAddress) {
1386155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        return doStringCommand("P2P_PEER " + deviceAddress);
1387155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    }
1388155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
1389155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    private String p2pGetParam(String deviceAddress, String key) {
1390155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        if (deviceAddress == null) return null;
1391155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
1392155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        String peerInfo = p2pPeer(deviceAddress);
1393155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        if (peerInfo == null) return null;
1394155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        String[] tokens= peerInfo.split("\n");
1395155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
1396155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        key += "=";
1397155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        for (String token : tokens) {
1398155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande            if (token.startsWith(key)) {
1399155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande                String[] nameValue = token.split("=");
1400155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande                if (nameValue.length != 2) break;
1401155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande                return nameValue[1];
1402155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande            }
1403155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        }
1404155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        return null;
1405155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    }
1406155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
1407155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    public boolean p2pServiceAdd(WifiP2pServiceInfo servInfo) {
1408155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        /*
1409155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande         * P2P_SERVICE_ADD bonjour <query hexdump> <RDATA hexdump>
1410155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande         * P2P_SERVICE_ADD upnp <version hex> <service>
1411155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande         *
1412155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande         * e.g)
1413155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande         * [Bonjour]
1414155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande         * # IP Printing over TCP (PTR) (RDATA=MyPrinter._ipp._tcp.local.)
1415155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande         * P2P_SERVICE_ADD bonjour 045f697070c00c000c01 094d795072696e746572c027
1416155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande         * # IP Printing over TCP (TXT) (RDATA=txtvers=1,pdl=application/postscript)
1417155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande         * P2P_SERVICE_ADD bonjour 096d797072696e746572045f697070c00c001001
1418155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande         *  09747874766572733d311a70646c3d6170706c69636174696f6e2f706f7374736372797074
1419155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande         *
1420155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande         * [UPnP]
1421155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande         * P2P_SERVICE_ADD upnp 10 uuid:6859dede-8574-59ab-9332-123456789012
1422155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande         * P2P_SERVICE_ADD upnp 10 uuid:6859dede-8574-59ab-9332-123456789012::upnp:rootdevice
1423155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande         * P2P_SERVICE_ADD upnp 10 uuid:6859dede-8574-59ab-9332-123456789012::urn:schemas-upnp
1424155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande         * -org:device:InternetGatewayDevice:1
1425155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande         * P2P_SERVICE_ADD upnp 10 uuid:6859dede-8574-59ab-9322-123456789012::urn:schemas-upnp
1426155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande         * -org:service:ContentDirectory:2
1427155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande         */
1428956f54b391677d78379729dd14518edddf3c7660Etan Cohen        synchronized (sLock) {
142918786eca942042388748b0d98979f21c9dff4a89Mitchell Wills            for (String s : servInfo.getSupplicantQueryList()) {
143018786eca942042388748b0d98979f21c9dff4a89Mitchell Wills                String command = "P2P_SERVICE_ADD";
143118786eca942042388748b0d98979f21c9dff4a89Mitchell Wills                command += (" " + s);
143218786eca942042388748b0d98979f21c9dff4a89Mitchell Wills                if (!doBooleanCommand(command)) {
143318786eca942042388748b0d98979f21c9dff4a89Mitchell Wills                    return false;
143418786eca942042388748b0d98979f21c9dff4a89Mitchell Wills                }
1435155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande            }
1436155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        }
1437155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        return true;
1438155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    }
1439155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
1440155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    public boolean p2pServiceDel(WifiP2pServiceInfo servInfo) {
1441155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        /*
1442155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande         * P2P_SERVICE_DEL bonjour <query hexdump>
1443155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande         * P2P_SERVICE_DEL upnp <version hex> <service>
1444155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande         */
1445956f54b391677d78379729dd14518edddf3c7660Etan Cohen        synchronized (sLock) {
144618786eca942042388748b0d98979f21c9dff4a89Mitchell Wills            for (String s : servInfo.getSupplicantQueryList()) {
144718786eca942042388748b0d98979f21c9dff4a89Mitchell Wills                String command = "P2P_SERVICE_DEL ";
1448155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
144918786eca942042388748b0d98979f21c9dff4a89Mitchell Wills                String[] data = s.split(" ");
145018786eca942042388748b0d98979f21c9dff4a89Mitchell Wills                if (data.length < 2) {
145118786eca942042388748b0d98979f21c9dff4a89Mitchell Wills                    return false;
145218786eca942042388748b0d98979f21c9dff4a89Mitchell Wills                }
145318786eca942042388748b0d98979f21c9dff4a89Mitchell Wills                if ("upnp".equals(data[0])) {
145418786eca942042388748b0d98979f21c9dff4a89Mitchell Wills                    command += s;
145518786eca942042388748b0d98979f21c9dff4a89Mitchell Wills                } else if ("bonjour".equals(data[0])) {
145618786eca942042388748b0d98979f21c9dff4a89Mitchell Wills                    command += data[0];
145718786eca942042388748b0d98979f21c9dff4a89Mitchell Wills                    command += (" " + data[1]);
145818786eca942042388748b0d98979f21c9dff4a89Mitchell Wills                } else {
145918786eca942042388748b0d98979f21c9dff4a89Mitchell Wills                    return false;
146018786eca942042388748b0d98979f21c9dff4a89Mitchell Wills                }
146118786eca942042388748b0d98979f21c9dff4a89Mitchell Wills                if (!doBooleanCommand(command)) {
146218786eca942042388748b0d98979f21c9dff4a89Mitchell Wills                    return false;
146318786eca942042388748b0d98979f21c9dff4a89Mitchell Wills                }
1464155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande            }
1465155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        }
1466155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        return true;
1467155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    }
1468155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
1469155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    public boolean p2pServiceFlush() {
1470155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        return doBooleanCommand("P2P_SERVICE_FLUSH");
1471155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    }
1472155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
1473155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    public String p2pServDiscReq(String addr, String query) {
1474155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        String command = "P2P_SERV_DISC_REQ";
1475155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        command += (" " + addr);
1476155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        command += (" " + query);
1477155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
1478155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        return doStringCommand(command);
1479155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    }
1480155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
1481155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    public boolean p2pServDiscCancelReq(String id) {
1482155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        return doBooleanCommand("P2P_SERV_DISC_CANCEL_REQ " + id);
1483155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    }
1484155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
1485155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    /* Set the current mode of miracast operation.
1486155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande     *  0 = disabled
1487155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande     *  1 = operating as source
1488155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande     *  2 = operating as sink
1489155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande     */
1490155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    public void setMiracastMode(int mode) {
1491155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        // Note: optional feature on the driver. It is ok for this to fail.
1492155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        doBooleanCommand("DRIVER MIRACAST " + mode);
1493155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    }
14943f7ef65ab71619040032aee96b5599849881d6fdAndres Morales
1495446db2d5457456743e4476029e14d7c3bb9f5bccYuhao Zheng    public boolean fetchAnqp(String bssid, String subtypes) {
1496446db2d5457456743e4476029e14d7c3bb9f5bccYuhao Zheng        return doBooleanCommand("ANQP_GET " + bssid + " " + subtypes);
1497446db2d5457456743e4476029e14d7c3bb9f5bccYuhao Zheng    }
1498446db2d5457456743e4476029e14d7c3bb9f5bccYuhao Zheng
1499f49a59bda006b13e0118d144e0a4a5f569b2251eVinit Deshpande    /*
1500f49a59bda006b13e0118d144e0a4a5f569b2251eVinit Deshpande     * NFC-related calls
1501f49a59bda006b13e0118d144e0a4a5f569b2251eVinit Deshpande     */
1502f49a59bda006b13e0118d144e0a4a5f569b2251eVinit Deshpande    public String getNfcWpsConfigurationToken(int netId) {
1503f49a59bda006b13e0118d144e0a4a5f569b2251eVinit Deshpande        return doStringCommand("WPS_NFC_CONFIG_TOKEN WPS " + netId);
1504f49a59bda006b13e0118d144e0a4a5f569b2251eVinit Deshpande    }
1505f49a59bda006b13e0118d144e0a4a5f569b2251eVinit Deshpande
1506f49a59bda006b13e0118d144e0a4a5f569b2251eVinit Deshpande    public String getNfcHandoverRequest() {
1507f49a59bda006b13e0118d144e0a4a5f569b2251eVinit Deshpande        return doStringCommand("NFC_GET_HANDOVER_REQ NDEF P2P-CR");
1508f49a59bda006b13e0118d144e0a4a5f569b2251eVinit Deshpande    }
1509f49a59bda006b13e0118d144e0a4a5f569b2251eVinit Deshpande
1510f49a59bda006b13e0118d144e0a4a5f569b2251eVinit Deshpande    public String getNfcHandoverSelect() {
1511f49a59bda006b13e0118d144e0a4a5f569b2251eVinit Deshpande        return doStringCommand("NFC_GET_HANDOVER_SEL NDEF P2P-CR");
1512f49a59bda006b13e0118d144e0a4a5f569b2251eVinit Deshpande    }
1513f49a59bda006b13e0118d144e0a4a5f569b2251eVinit Deshpande
1514f49a59bda006b13e0118d144e0a4a5f569b2251eVinit Deshpande    public boolean initiatorReportNfcHandover(String selectMessage) {
1515f49a59bda006b13e0118d144e0a4a5f569b2251eVinit Deshpande        return doBooleanCommand("NFC_REPORT_HANDOVER INIT P2P 00 " + selectMessage);
1516f49a59bda006b13e0118d144e0a4a5f569b2251eVinit Deshpande    }
1517f49a59bda006b13e0118d144e0a4a5f569b2251eVinit Deshpande
1518f49a59bda006b13e0118d144e0a4a5f569b2251eVinit Deshpande    public boolean responderReportNfcHandover(String requestMessage) {
1519f49a59bda006b13e0118d144e0a4a5f569b2251eVinit Deshpande        return doBooleanCommand("NFC_REPORT_HANDOVER RESP P2P " + requestMessage + " 00");
1520f49a59bda006b13e0118d144e0a4a5f569b2251eVinit Deshpande    }
1521f49a59bda006b13e0118d144e0a4a5f569b2251eVinit Deshpande
15227e0315140767e6f12200f7dbbe7aff43b4f75089Vinit Deshpande
15237e0315140767e6f12200f7dbbe7aff43b4f75089Vinit Deshpande    /* kernel logging support */
15247e0315140767e6f12200f7dbbe7aff43b4f75089Vinit Deshpande    private static native byte[] readKernelLogNative();
15257e0315140767e6f12200f7dbbe7aff43b4f75089Vinit Deshpande
15267e0315140767e6f12200f7dbbe7aff43b4f75089Vinit Deshpande    synchronized public String readKernelLog() {
15277e0315140767e6f12200f7dbbe7aff43b4f75089Vinit Deshpande        byte[] bytes = readKernelLogNative();
15287e0315140767e6f12200f7dbbe7aff43b4f75089Vinit Deshpande        if (bytes != null) {
15297e0315140767e6f12200f7dbbe7aff43b4f75089Vinit Deshpande            CharsetDecoder decoder = StandardCharsets.UTF_8.newDecoder();
15307e0315140767e6f12200f7dbbe7aff43b4f75089Vinit Deshpande            try {
15317e0315140767e6f12200f7dbbe7aff43b4f75089Vinit Deshpande                CharBuffer decoded = decoder.decode(ByteBuffer.wrap(bytes));
15327e0315140767e6f12200f7dbbe7aff43b4f75089Vinit Deshpande                return decoded.toString();
15337e0315140767e6f12200f7dbbe7aff43b4f75089Vinit Deshpande            } catch (CharacterCodingException cce) {
15347e0315140767e6f12200f7dbbe7aff43b4f75089Vinit Deshpande                return new String(bytes, StandardCharsets.ISO_8859_1);
15357e0315140767e6f12200f7dbbe7aff43b4f75089Vinit Deshpande            }
15367e0315140767e6f12200f7dbbe7aff43b4f75089Vinit Deshpande        } else {
15377e0315140767e6f12200f7dbbe7aff43b4f75089Vinit Deshpande            return "*** failed to read kernel log ***";
15387e0315140767e6f12200f7dbbe7aff43b4f75089Vinit Deshpande        }
15397e0315140767e6f12200f7dbbe7aff43b4f75089Vinit Deshpande    }
15407e0315140767e6f12200f7dbbe7aff43b4f75089Vinit Deshpande
15417f9a15d554f69311a0db43347d6473a7c4c46e2eVinit Deshapnde    /* WIFI HAL support */
15427f9a15d554f69311a0db43347d6473a7c4c46e2eVinit Deshapnde
154318786eca942042388748b0d98979f21c9dff4a89Mitchell Wills    // HAL command ids
154418786eca942042388748b0d98979f21c9dff4a89Mitchell Wills    private static int sCmdId = 1;
154518786eca942042388748b0d98979f21c9dff4a89Mitchell Wills    private static int getNewCmdIdLocked() {
154618786eca942042388748b0d98979f21c9dff4a89Mitchell Wills        return sCmdId++;
154718786eca942042388748b0d98979f21c9dff4a89Mitchell Wills    }
154818786eca942042388748b0d98979f21c9dff4a89Mitchell Wills
1549b0b1d59786de7ff1fc98e31a16a7d710458bf549Vinit Deshpande    private static final String TAG = "WifiNative-HAL";
1550f49a59bda006b13e0118d144e0a4a5f569b2251eVinit Deshpande    private static long sWifiHalHandle = 0;             /* used by JNI to save wifi_handle */
1551f49a59bda006b13e0118d144e0a4a5f569b2251eVinit Deshpande    private static long[] sWifiIfaceHandles = null;     /* used by JNI to save interface handles */
1552956f54b391677d78379729dd14518edddf3c7660Etan Cohen    public static int sWlan0Index = -1;
1553f49a59bda006b13e0118d144e0a4a5f569b2251eVinit Deshpande    private static MonitorThread sThread;
1554f49a59bda006b13e0118d144e0a4a5f569b2251eVinit Deshpande    private static final int STOP_HAL_TIMEOUT_MS = 1000;
15557f9a15d554f69311a0db43347d6473a7c4c46e2eVinit Deshapnde
1556b0b1d59786de7ff1fc98e31a16a7d710458bf549Vinit Deshpande    private static native boolean startHalNative();
1557b0b1d59786de7ff1fc98e31a16a7d710458bf549Vinit Deshpande    private static native void stopHalNative();
1558b0b1d59786de7ff1fc98e31a16a7d710458bf549Vinit Deshpande    private static native void waitForHalEventNative();
15597f9a15d554f69311a0db43347d6473a7c4c46e2eVinit Deshapnde
1560b0b1d59786de7ff1fc98e31a16a7d710458bf549Vinit Deshpande    private static class MonitorThread extends Thread {
15617ef73dd1b6e43c72b3841723504cd86dc402a134Vinit Deshapnde        public void run() {
1562b0b1d59786de7ff1fc98e31a16a7d710458bf549Vinit Deshpande            Log.i(TAG, "Waiting for HAL events mWifiHalHandle=" + Long.toString(sWifiHalHandle));
15637ef73dd1b6e43c72b3841723504cd86dc402a134Vinit Deshapnde            waitForHalEventNative();
15647ef73dd1b6e43c72b3841723504cd86dc402a134Vinit Deshapnde        }
15657ef73dd1b6e43c72b3841723504cd86dc402a134Vinit Deshapnde    }
15667ef73dd1b6e43c72b3841723504cd86dc402a134Vinit Deshapnde
156718786eca942042388748b0d98979f21c9dff4a89Mitchell Wills    public boolean startHal() {
1568d6307b404302949f6dadd14fa0860ff1aec432dcxinhe        String debugLog = "startHal stack: ";
1569d6307b404302949f6dadd14fa0860ff1aec432dcxinhe        java.lang.StackTraceElement[] elements = Thread.currentThread().getStackTrace();
1570d6307b404302949f6dadd14fa0860ff1aec432dcxinhe        for (int i = 2; i < elements.length && i <= 7; i++ ) {
1571d6307b404302949f6dadd14fa0860ff1aec432dcxinhe            debugLog = debugLog + " - " + elements[i].getMethodName();
1572d6307b404302949f6dadd14fa0860ff1aec432dcxinhe        }
1573d6307b404302949f6dadd14fa0860ff1aec432dcxinhe
1574956f54b391677d78379729dd14518edddf3c7660Etan Cohen        sLocalLog.log(debugLog);
1575d6307b404302949f6dadd14fa0860ff1aec432dcxinhe
1576956f54b391677d78379729dd14518edddf3c7660Etan Cohen        synchronized (sLock) {
1577918b4bdc2ae16438c5213320daeff1d551f4aed8Ningyuan Wang            if (startHalNative()) {
1578918b4bdc2ae16438c5213320daeff1d551f4aed8Ningyuan Wang                int wlan0Index = queryInterfaceIndex(mInterfaceName);
1579918b4bdc2ae16438c5213320daeff1d551f4aed8Ningyuan Wang                if (wlan0Index == -1) {
1580918b4bdc2ae16438c5213320daeff1d551f4aed8Ningyuan Wang                    if (DBG) sLocalLog.log("Could not find interface with name: " + mInterfaceName);
1581918b4bdc2ae16438c5213320daeff1d551f4aed8Ningyuan Wang                    return false;
1582918b4bdc2ae16438c5213320daeff1d551f4aed8Ningyuan Wang                }
1583918b4bdc2ae16438c5213320daeff1d551f4aed8Ningyuan Wang                sWlan0Index = wlan0Index;
1584f49a59bda006b13e0118d144e0a4a5f569b2251eVinit Deshpande                sThread = new MonitorThread();
1585f49a59bda006b13e0118d144e0a4a5f569b2251eVinit Deshpande                sThread.start();
1586aabe7a9f2f32915fd1f25416a6d2034a844005d6vandwalle                return true;
1587aabe7a9f2f32915fd1f25416a6d2034a844005d6vandwalle            } else {
1588956f54b391677d78379729dd14518edddf3c7660Etan Cohen                if (DBG) sLocalLog.log("Could not start hal");
1589f49a59bda006b13e0118d144e0a4a5f569b2251eVinit Deshpande                Log.e(TAG, "Could not start hal");
1590aabe7a9f2f32915fd1f25416a6d2034a844005d6vandwalle                return false;
1591aabe7a9f2f32915fd1f25416a6d2034a844005d6vandwalle            }
15927ef73dd1b6e43c72b3841723504cd86dc402a134Vinit Deshapnde        }
15937ef73dd1b6e43c72b3841723504cd86dc402a134Vinit Deshapnde    }
15947ef73dd1b6e43c72b3841723504cd86dc402a134Vinit Deshapnde
159518786eca942042388748b0d98979f21c9dff4a89Mitchell Wills    public void stopHal() {
1596956f54b391677d78379729dd14518edddf3c7660Etan Cohen        synchronized (sLock) {
159771d0be16c79791a005d60f9d5fa7d2d81e1f8c80xinhe            if (isHalStarted()) {
1598f49a59bda006b13e0118d144e0a4a5f569b2251eVinit Deshpande                stopHalNative();
1599f49a59bda006b13e0118d144e0a4a5f569b2251eVinit Deshpande                try {
1600f49a59bda006b13e0118d144e0a4a5f569b2251eVinit Deshpande                    sThread.join(STOP_HAL_TIMEOUT_MS);
1601f49a59bda006b13e0118d144e0a4a5f569b2251eVinit Deshpande                    Log.d(TAG, "HAL event thread stopped successfully");
1602f49a59bda006b13e0118d144e0a4a5f569b2251eVinit Deshpande                } catch (InterruptedException e) {
1603f49a59bda006b13e0118d144e0a4a5f569b2251eVinit Deshpande                    Log.e(TAG, "Could not stop HAL cleanly");
1604f49a59bda006b13e0118d144e0a4a5f569b2251eVinit Deshpande                }
1605f49a59bda006b13e0118d144e0a4a5f569b2251eVinit Deshpande                sThread = null;
1606f49a59bda006b13e0118d144e0a4a5f569b2251eVinit Deshpande                sWifiHalHandle = 0;
1607f49a59bda006b13e0118d144e0a4a5f569b2251eVinit Deshpande                sWifiIfaceHandles = null;
1608f49a59bda006b13e0118d144e0a4a5f569b2251eVinit Deshpande                sWlan0Index = -1;
1609f49a59bda006b13e0118d144e0a4a5f569b2251eVinit Deshpande            }
1610f49a59bda006b13e0118d144e0a4a5f569b2251eVinit Deshpande        }
16117ef73dd1b6e43c72b3841723504cd86dc402a134Vinit Deshapnde    }
16127f9a15d554f69311a0db43347d6473a7c4c46e2eVinit Deshapnde
161318786eca942042388748b0d98979f21c9dff4a89Mitchell Wills    public boolean isHalStarted() {
161471d0be16c79791a005d60f9d5fa7d2d81e1f8c80xinhe        return (sWifiHalHandle != 0);
161571d0be16c79791a005d60f9d5fa7d2d81e1f8c80xinhe    }
1616b0b1d59786de7ff1fc98e31a16a7d710458bf549Vinit Deshpande    private static native int getInterfacesNative();
16177f9a15d554f69311a0db43347d6473a7c4c46e2eVinit Deshapnde
1618918b4bdc2ae16438c5213320daeff1d551f4aed8Ningyuan Wang    public int queryInterfaceIndex(String interfaceName) {
1619956f54b391677d78379729dd14518edddf3c7660Etan Cohen        synchronized (sLock) {
162071d0be16c79791a005d60f9d5fa7d2d81e1f8c80xinhe            if (isHalStarted()) {
1621918b4bdc2ae16438c5213320daeff1d551f4aed8Ningyuan Wang                int num = getInterfacesNative();
1622918b4bdc2ae16438c5213320daeff1d551f4aed8Ningyuan Wang                for (int i = 0; i < num; i++) {
1623918b4bdc2ae16438c5213320daeff1d551f4aed8Ningyuan Wang                    String name = getInterfaceNameNative(i);
1624918b4bdc2ae16438c5213320daeff1d551f4aed8Ningyuan Wang                    if (name.equals(interfaceName)) {
1625918b4bdc2ae16438c5213320daeff1d551f4aed8Ningyuan Wang                        return i;
162602a1f98f2cecb8ae2d466d6f9fab06b473f970ddVinit Deshpande                    }
1627aabe7a9f2f32915fd1f25416a6d2034a844005d6vandwalle                }
1628e4e3750390bec0a849a9153348b7c21b2cc8b843Vinit Deshapnde            }
1629e4e3750390bec0a849a9153348b7c21b2cc8b843Vinit Deshapnde        }
1630918b4bdc2ae16438c5213320daeff1d551f4aed8Ningyuan Wang        return -1;
16317f9a15d554f69311a0db43347d6473a7c4c46e2eVinit Deshapnde    }
16327f9a15d554f69311a0db43347d6473a7c4c46e2eVinit Deshapnde
1633b0b1d59786de7ff1fc98e31a16a7d710458bf549Vinit Deshpande    private static native String getInterfaceNameNative(int index);
163418786eca942042388748b0d98979f21c9dff4a89Mitchell Wills    public String getInterfaceName(int index) {
1635956f54b391677d78379729dd14518edddf3c7660Etan Cohen        synchronized (sLock) {
163618786eca942042388748b0d98979f21c9dff4a89Mitchell Wills            return getInterfaceNameNative(index);
163718786eca942042388748b0d98979f21c9dff4a89Mitchell Wills        }
16387f9a15d554f69311a0db43347d6473a7c4c46e2eVinit Deshapnde    }
16397f9a15d554f69311a0db43347d6473a7c4c46e2eVinit Deshapnde
1640062e3f39e37874fedc01f267de5f4cf7dbebe2b4Randy Pan    // TODO: Change variable names to camel style.
1641e4e3750390bec0a849a9153348b7c21b2cc8b843Vinit Deshapnde    public static class ScanCapabilities {
1642297c3acabe7a85eb87240fe3ccf772e57ce6aef7Mitchell Wills        public int  max_scan_cache_size;
1643e4e3750390bec0a849a9153348b7c21b2cc8b843Vinit Deshapnde        public int  max_scan_buckets;
1644e4e3750390bec0a849a9153348b7c21b2cc8b843Vinit Deshapnde        public int  max_ap_cache_per_scan;
1645e4e3750390bec0a849a9153348b7c21b2cc8b843Vinit Deshapnde        public int  max_rssi_sample_size;
1646297c3acabe7a85eb87240fe3ccf772e57ce6aef7Mitchell Wills        public int  max_scan_reporting_threshold;
16477d6301ead19afdf3de37455e9ed133c25b4938cdVinit Deshpande        public int  max_hotlist_bssids;
1648e4e3750390bec0a849a9153348b7c21b2cc8b843Vinit Deshapnde        public int  max_significant_wifi_change_aps;
1649d34e3a7b98d51f684100483151fceb233fd95215Roshan Pius        public int  max_bssid_history_entries;
1650d34e3a7b98d51f684100483151fceb233fd95215Roshan Pius        public int  max_number_epno_networks;
1651d34e3a7b98d51f684100483151fceb233fd95215Roshan Pius        public int  max_number_epno_networks_by_ssid;
1652d34e3a7b98d51f684100483151fceb233fd95215Roshan Pius        public int  max_number_of_white_listed_ssid;
1653e4e3750390bec0a849a9153348b7c21b2cc8b843Vinit Deshapnde    }
1654e4e3750390bec0a849a9153348b7c21b2cc8b843Vinit Deshapnde
165518786eca942042388748b0d98979f21c9dff4a89Mitchell Wills    public boolean getScanCapabilities(ScanCapabilities capabilities) {
1656956f54b391677d78379729dd14518edddf3c7660Etan Cohen        synchronized (sLock) {
1657af5b49884f189bb171c9dc6c6a4405d97e7912acVinit Deshpande            return isHalStarted() && getScanCapabilitiesNative(sWlan0Index, capabilities);
1658af5b49884f189bb171c9dc6c6a4405d97e7912acVinit Deshpande        }
1659e4e3750390bec0a849a9153348b7c21b2cc8b843Vinit Deshapnde    }
1660e4e3750390bec0a849a9153348b7c21b2cc8b843Vinit Deshapnde
1661b0b1d59786de7ff1fc98e31a16a7d710458bf549Vinit Deshpande    private static native boolean getScanCapabilitiesNative(
1662b0b1d59786de7ff1fc98e31a16a7d710458bf549Vinit Deshpande            int iface, ScanCapabilities capabilities);
1663e4e3750390bec0a849a9153348b7c21b2cc8b843Vinit Deshapnde
1664b0b1d59786de7ff1fc98e31a16a7d710458bf549Vinit Deshpande    private static native boolean startScanNative(int iface, int id, ScanSettings settings);
1665b0b1d59786de7ff1fc98e31a16a7d710458bf549Vinit Deshpande    private static native boolean stopScanNative(int iface, int id);
166683a674a18c84ff4f01377bbfd8988699dec93bc2Vinit Deshpande    private static native WifiScanner.ScanData[] getScanResultsNative(int iface, boolean flush);
1667b0b1d59786de7ff1fc98e31a16a7d710458bf549Vinit Deshpande    private static native WifiLinkLayerStats getWifiLinkLayerStatsNative(int iface);
1668d745a52dc4f929d4a4030f205ed173bdf60eaf10Pierre Vandwalle    private static native void setWifiLinkLayerStatsNative(int iface, int enable);
16697f9a15d554f69311a0db43347d6473a7c4c46e2eVinit Deshapnde
1670e4e3750390bec0a849a9153348b7c21b2cc8b843Vinit Deshapnde    public static class ChannelSettings {
1671712ef6246834caeac3d5b06bea08e85d6b29cd7aMitchell Wills        public int frequency;
1672712ef6246834caeac3d5b06bea08e85d6b29cd7aMitchell Wills        public int dwell_time_ms;
1673712ef6246834caeac3d5b06bea08e85d6b29cd7aMitchell Wills        public boolean passive;
16747f9a15d554f69311a0db43347d6473a7c4c46e2eVinit Deshapnde    }
16757f9a15d554f69311a0db43347d6473a7c4c46e2eVinit Deshapnde
1676e4e3750390bec0a849a9153348b7c21b2cc8b843Vinit Deshapnde    public static class BucketSettings {
1677712ef6246834caeac3d5b06bea08e85d6b29cd7aMitchell Wills        public int bucket;
1678712ef6246834caeac3d5b06bea08e85d6b29cd7aMitchell Wills        public int band;
1679712ef6246834caeac3d5b06bea08e85d6b29cd7aMitchell Wills        public int period_ms;
1680712ef6246834caeac3d5b06bea08e85d6b29cd7aMitchell Wills        public int max_period_ms;
1681712ef6246834caeac3d5b06bea08e85d6b29cd7aMitchell Wills        public int step_count;
1682712ef6246834caeac3d5b06bea08e85d6b29cd7aMitchell Wills        public int report_events;
1683712ef6246834caeac3d5b06bea08e85d6b29cd7aMitchell Wills        public int num_channels;
1684712ef6246834caeac3d5b06bea08e85d6b29cd7aMitchell Wills        public ChannelSettings[] channels;
1685e4e3750390bec0a849a9153348b7c21b2cc8b843Vinit Deshapnde    }
16867f9a15d554f69311a0db43347d6473a7c4c46e2eVinit Deshapnde
1687e4e3750390bec0a849a9153348b7c21b2cc8b843Vinit Deshapnde    public static class ScanSettings {
1688712ef6246834caeac3d5b06bea08e85d6b29cd7aMitchell Wills        public int base_period_ms;
1689712ef6246834caeac3d5b06bea08e85d6b29cd7aMitchell Wills        public int max_ap_per_scan;
1690712ef6246834caeac3d5b06bea08e85d6b29cd7aMitchell Wills        public int report_threshold_percent;
1691712ef6246834caeac3d5b06bea08e85d6b29cd7aMitchell Wills        public int report_threshold_num_scans;
1692712ef6246834caeac3d5b06bea08e85d6b29cd7aMitchell Wills        public int num_buckets;
1693d74ff28cdb01d3f711e0b0f0e99e51fdb221eaf2Roshan Pius        /* Not part of gscan HAL API. Used only for wpa_supplicant scanning */
1694d74ff28cdb01d3f711e0b0f0e99e51fdb221eaf2Roshan Pius        public int[] hiddenNetworkIds;
1695712ef6246834caeac3d5b06bea08e85d6b29cd7aMitchell Wills        public BucketSettings[] buckets;
1696e4e3750390bec0a849a9153348b7c21b2cc8b843Vinit Deshapnde    }
16977f9a15d554f69311a0db43347d6473a7c4c46e2eVinit Deshapnde
169868cb8c02b4ad079c54a2ffd4407da921d18c7af9Wei Wang    /**
16999bd01d6cb65ea96d6729d35a1fc43bb2bb40e3bdRoshan Pius     * Network parameters to start PNO scan.
17009bd01d6cb65ea96d6729d35a1fc43bb2bb40e3bdRoshan Pius     */
17019bd01d6cb65ea96d6729d35a1fc43bb2bb40e3bdRoshan Pius    public static class PnoNetwork {
17029bd01d6cb65ea96d6729d35a1fc43bb2bb40e3bdRoshan Pius        public String ssid;
17039bd01d6cb65ea96d6729d35a1fc43bb2bb40e3bdRoshan Pius        public int networkId;
17049bd01d6cb65ea96d6729d35a1fc43bb2bb40e3bdRoshan Pius        public int priority;
17059bd01d6cb65ea96d6729d35a1fc43bb2bb40e3bdRoshan Pius        public byte flags;
1706ef3ea1092bc17673c0a85a845b053151b7c10e07Roshan Pius        public byte auth_bit_field;
17079bd01d6cb65ea96d6729d35a1fc43bb2bb40e3bdRoshan Pius    }
17089bd01d6cb65ea96d6729d35a1fc43bb2bb40e3bdRoshan Pius
17099bd01d6cb65ea96d6729d35a1fc43bb2bb40e3bdRoshan Pius    /**
17109bd01d6cb65ea96d6729d35a1fc43bb2bb40e3bdRoshan Pius     * Parameters to start PNO scan. This holds the list of networks which are going to used for
17119bd01d6cb65ea96d6729d35a1fc43bb2bb40e3bdRoshan Pius     * PNO scan.
17129bd01d6cb65ea96d6729d35a1fc43bb2bb40e3bdRoshan Pius     */
17139bd01d6cb65ea96d6729d35a1fc43bb2bb40e3bdRoshan Pius    public static class PnoSettings {
17149bd01d6cb65ea96d6729d35a1fc43bb2bb40e3bdRoshan Pius        public int min5GHzRssi;
17159bd01d6cb65ea96d6729d35a1fc43bb2bb40e3bdRoshan Pius        public int min24GHzRssi;
17169bd01d6cb65ea96d6729d35a1fc43bb2bb40e3bdRoshan Pius        public int initialScoreMax;
17179bd01d6cb65ea96d6729d35a1fc43bb2bb40e3bdRoshan Pius        public int currentConnectionBonus;
17189bd01d6cb65ea96d6729d35a1fc43bb2bb40e3bdRoshan Pius        public int sameNetworkBonus;
17199bd01d6cb65ea96d6729d35a1fc43bb2bb40e3bdRoshan Pius        public int secureBonus;
17209bd01d6cb65ea96d6729d35a1fc43bb2bb40e3bdRoshan Pius        public int band5GHzBonus;
1721dcd877d6c143db557884993ea437e2a432cb0ba3Roshan Pius        public boolean isConnected;
17229bd01d6cb65ea96d6729d35a1fc43bb2bb40e3bdRoshan Pius        public PnoNetwork[] networkList;
17239bd01d6cb65ea96d6729d35a1fc43bb2bb40e3bdRoshan Pius    }
17249bd01d6cb65ea96d6729d35a1fc43bb2bb40e3bdRoshan Pius
17259bd01d6cb65ea96d6729d35a1fc43bb2bb40e3bdRoshan Pius    /**
172668cb8c02b4ad079c54a2ffd4407da921d18c7af9Wei Wang     * Wi-Fi channel information.
172768cb8c02b4ad079c54a2ffd4407da921d18c7af9Wei Wang     */
172868cb8c02b4ad079c54a2ffd4407da921d18c7af9Wei Wang    public static class WifiChannelInfo {
172968cb8c02b4ad079c54a2ffd4407da921d18c7af9Wei Wang        int mPrimaryFrequency;
173068cb8c02b4ad079c54a2ffd4407da921d18c7af9Wei Wang        int mCenterFrequency0;
173168cb8c02b4ad079c54a2ffd4407da921d18c7af9Wei Wang        int mCenterFrequency1;
173268cb8c02b4ad079c54a2ffd4407da921d18c7af9Wei Wang        int mChannelWidth;
173368cb8c02b4ad079c54a2ffd4407da921d18c7af9Wei Wang        // TODO: add preamble once available in HAL.
173468cb8c02b4ad079c54a2ffd4407da921d18c7af9Wei Wang    }
173568cb8c02b4ad079c54a2ffd4407da921d18c7af9Wei Wang
1736b0b1d59786de7ff1fc98e31a16a7d710458bf549Vinit Deshpande    public static interface ScanEventHandler {
173763539f1283899fbbf83ab90757961b4be51d5034Mitchell Wills        /**
173863539f1283899fbbf83ab90757961b4be51d5034Mitchell Wills         * Called for each AP as it is found with the entire contents of the beacon/probe response.
173963539f1283899fbbf83ab90757961b4be51d5034Mitchell Wills         * Only called when WifiScanner.REPORT_EVENT_FULL_SCAN_RESULT is specified.
174063539f1283899fbbf83ab90757961b4be51d5034Mitchell Wills         */
1741c9e6069eb941d282af213dc20b171877db6b567bMitchell Wills        void onFullScanResult(ScanResult fullScanResult, int bucketsScanned);
174263539f1283899fbbf83ab90757961b4be51d5034Mitchell Wills        /**
174363539f1283899fbbf83ab90757961b4be51d5034Mitchell Wills         * Callback on an event during a gscan scan.
174463539f1283899fbbf83ab90757961b4be51d5034Mitchell Wills         * See WifiNative.WIFI_SCAN_* for possible values.
174563539f1283899fbbf83ab90757961b4be51d5034Mitchell Wills         */
174663539f1283899fbbf83ab90757961b4be51d5034Mitchell Wills        void onScanStatus(int event);
174763539f1283899fbbf83ab90757961b4be51d5034Mitchell Wills        /**
174863539f1283899fbbf83ab90757961b4be51d5034Mitchell Wills         * Called with the current cached scan results when gscan is paused.
174963539f1283899fbbf83ab90757961b4be51d5034Mitchell Wills         */
175083a674a18c84ff4f01377bbfd8988699dec93bc2Vinit Deshpande        void onScanPaused(WifiScanner.ScanData[] data);
175163539f1283899fbbf83ab90757961b4be51d5034Mitchell Wills        /**
175263539f1283899fbbf83ab90757961b4be51d5034Mitchell Wills         * Called with the current cached scan results when gscan is resumed.
175363539f1283899fbbf83ab90757961b4be51d5034Mitchell Wills         */
1754b0b1d59786de7ff1fc98e31a16a7d710458bf549Vinit Deshpande        void onScanRestarted();
1755e4e3750390bec0a849a9153348b7c21b2cc8b843Vinit Deshapnde    }
1756e4e3750390bec0a849a9153348b7c21b2cc8b843Vinit Deshapnde
17579bd01d6cb65ea96d6729d35a1fc43bb2bb40e3bdRoshan Pius    /**
17589bd01d6cb65ea96d6729d35a1fc43bb2bb40e3bdRoshan Pius     * Handler to notify the occurrence of various events during PNO scan.
17599bd01d6cb65ea96d6729d35a1fc43bb2bb40e3bdRoshan Pius     */
17609bd01d6cb65ea96d6729d35a1fc43bb2bb40e3bdRoshan Pius    public interface PnoEventHandler {
17619bd01d6cb65ea96d6729d35a1fc43bb2bb40e3bdRoshan Pius        /**
17629bd01d6cb65ea96d6729d35a1fc43bb2bb40e3bdRoshan Pius         * Callback to notify when one of the shortlisted networks is found during PNO scan.
17639bd01d6cb65ea96d6729d35a1fc43bb2bb40e3bdRoshan Pius         * @param results List of Scan results received.
17649bd01d6cb65ea96d6729d35a1fc43bb2bb40e3bdRoshan Pius         */
17659bd01d6cb65ea96d6729d35a1fc43bb2bb40e3bdRoshan Pius        void onPnoNetworkFound(ScanResult[] results);
1766063cfc7e3eef78fcbda24a66f0c473828b39c854Roshan Pius
1767063cfc7e3eef78fcbda24a66f0c473828b39c854Roshan Pius        /**
1768063cfc7e3eef78fcbda24a66f0c473828b39c854Roshan Pius         * Callback to notify when the PNO scan schedule fails.
1769063cfc7e3eef78fcbda24a66f0c473828b39c854Roshan Pius         */
1770063cfc7e3eef78fcbda24a66f0c473828b39c854Roshan Pius        void onPnoScanFailed();
17719bd01d6cb65ea96d6729d35a1fc43bb2bb40e3bdRoshan Pius    }
17729bd01d6cb65ea96d6729d35a1fc43bb2bb40e3bdRoshan Pius
1773b7cc309f06da8a18224057c21ba086f8550367d6Vinit Deshpande    /* scan status, keep these values in sync with gscan.h */
177471af8bb37fee5852ae458edabdef3b487d62a5b8Mitchell Wills    public static final int WIFI_SCAN_RESULTS_AVAILABLE = 0;
177571af8bb37fee5852ae458edabdef3b487d62a5b8Mitchell Wills    public static final int WIFI_SCAN_THRESHOLD_NUM_SCANS = 1;
177671af8bb37fee5852ae458edabdef3b487d62a5b8Mitchell Wills    public static final int WIFI_SCAN_THRESHOLD_PERCENT = 2;
177771af8bb37fee5852ae458edabdef3b487d62a5b8Mitchell Wills    public static final int WIFI_SCAN_FAILED = 3;
1778b7cc309f06da8a18224057c21ba086f8550367d6Vinit Deshpande
177918786eca942042388748b0d98979f21c9dff4a89Mitchell Wills    // Callback from native
17802a6d76f0899289cd3b96e3428f02076fdbc0363eMitchell Wills    private static void onScanStatus(int id, int event) {
178118786eca942042388748b0d98979f21c9dff4a89Mitchell Wills        ScanEventHandler handler = sScanEventHandler;
178263539f1283899fbbf83ab90757961b4be51d5034Mitchell Wills        if (handler != null) {
178363539f1283899fbbf83ab90757961b4be51d5034Mitchell Wills            handler.onScanStatus(event);
17842a6d76f0899289cd3b96e3428f02076fdbc0363eMitchell Wills        }
1785e4e3750390bec0a849a9153348b7c21b2cc8b843Vinit Deshapnde    }
1786e4e3750390bec0a849a9153348b7c21b2cc8b843Vinit Deshapnde
178718786eca942042388748b0d98979f21c9dff4a89Mitchell Wills    public static  WifiSsid createWifiSsid(byte[] rawSsid) {
17885cfd8d8b9f241dcad874125a1b5538ee0d6860fexinhe        String ssidHexString = String.valueOf(HexEncoding.encode(rawSsid));
17895cfd8d8b9f241dcad874125a1b5538ee0d6860fexinhe
17905cfd8d8b9f241dcad874125a1b5538ee0d6860fexinhe        if (ssidHexString == null) {
17915cfd8d8b9f241dcad874125a1b5538ee0d6860fexinhe            return null;
17925cfd8d8b9f241dcad874125a1b5538ee0d6860fexinhe        }
17935cfd8d8b9f241dcad874125a1b5538ee0d6860fexinhe
17945cfd8d8b9f241dcad874125a1b5538ee0d6860fexinhe        WifiSsid wifiSsid = WifiSsid.createFromHex(ssidHexString);
17955cfd8d8b9f241dcad874125a1b5538ee0d6860fexinhe
17965cfd8d8b9f241dcad874125a1b5538ee0d6860fexinhe        return wifiSsid;
17975cfd8d8b9f241dcad874125a1b5538ee0d6860fexinhe    }
17985cfd8d8b9f241dcad874125a1b5538ee0d6860fexinhe
17995cfd8d8b9f241dcad874125a1b5538ee0d6860fexinhe    public static String ssidConvert(byte[] rawSsid) {
18005cfd8d8b9f241dcad874125a1b5538ee0d6860fexinhe        String ssid;
18015cfd8d8b9f241dcad874125a1b5538ee0d6860fexinhe
18025cfd8d8b9f241dcad874125a1b5538ee0d6860fexinhe        CharsetDecoder decoder = StandardCharsets.UTF_8.newDecoder();
18035cfd8d8b9f241dcad874125a1b5538ee0d6860fexinhe        try {
18045cfd8d8b9f241dcad874125a1b5538ee0d6860fexinhe            CharBuffer decoded = decoder.decode(ByteBuffer.wrap(rawSsid));
18055cfd8d8b9f241dcad874125a1b5538ee0d6860fexinhe            ssid = decoded.toString();
18065cfd8d8b9f241dcad874125a1b5538ee0d6860fexinhe        } catch (CharacterCodingException cce) {
18075cfd8d8b9f241dcad874125a1b5538ee0d6860fexinhe            ssid = null;
18085cfd8d8b9f241dcad874125a1b5538ee0d6860fexinhe        }
18095cfd8d8b9f241dcad874125a1b5538ee0d6860fexinhe
18105cfd8d8b9f241dcad874125a1b5538ee0d6860fexinhe        if (ssid == null) {
18115cfd8d8b9f241dcad874125a1b5538ee0d6860fexinhe            ssid = new String(rawSsid, StandardCharsets.ISO_8859_1);
18125cfd8d8b9f241dcad874125a1b5538ee0d6860fexinhe        }
18135cfd8d8b9f241dcad874125a1b5538ee0d6860fexinhe
18145cfd8d8b9f241dcad874125a1b5538ee0d6860fexinhe        return ssid;
18155cfd8d8b9f241dcad874125a1b5538ee0d6860fexinhe    }
18165cfd8d8b9f241dcad874125a1b5538ee0d6860fexinhe
181718786eca942042388748b0d98979f21c9dff4a89Mitchell Wills    // Called from native
18185cfd8d8b9f241dcad874125a1b5538ee0d6860fexinhe    public static boolean setSsid(byte[] rawSsid, ScanResult result) {
18195cfd8d8b9f241dcad874125a1b5538ee0d6860fexinhe        if (rawSsid == null || rawSsid.length == 0 || result == null) {
18205cfd8d8b9f241dcad874125a1b5538ee0d6860fexinhe            return false;
18215cfd8d8b9f241dcad874125a1b5538ee0d6860fexinhe        }
18225cfd8d8b9f241dcad874125a1b5538ee0d6860fexinhe
18235cfd8d8b9f241dcad874125a1b5538ee0d6860fexinhe        result.SSID = ssidConvert(rawSsid);
18245cfd8d8b9f241dcad874125a1b5538ee0d6860fexinhe        result.wifiSsid = createWifiSsid(rawSsid);
18255cfd8d8b9f241dcad874125a1b5538ee0d6860fexinhe        return true;
18265cfd8d8b9f241dcad874125a1b5538ee0d6860fexinhe    }
18275cfd8d8b9f241dcad874125a1b5538ee0d6860fexinhe
18281ab129e587d334a144a0bca5323c27985397a403Randy Pan    private static void populateScanResult(ScanResult result, int beaconCap, String dbg) {
1829dd490cf1df37b70a8625dc4ec8e712f740dd1e4aPierre Vandwalle        if (dbg == null) dbg = "";
18305d31cedf4024e0f038b4dfc2081016c8631ee8feMitchell Wills
18315d31cedf4024e0f038b4dfc2081016c8631ee8feMitchell Wills        InformationElementUtil.HtOperation htOperation = new InformationElementUtil.HtOperation();
18325d31cedf4024e0f038b4dfc2081016c8631ee8feMitchell Wills        InformationElementUtil.VhtOperation vhtOperation =
18335d31cedf4024e0f038b4dfc2081016c8631ee8feMitchell Wills                new InformationElementUtil.VhtOperation();
18345d31cedf4024e0f038b4dfc2081016c8631ee8feMitchell Wills        InformationElementUtil.ExtendedCapabilities extendedCaps =
18355d31cedf4024e0f038b4dfc2081016c8631ee8feMitchell Wills                new InformationElementUtil.ExtendedCapabilities();
18365d31cedf4024e0f038b4dfc2081016c8631ee8feMitchell Wills
18375d31cedf4024e0f038b4dfc2081016c8631ee8feMitchell Wills        ScanResult.InformationElement elements[] =
18381ab129e587d334a144a0bca5323c27985397a403Randy Pan                InformationElementUtil.parseInformationElements(result.bytes);
18395d31cedf4024e0f038b4dfc2081016c8631ee8feMitchell Wills        for (ScanResult.InformationElement ie : elements) {
18405d31cedf4024e0f038b4dfc2081016c8631ee8feMitchell Wills            if(ie.id == ScanResult.InformationElement.EID_HT_OPERATION) {
18415d31cedf4024e0f038b4dfc2081016c8631ee8feMitchell Wills                htOperation.from(ie);
18425d31cedf4024e0f038b4dfc2081016c8631ee8feMitchell Wills            } else if(ie.id == ScanResult.InformationElement.EID_VHT_OPERATION) {
18435d31cedf4024e0f038b4dfc2081016c8631ee8feMitchell Wills                vhtOperation.from(ie);
18445d31cedf4024e0f038b4dfc2081016c8631ee8feMitchell Wills            } else if (ie.id == ScanResult.InformationElement.EID_EXTENDED_CAPS) {
18455d31cedf4024e0f038b4dfc2081016c8631ee8feMitchell Wills                extendedCaps.from(ie);
1846243931f3474f6235cfcf5c1a55fa2f192ee264aexinhe            }
1847243931f3474f6235cfcf5c1a55fa2f192ee264aexinhe        }
1848d9f37b2f3df18e96246db93ec4c2a5159b5d3915Vinit Deshpande
18495d31cedf4024e0f038b4dfc2081016c8631ee8feMitchell Wills        if (extendedCaps.is80211McRTTResponder) {
1850d9f37b2f3df18e96246db93ec4c2a5159b5d3915Vinit Deshpande            result.setFlag(ScanResult.FLAG_80211mc_RESPONDER);
1851d9f37b2f3df18e96246db93ec4c2a5159b5d3915Vinit Deshpande        } else {
1852d9f37b2f3df18e96246db93ec4c2a5159b5d3915Vinit Deshpande            result.clearFlag(ScanResult.FLAG_80211mc_RESPONDER);
1853d9f37b2f3df18e96246db93ec4c2a5159b5d3915Vinit Deshpande        }
1854d9f37b2f3df18e96246db93ec4c2a5159b5d3915Vinit Deshpande
1855243931f3474f6235cfcf5c1a55fa2f192ee264aexinhe        //handle RTT related information
18565d31cedf4024e0f038b4dfc2081016c8631ee8feMitchell Wills        if (vhtOperation.isValid()) {
18575d31cedf4024e0f038b4dfc2081016c8631ee8feMitchell Wills            result.channelWidth = vhtOperation.getChannelWidth();
18585d31cedf4024e0f038b4dfc2081016c8631ee8feMitchell Wills            result.centerFreq0 = vhtOperation.getCenterFreq0();
18595d31cedf4024e0f038b4dfc2081016c8631ee8feMitchell Wills            result.centerFreq1 = vhtOperation.getCenterFreq1();
1860243931f3474f6235cfcf5c1a55fa2f192ee264aexinhe        } else {
18615d31cedf4024e0f038b4dfc2081016c8631ee8feMitchell Wills            result.channelWidth = htOperation.getChannelWidth();
18625d31cedf4024e0f038b4dfc2081016c8631ee8feMitchell Wills            result.centerFreq0 = htOperation.getCenterFreq0(result.frequency);
1863243931f3474f6235cfcf5c1a55fa2f192ee264aexinhe            result.centerFreq1  = 0;
1864243931f3474f6235cfcf5c1a55fa2f192ee264aexinhe        }
18653571366ac36c70746b9f013ec2b54482861c9292Randy Pan
18663571366ac36c70746b9f013ec2b54482861c9292Randy Pan        // build capabilities string
18673571366ac36c70746b9f013ec2b54482861c9292Randy Pan        BitSet beaconCapBits = new BitSet(16);
18683571366ac36c70746b9f013ec2b54482861c9292Randy Pan        for (int i = 0; i < 16; i++) {
18693571366ac36c70746b9f013ec2b54482861c9292Randy Pan            if ((beaconCap & (1 << i)) != 0) {
18703571366ac36c70746b9f013ec2b54482861c9292Randy Pan                beaconCapBits.set(i);
18713571366ac36c70746b9f013ec2b54482861c9292Randy Pan            }
18723571366ac36c70746b9f013ec2b54482861c9292Randy Pan        }
18733571366ac36c70746b9f013ec2b54482861c9292Randy Pan        result.capabilities = InformationElementUtil.Capabilities.buildCapabilities(elements,
18743571366ac36c70746b9f013ec2b54482861c9292Randy Pan                                               beaconCapBits);
18753571366ac36c70746b9f013ec2b54482861c9292Randy Pan
1876243931f3474f6235cfcf5c1a55fa2f192ee264aexinhe        if(DBG) {
18773571366ac36c70746b9f013ec2b54482861c9292Randy Pan            Log.d(TAG, dbg + "SSID: " + result.SSID + " ChannelWidth is: " + result.channelWidth
18783571366ac36c70746b9f013ec2b54482861c9292Randy Pan                    + " PrimaryFreq: " + result.frequency + " mCenterfreq0: " + result.centerFreq0
18793571366ac36c70746b9f013ec2b54482861c9292Randy Pan                    + " mCenterfreq1: " + result.centerFreq1 + (extendedCaps.is80211McRTTResponder
18803571366ac36c70746b9f013ec2b54482861c9292Randy Pan                    ? "Support RTT reponder: " : "Do not support RTT responder")
18813571366ac36c70746b9f013ec2b54482861c9292Randy Pan                    + " Capabilities: " + result.capabilities);
1882f1daf9342b66bf134d13fa0a42e929a008f1ca62Vinit Deshapnde        }
1883f1daf9342b66bf134d13fa0a42e929a008f1ca62Vinit Deshapnde
1884476bee2fef10d060c25c35858b1f7f60803d9f49Vinit Deshpande        result.informationElements = elements;
1885dd490cf1df37b70a8625dc4ec8e712f740dd1e4aPierre Vandwalle    }
1886dd490cf1df37b70a8625dc4ec8e712f740dd1e4aPierre Vandwalle
188718786eca942042388748b0d98979f21c9dff4a89Mitchell Wills    // Callback from native
18881ab129e587d334a144a0bca5323c27985397a403Randy Pan    private static void onFullScanResult(int id, ScanResult result,
18893571366ac36c70746b9f013ec2b54482861c9292Randy Pan            int bucketsScanned, int beaconCap) {
18901ab129e587d334a144a0bca5323c27985397a403Randy Pan        if (DBG) Log.i(TAG, "Got a full scan results event, ssid = " + result.SSID);
1891dd490cf1df37b70a8625dc4ec8e712f740dd1e4aPierre Vandwalle
189218786eca942042388748b0d98979f21c9dff4a89Mitchell Wills        ScanEventHandler handler = sScanEventHandler;
189318786eca942042388748b0d98979f21c9dff4a89Mitchell Wills        if (handler != null) {
18941ab129e587d334a144a0bca5323c27985397a403Randy Pan            populateScanResult(result, beaconCap, " onFullScanResult ");
1895c9e6069eb941d282af213dc20b171877db6b567bMitchell Wills            handler.onFullScanResult(result, bucketsScanned);
1896dd490cf1df37b70a8625dc4ec8e712f740dd1e4aPierre Vandwalle        }
18977f9a15d554f69311a0db43347d6473a7c4c46e2eVinit Deshapnde    }
18987f9a15d554f69311a0db43347d6473a7c4c46e2eVinit Deshapnde
1899b0b1d59786de7ff1fc98e31a16a7d710458bf549Vinit Deshpande    private static int sScanCmdId = 0;
1900b0b1d59786de7ff1fc98e31a16a7d710458bf549Vinit Deshpande    private static ScanEventHandler sScanEventHandler;
1901b0b1d59786de7ff1fc98e31a16a7d710458bf549Vinit Deshpande    private static ScanSettings sScanSettings;
19027f9a15d554f69311a0db43347d6473a7c4c46e2eVinit Deshapnde
190318786eca942042388748b0d98979f21c9dff4a89Mitchell Wills    public boolean startScan(ScanSettings settings, ScanEventHandler eventHandler) {
1904956f54b391677d78379729dd14518edddf3c7660Etan Cohen        synchronized (sLock) {
190571d0be16c79791a005d60f9d5fa7d2d81e1f8c80xinhe            if (isHalStarted()) {
1906f49a59bda006b13e0118d144e0a4a5f569b2251eVinit Deshpande                if (sScanCmdId != 0) {
1907f49a59bda006b13e0118d144e0a4a5f569b2251eVinit Deshpande                    stopScan();
1908f49a59bda006b13e0118d144e0a4a5f569b2251eVinit Deshpande                } else if (sScanSettings != null || sScanEventHandler != null) {
1909b0b1d59786de7ff1fc98e31a16a7d710458bf549Vinit Deshpande                /* current scan is paused; no need to stop it */
1910f49a59bda006b13e0118d144e0a4a5f569b2251eVinit Deshpande                }
19117f9a15d554f69311a0db43347d6473a7c4c46e2eVinit Deshapnde
1912f49a59bda006b13e0118d144e0a4a5f569b2251eVinit Deshpande                sScanCmdId = getNewCmdIdLocked();
1913e4e3750390bec0a849a9153348b7c21b2cc8b843Vinit Deshapnde
1914f49a59bda006b13e0118d144e0a4a5f569b2251eVinit Deshpande                sScanSettings = settings;
1915f49a59bda006b13e0118d144e0a4a5f569b2251eVinit Deshpande                sScanEventHandler = eventHandler;
1916b0b1d59786de7ff1fc98e31a16a7d710458bf549Vinit Deshpande
1917f49a59bda006b13e0118d144e0a4a5f569b2251eVinit Deshpande                if (startScanNative(sWlan0Index, sScanCmdId, settings) == false) {
1918f49a59bda006b13e0118d144e0a4a5f569b2251eVinit Deshpande                    sScanEventHandler = null;
1919f49a59bda006b13e0118d144e0a4a5f569b2251eVinit Deshpande                    sScanSettings = null;
1920f49a59bda006b13e0118d144e0a4a5f569b2251eVinit Deshpande                    sScanCmdId = 0;
1921f49a59bda006b13e0118d144e0a4a5f569b2251eVinit Deshpande                    return false;
1922f49a59bda006b13e0118d144e0a4a5f569b2251eVinit Deshpande                }
1923f49a59bda006b13e0118d144e0a4a5f569b2251eVinit Deshpande
1924f49a59bda006b13e0118d144e0a4a5f569b2251eVinit Deshpande                return true;
1925f49a59bda006b13e0118d144e0a4a5f569b2251eVinit Deshpande            } else {
1926e4e3750390bec0a849a9153348b7c21b2cc8b843Vinit Deshapnde                return false;
1927e4e3750390bec0a849a9153348b7c21b2cc8b843Vinit Deshapnde            }
1928e4e3750390bec0a849a9153348b7c21b2cc8b843Vinit Deshapnde        }
19297f9a15d554f69311a0db43347d6473a7c4c46e2eVinit Deshapnde    }
19307f9a15d554f69311a0db43347d6473a7c4c46e2eVinit Deshapnde
193118786eca942042388748b0d98979f21c9dff4a89Mitchell Wills    public void stopScan() {
1932956f54b391677d78379729dd14518edddf3c7660Etan Cohen        synchronized (sLock) {
193371d0be16c79791a005d60f9d5fa7d2d81e1f8c80xinhe            if (isHalStarted()) {
1934ed34692d4c4e202748d41f90db397f1cc3dc8320Mitchell Wills                if (sScanCmdId != 0) {
1935ed34692d4c4e202748d41f90db397f1cc3dc8320Mitchell Wills                    stopScanNative(sWlan0Index, sScanCmdId);
1936ed34692d4c4e202748d41f90db397f1cc3dc8320Mitchell Wills                }
1937f49a59bda006b13e0118d144e0a4a5f569b2251eVinit Deshpande                sScanSettings = null;
1938f49a59bda006b13e0118d144e0a4a5f569b2251eVinit Deshpande                sScanEventHandler = null;
1939f49a59bda006b13e0118d144e0a4a5f569b2251eVinit Deshpande                sScanCmdId = 0;
1940f49a59bda006b13e0118d144e0a4a5f569b2251eVinit Deshpande            }
1941b0b1d59786de7ff1fc98e31a16a7d710458bf549Vinit Deshpande        }
1942b0b1d59786de7ff1fc98e31a16a7d710458bf549Vinit Deshpande    }
1943b0b1d59786de7ff1fc98e31a16a7d710458bf549Vinit Deshpande
194418786eca942042388748b0d98979f21c9dff4a89Mitchell Wills    public void pauseScan() {
1945956f54b391677d78379729dd14518edddf3c7660Etan Cohen        synchronized (sLock) {
194671d0be16c79791a005d60f9d5fa7d2d81e1f8c80xinhe            if (isHalStarted()) {
1947f49a59bda006b13e0118d144e0a4a5f569b2251eVinit Deshpande                if (sScanCmdId != 0 && sScanSettings != null && sScanEventHandler != null) {
1948f49a59bda006b13e0118d144e0a4a5f569b2251eVinit Deshpande                    Log.d(TAG, "Pausing scan");
1949f49a59bda006b13e0118d144e0a4a5f569b2251eVinit Deshpande                    WifiScanner.ScanData scanData[] = getScanResultsNative(sWlan0Index, true);
1950f49a59bda006b13e0118d144e0a4a5f569b2251eVinit Deshpande                    stopScanNative(sWlan0Index, sScanCmdId);
1951f49a59bda006b13e0118d144e0a4a5f569b2251eVinit Deshpande                    sScanCmdId = 0;
1952f49a59bda006b13e0118d144e0a4a5f569b2251eVinit Deshpande                    sScanEventHandler.onScanPaused(scanData);
1953f49a59bda006b13e0118d144e0a4a5f569b2251eVinit Deshpande                }
1954b0b1d59786de7ff1fc98e31a16a7d710458bf549Vinit Deshpande            }
1955b0b1d59786de7ff1fc98e31a16a7d710458bf549Vinit Deshpande        }
1956b0b1d59786de7ff1fc98e31a16a7d710458bf549Vinit Deshpande    }
1957b0b1d59786de7ff1fc98e31a16a7d710458bf549Vinit Deshpande
195818786eca942042388748b0d98979f21c9dff4a89Mitchell Wills    public void restartScan() {
1959956f54b391677d78379729dd14518edddf3c7660Etan Cohen        synchronized (sLock) {
196071d0be16c79791a005d60f9d5fa7d2d81e1f8c80xinhe            if (isHalStarted()) {
1961f49a59bda006b13e0118d144e0a4a5f569b2251eVinit Deshpande                if (sScanCmdId == 0 && sScanSettings != null && sScanEventHandler != null) {
19626f3626faf50499dd95aa299bb1011c27ab05776dPierre Vandwalle                    Log.d(TAG, "Restarting scan");
1963f49a59bda006b13e0118d144e0a4a5f569b2251eVinit Deshpande                    ScanEventHandler handler = sScanEventHandler;
1964f49a59bda006b13e0118d144e0a4a5f569b2251eVinit Deshpande                    ScanSettings settings = sScanSettings;
1965f49a59bda006b13e0118d144e0a4a5f569b2251eVinit Deshpande                    if (startScan(sScanSettings, sScanEventHandler)) {
1966f49a59bda006b13e0118d144e0a4a5f569b2251eVinit Deshpande                        sScanEventHandler.onScanRestarted();
1967f49a59bda006b13e0118d144e0a4a5f569b2251eVinit Deshpande                    } else {
196883a674a18c84ff4f01377bbfd8988699dec93bc2Vinit Deshpande                    /* we are still paused; don't change state */
1969f49a59bda006b13e0118d144e0a4a5f569b2251eVinit Deshpande                        sScanEventHandler = handler;
1970f49a59bda006b13e0118d144e0a4a5f569b2251eVinit Deshpande                        sScanSettings = settings;
1971f49a59bda006b13e0118d144e0a4a5f569b2251eVinit Deshpande                    }
197283a674a18c84ff4f01377bbfd8988699dec93bc2Vinit Deshpande                }
1973b0b1d59786de7ff1fc98e31a16a7d710458bf549Vinit Deshpande            }
1974e4e3750390bec0a849a9153348b7c21b2cc8b843Vinit Deshapnde        }
1975e4e3750390bec0a849a9153348b7c21b2cc8b843Vinit Deshapnde    }
1976e4e3750390bec0a849a9153348b7c21b2cc8b843Vinit Deshapnde
197718786eca942042388748b0d98979f21c9dff4a89Mitchell Wills    public WifiScanner.ScanData[] getScanResults(boolean flush) {
1978956f54b391677d78379729dd14518edddf3c7660Etan Cohen        synchronized (sLock) {
19793ff269ca67e73f66ac22049fc318b2f86eafb253Vinit Deshpande            WifiScanner.ScanData[] sd = null;
198071d0be16c79791a005d60f9d5fa7d2d81e1f8c80xinhe            if (isHalStarted()) {
19813ff269ca67e73f66ac22049fc318b2f86eafb253Vinit Deshpande                sd = getScanResultsNative(sWlan0Index, flush);
19823ff269ca67e73f66ac22049fc318b2f86eafb253Vinit Deshpande            }
19833ff269ca67e73f66ac22049fc318b2f86eafb253Vinit Deshpande
19843ff269ca67e73f66ac22049fc318b2f86eafb253Vinit Deshpande            if (sd != null) {
19853ff269ca67e73f66ac22049fc318b2f86eafb253Vinit Deshpande                return sd;
1986f49a59bda006b13e0118d144e0a4a5f569b2251eVinit Deshpande            } else {
19873ff269ca67e73f66ac22049fc318b2f86eafb253Vinit Deshpande                return new WifiScanner.ScanData[0];
1988f49a59bda006b13e0118d144e0a4a5f569b2251eVinit Deshpande            }
1989aabe7a9f2f32915fd1f25416a6d2034a844005d6vandwalle        }
1990e4e3750390bec0a849a9153348b7c21b2cc8b843Vinit Deshapnde    }
1991e4e3750390bec0a849a9153348b7c21b2cc8b843Vinit Deshapnde
1992b0b1d59786de7ff1fc98e31a16a7d710458bf549Vinit Deshpande    public static interface HotlistEventHandler {
1993d4762401ec14be6bdd2d27aff2478ddbf8d6ce2aVinit Deshpande        void onHotlistApFound (ScanResult[] result);
1994d4762401ec14be6bdd2d27aff2478ddbf8d6ce2aVinit Deshpande        void onHotlistApLost  (ScanResult[] result);
1995e4e3750390bec0a849a9153348b7c21b2cc8b843Vinit Deshapnde    }
1996e4e3750390bec0a849a9153348b7c21b2cc8b843Vinit Deshapnde
1997b0b1d59786de7ff1fc98e31a16a7d710458bf549Vinit Deshpande    private static int sHotlistCmdId = 0;
1998b0b1d59786de7ff1fc98e31a16a7d710458bf549Vinit Deshpande    private static HotlistEventHandler sHotlistEventHandler;
1999e4e3750390bec0a849a9153348b7c21b2cc8b843Vinit Deshapnde
2000b0b1d59786de7ff1fc98e31a16a7d710458bf549Vinit Deshpande    private native static boolean setHotlistNative(int iface, int id,
2001e4e3750390bec0a849a9153348b7c21b2cc8b843Vinit Deshapnde            WifiScanner.HotlistSettings settings);
2002b0b1d59786de7ff1fc98e31a16a7d710458bf549Vinit Deshpande    private native static boolean resetHotlistNative(int iface, int id);
2003e4e3750390bec0a849a9153348b7c21b2cc8b843Vinit Deshapnde
200418786eca942042388748b0d98979f21c9dff4a89Mitchell Wills    public boolean setHotlist(WifiScanner.HotlistSettings settings,
200518786eca942042388748b0d98979f21c9dff4a89Mitchell Wills            HotlistEventHandler eventHandler) {
2006956f54b391677d78379729dd14518edddf3c7660Etan Cohen        synchronized (sLock) {
200771d0be16c79791a005d60f9d5fa7d2d81e1f8c80xinhe            if (isHalStarted()) {
2008f49a59bda006b13e0118d144e0a4a5f569b2251eVinit Deshpande                if (sHotlistCmdId != 0) {
2009f49a59bda006b13e0118d144e0a4a5f569b2251eVinit Deshpande                    return false;
2010f49a59bda006b13e0118d144e0a4a5f569b2251eVinit Deshpande                } else {
2011f49a59bda006b13e0118d144e0a4a5f569b2251eVinit Deshpande                    sHotlistCmdId = getNewCmdIdLocked();
2012f49a59bda006b13e0118d144e0a4a5f569b2251eVinit Deshpande                }
2013f49a59bda006b13e0118d144e0a4a5f569b2251eVinit Deshpande
2014f49a59bda006b13e0118d144e0a4a5f569b2251eVinit Deshpande                sHotlistEventHandler = eventHandler;
2015f49a59bda006b13e0118d144e0a4a5f569b2251eVinit Deshpande                if (setHotlistNative(sWlan0Index, sHotlistCmdId, settings) == false) {
2016f49a59bda006b13e0118d144e0a4a5f569b2251eVinit Deshpande                    sHotlistEventHandler = null;
2017f49a59bda006b13e0118d144e0a4a5f569b2251eVinit Deshpande                    return false;
2018f49a59bda006b13e0118d144e0a4a5f569b2251eVinit Deshpande                }
2019e4e3750390bec0a849a9153348b7c21b2cc8b843Vinit Deshapnde
2020f49a59bda006b13e0118d144e0a4a5f569b2251eVinit Deshpande                return true;
2021f49a59bda006b13e0118d144e0a4a5f569b2251eVinit Deshpande            } else {
2022e4e3750390bec0a849a9153348b7c21b2cc8b843Vinit Deshapnde                return false;
2023e4e3750390bec0a849a9153348b7c21b2cc8b843Vinit Deshapnde            }
2024e4e3750390bec0a849a9153348b7c21b2cc8b843Vinit Deshapnde        }
2025e4e3750390bec0a849a9153348b7c21b2cc8b843Vinit Deshapnde    }
2026e4e3750390bec0a849a9153348b7c21b2cc8b843Vinit Deshapnde
202718786eca942042388748b0d98979f21c9dff4a89Mitchell Wills    public void resetHotlist() {
2028956f54b391677d78379729dd14518edddf3c7660Etan Cohen        synchronized (sLock) {
202971d0be16c79791a005d60f9d5fa7d2d81e1f8c80xinhe            if (isHalStarted()) {
2030f49a59bda006b13e0118d144e0a4a5f569b2251eVinit Deshpande                if (sHotlistCmdId != 0) {
2031f49a59bda006b13e0118d144e0a4a5f569b2251eVinit Deshpande                    resetHotlistNative(sWlan0Index, sHotlistCmdId);
2032f49a59bda006b13e0118d144e0a4a5f569b2251eVinit Deshpande                    sHotlistCmdId = 0;
2033f49a59bda006b13e0118d144e0a4a5f569b2251eVinit Deshpande                    sHotlistEventHandler = null;
2034f49a59bda006b13e0118d144e0a4a5f569b2251eVinit Deshpande                }
2035e4e3750390bec0a849a9153348b7c21b2cc8b843Vinit Deshapnde            }
20367f9a15d554f69311a0db43347d6473a7c4c46e2eVinit Deshapnde        }
20377f9a15d554f69311a0db43347d6473a7c4c46e2eVinit Deshapnde    }
2038e4e3750390bec0a849a9153348b7c21b2cc8b843Vinit Deshapnde
203918786eca942042388748b0d98979f21c9dff4a89Mitchell Wills    // Callback from native
204018786eca942042388748b0d98979f21c9dff4a89Mitchell Wills    private static void onHotlistApFound(int id, ScanResult[] results) {
204118786eca942042388748b0d98979f21c9dff4a89Mitchell Wills        HotlistEventHandler handler = sHotlistEventHandler;
204218786eca942042388748b0d98979f21c9dff4a89Mitchell Wills        if (handler != null) {
204318786eca942042388748b0d98979f21c9dff4a89Mitchell Wills            handler.onHotlistApFound(results);
204418786eca942042388748b0d98979f21c9dff4a89Mitchell Wills        } else {
204518786eca942042388748b0d98979f21c9dff4a89Mitchell Wills            /* this can happen because of race conditions */
204618786eca942042388748b0d98979f21c9dff4a89Mitchell Wills            Log.d(TAG, "Ignoring hotlist AP found event");
2047d4762401ec14be6bdd2d27aff2478ddbf8d6ce2aVinit Deshpande        }
2048d4762401ec14be6bdd2d27aff2478ddbf8d6ce2aVinit Deshpande    }
2049d4762401ec14be6bdd2d27aff2478ddbf8d6ce2aVinit Deshpande
205018786eca942042388748b0d98979f21c9dff4a89Mitchell Wills    // Callback from native
205118786eca942042388748b0d98979f21c9dff4a89Mitchell Wills    private static void onHotlistApLost(int id, ScanResult[] results) {
205218786eca942042388748b0d98979f21c9dff4a89Mitchell Wills        HotlistEventHandler handler = sHotlistEventHandler;
205318786eca942042388748b0d98979f21c9dff4a89Mitchell Wills        if (handler != null) {
205418786eca942042388748b0d98979f21c9dff4a89Mitchell Wills            handler.onHotlistApLost(results);
205518786eca942042388748b0d98979f21c9dff4a89Mitchell Wills        } else {
205618786eca942042388748b0d98979f21c9dff4a89Mitchell Wills            /* this can happen because of race conditions */
205718786eca942042388748b0d98979f21c9dff4a89Mitchell Wills            Log.d(TAG, "Ignoring hotlist AP lost event");
2058aabe7a9f2f32915fd1f25416a6d2034a844005d6vandwalle        }
2059e4e3750390bec0a849a9153348b7c21b2cc8b843Vinit Deshapnde    }
2060e4e3750390bec0a849a9153348b7c21b2cc8b843Vinit Deshapnde
2061b0b1d59786de7ff1fc98e31a16a7d710458bf549Vinit Deshpande    public static interface SignificantWifiChangeEventHandler {
2062e4e3750390bec0a849a9153348b7c21b2cc8b843Vinit Deshapnde        void onChangesFound(ScanResult[] result);
2063e4e3750390bec0a849a9153348b7c21b2cc8b843Vinit Deshapnde    }
2064e4e3750390bec0a849a9153348b7c21b2cc8b843Vinit Deshapnde
2065b0b1d59786de7ff1fc98e31a16a7d710458bf549Vinit Deshpande    private static SignificantWifiChangeEventHandler sSignificantWifiChangeHandler;
2066b0b1d59786de7ff1fc98e31a16a7d710458bf549Vinit Deshpande    private static int sSignificantWifiChangeCmdId;
2067e4e3750390bec0a849a9153348b7c21b2cc8b843Vinit Deshapnde
2068b0b1d59786de7ff1fc98e31a16a7d710458bf549Vinit Deshpande    private static native boolean trackSignificantWifiChangeNative(
2069e4e3750390bec0a849a9153348b7c21b2cc8b843Vinit Deshapnde            int iface, int id, WifiScanner.WifiChangeSettings settings);
2070b0b1d59786de7ff1fc98e31a16a7d710458bf549Vinit Deshpande    private static native boolean untrackSignificantWifiChangeNative(int iface, int id);
2071e4e3750390bec0a849a9153348b7c21b2cc8b843Vinit Deshapnde
207218786eca942042388748b0d98979f21c9dff4a89Mitchell Wills    public boolean trackSignificantWifiChange(
2073b0b1d59786de7ff1fc98e31a16a7d710458bf549Vinit Deshpande            WifiScanner.WifiChangeSettings settings, SignificantWifiChangeEventHandler handler) {
2074956f54b391677d78379729dd14518edddf3c7660Etan Cohen        synchronized (sLock) {
207571d0be16c79791a005d60f9d5fa7d2d81e1f8c80xinhe            if (isHalStarted()) {
2076f49a59bda006b13e0118d144e0a4a5f569b2251eVinit Deshpande                if (sSignificantWifiChangeCmdId != 0) {
2077f49a59bda006b13e0118d144e0a4a5f569b2251eVinit Deshpande                    return false;
2078f49a59bda006b13e0118d144e0a4a5f569b2251eVinit Deshpande                } else {
2079f49a59bda006b13e0118d144e0a4a5f569b2251eVinit Deshpande                    sSignificantWifiChangeCmdId = getNewCmdIdLocked();
2080f49a59bda006b13e0118d144e0a4a5f569b2251eVinit Deshpande                }
2081f49a59bda006b13e0118d144e0a4a5f569b2251eVinit Deshpande
2082f49a59bda006b13e0118d144e0a4a5f569b2251eVinit Deshpande                sSignificantWifiChangeHandler = handler;
2083851075f4f345d98de885bd2f6b833944b282097eMitchell Wills                if (trackSignificantWifiChangeNative(sWlan0Index, sSignificantWifiChangeCmdId,
2084851075f4f345d98de885bd2f6b833944b282097eMitchell Wills                        settings) == false) {
2085f49a59bda006b13e0118d144e0a4a5f569b2251eVinit Deshpande                    sSignificantWifiChangeHandler = null;
2086f49a59bda006b13e0118d144e0a4a5f569b2251eVinit Deshpande                    return false;
2087f49a59bda006b13e0118d144e0a4a5f569b2251eVinit Deshpande                }
2088e4e3750390bec0a849a9153348b7c21b2cc8b843Vinit Deshapnde
2089f49a59bda006b13e0118d144e0a4a5f569b2251eVinit Deshpande                return true;
2090f49a59bda006b13e0118d144e0a4a5f569b2251eVinit Deshpande            } else {
2091e4e3750390bec0a849a9153348b7c21b2cc8b843Vinit Deshapnde                return false;
2092e4e3750390bec0a849a9153348b7c21b2cc8b843Vinit Deshapnde            }
2093e4e3750390bec0a849a9153348b7c21b2cc8b843Vinit Deshapnde
2094e4e3750390bec0a849a9153348b7c21b2cc8b843Vinit Deshapnde        }
2095e4e3750390bec0a849a9153348b7c21b2cc8b843Vinit Deshapnde    }
2096e4e3750390bec0a849a9153348b7c21b2cc8b843Vinit Deshapnde
209718786eca942042388748b0d98979f21c9dff4a89Mitchell Wills    public void untrackSignificantWifiChange() {
2098956f54b391677d78379729dd14518edddf3c7660Etan Cohen        synchronized (sLock) {
209971d0be16c79791a005d60f9d5fa7d2d81e1f8c80xinhe            if (isHalStarted()) {
2100f49a59bda006b13e0118d144e0a4a5f569b2251eVinit Deshpande                if (sSignificantWifiChangeCmdId != 0) {
2101f49a59bda006b13e0118d144e0a4a5f569b2251eVinit Deshpande                    untrackSignificantWifiChangeNative(sWlan0Index, sSignificantWifiChangeCmdId);
2102f49a59bda006b13e0118d144e0a4a5f569b2251eVinit Deshpande                    sSignificantWifiChangeCmdId = 0;
2103f49a59bda006b13e0118d144e0a4a5f569b2251eVinit Deshpande                    sSignificantWifiChangeHandler = null;
2104f49a59bda006b13e0118d144e0a4a5f569b2251eVinit Deshpande                }
2105e4e3750390bec0a849a9153348b7c21b2cc8b843Vinit Deshapnde            }
2106e4e3750390bec0a849a9153348b7c21b2cc8b843Vinit Deshapnde        }
2107e4e3750390bec0a849a9153348b7c21b2cc8b843Vinit Deshapnde    }
2108e4e3750390bec0a849a9153348b7c21b2cc8b843Vinit Deshapnde
210918786eca942042388748b0d98979f21c9dff4a89Mitchell Wills    // Callback from native
211018786eca942042388748b0d98979f21c9dff4a89Mitchell Wills    private static void onSignificantWifiChange(int id, ScanResult[] results) {
211118786eca942042388748b0d98979f21c9dff4a89Mitchell Wills        SignificantWifiChangeEventHandler handler = sSignificantWifiChangeHandler;
211218786eca942042388748b0d98979f21c9dff4a89Mitchell Wills        if (handler != null) {
211318786eca942042388748b0d98979f21c9dff4a89Mitchell Wills            handler.onChangesFound(results);
211418786eca942042388748b0d98979f21c9dff4a89Mitchell Wills        } else {
2115f49a59bda006b13e0118d144e0a4a5f569b2251eVinit Deshpande            /* this can happen because of race conditions */
211618786eca942042388748b0d98979f21c9dff4a89Mitchell Wills            Log.d(TAG, "Ignoring significant wifi change");
2117aabe7a9f2f32915fd1f25416a6d2034a844005d6vandwalle        }
2118e4e3750390bec0a849a9153348b7c21b2cc8b843Vinit Deshapnde    }
2119e4e3750390bec0a849a9153348b7c21b2cc8b843Vinit Deshapnde
212018786eca942042388748b0d98979f21c9dff4a89Mitchell Wills    public WifiLinkLayerStats getWifiLinkLayerStats(String iface) {
2121200e8ee5097134010a6edee8d031bb02ff7eeb5avandwalle        // TODO: use correct iface name to Index translation
2122200e8ee5097134010a6edee8d031bb02ff7eeb5avandwalle        if (iface == null) return null;
2123956f54b391677d78379729dd14518edddf3c7660Etan Cohen        synchronized (sLock) {
212471d0be16c79791a005d60f9d5fa7d2d81e1f8c80xinhe            if (isHalStarted()) {
2125aabe7a9f2f32915fd1f25416a6d2034a844005d6vandwalle                return getWifiLinkLayerStatsNative(sWlan0Index);
2126f49a59bda006b13e0118d144e0a4a5f569b2251eVinit Deshpande            } else {
2127f49a59bda006b13e0118d144e0a4a5f569b2251eVinit Deshpande                return null;
2128f49a59bda006b13e0118d144e0a4a5f569b2251eVinit Deshpande            }
2129aabe7a9f2f32915fd1f25416a6d2034a844005d6vandwalle        }
21305c08cc119b92af69997af194cc8b6d0111e37d31Andres Morales    }
21315c08cc119b92af69997af194cc8b6d0111e37d31Andres Morales
213218786eca942042388748b0d98979f21c9dff4a89Mitchell Wills    public void setWifiLinkLayerStats(String iface, int enable) {
2133d745a52dc4f929d4a4030f205ed173bdf60eaf10Pierre Vandwalle        if (iface == null) return;
2134956f54b391677d78379729dd14518edddf3c7660Etan Cohen        synchronized (sLock) {
213571d0be16c79791a005d60f9d5fa7d2d81e1f8c80xinhe            if (isHalStarted()) {
2136d745a52dc4f929d4a4030f205ed173bdf60eaf10Pierre Vandwalle                setWifiLinkLayerStatsNative(sWlan0Index, enable);
2137d745a52dc4f929d4a4030f205ed173bdf60eaf10Pierre Vandwalle            }
2138d745a52dc4f929d4a4030f205ed173bdf60eaf10Pierre Vandwalle        }
2139d745a52dc4f929d4a4030f205ed173bdf60eaf10Pierre Vandwalle    }
2140d745a52dc4f929d4a4030f205ed173bdf60eaf10Pierre Vandwalle
2141c35361d54d4885c3174499e4ad46d3324387a9bbVinit Deshpande    public static native int getSupportedFeatureSetNative(int iface);
214218786eca942042388748b0d98979f21c9dff4a89Mitchell Wills    public int getSupportedFeatureSet() {
2143956f54b391677d78379729dd14518edddf3c7660Etan Cohen        synchronized (sLock) {
214471d0be16c79791a005d60f9d5fa7d2d81e1f8c80xinhe            if (isHalStarted()) {
2145f49a59bda006b13e0118d144e0a4a5f569b2251eVinit Deshpande                return getSupportedFeatureSetNative(sWlan0Index);
2146f49a59bda006b13e0118d144e0a4a5f569b2251eVinit Deshpande            } else {
2147f49a59bda006b13e0118d144e0a4a5f569b2251eVinit Deshpande                Log.d(TAG, "Failing getSupportedFeatureset because HAL isn't started");
2148f49a59bda006b13e0118d144e0a4a5f569b2251eVinit Deshpande                return 0;
2149f49a59bda006b13e0118d144e0a4a5f569b2251eVinit Deshpande            }
2150f49a59bda006b13e0118d144e0a4a5f569b2251eVinit Deshpande        }
2151a632d8a6edd350c7644b593b18eceaa5b368505bVinit Deshpande    }
2152143657392bf0702a155fe688171a5abbf4c86570Vinit Deshpande
2153143657392bf0702a155fe688171a5abbf4c86570Vinit Deshpande    /* Rtt related commands/events */
2154143657392bf0702a155fe688171a5abbf4c86570Vinit Deshpande    public static interface RttEventHandler {
2155143657392bf0702a155fe688171a5abbf4c86570Vinit Deshpande        void onRttResults(RttManager.RttResult[] result);
2156143657392bf0702a155fe688171a5abbf4c86570Vinit Deshpande    }
2157143657392bf0702a155fe688171a5abbf4c86570Vinit Deshpande
2158143657392bf0702a155fe688171a5abbf4c86570Vinit Deshpande    private static RttEventHandler sRttEventHandler;
2159143657392bf0702a155fe688171a5abbf4c86570Vinit Deshpande    private static int sRttCmdId;
2160143657392bf0702a155fe688171a5abbf4c86570Vinit Deshpande
216118786eca942042388748b0d98979f21c9dff4a89Mitchell Wills    // Callback from native
216218786eca942042388748b0d98979f21c9dff4a89Mitchell Wills    private static void onRttResults(int id, RttManager.RttResult[] results) {
216318786eca942042388748b0d98979f21c9dff4a89Mitchell Wills        RttEventHandler handler = sRttEventHandler;
216418786eca942042388748b0d98979f21c9dff4a89Mitchell Wills        if (handler != null && id == sRttCmdId) {
216502a1f98f2cecb8ae2d466d6f9fab06b473f970ddVinit Deshpande            Log.d(TAG, "Received " + results.length + " rtt results");
216618786eca942042388748b0d98979f21c9dff4a89Mitchell Wills            handler.onRttResults(results);
2167143657392bf0702a155fe688171a5abbf4c86570Vinit Deshpande            sRttCmdId = 0;
2168143657392bf0702a155fe688171a5abbf4c86570Vinit Deshpande        } else {
216918786eca942042388748b0d98979f21c9dff4a89Mitchell Wills            Log.d(TAG, "RTT Received event for unknown cmd = " + id +
217018786eca942042388748b0d98979f21c9dff4a89Mitchell Wills                    ", current id = " + sRttCmdId);
2171143657392bf0702a155fe688171a5abbf4c86570Vinit Deshpande        }
2172143657392bf0702a155fe688171a5abbf4c86570Vinit Deshpande    }
2173143657392bf0702a155fe688171a5abbf4c86570Vinit Deshpande
2174143657392bf0702a155fe688171a5abbf4c86570Vinit Deshpande    private static native boolean requestRangeNative(
2175143657392bf0702a155fe688171a5abbf4c86570Vinit Deshpande            int iface, int id, RttManager.RttParams[] params);
2176143657392bf0702a155fe688171a5abbf4c86570Vinit Deshpande    private static native boolean cancelRangeRequestNative(
2177143657392bf0702a155fe688171a5abbf4c86570Vinit Deshpande            int iface, int id, RttManager.RttParams[] params);
2178143657392bf0702a155fe688171a5abbf4c86570Vinit Deshpande
217918786eca942042388748b0d98979f21c9dff4a89Mitchell Wills    public boolean requestRtt(
2180143657392bf0702a155fe688171a5abbf4c86570Vinit Deshpande            RttManager.RttParams[] params, RttEventHandler handler) {
2181956f54b391677d78379729dd14518edddf3c7660Etan Cohen        synchronized (sLock) {
218271d0be16c79791a005d60f9d5fa7d2d81e1f8c80xinhe            if (isHalStarted()) {
2183f49a59bda006b13e0118d144e0a4a5f569b2251eVinit Deshpande                if (sRttCmdId != 0) {
2184f49a59bda006b13e0118d144e0a4a5f569b2251eVinit Deshpande                    Log.v("TAG", "Last one is still under measurement!");
2185f49a59bda006b13e0118d144e0a4a5f569b2251eVinit Deshpande                    return false;
2186f49a59bda006b13e0118d144e0a4a5f569b2251eVinit Deshpande                } else {
2187f49a59bda006b13e0118d144e0a4a5f569b2251eVinit Deshpande                    sRttCmdId = getNewCmdIdLocked();
2188f49a59bda006b13e0118d144e0a4a5f569b2251eVinit Deshpande                }
2189f49a59bda006b13e0118d144e0a4a5f569b2251eVinit Deshpande                sRttEventHandler = handler;
2190f49a59bda006b13e0118d144e0a4a5f569b2251eVinit Deshpande                Log.v(TAG, "native issue RTT request");
2191f49a59bda006b13e0118d144e0a4a5f569b2251eVinit Deshpande                return requestRangeNative(sWlan0Index, sRttCmdId, params);
2192143657392bf0702a155fe688171a5abbf4c86570Vinit Deshpande            } else {
2193f49a59bda006b13e0118d144e0a4a5f569b2251eVinit Deshpande                return false;
2194143657392bf0702a155fe688171a5abbf4c86570Vinit Deshpande            }
2195143657392bf0702a155fe688171a5abbf4c86570Vinit Deshpande        }
2196143657392bf0702a155fe688171a5abbf4c86570Vinit Deshpande    }
2197143657392bf0702a155fe688171a5abbf4c86570Vinit Deshpande
219818786eca942042388748b0d98979f21c9dff4a89Mitchell Wills    public boolean cancelRtt(RttManager.RttParams[] params) {
2199956f54b391677d78379729dd14518edddf3c7660Etan Cohen        synchronized (sLock) {
220071d0be16c79791a005d60f9d5fa7d2d81e1f8c80xinhe            if (isHalStarted()) {
2201f49a59bda006b13e0118d144e0a4a5f569b2251eVinit Deshpande                if (sRttCmdId == 0) {
2202f49a59bda006b13e0118d144e0a4a5f569b2251eVinit Deshpande                    return false;
2203f49a59bda006b13e0118d144e0a4a5f569b2251eVinit Deshpande                }
2204143657392bf0702a155fe688171a5abbf4c86570Vinit Deshpande
2205f49a59bda006b13e0118d144e0a4a5f569b2251eVinit Deshpande                sRttCmdId = 0;
2206f95649f33db0a328cb4c0bb5e10c7075e6c828f8xinhe
2207f49a59bda006b13e0118d144e0a4a5f569b2251eVinit Deshpande                if (cancelRangeRequestNative(sWlan0Index, sRttCmdId, params)) {
2208f49a59bda006b13e0118d144e0a4a5f569b2251eVinit Deshpande                    sRttEventHandler = null;
22095cfd8d8b9f241dcad874125a1b5538ee0d6860fexinhe                    Log.v(TAG, "RTT cancel Request Successfully");
2210f49a59bda006b13e0118d144e0a4a5f569b2251eVinit Deshpande                    return true;
2211f49a59bda006b13e0118d144e0a4a5f569b2251eVinit Deshpande                } else {
22125cfd8d8b9f241dcad874125a1b5538ee0d6860fexinhe                    Log.e(TAG, "RTT cancel Request failed");
2213f49a59bda006b13e0118d144e0a4a5f569b2251eVinit Deshpande                    return false;
2214f49a59bda006b13e0118d144e0a4a5f569b2251eVinit Deshpande                }
2215143657392bf0702a155fe688171a5abbf4c86570Vinit Deshpande            } else {
2216143657392bf0702a155fe688171a5abbf4c86570Vinit Deshpande                return false;
2217143657392bf0702a155fe688171a5abbf4c86570Vinit Deshpande            }
2218143657392bf0702a155fe688171a5abbf4c86570Vinit Deshpande        }
2219143657392bf0702a155fe688171a5abbf4c86570Vinit Deshpande    }
2220042c54bfd5144ce8e720585b2093796e7e28de5eVinit Deshpande
222168cb8c02b4ad079c54a2ffd4407da921d18c7af9Wei Wang    private static int sRttResponderCmdId = 0;
222268cb8c02b4ad079c54a2ffd4407da921d18c7af9Wei Wang
222368cb8c02b4ad079c54a2ffd4407da921d18c7af9Wei Wang    private static native ResponderConfig enableRttResponderNative(int iface, int commandId,
222468cb8c02b4ad079c54a2ffd4407da921d18c7af9Wei Wang            int timeoutSeconds, WifiChannelInfo channelHint);
222568cb8c02b4ad079c54a2ffd4407da921d18c7af9Wei Wang    /**
222668cb8c02b4ad079c54a2ffd4407da921d18c7af9Wei Wang     * Enable RTT responder role on the device. Returns {@link ResponderConfig} if the responder
222768cb8c02b4ad079c54a2ffd4407da921d18c7af9Wei Wang     * role is successfully enabled, {@code null} otherwise.
222868cb8c02b4ad079c54a2ffd4407da921d18c7af9Wei Wang     */
222968cb8c02b4ad079c54a2ffd4407da921d18c7af9Wei Wang    @Nullable
223068cb8c02b4ad079c54a2ffd4407da921d18c7af9Wei Wang    public ResponderConfig enableRttResponder(int timeoutSeconds) {
223168cb8c02b4ad079c54a2ffd4407da921d18c7af9Wei Wang        synchronized (sLock) {
223268cb8c02b4ad079c54a2ffd4407da921d18c7af9Wei Wang            if (!isHalStarted()) return null;
223368cb8c02b4ad079c54a2ffd4407da921d18c7af9Wei Wang            if (sRttResponderCmdId != 0) {
223468cb8c02b4ad079c54a2ffd4407da921d18c7af9Wei Wang                if (DBG) Log.e(mTAG, "responder mode already enabled - this shouldn't happen");
223568cb8c02b4ad079c54a2ffd4407da921d18c7af9Wei Wang                return null;
223668cb8c02b4ad079c54a2ffd4407da921d18c7af9Wei Wang            }
223768cb8c02b4ad079c54a2ffd4407da921d18c7af9Wei Wang            int id = getNewCmdIdLocked();
223868cb8c02b4ad079c54a2ffd4407da921d18c7af9Wei Wang            ResponderConfig config = enableRttResponderNative(
223968cb8c02b4ad079c54a2ffd4407da921d18c7af9Wei Wang                    sWlan0Index, id, timeoutSeconds, null);
224068cb8c02b4ad079c54a2ffd4407da921d18c7af9Wei Wang            if (config != null) sRttResponderCmdId = id;
224168cb8c02b4ad079c54a2ffd4407da921d18c7af9Wei Wang            if (DBG) Log.d(TAG, "enabling rtt " + (config != null));
224268cb8c02b4ad079c54a2ffd4407da921d18c7af9Wei Wang            return config;
224368cb8c02b4ad079c54a2ffd4407da921d18c7af9Wei Wang        }
224468cb8c02b4ad079c54a2ffd4407da921d18c7af9Wei Wang    }
224568cb8c02b4ad079c54a2ffd4407da921d18c7af9Wei Wang
224668cb8c02b4ad079c54a2ffd4407da921d18c7af9Wei Wang    private static native boolean disableRttResponderNative(int iface, int commandId);
224768cb8c02b4ad079c54a2ffd4407da921d18c7af9Wei Wang    /**
224868cb8c02b4ad079c54a2ffd4407da921d18c7af9Wei Wang     * Disable RTT responder role. Returns {@code true} if responder role is successfully disabled,
224968cb8c02b4ad079c54a2ffd4407da921d18c7af9Wei Wang     * {@code false} otherwise.
225068cb8c02b4ad079c54a2ffd4407da921d18c7af9Wei Wang     */
225168cb8c02b4ad079c54a2ffd4407da921d18c7af9Wei Wang    public boolean disableRttResponder() {
225268cb8c02b4ad079c54a2ffd4407da921d18c7af9Wei Wang        synchronized (sLock) {
225368cb8c02b4ad079c54a2ffd4407da921d18c7af9Wei Wang            if (!isHalStarted()) return false;
225468cb8c02b4ad079c54a2ffd4407da921d18c7af9Wei Wang            if (sRttResponderCmdId == 0) {
225568cb8c02b4ad079c54a2ffd4407da921d18c7af9Wei Wang                Log.e(mTAG, "responder role not enabled yet");
225668cb8c02b4ad079c54a2ffd4407da921d18c7af9Wei Wang                return true;
225768cb8c02b4ad079c54a2ffd4407da921d18c7af9Wei Wang            }
225868cb8c02b4ad079c54a2ffd4407da921d18c7af9Wei Wang            sRttResponderCmdId = 0;
225968cb8c02b4ad079c54a2ffd4407da921d18c7af9Wei Wang            return disableRttResponderNative(sWlan0Index, sRttResponderCmdId);
226068cb8c02b4ad079c54a2ffd4407da921d18c7af9Wei Wang        }
226168cb8c02b4ad079c54a2ffd4407da921d18c7af9Wei Wang    }
226268cb8c02b4ad079c54a2ffd4407da921d18c7af9Wei Wang
2263042c54bfd5144ce8e720585b2093796e7e28de5eVinit Deshpande    private static native boolean setScanningMacOuiNative(int iface, byte[] oui);
2264042c54bfd5144ce8e720585b2093796e7e28de5eVinit Deshpande
226518786eca942042388748b0d98979f21c9dff4a89Mitchell Wills    public boolean setScanningMacOui(byte[] oui) {
2266956f54b391677d78379729dd14518edddf3c7660Etan Cohen        synchronized (sLock) {
226771d0be16c79791a005d60f9d5fa7d2d81e1f8c80xinhe            if (isHalStarted()) {
2268042c54bfd5144ce8e720585b2093796e7e28de5eVinit Deshpande                return setScanningMacOuiNative(sWlan0Index, oui);
2269042c54bfd5144ce8e720585b2093796e7e28de5eVinit Deshpande            } else {
2270042c54bfd5144ce8e720585b2093796e7e28de5eVinit Deshpande                return false;
2271042c54bfd5144ce8e720585b2093796e7e28de5eVinit Deshpande            }
2272042c54bfd5144ce8e720585b2093796e7e28de5eVinit Deshpande        }
2273042c54bfd5144ce8e720585b2093796e7e28de5eVinit Deshpande    }
2274efa77c1826499b0a3e57998bd6b3073b107e45d7Vinit Deshpande
2275efa77c1826499b0a3e57998bd6b3073b107e45d7Vinit Deshpande    private static native int[] getChannelsForBandNative(
2276efa77c1826499b0a3e57998bd6b3073b107e45d7Vinit Deshpande            int iface, int band);
2277efa77c1826499b0a3e57998bd6b3073b107e45d7Vinit Deshpande
227818786eca942042388748b0d98979f21c9dff4a89Mitchell Wills    public int [] getChannelsForBand(int band) {
2279956f54b391677d78379729dd14518edddf3c7660Etan Cohen        synchronized (sLock) {
2280ee0a0132ebb0d7e9baf42e778ea9c094966ffb14xinhe            if (isHalStarted()) {
2281ee0a0132ebb0d7e9baf42e778ea9c094966ffb14xinhe                return getChannelsForBandNative(sWlan0Index, band);
228218786eca942042388748b0d98979f21c9dff4a89Mitchell Wills            } else {
2283ee0a0132ebb0d7e9baf42e778ea9c094966ffb14xinhe                return null;
2284ee0a0132ebb0d7e9baf42e778ea9c094966ffb14xinhe            }
2285efa77c1826499b0a3e57998bd6b3073b107e45d7Vinit Deshpande        }
2286efa77c1826499b0a3e57998bd6b3073b107e45d7Vinit Deshpande    }
22870465ff513cdccf7e883a505e14c5b78758e7e458Vinit Deshpande
2288ee0a0132ebb0d7e9baf42e778ea9c094966ffb14xinhe    private static native boolean isGetChannelsForBandSupportedNative();
228918786eca942042388748b0d98979f21c9dff4a89Mitchell Wills    public boolean isGetChannelsForBandSupported(){
2290956f54b391677d78379729dd14518edddf3c7660Etan Cohen        synchronized (sLock) {
2291ee0a0132ebb0d7e9baf42e778ea9c094966ffb14xinhe            if (isHalStarted()) {
2292ee0a0132ebb0d7e9baf42e778ea9c094966ffb14xinhe                return isGetChannelsForBandSupportedNative();
229318786eca942042388748b0d98979f21c9dff4a89Mitchell Wills            } else {
2294ee0a0132ebb0d7e9baf42e778ea9c094966ffb14xinhe                return false;
2295ee0a0132ebb0d7e9baf42e778ea9c094966ffb14xinhe            }
2296ee0a0132ebb0d7e9baf42e778ea9c094966ffb14xinhe        }
2297ee0a0132ebb0d7e9baf42e778ea9c094966ffb14xinhe    }
22980465ff513cdccf7e883a505e14c5b78758e7e458Vinit Deshpande
22990465ff513cdccf7e883a505e14c5b78758e7e458Vinit Deshpande    private static native boolean setDfsFlagNative(int iface, boolean dfsOn);
230018786eca942042388748b0d98979f21c9dff4a89Mitchell Wills    public boolean setDfsFlag(boolean dfsOn) {
2301956f54b391677d78379729dd14518edddf3c7660Etan Cohen        synchronized (sLock) {
230271d0be16c79791a005d60f9d5fa7d2d81e1f8c80xinhe            if (isHalStarted()) {
23030465ff513cdccf7e883a505e14c5b78758e7e458Vinit Deshpande                return setDfsFlagNative(sWlan0Index, dfsOn);
23040465ff513cdccf7e883a505e14c5b78758e7e458Vinit Deshpande            } else {
23050465ff513cdccf7e883a505e14c5b78758e7e458Vinit Deshpande                return false;
23060465ff513cdccf7e883a505e14c5b78758e7e458Vinit Deshpande            }
23070465ff513cdccf7e883a505e14c5b78758e7e458Vinit Deshpande        }
23080465ff513cdccf7e883a505e14c5b78758e7e458Vinit Deshpande    }
2309b830d76fbf5fa4077531b516066faa2fdbb92e81xinhe
23103b51fd1bb8356b284822f4f677ad941524e616ebNingyuan Wang    private static native boolean setInterfaceUpNative(boolean up);
231164e816ff5483030636e1815f69a6cbaf8cef289eNingyuan Wang    public boolean setInterfaceUp(boolean up) {
2312956f54b391677d78379729dd14518edddf3c7660Etan Cohen        synchronized (sLock) {
231371d0be16c79791a005d60f9d5fa7d2d81e1f8c80xinhe            if (isHalStarted()) {
23143b51fd1bb8356b284822f4f677ad941524e616ebNingyuan Wang                return setInterfaceUpNative(up);
2315b830d76fbf5fa4077531b516066faa2fdbb92e81xinhe            } else {
2316b830d76fbf5fa4077531b516066faa2fdbb92e81xinhe                return false;
2317b830d76fbf5fa4077531b516066faa2fdbb92e81xinhe            }
2318b830d76fbf5fa4077531b516066faa2fdbb92e81xinhe        }
2319b830d76fbf5fa4077531b516066faa2fdbb92e81xinhe    }
232012cf388ecf3b2aa5ba66ed680b931fa356ab75b5xinhe
232112cf388ecf3b2aa5ba66ed680b931fa356ab75b5xinhe    private static native RttManager.RttCapabilities getRttCapabilitiesNative(int iface);
232218786eca942042388748b0d98979f21c9dff4a89Mitchell Wills    public RttManager.RttCapabilities getRttCapabilities() {
2323956f54b391677d78379729dd14518edddf3c7660Etan Cohen        synchronized (sLock) {
232471d0be16c79791a005d60f9d5fa7d2d81e1f8c80xinhe            if (isHalStarted()) {
232512cf388ecf3b2aa5ba66ed680b931fa356ab75b5xinhe                return getRttCapabilitiesNative(sWlan0Index);
232618786eca942042388748b0d98979f21c9dff4a89Mitchell Wills            } else {
232712cf388ecf3b2aa5ba66ed680b931fa356ab75b5xinhe                return null;
232812cf388ecf3b2aa5ba66ed680b931fa356ab75b5xinhe            }
232912cf388ecf3b2aa5ba66ed680b931fa356ab75b5xinhe        }
233012cf388ecf3b2aa5ba66ed680b931fa356ab75b5xinhe    }
2331939177ff615062ec826601d536466875d8457375xinhe
2332e1dab7a2e3ab5911f812a302b4beed1f6eb5aba7Paul Jensen    private static native ApfCapabilities getApfCapabilitiesNative(int iface);
2333e1dab7a2e3ab5911f812a302b4beed1f6eb5aba7Paul Jensen    public ApfCapabilities getApfCapabilities() {
23346609df5a9c14e4440c085567a27437a8cfc50f88Paul Jensen        synchronized (sLock) {
23356609df5a9c14e4440c085567a27437a8cfc50f88Paul Jensen            if (isHalStarted()) {
2336e1dab7a2e3ab5911f812a302b4beed1f6eb5aba7Paul Jensen                return getApfCapabilitiesNative(sWlan0Index);
23376609df5a9c14e4440c085567a27437a8cfc50f88Paul Jensen            } else {
23386609df5a9c14e4440c085567a27437a8cfc50f88Paul Jensen                return null;
23396609df5a9c14e4440c085567a27437a8cfc50f88Paul Jensen            }
23406609df5a9c14e4440c085567a27437a8cfc50f88Paul Jensen        }
23416609df5a9c14e4440c085567a27437a8cfc50f88Paul Jensen    }
23426609df5a9c14e4440c085567a27437a8cfc50f88Paul Jensen
23436609df5a9c14e4440c085567a27437a8cfc50f88Paul Jensen    private static native boolean installPacketFilterNative(int iface, byte[] filter);
23446609df5a9c14e4440c085567a27437a8cfc50f88Paul Jensen    public boolean installPacketFilter(byte[] filter) {
23456609df5a9c14e4440c085567a27437a8cfc50f88Paul Jensen        synchronized (sLock) {
23466609df5a9c14e4440c085567a27437a8cfc50f88Paul Jensen            if (isHalStarted()) {
23476609df5a9c14e4440c085567a27437a8cfc50f88Paul Jensen                return installPacketFilterNative(sWlan0Index, filter);
23486609df5a9c14e4440c085567a27437a8cfc50f88Paul Jensen            } else {
23496609df5a9c14e4440c085567a27437a8cfc50f88Paul Jensen                return false;
23506609df5a9c14e4440c085567a27437a8cfc50f88Paul Jensen            }
23516609df5a9c14e4440c085567a27437a8cfc50f88Paul Jensen        }
23526609df5a9c14e4440c085567a27437a8cfc50f88Paul Jensen    }
23536609df5a9c14e4440c085567a27437a8cfc50f88Paul Jensen
2354939177ff615062ec826601d536466875d8457375xinhe    private static native boolean setCountryCodeHalNative(int iface, String CountryCode);
235518786eca942042388748b0d98979f21c9dff4a89Mitchell Wills    public boolean setCountryCodeHal(String CountryCode) {
2356956f54b391677d78379729dd14518edddf3c7660Etan Cohen        synchronized (sLock) {
235771d0be16c79791a005d60f9d5fa7d2d81e1f8c80xinhe            if (isHalStarted()) {
2358939177ff615062ec826601d536466875d8457375xinhe                return setCountryCodeHalNative(sWlan0Index, CountryCode);
2359939177ff615062ec826601d536466875d8457375xinhe            } else {
2360939177ff615062ec826601d536466875d8457375xinhe                return false;
2361939177ff615062ec826601d536466875d8457375xinhe            }
2362939177ff615062ec826601d536466875d8457375xinhe        }
2363939177ff615062ec826601d536466875d8457375xinhe    }
2364939177ff615062ec826601d536466875d8457375xinhe
2365d57f630f6a7a5fca872d2b96fc4cce1905daee5dxinhe    /* Rtt related commands/events */
2366d57f630f6a7a5fca872d2b96fc4cce1905daee5dxinhe    public abstract class TdlsEventHandler {
2367d57f630f6a7a5fca872d2b96fc4cce1905daee5dxinhe        abstract public void onTdlsStatus(String macAddr, int status, int reason);
2368d57f630f6a7a5fca872d2b96fc4cce1905daee5dxinhe    }
2369d57f630f6a7a5fca872d2b96fc4cce1905daee5dxinhe
2370d57f630f6a7a5fca872d2b96fc4cce1905daee5dxinhe    private static TdlsEventHandler sTdlsEventHandler;
2371d57f630f6a7a5fca872d2b96fc4cce1905daee5dxinhe
2372d57f630f6a7a5fca872d2b96fc4cce1905daee5dxinhe    private static native boolean enableDisableTdlsNative(int iface, boolean enable,
2373d57f630f6a7a5fca872d2b96fc4cce1905daee5dxinhe            String macAddr);
237418786eca942042388748b0d98979f21c9dff4a89Mitchell Wills    public boolean enableDisableTdls(boolean enable, String macAdd, TdlsEventHandler tdlsCallBack) {
2375956f54b391677d78379729dd14518edddf3c7660Etan Cohen        synchronized (sLock) {
2376f49a59bda006b13e0118d144e0a4a5f569b2251eVinit Deshpande            sTdlsEventHandler = tdlsCallBack;
2377f49a59bda006b13e0118d144e0a4a5f569b2251eVinit Deshpande            return enableDisableTdlsNative(sWlan0Index, enable, macAdd);
2378d57f630f6a7a5fca872d2b96fc4cce1905daee5dxinhe        }
2379d57f630f6a7a5fca872d2b96fc4cce1905daee5dxinhe    }
2380d57f630f6a7a5fca872d2b96fc4cce1905daee5dxinhe
2381d57f630f6a7a5fca872d2b96fc4cce1905daee5dxinhe    // Once TDLS per mac and event feature is implemented, this class definition should be
2382d57f630f6a7a5fca872d2b96fc4cce1905daee5dxinhe    // moved to the right place, like WifiManager etc
2383d57f630f6a7a5fca872d2b96fc4cce1905daee5dxinhe    public static class TdlsStatus {
2384d57f630f6a7a5fca872d2b96fc4cce1905daee5dxinhe        int channel;
2385d57f630f6a7a5fca872d2b96fc4cce1905daee5dxinhe        int global_operating_class;
2386d57f630f6a7a5fca872d2b96fc4cce1905daee5dxinhe        int state;
2387d57f630f6a7a5fca872d2b96fc4cce1905daee5dxinhe        int reason;
2388d57f630f6a7a5fca872d2b96fc4cce1905daee5dxinhe    }
2389d57f630f6a7a5fca872d2b96fc4cce1905daee5dxinhe    private static native TdlsStatus getTdlsStatusNative(int iface, String macAddr);
239018786eca942042388748b0d98979f21c9dff4a89Mitchell Wills    public TdlsStatus getTdlsStatus(String macAdd) {
2391956f54b391677d78379729dd14518edddf3c7660Etan Cohen        synchronized (sLock) {
239271d0be16c79791a005d60f9d5fa7d2d81e1f8c80xinhe            if (isHalStarted()) {
2393d57f630f6a7a5fca872d2b96fc4cce1905daee5dxinhe                return getTdlsStatusNative(sWlan0Index, macAdd);
2394d57f630f6a7a5fca872d2b96fc4cce1905daee5dxinhe            } else {
2395d57f630f6a7a5fca872d2b96fc4cce1905daee5dxinhe                return null;
2396d57f630f6a7a5fca872d2b96fc4cce1905daee5dxinhe            }
2397d57f630f6a7a5fca872d2b96fc4cce1905daee5dxinhe        }
2398d57f630f6a7a5fca872d2b96fc4cce1905daee5dxinhe    }
2399d57f630f6a7a5fca872d2b96fc4cce1905daee5dxinhe
2400d57f630f6a7a5fca872d2b96fc4cce1905daee5dxinhe    //ToFix: Once TDLS per mac and event feature is implemented, this class definition should be
2401d57f630f6a7a5fca872d2b96fc4cce1905daee5dxinhe    // moved to the right place, like WifiStateMachine etc
2402d57f630f6a7a5fca872d2b96fc4cce1905daee5dxinhe    public static class TdlsCapabilities {
2403d57f630f6a7a5fca872d2b96fc4cce1905daee5dxinhe        /* Maximum TDLS session number can be supported by the Firmware and hardware */
2404d57f630f6a7a5fca872d2b96fc4cce1905daee5dxinhe        int maxConcurrentTdlsSessionNumber;
2405d57f630f6a7a5fca872d2b96fc4cce1905daee5dxinhe        boolean isGlobalTdlsSupported;
2406d57f630f6a7a5fca872d2b96fc4cce1905daee5dxinhe        boolean isPerMacTdlsSupported;
2407d57f630f6a7a5fca872d2b96fc4cce1905daee5dxinhe        boolean isOffChannelTdlsSupported;
2408d57f630f6a7a5fca872d2b96fc4cce1905daee5dxinhe    }
2409d57f630f6a7a5fca872d2b96fc4cce1905daee5dxinhe
2410d57f630f6a7a5fca872d2b96fc4cce1905daee5dxinhe
2411d57f630f6a7a5fca872d2b96fc4cce1905daee5dxinhe
2412d57f630f6a7a5fca872d2b96fc4cce1905daee5dxinhe    private static native TdlsCapabilities getTdlsCapabilitiesNative(int iface);
241318786eca942042388748b0d98979f21c9dff4a89Mitchell Wills    public TdlsCapabilities getTdlsCapabilities () {
2414956f54b391677d78379729dd14518edddf3c7660Etan Cohen        synchronized (sLock) {
241571d0be16c79791a005d60f9d5fa7d2d81e1f8c80xinhe            if (isHalStarted()) {
2416d57f630f6a7a5fca872d2b96fc4cce1905daee5dxinhe                return getTdlsCapabilitiesNative(sWlan0Index);
2417d57f630f6a7a5fca872d2b96fc4cce1905daee5dxinhe            } else {
2418d57f630f6a7a5fca872d2b96fc4cce1905daee5dxinhe                return null;
2419d57f630f6a7a5fca872d2b96fc4cce1905daee5dxinhe            }
2420d57f630f6a7a5fca872d2b96fc4cce1905daee5dxinhe        }
2421d57f630f6a7a5fca872d2b96fc4cce1905daee5dxinhe    }
2422d57f630f6a7a5fca872d2b96fc4cce1905daee5dxinhe
242318786eca942042388748b0d98979f21c9dff4a89Mitchell Wills    private static boolean onTdlsStatus(String macAddr, int status, int reason) {
242418786eca942042388748b0d98979f21c9dff4a89Mitchell Wills        TdlsEventHandler handler = sTdlsEventHandler;
242518786eca942042388748b0d98979f21c9dff4a89Mitchell Wills        if (handler == null) {
242618786eca942042388748b0d98979f21c9dff4a89Mitchell Wills            return false;
242718786eca942042388748b0d98979f21c9dff4a89Mitchell Wills        } else {
242818786eca942042388748b0d98979f21c9dff4a89Mitchell Wills            handler.onTdlsStatus(macAddr, status, reason);
242918786eca942042388748b0d98979f21c9dff4a89Mitchell Wills            return true;
243018786eca942042388748b0d98979f21c9dff4a89Mitchell Wills        }
2431d57f630f6a7a5fca872d2b96fc4cce1905daee5dxinhe    }
2432d57f630f6a7a5fca872d2b96fc4cce1905daee5dxinhe
2433a0d34d3626bd9631039a485d2d428b1b7de390e2Pierre Vandwalle    //---------------------------------------------------------------------------------
2434a0d34d3626bd9631039a485d2d428b1b7de390e2Pierre Vandwalle
2435a0d34d3626bd9631039a485d2d428b1b7de390e2Pierre Vandwalle    /* Wifi Logger commands/events */
24367d6301ead19afdf3de37455e9ed133c25b4938cdVinit Deshpande
2437a0d34d3626bd9631039a485d2d428b1b7de390e2Pierre Vandwalle    public static interface WifiLoggerEventHandler {
24380bf150b2d69ec62df905bc6f8603437dc4caee1fVinit Deshpande        void onRingBufferData(RingBufferStatus status, byte[] buffer);
24390bf150b2d69ec62df905bc6f8603437dc4caee1fVinit Deshpande        void onWifiAlert(int errorCode, byte[] buffer);
2440a0d34d3626bd9631039a485d2d428b1b7de390e2Pierre Vandwalle    }
2441a0d34d3626bd9631039a485d2d428b1b7de390e2Pierre Vandwalle
2442a0d34d3626bd9631039a485d2d428b1b7de390e2Pierre Vandwalle    private static WifiLoggerEventHandler sWifiLoggerEventHandler = null;
2443a0d34d3626bd9631039a485d2d428b1b7de390e2Pierre Vandwalle
244418786eca942042388748b0d98979f21c9dff4a89Mitchell Wills    // Callback from native
24450bf150b2d69ec62df905bc6f8603437dc4caee1fVinit Deshpande    private static void onRingBufferData(RingBufferStatus status, byte[] buffer) {
244618786eca942042388748b0d98979f21c9dff4a89Mitchell Wills        WifiLoggerEventHandler handler = sWifiLoggerEventHandler;
244718786eca942042388748b0d98979f21c9dff4a89Mitchell Wills        if (handler != null)
244818786eca942042388748b0d98979f21c9dff4a89Mitchell Wills            handler.onRingBufferData(status, buffer);
244903ba4a5b108ab2bbdb5d0e7be46ac45c00a6d9ffxinhe    }
245003ba4a5b108ab2bbdb5d0e7be46ac45c00a6d9ffxinhe
245118786eca942042388748b0d98979f21c9dff4a89Mitchell Wills    // Callback from native
24520bf150b2d69ec62df905bc6f8603437dc4caee1fVinit Deshpande    private static void onWifiAlert(byte[] buffer, int errorCode) {
245318786eca942042388748b0d98979f21c9dff4a89Mitchell Wills        WifiLoggerEventHandler handler = sWifiLoggerEventHandler;
245418786eca942042388748b0d98979f21c9dff4a89Mitchell Wills        if (handler != null)
245518786eca942042388748b0d98979f21c9dff4a89Mitchell Wills            handler.onWifiAlert(errorCode, buffer);
245603ba4a5b108ab2bbdb5d0e7be46ac45c00a6d9ffxinhe    }
245703ba4a5b108ab2bbdb5d0e7be46ac45c00a6d9ffxinhe
2458b797893fc1966803d0c013faac42e6396a37a384xinhe    private static int sLogCmdId = -1;
2459b797893fc1966803d0c013faac42e6396a37a384xinhe    private static native boolean setLoggingEventHandlerNative(int iface, int id);
246018786eca942042388748b0d98979f21c9dff4a89Mitchell Wills    public boolean setLoggingEventHandler(WifiLoggerEventHandler handler) {
2461956f54b391677d78379729dd14518edddf3c7660Etan Cohen        synchronized (sLock) {
2462b797893fc1966803d0c013faac42e6396a37a384xinhe            if (isHalStarted()) {
2463b797893fc1966803d0c013faac42e6396a37a384xinhe                int oldId =  sLogCmdId;
2464b797893fc1966803d0c013faac42e6396a37a384xinhe                sLogCmdId = getNewCmdIdLocked();
2465b797893fc1966803d0c013faac42e6396a37a384xinhe                if (!setLoggingEventHandlerNative(sWlan0Index, sLogCmdId)) {
2466b797893fc1966803d0c013faac42e6396a37a384xinhe                    sLogCmdId = oldId;
2467b797893fc1966803d0c013faac42e6396a37a384xinhe                    return false;
2468b797893fc1966803d0c013faac42e6396a37a384xinhe                }
2469b797893fc1966803d0c013faac42e6396a37a384xinhe                sWifiLoggerEventHandler = handler;
2470b797893fc1966803d0c013faac42e6396a37a384xinhe                return true;
2471b797893fc1966803d0c013faac42e6396a37a384xinhe            } else {
2472b797893fc1966803d0c013faac42e6396a37a384xinhe                return false;
2473b797893fc1966803d0c013faac42e6396a37a384xinhe            }
247403ba4a5b108ab2bbdb5d0e7be46ac45c00a6d9ffxinhe        }
247503ba4a5b108ab2bbdb5d0e7be46ac45c00a6d9ffxinhe    }
247603ba4a5b108ab2bbdb5d0e7be46ac45c00a6d9ffxinhe
247703ba4a5b108ab2bbdb5d0e7be46ac45c00a6d9ffxinhe    private static native boolean startLoggingRingBufferNative(int iface, int verboseLevel,
24780bf150b2d69ec62df905bc6f8603437dc4caee1fVinit Deshpande            int flags, int minIntervalSec ,int minDataSize, String ringName);
247918786eca942042388748b0d98979f21c9dff4a89Mitchell Wills    public boolean startLoggingRingBuffer(int verboseLevel, int flags, int maxInterval,
248003ba4a5b108ab2bbdb5d0e7be46ac45c00a6d9ffxinhe            int minDataSize, String ringName){
2481956f54b391677d78379729dd14518edddf3c7660Etan Cohen        synchronized (sLock) {
248271d0be16c79791a005d60f9d5fa7d2d81e1f8c80xinhe            if (isHalStarted()) {
248303ba4a5b108ab2bbdb5d0e7be46ac45c00a6d9ffxinhe                return startLoggingRingBufferNative(sWlan0Index, verboseLevel, flags, maxInterval,
248403ba4a5b108ab2bbdb5d0e7be46ac45c00a6d9ffxinhe                        minDataSize, ringName);
248503ba4a5b108ab2bbdb5d0e7be46ac45c00a6d9ffxinhe            } else {
248603ba4a5b108ab2bbdb5d0e7be46ac45c00a6d9ffxinhe                return false;
248703ba4a5b108ab2bbdb5d0e7be46ac45c00a6d9ffxinhe            }
248803ba4a5b108ab2bbdb5d0e7be46ac45c00a6d9ffxinhe        }
248903ba4a5b108ab2bbdb5d0e7be46ac45c00a6d9ffxinhe    }
249003ba4a5b108ab2bbdb5d0e7be46ac45c00a6d9ffxinhe
249103ba4a5b108ab2bbdb5d0e7be46ac45c00a6d9ffxinhe    private static native int getSupportedLoggerFeatureSetNative(int iface);
249218786eca942042388748b0d98979f21c9dff4a89Mitchell Wills    public int getSupportedLoggerFeatureSet() {
2493956f54b391677d78379729dd14518edddf3c7660Etan Cohen        synchronized (sLock) {
249471d0be16c79791a005d60f9d5fa7d2d81e1f8c80xinhe            if (isHalStarted()) {
249503ba4a5b108ab2bbdb5d0e7be46ac45c00a6d9ffxinhe                return getSupportedLoggerFeatureSetNative(sWlan0Index);
249603ba4a5b108ab2bbdb5d0e7be46ac45c00a6d9ffxinhe            } else {
2497f49a59bda006b13e0118d144e0a4a5f569b2251eVinit Deshpande                return 0;
249803ba4a5b108ab2bbdb5d0e7be46ac45c00a6d9ffxinhe            }
249903ba4a5b108ab2bbdb5d0e7be46ac45c00a6d9ffxinhe        }
250003ba4a5b108ab2bbdb5d0e7be46ac45c00a6d9ffxinhe    }
250103ba4a5b108ab2bbdb5d0e7be46ac45c00a6d9ffxinhe
2502b797893fc1966803d0c013faac42e6396a37a384xinhe    private static native boolean resetLogHandlerNative(int iface, int id);
250318786eca942042388748b0d98979f21c9dff4a89Mitchell Wills    public boolean resetLogHandler() {
2504956f54b391677d78379729dd14518edddf3c7660Etan Cohen        synchronized (sLock) {
2505b797893fc1966803d0c013faac42e6396a37a384xinhe            if (isHalStarted()) {
2506b797893fc1966803d0c013faac42e6396a37a384xinhe                if (sLogCmdId == -1) {
2507b797893fc1966803d0c013faac42e6396a37a384xinhe                    Log.e(TAG,"Can not reset handler Before set any handler");
2508b797893fc1966803d0c013faac42e6396a37a384xinhe                    return false;
2509b797893fc1966803d0c013faac42e6396a37a384xinhe                }
2510b797893fc1966803d0c013faac42e6396a37a384xinhe                sWifiLoggerEventHandler = null;
2511b797893fc1966803d0c013faac42e6396a37a384xinhe                if (resetLogHandlerNative(sWlan0Index, sLogCmdId)) {
2512b797893fc1966803d0c013faac42e6396a37a384xinhe                    sLogCmdId = -1;
2513b797893fc1966803d0c013faac42e6396a37a384xinhe                    return true;
2514b797893fc1966803d0c013faac42e6396a37a384xinhe                } else {
2515b797893fc1966803d0c013faac42e6396a37a384xinhe                    return false;
2516b797893fc1966803d0c013faac42e6396a37a384xinhe                }
2517b797893fc1966803d0c013faac42e6396a37a384xinhe            } else {
2518b797893fc1966803d0c013faac42e6396a37a384xinhe                return false;
2519b797893fc1966803d0c013faac42e6396a37a384xinhe            }
2520b797893fc1966803d0c013faac42e6396a37a384xinhe        }
2521b797893fc1966803d0c013faac42e6396a37a384xinhe    }
2522b797893fc1966803d0c013faac42e6396a37a384xinhe
252303ba4a5b108ab2bbdb5d0e7be46ac45c00a6d9ffxinhe    private static native String getDriverVersionNative(int iface);
252418786eca942042388748b0d98979f21c9dff4a89Mitchell Wills    public String getDriverVersion() {
2525956f54b391677d78379729dd14518edddf3c7660Etan Cohen        synchronized (sLock) {
252671d0be16c79791a005d60f9d5fa7d2d81e1f8c80xinhe            if (isHalStarted()) {
252703ba4a5b108ab2bbdb5d0e7be46ac45c00a6d9ffxinhe                return getDriverVersionNative(sWlan0Index);
252803ba4a5b108ab2bbdb5d0e7be46ac45c00a6d9ffxinhe            } else {
2529f49a59bda006b13e0118d144e0a4a5f569b2251eVinit Deshpande                return "";
253003ba4a5b108ab2bbdb5d0e7be46ac45c00a6d9ffxinhe            }
253103ba4a5b108ab2bbdb5d0e7be46ac45c00a6d9ffxinhe        }
253203ba4a5b108ab2bbdb5d0e7be46ac45c00a6d9ffxinhe    }
253303ba4a5b108ab2bbdb5d0e7be46ac45c00a6d9ffxinhe
253403ba4a5b108ab2bbdb5d0e7be46ac45c00a6d9ffxinhe
253503ba4a5b108ab2bbdb5d0e7be46ac45c00a6d9ffxinhe    private static native String getFirmwareVersionNative(int iface);
253618786eca942042388748b0d98979f21c9dff4a89Mitchell Wills    public String getFirmwareVersion() {
2537956f54b391677d78379729dd14518edddf3c7660Etan Cohen        synchronized (sLock) {
253871d0be16c79791a005d60f9d5fa7d2d81e1f8c80xinhe            if (isHalStarted()) {
253903ba4a5b108ab2bbdb5d0e7be46ac45c00a6d9ffxinhe                return getFirmwareVersionNative(sWlan0Index);
254003ba4a5b108ab2bbdb5d0e7be46ac45c00a6d9ffxinhe            } else {
2541f49a59bda006b13e0118d144e0a4a5f569b2251eVinit Deshpande                return "";
254203ba4a5b108ab2bbdb5d0e7be46ac45c00a6d9ffxinhe            }
254303ba4a5b108ab2bbdb5d0e7be46ac45c00a6d9ffxinhe        }
254403ba4a5b108ab2bbdb5d0e7be46ac45c00a6d9ffxinhe    }
254503ba4a5b108ab2bbdb5d0e7be46ac45c00a6d9ffxinhe
25460bf150b2d69ec62df905bc6f8603437dc4caee1fVinit Deshpande    public static class RingBufferStatus{
25470bf150b2d69ec62df905bc6f8603437dc4caee1fVinit Deshpande        String name;
25480bf150b2d69ec62df905bc6f8603437dc4caee1fVinit Deshpande        int flag;
25490bf150b2d69ec62df905bc6f8603437dc4caee1fVinit Deshpande        int ringBufferId;
25500bf150b2d69ec62df905bc6f8603437dc4caee1fVinit Deshpande        int ringBufferByteSize;
25510bf150b2d69ec62df905bc6f8603437dc4caee1fVinit Deshpande        int verboseLevel;
25520bf150b2d69ec62df905bc6f8603437dc4caee1fVinit Deshpande        int writtenBytes;
25530bf150b2d69ec62df905bc6f8603437dc4caee1fVinit Deshpande        int readBytes;
25540bf150b2d69ec62df905bc6f8603437dc4caee1fVinit Deshpande        int writtenRecords;
25550bf150b2d69ec62df905bc6f8603437dc4caee1fVinit Deshpande
25560bf150b2d69ec62df905bc6f8603437dc4caee1fVinit Deshpande        @Override
25570bf150b2d69ec62df905bc6f8603437dc4caee1fVinit Deshpande        public String toString() {
25580bf150b2d69ec62df905bc6f8603437dc4caee1fVinit Deshpande            return "name: " + name + " flag: " + flag + " ringBufferId: " + ringBufferId +
25590bf150b2d69ec62df905bc6f8603437dc4caee1fVinit Deshpande                    " ringBufferByteSize: " +ringBufferByteSize + " verboseLevel: " +verboseLevel +
25600bf150b2d69ec62df905bc6f8603437dc4caee1fVinit Deshpande                    " writtenBytes: " + writtenBytes + " readBytes: " + readBytes +
25610bf150b2d69ec62df905bc6f8603437dc4caee1fVinit Deshpande                    " writtenRecords: " + writtenRecords;
25620bf150b2d69ec62df905bc6f8603437dc4caee1fVinit Deshpande        }
25630bf150b2d69ec62df905bc6f8603437dc4caee1fVinit Deshpande    }
25640bf150b2d69ec62df905bc6f8603437dc4caee1fVinit Deshpande
25650bf150b2d69ec62df905bc6f8603437dc4caee1fVinit Deshpande    private static native RingBufferStatus[] getRingBufferStatusNative(int iface);
256618786eca942042388748b0d98979f21c9dff4a89Mitchell Wills    public RingBufferStatus[] getRingBufferStatus() {
2567956f54b391677d78379729dd14518edddf3c7660Etan Cohen        synchronized (sLock) {
256871d0be16c79791a005d60f9d5fa7d2d81e1f8c80xinhe            if (isHalStarted()) {
256903ba4a5b108ab2bbdb5d0e7be46ac45c00a6d9ffxinhe                return getRingBufferStatusNative(sWlan0Index);
257003ba4a5b108ab2bbdb5d0e7be46ac45c00a6d9ffxinhe            } else {
257103ba4a5b108ab2bbdb5d0e7be46ac45c00a6d9ffxinhe                return null;
257203ba4a5b108ab2bbdb5d0e7be46ac45c00a6d9ffxinhe            }
257303ba4a5b108ab2bbdb5d0e7be46ac45c00a6d9ffxinhe        }
257403ba4a5b108ab2bbdb5d0e7be46ac45c00a6d9ffxinhe    }
257503ba4a5b108ab2bbdb5d0e7be46ac45c00a6d9ffxinhe
257603ba4a5b108ab2bbdb5d0e7be46ac45c00a6d9ffxinhe    private static native boolean getRingBufferDataNative(int iface, String ringName);
257718786eca942042388748b0d98979f21c9dff4a89Mitchell Wills    public boolean getRingBufferData(String ringName) {
2578956f54b391677d78379729dd14518edddf3c7660Etan Cohen        synchronized (sLock) {
257971d0be16c79791a005d60f9d5fa7d2d81e1f8c80xinhe            if (isHalStarted()) {
258003ba4a5b108ab2bbdb5d0e7be46ac45c00a6d9ffxinhe                return getRingBufferDataNative(sWlan0Index, ringName);
258103ba4a5b108ab2bbdb5d0e7be46ac45c00a6d9ffxinhe            } else {
258203ba4a5b108ab2bbdb5d0e7be46ac45c00a6d9ffxinhe                return false;
258303ba4a5b108ab2bbdb5d0e7be46ac45c00a6d9ffxinhe            }
258403ba4a5b108ab2bbdb5d0e7be46ac45c00a6d9ffxinhe        }
258503ba4a5b108ab2bbdb5d0e7be46ac45c00a6d9ffxinhe    }
258603ba4a5b108ab2bbdb5d0e7be46ac45c00a6d9ffxinhe
258718786eca942042388748b0d98979f21c9dff4a89Mitchell Wills    private static byte[] mFwMemoryDump;
258818786eca942042388748b0d98979f21c9dff4a89Mitchell Wills    // Callback from native
258903ba4a5b108ab2bbdb5d0e7be46ac45c00a6d9ffxinhe    private static void onWifiFwMemoryAvailable(byte[] buffer) {
259098dca3397c0049d3cfb9c97e2b021c5ee3a2e1bdxinhe        mFwMemoryDump = buffer;
259198dca3397c0049d3cfb9c97e2b021c5ee3a2e1bdxinhe        if (DBG) {
259298dca3397c0049d3cfb9c97e2b021c5ee3a2e1bdxinhe            Log.d(TAG, "onWifiFwMemoryAvailable is called and buffer length is: " +
259398dca3397c0049d3cfb9c97e2b021c5ee3a2e1bdxinhe                    (buffer == null ? 0 :  buffer.length));
259498dca3397c0049d3cfb9c97e2b021c5ee3a2e1bdxinhe        }
259503ba4a5b108ab2bbdb5d0e7be46ac45c00a6d9ffxinhe    }
2596127f7244183786e6ccae09e81eeccdac31973e69xinhe
259703ba4a5b108ab2bbdb5d0e7be46ac45c00a6d9ffxinhe    private static native boolean getFwMemoryDumpNative(int iface);
259818786eca942042388748b0d98979f21c9dff4a89Mitchell Wills    public byte[] getFwMemoryDump() {
2599956f54b391677d78379729dd14518edddf3c7660Etan Cohen        synchronized (sLock) {
260071d0be16c79791a005d60f9d5fa7d2d81e1f8c80xinhe            if (isHalStarted()) {
26010bf150b2d69ec62df905bc6f8603437dc4caee1fVinit Deshpande                if(getFwMemoryDumpNative(sWlan0Index)) {
26020bf150b2d69ec62df905bc6f8603437dc4caee1fVinit Deshpande                    byte[] fwMemoryDump = mFwMemoryDump;
260303ba4a5b108ab2bbdb5d0e7be46ac45c00a6d9ffxinhe                    mFwMemoryDump = null;
26040bf150b2d69ec62df905bc6f8603437dc4caee1fVinit Deshpande                    return fwMemoryDump;
26050bf150b2d69ec62df905bc6f8603437dc4caee1fVinit Deshpande                } else {
26060bf150b2d69ec62df905bc6f8603437dc4caee1fVinit Deshpande                    return null;
260703ba4a5b108ab2bbdb5d0e7be46ac45c00a6d9ffxinhe                }
260803ba4a5b108ab2bbdb5d0e7be46ac45c00a6d9ffxinhe            }
2609f49a59bda006b13e0118d144e0a4a5f569b2251eVinit Deshpande            return null;
2610a0d34d3626bd9631039a485d2d428b1b7de390e2Pierre Vandwalle        }
2611a0d34d3626bd9631039a485d2d428b1b7de390e2Pierre Vandwalle    }
2612dd490cf1df37b70a8625dc4ec8e712f740dd1e4aPierre Vandwalle
2613d0cb2fae195ebb9f658095667f3c7b7b8d69a204mukesh agrawal    private static native byte[] getDriverStateDumpNative(int iface);
2614d0cb2fae195ebb9f658095667f3c7b7b8d69a204mukesh agrawal    /** Fetch the driver state, for driver debugging. */
2615d0cb2fae195ebb9f658095667f3c7b7b8d69a204mukesh agrawal    public byte[] getDriverStateDump() {
2616d0cb2fae195ebb9f658095667f3c7b7b8d69a204mukesh agrawal        synchronized (sLock) {
2617d0cb2fae195ebb9f658095667f3c7b7b8d69a204mukesh agrawal            if (isHalStarted()) {
2618d0cb2fae195ebb9f658095667f3c7b7b8d69a204mukesh agrawal                return getDriverStateDumpNative(sWlan0Index);
2619d0cb2fae195ebb9f658095667f3c7b7b8d69a204mukesh agrawal            } else {
2620d0cb2fae195ebb9f658095667f3c7b7b8d69a204mukesh agrawal                return null;
2621d0cb2fae195ebb9f658095667f3c7b7b8d69a204mukesh agrawal            }
2622d0cb2fae195ebb9f658095667f3c7b7b8d69a204mukesh agrawal        }
2623d0cb2fae195ebb9f658095667f3c7b7b8d69a204mukesh agrawal    }
2624d0cb2fae195ebb9f658095667f3c7b7b8d69a204mukesh agrawal
2625dd490cf1df37b70a8625dc4ec8e712f740dd1e4aPierre Vandwalle    //---------------------------------------------------------------------------------
262609b1d0786d05436d524d7556c269e665a0962ee6mukesh agrawal    /* Packet fate API */
262709b1d0786d05436d524d7556c269e665a0962ee6mukesh agrawal
262809b1d0786d05436d524d7556c269e665a0962ee6mukesh agrawal    @Immutable
262909b1d0786d05436d524d7556c269e665a0962ee6mukesh agrawal    abstract static class FateReport {
263009b1d0786d05436d524d7556c269e665a0962ee6mukesh agrawal        final byte mFate;
263109b1d0786d05436d524d7556c269e665a0962ee6mukesh agrawal        final long mDriverTimestampUSec;
263209b1d0786d05436d524d7556c269e665a0962ee6mukesh agrawal        final byte mFrameType;
263309b1d0786d05436d524d7556c269e665a0962ee6mukesh agrawal        final byte[] mFrameBytes;
263409b1d0786d05436d524d7556c269e665a0962ee6mukesh agrawal
263509b1d0786d05436d524d7556c269e665a0962ee6mukesh agrawal        FateReport(byte fate, long driverTimestampUSec, byte frameType, byte[] frameBytes) {
263609b1d0786d05436d524d7556c269e665a0962ee6mukesh agrawal            mFate = fate;
263709b1d0786d05436d524d7556c269e665a0962ee6mukesh agrawal            mDriverTimestampUSec = driverTimestampUSec;
263809b1d0786d05436d524d7556c269e665a0962ee6mukesh agrawal            mFrameType = frameType;
263909b1d0786d05436d524d7556c269e665a0962ee6mukesh agrawal            mFrameBytes = frameBytes;
264009b1d0786d05436d524d7556c269e665a0962ee6mukesh agrawal        }
26410fcb06473cbe2824e401a80c0520bb1c14ed8f41mukesh agrawal
2642590f3fc2045389d5ef274c4b3bd6162d93b1a0acSamuel Tan        public String toTableRowString() {
2643590f3fc2045389d5ef274c4b3bd6162d93b1a0acSamuel Tan            StringWriter sw = new StringWriter();
2644590f3fc2045389d5ef274c4b3bd6162d93b1a0acSamuel Tan            PrintWriter pw = new PrintWriter(sw);
2645590f3fc2045389d5ef274c4b3bd6162d93b1a0acSamuel Tan            FrameParser parser = new FrameParser(mFrameType, mFrameBytes);
2646590f3fc2045389d5ef274c4b3bd6162d93b1a0acSamuel Tan            pw.format("%-15s  %-9s  %-32s  %-12s  %s\n",
2647590f3fc2045389d5ef274c4b3bd6162d93b1a0acSamuel Tan                    mDriverTimestampUSec, directionToString(), fateToString(),
2648590f3fc2045389d5ef274c4b3bd6162d93b1a0acSamuel Tan                            parser.mMostSpecificProtocolString, parser.mTypeString);
2649590f3fc2045389d5ef274c4b3bd6162d93b1a0acSamuel Tan            return sw.toString();
2650590f3fc2045389d5ef274c4b3bd6162d93b1a0acSamuel Tan        }
2651590f3fc2045389d5ef274c4b3bd6162d93b1a0acSamuel Tan
2652590f3fc2045389d5ef274c4b3bd6162d93b1a0acSamuel Tan        public String toVerboseStringWithPiiAllowed() {
26530fcb06473cbe2824e401a80c0520bb1c14ed8f41mukesh agrawal            StringWriter sw = new StringWriter();
26540fcb06473cbe2824e401a80c0520bb1c14ed8f41mukesh agrawal            PrintWriter pw = new PrintWriter(sw);
2655590f3fc2045389d5ef274c4b3bd6162d93b1a0acSamuel Tan            FrameParser parser = new FrameParser(mFrameType, mFrameBytes);
26560fcb06473cbe2824e401a80c0520bb1c14ed8f41mukesh agrawal            pw.format("Frame direction: %s\n", directionToString());
26570fcb06473cbe2824e401a80c0520bb1c14ed8f41mukesh agrawal            pw.format("Frame timestamp: %d\n", mDriverTimestampUSec);
26580fcb06473cbe2824e401a80c0520bb1c14ed8f41mukesh agrawal            pw.format("Frame fate: %s\n", fateToString());
26590fcb06473cbe2824e401a80c0520bb1c14ed8f41mukesh agrawal            pw.format("Frame type: %s\n", frameTypeToString(mFrameType));
2660590f3fc2045389d5ef274c4b3bd6162d93b1a0acSamuel Tan            pw.format("Frame protocol: %s\n", parser.mMostSpecificProtocolString);
2661590f3fc2045389d5ef274c4b3bd6162d93b1a0acSamuel Tan            pw.format("Frame protocol type: %s\n", parser.mTypeString);
26620fcb06473cbe2824e401a80c0520bb1c14ed8f41mukesh agrawal            pw.format("Frame length: %d\n", mFrameBytes.length);
26630fcb06473cbe2824e401a80c0520bb1c14ed8f41mukesh agrawal            pw.append("Frame bytes");
2664590f3fc2045389d5ef274c4b3bd6162d93b1a0acSamuel Tan            pw.append(HexDump.dumpHexString(mFrameBytes));  // potentially contains PII
26650fcb06473cbe2824e401a80c0520bb1c14ed8f41mukesh agrawal            pw.append("\n");
26660fcb06473cbe2824e401a80c0520bb1c14ed8f41mukesh agrawal            return sw.toString();
26670fcb06473cbe2824e401a80c0520bb1c14ed8f41mukesh agrawal        }
26680fcb06473cbe2824e401a80c0520bb1c14ed8f41mukesh agrawal
2669590f3fc2045389d5ef274c4b3bd6162d93b1a0acSamuel Tan        /* Returns a header to match the output of toTableRowString(). */
2670590f3fc2045389d5ef274c4b3bd6162d93b1a0acSamuel Tan        public static String getTableHeader() {
2671590f3fc2045389d5ef274c4b3bd6162d93b1a0acSamuel Tan            StringWriter sw = new StringWriter();
2672590f3fc2045389d5ef274c4b3bd6162d93b1a0acSamuel Tan            PrintWriter pw = new PrintWriter(sw);
2673590f3fc2045389d5ef274c4b3bd6162d93b1a0acSamuel Tan            pw.format("\n%-15s  %-9s  %-32s  %-12s  %s\n",
2674590f3fc2045389d5ef274c4b3bd6162d93b1a0acSamuel Tan                    "Timestamp", "Direction", "Fate", "Protocol", "Type");
2675590f3fc2045389d5ef274c4b3bd6162d93b1a0acSamuel Tan            pw.format("%-15s  %-9s  %-32s  %-12s  %s\n",
2676590f3fc2045389d5ef274c4b3bd6162d93b1a0acSamuel Tan                    "---------", "---------", "----", "--------", "----");
2677590f3fc2045389d5ef274c4b3bd6162d93b1a0acSamuel Tan            return sw.toString();
2678590f3fc2045389d5ef274c4b3bd6162d93b1a0acSamuel Tan        }
2679590f3fc2045389d5ef274c4b3bd6162d93b1a0acSamuel Tan
26800fcb06473cbe2824e401a80c0520bb1c14ed8f41mukesh agrawal        protected abstract String directionToString();
26810fcb06473cbe2824e401a80c0520bb1c14ed8f41mukesh agrawal
26820fcb06473cbe2824e401a80c0520bb1c14ed8f41mukesh agrawal        protected abstract String fateToString();
26830fcb06473cbe2824e401a80c0520bb1c14ed8f41mukesh agrawal
26840fcb06473cbe2824e401a80c0520bb1c14ed8f41mukesh agrawal        private static String frameTypeToString(byte frameType) {
26850fcb06473cbe2824e401a80c0520bb1c14ed8f41mukesh agrawal            switch (frameType) {
26860fcb06473cbe2824e401a80c0520bb1c14ed8f41mukesh agrawal                case WifiLoggerHal.FRAME_TYPE_UNKNOWN:
26870fcb06473cbe2824e401a80c0520bb1c14ed8f41mukesh agrawal                    return "unknown";
26880fcb06473cbe2824e401a80c0520bb1c14ed8f41mukesh agrawal                case WifiLoggerHal.FRAME_TYPE_ETHERNET_II:
26890fcb06473cbe2824e401a80c0520bb1c14ed8f41mukesh agrawal                    return "data";
26900fcb06473cbe2824e401a80c0520bb1c14ed8f41mukesh agrawal                case WifiLoggerHal.FRAME_TYPE_80211_MGMT:
26910fcb06473cbe2824e401a80c0520bb1c14ed8f41mukesh agrawal                    return "802.11 management";
26920fcb06473cbe2824e401a80c0520bb1c14ed8f41mukesh agrawal                default:
26930fcb06473cbe2824e401a80c0520bb1c14ed8f41mukesh agrawal                    return Byte.toString(frameType);
26940fcb06473cbe2824e401a80c0520bb1c14ed8f41mukesh agrawal            }
26950fcb06473cbe2824e401a80c0520bb1c14ed8f41mukesh agrawal        }
269609b1d0786d05436d524d7556c269e665a0962ee6mukesh agrawal    }
269709b1d0786d05436d524d7556c269e665a0962ee6mukesh agrawal
269809b1d0786d05436d524d7556c269e665a0962ee6mukesh agrawal    /**
269909b1d0786d05436d524d7556c269e665a0962ee6mukesh agrawal     * Represents the fate information for one outbound packet.
270009b1d0786d05436d524d7556c269e665a0962ee6mukesh agrawal     */
270109b1d0786d05436d524d7556c269e665a0962ee6mukesh agrawal    @Immutable
270209b1d0786d05436d524d7556c269e665a0962ee6mukesh agrawal    public static final class TxFateReport extends FateReport {
270309b1d0786d05436d524d7556c269e665a0962ee6mukesh agrawal        TxFateReport(byte fate, long driverTimestampUSec, byte frameType, byte[] frameBytes) {
270409b1d0786d05436d524d7556c269e665a0962ee6mukesh agrawal            super(fate, driverTimestampUSec, frameType, frameBytes);
270509b1d0786d05436d524d7556c269e665a0962ee6mukesh agrawal        }
27060fcb06473cbe2824e401a80c0520bb1c14ed8f41mukesh agrawal
27070fcb06473cbe2824e401a80c0520bb1c14ed8f41mukesh agrawal        @Override
27080fcb06473cbe2824e401a80c0520bb1c14ed8f41mukesh agrawal        protected String directionToString() {
27090fcb06473cbe2824e401a80c0520bb1c14ed8f41mukesh agrawal            return "TX";
27100fcb06473cbe2824e401a80c0520bb1c14ed8f41mukesh agrawal        }
27110fcb06473cbe2824e401a80c0520bb1c14ed8f41mukesh agrawal
27120fcb06473cbe2824e401a80c0520bb1c14ed8f41mukesh agrawal        @Override
27130fcb06473cbe2824e401a80c0520bb1c14ed8f41mukesh agrawal        protected String fateToString() {
27140fcb06473cbe2824e401a80c0520bb1c14ed8f41mukesh agrawal            switch (mFate) {
27150fcb06473cbe2824e401a80c0520bb1c14ed8f41mukesh agrawal                case WifiLoggerHal.TX_PKT_FATE_ACKED:
27160fcb06473cbe2824e401a80c0520bb1c14ed8f41mukesh agrawal                    return "acked";
27170fcb06473cbe2824e401a80c0520bb1c14ed8f41mukesh agrawal                case WifiLoggerHal.TX_PKT_FATE_SENT:
27180fcb06473cbe2824e401a80c0520bb1c14ed8f41mukesh agrawal                    return "sent";
27190fcb06473cbe2824e401a80c0520bb1c14ed8f41mukesh agrawal                case WifiLoggerHal.TX_PKT_FATE_FW_QUEUED:
27200fcb06473cbe2824e401a80c0520bb1c14ed8f41mukesh agrawal                    return "firmware queued";
27210fcb06473cbe2824e401a80c0520bb1c14ed8f41mukesh agrawal                case WifiLoggerHal.TX_PKT_FATE_FW_DROP_INVALID:
27220fcb06473cbe2824e401a80c0520bb1c14ed8f41mukesh agrawal                    return "firmware dropped (invalid frame)";
27230fcb06473cbe2824e401a80c0520bb1c14ed8f41mukesh agrawal                case WifiLoggerHal.TX_PKT_FATE_FW_DROP_NOBUFS:
27240fcb06473cbe2824e401a80c0520bb1c14ed8f41mukesh agrawal                    return "firmware dropped (no bufs)";
27250fcb06473cbe2824e401a80c0520bb1c14ed8f41mukesh agrawal                case WifiLoggerHal.TX_PKT_FATE_FW_DROP_OTHER:
27260fcb06473cbe2824e401a80c0520bb1c14ed8f41mukesh agrawal                    return "firmware dropped (other)";
27270fcb06473cbe2824e401a80c0520bb1c14ed8f41mukesh agrawal                case WifiLoggerHal.TX_PKT_FATE_DRV_QUEUED:
27280fcb06473cbe2824e401a80c0520bb1c14ed8f41mukesh agrawal                    return "driver queued";
27290fcb06473cbe2824e401a80c0520bb1c14ed8f41mukesh agrawal                case WifiLoggerHal.TX_PKT_FATE_DRV_DROP_INVALID:
27300fcb06473cbe2824e401a80c0520bb1c14ed8f41mukesh agrawal                    return "driver dropped (invalid frame)";
27310fcb06473cbe2824e401a80c0520bb1c14ed8f41mukesh agrawal                case WifiLoggerHal.TX_PKT_FATE_DRV_DROP_NOBUFS:
27320fcb06473cbe2824e401a80c0520bb1c14ed8f41mukesh agrawal                    return "driver dropped (no bufs)";
27330fcb06473cbe2824e401a80c0520bb1c14ed8f41mukesh agrawal                case WifiLoggerHal.TX_PKT_FATE_DRV_DROP_OTHER:
27340fcb06473cbe2824e401a80c0520bb1c14ed8f41mukesh agrawal                    return "driver dropped (other)";
27350fcb06473cbe2824e401a80c0520bb1c14ed8f41mukesh agrawal                default:
27360fcb06473cbe2824e401a80c0520bb1c14ed8f41mukesh agrawal                    return Byte.toString(mFate);
27370fcb06473cbe2824e401a80c0520bb1c14ed8f41mukesh agrawal            }
27380fcb06473cbe2824e401a80c0520bb1c14ed8f41mukesh agrawal        }
273909b1d0786d05436d524d7556c269e665a0962ee6mukesh agrawal    }
274009b1d0786d05436d524d7556c269e665a0962ee6mukesh agrawal
274109b1d0786d05436d524d7556c269e665a0962ee6mukesh agrawal    /**
274209b1d0786d05436d524d7556c269e665a0962ee6mukesh agrawal     * Represents the fate information for one inbound packet.
274309b1d0786d05436d524d7556c269e665a0962ee6mukesh agrawal     */
274409b1d0786d05436d524d7556c269e665a0962ee6mukesh agrawal    @Immutable
274509b1d0786d05436d524d7556c269e665a0962ee6mukesh agrawal    public static final class RxFateReport extends FateReport {
274609b1d0786d05436d524d7556c269e665a0962ee6mukesh agrawal        RxFateReport(byte fate, long driverTimestampUSec, byte frameType, byte[] frameBytes) {
274709b1d0786d05436d524d7556c269e665a0962ee6mukesh agrawal            super(fate, driverTimestampUSec, frameType, frameBytes);
274809b1d0786d05436d524d7556c269e665a0962ee6mukesh agrawal        }
27490fcb06473cbe2824e401a80c0520bb1c14ed8f41mukesh agrawal
27500fcb06473cbe2824e401a80c0520bb1c14ed8f41mukesh agrawal        @Override
27510fcb06473cbe2824e401a80c0520bb1c14ed8f41mukesh agrawal        protected String directionToString() {
27520fcb06473cbe2824e401a80c0520bb1c14ed8f41mukesh agrawal            return "RX";
27530fcb06473cbe2824e401a80c0520bb1c14ed8f41mukesh agrawal        }
27540fcb06473cbe2824e401a80c0520bb1c14ed8f41mukesh agrawal
27550fcb06473cbe2824e401a80c0520bb1c14ed8f41mukesh agrawal        @Override
27560fcb06473cbe2824e401a80c0520bb1c14ed8f41mukesh agrawal        protected String fateToString() {
27570fcb06473cbe2824e401a80c0520bb1c14ed8f41mukesh agrawal            switch (mFate) {
27580fcb06473cbe2824e401a80c0520bb1c14ed8f41mukesh agrawal                case WifiLoggerHal.RX_PKT_FATE_SUCCESS:
27590fcb06473cbe2824e401a80c0520bb1c14ed8f41mukesh agrawal                    return "success";
27600fcb06473cbe2824e401a80c0520bb1c14ed8f41mukesh agrawal                case WifiLoggerHal.RX_PKT_FATE_FW_QUEUED:
27610fcb06473cbe2824e401a80c0520bb1c14ed8f41mukesh agrawal                    return "firmware queued";
27620fcb06473cbe2824e401a80c0520bb1c14ed8f41mukesh agrawal                case WifiLoggerHal.RX_PKT_FATE_FW_DROP_FILTER:
27630fcb06473cbe2824e401a80c0520bb1c14ed8f41mukesh agrawal                    return "firmware dropped (filter)";
27640fcb06473cbe2824e401a80c0520bb1c14ed8f41mukesh agrawal                case WifiLoggerHal.RX_PKT_FATE_FW_DROP_INVALID:
27650fcb06473cbe2824e401a80c0520bb1c14ed8f41mukesh agrawal                    return "firmware dropped (invalid frame)";
27660fcb06473cbe2824e401a80c0520bb1c14ed8f41mukesh agrawal                case WifiLoggerHal.RX_PKT_FATE_FW_DROP_NOBUFS:
27670fcb06473cbe2824e401a80c0520bb1c14ed8f41mukesh agrawal                    return "firmware dropped (no bufs)";
27680fcb06473cbe2824e401a80c0520bb1c14ed8f41mukesh agrawal                case WifiLoggerHal.RX_PKT_FATE_FW_DROP_OTHER:
27690fcb06473cbe2824e401a80c0520bb1c14ed8f41mukesh agrawal                    return "firmware dropped (other)";
27700fcb06473cbe2824e401a80c0520bb1c14ed8f41mukesh agrawal                case WifiLoggerHal.RX_PKT_FATE_DRV_QUEUED:
27710fcb06473cbe2824e401a80c0520bb1c14ed8f41mukesh agrawal                    return "driver queued";
27720fcb06473cbe2824e401a80c0520bb1c14ed8f41mukesh agrawal                case WifiLoggerHal.RX_PKT_FATE_DRV_DROP_FILTER:
27730fcb06473cbe2824e401a80c0520bb1c14ed8f41mukesh agrawal                    return "driver dropped (filter)";
27740fcb06473cbe2824e401a80c0520bb1c14ed8f41mukesh agrawal                case WifiLoggerHal.RX_PKT_FATE_DRV_DROP_INVALID:
27750fcb06473cbe2824e401a80c0520bb1c14ed8f41mukesh agrawal                    return "driver dropped (invalid frame)";
27760fcb06473cbe2824e401a80c0520bb1c14ed8f41mukesh agrawal                case WifiLoggerHal.RX_PKT_FATE_DRV_DROP_NOBUFS:
27770fcb06473cbe2824e401a80c0520bb1c14ed8f41mukesh agrawal                    return "driver dropped (no bufs)";
27780fcb06473cbe2824e401a80c0520bb1c14ed8f41mukesh agrawal                case WifiLoggerHal.RX_PKT_FATE_DRV_DROP_OTHER:
27790fcb06473cbe2824e401a80c0520bb1c14ed8f41mukesh agrawal                    return "driver dropped (other)";
27800fcb06473cbe2824e401a80c0520bb1c14ed8f41mukesh agrawal                default:
27810fcb06473cbe2824e401a80c0520bb1c14ed8f41mukesh agrawal                    return Byte.toString(mFate);
27820fcb06473cbe2824e401a80c0520bb1c14ed8f41mukesh agrawal            }
27830fcb06473cbe2824e401a80c0520bb1c14ed8f41mukesh agrawal        }
278409b1d0786d05436d524d7556c269e665a0962ee6mukesh agrawal    }
278509b1d0786d05436d524d7556c269e665a0962ee6mukesh agrawal
278609b1d0786d05436d524d7556c269e665a0962ee6mukesh agrawal    private static native int startPktFateMonitoringNative(int iface);
27870fcb06473cbe2824e401a80c0520bb1c14ed8f41mukesh agrawal    /**
27880fcb06473cbe2824e401a80c0520bb1c14ed8f41mukesh agrawal     * Ask the HAL to enable packet fate monitoring. Fails unless HAL is started.
27890fcb06473cbe2824e401a80c0520bb1c14ed8f41mukesh agrawal     */
27900fcb06473cbe2824e401a80c0520bb1c14ed8f41mukesh agrawal    public boolean startPktFateMonitoring() {
27910fcb06473cbe2824e401a80c0520bb1c14ed8f41mukesh agrawal        synchronized (sLock) {
27920fcb06473cbe2824e401a80c0520bb1c14ed8f41mukesh agrawal            if (isHalStarted()) {
27930fcb06473cbe2824e401a80c0520bb1c14ed8f41mukesh agrawal                return startPktFateMonitoringNative(sWlan0Index) == WIFI_SUCCESS;
27940fcb06473cbe2824e401a80c0520bb1c14ed8f41mukesh agrawal            } else {
27950fcb06473cbe2824e401a80c0520bb1c14ed8f41mukesh agrawal                return false;
27960fcb06473cbe2824e401a80c0520bb1c14ed8f41mukesh agrawal            }
27970fcb06473cbe2824e401a80c0520bb1c14ed8f41mukesh agrawal        }
27980fcb06473cbe2824e401a80c0520bb1c14ed8f41mukesh agrawal    }
27990fcb06473cbe2824e401a80c0520bb1c14ed8f41mukesh agrawal
280009b1d0786d05436d524d7556c269e665a0962ee6mukesh agrawal    private static native int getTxPktFatesNative(int iface, TxFateReport[] reportBufs);
28010fcb06473cbe2824e401a80c0520bb1c14ed8f41mukesh agrawal    /**
28020fcb06473cbe2824e401a80c0520bb1c14ed8f41mukesh agrawal     * Fetch the most recent TX packet fates from the HAL. Fails unless HAL is started.
28030fcb06473cbe2824e401a80c0520bb1c14ed8f41mukesh agrawal     */
28040fcb06473cbe2824e401a80c0520bb1c14ed8f41mukesh agrawal    public boolean getTxPktFates(TxFateReport[] reportBufs) {
28050fcb06473cbe2824e401a80c0520bb1c14ed8f41mukesh agrawal        synchronized (sLock) {
28060fcb06473cbe2824e401a80c0520bb1c14ed8f41mukesh agrawal            if (isHalStarted()) {
28070fcb06473cbe2824e401a80c0520bb1c14ed8f41mukesh agrawal                int res = getTxPktFatesNative(sWlan0Index, reportBufs);
28080fcb06473cbe2824e401a80c0520bb1c14ed8f41mukesh agrawal                if (res != WIFI_SUCCESS) {
28090fcb06473cbe2824e401a80c0520bb1c14ed8f41mukesh agrawal                    Log.e(TAG, "getTxPktFatesNative returned " + res);
28100fcb06473cbe2824e401a80c0520bb1c14ed8f41mukesh agrawal                    return false;
28110fcb06473cbe2824e401a80c0520bb1c14ed8f41mukesh agrawal                } else {
28120fcb06473cbe2824e401a80c0520bb1c14ed8f41mukesh agrawal                    return true;
28130fcb06473cbe2824e401a80c0520bb1c14ed8f41mukesh agrawal                }
28140fcb06473cbe2824e401a80c0520bb1c14ed8f41mukesh agrawal            } else {
28150fcb06473cbe2824e401a80c0520bb1c14ed8f41mukesh agrawal                return false;
28160fcb06473cbe2824e401a80c0520bb1c14ed8f41mukesh agrawal            }
28170fcb06473cbe2824e401a80c0520bb1c14ed8f41mukesh agrawal        }
28180fcb06473cbe2824e401a80c0520bb1c14ed8f41mukesh agrawal    }
28190fcb06473cbe2824e401a80c0520bb1c14ed8f41mukesh agrawal
282009b1d0786d05436d524d7556c269e665a0962ee6mukesh agrawal    private static native int getRxPktFatesNative(int iface, RxFateReport[] reportBufs);
28210fcb06473cbe2824e401a80c0520bb1c14ed8f41mukesh agrawal    /**
28220fcb06473cbe2824e401a80c0520bb1c14ed8f41mukesh agrawal     * Fetch the most recent RX packet fates from the HAL. Fails unless HAL is started.
28230fcb06473cbe2824e401a80c0520bb1c14ed8f41mukesh agrawal     */
28240fcb06473cbe2824e401a80c0520bb1c14ed8f41mukesh agrawal    public boolean getRxPktFates(RxFateReport[] reportBufs) {
28250fcb06473cbe2824e401a80c0520bb1c14ed8f41mukesh agrawal        synchronized (sLock) {
28260fcb06473cbe2824e401a80c0520bb1c14ed8f41mukesh agrawal            if (isHalStarted()) {
28270fcb06473cbe2824e401a80c0520bb1c14ed8f41mukesh agrawal                int res = getRxPktFatesNative(sWlan0Index, reportBufs);
28280fcb06473cbe2824e401a80c0520bb1c14ed8f41mukesh agrawal                if (res != WIFI_SUCCESS) {
28290fcb06473cbe2824e401a80c0520bb1c14ed8f41mukesh agrawal                    Log.e(TAG, "getRxPktFatesNative returned " + res);
28300fcb06473cbe2824e401a80c0520bb1c14ed8f41mukesh agrawal                    return false;
28310fcb06473cbe2824e401a80c0520bb1c14ed8f41mukesh agrawal                } else {
28320fcb06473cbe2824e401a80c0520bb1c14ed8f41mukesh agrawal                    return true;
28330fcb06473cbe2824e401a80c0520bb1c14ed8f41mukesh agrawal                }
28340fcb06473cbe2824e401a80c0520bb1c14ed8f41mukesh agrawal            } else {
28350fcb06473cbe2824e401a80c0520bb1c14ed8f41mukesh agrawal                return false;
28360fcb06473cbe2824e401a80c0520bb1c14ed8f41mukesh agrawal            }
28370fcb06473cbe2824e401a80c0520bb1c14ed8f41mukesh agrawal        }
28380fcb06473cbe2824e401a80c0520bb1c14ed8f41mukesh agrawal    }
283909b1d0786d05436d524d7556c269e665a0962ee6mukesh agrawal
284009b1d0786d05436d524d7556c269e665a0962ee6mukesh agrawal    //---------------------------------------------------------------------------------
28415c3c06082b24f9ff0d479e82a63b52220c86598bRoshan Pius    /* Configure ePNO/PNO */
28429bd01d6cb65ea96d6729d35a1fc43bb2bb40e3bdRoshan Pius    private static PnoEventHandler sPnoEventHandler;
28439bd01d6cb65ea96d6729d35a1fc43bb2bb40e3bdRoshan Pius    private static int sPnoCmdId = 0;
2844dd490cf1df37b70a8625dc4ec8e712f740dd1e4aPierre Vandwalle
28459bd01d6cb65ea96d6729d35a1fc43bb2bb40e3bdRoshan Pius    private static native boolean setPnoListNative(int iface, int id, PnoSettings settings);
284618786eca942042388748b0d98979f21c9dff4a89Mitchell Wills
28475c3c06082b24f9ff0d479e82a63b52220c86598bRoshan Pius    /**
28489bd01d6cb65ea96d6729d35a1fc43bb2bb40e3bdRoshan Pius     * Set the PNO settings & the network list in HAL to start PNO.
28499bd01d6cb65ea96d6729d35a1fc43bb2bb40e3bdRoshan Pius     * @param settings PNO settings and network list.
28509bd01d6cb65ea96d6729d35a1fc43bb2bb40e3bdRoshan Pius     * @param eventHandler Handler to receive notifications back during PNO scan.
28519bd01d6cb65ea96d6729d35a1fc43bb2bb40e3bdRoshan Pius     * @return true if success, false otherwise
28525c3c06082b24f9ff0d479e82a63b52220c86598bRoshan Pius     */
28539bd01d6cb65ea96d6729d35a1fc43bb2bb40e3bdRoshan Pius    public boolean setPnoList(PnoSettings settings, PnoEventHandler eventHandler) {
28549bd01d6cb65ea96d6729d35a1fc43bb2bb40e3bdRoshan Pius        Log.e(TAG, "setPnoList cmd " + sPnoCmdId);
2855dd490cf1df37b70a8625dc4ec8e712f740dd1e4aPierre Vandwalle
28569bd01d6cb65ea96d6729d35a1fc43bb2bb40e3bdRoshan Pius        synchronized (sLock) {
28579bd01d6cb65ea96d6729d35a1fc43bb2bb40e3bdRoshan Pius            if (isHalStarted()) {
28589bd01d6cb65ea96d6729d35a1fc43bb2bb40e3bdRoshan Pius                sPnoCmdId = getNewCmdIdLocked();
28599bd01d6cb65ea96d6729d35a1fc43bb2bb40e3bdRoshan Pius                sPnoEventHandler = eventHandler;
28609bd01d6cb65ea96d6729d35a1fc43bb2bb40e3bdRoshan Pius                if (setPnoListNative(sWlan0Index, sPnoCmdId, settings)) {
28619bd01d6cb65ea96d6729d35a1fc43bb2bb40e3bdRoshan Pius                    return true;
28629bd01d6cb65ea96d6729d35a1fc43bb2bb40e3bdRoshan Pius                }
2863dd490cf1df37b70a8625dc4ec8e712f740dd1e4aPierre Vandwalle            }
28649bd01d6cb65ea96d6729d35a1fc43bb2bb40e3bdRoshan Pius            sPnoEventHandler = null;
28659bd01d6cb65ea96d6729d35a1fc43bb2bb40e3bdRoshan Pius            return false;
2866dd490cf1df37b70a8625dc4ec8e712f740dd1e4aPierre Vandwalle        }
2867dd490cf1df37b70a8625dc4ec8e712f740dd1e4aPierre Vandwalle    }
2868dd490cf1df37b70a8625dc4ec8e712f740dd1e4aPierre Vandwalle
28699bd01d6cb65ea96d6729d35a1fc43bb2bb40e3bdRoshan Pius    /**
28709bd01d6cb65ea96d6729d35a1fc43bb2bb40e3bdRoshan Pius     * Set the PNO network list in HAL to start PNO.
28719bd01d6cb65ea96d6729d35a1fc43bb2bb40e3bdRoshan Pius     * @param list PNO network list.
28729bd01d6cb65ea96d6729d35a1fc43bb2bb40e3bdRoshan Pius     * @param eventHandler Handler to receive notifications back during PNO scan.
28739bd01d6cb65ea96d6729d35a1fc43bb2bb40e3bdRoshan Pius     * @return true if success, false otherwise
28749bd01d6cb65ea96d6729d35a1fc43bb2bb40e3bdRoshan Pius     */
28759bd01d6cb65ea96d6729d35a1fc43bb2bb40e3bdRoshan Pius    public boolean setPnoList(PnoNetwork[] list, PnoEventHandler eventHandler) {
28769bd01d6cb65ea96d6729d35a1fc43bb2bb40e3bdRoshan Pius        PnoSettings settings = new PnoSettings();
28779bd01d6cb65ea96d6729d35a1fc43bb2bb40e3bdRoshan Pius        settings.networkList = list;
28789bd01d6cb65ea96d6729d35a1fc43bb2bb40e3bdRoshan Pius        return setPnoList(settings, eventHandler);
2879dd490cf1df37b70a8625dc4ec8e712f740dd1e4aPierre Vandwalle    }
2880dd490cf1df37b70a8625dc4ec8e712f740dd1e4aPierre Vandwalle
28819bd01d6cb65ea96d6729d35a1fc43bb2bb40e3bdRoshan Pius    private static native boolean resetPnoListNative(int iface, int id);
2882dd490cf1df37b70a8625dc4ec8e712f740dd1e4aPierre Vandwalle
28839bd01d6cb65ea96d6729d35a1fc43bb2bb40e3bdRoshan Pius    /**
28849bd01d6cb65ea96d6729d35a1fc43bb2bb40e3bdRoshan Pius     * Reset the PNO settings in HAL to stop PNO.
28859bd01d6cb65ea96d6729d35a1fc43bb2bb40e3bdRoshan Pius     * @return true if success, false otherwise
28869bd01d6cb65ea96d6729d35a1fc43bb2bb40e3bdRoshan Pius     */
28879bd01d6cb65ea96d6729d35a1fc43bb2bb40e3bdRoshan Pius    public boolean resetPnoList() {
28889bd01d6cb65ea96d6729d35a1fc43bb2bb40e3bdRoshan Pius        Log.e(TAG, "resetPnoList cmd " + sPnoCmdId);
2889dd490cf1df37b70a8625dc4ec8e712f740dd1e4aPierre Vandwalle
2890956f54b391677d78379729dd14518edddf3c7660Etan Cohen        synchronized (sLock) {
289171d0be16c79791a005d60f9d5fa7d2d81e1f8c80xinhe            if (isHalStarted()) {
2892f49a59bda006b13e0118d144e0a4a5f569b2251eVinit Deshpande                sPnoCmdId = getNewCmdIdLocked();
28939bd01d6cb65ea96d6729d35a1fc43bb2bb40e3bdRoshan Pius                sPnoEventHandler = null;
28949bd01d6cb65ea96d6729d35a1fc43bb2bb40e3bdRoshan Pius                if (resetPnoListNative(sWlan0Index, sPnoCmdId)) {
2895f49a59bda006b13e0118d144e0a4a5f569b2251eVinit Deshpande                    return true;
2896f49a59bda006b13e0118d144e0a4a5f569b2251eVinit Deshpande                }
2897dd490cf1df37b70a8625dc4ec8e712f740dd1e4aPierre Vandwalle            }
2898f49a59bda006b13e0118d144e0a4a5f569b2251eVinit Deshpande            return false;
2899dd490cf1df37b70a8625dc4ec8e712f740dd1e4aPierre Vandwalle        }
2900dd490cf1df37b70a8625dc4ec8e712f740dd1e4aPierre Vandwalle    }
2901dd490cf1df37b70a8625dc4ec8e712f740dd1e4aPierre Vandwalle
290218786eca942042388748b0d98979f21c9dff4a89Mitchell Wills    // Callback from native
29033571366ac36c70746b9f013ec2b54482861c9292Randy Pan    private static void onPnoNetworkFound(int id, ScanResult[] results, int[] beaconCaps) {
2904dd490cf1df37b70a8625dc4ec8e712f740dd1e4aPierre Vandwalle        if (results == null) {
2905dd490cf1df37b70a8625dc4ec8e712f740dd1e4aPierre Vandwalle            Log.e(TAG, "onPnoNetworkFound null results");
2906dd490cf1df37b70a8625dc4ec8e712f740dd1e4aPierre Vandwalle            return;
2907dd490cf1df37b70a8625dc4ec8e712f740dd1e4aPierre Vandwalle
2908dd490cf1df37b70a8625dc4ec8e712f740dd1e4aPierre Vandwalle        }
2909dd490cf1df37b70a8625dc4ec8e712f740dd1e4aPierre Vandwalle        Log.d(TAG, "WifiNative.onPnoNetworkFound result " + results.length);
2910dd490cf1df37b70a8625dc4ec8e712f740dd1e4aPierre Vandwalle
29119bd01d6cb65ea96d6729d35a1fc43bb2bb40e3bdRoshan Pius        PnoEventHandler handler = sPnoEventHandler;
291218786eca942042388748b0d98979f21c9dff4a89Mitchell Wills        if (sPnoCmdId != 0 && handler != null) {
291318786eca942042388748b0d98979f21c9dff4a89Mitchell Wills            for (int i=0; i<results.length; i++) {
291418786eca942042388748b0d98979f21c9dff4a89Mitchell Wills                Log.e(TAG, "onPnoNetworkFound SSID " + results[i].SSID
291518786eca942042388748b0d98979f21c9dff4a89Mitchell Wills                        + " " + results[i].level + " " + results[i].frequency);
2916dd490cf1df37b70a8625dc4ec8e712f740dd1e4aPierre Vandwalle
29171ab129e587d334a144a0bca5323c27985397a403Randy Pan                populateScanResult(results[i], beaconCaps[i], "onPnoNetworkFound ");
291818786eca942042388748b0d98979f21c9dff4a89Mitchell Wills                results[i].wifiSsid = WifiSsid.createFromAsciiEncoded(results[i].SSID);
2919dd490cf1df37b70a8625dc4ec8e712f740dd1e4aPierre Vandwalle            }
292018786eca942042388748b0d98979f21c9dff4a89Mitchell Wills
292118786eca942042388748b0d98979f21c9dff4a89Mitchell Wills            handler.onPnoNetworkFound(results);
292218786eca942042388748b0d98979f21c9dff4a89Mitchell Wills        } else {
292318786eca942042388748b0d98979f21c9dff4a89Mitchell Wills            /* this can happen because of race conditions */
292418786eca942042388748b0d98979f21c9dff4a89Mitchell Wills            Log.d(TAG, "Ignoring Pno Network found event");
2925dd490cf1df37b70a8625dc4ec8e712f740dd1e4aPierre Vandwalle        }
2926d4c25fd76644d5490700ae69fada4669de5193a1Pierre Vandwalle    }
2927d4c25fd76644d5490700ae69fada4669de5193a1Pierre Vandwalle
29289ccffbb6c3fe8eaa7d70cc3076d0dc3c6a9073e7Pierre Vandwalle    private native static boolean setBssidBlacklistNative(int iface, int id,
29299ccffbb6c3fe8eaa7d70cc3076d0dc3c6a9073e7Pierre Vandwalle                                              String list[]);
29309ccffbb6c3fe8eaa7d70cc3076d0dc3c6a9073e7Pierre Vandwalle
293118786eca942042388748b0d98979f21c9dff4a89Mitchell Wills    public boolean setBssidBlacklist(String list[]) {
29329ccffbb6c3fe8eaa7d70cc3076d0dc3c6a9073e7Pierre Vandwalle        int size = 0;
29339ccffbb6c3fe8eaa7d70cc3076d0dc3c6a9073e7Pierre Vandwalle        if (list != null) {
29349ccffbb6c3fe8eaa7d70cc3076d0dc3c6a9073e7Pierre Vandwalle            size = list.length;
29359ccffbb6c3fe8eaa7d70cc3076d0dc3c6a9073e7Pierre Vandwalle        }
29369ccffbb6c3fe8eaa7d70cc3076d0dc3c6a9073e7Pierre Vandwalle        Log.e(TAG, "setBssidBlacklist cmd " + sPnoCmdId + " size " + size);
29379ccffbb6c3fe8eaa7d70cc3076d0dc3c6a9073e7Pierre Vandwalle
2938956f54b391677d78379729dd14518edddf3c7660Etan Cohen        synchronized (sLock) {
293971d0be16c79791a005d60f9d5fa7d2d81e1f8c80xinhe            if (isHalStarted()) {
2940f49a59bda006b13e0118d144e0a4a5f569b2251eVinit Deshpande                sPnoCmdId = getNewCmdIdLocked();
2941f49a59bda006b13e0118d144e0a4a5f569b2251eVinit Deshpande                return setBssidBlacklistNative(sWlan0Index, sPnoCmdId, list);
2942f49a59bda006b13e0118d144e0a4a5f569b2251eVinit Deshpande            } else {
29439ccffbb6c3fe8eaa7d70cc3076d0dc3c6a9073e7Pierre Vandwalle                return false;
29449ccffbb6c3fe8eaa7d70cc3076d0dc3c6a9073e7Pierre Vandwalle            }
29459ccffbb6c3fe8eaa7d70cc3076d0dc3c6a9073e7Pierre Vandwalle        }
29469ccffbb6c3fe8eaa7d70cc3076d0dc3c6a9073e7Pierre Vandwalle    }
29479ccffbb6c3fe8eaa7d70cc3076d0dc3c6a9073e7Pierre Vandwalle
2948c55e88163b223db0ca6a99ed6ffe91845c30a576Prerepa Viswanadham    private native static int startSendingOffloadedPacketNative(int iface, int idx,
2949c55e88163b223db0ca6a99ed6ffe91845c30a576Prerepa Viswanadham                                    byte[] srcMac, byte[] dstMac, byte[] pktData, int period);
2950c55e88163b223db0ca6a99ed6ffe91845c30a576Prerepa Viswanadham
295118786eca942042388748b0d98979f21c9dff4a89Mitchell Wills    public int
2952c55e88163b223db0ca6a99ed6ffe91845c30a576Prerepa Viswanadham    startSendingOffloadedPacket(int slot, KeepalivePacketData keepAlivePacket, int period) {
2953c55e88163b223db0ca6a99ed6ffe91845c30a576Prerepa Viswanadham        Log.d(TAG, "startSendingOffloadedPacket slot=" + slot + " period=" + period);
29543bd22cebc41ed0786d0e7bd2970a634a8bb0093ePrerepa Viswanadham
29553bd22cebc41ed0786d0e7bd2970a634a8bb0093ePrerepa Viswanadham        String[] macAddrStr = getMacAddress().split(":");
29563bd22cebc41ed0786d0e7bd2970a634a8bb0093ePrerepa Viswanadham        byte[] srcMac = new byte[6];
29573bd22cebc41ed0786d0e7bd2970a634a8bb0093ePrerepa Viswanadham        for(int i = 0; i < 6; i++) {
29583bd22cebc41ed0786d0e7bd2970a634a8bb0093ePrerepa Viswanadham            Integer hexVal = Integer.parseInt(macAddrStr[i], 16);
29593bd22cebc41ed0786d0e7bd2970a634a8bb0093ePrerepa Viswanadham            srcMac[i] = hexVal.byteValue();
29603bd22cebc41ed0786d0e7bd2970a634a8bb0093ePrerepa Viswanadham        }
2961956f54b391677d78379729dd14518edddf3c7660Etan Cohen        synchronized (sLock) {
2962c55e88163b223db0ca6a99ed6ffe91845c30a576Prerepa Viswanadham            if (isHalStarted()) {
2963c55e88163b223db0ca6a99ed6ffe91845c30a576Prerepa Viswanadham                return startSendingOffloadedPacketNative(sWlan0Index, slot, srcMac,
296418786eca942042388748b0d98979f21c9dff4a89Mitchell Wills                        keepAlivePacket.dstMac, keepAlivePacket.data, period);
2965c55e88163b223db0ca6a99ed6ffe91845c30a576Prerepa Viswanadham            } else {
2966c55e88163b223db0ca6a99ed6ffe91845c30a576Prerepa Viswanadham                return -1;
2967c55e88163b223db0ca6a99ed6ffe91845c30a576Prerepa Viswanadham            }
2968c55e88163b223db0ca6a99ed6ffe91845c30a576Prerepa Viswanadham        }
2969c55e88163b223db0ca6a99ed6ffe91845c30a576Prerepa Viswanadham    }
2970c55e88163b223db0ca6a99ed6ffe91845c30a576Prerepa Viswanadham
2971c55e88163b223db0ca6a99ed6ffe91845c30a576Prerepa Viswanadham    private native static int stopSendingOffloadedPacketNative(int iface, int idx);
2972c55e88163b223db0ca6a99ed6ffe91845c30a576Prerepa Viswanadham
297318786eca942042388748b0d98979f21c9dff4a89Mitchell Wills    public int
2974c55e88163b223db0ca6a99ed6ffe91845c30a576Prerepa Viswanadham    stopSendingOffloadedPacket(int slot) {
2975c55e88163b223db0ca6a99ed6ffe91845c30a576Prerepa Viswanadham        Log.d(TAG, "stopSendingOffloadedPacket " + slot);
2976956f54b391677d78379729dd14518edddf3c7660Etan Cohen        synchronized (sLock) {
2977c55e88163b223db0ca6a99ed6ffe91845c30a576Prerepa Viswanadham            if (isHalStarted()) {
2978c55e88163b223db0ca6a99ed6ffe91845c30a576Prerepa Viswanadham                return stopSendingOffloadedPacketNative(sWlan0Index, slot);
2979c55e88163b223db0ca6a99ed6ffe91845c30a576Prerepa Viswanadham            } else {
2980c55e88163b223db0ca6a99ed6ffe91845c30a576Prerepa Viswanadham                return -1;
2981c55e88163b223db0ca6a99ed6ffe91845c30a576Prerepa Viswanadham            }
2982c55e88163b223db0ca6a99ed6ffe91845c30a576Prerepa Viswanadham        }
2983c55e88163b223db0ca6a99ed6ffe91845c30a576Prerepa Viswanadham    }
2984aed5cb65d8eefb2eee56a29e33ac3cd8b03312e7Prerepa Viswanadham
2985aed5cb65d8eefb2eee56a29e33ac3cd8b03312e7Prerepa Viswanadham    public static interface WifiRssiEventHandler {
2986aed5cb65d8eefb2eee56a29e33ac3cd8b03312e7Prerepa Viswanadham        void onRssiThresholdBreached(byte curRssi);
2987aed5cb65d8eefb2eee56a29e33ac3cd8b03312e7Prerepa Viswanadham    }
2988aed5cb65d8eefb2eee56a29e33ac3cd8b03312e7Prerepa Viswanadham
2989aed5cb65d8eefb2eee56a29e33ac3cd8b03312e7Prerepa Viswanadham    private static WifiRssiEventHandler sWifiRssiEventHandler;
2990aed5cb65d8eefb2eee56a29e33ac3cd8b03312e7Prerepa Viswanadham
299118786eca942042388748b0d98979f21c9dff4a89Mitchell Wills    // Callback from native
299218786eca942042388748b0d98979f21c9dff4a89Mitchell Wills    private static void onRssiThresholdBreached(int id, byte curRssi) {
299318786eca942042388748b0d98979f21c9dff4a89Mitchell Wills        WifiRssiEventHandler handler = sWifiRssiEventHandler;
299418786eca942042388748b0d98979f21c9dff4a89Mitchell Wills        if (handler != null) {
299518786eca942042388748b0d98979f21c9dff4a89Mitchell Wills            handler.onRssiThresholdBreached(curRssi);
299618786eca942042388748b0d98979f21c9dff4a89Mitchell Wills        }
2997aed5cb65d8eefb2eee56a29e33ac3cd8b03312e7Prerepa Viswanadham    }
2998aed5cb65d8eefb2eee56a29e33ac3cd8b03312e7Prerepa Viswanadham
2999aed5cb65d8eefb2eee56a29e33ac3cd8b03312e7Prerepa Viswanadham    private native static int startRssiMonitoringNative(int iface, int id,
3000aed5cb65d8eefb2eee56a29e33ac3cd8b03312e7Prerepa Viswanadham                                        byte maxRssi, byte minRssi);
3001aed5cb65d8eefb2eee56a29e33ac3cd8b03312e7Prerepa Viswanadham
3002aed5cb65d8eefb2eee56a29e33ac3cd8b03312e7Prerepa Viswanadham    private static int sRssiMonitorCmdId = 0;
3003aed5cb65d8eefb2eee56a29e33ac3cd8b03312e7Prerepa Viswanadham
300418786eca942042388748b0d98979f21c9dff4a89Mitchell Wills    public int startRssiMonitoring(byte maxRssi, byte minRssi,
3005aed5cb65d8eefb2eee56a29e33ac3cd8b03312e7Prerepa Viswanadham                                                WifiRssiEventHandler rssiEventHandler) {
3006aed5cb65d8eefb2eee56a29e33ac3cd8b03312e7Prerepa Viswanadham        Log.d(TAG, "startRssiMonitoring: maxRssi=" + maxRssi + " minRssi=" + minRssi);
3007956f54b391677d78379729dd14518edddf3c7660Etan Cohen        synchronized (sLock) {
300818786eca942042388748b0d98979f21c9dff4a89Mitchell Wills            sWifiRssiEventHandler = rssiEventHandler;
3009aed5cb65d8eefb2eee56a29e33ac3cd8b03312e7Prerepa Viswanadham            if (isHalStarted()) {
3010ed34692d4c4e202748d41f90db397f1cc3dc8320Mitchell Wills                if (sRssiMonitorCmdId != 0) {
3011ed34692d4c4e202748d41f90db397f1cc3dc8320Mitchell Wills                    stopRssiMonitoring();
3012ed34692d4c4e202748d41f90db397f1cc3dc8320Mitchell Wills                }
3013ed34692d4c4e202748d41f90db397f1cc3dc8320Mitchell Wills
3014aed5cb65d8eefb2eee56a29e33ac3cd8b03312e7Prerepa Viswanadham                sRssiMonitorCmdId = getNewCmdIdLocked();
3015aed5cb65d8eefb2eee56a29e33ac3cd8b03312e7Prerepa Viswanadham                Log.d(TAG, "sRssiMonitorCmdId = " + sRssiMonitorCmdId);
3016ed34692d4c4e202748d41f90db397f1cc3dc8320Mitchell Wills                int ret = startRssiMonitoringNative(sWlan0Index, sRssiMonitorCmdId,
3017ed34692d4c4e202748d41f90db397f1cc3dc8320Mitchell Wills                        maxRssi, minRssi);
3018ed34692d4c4e202748d41f90db397f1cc3dc8320Mitchell Wills                if (ret != 0) { // if not success
3019ed34692d4c4e202748d41f90db397f1cc3dc8320Mitchell Wills                    sRssiMonitorCmdId = 0;
3020ed34692d4c4e202748d41f90db397f1cc3dc8320Mitchell Wills                }
3021ed34692d4c4e202748d41f90db397f1cc3dc8320Mitchell Wills                return ret;
3022aed5cb65d8eefb2eee56a29e33ac3cd8b03312e7Prerepa Viswanadham            } else {
3023aed5cb65d8eefb2eee56a29e33ac3cd8b03312e7Prerepa Viswanadham                return -1;
3024aed5cb65d8eefb2eee56a29e33ac3cd8b03312e7Prerepa Viswanadham            }
3025aed5cb65d8eefb2eee56a29e33ac3cd8b03312e7Prerepa Viswanadham        }
3026aed5cb65d8eefb2eee56a29e33ac3cd8b03312e7Prerepa Viswanadham    }
3027aed5cb65d8eefb2eee56a29e33ac3cd8b03312e7Prerepa Viswanadham
3028aed5cb65d8eefb2eee56a29e33ac3cd8b03312e7Prerepa Viswanadham    private native static int stopRssiMonitoringNative(int iface, int idx);
3029aed5cb65d8eefb2eee56a29e33ac3cd8b03312e7Prerepa Viswanadham
303018786eca942042388748b0d98979f21c9dff4a89Mitchell Wills    public int stopRssiMonitoring() {
3031aed5cb65d8eefb2eee56a29e33ac3cd8b03312e7Prerepa Viswanadham        Log.d(TAG, "stopRssiMonitoring, cmdId " + sRssiMonitorCmdId);
3032956f54b391677d78379729dd14518edddf3c7660Etan Cohen        synchronized (sLock) {
3033aed5cb65d8eefb2eee56a29e33ac3cd8b03312e7Prerepa Viswanadham            if (isHalStarted()) {
3034ed34692d4c4e202748d41f90db397f1cc3dc8320Mitchell Wills                int ret = 0;
3035ed34692d4c4e202748d41f90db397f1cc3dc8320Mitchell Wills                if (sRssiMonitorCmdId != 0) {
3036ed34692d4c4e202748d41f90db397f1cc3dc8320Mitchell Wills                    ret = stopRssiMonitoringNative(sWlan0Index, sRssiMonitorCmdId);
3037ed34692d4c4e202748d41f90db397f1cc3dc8320Mitchell Wills                }
3038ed34692d4c4e202748d41f90db397f1cc3dc8320Mitchell Wills                sRssiMonitorCmdId = 0;
3039ed34692d4c4e202748d41f90db397f1cc3dc8320Mitchell Wills                return ret;
3040aed5cb65d8eefb2eee56a29e33ac3cd8b03312e7Prerepa Viswanadham            } else {
3041aed5cb65d8eefb2eee56a29e33ac3cd8b03312e7Prerepa Viswanadham                return -1;
3042aed5cb65d8eefb2eee56a29e33ac3cd8b03312e7Prerepa Viswanadham            }
3043aed5cb65d8eefb2eee56a29e33ac3cd8b03312e7Prerepa Viswanadham        }
3044aed5cb65d8eefb2eee56a29e33ac3cd8b03312e7Prerepa Viswanadham    }
30455ea42964ba17901a8d724736b450ace6ed48880fPrerepa Viswanadham
30465ea42964ba17901a8d724736b450ace6ed48880fPrerepa Viswanadham    private static native WifiWakeReasonAndCounts getWlanWakeReasonCountNative(int iface);
30476bf6986d359556010638dfae332b585162f06520Roshan Pius
30486bf6986d359556010638dfae332b585162f06520Roshan Pius    /**
30496bf6986d359556010638dfae332b585162f06520Roshan Pius     * Fetch the host wakeup reasons stats from wlan driver.
30506bf6986d359556010638dfae332b585162f06520Roshan Pius     * @return the |WifiWakeReasonAndCounts| object retrieved from the wlan driver.
30516bf6986d359556010638dfae332b585162f06520Roshan Pius     */
30526bf6986d359556010638dfae332b585162f06520Roshan Pius    public WifiWakeReasonAndCounts getWlanWakeReasonCount() {
30536bf6986d359556010638dfae332b585162f06520Roshan Pius        Log.d(TAG, "getWlanWakeReasonCount " + sWlan0Index);
30546bf6986d359556010638dfae332b585162f06520Roshan Pius        synchronized (sLock) {
30556bf6986d359556010638dfae332b585162f06520Roshan Pius            if (isHalStarted()) {
30566bf6986d359556010638dfae332b585162f06520Roshan Pius                return getWlanWakeReasonCountNative(sWlan0Index);
30576bf6986d359556010638dfae332b585162f06520Roshan Pius            } else {
30586bf6986d359556010638dfae332b585162f06520Roshan Pius                return null;
30596bf6986d359556010638dfae332b585162f06520Roshan Pius            }
30606bf6986d359556010638dfae332b585162f06520Roshan Pius        }
30616bf6986d359556010638dfae332b585162f06520Roshan Pius    }
30623dd6f75ff2ab823bd0c14581f2e047c74916f16fErik Kline
30633dd6f75ff2ab823bd0c14581f2e047c74916f16fErik Kline    private static native int configureNeighborDiscoveryOffload(int iface, boolean enabled);
30643dd6f75ff2ab823bd0c14581f2e047c74916f16fErik Kline
30653dd6f75ff2ab823bd0c14581f2e047c74916f16fErik Kline    public boolean configureNeighborDiscoveryOffload(boolean enabled) {
30663dd6f75ff2ab823bd0c14581f2e047c74916f16fErik Kline        final String logMsg =  "configureNeighborDiscoveryOffload(" + enabled + ")";
30673dd6f75ff2ab823bd0c14581f2e047c74916f16fErik Kline        Log.d(mTAG, logMsg);
30683dd6f75ff2ab823bd0c14581f2e047c74916f16fErik Kline        synchronized (sLock) {
30693dd6f75ff2ab823bd0c14581f2e047c74916f16fErik Kline            if (isHalStarted()) {
30703dd6f75ff2ab823bd0c14581f2e047c74916f16fErik Kline                final int ret = configureNeighborDiscoveryOffload(sWlan0Index, enabled);
30713dd6f75ff2ab823bd0c14581f2e047c74916f16fErik Kline                if (ret != 0) {
30723dd6f75ff2ab823bd0c14581f2e047c74916f16fErik Kline                    Log.d(mTAG, logMsg + " returned: " + ret);
30733dd6f75ff2ab823bd0c14581f2e047c74916f16fErik Kline                }
30743dd6f75ff2ab823bd0c14581f2e047c74916f16fErik Kline                return (ret == 0);
30753dd6f75ff2ab823bd0c14581f2e047c74916f16fErik Kline            }
30763dd6f75ff2ab823bd0c14581f2e047c74916f16fErik Kline        }
30773dd6f75ff2ab823bd0c14581f2e047c74916f16fErik Kline        return false;
30783dd6f75ff2ab823bd0c14581f2e047c74916f16fErik Kline    }
3079155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande}
3080