/* * Copyright (C) 2017 The Android Open Source Project * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. * You may obtain a copy of the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. * See the License for the specific language governing permissions and * limitations under the License. */ #include #include #include "chre/util/nanoapp/log.h" #include "chre/util/time.h" #include "chre/util/nanoapp/wifi.h" #define LOG_TAG "[WifiWorld]" //#define WIFI_WORLD_VERBOSE_WIFI_RESULT_LOGS #ifdef CHRE_NANOAPP_INTERNAL namespace chre { namespace { #endif // CHRE_NANOAPP_INTERNAL //! A dummy cookie to pass into the configure scan monitoring async request. const uint32_t kScanMonitoringCookie = 0x1337; //! A dummy cookie to pass into request scan async. const uint32_t kOnDemandScanCookie = 0xcafe; //! The interval for on-demand wifi scans. const Nanoseconds kWifiScanInterval = Nanoseconds(Seconds(10)); //! A handle for the cyclic timer to request periodic on-demand wifi-scans. uint32_t gWifiScanTimerHandle; namespace { /** * Logs a CHRE wifi scan result. * * @param result the scan result to log. */ void logChreWifiResult(const chreWifiScanResult& result) { const char *ssidStr = ""; char ssidBuffer[kMaxSsidStrLen]; if (result.ssidLen == 0) { ssidStr = ""; } else if (parseSsidToStr(ssidBuffer, kMaxSsidStrLen, result.ssid, result.ssidLen)) { ssidStr = ssidBuffer; } const char *bssidStr = ""; char bssidBuffer[kBssidStrLen]; if (parseBssidToStr(result.bssid, bssidBuffer, kBssidStrLen)) { bssidStr = bssidBuffer; } LOGI("Found network with SSID: %s", ssidStr); #ifdef WIFI_WORLD_VERBOSE_WIFI_RESULT_LOGS LOGI(" age (ms): %" PRIu32, result.ageMs); LOGI(" capability info: %" PRIx16, result.capabilityInfo); LOGI(" bssid: %s", bssidStr); LOGI(" flags: %" PRIx8, result.flags); LOGI(" rssi: %" PRId8 "dBm", result.rssi); LOGI(" band: %s (%" PRIu8 ")", parseChreWifiBand(result.band), result.band); LOGI(" primary channel: %" PRIu32, result.primaryChannel); LOGI(" center frequency primary: %" PRIu32, result.centerFreqPrimary); LOGI(" center frequency secondary: %" PRIu32, result.centerFreqSecondary); LOGI(" channel width: %" PRIu8, result.channelWidth); LOGI(" security mode: %" PRIx8, result.securityMode); #endif // WIFI_WORLD_VERBOSE_WIFI_RESULT_LOGS } /** * Handles the result of an asynchronous request for a wifi resource. * * @param result a pointer to the event structure containing the result of the * request. */ void handleWifiAsyncResult(const chreAsyncResult *result) { if (result->requestType == CHRE_WIFI_REQUEST_TYPE_CONFIGURE_SCAN_MONITOR) { if (result->success) { LOGI("Successfully requested wifi scan monitoring"); } else { LOGI("Error requesting wifi scan monitoring with %" PRIu8, result->errorCode); } if (result->cookie != &kScanMonitoringCookie) { LOGE("Scan monitoring request cookie mismatch"); } } else if (result->requestType == CHRE_WIFI_REQUEST_TYPE_REQUEST_SCAN) { if (result->success) { LOGI("Successfully requested an on-demand wifi scan"); } else { LOGE("Error requesting an on-demand wifi scan with %" PRIu8, result->errorCode); } if (result->cookie != &kOnDemandScanCookie) { LOGE("On-demand scan cookie mismatch"); } } } /** * Handles a wifi scan event. * * @param event a pointer to the details of the wifi scan event. */ void handleWifiScanEvent(const chreWifiScanEvent *event) { for (uint8_t i = 0; i < event->resultCount; i++) { const chreWifiScanResult& result = event->results[i]; logChreWifiResult(result); } } /** * Handles a timer event. * * @param eventData The cookie passed to the timer request. */ void handleTimerEvent(const void *eventData) { const uint32_t *timerHandle = static_cast(eventData); if (*timerHandle == gWifiScanTimerHandle) { if (chreWifiRequestScanAsyncDefault(&kOnDemandScanCookie)) { LOGI("Requested a wifi scan successfully"); } else { LOGE("Failed to request a wifi scan"); } } else { LOGE("Received invalid timer handle"); } } } // namespace bool nanoappStart() { LOGI("App started as instance %" PRIu32, chreGetInstanceId()); const char *wifiCapabilitiesStr; uint32_t wifiCapabilities = chreWifiGetCapabilities(); switch (wifiCapabilities) { case CHRE_WIFI_CAPABILITIES_ON_DEMAND_SCAN | CHRE_WIFI_CAPABILITIES_SCAN_MONITORING: wifiCapabilitiesStr = "ON_DEMAND_SCAN | SCAN_MONITORING"; break; case CHRE_WIFI_CAPABILITIES_ON_DEMAND_SCAN: wifiCapabilitiesStr = "ON_DEMAND_SCAN"; break; case CHRE_WIFI_CAPABILITIES_SCAN_MONITORING: wifiCapabilitiesStr = "SCAN_MONITORING"; break; case CHRE_WIFI_CAPABILITIES_NONE: wifiCapabilitiesStr = "NONE"; break; default: wifiCapabilitiesStr = "INVALID"; } LOGI("Detected WiFi support as: %s (%" PRIu32 ")", wifiCapabilitiesStr, wifiCapabilities); if (wifiCapabilities & CHRE_WIFI_CAPABILITIES_SCAN_MONITORING) { if (chreWifiConfigureScanMonitorAsync(true, &kScanMonitoringCookie)) { LOGI("Scan monitor enable request successful"); } else { LOGE("Error sending scan monitoring request"); } } if (wifiCapabilities & CHRE_WIFI_CAPABILITIES_ON_DEMAND_SCAN) { // Schedule a timer to send an active wifi scan. gWifiScanTimerHandle = chreTimerSet(kWifiScanInterval.toRawNanoseconds(), &gWifiScanTimerHandle /* data */, false /* oneShot */); if (gWifiScanTimerHandle == CHRE_TIMER_INVALID) { LOGE("Failed to set periodic scan timer"); } else { LOGI("Set a timer to request periodic WiFi scans"); } } return true; } void nanoappHandleEvent(uint32_t senderInstanceId, uint16_t eventType, const void *eventData) { switch (eventType) { case CHRE_EVENT_WIFI_ASYNC_RESULT: handleWifiAsyncResult(static_cast(eventData)); break; case CHRE_EVENT_WIFI_SCAN_RESULT: handleWifiScanEvent(static_cast(eventData)); break; case CHRE_EVENT_TIMER: handleTimerEvent(eventData); break; default: LOGW("Unhandled event type %" PRIu16, eventType); } } void nanoappEnd() { LOGI("Wifi world app stopped"); } #ifdef CHRE_NANOAPP_INTERNAL } // anonymous namespace } // namespace chre #include "chre/util/nanoapp/app_id.h" #include "chre/platform/static_nanoapp_init.h" CHRE_STATIC_NANOAPP_INIT(WifiWorld, chre::kWifiWorldAppId, 0); #endif // CHRE_NANOAPP_INTERNAL