Path: blob/main/sys/contrib/dev/athk/ath11k/debugfs_sta.c
106275 views
// SPDX-License-Identifier: BSD-3-Clause-Clear1/*2* Copyright (c) 2018-2019 The Linux Foundation. All rights reserved.3* Copyright (c) 2021-2023 Qualcomm Innovation Center, Inc. All rights reserved.4* Copyright (c) Qualcomm Technologies, Inc. and/or its subsidiaries.5*/67#include <linux/vmalloc.h>89#include "debugfs_sta.h"10#include "core.h"11#include "peer.h"12#include "debug.h"13#include "dp_tx.h"14#include "debugfs_htt_stats.h"1516void ath11k_debugfs_sta_add_tx_stats(struct ath11k_sta *arsta,17struct ath11k_per_peer_tx_stats *peer_stats,18u8 legacy_rate_idx)19{20struct rate_info *txrate = &arsta->txrate;21struct ath11k_htt_tx_stats *tx_stats;22int gi, mcs, bw, nss;2324if (!arsta->tx_stats)25return;2627tx_stats = arsta->tx_stats;28gi = FIELD_GET(RATE_INFO_FLAGS_SHORT_GI, arsta->txrate.flags);29mcs = txrate->mcs;30bw = ath11k_mac_mac80211_bw_to_ath11k_bw(txrate->bw);31nss = txrate->nss - 1;3233#define STATS_OP_FMT(name) tx_stats->stats[ATH11K_STATS_TYPE_##name]3435if (txrate->flags & RATE_INFO_FLAGS_HE_MCS) {36STATS_OP_FMT(SUCC).he[0][mcs] += peer_stats->succ_bytes;37STATS_OP_FMT(SUCC).he[1][mcs] += peer_stats->succ_pkts;38STATS_OP_FMT(FAIL).he[0][mcs] += peer_stats->failed_bytes;39STATS_OP_FMT(FAIL).he[1][mcs] += peer_stats->failed_pkts;40STATS_OP_FMT(RETRY).he[0][mcs] += peer_stats->retry_bytes;41STATS_OP_FMT(RETRY).he[1][mcs] += peer_stats->retry_pkts;42} else if (txrate->flags & RATE_INFO_FLAGS_VHT_MCS) {43STATS_OP_FMT(SUCC).vht[0][mcs] += peer_stats->succ_bytes;44STATS_OP_FMT(SUCC).vht[1][mcs] += peer_stats->succ_pkts;45STATS_OP_FMT(FAIL).vht[0][mcs] += peer_stats->failed_bytes;46STATS_OP_FMT(FAIL).vht[1][mcs] += peer_stats->failed_pkts;47STATS_OP_FMT(RETRY).vht[0][mcs] += peer_stats->retry_bytes;48STATS_OP_FMT(RETRY).vht[1][mcs] += peer_stats->retry_pkts;49} else if (txrate->flags & RATE_INFO_FLAGS_MCS) {50STATS_OP_FMT(SUCC).ht[0][mcs] += peer_stats->succ_bytes;51STATS_OP_FMT(SUCC).ht[1][mcs] += peer_stats->succ_pkts;52STATS_OP_FMT(FAIL).ht[0][mcs] += peer_stats->failed_bytes;53STATS_OP_FMT(FAIL).ht[1][mcs] += peer_stats->failed_pkts;54STATS_OP_FMT(RETRY).ht[0][mcs] += peer_stats->retry_bytes;55STATS_OP_FMT(RETRY).ht[1][mcs] += peer_stats->retry_pkts;56} else {57mcs = legacy_rate_idx;5859STATS_OP_FMT(SUCC).legacy[0][mcs] += peer_stats->succ_bytes;60STATS_OP_FMT(SUCC).legacy[1][mcs] += peer_stats->succ_pkts;61STATS_OP_FMT(FAIL).legacy[0][mcs] += peer_stats->failed_bytes;62STATS_OP_FMT(FAIL).legacy[1][mcs] += peer_stats->failed_pkts;63STATS_OP_FMT(RETRY).legacy[0][mcs] += peer_stats->retry_bytes;64STATS_OP_FMT(RETRY).legacy[1][mcs] += peer_stats->retry_pkts;65}6667if (peer_stats->is_ampdu) {68tx_stats->ba_fails += peer_stats->ba_fails;6970if (txrate->flags & RATE_INFO_FLAGS_HE_MCS) {71STATS_OP_FMT(AMPDU).he[0][mcs] +=72peer_stats->succ_bytes + peer_stats->retry_bytes;73STATS_OP_FMT(AMPDU).he[1][mcs] +=74peer_stats->succ_pkts + peer_stats->retry_pkts;75} else if (txrate->flags & RATE_INFO_FLAGS_MCS) {76STATS_OP_FMT(AMPDU).ht[0][mcs] +=77peer_stats->succ_bytes + peer_stats->retry_bytes;78STATS_OP_FMT(AMPDU).ht[1][mcs] +=79peer_stats->succ_pkts + peer_stats->retry_pkts;80} else {81STATS_OP_FMT(AMPDU).vht[0][mcs] +=82peer_stats->succ_bytes + peer_stats->retry_bytes;83STATS_OP_FMT(AMPDU).vht[1][mcs] +=84peer_stats->succ_pkts + peer_stats->retry_pkts;85}86STATS_OP_FMT(AMPDU).bw[0][bw] +=87peer_stats->succ_bytes + peer_stats->retry_bytes;88STATS_OP_FMT(AMPDU).nss[0][nss] +=89peer_stats->succ_bytes + peer_stats->retry_bytes;90STATS_OP_FMT(AMPDU).gi[0][gi] +=91peer_stats->succ_bytes + peer_stats->retry_bytes;92STATS_OP_FMT(AMPDU).bw[1][bw] +=93peer_stats->succ_pkts + peer_stats->retry_pkts;94STATS_OP_FMT(AMPDU).nss[1][nss] +=95peer_stats->succ_pkts + peer_stats->retry_pkts;96STATS_OP_FMT(AMPDU).gi[1][gi] +=97peer_stats->succ_pkts + peer_stats->retry_pkts;98} else {99tx_stats->ack_fails += peer_stats->ba_fails;100}101102STATS_OP_FMT(SUCC).bw[0][bw] += peer_stats->succ_bytes;103STATS_OP_FMT(SUCC).nss[0][nss] += peer_stats->succ_bytes;104STATS_OP_FMT(SUCC).gi[0][gi] += peer_stats->succ_bytes;105106STATS_OP_FMT(SUCC).bw[1][bw] += peer_stats->succ_pkts;107STATS_OP_FMT(SUCC).nss[1][nss] += peer_stats->succ_pkts;108STATS_OP_FMT(SUCC).gi[1][gi] += peer_stats->succ_pkts;109110STATS_OP_FMT(FAIL).bw[0][bw] += peer_stats->failed_bytes;111STATS_OP_FMT(FAIL).nss[0][nss] += peer_stats->failed_bytes;112STATS_OP_FMT(FAIL).gi[0][gi] += peer_stats->failed_bytes;113114STATS_OP_FMT(FAIL).bw[1][bw] += peer_stats->failed_pkts;115STATS_OP_FMT(FAIL).nss[1][nss] += peer_stats->failed_pkts;116STATS_OP_FMT(FAIL).gi[1][gi] += peer_stats->failed_pkts;117118STATS_OP_FMT(RETRY).bw[0][bw] += peer_stats->retry_bytes;119STATS_OP_FMT(RETRY).nss[0][nss] += peer_stats->retry_bytes;120STATS_OP_FMT(RETRY).gi[0][gi] += peer_stats->retry_bytes;121122STATS_OP_FMT(RETRY).bw[1][bw] += peer_stats->retry_pkts;123STATS_OP_FMT(RETRY).nss[1][nss] += peer_stats->retry_pkts;124STATS_OP_FMT(RETRY).gi[1][gi] += peer_stats->retry_pkts;125126tx_stats->tx_duration += peer_stats->duration;127}128129void ath11k_debugfs_sta_update_txcompl(struct ath11k *ar,130struct hal_tx_status *ts)131{132ath11k_dp_tx_update_txcompl(ar, ts);133}134135static ssize_t ath11k_dbg_sta_dump_tx_stats(struct file *file,136char __user *user_buf,137size_t count, loff_t *ppos)138{139struct ieee80211_sta *sta = file->private_data;140struct ath11k_sta *arsta = ath11k_sta_to_arsta(sta);141struct ath11k *ar = arsta->arvif->ar;142struct ath11k_htt_data_stats *stats;143static const char *str_name[ATH11K_STATS_TYPE_MAX] = {"succ", "fail",144"retry", "ampdu"};145static const char *str[ATH11K_COUNTER_TYPE_MAX] = {"bytes", "packets"};146int len = 0, i, j, k, retval = 0;147const int size = 2 * 4096;148char *buf;149150if (!arsta->tx_stats)151return -ENOENT;152153buf = kzalloc(size, GFP_KERNEL);154if (!buf)155return -ENOMEM;156157mutex_lock(&ar->conf_mutex);158159spin_lock_bh(&ar->data_lock);160for (k = 0; k < ATH11K_STATS_TYPE_MAX; k++) {161for (j = 0; j < ATH11K_COUNTER_TYPE_MAX; j++) {162stats = &arsta->tx_stats->stats[k];163len += scnprintf(buf + len, size - len, "%s_%s\n",164str_name[k],165str[j]);166len += scnprintf(buf + len, size - len,167" HE MCS %s\n",168str[j]);169for (i = 0; i < ATH11K_HE_MCS_NUM; i++)170len += scnprintf(buf + len, size - len,171" %llu ",172stats->he[j][i]);173len += scnprintf(buf + len, size - len, "\n");174len += scnprintf(buf + len, size - len,175" VHT MCS %s\n",176str[j]);177for (i = 0; i < ATH11K_VHT_MCS_NUM; i++)178len += scnprintf(buf + len, size - len,179" %llu ",180stats->vht[j][i]);181len += scnprintf(buf + len, size - len, "\n");182len += scnprintf(buf + len, size - len, " HT MCS %s\n",183str[j]);184for (i = 0; i < ATH11K_HT_MCS_NUM; i++)185len += scnprintf(buf + len, size - len,186" %llu ", stats->ht[j][i]);187len += scnprintf(buf + len, size - len, "\n");188len += scnprintf(buf + len, size - len,189" BW %s (20,40,80,160 MHz)\n", str[j]);190len += scnprintf(buf + len, size - len,191" %llu %llu %llu %llu\n",192stats->bw[j][0], stats->bw[j][1],193stats->bw[j][2], stats->bw[j][3]);194len += scnprintf(buf + len, size - len,195" NSS %s (1x1,2x2,3x3,4x4)\n", str[j]);196len += scnprintf(buf + len, size - len,197" %llu %llu %llu %llu\n",198stats->nss[j][0], stats->nss[j][1],199stats->nss[j][2], stats->nss[j][3]);200len += scnprintf(buf + len, size - len,201" GI %s (0.4us,0.8us,1.6us,3.2us)\n",202str[j]);203len += scnprintf(buf + len, size - len,204" %llu %llu %llu %llu\n",205stats->gi[j][0], stats->gi[j][1],206stats->gi[j][2], stats->gi[j][3]);207len += scnprintf(buf + len, size - len,208" legacy rate %s (1,2 ... Mbps)\n ",209str[j]);210for (i = 0; i < ATH11K_LEGACY_NUM; i++)211len += scnprintf(buf + len, size - len, "%llu ",212stats->legacy[j][i]);213len += scnprintf(buf + len, size - len, "\n");214}215}216217len += scnprintf(buf + len, size - len,218"\nTX duration\n %llu usecs\n",219arsta->tx_stats->tx_duration);220len += scnprintf(buf + len, size - len,221"BA fails\n %llu\n", arsta->tx_stats->ba_fails);222len += scnprintf(buf + len, size - len,223"ack fails\n %llu\n", arsta->tx_stats->ack_fails);224spin_unlock_bh(&ar->data_lock);225226if (len > size)227len = size;228retval = simple_read_from_buffer(user_buf, count, ppos, buf, len);229kfree(buf);230231mutex_unlock(&ar->conf_mutex);232return retval;233}234235static const struct file_operations fops_tx_stats = {236.read = ath11k_dbg_sta_dump_tx_stats,237.open = simple_open,238.owner = THIS_MODULE,239.llseek = default_llseek,240};241242static ssize_t ath11k_dbg_sta_dump_rx_stats(struct file *file,243char __user *user_buf,244size_t count, loff_t *ppos)245{246struct ieee80211_sta *sta = file->private_data;247struct ath11k_sta *arsta = ath11k_sta_to_arsta(sta);248struct ath11k *ar = arsta->arvif->ar;249struct ath11k_rx_peer_stats *rx_stats = arsta->rx_stats;250int len = 0, i, retval = 0;251const int size = 4096;252char *buf;253254if (!rx_stats)255return -ENOENT;256257buf = kzalloc(size, GFP_KERNEL);258if (!buf)259return -ENOMEM;260261mutex_lock(&ar->conf_mutex);262spin_lock_bh(&ar->ab->base_lock);263264len += scnprintf(buf + len, size - len, "RX peer stats:\n");265len += scnprintf(buf + len, size - len, "Num of MSDUs: %llu\n",266rx_stats->num_msdu);267len += scnprintf(buf + len, size - len, "Num of MSDUs with TCP L4: %llu\n",268rx_stats->tcp_msdu_count);269len += scnprintf(buf + len, size - len, "Num of MSDUs with UDP L4: %llu\n",270rx_stats->udp_msdu_count);271len += scnprintf(buf + len, size - len, "Num of MSDUs part of AMPDU: %llu\n",272rx_stats->ampdu_msdu_count);273len += scnprintf(buf + len, size - len, "Num of MSDUs not part of AMPDU: %llu\n",274rx_stats->non_ampdu_msdu_count);275len += scnprintf(buf + len, size - len, "Num of MSDUs using STBC: %llu\n",276rx_stats->stbc_count);277len += scnprintf(buf + len, size - len, "Num of MSDUs beamformed: %llu\n",278rx_stats->beamformed_count);279len += scnprintf(buf + len, size - len, "Num of MPDUs with FCS ok: %llu\n",280rx_stats->num_mpdu_fcs_ok);281len += scnprintf(buf + len, size - len, "Num of MPDUs with FCS error: %llu\n",282rx_stats->num_mpdu_fcs_err);283len += scnprintf(buf + len, size - len,284"GI: 0.8us %llu 0.4us %llu 1.6us %llu 3.2us %llu\n",285rx_stats->gi_count[0], rx_stats->gi_count[1],286rx_stats->gi_count[2], rx_stats->gi_count[3]);287len += scnprintf(buf + len, size - len,288"BW: 20Mhz %llu 40Mhz %llu 80Mhz %llu 160Mhz %llu\n",289rx_stats->bw_count[0], rx_stats->bw_count[1],290rx_stats->bw_count[2], rx_stats->bw_count[3]);291len += scnprintf(buf + len, size - len, "BCC %llu LDPC %llu\n",292rx_stats->coding_count[0], rx_stats->coding_count[1]);293len += scnprintf(buf + len, size - len,294"preamble: 11A %llu 11B %llu 11N %llu 11AC %llu 11AX %llu\n",295rx_stats->pream_cnt[0], rx_stats->pream_cnt[1],296rx_stats->pream_cnt[2], rx_stats->pream_cnt[3],297rx_stats->pream_cnt[4]);298len += scnprintf(buf + len, size - len,299"reception type: SU %llu MU_MIMO %llu MU_OFDMA %llu MU_OFDMA_MIMO %llu\n",300rx_stats->reception_type[0], rx_stats->reception_type[1],301rx_stats->reception_type[2], rx_stats->reception_type[3]);302len += scnprintf(buf + len, size - len, "TID(0-15) Legacy TID(16):");303for (i = 0; i <= IEEE80211_NUM_TIDS; i++)304len += scnprintf(buf + len, size - len, "%llu ", rx_stats->tid_count[i]);305len += scnprintf(buf + len, size - len, "\nMCS(0-11) Legacy MCS(12):");306for (i = 0; i < HAL_RX_MAX_MCS + 1; i++)307len += scnprintf(buf + len, size - len, "%llu ", rx_stats->mcs_count[i]);308len += scnprintf(buf + len, size - len, "\nNSS(1-8):");309for (i = 0; i < HAL_RX_MAX_NSS; i++)310len += scnprintf(buf + len, size - len, "%llu ", rx_stats->nss_count[i]);311len += scnprintf(buf + len, size - len, "\nRX Duration:%llu ",312rx_stats->rx_duration);313len += scnprintf(buf + len, size - len,314"\nDCM: %llu\nRU: 26 %llu 52: %llu 106: %llu 242: %llu 484: %llu 996: %llu\n",315rx_stats->dcm_count, rx_stats->ru_alloc_cnt[0],316rx_stats->ru_alloc_cnt[1], rx_stats->ru_alloc_cnt[2],317rx_stats->ru_alloc_cnt[3], rx_stats->ru_alloc_cnt[4],318rx_stats->ru_alloc_cnt[5]);319320len += scnprintf(buf + len, size - len, "\n");321322spin_unlock_bh(&ar->ab->base_lock);323324if (len > size)325len = size;326retval = simple_read_from_buffer(user_buf, count, ppos, buf, len);327kfree(buf);328329mutex_unlock(&ar->conf_mutex);330return retval;331}332333static const struct file_operations fops_rx_stats = {334.read = ath11k_dbg_sta_dump_rx_stats,335.open = simple_open,336.owner = THIS_MODULE,337.llseek = default_llseek,338};339340static int341ath11k_dbg_sta_open_htt_peer_stats(struct inode *inode, struct file *file)342{343struct ieee80211_sta *sta = inode->i_private;344struct ath11k_sta *arsta = ath11k_sta_to_arsta(sta);345struct ath11k *ar = arsta->arvif->ar;346struct debug_htt_stats_req *stats_req;347int type = ar->debug.htt_stats.type;348int ret;349350if ((type != ATH11K_DBG_HTT_EXT_STATS_PEER_INFO &&351type != ATH11K_DBG_HTT_EXT_STATS_PEER_CTRL_PATH_TXRX_STATS) ||352type == ATH11K_DBG_HTT_EXT_STATS_RESET)353return -EPERM;354355stats_req = vzalloc(sizeof(*stats_req) + ATH11K_HTT_STATS_BUF_SIZE);356if (!stats_req)357return -ENOMEM;358359mutex_lock(&ar->conf_mutex);360ar->debug.htt_stats.stats_req = stats_req;361stats_req->type = type;362memcpy(stats_req->peer_addr, sta->addr, ETH_ALEN);363ret = ath11k_debugfs_htt_stats_req(ar);364mutex_unlock(&ar->conf_mutex);365if (ret < 0)366goto out;367368file->private_data = stats_req;369return 0;370out:371vfree(stats_req);372ar->debug.htt_stats.stats_req = NULL;373return ret;374}375376static int377ath11k_dbg_sta_release_htt_peer_stats(struct inode *inode, struct file *file)378{379struct ieee80211_sta *sta = inode->i_private;380struct ath11k_sta *arsta = ath11k_sta_to_arsta(sta);381struct ath11k *ar = arsta->arvif->ar;382383mutex_lock(&ar->conf_mutex);384vfree(file->private_data);385ar->debug.htt_stats.stats_req = NULL;386mutex_unlock(&ar->conf_mutex);387388return 0;389}390391static ssize_t ath11k_dbg_sta_read_htt_peer_stats(struct file *file,392char __user *user_buf,393size_t count, loff_t *ppos)394{395struct debug_htt_stats_req *stats_req = file->private_data;396char *buf;397u32 length = 0;398399buf = stats_req->buf;400length = min_t(u32, stats_req->buf_len, ATH11K_HTT_STATS_BUF_SIZE);401return simple_read_from_buffer(user_buf, count, ppos, buf, length);402}403404static const struct file_operations fops_htt_peer_stats = {405.open = ath11k_dbg_sta_open_htt_peer_stats,406.release = ath11k_dbg_sta_release_htt_peer_stats,407.read = ath11k_dbg_sta_read_htt_peer_stats,408.owner = THIS_MODULE,409.llseek = default_llseek,410};411412static ssize_t ath11k_dbg_sta_write_peer_pktlog(struct file *file,413const char __user *buf,414size_t count, loff_t *ppos)415{416struct ieee80211_sta *sta = file->private_data;417struct ath11k_sta *arsta = ath11k_sta_to_arsta(sta);418struct ath11k *ar = arsta->arvif->ar;419int ret, enable;420421mutex_lock(&ar->conf_mutex);422423if (ar->state != ATH11K_STATE_ON) {424ret = -ENETDOWN;425goto out;426}427428ret = kstrtoint_from_user(buf, count, 0, &enable);429if (ret)430goto out;431432ar->debug.pktlog_peer_valid = enable;433memcpy(ar->debug.pktlog_peer_addr, sta->addr, ETH_ALEN);434435/* Send peer based pktlog enable/disable */436ret = ath11k_wmi_pdev_peer_pktlog_filter(ar, sta->addr, enable);437if (ret) {438ath11k_warn(ar->ab, "failed to set peer pktlog filter %pM: %d\n",439sta->addr, ret);440goto out;441}442443ath11k_dbg(ar->ab, ATH11K_DBG_WMI, "peer pktlog filter set to %d\n",444enable);445ret = count;446447out:448mutex_unlock(&ar->conf_mutex);449return ret;450}451452static ssize_t ath11k_dbg_sta_read_peer_pktlog(struct file *file,453char __user *ubuf,454size_t count, loff_t *ppos)455{456struct ieee80211_sta *sta = file->private_data;457struct ath11k_sta *arsta = ath11k_sta_to_arsta(sta);458struct ath11k *ar = arsta->arvif->ar;459char buf[32] = {};460int len;461462mutex_lock(&ar->conf_mutex);463len = scnprintf(buf, sizeof(buf), "%08x %pM\n",464ar->debug.pktlog_peer_valid,465ar->debug.pktlog_peer_addr);466mutex_unlock(&ar->conf_mutex);467468return simple_read_from_buffer(ubuf, count, ppos, buf, len);469}470471static const struct file_operations fops_peer_pktlog = {472.write = ath11k_dbg_sta_write_peer_pktlog,473.read = ath11k_dbg_sta_read_peer_pktlog,474.open = simple_open,475.owner = THIS_MODULE,476.llseek = default_llseek,477};478479static ssize_t ath11k_dbg_sta_write_delba(struct file *file,480const char __user *user_buf,481size_t count, loff_t *ppos)482{483struct ieee80211_sta *sta = file->private_data;484struct ath11k_sta *arsta = ath11k_sta_to_arsta(sta);485struct ath11k *ar = arsta->arvif->ar;486u32 tid, initiator, reason;487int ret;488char buf[64] = {};489490ret = simple_write_to_buffer(buf, sizeof(buf) - 1, ppos,491user_buf, count);492if (ret <= 0)493return ret;494495ret = sscanf(buf, "%u %u %u", &tid, &initiator, &reason);496if (ret != 3)497return -EINVAL;498499/* Valid TID values are 0 through 15 */500if (tid > HAL_DESC_REO_NON_QOS_TID - 1)501return -EINVAL;502503mutex_lock(&ar->conf_mutex);504if (ar->state != ATH11K_STATE_ON ||505arsta->aggr_mode != ATH11K_DBG_AGGR_MODE_MANUAL) {506ret = count;507goto out;508}509510ret = ath11k_wmi_delba_send(ar, arsta->arvif->vdev_id, sta->addr,511tid, initiator, reason);512if (ret) {513ath11k_warn(ar->ab, "failed to send delba: vdev_id %u peer %pM tid %u initiator %u reason %u\n",514arsta->arvif->vdev_id, sta->addr, tid, initiator,515reason);516}517ret = count;518out:519mutex_unlock(&ar->conf_mutex);520return ret;521}522523static const struct file_operations fops_delba = {524.write = ath11k_dbg_sta_write_delba,525.open = simple_open,526.owner = THIS_MODULE,527.llseek = default_llseek,528};529530static ssize_t ath11k_dbg_sta_write_addba_resp(struct file *file,531const char __user *user_buf,532size_t count, loff_t *ppos)533{534struct ieee80211_sta *sta = file->private_data;535struct ath11k_sta *arsta = ath11k_sta_to_arsta(sta);536struct ath11k *ar = arsta->arvif->ar;537u32 tid, status;538int ret;539char buf[64] = {};540541ret = simple_write_to_buffer(buf, sizeof(buf) - 1, ppos,542user_buf, count);543if (ret <= 0)544return ret;545546ret = sscanf(buf, "%u %u", &tid, &status);547if (ret != 2)548return -EINVAL;549550/* Valid TID values are 0 through 15 */551if (tid > HAL_DESC_REO_NON_QOS_TID - 1)552return -EINVAL;553554mutex_lock(&ar->conf_mutex);555if (ar->state != ATH11K_STATE_ON ||556arsta->aggr_mode != ATH11K_DBG_AGGR_MODE_MANUAL) {557ret = count;558goto out;559}560561ret = ath11k_wmi_addba_set_resp(ar, arsta->arvif->vdev_id, sta->addr,562tid, status);563if (ret) {564ath11k_warn(ar->ab, "failed to send addba response: vdev_id %u peer %pM tid %u status%u\n",565arsta->arvif->vdev_id, sta->addr, tid, status);566}567ret = count;568out:569mutex_unlock(&ar->conf_mutex);570return ret;571}572573static const struct file_operations fops_addba_resp = {574.write = ath11k_dbg_sta_write_addba_resp,575.open = simple_open,576.owner = THIS_MODULE,577.llseek = default_llseek,578};579580static ssize_t ath11k_dbg_sta_write_addba(struct file *file,581const char __user *user_buf,582size_t count, loff_t *ppos)583{584struct ieee80211_sta *sta = file->private_data;585struct ath11k_sta *arsta = ath11k_sta_to_arsta(sta);586struct ath11k *ar = arsta->arvif->ar;587u32 tid, buf_size;588int ret;589char buf[64] = {};590591ret = simple_write_to_buffer(buf, sizeof(buf) - 1, ppos,592user_buf, count);593if (ret <= 0)594return ret;595596ret = sscanf(buf, "%u %u", &tid, &buf_size);597if (ret != 2)598return -EINVAL;599600/* Valid TID values are 0 through 15 */601if (tid > HAL_DESC_REO_NON_QOS_TID - 1)602return -EINVAL;603604mutex_lock(&ar->conf_mutex);605if (ar->state != ATH11K_STATE_ON ||606arsta->aggr_mode != ATH11K_DBG_AGGR_MODE_MANUAL) {607ret = count;608goto out;609}610611ret = ath11k_wmi_addba_send(ar, arsta->arvif->vdev_id, sta->addr,612tid, buf_size);613if (ret) {614ath11k_warn(ar->ab, "failed to send addba request: vdev_id %u peer %pM tid %u buf_size %u\n",615arsta->arvif->vdev_id, sta->addr, tid, buf_size);616}617618ret = count;619out:620mutex_unlock(&ar->conf_mutex);621return ret;622}623624static const struct file_operations fops_addba = {625.write = ath11k_dbg_sta_write_addba,626.open = simple_open,627.owner = THIS_MODULE,628.llseek = default_llseek,629};630631static ssize_t ath11k_dbg_sta_read_aggr_mode(struct file *file,632char __user *user_buf,633size_t count, loff_t *ppos)634{635struct ieee80211_sta *sta = file->private_data;636struct ath11k_sta *arsta = ath11k_sta_to_arsta(sta);637struct ath11k *ar = arsta->arvif->ar;638char buf[64];639int len = 0;640641mutex_lock(&ar->conf_mutex);642len = scnprintf(buf, sizeof(buf) - len,643"aggregation mode: %s\n\n%s\n%s\n",644(arsta->aggr_mode == ATH11K_DBG_AGGR_MODE_AUTO) ?645"auto" : "manual", "auto = 0", "manual = 1");646mutex_unlock(&ar->conf_mutex);647648return simple_read_from_buffer(user_buf, count, ppos, buf, len);649}650651static ssize_t ath11k_dbg_sta_write_aggr_mode(struct file *file,652const char __user *user_buf,653size_t count, loff_t *ppos)654{655struct ieee80211_sta *sta = file->private_data;656struct ath11k_sta *arsta = ath11k_sta_to_arsta(sta);657struct ath11k *ar = arsta->arvif->ar;658u32 aggr_mode;659int ret;660661if (kstrtouint_from_user(user_buf, count, 0, &aggr_mode))662return -EINVAL;663664if (aggr_mode >= ATH11K_DBG_AGGR_MODE_MAX)665return -EINVAL;666667mutex_lock(&ar->conf_mutex);668if (ar->state != ATH11K_STATE_ON ||669aggr_mode == arsta->aggr_mode) {670ret = count;671goto out;672}673674ret = ath11k_wmi_addba_clear_resp(ar, arsta->arvif->vdev_id, sta->addr);675if (ret) {676ath11k_warn(ar->ab, "failed to clear addba session ret: %d\n",677ret);678goto out;679}680681arsta->aggr_mode = aggr_mode;682out:683mutex_unlock(&ar->conf_mutex);684return ret;685}686687static const struct file_operations fops_aggr_mode = {688.read = ath11k_dbg_sta_read_aggr_mode,689.write = ath11k_dbg_sta_write_aggr_mode,690.open = simple_open,691.owner = THIS_MODULE,692.llseek = default_llseek,693};694695static ssize_t696ath11k_write_htt_peer_stats_reset(struct file *file,697const char __user *user_buf,698size_t count, loff_t *ppos)699{700struct ieee80211_sta *sta = file->private_data;701struct ath11k_sta *arsta = ath11k_sta_to_arsta(sta);702struct ath11k *ar = arsta->arvif->ar;703struct htt_ext_stats_cfg_params cfg_params = {};704int ret;705u8 type;706707ret = kstrtou8_from_user(user_buf, count, 0, &type);708if (ret)709return ret;710711if (!type)712return ret;713714mutex_lock(&ar->conf_mutex);715cfg_params.cfg0 = HTT_STAT_PEER_INFO_MAC_ADDR;716cfg_params.cfg0 |= FIELD_PREP(GENMASK(15, 1),717HTT_PEER_STATS_REQ_MODE_FLUSH_TQM);718719cfg_params.cfg1 = HTT_STAT_DEFAULT_PEER_REQ_TYPE;720721cfg_params.cfg2 |= FIELD_PREP(GENMASK(7, 0), sta->addr[0]);722cfg_params.cfg2 |= FIELD_PREP(GENMASK(15, 8), sta->addr[1]);723cfg_params.cfg2 |= FIELD_PREP(GENMASK(23, 16), sta->addr[2]);724cfg_params.cfg2 |= FIELD_PREP(GENMASK(31, 24), sta->addr[3]);725726cfg_params.cfg3 |= FIELD_PREP(GENMASK(7, 0), sta->addr[4]);727cfg_params.cfg3 |= FIELD_PREP(GENMASK(15, 8), sta->addr[5]);728729cfg_params.cfg3 |= ATH11K_HTT_PEER_STATS_RESET;730731ret = ath11k_dp_tx_htt_h2t_ext_stats_req(ar,732ATH11K_DBG_HTT_EXT_STATS_PEER_INFO,733&cfg_params,7340ULL);735if (ret) {736ath11k_warn(ar->ab, "failed to send htt peer stats request: %d\n", ret);737mutex_unlock(&ar->conf_mutex);738return ret;739}740741mutex_unlock(&ar->conf_mutex);742743ret = count;744745return ret;746}747748static const struct file_operations fops_htt_peer_stats_reset = {749.write = ath11k_write_htt_peer_stats_reset,750.open = simple_open,751.owner = THIS_MODULE,752.llseek = default_llseek,753};754755static ssize_t ath11k_dbg_sta_read_peer_ps_state(struct file *file,756char __user *user_buf,757size_t count, loff_t *ppos)758{759struct ieee80211_sta *sta = file->private_data;760struct ath11k_sta *arsta = ath11k_sta_to_arsta(sta);761struct ath11k *ar = arsta->arvif->ar;762char buf[20];763int len;764765spin_lock_bh(&ar->data_lock);766767len = scnprintf(buf, sizeof(buf), "%d\n", arsta->peer_ps_state);768769spin_unlock_bh(&ar->data_lock);770771return simple_read_from_buffer(user_buf, count, ppos, buf, len);772}773774static const struct file_operations fops_peer_ps_state = {775.open = simple_open,776.read = ath11k_dbg_sta_read_peer_ps_state,777.owner = THIS_MODULE,778.llseek = default_llseek,779};780781static ssize_t ath11k_dbg_sta_read_current_ps_duration(struct file *file,782char __user *user_buf,783size_t count,784loff_t *ppos)785{786struct ieee80211_sta *sta = file->private_data;787struct ath11k_sta *arsta = ath11k_sta_to_arsta(sta);788struct ath11k *ar = arsta->arvif->ar;789u64 time_since_station_in_power_save;790char buf[20];791int len;792793spin_lock_bh(&ar->data_lock);794795if (arsta->peer_ps_state == WMI_PEER_PS_STATE_ON &&796arsta->peer_current_ps_valid)797time_since_station_in_power_save = jiffies_to_msecs(jiffies798- arsta->ps_start_jiffies);799else800time_since_station_in_power_save = 0;801802len = scnprintf(buf, sizeof(buf), "%llu\n",803time_since_station_in_power_save);804spin_unlock_bh(&ar->data_lock);805806return simple_read_from_buffer(user_buf, count, ppos, buf, len);807}808809static const struct file_operations fops_current_ps_duration = {810.open = simple_open,811.read = ath11k_dbg_sta_read_current_ps_duration,812.owner = THIS_MODULE,813.llseek = default_llseek,814};815816static ssize_t ath11k_dbg_sta_read_total_ps_duration(struct file *file,817char __user *user_buf,818size_t count, loff_t *ppos)819{820struct ieee80211_sta *sta = file->private_data;821struct ath11k_sta *arsta = ath11k_sta_to_arsta(sta);822struct ath11k *ar = arsta->arvif->ar;823char buf[20];824u64 power_save_duration;825int len;826827spin_lock_bh(&ar->data_lock);828829if (arsta->peer_ps_state == WMI_PEER_PS_STATE_ON &&830arsta->peer_current_ps_valid)831power_save_duration = jiffies_to_msecs(jiffies832- arsta->ps_start_jiffies)833+ arsta->ps_total_duration;834else835power_save_duration = arsta->ps_total_duration;836837len = scnprintf(buf, sizeof(buf), "%llu\n", power_save_duration);838839spin_unlock_bh(&ar->data_lock);840841return simple_read_from_buffer(user_buf, count, ppos, buf, len);842}843844static const struct file_operations fops_total_ps_duration = {845.open = simple_open,846.read = ath11k_dbg_sta_read_total_ps_duration,847.owner = THIS_MODULE,848.llseek = default_llseek,849};850851void ath11k_debugfs_sta_op_add(struct ieee80211_hw *hw, struct ieee80211_vif *vif,852struct ieee80211_sta *sta, struct dentry *dir)853{854struct ath11k *ar = hw->priv;855856if (ath11k_debugfs_is_extd_tx_stats_enabled(ar))857debugfs_create_file("tx_stats", 0400, dir, sta,858&fops_tx_stats);859if (ath11k_debugfs_is_extd_rx_stats_enabled(ar))860debugfs_create_file("rx_stats", 0400, dir, sta,861&fops_rx_stats);862863debugfs_create_file("htt_peer_stats", 0400, dir, sta,864&fops_htt_peer_stats);865866debugfs_create_file("peer_pktlog", 0644, dir, sta,867&fops_peer_pktlog);868869debugfs_create_file("aggr_mode", 0644, dir, sta, &fops_aggr_mode);870debugfs_create_file("addba", 0200, dir, sta, &fops_addba);871debugfs_create_file("addba_resp", 0200, dir, sta, &fops_addba_resp);872debugfs_create_file("delba", 0200, dir, sta, &fops_delba);873874if (test_bit(WMI_TLV_SERVICE_PER_PEER_HTT_STATS_RESET,875ar->ab->wmi_ab.svc_map))876debugfs_create_file("htt_peer_stats_reset", 0600, dir, sta,877&fops_htt_peer_stats_reset);878879debugfs_create_file("peer_ps_state", 0400, dir, sta,880&fops_peer_ps_state);881882if (test_bit(WMI_TLV_SERVICE_PEER_POWER_SAVE_DURATION_SUPPORT,883ar->ab->wmi_ab.svc_map)) {884debugfs_create_file("current_ps_duration", 0440, dir, sta,885&fops_current_ps_duration);886debugfs_create_file("total_ps_duration", 0440, dir, sta,887&fops_total_ps_duration);888}889}890891892