net80211: ieee80211_ratectl*: switch to reusable KPI

Replace various void * / int argument combinations with common structures:
- ieee80211_ratectl_tx_status for *_tx_complete();
- ieee80211_ratectl_tx_stats for *_tx_update();

While here, improve amrr_tx_update() for a bit:
1. In case, if receiver is not known (typical for Ralink USB drivers),
refresh Tx rate for all nodes on the interface.
2. There was a misuse:
- otus(4) sends non-decreasing counters (as originally intended);
- but ural(4), rum(4) and run(4) are using 'read & clear' registers
to obtain statistics for some period of time (and those 'last period'
values are used as arguments for tx_update()). If arguments are not big
enough, they are just discarded after the next call.

Fix: move counting into *_tx_update()
(now otus(4) will zero out all node counters after every tx_update() call)

Tested with:
- Intel 3945BG (wpi(4)), STA mode.
- WUSB54GC (rum(4)), STA / HOSTAP mode.
- RTL8188EU (urtwn(4)), STA mode.

Reviewed by:	adrian
Differential Revision:	https://reviews.freebsd.org/D8037
This commit is contained in:
avos 2016-10-02 20:35:55 +00:00
parent 43f9a6a577
commit a6044b10a0
32 changed files with 385 additions and 232 deletions

View File

