summaryrefslogtreecommitdiff
path: root/sys
diff options
context:
space:
mode:
Diffstat (limited to 'sys')
-rw-r--r--sys/dev/ic/malo.c233
1 files changed, 137 insertions, 96 deletions
diff --git a/sys/dev/ic/malo.c b/sys/dev/ic/malo.c
index 9f066fc0643..a18c811ae52 100644
--- a/sys/dev/ic/malo.c
+++ b/sys/dev/ic/malo.c
@@ -1,4 +1,4 @@
-/* $OpenBSD: malo.c,v 1.20 2006/11/09 10:25:12 mglocker Exp $ */
+/* $OpenBSD: malo.c,v 1.21 2006/11/10 22:29:44 mglocker Exp $ */
/*
* Copyright (c) 2006 Claudio Jeker <claudio@openbsd.org>
@@ -111,12 +111,14 @@ struct malo_tx_desc {
#define MALO_RX_RING_COUNT 256
#define MALO_TX_RING_COUNT 256
#define MALO_MAX_SCATTER 8 /* XXX unknown, wild guess */
+#define MALO_RATE_IS_OFDM(rate) ((rate) >= 12 && (rate) != 22)
/*
* Firmware commands
*/
#define MALO_CMD_GET_HW_SPEC 0x0003
#define MALO_CMD_SET_RADIO 0x001c
+#define MALO_CMD_SET_AID 0x010d
#define MALO_CMD_SET_TXPOWER 0x001e
#define MALO_CMD_SET_ANTENNA 0x0020
#define MALO_CMD_SET_PRESCAN 0x0107
@@ -156,6 +158,13 @@ struct malo_cmd_radio {
uint16_t enable;
} __packed;
+struct malo_cmd_aid {
+ uint16_t associd;
+ uint8_t macaddr[6];
+ uint32_t gprotection;
+ uint8_t aprates[14];
+} __packed;
+
struct malo_cmd_txpower {
uint16_t action;
uint16_t supportpowerlvl;
@@ -169,6 +178,11 @@ struct malo_cmd_antenna {
uint16_t mode;
} __packed;
+struct malo_cmd_postscan {
+ uint32_t isibss;
+ uint8_t bssid[6];
+} __packed;
+
struct malo_cmd_channel {
uint16_t action;
uint8_t channel;
@@ -200,6 +214,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_send_cmd_dma(struct malo_softc *sc, bus_addr_t addr);
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);
@@ -228,6 +243,8 @@ int malo_tx_mgt(struct malo_softc *sc, struct mbuf *m0,
void malo_tx_setup_desc(struct malo_softc *sc, struct malo_tx_desc *desc,
uint32_t flags, uint16_t xflags, int len, int rate,
const bus_dma_segment_t *segs, int nsegs, int ac);
+uint16_t
+ malo_txtime(int len, int rate, uint32_t flags);
void malo_rx_intr(struct malo_softc *sc);
int malo_load_bootimg(struct malo_softc *sc);
int malo_load_firmware(struct malo_softc *sc);
@@ -235,11 +252,14 @@ 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);
int malo_cmd_set_prescan(struct malo_softc *sc);
-int malo_cmd_set_postscan(struct malo_softc *sc);
+int malo_cmd_set_postscan(struct malo_softc *sc, uint8_t *macaddr,
+ uint8_t ibsson);
int malo_cmd_set_channel(struct malo_softc *sc, uint8_t channel);
int malo_cmd_set_antenna(struct malo_softc *sc, uint16_t antenna_type);
int malo_cmd_set_radio(struct malo_softc *sc, uint16_t mode,
uint16_t preamble);
+int malo_cmd_set_aid(struct malo_softc *sc, uint8_t *bssid,
+ uint16_t associd);
int malo_cmd_set_txpower(struct malo_softc *sc, unsigned int powerlevel);
int malo_cmd_set_rts(struct malo_softc *sc, uint32_t threshold);
@@ -291,13 +311,8 @@ malo_intr(void *arg)
case MALO_CMD_SET_POSTSCAN:
DPRINTF(("%s: cmd answer for MALO_CMD_SET_POSTSCAN=%x",
sc->sc_dev.dv_xname, hdr->result));
- /* wakeup caller */
- wakeup(sc);
break;
case MALO_CMD_SET_CHANNEL:
- bus_dmamap_sync(sc->sc_dmat, sc->sc_cmd_dmam, 0,
- PAGE_SIZE, BUS_DMASYNC_POSTWRITE |
- BUS_DMASYNC_POSTREAD);
DPRINTF(("%s: cmd answer for MALO_CMD_SET_CHANNEL=%x",
sc->sc_dev.dv_xname, hdr->result));
break;
@@ -522,6 +537,31 @@ malo_send_cmd(struct malo_softc *sc, bus_addr_t addr, uint32_t waitfor)
}
int
+malo_send_cmd_dma(struct malo_softc *sc, bus_addr_t addr)
+{
+ int i;
+ struct malo_cmdheader *hdr = sc->sc_cmd_mem;
+
+ 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);
+
+ for (i = 0; i < 100; i++) {
+ delay(50);
+ bus_dmamap_sync(sc->sc_dmat, sc->sc_cmd_dmam, 0, PAGE_SIZE,
+ BUS_DMASYNC_POSTWRITE | BUS_DMASYNC_POSTREAD);
+ if (hdr->cmd & 0x8000)
+ 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;
@@ -896,6 +936,9 @@ malo_init(struct ifnet *ifp)
ifp->if_flags |= IFF_RUNNING;
+ /* start background scanning */
+ ieee80211_new_state(ic, IEEE80211_S_SCAN, -1);
+
return (0);
}
@@ -1043,14 +1086,6 @@ malo_newstate(struct ieee80211com *ic, enum ieee80211_state nstate, int arg)
switch (nstate) {
case IEEE80211_S_INIT:
- if (ostate == IEEE80211_S_SCAN) {
- if (malo_cmd_set_postscan(sc) != 0)
- DPRINTF(("%s: can't set postscan\n",
- sc->sc_dev.dv_xname));
- else
- DPRINTF(("%s: postscan done\n",
- sc->sc_dev.dv_xname));
- }
break;
case IEEE80211_S_SCAN:
if (ostate == IEEE80211_S_INIT) {
@@ -1067,6 +1102,27 @@ malo_newstate(struct ieee80211com *ic, enum ieee80211_state nstate, int arg)
}
timeout_add(&sc->sc_scan_to, hz / 2);
break;
+ case IEEE80211_S_AUTH:
+ DPRINTF(("AUTH\n"));
+ malo_cmd_set_postscan(sc, ic->ic_myaddr, 1);
+ chan = ieee80211_chan2ieee(ic, ic->ic_bss->ni_chan);
+ malo_cmd_set_channel(sc, chan);
+ break;
+ case IEEE80211_S_ASSOC:
+ DPRINTF(("ASSOC\n"));
+#if 0
+ if (ic->ic_flags & IEEE80211_F_SHPREAMBLE)
+ malo_cmd_set_radio(sc, 1, 3); /* short preamble */
+ else
+ malo_cmd_set_radio(sc, 1, 1); /* long preamble */
+
+ malo_cmd_set_aid(sc, ic->ic_bss->ni_bssid,
+ ic->ic_bss->ni_associd);
+#endif
+ break;
+ case IEEE80211_S_RUN:
+ DPRINTF(("RUN\n"));
+ break;
default:
break;
}
@@ -1146,7 +1202,7 @@ malo_reset(struct ifnet *ifp)
return (EIO);
}
- if (malo_cmd_set_postscan(sc) != 0) {
+ if (malo_cmd_set_postscan(sc, ic->ic_myaddr, 1) != 0) {
DPRINTF(("%s: can't postscan\n", sc->sc_dev.dv_xname));
return (EIO);
}
@@ -1236,6 +1292,7 @@ malo_tx_mgt(struct malo_softc *sc, struct mbuf *m0, struct ieee80211_node *ni)
struct malo_tx_desc *desc;
struct malo_tx_data *data;
struct ieee80211_frame *wh;
+ uint16_t dur;
uint32_t flags = 0;
int rate, error;
@@ -1277,6 +1334,11 @@ malo_tx_mgt(struct malo_softc *sc, struct mbuf *m0, struct ieee80211_node *ni)
}
#endif
+ if (!IEEE80211_IS_MULTICAST(wh->i_addr1)) {
+ dur = malo_txtime(14, rate, ic->ic_flags);
+ *(uint16_t *)wh->i_dur = htole16(dur);
+ }
+
/*
* inject FW specific fields into the 802.11 frame
*
@@ -1285,6 +1347,7 @@ malo_tx_mgt(struct malo_softc *sc, struct mbuf *m0, struct ieee80211_node *ni)
* 6 bytes addr4 (inject)
* n bytes 802.11 frame body
*/
+ m0 = m_pullup(m0, sizeof(*wh));
if (M_TRAILINGSPACE(m0) < 8)
return (1);
bcopy(m0->m_data + 24, m0->m_data + 32, m0->m_len - 24);
@@ -1340,6 +1403,27 @@ malo_tx_setup_desc(struct malo_softc *sc, struct malo_tx_desc *desc,
desc->status = htole32(0x00000001 | 0x80000000);
}
+uint16_t
+malo_txtime(int len, int rate, uint32_t flags)
+{
+ uint16_t txtime;
+
+ if (MALO_RATE_IS_OFDM(rate)) {
+ /* IEEE Std 802.11g-2003, pp. 44 */
+ txtime = (8 + 4 * len + 3 + rate - 1) / rate;
+ txtime = 16 + 4 + 4 * txtime + 6;
+ } else {
+ /* IEEE Std 802.11b-1999, pp. 28 */
+ txtime = (16 * len + rate - 1) / rate;
+ if (rate != 2 && (flags & IEEE80211_F_SHPREAMBLE))
+ txtime += 72 + 24;
+ else
+ txtime += 144 + 48;
+ }
+
+ return (txtime);
+}
+
void
malo_rx_intr(struct malo_softc *sc)
{
@@ -1636,13 +1720,7 @@ malo_cmd_get_spec(struct malo_softc *sc)
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, "malocmd", 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)
+ if (malo_send_cmd_dma(sc, sc->sc_cmd_dmaaddr) != 0)
return (ETIMEDOUT);
/* XXX get the data from the buffer and feed it to ieee80211 */
@@ -1680,16 +1758,7 @@ malo_cmd_reset(struct malo_softc *sc)
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, "malocmd", 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);
+ return (malo_send_cmd_dma(sc, sc->sc_cmd_dmaaddr));
}
int
@@ -1705,41 +1774,29 @@ malo_cmd_set_prescan(struct malo_softc *sc)
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, "malocmd", 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);
+ return (malo_send_cmd_dma(sc, sc->sc_cmd_dmaaddr));
}
int
-malo_cmd_set_postscan(struct malo_softc *sc)
+malo_cmd_set_postscan(struct malo_softc *sc, uint8_t *macaddr, uint8_t ibsson)
{
struct malo_cmdheader *hdr = sc->sc_cmd_mem;
+ struct malo_cmd_postscan *body;
hdr->cmd = htole16(MALO_CMD_SET_POSTSCAN);
- hdr->size = htole16(sizeof(*hdr));
+ hdr->size = htole16(sizeof(*hdr) + sizeof(*body));
hdr->seqnum = 1;
hdr->result = 0;
+ body = (struct malo_cmd_postscan *)(hdr + 1);
- 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, "malocmd", hz);
+ bzero(body, sizeof(*body));
+ memcpy(&body->bssid, macaddr, ETHER_ADDR_LEN);
+ body->isibss = htole32(ibsson);
bus_dmamap_sync(sc->sc_dmat, sc->sc_cmd_dmam, 0, PAGE_SIZE,
- BUS_DMASYNC_POSTWRITE | BUS_DMASYNC_POSTREAD);
+ BUS_DMASYNC_PREWRITE | BUS_DMASYNC_PREREAD);
- if (hdr->cmd & MALO_CMD_RESPONSE)
- return (0);
- else
- return (ETIMEDOUT);
+ return (malo_send_cmd_dma(sc, sc->sc_cmd_dmaaddr));
}
int
@@ -1755,15 +1812,13 @@ malo_cmd_set_channel(struct malo_softc *sc, uint8_t channel)
body = (struct malo_cmd_channel *)(hdr + 1);
bzero(body, sizeof(*body));
- body->action = 0; /* seems this is needed */
+ body->action = htole16(1); /* seems this is needed */
body->channel = channel;
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);
-
- return (0);
+ return (malo_send_cmd_dma(sc, sc->sc_cmd_dmaaddr));
}
int
@@ -1788,16 +1843,7 @@ malo_cmd_set_antenna(struct malo_softc *sc, uint16_t antenna)
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, "malocmd", 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);
+ return (malo_send_cmd_dma(sc, sc->sc_cmd_dmaaddr));
}
int
@@ -1821,16 +1867,29 @@ malo_cmd_set_radio(struct malo_softc *sc, uint16_t enable,
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, "malocmd", hz);
+ return (malo_send_cmd_dma(sc, sc->sc_cmd_dmaaddr));
+}
+
+int
+malo_cmd_set_aid(struct malo_softc *sc, uint8_t *bssid, uint16_t associd)
+{
+ struct malo_cmdheader *hdr = sc->sc_cmd_mem;
+ struct malo_cmd_aid *body;
+
+ hdr->cmd = htole16(MALO_CMD_SET_AID);
+ hdr->size = htole16(sizeof(*hdr) + sizeof(*body));
+ hdr->seqnum = 1;
+ hdr->result = 0;
+ body = (struct malo_cmd_aid *)(hdr + 1);
+
+ bzero(body, sizeof(*body));
+ body->associd = htole16(associd);
+ memcpy(&body->macaddr[0], bssid, IEEE80211_ADDR_LEN);
bus_dmamap_sync(sc->sc_dmat, sc->sc_cmd_dmam, 0, PAGE_SIZE,
- BUS_DMASYNC_POSTWRITE | BUS_DMASYNC_POSTREAD);
+ BUS_DMASYNC_PREWRITE | BUS_DMASYNC_PREREAD);
- if (hdr->cmd & MALO_CMD_RESPONSE)
- return (0);
- else
- return (ETIMEDOUT);
+ return (malo_send_cmd_dma(sc, sc->sc_cmd_dmaaddr));
}
int
@@ -1857,16 +1916,7 @@ malo_cmd_set_txpower(struct malo_softc *sc, unsigned int powerlevel)
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, "malocmd", 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);
+ return (malo_send_cmd_dma(sc, sc->sc_cmd_dmaaddr));
}
int
@@ -1884,14 +1934,5 @@ malo_cmd_set_rts(struct malo_softc *sc, uint32_t threshold)
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, "malocmd", 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);
+ return (malo_send_cmd_dma(sc, sc->sc_cmd_dmaaddr));
}