WificondPnoScannerTest.java revision e7ba2963bedf426a1d8ba09ab535260ef364512b
162bc101940ae1f5e60c4d8861a149b900dbf5e5cRoshan Pius/*
262bc101940ae1f5e60c4d8861a149b900dbf5e5cRoshan Pius * Copyright (C) 2016 The Android Open Source Project
362bc101940ae1f5e60c4d8861a149b900dbf5e5cRoshan Pius *
462bc101940ae1f5e60c4d8861a149b900dbf5e5cRoshan Pius * Licensed under the Apache License, Version 2.0 (the "License");
562bc101940ae1f5e60c4d8861a149b900dbf5e5cRoshan Pius * you may not use this file except in compliance with the License.
662bc101940ae1f5e60c4d8861a149b900dbf5e5cRoshan Pius * You may obtain a copy of the License at
762bc101940ae1f5e60c4d8861a149b900dbf5e5cRoshan Pius *
862bc101940ae1f5e60c4d8861a149b900dbf5e5cRoshan Pius *      http://www.apache.org/licenses/LICENSE-2.0
962bc101940ae1f5e60c4d8861a149b900dbf5e5cRoshan Pius *
1062bc101940ae1f5e60c4d8861a149b900dbf5e5cRoshan Pius * Unless required by applicable law or agreed to in writing, software
1162bc101940ae1f5e60c4d8861a149b900dbf5e5cRoshan Pius * distributed under the License is distributed on an "AS IS" BASIS,
1262bc101940ae1f5e60c4d8861a149b900dbf5e5cRoshan Pius * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
1362bc101940ae1f5e60c4d8861a149b900dbf5e5cRoshan Pius * See the License for the specific language governing permissions and
1462bc101940ae1f5e60c4d8861a149b900dbf5e5cRoshan Pius * limitations under the License
1562bc101940ae1f5e60c4d8861a149b900dbf5e5cRoshan Pius */
1662bc101940ae1f5e60c4d8861a149b900dbf5e5cRoshan Pius
1762bc101940ae1f5e60c4d8861a149b900dbf5e5cRoshan Piuspackage com.android.server.wifi;
1862bc101940ae1f5e60c4d8861a149b900dbf5e5cRoshan Pius
19e7ba2963bedf426a1d8ba09ab535260ef364512bRoshan Piusimport static com.android.server.wifi.ScanTestUtil.NativeScanSettingsBuilder;
20e7ba2963bedf426a1d8ba09ab535260ef364512bRoshan Piusimport static com.android.server.wifi.ScanTestUtil.assertScanDataEquals;
21e7ba2963bedf426a1d8ba09ab535260ef364512bRoshan Pius
22e7ba2963bedf426a1d8ba09ab535260ef364512bRoshan Piusimport static org.junit.Assert.*;
23e7ba2963bedf426a1d8ba09ab535260ef364512bRoshan Piusimport static org.mockito.Mockito.*;
24e7ba2963bedf426a1d8ba09ab535260ef364512bRoshan Pius
2562bc101940ae1f5e60c4d8861a149b900dbf5e5cRoshan Piusimport android.content.Context;
2662bc101940ae1f5e60c4d8861a149b900dbf5e5cRoshan Piusimport android.net.wifi.WifiConfiguration;
2762bc101940ae1f5e60c4d8861a149b900dbf5e5cRoshan Piusimport android.net.wifi.WifiScanner;
2862bc101940ae1f5e60c4d8861a149b900dbf5e5cRoshan Piusimport android.test.suitebuilder.annotation.SmallTest;
2962bc101940ae1f5e60c4d8861a149b900dbf5e5cRoshan Pius
3062bc101940ae1f5e60c4d8861a149b900dbf5e5cRoshan Piusimport com.android.internal.R;
3162bc101940ae1f5e60c4d8861a149b900dbf5e5cRoshan Piusimport com.android.server.wifi.scanner.ChannelHelper.ChannelCollection;
3262bc101940ae1f5e60c4d8861a149b900dbf5e5cRoshan Pius
3362bc101940ae1f5e60c4d8861a149b900dbf5e5cRoshan Piusimport org.junit.Before;
3462bc101940ae1f5e60c4d8861a149b900dbf5e5cRoshan Piusimport org.junit.Test;
3562bc101940ae1f5e60c4d8861a149b900dbf5e5cRoshan Piusimport org.mockito.InOrder;
3662bc101940ae1f5e60c4d8861a149b900dbf5e5cRoshan Piusimport org.mockito.Mock;
3762bc101940ae1f5e60c4d8861a149b900dbf5e5cRoshan Piusimport org.mockito.MockitoAnnotations;
3862bc101940ae1f5e60c4d8861a149b900dbf5e5cRoshan Pius
3962bc101940ae1f5e60c4d8861a149b900dbf5e5cRoshan Piusimport java.util.HashSet;
4062bc101940ae1f5e60c4d8861a149b900dbf5e5cRoshan Piusimport java.util.Set;
4162bc101940ae1f5e60c4d8861a149b900dbf5e5cRoshan Pius
4262bc101940ae1f5e60c4d8861a149b900dbf5e5cRoshan Pius
4362bc101940ae1f5e60c4d8861a149b900dbf5e5cRoshan Pius/**
4462bc101940ae1f5e60c4d8861a149b900dbf5e5cRoshan Pius * Unit tests for {@link com.android.server.wifi.SupplicantWifiScannerImpl.setPnoList}.
4562bc101940ae1f5e60c4d8861a149b900dbf5e5cRoshan Pius */
4662bc101940ae1f5e60c4d8861a149b900dbf5e5cRoshan Pius@SmallTest
4762bc101940ae1f5e60c4d8861a149b900dbf5e5cRoshan Piuspublic class SupplicantPnoScannerTest {
4862bc101940ae1f5e60c4d8861a149b900dbf5e5cRoshan Pius
4962bc101940ae1f5e60c4d8861a149b900dbf5e5cRoshan Pius    @Mock Context mContext;
5062bc101940ae1f5e60c4d8861a149b900dbf5e5cRoshan Pius    MockAlarmManager mAlarmManager;
5162bc101940ae1f5e60c4d8861a149b900dbf5e5cRoshan Pius    MockWifiMonitor mWifiMonitor;
5262bc101940ae1f5e60c4d8861a149b900dbf5e5cRoshan Pius    MockLooper mLooper;
5362bc101940ae1f5e60c4d8861a149b900dbf5e5cRoshan Pius    @Mock WifiNative mWifiNative;
5462bc101940ae1f5e60c4d8861a149b900dbf5e5cRoshan Pius    MockResources mResources;
5562bc101940ae1f5e60c4d8861a149b900dbf5e5cRoshan Pius    SupplicantWifiScannerImpl mScanner;
5662bc101940ae1f5e60c4d8861a149b900dbf5e5cRoshan Pius
5762bc101940ae1f5e60c4d8861a149b900dbf5e5cRoshan Pius    @Before
5862bc101940ae1f5e60c4d8861a149b900dbf5e5cRoshan Pius    public void setup() throws Exception {
5962bc101940ae1f5e60c4d8861a149b900dbf5e5cRoshan Pius        MockitoAnnotations.initMocks(this);
6062bc101940ae1f5e60c4d8861a149b900dbf5e5cRoshan Pius
6162bc101940ae1f5e60c4d8861a149b900dbf5e5cRoshan Pius        mLooper = new MockLooper();
6262bc101940ae1f5e60c4d8861a149b900dbf5e5cRoshan Pius        mAlarmManager = new MockAlarmManager();
6362bc101940ae1f5e60c4d8861a149b900dbf5e5cRoshan Pius        mWifiMonitor = new MockWifiMonitor();
6462bc101940ae1f5e60c4d8861a149b900dbf5e5cRoshan Pius        mResources = new MockResources();
6562bc101940ae1f5e60c4d8861a149b900dbf5e5cRoshan Pius
6662bc101940ae1f5e60c4d8861a149b900dbf5e5cRoshan Pius        when(mWifiNative.getInterfaceName()).thenReturn("a_test_interface_name");
6762bc101940ae1f5e60c4d8861a149b900dbf5e5cRoshan Pius        when(mContext.getSystemService(Context.ALARM_SERVICE))
6862bc101940ae1f5e60c4d8861a149b900dbf5e5cRoshan Pius                .thenReturn(mAlarmManager.getAlarmManager());
6962bc101940ae1f5e60c4d8861a149b900dbf5e5cRoshan Pius        when(mContext.getResources()).thenReturn(mResources);
7062bc101940ae1f5e60c4d8861a149b900dbf5e5cRoshan Pius    }
7162bc101940ae1f5e60c4d8861a149b900dbf5e5cRoshan Pius
7262bc101940ae1f5e60c4d8861a149b900dbf5e5cRoshan Pius    /**
736c5018cef1eb7acbcfa7fc6c9b7c018bab7ba7baRoshan Pius     * Verify that the HW disconnected PNO scan triggers a supplicant PNO scan and invokes the
746c5018cef1eb7acbcfa7fc6c9b7c018bab7ba7baRoshan Pius     * OnPnoNetworkFound callback when the scan results are received.
7562bc101940ae1f5e60c4d8861a149b900dbf5e5cRoshan Pius     */
7662bc101940ae1f5e60c4d8861a149b900dbf5e5cRoshan Pius    @Test
77dcd877d6c143db557884993ea437e2a432cb0ba3Roshan Pius    public void startHwDisconnectedPnoScan() {
7862bc101940ae1f5e60c4d8861a149b900dbf5e5cRoshan Pius        createScannerWithHwPnoScanSupport();
7962bc101940ae1f5e60c4d8861a149b900dbf5e5cRoshan Pius
8062bc101940ae1f5e60c4d8861a149b900dbf5e5cRoshan Pius        WifiNative.PnoEventHandler pnoEventHandler = mock(WifiNative.PnoEventHandler.class);
81dcd877d6c143db557884993ea437e2a432cb0ba3Roshan Pius        WifiNative.PnoSettings pnoSettings = createDummyPnoSettings(false);
8262bc101940ae1f5e60c4d8861a149b900dbf5e5cRoshan Pius        ScanResults scanResults = createDummyScanResults();
8362bc101940ae1f5e60c4d8861a149b900dbf5e5cRoshan Pius
8462bc101940ae1f5e60c4d8861a149b900dbf5e5cRoshan Pius        InOrder order = inOrder(pnoEventHandler, mWifiNative);
8562bc101940ae1f5e60c4d8861a149b900dbf5e5cRoshan Pius        // Start PNO scan
8662bc101940ae1f5e60c4d8861a149b900dbf5e5cRoshan Pius        startSuccessfulPnoScan(null, pnoSettings, null, pnoEventHandler);
87dcd877d6c143db557884993ea437e2a432cb0ba3Roshan Pius        expectSuccessfulHwDisconnectedPnoScan(order, pnoSettings, pnoEventHandler, scanResults);
8862bc101940ae1f5e60c4d8861a149b900dbf5e5cRoshan Pius        verifyNoMoreInteractions(pnoEventHandler);
8962bc101940ae1f5e60c4d8861a149b900dbf5e5cRoshan Pius    }
9062bc101940ae1f5e60c4d8861a149b900dbf5e5cRoshan Pius
9162bc101940ae1f5e60c4d8861a149b900dbf5e5cRoshan Pius    /**
926c5018cef1eb7acbcfa7fc6c9b7c018bab7ba7baRoshan Pius     * Verify that we pause & resume HW PNO scan when a single scan is scheduled and invokes the
936c5018cef1eb7acbcfa7fc6c9b7c018bab7ba7baRoshan Pius     * OnPnoNetworkFound callback when the scan results are received.
9462bc101940ae1f5e60c4d8861a149b900dbf5e5cRoshan Pius     */
9562bc101940ae1f5e60c4d8861a149b900dbf5e5cRoshan Pius    @Test
96dcd877d6c143db557884993ea437e2a432cb0ba3Roshan Pius    public void pauseResumeHwDisconnectedPnoScanForSingleScan() {
9762bc101940ae1f5e60c4d8861a149b900dbf5e5cRoshan Pius        createScannerWithHwPnoScanSupport();
9862bc101940ae1f5e60c4d8861a149b900dbf5e5cRoshan Pius
9962bc101940ae1f5e60c4d8861a149b900dbf5e5cRoshan Pius        WifiNative.PnoEventHandler pnoEventHandler = mock(WifiNative.PnoEventHandler.class);
100dcd877d6c143db557884993ea437e2a432cb0ba3Roshan Pius        WifiNative.PnoSettings pnoSettings = createDummyPnoSettings(false);
10162bc101940ae1f5e60c4d8861a149b900dbf5e5cRoshan Pius        WifiNative.ScanEventHandler eventHandler = mock(WifiNative.ScanEventHandler.class);
10262bc101940ae1f5e60c4d8861a149b900dbf5e5cRoshan Pius        WifiNative.ScanSettings settings = createDummyScanSettings();
10362bc101940ae1f5e60c4d8861a149b900dbf5e5cRoshan Pius        ScanResults scanResults = createDummyScanResults();
10462bc101940ae1f5e60c4d8861a149b900dbf5e5cRoshan Pius
10562bc101940ae1f5e60c4d8861a149b900dbf5e5cRoshan Pius        InOrder order = inOrder(eventHandler, mWifiNative);
10662bc101940ae1f5e60c4d8861a149b900dbf5e5cRoshan Pius        // Start PNO scan
10762bc101940ae1f5e60c4d8861a149b900dbf5e5cRoshan Pius        startSuccessfulPnoScan(null, pnoSettings, null, pnoEventHandler);
10862bc101940ae1f5e60c4d8861a149b900dbf5e5cRoshan Pius        // Start single scan
10962bc101940ae1f5e60c4d8861a149b900dbf5e5cRoshan Pius        assertTrue(mScanner.startSingleScan(settings, eventHandler));
11062bc101940ae1f5e60c4d8861a149b900dbf5e5cRoshan Pius        // Verify that the PNO scan was paused and single scan runs successfully
11162bc101940ae1f5e60c4d8861a149b900dbf5e5cRoshan Pius        expectSuccessfulSingleScanWithHwPnoEnabled(order, eventHandler,
11262bc101940ae1f5e60c4d8861a149b900dbf5e5cRoshan Pius                expectedBandScanFreqs(WifiScanner.WIFI_BAND_24_GHZ), new HashSet<Integer>(),
11362bc101940ae1f5e60c4d8861a149b900dbf5e5cRoshan Pius                scanResults);
11462bc101940ae1f5e60c4d8861a149b900dbf5e5cRoshan Pius        verifyNoMoreInteractions(eventHandler);
11562bc101940ae1f5e60c4d8861a149b900dbf5e5cRoshan Pius
11662bc101940ae1f5e60c4d8861a149b900dbf5e5cRoshan Pius        order = inOrder(pnoEventHandler, mWifiNative);
117e7ba2963bedf426a1d8ba09ab535260ef364512bRoshan Pius        // Resume PNO scan after the single scan results are received and PNO monitor debounce
118e7ba2963bedf426a1d8ba09ab535260ef364512bRoshan Pius        // alarm fires.
119e7ba2963bedf426a1d8ba09ab535260ef364512bRoshan Pius        assertTrue("dispatch pno monitor alarm",
120e7ba2963bedf426a1d8ba09ab535260ef364512bRoshan Pius                mAlarmManager.dispatch(
121e7ba2963bedf426a1d8ba09ab535260ef364512bRoshan Pius                        SupplicantWifiScannerImpl.HwPnoDebouncer.PNO_DEBOUNCER_ALARM_TAG));
122e7ba2963bedf426a1d8ba09ab535260ef364512bRoshan Pius        assertEquals("dispatch message after alarm", 1, mLooper.dispatchAll());
12362bc101940ae1f5e60c4d8861a149b900dbf5e5cRoshan Pius        // Now verify that PNO scan is resumed successfully
124dcd877d6c143db557884993ea437e2a432cb0ba3Roshan Pius        expectSuccessfulHwDisconnectedPnoScan(order, pnoSettings, pnoEventHandler, scanResults);
12562bc101940ae1f5e60c4d8861a149b900dbf5e5cRoshan Pius        verifyNoMoreInteractions(pnoEventHandler);
12662bc101940ae1f5e60c4d8861a149b900dbf5e5cRoshan Pius    }
12762bc101940ae1f5e60c4d8861a149b900dbf5e5cRoshan Pius
12862bc101940ae1f5e60c4d8861a149b900dbf5e5cRoshan Pius    /**
1296c5018cef1eb7acbcfa7fc6c9b7c018bab7ba7baRoshan Pius     * Verify that the SW disconnected PNO scan triggers a background scan and invokes the
1306c5018cef1eb7acbcfa7fc6c9b7c018bab7ba7baRoshan Pius     * background scan callbacks when scan results are received.
13162bc101940ae1f5e60c4d8861a149b900dbf5e5cRoshan Pius     */
13262bc101940ae1f5e60c4d8861a149b900dbf5e5cRoshan Pius    @Test
133dcd877d6c143db557884993ea437e2a432cb0ba3Roshan Pius    public void startSwDisconnectedPnoScan() {
13462bc101940ae1f5e60c4d8861a149b900dbf5e5cRoshan Pius        createScannerWithSwPnoScanSupport();
1356c5018cef1eb7acbcfa7fc6c9b7c018bab7ba7baRoshan Pius        doSuccessfulSwPnoScanTest(false);
136dcd877d6c143db557884993ea437e2a432cb0ba3Roshan Pius    }
137dcd877d6c143db557884993ea437e2a432cb0ba3Roshan Pius
138dcd877d6c143db557884993ea437e2a432cb0ba3Roshan Pius    /**
1396c5018cef1eb7acbcfa7fc6c9b7c018bab7ba7baRoshan Pius     * Verify that the HW connected PNO scan triggers a background scan and invokes the
1406c5018cef1eb7acbcfa7fc6c9b7c018bab7ba7baRoshan Pius     * background scan callbacks when scan results are received.
141dcd877d6c143db557884993ea437e2a432cb0ba3Roshan Pius     */
142dcd877d6c143db557884993ea437e2a432cb0ba3Roshan Pius    @Test
143dcd877d6c143db557884993ea437e2a432cb0ba3Roshan Pius    public void startHwConnectedPnoScan() {
144dcd877d6c143db557884993ea437e2a432cb0ba3Roshan Pius        createScannerWithHwPnoScanSupport();
1456c5018cef1eb7acbcfa7fc6c9b7c018bab7ba7baRoshan Pius        doSuccessfulSwPnoScanTest(true);
146dcd877d6c143db557884993ea437e2a432cb0ba3Roshan Pius    }
147dcd877d6c143db557884993ea437e2a432cb0ba3Roshan Pius
148dcd877d6c143db557884993ea437e2a432cb0ba3Roshan Pius    /**
1496c5018cef1eb7acbcfa7fc6c9b7c018bab7ba7baRoshan Pius     * Verify that the SW connected PNO scan triggers a background scan and invokes the
1506c5018cef1eb7acbcfa7fc6c9b7c018bab7ba7baRoshan Pius     * background scan callbacks when scan results are received.
151dcd877d6c143db557884993ea437e2a432cb0ba3Roshan Pius     */
152dcd877d6c143db557884993ea437e2a432cb0ba3Roshan Pius    @Test
153dcd877d6c143db557884993ea437e2a432cb0ba3Roshan Pius    public void startSwConnectedPnoScan() {
154dcd877d6c143db557884993ea437e2a432cb0ba3Roshan Pius        createScannerWithSwPnoScanSupport();
1556c5018cef1eb7acbcfa7fc6c9b7c018bab7ba7baRoshan Pius        doSuccessfulSwPnoScanTest(true);
1566c5018cef1eb7acbcfa7fc6c9b7c018bab7ba7baRoshan Pius    }
157dcd877d6c143db557884993ea437e2a432cb0ba3Roshan Pius
158e7ba2963bedf426a1d8ba09ab535260ef364512bRoshan Pius    /**
159e7ba2963bedf426a1d8ba09ab535260ef364512bRoshan Pius     * Verify that the HW PNO delayed failure cleans up the scan settings cleanly.
160e7ba2963bedf426a1d8ba09ab535260ef364512bRoshan Pius     * 1. Start Hw PNO.
161e7ba2963bedf426a1d8ba09ab535260ef364512bRoshan Pius     * 2. Start Single Scan which should pause PNO scan.
162e7ba2963bedf426a1d8ba09ab535260ef364512bRoshan Pius     * 3. Fail the PNO scan resume and verify that the OnPnoScanFailed callback is invoked.
163e7ba2963bedf426a1d8ba09ab535260ef364512bRoshan Pius     * 4. Now restart a new PNO scan to ensure that the failure was cleanly handled.
164e7ba2963bedf426a1d8ba09ab535260ef364512bRoshan Pius     */
165e7ba2963bedf426a1d8ba09ab535260ef364512bRoshan Pius    @Test
166e7ba2963bedf426a1d8ba09ab535260ef364512bRoshan Pius    public void delayedHwDisconnectedPnoScanFailure() {
167e7ba2963bedf426a1d8ba09ab535260ef364512bRoshan Pius        createScannerWithHwPnoScanSupport();
168e7ba2963bedf426a1d8ba09ab535260ef364512bRoshan Pius
169e7ba2963bedf426a1d8ba09ab535260ef364512bRoshan Pius        WifiNative.PnoEventHandler pnoEventHandler = mock(WifiNative.PnoEventHandler.class);
170e7ba2963bedf426a1d8ba09ab535260ef364512bRoshan Pius        WifiNative.PnoSettings pnoSettings = createDummyPnoSettings(false);
171e7ba2963bedf426a1d8ba09ab535260ef364512bRoshan Pius        WifiNative.ScanEventHandler eventHandler = mock(WifiNative.ScanEventHandler.class);
172e7ba2963bedf426a1d8ba09ab535260ef364512bRoshan Pius        WifiNative.ScanSettings settings = createDummyScanSettings();
173e7ba2963bedf426a1d8ba09ab535260ef364512bRoshan Pius        ScanResults scanResults = createDummyScanResults();
174e7ba2963bedf426a1d8ba09ab535260ef364512bRoshan Pius
175e7ba2963bedf426a1d8ba09ab535260ef364512bRoshan Pius        InOrder order = inOrder(eventHandler, mWifiNative);
176e7ba2963bedf426a1d8ba09ab535260ef364512bRoshan Pius        // Start PNO scan
177e7ba2963bedf426a1d8ba09ab535260ef364512bRoshan Pius        startSuccessfulPnoScan(null, pnoSettings, null, pnoEventHandler);
178e7ba2963bedf426a1d8ba09ab535260ef364512bRoshan Pius        // Start single scan
179e7ba2963bedf426a1d8ba09ab535260ef364512bRoshan Pius        assertTrue(mScanner.startSingleScan(settings, eventHandler));
180e7ba2963bedf426a1d8ba09ab535260ef364512bRoshan Pius        // Verify that the PNO scan was paused and single scan runs successfully
181e7ba2963bedf426a1d8ba09ab535260ef364512bRoshan Pius        expectSuccessfulSingleScanWithHwPnoEnabled(order, eventHandler,
182e7ba2963bedf426a1d8ba09ab535260ef364512bRoshan Pius                expectedBandScanFreqs(WifiScanner.WIFI_BAND_24_GHZ), new HashSet<Integer>(),
183e7ba2963bedf426a1d8ba09ab535260ef364512bRoshan Pius                scanResults);
184e7ba2963bedf426a1d8ba09ab535260ef364512bRoshan Pius        verifyNoMoreInteractions(eventHandler);
185e7ba2963bedf426a1d8ba09ab535260ef364512bRoshan Pius
186e7ba2963bedf426a1d8ba09ab535260ef364512bRoshan Pius        // Fail the PNO resume and check that the OnPnoScanFailed callback is invoked.
187e7ba2963bedf426a1d8ba09ab535260ef364512bRoshan Pius        order = inOrder(pnoEventHandler, mWifiNative);
188e7ba2963bedf426a1d8ba09ab535260ef364512bRoshan Pius        when(mWifiNative.setPnoScan(true)).thenReturn(false);
189e7ba2963bedf426a1d8ba09ab535260ef364512bRoshan Pius        assertTrue("dispatch pno monitor alarm",
190e7ba2963bedf426a1d8ba09ab535260ef364512bRoshan Pius                mAlarmManager.dispatch(
191e7ba2963bedf426a1d8ba09ab535260ef364512bRoshan Pius                        SupplicantWifiScannerImpl.HwPnoDebouncer.PNO_DEBOUNCER_ALARM_TAG));
192e7ba2963bedf426a1d8ba09ab535260ef364512bRoshan Pius        assertEquals("dispatch message after alarm", 1, mLooper.dispatchAll());
193e7ba2963bedf426a1d8ba09ab535260ef364512bRoshan Pius        order.verify(pnoEventHandler).onPnoScanFailed();
194e7ba2963bedf426a1d8ba09ab535260ef364512bRoshan Pius        verifyNoMoreInteractions(pnoEventHandler);
195e7ba2963bedf426a1d8ba09ab535260ef364512bRoshan Pius
196e7ba2963bedf426a1d8ba09ab535260ef364512bRoshan Pius        // Add a new PNO scan request
197e7ba2963bedf426a1d8ba09ab535260ef364512bRoshan Pius        startSuccessfulPnoScan(null, pnoSettings, null, pnoEventHandler);
198e7ba2963bedf426a1d8ba09ab535260ef364512bRoshan Pius        assertTrue("dispatch pno monitor alarm",
199e7ba2963bedf426a1d8ba09ab535260ef364512bRoshan Pius                mAlarmManager.dispatch(
200e7ba2963bedf426a1d8ba09ab535260ef364512bRoshan Pius                        SupplicantWifiScannerImpl.HwPnoDebouncer.PNO_DEBOUNCER_ALARM_TAG));
201e7ba2963bedf426a1d8ba09ab535260ef364512bRoshan Pius        assertEquals("dispatch message after alarm", 1, mLooper.dispatchAll());
202e7ba2963bedf426a1d8ba09ab535260ef364512bRoshan Pius        expectSuccessfulHwDisconnectedPnoScan(order, pnoSettings, pnoEventHandler, scanResults);
203e7ba2963bedf426a1d8ba09ab535260ef364512bRoshan Pius        verifyNoMoreInteractions(pnoEventHandler);
204e7ba2963bedf426a1d8ba09ab535260ef364512bRoshan Pius    }
205e7ba2963bedf426a1d8ba09ab535260ef364512bRoshan Pius
2066c5018cef1eb7acbcfa7fc6c9b7c018bab7ba7baRoshan Pius    private void doSuccessfulSwPnoScanTest(boolean isConnectedPno) {
207dcd877d6c143db557884993ea437e2a432cb0ba3Roshan Pius        WifiNative.PnoEventHandler pnoEventHandler = mock(WifiNative.PnoEventHandler.class);
2086c5018cef1eb7acbcfa7fc6c9b7c018bab7ba7baRoshan Pius        WifiNative.PnoSettings pnoSettings = createDummyPnoSettings(isConnectedPno);
20962bc101940ae1f5e60c4d8861a149b900dbf5e5cRoshan Pius        WifiNative.ScanEventHandler scanEventHandler = mock(WifiNative.ScanEventHandler.class);
21062bc101940ae1f5e60c4d8861a149b900dbf5e5cRoshan Pius        WifiNative.ScanSettings scanSettings = createDummyScanSettings();
21162bc101940ae1f5e60c4d8861a149b900dbf5e5cRoshan Pius        ScanResults scanResults = createDummyScanResults();
21262bc101940ae1f5e60c4d8861a149b900dbf5e5cRoshan Pius
2136c5018cef1eb7acbcfa7fc6c9b7c018bab7ba7baRoshan Pius        InOrder order = inOrder(scanEventHandler, mWifiNative);
21462bc101940ae1f5e60c4d8861a149b900dbf5e5cRoshan Pius
21562bc101940ae1f5e60c4d8861a149b900dbf5e5cRoshan Pius        // Start PNO scan
21662bc101940ae1f5e60c4d8861a149b900dbf5e5cRoshan Pius        startSuccessfulPnoScan(scanSettings, pnoSettings, scanEventHandler, pnoEventHandler);
21762bc101940ae1f5e60c4d8861a149b900dbf5e5cRoshan Pius
2186c5018cef1eb7acbcfa7fc6c9b7c018bab7ba7baRoshan Pius        expectSuccessfulSwPnoScan(order, scanEventHandler, scanResults);
21962bc101940ae1f5e60c4d8861a149b900dbf5e5cRoshan Pius
22062bc101940ae1f5e60c4d8861a149b900dbf5e5cRoshan Pius        verifyNoMoreInteractions(pnoEventHandler);
22162bc101940ae1f5e60c4d8861a149b900dbf5e5cRoshan Pius    }
22262bc101940ae1f5e60c4d8861a149b900dbf5e5cRoshan Pius
22362bc101940ae1f5e60c4d8861a149b900dbf5e5cRoshan Pius    private void createScannerWithHwPnoScanSupport() {
22462bc101940ae1f5e60c4d8861a149b900dbf5e5cRoshan Pius        mResources.setBoolean(R.bool.config_wifi_background_scan_support, true);
22562bc101940ae1f5e60c4d8861a149b900dbf5e5cRoshan Pius        mScanner = new SupplicantWifiScannerImpl(mContext, mWifiNative, mLooper.getLooper());
22662bc101940ae1f5e60c4d8861a149b900dbf5e5cRoshan Pius    }
22762bc101940ae1f5e60c4d8861a149b900dbf5e5cRoshan Pius
22862bc101940ae1f5e60c4d8861a149b900dbf5e5cRoshan Pius    private void createScannerWithSwPnoScanSupport() {
22962bc101940ae1f5e60c4d8861a149b900dbf5e5cRoshan Pius        mResources.setBoolean(R.bool.config_wifi_background_scan_support, false);
23062bc101940ae1f5e60c4d8861a149b900dbf5e5cRoshan Pius        mScanner = new SupplicantWifiScannerImpl(mContext, mWifiNative, mLooper.getLooper());
23162bc101940ae1f5e60c4d8861a149b900dbf5e5cRoshan Pius    }
23262bc101940ae1f5e60c4d8861a149b900dbf5e5cRoshan Pius
233dcd877d6c143db557884993ea437e2a432cb0ba3Roshan Pius    private WifiNative.PnoSettings createDummyPnoSettings(boolean isConnected) {
23462bc101940ae1f5e60c4d8861a149b900dbf5e5cRoshan Pius        WifiNative.PnoSettings pnoSettings = new WifiNative.PnoSettings();
235dcd877d6c143db557884993ea437e2a432cb0ba3Roshan Pius        pnoSettings.isConnected = isConnected;
23662bc101940ae1f5e60c4d8861a149b900dbf5e5cRoshan Pius        pnoSettings.networkList = new WifiNative.PnoNetwork[2];
23762bc101940ae1f5e60c4d8861a149b900dbf5e5cRoshan Pius        pnoSettings.networkList[0] = new WifiNative.PnoNetwork();
23862bc101940ae1f5e60c4d8861a149b900dbf5e5cRoshan Pius        pnoSettings.networkList[0].ssid = "ssid_pno_1";
23962bc101940ae1f5e60c4d8861a149b900dbf5e5cRoshan Pius        pnoSettings.networkList[0].networkId = 1;
24062bc101940ae1f5e60c4d8861a149b900dbf5e5cRoshan Pius        pnoSettings.networkList[0].priority = 1;
24162bc101940ae1f5e60c4d8861a149b900dbf5e5cRoshan Pius        pnoSettings.networkList[1] = new WifiNative.PnoNetwork();
24262bc101940ae1f5e60c4d8861a149b900dbf5e5cRoshan Pius        pnoSettings.networkList[1].ssid = "ssid_pno_2";
24362bc101940ae1f5e60c4d8861a149b900dbf5e5cRoshan Pius        pnoSettings.networkList[1].networkId = 2;
24462bc101940ae1f5e60c4d8861a149b900dbf5e5cRoshan Pius        pnoSettings.networkList[1].priority = 2;
24562bc101940ae1f5e60c4d8861a149b900dbf5e5cRoshan Pius        return pnoSettings;
24662bc101940ae1f5e60c4d8861a149b900dbf5e5cRoshan Pius    }
24762bc101940ae1f5e60c4d8861a149b900dbf5e5cRoshan Pius
24862bc101940ae1f5e60c4d8861a149b900dbf5e5cRoshan Pius    private WifiNative.ScanSettings createDummyScanSettings() {
24962bc101940ae1f5e60c4d8861a149b900dbf5e5cRoshan Pius        WifiNative.ScanSettings settings = new NativeScanSettingsBuilder()
25062bc101940ae1f5e60c4d8861a149b900dbf5e5cRoshan Pius                .withBasePeriod(10000)
25162bc101940ae1f5e60c4d8861a149b900dbf5e5cRoshan Pius                .withMaxApPerScan(10)
25262bc101940ae1f5e60c4d8861a149b900dbf5e5cRoshan Pius                .addBucketWithBand(10000, WifiScanner.REPORT_EVENT_AFTER_EACH_SCAN,
25362bc101940ae1f5e60c4d8861a149b900dbf5e5cRoshan Pius                        WifiScanner.WIFI_BAND_24_GHZ)
25462bc101940ae1f5e60c4d8861a149b900dbf5e5cRoshan Pius                .build();
25562bc101940ae1f5e60c4d8861a149b900dbf5e5cRoshan Pius        return settings;
25662bc101940ae1f5e60c4d8861a149b900dbf5e5cRoshan Pius    }
25762bc101940ae1f5e60c4d8861a149b900dbf5e5cRoshan Pius
25862bc101940ae1f5e60c4d8861a149b900dbf5e5cRoshan Pius    private ScanResults createDummyScanResults() {
25962bc101940ae1f5e60c4d8861a149b900dbf5e5cRoshan Pius        return ScanResults.create(0, 2400, 2450, 2450, 2400, 2450, 2450, 2400, 2450, 2450);
26062bc101940ae1f5e60c4d8861a149b900dbf5e5cRoshan Pius    }
26162bc101940ae1f5e60c4d8861a149b900dbf5e5cRoshan Pius
26262bc101940ae1f5e60c4d8861a149b900dbf5e5cRoshan Pius    private void startSuccessfulPnoScan(WifiNative.ScanSettings scanSettings,
26362bc101940ae1f5e60c4d8861a149b900dbf5e5cRoshan Pius            WifiNative.PnoSettings pnoSettings, WifiNative.ScanEventHandler scanEventHandler,
26462bc101940ae1f5e60c4d8861a149b900dbf5e5cRoshan Pius            WifiNative.PnoEventHandler pnoEventHandler) {
26562bc101940ae1f5e60c4d8861a149b900dbf5e5cRoshan Pius        when(mWifiNative.setNetworkVariable(anyInt(), anyString(), anyString())).thenReturn(true);
26662bc101940ae1f5e60c4d8861a149b900dbf5e5cRoshan Pius        when(mWifiNative.enableNetworkWithoutConnect(anyInt())).thenReturn(true);
26762bc101940ae1f5e60c4d8861a149b900dbf5e5cRoshan Pius        // Scans succeed
26862bc101940ae1f5e60c4d8861a149b900dbf5e5cRoshan Pius        when(mWifiNative.scan(any(Set.class), any(Set.class))).thenReturn(true);
269e7ba2963bedf426a1d8ba09ab535260ef364512bRoshan Pius        when(mWifiNative.setPnoScan(anyBoolean())).thenReturn(true);
2706c5018cef1eb7acbcfa7fc6c9b7c018bab7ba7baRoshan Pius
271e7ba2963bedf426a1d8ba09ab535260ef364512bRoshan Pius        if (mScanner.isHwPnoSupported(pnoSettings.isConnected)) {
272e7ba2963bedf426a1d8ba09ab535260ef364512bRoshan Pius            // This should happen only for HW PNO scan
273e7ba2963bedf426a1d8ba09ab535260ef364512bRoshan Pius            assertTrue(mScanner.setHwPnoList(pnoSettings, pnoEventHandler));
274e7ba2963bedf426a1d8ba09ab535260ef364512bRoshan Pius        } else {
275e7ba2963bedf426a1d8ba09ab535260ef364512bRoshan Pius            // This should happen only for SW PNO scan
27662bc101940ae1f5e60c4d8861a149b900dbf5e5cRoshan Pius            assertTrue(mScanner.startBatchedScan(scanSettings, scanEventHandler));
277e7ba2963bedf426a1d8ba09ab535260ef364512bRoshan Pius
27862bc101940ae1f5e60c4d8861a149b900dbf5e5cRoshan Pius        }
27962bc101940ae1f5e60c4d8861a149b900dbf5e5cRoshan Pius    }
28062bc101940ae1f5e60c4d8861a149b900dbf5e5cRoshan Pius
28162bc101940ae1f5e60c4d8861a149b900dbf5e5cRoshan Pius    private Set<Integer> expectedBandScanFreqs(int band) {
28262bc101940ae1f5e60c4d8861a149b900dbf5e5cRoshan Pius        ChannelCollection collection = mScanner.getChannelHelper().createChannelCollection();
28362bc101940ae1f5e60c4d8861a149b900dbf5e5cRoshan Pius        collection.addBand(band);
28462bc101940ae1f5e60c4d8861a149b900dbf5e5cRoshan Pius        return collection.getSupplicantScanFreqs();
28562bc101940ae1f5e60c4d8861a149b900dbf5e5cRoshan Pius    }
28662bc101940ae1f5e60c4d8861a149b900dbf5e5cRoshan Pius
28762bc101940ae1f5e60c4d8861a149b900dbf5e5cRoshan Pius    /**
28862bc101940ae1f5e60c4d8861a149b900dbf5e5cRoshan Pius     * Verify that the PNO scan was successfully started.
28962bc101940ae1f5e60c4d8861a149b900dbf5e5cRoshan Pius     */
290dcd877d6c143db557884993ea437e2a432cb0ba3Roshan Pius    private void expectSuccessfulHwDisconnectedPnoScan(InOrder order,
291dcd877d6c143db557884993ea437e2a432cb0ba3Roshan Pius            WifiNative.PnoSettings pnoSettings, WifiNative.PnoEventHandler eventHandler,
292dcd877d6c143db557884993ea437e2a432cb0ba3Roshan Pius            ScanResults scanResults) {
29362bc101940ae1f5e60c4d8861a149b900dbf5e5cRoshan Pius        for (int i = 0; i < pnoSettings.networkList.length; i++) {
29462bc101940ae1f5e60c4d8861a149b900dbf5e5cRoshan Pius            WifiNative.PnoNetwork network = pnoSettings.networkList[i];
29562bc101940ae1f5e60c4d8861a149b900dbf5e5cRoshan Pius            order.verify(mWifiNative).setNetworkVariable(network.networkId,
29662bc101940ae1f5e60c4d8861a149b900dbf5e5cRoshan Pius                    WifiConfiguration.priorityVarName, Integer.toString(network.priority));
29762bc101940ae1f5e60c4d8861a149b900dbf5e5cRoshan Pius            order.verify(mWifiNative).enableNetworkWithoutConnect(network.networkId);
29862bc101940ae1f5e60c4d8861a149b900dbf5e5cRoshan Pius        }
29962bc101940ae1f5e60c4d8861a149b900dbf5e5cRoshan Pius        // Verify  HW PNO scan started
300e7ba2963bedf426a1d8ba09ab535260ef364512bRoshan Pius        order.verify(mWifiNative).setPnoScan(true);
30162bc101940ae1f5e60c4d8861a149b900dbf5e5cRoshan Pius
30262bc101940ae1f5e60c4d8861a149b900dbf5e5cRoshan Pius        // Setup scan results
30362bc101940ae1f5e60c4d8861a149b900dbf5e5cRoshan Pius        when(mWifiNative.getScanResults()).thenReturn(scanResults.getScanDetailArrayList());
30462bc101940ae1f5e60c4d8861a149b900dbf5e5cRoshan Pius
30562bc101940ae1f5e60c4d8861a149b900dbf5e5cRoshan Pius        // Notify scan has finished
30662bc101940ae1f5e60c4d8861a149b900dbf5e5cRoshan Pius        mWifiMonitor.sendMessage(mWifiNative.getInterfaceName(), WifiMonitor.SCAN_RESULTS_EVENT);
30762bc101940ae1f5e60c4d8861a149b900dbf5e5cRoshan Pius        assertEquals("dispatch message after results event", 1, mLooper.dispatchAll());
30862bc101940ae1f5e60c4d8861a149b900dbf5e5cRoshan Pius
30962bc101940ae1f5e60c4d8861a149b900dbf5e5cRoshan Pius        order.verify(eventHandler).onPnoNetworkFound(scanResults.getRawScanResults());
31062bc101940ae1f5e60c4d8861a149b900dbf5e5cRoshan Pius    }
31162bc101940ae1f5e60c4d8861a149b900dbf5e5cRoshan Pius
31262bc101940ae1f5e60c4d8861a149b900dbf5e5cRoshan Pius    /**
31362bc101940ae1f5e60c4d8861a149b900dbf5e5cRoshan Pius     * Verify that the single scan results were delivered and that the PNO scan was paused and
31462bc101940ae1f5e60c4d8861a149b900dbf5e5cRoshan Pius     * resumed either side of it.
31562bc101940ae1f5e60c4d8861a149b900dbf5e5cRoshan Pius     */
31662bc101940ae1f5e60c4d8861a149b900dbf5e5cRoshan Pius    private void expectSuccessfulSingleScanWithHwPnoEnabled(InOrder order,
31762bc101940ae1f5e60c4d8861a149b900dbf5e5cRoshan Pius            WifiNative.ScanEventHandler eventHandler, Set<Integer> expectedScanFreqs,
31862bc101940ae1f5e60c4d8861a149b900dbf5e5cRoshan Pius            Set<Integer> expectedHiddenNetIds, ScanResults scanResults) {
31962bc101940ae1f5e60c4d8861a149b900dbf5e5cRoshan Pius        // Pause PNO scan first
320e7ba2963bedf426a1d8ba09ab535260ef364512bRoshan Pius        order.verify(mWifiNative).setPnoScan(false);
32162bc101940ae1f5e60c4d8861a149b900dbf5e5cRoshan Pius
32262bc101940ae1f5e60c4d8861a149b900dbf5e5cRoshan Pius        order.verify(mWifiNative).scan(eq(expectedScanFreqs), eq(expectedHiddenNetIds));
32362bc101940ae1f5e60c4d8861a149b900dbf5e5cRoshan Pius
32462bc101940ae1f5e60c4d8861a149b900dbf5e5cRoshan Pius        when(mWifiNative.getScanResults()).thenReturn(scanResults.getScanDetailArrayList());
32562bc101940ae1f5e60c4d8861a149b900dbf5e5cRoshan Pius
32662bc101940ae1f5e60c4d8861a149b900dbf5e5cRoshan Pius        // Notify scan has finished
32762bc101940ae1f5e60c4d8861a149b900dbf5e5cRoshan Pius        mWifiMonitor.sendMessage(mWifiNative.getInterfaceName(), WifiMonitor.SCAN_RESULTS_EVENT);
32862bc101940ae1f5e60c4d8861a149b900dbf5e5cRoshan Pius        assertEquals("dispatch message after results event", 1, mLooper.dispatchAll());
32962bc101940ae1f5e60c4d8861a149b900dbf5e5cRoshan Pius
33062bc101940ae1f5e60c4d8861a149b900dbf5e5cRoshan Pius        order.verify(eventHandler).onScanStatus(WifiNative.WIFI_SCAN_RESULTS_AVAILABLE);
33162bc101940ae1f5e60c4d8861a149b900dbf5e5cRoshan Pius        assertScanDataEquals(scanResults.getScanData(), mScanner.getLatestSingleScanResults());
33262bc101940ae1f5e60c4d8861a149b900dbf5e5cRoshan Pius    }
33362bc101940ae1f5e60c4d8861a149b900dbf5e5cRoshan Pius
33462bc101940ae1f5e60c4d8861a149b900dbf5e5cRoshan Pius    /**
335dcd877d6c143db557884993ea437e2a432cb0ba3Roshan Pius     * Verify that the SW PNO scan was successfully started. This could either be disconnected
336dcd877d6c143db557884993ea437e2a432cb0ba3Roshan Pius     * or connected PNO.
3376c5018cef1eb7acbcfa7fc6c9b7c018bab7ba7baRoshan Pius     * This is basically ensuring that the background scan runs successfully and returns the
3386c5018cef1eb7acbcfa7fc6c9b7c018bab7ba7baRoshan Pius     * expected result.
33962bc101940ae1f5e60c4d8861a149b900dbf5e5cRoshan Pius     */
34062bc101940ae1f5e60c4d8861a149b900dbf5e5cRoshan Pius    private void expectSuccessfulSwPnoScan(InOrder order,
3416c5018cef1eb7acbcfa7fc6c9b7c018bab7ba7baRoshan Pius            WifiNative.ScanEventHandler eventHandler, ScanResults scanResults) {
34262bc101940ae1f5e60c4d8861a149b900dbf5e5cRoshan Pius
34362bc101940ae1f5e60c4d8861a149b900dbf5e5cRoshan Pius        // Verify scan started
34462bc101940ae1f5e60c4d8861a149b900dbf5e5cRoshan Pius        order.verify(mWifiNative).scan(any(Set.class), any(Set.class));
34562bc101940ae1f5e60c4d8861a149b900dbf5e5cRoshan Pius
34662bc101940ae1f5e60c4d8861a149b900dbf5e5cRoshan Pius        // Make sure that HW PNO scan was not started
34762bc101940ae1f5e60c4d8861a149b900dbf5e5cRoshan Pius        verify(mWifiNative, never()).enableBackgroundScan(anyBoolean(), anyObject());
34862bc101940ae1f5e60c4d8861a149b900dbf5e5cRoshan Pius
34962bc101940ae1f5e60c4d8861a149b900dbf5e5cRoshan Pius        // Setup scan results
35062bc101940ae1f5e60c4d8861a149b900dbf5e5cRoshan Pius        when(mWifiNative.getScanResults()).thenReturn(scanResults.getScanDetailArrayList());
35162bc101940ae1f5e60c4d8861a149b900dbf5e5cRoshan Pius
35262bc101940ae1f5e60c4d8861a149b900dbf5e5cRoshan Pius        // Notify scan has finished
35362bc101940ae1f5e60c4d8861a149b900dbf5e5cRoshan Pius        mWifiMonitor.sendMessage(mWifiNative.getInterfaceName(), WifiMonitor.SCAN_RESULTS_EVENT);
35462bc101940ae1f5e60c4d8861a149b900dbf5e5cRoshan Pius        assertEquals("dispatch message after results event", 1, mLooper.dispatchAll());
35562bc101940ae1f5e60c4d8861a149b900dbf5e5cRoshan Pius
3566c5018cef1eb7acbcfa7fc6c9b7c018bab7ba7baRoshan Pius        // Verify background scan results delivered
3576c5018cef1eb7acbcfa7fc6c9b7c018bab7ba7baRoshan Pius        order.verify(eventHandler).onScanStatus(WifiNative.WIFI_SCAN_RESULTS_AVAILABLE);
3586c5018cef1eb7acbcfa7fc6c9b7c018bab7ba7baRoshan Pius        WifiScanner.ScanData[] scanData = mScanner.getLatestBatchedScanResults(true);
3596c5018cef1eb7acbcfa7fc6c9b7c018bab7ba7baRoshan Pius        WifiScanner.ScanData lastScanData = scanData[scanData.length -1];
3606c5018cef1eb7acbcfa7fc6c9b7c018bab7ba7baRoshan Pius        assertScanDataEquals(scanResults.getScanData(), lastScanData);
36162bc101940ae1f5e60c4d8861a149b900dbf5e5cRoshan Pius    }
36262bc101940ae1f5e60c4d8861a149b900dbf5e5cRoshan Pius}
363