summaryrefslogtreecommitdiff
diff options
context:
space:
mode:
-rw-r--r--sys/dev/ic/malo.c468
1 files changed, 234 insertions, 234 deletions
diff --git a/sys/dev/ic/malo.c b/sys/dev/ic/malo.c
index af4a86ad1f7..39b64eb5f99 100644
--- a/sys/dev/ic/malo.c
+++ b/sys/dev/ic/malo.c
@@ -1,4 +1,4 @@
-/* $OpenBSD: malo.c,v 1.10 2006/10/24 19:20:01 mglocker Exp $ */
+/* $OpenBSD: malo.c,v 1.11 2006/10/25 20:42:57 mglocker Exp $ */
/*
* Copyright (c) 2006 Claudio Jeker <claudio@openbsd.org>
@@ -138,7 +138,6 @@ struct malo_hw_spec {
uint32_t WcbBase3;
} __packed;
-
#define malo_mem_write4(sc, off, x) \
bus_space_write_4((sc)->sc_mem1_bt, (sc)->sc_mem1_bh, (off), (x))
#define malo_mem_write2(sc, off, x) \
@@ -164,6 +163,7 @@ struct cfdriver malo_cd = {
int malo_alloc_cmd(struct malo_softc *sc);
void malo_free_cmd(struct malo_softc *sc);
+int malo_send_cmd(struct malo_softc *sc, bus_addr_t addr, uint32_t waitfor);
int malo_alloc_rx_ring(struct malo_softc *sc, struct malo_rx_ring *ring,
int count);
void malo_reset_rx_ring(struct malo_softc *sc, struct malo_rx_ring *ring);
@@ -171,11 +171,6 @@ void malo_free_rx_ring(struct malo_softc *sc, struct malo_rx_ring *ring);
int malo_alloc_tx_ring(struct malo_softc *sc, struct malo_tx_ring *ring,
int count);
void malo_free_tx_ring(struct malo_softc *sc, struct malo_tx_ring *ring);
-int malo_load_bootimg(struct malo_softc *sc);
-int malo_load_firmware(struct malo_softc *sc);
-int malo_send_cmd(struct malo_softc *sc, bus_addr_t addr, uint32_t waitfor);
-int malo_reset(struct malo_softc *sc);
-int malo_get_spec(struct malo_softc *sc);
int malo_init(struct ifnet *ifp);
int malo_ioctl(struct ifnet *ifp, u_long cmd, caddr_t data);
void malo_start(struct ifnet *ifp);
@@ -188,6 +183,10 @@ void malo_newassoc(struct ieee80211com *ic, struct ieee80211_node *ni,
struct ieee80211_node *
malo_node_alloc(struct ieee80211com *ic);
void malo_rx_intr(struct malo_softc *sc);
+int malo_load_bootimg(struct malo_softc *sc);
+int malo_load_firmware(struct malo_softc *sc);
+int malo_cmd_get_spec(struct malo_softc *sc);
+int malo_cmd_reset(struct malo_softc *sc);
/* supported rates */
const struct ieee80211_rateset malo_rates_11b =
@@ -232,6 +231,7 @@ malo_intr(void *arg)
/* just ack the interrupt */
malo_ctl_write4(sc, 0x0c30, 0);
+
return (1);
}
@@ -242,8 +242,8 @@ malo_attach(struct malo_softc *sc)
struct ifnet *ifp = &sc->sc_ic.ic_if;
int i;
#if 0
- /* ??? */
- //malo_ctl_write4(sc, 0x0c38, 0x1f);
+ /* ???, what is this for, seems unnecessary */
+ /* malo_ctl_write4(sc, 0x0c38, 0x1f); */
/* disable interrupts */
malo_ctl_read4(sc, 0x0c30);
malo_ctl_write4(sc, 0x0c30, 0);
@@ -388,6 +388,32 @@ malo_free_cmd(struct malo_softc *sc)
}
int
+malo_send_cmd(struct malo_softc *sc, bus_addr_t addr, uint32_t waitfor)
+{
+ int i;
+
+ malo_ctl_write4(sc, 0x0c10, (uint32_t)addr);
+ malo_ctl_read4(sc, 0x0c14);
+ malo_ctl_write4(sc, 0x0c18, 2); /* CPU_TRANSFER_CMD */
+ malo_ctl_read4(sc, 0x0c14);
+
+ if (waitfor == 0)
+ return (0);
+
+ /* wait for the DMA engine to finish the transfer */
+ for (i = 0; i < 100; i++) {
+ delay(50);
+ if (malo_ctl_read4(sc, 0x0c14) == waitfor);
+ break;
+ }
+
+ if (i == 100)
+ return (ETIMEDOUT);
+
+ return (0);
+}
+
+int
malo_alloc_rx_ring(struct malo_softc *sc, struct malo_rx_ring *ring, int count)
{
struct malo_rx_desc *desc;
@@ -668,229 +694,6 @@ malo_free_tx_ring(struct malo_softc *sc, struct malo_tx_ring *ring)
}
int
-malo_load_bootimg(struct malo_softc *sc)
-{
- char *name = "mrv8k-b.fw";
- uint8_t *ucode;
- size_t size, count;
- int error;
-
- /* load boot firmware */
- if ((error = loadfirmware(name, &ucode, &size)) != 0) {
- DPRINTF(("%s: error %d, could not read microcode %s!\n",
- sc->sc_dev.dv_xname, error, name));
- return (EIO);
- }
-
- /*
- * It seems we are putting this code directly onto the stack of
- * the ARM cpu. I don't know why we need to instruct the DMA
- * engine to move the code. This is a big riddle without docu.
- */
- DPRINTF(("%s: loading boot firmware\n", sc->sc_dev.dv_xname));
- malo_mem_write2(sc, 0xbef8, 0x001);
- malo_mem_write2(sc, 0xbefa, size);
- malo_mem_write4(sc, 0xbefc, 0);
-
- for (count = 0; count < size; count++)
- malo_mem_write1(sc, 0xbf00 + count, ucode[count]);
-
- /*
- * we loaded the firmware into card memory now tell the CPU
- * to fetch the code and execute it. The memory mapped via the
- * first bar is internaly mapped to 0xc0000000.
- */
- if (malo_send_cmd(sc, 0xc000bef8, 5) != 0) {
- printf("%s: timeout at boot firmware load!\n",
- sc->sc_dev.dv_xname);
- free(ucode, M_DEVBUF);
- return (ETIMEDOUT);
- }
- free(ucode, M_DEVBUF);
-
- /* tell the card we're done and... */
- malo_mem_write2(sc, 0xbef8, 0x001);
- malo_mem_write2(sc, 0xbefa, 0);
- malo_mem_write4(sc, 0xbefc, 0);
- malo_send_cmd(sc, 0xc000bef8, 0);
-
- /* give card a bit time to init */
- delay(50);
- DPRINTF(("%s: boot firmware loaded\n", sc->sc_dev.dv_xname));
-
- return (0);
-}
-
-int
-malo_load_firmware(struct malo_softc *sc)
-{
- struct malo_cmdheader *hdr;
- char *name = "mrv8k-f.fw";
- void *data;
- uint8_t *ucode;
- size_t size, count, bsize;
- int sn, error;
-
- /* load real firmware now */
- if ((error = loadfirmware(name, &ucode, &size)) != 0) {
- DPRINTF(("%s: error %d, could not read microcode %s!\n",
- sc->sc_dev.dv_xname, error, name));
- return (EIO);
- }
-
- DPRINTF(("%s: loading firmware\n", sc->sc_dev.dv_xname));
- hdr = sc->sc_cmd_mem;
- data = hdr + 1;
- sn = 1;
- for (count = 0; count < size; count += bsize) {
- bsize = MIN(256, size - count);
-
- hdr->cmd = htole16(0x0001);
- hdr->size = htole16(bsize);
- hdr->seqnum = htole16(sn++);
- hdr->result = 0;
-
- bcopy(ucode + count, data, bsize);
-
- bus_dmamap_sync(sc->sc_dmat, sc->sc_cmd_dmam, 0, PAGE_SIZE,
- BUS_DMASYNC_PREWRITE|BUS_DMASYNC_PREREAD);
- if (malo_send_cmd(sc, sc->sc_cmd_dmaaddr, 5) != 0) {
- printf("%s: timeout at firmware upload!\n",
- sc->sc_dev.dv_xname);
- free(ucode, M_DEVBUF);
- return (ETIMEDOUT);
- }
-
- delay(100);
- }
- free(ucode, M_DEVBUF);
-
- DPRINTF(("%s: firmware upload finished\n", sc->sc_dev.dv_xname));
-
- hdr->cmd = htole16(0x0001);
- hdr->size = 0;
- hdr->seqnum = htole16(sn++);
- hdr->result = 0;
-
- bus_dmamap_sync(sc->sc_dmat, sc->sc_cmd_dmam, 0, PAGE_SIZE,
- BUS_DMASYNC_PREWRITE|BUS_DMASYNC_PREREAD);
- if (malo_send_cmd(sc, sc->sc_cmd_dmaaddr, 0xf0f1f2f4) != 0) {
- printf("%s: timeout at firmware load!\n", sc->sc_dev.dv_xname);
- return (ETIMEDOUT);
- }
-
- /* give card a bit time to load firmware */
- delay(20000);
- DPRINTF(("%s: firmware loaded\n", sc->sc_dev.dv_xname));
-
- return (0);
-}
-
-int
-malo_send_cmd(struct malo_softc *sc, bus_addr_t addr, uint32_t waitfor)
-{
- int i;
-
- malo_ctl_write4(sc, 0x0c10, (uint32_t)addr);
- malo_ctl_read4(sc, 0x0c14);
- malo_ctl_write4(sc, 0x0c18, 2); /* CPU_TRANSFER_CMD */
- malo_ctl_read4(sc, 0x0c14);
-
- if (waitfor == 0)
- return (0);
-
- /* wait for the DMA engine to finish the transfer */
- for (i = 0; i < 100; i++) {
- delay(50);
- if (malo_ctl_read4(sc, 0x0c14) == waitfor);
- break;
- }
-
- if (i == 100)
- return (ETIMEDOUT);
-
- return (0);
-}
-
-int
-malo_reset(struct malo_softc *sc)
-{
- struct malo_cmdheader *hdr = sc->sc_cmd_mem;
-
- hdr->cmd = htole16(5);
- hdr->size = htole16(sizeof(*hdr));
- hdr->seqnum = 1;
- hdr->result = 0;
-
- bus_dmamap_sync(sc->sc_dmat, sc->sc_cmd_dmam, 0, PAGE_SIZE,
- BUS_DMASYNC_PREWRITE|BUS_DMASYNC_PREREAD);
-
- malo_send_cmd(sc, sc->sc_cmd_dmaaddr, 0);
- tsleep(sc, 0, "malorst", hz);
-
- bus_dmamap_sync(sc->sc_dmat, sc->sc_cmd_dmam, 0, PAGE_SIZE,
- BUS_DMASYNC_POSTWRITE|BUS_DMASYNC_POSTREAD);
-
- if (hdr->cmd & MALO_CMD_RESPONSE)
- return (0);
- else
- return (ETIMEDOUT);
-}
-
-int
-malo_get_spec(struct malo_softc *sc)
-{
- struct malo_cmdheader *hdr = sc->sc_cmd_mem;
- struct malo_hw_spec *spec;
-
- hdr->cmd = htole16(MALO_CMD_GET_HW_SPEC);
- hdr->size = htole16(sizeof(*hdr) + sizeof(*spec));
- hdr->seqnum = htole16(42); /* the one and only */
- hdr->result = 0;
- spec = (struct malo_hw_spec *)(hdr + 1);
-
- DPRINTF(("%s: fw cmd %04x size %d\n", sc->sc_dev.dv_xname,
- hdr->cmd, hdr->size));
-
- bzero(spec, sizeof(*spec));
- memset(spec->PermanentAddress, 0xff, ETHER_ADDR_LEN);
- spec->CookiePtr = htole32(sc->sc_cookie_dmaaddr);
-
- bus_dmamap_sync(sc->sc_dmat, sc->sc_cmd_dmam, 0, PAGE_SIZE,
- BUS_DMASYNC_PREWRITE|BUS_DMASYNC_PREREAD);
-
- malo_send_cmd(sc, sc->sc_cmd_dmaaddr, 0);
- tsleep(sc, 0, "malospc", hz);
-
- bus_dmamap_sync(sc->sc_dmat, sc->sc_cmd_dmam, 0, PAGE_SIZE,
- BUS_DMASYNC_POSTWRITE|BUS_DMASYNC_POSTREAD);
-
- if ((hdr->cmd & MALO_CMD_RESPONSE) == 0)
- return (ETIMEDOUT);
-
- /* XXX get the data form the buffer and feed it to ieee80211 */
- DPRINTF(("%s: get_hw_spec: V%x R%x, #WCB %d, #Mcast %d, Regcode %d, "
- "#Ant %d\n", sc->sc_dev.dv_xname, htole16(spec->HwVersion),
- htole32(spec->FWReleaseNumber), htole16(spec->NumOfWCB),
- htole16(spec->NumOfMCastAdr), htole16(spec->RegionCode),
- htole16(spec->NumberOfAntenna)));
-
- /* tell the DMA engine where our rings are */
- malo_mem_write4(sc, letoh32(spec->RxPdRdPtr) & 0xffff,
- htole32(sc->sc_rxring.physaddr));
- malo_mem_write4(sc, letoh32(spec->RxPdWrPtr) & 0xffff,
- htole32(sc->sc_rxring.physaddr));
- malo_mem_write4(sc, letoh32(spec->WcbBase0) & 0xffff,
- htole32(sc->sc_txring.physaddr));
-
- /* save DMA RX pointers for later use */
- sc->sc_RxPdRdPtr = letoh32(spec->RxPdRdPtr) & 0xffff;
- sc->sc_RxPdWrPtr = letoh32(spec->RxPdWrPtr) & 0xffff;
-
- return (0);
-}
-
-int
malo_init(struct ifnet *ifp)
{
struct malo_softc *sc = ifp->if_softc;
@@ -918,7 +721,7 @@ malo_init(struct ifnet *ifp)
malo_ctl_write4(sc, 0x0c3c, 0x1f);
malo_ctl_read4(sc, 0x0c14);
- if ((error = malo_get_spec(sc)))
+ if ((error = malo_cmd_get_spec(sc)))
return (error);
ifp->if_flags |= IFF_RUNNING;
@@ -978,7 +781,7 @@ malo_stop(struct malo_softc *sc)
/* try to reset card, if the firmware is loaded */
if (ifp->if_flags & IFF_RUNNING)
- malo_reset(sc);
+ malo_cmd_reset(sc);
ifp->if_flags &= ~(IFF_RUNNING | IFF_OACTIVE);
@@ -1161,3 +964,200 @@ skip:
malo_mem_write4(sc, sc->sc_RxPdRdPtr, rxRdPtr);
}
+
+int
+malo_load_bootimg(struct malo_softc *sc)
+{
+ char *name = "mrv8k-b.fw";
+ uint8_t *ucode;
+ size_t size, count;
+ int error;
+
+ /* load boot firmware */
+ if ((error = loadfirmware(name, &ucode, &size)) != 0) {
+ DPRINTF(("%s: error %d, could not read microcode %s!\n",
+ sc->sc_dev.dv_xname, error, name));
+ return (EIO);
+ }
+
+ /*
+ * It seems we are putting this code directly onto the stack of
+ * the ARM cpu. I don't know why we need to instruct the DMA
+ * engine to move the code. This is a big riddle without docu.
+ */
+ DPRINTF(("%s: loading boot firmware\n", sc->sc_dev.dv_xname));
+ malo_mem_write2(sc, 0xbef8, 0x001);
+ malo_mem_write2(sc, 0xbefa, size);
+ malo_mem_write4(sc, 0xbefc, 0);
+
+ for (count = 0; count < size; count++)
+ malo_mem_write1(sc, 0xbf00 + count, ucode[count]);
+
+ /*
+ * we loaded the firmware into card memory now tell the CPU
+ * to fetch the code and execute it. The memory mapped via the
+ * first bar is internaly mapped to 0xc0000000.
+ */
+ if (malo_send_cmd(sc, 0xc000bef8, 5) != 0) {
+ printf("%s: timeout at boot firmware load!\n",
+ sc->sc_dev.dv_xname);
+ free(ucode, M_DEVBUF);
+ return (ETIMEDOUT);
+ }
+ free(ucode, M_DEVBUF);
+
+ /* tell the card we're done and... */
+ malo_mem_write2(sc, 0xbef8, 0x001);
+ malo_mem_write2(sc, 0xbefa, 0);
+ malo_mem_write4(sc, 0xbefc, 0);
+ malo_send_cmd(sc, 0xc000bef8, 0);
+
+ /* give card a bit time to init */
+ delay(50);
+ DPRINTF(("%s: boot firmware loaded\n", sc->sc_dev.dv_xname));
+
+ return (0);
+}
+
+int
+malo_load_firmware(struct malo_softc *sc)
+{
+ struct malo_cmdheader *hdr;
+ char *name = "mrv8k-f.fw";
+ void *data;
+ uint8_t *ucode;
+ size_t size, count, bsize;
+ int sn, error;
+
+ /* load real firmware now */
+ if ((error = loadfirmware(name, &ucode, &size)) != 0) {
+ DPRINTF(("%s: error %d, could not read microcode %s!\n",
+ sc->sc_dev.dv_xname, error, name));
+ return (EIO);
+ }
+
+ DPRINTF(("%s: loading firmware\n", sc->sc_dev.dv_xname));
+ hdr = sc->sc_cmd_mem;
+ data = hdr + 1;
+ sn = 1;
+ for (count = 0; count < size; count += bsize) {
+ bsize = MIN(256, size - count);
+
+ hdr->cmd = htole16(0x0001);
+ hdr->size = htole16(bsize);
+ hdr->seqnum = htole16(sn++);
+ hdr->result = 0;
+
+ bcopy(ucode + count, data, bsize);
+
+ bus_dmamap_sync(sc->sc_dmat, sc->sc_cmd_dmam, 0, PAGE_SIZE,
+ BUS_DMASYNC_PREWRITE|BUS_DMASYNC_PREREAD);
+ if (malo_send_cmd(sc, sc->sc_cmd_dmaaddr, 5) != 0) {
+ printf("%s: timeout at firmware upload!\n",
+ sc->sc_dev.dv_xname);
+ free(ucode, M_DEVBUF);
+ return (ETIMEDOUT);
+ }
+
+ delay(100);
+ }
+ free(ucode, M_DEVBUF);
+
+ DPRINTF(("%s: firmware upload finished\n", sc->sc_dev.dv_xname));
+
+ hdr->cmd = htole16(0x0001);
+ hdr->size = 0;
+ hdr->seqnum = htole16(sn++);
+ hdr->result = 0;
+
+ bus_dmamap_sync(sc->sc_dmat, sc->sc_cmd_dmam, 0, PAGE_SIZE,
+ BUS_DMASYNC_PREWRITE|BUS_DMASYNC_PREREAD);
+ if (malo_send_cmd(sc, sc->sc_cmd_dmaaddr, 0xf0f1f2f4) != 0) {
+ printf("%s: timeout at firmware load!\n", sc->sc_dev.dv_xname);
+ return (ETIMEDOUT);
+ }
+
+ /* give card a bit time to load firmware */
+ delay(20000);
+ DPRINTF(("%s: firmware loaded\n", sc->sc_dev.dv_xname));
+
+ return (0);
+}
+
+int
+malo_cmd_get_spec(struct malo_softc *sc)
+{
+ struct malo_cmdheader *hdr = sc->sc_cmd_mem;
+ struct malo_hw_spec *spec;
+
+ hdr->cmd = htole16(MALO_CMD_GET_HW_SPEC);
+ hdr->size = htole16(sizeof(*hdr) + sizeof(*spec));
+ hdr->seqnum = htole16(42); /* the one and only */
+ hdr->result = 0;
+ spec = (struct malo_hw_spec *)(hdr + 1);
+
+ DPRINTF(("%s: fw cmd %04x size %d\n", sc->sc_dev.dv_xname,
+ hdr->cmd, hdr->size));
+
+ bzero(spec, sizeof(*spec));
+ memset(spec->PermanentAddress, 0xff, ETHER_ADDR_LEN);
+ spec->CookiePtr = htole32(sc->sc_cookie_dmaaddr);
+
+ bus_dmamap_sync(sc->sc_dmat, sc->sc_cmd_dmam, 0, PAGE_SIZE,
+ BUS_DMASYNC_PREWRITE|BUS_DMASYNC_PREREAD);
+
+ malo_send_cmd(sc, sc->sc_cmd_dmaaddr, 0);
+ tsleep(sc, 0, "malospc", hz);
+
+ bus_dmamap_sync(sc->sc_dmat, sc->sc_cmd_dmam, 0, PAGE_SIZE,
+ BUS_DMASYNC_POSTWRITE|BUS_DMASYNC_POSTREAD);
+
+ if ((hdr->cmd & MALO_CMD_RESPONSE) == 0)
+ return (ETIMEDOUT);
+
+ /* XXX get the data form the buffer and feed it to ieee80211 */
+ DPRINTF(("%s: get_hw_spec: V%x R%x, #WCB %d, #Mcast %d, Regcode %d, "
+ "#Ant %d\n", sc->sc_dev.dv_xname, htole16(spec->HwVersion),
+ htole32(spec->FWReleaseNumber), htole16(spec->NumOfWCB),
+ htole16(spec->NumOfMCastAdr), htole16(spec->RegionCode),
+ htole16(spec->NumberOfAntenna)));
+
+ /* tell the DMA engine where our rings are */
+ malo_mem_write4(sc, letoh32(spec->RxPdRdPtr) & 0xffff,
+ htole32(sc->sc_rxring.physaddr));
+ malo_mem_write4(sc, letoh32(spec->RxPdWrPtr) & 0xffff,
+ htole32(sc->sc_rxring.physaddr));
+ malo_mem_write4(sc, letoh32(spec->WcbBase0) & 0xffff,
+ htole32(sc->sc_txring.physaddr));
+
+ /* save DMA RX pointers for later use */
+ sc->sc_RxPdRdPtr = letoh32(spec->RxPdRdPtr) & 0xffff;
+ sc->sc_RxPdWrPtr = letoh32(spec->RxPdWrPtr) & 0xffff;
+
+ return (0);
+}
+
+int
+malo_cmd_reset(struct malo_softc *sc)
+{
+ struct malo_cmdheader *hdr = sc->sc_cmd_mem;
+
+ hdr->cmd = htole16(5);
+ hdr->size = htole16(sizeof(*hdr));
+ hdr->seqnum = 1;
+ hdr->result = 0;
+
+ bus_dmamap_sync(sc->sc_dmat, sc->sc_cmd_dmam, 0, PAGE_SIZE,
+ BUS_DMASYNC_PREWRITE | BUS_DMASYNC_PREREAD);
+
+ malo_send_cmd(sc, sc->sc_cmd_dmaaddr, 0);
+ tsleep(sc, 0, "malorst", hz);
+
+ bus_dmamap_sync(sc->sc_dmat, sc->sc_cmd_dmam, 0, PAGE_SIZE,
+ BUS_DMASYNC_POSTWRITE | BUS_DMASYNC_POSTREAD);
+
+ if (hdr->cmd & MALO_CMD_RESPONSE)
+ return (0);
+ else
+ return (ETIMEDOUT);
+}