Commit 400ece6c authored by Sven Eckelmann's avatar Sven Eckelmann Committed by Kalle Valo

wifi: ath11k: Don't drop tx_status when peer cannot be found

When a station idles for a long time, hostapd will try to send a QoS Null
frame to the station as "poll". NL80211_CMD_PROBE_CLIENT is used for this
purpose. And the skb will be added to ack_status_frame - waiting for a
completion via ieee80211_report_ack_skb().

But when the peer was already removed before the tx_complete arrives, the
peer will be missing. And when using dev_kfree_skb_any (instead of going
through mac80211), the entry will stay inside ack_status_frames. This IDR
will therefore run full after 8K request were generated for such clients.
At this point, the access point will then just stall and not allow any new
clients because idr_alloc() for ack_status_frame will fail.

ieee80211_free_txskb() on the other hand will (when required) call
ieee80211_report_ack_skb() and make sure that (when required) remove the
entry from the ack_status_frame.

Tested-on: IPQ6018 hw1.0 WLAN.HK.2.5.0.1-01100-QCAHKSWPL_SILICONZ-1

Fixes: 6257c702 ("wifi: ath11k: fix tx status reporting in encap offload mode")
Fixes: 94739d45 ("ath11k: switch to using ieee80211_tx_status_ext()")
Cc: stable@vger.kernel.org
Signed-off-by: default avatarSven Eckelmann <sven@narfation.org>
Signed-off-by: default avatarKalle Valo <quic_kvalo@quicinc.com>
Link: https://lore.kernel.org/r/20230802-ath11k-ack_status_leak-v2-1-c0af729d6229@narfation.org
parent 383e1b6a
...@@ -369,7 +369,7 @@ ath11k_dp_tx_htt_tx_complete_buf(struct ath11k_base *ab, ...@@ -369,7 +369,7 @@ ath11k_dp_tx_htt_tx_complete_buf(struct ath11k_base *ab,
"dp_tx: failed to find the peer with peer_id %d\n", "dp_tx: failed to find the peer with peer_id %d\n",
ts->peer_id); ts->peer_id);
spin_unlock_bh(&ab->base_lock); spin_unlock_bh(&ab->base_lock);
dev_kfree_skb_any(msdu); ieee80211_free_txskb(ar->hw, msdu);
return; return;
} }
spin_unlock_bh(&ab->base_lock); spin_unlock_bh(&ab->base_lock);
...@@ -624,7 +624,7 @@ static void ath11k_dp_tx_complete_msdu(struct ath11k *ar, ...@@ -624,7 +624,7 @@ static void ath11k_dp_tx_complete_msdu(struct ath11k *ar,
"dp_tx: failed to find the peer with peer_id %d\n", "dp_tx: failed to find the peer with peer_id %d\n",
ts->peer_id); ts->peer_id);
spin_unlock_bh(&ab->base_lock); spin_unlock_bh(&ab->base_lock);
dev_kfree_skb_any(msdu); ieee80211_free_txskb(ar->hw, msdu);
return; return;
} }
arsta = (struct ath11k_sta *)peer->sta->drv_priv; arsta = (struct ath11k_sta *)peer->sta->drv_priv;
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment