Commit 723e7113 authored by Felix Fietkau's avatar Felix Fietkau Committed by John W. Linville

ath9k: fix handling of broken descriptors

As the comment in ath_get_next_rx_buf indicates, if a descriptor with
the done bit set follows one with the done bit cleared, both descriptors
should be discarded, however the driver is not doing that yet.

To fix this, use the rs->rs_more flag as an indicator that the following
frame should be discarded. This also helps with the split buffer case:
if the first part of the frame is discarded, the following parts need to
be discarded as well, since they contain no valid header or usable data.
Signed-off-by: default avatarFelix Fietkau <nbd@openwrt.org>
Signed-off-by: default avatarJohn W. Linville <linville@tuxdriver.com>
parent 2e1cd495
...@@ -311,6 +311,7 @@ struct ath_rx_edma { ...@@ -311,6 +311,7 @@ struct ath_rx_edma {
struct ath_rx { struct ath_rx {
u8 defant; u8 defant;
u8 rxotherant; u8 rxotherant;
bool discard_next;
u32 *rxlink; u32 *rxlink;
u32 num_pkts; u32 num_pkts;
unsigned int rxfilter; unsigned int rxfilter;
......
...@@ -727,6 +727,13 @@ static struct ath_buf *ath_get_next_rx_buf(struct ath_softc *sc, ...@@ -727,6 +727,13 @@ static struct ath_buf *ath_get_next_rx_buf(struct ath_softc *sc,
ret = ath9k_hw_rxprocdesc(ah, tds, &trs); ret = ath9k_hw_rxprocdesc(ah, tds, &trs);
if (ret == -EINPROGRESS) if (ret == -EINPROGRESS)
return NULL; return NULL;
/*
* mark descriptor as zero-length and set the 'more'
* flag to ensure that both buffers get discarded
*/
rs->rs_datalen = 0;
rs->rs_more = true;
} }
list_del(&bf->list); list_del(&bf->list);
...@@ -933,14 +940,20 @@ static void ath9k_process_rssi(struct ath_common *common, ...@@ -933,14 +940,20 @@ static void ath9k_process_rssi(struct ath_common *common,
* up the frame up to let mac80211 handle the actual error case, be it no * up the frame up to let mac80211 handle the actual error case, be it no
* decryption key or real decryption error. This let us keep statistics there. * decryption key or real decryption error. This let us keep statistics there.
*/ */
static int ath9k_rx_skb_preprocess(struct ath_common *common, static int ath9k_rx_skb_preprocess(struct ath_softc *sc,
struct ieee80211_hw *hw,
struct ieee80211_hdr *hdr, struct ieee80211_hdr *hdr,
struct ath_rx_status *rx_stats, struct ath_rx_status *rx_stats,
struct ieee80211_rx_status *rx_status, struct ieee80211_rx_status *rx_status,
bool *decrypt_error) bool *decrypt_error)
{ {
struct ath_hw *ah = common->ah; struct ieee80211_hw *hw = sc->hw;
struct ath_hw *ah = sc->sc_ah;
struct ath_common *common = ath9k_hw_common(ah);
bool discard_current = sc->rx.discard_next;
sc->rx.discard_next = rx_stats->rs_more;
if (discard_current)
return -EINVAL;
/* /*
* everything but the rate is checked here, the rate check is done * everything but the rate is checked here, the rate check is done
...@@ -966,6 +979,7 @@ static int ath9k_rx_skb_preprocess(struct ath_common *common, ...@@ -966,6 +979,7 @@ static int ath9k_rx_skb_preprocess(struct ath_common *common,
if (rx_stats->rs_moreaggr) if (rx_stats->rs_moreaggr)
rx_status->flag |= RX_FLAG_NO_SIGNAL_VAL; rx_status->flag |= RX_FLAG_NO_SIGNAL_VAL;
sc->rx.discard_next = false;
return 0; return 0;
} }
...@@ -1243,8 +1257,8 @@ int ath_rx_tasklet(struct ath_softc *sc, int flush, bool hp) ...@@ -1243,8 +1257,8 @@ int ath_rx_tasklet(struct ath_softc *sc, int flush, bool hp)
} }
} }
retval = ath9k_rx_skb_preprocess(common, hw, hdr, &rs, retval = ath9k_rx_skb_preprocess(sc, hdr, &rs, rxs,
rxs, &decrypt_error); &decrypt_error);
if (retval) if (retval)
goto requeue_drop_frag; goto requeue_drop_frag;
......
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