/* Copyright (c) 2015, The Linux Foundation. All rights reserved.
*
* Redistribution and use in source and binary forms, with or without
* modification, are permitted provided that the following conditions
* are met:
* * Redistributions of source code must retain the above copyright
* notice, this list of conditions and the following disclaimer.
* * Redistributions in binary form must reproduce the above copyright
* notice, this list of conditions and the following disclaimer in
* the documentation and/or other materials provided with the
* distribution.
* * Neither the name of The Linux Foundation nor the names of its
* contributors may be used to endorse or promote products derived
* from this software without specific prior written permission.
*
* THIS SOFTWARE IS PROVIDED "AS IS" AND ANY EXPRESS OR IMPLIED
* WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES OF
* MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NON-INFRINGEMENT
* ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT OWNER OR CONTRIBUTORS
* BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR
* CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF
* SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR
* BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY,
* WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE
* OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN
* IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
*/
#include "sync.h"
#define LOG_TAG "WifiHAL"
#include <utils/Log.h>
#include "wifi_hal.h"
#include "common.h"
#include "cpp_bindings.h"
#include "rssi_monitor.h"
#include "qca-vendor.h"
#include "vendor_definitions.h"
//Singleton Static Instance
RSSIMonitorCommand* RSSIMonitorCommand::mRSSIMonitorCommandInstance = NULL;
RSSIMonitorCommand::RSSIMonitorCommand(wifi_handle handle, int id,
u32 vendor_id, u32 subcmd)
: WifiVendorCommand(handle, id, vendor_id, subcmd)
{
mRSSIMonitorCommandInstance = NULL;
memset(&mHandler, 0, sizeof(mHandler));
}
RSSIMonitorCommand::~RSSIMonitorCommand()
{
mRSSIMonitorCommandInstance = NULL;
}
void RSSIMonitorCommand::setReqId(wifi_request_id reqid)
{
mId = reqid;
}
RSSIMonitorCommand* RSSIMonitorCommand::instance(wifi_handle handle,
wifi_request_id id)
{
if (handle == NULL) {
ALOGE("Interface Handle is invalid");
return NULL;
}
if (mRSSIMonitorCommandInstance == NULL) {
mRSSIMonitorCommandInstance = new RSSIMonitorCommand(handle, id,
OUI_QCA,
QCA_NL80211_VENDOR_SUBCMD_MONITOR_RSSI);
return mRSSIMonitorCommandInstance;
}
else
{
if (handle != getWifiHandle(mRSSIMonitorCommandInstance->mInfo))
{
/* upper layer must have cleaned up the handle and reinitialized,
so we need to update the same */
ALOGV("Handle different, update the handle");
mRSSIMonitorCommandInstance->mInfo = (hal_info *)handle;
}
mRSSIMonitorCommandInstance->setReqId(id);
}
return mRSSIMonitorCommandInstance;
}
/* This function will be the main handler for incoming event.
* Call the appropriate callback handler after parsing the vendor data.
*/
int RSSIMonitorCommand::handleEvent(WifiEvent &event)
{
int ret = WIFI_SUCCESS;
WifiVendorCommand::handleEvent(event);
/* Parse the vendordata and get the attribute */
switch(mSubcmd)
{
case QCA_NL80211_VENDOR_SUBCMD_MONITOR_RSSI:
{
mac_addr addr;
s8 rssi;
wifi_request_id reqId;
struct nlattr *tb_vendor[QCA_WLAN_VENDOR_ATTR_RSSI_MONITORING_MAX
+ 1];
nla_parse(tb_vendor, QCA_WLAN_VENDOR_ATTR_RSSI_MONITORING_MAX,
(struct nlattr *)mVendorData,
mDataLen, NULL);
memset(addr, 0, sizeof(mac_addr));
if (!tb_vendor[
QCA_WLAN_VENDOR_ATTR_RSSI_MONITORING_REQUEST_ID])
{
ALOGE("%s: ATTR_RSSI_MONITORING_REQUEST_ID not found. Exit.",
__FUNCTION__);
ret = WIFI_ERROR_INVALID_ARGS;
break;
}
reqId = nla_get_u32(
tb_vendor[QCA_WLAN_VENDOR_ATTR_RSSI_MONITORING_REQUEST_ID]
);
/* If event has a different request_id, ignore that and use the
* request_id value which we're maintaining.
*/
if (reqId != id()) {
ALOGV("%s: Event has Req. ID:%d <> Ours:%d, continue...",
__FUNCTION__, reqId, id());
reqId = id();
}
ret = get_mac_addr(tb_vendor,
QCA_WLAN_VENDOR_ATTR_RSSI_MONITORING_CUR_BSSID,
addr);
if (ret != WIFI_SUCCESS) {
return ret;
}
ALOGV(MAC_ADDR_STR, MAC_ADDR_ARRAY(addr));
if (!tb_vendor[QCA_WLAN_VENDOR_ATTR_RSSI_MONITORING_CUR_RSSI])
{
ALOGE("%s: QCA_WLAN_VENDOR_ATTR_RSSI_MONITORING_CUR_RSSI"
" not found", __FUNCTION__);
return WIFI_ERROR_INVALID_ARGS;
}
rssi = get_s8(tb_vendor[
QCA_WLAN_VENDOR_ATTR_RSSI_MONITORING_CUR_RSSI]);
ALOGV("Current RSSI : %d ", rssi);
if (mHandler.on_rssi_threshold_breached)
(*mHandler.on_rssi_threshold_breached)(reqId, addr, rssi);
else
ALOGE("RSSI Monitoring: No Callback registered: ");
}
break;
default:
/* Error case should not happen print log */
ALOGE("%s: Wrong subcmd received %d", __FUNCTION__, mSubcmd);
}
return NL_SKIP;
}
int RSSIMonitorCommand::setCallbackHandler(wifi_rssi_event_handler nHandler,
u32 event)
{
int ret;
mHandler = nHandler;
ret = registerVendorHandler(mVendor_id, event);
if (ret != 0) {
/* Error case should not happen print log */
ALOGE("%s: Unable to register Vendor Handler Vendor Id=0x%x subcmd=%u",
__FUNCTION__, mVendor_id, mSubcmd);
}
return ret;
}
wifi_error RSSIMonitorCommand::unregisterHandler(u32 subCmd)
{
unregisterVendorHandler(mVendor_id, subCmd);
return WIFI_SUCCESS;
}
wifi_error wifi_start_rssi_monitoring(wifi_request_id id,
wifi_interface_handle iface,
s8 max_rssi,
s8 min_rssi,
wifi_rssi_event_handler eh)
{
int ret = WIFI_SUCCESS;
struct nlattr *nlData;
WifiVendorCommand *vCommand = NULL;
wifi_handle wifiHandle = getWifiHandle(iface);
RSSIMonitorCommand *rssiCommand;
ret = initialize_vendor_cmd(iface, id,
QCA_NL80211_VENDOR_SUBCMD_MONITOR_RSSI,
&vCommand);
if (ret != WIFI_SUCCESS) {
ALOGE("%s: Initialization failed", __FUNCTION__);
return (wifi_error)ret;
}
ALOGV("%s: Max RSSI:%d Min RSSI:%d", __FUNCTION__,
max_rssi, min_rssi);
/* Add the vendor specific attributes for the NL command. */
nlData = vCommand->attr_start(NL80211_ATTR_VENDOR_DATA);
if (!nlData)
goto cleanup;
if (vCommand->put_u32(
QCA_WLAN_VENDOR_ATTR_RSSI_MONITORING_CONTROL,
QCA_WLAN_RSSI_MONITORING_START) ||
vCommand->put_u32(
QCA_WLAN_VENDOR_ATTR_RSSI_MONITORING_REQUEST_ID,
id) ||
vCommand->put_s8(
QCA_WLAN_VENDOR_ATTR_RSSI_MONITORING_MAX_RSSI,
max_rssi) ||
vCommand->put_s8(
QCA_WLAN_VENDOR_ATTR_RSSI_MONITORING_MIN_RSSI,
min_rssi))
{
goto cleanup;
}
vCommand->attr_end(nlData);
rssiCommand = RSSIMonitorCommand::instance(wifiHandle, id);
if (rssiCommand == NULL) {
ALOGE("%s: Error rssiCommand NULL", __FUNCTION__);
return WIFI_ERROR_OUT_OF_MEMORY;
}
ret = rssiCommand->setCallbackHandler(eh,
QCA_NL80211_VENDOR_SUBCMD_MONITOR_RSSI);
if (ret < 0)
goto cleanup;
ret = vCommand->requestResponse();
if (ret < 0)
goto cleanup;
cleanup:
delete vCommand;
return (wifi_error)ret;
}
wifi_error wifi_stop_rssi_monitoring(wifi_request_id id,
wifi_interface_handle iface)
{
int ret = WIFI_SUCCESS;
struct nlattr *nlData;
WifiVendorCommand *vCommand = NULL;
wifi_handle wifiHandle = getWifiHandle(iface);
RSSIMonitorCommand *rssiCommand;
ret = initialize_vendor_cmd(iface, id,
QCA_NL80211_VENDOR_SUBCMD_MONITOR_RSSI,
&vCommand);
if (ret != WIFI_SUCCESS) {
ALOGE("%s: Initialization failed", __FUNCTION__);
return (wifi_error)ret;
}
/* Add the vendor specific attributes for the NL command. */
nlData = vCommand->attr_start(NL80211_ATTR_VENDOR_DATA);
if (!nlData)
goto cleanup;
if (vCommand->put_u32(
QCA_WLAN_VENDOR_ATTR_RSSI_MONITORING_CONTROL,
QCA_WLAN_RSSI_MONITORING_STOP) ||
vCommand->put_u32(
QCA_WLAN_VENDOR_ATTR_RSSI_MONITORING_REQUEST_ID,
id))
{
goto cleanup;
}
vCommand->attr_end(nlData);
ret = vCommand->requestResponse();
if (ret < 0)
goto cleanup;
rssiCommand = RSSIMonitorCommand::instance(wifiHandle, id);
if (rssiCommand == NULL) {
ALOGE("%s: Error rssiCommand NULL", __FUNCTION__);
ret = WIFI_ERROR_OUT_OF_MEMORY;
goto cleanup;
}
ret = rssiCommand->unregisterHandler(
QCA_NL80211_VENDOR_SUBCMD_MONITOR_RSSI);
if (ret != WIFI_SUCCESS)
goto cleanup;
delete rssiCommand;
cleanup:
delete vCommand;
return (wifi_error)ret;
}