mirror of
https://github.com/nxp-imx/mwifiex.git
synced 2025-01-15 16:25:35 +00:00
3df11f19c8
Reason: integrate WCS 2024 Q3 CF release patch, WiFi driver FW info: SDIO_WLAN_UART_BT_8997_16.92.21.p137.1_16.92.21.p119.3 PCIE_WLAN_UART_BT_8997_16.92.21.p137.1_16.92.21.p119.3 PCIE_WLAN_UART_BT_9098_17.92.1.p149.44_17.92.1.p149.44 SDIO_WLAN_UART_BT_8987_16.92.21.p137_16.92.21.p137 SDIO_WLAN_UART_BT_9098_17.92.1.p149.44_17.92.1.p149.44 SDIO_WLAN_UART_BT_IW416_16.92.21.p137_16.92.21.p137 SDIO_WLAN_UART_BT_IW612_18.99.3.p15_18.99.3.p15 Change-Id: Ifb7bb00203eb5dbc614a885e4483ed8323d3b0a4 Signed-off-by: yunjie <yunjie.jia@nxp.com>
1028 lines
31 KiB
C
1028 lines
31 KiB
C
/** @file mlan_11n_aggr.c
|
|
*
|
|
* @brief This file contains functions for 11n Aggregation.
|
|
*
|
|
*
|
|
* Copyright 2008-2021, 2024 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:
|
|
11/10/2008: initial version
|
|
********************************************************/
|
|
|
|
#include "mlan.h"
|
|
#include "mlan_join.h"
|
|
#include "mlan_util.h"
|
|
#include "mlan_fw.h"
|
|
#include "mlan_main.h"
|
|
#include "mlan_wmm.h"
|
|
#include "mlan_11n.h"
|
|
#include "mlan_11n_aggr.h"
|
|
|
|
/********************************************************
|
|
Local Variables
|
|
********************************************************/
|
|
|
|
/********************************************************
|
|
Global Variables
|
|
********************************************************/
|
|
|
|
/********************************************************
|
|
Local Functions
|
|
********************************************************/
|
|
/**
|
|
* @brief Aggregate individual packets into one AMSDU packet
|
|
*
|
|
* @param pmadapter A pointer to mlan_adapter structure
|
|
* @param amsdu_buf A pointer to packet buffer
|
|
* @param data A pointer to aggregated data packet being formed
|
|
* @param pkt_len Length of current packet to aggregate
|
|
* @param pad Pad
|
|
*
|
|
* @return Final packet size
|
|
*/
|
|
static int wlan_11n_form_amsdu_pkt(pmlan_adapter pmadapter, t_u8 *amsdu_buf,
|
|
t_u8 *data, int pkt_len, int *pad)
|
|
{
|
|
int dt_offset, amsdu_buf_offset;
|
|
Rfc1042Hdr_t snap = {
|
|
0xaa, /* LLC DSAP */
|
|
0xaa, /* LLC SSAP */
|
|
0x03, /* LLC CTRL */
|
|
{0x00, 0x00, 0x00}, /* SNAP OUI */
|
|
0x0000 /* SNAP type */
|
|
/*
|
|
* This field will be overwritten
|
|
* later with ethertype
|
|
*/
|
|
};
|
|
|
|
ENTER();
|
|
|
|
memcpy_ext(pmadapter, amsdu_buf, data, (MLAN_MAC_ADDR_LENGTH)*2,
|
|
(MLAN_MAC_ADDR_LENGTH)*2);
|
|
dt_offset = amsdu_buf_offset = (MLAN_MAC_ADDR_LENGTH)*2;
|
|
|
|
snap.snap_type = *(t_u16 *)(data + dt_offset);
|
|
dt_offset += sizeof(t_u16);
|
|
*(t_u16 *)(amsdu_buf + amsdu_buf_offset) =
|
|
mlan_htons(pkt_len + LLC_SNAP_LEN -
|
|
((2 * MLAN_MAC_ADDR_LENGTH) + sizeof(t_u16)));
|
|
amsdu_buf_offset += sizeof(t_u16);
|
|
memcpy_ext(pmadapter, amsdu_buf + amsdu_buf_offset, &snap, LLC_SNAP_LEN,
|
|
LLC_SNAP_LEN);
|
|
amsdu_buf_offset += LLC_SNAP_LEN;
|
|
|
|
memcpy_ext(pmadapter, amsdu_buf + amsdu_buf_offset, data + dt_offset,
|
|
pkt_len - dt_offset, pkt_len - dt_offset);
|
|
*pad = (((pkt_len + LLC_SNAP_LEN) & 3)) ?
|
|
(4 - (((pkt_len + LLC_SNAP_LEN)) & 3)) :
|
|
0;
|
|
|
|
LEAVE();
|
|
return pkt_len + LLC_SNAP_LEN + *pad;
|
|
}
|
|
|
|
/**
|
|
* @brief Add TxPD to AMSDU header
|
|
*
|
|
* @param priv A pointer to mlan_private structure
|
|
* @param mbuf Pointer to buffer where the TxPD will be formed
|
|
*
|
|
* @return N/A
|
|
*/
|
|
static void wlan_11n_form_amsdu_txpd(mlan_private *priv, mlan_buffer *mbuf)
|
|
{
|
|
TxPD *ptx_pd;
|
|
mlan_adapter *pmadapter = priv->adapter;
|
|
|
|
ENTER();
|
|
|
|
ptx_pd = (TxPD *)mbuf->pbuf;
|
|
// coverity[bad_memset:SUPPRESS]
|
|
memset(pmadapter, ptx_pd, 0, Tx_PD_SIZEOF(pmadapter));
|
|
|
|
/*
|
|
* Original priority has been overwritten
|
|
*/
|
|
ptx_pd->priority = (t_u8)mbuf->priority;
|
|
ptx_pd->bss_num = GET_BSS_NUM(priv);
|
|
ptx_pd->bss_type = priv->bss_type;
|
|
/* Always zero as the data is followed by TxPD */
|
|
ptx_pd->tx_pkt_offset = Tx_PD_SIZEOF(pmadapter);
|
|
ptx_pd->tx_pkt_type = PKT_TYPE_AMSDU;
|
|
if (mbuf->flags & MLAN_BUF_FLAG_TDLS)
|
|
ptx_pd->flags = MRVDRV_TxPD_FLAGS_TDLS_PACKET;
|
|
|
|
if (mbuf->flags & MLAN_BUF_FLAG_EASYMESH) {
|
|
ptx_pd->flags |= MRVDRV_TxPD_FLAGS_EASYMESH;
|
|
memcpy_ext(priv->adapter, ptx_pd->ra_mac, mbuf->mac,
|
|
MLAN_MAC_ADDR_LENGTH, MLAN_MAC_ADDR_LENGTH);
|
|
}
|
|
if (ptx_pd->tx_control == 0)
|
|
/* TxCtrl set by user or default */
|
|
ptx_pd->tx_control = priv->pkt_tx_ctrl;
|
|
|
|
endian_convert_TxPD(ptx_pd);
|
|
|
|
LEAVE();
|
|
}
|
|
|
|
/**
|
|
* @brief free pkts in amsdu_txq
|
|
*
|
|
* @param pmadapter A pointer to mlan_adapter structure
|
|
*
|
|
* @return N/A
|
|
*/
|
|
static INLINE void wlan_free_amsdu_txq(pmlan_adapter pmadapter)
|
|
{
|
|
pmlan_buffer pmbuf;
|
|
|
|
ENTER();
|
|
while ((pmbuf = (pmlan_buffer)util_peek_list(pmadapter->pmoal_handle,
|
|
&pmadapter->amsdu_txq,
|
|
MNULL, MNULL))) {
|
|
util_unlink_list(pmadapter->pmoal_handle, &pmadapter->amsdu_txq,
|
|
(pmlan_linked_list)pmbuf, MNULL, MNULL);
|
|
wlan_write_data_complete(pmadapter, pmbuf, MLAN_STATUS_FAILURE);
|
|
}
|
|
LEAVE();
|
|
}
|
|
|
|
#ifdef PCIEAW693
|
|
/**
|
|
* @brief Add TxPD to AMSDU header
|
|
*
|
|
* @param priv A pointer to mlan_private structure
|
|
* @param pmbuf A pointer to buffer where the TxPD will be formed
|
|
* @param amsdu_pkt_len amsdu packet length
|
|
*
|
|
* @return increased length (TxPD + intf_hr_len + padding for DMA
|
|
* alignment)
|
|
*/
|
|
static t_u16 wlan_form_amsdu_txpd(mlan_private *priv, mlan_buffer *pmbuf,
|
|
t_u16 amsdu_pkt_len)
|
|
{
|
|
mlan_adapter *pmadapter = priv->adapter;
|
|
TxPD *ptx_pd;
|
|
t_u8 *head_ptr = MNULL;
|
|
t_u16 data_len = pmbuf->data_len;
|
|
ENTER();
|
|
|
|
head_ptr = pmbuf->pbuf + pmbuf->data_offset - Tx_PD_SIZEOF(pmadapter) -
|
|
priv->intf_hr_len;
|
|
ptx_pd = (TxPD *)(head_ptr + priv->intf_hr_len);
|
|
// coverity[bad_memset:SUPPRESS]
|
|
memset(pmadapter, ptx_pd, 0, Tx_PD_SIZEOF(pmadapter));
|
|
|
|
/* Set the BSS number to TxPD */
|
|
ptx_pd->bss_num = GET_BSS_NUM(priv);
|
|
ptx_pd->bss_type = priv->bss_type;
|
|
ptx_pd->priority = (t_u8)pmbuf->priority;
|
|
ptx_pd->tx_pkt_type = PKT_TYPE_AMSDU;
|
|
ptx_pd->tx_pkt_length = amsdu_pkt_len;
|
|
|
|
ptx_pd->tx_pkt_offset = (t_u16)((t_ptr)pmbuf->pbuf +
|
|
pmbuf->data_offset - (t_ptr)ptx_pd);
|
|
if (pmbuf->flags & MLAN_BUF_FLAG_TDLS)
|
|
ptx_pd->flags = MRVDRV_TxPD_FLAGS_TDLS_PACKET;
|
|
if (ptx_pd->tx_control == 0)
|
|
/* TxCtrl set by user or default */
|
|
ptx_pd->tx_control = priv->pkt_tx_ctrl;
|
|
|
|
if (pmbuf->flags & MLAN_BUF_FLAG_EASYMESH) {
|
|
ptx_pd->flags |= MRVDRV_TxPD_FLAGS_EASYMESH;
|
|
memcpy_ext(priv->adapter, ptx_pd->ra_mac, pmbuf->mac,
|
|
MLAN_MAC_ADDR_LENGTH, MLAN_MAC_ADDR_LENGTH);
|
|
}
|
|
|
|
endian_convert_TxPD(ptx_pd);
|
|
|
|
/* Adjust the data offset and length to include TxPD in pmbuf */
|
|
pmbuf->data_len += pmbuf->data_offset;
|
|
pmbuf->data_offset = (t_u32)(head_ptr - pmbuf->pbuf);
|
|
pmbuf->data_len -= pmbuf->data_offset;
|
|
|
|
PRINTM(MDATA, "amsdu_pkt_len=%d, extra_len=%d\n", amsdu_pkt_len,
|
|
pmbuf->data_len - data_len);
|
|
DBG_HEXDUMP(MDAT_D, "AMSDU TxPD", ptx_pd, Tx_PD_SIZEOF(pmadapter));
|
|
|
|
LEAVE();
|
|
return (pmbuf->data_len - data_len);
|
|
}
|
|
|
|
/**
|
|
* @brief Form amsdu subframe
|
|
*
|
|
* @param pmadapter A pointer to mlan_adapter structure
|
|
* @param pmbuf A pointer to mlan_buffer
|
|
* @param pad Pad
|
|
*
|
|
* @return Final packet size
|
|
*/
|
|
static int wlan_form_amsdu_subframe(pmlan_adapter pmadapter, mlan_buffer *pmbuf,
|
|
int *pad)
|
|
{
|
|
int dt_offset, amsdu_buf_offset;
|
|
Rfc1042Hdr_t snap = {
|
|
0xaa, /* LLC DSAP */
|
|
0xaa, /* LLC SSAP */
|
|
0x03, /* LLC CTRL */
|
|
{0x00, 0x00, 0x00}, /* SNAP OUI */
|
|
0x0000 /* SNAP type */
|
|
/*
|
|
* This field will be overwritten
|
|
* later with ethertype
|
|
*/
|
|
};
|
|
t_u8 *amsdu_buf = MNULL;
|
|
t_u8 *data = pmbuf->pbuf + pmbuf->data_offset;
|
|
int pkt_len = pmbuf->data_len;
|
|
|
|
ENTER();
|
|
|
|
pmbuf->data_offset -= sizeof(Rfc1042Hdr_t);
|
|
pmbuf->data_len += sizeof(Rfc1042Hdr_t);
|
|
|
|
amsdu_buf = pmbuf->pbuf + pmbuf->data_offset;
|
|
|
|
memcpy_ext(pmadapter, amsdu_buf, data, (MLAN_MAC_ADDR_LENGTH)*2,
|
|
(MLAN_MAC_ADDR_LENGTH)*2);
|
|
dt_offset = amsdu_buf_offset = (MLAN_MAC_ADDR_LENGTH)*2;
|
|
|
|
snap.snap_type = *(t_u16 *)(data + dt_offset);
|
|
dt_offset += sizeof(t_u16);
|
|
*(t_u16 *)(amsdu_buf + amsdu_buf_offset) =
|
|
mlan_htons(pkt_len + LLC_SNAP_LEN -
|
|
((2 * MLAN_MAC_ADDR_LENGTH) + sizeof(t_u16)));
|
|
amsdu_buf_offset += sizeof(t_u16);
|
|
memcpy_ext(pmadapter, amsdu_buf + amsdu_buf_offset, &snap, LLC_SNAP_LEN,
|
|
LLC_SNAP_LEN);
|
|
amsdu_buf_offset += LLC_SNAP_LEN;
|
|
*pad = (((pkt_len + LLC_SNAP_LEN) & 3)) ?
|
|
(4 - (((pkt_len + LLC_SNAP_LEN)) & 3)) :
|
|
0;
|
|
pmbuf->data_len += *pad;
|
|
|
|
DBG_HEXDUMP(MDAT_D, "AMSDU subframe", pmbuf->pbuf + pmbuf->data_offset,
|
|
MIN(pmbuf->data_len, MAX_DATA_DUMP_LEN));
|
|
PRINTM(MDATA, "AMSDU subframe len=%d\n", pkt_len + LLC_SNAP_LEN + *pad);
|
|
LEAVE();
|
|
return pkt_len + LLC_SNAP_LEN + *pad;
|
|
}
|
|
#endif
|
|
|
|
/**
|
|
* @brief Update the TxPktLength field in TxPD after the complete AMSDU
|
|
* packet is formed
|
|
*
|
|
* @param priv A pointer to mlan_private structure
|
|
* @param mbuf TxPD buffer
|
|
*
|
|
* @return N/A
|
|
*/
|
|
static INLINE void wlan_11n_update_pktlen_amsdu_txpd(mlan_private *priv,
|
|
pmlan_buffer mbuf)
|
|
{
|
|
TxPD *ptx_pd;
|
|
ENTER();
|
|
|
|
ptx_pd = (TxPD *)mbuf->pbuf;
|
|
ptx_pd->tx_pkt_length = (t_u16)wlan_cpu_to_le16(
|
|
mbuf->data_len - Tx_PD_SIZEOF(priv->adapter));
|
|
ptx_pd->pkt_delay_2ms =
|
|
wlan_wmm_compute_driver_packet_delay(priv, mbuf);
|
|
|
|
#ifdef STA_SUPPORT
|
|
if ((GET_BSS_ROLE(priv) == MLAN_BSS_ROLE_STA) &&
|
|
(priv->adapter->pps_uapsd_mode)) {
|
|
if (MTRUE == wlan_check_last_packet_indication(priv)) {
|
|
priv->adapter->tx_lock_flag = MTRUE;
|
|
ptx_pd->flags |= MRVDRV_TxPD_POWER_MGMT_LAST_PACKET;
|
|
}
|
|
}
|
|
#endif /* STA_SUPPORT */
|
|
LEAVE();
|
|
}
|
|
|
|
/**
|
|
* @brief check if UAP AMSDU packet need forward out to connected peers
|
|
*
|
|
* @param priv A pointer to mlan_private
|
|
*
|
|
* @return MTRUE--packet need forward
|
|
*
|
|
*/
|
|
static t_u8 wlan_uap_check_forward(mlan_private *priv, Eth803Hdr_t *hdr)
|
|
{
|
|
/** include multicast packet */
|
|
if (hdr->dest_addr[0] & 0x01)
|
|
return MTRUE;
|
|
/** include unicast packet to another station */
|
|
if (wlan_get_station_entry(priv, hdr->dest_addr))
|
|
return MTRUE;
|
|
return MFALSE;
|
|
}
|
|
|
|
/**
|
|
* @brief Get number of aggregated packets
|
|
*
|
|
* @param priv A pointer to mlan_private structure
|
|
* @param data A pointer to packet data
|
|
* @param total_pkt_len Total packet length
|
|
* @param forward A pointer forward flag
|
|
*
|
|
* @return Number of packets
|
|
*/
|
|
static int wlan_11n_get_num_aggrpkts(mlan_private *priv, t_u8 *data,
|
|
int total_pkt_len, t_u8 *forward)
|
|
{
|
|
int pkt_count = 0, pkt_len, pad;
|
|
t_u8 hdr_len = sizeof(Eth803Hdr_t);
|
|
|
|
t_u8 forward_flag = MFALSE;
|
|
|
|
ENTER();
|
|
while (total_pkt_len >= hdr_len) {
|
|
if (priv->bss_role == MLAN_BSS_ROLE_UAP &&
|
|
wlan_uap_check_forward(priv, (Eth803Hdr_t *)data))
|
|
forward_flag = MTRUE;
|
|
/* Length will be in network format, change it to host */
|
|
pkt_len = mlan_ntohs(
|
|
(*(t_u16 *)(data + (2 * MLAN_MAC_ADDR_LENGTH))));
|
|
if (pkt_len > total_pkt_len) {
|
|
PRINTM(MERROR, "Error in packet length.\n");
|
|
break;
|
|
}
|
|
|
|
pad = (((pkt_len + sizeof(Eth803Hdr_t)) & 3)) ?
|
|
(4 - ((pkt_len + sizeof(Eth803Hdr_t)) & 3)) :
|
|
0;
|
|
data += pkt_len + pad + sizeof(Eth803Hdr_t);
|
|
total_pkt_len -= pkt_len + pad + sizeof(Eth803Hdr_t);
|
|
++pkt_count;
|
|
}
|
|
*forward = forward_flag;
|
|
LEAVE();
|
|
return pkt_count;
|
|
}
|
|
|
|
/********************************************************
|
|
Global Functions
|
|
********************************************************/
|
|
|
|
/**
|
|
* @brief Deaggregate the received AMSDU packet
|
|
*
|
|
* @param priv A pointer to mlan_private structure
|
|
* @param pmbuf A pointer to aggregated data packet
|
|
*
|
|
* @return MLAN_STATUS_SUCCESS --success, otherwise fail
|
|
*/
|
|
mlan_status wlan_11n_deaggregate_pkt(mlan_private *priv, pmlan_buffer pmbuf)
|
|
{
|
|
t_u16 pkt_len;
|
|
int total_pkt_len;
|
|
t_u8 *data;
|
|
mlan_adapter *pmadapter = priv->adapter;
|
|
t_u32 max_rx_data_size = MLAN_RX_DATA_BUF_SIZE;
|
|
int pad;
|
|
mlan_status ret = MLAN_STATUS_FAILURE;
|
|
RxPacketHdr_t *prx_pkt;
|
|
mlan_buffer *daggr_mbuf = MNULL;
|
|
t_u8 rfc1042_eth_hdr[MLAN_MAC_ADDR_LENGTH] = {0xaa, 0xaa, 0x03,
|
|
0x00, 0x00, 0x00};
|
|
t_u8 hdr_len = sizeof(Eth803Hdr_t);
|
|
t_u8 forward = MFALSE;
|
|
t_u8 eapol_type[2] = {0x88, 0x8e};
|
|
t_u8 tdls_action_type[2] = {0x89, 0x0d};
|
|
t_u32 in_ts_sec, in_ts_usec;
|
|
t_u32 out_ts_sec, out_ts_usec;
|
|
t_u32 in_copy_ts_sec, in_copy_ts_usec;
|
|
t_u32 out_copy_ts_sec, out_copy_ts_usec;
|
|
t_u32 copy_delay = 0;
|
|
t_u32 delay = 0;
|
|
t_u8 num_subframes = 0;
|
|
|
|
ENTER();
|
|
|
|
data = (t_u8 *)(pmbuf->pbuf + pmbuf->data_offset);
|
|
total_pkt_len = pmbuf->data_len;
|
|
|
|
/* Sanity test */
|
|
#if defined(USB)
|
|
if (IS_USB(pmadapter->card_type) &&
|
|
pmadapter->pcard_usb->usb_rx_deaggr.aggr_ctrl.enable) {
|
|
max_rx_data_size =
|
|
pmadapter->pcard_usb->usb_rx_deaggr.aggr_ctrl.aggr_max;
|
|
if (pmadapter->pcard_usb->usb_rx_deaggr.aggr_ctrl.aggr_mode ==
|
|
MLAN_USB_AGGR_MODE_NUM) {
|
|
max_rx_data_size *=
|
|
MAX(MLAN_USB_MAX_PKT_SIZE,
|
|
pmadapter->pcard_usb->usb_rx_deaggr
|
|
.aggr_ctrl.aggr_align);
|
|
max_rx_data_size =
|
|
MAX(max_rx_data_size, MLAN_RX_DATA_BUF_SIZE);
|
|
}
|
|
}
|
|
#endif
|
|
if (total_pkt_len > (int)max_rx_data_size) {
|
|
PRINTM(MERROR,
|
|
"Total packet length greater than tx buffer"
|
|
" size %d\n",
|
|
total_pkt_len);
|
|
goto done;
|
|
}
|
|
if (pmadapter->tp_state_on)
|
|
pmadapter->callbacks.moal_get_system_time(
|
|
pmadapter->pmoal_handle, &in_ts_sec, &in_ts_usec);
|
|
num_subframes = pmbuf->use_count =
|
|
wlan_11n_get_num_aggrpkts(priv, data, total_pkt_len, &forward);
|
|
|
|
// rx_trace 7
|
|
if (pmadapter->tp_state_on) {
|
|
pmadapter->callbacks.moal_tp_accounting(
|
|
pmadapter->pmoal_handle, pmbuf, 7 /*RX_DROP_P3*/);
|
|
pmadapter->callbacks.moal_tp_accounting_rx_param(
|
|
pmadapter->pmoal_handle, 4, pmbuf->use_count);
|
|
}
|
|
if (pmadapter->tp_state_drop_point == 7 /*RX_DROP_P3*/)
|
|
goto done;
|
|
prx_pkt = (RxPacketHdr_t *)data;
|
|
/** check if packet need send to host only */
|
|
if (pmbuf->pdesc && !forward) {
|
|
if (pmadapter->callbacks.moal_recv_amsdu_packet) {
|
|
ret = pmadapter->callbacks.moal_recv_amsdu_packet(
|
|
pmadapter->pmoal_handle, pmbuf);
|
|
if (ret == MLAN_STATUS_PENDING) {
|
|
#ifdef USB
|
|
if (IS_USB(pmadapter->card_type))
|
|
pmadapter->callbacks.moal_recv_complete(
|
|
pmadapter->pmoal_handle, MNULL,
|
|
pmadapter->rx_data_ep, ret);
|
|
#endif
|
|
priv->msdu_in_rx_amsdu_cnt += num_subframes;
|
|
priv->amsdu_rx_cnt++;
|
|
return ret;
|
|
}
|
|
goto done;
|
|
}
|
|
}
|
|
while (total_pkt_len >= hdr_len) {
|
|
prx_pkt = (RxPacketHdr_t *)data;
|
|
/* Length will be in network format, change it to host */
|
|
pkt_len = mlan_ntohs(
|
|
(*(t_u16 *)(data + (2 * MLAN_MAC_ADDR_LENGTH))));
|
|
if (pkt_len > total_pkt_len) {
|
|
PRINTM(MERROR,
|
|
"Error in packet length: total_pkt_len = %d, pkt_len = %d\n",
|
|
total_pkt_len, pkt_len);
|
|
ret = MLAN_STATUS_FAILURE;
|
|
break;
|
|
}
|
|
|
|
pad = (((pkt_len + sizeof(Eth803Hdr_t)) & 3)) ?
|
|
(4 - ((pkt_len + sizeof(Eth803Hdr_t)) & 3)) :
|
|
0;
|
|
|
|
total_pkt_len -= pkt_len + pad + sizeof(Eth803Hdr_t);
|
|
|
|
if (memcmp(pmadapter, &prx_pkt->rfc1042_hdr, rfc1042_eth_hdr,
|
|
sizeof(rfc1042_eth_hdr)) == 0) {
|
|
memmove(pmadapter, data + LLC_SNAP_LEN, data,
|
|
(2 * MLAN_MAC_ADDR_LENGTH));
|
|
data += LLC_SNAP_LEN;
|
|
pkt_len += sizeof(Eth803Hdr_t) - LLC_SNAP_LEN;
|
|
} else {
|
|
*(t_u16 *)(data + (2 * MLAN_MAC_ADDR_LENGTH)) =
|
|
(t_u16)0;
|
|
pkt_len += sizeof(Eth803Hdr_t);
|
|
}
|
|
daggr_mbuf = wlan_alloc_mlan_buffer(pmadapter,
|
|
pkt_len + MLAN_NET_IP_ALIGN,
|
|
0, MOAL_ALLOC_MLAN_BUFFER);
|
|
if (daggr_mbuf == MNULL) {
|
|
PRINTM(MERROR, "Error allocating daggr mlan_buffer\n");
|
|
ret = MLAN_STATUS_FAILURE;
|
|
break;
|
|
}
|
|
daggr_mbuf->data_offset += MLAN_NET_IP_ALIGN;
|
|
daggr_mbuf->bss_index = pmbuf->bss_index;
|
|
daggr_mbuf->buf_type = pmbuf->buf_type;
|
|
daggr_mbuf->data_len = pkt_len;
|
|
daggr_mbuf->in_ts_sec = pmbuf->in_ts_sec;
|
|
daggr_mbuf->in_ts_usec = pmbuf->in_ts_usec;
|
|
daggr_mbuf->flags = pmbuf->flags;
|
|
daggr_mbuf->extra_ts_sec = pmbuf->extra_ts_sec;
|
|
daggr_mbuf->extra_ts_usec = pmbuf->extra_ts_usec;
|
|
daggr_mbuf->pparent = pmbuf;
|
|
daggr_mbuf->priority = pmbuf->priority;
|
|
if (pmadapter->tp_state_on)
|
|
pmadapter->callbacks.moal_get_system_time(
|
|
pmadapter->pmoal_handle, &in_copy_ts_sec,
|
|
&in_copy_ts_usec);
|
|
memcpy_ext(pmadapter,
|
|
daggr_mbuf->pbuf + daggr_mbuf->data_offset, data,
|
|
pkt_len, daggr_mbuf->data_len);
|
|
if (pmadapter->tp_state_on) {
|
|
pmadapter->callbacks.moal_get_system_time(
|
|
pmadapter->pmoal_handle, &out_copy_ts_sec,
|
|
&out_copy_ts_usec);
|
|
copy_delay +=
|
|
(t_s32)(out_copy_ts_sec - in_copy_ts_sec) *
|
|
1000000;
|
|
copy_delay +=
|
|
(t_s32)(out_copy_ts_usec - in_copy_ts_usec);
|
|
}
|
|
#ifdef UAP_SUPPORT
|
|
if (GET_BSS_ROLE(priv) == MLAN_BSS_ROLE_UAP) {
|
|
if (pmbuf->flags & MLAN_BUF_FLAG_EASYMESH)
|
|
memcpy_ext(priv->adapter, daggr_mbuf->mac,
|
|
pmbuf->mac, MLAN_MAC_ADDR_LENGTH,
|
|
MLAN_MAC_ADDR_LENGTH);
|
|
ret = wlan_uap_recv_packet(priv, daggr_mbuf);
|
|
} else {
|
|
#endif /* UAP_SUPPORT */
|
|
/** send EAPOL from AMSDU pkt to firmware */
|
|
if (priv->sec_info.ewpa_enabled &&
|
|
(!memcmp(pmadapter,
|
|
daggr_mbuf->pbuf +
|
|
daggr_mbuf->data_offset +
|
|
MLAN_ETHER_PKT_TYPE_OFFSET,
|
|
eapol_type, sizeof(eapol_type)))) {
|
|
ret = wlan_prepare_cmd(
|
|
priv, HostCmd_CMD_802_11_EAPOL_PKT, 0,
|
|
0, MNULL, daggr_mbuf);
|
|
if (ret == MLAN_STATUS_SUCCESS)
|
|
wlan_recv_event(
|
|
priv,
|
|
MLAN_EVENT_ID_DRV_DEFER_HANDLING,
|
|
MNULL);
|
|
wlan_free_mlan_buffer(pmadapter, daggr_mbuf);
|
|
data += pkt_len + pad;
|
|
continue;
|
|
}
|
|
/**process tdls packet*/
|
|
if (!memcmp(pmadapter,
|
|
daggr_mbuf->pbuf + daggr_mbuf->data_offset +
|
|
MLAN_ETHER_PKT_TYPE_OFFSET,
|
|
tdls_action_type,
|
|
sizeof(tdls_action_type))) {
|
|
PRINTM(MEVENT,
|
|
"Recevie AMSDU TDLS action frame\n");
|
|
wlan_process_tdls_action_frame(
|
|
priv,
|
|
daggr_mbuf->pbuf +
|
|
daggr_mbuf->data_offset,
|
|
daggr_mbuf->data_len);
|
|
}
|
|
|
|
ret = pmadapter->callbacks.moal_recv_packet(
|
|
pmadapter->pmoal_handle, daggr_mbuf);
|
|
#ifdef UAP_SUPPORT
|
|
}
|
|
#endif /* UAP_SUPPORT */
|
|
switch (ret) {
|
|
case MLAN_STATUS_PENDING:
|
|
break;
|
|
case MLAN_STATUS_FAILURE:
|
|
PRINTM(MERROR, "Deaggr, send to moal failed\n");
|
|
daggr_mbuf->status_code = MLAN_ERROR_PKT_INVALID;
|
|
/* fall through */
|
|
case MLAN_STATUS_SUCCESS:
|
|
wlan_recv_packet_complete(pmadapter, daggr_mbuf, ret);
|
|
break;
|
|
default:
|
|
break;
|
|
}
|
|
|
|
data += pkt_len + pad;
|
|
}
|
|
if (pmadapter->tp_state_on) {
|
|
pmadapter->callbacks.moal_get_system_time(
|
|
pmadapter->pmoal_handle, &out_ts_sec, &out_ts_usec);
|
|
delay += (t_s32)(out_ts_sec - in_ts_sec) * 1000000;
|
|
delay += (t_s32)(out_ts_usec - in_ts_usec);
|
|
pmadapter->callbacks.moal_amsdu_tp_accounting(
|
|
pmadapter->pmoal_handle, delay, copy_delay);
|
|
}
|
|
|
|
done:
|
|
priv->msdu_in_rx_amsdu_cnt += pmbuf->use_count;
|
|
priv->amsdu_rx_cnt++;
|
|
/** we should free the aggr buffer after deaggr */
|
|
pmadapter->ops.data_complete(pmadapter, pmbuf, ret);
|
|
LEAVE();
|
|
return ret;
|
|
}
|
|
|
|
#ifdef PCIEAW693
|
|
/**
|
|
* @brief Send amsdu subframe list to interface
|
|
*
|
|
* @param priv A pointer to mlan_private structure
|
|
* @param pra_list Pointer to the RA List table containing the pointers
|
|
* to packets.
|
|
* @param headroom Any interface specific headroom that may be need. TxPD
|
|
* will be formed leaving this headroom.
|
|
* @param ptrindex Pointer index
|
|
*
|
|
* @return Final packet size or MLAN_STATUS_FAILURE
|
|
*/
|
|
static int wlan_send_amsdu_subframe_list(mlan_private *priv,
|
|
raListTbl *pra_list, int headroom,
|
|
int ptrindex)
|
|
{
|
|
int pkt_size = 0;
|
|
pmlan_adapter pmadapter = priv->adapter;
|
|
mlan_buffer *pmbuf_src;
|
|
mlan_buffer *pmbuf_last = MNULL;
|
|
mlan_buffer *pmbuf_first = MNULL;
|
|
int pad = 0;
|
|
mlan_status ret = MLAN_STATUS_SUCCESS;
|
|
#ifdef DEBUG_LEVEL1
|
|
t_u32 sec = 0, usec = 0;
|
|
#endif
|
|
t_u32 max_amsdu_size = MIN(pra_list->max_amsdu, pmadapter->tx_buf_size);
|
|
t_u32 max_msdu_count = 0;
|
|
t_u32 msdu_in_tx_amsdu_cnt = 0;
|
|
ENTER();
|
|
|
|
max_msdu_count = pmadapter->ops.get_max_msdu_cnt(pmadapter);
|
|
pmbuf_src = (pmlan_buffer)util_peek_list(
|
|
pmadapter->pmoal_handle, &pra_list->buf_head, MNULL, MNULL);
|
|
if (pmbuf_src) {
|
|
pmbuf_first = pmbuf_src;
|
|
} else {
|
|
pmadapter->callbacks.moal_spin_unlock(
|
|
pmadapter->pmoal_handle, priv->wmm.ra_list_spinlock);
|
|
goto exit;
|
|
}
|
|
|
|
while (pmbuf_src &&
|
|
((pkt_size + (pmbuf_src->data_len + LLC_SNAP_LEN) + headroom) <=
|
|
max_amsdu_size) &&
|
|
(msdu_in_tx_amsdu_cnt < max_msdu_count)) {
|
|
pmbuf_src =
|
|
(pmlan_buffer)util_dequeue_list(pmadapter->pmoal_handle,
|
|
&pra_list->buf_head,
|
|
MNULL, MNULL);
|
|
/* Collects TP statistics */
|
|
if (pmadapter->tp_state_on &&
|
|
(pkt_size > Tx_PD_SIZEOF(pmadapter)))
|
|
pmadapter->callbacks.moal_tp_accounting(
|
|
pmadapter->pmoal_handle, pmbuf_src, 3);
|
|
pra_list->total_pkts--;
|
|
|
|
/* decrement for every PDU taken from the list */
|
|
priv->wmm.pkts_queued[ptrindex]--;
|
|
util_scalar_decrement(pmadapter->pmoal_handle,
|
|
&priv->wmm.tx_pkts_queued, MNULL, MNULL);
|
|
|
|
pmadapter->callbacks.moal_spin_unlock(
|
|
pmadapter->pmoal_handle, priv->wmm.ra_list_spinlock);
|
|
|
|
if (pmbuf_src) {
|
|
pkt_size += wlan_form_amsdu_subframe(pmadapter,
|
|
pmbuf_src, &pad);
|
|
util_enqueue_list_tail(pmadapter->pmoal_handle,
|
|
&pmadapter->amsdu_txq,
|
|
(pmlan_linked_list)pmbuf_src,
|
|
MNULL, MNULL);
|
|
pmbuf_last = pmbuf_src;
|
|
priv->msdu_in_tx_amsdu_cnt++;
|
|
msdu_in_tx_amsdu_cnt++;
|
|
}
|
|
pmadapter->callbacks.moal_spin_lock(pmadapter->pmoal_handle,
|
|
priv->wmm.ra_list_spinlock);
|
|
|
|
if (!wlan_is_ralist_valid(priv, pra_list, ptrindex)) {
|
|
pmadapter->callbacks.moal_spin_unlock(
|
|
pmadapter->pmoal_handle,
|
|
priv->wmm.ra_list_spinlock);
|
|
LEAVE();
|
|
return MLAN_STATUS_FAILURE;
|
|
}
|
|
|
|
pmbuf_src =
|
|
(pmlan_buffer)util_peek_list(pmadapter->pmoal_handle,
|
|
&pra_list->buf_head, MNULL,
|
|
MNULL);
|
|
}
|
|
|
|
pmadapter->callbacks.moal_spin_unlock(pmadapter->pmoal_handle,
|
|
priv->wmm.ra_list_spinlock);
|
|
|
|
/* Last AMSDU packet does not need padding */
|
|
pkt_size -= pad;
|
|
pmbuf_last->data_len -= pad;
|
|
|
|
pkt_size += wlan_form_amsdu_txpd(priv, pmbuf_first, pkt_size);
|
|
/* Collects TP statistics */
|
|
if (pmadapter->tp_state_on) {
|
|
mlan_buffer mbuf;
|
|
mbuf.data_len = pkt_size;
|
|
pmadapter->callbacks.moal_tp_accounting(pmadapter->pmoal_handle,
|
|
&mbuf, 4);
|
|
pmadapter->callbacks.moal_tp_accounting_rx_param(
|
|
pmadapter->pmoal_handle, 5, msdu_in_tx_amsdu_cnt);
|
|
}
|
|
/* Drop Tx packets at drop point 4 */
|
|
if (pmadapter->tp_state_drop_point == 4) {
|
|
wlan_free_amsdu_txq(pmadapter);
|
|
goto exit;
|
|
}
|
|
ret = pmadapter->ops.send_data_list(pmadapter, MLAN_TYPE_DATA,
|
|
msdu_in_tx_amsdu_cnt, pkt_size);
|
|
|
|
if (ret != MLAN_STATUS_RESOURCE) {
|
|
pmadapter->callbacks.moal_spin_lock(pmadapter->pmoal_handle,
|
|
priv->wmm.ra_list_spinlock);
|
|
if (wlan_is_ralist_valid(priv, pra_list, ptrindex)) {
|
|
priv->wmm.packets_out[ptrindex]++;
|
|
priv->wmm.tid_tbl_ptr[ptrindex].ra_list_curr = pra_list;
|
|
}
|
|
pmadapter->bssprio_tbl[priv->bss_priority].bssprio_cur =
|
|
pmadapter->bssprio_tbl[priv->bss_priority]
|
|
.bssprio_cur->pnext;
|
|
pmadapter->callbacks.moal_spin_unlock(
|
|
pmadapter->pmoal_handle, priv->wmm.ra_list_spinlock);
|
|
}
|
|
|
|
PRINTM_GET_SYS_TIME(MDATA, &sec, &usec);
|
|
PRINTM_NETINTF(MDATA, priv);
|
|
PRINTM(MDATA, "%lu.%06lu : Data => FW\n", sec, usec);
|
|
priv->amsdu_tx_cnt++;
|
|
exit:
|
|
LEAVE();
|
|
return pkt_size;
|
|
}
|
|
#endif
|
|
/**
|
|
* @brief Aggregate multiple packets into one single AMSDU packet
|
|
*
|
|
* @param priv A pointer to mlan_private structure
|
|
* @param pra_list Pointer to the RA List table containing the pointers
|
|
* to packets.
|
|
* @param headroom Any interface specific headroom that may be need. TxPD
|
|
* will be formed leaving this headroom.
|
|
* @param ptrindex Pointer index
|
|
*
|
|
* @return Final packet size or MLAN_STATUS_FAILURE
|
|
*/
|
|
int wlan_11n_aggregate_pkt(mlan_private *priv, raListTbl *pra_list,
|
|
int headroom, int ptrindex)
|
|
{
|
|
int pkt_size = 0;
|
|
pmlan_adapter pmadapter = priv->adapter;
|
|
mlan_buffer *pmbuf_aggr, *pmbuf_src;
|
|
t_u8 *data;
|
|
int pad = 0;
|
|
mlan_status ret = MLAN_STATUS_SUCCESS;
|
|
#ifdef DEBUG_LEVEL1
|
|
t_u32 sec = 0, usec = 0;
|
|
#endif
|
|
mlan_tx_param tx_param;
|
|
#ifdef STA_SUPPORT
|
|
TxPD *ptx_pd = MNULL;
|
|
#endif
|
|
t_u32 max_amsdu_size = MIN(pra_list->max_amsdu, pmadapter->tx_buf_size);
|
|
t_u32 msdu_in_tx_amsdu_cnt = 0;
|
|
ENTER();
|
|
|
|
PRINTM(MDAT_D, "Handling Aggr packet\n");
|
|
|
|
#ifdef PCIEAW693
|
|
if (IS_PCIEAW693(pmadapter->card_type)) {
|
|
return wlan_send_amsdu_subframe_list(priv, pra_list, headroom,
|
|
ptrindex);
|
|
}
|
|
#endif
|
|
pmbuf_src = (pmlan_buffer)util_peek_list(
|
|
pmadapter->pmoal_handle, &pra_list->buf_head, MNULL, MNULL);
|
|
if (pmbuf_src) {
|
|
pmbuf_aggr = wlan_alloc_mlan_buffer(
|
|
pmadapter, pmadapter->tx_buf_size, headroom,
|
|
MOAL_MEM_FLAG_DIRTY | MOAL_MALLOC_BUFFER |
|
|
MOAL_MEM_FLAG_ATOMIC);
|
|
if (!pmbuf_aggr) {
|
|
PRINTM(MERROR, "Error allocating mlan_buffer\n");
|
|
pmadapter->callbacks.moal_spin_unlock(
|
|
pmadapter->pmoal_handle,
|
|
priv->wmm.ra_list_spinlock);
|
|
LEAVE();
|
|
return MLAN_STATUS_FAILURE;
|
|
}
|
|
|
|
data = pmbuf_aggr->pbuf + headroom;
|
|
pmbuf_aggr->bss_index = pmbuf_src->bss_index;
|
|
pmbuf_aggr->buf_type = pmbuf_src->buf_type;
|
|
pmbuf_aggr->priority = pmbuf_src->priority;
|
|
pmbuf_aggr->pbuf = data;
|
|
pmbuf_aggr->data_offset = 0;
|
|
pmbuf_aggr->in_ts_sec = pmbuf_src->in_ts_sec;
|
|
pmbuf_aggr->in_ts_usec = pmbuf_src->in_ts_usec;
|
|
pmbuf_aggr->extra_ts_sec = pmbuf_src->extra_ts_sec;
|
|
pmbuf_aggr->extra_ts_usec = pmbuf_src->extra_ts_usec;
|
|
if (pmbuf_src->flags & MLAN_BUF_FLAG_TDLS)
|
|
pmbuf_aggr->flags |= MLAN_BUF_FLAG_TDLS;
|
|
if (pmbuf_src->flags & MLAN_BUF_FLAG_TCP_ACK)
|
|
pmbuf_aggr->flags |= MLAN_BUF_FLAG_TCP_ACK;
|
|
|
|
if (pmbuf_src->flags & MLAN_BUF_FLAG_EASYMESH) {
|
|
pmbuf_aggr->flags |= MLAN_BUF_FLAG_EASYMESH;
|
|
memcpy_ext(priv->adapter, pmbuf_aggr->mac,
|
|
pmbuf_src->mac, MLAN_MAC_ADDR_LENGTH,
|
|
MLAN_MAC_ADDR_LENGTH);
|
|
}
|
|
/* Form AMSDU */
|
|
wlan_11n_form_amsdu_txpd(priv, pmbuf_aggr);
|
|
pkt_size = Tx_PD_SIZEOF(pmadapter);
|
|
#ifdef STA_SUPPORT
|
|
if (GET_BSS_ROLE(priv) == MLAN_BSS_ROLE_STA)
|
|
ptx_pd = (TxPD *)pmbuf_aggr->pbuf;
|
|
#endif
|
|
priv->msdu_in_tx_amsdu_cnt++;
|
|
} else {
|
|
pmadapter->callbacks.moal_spin_unlock(
|
|
pmadapter->pmoal_handle, priv->wmm.ra_list_spinlock);
|
|
goto exit;
|
|
}
|
|
|
|
while (pmbuf_src && ((pkt_size + (pmbuf_src->data_len + LLC_SNAP_LEN) +
|
|
headroom) <= max_amsdu_size)) {
|
|
pmbuf_src =
|
|
(pmlan_buffer)util_dequeue_list(pmadapter->pmoal_handle,
|
|
&pra_list->buf_head,
|
|
MNULL, MNULL);
|
|
/* Collects TP statistics */
|
|
if (pmadapter->tp_state_on &&
|
|
(pkt_size > Tx_PD_SIZEOF(pmadapter)))
|
|
pmadapter->callbacks.moal_tp_accounting(
|
|
pmadapter->pmoal_handle, pmbuf_src, 3);
|
|
pra_list->total_pkts--;
|
|
|
|
/* decrement for every PDU taken from the list */
|
|
priv->wmm.pkts_queued[ptrindex]--;
|
|
util_scalar_decrement(pmadapter->pmoal_handle,
|
|
&priv->wmm.tx_pkts_queued, MNULL, MNULL);
|
|
|
|
pmadapter->callbacks.moal_spin_unlock(
|
|
pmadapter->pmoal_handle, priv->wmm.ra_list_spinlock);
|
|
|
|
if (pmbuf_src) {
|
|
pkt_size += wlan_11n_form_amsdu_pkt(
|
|
pmadapter, (data + pkt_size),
|
|
pmbuf_src->pbuf + pmbuf_src->data_offset,
|
|
pmbuf_src->data_len, &pad);
|
|
|
|
DBG_HEXDUMP(MDAT_D, "pmbuf_src", pmbuf_src,
|
|
sizeof(mlan_buffer));
|
|
wlan_write_data_complete(pmadapter, pmbuf_src,
|
|
MLAN_STATUS_SUCCESS);
|
|
}
|
|
|
|
pmadapter->callbacks.moal_spin_lock(pmadapter->pmoal_handle,
|
|
priv->wmm.ra_list_spinlock);
|
|
|
|
if (!wlan_is_ralist_valid(priv, pra_list, ptrindex)) {
|
|
pmadapter->callbacks.moal_spin_unlock(
|
|
pmadapter->pmoal_handle,
|
|
priv->wmm.ra_list_spinlock);
|
|
LEAVE();
|
|
return MLAN_STATUS_FAILURE;
|
|
}
|
|
|
|
pmbuf_src =
|
|
(pmlan_buffer)util_peek_list(pmadapter->pmoal_handle,
|
|
&pra_list->buf_head, MNULL,
|
|
MNULL);
|
|
priv->msdu_in_tx_amsdu_cnt++;
|
|
msdu_in_tx_amsdu_cnt++;
|
|
}
|
|
|
|
pmadapter->callbacks.moal_spin_unlock(pmadapter->pmoal_handle,
|
|
priv->wmm.ra_list_spinlock);
|
|
|
|
/* Last AMSDU packet does not need padding */
|
|
pkt_size -= pad;
|
|
pmbuf_aggr->data_len = pkt_size;
|
|
wlan_11n_update_pktlen_amsdu_txpd(priv, pmbuf_aggr);
|
|
pmbuf_aggr->data_len += headroom;
|
|
pmbuf_aggr->pbuf = data - headroom;
|
|
tx_param.next_pkt_len =
|
|
((pmbuf_src) ? pmbuf_src->data_len + Tx_PD_SIZEOF(pmadapter) :
|
|
0);
|
|
/* Collects TP statistics */
|
|
if (pmadapter->tp_state_on) {
|
|
pmadapter->callbacks.moal_tp_accounting(pmadapter->pmoal_handle,
|
|
pmbuf_aggr, 4);
|
|
pmadapter->callbacks.moal_tp_accounting_rx_param(
|
|
pmadapter->pmoal_handle, 5, msdu_in_tx_amsdu_cnt);
|
|
}
|
|
|
|
/* Drop Tx packets at drop point 4 */
|
|
if (pmadapter->tp_state_drop_point == 4) {
|
|
wlan_write_data_complete(pmadapter, pmbuf_aggr, ret);
|
|
goto exit;
|
|
} else
|
|
ret = pmadapter->ops.host_to_card(priv, MLAN_TYPE_DATA,
|
|
pmbuf_aggr, &tx_param);
|
|
switch (ret) {
|
|
#ifdef USB
|
|
case MLAN_STATUS_PRESOURCE:
|
|
PRINTM(MINFO, "MLAN_STATUS_PRESOURCE is returned\n");
|
|
break;
|
|
#endif
|
|
case MLAN_STATUS_RESOURCE:
|
|
pmadapter->callbacks.moal_spin_lock(pmadapter->pmoal_handle,
|
|
priv->wmm.ra_list_spinlock);
|
|
|
|
if (!wlan_is_ralist_valid(priv, pra_list, ptrindex)) {
|
|
pmadapter->callbacks.moal_spin_unlock(
|
|
pmadapter->pmoal_handle,
|
|
priv->wmm.ra_list_spinlock);
|
|
pmbuf_aggr->status_code = MLAN_ERROR_PKT_INVALID;
|
|
wlan_write_data_complete(pmadapter, pmbuf_aggr,
|
|
MLAN_STATUS_FAILURE);
|
|
LEAVE();
|
|
return MLAN_STATUS_FAILURE;
|
|
}
|
|
#ifdef STA_SUPPORT
|
|
/* reset tx_lock_flag */
|
|
if ((GET_BSS_ROLE(priv) == MLAN_BSS_ROLE_STA) &&
|
|
pmadapter->pps_uapsd_mode &&
|
|
(pmadapter->tx_lock_flag == MTRUE)) {
|
|
pmadapter->tx_lock_flag = MFALSE;
|
|
if (ptx_pd != MNULL)
|
|
ptx_pd->flags = 0;
|
|
}
|
|
#endif
|
|
util_enqueue_list_head(pmadapter->pmoal_handle,
|
|
&pra_list->buf_head,
|
|
(pmlan_linked_list)pmbuf_aggr, MNULL,
|
|
MNULL);
|
|
|
|
pra_list->total_pkts++;
|
|
|
|
/* add back only one: aggregated packet is requeued as one */
|
|
priv->wmm.pkts_queued[ptrindex]++;
|
|
util_scalar_increment(pmadapter->pmoal_handle,
|
|
&priv->wmm.tx_pkts_queued, MNULL, MNULL);
|
|
pmbuf_aggr->flags |= MLAN_BUF_FLAG_REQUEUED_PKT;
|
|
pmadapter->callbacks.moal_spin_unlock(
|
|
pmadapter->pmoal_handle, priv->wmm.ra_list_spinlock);
|
|
PRINTM(MINFO, "MLAN_STATUS_RESOURCE is returned\n");
|
|
pmbuf_aggr->status_code = MLAN_ERROR_PKT_INVALID;
|
|
break;
|
|
case MLAN_STATUS_FAILURE:
|
|
pmbuf_aggr->status_code = MLAN_ERROR_DATA_TX_FAIL;
|
|
pmadapter->dbg.num_tx_host_to_card_failure++;
|
|
wlan_write_data_complete(pmadapter, pmbuf_aggr, ret);
|
|
goto exit;
|
|
case MLAN_STATUS_PENDING:
|
|
break;
|
|
case MLAN_STATUS_SUCCESS:
|
|
wlan_write_data_complete(pmadapter, pmbuf_aggr, ret);
|
|
break;
|
|
default:
|
|
break;
|
|
}
|
|
if (ret != MLAN_STATUS_RESOURCE) {
|
|
pmadapter->callbacks.moal_spin_lock(pmadapter->pmoal_handle,
|
|
priv->wmm.ra_list_spinlock);
|
|
if (wlan_is_ralist_valid(priv, pra_list, ptrindex)) {
|
|
priv->wmm.packets_out[ptrindex]++;
|
|
priv->wmm.tid_tbl_ptr[ptrindex].ra_list_curr = pra_list;
|
|
}
|
|
pmadapter->bssprio_tbl[priv->bss_priority].bssprio_cur =
|
|
pmadapter->bssprio_tbl[priv->bss_priority]
|
|
.bssprio_cur->pnext;
|
|
pmadapter->callbacks.moal_spin_unlock(
|
|
pmadapter->pmoal_handle, priv->wmm.ra_list_spinlock);
|
|
}
|
|
PRINTM_GET_SYS_TIME(MDATA, &sec, &usec);
|
|
PRINTM_NETINTF(MDATA, priv);
|
|
PRINTM(MDATA, "%lu.%06lu : Data => FW\n", sec, usec);
|
|
priv->amsdu_tx_cnt++;
|
|
|
|
exit:
|
|
LEAVE();
|
|
return MIN((pkt_size + headroom), INT_MAX);
|
|
}
|