2009-05-07 02:16:52 +08:00
|
|
|
/*
|
|
|
|
* Copyright (C) 2008 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.
|
|
|
|
*/
|
2009-05-21 06:28:43 +08:00
|
|
|
|
|
|
|
#include <stdlib.h>
|
2009-05-07 02:16:52 +08:00
|
|
|
#include <string.h>
|
|
|
|
#include <errno.h>
|
|
|
|
|
|
|
|
#define LOG_TAG "WifiController"
|
|
|
|
#include <cutils/log.h>
|
|
|
|
|
|
|
|
#include "Supplicant.h"
|
|
|
|
#include "WifiController.h"
|
2009-05-08 02:37:10 +08:00
|
|
|
#include "NetworkManager.h"
|
2009-06-24 12:10:57 +08:00
|
|
|
#include "ResponseCode.h"
|
2009-05-23 06:36:13 +08:00
|
|
|
#include "WifiNetwork.h"
|
2009-06-16 05:10:44 +08:00
|
|
|
#include "ISupplicantEventHandler.h"
|
|
|
|
#include "SupplicantState.h"
|
|
|
|
#include "SupplicantStatus.h"
|
|
|
|
#include "SupplicantAssociatingEvent.h"
|
|
|
|
#include "SupplicantAssociatedEvent.h"
|
|
|
|
#include "SupplicantConnectedEvent.h"
|
|
|
|
#include "SupplicantScanResultsEvent.h"
|
|
|
|
#include "SupplicantStateChangeEvent.h"
|
|
|
|
#include "SupplicantConnectionTimeoutEvent.h"
|
|
|
|
#include "SupplicantDisconnectedEvent.h"
|
2009-06-24 12:10:57 +08:00
|
|
|
#include "WifiStatusPoller.h"
|
2009-06-16 05:10:44 +08:00
|
|
|
|
|
|
|
WifiController::WifiController(PropertyManager *mPropMngr,
|
|
|
|
IControllerHandler *handlers,
|
|
|
|
char *modpath, char *modname, char *modargs) :
|
2009-06-24 12:10:57 +08:00
|
|
|
Controller("wifi", mPropMngr, handlers) {
|
2009-05-07 02:16:52 +08:00
|
|
|
strncpy(mModulePath, modpath, sizeof(mModulePath));
|
|
|
|
strncpy(mModuleName, modname, sizeof(mModuleName));
|
|
|
|
strncpy(mModuleArgs, modargs, sizeof(mModuleArgs));
|
|
|
|
|
2009-06-16 05:10:44 +08:00
|
|
|
mLatestScanResults = new ScanResultCollection();
|
|
|
|
pthread_mutex_init(&mLatestScanResultsLock, NULL);
|
|
|
|
|
2009-06-24 12:10:57 +08:00
|
|
|
pthread_mutex_init(&mLock, NULL);
|
2009-05-21 06:28:43 +08:00
|
|
|
|
2009-06-24 12:10:57 +08:00
|
|
|
mSupplicant = new Supplicant(this, this);
|
|
|
|
mActiveScan = false;
|
2009-05-23 06:36:13 +08:00
|
|
|
mEnabled = false;
|
2009-06-24 12:10:57 +08:00
|
|
|
mScanOnly = false;
|
|
|
|
mPacketFilter = false;
|
|
|
|
mBluetoothCoexScan = false;
|
|
|
|
mBluetoothCoexMode = 0;
|
|
|
|
mCurrentlyConnectedNetworkId = -1;
|
|
|
|
mStatusPoller = new WifiStatusPoller(this);
|
|
|
|
mRssiEventThreshold = 5;
|
|
|
|
mLastLinkSpeed = 0;
|
2009-05-23 06:36:13 +08:00
|
|
|
|
2009-06-16 05:10:44 +08:00
|
|
|
mSupplicantState = SupplicantState::UNKNOWN;
|
2009-06-24 12:10:57 +08:00
|
|
|
|
|
|
|
mStaticProperties.propEnabled = new WifiEnabledProperty(this);
|
|
|
|
mStaticProperties.propScanOnly = new WifiScanOnlyProperty(this);
|
|
|
|
mStaticProperties.propAllowedChannels = new WifiAllowedChannelsProperty(this);
|
|
|
|
|
|
|
|
mStaticProperties.propRssiEventThreshold =
|
|
|
|
new IntegerPropertyHelper("RssiEventThreshold", false, &mRssiEventThreshold);
|
|
|
|
|
|
|
|
mDynamicProperties.propSupplicantState = new WifiSupplicantStateProperty(this);
|
|
|
|
mDynamicProperties.propActiveScan = new WifiActiveScanProperty(this);
|
|
|
|
mDynamicProperties.propInterface = new WifiInterfaceProperty(this);
|
|
|
|
mDynamicProperties.propSearching = new WifiSearchingProperty(this);
|
|
|
|
mDynamicProperties.propPacketFilter = new WifiPacketFilterProperty(this);
|
|
|
|
mDynamicProperties.propBluetoothCoexScan = new WifiBluetoothCoexScanProperty(this);
|
|
|
|
mDynamicProperties.propBluetoothCoexMode = new WifiBluetoothCoexModeProperty(this);
|
|
|
|
mDynamicProperties.propCurrentNetwork = new WifiCurrentNetworkProperty(this);
|
|
|
|
|
|
|
|
mDynamicProperties.propRssi = new IntegerPropertyHelper("Rssi", true, &mLastRssi);
|
|
|
|
mDynamicProperties.propLinkSpeed = new IntegerPropertyHelper("LinkSpeed", true, &mLastLinkSpeed);
|
|
|
|
|
|
|
|
mDynamicProperties.propSuspended = new WifiSuspendedProperty(this);
|
|
|
|
mDynamicProperties.propNetCount = new WifiNetCountProperty(this);
|
|
|
|
mDynamicProperties.propTriggerScan = new WifiTriggerScanProperty(this);
|
2009-05-07 02:16:52 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
int WifiController::start() {
|
2009-06-24 12:10:57 +08:00
|
|
|
mPropMngr->attachProperty("wifi", mStaticProperties.propEnabled);
|
|
|
|
mPropMngr->attachProperty("wifi", mStaticProperties.propScanOnly);
|
|
|
|
mPropMngr->attachProperty("wifi", mStaticProperties.propAllowedChannels);
|
|
|
|
mPropMngr->attachProperty("wifi", mStaticProperties.propRssiEventThreshold);
|
2009-05-07 02:16:52 +08:00
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
int WifiController::stop() {
|
2009-06-24 12:10:57 +08:00
|
|
|
mPropMngr->detachProperty("wifi", mStaticProperties.propEnabled);
|
|
|
|
mPropMngr->detachProperty("wifi", mStaticProperties.propScanOnly);
|
|
|
|
mPropMngr->detachProperty("wifi", mStaticProperties.propAllowedChannels);
|
|
|
|
mPropMngr->detachProperty("wifi", mStaticProperties.propRssiEventThreshold);
|
2009-06-16 05:10:44 +08:00
|
|
|
return 0;
|
2009-05-07 02:16:52 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
int WifiController::enable() {
|
2009-06-17 01:50:47 +08:00
|
|
|
|
2009-05-08 02:37:10 +08:00
|
|
|
if (!isPoweredUp()) {
|
2009-06-17 01:50:47 +08:00
|
|
|
LOGI("Powering up");
|
2009-06-16 05:10:44 +08:00
|
|
|
sendStatusBroadcast("Powering up WiFi hardware");
|
2009-05-08 02:37:10 +08:00
|
|
|
if (powerUp()) {
|
|
|
|
LOGE("Powerup failed (%s)", strerror(errno));
|
|
|
|
return -1;
|
|
|
|
}
|
2009-05-07 02:16:52 +08:00
|
|
|
}
|
2009-05-08 02:37:10 +08:00
|
|
|
|
2009-05-07 02:16:52 +08:00
|
|
|
if (mModuleName[0] != '\0' && !isKernelModuleLoaded(mModuleName)) {
|
2009-06-17 01:50:47 +08:00
|
|
|
LOGI("Loading driver");
|
2009-06-16 05:10:44 +08:00
|
|
|
sendStatusBroadcast("Loading WiFi driver");
|
2009-05-07 02:16:52 +08:00
|
|
|
if (loadKernelModule(mModulePath, mModuleArgs)) {
|
|
|
|
LOGE("Kernel module load failed (%s)", strerror(errno));
|
|
|
|
goto out_powerdown;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2009-05-08 02:37:10 +08:00
|
|
|
if (!isFirmwareLoaded()) {
|
2009-06-17 01:50:47 +08:00
|
|
|
LOGI("Loading firmware");
|
2009-06-16 05:10:44 +08:00
|
|
|
sendStatusBroadcast("Loading WiFI firmware");
|
2009-05-08 02:37:10 +08:00
|
|
|
if (loadFirmware()) {
|
|
|
|
LOGE("Firmware load failed (%s)", strerror(errno));
|
|
|
|
goto out_powerdown;
|
|
|
|
}
|
2009-05-07 02:16:52 +08:00
|
|
|
}
|
|
|
|
|
2009-05-08 02:37:10 +08:00
|
|
|
if (!mSupplicant->isStarted()) {
|
2009-06-17 01:50:47 +08:00
|
|
|
LOGI("Starting WPA Supplicant");
|
2009-06-16 05:10:44 +08:00
|
|
|
sendStatusBroadcast("Starting WPA Supplicant");
|
2009-05-08 02:37:10 +08:00
|
|
|
if (mSupplicant->start()) {
|
|
|
|
LOGE("Supplicant start failed (%s)", strerror(errno));
|
|
|
|
goto out_unloadmodule;
|
|
|
|
}
|
2009-05-07 02:16:52 +08:00
|
|
|
}
|
|
|
|
|
2009-05-23 06:36:13 +08:00
|
|
|
if (Controller::bindInterface(mSupplicant->getInterfaceName())) {
|
|
|
|
LOGE("Error binding interface (%s)", strerror(errno));
|
|
|
|
goto out_unloadmodule;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (mSupplicant->refreshNetworkList())
|
|
|
|
LOGW("Error getting list of networks (%s)", strerror(errno));
|
|
|
|
|
2009-06-24 12:10:57 +08:00
|
|
|
LOGW("TODO: Set # of allowed regulatory channels!");
|
|
|
|
|
|
|
|
mPropMngr->attachProperty("wifi", mDynamicProperties.propSupplicantState);
|
|
|
|
mPropMngr->attachProperty("wifi", mDynamicProperties.propActiveScan);
|
|
|
|
mPropMngr->attachProperty("wifi", mDynamicProperties.propInterface);
|
|
|
|
mPropMngr->attachProperty("wifi", mDynamicProperties.propSearching);
|
|
|
|
mPropMngr->attachProperty("wifi", mDynamicProperties.propPacketFilter);
|
|
|
|
mPropMngr->attachProperty("wifi", mDynamicProperties.propBluetoothCoexScan);
|
|
|
|
mPropMngr->attachProperty("wifi", mDynamicProperties.propBluetoothCoexMode);
|
|
|
|
mPropMngr->attachProperty("wifi", mDynamicProperties.propCurrentNetwork);
|
|
|
|
mPropMngr->attachProperty("wifi", mDynamicProperties.propRssi);
|
|
|
|
mPropMngr->attachProperty("wifi", mDynamicProperties.propLinkSpeed);
|
|
|
|
mPropMngr->attachProperty("wifi", mDynamicProperties.propSuspended);
|
|
|
|
mPropMngr->attachProperty("wifi", mDynamicProperties.propNetCount);
|
|
|
|
mPropMngr->attachProperty("wifi", mDynamicProperties.propTriggerScan);
|
2009-05-23 06:36:13 +08:00
|
|
|
|
2009-06-17 01:50:47 +08:00
|
|
|
LOGI("Enabled successfully");
|
2009-05-07 02:16:52 +08:00
|
|
|
return 0;
|
|
|
|
|
|
|
|
out_unloadmodule:
|
|
|
|
if (mModuleName[0] != '\0' && !isKernelModuleLoaded(mModuleName)) {
|
|
|
|
if (unloadKernelModule(mModuleName)) {
|
|
|
|
LOGE("Unable to unload module after failure!");
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
out_powerdown:
|
|
|
|
if (powerDown()) {
|
|
|
|
LOGE("Unable to powerdown after failure!");
|
|
|
|
}
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
2009-06-24 12:10:57 +08:00
|
|
|
bool WifiController::getSuspended() {
|
|
|
|
pthread_mutex_lock(&mLock);
|
|
|
|
bool r = mSuspended;
|
|
|
|
pthread_mutex_unlock(&mLock);
|
|
|
|
return r;
|
|
|
|
}
|
|
|
|
|
|
|
|
int WifiController::setSuspend(bool suspend) {
|
|
|
|
|
|
|
|
pthread_mutex_lock(&mLock);
|
|
|
|
if (suspend == mSuspended) {
|
|
|
|
LOGW("Suspended state already = %d", suspend);
|
|
|
|
pthread_mutex_unlock(&mLock);
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (suspend) {
|
|
|
|
mHandlers->onControllerSuspending(this);
|
|
|
|
|
|
|
|
char tmp[80];
|
|
|
|
LOGD("Suspending from supplicant state %s",
|
|
|
|
SupplicantState::toString(mSupplicantState,
|
|
|
|
tmp,
|
|
|
|
sizeof(tmp)));
|
|
|
|
|
|
|
|
if (mSupplicantState != SupplicantState::IDLE) {
|
|
|
|
LOGD("Forcing Supplicant disconnect");
|
|
|
|
if (mSupplicant->disconnect()) {
|
|
|
|
LOGW("Error disconnecting (%s)", strerror(errno));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
LOGD("Stopping Supplicant driver");
|
|
|
|
if (mSupplicant->stopDriver()) {
|
|
|
|
LOGE("Error stopping driver (%s)", strerror(errno));
|
|
|
|
pthread_mutex_unlock(&mLock);
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
LOGD("Resuming");
|
|
|
|
|
|
|
|
if (mSupplicant->startDriver()) {
|
|
|
|
LOGE("Error resuming driver (%s)", strerror(errno));
|
|
|
|
pthread_mutex_unlock(&mLock);
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
// XXX: set regulatory max channels
|
|
|
|
if (mScanOnly)
|
|
|
|
mSupplicant->triggerScan();
|
|
|
|
else
|
|
|
|
mSupplicant->reconnect();
|
|
|
|
|
|
|
|
mHandlers->onControllerResumed(this);
|
|
|
|
}
|
|
|
|
|
|
|
|
mSuspended = suspend;
|
|
|
|
pthread_mutex_unlock(&mLock);
|
|
|
|
LOGD("Suspend / Resume completed");
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2009-05-21 06:28:43 +08:00
|
|
|
void WifiController::sendStatusBroadcast(const char *msg) {
|
2009-05-13 05:36:32 +08:00
|
|
|
NetworkManager::Instance()->
|
|
|
|
getBroadcaster()->
|
2009-06-24 12:10:57 +08:00
|
|
|
sendBroadcast(ResponseCode::UnsolicitedInformational, msg, false);
|
2009-05-08 02:37:10 +08:00
|
|
|
}
|
|
|
|
|
2009-05-07 02:16:52 +08:00
|
|
|
int WifiController::disable() {
|
|
|
|
|
2009-06-24 12:10:57 +08:00
|
|
|
mPropMngr->detachProperty("wifi", mDynamicProperties.propSupplicantState);
|
|
|
|
mPropMngr->detachProperty("wifi", mDynamicProperties.propActiveScan);
|
|
|
|
mPropMngr->detachProperty("wifi", mDynamicProperties.propInterface);
|
|
|
|
mPropMngr->detachProperty("wifi", mDynamicProperties.propSearching);
|
|
|
|
mPropMngr->detachProperty("wifi", mDynamicProperties.propPacketFilter);
|
|
|
|
mPropMngr->detachProperty("wifi", mDynamicProperties.propBluetoothCoexScan);
|
|
|
|
mPropMngr->detachProperty("wifi", mDynamicProperties.propBluetoothCoexMode);
|
|
|
|
mPropMngr->detachProperty("wifi", mDynamicProperties.propCurrentNetwork);
|
|
|
|
mPropMngr->detachProperty("wifi", mDynamicProperties.propRssi);
|
|
|
|
mPropMngr->detachProperty("wifi", mDynamicProperties.propLinkSpeed);
|
|
|
|
mPropMngr->detachProperty("wifi", mDynamicProperties.propSuspended);
|
|
|
|
mPropMngr->detachProperty("wifi", mDynamicProperties.propNetCount);
|
2009-06-16 05:10:44 +08:00
|
|
|
|
2009-05-08 02:37:10 +08:00
|
|
|
if (mSupplicant->isStarted()) {
|
2009-06-16 05:10:44 +08:00
|
|
|
sendStatusBroadcast("Stopping WPA Supplicant");
|
2009-05-08 02:37:10 +08:00
|
|
|
if (mSupplicant->stop()) {
|
|
|
|
LOGE("Supplicant stop failed (%s)", strerror(errno));
|
|
|
|
return -1;
|
|
|
|
}
|
2009-05-23 06:36:13 +08:00
|
|
|
} else
|
2009-05-08 02:37:10 +08:00
|
|
|
LOGW("disable(): Supplicant not running?");
|
2009-05-07 02:16:52 +08:00
|
|
|
|
|
|
|
if (mModuleName[0] != '\0' && isKernelModuleLoaded(mModuleName)) {
|
2009-06-16 05:10:44 +08:00
|
|
|
sendStatusBroadcast("Unloading WiFi driver");
|
2009-05-07 02:16:52 +08:00
|
|
|
if (unloadKernelModule(mModuleName)) {
|
|
|
|
LOGE("Unable to unload module (%s)", strerror(errno));
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2009-05-08 02:37:10 +08:00
|
|
|
if (isPoweredUp()) {
|
2009-06-16 05:10:44 +08:00
|
|
|
sendStatusBroadcast("Powering down WiFi hardware");
|
2009-05-08 02:37:10 +08:00
|
|
|
if (powerDown()) {
|
|
|
|
LOGE("Powerdown failed (%s)", strerror(errno));
|
|
|
|
return -1;
|
|
|
|
}
|
2009-05-07 02:16:52 +08:00
|
|
|
}
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
int WifiController::loadFirmware() {
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2009-06-24 12:10:57 +08:00
|
|
|
int WifiController::triggerScan() {
|
|
|
|
pthread_mutex_lock(&mLock);
|
|
|
|
if (verifyNotSuspended()) {
|
|
|
|
pthread_mutex_unlock(&mLock);
|
|
|
|
return -1;
|
|
|
|
}
|
2009-05-07 02:16:52 +08:00
|
|
|
|
2009-06-24 12:10:57 +08:00
|
|
|
switch (mSupplicantState) {
|
|
|
|
case SupplicantState::DISCONNECTED:
|
|
|
|
case SupplicantState::INACTIVE:
|
|
|
|
case SupplicantState::SCANNING:
|
|
|
|
case SupplicantState::IDLE:
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
// Switch to scan only mode
|
|
|
|
mSupplicant->setApScanMode(2);
|
|
|
|
break;
|
|
|
|
}
|
2009-05-07 02:16:52 +08:00
|
|
|
|
2009-06-24 12:10:57 +08:00
|
|
|
int rc = mSupplicant->triggerScan();
|
|
|
|
pthread_mutex_unlock(&mLock);
|
|
|
|
return rc;
|
|
|
|
}
|
2009-05-13 06:50:49 +08:00
|
|
|
|
2009-06-24 12:10:57 +08:00
|
|
|
int WifiController::setActiveScan(bool active) {
|
|
|
|
pthread_mutex_lock(&mLock);
|
|
|
|
if (mActiveScan == active) {
|
|
|
|
pthread_mutex_unlock(&mLock);
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
mActiveScan = active;
|
|
|
|
|
|
|
|
int rc = mSupplicant->setScanMode(active);
|
|
|
|
pthread_mutex_unlock(&mLock);
|
2009-05-07 02:16:52 +08:00
|
|
|
return rc;
|
|
|
|
}
|
|
|
|
|
2009-05-23 06:36:13 +08:00
|
|
|
WifiNetwork *WifiController::createNetwork() {
|
2009-06-24 12:10:57 +08:00
|
|
|
pthread_mutex_lock(&mLock);
|
2009-05-23 06:36:13 +08:00
|
|
|
WifiNetwork *wn = mSupplicant->createNetwork();
|
2009-06-24 12:10:57 +08:00
|
|
|
pthread_mutex_unlock(&mLock);
|
2009-05-23 06:36:13 +08:00
|
|
|
return wn;
|
2009-05-13 08:26:28 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
int WifiController::removeNetwork(int networkId) {
|
2009-06-24 12:10:57 +08:00
|
|
|
pthread_mutex_lock(&mLock);
|
2009-05-23 06:36:13 +08:00
|
|
|
WifiNetwork *wn = mSupplicant->lookupNetwork(networkId);
|
|
|
|
|
2009-06-24 12:10:57 +08:00
|
|
|
if (!wn) {
|
|
|
|
pthread_mutex_unlock(&mLock);
|
2009-05-23 06:36:13 +08:00
|
|
|
return -1;
|
2009-06-24 12:10:57 +08:00
|
|
|
}
|
|
|
|
int rc = mSupplicant->removeNetwork(wn);
|
|
|
|
pthread_mutex_unlock(&mLock);
|
|
|
|
return rc;
|
2009-05-13 08:26:28 +08:00
|
|
|
}
|
|
|
|
|
2009-05-08 02:37:10 +08:00
|
|
|
ScanResultCollection *WifiController::createScanResults() {
|
2009-06-16 05:10:44 +08:00
|
|
|
ScanResultCollection *d = new ScanResultCollection();
|
|
|
|
ScanResultCollection::iterator i;
|
|
|
|
|
|
|
|
pthread_mutex_lock(&mLatestScanResultsLock);
|
|
|
|
for (i = mLatestScanResults->begin(); i != mLatestScanResults->end(); ++i)
|
|
|
|
d->push_back((*i)->clone());
|
|
|
|
|
|
|
|
pthread_mutex_unlock(&mLatestScanResultsLock);
|
|
|
|
return d;
|
2009-05-07 02:16:52 +08:00
|
|
|
}
|
2009-05-13 08:26:28 +08:00
|
|
|
|
|
|
|
WifiNetworkCollection *WifiController::createNetworkList() {
|
|
|
|
return mSupplicant->createNetworkList();
|
|
|
|
}
|
2009-05-21 06:28:43 +08:00
|
|
|
|
2009-06-24 12:10:57 +08:00
|
|
|
int WifiController::setPacketFilter(bool enable) {
|
2009-05-23 06:36:13 +08:00
|
|
|
int rc;
|
|
|
|
|
2009-06-24 12:10:57 +08:00
|
|
|
pthread_mutex_lock(&mLock);
|
|
|
|
if (enable)
|
|
|
|
rc = mSupplicant->enablePacketFilter();
|
|
|
|
else
|
|
|
|
rc = mSupplicant->disablePacketFilter();
|
2009-05-23 06:36:13 +08:00
|
|
|
|
2009-06-24 12:10:57 +08:00
|
|
|
if (!rc)
|
|
|
|
mPacketFilter = enable;
|
|
|
|
pthread_mutex_unlock(&mLock);
|
2009-05-23 06:36:13 +08:00
|
|
|
return rc;
|
2009-05-21 06:28:43 +08:00
|
|
|
}
|
|
|
|
|
2009-06-24 12:10:57 +08:00
|
|
|
int WifiController::setBluetoothCoexistenceScan(bool enable) {
|
|
|
|
int rc;
|
|
|
|
|
|
|
|
pthread_mutex_lock(&mLock);
|
2009-05-23 06:36:13 +08:00
|
|
|
|
2009-06-24 12:10:57 +08:00
|
|
|
if (enable)
|
|
|
|
rc = mSupplicant->enableBluetoothCoexistenceScan();
|
2009-05-23 06:36:13 +08:00
|
|
|
else
|
2009-06-24 12:10:57 +08:00
|
|
|
rc = mSupplicant->disableBluetoothCoexistenceScan();
|
|
|
|
|
|
|
|
if (!rc)
|
|
|
|
mBluetoothCoexScan = enable;
|
|
|
|
pthread_mutex_unlock(&mLock);
|
|
|
|
return rc;
|
|
|
|
}
|
2009-05-21 06:28:43 +08:00
|
|
|
|
2009-06-24 12:10:57 +08:00
|
|
|
int WifiController::setScanOnly(bool scanOnly) {
|
|
|
|
pthread_mutex_lock(&mLock);
|
|
|
|
int rc = mSupplicant->setApScanMode((scanOnly ? 2 : 1));
|
|
|
|
if (!rc)
|
|
|
|
mScanOnly = scanOnly;
|
|
|
|
if (!mSuspended) {
|
|
|
|
if (scanOnly)
|
|
|
|
mSupplicant->disconnect();
|
|
|
|
else
|
|
|
|
mSupplicant->reconnect();
|
|
|
|
}
|
|
|
|
pthread_mutex_unlock(&mLock);
|
|
|
|
return rc;
|
|
|
|
}
|
|
|
|
|
|
|
|
int WifiController::setBluetoothCoexistenceMode(int mode) {
|
|
|
|
pthread_mutex_lock(&mLock);
|
|
|
|
int rc = mSupplicant->setBluetoothCoexistenceMode(mode);
|
|
|
|
if (!rc)
|
|
|
|
mBluetoothCoexMode = mode;
|
|
|
|
pthread_mutex_unlock(&mLock);
|
|
|
|
return rc;
|
2009-05-21 06:28:43 +08:00
|
|
|
}
|
|
|
|
|
2009-06-16 05:10:44 +08:00
|
|
|
void WifiController::onAssociatingEvent(SupplicantAssociatingEvent *evt) {
|
|
|
|
LOGD("onAssociatingEvent(%s, %s, %d)",
|
|
|
|
(evt->getBssid() ? evt->getBssid() : "n/a"),
|
|
|
|
(evt->getSsid() ? evt->getSsid() : "n/a"),
|
|
|
|
evt->getFreq());
|
|
|
|
}
|
|
|
|
|
|
|
|
void WifiController::onAssociatedEvent(SupplicantAssociatedEvent *evt) {
|
|
|
|
LOGD("onAssociatedEvent(%s)", evt->getBssid());
|
|
|
|
}
|
|
|
|
|
|
|
|
void WifiController::onConnectedEvent(SupplicantConnectedEvent *evt) {
|
|
|
|
LOGD("onConnectedEvent(%s, %d)", evt->getBssid(), evt->getReassociated());
|
2009-06-23 01:39:36 +08:00
|
|
|
SupplicantStatus *ss = mSupplicant->getStatus();
|
|
|
|
WifiNetwork *wn;
|
2009-06-16 05:10:44 +08:00
|
|
|
|
2009-06-23 01:39:36 +08:00
|
|
|
if (ss->getWpaState() != SupplicantState::COMPLETED) {
|
|
|
|
char tmp[32];
|
2009-06-16 05:10:44 +08:00
|
|
|
|
2009-06-23 01:39:36 +08:00
|
|
|
LOGW("onConnected() with SupplicantState = %s!",
|
|
|
|
SupplicantState::toString(ss->getWpaState(), tmp,
|
|
|
|
sizeof(tmp)));
|
|
|
|
return;
|
|
|
|
}
|
2009-06-16 05:10:44 +08:00
|
|
|
|
2009-06-23 01:39:36 +08:00
|
|
|
if (ss->getId() == -1) {
|
|
|
|
LOGW("onConnected() with id = -1!");
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2009-06-24 12:10:57 +08:00
|
|
|
mCurrentlyConnectedNetworkId = ss->getId();
|
2009-06-23 01:39:36 +08:00
|
|
|
if (!(wn = mSupplicant->lookupNetwork(ss->getId()))) {
|
|
|
|
LOGW("Error looking up connected network id %d (%s)",
|
|
|
|
ss->getId(), strerror(errno));
|
|
|
|
return;
|
2009-06-16 05:10:44 +08:00
|
|
|
}
|
2009-06-23 01:39:36 +08:00
|
|
|
|
|
|
|
delete ss;
|
2009-06-24 12:10:57 +08:00
|
|
|
mHandlers->onInterfaceConnected(this);
|
2009-06-16 05:10:44 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
void WifiController::onScanResultsEvent(SupplicantScanResultsEvent *evt) {
|
|
|
|
char *reply;
|
|
|
|
|
|
|
|
if (!(reply = (char *) malloc(4096))) {
|
|
|
|
LOGE("Out of memory");
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2009-06-24 12:10:57 +08:00
|
|
|
mNumScanResultsSinceLastStateChange++;
|
|
|
|
if (mNumScanResultsSinceLastStateChange >= 3)
|
|
|
|
mIsSupplicantSearching = false;
|
|
|
|
|
2009-06-16 05:10:44 +08:00
|
|
|
size_t len = 4096;
|
|
|
|
|
|
|
|
if (mSupplicant->sendCommand("SCAN_RESULTS", reply, &len)) {
|
|
|
|
LOGW("onScanResultsEvent: Error getting scan results (%s)",
|
|
|
|
strerror(errno));
|
|
|
|
free(reply);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
pthread_mutex_lock(&mLatestScanResultsLock);
|
|
|
|
if (!mLatestScanResults->empty()) {
|
|
|
|
ScanResultCollection::iterator i;
|
|
|
|
|
|
|
|
for (i = mLatestScanResults->begin();
|
|
|
|
i !=mLatestScanResults->end(); ++i) {
|
|
|
|
delete *i;
|
|
|
|
}
|
|
|
|
mLatestScanResults->clear();
|
|
|
|
}
|
|
|
|
|
|
|
|
char *linep;
|
|
|
|
char *linep_next = NULL;
|
|
|
|
|
|
|
|
if (!strtok_r(reply, "\n", &linep_next)) {
|
|
|
|
free(reply);
|
|
|
|
pthread_mutex_unlock(&mLatestScanResultsLock);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
while((linep = strtok_r(NULL, "\n", &linep_next)))
|
|
|
|
mLatestScanResults->push_back(new ScanResult(linep));
|
|
|
|
|
2009-06-24 12:10:57 +08:00
|
|
|
// Switch handling of scan results back to normal mode
|
|
|
|
mSupplicant->setApScanMode(1);
|
|
|
|
|
2009-06-16 05:10:44 +08:00
|
|
|
char *tmp;
|
|
|
|
asprintf(&tmp, "Scan results ready (%d)", mLatestScanResults->size());
|
|
|
|
NetworkManager::Instance()->getBroadcaster()->
|
2009-06-24 12:10:57 +08:00
|
|
|
sendBroadcast(ResponseCode::ScanResultsReady,
|
|
|
|
tmp, false);
|
2009-06-16 05:10:44 +08:00
|
|
|
free(tmp);
|
|
|
|
pthread_mutex_unlock(&mLatestScanResultsLock);
|
|
|
|
free(reply);
|
|
|
|
}
|
|
|
|
|
|
|
|
void WifiController::onStateChangeEvent(SupplicantStateChangeEvent *evt) {
|
|
|
|
char tmp[32];
|
|
|
|
char tmp2[32];
|
|
|
|
|
2009-06-24 12:10:57 +08:00
|
|
|
if (evt->getState() == mSupplicantState)
|
|
|
|
return;
|
|
|
|
|
2009-06-16 05:10:44 +08:00
|
|
|
LOGD("onStateChangeEvent(%s -> %s)",
|
|
|
|
SupplicantState::toString(mSupplicantState, tmp, sizeof(tmp)),
|
|
|
|
SupplicantState::toString(evt->getState(), tmp2, sizeof(tmp2)));
|
|
|
|
|
2009-06-24 12:10:57 +08:00
|
|
|
if (evt->getState() != SupplicantState::SCANNING) {
|
|
|
|
mIsSupplicantSearching = true;
|
|
|
|
mNumScanResultsSinceLastStateChange = 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
char *tmp3;
|
|
|
|
asprintf(&tmp3,
|
|
|
|
"Supplicant state changed from %d (%s) -> %d (%s)",
|
|
|
|
mSupplicantState, tmp, evt->getState(), tmp2);
|
|
|
|
|
2009-06-16 05:10:44 +08:00
|
|
|
mSupplicantState = evt->getState();
|
2009-06-24 12:10:57 +08:00
|
|
|
|
|
|
|
if (mSupplicantState == SupplicantState::COMPLETED) {
|
|
|
|
mStatusPoller->start();
|
|
|
|
} else if (mStatusPoller->isStarted()) {
|
|
|
|
mStatusPoller->stop();
|
|
|
|
}
|
|
|
|
|
|
|
|
NetworkManager::Instance()->getBroadcaster()->
|
|
|
|
sendBroadcast(ResponseCode::SupplicantStateChange,
|
|
|
|
tmp3, false);
|
|
|
|
free(tmp3);
|
2009-06-16 05:10:44 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
void WifiController::onConnectionTimeoutEvent(SupplicantConnectionTimeoutEvent *evt) {
|
|
|
|
LOGD("onConnectionTimeoutEvent(%s)", evt->getBssid());
|
|
|
|
}
|
|
|
|
|
|
|
|
void WifiController::onDisconnectedEvent(SupplicantDisconnectedEvent *evt) {
|
2009-06-24 12:10:57 +08:00
|
|
|
mCurrentlyConnectedNetworkId = -1;
|
|
|
|
mHandlers->onInterfaceDisconnected(this);
|
2009-06-16 05:10:44 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
#if 0
|
|
|
|
void WifiController::onTerminatingEvent(SupplicantEvent *evt) {
|
|
|
|
LOGD("onTerminatingEvent(%s)", evt->getEvent());
|
|
|
|
}
|
|
|
|
|
|
|
|
void WifiController::onPasswordChangedEvent(SupplicantEvent *evt) {
|
|
|
|
LOGD("onPasswordChangedEvent(%s)", evt->getEvent());
|
|
|
|
}
|
|
|
|
|
|
|
|
void WifiController::onEapNotificationEvent(SupplicantEvent *evt) {
|
|
|
|
LOGD("onEapNotificationEvent(%s)", evt->getEvent());
|
|
|
|
}
|
|
|
|
|
|
|
|
void WifiController::onEapStartedEvent(SupplicantEvent *evt) {
|
|
|
|
LOGD("onEapStartedEvent(%s)", evt->getEvent());
|
|
|
|
}
|
|
|
|
|
|
|
|
void WifiController::onEapMethodEvent(SupplicantEvent *evt) {
|
|
|
|
LOGD("onEapMethodEvent(%s)", evt->getEvent());
|
|
|
|
}
|
|
|
|
|
|
|
|
void WifiController::onEapSuccessEvent(SupplicantEvent *evt) {
|
|
|
|
LOGD("onEapSuccessEvent(%s)", evt->getEvent());
|
|
|
|
}
|
|
|
|
|
|
|
|
void WifiController::onEapFailureEvent(SupplicantEvent *evt) {
|
|
|
|
LOGD("onEapFailureEvent(%s)", evt->getEvent());
|
|
|
|
}
|
|
|
|
|
|
|
|
void WifiController::onLinkSpeedEvent(SupplicantEvent *evt) {
|
|
|
|
LOGD("onLinkSpeedEvent(%s)", evt->getEvent());
|
|
|
|
}
|
|
|
|
|
|
|
|
void WifiController::onDriverStateEvent(SupplicantEvent *evt) {
|
|
|
|
LOGD("onDriverStateEvent(%s)", evt->getEvent());
|
|
|
|
}
|
|
|
|
#endif
|
2009-06-24 12:10:57 +08:00
|
|
|
|
|
|
|
void WifiController::onStatusPollInterval() {
|
|
|
|
pthread_mutex_lock(&mLock);
|
|
|
|
int rssi;
|
|
|
|
if (mSupplicant->getRssi(&rssi)) {
|
|
|
|
LOGE("Failed to get rssi (%s)", strerror(errno));
|
|
|
|
pthread_mutex_unlock(&mLock);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (abs(mLastRssi - rssi) > mRssiEventThreshold) {
|
|
|
|
char *tmp3;
|
|
|
|
asprintf(&tmp3, "RSSI changed from %d -> %d",
|
|
|
|
mLastRssi, rssi);
|
|
|
|
mLastRssi = rssi;
|
|
|
|
NetworkManager::Instance()->getBroadcaster()->
|
|
|
|
sendBroadcast(ResponseCode::RssiChange,
|
|
|
|
tmp3, false);
|
|
|
|
free(tmp3);
|
|
|
|
}
|
|
|
|
|
|
|
|
int linkspeed = mSupplicant->getLinkSpeed();
|
|
|
|
if (linkspeed != mLastLinkSpeed) {
|
|
|
|
char *tmp3;
|
|
|
|
asprintf(&tmp3, "Link speed changed from %d -> %d",
|
|
|
|
mLastLinkSpeed, linkspeed);
|
|
|
|
mLastLinkSpeed = linkspeed;
|
|
|
|
NetworkManager::Instance()->getBroadcaster()->
|
|
|
|
sendBroadcast(ResponseCode::LinkSpeedChange,
|
|
|
|
tmp3, false);
|
|
|
|
free(tmp3);
|
|
|
|
|
|
|
|
}
|
|
|
|
pthread_mutex_unlock(&mLock);
|
|
|
|
}
|
|
|
|
|
|
|
|
int WifiController::verifyNotSuspended() {
|
|
|
|
if (mSuspended) {
|
|
|
|
errno = ESHUTDOWN;
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Property inner classes
|
|
|
|
*/
|
|
|
|
|
|
|
|
WifiController::WifiIntegerProperty::WifiIntegerProperty(WifiController *c,
|
|
|
|
const char *name,
|
|
|
|
bool ro,
|
|
|
|
int elements) :
|
|
|
|
IntegerProperty(name, ro, elements) {
|
|
|
|
mWc = c;
|
|
|
|
}
|
|
|
|
|
|
|
|
WifiController::WifiStringProperty::WifiStringProperty(WifiController *c,
|
|
|
|
const char *name,
|
|
|
|
bool ro, int elements) :
|
|
|
|
StringProperty(name, ro, elements) {
|
|
|
|
mWc = c;
|
|
|
|
}
|
|
|
|
|
|
|
|
WifiController::WifiEnabledProperty::WifiEnabledProperty(WifiController *c) :
|
|
|
|
WifiIntegerProperty(c, "Enabled", false, 1) {
|
|
|
|
}
|
|
|
|
|
|
|
|
int WifiController::WifiEnabledProperty::get(int idx, int *buffer) {
|
|
|
|
*buffer = mWc->mEnabled;
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
int WifiController::WifiEnabledProperty::set(int idx, int value) {
|
|
|
|
int rc = (value ? mWc->enable() : mWc->disable());
|
|
|
|
if (!rc)
|
|
|
|
mWc->mEnabled = value;
|
|
|
|
return rc;
|
|
|
|
}
|
|
|
|
|
|
|
|
WifiController::WifiScanOnlyProperty::WifiScanOnlyProperty(WifiController *c) :
|
|
|
|
WifiIntegerProperty(c, "ScanOnly", false, 1) {
|
|
|
|
}
|
|
|
|
int WifiController::WifiScanOnlyProperty::get(int idx, int *buffer) {
|
|
|
|
*buffer = mWc->mScanOnly;
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
int WifiController::WifiScanOnlyProperty::set(int idx, int value) {
|
|
|
|
return mWc->setScanOnly(value == 1);
|
|
|
|
}
|
|
|
|
|
|
|
|
WifiController::WifiAllowedChannelsProperty::WifiAllowedChannelsProperty(WifiController *c) :
|
|
|
|
WifiIntegerProperty(c, "AllowedChannels", false, 1) {
|
|
|
|
}
|
|
|
|
int WifiController::WifiAllowedChannelsProperty::get(int idx, int *buffer) {
|
|
|
|
*buffer = mWc->mNumAllowedChannels;
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
int WifiController::WifiAllowedChannelsProperty::set(int idx, int value) {
|
|
|
|
// XXX: IMPL
|
|
|
|
errno = ENOSYS;
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
WifiController::WifiSupplicantStateProperty::WifiSupplicantStateProperty(WifiController *c) :
|
|
|
|
WifiStringProperty(c, "SupplicantState", true, 1) {
|
|
|
|
}
|
|
|
|
int WifiController::WifiSupplicantStateProperty::get(int idx, char *buffer, size_t max) {
|
|
|
|
if (!SupplicantState::toString(mWc->mSupplicantState, buffer, max))
|
|
|
|
return -1;
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
WifiController::WifiActiveScanProperty::WifiActiveScanProperty(WifiController *c) :
|
|
|
|
WifiIntegerProperty(c, "ActiveScan", false, 1) {
|
|
|
|
}
|
|
|
|
int WifiController::WifiActiveScanProperty::get(int idx, int *buffer) {
|
|
|
|
*buffer = mWc->mActiveScan;
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
int WifiController::WifiActiveScanProperty::set(int idx, int value) {
|
|
|
|
return mWc->setActiveScan(value);
|
|
|
|
}
|
|
|
|
|
|
|
|
WifiController::WifiInterfaceProperty::WifiInterfaceProperty(WifiController *c) :
|
|
|
|
WifiStringProperty(c, "Interface", true, 1) {
|
|
|
|
}
|
|
|
|
int WifiController::WifiInterfaceProperty::get(int idx, char *buffer, size_t max) {
|
|
|
|
strncpy(buffer, (mWc->getBoundInterface() ? mWc->getBoundInterface() : "none"), max);
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
WifiController::WifiSearchingProperty::WifiSearchingProperty(WifiController *c) :
|
|
|
|
WifiIntegerProperty(c, "Searching", true, 1) {
|
|
|
|
}
|
|
|
|
int WifiController::WifiSearchingProperty::get(int idx, int *buffer) {
|
|
|
|
*buffer = mWc->mIsSupplicantSearching;
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
WifiController::WifiPacketFilterProperty::WifiPacketFilterProperty(WifiController *c) :
|
|
|
|
WifiIntegerProperty(c, "PacketFilter", false, 1) {
|
|
|
|
}
|
|
|
|
int WifiController::WifiPacketFilterProperty::get(int idx, int *buffer) {
|
|
|
|
*buffer = mWc->mPacketFilter;
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
int WifiController::WifiPacketFilterProperty::set(int idx, int value) {
|
|
|
|
return mWc->setPacketFilter(value);
|
|
|
|
}
|
|
|
|
|
|
|
|
WifiController::WifiBluetoothCoexScanProperty::WifiBluetoothCoexScanProperty(WifiController *c) :
|
|
|
|
WifiIntegerProperty(c, "BluetoothCoexScan", false, 1) {
|
|
|
|
}
|
|
|
|
int WifiController::WifiBluetoothCoexScanProperty::get(int idx, int *buffer) {
|
|
|
|
*buffer = mWc->mBluetoothCoexScan;
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
int WifiController::WifiBluetoothCoexScanProperty::set(int idx, int value) {
|
|
|
|
return mWc->setBluetoothCoexistenceScan(value == 1);
|
|
|
|
}
|
|
|
|
|
|
|
|
WifiController::WifiBluetoothCoexModeProperty::WifiBluetoothCoexModeProperty(WifiController *c) :
|
|
|
|
WifiIntegerProperty(c, "BluetoothCoexMode", false, 1) {
|
|
|
|
}
|
|
|
|
int WifiController::WifiBluetoothCoexModeProperty::get(int idx, int *buffer) {
|
|
|
|
*buffer = mWc->mBluetoothCoexMode;
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
int WifiController::WifiBluetoothCoexModeProperty::set(int idx, int value) {
|
|
|
|
return mWc->setBluetoothCoexistenceMode(value);
|
|
|
|
}
|
|
|
|
|
|
|
|
WifiController::WifiCurrentNetworkProperty::WifiCurrentNetworkProperty(WifiController *c) :
|
|
|
|
WifiIntegerProperty(c, "CurrentlyConnectedNetworkId", true, 1) {
|
|
|
|
}
|
|
|
|
int WifiController::WifiCurrentNetworkProperty::get(int idx, int *buffer) {
|
|
|
|
*buffer = mWc->mCurrentlyConnectedNetworkId;
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
WifiController::WifiSuspendedProperty::WifiSuspendedProperty(WifiController *c) :
|
|
|
|
WifiIntegerProperty(c, "Suspended", false, 1) {
|
|
|
|
}
|
|
|
|
int WifiController::WifiSuspendedProperty::get(int idx, int *buffer) {
|
|
|
|
*buffer = mWc->getSuspended();
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
int WifiController::WifiSuspendedProperty::set(int idx, int value) {
|
|
|
|
return mWc->setSuspend(value == 1);
|
|
|
|
}
|
|
|
|
|
|
|
|
WifiController::WifiNetCountProperty::WifiNetCountProperty(WifiController *c) :
|
|
|
|
WifiIntegerProperty(c, "NetCount", true, 1) {
|
|
|
|
}
|
|
|
|
int WifiController::WifiNetCountProperty::get(int idx, int *buffer) {
|
|
|
|
pthread_mutex_lock(&mWc->mLock);
|
|
|
|
*buffer = mWc->mSupplicant->getNetworkCount();
|
|
|
|
pthread_mutex_unlock(&mWc->mLock);
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
WifiController::WifiTriggerScanProperty::WifiTriggerScanProperty(WifiController *c) :
|
|
|
|
WifiIntegerProperty(c, "TriggerScan", false, 1) {
|
|
|
|
}
|
|
|
|
int WifiController::WifiTriggerScanProperty::get(int idx, int *buffer) {
|
|
|
|
// XXX: Need action type
|
|
|
|
*buffer = 0;
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
int WifiController::WifiTriggerScanProperty::set(int idx, int value) {
|
|
|
|
return mWc->triggerScan();
|
|
|
|
}
|
|
|
|
|