162306a36Sopenharmony_ci// SPDX-License-Identifier: ISC 262306a36Sopenharmony_ci/* 362306a36Sopenharmony_ci * Copyright (c) 2005-2011 Atheros Communications Inc. 462306a36Sopenharmony_ci * Copyright (c) 2011-2016 Qualcomm Atheros, Inc. 562306a36Sopenharmony_ci * Copyright (c) 2018, The Linux Foundation. All rights reserved. 662306a36Sopenharmony_ci */ 762306a36Sopenharmony_ci 862306a36Sopenharmony_ci#include "core.h" 962306a36Sopenharmony_ci#include "txrx.h" 1062306a36Sopenharmony_ci#include "htt.h" 1162306a36Sopenharmony_ci#include "mac.h" 1262306a36Sopenharmony_ci#include "debug.h" 1362306a36Sopenharmony_ci 1462306a36Sopenharmony_cistatic void ath10k_report_offchan_tx(struct ath10k *ar, struct sk_buff *skb) 1562306a36Sopenharmony_ci{ 1662306a36Sopenharmony_ci struct ieee80211_tx_info *info = IEEE80211_SKB_CB(skb); 1762306a36Sopenharmony_ci 1862306a36Sopenharmony_ci if (likely(!(info->flags & IEEE80211_TX_CTL_TX_OFFCHAN))) 1962306a36Sopenharmony_ci return; 2062306a36Sopenharmony_ci 2162306a36Sopenharmony_ci if (ath10k_mac_tx_frm_has_freq(ar)) 2262306a36Sopenharmony_ci return; 2362306a36Sopenharmony_ci 2462306a36Sopenharmony_ci /* If the original wait_for_completion() timed out before 2562306a36Sopenharmony_ci * {data,mgmt}_tx_completed() was called then we could complete 2662306a36Sopenharmony_ci * offchan_tx_completed for a different skb. Prevent this by using 2762306a36Sopenharmony_ci * offchan_tx_skb. 2862306a36Sopenharmony_ci */ 2962306a36Sopenharmony_ci spin_lock_bh(&ar->data_lock); 3062306a36Sopenharmony_ci if (ar->offchan_tx_skb != skb) { 3162306a36Sopenharmony_ci ath10k_warn(ar, "completed old offchannel frame\n"); 3262306a36Sopenharmony_ci goto out; 3362306a36Sopenharmony_ci } 3462306a36Sopenharmony_ci 3562306a36Sopenharmony_ci complete(&ar->offchan_tx_completed); 3662306a36Sopenharmony_ci ar->offchan_tx_skb = NULL; /* just for sanity */ 3762306a36Sopenharmony_ci 3862306a36Sopenharmony_ci ath10k_dbg(ar, ATH10K_DBG_HTT, "completed offchannel skb %pK\n", skb); 3962306a36Sopenharmony_ciout: 4062306a36Sopenharmony_ci spin_unlock_bh(&ar->data_lock); 4162306a36Sopenharmony_ci} 4262306a36Sopenharmony_ci 4362306a36Sopenharmony_ciint ath10k_txrx_tx_unref(struct ath10k_htt *htt, 4462306a36Sopenharmony_ci const struct htt_tx_done *tx_done) 4562306a36Sopenharmony_ci{ 4662306a36Sopenharmony_ci struct ieee80211_tx_status status; 4762306a36Sopenharmony_ci struct ath10k *ar = htt->ar; 4862306a36Sopenharmony_ci struct device *dev = ar->dev; 4962306a36Sopenharmony_ci struct ieee80211_tx_info *info; 5062306a36Sopenharmony_ci struct ieee80211_txq *txq; 5162306a36Sopenharmony_ci struct ath10k_skb_cb *skb_cb; 5262306a36Sopenharmony_ci struct ath10k_txq *artxq; 5362306a36Sopenharmony_ci struct sk_buff *msdu; 5462306a36Sopenharmony_ci u8 flags; 5562306a36Sopenharmony_ci 5662306a36Sopenharmony_ci ath10k_dbg(ar, ATH10K_DBG_HTT, 5762306a36Sopenharmony_ci "htt tx completion msdu_id %u status %d\n", 5862306a36Sopenharmony_ci tx_done->msdu_id, tx_done->status); 5962306a36Sopenharmony_ci 6062306a36Sopenharmony_ci if (tx_done->msdu_id >= htt->max_num_pending_tx) { 6162306a36Sopenharmony_ci ath10k_warn(ar, "warning: msdu_id %d too big, ignoring\n", 6262306a36Sopenharmony_ci tx_done->msdu_id); 6362306a36Sopenharmony_ci return -EINVAL; 6462306a36Sopenharmony_ci } 6562306a36Sopenharmony_ci 6662306a36Sopenharmony_ci spin_lock_bh(&htt->tx_lock); 6762306a36Sopenharmony_ci msdu = idr_find(&htt->pending_tx, tx_done->msdu_id); 6862306a36Sopenharmony_ci if (!msdu) { 6962306a36Sopenharmony_ci ath10k_warn(ar, "received tx completion for invalid msdu_id: %d\n", 7062306a36Sopenharmony_ci tx_done->msdu_id); 7162306a36Sopenharmony_ci spin_unlock_bh(&htt->tx_lock); 7262306a36Sopenharmony_ci return -ENOENT; 7362306a36Sopenharmony_ci } 7462306a36Sopenharmony_ci 7562306a36Sopenharmony_ci skb_cb = ATH10K_SKB_CB(msdu); 7662306a36Sopenharmony_ci txq = skb_cb->txq; 7762306a36Sopenharmony_ci 7862306a36Sopenharmony_ci if (txq) { 7962306a36Sopenharmony_ci artxq = (void *)txq->drv_priv; 8062306a36Sopenharmony_ci artxq->num_fw_queued--; 8162306a36Sopenharmony_ci } 8262306a36Sopenharmony_ci 8362306a36Sopenharmony_ci flags = skb_cb->flags; 8462306a36Sopenharmony_ci ath10k_htt_tx_free_msdu_id(htt, tx_done->msdu_id); 8562306a36Sopenharmony_ci ath10k_htt_tx_dec_pending(htt); 8662306a36Sopenharmony_ci spin_unlock_bh(&htt->tx_lock); 8762306a36Sopenharmony_ci 8862306a36Sopenharmony_ci rcu_read_lock(); 8962306a36Sopenharmony_ci if (txq && txq->sta && skb_cb->airtime_est) 9062306a36Sopenharmony_ci ieee80211_sta_register_airtime(txq->sta, txq->tid, 9162306a36Sopenharmony_ci skb_cb->airtime_est, 0); 9262306a36Sopenharmony_ci rcu_read_unlock(); 9362306a36Sopenharmony_ci 9462306a36Sopenharmony_ci if (ar->bus_param.dev_type != ATH10K_DEV_TYPE_HL) 9562306a36Sopenharmony_ci dma_unmap_single(dev, skb_cb->paddr, msdu->len, DMA_TO_DEVICE); 9662306a36Sopenharmony_ci 9762306a36Sopenharmony_ci ath10k_report_offchan_tx(htt->ar, msdu); 9862306a36Sopenharmony_ci 9962306a36Sopenharmony_ci info = IEEE80211_SKB_CB(msdu); 10062306a36Sopenharmony_ci memset(&info->status, 0, sizeof(info->status)); 10162306a36Sopenharmony_ci info->status.rates[0].idx = -1; 10262306a36Sopenharmony_ci 10362306a36Sopenharmony_ci trace_ath10k_txrx_tx_unref(ar, tx_done->msdu_id); 10462306a36Sopenharmony_ci 10562306a36Sopenharmony_ci if (!(info->flags & IEEE80211_TX_CTL_NO_ACK) && 10662306a36Sopenharmony_ci !(flags & ATH10K_SKB_F_NOACK_TID)) 10762306a36Sopenharmony_ci info->flags |= IEEE80211_TX_STAT_ACK; 10862306a36Sopenharmony_ci 10962306a36Sopenharmony_ci if (tx_done->status == HTT_TX_COMPL_STATE_NOACK) 11062306a36Sopenharmony_ci info->flags &= ~IEEE80211_TX_STAT_ACK; 11162306a36Sopenharmony_ci 11262306a36Sopenharmony_ci if ((tx_done->status == HTT_TX_COMPL_STATE_ACK) && 11362306a36Sopenharmony_ci ((info->flags & IEEE80211_TX_CTL_NO_ACK) || 11462306a36Sopenharmony_ci (flags & ATH10K_SKB_F_NOACK_TID))) 11562306a36Sopenharmony_ci info->flags |= IEEE80211_TX_STAT_NOACK_TRANSMITTED; 11662306a36Sopenharmony_ci 11762306a36Sopenharmony_ci if (tx_done->status == HTT_TX_COMPL_STATE_DISCARD) { 11862306a36Sopenharmony_ci if ((info->flags & IEEE80211_TX_CTL_NO_ACK) || 11962306a36Sopenharmony_ci (flags & ATH10K_SKB_F_NOACK_TID)) 12062306a36Sopenharmony_ci info->flags &= ~IEEE80211_TX_STAT_NOACK_TRANSMITTED; 12162306a36Sopenharmony_ci else 12262306a36Sopenharmony_ci info->flags &= ~IEEE80211_TX_STAT_ACK; 12362306a36Sopenharmony_ci } 12462306a36Sopenharmony_ci 12562306a36Sopenharmony_ci if (tx_done->status == HTT_TX_COMPL_STATE_ACK && 12662306a36Sopenharmony_ci tx_done->ack_rssi != ATH10K_INVALID_RSSI) { 12762306a36Sopenharmony_ci info->status.ack_signal = ATH10K_DEFAULT_NOISE_FLOOR + 12862306a36Sopenharmony_ci tx_done->ack_rssi; 12962306a36Sopenharmony_ci info->status.flags |= IEEE80211_TX_STATUS_ACK_SIGNAL_VALID; 13062306a36Sopenharmony_ci } 13162306a36Sopenharmony_ci 13262306a36Sopenharmony_ci memset(&status, 0, sizeof(status)); 13362306a36Sopenharmony_ci status.skb = msdu; 13462306a36Sopenharmony_ci status.info = info; 13562306a36Sopenharmony_ci 13662306a36Sopenharmony_ci rcu_read_lock(); 13762306a36Sopenharmony_ci 13862306a36Sopenharmony_ci if (txq) 13962306a36Sopenharmony_ci status.sta = txq->sta; 14062306a36Sopenharmony_ci 14162306a36Sopenharmony_ci ieee80211_tx_status_ext(htt->ar->hw, &status); 14262306a36Sopenharmony_ci 14362306a36Sopenharmony_ci rcu_read_unlock(); 14462306a36Sopenharmony_ci 14562306a36Sopenharmony_ci /* we do not own the msdu anymore */ 14662306a36Sopenharmony_ci 14762306a36Sopenharmony_ci return 0; 14862306a36Sopenharmony_ci} 14962306a36Sopenharmony_ci 15062306a36Sopenharmony_cistruct ath10k_peer *ath10k_peer_find(struct ath10k *ar, int vdev_id, 15162306a36Sopenharmony_ci const u8 *addr) 15262306a36Sopenharmony_ci{ 15362306a36Sopenharmony_ci struct ath10k_peer *peer; 15462306a36Sopenharmony_ci 15562306a36Sopenharmony_ci lockdep_assert_held(&ar->data_lock); 15662306a36Sopenharmony_ci 15762306a36Sopenharmony_ci list_for_each_entry(peer, &ar->peers, list) { 15862306a36Sopenharmony_ci if (peer->vdev_id != vdev_id) 15962306a36Sopenharmony_ci continue; 16062306a36Sopenharmony_ci if (!ether_addr_equal(peer->addr, addr)) 16162306a36Sopenharmony_ci continue; 16262306a36Sopenharmony_ci 16362306a36Sopenharmony_ci return peer; 16462306a36Sopenharmony_ci } 16562306a36Sopenharmony_ci 16662306a36Sopenharmony_ci return NULL; 16762306a36Sopenharmony_ci} 16862306a36Sopenharmony_ci 16962306a36Sopenharmony_cistruct ath10k_peer *ath10k_peer_find_by_id(struct ath10k *ar, int peer_id) 17062306a36Sopenharmony_ci{ 17162306a36Sopenharmony_ci struct ath10k_peer *peer; 17262306a36Sopenharmony_ci 17362306a36Sopenharmony_ci if (peer_id >= BITS_PER_TYPE(peer->peer_ids)) 17462306a36Sopenharmony_ci return NULL; 17562306a36Sopenharmony_ci 17662306a36Sopenharmony_ci lockdep_assert_held(&ar->data_lock); 17762306a36Sopenharmony_ci 17862306a36Sopenharmony_ci list_for_each_entry(peer, &ar->peers, list) 17962306a36Sopenharmony_ci if (test_bit(peer_id, peer->peer_ids)) 18062306a36Sopenharmony_ci return peer; 18162306a36Sopenharmony_ci 18262306a36Sopenharmony_ci return NULL; 18362306a36Sopenharmony_ci} 18462306a36Sopenharmony_ci 18562306a36Sopenharmony_cistatic int ath10k_wait_for_peer_common(struct ath10k *ar, int vdev_id, 18662306a36Sopenharmony_ci const u8 *addr, bool expect_mapped) 18762306a36Sopenharmony_ci{ 18862306a36Sopenharmony_ci long time_left; 18962306a36Sopenharmony_ci 19062306a36Sopenharmony_ci time_left = wait_event_timeout(ar->peer_mapping_wq, ({ 19162306a36Sopenharmony_ci bool mapped; 19262306a36Sopenharmony_ci 19362306a36Sopenharmony_ci spin_lock_bh(&ar->data_lock); 19462306a36Sopenharmony_ci mapped = !!ath10k_peer_find(ar, vdev_id, addr); 19562306a36Sopenharmony_ci spin_unlock_bh(&ar->data_lock); 19662306a36Sopenharmony_ci 19762306a36Sopenharmony_ci (mapped == expect_mapped || 19862306a36Sopenharmony_ci test_bit(ATH10K_FLAG_CRASH_FLUSH, &ar->dev_flags)); 19962306a36Sopenharmony_ci }), 3 * HZ); 20062306a36Sopenharmony_ci 20162306a36Sopenharmony_ci if (time_left == 0) 20262306a36Sopenharmony_ci return -ETIMEDOUT; 20362306a36Sopenharmony_ci 20462306a36Sopenharmony_ci return 0; 20562306a36Sopenharmony_ci} 20662306a36Sopenharmony_ci 20762306a36Sopenharmony_ciint ath10k_wait_for_peer_created(struct ath10k *ar, int vdev_id, const u8 *addr) 20862306a36Sopenharmony_ci{ 20962306a36Sopenharmony_ci return ath10k_wait_for_peer_common(ar, vdev_id, addr, true); 21062306a36Sopenharmony_ci} 21162306a36Sopenharmony_ci 21262306a36Sopenharmony_ciint ath10k_wait_for_peer_deleted(struct ath10k *ar, int vdev_id, const u8 *addr) 21362306a36Sopenharmony_ci{ 21462306a36Sopenharmony_ci return ath10k_wait_for_peer_common(ar, vdev_id, addr, false); 21562306a36Sopenharmony_ci} 21662306a36Sopenharmony_ci 21762306a36Sopenharmony_civoid ath10k_peer_map_event(struct ath10k_htt *htt, 21862306a36Sopenharmony_ci struct htt_peer_map_event *ev) 21962306a36Sopenharmony_ci{ 22062306a36Sopenharmony_ci struct ath10k *ar = htt->ar; 22162306a36Sopenharmony_ci struct ath10k_peer *peer; 22262306a36Sopenharmony_ci 22362306a36Sopenharmony_ci if (ev->peer_id >= ATH10K_MAX_NUM_PEER_IDS) { 22462306a36Sopenharmony_ci ath10k_warn(ar, 22562306a36Sopenharmony_ci "received htt peer map event with idx out of bounds: %u\n", 22662306a36Sopenharmony_ci ev->peer_id); 22762306a36Sopenharmony_ci return; 22862306a36Sopenharmony_ci } 22962306a36Sopenharmony_ci 23062306a36Sopenharmony_ci spin_lock_bh(&ar->data_lock); 23162306a36Sopenharmony_ci peer = ath10k_peer_find(ar, ev->vdev_id, ev->addr); 23262306a36Sopenharmony_ci if (!peer) { 23362306a36Sopenharmony_ci peer = kzalloc(sizeof(*peer), GFP_ATOMIC); 23462306a36Sopenharmony_ci if (!peer) 23562306a36Sopenharmony_ci goto exit; 23662306a36Sopenharmony_ci 23762306a36Sopenharmony_ci peer->vdev_id = ev->vdev_id; 23862306a36Sopenharmony_ci ether_addr_copy(peer->addr, ev->addr); 23962306a36Sopenharmony_ci list_add(&peer->list, &ar->peers); 24062306a36Sopenharmony_ci wake_up(&ar->peer_mapping_wq); 24162306a36Sopenharmony_ci } 24262306a36Sopenharmony_ci 24362306a36Sopenharmony_ci ath10k_dbg(ar, ATH10K_DBG_HTT, "htt peer map vdev %d peer %pM id %d\n", 24462306a36Sopenharmony_ci ev->vdev_id, ev->addr, ev->peer_id); 24562306a36Sopenharmony_ci 24662306a36Sopenharmony_ci WARN_ON(ar->peer_map[ev->peer_id] && (ar->peer_map[ev->peer_id] != peer)); 24762306a36Sopenharmony_ci ar->peer_map[ev->peer_id] = peer; 24862306a36Sopenharmony_ci set_bit(ev->peer_id, peer->peer_ids); 24962306a36Sopenharmony_ciexit: 25062306a36Sopenharmony_ci spin_unlock_bh(&ar->data_lock); 25162306a36Sopenharmony_ci} 25262306a36Sopenharmony_ci 25362306a36Sopenharmony_civoid ath10k_peer_unmap_event(struct ath10k_htt *htt, 25462306a36Sopenharmony_ci struct htt_peer_unmap_event *ev) 25562306a36Sopenharmony_ci{ 25662306a36Sopenharmony_ci struct ath10k *ar = htt->ar; 25762306a36Sopenharmony_ci struct ath10k_peer *peer; 25862306a36Sopenharmony_ci 25962306a36Sopenharmony_ci if (ev->peer_id >= ATH10K_MAX_NUM_PEER_IDS) { 26062306a36Sopenharmony_ci ath10k_warn(ar, 26162306a36Sopenharmony_ci "received htt peer unmap event with idx out of bounds: %u\n", 26262306a36Sopenharmony_ci ev->peer_id); 26362306a36Sopenharmony_ci return; 26462306a36Sopenharmony_ci } 26562306a36Sopenharmony_ci 26662306a36Sopenharmony_ci spin_lock_bh(&ar->data_lock); 26762306a36Sopenharmony_ci peer = ath10k_peer_find_by_id(ar, ev->peer_id); 26862306a36Sopenharmony_ci if (!peer) { 26962306a36Sopenharmony_ci ath10k_warn(ar, "peer-unmap-event: unknown peer id %d\n", 27062306a36Sopenharmony_ci ev->peer_id); 27162306a36Sopenharmony_ci goto exit; 27262306a36Sopenharmony_ci } 27362306a36Sopenharmony_ci 27462306a36Sopenharmony_ci ath10k_dbg(ar, ATH10K_DBG_HTT, "htt peer unmap vdev %d peer %pM id %d\n", 27562306a36Sopenharmony_ci peer->vdev_id, peer->addr, ev->peer_id); 27662306a36Sopenharmony_ci 27762306a36Sopenharmony_ci ar->peer_map[ev->peer_id] = NULL; 27862306a36Sopenharmony_ci clear_bit(ev->peer_id, peer->peer_ids); 27962306a36Sopenharmony_ci 28062306a36Sopenharmony_ci if (bitmap_empty(peer->peer_ids, ATH10K_MAX_NUM_PEER_IDS)) { 28162306a36Sopenharmony_ci list_del(&peer->list); 28262306a36Sopenharmony_ci kfree(peer); 28362306a36Sopenharmony_ci wake_up(&ar->peer_mapping_wq); 28462306a36Sopenharmony_ci } 28562306a36Sopenharmony_ci 28662306a36Sopenharmony_ciexit: 28762306a36Sopenharmony_ci spin_unlock_bh(&ar->data_lock); 28862306a36Sopenharmony_ci} 289