summaryrefslogtreecommitdiff
path: root/sys/dev/pci
diff options
context:
space:
mode:
Diffstat (limited to 'sys/dev/pci')
-rw-r--r--sys/dev/pci/if_iwi.c789
-rw-r--r--sys/dev/pci/if_iwireg.h101
-rw-r--r--sys/dev/pci/if_iwivar.h69
3 files changed, 403 insertions, 556 deletions
diff --git a/sys/dev/pci/if_iwi.c b/sys/dev/pci/if_iwi.c
index bdefff19b1a..e744161efe1 100644
--- a/sys/dev/pci/if_iwi.c
+++ b/sys/dev/pci/if_iwi.c
@@ -1,4 +1,4 @@
-/* $OpenBSD: if_iwi.c,v 1.62 2006/03/27 20:46:35 damien Exp $ */
+/* $OpenBSD: if_iwi.c,v 1.63 2006/03/31 17:18:36 pedro Exp $ */
/*-
* Copyright (c) 2004-2006
@@ -97,27 +97,19 @@ int iwi_match(struct device *, void *, void *);
void iwi_attach(struct device *, struct device *, void *);
int iwi_detach(struct device *, int);
void iwi_power(int, void *);
-int iwi_alloc_cmd_ring(struct iwi_softc *, struct iwi_cmd_ring *);
-void iwi_reset_cmd_ring(struct iwi_softc *, struct iwi_cmd_ring *);
-void iwi_free_cmd_ring(struct iwi_softc *, struct iwi_cmd_ring *);
-int iwi_alloc_tx_ring(struct iwi_softc *, struct iwi_tx_ring *,
- bus_addr_t, bus_addr_t);
-void iwi_reset_tx_ring(struct iwi_softc *, struct iwi_tx_ring *);
-void iwi_free_tx_ring(struct iwi_softc *, struct iwi_tx_ring *);
-int iwi_alloc_rx_ring(struct iwi_softc *, struct iwi_rx_ring *);
-void iwi_reset_rx_ring(struct iwi_softc *, struct iwi_rx_ring *);
-void iwi_free_rx_ring(struct iwi_softc *, struct iwi_rx_ring *);
+int iwi_dma_alloc(struct iwi_softc *);
+void iwi_release(struct iwi_softc *);
int iwi_media_change(struct ifnet *);
void iwi_media_status(struct ifnet *, struct ifmediareq *);
uint16_t iwi_read_prom_word(struct iwi_softc *, uint8_t);
int iwi_find_txnode(struct iwi_softc *, const uint8_t *);
int iwi_newstate(struct ieee80211com *, enum ieee80211_state, int);
-void iwi_frame_intr(struct iwi_softc *, struct iwi_rx_data *,
+void iwi_frame_intr(struct iwi_softc *, struct iwi_rx_buf *, int,
struct iwi_frame *);
-void iwi_notification_intr(struct iwi_softc *, struct iwi_rx_data *,
+void iwi_notification_intr(struct iwi_softc *, struct iwi_rx_buf *,
struct iwi_notif *);
void iwi_rx_intr(struct iwi_softc *);
-void iwi_tx_intr(struct iwi_softc *, struct iwi_tx_ring *);
+void iwi_tx_intr(struct iwi_softc *);
int iwi_intr(void *);
int iwi_cmd(struct iwi_softc *, uint8_t, void *, uint8_t, int);
int iwi_tx_start(struct ifnet *, struct mbuf *,
@@ -129,8 +121,8 @@ int iwi_get_radio(struct iwi_softc *, int *);
int iwi_ioctl(struct ifnet *, u_long, caddr_t);
void iwi_stop_master(struct iwi_softc *);
int iwi_reset(struct iwi_softc *);
-int iwi_load_ucode(struct iwi_softc *, const char *, int);
-int iwi_load_firmware(struct iwi_softc *, const char *, int);
+int iwi_load_ucode(struct iwi_softc *, const char *);
+int iwi_load_firmware(struct iwi_softc *, const char *);
int iwi_config(struct iwi_softc *);
int iwi_set_chan(struct iwi_softc *, struct ieee80211_channel *);
int iwi_scan(struct iwi_softc *);
@@ -211,6 +203,9 @@ iwi_attach(struct device *parent, struct device *self, void *aux)
sc->sc_sh = memh;
sc->sc_dmat = pa->pa_dmat;
+ /* disable interrupts */
+ CSR_WRITE_4(sc, IWI_CSR_INTR_MASK, 0);
+
if (pci_intr_map(pa, &ih) != 0) {
printf(": could not map interrupt\n");
return;
@@ -233,61 +228,19 @@ iwi_attach(struct device *parent, struct device *self, void *aux)
return;
}
- /*
- * Allocate rings.
- */
- error = iwi_alloc_cmd_ring(sc, &sc->cmdq);
- if (error != 0) {
- printf(": could not allocate Cmd ring\n");
+ if (iwi_dma_alloc(sc) != 0) {
+ printf(": could not allocate DMA resources\n");
return;
}
- error = iwi_alloc_tx_ring(sc, &sc->txq[0], IWI_CSR_TX1_RIDX,
- IWI_CSR_TX1_WIDX);
- if (error != 0) {
- printf(": could not allocate Tx ring 1\n");
- goto fail1;
- }
-
- error = iwi_alloc_tx_ring(sc, &sc->txq[1], IWI_CSR_TX2_RIDX,
- IWI_CSR_TX2_WIDX);
- if (error != 0) {
- printf(": could not allocate Tx ring 2\n");
- goto fail2;
- }
-
- error = iwi_alloc_tx_ring(sc, &sc->txq[2], IWI_CSR_TX3_RIDX,
- IWI_CSR_TX3_WIDX);
- if (error != 0) {
- printf(": could not allocate Tx ring 3\n");
- goto fail3;
- }
-
- error = iwi_alloc_tx_ring(sc, &sc->txq[3], IWI_CSR_TX4_RIDX,
- IWI_CSR_TX4_WIDX);
- if (error != 0) {
- printf(": could not allocate Tx ring 4\n");
- goto fail4;
- }
-
- error = iwi_alloc_rx_ring(sc, &sc->rxq);
- if (error != 0) {
- printf(": could not allocate Rx ring\n");
- goto fail5;
- }
-
- ic->ic_phytype = IEEE80211_T_OFDM; /* not only, but not used */
- ic->ic_opmode = IEEE80211_M_STA; /* default to BSS mode */
+ ic->ic_phytype = IEEE80211_T_OFDM;
+ ic->ic_opmode = IEEE80211_M_STA;
ic->ic_state = IEEE80211_S_INIT;
/* set device capabilities */
- ic->ic_caps =
- IEEE80211_C_IBSS | /* IBSS mode supported */
- IEEE80211_C_MONITOR | /* monitor mode supported */
- IEEE80211_C_WEP | /* h/w WEP supported */
- IEEE80211_C_TXPMGT | /* tx power management */
- IEEE80211_C_SHPREAMBLE | /* short preamble supported */
- IEEE80211_C_SCANALL; /* h/w scanning */
+ ic->ic_caps = IEEE80211_C_IBSS | IEEE80211_C_PMGT | IEEE80211_C_WEP |
+ IEEE80211_C_TXPMGT | IEEE80211_C_SHPREAMBLE | IEEE80211_C_MONITOR |
+ IEEE80211_C_SCANALL;
/* read MAC address from EEPROM */
val = iwi_read_prom_word(sc, IWI_EEPROM_MAC + 0);
@@ -368,14 +321,6 @@ iwi_attach(struct device *parent, struct device *self, void *aux)
sc->sc_txtap.wt_ihdr.it_len = htole16(sc->sc_txtap_len);
sc->sc_txtap.wt_ihdr.it_present = htole32(IWI_TX_RADIOTAP_PRESENT);
#endif
-
- return;
-
-fail5: iwi_free_tx_ring(sc, &sc->txq[3]);
-fail4: iwi_free_tx_ring(sc, &sc->txq[2]);
-fail3: iwi_free_tx_ring(sc, &sc->txq[1]);
-fail2: iwi_free_tx_ring(sc, &sc->txq[0]);
-fail1: iwi_free_cmd_ring(sc, &sc->cmdq);
}
int
@@ -389,12 +334,7 @@ iwi_detach(struct device* self, int flags)
ieee80211_ifdetach(ifp);
if_detach(ifp);
- iwi_free_cmd_ring(sc, &sc->cmdq);
- iwi_free_tx_ring(sc, &sc->txq[0]);
- iwi_free_tx_ring(sc, &sc->txq[1]);
- iwi_free_tx_ring(sc, &sc->txq[2]);
- iwi_free_tx_ring(sc, &sc->txq[3]);
- iwi_free_rx_ring(sc, &sc->rxq);
+ iwi_release(sc);
if (sc->sc_ih != NULL) {
pci_intr_disestablish(sc->sc_pct, sc->sc_ih);
@@ -430,268 +370,193 @@ iwi_power(int why, void *arg)
}
int
-iwi_alloc_cmd_ring(struct iwi_softc *sc, struct iwi_cmd_ring *ring)
+iwi_dma_alloc(struct iwi_softc *sc)
{
- int nsegs, error;
-
- ring->queued = 0;
- ring->cur = ring->next = 0;
+ int i, nsegs, error;
+ /*
+ * Allocate and map Tx ring.
+ */
error = bus_dmamap_create(sc->sc_dmat,
- sizeof (struct iwi_cmd_desc) * IWI_CMD_RING_COUNT, 1,
- sizeof (struct iwi_cmd_desc) * IWI_CMD_RING_COUNT, 0,
- BUS_DMA_NOWAIT, &ring->map);
+ sizeof (struct iwi_tx_desc) * IWI_TX_RING_SIZE, 1,
+ sizeof (struct iwi_tx_desc) * IWI_TX_RING_SIZE, 0, BUS_DMA_NOWAIT,
+ &sc->tx_ring_map);
if (error != 0) {
- printf("%s: could not create cmd ring DMA map\n",
+ printf("%s: could not create tx ring DMA map\n",
sc->sc_dev.dv_xname);
goto fail;
}
error = bus_dmamem_alloc(sc->sc_dmat,
- sizeof (struct iwi_cmd_desc) * IWI_CMD_RING_COUNT, PAGE_SIZE, 0,
- &ring->seg, 1, &nsegs, BUS_DMA_NOWAIT);
+ sizeof (struct iwi_tx_desc) * IWI_TX_RING_SIZE, PAGE_SIZE, 0,
+ &sc->tx_ring_seg, 1, &nsegs, BUS_DMA_NOWAIT);
if (error != 0) {
- printf("%s: could not allocate cmd ring DMA memory\n",
+ printf("%s: could not allocate tx ring DMA memory\n",
sc->sc_dev.dv_xname);
goto fail;
}
- error = bus_dmamem_map(sc->sc_dmat, &ring->seg, nsegs,
- sizeof (struct iwi_cmd_desc) * IWI_CMD_RING_COUNT,
- (caddr_t *)&ring->desc, BUS_DMA_NOWAIT);
+ error = bus_dmamem_map(sc->sc_dmat, &sc->tx_ring_seg, nsegs,
+ sizeof (struct iwi_tx_desc) * IWI_TX_RING_SIZE,
+ (caddr_t *)&sc->tx_desc, BUS_DMA_NOWAIT);
if (error != 0) {
- printf("%s: could not map cmd ring DMA memory\n",
+ printf("%s: could not map tx ring DMA memory\n",
sc->sc_dev.dv_xname);
goto fail;
}
- error = bus_dmamap_load(sc->sc_dmat, ring->map, ring->desc,
- sizeof (struct iwi_cmd_desc) * IWI_CMD_RING_COUNT, NULL,
+ error = bus_dmamap_load(sc->sc_dmat, sc->tx_ring_map, sc->tx_desc,
+ sizeof (struct iwi_tx_desc) * IWI_TX_RING_SIZE, NULL,
BUS_DMA_NOWAIT);
if (error != 0) {
- printf("%s: could not load cmd ring DMA map\n",
+ printf("%s: could not load tx ring DMA map\n",
sc->sc_dev.dv_xname);
goto fail;
}
- bzero(ring->desc, sizeof (struct iwi_cmd_desc) * IWI_CMD_RING_COUNT);
- return 0;
-
-fail: iwi_free_cmd_ring(sc, ring);
- return error;
-}
-
-void
-iwi_reset_cmd_ring(struct iwi_softc *sc, struct iwi_cmd_ring *ring)
-{
- ring->queued = 0;
- ring->cur = ring->next = 0;
-}
-
-void
-iwi_free_cmd_ring(struct iwi_softc *sc, struct iwi_cmd_ring *ring)
-{
- if (ring->map != NULL) {
- if (ring->desc != NULL) {
- bus_dmamap_unload(sc->sc_dmat, ring->map);
- bus_dmamem_unmap(sc->sc_dmat, (caddr_t)ring->desc,
- sizeof (struct iwi_cmd_desc) * IWI_CMD_RING_COUNT);
- bus_dmamem_free(sc->sc_dmat, &ring->seg, 1);
- }
- bus_dmamap_destroy(sc->sc_dmat, ring->map);
- }
-}
-
-int
-iwi_alloc_tx_ring(struct iwi_softc *sc, struct iwi_tx_ring *ring,
- bus_addr_t csr_ridx, bus_addr_t csr_widx)
-{
- struct iwi_tx_data *data;
- int i, nsegs, error;
-
- ring->queued = 0;
- ring->cur = ring->next = 0;
- ring->csr_ridx = csr_ridx;
- ring->csr_widx = csr_widx;
+ bzero(sc->tx_desc, sizeof (struct iwi_tx_desc) * IWI_TX_RING_SIZE);
+ /*
+ * Allocate and map command ring.
+ */
error = bus_dmamap_create(sc->sc_dmat,
- sizeof (struct iwi_tx_desc) * IWI_TX_RING_COUNT, 1,
- sizeof (struct iwi_tx_desc) * IWI_TX_RING_COUNT, 0, BUS_DMA_NOWAIT,
- &ring->map);
+ sizeof (struct iwi_cmd_desc) * IWI_CMD_RING_SIZE, 1,
+ sizeof (struct iwi_cmd_desc) * IWI_CMD_RING_SIZE, 0,
+ BUS_DMA_NOWAIT, &sc->cmd_ring_map);
if (error != 0) {
- printf("%s: could not create tx ring DMA map\n",
+ printf("%s: could not create command ring DMA map\n",
sc->sc_dev.dv_xname);
goto fail;
}
error = bus_dmamem_alloc(sc->sc_dmat,
- sizeof (struct iwi_tx_desc) * IWI_TX_RING_COUNT, PAGE_SIZE, 0,
- &ring->seg, 1, &nsegs, BUS_DMA_NOWAIT);
+ sizeof (struct iwi_cmd_desc) * IWI_CMD_RING_SIZE, PAGE_SIZE, 0,
+ &sc->cmd_ring_seg, 1, &nsegs, BUS_DMA_NOWAIT);
if (error != 0) {
- printf("%s: could not allocate tx ring DMA memory\n",
+ printf("%s: could not allocate command ring DMA memory\n",
sc->sc_dev.dv_xname);
goto fail;
}
- error = bus_dmamem_map(sc->sc_dmat, &ring->seg, nsegs,
- sizeof (struct iwi_tx_desc) * IWI_TX_RING_COUNT,
- (caddr_t *)&ring->desc, BUS_DMA_NOWAIT);
+ error = bus_dmamem_map(sc->sc_dmat, &sc->cmd_ring_seg, nsegs,
+ sizeof (struct iwi_cmd_desc) * IWI_CMD_RING_SIZE,
+ (caddr_t *)&sc->cmd_desc, BUS_DMA_NOWAIT);
if (error != 0) {
- printf("%s: could not map tx ring DMA memory\n",
+ printf("%s: could not map command ring DMA memory\n",
sc->sc_dev.dv_xname);
goto fail;
}
- error = bus_dmamap_load(sc->sc_dmat, ring->map, ring->desc,
- sizeof (struct iwi_tx_desc) * IWI_TX_RING_COUNT, NULL,
+ error = bus_dmamap_load(sc->sc_dmat, sc->cmd_ring_map, sc->cmd_desc,
+ sizeof (struct iwi_cmd_desc) * IWI_CMD_RING_SIZE, NULL,
BUS_DMA_NOWAIT);
if (error != 0) {
- printf("%s: could not load tx ring DMA map\n",
+ printf("%s: could not load command ring DMA map\n",
sc->sc_dev.dv_xname);
goto fail;
}
- bzero(ring->desc, sizeof (struct iwi_tx_desc) * IWI_TX_RING_COUNT);
-
- for (i = 0; i < IWI_TX_RING_COUNT; i++) {
- data = &ring->data[i];
+ bzero(sc->cmd_desc, sizeof (struct iwi_cmd_desc) * IWI_CMD_RING_SIZE);
+ /*
+ * Allocate Tx buffers DMA maps.
+ */
+ for (i = 0; i < IWI_TX_RING_SIZE; i++) {
error = bus_dmamap_create(sc->sc_dmat, MCLBYTES,
- IWI_MAX_SCATTER, MCLBYTES, 0, BUS_DMA_NOWAIT, &data->map);
+ IWI_MAX_NSEG - 2, MCLBYTES, 0, BUS_DMA_NOWAIT,
+ &sc->tx_buf[i].map);
if (error != 0) {
- printf("%s: could not create tx buf DMA map\n",
+ printf("%s: could not create tx buf DMA map",
sc->sc_dev.dv_xname);
goto fail;
}
}
- return 0;
-
-fail: iwi_free_tx_ring(sc, ring);
- return error;
-}
-
-void
-iwi_reset_tx_ring(struct iwi_softc *sc, struct iwi_tx_ring *ring)
-{
- struct iwi_tx_data *data;
- int i;
-
- for (i = 0; i < IWI_TX_RING_COUNT; i++) {
- data = &ring->data[i];
-
- if (data->m != NULL) {
- bus_dmamap_unload(sc->sc_dmat, data->map);
- m_freem(data->m);
- data->m = NULL;
- }
- }
-
- ring->queued = 0;
- ring->cur = ring->next = 0;
-}
-
-void
-iwi_free_tx_ring(struct iwi_softc *sc, struct iwi_tx_ring *ring)
-{
- struct iwi_tx_data *data;
- int i;
-
- if (ring->map != NULL) {
- if (ring->desc != NULL) {
- bus_dmamap_unload(sc->sc_dmat, ring->map);
- bus_dmamem_unmap(sc->sc_dmat, (caddr_t)ring->desc,
- sizeof (struct iwi_tx_desc) * IWI_TX_RING_COUNT);
- bus_dmamem_free(sc->sc_dmat, &ring->seg, 1);
- }
- bus_dmamap_destroy(sc->sc_dmat, ring->map);
- }
-
- for (i = 0; i < IWI_TX_RING_COUNT; i++) {
- data = &ring->data[i];
-
- if (data->m != NULL) {
- bus_dmamap_unload(sc->sc_dmat, data->map);
- m_freem(data->m);
- }
- bus_dmamap_destroy(sc->sc_dmat, data->map);
- }
-}
-
-int
-iwi_alloc_rx_ring(struct iwi_softc *sc, struct iwi_rx_ring *ring)
-{
- struct iwi_rx_data *data;
- int i, error;
-
- ring->cur = 0;
-
- for (i = 0; i < IWI_RX_RING_COUNT; i++) {
- data = &sc->rxq.data[i];
+ /*
+ * Allocate and map Rx buffers.
+ */
+ for (i = 0; i < IWI_RX_RING_SIZE; i++) {
error = bus_dmamap_create(sc->sc_dmat, MCLBYTES, 1, MCLBYTES,
- 0, BUS_DMA_NOWAIT, &data->map);
+ 0, BUS_DMA_NOWAIT, &sc->rx_buf[i].map);
if (error != 0) {
- printf("%s: could not create rx buf DMA map\n",
+ printf("%s: could not create rx buf DMA map",
sc->sc_dev.dv_xname);
goto fail;
}
- MGETHDR(data->m, M_DONTWAIT, MT_DATA);
- if (data->m == NULL) {
+ MGETHDR(sc->rx_buf[i].m, M_DONTWAIT, MT_DATA);
+ if (sc->rx_buf[i].m == NULL) {
printf("%s: could not allocate rx mbuf\n",
sc->sc_dev.dv_xname);
error = ENOMEM;
goto fail;
}
- MCLGET(data->m, M_DONTWAIT);
- if (!(data->m->m_flags & M_EXT)) {
- m_freem(data->m);
- data->m = NULL;
+ MCLGET(sc->rx_buf[i].m, M_DONTWAIT);
+ if (!(sc->rx_buf[i].m->m_flags & M_EXT)) {
+ m_freem(sc->rx_buf[i].m);
printf("%s: could not allocate rx mbuf cluster\n",
sc->sc_dev.dv_xname);
error = ENOMEM;
goto fail;
}
- error = bus_dmamap_load(sc->sc_dmat, data->map,
- mtod(data->m, void *), MCLBYTES, NULL, BUS_DMA_NOWAIT);
+ error = bus_dmamap_load(sc->sc_dmat, sc->rx_buf[i].map,
+ mtod(sc->rx_buf[i].m, void *), MCLBYTES, NULL,
+ BUS_DMA_NOWAIT);
if (error != 0) {
printf("%s: could not load rx buf DMA map\n",
sc->sc_dev.dv_xname);
goto fail;
}
-
- data->reg = IWI_CSR_RX_BASE + i * 4;
}
return 0;
-fail: iwi_free_rx_ring(sc, ring);
+fail: iwi_release(sc);
return error;
}
void
-iwi_reset_rx_ring(struct iwi_softc *sc, struct iwi_rx_ring *ring)
-{
- ring->cur = 0;
-}
-
-void
-iwi_free_rx_ring(struct iwi_softc *sc, struct iwi_rx_ring *ring)
+iwi_release(struct iwi_softc *sc)
{
- struct iwi_rx_data *data;
int i;
- for (i = 0; i < IWI_RX_RING_COUNT; i++) {
- data = &sc->rxq.data[i];
+ if (sc->tx_ring_map != NULL) {
+ if (sc->tx_desc != NULL) {
+ bus_dmamap_unload(sc->sc_dmat, sc->tx_ring_map);
+ bus_dmamem_unmap(sc->sc_dmat, (caddr_t)sc->tx_desc,
+ sizeof (struct iwi_tx_desc) * IWI_TX_RING_SIZE);
+ bus_dmamem_free(sc->sc_dmat, &sc->tx_ring_seg, 1);
+ }
+ bus_dmamap_destroy(sc->sc_dmat, sc->tx_ring_map);
+ }
+
+ if (sc->cmd_ring_map != NULL) {
+ if (sc->cmd_desc != NULL) {
+ bus_dmamap_unload(sc->sc_dmat, sc->cmd_ring_map);
+ bus_dmamem_unmap(sc->sc_dmat, (caddr_t)sc->cmd_desc,
+ sizeof (struct iwi_cmd_desc) * IWI_CMD_RING_SIZE);
+ bus_dmamem_free(sc->sc_dmat, &sc->cmd_ring_seg, 1);
+ }
+ bus_dmamap_destroy(sc->sc_dmat, sc->cmd_ring_map);
+ }
+
+ for (i = 0; i < IWI_TX_RING_SIZE; i++) {
+ if (sc->tx_buf[i].m != NULL) {
+ bus_dmamap_unload(sc->sc_dmat, sc->tx_buf[i].map);
+ m_freem(sc->tx_buf[i].m);
+ }
+ bus_dmamap_destroy(sc->sc_dmat, sc->tx_buf[i].map);
+ }
- if (data->m != NULL) {
- bus_dmamap_unload(sc->sc_dmat, data->map);
- m_freem(data->m);
+ for (i = 0; i < IWI_RX_RING_SIZE; i++) {
+ if (sc->rx_buf[i].m != NULL) {
+ bus_dmamap_unload(sc->sc_dmat, sc->rx_buf[i].map);
+ m_freem(sc->rx_buf[i].m);
}
- bus_dmamap_destroy(sc->sc_dmat, data->map);
+ bus_dmamap_destroy(sc->sc_dmat, sc->rx_buf[i].map);
}
}
@@ -904,7 +769,7 @@ iwi_read_prom_word(struct iwi_softc *sc, uint8_t addr)
}
void
-iwi_frame_intr(struct iwi_softc *sc, struct iwi_rx_data *data,
+iwi_frame_intr(struct iwi_softc *sc, struct iwi_rx_buf *buf, int i,
struct iwi_frame *frame)
{
struct ieee80211com *ic = &sc->sc_ic;
@@ -914,10 +779,10 @@ iwi_frame_intr(struct iwi_softc *sc, struct iwi_rx_data *data,
struct ieee80211_node *ni;
int error;
- DPRINTFN(5, ("received frame len=%u chan=%u rssi=%u\n",
- letoh16(frame->len), frame->chan, frame->rssi_dbm));
+ DPRINTFN(5, ("RX!DATA!%u!%u!%u\n", letoh16(frame->len), frame->chan,
+ frame->rssi_dbm));
- bus_dmamap_sync(sc->sc_dmat, data->map, sizeof (struct iwi_hdr),
+ bus_dmamap_sync(sc->sc_dmat, buf->map, sizeof (struct iwi_hdr),
sizeof (struct iwi_frame) + letoh16(frame->len),
BUS_DMASYNC_POSTREAD);
@@ -947,16 +812,16 @@ iwi_frame_intr(struct iwi_softc *sc, struct iwi_rx_data *data,
return;
}
- bus_dmamap_unload(sc->sc_dmat, data->map);
+ bus_dmamap_unload(sc->sc_dmat, buf->map);
- error = bus_dmamap_load(sc->sc_dmat, data->map, mtod(mnew, void *),
+ error = bus_dmamap_load(sc->sc_dmat, buf->map, mtod(mnew, void *),
MCLBYTES, NULL, BUS_DMA_NOWAIT);
if (error != 0) {
m_freem(mnew);
/* try to reload the old mbuf */
- error = bus_dmamap_load(sc->sc_dmat, data->map,
- mtod(data->m, void *), MCLBYTES, NULL, BUS_DMA_NOWAIT);
+ error = bus_dmamap_load(sc->sc_dmat, buf->map,
+ mtod(buf->m, void *), MCLBYTES, NULL, BUS_DMA_NOWAIT);
if (error != 0) {
/* very unlikely that it will fail... */
panic("%s: could not load old rx mbuf",
@@ -966,9 +831,9 @@ iwi_frame_intr(struct iwi_softc *sc, struct iwi_rx_data *data,
return;
}
- m = data->m;
- data->m = mnew;
- CSR_WRITE_4(sc, data->reg, data->map->dm_segs[0].ds_addr);
+ m = buf->m;
+ buf->m = mnew;
+ CSR_WRITE_4(sc, IWI_CSR_RX_BASE + i * 4, buf->map->dm_segs->ds_addr);
/* finalize mbuf */
m->m_pkthdr.rcvif = ifp;
@@ -1021,12 +886,11 @@ iwi_frame_intr(struct iwi_softc *sc, struct iwi_rx_data *data,
/* send the frame to the upper layer */
ieee80211_input(ifp, m, ni, frame->rssi_dbm, 0);
- /* node is no longer needed */
ieee80211_release_node(ic, ni);
}
void
-iwi_notification_intr(struct iwi_softc *sc, struct iwi_rx_data *data,
+iwi_notification_intr(struct iwi_softc *sc, struct iwi_rx_buf *buf,
struct iwi_notif *notif)
{
struct ieee80211com *ic = &sc->sc_ic;
@@ -1036,7 +900,7 @@ iwi_notification_intr(struct iwi_softc *sc, struct iwi_rx_data *data,
struct iwi_notif_authentication *auth;
struct iwi_notif_association *assoc;
- bus_dmamap_sync(sc->sc_dmat, data->map, sizeof (struct iwi_hdr),
+ bus_dmamap_sync(sc->sc_dmat, buf->map, sizeof (struct iwi_hdr),
sizeof (struct iwi_notif) + letoh16(notif->len),
BUS_DMASYNC_POSTREAD);
@@ -1044,7 +908,7 @@ iwi_notification_intr(struct iwi_softc *sc, struct iwi_rx_data *data,
case IWI_NOTIF_TYPE_SCAN_CHANNEL:
chan = (struct iwi_notif_scan_channel *)(notif + 1);
- DPRINTFN(2, ("Scanning channel (%u)\n", chan->nchan));
+ DPRINTFN(2, ("Scan channel (%u)\n", chan->nchan));
break;
case IWI_NOTIF_TYPE_SCAN_COMPLETE:
@@ -1112,28 +976,30 @@ iwi_notification_intr(struct iwi_softc *sc, struct iwi_rx_data *data,
void
iwi_rx_intr(struct iwi_softc *sc)
{
- struct iwi_rx_data *data;
+ struct iwi_rx_buf *buf;
struct iwi_hdr *hdr;
- uint32_t hw;
+ uint32_t r, i;
- hw = CSR_READ_4(sc, IWI_CSR_RX_RIDX);
+ r = CSR_READ_4(sc, IWI_CSR_RX_READ_INDEX);
- for (; sc->rxq.cur != hw;) {
- data = &sc->rxq.data[sc->rxq.cur];
+ for (i = (sc->rx_cur + 1) % IWI_RX_RING_SIZE; i != r;
+ i = (i + 1) % IWI_RX_RING_SIZE) {
- bus_dmamap_sync(sc->sc_dmat, data->map, 0,
+ buf = &sc->rx_buf[i];
+
+ bus_dmamap_sync(sc->sc_dmat, buf->map, 0,
sizeof (struct iwi_hdr), BUS_DMASYNC_POSTREAD);
- hdr = mtod(data->m, struct iwi_hdr *);
+ hdr = mtod(buf->m, struct iwi_hdr *);
switch (hdr->type) {
case IWI_HDR_TYPE_FRAME:
- iwi_frame_intr(sc, data,
+ iwi_frame_intr(sc, buf, i,
(struct iwi_frame *)(hdr + 1));
break;
case IWI_HDR_TYPE_NOTIF:
- iwi_notification_intr(sc, data,
+ iwi_notification_intr(sc, buf,
(struct iwi_notif *)(hdr + 1));
break;
@@ -1141,41 +1007,45 @@ iwi_rx_intr(struct iwi_softc *sc)
printf("%s: unknown hdr type %u\n",
sc->sc_dev.dv_xname, hdr->type);
}
-
- sc->rxq.cur = (sc->rxq.cur + 1) % IWI_RX_RING_COUNT;
}
/* tell the firmware what we have processed */
- hw = (hw == 0) ? IWI_RX_RING_COUNT - 1 : hw - 1;
- CSR_WRITE_4(sc, IWI_CSR_RX_WIDX, hw);
+ sc->rx_cur = (r == 0) ? IWI_RX_RING_SIZE - 1 : r - 1;
+ CSR_WRITE_4(sc, IWI_CSR_RX_WRITE_INDEX, sc->rx_cur);
}
void
-iwi_tx_intr(struct iwi_softc *sc, struct iwi_tx_ring *txq)
+iwi_tx_intr(struct iwi_softc *sc)
{
struct ieee80211com *ic = &sc->sc_ic;
struct ifnet *ifp = &ic->ic_if;
- struct iwi_tx_data *data;
- uint32_t hw;
+ struct iwi_tx_buf *buf;
+ uint32_t r, i;
+
+ r = CSR_READ_4(sc, IWI_CSR_TX1_READ_INDEX);
- hw = CSR_READ_4(sc, txq->csr_ridx);
+ for (i = (sc->tx_old + 1) % IWI_TX_RING_SIZE; i != r;
+ i = (i + 1) % IWI_TX_RING_SIZE) {
- for (; txq->next != hw;) {
- data = &txq->data[txq->next];
+ buf = &sc->tx_buf[i];
- bus_dmamap_unload(sc->sc_dmat, data->map);
- m_freem(data->m);
- data->m = NULL;
- ieee80211_release_node(ic, data->ni);
- data->ni = NULL;
+ bus_dmamap_unload(sc->sc_dmat, buf->map);
+ m_freem(buf->m);
+ buf->m = NULL;
+ ieee80211_release_node(ic, buf->ni);
+ buf->ni = NULL;
ifp->if_opackets++;
+ sc->tx_queued--;
- txq->queued--;
- txq->next = (txq->next + 1) % IWI_TX_RING_COUNT;
+ /* kill watchdog timer */
+ sc->sc_tx_timer = 0;
}
- sc->sc_tx_timer = 0;
+ /* remember what the firmware has processed */
+ sc->tx_old = (r == 0) ? IWI_TX_RING_SIZE - 1 : r - 1;
+
+ /* call start() since some buffer descriptors have been released */
ifp->if_flags &= ~IFF_OACTIVE;
(*ifp->if_start)(ifp);
}
@@ -1192,13 +1062,11 @@ iwi_intr(void *arg)
/* disable interrupts */
CSR_WRITE_4(sc, IWI_CSR_INTR_MASK, 0);
- /* acknowledge interrupts */
- CSR_WRITE_4(sc, IWI_CSR_INTR, r);
+ DPRINTFN(8, ("INTR!0x%08x\n", r));
if (r & (IWI_INTR_FATAL_ERROR | IWI_INTR_PARITY_ERROR)) {
printf("%s: fatal error\n", sc->sc_dev.dv_xname);
iwi_stop(&sc->sc_ic.ic_if, 1);
- r = 0; /* don't process more interrupts */
}
if (r & IWI_INTR_FW_INITED) {
@@ -1209,26 +1077,19 @@ iwi_intr(void *arg)
if (r & IWI_INTR_RADIO_OFF) {
DPRINTF(("radio transmitter off\n"));
iwi_stop(&sc->sc_ic.ic_if, 1);
- r = 0; /* don't process more interrupts */
}
- if (r & IWI_INTR_CMD_DONE)
- wakeup(sc);
-
- if (r & IWI_INTR_TX1_DONE)
- iwi_tx_intr(sc, &sc->txq[0]);
-
- if (r & IWI_INTR_TX2_DONE)
- iwi_tx_intr(sc, &sc->txq[1]);
+ if (r & IWI_INTR_RX_TRANSFER)
+ iwi_rx_intr(sc);
- if (r & IWI_INTR_TX3_DONE)
- iwi_tx_intr(sc, &sc->txq[2]);
+ if (r & IWI_INTR_CMD_TRANSFER)
+ wakeup(sc);
- if (r & IWI_INTR_TX4_DONE)
- iwi_tx_intr(sc, &sc->txq[3]);
+ if (r & IWI_INTR_TX1_TRANSFER)
+ iwi_tx_intr(sc);
- if (r & IWI_INTR_RX_DONE)
- iwi_rx_intr(sc);
+ /* acknowledge interrupts */
+ CSR_WRITE_4(sc, IWI_CSR_INTR, r);
/* re-enable interrupts */
CSR_WRITE_4(sc, IWI_CSR_INTR_MASK, IWI_INTR_MASK);
@@ -1241,22 +1102,21 @@ iwi_cmd(struct iwi_softc *sc, uint8_t type, void *data, uint8_t len, int async)
{
struct iwi_cmd_desc *desc;
- desc = &sc->cmdq.desc[sc->cmdq.cur];
+ DPRINTFN(2, ("TX!CMD!%u!%u\n", type, len));
+
+ desc = &sc->cmd_desc[sc->cmd_cur];
desc->hdr.type = IWI_HDR_TYPE_COMMAND;
desc->hdr.flags = IWI_HDR_FLAG_IRQ;
desc->type = type;
desc->len = len;
bcopy(data, desc->data, len);
- bus_dmamap_sync(sc->sc_dmat, sc->cmdq.map,
- sc->cmdq.cur * sizeof (struct iwi_cmd_desc),
+ bus_dmamap_sync(sc->sc_dmat, sc->cmd_ring_map,
+ sc->cmd_cur * sizeof (struct iwi_cmd_desc),
sizeof (struct iwi_cmd_desc), BUS_DMASYNC_PREWRITE);
- DPRINTFN(2, ("sending command idx=%u type=%u len=%u\n", sc->cmdq.cur,
- type, len));
-
- sc->cmdq.cur = (sc->cmdq.cur + 1) % IWI_CMD_RING_COUNT;
- CSR_WRITE_4(sc, IWI_CSR_CMD_WIDX, sc->cmdq.cur);
+ sc->cmd_cur = (sc->cmd_cur + 1) % IWI_CMD_RING_SIZE;
+ CSR_WRITE_4(sc, IWI_CSR_CMD_WRITE_INDEX, sc->cmd_cur);
return async ? 0 : tsleep(sc, 0, "iwicmd", hz);
}
@@ -1266,9 +1126,8 @@ iwi_tx_start(struct ifnet *ifp, struct mbuf *m0, struct ieee80211_node *ni)
{
struct iwi_softc *sc = ifp->if_softc;
struct ieee80211com *ic = &sc->sc_ic;
- struct iwi_tx_data *data;
+ struct iwi_tx_buf *buf;
struct iwi_tx_desc *desc;
- struct iwi_tx_ring *txq = &sc->txq[0];
struct mbuf *mnew;
int error, i, station = 0;
@@ -1290,8 +1149,8 @@ iwi_tx_start(struct ifnet *ifp, struct mbuf *m0, struct ieee80211_node *ni)
}
#endif
- data = &txq->data[txq->cur];
- desc = &txq->desc[txq->cur];
+ buf = &sc->tx_buf[sc->tx_cur];
+ desc = &sc->tx_desc[sc->tx_cur];
/* save and trim IEEE802.11 header */
m_copydata(m0, 0, sizeof (struct ieee80211_frame), (caddr_t)&desc->wh);
@@ -1307,8 +1166,7 @@ iwi_tx_start(struct ifnet *ifp, struct mbuf *m0, struct ieee80211_node *ni)
}
}
- error = bus_dmamap_load_mbuf(sc->sc_dmat, data->map, m0,
- BUS_DMA_NOWAIT);
+ error = bus_dmamap_load_mbuf(sc->sc_dmat, buf->map, m0, BUS_DMA_NOWAIT);
if (error != 0 && error != EFBIG) {
printf("%s: could not map mbuf (error %d)\n",
sc->sc_dev.dv_xname, error);
@@ -1339,7 +1197,7 @@ iwi_tx_start(struct ifnet *ifp, struct mbuf *m0, struct ieee80211_node *ni)
mnew->m_len = mnew->m_pkthdr.len;
m0 = mnew;
- error = bus_dmamap_load_mbuf(sc->sc_dmat, data->map, m0,
+ error = bus_dmamap_load_mbuf(sc->sc_dmat, buf->map, m0,
BUS_DMA_NOWAIT);
if (error != 0) {
printf("%s: could not map mbuf (error %d)\n",
@@ -1349,8 +1207,8 @@ iwi_tx_start(struct ifnet *ifp, struct mbuf *m0, struct ieee80211_node *ni)
}
}
- data->m = m0;
- data->ni = ni;
+ buf->m = m0;
+ buf->ni = ni;
desc->hdr.type = IWI_HDR_TYPE_DATA;
desc->hdr.flags = IWI_HDR_FLAG_IRQ;
@@ -1371,25 +1229,25 @@ iwi_tx_start(struct ifnet *ifp, struct mbuf *m0, struct ieee80211_node *ni)
if (ic->ic_flags & IEEE80211_F_SHPREAMBLE)
desc->flags |= IWI_DATA_FLAG_SHPREAMBLE;
- desc->nseg = htole32(data->map->dm_nsegs);
- for (i = 0; i < data->map->dm_nsegs; i++) {
- desc->seg_addr[i] = htole32(data->map->dm_segs[i].ds_addr);
- desc->seg_len[i] = htole16(data->map->dm_segs[i].ds_len);
+ desc->nseg = htole32(buf->map->dm_nsegs);
+ for (i = 0; i < buf->map->dm_nsegs; i++) {
+ desc->seg_addr[i] = htole32(buf->map->dm_segs[i].ds_addr);
+ desc->seg_len[i] = htole16(buf->map->dm_segs[i].ds_len);
}
- bus_dmamap_sync(sc->sc_dmat, txq->map,
- txq->cur * sizeof (struct iwi_tx_desc),
+ bus_dmamap_sync(sc->sc_dmat, sc->tx_ring_map,
+ sc->tx_cur * sizeof (struct iwi_tx_desc),
sizeof (struct iwi_tx_desc), BUS_DMASYNC_PREWRITE);
- bus_dmamap_sync(sc->sc_dmat, data->map, 0, MCLBYTES,
+ bus_dmamap_sync(sc->sc_dmat, buf->map, 0, MCLBYTES,
BUS_DMASYNC_PREWRITE);
- DPRINTFN(5, ("sending data frame idx=%u len=%u nseg=%u\n", txq->cur,
- letoh16(desc->len), data->map->dm_nsegs));
+ DPRINTFN(5, ("TX!DATA!%u!%u\n", letoh16(desc->len), desc->nseg));
- txq->queued++;
- txq->cur = (txq->cur + 1) % IWI_TX_RING_COUNT;
- CSR_WRITE_4(sc, txq->csr_widx, txq->cur);
+ /* inform firmware about this new packet */
+ sc->tx_queued++;
+ sc->tx_cur = (sc->tx_cur + 1) % IWI_TX_RING_SIZE;
+ CSR_WRITE_4(sc, IWI_CSR_TX1_WRITE_INDEX, sc->tx_cur);
return 0;
}
@@ -1410,7 +1268,7 @@ iwi_start(struct ifnet *ifp)
if (m0 == NULL)
break;
- if (sc->txq[0].queued >= IWI_TX_RING_COUNT - 8) {
+ if (sc->tx_queued >= IWI_TX_RING_SIZE - 8) {
IF_PREPEND(&ifp->if_snd, m0);
ifp->if_flags |= IFF_OACTIVE;
break;
@@ -1556,10 +1414,8 @@ iwi_stop_master(struct iwi_softc *sc)
break;
DELAY(10);
}
- if (ntries == 5) {
- printf("%s: timeout waiting for master\n",
- sc->sc_dev.dv_xname);
- }
+ if (ntries == 5)
+ printf("%s: timeout waiting for master\n", sc->sc_dev.dv_xname);
CSR_WRITE_4(sc, IWI_CSR_RST, CSR_READ_4(sc, IWI_CSR_RST) |
IWI_RST_PRINCETON_RESET);
@@ -1578,6 +1434,7 @@ iwi_reset(struct iwi_softc *sc)
CSR_WRITE_4(sc, IWI_CSR_CTL, CSR_READ_4(sc, IWI_CSR_CTL) |
IWI_CTL_INIT);
+ /* initialize phase-locked level (PLL) */
CSR_WRITE_4(sc, IWI_CSR_READ_INT, IWI_READ_INT_INIT_HOST);
/* wait for clock stabilization */
@@ -1586,11 +1443,8 @@ iwi_reset(struct iwi_softc *sc)
break;
DELAY(200);
}
- if (ntries == 1000) {
- printf("%s: timeout waiting for clock stabilization\n",
- sc->sc_dev.dv_xname);
+ if (ntries == 1000)
return EIO;
- }
CSR_WRITE_4(sc, IWI_CSR_RST, CSR_READ_4(sc, IWI_CSR_RST) |
IWI_RST_SW_RESET);
@@ -1609,10 +1463,27 @@ iwi_reset(struct iwi_softc *sc)
}
int
-iwi_load_ucode(struct iwi_softc *sc, const char *data, int size)
+iwi_load_ucode(struct iwi_softc *sc, const char *name)
{
- const uint16_t *w;
- int ntries, i;
+ u_char *uc, *data;
+ size_t size;
+ uint16_t *w;
+ int error, ntries, i;
+
+ if ((error = loadfirmware(name, &data, &size)) != 0) {
+ printf("%s: could not read ucode %s, error %d\n",
+ sc->sc_dev.dv_xname, name, error);
+ goto fail1;
+ }
+
+ if (size < sizeof (struct iwi_firmware_hdr)) {
+ error = EINVAL;
+ goto fail2;
+ }
+
+ uc = data;
+ uc += sizeof (struct iwi_firmware_hdr);
+ size -= sizeof (struct iwi_firmware_hdr);
CSR_WRITE_4(sc, IWI_CSR_RST, CSR_READ_4(sc, IWI_CSR_RST) |
IWI_RST_STOP_MASTER);
@@ -1622,17 +1493,15 @@ iwi_load_ucode(struct iwi_softc *sc, const char *data, int size)
DELAY(10);
}
if (ntries == 5) {
- printf("%s: timeout waiting for master\n",
- sc->sc_dev.dv_xname);
- return EIO;
+ printf("%s: timeout waiting for master\n", sc->sc_dev.dv_xname);
+ error = EIO;
+ goto fail2;
}
MEM_WRITE_4(sc, 0x3000e0, 0x80000000);
DELAY(5000);
-
CSR_WRITE_4(sc, IWI_CSR_RST, CSR_READ_4(sc, IWI_CSR_RST) &
~IWI_RST_PRINCETON_RESET);
-
DELAY(5000);
MEM_WRITE_4(sc, 0x3000e0, 0);
DELAY(1000);
@@ -1642,16 +1511,15 @@ iwi_load_ucode(struct iwi_softc *sc, const char *data, int size)
DELAY(1000);
MEM_WRITE_1(sc, 0x200000, 0x00);
MEM_WRITE_1(sc, 0x200000, 0x40);
- DELAY(1000);
/* adapter is buggy, we must set the address for each word */
- for (w = (const uint16_t *)data; size > 0; w++, size -= 2)
+ for (w = (uint16_t *)uc; size > 0; w++, size -= 2)
MEM_WRITE_2(sc, 0x200010, htole16(*w));
MEM_WRITE_1(sc, 0x200000, 0x00);
MEM_WRITE_1(sc, 0x200000, 0x80);
- /* wait until we get an answer */
+ /* wait until we get a response in the uc queue */
for (ntries = 0; ntries < 100; ntries++) {
if (MEM_READ_1(sc, 0x200000) & 1)
break;
@@ -1660,24 +1528,27 @@ iwi_load_ucode(struct iwi_softc *sc, const char *data, int size)
if (ntries == 100) {
printf("%s: timeout waiting for ucode to initialize\n",
sc->sc_dev.dv_xname);
- return EIO;
+ error = EIO;
+ goto fail2;
}
- /* read the answer or the firmware will not initialize properly */
+ /* empty the uc queue or the firmware will not initialize properly */
for (i = 0; i < 7; i++)
MEM_READ_4(sc, 0x200004);
MEM_WRITE_1(sc, 0x200000, 0x00);
- return 0;
+fail2: free(data, M_DEVBUF);
+fail1: return error;
}
/* macro to handle unaligned little endian data in firmware image */
#define GETLE32(p) ((p)[0] | (p)[1] << 8 | (p)[2] << 16 | (p)[3] << 24)
-
int
-iwi_load_firmware(struct iwi_softc *sc, const char *data, int size)
+iwi_load_firmware(struct iwi_softc *sc, const char *name)
{
+ u_char *fw, *data;
+ size_t size;
bus_dmamap_t map;
bus_dma_segment_t seg;
caddr_t virtaddr;
@@ -1685,21 +1556,40 @@ iwi_load_firmware(struct iwi_softc *sc, const char *data, int size)
uint32_t sentinel, ctl, src, dst, sum, len, mlen;
int ntries, nsegs, error;
+ if ((error = loadfirmware(name, &data, &size)) != 0) {
+ printf("%s: could not read firmware %s, error %d\n",
+ sc->sc_dev.dv_xname, name, error);
+ goto fail1;
+ }
+
+ if (size < sizeof (struct iwi_firmware_hdr)) {
+ error = EINVAL;
+ goto fail2;
+ }
+
+ fw = data;
+ fw += sizeof (struct iwi_firmware_hdr);
+ size -= sizeof (struct iwi_firmware_hdr);
+
/* allocate DMA memory for storing firmware image */
error = bus_dmamap_create(sc->sc_dmat, size, 1, size, 0,
BUS_DMA_NOWAIT, &map);
if (error != 0) {
printf("%s: could not create firmware DMA map\n",
sc->sc_dev.dv_xname);
- goto fail1;
+ goto fail2;
}
+ /*
+ * We cannot map fw directly because of some hardware constraints on
+ * the mapping address.
+ */
error = bus_dmamem_alloc(sc->sc_dmat, size, PAGE_SIZE, 0, &seg, 1,
&nsegs, BUS_DMA_NOWAIT);
if (error != 0) {
printf("%s: could allocate firmware DMA memory\n",
sc->sc_dev.dv_xname);
- goto fail2;
+ goto fail3;
}
error = bus_dmamem_map(sc->sc_dmat, &seg, nsegs, size, &virtaddr,
@@ -1707,7 +1597,7 @@ iwi_load_firmware(struct iwi_softc *sc, const char *data, int size)
if (error != 0) {
printf("%s: could not map firmware DMA memory\n",
sc->sc_dev.dv_xname);
- goto fail3;
+ goto fail4;
}
error = bus_dmamap_load(sc->sc_dmat, map, virtaddr, size, NULL,
@@ -1715,11 +1605,11 @@ iwi_load_firmware(struct iwi_softc *sc, const char *data, int size)
if (error != 0) {
printf("%s: could not load firmware DMA map\n",
sc->sc_dev.dv_xname);
- goto fail4;
+ goto fail5;
}
/* copy firmware image to DMA memory */
- bcopy(data, virtaddr, size);
+ bcopy(fw, virtaddr, size);
/* make sure the adapter will get up-to-date values */
bus_dmamap_sync(sc->sc_dmat, map, 0, size, BUS_DMASYNC_PREWRITE);
@@ -1732,7 +1622,7 @@ iwi_load_firmware(struct iwi_softc *sc, const char *data, int size)
* indirections. The adapter will read the firmware image through DMA
* using information stored in command blocks.
*/
- src = map->dm_segs[0].ds_addr;
+ src = map->dm_segs->ds_addr;
p = virtaddr;
end = p + size;
CSR_WRITE_4(sc, IWI_CSR_AUTOINC_ADDR, 0x27000);
@@ -1779,7 +1669,7 @@ iwi_load_firmware(struct iwi_softc *sc, const char *data, int size)
if (ntries == 400) {
printf("%s: timeout processing cb\n", sc->sc_dev.dv_xname);
error = EIO;
- goto fail5;
+ goto fail6;
}
/* we're done with command blocks processing */
@@ -1790,7 +1680,6 @@ iwi_load_firmware(struct iwi_softc *sc, const char *data, int size)
/* tell the adapter to initialize the firmware */
CSR_WRITE_4(sc, IWI_CSR_RST, 0);
-
CSR_WRITE_4(sc, IWI_CSR_CTL, CSR_READ_4(sc, IWI_CSR_CTL) |
IWI_CTL_ALLOW_STANDBY);
@@ -1798,14 +1687,16 @@ iwi_load_firmware(struct iwi_softc *sc, const char *data, int size)
if ((error = tsleep(sc, 0, "iwiinit", hz)) != 0) {
printf("%s: timeout waiting for firmware initialization to "
"complete\n", sc->sc_dev.dv_xname);
- goto fail5;
+ goto fail6;
}
-fail5: bus_dmamap_sync(sc->sc_dmat, map, 0, size, BUS_DMASYNC_POSTWRITE);
+fail6: bus_dmamap_sync(sc->sc_dmat, map, 0, size, BUS_DMASYNC_POSTWRITE);
bus_dmamap_unload(sc->sc_dmat, map);
-fail4: bus_dmamem_unmap(sc->sc_dmat, virtaddr, size);
-fail3: bus_dmamem_free(sc->sc_dmat, &seg, 1);
-fail2: bus_dmamap_destroy(sc->sc_dmat, map);
+fail5: bus_dmamem_unmap(sc->sc_dmat, virtaddr, size);
+fail4: bus_dmamem_free(sc->sc_dmat, &seg, 1);
+fail3: bus_dmamap_destroy(sc->sc_dmat, map);
+fail2: free(data, M_DEVBUF);
+
fail1: return error;
}
@@ -1831,11 +1722,11 @@ iwi_config(struct iwi_softc *sc)
bzero(&config, sizeof config);
config.bluetooth_coexistence = 1;
- config.antenna = 2; /* antenna diversity */
config.multicast_enabled = 1;
config.answer_pbreq = (ic->ic_opmode == IEEE80211_M_IBSS) ? 1 : 0;
DPRINTF(("Configuring adapter\n"));
- error = iwi_cmd(sc, IWI_CMD_SET_CONFIG, &config, sizeof config, 0);
+ error = iwi_cmd(sc, IWI_CMD_SET_CONFIGURATION, &config, sizeof config,
+ 0);
if (error != 0)
return error;
@@ -2019,14 +1910,13 @@ iwi_auth_and_assoc(struct iwi_softc *sc)
/* enable b/g autodection */
bzero(&config, sizeof config);
config.bluetooth_coexistence = 1;
- config.antenna = 2; /* antenna diversity */
config.multicast_enabled = 1;
config.bg_autodetection = 1;
config.answer_pbreq =
(ic->ic_opmode == IEEE80211_M_IBSS) ? 1 : 0;
DPRINTF(("Configuring adapter\n"));
- error = iwi_cmd(sc, IWI_CMD_SET_CONFIG, &config, sizeof config,
- 1);
+ error = iwi_cmd(sc, IWI_CMD_SET_CONFIGURATION, &config,
+ sizeof config, 1);
if (error != 0)
return error;
}
@@ -2096,112 +1986,90 @@ iwi_init(struct ifnet *ifp)
{
struct iwi_softc *sc = ifp->if_softc;
struct ieee80211com *ic = &sc->sc_ic;
- struct iwi_firmware_hdr *hdr;
- const char *name, *fw;
- u_char *data;
- size_t size;
+ const char *ucode, *main;
int i, error;
if ((error = iwi_reset(sc)) != 0) {
printf("%s: could not reset adapter\n", sc->sc_dev.dv_xname);
- goto fail1;
+ goto fail;
+ }
+
+ if ((error = iwi_load_firmware(sc, "iwi-boot")) != 0) {
+ printf("%s: could not load boot firmware\n",
+ sc->sc_dev.dv_xname);
+ goto fail;
}
switch (sc->sc_ic.ic_opmode) {
case IEEE80211_M_STA:
case IEEE80211_M_HOSTAP:
- name = "iwi-bss";
+ ucode = "iwi-ucode-bss";
+ main = "iwi-bss";
break;
+
case IEEE80211_M_IBSS:
case IEEE80211_M_AHDEMO:
- name = "iwi-ibss";
+ ucode = "iwi-ucode-ibss";
+ main = "iwi-ibss";
break;
+
case IEEE80211_M_MONITOR:
- name = "iwi-monitor";
+ ucode = "iwi-ucode-monitor";
+ main = "iwi-monitor";
break;
- default:
- name = NULL; /* should not get there */
}
- if ((error = loadfirmware(name, &data, &size)) != 0) {
- printf("%s: could not read firmware %s\n",
- sc->sc_dev.dv_xname, name);
- goto fail1;
- }
-
- if (size < sizeof (struct iwi_firmware_hdr)) {
- printf("%s: firmware image too short: %zu bytes\n",
- sc->sc_dev.dv_xname, size);
- goto fail2;
- }
-
- hdr = (struct iwi_firmware_hdr *)data;
-
- if (size < sizeof (struct iwi_firmware_hdr) + letoh32(hdr->bootsz) +
- letoh32(hdr->ucodesz) + letoh32(hdr->mainsz)) {
- printf("%s: firmware image too short: %zu bytes\n",
- sc->sc_dev.dv_xname, size);
- goto fail2;
- }
-
- fw = (const char *)data + sizeof (struct iwi_firmware_hdr);
- if ((error = iwi_load_firmware(sc, fw, letoh32(hdr->bootsz))) != 0) {
- printf("%s: could not load boot firmware\n",
- sc->sc_dev.dv_xname);
- goto fail2;
- }
-
- fw = (const char *)data + sizeof (struct iwi_firmware_hdr) +
- letoh32(hdr->bootsz);
- if ((error = iwi_load_ucode(sc, fw, letoh32(hdr->ucodesz))) != 0) {
+ if ((error = iwi_load_ucode(sc, ucode)) != 0) {
printf("%s: could not load microcode\n", sc->sc_dev.dv_xname);
- goto fail2;
+ goto fail;
}
iwi_stop_master(sc);
- CSR_WRITE_4(sc, IWI_CSR_CMD_BASE, sc->cmdq.map->dm_segs[0].ds_addr);
- CSR_WRITE_4(sc, IWI_CSR_CMD_SIZE, IWI_CMD_RING_COUNT);
- CSR_WRITE_4(sc, IWI_CSR_CMD_WIDX, sc->cmdq.cur);
+ sc->tx_cur = 0;
+ sc->tx_queued = 0;
+ sc->tx_old = IWI_TX_RING_SIZE - 1;
+ sc->cmd_cur = 0;
+ sc->rx_cur = IWI_RX_RING_SIZE - 1;
- CSR_WRITE_4(sc, IWI_CSR_TX1_BASE, sc->txq[0].map->dm_segs[0].ds_addr);
- CSR_WRITE_4(sc, IWI_CSR_TX1_SIZE, IWI_TX_RING_COUNT);
- CSR_WRITE_4(sc, IWI_CSR_TX1_WIDX, sc->txq[0].cur);
+ CSR_WRITE_4(sc, IWI_CSR_CMD_BASE, sc->cmd_ring_map->dm_segs->ds_addr);
+ CSR_WRITE_4(sc, IWI_CSR_CMD_SIZE, IWI_CMD_RING_SIZE);
+ CSR_WRITE_4(sc, IWI_CSR_CMD_WRITE_INDEX, sc->cmd_cur);
- CSR_WRITE_4(sc, IWI_CSR_TX2_BASE, sc->txq[1].map->dm_segs[0].ds_addr);
- CSR_WRITE_4(sc, IWI_CSR_TX2_SIZE, IWI_TX_RING_COUNT);
- CSR_WRITE_4(sc, IWI_CSR_TX2_WIDX, sc->txq[1].cur);
+ CSR_WRITE_4(sc, IWI_CSR_TX1_BASE, sc->tx_ring_map->dm_segs->ds_addr);
+ CSR_WRITE_4(sc, IWI_CSR_TX1_SIZE, IWI_TX_RING_SIZE);
+ CSR_WRITE_4(sc, IWI_CSR_TX1_WRITE_INDEX, sc->tx_cur);
- CSR_WRITE_4(sc, IWI_CSR_TX3_BASE, sc->txq[2].map->dm_segs[0].ds_addr);
- CSR_WRITE_4(sc, IWI_CSR_TX3_SIZE, IWI_TX_RING_COUNT);
- CSR_WRITE_4(sc, IWI_CSR_TX3_WIDX, sc->txq[2].cur);
+ CSR_WRITE_4(sc, IWI_CSR_TX2_BASE, sc->tx_ring_map->dm_segs->ds_addr);
+ CSR_WRITE_4(sc, IWI_CSR_TX2_SIZE, IWI_TX_RING_SIZE);
+ CSR_WRITE_4(sc, IWI_CSR_TX2_WRITE_INDEX, sc->tx_cur);
- CSR_WRITE_4(sc, IWI_CSR_TX4_BASE, sc->txq[3].map->dm_segs[0].ds_addr);
- CSR_WRITE_4(sc, IWI_CSR_TX4_SIZE, IWI_TX_RING_COUNT);
- CSR_WRITE_4(sc, IWI_CSR_TX4_WIDX, sc->txq[3].cur);
+ CSR_WRITE_4(sc, IWI_CSR_TX3_BASE, sc->tx_ring_map->dm_segs->ds_addr);
+ CSR_WRITE_4(sc, IWI_CSR_TX3_SIZE, IWI_TX_RING_SIZE);
+ CSR_WRITE_4(sc, IWI_CSR_TX3_WRITE_INDEX, sc->tx_cur);
- for (i = 0; i < IWI_RX_RING_COUNT; i++) {
- struct iwi_rx_data *data = &sc->rxq.data[i];
- CSR_WRITE_4(sc, data->reg, data->map->dm_segs[0].ds_addr);
- }
+ CSR_WRITE_4(sc, IWI_CSR_TX4_BASE, sc->tx_ring_map->dm_segs->ds_addr);
+ CSR_WRITE_4(sc, IWI_CSR_TX4_SIZE, IWI_TX_RING_SIZE);
+ CSR_WRITE_4(sc, IWI_CSR_TX4_WRITE_INDEX, sc->tx_cur);
- CSR_WRITE_4(sc, IWI_CSR_RX_WIDX, IWI_RX_RING_COUNT - 1);
+ for (i = 0; i < IWI_RX_RING_SIZE; i++)
+ CSR_WRITE_4(sc, IWI_CSR_RX_BASE + i * 4,
+ sc->rx_buf[i].map->dm_segs->ds_addr);
- fw = (const char *)data + sizeof (struct iwi_firmware_hdr) +
- letoh32(hdr->bootsz) + letoh32(hdr->ucodesz);
- if ((error = iwi_load_firmware(sc, fw, letoh32(hdr->mainsz))) != 0) {
+ CSR_WRITE_4(sc, IWI_CSR_RX_WRITE_INDEX, sc->rx_cur);
+
+ if ((error = iwi_load_firmware(sc, main)) != 0) {
printf("%s: could not load main firmware\n",
sc->sc_dev.dv_xname);
- goto fail2;
+ goto fail;
}
- free(data, M_DEVBUF);
sc->flags |= IWI_FLAG_FW_INITED;
if ((error = iwi_config(sc)) != 0) {
printf("%s: device configuration failed\n",
sc->sc_dev.dv_xname);
- goto fail1;
+ goto fail;
}
if (ic->ic_opmode != IEEE80211_M_MONITOR)
@@ -2214,8 +2082,8 @@ iwi_init(struct ifnet *ifp)
return 0;
-fail2: free(data, M_DEVBUF);
-fail1: iwi_stop(ifp, 0);
+fail: iwi_stop(ifp, 0);
+
return error;
}
@@ -2224,6 +2092,8 @@ iwi_stop(struct ifnet *ifp, int disable)
{
struct iwi_softc *sc = ifp->if_softc;
struct ieee80211com *ic = &sc->sc_ic;
+ struct iwi_tx_buf *buf;
+ int i;
sc->sc_tx_timer = 0;
ifp->if_timer = 0;
@@ -2232,16 +2102,21 @@ iwi_stop(struct ifnet *ifp, int disable)
ieee80211_new_state(ic, IEEE80211_S_INIT, -1);
iwi_stop_master(sc);
-
CSR_WRITE_4(sc, IWI_CSR_RST, IWI_RST_SW_RESET);
- /* reset rings */
- iwi_reset_cmd_ring(sc, &sc->cmdq);
- iwi_reset_tx_ring(sc, &sc->txq[0]);
- iwi_reset_tx_ring(sc, &sc->txq[1]);
- iwi_reset_tx_ring(sc, &sc->txq[2]);
- iwi_reset_tx_ring(sc, &sc->txq[3]);
- iwi_reset_rx_ring(sc, &sc->rxq);
+ /*
+ * Release Tx buffers.
+ */
+ for (i = 0; i < IWI_TX_RING_SIZE; i++) {
+ buf = &sc->tx_buf[i];
+
+ if (buf->m != NULL) {
+ bus_dmamap_unload(sc->sc_dmat, buf->map);
+ m_freem(buf->m);
+ buf->m = NULL;
+ }
+ buf->ni = NULL;
+ }
}
struct cfdriver iwi_cd = {
diff --git a/sys/dev/pci/if_iwireg.h b/sys/dev/pci/if_iwireg.h
index 99fe34ee0d8..e01947ad329 100644
--- a/sys/dev/pci/if_iwireg.h
+++ b/sys/dev/pci/if_iwireg.h
@@ -1,4 +1,4 @@
-/* $OpenBSD: if_iwireg.h,v 1.21 2006/03/27 20:46:35 damien Exp $ */
+/* $OpenBSD: if_iwireg.h,v 1.22 2006/03/31 17:18:37 pedro Exp $ */
/*-
* Copyright (c) 2004-2006
@@ -27,9 +27,9 @@
* SUCH DAMAGE.
*/
-#define IWI_CMD_RING_COUNT 16
-#define IWI_TX_RING_COUNT 64
-#define IWI_RX_RING_COUNT 32
+#define IWI_TX_RING_SIZE 64
+#define IWI_CMD_RING_SIZE 16
+#define IWI_RX_RING_SIZE 32
#define IWI_CSR_INTR 0x0008
#define IWI_CSR_INTR_MASK 0x000c
@@ -50,42 +50,41 @@
#define IWI_CSR_TX3_SIZE 0x021c
#define IWI_CSR_TX4_BASE 0x0220
#define IWI_CSR_TX4_SIZE 0x0224
-#define IWI_CSR_CMD_RIDX 0x0280
-#define IWI_CSR_TX1_RIDX 0x0284
-#define IWI_CSR_TX2_RIDX 0x0288
-#define IWI_CSR_TX3_RIDX 0x028c
-#define IWI_CSR_TX4_RIDX 0x0290
-#define IWI_CSR_RX_RIDX 0x02a0
+#define IWI_CSR_CMD_READ_INDEX 0x0280
+#define IWI_CSR_TX1_READ_INDEX 0x0284
+#define IWI_CSR_TX2_READ_INDEX 0x0288
+#define IWI_CSR_TX3_READ_INDEX 0x028c
+#define IWI_CSR_TX4_READ_INDEX 0x0290
+#define IWI_CSR_RX_READ_INDEX 0x02a0
#define IWI_CSR_RX_BASE 0x0500
#define IWI_CSR_TABLE0_SIZE 0x0700
#define IWI_CSR_TABLE0_BASE 0x0704
#define IWI_CSR_NODE_BASE 0x0c0c
-#define IWI_CSR_CMD_WIDX 0x0f80
-#define IWI_CSR_TX1_WIDX 0x0f84
-#define IWI_CSR_TX2_WIDX 0x0f88
-#define IWI_CSR_TX3_WIDX 0x0f8c
-#define IWI_CSR_TX4_WIDX 0x0f90
-#define IWI_CSR_RX_WIDX 0x0fa0
-#define IWI_CSR_READ_INT 0x0ff4
-
-/* aliases */
#define IWI_CSR_CURRENT_TX_RATE IWI_CSR_TABLE0_BASE
+#define IWI_CSR_CMD_WRITE_INDEX 0x0f80
+#define IWI_CSR_TX1_WRITE_INDEX 0x0f84
+#define IWI_CSR_TX2_WRITE_INDEX 0x0f88
+#define IWI_CSR_TX3_WRITE_INDEX 0x0f8c
+#define IWI_CSR_TX4_WRITE_INDEX 0x0f90
+#define IWI_CSR_RX_WRITE_INDEX 0x0fa0
+#define IWI_CSR_READ_INT 0x0ff4
/* possible flags for IWI_CSR_INTR */
-#define IWI_INTR_RX_DONE 0x00000002
-#define IWI_INTR_CMD_DONE 0x00000800
-#define IWI_INTR_TX1_DONE 0x00001000
-#define IWI_INTR_TX2_DONE 0x00002000
-#define IWI_INTR_TX3_DONE 0x00004000
-#define IWI_INTR_TX4_DONE 0x00008000
+#define IWI_INTR_RX_TRANSFER 0x00000002
+#define IWI_INTR_CMD_TRANSFER 0x00000800
+#define IWI_INTR_TX1_TRANSFER 0x00001000
+#define IWI_INTR_TX2_TRANSFER 0x00002000
+#define IWI_INTR_TX3_TRANSFER 0x00004000
+#define IWI_INTR_TX4_TRANSFER 0x00008000
#define IWI_INTR_FW_INITED 0x01000000
#define IWI_INTR_RADIO_OFF 0x04000000
#define IWI_INTR_FATAL_ERROR 0x40000000
#define IWI_INTR_PARITY_ERROR 0x80000000
#define IWI_INTR_MASK \
- (IWI_INTR_RX_DONE | IWI_INTR_CMD_DONE | IWI_INTR_TX1_DONE | \
- IWI_INTR_TX2_DONE | IWI_INTR_TX3_DONE | IWI_INTR_TX4_DONE | \
+ (IWI_INTR_RX_TRANSFER | IWI_INTR_CMD_TRANSFER | \
+ IWI_INTR_TX1_TRANSFER | IWI_INTR_TX2_TRANSFER | \
+ IWI_INTR_TX3_TRANSFER | IWI_INTR_TX4_TRANSFER | \
IWI_INTR_FW_INITED | IWI_INTR_RADIO_OFF | \
IWI_INTR_FATAL_ERROR | IWI_INTR_PARITY_ERROR)
@@ -130,9 +129,7 @@
/* firmware binary image header */
struct iwi_firmware_hdr {
uint32_t version;
- uint32_t bootsz;
- uint32_t ucodesz;
- uint32_t mainsz;
+ uint32_t mode;
} __packed;
struct iwi_hdr {
@@ -141,11 +138,9 @@ struct iwi_hdr {
#define IWI_HDR_TYPE_COMMAND 1
#define IWI_HDR_TYPE_NOTIF 3
#define IWI_HDR_TYPE_FRAME 9
-
uint8_t seq;
uint8_t flags;
#define IWI_HDR_FLAG_IRQ 0x04
-
uint8_t reserved;
} __packed;
@@ -159,7 +154,6 @@ struct iwi_notif {
#define IWI_NOTIF_TYPE_BEACON 17
#define IWI_NOTIF_TYPE_CALIBRATION 20
#define IWI_NOTIF_TYPE_NOISE 25
-
uint8_t flags;
uint16_t len;
} __packed;
@@ -176,7 +170,6 @@ struct iwi_notif_association {
uint8_t state;
#define IWI_DEASSOCIATED 0
#define IWI_ASSOCIATED 12
-
struct ieee80211_frame frame;
uint16_t capinfo;
uint16_t status;
@@ -222,7 +215,6 @@ struct iwi_tx_desc {
uint8_t reserved2[3];
uint8_t cmd;
#define IWI_DATA_CMD_TX 0x0b
-
uint8_t seq;
uint16_t len;
uint8_t priority;
@@ -230,7 +222,6 @@ struct iwi_tx_desc {
#define IWI_DATA_FLAG_SHPREAMBLE 0x04
#define IWI_DATA_FLAG_NO_WEP 0x20
#define IWI_DATA_FLAG_NEED_ACK 0x80
-
uint8_t xflags;
uint8_t wep_txkey;
uint8_t wepkey[IEEE80211_KEYBUF_SIZE];
@@ -243,7 +234,6 @@ struct iwi_tx_desc {
uint32_t nseg;
#define IWI_MAX_NSEG 6
-#define IWI_MAX_SCATTER (IWI_MAX_NSEG - 2)
uint32_t seg_addr[IWI_MAX_NSEG];
uint16_t seg_len[IWI_MAX_NSEG];
@@ -253,22 +243,21 @@ struct iwi_tx_desc {
struct iwi_cmd_desc {
struct iwi_hdr hdr;
uint8_t type;
-#define IWI_CMD_ENABLE 2
-#define IWI_CMD_SET_CONFIG 6
-#define IWI_CMD_SET_ESSID 8
-#define IWI_CMD_SET_MAC_ADDRESS 11
-#define IWI_CMD_SET_RTS_THRESHOLD 15
-#define IWI_CMD_SET_FRAG_THRESHOLD 16
-#define IWI_CMD_SET_POWER_MODE 17
-#define IWI_CMD_SET_WEP_KEY 18
-#define IWI_CMD_ASSOCIATE 21
-#define IWI_CMD_SET_RATES 22
-#define IWI_CMD_SCAN 26
-#define IWI_CMD_DISABLE 33
-#define IWI_CMD_SET_IV 34
-#define IWI_CMD_SET_TX_POWER 35
-#define IWI_CMD_SET_SENSITIVITY 42
-
+#define IWI_CMD_ENABLE 2
+#define IWI_CMD_SET_CONFIGURATION 6
+#define IWI_CMD_SET_ESSID 8
+#define IWI_CMD_SET_MAC_ADDRESS 11
+#define IWI_CMD_SET_RTS_THRESHOLD 15
+#define IWI_CMD_SET_FRAG_THRESHOLD 16
+#define IWI_CMD_SET_POWER_MODE 17
+#define IWI_CMD_SET_WEP_KEY 18
+#define IWI_CMD_ASSOCIATE 21
+#define IWI_CMD_SET_RATES 22
+#define IWI_CMD_SCAN 26
+#define IWI_CMD_DISABLE 33
+#define IWI_CMD_SET_IV 34
+#define IWI_CMD_SET_TX_POWER 35
+#define IWI_CMD_SET_SENSITIVITY 42
uint8_t len;
uint16_t reserved;
uint8_t data[120];
@@ -285,6 +274,9 @@ struct iwi_node {
#define IWI_MODE_11B 1
#define IWI_MODE_11G 2
+/* macro for command IWI_CMD_SET_SENSITIVITY */
+#define IWI_RSSIDBM2RAW(rssi) ((rssi) - 112)
+
/* possible values for command IWI_CMD_SET_POWER_MODE */
#define IWI_POWER_MODE_CAM 0
@@ -295,7 +287,6 @@ struct iwi_rateset {
uint8_t type;
#define IWI_RATESET_TYPE_NEGOTIATED 0
#define IWI_RATESET_TYPE_SUPPORTED 1
-
uint8_t reserved;
uint8_t rates[12];
} __packed;
@@ -319,7 +310,6 @@ struct iwi_associate {
#define IWI_AUTH_OPEN 0
#define IWI_AUTH_SHARED 1
#define IWI_AUTH_NONE 3
-
uint8_t type;
uint8_t reserved1;
uint16_t reserved2;
@@ -384,7 +374,6 @@ struct iwi_configuration {
struct iwi_wep_key {
uint8_t cmd;
#define IWI_WEP_KEY_CMD_SETKEY 0x08
-
uint8_t seq;
uint8_t idx;
uint8_t len;
diff --git a/sys/dev/pci/if_iwivar.h b/sys/dev/pci/if_iwivar.h
index 3218973f381..ed0eb4b8a79 100644
--- a/sys/dev/pci/if_iwivar.h
+++ b/sys/dev/pci/if_iwivar.h
@@ -1,4 +1,4 @@
-/* $OpenBSD: if_iwivar.h,v 1.14 2006/03/27 20:46:35 damien Exp $ */
+/* $OpenBSD: if_iwivar.h,v 1.15 2006/03/31 17:18:37 pedro Exp $ */
/*-
* Copyright (c) 2004-2006
@@ -55,45 +55,6 @@ struct iwi_tx_radiotap_header {
((1 << IEEE80211_RADIOTAP_FLAGS) | \
(1 << IEEE80211_RADIOTAP_CHANNEL))
-
-struct iwi_cmd_ring {
- bus_dmamap_t map;
- bus_dma_segment_t seg;
- struct iwi_cmd_desc desc[IWI_CMD_RING_COUNT];
- int queued;
- int cur;
- int next;
-};
-
-struct iwi_tx_data {
- bus_dmamap_t map;
- struct mbuf *m;
- struct ieee80211_node *ni;
-};
-
-struct iwi_tx_ring {
- bus_dmamap_t map;
- bus_dma_segment_t seg;
- bus_addr_t csr_ridx;
- bus_addr_t csr_widx;
- struct iwi_tx_desc *desc;
- struct iwi_tx_data data[IWI_TX_RING_COUNT];
- int queued;
- int cur;
- int next;
-};
-
-struct iwi_rx_data {
- bus_dmamap_t map;
- struct mbuf *m;
- uint32_t reg;
-};
-
-struct iwi_rx_ring {
- struct iwi_rx_data data[IWI_RX_RING_COUNT];
- int cur;
-};
-
struct iwi_softc {
struct device sc_dev;
@@ -106,9 +67,31 @@ struct iwi_softc {
bus_dma_tag_t sc_dmat;
- struct iwi_cmd_ring cmdq;
- struct iwi_tx_ring txq[4];
- struct iwi_rx_ring rxq;
+ struct iwi_tx_desc *tx_desc;
+ bus_dmamap_t tx_ring_map;
+ bus_dma_segment_t tx_ring_seg;
+
+ struct iwi_tx_buf {
+ bus_dmamap_t map;
+ struct mbuf *m;
+ struct ieee80211_node *ni;
+ } tx_buf[IWI_TX_RING_SIZE];
+
+ int tx_cur;
+ int tx_old;
+ int tx_queued;
+
+ struct iwi_cmd_desc *cmd_desc;
+ bus_dmamap_t cmd_ring_map;
+ bus_dma_segment_t cmd_ring_seg;
+ int cmd_cur;
+
+ struct iwi_rx_buf {
+ bus_dmamap_t map;
+ struct mbuf *m;
+ } rx_buf[IWI_RX_RING_SIZE];
+
+ int rx_cur;
#define IWI_MAX_NODE 32
uint8_t sta[IWI_MAX_NODE][IEEE80211_ADDR_LEN];