xref: /aosp_15_r20/system/chre/test/simulation/wifi_timeout_test.cc (revision 84e339476a462649f82315436d70fd732297a399)
1*84e33947SAndroid Build Coastguard Worker /*
2*84e33947SAndroid Build Coastguard Worker  * Copyright (C) 2022 The Android Open Source Project
3*84e33947SAndroid Build Coastguard Worker  *
4*84e33947SAndroid Build Coastguard Worker  * Licensed under the Apache License, Version 2.0 (the "License");
5*84e33947SAndroid Build Coastguard Worker  * you may not use this file except in compliance with the License.
6*84e33947SAndroid Build Coastguard Worker  * You may obtain a copy of the License at
7*84e33947SAndroid Build Coastguard Worker  *
8*84e33947SAndroid Build Coastguard Worker  *      http://www.apache.org/licenses/LICENSE-2.0
9*84e33947SAndroid Build Coastguard Worker  *
10*84e33947SAndroid Build Coastguard Worker  * Unless required by applicable law or agreed to in writing, software
11*84e33947SAndroid Build Coastguard Worker  * distributed under the License is distributed on an "AS IS" BASIS,
12*84e33947SAndroid Build Coastguard Worker  * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
13*84e33947SAndroid Build Coastguard Worker  * See the License for the specific language governing permissions and
14*84e33947SAndroid Build Coastguard Worker  * limitations under the License.
15*84e33947SAndroid Build Coastguard Worker  */
16*84e33947SAndroid Build Coastguard Worker 
17*84e33947SAndroid Build Coastguard Worker #include <cstdint>
18*84e33947SAndroid Build Coastguard Worker 
19*84e33947SAndroid Build Coastguard Worker #include "chre/core/event_loop_manager.h"
20*84e33947SAndroid Build Coastguard Worker #include "chre/core/settings.h"
21*84e33947SAndroid Build Coastguard Worker #include "chre/platform/linux/pal_wifi.h"
22*84e33947SAndroid Build Coastguard Worker #include "chre/platform/log.h"
23*84e33947SAndroid Build Coastguard Worker #include "chre/util/nanoapp/app_id.h"
24*84e33947SAndroid Build Coastguard Worker #include "chre/util/system/napp_permissions.h"
25*84e33947SAndroid Build Coastguard Worker #include "chre_api/chre/event.h"
26*84e33947SAndroid Build Coastguard Worker #include "chre_api/chre/re.h"
27*84e33947SAndroid Build Coastguard Worker #include "chre_api/chre/wifi.h"
28*84e33947SAndroid Build Coastguard Worker #include "gtest/gtest.h"
29*84e33947SAndroid Build Coastguard Worker #include "test_base.h"
30*84e33947SAndroid Build Coastguard Worker #include "test_event.h"
31*84e33947SAndroid Build Coastguard Worker #include "test_event_queue.h"
32*84e33947SAndroid Build Coastguard Worker #include "test_util.h"
33*84e33947SAndroid Build Coastguard Worker 
34*84e33947SAndroid Build Coastguard Worker namespace chre {
35*84e33947SAndroid Build Coastguard Worker namespace {
36*84e33947SAndroid Build Coastguard Worker // WifiTimeoutTestBase needs to set timeout more than the max waitForEvent()
37*84e33947SAndroid Build Coastguard Worker // should process (Currently it is
38*84e33947SAndroid Build Coastguard Worker // WifiCanDispatchSecondScanRequestInQueueAfterFirstTimeout). If not,
39*84e33947SAndroid Build Coastguard Worker // waitForEvent will timeout before actual timeout happens in CHRE, making us
40*84e33947SAndroid Build Coastguard Worker // unable to observe how system handles timeout.
41*84e33947SAndroid Build Coastguard Worker class WifiTimeoutTestBase : public TestBase {
42*84e33947SAndroid Build Coastguard Worker  protected:
getTimeoutNs() const43*84e33947SAndroid Build Coastguard Worker   uint64_t getTimeoutNs() const override {
44*84e33947SAndroid Build Coastguard Worker     return 3 * CHRE_TEST_WIFI_SCAN_RESULT_TIMEOUT_NS;
45*84e33947SAndroid Build Coastguard Worker   }
46*84e33947SAndroid Build Coastguard Worker };
47*84e33947SAndroid Build Coastguard Worker 
48*84e33947SAndroid Build Coastguard Worker CREATE_CHRE_TEST_EVENT(SCAN_REQUEST, 20);
49*84e33947SAndroid Build Coastguard Worker CREATE_CHRE_TEST_EVENT(REQUEST_TIMED_OUT, 21);
50*84e33947SAndroid Build Coastguard Worker 
TEST_F(WifiTimeoutTestBase,WifiScanRequestTimeoutTest)51*84e33947SAndroid Build Coastguard Worker TEST_F(WifiTimeoutTestBase, WifiScanRequestTimeoutTest) {
52*84e33947SAndroid Build Coastguard Worker   class ScanTestNanoapp : public TestNanoapp {
53*84e33947SAndroid Build Coastguard Worker    public:
54*84e33947SAndroid Build Coastguard Worker     explicit ScanTestNanoapp()
55*84e33947SAndroid Build Coastguard Worker         : TestNanoapp(
56*84e33947SAndroid Build Coastguard Worker               TestNanoappInfo{.perms = NanoappPermissions::CHRE_PERMS_WIFI}) {}
57*84e33947SAndroid Build Coastguard Worker 
58*84e33947SAndroid Build Coastguard Worker     bool start() override {
59*84e33947SAndroid Build Coastguard Worker       mRequestTimer = CHRE_TIMER_INVALID;
60*84e33947SAndroid Build Coastguard Worker       return true;
61*84e33947SAndroid Build Coastguard Worker     }
62*84e33947SAndroid Build Coastguard Worker 
63*84e33947SAndroid Build Coastguard Worker     void handleEvent(uint32_t, uint16_t eventType,
64*84e33947SAndroid Build Coastguard Worker                      const void *eventData) override {
65*84e33947SAndroid Build Coastguard Worker       switch (eventType) {
66*84e33947SAndroid Build Coastguard Worker         case CHRE_EVENT_WIFI_ASYNC_RESULT: {
67*84e33947SAndroid Build Coastguard Worker           auto *event = static_cast<const chreAsyncResult *>(eventData);
68*84e33947SAndroid Build Coastguard Worker           if (mRequestTimer != CHRE_TIMER_INVALID) {
69*84e33947SAndroid Build Coastguard Worker             chreTimerCancel(mRequestTimer);
70*84e33947SAndroid Build Coastguard Worker             mRequestTimer = CHRE_TIMER_INVALID;
71*84e33947SAndroid Build Coastguard Worker           }
72*84e33947SAndroid Build Coastguard Worker           if (event->success) {
73*84e33947SAndroid Build Coastguard Worker             TestEventQueueSingleton::get()->pushEvent(
74*84e33947SAndroid Build Coastguard Worker                 CHRE_EVENT_WIFI_ASYNC_RESULT,
75*84e33947SAndroid Build Coastguard Worker                 *(static_cast<const uint32_t *>(event->cookie)));
76*84e33947SAndroid Build Coastguard Worker           }
77*84e33947SAndroid Build Coastguard Worker           break;
78*84e33947SAndroid Build Coastguard Worker         }
79*84e33947SAndroid Build Coastguard Worker 
80*84e33947SAndroid Build Coastguard Worker         case CHRE_EVENT_WIFI_SCAN_RESULT: {
81*84e33947SAndroid Build Coastguard Worker           TestEventQueueSingleton::get()->pushEvent(
82*84e33947SAndroid Build Coastguard Worker               CHRE_EVENT_WIFI_SCAN_RESULT);
83*84e33947SAndroid Build Coastguard Worker           break;
84*84e33947SAndroid Build Coastguard Worker         }
85*84e33947SAndroid Build Coastguard Worker 
86*84e33947SAndroid Build Coastguard Worker         case CHRE_EVENT_TIMER: {
87*84e33947SAndroid Build Coastguard Worker           TestEventQueueSingleton::get()->pushEvent(REQUEST_TIMED_OUT);
88*84e33947SAndroid Build Coastguard Worker           mRequestTimer = CHRE_TIMER_INVALID;
89*84e33947SAndroid Build Coastguard Worker           break;
90*84e33947SAndroid Build Coastguard Worker         }
91*84e33947SAndroid Build Coastguard Worker 
92*84e33947SAndroid Build Coastguard Worker         case CHRE_EVENT_TEST_EVENT: {
93*84e33947SAndroid Build Coastguard Worker           auto event = static_cast<const TestEvent *>(eventData);
94*84e33947SAndroid Build Coastguard Worker           switch (event->type) {
95*84e33947SAndroid Build Coastguard Worker             case SCAN_REQUEST:
96*84e33947SAndroid Build Coastguard Worker               bool success = false;
97*84e33947SAndroid Build Coastguard Worker               mCookie = *static_cast<uint32_t *>(event->data);
98*84e33947SAndroid Build Coastguard Worker               if (chreWifiRequestScanAsyncDefault(&mCookie)) {
99*84e33947SAndroid Build Coastguard Worker                 mRequestTimer =
100*84e33947SAndroid Build Coastguard Worker                     chreTimerSet(CHRE_TEST_WIFI_SCAN_RESULT_TIMEOUT_NS, nullptr,
101*84e33947SAndroid Build Coastguard Worker                                  true /* oneShot */);
102*84e33947SAndroid Build Coastguard Worker                 success = mRequestTimer != CHRE_TIMER_INVALID;
103*84e33947SAndroid Build Coastguard Worker               }
104*84e33947SAndroid Build Coastguard Worker               TestEventQueueSingleton::get()->pushEvent(SCAN_REQUEST, success);
105*84e33947SAndroid Build Coastguard Worker               break;
106*84e33947SAndroid Build Coastguard Worker           }
107*84e33947SAndroid Build Coastguard Worker           break;
108*84e33947SAndroid Build Coastguard Worker         }
109*84e33947SAndroid Build Coastguard Worker       }
110*84e33947SAndroid Build Coastguard Worker     }
111*84e33947SAndroid Build Coastguard Worker 
112*84e33947SAndroid Build Coastguard Worker    protected:
113*84e33947SAndroid Build Coastguard Worker     uint32_t mCookie;
114*84e33947SAndroid Build Coastguard Worker     uint32_t mRequestTimer;
115*84e33947SAndroid Build Coastguard Worker   };
116*84e33947SAndroid Build Coastguard Worker 
117*84e33947SAndroid Build Coastguard Worker   uint64_t appId = loadNanoapp(MakeUnique<ScanTestNanoapp>());
118*84e33947SAndroid Build Coastguard Worker 
119*84e33947SAndroid Build Coastguard Worker   constexpr uint32_t timeOutCookie = 0xdead;
120*84e33947SAndroid Build Coastguard Worker   chrePalWifiEnableResponse(PalWifiAsyncRequestTypes::SCAN,
121*84e33947SAndroid Build Coastguard Worker                             false /* enableResponse */);
122*84e33947SAndroid Build Coastguard Worker   sendEventToNanoapp(appId, SCAN_REQUEST, timeOutCookie);
123*84e33947SAndroid Build Coastguard Worker   bool success;
124*84e33947SAndroid Build Coastguard Worker   waitForEvent(SCAN_REQUEST, &success);
125*84e33947SAndroid Build Coastguard Worker   EXPECT_TRUE(success);
126*84e33947SAndroid Build Coastguard Worker 
127*84e33947SAndroid Build Coastguard Worker   waitForEvent(REQUEST_TIMED_OUT);
128*84e33947SAndroid Build Coastguard Worker 
129*84e33947SAndroid Build Coastguard Worker   // Make sure that we can still request scan after a timedout
130*84e33947SAndroid Build Coastguard Worker   // request.
131*84e33947SAndroid Build Coastguard Worker   constexpr uint32_t successCookie = 0x0101;
132*84e33947SAndroid Build Coastguard Worker   chrePalWifiEnableResponse(PalWifiAsyncRequestTypes::SCAN,
133*84e33947SAndroid Build Coastguard Worker                             true /* enableResponse */);
134*84e33947SAndroid Build Coastguard Worker   sendEventToNanoapp(appId, SCAN_REQUEST, successCookie);
135*84e33947SAndroid Build Coastguard Worker   waitForEvent(SCAN_REQUEST, &success);
136*84e33947SAndroid Build Coastguard Worker   EXPECT_TRUE(success);
137*84e33947SAndroid Build Coastguard Worker   waitForEvent(CHRE_EVENT_WIFI_SCAN_RESULT);
138*84e33947SAndroid Build Coastguard Worker 
139*84e33947SAndroid Build Coastguard Worker   unloadNanoapp(appId);
140*84e33947SAndroid Build Coastguard Worker }
141*84e33947SAndroid Build Coastguard Worker 
TEST_F(WifiTimeoutTestBase,WifiCanDispatchQueuedRequestAfterOneTimeout)142*84e33947SAndroid Build Coastguard Worker TEST_F(WifiTimeoutTestBase, WifiCanDispatchQueuedRequestAfterOneTimeout) {
143*84e33947SAndroid Build Coastguard Worker   constexpr uint8_t kNanoappNum = 2;
144*84e33947SAndroid Build Coastguard Worker   // receivedTimeout is shared across apps and must be static.
145*84e33947SAndroid Build Coastguard Worker   // But we want it initialized each time the test is executed.
146*84e33947SAndroid Build Coastguard Worker   static uint8_t receivedTimeout;
147*84e33947SAndroid Build Coastguard Worker   receivedTimeout = 0;
148*84e33947SAndroid Build Coastguard Worker 
149*84e33947SAndroid Build Coastguard Worker   class ScanTestNanoapp : public TestNanoapp {
150*84e33947SAndroid Build Coastguard Worker    public:
151*84e33947SAndroid Build Coastguard Worker     explicit ScanTestNanoapp(uint64_t id = kDefaultTestNanoappId)
152*84e33947SAndroid Build Coastguard Worker         : TestNanoapp(TestNanoappInfo{
153*84e33947SAndroid Build Coastguard Worker               .id = id, .perms = NanoappPermissions::CHRE_PERMS_WIFI}) {}
154*84e33947SAndroid Build Coastguard Worker 
155*84e33947SAndroid Build Coastguard Worker     bool start() override {
156*84e33947SAndroid Build Coastguard Worker       for (uint8_t i = 0; i < kNanoappNum; ++i) {
157*84e33947SAndroid Build Coastguard Worker         mRequestTimers[i] = CHRE_TIMER_INVALID;
158*84e33947SAndroid Build Coastguard Worker       }
159*84e33947SAndroid Build Coastguard Worker       return true;
160*84e33947SAndroid Build Coastguard Worker     }
161*84e33947SAndroid Build Coastguard Worker 
162*84e33947SAndroid Build Coastguard Worker     void handleEvent(uint32_t, uint16_t eventType,
163*84e33947SAndroid Build Coastguard Worker                      const void *eventData) override {
164*84e33947SAndroid Build Coastguard Worker       size_t index = id() - CHRE_VENDOR_ID_EXAMPLE - 1;
165*84e33947SAndroid Build Coastguard Worker       switch (eventType) {
166*84e33947SAndroid Build Coastguard Worker         case CHRE_EVENT_WIFI_ASYNC_RESULT: {
167*84e33947SAndroid Build Coastguard Worker           auto *event = static_cast<const chreAsyncResult *>(eventData);
168*84e33947SAndroid Build Coastguard Worker           if (mRequestTimers[index] != CHRE_TIMER_INVALID) {
169*84e33947SAndroid Build Coastguard Worker             chreTimerCancel(mRequestTimers[index]);
170*84e33947SAndroid Build Coastguard Worker             mRequestTimers[index] = CHRE_TIMER_INVALID;
171*84e33947SAndroid Build Coastguard Worker           }
172*84e33947SAndroid Build Coastguard Worker           if (event->success) {
173*84e33947SAndroid Build Coastguard Worker             TestEventQueueSingleton::get()->pushEvent(
174*84e33947SAndroid Build Coastguard Worker                 CHRE_EVENT_WIFI_ASYNC_RESULT,
175*84e33947SAndroid Build Coastguard Worker                 *(static_cast<const uint32_t *>(event->cookie)));
176*84e33947SAndroid Build Coastguard Worker           }
177*84e33947SAndroid Build Coastguard Worker           break;
178*84e33947SAndroid Build Coastguard Worker         }
179*84e33947SAndroid Build Coastguard Worker 
180*84e33947SAndroid Build Coastguard Worker         case CHRE_EVENT_WIFI_SCAN_RESULT: {
181*84e33947SAndroid Build Coastguard Worker           TestEventQueueSingleton::get()->pushEvent(
182*84e33947SAndroid Build Coastguard Worker               CHRE_EVENT_WIFI_SCAN_RESULT);
183*84e33947SAndroid Build Coastguard Worker           break;
184*84e33947SAndroid Build Coastguard Worker         }
185*84e33947SAndroid Build Coastguard Worker 
186*84e33947SAndroid Build Coastguard Worker         case CHRE_EVENT_TIMER: {
187*84e33947SAndroid Build Coastguard Worker           if (eventData == &mCookie[index]) {
188*84e33947SAndroid Build Coastguard Worker             receivedTimeout++;
189*84e33947SAndroid Build Coastguard Worker             mRequestTimers[index] = CHRE_TIMER_INVALID;
190*84e33947SAndroid Build Coastguard Worker           }
191*84e33947SAndroid Build Coastguard Worker           if (receivedTimeout == 2) {
192*84e33947SAndroid Build Coastguard Worker             TestEventQueueSingleton::get()->pushEvent(REQUEST_TIMED_OUT);
193*84e33947SAndroid Build Coastguard Worker           }
194*84e33947SAndroid Build Coastguard Worker           break;
195*84e33947SAndroid Build Coastguard Worker         }
196*84e33947SAndroid Build Coastguard Worker 
197*84e33947SAndroid Build Coastguard Worker         case CHRE_EVENT_TEST_EVENT: {
198*84e33947SAndroid Build Coastguard Worker           auto event = static_cast<const TestEvent *>(eventData);
199*84e33947SAndroid Build Coastguard Worker           switch (event->type) {
200*84e33947SAndroid Build Coastguard Worker             case SCAN_REQUEST:
201*84e33947SAndroid Build Coastguard Worker               bool success = false;
202*84e33947SAndroid Build Coastguard Worker               mCookie[index] = *static_cast<uint32_t *>(event->data);
203*84e33947SAndroid Build Coastguard Worker               if (chreWifiRequestScanAsyncDefault(&mCookie[index])) {
204*84e33947SAndroid Build Coastguard Worker                 mRequestTimers[index] =
205*84e33947SAndroid Build Coastguard Worker                     chreTimerSet(CHRE_TEST_WIFI_SCAN_RESULT_TIMEOUT_NS,
206*84e33947SAndroid Build Coastguard Worker                                  &mCookie[index], true /* oneShot */);
207*84e33947SAndroid Build Coastguard Worker                 success = mRequestTimers[index] != CHRE_TIMER_INVALID;
208*84e33947SAndroid Build Coastguard Worker               }
209*84e33947SAndroid Build Coastguard Worker               TestEventQueueSingleton::get()->pushEvent(SCAN_REQUEST, success);
210*84e33947SAndroid Build Coastguard Worker               break;
211*84e33947SAndroid Build Coastguard Worker           }
212*84e33947SAndroid Build Coastguard Worker           break;
213*84e33947SAndroid Build Coastguard Worker         }
214*84e33947SAndroid Build Coastguard Worker       }
215*84e33947SAndroid Build Coastguard Worker     }
216*84e33947SAndroid Build Coastguard Worker 
217*84e33947SAndroid Build Coastguard Worker    protected:
218*84e33947SAndroid Build Coastguard Worker     uint32_t mCookie[kNanoappNum];
219*84e33947SAndroid Build Coastguard Worker     uint32_t mRequestTimers[kNanoappNum];
220*84e33947SAndroid Build Coastguard Worker   };
221*84e33947SAndroid Build Coastguard Worker   constexpr uint64_t kAppOneId = makeExampleNanoappId(1);
222*84e33947SAndroid Build Coastguard Worker   constexpr uint64_t kAppTwoId = makeExampleNanoappId(2);
223*84e33947SAndroid Build Coastguard Worker 
224*84e33947SAndroid Build Coastguard Worker   uint64_t firstAppId = loadNanoapp(MakeUnique<ScanTestNanoapp>(kAppOneId));
225*84e33947SAndroid Build Coastguard Worker   uint64_t secondAppId = loadNanoapp(MakeUnique<ScanTestNanoapp>(kAppTwoId));
226*84e33947SAndroid Build Coastguard Worker 
227*84e33947SAndroid Build Coastguard Worker   constexpr uint32_t timeOutCookie = 0xdead;
228*84e33947SAndroid Build Coastguard Worker   chrePalWifiEnableResponse(PalWifiAsyncRequestTypes::SCAN,
229*84e33947SAndroid Build Coastguard Worker                             false /* enableResponse */);
230*84e33947SAndroid Build Coastguard Worker   bool success;
231*84e33947SAndroid Build Coastguard Worker   sendEventToNanoapp(firstAppId, SCAN_REQUEST, timeOutCookie);
232*84e33947SAndroid Build Coastguard Worker   waitForEvent(SCAN_REQUEST, &success);
233*84e33947SAndroid Build Coastguard Worker   EXPECT_TRUE(success);
234*84e33947SAndroid Build Coastguard Worker   sendEventToNanoapp(secondAppId, SCAN_REQUEST, timeOutCookie);
235*84e33947SAndroid Build Coastguard Worker   waitForEvent(SCAN_REQUEST, &success);
236*84e33947SAndroid Build Coastguard Worker   EXPECT_TRUE(success);
237*84e33947SAndroid Build Coastguard Worker 
238*84e33947SAndroid Build Coastguard Worker   waitForEvent(REQUEST_TIMED_OUT);
239*84e33947SAndroid Build Coastguard Worker 
240*84e33947SAndroid Build Coastguard Worker   // Make sure that we can still request scan for both nanoapps after a timedout
241*84e33947SAndroid Build Coastguard Worker   // request.
242*84e33947SAndroid Build Coastguard Worker   constexpr uint32_t successCookie = 0x0101;
243*84e33947SAndroid Build Coastguard Worker   chrePalWifiEnableResponse(PalWifiAsyncRequestTypes::SCAN,
244*84e33947SAndroid Build Coastguard Worker                             true /* enableResponse */);
245*84e33947SAndroid Build Coastguard Worker   sendEventToNanoapp(firstAppId, SCAN_REQUEST, successCookie);
246*84e33947SAndroid Build Coastguard Worker   waitForEvent(SCAN_REQUEST, &success);
247*84e33947SAndroid Build Coastguard Worker   EXPECT_TRUE(success);
248*84e33947SAndroid Build Coastguard Worker   waitForEvent(CHRE_EVENT_WIFI_SCAN_RESULT);
249*84e33947SAndroid Build Coastguard Worker   sendEventToNanoapp(secondAppId, SCAN_REQUEST, successCookie);
250*84e33947SAndroid Build Coastguard Worker   waitForEvent(SCAN_REQUEST, &success);
251*84e33947SAndroid Build Coastguard Worker   EXPECT_TRUE(success);
252*84e33947SAndroid Build Coastguard Worker   waitForEvent(CHRE_EVENT_WIFI_SCAN_RESULT);
253*84e33947SAndroid Build Coastguard Worker 
254*84e33947SAndroid Build Coastguard Worker   unloadNanoapp(firstAppId);
255*84e33947SAndroid Build Coastguard Worker   unloadNanoapp(secondAppId);
256*84e33947SAndroid Build Coastguard Worker }
257*84e33947SAndroid Build Coastguard Worker 
TEST_F(WifiTimeoutTestBase,WifiScanMonitorTimeoutTest)258*84e33947SAndroid Build Coastguard Worker TEST_F(WifiTimeoutTestBase, WifiScanMonitorTimeoutTest) {
259*84e33947SAndroid Build Coastguard Worker   CREATE_CHRE_TEST_EVENT(SCAN_MONITOR_REQUEST, 1);
260*84e33947SAndroid Build Coastguard Worker 
261*84e33947SAndroid Build Coastguard Worker   struct MonitoringRequest {
262*84e33947SAndroid Build Coastguard Worker     bool enable;
263*84e33947SAndroid Build Coastguard Worker     uint32_t cookie;
264*84e33947SAndroid Build Coastguard Worker   };
265*84e33947SAndroid Build Coastguard Worker 
266*84e33947SAndroid Build Coastguard Worker   class App : public TestNanoapp {
267*84e33947SAndroid Build Coastguard Worker    public:
268*84e33947SAndroid Build Coastguard Worker     App()
269*84e33947SAndroid Build Coastguard Worker         : TestNanoapp(
270*84e33947SAndroid Build Coastguard Worker               TestNanoappInfo{.perms = NanoappPermissions::CHRE_PERMS_WIFI}) {}
271*84e33947SAndroid Build Coastguard Worker 
272*84e33947SAndroid Build Coastguard Worker     bool start() override {
273*84e33947SAndroid Build Coastguard Worker       mRequestTimer = CHRE_TIMER_INVALID;
274*84e33947SAndroid Build Coastguard Worker       return true;
275*84e33947SAndroid Build Coastguard Worker     }
276*84e33947SAndroid Build Coastguard Worker 
277*84e33947SAndroid Build Coastguard Worker     void handleEvent(uint32_t, uint16_t eventType,
278*84e33947SAndroid Build Coastguard Worker                      const void *eventData) override {
279*84e33947SAndroid Build Coastguard Worker       switch (eventType) {
280*84e33947SAndroid Build Coastguard Worker         case CHRE_EVENT_WIFI_ASYNC_RESULT: {
281*84e33947SAndroid Build Coastguard Worker           auto *event = static_cast<const chreAsyncResult *>(eventData);
282*84e33947SAndroid Build Coastguard Worker           if (event->success) {
283*84e33947SAndroid Build Coastguard Worker             if (mRequestTimer != CHRE_TIMER_INVALID) {
284*84e33947SAndroid Build Coastguard Worker               chreTimerCancel(mRequestTimer);
285*84e33947SAndroid Build Coastguard Worker               mRequestTimer = CHRE_TIMER_INVALID;
286*84e33947SAndroid Build Coastguard Worker             }
287*84e33947SAndroid Build Coastguard Worker             TestEventQueueSingleton::get()->pushEvent(
288*84e33947SAndroid Build Coastguard Worker                 CHRE_EVENT_WIFI_ASYNC_RESULT,
289*84e33947SAndroid Build Coastguard Worker                 *(static_cast<const uint32_t *>(event->cookie)));
290*84e33947SAndroid Build Coastguard Worker           }
291*84e33947SAndroid Build Coastguard Worker           break;
292*84e33947SAndroid Build Coastguard Worker         }
293*84e33947SAndroid Build Coastguard Worker 
294*84e33947SAndroid Build Coastguard Worker         case CHRE_EVENT_TIMER: {
295*84e33947SAndroid Build Coastguard Worker           mRequestTimer = CHRE_TIMER_INVALID;
296*84e33947SAndroid Build Coastguard Worker           TestEventQueueSingleton::get()->pushEvent(REQUEST_TIMED_OUT);
297*84e33947SAndroid Build Coastguard Worker           break;
298*84e33947SAndroid Build Coastguard Worker         }
299*84e33947SAndroid Build Coastguard Worker 
300*84e33947SAndroid Build Coastguard Worker         case CHRE_EVENT_TEST_EVENT: {
301*84e33947SAndroid Build Coastguard Worker           auto event = static_cast<const TestEvent *>(eventData);
302*84e33947SAndroid Build Coastguard Worker           switch (event->type) {
303*84e33947SAndroid Build Coastguard Worker             case SCAN_MONITOR_REQUEST:
304*84e33947SAndroid Build Coastguard Worker               bool success = false;
305*84e33947SAndroid Build Coastguard Worker               auto request =
306*84e33947SAndroid Build Coastguard Worker                   static_cast<const MonitoringRequest *>(event->data);
307*84e33947SAndroid Build Coastguard Worker               if (chreWifiConfigureScanMonitorAsync(request->enable,
308*84e33947SAndroid Build Coastguard Worker                                                     &mCookie)) {
309*84e33947SAndroid Build Coastguard Worker                 mCookie = request->cookie;
310*84e33947SAndroid Build Coastguard Worker                 mRequestTimer = chreTimerSet(CHRE_TEST_ASYNC_RESULT_TIMEOUT_NS,
311*84e33947SAndroid Build Coastguard Worker                                              nullptr, true /* oneShot */);
312*84e33947SAndroid Build Coastguard Worker                 success = mRequestTimer != CHRE_TIMER_INVALID;
313*84e33947SAndroid Build Coastguard Worker               }
314*84e33947SAndroid Build Coastguard Worker 
315*84e33947SAndroid Build Coastguard Worker               TestEventQueueSingleton::get()->pushEvent(SCAN_MONITOR_REQUEST,
316*84e33947SAndroid Build Coastguard Worker                                                         success);
317*84e33947SAndroid Build Coastguard Worker           }
318*84e33947SAndroid Build Coastguard Worker         }
319*84e33947SAndroid Build Coastguard Worker       }
320*84e33947SAndroid Build Coastguard Worker     }
321*84e33947SAndroid Build Coastguard Worker 
322*84e33947SAndroid Build Coastguard Worker    protected:
323*84e33947SAndroid Build Coastguard Worker     uint32_t mCookie;
324*84e33947SAndroid Build Coastguard Worker     uint32_t mRequestTimer;
325*84e33947SAndroid Build Coastguard Worker   };
326*84e33947SAndroid Build Coastguard Worker 
327*84e33947SAndroid Build Coastguard Worker   uint64_t appId = loadNanoapp(MakeUnique<App>());
328*84e33947SAndroid Build Coastguard Worker 
329*84e33947SAndroid Build Coastguard Worker   MonitoringRequest timeoutRequest{.enable = true, .cookie = 0xdead};
330*84e33947SAndroid Build Coastguard Worker   chrePalWifiEnableResponse(PalWifiAsyncRequestTypes::SCAN_MONITORING, false);
331*84e33947SAndroid Build Coastguard Worker   sendEventToNanoapp(appId, SCAN_MONITOR_REQUEST, timeoutRequest);
332*84e33947SAndroid Build Coastguard Worker   bool success;
333*84e33947SAndroid Build Coastguard Worker   waitForEvent(SCAN_MONITOR_REQUEST, &success);
334*84e33947SAndroid Build Coastguard Worker   EXPECT_TRUE(success);
335*84e33947SAndroid Build Coastguard Worker 
336*84e33947SAndroid Build Coastguard Worker   waitForEvent(REQUEST_TIMED_OUT);
337*84e33947SAndroid Build Coastguard Worker 
338*84e33947SAndroid Build Coastguard Worker   // Make sure that we can still request to change scan monitor after a timedout
339*84e33947SAndroid Build Coastguard Worker   // request.
340*84e33947SAndroid Build Coastguard Worker   MonitoringRequest enableRequest{.enable = true, .cookie = 0x1010};
341*84e33947SAndroid Build Coastguard Worker   chrePalWifiEnableResponse(PalWifiAsyncRequestTypes::SCAN_MONITORING, true);
342*84e33947SAndroid Build Coastguard Worker   sendEventToNanoapp(appId, SCAN_MONITOR_REQUEST, enableRequest);
343*84e33947SAndroid Build Coastguard Worker   waitForEvent(SCAN_MONITOR_REQUEST, &success);
344*84e33947SAndroid Build Coastguard Worker   EXPECT_TRUE(success);
345*84e33947SAndroid Build Coastguard Worker 
346*84e33947SAndroid Build Coastguard Worker   uint32_t cookie;
347*84e33947SAndroid Build Coastguard Worker   waitForEvent(CHRE_EVENT_WIFI_ASYNC_RESULT, &cookie);
348*84e33947SAndroid Build Coastguard Worker   EXPECT_EQ(cookie, enableRequest.cookie);
349*84e33947SAndroid Build Coastguard Worker   EXPECT_TRUE(chrePalWifiIsScanMonitoringActive());
350*84e33947SAndroid Build Coastguard Worker 
351*84e33947SAndroid Build Coastguard Worker   MonitoringRequest disableRequest{.enable = false, .cookie = 0x0101};
352*84e33947SAndroid Build Coastguard Worker   sendEventToNanoapp(appId, SCAN_MONITOR_REQUEST, disableRequest);
353*84e33947SAndroid Build Coastguard Worker   waitForEvent(SCAN_MONITOR_REQUEST, &success);
354*84e33947SAndroid Build Coastguard Worker   EXPECT_TRUE(success);
355*84e33947SAndroid Build Coastguard Worker 
356*84e33947SAndroid Build Coastguard Worker   waitForEvent(CHRE_EVENT_WIFI_ASYNC_RESULT, &cookie);
357*84e33947SAndroid Build Coastguard Worker   EXPECT_EQ(cookie, disableRequest.cookie);
358*84e33947SAndroid Build Coastguard Worker   EXPECT_FALSE(chrePalWifiIsScanMonitoringActive());
359*84e33947SAndroid Build Coastguard Worker 
360*84e33947SAndroid Build Coastguard Worker   unloadNanoapp(appId);
361*84e33947SAndroid Build Coastguard Worker }
362*84e33947SAndroid Build Coastguard Worker 
TEST_F(WifiTimeoutTestBase,WifiRequestRangingTimeoutTest)363*84e33947SAndroid Build Coastguard Worker TEST_F(WifiTimeoutTestBase, WifiRequestRangingTimeoutTest) {
364*84e33947SAndroid Build Coastguard Worker   CREATE_CHRE_TEST_EVENT(RANGING_REQUEST, 0);
365*84e33947SAndroid Build Coastguard Worker 
366*84e33947SAndroid Build Coastguard Worker   class App : public TestNanoapp {
367*84e33947SAndroid Build Coastguard Worker    public:
368*84e33947SAndroid Build Coastguard Worker     App()
369*84e33947SAndroid Build Coastguard Worker         : TestNanoapp(
370*84e33947SAndroid Build Coastguard Worker               TestNanoappInfo{.perms = NanoappPermissions::CHRE_PERMS_WIFI}) {}
371*84e33947SAndroid Build Coastguard Worker 
372*84e33947SAndroid Build Coastguard Worker     bool start() override {
373*84e33947SAndroid Build Coastguard Worker       mRequestTimer = CHRE_TIMER_INVALID;
374*84e33947SAndroid Build Coastguard Worker       return true;
375*84e33947SAndroid Build Coastguard Worker     }
376*84e33947SAndroid Build Coastguard Worker 
377*84e33947SAndroid Build Coastguard Worker     void handleEvent(uint32_t, uint16_t eventType,
378*84e33947SAndroid Build Coastguard Worker                      const void *eventData) override {
379*84e33947SAndroid Build Coastguard Worker       switch (eventType) {
380*84e33947SAndroid Build Coastguard Worker         case CHRE_EVENT_WIFI_ASYNC_RESULT: {
381*84e33947SAndroid Build Coastguard Worker           if (mRequestTimer != CHRE_TIMER_INVALID) {
382*84e33947SAndroid Build Coastguard Worker             chreTimerCancel(mRequestTimer);
383*84e33947SAndroid Build Coastguard Worker             mRequestTimer = CHRE_TIMER_INVALID;
384*84e33947SAndroid Build Coastguard Worker           }
385*84e33947SAndroid Build Coastguard Worker 
386*84e33947SAndroid Build Coastguard Worker           auto *event = static_cast<const chreAsyncResult *>(eventData);
387*84e33947SAndroid Build Coastguard Worker           if (event->success) {
388*84e33947SAndroid Build Coastguard Worker             if (event->errorCode == 0) {
389*84e33947SAndroid Build Coastguard Worker               TestEventQueueSingleton::get()->pushEvent(
390*84e33947SAndroid Build Coastguard Worker                   CHRE_EVENT_WIFI_ASYNC_RESULT,
391*84e33947SAndroid Build Coastguard Worker                   *(static_cast<const uint32_t *>(event->cookie)));
392*84e33947SAndroid Build Coastguard Worker             }
393*84e33947SAndroid Build Coastguard Worker           }
394*84e33947SAndroid Build Coastguard Worker           break;
395*84e33947SAndroid Build Coastguard Worker         }
396*84e33947SAndroid Build Coastguard Worker 
397*84e33947SAndroid Build Coastguard Worker         case CHRE_EVENT_TIMER: {
398*84e33947SAndroid Build Coastguard Worker           mRequestTimer = CHRE_TIMER_INVALID;
399*84e33947SAndroid Build Coastguard Worker           TestEventQueueSingleton::get()->pushEvent(REQUEST_TIMED_OUT);
400*84e33947SAndroid Build Coastguard Worker           break;
401*84e33947SAndroid Build Coastguard Worker         }
402*84e33947SAndroid Build Coastguard Worker 
403*84e33947SAndroid Build Coastguard Worker         case CHRE_EVENT_TEST_EVENT: {
404*84e33947SAndroid Build Coastguard Worker           auto event = static_cast<const TestEvent *>(eventData);
405*84e33947SAndroid Build Coastguard Worker           switch (event->type) {
406*84e33947SAndroid Build Coastguard Worker             case RANGING_REQUEST:
407*84e33947SAndroid Build Coastguard Worker               bool success = false;
408*84e33947SAndroid Build Coastguard Worker               mCookie = *static_cast<uint32_t *>(event->data);
409*84e33947SAndroid Build Coastguard Worker 
410*84e33947SAndroid Build Coastguard Worker               // Placeholder parameters since linux PAL does not use this to
411*84e33947SAndroid Build Coastguard Worker               // generate response
412*84e33947SAndroid Build Coastguard Worker               struct chreWifiRangingTarget dummyRangingTarget = {
413*84e33947SAndroid Build Coastguard Worker                   .macAddress = {0x12, 0x34, 0x56, 0x78, 0x9a, 0xbc},
414*84e33947SAndroid Build Coastguard Worker                   .primaryChannel = 0xdef02468,
415*84e33947SAndroid Build Coastguard Worker                   .centerFreqPrimary = 0xace13579,
416*84e33947SAndroid Build Coastguard Worker                   .centerFreqSecondary = 0xbdf369cf,
417*84e33947SAndroid Build Coastguard Worker                   .channelWidth = 0x48,
418*84e33947SAndroid Build Coastguard Worker               };
419*84e33947SAndroid Build Coastguard Worker 
420*84e33947SAndroid Build Coastguard Worker               struct chreWifiRangingParams dummyRangingParams = {
421*84e33947SAndroid Build Coastguard Worker                   .targetListLen = 1,
422*84e33947SAndroid Build Coastguard Worker                   .targetList = &dummyRangingTarget,
423*84e33947SAndroid Build Coastguard Worker               };
424*84e33947SAndroid Build Coastguard Worker 
425*84e33947SAndroid Build Coastguard Worker               if (chreWifiRequestRangingAsync(&dummyRangingParams, &mCookie)) {
426*84e33947SAndroid Build Coastguard Worker                 mRequestTimer =
427*84e33947SAndroid Build Coastguard Worker                     chreTimerSet(CHRE_TEST_WIFI_RANGING_RESULT_TIMEOUT_NS,
428*84e33947SAndroid Build Coastguard Worker                                  nullptr, true /* oneShot */);
429*84e33947SAndroid Build Coastguard Worker                 success = mRequestTimer != CHRE_TIMER_INVALID;
430*84e33947SAndroid Build Coastguard Worker               }
431*84e33947SAndroid Build Coastguard Worker               TestEventQueueSingleton::get()->pushEvent(RANGING_REQUEST,
432*84e33947SAndroid Build Coastguard Worker                                                         success);
433*84e33947SAndroid Build Coastguard Worker           }
434*84e33947SAndroid Build Coastguard Worker         }
435*84e33947SAndroid Build Coastguard Worker       }
436*84e33947SAndroid Build Coastguard Worker     }
437*84e33947SAndroid Build Coastguard Worker 
438*84e33947SAndroid Build Coastguard Worker    protected:
439*84e33947SAndroid Build Coastguard Worker     uint32_t mCookie;
440*84e33947SAndroid Build Coastguard Worker     uint32_t mRequestTimer;
441*84e33947SAndroid Build Coastguard Worker   };
442*84e33947SAndroid Build Coastguard Worker 
443*84e33947SAndroid Build Coastguard Worker   uint64_t appId = loadNanoapp(MakeUnique<App>());
444*84e33947SAndroid Build Coastguard Worker 
445*84e33947SAndroid Build Coastguard Worker   uint32_t timeOutCookie = 0xdead;
446*84e33947SAndroid Build Coastguard Worker 
447*84e33947SAndroid Build Coastguard Worker   chrePalWifiEnableResponse(PalWifiAsyncRequestTypes::RANGING, false);
448*84e33947SAndroid Build Coastguard Worker   sendEventToNanoapp(appId, RANGING_REQUEST, timeOutCookie);
449*84e33947SAndroid Build Coastguard Worker   bool success;
450*84e33947SAndroid Build Coastguard Worker   waitForEvent(RANGING_REQUEST, &success);
451*84e33947SAndroid Build Coastguard Worker   EXPECT_TRUE(success);
452*84e33947SAndroid Build Coastguard Worker 
453*84e33947SAndroid Build Coastguard Worker   waitForEvent(REQUEST_TIMED_OUT);
454*84e33947SAndroid Build Coastguard Worker 
455*84e33947SAndroid Build Coastguard Worker   // Make sure that we can still request ranging after a timedout request
456*84e33947SAndroid Build Coastguard Worker   uint32_t successCookie = 0x0101;
457*84e33947SAndroid Build Coastguard Worker   chrePalWifiEnableResponse(PalWifiAsyncRequestTypes::RANGING, true);
458*84e33947SAndroid Build Coastguard Worker   sendEventToNanoapp(appId, RANGING_REQUEST, successCookie);
459*84e33947SAndroid Build Coastguard Worker   waitForEvent(RANGING_REQUEST, &success);
460*84e33947SAndroid Build Coastguard Worker   EXPECT_TRUE(success);
461*84e33947SAndroid Build Coastguard Worker 
462*84e33947SAndroid Build Coastguard Worker   uint32_t cookie;
463*84e33947SAndroid Build Coastguard Worker   waitForEvent(CHRE_EVENT_WIFI_ASYNC_RESULT, &cookie);
464*84e33947SAndroid Build Coastguard Worker   EXPECT_EQ(cookie, successCookie);
465*84e33947SAndroid Build Coastguard Worker 
466*84e33947SAndroid Build Coastguard Worker   unloadNanoapp(appId);
467*84e33947SAndroid Build Coastguard Worker }
468*84e33947SAndroid Build Coastguard Worker 
469*84e33947SAndroid Build Coastguard Worker }  // namespace
470*84e33947SAndroid Build Coastguard Worker }  // namespace chre
471