Path: blob/main/sys/contrib/dev/athk/ath11k/hal_rx.c
48378 views
// SPDX-License-Identifier: BSD-3-Clause-Clear1/*2* Copyright (c) 2018-2019 The Linux Foundation. All rights reserved.3*/45#include "debug.h"6#include "hal.h"7#include "hal_tx.h"8#include "hal_rx.h"9#include "hal_desc.h"10#include "hif.h"1112static void ath11k_hal_reo_set_desc_hdr(struct hal_desc_header *hdr,13u8 owner, u8 buffer_type, u32 magic)14{15hdr->info0 = FIELD_PREP(HAL_DESC_HDR_INFO0_OWNER, owner) |16FIELD_PREP(HAL_DESC_HDR_INFO0_BUF_TYPE, buffer_type);1718/* Magic pattern in reserved bits for debugging */19hdr->info0 |= FIELD_PREP(HAL_DESC_HDR_INFO0_DBG_RESERVED, magic);20}2122static int ath11k_hal_reo_cmd_queue_stats(struct hal_tlv_hdr *tlv,23struct ath11k_hal_reo_cmd *cmd)24{25struct hal_reo_get_queue_stats *desc;2627tlv->tl = FIELD_PREP(HAL_TLV_HDR_TAG, HAL_REO_GET_QUEUE_STATS) |28FIELD_PREP(HAL_TLV_HDR_LEN, sizeof(*desc));2930desc = (struct hal_reo_get_queue_stats *)tlv->value;31memset_startat(desc, 0, queue_addr_lo);3233desc->cmd.info0 &= ~HAL_REO_CMD_HDR_INFO0_STATUS_REQUIRED;34if (cmd->flag & HAL_REO_CMD_FLG_NEED_STATUS)35desc->cmd.info0 |= HAL_REO_CMD_HDR_INFO0_STATUS_REQUIRED;3637desc->queue_addr_lo = cmd->addr_lo;38desc->info0 = FIELD_PREP(HAL_REO_GET_QUEUE_STATS_INFO0_QUEUE_ADDR_HI,39cmd->addr_hi);40if (cmd->flag & HAL_REO_CMD_FLG_STATS_CLEAR)41desc->info0 |= HAL_REO_GET_QUEUE_STATS_INFO0_CLEAR_STATS;4243return FIELD_GET(HAL_REO_CMD_HDR_INFO0_CMD_NUMBER, desc->cmd.info0);44}4546static int ath11k_hal_reo_cmd_flush_cache(struct ath11k_hal *hal, struct hal_tlv_hdr *tlv,47struct ath11k_hal_reo_cmd *cmd)48{49struct hal_reo_flush_cache *desc;50u8 avail_slot = ffz(hal->avail_blk_resource);5152if (cmd->flag & HAL_REO_CMD_FLG_FLUSH_BLOCK_LATER) {53if (avail_slot >= HAL_MAX_AVAIL_BLK_RES)54return -ENOSPC;5556hal->current_blk_index = avail_slot;57}5859tlv->tl = FIELD_PREP(HAL_TLV_HDR_TAG, HAL_REO_FLUSH_CACHE) |60FIELD_PREP(HAL_TLV_HDR_LEN, sizeof(*desc));6162desc = (struct hal_reo_flush_cache *)tlv->value;63memset_startat(desc, 0, cache_addr_lo);6465desc->cmd.info0 &= ~HAL_REO_CMD_HDR_INFO0_STATUS_REQUIRED;66if (cmd->flag & HAL_REO_CMD_FLG_NEED_STATUS)67desc->cmd.info0 |= HAL_REO_CMD_HDR_INFO0_STATUS_REQUIRED;6869desc->cache_addr_lo = cmd->addr_lo;70desc->info0 = FIELD_PREP(HAL_REO_FLUSH_CACHE_INFO0_CACHE_ADDR_HI,71cmd->addr_hi);7273if (cmd->flag & HAL_REO_CMD_FLG_FLUSH_FWD_ALL_MPDUS)74desc->info0 |= HAL_REO_FLUSH_CACHE_INFO0_FWD_ALL_MPDUS;7576if (cmd->flag & HAL_REO_CMD_FLG_FLUSH_BLOCK_LATER) {77desc->info0 |= HAL_REO_FLUSH_CACHE_INFO0_BLOCK_CACHE_USAGE;78desc->info0 |=79FIELD_PREP(HAL_REO_FLUSH_CACHE_INFO0_BLOCK_RESRC_IDX,80avail_slot);81}8283if (cmd->flag & HAL_REO_CMD_FLG_FLUSH_NO_INVAL)84desc->info0 |= HAL_REO_FLUSH_CACHE_INFO0_FLUSH_WO_INVALIDATE;8586if (cmd->flag & HAL_REO_CMD_FLG_FLUSH_ALL)87desc->info0 |= HAL_REO_FLUSH_CACHE_INFO0_FLUSH_ALL;8889return FIELD_GET(HAL_REO_CMD_HDR_INFO0_CMD_NUMBER, desc->cmd.info0);90}9192static int ath11k_hal_reo_cmd_update_rx_queue(struct hal_tlv_hdr *tlv,93struct ath11k_hal_reo_cmd *cmd)94{95struct hal_reo_update_rx_queue *desc;9697tlv->tl = FIELD_PREP(HAL_TLV_HDR_TAG, HAL_REO_UPDATE_RX_REO_QUEUE) |98FIELD_PREP(HAL_TLV_HDR_LEN, sizeof(*desc));99100desc = (struct hal_reo_update_rx_queue *)tlv->value;101memset_startat(desc, 0, queue_addr_lo);102103desc->cmd.info0 &= ~HAL_REO_CMD_HDR_INFO0_STATUS_REQUIRED;104if (cmd->flag & HAL_REO_CMD_FLG_NEED_STATUS)105desc->cmd.info0 |= HAL_REO_CMD_HDR_INFO0_STATUS_REQUIRED;106107desc->queue_addr_lo = cmd->addr_lo;108desc->info0 =109FIELD_PREP(HAL_REO_UPD_RX_QUEUE_INFO0_QUEUE_ADDR_HI,110cmd->addr_hi) |111FIELD_PREP(HAL_REO_UPD_RX_QUEUE_INFO0_UPD_RX_QUEUE_NUM,112!!(cmd->upd0 & HAL_REO_CMD_UPD0_RX_QUEUE_NUM)) |113FIELD_PREP(HAL_REO_UPD_RX_QUEUE_INFO0_UPD_VLD,114!!(cmd->upd0 & HAL_REO_CMD_UPD0_VLD)) |115FIELD_PREP(HAL_REO_UPD_RX_QUEUE_INFO0_UPD_ASSOC_LNK_DESC_CNT,116!!(cmd->upd0 & HAL_REO_CMD_UPD0_ALDC)) |117FIELD_PREP(HAL_REO_UPD_RX_QUEUE_INFO0_UPD_DIS_DUP_DETECTION,118!!(cmd->upd0 & HAL_REO_CMD_UPD0_DIS_DUP_DETECTION)) |119FIELD_PREP(HAL_REO_UPD_RX_QUEUE_INFO0_UPD_SOFT_REORDER_EN,120!!(cmd->upd0 & HAL_REO_CMD_UPD0_SOFT_REORDER_EN)) |121FIELD_PREP(HAL_REO_UPD_RX_QUEUE_INFO0_UPD_AC,122!!(cmd->upd0 & HAL_REO_CMD_UPD0_AC)) |123FIELD_PREP(HAL_REO_UPD_RX_QUEUE_INFO0_UPD_BAR,124!!(cmd->upd0 & HAL_REO_CMD_UPD0_BAR)) |125FIELD_PREP(HAL_REO_UPD_RX_QUEUE_INFO0_UPD_RETRY,126!!(cmd->upd0 & HAL_REO_CMD_UPD0_RETRY)) |127FIELD_PREP(HAL_REO_UPD_RX_QUEUE_INFO0_UPD_CHECK_2K_MODE,128!!(cmd->upd0 & HAL_REO_CMD_UPD0_CHECK_2K_MODE)) |129FIELD_PREP(HAL_REO_UPD_RX_QUEUE_INFO0_UPD_OOR_MODE,130!!(cmd->upd0 & HAL_REO_CMD_UPD0_OOR_MODE)) |131FIELD_PREP(HAL_REO_UPD_RX_QUEUE_INFO0_UPD_BA_WINDOW_SIZE,132!!(cmd->upd0 & HAL_REO_CMD_UPD0_BA_WINDOW_SIZE)) |133FIELD_PREP(HAL_REO_UPD_RX_QUEUE_INFO0_UPD_PN_CHECK,134!!(cmd->upd0 & HAL_REO_CMD_UPD0_PN_CHECK)) |135FIELD_PREP(HAL_REO_UPD_RX_QUEUE_INFO0_UPD_EVEN_PN,136!!(cmd->upd0 & HAL_REO_CMD_UPD0_EVEN_PN)) |137FIELD_PREP(HAL_REO_UPD_RX_QUEUE_INFO0_UPD_UNEVEN_PN,138!!(cmd->upd0 & HAL_REO_CMD_UPD0_UNEVEN_PN)) |139FIELD_PREP(HAL_REO_UPD_RX_QUEUE_INFO0_UPD_PN_HANDLE_ENABLE,140!!(cmd->upd0 & HAL_REO_CMD_UPD0_PN_HANDLE_ENABLE)) |141FIELD_PREP(HAL_REO_UPD_RX_QUEUE_INFO0_UPD_PN_SIZE,142!!(cmd->upd0 & HAL_REO_CMD_UPD0_PN_SIZE)) |143FIELD_PREP(HAL_REO_UPD_RX_QUEUE_INFO0_UPD_IGNORE_AMPDU_FLG,144!!(cmd->upd0 & HAL_REO_CMD_UPD0_IGNORE_AMPDU_FLG)) |145FIELD_PREP(HAL_REO_UPD_RX_QUEUE_INFO0_UPD_SVLD,146!!(cmd->upd0 & HAL_REO_CMD_UPD0_SVLD)) |147FIELD_PREP(HAL_REO_UPD_RX_QUEUE_INFO0_UPD_SSN,148!!(cmd->upd0 & HAL_REO_CMD_UPD0_SSN)) |149FIELD_PREP(HAL_REO_UPD_RX_QUEUE_INFO0_UPD_SEQ_2K_ERR,150!!(cmd->upd0 & HAL_REO_CMD_UPD0_SEQ_2K_ERR)) |151FIELD_PREP(HAL_REO_UPD_RX_QUEUE_INFO0_UPD_PN_VALID,152!!(cmd->upd0 & HAL_REO_CMD_UPD0_PN_VALID)) |153FIELD_PREP(HAL_REO_UPD_RX_QUEUE_INFO0_UPD_PN,154!!(cmd->upd0 & HAL_REO_CMD_UPD0_PN));155156desc->info1 =157FIELD_PREP(HAL_REO_UPD_RX_QUEUE_INFO1_RX_QUEUE_NUMBER,158cmd->rx_queue_num) |159FIELD_PREP(HAL_REO_UPD_RX_QUEUE_INFO1_VLD,160!!(cmd->upd1 & HAL_REO_CMD_UPD1_VLD)) |161FIELD_PREP(HAL_REO_UPD_RX_QUEUE_INFO1_ASSOC_LNK_DESC_COUNTER,162FIELD_GET(HAL_REO_CMD_UPD1_ALDC, cmd->upd1)) |163FIELD_PREP(HAL_REO_UPD_RX_QUEUE_INFO1_DIS_DUP_DETECTION,164!!(cmd->upd1 & HAL_REO_CMD_UPD1_DIS_DUP_DETECTION)) |165FIELD_PREP(HAL_REO_UPD_RX_QUEUE_INFO1_SOFT_REORDER_EN,166!!(cmd->upd1 & HAL_REO_CMD_UPD1_SOFT_REORDER_EN)) |167FIELD_PREP(HAL_REO_UPD_RX_QUEUE_INFO1_AC,168FIELD_GET(HAL_REO_CMD_UPD1_AC, cmd->upd1)) |169FIELD_PREP(HAL_REO_UPD_RX_QUEUE_INFO1_BAR,170!!(cmd->upd1 & HAL_REO_CMD_UPD1_BAR)) |171FIELD_PREP(HAL_REO_UPD_RX_QUEUE_INFO1_CHECK_2K_MODE,172!!(cmd->upd1 & HAL_REO_CMD_UPD1_CHECK_2K_MODE)) |173FIELD_PREP(HAL_REO_UPD_RX_QUEUE_INFO1_RETRY,174!!(cmd->upd1 & HAL_REO_CMD_UPD1_RETRY)) |175FIELD_PREP(HAL_REO_UPD_RX_QUEUE_INFO1_OOR_MODE,176!!(cmd->upd1 & HAL_REO_CMD_UPD1_OOR_MODE)) |177FIELD_PREP(HAL_REO_UPD_RX_QUEUE_INFO1_PN_CHECK,178!!(cmd->upd1 & HAL_REO_CMD_UPD1_PN_CHECK)) |179FIELD_PREP(HAL_REO_UPD_RX_QUEUE_INFO1_EVEN_PN,180!!(cmd->upd1 & HAL_REO_CMD_UPD1_EVEN_PN)) |181FIELD_PREP(HAL_REO_UPD_RX_QUEUE_INFO1_UNEVEN_PN,182!!(cmd->upd1 & HAL_REO_CMD_UPD1_UNEVEN_PN)) |183FIELD_PREP(HAL_REO_UPD_RX_QUEUE_INFO1_PN_HANDLE_ENABLE,184!!(cmd->upd1 & HAL_REO_CMD_UPD1_PN_HANDLE_ENABLE)) |185FIELD_PREP(HAL_REO_UPD_RX_QUEUE_INFO1_IGNORE_AMPDU_FLG,186!!(cmd->upd1 & HAL_REO_CMD_UPD1_IGNORE_AMPDU_FLG));187188if (cmd->pn_size == 24)189cmd->pn_size = HAL_RX_REO_QUEUE_PN_SIZE_24;190else if (cmd->pn_size == 48)191cmd->pn_size = HAL_RX_REO_QUEUE_PN_SIZE_48;192else if (cmd->pn_size == 128)193cmd->pn_size = HAL_RX_REO_QUEUE_PN_SIZE_128;194195if (cmd->ba_window_size < 1)196cmd->ba_window_size = 1;197198if (cmd->ba_window_size == 1)199cmd->ba_window_size++;200201desc->info2 =202FIELD_PREP(HAL_REO_UPD_RX_QUEUE_INFO2_BA_WINDOW_SIZE,203cmd->ba_window_size - 1) |204FIELD_PREP(HAL_REO_UPD_RX_QUEUE_INFO2_PN_SIZE, cmd->pn_size) |205FIELD_PREP(HAL_REO_UPD_RX_QUEUE_INFO2_SVLD,206!!(cmd->upd2 & HAL_REO_CMD_UPD2_SVLD)) |207FIELD_PREP(HAL_REO_UPD_RX_QUEUE_INFO2_SSN,208FIELD_GET(HAL_REO_CMD_UPD2_SSN, cmd->upd2)) |209FIELD_PREP(HAL_REO_UPD_RX_QUEUE_INFO2_SEQ_2K_ERR,210!!(cmd->upd2 & HAL_REO_CMD_UPD2_SEQ_2K_ERR)) |211FIELD_PREP(HAL_REO_UPD_RX_QUEUE_INFO2_PN_ERR,212!!(cmd->upd2 & HAL_REO_CMD_UPD2_PN_ERR));213214return FIELD_GET(HAL_REO_CMD_HDR_INFO0_CMD_NUMBER, desc->cmd.info0);215}216217int ath11k_hal_reo_cmd_send(struct ath11k_base *ab, struct hal_srng *srng,218enum hal_reo_cmd_type type,219struct ath11k_hal_reo_cmd *cmd)220{221struct hal_tlv_hdr *reo_desc;222int ret;223224spin_lock_bh(&srng->lock);225226ath11k_hal_srng_access_begin(ab, srng);227reo_desc = (struct hal_tlv_hdr *)ath11k_hal_srng_src_get_next_entry(ab, srng);228if (!reo_desc) {229ret = -ENOBUFS;230goto out;231}232233switch (type) {234case HAL_REO_CMD_GET_QUEUE_STATS:235ret = ath11k_hal_reo_cmd_queue_stats(reo_desc, cmd);236break;237case HAL_REO_CMD_FLUSH_CACHE:238ret = ath11k_hal_reo_cmd_flush_cache(&ab->hal, reo_desc, cmd);239break;240case HAL_REO_CMD_UPDATE_RX_QUEUE:241ret = ath11k_hal_reo_cmd_update_rx_queue(reo_desc, cmd);242break;243case HAL_REO_CMD_FLUSH_QUEUE:244case HAL_REO_CMD_UNBLOCK_CACHE:245case HAL_REO_CMD_FLUSH_TIMEOUT_LIST:246ath11k_warn(ab, "Unsupported reo command %d\n", type);247ret = -ENOTSUPP;248break;249default:250ath11k_warn(ab, "Unknown reo command %d\n", type);251ret = -EINVAL;252break;253}254255ath11k_dp_shadow_start_timer(ab, srng, &ab->dp.reo_cmd_timer);256257out:258ath11k_hal_srng_access_end(ab, srng);259spin_unlock_bh(&srng->lock);260261return ret;262}263264void ath11k_hal_rx_buf_addr_info_set(void *desc, dma_addr_t paddr,265u32 cookie, u8 manager)266{267struct ath11k_buffer_addr *binfo = (struct ath11k_buffer_addr *)desc;268u32 paddr_lo, paddr_hi;269270paddr_lo = lower_32_bits(paddr);271paddr_hi = upper_32_bits(paddr);272binfo->info0 = FIELD_PREP(BUFFER_ADDR_INFO0_ADDR, paddr_lo);273binfo->info1 = FIELD_PREP(BUFFER_ADDR_INFO1_ADDR, paddr_hi) |274FIELD_PREP(BUFFER_ADDR_INFO1_SW_COOKIE, cookie) |275FIELD_PREP(BUFFER_ADDR_INFO1_RET_BUF_MGR, manager);276}277278void ath11k_hal_rx_buf_addr_info_get(void *desc, dma_addr_t *paddr,279u32 *cookie, u8 *rbm)280{281struct ath11k_buffer_addr *binfo = (struct ath11k_buffer_addr *)desc;282283*paddr =284(((u64)FIELD_GET(BUFFER_ADDR_INFO1_ADDR, binfo->info1)) << 32) |285FIELD_GET(BUFFER_ADDR_INFO0_ADDR, binfo->info0);286*cookie = FIELD_GET(BUFFER_ADDR_INFO1_SW_COOKIE, binfo->info1);287*rbm = FIELD_GET(BUFFER_ADDR_INFO1_RET_BUF_MGR, binfo->info1);288}289290void ath11k_hal_rx_msdu_link_info_get(void *link_desc, u32 *num_msdus,291u32 *msdu_cookies,292enum hal_rx_buf_return_buf_manager *rbm)293{294struct hal_rx_msdu_link *link = (struct hal_rx_msdu_link *)link_desc;295struct hal_rx_msdu_details *msdu;296int i;297298*num_msdus = HAL_NUM_RX_MSDUS_PER_LINK_DESC;299300msdu = &link->msdu_link[0];301*rbm = FIELD_GET(BUFFER_ADDR_INFO1_RET_BUF_MGR,302msdu->buf_addr_info.info1);303304for (i = 0; i < *num_msdus; i++) {305msdu = &link->msdu_link[i];306307if (!FIELD_GET(BUFFER_ADDR_INFO0_ADDR,308msdu->buf_addr_info.info0)) {309*num_msdus = i;310break;311}312*msdu_cookies = FIELD_GET(BUFFER_ADDR_INFO1_SW_COOKIE,313msdu->buf_addr_info.info1);314msdu_cookies++;315}316}317318int ath11k_hal_desc_reo_parse_err(struct ath11k_base *ab, u32 *rx_desc,319dma_addr_t *paddr, u32 *desc_bank)320{321struct hal_reo_dest_ring *desc = (struct hal_reo_dest_ring *)rx_desc;322enum hal_reo_dest_ring_push_reason push_reason;323enum hal_reo_dest_ring_error_code err_code;324325push_reason = FIELD_GET(HAL_REO_DEST_RING_INFO0_PUSH_REASON,326desc->info0);327err_code = FIELD_GET(HAL_REO_DEST_RING_INFO0_ERROR_CODE,328desc->info0);329ab->soc_stats.reo_error[err_code]++;330331if (push_reason != HAL_REO_DEST_RING_PUSH_REASON_ERR_DETECTED &&332push_reason != HAL_REO_DEST_RING_PUSH_REASON_ROUTING_INSTRUCTION) {333ath11k_warn(ab, "expected error push reason code, received %d\n",334push_reason);335return -EINVAL;336}337338if (FIELD_GET(HAL_REO_DEST_RING_INFO0_BUFFER_TYPE, desc->info0) !=339HAL_REO_DEST_RING_BUFFER_TYPE_LINK_DESC) {340ath11k_warn(ab, "expected buffer type link_desc");341return -EINVAL;342}343344ath11k_hal_rx_reo_ent_paddr_get(ab, rx_desc, paddr, desc_bank);345346return 0;347}348349int ath11k_hal_wbm_desc_parse_err(struct ath11k_base *ab, void *desc,350struct hal_rx_wbm_rel_info *rel_info)351{352struct hal_wbm_release_ring *wbm_desc = desc;353enum hal_wbm_rel_desc_type type;354enum hal_wbm_rel_src_module rel_src;355enum hal_rx_buf_return_buf_manager ret_buf_mgr;356357type = FIELD_GET(HAL_WBM_RELEASE_INFO0_DESC_TYPE,358wbm_desc->info0);359/* We expect only WBM_REL buffer type */360if (type != HAL_WBM_REL_DESC_TYPE_REL_MSDU) {361WARN_ON(1);362return -EINVAL;363}364365rel_src = FIELD_GET(HAL_WBM_RELEASE_INFO0_REL_SRC_MODULE,366wbm_desc->info0);367if (rel_src != HAL_WBM_REL_SRC_MODULE_RXDMA &&368rel_src != HAL_WBM_REL_SRC_MODULE_REO)369return -EINVAL;370371ret_buf_mgr = FIELD_GET(BUFFER_ADDR_INFO1_RET_BUF_MGR,372wbm_desc->buf_addr_info.info1);373if (ret_buf_mgr != HAL_RX_BUF_RBM_SW3_BM) {374ab->soc_stats.invalid_rbm++;375return -EINVAL;376}377378rel_info->cookie = FIELD_GET(BUFFER_ADDR_INFO1_SW_COOKIE,379wbm_desc->buf_addr_info.info1);380rel_info->err_rel_src = rel_src;381if (rel_src == HAL_WBM_REL_SRC_MODULE_REO) {382rel_info->push_reason =383FIELD_GET(HAL_WBM_RELEASE_INFO0_REO_PUSH_REASON,384wbm_desc->info0);385rel_info->err_code =386FIELD_GET(HAL_WBM_RELEASE_INFO0_REO_ERROR_CODE,387wbm_desc->info0);388} else {389rel_info->push_reason =390FIELD_GET(HAL_WBM_RELEASE_INFO0_RXDMA_PUSH_REASON,391wbm_desc->info0);392rel_info->err_code =393FIELD_GET(HAL_WBM_RELEASE_INFO0_RXDMA_ERROR_CODE,394wbm_desc->info0);395}396397rel_info->first_msdu = FIELD_GET(HAL_WBM_RELEASE_INFO2_FIRST_MSDU,398wbm_desc->info2);399rel_info->last_msdu = FIELD_GET(HAL_WBM_RELEASE_INFO2_LAST_MSDU,400wbm_desc->info2);401return 0;402}403404void ath11k_hal_rx_reo_ent_paddr_get(struct ath11k_base *ab, void *desc,405dma_addr_t *paddr, u32 *desc_bank)406{407struct ath11k_buffer_addr *buff_addr = desc;408409*paddr = ((u64)(FIELD_GET(BUFFER_ADDR_INFO1_ADDR, buff_addr->info1)) << 32) |410FIELD_GET(BUFFER_ADDR_INFO0_ADDR, buff_addr->info0);411412*desc_bank = FIELD_GET(BUFFER_ADDR_INFO1_SW_COOKIE, buff_addr->info1);413}414415void ath11k_hal_rx_msdu_link_desc_set(struct ath11k_base *ab, void *desc,416void *link_desc,417enum hal_wbm_rel_bm_act action)418{419struct hal_wbm_release_ring *dst_desc = desc;420struct hal_wbm_release_ring *src_desc = link_desc;421422dst_desc->buf_addr_info = src_desc->buf_addr_info;423dst_desc->info0 |= FIELD_PREP(HAL_WBM_RELEASE_INFO0_REL_SRC_MODULE,424HAL_WBM_REL_SRC_MODULE_SW) |425FIELD_PREP(HAL_WBM_RELEASE_INFO0_BM_ACTION, action) |426FIELD_PREP(HAL_WBM_RELEASE_INFO0_DESC_TYPE,427HAL_WBM_REL_DESC_TYPE_MSDU_LINK);428}429430void ath11k_hal_reo_status_queue_stats(struct ath11k_base *ab, u32 *reo_desc,431struct hal_reo_status *status)432{433struct hal_tlv_hdr *tlv = (struct hal_tlv_hdr *)reo_desc;434struct hal_reo_get_queue_stats_status *desc =435(struct hal_reo_get_queue_stats_status *)tlv->value;436437status->uniform_hdr.cmd_num =438FIELD_GET(HAL_REO_STATUS_HDR_INFO0_STATUS_NUM,439desc->hdr.info0);440status->uniform_hdr.cmd_status =441FIELD_GET(HAL_REO_STATUS_HDR_INFO0_EXEC_STATUS,442desc->hdr.info0);443444ath11k_dbg(ab, ATH11K_DBG_HAL, "Queue stats status:\n");445ath11k_dbg(ab, ATH11K_DBG_HAL, "header: cmd_num %d status %d\n",446status->uniform_hdr.cmd_num,447status->uniform_hdr.cmd_status);448ath11k_dbg(ab, ATH11K_DBG_HAL, "ssn %ld cur_idx %ld\n",449FIELD_GET(HAL_REO_GET_QUEUE_STATS_STATUS_INFO0_SSN,450desc->info0),451FIELD_GET(HAL_REO_GET_QUEUE_STATS_STATUS_INFO0_CUR_IDX,452desc->info0));453ath11k_dbg(ab, ATH11K_DBG_HAL, "pn = [%08x, %08x, %08x, %08x]\n",454desc->pn[0], desc->pn[1], desc->pn[2], desc->pn[3]);455ath11k_dbg(ab, ATH11K_DBG_HAL,456"last_rx: enqueue_tstamp %08x dequeue_tstamp %08x\n",457desc->last_rx_enqueue_timestamp,458desc->last_rx_dequeue_timestamp);459ath11k_dbg(ab, ATH11K_DBG_HAL,460"rx_bitmap [%08x %08x %08x %08x %08x %08x %08x %08x]\n",461desc->rx_bitmap[0], desc->rx_bitmap[1], desc->rx_bitmap[2],462desc->rx_bitmap[3], desc->rx_bitmap[4], desc->rx_bitmap[5],463desc->rx_bitmap[6], desc->rx_bitmap[7]);464ath11k_dbg(ab, ATH11K_DBG_HAL, "count: cur_mpdu %ld cur_msdu %ld\n",465FIELD_GET(HAL_REO_GET_QUEUE_STATS_STATUS_INFO1_MPDU_COUNT,466desc->info1),467FIELD_GET(HAL_REO_GET_QUEUE_STATS_STATUS_INFO1_MSDU_COUNT,468desc->info1));469ath11k_dbg(ab, ATH11K_DBG_HAL, "fwd_timeout %ld fwd_bar %ld dup_count %ld\n",470FIELD_GET(HAL_REO_GET_QUEUE_STATS_STATUS_INFO2_TIMEOUT_COUNT,471desc->info2),472FIELD_GET(HAL_REO_GET_QUEUE_STATS_STATUS_INFO2_FDTB_COUNT,473desc->info2),474FIELD_GET(HAL_REO_GET_QUEUE_STATS_STATUS_INFO2_DUPLICATE_COUNT,475desc->info2));476ath11k_dbg(ab, ATH11K_DBG_HAL, "frames_in_order %ld bar_rcvd %ld\n",477FIELD_GET(HAL_REO_GET_QUEUE_STATS_STATUS_INFO3_FIO_COUNT,478desc->info3),479FIELD_GET(HAL_REO_GET_QUEUE_STATS_STATUS_INFO3_BAR_RCVD_CNT,480desc->info3));481ath11k_dbg(ab, ATH11K_DBG_HAL, "num_mpdus %d num_msdus %d total_bytes %d\n",482desc->num_mpdu_frames, desc->num_msdu_frames,483desc->total_bytes);484ath11k_dbg(ab, ATH11K_DBG_HAL, "late_rcvd %ld win_jump_2k %ld hole_cnt %ld\n",485FIELD_GET(HAL_REO_GET_QUEUE_STATS_STATUS_INFO4_LATE_RX_MPDU,486desc->info4),487FIELD_GET(HAL_REO_GET_QUEUE_STATS_STATUS_INFO4_WINDOW_JMP2K,488desc->info4),489FIELD_GET(HAL_REO_GET_QUEUE_STATS_STATUS_INFO4_HOLE_COUNT,490desc->info4));491ath11k_dbg(ab, ATH11K_DBG_HAL, "looping count %ld\n",492FIELD_GET(HAL_REO_GET_QUEUE_STATS_STATUS_INFO5_LOOPING_CNT,493desc->info5));494}495496int ath11k_hal_reo_process_status(u8 *reo_desc, u8 *status)497{498struct hal_tlv_hdr *tlv = (struct hal_tlv_hdr *)reo_desc;499struct hal_reo_status_hdr *hdr;500501hdr = (struct hal_reo_status_hdr *)tlv->value;502*status = FIELD_GET(HAL_REO_STATUS_HDR_INFO0_EXEC_STATUS, hdr->info0);503504return FIELD_GET(HAL_REO_STATUS_HDR_INFO0_STATUS_NUM, hdr->info0);505}506507void ath11k_hal_reo_flush_queue_status(struct ath11k_base *ab, u32 *reo_desc,508struct hal_reo_status *status)509{510struct hal_tlv_hdr *tlv = (struct hal_tlv_hdr *)reo_desc;511struct hal_reo_flush_queue_status *desc =512(struct hal_reo_flush_queue_status *)tlv->value;513514status->uniform_hdr.cmd_num =515FIELD_GET(HAL_REO_STATUS_HDR_INFO0_STATUS_NUM,516desc->hdr.info0);517status->uniform_hdr.cmd_status =518FIELD_GET(HAL_REO_STATUS_HDR_INFO0_EXEC_STATUS,519desc->hdr.info0);520status->u.flush_queue.err_detected =521FIELD_GET(HAL_REO_FLUSH_QUEUE_INFO0_ERR_DETECTED,522desc->info0);523}524525void ath11k_hal_reo_flush_cache_status(struct ath11k_base *ab, u32 *reo_desc,526struct hal_reo_status *status)527{528struct ath11k_hal *hal = &ab->hal;529struct hal_tlv_hdr *tlv = (struct hal_tlv_hdr *)reo_desc;530struct hal_reo_flush_cache_status *desc =531(struct hal_reo_flush_cache_status *)tlv->value;532533status->uniform_hdr.cmd_num =534FIELD_GET(HAL_REO_STATUS_HDR_INFO0_STATUS_NUM,535desc->hdr.info0);536status->uniform_hdr.cmd_status =537FIELD_GET(HAL_REO_STATUS_HDR_INFO0_EXEC_STATUS,538desc->hdr.info0);539540status->u.flush_cache.err_detected =541FIELD_GET(HAL_REO_FLUSH_CACHE_STATUS_INFO0_IS_ERR,542desc->info0);543status->u.flush_cache.err_code =544FIELD_GET(HAL_REO_FLUSH_CACHE_STATUS_INFO0_BLOCK_ERR_CODE,545desc->info0);546if (!status->u.flush_cache.err_code)547hal->avail_blk_resource |= BIT(hal->current_blk_index);548549status->u.flush_cache.cache_controller_flush_status_hit =550FIELD_GET(HAL_REO_FLUSH_CACHE_STATUS_INFO0_FLUSH_STATUS_HIT,551desc->info0);552553status->u.flush_cache.cache_controller_flush_status_desc_type =554FIELD_GET(HAL_REO_FLUSH_CACHE_STATUS_INFO0_FLUSH_DESC_TYPE,555desc->info0);556status->u.flush_cache.cache_controller_flush_status_client_id =557FIELD_GET(HAL_REO_FLUSH_CACHE_STATUS_INFO0_FLUSH_CLIENT_ID,558desc->info0);559status->u.flush_cache.cache_controller_flush_status_err =560FIELD_GET(HAL_REO_FLUSH_CACHE_STATUS_INFO0_FLUSH_ERR,561desc->info0);562status->u.flush_cache.cache_controller_flush_status_cnt =563FIELD_GET(HAL_REO_FLUSH_CACHE_STATUS_INFO0_FLUSH_COUNT,564desc->info0);565}566567void ath11k_hal_reo_unblk_cache_status(struct ath11k_base *ab, u32 *reo_desc,568struct hal_reo_status *status)569{570struct ath11k_hal *hal = &ab->hal;571struct hal_tlv_hdr *tlv = (struct hal_tlv_hdr *)reo_desc;572struct hal_reo_unblock_cache_status *desc =573(struct hal_reo_unblock_cache_status *)tlv->value;574575status->uniform_hdr.cmd_num =576FIELD_GET(HAL_REO_STATUS_HDR_INFO0_STATUS_NUM,577desc->hdr.info0);578status->uniform_hdr.cmd_status =579FIELD_GET(HAL_REO_STATUS_HDR_INFO0_EXEC_STATUS,580desc->hdr.info0);581582status->u.unblock_cache.err_detected =583FIELD_GET(HAL_REO_UNBLOCK_CACHE_STATUS_INFO0_IS_ERR,584desc->info0);585status->u.unblock_cache.unblock_type =586FIELD_GET(HAL_REO_UNBLOCK_CACHE_STATUS_INFO0_TYPE,587desc->info0);588589if (!status->u.unblock_cache.err_detected &&590status->u.unblock_cache.unblock_type ==591HAL_REO_STATUS_UNBLOCK_BLOCKING_RESOURCE)592hal->avail_blk_resource &= ~BIT(hal->current_blk_index);593}594595void ath11k_hal_reo_flush_timeout_list_status(struct ath11k_base *ab,596u32 *reo_desc,597struct hal_reo_status *status)598{599struct hal_tlv_hdr *tlv = (struct hal_tlv_hdr *)reo_desc;600struct hal_reo_flush_timeout_list_status *desc =601(struct hal_reo_flush_timeout_list_status *)tlv->value;602603status->uniform_hdr.cmd_num =604FIELD_GET(HAL_REO_STATUS_HDR_INFO0_STATUS_NUM,605desc->hdr.info0);606status->uniform_hdr.cmd_status =607FIELD_GET(HAL_REO_STATUS_HDR_INFO0_EXEC_STATUS,608desc->hdr.info0);609610status->u.timeout_list.err_detected =611FIELD_GET(HAL_REO_FLUSH_TIMEOUT_STATUS_INFO0_IS_ERR,612desc->info0);613status->u.timeout_list.list_empty =614FIELD_GET(HAL_REO_FLUSH_TIMEOUT_STATUS_INFO0_LIST_EMPTY,615desc->info0);616617status->u.timeout_list.release_desc_cnt =618FIELD_GET(HAL_REO_FLUSH_TIMEOUT_STATUS_INFO1_REL_DESC_COUNT,619desc->info1);620status->u.timeout_list.fwd_buf_cnt =621FIELD_GET(HAL_REO_FLUSH_TIMEOUT_STATUS_INFO1_FWD_BUF_COUNT,622desc->info1);623}624625void ath11k_hal_reo_desc_thresh_reached_status(struct ath11k_base *ab,626u32 *reo_desc,627struct hal_reo_status *status)628{629struct hal_tlv_hdr *tlv = (struct hal_tlv_hdr *)reo_desc;630struct hal_reo_desc_thresh_reached_status *desc =631(struct hal_reo_desc_thresh_reached_status *)tlv->value;632633status->uniform_hdr.cmd_num =634FIELD_GET(HAL_REO_STATUS_HDR_INFO0_STATUS_NUM,635desc->hdr.info0);636status->uniform_hdr.cmd_status =637FIELD_GET(HAL_REO_STATUS_HDR_INFO0_EXEC_STATUS,638desc->hdr.info0);639640status->u.desc_thresh_reached.threshold_idx =641FIELD_GET(HAL_REO_DESC_THRESH_STATUS_INFO0_THRESH_INDEX,642desc->info0);643644status->u.desc_thresh_reached.link_desc_counter0 =645FIELD_GET(HAL_REO_DESC_THRESH_STATUS_INFO1_LINK_DESC_COUNTER0,646desc->info1);647648status->u.desc_thresh_reached.link_desc_counter1 =649FIELD_GET(HAL_REO_DESC_THRESH_STATUS_INFO2_LINK_DESC_COUNTER1,650desc->info2);651652status->u.desc_thresh_reached.link_desc_counter2 =653FIELD_GET(HAL_REO_DESC_THRESH_STATUS_INFO3_LINK_DESC_COUNTER2,654desc->info3);655656status->u.desc_thresh_reached.link_desc_counter_sum =657FIELD_GET(HAL_REO_DESC_THRESH_STATUS_INFO4_LINK_DESC_COUNTER_SUM,658desc->info4);659}660661void ath11k_hal_reo_update_rx_reo_queue_status(struct ath11k_base *ab,662u32 *reo_desc,663struct hal_reo_status *status)664{665struct hal_tlv_hdr *tlv = (struct hal_tlv_hdr *)reo_desc;666struct hal_reo_status_hdr *desc =667(struct hal_reo_status_hdr *)tlv->value;668669status->uniform_hdr.cmd_num =670FIELD_GET(HAL_REO_STATUS_HDR_INFO0_STATUS_NUM,671desc->info0);672status->uniform_hdr.cmd_status =673FIELD_GET(HAL_REO_STATUS_HDR_INFO0_EXEC_STATUS,674desc->info0);675}676677u32 ath11k_hal_reo_qdesc_size(u32 ba_window_size, u8 tid)678{679u32 num_ext_desc;680681if (ba_window_size <= 1) {682if (tid != HAL_DESC_REO_NON_QOS_TID)683num_ext_desc = 1;684else685num_ext_desc = 0;686} else if (ba_window_size <= 105) {687num_ext_desc = 1;688} else if (ba_window_size <= 210) {689num_ext_desc = 2;690} else {691num_ext_desc = 3;692}693694return sizeof(struct hal_rx_reo_queue) +695(num_ext_desc * sizeof(struct hal_rx_reo_queue_ext));696}697698void ath11k_hal_reo_qdesc_setup(void *vaddr, int tid, u32 ba_window_size,699u32 start_seq, enum hal_pn_type type)700{701struct hal_rx_reo_queue *qdesc = (struct hal_rx_reo_queue *)vaddr;702struct hal_rx_reo_queue_ext *ext_desc;703704memset(qdesc, 0, sizeof(*qdesc));705706ath11k_hal_reo_set_desc_hdr(&qdesc->desc_hdr, HAL_DESC_REO_OWNED,707HAL_DESC_REO_QUEUE_DESC,708REO_QUEUE_DESC_MAGIC_DEBUG_PATTERN_0);709710qdesc->rx_queue_num = FIELD_PREP(HAL_RX_REO_QUEUE_RX_QUEUE_NUMBER, tid);711712qdesc->info0 =713FIELD_PREP(HAL_RX_REO_QUEUE_INFO0_VLD, 1) |714FIELD_PREP(HAL_RX_REO_QUEUE_INFO0_ASSOC_LNK_DESC_COUNTER, 1) |715FIELD_PREP(HAL_RX_REO_QUEUE_INFO0_AC, ath11k_tid_to_ac(tid));716717if (ba_window_size < 1)718ba_window_size = 1;719720if (ba_window_size == 1 && tid != HAL_DESC_REO_NON_QOS_TID)721ba_window_size++;722723if (ba_window_size == 1)724qdesc->info0 |= FIELD_PREP(HAL_RX_REO_QUEUE_INFO0_RETRY, 1);725726qdesc->info0 |= FIELD_PREP(HAL_RX_REO_QUEUE_INFO0_BA_WINDOW_SIZE,727ba_window_size - 1);728switch (type) {729case HAL_PN_TYPE_NONE:730case HAL_PN_TYPE_WAPI_EVEN:731case HAL_PN_TYPE_WAPI_UNEVEN:732break;733case HAL_PN_TYPE_WPA:734qdesc->info0 |=735FIELD_PREP(HAL_RX_REO_QUEUE_INFO0_PN_CHECK, 1) |736FIELD_PREP(HAL_RX_REO_QUEUE_INFO0_PN_SIZE,737HAL_RX_REO_QUEUE_PN_SIZE_48);738break;739}740741/* TODO: Set Ignore ampdu flags based on BA window size and/or742* AMPDU capabilities743*/744qdesc->info0 |= FIELD_PREP(HAL_RX_REO_QUEUE_INFO0_IGNORE_AMPDU_FLG, 1);745746qdesc->info1 |= FIELD_PREP(HAL_RX_REO_QUEUE_INFO1_SVLD, 0);747748if (start_seq <= 0xfff)749qdesc->info1 = FIELD_PREP(HAL_RX_REO_QUEUE_INFO1_SSN,750start_seq);751752if (tid == HAL_DESC_REO_NON_QOS_TID)753return;754755ext_desc = qdesc->ext_desc;756757/* TODO: HW queue descriptors are currently allocated for max BA758* window size for all QOS TIDs so that same descriptor can be used759* later when ADDBA request is received. This should be changed to760* allocate HW queue descriptors based on BA window size being761* negotiated (0 for non BA cases), and reallocate when BA window762* size changes and also send WMI message to FW to change the REO763* queue descriptor in Rx peer entry as part of dp_rx_tid_update.764*/765memset(ext_desc, 0, sizeof(*ext_desc));766ath11k_hal_reo_set_desc_hdr(&ext_desc->desc_hdr, HAL_DESC_REO_OWNED,767HAL_DESC_REO_QUEUE_EXT_DESC,768REO_QUEUE_DESC_MAGIC_DEBUG_PATTERN_1);769ext_desc++;770memset(ext_desc, 0, sizeof(*ext_desc));771ath11k_hal_reo_set_desc_hdr(&ext_desc->desc_hdr, HAL_DESC_REO_OWNED,772HAL_DESC_REO_QUEUE_EXT_DESC,773REO_QUEUE_DESC_MAGIC_DEBUG_PATTERN_2);774ext_desc++;775memset(ext_desc, 0, sizeof(*ext_desc));776ath11k_hal_reo_set_desc_hdr(&ext_desc->desc_hdr, HAL_DESC_REO_OWNED,777HAL_DESC_REO_QUEUE_EXT_DESC,778REO_QUEUE_DESC_MAGIC_DEBUG_PATTERN_3);779}780781void ath11k_hal_reo_init_cmd_ring(struct ath11k_base *ab,782struct hal_srng *srng)783{784struct hal_srng_params params;785struct hal_tlv_hdr *tlv;786struct hal_reo_get_queue_stats *desc;787int i, cmd_num = 1;788int entry_size;789u8 *entry;790791memset(¶ms, 0, sizeof(params));792793entry_size = ath11k_hal_srng_get_entrysize(ab, HAL_REO_CMD);794ath11k_hal_srng_get_params(ab, srng, ¶ms);795entry = (u8 *)params.ring_base_vaddr;796797for (i = 0; i < params.num_entries; i++) {798tlv = (struct hal_tlv_hdr *)entry;799desc = (struct hal_reo_get_queue_stats *)tlv->value;800desc->cmd.info0 =801FIELD_PREP(HAL_REO_CMD_HDR_INFO0_CMD_NUMBER, cmd_num++);802entry += entry_size;803}804}805806#define HAL_MAX_UL_MU_USERS 37807static inline void808ath11k_hal_rx_handle_ofdma_info(void *rx_tlv,809struct hal_rx_user_status *rx_user_status)810{811struct hal_rx_ppdu_end_user_stats *ppdu_end_user =812(struct hal_rx_ppdu_end_user_stats *)rx_tlv;813814rx_user_status->ul_ofdma_user_v0_word0 = __le32_to_cpu(ppdu_end_user->info6);815816rx_user_status->ul_ofdma_user_v0_word1 = __le32_to_cpu(ppdu_end_user->rsvd2[10]);817}818819static inline void820ath11k_hal_rx_populate_byte_count(void *rx_tlv, void *ppduinfo,821struct hal_rx_user_status *rx_user_status)822{823struct hal_rx_ppdu_end_user_stats *ppdu_end_user =824(struct hal_rx_ppdu_end_user_stats *)rx_tlv;825826rx_user_status->mpdu_ok_byte_count =827FIELD_GET(HAL_RX_PPDU_END_USER_STATS_RSVD2_6_MPDU_OK_BYTE_COUNT,828__le32_to_cpu(ppdu_end_user->rsvd2[6]));829rx_user_status->mpdu_err_byte_count =830FIELD_GET(HAL_RX_PPDU_END_USER_STATS_RSVD2_8_MPDU_ERR_BYTE_COUNT,831__le32_to_cpu(ppdu_end_user->rsvd2[8]));832}833834static inline void835ath11k_hal_rx_populate_mu_user_info(void *rx_tlv, struct hal_rx_mon_ppdu_info *ppdu_info,836struct hal_rx_user_status *rx_user_status)837{838rx_user_status->ast_index = ppdu_info->ast_index;839rx_user_status->tid = ppdu_info->tid;840rx_user_status->tcp_msdu_count =841ppdu_info->tcp_msdu_count;842rx_user_status->udp_msdu_count =843ppdu_info->udp_msdu_count;844rx_user_status->other_msdu_count =845ppdu_info->other_msdu_count;846rx_user_status->frame_control = ppdu_info->frame_control;847rx_user_status->frame_control_info_valid =848ppdu_info->frame_control_info_valid;849rx_user_status->data_sequence_control_info_valid =850ppdu_info->data_sequence_control_info_valid;851rx_user_status->first_data_seq_ctrl =852ppdu_info->first_data_seq_ctrl;853rx_user_status->preamble_type = ppdu_info->preamble_type;854rx_user_status->ht_flags = ppdu_info->ht_flags;855rx_user_status->vht_flags = ppdu_info->vht_flags;856rx_user_status->he_flags = ppdu_info->he_flags;857rx_user_status->rs_flags = ppdu_info->rs_flags;858859rx_user_status->mpdu_cnt_fcs_ok =860ppdu_info->num_mpdu_fcs_ok;861rx_user_status->mpdu_cnt_fcs_err =862ppdu_info->num_mpdu_fcs_err;863864ath11k_hal_rx_populate_byte_count(rx_tlv, ppdu_info, rx_user_status);865}866867static u16 ath11k_hal_rx_mpduinfo_get_peerid(struct ath11k_base *ab,868struct hal_rx_mpdu_info *mpdu_info)869{870return ab->hw_params.hw_ops->mpdu_info_get_peerid(mpdu_info);871}872873static enum hal_rx_mon_status874ath11k_hal_rx_parse_mon_status_tlv(struct ath11k_base *ab,875struct hal_rx_mon_ppdu_info *ppdu_info,876u32 tlv_tag, u8 *tlv_data, u32 userid)877{878u32 info0, info1, value;879u8 he_dcm = 0, he_stbc = 0;880u16 he_gi = 0, he_ltf = 0;881882switch (tlv_tag) {883case HAL_RX_PPDU_START: {884struct hal_rx_ppdu_start *ppdu_start =885(struct hal_rx_ppdu_start *)tlv_data;886887ppdu_info->ppdu_id =888FIELD_GET(HAL_RX_PPDU_START_INFO0_PPDU_ID,889__le32_to_cpu(ppdu_start->info0));890ppdu_info->chan_num = __le32_to_cpu(ppdu_start->chan_num);891ppdu_info->ppdu_ts = __le32_to_cpu(ppdu_start->ppdu_start_ts);892break;893}894case HAL_RX_PPDU_END_USER_STATS: {895struct hal_rx_ppdu_end_user_stats *eu_stats =896(struct hal_rx_ppdu_end_user_stats *)tlv_data;897898info0 = __le32_to_cpu(eu_stats->info0);899info1 = __le32_to_cpu(eu_stats->info1);900901ppdu_info->ast_index =902FIELD_GET(HAL_RX_PPDU_END_USER_STATS_INFO2_AST_INDEX,903__le32_to_cpu(eu_stats->info2));904ppdu_info->tid =905ffs(FIELD_GET(HAL_RX_PPDU_END_USER_STATS_INFO6_TID_BITMAP,906__le32_to_cpu(eu_stats->info6))) - 1;907ppdu_info->tcp_msdu_count =908FIELD_GET(HAL_RX_PPDU_END_USER_STATS_INFO4_TCP_MSDU_CNT,909__le32_to_cpu(eu_stats->info4));910ppdu_info->udp_msdu_count =911FIELD_GET(HAL_RX_PPDU_END_USER_STATS_INFO4_UDP_MSDU_CNT,912__le32_to_cpu(eu_stats->info4));913ppdu_info->other_msdu_count =914FIELD_GET(HAL_RX_PPDU_END_USER_STATS_INFO5_OTHER_MSDU_CNT,915__le32_to_cpu(eu_stats->info5));916ppdu_info->tcp_ack_msdu_count =917FIELD_GET(HAL_RX_PPDU_END_USER_STATS_INFO5_TCP_ACK_MSDU_CNT,918__le32_to_cpu(eu_stats->info5));919ppdu_info->preamble_type =920FIELD_GET(HAL_RX_PPDU_END_USER_STATS_INFO1_PKT_TYPE, info1);921ppdu_info->num_mpdu_fcs_ok =922FIELD_GET(HAL_RX_PPDU_END_USER_STATS_INFO1_MPDU_CNT_FCS_OK,923info1);924ppdu_info->num_mpdu_fcs_err =925FIELD_GET(HAL_RX_PPDU_END_USER_STATS_INFO0_MPDU_CNT_FCS_ERR,926info0);927switch (ppdu_info->preamble_type) {928case HAL_RX_PREAMBLE_11N:929ppdu_info->ht_flags = 1;930break;931case HAL_RX_PREAMBLE_11AC:932ppdu_info->vht_flags = 1;933break;934case HAL_RX_PREAMBLE_11AX:935ppdu_info->he_flags = 1;936break;937default:938break;939}940941if (userid < HAL_MAX_UL_MU_USERS) {942struct hal_rx_user_status *rxuser_stats =943&ppdu_info->userstats;944945ath11k_hal_rx_handle_ofdma_info(tlv_data, rxuser_stats);946ath11k_hal_rx_populate_mu_user_info(tlv_data, ppdu_info,947rxuser_stats);948}949ppdu_info->userstats.mpdu_fcs_ok_bitmap[0] =950__le32_to_cpu(eu_stats->rsvd1[0]);951ppdu_info->userstats.mpdu_fcs_ok_bitmap[1] =952__le32_to_cpu(eu_stats->rsvd1[1]);953954break;955}956case HAL_RX_PPDU_END_USER_STATS_EXT: {957struct hal_rx_ppdu_end_user_stats_ext *eu_stats =958(struct hal_rx_ppdu_end_user_stats_ext *)tlv_data;959ppdu_info->userstats.mpdu_fcs_ok_bitmap[2] = eu_stats->info1;960ppdu_info->userstats.mpdu_fcs_ok_bitmap[3] = eu_stats->info2;961ppdu_info->userstats.mpdu_fcs_ok_bitmap[4] = eu_stats->info3;962ppdu_info->userstats.mpdu_fcs_ok_bitmap[5] = eu_stats->info4;963ppdu_info->userstats.mpdu_fcs_ok_bitmap[6] = eu_stats->info5;964ppdu_info->userstats.mpdu_fcs_ok_bitmap[7] = eu_stats->info6;965break;966}967case HAL_PHYRX_HT_SIG: {968struct hal_rx_ht_sig_info *ht_sig =969(struct hal_rx_ht_sig_info *)tlv_data;970971info0 = __le32_to_cpu(ht_sig->info0);972info1 = __le32_to_cpu(ht_sig->info1);973974ppdu_info->mcs = FIELD_GET(HAL_RX_HT_SIG_INFO_INFO0_MCS, info0);975ppdu_info->bw = FIELD_GET(HAL_RX_HT_SIG_INFO_INFO0_BW, info0);976ppdu_info->is_stbc = FIELD_GET(HAL_RX_HT_SIG_INFO_INFO1_STBC,977info1);978ppdu_info->ldpc = FIELD_GET(HAL_RX_HT_SIG_INFO_INFO1_FEC_CODING, info1);979ppdu_info->gi = info1 & HAL_RX_HT_SIG_INFO_INFO1_GI;980981switch (ppdu_info->mcs) {982case 0 ... 7:983ppdu_info->nss = 1;984break;985case 8 ... 15:986ppdu_info->nss = 2;987break;988case 16 ... 23:989ppdu_info->nss = 3;990break;991case 24 ... 31:992ppdu_info->nss = 4;993break;994}995996if (ppdu_info->nss > 1)997ppdu_info->mcs = ppdu_info->mcs % 8;998999ppdu_info->reception_type = HAL_RX_RECEPTION_TYPE_SU;1000break;1001}1002case HAL_PHYRX_L_SIG_B: {1003struct hal_rx_lsig_b_info *lsigb =1004(struct hal_rx_lsig_b_info *)tlv_data;10051006ppdu_info->rate = FIELD_GET(HAL_RX_LSIG_B_INFO_INFO0_RATE,1007__le32_to_cpu(lsigb->info0));1008ppdu_info->reception_type = HAL_RX_RECEPTION_TYPE_SU;1009break;1010}1011case HAL_PHYRX_L_SIG_A: {1012struct hal_rx_lsig_a_info *lsiga =1013(struct hal_rx_lsig_a_info *)tlv_data;10141015ppdu_info->rate = FIELD_GET(HAL_RX_LSIG_A_INFO_INFO0_RATE,1016__le32_to_cpu(lsiga->info0));1017ppdu_info->reception_type = HAL_RX_RECEPTION_TYPE_SU;1018break;1019}1020case HAL_PHYRX_VHT_SIG_A: {1021struct hal_rx_vht_sig_a_info *vht_sig =1022(struct hal_rx_vht_sig_a_info *)tlv_data;1023u32 nsts;1024u32 group_id;1025u8 gi_setting;10261027info0 = __le32_to_cpu(vht_sig->info0);1028info1 = __le32_to_cpu(vht_sig->info1);10291030ppdu_info->ldpc = FIELD_GET(HAL_RX_VHT_SIG_A_INFO_INFO1_SU_MU_CODING,1031info1);1032ppdu_info->mcs = FIELD_GET(HAL_RX_VHT_SIG_A_INFO_INFO1_MCS,1033info1);1034gi_setting = FIELD_GET(HAL_RX_VHT_SIG_A_INFO_INFO1_GI_SETTING,1035info1);1036switch (gi_setting) {1037case HAL_RX_VHT_SIG_A_NORMAL_GI:1038ppdu_info->gi = HAL_RX_GI_0_8_US;1039break;1040case HAL_RX_VHT_SIG_A_SHORT_GI:1041case HAL_RX_VHT_SIG_A_SHORT_GI_AMBIGUITY:1042ppdu_info->gi = HAL_RX_GI_0_4_US;1043break;1044}10451046ppdu_info->is_stbc = info0 & HAL_RX_VHT_SIG_A_INFO_INFO0_STBC;1047nsts = FIELD_GET(HAL_RX_VHT_SIG_A_INFO_INFO0_NSTS, info0);1048if (ppdu_info->is_stbc && nsts > 0)1049nsts = ((nsts + 1) >> 1) - 1;10501051ppdu_info->nss = (nsts & VHT_SIG_SU_NSS_MASK) + 1;1052ppdu_info->bw = FIELD_GET(HAL_RX_VHT_SIG_A_INFO_INFO0_BW,1053info0);1054ppdu_info->beamformed = info1 &1055HAL_RX_VHT_SIG_A_INFO_INFO1_BEAMFORMED;1056group_id = FIELD_GET(HAL_RX_VHT_SIG_A_INFO_INFO0_GROUP_ID,1057info0);1058if (group_id == 0 || group_id == 63)1059ppdu_info->reception_type = HAL_RX_RECEPTION_TYPE_SU;1060else1061ppdu_info->reception_type =1062HAL_RX_RECEPTION_TYPE_MU_MIMO;1063ppdu_info->vht_flag_values5 = group_id;1064ppdu_info->vht_flag_values3[0] = (((ppdu_info->mcs) << 4) |1065ppdu_info->nss);1066ppdu_info->vht_flag_values2 = ppdu_info->bw;1067ppdu_info->vht_flag_values4 =1068FIELD_GET(HAL_RX_VHT_SIG_A_INFO_INFO1_SU_MU_CODING, info1);1069break;1070}1071case HAL_PHYRX_HE_SIG_A_SU: {1072struct hal_rx_he_sig_a_su_info *he_sig_a =1073(struct hal_rx_he_sig_a_su_info *)tlv_data;10741075ppdu_info->he_flags = 1;1076info0 = __le32_to_cpu(he_sig_a->info0);1077info1 = __le32_to_cpu(he_sig_a->info1);10781079value = FIELD_GET(HAL_RX_HE_SIG_A_SU_INFO_INFO0_FORMAT_IND, info0);10801081if (value == 0)1082ppdu_info->he_data1 = IEEE80211_RADIOTAP_HE_DATA1_FORMAT_TRIG;1083else1084ppdu_info->he_data1 = IEEE80211_RADIOTAP_HE_DATA1_FORMAT_SU;10851086ppdu_info->he_data1 |=1087IEEE80211_RADIOTAP_HE_DATA1_BSS_COLOR_KNOWN |1088IEEE80211_RADIOTAP_HE_DATA1_BEAM_CHANGE_KNOWN |1089IEEE80211_RADIOTAP_HE_DATA1_UL_DL_KNOWN |1090IEEE80211_RADIOTAP_HE_DATA1_DATA_MCS_KNOWN |1091IEEE80211_RADIOTAP_HE_DATA1_DATA_DCM_KNOWN |1092IEEE80211_RADIOTAP_HE_DATA1_CODING_KNOWN |1093IEEE80211_RADIOTAP_HE_DATA1_LDPC_XSYMSEG_KNOWN |1094IEEE80211_RADIOTAP_HE_DATA1_STBC_KNOWN |1095IEEE80211_RADIOTAP_HE_DATA1_BW_RU_ALLOC_KNOWN |1096IEEE80211_RADIOTAP_HE_DATA1_DOPPLER_KNOWN;10971098ppdu_info->he_data2 |=1099IEEE80211_RADIOTAP_HE_DATA2_GI_KNOWN |1100IEEE80211_RADIOTAP_HE_DATA2_TXBF_KNOWN |1101IEEE80211_RADIOTAP_HE_DATA2_PE_DISAMBIG_KNOWN |1102IEEE80211_RADIOTAP_HE_DATA2_TXOP_KNOWN |1103IEEE80211_RADIOTAP_HE_DATA2_NUM_LTF_SYMS_KNOWN |1104IEEE80211_RADIOTAP_HE_DATA2_PRE_FEC_PAD_KNOWN |1105IEEE80211_RADIOTAP_HE_DATA2_MIDAMBLE_KNOWN;11061107value = FIELD_GET(HAL_RX_HE_SIG_A_SU_INFO_INFO0_BSS_COLOR, info0);1108ppdu_info->he_data3 =1109FIELD_PREP(IEEE80211_RADIOTAP_HE_DATA3_BSS_COLOR, value);1110value = FIELD_GET(HAL_RX_HE_SIG_A_SU_INFO_INFO0_BEAM_CHANGE, info0);1111ppdu_info->he_data3 |=1112FIELD_PREP(IEEE80211_RADIOTAP_HE_DATA3_BEAM_CHANGE, value);1113value = FIELD_GET(HAL_RX_HE_SIG_A_SU_INFO_INFO0_DL_UL_FLAG, info0);1114ppdu_info->he_data3 |=1115FIELD_PREP(IEEE80211_RADIOTAP_HE_DATA3_UL_DL, value);1116value = FIELD_GET(HAL_RX_HE_SIG_A_SU_INFO_INFO0_TRANSMIT_MCS, info0);1117ppdu_info->mcs = value;1118ppdu_info->he_data3 |=1119FIELD_PREP(IEEE80211_RADIOTAP_HE_DATA3_DATA_MCS, value);11201121he_dcm = FIELD_GET(HAL_RX_HE_SIG_A_SU_INFO_INFO0_DCM, info0);1122ppdu_info->dcm = he_dcm;1123ppdu_info->he_data3 |=1124FIELD_PREP(IEEE80211_RADIOTAP_HE_DATA3_DATA_DCM, he_dcm);1125value = FIELD_GET(HAL_RX_HE_SIG_A_SU_INFO_INFO1_CODING, info1);1126ppdu_info->ldpc = (value == HAL_RX_SU_MU_CODING_LDPC) ? 1 : 0;1127ppdu_info->he_data3 |=1128FIELD_PREP(IEEE80211_RADIOTAP_HE_DATA3_CODING, value);1129value = FIELD_GET(HAL_RX_HE_SIG_A_SU_INFO_INFO1_LDPC_EXTRA, info1);1130ppdu_info->he_data3 |=1131FIELD_PREP(IEEE80211_RADIOTAP_HE_DATA3_LDPC_XSYMSEG, value);1132he_stbc = FIELD_GET(HAL_RX_HE_SIG_A_SU_INFO_INFO1_STBC, info1);1133ppdu_info->is_stbc = he_stbc;1134ppdu_info->he_data3 |=1135FIELD_PREP(IEEE80211_RADIOTAP_HE_DATA3_STBC, he_stbc);11361137/* data4 */1138value = FIELD_GET(HAL_RX_HE_SIG_A_SU_INFO_INFO0_SPATIAL_REUSE, info0);1139ppdu_info->he_data4 =1140FIELD_PREP(IEEE80211_RADIOTAP_HE_DATA4_SU_MU_SPTL_REUSE, value);11411142/* data5 */1143value = FIELD_GET(HAL_RX_HE_SIG_A_SU_INFO_INFO0_TRANSMIT_BW, info0);1144ppdu_info->bw = value;1145ppdu_info->he_data5 =1146FIELD_PREP(IEEE80211_RADIOTAP_HE_DATA5_DATA_BW_RU_ALLOC, value);1147value = FIELD_GET(HAL_RX_HE_SIG_A_SU_INFO_INFO0_CP_LTF_SIZE, info0);1148switch (value) {1149case 0:1150he_gi = HE_GI_0_8;1151he_ltf = HE_LTF_1_X;1152break;1153case 1:1154he_gi = HE_GI_0_8;1155he_ltf = HE_LTF_2_X;1156break;1157case 2:1158he_gi = HE_GI_1_6;1159he_ltf = HE_LTF_2_X;1160break;1161case 3:1162if (he_dcm && he_stbc) {1163he_gi = HE_GI_0_8;1164he_ltf = HE_LTF_4_X;1165} else {1166he_gi = HE_GI_3_2;1167he_ltf = HE_LTF_4_X;1168}1169break;1170}1171ppdu_info->gi = he_gi;1172he_gi = (he_gi != 0) ? he_gi - 1 : 0;1173ppdu_info->he_data5 |= FIELD_PREP(IEEE80211_RADIOTAP_HE_DATA5_GI, he_gi);1174ppdu_info->ltf_size = he_ltf;1175ppdu_info->he_data5 |=1176FIELD_PREP(IEEE80211_RADIOTAP_HE_DATA5_LTF_SIZE,1177(he_ltf == HE_LTF_4_X) ? he_ltf - 1 : he_ltf);11781179value = FIELD_GET(HAL_RX_HE_SIG_A_SU_INFO_INFO0_NSTS, info0);1180ppdu_info->he_data5 |=1181FIELD_PREP(IEEE80211_RADIOTAP_HE_DATA5_NUM_LTF_SYMS, value);11821183value = FIELD_GET(HAL_RX_HE_SIG_A_SU_INFO_INFO1_PKT_EXT_FACTOR, info1);1184ppdu_info->he_data5 |=1185FIELD_PREP(IEEE80211_RADIOTAP_HE_DATA5_PRE_FEC_PAD, value);11861187value = FIELD_GET(HAL_RX_HE_SIG_A_SU_INFO_INFO1_TXBF, info1);1188ppdu_info->beamformed = value;1189ppdu_info->he_data5 |=1190FIELD_PREP(IEEE80211_RADIOTAP_HE_DATA5_TXBF, value);1191value = FIELD_GET(HAL_RX_HE_SIG_A_SU_INFO_INFO1_PKT_EXT_PE_DISAM, info1);1192ppdu_info->he_data5 |=1193FIELD_PREP(IEEE80211_RADIOTAP_HE_DATA5_PE_DISAMBIG, value);11941195/* data6 */1196value = FIELD_GET(HAL_RX_HE_SIG_A_SU_INFO_INFO0_NSTS, info0);1197value++;1198ppdu_info->nss = value;1199ppdu_info->he_data6 =1200FIELD_PREP(IEEE80211_RADIOTAP_HE_DATA6_NSTS, value);1201value = FIELD_GET(HAL_RX_HE_SIG_A_SU_INFO_INFO1_DOPPLER_IND, info1);1202ppdu_info->he_data6 |=1203FIELD_PREP(IEEE80211_RADIOTAP_HE_DATA6_DOPPLER, value);1204value = FIELD_GET(HAL_RX_HE_SIG_A_SU_INFO_INFO1_TXOP_DURATION, info1);1205ppdu_info->he_data6 |=1206FIELD_PREP(IEEE80211_RADIOTAP_HE_DATA6_TXOP, value);12071208ppdu_info->reception_type = HAL_RX_RECEPTION_TYPE_SU;1209break;1210}1211case HAL_PHYRX_HE_SIG_A_MU_DL: {1212struct hal_rx_he_sig_a_mu_dl_info *he_sig_a_mu_dl =1213(struct hal_rx_he_sig_a_mu_dl_info *)tlv_data;12141215info0 = __le32_to_cpu(he_sig_a_mu_dl->info0);1216info1 = __le32_to_cpu(he_sig_a_mu_dl->info1);12171218ppdu_info->he_mu_flags = 1;12191220ppdu_info->he_data1 = IEEE80211_RADIOTAP_HE_DATA1_FORMAT_MU;1221ppdu_info->he_data1 |=1222IEEE80211_RADIOTAP_HE_DATA1_BSS_COLOR_KNOWN |1223IEEE80211_RADIOTAP_HE_DATA1_UL_DL_KNOWN |1224IEEE80211_RADIOTAP_HE_DATA1_LDPC_XSYMSEG_KNOWN |1225IEEE80211_RADIOTAP_HE_DATA1_STBC_KNOWN |1226IEEE80211_RADIOTAP_HE_DATA1_BW_RU_ALLOC_KNOWN |1227IEEE80211_RADIOTAP_HE_DATA1_DOPPLER_KNOWN;12281229ppdu_info->he_data2 =1230IEEE80211_RADIOTAP_HE_DATA2_GI_KNOWN |1231IEEE80211_RADIOTAP_HE_DATA2_NUM_LTF_SYMS_KNOWN |1232IEEE80211_RADIOTAP_HE_DATA2_PRE_FEC_PAD_KNOWN |1233IEEE80211_RADIOTAP_HE_DATA2_PE_DISAMBIG_KNOWN |1234IEEE80211_RADIOTAP_HE_DATA2_TXOP_KNOWN |1235IEEE80211_RADIOTAP_HE_DATA2_MIDAMBLE_KNOWN;12361237/*data3*/1238value = FIELD_GET(HAL_RX_HE_SIG_A_MU_DL_INFO_INFO0_BSS_COLOR, info0);1239ppdu_info->he_data3 =1240FIELD_PREP(IEEE80211_RADIOTAP_HE_DATA3_BSS_COLOR, value);12411242value = FIELD_GET(HAL_RX_HE_SIG_A_MU_DL_INFO_INFO0_UL_FLAG, info0);1243ppdu_info->he_data3 |=1244FIELD_PREP(IEEE80211_RADIOTAP_HE_DATA3_UL_DL, value);12451246value = FIELD_GET(HAL_RX_HE_SIG_A_MU_DL_INFO_INFO1_LDPC_EXTRA, info1);1247ppdu_info->he_data3 |=1248FIELD_PREP(IEEE80211_RADIOTAP_HE_DATA3_LDPC_XSYMSEG, value);12491250value = FIELD_GET(HAL_RX_HE_SIG_A_MU_DL_INFO_INFO1_STBC, info1);1251he_stbc = value;1252ppdu_info->he_data3 |=1253FIELD_PREP(IEEE80211_RADIOTAP_HE_DATA3_STBC, value);12541255/*data4*/1256value = FIELD_GET(HAL_RX_HE_SIG_A_MU_DL_INFO_INFO0_SPATIAL_REUSE, info0);1257ppdu_info->he_data4 =1258FIELD_PREP(IEEE80211_RADIOTAP_HE_DATA4_SU_MU_SPTL_REUSE, value);12591260/*data5*/1261value = FIELD_GET(HAL_RX_HE_SIG_A_MU_DL_INFO_INFO0_TRANSMIT_BW, info0);1262ppdu_info->bw = value;1263ppdu_info->he_data5 =1264FIELD_PREP(IEEE80211_RADIOTAP_HE_DATA5_DATA_BW_RU_ALLOC, value);12651266value = FIELD_GET(HAL_RX_HE_SIG_A_MU_DL_INFO_INFO0_CP_LTF_SIZE, info0);1267switch (value) {1268case 0:1269he_gi = HE_GI_0_8;1270he_ltf = HE_LTF_4_X;1271break;1272case 1:1273he_gi = HE_GI_0_8;1274he_ltf = HE_LTF_2_X;1275break;1276case 2:1277he_gi = HE_GI_1_6;1278he_ltf = HE_LTF_2_X;1279break;1280case 3:1281he_gi = HE_GI_3_2;1282he_ltf = HE_LTF_4_X;1283break;1284}1285ppdu_info->gi = he_gi;1286he_gi = (he_gi != 0) ? he_gi - 1 : 0;1287ppdu_info->he_data5 |= FIELD_PREP(IEEE80211_RADIOTAP_HE_DATA5_GI, he_gi);1288ppdu_info->ltf_size = he_ltf;1289ppdu_info->he_data5 |=1290FIELD_PREP(IEEE80211_RADIOTAP_HE_DATA5_LTF_SIZE,1291(he_ltf == HE_LTF_4_X) ? he_ltf - 1 : he_ltf);12921293value = FIELD_GET(HAL_RX_HE_SIG_A_MU_DL_INFO_INFO1_NUM_LTF_SYMB, info1);1294ppdu_info->he_data5 |=1295FIELD_PREP(IEEE80211_RADIOTAP_HE_DATA5_NUM_LTF_SYMS, value);12961297value = FIELD_GET(HAL_RX_HE_SIG_A_MU_DL_INFO_INFO1_PKT_EXT_FACTOR,1298info1);1299ppdu_info->he_data5 |=1300FIELD_PREP(IEEE80211_RADIOTAP_HE_DATA5_PRE_FEC_PAD, value);13011302value = FIELD_GET(HAL_RX_HE_SIG_A_MU_DL_INFO_INFO1_PKT_EXT_PE_DISAM,1303info1);1304ppdu_info->he_data5 |=1305FIELD_PREP(IEEE80211_RADIOTAP_HE_DATA5_PE_DISAMBIG, value);13061307/*data6*/1308value = FIELD_GET(HAL_RX_HE_SIG_A_MU_DL_INFO_INFO0_DOPPLER_INDICATION,1309info0);1310ppdu_info->he_data6 |=1311FIELD_PREP(IEEE80211_RADIOTAP_HE_DATA6_DOPPLER, value);13121313value = FIELD_GET(HAL_RX_HE_SIG_A_MU_DL_INFO_INFO1_TXOP_DURATION, info1);1314ppdu_info->he_data6 |=1315FIELD_PREP(IEEE80211_RADIOTAP_HE_DATA6_TXOP, value);13161317/* HE-MU Flags */1318/* HE-MU-flags1 */1319ppdu_info->he_flags1 =1320IEEE80211_RADIOTAP_HE_MU_FLAGS1_SIG_B_MCS_KNOWN |1321IEEE80211_RADIOTAP_HE_MU_FLAGS1_SIG_B_DCM_KNOWN |1322IEEE80211_RADIOTAP_HE_MU_FLAGS1_SIG_B_COMP_KNOWN |1323IEEE80211_RADIOTAP_HE_MU_FLAGS1_SIG_B_SYMS_USERS_KNOWN |1324IEEE80211_RADIOTAP_HE_MU_FLAGS1_CH1_RU_KNOWN;13251326value = FIELD_GET(HAL_RX_HE_SIG_A_MU_DL_INFO_INFO0_MCS_OF_SIGB, info0);1327ppdu_info->he_flags1 |=1328FIELD_PREP(IEEE80211_RADIOTAP_HE_MU_FLAGS1_SIG_B_MCS_KNOWN,1329value);1330value = FIELD_GET(HAL_RX_HE_SIG_A_MU_DL_INFO_INFO0_DCM_OF_SIGB, info0);1331ppdu_info->he_flags1 |=1332FIELD_PREP(IEEE80211_RADIOTAP_HE_MU_FLAGS1_SIG_B_DCM_KNOWN,1333value);13341335/* HE-MU-flags2 */1336ppdu_info->he_flags2 =1337IEEE80211_RADIOTAP_HE_MU_FLAGS2_BW_FROM_SIG_A_BW_KNOWN;13381339value = FIELD_GET(HAL_RX_HE_SIG_A_MU_DL_INFO_INFO0_TRANSMIT_BW, info0);1340ppdu_info->he_flags2 |=1341FIELD_PREP(IEEE80211_RADIOTAP_HE_MU_FLAGS2_BW_FROM_SIG_A_BW,1342value);1343value = FIELD_GET(HAL_RX_HE_SIG_A_MU_DL_INFO_INFO0_COMP_MODE_SIGB, info0);1344ppdu_info->he_flags2 |=1345FIELD_PREP(IEEE80211_RADIOTAP_HE_MU_FLAGS2_SIG_B_COMP, value);1346value = FIELD_GET(HAL_RX_HE_SIG_A_MU_DL_INFO_INFO0_NUM_SIGB_SYMB, info0);1347value = value - 1;1348ppdu_info->he_flags2 |=1349FIELD_PREP(IEEE80211_RADIOTAP_HE_MU_FLAGS2_SIG_B_SYMS_USERS,1350value);13511352ppdu_info->is_stbc = info1 &1353HAL_RX_HE_SIG_A_MU_DL_INFO_INFO1_STBC;1354ppdu_info->reception_type = HAL_RX_RECEPTION_TYPE_MU_MIMO;1355break;1356}1357case HAL_PHYRX_HE_SIG_B1_MU: {1358struct hal_rx_he_sig_b1_mu_info *he_sig_b1_mu =1359(struct hal_rx_he_sig_b1_mu_info *)tlv_data;1360u16 ru_tones;13611362info0 = __le32_to_cpu(he_sig_b1_mu->info0);13631364ru_tones = FIELD_GET(HAL_RX_HE_SIG_B1_MU_INFO_INFO0_RU_ALLOCATION,1365info0);1366ppdu_info->ru_alloc =1367ath11k_mac_phy_he_ru_to_nl80211_he_ru_alloc(ru_tones);1368ppdu_info->he_RU[0] = ru_tones;1369ppdu_info->reception_type = HAL_RX_RECEPTION_TYPE_MU_MIMO;1370break;1371}1372case HAL_PHYRX_HE_SIG_B2_MU: {1373struct hal_rx_he_sig_b2_mu_info *he_sig_b2_mu =1374(struct hal_rx_he_sig_b2_mu_info *)tlv_data;13751376info0 = __le32_to_cpu(he_sig_b2_mu->info0);13771378ppdu_info->he_data1 |= IEEE80211_RADIOTAP_HE_DATA1_DATA_MCS_KNOWN |1379IEEE80211_RADIOTAP_HE_DATA1_CODING_KNOWN;13801381ppdu_info->mcs =1382FIELD_GET(HAL_RX_HE_SIG_B2_MU_INFO_INFO0_STA_MCS, info0);1383ppdu_info->he_data3 |=1384FIELD_PREP(IEEE80211_RADIOTAP_HE_DATA3_DATA_MCS, ppdu_info->mcs);13851386value = FIELD_GET(HAL_RX_HE_SIG_B2_MU_INFO_INFO0_STA_CODING, info0);1387ppdu_info->ldpc = value;1388ppdu_info->he_data3 |=1389FIELD_PREP(IEEE80211_RADIOTAP_HE_DATA3_CODING, value);13901391value = FIELD_GET(HAL_RX_HE_SIG_B2_MU_INFO_INFO0_STA_ID, info0);1392ppdu_info->he_data4 |=1393FIELD_PREP(IEEE80211_RADIOTAP_HE_DATA4_MU_STA_ID, value);13941395ppdu_info->nss =1396FIELD_GET(HAL_RX_HE_SIG_B2_MU_INFO_INFO0_STA_NSTS, info0) + 1;1397break;1398}1399case HAL_PHYRX_HE_SIG_B2_OFDMA: {1400struct hal_rx_he_sig_b2_ofdma_info *he_sig_b2_ofdma =1401(struct hal_rx_he_sig_b2_ofdma_info *)tlv_data;14021403info0 = __le32_to_cpu(he_sig_b2_ofdma->info0);14041405ppdu_info->he_data1 |=1406IEEE80211_RADIOTAP_HE_DATA1_DATA_MCS_KNOWN |1407IEEE80211_RADIOTAP_HE_DATA1_DATA_DCM_KNOWN |1408IEEE80211_RADIOTAP_HE_DATA1_CODING_KNOWN;14091410/* HE-data2 */1411ppdu_info->he_data2 |= IEEE80211_RADIOTAP_HE_DATA2_TXBF_KNOWN;14121413ppdu_info->mcs =1414FIELD_GET(HAL_RX_HE_SIG_B2_OFDMA_INFO_INFO0_STA_MCS,1415info0);1416ppdu_info->he_data3 |=1417FIELD_PREP(IEEE80211_RADIOTAP_HE_DATA3_DATA_MCS, ppdu_info->mcs);14181419value = FIELD_GET(HAL_RX_HE_SIG_B2_OFDMA_INFO_INFO0_STA_DCM, info0);1420he_dcm = value;1421ppdu_info->he_data3 |=1422FIELD_PREP(IEEE80211_RADIOTAP_HE_DATA3_DATA_DCM, value);14231424value = FIELD_GET(HAL_RX_HE_SIG_B2_OFDMA_INFO_INFO0_STA_CODING, info0);1425ppdu_info->ldpc = value;1426ppdu_info->he_data3 |=1427FIELD_PREP(IEEE80211_RADIOTAP_HE_DATA3_CODING, value);14281429/* HE-data4 */1430value = FIELD_GET(HAL_RX_HE_SIG_B2_OFDMA_INFO_INFO0_STA_ID, info0);1431ppdu_info->he_data4 |=1432FIELD_PREP(IEEE80211_RADIOTAP_HE_DATA4_MU_STA_ID, value);14331434ppdu_info->nss =1435FIELD_GET(HAL_RX_HE_SIG_B2_OFDMA_INFO_INFO0_STA_NSTS,1436info0) + 1;1437ppdu_info->beamformed =1438info0 & HAL_RX_HE_SIG_B2_OFDMA_INFO_INFO0_STA_TXBF;1439ppdu_info->reception_type = HAL_RX_RECEPTION_TYPE_MU_OFDMA;1440break;1441}1442case HAL_PHYRX_RSSI_LEGACY: {1443int i;1444bool db2dbm = test_bit(WMI_TLV_SERVICE_HW_DB2DBM_CONVERSION_SUPPORT,1445ab->wmi_ab.svc_map);1446struct hal_rx_phyrx_rssi_legacy_info *rssi =1447(struct hal_rx_phyrx_rssi_legacy_info *)tlv_data;14481449/* TODO: Please note that the combined rssi will not be accurate1450* in MU case. Rssi in MU needs to be retrieved from1451* PHYRX_OTHER_RECEIVE_INFO TLV.1452*/1453ppdu_info->rssi_comb =1454FIELD_GET(HAL_RX_PHYRX_RSSI_LEGACY_INFO_INFO0_RSSI_COMB,1455__le32_to_cpu(rssi->info0));14561457if (db2dbm) {1458for (i = 0; i < ARRAY_SIZE(rssi->preamble); i++) {1459ppdu_info->rssi_chain_pri20[i] =1460le32_get_bits(rssi->preamble[i].rssi_2040,1461HAL_RX_PHYRX_RSSI_PREAMBLE_PRI20);1462}1463}1464break;1465}1466case HAL_RX_MPDU_START: {1467struct hal_rx_mpdu_info *mpdu_info =1468(struct hal_rx_mpdu_info *)tlv_data;1469u16 peer_id;14701471peer_id = ath11k_hal_rx_mpduinfo_get_peerid(ab, mpdu_info);1472if (peer_id)1473ppdu_info->peer_id = peer_id;1474break;1475}1476case HAL_RXPCU_PPDU_END_INFO: {1477struct hal_rx_ppdu_end_duration *ppdu_rx_duration =1478(struct hal_rx_ppdu_end_duration *)tlv_data;1479ppdu_info->rx_duration =1480FIELD_GET(HAL_RX_PPDU_END_DURATION,1481__le32_to_cpu(ppdu_rx_duration->info0));1482ppdu_info->tsft = __le32_to_cpu(ppdu_rx_duration->rsvd0[1]);1483ppdu_info->tsft = (ppdu_info->tsft << 32) |1484__le32_to_cpu(ppdu_rx_duration->rsvd0[0]);1485break;1486}1487case HAL_DUMMY:1488return HAL_RX_MON_STATUS_BUF_DONE;1489case HAL_RX_PPDU_END_STATUS_DONE:1490case 0:1491return HAL_RX_MON_STATUS_PPDU_DONE;1492default:1493break;1494}14951496return HAL_RX_MON_STATUS_PPDU_NOT_DONE;1497}14981499enum hal_rx_mon_status1500ath11k_hal_rx_parse_mon_status(struct ath11k_base *ab,1501struct hal_rx_mon_ppdu_info *ppdu_info,1502struct sk_buff *skb)1503{1504struct hal_tlv_hdr *tlv;1505enum hal_rx_mon_status hal_status = HAL_RX_MON_STATUS_BUF_DONE;1506u16 tlv_tag;1507u16 tlv_len;1508u32 tlv_userid = 0;1509u8 *ptr = skb->data;15101511do {1512tlv = (struct hal_tlv_hdr *)ptr;1513tlv_tag = FIELD_GET(HAL_TLV_HDR_TAG, tlv->tl);1514tlv_len = FIELD_GET(HAL_TLV_HDR_LEN, tlv->tl);1515tlv_userid = FIELD_GET(HAL_TLV_USR_ID, tlv->tl);1516ptr += sizeof(*tlv);15171518/* The actual length of PPDU_END is the combined length of many PHY1519* TLVs that follow. Skip the TLV header and1520* rx_rxpcu_classification_overview that follows the header to get to1521* next TLV.1522*/1523if (tlv_tag == HAL_RX_PPDU_END)1524tlv_len = sizeof(struct hal_rx_rxpcu_classification_overview);15251526hal_status = ath11k_hal_rx_parse_mon_status_tlv(ab, ppdu_info,1527tlv_tag, ptr, tlv_userid);1528ptr += tlv_len;1529ptr = PTR_ALIGN(ptr, HAL_TLV_ALIGN);15301531if ((ptr - skb->data) >= DP_RX_BUFFER_SIZE)1532break;1533} while (hal_status == HAL_RX_MON_STATUS_PPDU_NOT_DONE);15341535return hal_status;1536}15371538void ath11k_hal_rx_reo_ent_buf_paddr_get(void *rx_desc, dma_addr_t *paddr,1539u32 *sw_cookie, void **pp_buf_addr,1540u8 *rbm, u32 *msdu_cnt)1541{1542struct hal_reo_entrance_ring *reo_ent_ring =1543(struct hal_reo_entrance_ring *)rx_desc;1544struct ath11k_buffer_addr *buf_addr_info;1545struct rx_mpdu_desc *rx_mpdu_desc_info_details;15461547rx_mpdu_desc_info_details =1548(struct rx_mpdu_desc *)&reo_ent_ring->rx_mpdu_info;15491550*msdu_cnt = FIELD_GET(RX_MPDU_DESC_INFO0_MSDU_COUNT,1551rx_mpdu_desc_info_details->info0);15521553buf_addr_info = (struct ath11k_buffer_addr *)&reo_ent_ring->buf_addr_info;15541555*paddr = (((u64)FIELD_GET(BUFFER_ADDR_INFO1_ADDR,1556buf_addr_info->info1)) << 32) |1557FIELD_GET(BUFFER_ADDR_INFO0_ADDR,1558buf_addr_info->info0);15591560*sw_cookie = FIELD_GET(BUFFER_ADDR_INFO1_SW_COOKIE,1561buf_addr_info->info1);1562*rbm = FIELD_GET(BUFFER_ADDR_INFO1_RET_BUF_MGR,1563buf_addr_info->info1);15641565*pp_buf_addr = (void *)buf_addr_info;1566}15671568void1569ath11k_hal_rx_sw_mon_ring_buf_paddr_get(void *rx_desc,1570struct hal_sw_mon_ring_entries *sw_mon_entries)1571{1572struct hal_sw_monitor_ring *sw_mon_ring = rx_desc;1573struct ath11k_buffer_addr *buf_addr_info;1574struct ath11k_buffer_addr *status_buf_addr_info;1575struct rx_mpdu_desc *rx_mpdu_desc_info_details;15761577rx_mpdu_desc_info_details = &sw_mon_ring->rx_mpdu_info;15781579sw_mon_entries->msdu_cnt = FIELD_GET(RX_MPDU_DESC_INFO0_MSDU_COUNT,1580rx_mpdu_desc_info_details->info0);15811582buf_addr_info = &sw_mon_ring->buf_addr_info;1583status_buf_addr_info = &sw_mon_ring->status_buf_addr_info;15841585sw_mon_entries->mon_dst_paddr = (((u64)FIELD_GET(BUFFER_ADDR_INFO1_ADDR,1586buf_addr_info->info1)) << 32) |1587FIELD_GET(BUFFER_ADDR_INFO0_ADDR,1588buf_addr_info->info0);15891590sw_mon_entries->mon_status_paddr =1591(((u64)FIELD_GET(BUFFER_ADDR_INFO1_ADDR,1592status_buf_addr_info->info1)) << 32) |1593FIELD_GET(BUFFER_ADDR_INFO0_ADDR,1594status_buf_addr_info->info0);15951596sw_mon_entries->mon_dst_sw_cookie = FIELD_GET(BUFFER_ADDR_INFO1_SW_COOKIE,1597buf_addr_info->info1);15981599sw_mon_entries->mon_status_sw_cookie = FIELD_GET(BUFFER_ADDR_INFO1_SW_COOKIE,1600status_buf_addr_info->info1);16011602sw_mon_entries->status_buf_count = FIELD_GET(HAL_SW_MON_RING_INFO0_STATUS_BUF_CNT,1603sw_mon_ring->info0);16041605sw_mon_entries->dst_buf_addr_info = buf_addr_info;1606sw_mon_entries->status_buf_addr_info = status_buf_addr_info;16071608sw_mon_entries->ppdu_id =1609FIELD_GET(HAL_SW_MON_RING_INFO1_PHY_PPDU_ID, sw_mon_ring->info1);1610}161116121613