mirror of
https://github.com/nxp-imx/mwifiex.git
synced 2025-01-26 21:40:08 +00:00
983c0739ec
Corresponding firmware version: SDIO-UART W8987 Firmware version 16.92.21.p99.2 PCIE-UART W8997 Firmware version 16.92.21.p84.4 SDIO-UART W8997 Firmware version 16.92.21.p84.4 SDIO-UART IW416 Firmware version 16.92.21.p84.128 SDIO_UART IW612 Firmware version 18.99.2.p66 SDIO-UART W8801 Firmware version 14.92.36.p186 SDIO-UART W9098 Firmware version 17.92.1.p136.131 PCIE-UART W9098 Firmware version 17.92.1.p136.131 Signed-off-by: Sherry Sun <sherry.sun@nxp.com>
586 lines
17 KiB
C
586 lines
17 KiB
C
/**
|
|
* @file mlan_txrx.c
|
|
*
|
|
* @brief This file contains the handling of TX/RX in MLAN
|
|
*
|
|
*
|
|
* 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:
|
|
05/11/2009: initial version
|
|
************************************************************/
|
|
|
|
#include "mlan.h"
|
|
#ifdef STA_SUPPORT
|
|
#include "mlan_join.h"
|
|
#endif
|
|
#include "mlan_util.h"
|
|
#include "mlan_fw.h"
|
|
#include "mlan_main.h"
|
|
#include "mlan_wmm.h"
|
|
|
|
/********************************************************
|
|
Local Variables
|
|
********************************************************/
|
|
|
|
/********************************************************
|
|
Global Variables
|
|
********************************************************/
|
|
|
|
/********************************************************
|
|
Local Functions
|
|
********************************************************/
|
|
|
|
/********************************************************
|
|
Global Functions
|
|
********************************************************/
|
|
/**
|
|
* @brief This function processes the received buffer
|
|
*
|
|
* @param pmadapter A pointer to mlan_adapter
|
|
* @param pmbuf A pointer to the received buffer
|
|
*
|
|
* @return MLAN_STATUS_SUCCESS or MLAN_STATUS_FAILURE
|
|
*/
|
|
mlan_status wlan_handle_rx_packet(pmlan_adapter pmadapter, pmlan_buffer pmbuf)
|
|
{
|
|
mlan_status ret = MLAN_STATUS_SUCCESS;
|
|
pmlan_private priv = MNULL;
|
|
RxPD *prx_pd;
|
|
#ifdef DEBUG_LEVEL1
|
|
t_u32 sec = 0, usec = 0;
|
|
#endif
|
|
|
|
ENTER();
|
|
|
|
prx_pd = (RxPD *)(pmbuf->pbuf + pmbuf->data_offset);
|
|
/* Get the BSS number from RxPD, get corresponding priv */
|
|
priv = wlan_get_priv_by_id(pmadapter, prx_pd->bss_num & BSS_NUM_MASK,
|
|
prx_pd->bss_type);
|
|
if (!priv)
|
|
priv = wlan_get_priv(pmadapter, MLAN_BSS_ROLE_ANY);
|
|
if (!priv) {
|
|
ret = MLAN_STATUS_FAILURE;
|
|
goto done;
|
|
}
|
|
pmbuf->bss_index = priv->bss_index;
|
|
PRINTM_GET_SYS_TIME(MDATA, &sec, &usec);
|
|
PRINTM_NETINTF(MDATA, priv);
|
|
PRINTM(MDATA, "%lu.%06lu : Data <= FW\n", sec, usec);
|
|
ret = priv->ops.process_rx_packet(pmadapter, pmbuf);
|
|
|
|
done:
|
|
LEAVE();
|
|
return ret;
|
|
}
|
|
Stats_mcast_drv_t gmcast_stats = {0};
|
|
/* This flag is used to protect the mcast drv stat update
|
|
* when it's value is copied to provide to mlanutl
|
|
* MTRUE = Update is allowed
|
|
* MFALSE = Update is not allowed
|
|
*/
|
|
t_u8 mcast_drv_update_allow_flag = MTRUE;
|
|
#define DEST_MAC_OFFSET 10
|
|
#define CYCLE_START 1
|
|
#define FIVE_SEC 5000000 /* 1000000 usec = 1sec*/
|
|
/**
|
|
* @brief This function calculates the cycle delta and driver time delta
|
|
* for Mcast packets
|
|
*
|
|
* @param pmadapter A pointer to pmlan_adapter structure
|
|
* @param pmbuf A pointer to the mlan_buffer for process
|
|
*
|
|
* @return Nothing
|
|
*/
|
|
void wlan_drv_mcast_cycle_delay_calulation(pmlan_adapter pmadapter,
|
|
pmlan_buffer pmbuf)
|
|
{
|
|
static t_u32 prev_mcast_sec = 0;
|
|
static t_u32 prev_mcast_usec = 0;
|
|
t_u32 curr_ts_sec = 0;
|
|
t_u32 curr_ts_usec = 0;
|
|
t_u64 cycle_delta = 0;
|
|
t_u64 profile_delta = 0;
|
|
|
|
if (mcast_drv_update_allow_flag == MFALSE)
|
|
return;
|
|
/* Take current time */
|
|
if (pmadapter && pmadapter->pmoal_handle)
|
|
pmadapter->callbacks.moal_get_system_time(
|
|
pmadapter->pmoal_handle, &curr_ts_sec, &curr_ts_usec);
|
|
else
|
|
PRINTM(MERROR, "ERR: pmadapter or pmoal_handle NULL\n",
|
|
__func__);
|
|
|
|
if (curr_ts_sec || curr_ts_usec) {
|
|
/* Calculate profile delta */
|
|
profile_delta = (curr_ts_sec - pmbuf->in_ts_sec) * 1000000;
|
|
profile_delta += (t_s32)(curr_ts_usec - pmbuf->in_ts_usec);
|
|
|
|
if ((profile_delta >= 0) && (profile_delta <= 1000))
|
|
gmcast_stats.spent_time_under_1000usec++;
|
|
else if ((profile_delta > 1000) && (profile_delta <= 2000))
|
|
gmcast_stats.spent_time_over_1000usec++;
|
|
else if ((profile_delta > 2000) && (profile_delta <= 3000))
|
|
gmcast_stats.spent_time_over_2000usec++;
|
|
else if (profile_delta > 3000)
|
|
gmcast_stats.spent_time_over_3000usec++;
|
|
}
|
|
/* Process the start cycle data */
|
|
cycle_delta = (pmbuf->in_ts_sec - prev_mcast_sec) * 1000000;
|
|
cycle_delta += (t_s32)(pmbuf->in_ts_usec - prev_mcast_usec);
|
|
|
|
/* If start cycle delta is more than 5 sec ignore*/
|
|
if ((pmbuf->u.mc_tx_info.mc_pkt_flags & (1 << CYCLE_START)) &&
|
|
(cycle_delta < FIVE_SEC)) {
|
|
if ((cycle_delta >= 0) && (cycle_delta <= 2300))
|
|
gmcast_stats.cycle_recv_under_2300usec++;
|
|
if ((cycle_delta > 2300) && (cycle_delta <= 2900))
|
|
gmcast_stats.cycle_recv_in_time++;
|
|
if ((cycle_delta > 2900) && (cycle_delta <= 3500))
|
|
gmcast_stats.cycle_recv_over_2900usec++;
|
|
if ((cycle_delta > 3500) && (cycle_delta <= 5000))
|
|
gmcast_stats.cycle_recv_over_3500usec++;
|
|
if ((cycle_delta > 5000) && (cycle_delta <= 10000))
|
|
gmcast_stats.cycle_recv_over_5000usec++;
|
|
if ((cycle_delta > 10000) && (cycle_delta <= 15000))
|
|
gmcast_stats.cycle_recv_over_10000usec++;
|
|
if (cycle_delta > 15000)
|
|
gmcast_stats.cycle_recv_over_15000usec++;
|
|
}
|
|
/* Update the last received mcast cycle value */
|
|
prev_mcast_sec = pmbuf->in_ts_sec;
|
|
prev_mcast_usec = pmbuf->in_ts_usec;
|
|
}
|
|
/**
|
|
* @brief This function checks the conditions and sends packet to device
|
|
*
|
|
* @param priv A pointer to mlan_private structure
|
|
* @param pmbuf A pointer to the mlan_buffer for process
|
|
* @param tx_param A pointer to mlan_tx_param structure
|
|
*
|
|
* @return MLAN_STATUS_SUCCESS/MLAN_STATUS_PENDING --success, otherwise
|
|
* failure
|
|
*/
|
|
mlan_status wlan_process_tx(pmlan_private priv, pmlan_buffer pmbuf,
|
|
mlan_tx_param *tx_param)
|
|
{
|
|
mlan_status ret = MLAN_STATUS_SUCCESS;
|
|
pmlan_adapter pmadapter = priv->adapter;
|
|
t_u8 *head_ptr = MNULL;
|
|
#ifdef DEBUG_LEVEL1
|
|
t_u32 sec = 0, usec = 0;
|
|
#endif
|
|
#ifdef STA_SUPPORT
|
|
PTxPD plocal_tx_pd = MNULL;
|
|
#endif
|
|
t_u8 dest_mac_first_octet = 0;
|
|
|
|
ENTER();
|
|
head_ptr = (t_u8 *)priv->ops.process_txpd(priv, pmbuf);
|
|
if (!head_ptr) {
|
|
pmbuf->status_code = MLAN_ERROR_PKT_INVALID;
|
|
ret = MLAN_STATUS_FAILURE;
|
|
goto done;
|
|
}
|
|
#ifdef STA_SUPPORT
|
|
if (GET_BSS_ROLE(priv) == MLAN_BSS_ROLE_STA)
|
|
plocal_tx_pd = (TxPD *)(head_ptr + priv->intf_hr_len);
|
|
#endif
|
|
dest_mac_first_octet = *(head_ptr + priv->intf_hr_len + sizeof(TxPD) +
|
|
DEST_MAC_OFFSET);
|
|
|
|
if (dest_mac_first_octet & 0x01)
|
|
wlan_drv_mcast_cycle_delay_calulation(pmadapter, pmbuf);
|
|
if (pmadapter->tp_state_on)
|
|
pmadapter->callbacks.moal_tp_accounting(pmadapter->pmoal_handle,
|
|
pmbuf, 4);
|
|
if (pmadapter->tp_state_drop_point == 4)
|
|
goto done;
|
|
else {
|
|
ret = pmadapter->ops.host_to_card(priv, MLAN_TYPE_DATA, pmbuf,
|
|
tx_param);
|
|
}
|
|
done:
|
|
switch (ret) {
|
|
#ifdef USB
|
|
case MLAN_STATUS_PRESOURCE:
|
|
PRINTM(MINFO, "MLAN_STATUS_PRESOURCE is returned\n");
|
|
DBG_HEXDUMP(MDAT_D, "Tx", head_ptr + priv->intf_hr_len,
|
|
MIN(pmbuf->data_len + sizeof(TxPD),
|
|
MAX_DATA_DUMP_LEN));
|
|
break;
|
|
#endif
|
|
case MLAN_STATUS_RESOURCE:
|
|
#ifdef STA_SUPPORT
|
|
if ((GET_BSS_ROLE(priv) == MLAN_BSS_ROLE_STA) &&
|
|
pmadapter->pps_uapsd_mode &&
|
|
(pmadapter->tx_lock_flag == MTRUE)) {
|
|
pmadapter->tx_lock_flag = MFALSE;
|
|
if (plocal_tx_pd != MNULL)
|
|
plocal_tx_pd->flags = 0;
|
|
}
|
|
#endif
|
|
PRINTM(MINFO, "MLAN_STATUS_RESOURCE is returned\n");
|
|
break;
|
|
case MLAN_STATUS_FAILURE:
|
|
pmadapter->dbg.num_tx_host_to_card_failure++;
|
|
pmbuf->status_code = MLAN_ERROR_DATA_TX_FAIL;
|
|
wlan_write_data_complete(pmadapter, pmbuf, ret);
|
|
break;
|
|
case MLAN_STATUS_PENDING:
|
|
DBG_HEXDUMP(MDAT_D, "Tx", head_ptr + priv->intf_hr_len,
|
|
MIN(pmbuf->data_len + sizeof(TxPD),
|
|
MAX_DATA_DUMP_LEN));
|
|
break;
|
|
case MLAN_STATUS_SUCCESS:
|
|
DBG_HEXDUMP(MDAT_D, "Tx", head_ptr + priv->intf_hr_len,
|
|
MIN(pmbuf->data_len + sizeof(TxPD),
|
|
MAX_DATA_DUMP_LEN));
|
|
wlan_write_data_complete(pmadapter, pmbuf, ret);
|
|
break;
|
|
default:
|
|
break;
|
|
}
|
|
|
|
if ((ret == MLAN_STATUS_SUCCESS) || (ret == MLAN_STATUS_PENDING)) {
|
|
PRINTM_GET_SYS_TIME(MDATA, &sec, &usec);
|
|
PRINTM_NETINTF(MDATA, priv);
|
|
PRINTM(MDATA, "%lu.%06lu : Data => FW\n", sec, usec);
|
|
}
|
|
LEAVE();
|
|
return ret;
|
|
}
|
|
|
|
/**
|
|
* @brief Packet send completion handling
|
|
*
|
|
* @param pmadapter A pointer to mlan_adapter structure
|
|
* @param pmbuf A pointer to mlan_buffer structure
|
|
* @param status Callback status
|
|
*
|
|
* @return MLAN_STATUS_SUCCESS
|
|
*/
|
|
mlan_status wlan_write_data_complete(pmlan_adapter pmadapter,
|
|
pmlan_buffer pmbuf, mlan_status status)
|
|
{
|
|
mlan_status ret = MLAN_STATUS_SUCCESS;
|
|
pmlan_callbacks pcb;
|
|
|
|
ENTER();
|
|
|
|
MASSERT(pmadapter && pmbuf);
|
|
if (!pmadapter || !pmbuf) {
|
|
LEAVE();
|
|
return MLAN_STATUS_FAILURE;
|
|
}
|
|
|
|
pcb = &pmadapter->callbacks;
|
|
|
|
if ((pmbuf->buf_type == MLAN_BUF_TYPE_DATA) ||
|
|
(pmbuf->buf_type == MLAN_BUF_TYPE_RAW_DATA)) {
|
|
PRINTM(MINFO, "wlan_write_data_complete: DATA %p\n", pmbuf);
|
|
#if defined(USB)
|
|
if ((pmbuf->flags & MLAN_BUF_FLAG_USB_TX_AGGR) &&
|
|
pmbuf->use_count) {
|
|
pmlan_buffer pmbuf_next;
|
|
t_u32 i, use_count = pmbuf->use_count;
|
|
for (i = 0; i <= use_count; i++) {
|
|
pmbuf_next = pmbuf->pnext;
|
|
if (pmbuf->flags & MLAN_BUF_FLAG_MOAL_TX_BUF)
|
|
pcb->moal_send_packet_complete(
|
|
pmadapter->pmoal_handle, pmbuf,
|
|
status);
|
|
else
|
|
wlan_free_mlan_buffer(pmadapter, pmbuf);
|
|
pmbuf = pmbuf_next;
|
|
}
|
|
} else {
|
|
#endif
|
|
if (pmbuf->flags & MLAN_BUF_FLAG_MOAL_TX_BUF) {
|
|
/* pmbuf was allocated by MOAL */
|
|
pcb->moal_send_packet_complete(
|
|
pmadapter->pmoal_handle, pmbuf, status);
|
|
} else {
|
|
/* pmbuf was allocated by MLAN */
|
|
wlan_free_mlan_buffer(pmadapter, pmbuf);
|
|
}
|
|
#if defined(USB)
|
|
}
|
|
#endif
|
|
}
|
|
|
|
LEAVE();
|
|
return ret;
|
|
}
|
|
|
|
/**
|
|
* @brief Packet receive completion callback handler
|
|
*
|
|
* @param pmadapter A pointer to mlan_adapter structure
|
|
* @param pmbuf A pointer to mlan_buffer structure
|
|
* @param status Callback status
|
|
*
|
|
* @return MLAN_STATUS_SUCCESS
|
|
*/
|
|
mlan_status wlan_recv_packet_complete(pmlan_adapter pmadapter,
|
|
pmlan_buffer pmbuf, mlan_status status)
|
|
{
|
|
mlan_status ret = MLAN_STATUS_SUCCESS;
|
|
|
|
ENTER();
|
|
|
|
MASSERT(pmadapter && pmbuf);
|
|
if (!pmadapter || !pmbuf) {
|
|
LEAVE();
|
|
return MLAN_STATUS_FAILURE;
|
|
}
|
|
|
|
MASSERT(pmbuf->bss_index < pmadapter->priv_num);
|
|
|
|
if (pmbuf->pparent) {
|
|
/** we will free the pparaent at the end of deaggr */
|
|
wlan_free_mlan_buffer(pmadapter, pmbuf);
|
|
} else {
|
|
pmadapter->ops.data_complete(pmadapter, pmbuf, status);
|
|
}
|
|
|
|
LEAVE();
|
|
return ret;
|
|
}
|
|
|
|
/**
|
|
* @brief Add packet to Bypass TX queue
|
|
*
|
|
* @param pmadapter Pointer to the mlan_adapter driver data struct
|
|
* @param pmbuf Pointer to the mlan_buffer data struct
|
|
*
|
|
* @return N/A
|
|
*/
|
|
t_void wlan_add_buf_bypass_txqueue(mlan_adapter *pmadapter, pmlan_buffer pmbuf)
|
|
{
|
|
pmlan_private priv = pmadapter->priv[pmbuf->bss_index];
|
|
ENTER();
|
|
|
|
if (pmbuf->buf_type != MLAN_BUF_TYPE_RAW_DATA)
|
|
pmbuf->buf_type = MLAN_BUF_TYPE_DATA;
|
|
pmadapter->callbacks.moal_spin_lock(pmadapter->pmoal_handle,
|
|
priv->bypass_txq.plock);
|
|
pmadapter->bypass_pkt_count++;
|
|
util_enqueue_list_tail(pmadapter->pmoal_handle, &priv->bypass_txq,
|
|
(pmlan_linked_list)pmbuf, MNULL, MNULL);
|
|
pmadapter->callbacks.moal_spin_unlock(pmadapter->pmoal_handle,
|
|
priv->bypass_txq.plock);
|
|
LEAVE();
|
|
}
|
|
|
|
/**
|
|
* @brief Check if packets are available in Bypass TX queue
|
|
*
|
|
* @param pmadapter Pointer to the mlan_adapter driver data struct
|
|
*
|
|
* @return MFALSE if not empty; MTRUE if empty
|
|
*/
|
|
INLINE t_u8 wlan_bypass_tx_list_empty(mlan_adapter *pmadapter)
|
|
{
|
|
#if defined(USB)
|
|
if (IS_USB(pmadapter->card_type)) {
|
|
pmlan_callbacks pcb = &pmadapter->callbacks;
|
|
pmlan_private priv;
|
|
int j = 0;
|
|
for (j = 0; j < pmadapter->priv_num; ++j) {
|
|
priv = pmadapter->priv[j];
|
|
if (priv) {
|
|
if (!wlan_is_port_ready(pmadapter,
|
|
priv->port_index))
|
|
continue;
|
|
if (util_peek_list(pmadapter->pmoal_handle,
|
|
&priv->bypass_txq,
|
|
pcb->moal_spin_lock,
|
|
pcb->moal_spin_unlock))
|
|
return MFALSE;
|
|
}
|
|
}
|
|
return MTRUE;
|
|
} else
|
|
#endif
|
|
return (pmadapter->bypass_pkt_count) ? MFALSE : MTRUE;
|
|
}
|
|
|
|
/**
|
|
* @brief Clean up the By-pass TX queue
|
|
*
|
|
* @param priv Pointer to the mlan_private data struct
|
|
*
|
|
* @return N/A
|
|
*/
|
|
t_void wlan_cleanup_bypass_txq(mlan_private *priv)
|
|
{
|
|
pmlan_buffer pmbuf;
|
|
mlan_adapter *pmadapter = priv->adapter;
|
|
ENTER();
|
|
pmadapter->callbacks.moal_spin_lock(pmadapter->pmoal_handle,
|
|
priv->bypass_txq.plock);
|
|
while ((pmbuf = (pmlan_buffer)util_peek_list(pmadapter->pmoal_handle,
|
|
&priv->bypass_txq, MNULL,
|
|
MNULL))) {
|
|
util_unlink_list(pmadapter->pmoal_handle, &priv->bypass_txq,
|
|
(pmlan_linked_list)pmbuf, MNULL, MNULL);
|
|
wlan_write_data_complete(pmadapter, pmbuf, MLAN_STATUS_FAILURE);
|
|
pmadapter->bypass_pkt_count--;
|
|
}
|
|
pmadapter->callbacks.moal_spin_unlock(pmadapter->pmoal_handle,
|
|
priv->bypass_txq.plock);
|
|
LEAVE();
|
|
}
|
|
|
|
/**
|
|
* @brief Transmit the By-passed packet awaiting in by-pass queue
|
|
*
|
|
* @param pmadapter Pointer to the mlan_adapter driver data struct
|
|
*
|
|
* @return N/A
|
|
*/
|
|
t_void wlan_process_bypass_tx(pmlan_adapter pmadapter)
|
|
{
|
|
pmlan_buffer pmbuf;
|
|
mlan_tx_param tx_param;
|
|
mlan_status status = MLAN_STATUS_SUCCESS;
|
|
pmlan_private priv;
|
|
int j = 0;
|
|
ENTER();
|
|
do {
|
|
for (j = 0; j < pmadapter->priv_num; ++j) {
|
|
priv = pmadapter->priv[j];
|
|
if (priv) {
|
|
#if defined(USB)
|
|
if (!wlan_is_port_ready(pmadapter,
|
|
priv->port_index))
|
|
continue;
|
|
#endif
|
|
pmbuf = (pmlan_buffer)util_dequeue_list(
|
|
pmadapter->pmoal_handle,
|
|
&priv->bypass_txq,
|
|
pmadapter->callbacks.moal_spin_lock,
|
|
pmadapter->callbacks.moal_spin_unlock);
|
|
if (pmbuf) {
|
|
pmadapter->callbacks.moal_spin_lock(
|
|
pmadapter->pmoal_handle,
|
|
priv->bypass_txq.plock);
|
|
pmadapter->bypass_pkt_count--;
|
|
pmadapter->callbacks.moal_spin_unlock(
|
|
pmadapter->pmoal_handle,
|
|
priv->bypass_txq.plock);
|
|
PRINTM(MINFO,
|
|
"Dequeuing bypassed packet %p\n",
|
|
pmbuf);
|
|
if (wlan_bypass_tx_list_empty(
|
|
pmadapter))
|
|
tx_param.next_pkt_len = 0;
|
|
else
|
|
tx_param.next_pkt_len =
|
|
pmbuf->data_len;
|
|
status = wlan_process_tx(
|
|
pmadapter->priv[pmbuf->bss_index],
|
|
pmbuf, &tx_param);
|
|
|
|
if (status == MLAN_STATUS_RESOURCE) {
|
|
/* Queue the packet again so
|
|
* that it will be TX'ed later
|
|
*/
|
|
pmadapter->callbacks.moal_spin_lock(
|
|
pmadapter->pmoal_handle,
|
|
priv->bypass_txq.plock);
|
|
pmadapter->bypass_pkt_count++;
|
|
util_enqueue_list_head(
|
|
pmadapter->pmoal_handle,
|
|
&priv->bypass_txq,
|
|
(pmlan_linked_list)pmbuf,
|
|
pmadapter->callbacks
|
|
.moal_spin_lock,
|
|
pmadapter->callbacks
|
|
.moal_spin_unlock);
|
|
pmadapter->callbacks.moal_spin_unlock(
|
|
pmadapter->pmoal_handle,
|
|
priv->bypass_txq.plock);
|
|
}
|
|
break;
|
|
} else {
|
|
PRINTM(MINFO, "Nothing to send\n");
|
|
}
|
|
}
|
|
}
|
|
} while (!pmadapter->data_sent && !pmadapter->tx_lock_flag &&
|
|
!wlan_bypass_tx_list_empty(pmadapter));
|
|
LEAVE();
|
|
}
|
|
|
|
/**
|
|
* @brief This function will convert 802.11 header to 802.3 header
|
|
and save the backhaul station aid to pmbuf
|
|
*
|
|
* @param priv A pointer to mlan_private
|
|
* @param pmbuf A pointer to mlan_buffer
|
|
* @param prx_pd A pointer to RxPD
|
|
*
|
|
* @return MLAN_STATUS_PENDING --success, otherwise fail
|
|
*/
|
|
mlan_status wlan_check_easymesh_pkt(mlan_private *priv, pmlan_buffer pmbuf,
|
|
RxPD *prx_pd)
|
|
{
|
|
Eth803Hdr_t *eth_header = MNULL;
|
|
sta_node *sta_ptr = MNULL;
|
|
mlan_status ret = MLAN_STATUS_SUCCESS;
|
|
t_u8 *pos = MNULL;
|
|
t_u32 tmp = 0;
|
|
|
|
ENTER();
|
|
|
|
pos = (t_u8 *)prx_pd + prx_pd->rx_pkt_offset;
|
|
eth_header = (Eth803Hdr_t *)pos;
|
|
|
|
PRINTM(MDAT_D,
|
|
"Rx Easymesh ETH header destination address: " FULL_MACSTR
|
|
"\nETH header source address: " FULL_MACSTR "\n",
|
|
FULL_MAC2STR(eth_header->dest_addr),
|
|
FULL_MAC2STR(eth_header->src_addr));
|
|
|
|
if (priv->bss_type == MLAN_BSS_TYPE_UAP) {
|
|
pmbuf->flags |= MLAN_BUF_FLAG_EASYMESH;
|
|
memcpy_ext(priv->adapter, pmbuf->mac, prx_pd->ta_mac,
|
|
MLAN_MAC_ADDR_LENGTH, MLAN_MAC_ADDR_LENGTH);
|
|
|
|
sta_ptr = wlan_get_station_entry(priv, prx_pd->ta_mac);
|
|
if (!sta_ptr) {
|
|
PRINTM(MERROR,
|
|
"Easymesh Error! Can't find station in the station list\n");
|
|
ret = MLAN_STATUS_FAILURE;
|
|
goto done;
|
|
}
|
|
|
|
/* Save station aid to pmbuf and send it to moal */
|
|
tmp = (t_u32)sta_ptr->aid;
|
|
pmbuf->priority |= (tmp << 24);
|
|
PRINTM(MDAT_D, "Easymesh: Rx for VLAN " FULL_MACSTR "\n",
|
|
FULL_MAC2STR(prx_pd->ta_mac));
|
|
}
|
|
done:
|
|
LEAVE();
|
|
return ret;
|
|
}
|