WifiNative.java revision b66b29a00da970ee75052e24f592c8d6c16bd0ed
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;
32f22d23092ab37286a5ef9d257d5bb32c421d2669vandwalleimport android.os.SystemClock;
33155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpandeimport android.text.TextUtils;
3403ba4a5b108ab2bbdb5d0e7be46ac45c00a6d9ffxinheimport android.util.Base64;
35155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpandeimport android.util.LocalLog;
36155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpandeimport android.util.Log;
37155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
3898dca3397c0049d3cfb9c97e2b021c5ee3a2e1bdxinheimport java.io.ByteArrayOutputStream;
3998dca3397c0049d3cfb9c97e2b021c5ee3a2e1bdxinheimport java.io.IOException;
40155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpandeimport java.util.ArrayList;
41155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpandeimport java.util.List;
42155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpandeimport java.util.Locale;
4398dca3397c0049d3cfb9c97e2b021c5ee3a2e1bdxinheimport java.util.zip.Deflater;
44155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
45155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande/**
46155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande * Native calls for bring up/shut down of the supplicant daemon and for
47155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande * sending requests to the supplicant daemon
48155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande *
49155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande * waitForEvent() is called on the monitor thread for events. All other methods
50155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande * must be serialized from the framework.
51155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande *
52155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande * {@hide}
53155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande */
54155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpandepublic class WifiNative {
55155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
56b66b29a00da970ee75052e24f592c8d6c16bd0edxinhe    private static boolean DBG = false;
57155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    private final String mTAG;
58155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    private static final int DEFAULT_GROUP_OWNER_INTENT     = 6;
59155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
60155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    static final int BLUETOOTH_COEXISTENCE_MODE_ENABLED     = 0;
61155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    static final int BLUETOOTH_COEXISTENCE_MODE_DISABLED    = 1;
62155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    static final int BLUETOOTH_COEXISTENCE_MODE_SENSE       = 2;
63155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
64155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    static final int SCAN_WITHOUT_CONNECTION_SETUP          = 1;
65155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    static final int SCAN_WITH_CONNECTION_SETUP             = 2;
66155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
67155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    // Hold this lock before calling supplicant - it is required to
68155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    // mutually exclude access from Wifi and P2p state machines
69155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    static final Object mLock = new Object();
70155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
71155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    public final String mInterfaceName;
72155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    public final String mInterfacePrefix;
73155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
74155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    private boolean mSuspendOptEnabled = false;
75155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
76243931f3474f6235cfcf5c1a55fa2f192ee264aexinhe    private static final int EID_HT_OPERATION = 61;
77243931f3474f6235cfcf5c1a55fa2f192ee264aexinhe    private static final int EID_VHT_OPERATION = 192;
78243931f3474f6235cfcf5c1a55fa2f192ee264aexinhe    private static final int EID_EXTENDED_CAPS = 127;
79243931f3474f6235cfcf5c1a55fa2f192ee264aexinhe    private static final int RTT_RESP_ENABLE_BIT = 70;
80155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    /* Register native functions */
81155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
82155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    static {
83155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        /* Native functions are defined in libwifi-service.so */
84155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        System.loadLibrary("wifi-service");
85155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        registerNatives();
86155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    }
87155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
88155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    private static native int registerNatives();
89155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
90155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    public native static boolean loadDriver();
91155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
92155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    public native static boolean isDriverLoaded();
93155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
94155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    public native static boolean unloadDriver();
95155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
96155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    public native static boolean startSupplicant(boolean p2pSupported);
97155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
98155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    /* Sends a kill signal to supplicant. To be used when we have lost connection
99155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande       or when the supplicant is hung */
100155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    public native static boolean killSupplicant(boolean p2pSupported);
101155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
102155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    private native boolean connectToSupplicantNative();
103155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
104155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    private native void closeSupplicantConnectionNative();
105155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
106155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    /**
107155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande     * Wait for the supplicant to send an event, returning the event string.
108155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande     * @return the event string sent by the supplicant.
109155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande     */
110155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    private native String waitForEventNative();
111155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
112155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    private native boolean doBooleanCommandNative(String command);
113155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
114155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    private native int doIntCommandNative(String command);
115155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
116155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    private native String doStringCommandNative(String command);
117155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
118155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    public WifiNative(String interfaceName) {
119155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        mInterfaceName = interfaceName;
120155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        mTAG = "WifiNative-" + interfaceName;
121155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        if (!interfaceName.equals("p2p0")) {
122155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande            mInterfacePrefix = "IFNAME=" + interfaceName + " ";
123155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        } else {
124155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande            // commands for p2p0 interface don't need prefix
125155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande            mInterfacePrefix = "";
126155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        }
127155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    }
128155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
129ed9938883ae2dade81c8be6cd6ceaef3febd5239vandwalle    void enableVerboseLogging(int verbose) {
130ed9938883ae2dade81c8be6cd6ceaef3febd5239vandwalle        if (verbose > 0) {
131ed9938883ae2dade81c8be6cd6ceaef3febd5239vandwalle            DBG = true;
132ed9938883ae2dade81c8be6cd6ceaef3febd5239vandwalle        } else {
133ed9938883ae2dade81c8be6cd6ceaef3febd5239vandwalle            DBG = false;
134ed9938883ae2dade81c8be6cd6ceaef3febd5239vandwalle        }
135ed9938883ae2dade81c8be6cd6ceaef3febd5239vandwalle    }
136ed9938883ae2dade81c8be6cd6ceaef3febd5239vandwalle
137155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    private static final LocalLog mLocalLog = new LocalLog(1024);
138155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
139155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    // hold mLock before accessing mCmdIdLock
140b0b1d59786de7ff1fc98e31a16a7d710458bf549Vinit Deshpande    private static int sCmdId;
141155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
142155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    public LocalLog getLocalLog() {
143155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        return mLocalLog;
144155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    }
145155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
146b0b1d59786de7ff1fc98e31a16a7d710458bf549Vinit Deshpande    private static int getNewCmdIdLocked() {
147b0b1d59786de7ff1fc98e31a16a7d710458bf549Vinit Deshpande        return sCmdId++;
148155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    }
149155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
150155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    private void localLog(String s) {
151155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        if (mLocalLog != null)
152155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande            mLocalLog.log(mInterfaceName + ": " + s);
153155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    }
154155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
155155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    public boolean connectToSupplicant() {
156155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        // No synchronization necessary .. it is implemented in WifiMonitor
157155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        localLog(mInterfacePrefix + "connectToSupplicant");
158155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        return connectToSupplicantNative();
159155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    }
160155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
161155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    public void closeSupplicantConnection() {
162155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        localLog(mInterfacePrefix + "closeSupplicantConnection");
163155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        closeSupplicantConnectionNative();
164155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    }
165155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
166155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    public String waitForEvent() {
167155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        // No synchronization necessary .. it is implemented in WifiMonitor
168155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        return waitForEventNative();
169155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    }
170155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
171155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    private boolean doBooleanCommand(String command) {
172155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        if (DBG) Log.d(mTAG, "doBoolean: " + command);
173155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        synchronized (mLock) {
174155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande            int cmdId = getNewCmdIdLocked();
1757b581f46f6c9bc6edf0edd287d47106712fb2144vandwalle            String toLog = Integer.toString(cmdId) + ":" + mInterfacePrefix + command;
176155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande            boolean result = doBooleanCommandNative(mInterfacePrefix + command);
1777b581f46f6c9bc6edf0edd287d47106712fb2144vandwalle            localLog(toLog + " -> " + result);
1780888ce6f90bdaeee799dd8361ea4781e23a33b87vandwalle            if (DBG) Log.d(mTAG, command + ": returned " + result);
179155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande            return result;
180155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        }
181155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    }
182155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
183155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    private int doIntCommand(String command) {
184155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        if (DBG) Log.d(mTAG, "doInt: " + command);
185155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        synchronized (mLock) {
186155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande            int cmdId = getNewCmdIdLocked();
1877b581f46f6c9bc6edf0edd287d47106712fb2144vandwalle            String toLog = Integer.toString(cmdId) + ":" + mInterfacePrefix + command;
188155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande            int result = doIntCommandNative(mInterfacePrefix + command);
1897b581f46f6c9bc6edf0edd287d47106712fb2144vandwalle            localLog(toLog + " -> " + result);
190155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande            if (DBG) Log.d(mTAG, "   returned " + result);
191155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande            return result;
192155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        }
193155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    }
194155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
195155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    private String doStringCommand(String command) {
1960888ce6f90bdaeee799dd8361ea4781e23a33b87vandwalle        if (DBG) {
1970888ce6f90bdaeee799dd8361ea4781e23a33b87vandwalle            //GET_NETWORK commands flood the logs
1980888ce6f90bdaeee799dd8361ea4781e23a33b87vandwalle            if (!command.startsWith("GET_NETWORK")) {
1990888ce6f90bdaeee799dd8361ea4781e23a33b87vandwalle                Log.d(mTAG, "doString: [" + command + "]");
2000888ce6f90bdaeee799dd8361ea4781e23a33b87vandwalle            }
2010888ce6f90bdaeee799dd8361ea4781e23a33b87vandwalle        }
202155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        synchronized (mLock) {
203155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande            int cmdId = getNewCmdIdLocked();
2047b581f46f6c9bc6edf0edd287d47106712fb2144vandwalle            String toLog = Integer.toString(cmdId) + ":" + mInterfacePrefix + command;
205155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande            String result = doStringCommandNative(mInterfacePrefix + command);
20640ff222cec1bd05879edb53abc75c6deead734cavandwalle            if (result == null) {
20740ff222cec1bd05879edb53abc75c6deead734cavandwalle                if (DBG) Log.d(mTAG, "doStringCommandNative no result");
20840ff222cec1bd05879edb53abc75c6deead734cavandwalle            } else {
2097b581f46f6c9bc6edf0edd287d47106712fb2144vandwalle                if (!command.startsWith("STATUS-")) {
2107b581f46f6c9bc6edf0edd287d47106712fb2144vandwalle                    localLog(toLog + " -> " + result);
2117b581f46f6c9bc6edf0edd287d47106712fb2144vandwalle                }
21240ff222cec1bd05879edb53abc75c6deead734cavandwalle                if (DBG) Log.d(mTAG, "   returned " + result.replace("\n", " "));
21340ff222cec1bd05879edb53abc75c6deead734cavandwalle            }
214155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande            return result;
215155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        }
216155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    }
217155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
218155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    private String doStringCommandWithoutLogging(String command) {
2190888ce6f90bdaeee799dd8361ea4781e23a33b87vandwalle        if (DBG) {
2200888ce6f90bdaeee799dd8361ea4781e23a33b87vandwalle            //GET_NETWORK commands flood the logs
2210888ce6f90bdaeee799dd8361ea4781e23a33b87vandwalle            if (!command.startsWith("GET_NETWORK")) {
2220888ce6f90bdaeee799dd8361ea4781e23a33b87vandwalle                Log.d(mTAG, "doString: [" + command + "]");
2230888ce6f90bdaeee799dd8361ea4781e23a33b87vandwalle            }
22427355a942653264388e909a4276196ee63e57811vandwalle        }
22527355a942653264388e909a4276196ee63e57811vandwalle        synchronized (mLock) {
226155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande            return doStringCommandNative(mInterfacePrefix + command);
227155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        }
228155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    }
229155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
230155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    public boolean ping() {
231155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        String pong = doStringCommand("PING");
232155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        return (pong != null && pong.equals("PONG"));
233155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    }
234155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
235ad7319939c424d42fa6a3791c47f613db8ef3cd8vandwalle    public void setSupplicantLogLevel(String level) {
236ad7319939c424d42fa6a3791c47f613db8ef3cd8vandwalle        doStringCommand("LOG_LEVEL " + level);
237ad7319939c424d42fa6a3791c47f613db8ef3cd8vandwalle    }
238ad7319939c424d42fa6a3791c47f613db8ef3cd8vandwalle
239a0009d14075b5345b8f916c3fb3f2260c938cb9dYuhao Zheng    public String getFreqCapability() {
240a0009d14075b5345b8f916c3fb3f2260c938cb9dYuhao Zheng        return doStringCommand("GET_CAPABILITY freq");
241a0009d14075b5345b8f916c3fb3f2260c938cb9dYuhao Zheng    }
242a0009d14075b5345b8f916c3fb3f2260c938cb9dYuhao Zheng
243a0009d14075b5345b8f916c3fb3f2260c938cb9dYuhao Zheng    public boolean scan(int type, String freqList) {
244155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        if (type == SCAN_WITHOUT_CONNECTION_SETUP) {
245a0009d14075b5345b8f916c3fb3f2260c938cb9dYuhao Zheng            if (freqList == null) return doBooleanCommand("SCAN TYPE=ONLY");
246a0009d14075b5345b8f916c3fb3f2260c938cb9dYuhao Zheng            else return doBooleanCommand("SCAN TYPE=ONLY freq=" + freqList);
247155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        } else if (type == SCAN_WITH_CONNECTION_SETUP) {
248a0009d14075b5345b8f916c3fb3f2260c938cb9dYuhao Zheng            if (freqList == null) return doBooleanCommand("SCAN");
249a0009d14075b5345b8f916c3fb3f2260c938cb9dYuhao Zheng            else return doBooleanCommand("SCAN freq=" + freqList);
250155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        } else {
251155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande            throw new IllegalArgumentException("Invalid scan type");
252155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        }
253155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    }
254155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
255155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    /* Does a graceful shutdown of supplicant. Is a common stop function for both p2p and sta.
256155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande     *
257155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande     * Note that underneath we use a harsh-sounding "terminate" supplicant command
258155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande     * for a graceful stop and a mild-sounding "stop" interface
259155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande     * to kill the process
260155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande     */
261155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    public boolean stopSupplicant() {
262155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        return doBooleanCommand("TERMINATE");
263155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    }
264155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
265155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    public String listNetworks() {
266155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        return doStringCommand("LIST_NETWORKS");
267155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    }
268155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
269e3939cb40d9ba3842be105a6e85172dc06e14758Vinit Deshpande    public String listNetworks(int last_id) {
270e3939cb40d9ba3842be105a6e85172dc06e14758Vinit Deshpande        return doStringCommand("LIST_NETWORKS LAST_ID=" + last_id);
271e3939cb40d9ba3842be105a6e85172dc06e14758Vinit Deshpande    }
272e3939cb40d9ba3842be105a6e85172dc06e14758Vinit Deshpande
273155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    public int addNetwork() {
274155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        return doIntCommand("ADD_NETWORK");
275155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    }
276155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
277155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    public boolean setNetworkVariable(int netId, String name, String value) {
278155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        if (TextUtils.isEmpty(name) || TextUtils.isEmpty(value)) return false;
279155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        return doBooleanCommand("SET_NETWORK " + netId + " " + name + " " + value);
280155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    }
281155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
282155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    public String getNetworkVariable(int netId, String name) {
283155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        if (TextUtils.isEmpty(name)) return null;
284155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
285155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        // GET_NETWORK will likely flood the logs ...
286155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        return doStringCommandWithoutLogging("GET_NETWORK " + netId + " " + name);
287155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    }
288155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
289155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    public boolean removeNetwork(int netId) {
290155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        return doBooleanCommand("REMOVE_NETWORK " + netId);
291155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    }
292155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
293f22d23092ab37286a5ef9d257d5bb32c421d2669vandwalle
294f22d23092ab37286a5ef9d257d5bb32c421d2669vandwalle    private void logDbg(String debug) {
295f22d23092ab37286a5ef9d257d5bb32c421d2669vandwalle        long now = SystemClock.elapsedRealtimeNanos();
296f22d23092ab37286a5ef9d257d5bb32c421d2669vandwalle        String ts = String.format("[%,d us] ", now/1000);
297ecd2b88214b5d214fd1f63a9560caff9058912ddvandwalle        Log.e("WifiNative: ", ts+debug+ " stack:"
298ecd2b88214b5d214fd1f63a9560caff9058912ddvandwalle                + Thread.currentThread().getStackTrace()[2].getMethodName() +" - "
299ecd2b88214b5d214fd1f63a9560caff9058912ddvandwalle                + Thread.currentThread().getStackTrace()[3].getMethodName() +" - "
300ecd2b88214b5d214fd1f63a9560caff9058912ddvandwalle                + Thread.currentThread().getStackTrace()[4].getMethodName() +" - "
301ecd2b88214b5d214fd1f63a9560caff9058912ddvandwalle                + Thread.currentThread().getStackTrace()[5].getMethodName()+" - "
302ecd2b88214b5d214fd1f63a9560caff9058912ddvandwalle                + Thread.currentThread().getStackTrace()[6].getMethodName());
303f22d23092ab37286a5ef9d257d5bb32c421d2669vandwalle
304f22d23092ab37286a5ef9d257d5bb32c421d2669vandwalle    }
305155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    public boolean enableNetwork(int netId, boolean disableOthers) {
306ecd2b88214b5d214fd1f63a9560caff9058912ddvandwalle        if (DBG) logDbg("enableNetwork nid=" + Integer.toString(netId)
307ecd2b88214b5d214fd1f63a9560caff9058912ddvandwalle                + " disableOthers=" + disableOthers);
308155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        if (disableOthers) {
309155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande            return doBooleanCommand("SELECT_NETWORK " + netId);
310155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        } else {
311155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande            return doBooleanCommand("ENABLE_NETWORK " + netId);
312155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        }
313155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    }
314155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
315155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    public boolean disableNetwork(int netId) {
316f22d23092ab37286a5ef9d257d5bb32c421d2669vandwalle        if (DBG) logDbg("disableNetwork nid=" + Integer.toString(netId));
317155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        return doBooleanCommand("DISABLE_NETWORK " + netId);
318155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    }
319155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
3200047ccf563baa288777e06c6fe95d3681fcf5ccdVinit Deshpande    public boolean selectNetwork(int netId) {
3210047ccf563baa288777e06c6fe95d3681fcf5ccdVinit Deshpande        if (DBG) logDbg("selectNetwork nid=" + Integer.toString(netId));
3220047ccf563baa288777e06c6fe95d3681fcf5ccdVinit Deshpande        return doBooleanCommand("SELECT_NETWORK " + netId);
3230047ccf563baa288777e06c6fe95d3681fcf5ccdVinit Deshpande    }
3240047ccf563baa288777e06c6fe95d3681fcf5ccdVinit Deshpande
325155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    public boolean reconnect() {
326f22d23092ab37286a5ef9d257d5bb32c421d2669vandwalle        if (DBG) logDbg("RECONNECT ");
327155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        return doBooleanCommand("RECONNECT");
328155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    }
329155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
330155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    public boolean reassociate() {
331f22d23092ab37286a5ef9d257d5bb32c421d2669vandwalle        if (DBG) logDbg("REASSOCIATE ");
332155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        return doBooleanCommand("REASSOCIATE");
333155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    }
334155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
335155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    public boolean disconnect() {
33621bc54cb37a0085b1c909cb4d55ebb12a2facefbvandwalle        if (DBG) logDbg("DISCONNECT ");
337155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        return doBooleanCommand("DISCONNECT");
338155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    }
339155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
340155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    public String status() {
34199d385e3b4d34841d6efcfd7cc9bf1d5ae25de14vandwalle        return status(false);
342155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    }
343155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
34499d385e3b4d34841d6efcfd7cc9bf1d5ae25de14vandwalle    public String status(boolean noEvents) {
34599d385e3b4d34841d6efcfd7cc9bf1d5ae25de14vandwalle        if (noEvents) {
34699d385e3b4d34841d6efcfd7cc9bf1d5ae25de14vandwalle            return doStringCommand("STATUS-NO_EVENTS");
34799d385e3b4d34841d6efcfd7cc9bf1d5ae25de14vandwalle        } else {
34899d385e3b4d34841d6efcfd7cc9bf1d5ae25de14vandwalle            return doStringCommand("STATUS");
34999d385e3b4d34841d6efcfd7cc9bf1d5ae25de14vandwalle        }
35099d385e3b4d34841d6efcfd7cc9bf1d5ae25de14vandwalle    }
35199d385e3b4d34841d6efcfd7cc9bf1d5ae25de14vandwalle
352155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    public String getMacAddress() {
353155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        //Macaddr = XX.XX.XX.XX.XX.XX
354155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        String ret = doStringCommand("DRIVER MACADDR");
355155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        if (!TextUtils.isEmpty(ret)) {
356155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande            String[] tokens = ret.split(" = ");
357155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande            if (tokens.length == 2) return tokens[1];
358155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        }
359155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        return null;
360155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    }
361155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
362a0d34d3626bd9631039a485d2d428b1b7de390e2Pierre Vandwalle
363a0d34d3626bd9631039a485d2d428b1b7de390e2Pierre Vandwalle
364155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    /**
365155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande     * Format of results:
366155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande     * =================
367155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande     * id=1
368155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande     * bssid=68:7f:74:d7:1b:6e
369155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande     * freq=2412
370155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande     * level=-43
371155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande     * tsf=1344621975160944
372155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande     * age=2623
373155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande     * flags=[WPA2-PSK-CCMP][WPS][ESS]
374155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande     * ssid=zubyb
375155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande     * ====
376155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande     *
377155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande     * RANGE=ALL gets all scan results
378155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande     * RANGE=ID- gets results from ID
379155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande     * MASK=<N> see wpa_supplicant/src/common/wpa_ctrl.h for details
38077f2b82a2e80af8da52c22d69a76def6d4209757Jan Nordqvist     * 0                         0                        1                       0     2
38177f2b82a2e80af8da52c22d69a76def6d4209757Jan Nordqvist     *                           WPA_BSS_MASK_MESH_SCAN | WPA_BSS_MASK_DELIM    | WPA_BSS_MASK_WIFI_DISPLAY
38277f2b82a2e80af8da52c22d69a76def6d4209757Jan Nordqvist     * 0                         0                        0                       1     1   -> 9
38377f2b82a2e80af8da52c22d69a76def6d4209757Jan Nordqvist     * WPA_BSS_MASK_INTERNETW  | WPA_BSS_MASK_P2P_SCAN  | WPA_BSS_MASK_WPS_SCAN | WPA_BSS_MASK_SSID
38477f2b82a2e80af8da52c22d69a76def6d4209757Jan Nordqvist     * 1                         0                        0                       1     9   -> d
38577f2b82a2e80af8da52c22d69a76def6d4209757Jan Nordqvist     * WPA_BSS_MASK_FLAGS      | WPA_BSS_MASK_IE        | WPA_BSS_MASK_AGE      | WPA_BSS_MASK_TSF
38677f2b82a2e80af8da52c22d69a76def6d4209757Jan Nordqvist     * 1                         0                        0                       0     8
38777f2b82a2e80af8da52c22d69a76def6d4209757Jan Nordqvist     * WPA_BSS_MASK_LEVEL      | WPA_BSS_MASK_NOISE     | WPA_BSS_MASK_QUAL     | WPA_BSS_MASK_CAPABILITIES
38877f2b82a2e80af8da52c22d69a76def6d4209757Jan Nordqvist     * 0                         1                        1                       1     7
38977f2b82a2e80af8da52c22d69a76def6d4209757Jan Nordqvist     * WPA_BSS_MASK_BEACON_INT | WPA_BSS_MASK_FREQ      | WPA_BSS_MASK_BSSID    | WPA_BSS_MASK_ID
39077f2b82a2e80af8da52c22d69a76def6d4209757Jan Nordqvist     *
39177f2b82a2e80af8da52c22d69a76def6d4209757Jan Nordqvist     * WPA_BSS_MASK_INTERNETW adds ANQP info (ctrl_iface:4151-4176)
39277f2b82a2e80af8da52c22d69a76def6d4209757Jan Nordqvist     *
39377f2b82a2e80af8da52c22d69a76def6d4209757Jan Nordqvist     * ctrl_iface.c:wpa_supplicant_ctrl_iface_process:7884
39477f2b82a2e80af8da52c22d69a76def6d4209757Jan Nordqvist     *  wpa_supplicant_ctrl_iface_bss:4315
39577f2b82a2e80af8da52c22d69a76def6d4209757Jan Nordqvist     *  print_bss_info
396155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande     */
397155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    public String scanResults(int sid) {
39877f2b82a2e80af8da52c22d69a76def6d4209757Jan Nordqvist        return doStringCommandWithoutLogging("BSS RANGE=" + sid + "- MASK=0x29d87");
39977f2b82a2e80af8da52c22d69a76def6d4209757Jan Nordqvist    }
40077f2b82a2e80af8da52c22d69a76def6d4209757Jan Nordqvist
40177f2b82a2e80af8da52c22d69a76def6d4209757Jan Nordqvist    public String doCustomCommand(String command) {
40277f2b82a2e80af8da52c22d69a76def6d4209757Jan Nordqvist        return doStringCommand(command);
403155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    }
404155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
405155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    /**
406446db2d5457456743e4476029e14d7c3bb9f5bccYuhao Zheng     * Format of result:
407446db2d5457456743e4476029e14d7c3bb9f5bccYuhao Zheng     * id=1016
408446db2d5457456743e4476029e14d7c3bb9f5bccYuhao Zheng     * bssid=00:03:7f:40:84:10
409446db2d5457456743e4476029e14d7c3bb9f5bccYuhao Zheng     * freq=2462
410446db2d5457456743e4476029e14d7c3bb9f5bccYuhao Zheng     * beacon_int=200
411446db2d5457456743e4476029e14d7c3bb9f5bccYuhao Zheng     * capabilities=0x0431
412446db2d5457456743e4476029e14d7c3bb9f5bccYuhao Zheng     * qual=0
413446db2d5457456743e4476029e14d7c3bb9f5bccYuhao Zheng     * noise=0
414446db2d5457456743e4476029e14d7c3bb9f5bccYuhao Zheng     * level=-46
415446db2d5457456743e4476029e14d7c3bb9f5bccYuhao Zheng     * tsf=0000002669008476
416446db2d5457456743e4476029e14d7c3bb9f5bccYuhao Zheng     * age=5
417446db2d5457456743e4476029e14d7c3bb9f5bccYuhao Zheng     * ie=00105143412d485332302d52322d54455354010882848b960c12182403010b0706555...
418446db2d5457456743e4476029e14d7c3bb9f5bccYuhao Zheng     * flags=[WPA2-EAP-CCMP][ESS][P2P][HS20]
419446db2d5457456743e4476029e14d7c3bb9f5bccYuhao Zheng     * ssid=QCA-HS20-R2-TEST
420446db2d5457456743e4476029e14d7c3bb9f5bccYuhao Zheng     * p2p_device_name=
42156d0178183460eed9afbd85e5c0d215e27d5f5bcvandwalle     * p2p_config_methods=0x0SET_NE
422446db2d5457456743e4476029e14d7c3bb9f5bccYuhao Zheng     * anqp_venue_name=02083d656e6757692d466920416c6c69616e63650a3239383920436f...
423446db2d5457456743e4476029e14d7c3bb9f5bccYuhao Zheng     * anqp_network_auth_type=010000
424446db2d5457456743e4476029e14d7c3bb9f5bccYuhao Zheng     * anqp_roaming_consortium=03506f9a05001bc504bd
425446db2d5457456743e4476029e14d7c3bb9f5bccYuhao Zheng     * anqp_ip_addr_type_availability=0c
426446db2d5457456743e4476029e14d7c3bb9f5bccYuhao Zheng     * anqp_nai_realm=0200300000246d61696c2e6578616d706c652e636f6d3b636973636f2...
427446db2d5457456743e4476029e14d7c3bb9f5bccYuhao Zheng     * anqp_3gpp=000600040132f465
428446db2d5457456743e4476029e14d7c3bb9f5bccYuhao Zheng     * anqp_domain_name=0b65786d61706c652e636f6d
429446db2d5457456743e4476029e14d7c3bb9f5bccYuhao Zheng     * hs20_operator_friendly_name=11656e6757692d466920416c6c69616e63650e636869...
430446db2d5457456743e4476029e14d7c3bb9f5bccYuhao Zheng     * hs20_wan_metrics=01c40900008001000000000a00
431446db2d5457456743e4476029e14d7c3bb9f5bccYuhao Zheng     * hs20_connection_capability=0100000006140001061600000650000106bb010106bb0...
432446db2d5457456743e4476029e14d7c3bb9f5bccYuhao Zheng     * hs20_osu_providers_list=0b5143412d4f53552d425353010901310015656e6757692d...
433446db2d5457456743e4476029e14d7c3bb9f5bccYuhao Zheng     */
434446db2d5457456743e4476029e14d7c3bb9f5bccYuhao Zheng    public String scanResult(String bssid) {
435446db2d5457456743e4476029e14d7c3bb9f5bccYuhao Zheng        return doStringCommand("BSS " + bssid);
436446db2d5457456743e4476029e14d7c3bb9f5bccYuhao Zheng    }
437446db2d5457456743e4476029e14d7c3bb9f5bccYuhao Zheng
438446db2d5457456743e4476029e14d7c3bb9f5bccYuhao Zheng    /**
439155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande     * Format of command
440155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande     * DRIVER WLS_BATCHING SET SCANFREQ=x MSCAN=r BESTN=y CHANNEL=<z, w, t> RTT=s
441155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande     * where x is an ascii representation of an integer number of seconds between scans
442155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande     *       r is an ascii representation of an integer number of scans per batch
443155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande     *       y is an ascii representation of an integer number of the max AP to remember per scan
444155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande     *       z, w, t represent a 1..n size list of channel numbers and/or 'A', 'B' values
445155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande     *           indicating entire ranges of channels
446155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande     *       s is an ascii representation of an integer number of highest-strength AP
447155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande     *           for which we'd like approximate distance reported
448155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande     *
449155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande     * The return value is an ascii integer representing a guess of the number of scans
450155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande     * the firmware can remember before it runs out of buffer space or -1 on error
451155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande     */
452155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    public String setBatchedScanSettings(BatchedScanSettings settings) {
453155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        if (settings == null) {
454155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande            return doStringCommand("DRIVER WLS_BATCHING STOP");
455155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        }
456155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        String cmd = "DRIVER WLS_BATCHING SET SCANFREQ=" + settings.scanIntervalSec;
457155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        cmd += " MSCAN=" + settings.maxScansPerBatch;
458155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        if (settings.maxApPerScan != BatchedScanSettings.UNSPECIFIED) {
459155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande            cmd += " BESTN=" + settings.maxApPerScan;
460155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        }
461155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        if (settings.channelSet != null && !settings.channelSet.isEmpty()) {
462155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande            cmd += " CHANNEL=<";
463155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande            int i = 0;
464155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande            for (String channel : settings.channelSet) {
465155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande                cmd += (i > 0 ? "," : "") + channel;
466155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande                ++i;
467155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande            }
468155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande            cmd += ">";
469155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        }
470155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        if (settings.maxApForDistance != BatchedScanSettings.UNSPECIFIED) {
471155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande            cmd += " RTT=" + settings.maxApForDistance;
472155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        }
473155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        return doStringCommand(cmd);
474155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    }
475155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
476155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    public String getBatchedScanResults() {
477155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        return doStringCommand("DRIVER WLS_BATCHING GET");
478155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    }
479155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
480155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    public boolean startDriver() {
481155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        return doBooleanCommand("DRIVER START");
482155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    }
483155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
484155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    public boolean stopDriver() {
485155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        return doBooleanCommand("DRIVER STOP");
486155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    }
487155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
488155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
489155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    /**
490155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande     * Start filtering out Multicast V4 packets
491155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande     * @return {@code true} if the operation succeeded, {@code false} otherwise
492155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande     *
493155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande     * Multicast filtering rules work as follows:
494155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande     *
495155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande     * The driver can filter multicast (v4 and/or v6) and broadcast packets when in
496155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande     * a power optimized mode (typically when screen goes off).
497155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande     *
498155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande     * In order to prevent the driver from filtering the multicast/broadcast packets, we have to
499155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande     * add a DRIVER RXFILTER-ADD rule followed by DRIVER RXFILTER-START to make the rule effective
500155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande     *
501155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande     * DRIVER RXFILTER-ADD Num
502155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande     *   where Num = 0 - Unicast, 1 - Broadcast, 2 - Mutil4 or 3 - Multi6
503155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande     *
504155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande     * and DRIVER RXFILTER-START
505155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande     * In order to stop the usage of these rules, we do
506155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande     *
507155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande     * DRIVER RXFILTER-STOP
508155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande     * DRIVER RXFILTER-REMOVE Num
509155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande     *   where Num is as described for RXFILTER-ADD
510155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande     *
511155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande     * The  SETSUSPENDOPT driver command overrides the filtering rules
512155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande     */
513155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    public boolean startFilteringMulticastV4Packets() {
514155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        return doBooleanCommand("DRIVER RXFILTER-STOP")
515155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande            && doBooleanCommand("DRIVER RXFILTER-REMOVE 2")
516155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande            && doBooleanCommand("DRIVER RXFILTER-START");
517155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    }
518155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
519155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    /**
520155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande     * Stop filtering out Multicast V4 packets.
521155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande     * @return {@code true} if the operation succeeded, {@code false} otherwise
522155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande     */
523155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    public boolean stopFilteringMulticastV4Packets() {
524155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        return doBooleanCommand("DRIVER RXFILTER-STOP")
525155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande            && doBooleanCommand("DRIVER RXFILTER-ADD 2")
526155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande            && doBooleanCommand("DRIVER RXFILTER-START");
527155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    }
528155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
529155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    /**
530155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande     * Start filtering out Multicast V6 packets
531155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande     * @return {@code true} if the operation succeeded, {@code false} otherwise
532155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande     */
533155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    public boolean startFilteringMulticastV6Packets() {
534155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        return doBooleanCommand("DRIVER RXFILTER-STOP")
535155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande            && doBooleanCommand("DRIVER RXFILTER-REMOVE 3")
536155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande            && doBooleanCommand("DRIVER RXFILTER-START");
537155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    }
538155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
539155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    /**
540155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande     * Stop filtering out Multicast V6 packets.
541155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande     * @return {@code true} if the operation succeeded, {@code false} otherwise
542155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande     */
543155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    public boolean stopFilteringMulticastV6Packets() {
544155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        return doBooleanCommand("DRIVER RXFILTER-STOP")
545155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande            && doBooleanCommand("DRIVER RXFILTER-ADD 3")
546155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande            && doBooleanCommand("DRIVER RXFILTER-START");
547155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    }
548155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
54903cdd624f5da6d62cb731304aa7505921566f69dDmitry Shmidt    /**
55003cdd624f5da6d62cb731304aa7505921566f69dDmitry Shmidt     * Set the operational frequency band
55103cdd624f5da6d62cb731304aa7505921566f69dDmitry Shmidt     * @param band One of
55203cdd624f5da6d62cb731304aa7505921566f69dDmitry Shmidt     *     {@link WifiManager#WIFI_FREQUENCY_BAND_AUTO},
55303cdd624f5da6d62cb731304aa7505921566f69dDmitry Shmidt     *     {@link WifiManager#WIFI_FREQUENCY_BAND_5GHZ},
55403cdd624f5da6d62cb731304aa7505921566f69dDmitry Shmidt     *     {@link WifiManager#WIFI_FREQUENCY_BAND_2GHZ},
55503cdd624f5da6d62cb731304aa7505921566f69dDmitry Shmidt     * @return {@code true} if the operation succeeded, {@code false} otherwise
55603cdd624f5da6d62cb731304aa7505921566f69dDmitry Shmidt     */
557155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    public boolean setBand(int band) {
55803cdd624f5da6d62cb731304aa7505921566f69dDmitry Shmidt        String bandstr;
55903cdd624f5da6d62cb731304aa7505921566f69dDmitry Shmidt
56003cdd624f5da6d62cb731304aa7505921566f69dDmitry Shmidt        if (band == WifiManager.WIFI_FREQUENCY_BAND_5GHZ)
56103cdd624f5da6d62cb731304aa7505921566f69dDmitry Shmidt            bandstr = "5G";
56203cdd624f5da6d62cb731304aa7505921566f69dDmitry Shmidt        else if (band == WifiManager.WIFI_FREQUENCY_BAND_2GHZ)
56303cdd624f5da6d62cb731304aa7505921566f69dDmitry Shmidt            bandstr = "2G";
56403cdd624f5da6d62cb731304aa7505921566f69dDmitry Shmidt        else
56503cdd624f5da6d62cb731304aa7505921566f69dDmitry Shmidt            bandstr = "AUTO";
56603cdd624f5da6d62cb731304aa7505921566f69dDmitry Shmidt        return doBooleanCommand("SET SETBAND " + bandstr);
567155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    }
568155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
5697ef73dd1b6e43c72b3841723504cd86dc402a134Vinit Deshapnde    /**
5707ef73dd1b6e43c72b3841723504cd86dc402a134Vinit Deshapnde      * Sets the bluetooth coexistence mode.
5717ef73dd1b6e43c72b3841723504cd86dc402a134Vinit Deshapnde      *
5727ef73dd1b6e43c72b3841723504cd86dc402a134Vinit Deshapnde      * @param mode One of {@link #BLUETOOTH_COEXISTENCE_MODE_DISABLED},
5737ef73dd1b6e43c72b3841723504cd86dc402a134Vinit Deshapnde      *            {@link #BLUETOOTH_COEXISTENCE_MODE_ENABLED}, or
5747ef73dd1b6e43c72b3841723504cd86dc402a134Vinit Deshapnde      *            {@link #BLUETOOTH_COEXISTENCE_MODE_SENSE}.
5757ef73dd1b6e43c72b3841723504cd86dc402a134Vinit Deshapnde      * @return Whether the mode was successfully set.
5767ef73dd1b6e43c72b3841723504cd86dc402a134Vinit Deshapnde      */
577155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    public boolean setBluetoothCoexistenceMode(int mode) {
578155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        return doBooleanCommand("DRIVER BTCOEXMODE " + mode);
579155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    }
580155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
581155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    /**
582155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande     * Enable or disable Bluetooth coexistence scan mode. When this mode is on,
583155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande     * some of the low-level scan parameters used by the driver are changed to
584155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande     * reduce interference with A2DP streaming.
585155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande     *
586155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande     * @param isSet whether to enable or disable this mode
587155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande     * @return {@code true} if the command succeeded, {@code false} otherwise.
588155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande     */
589155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    public boolean setBluetoothCoexistenceScanMode(boolean setCoexScanMode) {
590155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        if (setCoexScanMode) {
591155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande            return doBooleanCommand("DRIVER BTCOEXSCAN-START");
592155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        } else {
593155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande            return doBooleanCommand("DRIVER BTCOEXSCAN-STOP");
594155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        }
595155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    }
596155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
5970a696d168d7ad98ab5084d2a16e3d02c545a85aaVinit Deshapnde    public void enableSaveConfig() {
5980a696d168d7ad98ab5084d2a16e3d02c545a85aaVinit Deshapnde        doBooleanCommand("SET update_config 1");
5990a696d168d7ad98ab5084d2a16e3d02c545a85aaVinit Deshapnde    }
6000a696d168d7ad98ab5084d2a16e3d02c545a85aaVinit Deshapnde
601155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    public boolean saveConfig() {
602155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        return doBooleanCommand("SAVE_CONFIG");
603155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    }
604155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
605155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    public boolean addToBlacklist(String bssid) {
606155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        if (TextUtils.isEmpty(bssid)) return false;
607155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        return doBooleanCommand("BLACKLIST " + bssid);
608155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    }
609155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
610155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    public boolean clearBlacklist() {
611155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        return doBooleanCommand("BLACKLIST clear");
612155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    }
613155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
614155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    public boolean setSuspendOptimizations(boolean enabled) {
615f22d23092ab37286a5ef9d257d5bb32c421d2669vandwalle       // if (mSuspendOptEnabled == enabled) return true;
616155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        mSuspendOptEnabled = enabled;
617f22d23092ab37286a5ef9d257d5bb32c421d2669vandwalle
618f22d23092ab37286a5ef9d257d5bb32c421d2669vandwalle        Log.e("native", "do suspend " + enabled);
619155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        if (enabled) {
620155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande            return doBooleanCommand("DRIVER SETSUSPENDMODE 1");
621155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        } else {
622155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande            return doBooleanCommand("DRIVER SETSUSPENDMODE 0");
623155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        }
624155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    }
625155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
626155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    public boolean setCountryCode(String countryCode) {
6270465ff513cdccf7e883a505e14c5b78758e7e458Vinit Deshpande        if (countryCode != null)
6280465ff513cdccf7e883a505e14c5b78758e7e458Vinit Deshpande            return doBooleanCommand("DRIVER COUNTRY " + countryCode.toUpperCase(Locale.ROOT));
6290465ff513cdccf7e883a505e14c5b78758e7e458Vinit Deshpande        else
6300465ff513cdccf7e883a505e14c5b78758e7e458Vinit Deshpande            return doBooleanCommand("DRIVER COUNTRY");
631155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    }
632155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
633155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    public void enableBackgroundScan(boolean enable) {
634155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        if (enable) {
635155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande            doBooleanCommand("SET pno 1");
636155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        } else {
637155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande            doBooleanCommand("SET pno 0");
638155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        }
639155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    }
640155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
641f22d23092ab37286a5ef9d257d5bb32c421d2669vandwalle    public void enableAutoConnect(boolean enable) {
642f22d23092ab37286a5ef9d257d5bb32c421d2669vandwalle        if (enable) {
643f22d23092ab37286a5ef9d257d5bb32c421d2669vandwalle            doBooleanCommand("STA_AUTOCONNECT 1");
644f22d23092ab37286a5ef9d257d5bb32c421d2669vandwalle        } else {
645f22d23092ab37286a5ef9d257d5bb32c421d2669vandwalle            doBooleanCommand("STA_AUTOCONNECT 0");
646f22d23092ab37286a5ef9d257d5bb32c421d2669vandwalle        }
647f22d23092ab37286a5ef9d257d5bb32c421d2669vandwalle    }
648f22d23092ab37286a5ef9d257d5bb32c421d2669vandwalle
649155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    public void setScanInterval(int scanInterval) {
650155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        doBooleanCommand("SCAN_INTERVAL " + scanInterval);
651155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    }
652155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
653155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    public void startTdls(String macAddr, boolean enable) {
654155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        if (enable) {
655155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande            doBooleanCommand("TDLS_DISCOVER " + macAddr);
656155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande            doBooleanCommand("TDLS_SETUP " + macAddr);
657155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        } else {
658155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande            doBooleanCommand("TDLS_TEARDOWN " + macAddr);
659155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        }
660155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    }
661155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
662155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    /** Example output:
663155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande     * RSSI=-65
664155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande     * LINKSPEED=48
665155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande     * NOISE=9999
666155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande     * FREQUENCY=0
667155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande     */
668155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    public String signalPoll() {
669155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        return doStringCommandWithoutLogging("SIGNAL_POLL");
670155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    }
671155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
672155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    /** Example outout:
673155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande     * TXGOOD=396
674155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande     * TXBAD=1
675155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande     */
676155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    public String pktcntPoll() {
677155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        return doStringCommand("PKTCNT_POLL");
678155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    }
679155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
680155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    public void bssFlush() {
681155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        doBooleanCommand("BSS_FLUSH 0");
682155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    }
683155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
684155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    public boolean startWpsPbc(String bssid) {
685155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        if (TextUtils.isEmpty(bssid)) {
686155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande            return doBooleanCommand("WPS_PBC");
687155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        } else {
688155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande            return doBooleanCommand("WPS_PBC " + bssid);
689155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        }
690155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    }
691155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
692155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    public boolean startWpsPbc(String iface, String bssid) {
693155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        synchronized (mLock) {
694155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande            if (TextUtils.isEmpty(bssid)) {
695155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande                return doBooleanCommandNative("IFNAME=" + iface + " WPS_PBC");
696155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande            } else {
697155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande                return doBooleanCommandNative("IFNAME=" + iface + " WPS_PBC " + bssid);
698155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande            }
699155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        }
700155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    }
701155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
702155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    public boolean startWpsPinKeypad(String pin) {
703155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        if (TextUtils.isEmpty(pin)) return false;
704155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        return doBooleanCommand("WPS_PIN any " + pin);
705155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    }
706155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
707155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    public boolean startWpsPinKeypad(String iface, String pin) {
708155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        if (TextUtils.isEmpty(pin)) return false;
709155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        synchronized (mLock) {
710155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande            return doBooleanCommandNative("IFNAME=" + iface + " WPS_PIN any " + pin);
711155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        }
712155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    }
713155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
714155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
715155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    public String startWpsPinDisplay(String bssid) {
716155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        if (TextUtils.isEmpty(bssid)) {
717155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande            return doStringCommand("WPS_PIN any");
718155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        } else {
719155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande            return doStringCommand("WPS_PIN " + bssid);
720155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        }
721155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    }
722155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
723155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    public String startWpsPinDisplay(String iface, String bssid) {
724155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        synchronized (mLock) {
725155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande            if (TextUtils.isEmpty(bssid)) {
726155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande                return doStringCommandNative("IFNAME=" + iface + " WPS_PIN any");
727155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande            } else {
728155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande                return doStringCommandNative("IFNAME=" + iface + " WPS_PIN " + bssid);
729155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande            }
730155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        }
731155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    }
732155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
73333b575ca6bee66183929f9474b5a161432918604Vinit Deshpande    public boolean setExternalSim(boolean external) {
73433b575ca6bee66183929f9474b5a161432918604Vinit Deshpande        synchronized (mLock) {
73533b575ca6bee66183929f9474b5a161432918604Vinit Deshpande            String value = external ? "1" : "0";
7364d701eca56d62586b0ab8af6ad864bac74a1dcd0Vinit Deshpande            Log.d(TAG, "Setting external_sim to " + value);
7374d701eca56d62586b0ab8af6ad864bac74a1dcd0Vinit Deshpande            return doBooleanCommand("SET external_sim " + value);
73833b575ca6bee66183929f9474b5a161432918604Vinit Deshpande        }
73933b575ca6bee66183929f9474b5a161432918604Vinit Deshpande    }
74033b575ca6bee66183929f9474b5a161432918604Vinit Deshpande
741f97140d51d14ce0659d381f443c08dbd94dfea28Honore Tricot    public boolean simAuthResponse(int id, String type, String response) {
742f97140d51d14ce0659d381f443c08dbd94dfea28Honore Tricot        // with type = GSM-AUTH, UMTS-AUTH or UMTS-AUTS
74333b575ca6bee66183929f9474b5a161432918604Vinit Deshpande        synchronized (mLock) {
744f97140d51d14ce0659d381f443c08dbd94dfea28Honore Tricot            return doBooleanCommand("CTRL-RSP-SIM-" + id + ":" + type + response);
74533b575ca6bee66183929f9474b5a161432918604Vinit Deshpande        }
74633b575ca6bee66183929f9474b5a161432918604Vinit Deshpande    }
74733b575ca6bee66183929f9474b5a161432918604Vinit Deshpande
748ad607d99c372160c7d4b38e755e1b47d6419856eHonore Tricot    public boolean simIdentityResponse(int id, String response) {
749ad607d99c372160c7d4b38e755e1b47d6419856eHonore Tricot        synchronized (mLock) {
750ad607d99c372160c7d4b38e755e1b47d6419856eHonore Tricot            return doBooleanCommand("CTRL-RSP-IDENTITY-" + id + ":" + response);
751ad607d99c372160c7d4b38e755e1b47d6419856eHonore Tricot        }
752ad607d99c372160c7d4b38e755e1b47d6419856eHonore Tricot    }
753ad607d99c372160c7d4b38e755e1b47d6419856eHonore Tricot
754155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    /* Configures an access point connection */
755155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    public boolean startWpsRegistrar(String bssid, String pin) {
756155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        if (TextUtils.isEmpty(bssid) || TextUtils.isEmpty(pin)) return false;
757155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        return doBooleanCommand("WPS_REG " + bssid + " " + pin);
758155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    }
759155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
760155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    public boolean cancelWps() {
761155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        return doBooleanCommand("WPS_CANCEL");
762155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    }
763155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
764155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    public boolean setPersistentReconnect(boolean enabled) {
765155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        int value = (enabled == true) ? 1 : 0;
766155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        return doBooleanCommand("SET persistent_reconnect " + value);
767155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    }
768155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
769155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    public boolean setDeviceName(String name) {
770155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        return doBooleanCommand("SET device_name " + name);
771155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    }
772155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
773155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    public boolean setDeviceType(String type) {
774155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        return doBooleanCommand("SET device_type " + type);
775155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    }
776155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
777155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    public boolean setConfigMethods(String cfg) {
778155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        return doBooleanCommand("SET config_methods " + cfg);
779155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    }
780155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
781155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    public boolean setManufacturer(String value) {
782155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        return doBooleanCommand("SET manufacturer " + value);
783155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    }
784155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
785155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    public boolean setModelName(String value) {
786155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        return doBooleanCommand("SET model_name " + value);
787155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    }
788155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
789155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    public boolean setModelNumber(String value) {
790155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        return doBooleanCommand("SET model_number " + value);
791155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    }
792155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
793155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    public boolean setSerialNumber(String value) {
794155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        return doBooleanCommand("SET serial_number " + value);
795155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    }
796155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
797155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    public boolean setP2pSsidPostfix(String postfix) {
798155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        return doBooleanCommand("SET p2p_ssid_postfix " + postfix);
799155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    }
800155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
801155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    public boolean setP2pGroupIdle(String iface, int time) {
802155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        synchronized (mLock) {
803155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande            return doBooleanCommandNative("IFNAME=" + iface + " SET p2p_group_idle " + time);
804155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        }
805155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    }
806155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
807155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    public void setPowerSave(boolean enabled) {
808155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        if (enabled) {
809155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande            doBooleanCommand("SET ps 1");
810155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        } else {
811155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande            doBooleanCommand("SET ps 0");
812155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        }
813155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    }
814155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
815155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    public boolean setP2pPowerSave(String iface, boolean enabled) {
816155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        synchronized (mLock) {
817155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande            if (enabled) {
818155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande                return doBooleanCommandNative("IFNAME=" + iface + " P2P_SET ps 1");
819155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande            } else {
820155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande                return doBooleanCommandNative("IFNAME=" + iface + " P2P_SET ps 0");
821155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande            }
822155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        }
823155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    }
824155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
825155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    public boolean setWfdEnable(boolean enable) {
826155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        return doBooleanCommand("SET wifi_display " + (enable ? "1" : "0"));
827155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    }
828155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
829155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    public boolean setWfdDeviceInfo(String hex) {
830155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        return doBooleanCommand("WFD_SUBELEM_SET 0 " + hex);
831155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    }
832155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
833155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    /**
834155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande     * "sta" prioritizes STA connection over P2P and "p2p" prioritizes
835155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande     * P2P connection over STA
836155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande     */
837155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    public boolean setConcurrencyPriority(String s) {
838155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        return doBooleanCommand("P2P_SET conc_pref " + s);
839155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    }
840155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
841155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    public boolean p2pFind() {
842155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        return doBooleanCommand("P2P_FIND");
843155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    }
844155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
845155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    public boolean p2pFind(int timeout) {
846155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        if (timeout <= 0) {
847155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande            return p2pFind();
848155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        }
849155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        return doBooleanCommand("P2P_FIND " + timeout);
850155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    }
851155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
852155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    public boolean p2pStopFind() {
853155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande       return doBooleanCommand("P2P_STOP_FIND");
854155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    }
855155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
856155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    public boolean p2pListen() {
857155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        return doBooleanCommand("P2P_LISTEN");
858155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    }
859155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
860155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    public boolean p2pListen(int timeout) {
861155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        if (timeout <= 0) {
862155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande            return p2pListen();
863155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        }
864155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        return doBooleanCommand("P2P_LISTEN " + timeout);
865155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    }
866155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
867155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    public boolean p2pExtListen(boolean enable, int period, int interval) {
868155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        if (enable && interval < period) {
869155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande            return false;
870155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        }
871155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        return doBooleanCommand("P2P_EXT_LISTEN"
872155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande                    + (enable ? (" " + period + " " + interval) : ""));
873155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    }
874155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
875155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    public boolean p2pSetChannel(int lc, int oc) {
876155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        if (DBG) Log.d(mTAG, "p2pSetChannel: lc="+lc+", oc="+oc);
877155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
878155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        if (lc >=1 && lc <= 11) {
879155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande            if (!doBooleanCommand("P2P_SET listen_channel " + lc)) {
880155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande                return false;
881155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande            }
882155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        } else if (lc != 0) {
883155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande            return false;
884155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        }
885155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
886155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        if (oc >= 1 && oc <= 165 ) {
887155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande            int freq = (oc <= 14 ? 2407 : 5000) + oc * 5;
888155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande            return doBooleanCommand("P2P_SET disallow_freq 1000-"
889155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande                    + (freq - 5) + "," + (freq + 5) + "-6000");
890155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        } else if (oc == 0) {
891155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande            /* oc==0 disables "P2P_SET disallow_freq" (enables all freqs) */
892155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande            return doBooleanCommand("P2P_SET disallow_freq \"\"");
893155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        }
894155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
895155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        return false;
896155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    }
897155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
898155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    public boolean p2pFlush() {
899155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        return doBooleanCommand("P2P_FLUSH");
900155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    }
901155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
902155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    /* p2p_connect <peer device address> <pbc|pin|PIN#> [label|display|keypad]
903155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        [persistent] [join|auth] [go_intent=<0..15>] [freq=<in MHz>] */
904155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    public String p2pConnect(WifiP2pConfig config, boolean joinExistingGroup) {
905155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        if (config == null) return null;
906155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        List<String> args = new ArrayList<String>();
907155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        WpsInfo wps = config.wps;
908155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        args.add(config.deviceAddress);
909155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
910155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        switch (wps.setup) {
911155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande            case WpsInfo.PBC:
912155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande                args.add("pbc");
913155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande                break;
914155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande            case WpsInfo.DISPLAY:
915155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande                if (TextUtils.isEmpty(wps.pin)) {
916155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande                    args.add("pin");
917155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande                } else {
918155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande                    args.add(wps.pin);
919155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande                }
920155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande                args.add("display");
921155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande                break;
922155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande            case WpsInfo.KEYPAD:
923155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande                args.add(wps.pin);
924155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande                args.add("keypad");
925155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande                break;
926155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande            case WpsInfo.LABEL:
927155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande                args.add(wps.pin);
928155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande                args.add("label");
929155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande            default:
930155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande                break;
931155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        }
932155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
933155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        if (config.netId == WifiP2pGroup.PERSISTENT_NET_ID) {
934155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande            args.add("persistent");
935155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        }
936155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
937155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        if (joinExistingGroup) {
938155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande            args.add("join");
939155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        } else {
940155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande            //TODO: This can be adapted based on device plugged in state and
941155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande            //device battery state
942155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande            int groupOwnerIntent = config.groupOwnerIntent;
943155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande            if (groupOwnerIntent < 0 || groupOwnerIntent > 15) {
944155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande                groupOwnerIntent = DEFAULT_GROUP_OWNER_INTENT;
945155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande            }
946155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande            args.add("go_intent=" + groupOwnerIntent);
947155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        }
948155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
949155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        String command = "P2P_CONNECT ";
950155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        for (String s : args) command += s + " ";
951155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
952155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        return doStringCommand(command);
953155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    }
954155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
955155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    public boolean p2pCancelConnect() {
956155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        return doBooleanCommand("P2P_CANCEL");
957155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    }
958155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
959155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    public boolean p2pProvisionDiscovery(WifiP2pConfig config) {
960155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        if (config == null) return false;
961155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
962155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        switch (config.wps.setup) {
963155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande            case WpsInfo.PBC:
964155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande                return doBooleanCommand("P2P_PROV_DISC " + config.deviceAddress + " pbc");
965155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande            case WpsInfo.DISPLAY:
966155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande                //We are doing display, so provision discovery is keypad
967155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande                return doBooleanCommand("P2P_PROV_DISC " + config.deviceAddress + " keypad");
968155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande            case WpsInfo.KEYPAD:
969155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande                //We are doing keypad, so provision discovery is display
970155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande                return doBooleanCommand("P2P_PROV_DISC " + config.deviceAddress + " display");
971155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande            default:
972155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande                break;
973155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        }
974155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        return false;
975155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    }
976155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
977155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    public boolean p2pGroupAdd(boolean persistent) {
978155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        if (persistent) {
979155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande            return doBooleanCommand("P2P_GROUP_ADD persistent");
980155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        }
981155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        return doBooleanCommand("P2P_GROUP_ADD");
982155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    }
983155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
984155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    public boolean p2pGroupAdd(int netId) {
985155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        return doBooleanCommand("P2P_GROUP_ADD persistent=" + netId);
986155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    }
987155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
988155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    public boolean p2pGroupRemove(String iface) {
989155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        if (TextUtils.isEmpty(iface)) return false;
990155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        synchronized (mLock) {
991155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande            return doBooleanCommandNative("IFNAME=" + iface + " P2P_GROUP_REMOVE " + iface);
992155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        }
993155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    }
994155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
995155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    public boolean p2pReject(String deviceAddress) {
996155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        return doBooleanCommand("P2P_REJECT " + deviceAddress);
997155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    }
998155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
999155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    /* Invite a peer to a group */
1000155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    public boolean p2pInvite(WifiP2pGroup group, String deviceAddress) {
1001155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        if (TextUtils.isEmpty(deviceAddress)) return false;
1002155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
1003155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        if (group == null) {
1004155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande            return doBooleanCommand("P2P_INVITE peer=" + deviceAddress);
1005155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        } else {
1006155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande            return doBooleanCommand("P2P_INVITE group=" + group.getInterface()
1007155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande                    + " peer=" + deviceAddress + " go_dev_addr=" + group.getOwner().deviceAddress);
1008155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        }
1009155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    }
1010155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
1011155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    /* Reinvoke a persistent connection */
1012155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    public boolean p2pReinvoke(int netId, String deviceAddress) {
1013155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        if (TextUtils.isEmpty(deviceAddress) || netId < 0) return false;
1014155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
1015155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        return doBooleanCommand("P2P_INVITE persistent=" + netId + " peer=" + deviceAddress);
1016155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    }
1017155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
1018155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    public String p2pGetSsid(String deviceAddress) {
1019155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        return p2pGetParam(deviceAddress, "oper_ssid");
1020155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    }
1021155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
1022155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    public String p2pGetDeviceAddress() {
102327f4b0c6ea9533e91863da48cefc80f8b5a88d1eVinit Deshpande
102436286b23c4e30f042ed753a670c2b462ebf13a48Vinit Deshpande        Log.d(TAG, "p2pGetDeviceAddress");
102536286b23c4e30f042ed753a670c2b462ebf13a48Vinit Deshpande
102627f4b0c6ea9533e91863da48cefc80f8b5a88d1eVinit Deshpande        String status = null;
102727f4b0c6ea9533e91863da48cefc80f8b5a88d1eVinit Deshpande
102836286b23c4e30f042ed753a670c2b462ebf13a48Vinit Deshpande        /* Explicitly calling the API without IFNAME= prefix to take care of the devices that
102936286b23c4e30f042ed753a670c2b462ebf13a48Vinit Deshpande        don't have p2p0 interface. Supplicant seems to be returning the correct address anyway. */
103036286b23c4e30f042ed753a670c2b462ebf13a48Vinit Deshpande
103127f4b0c6ea9533e91863da48cefc80f8b5a88d1eVinit Deshpande        synchronized (mLock) {
103227f4b0c6ea9533e91863da48cefc80f8b5a88d1eVinit Deshpande            status = doStringCommandNative("STATUS");
103327f4b0c6ea9533e91863da48cefc80f8b5a88d1eVinit Deshpande        }
103427f4b0c6ea9533e91863da48cefc80f8b5a88d1eVinit Deshpande
103527f4b0c6ea9533e91863da48cefc80f8b5a88d1eVinit Deshpande        String result = "";
103636286b23c4e30f042ed753a670c2b462ebf13a48Vinit Deshpande        if (status != null) {
103736286b23c4e30f042ed753a670c2b462ebf13a48Vinit Deshpande            String[] tokens = status.split("\n");
103836286b23c4e30f042ed753a670c2b462ebf13a48Vinit Deshpande            for (String token : tokens) {
103936286b23c4e30f042ed753a670c2b462ebf13a48Vinit Deshpande                if (token.startsWith("p2p_device_address=")) {
104036286b23c4e30f042ed753a670c2b462ebf13a48Vinit Deshpande                    String[] nameValue = token.split("=");
104136286b23c4e30f042ed753a670c2b462ebf13a48Vinit Deshpande                    if (nameValue.length != 2)
104236286b23c4e30f042ed753a670c2b462ebf13a48Vinit Deshpande                        break;
104336286b23c4e30f042ed753a670c2b462ebf13a48Vinit Deshpande                    result = nameValue[1];
104436286b23c4e30f042ed753a670c2b462ebf13a48Vinit Deshpande                }
1045155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande            }
1046155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        }
104736286b23c4e30f042ed753a670c2b462ebf13a48Vinit Deshpande
104836286b23c4e30f042ed753a670c2b462ebf13a48Vinit Deshpande        Log.d(TAG, "p2pGetDeviceAddress returning " + result);
104936286b23c4e30f042ed753a670c2b462ebf13a48Vinit Deshpande        return result;
1050155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    }
1051155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
1052155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    public int getGroupCapability(String deviceAddress) {
1053155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        int gc = 0;
1054155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        if (TextUtils.isEmpty(deviceAddress)) return gc;
1055155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        String peerInfo = p2pPeer(deviceAddress);
1056155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        if (TextUtils.isEmpty(peerInfo)) return gc;
1057155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
1058155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        String[] tokens = peerInfo.split("\n");
1059155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        for (String token : tokens) {
1060155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande            if (token.startsWith("group_capab=")) {
1061155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande                String[] nameValue = token.split("=");
1062155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande                if (nameValue.length != 2) break;
1063155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande                try {
1064155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande                    return Integer.decode(nameValue[1]);
1065155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande                } catch(NumberFormatException e) {
1066155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande                    return gc;
1067155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande                }
1068155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande            }
1069155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        }
1070155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        return gc;
1071155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    }
1072155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
1073155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    public String p2pPeer(String deviceAddress) {
1074155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        return doStringCommand("P2P_PEER " + deviceAddress);
1075155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    }
1076155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
1077155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    private String p2pGetParam(String deviceAddress, String key) {
1078155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        if (deviceAddress == null) return null;
1079155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
1080155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        String peerInfo = p2pPeer(deviceAddress);
1081155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        if (peerInfo == null) return null;
1082155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        String[] tokens= peerInfo.split("\n");
1083155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
1084155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        key += "=";
1085155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        for (String token : tokens) {
1086155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande            if (token.startsWith(key)) {
1087155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande                String[] nameValue = token.split("=");
1088155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande                if (nameValue.length != 2) break;
1089155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande                return nameValue[1];
1090155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande            }
1091155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        }
1092155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        return null;
1093155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    }
1094155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
1095155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    public boolean p2pServiceAdd(WifiP2pServiceInfo servInfo) {
1096155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        /*
1097155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande         * P2P_SERVICE_ADD bonjour <query hexdump> <RDATA hexdump>
1098155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande         * P2P_SERVICE_ADD upnp <version hex> <service>
1099155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande         *
1100155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande         * e.g)
1101155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande         * [Bonjour]
1102155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande         * # IP Printing over TCP (PTR) (RDATA=MyPrinter._ipp._tcp.local.)
1103155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande         * P2P_SERVICE_ADD bonjour 045f697070c00c000c01 094d795072696e746572c027
1104155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande         * # IP Printing over TCP (TXT) (RDATA=txtvers=1,pdl=application/postscript)
1105155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande         * P2P_SERVICE_ADD bonjour 096d797072696e746572045f697070c00c001001
1106155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande         *  09747874766572733d311a70646c3d6170706c69636174696f6e2f706f7374736372797074
1107155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande         *
1108155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande         * [UPnP]
1109155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande         * P2P_SERVICE_ADD upnp 10 uuid:6859dede-8574-59ab-9332-123456789012
1110155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande         * P2P_SERVICE_ADD upnp 10 uuid:6859dede-8574-59ab-9332-123456789012::upnp:rootdevice
1111155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande         * P2P_SERVICE_ADD upnp 10 uuid:6859dede-8574-59ab-9332-123456789012::urn:schemas-upnp
1112155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande         * -org:device:InternetGatewayDevice:1
1113155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande         * P2P_SERVICE_ADD upnp 10 uuid:6859dede-8574-59ab-9322-123456789012::urn:schemas-upnp
1114155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande         * -org:service:ContentDirectory:2
1115155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande         */
1116155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        for (String s : servInfo.getSupplicantQueryList()) {
1117155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande            String command = "P2P_SERVICE_ADD";
1118155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande            command += (" " + s);
1119155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande            if (!doBooleanCommand(command)) {
1120155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande                return false;
1121155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande            }
1122155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        }
1123155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        return true;
1124155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    }
1125155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
1126155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    public boolean p2pServiceDel(WifiP2pServiceInfo servInfo) {
1127155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        /*
1128155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande         * P2P_SERVICE_DEL bonjour <query hexdump>
1129155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande         * P2P_SERVICE_DEL upnp <version hex> <service>
1130155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande         */
1131155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        for (String s : servInfo.getSupplicantQueryList()) {
1132155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande            String command = "P2P_SERVICE_DEL ";
1133155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
1134155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande            String[] data = s.split(" ");
1135155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande            if (data.length < 2) {
1136155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande                return false;
1137155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande            }
1138155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande            if ("upnp".equals(data[0])) {
1139155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande                command += s;
1140155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande            } else if ("bonjour".equals(data[0])) {
1141155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande                command += data[0];
1142155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande                command += (" " + data[1]);
1143155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande            } else {
1144155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande                return false;
1145155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande            }
1146155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande            if (!doBooleanCommand(command)) {
1147155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande                return false;
1148155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande            }
1149155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        }
1150155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        return true;
1151155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    }
1152155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
1153155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    public boolean p2pServiceFlush() {
1154155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        return doBooleanCommand("P2P_SERVICE_FLUSH");
1155155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    }
1156155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
1157155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    public String p2pServDiscReq(String addr, String query) {
1158155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        String command = "P2P_SERV_DISC_REQ";
1159155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        command += (" " + addr);
1160155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        command += (" " + query);
1161155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
1162155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        return doStringCommand(command);
1163155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    }
1164155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
1165155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    public boolean p2pServDiscCancelReq(String id) {
1166155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        return doBooleanCommand("P2P_SERV_DISC_CANCEL_REQ " + id);
1167155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    }
1168155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande
1169155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    /* Set the current mode of miracast operation.
1170155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande     *  0 = disabled
1171155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande     *  1 = operating as source
1172155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande     *  2 = operating as sink
1173155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande     */
1174155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    public void setMiracastMode(int mode) {
1175155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        // Note: optional feature on the driver. It is ok for this to fail.
1176155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande        doBooleanCommand("DRIVER MIRACAST " + mode);
1177155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande    }
11783f7ef65ab71619040032aee96b5599849881d6fdAndres Morales
1179446db2d5457456743e4476029e14d7c3bb9f5bccYuhao Zheng    public boolean fetchAnqp(String bssid, String subtypes) {
1180446db2d5457456743e4476029e14d7c3bb9f5bccYuhao Zheng        return doBooleanCommand("ANQP_GET " + bssid + " " + subtypes);
1181446db2d5457456743e4476029e14d7c3bb9f5bccYuhao Zheng    }
1182446db2d5457456743e4476029e14d7c3bb9f5bccYuhao Zheng
11837f9a15d554f69311a0db43347d6473a7c4c46e2eVinit Deshapnde    /* WIFI HAL support */
11847f9a15d554f69311a0db43347d6473a7c4c46e2eVinit Deshapnde
1185b0b1d59786de7ff1fc98e31a16a7d710458bf549Vinit Deshpande    private static final String TAG = "WifiNative-HAL";
1186aabe7a9f2f32915fd1f25416a6d2034a844005d6vandwalle    private static long sWifiHalHandle = 0;  /* used by JNI to save wifi_handle */
1187aabe7a9f2f32915fd1f25416a6d2034a844005d6vandwalle    private static long[] sWifiIfaceHandles = null;  /* used by JNI to save interface handles */
1188aabe7a9f2f32915fd1f25416a6d2034a844005d6vandwalle    private static int sWlan0Index = -1;
1189aabe7a9f2f32915fd1f25416a6d2034a844005d6vandwalle    private static int sP2p0Index = -1;
1190aabe7a9f2f32915fd1f25416a6d2034a844005d6vandwalle
1191aabe7a9f2f32915fd1f25416a6d2034a844005d6vandwalle    private static boolean sHalIsStarted = false;
1192cce360ad854cabb238ba0d9290785c26e837749cVinit Deshpande    private static boolean sHalFailed = false;
11937f9a15d554f69311a0db43347d6473a7c4c46e2eVinit Deshapnde
1194b0b1d59786de7ff1fc98e31a16a7d710458bf549Vinit Deshpande    private static native boolean startHalNative();
1195b0b1d59786de7ff1fc98e31a16a7d710458bf549Vinit Deshpande    private static native void stopHalNative();
1196b0b1d59786de7ff1fc98e31a16a7d710458bf549Vinit Deshpande    private static native void waitForHalEventNative();
11977f9a15d554f69311a0db43347d6473a7c4c46e2eVinit Deshapnde
1198b0b1d59786de7ff1fc98e31a16a7d710458bf549Vinit Deshpande    private static class MonitorThread extends Thread {
11997ef73dd1b6e43c72b3841723504cd86dc402a134Vinit Deshapnde        public void run() {
1200b0b1d59786de7ff1fc98e31a16a7d710458bf549Vinit Deshpande            Log.i(TAG, "Waiting for HAL events mWifiHalHandle=" + Long.toString(sWifiHalHandle));
12017ef73dd1b6e43c72b3841723504cd86dc402a134Vinit Deshapnde            waitForHalEventNative();
12027ef73dd1b6e43c72b3841723504cd86dc402a134Vinit Deshapnde        }
12037ef73dd1b6e43c72b3841723504cd86dc402a134Vinit Deshapnde    }
12047ef73dd1b6e43c72b3841723504cd86dc402a134Vinit Deshapnde
1205b0b1d59786de7ff1fc98e31a16a7d710458bf549Vinit Deshpande    synchronized public static boolean startHal() {
1206d6307b404302949f6dadd14fa0860ff1aec432dcxinhe
1207d6307b404302949f6dadd14fa0860ff1aec432dcxinhe        String debugLog = "startHal stack: ";
1208d6307b404302949f6dadd14fa0860ff1aec432dcxinhe        java.lang.StackTraceElement[] elements = Thread.currentThread().getStackTrace();
1209d6307b404302949f6dadd14fa0860ff1aec432dcxinhe        for (int i = 2; i < elements.length && i <= 7; i++ ) {
1210d6307b404302949f6dadd14fa0860ff1aec432dcxinhe            debugLog = debugLog + " - " + elements[i].getMethodName();
1211d6307b404302949f6dadd14fa0860ff1aec432dcxinhe        }
1212d6307b404302949f6dadd14fa0860ff1aec432dcxinhe
1213d7cbebf7fa9081e294ecab38cfe82709f381e1c9Vinit Deshpande        if (DBG) mLocalLog.log(debugLog);
1214d6307b404302949f6dadd14fa0860ff1aec432dcxinhe
1215aabe7a9f2f32915fd1f25416a6d2034a844005d6vandwalle        synchronized (mLock) {
1216cce360ad854cabb238ba0d9290785c26e837749cVinit Deshpande            if (sHalFailed)
1217cce360ad854cabb238ba0d9290785c26e837749cVinit Deshpande                return false;
1218cce360ad854cabb238ba0d9290785c26e837749cVinit Deshpande            if (startHalNative() && (getInterfaces() != 0) && (sWlan0Index != -1)) {
1219f75cae779514830d18a3e69e03e5e4c6aa14d0b9xinhe                if(!sHalIsStarted){
1220f75cae779514830d18a3e69e03e5e4c6aa14d0b9xinhe                    new MonitorThread().start();
1221f75cae779514830d18a3e69e03e5e4c6aa14d0b9xinhe                }
1222aabe7a9f2f32915fd1f25416a6d2034a844005d6vandwalle                sHalIsStarted = true;
1223aabe7a9f2f32915fd1f25416a6d2034a844005d6vandwalle                return true;
1224aabe7a9f2f32915fd1f25416a6d2034a844005d6vandwalle            } else {
1225b0b1d59786de7ff1fc98e31a16a7d710458bf549Vinit Deshpande                Log.i(TAG, "Could not start hal");
1226aabe7a9f2f32915fd1f25416a6d2034a844005d6vandwalle                sHalIsStarted = false;
1227cce360ad854cabb238ba0d9290785c26e837749cVinit Deshpande                sHalFailed = true;
1228aabe7a9f2f32915fd1f25416a6d2034a844005d6vandwalle                return false;
1229aabe7a9f2f32915fd1f25416a6d2034a844005d6vandwalle            }
12307ef73dd1b6e43c72b3841723504cd86dc402a134Vinit Deshapnde        }
12317ef73dd1b6e43c72b3841723504cd86dc402a134Vinit Deshapnde    }
12327ef73dd1b6e43c72b3841723504cd86dc402a134Vinit Deshapnde
1233b0b1d59786de7ff1fc98e31a16a7d710458bf549Vinit Deshpande    synchronized public static void stopHal() {
12347ef73dd1b6e43c72b3841723504cd86dc402a134Vinit Deshapnde        stopHalNative();
12357ef73dd1b6e43c72b3841723504cd86dc402a134Vinit Deshapnde    }
12367f9a15d554f69311a0db43347d6473a7c4c46e2eVinit Deshapnde
1237b0b1d59786de7ff1fc98e31a16a7d710458bf549Vinit Deshpande    private static native int getInterfacesNative();
12387f9a15d554f69311a0db43347d6473a7c4c46e2eVinit Deshapnde
1239b0b1d59786de7ff1fc98e31a16a7d710458bf549Vinit Deshpande    synchronized public static int getInterfaces() {
1240aabe7a9f2f32915fd1f25416a6d2034a844005d6vandwalle        synchronized (mLock) {
124102a1f98f2cecb8ae2d466d6f9fab06b473f970ddVinit Deshpande            if (sWifiIfaceHandles == null) {
124202a1f98f2cecb8ae2d466d6f9fab06b473f970ddVinit Deshpande                int num = getInterfacesNative();
124302a1f98f2cecb8ae2d466d6f9fab06b473f970ddVinit Deshpande                int wifi_num = 0;
124402a1f98f2cecb8ae2d466d6f9fab06b473f970ddVinit Deshpande                for (int i = 0; i < num; i++) {
124502a1f98f2cecb8ae2d466d6f9fab06b473f970ddVinit Deshpande                    String name = getInterfaceNameNative(i);
124602a1f98f2cecb8ae2d466d6f9fab06b473f970ddVinit Deshpande                    Log.i(TAG, "interface[" + i + "] = " + name);
124702a1f98f2cecb8ae2d466d6f9fab06b473f970ddVinit Deshpande                    if (name.equals("wlan0")) {
124802a1f98f2cecb8ae2d466d6f9fab06b473f970ddVinit Deshpande                        sWlan0Index = i;
124902a1f98f2cecb8ae2d466d6f9fab06b473f970ddVinit Deshpande                        wifi_num++;
125002a1f98f2cecb8ae2d466d6f9fab06b473f970ddVinit Deshpande                    } else if (name.equals("p2p0")) {
125102a1f98f2cecb8ae2d466d6f9fab06b473f970ddVinit Deshpande                        sP2p0Index = i;
125202a1f98f2cecb8ae2d466d6f9fab06b473f970ddVinit Deshpande                        wifi_num++;
125302a1f98f2cecb8ae2d466d6f9fab06b473f970ddVinit Deshpande                    }
1254aabe7a9f2f32915fd1f25416a6d2034a844005d6vandwalle                }
125502a1f98f2cecb8ae2d466d6f9fab06b473f970ddVinit Deshpande                return wifi_num;
125602a1f98f2cecb8ae2d466d6f9fab06b473f970ddVinit Deshpande            } else {
125702a1f98f2cecb8ae2d466d6f9fab06b473f970ddVinit Deshpande                return sWifiIfaceHandles.length;
1258e4e3750390bec0a849a9153348b7c21b2cc8b843Vinit Deshapnde            }
1259e4e3750390bec0a849a9153348b7c21b2cc8b843Vinit Deshapnde        }
12607f9a15d554f69311a0db43347d6473a7c4c46e2eVinit Deshapnde    }
12617f9a15d554f69311a0db43347d6473a7c4c46e2eVinit Deshapnde
1262b0b1d59786de7ff1fc98e31a16a7d710458bf549Vinit Deshpande    private static native String getInterfaceNameNative(int index);
1263a632d8a6edd350c7644b593b18eceaa5b368505bVinit Deshpande    synchronized public static String getInterfaceName(int index) {
1264a632d8a6edd350c7644b593b18eceaa5b368505bVinit Deshpande        return getInterfaceNameNative(index);
12657f9a15d554f69311a0db43347d6473a7c4c46e2eVinit Deshapnde    }
12667f9a15d554f69311a0db43347d6473a7c4c46e2eVinit Deshapnde
1267e4e3750390bec0a849a9153348b7c21b2cc8b843Vinit Deshapnde    public static class ScanCapabilities {
1268e4e3750390bec0a849a9153348b7c21b2cc8b843Vinit Deshapnde        public int  max_scan_cache_size;                 // in number of scan results??
1269e4e3750390bec0a849a9153348b7c21b2cc8b843Vinit Deshapnde        public int  max_scan_buckets;
1270e4e3750390bec0a849a9153348b7c21b2cc8b843Vinit Deshapnde        public int  max_ap_cache_per_scan;
1271e4e3750390bec0a849a9153348b7c21b2cc8b843Vinit Deshapnde        public int  max_rssi_sample_size;
1272e4e3750390bec0a849a9153348b7c21b2cc8b843Vinit Deshapnde        public int  max_scan_reporting_threshold;        // in number of scan results??
12737d6301ead19afdf3de37455e9ed133c25b4938cdVinit Deshpande        public int  max_hotlist_bssids;
1274e4e3750390bec0a849a9153348b7c21b2cc8b843Vinit Deshapnde        public int  max_significant_wifi_change_aps;
1275e4e3750390bec0a849a9153348b7c21b2cc8b843Vinit Deshapnde    }
1276e4e3750390bec0a849a9153348b7c21b2cc8b843Vinit Deshapnde
1277b0b1d59786de7ff1fc98e31a16a7d710458bf549Vinit Deshpande    public static boolean getScanCapabilities(ScanCapabilities capabilities) {
1278aabe7a9f2f32915fd1f25416a6d2034a844005d6vandwalle        return getScanCapabilitiesNative(sWlan0Index, capabilities);
1279e4e3750390bec0a849a9153348b7c21b2cc8b843Vinit Deshapnde    }
1280e4e3750390bec0a849a9153348b7c21b2cc8b843Vinit Deshapnde
1281b0b1d59786de7ff1fc98e31a16a7d710458bf549Vinit Deshpande    private static native boolean getScanCapabilitiesNative(
1282b0b1d59786de7ff1fc98e31a16a7d710458bf549Vinit Deshpande            int iface, ScanCapabilities capabilities);
1283e4e3750390bec0a849a9153348b7c21b2cc8b843Vinit Deshapnde
1284b0b1d59786de7ff1fc98e31a16a7d710458bf549Vinit Deshpande    private static native boolean startScanNative(int iface, int id, ScanSettings settings);
1285b0b1d59786de7ff1fc98e31a16a7d710458bf549Vinit Deshpande    private static native boolean stopScanNative(int iface, int id);
128683a674a18c84ff4f01377bbfd8988699dec93bc2Vinit Deshpande    private static native WifiScanner.ScanData[] getScanResultsNative(int iface, boolean flush);
1287b0b1d59786de7ff1fc98e31a16a7d710458bf549Vinit Deshpande    private static native WifiLinkLayerStats getWifiLinkLayerStatsNative(int iface);
12887f9a15d554f69311a0db43347d6473a7c4c46e2eVinit Deshapnde
1289e4e3750390bec0a849a9153348b7c21b2cc8b843Vinit Deshapnde    public static class ChannelSettings {
1290e4e3750390bec0a849a9153348b7c21b2cc8b843Vinit Deshapnde        int frequency;
1291e4e3750390bec0a849a9153348b7c21b2cc8b843Vinit Deshapnde        int dwell_time_ms;
1292e4e3750390bec0a849a9153348b7c21b2cc8b843Vinit Deshapnde        boolean passive;
12937f9a15d554f69311a0db43347d6473a7c4c46e2eVinit Deshapnde    }
12947f9a15d554f69311a0db43347d6473a7c4c46e2eVinit Deshapnde
1295e4e3750390bec0a849a9153348b7c21b2cc8b843Vinit Deshapnde    public static class BucketSettings {
1296e4e3750390bec0a849a9153348b7c21b2cc8b843Vinit Deshapnde        int bucket;
1297e4e3750390bec0a849a9153348b7c21b2cc8b843Vinit Deshapnde        int band;
1298e4e3750390bec0a849a9153348b7c21b2cc8b843Vinit Deshapnde        int period_ms;
1299e4e3750390bec0a849a9153348b7c21b2cc8b843Vinit Deshapnde        int report_events;
1300e4e3750390bec0a849a9153348b7c21b2cc8b843Vinit Deshapnde        int num_channels;
1301daac2ad767f6047409987bb22812ab5f295e54dfVinit Deshpande        ChannelSettings channels[];
1302e4e3750390bec0a849a9153348b7c21b2cc8b843Vinit Deshapnde    }
13037f9a15d554f69311a0db43347d6473a7c4c46e2eVinit Deshapnde
1304e4e3750390bec0a849a9153348b7c21b2cc8b843Vinit Deshapnde    public static class ScanSettings {
1305e4e3750390bec0a849a9153348b7c21b2cc8b843Vinit Deshapnde        int base_period_ms;
1306e4e3750390bec0a849a9153348b7c21b2cc8b843Vinit Deshapnde        int max_ap_per_scan;
130783a674a18c84ff4f01377bbfd8988699dec93bc2Vinit Deshpande        int report_threshold_percent;
130883a674a18c84ff4f01377bbfd8988699dec93bc2Vinit Deshpande        int report_threshold_num_scans;
1309e4e3750390bec0a849a9153348b7c21b2cc8b843Vinit Deshapnde        int num_buckets;
1310daac2ad767f6047409987bb22812ab5f295e54dfVinit Deshpande        BucketSettings buckets[];
1311e4e3750390bec0a849a9153348b7c21b2cc8b843Vinit Deshapnde    }
13127f9a15d554f69311a0db43347d6473a7c4c46e2eVinit Deshapnde
1313b0b1d59786de7ff1fc98e31a16a7d710458bf549Vinit Deshpande    public static interface ScanEventHandler {
1314e4e3750390bec0a849a9153348b7c21b2cc8b843Vinit Deshapnde        void onScanResultsAvailable();
1315476bee2fef10d060c25c35858b1f7f60803d9f49Vinit Deshpande        void onFullScanResult(ScanResult fullScanResult);
131683a674a18c84ff4f01377bbfd8988699dec93bc2Vinit Deshpande        void onScanStatus();
131783a674a18c84ff4f01377bbfd8988699dec93bc2Vinit Deshpande        void onScanPaused(WifiScanner.ScanData[] data);
1318b0b1d59786de7ff1fc98e31a16a7d710458bf549Vinit Deshpande        void onScanRestarted();
1319e4e3750390bec0a849a9153348b7c21b2cc8b843Vinit Deshapnde    }
1320e4e3750390bec0a849a9153348b7c21b2cc8b843Vinit Deshapnde
1321b0b1d59786de7ff1fc98e31a16a7d710458bf549Vinit Deshpande    synchronized static void onScanResultsAvailable(int id) {
1322b7cc309f06da8a18224057c21ba086f8550367d6Vinit Deshpande        if (sScanEventHandler  != null) {
1323b7cc309f06da8a18224057c21ba086f8550367d6Vinit Deshpande            sScanEventHandler.onScanResultsAvailable();
1324b7cc309f06da8a18224057c21ba086f8550367d6Vinit Deshpande        }
1325b7cc309f06da8a18224057c21ba086f8550367d6Vinit Deshpande    }
1326b7cc309f06da8a18224057c21ba086f8550367d6Vinit Deshpande
1327b7cc309f06da8a18224057c21ba086f8550367d6Vinit Deshpande    /* scan status, keep these values in sync with gscan.h */
1328b7cc309f06da8a18224057c21ba086f8550367d6Vinit Deshpande    private static int WIFI_SCAN_BUFFER_FULL = 0;
1329b7cc309f06da8a18224057c21ba086f8550367d6Vinit Deshpande    private static int WIFI_SCAN_COMPLETE = 1;
1330b7cc309f06da8a18224057c21ba086f8550367d6Vinit Deshpande
1331b7cc309f06da8a18224057c21ba086f8550367d6Vinit Deshpande    synchronized static void onScanStatus(int status) {
1332b7cc309f06da8a18224057c21ba086f8550367d6Vinit Deshpande        Log.i(TAG, "Got a scan status changed event, status = " + status);
1333b7cc309f06da8a18224057c21ba086f8550367d6Vinit Deshpande
1334b7cc309f06da8a18224057c21ba086f8550367d6Vinit Deshpande        if (status == WIFI_SCAN_BUFFER_FULL) {
1335b7cc309f06da8a18224057c21ba086f8550367d6Vinit Deshpande            /* we have a separate event to take care of this */
1336b7cc309f06da8a18224057c21ba086f8550367d6Vinit Deshpande        } else if (status == WIFI_SCAN_COMPLETE) {
1337b7cc309f06da8a18224057c21ba086f8550367d6Vinit Deshpande            if (sScanEventHandler  != null) {
133883a674a18c84ff4f01377bbfd8988699dec93bc2Vinit Deshpande                sScanEventHandler.onScanStatus();
1339b7cc309f06da8a18224057c21ba086f8550367d6Vinit Deshpande            }
1340b7cc309f06da8a18224057c21ba086f8550367d6Vinit Deshpande        }
1341e4e3750390bec0a849a9153348b7c21b2cc8b843Vinit Deshapnde    }
1342e4e3750390bec0a849a9153348b7c21b2cc8b843Vinit Deshapnde
1343dd490cf1df37b70a8625dc4ec8e712f740dd1e4aPierre Vandwalle    static void populateScanResult(ScanResult result, byte bytes[], String dbg) {
1344f1daf9342b66bf134d13fa0a42e929a008f1ca62Vinit Deshapnde        int num = 0;
1345dd490cf1df37b70a8625dc4ec8e712f740dd1e4aPierre Vandwalle        if (bytes == null) return;
1346dd490cf1df37b70a8625dc4ec8e712f740dd1e4aPierre Vandwalle        if (dbg == null) dbg = "";
1347f1daf9342b66bf134d13fa0a42e929a008f1ca62Vinit Deshapnde        for (int i = 0; i < bytes.length; ) {
1348e7b9e74a9a699e74881e5c98c684567763dfaa35Vinit Deshpande            int type  = bytes[i] & 0xFF;
1349e7b9e74a9a699e74881e5c98c684567763dfaa35Vinit Deshpande            int len = bytes[i + 1] & 0xFF;
1350e7b9e74a9a699e74881e5c98c684567763dfaa35Vinit Deshpande
1351e7b9e74a9a699e74881e5c98c684567763dfaa35Vinit Deshpande            if (i + len + 2 > bytes.length) {
1352dd490cf1df37b70a8625dc4ec8e712f740dd1e4aPierre Vandwalle                Log.w(TAG, dbg + "bad length " + len + " of IE " + type + " from " + result.BSSID);
1353dd490cf1df37b70a8625dc4ec8e712f740dd1e4aPierre Vandwalle                Log.w(TAG, dbg + "ignoring the rest of the IEs");
1354e7b9e74a9a699e74881e5c98c684567763dfaa35Vinit Deshpande                break;
1355f1daf9342b66bf134d13fa0a42e929a008f1ca62Vinit Deshapnde            }
1356e7b9e74a9a699e74881e5c98c684567763dfaa35Vinit Deshpande            num++;
1357f1daf9342b66bf134d13fa0a42e929a008f1ca62Vinit Deshapnde            i += len + 2;
1358dd490cf1df37b70a8625dc4ec8e712f740dd1e4aPierre Vandwalle            if (DBG) Log.i(TAG, dbg + "bytes[" + i + "] = [" + type + ", " + len + "]" + ", " +
1359b7cc309f06da8a18224057c21ba086f8550367d6Vinit Deshpande                    "next = " + i);
1360f1daf9342b66bf134d13fa0a42e929a008f1ca62Vinit Deshapnde        }
1361f1daf9342b66bf134d13fa0a42e929a008f1ca62Vinit Deshapnde
1362243931f3474f6235cfcf5c1a55fa2f192ee264aexinhe        int secondChanelOffset = 0;
1363243931f3474f6235cfcf5c1a55fa2f192ee264aexinhe        byte channelMode = 0;
1364243931f3474f6235cfcf5c1a55fa2f192ee264aexinhe        byte centerFreqIndex1 = 0;
1365243931f3474f6235cfcf5c1a55fa2f192ee264aexinhe        byte centerFreqIndex2 = 0;
1366243931f3474f6235cfcf5c1a55fa2f192ee264aexinhe        result.is80211McRTTResponder = false;
1367243931f3474f6235cfcf5c1a55fa2f192ee264aexinhe
1368476bee2fef10d060c25c35858b1f7f60803d9f49Vinit Deshpande        ScanResult.InformationElement elements[] = new ScanResult.InformationElement[num];
1369f1daf9342b66bf134d13fa0a42e929a008f1ca62Vinit Deshapnde        for (int i = 0, index = 0; i < num; i++) {
1370e7b9e74a9a699e74881e5c98c684567763dfaa35Vinit Deshpande            int type  = bytes[index] & 0xFF;
1371e7b9e74a9a699e74881e5c98c684567763dfaa35Vinit Deshpande            int len = bytes[index + 1] & 0xFF;
1372dd490cf1df37b70a8625dc4ec8e712f740dd1e4aPierre Vandwalle            if (DBG) Log.i(TAG, dbg + "index = " + index + ", type = " + type + ", len = " + len);
1373476bee2fef10d060c25c35858b1f7f60803d9f49Vinit Deshpande            ScanResult.InformationElement elem = new ScanResult.InformationElement();
1374f1daf9342b66bf134d13fa0a42e929a008f1ca62Vinit Deshapnde            elem.id = type;
1375f1daf9342b66bf134d13fa0a42e929a008f1ca62Vinit Deshapnde            elem.bytes = new byte[len];
1376f1daf9342b66bf134d13fa0a42e929a008f1ca62Vinit Deshapnde            for (int j = 0; j < len; j++) {
1377f1daf9342b66bf134d13fa0a42e929a008f1ca62Vinit Deshapnde                elem.bytes[j] = bytes[index + j + 2];
1378f1daf9342b66bf134d13fa0a42e929a008f1ca62Vinit Deshapnde            }
1379f1daf9342b66bf134d13fa0a42e929a008f1ca62Vinit Deshapnde            elements[i] = elem;
1380243931f3474f6235cfcf5c1a55fa2f192ee264aexinhe            int inforStart = index + 2;
1381f1daf9342b66bf134d13fa0a42e929a008f1ca62Vinit Deshapnde            index += (len + 2);
1382243931f3474f6235cfcf5c1a55fa2f192ee264aexinhe
1383243931f3474f6235cfcf5c1a55fa2f192ee264aexinhe            if(type == EID_HT_OPERATION) {
1384243931f3474f6235cfcf5c1a55fa2f192ee264aexinhe                secondChanelOffset = bytes[inforStart + 1] & 0x3;
1385243931f3474f6235cfcf5c1a55fa2f192ee264aexinhe            } else if(type == EID_VHT_OPERATION) {
1386243931f3474f6235cfcf5c1a55fa2f192ee264aexinhe                channelMode = bytes[inforStart];
1387243931f3474f6235cfcf5c1a55fa2f192ee264aexinhe                centerFreqIndex1 = bytes[inforStart + 1];
1388243931f3474f6235cfcf5c1a55fa2f192ee264aexinhe                centerFreqIndex2 = bytes[inforStart + 2];
1389243931f3474f6235cfcf5c1a55fa2f192ee264aexinhe            } else if (type == EID_EXTENDED_CAPS) {
1390dd490cf1df37b70a8625dc4ec8e712f740dd1e4aPierre Vandwalle                int tempIndex = RTT_RESP_ENABLE_BIT / 8;
1391dd490cf1df37b70a8625dc4ec8e712f740dd1e4aPierre Vandwalle                byte offset = RTT_RESP_ENABLE_BIT % 8;
1392dd490cf1df37b70a8625dc4ec8e712f740dd1e4aPierre Vandwalle
1393dd490cf1df37b70a8625dc4ec8e712f740dd1e4aPierre Vandwalle                if(len < tempIndex + 1) {
1394dd490cf1df37b70a8625dc4ec8e712f740dd1e4aPierre Vandwalle                    result.is80211McRTTResponder = false;
1395dd490cf1df37b70a8625dc4ec8e712f740dd1e4aPierre Vandwalle                } else {
1396dd490cf1df37b70a8625dc4ec8e712f740dd1e4aPierre Vandwalle                    if ((bytes[inforStart + tempIndex] & ((byte)0x1 << offset)) != 0) {
1397dd490cf1df37b70a8625dc4ec8e712f740dd1e4aPierre Vandwalle                        result.is80211McRTTResponder = true;
1398dd490cf1df37b70a8625dc4ec8e712f740dd1e4aPierre Vandwalle                    } else {
1399dd490cf1df37b70a8625dc4ec8e712f740dd1e4aPierre Vandwalle                        result.is80211McRTTResponder = false;
1400dd490cf1df37b70a8625dc4ec8e712f740dd1e4aPierre Vandwalle                    }
1401dd490cf1df37b70a8625dc4ec8e712f740dd1e4aPierre Vandwalle                }
1402243931f3474f6235cfcf5c1a55fa2f192ee264aexinhe            }
1403243931f3474f6235cfcf5c1a55fa2f192ee264aexinhe        }
1404243931f3474f6235cfcf5c1a55fa2f192ee264aexinhe        //handle RTT related information
1405243931f3474f6235cfcf5c1a55fa2f192ee264aexinhe        if (channelMode != 0) {
1406243931f3474f6235cfcf5c1a55fa2f192ee264aexinhe            // 80 or 160 MHz
1407243931f3474f6235cfcf5c1a55fa2f192ee264aexinhe            result.channelWidth = channelMode + 1;
1408243931f3474f6235cfcf5c1a55fa2f192ee264aexinhe
1409243931f3474f6235cfcf5c1a55fa2f192ee264aexinhe            //convert channel index to frequency in MHz, channel 36 is 5180MHz
1410243931f3474f6235cfcf5c1a55fa2f192ee264aexinhe            result.centerFreq0 = (centerFreqIndex1 - 36) * 5 + 5180;
1411243931f3474f6235cfcf5c1a55fa2f192ee264aexinhe
1412243931f3474f6235cfcf5c1a55fa2f192ee264aexinhe            if(channelMode > 1) { //160MHz
1413243931f3474f6235cfcf5c1a55fa2f192ee264aexinhe                result.centerFreq1 = (centerFreqIndex2 - 36) * 5 + 5180;
1414243931f3474f6235cfcf5c1a55fa2f192ee264aexinhe            } else {
1415243931f3474f6235cfcf5c1a55fa2f192ee264aexinhe                result.centerFreq1 = 0;
1416243931f3474f6235cfcf5c1a55fa2f192ee264aexinhe            }
1417243931f3474f6235cfcf5c1a55fa2f192ee264aexinhe        } else {
1418243931f3474f6235cfcf5c1a55fa2f192ee264aexinhe            //20 or 40 MHz
1419243931f3474f6235cfcf5c1a55fa2f192ee264aexinhe            if (secondChanelOffset != 0) {//40MHz
1420243931f3474f6235cfcf5c1a55fa2f192ee264aexinhe                result.channelWidth = 1;
1421243931f3474f6235cfcf5c1a55fa2f192ee264aexinhe                if (secondChanelOffset == 1) {
1422243931f3474f6235cfcf5c1a55fa2f192ee264aexinhe                    result.centerFreq0 = result.frequency + 20;
1423243931f3474f6235cfcf5c1a55fa2f192ee264aexinhe                } else if (secondChanelOffset == 3) {
1424243931f3474f6235cfcf5c1a55fa2f192ee264aexinhe                    result.centerFreq0 = result.frequency - 20;
1425243931f3474f6235cfcf5c1a55fa2f192ee264aexinhe                } else {
1426243931f3474f6235cfcf5c1a55fa2f192ee264aexinhe                    result.centerFreq0 = 0;
1427dd490cf1df37b70a8625dc4ec8e712f740dd1e4aPierre Vandwalle                    Log.e(TAG, dbg + ": Error on secondChanelOffset");
1428243931f3474f6235cfcf5c1a55fa2f192ee264aexinhe                }
1429243931f3474f6235cfcf5c1a55fa2f192ee264aexinhe            } else {
1430243931f3474f6235cfcf5c1a55fa2f192ee264aexinhe                result.centerFreq0  = 0;
1431243931f3474f6235cfcf5c1a55fa2f192ee264aexinhe                result.centerFreq1  = 0;
1432243931f3474f6235cfcf5c1a55fa2f192ee264aexinhe            }
1433243931f3474f6235cfcf5c1a55fa2f192ee264aexinhe            result.centerFreq1  = 0;
1434243931f3474f6235cfcf5c1a55fa2f192ee264aexinhe        }
1435243931f3474f6235cfcf5c1a55fa2f192ee264aexinhe        if(DBG) {
1436dd490cf1df37b70a8625dc4ec8e712f740dd1e4aPierre Vandwalle            Log.d(TAG, dbg + "SSID: " + result.SSID + " ChannelWidth is: " + result.channelWidth +
1437243931f3474f6235cfcf5c1a55fa2f192ee264aexinhe                    " PrimaryFreq: " + result.frequency +" mCenterfreq0: " + result.centerFreq0 +
1438243931f3474f6235cfcf5c1a55fa2f192ee264aexinhe                    " mCenterfreq1: " + result.centerFreq1 + (result.is80211McRTTResponder ?
1439243931f3474f6235cfcf5c1a55fa2f192ee264aexinhe                    "Support RTT reponder: " : "Do not support RTT responder"));
1440f1daf9342b66bf134d13fa0a42e929a008f1ca62Vinit Deshapnde        }
1441f1daf9342b66bf134d13fa0a42e929a008f1ca62Vinit Deshapnde
1442476bee2fef10d060c25c35858b1f7f60803d9f49Vinit Deshpande        result.informationElements = elements;
1443dd490cf1df37b70a8625dc4ec8e712f740dd1e4aPierre Vandwalle    }
1444dd490cf1df37b70a8625dc4ec8e712f740dd1e4aPierre Vandwalle
1445dd490cf1df37b70a8625dc4ec8e712f740dd1e4aPierre Vandwalle    synchronized static void onFullScanResult(int id, ScanResult result, byte bytes[]) {
1446dd490cf1df37b70a8625dc4ec8e712f740dd1e4aPierre Vandwalle        if (DBG) Log.i(TAG, "Got a full scan results event, ssid = " + result.SSID + ", " +
1447dd490cf1df37b70a8625dc4ec8e712f740dd1e4aPierre Vandwalle                "num = " + bytes.length);
1448dd490cf1df37b70a8625dc4ec8e712f740dd1e4aPierre Vandwalle
1449dd490cf1df37b70a8625dc4ec8e712f740dd1e4aPierre Vandwalle        if (sScanEventHandler == null) {
1450dd490cf1df37b70a8625dc4ec8e712f740dd1e4aPierre Vandwalle            return;
1451dd490cf1df37b70a8625dc4ec8e712f740dd1e4aPierre Vandwalle        }
1452dd490cf1df37b70a8625dc4ec8e712f740dd1e4aPierre Vandwalle        populateScanResult(result, bytes, " onFullScanResult ");
1453dd490cf1df37b70a8625dc4ec8e712f740dd1e4aPierre Vandwalle
1454e7b9e74a9a699e74881e5c98c684567763dfaa35Vinit Deshpande        sScanEventHandler.onFullScanResult(result);
14557f9a15d554f69311a0db43347d6473a7c4c46e2eVinit Deshapnde    }
14567f9a15d554f69311a0db43347d6473a7c4c46e2eVinit Deshapnde
1457b0b1d59786de7ff1fc98e31a16a7d710458bf549Vinit Deshpande    private static int sScanCmdId = 0;
1458b0b1d59786de7ff1fc98e31a16a7d710458bf549Vinit Deshpande    private static ScanEventHandler sScanEventHandler;
1459b0b1d59786de7ff1fc98e31a16a7d710458bf549Vinit Deshpande    private static ScanSettings sScanSettings;
14607f9a15d554f69311a0db43347d6473a7c4c46e2eVinit Deshapnde
1461b0b1d59786de7ff1fc98e31a16a7d710458bf549Vinit Deshpande    synchronized public static boolean startScan(
1462b0b1d59786de7ff1fc98e31a16a7d710458bf549Vinit Deshpande            ScanSettings settings, ScanEventHandler eventHandler) {
14637f9a15d554f69311a0db43347d6473a7c4c46e2eVinit Deshapnde        synchronized (mLock) {
1464b0b1d59786de7ff1fc98e31a16a7d710458bf549Vinit Deshpande
1465b0b1d59786de7ff1fc98e31a16a7d710458bf549Vinit Deshpande            if (sScanCmdId != 0) {
1466741953368eafa247f2820496aaa521bc0e86e9e1Navtej Singh Mann                stopScan();
1467b0b1d59786de7ff1fc98e31a16a7d710458bf549Vinit Deshpande            } else if (sScanSettings != null || sScanEventHandler != null) {
1468b0b1d59786de7ff1fc98e31a16a7d710458bf549Vinit Deshpande                /* current scan is paused; no need to stop it */
1469b0b1d59786de7ff1fc98e31a16a7d710458bf549Vinit Deshpande            }
14707f9a15d554f69311a0db43347d6473a7c4c46e2eVinit Deshapnde
1471b0b1d59786de7ff1fc98e31a16a7d710458bf549Vinit Deshpande            sScanCmdId = getNewCmdIdLocked();
1472e4e3750390bec0a849a9153348b7c21b2cc8b843Vinit Deshapnde
1473b0b1d59786de7ff1fc98e31a16a7d710458bf549Vinit Deshpande            sScanSettings = settings;
1474b0b1d59786de7ff1fc98e31a16a7d710458bf549Vinit Deshpande            sScanEventHandler = eventHandler;
1475b0b1d59786de7ff1fc98e31a16a7d710458bf549Vinit Deshpande
1476b0b1d59786de7ff1fc98e31a16a7d710458bf549Vinit Deshpande            if (startScanNative(sWlan0Index, sScanCmdId, settings) == false) {
1477b0b1d59786de7ff1fc98e31a16a7d710458bf549Vinit Deshpande                sScanEventHandler = null;
1478b0b1d59786de7ff1fc98e31a16a7d710458bf549Vinit Deshpande                sScanSettings = null;
147983a674a18c84ff4f01377bbfd8988699dec93bc2Vinit Deshpande                sScanCmdId = 0;
1480e4e3750390bec0a849a9153348b7c21b2cc8b843Vinit Deshapnde                return false;
1481e4e3750390bec0a849a9153348b7c21b2cc8b843Vinit Deshapnde            }
1482e4e3750390bec0a849a9153348b7c21b2cc8b843Vinit Deshapnde
1483e4e3750390bec0a849a9153348b7c21b2cc8b843Vinit Deshapnde            return true;
1484e4e3750390bec0a849a9153348b7c21b2cc8b843Vinit Deshapnde        }
14857f9a15d554f69311a0db43347d6473a7c4c46e2eVinit Deshapnde    }
14867f9a15d554f69311a0db43347d6473a7c4c46e2eVinit Deshapnde
1487b0b1d59786de7ff1fc98e31a16a7d710458bf549Vinit Deshpande    synchronized public static void stopScan() {
1488b0b1d59786de7ff1fc98e31a16a7d710458bf549Vinit Deshpande        synchronized (mLock) {
1489b0b1d59786de7ff1fc98e31a16a7d710458bf549Vinit Deshpande            stopScanNative(sWlan0Index, sScanCmdId);
1490b0b1d59786de7ff1fc98e31a16a7d710458bf549Vinit Deshpande            sScanSettings = null;
1491b0b1d59786de7ff1fc98e31a16a7d710458bf549Vinit Deshpande            sScanEventHandler = null;
1492b0b1d59786de7ff1fc98e31a16a7d710458bf549Vinit Deshpande            sScanCmdId = 0;
1493b0b1d59786de7ff1fc98e31a16a7d710458bf549Vinit Deshpande        }
1494b0b1d59786de7ff1fc98e31a16a7d710458bf549Vinit Deshpande    }
1495b0b1d59786de7ff1fc98e31a16a7d710458bf549Vinit Deshpande
1496b0b1d59786de7ff1fc98e31a16a7d710458bf549Vinit Deshpande    synchronized public static void pauseScan() {
14977f9a15d554f69311a0db43347d6473a7c4c46e2eVinit Deshapnde        synchronized (mLock) {
1498b0b1d59786de7ff1fc98e31a16a7d710458bf549Vinit Deshpande            if (sScanCmdId != 0 && sScanSettings != null && sScanEventHandler != null) {
1499b0b1d59786de7ff1fc98e31a16a7d710458bf549Vinit Deshpande                Log.d(TAG, "Pausing scan");
150083a674a18c84ff4f01377bbfd8988699dec93bc2Vinit Deshpande                WifiScanner.ScanData scanData[] = getScanResultsNative(sWlan0Index, true);
1501b0b1d59786de7ff1fc98e31a16a7d710458bf549Vinit Deshpande                stopScanNative(sWlan0Index, sScanCmdId);
1502b0b1d59786de7ff1fc98e31a16a7d710458bf549Vinit Deshpande                sScanCmdId = 0;
150383a674a18c84ff4f01377bbfd8988699dec93bc2Vinit Deshpande                sScanEventHandler.onScanPaused(scanData);
1504b0b1d59786de7ff1fc98e31a16a7d710458bf549Vinit Deshpande            }
1505b0b1d59786de7ff1fc98e31a16a7d710458bf549Vinit Deshpande        }
1506b0b1d59786de7ff1fc98e31a16a7d710458bf549Vinit Deshpande    }
1507b0b1d59786de7ff1fc98e31a16a7d710458bf549Vinit Deshpande
1508b0b1d59786de7ff1fc98e31a16a7d710458bf549Vinit Deshpande    synchronized public static void restartScan() {
1509b0b1d59786de7ff1fc98e31a16a7d710458bf549Vinit Deshpande        synchronized (mLock) {
1510b0b1d59786de7ff1fc98e31a16a7d710458bf549Vinit Deshpande            if (sScanCmdId == 0 && sScanSettings != null && sScanEventHandler != null) {
1511b0b1d59786de7ff1fc98e31a16a7d710458bf549Vinit Deshpande                Log.d(TAG, "Restarting scan");
151283a674a18c84ff4f01377bbfd8988699dec93bc2Vinit Deshpande                ScanEventHandler handler = sScanEventHandler;
151383a674a18c84ff4f01377bbfd8988699dec93bc2Vinit Deshpande                ScanSettings settings = sScanSettings;
151483a674a18c84ff4f01377bbfd8988699dec93bc2Vinit Deshpande                if (startScan(sScanSettings, sScanEventHandler)) {
151583a674a18c84ff4f01377bbfd8988699dec93bc2Vinit Deshpande                    sScanEventHandler.onScanRestarted();
151683a674a18c84ff4f01377bbfd8988699dec93bc2Vinit Deshpande                } else {
151783a674a18c84ff4f01377bbfd8988699dec93bc2Vinit Deshpande                    /* we are still paused; don't change state */
151883a674a18c84ff4f01377bbfd8988699dec93bc2Vinit Deshpande                    sScanEventHandler = handler;
151983a674a18c84ff4f01377bbfd8988699dec93bc2Vinit Deshpande                    sScanSettings = settings;
152083a674a18c84ff4f01377bbfd8988699dec93bc2Vinit Deshpande                }
1521b0b1d59786de7ff1fc98e31a16a7d710458bf549Vinit Deshpande            }
1522e4e3750390bec0a849a9153348b7c21b2cc8b843Vinit Deshapnde        }
1523e4e3750390bec0a849a9153348b7c21b2cc8b843Vinit Deshapnde    }
1524e4e3750390bec0a849a9153348b7c21b2cc8b843Vinit Deshapnde
152583a674a18c84ff4f01377bbfd8988699dec93bc2Vinit Deshpande    synchronized public static WifiScanner.ScanData[] getScanResults(boolean flush) {
1526aabe7a9f2f32915fd1f25416a6d2034a844005d6vandwalle        synchronized (mLock) {
152783a674a18c84ff4f01377bbfd8988699dec93bc2Vinit Deshpande            return getScanResultsNative(sWlan0Index, flush);
1528aabe7a9f2f32915fd1f25416a6d2034a844005d6vandwalle        }
1529e4e3750390bec0a849a9153348b7c21b2cc8b843Vinit Deshapnde    }
1530e4e3750390bec0a849a9153348b7c21b2cc8b843Vinit Deshapnde
1531b0b1d59786de7ff1fc98e31a16a7d710458bf549Vinit Deshpande    public static interface HotlistEventHandler {
1532d4762401ec14be6bdd2d27aff2478ddbf8d6ce2aVinit Deshpande        void onHotlistApFound (ScanResult[] result);
1533d4762401ec14be6bdd2d27aff2478ddbf8d6ce2aVinit Deshpande        void onHotlistApLost  (ScanResult[] result);
1534e4e3750390bec0a849a9153348b7c21b2cc8b843Vinit Deshapnde    }
1535e4e3750390bec0a849a9153348b7c21b2cc8b843Vinit Deshapnde
1536b0b1d59786de7ff1fc98e31a16a7d710458bf549Vinit Deshpande    private static int sHotlistCmdId = 0;
1537b0b1d59786de7ff1fc98e31a16a7d710458bf549Vinit Deshpande    private static HotlistEventHandler sHotlistEventHandler;
1538e4e3750390bec0a849a9153348b7c21b2cc8b843Vinit Deshapnde
1539b0b1d59786de7ff1fc98e31a16a7d710458bf549Vinit Deshpande    private native static boolean setHotlistNative(int iface, int id,
1540e4e3750390bec0a849a9153348b7c21b2cc8b843Vinit Deshapnde            WifiScanner.HotlistSettings settings);
1541b0b1d59786de7ff1fc98e31a16a7d710458bf549Vinit Deshpande    private native static boolean resetHotlistNative(int iface, int id);
1542e4e3750390bec0a849a9153348b7c21b2cc8b843Vinit Deshapnde
1543b0b1d59786de7ff1fc98e31a16a7d710458bf549Vinit Deshpande    synchronized public static boolean setHotlist(WifiScanner.HotlistSettings settings,
1544aabe7a9f2f32915fd1f25416a6d2034a844005d6vandwalle                                    HotlistEventHandler eventHandler) {
1545e4e3750390bec0a849a9153348b7c21b2cc8b843Vinit Deshapnde        synchronized (mLock) {
1546b0b1d59786de7ff1fc98e31a16a7d710458bf549Vinit Deshpande            if (sHotlistCmdId != 0) {
1547e4e3750390bec0a849a9153348b7c21b2cc8b843Vinit Deshapnde                return false;
1548e4e3750390bec0a849a9153348b7c21b2cc8b843Vinit Deshapnde            } else {
1549b0b1d59786de7ff1fc98e31a16a7d710458bf549Vinit Deshpande                sHotlistCmdId = getNewCmdIdLocked();
1550e4e3750390bec0a849a9153348b7c21b2cc8b843Vinit Deshapnde            }
1551e4e3750390bec0a849a9153348b7c21b2cc8b843Vinit Deshapnde
1552b0b1d59786de7ff1fc98e31a16a7d710458bf549Vinit Deshpande            sHotlistEventHandler = eventHandler;
1553b0b1d59786de7ff1fc98e31a16a7d710458bf549Vinit Deshpande            if (setHotlistNative(sWlan0Index, sScanCmdId, settings) == false) {
1554b0b1d59786de7ff1fc98e31a16a7d710458bf549Vinit Deshpande                sHotlistEventHandler = null;
1555e4e3750390bec0a849a9153348b7c21b2cc8b843Vinit Deshapnde                return false;
1556e4e3750390bec0a849a9153348b7c21b2cc8b843Vinit Deshapnde            }
1557e4e3750390bec0a849a9153348b7c21b2cc8b843Vinit Deshapnde
1558e4e3750390bec0a849a9153348b7c21b2cc8b843Vinit Deshapnde            return true;
1559e4e3750390bec0a849a9153348b7c21b2cc8b843Vinit Deshapnde        }
1560e4e3750390bec0a849a9153348b7c21b2cc8b843Vinit Deshapnde    }
1561e4e3750390bec0a849a9153348b7c21b2cc8b843Vinit Deshapnde
1562b0b1d59786de7ff1fc98e31a16a7d710458bf549Vinit Deshpande    synchronized public static void resetHotlist() {
1563e4e3750390bec0a849a9153348b7c21b2cc8b843Vinit Deshapnde        synchronized (mLock) {
1564b0b1d59786de7ff1fc98e31a16a7d710458bf549Vinit Deshpande            if (sHotlistCmdId != 0) {
1565b0b1d59786de7ff1fc98e31a16a7d710458bf549Vinit Deshpande                resetHotlistNative(sWlan0Index, sHotlistCmdId);
1566b0b1d59786de7ff1fc98e31a16a7d710458bf549Vinit Deshpande                sHotlistCmdId = 0;
1567b0b1d59786de7ff1fc98e31a16a7d710458bf549Vinit Deshpande                sHotlistEventHandler = null;
1568e4e3750390bec0a849a9153348b7c21b2cc8b843Vinit Deshapnde            }
15697f9a15d554f69311a0db43347d6473a7c4c46e2eVinit Deshapnde        }
15707f9a15d554f69311a0db43347d6473a7c4c46e2eVinit Deshapnde    }
1571e4e3750390bec0a849a9153348b7c21b2cc8b843Vinit Deshapnde
1572b0b1d59786de7ff1fc98e31a16a7d710458bf549Vinit Deshpande    synchronized public static void onHotlistApFound(int id, ScanResult[] results) {
1573aabe7a9f2f32915fd1f25416a6d2034a844005d6vandwalle        synchronized (mLock) {
15741814928371a30b11af31e2bbe5210c4337ed16f3Vinit Deshpande            if (sHotlistCmdId != 0) {
15751814928371a30b11af31e2bbe5210c4337ed16f3Vinit Deshpande                sHotlistEventHandler.onHotlistApFound(results);
15761814928371a30b11af31e2bbe5210c4337ed16f3Vinit Deshpande            } else {
15771814928371a30b11af31e2bbe5210c4337ed16f3Vinit Deshpande                /* this can happen because of race conditions */
1578d4762401ec14be6bdd2d27aff2478ddbf8d6ce2aVinit Deshpande                Log.d(TAG, "Ignoring hotlist AP found event");
1579d4762401ec14be6bdd2d27aff2478ddbf8d6ce2aVinit Deshpande            }
1580d4762401ec14be6bdd2d27aff2478ddbf8d6ce2aVinit Deshpande        }
1581d4762401ec14be6bdd2d27aff2478ddbf8d6ce2aVinit Deshpande    }
1582d4762401ec14be6bdd2d27aff2478ddbf8d6ce2aVinit Deshpande
1583d4762401ec14be6bdd2d27aff2478ddbf8d6ce2aVinit Deshpande    synchronized public static void onHotlistApLost(int id, ScanResult[] results) {
1584d4762401ec14be6bdd2d27aff2478ddbf8d6ce2aVinit Deshpande        synchronized (mLock) {
1585d4762401ec14be6bdd2d27aff2478ddbf8d6ce2aVinit Deshpande            if (sHotlistCmdId != 0) {
1586d4762401ec14be6bdd2d27aff2478ddbf8d6ce2aVinit Deshpande                sHotlistEventHandler.onHotlistApLost(results);
1587d4762401ec14be6bdd2d27aff2478ddbf8d6ce2aVinit Deshpande            } else {
1588d4762401ec14be6bdd2d27aff2478ddbf8d6ce2aVinit Deshpande                /* this can happen because of race conditions */
1589d4762401ec14be6bdd2d27aff2478ddbf8d6ce2aVinit Deshpande                Log.d(TAG, "Ignoring hotlist AP lost event");
15901814928371a30b11af31e2bbe5210c4337ed16f3Vinit Deshpande            }
1591aabe7a9f2f32915fd1f25416a6d2034a844005d6vandwalle        }
1592e4e3750390bec0a849a9153348b7c21b2cc8b843Vinit Deshapnde    }
1593e4e3750390bec0a849a9153348b7c21b2cc8b843Vinit Deshapnde
1594b0b1d59786de7ff1fc98e31a16a7d710458bf549Vinit Deshpande    public static interface SignificantWifiChangeEventHandler {
1595e4e3750390bec0a849a9153348b7c21b2cc8b843Vinit Deshapnde        void onChangesFound(ScanResult[] result);
1596e4e3750390bec0a849a9153348b7c21b2cc8b843Vinit Deshapnde    }
1597e4e3750390bec0a849a9153348b7c21b2cc8b843Vinit Deshapnde
1598b0b1d59786de7ff1fc98e31a16a7d710458bf549Vinit Deshpande    private static SignificantWifiChangeEventHandler sSignificantWifiChangeHandler;
1599b0b1d59786de7ff1fc98e31a16a7d710458bf549Vinit Deshpande    private static int sSignificantWifiChangeCmdId;
1600e4e3750390bec0a849a9153348b7c21b2cc8b843Vinit Deshapnde
1601b0b1d59786de7ff1fc98e31a16a7d710458bf549Vinit Deshpande    private static native boolean trackSignificantWifiChangeNative(
1602e4e3750390bec0a849a9153348b7c21b2cc8b843Vinit Deshapnde            int iface, int id, WifiScanner.WifiChangeSettings settings);
1603b0b1d59786de7ff1fc98e31a16a7d710458bf549Vinit Deshpande    private static native boolean untrackSignificantWifiChangeNative(int iface, int id);
1604e4e3750390bec0a849a9153348b7c21b2cc8b843Vinit Deshapnde
1605b0b1d59786de7ff1fc98e31a16a7d710458bf549Vinit Deshpande    synchronized public static boolean trackSignificantWifiChange(
1606b0b1d59786de7ff1fc98e31a16a7d710458bf549Vinit Deshpande            WifiScanner.WifiChangeSettings settings, SignificantWifiChangeEventHandler handler) {
1607e4e3750390bec0a849a9153348b7c21b2cc8b843Vinit Deshapnde        synchronized (mLock) {
1608b0b1d59786de7ff1fc98e31a16a7d710458bf549Vinit Deshpande            if (sSignificantWifiChangeCmdId != 0) {
1609e4e3750390bec0a849a9153348b7c21b2cc8b843Vinit Deshapnde                return false;
1610e4e3750390bec0a849a9153348b7c21b2cc8b843Vinit Deshapnde            } else {
1611b0b1d59786de7ff1fc98e31a16a7d710458bf549Vinit Deshpande                sSignificantWifiChangeCmdId = getNewCmdIdLocked();
1612e4e3750390bec0a849a9153348b7c21b2cc8b843Vinit Deshapnde            }
1613e4e3750390bec0a849a9153348b7c21b2cc8b843Vinit Deshapnde
1614b0b1d59786de7ff1fc98e31a16a7d710458bf549Vinit Deshpande            sSignificantWifiChangeHandler = handler;
1615b0b1d59786de7ff1fc98e31a16a7d710458bf549Vinit Deshpande            if (trackSignificantWifiChangeNative(sWlan0Index, sScanCmdId, settings) == false) {
16161814928371a30b11af31e2bbe5210c4337ed16f3Vinit Deshpande                sSignificantWifiChangeHandler = null;
1617e4e3750390bec0a849a9153348b7c21b2cc8b843Vinit Deshapnde                return false;
1618e4e3750390bec0a849a9153348b7c21b2cc8b843Vinit Deshapnde            }
1619e4e3750390bec0a849a9153348b7c21b2cc8b843Vinit Deshapnde
1620e4e3750390bec0a849a9153348b7c21b2cc8b843Vinit Deshapnde            return true;
1621e4e3750390bec0a849a9153348b7c21b2cc8b843Vinit Deshapnde        }
1622e4e3750390bec0a849a9153348b7c21b2cc8b843Vinit Deshapnde    }
1623e4e3750390bec0a849a9153348b7c21b2cc8b843Vinit Deshapnde
1624b0b1d59786de7ff1fc98e31a16a7d710458bf549Vinit Deshpande    synchronized static void untrackSignificantWifiChange() {
1625e4e3750390bec0a849a9153348b7c21b2cc8b843Vinit Deshapnde        synchronized (mLock) {
1626b0b1d59786de7ff1fc98e31a16a7d710458bf549Vinit Deshpande            if (sSignificantWifiChangeCmdId != 0) {
1627b0b1d59786de7ff1fc98e31a16a7d710458bf549Vinit Deshpande                untrackSignificantWifiChangeNative(sWlan0Index, sSignificantWifiChangeCmdId);
1628b0b1d59786de7ff1fc98e31a16a7d710458bf549Vinit Deshpande                sSignificantWifiChangeCmdId = 0;
1629b0b1d59786de7ff1fc98e31a16a7d710458bf549Vinit Deshpande                sSignificantWifiChangeHandler = null;
1630e4e3750390bec0a849a9153348b7c21b2cc8b843Vinit Deshapnde            }
1631e4e3750390bec0a849a9153348b7c21b2cc8b843Vinit Deshapnde        }
1632e4e3750390bec0a849a9153348b7c21b2cc8b843Vinit Deshapnde    }
1633e4e3750390bec0a849a9153348b7c21b2cc8b843Vinit Deshapnde
1634b0b1d59786de7ff1fc98e31a16a7d710458bf549Vinit Deshpande    synchronized static void onSignificantWifiChange(int id, ScanResult[] results) {
1635aabe7a9f2f32915fd1f25416a6d2034a844005d6vandwalle        synchronized (mLock) {
16361814928371a30b11af31e2bbe5210c4337ed16f3Vinit Deshpande            if (sSignificantWifiChangeCmdId != 0) {
16371814928371a30b11af31e2bbe5210c4337ed16f3Vinit Deshpande                sSignificantWifiChangeHandler.onChangesFound(results);
16381814928371a30b11af31e2bbe5210c4337ed16f3Vinit Deshpande            } else {
16391814928371a30b11af31e2bbe5210c4337ed16f3Vinit Deshpande                /* this can happen because of race conditions */
16401814928371a30b11af31e2bbe5210c4337ed16f3Vinit Deshpande                Log.d(TAG, "Ignoring significant wifi change");
16411814928371a30b11af31e2bbe5210c4337ed16f3Vinit Deshpande            }
1642aabe7a9f2f32915fd1f25416a6d2034a844005d6vandwalle        }
1643e4e3750390bec0a849a9153348b7c21b2cc8b843Vinit Deshapnde    }
1644e4e3750390bec0a849a9153348b7c21b2cc8b843Vinit Deshapnde
1645200e8ee5097134010a6edee8d031bb02ff7eeb5avandwalle    synchronized public static WifiLinkLayerStats getWifiLinkLayerStats(String iface) {
1646200e8ee5097134010a6edee8d031bb02ff7eeb5avandwalle        // TODO: use correct iface name to Index translation
1647200e8ee5097134010a6edee8d031bb02ff7eeb5avandwalle        if (iface == null) return null;
1648aabe7a9f2f32915fd1f25416a6d2034a844005d6vandwalle        synchronized (mLock) {
1649aabe7a9f2f32915fd1f25416a6d2034a844005d6vandwalle            if (!sHalIsStarted)
1650aabe7a9f2f32915fd1f25416a6d2034a844005d6vandwalle                startHal();
1651aabe7a9f2f32915fd1f25416a6d2034a844005d6vandwalle            if (sHalIsStarted)
1652aabe7a9f2f32915fd1f25416a6d2034a844005d6vandwalle                return getWifiLinkLayerStatsNative(sWlan0Index);
1653aabe7a9f2f32915fd1f25416a6d2034a844005d6vandwalle        }
1654aabe7a9f2f32915fd1f25416a6d2034a844005d6vandwalle        return null;
1655aabe7a9f2f32915fd1f25416a6d2034a844005d6vandwalle    }
16565c08cc119b92af69997af194cc8b6d0111e37d31Andres Morales
16575c08cc119b92af69997af194cc8b6d0111e37d31Andres Morales    /*
16585c08cc119b92af69997af194cc8b6d0111e37d31Andres Morales     * NFC-related calls
16595c08cc119b92af69997af194cc8b6d0111e37d31Andres Morales     */
16605c08cc119b92af69997af194cc8b6d0111e37d31Andres Morales    public String getNfcWpsConfigurationToken(int netId) {
16615c08cc119b92af69997af194cc8b6d0111e37d31Andres Morales        return doStringCommand("WPS_NFC_CONFIG_TOKEN WPS " + netId);
16625c08cc119b92af69997af194cc8b6d0111e37d31Andres Morales    }
16635c08cc119b92af69997af194cc8b6d0111e37d31Andres Morales
16645c08cc119b92af69997af194cc8b6d0111e37d31Andres Morales    public String getNfcHandoverRequest() {
16655c08cc119b92af69997af194cc8b6d0111e37d31Andres Morales        return doStringCommand("NFC_GET_HANDOVER_REQ NDEF P2P-CR");
16665c08cc119b92af69997af194cc8b6d0111e37d31Andres Morales    }
16675c08cc119b92af69997af194cc8b6d0111e37d31Andres Morales
16685c08cc119b92af69997af194cc8b6d0111e37d31Andres Morales    public String getNfcHandoverSelect() {
16695c08cc119b92af69997af194cc8b6d0111e37d31Andres Morales        return doStringCommand("NFC_GET_HANDOVER_SEL NDEF P2P-CR");
16705c08cc119b92af69997af194cc8b6d0111e37d31Andres Morales    }
16715c08cc119b92af69997af194cc8b6d0111e37d31Andres Morales
16725c08cc119b92af69997af194cc8b6d0111e37d31Andres Morales    public boolean initiatorReportNfcHandover(String selectMessage) {
16735c08cc119b92af69997af194cc8b6d0111e37d31Andres Morales        return doBooleanCommand("NFC_REPORT_HANDOVER INIT P2P 00 " + selectMessage);
16745c08cc119b92af69997af194cc8b6d0111e37d31Andres Morales    }
16755c08cc119b92af69997af194cc8b6d0111e37d31Andres Morales
16765c08cc119b92af69997af194cc8b6d0111e37d31Andres Morales    public boolean responderReportNfcHandover(String requestMessage) {
16775c08cc119b92af69997af194cc8b6d0111e37d31Andres Morales        return doBooleanCommand("NFC_REPORT_HANDOVER RESP P2P " + requestMessage + " 00");
16785c08cc119b92af69997af194cc8b6d0111e37d31Andres Morales    }
16795c08cc119b92af69997af194cc8b6d0111e37d31Andres Morales
1680c35361d54d4885c3174499e4ad46d3324387a9bbVinit Deshpande    public static native int getSupportedFeatureSetNative(int iface);
1681a632d8a6edd350c7644b593b18eceaa5b368505bVinit Deshpande    synchronized public static int getSupportedFeatureSet() {
1682c35361d54d4885c3174499e4ad46d3324387a9bbVinit Deshpande        return getSupportedFeatureSetNative(sWlan0Index);
1683a632d8a6edd350c7644b593b18eceaa5b368505bVinit Deshpande    }
1684143657392bf0702a155fe688171a5abbf4c86570Vinit Deshpande
1685143657392bf0702a155fe688171a5abbf4c86570Vinit Deshpande    /* Rtt related commands/events */
1686143657392bf0702a155fe688171a5abbf4c86570Vinit Deshpande    public static interface RttEventHandler {
1687143657392bf0702a155fe688171a5abbf4c86570Vinit Deshpande        void onRttResults(RttManager.RttResult[] result);
1688143657392bf0702a155fe688171a5abbf4c86570Vinit Deshpande    }
1689143657392bf0702a155fe688171a5abbf4c86570Vinit Deshpande
1690143657392bf0702a155fe688171a5abbf4c86570Vinit Deshpande    private static RttEventHandler sRttEventHandler;
1691143657392bf0702a155fe688171a5abbf4c86570Vinit Deshpande    private static int sRttCmdId;
1692143657392bf0702a155fe688171a5abbf4c86570Vinit Deshpande
1693143657392bf0702a155fe688171a5abbf4c86570Vinit Deshpande    synchronized private static void onRttResults(int id, RttManager.RttResult[] results) {
1694143657392bf0702a155fe688171a5abbf4c86570Vinit Deshpande        if (id == sRttCmdId) {
169502a1f98f2cecb8ae2d466d6f9fab06b473f970ddVinit Deshpande            Log.d(TAG, "Received " + results.length + " rtt results");
1696143657392bf0702a155fe688171a5abbf4c86570Vinit Deshpande            sRttEventHandler.onRttResults(results);
1697143657392bf0702a155fe688171a5abbf4c86570Vinit Deshpande            sRttCmdId = 0;
1698143657392bf0702a155fe688171a5abbf4c86570Vinit Deshpande        } else {
1699f95649f33db0a328cb4c0bb5e10c7075e6c828f8xinhe            Log.d(TAG, "RTT Received event for unknown cmd = " + id + ", current id = " + sRttCmdId);
1700143657392bf0702a155fe688171a5abbf4c86570Vinit Deshpande        }
1701143657392bf0702a155fe688171a5abbf4c86570Vinit Deshpande    }
1702143657392bf0702a155fe688171a5abbf4c86570Vinit Deshpande
1703143657392bf0702a155fe688171a5abbf4c86570Vinit Deshpande    private static native boolean requestRangeNative(
1704143657392bf0702a155fe688171a5abbf4c86570Vinit Deshpande            int iface, int id, RttManager.RttParams[] params);
1705143657392bf0702a155fe688171a5abbf4c86570Vinit Deshpande    private static native boolean cancelRangeRequestNative(
1706143657392bf0702a155fe688171a5abbf4c86570Vinit Deshpande            int iface, int id, RttManager.RttParams[] params);
1707143657392bf0702a155fe688171a5abbf4c86570Vinit Deshpande
1708143657392bf0702a155fe688171a5abbf4c86570Vinit Deshpande    synchronized public static boolean requestRtt(
1709143657392bf0702a155fe688171a5abbf4c86570Vinit Deshpande            RttManager.RttParams[] params, RttEventHandler handler) {
1710143657392bf0702a155fe688171a5abbf4c86570Vinit Deshpande        synchronized (mLock) {
1711143657392bf0702a155fe688171a5abbf4c86570Vinit Deshpande            if (sRttCmdId != 0) {
1712f95649f33db0a328cb4c0bb5e10c7075e6c828f8xinhe                Log.v("TAG", "Last one is still under measurement!");
1713143657392bf0702a155fe688171a5abbf4c86570Vinit Deshpande                return false;
1714143657392bf0702a155fe688171a5abbf4c86570Vinit Deshpande            } else {
1715143657392bf0702a155fe688171a5abbf4c86570Vinit Deshpande                sRttCmdId = getNewCmdIdLocked();
1716143657392bf0702a155fe688171a5abbf4c86570Vinit Deshpande            }
1717143657392bf0702a155fe688171a5abbf4c86570Vinit Deshpande            sRttEventHandler = handler;
1718f95649f33db0a328cb4c0bb5e10c7075e6c828f8xinhe            Log.v(TAG, "native issue RTT request");
1719143657392bf0702a155fe688171a5abbf4c86570Vinit Deshpande            return requestRangeNative(sWlan0Index, sRttCmdId, params);
1720143657392bf0702a155fe688171a5abbf4c86570Vinit Deshpande        }
1721143657392bf0702a155fe688171a5abbf4c86570Vinit Deshpande    }
1722143657392bf0702a155fe688171a5abbf4c86570Vinit Deshpande
1723143657392bf0702a155fe688171a5abbf4c86570Vinit Deshpande    synchronized public static boolean cancelRtt(RttManager.RttParams[] params) {
1724143657392bf0702a155fe688171a5abbf4c86570Vinit Deshpande        synchronized(mLock) {
1725143657392bf0702a155fe688171a5abbf4c86570Vinit Deshpande            if (sRttCmdId == 0) {
1726143657392bf0702a155fe688171a5abbf4c86570Vinit Deshpande                return false;
1727143657392bf0702a155fe688171a5abbf4c86570Vinit Deshpande            }
1728143657392bf0702a155fe688171a5abbf4c86570Vinit Deshpande
1729f95649f33db0a328cb4c0bb5e10c7075e6c828f8xinhe            sRttCmdId = 0;
1730f95649f33db0a328cb4c0bb5e10c7075e6c828f8xinhe
1731143657392bf0702a155fe688171a5abbf4c86570Vinit Deshpande            if (cancelRangeRequestNative(sWlan0Index, sRttCmdId, params)) {
1732143657392bf0702a155fe688171a5abbf4c86570Vinit Deshpande                sRttEventHandler = null;
1733f95649f33db0a328cb4c0bb5e10c7075e6c828f8xinhe                Log.v(TAG, "Xin: RTT cancel Request Successfully");
1734143657392bf0702a155fe688171a5abbf4c86570Vinit Deshpande                return true;
1735143657392bf0702a155fe688171a5abbf4c86570Vinit Deshpande            } else {
1736f95649f33db0a328cb4c0bb5e10c7075e6c828f8xinhe                Log.e(TAG, "Xin:RTT cancel Request failed");
1737143657392bf0702a155fe688171a5abbf4c86570Vinit Deshpande                return false;
1738143657392bf0702a155fe688171a5abbf4c86570Vinit Deshpande            }
1739143657392bf0702a155fe688171a5abbf4c86570Vinit Deshpande        }
1740143657392bf0702a155fe688171a5abbf4c86570Vinit Deshpande    }
1741042c54bfd5144ce8e720585b2093796e7e28de5eVinit Deshpande
1742042c54bfd5144ce8e720585b2093796e7e28de5eVinit Deshpande    private static native boolean setScanningMacOuiNative(int iface, byte[] oui);
1743042c54bfd5144ce8e720585b2093796e7e28de5eVinit Deshpande
1744042c54bfd5144ce8e720585b2093796e7e28de5eVinit Deshpande    synchronized public static boolean setScanningMacOui(byte[] oui) {
1745042c54bfd5144ce8e720585b2093796e7e28de5eVinit Deshpande        synchronized (mLock) {
1746042c54bfd5144ce8e720585b2093796e7e28de5eVinit Deshpande            if (startHal()) {
1747042c54bfd5144ce8e720585b2093796e7e28de5eVinit Deshpande                return setScanningMacOuiNative(sWlan0Index, oui);
1748042c54bfd5144ce8e720585b2093796e7e28de5eVinit Deshpande            } else {
1749042c54bfd5144ce8e720585b2093796e7e28de5eVinit Deshpande                return false;
1750042c54bfd5144ce8e720585b2093796e7e28de5eVinit Deshpande            }
1751042c54bfd5144ce8e720585b2093796e7e28de5eVinit Deshpande        }
1752042c54bfd5144ce8e720585b2093796e7e28de5eVinit Deshpande    }
1753efa77c1826499b0a3e57998bd6b3073b107e45d7Vinit Deshpande
1754efa77c1826499b0a3e57998bd6b3073b107e45d7Vinit Deshpande    private static native int[] getChannelsForBandNative(
1755efa77c1826499b0a3e57998bd6b3073b107e45d7Vinit Deshpande            int iface, int band);
1756efa77c1826499b0a3e57998bd6b3073b107e45d7Vinit Deshpande
1757efa77c1826499b0a3e57998bd6b3073b107e45d7Vinit Deshpande    synchronized public static int [] getChannelsForBand(int band) {
1758efa77c1826499b0a3e57998bd6b3073b107e45d7Vinit Deshpande        synchronized (mLock) {
1759efa77c1826499b0a3e57998bd6b3073b107e45d7Vinit Deshpande            if (startHal()) {
1760efa77c1826499b0a3e57998bd6b3073b107e45d7Vinit Deshpande                return getChannelsForBandNative(sWlan0Index, band);
1761efa77c1826499b0a3e57998bd6b3073b107e45d7Vinit Deshpande            } else {
1762efa77c1826499b0a3e57998bd6b3073b107e45d7Vinit Deshpande                return null;
1763efa77c1826499b0a3e57998bd6b3073b107e45d7Vinit Deshpande            }
1764efa77c1826499b0a3e57998bd6b3073b107e45d7Vinit Deshpande        }
1765efa77c1826499b0a3e57998bd6b3073b107e45d7Vinit Deshpande    }
17660465ff513cdccf7e883a505e14c5b78758e7e458Vinit Deshpande
17670465ff513cdccf7e883a505e14c5b78758e7e458Vinit Deshpande
17680465ff513cdccf7e883a505e14c5b78758e7e458Vinit Deshpande    private static native boolean setDfsFlagNative(int iface, boolean dfsOn);
17690465ff513cdccf7e883a505e14c5b78758e7e458Vinit Deshpande    synchronized public static boolean setDfsFlag(boolean dfsOn) {
17700465ff513cdccf7e883a505e14c5b78758e7e458Vinit Deshpande        synchronized (mLock) {
17710465ff513cdccf7e883a505e14c5b78758e7e458Vinit Deshpande            if (startHal()) {
17720465ff513cdccf7e883a505e14c5b78758e7e458Vinit Deshpande                return setDfsFlagNative(sWlan0Index, dfsOn);
17730465ff513cdccf7e883a505e14c5b78758e7e458Vinit Deshpande            } else {
17740465ff513cdccf7e883a505e14c5b78758e7e458Vinit Deshpande                return false;
17750465ff513cdccf7e883a505e14c5b78758e7e458Vinit Deshpande            }
17760465ff513cdccf7e883a505e14c5b78758e7e458Vinit Deshpande        }
17770465ff513cdccf7e883a505e14c5b78758e7e458Vinit Deshpande    }
1778b830d76fbf5fa4077531b516066faa2fdbb92e81xinhe
1779b830d76fbf5fa4077531b516066faa2fdbb92e81xinhe    private static native boolean toggleInterfaceNative(int on);
1780b830d76fbf5fa4077531b516066faa2fdbb92e81xinhe    synchronized public static boolean toggleInterface(int on) {
1781b830d76fbf5fa4077531b516066faa2fdbb92e81xinhe        synchronized (mLock) {
1782b830d76fbf5fa4077531b516066faa2fdbb92e81xinhe            if (startHal()) {
1783b830d76fbf5fa4077531b516066faa2fdbb92e81xinhe                return toggleInterfaceNative(0);
1784b830d76fbf5fa4077531b516066faa2fdbb92e81xinhe            } else {
1785b830d76fbf5fa4077531b516066faa2fdbb92e81xinhe
1786b830d76fbf5fa4077531b516066faa2fdbb92e81xinhe                return false;
1787b830d76fbf5fa4077531b516066faa2fdbb92e81xinhe            }
1788b830d76fbf5fa4077531b516066faa2fdbb92e81xinhe        }
1789b830d76fbf5fa4077531b516066faa2fdbb92e81xinhe    }
179012cf388ecf3b2aa5ba66ed680b931fa356ab75b5xinhe
179112cf388ecf3b2aa5ba66ed680b931fa356ab75b5xinhe    private static native RttManager.RttCapabilities getRttCapabilitiesNative(int iface);
179212cf388ecf3b2aa5ba66ed680b931fa356ab75b5xinhe    synchronized public static RttManager.RttCapabilities getRttCapabilities() {
179312cf388ecf3b2aa5ba66ed680b931fa356ab75b5xinhe        synchronized (mLock) {
179412cf388ecf3b2aa5ba66ed680b931fa356ab75b5xinhe            if (startHal()) {
179512cf388ecf3b2aa5ba66ed680b931fa356ab75b5xinhe                return getRttCapabilitiesNative(sWlan0Index);
179612cf388ecf3b2aa5ba66ed680b931fa356ab75b5xinhe            } else {
179712cf388ecf3b2aa5ba66ed680b931fa356ab75b5xinhe                return null;
179812cf388ecf3b2aa5ba66ed680b931fa356ab75b5xinhe            }
179912cf388ecf3b2aa5ba66ed680b931fa356ab75b5xinhe        }
180012cf388ecf3b2aa5ba66ed680b931fa356ab75b5xinhe    }
1801939177ff615062ec826601d536466875d8457375xinhe
1802939177ff615062ec826601d536466875d8457375xinhe    private static native boolean setCountryCodeHalNative(int iface, String CountryCode);
1803939177ff615062ec826601d536466875d8457375xinhe    synchronized public static boolean setCountryCodeHal( String CountryCode) {
1804939177ff615062ec826601d536466875d8457375xinhe        synchronized (mLock) {
1805939177ff615062ec826601d536466875d8457375xinhe            if (startHal()) {
1806939177ff615062ec826601d536466875d8457375xinhe                return setCountryCodeHalNative(sWlan0Index, CountryCode);
1807939177ff615062ec826601d536466875d8457375xinhe            } else {
1808939177ff615062ec826601d536466875d8457375xinhe                return false;
1809939177ff615062ec826601d536466875d8457375xinhe            }
1810939177ff615062ec826601d536466875d8457375xinhe        }
1811939177ff615062ec826601d536466875d8457375xinhe    }
1812939177ff615062ec826601d536466875d8457375xinhe
1813d57f630f6a7a5fca872d2b96fc4cce1905daee5dxinhe    /* Rtt related commands/events */
1814d57f630f6a7a5fca872d2b96fc4cce1905daee5dxinhe    public abstract class TdlsEventHandler {
1815d57f630f6a7a5fca872d2b96fc4cce1905daee5dxinhe        abstract public void onTdlsStatus(String macAddr, int status, int reason);
1816d57f630f6a7a5fca872d2b96fc4cce1905daee5dxinhe    }
1817d57f630f6a7a5fca872d2b96fc4cce1905daee5dxinhe
1818d57f630f6a7a5fca872d2b96fc4cce1905daee5dxinhe    private static TdlsEventHandler sTdlsEventHandler;
1819d57f630f6a7a5fca872d2b96fc4cce1905daee5dxinhe
1820d57f630f6a7a5fca872d2b96fc4cce1905daee5dxinhe
1821d57f630f6a7a5fca872d2b96fc4cce1905daee5dxinhe    private static native boolean enableDisableTdlsNative(int iface, boolean enable,
1822d57f630f6a7a5fca872d2b96fc4cce1905daee5dxinhe            String macAddr);
1823d57f630f6a7a5fca872d2b96fc4cce1905daee5dxinhe    synchronized public static boolean enableDisableTdls(boolean enable, String macAdd,
1824d57f630f6a7a5fca872d2b96fc4cce1905daee5dxinhe            TdlsEventHandler tdlsCallBack) {
1825d57f630f6a7a5fca872d2b96fc4cce1905daee5dxinhe        synchronized (mLock) {
1826d57f630f6a7a5fca872d2b96fc4cce1905daee5dxinhe            if (startHal()) {
1827d57f630f6a7a5fca872d2b96fc4cce1905daee5dxinhe                sTdlsEventHandler = tdlsCallBack;
1828d57f630f6a7a5fca872d2b96fc4cce1905daee5dxinhe                return enableDisableTdlsNative(sWlan0Index, enable, macAdd);
1829d57f630f6a7a5fca872d2b96fc4cce1905daee5dxinhe            } else {
1830d57f630f6a7a5fca872d2b96fc4cce1905daee5dxinhe                return false;
1831d57f630f6a7a5fca872d2b96fc4cce1905daee5dxinhe            }
1832d57f630f6a7a5fca872d2b96fc4cce1905daee5dxinhe        }
1833d57f630f6a7a5fca872d2b96fc4cce1905daee5dxinhe    }
1834d57f630f6a7a5fca872d2b96fc4cce1905daee5dxinhe
1835d57f630f6a7a5fca872d2b96fc4cce1905daee5dxinhe    // Once TDLS per mac and event feature is implemented, this class definition should be
1836d57f630f6a7a5fca872d2b96fc4cce1905daee5dxinhe    // moved to the right place, like WifiManager etc
1837d57f630f6a7a5fca872d2b96fc4cce1905daee5dxinhe    public static class TdlsStatus {
1838d57f630f6a7a5fca872d2b96fc4cce1905daee5dxinhe        int channel;
1839d57f630f6a7a5fca872d2b96fc4cce1905daee5dxinhe        int global_operating_class;
1840d57f630f6a7a5fca872d2b96fc4cce1905daee5dxinhe        int state;
1841d57f630f6a7a5fca872d2b96fc4cce1905daee5dxinhe        int reason;
1842d57f630f6a7a5fca872d2b96fc4cce1905daee5dxinhe    }
1843d57f630f6a7a5fca872d2b96fc4cce1905daee5dxinhe    private static native TdlsStatus getTdlsStatusNative(int iface, String macAddr);
1844d57f630f6a7a5fca872d2b96fc4cce1905daee5dxinhe    synchronized public static TdlsStatus getTdlsStatus (String macAdd) {
1845d57f630f6a7a5fca872d2b96fc4cce1905daee5dxinhe        synchronized (mLock) {
1846d57f630f6a7a5fca872d2b96fc4cce1905daee5dxinhe            if (startHal()) {
1847d57f630f6a7a5fca872d2b96fc4cce1905daee5dxinhe                return getTdlsStatusNative(sWlan0Index, macAdd);
1848d57f630f6a7a5fca872d2b96fc4cce1905daee5dxinhe            } else {
1849d57f630f6a7a5fca872d2b96fc4cce1905daee5dxinhe                return null;
1850d57f630f6a7a5fca872d2b96fc4cce1905daee5dxinhe            }
1851d57f630f6a7a5fca872d2b96fc4cce1905daee5dxinhe        }
1852d57f630f6a7a5fca872d2b96fc4cce1905daee5dxinhe    }
1853d57f630f6a7a5fca872d2b96fc4cce1905daee5dxinhe
1854d57f630f6a7a5fca872d2b96fc4cce1905daee5dxinhe    //ToFix: Once TDLS per mac and event feature is implemented, this class definition should be
1855d57f630f6a7a5fca872d2b96fc4cce1905daee5dxinhe    // moved to the right place, like WifiStateMachine etc
1856d57f630f6a7a5fca872d2b96fc4cce1905daee5dxinhe    public static class TdlsCapabilities {
1857d57f630f6a7a5fca872d2b96fc4cce1905daee5dxinhe        /* Maximum TDLS session number can be supported by the Firmware and hardware */
1858d57f630f6a7a5fca872d2b96fc4cce1905daee5dxinhe        int maxConcurrentTdlsSessionNumber;
1859d57f630f6a7a5fca872d2b96fc4cce1905daee5dxinhe        boolean isGlobalTdlsSupported;
1860d57f630f6a7a5fca872d2b96fc4cce1905daee5dxinhe        boolean isPerMacTdlsSupported;
1861d57f630f6a7a5fca872d2b96fc4cce1905daee5dxinhe        boolean isOffChannelTdlsSupported;
1862d57f630f6a7a5fca872d2b96fc4cce1905daee5dxinhe    }
1863d57f630f6a7a5fca872d2b96fc4cce1905daee5dxinhe
1864d57f630f6a7a5fca872d2b96fc4cce1905daee5dxinhe
1865d57f630f6a7a5fca872d2b96fc4cce1905daee5dxinhe
1866d57f630f6a7a5fca872d2b96fc4cce1905daee5dxinhe    private static native TdlsCapabilities getTdlsCapabilitiesNative(int iface);
1867d57f630f6a7a5fca872d2b96fc4cce1905daee5dxinhe    synchronized public static TdlsCapabilities getTdlsCapabilities () {
1868d57f630f6a7a5fca872d2b96fc4cce1905daee5dxinhe        synchronized (mLock) {
1869d57f630f6a7a5fca872d2b96fc4cce1905daee5dxinhe            if (startHal()) {
1870d57f630f6a7a5fca872d2b96fc4cce1905daee5dxinhe                return getTdlsCapabilitiesNative(sWlan0Index);
1871d57f630f6a7a5fca872d2b96fc4cce1905daee5dxinhe            } else {
1872d57f630f6a7a5fca872d2b96fc4cce1905daee5dxinhe                return null;
1873d57f630f6a7a5fca872d2b96fc4cce1905daee5dxinhe            }
1874d57f630f6a7a5fca872d2b96fc4cce1905daee5dxinhe        }
1875d57f630f6a7a5fca872d2b96fc4cce1905daee5dxinhe    }
1876d57f630f6a7a5fca872d2b96fc4cce1905daee5dxinhe
1877d57f630f6a7a5fca872d2b96fc4cce1905daee5dxinhe    synchronized private static boolean onTdlsStatus(String macAddr, int status, int reason) {
1878d57f630f6a7a5fca872d2b96fc4cce1905daee5dxinhe         if (sTdlsEventHandler == null) {
1879d57f630f6a7a5fca872d2b96fc4cce1905daee5dxinhe             return false;
1880d57f630f6a7a5fca872d2b96fc4cce1905daee5dxinhe         } else {
1881d57f630f6a7a5fca872d2b96fc4cce1905daee5dxinhe             sTdlsEventHandler.onTdlsStatus(macAddr, status, reason);
1882d57f630f6a7a5fca872d2b96fc4cce1905daee5dxinhe             return true;
1883d57f630f6a7a5fca872d2b96fc4cce1905daee5dxinhe         }
1884d57f630f6a7a5fca872d2b96fc4cce1905daee5dxinhe    }
1885d57f630f6a7a5fca872d2b96fc4cce1905daee5dxinhe
1886a0d34d3626bd9631039a485d2d428b1b7de390e2Pierre Vandwalle    //---------------------------------------------------------------------------------
1887a0d34d3626bd9631039a485d2d428b1b7de390e2Pierre Vandwalle
1888a0d34d3626bd9631039a485d2d428b1b7de390e2Pierre Vandwalle    /* Wifi Logger commands/events */
18897d6301ead19afdf3de37455e9ed133c25b4938cdVinit Deshpande
189003ba4a5b108ab2bbdb5d0e7be46ac45c00a6d9ffxinhe    public static native boolean startLogging(int iface);
18917d6301ead19afdf3de37455e9ed133c25b4938cdVinit Deshpande
1892a0d34d3626bd9631039a485d2d428b1b7de390e2Pierre Vandwalle    public static interface WifiLoggerEventHandler {
1893a0d34d3626bd9631039a485d2d428b1b7de390e2Pierre Vandwalle        void onDataAvailable(char data[], int len);
1894a0d34d3626bd9631039a485d2d428b1b7de390e2Pierre Vandwalle    }
1895a0d34d3626bd9631039a485d2d428b1b7de390e2Pierre Vandwalle
1896a0d34d3626bd9631039a485d2d428b1b7de390e2Pierre Vandwalle    private static WifiLoggerEventHandler sWifiLoggerEventHandler = null;
1897a0d34d3626bd9631039a485d2d428b1b7de390e2Pierre Vandwalle
189803ba4a5b108ab2bbdb5d0e7be46ac45c00a6d9ffxinhe    public static class WifiLoggerEvent {
189903ba4a5b108ab2bbdb5d0e7be46ac45c00a6d9ffxinhe        WifiLogger.RingBufferStatus status;
190003ba4a5b108ab2bbdb5d0e7be46ac45c00a6d9ffxinhe        int entrySize;
190103ba4a5b108ab2bbdb5d0e7be46ac45c00a6d9ffxinhe        int flags;
190203ba4a5b108ab2bbdb5d0e7be46ac45c00a6d9ffxinhe        int type;
190303ba4a5b108ab2bbdb5d0e7be46ac45c00a6d9ffxinhe        long timestamp;
190403ba4a5b108ab2bbdb5d0e7be46ac45c00a6d9ffxinhe        byte[] entry;
190503ba4a5b108ab2bbdb5d0e7be46ac45c00a6d9ffxinhe
190603ba4a5b108ab2bbdb5d0e7be46ac45c00a6d9ffxinhe        public String toString() {
190703ba4a5b108ab2bbdb5d0e7be46ac45c00a6d9ffxinhe            StringBuilder str =  new StringBuilder();
190803ba4a5b108ab2bbdb5d0e7be46ac45c00a6d9ffxinhe            str.append(status + "\n entry size: " + entrySize + " flags: " + flags);
190903ba4a5b108ab2bbdb5d0e7be46ac45c00a6d9ffxinhe            str.append(" type: " + type + " timestamp: " + timestamp +"\n");
191003ba4a5b108ab2bbdb5d0e7be46ac45c00a6d9ffxinhe            if (entry != null) {
191103ba4a5b108ab2bbdb5d0e7be46ac45c00a6d9ffxinhe                str.append(android.util.Base64.encodeToString(entry, Base64.DEFAULT));
191203ba4a5b108ab2bbdb5d0e7be46ac45c00a6d9ffxinhe            } else {
191303ba4a5b108ab2bbdb5d0e7be46ac45c00a6d9ffxinhe                str.append(" empty bytes[]");
191403ba4a5b108ab2bbdb5d0e7be46ac45c00a6d9ffxinhe            }
191503ba4a5b108ab2bbdb5d0e7be46ac45c00a6d9ffxinhe            str.append("\n\n");
191603ba4a5b108ab2bbdb5d0e7be46ac45c00a6d9ffxinhe            return str.toString();
191703ba4a5b108ab2bbdb5d0e7be46ac45c00a6d9ffxinhe        }
191803ba4a5b108ab2bbdb5d0e7be46ac45c00a6d9ffxinhe    }
191903ba4a5b108ab2bbdb5d0e7be46ac45c00a6d9ffxinhe
192003ba4a5b108ab2bbdb5d0e7be46ac45c00a6d9ffxinhe    private static void onWifiLoggerEvent(WifiLoggerEvent event) {
192103ba4a5b108ab2bbdb5d0e7be46ac45c00a6d9ffxinhe        if (event != null) {
192203ba4a5b108ab2bbdb5d0e7be46ac45c00a6d9ffxinhe            Log.d(TAG, "Logger Event:" + event);
192303ba4a5b108ab2bbdb5d0e7be46ac45c00a6d9ffxinhe        }
192403ba4a5b108ab2bbdb5d0e7be46ac45c00a6d9ffxinhe    }
192503ba4a5b108ab2bbdb5d0e7be46ac45c00a6d9ffxinhe
192603ba4a5b108ab2bbdb5d0e7be46ac45c00a6d9ffxinhe    private static void onWifiAlert(byte[] buffer, int errorCode) {
192703ba4a5b108ab2bbdb5d0e7be46ac45c00a6d9ffxinhe        Log.d(TAG, "Logger Alert error code:" + errorCode);
192803ba4a5b108ab2bbdb5d0e7be46ac45c00a6d9ffxinhe        if (buffer != null) {
192903ba4a5b108ab2bbdb5d0e7be46ac45c00a6d9ffxinhe            StringBuilder str =  new StringBuilder();
193003ba4a5b108ab2bbdb5d0e7be46ac45c00a6d9ffxinhe            str.append(android.util.Base64.encodeToString(buffer, android.util.Base64.NO_WRAP));
193103ba4a5b108ab2bbdb5d0e7be46ac45c00a6d9ffxinhe            Log.e(TAG,"Logger Alert event:");
193203ba4a5b108ab2bbdb5d0e7be46ac45c00a6d9ffxinhe            Log.e(TAG, str.toString());
193303ba4a5b108ab2bbdb5d0e7be46ac45c00a6d9ffxinhe        } else {
193403ba4a5b108ab2bbdb5d0e7be46ac45c00a6d9ffxinhe            Log.e(TAG," empty Alert");
193503ba4a5b108ab2bbdb5d0e7be46ac45c00a6d9ffxinhe        }
193603ba4a5b108ab2bbdb5d0e7be46ac45c00a6d9ffxinhe    }
193703ba4a5b108ab2bbdb5d0e7be46ac45c00a6d9ffxinhe
193803ba4a5b108ab2bbdb5d0e7be46ac45c00a6d9ffxinhe    private static native boolean startLoggingRingBufferNative(int iface, int verboseLevel,
193903ba4a5b108ab2bbdb5d0e7be46ac45c00a6d9ffxinhe            int flags, int maxInterval,int minDataSize, String ringName);
194003ba4a5b108ab2bbdb5d0e7be46ac45c00a6d9ffxinhe    synchronized public static boolean startLoggingRingBuffer(int verboseLevel, int flags, int maxInterval,
194103ba4a5b108ab2bbdb5d0e7be46ac45c00a6d9ffxinhe            int minDataSize, String ringName){
194203ba4a5b108ab2bbdb5d0e7be46ac45c00a6d9ffxinhe        synchronized (mLock) {
194303ba4a5b108ab2bbdb5d0e7be46ac45c00a6d9ffxinhe            if (startHal()) {
194403ba4a5b108ab2bbdb5d0e7be46ac45c00a6d9ffxinhe                return startLoggingRingBufferNative(sWlan0Index, verboseLevel, flags, maxInterval,
194503ba4a5b108ab2bbdb5d0e7be46ac45c00a6d9ffxinhe                        minDataSize, ringName);
194603ba4a5b108ab2bbdb5d0e7be46ac45c00a6d9ffxinhe            } else {
194703ba4a5b108ab2bbdb5d0e7be46ac45c00a6d9ffxinhe                return false;
194803ba4a5b108ab2bbdb5d0e7be46ac45c00a6d9ffxinhe            }
194903ba4a5b108ab2bbdb5d0e7be46ac45c00a6d9ffxinhe        }
195003ba4a5b108ab2bbdb5d0e7be46ac45c00a6d9ffxinhe    }
195103ba4a5b108ab2bbdb5d0e7be46ac45c00a6d9ffxinhe
195203ba4a5b108ab2bbdb5d0e7be46ac45c00a6d9ffxinhe    private static native int getSupportedLoggerFeatureSetNative(int iface);
195303ba4a5b108ab2bbdb5d0e7be46ac45c00a6d9ffxinhe    synchronized public static int getSupportedLoggerFeatureSet() {
195403ba4a5b108ab2bbdb5d0e7be46ac45c00a6d9ffxinhe        synchronized (mLock) {
195503ba4a5b108ab2bbdb5d0e7be46ac45c00a6d9ffxinhe            if (startHal()) {
195603ba4a5b108ab2bbdb5d0e7be46ac45c00a6d9ffxinhe                return getSupportedLoggerFeatureSetNative(sWlan0Index);
195703ba4a5b108ab2bbdb5d0e7be46ac45c00a6d9ffxinhe            } else {
195803ba4a5b108ab2bbdb5d0e7be46ac45c00a6d9ffxinhe                return -1;
195903ba4a5b108ab2bbdb5d0e7be46ac45c00a6d9ffxinhe            }
196003ba4a5b108ab2bbdb5d0e7be46ac45c00a6d9ffxinhe        }
196103ba4a5b108ab2bbdb5d0e7be46ac45c00a6d9ffxinhe    }
196203ba4a5b108ab2bbdb5d0e7be46ac45c00a6d9ffxinhe
196303ba4a5b108ab2bbdb5d0e7be46ac45c00a6d9ffxinhe    private static native String getDriverVersionNative(int iface);
196403ba4a5b108ab2bbdb5d0e7be46ac45c00a6d9ffxinhe    synchronized public static String getDriverVersion() {
196503ba4a5b108ab2bbdb5d0e7be46ac45c00a6d9ffxinhe        synchronized (mLock) {
196603ba4a5b108ab2bbdb5d0e7be46ac45c00a6d9ffxinhe            if (startHal()) {
196703ba4a5b108ab2bbdb5d0e7be46ac45c00a6d9ffxinhe                return getDriverVersionNative(sWlan0Index);
196803ba4a5b108ab2bbdb5d0e7be46ac45c00a6d9ffxinhe            } else {
196903ba4a5b108ab2bbdb5d0e7be46ac45c00a6d9ffxinhe                return null;
197003ba4a5b108ab2bbdb5d0e7be46ac45c00a6d9ffxinhe            }
197103ba4a5b108ab2bbdb5d0e7be46ac45c00a6d9ffxinhe        }
197203ba4a5b108ab2bbdb5d0e7be46ac45c00a6d9ffxinhe    }
197303ba4a5b108ab2bbdb5d0e7be46ac45c00a6d9ffxinhe
197403ba4a5b108ab2bbdb5d0e7be46ac45c00a6d9ffxinhe
197503ba4a5b108ab2bbdb5d0e7be46ac45c00a6d9ffxinhe    private static native String getFirmwareVersionNative(int iface);
197603ba4a5b108ab2bbdb5d0e7be46ac45c00a6d9ffxinhe    synchronized public static String getFirmwareVersion() {
197703ba4a5b108ab2bbdb5d0e7be46ac45c00a6d9ffxinhe        synchronized (mLock) {
197803ba4a5b108ab2bbdb5d0e7be46ac45c00a6d9ffxinhe            if (startHal()) {
197903ba4a5b108ab2bbdb5d0e7be46ac45c00a6d9ffxinhe                return getFirmwareVersionNative(sWlan0Index);
198003ba4a5b108ab2bbdb5d0e7be46ac45c00a6d9ffxinhe            } else {
198103ba4a5b108ab2bbdb5d0e7be46ac45c00a6d9ffxinhe                return null;
198203ba4a5b108ab2bbdb5d0e7be46ac45c00a6d9ffxinhe            }
198303ba4a5b108ab2bbdb5d0e7be46ac45c00a6d9ffxinhe        }
198403ba4a5b108ab2bbdb5d0e7be46ac45c00a6d9ffxinhe    }
198503ba4a5b108ab2bbdb5d0e7be46ac45c00a6d9ffxinhe
198603ba4a5b108ab2bbdb5d0e7be46ac45c00a6d9ffxinhe    private static native WifiLogger.RingBufferStatus[] getRingBufferStatusNative(int iface);
198703ba4a5b108ab2bbdb5d0e7be46ac45c00a6d9ffxinhe    synchronized public static WifiLogger.RingBufferStatus[] getRingBufferStatus() {
198803ba4a5b108ab2bbdb5d0e7be46ac45c00a6d9ffxinhe        synchronized (mLock) {
198903ba4a5b108ab2bbdb5d0e7be46ac45c00a6d9ffxinhe            if (startHal()) {
199003ba4a5b108ab2bbdb5d0e7be46ac45c00a6d9ffxinhe                return getRingBufferStatusNative(sWlan0Index);
199103ba4a5b108ab2bbdb5d0e7be46ac45c00a6d9ffxinhe            } else {
199203ba4a5b108ab2bbdb5d0e7be46ac45c00a6d9ffxinhe                return null;
199303ba4a5b108ab2bbdb5d0e7be46ac45c00a6d9ffxinhe            }
199403ba4a5b108ab2bbdb5d0e7be46ac45c00a6d9ffxinhe        }
199503ba4a5b108ab2bbdb5d0e7be46ac45c00a6d9ffxinhe    }
199603ba4a5b108ab2bbdb5d0e7be46ac45c00a6d9ffxinhe
199703ba4a5b108ab2bbdb5d0e7be46ac45c00a6d9ffxinhe    private static native boolean getRingBufferDataNative(int iface, String ringName);
199803ba4a5b108ab2bbdb5d0e7be46ac45c00a6d9ffxinhe    synchronized public static boolean getRingBufferData(String ringName) {
199903ba4a5b108ab2bbdb5d0e7be46ac45c00a6d9ffxinhe        synchronized (mLock) {
200003ba4a5b108ab2bbdb5d0e7be46ac45c00a6d9ffxinhe            if (startHal()) {
200103ba4a5b108ab2bbdb5d0e7be46ac45c00a6d9ffxinhe                return getRingBufferDataNative(sWlan0Index, ringName);
200203ba4a5b108ab2bbdb5d0e7be46ac45c00a6d9ffxinhe            } else {
200303ba4a5b108ab2bbdb5d0e7be46ac45c00a6d9ffxinhe                return false;
200403ba4a5b108ab2bbdb5d0e7be46ac45c00a6d9ffxinhe            }
200503ba4a5b108ab2bbdb5d0e7be46ac45c00a6d9ffxinhe        }
200603ba4a5b108ab2bbdb5d0e7be46ac45c00a6d9ffxinhe    }
200703ba4a5b108ab2bbdb5d0e7be46ac45c00a6d9ffxinhe
200898dca3397c0049d3cfb9c97e2b021c5ee3a2e1bdxinhe    static private byte[] mFwMemoryDump;
200903ba4a5b108ab2bbdb5d0e7be46ac45c00a6d9ffxinhe    private static void onWifiFwMemoryAvailable(byte[] buffer) {
201098dca3397c0049d3cfb9c97e2b021c5ee3a2e1bdxinhe        mFwMemoryDump = buffer;
201198dca3397c0049d3cfb9c97e2b021c5ee3a2e1bdxinhe        if (DBG) {
201298dca3397c0049d3cfb9c97e2b021c5ee3a2e1bdxinhe            Log.d(TAG, "onWifiFwMemoryAvailable is called and buffer length is: " +
201398dca3397c0049d3cfb9c97e2b021c5ee3a2e1bdxinhe                    (buffer == null ? 0 :  buffer.length));
201498dca3397c0049d3cfb9c97e2b021c5ee3a2e1bdxinhe        }
201503ba4a5b108ab2bbdb5d0e7be46ac45c00a6d9ffxinhe    }
201603ba4a5b108ab2bbdb5d0e7be46ac45c00a6d9ffxinhe    private static native boolean getFwMemoryDumpNative(int iface);
201703ba4a5b108ab2bbdb5d0e7be46ac45c00a6d9ffxinhe    synchronized public static String getFwMemoryDump() {
201803ba4a5b108ab2bbdb5d0e7be46ac45c00a6d9ffxinhe        synchronized (mLock) {
201903ba4a5b108ab2bbdb5d0e7be46ac45c00a6d9ffxinhe            if (startHal()) {
202003ba4a5b108ab2bbdb5d0e7be46ac45c00a6d9ffxinhe                if(getFwMemoryDumpNative(sWlan0Index)) {
202198dca3397c0049d3cfb9c97e2b021c5ee3a2e1bdxinhe                    if(mFwMemoryDump == null || mFwMemoryDump.length == 0) {
202298dca3397c0049d3cfb9c97e2b021c5ee3a2e1bdxinhe                         return null;
202398dca3397c0049d3cfb9c97e2b021c5ee3a2e1bdxinhe                    }
202498dca3397c0049d3cfb9c97e2b021c5ee3a2e1bdxinhe                    String result;
202598dca3397c0049d3cfb9c97e2b021c5ee3a2e1bdxinhe                    //compress
202698dca3397c0049d3cfb9c97e2b021c5ee3a2e1bdxinhe                    Deflater compressor = new Deflater();
202798dca3397c0049d3cfb9c97e2b021c5ee3a2e1bdxinhe                    compressor.setLevel(Deflater.BEST_COMPRESSION);
202898dca3397c0049d3cfb9c97e2b021c5ee3a2e1bdxinhe                    compressor.setInput(mFwMemoryDump);
202998dca3397c0049d3cfb9c97e2b021c5ee3a2e1bdxinhe                    compressor.finish();
203098dca3397c0049d3cfb9c97e2b021c5ee3a2e1bdxinhe                    ByteArrayOutputStream bos = new ByteArrayOutputStream(mFwMemoryDump.length);
203198dca3397c0049d3cfb9c97e2b021c5ee3a2e1bdxinhe                    final byte[] buf = new byte[1024];
203298dca3397c0049d3cfb9c97e2b021c5ee3a2e1bdxinhe
203398dca3397c0049d3cfb9c97e2b021c5ee3a2e1bdxinhe                    while (!compressor.finished()) {
203498dca3397c0049d3cfb9c97e2b021c5ee3a2e1bdxinhe                        int count = compressor.deflate(buf);
203598dca3397c0049d3cfb9c97e2b021c5ee3a2e1bdxinhe                         bos.write(buf, 0, count);
203698dca3397c0049d3cfb9c97e2b021c5ee3a2e1bdxinhe                    }
203798dca3397c0049d3cfb9c97e2b021c5ee3a2e1bdxinhe
203898dca3397c0049d3cfb9c97e2b021c5ee3a2e1bdxinhe                    try {
203998dca3397c0049d3cfb9c97e2b021c5ee3a2e1bdxinhe                        bos.close();
204098dca3397c0049d3cfb9c97e2b021c5ee3a2e1bdxinhe                    } catch (IOException e) {
204198dca3397c0049d3cfb9c97e2b021c5ee3a2e1bdxinhe                        Log.e(TAG, "ByteArrayOutputStream close error");
204298dca3397c0049d3cfb9c97e2b021c5ee3a2e1bdxinhe                        result =  android.util.Base64.encodeToString(mFwMemoryDump, Base64.DEFAULT);
204398dca3397c0049d3cfb9c97e2b021c5ee3a2e1bdxinhe                        return result;
204498dca3397c0049d3cfb9c97e2b021c5ee3a2e1bdxinhe                    }
204598dca3397c0049d3cfb9c97e2b021c5ee3a2e1bdxinhe
204698dca3397c0049d3cfb9c97e2b021c5ee3a2e1bdxinhe                    byte[] compressData = bos.toByteArray();
204798dca3397c0049d3cfb9c97e2b021c5ee3a2e1bdxinhe                    if(DBG) {
204898dca3397c0049d3cfb9c97e2b021c5ee3a2e1bdxinhe                        Log.d(TAG," length is:" + (compressData == null? "0" :
204998dca3397c0049d3cfb9c97e2b021c5ee3a2e1bdxinhe                                compressData.length));
205098dca3397c0049d3cfb9c97e2b021c5ee3a2e1bdxinhe                    }
205198dca3397c0049d3cfb9c97e2b021c5ee3a2e1bdxinhe                    //encode
205298dca3397c0049d3cfb9c97e2b021c5ee3a2e1bdxinhe                    result =  android.util.Base64.encodeToString(compressData.length <
205398dca3397c0049d3cfb9c97e2b021c5ee3a2e1bdxinhe                            mFwMemoryDump.length ? compressData :mFwMemoryDump , Base64.DEFAULT);
205498dca3397c0049d3cfb9c97e2b021c5ee3a2e1bdxinhe                    if(DBG) {
205598dca3397c0049d3cfb9c97e2b021c5ee3a2e1bdxinhe                        Log.d(TAG, "FwMemoryDump length is :" + result.length());
205698dca3397c0049d3cfb9c97e2b021c5ee3a2e1bdxinhe                    }
205798dca3397c0049d3cfb9c97e2b021c5ee3a2e1bdxinhe
205803ba4a5b108ab2bbdb5d0e7be46ac45c00a6d9ffxinhe                    mFwMemoryDump = null;
205998dca3397c0049d3cfb9c97e2b021c5ee3a2e1bdxinhe                    return result;
206003ba4a5b108ab2bbdb5d0e7be46ac45c00a6d9ffxinhe                } else {
206103ba4a5b108ab2bbdb5d0e7be46ac45c00a6d9ffxinhe                    return null;
206203ba4a5b108ab2bbdb5d0e7be46ac45c00a6d9ffxinhe                }
206303ba4a5b108ab2bbdb5d0e7be46ac45c00a6d9ffxinhe            } else {
206403ba4a5b108ab2bbdb5d0e7be46ac45c00a6d9ffxinhe                return null;
206503ba4a5b108ab2bbdb5d0e7be46ac45c00a6d9ffxinhe            }
2066a0d34d3626bd9631039a485d2d428b1b7de390e2Pierre Vandwalle        }
2067a0d34d3626bd9631039a485d2d428b1b7de390e2Pierre Vandwalle    }
2068dd490cf1df37b70a8625dc4ec8e712f740dd1e4aPierre Vandwalle
2069dd490cf1df37b70a8625dc4ec8e712f740dd1e4aPierre Vandwalle    //---------------------------------------------------------------------------------
2070dd490cf1df37b70a8625dc4ec8e712f740dd1e4aPierre Vandwalle    /* Configure ePNO */
2071dd490cf1df37b70a8625dc4ec8e712f740dd1e4aPierre Vandwalle
2072dd490cf1df37b70a8625dc4ec8e712f740dd1e4aPierre Vandwalle    public class WifiPnoNetwork {
2073dd490cf1df37b70a8625dc4ec8e712f740dd1e4aPierre Vandwalle        String SSID;
2074dd490cf1df37b70a8625dc4ec8e712f740dd1e4aPierre Vandwalle        int rssi_threshold;
2075dd490cf1df37b70a8625dc4ec8e712f740dd1e4aPierre Vandwalle        int flags;
2076dd490cf1df37b70a8625dc4ec8e712f740dd1e4aPierre Vandwalle        int auth;
2077dd490cf1df37b70a8625dc4ec8e712f740dd1e4aPierre Vandwalle        String configKey; // kept for reference
2078dd490cf1df37b70a8625dc4ec8e712f740dd1e4aPierre Vandwalle
2079dd490cf1df37b70a8625dc4ec8e712f740dd1e4aPierre Vandwalle        WifiPnoNetwork(WifiConfiguration config, int threshold) {
2080dd490cf1df37b70a8625dc4ec8e712f740dd1e4aPierre Vandwalle            if (config.SSID == null) {
2081dd490cf1df37b70a8625dc4ec8e712f740dd1e4aPierre Vandwalle                this.SSID = "";
2082dd490cf1df37b70a8625dc4ec8e712f740dd1e4aPierre Vandwalle                this.flags = 1;
2083dd490cf1df37b70a8625dc4ec8e712f740dd1e4aPierre Vandwalle            } else {
2084dd490cf1df37b70a8625dc4ec8e712f740dd1e4aPierre Vandwalle                this.SSID = config.SSID;
2085dd490cf1df37b70a8625dc4ec8e712f740dd1e4aPierre Vandwalle            }
2086dd490cf1df37b70a8625dc4ec8e712f740dd1e4aPierre Vandwalle            this.rssi_threshold = threshold;
2087dd490cf1df37b70a8625dc4ec8e712f740dd1e4aPierre Vandwalle            if (config.allowedKeyManagement.get(WifiConfiguration.KeyMgmt.WPA_PSK)) {
2088dd490cf1df37b70a8625dc4ec8e712f740dd1e4aPierre Vandwalle                auth |= 2;
2089dd490cf1df37b70a8625dc4ec8e712f740dd1e4aPierre Vandwalle            } else if (config.allowedKeyManagement.get(WifiConfiguration.KeyMgmt.WPA_EAP) ||
2090dd490cf1df37b70a8625dc4ec8e712f740dd1e4aPierre Vandwalle                    config.allowedKeyManagement.get(WifiConfiguration.KeyMgmt.IEEE8021X)) {
2091dd490cf1df37b70a8625dc4ec8e712f740dd1e4aPierre Vandwalle                auth |= 4;
2092dd490cf1df37b70a8625dc4ec8e712f740dd1e4aPierre Vandwalle            } else if (config.wepKeys[0] != null) {
2093dd490cf1df37b70a8625dc4ec8e712f740dd1e4aPierre Vandwalle                auth |= 1;
2094dd490cf1df37b70a8625dc4ec8e712f740dd1e4aPierre Vandwalle            } else {
2095dd490cf1df37b70a8625dc4ec8e712f740dd1e4aPierre Vandwalle                auth |= 1;
2096dd490cf1df37b70a8625dc4ec8e712f740dd1e4aPierre Vandwalle            }
2097dd490cf1df37b70a8625dc4ec8e712f740dd1e4aPierre Vandwalle//            auth = 0;
2098dd490cf1df37b70a8625dc4ec8e712f740dd1e4aPierre Vandwalle            flags |= 6; //A and G
2099dd490cf1df37b70a8625dc4ec8e712f740dd1e4aPierre Vandwalle            configKey = config.configKey();
2100dd490cf1df37b70a8625dc4ec8e712f740dd1e4aPierre Vandwalle        }
2101dd490cf1df37b70a8625dc4ec8e712f740dd1e4aPierre Vandwalle
2102dd490cf1df37b70a8625dc4ec8e712f740dd1e4aPierre Vandwalle        @Override
2103dd490cf1df37b70a8625dc4ec8e712f740dd1e4aPierre Vandwalle        public String toString() {
2104dd490cf1df37b70a8625dc4ec8e712f740dd1e4aPierre Vandwalle            StringBuilder sbuf = new StringBuilder();
2105dd490cf1df37b70a8625dc4ec8e712f740dd1e4aPierre Vandwalle            sbuf.append(this.SSID);
2106dd490cf1df37b70a8625dc4ec8e712f740dd1e4aPierre Vandwalle            sbuf.append(" flags=").append(this.flags);
2107dd490cf1df37b70a8625dc4ec8e712f740dd1e4aPierre Vandwalle            sbuf.append(" rssi=").append(this.rssi_threshold);
2108dd490cf1df37b70a8625dc4ec8e712f740dd1e4aPierre Vandwalle            sbuf.append(" auth=").append(this.auth);
2109dd490cf1df37b70a8625dc4ec8e712f740dd1e4aPierre Vandwalle            return sbuf.toString();
2110dd490cf1df37b70a8625dc4ec8e712f740dd1e4aPierre Vandwalle        }
2111dd490cf1df37b70a8625dc4ec8e712f740dd1e4aPierre Vandwalle    }
2112dd490cf1df37b70a8625dc4ec8e712f740dd1e4aPierre Vandwalle
2113dd490cf1df37b70a8625dc4ec8e712f740dd1e4aPierre Vandwalle    public static interface WifiPnoEventHandler {
2114dd490cf1df37b70a8625dc4ec8e712f740dd1e4aPierre Vandwalle        void onPnoNetworkFound(ScanResult results[]);
2115dd490cf1df37b70a8625dc4ec8e712f740dd1e4aPierre Vandwalle    }
2116dd490cf1df37b70a8625dc4ec8e712f740dd1e4aPierre Vandwalle
2117dd490cf1df37b70a8625dc4ec8e712f740dd1e4aPierre Vandwalle    private static WifiPnoEventHandler sWifiPnoEventHandler;
2118dd490cf1df37b70a8625dc4ec8e712f740dd1e4aPierre Vandwalle
2119dd490cf1df37b70a8625dc4ec8e712f740dd1e4aPierre Vandwalle    private static int sPnoCmdId = 0;
2120dd490cf1df37b70a8625dc4ec8e712f740dd1e4aPierre Vandwalle
2121dd490cf1df37b70a8625dc4ec8e712f740dd1e4aPierre Vandwalle    private native static boolean setPnoListNative(int iface, int id, WifiPnoNetwork list[]);
2122dd490cf1df37b70a8625dc4ec8e712f740dd1e4aPierre Vandwalle
2123dd490cf1df37b70a8625dc4ec8e712f740dd1e4aPierre Vandwalle    synchronized public static boolean setPnoList(WifiPnoNetwork list[],
2124dd490cf1df37b70a8625dc4ec8e712f740dd1e4aPierre Vandwalle                                                  WifiPnoEventHandler eventHandler) {
2125dd490cf1df37b70a8625dc4ec8e712f740dd1e4aPierre Vandwalle        Log.e(TAG, "setPnoList cmd " + sPnoCmdId);
2126dd490cf1df37b70a8625dc4ec8e712f740dd1e4aPierre Vandwalle
2127dd490cf1df37b70a8625dc4ec8e712f740dd1e4aPierre Vandwalle        synchronized (mLock) {
2128dd490cf1df37b70a8625dc4ec8e712f740dd1e4aPierre Vandwalle
2129dd490cf1df37b70a8625dc4ec8e712f740dd1e4aPierre Vandwalle            sPnoCmdId = getNewCmdIdLocked();
2130dd490cf1df37b70a8625dc4ec8e712f740dd1e4aPierre Vandwalle
2131dd490cf1df37b70a8625dc4ec8e712f740dd1e4aPierre Vandwalle            sWifiPnoEventHandler = eventHandler;
2132dd490cf1df37b70a8625dc4ec8e712f740dd1e4aPierre Vandwalle            if (setPnoListNative(sWlan0Index, sPnoCmdId, list) == false) {
2133dd490cf1df37b70a8625dc4ec8e712f740dd1e4aPierre Vandwalle                sWifiPnoEventHandler = null;
2134dd490cf1df37b70a8625dc4ec8e712f740dd1e4aPierre Vandwalle                return false;
2135dd490cf1df37b70a8625dc4ec8e712f740dd1e4aPierre Vandwalle            }
2136dd490cf1df37b70a8625dc4ec8e712f740dd1e4aPierre Vandwalle
2137dd490cf1df37b70a8625dc4ec8e712f740dd1e4aPierre Vandwalle            return true;
2138dd490cf1df37b70a8625dc4ec8e712f740dd1e4aPierre Vandwalle        }
2139dd490cf1df37b70a8625dc4ec8e712f740dd1e4aPierre Vandwalle    }
2140dd490cf1df37b70a8625dc4ec8e712f740dd1e4aPierre Vandwalle
2141dd490cf1df37b70a8625dc4ec8e712f740dd1e4aPierre Vandwalle    synchronized public static void onPnoNetworkFound(int id, ScanResult[] results) {
2142dd490cf1df37b70a8625dc4ec8e712f740dd1e4aPierre Vandwalle
2143dd490cf1df37b70a8625dc4ec8e712f740dd1e4aPierre Vandwalle        if (results == null) {
2144dd490cf1df37b70a8625dc4ec8e712f740dd1e4aPierre Vandwalle            Log.e(TAG, "onPnoNetworkFound null results");
2145dd490cf1df37b70a8625dc4ec8e712f740dd1e4aPierre Vandwalle            return;
2146dd490cf1df37b70a8625dc4ec8e712f740dd1e4aPierre Vandwalle
2147dd490cf1df37b70a8625dc4ec8e712f740dd1e4aPierre Vandwalle        }
2148dd490cf1df37b70a8625dc4ec8e712f740dd1e4aPierre Vandwalle        Log.d(TAG, "WifiNative.onPnoNetworkFound result " + results.length);
2149dd490cf1df37b70a8625dc4ec8e712f740dd1e4aPierre Vandwalle
2150dd490cf1df37b70a8625dc4ec8e712f740dd1e4aPierre Vandwalle        //Log.e(TAG, "onPnoNetworkFound length " + results.length);
2151dd490cf1df37b70a8625dc4ec8e712f740dd1e4aPierre Vandwalle        //return;
2152dd490cf1df37b70a8625dc4ec8e712f740dd1e4aPierre Vandwalle        for (int i=0; i<results.length; i++) {
2153dd490cf1df37b70a8625dc4ec8e712f740dd1e4aPierre Vandwalle            Log.e(TAG, "onPnoNetworkFound SSID " + results[i].SSID
2154dd490cf1df37b70a8625dc4ec8e712f740dd1e4aPierre Vandwalle                    + " " + results[i].level + " " + results[i].frequency);
2155dd490cf1df37b70a8625dc4ec8e712f740dd1e4aPierre Vandwalle
2156dd490cf1df37b70a8625dc4ec8e712f740dd1e4aPierre Vandwalle            populateScanResult(results[i], results[i].bytes, "onPnoNetworkFound ");
2157dd490cf1df37b70a8625dc4ec8e712f740dd1e4aPierre Vandwalle            results[i].wifiSsid = WifiSsid.createFromAsciiEncoded(results[i].SSID);
2158dd490cf1df37b70a8625dc4ec8e712f740dd1e4aPierre Vandwalle        }
2159dd490cf1df37b70a8625dc4ec8e712f740dd1e4aPierre Vandwalle        synchronized (mLock) {
2160dd490cf1df37b70a8625dc4ec8e712f740dd1e4aPierre Vandwalle            if (sPnoCmdId != 0 && sWifiPnoEventHandler != null) {
2161dd490cf1df37b70a8625dc4ec8e712f740dd1e4aPierre Vandwalle                sWifiPnoEventHandler.onPnoNetworkFound(results);
2162dd490cf1df37b70a8625dc4ec8e712f740dd1e4aPierre Vandwalle            } else {
2163dd490cf1df37b70a8625dc4ec8e712f740dd1e4aPierre Vandwalle                /* this can happen because of race conditions */
2164dd490cf1df37b70a8625dc4ec8e712f740dd1e4aPierre Vandwalle                Log.d(TAG, "Ignoring Pno Network found event");
2165dd490cf1df37b70a8625dc4ec8e712f740dd1e4aPierre Vandwalle            }
2166dd490cf1df37b70a8625dc4ec8e712f740dd1e4aPierre Vandwalle        }
2167d4c25fd76644d5490700ae69fada4669de5193a1Pierre Vandwalle    }
2168d4c25fd76644d5490700ae69fada4669de5193a1Pierre Vandwalle
2169d4c25fd76644d5490700ae69fada4669de5193a1Pierre Vandwalle    public class WifiLazyRoamParams {
2170d4c25fd76644d5490700ae69fada4669de5193a1Pierre Vandwalle        int A_band_boost_threshold;
2171d4c25fd76644d5490700ae69fada4669de5193a1Pierre Vandwalle        int A_band_penalty_threshold;
2172d4c25fd76644d5490700ae69fada4669de5193a1Pierre Vandwalle        int A_band_boost_factor;
2173d4c25fd76644d5490700ae69fada4669de5193a1Pierre Vandwalle        int A_band_penalty_factor;
2174d4c25fd76644d5490700ae69fada4669de5193a1Pierre Vandwalle        int A_band_max_boost;
2175d4c25fd76644d5490700ae69fada4669de5193a1Pierre Vandwalle        int lazy_roam_hysteresis;
2176d4c25fd76644d5490700ae69fada4669de5193a1Pierre Vandwalle        int alert_roam_rssi_trigger;
2177dd490cf1df37b70a8625dc4ec8e712f740dd1e4aPierre Vandwalle
2178d4c25fd76644d5490700ae69fada4669de5193a1Pierre Vandwalle        WifiLazyRoamParams() {
2179d4c25fd76644d5490700ae69fada4669de5193a1Pierre Vandwalle        }
2180d4c25fd76644d5490700ae69fada4669de5193a1Pierre Vandwalle
2181d4c25fd76644d5490700ae69fada4669de5193a1Pierre Vandwalle        @Override
2182d4c25fd76644d5490700ae69fada4669de5193a1Pierre Vandwalle        public String toString() {
2183d4c25fd76644d5490700ae69fada4669de5193a1Pierre Vandwalle            StringBuilder sbuf = new StringBuilder();
2184d4c25fd76644d5490700ae69fada4669de5193a1Pierre Vandwalle            sbuf.append(" A_band_boost_threshold=").append(this.A_band_boost_threshold);
2185d4c25fd76644d5490700ae69fada4669de5193a1Pierre Vandwalle            sbuf.append(" A_band_penalty_threshold=").append(this.A_band_penalty_threshold);
2186d4c25fd76644d5490700ae69fada4669de5193a1Pierre Vandwalle            sbuf.append(" A_band_boost_factor=").append(this.A_band_boost_factor);
2187d4c25fd76644d5490700ae69fada4669de5193a1Pierre Vandwalle            sbuf.append(" A_band_penalty_factor=").append(this.A_band_penalty_factor);
2188d4c25fd76644d5490700ae69fada4669de5193a1Pierre Vandwalle            sbuf.append(" A_band_max_boost=").append(this.A_band_max_boost);
2189d4c25fd76644d5490700ae69fada4669de5193a1Pierre Vandwalle            sbuf.append(" lazy_roam_hysteresis=").append(this.lazy_roam_hysteresis);
2190d4c25fd76644d5490700ae69fada4669de5193a1Pierre Vandwalle            sbuf.append(" alert_roam_rssi_trigger=").append(this.alert_roam_rssi_trigger);
2191d4c25fd76644d5490700ae69fada4669de5193a1Pierre Vandwalle            return sbuf.toString();
2192d4c25fd76644d5490700ae69fada4669de5193a1Pierre Vandwalle        }
2193d4c25fd76644d5490700ae69fada4669de5193a1Pierre Vandwalle    }
2194d4c25fd76644d5490700ae69fada4669de5193a1Pierre Vandwalle
21959ccffbb6c3fe8eaa7d70cc3076d0dc3c6a9073e7Pierre Vandwalle    private native static boolean setLazyRoamNative(int iface, int id,
2196d4c25fd76644d5490700ae69fada4669de5193a1Pierre Vandwalle                                              boolean enabled, WifiLazyRoamParams param);
2197d4c25fd76644d5490700ae69fada4669de5193a1Pierre Vandwalle
2198d4c25fd76644d5490700ae69fada4669de5193a1Pierre Vandwalle    synchronized public static boolean setLazyRoam(boolean enabled, WifiLazyRoamParams params) {
2199d4c25fd76644d5490700ae69fada4669de5193a1Pierre Vandwalle        synchronized (mLock) {
2200d4c25fd76644d5490700ae69fada4669de5193a1Pierre Vandwalle            if (startHal()) {
2201d4c25fd76644d5490700ae69fada4669de5193a1Pierre Vandwalle                sPnoCmdId = getNewCmdIdLocked();
2202d4c25fd76644d5490700ae69fada4669de5193a1Pierre Vandwalle
22039ccffbb6c3fe8eaa7d70cc3076d0dc3c6a9073e7Pierre Vandwalle                return setLazyRoamNative(sWlan0Index, sPnoCmdId, enabled, params);
2204d4c25fd76644d5490700ae69fada4669de5193a1Pierre Vandwalle            } else {
2205d4c25fd76644d5490700ae69fada4669de5193a1Pierre Vandwalle                return false;
2206d4c25fd76644d5490700ae69fada4669de5193a1Pierre Vandwalle            }
2207d4c25fd76644d5490700ae69fada4669de5193a1Pierre Vandwalle        }
2208d4c25fd76644d5490700ae69fada4669de5193a1Pierre Vandwalle    }
22099ccffbb6c3fe8eaa7d70cc3076d0dc3c6a9073e7Pierre Vandwalle
22109ccffbb6c3fe8eaa7d70cc3076d0dc3c6a9073e7Pierre Vandwalle    private native static boolean setBssidBlacklistNative(int iface, int id,
22119ccffbb6c3fe8eaa7d70cc3076d0dc3c6a9073e7Pierre Vandwalle                                              String list[]);
22129ccffbb6c3fe8eaa7d70cc3076d0dc3c6a9073e7Pierre Vandwalle
22139ccffbb6c3fe8eaa7d70cc3076d0dc3c6a9073e7Pierre Vandwalle    synchronized public static boolean setBssidBlacklist(String list[]) {
22149ccffbb6c3fe8eaa7d70cc3076d0dc3c6a9073e7Pierre Vandwalle        int size = 0;
22159ccffbb6c3fe8eaa7d70cc3076d0dc3c6a9073e7Pierre Vandwalle        if (list != null) {
22169ccffbb6c3fe8eaa7d70cc3076d0dc3c6a9073e7Pierre Vandwalle            size = list.length;
22179ccffbb6c3fe8eaa7d70cc3076d0dc3c6a9073e7Pierre Vandwalle        }
22189ccffbb6c3fe8eaa7d70cc3076d0dc3c6a9073e7Pierre Vandwalle        Log.e(TAG, "setBssidBlacklist cmd " + sPnoCmdId + " size " + size);
22199ccffbb6c3fe8eaa7d70cc3076d0dc3c6a9073e7Pierre Vandwalle
22209ccffbb6c3fe8eaa7d70cc3076d0dc3c6a9073e7Pierre Vandwalle        synchronized (mLock) {
22219ccffbb6c3fe8eaa7d70cc3076d0dc3c6a9073e7Pierre Vandwalle            sPnoCmdId = getNewCmdIdLocked();
22229ccffbb6c3fe8eaa7d70cc3076d0dc3c6a9073e7Pierre Vandwalle
22239ccffbb6c3fe8eaa7d70cc3076d0dc3c6a9073e7Pierre Vandwalle            if (setBssidBlacklistNative(sWlan0Index, sPnoCmdId, list) == false) {
22249ccffbb6c3fe8eaa7d70cc3076d0dc3c6a9073e7Pierre Vandwalle                return false;
22259ccffbb6c3fe8eaa7d70cc3076d0dc3c6a9073e7Pierre Vandwalle            }
22269ccffbb6c3fe8eaa7d70cc3076d0dc3c6a9073e7Pierre Vandwalle            return true;
22279ccffbb6c3fe8eaa7d70cc3076d0dc3c6a9073e7Pierre Vandwalle        }
22289ccffbb6c3fe8eaa7d70cc3076d0dc3c6a9073e7Pierre Vandwalle    }
22299ccffbb6c3fe8eaa7d70cc3076d0dc3c6a9073e7Pierre Vandwalle
22305caa43b34e0445ff08bff5931b84ffbc850b2a66Pierre Vandwalle    private native static boolean setSsidWhitelistNative(int iface, int id, String list[]);
22315caa43b34e0445ff08bff5931b84ffbc850b2a66Pierre Vandwalle
22325caa43b34e0445ff08bff5931b84ffbc850b2a66Pierre Vandwalle    synchronized public static boolean setSsidWhitelist(String list[]) {
22335caa43b34e0445ff08bff5931b84ffbc850b2a66Pierre Vandwalle        int size = 0;
22345caa43b34e0445ff08bff5931b84ffbc850b2a66Pierre Vandwalle        if (list != null) {
22355caa43b34e0445ff08bff5931b84ffbc850b2a66Pierre Vandwalle            size = list.length;
22365caa43b34e0445ff08bff5931b84ffbc850b2a66Pierre Vandwalle        }
22375caa43b34e0445ff08bff5931b84ffbc850b2a66Pierre Vandwalle        Log.e(TAG, "setSsidWhitelist cmd " + sPnoCmdId + " size " + size);
22385caa43b34e0445ff08bff5931b84ffbc850b2a66Pierre Vandwalle
22395caa43b34e0445ff08bff5931b84ffbc850b2a66Pierre Vandwalle        synchronized (mLock) {
22405caa43b34e0445ff08bff5931b84ffbc850b2a66Pierre Vandwalle            sPnoCmdId = getNewCmdIdLocked();
22415caa43b34e0445ff08bff5931b84ffbc850b2a66Pierre Vandwalle
22425caa43b34e0445ff08bff5931b84ffbc850b2a66Pierre Vandwalle            if (setSsidWhitelistNative(sWlan0Index, sPnoCmdId, list) == false) {
22435caa43b34e0445ff08bff5931b84ffbc850b2a66Pierre Vandwalle                return false;
22445caa43b34e0445ff08bff5931b84ffbc850b2a66Pierre Vandwalle            }
22455caa43b34e0445ff08bff5931b84ffbc850b2a66Pierre Vandwalle            return true;
22465caa43b34e0445ff08bff5931b84ffbc850b2a66Pierre Vandwalle        }
22475caa43b34e0445ff08bff5931b84ffbc850b2a66Pierre Vandwalle    }
22485caa43b34e0445ff08bff5931b84ffbc850b2a66Pierre Vandwalle
2249155b9d09ef9b8ead3ca617afdd91e74070d3f0cbVinit Deshpande}
2250