summaryrefslogtreecommitdiff
path: root/sys/dev/pci/if_iwn.c
diff options
context:
space:
mode:
Diffstat (limited to 'sys/dev/pci/if_iwn.c')
-rw-r--r--sys/dev/pci/if_iwn.c130
1 files changed, 123 insertions, 7 deletions
diff --git a/sys/dev/pci/if_iwn.c b/sys/dev/pci/if_iwn.c
index 4347d4c19b9..e60825b5877 100644
--- a/sys/dev/pci/if_iwn.c
+++ b/sys/dev/pci/if_iwn.c
@@ -1,4 +1,4 @@
-/* $OpenBSD: if_iwn.c,v 1.37 2008/11/22 08:23:52 brad Exp $ */
+/* $OpenBSD: if_iwn.c,v 1.38 2008/11/25 17:06:27 damien Exp $ */
/*-
* Copyright (c) 2007, 2008
@@ -793,6 +793,7 @@ iwn_dma_contig_alloc(bus_dma_tag_t tag, struct iwn_dma_info *dma, void **kvap,
goto fail;
memset(dma->vaddr, 0, size);
+ bus_dmamap_sync(tag, dma->map, 0, size, BUS_DMASYNC_PREWRITE);
dma->paddr = dma->map->dm_segs[0].ds_addr;
if (kvap != NULL)
@@ -809,6 +810,8 @@ iwn_dma_contig_free(struct iwn_dma_info *dma)
{
if (dma->map != NULL) {
if (dma->vaddr != NULL) {
+ bus_dmamap_sync(dma->tag, dma->map, 0, dma->size,
+ BUS_DMASYNC_POSTREAD | BUS_DMASYNC_POSTWRITE);
bus_dmamap_unload(dma->tag, dma->map);
bus_dmamem_unmap(dma->tag, dma->vaddr, dma->size);
bus_dmamem_free(dma->tag, &dma->seg, 1);
@@ -978,6 +981,10 @@ iwn_alloc_rx_ring(struct iwn_softc *sc, struct iwn_rx_ring *ring)
/* Set physical address of RX buffer (256-byte aligned.) */
ring->desc[i] = htole32(rbuf->paddr >> 8);
}
+
+ bus_dmamap_sync(sc->sc_dmat, ring->desc_dma.map,
+ 0, ring->desc_dma.size, BUS_DMASYNC_PREWRITE);
+
return 0;
fail: iwn_free_rx_ring(sc, ring);
@@ -1098,6 +1105,8 @@ iwn_reset_tx_ring(struct iwn_softc *sc, struct iwn_tx_ring *ring)
struct iwn_tx_data *data = &ring->data[i];
if (data->m != NULL) {
+ bus_dmamap_sync(sc->sc_dmat, data->map, 0,
+ data->map->dm_mapsize, BUS_DMASYNC_POSTWRITE);
bus_dmamap_unload(sc->sc_dmat, data->map);
m_freem(data->m);
data->m = NULL;
@@ -1105,6 +1114,8 @@ iwn_reset_tx_ring(struct iwn_softc *sc, struct iwn_tx_ring *ring)
}
/* Clear TX descriptors. */
memset(ring->desc, 0, ring->desc_dma.size);
+ bus_dmamap_sync(sc->sc_dmat, ring->desc_dma.map, 0,
+ ring->desc_dma.size, BUS_DMASYNC_PREWRITE);
sc->qfullmsk &= ~(1 << ring->qid);
ring->queued = 0;
ring->cur = 0;
@@ -1122,6 +1133,8 @@ iwn_free_tx_ring(struct iwn_softc *sc, struct iwn_tx_ring *ring)
struct iwn_tx_data *data = &ring->data[i];
if (data->m != NULL) {
+ bus_dmamap_sync(sc->sc_dmat, data->map, 0,
+ data->map->dm_mapsize, BUS_DMASYNC_POSTWRITE);
bus_dmamap_unload(sc->sc_dmat, data->map);
m_freem(data->m);
}
@@ -1519,11 +1532,14 @@ iwn_ccmp_decap(struct iwn_softc *sc, struct mbuf *m, struct ieee80211_key *k)
void
iwn_rx_phy(struct iwn_softc *sc, struct iwn_rx_desc *desc)
{
- struct iwn_rx_stat *stat;
+ struct iwn_rx_stat *stat = (struct iwn_rx_stat *)(desc + 1);
DPRINTFN(2, ("received PHY stats\n"));
+ bus_dmamap_sync(sc->sc_dmat, sc->rxq.buf_dma.map,
+ (caddr_t)stat - sc->rxq.buf_dma.vaddr, sizeof (*stat),
+ BUS_DMASYNC_POSTREAD);
+
/* Save RX statistics, they will be used on MPDU_RX_DONE. */
- stat = (struct iwn_rx_stat *)(desc + 1);
memcpy(&sc->last_rx_stat, stat, sizeof (*stat));
sc->last_rx_valid = 1;
}
@@ -1562,6 +1578,10 @@ iwn_rx_done(struct iwn_softc *sc, struct iwn_rx_desc *desc,
} else
stat = (struct iwn_rx_stat *)(desc + 1);
+ bus_dmamap_sync(sc->sc_dmat, ring->buf_dma.map,
+ (caddr_t)(desc + 1) - ring->buf_dma.vaddr, IWN_RBUF_SIZE,
+ BUS_DMASYNC_POSTREAD);
+
if (stat->cfg_phy_len > IWN_STAT_MAXLEN) {
printf("%s: invalid RX statistic header\n",
sc->sc_dev.dv_xname);
@@ -1649,6 +1669,9 @@ iwn_rx_done(struct iwn_softc *sc, struct iwn_rx_desc *desc,
/* Update RX descriptor. */
ring->desc[ring->cur] = htole32(rbuf->paddr >> 8);
+ bus_dmamap_sync(sc->sc_dmat, ring->desc_dma.map,
+ ring->cur * sizeof (uint32_t), sizeof (uint32_t),
+ BUS_DMASYNC_PREWRITE);
} else {
/* No free rbufs, copy frame into an mbuf. */
m = m_copym2(m, 0, M_COPYALL, M_DONTWAIT);
@@ -1733,6 +1756,11 @@ iwn5000_rx_calib_results(struct iwn_softc *sc, struct iwn_rx_desc *desc)
if (!(sc->sc_flags & IWN_FLAG_FIRST_BOOT))
return;
+ len = (letoh32(desc->len) & 0x3fff) - 4;
+ bus_dmamap_sync(sc->sc_dmat, sc->rxq.buf_dma.map,
+ (caddr_t)calib - sc->rxq.buf_dma.vaddr, len,
+ BUS_DMASYNC_POSTREAD);
+
switch (calib->code) {
case IWN5000_PHY_CALIB_LO:
idx = 0;
@@ -1753,7 +1781,6 @@ iwn5000_rx_calib_results(struct iwn_softc *sc, struct iwn_rx_desc *desc)
/* Save calibration result. */
if (sc->calibcmd[idx].buf != NULL)
free(sc->calibcmd[idx].buf, M_DEVBUF);
- len = (letoh32(desc->len) & 0x3fff) - 4;
sc->calibcmd[idx].buf = malloc(len, M_DEVBUF, M_NOWAIT);
if (sc->calibcmd[idx].buf == NULL) {
DPRINTF(("not enough memory for calibration result %d\n",
@@ -1783,6 +1810,10 @@ iwn_rx_statistics(struct iwn_softc *sc, struct iwn_rx_desc *desc)
if (ic->ic_state != IEEE80211_S_RUN)
return;
+ bus_dmamap_sync(sc->sc_dmat, sc->rxq.buf_dma.map,
+ (caddr_t)stats - sc->rxq.buf_dma.vaddr, sizeof (*stats),
+ BUS_DMASYNC_POSTREAD);
+
DPRINTFN(3, ("received statistics (cmd=%d)\n", desc->type));
sc->calib_cnt = 0; /* Reset TX power calibration timeout. */
@@ -1828,6 +1859,9 @@ iwn4965_tx_done(struct iwn_softc *sc, struct iwn_rx_desc *desc)
{
struct iwn4965_tx_stat *stat = (struct iwn4965_tx_stat *)(desc + 1);
+ bus_dmamap_sync(sc->sc_dmat, sc->rxq.buf_dma.map,
+ (caddr_t)stat - sc->rxq.buf_dma.vaddr, sizeof (*stat),
+ BUS_DMASYNC_POSTREAD);
iwn_tx_done(sc, desc, stat->retrycnt, letoh32(stat->status) & 0xff);
}
@@ -1838,6 +1872,10 @@ iwn5000_tx_done(struct iwn_softc *sc, struct iwn_rx_desc *desc)
/* Reset TX scheduler slot. */
iwn5000_reset_sched(sc, desc->qid & 0xf, desc->idx);
+
+ bus_dmamap_sync(sc->sc_dmat, sc->rxq.buf_dma.map,
+ (caddr_t)stat - sc->rxq.buf_dma.vaddr, sizeof (*stat),
+ BUS_DMASYNC_POSTREAD);
iwn_tx_done(sc, desc, stat->retrycnt, letoh16(stat->status) & 0xff);
}
@@ -1865,6 +1903,8 @@ iwn_tx_done(struct iwn_softc *sc, struct iwn_rx_desc *desc, int retrycnt,
ifp->if_opackets++;
/* Unmap and free mbuf. */
+ bus_dmamap_sync(sc->sc_dmat, data->map, 0, data->map->dm_mapsize,
+ BUS_DMASYNC_POSTWRITE);
bus_dmamap_unload(sc->sc_dmat, data->map);
m_freem(data->m);
data->m = NULL;
@@ -1898,6 +1938,8 @@ iwn_cmd_done(struct iwn_softc *sc, struct iwn_rx_desc *desc)
/* If the command was mapped in an mbuf, free it. */
if (data->m != NULL) {
+ bus_dmamap_sync(sc->sc_dmat, data->map, 0,
+ data->map->dm_mapsize, BUS_DMASYNC_POSTWRITE);
bus_dmamap_unload(sc->sc_dmat, data->map);
m_freem(data->m);
data->m = NULL;
@@ -1915,11 +1957,18 @@ iwn_notif_intr(struct iwn_softc *sc)
struct ifnet *ifp = &ic->ic_if;
uint16_t hw;
+ bus_dmamap_sync(sc->sc_dmat, sc->rxq.stat_dma.map,
+ 0, sc->rxq.stat_dma.size, BUS_DMASYNC_POSTREAD);
+
hw = letoh16(sc->rxq.stat->closed_count) & 0xfff;
while (sc->rxq.cur != hw) {
struct iwn_rx_data *data = &sc->rxq.data[sc->rxq.cur];
struct iwn_rx_desc *desc = (void *)data->m->m_ext.ext_buf;
+ bus_dmamap_sync(sc->sc_dmat, sc->rxq.buf_dma.map,
+ (caddr_t)desc - sc->rxq.buf_dma.vaddr, sizeof (*desc),
+ BUS_DMASYNC_POSTREAD);
+
DPRINTFN(4, ("notification qid=%d idx=%d flags=%x type=%d\n",
desc->qid & 0xf, desc->idx, desc->flags, desc->type));
@@ -1951,6 +2000,10 @@ iwn_notif_intr(struct iwn_softc *sc)
{
struct iwn_beacon_missed *miss =
(struct iwn_beacon_missed *)(desc + 1);
+
+ bus_dmamap_sync(sc->sc_dmat, sc->rxq.buf_dma.map,
+ (caddr_t)miss - sc->rxq.buf_dma.vaddr,
+ sizeof (*miss), BUS_DMASYNC_POSTREAD);
/*
* If more than 5 consecutive beacons are missed,
* reinitialize the sensitivity state machine.
@@ -1968,6 +2021,9 @@ iwn_notif_intr(struct iwn_softc *sc)
(struct iwn_ucode_info *)(desc + 1);
/* The microcontroller is ready. */
+ bus_dmamap_sync(sc->sc_dmat, sc->rxq.buf_dma.map,
+ (caddr_t)uc - sc->rxq.buf_dma.vaddr,
+ sizeof (*uc), BUS_DMASYNC_POSTREAD);
DPRINTF(("microcode alive notification version=%d.%d "
"subtype=%x alive=%x\n", uc->major, uc->minor,
uc->subtype, letoh32(uc->valid)));
@@ -1990,6 +2046,9 @@ iwn_notif_intr(struct iwn_softc *sc)
uint32_t *status = (uint32_t *)(desc + 1);
/* Enabled/disabled notification. */
+ bus_dmamap_sync(sc->sc_dmat, sc->rxq.buf_dma.map,
+ (caddr_t)status - sc->rxq.buf_dma.vaddr,
+ sizeof (*status), BUS_DMASYNC_POSTREAD);
DPRINTF(("state changed to %x\n", letoh32(*status)));
if (letoh32(*status) & 1) {
@@ -2008,6 +2067,9 @@ iwn_notif_intr(struct iwn_softc *sc)
struct iwn_start_scan *scan =
(struct iwn_start_scan *)(desc + 1);
+ bus_dmamap_sync(sc->sc_dmat, sc->rxq.buf_dma.map,
+ (caddr_t)scan - sc->rxq.buf_dma.vaddr,
+ sizeof (*scan), BUS_DMASYNC_POSTREAD);
DPRINTFN(2, ("scanning channel %d status %x\n",
scan->chan, letoh32(scan->status)));
@@ -2020,6 +2082,9 @@ iwn_notif_intr(struct iwn_softc *sc)
struct iwn_stop_scan *scan =
(struct iwn_stop_scan *)(desc + 1);
+ bus_dmamap_sync(sc->sc_dmat, sc->rxq.buf_dma.map,
+ (caddr_t)scan - sc->rxq.buf_dma.vaddr,
+ sizeof (*scan), BUS_DMASYNC_POSTREAD);
DPRINTF(("scan finished nchan=%d status=%d chan=%d\n",
scan->nchan, scan->status, scan->chan));
@@ -2208,8 +2273,15 @@ iwn4965_update_sched(struct iwn_softc *sc, int qid, int idx, uint8_t id,
uint16_t *w = &sc->sched[qid * IWN4965_SCHED_COUNT + idx];
*w = htole16(len + 8);
- if (idx < IWN_SCHED_WINSZ)
+ bus_dmamap_sync(sc->sc_dmat, sc->sched_dma.map,
+ (caddr_t)w - sc->sched_dma.vaddr, sizeof (uint16_t),
+ BUS_DMASYNC_PREWRITE);
+ if (idx < IWN_SCHED_WINSZ) {
*(w + IWN_TX_RING_COUNT) = *w;
+ bus_dmamap_sync(sc->sc_dmat, sc->sched_dma.map,
+ (caddr_t)(w + IWN_TX_RING_COUNT) - sc->sched_dma.vaddr,
+ sizeof (uint16_t), BUS_DMASYNC_PREWRITE);
+ }
}
void
@@ -2219,8 +2291,15 @@ iwn5000_update_sched(struct iwn_softc *sc, int qid, int idx, uint8_t id,
uint16_t *w = &sc->sched[qid * IWN5000_SCHED_COUNT + idx];
*w = htole16(id << 12 | (len + 8));
- if (idx < IWN_SCHED_WINSZ)
+ bus_dmamap_sync(sc->sc_dmat, sc->sched_dma.map,
+ (caddr_t)w - sc->sched_dma.vaddr, sizeof (uint16_t),
+ BUS_DMASYNC_PREWRITE);
+ if (idx < IWN_SCHED_WINSZ) {
*(w + IWN_TX_RING_COUNT) = *w;
+ bus_dmamap_sync(sc->sc_dmat, sc->sched_dma.map,
+ (caddr_t)(w + IWN_TX_RING_COUNT) - sc->sched_dma.vaddr,
+ sizeof (uint16_t), BUS_DMASYNC_PREWRITE);
+ }
}
void
@@ -2229,8 +2308,15 @@ iwn5000_reset_sched(struct iwn_softc *sc, int qid, int idx)
uint16_t *w = &sc->sched[qid * IWN5000_SCHED_COUNT + idx];
*w = (*w & htole16(0xf000)) | htole16(1);
- if (idx < IWN_SCHED_WINSZ)
+ bus_dmamap_sync(sc->sc_dmat, sc->sched_dma.map,
+ (caddr_t)w - sc->sched_dma.vaddr, sizeof (uint16_t),
+ BUS_DMASYNC_PREWRITE);
+ if (idx < IWN_SCHED_WINSZ) {
*(w + IWN_TX_RING_COUNT) = *w;
+ bus_dmamap_sync(sc->sc_dmat, sc->sched_dma.map,
+ (caddr_t)(w + IWN_TX_RING_COUNT) - sc->sched_dma.vaddr,
+ sizeof (uint16_t), BUS_DMASYNC_PREWRITE);
+ }
}
int
@@ -2508,6 +2594,15 @@ iwn_tx(struct iwn_softc *sc, struct mbuf *m, struct ieee80211_node *ni)
data->map->dm_segs[i - 1].ds_len);
}
+ bus_dmamap_sync(sc->sc_dmat, data->map, 0, data->map->dm_mapsize,
+ BUS_DMASYNC_PREWRITE);
+ bus_dmamap_sync(sc->sc_dmat, ring->cmd_dma.map,
+ (caddr_t)cmd - ring->cmd_dma.vaddr, sizeof (*cmd),
+ BUS_DMASYNC_PREWRITE);
+ bus_dmamap_sync(sc->sc_dmat, ring->desc_dma.map,
+ (caddr_t)desc - ring->desc_dma.vaddr, sizeof (*desc),
+ BUS_DMASYNC_PREWRITE);
+
/* Update TX scheduler. */
hal->update_sched(sc, ring->qid, ring->cur, tx->id, totlen);
@@ -2723,6 +2818,18 @@ iwn_cmd(struct iwn_softc *sc, int code, const void *buf, int size, int async)
IWN_SET_DESC_NSEGS(desc, 1);
IWN_SET_DESC_SEG(desc, 0, paddr, totlen);
+ if (size > sizeof cmd->data) {
+ bus_dmamap_sync(sc->sc_dmat, data->map, 0, totlen,
+ BUS_DMASYNC_PREWRITE);
+ } else {
+ bus_dmamap_sync(sc->sc_dmat, ring->cmd_dma.map,
+ (caddr_t)cmd - ring->cmd_dma.vaddr, totlen,
+ BUS_DMASYNC_PREWRITE);
+ }
+ bus_dmamap_sync(sc->sc_dmat, ring->desc_dma.map,
+ (caddr_t)desc - ring->desc_dma.vaddr, sizeof (*desc),
+ BUS_DMASYNC_PREWRITE);
+
/* Update TX scheduler. */
hal->update_sched(sc, ring->qid, ring->cur, 0, 0);
@@ -4481,8 +4588,12 @@ iwn4965_load_firmware(struct iwn_softc *sc)
/* Copy initialization sections into pre-allocated DMA-safe memory. */
memcpy(dma->vaddr, fw->init.data, fw->init.datasz);
+ bus_dmamap_sync(sc->sc_dmat, dma->map, 0, fw->init.datasz,
+ BUS_DMASYNC_PREWRITE);
memcpy(dma->vaddr + IWN4965_FW_DATA_MAXSZ,
fw->init.text, fw->init.textsz);
+ bus_dmamap_sync(sc->sc_dmat, dma->map, IWN4965_FW_DATA_MAXSZ,
+ fw->init.textsz, BUS_DMASYNC_PREWRITE);
/* Tell adapter where to find initialization sections. */
if ((error = iwn_nic_lock(sc)) != 0)
@@ -4517,8 +4628,12 @@ iwn4965_load_firmware(struct iwn_softc *sc)
/* Copy runtime sections into pre-allocated DMA-safe memory. */
memcpy(dma->vaddr, fw->main.data, fw->main.datasz);
+ bus_dmamap_sync(sc->sc_dmat, dma->map, 0, fw->main.datasz,
+ BUS_DMASYNC_PREWRITE);
memcpy(dma->vaddr + IWN4965_FW_DATA_MAXSZ,
fw->main.text, fw->main.textsz);
+ bus_dmamap_sync(sc->sc_dmat, dma->map, IWN4965_FW_DATA_MAXSZ,
+ fw->main.textsz, BUS_DMASYNC_PREWRITE);
/* Tell adapter where to find runtime sections. */
if ((error = iwn_nic_lock(sc)) != 0)
@@ -4543,6 +4658,7 @@ iwn5000_load_firmware_section(struct iwn_softc *sc, uint32_t dst,
/* Copy firmware section into pre-allocated DMA-safe memory. */
memcpy(dma->vaddr, section, size);
+ bus_dmamap_sync(sc->sc_dmat, dma->map, 0, size, BUS_DMASYNC_PREWRITE);
if ((error = iwn_nic_lock(sc)) != 0)
return error;