1 /*
2 * Copyright (C) 2017 The Android Open Source Project
3 *
4 * Licensed under the Apache License, Version 2.0 (the "License");
5 * you may not use this file except in compliance with the License.
6 * You may obtain a copy of the License at
7 *
8 * http://www.apache.org/licenses/LICENSE-2.0
9 *
10 * Unless required by applicable law or agreed to in writing, software
11 * distributed under the License is distributed on an "AS IS" BASIS,
12 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
13 * See the License for the specific language governing permissions and
14 * limitations under the License.
15 */
16
17 #include <chre.h>
18 #include <cinttypes>
19
20 #include "chre/util/macros.h"
21 #include "chre/util/nanoapp/log.h"
22 #include "chre/util/time.h"
23 #include "chre/util/nanoapp/wifi.h"
24
25 using chre::kOneMillisecondInNanoseconds;
26 using chre::Nanoseconds;
27 using chre::Seconds;
28
29 #define LOG_TAG "[WifiWorld]"
30
31 //#define WIFI_WORLD_VERBOSE_WIFI_RESULT_LOGS
32
33 #ifdef CHRE_NANOAPP_INTERNAL
34 namespace chre {
35 namespace {
36 #endif // CHRE_NANOAPP_INTERNAL
37
38 namespace {
39
40 //! A dummy cookie to pass into the configure scan monitoring async request.
41 constexpr uint32_t kScanMonitoringCookie = 0x1337;
42
43 //! A dummy cookie to pass into request scan async.
44 constexpr uint32_t kOnDemandScanCookie = 0xcafe;
45
46 //! The interval for on-demand wifi scans.
47 constexpr Nanoseconds kWifiScanInterval = Nanoseconds(Seconds(10));
48
49 //! A handle for the cyclic timer to request periodic on-demand wifi-scans.
50 uint32_t gWifiScanTimerHandle;
51
52 //! A global instance of wifi capabilities to use when reqeuesting wifi
53 //! functionality. This is populated at startup.
54 uint32_t gWifiCapabilities;
55
56 //! The last time in nanoseconds a wifi scan request was sucessfully made.
57 uint64_t gLastRequestTimeNs = 0;
58
59 //! True if CHRE_WIFI_REQUEST_TYPE_REQUEST_SCAN mode is requested.
60 bool gPendingOnDemandScan = false;
61
62 //! Accumulating count of the scan request results so far.
63 uint32_t gScanResultAcc = 0;
64
65 //! The currently requested on-demand wifi scan parameters.
66 chreWifiScanParams gWifiScanParams = {};
67
68 //! The sequence of on-demand wifi scan types to request for.
69 constexpr chreWifiScanType gWifiScanTypes[] = {
70 CHRE_WIFI_SCAN_TYPE_ACTIVE,
71 CHRE_WIFI_SCAN_TYPE_ACTIVE_PLUS_PASSIVE_DFS,
72 CHRE_WIFI_SCAN_TYPE_PASSIVE
73 };
74
75 //! The index of the next wifi scan type to request for.
76 uint8_t gScanTypeIndex = 0;
77
78 /**
79 * Logs a CHRE wifi scan result.
80 *
81 * @param result the scan result to log.
82 */
logChreWifiResult(const chreWifiScanResult & result)83 void logChreWifiResult(const chreWifiScanResult& result) {
84 const char *ssidStr = "<non-printable>";
85 char ssidBuffer[chre::kMaxSsidStrLen];
86 if (result.ssidLen == 0) {
87 ssidStr = "<empty>";
88 } else if (chre::parseSsidToStr(ssidBuffer, sizeof(ssidBuffer),
89 result.ssid, result.ssidLen)) {
90 ssidStr = ssidBuffer;
91 }
92
93 LOGI("Found network with SSID: %s", ssidStr);
94 #ifdef WIFI_WORLD_VERBOSE_WIFI_RESULT_LOGS
95 const char *bssidStr = "<non-printable>";
96 char bssidBuffer[chre::kBssidStrLen];
97 if (chre::parseBssidToStr(result.bssid, bssidBuffer, sizeof(bssidBuffer))) {
98 bssidStr = bssidBuffer;
99 }
100
101 LOGI(" age (ms): %" PRIu32, result.ageMs);
102 LOGI(" capability info: %" PRIx16, result.capabilityInfo);
103 LOGI(" bssid: %s", bssidStr);
104 LOGI(" flags: %" PRIx8, result.flags);
105 LOGI(" rssi: %" PRId8 "dBm", result.rssi);
106 LOGI(" band: %s (%" PRIu8 ")",
107 chre::parseChreWifiBand(result.band), result.band);
108 LOGI(" primary channel: %" PRIu32, result.primaryChannel);
109 LOGI(" center frequency primary: %" PRIu32, result.centerFreqPrimary);
110 LOGI(" center frequency secondary: %" PRIu32, result.centerFreqSecondary);
111 LOGI(" channel width: %" PRIu8, result.channelWidth);
112 LOGI(" security mode: %" PRIx8, result.securityMode);
113 #endif // WIFI_WORLD_VERBOSE_WIFI_RESULT_LOGS
114 }
115
116 /**
117 * Requests a delayed wifi scan using a one-shot timer. The interval is
118 * specified as a constant at the top of this file.
119 */
requestDelayedWifiScan()120 void requestDelayedWifiScan() {
121 if (gWifiCapabilities & CHRE_WIFI_CAPABILITIES_ON_DEMAND_SCAN) {
122 // Schedule a timer to send an active wifi scan.
123 gWifiScanTimerHandle = chreTimerSet(kWifiScanInterval.toRawNanoseconds(),
124 &gWifiScanTimerHandle /* data */,
125 true /* oneShot */);
126 if (gWifiScanTimerHandle == CHRE_TIMER_INVALID) {
127 LOGE("Failed to set timer for delayed wifi scan");
128 } else {
129 LOGI("Set a timer to request a WiFi scan");
130 }
131 }
132 }
133
134 /**
135 * Handles the result of an asynchronous request for a wifi resource.
136 *
137 * @param result a pointer to the event structure containing the result of the
138 * request.
139 */
handleWifiAsyncResult(const chreAsyncResult * result)140 void handleWifiAsyncResult(const chreAsyncResult *result) {
141 if (result->requestType == CHRE_WIFI_REQUEST_TYPE_CONFIGURE_SCAN_MONITOR) {
142 if (result->success) {
143 LOGI("Successfully requested wifi scan monitoring");
144 } else {
145 LOGE("Error requesting wifi scan monitoring with %" PRIu8,
146 result->errorCode);
147 }
148
149 if (result->cookie != &kScanMonitoringCookie) {
150 LOGE("Scan monitoring request cookie mismatch");
151 }
152 } else if (result->requestType == CHRE_WIFI_REQUEST_TYPE_REQUEST_SCAN) {
153 uint64_t timeSinceRequest = chreGetTime() - gLastRequestTimeNs;
154 if (result->success) {
155 LOGI("Successfully requested an on-demand wifi scan (response time %"
156 PRIu64 " ms)", timeSinceRequest / kOneMillisecondInNanoseconds);
157 gPendingOnDemandScan = true;
158 } else {
159 LOGE("Error requesting an on-demand wifi scan with %" PRIu8,
160 result->errorCode);
161 }
162
163 if (result->cookie != &kOnDemandScanCookie) {
164 LOGE("On-demand scan cookie mismatch");
165 }
166
167 requestDelayedWifiScan();
168 } else {
169 LOGE("Received invalid async result");
170 }
171 }
172
173 /**
174 * Handles a wifi scan event.
175 *
176 * @param event a pointer to the details of the wifi scan event.
177 */
handleWifiScanEvent(const chreWifiScanEvent * event)178 void handleWifiScanEvent(const chreWifiScanEvent *event) {
179 LOGI("Received Wifi scan event of type %" PRIu8 " with %" PRIu8
180 " results at %" PRIu64 "ns", event->scanType, event->resultCount,
181 event->referenceTime);
182
183 if (gPendingOnDemandScan) {
184 uint64_t timeSinceRequest = chreGetTime() - gLastRequestTimeNs;
185 LOGI("Time since scan request = %" PRIu64 " ms",
186 timeSinceRequest / kOneMillisecondInNanoseconds);
187
188 if (event->scanType != gWifiScanParams.scanType) {
189 LOGE("Invalid scan event type (expected %" PRIu8 ", received %" PRIu8 ")",
190 gWifiScanParams.scanType, event->scanType);
191 }
192
193 gScanResultAcc += event->resultCount;
194 if (gScanResultAcc >= event->resultTotal) {
195 gPendingOnDemandScan = false;
196 gScanResultAcc = 0;
197 }
198 }
199
200 for (uint8_t i = 0; i < event->resultCount; i++) {
201 const chreWifiScanResult& result = event->results[i];
202 logChreWifiResult(result);
203 }
204 }
205
206 /**
207 * Handles a timer event.
208 *
209 * @param eventData The cookie passed to the timer request.
210 */
handleTimerEvent(const void * eventData)211 void handleTimerEvent(const void *eventData) {
212 const uint32_t *timerHandle = static_cast<const uint32_t *>(eventData);
213 if (*timerHandle == gWifiScanTimerHandle) {
214 gWifiScanParams.scanType = gWifiScanTypes[gScanTypeIndex];
215 gWifiScanParams.maxScanAgeMs = 5000; // 5 seconds
216 gWifiScanParams.frequencyListLen = 0;
217 gWifiScanParams.ssidListLen = 0;
218 gScanTypeIndex = (gScanTypeIndex + 1) % ARRAY_SIZE(gWifiScanTypes);
219
220 if (chreWifiRequestScanAsync(&gWifiScanParams, &kOnDemandScanCookie)) {
221 LOGI("Requested a wifi scan successfully");
222 gLastRequestTimeNs = chreGetTime();
223 } else {
224 LOGE("Failed to request a wifi scan");
225 }
226 } else {
227 LOGE("Received invalid timer handle");
228 }
229 }
230
231 } // namespace
232
nanoappStart()233 bool nanoappStart() {
234 LOGI("App started as instance %" PRIu32, chreGetInstanceId());
235
236 gWifiCapabilities = chreWifiGetCapabilities();
237 LOGI("Detected WiFi support as: 0x%" PRIx32, gWifiCapabilities);
238
239 if (gWifiCapabilities & CHRE_WIFI_CAPABILITIES_SCAN_MONITORING) {
240 if (chreWifiConfigureScanMonitorAsync(true, &kScanMonitoringCookie)) {
241 LOGI("Scan monitor enable request successful");
242 } else {
243 LOGE("Error sending scan monitoring request");
244 }
245 }
246
247 requestDelayedWifiScan();
248 return true;
249 }
250
nanoappHandleEvent(uint32_t senderInstanceId,uint16_t eventType,const void * eventData)251 void nanoappHandleEvent(uint32_t senderInstanceId,
252 uint16_t eventType,
253 const void *eventData) {
254 switch (eventType) {
255 case CHRE_EVENT_WIFI_ASYNC_RESULT:
256 handleWifiAsyncResult(static_cast<const chreAsyncResult *>(eventData));
257 break;
258 case CHRE_EVENT_WIFI_SCAN_RESULT:
259 handleWifiScanEvent(static_cast<const chreWifiScanEvent *>(eventData));
260 break;
261 case CHRE_EVENT_TIMER:
262 handleTimerEvent(eventData);
263 break;
264 default:
265 LOGW("Unhandled event type %" PRIu16, eventType);
266 }
267 }
268
nanoappEnd()269 void nanoappEnd() {
270 LOGI("Wifi world app stopped");
271 }
272
273 #ifdef CHRE_NANOAPP_INTERNAL
274 } // anonymous namespace
275 } // namespace chre
276
277 #include "chre/util/nanoapp/app_id.h"
278 #include "chre/platform/static_nanoapp_init.h"
279
280 CHRE_STATIC_NANOAPP_INIT(WifiWorld, chre::kWifiWorldAppId, 0);
281 #endif // CHRE_NANOAPP_INTERNAL
282