WifiNative.java revision 127f7244183786e6ccae09e81eeccdac31973e69
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
19155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpandeimport android.net.wifi.BatchedScanSettings;
20143657392bf0702a155fe688171a5abbf4c86570Vinit Deshpandeimport android.net.wifi.RttManager;
21e4e3750390bec0a849a9153348b7c21b2cc8b843Vinit Deshapndeimport android.net.wifi.ScanResult;
22dd490cf1df37b70a8625dc4ec8e712f740dd1e4aPierre Vandwalleimport android.net.wifi.WifiConfiguration;
23aabe7a9f2f32915fd1f25416a6d2034a844005d6vandwalleimport android.net.wifi.WifiLinkLayerStats;
2403cdd624f5da6d62cb731304aa7505921566f69dDmitry Shmidtimport android.net.wifi.WifiManager;
25e4e3750390bec0a849a9153348b7c21b2cc8b843Vinit Deshapndeimport android.net.wifi.WifiScanner;
2612cf388ecf3b2aa5ba66ed680b931fa356ab75b5xinheimport android.net.wifi.RttManager;
27dd490cf1df37b70a8625dc4ec8e712f740dd1e4aPierre Vandwalleimport android.net.wifi.WifiSsid;
28155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpandeimport android.net.wifi.WpsInfo;
29155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpandeimport android.net.wifi.p2p.WifiP2pConfig;
30155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpandeimport android.net.wifi.p2p.WifiP2pGroup;
3103cdd624f5da6d62cb731304aa7505921566f69dDmitry Shmidtimport android.net.wifi.p2p.nsd.WifiP2pServiceInfo;
32a6777abfc90496801e9942f57fbfa091ba85ae82Prerepa Viswanadhamimport android.net.wifi.WifiEnterpriseConfig;
33f22d23092ab37286a5ef9d257d5bb32c421d2669vandwalleimport android.os.SystemClock;
34155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpandeimport android.text.TextUtils;
3503ba4a5b108ab2bbdb5d0e7be46ac45c00a6d9ffxinheimport android.util.Base64;
36155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpandeimport android.util.LocalLog;
37155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpandeimport android.util.Log;
38155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
3998dca3397c0049d3cfb9c97e2b021c5ee3a2e1bdxinheimport java.io.ByteArrayOutputStream;
4098dca3397c0049d3cfb9c97e2b021c5ee3a2e1bdxinheimport java.io.IOException;
41155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpandeimport java.util.ArrayList;
42155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpandeimport java.util.List;
43155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpandeimport java.util.Locale;
4498dca3397c0049d3cfb9c97e2b021c5ee3a2e1bdxinheimport java.util.zip.Deflater;
45155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
46155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande/**
47155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande * Native calls for bring up/shut down of the supplicant daemon and for
48155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande * sending requests to the supplicant daemon
49155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande *
50155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande * waitForEvent() is called on the monitor thread for events. All other methods
51155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande * must be serialized from the framework.
52155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande *
53155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande * {@hide}
54155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande */
55155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpandepublic class WifiNative {
56155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
57b66b29a00da970ee75052e24f592c8d6c16bd0edxinhe    private static boolean DBG = false;
58155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    private final String mTAG;
59155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    private static final int DEFAULT_GROUP_OWNER_INTENT     = 6;
60155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
61155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    static final int BLUETOOTH_COEXISTENCE_MODE_ENABLED     = 0;
62155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    static final int BLUETOOTH_COEXISTENCE_MODE_DISABLED    = 1;
63155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    static final int BLUETOOTH_COEXISTENCE_MODE_SENSE       = 2;
64155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
65155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    static final int SCAN_WITHOUT_CONNECTION_SETUP          = 1;
66155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    static final int SCAN_WITH_CONNECTION_SETUP             = 2;
67155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
68155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    // Hold this lock before calling supplicant - it is required to
69155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    // mutually exclude access from Wifi and P2p state machines
70155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    static final Object mLock = new Object();
71155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
72155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    public final String mInterfaceName;
73155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    public final String mInterfacePrefix;
74155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
75155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    private boolean mSuspendOptEnabled = false;
76155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
77243931f3474f6235cfcf5c1a55fa2f192ee264aexinhe    private static final int EID_HT_OPERATION = 61;
78243931f3474f6235cfcf5c1a55fa2f192ee264aexinhe    private static final int EID_VHT_OPERATION = 192;
79243931f3474f6235cfcf5c1a55fa2f192ee264aexinhe    private static final int EID_EXTENDED_CAPS = 127;
80243931f3474f6235cfcf5c1a55fa2f192ee264aexinhe    private static final int RTT_RESP_ENABLE_BIT = 70;
81155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    /* Register native functions */
82155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
83155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    static {
84155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        /* Native functions are defined in libwifi-service.so */
85155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        System.loadLibrary("wifi-service");
86155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        registerNatives();
87155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    }
88155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
89155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    private static native int registerNatives();
90155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
91155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    public native static boolean loadDriver();
92155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
93155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    public native static boolean isDriverLoaded();
94155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
95155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    public native static boolean unloadDriver();
96155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
97155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    public native static boolean startSupplicant(boolean p2pSupported);
98155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
99155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    /* Sends a kill signal to supplicant. To be used when we have lost connection
100155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande       or when the supplicant is hung */
101155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    public native static boolean killSupplicant(boolean p2pSupported);
102155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
103155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    private native boolean connectToSupplicantNative();
104155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
105155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    private native void closeSupplicantConnectionNative();
106155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
107155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    /**
108155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande     * Wait for the supplicant to send an event, returning the event string.
109155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande     * @return the event string sent by the supplicant.
110155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande     */
111155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    private native String waitForEventNative();
112155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
113155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    private native boolean doBooleanCommandNative(String command);
114155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
115155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    private native int doIntCommandNative(String command);
116155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
117155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    private native String doStringCommandNative(String command);
118155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
119155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    public WifiNative(String interfaceName) {
120155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        mInterfaceName = interfaceName;
121155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        mTAG = "WifiNative-" + interfaceName;
122155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        if (!interfaceName.equals("p2p0")) {
123155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande            mInterfacePrefix = "IFNAME=" + interfaceName + " ";
124155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        } else {
125155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande            // commands for p2p0 interface don't need prefix
126155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande            mInterfacePrefix = "";
127155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        }
128155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    }
129155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
130ed9938883ae2dade81c8be6cd6ceaef3febd5239vandwalle    void enableVerboseLogging(int verbose) {
131ed9938883ae2dade81c8be6cd6ceaef3febd5239vandwalle        if (verbose > 0) {
132ed9938883ae2dade81c8be6cd6ceaef3febd5239vandwalle            DBG = true;
133ed9938883ae2dade81c8be6cd6ceaef3febd5239vandwalle        } else {
134ed9938883ae2dade81c8be6cd6ceaef3febd5239vandwalle            DBG = false;
135ed9938883ae2dade81c8be6cd6ceaef3febd5239vandwalle        }
136ed9938883ae2dade81c8be6cd6ceaef3febd5239vandwalle    }
137ed9938883ae2dade81c8be6cd6ceaef3febd5239vandwalle
1386414301b7b685307e8ae8cea6976ed05cc49b254Vinit Deshpande    private static final LocalLog mLocalLog = new LocalLog(16384);
139155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
140155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    // hold mLock before accessing mCmdIdLock
141b0b1d59786de7ff1fc98e31a16a7d710458bf549Vinit Deshpande    private static int sCmdId;
142155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
14304d29dfffd4c64bc0eb3f109269c1ad2bde6e363Vinit Deshpande    public static LocalLog getLocalLog() {
144155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        return mLocalLog;
145155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    }
146155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
147b0b1d59786de7ff1fc98e31a16a7d710458bf549Vinit Deshpande    private static int getNewCmdIdLocked() {
148b0b1d59786de7ff1fc98e31a16a7d710458bf549Vinit Deshpande        return sCmdId++;
149155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    }
150155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
151155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    private void localLog(String s) {
152155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        if (mLocalLog != null)
153155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande            mLocalLog.log(mInterfaceName + ": " + s);
154155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    }
155155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
156155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    public boolean connectToSupplicant() {
157155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        // No synchronization necessary .. it is implemented in WifiMonitor
158155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        localLog(mInterfacePrefix + "connectToSupplicant");
159155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        return connectToSupplicantNative();
160155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    }
161155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
162155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    public void closeSupplicantConnection() {
163155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        localLog(mInterfacePrefix + "closeSupplicantConnection");
164155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        closeSupplicantConnectionNative();
165155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    }
166155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
167155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    public String waitForEvent() {
168155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        // No synchronization necessary .. it is implemented in WifiMonitor
169155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        return waitForEventNative();
170155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    }
171155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
172155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    private boolean doBooleanCommand(String command) {
173155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        if (DBG) Log.d(mTAG, "doBoolean: " + command);
174155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        synchronized (mLock) {
175155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande            int cmdId = getNewCmdIdLocked();
1767b581f46f6c9bc6edf0edd287d47106712fb2144vandwalle            String toLog = Integer.toString(cmdId) + ":" + mInterfacePrefix + command;
177155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande            boolean result = doBooleanCommandNative(mInterfacePrefix + command);
1787b581f46f6c9bc6edf0edd287d47106712fb2144vandwalle            localLog(toLog + " -> " + result);
1790888ce6f90bdaeee799dd8361ea4781e23a33b87vandwalle            if (DBG) Log.d(mTAG, command + ": returned " + result);
180155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande            return result;
181155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        }
182155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    }
183155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
184a6777abfc90496801e9942f57fbfa091ba85ae82Prerepa Viswanadham    private boolean doBooleanCommandWithoutLogging(String command) {
185a6777abfc90496801e9942f57fbfa091ba85ae82Prerepa Viswanadham        if (DBG) Log.d(mTAG, "doBooleanCommandWithoutLogging: " + command);
186a6777abfc90496801e9942f57fbfa091ba85ae82Prerepa Viswanadham        synchronized (mLock) {
187a6777abfc90496801e9942f57fbfa091ba85ae82Prerepa Viswanadham            int cmdId = getNewCmdIdLocked();
188a6777abfc90496801e9942f57fbfa091ba85ae82Prerepa Viswanadham            boolean result = doBooleanCommandNative(mInterfacePrefix + command);
189a6777abfc90496801e9942f57fbfa091ba85ae82Prerepa Viswanadham            if (DBG) Log.d(mTAG, command + ": returned " + result);
190a6777abfc90496801e9942f57fbfa091ba85ae82Prerepa Viswanadham            return result;
191a6777abfc90496801e9942f57fbfa091ba85ae82Prerepa Viswanadham        }
192a6777abfc90496801e9942f57fbfa091ba85ae82Prerepa Viswanadham    }
193a6777abfc90496801e9942f57fbfa091ba85ae82Prerepa Viswanadham
194155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    private int doIntCommand(String command) {
195155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        if (DBG) Log.d(mTAG, "doInt: " + command);
196155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        synchronized (mLock) {
197155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande            int cmdId = getNewCmdIdLocked();
1987b581f46f6c9bc6edf0edd287d47106712fb2144vandwalle            String toLog = Integer.toString(cmdId) + ":" + mInterfacePrefix + command;
199155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande            int result = doIntCommandNative(mInterfacePrefix + command);
2007b581f46f6c9bc6edf0edd287d47106712fb2144vandwalle            localLog(toLog + " -> " + result);
201155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande            if (DBG) Log.d(mTAG, "   returned " + result);
202155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande            return result;
203155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        }
204155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    }
205155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
206155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    private String doStringCommand(String command) {
2070888ce6f90bdaeee799dd8361ea4781e23a33b87vandwalle        if (DBG) {
2080888ce6f90bdaeee799dd8361ea4781e23a33b87vandwalle            //GET_NETWORK commands flood the logs
2090888ce6f90bdaeee799dd8361ea4781e23a33b87vandwalle            if (!command.startsWith("GET_NETWORK")) {
2100888ce6f90bdaeee799dd8361ea4781e23a33b87vandwalle                Log.d(mTAG, "doString: [" + command + "]");
2110888ce6f90bdaeee799dd8361ea4781e23a33b87vandwalle            }
2120888ce6f90bdaeee799dd8361ea4781e23a33b87vandwalle        }
213155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        synchronized (mLock) {
214155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande            int cmdId = getNewCmdIdLocked();
2157b581f46f6c9bc6edf0edd287d47106712fb2144vandwalle            String toLog = Integer.toString(cmdId) + ":" + mInterfacePrefix + command;
216155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande            String result = doStringCommandNative(mInterfacePrefix + command);
21740ff222cec1bd05879edb53abc75c6deead734cavandwalle            if (result == null) {
21840ff222cec1bd05879edb53abc75c6deead734cavandwalle                if (DBG) Log.d(mTAG, "doStringCommandNative no result");
21940ff222cec1bd05879edb53abc75c6deead734cavandwalle            } else {
2207b581f46f6c9bc6edf0edd287d47106712fb2144vandwalle                if (!command.startsWith("STATUS-")) {
2217b581f46f6c9bc6edf0edd287d47106712fb2144vandwalle                    localLog(toLog + " -> " + result);
2227b581f46f6c9bc6edf0edd287d47106712fb2144vandwalle                }
22340ff222cec1bd05879edb53abc75c6deead734cavandwalle                if (DBG) Log.d(mTAG, "   returned " + result.replace("\n", " "));
22440ff222cec1bd05879edb53abc75c6deead734cavandwalle            }
225155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande            return result;
226155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        }
227155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    }
228155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
229155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    private String doStringCommandWithoutLogging(String command) {
2300888ce6f90bdaeee799dd8361ea4781e23a33b87vandwalle        if (DBG) {
2310888ce6f90bdaeee799dd8361ea4781e23a33b87vandwalle            //GET_NETWORK commands flood the logs
2320888ce6f90bdaeee799dd8361ea4781e23a33b87vandwalle            if (!command.startsWith("GET_NETWORK")) {
2330888ce6f90bdaeee799dd8361ea4781e23a33b87vandwalle                Log.d(mTAG, "doString: [" + command + "]");
2340888ce6f90bdaeee799dd8361ea4781e23a33b87vandwalle            }
23527355a942653264388e909a4276196ee63e57811vandwalle        }
23627355a942653264388e909a4276196ee63e57811vandwalle        synchronized (mLock) {
237155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande            return doStringCommandNative(mInterfacePrefix + command);
238155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        }
239155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    }
240155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
241155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    public boolean ping() {
242155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        String pong = doStringCommand("PING");
243155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        return (pong != null && pong.equals("PONG"));
244155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    }
245155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
246ad7319939c424d42fa6a3791c47f613db8ef3cd8vandwalle    public void setSupplicantLogLevel(String level) {
247ad7319939c424d42fa6a3791c47f613db8ef3cd8vandwalle        doStringCommand("LOG_LEVEL " + level);
248ad7319939c424d42fa6a3791c47f613db8ef3cd8vandwalle    }
249ad7319939c424d42fa6a3791c47f613db8ef3cd8vandwalle
250a0009d14075b5345b8f916c3fb3f2260c938cb9dYuhao Zheng    public String getFreqCapability() {
251a0009d14075b5345b8f916c3fb3f2260c938cb9dYuhao Zheng        return doStringCommand("GET_CAPABILITY freq");
252a0009d14075b5345b8f916c3fb3f2260c938cb9dYuhao Zheng    }
253a0009d14075b5345b8f916c3fb3f2260c938cb9dYuhao Zheng
254a0009d14075b5345b8f916c3fb3f2260c938cb9dYuhao Zheng    public boolean scan(int type, String freqList) {
255155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        if (type == SCAN_WITHOUT_CONNECTION_SETUP) {
256a0009d14075b5345b8f916c3fb3f2260c938cb9dYuhao Zheng            if (freqList == null) return doBooleanCommand("SCAN TYPE=ONLY");
257a0009d14075b5345b8f916c3fb3f2260c938cb9dYuhao Zheng            else return doBooleanCommand("SCAN TYPE=ONLY freq=" + freqList);
258155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        } else if (type == SCAN_WITH_CONNECTION_SETUP) {
259a0009d14075b5345b8f916c3fb3f2260c938cb9dYuhao Zheng            if (freqList == null) return doBooleanCommand("SCAN");
260a0009d14075b5345b8f916c3fb3f2260c938cb9dYuhao Zheng            else return doBooleanCommand("SCAN freq=" + freqList);
261155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        } else {
262155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande            throw new IllegalArgumentException("Invalid scan type");
263155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        }
264155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    }
265155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
266155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    /* Does a graceful shutdown of supplicant. Is a common stop function for both p2p and sta.
267155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande     *
268155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande     * Note that underneath we use a harsh-sounding "terminate" supplicant command
269155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande     * for a graceful stop and a mild-sounding "stop" interface
270155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande     * to kill the process
271155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande     */
272155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    public boolean stopSupplicant() {
273155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        return doBooleanCommand("TERMINATE");
274155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    }
275155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
276155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    public String listNetworks() {
277155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        return doStringCommand("LIST_NETWORKS");
278155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    }
279155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
280e3939cb40d9ba3842be105a6e85172dc06e14758Vinit Deshpande    public String listNetworks(int last_id) {
281e3939cb40d9ba3842be105a6e85172dc06e14758Vinit Deshpande        return doStringCommand("LIST_NETWORKS LAST_ID=" + last_id);
282e3939cb40d9ba3842be105a6e85172dc06e14758Vinit Deshpande    }
283e3939cb40d9ba3842be105a6e85172dc06e14758Vinit Deshpande
284155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    public int addNetwork() {
285155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        return doIntCommand("ADD_NETWORK");
286155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    }
287155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
288155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    public boolean setNetworkVariable(int netId, String name, String value) {
289155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        if (TextUtils.isEmpty(name) || TextUtils.isEmpty(value)) return false;
290a6777abfc90496801e9942f57fbfa091ba85ae82Prerepa Viswanadham        if (name.equals(WifiConfiguration.pskVarName)
291a6777abfc90496801e9942f57fbfa091ba85ae82Prerepa Viswanadham                || name.equals(WifiEnterpriseConfig.PASSWORD_KEY)) {
292a6777abfc90496801e9942f57fbfa091ba85ae82Prerepa Viswanadham            return doBooleanCommandWithoutLogging("SET_NETWORK " + netId + " " + name + " " + value);
293a6777abfc90496801e9942f57fbfa091ba85ae82Prerepa Viswanadham        } else {
294a6777abfc90496801e9942f57fbfa091ba85ae82Prerepa Viswanadham            return doBooleanCommand("SET_NETWORK " + netId + " " + name + " " + value);
295a6777abfc90496801e9942f57fbfa091ba85ae82Prerepa Viswanadham        }
296155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    }
297155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
298155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    public String getNetworkVariable(int netId, String name) {
299155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        if (TextUtils.isEmpty(name)) return null;
300155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
301155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        // GET_NETWORK will likely flood the logs ...
302155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        return doStringCommandWithoutLogging("GET_NETWORK " + netId + " " + name);
303155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    }
304155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
305155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    public boolean removeNetwork(int netId) {
306155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        return doBooleanCommand("REMOVE_NETWORK " + netId);
307155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    }
308155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
309f22d23092ab37286a5ef9d257d5bb32c421d2669vandwalle
310f22d23092ab37286a5ef9d257d5bb32c421d2669vandwalle    private void logDbg(String debug) {
311f22d23092ab37286a5ef9d257d5bb32c421d2669vandwalle        long now = SystemClock.elapsedRealtimeNanos();
312f22d23092ab37286a5ef9d257d5bb32c421d2669vandwalle        String ts = String.format("[%,d us] ", now/1000);
313ecd2b88214b5d214fd1f63a9560caff9058912ddvandwalle        Log.e("WifiNative: ", ts+debug+ " stack:"
314ecd2b88214b5d214fd1f63a9560caff9058912ddvandwalle                + Thread.currentThread().getStackTrace()[2].getMethodName() +" - "
315ecd2b88214b5d214fd1f63a9560caff9058912ddvandwalle                + Thread.currentThread().getStackTrace()[3].getMethodName() +" - "
316ecd2b88214b5d214fd1f63a9560caff9058912ddvandwalle                + Thread.currentThread().getStackTrace()[4].getMethodName() +" - "
317ecd2b88214b5d214fd1f63a9560caff9058912ddvandwalle                + Thread.currentThread().getStackTrace()[5].getMethodName()+" - "
318ecd2b88214b5d214fd1f63a9560caff9058912ddvandwalle                + Thread.currentThread().getStackTrace()[6].getMethodName());
319f22d23092ab37286a5ef9d257d5bb32c421d2669vandwalle
320f22d23092ab37286a5ef9d257d5bb32c421d2669vandwalle    }
321155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    public boolean enableNetwork(int netId, boolean disableOthers) {
322ecd2b88214b5d214fd1f63a9560caff9058912ddvandwalle        if (DBG) logDbg("enableNetwork nid=" + Integer.toString(netId)
323ecd2b88214b5d214fd1f63a9560caff9058912ddvandwalle                + " disableOthers=" + disableOthers);
324155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        if (disableOthers) {
325155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande            return doBooleanCommand("SELECT_NETWORK " + netId);
326155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        } else {
327155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande            return doBooleanCommand("ENABLE_NETWORK " + netId);
328155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        }
329155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    }
330155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
331155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    public boolean disableNetwork(int netId) {
332f22d23092ab37286a5ef9d257d5bb32c421d2669vandwalle        if (DBG) logDbg("disableNetwork nid=" + Integer.toString(netId));
333155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        return doBooleanCommand("DISABLE_NETWORK " + netId);
334155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    }
335155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
3360047ccf563baa288777e06c6fe95d3681fcf5ccdVinit Deshpande    public boolean selectNetwork(int netId) {
3370047ccf563baa288777e06c6fe95d3681fcf5ccdVinit Deshpande        if (DBG) logDbg("selectNetwork nid=" + Integer.toString(netId));
3380047ccf563baa288777e06c6fe95d3681fcf5ccdVinit Deshpande        return doBooleanCommand("SELECT_NETWORK " + netId);
3390047ccf563baa288777e06c6fe95d3681fcf5ccdVinit Deshpande    }
3400047ccf563baa288777e06c6fe95d3681fcf5ccdVinit Deshpande
341155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    public boolean reconnect() {
342f22d23092ab37286a5ef9d257d5bb32c421d2669vandwalle        if (DBG) logDbg("RECONNECT ");
343155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        return doBooleanCommand("RECONNECT");
344155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    }
345155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
346155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    public boolean reassociate() {
347f22d23092ab37286a5ef9d257d5bb32c421d2669vandwalle        if (DBG) logDbg("REASSOCIATE ");
348155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        return doBooleanCommand("REASSOCIATE");
349155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    }
350155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
351155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    public boolean disconnect() {
35221bc54cb37a0085b1c909cb4d55ebb12a2facefbvandwalle        if (DBG) logDbg("DISCONNECT ");
353155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        return doBooleanCommand("DISCONNECT");
354155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    }
355155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
356155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    public String status() {
35799d385e3b4d34841d6efcfd7cc9bf1d5ae25de14vandwalle        return status(false);
358155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    }
359155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
36099d385e3b4d34841d6efcfd7cc9bf1d5ae25de14vandwalle    public String status(boolean noEvents) {
36199d385e3b4d34841d6efcfd7cc9bf1d5ae25de14vandwalle        if (noEvents) {
36299d385e3b4d34841d6efcfd7cc9bf1d5ae25de14vandwalle            return doStringCommand("STATUS-NO_EVENTS");
36399d385e3b4d34841d6efcfd7cc9bf1d5ae25de14vandwalle        } else {
36499d385e3b4d34841d6efcfd7cc9bf1d5ae25de14vandwalle            return doStringCommand("STATUS");
36599d385e3b4d34841d6efcfd7cc9bf1d5ae25de14vandwalle        }
36699d385e3b4d34841d6efcfd7cc9bf1d5ae25de14vandwalle    }
36799d385e3b4d34841d6efcfd7cc9bf1d5ae25de14vandwalle
368155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    public String getMacAddress() {
369155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        //Macaddr = XX.XX.XX.XX.XX.XX
370155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        String ret = doStringCommand("DRIVER MACADDR");
371155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        if (!TextUtils.isEmpty(ret)) {
372155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande            String[] tokens = ret.split(" = ");
373155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande            if (tokens.length == 2) return tokens[1];
374155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        }
375155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        return null;
376155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    }
377155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
378a0d34d3626bd9631039a485d2d428b1b7de390e2Pierre Vandwalle
379a0d34d3626bd9631039a485d2d428b1b7de390e2Pierre Vandwalle
380155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    /**
381155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande     * Format of results:
382155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande     * =================
383155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande     * id=1
384155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande     * bssid=68:7f:74:d7:1b:6e
385155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande     * freq=2412
386155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande     * level=-43
387155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande     * tsf=1344621975160944
388155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande     * age=2623
389155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande     * flags=[WPA2-PSK-CCMP][WPS][ESS]
390155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande     * ssid=zubyb
391155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande     * ====
392155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande     *
393155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande     * RANGE=ALL gets all scan results
394155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande     * RANGE=ID- gets results from ID
395155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande     * MASK=<N> see wpa_supplicant/src/common/wpa_ctrl.h for details
39677f2b82a2e80af8da52c22d69a76def6d4209757Jan Nordqvist     * 0                         0                        1                       0     2
39777f2b82a2e80af8da52c22d69a76def6d4209757Jan Nordqvist     *                           WPA_BSS_MASK_MESH_SCAN | WPA_BSS_MASK_DELIM    | WPA_BSS_MASK_WIFI_DISPLAY
39877f2b82a2e80af8da52c22d69a76def6d4209757Jan Nordqvist     * 0                         0                        0                       1     1   -> 9
39977f2b82a2e80af8da52c22d69a76def6d4209757Jan Nordqvist     * WPA_BSS_MASK_INTERNETW  | WPA_BSS_MASK_P2P_SCAN  | WPA_BSS_MASK_WPS_SCAN | WPA_BSS_MASK_SSID
40077f2b82a2e80af8da52c22d69a76def6d4209757Jan Nordqvist     * 1                         0                        0                       1     9   -> d
40177f2b82a2e80af8da52c22d69a76def6d4209757Jan Nordqvist     * WPA_BSS_MASK_FLAGS      | WPA_BSS_MASK_IE        | WPA_BSS_MASK_AGE      | WPA_BSS_MASK_TSF
40277f2b82a2e80af8da52c22d69a76def6d4209757Jan Nordqvist     * 1                         0                        0                       0     8
40377f2b82a2e80af8da52c22d69a76def6d4209757Jan Nordqvist     * WPA_BSS_MASK_LEVEL      | WPA_BSS_MASK_NOISE     | WPA_BSS_MASK_QUAL     | WPA_BSS_MASK_CAPABILITIES
40477f2b82a2e80af8da52c22d69a76def6d4209757Jan Nordqvist     * 0                         1                        1                       1     7
40577f2b82a2e80af8da52c22d69a76def6d4209757Jan Nordqvist     * WPA_BSS_MASK_BEACON_INT | WPA_BSS_MASK_FREQ      | WPA_BSS_MASK_BSSID    | WPA_BSS_MASK_ID
40677f2b82a2e80af8da52c22d69a76def6d4209757Jan Nordqvist     *
40777f2b82a2e80af8da52c22d69a76def6d4209757Jan Nordqvist     * WPA_BSS_MASK_INTERNETW adds ANQP info (ctrl_iface:4151-4176)
40877f2b82a2e80af8da52c22d69a76def6d4209757Jan Nordqvist     *
40977f2b82a2e80af8da52c22d69a76def6d4209757Jan Nordqvist     * ctrl_iface.c:wpa_supplicant_ctrl_iface_process:7884
41077f2b82a2e80af8da52c22d69a76def6d4209757Jan Nordqvist     *  wpa_supplicant_ctrl_iface_bss:4315
41177f2b82a2e80af8da52c22d69a76def6d4209757Jan Nordqvist     *  print_bss_info
412155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande     */
413155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    public String scanResults(int sid) {
41477f2b82a2e80af8da52c22d69a76def6d4209757Jan Nordqvist        return doStringCommandWithoutLogging("BSS RANGE=" + sid + "- MASK=0x29d87");
41577f2b82a2e80af8da52c22d69a76def6d4209757Jan Nordqvist    }
41677f2b82a2e80af8da52c22d69a76def6d4209757Jan Nordqvist
41777f2b82a2e80af8da52c22d69a76def6d4209757Jan Nordqvist    public String doCustomCommand(String command) {
41877f2b82a2e80af8da52c22d69a76def6d4209757Jan Nordqvist        return doStringCommand(command);
419155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    }
420155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
421155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    /**
422446db2d5457456743e4476029e14d7c3bb9f5bccYuhao Zheng     * Format of result:
423446db2d5457456743e4476029e14d7c3bb9f5bccYuhao Zheng     * id=1016
424446db2d5457456743e4476029e14d7c3bb9f5bccYuhao Zheng     * bssid=00:03:7f:40:84:10
425446db2d5457456743e4476029e14d7c3bb9f5bccYuhao Zheng     * freq=2462
426446db2d5457456743e4476029e14d7c3bb9f5bccYuhao Zheng     * beacon_int=200
427446db2d5457456743e4476029e14d7c3bb9f5bccYuhao Zheng     * capabilities=0x0431
428446db2d5457456743e4476029e14d7c3bb9f5bccYuhao Zheng     * qual=0
429446db2d5457456743e4476029e14d7c3bb9f5bccYuhao Zheng     * noise=0
430446db2d5457456743e4476029e14d7c3bb9f5bccYuhao Zheng     * level=-46
431446db2d5457456743e4476029e14d7c3bb9f5bccYuhao Zheng     * tsf=0000002669008476
432446db2d5457456743e4476029e14d7c3bb9f5bccYuhao Zheng     * age=5
433446db2d5457456743e4476029e14d7c3bb9f5bccYuhao Zheng     * ie=00105143412d485332302d52322d54455354010882848b960c12182403010b0706555...
434446db2d5457456743e4476029e14d7c3bb9f5bccYuhao Zheng     * flags=[WPA2-EAP-CCMP][ESS][P2P][HS20]
435446db2d5457456743e4476029e14d7c3bb9f5bccYuhao Zheng     * ssid=QCA-HS20-R2-TEST
436446db2d5457456743e4476029e14d7c3bb9f5bccYuhao Zheng     * p2p_device_name=
43756d0178183460eed9afbd85e5c0d215e27d5f5bcvandwalle     * p2p_config_methods=0x0SET_NE
438446db2d5457456743e4476029e14d7c3bb9f5bccYuhao Zheng     * anqp_venue_name=02083d656e6757692d466920416c6c69616e63650a3239383920436f...
439446db2d5457456743e4476029e14d7c3bb9f5bccYuhao Zheng     * anqp_network_auth_type=010000
440446db2d5457456743e4476029e14d7c3bb9f5bccYuhao Zheng     * anqp_roaming_consortium=03506f9a05001bc504bd
441446db2d5457456743e4476029e14d7c3bb9f5bccYuhao Zheng     * anqp_ip_addr_type_availability=0c
442446db2d5457456743e4476029e14d7c3bb9f5bccYuhao Zheng     * anqp_nai_realm=0200300000246d61696c2e6578616d706c652e636f6d3b636973636f2...
443446db2d5457456743e4476029e14d7c3bb9f5bccYuhao Zheng     * anqp_3gpp=000600040132f465
444446db2d5457456743e4476029e14d7c3bb9f5bccYuhao Zheng     * anqp_domain_name=0b65786d61706c652e636f6d
445446db2d5457456743e4476029e14d7c3bb9f5bccYuhao Zheng     * hs20_operator_friendly_name=11656e6757692d466920416c6c69616e63650e636869...
446446db2d5457456743e4476029e14d7c3bb9f5bccYuhao Zheng     * hs20_wan_metrics=01c40900008001000000000a00
447446db2d5457456743e4476029e14d7c3bb9f5bccYuhao Zheng     * hs20_connection_capability=0100000006140001061600000650000106bb010106bb0...
448446db2d5457456743e4476029e14d7c3bb9f5bccYuhao Zheng     * hs20_osu_providers_list=0b5143412d4f53552d425353010901310015656e6757692d...
449446db2d5457456743e4476029e14d7c3bb9f5bccYuhao Zheng     */
450446db2d5457456743e4476029e14d7c3bb9f5bccYuhao Zheng    public String scanResult(String bssid) {
451446db2d5457456743e4476029e14d7c3bb9f5bccYuhao Zheng        return doStringCommand("BSS " + bssid);
452446db2d5457456743e4476029e14d7c3bb9f5bccYuhao Zheng    }
453446db2d5457456743e4476029e14d7c3bb9f5bccYuhao Zheng
454446db2d5457456743e4476029e14d7c3bb9f5bccYuhao Zheng    /**
455155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande     * Format of command
456155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande     * DRIVER WLS_BATCHING SET SCANFREQ=x MSCAN=r BESTN=y CHANNEL=<z, w, t> RTT=s
457155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande     * where x is an ascii representation of an integer number of seconds between scans
458155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande     *       r is an ascii representation of an integer number of scans per batch
459155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande     *       y is an ascii representation of an integer number of the max AP to remember per scan
460155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande     *       z, w, t represent a 1..n size list of channel numbers and/or 'A', 'B' values
461155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande     *           indicating entire ranges of channels
462155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande     *       s is an ascii representation of an integer number of highest-strength AP
463155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande     *           for which we'd like approximate distance reported
464155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande     *
465155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande     * The return value is an ascii integer representing a guess of the number of scans
466155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande     * the firmware can remember before it runs out of buffer space or -1 on error
467155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande     */
468155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    public String setBatchedScanSettings(BatchedScanSettings settings) {
469155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        if (settings == null) {
470155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande            return doStringCommand("DRIVER WLS_BATCHING STOP");
471155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        }
472155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        String cmd = "DRIVER WLS_BATCHING SET SCANFREQ=" + settings.scanIntervalSec;
473155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        cmd += " MSCAN=" + settings.maxScansPerBatch;
474155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        if (settings.maxApPerScan != BatchedScanSettings.UNSPECIFIED) {
475155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande            cmd += " BESTN=" + settings.maxApPerScan;
476155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        }
477155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        if (settings.channelSet != null && !settings.channelSet.isEmpty()) {
478155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande            cmd += " CHANNEL=<";
479155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande            int i = 0;
480155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande            for (String channel : settings.channelSet) {
481155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande                cmd += (i > 0 ? "," : "") + channel;
482155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande                ++i;
483155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande            }
484155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande            cmd += ">";
485155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        }
486155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        if (settings.maxApForDistance != BatchedScanSettings.UNSPECIFIED) {
487155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande            cmd += " RTT=" + settings.maxApForDistance;
488155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        }
489155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        return doStringCommand(cmd);
490155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    }
491155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
492155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    public String getBatchedScanResults() {
493155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        return doStringCommand("DRIVER WLS_BATCHING GET");
494155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    }
495155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
496155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    public boolean startDriver() {
497155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        return doBooleanCommand("DRIVER START");
498155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    }
499155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
500155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    public boolean stopDriver() {
501155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        return doBooleanCommand("DRIVER STOP");
502155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    }
503155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
504155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
505155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    /**
506155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande     * Start filtering out Multicast V4 packets
507155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande     * @return {@code true} if the operation succeeded, {@code false} otherwise
508155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande     *
509155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande     * Multicast filtering rules work as follows:
510155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande     *
511155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande     * The driver can filter multicast (v4 and/or v6) and broadcast packets when in
512155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande     * a power optimized mode (typically when screen goes off).
513155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande     *
514155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande     * In order to prevent the driver from filtering the multicast/broadcast packets, we have to
515155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande     * add a DRIVER RXFILTER-ADD rule followed by DRIVER RXFILTER-START to make the rule effective
516155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande     *
517155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande     * DRIVER RXFILTER-ADD Num
518155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande     *   where Num = 0 - Unicast, 1 - Broadcast, 2 - Mutil4 or 3 - Multi6
519155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande     *
520155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande     * and DRIVER RXFILTER-START
521155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande     * In order to stop the usage of these rules, we do
522155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande     *
523155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande     * DRIVER RXFILTER-STOP
524155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande     * DRIVER RXFILTER-REMOVE Num
525155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande     *   where Num is as described for RXFILTER-ADD
526155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande     *
527155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande     * The  SETSUSPENDOPT driver command overrides the filtering rules
528155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande     */
529155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    public boolean startFilteringMulticastV4Packets() {
530155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        return doBooleanCommand("DRIVER RXFILTER-STOP")
531155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande            && doBooleanCommand("DRIVER RXFILTER-REMOVE 2")
532155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande            && doBooleanCommand("DRIVER RXFILTER-START");
533155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    }
534155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
535155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    /**
536155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande     * Stop filtering out Multicast V4 packets.
537155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande     * @return {@code true} if the operation succeeded, {@code false} otherwise
538155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande     */
539155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    public boolean stopFilteringMulticastV4Packets() {
540155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        return doBooleanCommand("DRIVER RXFILTER-STOP")
541155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande            && doBooleanCommand("DRIVER RXFILTER-ADD 2")
542155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande            && doBooleanCommand("DRIVER RXFILTER-START");
543155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    }
544155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
545155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    /**
546155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande     * Start filtering out Multicast V6 packets
547155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande     * @return {@code true} if the operation succeeded, {@code false} otherwise
548155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande     */
549155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    public boolean startFilteringMulticastV6Packets() {
550155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        return doBooleanCommand("DRIVER RXFILTER-STOP")
551155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande            && doBooleanCommand("DRIVER RXFILTER-REMOVE 3")
552155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande            && doBooleanCommand("DRIVER RXFILTER-START");
553155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    }
554155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
555155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    /**
556155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande     * Stop filtering out Multicast V6 packets.
557155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande     * @return {@code true} if the operation succeeded, {@code false} otherwise
558155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande     */
559155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    public boolean stopFilteringMulticastV6Packets() {
560155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        return doBooleanCommand("DRIVER RXFILTER-STOP")
561155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande            && doBooleanCommand("DRIVER RXFILTER-ADD 3")
562155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande            && doBooleanCommand("DRIVER RXFILTER-START");
563155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    }
564155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
56503cdd624f5da6d62cb731304aa7505921566f69dDmitry Shmidt    /**
56603cdd624f5da6d62cb731304aa7505921566f69dDmitry Shmidt     * Set the operational frequency band
56703cdd624f5da6d62cb731304aa7505921566f69dDmitry Shmidt     * @param band One of
56803cdd624f5da6d62cb731304aa7505921566f69dDmitry Shmidt     *     {@link WifiManager#WIFI_FREQUENCY_BAND_AUTO},
56903cdd624f5da6d62cb731304aa7505921566f69dDmitry Shmidt     *     {@link WifiManager#WIFI_FREQUENCY_BAND_5GHZ},
57003cdd624f5da6d62cb731304aa7505921566f69dDmitry Shmidt     *     {@link WifiManager#WIFI_FREQUENCY_BAND_2GHZ},
57103cdd624f5da6d62cb731304aa7505921566f69dDmitry Shmidt     * @return {@code true} if the operation succeeded, {@code false} otherwise
57203cdd624f5da6d62cb731304aa7505921566f69dDmitry Shmidt     */
573155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    public boolean setBand(int band) {
57403cdd624f5da6d62cb731304aa7505921566f69dDmitry Shmidt        String bandstr;
57503cdd624f5da6d62cb731304aa7505921566f69dDmitry Shmidt
57603cdd624f5da6d62cb731304aa7505921566f69dDmitry Shmidt        if (band == WifiManager.WIFI_FREQUENCY_BAND_5GHZ)
57703cdd624f5da6d62cb731304aa7505921566f69dDmitry Shmidt            bandstr = "5G";
57803cdd624f5da6d62cb731304aa7505921566f69dDmitry Shmidt        else if (band == WifiManager.WIFI_FREQUENCY_BAND_2GHZ)
57903cdd624f5da6d62cb731304aa7505921566f69dDmitry Shmidt            bandstr = "2G";
58003cdd624f5da6d62cb731304aa7505921566f69dDmitry Shmidt        else
58103cdd624f5da6d62cb731304aa7505921566f69dDmitry Shmidt            bandstr = "AUTO";
58203cdd624f5da6d62cb731304aa7505921566f69dDmitry Shmidt        return doBooleanCommand("SET SETBAND " + bandstr);
583155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    }
584155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
5857ef73dd1b6e43c72b3841723504cd86dc402a134Vinit Deshapnde    /**
5867ef73dd1b6e43c72b3841723504cd86dc402a134Vinit Deshapnde      * Sets the bluetooth coexistence mode.
5877ef73dd1b6e43c72b3841723504cd86dc402a134Vinit Deshapnde      *
5887ef73dd1b6e43c72b3841723504cd86dc402a134Vinit Deshapnde      * @param mode One of {@link #BLUETOOTH_COEXISTENCE_MODE_DISABLED},
5897ef73dd1b6e43c72b3841723504cd86dc402a134Vinit Deshapnde      *            {@link #BLUETOOTH_COEXISTENCE_MODE_ENABLED}, or
5907ef73dd1b6e43c72b3841723504cd86dc402a134Vinit Deshapnde      *            {@link #BLUETOOTH_COEXISTENCE_MODE_SENSE}.
5917ef73dd1b6e43c72b3841723504cd86dc402a134Vinit Deshapnde      * @return Whether the mode was successfully set.
5927ef73dd1b6e43c72b3841723504cd86dc402a134Vinit Deshapnde      */
593155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    public boolean setBluetoothCoexistenceMode(int mode) {
594155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        return doBooleanCommand("DRIVER BTCOEXMODE " + mode);
595155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    }
596155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
597155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    /**
598155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande     * Enable or disable Bluetooth coexistence scan mode. When this mode is on,
599155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande     * some of the low-level scan parameters used by the driver are changed to
600155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande     * reduce interference with A2DP streaming.
601155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande     *
602155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande     * @param isSet whether to enable or disable this mode
603155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande     * @return {@code true} if the command succeeded, {@code false} otherwise.
604155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande     */
605155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    public boolean setBluetoothCoexistenceScanMode(boolean setCoexScanMode) {
606155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        if (setCoexScanMode) {
607155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande            return doBooleanCommand("DRIVER BTCOEXSCAN-START");
608155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        } else {
609155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande            return doBooleanCommand("DRIVER BTCOEXSCAN-STOP");
610155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        }
611155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    }
612155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
6130a696d168d7ad98ab5084d2a16e3d02c545a85aaVinit Deshapnde    public void enableSaveConfig() {
6140a696d168d7ad98ab5084d2a16e3d02c545a85aaVinit Deshapnde        doBooleanCommand("SET update_config 1");
6150a696d168d7ad98ab5084d2a16e3d02c545a85aaVinit Deshapnde    }
6160a696d168d7ad98ab5084d2a16e3d02c545a85aaVinit Deshapnde
617155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    public boolean saveConfig() {
618155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        return doBooleanCommand("SAVE_CONFIG");
619155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    }
620155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
621155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    public boolean addToBlacklist(String bssid) {
622155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        if (TextUtils.isEmpty(bssid)) return false;
623155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        return doBooleanCommand("BLACKLIST " + bssid);
624155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    }
625155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
626155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    public boolean clearBlacklist() {
627155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        return doBooleanCommand("BLACKLIST clear");
628155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    }
629155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
630155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    public boolean setSuspendOptimizations(boolean enabled) {
631f22d23092ab37286a5ef9d257d5bb32c421d2669vandwalle       // if (mSuspendOptEnabled == enabled) return true;
632155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        mSuspendOptEnabled = enabled;
633f22d23092ab37286a5ef9d257d5bb32c421d2669vandwalle
634f22d23092ab37286a5ef9d257d5bb32c421d2669vandwalle        Log.e("native", "do suspend " + enabled);
635155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        if (enabled) {
636155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande            return doBooleanCommand("DRIVER SETSUSPENDMODE 1");
637155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        } else {
638155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande            return doBooleanCommand("DRIVER SETSUSPENDMODE 0");
639155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        }
640155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    }
641155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
642155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    public boolean setCountryCode(String countryCode) {
6430465ff513cdccf7e883a505e14c5b78758e7e458Vinit Deshpande        if (countryCode != null)
6440465ff513cdccf7e883a505e14c5b78758e7e458Vinit Deshpande            return doBooleanCommand("DRIVER COUNTRY " + countryCode.toUpperCase(Locale.ROOT));
6450465ff513cdccf7e883a505e14c5b78758e7e458Vinit Deshpande        else
6460465ff513cdccf7e883a505e14c5b78758e7e458Vinit Deshpande            return doBooleanCommand("DRIVER COUNTRY");
647155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    }
648155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
649c97b98d6877f6603a6a0eee820576f59e7da5e52Pierre Vandwalle    public boolean enableBackgroundScan(boolean enable) {
650c97b98d6877f6603a6a0eee820576f59e7da5e52Pierre Vandwalle        boolean ret;
651155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        if (enable) {
652c97b98d6877f6603a6a0eee820576f59e7da5e52Pierre Vandwalle            ret = doBooleanCommand("SET pno 1");
653155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        } else {
654c97b98d6877f6603a6a0eee820576f59e7da5e52Pierre Vandwalle            ret = doBooleanCommand("SET pno 0");
655155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        }
656c97b98d6877f6603a6a0eee820576f59e7da5e52Pierre Vandwalle        return ret;
657155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    }
658155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
659f22d23092ab37286a5ef9d257d5bb32c421d2669vandwalle    public void enableAutoConnect(boolean enable) {
660f22d23092ab37286a5ef9d257d5bb32c421d2669vandwalle        if (enable) {
661f22d23092ab37286a5ef9d257d5bb32c421d2669vandwalle            doBooleanCommand("STA_AUTOCONNECT 1");
662f22d23092ab37286a5ef9d257d5bb32c421d2669vandwalle        } else {
663f22d23092ab37286a5ef9d257d5bb32c421d2669vandwalle            doBooleanCommand("STA_AUTOCONNECT 0");
664f22d23092ab37286a5ef9d257d5bb32c421d2669vandwalle        }
665f22d23092ab37286a5ef9d257d5bb32c421d2669vandwalle    }
666f22d23092ab37286a5ef9d257d5bb32c421d2669vandwalle
667155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    public void setScanInterval(int scanInterval) {
668155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        doBooleanCommand("SCAN_INTERVAL " + scanInterval);
669155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    }
670155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
671155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    public void startTdls(String macAddr, boolean enable) {
672155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        if (enable) {
673155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande            doBooleanCommand("TDLS_DISCOVER " + macAddr);
674155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande            doBooleanCommand("TDLS_SETUP " + macAddr);
675155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        } else {
676155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande            doBooleanCommand("TDLS_TEARDOWN " + macAddr);
677155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        }
678155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    }
679155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
680155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    /** Example output:
681155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande     * RSSI=-65
682155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande     * LINKSPEED=48
683155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande     * NOISE=9999
684155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande     * FREQUENCY=0
685155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande     */
686155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    public String signalPoll() {
687155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        return doStringCommandWithoutLogging("SIGNAL_POLL");
688155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    }
689155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
690155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    /** Example outout:
691155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande     * TXGOOD=396
692155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande     * TXBAD=1
693155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande     */
694155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    public String pktcntPoll() {
695155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        return doStringCommand("PKTCNT_POLL");
696155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    }
697155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
698155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    public void bssFlush() {
699155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        doBooleanCommand("BSS_FLUSH 0");
700155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    }
701155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
702155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    public boolean startWpsPbc(String bssid) {
703155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        if (TextUtils.isEmpty(bssid)) {
704155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande            return doBooleanCommand("WPS_PBC");
705155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        } else {
706155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande            return doBooleanCommand("WPS_PBC " + bssid);
707155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        }
708155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    }
709155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
710155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    public boolean startWpsPbc(String iface, String bssid) {
711155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        synchronized (mLock) {
712155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande            if (TextUtils.isEmpty(bssid)) {
713155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande                return doBooleanCommandNative("IFNAME=" + iface + " WPS_PBC");
714155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande            } else {
715155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande                return doBooleanCommandNative("IFNAME=" + iface + " WPS_PBC " + bssid);
716155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande            }
717155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        }
718155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    }
719155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
720155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    public boolean startWpsPinKeypad(String pin) {
721155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        if (TextUtils.isEmpty(pin)) return false;
722155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        return doBooleanCommand("WPS_PIN any " + pin);
723155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    }
724155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
725155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    public boolean startWpsPinKeypad(String iface, String pin) {
726155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        if (TextUtils.isEmpty(pin)) return false;
727155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        synchronized (mLock) {
728155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande            return doBooleanCommandNative("IFNAME=" + iface + " WPS_PIN any " + pin);
729155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        }
730155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    }
731155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
732155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
733155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    public String startWpsPinDisplay(String bssid) {
734155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        if (TextUtils.isEmpty(bssid)) {
735155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande            return doStringCommand("WPS_PIN any");
736155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        } else {
737155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande            return doStringCommand("WPS_PIN " + bssid);
738155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        }
739155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    }
740155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
741155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    public String startWpsPinDisplay(String iface, String bssid) {
742155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        synchronized (mLock) {
743155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande            if (TextUtils.isEmpty(bssid)) {
744155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande                return doStringCommandNative("IFNAME=" + iface + " WPS_PIN any");
745155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande            } else {
746155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande                return doStringCommandNative("IFNAME=" + iface + " WPS_PIN " + bssid);
747155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande            }
748155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        }
749155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    }
750155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
75133b575ca6bee66183929f9474b5a161432918604Vinit Deshpande    public boolean setExternalSim(boolean external) {
75233b575ca6bee66183929f9474b5a161432918604Vinit Deshpande        synchronized (mLock) {
75333b575ca6bee66183929f9474b5a161432918604Vinit Deshpande            String value = external ? "1" : "0";
7544d701eca56d62586b0ab8af6ad864bac74a1dcd0Vinit Deshpande            Log.d(TAG, "Setting external_sim to " + value);
7554d701eca56d62586b0ab8af6ad864bac74a1dcd0Vinit Deshpande            return doBooleanCommand("SET external_sim " + value);
75633b575ca6bee66183929f9474b5a161432918604Vinit Deshpande        }
75733b575ca6bee66183929f9474b5a161432918604Vinit Deshpande    }
75833b575ca6bee66183929f9474b5a161432918604Vinit Deshpande
759f97140d51d14ce0659d381f443c08dbd94dfea28Honore Tricot    public boolean simAuthResponse(int id, String type, String response) {
760f97140d51d14ce0659d381f443c08dbd94dfea28Honore Tricot        // with type = GSM-AUTH, UMTS-AUTH or UMTS-AUTS
76133b575ca6bee66183929f9474b5a161432918604Vinit Deshpande        synchronized (mLock) {
762f97140d51d14ce0659d381f443c08dbd94dfea28Honore Tricot            return doBooleanCommand("CTRL-RSP-SIM-" + id + ":" + type + response);
76333b575ca6bee66183929f9474b5a161432918604Vinit Deshpande        }
76433b575ca6bee66183929f9474b5a161432918604Vinit Deshpande    }
76533b575ca6bee66183929f9474b5a161432918604Vinit Deshpande
766ad607d99c372160c7d4b38e755e1b47d6419856eHonore Tricot    public boolean simIdentityResponse(int id, String response) {
767ad607d99c372160c7d4b38e755e1b47d6419856eHonore Tricot        synchronized (mLock) {
768ad607d99c372160c7d4b38e755e1b47d6419856eHonore Tricot            return doBooleanCommand("CTRL-RSP-IDENTITY-" + id + ":" + response);
769ad607d99c372160c7d4b38e755e1b47d6419856eHonore Tricot        }
770ad607d99c372160c7d4b38e755e1b47d6419856eHonore Tricot    }
771ad607d99c372160c7d4b38e755e1b47d6419856eHonore Tricot
772155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    /* Configures an access point connection */
773155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    public boolean startWpsRegistrar(String bssid, String pin) {
774155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        if (TextUtils.isEmpty(bssid) || TextUtils.isEmpty(pin)) return false;
775155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        return doBooleanCommand("WPS_REG " + bssid + " " + pin);
776155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    }
777155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
778155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    public boolean cancelWps() {
779155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        return doBooleanCommand("WPS_CANCEL");
780155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    }
781155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
782155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    public boolean setPersistentReconnect(boolean enabled) {
783155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        int value = (enabled == true) ? 1 : 0;
784155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        return doBooleanCommand("SET persistent_reconnect " + value);
785155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    }
786155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
787155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    public boolean setDeviceName(String name) {
788155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        return doBooleanCommand("SET device_name " + name);
789155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    }
790155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
791155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    public boolean setDeviceType(String type) {
792155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        return doBooleanCommand("SET device_type " + type);
793155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    }
794155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
795155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    public boolean setConfigMethods(String cfg) {
796155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        return doBooleanCommand("SET config_methods " + cfg);
797155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    }
798155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
799155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    public boolean setManufacturer(String value) {
800155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        return doBooleanCommand("SET manufacturer " + value);
801155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    }
802155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
803155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    public boolean setModelName(String value) {
804155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        return doBooleanCommand("SET model_name " + value);
805155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    }
806155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
807155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    public boolean setModelNumber(String value) {
808155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        return doBooleanCommand("SET model_number " + value);
809155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    }
810155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
811155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    public boolean setSerialNumber(String value) {
812155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        return doBooleanCommand("SET serial_number " + value);
813155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    }
814155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
815155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    public boolean setP2pSsidPostfix(String postfix) {
816155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        return doBooleanCommand("SET p2p_ssid_postfix " + postfix);
817155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    }
818155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
819155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    public boolean setP2pGroupIdle(String iface, int time) {
820155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        synchronized (mLock) {
821155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande            return doBooleanCommandNative("IFNAME=" + iface + " SET p2p_group_idle " + time);
822155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        }
823155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    }
824155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
825155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    public void setPowerSave(boolean enabled) {
826155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        if (enabled) {
827155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande            doBooleanCommand("SET ps 1");
828155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        } else {
829155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande            doBooleanCommand("SET ps 0");
830155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        }
831155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    }
832155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
833155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    public boolean setP2pPowerSave(String iface, boolean enabled) {
834155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        synchronized (mLock) {
835155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande            if (enabled) {
836155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande                return doBooleanCommandNative("IFNAME=" + iface + " P2P_SET ps 1");
837155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande            } else {
838155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande                return doBooleanCommandNative("IFNAME=" + iface + " P2P_SET ps 0");
839155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande            }
840155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        }
841155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    }
842155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
843155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    public boolean setWfdEnable(boolean enable) {
844155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        return doBooleanCommand("SET wifi_display " + (enable ? "1" : "0"));
845155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    }
846155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
847155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    public boolean setWfdDeviceInfo(String hex) {
848155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        return doBooleanCommand("WFD_SUBELEM_SET 0 " + hex);
849155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    }
850155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
851155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    /**
852155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande     * "sta" prioritizes STA connection over P2P and "p2p" prioritizes
853155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande     * P2P connection over STA
854155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande     */
855155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    public boolean setConcurrencyPriority(String s) {
856155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        return doBooleanCommand("P2P_SET conc_pref " + s);
857155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    }
858155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
859155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    public boolean p2pFind() {
860155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        return doBooleanCommand("P2P_FIND");
861155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    }
862155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
863155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    public boolean p2pFind(int timeout) {
864155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        if (timeout <= 0) {
865155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande            return p2pFind();
866155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        }
867155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        return doBooleanCommand("P2P_FIND " + timeout);
868155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    }
869155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
870155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    public boolean p2pStopFind() {
871155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande       return doBooleanCommand("P2P_STOP_FIND");
872155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    }
873155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
874155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    public boolean p2pListen() {
875155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        return doBooleanCommand("P2P_LISTEN");
876155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    }
877155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
878155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    public boolean p2pListen(int timeout) {
879155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        if (timeout <= 0) {
880155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande            return p2pListen();
881155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        }
882155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        return doBooleanCommand("P2P_LISTEN " + timeout);
883155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    }
884155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
885155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    public boolean p2pExtListen(boolean enable, int period, int interval) {
886155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        if (enable && interval < period) {
887155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande            return false;
888155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        }
889155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        return doBooleanCommand("P2P_EXT_LISTEN"
890155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande                    + (enable ? (" " + period + " " + interval) : ""));
891155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    }
892155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
893155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    public boolean p2pSetChannel(int lc, int oc) {
894155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        if (DBG) Log.d(mTAG, "p2pSetChannel: lc="+lc+", oc="+oc);
895155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
896155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        if (lc >=1 && lc <= 11) {
897155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande            if (!doBooleanCommand("P2P_SET listen_channel " + lc)) {
898155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande                return false;
899155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande            }
900155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        } else if (lc != 0) {
901155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande            return false;
902155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        }
903155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
904155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        if (oc >= 1 && oc <= 165 ) {
905155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande            int freq = (oc <= 14 ? 2407 : 5000) + oc * 5;
906155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande            return doBooleanCommand("P2P_SET disallow_freq 1000-"
907155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande                    + (freq - 5) + "," + (freq + 5) + "-6000");
908155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        } else if (oc == 0) {
909155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande            /* oc==0 disables "P2P_SET disallow_freq" (enables all freqs) */
910155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande            return doBooleanCommand("P2P_SET disallow_freq \"\"");
911155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        }
912155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
913155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        return false;
914155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    }
915155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
916155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    public boolean p2pFlush() {
917155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        return doBooleanCommand("P2P_FLUSH");
918155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    }
919155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
920155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    /* p2p_connect <peer device address> <pbc|pin|PIN#> [label|display|keypad]
921155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        [persistent] [join|auth] [go_intent=<0..15>] [freq=<in MHz>] */
922155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    public String p2pConnect(WifiP2pConfig config, boolean joinExistingGroup) {
923155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        if (config == null) return null;
924155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        List<String> args = new ArrayList<String>();
925155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        WpsInfo wps = config.wps;
926155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        args.add(config.deviceAddress);
927155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
928155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        switch (wps.setup) {
929155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande            case WpsInfo.PBC:
930155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande                args.add("pbc");
931155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande                break;
932155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande            case WpsInfo.DISPLAY:
933155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande                if (TextUtils.isEmpty(wps.pin)) {
934155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande                    args.add("pin");
935155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande                } else {
936155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande                    args.add(wps.pin);
937155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande                }
938155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande                args.add("display");
939155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande                break;
940155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande            case WpsInfo.KEYPAD:
941155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande                args.add(wps.pin);
942155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande                args.add("keypad");
943155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande                break;
944155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande            case WpsInfo.LABEL:
945155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande                args.add(wps.pin);
946155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande                args.add("label");
947155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande            default:
948155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande                break;
949155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        }
950155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
951155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        if (config.netId == WifiP2pGroup.PERSISTENT_NET_ID) {
952155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande            args.add("persistent");
953155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        }
954155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
955155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        if (joinExistingGroup) {
956155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande            args.add("join");
957155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        } else {
958155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande            //TODO: This can be adapted based on device plugged in state and
959155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande            //device battery state
960155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande            int groupOwnerIntent = config.groupOwnerIntent;
961155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande            if (groupOwnerIntent < 0 || groupOwnerIntent > 15) {
962155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande                groupOwnerIntent = DEFAULT_GROUP_OWNER_INTENT;
963155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande            }
964155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande            args.add("go_intent=" + groupOwnerIntent);
965155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        }
966155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
967155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        String command = "P2P_CONNECT ";
968155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        for (String s : args) command += s + " ";
969155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
970155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        return doStringCommand(command);
971155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    }
972155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
973155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    public boolean p2pCancelConnect() {
974155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        return doBooleanCommand("P2P_CANCEL");
975155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    }
976155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
977155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    public boolean p2pProvisionDiscovery(WifiP2pConfig config) {
978155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        if (config == null) return false;
979155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
980155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        switch (config.wps.setup) {
981155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande            case WpsInfo.PBC:
982155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande                return doBooleanCommand("P2P_PROV_DISC " + config.deviceAddress + " pbc");
983155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande            case WpsInfo.DISPLAY:
984155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande                //We are doing display, so provision discovery is keypad
985155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande                return doBooleanCommand("P2P_PROV_DISC " + config.deviceAddress + " keypad");
986155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande            case WpsInfo.KEYPAD:
987155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande                //We are doing keypad, so provision discovery is display
988155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande                return doBooleanCommand("P2P_PROV_DISC " + config.deviceAddress + " display");
989155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande            default:
990155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande                break;
991155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        }
992155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        return false;
993155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    }
994155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
995155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    public boolean p2pGroupAdd(boolean persistent) {
996155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        if (persistent) {
997155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande            return doBooleanCommand("P2P_GROUP_ADD persistent");
998155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        }
999155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        return doBooleanCommand("P2P_GROUP_ADD");
1000155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    }
1001155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
1002155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    public boolean p2pGroupAdd(int netId) {
1003155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        return doBooleanCommand("P2P_GROUP_ADD persistent=" + netId);
1004155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    }
1005155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
1006155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    public boolean p2pGroupRemove(String iface) {
1007155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        if (TextUtils.isEmpty(iface)) return false;
1008155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        synchronized (mLock) {
1009155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande            return doBooleanCommandNative("IFNAME=" + iface + " P2P_GROUP_REMOVE " + iface);
1010155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        }
1011155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    }
1012155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
1013155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    public boolean p2pReject(String deviceAddress) {
1014155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        return doBooleanCommand("P2P_REJECT " + deviceAddress);
1015155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    }
1016155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
1017155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    /* Invite a peer to a group */
1018155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    public boolean p2pInvite(WifiP2pGroup group, String deviceAddress) {
1019155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        if (TextUtils.isEmpty(deviceAddress)) return false;
1020155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
1021155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        if (group == null) {
1022155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande            return doBooleanCommand("P2P_INVITE peer=" + deviceAddress);
1023155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        } else {
1024155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande            return doBooleanCommand("P2P_INVITE group=" + group.getInterface()
1025155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande                    + " peer=" + deviceAddress + " go_dev_addr=" + group.getOwner().deviceAddress);
1026155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        }
1027155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    }
1028155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
1029155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    /* Reinvoke a persistent connection */
1030155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    public boolean p2pReinvoke(int netId, String deviceAddress) {
1031155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        if (TextUtils.isEmpty(deviceAddress) || netId < 0) return false;
1032155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
1033155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        return doBooleanCommand("P2P_INVITE persistent=" + netId + " peer=" + deviceAddress);
1034155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    }
1035155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
1036155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    public String p2pGetSsid(String deviceAddress) {
1037155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        return p2pGetParam(deviceAddress, "oper_ssid");
1038155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    }
1039155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
1040155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    public String p2pGetDeviceAddress() {
104127f4b0c6ea9533e91863da48cefc80f8b5a88d1eVinit Deshpande
104236286b23c4e30f042ed753a670c2b462ebf13a48Vinit Deshpande        Log.d(TAG, "p2pGetDeviceAddress");
104336286b23c4e30f042ed753a670c2b462ebf13a48Vinit Deshpande
104427f4b0c6ea9533e91863da48cefc80f8b5a88d1eVinit Deshpande        String status = null;
104527f4b0c6ea9533e91863da48cefc80f8b5a88d1eVinit Deshpande
104636286b23c4e30f042ed753a670c2b462ebf13a48Vinit Deshpande        /* Explicitly calling the API without IFNAME= prefix to take care of the devices that
104736286b23c4e30f042ed753a670c2b462ebf13a48Vinit Deshpande        don't have p2p0 interface. Supplicant seems to be returning the correct address anyway. */
104836286b23c4e30f042ed753a670c2b462ebf13a48Vinit Deshpande
104927f4b0c6ea9533e91863da48cefc80f8b5a88d1eVinit Deshpande        synchronized (mLock) {
105027f4b0c6ea9533e91863da48cefc80f8b5a88d1eVinit Deshpande            status = doStringCommandNative("STATUS");
105127f4b0c6ea9533e91863da48cefc80f8b5a88d1eVinit Deshpande        }
105227f4b0c6ea9533e91863da48cefc80f8b5a88d1eVinit Deshpande
105327f4b0c6ea9533e91863da48cefc80f8b5a88d1eVinit Deshpande        String result = "";
105436286b23c4e30f042ed753a670c2b462ebf13a48Vinit Deshpande        if (status != null) {
105536286b23c4e30f042ed753a670c2b462ebf13a48Vinit Deshpande            String[] tokens = status.split("\n");
105636286b23c4e30f042ed753a670c2b462ebf13a48Vinit Deshpande            for (String token : tokens) {
105736286b23c4e30f042ed753a670c2b462ebf13a48Vinit Deshpande                if (token.startsWith("p2p_device_address=")) {
105836286b23c4e30f042ed753a670c2b462ebf13a48Vinit Deshpande                    String[] nameValue = token.split("=");
105936286b23c4e30f042ed753a670c2b462ebf13a48Vinit Deshpande                    if (nameValue.length != 2)
106036286b23c4e30f042ed753a670c2b462ebf13a48Vinit Deshpande                        break;
106136286b23c4e30f042ed753a670c2b462ebf13a48Vinit Deshpande                    result = nameValue[1];
106236286b23c4e30f042ed753a670c2b462ebf13a48Vinit Deshpande                }
1063155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande            }
1064155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        }
106536286b23c4e30f042ed753a670c2b462ebf13a48Vinit Deshpande
106636286b23c4e30f042ed753a670c2b462ebf13a48Vinit Deshpande        Log.d(TAG, "p2pGetDeviceAddress returning " + result);
106736286b23c4e30f042ed753a670c2b462ebf13a48Vinit Deshpande        return result;
1068155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    }
1069155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
1070155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    public int getGroupCapability(String deviceAddress) {
1071155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        int gc = 0;
1072155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        if (TextUtils.isEmpty(deviceAddress)) return gc;
1073155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        String peerInfo = p2pPeer(deviceAddress);
1074155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        if (TextUtils.isEmpty(peerInfo)) return gc;
1075155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
1076155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        String[] tokens = peerInfo.split("\n");
1077155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        for (String token : tokens) {
1078155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande            if (token.startsWith("group_capab=")) {
1079155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande                String[] nameValue = token.split("=");
1080155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande                if (nameValue.length != 2) break;
1081155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande                try {
1082155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande                    return Integer.decode(nameValue[1]);
1083155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande                } catch(NumberFormatException e) {
1084155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande                    return gc;
1085155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande                }
1086155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande            }
1087155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        }
1088155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        return gc;
1089155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    }
1090155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
1091155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    public String p2pPeer(String deviceAddress) {
1092155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        return doStringCommand("P2P_PEER " + deviceAddress);
1093155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    }
1094155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
1095155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    private String p2pGetParam(String deviceAddress, String key) {
1096155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        if (deviceAddress == null) return null;
1097155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
1098155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        String peerInfo = p2pPeer(deviceAddress);
1099155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        if (peerInfo == null) return null;
1100155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        String[] tokens= peerInfo.split("\n");
1101155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
1102155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        key += "=";
1103155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        for (String token : tokens) {
1104155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande            if (token.startsWith(key)) {
1105155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande                String[] nameValue = token.split("=");
1106155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande                if (nameValue.length != 2) break;
1107155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande                return nameValue[1];
1108155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande            }
1109155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        }
1110155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        return null;
1111155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    }
1112155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
1113155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    public boolean p2pServiceAdd(WifiP2pServiceInfo servInfo) {
1114155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        /*
1115155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande         * P2P_SERVICE_ADD bonjour <query hexdump> <RDATA hexdump>
1116155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande         * P2P_SERVICE_ADD upnp <version hex> <service>
1117155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande         *
1118155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande         * e.g)
1119155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande         * [Bonjour]
1120155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande         * # IP Printing over TCP (PTR) (RDATA=MyPrinter._ipp._tcp.local.)
1121155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande         * P2P_SERVICE_ADD bonjour 045f697070c00c000c01 094d795072696e746572c027
1122155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande         * # IP Printing over TCP (TXT) (RDATA=txtvers=1,pdl=application/postscript)
1123155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande         * P2P_SERVICE_ADD bonjour 096d797072696e746572045f697070c00c001001
1124155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande         *  09747874766572733d311a70646c3d6170706c69636174696f6e2f706f7374736372797074
1125155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande         *
1126155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande         * [UPnP]
1127155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande         * P2P_SERVICE_ADD upnp 10 uuid:6859dede-8574-59ab-9332-123456789012
1128155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande         * P2P_SERVICE_ADD upnp 10 uuid:6859dede-8574-59ab-9332-123456789012::upnp:rootdevice
1129155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande         * P2P_SERVICE_ADD upnp 10 uuid:6859dede-8574-59ab-9332-123456789012::urn:schemas-upnp
1130155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande         * -org:device:InternetGatewayDevice:1
1131155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande         * P2P_SERVICE_ADD upnp 10 uuid:6859dede-8574-59ab-9322-123456789012::urn:schemas-upnp
1132155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande         * -org:service:ContentDirectory:2
1133155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande         */
1134155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        for (String s : servInfo.getSupplicantQueryList()) {
1135155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande            String command = "P2P_SERVICE_ADD";
1136155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande            command += (" " + s);
1137155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande            if (!doBooleanCommand(command)) {
1138155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande                return false;
1139155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande            }
1140155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        }
1141155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        return true;
1142155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    }
1143155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
1144155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    public boolean p2pServiceDel(WifiP2pServiceInfo servInfo) {
1145155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        /*
1146155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande         * P2P_SERVICE_DEL bonjour <query hexdump>
1147155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande         * P2P_SERVICE_DEL upnp <version hex> <service>
1148155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande         */
1149155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        for (String s : servInfo.getSupplicantQueryList()) {
1150155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande            String command = "P2P_SERVICE_DEL ";
1151155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
1152155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande            String[] data = s.split(" ");
1153155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande            if (data.length < 2) {
1154155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande                return false;
1155155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande            }
1156155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande            if ("upnp".equals(data[0])) {
1157155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande                command += s;
1158155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande            } else if ("bonjour".equals(data[0])) {
1159155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande                command += data[0];
1160155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande                command += (" " + data[1]);
1161155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande            } else {
1162155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande                return false;
1163155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande            }
1164155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande            if (!doBooleanCommand(command)) {
1165155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande                return false;
1166155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande            }
1167155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        }
1168155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        return true;
1169155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    }
1170155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
1171155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    public boolean p2pServiceFlush() {
1172155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        return doBooleanCommand("P2P_SERVICE_FLUSH");
1173155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    }
1174155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
1175155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    public String p2pServDiscReq(String addr, String query) {
1176155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        String command = "P2P_SERV_DISC_REQ";
1177155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        command += (" " + addr);
1178155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        command += (" " + query);
1179155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
1180155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        return doStringCommand(command);
1181155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    }
1182155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
1183155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    public boolean p2pServDiscCancelReq(String id) {
1184155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        return doBooleanCommand("P2P_SERV_DISC_CANCEL_REQ " + id);
1185155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    }
1186155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
1187155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    /* Set the current mode of miracast operation.
1188155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande     *  0 = disabled
1189155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande     *  1 = operating as source
1190155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande     *  2 = operating as sink
1191155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande     */
1192155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    public void setMiracastMode(int mode) {
1193155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        // Note: optional feature on the driver. It is ok for this to fail.
1194155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        doBooleanCommand("DRIVER MIRACAST " + mode);
1195155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    }
11963f7ef65ab71619040032aee96b5599849881d6fdAndres Morales
1197446db2d5457456743e4476029e14d7c3bb9f5bccYuhao Zheng    public boolean fetchAnqp(String bssid, String subtypes) {
1198446db2d5457456743e4476029e14d7c3bb9f5bccYuhao Zheng        return doBooleanCommand("ANQP_GET " + bssid + " " + subtypes);
1199446db2d5457456743e4476029e14d7c3bb9f5bccYuhao Zheng    }
1200446db2d5457456743e4476029e14d7c3bb9f5bccYuhao Zheng
1201f49a59bda006b13e0118d144e0a4a5f569b2251eVinit Deshpande    /*
1202f49a59bda006b13e0118d144e0a4a5f569b2251eVinit Deshpande     * NFC-related calls
1203f49a59bda006b13e0118d144e0a4a5f569b2251eVinit Deshpande     */
1204f49a59bda006b13e0118d144e0a4a5f569b2251eVinit Deshpande    public String getNfcWpsConfigurationToken(int netId) {
1205f49a59bda006b13e0118d144e0a4a5f569b2251eVinit Deshpande        return doStringCommand("WPS_NFC_CONFIG_TOKEN WPS " + netId);
1206f49a59bda006b13e0118d144e0a4a5f569b2251eVinit Deshpande    }
1207f49a59bda006b13e0118d144e0a4a5f569b2251eVinit Deshpande
1208f49a59bda006b13e0118d144e0a4a5f569b2251eVinit Deshpande    public String getNfcHandoverRequest() {
1209f49a59bda006b13e0118d144e0a4a5f569b2251eVinit Deshpande        return doStringCommand("NFC_GET_HANDOVER_REQ NDEF P2P-CR");
1210f49a59bda006b13e0118d144e0a4a5f569b2251eVinit Deshpande    }
1211f49a59bda006b13e0118d144e0a4a5f569b2251eVinit Deshpande
1212f49a59bda006b13e0118d144e0a4a5f569b2251eVinit Deshpande    public String getNfcHandoverSelect() {
1213f49a59bda006b13e0118d144e0a4a5f569b2251eVinit Deshpande        return doStringCommand("NFC_GET_HANDOVER_SEL NDEF P2P-CR");
1214f49a59bda006b13e0118d144e0a4a5f569b2251eVinit Deshpande    }
1215f49a59bda006b13e0118d144e0a4a5f569b2251eVinit Deshpande
1216f49a59bda006b13e0118d144e0a4a5f569b2251eVinit Deshpande    public boolean initiatorReportNfcHandover(String selectMessage) {
1217f49a59bda006b13e0118d144e0a4a5f569b2251eVinit Deshpande        return doBooleanCommand("NFC_REPORT_HANDOVER INIT P2P 00 " + selectMessage);
1218f49a59bda006b13e0118d144e0a4a5f569b2251eVinit Deshpande    }
1219f49a59bda006b13e0118d144e0a4a5f569b2251eVinit Deshpande
1220f49a59bda006b13e0118d144e0a4a5f569b2251eVinit Deshpande    public boolean responderReportNfcHandover(String requestMessage) {
1221f49a59bda006b13e0118d144e0a4a5f569b2251eVinit Deshpande        return doBooleanCommand("NFC_REPORT_HANDOVER RESP P2P " + requestMessage + " 00");
1222f49a59bda006b13e0118d144e0a4a5f569b2251eVinit Deshpande    }
1223f49a59bda006b13e0118d144e0a4a5f569b2251eVinit Deshpande
12247f9a15d554f69311a0db43347d6473a7c4c46e2eVinit Deshapnde    /* WIFI HAL support */
12257f9a15d554f69311a0db43347d6473a7c4c46e2eVinit Deshapnde
1226b0b1d59786de7ff1fc98e31a16a7d710458bf549Vinit Deshpande    private static final String TAG = "WifiNative-HAL";
1227f49a59bda006b13e0118d144e0a4a5f569b2251eVinit Deshpande    private static long sWifiHalHandle = 0;             /* used by JNI to save wifi_handle */
1228f49a59bda006b13e0118d144e0a4a5f569b2251eVinit Deshpande    private static long[] sWifiIfaceHandles = null;     /* used by JNI to save interface handles */
1229aabe7a9f2f32915fd1f25416a6d2034a844005d6vandwalle    private static int sWlan0Index = -1;
1230aabe7a9f2f32915fd1f25416a6d2034a844005d6vandwalle    private static int sP2p0Index = -1;
1231f49a59bda006b13e0118d144e0a4a5f569b2251eVinit Deshpande    private static MonitorThread sThread;
1232f49a59bda006b13e0118d144e0a4a5f569b2251eVinit Deshpande    private static final int STOP_HAL_TIMEOUT_MS = 1000;
12337f9a15d554f69311a0db43347d6473a7c4c46e2eVinit Deshapnde
1234b0b1d59786de7ff1fc98e31a16a7d710458bf549Vinit Deshpande    private static native boolean startHalNative();
1235b0b1d59786de7ff1fc98e31a16a7d710458bf549Vinit Deshpande    private static native void stopHalNative();
1236b0b1d59786de7ff1fc98e31a16a7d710458bf549Vinit Deshpande    private static native void waitForHalEventNative();
12377f9a15d554f69311a0db43347d6473a7c4c46e2eVinit Deshapnde
1238b0b1d59786de7ff1fc98e31a16a7d710458bf549Vinit Deshpande    private static class MonitorThread extends Thread {
12397ef73dd1b6e43c72b3841723504cd86dc402a134Vinit Deshapnde        public void run() {
1240b0b1d59786de7ff1fc98e31a16a7d710458bf549Vinit Deshpande            Log.i(TAG, "Waiting for HAL events mWifiHalHandle=" + Long.toString(sWifiHalHandle));
12417ef73dd1b6e43c72b3841723504cd86dc402a134Vinit Deshapnde            waitForHalEventNative();
12427ef73dd1b6e43c72b3841723504cd86dc402a134Vinit Deshapnde        }
12437ef73dd1b6e43c72b3841723504cd86dc402a134Vinit Deshapnde    }
12447ef73dd1b6e43c72b3841723504cd86dc402a134Vinit Deshapnde
1245b0b1d59786de7ff1fc98e31a16a7d710458bf549Vinit Deshpande    synchronized public static boolean startHal() {
1246d6307b404302949f6dadd14fa0860ff1aec432dcxinhe
1247d6307b404302949f6dadd14fa0860ff1aec432dcxinhe        String debugLog = "startHal stack: ";
1248d6307b404302949f6dadd14fa0860ff1aec432dcxinhe        java.lang.StackTraceElement[] elements = Thread.currentThread().getStackTrace();
1249d6307b404302949f6dadd14fa0860ff1aec432dcxinhe        for (int i = 2; i < elements.length && i <= 7; i++ ) {
1250d6307b404302949f6dadd14fa0860ff1aec432dcxinhe            debugLog = debugLog + " - " + elements[i].getMethodName();
1251d6307b404302949f6dadd14fa0860ff1aec432dcxinhe        }
1252d6307b404302949f6dadd14fa0860ff1aec432dcxinhe
1253f49a59bda006b13e0118d144e0a4a5f569b2251eVinit Deshpande        mLocalLog.log(debugLog);
1254d6307b404302949f6dadd14fa0860ff1aec432dcxinhe
1255aabe7a9f2f32915fd1f25416a6d2034a844005d6vandwalle        synchronized (mLock) {
1256cce360ad854cabb238ba0d9290785c26e837749cVinit Deshpande            if (startHalNative() && (getInterfaces() != 0) && (sWlan0Index != -1)) {
1257f49a59bda006b13e0118d144e0a4a5f569b2251eVinit Deshpande                sThread = new MonitorThread();
1258f49a59bda006b13e0118d144e0a4a5f569b2251eVinit Deshpande                sThread.start();
1259aabe7a9f2f32915fd1f25416a6d2034a844005d6vandwalle                return true;
1260aabe7a9f2f32915fd1f25416a6d2034a844005d6vandwalle            } else {
1261f49a59bda006b13e0118d144e0a4a5f569b2251eVinit Deshpande                if (DBG) mLocalLog.log("Could not start hal");
1262f49a59bda006b13e0118d144e0a4a5f569b2251eVinit Deshpande                Log.e(TAG, "Could not start hal");
1263aabe7a9f2f32915fd1f25416a6d2034a844005d6vandwalle                return false;
1264aabe7a9f2f32915fd1f25416a6d2034a844005d6vandwalle            }
12657ef73dd1b6e43c72b3841723504cd86dc402a134Vinit Deshapnde        }
12667ef73dd1b6e43c72b3841723504cd86dc402a134Vinit Deshapnde    }
12677ef73dd1b6e43c72b3841723504cd86dc402a134Vinit Deshapnde
1268b0b1d59786de7ff1fc98e31a16a7d710458bf549Vinit Deshpande    synchronized public static void stopHal() {
1269f49a59bda006b13e0118d144e0a4a5f569b2251eVinit Deshpande        synchronized (mLock) {
127071d0be16c79791a005d60f9d5fa7d2d81e1f8c80xinhe            if (isHalStarted()) {
1271f49a59bda006b13e0118d144e0a4a5f569b2251eVinit Deshpande                stopHalNative();
1272f49a59bda006b13e0118d144e0a4a5f569b2251eVinit Deshpande                try {
1273f49a59bda006b13e0118d144e0a4a5f569b2251eVinit Deshpande                    sThread.join(STOP_HAL_TIMEOUT_MS);
1274f49a59bda006b13e0118d144e0a4a5f569b2251eVinit Deshpande                    Log.d(TAG, "HAL event thread stopped successfully");
1275f49a59bda006b13e0118d144e0a4a5f569b2251eVinit Deshpande                } catch (InterruptedException e) {
1276f49a59bda006b13e0118d144e0a4a5f569b2251eVinit Deshpande                    Log.e(TAG, "Could not stop HAL cleanly");
1277f49a59bda006b13e0118d144e0a4a5f569b2251eVinit Deshpande                }
1278f49a59bda006b13e0118d144e0a4a5f569b2251eVinit Deshpande                sThread = null;
1279f49a59bda006b13e0118d144e0a4a5f569b2251eVinit Deshpande                sWifiHalHandle = 0;
1280f49a59bda006b13e0118d144e0a4a5f569b2251eVinit Deshpande                sWifiIfaceHandles = null;
1281f49a59bda006b13e0118d144e0a4a5f569b2251eVinit Deshpande                sWlan0Index = -1;
1282f49a59bda006b13e0118d144e0a4a5f569b2251eVinit Deshpande                sP2p0Index = -1;
1283f49a59bda006b13e0118d144e0a4a5f569b2251eVinit Deshpande            }
1284f49a59bda006b13e0118d144e0a4a5f569b2251eVinit Deshpande        }
12857ef73dd1b6e43c72b3841723504cd86dc402a134Vinit Deshapnde    }
12867f9a15d554f69311a0db43347d6473a7c4c46e2eVinit Deshapnde
128771d0be16c79791a005d60f9d5fa7d2d81e1f8c80xinhe    public static boolean isHalStarted() {
128871d0be16c79791a005d60f9d5fa7d2d81e1f8c80xinhe        return (sWifiHalHandle != 0);
128971d0be16c79791a005d60f9d5fa7d2d81e1f8c80xinhe    }
1290b0b1d59786de7ff1fc98e31a16a7d710458bf549Vinit Deshpande    private static native int getInterfacesNative();
12917f9a15d554f69311a0db43347d6473a7c4c46e2eVinit Deshapnde
1292b0b1d59786de7ff1fc98e31a16a7d710458bf549Vinit Deshpande    synchronized public static int getInterfaces() {
1293aabe7a9f2f32915fd1f25416a6d2034a844005d6vandwalle        synchronized (mLock) {
129471d0be16c79791a005d60f9d5fa7d2d81e1f8c80xinhe            if (isHalStarted()) {
1295f49a59bda006b13e0118d144e0a4a5f569b2251eVinit Deshpande                if (sWifiIfaceHandles == null) {
1296f49a59bda006b13e0118d144e0a4a5f569b2251eVinit Deshpande                    int num = getInterfacesNative();
1297f49a59bda006b13e0118d144e0a4a5f569b2251eVinit Deshpande                    int wifi_num = 0;
1298f49a59bda006b13e0118d144e0a4a5f569b2251eVinit Deshpande                    for (int i = 0; i < num; i++) {
1299f49a59bda006b13e0118d144e0a4a5f569b2251eVinit Deshpande                        String name = getInterfaceNameNative(i);
1300f49a59bda006b13e0118d144e0a4a5f569b2251eVinit Deshpande                        Log.i(TAG, "interface[" + i + "] = " + name);
1301f49a59bda006b13e0118d144e0a4a5f569b2251eVinit Deshpande                        if (name.equals("wlan0")) {
1302f49a59bda006b13e0118d144e0a4a5f569b2251eVinit Deshpande                            sWlan0Index = i;
1303f49a59bda006b13e0118d144e0a4a5f569b2251eVinit Deshpande                            wifi_num++;
1304f49a59bda006b13e0118d144e0a4a5f569b2251eVinit Deshpande                        } else if (name.equals("p2p0")) {
1305f49a59bda006b13e0118d144e0a4a5f569b2251eVinit Deshpande                            sP2p0Index = i;
1306f49a59bda006b13e0118d144e0a4a5f569b2251eVinit Deshpande                            wifi_num++;
1307f49a59bda006b13e0118d144e0a4a5f569b2251eVinit Deshpande                        }
130802a1f98f2cecb8ae2d466d6f9fab06b473f970ddVinit Deshpande                    }
1309f49a59bda006b13e0118d144e0a4a5f569b2251eVinit Deshpande                    return wifi_num;
1310f49a59bda006b13e0118d144e0a4a5f569b2251eVinit Deshpande                } else {
1311f49a59bda006b13e0118d144e0a4a5f569b2251eVinit Deshpande                    return sWifiIfaceHandles.length;
1312aabe7a9f2f32915fd1f25416a6d2034a844005d6vandwalle                }
131302a1f98f2cecb8ae2d466d6f9fab06b473f970ddVinit Deshpande            } else {
1314f49a59bda006b13e0118d144e0a4a5f569b2251eVinit Deshpande                return 0;
1315e4e3750390bec0a849a9153348b7c21b2cc8b843Vinit Deshapnde            }
1316e4e3750390bec0a849a9153348b7c21b2cc8b843Vinit Deshapnde        }
13177f9a15d554f69311a0db43347d6473a7c4c46e2eVinit Deshapnde    }
13187f9a15d554f69311a0db43347d6473a7c4c46e2eVinit Deshapnde
1319b0b1d59786de7ff1fc98e31a16a7d710458bf549Vinit Deshpande    private static native String getInterfaceNameNative(int index);
1320a632d8a6edd350c7644b593b18eceaa5b368505bVinit Deshpande    synchronized public static String getInterfaceName(int index) {
1321a632d8a6edd350c7644b593b18eceaa5b368505bVinit Deshpande        return getInterfaceNameNative(index);
13227f9a15d554f69311a0db43347d6473a7c4c46e2eVinit Deshapnde    }
13237f9a15d554f69311a0db43347d6473a7c4c46e2eVinit Deshapnde
1324e4e3750390bec0a849a9153348b7c21b2cc8b843Vinit Deshapnde    public static class ScanCapabilities {
1325e4e3750390bec0a849a9153348b7c21b2cc8b843Vinit Deshapnde        public int  max_scan_cache_size;                 // in number of scan results??
1326e4e3750390bec0a849a9153348b7c21b2cc8b843Vinit Deshapnde        public int  max_scan_buckets;
1327e4e3750390bec0a849a9153348b7c21b2cc8b843Vinit Deshapnde        public int  max_ap_cache_per_scan;
1328e4e3750390bec0a849a9153348b7c21b2cc8b843Vinit Deshapnde        public int  max_rssi_sample_size;
1329e4e3750390bec0a849a9153348b7c21b2cc8b843Vinit Deshapnde        public int  max_scan_reporting_threshold;        // in number of scan results??
13307d6301ead19afdf3de37455e9ed133c25b4938cdVinit Deshpande        public int  max_hotlist_bssids;
1331e4e3750390bec0a849a9153348b7c21b2cc8b843Vinit Deshapnde        public int  max_significant_wifi_change_aps;
1332e4e3750390bec0a849a9153348b7c21b2cc8b843Vinit Deshapnde    }
1333e4e3750390bec0a849a9153348b7c21b2cc8b843Vinit Deshapnde
1334af5b49884f189bb171c9dc6c6a4405d97e7912acVinit Deshpande    synchronized public static boolean getScanCapabilities(ScanCapabilities capabilities) {
1335af5b49884f189bb171c9dc6c6a4405d97e7912acVinit Deshpande        synchronized (mLock) {
1336af5b49884f189bb171c9dc6c6a4405d97e7912acVinit Deshpande            return isHalStarted() && getScanCapabilitiesNative(sWlan0Index, capabilities);
1337af5b49884f189bb171c9dc6c6a4405d97e7912acVinit Deshpande        }
1338e4e3750390bec0a849a9153348b7c21b2cc8b843Vinit Deshapnde    }
1339e4e3750390bec0a849a9153348b7c21b2cc8b843Vinit Deshapnde
1340b0b1d59786de7ff1fc98e31a16a7d710458bf549Vinit Deshpande    private static native boolean getScanCapabilitiesNative(
1341b0b1d59786de7ff1fc98e31a16a7d710458bf549Vinit Deshpande            int iface, ScanCapabilities capabilities);
1342e4e3750390bec0a849a9153348b7c21b2cc8b843Vinit Deshapnde
1343b0b1d59786de7ff1fc98e31a16a7d710458bf549Vinit Deshpande    private static native boolean startScanNative(int iface, int id, ScanSettings settings);
1344b0b1d59786de7ff1fc98e31a16a7d710458bf549Vinit Deshpande    private static native boolean stopScanNative(int iface, int id);
134583a674a18c84ff4f01377bbfd8988699dec93bc2Vinit Deshpande    private static native WifiScanner.ScanData[] getScanResultsNative(int iface, boolean flush);
1346b0b1d59786de7ff1fc98e31a16a7d710458bf549Vinit Deshpande    private static native WifiLinkLayerStats getWifiLinkLayerStatsNative(int iface);
1347d745a52dc4f929d4a4030f205ed173bdf60eaf10Pierre Vandwalle    private static native void setWifiLinkLayerStatsNative(int iface, int enable);
13487f9a15d554f69311a0db43347d6473a7c4c46e2eVinit Deshapnde
1349e4e3750390bec0a849a9153348b7c21b2cc8b843Vinit Deshapnde    public static class ChannelSettings {
1350e4e3750390bec0a849a9153348b7c21b2cc8b843Vinit Deshapnde        int frequency;
1351e4e3750390bec0a849a9153348b7c21b2cc8b843Vinit Deshapnde        int dwell_time_ms;
1352e4e3750390bec0a849a9153348b7c21b2cc8b843Vinit Deshapnde        boolean passive;
13537f9a15d554f69311a0db43347d6473a7c4c46e2eVinit Deshapnde    }
13547f9a15d554f69311a0db43347d6473a7c4c46e2eVinit Deshapnde
1355e4e3750390bec0a849a9153348b7c21b2cc8b843Vinit Deshapnde    public static class BucketSettings {
1356e4e3750390bec0a849a9153348b7c21b2cc8b843Vinit Deshapnde        int bucket;
1357e4e3750390bec0a849a9153348b7c21b2cc8b843Vinit Deshapnde        int band;
1358e4e3750390bec0a849a9153348b7c21b2cc8b843Vinit Deshapnde        int period_ms;
1359e4e3750390bec0a849a9153348b7c21b2cc8b843Vinit Deshapnde        int report_events;
1360e4e3750390bec0a849a9153348b7c21b2cc8b843Vinit Deshapnde        int num_channels;
1361daac2ad767f6047409987bb22812ab5f295e54dfVinit Deshpande        ChannelSettings channels[];
1362e4e3750390bec0a849a9153348b7c21b2cc8b843Vinit Deshapnde    }
13637f9a15d554f69311a0db43347d6473a7c4c46e2eVinit Deshapnde
1364e4e3750390bec0a849a9153348b7c21b2cc8b843Vinit Deshapnde    public static class ScanSettings {
1365e4e3750390bec0a849a9153348b7c21b2cc8b843Vinit Deshapnde        int base_period_ms;
1366e4e3750390bec0a849a9153348b7c21b2cc8b843Vinit Deshapnde        int max_ap_per_scan;
136783a674a18c84ff4f01377bbfd8988699dec93bc2Vinit Deshpande        int report_threshold_percent;
136883a674a18c84ff4f01377bbfd8988699dec93bc2Vinit Deshpande        int report_threshold_num_scans;
1369e4e3750390bec0a849a9153348b7c21b2cc8b843Vinit Deshapnde        int num_buckets;
1370daac2ad767f6047409987bb22812ab5f295e54dfVinit Deshpande        BucketSettings buckets[];
1371e4e3750390bec0a849a9153348b7c21b2cc8b843Vinit Deshapnde    }
13727f9a15d554f69311a0db43347d6473a7c4c46e2eVinit Deshapnde
1373b0b1d59786de7ff1fc98e31a16a7d710458bf549Vinit Deshpande    public static interface ScanEventHandler {
1374e4e3750390bec0a849a9153348b7c21b2cc8b843Vinit Deshapnde        void onScanResultsAvailable();
1375476bee2fef10d060c25c35858b1f7f60803d9f49Vinit Deshpande        void onFullScanResult(ScanResult fullScanResult);
137683a674a18c84ff4f01377bbfd8988699dec93bc2Vinit Deshpande        void onScanStatus();
137783a674a18c84ff4f01377bbfd8988699dec93bc2Vinit Deshpande        void onScanPaused(WifiScanner.ScanData[] data);
1378b0b1d59786de7ff1fc98e31a16a7d710458bf549Vinit Deshpande        void onScanRestarted();
1379e4e3750390bec0a849a9153348b7c21b2cc8b843Vinit Deshapnde    }
1380e4e3750390bec0a849a9153348b7c21b2cc8b843Vinit Deshapnde
1381b0b1d59786de7ff1fc98e31a16a7d710458bf549Vinit Deshpande    synchronized static void onScanResultsAvailable(int id) {
1382b7cc309f06da8a18224057c21ba086f8550367d6Vinit Deshpande        if (sScanEventHandler  != null) {
1383b7cc309f06da8a18224057c21ba086f8550367d6Vinit Deshpande            sScanEventHandler.onScanResultsAvailable();
1384b7cc309f06da8a18224057c21ba086f8550367d6Vinit Deshpande        }
1385b7cc309f06da8a18224057c21ba086f8550367d6Vinit Deshpande    }
1386b7cc309f06da8a18224057c21ba086f8550367d6Vinit Deshpande
1387b7cc309f06da8a18224057c21ba086f8550367d6Vinit Deshpande    /* scan status, keep these values in sync with gscan.h */
1388b7cc309f06da8a18224057c21ba086f8550367d6Vinit Deshpande    private static int WIFI_SCAN_BUFFER_FULL = 0;
1389b7cc309f06da8a18224057c21ba086f8550367d6Vinit Deshpande    private static int WIFI_SCAN_COMPLETE = 1;
1390b7cc309f06da8a18224057c21ba086f8550367d6Vinit Deshpande
1391b7cc309f06da8a18224057c21ba086f8550367d6Vinit Deshpande    synchronized static void onScanStatus(int status) {
1392b7cc309f06da8a18224057c21ba086f8550367d6Vinit Deshpande        if (status == WIFI_SCAN_BUFFER_FULL) {
1393b7cc309f06da8a18224057c21ba086f8550367d6Vinit Deshpande            /* we have a separate event to take care of this */
1394b7cc309f06da8a18224057c21ba086f8550367d6Vinit Deshpande        } else if (status == WIFI_SCAN_COMPLETE) {
1395b7cc309f06da8a18224057c21ba086f8550367d6Vinit Deshpande            if (sScanEventHandler  != null) {
139683a674a18c84ff4f01377bbfd8988699dec93bc2Vinit Deshpande                sScanEventHandler.onScanStatus();
1397b7cc309f06da8a18224057c21ba086f8550367d6Vinit Deshpande            }
1398b7cc309f06da8a18224057c21ba086f8550367d6Vinit Deshpande        }
1399e4e3750390bec0a849a9153348b7c21b2cc8b843Vinit Deshapnde    }
1400e4e3750390bec0a849a9153348b7c21b2cc8b843Vinit Deshapnde
1401dd490cf1df37b70a8625dc4ec8e712f740dd1e4aPierre Vandwalle    static void populateScanResult(ScanResult result, byte bytes[], String dbg) {
1402f1daf9342b66bf134d13fa0a42e929a008f1ca62Vinit Deshapnde        int num = 0;
1403dd490cf1df37b70a8625dc4ec8e712f740dd1e4aPierre Vandwalle        if (bytes == null) return;
1404dd490cf1df37b70a8625dc4ec8e712f740dd1e4aPierre Vandwalle        if (dbg == null) dbg = "";
1405127f7244183786e6ccae09e81eeccdac31973e69xinhe        for (int i = 0; i < bytes.length - 1; ) {
1406e7b9e74a9a699e74881e5c98c684567763dfaa35Vinit Deshpande            int type  = bytes[i] & 0xFF;
1407e7b9e74a9a699e74881e5c98c684567763dfaa35Vinit Deshpande            int len = bytes[i + 1] & 0xFF;
1408e7b9e74a9a699e74881e5c98c684567763dfaa35Vinit Deshpande            if (i + len + 2 > bytes.length) {
1409dd490cf1df37b70a8625dc4ec8e712f740dd1e4aPierre Vandwalle                Log.w(TAG, dbg + "bad length " + len + " of IE " + type + " from " + result.BSSID);
1410dd490cf1df37b70a8625dc4ec8e712f740dd1e4aPierre Vandwalle                Log.w(TAG, dbg + "ignoring the rest of the IEs");
1411e7b9e74a9a699e74881e5c98c684567763dfaa35Vinit Deshpande                break;
1412f1daf9342b66bf134d13fa0a42e929a008f1ca62Vinit Deshapnde            }
1413e7b9e74a9a699e74881e5c98c684567763dfaa35Vinit Deshpande            num++;
1414dd490cf1df37b70a8625dc4ec8e712f740dd1e4aPierre Vandwalle            if (DBG) Log.i(TAG, dbg + "bytes[" + i + "] = [" + type + ", " + len + "]" + ", " +
1415127f7244183786e6ccae09e81eeccdac31973e69xinhe                    "next = " + (i + len + 2));
1416127f7244183786e6ccae09e81eeccdac31973e69xinhe            i += len + 2;
1417f1daf9342b66bf134d13fa0a42e929a008f1ca62Vinit Deshapnde        }
1418f1daf9342b66bf134d13fa0a42e929a008f1ca62Vinit Deshapnde
1419243931f3474f6235cfcf5c1a55fa2f192ee264aexinhe        int secondChanelOffset = 0;
1420243931f3474f6235cfcf5c1a55fa2f192ee264aexinhe        byte channelMode = 0;
1421243931f3474f6235cfcf5c1a55fa2f192ee264aexinhe        byte centerFreqIndex1 = 0;
1422243931f3474f6235cfcf5c1a55fa2f192ee264aexinhe        byte centerFreqIndex2 = 0;
1423d9f37b2f3df18e96246db93ec4c2a5159b5d3915Vinit Deshpande
1424d9f37b2f3df18e96246db93ec4c2a5159b5d3915Vinit Deshpande        boolean is80211McRTTResponder = false;
1425243931f3474f6235cfcf5c1a55fa2f192ee264aexinhe
1426476bee2fef10d060c25c35858b1f7f60803d9f49Vinit Deshpande        ScanResult.InformationElement elements[] = new ScanResult.InformationElement[num];
1427f1daf9342b66bf134d13fa0a42e929a008f1ca62Vinit Deshapnde        for (int i = 0, index = 0; i < num; i++) {
1428e7b9e74a9a699e74881e5c98c684567763dfaa35Vinit Deshpande            int type  = bytes[index] & 0xFF;
1429e7b9e74a9a699e74881e5c98c684567763dfaa35Vinit Deshpande            int len = bytes[index + 1] & 0xFF;
1430dd490cf1df37b70a8625dc4ec8e712f740dd1e4aPierre Vandwalle            if (DBG) Log.i(TAG, dbg + "index = " + index + ", type = " + type + ", len = " + len);
1431476bee2fef10d060c25c35858b1f7f60803d9f49Vinit Deshpande            ScanResult.InformationElement elem = new ScanResult.InformationElement();
1432f1daf9342b66bf134d13fa0a42e929a008f1ca62Vinit Deshapnde            elem.id = type;
1433f1daf9342b66bf134d13fa0a42e929a008f1ca62Vinit Deshapnde            elem.bytes = new byte[len];
1434f1daf9342b66bf134d13fa0a42e929a008f1ca62Vinit Deshapnde            for (int j = 0; j < len; j++) {
1435f1daf9342b66bf134d13fa0a42e929a008f1ca62Vinit Deshapnde                elem.bytes[j] = bytes[index + j + 2];
1436f1daf9342b66bf134d13fa0a42e929a008f1ca62Vinit Deshapnde            }
1437f1daf9342b66bf134d13fa0a42e929a008f1ca62Vinit Deshapnde            elements[i] = elem;
1438243931f3474f6235cfcf5c1a55fa2f192ee264aexinhe            int inforStart = index + 2;
1439f1daf9342b66bf134d13fa0a42e929a008f1ca62Vinit Deshapnde            index += (len + 2);
1440243931f3474f6235cfcf5c1a55fa2f192ee264aexinhe
1441243931f3474f6235cfcf5c1a55fa2f192ee264aexinhe            if(type == EID_HT_OPERATION) {
1442243931f3474f6235cfcf5c1a55fa2f192ee264aexinhe                secondChanelOffset = bytes[inforStart + 1] & 0x3;
1443243931f3474f6235cfcf5c1a55fa2f192ee264aexinhe            } else if(type == EID_VHT_OPERATION) {
1444243931f3474f6235cfcf5c1a55fa2f192ee264aexinhe                channelMode = bytes[inforStart];
1445243931f3474f6235cfcf5c1a55fa2f192ee264aexinhe                centerFreqIndex1 = bytes[inforStart + 1];
1446243931f3474f6235cfcf5c1a55fa2f192ee264aexinhe                centerFreqIndex2 = bytes[inforStart + 2];
1447243931f3474f6235cfcf5c1a55fa2f192ee264aexinhe            } else if (type == EID_EXTENDED_CAPS) {
1448dd490cf1df37b70a8625dc4ec8e712f740dd1e4aPierre Vandwalle                int tempIndex = RTT_RESP_ENABLE_BIT / 8;
1449dd490cf1df37b70a8625dc4ec8e712f740dd1e4aPierre Vandwalle                byte offset = RTT_RESP_ENABLE_BIT % 8;
1450dd490cf1df37b70a8625dc4ec8e712f740dd1e4aPierre Vandwalle
1451dd490cf1df37b70a8625dc4ec8e712f740dd1e4aPierre Vandwalle                if(len < tempIndex + 1) {
1452d9f37b2f3df18e96246db93ec4c2a5159b5d3915Vinit Deshpande                    is80211McRTTResponder = false;
1453dd490cf1df37b70a8625dc4ec8e712f740dd1e4aPierre Vandwalle                } else {
1454dd490cf1df37b70a8625dc4ec8e712f740dd1e4aPierre Vandwalle                    if ((bytes[inforStart + tempIndex] & ((byte)0x1 << offset)) != 0) {
1455d9f37b2f3df18e96246db93ec4c2a5159b5d3915Vinit Deshpande                        is80211McRTTResponder = true;
1456dd490cf1df37b70a8625dc4ec8e712f740dd1e4aPierre Vandwalle                    } else {
1457d9f37b2f3df18e96246db93ec4c2a5159b5d3915Vinit Deshpande                        is80211McRTTResponder = false;
1458dd490cf1df37b70a8625dc4ec8e712f740dd1e4aPierre Vandwalle                    }
1459dd490cf1df37b70a8625dc4ec8e712f740dd1e4aPierre Vandwalle                }
1460243931f3474f6235cfcf5c1a55fa2f192ee264aexinhe            }
1461243931f3474f6235cfcf5c1a55fa2f192ee264aexinhe        }
1462d9f37b2f3df18e96246db93ec4c2a5159b5d3915Vinit Deshpande
1463d9f37b2f3df18e96246db93ec4c2a5159b5d3915Vinit Deshpande        if (is80211McRTTResponder) {
1464d9f37b2f3df18e96246db93ec4c2a5159b5d3915Vinit Deshpande            result.setFlag(ScanResult.FLAG_80211mc_RESPONDER);
1465d9f37b2f3df18e96246db93ec4c2a5159b5d3915Vinit Deshpande        } else {
1466d9f37b2f3df18e96246db93ec4c2a5159b5d3915Vinit Deshpande            result.clearFlag(ScanResult.FLAG_80211mc_RESPONDER);
1467d9f37b2f3df18e96246db93ec4c2a5159b5d3915Vinit Deshpande        }
1468d9f37b2f3df18e96246db93ec4c2a5159b5d3915Vinit Deshpande
1469243931f3474f6235cfcf5c1a55fa2f192ee264aexinhe        //handle RTT related information
1470243931f3474f6235cfcf5c1a55fa2f192ee264aexinhe        if (channelMode != 0) {
1471243931f3474f6235cfcf5c1a55fa2f192ee264aexinhe            // 80 or 160 MHz
1472243931f3474f6235cfcf5c1a55fa2f192ee264aexinhe            result.channelWidth = channelMode + 1;
1473243931f3474f6235cfcf5c1a55fa2f192ee264aexinhe
1474243931f3474f6235cfcf5c1a55fa2f192ee264aexinhe            //convert channel index to frequency in MHz, channel 36 is 5180MHz
1475243931f3474f6235cfcf5c1a55fa2f192ee264aexinhe            result.centerFreq0 = (centerFreqIndex1 - 36) * 5 + 5180;
1476243931f3474f6235cfcf5c1a55fa2f192ee264aexinhe
1477243931f3474f6235cfcf5c1a55fa2f192ee264aexinhe            if(channelMode > 1) { //160MHz
1478243931f3474f6235cfcf5c1a55fa2f192ee264aexinhe                result.centerFreq1 = (centerFreqIndex2 - 36) * 5 + 5180;
1479243931f3474f6235cfcf5c1a55fa2f192ee264aexinhe            } else {
1480243931f3474f6235cfcf5c1a55fa2f192ee264aexinhe                result.centerFreq1 = 0;
1481243931f3474f6235cfcf5c1a55fa2f192ee264aexinhe            }
1482243931f3474f6235cfcf5c1a55fa2f192ee264aexinhe        } else {
1483243931f3474f6235cfcf5c1a55fa2f192ee264aexinhe            //20 or 40 MHz
1484243931f3474f6235cfcf5c1a55fa2f192ee264aexinhe            if (secondChanelOffset != 0) {//40MHz
1485243931f3474f6235cfcf5c1a55fa2f192ee264aexinhe                result.channelWidth = 1;
1486243931f3474f6235cfcf5c1a55fa2f192ee264aexinhe                if (secondChanelOffset == 1) {
1487243931f3474f6235cfcf5c1a55fa2f192ee264aexinhe                    result.centerFreq0 = result.frequency + 20;
1488243931f3474f6235cfcf5c1a55fa2f192ee264aexinhe                } else if (secondChanelOffset == 3) {
1489243931f3474f6235cfcf5c1a55fa2f192ee264aexinhe                    result.centerFreq0 = result.frequency - 20;
1490243931f3474f6235cfcf5c1a55fa2f192ee264aexinhe                } else {
1491243931f3474f6235cfcf5c1a55fa2f192ee264aexinhe                    result.centerFreq0 = 0;
1492dd490cf1df37b70a8625dc4ec8e712f740dd1e4aPierre Vandwalle                    Log.e(TAG, dbg + ": Error on secondChanelOffset");
1493243931f3474f6235cfcf5c1a55fa2f192ee264aexinhe                }
1494243931f3474f6235cfcf5c1a55fa2f192ee264aexinhe            } else {
1495243931f3474f6235cfcf5c1a55fa2f192ee264aexinhe                result.centerFreq0  = 0;
1496243931f3474f6235cfcf5c1a55fa2f192ee264aexinhe                result.centerFreq1  = 0;
1497243931f3474f6235cfcf5c1a55fa2f192ee264aexinhe            }
1498243931f3474f6235cfcf5c1a55fa2f192ee264aexinhe            result.centerFreq1  = 0;
1499243931f3474f6235cfcf5c1a55fa2f192ee264aexinhe        }
1500243931f3474f6235cfcf5c1a55fa2f192ee264aexinhe        if(DBG) {
1501dd490cf1df37b70a8625dc4ec8e712f740dd1e4aPierre Vandwalle            Log.d(TAG, dbg + "SSID: " + result.SSID + " ChannelWidth is: " + result.channelWidth +
1502243931f3474f6235cfcf5c1a55fa2f192ee264aexinhe                    " PrimaryFreq: " + result.frequency +" mCenterfreq0: " + result.centerFreq0 +
1503d9f37b2f3df18e96246db93ec4c2a5159b5d3915Vinit Deshpande                    " mCenterfreq1: " + result.centerFreq1 + (is80211McRTTResponder ?
1504243931f3474f6235cfcf5c1a55fa2f192ee264aexinhe                    "Support RTT reponder: " : "Do not support RTT responder"));
1505f1daf9342b66bf134d13fa0a42e929a008f1ca62Vinit Deshapnde        }
1506f1daf9342b66bf134d13fa0a42e929a008f1ca62Vinit Deshapnde
1507476bee2fef10d060c25c35858b1f7f60803d9f49Vinit Deshpande        result.informationElements = elements;
1508dd490cf1df37b70a8625dc4ec8e712f740dd1e4aPierre Vandwalle    }
1509dd490cf1df37b70a8625dc4ec8e712f740dd1e4aPierre Vandwalle
1510dd490cf1df37b70a8625dc4ec8e712f740dd1e4aPierre Vandwalle    synchronized static void onFullScanResult(int id, ScanResult result, byte bytes[]) {
1511dd490cf1df37b70a8625dc4ec8e712f740dd1e4aPierre Vandwalle        if (DBG) Log.i(TAG, "Got a full scan results event, ssid = " + result.SSID + ", " +
1512dd490cf1df37b70a8625dc4ec8e712f740dd1e4aPierre Vandwalle                "num = " + bytes.length);
1513dd490cf1df37b70a8625dc4ec8e712f740dd1e4aPierre Vandwalle
1514dd490cf1df37b70a8625dc4ec8e712f740dd1e4aPierre Vandwalle        if (sScanEventHandler == null) {
1515dd490cf1df37b70a8625dc4ec8e712f740dd1e4aPierre Vandwalle            return;
1516dd490cf1df37b70a8625dc4ec8e712f740dd1e4aPierre Vandwalle        }
1517dd490cf1df37b70a8625dc4ec8e712f740dd1e4aPierre Vandwalle        populateScanResult(result, bytes, " onFullScanResult ");
1518dd490cf1df37b70a8625dc4ec8e712f740dd1e4aPierre Vandwalle
1519e7b9e74a9a699e74881e5c98c684567763dfaa35Vinit Deshpande        sScanEventHandler.onFullScanResult(result);
15207f9a15d554f69311a0db43347d6473a7c4c46e2eVinit Deshapnde    }
15217f9a15d554f69311a0db43347d6473a7c4c46e2eVinit Deshapnde
1522b0b1d59786de7ff1fc98e31a16a7d710458bf549Vinit Deshpande    private static int sScanCmdId = 0;
1523b0b1d59786de7ff1fc98e31a16a7d710458bf549Vinit Deshpande    private static ScanEventHandler sScanEventHandler;
1524b0b1d59786de7ff1fc98e31a16a7d710458bf549Vinit Deshpande    private static ScanSettings sScanSettings;
15257f9a15d554f69311a0db43347d6473a7c4c46e2eVinit Deshapnde
1526b0b1d59786de7ff1fc98e31a16a7d710458bf549Vinit Deshpande    synchronized public static boolean startScan(
1527b0b1d59786de7ff1fc98e31a16a7d710458bf549Vinit Deshpande            ScanSettings settings, ScanEventHandler eventHandler) {
15287f9a15d554f69311a0db43347d6473a7c4c46e2eVinit Deshapnde        synchronized (mLock) {
152971d0be16c79791a005d60f9d5fa7d2d81e1f8c80xinhe            if (isHalStarted()) {
1530b0b1d59786de7ff1fc98e31a16a7d710458bf549Vinit Deshpande
1531f49a59bda006b13e0118d144e0a4a5f569b2251eVinit Deshpande                if (sScanCmdId != 0) {
1532f49a59bda006b13e0118d144e0a4a5f569b2251eVinit Deshpande                    stopScan();
1533f49a59bda006b13e0118d144e0a4a5f569b2251eVinit Deshpande                } else if (sScanSettings != null || sScanEventHandler != null) {
1534b0b1d59786de7ff1fc98e31a16a7d710458bf549Vinit Deshpande                /* current scan is paused; no need to stop it */
1535f49a59bda006b13e0118d144e0a4a5f569b2251eVinit Deshpande                }
15367f9a15d554f69311a0db43347d6473a7c4c46e2eVinit Deshapnde
1537f49a59bda006b13e0118d144e0a4a5f569b2251eVinit Deshpande                sScanCmdId = getNewCmdIdLocked();
1538e4e3750390bec0a849a9153348b7c21b2cc8b843Vinit Deshapnde
1539f49a59bda006b13e0118d144e0a4a5f569b2251eVinit Deshpande                sScanSettings = settings;
1540f49a59bda006b13e0118d144e0a4a5f569b2251eVinit Deshpande                sScanEventHandler = eventHandler;
1541b0b1d59786de7ff1fc98e31a16a7d710458bf549Vinit Deshpande
1542f49a59bda006b13e0118d144e0a4a5f569b2251eVinit Deshpande                if (startScanNative(sWlan0Index, sScanCmdId, settings) == false) {
1543f49a59bda006b13e0118d144e0a4a5f569b2251eVinit Deshpande                    sScanEventHandler = null;
1544f49a59bda006b13e0118d144e0a4a5f569b2251eVinit Deshpande                    sScanSettings = null;
1545f49a59bda006b13e0118d144e0a4a5f569b2251eVinit Deshpande                    sScanCmdId = 0;
1546f49a59bda006b13e0118d144e0a4a5f569b2251eVinit Deshpande                    return false;
1547f49a59bda006b13e0118d144e0a4a5f569b2251eVinit Deshpande                }
1548f49a59bda006b13e0118d144e0a4a5f569b2251eVinit Deshpande
1549f49a59bda006b13e0118d144e0a4a5f569b2251eVinit Deshpande                return true;
1550f49a59bda006b13e0118d144e0a4a5f569b2251eVinit Deshpande            } else {
1551e4e3750390bec0a849a9153348b7c21b2cc8b843Vinit Deshapnde                return false;
1552e4e3750390bec0a849a9153348b7c21b2cc8b843Vinit Deshapnde            }
1553e4e3750390bec0a849a9153348b7c21b2cc8b843Vinit Deshapnde        }
15547f9a15d554f69311a0db43347d6473a7c4c46e2eVinit Deshapnde    }
15557f9a15d554f69311a0db43347d6473a7c4c46e2eVinit Deshapnde
1556b0b1d59786de7ff1fc98e31a16a7d710458bf549Vinit Deshpande    synchronized public static void stopScan() {
1557b0b1d59786de7ff1fc98e31a16a7d710458bf549Vinit Deshpande        synchronized (mLock) {
155871d0be16c79791a005d60f9d5fa7d2d81e1f8c80xinhe            if (isHalStarted()) {
1559f49a59bda006b13e0118d144e0a4a5f569b2251eVinit Deshpande                stopScanNative(sWlan0Index, sScanCmdId);
1560f49a59bda006b13e0118d144e0a4a5f569b2251eVinit Deshpande                sScanSettings = null;
1561f49a59bda006b13e0118d144e0a4a5f569b2251eVinit Deshpande                sScanEventHandler = null;
1562f49a59bda006b13e0118d144e0a4a5f569b2251eVinit Deshpande                sScanCmdId = 0;
1563f49a59bda006b13e0118d144e0a4a5f569b2251eVinit Deshpande            }
1564b0b1d59786de7ff1fc98e31a16a7d710458bf549Vinit Deshpande        }
1565b0b1d59786de7ff1fc98e31a16a7d710458bf549Vinit Deshpande    }
1566b0b1d59786de7ff1fc98e31a16a7d710458bf549Vinit Deshpande
1567b0b1d59786de7ff1fc98e31a16a7d710458bf549Vinit Deshpande    synchronized public static void pauseScan() {
15687f9a15d554f69311a0db43347d6473a7c4c46e2eVinit Deshapnde        synchronized (mLock) {
156971d0be16c79791a005d60f9d5fa7d2d81e1f8c80xinhe            if (isHalStarted()) {
1570f49a59bda006b13e0118d144e0a4a5f569b2251eVinit Deshpande                if (sScanCmdId != 0 && sScanSettings != null && sScanEventHandler != null) {
1571f49a59bda006b13e0118d144e0a4a5f569b2251eVinit Deshpande                    Log.d(TAG, "Pausing scan");
1572f49a59bda006b13e0118d144e0a4a5f569b2251eVinit Deshpande                    WifiScanner.ScanData scanData[] = getScanResultsNative(sWlan0Index, true);
1573f49a59bda006b13e0118d144e0a4a5f569b2251eVinit Deshpande                    stopScanNative(sWlan0Index, sScanCmdId);
1574f49a59bda006b13e0118d144e0a4a5f569b2251eVinit Deshpande                    sScanCmdId = 0;
1575f49a59bda006b13e0118d144e0a4a5f569b2251eVinit Deshpande                    sScanEventHandler.onScanPaused(scanData);
1576f49a59bda006b13e0118d144e0a4a5f569b2251eVinit Deshpande                }
1577b0b1d59786de7ff1fc98e31a16a7d710458bf549Vinit Deshpande            }
1578b0b1d59786de7ff1fc98e31a16a7d710458bf549Vinit Deshpande        }
1579b0b1d59786de7ff1fc98e31a16a7d710458bf549Vinit Deshpande    }
1580b0b1d59786de7ff1fc98e31a16a7d710458bf549Vinit Deshpande
1581b0b1d59786de7ff1fc98e31a16a7d710458bf549Vinit Deshpande    synchronized public static void restartScan() {
1582b0b1d59786de7ff1fc98e31a16a7d710458bf549Vinit Deshpande        synchronized (mLock) {
158371d0be16c79791a005d60f9d5fa7d2d81e1f8c80xinhe            if (isHalStarted()) {
1584f49a59bda006b13e0118d144e0a4a5f569b2251eVinit Deshpande                if (sScanCmdId == 0 && sScanSettings != null && sScanEventHandler != null) {
15856f3626faf50499dd95aa299bb1011c27ab05776dPierre Vandwalle                    Log.d(TAG, "Restarting scan");
1586f49a59bda006b13e0118d144e0a4a5f569b2251eVinit Deshpande                    ScanEventHandler handler = sScanEventHandler;
1587f49a59bda006b13e0118d144e0a4a5f569b2251eVinit Deshpande                    ScanSettings settings = sScanSettings;
1588f49a59bda006b13e0118d144e0a4a5f569b2251eVinit Deshpande                    if (startScan(sScanSettings, sScanEventHandler)) {
1589f49a59bda006b13e0118d144e0a4a5f569b2251eVinit Deshpande                        sScanEventHandler.onScanRestarted();
1590f49a59bda006b13e0118d144e0a4a5f569b2251eVinit Deshpande                    } else {
159183a674a18c84ff4f01377bbfd8988699dec93bc2Vinit Deshpande                    /* we are still paused; don't change state */
1592f49a59bda006b13e0118d144e0a4a5f569b2251eVinit Deshpande                        sScanEventHandler = handler;
1593f49a59bda006b13e0118d144e0a4a5f569b2251eVinit Deshpande                        sScanSettings = settings;
1594f49a59bda006b13e0118d144e0a4a5f569b2251eVinit Deshpande                    }
159583a674a18c84ff4f01377bbfd8988699dec93bc2Vinit Deshpande                }
1596b0b1d59786de7ff1fc98e31a16a7d710458bf549Vinit Deshpande            }
1597e4e3750390bec0a849a9153348b7c21b2cc8b843Vinit Deshapnde        }
1598e4e3750390bec0a849a9153348b7c21b2cc8b843Vinit Deshapnde    }
1599e4e3750390bec0a849a9153348b7c21b2cc8b843Vinit Deshapnde
160083a674a18c84ff4f01377bbfd8988699dec93bc2Vinit Deshpande    synchronized public static WifiScanner.ScanData[] getScanResults(boolean flush) {
1601aabe7a9f2f32915fd1f25416a6d2034a844005d6vandwalle        synchronized (mLock) {
160271d0be16c79791a005d60f9d5fa7d2d81e1f8c80xinhe            if (isHalStarted()) {
1603f49a59bda006b13e0118d144e0a4a5f569b2251eVinit Deshpande                return getScanResultsNative(sWlan0Index, flush);
1604f49a59bda006b13e0118d144e0a4a5f569b2251eVinit Deshpande            } else {
1605f49a59bda006b13e0118d144e0a4a5f569b2251eVinit Deshpande                return null;
1606f49a59bda006b13e0118d144e0a4a5f569b2251eVinit Deshpande            }
1607aabe7a9f2f32915fd1f25416a6d2034a844005d6vandwalle        }
1608e4e3750390bec0a849a9153348b7c21b2cc8b843Vinit Deshapnde    }
1609e4e3750390bec0a849a9153348b7c21b2cc8b843Vinit Deshapnde
1610b0b1d59786de7ff1fc98e31a16a7d710458bf549Vinit Deshpande    public static interface HotlistEventHandler {
1611d4762401ec14be6bdd2d27aff2478ddbf8d6ce2aVinit Deshpande        void onHotlistApFound (ScanResult[] result);
1612d4762401ec14be6bdd2d27aff2478ddbf8d6ce2aVinit Deshpande        void onHotlistApLost  (ScanResult[] result);
1613e4e3750390bec0a849a9153348b7c21b2cc8b843Vinit Deshapnde    }
1614e4e3750390bec0a849a9153348b7c21b2cc8b843Vinit Deshapnde
1615b0b1d59786de7ff1fc98e31a16a7d710458bf549Vinit Deshpande    private static int sHotlistCmdId = 0;
1616b0b1d59786de7ff1fc98e31a16a7d710458bf549Vinit Deshpande    private static HotlistEventHandler sHotlistEventHandler;
1617e4e3750390bec0a849a9153348b7c21b2cc8b843Vinit Deshapnde
1618b0b1d59786de7ff1fc98e31a16a7d710458bf549Vinit Deshpande    private native static boolean setHotlistNative(int iface, int id,
1619e4e3750390bec0a849a9153348b7c21b2cc8b843Vinit Deshapnde            WifiScanner.HotlistSettings settings);
1620b0b1d59786de7ff1fc98e31a16a7d710458bf549Vinit Deshpande    private native static boolean resetHotlistNative(int iface, int id);
1621e4e3750390bec0a849a9153348b7c21b2cc8b843Vinit Deshapnde
1622b0b1d59786de7ff1fc98e31a16a7d710458bf549Vinit Deshpande    synchronized public static boolean setHotlist(WifiScanner.HotlistSettings settings,
1623aabe7a9f2f32915fd1f25416a6d2034a844005d6vandwalle                                    HotlistEventHandler eventHandler) {
1624e4e3750390bec0a849a9153348b7c21b2cc8b843Vinit Deshapnde        synchronized (mLock) {
162571d0be16c79791a005d60f9d5fa7d2d81e1f8c80xinhe            if (isHalStarted()) {
1626f49a59bda006b13e0118d144e0a4a5f569b2251eVinit Deshpande                if (sHotlistCmdId != 0) {
1627f49a59bda006b13e0118d144e0a4a5f569b2251eVinit Deshpande                    return false;
1628f49a59bda006b13e0118d144e0a4a5f569b2251eVinit Deshpande                } else {
1629f49a59bda006b13e0118d144e0a4a5f569b2251eVinit Deshpande                    sHotlistCmdId = getNewCmdIdLocked();
1630f49a59bda006b13e0118d144e0a4a5f569b2251eVinit Deshpande                }
1631f49a59bda006b13e0118d144e0a4a5f569b2251eVinit Deshpande
1632f49a59bda006b13e0118d144e0a4a5f569b2251eVinit Deshpande                sHotlistEventHandler = eventHandler;
1633f49a59bda006b13e0118d144e0a4a5f569b2251eVinit Deshpande                if (setHotlistNative(sWlan0Index, sHotlistCmdId, settings) == false) {
1634f49a59bda006b13e0118d144e0a4a5f569b2251eVinit Deshpande                    sHotlistEventHandler = null;
1635f49a59bda006b13e0118d144e0a4a5f569b2251eVinit Deshpande                    return false;
1636f49a59bda006b13e0118d144e0a4a5f569b2251eVinit Deshpande                }
1637e4e3750390bec0a849a9153348b7c21b2cc8b843Vinit Deshapnde
1638f49a59bda006b13e0118d144e0a4a5f569b2251eVinit Deshpande                return true;
1639f49a59bda006b13e0118d144e0a4a5f569b2251eVinit Deshpande            } else {
1640e4e3750390bec0a849a9153348b7c21b2cc8b843Vinit Deshapnde                return false;
1641e4e3750390bec0a849a9153348b7c21b2cc8b843Vinit Deshapnde            }
1642e4e3750390bec0a849a9153348b7c21b2cc8b843Vinit Deshapnde        }
1643e4e3750390bec0a849a9153348b7c21b2cc8b843Vinit Deshapnde    }
1644e4e3750390bec0a849a9153348b7c21b2cc8b843Vinit Deshapnde
1645b0b1d59786de7ff1fc98e31a16a7d710458bf549Vinit Deshpande    synchronized public static void resetHotlist() {
1646e4e3750390bec0a849a9153348b7c21b2cc8b843Vinit Deshapnde        synchronized (mLock) {
164771d0be16c79791a005d60f9d5fa7d2d81e1f8c80xinhe            if (isHalStarted()) {
1648f49a59bda006b13e0118d144e0a4a5f569b2251eVinit Deshpande                if (sHotlistCmdId != 0) {
1649f49a59bda006b13e0118d144e0a4a5f569b2251eVinit Deshpande                    resetHotlistNative(sWlan0Index, sHotlistCmdId);
1650f49a59bda006b13e0118d144e0a4a5f569b2251eVinit Deshpande                    sHotlistCmdId = 0;
1651f49a59bda006b13e0118d144e0a4a5f569b2251eVinit Deshpande                    sHotlistEventHandler = null;
1652f49a59bda006b13e0118d144e0a4a5f569b2251eVinit Deshpande                }
1653e4e3750390bec0a849a9153348b7c21b2cc8b843Vinit Deshapnde            }
16547f9a15d554f69311a0db43347d6473a7c4c46e2eVinit Deshapnde        }
16557f9a15d554f69311a0db43347d6473a7c4c46e2eVinit Deshapnde    }
1656e4e3750390bec0a849a9153348b7c21b2cc8b843Vinit Deshapnde
1657b0b1d59786de7ff1fc98e31a16a7d710458bf549Vinit Deshpande    synchronized public static void onHotlistApFound(int id, ScanResult[] results) {
1658aabe7a9f2f32915fd1f25416a6d2034a844005d6vandwalle        synchronized (mLock) {
165971d0be16c79791a005d60f9d5fa7d2d81e1f8c80xinhe            if (isHalStarted()) {
1660f49a59bda006b13e0118d144e0a4a5f569b2251eVinit Deshpande                if (sHotlistCmdId != 0) {
1661f49a59bda006b13e0118d144e0a4a5f569b2251eVinit Deshpande                    sHotlistEventHandler.onHotlistApFound(results);
1662f49a59bda006b13e0118d144e0a4a5f569b2251eVinit Deshpande                } else {
16631814928371a30b11af31e2bbe5210c4337ed16f3Vinit Deshpande                /* this can happen because of race conditions */
1664f49a59bda006b13e0118d144e0a4a5f569b2251eVinit Deshpande                    Log.d(TAG, "Ignoring hotlist AP found event");
1665f49a59bda006b13e0118d144e0a4a5f569b2251eVinit Deshpande                }
1666d4762401ec14be6bdd2d27aff2478ddbf8d6ce2aVinit Deshpande            }
1667d4762401ec14be6bdd2d27aff2478ddbf8d6ce2aVinit Deshpande        }
1668d4762401ec14be6bdd2d27aff2478ddbf8d6ce2aVinit Deshpande    }
1669d4762401ec14be6bdd2d27aff2478ddbf8d6ce2aVinit Deshpande
1670d4762401ec14be6bdd2d27aff2478ddbf8d6ce2aVinit Deshpande    synchronized public static void onHotlistApLost(int id, ScanResult[] results) {
1671d4762401ec14be6bdd2d27aff2478ddbf8d6ce2aVinit Deshpande        synchronized (mLock) {
167271d0be16c79791a005d60f9d5fa7d2d81e1f8c80xinhe            if (isHalStarted()) {
1673f49a59bda006b13e0118d144e0a4a5f569b2251eVinit Deshpande                if (sHotlistCmdId != 0) {
1674f49a59bda006b13e0118d144e0a4a5f569b2251eVinit Deshpande                    sHotlistEventHandler.onHotlistApLost(results);
1675f49a59bda006b13e0118d144e0a4a5f569b2251eVinit Deshpande                } else {
1676d4762401ec14be6bdd2d27aff2478ddbf8d6ce2aVinit Deshpande                /* this can happen because of race conditions */
1677f49a59bda006b13e0118d144e0a4a5f569b2251eVinit Deshpande                    Log.d(TAG, "Ignoring hotlist AP lost event");
1678f49a59bda006b13e0118d144e0a4a5f569b2251eVinit Deshpande                }
16791814928371a30b11af31e2bbe5210c4337ed16f3Vinit Deshpande            }
1680aabe7a9f2f32915fd1f25416a6d2034a844005d6vandwalle        }
1681e4e3750390bec0a849a9153348b7c21b2cc8b843Vinit Deshapnde    }
1682e4e3750390bec0a849a9153348b7c21b2cc8b843Vinit Deshapnde
1683b0b1d59786de7ff1fc98e31a16a7d710458bf549Vinit Deshpande    public static interface SignificantWifiChangeEventHandler {
1684e4e3750390bec0a849a9153348b7c21b2cc8b843Vinit Deshapnde        void onChangesFound(ScanResult[] result);
1685e4e3750390bec0a849a9153348b7c21b2cc8b843Vinit Deshapnde    }
1686e4e3750390bec0a849a9153348b7c21b2cc8b843Vinit Deshapnde
1687b0b1d59786de7ff1fc98e31a16a7d710458bf549Vinit Deshpande    private static SignificantWifiChangeEventHandler sSignificantWifiChangeHandler;
1688b0b1d59786de7ff1fc98e31a16a7d710458bf549Vinit Deshpande    private static int sSignificantWifiChangeCmdId;
1689e4e3750390bec0a849a9153348b7c21b2cc8b843Vinit Deshapnde
1690b0b1d59786de7ff1fc98e31a16a7d710458bf549Vinit Deshpande    private static native boolean trackSignificantWifiChangeNative(
1691e4e3750390bec0a849a9153348b7c21b2cc8b843Vinit Deshapnde            int iface, int id, WifiScanner.WifiChangeSettings settings);
1692b0b1d59786de7ff1fc98e31a16a7d710458bf549Vinit Deshpande    private static native boolean untrackSignificantWifiChangeNative(int iface, int id);
1693e4e3750390bec0a849a9153348b7c21b2cc8b843Vinit Deshapnde
1694b0b1d59786de7ff1fc98e31a16a7d710458bf549Vinit Deshpande    synchronized public static boolean trackSignificantWifiChange(
1695b0b1d59786de7ff1fc98e31a16a7d710458bf549Vinit Deshpande            WifiScanner.WifiChangeSettings settings, SignificantWifiChangeEventHandler handler) {
1696e4e3750390bec0a849a9153348b7c21b2cc8b843Vinit Deshapnde        synchronized (mLock) {
169771d0be16c79791a005d60f9d5fa7d2d81e1f8c80xinhe            if (isHalStarted()) {
1698f49a59bda006b13e0118d144e0a4a5f569b2251eVinit Deshpande                if (sSignificantWifiChangeCmdId != 0) {
1699f49a59bda006b13e0118d144e0a4a5f569b2251eVinit Deshpande                    return false;
1700f49a59bda006b13e0118d144e0a4a5f569b2251eVinit Deshpande                } else {
1701f49a59bda006b13e0118d144e0a4a5f569b2251eVinit Deshpande                    sSignificantWifiChangeCmdId = getNewCmdIdLocked();
1702f49a59bda006b13e0118d144e0a4a5f569b2251eVinit Deshpande                }
1703f49a59bda006b13e0118d144e0a4a5f569b2251eVinit Deshpande
1704f49a59bda006b13e0118d144e0a4a5f569b2251eVinit Deshpande                sSignificantWifiChangeHandler = handler;
1705f49a59bda006b13e0118d144e0a4a5f569b2251eVinit Deshpande                if (trackSignificantWifiChangeNative(sWlan0Index, sScanCmdId, settings) == false) {
1706f49a59bda006b13e0118d144e0a4a5f569b2251eVinit Deshpande                    sSignificantWifiChangeHandler = null;
1707f49a59bda006b13e0118d144e0a4a5f569b2251eVinit Deshpande                    return false;
1708f49a59bda006b13e0118d144e0a4a5f569b2251eVinit Deshpande                }
1709e4e3750390bec0a849a9153348b7c21b2cc8b843Vinit Deshapnde
1710f49a59bda006b13e0118d144e0a4a5f569b2251eVinit Deshpande                return true;
1711f49a59bda006b13e0118d144e0a4a5f569b2251eVinit Deshpande            } else {
1712e4e3750390bec0a849a9153348b7c21b2cc8b843Vinit Deshapnde                return false;
1713e4e3750390bec0a849a9153348b7c21b2cc8b843Vinit Deshapnde            }
1714e4e3750390bec0a849a9153348b7c21b2cc8b843Vinit Deshapnde
1715e4e3750390bec0a849a9153348b7c21b2cc8b843Vinit Deshapnde        }
1716e4e3750390bec0a849a9153348b7c21b2cc8b843Vinit Deshapnde    }
1717e4e3750390bec0a849a9153348b7c21b2cc8b843Vinit Deshapnde
1718b0b1d59786de7ff1fc98e31a16a7d710458bf549Vinit Deshpande    synchronized static void untrackSignificantWifiChange() {
1719e4e3750390bec0a849a9153348b7c21b2cc8b843Vinit Deshapnde        synchronized (mLock) {
172071d0be16c79791a005d60f9d5fa7d2d81e1f8c80xinhe            if (isHalStarted()) {
1721f49a59bda006b13e0118d144e0a4a5f569b2251eVinit Deshpande                if (sSignificantWifiChangeCmdId != 0) {
1722f49a59bda006b13e0118d144e0a4a5f569b2251eVinit Deshpande                    untrackSignificantWifiChangeNative(sWlan0Index, sSignificantWifiChangeCmdId);
1723f49a59bda006b13e0118d144e0a4a5f569b2251eVinit Deshpande                    sSignificantWifiChangeCmdId = 0;
1724f49a59bda006b13e0118d144e0a4a5f569b2251eVinit Deshpande                    sSignificantWifiChangeHandler = null;
1725f49a59bda006b13e0118d144e0a4a5f569b2251eVinit Deshpande                }
1726e4e3750390bec0a849a9153348b7c21b2cc8b843Vinit Deshapnde            }
1727e4e3750390bec0a849a9153348b7c21b2cc8b843Vinit Deshapnde        }
1728e4e3750390bec0a849a9153348b7c21b2cc8b843Vinit Deshapnde    }
1729e4e3750390bec0a849a9153348b7c21b2cc8b843Vinit Deshapnde
1730b0b1d59786de7ff1fc98e31a16a7d710458bf549Vinit Deshpande    synchronized static void onSignificantWifiChange(int id, ScanResult[] results) {
1731aabe7a9f2f32915fd1f25416a6d2034a844005d6vandwalle        synchronized (mLock) {
17321814928371a30b11af31e2bbe5210c4337ed16f3Vinit Deshpande            if (sSignificantWifiChangeCmdId != 0) {
17331814928371a30b11af31e2bbe5210c4337ed16f3Vinit Deshpande                sSignificantWifiChangeHandler.onChangesFound(results);
17341814928371a30b11af31e2bbe5210c4337ed16f3Vinit Deshpande            } else {
1735f49a59bda006b13e0118d144e0a4a5f569b2251eVinit Deshpande            /* this can happen because of race conditions */
17361814928371a30b11af31e2bbe5210c4337ed16f3Vinit Deshpande                Log.d(TAG, "Ignoring significant wifi change");
17371814928371a30b11af31e2bbe5210c4337ed16f3Vinit Deshpande            }
1738aabe7a9f2f32915fd1f25416a6d2034a844005d6vandwalle        }
1739e4e3750390bec0a849a9153348b7c21b2cc8b843Vinit Deshapnde    }
1740e4e3750390bec0a849a9153348b7c21b2cc8b843Vinit Deshapnde
1741200e8ee5097134010a6edee8d031bb02ff7eeb5avandwalle    synchronized public static WifiLinkLayerStats getWifiLinkLayerStats(String iface) {
1742200e8ee5097134010a6edee8d031bb02ff7eeb5avandwalle        // TODO: use correct iface name to Index translation
1743200e8ee5097134010a6edee8d031bb02ff7eeb5avandwalle        if (iface == null) return null;
1744aabe7a9f2f32915fd1f25416a6d2034a844005d6vandwalle        synchronized (mLock) {
174571d0be16c79791a005d60f9d5fa7d2d81e1f8c80xinhe            if (isHalStarted()) {
1746aabe7a9f2f32915fd1f25416a6d2034a844005d6vandwalle                return getWifiLinkLayerStatsNative(sWlan0Index);
1747f49a59bda006b13e0118d144e0a4a5f569b2251eVinit Deshpande            } else {
1748f49a59bda006b13e0118d144e0a4a5f569b2251eVinit Deshpande                return null;
1749f49a59bda006b13e0118d144e0a4a5f569b2251eVinit Deshpande            }
1750aabe7a9f2f32915fd1f25416a6d2034a844005d6vandwalle        }
17515c08cc119b92af69997af194cc8b6d0111e37d31Andres Morales    }
17525c08cc119b92af69997af194cc8b6d0111e37d31Andres Morales
1753d745a52dc4f929d4a4030f205ed173bdf60eaf10Pierre Vandwalle    synchronized public static void setWifiLinkLayerStats(String iface, int enable) {
1754d745a52dc4f929d4a4030f205ed173bdf60eaf10Pierre Vandwalle        if (iface == null) return;
1755d745a52dc4f929d4a4030f205ed173bdf60eaf10Pierre Vandwalle        synchronized (mLock) {
175671d0be16c79791a005d60f9d5fa7d2d81e1f8c80xinhe            if (isHalStarted()) {
1757d745a52dc4f929d4a4030f205ed173bdf60eaf10Pierre Vandwalle                setWifiLinkLayerStatsNative(sWlan0Index, enable);
1758d745a52dc4f929d4a4030f205ed173bdf60eaf10Pierre Vandwalle            }
1759d745a52dc4f929d4a4030f205ed173bdf60eaf10Pierre Vandwalle        }
1760d745a52dc4f929d4a4030f205ed173bdf60eaf10Pierre Vandwalle    }
1761d745a52dc4f929d4a4030f205ed173bdf60eaf10Pierre Vandwalle
1762c35361d54d4885c3174499e4ad46d3324387a9bbVinit Deshpande    public static native int getSupportedFeatureSetNative(int iface);
1763a632d8a6edd350c7644b593b18eceaa5b368505bVinit Deshpande    synchronized public static int getSupportedFeatureSet() {
1764f49a59bda006b13e0118d144e0a4a5f569b2251eVinit Deshpande        synchronized (mLock) {
176571d0be16c79791a005d60f9d5fa7d2d81e1f8c80xinhe            if (isHalStarted()) {
1766f49a59bda006b13e0118d144e0a4a5f569b2251eVinit Deshpande                return getSupportedFeatureSetNative(sWlan0Index);
1767f49a59bda006b13e0118d144e0a4a5f569b2251eVinit Deshpande            } else {
1768f49a59bda006b13e0118d144e0a4a5f569b2251eVinit Deshpande                Log.d(TAG, "Failing getSupportedFeatureset because HAL isn't started");
1769f49a59bda006b13e0118d144e0a4a5f569b2251eVinit Deshpande                return 0;
1770f49a59bda006b13e0118d144e0a4a5f569b2251eVinit Deshpande            }
1771f49a59bda006b13e0118d144e0a4a5f569b2251eVinit Deshpande        }
1772a632d8a6edd350c7644b593b18eceaa5b368505bVinit Deshpande    }
1773143657392bf0702a155fe688171a5abbf4c86570Vinit Deshpande
1774143657392bf0702a155fe688171a5abbf4c86570Vinit Deshpande    /* Rtt related commands/events */
1775143657392bf0702a155fe688171a5abbf4c86570Vinit Deshpande    public static interface RttEventHandler {
1776143657392bf0702a155fe688171a5abbf4c86570Vinit Deshpande        void onRttResults(RttManager.RttResult[] result);
1777143657392bf0702a155fe688171a5abbf4c86570Vinit Deshpande    }
1778143657392bf0702a155fe688171a5abbf4c86570Vinit Deshpande
1779143657392bf0702a155fe688171a5abbf4c86570Vinit Deshpande    private static RttEventHandler sRttEventHandler;
1780143657392bf0702a155fe688171a5abbf4c86570Vinit Deshpande    private static int sRttCmdId;
1781143657392bf0702a155fe688171a5abbf4c86570Vinit Deshpande
1782143657392bf0702a155fe688171a5abbf4c86570Vinit Deshpande    synchronized private static void onRttResults(int id, RttManager.RttResult[] results) {
1783143657392bf0702a155fe688171a5abbf4c86570Vinit Deshpande        if (id == sRttCmdId) {
178402a1f98f2cecb8ae2d466d6f9fab06b473f970ddVinit Deshpande            Log.d(TAG, "Received " + results.length + " rtt results");
1785143657392bf0702a155fe688171a5abbf4c86570Vinit Deshpande            sRttEventHandler.onRttResults(results);
1786143657392bf0702a155fe688171a5abbf4c86570Vinit Deshpande            sRttCmdId = 0;
1787143657392bf0702a155fe688171a5abbf4c86570Vinit Deshpande        } else {
1788f95649f33db0a328cb4c0bb5e10c7075e6c828f8xinhe            Log.d(TAG, "RTT Received event for unknown cmd = " + id + ", current id = " + sRttCmdId);
1789143657392bf0702a155fe688171a5abbf4c86570Vinit Deshpande        }
1790143657392bf0702a155fe688171a5abbf4c86570Vinit Deshpande    }
1791143657392bf0702a155fe688171a5abbf4c86570Vinit Deshpande
1792143657392bf0702a155fe688171a5abbf4c86570Vinit Deshpande    private static native boolean requestRangeNative(
1793143657392bf0702a155fe688171a5abbf4c86570Vinit Deshpande            int iface, int id, RttManager.RttParams[] params);
1794143657392bf0702a155fe688171a5abbf4c86570Vinit Deshpande    private static native boolean cancelRangeRequestNative(
1795143657392bf0702a155fe688171a5abbf4c86570Vinit Deshpande            int iface, int id, RttManager.RttParams[] params);
1796143657392bf0702a155fe688171a5abbf4c86570Vinit Deshpande
1797143657392bf0702a155fe688171a5abbf4c86570Vinit Deshpande    synchronized public static boolean requestRtt(
1798143657392bf0702a155fe688171a5abbf4c86570Vinit Deshpande            RttManager.RttParams[] params, RttEventHandler handler) {
1799143657392bf0702a155fe688171a5abbf4c86570Vinit Deshpande        synchronized (mLock) {
180071d0be16c79791a005d60f9d5fa7d2d81e1f8c80xinhe            if (isHalStarted()) {
1801f49a59bda006b13e0118d144e0a4a5f569b2251eVinit Deshpande                if (sRttCmdId != 0) {
1802f49a59bda006b13e0118d144e0a4a5f569b2251eVinit Deshpande                    Log.v("TAG", "Last one is still under measurement!");
1803f49a59bda006b13e0118d144e0a4a5f569b2251eVinit Deshpande                    return false;
1804f49a59bda006b13e0118d144e0a4a5f569b2251eVinit Deshpande                } else {
1805f49a59bda006b13e0118d144e0a4a5f569b2251eVinit Deshpande                    sRttCmdId = getNewCmdIdLocked();
1806f49a59bda006b13e0118d144e0a4a5f569b2251eVinit Deshpande                }
1807f49a59bda006b13e0118d144e0a4a5f569b2251eVinit Deshpande                sRttEventHandler = handler;
1808f49a59bda006b13e0118d144e0a4a5f569b2251eVinit Deshpande                Log.v(TAG, "native issue RTT request");
1809f49a59bda006b13e0118d144e0a4a5f569b2251eVinit Deshpande                return requestRangeNative(sWlan0Index, sRttCmdId, params);
1810143657392bf0702a155fe688171a5abbf4c86570Vinit Deshpande            } else {
1811f49a59bda006b13e0118d144e0a4a5f569b2251eVinit Deshpande                return false;
1812143657392bf0702a155fe688171a5abbf4c86570Vinit Deshpande            }
1813143657392bf0702a155fe688171a5abbf4c86570Vinit Deshpande        }
1814143657392bf0702a155fe688171a5abbf4c86570Vinit Deshpande    }
1815143657392bf0702a155fe688171a5abbf4c86570Vinit Deshpande
1816143657392bf0702a155fe688171a5abbf4c86570Vinit Deshpande    synchronized public static boolean cancelRtt(RttManager.RttParams[] params) {
1817143657392bf0702a155fe688171a5abbf4c86570Vinit Deshpande        synchronized(mLock) {
181871d0be16c79791a005d60f9d5fa7d2d81e1f8c80xinhe            if (isHalStarted()) {
1819f49a59bda006b13e0118d144e0a4a5f569b2251eVinit Deshpande                if (sRttCmdId == 0) {
1820f49a59bda006b13e0118d144e0a4a5f569b2251eVinit Deshpande                    return false;
1821f49a59bda006b13e0118d144e0a4a5f569b2251eVinit Deshpande                }
1822143657392bf0702a155fe688171a5abbf4c86570Vinit Deshpande
1823f49a59bda006b13e0118d144e0a4a5f569b2251eVinit Deshpande                sRttCmdId = 0;
1824f95649f33db0a328cb4c0bb5e10c7075e6c828f8xinhe
1825f49a59bda006b13e0118d144e0a4a5f569b2251eVinit Deshpande                if (cancelRangeRequestNative(sWlan0Index, sRttCmdId, params)) {
1826f49a59bda006b13e0118d144e0a4a5f569b2251eVinit Deshpande                    sRttEventHandler = null;
1827f49a59bda006b13e0118d144e0a4a5f569b2251eVinit Deshpande                    Log.v(TAG, "Xin: RTT cancel Request Successfully");
1828f49a59bda006b13e0118d144e0a4a5f569b2251eVinit Deshpande                    return true;
1829f49a59bda006b13e0118d144e0a4a5f569b2251eVinit Deshpande                } else {
1830f49a59bda006b13e0118d144e0a4a5f569b2251eVinit Deshpande                    Log.e(TAG, "Xin:RTT cancel Request failed");
1831f49a59bda006b13e0118d144e0a4a5f569b2251eVinit Deshpande                    return false;
1832f49a59bda006b13e0118d144e0a4a5f569b2251eVinit Deshpande                }
1833143657392bf0702a155fe688171a5abbf4c86570Vinit Deshpande            } else {
1834143657392bf0702a155fe688171a5abbf4c86570Vinit Deshpande                return false;
1835143657392bf0702a155fe688171a5abbf4c86570Vinit Deshpande            }
1836143657392bf0702a155fe688171a5abbf4c86570Vinit Deshpande        }
1837143657392bf0702a155fe688171a5abbf4c86570Vinit Deshpande    }
1838042c54bfd5144ce8e720585b2093796e7e28de5eVinit Deshpande
1839042c54bfd5144ce8e720585b2093796e7e28de5eVinit Deshpande    private static native boolean setScanningMacOuiNative(int iface, byte[] oui);
1840042c54bfd5144ce8e720585b2093796e7e28de5eVinit Deshpande
1841042c54bfd5144ce8e720585b2093796e7e28de5eVinit Deshpande    synchronized public static boolean setScanningMacOui(byte[] oui) {
1842042c54bfd5144ce8e720585b2093796e7e28de5eVinit Deshpande        synchronized (mLock) {
184371d0be16c79791a005d60f9d5fa7d2d81e1f8c80xinhe            if (isHalStarted()) {
1844042c54bfd5144ce8e720585b2093796e7e28de5eVinit Deshpande                return setScanningMacOuiNative(sWlan0Index, oui);
1845042c54bfd5144ce8e720585b2093796e7e28de5eVinit Deshpande            } else {
1846042c54bfd5144ce8e720585b2093796e7e28de5eVinit Deshpande                return false;
1847042c54bfd5144ce8e720585b2093796e7e28de5eVinit Deshpande            }
1848042c54bfd5144ce8e720585b2093796e7e28de5eVinit Deshpande        }
1849042c54bfd5144ce8e720585b2093796e7e28de5eVinit Deshpande    }
1850efa77c1826499b0a3e57998bd6b3073b107e45d7Vinit Deshpande
1851efa77c1826499b0a3e57998bd6b3073b107e45d7Vinit Deshpande    private static native int[] getChannelsForBandNative(
1852efa77c1826499b0a3e57998bd6b3073b107e45d7Vinit Deshpande            int iface, int band);
1853efa77c1826499b0a3e57998bd6b3073b107e45d7Vinit Deshpande
1854efa77c1826499b0a3e57998bd6b3073b107e45d7Vinit Deshpande    synchronized public static int [] getChannelsForBand(int band) {
1855efa77c1826499b0a3e57998bd6b3073b107e45d7Vinit Deshpande        synchronized (mLock) {
1856f49a59bda006b13e0118d144e0a4a5f569b2251eVinit Deshpande            return getChannelsForBandNative(sWlan0Index, band);
1857efa77c1826499b0a3e57998bd6b3073b107e45d7Vinit Deshpande        }
1858efa77c1826499b0a3e57998bd6b3073b107e45d7Vinit Deshpande    }
18590465ff513cdccf7e883a505e14c5b78758e7e458Vinit Deshpande
18600465ff513cdccf7e883a505e14c5b78758e7e458Vinit Deshpande
18610465ff513cdccf7e883a505e14c5b78758e7e458Vinit Deshpande    private static native boolean setDfsFlagNative(int iface, boolean dfsOn);
18620465ff513cdccf7e883a505e14c5b78758e7e458Vinit Deshpande    synchronized public static boolean setDfsFlag(boolean dfsOn) {
18630465ff513cdccf7e883a505e14c5b78758e7e458Vinit Deshpande        synchronized (mLock) {
186471d0be16c79791a005d60f9d5fa7d2d81e1f8c80xinhe            if (isHalStarted()) {
18650465ff513cdccf7e883a505e14c5b78758e7e458Vinit Deshpande                return setDfsFlagNative(sWlan0Index, dfsOn);
18660465ff513cdccf7e883a505e14c5b78758e7e458Vinit Deshpande            } else {
18670465ff513cdccf7e883a505e14c5b78758e7e458Vinit Deshpande                return false;
18680465ff513cdccf7e883a505e14c5b78758e7e458Vinit Deshpande            }
18690465ff513cdccf7e883a505e14c5b78758e7e458Vinit Deshpande        }
18700465ff513cdccf7e883a505e14c5b78758e7e458Vinit Deshpande    }
1871b830d76fbf5fa4077531b516066faa2fdbb92e81xinhe
1872b830d76fbf5fa4077531b516066faa2fdbb92e81xinhe    private static native boolean toggleInterfaceNative(int on);
1873b830d76fbf5fa4077531b516066faa2fdbb92e81xinhe    synchronized public static boolean toggleInterface(int on) {
1874b830d76fbf5fa4077531b516066faa2fdbb92e81xinhe        synchronized (mLock) {
187571d0be16c79791a005d60f9d5fa7d2d81e1f8c80xinhe            if (isHalStarted()) {
1876b830d76fbf5fa4077531b516066faa2fdbb92e81xinhe                return toggleInterfaceNative(0);
1877b830d76fbf5fa4077531b516066faa2fdbb92e81xinhe            } else {
1878b830d76fbf5fa4077531b516066faa2fdbb92e81xinhe                return false;
1879b830d76fbf5fa4077531b516066faa2fdbb92e81xinhe            }
1880b830d76fbf5fa4077531b516066faa2fdbb92e81xinhe        }
1881b830d76fbf5fa4077531b516066faa2fdbb92e81xinhe    }
188212cf388ecf3b2aa5ba66ed680b931fa356ab75b5xinhe
188312cf388ecf3b2aa5ba66ed680b931fa356ab75b5xinhe    private static native RttManager.RttCapabilities getRttCapabilitiesNative(int iface);
188412cf388ecf3b2aa5ba66ed680b931fa356ab75b5xinhe    synchronized public static RttManager.RttCapabilities getRttCapabilities() {
188512cf388ecf3b2aa5ba66ed680b931fa356ab75b5xinhe        synchronized (mLock) {
188671d0be16c79791a005d60f9d5fa7d2d81e1f8c80xinhe            if (isHalStarted()) {
188712cf388ecf3b2aa5ba66ed680b931fa356ab75b5xinhe                return getRttCapabilitiesNative(sWlan0Index);
1888f49a59bda006b13e0118d144e0a4a5f569b2251eVinit Deshpande            }else {
188912cf388ecf3b2aa5ba66ed680b931fa356ab75b5xinhe                return null;
189012cf388ecf3b2aa5ba66ed680b931fa356ab75b5xinhe            }
189112cf388ecf3b2aa5ba66ed680b931fa356ab75b5xinhe        }
189212cf388ecf3b2aa5ba66ed680b931fa356ab75b5xinhe    }
1893939177ff615062ec826601d536466875d8457375xinhe
1894939177ff615062ec826601d536466875d8457375xinhe    private static native boolean setCountryCodeHalNative(int iface, String CountryCode);
1895939177ff615062ec826601d536466875d8457375xinhe    synchronized public static boolean setCountryCodeHal( String CountryCode) {
1896939177ff615062ec826601d536466875d8457375xinhe        synchronized (mLock) {
189771d0be16c79791a005d60f9d5fa7d2d81e1f8c80xinhe            if (isHalStarted()) {
1898939177ff615062ec826601d536466875d8457375xinhe                return setCountryCodeHalNative(sWlan0Index, CountryCode);
1899939177ff615062ec826601d536466875d8457375xinhe            } else {
1900939177ff615062ec826601d536466875d8457375xinhe                return false;
1901939177ff615062ec826601d536466875d8457375xinhe            }
1902939177ff615062ec826601d536466875d8457375xinhe        }
1903939177ff615062ec826601d536466875d8457375xinhe    }
1904939177ff615062ec826601d536466875d8457375xinhe
1905d57f630f6a7a5fca872d2b96fc4cce1905daee5dxinhe    /* Rtt related commands/events */
1906d57f630f6a7a5fca872d2b96fc4cce1905daee5dxinhe    public abstract class TdlsEventHandler {
1907d57f630f6a7a5fca872d2b96fc4cce1905daee5dxinhe        abstract public void onTdlsStatus(String macAddr, int status, int reason);
1908d57f630f6a7a5fca872d2b96fc4cce1905daee5dxinhe    }
1909d57f630f6a7a5fca872d2b96fc4cce1905daee5dxinhe
1910d57f630f6a7a5fca872d2b96fc4cce1905daee5dxinhe    private static TdlsEventHandler sTdlsEventHandler;
1911d57f630f6a7a5fca872d2b96fc4cce1905daee5dxinhe
1912d57f630f6a7a5fca872d2b96fc4cce1905daee5dxinhe    private static native boolean enableDisableTdlsNative(int iface, boolean enable,
1913d57f630f6a7a5fca872d2b96fc4cce1905daee5dxinhe            String macAddr);
1914d57f630f6a7a5fca872d2b96fc4cce1905daee5dxinhe    synchronized public static boolean enableDisableTdls(boolean enable, String macAdd,
1915d57f630f6a7a5fca872d2b96fc4cce1905daee5dxinhe            TdlsEventHandler tdlsCallBack) {
1916d57f630f6a7a5fca872d2b96fc4cce1905daee5dxinhe        synchronized (mLock) {
1917f49a59bda006b13e0118d144e0a4a5f569b2251eVinit Deshpande            sTdlsEventHandler = tdlsCallBack;
1918f49a59bda006b13e0118d144e0a4a5f569b2251eVinit Deshpande            return enableDisableTdlsNative(sWlan0Index, enable, macAdd);
1919d57f630f6a7a5fca872d2b96fc4cce1905daee5dxinhe        }
1920d57f630f6a7a5fca872d2b96fc4cce1905daee5dxinhe    }
1921d57f630f6a7a5fca872d2b96fc4cce1905daee5dxinhe
1922d57f630f6a7a5fca872d2b96fc4cce1905daee5dxinhe    // Once TDLS per mac and event feature is implemented, this class definition should be
1923d57f630f6a7a5fca872d2b96fc4cce1905daee5dxinhe    // moved to the right place, like WifiManager etc
1924d57f630f6a7a5fca872d2b96fc4cce1905daee5dxinhe    public static class TdlsStatus {
1925d57f630f6a7a5fca872d2b96fc4cce1905daee5dxinhe        int channel;
1926d57f630f6a7a5fca872d2b96fc4cce1905daee5dxinhe        int global_operating_class;
1927d57f630f6a7a5fca872d2b96fc4cce1905daee5dxinhe        int state;
1928d57f630f6a7a5fca872d2b96fc4cce1905daee5dxinhe        int reason;
1929d57f630f6a7a5fca872d2b96fc4cce1905daee5dxinhe    }
1930d57f630f6a7a5fca872d2b96fc4cce1905daee5dxinhe    private static native TdlsStatus getTdlsStatusNative(int iface, String macAddr);
1931d57f630f6a7a5fca872d2b96fc4cce1905daee5dxinhe    synchronized public static TdlsStatus getTdlsStatus (String macAdd) {
1932d57f630f6a7a5fca872d2b96fc4cce1905daee5dxinhe        synchronized (mLock) {
193371d0be16c79791a005d60f9d5fa7d2d81e1f8c80xinhe            if (isHalStarted()) {
1934d57f630f6a7a5fca872d2b96fc4cce1905daee5dxinhe                return getTdlsStatusNative(sWlan0Index, macAdd);
1935d57f630f6a7a5fca872d2b96fc4cce1905daee5dxinhe            } else {
1936d57f630f6a7a5fca872d2b96fc4cce1905daee5dxinhe                return null;
1937d57f630f6a7a5fca872d2b96fc4cce1905daee5dxinhe            }
1938d57f630f6a7a5fca872d2b96fc4cce1905daee5dxinhe        }
1939d57f630f6a7a5fca872d2b96fc4cce1905daee5dxinhe    }
1940d57f630f6a7a5fca872d2b96fc4cce1905daee5dxinhe
1941d57f630f6a7a5fca872d2b96fc4cce1905daee5dxinhe    //ToFix: Once TDLS per mac and event feature is implemented, this class definition should be
1942d57f630f6a7a5fca872d2b96fc4cce1905daee5dxinhe    // moved to the right place, like WifiStateMachine etc
1943d57f630f6a7a5fca872d2b96fc4cce1905daee5dxinhe    public static class TdlsCapabilities {
1944d57f630f6a7a5fca872d2b96fc4cce1905daee5dxinhe        /* Maximum TDLS session number can be supported by the Firmware and hardware */
1945d57f630f6a7a5fca872d2b96fc4cce1905daee5dxinhe        int maxConcurrentTdlsSessionNumber;
1946d57f630f6a7a5fca872d2b96fc4cce1905daee5dxinhe        boolean isGlobalTdlsSupported;
1947d57f630f6a7a5fca872d2b96fc4cce1905daee5dxinhe        boolean isPerMacTdlsSupported;
1948d57f630f6a7a5fca872d2b96fc4cce1905daee5dxinhe        boolean isOffChannelTdlsSupported;
1949d57f630f6a7a5fca872d2b96fc4cce1905daee5dxinhe    }
1950d57f630f6a7a5fca872d2b96fc4cce1905daee5dxinhe
1951d57f630f6a7a5fca872d2b96fc4cce1905daee5dxinhe
1952d57f630f6a7a5fca872d2b96fc4cce1905daee5dxinhe
1953d57f630f6a7a5fca872d2b96fc4cce1905daee5dxinhe    private static native TdlsCapabilities getTdlsCapabilitiesNative(int iface);
1954d57f630f6a7a5fca872d2b96fc4cce1905daee5dxinhe    synchronized public static TdlsCapabilities getTdlsCapabilities () {
1955d57f630f6a7a5fca872d2b96fc4cce1905daee5dxinhe        synchronized (mLock) {
195671d0be16c79791a005d60f9d5fa7d2d81e1f8c80xinhe            if (isHalStarted()) {
1957d57f630f6a7a5fca872d2b96fc4cce1905daee5dxinhe                return getTdlsCapabilitiesNative(sWlan0Index);
1958d57f630f6a7a5fca872d2b96fc4cce1905daee5dxinhe            } else {
1959d57f630f6a7a5fca872d2b96fc4cce1905daee5dxinhe                return null;
1960d57f630f6a7a5fca872d2b96fc4cce1905daee5dxinhe            }
1961d57f630f6a7a5fca872d2b96fc4cce1905daee5dxinhe        }
1962d57f630f6a7a5fca872d2b96fc4cce1905daee5dxinhe    }
1963d57f630f6a7a5fca872d2b96fc4cce1905daee5dxinhe
1964d57f630f6a7a5fca872d2b96fc4cce1905daee5dxinhe    synchronized private static boolean onTdlsStatus(String macAddr, int status, int reason) {
1965d57f630f6a7a5fca872d2b96fc4cce1905daee5dxinhe         if (sTdlsEventHandler == null) {
1966d57f630f6a7a5fca872d2b96fc4cce1905daee5dxinhe             return false;
1967d57f630f6a7a5fca872d2b96fc4cce1905daee5dxinhe         } else {
1968d57f630f6a7a5fca872d2b96fc4cce1905daee5dxinhe             sTdlsEventHandler.onTdlsStatus(macAddr, status, reason);
1969d57f630f6a7a5fca872d2b96fc4cce1905daee5dxinhe             return true;
1970d57f630f6a7a5fca872d2b96fc4cce1905daee5dxinhe         }
1971d57f630f6a7a5fca872d2b96fc4cce1905daee5dxinhe    }
1972d57f630f6a7a5fca872d2b96fc4cce1905daee5dxinhe
1973a0d34d3626bd9631039a485d2d428b1b7de390e2Pierre Vandwalle    //---------------------------------------------------------------------------------
1974a0d34d3626bd9631039a485d2d428b1b7de390e2Pierre Vandwalle
1975a0d34d3626bd9631039a485d2d428b1b7de390e2Pierre Vandwalle    /* Wifi Logger commands/events */
19767d6301ead19afdf3de37455e9ed133c25b4938cdVinit Deshpande
197703ba4a5b108ab2bbdb5d0e7be46ac45c00a6d9ffxinhe    public static native boolean startLogging(int iface);
19787d6301ead19afdf3de37455e9ed133c25b4938cdVinit Deshpande
1979a0d34d3626bd9631039a485d2d428b1b7de390e2Pierre Vandwalle    public static interface WifiLoggerEventHandler {
19800bf150b2d69ec62df905bc6f8603437dc4caee1fVinit Deshpande        void onRingBufferData(RingBufferStatus status, byte[] buffer);
19810bf150b2d69ec62df905bc6f8603437dc4caee1fVinit Deshpande        void onWifiAlert(int errorCode, byte[] buffer);
1982a0d34d3626bd9631039a485d2d428b1b7de390e2Pierre Vandwalle    }
1983a0d34d3626bd9631039a485d2d428b1b7de390e2Pierre Vandwalle
1984a0d34d3626bd9631039a485d2d428b1b7de390e2Pierre Vandwalle    private static WifiLoggerEventHandler sWifiLoggerEventHandler = null;
1985a0d34d3626bd9631039a485d2d428b1b7de390e2Pierre Vandwalle
19860bf150b2d69ec62df905bc6f8603437dc4caee1fVinit Deshpande    private static void onRingBufferData(RingBufferStatus status, byte[] buffer) {
19870bf150b2d69ec62df905bc6f8603437dc4caee1fVinit Deshpande        if (sWifiLoggerEventHandler != null)
19880bf150b2d69ec62df905bc6f8603437dc4caee1fVinit Deshpande            sWifiLoggerEventHandler.onRingBufferData(status, buffer);
198903ba4a5b108ab2bbdb5d0e7be46ac45c00a6d9ffxinhe    }
199003ba4a5b108ab2bbdb5d0e7be46ac45c00a6d9ffxinhe
19910bf150b2d69ec62df905bc6f8603437dc4caee1fVinit Deshpande    private static void onWifiAlert(byte[] buffer, int errorCode) {
19920bf150b2d69ec62df905bc6f8603437dc4caee1fVinit Deshpande        if (sWifiLoggerEventHandler != null)
19930bf150b2d69ec62df905bc6f8603437dc4caee1fVinit Deshpande            sWifiLoggerEventHandler.onWifiAlert(errorCode, buffer);
199403ba4a5b108ab2bbdb5d0e7be46ac45c00a6d9ffxinhe    }
199503ba4a5b108ab2bbdb5d0e7be46ac45c00a6d9ffxinhe
1996b797893fc1966803d0c013faac42e6396a37a384xinhe    private static int sLogCmdId = -1;
1997b797893fc1966803d0c013faac42e6396a37a384xinhe    private static native boolean setLoggingEventHandlerNative(int iface, int id);
1998b797893fc1966803d0c013faac42e6396a37a384xinhe    synchronized public static boolean setLoggingEventHandler(WifiLoggerEventHandler handler) {
19990bf150b2d69ec62df905bc6f8603437dc4caee1fVinit Deshpande        synchronized (mLock) {
2000b797893fc1966803d0c013faac42e6396a37a384xinhe            if (isHalStarted()) {
2001b797893fc1966803d0c013faac42e6396a37a384xinhe                int oldId =  sLogCmdId;
2002b797893fc1966803d0c013faac42e6396a37a384xinhe                sLogCmdId = getNewCmdIdLocked();
2003b797893fc1966803d0c013faac42e6396a37a384xinhe                if (!setLoggingEventHandlerNative(sWlan0Index, sLogCmdId)) {
2004b797893fc1966803d0c013faac42e6396a37a384xinhe                    sLogCmdId = oldId;
2005b797893fc1966803d0c013faac42e6396a37a384xinhe                    return false;
2006b797893fc1966803d0c013faac42e6396a37a384xinhe                }
2007b797893fc1966803d0c013faac42e6396a37a384xinhe                sWifiLoggerEventHandler = handler;
2008b797893fc1966803d0c013faac42e6396a37a384xinhe                return true;
2009b797893fc1966803d0c013faac42e6396a37a384xinhe            } else {
2010b797893fc1966803d0c013faac42e6396a37a384xinhe                return false;
2011b797893fc1966803d0c013faac42e6396a37a384xinhe            }
201203ba4a5b108ab2bbdb5d0e7be46ac45c00a6d9ffxinhe        }
201303ba4a5b108ab2bbdb5d0e7be46ac45c00a6d9ffxinhe    }
201403ba4a5b108ab2bbdb5d0e7be46ac45c00a6d9ffxinhe
201503ba4a5b108ab2bbdb5d0e7be46ac45c00a6d9ffxinhe    private static native boolean startLoggingRingBufferNative(int iface, int verboseLevel,
20160bf150b2d69ec62df905bc6f8603437dc4caee1fVinit Deshpande            int flags, int minIntervalSec ,int minDataSize, String ringName);
201703ba4a5b108ab2bbdb5d0e7be46ac45c00a6d9ffxinhe    synchronized public static boolean startLoggingRingBuffer(int verboseLevel, int flags, int maxInterval,
201803ba4a5b108ab2bbdb5d0e7be46ac45c00a6d9ffxinhe            int minDataSize, String ringName){
201903ba4a5b108ab2bbdb5d0e7be46ac45c00a6d9ffxinhe        synchronized (mLock) {
202071d0be16c79791a005d60f9d5fa7d2d81e1f8c80xinhe            if (isHalStarted()) {
202103ba4a5b108ab2bbdb5d0e7be46ac45c00a6d9ffxinhe                return startLoggingRingBufferNative(sWlan0Index, verboseLevel, flags, maxInterval,
202203ba4a5b108ab2bbdb5d0e7be46ac45c00a6d9ffxinhe                        minDataSize, ringName);
202303ba4a5b108ab2bbdb5d0e7be46ac45c00a6d9ffxinhe            } else {
202403ba4a5b108ab2bbdb5d0e7be46ac45c00a6d9ffxinhe                return false;
202503ba4a5b108ab2bbdb5d0e7be46ac45c00a6d9ffxinhe            }
202603ba4a5b108ab2bbdb5d0e7be46ac45c00a6d9ffxinhe        }
202703ba4a5b108ab2bbdb5d0e7be46ac45c00a6d9ffxinhe    }
202803ba4a5b108ab2bbdb5d0e7be46ac45c00a6d9ffxinhe
202903ba4a5b108ab2bbdb5d0e7be46ac45c00a6d9ffxinhe    private static native int getSupportedLoggerFeatureSetNative(int iface);
203003ba4a5b108ab2bbdb5d0e7be46ac45c00a6d9ffxinhe    synchronized public static int getSupportedLoggerFeatureSet() {
203103ba4a5b108ab2bbdb5d0e7be46ac45c00a6d9ffxinhe        synchronized (mLock) {
203271d0be16c79791a005d60f9d5fa7d2d81e1f8c80xinhe            if (isHalStarted()) {
203303ba4a5b108ab2bbdb5d0e7be46ac45c00a6d9ffxinhe                return getSupportedLoggerFeatureSetNative(sWlan0Index);
203403ba4a5b108ab2bbdb5d0e7be46ac45c00a6d9ffxinhe            } else {
2035f49a59bda006b13e0118d144e0a4a5f569b2251eVinit Deshpande                return 0;
203603ba4a5b108ab2bbdb5d0e7be46ac45c00a6d9ffxinhe            }
203703ba4a5b108ab2bbdb5d0e7be46ac45c00a6d9ffxinhe        }
203803ba4a5b108ab2bbdb5d0e7be46ac45c00a6d9ffxinhe    }
203903ba4a5b108ab2bbdb5d0e7be46ac45c00a6d9ffxinhe
2040b797893fc1966803d0c013faac42e6396a37a384xinhe    private static native boolean resetLogHandlerNative(int iface, int id);
2041b797893fc1966803d0c013faac42e6396a37a384xinhe    synchronized public static boolean resetLogHandler() {
2042b797893fc1966803d0c013faac42e6396a37a384xinhe        synchronized (mLock) {
2043b797893fc1966803d0c013faac42e6396a37a384xinhe            if (isHalStarted()) {
2044b797893fc1966803d0c013faac42e6396a37a384xinhe                if (sLogCmdId == -1) {
2045b797893fc1966803d0c013faac42e6396a37a384xinhe                    Log.e(TAG,"Can not reset handler Before set any handler");
2046b797893fc1966803d0c013faac42e6396a37a384xinhe                    return false;
2047b797893fc1966803d0c013faac42e6396a37a384xinhe                }
2048b797893fc1966803d0c013faac42e6396a37a384xinhe                sWifiLoggerEventHandler = null;
2049b797893fc1966803d0c013faac42e6396a37a384xinhe                if (resetLogHandlerNative(sWlan0Index, sLogCmdId)) {
2050b797893fc1966803d0c013faac42e6396a37a384xinhe                    sLogCmdId = -1;
2051b797893fc1966803d0c013faac42e6396a37a384xinhe                    return true;
2052b797893fc1966803d0c013faac42e6396a37a384xinhe                } else {
2053b797893fc1966803d0c013faac42e6396a37a384xinhe                    return false;
2054b797893fc1966803d0c013faac42e6396a37a384xinhe                }
2055b797893fc1966803d0c013faac42e6396a37a384xinhe            } else {
2056b797893fc1966803d0c013faac42e6396a37a384xinhe                return false;
2057b797893fc1966803d0c013faac42e6396a37a384xinhe            }
2058b797893fc1966803d0c013faac42e6396a37a384xinhe        }
2059b797893fc1966803d0c013faac42e6396a37a384xinhe    }
2060b797893fc1966803d0c013faac42e6396a37a384xinhe
206103ba4a5b108ab2bbdb5d0e7be46ac45c00a6d9ffxinhe    private static native String getDriverVersionNative(int iface);
206203ba4a5b108ab2bbdb5d0e7be46ac45c00a6d9ffxinhe    synchronized public static String getDriverVersion() {
206303ba4a5b108ab2bbdb5d0e7be46ac45c00a6d9ffxinhe        synchronized (mLock) {
206471d0be16c79791a005d60f9d5fa7d2d81e1f8c80xinhe            if (isHalStarted()) {
206503ba4a5b108ab2bbdb5d0e7be46ac45c00a6d9ffxinhe                return getDriverVersionNative(sWlan0Index);
206603ba4a5b108ab2bbdb5d0e7be46ac45c00a6d9ffxinhe            } else {
2067f49a59bda006b13e0118d144e0a4a5f569b2251eVinit Deshpande                return "";
206803ba4a5b108ab2bbdb5d0e7be46ac45c00a6d9ffxinhe            }
206903ba4a5b108ab2bbdb5d0e7be46ac45c00a6d9ffxinhe        }
207003ba4a5b108ab2bbdb5d0e7be46ac45c00a6d9ffxinhe    }
207103ba4a5b108ab2bbdb5d0e7be46ac45c00a6d9ffxinhe
207203ba4a5b108ab2bbdb5d0e7be46ac45c00a6d9ffxinhe
207303ba4a5b108ab2bbdb5d0e7be46ac45c00a6d9ffxinhe    private static native String getFirmwareVersionNative(int iface);
207403ba4a5b108ab2bbdb5d0e7be46ac45c00a6d9ffxinhe    synchronized public static String getFirmwareVersion() {
207503ba4a5b108ab2bbdb5d0e7be46ac45c00a6d9ffxinhe        synchronized (mLock) {
207671d0be16c79791a005d60f9d5fa7d2d81e1f8c80xinhe            if (isHalStarted()) {
207703ba4a5b108ab2bbdb5d0e7be46ac45c00a6d9ffxinhe                return getFirmwareVersionNative(sWlan0Index);
207803ba4a5b108ab2bbdb5d0e7be46ac45c00a6d9ffxinhe            } else {
2079f49a59bda006b13e0118d144e0a4a5f569b2251eVinit Deshpande                return "";
208003ba4a5b108ab2bbdb5d0e7be46ac45c00a6d9ffxinhe            }
208103ba4a5b108ab2bbdb5d0e7be46ac45c00a6d9ffxinhe        }
208203ba4a5b108ab2bbdb5d0e7be46ac45c00a6d9ffxinhe    }
208303ba4a5b108ab2bbdb5d0e7be46ac45c00a6d9ffxinhe
20840bf150b2d69ec62df905bc6f8603437dc4caee1fVinit Deshpande    public static class RingBufferStatus{
20850bf150b2d69ec62df905bc6f8603437dc4caee1fVinit Deshpande        String name;
20860bf150b2d69ec62df905bc6f8603437dc4caee1fVinit Deshpande        int flag;
20870bf150b2d69ec62df905bc6f8603437dc4caee1fVinit Deshpande        int ringBufferId;
20880bf150b2d69ec62df905bc6f8603437dc4caee1fVinit Deshpande        int ringBufferByteSize;
20890bf150b2d69ec62df905bc6f8603437dc4caee1fVinit Deshpande        int verboseLevel;
20900bf150b2d69ec62df905bc6f8603437dc4caee1fVinit Deshpande        int writtenBytes;
20910bf150b2d69ec62df905bc6f8603437dc4caee1fVinit Deshpande        int readBytes;
20920bf150b2d69ec62df905bc6f8603437dc4caee1fVinit Deshpande        int writtenRecords;
20930bf150b2d69ec62df905bc6f8603437dc4caee1fVinit Deshpande
20940bf150b2d69ec62df905bc6f8603437dc4caee1fVinit Deshpande        @Override
20950bf150b2d69ec62df905bc6f8603437dc4caee1fVinit Deshpande        public String toString() {
20960bf150b2d69ec62df905bc6f8603437dc4caee1fVinit Deshpande            return "name: " + name + " flag: " + flag + " ringBufferId: " + ringBufferId +
20970bf150b2d69ec62df905bc6f8603437dc4caee1fVinit Deshpande                    " ringBufferByteSize: " +ringBufferByteSize + " verboseLevel: " +verboseLevel +
20980bf150b2d69ec62df905bc6f8603437dc4caee1fVinit Deshpande                    " writtenBytes: " + writtenBytes + " readBytes: " + readBytes +
20990bf150b2d69ec62df905bc6f8603437dc4caee1fVinit Deshpande                    " writtenRecords: " + writtenRecords;
21000bf150b2d69ec62df905bc6f8603437dc4caee1fVinit Deshpande        }
21010bf150b2d69ec62df905bc6f8603437dc4caee1fVinit Deshpande    }
21020bf150b2d69ec62df905bc6f8603437dc4caee1fVinit Deshpande
21030bf150b2d69ec62df905bc6f8603437dc4caee1fVinit Deshpande    private static native RingBufferStatus[] getRingBufferStatusNative(int iface);
21040bf150b2d69ec62df905bc6f8603437dc4caee1fVinit Deshpande    synchronized public static RingBufferStatus[] getRingBufferStatus() {
210503ba4a5b108ab2bbdb5d0e7be46ac45c00a6d9ffxinhe        synchronized (mLock) {
210671d0be16c79791a005d60f9d5fa7d2d81e1f8c80xinhe            if (isHalStarted()) {
210703ba4a5b108ab2bbdb5d0e7be46ac45c00a6d9ffxinhe                return getRingBufferStatusNative(sWlan0Index);
210803ba4a5b108ab2bbdb5d0e7be46ac45c00a6d9ffxinhe            } else {
210903ba4a5b108ab2bbdb5d0e7be46ac45c00a6d9ffxinhe                return null;
211003ba4a5b108ab2bbdb5d0e7be46ac45c00a6d9ffxinhe            }
211103ba4a5b108ab2bbdb5d0e7be46ac45c00a6d9ffxinhe        }
211203ba4a5b108ab2bbdb5d0e7be46ac45c00a6d9ffxinhe    }
211303ba4a5b108ab2bbdb5d0e7be46ac45c00a6d9ffxinhe
211403ba4a5b108ab2bbdb5d0e7be46ac45c00a6d9ffxinhe    private static native boolean getRingBufferDataNative(int iface, String ringName);
211503ba4a5b108ab2bbdb5d0e7be46ac45c00a6d9ffxinhe    synchronized public static boolean getRingBufferData(String ringName) {
211603ba4a5b108ab2bbdb5d0e7be46ac45c00a6d9ffxinhe        synchronized (mLock) {
211771d0be16c79791a005d60f9d5fa7d2d81e1f8c80xinhe            if (isHalStarted()) {
211803ba4a5b108ab2bbdb5d0e7be46ac45c00a6d9ffxinhe                return getRingBufferDataNative(sWlan0Index, ringName);
211903ba4a5b108ab2bbdb5d0e7be46ac45c00a6d9ffxinhe            } else {
212003ba4a5b108ab2bbdb5d0e7be46ac45c00a6d9ffxinhe                return false;
212103ba4a5b108ab2bbdb5d0e7be46ac45c00a6d9ffxinhe            }
212203ba4a5b108ab2bbdb5d0e7be46ac45c00a6d9ffxinhe        }
212303ba4a5b108ab2bbdb5d0e7be46ac45c00a6d9ffxinhe    }
212403ba4a5b108ab2bbdb5d0e7be46ac45c00a6d9ffxinhe
212598dca3397c0049d3cfb9c97e2b021c5ee3a2e1bdxinhe    static private byte[] mFwMemoryDump;
212603ba4a5b108ab2bbdb5d0e7be46ac45c00a6d9ffxinhe    private static void onWifiFwMemoryAvailable(byte[] buffer) {
212798dca3397c0049d3cfb9c97e2b021c5ee3a2e1bdxinhe        mFwMemoryDump = buffer;
212898dca3397c0049d3cfb9c97e2b021c5ee3a2e1bdxinhe        if (DBG) {
212998dca3397c0049d3cfb9c97e2b021c5ee3a2e1bdxinhe            Log.d(TAG, "onWifiFwMemoryAvailable is called and buffer length is: " +
213098dca3397c0049d3cfb9c97e2b021c5ee3a2e1bdxinhe                    (buffer == null ? 0 :  buffer.length));
213198dca3397c0049d3cfb9c97e2b021c5ee3a2e1bdxinhe        }
213203ba4a5b108ab2bbdb5d0e7be46ac45c00a6d9ffxinhe    }
2133127f7244183786e6ccae09e81eeccdac31973e69xinhe
213403ba4a5b108ab2bbdb5d0e7be46ac45c00a6d9ffxinhe    private static native boolean getFwMemoryDumpNative(int iface);
21350bf150b2d69ec62df905bc6f8603437dc4caee1fVinit Deshpande    synchronized public static byte[] getFwMemoryDump() {
213603ba4a5b108ab2bbdb5d0e7be46ac45c00a6d9ffxinhe        synchronized (mLock) {
213771d0be16c79791a005d60f9d5fa7d2d81e1f8c80xinhe            if (isHalStarted()) {
21380bf150b2d69ec62df905bc6f8603437dc4caee1fVinit Deshpande                if(getFwMemoryDumpNative(sWlan0Index)) {
21390bf150b2d69ec62df905bc6f8603437dc4caee1fVinit Deshpande                    byte[] fwMemoryDump = mFwMemoryDump;
214003ba4a5b108ab2bbdb5d0e7be46ac45c00a6d9ffxinhe                    mFwMemoryDump = null;
21410bf150b2d69ec62df905bc6f8603437dc4caee1fVinit Deshpande                    return fwMemoryDump;
21420bf150b2d69ec62df905bc6f8603437dc4caee1fVinit Deshpande                } else {
21430bf150b2d69ec62df905bc6f8603437dc4caee1fVinit Deshpande                    return null;
214403ba4a5b108ab2bbdb5d0e7be46ac45c00a6d9ffxinhe                }
214503ba4a5b108ab2bbdb5d0e7be46ac45c00a6d9ffxinhe            }
2146f49a59bda006b13e0118d144e0a4a5f569b2251eVinit Deshpande
2147f49a59bda006b13e0118d144e0a4a5f569b2251eVinit Deshpande            return null;
2148a0d34d3626bd9631039a485d2d428b1b7de390e2Pierre Vandwalle        }
2149a0d34d3626bd9631039a485d2d428b1b7de390e2Pierre Vandwalle    }
2150dd490cf1df37b70a8625dc4ec8e712f740dd1e4aPierre Vandwalle
2151dd490cf1df37b70a8625dc4ec8e712f740dd1e4aPierre Vandwalle    //---------------------------------------------------------------------------------
2152dd490cf1df37b70a8625dc4ec8e712f740dd1e4aPierre Vandwalle    /* Configure ePNO */
2153dd490cf1df37b70a8625dc4ec8e712f740dd1e4aPierre Vandwalle
2154dd490cf1df37b70a8625dc4ec8e712f740dd1e4aPierre Vandwalle    public class WifiPnoNetwork {
2155dd490cf1df37b70a8625dc4ec8e712f740dd1e4aPierre Vandwalle        String SSID;
2156dd490cf1df37b70a8625dc4ec8e712f740dd1e4aPierre Vandwalle        int rssi_threshold;
2157dd490cf1df37b70a8625dc4ec8e712f740dd1e4aPierre Vandwalle        int flags;
2158dd490cf1df37b70a8625dc4ec8e712f740dd1e4aPierre Vandwalle        int auth;
2159dd490cf1df37b70a8625dc4ec8e712f740dd1e4aPierre Vandwalle        String configKey; // kept for reference
2160dd490cf1df37b70a8625dc4ec8e712f740dd1e4aPierre Vandwalle
2161dd490cf1df37b70a8625dc4ec8e712f740dd1e4aPierre Vandwalle        WifiPnoNetwork(WifiConfiguration config, int threshold) {
2162dd490cf1df37b70a8625dc4ec8e712f740dd1e4aPierre Vandwalle            if (config.SSID == null) {
2163dd490cf1df37b70a8625dc4ec8e712f740dd1e4aPierre Vandwalle                this.SSID = "";
2164dd490cf1df37b70a8625dc4ec8e712f740dd1e4aPierre Vandwalle                this.flags = 1;
2165dd490cf1df37b70a8625dc4ec8e712f740dd1e4aPierre Vandwalle            } else {
2166dd490cf1df37b70a8625dc4ec8e712f740dd1e4aPierre Vandwalle                this.SSID = config.SSID;
2167dd490cf1df37b70a8625dc4ec8e712f740dd1e4aPierre Vandwalle            }
2168dd490cf1df37b70a8625dc4ec8e712f740dd1e4aPierre Vandwalle            this.rssi_threshold = threshold;
2169dd490cf1df37b70a8625dc4ec8e712f740dd1e4aPierre Vandwalle            if (config.allowedKeyManagement.get(WifiConfiguration.KeyMgmt.WPA_PSK)) {
2170dd490cf1df37b70a8625dc4ec8e712f740dd1e4aPierre Vandwalle                auth |= 2;
2171dd490cf1df37b70a8625dc4ec8e712f740dd1e4aPierre Vandwalle            } else if (config.allowedKeyManagement.get(WifiConfiguration.KeyMgmt.WPA_EAP) ||
2172dd490cf1df37b70a8625dc4ec8e712f740dd1e4aPierre Vandwalle                    config.allowedKeyManagement.get(WifiConfiguration.KeyMgmt.IEEE8021X)) {
2173dd490cf1df37b70a8625dc4ec8e712f740dd1e4aPierre Vandwalle                auth |= 4;
2174dd490cf1df37b70a8625dc4ec8e712f740dd1e4aPierre Vandwalle            } else if (config.wepKeys[0] != null) {
2175dd490cf1df37b70a8625dc4ec8e712f740dd1e4aPierre Vandwalle                auth |= 1;
2176dd490cf1df37b70a8625dc4ec8e712f740dd1e4aPierre Vandwalle            } else {
2177dd490cf1df37b70a8625dc4ec8e712f740dd1e4aPierre Vandwalle                auth |= 1;
2178dd490cf1df37b70a8625dc4ec8e712f740dd1e4aPierre Vandwalle            }
2179dd490cf1df37b70a8625dc4ec8e712f740dd1e4aPierre Vandwalle//            auth = 0;
2180dd490cf1df37b70a8625dc4ec8e712f740dd1e4aPierre Vandwalle            flags |= 6; //A and G
2181dd490cf1df37b70a8625dc4ec8e712f740dd1e4aPierre Vandwalle            configKey = config.configKey();
2182dd490cf1df37b70a8625dc4ec8e712f740dd1e4aPierre Vandwalle        }
2183dd490cf1df37b70a8625dc4ec8e712f740dd1e4aPierre Vandwalle
2184dd490cf1df37b70a8625dc4ec8e712f740dd1e4aPierre Vandwalle        @Override
2185dd490cf1df37b70a8625dc4ec8e712f740dd1e4aPierre Vandwalle        public String toString() {
2186dd490cf1df37b70a8625dc4ec8e712f740dd1e4aPierre Vandwalle            StringBuilder sbuf = new StringBuilder();
2187dd490cf1df37b70a8625dc4ec8e712f740dd1e4aPierre Vandwalle            sbuf.append(this.SSID);
2188dd490cf1df37b70a8625dc4ec8e712f740dd1e4aPierre Vandwalle            sbuf.append(" flags=").append(this.flags);
2189dd490cf1df37b70a8625dc4ec8e712f740dd1e4aPierre Vandwalle            sbuf.append(" rssi=").append(this.rssi_threshold);
2190dd490cf1df37b70a8625dc4ec8e712f740dd1e4aPierre Vandwalle            sbuf.append(" auth=").append(this.auth);
2191dd490cf1df37b70a8625dc4ec8e712f740dd1e4aPierre Vandwalle            return sbuf.toString();
2192dd490cf1df37b70a8625dc4ec8e712f740dd1e4aPierre Vandwalle        }
2193dd490cf1df37b70a8625dc4ec8e712f740dd1e4aPierre Vandwalle    }
2194dd490cf1df37b70a8625dc4ec8e712f740dd1e4aPierre Vandwalle
2195dd490cf1df37b70a8625dc4ec8e712f740dd1e4aPierre Vandwalle    public static interface WifiPnoEventHandler {
2196dd490cf1df37b70a8625dc4ec8e712f740dd1e4aPierre Vandwalle        void onPnoNetworkFound(ScanResult results[]);
2197dd490cf1df37b70a8625dc4ec8e712f740dd1e4aPierre Vandwalle    }
2198dd490cf1df37b70a8625dc4ec8e712f740dd1e4aPierre Vandwalle
2199dd490cf1df37b70a8625dc4ec8e712f740dd1e4aPierre Vandwalle    private static WifiPnoEventHandler sWifiPnoEventHandler;
2200dd490cf1df37b70a8625dc4ec8e712f740dd1e4aPierre Vandwalle
2201dd490cf1df37b70a8625dc4ec8e712f740dd1e4aPierre Vandwalle    private static int sPnoCmdId = 0;
2202dd490cf1df37b70a8625dc4ec8e712f740dd1e4aPierre Vandwalle
2203dd490cf1df37b70a8625dc4ec8e712f740dd1e4aPierre Vandwalle    private native static boolean setPnoListNative(int iface, int id, WifiPnoNetwork list[]);
2204dd490cf1df37b70a8625dc4ec8e712f740dd1e4aPierre Vandwalle
2205dd490cf1df37b70a8625dc4ec8e712f740dd1e4aPierre Vandwalle    synchronized public static boolean setPnoList(WifiPnoNetwork list[],
2206dd490cf1df37b70a8625dc4ec8e712f740dd1e4aPierre Vandwalle                                                  WifiPnoEventHandler eventHandler) {
2207dd490cf1df37b70a8625dc4ec8e712f740dd1e4aPierre Vandwalle        Log.e(TAG, "setPnoList cmd " + sPnoCmdId);
2208dd490cf1df37b70a8625dc4ec8e712f740dd1e4aPierre Vandwalle
2209dd490cf1df37b70a8625dc4ec8e712f740dd1e4aPierre Vandwalle        synchronized (mLock) {
221071d0be16c79791a005d60f9d5fa7d2d81e1f8c80xinhe            if (isHalStarted()) {
2211dd490cf1df37b70a8625dc4ec8e712f740dd1e4aPierre Vandwalle
2212f49a59bda006b13e0118d144e0a4a5f569b2251eVinit Deshpande                sPnoCmdId = getNewCmdIdLocked();
2213dd490cf1df37b70a8625dc4ec8e712f740dd1e4aPierre Vandwalle
2214f49a59bda006b13e0118d144e0a4a5f569b2251eVinit Deshpande                sWifiPnoEventHandler = eventHandler;
2215f49a59bda006b13e0118d144e0a4a5f569b2251eVinit Deshpande                if (setPnoListNative(sWlan0Index, sPnoCmdId, list)) {
2216f49a59bda006b13e0118d144e0a4a5f569b2251eVinit Deshpande                    return true;
2217f49a59bda006b13e0118d144e0a4a5f569b2251eVinit Deshpande                }
2218dd490cf1df37b70a8625dc4ec8e712f740dd1e4aPierre Vandwalle            }
2219dd490cf1df37b70a8625dc4ec8e712f740dd1e4aPierre Vandwalle
2220f49a59bda006b13e0118d144e0a4a5f569b2251eVinit Deshpande            sWifiPnoEventHandler = null;
2221f49a59bda006b13e0118d144e0a4a5f569b2251eVinit Deshpande            return false;
2222dd490cf1df37b70a8625dc4ec8e712f740dd1e4aPierre Vandwalle        }
2223dd490cf1df37b70a8625dc4ec8e712f740dd1e4aPierre Vandwalle    }
2224dd490cf1df37b70a8625dc4ec8e712f740dd1e4aPierre Vandwalle
2225dd490cf1df37b70a8625dc4ec8e712f740dd1e4aPierre Vandwalle    synchronized public static void onPnoNetworkFound(int id, ScanResult[] results) {
2226dd490cf1df37b70a8625dc4ec8e712f740dd1e4aPierre Vandwalle
2227dd490cf1df37b70a8625dc4ec8e712f740dd1e4aPierre Vandwalle        if (results == null) {
2228dd490cf1df37b70a8625dc4ec8e712f740dd1e4aPierre Vandwalle            Log.e(TAG, "onPnoNetworkFound null results");
2229dd490cf1df37b70a8625dc4ec8e712f740dd1e4aPierre Vandwalle            return;
2230dd490cf1df37b70a8625dc4ec8e712f740dd1e4aPierre Vandwalle
2231dd490cf1df37b70a8625dc4ec8e712f740dd1e4aPierre Vandwalle        }
2232dd490cf1df37b70a8625dc4ec8e712f740dd1e4aPierre Vandwalle        Log.d(TAG, "WifiNative.onPnoNetworkFound result " + results.length);
2233dd490cf1df37b70a8625dc4ec8e712f740dd1e4aPierre Vandwalle
2234dd490cf1df37b70a8625dc4ec8e712f740dd1e4aPierre Vandwalle        //Log.e(TAG, "onPnoNetworkFound length " + results.length);
2235dd490cf1df37b70a8625dc4ec8e712f740dd1e4aPierre Vandwalle        //return;
2236dd490cf1df37b70a8625dc4ec8e712f740dd1e4aPierre Vandwalle        for (int i=0; i<results.length; i++) {
2237dd490cf1df37b70a8625dc4ec8e712f740dd1e4aPierre Vandwalle            Log.e(TAG, "onPnoNetworkFound SSID " + results[i].SSID
2238dd490cf1df37b70a8625dc4ec8e712f740dd1e4aPierre Vandwalle                    + " " + results[i].level + " " + results[i].frequency);
2239dd490cf1df37b70a8625dc4ec8e712f740dd1e4aPierre Vandwalle
2240dd490cf1df37b70a8625dc4ec8e712f740dd1e4aPierre Vandwalle            populateScanResult(results[i], results[i].bytes, "onPnoNetworkFound ");
2241dd490cf1df37b70a8625dc4ec8e712f740dd1e4aPierre Vandwalle            results[i].wifiSsid = WifiSsid.createFromAsciiEncoded(results[i].SSID);
2242dd490cf1df37b70a8625dc4ec8e712f740dd1e4aPierre Vandwalle        }
2243dd490cf1df37b70a8625dc4ec8e712f740dd1e4aPierre Vandwalle        synchronized (mLock) {
2244dd490cf1df37b70a8625dc4ec8e712f740dd1e4aPierre Vandwalle            if (sPnoCmdId != 0 && sWifiPnoEventHandler != null) {
2245dd490cf1df37b70a8625dc4ec8e712f740dd1e4aPierre Vandwalle                sWifiPnoEventHandler.onPnoNetworkFound(results);
2246dd490cf1df37b70a8625dc4ec8e712f740dd1e4aPierre Vandwalle            } else {
2247dd490cf1df37b70a8625dc4ec8e712f740dd1e4aPierre Vandwalle                /* this can happen because of race conditions */
2248dd490cf1df37b70a8625dc4ec8e712f740dd1e4aPierre Vandwalle                Log.d(TAG, "Ignoring Pno Network found event");
2249dd490cf1df37b70a8625dc4ec8e712f740dd1e4aPierre Vandwalle            }
2250dd490cf1df37b70a8625dc4ec8e712f740dd1e4aPierre Vandwalle        }
2251d4c25fd76644d5490700ae69fada4669de5193a1Pierre Vandwalle    }
2252d4c25fd76644d5490700ae69fada4669de5193a1Pierre Vandwalle
2253d4c25fd76644d5490700ae69fada4669de5193a1Pierre Vandwalle    public class WifiLazyRoamParams {
2254d4c25fd76644d5490700ae69fada4669de5193a1Pierre Vandwalle        int A_band_boost_threshold;
2255d4c25fd76644d5490700ae69fada4669de5193a1Pierre Vandwalle        int A_band_penalty_threshold;
2256d4c25fd76644d5490700ae69fada4669de5193a1Pierre Vandwalle        int A_band_boost_factor;
2257d4c25fd76644d5490700ae69fada4669de5193a1Pierre Vandwalle        int A_band_penalty_factor;
2258d4c25fd76644d5490700ae69fada4669de5193a1Pierre Vandwalle        int A_band_max_boost;
2259d4c25fd76644d5490700ae69fada4669de5193a1Pierre Vandwalle        int lazy_roam_hysteresis;
2260d4c25fd76644d5490700ae69fada4669de5193a1Pierre Vandwalle        int alert_roam_rssi_trigger;
2261dd490cf1df37b70a8625dc4ec8e712f740dd1e4aPierre Vandwalle
2262d4c25fd76644d5490700ae69fada4669de5193a1Pierre Vandwalle        WifiLazyRoamParams() {
2263d4c25fd76644d5490700ae69fada4669de5193a1Pierre Vandwalle        }
2264d4c25fd76644d5490700ae69fada4669de5193a1Pierre Vandwalle
2265d4c25fd76644d5490700ae69fada4669de5193a1Pierre Vandwalle        @Override
2266d4c25fd76644d5490700ae69fada4669de5193a1Pierre Vandwalle        public String toString() {
2267d4c25fd76644d5490700ae69fada4669de5193a1Pierre Vandwalle            StringBuilder sbuf = new StringBuilder();
2268d4c25fd76644d5490700ae69fada4669de5193a1Pierre Vandwalle            sbuf.append(" A_band_boost_threshold=").append(this.A_band_boost_threshold);
2269d4c25fd76644d5490700ae69fada4669de5193a1Pierre Vandwalle            sbuf.append(" A_band_penalty_threshold=").append(this.A_band_penalty_threshold);
2270d4c25fd76644d5490700ae69fada4669de5193a1Pierre Vandwalle            sbuf.append(" A_band_boost_factor=").append(this.A_band_boost_factor);
2271d4c25fd76644d5490700ae69fada4669de5193a1Pierre Vandwalle            sbuf.append(" A_band_penalty_factor=").append(this.A_band_penalty_factor);
2272d4c25fd76644d5490700ae69fada4669de5193a1Pierre Vandwalle            sbuf.append(" A_band_max_boost=").append(this.A_band_max_boost);
2273d4c25fd76644d5490700ae69fada4669de5193a1Pierre Vandwalle            sbuf.append(" lazy_roam_hysteresis=").append(this.lazy_roam_hysteresis);
2274d4c25fd76644d5490700ae69fada4669de5193a1Pierre Vandwalle            sbuf.append(" alert_roam_rssi_trigger=").append(this.alert_roam_rssi_trigger);
2275d4c25fd76644d5490700ae69fada4669de5193a1Pierre Vandwalle            return sbuf.toString();
2276d4c25fd76644d5490700ae69fada4669de5193a1Pierre Vandwalle        }
2277d4c25fd76644d5490700ae69fada4669de5193a1Pierre Vandwalle    }
2278d4c25fd76644d5490700ae69fada4669de5193a1Pierre Vandwalle
22799ccffbb6c3fe8eaa7d70cc3076d0dc3c6a9073e7Pierre Vandwalle    private native static boolean setLazyRoamNative(int iface, int id,
2280d4c25fd76644d5490700ae69fada4669de5193a1Pierre Vandwalle                                              boolean enabled, WifiLazyRoamParams param);
2281d4c25fd76644d5490700ae69fada4669de5193a1Pierre Vandwalle
2282d4c25fd76644d5490700ae69fada4669de5193a1Pierre Vandwalle    synchronized public static boolean setLazyRoam(boolean enabled, WifiLazyRoamParams params) {
2283d4c25fd76644d5490700ae69fada4669de5193a1Pierre Vandwalle        synchronized (mLock) {
228471d0be16c79791a005d60f9d5fa7d2d81e1f8c80xinhe            if (isHalStarted()) {
2285d4c25fd76644d5490700ae69fada4669de5193a1Pierre Vandwalle                sPnoCmdId = getNewCmdIdLocked();
22869ccffbb6c3fe8eaa7d70cc3076d0dc3c6a9073e7Pierre Vandwalle                return setLazyRoamNative(sWlan0Index, sPnoCmdId, enabled, params);
2287d4c25fd76644d5490700ae69fada4669de5193a1Pierre Vandwalle            } else {
2288d4c25fd76644d5490700ae69fada4669de5193a1Pierre Vandwalle                return false;
2289d4c25fd76644d5490700ae69fada4669de5193a1Pierre Vandwalle            }
2290d4c25fd76644d5490700ae69fada4669de5193a1Pierre Vandwalle        }
2291d4c25fd76644d5490700ae69fada4669de5193a1Pierre Vandwalle    }
22929ccffbb6c3fe8eaa7d70cc3076d0dc3c6a9073e7Pierre Vandwalle
22939ccffbb6c3fe8eaa7d70cc3076d0dc3c6a9073e7Pierre Vandwalle    private native static boolean setBssidBlacklistNative(int iface, int id,
22949ccffbb6c3fe8eaa7d70cc3076d0dc3c6a9073e7Pierre Vandwalle                                              String list[]);
22959ccffbb6c3fe8eaa7d70cc3076d0dc3c6a9073e7Pierre Vandwalle
22969ccffbb6c3fe8eaa7d70cc3076d0dc3c6a9073e7Pierre Vandwalle    synchronized public static boolean setBssidBlacklist(String list[]) {
22979ccffbb6c3fe8eaa7d70cc3076d0dc3c6a9073e7Pierre Vandwalle        int size = 0;
22989ccffbb6c3fe8eaa7d70cc3076d0dc3c6a9073e7Pierre Vandwalle        if (list != null) {
22999ccffbb6c3fe8eaa7d70cc3076d0dc3c6a9073e7Pierre Vandwalle            size = list.length;
23009ccffbb6c3fe8eaa7d70cc3076d0dc3c6a9073e7Pierre Vandwalle        }
23019ccffbb6c3fe8eaa7d70cc3076d0dc3c6a9073e7Pierre Vandwalle        Log.e(TAG, "setBssidBlacklist cmd " + sPnoCmdId + " size " + size);
23029ccffbb6c3fe8eaa7d70cc3076d0dc3c6a9073e7Pierre Vandwalle
23039ccffbb6c3fe8eaa7d70cc3076d0dc3c6a9073e7Pierre Vandwalle        synchronized (mLock) {
230471d0be16c79791a005d60f9d5fa7d2d81e1f8c80xinhe            if (isHalStarted()) {
2305f49a59bda006b13e0118d144e0a4a5f569b2251eVinit Deshpande                sPnoCmdId = getNewCmdIdLocked();
2306f49a59bda006b13e0118d144e0a4a5f569b2251eVinit Deshpande                return setBssidBlacklistNative(sWlan0Index, sPnoCmdId, list);
2307f49a59bda006b13e0118d144e0a4a5f569b2251eVinit Deshpande            } else {
23089ccffbb6c3fe8eaa7d70cc3076d0dc3c6a9073e7Pierre Vandwalle                return false;
23099ccffbb6c3fe8eaa7d70cc3076d0dc3c6a9073e7Pierre Vandwalle            }
23109ccffbb6c3fe8eaa7d70cc3076d0dc3c6a9073e7Pierre Vandwalle        }
23119ccffbb6c3fe8eaa7d70cc3076d0dc3c6a9073e7Pierre Vandwalle    }
23129ccffbb6c3fe8eaa7d70cc3076d0dc3c6a9073e7Pierre Vandwalle
23135caa43b34e0445ff08bff5931b84ffbc850b2a66Pierre Vandwalle    private native static boolean setSsidWhitelistNative(int iface, int id, String list[]);
23145caa43b34e0445ff08bff5931b84ffbc850b2a66Pierre Vandwalle
23155caa43b34e0445ff08bff5931b84ffbc850b2a66Pierre Vandwalle    synchronized public static boolean setSsidWhitelist(String list[]) {
23165caa43b34e0445ff08bff5931b84ffbc850b2a66Pierre Vandwalle        int size = 0;
23175caa43b34e0445ff08bff5931b84ffbc850b2a66Pierre Vandwalle        if (list != null) {
23185caa43b34e0445ff08bff5931b84ffbc850b2a66Pierre Vandwalle            size = list.length;
23195caa43b34e0445ff08bff5931b84ffbc850b2a66Pierre Vandwalle        }
23205caa43b34e0445ff08bff5931b84ffbc850b2a66Pierre Vandwalle        Log.e(TAG, "setSsidWhitelist cmd " + sPnoCmdId + " size " + size);
23215caa43b34e0445ff08bff5931b84ffbc850b2a66Pierre Vandwalle
23225caa43b34e0445ff08bff5931b84ffbc850b2a66Pierre Vandwalle        synchronized (mLock) {
232371d0be16c79791a005d60f9d5fa7d2d81e1f8c80xinhe            if (isHalStarted()) {
2324f49a59bda006b13e0118d144e0a4a5f569b2251eVinit Deshpande                sPnoCmdId = getNewCmdIdLocked();
23255caa43b34e0445ff08bff5931b84ffbc850b2a66Pierre Vandwalle
2326f49a59bda006b13e0118d144e0a4a5f569b2251eVinit Deshpande                return setSsidWhitelistNative(sWlan0Index, sPnoCmdId, list);
2327f49a59bda006b13e0118d144e0a4a5f569b2251eVinit Deshpande            } else {
23285caa43b34e0445ff08bff5931b84ffbc850b2a66Pierre Vandwalle                return false;
23295caa43b34e0445ff08bff5931b84ffbc850b2a66Pierre Vandwalle            }
23305caa43b34e0445ff08bff5931b84ffbc850b2a66Pierre Vandwalle        }
23315caa43b34e0445ff08bff5931b84ffbc850b2a66Pierre Vandwalle    }
2332155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande}
2333