mirror of
https://github.com/nxp-imx/mwifiex.git
synced 2025-01-15 16:25:35 +00:00
8ffae47921
changes: 1. WCSWREL-191: Fixed the error when loading module param from user config for SD8801 2. WCSWREL-186: Fixed the issue of mlanutl failing on kernel higher than L5.15 3. Fixed low throughput issue for WPA3 SAE 4. Added driver change for WLAN throughput improvement on 8997 SoC 5. Updated README to recommend not to use WEP/TKIP for all chipsets 6. WCSWREL-180: Fix P2P test fail on kernel higher than L5.12 7. WCSWREL-156: kernel_write/kernel_read not allowed by drivers for L5.10 kernel GKI buildou 8. Alternative for pm_qos_add_request/pm_qos_remove_request Signed-off-by: Sherry Sun <sherry.sun@nxp.com> Approved-by: Tian Yang <yang.tian@nxp.com>
5703 lines
188 KiB
C
5703 lines
188 KiB
C
/** @file mlan_uap_cmdevent.c
|
|
*
|
|
* @brief This file contains the handling of AP mode command and event
|
|
*
|
|
*
|
|
* Copyright 2009-2021 NXP
|
|
*
|
|
* This software file (the File) is distributed by NXP
|
|
* under the terms of the GNU General Public License Version 2, June 1991
|
|
* (the License). You may use, redistribute and/or modify the File in
|
|
* accordance with the terms and conditions of the License, a copy of which
|
|
* is available by writing to the Free Software Foundation, Inc.,
|
|
* 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA or on the
|
|
* worldwide web at http://www.gnu.org/licenses/old-licenses/gpl-2.0.txt.
|
|
*
|
|
* THE FILE IS DISTRIBUTED AS-IS, WITHOUT WARRANTY OF ANY KIND, AND THE
|
|
* IMPLIED WARRANTIES OF MERCHANTABILITY OR FITNESS FOR A PARTICULAR PURPOSE
|
|
* ARE EXPRESSLY DISCLAIMED. The License provides additional details about
|
|
* this warranty disclaimer.
|
|
*
|
|
*/
|
|
|
|
/********************************************************
|
|
Change log:
|
|
02/05/2009: initial version
|
|
********************************************************/
|
|
|
|
#include "mlan.h"
|
|
#include "mlan_util.h"
|
|
#include "mlan_fw.h"
|
|
#ifdef STA_SUPPORT
|
|
#include "mlan_join.h"
|
|
#endif
|
|
#include "mlan_main.h"
|
|
#include "mlan_uap.h"
|
|
#ifdef SDIO
|
|
#include "mlan_sdio.h"
|
|
#endif /* SDIO */
|
|
#include "mlan_11n.h"
|
|
#include "mlan_11h.h"
|
|
#include "mlan_11ac.h"
|
|
#include "mlan_11ax.h"
|
|
#ifdef DRV_EMBEDDED_AUTHENTICATOR
|
|
#include "authenticator_api.h"
|
|
#endif
|
|
#ifdef PCIE
|
|
#include "mlan_pcie.h"
|
|
#endif /* PCIE */
|
|
/********************************************************
|
|
Local Functions
|
|
********************************************************/
|
|
|
|
/**
|
|
* @brief This function prepares command of BAND_STEERING_CFG
|
|
*
|
|
* @param pmpriv A pointer to mlan_private structure
|
|
* @param cmd A pointer to HostCmd_DS_COMMAND structure
|
|
* @param cmd_action
|
|
* @param pdata_buf A pointer to data buffer
|
|
* @return MLAN_STATUS_SUCCESS or MLAN_STATUS_FAILURE
|
|
*/
|
|
static mlan_status wlan_cmd_set_get_band_steering_cfg(pmlan_private pmpriv,
|
|
HostCmd_DS_COMMAND *cmd,
|
|
t_u16 cmd_action,
|
|
t_void *pdata_buf)
|
|
{
|
|
mlan_ds_band_steer_cfg *pband_steer_cfg =
|
|
(mlan_ds_band_steer_cfg *)pdata_buf;
|
|
ENTER();
|
|
|
|
cmd->command = wlan_cpu_to_le16(HostCmd_CMD_802_11_BAND_STEERING);
|
|
cmd->size =
|
|
wlan_cpu_to_le16(sizeof(HostCmd_DS_BAND_STEERING) + S_DS_GEN);
|
|
cmd->params.band_steer_info.state = pband_steer_cfg->state;
|
|
cmd->params.band_steer_info.block_2g_prb_req =
|
|
pband_steer_cfg->block_2g_prb_req;
|
|
cmd->params.band_steer_info.max_btm_req_allowed =
|
|
pband_steer_cfg->max_btm_req_allowed;
|
|
cmd->params.band_steer_info.action = cmd_action;
|
|
|
|
LEAVE();
|
|
return MLAN_STATUS_SUCCESS;
|
|
}
|
|
|
|
/**
|
|
* @brief This function handle response of HostCmd_CMD_802_11_BAND_STEERING
|
|
*
|
|
* @param pmpriv A pointer to mlan_private structure
|
|
* @param resp Pointer to command response buffer
|
|
*
|
|
* @return MLAN_STATUS_SUCCESS or MLAN_STATUS_FAILURE
|
|
*/
|
|
static mlan_status
|
|
wlan_ret_set_get_band_steering_cfg(mlan_private *pmpriv,
|
|
HostCmd_DS_COMMAND *resp,
|
|
mlan_ioctl_req *pioctl_buf)
|
|
{
|
|
mlan_status ret = MLAN_STATUS_SUCCESS;
|
|
HostCmd_DS_BAND_STEERING *pband_steer_info =
|
|
&resp->params.band_steer_info;
|
|
mlan_ds_misc_cfg *pband_steer;
|
|
|
|
ENTER();
|
|
|
|
pband_steer = (mlan_ds_misc_cfg *)pioctl_buf->pbuf;
|
|
|
|
pband_steer->param.band_steer_cfg.action = pband_steer_info->action;
|
|
pband_steer->param.band_steer_cfg.state = pband_steer_info->state;
|
|
pband_steer->param.band_steer_cfg.block_2g_prb_req =
|
|
pband_steer_info->block_2g_prb_req;
|
|
pband_steer->param.band_steer_cfg.max_btm_req_allowed =
|
|
pband_steer_info->max_btm_req_allowed;
|
|
|
|
LEAVE();
|
|
return ret;
|
|
}
|
|
|
|
/**
|
|
* @brief This function prepares command of BEACON_STUCK_CFG
|
|
*
|
|
* @param pmpriv A pointer to mlan_private structure
|
|
* @param cmd A pointer to HostCmd_DS_COMMAND structure
|
|
* @param cmd_action
|
|
* @param pdata_buf A pointer to data buffer
|
|
* @return MLAN_STATUS_SUCCESS or MLAN_STATUS_FAILURE
|
|
*/
|
|
static mlan_status wlan_cmd_set_get_beacon_stuck_cfg(IN pmlan_private pmpriv,
|
|
IN HostCmd_DS_COMMAND *cmd,
|
|
IN t_u16 cmd_action,
|
|
IN t_void *pdata_buf)
|
|
{
|
|
HostCmd_DS_BEACON_STUCK_CFG *pbeacon_stuck_param_cfg =
|
|
(HostCmd_DS_BEACON_STUCK_CFG *)(pdata_buf + sizeof(t_u32));
|
|
|
|
ENTER();
|
|
|
|
cmd->command = wlan_cpu_to_le16(HostCmd_CMD_UAP_BEACON_STUCK_CFG);
|
|
cmd->size = wlan_cpu_to_le16(sizeof(HostCmd_DS_BEACON_STUCK_CFG) +
|
|
S_DS_GEN);
|
|
cmd->params.beacon_stuck_cfg.beacon_stuck_detect_count =
|
|
pbeacon_stuck_param_cfg->beacon_stuck_detect_count;
|
|
cmd->params.beacon_stuck_cfg.recovery_confirm_count =
|
|
pbeacon_stuck_param_cfg->recovery_confirm_count;
|
|
cmd->params.beacon_stuck_cfg.action = cmd_action;
|
|
|
|
LEAVE();
|
|
return MLAN_STATUS_SUCCESS;
|
|
}
|
|
|
|
/**
|
|
* @brief This function handle response of HostCmd_CMD_UAP_BEACON_STUCK_CFG
|
|
*
|
|
* @param pmpriv A pointer to mlan_private structure
|
|
* @param resp Pointer to command response buffer
|
|
*
|
|
* @return MLAN_STATUS_SUCCESS or MLAN_STATUS_FAILURE
|
|
*/
|
|
static mlan_status wlan_ret_set_get_beacon_stuck_cfg(mlan_private *pmpriv,
|
|
HostCmd_DS_COMMAND *resp,
|
|
mlan_ioctl_req *pioctl_buf)
|
|
{
|
|
mlan_status ret = MLAN_STATUS_SUCCESS;
|
|
HostCmd_DS_BEACON_STUCK_CFG *pbeacon_stuck_param_cfg =
|
|
&resp->params.beacon_stuck_cfg;
|
|
mlan_ds_misc_cfg *pbeacon_stuck;
|
|
|
|
ENTER();
|
|
|
|
pbeacon_stuck = (mlan_ds_misc_cfg *)pioctl_buf->pbuf;
|
|
|
|
pbeacon_stuck->param.beacon_stuck_cfg.action =
|
|
pbeacon_stuck_param_cfg->action;
|
|
pbeacon_stuck->param.beacon_stuck_cfg.beacon_stuck_detect_count =
|
|
pbeacon_stuck_param_cfg->beacon_stuck_detect_count;
|
|
pbeacon_stuck->param.beacon_stuck_cfg.recovery_confirm_count =
|
|
pbeacon_stuck_param_cfg->recovery_confirm_count;
|
|
|
|
LEAVE();
|
|
return ret;
|
|
}
|
|
|
|
/**
|
|
* @brief This function handles the command response error
|
|
*
|
|
* @param pmpriv A pointer to mlan_private structure
|
|
* @param resp A pointer to HostCmd_DS_COMMAND
|
|
* @param pioctl_buf A pointer to command buffer
|
|
*
|
|
* @return N/A
|
|
*/
|
|
static mlan_status uap_process_cmdresp_error(mlan_private *pmpriv,
|
|
HostCmd_DS_COMMAND *resp,
|
|
mlan_ioctl_req *pioctl_buf)
|
|
{
|
|
mlan_adapter *pmadapter = pmpriv->adapter;
|
|
#if defined(USB)
|
|
t_u8 i;
|
|
#endif
|
|
mlan_status ret = MLAN_STATUS_FAILURE;
|
|
|
|
ENTER();
|
|
if (resp->command != HostCmd_CMD_WMM_PARAM_CONFIG ||
|
|
resp->command != HostCmd_CMD_CHAN_REGION_CFG)
|
|
PRINTM(MERROR, "CMD_RESP: cmd %#x error, result=%#x\n",
|
|
resp->command, resp->result);
|
|
if (pioctl_buf)
|
|
pioctl_buf->status_code = resp->result;
|
|
/*
|
|
* Handling errors here
|
|
*/
|
|
switch (resp->command) {
|
|
#ifdef SDIO
|
|
case HostCmd_CMD_SDIO_SP_RX_AGGR_CFG:
|
|
pmadapter->pcard_sd->sdio_rx_aggr_enable = MFALSE;
|
|
PRINTM(MMSG, "FW don't support SDIO single port rx aggr\n");
|
|
break;
|
|
#endif
|
|
|
|
case HOST_CMD_APCMD_SYS_CONFIGURE: {
|
|
HostCmd_DS_SYS_CONFIG *sys_config =
|
|
(HostCmd_DS_SYS_CONFIG *)&resp->params.sys_config;
|
|
t_u16 resp_len = 0, travel_len = 0, index;
|
|
mlan_ds_misc_custom_ie *cust_ie = MNULL;
|
|
mlan_ds_misc_cfg *misc = MNULL;
|
|
custom_ie *cptr;
|
|
|
|
if (!pioctl_buf || (pioctl_buf->req_id != MLAN_IOCTL_MISC_CFG))
|
|
break;
|
|
misc = (mlan_ds_misc_cfg *)pioctl_buf->pbuf;
|
|
if ((pioctl_buf->action == MLAN_ACT_SET) &&
|
|
(misc->sub_command == MLAN_OID_MISC_CUSTOM_IE)) {
|
|
cust_ie = (mlan_ds_misc_custom_ie *)
|
|
sys_config->tlv_buffer;
|
|
if (cust_ie) {
|
|
cust_ie->type = wlan_le16_to_cpu(cust_ie->type);
|
|
resp_len = cust_ie->len =
|
|
wlan_le16_to_cpu(cust_ie->len);
|
|
travel_len = 0;
|
|
/* conversion for index, mask, len */
|
|
if (resp_len == sizeof(t_u16))
|
|
cust_ie->ie_data_list[0].ie_index =
|
|
wlan_cpu_to_le16(
|
|
cust_ie->ie_data_list[0]
|
|
.ie_index);
|
|
|
|
while (resp_len > sizeof(t_u16)) {
|
|
cptr = (custom_ie
|
|
*)(((t_u8 *)cust_ie
|
|
->ie_data_list) +
|
|
travel_len);
|
|
index = cptr->ie_index =
|
|
wlan_le16_to_cpu(
|
|
cptr->ie_index);
|
|
cptr->mgmt_subtype_mask =
|
|
wlan_le16_to_cpu(
|
|
cptr->mgmt_subtype_mask);
|
|
cptr->ie_length = wlan_le16_to_cpu(
|
|
cptr->ie_length);
|
|
travel_len += cptr->ie_length +
|
|
sizeof(custom_ie) -
|
|
MAX_IE_SIZE;
|
|
resp_len -= cptr->ie_length +
|
|
sizeof(custom_ie) -
|
|
MAX_IE_SIZE;
|
|
if ((pmpriv->mgmt_ie[index]
|
|
.mgmt_subtype_mask ==
|
|
cptr->mgmt_subtype_mask) &&
|
|
(pmpriv->mgmt_ie[index].ie_length ==
|
|
cptr->ie_length) &&
|
|
!memcmp(pmpriv->adapter,
|
|
pmpriv->mgmt_ie[index]
|
|
.ie_buffer,
|
|
cptr->ie_buffer,
|
|
cptr->ie_length)) {
|
|
PRINTM(MERROR,
|
|
"set custom ie fail, remove ie index :%d\n",
|
|
index);
|
|
memset(pmadapter,
|
|
&pmpriv->mgmt_ie[index],
|
|
0, sizeof(custom_ie));
|
|
}
|
|
}
|
|
}
|
|
}
|
|
} break;
|
|
case HostCmd_CMD_PACKET_AGGR_CTRL:
|
|
#ifdef USB
|
|
if (IS_USB(pmadapter->card_type)) {
|
|
for (i = 0; i < MAX_USB_TX_PORT_NUM; i++)
|
|
pmadapter->pcard_usb->usb_tx_aggr[i]
|
|
.aggr_ctrl.enable = MFALSE;
|
|
pmadapter->pcard_usb->usb_rx_deaggr.aggr_ctrl.enable =
|
|
MFALSE;
|
|
}
|
|
#endif
|
|
break;
|
|
case HostCmd_CMD_CHAN_REGION_CFG:
|
|
ret = MLAN_STATUS_SUCCESS;
|
|
PRINTM(MCMND, "FW don't support chan region cfg command!\n");
|
|
break;
|
|
#if defined(DRV_EMBEDDED_AUTHENTICATOR)
|
|
case HostCmd_CMD_CRYPTO:
|
|
PRINTM(MCMND, "crypto cmd result=0x%x!\n", resp->result);
|
|
ret = wlan_ret_crypto(pmpriv, resp, pioctl_buf);
|
|
break;
|
|
#endif
|
|
default:
|
|
break;
|
|
}
|
|
|
|
wlan_request_cmd_lock(pmadapter);
|
|
wlan_insert_cmd_to_free_q(pmadapter, pmadapter->curr_cmd);
|
|
pmadapter->curr_cmd = MNULL;
|
|
wlan_release_cmd_lock(pmadapter);
|
|
|
|
LEAVE();
|
|
return ret;
|
|
}
|
|
|
|
/**
|
|
* @brief This function will return the pointer to station entry in station
|
|
* list table which matches the give mac address
|
|
*
|
|
* @param priv A pointer to mlan_private
|
|
*
|
|
* @return A pointer to structure sta_node
|
|
*/
|
|
static void wlan_notify_station_deauth(mlan_private *priv)
|
|
{
|
|
sta_node *sta_ptr;
|
|
t_u8 event_buf[100];
|
|
mlan_event *pevent = (mlan_event *)event_buf;
|
|
t_u8 *pbuf;
|
|
|
|
ENTER();
|
|
sta_ptr = (sta_node *)util_peek_list(
|
|
priv->adapter->pmoal_handle, &priv->sta_list,
|
|
priv->adapter->callbacks.moal_spin_lock,
|
|
priv->adapter->callbacks.moal_spin_unlock);
|
|
if (!sta_ptr) {
|
|
LEAVE();
|
|
return;
|
|
}
|
|
while (sta_ptr != (sta_node *)&priv->sta_list) {
|
|
memset(priv->adapter, event_buf, 0, sizeof(event_buf));
|
|
pevent->bss_index = priv->bss_index;
|
|
pevent->event_id = MLAN_EVENT_ID_UAP_FW_STA_DISCONNECT;
|
|
pevent->event_len = MLAN_MAC_ADDR_LENGTH + 2;
|
|
pbuf = (t_u8 *)pevent->event_buf;
|
|
/* reason field set to 0, Unspecified */
|
|
memcpy_ext(priv->adapter, pbuf + 2, sta_ptr->mac_addr,
|
|
MLAN_MAC_ADDR_LENGTH, MLAN_MAC_ADDR_LENGTH);
|
|
wlan_recv_event(priv, pevent->event_id, pevent);
|
|
sta_ptr = sta_ptr->pnext;
|
|
}
|
|
LEAVE();
|
|
return;
|
|
}
|
|
|
|
/**
|
|
* @brief This function prepares command of hs_cfg.
|
|
*
|
|
* @param pmpriv A pointer to mlan_private structure
|
|
* @param cmd A pointer to HostCmd_DS_COMMAND structure
|
|
* @param cmd_action The action: GET or SET
|
|
* @param pdata_buf A pointer to data buffer
|
|
*
|
|
* @return MLAN_STATUS_SUCCESS
|
|
*/
|
|
static mlan_status wlan_uap_cmd_802_11_hs_cfg(pmlan_private pmpriv,
|
|
HostCmd_DS_COMMAND *cmd,
|
|
t_u16 cmd_action,
|
|
hs_config_param *pdata_buf)
|
|
{
|
|
pmlan_adapter pmadapter = pmpriv->adapter;
|
|
HostCmd_DS_802_11_HS_CFG_ENH *phs_cfg =
|
|
(HostCmd_DS_802_11_HS_CFG_ENH *)&(cmd->params.opt_hs_cfg);
|
|
t_u8 *tlv = (t_u8 *)phs_cfg + sizeof(HostCmd_DS_802_11_HS_CFG_ENH);
|
|
MrvlIEtypes_HsWakeHoldoff_t *holdoff_tlv = MNULL;
|
|
MrvlIEtypes_HS_Antmode_t *antmode_tlv = MNULL;
|
|
MrvlIEtypes_WakeupSourceGPIO_t *gpio_tlv = MNULL;
|
|
MrvlIEtypes_MgmtFrameFilter_t *mgmt_filter_tlv = MNULL;
|
|
MrvlIEtypes_WakeupExtend_t *ext_tlv = MNULL;
|
|
|
|
ENTER();
|
|
cmd->command = wlan_cpu_to_le16(HostCmd_CMD_802_11_HS_CFG_ENH);
|
|
cmd->size = wlan_cpu_to_le16(S_DS_GEN +
|
|
sizeof(HostCmd_DS_802_11_HS_CFG_ENH));
|
|
|
|
if (pdata_buf == MNULL) {
|
|
phs_cfg->action = wlan_cpu_to_le16(HS_ACTIVATE);
|
|
phs_cfg->params.hs_activate.resp_ctrl =
|
|
wlan_cpu_to_le16(RESP_NEEDED);
|
|
} else {
|
|
phs_cfg->action = wlan_cpu_to_le16(HS_CONFIGURE);
|
|
phs_cfg->params.hs_config.conditions =
|
|
wlan_cpu_to_le32(pdata_buf->conditions);
|
|
phs_cfg->params.hs_config.gpio = pdata_buf->gpio;
|
|
phs_cfg->params.hs_config.gap = pdata_buf->gap;
|
|
if (pmpriv->adapter->min_wake_holdoff) {
|
|
cmd->size = wlan_cpu_to_le16(
|
|
S_DS_GEN +
|
|
sizeof(HostCmd_DS_802_11_HS_CFG_ENH) +
|
|
sizeof(MrvlIEtypes_HsWakeHoldoff_t));
|
|
holdoff_tlv = (MrvlIEtypes_HsWakeHoldoff_t *)tlv;
|
|
holdoff_tlv->header.type =
|
|
wlan_cpu_to_le16(TLV_TYPE_HS_WAKE_HOLDOFF);
|
|
holdoff_tlv->header.len = wlan_cpu_to_le16(
|
|
sizeof(MrvlIEtypes_HsWakeHoldoff_t) -
|
|
sizeof(MrvlIEtypesHeader_t));
|
|
holdoff_tlv->min_wake_holdoff = wlan_cpu_to_le16(
|
|
pmpriv->adapter->min_wake_holdoff);
|
|
tlv += sizeof(MrvlIEtypes_HsWakeHoldoff_t);
|
|
}
|
|
PRINTM(MCMND,
|
|
"HS_CFG_CMD: condition:0x%x gpio:0x%x gap:0x%x holdoff=%d\n",
|
|
phs_cfg->params.hs_config.conditions,
|
|
phs_cfg->params.hs_config.gpio,
|
|
phs_cfg->params.hs_config.gap,
|
|
pmpriv->adapter->min_wake_holdoff);
|
|
|
|
if (pmadapter->param_type_ind == 1) {
|
|
cmd->size += sizeof(MrvlIEtypes_WakeupSourceGPIO_t);
|
|
gpio_tlv = (MrvlIEtypes_WakeupSourceGPIO_t *)tlv;
|
|
gpio_tlv->header.type = wlan_cpu_to_le16(
|
|
TLV_TYPE_HS_WAKEUP_SOURCE_GPIO);
|
|
gpio_tlv->header.len = wlan_cpu_to_le16(
|
|
sizeof(MrvlIEtypes_WakeupSourceGPIO_t) -
|
|
sizeof(MrvlIEtypesHeader_t));
|
|
gpio_tlv->ind_gpio = (t_u8)pmadapter->ind_gpio;
|
|
gpio_tlv->level = (t_u8)pmadapter->level;
|
|
tlv += sizeof(MrvlIEtypes_WakeupSourceGPIO_t);
|
|
}
|
|
if (pmadapter->param_type_ext == 2) {
|
|
cmd->size += sizeof(MrvlIEtypes_WakeupExtend_t);
|
|
ext_tlv = (MrvlIEtypes_WakeupExtend_t *)tlv;
|
|
ext_tlv->header.type =
|
|
wlan_cpu_to_le16(TLV_TYPE_WAKEUP_EXTEND);
|
|
ext_tlv->header.len = wlan_cpu_to_le16(
|
|
sizeof(MrvlIEtypes_WakeupExtend_t) -
|
|
sizeof(MrvlIEtypesHeader_t));
|
|
ext_tlv->event_force_ignore =
|
|
wlan_cpu_to_le32(pmadapter->event_force_ignore);
|
|
ext_tlv->event_use_ext_gap =
|
|
wlan_cpu_to_le32(pmadapter->event_use_ext_gap);
|
|
ext_tlv->ext_gap = pmadapter->ext_gap;
|
|
ext_tlv->gpio_wave = pmadapter->gpio_wave;
|
|
tlv += sizeof(MrvlIEtypes_WakeupExtend_t);
|
|
}
|
|
if (pmadapter->mgmt_filter[0].type) {
|
|
int i = 0;
|
|
mgmt_frame_filter mgmt_filter[MAX_MGMT_FRAME_FILTER];
|
|
memset(pmadapter, mgmt_filter, 0,
|
|
MAX_MGMT_FRAME_FILTER *
|
|
sizeof(mgmt_frame_filter));
|
|
mgmt_filter_tlv = (MrvlIEtypes_MgmtFrameFilter_t *)tlv;
|
|
mgmt_filter_tlv->header.type =
|
|
wlan_cpu_to_le16(TLV_TYPE_MGMT_FRAME_WAKEUP);
|
|
tlv += sizeof(MrvlIEtypesHeader_t);
|
|
while (i < MAX_MGMT_FRAME_FILTER &&
|
|
pmadapter->mgmt_filter[i].type) {
|
|
mgmt_filter[i].action =
|
|
(t_u8)pmadapter->mgmt_filter[i].action;
|
|
mgmt_filter[i].type =
|
|
(t_u8)pmadapter->mgmt_filter[i].type;
|
|
mgmt_filter[i].frame_mask = wlan_cpu_to_le32(
|
|
pmadapter->mgmt_filter[i].frame_mask);
|
|
i++;
|
|
}
|
|
memcpy_ext(pmadapter, (t_u8 *)mgmt_filter_tlv->filter,
|
|
(t_u8 *)mgmt_filter,
|
|
i * sizeof(mgmt_frame_filter),
|
|
sizeof(mgmt_filter_tlv->filter));
|
|
tlv += i * sizeof(mgmt_frame_filter);
|
|
mgmt_filter_tlv->header.len =
|
|
wlan_cpu_to_le16(i * sizeof(mgmt_frame_filter));
|
|
cmd->size += i * sizeof(mgmt_frame_filter) +
|
|
sizeof(MrvlIEtypesHeader_t);
|
|
}
|
|
if (pmadapter->hs_mimo_switch) {
|
|
cmd->size += sizeof(MrvlIEtypes_HS_Antmode_t);
|
|
antmode_tlv = (MrvlIEtypes_HS_Antmode_t *)tlv;
|
|
antmode_tlv->header.type =
|
|
wlan_cpu_to_le16(TLV_TYPE_HS_ANTMODE);
|
|
antmode_tlv->header.len = wlan_cpu_to_le16(
|
|
sizeof(MrvlIEtypes_HS_Antmode_t) -
|
|
sizeof(MrvlIEtypesHeader_t));
|
|
antmode_tlv->txpath_antmode = ANTMODE_FW_DECISION;
|
|
antmode_tlv->rxpath_antmode = ANTMODE_FW_DECISION;
|
|
tlv += sizeof(MrvlIEtypes_HS_Antmode_t);
|
|
PRINTM(MCMND,
|
|
"hs_mimo_switch=%d, txpath_antmode=%d, rxpath_antmode=%d\n",
|
|
pmadapter->hs_mimo_switch,
|
|
antmode_tlv->txpath_antmode,
|
|
antmode_tlv->rxpath_antmode);
|
|
}
|
|
}
|
|
LEAVE();
|
|
return MLAN_STATUS_SUCCESS;
|
|
}
|
|
|
|
/**
|
|
* @brief This function prepares command of Tx data pause
|
|
*
|
|
* @param pmpriv A pointer to mlan_private structure
|
|
* @param cmd A pointer to HostCmd_DS_COMMAND structure
|
|
* @param cmd_action the action: GET or SET
|
|
* @param pdata_buf A pointer to data buffer
|
|
* @return MLAN_STATUS_SUCCESS
|
|
*/
|
|
static mlan_status wlan_uap_cmd_txdatapause(pmlan_private pmpriv,
|
|
HostCmd_DS_COMMAND *cmd,
|
|
t_u16 cmd_action, t_void *pdata_buf)
|
|
{
|
|
HostCmd_DS_CMD_TX_DATA_PAUSE *pause_cmd =
|
|
(HostCmd_DS_CMD_TX_DATA_PAUSE *)&cmd->params.tx_data_pause;
|
|
mlan_ds_misc_tx_datapause *data_pause =
|
|
(mlan_ds_misc_tx_datapause *)pdata_buf;
|
|
|
|
ENTER();
|
|
|
|
cmd->command = wlan_cpu_to_le16(HostCmd_CMD_CFG_TX_DATA_PAUSE);
|
|
cmd->size = wlan_cpu_to_le16(sizeof(HostCmd_DS_CMD_TX_DATA_PAUSE) +
|
|
S_DS_GEN);
|
|
pause_cmd->action = wlan_cpu_to_le16(cmd_action);
|
|
|
|
if (cmd_action == HostCmd_ACT_GEN_SET) {
|
|
pause_cmd->enable_tx_pause = (t_u8)data_pause->tx_pause;
|
|
pause_cmd->pause_tx_count = (t_u8)data_pause->tx_buf_cnt;
|
|
}
|
|
|
|
LEAVE();
|
|
return MLAN_STATUS_SUCCESS;
|
|
}
|
|
|
|
/**
|
|
* @brief This function handles the command response of Tx data pause
|
|
*
|
|
* @param pmpriv A pointer to mlan_private structure
|
|
* @param resp A pointer to HostCmd_DS_COMMAND
|
|
* @param pioctl_buf A pointer to mlan_ioctl_req structure
|
|
*
|
|
* @return MLAN_STATUS_SUCCESS
|
|
*/
|
|
static mlan_status wlan_uap_ret_txdatapause(pmlan_private pmpriv,
|
|
HostCmd_DS_COMMAND *resp,
|
|
mlan_ioctl_req *pioctl_buf)
|
|
{
|
|
HostCmd_DS_CMD_TX_DATA_PAUSE *pause_cmd =
|
|
(HostCmd_DS_CMD_TX_DATA_PAUSE *)&resp->params.tx_data_pause;
|
|
mlan_ds_misc_cfg *misc_cfg = MNULL;
|
|
|
|
ENTER();
|
|
|
|
if (pioctl_buf) {
|
|
misc_cfg = (mlan_ds_misc_cfg *)pioctl_buf->pbuf;
|
|
misc_cfg->param.tx_datapause.tx_pause =
|
|
pause_cmd->enable_tx_pause;
|
|
misc_cfg->param.tx_datapause.tx_buf_cnt =
|
|
pause_cmd->pause_tx_count;
|
|
}
|
|
LEAVE();
|
|
return MLAN_STATUS_SUCCESS;
|
|
}
|
|
|
|
/**
|
|
* @brief This function will process tx pause event
|
|
*
|
|
* @param priv A pointer to mlan_private
|
|
* @param pevent A pointer to event buf
|
|
*
|
|
* @return N/A
|
|
*/
|
|
static void wlan_process_tx_pause_event(pmlan_private priv, pmlan_buffer pevent)
|
|
{
|
|
t_u16 tlv_type, tlv_len;
|
|
int tlv_buf_left = pevent->data_len - sizeof(t_u32);
|
|
MrvlIEtypesHeader_t *tlv =
|
|
(MrvlIEtypesHeader_t *)(pevent->pbuf + pevent->data_offset +
|
|
sizeof(t_u32));
|
|
MrvlIEtypes_tx_pause_t *tx_pause_tlv;
|
|
sta_node *sta_ptr = MNULL;
|
|
t_u8 bc_mac[] = {0xff, 0xff, 0xff, 0xff, 0xff, 0xff};
|
|
t_u32 total_pkts_queued;
|
|
t_u16 tx_pkts_queued = 0;
|
|
;
|
|
|
|
ENTER();
|
|
|
|
total_pkts_queued =
|
|
util_scalar_read(priv->adapter->pmoal_handle,
|
|
&priv->wmm.tx_pkts_queued, MNULL, MNULL);
|
|
while (tlv_buf_left >= (int)sizeof(MrvlIEtypesHeader_t)) {
|
|
tlv_type = wlan_le16_to_cpu(tlv->type);
|
|
tlv_len = wlan_le16_to_cpu(tlv->len);
|
|
if ((sizeof(MrvlIEtypesHeader_t) + tlv_len) >
|
|
(unsigned int)tlv_buf_left) {
|
|
PRINTM(MERROR, "wrong tlv: tlvLen=%d, tlvBufLeft=%d\n",
|
|
tlv_len, tlv_buf_left);
|
|
break;
|
|
}
|
|
if (tlv_type == TLV_TYPE_TX_PAUSE) {
|
|
tx_pause_tlv = (MrvlIEtypes_tx_pause_t *)tlv;
|
|
|
|
if (!memcmp(priv->adapter, bc_mac,
|
|
tx_pause_tlv->peermac,
|
|
MLAN_MAC_ADDR_LENGTH))
|
|
tx_pkts_queued = wlan_update_ralist_tx_pause(
|
|
priv, tx_pause_tlv->peermac,
|
|
tx_pause_tlv->tx_pause);
|
|
else if (!memcmp(priv->adapter, priv->curr_addr,
|
|
tx_pause_tlv->peermac,
|
|
MLAN_MAC_ADDR_LENGTH)) {
|
|
if (tx_pause_tlv->tx_pause)
|
|
priv->tx_pause = MTRUE;
|
|
else
|
|
priv->tx_pause = MFALSE;
|
|
} else {
|
|
sta_ptr = wlan_get_station_entry(
|
|
priv, tx_pause_tlv->peermac);
|
|
if (sta_ptr) {
|
|
if (sta_ptr->tx_pause !=
|
|
tx_pause_tlv->tx_pause) {
|
|
sta_ptr->tx_pause =
|
|
tx_pause_tlv->tx_pause;
|
|
tx_pkts_queued =
|
|
wlan_update_ralist_tx_pause(
|
|
priv,
|
|
tx_pause_tlv
|
|
->peermac,
|
|
tx_pause_tlv
|
|
->tx_pause);
|
|
}
|
|
}
|
|
}
|
|
if (!tx_pause_tlv->tx_pause)
|
|
total_pkts_queued += tx_pkts_queued;
|
|
PRINTM(MCMND,
|
|
"TxPause: " MACSTR
|
|
" pause=%d, pkts=%d pending=%d total=%d\n",
|
|
MAC2STR(tx_pause_tlv->peermac),
|
|
tx_pause_tlv->tx_pause, tx_pause_tlv->pkt_cnt,
|
|
tx_pkts_queued, total_pkts_queued);
|
|
}
|
|
tlv_buf_left -= (sizeof(MrvlIEtypesHeader_t) + tlv_len);
|
|
tlv = (MrvlIEtypesHeader_t *)((t_u8 *)tlv + tlv_len +
|
|
sizeof(MrvlIEtypesHeader_t));
|
|
}
|
|
|
|
LEAVE();
|
|
return;
|
|
}
|
|
|
|
/**
|
|
* @brief This function prepares command for config uap settings
|
|
*
|
|
* @param pmpriv A pointer to mlan_private structure
|
|
* @param cmd A pointer to HostCmd_DS_COMMAND structure
|
|
* @param cmd_action the action: GET or SET
|
|
* @param pioctl_buf A pointer to mlan_ioctl_req structure
|
|
* @return MLAN_STATUS_SUCCESS or MLAN_STATUS_FAILURE
|
|
*/
|
|
static mlan_status wlan_uap_cmd_ap_config(pmlan_private pmpriv,
|
|
HostCmd_DS_COMMAND *cmd,
|
|
t_u16 cmd_action,
|
|
pmlan_ioctl_req pioctl_buf)
|
|
{
|
|
mlan_ds_bss *bss = MNULL;
|
|
HostCmd_DS_SYS_CONFIG *sys_config =
|
|
(HostCmd_DS_SYS_CONFIG *)&cmd->params.sys_config;
|
|
t_u8 *tlv = MNULL;
|
|
MrvlIEtypes_MacAddr_t *tlv_mac = MNULL;
|
|
MrvlIEtypes_SsIdParamSet_t *tlv_ssid = MNULL;
|
|
MrvlIEtypes_beacon_period_t *tlv_beacon_period = MNULL;
|
|
MrvlIEtypes_dtim_period_t *tlv_dtim_period = MNULL;
|
|
MrvlIEtypes_RatesParamSet_t *tlv_rates = MNULL;
|
|
MrvlIEtypes_tx_rate_t *tlv_txrate = MNULL;
|
|
MrvlIEtypes_mcbc_rate_t *tlv_mcbc_rate = MNULL;
|
|
MrvlIEtypes_tx_power_t *tlv_tx_power = MNULL;
|
|
MrvlIEtypes_bcast_ssid_t *tlv_bcast_ssid = MNULL;
|
|
MrvlIEtypes_antenna_mode_t *tlv_antenna = MNULL;
|
|
MrvlIEtypes_pkt_forward_t *tlv_pkt_forward = MNULL;
|
|
MrvlIEtypes_max_sta_count_t *tlv_sta_count = MNULL;
|
|
MrvlIEtypes_sta_ageout_t *tlv_sta_ageout = MNULL;
|
|
MrvlIEtypes_ps_sta_ageout_t *tlv_ps_sta_ageout = MNULL;
|
|
MrvlIEtypes_rts_threshold_t *tlv_rts_threshold = MNULL;
|
|
MrvlIEtypes_frag_threshold_t *tlv_frag_threshold = MNULL;
|
|
MrvlIEtypes_retry_limit_t *tlv_retry_limit = MNULL;
|
|
MrvlIEtypes_eapol_pwk_hsk_timeout_t *tlv_pairwise_timeout = MNULL;
|
|
MrvlIEtypes_eapol_pwk_hsk_retries_t *tlv_pairwise_retries = MNULL;
|
|
MrvlIEtypes_eapol_gwk_hsk_timeout_t *tlv_groupwise_timeout = MNULL;
|
|
MrvlIEtypes_eapol_gwk_hsk_retries_t *tlv_groupwise_retries = MNULL;
|
|
MrvlIEtypes_mgmt_ie_passthru_t *tlv_mgmt_ie_passthru = MNULL;
|
|
MrvlIEtypes_2040_coex_enable_t *tlv_2040_coex_enable = MNULL;
|
|
MrvlIEtypes_mac_filter_t *tlv_mac_filter = MNULL;
|
|
MrvlIEtypes_channel_band_t *tlv_chan_band = MNULL;
|
|
MrvlIEtypes_ChanListParamSet_t *tlv_chan_list = MNULL;
|
|
ChanScanParamSet_t *pscan_chan = MNULL;
|
|
MrvlIEtypes_auth_type_t *tlv_auth_type = MNULL;
|
|
MrvlIEtypes_encrypt_protocol_t *tlv_encrypt_protocol = MNULL;
|
|
MrvlIEtypes_akmp_t *tlv_akmp = MNULL;
|
|
MrvlIEtypes_pwk_cipher_t *tlv_pwk_cipher = MNULL;
|
|
MrvlIEtypes_gwk_cipher_t *tlv_gwk_cipher = MNULL;
|
|
MrvlIEtypes_rsn_replay_prot_t *tlv_rsn_prot = MNULL;
|
|
MrvlIEtypes_passphrase_t *tlv_passphrase = MNULL;
|
|
MrvlIEtypes_group_rekey_time_t *tlv_rekey_time = MNULL;
|
|
MrvlIEtypes_wep_key_t *tlv_wep_key = MNULL;
|
|
MrvlIETypes_HTCap_t *tlv_htcap = MNULL;
|
|
MrvlIEtypes_wmm_parameter_t *tlv_wmm_parameter = MNULL;
|
|
MrvlIEtypes_preamble_t *tlv_preamble = MNULL;
|
|
|
|
t_u32 cmd_size = 0;
|
|
t_u8 zero_mac[] = {0, 0, 0, 0, 0, 0};
|
|
t_u16 i;
|
|
t_u16 ac;
|
|
#if defined(PCIE9098) || defined(SD9098) || defined(USB9098) || \
|
|
defined(PCIE9097) || defined(SD9097) || defined(USB9097)
|
|
int rx_mcs_supp = 0;
|
|
#endif
|
|
|
|
ENTER();
|
|
if (pioctl_buf == MNULL) {
|
|
LEAVE();
|
|
return MLAN_STATUS_FAILURE;
|
|
}
|
|
|
|
bss = (mlan_ds_bss *)pioctl_buf->pbuf;
|
|
|
|
cmd->command = wlan_cpu_to_le16(HOST_CMD_APCMD_SYS_CONFIGURE);
|
|
sys_config->action = wlan_cpu_to_le16(cmd_action);
|
|
cmd_size = sizeof(HostCmd_DS_SYS_CONFIG) - 1 + S_DS_GEN;
|
|
|
|
tlv = (t_u8 *)sys_config->tlv_buffer;
|
|
if (memcmp(pmpriv->adapter, zero_mac, &bss->param.bss_config.mac_addr,
|
|
MLAN_MAC_ADDR_LENGTH)) {
|
|
tlv_mac = (MrvlIEtypes_MacAddr_t *)tlv;
|
|
tlv_mac->header.type =
|
|
wlan_cpu_to_le16(TLV_TYPE_UAP_MAC_ADDRESS);
|
|
tlv_mac->header.len = wlan_cpu_to_le16(MLAN_MAC_ADDR_LENGTH);
|
|
memcpy_ext(pmpriv->adapter, tlv_mac->mac,
|
|
&bss->param.bss_config.mac_addr,
|
|
MLAN_MAC_ADDR_LENGTH, MLAN_MAC_ADDR_LENGTH);
|
|
cmd_size += sizeof(MrvlIEtypes_MacAddr_t);
|
|
tlv += sizeof(MrvlIEtypes_MacAddr_t);
|
|
}
|
|
|
|
if (bss->param.bss_config.bandcfg.scanMode == SCAN_MODE_ACS) {
|
|
/* ACS is not allowed when DFS repeater mode is on */
|
|
if (pmpriv->adapter->dfs_repeater) {
|
|
PRINTM(MERROR, "ACS is not allowed when"
|
|
"DFS repeater mode is on.\n");
|
|
return MLAN_STATUS_FAILURE;
|
|
}
|
|
}
|
|
|
|
if (bss->param.bss_config.ssid.ssid_len) {
|
|
tlv_ssid = (MrvlIEtypes_SsIdParamSet_t *)tlv;
|
|
tlv_ssid->header.type = wlan_cpu_to_le16(TLV_TYPE_SSID);
|
|
tlv_ssid->header.len = wlan_cpu_to_le16(
|
|
(t_u16)bss->param.bss_config.ssid.ssid_len);
|
|
memcpy_ext(pmpriv->adapter, tlv_ssid->ssid,
|
|
bss->param.bss_config.ssid.ssid,
|
|
bss->param.bss_config.ssid.ssid_len,
|
|
MLAN_MAX_SSID_LENGTH);
|
|
cmd_size += sizeof(MrvlIEtypesHeader_t) +
|
|
bss->param.bss_config.ssid.ssid_len;
|
|
tlv += sizeof(MrvlIEtypesHeader_t) +
|
|
bss->param.bss_config.ssid.ssid_len;
|
|
}
|
|
|
|
if ((bss->param.bss_config.beacon_period >= MIN_BEACON_PERIOD) &&
|
|
(bss->param.bss_config.beacon_period <= MAX_BEACON_PERIOD)) {
|
|
tlv_beacon_period = (MrvlIEtypes_beacon_period_t *)tlv;
|
|
tlv_beacon_period->header.type =
|
|
wlan_cpu_to_le16(TLV_TYPE_UAP_BEACON_PERIOD);
|
|
tlv_beacon_period->header.len = wlan_cpu_to_le16(sizeof(t_u16));
|
|
tlv_beacon_period->beacon_period =
|
|
wlan_cpu_to_le16(bss->param.bss_config.beacon_period);
|
|
cmd_size += sizeof(MrvlIEtypes_beacon_period_t);
|
|
tlv += sizeof(MrvlIEtypes_beacon_period_t);
|
|
}
|
|
|
|
if ((bss->param.bss_config.dtim_period >= MIN_DTIM_PERIOD) &&
|
|
(bss->param.bss_config.dtim_period <= MAX_DTIM_PERIOD)) {
|
|
tlv_dtim_period = (MrvlIEtypes_dtim_period_t *)tlv;
|
|
tlv_dtim_period->header.type =
|
|
wlan_cpu_to_le16(TLV_TYPE_UAP_DTIM_PERIOD);
|
|
tlv_dtim_period->header.len = wlan_cpu_to_le16(sizeof(t_u8));
|
|
tlv_dtim_period->dtim_period =
|
|
bss->param.bss_config.dtim_period;
|
|
cmd_size += sizeof(MrvlIEtypes_dtim_period_t);
|
|
tlv += sizeof(MrvlIEtypes_dtim_period_t);
|
|
}
|
|
|
|
if (bss->param.bss_config.rates[0]) {
|
|
tlv_rates = (MrvlIEtypes_RatesParamSet_t *)tlv;
|
|
tlv_rates->header.type = wlan_cpu_to_le16(TLV_TYPE_RATES);
|
|
for (i = 0;
|
|
i < MAX_DATA_RATES && bss->param.bss_config.rates[i];
|
|
i++) {
|
|
tlv_rates->rates[i] = bss->param.bss_config.rates[i];
|
|
}
|
|
tlv_rates->header.len = wlan_cpu_to_le16(i);
|
|
cmd_size += sizeof(MrvlIEtypesHeader_t) + i;
|
|
tlv += sizeof(MrvlIEtypesHeader_t) + i;
|
|
}
|
|
|
|
if (bss->param.bss_config.tx_data_rate <= DATA_RATE_54M) {
|
|
tlv_txrate = (MrvlIEtypes_tx_rate_t *)tlv;
|
|
tlv_txrate->header.type =
|
|
wlan_cpu_to_le16(TLV_TYPE_UAP_TX_DATA_RATE);
|
|
tlv_txrate->header.len = wlan_cpu_to_le16(sizeof(t_u16));
|
|
tlv_txrate->tx_data_rate =
|
|
wlan_cpu_to_le16(bss->param.bss_config.tx_data_rate);
|
|
cmd_size += sizeof(MrvlIEtypes_tx_rate_t);
|
|
tlv += sizeof(MrvlIEtypes_tx_rate_t);
|
|
}
|
|
|
|
if (bss->param.bss_config.tx_beacon_rate <= DATA_RATE_54M) {
|
|
tlv_txrate = (MrvlIEtypes_tx_rate_t *)tlv;
|
|
tlv_txrate->header.type =
|
|
wlan_cpu_to_le16(TLV_TYPE_UAP_TX_BEACON_RATE);
|
|
tlv_txrate->header.len = wlan_cpu_to_le16(sizeof(t_u16));
|
|
tlv_txrate->tx_data_rate =
|
|
wlan_cpu_to_le16(bss->param.bss_config.tx_beacon_rate);
|
|
cmd_size += sizeof(MrvlIEtypes_tx_rate_t);
|
|
tlv += sizeof(MrvlIEtypes_tx_rate_t);
|
|
}
|
|
|
|
if (bss->param.bss_config.mcbc_data_rate <= DATA_RATE_54M) {
|
|
tlv_mcbc_rate = (MrvlIEtypes_mcbc_rate_t *)tlv;
|
|
tlv_mcbc_rate->header.type =
|
|
wlan_cpu_to_le16(TLV_TYPE_UAP_MCBC_DATA_RATE);
|
|
tlv_mcbc_rate->header.len = wlan_cpu_to_le16(sizeof(t_u16));
|
|
tlv_mcbc_rate->mcbc_data_rate =
|
|
wlan_cpu_to_le16(bss->param.bss_config.mcbc_data_rate);
|
|
cmd_size += sizeof(MrvlIEtypes_mcbc_rate_t);
|
|
tlv += sizeof(MrvlIEtypes_mcbc_rate_t);
|
|
}
|
|
|
|
if (bss->param.bss_config.tx_power_level <= MAX_TX_POWER) {
|
|
tlv_tx_power = (MrvlIEtypes_tx_power_t *)tlv;
|
|
tlv_tx_power->header.type =
|
|
wlan_cpu_to_le16(TLV_TYPE_UAP_TX_POWER);
|
|
tlv_tx_power->header.len = wlan_cpu_to_le16(sizeof(t_u8));
|
|
tlv_tx_power->tx_power = bss->param.bss_config.tx_power_level;
|
|
cmd_size += sizeof(MrvlIEtypes_tx_power_t);
|
|
tlv += sizeof(MrvlIEtypes_tx_power_t);
|
|
}
|
|
|
|
if (bss->param.bss_config.bcast_ssid_ctl <= MAX_BCAST_SSID_CTL) {
|
|
tlv_bcast_ssid = (MrvlIEtypes_bcast_ssid_t *)tlv;
|
|
tlv_bcast_ssid->header.type =
|
|
wlan_cpu_to_le16(TLV_TYPE_UAP_BCAST_SSID_CTL);
|
|
tlv_bcast_ssid->header.len = wlan_cpu_to_le16(sizeof(t_u8));
|
|
tlv_bcast_ssid->bcast_ssid_ctl =
|
|
bss->param.bss_config.bcast_ssid_ctl;
|
|
cmd_size += sizeof(MrvlIEtypes_bcast_ssid_t);
|
|
tlv += sizeof(MrvlIEtypes_bcast_ssid_t);
|
|
}
|
|
|
|
if ((bss->param.bss_config.tx_antenna == ANTENNA_MODE_A) ||
|
|
(bss->param.bss_config.tx_antenna == ANTENNA_MODE_B)) {
|
|
tlv_antenna = (MrvlIEtypes_antenna_mode_t *)tlv;
|
|
tlv_antenna->header.type =
|
|
wlan_cpu_to_le16(TLV_TYPE_UAP_ANTENNA_CTL);
|
|
tlv_antenna->header.len =
|
|
wlan_cpu_to_le16(sizeof(t_u8) + sizeof(t_u8));
|
|
tlv_antenna->which_antenna = TX_ANTENNA;
|
|
tlv_antenna->antenna_mode = bss->param.bss_config.tx_antenna;
|
|
cmd_size += sizeof(MrvlIEtypes_antenna_mode_t);
|
|
tlv += sizeof(MrvlIEtypes_antenna_mode_t);
|
|
}
|
|
|
|
if ((bss->param.bss_config.rx_antenna == ANTENNA_MODE_A) ||
|
|
(bss->param.bss_config.rx_antenna == ANTENNA_MODE_B)) {
|
|
tlv_antenna = (MrvlIEtypes_antenna_mode_t *)tlv;
|
|
tlv_antenna->header.type =
|
|
wlan_cpu_to_le16(TLV_TYPE_UAP_ANTENNA_CTL);
|
|
tlv_antenna->header.len =
|
|
wlan_cpu_to_le16(sizeof(t_u8) + sizeof(t_u8));
|
|
tlv_antenna->which_antenna = RX_ANTENNA;
|
|
tlv_antenna->antenna_mode = bss->param.bss_config.rx_antenna;
|
|
cmd_size += sizeof(MrvlIEtypes_antenna_mode_t);
|
|
tlv += sizeof(MrvlIEtypes_antenna_mode_t);
|
|
}
|
|
|
|
if (bss->param.bss_config.pkt_forward_ctl <= MAX_PKT_FWD_CTRL) {
|
|
tlv_pkt_forward = (MrvlIEtypes_pkt_forward_t *)tlv;
|
|
tlv_pkt_forward->header.type =
|
|
wlan_cpu_to_le16(TLV_TYPE_UAP_PKT_FWD_CTL);
|
|
tlv_pkt_forward->header.len = wlan_cpu_to_le16(sizeof(t_u8));
|
|
tlv_pkt_forward->pkt_forward_ctl =
|
|
bss->param.bss_config.pkt_forward_ctl;
|
|
cmd_size += sizeof(MrvlIEtypes_pkt_forward_t);
|
|
tlv += sizeof(MrvlIEtypes_pkt_forward_t);
|
|
}
|
|
|
|
if (bss->param.bss_config.max_sta_count <= MAX_STA_COUNT) {
|
|
tlv_sta_count = (MrvlIEtypes_max_sta_count_t *)tlv;
|
|
tlv_sta_count->header.type =
|
|
wlan_cpu_to_le16(TLV_TYPE_UAP_MAX_STA_CNT);
|
|
tlv_sta_count->header.len = wlan_cpu_to_le16(sizeof(t_u16));
|
|
tlv_sta_count->max_sta_count =
|
|
wlan_cpu_to_le16(bss->param.bss_config.max_sta_count);
|
|
cmd_size += sizeof(MrvlIEtypes_max_sta_count_t);
|
|
tlv += sizeof(MrvlIEtypes_max_sta_count_t);
|
|
}
|
|
|
|
if (((bss->param.bss_config.sta_ageout_timer >= MIN_STAGE_OUT_TIME) &&
|
|
(bss->param.bss_config.sta_ageout_timer <= MAX_STAGE_OUT_TIME)) ||
|
|
(bss->param.bss_config.sta_ageout_timer == 0)) {
|
|
tlv_sta_ageout = (MrvlIEtypes_sta_ageout_t *)tlv;
|
|
tlv_sta_ageout->header.type =
|
|
wlan_cpu_to_le16(TLV_TYPE_UAP_STA_AGEOUT_TIMER);
|
|
tlv_sta_ageout->header.len = wlan_cpu_to_le16(sizeof(t_u32));
|
|
tlv_sta_ageout->sta_ageout_timer = wlan_cpu_to_le32(
|
|
bss->param.bss_config.sta_ageout_timer);
|
|
cmd_size += sizeof(MrvlIEtypes_sta_ageout_t);
|
|
tlv += sizeof(MrvlIEtypes_sta_ageout_t);
|
|
}
|
|
|
|
if (((bss->param.bss_config.ps_sta_ageout_timer >= MIN_STAGE_OUT_TIME) &&
|
|
(bss->param.bss_config.ps_sta_ageout_timer <=
|
|
MAX_STAGE_OUT_TIME)) ||
|
|
(bss->param.bss_config.ps_sta_ageout_timer == 0)) {
|
|
tlv_ps_sta_ageout = (MrvlIEtypes_ps_sta_ageout_t *)tlv;
|
|
tlv_ps_sta_ageout->header.type =
|
|
wlan_cpu_to_le16(TLV_TYPE_UAP_PS_STA_AGEOUT_TIMER);
|
|
tlv_ps_sta_ageout->header.len = wlan_cpu_to_le16(sizeof(t_u32));
|
|
tlv_ps_sta_ageout->ps_sta_ageout_timer = wlan_cpu_to_le32(
|
|
bss->param.bss_config.ps_sta_ageout_timer);
|
|
cmd_size += sizeof(MrvlIEtypes_ps_sta_ageout_t);
|
|
tlv += sizeof(MrvlIEtypes_ps_sta_ageout_t);
|
|
}
|
|
if (bss->param.bss_config.rts_threshold <= MAX_RTS_THRESHOLD) {
|
|
tlv_rts_threshold = (MrvlIEtypes_rts_threshold_t *)tlv;
|
|
tlv_rts_threshold->header.type =
|
|
wlan_cpu_to_le16(TLV_TYPE_UAP_RTS_THRESHOLD);
|
|
tlv_rts_threshold->header.len = wlan_cpu_to_le16(sizeof(t_u16));
|
|
tlv_rts_threshold->rts_threshold =
|
|
wlan_cpu_to_le16(bss->param.bss_config.rts_threshold);
|
|
cmd_size += sizeof(MrvlIEtypes_rts_threshold_t);
|
|
tlv += sizeof(MrvlIEtypes_rts_threshold_t);
|
|
}
|
|
|
|
if ((bss->param.bss_config.frag_threshold >= MIN_FRAG_THRESHOLD) &&
|
|
(bss->param.bss_config.frag_threshold <= MAX_FRAG_THRESHOLD)) {
|
|
tlv_frag_threshold = (MrvlIEtypes_frag_threshold_t *)tlv;
|
|
tlv_frag_threshold->header.type =
|
|
wlan_cpu_to_le16(TLV_TYPE_UAP_FRAG_THRESHOLD);
|
|
tlv_frag_threshold->header.len =
|
|
wlan_cpu_to_le16(sizeof(t_u16));
|
|
tlv_frag_threshold->frag_threshold =
|
|
wlan_cpu_to_le16(bss->param.bss_config.frag_threshold);
|
|
cmd_size += sizeof(MrvlIEtypes_frag_threshold_t);
|
|
tlv += sizeof(MrvlIEtypes_frag_threshold_t);
|
|
}
|
|
|
|
if (bss->param.bss_config.retry_limit <= MAX_RETRY_LIMIT) {
|
|
tlv_retry_limit = (MrvlIEtypes_retry_limit_t *)tlv;
|
|
tlv_retry_limit->header.type =
|
|
wlan_cpu_to_le16(TLV_TYPE_UAP_RETRY_LIMIT);
|
|
tlv_retry_limit->header.len = wlan_cpu_to_le16(sizeof(t_u8));
|
|
tlv_retry_limit->retry_limit =
|
|
(t_u8)bss->param.bss_config.retry_limit;
|
|
cmd_size += sizeof(MrvlIEtypes_retry_limit_t);
|
|
tlv += sizeof(MrvlIEtypes_retry_limit_t);
|
|
}
|
|
|
|
#ifdef DRV_EMBEDDED_AUTHENTICATOR
|
|
if (IS_FW_SUPPORT_AUTHENTICATOR(pmpriv->adapter)) {
|
|
#endif
|
|
if (bss->param.bss_config.pairwise_update_timeout <
|
|
(MAX_VALID_DWORD)) {
|
|
tlv_pairwise_timeout =
|
|
(MrvlIEtypes_eapol_pwk_hsk_timeout_t *)tlv;
|
|
tlv_pairwise_timeout->header.type = wlan_cpu_to_le16(
|
|
TLV_TYPE_UAP_EAPOL_PWK_HSK_TIMEOUT);
|
|
tlv_pairwise_timeout->header.len =
|
|
wlan_cpu_to_le16(sizeof(t_u32));
|
|
tlv_pairwise_timeout
|
|
->pairwise_update_timeout = wlan_cpu_to_le32(
|
|
bss->param.bss_config.pairwise_update_timeout);
|
|
cmd_size += sizeof(MrvlIEtypes_eapol_pwk_hsk_timeout_t);
|
|
tlv += sizeof(MrvlIEtypes_eapol_pwk_hsk_timeout_t);
|
|
}
|
|
|
|
if (bss->param.bss_config.pwk_retries < (MAX_VALID_DWORD)) {
|
|
tlv_pairwise_retries =
|
|
(MrvlIEtypes_eapol_pwk_hsk_retries_t *)tlv;
|
|
tlv_pairwise_retries->header.type = wlan_cpu_to_le16(
|
|
TLV_TYPE_UAP_EAPOL_PWK_HSK_RETRIES);
|
|
tlv_pairwise_retries->header.len =
|
|
wlan_cpu_to_le16(sizeof(t_u32));
|
|
tlv_pairwise_retries->pwk_retries = wlan_cpu_to_le32(
|
|
bss->param.bss_config.pwk_retries);
|
|
cmd_size += sizeof(MrvlIEtypes_eapol_pwk_hsk_retries_t);
|
|
tlv += sizeof(MrvlIEtypes_eapol_pwk_hsk_retries_t);
|
|
}
|
|
|
|
if (bss->param.bss_config.groupwise_update_timeout <
|
|
(MAX_VALID_DWORD)) {
|
|
tlv_groupwise_timeout =
|
|
(MrvlIEtypes_eapol_gwk_hsk_timeout_t *)tlv;
|
|
tlv_groupwise_timeout->header.type = wlan_cpu_to_le16(
|
|
TLV_TYPE_UAP_EAPOL_GWK_HSK_TIMEOUT);
|
|
tlv_groupwise_timeout->header.len =
|
|
wlan_cpu_to_le16(sizeof(t_u32));
|
|
tlv_groupwise_timeout
|
|
->groupwise_update_timeout = wlan_cpu_to_le32(
|
|
bss->param.bss_config.groupwise_update_timeout);
|
|
cmd_size += sizeof(MrvlIEtypes_eapol_gwk_hsk_timeout_t);
|
|
tlv += sizeof(MrvlIEtypes_eapol_gwk_hsk_timeout_t);
|
|
}
|
|
|
|
if (bss->param.bss_config.gwk_retries < (MAX_VALID_DWORD)) {
|
|
tlv_groupwise_retries =
|
|
(MrvlIEtypes_eapol_gwk_hsk_retries_t *)tlv;
|
|
tlv_groupwise_retries->header.type = wlan_cpu_to_le16(
|
|
TLV_TYPE_UAP_EAPOL_GWK_HSK_RETRIES);
|
|
tlv_groupwise_retries->header.len =
|
|
wlan_cpu_to_le16(sizeof(t_u32));
|
|
tlv_groupwise_retries->gwk_retries = wlan_cpu_to_le32(
|
|
bss->param.bss_config.gwk_retries);
|
|
cmd_size += sizeof(MrvlIEtypes_eapol_gwk_hsk_retries_t);
|
|
tlv += sizeof(MrvlIEtypes_eapol_gwk_hsk_retries_t);
|
|
}
|
|
#ifdef DRV_EMBEDDED_AUTHENTICATOR
|
|
}
|
|
#endif
|
|
if ((bss->param.bss_config.filter.filter_mode <=
|
|
MAC_FILTER_MODE_BLOCK_MAC) &&
|
|
(bss->param.bss_config.filter.mac_count <= MAX_MAC_FILTER_NUM)) {
|
|
tlv_mac_filter = (MrvlIEtypes_mac_filter_t *)tlv;
|
|
tlv_mac_filter->header.type =
|
|
wlan_cpu_to_le16(TLV_TYPE_UAP_STA_MAC_ADDR_FILTER);
|
|
tlv_mac_filter->header.len = wlan_cpu_to_le16(
|
|
2 + MLAN_MAC_ADDR_LENGTH *
|
|
bss->param.bss_config.filter.mac_count);
|
|
tlv_mac_filter->count =
|
|
(t_u8)bss->param.bss_config.filter.mac_count;
|
|
tlv_mac_filter->filter_mode =
|
|
(t_u8)bss->param.bss_config.filter.filter_mode;
|
|
memcpy_ext(pmpriv->adapter, tlv_mac_filter->mac_address,
|
|
(t_u8 *)bss->param.bss_config.filter.mac_list,
|
|
MLAN_MAC_ADDR_LENGTH *
|
|
bss->param.bss_config.filter.mac_count,
|
|
MLAN_MAC_ADDR_LENGTH * MAX_MAC_FILTER_NUM);
|
|
cmd_size += sizeof(MrvlIEtypesHeader_t) + 2 +
|
|
MLAN_MAC_ADDR_LENGTH *
|
|
bss->param.bss_config.filter.mac_count;
|
|
tlv += sizeof(MrvlIEtypesHeader_t) + 2 +
|
|
MLAN_MAC_ADDR_LENGTH *
|
|
bss->param.bss_config.filter.mac_count;
|
|
}
|
|
|
|
if (((bss->param.bss_config.bandcfg.scanMode == SCAN_MODE_MANUAL) &&
|
|
(bss->param.bss_config.channel > 0) &&
|
|
(bss->param.bss_config.channel <= MLAN_MAX_CHANNEL)) ||
|
|
(bss->param.bss_config.bandcfg.scanMode == SCAN_MODE_ACS)) {
|
|
tlv_chan_band = (MrvlIEtypes_channel_band_t *)tlv;
|
|
tlv_chan_band->header.type =
|
|
wlan_cpu_to_le16(TLV_TYPE_UAP_CHAN_BAND_CONFIG);
|
|
tlv_chan_band->header.len =
|
|
wlan_cpu_to_le16(sizeof(t_u8) + sizeof(t_u8));
|
|
tlv_chan_band->bandcfg = bss->param.bss_config.bandcfg;
|
|
tlv_chan_band->channel = bss->param.bss_config.channel;
|
|
cmd_size += sizeof(MrvlIEtypes_channel_band_t);
|
|
tlv += sizeof(MrvlIEtypes_channel_band_t);
|
|
}
|
|
|
|
if ((bss->param.bss_config.num_of_chan) &&
|
|
(bss->param.bss_config.num_of_chan <= MLAN_MAX_CHANNEL)) {
|
|
tlv_chan_list = (MrvlIEtypes_ChanListParamSet_t *)tlv;
|
|
tlv_chan_list->header.type =
|
|
wlan_cpu_to_le16(TLV_TYPE_CHANLIST);
|
|
tlv_chan_list->header.len = wlan_cpu_to_le16(
|
|
(t_u16)(sizeof(ChanScanParamSet_t) *
|
|
bss->param.bss_config.num_of_chan));
|
|
pscan_chan = tlv_chan_list->chan_scan_param;
|
|
for (i = 0; i < bss->param.bss_config.num_of_chan; i++) {
|
|
pscan_chan->chan_number =
|
|
bss->param.bss_config.chan_list[i].chan_number;
|
|
pscan_chan->bandcfg =
|
|
bss->param.bss_config.chan_list[i].bandcfg;
|
|
pscan_chan++;
|
|
}
|
|
cmd_size += sizeof(tlv_chan_list->header) +
|
|
(sizeof(ChanScanParamSet_t) *
|
|
bss->param.bss_config.num_of_chan);
|
|
tlv += sizeof(tlv_chan_list->header) +
|
|
(sizeof(ChanScanParamSet_t) *
|
|
bss->param.bss_config.num_of_chan);
|
|
}
|
|
|
|
if ((bss->param.bss_config.auth_mode <= MLAN_AUTH_MODE_SHARED) ||
|
|
(bss->param.bss_config.auth_mode == MLAN_AUTH_MODE_AUTO)) {
|
|
tlv_auth_type = (MrvlIEtypes_auth_type_t *)tlv;
|
|
tlv_auth_type->header.type =
|
|
wlan_cpu_to_le16(TLV_TYPE_AUTH_TYPE);
|
|
tlv_auth_type->header.len = wlan_cpu_to_le16(sizeof(t_u8));
|
|
tlv_auth_type->auth_type =
|
|
(t_u8)bss->param.bss_config.auth_mode;
|
|
cmd_size += sizeof(MrvlIEtypes_auth_type_t);
|
|
tlv += sizeof(MrvlIEtypes_auth_type_t);
|
|
}
|
|
|
|
if (bss->param.bss_config.protocol) {
|
|
tlv_encrypt_protocol = (MrvlIEtypes_encrypt_protocol_t *)tlv;
|
|
tlv_encrypt_protocol->header.type =
|
|
wlan_cpu_to_le16(TLV_TYPE_UAP_ENCRYPT_PROTOCOL);
|
|
tlv_encrypt_protocol->header.len =
|
|
wlan_cpu_to_le16(sizeof(t_u16));
|
|
tlv_encrypt_protocol->protocol =
|
|
wlan_cpu_to_le16(bss->param.bss_config.protocol);
|
|
cmd_size += sizeof(MrvlIEtypes_encrypt_protocol_t);
|
|
tlv += sizeof(MrvlIEtypes_encrypt_protocol_t);
|
|
}
|
|
|
|
if ((bss->param.bss_config.protocol & PROTOCOL_WPA) ||
|
|
(bss->param.bss_config.protocol & PROTOCOL_WPA2) ||
|
|
(bss->param.bss_config.protocol & PROTOCOL_EAP)) {
|
|
tlv_akmp = (MrvlIEtypes_akmp_t *)tlv;
|
|
tlv_akmp->header.type = wlan_cpu_to_le16(TLV_TYPE_UAP_AKMP);
|
|
tlv_akmp->key_mgmt =
|
|
wlan_cpu_to_le16(bss->param.bss_config.key_mgmt);
|
|
tlv_akmp->header.len = sizeof(t_u16);
|
|
tlv_akmp->key_mgmt_operation = wlan_cpu_to_le16(
|
|
bss->param.bss_config.key_mgmt_operation);
|
|
tlv_akmp->header.len += sizeof(t_u16);
|
|
tlv_akmp->header.len = wlan_cpu_to_le16(tlv_akmp->header.len);
|
|
cmd_size += sizeof(MrvlIEtypes_akmp_t);
|
|
tlv += sizeof(MrvlIEtypes_akmp_t);
|
|
|
|
if (bss->param.bss_config.wpa_cfg.pairwise_cipher_wpa &
|
|
VALID_CIPHER_BITMAP) {
|
|
tlv_pwk_cipher = (MrvlIEtypes_pwk_cipher_t *)tlv;
|
|
tlv_pwk_cipher->header.type =
|
|
wlan_cpu_to_le16(TLV_TYPE_PWK_CIPHER);
|
|
tlv_pwk_cipher->header.len = wlan_cpu_to_le16(
|
|
sizeof(t_u16) + sizeof(t_u8) + sizeof(t_u8));
|
|
tlv_pwk_cipher->protocol =
|
|
wlan_cpu_to_le16(PROTOCOL_WPA);
|
|
tlv_pwk_cipher->pairwise_cipher =
|
|
bss->param.bss_config.wpa_cfg
|
|
.pairwise_cipher_wpa;
|
|
cmd_size += sizeof(MrvlIEtypes_pwk_cipher_t);
|
|
tlv += sizeof(MrvlIEtypes_pwk_cipher_t);
|
|
}
|
|
|
|
if (bss->param.bss_config.wpa_cfg.pairwise_cipher_wpa2 &
|
|
VALID_CIPHER_BITMAP) {
|
|
tlv_pwk_cipher = (MrvlIEtypes_pwk_cipher_t *)tlv;
|
|
tlv_pwk_cipher->header.type =
|
|
wlan_cpu_to_le16(TLV_TYPE_PWK_CIPHER);
|
|
tlv_pwk_cipher->header.len = wlan_cpu_to_le16(
|
|
sizeof(t_u16) + sizeof(t_u8) + sizeof(t_u8));
|
|
tlv_pwk_cipher->protocol =
|
|
wlan_cpu_to_le16(PROTOCOL_WPA2);
|
|
tlv_pwk_cipher->pairwise_cipher =
|
|
bss->param.bss_config.wpa_cfg
|
|
.pairwise_cipher_wpa2;
|
|
cmd_size += sizeof(MrvlIEtypes_pwk_cipher_t);
|
|
tlv += sizeof(MrvlIEtypes_pwk_cipher_t);
|
|
}
|
|
|
|
if (bss->param.bss_config.wpa_cfg.group_cipher &
|
|
VALID_CIPHER_BITMAP) {
|
|
tlv_gwk_cipher = (MrvlIEtypes_gwk_cipher_t *)tlv;
|
|
tlv_gwk_cipher->header.type =
|
|
wlan_cpu_to_le16(TLV_TYPE_GWK_CIPHER);
|
|
tlv_gwk_cipher->header.len =
|
|
wlan_cpu_to_le16(sizeof(t_u8) + sizeof(t_u8));
|
|
tlv_gwk_cipher->group_cipher =
|
|
bss->param.bss_config.wpa_cfg.group_cipher;
|
|
cmd_size += sizeof(MrvlIEtypes_gwk_cipher_t);
|
|
tlv += sizeof(MrvlIEtypes_gwk_cipher_t);
|
|
}
|
|
|
|
if (bss->param.bss_config.wpa_cfg.rsn_protection <= MTRUE) {
|
|
tlv_rsn_prot = (MrvlIEtypes_rsn_replay_prot_t *)tlv;
|
|
tlv_rsn_prot->header.type = wlan_cpu_to_le16(
|
|
TLV_TYPE_UAP_RSN_REPLAY_PROTECT);
|
|
tlv_rsn_prot->header.len =
|
|
wlan_cpu_to_le16(sizeof(t_u8));
|
|
tlv_rsn_prot->rsn_replay_prot =
|
|
bss->param.bss_config.wpa_cfg.rsn_protection;
|
|
cmd_size += sizeof(MrvlIEtypes_rsn_replay_prot_t);
|
|
tlv += sizeof(MrvlIEtypes_rsn_replay_prot_t);
|
|
}
|
|
|
|
#ifdef DRV_EMBEDDED_AUTHENTICATOR
|
|
if (IS_FW_SUPPORT_AUTHENTICATOR(pmpriv->adapter)) {
|
|
#endif
|
|
if (bss->param.bss_config.wpa_cfg.length) {
|
|
tlv_passphrase =
|
|
(MrvlIEtypes_passphrase_t *)tlv;
|
|
tlv_passphrase->header.type = wlan_cpu_to_le16(
|
|
TLV_TYPE_UAP_WPA_PASSPHRASE);
|
|
tlv_passphrase->header
|
|
.len = (t_u16)wlan_cpu_to_le16(
|
|
bss->param.bss_config.wpa_cfg.length);
|
|
memcpy_ext(
|
|
pmpriv->adapter,
|
|
tlv_passphrase->passphrase,
|
|
bss->param.bss_config.wpa_cfg.passphrase,
|
|
bss->param.bss_config.wpa_cfg.length,
|
|
bss->param.bss_config.wpa_cfg.length);
|
|
cmd_size +=
|
|
sizeof(MrvlIEtypesHeader_t) +
|
|
bss->param.bss_config.wpa_cfg.length;
|
|
tlv += sizeof(MrvlIEtypesHeader_t) +
|
|
bss->param.bss_config.wpa_cfg.length;
|
|
}
|
|
|
|
if (bss->param.bss_config.wpa_cfg.gk_rekey_time <
|
|
MAX_GRP_TIMER) {
|
|
tlv_rekey_time =
|
|
(MrvlIEtypes_group_rekey_time_t *)tlv;
|
|
tlv_rekey_time->header.type = wlan_cpu_to_le16(
|
|
TLV_TYPE_UAP_GRP_REKEY_TIME);
|
|
tlv_rekey_time->header.len =
|
|
wlan_cpu_to_le16(sizeof(t_u32));
|
|
tlv_rekey_time->gk_rekey_time =
|
|
wlan_cpu_to_le32(
|
|
bss->param.bss_config.wpa_cfg
|
|
.gk_rekey_time);
|
|
cmd_size +=
|
|
sizeof(MrvlIEtypes_group_rekey_time_t);
|
|
tlv += sizeof(MrvlIEtypes_group_rekey_time_t);
|
|
}
|
|
#ifdef DRV_EMBEDDED_AUTHENTICATOR
|
|
}
|
|
#endif
|
|
} else {
|
|
if ((bss->param.bss_config.wep_cfg.key0.length) &&
|
|
((bss->param.bss_config.wep_cfg.key0.length == 5) ||
|
|
(bss->param.bss_config.wep_cfg.key0.length == 10) ||
|
|
(bss->param.bss_config.wep_cfg.key0.length == 13) ||
|
|
(bss->param.bss_config.wep_cfg.key0.length == 26))) {
|
|
tlv_wep_key = (MrvlIEtypes_wep_key_t *)tlv;
|
|
tlv_wep_key->header.type =
|
|
wlan_cpu_to_le16(TLV_TYPE_UAP_WEP_KEY);
|
|
tlv_wep_key->header.len = wlan_cpu_to_le16(
|
|
2 + bss->param.bss_config.wep_cfg.key0.length);
|
|
tlv_wep_key->key_index =
|
|
bss->param.bss_config.wep_cfg.key0.key_index;
|
|
tlv_wep_key->is_default =
|
|
bss->param.bss_config.wep_cfg.key0.is_default;
|
|
memcpy_ext(pmpriv->adapter, tlv_wep_key->key,
|
|
bss->param.bss_config.wep_cfg.key0.key,
|
|
bss->param.bss_config.wep_cfg.key0.length,
|
|
bss->param.bss_config.wep_cfg.key0.length);
|
|
cmd_size += sizeof(MrvlIEtypesHeader_t) + 2 +
|
|
bss->param.bss_config.wep_cfg.key0.length;
|
|
tlv += sizeof(MrvlIEtypesHeader_t) + 2 +
|
|
bss->param.bss_config.wep_cfg.key0.length;
|
|
}
|
|
|
|
if ((bss->param.bss_config.wep_cfg.key1.length) &&
|
|
((bss->param.bss_config.wep_cfg.key1.length == 5) ||
|
|
(bss->param.bss_config.wep_cfg.key1.length == 10) ||
|
|
(bss->param.bss_config.wep_cfg.key1.length == 13) ||
|
|
(bss->param.bss_config.wep_cfg.key1.length == 26))) {
|
|
tlv_wep_key = (MrvlIEtypes_wep_key_t *)tlv;
|
|
tlv_wep_key->header.type =
|
|
wlan_cpu_to_le16(TLV_TYPE_UAP_WEP_KEY);
|
|
tlv_wep_key->header.len = wlan_cpu_to_le16(
|
|
2 + bss->param.bss_config.wep_cfg.key1.length);
|
|
tlv_wep_key->key_index =
|
|
bss->param.bss_config.wep_cfg.key1.key_index;
|
|
tlv_wep_key->is_default =
|
|
bss->param.bss_config.wep_cfg.key1.is_default;
|
|
memcpy_ext(pmpriv->adapter, tlv_wep_key->key,
|
|
bss->param.bss_config.wep_cfg.key1.key,
|
|
bss->param.bss_config.wep_cfg.key1.length,
|
|
bss->param.bss_config.wep_cfg.key1.length);
|
|
cmd_size += sizeof(MrvlIEtypesHeader_t) + 2 +
|
|
bss->param.bss_config.wep_cfg.key1.length;
|
|
tlv += sizeof(MrvlIEtypesHeader_t) + 2 +
|
|
bss->param.bss_config.wep_cfg.key1.length;
|
|
}
|
|
|
|
if ((bss->param.bss_config.wep_cfg.key2.length) &&
|
|
((bss->param.bss_config.wep_cfg.key2.length == 5) ||
|
|
(bss->param.bss_config.wep_cfg.key2.length == 10) ||
|
|
(bss->param.bss_config.wep_cfg.key2.length == 13) ||
|
|
(bss->param.bss_config.wep_cfg.key2.length == 26))) {
|
|
tlv_wep_key = (MrvlIEtypes_wep_key_t *)tlv;
|
|
tlv_wep_key->header.type =
|
|
wlan_cpu_to_le16(TLV_TYPE_UAP_WEP_KEY);
|
|
tlv_wep_key->header.len = wlan_cpu_to_le16(
|
|
2 + bss->param.bss_config.wep_cfg.key2.length);
|
|
tlv_wep_key->key_index =
|
|
bss->param.bss_config.wep_cfg.key2.key_index;
|
|
tlv_wep_key->is_default =
|
|
bss->param.bss_config.wep_cfg.key2.is_default;
|
|
memcpy_ext(pmpriv->adapter, tlv_wep_key->key,
|
|
bss->param.bss_config.wep_cfg.key2.key,
|
|
bss->param.bss_config.wep_cfg.key2.length,
|
|
bss->param.bss_config.wep_cfg.key2.length);
|
|
cmd_size += sizeof(MrvlIEtypesHeader_t) + 2 +
|
|
bss->param.bss_config.wep_cfg.key2.length;
|
|
tlv += sizeof(MrvlIEtypesHeader_t) + 2 +
|
|
bss->param.bss_config.wep_cfg.key2.length;
|
|
}
|
|
|
|
if ((bss->param.bss_config.wep_cfg.key3.length) &&
|
|
((bss->param.bss_config.wep_cfg.key3.length == 5) ||
|
|
(bss->param.bss_config.wep_cfg.key3.length == 10) ||
|
|
(bss->param.bss_config.wep_cfg.key3.length == 13) ||
|
|
(bss->param.bss_config.wep_cfg.key3.length == 26))) {
|
|
tlv_wep_key = (MrvlIEtypes_wep_key_t *)tlv;
|
|
tlv_wep_key->header.type =
|
|
wlan_cpu_to_le16(TLV_TYPE_UAP_WEP_KEY);
|
|
tlv_wep_key->header.len = wlan_cpu_to_le16(
|
|
2 + bss->param.bss_config.wep_cfg.key3.length);
|
|
tlv_wep_key->key_index =
|
|
bss->param.bss_config.wep_cfg.key3.key_index;
|
|
tlv_wep_key->is_default =
|
|
bss->param.bss_config.wep_cfg.key3.is_default;
|
|
memcpy_ext(pmpriv->adapter, tlv_wep_key->key,
|
|
bss->param.bss_config.wep_cfg.key3.key,
|
|
bss->param.bss_config.wep_cfg.key3.length,
|
|
bss->param.bss_config.wep_cfg.key3.length);
|
|
cmd_size += sizeof(MrvlIEtypesHeader_t) + 2 +
|
|
bss->param.bss_config.wep_cfg.key3.length;
|
|
tlv += sizeof(MrvlIEtypesHeader_t) + 2 +
|
|
bss->param.bss_config.wep_cfg.key3.length;
|
|
}
|
|
}
|
|
if (bss->param.bss_config.ht_cap_info) {
|
|
tlv_htcap = (MrvlIETypes_HTCap_t *)tlv;
|
|
tlv_htcap->header.type = wlan_cpu_to_le16(HT_CAPABILITY);
|
|
tlv_htcap->header.len = wlan_cpu_to_le16(sizeof(HTCap_t));
|
|
tlv_htcap->ht_cap.ht_cap_info =
|
|
wlan_cpu_to_le16(bss->param.bss_config.ht_cap_info);
|
|
tlv_htcap->ht_cap.ampdu_param =
|
|
bss->param.bss_config.ampdu_param;
|
|
memcpy_ext(pmpriv->adapter, tlv_htcap->ht_cap.supported_mcs_set,
|
|
bss->param.bss_config.supported_mcs_set, 16,
|
|
sizeof(tlv_htcap->ht_cap.supported_mcs_set));
|
|
#if defined(PCIE9098) || defined(SD9098) || defined(USB9098) || \
|
|
defined(PCIE9097) || defined(SD9097) || defined(USB9097)
|
|
if (IS_CARD9098(pmpriv->adapter->card_type) ||
|
|
IS_CARD9097(pmpriv->adapter->card_type)) {
|
|
if (bss->param.bss_config.supported_mcs_set[0]) {
|
|
if (bss->param.bss_config.bandcfg.chanBand ==
|
|
BAND_5GHZ)
|
|
rx_mcs_supp = GET_RXMCSSUPP(
|
|
pmpriv->adapter->user_htstream >>
|
|
8);
|
|
else
|
|
rx_mcs_supp = GET_RXMCSSUPP(
|
|
pmpriv->adapter->user_htstream);
|
|
|
|
if (rx_mcs_supp == 0x1) {
|
|
tlv_htcap->ht_cap.supported_mcs_set[0] =
|
|
0xFF;
|
|
tlv_htcap->ht_cap.supported_mcs_set[1] =
|
|
0;
|
|
} else if (rx_mcs_supp == 0x2) {
|
|
tlv_htcap->ht_cap.supported_mcs_set[0] =
|
|
0xFF;
|
|
tlv_htcap->ht_cap.supported_mcs_set[1] =
|
|
0xFF;
|
|
}
|
|
}
|
|
}
|
|
#endif
|
|
tlv_htcap->ht_cap.ht_ext_cap =
|
|
wlan_cpu_to_le16(bss->param.bss_config.ht_ext_cap);
|
|
tlv_htcap->ht_cap.tx_bf_cap =
|
|
wlan_cpu_to_le32(bss->param.bss_config.tx_bf_cap);
|
|
tlv_htcap->ht_cap.asel = bss->param.bss_config.asel;
|
|
cmd_size += sizeof(MrvlIETypes_HTCap_t);
|
|
tlv += sizeof(MrvlIETypes_HTCap_t);
|
|
}
|
|
if (bss->param.bss_config.mgmt_ie_passthru_mask < (MAX_VALID_DWORD)) {
|
|
tlv_mgmt_ie_passthru = (MrvlIEtypes_mgmt_ie_passthru_t *)tlv;
|
|
tlv_mgmt_ie_passthru->header.type =
|
|
wlan_cpu_to_le16(TLV_TYPE_UAP_MGMT_IE_PASSTHRU_MASK);
|
|
tlv_mgmt_ie_passthru->header.len =
|
|
wlan_cpu_to_le16(sizeof(t_u32));
|
|
/* keep copy in private data */
|
|
pmpriv->mgmt_frame_passthru_mask =
|
|
bss->param.bss_config.mgmt_ie_passthru_mask;
|
|
tlv_mgmt_ie_passthru->mgmt_ie_mask = wlan_cpu_to_le32(
|
|
bss->param.bss_config.mgmt_ie_passthru_mask);
|
|
cmd_size += sizeof(MrvlIEtypes_mgmt_ie_passthru_t);
|
|
tlv += sizeof(MrvlIEtypes_mgmt_ie_passthru_t);
|
|
}
|
|
if ((bss->param.bss_config.enable_2040coex == 0) ||
|
|
(bss->param.bss_config.enable_2040coex == 1)) {
|
|
tlv_2040_coex_enable = (MrvlIEtypes_2040_coex_enable_t *)tlv;
|
|
tlv_2040_coex_enable->header.type =
|
|
wlan_cpu_to_le16(TLV_TYPE_2040_BSS_COEX_CONTROL);
|
|
tlv_2040_coex_enable->header.len =
|
|
wlan_cpu_to_le16(sizeof(t_u8));
|
|
tlv_2040_coex_enable->enable_2040coex =
|
|
bss->param.bss_config.enable_2040coex;
|
|
cmd_size += sizeof(MrvlIEtypes_2040_coex_enable_t);
|
|
tlv += sizeof(MrvlIEtypes_2040_coex_enable_t);
|
|
}
|
|
if ((bss->param.bss_config.uap_host_based_config == MTRUE) ||
|
|
(bss->param.bss_config.wmm_para.qos_info & 0x80 ||
|
|
bss->param.bss_config.wmm_para.qos_info == 0x00)) {
|
|
tlv_wmm_parameter = (MrvlIEtypes_wmm_parameter_t *)tlv;
|
|
tlv_wmm_parameter->header.type =
|
|
wlan_cpu_to_le16(TLV_TYPE_VENDOR_SPECIFIC_IE);
|
|
tlv_wmm_parameter->header.len = wlan_cpu_to_le16(
|
|
sizeof(bss->param.bss_config.wmm_para));
|
|
memcpy_ext(pmpriv->adapter, tlv_wmm_parameter->wmm_para.ouitype,
|
|
bss->param.bss_config.wmm_para.ouitype,
|
|
sizeof(tlv_wmm_parameter->wmm_para.ouitype),
|
|
sizeof(tlv_wmm_parameter->wmm_para.ouitype));
|
|
tlv_wmm_parameter->wmm_para.ouisubtype =
|
|
bss->param.bss_config.wmm_para.ouisubtype;
|
|
tlv_wmm_parameter->wmm_para.version =
|
|
bss->param.bss_config.wmm_para.version;
|
|
tlv_wmm_parameter->wmm_para.qos_info =
|
|
bss->param.bss_config.wmm_para.qos_info;
|
|
for (ac = 0; ac < 4; ac++) {
|
|
tlv_wmm_parameter->wmm_para.ac_params[ac]
|
|
.aci_aifsn.aifsn =
|
|
bss->param.bss_config.wmm_para.ac_params[ac]
|
|
.aci_aifsn.aifsn;
|
|
tlv_wmm_parameter->wmm_para.ac_params[ac].aci_aifsn.aci =
|
|
bss->param.bss_config.wmm_para.ac_params[ac]
|
|
.aci_aifsn.aci;
|
|
tlv_wmm_parameter->wmm_para.ac_params[ac].ecw.ecw_max =
|
|
bss->param.bss_config.wmm_para.ac_params[ac]
|
|
.ecw.ecw_max;
|
|
tlv_wmm_parameter->wmm_para.ac_params[ac].ecw.ecw_min =
|
|
bss->param.bss_config.wmm_para.ac_params[ac]
|
|
.ecw.ecw_min;
|
|
tlv_wmm_parameter->wmm_para.ac_params[ac].tx_op_limit =
|
|
wlan_cpu_to_le16(bss->param.bss_config.wmm_para
|
|
.ac_params[ac]
|
|
.tx_op_limit);
|
|
}
|
|
cmd_size += sizeof(MrvlIEtypes_wmm_parameter_t);
|
|
tlv += sizeof(MrvlIEtypes_wmm_parameter_t);
|
|
}
|
|
#ifdef DRV_EMBEDDED_AUTHENTICATOR
|
|
if (!IS_FW_SUPPORT_AUTHENTICATOR(pmpriv->adapter))
|
|
AuthenticatorBssConfig(pmpriv->psapriv,
|
|
(t_u8 *)&bss->param.bss_config, 0, 0, 0);
|
|
#endif
|
|
if (pmpriv->adapter->pcard_info->v17_fw_api &&
|
|
bss->param.bss_config.preamble_type) {
|
|
tlv_preamble = (MrvlIEtypes_preamble_t *)tlv;
|
|
tlv_preamble->header.type =
|
|
wlan_cpu_to_le16(TLV_TYPE_UAP_PREAMBLE_CTL);
|
|
tlv_preamble->header.len =
|
|
wlan_cpu_to_le16(sizeof(MrvlIEtypes_preamble_t) -
|
|
sizeof(MrvlIEtypesHeader_t));
|
|
tlv_preamble->preamble_type =
|
|
wlan_cpu_to_le16(bss->param.bss_config.preamble_type);
|
|
|
|
cmd_size += sizeof(MrvlIEtypes_preamble_t);
|
|
tlv += sizeof(MrvlIEtypes_preamble_t);
|
|
}
|
|
cmd->size = (t_u16)wlan_cpu_to_le16(cmd_size);
|
|
PRINTM(MCMND, "AP config: cmd_size=%d\n", cmd_size);
|
|
LEAVE();
|
|
return MLAN_STATUS_SUCCESS;
|
|
}
|
|
|
|
/**
|
|
* @brief This function prepares command of sys_config
|
|
*
|
|
* @param pmpriv A pointer to mlan_private structure
|
|
* @param cmd A pointer to HostCmd_DS_COMMAND structure
|
|
* @param cmd_action the action: GET or SET
|
|
* @param pioctl_buf A pointer to mlan_ioctl_req structure
|
|
* @return MLAN_STATUS_SUCCESS or MLAN_STATUS_FAILURE
|
|
*/
|
|
static mlan_status wlan_uap_cmd_sys_configure(pmlan_private pmpriv,
|
|
HostCmd_DS_COMMAND *cmd,
|
|
t_u16 cmd_action,
|
|
pmlan_ioctl_req pioctl_buf,
|
|
t_void *pdata_buf)
|
|
{
|
|
mlan_ds_bss *bss = MNULL;
|
|
HostCmd_DS_SYS_CONFIG *sys_config =
|
|
(HostCmd_DS_SYS_CONFIG *)&cmd->params.sys_config;
|
|
MrvlIEtypes_MacAddr_t *mac_tlv = MNULL;
|
|
MrvlIEtypes_channel_band_t *pdat_tlv_cb = MNULL;
|
|
MrvlIEtypes_beacon_period_t *bcn_pd_tlv = MNULL,
|
|
*pdat_tlv_bcnpd = MNULL;
|
|
MrvlIEtypes_dtim_period_t *dtim_pd_tlv = MNULL,
|
|
*pdat_tlv_dtimpd = MNULL;
|
|
MrvlIEtypes_wmm_parameter_t *tlv_wmm_parameter = MNULL;
|
|
MrvlIEtypes_ChanListParamSet_t *tlv_chan_list = MNULL;
|
|
ChanScanParamSet_t *pscan_chan = MNULL;
|
|
MrvlIEtypes_channel_band_t *chan_band_tlv = MNULL;
|
|
MrvlIEtypes_chan_bw_oper_t *poper_class_tlv = MNULL;
|
|
t_u8 length = 0;
|
|
t_u8 curr_oper_class = 1;
|
|
t_u8 *oper_class_ie = (t_u8 *)sys_config->tlv_buffer;
|
|
t_u16 i = 0;
|
|
t_u8 ac = 0;
|
|
mlan_ds_misc_custom_ie *cust_ie = MNULL;
|
|
mlan_ds_misc_cfg *misc = MNULL;
|
|
MrvlIEtypesHeader_t *ie_header =
|
|
(MrvlIEtypesHeader_t *)sys_config->tlv_buffer;
|
|
MrvlIEtypesHeader_t *pdata_header = (MrvlIEtypesHeader_t *)pdata_buf;
|
|
t_u8 *ie = (t_u8 *)sys_config->tlv_buffer + sizeof(MrvlIEtypesHeader_t);
|
|
t_u16 req_len = 0, travel_len = 0;
|
|
custom_ie *cptr = MNULL;
|
|
mlan_status ret = MLAN_STATUS_SUCCESS;
|
|
MrvlIEtypes_wacp_mode_t *tlv_wacp_mode = MNULL;
|
|
|
|
ENTER();
|
|
|
|
cmd->command = wlan_cpu_to_le16(HOST_CMD_APCMD_SYS_CONFIGURE);
|
|
sys_config->action = wlan_cpu_to_le16(cmd_action);
|
|
cmd->size =
|
|
wlan_cpu_to_le16(sizeof(HostCmd_DS_SYS_CONFIG) - 1 + S_DS_GEN);
|
|
if (pioctl_buf == MNULL) {
|
|
if (pdata_buf) {
|
|
switch (pdata_header->type) {
|
|
case TLV_TYPE_UAP_CHAN_BAND_CONFIG:
|
|
pdat_tlv_cb =
|
|
(MrvlIEtypes_channel_band_t *)pdata_buf;
|
|
chan_band_tlv = (MrvlIEtypes_channel_band_t *)
|
|
sys_config->tlv_buffer;
|
|
cmd->size = wlan_cpu_to_le16(
|
|
sizeof(HostCmd_DS_SYS_CONFIG) - 1 +
|
|
S_DS_GEN +
|
|
sizeof(MrvlIEtypes_channel_band_t));
|
|
chan_band_tlv->header.type = wlan_cpu_to_le16(
|
|
TLV_TYPE_UAP_CHAN_BAND_CONFIG);
|
|
chan_band_tlv->header.len = wlan_cpu_to_le16(
|
|
sizeof(MrvlIEtypes_channel_band_t) -
|
|
sizeof(MrvlIEtypesHeader_t));
|
|
if (cmd_action) {
|
|
chan_band_tlv->bandcfg =
|
|
pdat_tlv_cb->bandcfg;
|
|
chan_band_tlv->channel =
|
|
pdat_tlv_cb->channel;
|
|
}
|
|
ret = MLAN_STATUS_SUCCESS;
|
|
break;
|
|
case TLV_TYPE_UAP_BEACON_PERIOD:
|
|
pdat_tlv_bcnpd = (MrvlIEtypes_beacon_period_t *)
|
|
pdata_buf;
|
|
bcn_pd_tlv = (MrvlIEtypes_beacon_period_t *)
|
|
sys_config->tlv_buffer;
|
|
cmd->size = sizeof(HostCmd_DS_SYS_CONFIG) - 1 +
|
|
S_DS_GEN +
|
|
sizeof(MrvlIEtypes_beacon_period_t);
|
|
bcn_pd_tlv->header.type = wlan_cpu_to_le16(
|
|
TLV_TYPE_UAP_BEACON_PERIOD);
|
|
bcn_pd_tlv->header.len = wlan_cpu_to_le16(
|
|
sizeof(MrvlIEtypes_beacon_period_t) -
|
|
sizeof(MrvlIEtypesHeader_t));
|
|
if (cmd_action) {
|
|
bcn_pd_tlv->beacon_period =
|
|
wlan_cpu_to_le16(
|
|
pdat_tlv_bcnpd
|
|
->beacon_period);
|
|
}
|
|
/* Add TLV_UAP_DTIM_PERIOD if it follws in
|
|
* pdata_buf */
|
|
pdat_tlv_dtimpd =
|
|
(MrvlIEtypes_dtim_period_t
|
|
*)(((t_u8 *)pdata_buf) +
|
|
sizeof(MrvlIEtypes_beacon_period_t));
|
|
if (TLV_TYPE_UAP_DTIM_PERIOD ==
|
|
pdat_tlv_dtimpd->header.type) {
|
|
dtim_pd_tlv =
|
|
(MrvlIEtypes_dtim_period_t
|
|
*)(sys_config
|
|
->tlv_buffer +
|
|
sizeof(MrvlIEtypes_beacon_period_t));
|
|
cmd->size += sizeof(
|
|
MrvlIEtypes_dtim_period_t);
|
|
dtim_pd_tlv->header
|
|
.type = wlan_cpu_to_le16(
|
|
TLV_TYPE_UAP_DTIM_PERIOD);
|
|
dtim_pd_tlv->header
|
|
.len = wlan_cpu_to_le16(
|
|
sizeof(MrvlIEtypes_dtim_period_t) -
|
|
sizeof(MrvlIEtypesHeader_t));
|
|
if (cmd_action) {
|
|
dtim_pd_tlv->dtim_period =
|
|
pdat_tlv_dtimpd
|
|
->dtim_period;
|
|
}
|
|
}
|
|
/* Finalize cmd size */
|
|
cmd->size = wlan_cpu_to_le16(cmd->size);
|
|
ret = MLAN_STATUS_SUCCESS;
|
|
break;
|
|
case TLV_TYPE_MGMT_IE:
|
|
cust_ie = (mlan_ds_misc_custom_ie *)pdata_buf;
|
|
cmd->size = wlan_cpu_to_le16(
|
|
sizeof(HostCmd_DS_SYS_CONFIG) - 1 +
|
|
S_DS_GEN + sizeof(MrvlIEtypesHeader_t) +
|
|
cust_ie->len);
|
|
ie_header->type =
|
|
wlan_cpu_to_le16(TLV_TYPE_MGMT_IE);
|
|
ie_header->len = wlan_cpu_to_le16(cust_ie->len);
|
|
|
|
if (ie) {
|
|
req_len = cust_ie->len;
|
|
travel_len = 0;
|
|
/* conversion for index, mask, len */
|
|
if (req_len == sizeof(t_u16))
|
|
cust_ie->ie_data_list[0]
|
|
.ie_index = wlan_cpu_to_le16(
|
|
cust_ie->ie_data_list[0]
|
|
.ie_index);
|
|
while (req_len > sizeof(t_u16)) {
|
|
cptr = (custom_ie
|
|
*)(((t_u8 *)&cust_ie
|
|
->ie_data_list) +
|
|
travel_len);
|
|
travel_len +=
|
|
cptr->ie_length +
|
|
sizeof(custom_ie) -
|
|
MAX_IE_SIZE;
|
|
req_len -= cptr->ie_length +
|
|
sizeof(custom_ie) -
|
|
MAX_IE_SIZE;
|
|
cptr->ie_index =
|
|
wlan_cpu_to_le16(
|
|
cptr->ie_index);
|
|
cptr->mgmt_subtype_mask = wlan_cpu_to_le16(
|
|
cptr->mgmt_subtype_mask);
|
|
cptr->ie_length =
|
|
wlan_cpu_to_le16(
|
|
cptr->ie_length);
|
|
}
|
|
memcpy_ext(pmpriv->adapter, ie,
|
|
cust_ie->ie_data_list,
|
|
cust_ie->len, cust_ie->len);
|
|
}
|
|
break;
|
|
case REGULATORY_CLASS:
|
|
poper_class_tlv =
|
|
(MrvlIEtypes_chan_bw_oper_t *)pdata_buf;
|
|
ret = wlan_get_curr_oper_class(
|
|
pmpriv,
|
|
poper_class_tlv->ds_chan_bw_oper.channel,
|
|
poper_class_tlv->ds_chan_bw_oper
|
|
.bandwidth,
|
|
&curr_oper_class);
|
|
if (ret != MLAN_STATUS_SUCCESS) {
|
|
PRINTM(MERROR,
|
|
"Can not get current oper class! bandwidth = %d, channel = %d\n",
|
|
poper_class_tlv->ds_chan_bw_oper
|
|
.bandwidth,
|
|
poper_class_tlv->ds_chan_bw_oper
|
|
.channel);
|
|
}
|
|
|
|
if (cmd_action == HostCmd_ACT_GEN_SET)
|
|
length =
|
|
wlan_add_supported_oper_class_ie(
|
|
pmpriv, &oper_class_ie,
|
|
curr_oper_class);
|
|
cmd->size = wlan_cpu_to_le16(
|
|
sizeof(HostCmd_DS_SYS_CONFIG) - 1 +
|
|
S_DS_GEN + length);
|
|
break;
|
|
case TLV_TYPE_UAP_MAX_STA_CNT_PER_CHIP:
|
|
memcpy_ext(
|
|
pmpriv->adapter, sys_config->tlv_buffer,
|
|
pdata_buf,
|
|
sizeof(MrvlIEtypes_uap_max_sta_cnt_t),
|
|
sizeof(MrvlIEtypes_uap_max_sta_cnt_t));
|
|
cmd->size = wlan_cpu_to_le16(
|
|
sizeof(HostCmd_DS_SYS_CONFIG) - 1 +
|
|
S_DS_GEN +
|
|
sizeof(MrvlIEtypes_uap_max_sta_cnt_t));
|
|
break;
|
|
default:
|
|
PRINTM(MERROR,
|
|
"Wrong data, or missing TLV_TYPE 0x%04x handler.\n",
|
|
*(t_u16 *)pdata_buf);
|
|
break;
|
|
}
|
|
goto done;
|
|
} else {
|
|
mac_tlv =
|
|
(MrvlIEtypes_MacAddr_t *)sys_config->tlv_buffer;
|
|
cmd->size = wlan_cpu_to_le16(
|
|
sizeof(HostCmd_DS_SYS_CONFIG) - 1 + S_DS_GEN +
|
|
sizeof(MrvlIEtypes_MacAddr_t));
|
|
mac_tlv->header.type =
|
|
wlan_cpu_to_le16(TLV_TYPE_UAP_MAC_ADDRESS);
|
|
mac_tlv->header.len =
|
|
wlan_cpu_to_le16(MLAN_MAC_ADDR_LENGTH);
|
|
ret = MLAN_STATUS_SUCCESS;
|
|
goto done;
|
|
}
|
|
}
|
|
if (pioctl_buf->req_id == MLAN_IOCTL_BSS) {
|
|
bss = (mlan_ds_bss *)pioctl_buf->pbuf;
|
|
if (bss->sub_command == MLAN_OID_BSS_MAC_ADDR) {
|
|
mac_tlv =
|
|
(MrvlIEtypes_MacAddr_t *)sys_config->tlv_buffer;
|
|
cmd->size = wlan_cpu_to_le16(
|
|
sizeof(HostCmd_DS_SYS_CONFIG) - 1 + S_DS_GEN +
|
|
sizeof(MrvlIEtypes_MacAddr_t));
|
|
mac_tlv->header.type =
|
|
wlan_cpu_to_le16(TLV_TYPE_UAP_MAC_ADDRESS);
|
|
mac_tlv->header.len =
|
|
wlan_cpu_to_le16(MLAN_MAC_ADDR_LENGTH);
|
|
if (cmd_action == HostCmd_ACT_GEN_SET)
|
|
memcpy_ext(pmpriv->adapter, mac_tlv->mac,
|
|
&bss->param.mac_addr,
|
|
MLAN_MAC_ADDR_LENGTH,
|
|
MLAN_MAC_ADDR_LENGTH);
|
|
} else if (bss->sub_command == MLAN_OID_UAP_CFG_WMM_PARAM) {
|
|
tlv_wmm_parameter = (MrvlIEtypes_wmm_parameter_t *)
|
|
sys_config->tlv_buffer;
|
|
cmd->size = wlan_cpu_to_le16(
|
|
sizeof(HostCmd_DS_SYS_CONFIG) - 1 + S_DS_GEN +
|
|
sizeof(MrvlIEtypes_wmm_parameter_t));
|
|
tlv_wmm_parameter->header.type =
|
|
wlan_cpu_to_le16(TLV_TYPE_AP_WMM_PARAM);
|
|
tlv_wmm_parameter->header.len = wlan_cpu_to_le16(
|
|
sizeof(bss->param.ap_wmm_para));
|
|
if (cmd_action == HostCmd_ACT_GEN_SET) {
|
|
for (ac = 0; ac < 4; ac++) {
|
|
tlv_wmm_parameter->wmm_para
|
|
.ac_params[ac]
|
|
.aci_aifsn.aifsn =
|
|
bss->param.ap_wmm_para
|
|
.ac_params[ac]
|
|
.aci_aifsn.aifsn;
|
|
tlv_wmm_parameter->wmm_para
|
|
.ac_params[ac]
|
|
.aci_aifsn.aci =
|
|
bss->param.ap_wmm_para
|
|
.ac_params[ac]
|
|
.aci_aifsn.aci;
|
|
tlv_wmm_parameter->wmm_para
|
|
.ac_params[ac]
|
|
.ecw.ecw_max =
|
|
bss->param.ap_wmm_para
|
|
.ac_params[ac]
|
|
.ecw.ecw_max;
|
|
tlv_wmm_parameter->wmm_para
|
|
.ac_params[ac]
|
|
.ecw.ecw_min =
|
|
bss->param.ap_wmm_para
|
|
.ac_params[ac]
|
|
.ecw.ecw_min;
|
|
tlv_wmm_parameter->wmm_para
|
|
.ac_params[ac]
|
|
.tx_op_limit = wlan_cpu_to_le16(
|
|
bss->param.ap_wmm_para
|
|
.ac_params[ac]
|
|
.tx_op_limit);
|
|
}
|
|
}
|
|
} else if (bss->sub_command == MLAN_OID_UAP_SCAN_CHANNELS) {
|
|
tlv_chan_list = (MrvlIEtypes_ChanListParamSet_t *)
|
|
sys_config->tlv_buffer;
|
|
tlv_chan_list->header.type =
|
|
wlan_cpu_to_le16(TLV_TYPE_CHANLIST);
|
|
if (bss->param.ap_scan_channels.num_of_chan &&
|
|
bss->param.ap_scan_channels.num_of_chan <=
|
|
MLAN_MAX_CHANNEL) {
|
|
cmd->size = wlan_cpu_to_le16(
|
|
sizeof(HostCmd_DS_SYS_CONFIG) - 1 +
|
|
S_DS_GEN +
|
|
sizeof(tlv_chan_list->header) +
|
|
sizeof(ChanScanParamSet_t) *
|
|
bss->param.ap_scan_channels
|
|
.num_of_chan);
|
|
tlv_chan_list->header.len = wlan_cpu_to_le16(
|
|
(t_u16)(sizeof(ChanScanParamSet_t) *
|
|
bss->param.ap_scan_channels
|
|
.num_of_chan));
|
|
pscan_chan = tlv_chan_list->chan_scan_param;
|
|
for (i = 0;
|
|
i <
|
|
bss->param.ap_scan_channels.num_of_chan;
|
|
i++) {
|
|
pscan_chan->chan_number =
|
|
bss->param.ap_scan_channels
|
|
.chan_list[i]
|
|
.chan_number;
|
|
pscan_chan->bandcfg =
|
|
bss->param.ap_scan_channels
|
|
.chan_list[i]
|
|
.bandcfg;
|
|
pscan_chan++;
|
|
}
|
|
PRINTM(MCMND,
|
|
"Set AP scan channel list = %d\n",
|
|
bss->param.ap_scan_channels.num_of_chan);
|
|
} else {
|
|
tlv_chan_list->header.len = wlan_cpu_to_le16(
|
|
(t_u16)(sizeof(ChanScanParamSet_t) *
|
|
MLAN_MAX_CHANNEL));
|
|
cmd->size = wlan_cpu_to_le16(
|
|
sizeof(HostCmd_DS_SYS_CONFIG) - 1 +
|
|
S_DS_GEN +
|
|
sizeof(MrvlIEtypes_ChanListParamSet_t) +
|
|
sizeof(ChanScanParamSet_t) *
|
|
MLAN_MAX_CHANNEL);
|
|
}
|
|
} else if (bss->sub_command == MLAN_OID_UAP_CHANNEL) {
|
|
chan_band_tlv = (MrvlIEtypes_channel_band_t *)
|
|
sys_config->tlv_buffer;
|
|
cmd->size = wlan_cpu_to_le16(
|
|
sizeof(HostCmd_DS_SYS_CONFIG) - 1 + S_DS_GEN +
|
|
sizeof(MrvlIEtypes_channel_band_t));
|
|
chan_band_tlv->header.type =
|
|
wlan_cpu_to_le16(TLV_TYPE_UAP_CHAN_BAND_CONFIG);
|
|
chan_band_tlv->header.len = wlan_cpu_to_le16(
|
|
sizeof(MrvlIEtypes_channel_band_t) -
|
|
sizeof(MrvlIEtypesHeader_t));
|
|
if (cmd_action == HostCmd_ACT_GEN_SET) {
|
|
chan_band_tlv->bandcfg =
|
|
bss->param.ap_channel.bandcfg;
|
|
chan_band_tlv->channel =
|
|
bss->param.ap_channel.channel;
|
|
PRINTM(MCMND,
|
|
"Set AP channel, band=%d, channel=%d\n",
|
|
bss->param.ap_channel.bandcfg,
|
|
bss->param.ap_channel.channel);
|
|
}
|
|
} else if ((bss->sub_command == MLAN_OID_UAP_BSS_CONFIG) &&
|
|
(cmd_action == HostCmd_ACT_GEN_SET)) {
|
|
ret = wlan_uap_cmd_ap_config(pmpriv, cmd, cmd_action,
|
|
pioctl_buf);
|
|
goto done;
|
|
}
|
|
} else if (pioctl_buf->req_id == MLAN_IOCTL_MISC_CFG) {
|
|
misc = (mlan_ds_misc_cfg *)pioctl_buf->pbuf;
|
|
if ((misc->sub_command == MLAN_OID_MISC_GEN_IE) &&
|
|
(misc->param.gen_ie.type == MLAN_IE_TYPE_GEN_IE)) {
|
|
cmd->size = wlan_cpu_to_le16(
|
|
sizeof(HostCmd_DS_SYS_CONFIG) - 1 + S_DS_GEN +
|
|
sizeof(MrvlIEtypesHeader_t) +
|
|
misc->param.gen_ie.len);
|
|
ie_header->type = wlan_cpu_to_le16(TLV_TYPE_WAPI_IE);
|
|
ie_header->len =
|
|
wlan_cpu_to_le16(misc->param.gen_ie.len);
|
|
if (cmd_action == HostCmd_ACT_GEN_SET)
|
|
memcpy_ext(pmpriv->adapter, ie,
|
|
misc->param.gen_ie.ie_data,
|
|
misc->param.gen_ie.len,
|
|
misc->param.gen_ie.len);
|
|
}
|
|
if ((misc->sub_command == MLAN_OID_MISC_CUSTOM_IE) &&
|
|
(misc->param.cust_ie.type == TLV_TYPE_MGMT_IE)) {
|
|
cmd->size = wlan_cpu_to_le16(
|
|
sizeof(HostCmd_DS_SYS_CONFIG) - 1 + S_DS_GEN +
|
|
sizeof(MrvlIEtypesHeader_t) +
|
|
misc->param.cust_ie.len);
|
|
ie_header->type = wlan_cpu_to_le16(TLV_TYPE_MGMT_IE);
|
|
ie_header->len =
|
|
wlan_cpu_to_le16(misc->param.cust_ie.len);
|
|
|
|
if (ie) {
|
|
req_len = misc->param.cust_ie.len;
|
|
travel_len = 0;
|
|
/* conversion for index, mask, len */
|
|
if (req_len == sizeof(t_u16))
|
|
misc->param.cust_ie.ie_data_list[0]
|
|
.ie_index = wlan_cpu_to_le16(
|
|
misc->param.cust_ie
|
|
.ie_data_list[0]
|
|
.ie_index);
|
|
while (req_len > sizeof(t_u16)) {
|
|
cptr = (custom_ie
|
|
*)(((t_u8 *)&misc->param
|
|
.cust_ie
|
|
.ie_data_list) +
|
|
travel_len);
|
|
travel_len += cptr->ie_length +
|
|
sizeof(custom_ie) -
|
|
MAX_IE_SIZE;
|
|
req_len -= cptr->ie_length +
|
|
sizeof(custom_ie) -
|
|
MAX_IE_SIZE;
|
|
cptr->ie_index = wlan_cpu_to_le16(
|
|
cptr->ie_index);
|
|
cptr->mgmt_subtype_mask =
|
|
wlan_cpu_to_le16(
|
|
cptr->mgmt_subtype_mask);
|
|
cptr->ie_length = wlan_cpu_to_le16(
|
|
cptr->ie_length);
|
|
}
|
|
if (misc->param.cust_ie.len)
|
|
memcpy_ext(
|
|
pmpriv->adapter, ie,
|
|
misc->param.cust_ie.ie_data_list,
|
|
misc->param.cust_ie.len,
|
|
misc->param.cust_ie.len);
|
|
}
|
|
}
|
|
if (misc->sub_command == MLAN_OID_MISC_WACP_MODE) {
|
|
tlv_wacp_mode = (MrvlIEtypes_wacp_mode_t *)
|
|
sys_config->tlv_buffer;
|
|
tlv_wacp_mode->header.type =
|
|
wlan_cpu_to_le16(TLV_TYPE_UAP_WACP_MODE);
|
|
tlv_wacp_mode->header.len =
|
|
wlan_cpu_to_le16(sizeof(t_u8));
|
|
if (cmd_action == HostCmd_ACT_GEN_SET) {
|
|
tlv_wacp_mode->wacp_mode =
|
|
misc->param.wacp_mode;
|
|
}
|
|
cmd->size = wlan_cpu_to_le16(
|
|
sizeof(HostCmd_DS_SYS_CONFIG) - 1 + S_DS_GEN +
|
|
sizeof(MrvlIEtypes_wacp_mode_t));
|
|
}
|
|
}
|
|
done:
|
|
LEAVE();
|
|
return ret;
|
|
}
|
|
|
|
/**
|
|
* @brief This function handles command resp for get uap settings
|
|
*
|
|
* @param pmpriv A pointer to mlan_private structure
|
|
* @param resp A pointer to HostCmd_DS_COMMAND
|
|
* @param pioctl_buf A pointer to mlan_ioctl_req structure
|
|
*
|
|
* @return MLAN_STATUS_SUCCESS
|
|
*/
|
|
static mlan_status wlan_uap_ret_cmd_ap_config(pmlan_private pmpriv,
|
|
HostCmd_DS_COMMAND *resp,
|
|
mlan_ioctl_req *pioctl_buf)
|
|
{
|
|
HostCmd_DS_SYS_CONFIG *sys_config =
|
|
(HostCmd_DS_SYS_CONFIG *)&resp->params.sys_config;
|
|
mlan_ds_bss *bss = MNULL;
|
|
MrvlIEtypesHeader_t *tlv = MNULL;
|
|
t_u16 tlv_buf_left = 0;
|
|
t_u16 tlv_type = 0;
|
|
t_u16 tlv_len = 0;
|
|
MrvlIEtypes_MacAddr_t *tlv_mac = MNULL;
|
|
MrvlIEtypes_SsIdParamSet_t *tlv_ssid = MNULL;
|
|
MrvlIEtypes_beacon_period_t *tlv_beacon_period = MNULL;
|
|
MrvlIEtypes_dtim_period_t *tlv_dtim_period = MNULL;
|
|
MrvlIEtypes_RatesParamSet_t *tlv_rates = MNULL;
|
|
MrvlIEtypes_tx_rate_t *tlv_txrate = MNULL;
|
|
MrvlIEtypes_mcbc_rate_t *tlv_mcbc_rate = MNULL;
|
|
MrvlIEtypes_tx_power_t *tlv_tx_power = MNULL;
|
|
MrvlIEtypes_bcast_ssid_t *tlv_bcast_ssid = MNULL;
|
|
MrvlIEtypes_antenna_mode_t *tlv_antenna = MNULL;
|
|
MrvlIEtypes_pkt_forward_t *tlv_pkt_forward = MNULL;
|
|
MrvlIEtypes_max_sta_count_t *tlv_sta_count = MNULL;
|
|
MrvlIEtypes_sta_ageout_t *tlv_sta_ageout = MNULL;
|
|
MrvlIEtypes_ps_sta_ageout_t *tlv_ps_sta_ageout = MNULL;
|
|
MrvlIEtypes_rts_threshold_t *tlv_rts_threshold = MNULL;
|
|
MrvlIEtypes_frag_threshold_t *tlv_frag_threshold = MNULL;
|
|
MrvlIEtypes_retry_limit_t *tlv_retry_limit = MNULL;
|
|
MrvlIEtypes_eapol_pwk_hsk_timeout_t *tlv_pairwise_timeout = MNULL;
|
|
MrvlIEtypes_eapol_pwk_hsk_retries_t *tlv_pairwise_retries = MNULL;
|
|
MrvlIEtypes_eapol_gwk_hsk_timeout_t *tlv_groupwise_timeout = MNULL;
|
|
MrvlIEtypes_eapol_gwk_hsk_retries_t *tlv_groupwise_retries = MNULL;
|
|
MrvlIEtypes_mgmt_ie_passthru_t *tlv_mgmt_ie_passthru = MNULL;
|
|
MrvlIEtypes_2040_coex_enable_t *tlv_2040_coex_enable = MNULL;
|
|
MrvlIEtypes_mac_filter_t *tlv_mac_filter = MNULL;
|
|
MrvlIEtypes_channel_band_t *tlv_chan_band = MNULL;
|
|
MrvlIEtypes_ChanListParamSet_t *tlv_chan_list = MNULL;
|
|
ChanScanParamSet_t *pscan_chan = MNULL;
|
|
MrvlIEtypes_auth_type_t *tlv_auth_type = MNULL;
|
|
MrvlIEtypes_encrypt_protocol_t *tlv_encrypt_protocol = MNULL;
|
|
MrvlIEtypes_akmp_t *tlv_akmp = MNULL;
|
|
MrvlIEtypes_pwk_cipher_t *tlv_pwk_cipher = MNULL;
|
|
MrvlIEtypes_gwk_cipher_t *tlv_gwk_cipher = MNULL;
|
|
MrvlIEtypes_rsn_replay_prot_t *tlv_rsn_prot = MNULL;
|
|
MrvlIEtypes_passphrase_t *tlv_passphrase = MNULL;
|
|
#ifdef WIFI_DIRECT_SUPPORT
|
|
MrvlIEtypes_psk_t *tlv_psk = MNULL;
|
|
#endif /* WIFI_DIRECT_SUPPORT */
|
|
MrvlIEtypes_group_rekey_time_t *tlv_rekey_time = MNULL;
|
|
MrvlIEtypes_wep_key_t *tlv_wep_key = MNULL;
|
|
MrvlIEtypes_preamble_t *tlv_preamble = MNULL;
|
|
MrvlIEtypes_bss_status_t *tlv_bss_status = MNULL;
|
|
MrvlIETypes_HTCap_t *tlv_htcap = MNULL;
|
|
MrvlIEtypes_wmm_parameter_t *tlv_wmm_parameter = MNULL;
|
|
|
|
wep_key *pkey = MNULL;
|
|
t_u16 i;
|
|
t_u16 ac;
|
|
|
|
ENTER();
|
|
|
|
bss = (mlan_ds_bss *)pioctl_buf->pbuf;
|
|
tlv = (MrvlIEtypesHeader_t *)sys_config->tlv_buffer;
|
|
tlv_buf_left =
|
|
resp->size - (sizeof(HostCmd_DS_SYS_CONFIG) - 1 + S_DS_GEN);
|
|
|
|
while (tlv_buf_left >= sizeof(MrvlIEtypesHeader_t)) {
|
|
tlv_type = wlan_le16_to_cpu(tlv->type);
|
|
tlv_len = wlan_le16_to_cpu(tlv->len);
|
|
|
|
if (tlv_buf_left < (tlv_len + sizeof(MrvlIEtypesHeader_t))) {
|
|
PRINTM(MERROR,
|
|
"Error processing uAP sys config TLVs, bytes left < TLV length\n");
|
|
break;
|
|
}
|
|
|
|
switch (tlv_type) {
|
|
case TLV_TYPE_UAP_MAC_ADDRESS:
|
|
tlv_mac = (MrvlIEtypes_MacAddr_t *)tlv;
|
|
memcpy_ext(pmpriv->adapter,
|
|
&bss->param.bss_config.mac_addr,
|
|
tlv_mac->mac, MLAN_MAC_ADDR_LENGTH,
|
|
MLAN_MAC_ADDR_LENGTH);
|
|
break;
|
|
case TLV_TYPE_SSID:
|
|
tlv_ssid = (MrvlIEtypes_SsIdParamSet_t *)tlv;
|
|
bss->param.bss_config.ssid.ssid_len =
|
|
MIN(MLAN_MAX_SSID_LENGTH, tlv_len);
|
|
memcpy_ext(pmpriv->adapter,
|
|
bss->param.bss_config.ssid.ssid,
|
|
tlv_ssid->ssid, tlv_len,
|
|
MLAN_MAX_SSID_LENGTH);
|
|
break;
|
|
case TLV_TYPE_UAP_BEACON_PERIOD:
|
|
tlv_beacon_period = (MrvlIEtypes_beacon_period_t *)tlv;
|
|
bss->param.bss_config.beacon_period = wlan_le16_to_cpu(
|
|
tlv_beacon_period->beacon_period);
|
|
pmpriv->uap_state_chan_cb.beacon_period =
|
|
wlan_le16_to_cpu(
|
|
tlv_beacon_period->beacon_period);
|
|
break;
|
|
case TLV_TYPE_UAP_DTIM_PERIOD:
|
|
tlv_dtim_period = (MrvlIEtypes_dtim_period_t *)tlv;
|
|
bss->param.bss_config.dtim_period =
|
|
tlv_dtim_period->dtim_period;
|
|
pmpriv->uap_state_chan_cb.dtim_period =
|
|
tlv_dtim_period->dtim_period;
|
|
break;
|
|
case TLV_TYPE_RATES:
|
|
tlv_rates = (MrvlIEtypes_RatesParamSet_t *)tlv;
|
|
memcpy_ext(pmpriv->adapter, bss->param.bss_config.rates,
|
|
tlv_rates->rates, tlv_len, MAX_DATA_RATES);
|
|
break;
|
|
case TLV_TYPE_UAP_TX_DATA_RATE:
|
|
tlv_txrate = (MrvlIEtypes_tx_rate_t *)tlv;
|
|
bss->param.bss_config.tx_data_rate =
|
|
wlan_le16_to_cpu(tlv_txrate->tx_data_rate);
|
|
break;
|
|
case TLV_TYPE_UAP_TX_BEACON_RATE:
|
|
tlv_txrate = (MrvlIEtypes_tx_rate_t *)tlv;
|
|
bss->param.bss_config.tx_beacon_rate =
|
|
wlan_le16_to_cpu(tlv_txrate->tx_data_rate);
|
|
break;
|
|
case TLV_TYPE_UAP_MCBC_DATA_RATE:
|
|
tlv_mcbc_rate = (MrvlIEtypes_mcbc_rate_t *)tlv;
|
|
bss->param.bss_config.mcbc_data_rate =
|
|
wlan_le16_to_cpu(tlv_mcbc_rate->mcbc_data_rate);
|
|
break;
|
|
case TLV_TYPE_UAP_TX_POWER:
|
|
tlv_tx_power = (MrvlIEtypes_tx_power_t *)tlv;
|
|
bss->param.bss_config.tx_power_level =
|
|
tlv_tx_power->tx_power;
|
|
break;
|
|
case TLV_TYPE_UAP_BCAST_SSID_CTL:
|
|
tlv_bcast_ssid = (MrvlIEtypes_bcast_ssid_t *)tlv;
|
|
bss->param.bss_config.bcast_ssid_ctl =
|
|
tlv_bcast_ssid->bcast_ssid_ctl;
|
|
break;
|
|
case TLV_TYPE_UAP_ANTENNA_CTL:
|
|
tlv_antenna = (MrvlIEtypes_antenna_mode_t *)tlv;
|
|
if (tlv_antenna->which_antenna == TX_ANTENNA)
|
|
bss->param.bss_config.tx_antenna =
|
|
tlv_antenna->antenna_mode;
|
|
else if (tlv_antenna->which_antenna == RX_ANTENNA)
|
|
bss->param.bss_config.rx_antenna =
|
|
tlv_antenna->antenna_mode;
|
|
break;
|
|
case TLV_TYPE_UAP_PKT_FWD_CTL:
|
|
tlv_pkt_forward = (MrvlIEtypes_pkt_forward_t *)tlv;
|
|
bss->param.bss_config.pkt_forward_ctl =
|
|
tlv_pkt_forward->pkt_forward_ctl;
|
|
break;
|
|
case TLV_TYPE_UAP_MAX_STA_CNT:
|
|
tlv_sta_count = (MrvlIEtypes_max_sta_count_t *)tlv;
|
|
bss->param.bss_config.max_sta_count =
|
|
wlan_le16_to_cpu(tlv_sta_count->max_sta_count);
|
|
break;
|
|
case TLV_TYPE_UAP_STA_AGEOUT_TIMER:
|
|
tlv_sta_ageout = (MrvlIEtypes_sta_ageout_t *)tlv;
|
|
bss->param.bss_config.sta_ageout_timer =
|
|
wlan_le32_to_cpu(
|
|
tlv_sta_ageout->sta_ageout_timer);
|
|
break;
|
|
case TLV_TYPE_UAP_PS_STA_AGEOUT_TIMER:
|
|
tlv_ps_sta_ageout = (MrvlIEtypes_ps_sta_ageout_t *)tlv;
|
|
bss->param.bss_config.ps_sta_ageout_timer =
|
|
wlan_le32_to_cpu(
|
|
tlv_ps_sta_ageout->ps_sta_ageout_timer);
|
|
break;
|
|
case TLV_TYPE_UAP_RTS_THRESHOLD:
|
|
tlv_rts_threshold = (MrvlIEtypes_rts_threshold_t *)tlv;
|
|
bss->param.bss_config.rts_threshold = wlan_le16_to_cpu(
|
|
tlv_rts_threshold->rts_threshold);
|
|
break;
|
|
case TLV_TYPE_UAP_FRAG_THRESHOLD:
|
|
tlv_frag_threshold =
|
|
(MrvlIEtypes_frag_threshold_t *)tlv;
|
|
bss->param.bss_config.frag_threshold = wlan_le16_to_cpu(
|
|
tlv_frag_threshold->frag_threshold);
|
|
break;
|
|
case TLV_TYPE_UAP_RETRY_LIMIT:
|
|
tlv_retry_limit = (MrvlIEtypes_retry_limit_t *)tlv;
|
|
bss->param.bss_config.retry_limit =
|
|
tlv_retry_limit->retry_limit;
|
|
break;
|
|
case TLV_TYPE_UAP_EAPOL_PWK_HSK_TIMEOUT:
|
|
tlv_pairwise_timeout =
|
|
(MrvlIEtypes_eapol_pwk_hsk_timeout_t *)tlv;
|
|
bss->param.bss_config
|
|
.pairwise_update_timeout = wlan_le32_to_cpu(
|
|
tlv_pairwise_timeout->pairwise_update_timeout);
|
|
break;
|
|
case TLV_TYPE_UAP_EAPOL_PWK_HSK_RETRIES:
|
|
tlv_pairwise_retries =
|
|
(MrvlIEtypes_eapol_pwk_hsk_retries_t *)tlv;
|
|
bss->param.bss_config.pwk_retries = wlan_le32_to_cpu(
|
|
tlv_pairwise_retries->pwk_retries);
|
|
break;
|
|
case TLV_TYPE_UAP_EAPOL_GWK_HSK_TIMEOUT:
|
|
tlv_groupwise_timeout =
|
|
(MrvlIEtypes_eapol_gwk_hsk_timeout_t *)tlv;
|
|
bss->param.bss_config.groupwise_update_timeout =
|
|
wlan_le32_to_cpu(
|
|
tlv_groupwise_timeout
|
|
->groupwise_update_timeout);
|
|
break;
|
|
case TLV_TYPE_UAP_EAPOL_GWK_HSK_RETRIES:
|
|
tlv_groupwise_retries =
|
|
(MrvlIEtypes_eapol_gwk_hsk_retries_t *)tlv;
|
|
bss->param.bss_config.gwk_retries = wlan_le32_to_cpu(
|
|
tlv_groupwise_retries->gwk_retries);
|
|
break;
|
|
case TLV_TYPE_UAP_MGMT_IE_PASSTHRU_MASK:
|
|
tlv_mgmt_ie_passthru =
|
|
(MrvlIEtypes_mgmt_ie_passthru_t *)tlv;
|
|
bss->param.bss_config.mgmt_ie_passthru_mask =
|
|
wlan_le32_to_cpu(
|
|
tlv_mgmt_ie_passthru->mgmt_ie_mask);
|
|
break;
|
|
case TLV_TYPE_2040_BSS_COEX_CONTROL:
|
|
tlv_2040_coex_enable =
|
|
(MrvlIEtypes_2040_coex_enable_t *)tlv;
|
|
bss->param.bss_config.enable_2040coex =
|
|
tlv_2040_coex_enable->enable_2040coex;
|
|
break;
|
|
case TLV_TYPE_UAP_STA_MAC_ADDR_FILTER:
|
|
tlv_mac_filter = (MrvlIEtypes_mac_filter_t *)tlv;
|
|
bss->param.bss_config.filter.mac_count =
|
|
MIN(MAX_MAC_FILTER_NUM, tlv_mac_filter->count);
|
|
bss->param.bss_config.filter.filter_mode =
|
|
tlv_mac_filter->filter_mode;
|
|
memcpy_ext(
|
|
pmpriv->adapter,
|
|
(t_u8 *)bss->param.bss_config.filter.mac_list,
|
|
tlv_mac_filter->mac_address,
|
|
MLAN_MAC_ADDR_LENGTH *
|
|
bss->param.bss_config.filter.mac_count,
|
|
sizeof(bss->param.bss_config.filter.mac_list));
|
|
break;
|
|
case TLV_TYPE_UAP_CHAN_BAND_CONFIG:
|
|
tlv_chan_band = (MrvlIEtypes_channel_band_t *)tlv;
|
|
bss->param.bss_config.bandcfg = tlv_chan_band->bandcfg;
|
|
bss->param.bss_config.channel = tlv_chan_band->channel;
|
|
pmpriv->uap_state_chan_cb.bandcfg =
|
|
tlv_chan_band->bandcfg;
|
|
pmpriv->uap_state_chan_cb.channel =
|
|
tlv_chan_band->channel;
|
|
break;
|
|
case TLV_TYPE_CHANLIST:
|
|
tlv_chan_list = (MrvlIEtypes_ChanListParamSet_t *)tlv;
|
|
bss->param.bss_config.num_of_chan =
|
|
tlv_len / sizeof(ChanScanParamSet_t);
|
|
pscan_chan = tlv_chan_list->chan_scan_param;
|
|
for (i = 0; i < bss->param.bss_config.num_of_chan;
|
|
i++) {
|
|
bss->param.bss_config.chan_list[i].chan_number =
|
|
pscan_chan->chan_number;
|
|
bss->param.bss_config.chan_list[i].bandcfg =
|
|
pscan_chan->bandcfg;
|
|
pscan_chan++;
|
|
}
|
|
break;
|
|
case TLV_TYPE_AUTH_TYPE:
|
|
tlv_auth_type = (MrvlIEtypes_auth_type_t *)tlv;
|
|
bss->param.bss_config.auth_mode =
|
|
tlv_auth_type->auth_type;
|
|
break;
|
|
case TLV_TYPE_UAP_ENCRYPT_PROTOCOL:
|
|
tlv_encrypt_protocol =
|
|
(MrvlIEtypes_encrypt_protocol_t *)tlv;
|
|
bss->param.bss_config.protocol = wlan_le16_to_cpu(
|
|
tlv_encrypt_protocol->protocol);
|
|
break;
|
|
case TLV_TYPE_UAP_AKMP:
|
|
tlv_akmp = (MrvlIEtypes_akmp_t *)tlv;
|
|
bss->param.bss_config.key_mgmt =
|
|
wlan_le16_to_cpu(tlv_akmp->key_mgmt);
|
|
if (tlv_len > sizeof(t_u16))
|
|
bss->param.bss_config.key_mgmt_operation =
|
|
wlan_le16_to_cpu(
|
|
tlv_akmp->key_mgmt_operation);
|
|
break;
|
|
case TLV_TYPE_PWK_CIPHER:
|
|
tlv_pwk_cipher = (MrvlIEtypes_pwk_cipher_t *)tlv;
|
|
if (wlan_le16_to_cpu(tlv_pwk_cipher->protocol) &
|
|
PROTOCOL_WPA)
|
|
bss->param.bss_config.wpa_cfg
|
|
.pairwise_cipher_wpa =
|
|
tlv_pwk_cipher->pairwise_cipher;
|
|
if (wlan_le16_to_cpu(tlv_pwk_cipher->protocol) &
|
|
PROTOCOL_WPA2)
|
|
bss->param.bss_config.wpa_cfg
|
|
.pairwise_cipher_wpa2 =
|
|
tlv_pwk_cipher->pairwise_cipher;
|
|
if (wlan_le16_to_cpu(tlv_pwk_cipher->protocol) &
|
|
PROTOCOL_WPA3_SAE)
|
|
bss->param.bss_config.wpa_cfg
|
|
.pairwise_cipher_wpa2 =
|
|
tlv_pwk_cipher->pairwise_cipher;
|
|
break;
|
|
case TLV_TYPE_GWK_CIPHER:
|
|
tlv_gwk_cipher = (MrvlIEtypes_gwk_cipher_t *)tlv;
|
|
bss->param.bss_config.wpa_cfg.group_cipher =
|
|
tlv_gwk_cipher->group_cipher;
|
|
break;
|
|
case TLV_TYPE_UAP_RSN_REPLAY_PROTECT:
|
|
tlv_rsn_prot = (MrvlIEtypes_rsn_replay_prot_t *)tlv;
|
|
bss->param.bss_config.wpa_cfg.rsn_protection =
|
|
tlv_rsn_prot->rsn_replay_prot;
|
|
break;
|
|
case TLV_TYPE_UAP_WPA_PASSPHRASE:
|
|
tlv_passphrase = (MrvlIEtypes_passphrase_t *)tlv;
|
|
bss->param.bss_config.wpa_cfg.length =
|
|
MIN(MLAN_PMK_HEXSTR_LENGTH, tlv_len);
|
|
memcpy_ext(pmpriv->adapter,
|
|
bss->param.bss_config.wpa_cfg.passphrase,
|
|
tlv_passphrase->passphrase,
|
|
bss->param.bss_config.wpa_cfg.length,
|
|
sizeof(bss->param.bss_config.wpa_cfg
|
|
.passphrase));
|
|
break;
|
|
#ifdef WIFI_DIRECT_SUPPORT
|
|
case TLV_TYPE_UAP_PSK:
|
|
tlv_psk = (MrvlIEtypes_psk_t *)tlv;
|
|
memcpy_ext(pmpriv->adapter, bss->param.bss_config.psk,
|
|
tlv_psk->psk, tlv_len, MLAN_MAX_KEY_LENGTH);
|
|
break;
|
|
#endif /* WIFI_DIRECT_SUPPORT */
|
|
case TLV_TYPE_UAP_GRP_REKEY_TIME:
|
|
tlv_rekey_time = (MrvlIEtypes_group_rekey_time_t *)tlv;
|
|
bss->param.bss_config.wpa_cfg.gk_rekey_time =
|
|
wlan_le32_to_cpu(tlv_rekey_time->gk_rekey_time);
|
|
break;
|
|
case TLV_TYPE_UAP_WEP_KEY:
|
|
tlv_wep_key = (MrvlIEtypes_wep_key_t *)tlv;
|
|
pkey = MNULL;
|
|
if (tlv_wep_key->key_index == 0)
|
|
pkey = &bss->param.bss_config.wep_cfg.key0;
|
|
else if (tlv_wep_key->key_index == 1)
|
|
pkey = &bss->param.bss_config.wep_cfg.key1;
|
|
else if (tlv_wep_key->key_index == 2)
|
|
pkey = &bss->param.bss_config.wep_cfg.key2;
|
|
else if (tlv_wep_key->key_index == 3)
|
|
pkey = &bss->param.bss_config.wep_cfg.key3;
|
|
if (pkey) {
|
|
pkey->key_index = tlv_wep_key->key_index;
|
|
pkey->is_default = tlv_wep_key->is_default;
|
|
pkey->length =
|
|
MIN(MAX_WEP_KEY_SIZE, (tlv_len - 2));
|
|
memcpy_ext(pmpriv->adapter, pkey->key,
|
|
tlv_wep_key->key, pkey->length,
|
|
pkey->length);
|
|
}
|
|
break;
|
|
case TLV_TYPE_UAP_PREAMBLE_CTL:
|
|
tlv_preamble = (MrvlIEtypes_preamble_t *)tlv;
|
|
bss->param.bss_config.preamble_type =
|
|
tlv_preamble->preamble_type;
|
|
break;
|
|
case TLV_TYPE_BSS_STATUS:
|
|
tlv_bss_status = (MrvlIEtypes_bss_status_t *)tlv;
|
|
bss->param.bss_config.bss_status =
|
|
wlan_le16_to_cpu(tlv_bss_status->bss_status);
|
|
pmpriv->uap_bss_started =
|
|
(bss->param.bss_config.bss_status) ? MTRUE :
|
|
MFALSE;
|
|
break;
|
|
case TLV_TYPE_HT_CAPABILITY:
|
|
tlv_htcap = (MrvlIETypes_HTCap_t *)tlv;
|
|
bss->param.bss_config.ht_cap_info =
|
|
wlan_le16_to_cpu(tlv_htcap->ht_cap.ht_cap_info);
|
|
bss->param.bss_config.ampdu_param =
|
|
tlv_htcap->ht_cap.ampdu_param;
|
|
memcpy_ext(
|
|
pmpriv->adapter,
|
|
bss->param.bss_config.supported_mcs_set,
|
|
tlv_htcap->ht_cap.supported_mcs_set, 16,
|
|
sizeof(bss->param.bss_config.supported_mcs_set));
|
|
bss->param.bss_config.ht_ext_cap =
|
|
wlan_le16_to_cpu(tlv_htcap->ht_cap.ht_ext_cap);
|
|
bss->param.bss_config.tx_bf_cap =
|
|
wlan_le32_to_cpu(tlv_htcap->ht_cap.tx_bf_cap);
|
|
bss->param.bss_config.asel = tlv_htcap->ht_cap.asel;
|
|
break;
|
|
case TLV_TYPE_VENDOR_SPECIFIC_IE:
|
|
tlv_wmm_parameter = (MrvlIEtypes_wmm_parameter_t *)tlv;
|
|
bss->param.bss_config.wmm_para.qos_info =
|
|
tlv_wmm_parameter->wmm_para.qos_info;
|
|
for (ac = 0; ac < 4; ac++) {
|
|
bss->param.bss_config.wmm_para.ac_params[ac]
|
|
.aci_aifsn.aifsn =
|
|
tlv_wmm_parameter->wmm_para
|
|
.ac_params[ac]
|
|
.aci_aifsn.aifsn;
|
|
bss->param.bss_config.wmm_para.ac_params[ac]
|
|
.aci_aifsn.aci =
|
|
tlv_wmm_parameter->wmm_para
|
|
.ac_params[ac]
|
|
.aci_aifsn.aci;
|
|
bss->param.bss_config.wmm_para.ac_params[ac]
|
|
.ecw.ecw_max =
|
|
tlv_wmm_parameter->wmm_para
|
|
.ac_params[ac]
|
|
.ecw.ecw_max;
|
|
bss->param.bss_config.wmm_para.ac_params[ac]
|
|
.ecw.ecw_min =
|
|
tlv_wmm_parameter->wmm_para
|
|
.ac_params[ac]
|
|
.ecw.ecw_min;
|
|
bss->param.bss_config.wmm_para.ac_params[ac]
|
|
.tx_op_limit = wlan_le16_to_cpu(
|
|
tlv_wmm_parameter->wmm_para
|
|
.ac_params[ac]
|
|
.tx_op_limit);
|
|
}
|
|
break;
|
|
}
|
|
|
|
tlv_buf_left -= tlv_len + sizeof(MrvlIEtypesHeader_t);
|
|
tlv = (MrvlIEtypesHeader_t *)((t_u8 *)tlv + tlv_len +
|
|
sizeof(MrvlIEtypesHeader_t));
|
|
}
|
|
#ifdef DRV_EMBEDDED_AUTHENTICATOR
|
|
if (!IS_FW_SUPPORT_AUTHENTICATOR(pmpriv->adapter))
|
|
AuthenticatorBssConfig(pmpriv->psapriv,
|
|
(t_u8 *)&bss->param.bss_config, 0, 0, 1);
|
|
#endif
|
|
LEAVE();
|
|
return MLAN_STATUS_SUCCESS;
|
|
}
|
|
|
|
/**
|
|
* @brief This function handles the command response of sys_reset
|
|
* Clear various private state variables used by DFS.
|
|
*
|
|
* @param pmpriv A pointer to mlan_private structure
|
|
* @param resp A pointer to HostCmd_DS_COMMAND
|
|
* @param pioctl_buf A pointer to mlan_ioctl_req structure
|
|
*
|
|
* @return MLAN_STATUS_SUCCESS
|
|
*/
|
|
static mlan_status wlan_uap_ret_sys_reset(pmlan_private pmpriv,
|
|
HostCmd_DS_COMMAND *resp,
|
|
mlan_ioctl_req *pioctl_buf)
|
|
{
|
|
ENTER();
|
|
|
|
memset(pmpriv->adapter, &(pmpriv->uap_state_chan_cb.bandcfg), 0,
|
|
sizeof(pmpriv->uap_state_chan_cb.bandcfg));
|
|
pmpriv->uap_state_chan_cb.channel = 0;
|
|
pmpriv->uap_state_chan_cb.beacon_period = 0;
|
|
pmpriv->uap_state_chan_cb.dtim_period = 0;
|
|
|
|
/* assume default 11d/11h states are off, should check with FW */
|
|
/* currently don't clear domain_info... global, could be from STA */
|
|
wlan_11d_priv_init(pmpriv);
|
|
wlan_11h_priv_init(pmpriv);
|
|
|
|
LEAVE();
|
|
return MLAN_STATUS_SUCCESS;
|
|
}
|
|
|
|
/**
|
|
* @brief This function handles the command response of sys_config
|
|
*
|
|
* @param pmpriv A pointer to mlan_private structure
|
|
* @param resp A pointer to HostCmd_DS_COMMAND
|
|
* @param pioctl_buf A pointer to mlan_ioctl_req structure
|
|
*
|
|
* @return MLAN_STATUS_SUCCESS
|
|
*/
|
|
static mlan_status wlan_uap_ret_sys_config(pmlan_private pmpriv,
|
|
HostCmd_DS_COMMAND *resp,
|
|
mlan_ioctl_req *pioctl_buf)
|
|
{
|
|
int resp_len = 0, travel_len = 0;
|
|
t_u32 i = 0;
|
|
custom_ie *cptr;
|
|
HostCmd_DS_SYS_CONFIG *sys_config =
|
|
(HostCmd_DS_SYS_CONFIG *)&resp->params.sys_config;
|
|
mlan_ds_bss *bss = MNULL;
|
|
mlan_ds_misc_cfg *misc = MNULL;
|
|
MrvlIEtypes_MacAddr_t *tlv =
|
|
(MrvlIEtypes_MacAddr_t *)sys_config->tlv_buffer;
|
|
mlan_ds_misc_custom_ie *cust_ie = MNULL;
|
|
tlvbuf_max_mgmt_ie *max_mgmt_ie = MNULL;
|
|
MrvlIEtypes_wmm_parameter_t *tlv_wmm_parameter =
|
|
(MrvlIEtypes_wmm_parameter_t *)sys_config->tlv_buffer;
|
|
MrvlIEtypes_ChanListParamSet_t *tlv_chan_list =
|
|
(MrvlIEtypes_ChanListParamSet_t *)sys_config->tlv_buffer;
|
|
MrvlIEtypes_channel_band_t *chan_band_tlv =
|
|
(MrvlIEtypes_channel_band_t *)sys_config->tlv_buffer;
|
|
ChanScanParamSet_t *pscan_chan = MNULL;
|
|
t_u8 ac = 0;
|
|
MrvlIEtypes_channel_band_t *tlv_cb = MNULL;
|
|
MrvlIEtypes_beacon_period_t *tlv_bcnpd = MNULL;
|
|
MrvlIEtypes_dtim_period_t *tlv_dtimpd = MNULL;
|
|
MrvlIEtypes_uap_max_sta_cnt_t *tlv_uap_max_sta = MNULL;
|
|
|
|
ENTER();
|
|
if (pioctl_buf) {
|
|
if (pioctl_buf->req_id == MLAN_IOCTL_BSS) {
|
|
bss = (mlan_ds_bss *)pioctl_buf->pbuf;
|
|
if (bss->sub_command == MLAN_OID_BSS_MAC_ADDR) {
|
|
if (TLV_TYPE_UAP_MAC_ADDRESS ==
|
|
wlan_le16_to_cpu(tlv->header.type)) {
|
|
memcpy_ext(pmpriv->adapter,
|
|
&bss->param.mac_addr,
|
|
tlv->mac,
|
|
MLAN_MAC_ADDR_LENGTH,
|
|
MLAN_MAC_ADDR_LENGTH);
|
|
}
|
|
} else if (bss->sub_command ==
|
|
MLAN_OID_UAP_CFG_WMM_PARAM) {
|
|
if (TLV_TYPE_AP_WMM_PARAM ==
|
|
wlan_le16_to_cpu(
|
|
tlv_wmm_parameter->header.type)) {
|
|
if (wlan_le16_to_cpu(
|
|
tlv_wmm_parameter->header
|
|
.len) <
|
|
sizeof(bss->param.ap_wmm_para)) {
|
|
PRINTM(MCMND,
|
|
"FW don't support AP WMM PARAM\n");
|
|
} else {
|
|
bss->param.ap_wmm_para.reserved =
|
|
MLAN_STATUS_COMPLETE;
|
|
for (ac = 0; ac < 4; ac++) {
|
|
bss->param.ap_wmm_para
|
|
.ac_params[ac]
|
|
.aci_aifsn
|
|
.aifsn =
|
|
tlv_wmm_parameter
|
|
->wmm_para
|
|
.ac_params
|
|
[ac]
|
|
.aci_aifsn
|
|
.aifsn;
|
|
bss->param.ap_wmm_para
|
|
.ac_params[ac]
|
|
.aci_aifsn.aci =
|
|
tlv_wmm_parameter
|
|
->wmm_para
|
|
.ac_params
|
|
[ac]
|
|
.aci_aifsn
|
|
.aci;
|
|
bss->param.ap_wmm_para
|
|
.ac_params[ac]
|
|
.ecw.ecw_max =
|
|
tlv_wmm_parameter
|
|
->wmm_para
|
|
.ac_params
|
|
[ac]
|
|
.ecw
|
|
.ecw_max;
|
|
bss->param.ap_wmm_para
|
|
.ac_params[ac]
|
|
.ecw.ecw_min =
|
|
tlv_wmm_parameter
|
|
->wmm_para
|
|
.ac_params
|
|
[ac]
|
|
.ecw
|
|
.ecw_min;
|
|
bss->param.ap_wmm_para
|
|
.ac_params[ac]
|
|
.tx_op_limit = wlan_le16_to_cpu(
|
|
tlv_wmm_parameter
|
|
->wmm_para
|
|
.ac_params
|
|
[ac]
|
|
.tx_op_limit);
|
|
PRINTM(MCMND,
|
|
"ac=%d, aifsn=%d, aci=%d, ecw_max=%d, ecw_min=%d, tx_op=%d\n",
|
|
ac,
|
|
bss->param
|
|
.ap_wmm_para
|
|
.ac_params
|
|
[ac]
|
|
.aci_aifsn
|
|
.aifsn,
|
|
bss->param
|
|
.ap_wmm_para
|
|
.ac_params
|
|
[ac]
|
|
.aci_aifsn
|
|
.aci,
|
|
bss->param
|
|
.ap_wmm_para
|
|
.ac_params
|
|
[ac]
|
|
.ecw
|
|
.ecw_max,
|
|
bss->param
|
|
.ap_wmm_para
|
|
.ac_params
|
|
[ac]
|
|
.ecw
|
|
.ecw_min,
|
|
bss->param
|
|
.ap_wmm_para
|
|
.ac_params
|
|
[ac]
|
|
.tx_op_limit);
|
|
}
|
|
}
|
|
}
|
|
} else if (bss->sub_command ==
|
|
MLAN_OID_UAP_SCAN_CHANNELS) {
|
|
if (TLV_TYPE_CHANLIST ==
|
|
wlan_le16_to_cpu(
|
|
tlv_chan_list->header.type)) {
|
|
pscan_chan =
|
|
tlv_chan_list->chan_scan_param;
|
|
bss->param.ap_scan_channels.num_of_chan =
|
|
0;
|
|
for (i = 0;
|
|
i <
|
|
(int)wlan_le16_to_cpu(
|
|
tlv_chan_list->header.len) /
|
|
sizeof(ChanScanParamSet_t);
|
|
i++) {
|
|
if (bss->param.ap_scan_channels
|
|
.remove_nop_channel &&
|
|
wlan_11h_is_channel_under_nop(
|
|
pmpriv->adapter,
|
|
pscan_chan
|
|
->chan_number)) {
|
|
bss->param
|
|
.ap_scan_channels
|
|
.num_remvoed_channel++;
|
|
PRINTM(MCMND,
|
|
"Remove nop channel=%d\n",
|
|
pscan_chan
|
|
->chan_number);
|
|
pscan_chan++;
|
|
continue;
|
|
}
|
|
bss->param.ap_scan_channels
|
|
.chan_list
|
|
[bss->param
|
|
.ap_scan_channels
|
|
.num_of_chan]
|
|
.chan_number =
|
|
pscan_chan->chan_number;
|
|
bss->param.ap_scan_channels
|
|
.chan_list
|
|
[bss->param
|
|
.ap_scan_channels
|
|
.num_of_chan]
|
|
.bandcfg =
|
|
pscan_chan->bandcfg;
|
|
bss->param.ap_scan_channels
|
|
.num_of_chan++;
|
|
pscan_chan++;
|
|
}
|
|
PRINTM(MCMND,
|
|
"AP scan channel list=%d\n",
|
|
bss->param.ap_scan_channels
|
|
.num_of_chan);
|
|
}
|
|
} else if (bss->sub_command == MLAN_OID_UAP_CHANNEL) {
|
|
if (TLV_TYPE_UAP_CHAN_BAND_CONFIG ==
|
|
wlan_le16_to_cpu(
|
|
chan_band_tlv->header.type)) {
|
|
bss->param.ap_channel.bandcfg =
|
|
chan_band_tlv->bandcfg;
|
|
bss->param.ap_channel.channel =
|
|
chan_band_tlv->channel;
|
|
bss->param.ap_channel.is_11n_enabled =
|
|
pmpriv->is_11n_enabled;
|
|
bss->param.ap_channel.is_dfs_chan =
|
|
wlan_11h_radar_detect_required(
|
|
pmpriv,
|
|
bss->param.ap_channel
|
|
.channel);
|
|
if (chan_band_tlv->bandcfg.chanWidth ==
|
|
CHAN_BW_80MHZ)
|
|
bss->param.ap_channel
|
|
.center_chan =
|
|
wlan_get_center_freq_idx(
|
|
pmpriv,
|
|
BAND_AAC,
|
|
chan_band_tlv
|
|
->channel,
|
|
CHANNEL_BW_80MHZ);
|
|
PRINTM(MCMND,
|
|
"AP channel, band=0x%x, channel=%d, is_11n_enabled=%d center_chan=%d\n",
|
|
bss->param.ap_channel.bandcfg,
|
|
bss->param.ap_channel.channel,
|
|
bss->param.ap_channel
|
|
.is_11n_enabled,
|
|
bss->param.ap_channel
|
|
.center_chan);
|
|
}
|
|
} else if ((bss->sub_command ==
|
|
MLAN_OID_UAP_BSS_CONFIG) &&
|
|
(pioctl_buf->action == MLAN_ACT_GET)) {
|
|
wlan_uap_ret_cmd_ap_config(pmpriv, resp,
|
|
pioctl_buf);
|
|
}
|
|
}
|
|
if (pioctl_buf->req_id == MLAN_IOCTL_MISC_CFG) {
|
|
misc = (mlan_ds_misc_cfg *)pioctl_buf->pbuf;
|
|
cust_ie = (mlan_ds_misc_custom_ie *)
|
|
sys_config->tlv_buffer;
|
|
if ((pioctl_buf->action == MLAN_ACT_GET ||
|
|
pioctl_buf->action == MLAN_ACT_SET) &&
|
|
(misc->sub_command == MLAN_OID_MISC_CUSTOM_IE)) {
|
|
cust_ie->type = wlan_le16_to_cpu(cust_ie->type);
|
|
resp_len = cust_ie->len =
|
|
wlan_le16_to_cpu(cust_ie->len);
|
|
travel_len = 0;
|
|
/* conversion for index, mask, len */
|
|
if (resp_len == sizeof(t_u16))
|
|
cust_ie->ie_data_list[0].ie_index =
|
|
wlan_cpu_to_le16(
|
|
cust_ie->ie_data_list[0]
|
|
.ie_index);
|
|
|
|
while (resp_len > (int)sizeof(t_u16)) {
|
|
cptr = (custom_ie
|
|
*)(((t_u8 *)cust_ie
|
|
->ie_data_list) +
|
|
travel_len);
|
|
cptr->ie_index = wlan_le16_to_cpu(
|
|
cptr->ie_index);
|
|
cptr->mgmt_subtype_mask =
|
|
wlan_le16_to_cpu(
|
|
cptr->mgmt_subtype_mask);
|
|
cptr->ie_length = wlan_le16_to_cpu(
|
|
cptr->ie_length);
|
|
travel_len += cptr->ie_length +
|
|
sizeof(custom_ie) -
|
|
MAX_IE_SIZE;
|
|
resp_len -= cptr->ie_length +
|
|
sizeof(custom_ie) -
|
|
MAX_IE_SIZE;
|
|
}
|
|
memcpy_ext(pmpriv->adapter,
|
|
&misc->param.cust_ie, cust_ie,
|
|
cust_ie->len +
|
|
sizeof(MrvlIEtypesHeader_t),
|
|
sizeof(mlan_ds_misc_custom_ie) -
|
|
sizeof(tlvbuf_max_mgmt_ie));
|
|
max_mgmt_ie =
|
|
(tlvbuf_max_mgmt_ie
|
|
*)(sys_config->tlv_buffer +
|
|
cust_ie->len +
|
|
sizeof(MrvlIEtypesHeader_t));
|
|
if (max_mgmt_ie) {
|
|
max_mgmt_ie->type = wlan_le16_to_cpu(
|
|
max_mgmt_ie->type);
|
|
if (max_mgmt_ie->type ==
|
|
TLV_TYPE_MAX_MGMT_IE) {
|
|
max_mgmt_ie->len =
|
|
wlan_le16_to_cpu(
|
|
max_mgmt_ie
|
|
->len);
|
|
max_mgmt_ie->count =
|
|
wlan_le16_to_cpu(
|
|
max_mgmt_ie
|
|
->count);
|
|
for (i = 0;
|
|
i < max_mgmt_ie->count;
|
|
i++) {
|
|
max_mgmt_ie->info[i]
|
|
.buf_size = wlan_le16_to_cpu(
|
|
max_mgmt_ie
|
|
->info[i]
|
|
.buf_size);
|
|
max_mgmt_ie->info[i]
|
|
.buf_count = wlan_le16_to_cpu(
|
|
max_mgmt_ie
|
|
->info[i]
|
|
.buf_count);
|
|
}
|
|
/* Append max_mgmt_ie TLV after
|
|
* custom_ie */
|
|
memcpy_ext(
|
|
pmpriv->adapter,
|
|
(t_u8 *)&misc->param
|
|
.cust_ie +
|
|
(cust_ie->len +
|
|
sizeof(MrvlIEtypesHeader_t)),
|
|
max_mgmt_ie,
|
|
max_mgmt_ie->len +
|
|
sizeof(MrvlIEtypesHeader_t),
|
|
sizeof(tlvbuf_max_mgmt_ie));
|
|
}
|
|
}
|
|
}
|
|
}
|
|
} else { /* no ioctl: driver generated get/set */
|
|
switch (wlan_le16_to_cpu(tlv->header.type)) {
|
|
case TLV_TYPE_UAP_MAC_ADDRESS:
|
|
memcpy_ext(pmpriv->adapter, pmpriv->curr_addr, tlv->mac,
|
|
MLAN_MAC_ADDR_LENGTH, MLAN_MAC_ADDR_LENGTH);
|
|
break;
|
|
case TLV_TYPE_UAP_MAX_STA_CNT_PER_CHIP:
|
|
tlv_uap_max_sta = (MrvlIEtypes_uap_max_sta_cnt_t *)tlv;
|
|
pmpriv->adapter->max_sta_conn =
|
|
wlan_le16_to_cpu(tlv_uap_max_sta->uap_max_sta);
|
|
PRINTM(MCMND, "Uap max_sta per chip=%d\n",
|
|
wlan_le16_to_cpu(tlv_uap_max_sta->uap_max_sta));
|
|
break;
|
|
case TLV_TYPE_UAP_CHAN_BAND_CONFIG:
|
|
tlv_cb = (MrvlIEtypes_channel_band_t *)tlv;
|
|
pmpriv->uap_state_chan_cb.bandcfg = tlv_cb->bandcfg;
|
|
pmpriv->uap_state_chan_cb.channel = tlv_cb->channel;
|
|
/* call callback waiting for channel info */
|
|
if (pmpriv->uap_state_chan_cb.get_chan_callback)
|
|
pmpriv->uap_state_chan_cb.get_chan_callback(
|
|
pmpriv);
|
|
break;
|
|
case TLV_TYPE_UAP_BEACON_PERIOD:
|
|
tlv_bcnpd = (MrvlIEtypes_beacon_period_t *)tlv;
|
|
pmpriv->uap_state_chan_cb.beacon_period =
|
|
wlan_le16_to_cpu(tlv_bcnpd->beacon_period);
|
|
/* copy dtim_period as well if it follows */
|
|
tlv_dtimpd =
|
|
(MrvlIEtypes_dtim_period_t
|
|
*)(((t_u8 *)tlv) +
|
|
sizeof(MrvlIEtypes_beacon_period_t));
|
|
if (TLV_TYPE_UAP_DTIM_PERIOD ==
|
|
wlan_le16_to_cpu(tlv_dtimpd->header.type))
|
|
pmpriv->uap_state_chan_cb.dtim_period =
|
|
tlv_dtimpd->dtim_period;
|
|
/* call callback waiting for beacon/dtim info */
|
|
if (pmpriv->uap_state_chan_cb.get_chan_callback)
|
|
pmpriv->uap_state_chan_cb.get_chan_callback(
|
|
pmpriv);
|
|
break;
|
|
case TLV_TYPE_MGMT_IE:
|
|
if ((pmpriv->adapter->state_rdh.stage ==
|
|
RDH_SET_CUSTOM_IE) ||
|
|
(pmpriv->adapter->state_rdh.stage ==
|
|
RDH_REM_CUSTOM_IE)) {
|
|
if (!pmpriv->adapter->ecsa_enable)
|
|
wlan_11h_radar_detected_callback(
|
|
(t_void *)pmpriv);
|
|
}
|
|
break;
|
|
}
|
|
}
|
|
LEAVE();
|
|
return MLAN_STATUS_SUCCESS;
|
|
}
|
|
|
|
/**
|
|
* @brief This function prepares command of snmp_mib
|
|
*
|
|
* @param pmpriv A pointer to mlan_private structure
|
|
* @param cmd A pointer to HostCmd_DS_COMMAND structure
|
|
* @param cmd_action the action: GET or SET
|
|
* @param cmd_oid Cmd oid: treated as sub command
|
|
* @param pioctl_buf A pointer to mlan_ioctl_req structure
|
|
* @param pdata_buf A pointer to information buffer
|
|
* @return MLAN_STATUS_SUCCESS or MLAN_STATUS_FAILURE
|
|
*/
|
|
static mlan_status wlan_uap_cmd_snmp_mib(pmlan_private pmpriv,
|
|
HostCmd_DS_COMMAND *cmd,
|
|
t_u16 cmd_action, t_u32 cmd_oid,
|
|
pmlan_ioctl_req pioctl_buf,
|
|
t_void *pdata_buf)
|
|
{
|
|
HostCmd_DS_802_11_SNMP_MIB *psnmp_mib = &cmd->params.smib;
|
|
HostCmd_DS_UAP_802_11_SNMP_MIB *puap_snmp_mib = &cmd->params.uap_smib;
|
|
|
|
mlan_status ret = MLAN_STATUS_SUCCESS;
|
|
t_u8 *psnmp_oid = MNULL;
|
|
t_u32 ul_temp;
|
|
t_u8 i;
|
|
|
|
t_u8 snmp_oids[] = {
|
|
tkip_mic_failures,
|
|
ccmp_decrypt_errors,
|
|
wep_undecryptable_count,
|
|
wep_icv_error_count,
|
|
decrypt_failure_count,
|
|
dot11_mcast_tx_count,
|
|
dot11_failed_count,
|
|
dot11_retry_count,
|
|
dot11_multi_retry_count,
|
|
dot11_frame_dup_count,
|
|
dot11_rts_success_count,
|
|
dot11_rts_failure_count,
|
|
dot11_ack_failure_count,
|
|
dot11_rx_fragment_count,
|
|
dot11_mcast_rx_frame_count,
|
|
dot11_fcs_error_count,
|
|
dot11_tx_frame_count,
|
|
dot11_rsna_tkip_cm_invoked,
|
|
dot11_rsna_4way_hshk_failures,
|
|
};
|
|
|
|
ENTER();
|
|
|
|
if (cmd_action == HostCmd_ACT_GEN_GET) {
|
|
cmd->command = wlan_cpu_to_le16(HostCmd_CMD_802_11_SNMP_MIB);
|
|
psnmp_mib->query_type = wlan_cpu_to_le16(HostCmd_ACT_GEN_GET);
|
|
if (cmd_oid == StopDeauth_i) {
|
|
psnmp_mib->oid = wlan_cpu_to_le16((t_u16)StopDeauth_i);
|
|
psnmp_mib->buf_size = wlan_cpu_to_le16(sizeof(t_u8));
|
|
cmd->size = wlan_cpu_to_le16(
|
|
sizeof(HostCmd_DS_802_11_SNMP_MIB) + S_DS_GEN);
|
|
} else {
|
|
cmd->size = wlan_cpu_to_le16(
|
|
sizeof(t_u16) + S_DS_GEN +
|
|
sizeof(snmp_oids) *
|
|
sizeof(MrvlIEtypes_snmp_oid_t));
|
|
psnmp_oid = (t_u8 *)&puap_snmp_mib->snmp_data;
|
|
for (i = 0; i < sizeof(snmp_oids); i++) {
|
|
/* SNMP OID header type */
|
|
*(t_u16 *)psnmp_oid =
|
|
wlan_cpu_to_le16(snmp_oids[i]);
|
|
psnmp_oid += sizeof(t_u16);
|
|
/* SNMP OID header length */
|
|
*(t_u16 *)psnmp_oid =
|
|
wlan_cpu_to_le16(sizeof(t_u32));
|
|
psnmp_oid += sizeof(t_u16) + sizeof(t_u32);
|
|
}
|
|
}
|
|
} else { /* cmd_action == ACT_SET */
|
|
cmd->command = wlan_cpu_to_le16(HostCmd_CMD_802_11_SNMP_MIB);
|
|
cmd->size = sizeof(HostCmd_DS_802_11_SNMP_MIB) - 1 + S_DS_GEN;
|
|
psnmp_mib->query_type = wlan_cpu_to_le16(HostCmd_ACT_GEN_SET);
|
|
|
|
switch (cmd_oid) {
|
|
case Dot11D_i:
|
|
case Dot11H_i:
|
|
psnmp_mib->oid = wlan_cpu_to_le16((t_u16)cmd_oid);
|
|
psnmp_mib->buf_size = wlan_cpu_to_le16(sizeof(t_u16));
|
|
ul_temp = *(t_u32 *)pdata_buf;
|
|
*((t_u16 *)(psnmp_mib->value)) =
|
|
wlan_cpu_to_le16((t_u16)ul_temp);
|
|
cmd->size += sizeof(t_u16);
|
|
break;
|
|
case ECSAEnable_i:
|
|
psnmp_mib->oid = wlan_cpu_to_le16((t_u16)cmd_oid);
|
|
psnmp_mib->buf_size = wlan_cpu_to_le16(sizeof(t_u8));
|
|
psnmp_mib->value[0] = *((t_u8 *)pdata_buf);
|
|
cmd->size += sizeof(t_u8);
|
|
break;
|
|
case StopDeauth_i:
|
|
psnmp_mib->oid = wlan_cpu_to_le16((t_u16)cmd_oid);
|
|
psnmp_mib->buf_size = wlan_cpu_to_le16(sizeof(t_u8));
|
|
psnmp_mib->value[0] = *((t_u8 *)pdata_buf);
|
|
cmd->size += sizeof(t_u8);
|
|
break;
|
|
default:
|
|
PRINTM(MERROR, "Unsupported OID.\n");
|
|
ret = MLAN_STATUS_FAILURE;
|
|
break;
|
|
}
|
|
cmd->size = wlan_cpu_to_le16(cmd->size);
|
|
}
|
|
|
|
LEAVE();
|
|
return ret;
|
|
}
|
|
|
|
/**
|
|
* @brief This function prepares command of get_log.
|
|
*
|
|
* @param pmpriv A pointer to mlan_private structure
|
|
* @param cmd A pointer to HostCmd_DS_COMMAND structure
|
|
*
|
|
* @return MLAN_STATUS_SUCCESS
|
|
*/
|
|
static mlan_status wlan_uap_cmd_802_11_get_log(pmlan_private pmpriv,
|
|
HostCmd_DS_COMMAND *cmd)
|
|
{
|
|
ENTER();
|
|
cmd->command = wlan_cpu_to_le16(HostCmd_CMD_802_11_GET_LOG);
|
|
cmd->size =
|
|
wlan_cpu_to_le16(sizeof(HostCmd_DS_802_11_GET_LOG) + S_DS_GEN);
|
|
LEAVE();
|
|
return MLAN_STATUS_SUCCESS;
|
|
}
|
|
|
|
/**
|
|
* @brief This function prepares command of bss_start.
|
|
*
|
|
* @param pmpriv A pointer to mlan_private structure
|
|
* @param cmd A pointer to HostCmd_DS_COMMAND structure
|
|
*
|
|
* @return MLAN_STATUS_SUCCESS
|
|
*/
|
|
static mlan_status wlan_uap_cmd_bss_start(pmlan_private pmpriv,
|
|
HostCmd_DS_COMMAND *cmd)
|
|
{
|
|
MrvlIEtypes_HostMlme_t *tlv;
|
|
|
|
ENTER();
|
|
cmd->command = wlan_cpu_to_le16(HOST_CMD_APCMD_BSS_START);
|
|
cmd->size = S_DS_GEN;
|
|
if (pmpriv->uap_host_based & UAP_FLAG_HOST_MLME) {
|
|
tlv = (MrvlIEtypes_HostMlme_t *)((t_u8 *)cmd + cmd->size);
|
|
tlv->header.type = wlan_cpu_to_le16(TLV_TYPE_HOST_MLME);
|
|
tlv->header.len = wlan_cpu_to_le16(sizeof(tlv->host_mlme));
|
|
tlv->host_mlme = MTRUE;
|
|
cmd->size += sizeof(MrvlIEtypes_HostMlme_t);
|
|
}
|
|
cmd->size = wlan_cpu_to_le16(cmd->size);
|
|
LEAVE();
|
|
return MLAN_STATUS_SUCCESS;
|
|
}
|
|
|
|
/**
|
|
* @brief This function handles the command response of snmp_mib
|
|
*
|
|
* @param pmpriv A pointer to mlan_private structure
|
|
* @param resp A pointer to HostCmd_DS_COMMAND
|
|
* @param pioctl_buf A pointer to mlan_ioctl_req structure
|
|
*
|
|
* @return MLAN_STATUS_SUCCESS
|
|
*/
|
|
static mlan_status wlan_uap_ret_snmp_mib(pmlan_private pmpriv,
|
|
HostCmd_DS_COMMAND *resp,
|
|
mlan_ioctl_req *pioctl_buf)
|
|
{
|
|
pmlan_adapter pmadapter = pmpriv->adapter;
|
|
HostCmd_DS_802_11_SNMP_MIB *psnmp_mib =
|
|
(HostCmd_DS_802_11_SNMP_MIB *)&resp->params.smib;
|
|
mlan_ds_get_info *info;
|
|
mlan_ds_snmp_mib *mib = MNULL;
|
|
t_u16 oid = wlan_le16_to_cpu(psnmp_mib->oid);
|
|
t_u8 *psnmp_oid = MNULL;
|
|
t_u32 data;
|
|
t_u16 tlv_buf_left = 0;
|
|
t_u16 tlv_type = 0;
|
|
t_u16 query_type = wlan_le16_to_cpu(psnmp_mib->query_type);
|
|
|
|
ENTER();
|
|
if (query_type == HostCmd_ACT_GEN_GET) {
|
|
if (!pioctl_buf) {
|
|
LEAVE();
|
|
return MLAN_STATUS_SUCCESS;
|
|
}
|
|
if (oid == StopDeauth_i) {
|
|
mib = (mlan_ds_snmp_mib *)pioctl_buf->pbuf;
|
|
if (mib)
|
|
mib->param.deauthctrl = psnmp_mib->value[0];
|
|
LEAVE();
|
|
return MLAN_STATUS_SUCCESS;
|
|
}
|
|
info = (mlan_ds_get_info *)pioctl_buf->pbuf;
|
|
tlv_buf_left = resp->size - (sizeof(t_u16) + S_DS_GEN);
|
|
psnmp_oid = (t_u8 *)&psnmp_mib->oid;
|
|
while (tlv_buf_left >= sizeof(MrvlIEtypes_snmp_oid_t)) {
|
|
tlv_type = wlan_le16_to_cpu(*(t_u16 *)psnmp_oid);
|
|
psnmp_oid += sizeof(t_u16) + sizeof(t_u16);
|
|
memcpy_ext(pmadapter, &data, psnmp_oid, sizeof(t_u32),
|
|
sizeof(t_u32));
|
|
switch (tlv_type) {
|
|
case tkip_mic_failures:
|
|
info->param.ustats.tkip_mic_failures =
|
|
wlan_le32_to_cpu(data);
|
|
break;
|
|
case ccmp_decrypt_errors:
|
|
info->param.ustats.ccmp_decrypt_errors =
|
|
wlan_le32_to_cpu(data);
|
|
break;
|
|
case wep_undecryptable_count:
|
|
info->param.ustats.wep_undecryptable_count =
|
|
wlan_le32_to_cpu(data);
|
|
break;
|
|
case wep_icv_error_count:
|
|
info->param.ustats.wep_icv_error_count =
|
|
wlan_le32_to_cpu(data);
|
|
break;
|
|
case decrypt_failure_count:
|
|
info->param.ustats.decrypt_failure_count =
|
|
wlan_le32_to_cpu(data);
|
|
break;
|
|
case dot11_mcast_tx_count:
|
|
info->param.ustats.mcast_tx_count =
|
|
wlan_le32_to_cpu(data);
|
|
break;
|
|
case dot11_failed_count:
|
|
info->param.ustats.failed_count =
|
|
wlan_le32_to_cpu(data);
|
|
break;
|
|
case dot11_retry_count:
|
|
info->param.ustats.retry_count =
|
|
wlan_le32_to_cpu(data);
|
|
break;
|
|
case dot11_multi_retry_count:
|
|
info->param.ustats.multi_retry_count =
|
|
wlan_le32_to_cpu(data);
|
|
break;
|
|
case dot11_frame_dup_count:
|
|
info->param.ustats.frame_dup_count =
|
|
wlan_le32_to_cpu(data);
|
|
break;
|
|
case dot11_rts_success_count:
|
|
info->param.ustats.rts_success_count =
|
|
wlan_le32_to_cpu(data);
|
|
break;
|
|
case dot11_rts_failure_count:
|
|
info->param.ustats.rts_failure_count =
|
|
wlan_le32_to_cpu(data);
|
|
break;
|
|
case dot11_ack_failure_count:
|
|
info->param.ustats.ack_failure_count =
|
|
wlan_le32_to_cpu(data);
|
|
break;
|
|
case dot11_rx_fragment_count:
|
|
info->param.ustats.rx_fragment_count =
|
|
wlan_le32_to_cpu(data);
|
|
break;
|
|
case dot11_mcast_rx_frame_count:
|
|
info->param.ustats.mcast_rx_frame_count =
|
|
wlan_le32_to_cpu(data);
|
|
break;
|
|
case dot11_fcs_error_count:
|
|
info->param.ustats.fcs_error_count =
|
|
wlan_le32_to_cpu(data);
|
|
break;
|
|
case dot11_tx_frame_count:
|
|
info->param.ustats.tx_frame_count =
|
|
wlan_le32_to_cpu(data);
|
|
break;
|
|
case dot11_rsna_tkip_cm_invoked:
|
|
info->param.ustats.rsna_tkip_cm_invoked =
|
|
wlan_le32_to_cpu(data);
|
|
break;
|
|
case dot11_rsna_4way_hshk_failures:
|
|
info->param.ustats.rsna_4way_hshk_failures =
|
|
wlan_le32_to_cpu(data);
|
|
break;
|
|
}
|
|
tlv_buf_left -= sizeof(MrvlIEtypes_snmp_oid_t);
|
|
psnmp_oid += sizeof(t_u32);
|
|
}
|
|
} else { /* ACT_SET */
|
|
switch (wlan_le16_to_cpu(psnmp_mib->oid)) {
|
|
case Dot11D_i:
|
|
data = wlan_le16_to_cpu(*((t_u16 *)(psnmp_mib->value)));
|
|
/* Set 11d state to private */
|
|
pmpriv->state_11d.enable_11d = data;
|
|
/* Set user enable flag if called from ioctl */
|
|
if (pioctl_buf)
|
|
pmpriv->state_11d.user_enable_11d = data;
|
|
break;
|
|
case Dot11H_i:
|
|
data = wlan_le16_to_cpu(*((t_u16 *)(psnmp_mib->value)));
|
|
/* Set 11h state to priv */
|
|
pmpriv->intf_state_11h.is_11h_active =
|
|
(data & ENABLE_11H_MASK);
|
|
/* Set radar_det state to adapter */
|
|
pmpriv->adapter->state_11h.is_master_radar_det_active =
|
|
(data & MASTER_RADAR_DET_MASK) ? MTRUE : MFALSE;
|
|
pmpriv->adapter->state_11h.is_slave_radar_det_active =
|
|
(data & SLAVE_RADAR_DET_MASK) ? MTRUE : MFALSE;
|
|
break;
|
|
}
|
|
}
|
|
|
|
LEAVE();
|
|
return MLAN_STATUS_SUCCESS;
|
|
}
|
|
|
|
/**
|
|
* @brief This function handles the command response of get_log
|
|
*
|
|
* @param pmpriv A pointer to mlan_private structure
|
|
* @param resp A pointer to HostCmd_DS_COMMAND
|
|
* @param pioctl_buf A pointer to mlan_ioctl_req structure
|
|
*
|
|
* @return MLAN_STATUS_SUCCESS
|
|
*/
|
|
static mlan_status wlan_uap_ret_get_log(pmlan_private pmpriv,
|
|
HostCmd_DS_COMMAND *resp,
|
|
mlan_ioctl_req *pioctl_buf)
|
|
{
|
|
HostCmd_DS_802_11_GET_LOG *pget_log =
|
|
(HostCmd_DS_802_11_GET_LOG *)&resp->params.get_log;
|
|
mlan_ds_get_info *pget_info = MNULL;
|
|
int i = 0;
|
|
|
|
ENTER();
|
|
|
|
if (pioctl_buf) {
|
|
pget_info = (mlan_ds_get_info *)pioctl_buf->pbuf;
|
|
pget_info->param.stats.mcast_tx_frame =
|
|
wlan_le32_to_cpu(pget_log->mcast_tx_frame);
|
|
pget_info->param.stats.failed =
|
|
wlan_le32_to_cpu(pget_log->failed);
|
|
pget_info->param.stats.retry =
|
|
wlan_le32_to_cpu(pget_log->retry);
|
|
pget_info->param.stats.multi_retry =
|
|
wlan_le32_to_cpu(pget_log->multiretry);
|
|
pget_info->param.stats.frame_dup =
|
|
wlan_le32_to_cpu(pget_log->frame_dup);
|
|
pget_info->param.stats.rts_success =
|
|
wlan_le32_to_cpu(pget_log->rts_success);
|
|
pget_info->param.stats.rts_failure =
|
|
wlan_le32_to_cpu(pget_log->rts_failure);
|
|
pget_info->param.stats.ack_failure =
|
|
wlan_le32_to_cpu(pget_log->ack_failure);
|
|
pget_info->param.stats.rx_frag =
|
|
wlan_le32_to_cpu(pget_log->rx_frag);
|
|
pget_info->param.stats.mcast_rx_frame =
|
|
wlan_le32_to_cpu(pget_log->mcast_rx_frame);
|
|
pget_info->param.stats.fcs_error =
|
|
wlan_le32_to_cpu(pget_log->fcs_error);
|
|
pget_info->param.stats.tx_frame =
|
|
wlan_le32_to_cpu(pget_log->tx_frame);
|
|
pget_info->param.stats.wep_icv_error[0] =
|
|
wlan_le32_to_cpu(pget_log->wep_icv_err_cnt[0]);
|
|
pget_info->param.stats.wep_icv_error[1] =
|
|
wlan_le32_to_cpu(pget_log->wep_icv_err_cnt[1]);
|
|
pget_info->param.stats.wep_icv_error[2] =
|
|
wlan_le32_to_cpu(pget_log->wep_icv_err_cnt[2]);
|
|
pget_info->param.stats.wep_icv_error[3] =
|
|
wlan_le32_to_cpu(pget_log->wep_icv_err_cnt[3]);
|
|
pget_info->param.stats.bcn_rcv_cnt =
|
|
wlan_le32_to_cpu(pget_log->bcn_rcv_cnt);
|
|
pget_info->param.stats.bcn_miss_cnt =
|
|
wlan_le32_to_cpu(pget_log->bcn_miss_cnt);
|
|
pget_info->param.stats.amsdu_rx_cnt = pmpriv->amsdu_rx_cnt;
|
|
pget_info->param.stats.msdu_in_rx_amsdu_cnt =
|
|
pmpriv->msdu_in_rx_amsdu_cnt;
|
|
pget_info->param.stats.amsdu_tx_cnt = pmpriv->amsdu_tx_cnt;
|
|
pget_info->param.stats.msdu_in_tx_amsdu_cnt =
|
|
pmpriv->msdu_in_tx_amsdu_cnt;
|
|
pget_info->param.stats.rx_stuck_issue_cnt[0] =
|
|
wlan_le32_to_cpu(pget_log->rx_stuck_issue_cnt[0]);
|
|
pget_info->param.stats.rx_stuck_issue_cnt[1] =
|
|
wlan_le32_to_cpu(pget_log->rx_stuck_issue_cnt[1]);
|
|
pget_info->param.stats.rx_stuck_recovery_cnt =
|
|
wlan_le32_to_cpu(pget_log->rx_stuck_recovery_cnt);
|
|
pget_info->param.stats.rx_stuck_tsf[0] =
|
|
wlan_le64_to_cpu(pget_log->rx_stuck_tsf[0]);
|
|
pget_info->param.stats.rx_stuck_tsf[1] =
|
|
wlan_le64_to_cpu(pget_log->rx_stuck_tsf[1]);
|
|
pget_info->param.stats.tx_watchdog_recovery_cnt =
|
|
wlan_le32_to_cpu(pget_log->tx_watchdog_recovery_cnt);
|
|
pget_info->param.stats.tx_watchdog_tsf[0] =
|
|
wlan_le64_to_cpu(pget_log->tx_watchdog_tsf[0]);
|
|
pget_info->param.stats.tx_watchdog_tsf[1] =
|
|
wlan_le64_to_cpu(pget_log->tx_watchdog_tsf[1]);
|
|
pget_info->param.stats.channel_switch_ann_sent =
|
|
wlan_le32_to_cpu(pget_log->channel_switch_ann_sent);
|
|
pget_info->param.stats.channel_switch_state =
|
|
wlan_le32_to_cpu(pget_log->channel_switch_state);
|
|
pget_info->param.stats.reg_class =
|
|
wlan_le32_to_cpu(pget_log->reg_class);
|
|
pget_info->param.stats.channel_number =
|
|
wlan_le32_to_cpu(pget_log->channel_number);
|
|
pget_info->param.stats.channel_switch_mode =
|
|
wlan_le32_to_cpu(pget_log->channel_switch_mode);
|
|
pget_info->param.stats.rx_reset_mac_recovery_cnt =
|
|
wlan_le32_to_cpu(pget_log->rx_reset_mac_recovery_cnt);
|
|
pget_info->param.stats.rx_Isr2_NotDone_Cnt =
|
|
wlan_le32_to_cpu(pget_log->rx_Isr2_NotDone_Cnt);
|
|
pget_info->param.stats.gdma_abort_cnt =
|
|
wlan_le32_to_cpu(pget_log->gdma_abort_cnt);
|
|
pget_info->param.stats.g_reset_rx_mac_cnt =
|
|
wlan_le32_to_cpu(pget_log->g_reset_rx_mac_cnt);
|
|
// Ownership error counters
|
|
pget_info->param.stats.dwCtlErrCnt =
|
|
wlan_le32_to_cpu(pget_log->dwCtlErrCnt);
|
|
pget_info->param.stats.dwBcnErrCnt =
|
|
wlan_le32_to_cpu(pget_log->dwBcnErrCnt);
|
|
pget_info->param.stats.dwMgtErrCnt =
|
|
wlan_le32_to_cpu(pget_log->dwMgtErrCnt);
|
|
pget_info->param.stats.dwDatErrCnt =
|
|
wlan_le32_to_cpu(pget_log->dwDatErrCnt);
|
|
pget_info->param.stats.bigtk_mmeGoodCnt =
|
|
wlan_le32_to_cpu(pget_log->bigtk_mmeGoodCnt);
|
|
pget_info->param.stats.bigtk_replayErrCnt =
|
|
wlan_le32_to_cpu(pget_log->bigtk_replayErrCnt);
|
|
pget_info->param.stats.bigtk_micErrCnt =
|
|
wlan_le32_to_cpu(pget_log->bigtk_micErrCnt);
|
|
pget_info->param.stats.bigtk_mmeNotFoundCnt =
|
|
wlan_le32_to_cpu(pget_log->bigtk_mmeNotFoundCnt);
|
|
|
|
if (pmpriv->adapter->getlog_enable) {
|
|
pget_info->param.stats.tx_frag_cnt =
|
|
wlan_le32_to_cpu(pget_log->tx_frag_cnt);
|
|
for (i = 0; i < 8; i++) {
|
|
pget_info->param.stats.qos_tx_frag_cnt[i] =
|
|
wlan_le32_to_cpu(
|
|
pget_log->qos_tx_frag_cnt[i]);
|
|
pget_info->param.stats.qos_failed_cnt[i] =
|
|
wlan_le32_to_cpu(
|
|
pget_log->qos_failed_cnt[i]);
|
|
pget_info->param.stats.qos_retry_cnt[i] =
|
|
wlan_le32_to_cpu(
|
|
pget_log->qos_retry_cnt[i]);
|
|
pget_info->param.stats.qos_multi_retry_cnt[i] =
|
|
wlan_le32_to_cpu(
|
|
pget_log->qos_multi_retry_cnt[i]);
|
|
pget_info->param.stats.qos_frm_dup_cnt[i] =
|
|
wlan_le32_to_cpu(
|
|
pget_log->qos_frm_dup_cnt[i]);
|
|
pget_info->param.stats.qos_rts_suc_cnt[i] =
|
|
wlan_le32_to_cpu(
|
|
pget_log->qos_rts_suc_cnt[i]);
|
|
pget_info->param.stats.qos_rts_failure_cnt[i] =
|
|
wlan_le32_to_cpu(
|
|
pget_log->qos_rts_failure_cnt[i]);
|
|
pget_info->param.stats.qos_ack_failure_cnt[i] =
|
|
wlan_le32_to_cpu(
|
|
pget_log->qos_ack_failure_cnt[i]);
|
|
pget_info->param.stats.qos_rx_frag_cnt[i] =
|
|
wlan_le32_to_cpu(
|
|
pget_log->qos_rx_frag_cnt[i]);
|
|
pget_info->param.stats.qos_tx_frm_cnt[i] =
|
|
wlan_le32_to_cpu(
|
|
pget_log->qos_tx_frm_cnt[i]);
|
|
pget_info->param.stats.qos_discarded_frm_cnt
|
|
[i] = wlan_le32_to_cpu(
|
|
pget_log->qos_discarded_frm_cnt[i]);
|
|
pget_info->param.stats.qos_mpdus_rx_cnt[i] =
|
|
wlan_le32_to_cpu(
|
|
pget_log->qos_mpdus_rx_cnt[i]);
|
|
pget_info->param.stats.qos_retries_rx_cnt[i] =
|
|
wlan_le32_to_cpu(
|
|
pget_log->qos_retries_rx_cnt[i]);
|
|
}
|
|
pget_info->param.stats.mgmt_ccmp_replays =
|
|
wlan_le32_to_cpu(pget_log->mgmt_ccmp_replays);
|
|
pget_info->param.stats.tx_amsdu_cnt =
|
|
wlan_le32_to_cpu(pget_log->tx_amsdu_cnt);
|
|
pget_info->param.stats.failed_amsdu_cnt =
|
|
wlan_le32_to_cpu(pget_log->failed_amsdu_cnt);
|
|
pget_info->param.stats.retry_amsdu_cnt =
|
|
wlan_le32_to_cpu(pget_log->retry_amsdu_cnt);
|
|
pget_info->param.stats.multi_retry_amsdu_cnt =
|
|
wlan_le32_to_cpu(
|
|
pget_log->multi_retry_amsdu_cnt);
|
|
pget_info->param.stats.tx_octets_in_amsdu_cnt =
|
|
wlan_le64_to_cpu(
|
|
pget_log->tx_octets_in_amsdu_cnt);
|
|
pget_info->param.stats.amsdu_ack_failure_cnt =
|
|
wlan_le32_to_cpu(
|
|
pget_log->amsdu_ack_failure_cnt);
|
|
pget_info->param.stats.rx_amsdu_cnt =
|
|
wlan_le32_to_cpu(pget_log->rx_amsdu_cnt);
|
|
pget_info->param.stats.rx_octets_in_amsdu_cnt =
|
|
wlan_le64_to_cpu(
|
|
pget_log->rx_octets_in_amsdu_cnt);
|
|
pget_info->param.stats.tx_ampdu_cnt =
|
|
wlan_le32_to_cpu(pget_log->tx_ampdu_cnt);
|
|
pget_info->param.stats.tx_mpdus_in_ampdu_cnt =
|
|
wlan_le32_to_cpu(
|
|
pget_log->tx_mpdus_in_ampdu_cnt);
|
|
pget_info->param.stats.tx_octets_in_ampdu_cnt =
|
|
wlan_le64_to_cpu(
|
|
pget_log->tx_octets_in_ampdu_cnt);
|
|
pget_info->param.stats.ampdu_rx_cnt =
|
|
wlan_le32_to_cpu(pget_log->ampdu_rx_cnt);
|
|
pget_info->param.stats.mpdu_in_rx_ampdu_cnt =
|
|
wlan_le32_to_cpu(
|
|
pget_log->mpdu_in_rx_ampdu_cnt);
|
|
pget_info->param.stats.rx_octets_in_ampdu_cnt =
|
|
wlan_le64_to_cpu(
|
|
pget_log->rx_octets_in_ampdu_cnt);
|
|
pget_info->param.stats.ampdu_delimiter_crc_error_cnt =
|
|
wlan_le32_to_cpu(
|
|
pget_log->ampdu_delimiter_crc_error_cnt);
|
|
|
|
/* Indicate ioctl complete */
|
|
pioctl_buf->data_read_written =
|
|
sizeof(mlan_ds_get_info);
|
|
} else
|
|
pioctl_buf->data_read_written =
|
|
sizeof(mlan_ds_get_stats_org) +
|
|
sizeof(pget_info->sub_command);
|
|
}
|
|
LEAVE();
|
|
return MLAN_STATUS_SUCCESS;
|
|
}
|
|
|
|
/**
|
|
* @brief This function prepares command of deauth station
|
|
*
|
|
* @param pmpriv A pointer to mlan_private structure
|
|
* @param cmd A pointer to HostCmd_DS_COMMAND structure
|
|
* @param pdata_buf A pointer to data buffer
|
|
* @return MLAN_STATUS_SUCCESS
|
|
*/
|
|
static mlan_status wlan_uap_cmd_sta_deauth(pmlan_private pmpriv,
|
|
HostCmd_DS_COMMAND *cmd,
|
|
t_void *pdata_buf)
|
|
{
|
|
HostCmd_DS_STA_DEAUTH *pcmd_sta_deauth =
|
|
(HostCmd_DS_STA_DEAUTH *)&cmd->params.sta_deauth;
|
|
mlan_deauth_param *deauth = (mlan_deauth_param *)pdata_buf;
|
|
|
|
ENTER();
|
|
cmd->command = wlan_cpu_to_le16(HOST_CMD_APCMD_STA_DEAUTH);
|
|
cmd->size = wlan_cpu_to_le16(S_DS_GEN + sizeof(HostCmd_DS_STA_DEAUTH));
|
|
memcpy_ext(pmpriv->adapter, pcmd_sta_deauth->mac, deauth->mac_addr,
|
|
MLAN_MAC_ADDR_LENGTH, MLAN_MAC_ADDR_LENGTH);
|
|
pcmd_sta_deauth->reason = wlan_cpu_to_le16(deauth->reason_code);
|
|
LEAVE();
|
|
return MLAN_STATUS_SUCCESS;
|
|
}
|
|
|
|
/**
|
|
* @brief This function prepares command of report mic_err
|
|
*
|
|
* @param pmpriv A pointer to mlan_private structure
|
|
* @param cmd A pointer to HostCmd_DS_COMMAND structure
|
|
* @param pdata_buf A pointer to data buffer
|
|
* @return MLAN_STATUS_SUCCESS
|
|
*/
|
|
static mlan_status wlan_uap_cmd_report_mic(pmlan_private pmpriv,
|
|
HostCmd_DS_COMMAND *cmd,
|
|
t_void *pdata_buf)
|
|
{
|
|
HostCmd_DS_REPORT_MIC *pcmd_report_mic =
|
|
(HostCmd_DS_REPORT_MIC *)&cmd->params.report_mic;
|
|
|
|
ENTER();
|
|
cmd->command = wlan_cpu_to_le16(HOST_CMD_APCMD_REPORT_MIC);
|
|
cmd->size = wlan_cpu_to_le16(S_DS_GEN + sizeof(HostCmd_DS_REPORT_MIC));
|
|
memcpy_ext(pmpriv->adapter, pcmd_report_mic->mac, pdata_buf,
|
|
MLAN_MAC_ADDR_LENGTH, MLAN_MAC_ADDR_LENGTH);
|
|
LEAVE();
|
|
return MLAN_STATUS_SUCCESS;
|
|
}
|
|
|
|
/**
|
|
* @brief This function prepares command of key material
|
|
*
|
|
* @param pmpriv A pointer to mlan_private structure
|
|
* @param cmd A pointer to HostCmd_DS_COMMAND structure
|
|
* @param cmd_action The action: GET or SET
|
|
* @param cmd_oid OID: ENABLE or DISABLE
|
|
* @param pdata_buf A pointer to data buffer
|
|
* @return MLAN_STATUS_SUCCESS
|
|
*/
|
|
static mlan_status wlan_uap_cmd_key_material(pmlan_private pmpriv,
|
|
HostCmd_DS_COMMAND *cmd,
|
|
t_u16 cmd_action, t_u16 cmd_oid,
|
|
t_void *pdata_buf)
|
|
{
|
|
HostCmd_DS_802_11_KEY_MATERIAL *pkey_material =
|
|
&cmd->params.key_material;
|
|
mlan_ds_encrypt_key *pkey = (mlan_ds_encrypt_key *)pdata_buf;
|
|
mlan_status ret = MLAN_STATUS_SUCCESS;
|
|
sta_node *sta_ptr = MNULL;
|
|
|
|
ENTER();
|
|
if (!pkey) {
|
|
ret = MLAN_STATUS_FAILURE;
|
|
goto done;
|
|
}
|
|
cmd->command = wlan_cpu_to_le16(HostCmd_CMD_802_11_KEY_MATERIAL);
|
|
pkey_material->action = wlan_cpu_to_le16(cmd_action);
|
|
if (cmd_action == HostCmd_ACT_GEN_GET) {
|
|
cmd->size = wlan_cpu_to_le16(sizeof(pkey_material->action) +
|
|
S_DS_GEN);
|
|
goto done;
|
|
}
|
|
memset(pmpriv->adapter, &pkey_material->key_param_set, 0,
|
|
sizeof(MrvlIEtype_KeyParamSetV2_t));
|
|
if (pkey->key_flags & KEY_FLAG_REMOVE_KEY) {
|
|
pkey_material->action =
|
|
wlan_cpu_to_le16(HostCmd_ACT_GEN_REMOVE);
|
|
pkey_material->key_param_set.type =
|
|
wlan_cpu_to_le16(TLV_TYPE_KEY_PARAM_V2);
|
|
pkey_material->key_param_set.length =
|
|
wlan_cpu_to_le16(KEY_PARAMS_FIXED_LEN);
|
|
pkey_material->key_param_set.key_idx =
|
|
pkey->key_index & KEY_INDEX_MASK;
|
|
pkey_material->key_param_set.key_info = wlan_cpu_to_le16(
|
|
KEY_INFO_MCAST_KEY | KEY_INFO_UCAST_KEY);
|
|
memcpy_ext(pmpriv->adapter,
|
|
pkey_material->key_param_set.mac_addr,
|
|
pkey->mac_addr, MLAN_MAC_ADDR_LENGTH,
|
|
MLAN_MAC_ADDR_LENGTH);
|
|
cmd->size = wlan_cpu_to_le16(sizeof(MrvlIEtypesHeader_t) +
|
|
S_DS_GEN + KEY_PARAMS_FIXED_LEN +
|
|
sizeof(pkey_material->action));
|
|
PRINTM(MCMND, "Remove Key\n");
|
|
goto done;
|
|
}
|
|
pkey_material->action = wlan_cpu_to_le16(HostCmd_ACT_GEN_SET);
|
|
pkey_material->key_param_set.key_idx = pkey->key_index & KEY_INDEX_MASK;
|
|
pkey_material->key_param_set.type =
|
|
wlan_cpu_to_le16(TLV_TYPE_KEY_PARAM_V2);
|
|
pkey_material->key_param_set.key_info = KEY_INFO_ENABLE_KEY;
|
|
memcpy_ext(pmpriv->adapter, pkey_material->key_param_set.mac_addr,
|
|
pkey->mac_addr, MLAN_MAC_ADDR_LENGTH, MLAN_MAC_ADDR_LENGTH);
|
|
if (pkey->key_len <= MAX_WEP_KEY_SIZE) {
|
|
pkey_material->key_param_set.length = wlan_cpu_to_le16(
|
|
KEY_PARAMS_FIXED_LEN + sizeof(wep_param_t));
|
|
pkey_material->key_param_set.key_type = KEY_TYPE_ID_WEP;
|
|
pkey_material->key_param_set.key_info |=
|
|
KEY_INFO_MCAST_KEY | KEY_INFO_UCAST_KEY;
|
|
if (pkey_material->key_param_set.key_idx ==
|
|
(pmpriv->wep_key_curr_index & KEY_INDEX_MASK))
|
|
pkey_material->key_param_set.key_info |=
|
|
KEY_INFO_DEFAULT_KEY;
|
|
pkey_material->key_param_set.key_info =
|
|
wlan_cpu_to_le16(pkey_material->key_param_set.key_info);
|
|
pkey_material->key_param_set.key_params.wep.key_len =
|
|
wlan_cpu_to_le16(pkey->key_len);
|
|
memcpy_ext(pmpriv->adapter,
|
|
pkey_material->key_param_set.key_params.wep.key,
|
|
pkey->key_material, pkey->key_len, MAX_WEP_KEY_SIZE);
|
|
cmd->size = wlan_cpu_to_le16(sizeof(MrvlIEtypesHeader_t) +
|
|
S_DS_GEN + KEY_PARAMS_FIXED_LEN +
|
|
sizeof(wep_param_t) +
|
|
sizeof(pkey_material->action));
|
|
PRINTM(MCMND, "Set WEP Key\n");
|
|
goto done;
|
|
}
|
|
if (pkey->key_flags & KEY_FLAG_GROUP_KEY)
|
|
pkey_material->key_param_set.key_info |= KEY_INFO_MCAST_KEY;
|
|
else
|
|
pkey_material->key_param_set.key_info |= KEY_INFO_UCAST_KEY;
|
|
if (pkey->key_flags & KEY_FLAG_AES_MCAST_IGTK)
|
|
pkey_material->key_param_set.key_info |= KEY_INFO_CMAC_AES_KEY;
|
|
if (pkey->key_flags & KEY_FLAG_SET_TX_KEY)
|
|
pkey_material->key_param_set.key_info |=
|
|
KEY_INFO_TX_KEY | KEY_INFO_RX_KEY;
|
|
else
|
|
pkey_material->key_param_set.key_info |= KEY_INFO_TX_KEY;
|
|
if (pkey->is_wapi_key) {
|
|
pkey_material->key_param_set.key_type = KEY_TYPE_ID_WAPI;
|
|
memcpy_ext(pmpriv->adapter,
|
|
pkey_material->key_param_set.key_params.wapi.pn,
|
|
pkey->pn, PN_SIZE, PN_SIZE);
|
|
pkey_material->key_param_set.key_params.wapi.key_len =
|
|
wlan_cpu_to_le16(MIN(WAPI_KEY_SIZE, pkey->key_len));
|
|
memcpy_ext(pmpriv->adapter,
|
|
pkey_material->key_param_set.key_params.wapi.key,
|
|
pkey->key_material, pkey->key_len, WAPI_KEY_SIZE);
|
|
if (!pmpriv->sec_info.wapi_key_on)
|
|
pkey_material->key_param_set.key_info |=
|
|
KEY_INFO_DEFAULT_KEY;
|
|
if (pkey->key_flags & KEY_FLAG_GROUP_KEY) {
|
|
pmpriv->sec_info.wapi_key_on = MTRUE;
|
|
} else {
|
|
/* WAPI pairwise key: unicast */
|
|
sta_ptr =
|
|
wlan_add_station_entry(pmpriv, pkey->mac_addr);
|
|
if (sta_ptr) {
|
|
PRINTM(MCMND, "station: wapi_key_on\n");
|
|
sta_ptr->wapi_key_on = MTRUE;
|
|
}
|
|
}
|
|
pkey_material->key_param_set.key_info =
|
|
wlan_cpu_to_le16(pkey_material->key_param_set.key_info);
|
|
pkey_material->key_param_set.length = wlan_cpu_to_le16(
|
|
KEY_PARAMS_FIXED_LEN + sizeof(wapi_param));
|
|
cmd->size = wlan_cpu_to_le16(sizeof(MrvlIEtypesHeader_t) +
|
|
S_DS_GEN + KEY_PARAMS_FIXED_LEN +
|
|
sizeof(wapi_param) +
|
|
sizeof(pkey_material->action));
|
|
PRINTM(MCMND, "Set WAPI Key\n");
|
|
goto done;
|
|
}
|
|
pkey_material->key_param_set.key_info |= KEY_INFO_DEFAULT_KEY;
|
|
pkey_material->key_param_set.key_info =
|
|
wlan_cpu_to_le16(pkey_material->key_param_set.key_info);
|
|
if (pkey->key_len == WPA_AES_KEY_LEN &&
|
|
!(pkey->key_flags & KEY_FLAG_AES_MCAST_IGTK)) {
|
|
if (pkey->key_flags &
|
|
(KEY_FLAG_RX_SEQ_VALID | KEY_FLAG_TX_SEQ_VALID))
|
|
memcpy_ext(
|
|
pmpriv->adapter,
|
|
pkey_material->key_param_set.key_params.aes.pn,
|
|
pkey->pn, SEQ_MAX_SIZE, WPA_PN_SIZE);
|
|
pkey_material->key_param_set.key_type = KEY_TYPE_ID_AES;
|
|
pkey_material->key_param_set.key_params.aes.key_len =
|
|
wlan_cpu_to_le16(pkey->key_len);
|
|
memcpy_ext(pmpriv->adapter,
|
|
pkey_material->key_param_set.key_params.aes.key,
|
|
pkey->key_material, pkey->key_len, WPA_AES_KEY_LEN);
|
|
pkey_material->key_param_set.length = wlan_cpu_to_le16(
|
|
KEY_PARAMS_FIXED_LEN + sizeof(aes_param));
|
|
cmd->size = wlan_cpu_to_le16(sizeof(MrvlIEtypesHeader_t) +
|
|
S_DS_GEN + KEY_PARAMS_FIXED_LEN +
|
|
sizeof(aes_param) +
|
|
sizeof(pkey_material->action));
|
|
PRINTM(MCMND, "Set AES Key\n");
|
|
goto done;
|
|
}
|
|
if (pkey->key_len == WPA_IGTK_KEY_LEN &&
|
|
(pkey->key_flags & KEY_FLAG_AES_MCAST_IGTK)) {
|
|
if (pkey->key_flags &
|
|
(KEY_FLAG_RX_SEQ_VALID | KEY_FLAG_TX_SEQ_VALID))
|
|
memcpy_ext(pmpriv->adapter,
|
|
pkey_material->key_param_set.key_params
|
|
.cmac_aes.ipn,
|
|
pkey->pn, SEQ_MAX_SIZE, IGTK_PN_SIZE);
|
|
pkey_material->key_param_set.key_info &=
|
|
~(wlan_cpu_to_le16(KEY_INFO_MCAST_KEY));
|
|
pkey_material->key_param_set.key_info |=
|
|
wlan_cpu_to_le16(KEY_INFO_AES_MCAST_IGTK);
|
|
if (pkey->key_flags & KEY_FLAG_GMAC_128)
|
|
pkey_material->key_param_set.key_type =
|
|
KEY_TYPE_ID_BIP_GMAC_128;
|
|
else
|
|
pkey_material->key_param_set.key_type =
|
|
KEY_TYPE_ID_AES_CMAC;
|
|
pkey_material->key_param_set.key_params.cmac_aes.key_len =
|
|
wlan_cpu_to_le16(pkey->key_len);
|
|
memcpy_ext(pmpriv->adapter,
|
|
pkey_material->key_param_set.key_params.cmac_aes.key,
|
|
pkey->key_material, pkey->key_len, CMAC_AES_KEY_LEN);
|
|
pkey_material->key_param_set.length = wlan_cpu_to_le16(
|
|
KEY_PARAMS_FIXED_LEN + sizeof(cmac_aes_param));
|
|
cmd->size = wlan_cpu_to_le16(sizeof(MrvlIEtypesHeader_t) +
|
|
S_DS_GEN + KEY_PARAMS_FIXED_LEN +
|
|
sizeof(cmac_aes_param) +
|
|
sizeof(pkey_material->action));
|
|
if (pkey->key_flags & KEY_FLAG_GMAC_128)
|
|
PRINTM(MCMND, "Set AES 128 GMAC Key\n");
|
|
else
|
|
PRINTM(MCMND, "Set CMAC AES Key\n");
|
|
goto done;
|
|
}
|
|
if (pkey->key_len == WPA_IGTK_256_KEY_LEN &&
|
|
(pkey->key_flags & KEY_FLAG_AES_MCAST_IGTK)) {
|
|
if (pkey->key_flags &
|
|
(KEY_FLAG_RX_SEQ_VALID | KEY_FLAG_TX_SEQ_VALID))
|
|
memcpy_ext(pmpriv->adapter,
|
|
pkey_material->key_param_set.key_params
|
|
.cmac_aes.ipn,
|
|
pkey->pn, SEQ_MAX_SIZE, IGTK_PN_SIZE);
|
|
pkey_material->key_param_set.key_info &=
|
|
~(wlan_cpu_to_le16(KEY_INFO_MCAST_KEY));
|
|
pkey_material->key_param_set.key_info |=
|
|
wlan_cpu_to_le16(KEY_INFO_AES_MCAST_IGTK);
|
|
pkey_material->key_param_set.key_type =
|
|
KEY_TYPE_ID_BIP_GMAC_256;
|
|
pkey_material->key_param_set.key_params.cmac_aes.key_len =
|
|
wlan_cpu_to_le16(pkey->key_len);
|
|
memcpy_ext(pmpriv->adapter,
|
|
pkey_material->key_param_set.key_params.cmac_aes.key,
|
|
pkey->key_material, pkey->key_len,
|
|
WPA_IGTK_256_KEY_LEN);
|
|
pkey_material->key_param_set.length = wlan_cpu_to_le16(
|
|
KEY_PARAMS_FIXED_LEN + sizeof(gmac_aes_256_param));
|
|
cmd->size = wlan_cpu_to_le16(sizeof(MrvlIEtypesHeader_t) +
|
|
S_DS_GEN + KEY_PARAMS_FIXED_LEN +
|
|
sizeof(gmac_aes_256_param) +
|
|
sizeof(pkey_material->action));
|
|
PRINTM(MCMND, "Set AES 256 GMAC Key\n");
|
|
goto done;
|
|
}
|
|
if (pkey->key_len == WPA_TKIP_KEY_LEN) {
|
|
if (pkey->key_flags &
|
|
(KEY_FLAG_RX_SEQ_VALID | KEY_FLAG_TX_SEQ_VALID))
|
|
memcpy_ext(
|
|
pmpriv->adapter,
|
|
pkey_material->key_param_set.key_params.tkip.pn,
|
|
pkey->pn, SEQ_MAX_SIZE, WPA_PN_SIZE);
|
|
pkey_material->key_param_set.key_type = KEY_TYPE_ID_TKIP;
|
|
pkey_material->key_param_set.key_params.tkip.key_len =
|
|
wlan_cpu_to_le16(pkey->key_len);
|
|
memcpy_ext(pmpriv->adapter,
|
|
pkey_material->key_param_set.key_params.tkip.key,
|
|
pkey->key_material, pkey->key_len, WPA_TKIP_KEY_LEN);
|
|
pkey_material->key_param_set.length = wlan_cpu_to_le16(
|
|
KEY_PARAMS_FIXED_LEN + sizeof(tkip_param));
|
|
cmd->size = wlan_cpu_to_le16(sizeof(MrvlIEtypesHeader_t) +
|
|
S_DS_GEN + KEY_PARAMS_FIXED_LEN +
|
|
sizeof(tkip_param) +
|
|
sizeof(pkey_material->action));
|
|
PRINTM(MCMND, "Set TKIP Key\n");
|
|
}
|
|
done:
|
|
LEAVE();
|
|
return ret;
|
|
}
|
|
|
|
/**
|
|
* @brief This function handles the command response of sta_list
|
|
*
|
|
* @param pmpriv A pointer to mlan_private structure
|
|
* @param resp A pointer to HostCmd_DS_COMMAND
|
|
* @param pioctl_buf A pointer to mlan_ioctl_req structure
|
|
*
|
|
* @return MLAN_STATUS_SUCCESS
|
|
*/
|
|
static mlan_status wlan_uap_ret_sta_list(pmlan_private pmpriv,
|
|
HostCmd_DS_COMMAND *resp,
|
|
mlan_ioctl_req *pioctl_buf)
|
|
{
|
|
HostCmd_DS_STA_LIST *sta_list =
|
|
(HostCmd_DS_STA_LIST *)&resp->params.sta_list;
|
|
mlan_ds_get_info *info;
|
|
MrvlIEtypes_sta_info_t *tlv = MNULL;
|
|
t_u8 i = 0;
|
|
sta_node *sta_ptr;
|
|
t_u8 tlv_len = 0;
|
|
t_u8 *buf = MNULL;
|
|
|
|
ENTER();
|
|
if (pioctl_buf) {
|
|
info = (mlan_ds_get_info *)pioctl_buf->pbuf;
|
|
info->param.sta_list.sta_count =
|
|
wlan_le16_to_cpu(sta_list->sta_count);
|
|
buf = sta_list->tlv_buf;
|
|
tlv = (MrvlIEtypes_sta_info_t *)buf;
|
|
info->param.sta_list.sta_count =
|
|
MIN(info->param.sta_list.sta_count, MAX_NUM_CLIENTS);
|
|
pioctl_buf->data_read_written =
|
|
sizeof(info->param.sta_list.sta_count);
|
|
for (i = 0; i < info->param.sta_list.sta_count; i++) {
|
|
tlv_len = wlan_le16_to_cpu(tlv->header.len);
|
|
memcpy_ext(pmpriv->adapter,
|
|
info->param.sta_list.info[i].mac_address,
|
|
tlv->mac_address, MLAN_MAC_ADDR_LENGTH,
|
|
MLAN_MAC_ADDR_LENGTH);
|
|
info->param.sta_list.info[i].ie_len =
|
|
tlv_len + sizeof(MrvlIEtypesHeader_t) -
|
|
sizeof(MrvlIEtypes_sta_info_t);
|
|
if (info->param.sta_list.info[i].ie_len)
|
|
memcpy_ext(pmpriv->adapter,
|
|
info->param.sta_list.info[i].ie_buf,
|
|
tlv->ie_buf,
|
|
info->param.sta_list.info[i].ie_len,
|
|
info->param.sta_list.info[i].ie_len);
|
|
info->param.sta_list.info[i].power_mgmt_status =
|
|
tlv->power_mgmt_status;
|
|
info->param.sta_list.info[i].rssi = tlv->rssi;
|
|
sta_ptr = wlan_get_station_entry(pmpriv,
|
|
tlv->mac_address);
|
|
if (sta_ptr) {
|
|
info->param.sta_list.info[i].bandmode =
|
|
sta_ptr->bandmode;
|
|
memcpy_ext(
|
|
pmpriv->adapter,
|
|
&(info->param.sta_list.info[i].stats),
|
|
&(sta_ptr->stats), sizeof(sta_stats),
|
|
sizeof(sta_stats));
|
|
} else
|
|
info->param.sta_list.info[i].bandmode = 0xFF;
|
|
pioctl_buf->data_read_written +=
|
|
sizeof(sta_info_data) - 1 +
|
|
info->param.sta_list.info[i].ie_len;
|
|
buf += sizeof(MrvlIEtypes_sta_info_t) +
|
|
info->param.sta_list.info[i].ie_len;
|
|
tlv = (MrvlIEtypes_sta_info_t *)buf;
|
|
}
|
|
PRINTM(MCMND, "Total sta_list size=%d\n",
|
|
pioctl_buf->data_read_written);
|
|
}
|
|
|
|
LEAVE();
|
|
return MLAN_STATUS_SUCCESS;
|
|
}
|
|
|
|
/** Fixed size of bss start event */
|
|
#define BSS_START_EVENT_FIX_SIZE 12
|
|
|
|
/**
|
|
* @brief This function will search for the specific ie
|
|
*
|
|
* @param priv A pointer to mlan_private
|
|
* @param pevent A pointer to event buf
|
|
*
|
|
* @return N/A
|
|
*/
|
|
static void wlan_check_uap_capability(pmlan_private priv, pmlan_buffer pevent)
|
|
{
|
|
t_u16 tlv_type, tlv_len;
|
|
int tlv_buf_left = pevent->data_len - BSS_START_EVENT_FIX_SIZE;
|
|
MrvlIEtypesHeader_t *tlv =
|
|
(MrvlIEtypesHeader_t *)(pevent->pbuf + pevent->data_offset +
|
|
BSS_START_EVENT_FIX_SIZE);
|
|
|
|
const t_u8 wmm_oui[4] = {0x00, 0x50, 0xf2, 0x02};
|
|
IEEEtypes_WmmParameter_t wmm_param_ie;
|
|
MrvlIEtypes_channel_band_t *pchan_info;
|
|
t_u8 event_buf[100];
|
|
mlan_event *event = (mlan_event *)event_buf;
|
|
chan_band_info *pchan_band_info = (chan_band_info *)event->event_buf;
|
|
MrvlIEtypes_He_cap_t *pext_tlv = MNULL;
|
|
|
|
ENTER();
|
|
priv->wmm_enabled = MFALSE;
|
|
priv->pkt_fwd = MFALSE;
|
|
priv->is_11n_enabled = MFALSE;
|
|
priv->is_11ac_enabled = MFALSE;
|
|
priv->is_11ax_enabled = MFALSE;
|
|
|
|
while (tlv_buf_left >= (int)sizeof(MrvlIEtypesHeader_t)) {
|
|
tlv_type = wlan_le16_to_cpu(tlv->type);
|
|
tlv_len = wlan_le16_to_cpu(tlv->len);
|
|
if ((sizeof(MrvlIEtypesHeader_t) + tlv_len) >
|
|
(unsigned int)tlv_buf_left) {
|
|
PRINTM(MERROR, "wrong tlv: tlvLen=%d, tlvBufLeft=%d\n",
|
|
tlv_len, tlv_buf_left);
|
|
break;
|
|
}
|
|
if (tlv_type == HT_CAPABILITY) {
|
|
DBG_HEXDUMP(MCMD_D, "HT_CAP tlv", tlv,
|
|
tlv_len + sizeof(MrvlIEtypesHeader_t));
|
|
priv->is_11n_enabled = MTRUE;
|
|
}
|
|
if (tlv_type == VHT_CAPABILITY) {
|
|
DBG_HEXDUMP(MCMD_D, "VHT_CAP tlv", tlv,
|
|
tlv_len + sizeof(MrvlIEtypesHeader_t));
|
|
priv->is_11ac_enabled = MTRUE;
|
|
}
|
|
if (tlv_type == EXTENSION) {
|
|
pext_tlv = (MrvlIEtypes_He_cap_t *)tlv;
|
|
if (pext_tlv->ext_id == HE_CAPABILITY) {
|
|
DBG_HEXDUMP(
|
|
MCMD_D, "HE_CAP tlv", tlv,
|
|
tlv_len + sizeof(MrvlIEtypesHeader_t));
|
|
priv->is_11ax_enabled = MTRUE;
|
|
}
|
|
}
|
|
|
|
if (tlv_type == VENDOR_SPECIFIC_221) {
|
|
if (!memcmp(priv->adapter,
|
|
(t_u8 *)tlv + sizeof(MrvlIEtypesHeader_t),
|
|
wmm_oui, sizeof(wmm_oui))) {
|
|
DBG_HEXDUMP(
|
|
MCMD_D, "wmm ie tlv", tlv,
|
|
tlv_len + sizeof(MrvlIEtypesHeader_t));
|
|
priv->wmm_enabled = MFALSE;
|
|
wlan_wmm_setup_ac_downgrade(priv);
|
|
priv->wmm_enabled = MTRUE;
|
|
memcpy_ext(priv->adapter, &wmm_param_ie,
|
|
((t_u8 *)tlv + 2),
|
|
sizeof(IEEEtypes_WmmParameter_t),
|
|
sizeof(IEEEtypes_WmmParameter_t));
|
|
wmm_param_ie.vend_hdr.len = (t_u8)tlv_len;
|
|
wmm_param_ie.vend_hdr.element_id = WMM_IE;
|
|
wlan_wmm_setup_queue_priorities(priv,
|
|
&wmm_param_ie);
|
|
}
|
|
}
|
|
if (tlv_type == TLV_TYPE_UAP_PKT_FWD_CTL) {
|
|
DBG_HEXDUMP(MCMD_D, "pkt_fwd tlv", tlv,
|
|
tlv_len + sizeof(MrvlIEtypesHeader_t));
|
|
priv->pkt_fwd =
|
|
*((t_u8 *)tlv + sizeof(MrvlIEtypesHeader_t));
|
|
PRINTM(MCMND, "pkt_fwd FW: 0x%x\n", priv->pkt_fwd);
|
|
if (priv->pkt_fwd & PKT_FWD_FW_BIT)
|
|
priv->pkt_fwd = MFALSE;
|
|
else
|
|
priv->pkt_fwd |= PKT_FWD_ENABLE_BIT;
|
|
PRINTM(MCMND, "pkt_fwd DRV: 0x%x\n", priv->pkt_fwd);
|
|
}
|
|
if (tlv_type == TLV_TYPE_UAP_CHAN_BAND_CONFIG) {
|
|
DBG_HEXDUMP(MCMD_D, "chan_band_config tlv", tlv,
|
|
tlv_len + sizeof(MrvlIEtypesHeader_t));
|
|
pchan_info = (MrvlIEtypes_channel_band_t *)tlv;
|
|
priv->uap_channel = pchan_info->channel;
|
|
PRINTM(MCMND, "uap_channel FW: 0x%x\n",
|
|
priv->uap_channel);
|
|
event->bss_index = priv->bss_index;
|
|
event->event_id = MLAN_EVENT_ID_DRV_UAP_CHAN_INFO;
|
|
event->event_len = sizeof(chan_band_info);
|
|
memcpy_ext(priv->adapter,
|
|
(t_u8 *)&pchan_band_info->bandcfg,
|
|
(t_u8 *)&pchan_info->bandcfg, tlv_len,
|
|
tlv_len);
|
|
if (pchan_band_info->bandcfg.chanWidth == CHAN_BW_80MHZ)
|
|
pchan_band_info->center_chan =
|
|
wlan_get_center_freq_idx(
|
|
priv, BAND_AAC,
|
|
pchan_info->channel,
|
|
CHANNEL_BW_80MHZ);
|
|
if (priv->adapter->ecsa_enable) {
|
|
int ret;
|
|
t_u8 bandwidth = BW_20MHZ;
|
|
|
|
MrvlIEtypes_chan_bw_oper_t chan_bw_oper;
|
|
chan_bw_oper.header.type = REGULATORY_CLASS;
|
|
chan_bw_oper.header.len =
|
|
sizeof(MrvlIEtypes_chan_bw_oper_t);
|
|
chan_bw_oper.ds_chan_bw_oper.channel =
|
|
pchan_info->channel;
|
|
|
|
if (pchan_band_info->bandcfg.chanWidth ==
|
|
CHAN_BW_40MHZ)
|
|
bandwidth = BW_40MHZ;
|
|
else if (pchan_band_info->bandcfg.chanWidth ==
|
|
CHAN_BW_80MHZ)
|
|
bandwidth = BW_80MHZ;
|
|
chan_bw_oper.ds_chan_bw_oper.bandwidth =
|
|
bandwidth;
|
|
|
|
ret = wlan_prepare_cmd(
|
|
priv, HOST_CMD_APCMD_SYS_CONFIGURE,
|
|
HostCmd_ACT_GEN_SET, 0, MNULL,
|
|
&chan_bw_oper);
|
|
if (ret != MLAN_STATUS_SUCCESS &&
|
|
ret != MLAN_STATUS_PENDING) {
|
|
PRINTM(MERROR,
|
|
"%s(): Could not set supported operating class IE for priv=%p [priv_bss_idx=%d]!\n",
|
|
__func__, priv, priv->bss_index);
|
|
}
|
|
}
|
|
}
|
|
|
|
tlv_buf_left -= (sizeof(MrvlIEtypesHeader_t) + tlv_len);
|
|
tlv = (MrvlIEtypesHeader_t *)((t_u8 *)tlv + tlv_len +
|
|
sizeof(MrvlIEtypesHeader_t));
|
|
}
|
|
if (priv->wmm_enabled == MFALSE) {
|
|
/* Since WMM is not enabled, setup the queues with the defaults
|
|
*/
|
|
wlan_wmm_setup_queues(priv);
|
|
}
|
|
if (event->event_id == MLAN_EVENT_ID_DRV_UAP_CHAN_INFO) {
|
|
pchan_band_info->is_11n_enabled = priv->is_11n_enabled;
|
|
wlan_recv_event(priv, MLAN_EVENT_ID_DRV_UAP_CHAN_INFO, event);
|
|
}
|
|
|
|
LEAVE();
|
|
}
|
|
|
|
/**
|
|
* @brief This function will update WAPI PN in statation assoc event
|
|
*
|
|
* @param priv A pointer to mlan_private
|
|
* @param pevent A pointer to event buf
|
|
*
|
|
* @return MFALSE
|
|
*/
|
|
static t_u32 wlan_update_wapi_info_tlv(pmlan_private priv, pmlan_buffer pevent)
|
|
{
|
|
t_u32 ret = MFALSE;
|
|
t_u16 tlv_type, tlv_len;
|
|
t_u32 tx_pn[4];
|
|
t_u32 i = 0;
|
|
int tlv_buf_left = pevent->data_len - ASSOC_EVENT_FIX_SIZE;
|
|
MrvlIEtypesHeader_t *tlv =
|
|
(MrvlIEtypesHeader_t *)(pevent->pbuf + pevent->data_offset +
|
|
ASSOC_EVENT_FIX_SIZE);
|
|
MrvlIEtypes_wapi_info_t *wapi_tlv = MNULL;
|
|
|
|
ENTER();
|
|
while (tlv_buf_left >= (int)sizeof(MrvlIEtypesHeader_t)) {
|
|
tlv_type = wlan_le16_to_cpu(tlv->type);
|
|
tlv_len = wlan_le16_to_cpu(tlv->len);
|
|
if ((sizeof(MrvlIEtypesHeader_t) + tlv_len) >
|
|
(unsigned int)tlv_buf_left) {
|
|
PRINTM(MERROR, "wrong tlv: tlvLen=%d, tlvBufLeft=%d\n",
|
|
tlv_len, tlv_buf_left);
|
|
break;
|
|
}
|
|
if (tlv_type == TLV_TYPE_AP_WAPI_INFO) {
|
|
wapi_tlv = (MrvlIEtypes_wapi_info_t *)tlv;
|
|
DBG_HEXDUMP(MCMD_D, "Fw:multicast_PN",
|
|
wapi_tlv->multicast_PN, PN_SIZE);
|
|
memcpy_ext(priv->adapter, (t_u8 *)tx_pn,
|
|
wapi_tlv->multicast_PN, PN_SIZE,
|
|
sizeof(tx_pn));
|
|
for (i = 0; i < 4; i++)
|
|
tx_pn[i] = mlan_ntohl(tx_pn[i]);
|
|
memcpy_ext(priv->adapter, wapi_tlv->multicast_PN,
|
|
(t_u8 *)tx_pn, PN_SIZE,
|
|
sizeof(wapi_tlv->multicast_PN));
|
|
DBG_HEXDUMP(MCMD_D, "Host:multicast_PN",
|
|
wapi_tlv->multicast_PN, PN_SIZE);
|
|
break;
|
|
}
|
|
tlv_buf_left -= (sizeof(MrvlIEtypesHeader_t) + tlv_len);
|
|
tlv = (MrvlIEtypesHeader_t *)((t_u8 *)tlv + tlv_len +
|
|
sizeof(MrvlIEtypesHeader_t));
|
|
}
|
|
LEAVE();
|
|
|
|
return ret;
|
|
}
|
|
|
|
/**
|
|
* @brief This function send sta_assoc_event to moal
|
|
* payload with sta mac address and assoc ie.
|
|
*
|
|
* @param priv A pointer to mlan_private
|
|
* @param pevent A pointer to mlan_event buffer
|
|
* @param pbuf A pointer to mlan_buffer which has event content.
|
|
*
|
|
* @return MFALSE
|
|
*/
|
|
static t_u32 wlan_process_sta_assoc_event(pmlan_private priv,
|
|
mlan_event *pevent,
|
|
pmlan_buffer pmbuf)
|
|
{
|
|
t_u32 ret = MFALSE;
|
|
t_u16 tlv_type, tlv_len;
|
|
t_u16 frame_control, frame_sub_type = 0;
|
|
t_u8 *assoc_req_ie = MNULL;
|
|
t_u8 ie_len = 0, assoc_ie_len = 0;
|
|
int tlv_buf_left = pmbuf->data_len - ASSOC_EVENT_FIX_SIZE;
|
|
MrvlIEtypesHeader_t *tlv =
|
|
(MrvlIEtypesHeader_t *)(pmbuf->pbuf + pmbuf->data_offset +
|
|
ASSOC_EVENT_FIX_SIZE);
|
|
MrvlIETypes_MgmtFrameSet_t *mgmt_tlv = MNULL;
|
|
|
|
ENTER();
|
|
pevent->event_id = MLAN_EVENT_ID_UAP_FW_STA_CONNECT;
|
|
pevent->bss_index = priv->bss_index;
|
|
pevent->event_len = MLAN_MAC_ADDR_LENGTH;
|
|
memcpy_ext(priv->adapter, pevent->event_buf,
|
|
pmbuf->pbuf + pmbuf->data_offset + 6, pevent->event_len,
|
|
pevent->event_len);
|
|
while (tlv_buf_left >= (int)sizeof(MrvlIEtypesHeader_t)) {
|
|
tlv_type = wlan_le16_to_cpu(tlv->type);
|
|
tlv_len = wlan_le16_to_cpu(tlv->len);
|
|
if ((sizeof(MrvlIEtypesHeader_t) + tlv_len) >
|
|
(unsigned int)tlv_buf_left) {
|
|
PRINTM(MERROR, "wrong tlv: tlvLen=%d, tlvBufLeft=%d\n",
|
|
tlv_len, tlv_buf_left);
|
|
break;
|
|
}
|
|
if (tlv_type == TLV_TYPE_MGMT_FRAME) {
|
|
mgmt_tlv = (MrvlIETypes_MgmtFrameSet_t *)tlv;
|
|
memcpy_ext(priv->adapter, &frame_control,
|
|
(t_u8 *)&(mgmt_tlv->frame_control),
|
|
sizeof(frame_control),
|
|
sizeof(frame_control));
|
|
frame_sub_type = IEEE80211_GET_FC_MGMT_FRAME_SUBTYPE(
|
|
frame_control);
|
|
if ((mgmt_tlv->frame_control.type == 0) &&
|
|
((frame_sub_type == SUBTYPE_ASSOC_REQUEST) ||
|
|
(frame_sub_type == SUBTYPE_REASSOC_REQUEST))) {
|
|
if (frame_sub_type == SUBTYPE_ASSOC_REQUEST)
|
|
assoc_ie_len =
|
|
sizeof(IEEEtypes_AssocRqst_t);
|
|
else if (frame_sub_type ==
|
|
SUBTYPE_REASSOC_REQUEST)
|
|
assoc_ie_len =
|
|
sizeof(IEEEtypes_ReAssocRqst_t);
|
|
|
|
ie_len = tlv_len -
|
|
sizeof(IEEEtypes_FrameCtl_t) -
|
|
assoc_ie_len;
|
|
assoc_req_ie =
|
|
(t_u8 *)tlv +
|
|
sizeof(MrvlIETypes_MgmtFrameSet_t) +
|
|
assoc_ie_len;
|
|
memcpy_ext(priv->adapter,
|
|
pevent->event_buf +
|
|
pevent->event_len,
|
|
assoc_req_ie, ie_len, ie_len);
|
|
pevent->event_len += ie_len;
|
|
break;
|
|
}
|
|
}
|
|
tlv_buf_left -= (sizeof(MrvlIEtypesHeader_t) + tlv_len);
|
|
tlv = (MrvlIEtypesHeader_t *)((t_u8 *)tlv + tlv_len +
|
|
sizeof(MrvlIEtypesHeader_t));
|
|
}
|
|
PRINTM(MEVENT, "STA assoc event len=%d\n", pevent->event_len);
|
|
DBG_HEXDUMP(MCMD_D, "STA assoc event", pevent->event_buf,
|
|
pevent->event_len);
|
|
wlan_recv_event(priv, pevent->event_id, pevent);
|
|
LEAVE();
|
|
return ret;
|
|
}
|
|
|
|
/**
|
|
* @brief This function handles repsonse of acs_scan.
|
|
*
|
|
* @param pmpriv A pointer to mlan_private structure
|
|
* @param pcmd A pointer to HostCmd_DS_COMMAND structure
|
|
* @param pioctl_buf A pointer to mlan_ioctl_req structure
|
|
*
|
|
* @return MLAN_STATUS_SUCCESS
|
|
*/
|
|
static mlan_status wlan_ret_cmd_uap_acs_scan(pmlan_private pmpriv,
|
|
const HostCmd_DS_COMMAND *resp,
|
|
mlan_ioctl_req *pioctl_buf)
|
|
{
|
|
HostCMD_DS_APCMD_ACS_SCAN *acs_scan =
|
|
(HostCMD_DS_APCMD_ACS_SCAN *)&resp->params.acs_scan;
|
|
mlan_ds_bss *bss = MNULL;
|
|
|
|
ENTER();
|
|
PRINTM(MCMND, "ACS scan done: bandcfg=%x, channel=%d\n",
|
|
acs_scan->bandcfg, acs_scan->chan);
|
|
if (pioctl_buf) {
|
|
bss = (mlan_ds_bss *)pioctl_buf->pbuf;
|
|
bss->param.ap_acs_scan.chan = acs_scan->chan;
|
|
bss->param.ap_acs_scan.bandcfg = acs_scan->bandcfg;
|
|
pioctl_buf->data_read_written = sizeof(mlan_ds_bss);
|
|
}
|
|
|
|
LEAVE();
|
|
return MLAN_STATUS_SUCCESS;
|
|
}
|
|
|
|
/**
|
|
* @brief This function prepares command of uap operation control
|
|
*
|
|
* @param pmpriv A pointer to mlan_private structure
|
|
* @param cmd A pointer to HostCmd_DS_COMMAND structure
|
|
* @param cmd_action the action: GET or SET
|
|
* @param pdata_buf A pointer to data buffer
|
|
* @return MLAN_STATUS_SUCCESS
|
|
*/
|
|
static mlan_status wlan_uap_cmd_oper_ctrl(pmlan_private pmpriv,
|
|
HostCmd_DS_COMMAND *cmd,
|
|
t_u16 cmd_action, t_void *pdata_buf)
|
|
{
|
|
HostCmd_DS_UAP_OPER_CTRL *poper_ctl =
|
|
(HostCmd_DS_UAP_OPER_CTRL *)&cmd->params.uap_oper_ctrl;
|
|
mlan_ds_bss *bss = (mlan_ds_bss *)pdata_buf;
|
|
mlan_uap_oper_ctrl *uap_oper_ctrl = &bss->param.ap_oper_ctrl;
|
|
Band_Config_t *bandcfg = MNULL;
|
|
|
|
ENTER();
|
|
|
|
cmd->command = wlan_cpu_to_le16(HOST_CMD_APCMD_OPER_CTRL);
|
|
cmd->size =
|
|
wlan_cpu_to_le16(sizeof(HostCmd_DS_UAP_OPER_CTRL) + S_DS_GEN);
|
|
poper_ctl->action = wlan_cpu_to_le16(cmd_action);
|
|
|
|
if (cmd_action == HostCmd_ACT_GEN_SET) {
|
|
poper_ctl->ctrl = wlan_cpu_to_le16(uap_oper_ctrl->ctrl_value);
|
|
if (uap_oper_ctrl->ctrl_value == 2) {
|
|
poper_ctl->chan_opt =
|
|
wlan_cpu_to_le16(uap_oper_ctrl->chan_opt);
|
|
if (uap_oper_ctrl->chan_opt == 3) {
|
|
poper_ctl->channel_band.header.type =
|
|
wlan_cpu_to_le16(
|
|
TLV_TYPE_UAP_CHAN_BAND_CONFIG);
|
|
poper_ctl->channel_band.header
|
|
.len = wlan_cpu_to_le16(
|
|
sizeof(MrvlIEtypes_channel_band_t) -
|
|
sizeof(MrvlIEtypesHeader_t));
|
|
bandcfg = &poper_ctl->channel_band.bandcfg;
|
|
if (uap_oper_ctrl->channel > 14)
|
|
bandcfg->chanBand = BAND_5GHZ;
|
|
bandcfg->chanWidth = uap_oper_ctrl->band_cfg;
|
|
if (bandcfg->chanWidth)
|
|
bandcfg->chan2Offset =
|
|
wlan_get_second_channel_offset(
|
|
uap_oper_ctrl->channel);
|
|
bandcfg->scanMode = SCAN_MODE_MANUAL;
|
|
poper_ctl->channel_band.channel =
|
|
uap_oper_ctrl->channel;
|
|
}
|
|
}
|
|
}
|
|
|
|
LEAVE();
|
|
return MLAN_STATUS_SUCCESS;
|
|
}
|
|
|
|
/**
|
|
* @brief This function handles the command response of uap operation control
|
|
*
|
|
* @param pmpriv A pointer to mlan_private structure
|
|
* @param resp A pointer to HostCmd_DS_COMMAND
|
|
* @param pioctl_buf A pointer to mlan_ioctl_req structure
|
|
*
|
|
* @return MLAN_STATUS_SUCCESS
|
|
*/
|
|
static mlan_status wlan_uap_ret_oper_ctrl(pmlan_private pmpriv,
|
|
HostCmd_DS_COMMAND *resp,
|
|
mlan_ioctl_req *pioctl_buf)
|
|
{
|
|
HostCmd_DS_UAP_OPER_CTRL *poper_ctl =
|
|
(HostCmd_DS_UAP_OPER_CTRL *)&resp->params.uap_oper_ctrl;
|
|
mlan_ds_bss *bss = MNULL;
|
|
mlan_uap_oper_ctrl *uap_oper_ctrl = MNULL;
|
|
Band_Config_t *bandcfg = MNULL;
|
|
|
|
ENTER();
|
|
|
|
if (pioctl_buf && pioctl_buf->action == MLAN_ACT_GET) {
|
|
bss = (mlan_ds_bss *)pioctl_buf->pbuf;
|
|
uap_oper_ctrl = (mlan_uap_oper_ctrl *)&bss->param.ap_oper_ctrl;
|
|
uap_oper_ctrl->ctrl_value = wlan_le16_to_cpu(poper_ctl->ctrl);
|
|
uap_oper_ctrl->chan_opt = wlan_le16_to_cpu(poper_ctl->chan_opt);
|
|
uap_oper_ctrl->channel = poper_ctl->channel_band.channel;
|
|
bandcfg = &poper_ctl->channel_band.bandcfg;
|
|
uap_oper_ctrl->band_cfg = bandcfg->chanWidth;
|
|
}
|
|
|
|
LEAVE();
|
|
return MLAN_STATUS_SUCCESS;
|
|
}
|
|
|
|
/**
|
|
* @brief Check 11B support Rates
|
|
*
|
|
*
|
|
* @param pmadapter Private mlan adapter structure
|
|
*
|
|
* @return MTRUE/MFALSE
|
|
*
|
|
*/
|
|
static t_u8
|
|
wlan_check_11B_support_rates(MrvlIEtypes_RatesParamSet_t *prates_tlv)
|
|
{
|
|
int i;
|
|
t_u8 rate;
|
|
t_u8 ret = MTRUE;
|
|
for (i = 0; i < prates_tlv->header.len; i++) {
|
|
rate = prates_tlv->rates[i] & 0x7f;
|
|
if ((rate != 0x02) && (rate != 0x04) && (rate != 0x0b) &&
|
|
(rate != 0x16)) {
|
|
ret = MFALSE;
|
|
break;
|
|
}
|
|
}
|
|
return ret;
|
|
}
|
|
|
|
/**
|
|
* @brief This function prepares command of sys_config
|
|
*
|
|
* @param pmpriv A pointer to mlan_private structure
|
|
* @param cmd A pointer to HostCmd_DS_COMMAND structure
|
|
* @param cmd_action cmd action
|
|
* @param pioctl_buf A pointer to mlan_ioctl_req structure
|
|
* @return MLAN_STATUS_SUCCESS or MLAN_STATUS_FAILURE
|
|
*/
|
|
static mlan_status wlan_uap_cmd_add_station(pmlan_private pmpriv,
|
|
HostCmd_DS_COMMAND *cmd,
|
|
t_u16 cmd_action,
|
|
pmlan_ioctl_req pioctl_buf)
|
|
{
|
|
mlan_ds_bss *bss = MNULL;
|
|
HostCmd_DS_ADD_STATION *new_sta =
|
|
(HostCmd_DS_ADD_STATION *)&cmd->params.sta_info;
|
|
sta_node *sta_ptr = MNULL;
|
|
t_u16 tlv_buf_left;
|
|
t_u8 *pos = MNULL;
|
|
t_u8 *tlv_buf = MNULL;
|
|
t_u16 travel_len = 0;
|
|
MrvlIEtypesHeader_t *tlv;
|
|
t_u16 tlv_len = 0;
|
|
t_u8 b_only = MFALSE;
|
|
mlan_adapter *pmadapter = pmpriv->adapter;
|
|
MrvlIETypes_HTCap_t *phtcap;
|
|
MrvlIETypes_VHTCap_t *pvhtcap;
|
|
MrvlIEtypes_Extension_t *pext_tlv = MNULL;
|
|
MrvlIEtypes_StaFlag_t *pstaflag;
|
|
int i;
|
|
|
|
ENTER();
|
|
|
|
if (!pioctl_buf) {
|
|
LEAVE();
|
|
return MLAN_STATUS_FAILURE;
|
|
}
|
|
bss = (mlan_ds_bss *)pioctl_buf->pbuf;
|
|
cmd->command = wlan_cpu_to_le16(HostCmd_CMD_ADD_NEW_STATION);
|
|
new_sta->action = wlan_cpu_to_le16(cmd_action);
|
|
cmd->size = sizeof(HostCmd_DS_ADD_STATION) + S_DS_GEN;
|
|
if (cmd_action == HostCmd_ACT_ADD_STA) {
|
|
sta_ptr = wlan_get_station_entry(pmpriv,
|
|
bss->param.sta_info.peer_mac);
|
|
if (!sta_ptr)
|
|
sta_ptr = wlan_add_station_entry(
|
|
pmpriv, bss->param.sta_info.peer_mac);
|
|
} else {
|
|
sta_ptr = wlan_add_station_entry(pmpriv,
|
|
bss->param.sta_info.peer_mac);
|
|
}
|
|
if (!sta_ptr) {
|
|
LEAVE();
|
|
return MLAN_STATUS_FAILURE;
|
|
}
|
|
memcpy_ext(pmadapter, new_sta->peer_mac, bss->param.sta_info.peer_mac,
|
|
MLAN_MAC_ADDR_LENGTH, MLAN_MAC_ADDR_LENGTH);
|
|
if (cmd_action != HostCmd_ACT_ADD_STA)
|
|
goto done;
|
|
new_sta->aid = wlan_cpu_to_le16(bss->param.sta_info.aid);
|
|
new_sta->listen_interval =
|
|
wlan_cpu_to_le32(bss->param.sta_info.listen_interval);
|
|
if (bss->param.sta_info.cap_info)
|
|
new_sta->cap_info =
|
|
wlan_cpu_to_le16(bss->param.sta_info.cap_info);
|
|
else
|
|
new_sta->cap_info = wlan_cpu_to_le16(sta_ptr->capability);
|
|
tlv_buf_left = bss->param.sta_info.tlv_len;
|
|
pos = new_sta->tlv;
|
|
tlv_buf = bss->param.sta_info.tlv;
|
|
tlv = (MrvlIEtypesHeader_t *)tlv_buf;
|
|
if (bss->param.sta_info.sta_flags & STA_FLAG_WME) {
|
|
PRINTM(MCMND, "STA flags supports wmm \n");
|
|
sta_ptr->is_wmm_enabled = MTRUE;
|
|
}
|
|
// append sta_flag_flags.
|
|
pstaflag = (MrvlIEtypes_StaFlag_t *)pos;
|
|
pstaflag->header.type = wlan_cpu_to_le16(TLV_TYPE_UAP_STA_FLAGS);
|
|
pstaflag->header.len = wlan_cpu_to_le16(sizeof(t_u32));
|
|
pstaflag->sta_flags = wlan_cpu_to_le32(bss->param.sta_info.sta_flags);
|
|
pos += sizeof(MrvlIEtypes_StaFlag_t);
|
|
cmd->size += sizeof(MrvlIEtypes_StaFlag_t);
|
|
|
|
while (tlv_buf_left >= sizeof(MrvlIEtypesHeader_t)) {
|
|
if (tlv_buf_left < (sizeof(MrvlIEtypesHeader_t) + tlv->len))
|
|
break;
|
|
switch (tlv->type) {
|
|
case EXT_CAPABILITY:
|
|
break;
|
|
case SUPPORTED_RATES:
|
|
b_only = wlan_check_11B_support_rates(
|
|
(MrvlIEtypes_RatesParamSet_t *)tlv);
|
|
break;
|
|
case QOS_INFO:
|
|
PRINTM(MCMND, "STA supports wmm\n");
|
|
sta_ptr->is_wmm_enabled = MTRUE;
|
|
break;
|
|
case HT_CAPABILITY:
|
|
PRINTM(MCMND, "STA supports 11n\n");
|
|
sta_ptr->is_11n_enabled = MTRUE;
|
|
phtcap = (MrvlIETypes_HTCap_t *)tlv;
|
|
if (sta_ptr->HTcap.ieee_hdr.element_id ==
|
|
HT_CAPABILITY) {
|
|
if (GETHT_40MHZ_INTOLARANT(
|
|
sta_ptr->HTcap.ht_cap.ht_cap_info)) {
|
|
PRINTM(MCMND,
|
|
"SETHT_40MHZ_INTOLARANT\n");
|
|
SETHT_40MHZ_INTOLARANT(
|
|
phtcap->ht_cap.ht_cap_info);
|
|
}
|
|
}
|
|
if (GETHT_MAXAMSDU(phtcap->ht_cap.ht_cap_info))
|
|
sta_ptr->max_amsdu = MLAN_TX_DATA_BUF_SIZE_8K;
|
|
else
|
|
sta_ptr->max_amsdu = MLAN_TX_DATA_BUF_SIZE_4K;
|
|
break;
|
|
case VHT_CAPABILITY:
|
|
PRINTM(MCMND, "STA supports 11ac\n");
|
|
sta_ptr->is_11ac_enabled = MTRUE;
|
|
pvhtcap = (MrvlIETypes_VHTCap_t *)tlv;
|
|
if (GET_VHTCAP_MAXMPDULEN(
|
|
pvhtcap->vht_cap.vht_cap_info) == 2)
|
|
sta_ptr->max_amsdu = MLAN_TX_DATA_BUF_SIZE_12K;
|
|
else if (GET_VHTCAP_MAXMPDULEN(
|
|
pvhtcap->vht_cap.vht_cap_info) == 1)
|
|
sta_ptr->max_amsdu = MLAN_TX_DATA_BUF_SIZE_8K;
|
|
else
|
|
sta_ptr->max_amsdu = MLAN_TX_DATA_BUF_SIZE_4K;
|
|
break;
|
|
case OPER_MODE_NTF:
|
|
break;
|
|
case EXTENSION:
|
|
pext_tlv = (MrvlIEtypes_Extension_t *)tlv;
|
|
if (pext_tlv->ext_id == HE_CAPABILITY) {
|
|
sta_ptr->is_11ax_enabled = MTRUE;
|
|
PRINTM(MCMND, "STA supports 11ax\n");
|
|
} else {
|
|
pext_tlv = MNULL;
|
|
}
|
|
break;
|
|
default:
|
|
break;
|
|
}
|
|
tlv_len = tlv->len;
|
|
tlv->type = wlan_cpu_to_le16(tlv->type);
|
|
tlv->len = wlan_cpu_to_le16(tlv->len);
|
|
memcpy_ext(pmadapter, pos, (t_u8 *)tlv,
|
|
sizeof(MrvlIEtypesHeader_t) + tlv_len,
|
|
sizeof(MrvlIEtypesHeader_t) + tlv_len);
|
|
pos += sizeof(MrvlIEtypesHeader_t) + tlv_len;
|
|
tlv_buf += sizeof(MrvlIEtypesHeader_t) + tlv_len;
|
|
tlv = (MrvlIEtypesHeader_t *)tlv_buf;
|
|
travel_len += sizeof(MrvlIEtypesHeader_t) + tlv_len;
|
|
tlv_buf_left -= sizeof(MrvlIEtypesHeader_t) + tlv_len;
|
|
}
|
|
if (sta_ptr->is_11ax_enabled) {
|
|
if (pext_tlv == MNULL) {
|
|
tlv = (MrvlIEtypesHeader_t *)pos;
|
|
tlv->type = wlan_cpu_to_le16(EXTENSION);
|
|
tlv->len = wlan_cpu_to_le16(
|
|
MIN(sta_ptr->he_cap.ieee_hdr.len,
|
|
sizeof(IEEEtypes_HECap_t) -
|
|
sizeof(IEEEtypes_Header_t)));
|
|
|
|
pos += sizeof(MrvlIEtypesHeader_t);
|
|
memcpy_ext(pmadapter, pos,
|
|
(t_u8 *)&sta_ptr->he_cap.ext_id, tlv->len,
|
|
tlv->len);
|
|
travel_len += sizeof(MrvlIEtypesHeader_t) + tlv->len;
|
|
}
|
|
}
|
|
|
|
if (sta_ptr->is_11n_enabled) {
|
|
if (pmpriv->uap_channel <= 14)
|
|
sta_ptr->bandmode = BAND_GN;
|
|
else
|
|
sta_ptr->bandmode = BAND_AN;
|
|
} else if (!b_only) {
|
|
if (pmpriv->uap_channel <= 14)
|
|
sta_ptr->bandmode = BAND_G;
|
|
else
|
|
sta_ptr->bandmode = BAND_A;
|
|
} else
|
|
sta_ptr->bandmode = BAND_B;
|
|
if (sta_ptr->is_11ac_enabled) {
|
|
if (pmpriv->uap_channel <= 14)
|
|
sta_ptr->bandmode = BAND_GAC;
|
|
else
|
|
sta_ptr->bandmode = BAND_AAC;
|
|
}
|
|
if (sta_ptr->is_11ax_enabled) {
|
|
if (pmpriv->uap_channel <= 14)
|
|
sta_ptr->bandmode = BAND_GAX;
|
|
else
|
|
sta_ptr->bandmode = BAND_AAX;
|
|
}
|
|
|
|
for (i = 0; i < MAX_NUM_TID; i++) {
|
|
if (sta_ptr->is_11n_enabled)
|
|
sta_ptr->ampdu_sta[i] =
|
|
pmpriv->aggr_prio_tbl[i].ampdu_user;
|
|
else
|
|
sta_ptr->ampdu_sta[i] = BA_STREAM_NOT_ALLOWED;
|
|
}
|
|
memset(pmadapter, sta_ptr->rx_seq, 0xff, sizeof(sta_ptr->rx_seq));
|
|
done:
|
|
cmd->size += travel_len;
|
|
cmd->size = wlan_cpu_to_le16(cmd->size);
|
|
LEAVE();
|
|
return MLAN_STATUS_SUCCESS;
|
|
}
|
|
|
|
/********************************************************
|
|
Global Functions
|
|
********************************************************/
|
|
/**
|
|
* @brief This function prepare the command before sending to firmware.
|
|
*
|
|
* @param priv A pointer to mlan_private structure
|
|
* @param cmd_no Command number
|
|
* @param cmd_action Command action: GET or SET
|
|
* @param cmd_oid Cmd oid: treated as sub command
|
|
* @param pioctl_buf A pointer to MLAN IOCTL Request buffer
|
|
* @param pdata_buf A pointer to information buffer
|
|
* @param pcmd_buf A pointer to cmd buf
|
|
*
|
|
* @return MLAN_STATUS_SUCCESS or MLAN_STATUS_FAILURE
|
|
*/
|
|
mlan_status wlan_ops_uap_prepare_cmd(t_void *priv, t_u16 cmd_no,
|
|
t_u16 cmd_action, t_u32 cmd_oid,
|
|
t_void *pioctl_buf, t_void *pdata_buf,
|
|
t_void *pcmd_buf)
|
|
{
|
|
HostCmd_DS_COMMAND *cmd_ptr = (HostCmd_DS_COMMAND *)pcmd_buf;
|
|
mlan_private *pmpriv = (mlan_private *)priv;
|
|
mlan_status ret = MLAN_STATUS_SUCCESS;
|
|
pmlan_ioctl_req pioctl_req = (mlan_ioctl_req *)pioctl_buf;
|
|
|
|
ENTER();
|
|
|
|
/* Prepare command */
|
|
switch (cmd_no) {
|
|
case HostCMD_APCMD_ACS_SCAN:
|
|
case HostCmd_CMD_SOFT_RESET:
|
|
case HOST_CMD_APCMD_BSS_STOP:
|
|
case HOST_CMD_APCMD_SYS_INFO:
|
|
case HOST_CMD_APCMD_SYS_RESET:
|
|
case HOST_CMD_APCMD_STA_LIST:
|
|
cmd_ptr->command = wlan_cpu_to_le16(cmd_no);
|
|
cmd_ptr->size = wlan_cpu_to_le16(S_DS_GEN);
|
|
break;
|
|
case HOST_CMD_APCMD_BSS_START:
|
|
ret = wlan_uap_cmd_bss_start(pmpriv, cmd_ptr);
|
|
#ifdef DRV_EMBEDDED_AUTHENTICATOR
|
|
if (IsAuthenticatorEnabled(pmpriv->psapriv))
|
|
AuthenticatorBssConfig(pmpriv->psapriv, MNULL, 1, 0, 0);
|
|
#endif
|
|
break;
|
|
case HOST_CMD_APCMD_SYS_CONFIGURE:
|
|
ret = wlan_uap_cmd_sys_configure(pmpriv, cmd_ptr, cmd_action,
|
|
(pmlan_ioctl_req)pioctl_buf,
|
|
pdata_buf);
|
|
break;
|
|
case HostCmd_CMD_802_11_PS_MODE_ENH:
|
|
ret = wlan_cmd_enh_power_mode(pmpriv, cmd_ptr, cmd_action,
|
|
(t_u16)cmd_oid, pdata_buf);
|
|
break;
|
|
#if defined(SDIO)
|
|
case HostCmd_CMD_SDIO_GPIO_INT_CONFIG:
|
|
ret = wlan_cmd_sdio_gpio_int(pmpriv, cmd_ptr, cmd_action,
|
|
pdata_buf);
|
|
break;
|
|
#endif
|
|
case HostCmd_CMD_FUNC_INIT:
|
|
if (pmpriv->adapter->hw_status == WlanHardwareStatusReset)
|
|
pmpriv->adapter->hw_status =
|
|
WlanHardwareStatusInitializing;
|
|
cmd_ptr->command = wlan_cpu_to_le16(cmd_no);
|
|
cmd_ptr->size = wlan_cpu_to_le16(S_DS_GEN);
|
|
break;
|
|
case HostCmd_CMD_FUNC_SHUTDOWN:
|
|
pmpriv->adapter->hw_status = WlanHardwareStatusReset;
|
|
cmd_ptr->command = wlan_cpu_to_le16(cmd_no);
|
|
cmd_ptr->size = wlan_cpu_to_le16(S_DS_GEN);
|
|
break;
|
|
case HostCmd_CMD_CFG_DATA:
|
|
ret = wlan_cmd_cfg_data(pmpriv, cmd_ptr, cmd_action, cmd_oid,
|
|
pdata_buf);
|
|
break;
|
|
case HostCmd_CMD_MAC_CONTROL:
|
|
ret = wlan_cmd_mac_control(pmpriv, cmd_ptr, cmd_action,
|
|
pdata_buf);
|
|
break;
|
|
case HostCmd_CMD_802_11_SNMP_MIB:
|
|
ret = wlan_uap_cmd_snmp_mib(pmpriv, cmd_ptr, cmd_action,
|
|
cmd_oid,
|
|
(pmlan_ioctl_req)pioctl_buf,
|
|
pdata_buf);
|
|
break;
|
|
case HostCmd_CMD_802_11_GET_LOG:
|
|
ret = wlan_uap_cmd_802_11_get_log(pmpriv, cmd_ptr);
|
|
break;
|
|
case HostCmd_CMD_802_11D_DOMAIN_INFO:
|
|
ret = wlan_cmd_802_11d_domain_info(pmpriv, cmd_ptr, cmd_action);
|
|
break;
|
|
case HostCmd_CMD_CHAN_REPORT_REQUEST:
|
|
ret = wlan_11h_cmd_process(pmpriv, cmd_ptr, pdata_buf);
|
|
break;
|
|
case HOST_CMD_APCMD_STA_DEAUTH:
|
|
ret = wlan_uap_cmd_sta_deauth(pmpriv, cmd_ptr, pdata_buf);
|
|
break;
|
|
case HOST_CMD_APCMD_REPORT_MIC:
|
|
ret = wlan_uap_cmd_report_mic(pmpriv, cmd_ptr, pdata_buf);
|
|
break;
|
|
case HostCmd_CMD_802_11_KEY_MATERIAL:
|
|
ret = wlan_uap_cmd_key_material(pmpriv, cmd_ptr, cmd_action,
|
|
cmd_oid, pdata_buf);
|
|
break;
|
|
case HostCmd_CMD_GET_HW_SPEC:
|
|
ret = wlan_cmd_get_hw_spec(pmpriv, cmd_ptr);
|
|
break;
|
|
#ifdef SDIO
|
|
case HostCmd_CMD_SDIO_SP_RX_AGGR_CFG:
|
|
ret = wlan_cmd_sdio_rx_aggr_cfg(cmd_ptr, cmd_action, pdata_buf);
|
|
break;
|
|
#endif
|
|
case HostCmd_CMD_802_11_HS_CFG_ENH:
|
|
ret = wlan_uap_cmd_802_11_hs_cfg(pmpriv, cmd_ptr, cmd_action,
|
|
(hs_config_param *)pdata_buf);
|
|
break;
|
|
case HostCmd_CMD_HS_WAKEUP_REASON:
|
|
ret = wlan_cmd_hs_wakeup_reason(pmpriv, cmd_ptr, pdata_buf);
|
|
break;
|
|
case HostCmd_CMD_802_11_ROBUSTCOEX:
|
|
ret = wlan_cmd_robustcoex(pmpriv, cmd_ptr, cmd_action,
|
|
(t_u16 *)pdata_buf);
|
|
break;
|
|
case HostCmd_CMD_DMCS_CONFIG:
|
|
ret = wlan_cmd_dmcs_config(pmpriv, cmd_ptr, cmd_action,
|
|
pdata_buf);
|
|
break;
|
|
case HostCmd_CMD_RECONFIGURE_TX_BUFF:
|
|
ret = wlan_cmd_recfg_tx_buf(pmpriv, cmd_ptr, cmd_action,
|
|
pdata_buf);
|
|
break;
|
|
case HostCmd_CMD_AMSDU_AGGR_CTRL:
|
|
ret = wlan_cmd_amsdu_aggr_ctrl(pmpriv, cmd_ptr, cmd_action,
|
|
pdata_buf);
|
|
break;
|
|
case HostCmd_CMD_11N_CFG:
|
|
ret = wlan_cmd_11n_cfg(pmpriv, cmd_ptr, cmd_action, pdata_buf);
|
|
break;
|
|
case HostCmd_CMD_11N_ADDBA_REQ:
|
|
ret = wlan_cmd_11n_addba_req(pmpriv, cmd_ptr, pdata_buf);
|
|
break;
|
|
case HostCmd_CMD_11N_DELBA:
|
|
ret = wlan_cmd_11n_delba(pmpriv, cmd_ptr, pdata_buf);
|
|
break;
|
|
case HostCmd_CMD_11N_ADDBA_RSP:
|
|
ret = wlan_cmd_11n_addba_rspgen(pmpriv, cmd_ptr, pdata_buf);
|
|
break;
|
|
case HostCmd_CMD_REJECT_ADDBA_REQ:
|
|
ret = wlan_cmd_reject_addba_req(pmpriv, cmd_ptr, cmd_action,
|
|
pdata_buf);
|
|
break;
|
|
case HostCmd_CMD_TX_BF_CFG:
|
|
ret = wlan_cmd_tx_bf_cfg(pmpriv, cmd_ptr, cmd_action,
|
|
pdata_buf);
|
|
break;
|
|
#if defined(WIFI_DIRECT_SUPPORT)
|
|
case HostCmd_CMD_SET_BSS_MODE:
|
|
cmd_ptr->command = wlan_cpu_to_le16(cmd_no);
|
|
if (pdata_buf)
|
|
cmd_ptr->params.bss_mode.con_type = *(t_u8 *)pdata_buf;
|
|
else
|
|
cmd_ptr->params.bss_mode.con_type =
|
|
BSS_MODE_WIFIDIRECT_GO;
|
|
cmd_ptr->size = wlan_cpu_to_le16(
|
|
sizeof(HostCmd_DS_SET_BSS_MODE) + S_DS_GEN);
|
|
ret = MLAN_STATUS_SUCCESS;
|
|
break;
|
|
#endif
|
|
case HostCmd_CMD_VERSION_EXT:
|
|
cmd_ptr->command = wlan_cpu_to_le16(cmd_no);
|
|
cmd_ptr->params.verext.version_str_sel =
|
|
(t_u8)(*((t_u32 *)pdata_buf));
|
|
cmd_ptr->size = wlan_cpu_to_le16(
|
|
sizeof(HostCmd_DS_VERSION_EXT) + S_DS_GEN);
|
|
ret = MLAN_STATUS_SUCCESS;
|
|
break;
|
|
case HostCmd_CMD_RX_MGMT_IND:
|
|
cmd_ptr->command = wlan_cpu_to_le16(cmd_no);
|
|
cmd_ptr->params.rx_mgmt_ind.action =
|
|
wlan_cpu_to_le16(cmd_action);
|
|
cmd_ptr->params.rx_mgmt_ind.mgmt_subtype_mask =
|
|
(t_u32)(*((t_u32 *)pdata_buf));
|
|
cmd_ptr->size = wlan_cpu_to_le16(
|
|
sizeof(HostCmd_DS_RX_MGMT_IND) + S_DS_GEN);
|
|
break;
|
|
case HostCmd_CMD_CFG_TX_DATA_PAUSE:
|
|
ret = wlan_uap_cmd_txdatapause(pmpriv, cmd_ptr, cmd_action,
|
|
pdata_buf);
|
|
break;
|
|
case HostCmd_CMD_802_11_RADIO_CONTROL:
|
|
ret = wlan_cmd_802_11_radio_control(pmpriv, cmd_ptr, cmd_action,
|
|
pdata_buf);
|
|
break;
|
|
case HostCmd_CMD_TX_RATE_CFG:
|
|
ret = wlan_cmd_tx_rate_cfg(pmpriv, cmd_ptr, cmd_action,
|
|
pdata_buf, pioctl_buf);
|
|
break;
|
|
case HostCmd_CMD_802_11_TX_RATE_QUERY:
|
|
cmd_ptr->command =
|
|
wlan_cpu_to_le16(HostCmd_CMD_802_11_TX_RATE_QUERY);
|
|
cmd_ptr->size = wlan_cpu_to_le16(sizeof(HostCmd_TX_RATE_QUERY) +
|
|
S_DS_GEN);
|
|
pmpriv->tx_rate = 0;
|
|
ret = MLAN_STATUS_SUCCESS;
|
|
break;
|
|
case HostCmd_CMD_802_11_REMAIN_ON_CHANNEL:
|
|
ret = wlan_cmd_remain_on_channel(pmpriv, cmd_ptr, cmd_action,
|
|
pdata_buf);
|
|
break;
|
|
#ifdef WIFI_DIRECT_SUPPORT
|
|
case HOST_CMD_WIFI_DIRECT_MODE_CONFIG:
|
|
ret = wlan_cmd_wifi_direct_mode(pmpriv, cmd_ptr, cmd_action,
|
|
pdata_buf);
|
|
break;
|
|
case HOST_CMD_P2P_PARAMS_CONFIG:
|
|
ret = wlan_cmd_p2p_params_config(pmpriv, cmd_ptr, cmd_action,
|
|
pdata_buf);
|
|
break;
|
|
#endif
|
|
case HOST_CMD_GPIO_TSF_LATCH_PARAM_CONFIG:
|
|
ret = wlan_cmd_gpio_tsf_latch(pmpriv, cmd_ptr, cmd_action,
|
|
pioctl_buf, pdata_buf);
|
|
break;
|
|
case HostCmd_CMD_802_11_RF_ANTENNA:
|
|
ret = wlan_cmd_802_11_rf_antenna(pmpriv, cmd_ptr, cmd_action,
|
|
pdata_buf);
|
|
break;
|
|
case HostCmd_CMD_802_11_MIMO_SWITCH:
|
|
ret = wlan_cmd_802_11_mimo_switch(pmpriv, cmd_ptr, pdata_buf);
|
|
break;
|
|
case HostCmd_CMD_11AC_CFG:
|
|
ret = wlan_cmd_11ac_cfg(pmpriv, cmd_ptr, cmd_action, pdata_buf);
|
|
break;
|
|
case HostCmd_CMD_DYN_BW:
|
|
ret = wlan_cmd_config_dyn_bw(pmpriv, cmd_ptr, cmd_action,
|
|
pdata_buf);
|
|
break;
|
|
case HostCmd_CMD_MAC_REG_ACCESS:
|
|
case HostCmd_CMD_BBP_REG_ACCESS:
|
|
case HostCmd_CMD_RF_REG_ACCESS:
|
|
case HostCmd_CMD_CAU_REG_ACCESS:
|
|
case HostCmd_CMD_TARGET_ACCESS:
|
|
case HostCmd_CMD_802_11_EEPROM_ACCESS:
|
|
case HostCmd_CMD_BCA_REG_ACCESS:
|
|
ret = wlan_cmd_reg_access(pmpriv, cmd_ptr, cmd_action,
|
|
pdata_buf);
|
|
break;
|
|
case HostCmd_CMD_MEM_ACCESS:
|
|
ret = wlan_cmd_mem_access(cmd_ptr, cmd_action, pdata_buf);
|
|
break;
|
|
case HostCmd_CMD_WMM_QUEUE_CONFIG:
|
|
ret = wlan_cmd_wmm_queue_config(pmpriv, cmd_ptr, pdata_buf);
|
|
break;
|
|
#ifdef RX_PACKET_COALESCE
|
|
case HostCmd_CMD_RX_PKT_COALESCE_CFG:
|
|
ret = wlan_cmd_rx_pkt_coalesce_cfg(pmpriv, cmd_ptr, cmd_action,
|
|
pdata_buf);
|
|
break;
|
|
#endif
|
|
case HOST_CMD_APCMD_OPER_CTRL:
|
|
ret = wlan_uap_cmd_oper_ctrl(pmpriv, cmd_ptr, cmd_action,
|
|
pdata_buf);
|
|
break;
|
|
|
|
case HostCmd_CMD_INDEPENDENT_RESET_CFG:
|
|
ret = wlan_cmd_ind_rst_cfg(cmd_ptr, cmd_action, pdata_buf);
|
|
break;
|
|
case HostCmd_CMD_GET_TSF:
|
|
ret = wlan_cmd_get_tsf(pmpriv, cmd_ptr, cmd_action);
|
|
break;
|
|
|
|
case HostCmd_CMD_802_11_PS_INACTIVITY_TIMEOUT:
|
|
ret = wlan_cmd_ps_inactivity_timeout(pmpriv, cmd_ptr,
|
|
cmd_action, pdata_buf);
|
|
break;
|
|
|
|
case HostCmd_CMD_CHAN_REGION_CFG:
|
|
cmd_ptr->command = wlan_cpu_to_le16(cmd_no);
|
|
cmd_ptr->size = wlan_cpu_to_le16(
|
|
sizeof(HostCmd_DS_CHAN_REGION_CFG) + S_DS_GEN);
|
|
cmd_ptr->params.reg_cfg.action = wlan_cpu_to_le16(cmd_action);
|
|
break;
|
|
case HostCmd_CMD_PACKET_AGGR_CTRL:
|
|
ret = wlan_cmd_packet_aggr_ctrl(pmpriv, cmd_ptr, cmd_action,
|
|
pdata_buf);
|
|
break;
|
|
#if defined(PCIE)
|
|
#if defined(PCIE8997) || defined(PCIE8897)
|
|
case HostCmd_CMD_PCIE_HOST_BUF_DETAILS:
|
|
ret = wlan_cmd_pcie_host_buf_cfg(pmpriv, cmd_ptr, cmd_action,
|
|
pdata_buf);
|
|
break;
|
|
#endif
|
|
#endif
|
|
case HOST_CMD_TX_RX_PKT_STATS:
|
|
ret = wlan_cmd_tx_rx_pkt_stats(pmpriv, cmd_ptr,
|
|
(pmlan_ioctl_req)pioctl_buf,
|
|
pdata_buf);
|
|
break;
|
|
case HostCmd_CMD_FW_DUMP_EVENT:
|
|
ret = wlan_cmd_fw_dump_event(pmpriv, cmd_ptr, cmd_action,
|
|
pdata_buf);
|
|
break;
|
|
case HostCmd_CMD_802_11_LINK_STATS:
|
|
ret = wlan_cmd_802_11_link_statistic(pmpriv, cmd_ptr,
|
|
cmd_action, pioctl_buf);
|
|
break;
|
|
case HostCmd_CMD_ADD_NEW_STATION:
|
|
ret = wlan_uap_cmd_add_station(pmpriv, cmd_ptr, cmd_action,
|
|
(pmlan_ioctl_req)pioctl_buf);
|
|
break;
|
|
case HostCmd_CMD_BOOT_SLEEP:
|
|
ret = wlan_cmd_boot_sleep(pmpriv, cmd_ptr, cmd_action,
|
|
pdata_buf);
|
|
break;
|
|
#if defined(DRV_EMBEDDED_AUTHENTICATOR)
|
|
case HostCmd_CMD_CRYPTO:
|
|
ret = wlan_cmd_crypto(pmpriv, cmd_ptr, cmd_action, pdata_buf);
|
|
break;
|
|
#endif
|
|
case HostCmd_CMD_11AX_CFG:
|
|
ret = wlan_cmd_11ax_cfg(pmpriv, cmd_ptr, cmd_action, pdata_buf);
|
|
break;
|
|
case HostCmd_CMD_11AX_CMD:
|
|
ret = wlan_cmd_11ax_cmd(pmpriv, cmd_ptr, cmd_action, pdata_buf);
|
|
break;
|
|
case HostCmd_CMD_RANGE_EXT:
|
|
ret = wlan_cmd_range_ext(pmpriv, cmd_ptr, cmd_action,
|
|
pdata_buf);
|
|
break;
|
|
case HostCmd_CMD_RX_ABORT_CFG:
|
|
ret = wlan_cmd_rxabortcfg(pmpriv, cmd_ptr, cmd_action,
|
|
pdata_buf);
|
|
break;
|
|
case HostCmd_CMD_RX_ABORT_CFG_EXT:
|
|
ret = wlan_cmd_rxabortcfg_ext(pmpriv, cmd_ptr, cmd_action,
|
|
pdata_buf);
|
|
break;
|
|
case HostCmd_CMD_TX_AMPDU_PROT_MODE:
|
|
ret = wlan_cmd_tx_ampdu_prot_mode(pmpriv, cmd_ptr, cmd_action,
|
|
pdata_buf);
|
|
break;
|
|
case HostCmd_CMD_DOT11MC_UNASSOC_FTM_CFG:
|
|
ret = wlan_cmd_dot11mc_unassoc_ftm_cfg(pmpriv, cmd_ptr,
|
|
cmd_action, pdata_buf);
|
|
break;
|
|
case HostCmd_CMD_RATE_ADAPT_CFG:
|
|
ret = wlan_cmd_rate_adapt_cfg(pmpriv, cmd_ptr, cmd_action,
|
|
pdata_buf);
|
|
break;
|
|
case HostCmd_CMD_CCK_DESENSE_CFG:
|
|
ret = wlan_cmd_cck_desense_cfg(pmpriv, cmd_ptr, cmd_action,
|
|
pdata_buf);
|
|
break;
|
|
case HostCmd_CHANNEL_TRPC_CONFIG:
|
|
ret = wlan_cmd_get_chan_trpc_config(pmpriv, cmd_ptr, cmd_action,
|
|
pdata_buf);
|
|
break;
|
|
case HostCmd_CMD_LOW_POWER_MODE_CFG:
|
|
ret = wlan_cmd_set_get_low_power_mode_cfg(
|
|
pmpriv, cmd_ptr, cmd_action, pdata_buf);
|
|
break;
|
|
case HostCmd_CMD_802_11_BAND_STEERING:
|
|
ret = wlan_cmd_set_get_band_steering_cfg(pmpriv, cmd_ptr,
|
|
cmd_action, pdata_buf);
|
|
break;
|
|
case HostCmd_CMD_UAP_BEACON_STUCK_CFG:
|
|
ret = wlan_cmd_set_get_beacon_stuck_cfg(pmpriv, cmd_ptr,
|
|
cmd_action, pdata_buf);
|
|
break;
|
|
case HostCmd_CMD_HAL_PHY_CFG:
|
|
ret = wlan_cmd_hal_phy_cfg(pmpriv, cmd_ptr, cmd_action,
|
|
pdata_buf);
|
|
break;
|
|
default:
|
|
PRINTM(MERROR, "PREP_CMD: unknown command- %#x\n", cmd_no);
|
|
if (pioctl_req)
|
|
pioctl_req->status_code = MLAN_ERROR_CMD_INVALID;
|
|
ret = MLAN_STATUS_FAILURE;
|
|
break;
|
|
}
|
|
LEAVE();
|
|
return ret;
|
|
}
|
|
|
|
/**
|
|
* @brief This function handles the AP mode command response
|
|
*
|
|
* @param priv A pointer to mlan_private structure
|
|
* @param cmdresp_no cmd no
|
|
* @param pcmd_buf cmdresp buf
|
|
* @param pioctl A pointer to ioctl buf
|
|
*
|
|
* @return MLAN_STATUS_SUCCESS or MLAN_STATUS_FAILURE
|
|
*/
|
|
mlan_status wlan_ops_uap_process_cmdresp(t_void *priv, t_u16 cmdresp_no,
|
|
t_void *pcmd_buf, t_void *pioctl)
|
|
{
|
|
mlan_status ret = MLAN_STATUS_SUCCESS;
|
|
mlan_private *pmpriv = (mlan_private *)priv;
|
|
HostCmd_DS_COMMAND *resp = (HostCmd_DS_COMMAND *)pcmd_buf;
|
|
mlan_ioctl_req *pioctl_buf = (mlan_ioctl_req *)pioctl;
|
|
mlan_adapter *pmadapter = pmpriv->adapter;
|
|
#ifdef SDIO
|
|
int ctr;
|
|
#endif
|
|
wlan_dfs_device_state_t *pstate_dfs =
|
|
(wlan_dfs_device_state_t *)&pmpriv->adapter->state_dfs;
|
|
t_u32 sec, usec;
|
|
ENTER();
|
|
|
|
/* If the command is not successful, cleanup and return failure */
|
|
if (resp->result != HostCmd_RESULT_OK) {
|
|
ret = uap_process_cmdresp_error(pmpriv, resp, pioctl_buf);
|
|
LEAVE();
|
|
return ret;
|
|
}
|
|
|
|
/* Command successful, handle response */
|
|
switch (cmdresp_no) {
|
|
case HOST_CMD_APCMD_BSS_STOP:
|
|
pmpriv->uap_bss_started = MFALSE;
|
|
/* Timestamp update is required because bss_start after skip_cac
|
|
* enabled should not select non-current channel just because
|
|
* timestamp got expired
|
|
*/
|
|
if (!pmpriv->intf_state_11h.is_11h_host &&
|
|
!pstate_dfs->dfs_check_pending &&
|
|
pstate_dfs->dfs_check_channel) {
|
|
pmpriv->adapter->callbacks.moal_get_system_time(
|
|
pmpriv->adapter->pmoal_handle, &sec, &usec);
|
|
pstate_dfs->dfs_report_time_sec = sec;
|
|
}
|
|
if (pmpriv->intf_state_11h.is_11h_host)
|
|
pmpriv->intf_state_11h.tx_disabled = MFALSE;
|
|
else {
|
|
if (pmpriv->adapter->ecsa_enable)
|
|
wlan_11h_remove_custom_ie(pmpriv->adapter,
|
|
pmpriv);
|
|
wlan_11h_check_update_radar_det_state(pmpriv);
|
|
}
|
|
|
|
if (pmpriv->adapter->state_rdh.stage == RDH_STOP_INTFS)
|
|
wlan_11h_radar_detected_callback((t_void *)pmpriv);
|
|
wlan_coex_ampdu_rxwinsize(pmadapter);
|
|
#ifdef DRV_EMBEDDED_AUTHENTICATOR
|
|
if (IsAuthenticatorEnabled(pmpriv->psapriv)) {
|
|
AuthenticatorBssConfig(pmpriv->psapriv, MNULL, 0, 1, 0);
|
|
AuthenticatorkeyClear(pmpriv->psapriv);
|
|
}
|
|
#endif
|
|
pmpriv->uap_host_based = 0;
|
|
break;
|
|
case HOST_CMD_APCMD_BSS_START:
|
|
if (!pmpriv->intf_state_11h.is_11h_host &&
|
|
pmpriv->adapter->state_rdh.stage == RDH_RESTART_INTFS)
|
|
wlan_11h_radar_detected_callback((t_void *)pmpriv);
|
|
/* Stop pps_uapsd_mode once bss_start */
|
|
pmpriv->adapter->tx_lock_flag = MFALSE;
|
|
pmpriv->adapter->pps_uapsd_mode = MFALSE;
|
|
pmpriv->adapter->delay_null_pkt = MFALSE;
|
|
/* Clear AMSDU statistics*/
|
|
pmpriv->amsdu_rx_cnt = 0;
|
|
pmpriv->amsdu_tx_cnt = 0;
|
|
pmpriv->msdu_in_rx_amsdu_cnt = 0;
|
|
pmpriv->msdu_in_tx_amsdu_cnt = 0;
|
|
break;
|
|
case HOST_CMD_APCMD_SYS_RESET:
|
|
pmpriv->uap_bss_started = MFALSE;
|
|
pmpriv->uap_host_based = 0;
|
|
#ifdef DRV_EMBEDDED_AUTHENTICATOR
|
|
AuthenitcatorInitBssConfig(pmpriv->psapriv);
|
|
#endif
|
|
ret = wlan_uap_ret_sys_reset(pmpriv, resp, pioctl_buf);
|
|
wlan_11h_check_update_radar_det_state(pmpriv);
|
|
wlan_coex_ampdu_rxwinsize(pmadapter);
|
|
break;
|
|
case HOST_CMD_APCMD_SYS_INFO:
|
|
break;
|
|
case HOST_CMD_APCMD_SYS_CONFIGURE:
|
|
ret = wlan_uap_ret_sys_config(pmpriv, resp, pioctl_buf);
|
|
break;
|
|
case HostCmd_CMD_802_11_PS_MODE_ENH:
|
|
ret = wlan_ret_enh_power_mode(pmpriv, resp, pioctl_buf);
|
|
break;
|
|
#if defined(SDIO)
|
|
case HostCmd_CMD_SDIO_GPIO_INT_CONFIG:
|
|
break;
|
|
#endif
|
|
case HostCmd_CMD_FUNC_INIT:
|
|
case HostCmd_CMD_FUNC_SHUTDOWN:
|
|
break;
|
|
case HostCmd_CMD_802_11_SNMP_MIB:
|
|
ret = wlan_uap_ret_snmp_mib(pmpriv, resp, pioctl_buf);
|
|
break;
|
|
case HostCmd_CMD_802_11_GET_LOG:
|
|
ret = wlan_uap_ret_get_log(pmpriv, resp, pioctl_buf);
|
|
break;
|
|
case HostCmd_CMD_802_11D_DOMAIN_INFO:
|
|
ret = wlan_ret_802_11d_domain_info(pmpriv, resp);
|
|
break;
|
|
case HostCmd_CMD_CHAN_REPORT_REQUEST:
|
|
ret = wlan_11h_cmdresp_process(pmpriv, resp);
|
|
break;
|
|
case HOST_CMD_APCMD_STA_DEAUTH:
|
|
break;
|
|
case HOST_CMD_APCMD_REPORT_MIC:
|
|
break;
|
|
case HostCmd_CMD_802_11_KEY_MATERIAL:
|
|
break;
|
|
case HOST_CMD_APCMD_STA_LIST:
|
|
ret = wlan_uap_ret_sta_list(pmpriv, resp, pioctl_buf);
|
|
break;
|
|
case HostCmd_CMD_GET_HW_SPEC:
|
|
ret = wlan_ret_get_hw_spec(pmpriv, resp, pioctl_buf);
|
|
break;
|
|
#ifdef SDIO
|
|
case HostCmd_CMD_SDIO_SP_RX_AGGR_CFG:
|
|
ret = wlan_ret_sdio_rx_aggr_cfg(pmpriv, resp);
|
|
break;
|
|
#endif
|
|
case HostCmd_CMD_CFG_DATA:
|
|
ret = wlan_ret_cfg_data(pmpriv, resp, pioctl_buf);
|
|
break;
|
|
case HostCmd_CMD_MAC_CONTROL:
|
|
ret = wlan_ret_mac_control(pmpriv, resp, pioctl_buf);
|
|
break;
|
|
case HostCmd_CMD_802_11_HS_CFG_ENH:
|
|
ret = wlan_ret_802_11_hs_cfg(pmpriv, resp, pioctl_buf);
|
|
break;
|
|
case HostCmd_CMD_HS_WAKEUP_REASON:
|
|
ret = wlan_ret_hs_wakeup_reason(pmpriv, resp, pioctl_buf);
|
|
break;
|
|
case HostCmd_CMD_802_11_ROBUSTCOEX:
|
|
break;
|
|
case HostCmd_CMD_DMCS_CONFIG:
|
|
ret = wlan_ret_dmcs_config(pmpriv, resp, pioctl_buf);
|
|
break;
|
|
case HostCmd_CMD_11N_ADDBA_REQ:
|
|
ret = wlan_ret_11n_addba_req(pmpriv, resp);
|
|
break;
|
|
case HostCmd_CMD_11N_DELBA:
|
|
ret = wlan_ret_11n_delba(pmpriv, resp);
|
|
break;
|
|
case HostCmd_CMD_11N_ADDBA_RSP:
|
|
ret = wlan_ret_11n_addba_resp(pmpriv, resp);
|
|
break;
|
|
case HostCmd_CMD_SET_BSS_MODE:
|
|
break;
|
|
case HostCmd_CMD_RECONFIGURE_TX_BUFF:
|
|
wlan_set_tx_pause_flag(pmpriv, MFALSE);
|
|
|
|
pmadapter->tx_buf_size =
|
|
(t_u16)wlan_le16_to_cpu(resp->params.tx_buf.buff_size);
|
|
#ifdef SDIO
|
|
if (IS_SD(pmadapter->card_type)) {
|
|
pmadapter->tx_buf_size = (pmadapter->tx_buf_size /
|
|
MLAN_SDIO_BLOCK_SIZE) *
|
|
MLAN_SDIO_BLOCK_SIZE;
|
|
pmadapter->pcard_sd->mp_end_port = wlan_le16_to_cpu(
|
|
resp->params.tx_buf.mp_end_port);
|
|
pmadapter->pcard_sd->mp_data_port_mask =
|
|
pmadapter->pcard_sd->reg->data_port_mask;
|
|
|
|
for (ctr = 1;
|
|
ctr <= pmadapter->pcard_sd->max_ports -
|
|
pmadapter->pcard_sd->mp_end_port;
|
|
ctr++) {
|
|
pmadapter->pcard_sd->mp_data_port_mask &=
|
|
~(1 << (pmadapter->pcard_sd->max_ports -
|
|
ctr));
|
|
}
|
|
|
|
pmadapter->pcard_sd->curr_wr_port =
|
|
pmadapter->pcard_sd->reg->start_wr_port;
|
|
pmadapter->pcard_sd->mpa_tx.pkt_aggr_limit =
|
|
MIN(pmadapter->pcard_sd->mp_aggr_pkt_limit,
|
|
(pmadapter->pcard_sd->mp_end_port >> 1));
|
|
PRINTM(MCMND, "end port %d, data port mask %x\n",
|
|
wlan_le16_to_cpu(
|
|
resp->params.tx_buf.mp_end_port),
|
|
pmadapter->pcard_sd->mp_data_port_mask);
|
|
}
|
|
#endif
|
|
pmadapter->curr_tx_buf_size = pmadapter->tx_buf_size;
|
|
PRINTM(MCMND, "max_tx_buf_size=%d, tx_buf_size=%d\n",
|
|
pmadapter->max_tx_buf_size, pmadapter->tx_buf_size);
|
|
break;
|
|
case HostCmd_CMD_AMSDU_AGGR_CTRL:
|
|
ret = wlan_ret_amsdu_aggr_ctrl(pmpriv, resp, pioctl_buf);
|
|
break;
|
|
case HostCmd_CMD_11N_CFG:
|
|
ret = wlan_ret_11n_cfg(pmpriv, resp, pioctl_buf);
|
|
break;
|
|
case HostCmd_CMD_REJECT_ADDBA_REQ:
|
|
ret = wlan_ret_reject_addba_req(pmpriv, resp, pioctl_buf);
|
|
break;
|
|
case HostCmd_CMD_TX_BF_CFG:
|
|
ret = wlan_ret_tx_bf_cfg(pmpriv, resp, pioctl_buf);
|
|
break;
|
|
case HostCmd_CMD_VERSION_EXT:
|
|
ret = wlan_ret_ver_ext(pmpriv, resp, pioctl_buf);
|
|
break;
|
|
case HostCmd_CMD_RX_MGMT_IND:
|
|
ret = wlan_ret_rx_mgmt_ind(pmpriv, resp, pioctl_buf);
|
|
break;
|
|
case HostCmd_CMD_CFG_TX_DATA_PAUSE:
|
|
ret = wlan_uap_ret_txdatapause(pmpriv, resp, pioctl_buf);
|
|
break;
|
|
case HostCmd_CMD_802_11_RADIO_CONTROL:
|
|
ret = wlan_ret_802_11_radio_control(pmpriv, resp, pioctl_buf);
|
|
break;
|
|
case HostCmd_CMD_TX_RATE_CFG:
|
|
ret = wlan_ret_tx_rate_cfg(pmpriv, resp, pioctl_buf);
|
|
break;
|
|
case HostCmd_CMD_802_11_TX_RATE_QUERY:
|
|
ret = wlan_ret_802_11_tx_rate_query(pmpriv, resp, pioctl_buf);
|
|
break;
|
|
case HostCmd_CMD_802_11_REMAIN_ON_CHANNEL:
|
|
ret = wlan_ret_remain_on_channel(pmpriv, resp, pioctl_buf);
|
|
break;
|
|
#ifdef WIFI_DIRECT_SUPPORT
|
|
case HOST_CMD_WIFI_DIRECT_MODE_CONFIG:
|
|
ret = wlan_ret_wifi_direct_mode(pmpriv, resp, pioctl_buf);
|
|
break;
|
|
case HOST_CMD_P2P_PARAMS_CONFIG:
|
|
ret = wlan_ret_p2p_params_config(pmpriv, resp, pioctl_buf);
|
|
break;
|
|
#endif
|
|
case HOST_CMD_GPIO_TSF_LATCH_PARAM_CONFIG:
|
|
ret = wlan_ret_gpio_tsf_latch(pmpriv, resp, pioctl_buf);
|
|
break;
|
|
case HostCmd_CMD_802_11_RF_ANTENNA:
|
|
ret = wlan_ret_802_11_rf_antenna(pmpriv, resp, pioctl_buf);
|
|
break;
|
|
case HostCmd_CMD_802_11_MIMO_SWITCH:
|
|
break;
|
|
case HostCmd_CMD_11AC_CFG:
|
|
ret = wlan_ret_11ac_cfg(pmpriv, resp, pioctl_buf);
|
|
break;
|
|
case HostCmd_CMD_DYN_BW:
|
|
ret = wlan_ret_dyn_bw(pmpriv, resp, pioctl_buf);
|
|
break;
|
|
case HostCmd_CMD_MAC_REG_ACCESS:
|
|
case HostCmd_CMD_BBP_REG_ACCESS:
|
|
case HostCmd_CMD_RF_REG_ACCESS:
|
|
case HostCmd_CMD_CAU_REG_ACCESS:
|
|
case HostCmd_CMD_TARGET_ACCESS:
|
|
case HostCmd_CMD_802_11_EEPROM_ACCESS:
|
|
case HostCmd_CMD_BCA_REG_ACCESS:
|
|
ret = wlan_ret_reg_access(pmpriv->adapter, cmdresp_no, resp,
|
|
pioctl_buf);
|
|
break;
|
|
case HostCmd_CMD_MEM_ACCESS:
|
|
ret = wlan_ret_mem_access(pmpriv, resp, pioctl_buf);
|
|
break;
|
|
case HostCmd_CMD_WMM_QUEUE_CONFIG:
|
|
ret = wlan_ret_wmm_queue_config(pmpriv, resp, pioctl_buf);
|
|
break;
|
|
#ifdef RX_PACKET_COALESCE
|
|
case HostCmd_CMD_RX_PKT_COALESCE_CFG:
|
|
ret = wlan_ret_rx_pkt_coalesce_cfg(pmpriv, resp, pioctl_buf);
|
|
break;
|
|
#endif
|
|
case HostCMD_APCMD_ACS_SCAN:
|
|
ret = wlan_ret_cmd_uap_acs_scan(pmpriv, resp, pioctl_buf);
|
|
break;
|
|
case HOST_CMD_APCMD_OPER_CTRL:
|
|
ret = wlan_uap_ret_oper_ctrl(pmpriv, resp, pioctl_buf);
|
|
break;
|
|
case HostCmd_CMD_INDEPENDENT_RESET_CFG:
|
|
ret = wlan_ret_ind_rst_cfg(pmpriv, resp, pioctl_buf);
|
|
break;
|
|
case HostCmd_CMD_802_11_PS_INACTIVITY_TIMEOUT:
|
|
break;
|
|
case HostCmd_CMD_GET_TSF:
|
|
ret = wlan_ret_get_tsf(pmpriv, resp, pioctl_buf);
|
|
break;
|
|
|
|
case HostCmd_CMD_CHAN_REGION_CFG:
|
|
ret = wlan_ret_chan_region_cfg(pmpriv, resp, pioctl_buf);
|
|
break;
|
|
case HostCmd_CMD_PACKET_AGGR_CTRL:
|
|
ret = wlan_ret_packet_aggr_ctrl(pmpriv, resp, pioctl_buf);
|
|
break;
|
|
#if defined(PCIE)
|
|
#if defined(PCIE8997) || defined(PCIE8897)
|
|
case HostCmd_CMD_PCIE_HOST_BUF_DETAILS:
|
|
PRINTM(MINFO, "PCIE host buffer configuration successful.\n");
|
|
break;
|
|
#endif
|
|
#endif
|
|
case HOST_CMD_TX_RX_PKT_STATS:
|
|
ret = wlan_ret_tx_rx_pkt_stats(pmpriv, resp, pioctl_buf);
|
|
break;
|
|
case HostCmd_CMD_802_11_LINK_STATS:
|
|
ret = wlan_ret_get_link_statistic(pmpriv, resp, pioctl_buf);
|
|
break;
|
|
case HostCmd_CMD_BOOT_SLEEP:
|
|
ret = wlan_ret_boot_sleep(pmpriv, resp, pioctl_buf);
|
|
break;
|
|
case HostCmd_CMD_ADD_NEW_STATION:
|
|
break;
|
|
#if defined(DRV_EMBEDDED_AUTHENTICATOR)
|
|
case HostCmd_CMD_CRYPTO:
|
|
ret = wlan_ret_crypto(pmpriv, resp, pioctl_buf);
|
|
break;
|
|
#endif
|
|
case HostCmd_CMD_11AX_CFG:
|
|
ret = wlan_ret_11ax_cfg(pmpriv, resp, pioctl_buf);
|
|
break;
|
|
case HostCmd_CMD_11AX_CMD:
|
|
ret = wlan_ret_11ax_cmd(pmpriv, resp, pioctl_buf);
|
|
break;
|
|
case HostCmd_CMD_RANGE_EXT:
|
|
ret = wlan_ret_range_ext(pmpriv, resp, pioctl_buf);
|
|
break;
|
|
case HostCmd_CMD_RX_ABORT_CFG:
|
|
ret = wlan_ret_rxabortcfg(pmpriv, resp, pioctl_buf);
|
|
break;
|
|
case HostCmd_CMD_RX_ABORT_CFG_EXT:
|
|
ret = wlan_ret_rxabortcfg_ext(pmpriv, resp, pioctl_buf);
|
|
break;
|
|
case HostCmd_CMD_TX_AMPDU_PROT_MODE:
|
|
ret = wlan_ret_tx_ampdu_prot_mode(pmpriv, resp, pioctl_buf);
|
|
break;
|
|
case HostCmd_CMD_DOT11MC_UNASSOC_FTM_CFG:
|
|
ret = wlan_ret_dot11mc_unassoc_ftm_cfg(pmpriv, resp,
|
|
pioctl_buf);
|
|
break;
|
|
case HostCmd_CMD_HAL_PHY_CFG:
|
|
ret = wlan_ret_hal_phy_cfg(pmpriv, resp, pioctl_buf);
|
|
break;
|
|
case HostCmd_CMD_RATE_ADAPT_CFG:
|
|
ret = wlan_ret_rate_adapt_cfg(pmpriv, resp, pioctl_buf);
|
|
break;
|
|
case HostCmd_CMD_CCK_DESENSE_CFG:
|
|
ret = wlan_ret_cck_desense_cfg(pmpriv, resp, pioctl_buf);
|
|
break;
|
|
case HostCmd_CHANNEL_TRPC_CONFIG:
|
|
ret = wlan_ret_get_chan_trpc_config(pmpriv, resp, pioctl_buf);
|
|
break;
|
|
case HostCmd_CMD_LOW_POWER_MODE_CFG:
|
|
ret = wlan_ret_set_get_low_power_mode_cfg(pmpriv, resp,
|
|
pioctl_buf);
|
|
break;
|
|
case HostCmd_CMD_802_11_BAND_STEERING:
|
|
ret = wlan_ret_set_get_band_steering_cfg(pmpriv, resp,
|
|
pioctl_buf);
|
|
break;
|
|
case HostCmd_CMD_UAP_BEACON_STUCK_CFG:
|
|
ret = wlan_ret_set_get_beacon_stuck_cfg(pmpriv, resp,
|
|
pioctl_buf);
|
|
break;
|
|
default:
|
|
PRINTM(MERROR, "CMD_RESP: Unknown command response %#x\n",
|
|
resp->command);
|
|
if (pioctl_buf)
|
|
pioctl_buf->status_code = MLAN_ERROR_CMD_RESP_FAIL;
|
|
break;
|
|
}
|
|
LEAVE();
|
|
return ret;
|
|
}
|
|
|
|
/**
|
|
* @brief This function handles events generated by firmware
|
|
*
|
|
* @param priv A pointer to mlan_private structure
|
|
*
|
|
* @return MLAN_STATUS_SUCCESS or MLAN_STATUS_FAILURE
|
|
*/
|
|
mlan_status wlan_ops_uap_process_event(t_void *priv)
|
|
{
|
|
pmlan_private pmpriv = (pmlan_private)priv;
|
|
pmlan_adapter pmadapter = pmpriv->adapter;
|
|
pmlan_callbacks pcb = &pmadapter->callbacks;
|
|
mlan_status ret = MLAN_STATUS_SUCCESS;
|
|
t_u32 eventcause = pmadapter->event_cause;
|
|
pmlan_buffer pmbuf = pmadapter->pmlan_buffer_event;
|
|
t_u8 *event_buf = MNULL;
|
|
mlan_event *pevent = MNULL;
|
|
t_u8 sta_addr[MLAN_MAC_ADDR_LENGTH];
|
|
sta_node *sta_ptr = MNULL;
|
|
t_u8 i = 0;
|
|
t_u8 channel = 0;
|
|
MrvlIEtypes_channel_band_t *pchan_info = MNULL;
|
|
chan_band_info *pchan_band_info = MNULL;
|
|
event_exceed_max_p2p_conn *event_excd_p2p = MNULL;
|
|
t_u16 enable;
|
|
|
|
ENTER();
|
|
|
|
if (!pmbuf) {
|
|
LEAVE();
|
|
return MLAN_STATUS_FAILURE;
|
|
}
|
|
/* Event length check */
|
|
if (pmbuf && (pmbuf->data_len - sizeof(eventcause)) > MAX_EVENT_SIZE) {
|
|
pmbuf->status_code = MLAN_ERROR_PKT_SIZE_INVALID;
|
|
LEAVE();
|
|
return MLAN_STATUS_FAILURE;
|
|
}
|
|
|
|
/* Allocate memory for event buffer */
|
|
ret = pcb->moal_malloc(pmadapter->pmoal_handle,
|
|
MAX_EVENT_SIZE + sizeof(mlan_event),
|
|
MLAN_MEM_DEF, &event_buf);
|
|
if ((ret != MLAN_STATUS_SUCCESS) || !event_buf) {
|
|
PRINTM(MERROR, "Could not allocate buffer for event buf\n");
|
|
if (pmbuf)
|
|
pmbuf->status_code = MLAN_ERROR_NO_MEM;
|
|
goto done;
|
|
}
|
|
pevent = (pmlan_event)event_buf;
|
|
memset(pmadapter, &pevent->event_id, 0, sizeof(pevent->event_id));
|
|
|
|
if (eventcause != EVENT_PS_SLEEP && eventcause != EVENT_PS_AWAKE &&
|
|
pmbuf->data_len > sizeof(eventcause))
|
|
DBG_HEXDUMP(MEVT_D, "EVENT", pmbuf->pbuf + pmbuf->data_offset,
|
|
pmbuf->data_len);
|
|
|
|
switch (eventcause) {
|
|
case EVENT_MICRO_AP_BSS_START:
|
|
PRINTM(MEVENT, "EVENT: MICRO_AP_BSS_START\n");
|
|
pmpriv->uap_bss_started = MTRUE;
|
|
pmpriv->is_data_rate_auto = MTRUE;
|
|
memcpy_ext(pmadapter, pmpriv->curr_addr,
|
|
pmadapter->event_body + 2, MLAN_MAC_ADDR_LENGTH,
|
|
MLAN_MAC_ADDR_LENGTH);
|
|
pevent->event_id = MLAN_EVENT_ID_UAP_FW_BSS_START;
|
|
wlan_check_uap_capability(pmpriv, pmbuf);
|
|
wlan_coex_ampdu_rxwinsize(pmadapter);
|
|
#ifdef DRV_EMBEDDED_AUTHENTICATOR
|
|
if (IsAuthenticatorEnabled(pmpriv->psapriv)) {
|
|
pmadapter->authenticator_priv = pmpriv;
|
|
wlan_recv_event(pmpriv, MLAN_EVENT_ID_DRV_DEFER_RX_WORK,
|
|
MNULL);
|
|
}
|
|
#endif
|
|
break;
|
|
case EVENT_MICRO_AP_BSS_ACTIVE:
|
|
PRINTM(MEVENT, "EVENT: MICRO_AP_BSS_ACTIVE\n");
|
|
pmpriv->media_connected = MTRUE;
|
|
pmpriv->port_open = MTRUE;
|
|
pevent->event_id = MLAN_EVENT_ID_UAP_FW_BSS_ACTIVE;
|
|
break;
|
|
case EVENT_MICRO_AP_BSS_IDLE:
|
|
PRINTM(MEVENT, "EVENT: MICRO_AP_BSS_IDLE\n");
|
|
pevent->event_id = MLAN_EVENT_ID_UAP_FW_BSS_IDLE;
|
|
pmpriv->media_connected = MFALSE;
|
|
wlan_clean_txrx(pmpriv);
|
|
wlan_notify_station_deauth(pmpriv);
|
|
wlan_delete_station_list(pmpriv);
|
|
pmpriv->port_open = MFALSE;
|
|
pmpriv->amsdu_disable = MFALSE;
|
|
pmpriv->tx_pause = MFALSE;
|
|
break;
|
|
case EVENT_MICRO_AP_MIC_COUNTERMEASURES:
|
|
PRINTM(MEVENT, "EVENT: MICRO_AP_MIC_COUNTERMEASURES\n");
|
|
pevent->event_id = MLAN_EVENT_ID_UAP_FW_MIC_COUNTERMEASURES;
|
|
break;
|
|
case EVENT_PS_AWAKE:
|
|
PRINTM(MINFO, "EVENT: AWAKE\n");
|
|
PRINTM_NETINTF(MEVENT, pmpriv);
|
|
PRINTM(MEVENT, "||");
|
|
/* Handle unexpected PS AWAKE event */
|
|
if (pmadapter->ps_state == PS_STATE_SLEEP_CFM)
|
|
break;
|
|
pmadapter->pm_wakeup_card_req = MFALSE;
|
|
pmadapter->pm_wakeup_fw_try = MFALSE;
|
|
pmadapter->ps_state = PS_STATE_AWAKE;
|
|
|
|
break;
|
|
case EVENT_PS_SLEEP:
|
|
PRINTM(MINFO, "EVENT: SLEEP\n");
|
|
PRINTM_NETINTF(MEVENT, pmpriv);
|
|
PRINTM(MEVENT, "__");
|
|
/* Handle unexpected PS SLEEP event */
|
|
if (pmadapter->ps_state == PS_STATE_SLEEP_CFM)
|
|
break;
|
|
pmadapter->ps_state = PS_STATE_PRE_SLEEP;
|
|
wlan_check_ps_cond(pmadapter);
|
|
break;
|
|
case EVENT_MICRO_AP_STA_ASSOC:
|
|
wlan_process_sta_assoc_event(pmpriv, pevent, pmbuf);
|
|
memcpy_ext(pmadapter, sta_addr, pmadapter->event_body + 2,
|
|
MLAN_MAC_ADDR_LENGTH, MLAN_MAC_ADDR_LENGTH);
|
|
sta_ptr = wlan_add_station_entry(pmpriv, sta_addr);
|
|
PRINTM_NETINTF(MMSG, pmpriv);
|
|
PRINTM(MMSG, "wlan: EVENT: MICRO_AP_STA_ASSOC " MACSTR "\n",
|
|
MAC2STR(sta_addr));
|
|
if (!sta_ptr)
|
|
break;
|
|
if (pmpriv->is_11n_enabled
|
|
#ifdef DRV_EMBEDDED_AUTHENTICATOR
|
|
|| IsAuthenticatorEnabled(pmpriv->psapriv)
|
|
#endif
|
|
) {
|
|
wlan_check_sta_capability(pmpriv, pmbuf, sta_ptr);
|
|
for (i = 0; i < MAX_NUM_TID; i++) {
|
|
if (sta_ptr->is_11n_enabled)
|
|
sta_ptr->ampdu_sta[i] =
|
|
pmpriv->aggr_prio_tbl[i]
|
|
.ampdu_user;
|
|
else
|
|
sta_ptr->ampdu_sta[i] =
|
|
BA_STREAM_NOT_ALLOWED;
|
|
}
|
|
memset(pmadapter, sta_ptr->rx_seq, 0xff,
|
|
sizeof(sta_ptr->rx_seq));
|
|
}
|
|
if (pmpriv->sec_info.wapi_enabled)
|
|
wlan_update_wapi_info_tlv(pmpriv, pmbuf);
|
|
#ifdef DRV_EMBEDDED_AUTHENTICATOR
|
|
/**enter authenticator*/
|
|
if (IsAuthenticatorEnabled(pmpriv->psapriv))
|
|
AuthenticatorSendEapolPacket(
|
|
pmpriv->psapriv, sta_ptr->cm_connectioninfo);
|
|
#endif
|
|
pevent->event_id = MLAN_EVENT_ID_DRV_PASSTHRU;
|
|
break;
|
|
case EVENT_MICRO_AP_STA_DEAUTH:
|
|
pevent->event_id = MLAN_EVENT_ID_UAP_FW_STA_DISCONNECT;
|
|
pevent->bss_index = pmpriv->bss_index;
|
|
pevent->event_len = pmbuf->data_len - 4;
|
|
/* skip event length field */
|
|
memcpy_ext(pmadapter, (t_u8 *)pevent->event_buf,
|
|
pmbuf->pbuf + pmbuf->data_offset + 4,
|
|
pevent->event_len, pevent->event_len);
|
|
wlan_recv_event(pmpriv, pevent->event_id, pevent);
|
|
memcpy_ext(pmadapter, sta_addr, pmadapter->event_body + 2,
|
|
MLAN_MAC_ADDR_LENGTH, MLAN_MAC_ADDR_LENGTH);
|
|
PRINTM_NETINTF(MMSG, pmpriv);
|
|
PRINTM(MMSG, "wlan: EVENT: MICRO_AP_STA_DEAUTH " MACSTR "\n",
|
|
MAC2STR(sta_addr));
|
|
if (pmpriv->is_11n_enabled) {
|
|
wlan_cleanup_reorder_tbl(pmpriv, sta_addr);
|
|
wlan_11n_cleanup_txbastream_tbl(pmpriv, sta_addr);
|
|
}
|
|
wlan_wmm_delete_peer_ralist(pmpriv, sta_addr);
|
|
wlan_delete_station_entry(pmpriv, sta_addr);
|
|
pevent->event_id = MLAN_EVENT_ID_DRV_PASSTHRU;
|
|
break;
|
|
case EVENT_HS_ACT_REQ:
|
|
PRINTM(MEVENT, "EVENT: HS_ACT_REQ\n");
|
|
ret = wlan_prepare_cmd(pmpriv, HostCmd_CMD_802_11_HS_CFG_ENH, 0,
|
|
0, MNULL, MNULL);
|
|
break;
|
|
case EVENT_ADDBA:
|
|
PRINTM(MEVENT, "EVENT: ADDBA Request\n");
|
|
if (pmpriv->media_connected == MTRUE)
|
|
ret = wlan_prepare_cmd(pmpriv,
|
|
HostCmd_CMD_11N_ADDBA_RSP,
|
|
HostCmd_ACT_GEN_SET, 0, MNULL,
|
|
pmadapter->event_body);
|
|
else
|
|
PRINTM(MERROR,
|
|
"Ignore ADDBA Request event in BSS idle state\n");
|
|
break;
|
|
case EVENT_DELBA:
|
|
PRINTM(MEVENT, "EVENT: DELBA Request\n");
|
|
if (pmpriv->media_connected == MTRUE)
|
|
wlan_11n_delete_bastream(pmpriv, pmadapter->event_body);
|
|
else
|
|
PRINTM(MERROR,
|
|
"Ignore DELBA Request event in BSS idle state\n");
|
|
break;
|
|
case EVENT_BA_STREAM_TIMEOUT:
|
|
PRINTM(MEVENT, "EVENT: BA Stream timeout\n");
|
|
if (pmpriv->media_connected == MTRUE)
|
|
wlan_11n_ba_stream_timeout(
|
|
pmpriv, (HostCmd_DS_11N_BATIMEOUT *)
|
|
pmadapter->event_body);
|
|
else
|
|
PRINTM(MERROR,
|
|
"Ignore BA Stream timeout event in BSS idle state\n");
|
|
break;
|
|
case EVENT_RXBA_SYNC:
|
|
PRINTM(MEVENT, "EVENT: RXBA_SYNC\n");
|
|
wlan_11n_rxba_sync_event(pmpriv, pmadapter->event_body,
|
|
pmbuf->data_len - sizeof(eventcause));
|
|
break;
|
|
case EVENT_AMSDU_AGGR_CTRL:
|
|
PRINTM(MEVENT, "EVENT: AMSDU_AGGR_CTRL %d\n",
|
|
*(t_u16 *)pmadapter->event_body);
|
|
pmadapter->tx_buf_size =
|
|
MIN(pmadapter->curr_tx_buf_size,
|
|
wlan_le16_to_cpu(*(t_u16 *)pmadapter->event_body));
|
|
if (pmbuf->data_len == sizeof(eventcause) + sizeof(t_u32)) {
|
|
enable = wlan_le16_to_cpu(
|
|
*(t_u16 *)(pmadapter->event_body +
|
|
sizeof(t_u16)));
|
|
if (enable)
|
|
pmpriv->amsdu_disable = MFALSE;
|
|
else
|
|
pmpriv->amsdu_disable = MTRUE;
|
|
PRINTM(MEVENT, "amsdu_disable=%d\n",
|
|
pmpriv->amsdu_disable);
|
|
}
|
|
PRINTM(MEVENT, "tx_buf_size %d\n", pmadapter->tx_buf_size);
|
|
break;
|
|
case EVENT_TX_DATA_PAUSE:
|
|
PRINTM(MEVENT, "EVENT: TX_DATA_PAUSE\n");
|
|
wlan_process_tx_pause_event(priv, pmbuf);
|
|
break;
|
|
case EVENT_RADAR_DETECTED:
|
|
PRINTM_NETINTF(MEVENT, pmpriv);
|
|
PRINTM(MEVENT, "EVENT: Radar Detected\n");
|
|
if (pmpriv->adapter->dfs_test_params.cac_restart &&
|
|
pmpriv->adapter->state_dfs.dfs_check_pending) {
|
|
wlan_11h_cancel_radar_detect(pmpriv);
|
|
wlan_11h_issue_radar_detect(
|
|
pmpriv, MNULL,
|
|
pmpriv->adapter->dfs_test_params.chan,
|
|
pmpriv->adapter->dfs_test_params.bandcfg);
|
|
pevent->event_id = 0;
|
|
break;
|
|
}
|
|
/* Send as passthru first, this event can cause other events */
|
|
memset(pmadapter, event_buf, 0x00, MAX_EVENT_SIZE);
|
|
pevent->bss_index = pmpriv->bss_index;
|
|
pevent->event_id = MLAN_EVENT_ID_DRV_PASSTHRU;
|
|
pevent->event_len = pmbuf->data_len;
|
|
memcpy_ext(pmadapter, (t_u8 *)pevent->event_buf,
|
|
pmbuf->pbuf + pmbuf->data_offset, pevent->event_len,
|
|
pevent->event_len);
|
|
wlan_recv_event(pmpriv, pevent->event_id, pevent);
|
|
pevent->event_id = 0; /* clear to avoid resending at end of fcn
|
|
*/
|
|
|
|
/* Print event data */
|
|
pevent->event_id = MLAN_EVENT_ID_FW_RADAR_DETECTED;
|
|
pevent->event_len = pmbuf->data_len - sizeof(eventcause);
|
|
memcpy_ext(pmadapter, (t_u8 *)pevent->event_buf,
|
|
pmbuf->pbuf + pmbuf->data_offset +
|
|
sizeof(eventcause),
|
|
pevent->event_len, pevent->event_len);
|
|
wlan_11h_print_event_radar_detected(pmpriv, pevent, &channel);
|
|
*((t_u8 *)pevent->event_buf) = channel;
|
|
if (!pmpriv->intf_state_11h.is_11h_host) {
|
|
if (pmadapter->state_rdh.stage == RDH_OFF) {
|
|
pmadapter->state_rdh.stage = RDH_CHK_INTFS;
|
|
wlan_11h_radar_detected_handling(pmadapter,
|
|
pmpriv);
|
|
if (pmpriv->uap_host_based)
|
|
wlan_recv_event(
|
|
priv,
|
|
MLAN_EVENT_ID_FW_RADAR_DETECTED,
|
|
pevent);
|
|
} else {
|
|
PRINTM(MEVENT,
|
|
"Ignore Event Radar Detected - handling"
|
|
" already in progress.\n");
|
|
}
|
|
} else {
|
|
if (pmpriv->adapter->dfs_test_params
|
|
.no_channel_change_on_radar ||
|
|
pmpriv->adapter->dfs_test_params
|
|
.fixed_new_channel_on_radar) {
|
|
if (pmadapter->state_rdh.stage == RDH_OFF ||
|
|
pmadapter->state_rdh.stage ==
|
|
RDH_SET_CUSTOM_IE) {
|
|
pmadapter->state_rdh.stage =
|
|
RDH_CHK_INTFS;
|
|
wlan_11h_radar_detected_handling(
|
|
pmadapter, pmpriv);
|
|
} else
|
|
PRINTM(MEVENT,
|
|
"Ignore Event Radar Detected - handling already in progress.\n");
|
|
} else {
|
|
pmpriv->intf_state_11h.tx_disabled = MTRUE;
|
|
wlan_recv_event(priv,
|
|
MLAN_EVENT_ID_FW_RADAR_DETECTED,
|
|
pevent);
|
|
}
|
|
}
|
|
|
|
pevent->event_id = 0; /* clear to avoid resending at end of fcn
|
|
*/
|
|
break;
|
|
case EVENT_CHANNEL_REPORT_RDY:
|
|
PRINTM_NETINTF(MEVENT, pmpriv);
|
|
PRINTM(MEVENT, "EVENT: Channel Report Ready\n");
|
|
pmpriv->adapter->dfs_test_params.cac_restart = MFALSE;
|
|
memset(pmadapter, event_buf, 0x00, MAX_EVENT_SIZE);
|
|
/* Setup event buffer */
|
|
pevent->bss_index = pmpriv->bss_index;
|
|
pevent->event_id = MLAN_EVENT_ID_FW_CHANNEL_REPORT_RDY;
|
|
pevent->event_len = pmbuf->data_len - sizeof(eventcause);
|
|
/* Copy event data */
|
|
memcpy_ext(pmadapter, (t_u8 *)pevent->event_buf,
|
|
pmbuf->pbuf + pmbuf->data_offset +
|
|
sizeof(eventcause),
|
|
pevent->event_len, pevent->event_len);
|
|
/* Handle / pass event data, and free buffer */
|
|
ret = wlan_11h_handle_event_chanrpt_ready(pmpriv, pevent,
|
|
&channel);
|
|
if (pmpriv->intf_state_11h.is_11h_host) {
|
|
*((t_u8 *)pevent->event_buf) =
|
|
pmpriv->adapter->state_dfs.dfs_radar_found;
|
|
*((t_u8 *)pevent->event_buf + 1) = channel;
|
|
wlan_recv_event(pmpriv,
|
|
MLAN_EVENT_ID_FW_CHANNEL_REPORT_RDY,
|
|
pevent);
|
|
} else {
|
|
/* Send up this Event to unblock MOAL waitqueue */
|
|
wlan_recv_event(pmpriv, MLAN_EVENT_ID_DRV_MEAS_REPORT,
|
|
MNULL);
|
|
}
|
|
pevent->event_id = MLAN_EVENT_ID_DRV_PASSTHRU;
|
|
break;
|
|
case EVENT_CHANNEL_SWITCH:
|
|
pchan_info =
|
|
(MrvlIEtypes_channel_band_t *)(pmadapter->event_body);
|
|
channel = pchan_info->channel;
|
|
PRINTM_NETINTF(MEVENT, pmpriv);
|
|
PRINTM(MEVENT, "EVENT: CHANNEL_SWITCH new channel %d\n",
|
|
channel);
|
|
pmpriv->uap_channel = channel;
|
|
pmpriv->uap_state_chan_cb.channel = pchan_info->channel;
|
|
pmpriv->uap_state_chan_cb.bandcfg = pchan_info->bandcfg;
|
|
if (wlan_11h_radar_detect_required(pmpriv,
|
|
pchan_info->channel)) {
|
|
if (!wlan_11h_is_active(pmpriv)) {
|
|
/* active 11h extention in Fw */
|
|
ret = wlan_11h_activate(pmpriv, MNULL, MTRUE);
|
|
ret = wlan_11h_config_master_radar_det(pmpriv,
|
|
MTRUE);
|
|
ret = wlan_11h_check_update_radar_det_state(
|
|
pmpriv);
|
|
}
|
|
if (pmpriv->uap_host_based)
|
|
pmpriv->intf_state_11h.is_11h_host = MTRUE;
|
|
wlan_11h_set_dfs_check_chan(pmpriv,
|
|
pchan_info->channel);
|
|
}
|
|
if ((pmpriv->adapter->state_rdh.stage != RDH_OFF &&
|
|
!pmpriv->intf_state_11h.is_11h_host) ||
|
|
pmpriv->adapter->dfs_test_params.no_channel_change_on_radar ||
|
|
pmpriv->adapter->dfs_test_params.fixed_new_channel_on_radar) {
|
|
/* Handle embedded DFS */
|
|
if (pmpriv->adapter->state_rdh.stage ==
|
|
RDH_SET_CUSTOM_IE) {
|
|
pmadapter->state_rdh.stage =
|
|
RDH_RESTART_TRAFFIC;
|
|
wlan_11h_radar_detected_handling(pmadapter,
|
|
pmpriv);
|
|
}
|
|
|
|
} else {
|
|
/* Handle Host-based DFS and non-DFS(normal uap) case */
|
|
pmpriv->intf_state_11h.tx_disabled = MFALSE;
|
|
memset(pmadapter, event_buf, 0x00, MAX_EVENT_SIZE);
|
|
/* Setup event buffer */
|
|
pevent->bss_index = pmpriv->bss_index;
|
|
pevent->event_id =
|
|
MLAN_EVENT_ID_FW_CHAN_SWITCH_COMPLETE;
|
|
pevent->event_len = sizeof(chan_band_info);
|
|
pchan_band_info = (chan_band_info *)pevent->event_buf;
|
|
/* Copy event data */
|
|
memcpy_ext(pmadapter, (t_u8 *)&pchan_band_info->bandcfg,
|
|
(t_u8 *)&pchan_info->bandcfg,
|
|
sizeof(pchan_info->bandcfg),
|
|
sizeof(pchan_info->bandcfg));
|
|
pchan_band_info->channel = pchan_info->channel;
|
|
if (pchan_band_info->bandcfg.chanWidth == CHAN_BW_80MHZ)
|
|
pchan_band_info->center_chan =
|
|
wlan_get_center_freq_idx(
|
|
priv, BAND_AAC,
|
|
pchan_info->channel,
|
|
CHANNEL_BW_80MHZ);
|
|
pchan_band_info->is_11n_enabled =
|
|
pmpriv->is_11n_enabled;
|
|
wlan_recv_event(pmpriv,
|
|
MLAN_EVENT_ID_FW_CHAN_SWITCH_COMPLETE,
|
|
pevent);
|
|
pevent->event_id = 0;
|
|
}
|
|
break;
|
|
case EVENT_REMAIN_ON_CHANNEL_EXPIRED:
|
|
PRINTM_NETINTF(MEVENT, pmpriv);
|
|
PRINTM(MEVENT, "EVENT: REMAIN_ON_CHANNEL_EXPIRED reason=%d\n",
|
|
*(t_u16 *)pmadapter->event_body);
|
|
wlan_recv_event(pmpriv, MLAN_EVENT_ID_DRV_FLUSH_RX_WORK, MNULL);
|
|
pevent->event_id = MLAN_EVENT_ID_FW_REMAIN_ON_CHAN_EXPIRED;
|
|
break;
|
|
|
|
case EVENT_FW_DEBUG_INFO:
|
|
memset(pmadapter, event_buf, 0x00, MAX_EVENT_SIZE);
|
|
pevent->bss_index = pmpriv->bss_index;
|
|
pevent->event_id = MLAN_EVENT_ID_FW_DEBUG_INFO;
|
|
pevent->event_len = pmbuf->data_len - sizeof(eventcause);
|
|
memcpy_ext(pmadapter, (t_u8 *)pevent->event_buf,
|
|
pmbuf->pbuf + pmbuf->data_offset +
|
|
sizeof(eventcause),
|
|
pevent->event_len, pevent->event_len);
|
|
PRINTM(MEVENT, "EVENT: FW Debug Info %s\n",
|
|
(t_u8 *)pevent->event_buf);
|
|
wlan_recv_event(pmpriv, pevent->event_id, pevent);
|
|
pevent->event_id = 0; /* clear to avoid resending at end of fcn
|
|
*/
|
|
break;
|
|
case EVENT_TX_STATUS_REPORT:
|
|
PRINTM(MINFO, "EVENT: TX_STATUS\n");
|
|
pevent->event_id = MLAN_EVENT_ID_FW_TX_STATUS;
|
|
break;
|
|
case EVENT_BT_COEX_WLAN_PARA_CHANGE:
|
|
PRINTM(MEVENT, "EVENT: BT coex wlan param update\n");
|
|
wlan_bt_coex_wlan_param_update_event(pmpriv, pmbuf);
|
|
break;
|
|
case EVENT_EXCEED_MAX_P2P_CONN:
|
|
event_excd_p2p =
|
|
(event_exceed_max_p2p_conn *)(pmbuf->pbuf +
|
|
pmbuf->data_offset);
|
|
PRINTM(MEVENT, "EVENT: EXCEED MAX P2P CONNECTION\n");
|
|
PRINTM(MEVENT, "REQUEST P2P MAC: " MACSTR "\n",
|
|
MAC2STR(event_excd_p2p->peer_mac_addr));
|
|
pevent->event_id = MLAN_EVENT_ID_DRV_PASSTHRU;
|
|
break;
|
|
case EVENT_VDLL_IND:
|
|
wlan_process_vdll_event(pmpriv, pmbuf);
|
|
break;
|
|
|
|
case EVENT_FW_HANG_REPORT:
|
|
if (pmbuf->data_len < (sizeof(eventcause) + sizeof(t_u16))) {
|
|
PRINTM(MEVENT,
|
|
"EVENT: EVENT_FW_HANG_REPORT skip for len too short: %d\n",
|
|
pmbuf->data_len);
|
|
break;
|
|
}
|
|
PRINTM(MEVENT, "EVENT: EVENT_FW_HANG_REPORT reasoncode=%d\n",
|
|
wlan_le16_to_cpu(*(t_u16 *)(pmbuf->pbuf +
|
|
pmbuf->data_offset +
|
|
sizeof(eventcause))));
|
|
pmadapter->fw_hang_report = MTRUE;
|
|
wlan_recv_event(pmpriv, MLAN_EVENT_ID_DRV_DBG_DUMP, MNULL);
|
|
break;
|
|
case EVENT_WATCHDOG_TMOUT:
|
|
PRINTM(MEVENT, "EVENT: EVENT_WATCHDOG_TMOUT reasoncode=%d\n",
|
|
wlan_le16_to_cpu(*(t_u16 *)(pmbuf->pbuf +
|
|
pmbuf->data_offset +
|
|
sizeof(eventcause))));
|
|
pevent->event_id = MLAN_EVENT_ID_DRV_WIFI_STATUS;
|
|
pevent->event_len = sizeof(pevent->event_id) + sizeof(t_u16);
|
|
memcpy_ext(pmadapter, (t_u8 *)pevent->event_buf,
|
|
pmbuf->pbuf + pmbuf->data_offset +
|
|
sizeof(eventcause),
|
|
sizeof(t_u16), sizeof(t_u16));
|
|
break;
|
|
default:
|
|
pevent->event_id = MLAN_EVENT_ID_DRV_PASSTHRU;
|
|
break;
|
|
}
|
|
|
|
if (pevent->event_id) {
|
|
pevent->bss_index = pmpriv->bss_index;
|
|
pevent->event_len = pmbuf->data_len;
|
|
memcpy_ext(pmadapter, (t_u8 *)pevent->event_buf,
|
|
pmbuf->pbuf + pmbuf->data_offset, pevent->event_len,
|
|
pevent->event_len);
|
|
wlan_recv_event(pmpriv, pevent->event_id, pevent);
|
|
}
|
|
done:
|
|
if (event_buf)
|
|
pcb->moal_mfree(pmadapter->pmoal_handle, event_buf);
|
|
LEAVE();
|
|
return ret;
|
|
}
|
|
|
|
/**
|
|
* @brief This function issues commands to set uap max sta number
|
|
*
|
|
* @param priv A pointer to mlan_private structure
|
|
* @param uap_max_sta Max station number uAP can supported (per chip)
|
|
*
|
|
* @return MLAN_STATUS_SUCCESS or MLAN_STATUS_PENDING or MLAN_STATUS_FAILURE
|
|
*/
|
|
static mlan_status wlan_uap_set_uap_max_sta(pmlan_private pmpriv,
|
|
t_u8 uap_max_sta)
|
|
{
|
|
MrvlIEtypes_uap_max_sta_cnt_t tlv_uap_max_sta;
|
|
mlan_status ret = MLAN_STATUS_SUCCESS;
|
|
|
|
ENTER();
|
|
memset(pmpriv->adapter, &tlv_uap_max_sta, 0, sizeof(tlv_uap_max_sta));
|
|
tlv_uap_max_sta.header.type =
|
|
wlan_cpu_to_le16(TLV_TYPE_UAP_MAX_STA_CNT_PER_CHIP);
|
|
tlv_uap_max_sta.header.len = wlan_cpu_to_le16(sizeof(t_u16));
|
|
tlv_uap_max_sta.uap_max_sta = wlan_cpu_to_le16(uap_max_sta);
|
|
ret = wlan_prepare_cmd(pmpriv, HOST_CMD_APCMD_SYS_CONFIGURE,
|
|
HostCmd_ACT_GEN_SET, 0, MNULL, &tlv_uap_max_sta);
|
|
LEAVE();
|
|
return ret;
|
|
}
|
|
|
|
/**
|
|
* @brief This function issues commands to initialize firmware
|
|
*
|
|
* @param priv A pointer to mlan_private structure
|
|
* @param first_bss flag for first BSS
|
|
*
|
|
* @return MLAN_STATUS_SUCCESS or MLAN_STATUS_PENDING or MLAN_STATUS_FAILURE
|
|
*/
|
|
mlan_status wlan_ops_uap_init_cmd(t_void *priv, t_u8 first_bss)
|
|
{
|
|
mlan_status ret = MLAN_STATUS_SUCCESS;
|
|
pmlan_private pmpriv = (pmlan_private)priv;
|
|
t_u16 last_cmd = 0;
|
|
|
|
ENTER();
|
|
if (!pmpriv) {
|
|
LEAVE();
|
|
return MLAN_STATUS_FAILURE;
|
|
}
|
|
|
|
if (first_bss) {
|
|
if (wlan_adapter_init_cmd(pmpriv->adapter) ==
|
|
MLAN_STATUS_FAILURE) {
|
|
ret = MLAN_STATUS_FAILURE;
|
|
goto done;
|
|
}
|
|
}
|
|
if (pmpriv->adapter->init_para.uap_max_sta &&
|
|
(pmpriv->adapter->init_para.uap_max_sta <= MAX_STA_COUNT))
|
|
wlan_uap_set_uap_max_sta(
|
|
pmpriv, pmpriv->adapter->init_para.uap_max_sta);
|
|
|
|
ret = wlan_prepare_cmd(pmpriv, HOST_CMD_APCMD_SYS_CONFIGURE,
|
|
HostCmd_ACT_GEN_GET, 0, MNULL, MNULL);
|
|
if (ret) {
|
|
ret = MLAN_STATUS_FAILURE;
|
|
goto done;
|
|
}
|
|
last_cmd = HOST_CMD_APCMD_SYS_CONFIGURE;
|
|
/** set last_init_cmd */
|
|
if (last_cmd) {
|
|
pmpriv->adapter->last_init_cmd = last_cmd;
|
|
ret = MLAN_STATUS_PENDING;
|
|
}
|
|
done:
|
|
LEAVE();
|
|
return ret;
|
|
}
|