Commit de724a52 authored by Felix Fietkau's avatar Felix Fietkau Committed by Jiri Slaby

ath9k: fix invalid descriptor discarding

commit b7b146c9 upstream.

Only set sc->rx.discard_next to rx_stats->rs_more when actually
discarding the current descriptor.

Also, fix a detection of broken descriptors:
First the code checks if the current descriptor is not done.
Then it checks if the next descriptor is done.
Add a check that afterwards checks the first descriptor again, because
it might have been completed in the mean time.

This fixes a regression introduced in
commit 723e7113
"ath9k: fix handling of broken descriptors"
Reported-by: default avatarMarco André Dinis <marcoandredinis@gmail.com>
Signed-off-by: default avatarFelix Fietkau <nbd@openwrt.org>
Signed-off-by: default avatarJohn W. Linville <linville@tuxdriver.com>
Signed-off-by: default avatarJiri Slaby <jslaby@suse.cz>
parent 7f60220d
...@@ -729,6 +729,12 @@ static struct ath_buf *ath_get_next_rx_buf(struct ath_softc *sc, ...@@ -729,6 +729,12 @@ static struct ath_buf *ath_get_next_rx_buf(struct ath_softc *sc,
if (ret == -EINPROGRESS) if (ret == -EINPROGRESS)
return NULL; return NULL;
/*
* Re-check previous descriptor, in case it has been filled
* in the mean time.
*/
ret = ath9k_hw_rxprocdesc(ah, ds, rs);
if (ret == -EINPROGRESS) {
/* /*
* mark descriptor as zero-length and set the 'more' * mark descriptor as zero-length and set the 'more'
* flag to ensure that both buffers get discarded * flag to ensure that both buffers get discarded
...@@ -736,6 +742,7 @@ static struct ath_buf *ath_get_next_rx_buf(struct ath_softc *sc, ...@@ -736,6 +742,7 @@ static struct ath_buf *ath_get_next_rx_buf(struct ath_softc *sc,
rs->rs_datalen = 0; rs->rs_datalen = 0;
rs->rs_more = true; rs->rs_more = true;
} }
}
list_del(&bf->list); list_del(&bf->list);
if (!bf->bf_mpdu) if (!bf->bf_mpdu)
...@@ -1093,22 +1100,22 @@ static int ath9k_rx_skb_preprocess(struct ath_softc *sc, ...@@ -1093,22 +1100,22 @@ static int ath9k_rx_skb_preprocess(struct ath_softc *sc,
struct ath_common *common = ath9k_hw_common(ah); struct ath_common *common = ath9k_hw_common(ah);
struct ieee80211_hdr *hdr; struct ieee80211_hdr *hdr;
bool discard_current = sc->rx.discard_next; bool discard_current = sc->rx.discard_next;
int ret = 0;
/* /*
* Discard corrupt descriptors which are marked in * Discard corrupt descriptors which are marked in
* ath_get_next_rx_buf(). * ath_get_next_rx_buf().
*/ */
sc->rx.discard_next = rx_stats->rs_more;
if (discard_current) if (discard_current)
return -EINVAL; goto corrupt;
sc->rx.discard_next = false;
/* /*
* Discard zero-length packets. * Discard zero-length packets.
*/ */
if (!rx_stats->rs_datalen) { if (!rx_stats->rs_datalen) {
RX_STAT_INC(rx_len_err); RX_STAT_INC(rx_len_err);
return -EINVAL; goto corrupt;
} }
/* /*
...@@ -1118,7 +1125,7 @@ static int ath9k_rx_skb_preprocess(struct ath_softc *sc, ...@@ -1118,7 +1125,7 @@ static int ath9k_rx_skb_preprocess(struct ath_softc *sc,
*/ */
if (rx_stats->rs_datalen > (common->rx_bufsize - ah->caps.rx_status_len)) { if (rx_stats->rs_datalen > (common->rx_bufsize - ah->caps.rx_status_len)) {
RX_STAT_INC(rx_len_err); RX_STAT_INC(rx_len_err);
return -EINVAL; goto corrupt;
} }
/* Only use status info from the last fragment */ /* Only use status info from the last fragment */
...@@ -1132,10 +1139,8 @@ static int ath9k_rx_skb_preprocess(struct ath_softc *sc, ...@@ -1132,10 +1139,8 @@ static int ath9k_rx_skb_preprocess(struct ath_softc *sc,
* This is different from the other corrupt descriptor * This is different from the other corrupt descriptor
* condition handled above. * condition handled above.
*/ */
if (rx_stats->rs_status & ATH9K_RXERR_CORRUPT_DESC) { if (rx_stats->rs_status & ATH9K_RXERR_CORRUPT_DESC)
ret = -EINVAL; goto corrupt;
goto exit;
}
hdr = (struct ieee80211_hdr *) (skb->data + ah->caps.rx_status_len); hdr = (struct ieee80211_hdr *) (skb->data + ah->caps.rx_status_len);
...@@ -1151,18 +1156,15 @@ static int ath9k_rx_skb_preprocess(struct ath_softc *sc, ...@@ -1151,18 +1156,15 @@ static int ath9k_rx_skb_preprocess(struct ath_softc *sc,
if (ath_process_fft(sc, hdr, rx_stats, rx_status->mactime)) if (ath_process_fft(sc, hdr, rx_stats, rx_status->mactime))
RX_STAT_INC(rx_spectral); RX_STAT_INC(rx_spectral);
ret = -EINVAL; return -EINVAL;
goto exit;
} }
/* /*
* everything but the rate is checked here, the rate check is done * everything but the rate is checked here, the rate check is done
* separately to avoid doing two lookups for a rate for each frame. * separately to avoid doing two lookups for a rate for each frame.
*/ */
if (!ath9k_rx_accept(common, hdr, rx_status, rx_stats, decrypt_error)) { if (!ath9k_rx_accept(common, hdr, rx_status, rx_stats, decrypt_error))
ret = -EINVAL; return -EINVAL;
goto exit;
}
rx_stats->is_mybeacon = ath9k_is_mybeacon(sc, hdr); rx_stats->is_mybeacon = ath9k_is_mybeacon(sc, hdr);
if (rx_stats->is_mybeacon) { if (rx_stats->is_mybeacon) {
...@@ -1173,15 +1175,11 @@ static int ath9k_rx_skb_preprocess(struct ath_softc *sc, ...@@ -1173,15 +1175,11 @@ static int ath9k_rx_skb_preprocess(struct ath_softc *sc,
/* /*
* This shouldn't happen, but have a safety check anyway. * This shouldn't happen, but have a safety check anyway.
*/ */
if (WARN_ON(!ah->curchan)) { if (WARN_ON(!ah->curchan))
ret = -EINVAL; return -EINVAL;
goto exit;
}
if (ath9k_process_rate(common, hw, rx_stats, rx_status)) { if (ath9k_process_rate(common, hw, rx_stats, rx_status))
ret =-EINVAL; return -EINVAL;
goto exit;
}
ath9k_process_rssi(common, hw, rx_stats, rx_status); ath9k_process_rssi(common, hw, rx_stats, rx_status);
...@@ -1196,9 +1194,11 @@ static int ath9k_rx_skb_preprocess(struct ath_softc *sc, ...@@ -1196,9 +1194,11 @@ static int ath9k_rx_skb_preprocess(struct ath_softc *sc,
sc->rx.num_pkts++; sc->rx.num_pkts++;
#endif #endif
exit: return 0;
sc->rx.discard_next = false;
return ret; corrupt:
sc->rx.discard_next = rx_stats->rs_more;
return -EINVAL;
} }
static void ath9k_rx_skb_postprocess(struct ath_common *common, static void ath9k_rx_skb_postprocess(struct ath_common *common,
......
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