summaryrefslogtreecommitdiff
path: root/sys/dev/ic/rtw.c
diff options
context:
space:
mode:
Diffstat (limited to 'sys/dev/ic/rtw.c')
-rw-r--r--sys/dev/ic/rtw.c416
1 files changed, 295 insertions, 121 deletions
diff --git a/sys/dev/ic/rtw.c b/sys/dev/ic/rtw.c
index 8ac6208f753..90736291103 100644
--- a/sys/dev/ic/rtw.c
+++ b/sys/dev/ic/rtw.c
@@ -1,4 +1,4 @@
-/* $OpenBSD: rtw.c,v 1.50 2005/11/20 09:49:05 jsg Exp $ */
+/* $OpenBSD: rtw.c,v 1.51 2006/01/05 05:36:05 jsg Exp $ */
/* $NetBSD: rtw.c,v 1.29 2004/12/27 19:49:16 dyoung Exp $ */
/*-
@@ -79,6 +79,8 @@
int rtw_rfprog_fallback = 0;
int rtw_host_rfio = 0;
+int rtw_xmtr_restart = 0;
+int rtw_do_chip_reset = 0;
int rtw_dwelltime = 200; /* milliseconds per channel */
int rtw_macbangbits_timeout = 100;
@@ -90,9 +92,13 @@ int rtw_rxbufs_limit = RTW_RXQLEN;
void rtw_start(struct ifnet *);
void rtw_txdesc_blk_init_all(struct rtw_txdesc_blk *);
void rtw_txsoft_blk_init_all(struct rtw_txsoft_blk *);
+void rtw_txdesc_blk_init(struct rtw_txdesc_blk *);
void rtw_txdescs_sync(struct rtw_txdesc_blk *, u_int, u_int, int);
+u_int rtw_txring_next(struct rtw_regs *, struct rtw_txdesc_blk *);
+void rtw_txring_fixup(struct rtw_softc *);
void rtw_rxbufs_release(bus_dma_tag_t, struct rtw_rxsoft *);
void rtw_rxdesc_init(struct rtw_rxdesc_blk *, struct rtw_rxsoft *, int, int);
+void rtw_rxring_fixup(struct rtw_softc *);
void rtw_io_enable(struct rtw_regs *, u_int8_t, int);
void rtw_intr_rx(struct rtw_softc *, u_int16_t);
void rtw_intr_beacon(struct rtw_softc *, u_int16_t);
@@ -157,20 +163,21 @@ int rtw_chip_reset1(struct rtw_regs *, const char *);
int rtw_chip_reset(struct rtw_regs *, const char *);
int rtw_recall_eeprom(struct rtw_regs *, const char *);
int rtw_reset(struct rtw_softc *);
+void rtw_reset_oactive(struct rtw_softc *);
int rtw_txdesc_dmamaps_create(bus_dma_tag_t, struct rtw_txsoft *, u_int);
int rtw_rxdesc_dmamaps_create(bus_dma_tag_t, struct rtw_rxsoft *, u_int);
void rtw_rxdesc_dmamaps_destroy(bus_dma_tag_t, struct rtw_rxsoft *, u_int);
void rtw_txdesc_dmamaps_destroy(bus_dma_tag_t, struct rtw_txsoft *, u_int);
void rtw_init_channels(enum rtw_locale, struct ieee80211_channel (*)[],
const char*);
-void rtw_identify_country(struct rtw_regs *, enum rtw_locale *, const char *);
+void rtw_identify_country(struct rtw_regs *, enum rtw_locale *);
int rtw_identify_sta(struct rtw_regs *, u_int8_t (*)[], const char *);
void rtw_rxdescs_sync(struct rtw_rxdesc_blk *, int, int, int);
int rtw_rxsoft_alloc(bus_dma_tag_t, struct rtw_rxsoft *);
void rtw_collect_txpkt(struct rtw_softc *, struct rtw_txdesc_blk *,
struct rtw_txsoft *, int);
void rtw_collect_txring(struct rtw_softc *, struct rtw_txsoft_blk *,
- struct rtw_txdesc_blk *);
+ struct rtw_txdesc_blk *, int);
void rtw_suspend_ticks(struct rtw_softc *);
void rtw_resume_ticks(struct rtw_softc *);
void rtw_enable_interrupts(struct rtw_softc *);
@@ -184,12 +191,14 @@ void rtw_rxdesc_init_all(struct rtw_rxdesc_blk *, struct rtw_rxsoft *,
int);
int rtw_txring_choose(struct rtw_softc *, struct rtw_txsoft_blk **,
struct rtw_txdesc_blk **, int);
+u_int rtw_txring_next(struct rtw_regs *, struct rtw_txdesc_blk *);
struct mbuf *rtw_80211_dequeue(struct rtw_softc *, struct ifqueue *, int,
struct rtw_txsoft_blk **, struct rtw_txdesc_blk **,
struct ieee80211_node **, short *);
uint64_t rtw_tsf_extend(struct rtw_regs *, u_int32_t);
void rtw_ibss_merge(struct rtw_softc *, struct ieee80211_node *,
u_int32_t);
+void rtw_idle(struct rtw_regs *);
void rtw_led_attach(struct rtw_led_state *, void *);
void rtw_led_init(struct rtw_regs *);
void rtw_led_slowblink(void *);
@@ -796,8 +805,7 @@ rtw_init_channels(enum rtw_locale locale,
}
void
-rtw_identify_country(struct rtw_regs *regs, enum rtw_locale *locale,
- const char *dvname)
+rtw_identify_country(struct rtw_regs *regs, enum rtw_locale *locale)
{
u_int8_t cfg0 = RTW_READ8(regs, RTW_CONFIG0);
@@ -862,11 +870,32 @@ void
rtw_txdesc_blk_init_all(struct rtw_txdesc_blk *tdb)
{
int pri;
- u_int ndesc[RTW_NTXPRI] =
- {RTW_NTXDESCLO, RTW_NTXDESCMD, RTW_NTXDESCHI, RTW_NTXDESCBCN};
+ /* nfree: the number of free descriptors in each ring.
+ * The beacon ring is a special case: I do not let the
+ * driver use all of the descriptors on the beacon ring.
+ * The reasons are two-fold:
+ *
+ * (1) A BEACON descriptor's OWN bit is (apparently) not
+ * updated, so the driver cannot easily know if the descriptor
+ * belongs to it, or if it is racing the NIC. If the NIC
+ * does not OWN every descriptor, then the driver can safely
+ * update the descriptors when RTW_TBDA points at tdb_next.
+ *
+ * (2) I hope that the NIC will process more than one BEACON
+ * descriptor in a single beacon interval, since that will
+ * enable multiple-BSS support. Since the NIC does not
+ * clear the OWN bit, there is no natural place for it to
+ * stop processing BEACON desciptors. Maybe it will *not*
+ * stop processing them! I do not want to chance the NIC
+ * looping around and around a saturated beacon ring, so
+ * I will leave one descriptor unOWNed at all times.
+ */
+ u_int nfree[RTW_NTXPRI] =
+ {RTW_NTXDESCLO, RTW_NTXDESCMD, RTW_NTXDESCHI,
+ RTW_NTXDESCBCN - 1};
for (pri = 0; pri < RTW_NTXPRI; pri++) {
- tdb[pri].tdb_nfree = ndesc[pri];
+ tdb[pri].tdb_nfree = nfree[pri];
tdb[pri].tdb_next = 0;
}
}
@@ -884,6 +913,7 @@ rtw_txsoft_blk_init(struct rtw_txsoft_blk *tsb)
ts->ts_mbuf = NULL;
SIMPLEQ_INSERT_TAIL(&tsb->tsb_freeq, ts, ts_q);
}
+ tsb->tsb_tx_timer = 0;
return 0;
}
@@ -1066,7 +1096,6 @@ rtw_rxdesc_init_all(struct rtw_rxdesc_blk *rdb, struct rtw_rxsoft *ctl,
rs = &ctl[i];
rtw_rxdesc_init(rdb, rs, i, kick);
}
- rdb->rdb_next = 0;
}
void
@@ -1354,6 +1383,7 @@ rtw_txsofts_release(bus_dma_tag_t dmat, struct ieee80211com *ic,
SIMPLEQ_REMOVE_HEAD(&tsb->tsb_dirtyq, ts_q);
SIMPLEQ_INSERT_TAIL(&tsb->tsb_freeq, ts, ts_q);
}
+ tsb->tsb_tx_timer = 0;
}
void
@@ -1390,10 +1420,30 @@ rtw_collect_txpkt(struct rtw_softc *sc, struct rtw_txdesc_blk *tdb,
condstring, rts_retry, data_retry));
}
+void
+rtw_reset_oactive(struct rtw_softc *sc)
+{
+ short oflags;
+ int pri;
+ struct rtw_txsoft_blk *tsb;
+ struct rtw_txdesc_blk *tdb;
+ oflags = sc->sc_if.if_flags;
+ for (pri = 0; pri < RTW_NTXPRI; pri++) {
+ tsb = &sc->sc_txsoft_blk[pri];
+ tdb = &sc->sc_txdesc_blk[pri];
+ if (!SIMPLEQ_EMPTY(&tsb->tsb_freeq) && tdb->tdb_nfree > 0)
+ sc->sc_if.if_flags &= ~IFF_OACTIVE;
+ }
+ if (oflags != sc->sc_if.if_flags) {
+ DPRINTF(sc, RTW_DEBUG_OACTIVE,
+ ("%s: reset OACTIVE\n", __func__));
+ }
+}
+
/* Collect transmitted packets. */
void
rtw_collect_txring(struct rtw_softc *sc, struct rtw_txsoft_blk *tsb,
- struct rtw_txdesc_blk *tdb)
+ struct rtw_txdesc_blk *tdb, int force)
{
int ndesc;
struct rtw_txsoft *ts;
@@ -1408,22 +1458,28 @@ rtw_collect_txring(struct rtw_softc *sc, struct rtw_txsoft_blk *tsb,
rtw_txdescs_sync(tdb, ts->ts_first, ndesc,
BUS_DMASYNC_POSTREAD|BUS_DMASYNC_POSTWRITE);
- if ((tdb->tdb_desc[ts->ts_last].td_stat &
+ if (force) {
+ int i;
+ for (i = ts->ts_first; ; i = RTW_NEXT_IDX(tdb, i)) {
+ tdb->tdb_desc[i].td_stat &=
+ ~htole32(RTW_TXSTAT_OWN);
+ if (i == ts->ts_last)
+ break;
+ }
+ rtw_txdescs_sync(tdb, ts->ts_first, ndesc,
+ BUS_DMASYNC_PREREAD|BUS_DMASYNC_PREWRITE);
+ } else if ((tdb->tdb_desc[ts->ts_last].td_stat &
htole32(RTW_TXSTAT_OWN)) != 0)
break;
- if (&sc->sc_txdesc_blk[RTW_TXPRIBCN] == tdb) {
- RTW_DPRINTF(RTW_DEBUG_BEACON,
- ("%s: collected beacon\n", __func__));
- }
-
rtw_collect_txpkt(sc, tdb, ts, ndesc);
SIMPLEQ_REMOVE_HEAD(&tsb->tsb_dirtyq, ts_q);
SIMPLEQ_INSERT_TAIL(&tsb->tsb_freeq, ts, ts_q);
- sc->sc_if.if_flags &= ~IFF_OACTIVE;
}
+ /* no more pending transmissions, cancel watchdog */
if (ts == NULL)
tsb->tsb_tx_timer = 0;
+ rtw_reset_oactive(sc);
}
void
@@ -1437,21 +1493,63 @@ rtw_intr_tx(struct rtw_softc *sc, u_int16_t isr)
tsb = &sc->sc_txsoft_blk[pri];
tdb = &sc->sc_txdesc_blk[pri];
- rtw_collect_txring(sc, tsb, tdb);
+ rtw_collect_txring(sc, tsb, tdb, 0);
- if ((isr & RTW_INTR_TX) != 0)
- rtw_start(&sc->sc_if);
}
- /* TBD */
- return;
+ if ((isr & RTW_INTR_TX) != 0)
+ rtw_start(&sc->sc_if);
}
void
rtw_intr_beacon(struct rtw_softc *sc, u_int16_t isr)
{
- /* TBD */
- return;
+ u_int next;
+ uint32_t tsfth, tsftl;
+ struct ieee80211com *ic;
+ struct rtw_txdesc_blk *tdb = &sc->sc_txdesc_blk[RTW_TXPRIBCN];
+ struct rtw_txsoft_blk *tsb = &sc->sc_txsoft_blk[RTW_TXPRIBCN];
+ struct mbuf *m;
+
+ tsfth = RTW_READ(&sc->sc_regs, RTW_TSFTRH);
+ tsftl = RTW_READ(&sc->sc_regs, RTW_TSFTRL);
+
+ if ((isr & (RTW_INTR_TBDOK|RTW_INTR_TBDER)) != 0) {
+ next = rtw_txring_next(&sc->sc_regs, tdb);
+ RTW_DPRINTF(RTW_DEBUG_BEACON,
+ ("%s: beacon ring %sprocessed, isr = %#04hx"
+ ", next %u expected %u, %llu\n", __func__,
+ (next == tdb->tdb_next) ? "" : "un", isr, next,
+ tdb->tdb_next, (uint64_t)tsfth << 32 | tsftl));
+ if ((RTW_READ8(&sc->sc_regs, RTW_TPPOLL) & RTW_TPPOLL_BQ) == 0){
+ rtw_collect_txring(sc, tsb, tdb, 1);
+ tdb->tdb_next = 0;
+ }
+ }
+ /* Start beacon transmission. */
+
+ if ((isr & RTW_INTR_BCNINT) != 0 &&
+ sc->sc_ic.ic_state == IEEE80211_S_RUN &&
+ SIMPLEQ_EMPTY(&tsb->tsb_dirtyq)) {
+ RTW_DPRINTF(RTW_DEBUG_BEACON,
+ ("%s: beacon prep. time, isr = %#04hx"
+ ", %16llu\n", __func__, isr,
+ (uint64_t)tsfth << 32 | tsftl));
+ ic = &sc->sc_ic;
+ if ((m = ieee80211_beacon_alloc(ic, ic->ic_bss)) != NULL) {
+ RTW_DPRINTF(RTW_DEBUG_BEACON,
+ ("%s: m %p len %u\n", __func__, m, m->m_len));
+ }
+
+ if (m == NULL) {
+ printf("%s: could not allocate beacon\n",
+ sc->sc_dev.dv_xname);
+ return;
+ }
+ m->m_pkthdr.rcvif = (void *)ieee80211_ref_node(ic->ic_bss);
+ IF_ENQUEUE(&sc->sc_beaconq, m);
+ rtw_start(&sc->sc_if);
+ }
}
void
@@ -1497,30 +1595,34 @@ rtw_dump_rings(struct rtw_softc *sc)
void
rtw_hwring_setup(struct rtw_softc *sc)
{
+ int pri;
struct rtw_regs *regs = &sc->sc_regs;
+ struct rtw_txdesc_blk *tdb;
+
+ sc->sc_txdesc_blk[RTW_TXPRILO].tdb_basereg = RTW_TLPDA;
+ sc->sc_txdesc_blk[RTW_TXPRILO].tdb_base = RTW_RING_BASE(sc, hd_txlo);
+ sc->sc_txdesc_blk[RTW_TXPRIMD].tdb_basereg = RTW_TNPDA;
+ sc->sc_txdesc_blk[RTW_TXPRIMD].tdb_base = RTW_RING_BASE(sc, hd_txmd);
+ sc->sc_txdesc_blk[RTW_TXPRIHI].tdb_basereg = RTW_THPDA;
+ sc->sc_txdesc_blk[RTW_TXPRIHI].tdb_base = RTW_RING_BASE(sc, hd_txhi);
+ sc->sc_txdesc_blk[RTW_TXPRIBCN].tdb_basereg = RTW_TBDA;
+ sc->sc_txdesc_blk[RTW_TXPRIBCN].tdb_base = RTW_RING_BASE(sc, hd_bcn);
+
+ for (pri = 0; pri < RTW_NTXPRI; pri++) {
+ tdb = &sc->sc_txdesc_blk[pri];
+ RTW_WRITE(regs, tdb->tdb_basereg, tdb->tdb_base);
+ RTW_DPRINTF(RTW_DEBUG_XMIT_DESC,
+ ("%s: reg[tdb->tdb_basereg] <- %lx\n", __func__,
+ (u_int *)tdb->tdb_base));
+ }
+
RTW_WRITE(regs, RTW_RDSAR, RTW_RING_BASE(sc, hd_rx));
- RTW_WRITE(regs, RTW_TLPDA, RTW_RING_BASE(sc, hd_txlo));
- RTW_WRITE(regs, RTW_TNPDA, RTW_RING_BASE(sc, hd_txmd));
- RTW_WRITE(regs, RTW_THPDA, RTW_RING_BASE(sc, hd_txhi));
- RTW_WRITE(regs, RTW_TBDA, RTW_RING_BASE(sc, hd_bcn));
- RTW_SYNC(regs, RTW_TLPDA, RTW_RDSAR);
-#if 0
- RTW_DPRINTF(RTW_DEBUG_XMIT_DESC,
- ("%s: reg[TLPDA] <- %\n", __func__,
- (u_int *)RTW_RING_BASE(sc, hd_txlo)));
- RTW_DPRINTF(RTW_DEBUG_XMIT_DESC,
- ("%s: reg[TNPDA] <- %\n", __func__,
- (u_int *)RTW_RING_BASE(sc, hd_txmd)));
- RTW_DPRINTF(RTW_DEBUG_XMIT_DESC,
- ("%s: reg[THPDA] <- %\n", __func__,
- (u_int *)RTW_RING_BASE(sc, hd_txhi)));
- RTW_DPRINTF(RTW_DEBUG_XMIT_DESC,
- ("%s: reg[TBDA] <- %\n", __func__,
- (u_int *)RTW_RING_BASE(sc, hd_bcn)));
+
RTW_DPRINTF(RTW_DEBUG_RECV_DESC,
- ("%s: reg[RDSAR] <- %\n", __func__,
+ ("%s: reg[RDSAR] <- %lx\n", __func__,
(u_int *)RTW_RING_BASE(sc, hd_rx)));
-#endif
+
+ RTW_SYNC(regs, RTW_TLPDA, RTW_RDSAR);
}
int
@@ -1547,6 +1649,7 @@ rtw_swring_setup(struct rtw_softc *sc)
rtw_rxdescs_sync(rdb, 0, rdb->rdb_ndesc,
BUS_DMASYNC_POSTREAD|BUS_DMASYNC_POSTWRITE);
rtw_rxdesc_init_all(rdb, sc->sc_rxsoft, 1);
+ rdb->rdb_next = 0;
tdb = &sc->sc_txdesc_blk[0];
for (pri = 0; pri < RTW_NTXPRI; pri++) {
@@ -1557,75 +1660,118 @@ rtw_swring_setup(struct rtw_softc *sc)
}
void
-rtw_txdesc_blk_reset(struct rtw_txdesc_blk *tdb)
+rtw_txdesc_blk_init(struct rtw_txdesc_blk *tdb)
{
int i;
bzero(tdb->tdb_desc, sizeof(tdb->tdb_desc[0]) * tdb->tdb_ndesc);
for (i = 0; i < tdb->tdb_ndesc; i++)
tdb->tdb_desc[i].td_next = htole32(RTW_NEXT_DESC(tdb, i));
- tdb->tdb_nfree = tdb->tdb_ndesc;
- tdb->tdb_next = 0;
+}
+
+u_int
+rtw_txring_next(struct rtw_regs *regs, struct rtw_txdesc_blk *tdb)
+{
+ return (letoh32(RTW_READ(regs, tdb->tdb_basereg)) - tdb->tdb_base) /
+ sizeof(struct rtw_txdesc);
}
void
-rtw_txdescs_reset(struct rtw_softc *sc)
+rtw_txring_fixup(struct rtw_softc *sc)
{
int pri;
+ u_int next;
struct rtw_txdesc_blk *tdb;
+ struct rtw_regs *regs = &sc->sc_regs;
for (pri = 0; pri < RTW_NTXPRI; pri++) {
tdb = &sc->sc_txdesc_blk[pri];
- rtw_txsofts_release(sc->sc_dmat, &sc->sc_ic,
- &sc->sc_txsoft_blk[pri]);
- rtw_txdesc_blk_reset(tdb);
- rtw_txdescs_sync(tdb, 0, tdb->tdb_ndesc,
- BUS_DMASYNC_PREWRITE|BUS_DMASYNC_PREREAD);
+ next = rtw_txring_next(regs, tdb);
+ if (tdb->tdb_next == next)
+ continue;
+ RTW_DPRINTF(RTW_DEBUG_BUGS,
+ ("%s: tx-ring %d expected next %u, read %u\n", __func__,
+ pri, tdb->tdb_next, next));
+ tdb->tdb_next = next;
}
}
void
-rtw_rxdescs_reset(struct rtw_softc *sc)
+rtw_rxring_fixup(struct rtw_softc *sc)
{
- rtw_rxdesc_init_all(&sc->sc_rxdesc_blk, &sc->sc_rxsoft[0], 1);
+ u_int next;
+ uint32_t rdsar;
+ struct rtw_rxdesc_blk *rdb;
+
+ rdsar = letoh32(RTW_READ(&sc->sc_regs, RTW_RDSAR));
+ next = (rdsar - RTW_RING_BASE(sc, hd_rx)) / sizeof(struct rtw_rxdesc);
+
+ rdb = &sc->sc_rxdesc_blk;
+ if (rdb->rdb_next != next) {
+ RTW_DPRINTF(RTW_DEBUG_BUGS,
+ ("%s: rx-ring expected next %u, read %u\n", __func__,
+ rdb->rdb_next, next));
+ rdb->rdb_next = next;
+ }
+}
+
+void
+rtw_txdescs_reset(struct rtw_softc *sc)
+{
+ int pri;
+
+ for (pri = 0; pri < RTW_NTXPRI; pri++) {
+ rtw_collect_txring(sc, &sc->sc_txsoft_blk[pri],
+ &sc->sc_txdesc_blk[pri], 1);
+ }
}
void
rtw_intr_ioerror(struct rtw_softc *sc, u_int16_t isr)
{
+ uint8_t cr = 0;
+ int xmtr = 0, rcvr = 0;
struct rtw_regs *regs = &sc->sc_regs;
-#ifdef RTW_DEBUG
- if ((isr & RTW_INTR_TXFOVW) != 0)
- printf("%s: tx fifo overflow\n", sc->sc_dev.dv_xname);
-#endif
+ if ((isr & RTW_INTR_TXFOVW) != 0) {
+ RTW_DPRINTF(RTW_DEBUG_BUGS,
+ ("%s: tx fifo underflow\n", sc->sc_dev.dv_xname));
+ xmtr = 1;
+ cr |= RTW_CR_TE;
+ }
- if ((isr & (RTW_INTR_RDU|RTW_INTR_RXFOVW)) == 0)
+ if ((isr & (RTW_INTR_RDU|RTW_INTR_RXFOVW)) != 0) {
+ cr = RTW_CR_RE;
+ rcvr = 1;
return;
+ }
- RTW_DPRINTF(RTW_DEBUG_BUGS, ("%s: restarting xmit/recv, isr %"
+ RTW_DPRINTF(RTW_DEBUG_BUGS, ("%s: restarting xmit/recv, isr %hx"
"\n", sc->sc_dev.dv_xname, isr));
#ifdef RTW_DEBUG
rtw_dump_rings(sc);
#endif /* RTW_DEBUG */
- rtw_io_enable(regs, RTW_CR_RE | RTW_CR_TE, 0);
+ rtw_io_enable(regs, cr, 0);
- /* Collect rx'd packets. Refresh rx buffers. */
- rtw_intr_rx(sc, 0);
- /* Collect tx'd packets. */
- rtw_intr_tx(sc, 0);
+ /* Collect rx'd packets. Refresh rx buffers. */
+ if (rcvr)
+ rtw_intr_rx(sc, 0);
+ /* Collect tx'd packets. XXX let's hope this stops the transmit
+ * timeouts.
+ */
+ if (xmtr)
+ rtw_txdescs_reset(sc);
RTW_WRITE16(regs, RTW_IMR, 0);
RTW_SYNC(regs, RTW_IMR, RTW_IMR);
- rtw_chip_reset1(regs, sc->sc_dev.dv_xname);
-
- rtw_rxdescs_reset(sc);
- rtw_txdescs_reset(sc);
+ if (rtw_do_chip_reset) {
+ rtw_chip_reset1(regs, sc->sc_dev.dv_xname);
+ }
- rtw_hwring_setup(sc);
+ rtw_rxdesc_init_all(&sc->sc_rxdesc_blk, &sc->sc_rxsoft[0], 1);
#ifdef RTW_DEBUG
rtw_dump_rings(sc);
@@ -1633,8 +1779,11 @@ rtw_intr_ioerror(struct rtw_softc *sc, u_int16_t isr)
RTW_WRITE16(regs, RTW_IMR, sc->sc_inten);
RTW_SYNC(regs, RTW_IMR, RTW_IMR);
- rtw_io_enable(regs, RTW_CR_RE | RTW_CR_TE, 1);
- sc->sc_if.if_flags &= ~IFF_OACTIVE;
+ if (rcvr)
+ rtw_rxring_fixup(sc);
+ rtw_io_enable(regs, cr, 1);
+ if (xmtr)
+ rtw_txring_fixup(sc);
}
void
@@ -2568,6 +2717,8 @@ rtw_txring_choose(struct rtw_softc *sc, struct rtw_txsoft_blk **tsbp,
tdb = &sc->sc_txdesc_blk[pri];
if (SIMPLEQ_EMPTY(&tsb->tsb_freeq) || tdb->tdb_nfree == 0) {
+ if (tsb->tsb_tx_timer == 0)
+ tsb->tsb_tx_timer = 5;
*tsbp = NULL;
*tdbp = NULL;
return -1;
@@ -2587,7 +2738,10 @@ rtw_80211_dequeue(struct rtw_softc *sc, struct ifqueue *ifq, int pri,
if (IF_IS_EMPTY(ifq))
return NULL;
if (rtw_txring_choose(sc, tsbp, tdbp, pri) == -1) {
+ DPRINTF(sc, RTW_DEBUG_XMIT_RSRC, ("%s: no ring %d descriptor\n",
+ __func__, pri));
*if_flagsp |= IFF_OACTIVE;
+ sc->sc_if.if_timer = 1;
return NULL;
}
IF_DEQUEUE(ifq, m);
@@ -2642,14 +2796,27 @@ rtw_dequeue(struct ifnet *ifp, struct rtw_txsoft_blk **tsbp,
return 0;
}
+ if (sc->sc_ic.ic_state != IEEE80211_S_RUN) {
+ DPRINTF(sc, RTW_DEBUG_XMIT, ("%s: not running\n", __func__));
+ return 0;
+ }
+
+ *mp = NULL;
+
+ IFQ_POLL(&ifp->if_snd, m0);
+ if (m0 == NULL) {
+ DPRINTF(sc, RTW_DEBUG_XMIT, ("%s: no frame ready\n",
+ __func__));
+ return 0;
+ }
+
if (rtw_txring_choose(sc, tsbp, tdbp, RTW_TXPRIMD) == -1) {
DPRINTF(sc, RTW_DEBUG_XMIT, ("%s: no descriptor\n", __func__));
*if_flagsp |= IFF_OACTIVE;
+ sc->sc_if.if_timer = 1;
return 0;
}
- *mp = NULL;
-
IFQ_DEQUEUE(&ifp->if_snd, m0);
if (m0 == NULL) {
DPRINTF(sc, RTW_DEBUG_XMIT, ("%s: no frame/ring ready\n",
@@ -2746,7 +2913,7 @@ rtw_dmamap_load_txbuf(bus_dma_tag_t dmat, bus_dmamap_t dmam, struct mbuf *chain,
m_freem(m0);
return NULL;
} else if (dmam->dm_nsegs > ndescfree) {
- *ifflagsp |= IFF_OACTIVE;
+ printf("%s: too many tx segments\n", dvname);
bus_dmamap_unload(dmat, dmam);
m_freem(m0);
return NULL;
@@ -2760,11 +2927,11 @@ rtw_print_txdesc(struct rtw_softc *sc, const char *action,
struct rtw_txsoft *ts, struct rtw_txdesc_blk *tdb, int desc)
{
struct rtw_txdesc *td = &tdb->tdb_desc[desc];
- DPRINTF(sc, RTW_DEBUG_XMIT_DESC, ("%s: %p %s txdesc[%d] ctl0 %#08x "
- "ctl1 %#08x buf %#08x len %#08x\n",
+ DPRINTF(sc, RTW_DEBUG_XMIT_DESC, ("%s: %p %s txdesc[%d] next %#08x "
+ "buf %#08x ctl0 %#08x ctl1 %#08x len %#08x\n",
sc->sc_dev.dv_xname, ts, action, desc,
- letoh32(td->td_ctl0),
- letoh32(td->td_ctl1), letoh32(td->td_buf),
+ letoh32(td->td_buf), letoh32(td->td_next),
+ letoh32(td->td_ctl0), letoh32(td->td_ctl1),
letoh32(td->td_len)));
}
#endif /* RTW_DEBUG */
@@ -2976,12 +3143,10 @@ rtw_start(struct ifnet *ifp)
SIMPLEQ_REMOVE_HEAD(&tsb->tsb_freeq, ts_q);
SIMPLEQ_INSERT_TAIL(&tsb->tsb_dirtyq, ts, ts_q);
- if (tsb != &sc->sc_txsoft_blk[RTW_TXPRIBCN]) {
+ if (tsb != &sc->sc_txsoft_blk[RTW_TXPRIBCN])
sc->sc_led_state.ls_event |= RTW_LED_S_TX;
- tsb->tsb_tx_timer = 5;
- ifp->if_timer = 1;
- }
-
+ tsb->tsb_tx_timer = 5;
+ ifp->if_timer = 1;
tppoll = RTW_READ8(&sc->sc_regs, RTW_TPPOLL);
tppoll &= ~RTW_TPPOLL_SALL;
tppoll |= tsb->tsb_poll & RTW_TPPOLL_ALL;
@@ -3000,9 +3165,26 @@ post_dequeue_err:
}
void
+rtw_idle(struct rtw_regs *regs)
+{
+ int active;
+
+ /* request stop DMA; wait for packets to stop transmitting. */
+
+ RTW_WRITE8(regs, RTW_TPPOLL, RTW_TPPOLL_SALL);
+ RTW_WBR(regs, RTW_TPPOLL, RTW_TPPOLL);
+
+ for (active = 0; active < 300 &&
+ (RTW_READ8(regs, RTW_TPPOLL) & RTW_TPPOLL_ACTIVE) != 0; active++)
+ DELAY(10);
+ RTW_DPRINTF(RTW_DEBUG_BUGS,
+ ("%s: transmit DMA idle in %dus\n", __func__, active * 10));
+}
+
+void
rtw_watchdog(struct ifnet *ifp)
{
- int pri;
+ int pri, tx_timeouts = 0;
struct rtw_softc *sc;
struct rtw_txsoft_blk *tsb;
@@ -3018,30 +3200,32 @@ rtw_watchdog(struct ifnet *ifp)
if (tsb->tsb_tx_timer == 0)
continue;
-
- if (--tsb->tsb_tx_timer == 0) {
+ else if (--tsb->tsb_tx_timer == 0) {
if (SIMPLEQ_EMPTY(&tsb->tsb_dirtyq))
continue;
- printf("%s: transmit timeout, priority %d\n",
- ifp->if_xname, pri);
+ RTW_DPRINTF(RTW_DEBUG_BUGS,
+ ("%s: transmit timeout, priority %d\n",
+ ifp->if_xname, pri));
ifp->if_oerrors++;
- /* Stop Tx DMA, disable transmitter, clear
- * Tx rings, and restart.
- *
- * TBD Stop/restart just the broken ring?
- */
- RTW_WRITE8(&sc->sc_regs, RTW_TPPOLL, RTW_TPPOLL_SALL);
- RTW_SYNC(&sc->sc_regs, RTW_TPPOLL, RTW_TPPOLL);
- rtw_io_enable(&sc->sc_regs, RTW_CR_TE, 0);
- rtw_txdescs_reset(sc);
- rtw_io_enable(&sc->sc_regs, RTW_CR_TE, 1);
- sc->sc_if.if_flags &= ~IFF_OACTIVE;
- rtw_start(ifp);
+ tx_timeouts++;
} else
ifp->if_timer = 1;
}
+
+ if (tx_timeouts > 0) {
+ /* Stop Tx DMA, disable xmtr, flush Tx rings, enable xmtr,
+ * reset s/w tx-ring pointers, and start transmission.
+ *
+ * TBD Stop/restart just the broken rings?
+ */
+ rtw_idle(&sc->sc_regs);
+ rtw_io_enable(&sc->sc_regs, RTW_CR_TE, 0);
+ rtw_txdescs_reset(sc);
+ rtw_io_enable(&sc->sc_regs, RTW_CR_TE, 1);
+ rtw_txring_fixup(sc);
+ rtw_start(ifp);
+ }
ieee80211_watchdog(ifp);
- return;
}
void
@@ -3062,7 +3246,7 @@ rtw_next_scan(void *arg)
void
rtw_join_bss(struct rtw_softc *sc, u_int8_t *bssid, u_int16_t intval0)
{
- uint16_t bcnitv, intval;
+ uint16_t bcnitv, bintritv, intval;
int i;
struct rtw_regs *regs = &sc->sc_regs;
@@ -3078,10 +3262,13 @@ rtw_join_bss(struct rtw_softc *sc, u_int8_t *bssid, u_int16_t intval0)
bcnitv = RTW_READ16(regs, RTW_BCNITV) & ~RTW_BCNITV_BCNITV_MASK;
bcnitv |= LSHIFT(intval, RTW_BCNITV_BCNITV_MASK);
RTW_WRITE16(regs, RTW_BCNITV, bcnitv);
+ /* interrupt host 1ms before the TBTT */
+ bintritv = RTW_READ16(regs, RTW_BINTRITV) & ~RTW_BINTRITV_BINTRITV;
+ bintritv |= LSHIFT(1000, RTW_BINTRITV_BINTRITV);
+ RTW_WRITE16(regs, RTW_BINTRITV, bintritv);
/* magic from Linux */
RTW_WRITE16(regs, RTW_ATIMWND, LSHIFT(1, RTW_ATIMWND_ATIMWND));
RTW_WRITE16(regs, RTW_ATIMTRITV, LSHIFT(2, RTW_ATIMTRITV_ATIMTRITV));
-
rtw_set_access(regs, RTW_ACCESS_NONE);
/* TBD WEP */
@@ -3096,7 +3283,6 @@ rtw_newstate(struct ieee80211com *ic, enum ieee80211_state nstate, int arg)
{
struct ifnet *ifp = &ic->ic_if;
struct rtw_softc *sc = ifp->if_softc;
- struct mbuf *m;
enum ieee80211_state ostate;
int error;
@@ -3134,12 +3320,6 @@ rtw_newstate(struct ieee80211com *ic, enum ieee80211_state nstate, int arg)
case IEEE80211_M_HOSTAP:
case IEEE80211_M_IBSS:
rtw_set_nettype(sc, IEEE80211_M_MONITOR);
- m = ieee80211_beacon_alloc(ic, ic->ic_bss);
- if (m == NULL) {
- printf("%s: could not allocate beacon\n",
- sc->sc_dev.dv_xname);
- } else
- IF_ENQUEUE(&sc->sc_beaconq, m);
/*FALLTHROUGH*/
case IEEE80211_M_AHDEMO:
case IEEE80211_M_STA:
@@ -3159,12 +3339,6 @@ rtw_newstate(struct ieee80211com *ic, enum ieee80211_state nstate, int arg)
if (nstate != IEEE80211_S_SCAN)
timeout_del(&sc->sc_scan_to);
- /* Start beacon transmission. */
- if (nstate == IEEE80211_S_RUN &&
- (ic->ic_opmode == IEEE80211_M_HOSTAP ||
- ic->ic_opmode == IEEE80211_M_IBSS))
- rtw_start(&sc->sc_if);
-
return (*sc->sc_mtbl.mt_newstate)(ic, nstate, arg);
}
@@ -3405,7 +3579,8 @@ rtw_txdesc_blk_setup(struct rtw_txdesc_blk *tdb, struct rtw_txdesc *desc,
bzero(tdb->tdb_desc, sizeof(tdb->tdb_desc[0]) * tdb->tdb_ndesc);
- rtw_txdesc_blk_reset(tdb);
+ rtw_txdesc_blk_init(tdb);
+ tdb->tdb_next = 0;
}
void
@@ -3729,8 +3904,7 @@ rtw_attach(struct rtw_softc *sc)
("%s: PHY delay %d\n", sc->sc_dev.dv_xname, sc->sc_phydelay));
if (sc->sc_locale == RTW_LOCALE_UNKNOWN)
- rtw_identify_country(&sc->sc_regs, &sc->sc_locale,
- sc->sc_dev.dv_xname);
+ rtw_identify_country(&sc->sc_regs, &sc->sc_locale);
rtw_init_channels(sc->sc_locale, &sc->sc_ic.ic_channels,
sc->sc_dev.dv_xname);