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_nan.h"
22*84e33947SAndroid Build Coastguard Worker #include "chre/platform/linux/pal_wifi.h"
23*84e33947SAndroid Build Coastguard Worker #include "chre/platform/log.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/wifi.h"
27*84e33947SAndroid Build Coastguard Worker #include "gtest/gtest.h"
28*84e33947SAndroid Build Coastguard Worker #include "test_base.h"
29*84e33947SAndroid Build Coastguard Worker #include "test_event.h"
30*84e33947SAndroid Build Coastguard Worker #include "test_event_queue.h"
31*84e33947SAndroid Build Coastguard Worker #include "test_util.h"
32*84e33947SAndroid Build Coastguard Worker
33*84e33947SAndroid Build Coastguard Worker namespace chre {
34*84e33947SAndroid Build Coastguard Worker namespace {
35*84e33947SAndroid Build Coastguard Worker using namespace std::chrono_literals;
36*84e33947SAndroid Build Coastguard Worker
37*84e33947SAndroid Build Coastguard Worker CREATE_CHRE_TEST_EVENT(SCAN_REQUEST, 20);
38*84e33947SAndroid Build Coastguard Worker
39*84e33947SAndroid Build Coastguard Worker struct WifiAsyncData {
40*84e33947SAndroid Build Coastguard Worker const uint32_t *cookie;
41*84e33947SAndroid Build Coastguard Worker chreError errorCode;
42*84e33947SAndroid Build Coastguard Worker };
43*84e33947SAndroid Build Coastguard Worker
44*84e33947SAndroid Build Coastguard Worker constexpr uint64_t kAppOneId = 0x0123456789000001;
45*84e33947SAndroid Build Coastguard Worker constexpr uint64_t kAppTwoId = 0x0123456789000002;
46*84e33947SAndroid Build Coastguard Worker
47*84e33947SAndroid Build Coastguard Worker class WifiScanRequestQueueTestBase : public TestBase {
48*84e33947SAndroid Build Coastguard Worker public:
SetUp()49*84e33947SAndroid Build Coastguard Worker void SetUp() {
50*84e33947SAndroid Build Coastguard Worker TestBase::SetUp();
51*84e33947SAndroid Build Coastguard Worker // Add delay to make sure the requests are queued.
52*84e33947SAndroid Build Coastguard Worker chrePalWifiDelayResponse(PalWifiAsyncRequestTypes::SCAN,
53*84e33947SAndroid Build Coastguard Worker /* milliseconds= */ 100ms);
54*84e33947SAndroid Build Coastguard Worker }
55*84e33947SAndroid Build Coastguard Worker
TearDown()56*84e33947SAndroid Build Coastguard Worker void TearDown() {
57*84e33947SAndroid Build Coastguard Worker TestBase::TearDown();
58*84e33947SAndroid Build Coastguard Worker chrePalWifiDelayResponse(PalWifiAsyncRequestTypes::SCAN,
59*84e33947SAndroid Build Coastguard Worker /* milliseconds= */ 0ms);
60*84e33947SAndroid Build Coastguard Worker }
61*84e33947SAndroid Build Coastguard Worker };
62*84e33947SAndroid Build Coastguard Worker
63*84e33947SAndroid Build Coastguard Worker class WifiScanTestNanoapp : public TestNanoapp {
64*84e33947SAndroid Build Coastguard Worker public:
WifiScanTestNanoapp()65*84e33947SAndroid Build Coastguard Worker WifiScanTestNanoapp()
66*84e33947SAndroid Build Coastguard Worker : TestNanoapp(
67*84e33947SAndroid Build Coastguard Worker TestNanoappInfo{.perms = NanoappPermissions::CHRE_PERMS_WIFI}) {}
68*84e33947SAndroid Build Coastguard Worker
WifiScanTestNanoapp(uint64_t id)69*84e33947SAndroid Build Coastguard Worker explicit WifiScanTestNanoapp(uint64_t id)
70*84e33947SAndroid Build Coastguard Worker : TestNanoapp(TestNanoappInfo{
71*84e33947SAndroid Build Coastguard Worker .id = id, .perms = NanoappPermissions::CHRE_PERMS_WIFI}) {}
72*84e33947SAndroid Build Coastguard Worker
handleEvent(uint32_t,uint16_t eventType,const void * eventData)73*84e33947SAndroid Build Coastguard Worker void handleEvent(uint32_t, uint16_t eventType,
74*84e33947SAndroid Build Coastguard Worker const void *eventData) override {
75*84e33947SAndroid Build Coastguard Worker switch (eventType) {
76*84e33947SAndroid Build Coastguard Worker case CHRE_EVENT_WIFI_ASYNC_RESULT: {
77*84e33947SAndroid Build Coastguard Worker auto *event = static_cast<const chreAsyncResult *>(eventData);
78*84e33947SAndroid Build Coastguard Worker TestEventQueueSingleton::get()->pushEvent(
79*84e33947SAndroid Build Coastguard Worker CHRE_EVENT_WIFI_ASYNC_RESULT,
80*84e33947SAndroid Build Coastguard Worker WifiAsyncData{
81*84e33947SAndroid Build Coastguard Worker .cookie = static_cast<const uint32_t *>(event->cookie),
82*84e33947SAndroid Build Coastguard Worker .errorCode = static_cast<chreError>(event->errorCode)});
83*84e33947SAndroid Build Coastguard Worker break;
84*84e33947SAndroid Build Coastguard Worker }
85*84e33947SAndroid Build Coastguard Worker
86*84e33947SAndroid Build Coastguard Worker case CHRE_EVENT_WIFI_SCAN_RESULT: {
87*84e33947SAndroid Build Coastguard Worker TestEventQueueSingleton::get()->pushEvent(CHRE_EVENT_WIFI_SCAN_RESULT);
88*84e33947SAndroid Build Coastguard Worker break;
89*84e33947SAndroid Build Coastguard Worker }
90*84e33947SAndroid Build Coastguard Worker
91*84e33947SAndroid Build Coastguard Worker case CHRE_EVENT_TEST_EVENT: {
92*84e33947SAndroid Build Coastguard Worker auto event = static_cast<const TestEvent *>(eventData);
93*84e33947SAndroid Build Coastguard Worker switch (event->type) {
94*84e33947SAndroid Build Coastguard Worker case SCAN_REQUEST:
95*84e33947SAndroid Build Coastguard Worker bool success = false;
96*84e33947SAndroid Build Coastguard Worker if (mNextFreeCookieIndex < kMaxPendingCookie) {
97*84e33947SAndroid Build Coastguard Worker mCookies[mNextFreeCookieIndex] =
98*84e33947SAndroid Build Coastguard Worker *static_cast<uint32_t *>(event->data);
99*84e33947SAndroid Build Coastguard Worker success = chreWifiRequestScanAsyncDefault(
100*84e33947SAndroid Build Coastguard Worker &mCookies[mNextFreeCookieIndex]);
101*84e33947SAndroid Build Coastguard Worker mNextFreeCookieIndex++;
102*84e33947SAndroid Build Coastguard Worker } else {
103*84e33947SAndroid Build Coastguard Worker LOGE("Too many cookies passed from test body!");
104*84e33947SAndroid Build Coastguard Worker }
105*84e33947SAndroid Build Coastguard Worker TestEventQueueSingleton::get()->pushEvent(SCAN_REQUEST, success);
106*84e33947SAndroid Build Coastguard Worker }
107*84e33947SAndroid Build Coastguard Worker }
108*84e33947SAndroid Build Coastguard Worker }
109*84e33947SAndroid Build Coastguard Worker }
110*84e33947SAndroid Build Coastguard Worker
111*84e33947SAndroid Build Coastguard Worker protected:
112*84e33947SAndroid Build Coastguard Worker static constexpr uint8_t kMaxPendingCookie = 10;
113*84e33947SAndroid Build Coastguard Worker uint32_t mCookies[kMaxPendingCookie];
114*84e33947SAndroid Build Coastguard Worker uint8_t mNextFreeCookieIndex = 0;
115*84e33947SAndroid Build Coastguard Worker };
116*84e33947SAndroid Build Coastguard Worker
TEST_F(TestBase,WifiScanBasicSettingTest)117*84e33947SAndroid Build Coastguard Worker TEST_F(TestBase, WifiScanBasicSettingTest) {
118*84e33947SAndroid Build Coastguard Worker uint64_t appId = loadNanoapp(MakeUnique<WifiScanTestNanoapp>());
119*84e33947SAndroid Build Coastguard Worker
120*84e33947SAndroid Build Coastguard Worker EventLoopManagerSingleton::get()->getSettingManager().postSettingChange(
121*84e33947SAndroid Build Coastguard Worker Setting::WIFI_AVAILABLE, true /* enabled */);
122*84e33947SAndroid Build Coastguard Worker
123*84e33947SAndroid Build Coastguard Worker constexpr uint32_t firstCookie = 0x1010;
124*84e33947SAndroid Build Coastguard Worker bool success;
125*84e33947SAndroid Build Coastguard Worker WifiAsyncData wifiAsyncData;
126*84e33947SAndroid Build Coastguard Worker
127*84e33947SAndroid Build Coastguard Worker sendEventToNanoapp(appId, SCAN_REQUEST, firstCookie);
128*84e33947SAndroid Build Coastguard Worker waitForEvent(SCAN_REQUEST, &success);
129*84e33947SAndroid Build Coastguard Worker EXPECT_TRUE(success);
130*84e33947SAndroid Build Coastguard Worker
131*84e33947SAndroid Build Coastguard Worker waitForEvent(CHRE_EVENT_WIFI_ASYNC_RESULT, &wifiAsyncData);
132*84e33947SAndroid Build Coastguard Worker EXPECT_EQ(wifiAsyncData.errorCode, CHRE_ERROR_NONE);
133*84e33947SAndroid Build Coastguard Worker EXPECT_EQ(*wifiAsyncData.cookie, firstCookie);
134*84e33947SAndroid Build Coastguard Worker waitForEvent(CHRE_EVENT_WIFI_SCAN_RESULT);
135*84e33947SAndroid Build Coastguard Worker
136*84e33947SAndroid Build Coastguard Worker EventLoopManagerSingleton::get()->getSettingManager().postSettingChange(
137*84e33947SAndroid Build Coastguard Worker Setting::WIFI_AVAILABLE, false /* enabled */);
138*84e33947SAndroid Build Coastguard Worker
139*84e33947SAndroid Build Coastguard Worker constexpr uint32_t secondCookie = 0x2020;
140*84e33947SAndroid Build Coastguard Worker sendEventToNanoapp(appId, SCAN_REQUEST, secondCookie);
141*84e33947SAndroid Build Coastguard Worker waitForEvent(SCAN_REQUEST, &success);
142*84e33947SAndroid Build Coastguard Worker EXPECT_TRUE(success);
143*84e33947SAndroid Build Coastguard Worker
144*84e33947SAndroid Build Coastguard Worker waitForEvent(CHRE_EVENT_WIFI_ASYNC_RESULT, &wifiAsyncData);
145*84e33947SAndroid Build Coastguard Worker EXPECT_EQ(wifiAsyncData.errorCode, CHRE_ERROR_FUNCTION_DISABLED);
146*84e33947SAndroid Build Coastguard Worker EXPECT_EQ(*wifiAsyncData.cookie, secondCookie);
147*84e33947SAndroid Build Coastguard Worker
148*84e33947SAndroid Build Coastguard Worker EventLoopManagerSingleton::get()->getSettingManager().postSettingChange(
149*84e33947SAndroid Build Coastguard Worker Setting::WIFI_AVAILABLE, true /* enabled */);
150*84e33947SAndroid Build Coastguard Worker unloadNanoapp(appId);
151*84e33947SAndroid Build Coastguard Worker }
152*84e33947SAndroid Build Coastguard Worker
TEST_F(WifiScanRequestQueueTestBase,WifiQueuedScanSettingChangeTest)153*84e33947SAndroid Build Coastguard Worker TEST_F(WifiScanRequestQueueTestBase, WifiQueuedScanSettingChangeTest) {
154*84e33947SAndroid Build Coastguard Worker CREATE_CHRE_TEST_EVENT(CONCURRENT_NANOAPP_RECEIVED_EXPECTED_ASYNC_EVENT_COUNT,
155*84e33947SAndroid Build Coastguard Worker 1);
156*84e33947SAndroid Build Coastguard Worker CREATE_CHRE_TEST_EVENT(CONCURRENT_NANOAPP_READ_ASYNC_EVENT, 2);
157*84e33947SAndroid Build Coastguard Worker // Expecting to receive two event, one from each nanoapp.
158*84e33947SAndroid Build Coastguard Worker constexpr uint8_t kExpectedReceiveAsyncResultCount = 2;
159*84e33947SAndroid Build Coastguard Worker // receivedAsyncEventCount is shared across apps and must be static.
160*84e33947SAndroid Build Coastguard Worker // But we want it initialized each time the test is executed.
161*84e33947SAndroid Build Coastguard Worker static uint8_t receivedAsyncEventCount;
162*84e33947SAndroid Build Coastguard Worker receivedAsyncEventCount = 0;
163*84e33947SAndroid Build Coastguard Worker
164*84e33947SAndroid Build Coastguard Worker class WifiScanTestConcurrentNanoapp : public TestNanoapp {
165*84e33947SAndroid Build Coastguard Worker public:
166*84e33947SAndroid Build Coastguard Worker explicit WifiScanTestConcurrentNanoapp(uint64_t id)
167*84e33947SAndroid Build Coastguard Worker : TestNanoapp(TestNanoappInfo{
168*84e33947SAndroid Build Coastguard Worker .id = id, .perms = NanoappPermissions::CHRE_PERMS_WIFI}) {}
169*84e33947SAndroid Build Coastguard Worker
170*84e33947SAndroid Build Coastguard Worker void handleEvent(uint32_t, uint16_t eventType,
171*84e33947SAndroid Build Coastguard Worker const void *eventData) override {
172*84e33947SAndroid Build Coastguard Worker switch (eventType) {
173*84e33947SAndroid Build Coastguard Worker case CHRE_EVENT_WIFI_ASYNC_RESULT: {
174*84e33947SAndroid Build Coastguard Worker auto *event = static_cast<const chreAsyncResult *>(eventData);
175*84e33947SAndroid Build Coastguard Worker mReceivedAsyncResult = WifiAsyncData{
176*84e33947SAndroid Build Coastguard Worker .cookie = static_cast<const uint32_t *>(event->cookie),
177*84e33947SAndroid Build Coastguard Worker .errorCode = static_cast<chreError>(event->errorCode)};
178*84e33947SAndroid Build Coastguard Worker ++receivedAsyncEventCount;
179*84e33947SAndroid Build Coastguard Worker break;
180*84e33947SAndroid Build Coastguard Worker }
181*84e33947SAndroid Build Coastguard Worker
182*84e33947SAndroid Build Coastguard Worker case CHRE_EVENT_TEST_EVENT: {
183*84e33947SAndroid Build Coastguard Worker auto event = static_cast<const TestEvent *>(eventData);
184*84e33947SAndroid Build Coastguard Worker bool success = false;
185*84e33947SAndroid Build Coastguard Worker switch (event->type) {
186*84e33947SAndroid Build Coastguard Worker case SCAN_REQUEST:
187*84e33947SAndroid Build Coastguard Worker mSentCookie = *static_cast<uint32_t *>(event->data);
188*84e33947SAndroid Build Coastguard Worker success = chreWifiRequestScanAsyncDefault(&(mSentCookie));
189*84e33947SAndroid Build Coastguard Worker TestEventQueueSingleton::get()->pushEvent(SCAN_REQUEST, success);
190*84e33947SAndroid Build Coastguard Worker break;
191*84e33947SAndroid Build Coastguard Worker case CONCURRENT_NANOAPP_READ_ASYNC_EVENT:
192*84e33947SAndroid Build Coastguard Worker TestEventQueueSingleton::get()->pushEvent(
193*84e33947SAndroid Build Coastguard Worker CONCURRENT_NANOAPP_READ_ASYNC_EVENT, mReceivedAsyncResult);
194*84e33947SAndroid Build Coastguard Worker break;
195*84e33947SAndroid Build Coastguard Worker }
196*84e33947SAndroid Build Coastguard Worker }
197*84e33947SAndroid Build Coastguard Worker }
198*84e33947SAndroid Build Coastguard Worker
199*84e33947SAndroid Build Coastguard Worker if (receivedAsyncEventCount == kExpectedReceiveAsyncResultCount) {
200*84e33947SAndroid Build Coastguard Worker TestEventQueueSingleton::get()->pushEvent(
201*84e33947SAndroid Build Coastguard Worker CONCURRENT_NANOAPP_RECEIVED_EXPECTED_ASYNC_EVENT_COUNT);
202*84e33947SAndroid Build Coastguard Worker }
203*84e33947SAndroid Build Coastguard Worker }
204*84e33947SAndroid Build Coastguard Worker
205*84e33947SAndroid Build Coastguard Worker protected:
206*84e33947SAndroid Build Coastguard Worker uint32_t mSentCookie;
207*84e33947SAndroid Build Coastguard Worker WifiAsyncData mReceivedAsyncResult;
208*84e33947SAndroid Build Coastguard Worker };
209*84e33947SAndroid Build Coastguard Worker
210*84e33947SAndroid Build Coastguard Worker uint64_t appOneId =
211*84e33947SAndroid Build Coastguard Worker loadNanoapp(MakeUnique<WifiScanTestConcurrentNanoapp>(kAppOneId));
212*84e33947SAndroid Build Coastguard Worker uint64_t appTwoId =
213*84e33947SAndroid Build Coastguard Worker loadNanoapp(MakeUnique<WifiScanTestConcurrentNanoapp>(kAppTwoId));
214*84e33947SAndroid Build Coastguard Worker
215*84e33947SAndroid Build Coastguard Worker constexpr uint32_t appOneRequestCookie = 0x1010;
216*84e33947SAndroid Build Coastguard Worker constexpr uint32_t appTwoRequestCookie = 0x2020;
217*84e33947SAndroid Build Coastguard Worker bool success;
218*84e33947SAndroid Build Coastguard Worker sendEventToNanoapp(appOneId, SCAN_REQUEST, appOneRequestCookie);
219*84e33947SAndroid Build Coastguard Worker waitForEvent(SCAN_REQUEST, &success);
220*84e33947SAndroid Build Coastguard Worker EXPECT_TRUE(success);
221*84e33947SAndroid Build Coastguard Worker sendEventToNanoapp(appTwoId, SCAN_REQUEST, appTwoRequestCookie);
222*84e33947SAndroid Build Coastguard Worker waitForEvent(SCAN_REQUEST, &success);
223*84e33947SAndroid Build Coastguard Worker EXPECT_TRUE(success);
224*84e33947SAndroid Build Coastguard Worker
225*84e33947SAndroid Build Coastguard Worker EventLoopManagerSingleton::get()->getSettingManager().postSettingChange(
226*84e33947SAndroid Build Coastguard Worker Setting::WIFI_AVAILABLE, false /* enabled */);
227*84e33947SAndroid Build Coastguard Worker
228*84e33947SAndroid Build Coastguard Worker // We need to make sure that each nanoapp has received one async result before
229*84e33947SAndroid Build Coastguard Worker // further analysis.
230*84e33947SAndroid Build Coastguard Worker waitForEvent(CONCURRENT_NANOAPP_RECEIVED_EXPECTED_ASYNC_EVENT_COUNT);
231*84e33947SAndroid Build Coastguard Worker
232*84e33947SAndroid Build Coastguard Worker WifiAsyncData wifiAsyncData;
233*84e33947SAndroid Build Coastguard Worker sendEventToNanoapp(appOneId, CONCURRENT_NANOAPP_READ_ASYNC_EVENT);
234*84e33947SAndroid Build Coastguard Worker waitForEvent(CONCURRENT_NANOAPP_READ_ASYNC_EVENT, &wifiAsyncData);
235*84e33947SAndroid Build Coastguard Worker EXPECT_EQ(wifiAsyncData.errorCode, CHRE_ERROR_NONE);
236*84e33947SAndroid Build Coastguard Worker EXPECT_EQ(*wifiAsyncData.cookie, appOneRequestCookie);
237*84e33947SAndroid Build Coastguard Worker
238*84e33947SAndroid Build Coastguard Worker sendEventToNanoapp(appTwoId, CONCURRENT_NANOAPP_READ_ASYNC_EVENT);
239*84e33947SAndroid Build Coastguard Worker waitForEvent(CONCURRENT_NANOAPP_READ_ASYNC_EVENT, &wifiAsyncData);
240*84e33947SAndroid Build Coastguard Worker EXPECT_EQ(wifiAsyncData.errorCode, CHRE_ERROR_FUNCTION_DISABLED);
241*84e33947SAndroid Build Coastguard Worker EXPECT_EQ(*wifiAsyncData.cookie, appTwoRequestCookie);
242*84e33947SAndroid Build Coastguard Worker
243*84e33947SAndroid Build Coastguard Worker EventLoopManagerSingleton::get()->getSettingManager().postSettingChange(
244*84e33947SAndroid Build Coastguard Worker Setting::WIFI_AVAILABLE, true /* enabled */);
245*84e33947SAndroid Build Coastguard Worker
246*84e33947SAndroid Build Coastguard Worker unloadNanoapp(appOneId);
247*84e33947SAndroid Build Coastguard Worker unloadNanoapp(appTwoId);
248*84e33947SAndroid Build Coastguard Worker }
249*84e33947SAndroid Build Coastguard Worker
TEST_F(WifiScanRequestQueueTestBase,WifiScanRejectRequestFromSameNanoapp)250*84e33947SAndroid Build Coastguard Worker TEST_F(WifiScanRequestQueueTestBase, WifiScanRejectRequestFromSameNanoapp) {
251*84e33947SAndroid Build Coastguard Worker CREATE_CHRE_TEST_EVENT(RECEIVED_ALL_EXPECTED_EVENTS, 1);
252*84e33947SAndroid Build Coastguard Worker CREATE_CHRE_TEST_EVENT(READ_ASYNC_EVENT, 2);
253*84e33947SAndroid Build Coastguard Worker
254*84e33947SAndroid Build Coastguard Worker static constexpr uint8_t kExpectedReceivedScanRequestCount = 2;
255*84e33947SAndroid Build Coastguard Worker
256*84e33947SAndroid Build Coastguard Worker class WifiScanTestBufferedAsyncResultNanoapp : public TestNanoapp {
257*84e33947SAndroid Build Coastguard Worker public:
258*84e33947SAndroid Build Coastguard Worker WifiScanTestBufferedAsyncResultNanoapp()
259*84e33947SAndroid Build Coastguard Worker : TestNanoapp(
260*84e33947SAndroid Build Coastguard Worker TestNanoappInfo{.perms = NanoappPermissions::CHRE_PERMS_WIFI}) {}
261*84e33947SAndroid Build Coastguard Worker
262*84e33947SAndroid Build Coastguard Worker void handleEvent(uint32_t, uint16_t eventType,
263*84e33947SAndroid Build Coastguard Worker const void *eventData) override {
264*84e33947SAndroid Build Coastguard Worker switch (eventType) {
265*84e33947SAndroid Build Coastguard Worker case CHRE_EVENT_WIFI_ASYNC_RESULT: {
266*84e33947SAndroid Build Coastguard Worker auto *event = static_cast<const chreAsyncResult *>(eventData);
267*84e33947SAndroid Build Coastguard Worker mReceivedAsyncResult = WifiAsyncData{
268*84e33947SAndroid Build Coastguard Worker .cookie = static_cast<const uint32_t *>(event->cookie),
269*84e33947SAndroid Build Coastguard Worker .errorCode = static_cast<chreError>(event->errorCode)};
270*84e33947SAndroid Build Coastguard Worker ++mReceivedAsyncEventCount;
271*84e33947SAndroid Build Coastguard Worker break;
272*84e33947SAndroid Build Coastguard Worker }
273*84e33947SAndroid Build Coastguard Worker
274*84e33947SAndroid Build Coastguard Worker case CHRE_EVENT_TEST_EVENT: {
275*84e33947SAndroid Build Coastguard Worker auto event = static_cast<const TestEvent *>(eventData);
276*84e33947SAndroid Build Coastguard Worker bool success = false;
277*84e33947SAndroid Build Coastguard Worker switch (event->type) {
278*84e33947SAndroid Build Coastguard Worker case SCAN_REQUEST:
279*84e33947SAndroid Build Coastguard Worker if (mReceivedScanRequestCount >=
280*84e33947SAndroid Build Coastguard Worker kExpectedReceivedScanRequestCount) {
281*84e33947SAndroid Build Coastguard Worker LOGE("Asking too many scan request");
282*84e33947SAndroid Build Coastguard Worker } else {
283*84e33947SAndroid Build Coastguard Worker mReceivedCookies[mReceivedScanRequestCount] =
284*84e33947SAndroid Build Coastguard Worker *static_cast<uint32_t *>(event->data);
285*84e33947SAndroid Build Coastguard Worker success = chreWifiRequestScanAsyncDefault(
286*84e33947SAndroid Build Coastguard Worker &(mReceivedCookies[mReceivedScanRequestCount]));
287*84e33947SAndroid Build Coastguard Worker ++mReceivedScanRequestCount;
288*84e33947SAndroid Build Coastguard Worker TestEventQueueSingleton::get()->pushEvent(SCAN_REQUEST,
289*84e33947SAndroid Build Coastguard Worker success);
290*84e33947SAndroid Build Coastguard Worker }
291*84e33947SAndroid Build Coastguard Worker break;
292*84e33947SAndroid Build Coastguard Worker case READ_ASYNC_EVENT:
293*84e33947SAndroid Build Coastguard Worker TestEventQueueSingleton::get()->pushEvent(READ_ASYNC_EVENT,
294*84e33947SAndroid Build Coastguard Worker mReceivedAsyncResult);
295*84e33947SAndroid Build Coastguard Worker break;
296*84e33947SAndroid Build Coastguard Worker }
297*84e33947SAndroid Build Coastguard Worker }
298*84e33947SAndroid Build Coastguard Worker }
299*84e33947SAndroid Build Coastguard Worker if (mReceivedAsyncEventCount == kExpectedReceivedAsyncResultCount &&
300*84e33947SAndroid Build Coastguard Worker mReceivedScanRequestCount == kExpectedReceivedScanRequestCount) {
301*84e33947SAndroid Build Coastguard Worker TestEventQueueSingleton::get()->pushEvent(RECEIVED_ALL_EXPECTED_EVENTS);
302*84e33947SAndroid Build Coastguard Worker }
303*84e33947SAndroid Build Coastguard Worker }
304*84e33947SAndroid Build Coastguard Worker
305*84e33947SAndroid Build Coastguard Worker protected:
306*84e33947SAndroid Build Coastguard Worker // We are only expecting to receive one async result since the second
307*84e33947SAndroid Build Coastguard Worker // request is expected to fail.
308*84e33947SAndroid Build Coastguard Worker const uint8_t kExpectedReceivedAsyncResultCount = 1;
309*84e33947SAndroid Build Coastguard Worker uint8_t mReceivedAsyncEventCount = 0;
310*84e33947SAndroid Build Coastguard Worker uint8_t mReceivedScanRequestCount = 0;
311*84e33947SAndroid Build Coastguard Worker
312*84e33947SAndroid Build Coastguard Worker // We need to have two cookie storage to separate the two scan request.
313*84e33947SAndroid Build Coastguard Worker uint32_t mReceivedCookies[kExpectedReceivedScanRequestCount];
314*84e33947SAndroid Build Coastguard Worker WifiAsyncData mReceivedAsyncResult;
315*84e33947SAndroid Build Coastguard Worker };
316*84e33947SAndroid Build Coastguard Worker
317*84e33947SAndroid Build Coastguard Worker uint64_t appId =
318*84e33947SAndroid Build Coastguard Worker loadNanoapp(MakeUnique<WifiScanTestBufferedAsyncResultNanoapp>());
319*84e33947SAndroid Build Coastguard Worker
320*84e33947SAndroid Build Coastguard Worker constexpr uint32_t kFirstRequestCookie = 0x1010;
321*84e33947SAndroid Build Coastguard Worker constexpr uint32_t kSecondRequestCookie = 0x2020;
322*84e33947SAndroid Build Coastguard Worker bool success;
323*84e33947SAndroid Build Coastguard Worker sendEventToNanoapp(appId, SCAN_REQUEST, kFirstRequestCookie);
324*84e33947SAndroid Build Coastguard Worker waitForEvent(SCAN_REQUEST, &success);
325*84e33947SAndroid Build Coastguard Worker EXPECT_TRUE(success);
326*84e33947SAndroid Build Coastguard Worker sendEventToNanoapp(appId, SCAN_REQUEST, kSecondRequestCookie);
327*84e33947SAndroid Build Coastguard Worker waitForEvent(SCAN_REQUEST, &success);
328*84e33947SAndroid Build Coastguard Worker EXPECT_FALSE(success);
329*84e33947SAndroid Build Coastguard Worker
330*84e33947SAndroid Build Coastguard Worker // We need to make sure that the nanoapp has received one async result and did
331*84e33947SAndroid Build Coastguard Worker // two scan requests before further analysis.
332*84e33947SAndroid Build Coastguard Worker waitForEvent(RECEIVED_ALL_EXPECTED_EVENTS);
333*84e33947SAndroid Build Coastguard Worker
334*84e33947SAndroid Build Coastguard Worker WifiAsyncData wifiAsyncData;
335*84e33947SAndroid Build Coastguard Worker sendEventToNanoapp(appId, READ_ASYNC_EVENT);
336*84e33947SAndroid Build Coastguard Worker waitForEvent(READ_ASYNC_EVENT, &wifiAsyncData);
337*84e33947SAndroid Build Coastguard Worker EXPECT_EQ(wifiAsyncData.errorCode, CHRE_ERROR_NONE);
338*84e33947SAndroid Build Coastguard Worker EXPECT_EQ(*wifiAsyncData.cookie, kFirstRequestCookie);
339*84e33947SAndroid Build Coastguard Worker
340*84e33947SAndroid Build Coastguard Worker unloadNanoapp(appId);
341*84e33947SAndroid Build Coastguard Worker }
342*84e33947SAndroid Build Coastguard Worker
TEST_F(WifiScanRequestQueueTestBase,WifiScanActiveScanFromDistinctNanoapps)343*84e33947SAndroid Build Coastguard Worker TEST_F(WifiScanRequestQueueTestBase, WifiScanActiveScanFromDistinctNanoapps) {
344*84e33947SAndroid Build Coastguard Worker CREATE_CHRE_TEST_EVENT(CONCURRENT_NANOAPP_RECEIVED_EXPECTED_ASYNC_EVENT_COUNT,
345*84e33947SAndroid Build Coastguard Worker 1);
346*84e33947SAndroid Build Coastguard Worker CREATE_CHRE_TEST_EVENT(CONCURRENT_NANOAPP_READ_COOKIE, 2);
347*84e33947SAndroid Build Coastguard Worker
348*84e33947SAndroid Build Coastguard Worker constexpr uint8_t kExpectedReceiveAsyncResultCount = 2;
349*84e33947SAndroid Build Coastguard Worker // receivedCookieCount is shared across apps and must be static.
350*84e33947SAndroid Build Coastguard Worker // But we want it initialized each time the test is executed.
351*84e33947SAndroid Build Coastguard Worker static uint8_t receivedCookieCount;
352*84e33947SAndroid Build Coastguard Worker receivedCookieCount = 0;
353*84e33947SAndroid Build Coastguard Worker
354*84e33947SAndroid Build Coastguard Worker class WifiScanTestConcurrentNanoapp : public TestNanoapp {
355*84e33947SAndroid Build Coastguard Worker public:
356*84e33947SAndroid Build Coastguard Worker explicit WifiScanTestConcurrentNanoapp(uint64_t id)
357*84e33947SAndroid Build Coastguard Worker : TestNanoapp(TestNanoappInfo{
358*84e33947SAndroid Build Coastguard Worker .id = id, .perms = NanoappPermissions::CHRE_PERMS_WIFI}) {}
359*84e33947SAndroid Build Coastguard Worker
360*84e33947SAndroid Build Coastguard Worker void handleEvent(uint32_t, uint16_t eventType,
361*84e33947SAndroid Build Coastguard Worker const void *eventData) override {
362*84e33947SAndroid Build Coastguard Worker switch (eventType) {
363*84e33947SAndroid Build Coastguard Worker case CHRE_EVENT_WIFI_ASYNC_RESULT: {
364*84e33947SAndroid Build Coastguard Worker auto *event = static_cast<const chreAsyncResult *>(eventData);
365*84e33947SAndroid Build Coastguard Worker if (event->errorCode == CHRE_ERROR_NONE) {
366*84e33947SAndroid Build Coastguard Worker mReceivedCookie = *static_cast<const uint32_t *>(event->cookie);
367*84e33947SAndroid Build Coastguard Worker ++receivedCookieCount;
368*84e33947SAndroid Build Coastguard Worker } else {
369*84e33947SAndroid Build Coastguard Worker LOGE("Received failed async result");
370*84e33947SAndroid Build Coastguard Worker }
371*84e33947SAndroid Build Coastguard Worker
372*84e33947SAndroid Build Coastguard Worker if (receivedCookieCount == kExpectedReceiveAsyncResultCount) {
373*84e33947SAndroid Build Coastguard Worker TestEventQueueSingleton::get()->pushEvent(
374*84e33947SAndroid Build Coastguard Worker CONCURRENT_NANOAPP_RECEIVED_EXPECTED_ASYNC_EVENT_COUNT);
375*84e33947SAndroid Build Coastguard Worker }
376*84e33947SAndroid Build Coastguard Worker break;
377*84e33947SAndroid Build Coastguard Worker }
378*84e33947SAndroid Build Coastguard Worker
379*84e33947SAndroid Build Coastguard Worker case CHRE_EVENT_TEST_EVENT: {
380*84e33947SAndroid Build Coastguard Worker auto event = static_cast<const TestEvent *>(eventData);
381*84e33947SAndroid Build Coastguard Worker bool success = false;
382*84e33947SAndroid Build Coastguard Worker switch (event->type) {
383*84e33947SAndroid Build Coastguard Worker case SCAN_REQUEST:
384*84e33947SAndroid Build Coastguard Worker mSentCookie = *static_cast<uint32_t *>(event->data);
385*84e33947SAndroid Build Coastguard Worker success = chreWifiRequestScanAsyncDefault(&(mSentCookie));
386*84e33947SAndroid Build Coastguard Worker TestEventQueueSingleton::get()->pushEvent(SCAN_REQUEST, success);
387*84e33947SAndroid Build Coastguard Worker break;
388*84e33947SAndroid Build Coastguard Worker case CONCURRENT_NANOAPP_READ_COOKIE:
389*84e33947SAndroid Build Coastguard Worker TestEventQueueSingleton::get()->pushEvent(
390*84e33947SAndroid Build Coastguard Worker CONCURRENT_NANOAPP_READ_COOKIE, mReceivedCookie);
391*84e33947SAndroid Build Coastguard Worker break;
392*84e33947SAndroid Build Coastguard Worker }
393*84e33947SAndroid Build Coastguard Worker }
394*84e33947SAndroid Build Coastguard Worker }
395*84e33947SAndroid Build Coastguard Worker }
396*84e33947SAndroid Build Coastguard Worker
397*84e33947SAndroid Build Coastguard Worker protected:
398*84e33947SAndroid Build Coastguard Worker uint32_t mSentCookie;
399*84e33947SAndroid Build Coastguard Worker uint32_t mReceivedCookie;
400*84e33947SAndroid Build Coastguard Worker };
401*84e33947SAndroid Build Coastguard Worker
402*84e33947SAndroid Build Coastguard Worker uint64_t appOneId =
403*84e33947SAndroid Build Coastguard Worker loadNanoapp(MakeUnique<WifiScanTestConcurrentNanoapp>(kAppOneId));
404*84e33947SAndroid Build Coastguard Worker uint64_t appTwoId =
405*84e33947SAndroid Build Coastguard Worker loadNanoapp(MakeUnique<WifiScanTestConcurrentNanoapp>(kAppTwoId));
406*84e33947SAndroid Build Coastguard Worker
407*84e33947SAndroid Build Coastguard Worker constexpr uint32_t kAppOneRequestCookie = 0x1010;
408*84e33947SAndroid Build Coastguard Worker constexpr uint32_t kAppTwoRequestCookie = 0x2020;
409*84e33947SAndroid Build Coastguard Worker bool success;
410*84e33947SAndroid Build Coastguard Worker sendEventToNanoapp(appOneId, SCAN_REQUEST, kAppOneRequestCookie);
411*84e33947SAndroid Build Coastguard Worker waitForEvent(SCAN_REQUEST, &success);
412*84e33947SAndroid Build Coastguard Worker EXPECT_TRUE(success);
413*84e33947SAndroid Build Coastguard Worker sendEventToNanoapp(appTwoId, SCAN_REQUEST, kAppTwoRequestCookie);
414*84e33947SAndroid Build Coastguard Worker waitForEvent(SCAN_REQUEST, &success);
415*84e33947SAndroid Build Coastguard Worker EXPECT_TRUE(success);
416*84e33947SAndroid Build Coastguard Worker
417*84e33947SAndroid Build Coastguard Worker waitForEvent(CONCURRENT_NANOAPP_RECEIVED_EXPECTED_ASYNC_EVENT_COUNT);
418*84e33947SAndroid Build Coastguard Worker
419*84e33947SAndroid Build Coastguard Worker uint32_t receivedCookie;
420*84e33947SAndroid Build Coastguard Worker sendEventToNanoapp(appOneId, CONCURRENT_NANOAPP_READ_COOKIE);
421*84e33947SAndroid Build Coastguard Worker waitForEvent(CONCURRENT_NANOAPP_READ_COOKIE, &receivedCookie);
422*84e33947SAndroid Build Coastguard Worker EXPECT_EQ(kAppOneRequestCookie, receivedCookie);
423*84e33947SAndroid Build Coastguard Worker
424*84e33947SAndroid Build Coastguard Worker sendEventToNanoapp(appTwoId, CONCURRENT_NANOAPP_READ_COOKIE);
425*84e33947SAndroid Build Coastguard Worker waitForEvent(CONCURRENT_NANOAPP_READ_COOKIE, &receivedCookie);
426*84e33947SAndroid Build Coastguard Worker EXPECT_EQ(kAppTwoRequestCookie, receivedCookie);
427*84e33947SAndroid Build Coastguard Worker
428*84e33947SAndroid Build Coastguard Worker unloadNanoapp(appOneId);
429*84e33947SAndroid Build Coastguard Worker unloadNanoapp(appTwoId);
430*84e33947SAndroid Build Coastguard Worker }
431*84e33947SAndroid Build Coastguard Worker
432*84e33947SAndroid Build Coastguard Worker } // namespace
433*84e33947SAndroid Build Coastguard Worker } // namespace chre