@ -3321,7 +3321,6 @@ _bwi_txeof(struct bwi_softc *sc, uint16_t tx_id, int acked, int data_txcnt)
struct bwi_txbuf *tb; struct bwi_txbuf *tb;
int ring_idx, buf_idx; int ring_idx, buf_idx;
struct ieee80211_node *ni; struct ieee80211_node *ni;
struct ieee80211vap *vap;
if (tx_id == 0) { if (tx_id == 0) {
device_printf(sc->sc_dev, "%s: zero tx id\n", __func__); device_printf(sc->sc_dev, "%s: zero tx id\n", __func__);
@ -3348,7 +3347,7 @@ _bwi_txeof(struct bwi_softc *sc, uint16_t tx_id, int acked, int data_txcnt)
if ((ni = tb->tb_ni) != NULL) { if ((ni = tb->tb_ni) != NULL) {
const struct bwi_txbuf_hdr *hdr = const struct bwi_txbuf_hdr *hdr =
mtod(tb->tb_mbuf, const struct bwi_txbuf_hdr *); mtod(tb->tb_mbuf, const struct bwi_txbuf_hdr *);
vap = ni->ni_vap; struct ieee80211_ratectl_tx_status txs;
/* NB: update rate control only for unicast frames */ /* NB: update rate control only for unicast frames */
if (hdr->txh_mac_ctrl & htole32(BWI_TXH_MAC_C_ACK)) { if (hdr->txh_mac_ctrl & htole32(BWI_TXH_MAC_C_ACK)) {
@ -3359,9 +3358,15 @@ _bwi_txeof(struct bwi_softc *sc, uint16_t tx_id, int acked, int data_txcnt)
* well so to avoid over-aggressive downshifting we * well so to avoid over-aggressive downshifting we
* treat any number of retries as "1". * treat any number of retries as "1".
*/ */
ieee80211_ratectl_tx_complete(vap, ni, txs.flags = IEEE80211_RATECTL_STATUS_LONG_RETRY;
(data_txcnt > 1) ? IEEE80211_RATECTL_TX_SUCCESS : txs.long_retries = acked;
IEEE80211_RATECTL_TX_FAILURE, &acked, NULL); if (data_txcnt > 1)
txs.status = IEEE80211_RATECTL_TX_SUCCESS;
else {
txs.status =
IEEE80211_RATECTL_TX_FAIL_UNSPECIFIED;
}
ieee80211_ratectl_tx_complete(ni, &txs);
} }
ieee80211_tx_complete(ni, tb->tb_mbuf, !acked); ieee80211_tx_complete(ni, tb->tb_mbuf, !acked);
tb->tb_ni = NULL; tb->tb_ni = NULL;

View File

@ -258,6 +258,8 @@ static int bwn_dma_newbuf(struct bwn_dma_ring *,
static void bwn_dma_buf_addr(void *, bus_dma_segment_t *, int, static void bwn_dma_buf_addr(void *, bus_dma_segment_t *, int,
bus_size_t, int); bus_size_t, int);
static uint8_t bwn_dma_check_redzone(struct bwn_dma_ring *, struct mbuf *); static uint8_t bwn_dma_check_redzone(struct bwn_dma_ring *, struct mbuf *);
static void bwn_ratectl_tx_complete(const struct ieee80211_node *,
const struct bwn_txstatus *);
static void bwn_dma_handle_txeof(struct bwn_mac *, static void bwn_dma_handle_txeof(struct bwn_mac *,
const struct bwn_txstatus *); const struct bwn_txstatus *);
static int bwn_dma_tx_start(struct bwn_mac *, struct ieee80211_node *, static int bwn_dma_tx_start(struct bwn_mac *, struct ieee80211_node *,
@ -5890,6 +5892,33 @@ drop:
device_printf(sc->sc_dev, "%s: dropped\n", __func__); device_printf(sc->sc_dev, "%s: dropped\n", __func__);
} }
static void
bwn_ratectl_tx_complete(const struct ieee80211_node *ni,
const struct bwn_txstatus *status)
{
struct ieee80211_ratectl_tx_status txs;
int retrycnt = 0;
/*
* If we don't get an ACK, then we should log the
* full framecnt. That may be 0 if it's a PHY
* failure, so ensure that gets logged as some
* retry attempt.
*/
txs.flags = IEEE80211_RATECTL_STATUS_LONG_RETRY;
if (status->ack) {
txs.status = IEEE80211_RATECTL_TX_SUCCESS;
retrycnt = status->framecnt - 1;
} else {
txs.status = IEEE80211_RATECTL_TX_FAIL_UNSPECIFIED;
retrycnt = status->framecnt;
if (retrycnt == 0)
retrycnt = 1;
}
txs.long_retries = retrycnt;
ieee80211_ratectl_tx_complete(ni, &txs);
}
static void static void
bwn_dma_handle_txeof(struct bwn_mac *mac, bwn_dma_handle_txeof(struct bwn_mac *mac,
const struct bwn_txstatus *status) const struct bwn_txstatus *status)
@ -5900,7 +5929,6 @@ bwn_dma_handle_txeof(struct bwn_mac *mac,
struct bwn_dmadesc_meta *meta; struct bwn_dmadesc_meta *meta;
struct bwn_softc *sc = mac->mac_sc; struct bwn_softc *sc = mac->mac_sc;
int slot; int slot;
int retrycnt = 0;
BWN_ASSERT_LOCKED(sc); BWN_ASSERT_LOCKED(sc);
@ -5925,24 +5953,7 @@ bwn_dma_handle_txeof(struct bwn_mac *mac,
KASSERT(meta->mt_m != NULL, KASSERT(meta->mt_m != NULL,
("%s:%d: fail", __func__, __LINE__)); ("%s:%d: fail", __func__, __LINE__));
/* bwn_ratectl_tx_complete(meta->mt_ni, status);
* If we don't get an ACK, then we should log the
* full framecnt. That may be 0 if it's a PHY
* failure, so ensure that gets logged as some
* retry attempt.
*/
if (status->ack) {
retrycnt = status->framecnt - 1;
} else {
retrycnt = status->framecnt;
if (retrycnt == 0)
retrycnt = 1;
}
ieee80211_ratectl_tx_complete(meta->mt_ni->ni_vap, meta->mt_ni,
status->ack ?
IEEE80211_RATECTL_TX_SUCCESS :
IEEE80211_RATECTL_TX_FAILURE,
&retrycnt, 0);
ieee80211_tx_complete(meta->mt_ni, meta->mt_m, 0); ieee80211_tx_complete(meta->mt_ni, meta->mt_m, 0);
meta->mt_ni = NULL; meta->mt_ni = NULL;
meta->mt_m = NULL; meta->mt_m = NULL;
@ -5970,7 +5981,6 @@ bwn_pio_handle_txeof(struct bwn_mac *mac,
struct bwn_pio_txqueue *tq; struct bwn_pio_txqueue *tq;
struct bwn_pio_txpkt *tp = NULL; struct bwn_pio_txpkt *tp = NULL;
struct bwn_softc *sc = mac->mac_sc; struct bwn_softc *sc = mac->mac_sc;
int retrycnt = 0;
BWN_ASSERT_LOCKED(sc); BWN_ASSERT_LOCKED(sc);
@ -5981,31 +5991,14 @@ bwn_pio_handle_txeof(struct bwn_mac *mac,
tq->tq_used -= roundup(tp->tp_m->m_pkthdr.len + BWN_HDRSIZE(mac), 4); tq->tq_used -= roundup(tp->tp_m->m_pkthdr.len + BWN_HDRSIZE(mac), 4);
tq->tq_free++; tq->tq_free++;
/* XXX ieee80211_tx_complete()? */
if (tp->tp_ni != NULL) { if (tp->tp_ni != NULL) {
/* /*
* Do any tx complete callback. Note this must * Do any tx complete callback. Note this must
* be done before releasing the node reference. * be done before releasing the node reference.
*/ */
/* bwn_ratectl_tx_complete(tp->tp_ni, status);
* If we don't get an ACK, then we should log the
* full framecnt. That may be 0 if it's a PHY
* failure, so ensure that gets logged as some
* retry attempt.
*/
if (status->ack) {
retrycnt = status->framecnt - 1;
} else {
retrycnt = status->framecnt;
if (retrycnt == 0)
retrycnt = 1;
}
ieee80211_ratectl_tx_complete(tp->tp_ni->ni_vap, tp->tp_ni,
status->ack ?
IEEE80211_RATECTL_TX_SUCCESS :
IEEE80211_RATECTL_TX_FAILURE,
&retrycnt, 0);
if (tp->tp_m->m_flags & M_TXCB) if (tp->tp_m->m_flags & M_TXCB)
ieee80211_process_callback(tp->tp_ni, tp->tp_m, 0); ieee80211_process_callback(tp->tp_ni, tp->tp_m, 0);
ieee80211_free_node(tp->tp_ni); ieee80211_free_node(tp->tp_ni);

View File

@ -3027,10 +3027,9 @@ iwm_mvm_rx_tx_cmd_single(struct iwm_softc *sc, struct iwm_rx_packet *pkt,
struct iwm_node *in) struct iwm_node *in)
{ {
struct iwm_mvm_tx_resp *tx_resp = (void *)pkt->data; struct iwm_mvm_tx_resp *tx_resp = (void *)pkt->data;
struct ieee80211_ratectl_tx_status *txs = &sc->sc_txs;
struct ieee80211_node *ni = &in->in_ni; struct ieee80211_node *ni = &in->in_ni;
struct ieee80211vap *vap = ni->ni_vap;
int status = le16toh(tx_resp->status.status) & IWM_TX_STATUS_MSK; int status = le16toh(tx_resp->status.status) & IWM_TX_STATUS_MSK;
int failack = tx_resp->failure_frame;
KASSERT(tx_resp->frame_count == 1, ("too many frames")); KASSERT(tx_resp->frame_count == 1, ("too many frames"));
@ -3046,16 +3045,32 @@ iwm_mvm_rx_tx_cmd_single(struct iwm_softc *sc, struct iwm_rx_packet *pkt,
le32toh(tx_resp->initial_rate), le32toh(tx_resp->initial_rate),
(int) le16toh(tx_resp->wireless_media_time)); (int) le16toh(tx_resp->wireless_media_time));
txs->flags = IEEE80211_RATECTL_STATUS_SHORT_RETRY |
IEEE80211_RATECTL_STATUS_LONG_RETRY;
txs->short_retries = tx_resp->failure_rts;
txs->long_retries = tx_resp->failure_frame;
if (status != IWM_TX_STATUS_SUCCESS && if (status != IWM_TX_STATUS_SUCCESS &&
status != IWM_TX_STATUS_DIRECT_DONE) { status != IWM_TX_STATUS_DIRECT_DONE) {
ieee80211_ratectl_tx_complete(vap, ni, switch (status) {
IEEE80211_RATECTL_TX_FAILURE, &failack, NULL); case IWM_TX_STATUS_FAIL_SHORT_LIMIT:
return (1); txs->status = IEEE80211_RATECTL_TX_FAIL_SHORT;
} else { break;
ieee80211_ratectl_tx_complete(vap, ni, case IWM_TX_STATUS_FAIL_LONG_LIMIT:
IEEE80211_RATECTL_TX_SUCCESS, &failack, NULL); txs->status = IEEE80211_RATECTL_TX_FAIL_LONG;
return (0); break;
case IWM_TX_STATUS_FAIL_LIFE_EXPIRE:
txs->status = IEEE80211_RATECTL_TX_FAIL_EXPIRED;
break;
default:
txs->status = IEEE80211_RATECTL_TX_FAIL_UNSPECIFIED;
break;
} }
} else {
txs->status = IEEE80211_RATECTL_TX_SUCCESS;
}
ieee80211_ratectl_tx_complete(ni, txs);
return (txs->status != IEEE80211_RATECTL_TX_SUCCESS);
} }
static void static void

View File

@ -392,6 +392,7 @@ struct iwm_softc {
struct mtx sc_mtx; struct mtx sc_mtx;
struct mbufq sc_snd; struct mbufq sc_snd;
struct ieee80211com sc_ic; struct ieee80211com sc_ic;
struct ieee80211_ratectl_tx_status sc_txs;
int sc_flags; int sc_flags;
#define IWM_FLAG_USE_ICT (1 << 0) #define IWM_FLAG_USE_ICT (1 << 0)

View File

@ -210,9 +210,10 @@ static void iwn4965_tx_done(struct iwn_softc *, struct iwn_rx_desc *,
struct iwn_rx_data *); struct iwn_rx_data *);
static void iwn5000_tx_done(struct iwn_softc *, struct iwn_rx_desc *, static void iwn5000_tx_done(struct iwn_softc *, struct iwn_rx_desc *,
struct iwn_rx_data *); struct iwn_rx_data *);
static void iwn_tx_done(struct iwn_softc *, struct iwn_rx_desc *, int, static void iwn_tx_done(struct iwn_softc *, struct iwn_rx_desc *, int, int,
uint8_t); uint8_t);
static void iwn_ampdu_tx_done(struct iwn_softc *, int, int, int, int, void *); static void iwn_ampdu_tx_done(struct iwn_softc *, int, int, int, int, int,
void *);
static void iwn_cmd_done(struct iwn_softc *, struct iwn_rx_desc *); static void iwn_cmd_done(struct iwn_softc *, struct iwn_rx_desc *);
static void iwn_notif_intr(struct iwn_softc *); static void iwn_notif_intr(struct iwn_softc *);
static void iwn_wakeup_intr(struct iwn_softc *); static void iwn_wakeup_intr(struct iwn_softc *);
@ -3147,6 +3148,7 @@ static void
iwn_rx_compressed_ba(struct iwn_softc *sc, struct iwn_rx_desc *desc, iwn_rx_compressed_ba(struct iwn_softc *sc, struct iwn_rx_desc *desc,
struct iwn_rx_data *data) struct iwn_rx_data *data)
{ {
struct ieee80211_ratectl_tx_status *txs = &sc->sc_txs;
struct iwn_ops *ops = &sc->ops; struct iwn_ops *ops = &sc->ops;
struct iwn_node *wn; struct iwn_node *wn;
struct ieee80211_node *ni; struct ieee80211_node *ni;
@ -3158,7 +3160,7 @@ iwn_rx_compressed_ba(struct iwn_softc *sc, struct iwn_rx_desc *desc,
uint64_t bitmap; uint64_t bitmap;
uint16_t ssn; uint16_t ssn;
uint8_t tid; uint8_t tid;
int ackfailcnt = 0, i, lastidx, qid, *res, shift; int i, lastidx, qid, *res, shift;
int tx_ok = 0, tx_err = 0; int tx_ok = 0, tx_err = 0;
DPRINTF(sc, IWN_DEBUG_TRACE | IWN_DEBUG_XMIT, "->%s begin\n", __func__); DPRINTF(sc, IWN_DEBUG_TRACE | IWN_DEBUG_XMIT, "->%s begin\n", __func__);
@ -3227,15 +3229,15 @@ iwn_rx_compressed_ba(struct iwn_softc *sc, struct iwn_rx_desc *desc,
ni = tap->txa_ni; ni = tap->txa_ni;
bitmap = (le64toh(ba->bitmap) >> shift) & wn->agg[tid].bitmap; bitmap = (le64toh(ba->bitmap) >> shift) & wn->agg[tid].bitmap;
for (i = 0; bitmap; i++) { for (i = 0; bitmap; i++) {
txs->flags = 0; /* XXX TODO */
if ((bitmap & 1) == 0) { if ((bitmap & 1) == 0) {
tx_err ++; tx_err ++;
ieee80211_ratectl_tx_complete(ni->ni_vap, ni, txs->status = IEEE80211_RATECTL_TX_FAIL_UNSPECIFIED;
IEEE80211_RATECTL_TX_FAILURE, &ackfailcnt, NULL);
} else { } else {
tx_ok ++; tx_ok ++;
ieee80211_ratectl_tx_complete(ni->ni_vap, ni, txs->status = IEEE80211_RATECTL_TX_SUCCESS;
IEEE80211_RATECTL_TX_SUCCESS, &ackfailcnt, NULL);
} }
ieee80211_ratectl_tx_complete(ni, txs);
bitmap >>= 1; bitmap >>= 1;
} }
@ -3501,9 +3503,9 @@ iwn4965_tx_done(struct iwn_softc *sc, struct iwn_rx_desc *desc,
bus_dmamap_sync(ring->data_dmat, data->map, BUS_DMASYNC_POSTREAD); bus_dmamap_sync(ring->data_dmat, data->map, BUS_DMASYNC_POSTREAD);
if (qid >= sc->firstaggqueue) { if (qid >= sc->firstaggqueue) {
iwn_ampdu_tx_done(sc, qid, desc->idx, stat->nframes, iwn_ampdu_tx_done(sc, qid, desc->idx, stat->nframes,
stat->ackfailcnt, &stat->status); stat->rtsfailcnt, stat->ackfailcnt, &stat->status);
} else { } else {
iwn_tx_done(sc, desc, stat->ackfailcnt, iwn_tx_done(sc, desc, stat->rtsfailcnt, stat->ackfailcnt,
le32toh(stat->status) & 0xff); le32toh(stat->status) & 0xff);
} }
} }
@ -3536,9 +3538,9 @@ iwn5000_tx_done(struct iwn_softc *sc, struct iwn_rx_desc *desc,
bus_dmamap_sync(ring->data_dmat, data->map, BUS_DMASYNC_POSTREAD); bus_dmamap_sync(ring->data_dmat, data->map, BUS_DMASYNC_POSTREAD);
if (qid >= sc->firstaggqueue) { if (qid >= sc->firstaggqueue) {
iwn_ampdu_tx_done(sc, qid, desc->idx, stat->nframes, iwn_ampdu_tx_done(sc, qid, desc->idx, stat->nframes,
stat->ackfailcnt, &stat->status); stat->rtsfailcnt, stat->ackfailcnt, &stat->status);
} else { } else {
iwn_tx_done(sc, desc, stat->ackfailcnt, iwn_tx_done(sc, desc, stat->rtsfailcnt, stat->ackfailcnt,
le16toh(stat->status) & 0xff); le16toh(stat->status) & 0xff);
} }
} }
@ -3547,14 +3549,14 @@ iwn5000_tx_done(struct iwn_softc *sc, struct iwn_rx_desc *desc,
* Adapter-independent backend for TX_DONE firmware notifications. * Adapter-independent backend for TX_DONE firmware notifications.
*/ */
static void static void
iwn_tx_done(struct iwn_softc *sc, struct iwn_rx_desc *desc, int ackfailcnt, iwn_tx_done(struct iwn_softc *sc, struct iwn_rx_desc *desc, int rtsfailcnt,
uint8_t status) int ackfailcnt, uint8_t status)
{ {
struct ieee80211_ratectl_tx_status *txs = &sc->sc_txs;
struct iwn_tx_ring *ring = &sc->txq[desc->qid & 0xf]; struct iwn_tx_ring *ring = &sc->txq[desc->qid & 0xf];
struct iwn_tx_data *data = &ring->data[desc->idx]; struct iwn_tx_data *data = &ring->data[desc->idx];
struct mbuf *m; struct mbuf *m;
struct ieee80211_node *ni; struct ieee80211_node *ni;
struct ieee80211vap *vap;
KASSERT(data->ni != NULL, ("no node")); KASSERT(data->ni != NULL, ("no node"));
@ -3565,17 +3567,33 @@ iwn_tx_done(struct iwn_softc *sc, struct iwn_rx_desc *desc, int ackfailcnt,
bus_dmamap_unload(ring->data_dmat, data->map); bus_dmamap_unload(ring->data_dmat, data->map);
m = data->m, data->m = NULL; m = data->m, data->m = NULL;
ni = data->ni, data->ni = NULL; ni = data->ni, data->ni = NULL;
vap = ni->ni_vap;
/* /*
* Update rate control statistics for the node. * Update rate control statistics for the node.
*/ */
if (status & IWN_TX_FAIL) txs->flags = IEEE80211_RATECTL_STATUS_SHORT_RETRY |
ieee80211_ratectl_tx_complete(vap, ni, IEEE80211_RATECTL_STATUS_LONG_RETRY;
IEEE80211_RATECTL_TX_FAILURE, &ackfailcnt, NULL); txs->short_retries = rtsfailcnt;
else txs->long_retries = ackfailcnt;
ieee80211_ratectl_tx_complete(vap, ni, if (!(status & IWN_TX_FAIL))
IEEE80211_RATECTL_TX_SUCCESS, &ackfailcnt, NULL); txs->status = IEEE80211_RATECTL_TX_SUCCESS;
else {
switch (status) {
case IWN_TX_FAIL_SHORT_LIMIT:
txs->status = IEEE80211_RATECTL_TX_FAIL_SHORT;
break;
case IWN_TX_FAIL_LONG_LIMIT:
txs->status = IEEE80211_RATECTL_TX_FAIL_LONG;
break;
case IWN_TX_STATUS_FAIL_LIFE_EXPIRE:
txs->status = IEEE80211_RATECTL_TX_FAIL_EXPIRED;
break;
default:
txs->status = IEEE80211_RATECTL_TX_FAIL_UNSPECIFIED;
break;
}
}
ieee80211_ratectl_tx_complete(ni, txs);
/* /*
* Channels marked for "radar" require traffic to be received * Channels marked for "radar" require traffic to be received
@ -3640,10 +3658,11 @@ iwn_cmd_done(struct iwn_softc *sc, struct iwn_rx_desc *desc)
static void static void
iwn_ampdu_tx_done(struct iwn_softc *sc, int qid, int idx, int nframes, iwn_ampdu_tx_done(struct iwn_softc *sc, int qid, int idx, int nframes,
int ackfailcnt, void *stat) int rtsfailcnt, int ackfailcnt, void *stat)
{ {
struct iwn_ops *ops = &sc->ops; struct iwn_ops *ops = &sc->ops;
struct iwn_tx_ring *ring = &sc->txq[qid]; struct iwn_tx_ring *ring = &sc->txq[qid];
struct ieee80211_ratectl_tx_status *txs = &sc->sc_txs;
struct iwn_tx_data *data; struct iwn_tx_data *data;
struct mbuf *m; struct mbuf *m;
struct iwn_node *wn; struct iwn_node *wn;
@ -3682,6 +3701,10 @@ iwn_ampdu_tx_done(struct iwn_softc *sc, int qid, int idx, int nframes,
* handled differently. * handled differently.
*/ */
if (nframes == 1) { if (nframes == 1) {
txs->flags = IEEE80211_RATECTL_STATUS_SHORT_RETRY |
IEEE80211_RATECTL_STATUS_LONG_RETRY;
txs->short_retries = rtsfailcnt;
txs->long_retries = ackfailcnt;
if ((*status & 0xff) != 1 && (*status & 0xff) != 2) { if ((*status & 0xff) != 1 && (*status & 0xff) != 2) {
#ifdef NOT_YET #ifdef NOT_YET
printf("ieee80211_send_bar()\n"); printf("ieee80211_send_bar()\n");
@ -3691,11 +3714,8 @@ iwn_ampdu_tx_done(struct iwn_softc *sc, int qid, int idx, int nframes,
* notification is pushed up to the rate control * notification is pushed up to the rate control
* layer. * layer.
*/ */
ieee80211_ratectl_tx_complete(ni->ni_vap, /* XXX */
ni, txs->status = IEEE80211_RATECTL_TX_FAIL_UNSPECIFIED;
IEEE80211_RATECTL_TX_FAILURE,
&ackfailcnt,
NULL);
} else { } else {
/* /*
* If nframes=1, then we won't be getting a BA for * If nframes=1, then we won't be getting a BA for
@ -3703,12 +3723,9 @@ iwn_ampdu_tx_done(struct iwn_softc *sc, int qid, int idx, int nframes,
* rate control code with how many retries were * rate control code with how many retries were
* needed to send it. * needed to send it.
*/ */
ieee80211_ratectl_tx_complete(ni->ni_vap, txs->status = IEEE80211_RATECTL_TX_SUCCESS;
ni,
IEEE80211_RATECTL_TX_SUCCESS,
&ackfailcnt,
NULL);
} }
ieee80211_ratectl_tx_complete(ni, txs);
} }
bitmap = 0; bitmap = 0;

View File

@ -238,6 +238,7 @@ struct iwn_softc {
struct cdev *sc_cdev; struct cdev *sc_cdev;
struct mtx sc_mtx; struct mtx sc_mtx;
struct ieee80211com sc_ic; struct ieee80211com sc_ic;
struct ieee80211_ratectl_tx_status sc_txs;
u_int sc_flags; u_int sc_flags;
#define IWN_FLAG_HAS_OTPROM (1 << 1) #define IWN_FLAG_HAS_OTPROM (1 << 1)

View File

@ -2148,14 +2148,18 @@ otus_hw_rate_is_ofdm(struct otus_softc *sc, uint8_t hw_rate)
static void static void
otus_tx_update_ratectl(struct otus_softc *sc, struct ieee80211_node *ni) otus_tx_update_ratectl(struct otus_softc *sc, struct ieee80211_node *ni)
{ {
int tx, tx_success, tx_retry; struct ieee80211_ratectl_tx_stats *txs = &sc->sc_txs;
struct otus_node *on = OTUS_NODE(ni);
tx = OTUS_NODE(ni)->tx_done; txs->flags = IEEE80211_RATECTL_TX_STATS_NODE |
tx_success = OTUS_NODE(ni)->tx_done - OTUS_NODE(ni)->tx_err; IEEE80211_RATECTL_TX_STATS_RETRIES;
tx_retry = OTUS_NODE(ni)->tx_retries; txs->ni = ni;
txs->nframes = on->tx_done;
txs->nsuccess = on->tx_done - on->tx_err;
txs->nretries = on->tx_retries;
ieee80211_ratectl_tx_update(ni->ni_vap, ni, &tx, &tx_success, ieee80211_ratectl_tx_update(ni->ni_vap, txs);
&tx_retry); on->tx_done = on->tx_err = on->tx_retries = 0;
} }
/* /*

View File

@ -997,6 +997,7 @@ struct otus_vap {
struct otus_softc { struct otus_softc {
struct ieee80211com sc_ic; struct ieee80211com sc_ic;
struct ieee80211_ratectl_tx_stats sc_txs;
struct mbufq sc_snd; struct mbufq sc_snd;
device_t sc_dev; device_t sc_dev;
struct usb_device *sc_udev; struct usb_device *sc_udev;

View File

@ -44,6 +44,7 @@ __FBSDID("$FreeBSD$");
#include <net80211/ieee80211_var.h> #include <net80211/ieee80211_var.h>
#include <net80211/ieee80211_radiotap.h> #include <net80211/ieee80211_radiotap.h>
#include <net80211/ieee80211_ratectl.h>
#include <dev/pci/pcireg.h> #include <dev/pci/pcireg.h>
#include <dev/pci/pcivar.h> #include <dev/pci/pcivar.h>

View File

@ -911,17 +911,18 @@ rt2560_encryption_intr(struct rt2560_softc *sc)
static void static void
rt2560_tx_intr(struct rt2560_softc *sc) rt2560_tx_intr(struct rt2560_softc *sc)
{ {
struct ieee80211_ratectl_tx_status *txs = &sc->sc_txs;
struct rt2560_tx_desc *desc; struct rt2560_tx_desc *desc;
struct rt2560_tx_data *data; struct rt2560_tx_data *data;
struct mbuf *m; struct mbuf *m;
struct ieee80211vap *vap;
struct ieee80211_node *ni; struct ieee80211_node *ni;
uint32_t flags; uint32_t flags;
int retrycnt, status; int status;
bus_dmamap_sync(sc->txq.desc_dmat, sc->txq.desc_map, bus_dmamap_sync(sc->txq.desc_dmat, sc->txq.desc_map,
BUS_DMASYNC_POSTREAD); BUS_DMASYNC_POSTREAD);
txs->flags = IEEE80211_RATECTL_STATUS_LONG_RETRY;
for (;;) { for (;;) {
desc = &sc->txq.desc[sc->txq.next]; desc = &sc->txq.desc[sc->txq.next];
data = &sc->txq.data[sc->txq.next]; data = &sc->txq.data[sc->txq.next];
@ -934,41 +935,37 @@ rt2560_tx_intr(struct rt2560_softc *sc)
m = data->m; m = data->m;
ni = data->ni; ni = data->ni;
vap = ni->ni_vap;
switch (flags & RT2560_TX_RESULT_MASK) { switch (flags & RT2560_TX_RESULT_MASK) {
case RT2560_TX_SUCCESS: case RT2560_TX_SUCCESS:
retrycnt = 0; txs->status = IEEE80211_RATECTL_TX_SUCCESS;
txs->long_retries = 0;
DPRINTFN(sc, 10, "%s\n", "data frame sent successfully"); DPRINTFN(sc, 10, "%s\n", "data frame sent successfully");
if (data->rix != IEEE80211_FIXED_RATE_NONE) if (data->rix != IEEE80211_FIXED_RATE_NONE)
ieee80211_ratectl_tx_complete(vap, ni, ieee80211_ratectl_tx_complete(ni, txs);
IEEE80211_RATECTL_TX_SUCCESS,
&retrycnt, NULL);
status = 0; status = 0;
break; break;
case RT2560_TX_SUCCESS_RETRY: case RT2560_TX_SUCCESS_RETRY:
retrycnt = RT2560_TX_RETRYCNT(flags); txs->status = IEEE80211_RATECTL_TX_SUCCESS;
txs->long_retries = RT2560_TX_RETRYCNT(flags);
DPRINTFN(sc, 9, "data frame sent after %u retries\n", DPRINTFN(sc, 9, "data frame sent after %u retries\n",
retrycnt); txs->long_retries);
if (data->rix != IEEE80211_FIXED_RATE_NONE) if (data->rix != IEEE80211_FIXED_RATE_NONE)
ieee80211_ratectl_tx_complete(vap, ni, ieee80211_ratectl_tx_complete(ni, txs);
IEEE80211_RATECTL_TX_SUCCESS,
&retrycnt, NULL);
status = 0; status = 0;
break; break;
case RT2560_TX_FAIL_RETRY: case RT2560_TX_FAIL_RETRY:
retrycnt = RT2560_TX_RETRYCNT(flags); txs->status = IEEE80211_RATECTL_TX_FAIL_LONG;
txs->long_retries = RT2560_TX_RETRYCNT(flags);
DPRINTFN(sc, 9, "data frame failed after %d retries\n", DPRINTFN(sc, 9, "data frame failed after %d retries\n",
retrycnt); txs->long_retries);
if (data->rix != IEEE80211_FIXED_RATE_NONE) if (data->rix != IEEE80211_FIXED_RATE_NONE)
ieee80211_ratectl_tx_complete(vap, ni, ieee80211_ratectl_tx_complete(ni, txs);
IEEE80211_RATECTL_TX_FAILURE,
&retrycnt, NULL);
status = 1; status = 1;
break; break;

View File

@ -105,6 +105,7 @@ struct rt2560_vap {
struct rt2560_softc { struct rt2560_softc {
struct ieee80211com sc_ic; struct ieee80211com sc_ic;
struct ieee80211_ratectl_tx_status sc_txs;
struct mtx sc_mtx; struct mtx sc_mtx;
struct mbufq sc_snd; struct mbufq sc_snd;
device_t sc_dev; device_t sc_dev;

View File

@ -851,12 +851,13 @@ rt2661_eeprom_read(struct rt2661_softc *sc, uint8_t addr)
static void static void
rt2661_tx_intr(struct rt2661_softc *sc) rt2661_tx_intr(struct rt2661_softc *sc)
{ {
struct ieee80211_ratectl_tx_status *txs = &sc->sc_txs;
struct rt2661_tx_ring *txq; struct rt2661_tx_ring *txq;
struct rt2661_tx_data *data; struct rt2661_tx_data *data;
uint32_t val; uint32_t val;
int error, qid, retrycnt; int error, qid;
struct ieee80211vap *vap;
txs->flags = IEEE80211_RATECTL_TX_FAIL_LONG;
for (;;) { for (;;) {
struct ieee80211_node *ni; struct ieee80211_node *ni;
struct mbuf *m; struct mbuf *m;
@ -879,31 +880,27 @@ rt2661_tx_intr(struct rt2661_softc *sc)
/* if no frame has been sent, ignore */ /* if no frame has been sent, ignore */
if (ni == NULL) if (ni == NULL)
continue; continue;
else
vap = ni->ni_vap;
switch (RT2661_TX_RESULT(val)) { switch (RT2661_TX_RESULT(val)) {
case RT2661_TX_SUCCESS: case RT2661_TX_SUCCESS:
retrycnt = RT2661_TX_RETRYCNT(val); txs->status = IEEE80211_RATECTL_TX_SUCCESS;
txs->long_retries = RT2661_TX_RETRYCNT(val);
DPRINTFN(sc, 10, "data frame sent successfully after " DPRINTFN(sc, 10, "data frame sent successfully after "
"%d retries\n", retrycnt); "%d retries\n", txs->long_retries);
if (data->rix != IEEE80211_FIXED_RATE_NONE) if (data->rix != IEEE80211_FIXED_RATE_NONE)
ieee80211_ratectl_tx_complete(vap, ni, ieee80211_ratectl_tx_complete(ni, txs);
IEEE80211_RATECTL_TX_SUCCESS,
&retrycnt, NULL);
error = 0; error = 0;
break; break;
case RT2661_TX_RETRY_FAIL: case RT2661_TX_RETRY_FAIL:
retrycnt = RT2661_TX_RETRYCNT(val); txs->status = IEEE80211_RATECTL_TX_FAIL_LONG;
txs->long_retries = RT2661_TX_RETRYCNT(val);
DPRINTFN(sc, 9, "%s\n", DPRINTFN(sc, 9, "%s\n",
"sending data frame failed (too much retries)"); "sending data frame failed (too much retries)");
if (data->rix != IEEE80211_FIXED_RATE_NONE) if (data->rix != IEEE80211_FIXED_RATE_NONE)
ieee80211_ratectl_tx_complete(vap, ni, ieee80211_ratectl_tx_complete(ni, txs);
IEEE80211_RATECTL_TX_FAILURE,
&retrycnt, NULL);
error = 1; error = 1;
break; break;

View File

@ -98,6 +98,7 @@ struct rt2661_vap {
struct rt2661_softc { struct rt2661_softc {
struct ieee80211com sc_ic; struct ieee80211com sc_ic;
struct ieee80211_ratectl_tx_status sc_txs;
struct mtx sc_mtx; struct mtx sc_mtx;
struct mbufq sc_snd; struct mbufq sc_snd;
device_t sc_dev; device_t sc_dev;

View File

@ -1083,12 +1083,13 @@ rt2860_intr_coherent(struct rt2860_softc *sc)
static void static void
rt2860_drain_stats_fifo(struct rt2860_softc *sc) rt2860_drain_stats_fifo(struct rt2860_softc *sc)
{ {
struct ieee80211_ratectl_tx_status *txs = &sc->sc_txs;
struct ieee80211_node *ni; struct ieee80211_node *ni;
uint32_t stat; uint32_t stat;
int retrycnt;
uint8_t wcid, mcs, pid; uint8_t wcid, mcs, pid;
/* drain Tx status FIFO (maxsize = 16) */ /* drain Tx status FIFO (maxsize = 16) */
txs->flags = IEEE80211_RATECTL_STATUS_LONG_RETRY;
while ((stat = RAL_READ(sc, RT2860_TX_STAT_FIFO)) & RT2860_TXQ_VLD) { while ((stat = RAL_READ(sc, RT2860_TX_STAT_FIFO)) & RT2860_TXQ_VLD) {
DPRINTFN(4, ("tx stat 0x%08x\n", stat)); DPRINTFN(4, ("tx stat 0x%08x\n", stat));
@ -1110,14 +1111,15 @@ rt2860_drain_stats_fifo(struct rt2860_softc *sc)
mcs = (stat >> RT2860_TXQ_MCS_SHIFT) & 0x7f; mcs = (stat >> RT2860_TXQ_MCS_SHIFT) & 0x7f;
pid = (stat >> RT2860_TXQ_PID_SHIFT) & 0xf; pid = (stat >> RT2860_TXQ_PID_SHIFT) & 0xf;
if (mcs + 1 != pid) if (mcs + 1 != pid)
retrycnt = 1; txs->long_retries = 1;
else else
retrycnt = 0; txs->long_retries = 0;
ieee80211_ratectl_tx_complete(ni->ni_vap, ni, txs->status = IEEE80211_RATECTL_TX_SUCCESS;
IEEE80211_RATECTL_TX_SUCCESS, &retrycnt, NULL); ieee80211_ratectl_tx_complete(ni, txs);
} else { } else {
ieee80211_ratectl_tx_complete(ni->ni_vap, ni, txs->status = IEEE80211_RATECTL_TX_FAIL_UNSPECIFIED;
IEEE80211_RATECTL_TX_FAILURE, &retrycnt, NULL); txs->long_retries = 1; /* XXX */
ieee80211_ratectl_tx_complete(ni, txs);
if_inc_counter(ni->ni_vap->iv_ifp, if_inc_counter(ni->ni_vap->iv_ifp,
IFCOUNTER_OERRORS, 1); IFCOUNTER_OERRORS, 1);
} }

View File

@ -116,6 +116,7 @@ struct rt2860_vap {
struct rt2860_softc { struct rt2860_softc {
struct ieee80211com sc_ic; struct ieee80211com sc_ic;
struct ieee80211_ratectl_tx_status sc_txs;
struct mbufq sc_snd; struct mbufq sc_snd;
struct mtx sc_mtx; struct mtx sc_mtx;
device_t sc_dev; device_t sc_dev;

View File

@ -1016,7 +1016,7 @@ static void
urtwn_r88e_ratectl_tx_complete(struct urtwn_softc *sc, void *arg) urtwn_r88e_ratectl_tx_complete(struct urtwn_softc *sc, void *arg)
{ {
struct r88e_tx_rpt_ccx *rpt = arg; struct r88e_tx_rpt_ccx *rpt = arg;
struct ieee80211vap *vap; struct ieee80211_ratectl_tx_status *txs = &sc->sc_txs;
struct ieee80211_node *ni; struct ieee80211_node *ni;
uint8_t macid; uint8_t macid;
int ntries; int ntries;
@ -1027,19 +1027,28 @@ urtwn_r88e_ratectl_tx_complete(struct urtwn_softc *sc, void *arg)
URTWN_NT_LOCK(sc); URTWN_NT_LOCK(sc);
ni = sc->node_list[macid]; ni = sc->node_list[macid];
if (ni != NULL) { if (ni != NULL) {
vap = ni->ni_vap;
URTWN_DPRINTF(sc, URTWN_DEBUG_INTR, "%s: frame for macid %d was" URTWN_DPRINTF(sc, URTWN_DEBUG_INTR, "%s: frame for macid %d was"
"%s sent (%d retries)\n", __func__, macid, "%s sent (%d retries)\n", __func__, macid,
(rpt->rptb1 & R88E_RPTB1_PKT_OK) ? "" : " not", (rpt->rptb1 & R88E_RPTB1_PKT_OK) ? "" : " not",
ntries); ntries);
if (rpt->rptb1 & R88E_RPTB1_PKT_OK) { txs->flags = IEEE80211_RATECTL_STATUS_LONG_RETRY |
ieee80211_ratectl_tx_complete(vap, ni, IEEE80211_RATECTL_STATUS_FINAL_RATE;
IEEE80211_RATECTL_TX_SUCCESS, &ntries, NULL); txs->long_retries = ntries;
} else { if (rpt->final_rate > URTWN_RIDX_OFDM54) { /* MCS */
ieee80211_ratectl_tx_complete(vap, ni, txs->final_rate =
IEEE80211_RATECTL_TX_FAILURE, &ntries, NULL); (rpt->final_rate - 12) | IEEE80211_RATE_MCS;
} } else
txs->final_rate = ridx2rate[rpt->final_rate];
if (rpt->rptb1 & R88E_RPTB1_PKT_OK)
txs->status = IEEE80211_RATECTL_TX_SUCCESS;
else if (rpt->rptb2 & R88E_RPTB2_RETRY_OVER)
txs->status = IEEE80211_RATECTL_TX_FAIL_LONG;
else if (rpt->rptb2 & R88E_RPTB2_LIFE_EXPIRE)
txs->status = IEEE80211_RATECTL_TX_FAIL_EXPIRED;
else
txs->status = IEEE80211_RATECTL_TX_FAIL_UNSPECIFIED;
ieee80211_ratectl_tx_complete(ni, txs);
} else { } else {
URTWN_DPRINTF(sc, URTWN_DEBUG_INTR, "%s: macid %d, ni is NULL\n", URTWN_DPRINTF(sc, URTWN_DEBUG_INTR, "%s: macid %d, ni is NULL\n",
__func__, macid); __func__, macid);

View File

@ -1227,9 +1227,8 @@ struct r88e_tx_rpt_ccx {
#define R88E_RPTB2_LIFE_EXPIRE 0x40 #define R88E_RPTB2_LIFE_EXPIRE 0x40
#define R88E_RPTB2_RETRY_OVER 0x80 #define R88E_RPTB2_RETRY_OVER 0x80
uint8_t rptb3; uint16_t ccx_qtime;
uint8_t rptb4; uint8_t final_rate;
uint8_t rptb5;
uint8_t rptb6; uint8_t rptb6;
#define R88E_RPTB6_QSEL_M 0xf0 #define R88E_RPTB6_QSEL_M 0xf0
#define R88E_RPTB6_QSEL_S 4 #define R88E_RPTB6_QSEL_S 4

View File

@ -136,6 +136,7 @@ union urtwn_rom {
struct urtwn_softc { struct urtwn_softc {
struct ieee80211com sc_ic; struct ieee80211com sc_ic;
struct ieee80211_ratectl_tx_status sc_txs;
struct mbufq sc_snd; struct mbufq sc_snd;
device_t sc_dev; device_t sc_dev;
struct usb_device *sc_udev; struct usb_device *sc_udev;

View File

@ -1668,8 +1668,10 @@ rum_tx_data(struct rum_softc *sc, struct mbuf *m0, struct ieee80211_node *ni)
rate = tp->ucastrate; rate = tp->ucastrate;
else if (m0->m_flags & M_EAPOL) else if (m0->m_flags & M_EAPOL)
rate = tp->mgmtrate; rate = tp->mgmtrate;
else else {
(void) ieee80211_ratectl_rate(ni, NULL, 0);
rate = ni->ni_txrate; rate = ni->ni_txrate;
}
if (wh->i_fc[1] & IEEE80211_FC1_PROTECTED) { if (wh->i_fc[1] & IEEE80211_FC1_PROTECTED) {
k = ieee80211_crypto_get_txkey(ni, m0); k = ieee80211_crypto_get_txkey(ni, m0);
@ -3154,9 +3156,8 @@ rum_ratectl_task(void *arg, int pending)
struct rum_vap *rvp = arg; struct rum_vap *rvp = arg;
struct ieee80211vap *vap = &rvp->vap; struct ieee80211vap *vap = &rvp->vap;
struct rum_softc *sc = vap->iv_ic->ic_softc; struct rum_softc *sc = vap->iv_ic->ic_softc;
struct ieee80211_node *ni; struct ieee80211_ratectl_tx_stats *txs = &sc->sc_txs;
int ok[3], fail; int ok[3], fail;
int sum, success, retrycnt;
RUM_LOCK(sc); RUM_LOCK(sc);
/* read and clear statistic registers (STA_CSR0 to STA_CSR5) */ /* read and clear statistic registers (STA_CSR0 to STA_CSR5) */
@ -3167,17 +3168,14 @@ rum_ratectl_task(void *arg, int pending)
ok[2] = (le32toh(sc->sta[5]) & 0xffff); /* TX ok w/ multiple retries */ ok[2] = (le32toh(sc->sta[5]) & 0xffff); /* TX ok w/ multiple retries */
fail = (le32toh(sc->sta[5]) >> 16); /* TX retry-fail count */ fail = (le32toh(sc->sta[5]) >> 16); /* TX retry-fail count */
success = ok[0] + ok[1] + ok[2]; txs->flags = IEEE80211_RATECTL_TX_STATS_RETRIES;
sum = success + fail; txs->nframes = ok[0] + ok[1] + ok[2] + fail;
txs->nsuccess = txs->nframes - fail;
/* XXX at least */ /* XXX at least */
retrycnt = ok[1] + ok[2] * 2 + fail * (rvp->maxretry + 1); txs->nretries = ok[1] + ok[2] * 2 + fail * (rvp->maxretry + 1);
if (sum != 0) { if (txs->nframes != 0)
ni = ieee80211_ref_node(vap->iv_bss); ieee80211_ratectl_tx_update(vap, txs);
ieee80211_ratectl_tx_update(vap, ni, &sum, &ok, &retrycnt);
(void) ieee80211_ratectl_rate(ni, NULL, 0);
ieee80211_free_node(ni);
}
/* count TX retry-fail as Tx errors */ /* count TX retry-fail as Tx errors */
if_inc_counter(vap->iv_ifp, IFCOUNTER_OERRORS, fail); if_inc_counter(vap->iv_ifp, IFCOUNTER_OERRORS, fail);

View File

@ -110,6 +110,7 @@ enum {
struct rum_softc { struct rum_softc {
struct ieee80211com sc_ic; struct ieee80211com sc_ic;
struct ieee80211_ratectl_tx_stats sc_txs;
struct mbufq sc_snd; struct mbufq sc_snd;
device_t sc_dev; device_t sc_dev;
struct usb_device *sc_udev; struct usb_device *sc_udev;

View File

@ -2552,11 +2552,12 @@ static void
run_iter_func(void *arg, struct ieee80211_node *ni) run_iter_func(void *arg, struct ieee80211_node *ni)
{ {
struct run_softc *sc = arg; struct run_softc *sc = arg;
struct ieee80211_ratectl_tx_stats *txs = &sc->sc_txs;
struct ieee80211vap *vap = ni->ni_vap; struct ieee80211vap *vap = ni->ni_vap;
struct run_node *rn = RUN_NODE(ni); struct run_node *rn = RUN_NODE(ni);
union run_stats sta[2]; union run_stats sta[2];
uint16_t (*wstat)[3]; uint16_t (*wstat)[3];
int txcnt, success, retrycnt, error; int error;
RUN_LOCK(sc); RUN_LOCK(sc);
@ -2565,6 +2566,9 @@ run_iter_func(void *arg, struct ieee80211_node *ni)
ni != vap->iv_bss) ni != vap->iv_bss)
goto fail; goto fail;
txs->flags = IEEE80211_RATECTL_TX_STATS_NODE |
IEEE80211_RATECTL_TX_STATS_RETRIES;
txs->ni = ni;
if (sc->rvp_cnt <= 1 && (vap->iv_opmode == IEEE80211_M_IBSS || if (sc->rvp_cnt <= 1 && (vap->iv_opmode == IEEE80211_M_IBSS ||
vap->iv_opmode == IEEE80211_M_STA)) { vap->iv_opmode == IEEE80211_M_STA)) {
/* read statistic counters (clear on read) and update AMRR state */ /* read statistic counters (clear on read) and update AMRR state */
@ -2577,12 +2581,15 @@ run_iter_func(void *arg, struct ieee80211_node *ni)
if_inc_counter(vap->iv_ifp, IFCOUNTER_OERRORS, if_inc_counter(vap->iv_ifp, IFCOUNTER_OERRORS,
le16toh(sta[0].error.fail)); le16toh(sta[0].error.fail));
retrycnt = le16toh(sta[1].tx.retry); txs->nretries = le16toh(sta[1].tx.retry);
success = le16toh(sta[1].tx.success); txs->nsuccess = le16toh(sta[1].tx.success);
txcnt = retrycnt + success + le16toh(sta[0].error.fail); /* nretries??? */
txs->nframes = txs->nretries + txs->nsuccess +
le16toh(sta[0].error.fail);
DPRINTFN(3, "retrycnt=%d success=%d failcnt=%d\n", DPRINTFN(3, "retrycnt=%d success=%d failcnt=%d\n",
retrycnt, success, le16toh(sta[0].error.fail)); txs->nretries, txs->nsuccess,
le16toh(sta[0].error.fail));
} else { } else {
wstat = &(sc->wcid_stats[RUN_AID2WCID(ni->ni_associd)]); wstat = &(sc->wcid_stats[RUN_AID2WCID(ni->ni_associd)]);
@ -2590,16 +2597,16 @@ run_iter_func(void *arg, struct ieee80211_node *ni)
wstat > &(sc->wcid_stats[RT2870_WCID_MAX])) wstat > &(sc->wcid_stats[RT2870_WCID_MAX]))
goto fail; goto fail;
txcnt = (*wstat)[RUN_TXCNT]; txs->nretries = (*wstat)[RUN_RETRY];
success = (*wstat)[RUN_SUCCESS]; txs->nsuccess = (*wstat)[RUN_SUCCESS];
retrycnt = (*wstat)[RUN_RETRY]; txs->nframes = (*wstat)[RUN_TXCNT];
DPRINTFN(3, "retrycnt=%d txcnt=%d success=%d\n", DPRINTFN(3, "retrycnt=%d txcnt=%d success=%d\n",
retrycnt, txcnt, success); txs->nretries, txs->nframes, txs->nsuccess);
memset(wstat, 0, sizeof(*wstat)); memset(wstat, 0, sizeof(*wstat));
} }
ieee80211_ratectl_tx_update(vap, ni, &txcnt, &success, &retrycnt); ieee80211_ratectl_tx_update(vap, txs);
rn->amrr_ridx = ieee80211_ratectl_rate(ni, NULL, 0); rn->amrr_ridx = ieee80211_ratectl_rate(ni, NULL, 0);
fail: fail:

View File

@ -159,6 +159,7 @@ struct run_endpoint_queue {
struct run_softc { struct run_softc {
struct mtx sc_mtx; struct mtx sc_mtx;
struct ieee80211com sc_ic; struct ieee80211com sc_ic;
struct ieee80211_ratectl_tx_stats sc_txs;
struct mbufq sc_snd; struct mbufq sc_snd;
device_t sc_dev; device_t sc_dev;
struct usb_device *sc_udev; struct usb_device *sc_udev;

View File

@ -1254,8 +1254,10 @@ ural_tx_data(struct ural_softc *sc, struct mbuf *m0, struct ieee80211_node *ni)
rate = tp->mcastrate; rate = tp->mcastrate;
else if (tp->ucastrate != IEEE80211_FIXED_RATE_NONE) else if (tp->ucastrate != IEEE80211_FIXED_RATE_NONE)
rate = tp->ucastrate; rate = tp->ucastrate;
else else {
(void) ieee80211_ratectl_rate(ni, NULL, 0);
rate = ni->ni_txrate; rate = ni->ni_txrate;
}
if (wh->i_fc[1] & IEEE80211_FC1_PROTECTED) { if (wh->i_fc[1] & IEEE80211_FC1_PROTECTED) {
k = ieee80211_crypto_encap(ni, m0); k = ieee80211_crypto_encap(ni, m0);
@ -2208,32 +2210,29 @@ ural_ratectl_task(void *arg, int pending)
{ {
struct ural_vap *uvp = arg; struct ural_vap *uvp = arg;
struct ieee80211vap *vap = &uvp->vap; struct ieee80211vap *vap = &uvp->vap;
struct ieee80211com *ic = vap->iv_ic; struct ural_softc *sc = vap->iv_ic->ic_softc;
struct ural_softc *sc = ic->ic_softc; struct ieee80211_ratectl_tx_stats *txs = &sc->sc_txs;
struct ieee80211_node *ni; int fail;
int ok, fail;
int sum, retrycnt;
ni = ieee80211_ref_node(vap->iv_bss);
RAL_LOCK(sc); RAL_LOCK(sc);
/* read and clear statistic registers (STA_CSR0 to STA_CSR10) */ /* read and clear statistic registers (STA_CSR0 to STA_CSR10) */
ural_read_multi(sc, RAL_STA_CSR0, sc->sta, sizeof(sc->sta)); ural_read_multi(sc, RAL_STA_CSR0, sc->sta, sizeof(sc->sta));
ok = sc->sta[7] + /* TX ok w/o retry */ txs->flags = IEEE80211_RATECTL_TX_STATS_RETRIES;
txs->nsuccess = sc->sta[7] + /* TX ok w/o retry */
sc->sta[8]; /* TX ok w/ retry */ sc->sta[8]; /* TX ok w/ retry */
fail = sc->sta[9]; /* TX retry-fail count */ fail = sc->sta[9]; /* TX retry-fail count */
sum = ok+fail; txs->nframes = txs->nsuccess + fail;
retrycnt = sc->sta[8] + fail; /* XXX fail * maxretry */
txs->nretries = sc->sta[8] + fail;
ieee80211_ratectl_tx_update(vap, ni, &sum, &ok, &retrycnt); ieee80211_ratectl_tx_update(vap, txs);
(void) ieee80211_ratectl_rate(ni, NULL, 0);
/* count TX retry-fail as Tx errors */ /* count TX retry-fail as Tx errors */
if_inc_counter(ni->ni_vap->iv_ifp, IFCOUNTER_OERRORS, fail); if_inc_counter(vap->iv_ifp, IFCOUNTER_OERRORS, fail);
usb_callout_reset(&uvp->ratectl_ch, hz, ural_ratectl_timeout, uvp); usb_callout_reset(&uvp->ratectl_ch, hz, ural_ratectl_timeout, uvp);
RAL_UNLOCK(sc); RAL_UNLOCK(sc);
ieee80211_free_node(ni);
} }
static int static int

View File

@ -90,6 +90,7 @@ enum {
struct ural_softc { struct ural_softc {
struct ieee80211com sc_ic; struct ieee80211com sc_ic;
struct ieee80211_ratectl_tx_stats sc_txs;
struct mbufq sc_snd; struct mbufq sc_snd;
device_t sc_dev; device_t sc_dev;
struct usb_device *sc_udev; struct usb_device *sc_udev;

View File

@ -662,12 +662,24 @@ zyd_intr_read_callback(struct usb_xfer *xfer, usb_error_t error)
*/ */
ni = ieee80211_find_txnode(vap, retry->macaddr); ni = ieee80211_find_txnode(vap, retry->macaddr);
if (ni != NULL) { if (ni != NULL) {
struct ieee80211_ratectl_tx_status *txs =
&sc->sc_txs;
int retrycnt = int retrycnt =
(int)(le16toh(retry->count) & 0xff); (int)(le16toh(retry->count) & 0xff);
ieee80211_ratectl_tx_complete(vap, ni, txs->flags =
IEEE80211_RATECTL_TX_FAILURE, IEEE80211_RATECTL_STATUS_LONG_RETRY;
&retrycnt, NULL); txs->long_retries = retrycnt;
if (le16toh(retry->count) & 0x100) {
txs->status =
IEEE80211_RATECTL_TX_FAIL_LONG;
} else {
txs->status =
IEEE80211_RATECTL_TX_SUCCESS;
}
ieee80211_ratectl_tx_complete(ni, txs);
ieee80211_free_node(ni); ieee80211_free_node(ni);
} }
if (le16toh(retry->count) & 0x100) if (le16toh(retry->count) & 0x100)

View File

@ -1254,6 +1254,7 @@ enum {
struct zyd_softc { struct zyd_softc {
struct ieee80211com sc_ic; struct ieee80211com sc_ic;
struct ieee80211_ratectl_tx_status sc_txs;
struct mbufq sc_snd; struct mbufq sc_snd;
device_t sc_dev; device_t sc_dev;
struct usb_device *sc_udev; struct usb_device *sc_udev;

View File

@ -526,6 +526,10 @@ wpi_attach(device_t dev)
wpi_radiotap_attach(sc); wpi_radiotap_attach(sc);
/* Setup Tx status flags (constant). */
sc->sc_txs.flags = IEEE80211_RATECTL_STATUS_SHORT_RETRY |
IEEE80211_RATECTL_STATUS_LONG_RETRY;
callout_init_mtx(&sc->calib_to, &sc->rxon_mtx, 0); callout_init_mtx(&sc->calib_to, &sc->rxon_mtx, 0);
callout_init_mtx(&sc->scan_timeout, &sc->rxon_mtx, 0); callout_init_mtx(&sc->scan_timeout, &sc->rxon_mtx, 0);
callout_init_mtx(&sc->tx_timeout, &sc->txq_state_mtx, 0); callout_init_mtx(&sc->tx_timeout, &sc->txq_state_mtx, 0);
@ -2051,14 +2055,13 @@ wpi_rx_statistics(struct wpi_softc *sc, struct wpi_rx_desc *desc,
static void static void
wpi_tx_done(struct wpi_softc *sc, struct wpi_rx_desc *desc) wpi_tx_done(struct wpi_softc *sc, struct wpi_rx_desc *desc)
{ {
struct ieee80211_ratectl_tx_status *txs = &sc->sc_txs;
struct wpi_tx_ring *ring = &sc->txq[desc->qid & 0x3]; struct wpi_tx_ring *ring = &sc->txq[desc->qid & 0x3];
struct wpi_tx_data *data = &ring->data[desc->idx]; struct wpi_tx_data *data = &ring->data[desc->idx];
struct wpi_tx_stat *stat = (struct wpi_tx_stat *)(desc + 1); struct wpi_tx_stat *stat = (struct wpi_tx_stat *)(desc + 1);
struct mbuf *m; struct mbuf *m;
struct ieee80211_node *ni; struct ieee80211_node *ni;
struct ieee80211vap *vap;
uint32_t status = le32toh(stat->status); uint32_t status = le32toh(stat->status);
int ackfailcnt = stat->ackfailcnt / WPI_NTRIES_DEFAULT;
KASSERT(data->ni != NULL, ("no node")); KASSERT(data->ni != NULL, ("no node"));
KASSERT(data->m != NULL, ("no mbuf")); KASSERT(data->m != NULL, ("no mbuf"));
@ -2075,18 +2078,32 @@ wpi_tx_done(struct wpi_softc *sc, struct wpi_rx_desc *desc)
bus_dmamap_unload(ring->data_dmat, data->map); bus_dmamap_unload(ring->data_dmat, data->map);
m = data->m, data->m = NULL; m = data->m, data->m = NULL;
ni = data->ni, data->ni = NULL; ni = data->ni, data->ni = NULL;
vap = ni->ni_vap;
/* /*
* Update rate control statistics for the node. * Update rate control statistics for the node.
*/ */
if (status & WPI_TX_STATUS_FAIL) { txs->short_retries = stat->rtsfailcnt;
ieee80211_ratectl_tx_complete(vap, ni, txs->long_retries = stat->ackfailcnt / WPI_NTRIES_DEFAULT;
IEEE80211_RATECTL_TX_FAILURE, &ackfailcnt, NULL); if (!(status & WPI_TX_STATUS_FAIL))
} else txs->status = IEEE80211_RATECTL_TX_SUCCESS;
ieee80211_ratectl_tx_complete(vap, ni, else {
IEEE80211_RATECTL_TX_SUCCESS, &ackfailcnt, NULL); switch (status & 0xff) {
case WPI_TX_STATUS_FAIL_SHORT_LIMIT:
txs->status = IEEE80211_RATECTL_TX_FAIL_SHORT;
break;
case WPI_TX_STATUS_FAIL_LONG_LIMIT:
txs->status = IEEE80211_RATECTL_TX_FAIL_LONG;
break;
case WPI_TX_STATUS_FAIL_LIFE_EXPIRE:
txs->status = IEEE80211_RATECTL_TX_FAIL_EXPIRED;
break;
default:
txs->status = IEEE80211_RATECTL_TX_FAIL_UNSPECIFIED;
break;
}
}
ieee80211_ratectl_tx_complete(ni, txs);
ieee80211_tx_complete(ni, m, (status & WPI_TX_STATUS_FAIL) != 0); ieee80211_tx_complete(ni, m, (status & WPI_TX_STATUS_FAIL) != 0);
WPI_TXQ_STATE_LOCK(sc); WPI_TXQ_STATE_LOCK(sc);

View File

@ -170,6 +170,7 @@ struct wpi_softc {
struct mtx sc_mtx; struct mtx sc_mtx;
struct ieee80211com sc_ic; struct ieee80211com sc_ic;
struct ieee80211_ratectl_tx_status sc_txs;
struct mtx tx_mtx; struct mtx tx_mtx;

View File

@ -69,11 +69,11 @@ static void amrr_node_deinit(struct ieee80211_node *);
static int amrr_update(struct ieee80211_amrr *, static int amrr_update(struct ieee80211_amrr *,
struct ieee80211_amrr_node *, struct ieee80211_node *); struct ieee80211_amrr_node *, struct ieee80211_node *);
static int amrr_rate(struct ieee80211_node *, void *, uint32_t); static int amrr_rate(struct ieee80211_node *, void *, uint32_t);
static void amrr_tx_complete(const struct ieee80211vap *, static void amrr_tx_complete(const struct ieee80211_node *,
const struct ieee80211_node *, int, const struct ieee80211_ratectl_tx_status *);
void *, void *); static void amrr_tx_update_cb(void *, struct ieee80211_node *);
static void amrr_tx_update(const struct ieee80211vap *vap, static void amrr_tx_update(struct ieee80211vap *vap,
const struct ieee80211_node *, void *, void *, void *); struct ieee80211_ratectl_tx_stats *);
static void amrr_sysctlattach(struct ieee80211vap *, static void amrr_sysctlattach(struct ieee80211vap *,
struct sysctl_ctx_list *, struct sysctl_oid *); struct sysctl_ctx_list *, struct sysctl_oid *);
static void amrr_node_stats(struct ieee80211_node *ni, struct sbuf *s); static void amrr_node_stats(struct ieee80211_node *ni, struct sbuf *s);
@ -360,34 +360,56 @@ amrr_rate(struct ieee80211_node *ni, void *arg __unused, uint32_t iarg __unused)
* retransmissions (i.e. xmit attempts - 1). * retransmissions (i.e. xmit attempts - 1).
*/ */
static void static void
amrr_tx_complete(const struct ieee80211vap *vap, amrr_tx_complete(const struct ieee80211_node *ni,
const struct ieee80211_node *ni, int ok, const struct ieee80211_ratectl_tx_status *status)
void *arg1, void *arg2 __unused)
{ {
struct ieee80211_amrr_node *amn = ni->ni_rctls; struct ieee80211_amrr_node *amn = ni->ni_rctls;
int retries = *(int *)arg1; int retries;
retries = 0;
if (status->flags & IEEE80211_RATECTL_STATUS_LONG_RETRY)
retries = status->long_retries;
amn->amn_txcnt++; amn->amn_txcnt++;
if (ok) if (status->status == IEEE80211_RATECTL_TX_SUCCESS)
amn->amn_success++; amn->amn_success++;
amn->amn_retrycnt += retries; amn->amn_retrycnt += retries;
} }
static void
amrr_tx_update_cb(void *arg, struct ieee80211_node *ni)
{
struct ieee80211_ratectl_tx_stats *stats = arg;
struct ieee80211_amrr_node *amn = ni->ni_rctls;
int txcnt, success, retrycnt;
txcnt = stats->nframes;
success = stats->nsuccess;
retrycnt = 0;
if (stats->flags & IEEE80211_RATECTL_TX_STATS_RETRIES)
retrycnt = stats->nretries;
amn->amn_txcnt += txcnt;
amn->amn_success += success;
amn->amn_retrycnt += retrycnt;
}
/* /*
* Set tx count/retry statistics explicitly. Intended for * Set tx count/retry statistics explicitly. Intended for
* drivers that poll the device for statistics maintained * drivers that poll the device for statistics maintained
* in the device. * in the device.
*/ */
static void static void
amrr_tx_update(const struct ieee80211vap *vap, const struct ieee80211_node *ni, amrr_tx_update(struct ieee80211vap *vap,
void *arg1, void *arg2, void *arg3) struct ieee80211_ratectl_tx_stats *stats)
{ {
struct ieee80211_amrr_node *amn = ni->ni_rctls;
int txcnt = *(int *)arg1, success = *(int *)arg2, retrycnt = *(int *)arg3;
amn->amn_txcnt = txcnt; if (stats->flags & IEEE80211_RATECTL_TX_STATS_NODE)
amn->amn_success = success; amrr_tx_update_cb(stats, stats->ni);
amn->amn_retrycnt = retrycnt; else {
ieee80211_iterate_nodes_vap(&vap->iv_ic->ic_sta, vap,
amrr_tx_update_cb, stats);
}
} }
static int static int

View File

@ -34,8 +34,49 @@ enum ieee80211_ratealgs {
IEEE80211_RATECTL_MAX IEEE80211_RATECTL_MAX
}; };
#define IEEE80211_RATECTL_TX_SUCCESS 1 /* used fields for tx_complete() events */
#define IEEE80211_RATECTL_TX_FAILURE 0 #define IEEE80211_RATECTL_STATUS_PKTLEN 0x00000001
#define IEEE80211_RATECTL_STATUS_FINAL_RATE 0x00000002
#define IEEE80211_RATECTL_STATUS_SHORT_RETRY 0x00000004
#define IEEE80211_RATECTL_STATUS_LONG_RETRY 0x00000008
#define IEEE80211_RATECTL_STATUS_RSSI 0x00000010
/* failure reason */
enum ieee80211_ratectl_tx_fail_reason {
IEEE80211_RATECTL_TX_SUCCESS = 0,
IEEE80211_RATECTL_TX_FAIL_SHORT = 1, /* too many RTS retries */
IEEE80211_RATECTL_TX_FAIL_LONG = 2, /* too many retries */
IEEE80211_RATECTL_TX_FAIL_EXPIRED = 3, /* lifetime expired */
IEEE80211_RATECTL_TX_FAIL_UNSPECIFIED = 4, /* another reason */
};
#define IEEE80211_RATECTL_TX_FAIL_MAX \
(IEEE80211_RATECTL_TX_FAIL_UNSPECIFIED + 1)
struct ieee80211_ratectl_tx_status {
uint32_t flags; /* mark used fields */
enum ieee80211_ratectl_tx_fail_reason status; /* Tx status */
int pktlen; /* frame length */
int final_rate; /* transmission rate */
uint_fast8_t short_retries; /* RTS/CTS retries */
uint_fast8_t long_retries; /* ACK retries */
int8_t rssi; /* ACK RSSI */
uint8_t spare[15]; /* for future use */
};
/* used fields for tx_update() events */
#define IEEE80211_RATECTL_TX_STATS_NODE 0x00000001
#define IEEE80211_RATECTL_TX_STATS_RETRIES 0x00000002
struct ieee80211_ratectl_tx_stats {
uint32_t flags; /* mark used fields */
struct ieee80211_node *ni; /* receiver */
int nframes; /* transmitted frames */
int nsuccess; /* ACKed frames */
int nretries; /* number of retries */
};
struct ieee80211_ratectl { struct ieee80211_ratectl {
const char *ir_name; const char *ir_name;
@ -46,12 +87,10 @@ struct ieee80211_ratectl {
void (*ir_node_init)(struct ieee80211_node *); void (*ir_node_init)(struct ieee80211_node *);
void (*ir_node_deinit)(struct ieee80211_node *); void (*ir_node_deinit)(struct ieee80211_node *);
int (*ir_rate)(struct ieee80211_node *, void *, uint32_t); int (*ir_rate)(struct ieee80211_node *, void *, uint32_t);
void (*ir_tx_complete)(const struct ieee80211vap *, void (*ir_tx_complete)(const struct ieee80211_node *,
const struct ieee80211_node *, int, const struct ieee80211_ratectl_tx_status *);
void *, void *); void (*ir_tx_update)(struct ieee80211vap *,
void (*ir_tx_update)(const struct ieee80211vap *, struct ieee80211_ratectl_tx_stats *);
const struct ieee80211_node *,
void *, void *, void *);
void (*ir_setinterval)(const struct ieee80211vap *, int); void (*ir_setinterval)(const struct ieee80211vap *, int);
void (*ir_node_stats)(struct ieee80211_node *ni, struct sbuf *s); void (*ir_node_stats)(struct ieee80211_node *ni, struct sbuf *s);
}; };
@ -94,19 +133,21 @@ ieee80211_ratectl_rate(struct ieee80211_node *ni, void *arg, uint32_t iarg)
} }
static __inline void static __inline void
ieee80211_ratectl_tx_complete(const struct ieee80211vap *vap, ieee80211_ratectl_tx_complete(const struct ieee80211_node *ni,
const struct ieee80211_node *ni, int status, void *arg1, void *arg2) const struct ieee80211_ratectl_tx_status *status)
{ {
vap->iv_rate->ir_tx_complete(vap, ni, status, arg1, arg2); const struct ieee80211vap *vap = ni->ni_vap;
vap->iv_rate->ir_tx_complete(ni, status);
} }
static __inline void static __inline void
ieee80211_ratectl_tx_update(const struct ieee80211vap *vap, ieee80211_ratectl_tx_update(struct ieee80211vap *vap,
const struct ieee80211_node *ni, void *arg1, void *arg2, void *arg3) struct ieee80211_ratectl_tx_stats *stats)
{ {
if (vap->iv_rate->ir_tx_update == NULL) if (vap->iv_rate->ir_tx_update == NULL)
return; return;
vap->iv_rate->ir_tx_update(vap, ni, arg1, arg2, arg3); vap->iv_rate->ir_tx_update(vap, stats);
} }
static __inline void static __inline void

View File

@ -80,15 +80,14 @@ none_rate(struct ieee80211_node *ni, void *arg __unused, uint32_t iarg __unused)
} }
static void static void
none_tx_complete(const struct ieee80211vap *vap, none_tx_complete(const struct ieee80211_node *ni,
const struct ieee80211_node *ni, int ok, const struct ieee80211_ratectl_tx_status *status)
void *arg1, void *arg2 __unused)
{ {
} }
static void static void
none_tx_update(const struct ieee80211vap *vap, const struct ieee80211_node *ni, none_tx_update(struct ieee80211vap *vap,
void *arg1, void *arg2, void *arg3) struct ieee80211_ratectl_tx_stats *stats)
{ {
} }

View File

@ -87,9 +87,8 @@ static int rssadapt_rate(struct ieee80211_node *, void *, uint32_t);
static void rssadapt_lower_rate(struct ieee80211_rssadapt_node *, int, int); static void rssadapt_lower_rate(struct ieee80211_rssadapt_node *, int, int);
static void rssadapt_raise_rate(struct ieee80211_rssadapt_node *, static void rssadapt_raise_rate(struct ieee80211_rssadapt_node *,
int, int); int, int);
static void rssadapt_tx_complete(const struct ieee80211vap *, static void rssadapt_tx_complete(const struct ieee80211_node *,
const struct ieee80211_node *, int, const struct ieee80211_ratectl_tx_status *);
void *, void *);
static void rssadapt_sysctlattach(struct ieee80211vap *, static void rssadapt_sysctlattach(struct ieee80211vap *,
struct sysctl_ctx_list *, struct sysctl_oid *); struct sysctl_ctx_list *, struct sysctl_oid *);
@ -310,13 +309,21 @@ rssadapt_raise_rate(struct ieee80211_rssadapt_node *ra, int pktlen, int rssi)
} }
static void static void
rssadapt_tx_complete(const struct ieee80211vap *vap, rssadapt_tx_complete(const struct ieee80211_node *ni,
const struct ieee80211_node *ni, int success, void *arg1, void *arg2) const struct ieee80211_ratectl_tx_status *status)
{ {
struct ieee80211_rssadapt_node *ra = ni->ni_rctls; struct ieee80211_rssadapt_node *ra = ni->ni_rctls;
int pktlen = *(int *)arg1, rssi = *(int *)arg2; int pktlen, rssi;
if (success) { if ((status->flags &
(IEEE80211_RATECTL_STATUS_PKTLEN|IEEE80211_RATECTL_STATUS_RSSI)) !=
(IEEE80211_RATECTL_STATUS_PKTLEN|IEEE80211_RATECTL_STATUS_RSSI))
return;
pktlen = status->pktlen;
rssi = status->rssi;
if (status->status == IEEE80211_RATECTL_TX_SUCCESS) {
ra->ra_nok++; ra->ra_nok++;
if ((ra->ra_rix + 1) < ra->ra_rates.rs_nrates && if ((ra->ra_rix + 1) < ra->ra_rates.rs_nrates &&
(ticks - ra->ra_last_raise) >= ra->ra_raise_interval) (ticks - ra->ra_last_raise) >= ra->ra_raise_interval)