summaryrefslogtreecommitdiff
diff options
context:
space:
mode:
authorMarcus Glocker <mglocker@cvs.openbsd.org>2006-09-16 10:36:13 +0000
committerMarcus Glocker <mglocker@cvs.openbsd.org>2006-09-16 10:36:13 +0000
commit962235ae186f8a360bf20915335e757bf4b37071 (patch)
treefcdaf89983c1bb31b4e0495c3baaacd24d404cbc
parent9243409c548edae3af3e78dd91df949d153846bc (diff)
Bring pgt(4) in a working state. Much work also done by claudio@.
ok deraadt@
-rw-r--r--share/man/man4/pgt.410
-rw-r--r--sys/dev/ic/pgt.c1723
-rw-r--r--sys/dev/ic/pgtreg.h26
-rw-r--r--sys/dev/ic/pgtvar.h116
-rw-r--r--sys/dev/pci/if_pgt_pci.c7
5 files changed, 872 insertions, 1010 deletions
diff --git a/share/man/man4/pgt.4 b/share/man/man4/pgt.4
index 90e47d0def9..6b79647c0aa 100644
--- a/share/man/man4/pgt.4
+++ b/share/man/man4/pgt.4
@@ -1,4 +1,4 @@
-.\" $OpenBSD: pgt.4,v 1.4 2006/08/30 23:14:30 jmc Exp $
+.\" $OpenBSD: pgt.4,v 1.5 2006/09/16 10:36:12 mglocker Exp $
.\"
.\" Copyright (c) 2006 Theo de Raadt.
.\" Copyright (c) 2006 The DragonFly Project. All rights reserved.
@@ -81,7 +81,7 @@ This mode is the default.
.El
.Pp
.Nm
-supports software WEP.
+supports hardware WEP.
Wired Equivalent Privacy (WEP) is the de facto encryption standard
for wireless networks.
It can be typically configured in one of three modes:
@@ -231,8 +231,8 @@ The driver needs a set of firmware files which are loaded when
an interface is brought up:
.Pp
.Bl -tag -width Ds -offset indent -compact
-.It Pa /etc/firmware/1.0.4.3.arm
-.It Pa /etc/firmware/1.1.0.0.arm
+.It Pa /etc/firmware/pgt-isl3877
+.It Pa /etc/firmware/pgt-isl3890
.El
.Pp
These firmware files are not free because Connexant refuses
@@ -248,7 +248,7 @@ A prepackaged version of the firmware, designed to be used with
.Xr pkg_add 1 ,
can be found at:
.Bd -literal -offset indent
-http://www.nazgul.ch/pgt/pgt-firmware-1.1.tgz
+http://www.nazgul.ch/pgt/pgt-firmware-1.0.tgz
.Ed
.Sh HARDWARE
The following cards are among those supported by the
diff --git a/sys/dev/ic/pgt.c b/sys/dev/ic/pgt.c
index f39f32ef5e4..4700c039e61 100644
--- a/sys/dev/ic/pgt.c
+++ b/sys/dev/ic/pgt.c
@@ -1,4 +1,23 @@
-/*-
+/* $OpenBSD: pgt.c,v 1.10 2006/09/16 10:36:12 mglocker Exp $ */
+
+/*
+ * Copyright (c) 2006 Claudio Jeker <claudio@openbsd.org>
+ * Copyright (c) 2006 Marcus Glocker <mglocker@openbsd.org>
+ *
+ * Permission to use, copy, modify, and distribute this software for any
+ * purpose with or without fee is hereby granted, provided that the above
+ * copyright notice and this permission notice appear in all copies.
+ *
+ * THE SOFTWARE IS PROVIDED "AS IS" AND THE AUTHOR DISCLAIMS ALL WARRANTIES
+ * WITH REGARD TO THIS SOFTWARE INCLUDING ALL IMPLIED WARRANTIES OF
+ * MERCHANTABILITY AND FITNESS. IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR
+ * ANY SPECIAL, DIRECT, INDIRECT, OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES
+ * WHATSOEVER RESULTING FROM LOSS OF USE, DATA OR PROFITS, WHETHER IN AN
+ * ACTION OF CONTRACT, NEGLIGENCE OR OTHER TORTIOUS ACTION, ARISING OUT OF
+ * OR IN CONNECTION WITH THE USE OR PERFORMANCE OF THIS SOFTWARE.
+ */
+
+/*
* Copyright (c) 2004 Fujitsu Laboratories of America, Inc.
* Copyright (c) 2004 Brian Fundakowski Feldman
* All rights reserved.
@@ -74,14 +93,17 @@
#include <dev/ic/if_wi_ieee.h>
#include <dev/ic/if_wivar.h>
-#define PGT_DEBUG
-
#ifdef PGT_DEBUG
#define DPRINTF(x) do { printf x; } while (0)
#else
#define DPRINTF(x)
#endif
+#define SETOID(oid, var, size) { \
+ if (pgt_oid_set(sc, oid, var, size) != 0) \
+ break; \
+}
+
/*
* This is a driver for the Intersil Prism family of 802.11g network cards,
* based upon version 1.2 of the Linux driver and firmware found at
@@ -89,33 +111,22 @@
*/
/*
- * hack to get it compiled
+ * hack to get it compiled (from FreeBSD)
*/
-/* got from if_wi */
-#define WI_PRISM2_RES_SIZE 62
-#define WI_RID_SCAN_REQ 0xFCE1
-#define WI_RID_SCAN_APS 0x0600
-#define WI_RID_CHANNEL_LIST 0xFD10
-#define WI_RID_SCAN_RES 0xFD88
-#define WI_RID_COMMS_QUALITY 0xFD43
-/* got from FreeBSD */
-#define IEEE80211_WEP_OFF 0
-#define IEEE80211_WEP_ON 1
-#define IEEE80211_WEP_MIXED 2
#define IEEE80211_IOC_MLME 21
#define IEEE80211_IOC_AUTHMODE 7
-#define IEEE80211_IOC_WEP 3
-#define IEEE80211_IOC_WEPTXKEY 6
-#define IEEE80211_IOC_WEPKEY 4
#define IEEE80211_POWERSAVE_ON 1
#define IEEE80211_POWERSAVE_OFF 0
#define IEEE80211_MLME_UNAUTHORIZE 5
#define IEEE80211_MLME_AUTHORIZE 4
+#define SCAN_TIMEOUT 5 /* 5 seconds */
+
struct cfdriver pgt_cd = {
NULL, "pgt", DV_IFNET
};
+int pgt_media_change(struct ifnet *ifp);
void pgt_write_memory_barrier(struct pgt_softc *);
uint32_t pgt_read_4(struct pgt_softc *, uint16_t);
void pgt_write_4(struct pgt_softc *, uint16_t, uint32_t);
@@ -126,19 +137,13 @@ void pgt_reinit_rx_desc_frag(struct pgt_softc *, struct pgt_desc *);
int pgt_load_tx_desc_frag(struct pgt_softc *, enum pgt_queue,
struct pgt_desc *);
void pgt_unload_tx_desc_frag(struct pgt_softc *, struct pgt_desc *);
-void pgt_enter_critical(struct pgt_softc *);
-int pgt_try_enter_data_critical(struct pgt_softc *);
-void pgt_exit_critical(struct pgt_softc *);
-#if 0
-void pgt_try_exit_data_critical(struct pgt_softc *);
-#endif
int pgt_load_firmware(struct pgt_softc *);
void pgt_cleanup_queue(struct pgt_softc *, enum pgt_queue,
struct pgt_frag []);
int pgt_reset(struct pgt_softc *);
void pgt_disable(struct pgt_softc *, unsigned int);
void pgt_init_intr(struct pgt_softc *);
-void pgt_update_intr(struct pgt_softc *, struct mbuf ***, int);
+void pgt_update_intr(struct pgt_softc *, int);
struct mbuf
*pgt_ieee80211_encap(struct pgt_softc *, struct ether_header *,
struct mbuf *, struct ieee80211_node **);
@@ -147,19 +152,14 @@ void pgt_wakeup_intr(struct pgt_softc *);
void pgt_sleep_intr(struct pgt_softc *);
void pgt_empty_traps(struct pgt_softc_kthread *);
void pgt_per_device_kthread(void *);
-struct mbuf
- *pgt_alloc_async(size_t);
void pgt_async_reset(struct pgt_softc *);
-void pgt_async_trap(struct pgt_softc *, uint32_t, void *, size_t);
void pgt_async_update(struct pgt_softc *);
-//void pgt_poll(struct ifnet *, enum poll_cmd, int);
-void pgt_intr_body(struct pgt_softc *, struct mbuf **, int);
void pgt_txdone(struct pgt_softc *, enum pgt_queue);
void pgt_rxdone(struct pgt_softc *, enum pgt_queue);
void pgt_trap_received(struct pgt_softc *, uint32_t, void *, size_t);
void pgt_mgmtrx_completion(struct pgt_softc *, struct pgt_mgmt_desc *);
-int pgt_datarx_completion(struct pgt_softc *, enum pgt_queue,
- struct mbuf ***, int);
+struct mbuf
+ *pgt_datarx_completion(struct pgt_softc *, enum pgt_queue);
int pgt_oid_get(struct pgt_softc *, enum pgt_oid, void *, size_t);
int pgt_oid_retrieve(struct pgt_softc *, enum pgt_oid, void *, size_t);
int pgt_oid_set(struct pgt_softc *, enum pgt_oid, const void *, size_t);
@@ -180,27 +180,24 @@ void pgt_ieee80211_node_copy(struct ieee80211com *,
int pgt_ieee80211_send_mgmt(struct ieee80211com *,
struct ieee80211_node *, int, int);
int pgt_net_attach(struct pgt_softc *);
-void pgt_net_detach(struct pgt_softc *);
void pgt_start(struct ifnet *);
void pgt_start_body(struct pgt_softc *, struct ieee80211com *,
struct ifnet *);
int pgt_ioctl(struct ifnet *, u_long, caddr_t);
void pgt_obj_bss2scanres(struct pgt_softc *,
struct pgt_obj_bss *, struct wi_scan_res *, uint32_t);
+#if 0
int pgt_node_set_authorization(struct pgt_softc *,
struct pgt_ieee80211_node *,
enum pin_dot1x_authorization);
-#if 0
int pgt_do_mlme_sta(struct pgt_softc *, struct ieee80211req_mlme *);
int pgt_do_mlme_hostap(struct pgt_softc *, struct ieee80211req_mlme *);
int pgt_do_mlme_adhoc(struct pgt_softc *, struct ieee80211req_mlme *);
int pgt_80211_set(struct pgt_softc *, struct ieee80211req *);
#endif
-int pgt_wavelan_get(struct pgt_softc *, struct wi_req *);
-int pgt_wavelan_set(struct pgt_softc *, struct wi_req *);
void node_mark_active_ap(void *, struct ieee80211_node *);
void node_mark_active_adhoc(void *, struct ieee80211_node *);
-void pgt_periodic(struct ifnet *);
+void pgt_watchdog(struct ifnet *);
int pgt_init(struct ifnet *);
void pgt_update_hw_from_sw(struct pgt_softc *, int, int);
void pgt_update_hw_from_nodes(struct pgt_softc *);
@@ -208,9 +205,7 @@ void pgt_hostap_handle_mlme(struct pgt_softc *, uint32_t,
struct pgt_obj_mlme *);
void pgt_update_sw_from_hw(struct pgt_softc *,
struct pgt_async_trap *, struct mbuf *);
-int pgt_media_change(struct ifnet *);
-void pgt_media_status(struct ifnet *, struct ifmediareq *);
-int pgt_new_state(struct ieee80211com *, enum ieee80211_state, int);
+int pgt_newstate(struct ieee80211com *, enum ieee80211_state, int);
int pgt_drain_tx_queue(struct pgt_softc *, enum pgt_queue);
int pgt_dma_alloc(struct pgt_softc *);
int pgt_dma_alloc_queue(struct pgt_softc *sc, enum pgt_queue pq);
@@ -270,9 +265,9 @@ pgt_debug_events(struct pgt_softc *sc, const char *when)
letoh32(sc->sc_cb->pcb_driver_curfrag[i]) - \
letoh32(sc->sc_cb->pcb_device_curfrag[i])
if (sc->sc_debug & SC_DEBUG_EVENTS)
- printf("%s: ev%s: %u %u %u %u %u %u\n",
- sc->sc_dev.dv_xname, when,
- COUNT(0), COUNT(1), COUNT(2), COUNT(3), COUNT(4), COUNT(5));
+ DPRINTF(("%s: ev%s: %u %u %u %u %u %u\n",
+ sc->sc_dev.dv_xname, when, COUNT(0), COUNT(1), COUNT(2),
+ COUNT(3), COUNT(4), COUNT(5)));
#undef COUNT
}
@@ -288,7 +283,7 @@ pgt_reinit_rx_desc_frag(struct pgt_softc *sc, struct pgt_desc *pd)
{
pd->pd_fragp->pf_addr = htole32((uint32_t)pd->pd_dmaaddr);
pd->pd_fragp->pf_size = htole16(PGT_FRAG_SIZE);
- pd->pd_fragp->pf_flags = htole16(0);
+ pd->pd_fragp->pf_flags = 0;
bus_dmamap_sync(sc->sc_dmat, pd->pd_dmam, 0, pd->pd_dmam->dm_mapsize,
BUS_DMASYNC_POSTWRITE);
@@ -303,11 +298,12 @@ pgt_load_tx_desc_frag(struct pgt_softc *sc, enum pgt_queue pq,
error = bus_dmamap_load(sc->sc_dmat, pd->pd_dmam, pd->pd_mem,
PGT_FRAG_SIZE, NULL, BUS_DMA_NOWAIT);
if (error) {
- printf("%s: unable to load %s tx DMA: %d\n",
+ DPRINTF(("%s: unable to load %s tx DMA: %d\n",
sc->sc_dev.dv_xname,
- pgt_queue_is_data(pq) ? "data" : "mgmt", error);
+ pgt_queue_is_data(pq) ? "data" : "mgmt", error));
return (error);
}
+ pd->pd_dmaaddr = pd->pd_dmam->dm_segs[0].ds_addr;
pd->pd_fragp->pf_addr = htole32((uint32_t)pd->pd_dmaaddr);
pd->pd_fragp->pf_size = htole16(PGT_FRAG_SIZE);
pd->pd_fragp->pf_flags = htole16(0);
@@ -325,80 +321,12 @@ pgt_unload_tx_desc_frag(struct pgt_softc *sc, struct pgt_desc *pd)
pd->pd_dmaaddr = 0;
}
-/*
- * The critical lock uses int sc_critical to signify whether there are
- * currently transmissions in flight (> 0), is a management in progress
- * (<= -1), or the device is not critical (== 0). Management waiters
- * prevent further transmissions from contributing to sc_critical --
- * that is, make it monotonically decrease again back to 0 as
- * transmissions complete or time out.
- */
-void
-pgt_enter_critical(struct pgt_softc *sc)
-{
- //if (sc->sc_critical < 0 && sc->sc_critical_thread == curthread) {
- if (sc->sc_critical < 0) {
- sc->sc_critical--;
- } else {
- while (sc->sc_critical != 0) {
- sc->sc_refcnt++;
- //cv_wait(&sc->sc_critical_cv, &sc->sc_lock);
- sc->sc_refcnt--;
- }
- sc->sc_critical = -1;
- //sc->sc_critical_thread = curthread;
- }
-}
-
-void
-pgt_exit_critical(struct pgt_softc *sc)
-{
- if (++sc->sc_critical == 0) {
- sc->sc_critical_thread = NULL;
- //if (sc->sc_critical_cv.cv_waiters == 0 &&
- if (sc->sc_flags & SC_START_DESIRED) {
- sc->sc_flags &= ~SC_START_DESIRED;
- if (sc->sc_ic.ic_if.if_flags & IFF_RUNNING)
- pgt_start_body(sc, &sc->sc_ic,
- &sc->sc_ic.ic_if);
- } else {
- //cv_signal(&sc->sc_critical_cv);
- }
- }
-}
-
-int
-pgt_try_enter_data_critical(struct pgt_softc *sc)
-{
- if (sc->sc_critical <= -1) {
- // || sc->sc_critical_cv.cv_waiters > 0) {
- /*
- * Don't start new data packets if management
- * wants a chance.
- */
- sc->sc_flags |= SC_START_DESIRED;
- return (0);
- } else {
- sc->sc_critical++;
- return (1);
- }
-}
-
-#if 0
-void
-pgt_try_exit_data_critical(struct pgt_softc *sc)
-{
- if (--sc->sc_critical == 0)
- cv_signal(&sc->sc_critical_cv);
-}
-#endif
-
int
pgt_load_firmware(struct pgt_softc *sc)
{
int error, reg, dirreg, fwoff, ucodeoff, fwlen;
uint8_t *ucode;
- const uint32_t *uc;
+ uint32_t *uc;
size_t size;
char *name;
@@ -410,14 +338,14 @@ pgt_load_firmware(struct pgt_softc *sc)
error = loadfirmware(name, &ucode, &size);
if (error != 0) {
- printf("%s: error %d, could not read microcode %s!\n",
- sc->sc_dev.dv_xname, error, name);
+ DPRINTF(("%s: error %d, could not read microcode %s!\n",
+ sc->sc_dev.dv_xname, error, name));
return (EIO);
}
if (size & 3) {
- printf("%s: bad firmware size %u\n",
- sc->sc_dev.dv_xname, size);
+ DPRINTF(("%s: bad firmware size %u\n",
+ sc->sc_dev.dv_xname, size));
free(ucode, M_DEVBUF);
return (EINVAL);
}
@@ -426,7 +354,7 @@ pgt_load_firmware(struct pgt_softc *sc)
fwoff = 0;
ucodeoff = 0;
- uc = (const uint32_t *)ucode;
+ uc = (uint32_t *)ucode;
reg = PGT_FIRMWARE_INTERNAL_OFFSET;
while (fwoff < size) {
pgt_write_4_flush(sc, PGT_REG_DIR_MEM_BASE, reg);
@@ -484,8 +412,9 @@ pgt_cleanup_queue(struct pgt_softc *sc, enum pgt_queue pq,
struct pgt_desc *pd;
unsigned int i;
- sc->sc_cb->pcb_device_curfrag[pq] = htole32(0);
+ sc->sc_cb->pcb_device_curfrag[pq] = 0;
i = 0;
+ /* XXX why only freeq ??? */
TAILQ_FOREACH(pd, &sc->sc_freeq[pq], pd_link) {
pd->pd_fragnum = i;
pd->pd_fragp = &pqfrags[i];
@@ -502,7 +431,7 @@ pgt_cleanup_queue(struct pgt_softc *sc, enum pgt_queue pq,
if (pgt_queue_is_rx(pq))
sc->sc_cb->pcb_driver_curfrag[pq] = htole32(i);
else
- sc->sc_cb->pcb_driver_curfrag[pq] = htole32(0);
+ sc->sc_cb->pcb_driver_curfrag[pq] = 0;
}
/*
@@ -515,7 +444,7 @@ pgt_reset(struct pgt_softc *sc)
int error;
/* disable all interrupts */
- pgt_write_4_flush(sc, PGT_REG_INT_EN, 0x00000000);
+ pgt_write_4_flush(sc, PGT_REG_INT_EN, 0);
DELAY(PGT_WRITEIO_DELAY);
/*
@@ -554,9 +483,9 @@ pgt_reset(struct pgt_softc *sc)
}
/* upload the control block's DMA address */
- //pgt_write_4_flush(sc, PGT_REG_CTRL_BLK_BASE,
- // htole32((uint32_t)sc->sc_cbdmabusaddr));
- //DELAY(PGT_WRITEIO_DELAY);
+ pgt_write_4_flush(sc, PGT_REG_CTRL_BLK_BASE,
+ htole32((uint32_t)sc->sc_cbdmam->dm_segs[0].ds_addr));
+ DELAY(PGT_WRITEIO_DELAY);
/* send a reset event */
pgt_write_4_flush(sc, PGT_REG_DEV_INT, PGT_DEV_INT_RESET);
@@ -577,77 +506,27 @@ void
pgt_disable(struct pgt_softc *sc, unsigned int flag)
{
struct ieee80211com *ic;
- uint32_t reg;
unsigned int wokeup;
- int tries = 6, tryagain;
+ int tryagain = 0;
ic = &sc->sc_ic;
- if (flag == SC_DYING && sc->sc_flags & SC_DYING) {
- while (sc->sc_drainer != NULL);
- //(void)msleep(&sc->sc_drainer, &sc->sc_lock,
- // PZERO, "pffol1", hz / 10);
- //goto out2;
- return;
- }
- if (flag == SC_NEEDS_RESET) {
- if (sc->sc_drainer != NULL || sc->sc_flags & SC_GONE)
- /*
- * Multiple pending resets are not useful.
- */
- //goto out2;
- return;
- } else {
- while (sc->sc_drainer != NULL);
- //(void)msleep(&sc->sc_drainer, &sc->sc_lock,
- // PZERO, "pffol1", hz / 10);
- }
ic->ic_if.if_flags &= ~IFF_RUNNING;
sc->sc_flags |= SC_UNINITIALIZED;
-#ifdef DEVICE_POLLING
- ether_poll_deregister(&ic->ic_if);
- /* Turn back on interrupts. */
- pgt_write_4_flush(sc, PGT_REG_INT_EN, PGT_INT_STAT_SOURCES);
- DELAY(PGT_WRITEIO_DELAY);
-#endif
- //sc->sc_drainer = curthread;
sc->sc_flags |= flag;
- /*
- * The detacher has to wait for all activity to cease (the refcount
- * will reach 1 and sc_async_events will have been emptied). If
- * we were to drain while doing just a "reset" then this could
- * deadlock.
- */
+
pgt_drain_tx_queue(sc, PGT_QUEUE_DATA_LOW_TX);
pgt_drain_tx_queue(sc, PGT_QUEUE_DATA_HIGH_TX);
pgt_drain_tx_queue(sc, PGT_QUEUE_MGMT_TX);
- if (flag == SC_DYING) {
- while (sc->sc_refcnt > 1);
- //(void)msleep(&sc->sc_drainer, &sc->sc_lock,
- // PZERO, "pffol2", hz / 10);
- }
+
trying_again:
- tryagain = 0;
/* disable all interrupts */
- pgt_write_4_flush(sc, PGT_REG_INT_EN, 0x00000000);
- DELAY(PGT_WRITEIO_DELAY);
- if (sc->sc_intcookie != NULL) {
- //bus_teardown_intr(sc->sc_dev, sc->sc_intres, sc->sc_intcookie);
- sc->sc_intcookie = NULL;
- }
- reg = pgt_read_4(sc, PGT_REG_CTRL_STAT);
- reg &= ~(PGT_CTRL_STAT_RESET | PGT_CTRL_STAT_RAMBOOT);
- pgt_write_4(sc, PGT_REG_CTRL_STAT, reg);
- pgt_write_memory_barrier(sc);
- DELAY(PGT_WRITEIO_DELAY);
- reg |= PGT_CTRL_STAT_RESET;
- pgt_write_4(sc, PGT_REG_CTRL_STAT, reg);
- pgt_write_memory_barrier(sc);
- DELAY(PGT_WRITEIO_DELAY);
- reg &= ~PGT_CTRL_STAT_RESET;
- pgt_write_4(sc, PGT_REG_CTRL_STAT, reg);
- pgt_write_memory_barrier(sc);
+ pgt_write_4_flush(sc, PGT_REG_INT_EN, 0);
DELAY(PGT_WRITEIO_DELAY);
+
+ /* reboot card */
+ pgt_reboot(sc);
+
do {
wokeup = 0;
/*
@@ -663,51 +542,35 @@ trying_again:
pmd->pmd_error = ENETRESET;
wakeup_one(pmd);
if (sc->sc_debug & SC_DEBUG_MGMT)
- printf("%s: queue: mgmt %p <- 0x%x (drained)\n",
- sc->sc_dev.dv_xname, pmd, pmd->pmd_oid);
+ DPRINTF(("%s: queue: mgmt %p <- %#x "
+ "(drained)\n", sc->sc_dev.dv_xname,
+ pmd, pmd->pmd_oid));
wokeup++;
}
if (wokeup > 0) {
- //(void)msleep(&sc->sc_drainer, &sc->sc_lock,
- // PZERO, "pffol3", hz / 10);
if (flag == SC_NEEDS_RESET && sc->sc_flags & SC_DYING) {
sc->sc_flags &= ~flag;
- goto out;
+ return;
}
}
} while (wokeup > 0);
- if (flag == SC_NEEDS_RESET && !(sc->sc_flags & SC_GONE)) {
+
+ if (flag == SC_NEEDS_RESET) {
int error;
- printf("%s: resetting\n", sc->sc_dev.dv_xname);
- sc->sc_refcnt++;
+ DPRINTF(("%s: resetting\n", sc->sc_dev.dv_xname));
sc->sc_flags &= ~SC_POWERSAVE;
sc->sc_flags |= SC_NEEDS_FIRMWARE;
- //error = bus_setup_intr(sc->sc_dev, sc->sc_intres,
- // INTR_TYPE_NET | INTR_MPSAFE, pgt_intr,
- // &ic->ic_if, &sc->sc_intcookie);
- if (error != 0 || sc->sc_flags & SC_DYING) {
- if (error != 0) {
- printf("%s: failure establishing irq in "
- "reset: %d\n",
- sc->sc_dev.dv_xname, error);
- sc->sc_flags |= SC_DYING;
- }
- sc->sc_refcnt--;
- goto out;
- }
error = pgt_reset(sc);
if (error == 0) {
- //(void)msleep(&sc->sc_flags, &sc->sc_lock, PZERO,
- // "pffres", hz * 10);
+ tsleep(&sc->sc_flags, 0, "pftres", hz);
if (sc->sc_flags & SC_UNINITIALIZED) {
printf("%s: not responding\n",
sc->sc_dev.dv_xname);
/* Thud. It was probably removed. */
- if (--tries == 0)
- sc->sc_flags |= SC_GONE;
- else
- tryagain = 1;
+ if (tryagain)
+ panic("pgt went for lunch"); /* XXX */
+ tryagain = 1;
} else {
/* await all interrupts */
pgt_write_4_flush(sc, PGT_REG_INT_EN,
@@ -716,18 +579,19 @@ trying_again:
ic->ic_if.if_flags |= IFF_RUNNING;
}
}
- sc->sc_refcnt--;
+
if (tryagain)
goto trying_again;
+
sc->sc_flags &= ~flag;
if (ic->ic_if.if_flags & IFF_RUNNING)
pgt_update_hw_from_sw(sc,
ic->ic_state != IEEE80211_S_INIT,
ic->ic_opmode != IEEE80211_M_MONITOR);
}
-out:
- sc->sc_drainer = NULL;
- wakeup(&sc->sc_drainer);
+
+ ic->ic_if.if_flags &= ~(IFF_RUNNING | IFF_OACTIVE);
+ ieee80211_new_state(&sc->sc_ic, IEEE80211_S_INIT, -1);
}
int
@@ -735,34 +599,36 @@ pgt_attach(struct pgt_softc *sc)
{
int error;
+ /* debug flags */
+ //ieee80211_debug = 1;
+ //sc->sc_debug |= SC_DEBUG_QUEUES; /* super verbose */
+ //sc->sc_debug |= SC_DEBUG_MGMT;
+ sc->sc_debug |= SC_DEBUG_UNEXPECTED;
+ //sc->sc_debug |= SC_DEBUG_TRIGGER; /* verbose */
+ //sc->sc_debug |= SC_DEBUG_EVENTS; /* super verbose */
+ //sc->sc_debug |= SC_DEBUG_POWER;
+ sc->sc_debug |= SC_DEBUG_TRAP;
+ sc->sc_debug |= SC_DEBUG_LINK;
+ //sc->sc_debug |= SC_DEBUG_RXANNEX;
+ //sc->sc_debug |= SC_DEBUG_RXFRAG;
+ //sc->sc_debug |= SC_DEBUG_RXETHER;
+
error = pgt_dma_alloc(sc);
if (error)
return (error);
sc->sc_ic.ic_if.if_softc = sc;
- sc->sc_refcnt = 1;
TAILQ_INIT(&sc->sc_mgmtinprog);
TAILQ_INIT(&sc->sc_kthread.sck_traps);
sc->sc_flags |= SC_NEEDS_FIRMWARE | SC_UNINITIALIZED;
- /*
- error = bus_setup_intr(dev, sc->sc_intres, INTR_TYPE_NET | INTR_MPSAFE,
- pgt_intr, &sc->sc_ic.ic_if, &sc->sc_intcookie);
- if (error != 0) {
- printf("%s: failure establishing interrupt: %d\n",
- sc->sc_dev.dv_xname, error);
- goto failed;
- }
- */
- sc->sc_80211_ioc_wep = IEEE80211_WEP_OFF;
+
sc->sc_80211_ioc_auth = IEEE80211_AUTH_OPEN;
error = pgt_reset(sc);
if (error)
goto failed;
- sc->sc_refcnt++;
tsleep(&sc->sc_flags, 0, "pftres", hz);
- sc->sc_refcnt--;
if (sc->sc_flags & SC_UNINITIALIZED) {
printf("%s: not responding\n", sc->sc_dev.dv_xname);
error = ETIMEDOUT;
@@ -778,6 +644,10 @@ pgt_attach(struct pgt_softc *sc)
if (error)
goto failed;
+ if (kthread_create(pgt_per_device_kthread, sc, NULL,
+ sc->sc_dev.dv_xname) != 0)
+ goto failed;
+
ieee80211_new_state(&sc->sc_ic, IEEE80211_S_INIT, -1);
failed:
@@ -792,11 +662,15 @@ failed:
int
pgt_detach(struct pgt_softc *sc)
{
- pgt_net_detach(sc);
- sc->sc_flags |= SC_GONE;
+ /* stop card */
pgt_disable(sc, SC_DYING);
pgt_reboot(sc);
+ ieee80211_ifdetach(&sc->sc_ic.ic_if);
+ if_detach(&sc->sc_ic.ic_if);
+
+ pgt_dma_free(sc);
+
return (0);
}
@@ -827,8 +701,8 @@ pgt_init_intr(struct pgt_softc *sc)
{
if ((sc->sc_flags & SC_UNINITIALIZED) == 0) {
if (sc->sc_debug & SC_DEBUG_UNEXPECTED)
- printf("%s: spurious initialization\n",
- sc->sc_dev.dv_xname);
+ DPRINTF(("%s: spurious initialization\n",
+ sc->sc_dev.dv_xname));
} else {
sc->sc_flags &= ~SC_UNINITIALIZED;
wakeup(&sc->sc_flags);
@@ -840,8 +714,7 @@ pgt_init_intr(struct pgt_softc *sc)
* for new packets.
*/
void
-pgt_update_intr(struct pgt_softc *sc, struct mbuf ***last_nextpkt,
- int max_datarx_count)
+pgt_update_intr(struct pgt_softc *sc, int hack)
{
/* priority order */
enum pgt_queue pqs[PGT_QUEUE_COUNT] = {
@@ -849,9 +722,10 @@ pgt_update_intr(struct pgt_softc *sc, struct mbuf ***last_nextpkt,
PGT_QUEUE_DATA_HIGH_TX, PGT_QUEUE_DATA_HIGH_RX,
PGT_QUEUE_DATA_LOW_TX, PGT_QUEUE_DATA_LOW_RX
};
+ struct mbuf *m;
uint32_t npend;
unsigned int dirtycount;
- int i, prevwasmf;
+ int i;
bus_dmamap_sync(sc->sc_dmat, sc->sc_cbdmam, 0,
sc->sc_cbdmam->dm_mapsize,
@@ -875,7 +749,7 @@ pgt_update_intr(struct pgt_softc *sc, struct mbuf ***last_nextpkt,
data = pgt_queue_is_data(pqs[i]);
#ifdef PGT_BUGGY_INTERRUPT_RECOVERY
- if (last_nextpkt == NULL && data)
+ if (hack && data)
continue;
#endif
npend = pgt_queue_frags_pending(sc, pqs[i]);
@@ -884,31 +758,25 @@ pgt_update_intr(struct pgt_softc *sc, struct mbuf ***last_nextpkt,
* always be qtotal (qdirty is 0).
*/
if (npend > qfree) {
- if (sc->sc_flags & SC_DEBUG_UNEXPECTED)
- printf("%s: rx queue [%u] overflowed "
- "by %u\n",
- sc->sc_dev.dv_xname,
- pqs[i], npend - qfree);
+ if (sc->sc_debug & SC_DEBUG_UNEXPECTED)
+ DPRINTF(("%s: rx queue [%u] "
+ "overflowed by %u\n",
+ sc->sc_dev.dv_xname, pqs[i],
+ npend - qfree));
sc->sc_flags |= SC_INTR_RESET;
break;
}
while (qfree-- > npend) {
-#ifdef DEVICE_POLLING
- if (data && max_datarx_count != -1) {
- if (max_datarx_count-- == 0)
- break;
- }
-#endif
pgt_rxdone(sc, pqs[i]);
}
} else {
npend = pgt_queue_frags_pending(sc, pqs[i]);
if (npend > qdirty) {
- if (sc->sc_flags & SC_DEBUG_UNEXPECTED)
- printf("%s: tx queue [%u] underflowed "
- "by %u\n",
- sc->sc_dev.dv_xname,
- pqs[i], npend - qdirty);
+ if (sc->sc_debug & SC_DEBUG_UNEXPECTED)
+ DPRINTF(("%s: tx queue [%u] "
+ "underflowed by %u\n",
+ sc->sc_dev.dv_xname, pqs[i],
+ npend - qdirty));
sc->sc_flags |= SC_INTR_RESET;
break;
}
@@ -917,8 +785,11 @@ pgt_update_intr(struct pgt_softc *sc, struct mbuf ***last_nextpkt,
* queue just became empty, wake up any waiters.
*/
if (qdirty > npend) {
- if (TAILQ_EMPTY(&sc->sc_freeq[pqs[i]]))
- wakeup(&sc->sc_freeq[pqs[i]]);
+ if (pgt_queue_is_data(pqs[i])) {
+ sc->sc_ic.ic_if.if_timer = 0;
+ sc->sc_ic.ic_if.if_flags &=
+ ~IFF_OACTIVE;
+ }
while (qdirty-- > npend)
pgt_txdone(sc, pqs[i]);
}
@@ -946,19 +817,19 @@ pgt_update_intr(struct pgt_softc *sc, struct mbuf ***last_nextpkt,
letoh32(sc->sc_cb->pcb_driver_curfrag[PGT_QUEUE_MGMT_RX]));
dirtycount = sc->sc_dirtyq_count[PGT_QUEUE_DATA_HIGH_RX];
- prevwasmf = 0;
- while (!TAILQ_EMPTY(&sc->sc_dirtyq[PGT_QUEUE_DATA_HIGH_RX]))
- prevwasmf = pgt_datarx_completion(sc, PGT_QUEUE_DATA_HIGH_RX,
- last_nextpkt, prevwasmf);
+ while (!TAILQ_EMPTY(&sc->sc_dirtyq[PGT_QUEUE_DATA_HIGH_RX])) {
+ if ((m = pgt_datarx_completion(sc, PGT_QUEUE_DATA_HIGH_RX)))
+ pgt_input_frames(sc, m);
+ }
sc->sc_cb->pcb_driver_curfrag[PGT_QUEUE_DATA_HIGH_RX] =
htole32(dirtycount +
letoh32(sc->sc_cb->pcb_driver_curfrag[PGT_QUEUE_DATA_HIGH_RX]));
dirtycount = sc->sc_dirtyq_count[PGT_QUEUE_DATA_LOW_RX];
- prevwasmf = 0;
- while (!TAILQ_EMPTY(&sc->sc_dirtyq[PGT_QUEUE_DATA_LOW_RX]))
- prevwasmf = pgt_datarx_completion(sc, PGT_QUEUE_DATA_LOW_RX,
- last_nextpkt, prevwasmf);
+ while (!TAILQ_EMPTY(&sc->sc_dirtyq[PGT_QUEUE_DATA_LOW_RX])) {
+ if ((m = pgt_datarx_completion(sc, PGT_QUEUE_DATA_LOW_RX)))
+ pgt_input_frames(sc, m);
+ }
sc->sc_cb->pcb_driver_curfrag[PGT_QUEUE_DATA_LOW_RX] =
htole32(dirtycount +
letoh32(sc->sc_cb->pcb_driver_curfrag[PGT_QUEUE_DATA_LOW_RX]));
@@ -980,9 +851,9 @@ pgt_ieee80211_encap(struct pgt_softc *sc, struct ether_header *eh,
struct llc *snap;
ic = &sc->sc_ic;
- M_PREPEND(m, sizeof(frame) + sizeof(snap), M_DONTWAIT);
+ M_PREPEND(m, sizeof(*frame) + sizeof(*snap), M_DONTWAIT);
if (m != NULL)
- m = m_pullup(m, sizeof(frame) + sizeof(snap));
+ m = m_pullup(m, sizeof(*frame) + sizeof(*snap));
if (m == NULL)
return (m);
frame = mtod(m, struct ieee80211_frame *);
@@ -1065,7 +936,7 @@ pgt_input_frames(struct pgt_softc *sc, struct mbuf *m)
struct ieee80211com *ic;
struct pgt_data_frame pdf;
struct pgt_rx_annex *pra;
- struct mbuf *next, *m2;
+ struct mbuf *next;
unsigned int n;
uint32_t rstamp;
uint16_t dataoff;
@@ -1077,51 +948,7 @@ pgt_input_frames(struct pgt_softc *sc, struct mbuf *m)
for (next = m; m != NULL; m = next) {
next = m->m_nextpkt;
m->m_nextpkt = NULL;
- //if (m->m_flags & M_PROTO2) {
- if (m->m_flags) {
- /*
- * We either ended up losing the previous
- * frag, or we're trying to receive more than
- * two of them.
- */
- ifp->if_ierrors++;
- m_freem(m);
- continue;
- }
- if (m->m_flags & M_PROTO1) {
- if (next == NULL) {
- if (sc->sc_debug & SC_DEBUG_UNEXPECTED)
- printf("%s: more frags set, but not "
- "found\n",
- sc->sc_dev.dv_xname);
- ifp->if_ierrors++;
- m_freem(m);
- continue;
- } else {
- //if (!(next->m_flags & M_PROTO2)) {
- if (!(next->m_flags)) {
- if (sc->sc_debug & SC_DEBUG_UNEXPECTED)
- printf("%s: more frags set, "
- "but next is not one\n",
- sc->sc_dev.dv_xname);
- ifp->if_ierrors++;
- m_freem(m);
- continue;
- }
- /*
- * If there are yet more frags after the
- * second, we're not touching them here.
- */
- //next->m_flags &= ~(M_PROTO1 | M_PROTO2);
- }
- m->m_flags &= ~M_PROTO1;
- m2 = next;
- next = m2->m_nextpkt;
- m2->m_nextpkt = NULL;
- /* Remove any preceding junk from the latter frag. */
- m_adj(m2, *mtod(m2, uint16_t *));
- m_cat(m, m2);
- }
+
dataoff = *mtod(m, uint16_t *);
m_adj(m, 2);
if (dataoff < sizeof(pdf)) {
@@ -1134,17 +961,19 @@ pgt_input_frames(struct pgt_softc *sc, struct mbuf *m)
}
bcopy(mtod(m, struct pgt_data_frame *), &pdf, sizeof(pdf));
m_adj(m, dataoff);
- m = m_pullup(m, sizeof(*pra));
- if (m == NULL) {
- if (sc->sc_debug & SC_DEBUG_UNEXPECTED)
- printf("%s: m_pullup failure\n",
- sc->sc_dev.dv_xname);
- ifp->if_ierrors++;
- continue;
+ if (m->m_len < sizeof(*pra)) {
+ m = m_pullup(m, sizeof(*pra));
+ if (m == NULL) {
+ if (sc->sc_debug & SC_DEBUG_UNEXPECTED)
+ printf("%s: m_pullup failure\n",
+ sc->sc_dev.dv_xname);
+ ifp->if_ierrors++;
+ continue;
+ }
}
pra = mtod(m, struct pgt_rx_annex *);
if (sc->sc_debug & SC_DEBUG_RXANNEX)
- printf("%s: rx annex: ? %04x ? %04x "
+ DPRINTF(("%s: rx annex: ? %04x ? %04x "
"len %u clock %u flags %02x ? %02x rate %u ? %02x "
"freq %u ? %04x rssi %u pad %02x%02x%02x\n",
sc->sc_dev.dv_xname,
@@ -1155,9 +984,9 @@ pgt_input_frames(struct pgt_softc *sc, struct mbuf *m)
pra->pra_unknown1, pra->pra_rate,
pra->pra_unknown2, letoh32(pra->pra_frequency),
pra->pra_unknown3, pra->pra_rssi,
- pra->pra_pad[0], pra->pra_pad[1], pra->pra_pad[2]);
+ pra->pra_pad[0], pra->pra_pad[1], pra->pra_pad[2]));
if (sc->sc_debug & SC_DEBUG_RXETHER)
- printf("%s: rx ether: "
+ DPRINTF(("%s: rx ether: "
"%02x:%02x:%02x:%02x:%02x:%02x < "
"%02x:%02x:%02x:%02x:%02x:%02x 0x%04x\n",
sc->sc_dev.dv_xname,
@@ -1167,7 +996,7 @@ pgt_input_frames(struct pgt_softc *sc, struct mbuf *m)
pra->pra_ether_shost[0], pra->pra_ether_shost[1],
pra->pra_ether_shost[2], pra->pra_ether_shost[3],
pra->pra_ether_shost[4], pra->pra_ether_shost[5],
- ntohs(pra->pra_ether_type));
+ ntohs(pra->pra_ether_type)));
/*
* This flag is set if e.g. packet could not be decrypted.
*/
@@ -1193,7 +1022,7 @@ pgt_input_frames(struct pgt_softc *sc, struct mbuf *m)
* two bits seem to not be set in any known
* circumstances.
*/
- encrypted = sc->sc_80211_ioc_wep != IEEE80211_WEP_OFF &&
+ encrypted = sc->sc_ic.ic_flags == IEEE80211_F_WEPON &&
(letoh16(pdf.pdf_unknown) & 0xc) == 0x8;
memcpy(eh.ether_dhost, pra->pra_ether_dhost, ETHER_ADDR_LEN);
memcpy(eh.ether_shost, pra->pra_ether_shost, ETHER_ADDR_LEN);
@@ -1212,12 +1041,12 @@ pgt_input_frames(struct pgt_softc *sc, struct mbuf *m)
else
chan = ic->ic_bss->ni_chan;
/* Send to 802.3 listeners. */
- m_adj(m, sizeof(*pra) - sizeof(eh));
- memcpy(mtod(m, struct ether_header *), &eh, sizeof(eh));
- m_adj(m, sizeof(eh));
+ m_adj(m, sizeof(*pra));
m = pgt_ieee80211_encap(sc, &eh, m, &ni);
if (m != NULL) {
+#if NBPFILTER > 0
if (sc->sc_drvbpf != NULL) {
+#if 0
struct pgt_ieee80211_radiotap pir;
bzero(&pir, sizeof(pir));
@@ -1232,8 +1061,26 @@ pgt_input_frames(struct pgt_softc *sc, struct mbuf *m)
pir.pir_channel_flags = htole16(chan->ic_flags);
pir.pir_db_antsignal = rssi;
pir.pir_db_antnoise = sc->sc_noise;
- //bpf_mtap2(sc->sc_drvbpf, &pir, sizeof(pir), m);
+ bpf_mtap2(sc->sc_drvbpf, &pir, sizeof(pir), m);
+#endif
+ struct mbuf mb;
+ struct pgt_rx_radiotap_hdr *tap = &sc->sc_rxtap;
+
+ tap->wr_flags = 0;
+ tap->wr_chan_freq = htole16(chan->ic_freq);
+ tap->wr_chan_flags = htole16(chan->ic_flags);
+ tap->wr_rssi = rssi;
+ //tap->wr_max_rssi = ic->ic_max_rssi;
+
+ M_DUP_PKTHDR(&mb, m);
+ mb.m_data = (caddr_t)tap;
+ mb.m_len = sc->sc_rxtap_len;
+ mb.m_next = m;
+ mb.m_pkthdr.len += mb.m_len;
+
+ bpf_mtap(sc->sc_drvbpf, &mb, BPF_DIRECTION_IN);
}
+#endif
ni->ni_rssi = rssi;
ni->ni_rstamp = rstamp;
ieee80211_input(ifp, m, ni, rssi, rstamp);
@@ -1333,29 +1180,35 @@ pgt_per_device_kthread(void *argp)
struct pgt_softc_kthread *sck;
struct pgt_async_trap *pa;
struct mbuf *m;
+ int s;
sc = argp;
sck = &sc->sc_kthread;
while (!sck->sck_exit) {
if (!sck->sck_update && !sck->sck_reset &&
- TAILQ_EMPTY(&sck->sck_traps)) {
- //cv_wait(&sck->sck_needed, &sc->sc_lock);
- }
+ TAILQ_EMPTY(&sck->sck_traps))
+ tsleep(&sc->sc_kthread, 0, "pgtkth", 0);
if (sck->sck_reset) {
+ DPRINTF(("%s: [thread] async reset\n",
+ sc->sc_dev.dv_xname));
sck->sck_reset = 0;
sck->sck_update = 0;
pgt_empty_traps(sck);
- pgt_enter_critical(sc);
+ s = splnet();
pgt_disable(sc, SC_NEEDS_RESET);
- pgt_exit_critical(sc);
+ splx(s);
} else if (!TAILQ_EMPTY(&sck->sck_traps)) {
+ DPRINTF(("%s: [thread] got a trap\n",
+ sc->sc_dev.dv_xname));
pa = TAILQ_FIRST(&sck->sck_traps);
TAILQ_REMOVE(&sck->sck_traps, pa, pa_link);
m = pa->pa_mbuf;
m_adj(m, sizeof(*pa));
pgt_update_sw_from_hw(sc, pa, m);
m_freem(m);
- } else {
+ } else if (sck->sck_update) {
+ //DPRINTF(("%s: [thread] update_sw_from_hw\n",
+ // sc->sc_dev.dv_xname));
sck->sck_update = 0;
pgt_update_sw_from_hw(sc, NULL, NULL);
}
@@ -1364,51 +1217,13 @@ pgt_per_device_kthread(void *argp)
kthread_exit(0);
}
-struct mbuf *
-pgt_alloc_async(size_t trapdata)
-{
- struct mbuf *m;
- size_t total;
-
- total = trapdata + sizeof(struct pgt_async_trap);
- if (total >= MINCLSIZE)
- MGETHDR(m, MT_DATA, 0);
- else
- m = m_get(M_DONTWAIT, MT_DATA);
- if (m != NULL)
- m->m_len = total;
- return (m);
-}
-
void
pgt_async_reset(struct pgt_softc *sc)
{
if (sc->sc_flags & (SC_DYING | SC_NEEDS_RESET))
return;
sc->sc_kthread.sck_reset = 1;
- //cv_signal(&sc->sc_kthread.sck_needed);
-}
-
-void
-pgt_async_trap(struct pgt_softc *sc, uint32_t oid, void *data, size_t len)
-{
- struct pgt_async_trap *pa;
- struct mbuf *m;
- char *p;
-
- if (sc->sc_flags & SC_DYING)
- return;
- m = pgt_alloc_async(sizeof(oid) + len);
- if (m == NULL)
- return;
- pa = mtod(m, struct pgt_async_trap *);
- p = mtod(m, char *) + sizeof(*pa);
- *(uint32_t *)p = oid;
- p += sizeof(uint32_t);
- memcpy(p, data, len);
- pa->pa_mbuf = m;
- TAILQ_INSERT_TAIL(&sc->sc_kthread.sck_traps, pa, pa_link);
- //cv_signal(&sc->sc_kthread.sck_needed);
+ wakeup(&sc->sc_kthread);
}
void
@@ -1417,85 +1232,19 @@ pgt_async_update(struct pgt_softc *sc)
if (sc->sc_flags & SC_DYING)
return;
sc->sc_kthread.sck_update = 1;
- //cv_signal(&sc->sc_kthread.sck_needed);
+ wakeup(&sc->sc_kthread);
}
-#ifdef DEVICE_POLLING
-void
-pgt_poll(struct ifnet *ifp, enum poll_cmd cmd, int count)
-{
- struct pgt_softc *sc;
- struct mbuf *datarx = NULL;
-
- sc = ifp->if_softc;
- if (!(ifp->if_capenable & IFCAP_POLLING)) {
- ether_poll_deregister(ifp); /* already have Giant, no LOR */
- cmd = POLL_DEREGISTER;
- }
- if (cmd == POLL_DEREGISTER) { /* final call, enable interrupts */
- pgt_write_4_flush(sc, PGT_REG_INT_EN, PGT_INT_STAT_SOURCES);
- DELAY(PGT_WRITEIO_DELAY);
- return;
- }
- pgt_intr_body(sc, &datarx, count);
- if (cmd == POLL_AND_CHECK_STATUS) {
- /* Do more expensive periodic stuff. */
- pgt_async_update(sc);
- }
- /*
- * Now that we have unlocked the softc, decode and enter the
- * data frames we've received.
- */
- if (datarx != NULL)
- pgt_input_frames(sc, datarx);
- //if (!IFQ_DRV_IS_EMPTY(&sc->sc_ic.ic_if.if_snd))
- // pgt_start(&sc->sc_ic.ic_if);
-}
-#endif
-
int
pgt_intr(void *arg)
{
struct pgt_softc *sc;
struct ifnet *ifp;
- struct mbuf *datarx = NULL;
+ u_int32_t reg;
sc = arg;
ifp = &sc->sc_ic.ic_if;
-#ifdef DEVICE_POLLING
- if (ifp->if_flags & IFF_POLLING)
- return;
- if (ifp->if_capenable & IFCAP_POLLING &&
- !(sc->sc_flags & SC_UNINITIALIZED) &&
- ether_poll_register(pgt_poll, ifp)) {
- /* Turn off interrupts. */
- pgt_write_4_flush(sc, PGT_REG_INT_EN, 0);
- DELAY(PGT_WRITEIO_DELAY);
- pgt_poll(ifp, POLL_ONLY, 1);
- return;
- }
-#endif
- pgt_intr_body(sc, &datarx, -1);
-
- /*
- * Now that we have unlocked the softc, decode and enter the
- * data frames we've received.
- */
- if (datarx != NULL)
- pgt_input_frames(sc, datarx);
- //if (!IFQ_DRV_IS_EMPTY(&ifp->if_snd))
- // pgt_start(ifp);
-
- return (0);
-}
-
-void
-pgt_intr_body(struct pgt_softc *sc, struct mbuf **datarx,
- int max_datarx_count)
-{
- u_int32_t reg;
-
/*
* Here the Linux driver ands in the value of the INT_EN register,
* and masks off everything but the documented interrupt bits. Why?
@@ -1510,52 +1259,52 @@ pgt_intr_body(struct pgt_softc *sc, struct mbuf **datarx,
*/
reg = pgt_read_4(sc, PGT_REG_CTRL_STAT);
if (reg & PGT_CTRL_STAT_SLEEPMODE)
- return;
+ return (0);
}
-#ifdef DEVICE_POLLING
- if (sc->sc_ic.ic_if.if_flags & IFF_POLLING)
- reg = PGT_INT_STAT_UPDATE;
- else
-#endif
- reg = pgt_read_4(sc, PGT_REG_INT_STAT);
- if (reg != 0) {
-#ifdef DEVICE_POLLING
- if (!(sc->sc_ic.ic_if.if_flags & IFF_POLLING))
-#endif
- pgt_write_4_flush(sc, PGT_REG_INT_ACK, reg);
- if (reg & PGT_INT_STAT_INIT)
- pgt_init_intr(sc);
- if (reg & PGT_INT_STAT_UPDATE) {
- pgt_update_intr(sc, &datarx, max_datarx_count);
- /*
- * If we got an update, it's not really asleep.
- */
- sc->sc_flags &= ~SC_POWERSAVE;
- /*
- * Pretend I have any idea what the documentation
- * would say, and just give it a shot sending an
- * "update" after acknowledging the interrupt
- * bits and writing out the new control block.
- */
- pgt_write_4_flush(sc, PGT_REG_DEV_INT,
- PGT_DEV_INT_UPDATE);
- DELAY(PGT_WRITEIO_DELAY);
- }
- if (reg & PGT_INT_STAT_SLEEP && !(reg & PGT_INT_STAT_WAKEUP))
- pgt_sleep_intr(sc);
- if (reg & PGT_INT_STAT_WAKEUP)
- pgt_wakeup_intr(sc);
+ reg = pgt_read_4(sc, PGT_REG_INT_STAT);
+ if (reg == 0)
+ return (0); /* This interrupt is not from us */
+
+ pgt_write_4_flush(sc, PGT_REG_INT_ACK, reg);
+ if (reg & PGT_INT_STAT_INIT)
+ pgt_init_intr(sc);
+ if (reg & PGT_INT_STAT_UPDATE) {
+ pgt_update_intr(sc, 0);
+ /*
+ * If we got an update, it's not really asleep.
+ */
+ sc->sc_flags &= ~SC_POWERSAVE;
+ /*
+ * Pretend I have any idea what the documentation
+ * would say, and just give it a shot sending an
+ * "update" after acknowledging the interrupt
+ * bits and writing out the new control block.
+ */
+ pgt_write_4_flush(sc, PGT_REG_DEV_INT,
+ PGT_DEV_INT_UPDATE);
+ DELAY(PGT_WRITEIO_DELAY);
}
+ if (reg & PGT_INT_STAT_SLEEP && !(reg & PGT_INT_STAT_WAKEUP))
+ pgt_sleep_intr(sc);
+ if (reg & PGT_INT_STAT_WAKEUP)
+ pgt_wakeup_intr(sc);
+
if (sc->sc_flags & SC_INTR_RESET) {
sc->sc_flags &= ~SC_INTR_RESET;
pgt_async_reset(sc);
}
+
if (reg & ~PGT_INT_STAT_SOURCES && sc->sc_debug & SC_DEBUG_UNEXPECTED) {
printf("%s: unknown interrupt bits %#x (stat %#x)\n",
sc->sc_dev.dv_xname,
reg & ~PGT_INT_STAT_SOURCES,
pgt_read_4(sc, PGT_REG_CTRL_STAT));
}
+
+ if (!IFQ_IS_EMPTY(&ifp->if_snd))
+ pgt_start(ifp);
+
+ return (1);
}
void
@@ -1573,24 +1322,20 @@ pgt_txdone(struct pgt_softc *sc, enum pgt_queue pq)
BUS_DMASYNC_POSTREAD);
/* Management frames want completion information. */
if (sc->sc_debug & SC_DEBUG_QUEUES) {
- printf("%s: queue: tx %u <- [%u]\n",
- sc->sc_dev.dv_xname, pd->pd_fragnum, pq);
+ DPRINTF(("%s: queue: tx %u <- [%u]\n",
+ sc->sc_dev.dv_xname, pd->pd_fragnum, pq));
if (sc->sc_debug & SC_DEBUG_MGMT && pgt_queue_is_mgmt(pq)) {
struct pgt_mgmt_frame *pmf;
pmf = (struct pgt_mgmt_frame *)pd->pd_mem;
- printf("%s: queue: txmgmt %p <- "
- "(ver %u, op %u, flags 0x%x)\n",
+ DPRINTF(("%s: queue: txmgmt %p <- "
+ "(ver %u, op %u, flags %#x)\n",
sc->sc_dev.dv_xname,
pd, pmf->pmf_version, pmf->pmf_operation,
- pmf->pmf_flags);
+ pmf->pmf_flags));
}
}
pgt_unload_tx_desc_frag(sc, pd);
- /*
- if (pgt_queue_is_data(pq))
- pgt_try_exit_data_critical(sc);
- */
}
void
@@ -1607,12 +1352,12 @@ pgt_rxdone(struct pgt_softc *sc, enum pgt_queue pq)
pd->pd_dmam->dm_mapsize,
BUS_DMASYNC_POSTREAD);
if (sc->sc_debug & SC_DEBUG_QUEUES)
- printf("%s: queue: rx %u <- [%u]\n",
- sc->sc_dev.dv_xname, pd->pd_fragnum, pq);
+ DPRINTF(("%s: queue: rx %u <- [%u]\n",
+ sc->sc_dev.dv_xname, pd->pd_fragnum, pq));
if (sc->sc_debug & SC_DEBUG_UNEXPECTED &&
pd->pd_fragp->pf_flags & ~htole16(PF_FLAG_MF))
- printf("%s: unknown flags on rx [%u]: 0x%x\n",
- sc->sc_dev.dv_xname, pq, letoh16(pd->pd_fragp->pf_flags));
+ DPRINTF(("%s: unknown flags on rx [%u]: %#x\n",
+ sc->sc_dev.dv_xname, pq, letoh16(pd->pd_fragp->pf_flags)));
}
/*
@@ -1625,7 +1370,41 @@ void
pgt_trap_received(struct pgt_softc *sc, uint32_t oid, void *trapdata,
size_t size)
{
- pgt_async_trap(sc, oid, trapdata, size);
+ struct pgt_async_trap *pa;
+ struct mbuf *m;
+ char *p;
+ size_t total;
+
+ if (sc->sc_flags & SC_DYING)
+ return;
+
+ total = sizeof(oid) + size + sizeof(struct pgt_async_trap);
+ if (total >= MINCLSIZE) {
+ MGETHDR(m, M_DONTWAIT, MT_DATA);
+ if (m == NULL)
+ return;
+ MCLGET(m, M_DONTWAIT);
+ if (!(m->m_flags & M_EXT)) {
+ m_freem(m);
+ m = NULL;
+ }
+ } else
+ m = m_get(M_DONTWAIT, MT_DATA);
+
+ if (m == NULL)
+ return;
+ else
+ m->m_len = total;
+
+ pa = mtod(m, struct pgt_async_trap *);
+ p = mtod(m, char *) + sizeof(*pa);
+ *(uint32_t *)p = oid;
+ p += sizeof(uint32_t);
+ memcpy(p, trapdata, size);
+ pa->pa_mbuf = m;
+
+ TAILQ_INSERT_TAIL(&sc->sc_kthread.sck_traps, pa, pa_link);
+ wakeup(&sc->sc_kthread);
}
/*
@@ -1647,29 +1426,29 @@ pgt_mgmtrx_completion(struct pgt_softc *sc, struct pgt_mgmt_desc *pmd)
sc->sc_freeq_count[PGT_QUEUE_MGMT_RX]++;
if (letoh16(pd->pd_fragp->pf_size) < sizeof(*pmf)) {
if (sc->sc_debug & SC_DEBUG_UNEXPECTED)
- printf("%s: mgmt desc too small: %u\n",
+ DPRINTF(("%s: mgmt desc too small: %u\n",
sc->sc_dev.dv_xname,
- letoh16(pd->pd_fragp->pf_size));
+ letoh16(pd->pd_fragp->pf_size)));
goto out_nopmd;
}
pmf = (struct pgt_mgmt_frame *)pd->pd_mem;
if (pmf->pmf_version != PMF_VER) {
if (sc->sc_debug & SC_DEBUG_UNEXPECTED)
- printf("%s: unknown mgmt version %u\n",
- sc->sc_dev.dv_xname, pmf->pmf_version);
+ DPRINTF(("%s: unknown mgmt version %u\n",
+ sc->sc_dev.dv_xname, pmf->pmf_version));
goto out_nopmd;
}
if (pmf->pmf_device != PMF_DEV) {
if (sc->sc_debug & SC_DEBUG_UNEXPECTED)
- printf("%s: unknown mgmt dev %u\n",
- sc->sc_dev.dv_xname, pmf->pmf_device);
+ DPRINTF(("%s: unknown mgmt dev %u\n",
+ sc->sc_dev.dv_xname, pmf->pmf_device));
goto out;
}
if (pmf->pmf_flags & ~PMF_FLAG_VALID) {
if (sc->sc_debug & SC_DEBUG_UNEXPECTED)
- printf("%s: unknown mgmt flags %u\n",
+ DPRINTF(("%s: unknown mgmt flags %x\n",
sc->sc_dev.dv_xname,
- pmf->pmf_flags & ~PMF_FLAG_VALID);
+ pmf->pmf_flags & ~PMF_FLAG_VALID));
goto out;
}
if (pmf->pmf_flags & PMF_FLAG_LE) {
@@ -1681,15 +1460,18 @@ pgt_mgmtrx_completion(struct pgt_softc *sc, struct pgt_mgmt_desc *pmd)
}
if (pmf->pmf_operation == PMF_OP_TRAP) {
pmd = NULL; /* ignored */
+ DPRINTF(("%s: mgmt trap received "
+ "(op %u, oid %#x, len %u)\n", sc->sc_dev.dv_xname,
+ pmf->pmf_operation, oid, size));
pgt_trap_received(sc, oid, (char *)pmf + sizeof(*pmf),
min(size, PGT_FRAG_SIZE - sizeof(*pmf)));
goto out_nopmd;
}
if (pmd == NULL) {
if (sc->sc_debug & (SC_DEBUG_UNEXPECTED | SC_DEBUG_MGMT))
- printf("%s: spurious mgmt received "
- "(op %u, oid 0x%x, len %u\n",
- sc->sc_dev.dv_xname, pmf->pmf_operation, oid, size);
+ DPRINTF(("%s: spurious mgmt received "
+ "(op %u, oid %#x, len %u)\n", sc->sc_dev.dv_xname,
+ pmf->pmf_operation, oid, size));
goto out_nopmd;
}
switch (pmf->pmf_operation) {
@@ -1701,24 +1483,22 @@ pgt_mgmtrx_completion(struct pgt_softc *sc, struct pgt_mgmt_desc *pmd)
goto out;
default:
if (sc->sc_debug & SC_DEBUG_UNEXPECTED)
- printf("%s: unknown mgmt op %u\n",
- sc->sc_dev.dv_xname, pmf->pmf_operation);
+ DPRINTF(("%s: unknown mgmt op %u\n",
+ sc->sc_dev.dv_xname, pmf->pmf_operation));
pmd->pmd_error = EIO;
goto out;
}
if (oid != pmd->pmd_oid) {
if (sc->sc_debug & SC_DEBUG_UNEXPECTED)
- printf("%s: mgmt oid changed from 0x%x "
- "-> 0x%x\n",
- sc->sc_dev.dv_xname, pmd->pmd_oid, oid);
+ DPRINTF(("%s: mgmt oid changed from %#x -> %#x\n",
+ sc->sc_dev.dv_xname, pmd->pmd_oid, oid));
pmd->pmd_oid = oid;
}
if (pmd->pmd_recvbuf != NULL) {
if (size > PGT_FRAG_SIZE) {
if (sc->sc_debug & SC_DEBUG_UNEXPECTED)
- printf("%s: mgmt oid 0x%x "
- "has bad size %u\n",
- sc->sc_dev.dv_xname, oid, size);
+ DPRINTF(("%s: mgmt oid %#x has bad size %u\n",
+ sc->sc_dev.dv_xname, oid, size));
pmd->pmd_error = EIO;
goto out;
}
@@ -1729,17 +1509,14 @@ pgt_mgmtrx_completion(struct pgt_softc *sc, struct pgt_mgmt_desc *pmd)
size);
pmd->pmd_len = size;
}
+
out:
TAILQ_REMOVE(&sc->sc_mgmtinprog, pmd, pmd_link);
-#ifdef DEVICE_POLLING
- if (!(sc->sc_ic.ic_if.if_flags & IFF_POLLING))
-#endif
- wakeup_one(pmd);
+ wakeup_one(pmd);
if (sc->sc_debug & SC_DEBUG_MGMT)
- printf("%s: queue: mgmt %p <- (op %u, "
- "oid 0x%x, len %u)\n",
+ DPRINTF(("%s: queue: mgmt %p <- (op %u, oid %#x, len %u)\n",
sc->sc_dev.dv_xname, pmd, pmf->pmf_operation,
- pmd->pmd_oid, pmd->pmd_len);
+ pmd->pmd_oid, pmd->pmd_len));
out_nopmd:
pgt_reinit_rx_desc_frag(sc, pd);
}
@@ -1749,75 +1526,106 @@ out_nopmd:
* whether the rx queue being full enough to start, but not finish,
* queueing a fragmented packet, can happen.
*/
-int
-pgt_datarx_completion(struct pgt_softc *sc, enum pgt_queue pq,
- struct mbuf ***last_nextpkt, int prevwasmf)
+struct mbuf *
+pgt_datarx_completion(struct pgt_softc *sc, enum pgt_queue pq)
{
struct ifnet *ifp;
struct pgt_desc *pd;
- struct mbuf *m;
+ struct mbuf *top, **mp, *m;
size_t datalen;
- uint16_t dataoff;
- int morefrags;
+ uint16_t morefrags, dataoff;
+ int tlen = 0;
ifp = &sc->sc_ic.ic_if;
- pd = TAILQ_FIRST(&sc->sc_dirtyq[pq]);
- TAILQ_REMOVE(&sc->sc_dirtyq[pq], pd, pd_link);
- sc->sc_dirtyq_count[pq]--;
- datalen = letoh16(pd->pd_fragp->pf_size);
- dataoff = letoh32(pd->pd_fragp->pf_addr) - pd->pd_dmaaddr;
- morefrags = pd->pd_fragp->pf_flags & htole16(PF_FLAG_MF);
- if (sc->sc_debug & SC_DEBUG_RXFRAG)
- printf("%s: rx frag: len %u memoff %u\n",
- sc->sc_dev.dv_xname, datalen, dataoff);
- /* Add the (two+?) bytes for the header. */
- datalen += dataoff;
- if (datalen > PGT_FRAG_SIZE) {
- if (sc->sc_debug & SC_DEBUG_UNEXPECTED)
- printf("%s data rx too big: %u\n",
- sc->sc_dev.dv_xname, datalen);
- ifp->if_ierrors++;
- goto out;
- }
- /* Add a uint16_t at the beginning containing the actual data offset. */
- if (prevwasmf) {
- if (datalen + 2 >= MINCLSIZE)
- MGETHDR(m, MT_DATA, 0);
- else
- m = m_get(M_DONTWAIT, MT_DATA);
- if (m == NULL) {
- ifp->if_ierrors++;
- goto out;
+ m = NULL;
+ top = NULL;
+ mp = &top;
+
+ while ((pd = TAILQ_FIRST(&sc->sc_dirtyq[pq])) != NULL) {
+ TAILQ_REMOVE(&sc->sc_dirtyq[pq], pd, pd_link);
+ sc->sc_dirtyq_count[pq]--;
+ datalen = letoh16(pd->pd_fragp->pf_size);
+ dataoff = letoh32(pd->pd_fragp->pf_addr) - pd->pd_dmaaddr;
+ morefrags = pd->pd_fragp->pf_flags & htole16(PF_FLAG_MF);
+
+ if (sc->sc_debug & SC_DEBUG_RXFRAG)
+ DPRINTF(("%s: rx frag: len %u memoff %u flags %x\n",
+ sc->sc_dev.dv_xname, datalen, dataoff,
+ pd->pd_fragp->pf_flags));
+
+ /* Add the (two+?) bytes for the header. */
+ if (datalen + dataoff > PGT_FRAG_SIZE) {
+ if (sc->sc_debug & SC_DEBUG_UNEXPECTED)
+ DPRINTF(("%s data rx too big: %u\n",
+ sc->sc_dev.dv_xname, datalen));
+ goto fail;
}
- //m->m_flags |= M_PROTO2;
- bcopy(pd->pd_mem, mtod(m, char *) + 2, datalen);
- m->m_len = datalen;
- } else {
- m = m_devget(pd->pd_mem, datalen,
- sizeof(struct ieee80211_frame) + 2, ifp, NULL);
- if (m != NULL)
- M_PREPEND(m, 2, M_DONTWAIT);
+
if (m == NULL) {
- ifp->if_ierrors++;
- goto out;
+ datalen += dataoff;
+ MGETHDR(m, M_DONTWAIT, MT_DATA);
+ if (m == NULL)
+ goto fail;
+ if (datalen + 2 >= MINCLSIZE) {
+ MCLGET(m, M_DONTWAIT);
+ if (!(m->m_flags & M_EXT)) {
+ m_free(m);
+ goto fail;
+ }
+ }
+
+ /*
+ * Add a uint16_t at the beginning containing the
+ * actual data offset.
+ */
+ *mtod(m, uint16_t *) = dataoff;
+ bcopy(pd->pd_mem, mtod(m, char *) + 2, datalen);
+ m->m_len = datalen + 2;
+ tlen += datalen + 2;
+ } else {
+ m = m_get(M_DONTWAIT, MT_DATA);
+ if (m == NULL)
+ goto fail;
+ if (datalen >= MINCLSIZE) {
+ MCLGET(m, M_DONTWAIT);
+ if (!(m->m_flags & M_EXT)) {
+ m_free(m);
+ goto fail;
+ }
+ }
+ bcopy(pd->pd_mem + dataoff, mtod(m, char *), datalen);
+ m->m_len = datalen;
+ tlen += datalen;
}
+
+
+ *mp = m;
+ mp = &m->m_next;
+
+ TAILQ_INSERT_TAIL(&sc->sc_freeq[pq], pd, pd_link);
+ sc->sc_freeq_count[pq]++;
+ pgt_reinit_rx_desc_frag(sc, pd);
+
+ if (!morefrags)
+ break;
}
- *mtod(m, uint16_t *) = dataoff;
- if (morefrags)
- m->m_flags |= M_PROTO1;
- else
- /*
- * Count non-fragmented packets and the last fragment
- * in fragmented packets.
- */
+
+ if (top) {
ifp->if_ipackets++;
- **last_nextpkt = m;
- *last_nextpkt = &m->m_nextpkt;
-out:
+ top->m_pkthdr.len = tlen;
+ top->m_pkthdr.rcvif = ifp;
+ }
+ return (top);
+
+fail:
TAILQ_INSERT_TAIL(&sc->sc_freeq[pq], pd, pd_link);
sc->sc_freeq_count[pq]++;
pgt_reinit_rx_desc_frag(sc, pd);
- return (morefrags);
+
+ ifp->if_ierrors++;
+ if (top)
+ m_freem(top);
+ return (NULL);
}
int
@@ -1831,13 +1639,14 @@ pgt_oid_get(struct pgt_softc *sc, enum pgt_oid oid,
pmd.pmd_recvbuf = arg;
pmd.pmd_len = arglen;
pmd.pmd_oid = oid;
+
error = pgt_mgmt_request(sc, &pmd);
if (error == 0)
error = pmd.pmd_error;
if (error != 0 && error != EPERM && sc->sc_debug & SC_DEBUG_UNEXPECTED)
- printf("%s: failure getting oid 0x%x: %d\n",
- sc->sc_dev.dv_xname,
- oid, error);
+ DPRINTF(("%s: failure getting oid %#x: %d\n",
+ sc->sc_dev.dv_xname, oid, error));
+
return (error);
}
@@ -1853,12 +1662,14 @@ pgt_oid_retrieve(struct pgt_softc *sc, enum pgt_oid oid,
pmd.pmd_recvbuf = arg;
pmd.pmd_len = arglen;
pmd.pmd_oid = oid;
+
error = pgt_mgmt_request(sc, &pmd);
if (error == 0)
error = pmd.pmd_error;
if (error != 0 && error != EPERM && sc->sc_debug & SC_DEBUG_UNEXPECTED)
- printf("%s: failure retrieving oid 0x%x: %d\n",
- sc->sc_dev.dv_xname, oid, error);
+ DPRINTF(("%s: failure retrieving oid %#x: %d\n",
+ sc->sc_dev.dv_xname, oid, error));
+
return (error);
}
@@ -1873,20 +1684,22 @@ pgt_oid_set(struct pgt_softc *sc, enum pgt_oid oid,
pmd.pmd_sendbuf = arg;
pmd.pmd_len = arglen;
pmd.pmd_oid = oid;
+
error = pgt_mgmt_request(sc, &pmd);
if (error == 0)
error = pmd.pmd_error;
if (error != 0 && error != EPERM && sc->sc_debug & SC_DEBUG_UNEXPECTED)
- printf("%s: failure setting oid 0x%x: %d\n",
- sc->sc_dev.dv_xname, oid, error);
+ DPRINTF(("%s: failure setting oid %#x: %d\n",
+ sc->sc_dev.dv_xname, oid, error));
+
return (error);
}
void
pgt_state_dump(struct pgt_softc *sc)
{
- printf("%s: state dump: control 0x%08x "
- "interrupt 0x%08x\n", sc->sc_dev.dv_xname,
+ printf("%s: state dump: control 0x%08x interrupt 0x%08x\n",
+ sc->sc_dev.dv_xname,
pgt_read_4(sc, PGT_REG_CTRL_STAT),
pgt_read_4(sc, PGT_REG_INT_STAT));
@@ -1952,26 +1765,12 @@ pgt_mgmt_request(struct pgt_softc *sc, struct pgt_mgmt_desc *pmd)
pmd->pmd_error = EINPROGRESS;
TAILQ_INSERT_TAIL(&sc->sc_mgmtinprog, pmd, pmd_link);
if (sc->sc_debug & SC_DEBUG_MGMT)
- printf("%s: queue: mgmt %p -> (op %u, "
- "oid 0x%x, len %u)\n", sc->sc_dev.dv_xname,
+ DPRINTF(("%s: queue: mgmt %p -> (op %u, "
+ "oid %#x, len %u)\n", sc->sc_dev.dv_xname,
pmd, pmf->pmf_operation,
- pmd->pmd_oid, pmd->pmd_len);
+ pmd->pmd_oid, pmd->pmd_len));
pgt_desc_transmit(sc, PGT_QUEUE_MGMT_TX, pd,
sizeof(*pmf) + pmd->pmd_len, 0);
- sc->sc_refcnt++;
-#ifdef DEVICE_POLLING
- /*
- * If we're polling, try 1/10th second initially at the smallest
- * interval we can sleep for.
- */
- for (i = 0; sc->sc_ic.ic_if.if_flags & IFF_POLLING && i < 100; i++) {
- pgt_update_intr(sc, NULL, 0);
- if (pmd->pmd_error != EINPROGRESS)
- goto usedpoll;
- if (tsleep(pmd, 0, "pgtmgp", 1) != EWOULDBLOCK)
- break;
- }
-#endif
/*
* Try for one second, triggering 10 times.
*
@@ -1994,16 +1793,13 @@ pgt_mgmt_request(struct pgt_softc *sc, struct pgt_mgmt_desc *pmd)
if (i != 9)
pgt_maybe_trigger(sc, PGT_QUEUE_MGMT_RX);
#ifdef PGT_BUGGY_INTERRUPT_RECOVERY
- pgt_update_intr(sc, NULL, 0);
+ pgt_update_intr(sc, 0);
#endif
} while (i++ < 10);
-#ifdef DEVICE_POLLING
-usedpoll:
-#endif
if (pmd->pmd_error == EINPROGRESS) {
printf("%s: timeout waiting for management "
- "packet response to 0x%x\n",
+ "packet response to %#x\n",
sc->sc_dev.dv_xname, pmd->pmd_oid);
TAILQ_REMOVE(&sc->sc_mgmtinprog, pmd, pmd_link);
if (sc->sc_debug & SC_DEBUG_UNEXPECTED)
@@ -2012,7 +1808,6 @@ usedpoll:
error = ETIMEDOUT;
} else
error = 0;
- sc->sc_refcnt--;
return (error);
}
@@ -2026,8 +1821,8 @@ pgt_desc_transmit(struct pgt_softc *sc, enum pgt_queue pq, struct pgt_desc *pd,
TAILQ_INSERT_TAIL(&sc->sc_dirtyq[pq], pd, pd_link);
sc->sc_dirtyq_count[pq]++;
if (sc->sc_debug & SC_DEBUG_QUEUES)
- printf("%s: queue: tx %u -> [%u]\n", sc->sc_dev.dv_xname,
- pd->pd_fragnum, pq);
+ DPRINTF(("%s: queue: tx %u -> [%u]\n", sc->sc_dev.dv_xname,
+ pd->pd_fragnum, pq));
bus_dmamap_sync(sc->sc_dmat, sc->sc_cbdmam, 0,
sc->sc_cbdmam->dm_mapsize,
BUS_DMASYNC_POSTREAD | BUS_DMASYNC_PREWRITE);
@@ -2053,8 +1848,8 @@ pgt_maybe_trigger(struct pgt_softc *sc, enum pgt_queue pq)
uint32_t reg;
if (sc->sc_debug & SC_DEBUG_TRIGGER)
- printf("%s: triggered by queue [%u]\n",
- sc->sc_dev.dv_xname, pq);
+ DPRINTF(("%s: triggered by queue [%u]\n",
+ sc->sc_dev.dv_xname, pq));
pgt_debug_events(sc, "trig");
if (sc->sc_flags & SC_POWERSAVE) {
/* Magic values ahoy? */
@@ -2066,9 +1861,9 @@ pgt_maybe_trigger(struct pgt_softc *sc, enum pgt_queue pq)
} while (tries-- != 0);
if (!(reg & PGT_CTRL_STAT_SLEEPMODE)) {
if (sc->sc_debug & SC_DEBUG_UNEXPECTED)
- printf("%s: timeout triggering from "
+ DPRINTF(("%s: timeout triggering from "
"sleep mode\n",
- sc->sc_dev.dv_xname);
+ sc->sc_dev.dv_xname));
pgt_async_reset(sc);
return;
}
@@ -2094,12 +1889,13 @@ pgt_ieee80211_node_alloc(struct ieee80211com *ic)
bzero(pin, sizeof *pin);
if (pin != NULL)
pin->pin_dot1x_auth = PIN_DOT1X_UNAUTHORIZED;
+
return (struct ieee80211_node *)pin;
}
void
pgt_ieee80211_newassoc(struct ieee80211com *ic, struct ieee80211_node *ni,
- int reallynew)
+ int reallynew)
{
ieee80211_ref_node(ni);
}
@@ -2115,7 +1911,7 @@ pgt_ieee80211_node_free(struct ieee80211com *ic, struct ieee80211_node *ni)
void
pgt_ieee80211_node_copy(struct ieee80211com *ic, struct ieee80211_node *dst,
- const struct ieee80211_node *src)
+ const struct ieee80211_node *src)
{
const struct pgt_ieee80211_node *psrc;
struct pgt_ieee80211_node *pdst;
@@ -2127,7 +1923,7 @@ pgt_ieee80211_node_copy(struct ieee80211com *ic, struct ieee80211_node *dst,
int
pgt_ieee80211_send_mgmt(struct ieee80211com *ic, struct ieee80211_node *ni,
- int type, int arg)
+ int type, int arg)
{
return (EOPNOTSUPP);
}
@@ -2137,8 +1933,7 @@ int
pgt_net_attach(struct pgt_softc *sc)
{
struct ieee80211com *ic = &sc->sc_ic;
- struct arpcom *ac = &ic->ic_ac;
- struct ifnet *ifp = &ac->ac_if;
+ struct ifnet *ifp = &ic->ic_if;
struct ieee80211_rateset *rs;
uint8_t rates[IEEE80211_RATE_MAXSIZE];
struct pgt_obj_buffer psbuffer;
@@ -2148,37 +1943,32 @@ pgt_net_attach(struct pgt_softc *sc)
int error;
psbuffer.pob_size = htole32(PGT_FRAG_SIZE * PGT_PSM_BUFFER_FRAME_COUNT);
- psbuffer.pob_addr = htole32((uint32_t)sc->sc_psmdmabusaddr);
- error = pgt_oid_set(sc, PGT_OID_PSM_BUFFER, &psbuffer,
- sizeof(psbuffer));
- if (error == 0)
- error = pgt_oid_get(sc, PGT_OID_PHY, &phymode, sizeof(phymode));
- if (error == 0)
- error = pgt_oid_get(sc, PGT_OID_MAC_ADDRESS, ac->ac_enaddr,
- sizeof(ac->ac_enaddr));
- if (error == 0)
- error = pgt_oid_get(sc, PGT_OID_COUNTRY, &country,
- sizeof(country));
+ psbuffer.pob_addr = htole32(sc->sc_psmdmam->dm_segs[0].ds_addr);
+ error = pgt_oid_set(sc, PGT_OID_PSM_BUFFER, &psbuffer, sizeof(country));
+ if (error)
+ return (error);
+ error = pgt_oid_get(sc, PGT_OID_PHY, &phymode, sizeof(phymode));
+ if (error)
+ return (error);
+ error = pgt_oid_get(sc, PGT_OID_MAC_ADDRESS, ic->ic_myaddr,
+ sizeof(ic->ic_myaddr));
+ if (error)
+ return (error);
+ error = pgt_oid_get(sc, PGT_OID_COUNTRY, &country, sizeof(country));
if (error)
return (error);
- //if_initname(ifp, device_get_name(sc->sc_dev),
- // device_get_unit(sc->sc_dev));
- ifp->if_flags = IFF_BROADCAST | IFF_MULTICAST | IFF_RUNNING;
- ifp->if_capabilities = IFCAP_VLAN_MTU;
-#ifdef DEVICE_POLLING
- ifp->if_capabilities |= IFCAP_POLLING;
-#endif
- //ifp->if_capenable = IFCAP_VLAN_MTU;
- ifp->if_start = pgt_start;
- ifp->if_ioctl = pgt_ioctl;
- ifp->if_watchdog = pgt_periodic;
+ ifp->if_softc = sc;
ifp->if_init = pgt_init;
+ ifp->if_ioctl = pgt_ioctl;
+ ifp->if_start = pgt_start;
+ ifp->if_watchdog = pgt_watchdog;
+ ifp->if_flags = IFF_SIMPLEX | IFF_BROADCAST | IFF_MULTICAST;
+ strlcpy(ifp->if_xname, sc->sc_dev.dv_xname, IFNAMSIZ);
+
IFQ_SET_MAXLEN(&ifp->if_snd, PGT_QUEUE_FULL_THRESHOLD);
- //ifp->if_snd.ifq_drv_maxlen = PGT_QUEUE_FULL_THRESHOLD;
IFQ_SET_READY(&ifp->if_snd);
- IEEE80211_ADDR_COPY(ic->ic_myaddr, ac->ac_enaddr);
j = sizeof(*freqs) + (IEEE80211_CHAN_MAX + 1) * sizeof(uint16_t);
freqs = malloc(j, M_DEVBUF, M_WAITOK);
error = pgt_oid_get(sc, PGT_OID_SUPPORTED_FREQUENCIES, freqs, j);
@@ -2236,7 +2026,7 @@ pgt_net_attach(struct pgt_softc *sc)
case 4:
case 11:
case 22:
- case 44: /* maybe */
+ case 44: /* maybe */
if (phymode & htole32(PGT_OID_PHY_2400MHZ)) {
rs = &ic->ic_sup_rates[IEEE80211_MODE_11B];
rs->rs_rates[rs->rs_nrates++] = rates[i];
@@ -2254,39 +2044,58 @@ pgt_net_attach(struct pgt_softc *sc)
rs->rs_rates[rs->rs_nrates++] = rates[i];
}
}
+
ic->ic_caps = IEEE80211_C_WEP | IEEE80211_C_IBSS | IEEE80211_C_PMGT |
IEEE80211_C_HOSTAP | IEEE80211_C_TXPMGT | IEEE80211_C_SHSLOT |
IEEE80211_C_SHPREAMBLE | IEEE80211_C_MONITOR;
- ic->ic_phytype = IEEE80211_T_OFDM; /* XXX not really used */
+
ic->ic_opmode = IEEE80211_M_STA;
ic->ic_state = IEEE80211_S_INIT;
- ic->ic_protmode = IEEE80211_PROT_NONE;
+
if_attach(ifp);
ieee80211_ifattach(ifp);
- /* Set up post-attach/pre-lateattach vector functions. */
- ic->ic_newstate = pgt_new_state;
+
+ /* Set up post-attach/pre-lateattach vector functions */
+ sc->sc_newstate = ic->ic_newstate;
+ ic->ic_newstate = pgt_newstate;
ic->ic_node_alloc = pgt_ieee80211_node_alloc;
ic->ic_newassoc = pgt_ieee80211_newassoc;
ic->ic_node_free = pgt_ieee80211_node_free;
ic->ic_node_copy = pgt_ieee80211_node_copy;
ic->ic_send_mgmt = pgt_ieee80211_send_mgmt;
+
/* let net80211 handle switching around the media + resetting */
- ieee80211_media_init(ifp, pgt_media_change, pgt_media_status);
- //bpfattach2(ifp, DLT_IEEE802_11_RADIO, sizeof(struct ieee80211_frame) +
- // sizeof(struct pgt_ieee80211_radiotap), &sc->sc_drvbpf);
- /* default to the first channel we know of */
- ic->ic_bss->ni_chan = ic->ic_ibss_chan;
- sc->sc_if_flags = ifp->if_flags;
+ ieee80211_media_init(ifp, pgt_media_change, ieee80211_media_status);
+
+#if NBPFILTER > 0
+ bpfattach(&sc->sc_drvbpf, ifp, DLT_IEEE802_11_RADIO,
+ sizeof(struct ieee80211_frame) + 64);
+
+ sc->sc_rxtap_len = sizeof(sc->sc_rxtapu);
+ sc->sc_rxtap.wr_ihdr.it_len = htole16(sc->sc_rxtap_len);
+ sc->sc_rxtap.wr_ihdr.it_present = htole32(PGT_RX_RADIOTAP_PRESENT);
+
+ sc->sc_txtap_len = sizeof(sc->sc_txtapu);
+ sc->sc_txtap.wt_ihdr.it_len = htole16(sc->sc_txtap_len);
+ sc->sc_txtap.wt_ihdr.it_present = htole32(PGT_TX_RADIOTAP_PRESENT);
+#endif
return (0);
}
-void
-pgt_net_detach(struct pgt_softc *sc)
+int
+pgt_media_change(struct ifnet *ifp)
{
- ieee80211_new_state(&sc->sc_ic, IEEE80211_S_INIT, -1);
- bpfdetach(&sc->sc_ic.ic_if);
- ieee80211_ifdetach(&sc->sc_ic.ic_if);
+ struct pgt_softc *sc = ifp->if_softc;
+ int error;
+
+ error = ieee80211_media_change(ifp);
+ if (error == ENETRESET) {
+ pgt_update_hw_from_sw(sc, 0, 0);
+ error = 0;
+ }
+
+ return (error);
}
void
@@ -2297,11 +2106,13 @@ pgt_start(struct ifnet *ifp)
sc = ifp->if_softc;
ic = &sc->sc_ic;
+
if (sc->sc_flags & (SC_DYING | SC_NEEDS_RESET) ||
!(ifp->if_flags & IFF_RUNNING) ||
ic->ic_state != IEEE80211_S_RUN) {
return;
}
+
pgt_start_body(sc, ic, ifp);
}
@@ -2317,33 +2128,27 @@ pgt_start_body(struct pgt_softc *sc, struct ieee80211com *ic, struct ifnet *ifp)
struct mbuf *m;
int error;
- if (!pgt_try_enter_data_critical(sc))
- return;
/*
* Management packets should probably be MLME frames
* (i.e. hostap "managed" mode); we don't touch the
* net80211 management queue.
*/
for (; sc->sc_dirtyq_count[PGT_QUEUE_DATA_LOW_TX] <
- //PGT_QUEUE_FULL_THRESHOLD && !IFQ_DRV_IS_EMPTY(&ifp->if_snd);) {
- PGT_QUEUE_FULL_THRESHOLD;) {
+ PGT_QUEUE_FULL_THRESHOLD && !IFQ_IS_EMPTY(&ifp->if_snd);) {
pd = TAILQ_FIRST(&sc->sc_freeq[PGT_QUEUE_DATA_LOW_TX]);
- //IFQ_DRV_DEQUEUE(&ifp->if_snd, m);
+ IFQ_DEQUEUE(&ifp->if_snd, m);
if (m == NULL)
break;
if (m->m_pkthdr.len <= PGT_FRAG_SIZE) {
error = pgt_load_tx_desc_frag(sc,
PGT_QUEUE_DATA_LOW_TX, pd);
if (error) {
- //IFQ_DRV_PREPEND(&ifp->if_snd, m);
+ IF_PREPEND(&ifp->if_snd, m);
break;
}
m_copydata(m, 0, m->m_pkthdr.len, pd->pd_mem);
pgt_desc_transmit(sc, PGT_QUEUE_DATA_LOW_TX,
pd, m->m_pkthdr.len, 0);
- //BPF_MTAP(ifp, m);
- ifp->if_opackets++;
- sc->sc_critical++;
} else if (m->m_pkthdr.len <= PGT_FRAG_SIZE * 2) {
struct pgt_desc *pd2;
@@ -2355,7 +2160,7 @@ pgt_start_body(struct pgt_softc *sc, struct ieee80211com *ic, struct ifnet *ifp)
*/
if (sc->sc_dirtyq_count[PGT_QUEUE_DATA_LOW_TX] + 2 >
PGT_QUEUE_FULL_THRESHOLD) {
- //IFQ_DRV_PREPEND(&ifp->if_snd, m);
+ IF_PREPEND(&ifp->if_snd, m);
break;
}
pd2 = TAILQ_NEXT(pd, pd_link);
@@ -2372,7 +2177,7 @@ pgt_start_body(struct pgt_softc *sc, struct ieee80211com *ic, struct ifnet *ifp)
}
}
if (error) {
- //IFQ_DRV_PREPEND(&ifp->if_snd, m);
+ IF_PREPEND(&ifp->if_snd, m);
break;
}
m_copydata(m, 0, PGT_FRAG_SIZE, pd->pd_mem);
@@ -2382,9 +2187,6 @@ pgt_start_body(struct pgt_softc *sc, struct ieee80211com *ic, struct ifnet *ifp)
m->m_pkthdr.len - PGT_FRAG_SIZE, pd2->pd_mem);
pgt_desc_transmit(sc, PGT_QUEUE_DATA_LOW_TX,
pd2, m->m_pkthdr.len - PGT_FRAG_SIZE, 0);
- //BPF_MTAP(ifp, m);
- ifp->if_opackets++;
- sc->sc_critical += 2;
} else {
ifp->if_oerrors++;
m_freem(m);
@@ -2392,9 +2194,13 @@ pgt_start_body(struct pgt_softc *sc, struct ieee80211com *ic, struct ifnet *ifp)
}
if (m != NULL) {
struct ieee80211_node *ni;
-
+#if NBPFILTER > 0
+ if (ifp->if_bpf != NULL)
+ bpf_mtap(ifp->if_bpf, m, BPF_DIRECTION_OUT);
+#endif
+ ifp->if_opackets++;
ifp->if_timer = 1;
- //getbinuptime(&sc->sc_data_tx_started);
+ sc->sc_txtimer = 5;
ni = ieee80211_find_txnode(&sc->sc_ic,
mtod(m, struct ether_header *)->ether_dhost);
if (ni != NULL) {
@@ -2402,7 +2208,9 @@ pgt_start_body(struct pgt_softc *sc, struct ieee80211com *ic, struct ifnet *ifp)
if (ni != ic->ic_bss)
ieee80211_release_node(&sc->sc_ic, ni);
}
+#if NBPFILTER > 0
if (sc->sc_drvbpf != NULL) {
+#if 0
struct pgt_ieee80211_radiotap pir;
struct ether_header eh;
@@ -2427,34 +2235,74 @@ pgt_start_body(struct pgt_softc *sc, struct ieee80211com *ic, struct ifnet *ifp)
m_adj(m, sizeof(eh));
m = pgt_ieee80211_encap(sc, &eh, m, NULL);
if (m != NULL) {
- //bpf_mtap2(sc->sc_drvbpf, &pir,
- // sizeof(pir), m);
+ bpf_mtap2(sc->sc_drvbpf, &pir,
+ sizeof(pir), m);
m_freem(m);
}
- } else {
- m_freem(m);
+#endif
+ struct mbuf mb;
+ struct ether_header eh;
+ struct pgt_tx_radiotap_hdr *tap = &sc->sc_txtap;
+
+ bcopy(mtod(m, struct ether_header *), &eh,
+ sizeof(eh));
+ m_adj(m, sizeof(eh));
+ m = pgt_ieee80211_encap(sc, &eh, m, NULL);
+
+ tap->wt_flags = 0;
+ //tap->wt_rate = rate;
+ tap->wt_rate = 0;
+ tap->wt_chan_freq =
+ htole16(ic->ic_bss->ni_chan->ic_freq);
+ tap->wt_chan_flags =
+ htole16(ic->ic_bss->ni_chan->ic_flags);
+
+ if (m != NULL) {
+ M_DUP_PKTHDR(&mb, m);
+ mb.m_data = (caddr_t)tap;
+ mb.m_len = sc->sc_txtap_len;
+ mb.m_next = m;
+ mb.m_pkthdr.len += mb.m_len;
+
+ bpf_mtap(sc->sc_drvbpf, &mb,
+ BPF_DIRECTION_OUT);
+ }
}
+#endif
+ if (m != NULL)
+ m_freem(m);
}
}
- //pgt_try_exit_data_critical(sc);
}
int
pgt_ioctl(struct ifnet *ifp, u_long cmd, caddr_t req)
{
+#if 0
+ struct ifprismoidreq *preq;
+ struct ieee80211req *ireq;
+#endif
struct pgt_softc *sc = ifp->if_softc;
+ struct ifaddr *ifa;
struct ifreq *ifr;
struct wi_req *wreq;
- //struct ifprismoidreq *preq;
- //struct ieee80211req *ireq;
- int error, oldflags;
+ struct ieee80211_nodereq_all *na;
+ struct ieee80211com *ic;
+ struct pgt_obj_bsslist *pob;
+ struct wi_scan_p2_hdr *p2hdr;
+ struct wi_scan_res *res;
+ uint32_t noise;
+ int maxscan, i, j, s, error = 0;
+ ic = &sc->sc_ic;
ifr = (struct ifreq *)req;
+
+ s = splnet();
switch (cmd) {
#if 0
case SIOCGPRISMOID:
case SIOCSPRISMOID:
- //error = suser(curthread);
+ error = suser(curthread);
if (error)
return (error);
preq = (struct ifprismoidreq *)req;
@@ -2470,66 +2318,133 @@ pgt_ioctl(struct ifnet *ifp, u_long cmd, caddr_t req)
pgt_exit_critical(sc);
break;
#endif
- case SIOCGWAVELAN:
- case SIOCSWAVELAN:
+ case SIOCS80211SCAN:
+ /*
+ * This chip scans always as soon as it gets initialized.
+ */
+
+ /*
+ * Give us a bit time to scan in case we were not
+ * initialized before and let the userland process wait.
+ */
+ tsleep(&sc->sc_flags, 0, "pgtsca", hz * SCAN_TIMEOUT);
+
+ break;
+ case SIOCG80211ALLNODES: {
+ struct ieee80211_nodereq *nr = NULL;
+ na = (struct ieee80211_nodereq_all *)req;
wreq = malloc(sizeof(*wreq), M_DEVBUF, M_WAITOK);
- error = copyin(ifr->ifr_data, wreq, sizeof(*wreq));
+ bzero(wreq, sizeof(*wreq));
+
+ maxscan = PGT_OBJ_BSSLIST_NBSS;
+ pob = malloc(sizeof(*pob) +
+ sizeof(struct pgt_obj_bss) * maxscan, M_DEVBUF, M_WAITOK);
+ error = pgt_oid_get(sc, PGT_OID_NOISE_FLOOR, &noise,
+ sizeof(noise));
+
if (error == 0) {
- if (cmd == SIOCGWAVELAN) {
- error = pgt_wavelan_get(sc, wreq);
- if (error == 0)
- error = copyout(wreq, ifr->ifr_data,
- sizeof(*wreq));
- } else {
- error = pgt_wavelan_set(sc, wreq);
+ noise = letoh32(noise);
+ error = pgt_oid_get(sc, PGT_OID_BSS_LIST, pob,
+ sizeof(*pob) +
+ sizeof(struct pgt_obj_bss) * maxscan);
+ }
+
+ if (error == 0) {
+ maxscan = min(PGT_OBJ_BSSLIST_NBSS,
+ letoh32(pob->pob_count));
+ maxscan = min(maxscan,
+ (sizeof(wreq->wi_val) - sizeof(*p2hdr)) /
+ WI_PRISM2_RES_SIZE);
+ p2hdr = (struct wi_scan_p2_hdr *)&wreq->wi_val;
+ p2hdr->wi_rsvd = 0;
+ p2hdr->wi_reason = 1;
+ wreq->wi_len = (maxscan * WI_PRISM2_RES_SIZE) / 2 +
+ sizeof(*p2hdr) / 2;
+ wreq->wi_type = WI_RID_SCAN_RES;
+ }
+
+ for (na->na_nodes = j = i = 0; i < maxscan &&
+ (na->na_size >= j + sizeof(struct ieee80211_nodereq));
+ i++) {
+ /* allocate node space */
+ if (nr == NULL)
+ nr = malloc(sizeof(*nr), M_DEVBUF, M_WAITOK);
+
+ /* get next BSS scan result */
+ res = (struct wi_scan_res *)
+ ((char *)&wreq->wi_val + sizeof(*p2hdr) +
+ i * WI_PRISM2_RES_SIZE);
+ pgt_obj_bss2scanres(sc, &pob->pob_bsslist[i],
+ res, noise);
+
+ /* copy it to node structure for ifconfig to read */
+ bzero(nr, sizeof(*nr));
+ IEEE80211_ADDR_COPY(nr->nr_macaddr, res->wi_bssid);
+ IEEE80211_ADDR_COPY(nr->nr_bssid, res->wi_bssid);
+ nr->nr_channel = letoh16(res->wi_chan);
+ nr->nr_chan_flags = IEEE80211_CHAN_B;
+ nr->nr_rssi = letoh16(res->wi_signal);
+ nr->nr_max_rssi = 0; /* XXX */
+ nr->nr_nwid_len = letoh16(res->wi_ssid_len);
+ bcopy(res->wi_ssid, nr->nr_nwid, nr->nr_nwid_len);
+ nr->nr_intval = letoh16(res->wi_interval);
+ nr->nr_capinfo = letoh16(res->wi_capinfo);
+ nr->nr_txrate = res->wi_rate == WI_WAVELAN_RES_1M ? 2 :
+ (res->wi_rate == WI_WAVELAN_RES_2M ? 4 :
+ (res->wi_rate == WI_WAVELAN_RES_5M ? 11 :
+ (res->wi_rate == WI_WAVELAN_RES_11M ? 22 : 0)));
+ nr->nr_nrates = 0;
+ while (res->wi_srates[nr->nr_nrates] != 0) {
+ nr->nr_rates[nr->nr_nrates] =
+ res->wi_srates[nr->nr_nrates] &
+ WI_VAR_SRATES_MASK;
+ nr->nr_nrates++;
}
+ nr->nr_flags = 0;
+ if (bcmp(nr->nr_macaddr, nr->nr_bssid,
+ IEEE80211_ADDR_LEN) == 0)
+ nr->nr_flags |= IEEE80211_NODEREQ_AP;
+ error = copyout(nr, (caddr_t)na->na_node + j,
+ sizeof(struct ieee80211_nodereq));
+ if (error)
+ break;
+
+ /* point to next node entry */
+ j += sizeof(struct ieee80211_nodereq);
+ na->na_nodes++;
}
- free(wreq, M_DEVBUF);
- if (error == EOPNOTSUPP)
- goto notours;
+ if (nr)
+ free(nr, M_DEVBUF);
+ free(pob, M_DEVBUF);
break;
+ }
+ case SIOCSIFADDR: ifa = (struct ifaddr *)req;
+ ifp->if_flags |= IFF_UP;
+#ifdef INET
+ if (ifa->ifa_addr->sa_family == AF_INET)
+ arp_ifinit(&sc->sc_ic.ic_ac, ifa);
+#endif
+ /* FALLTHROUGH */
case SIOCSIFFLAGS:
- error = 0;
- oldflags = sc->sc_if_flags;
- sc->sc_if_flags = ifp->if_flags;
- if ((oldflags & (IFF_PROMISC | IFF_UP)) !=
- (ifp->if_flags & (IFF_PROMISC | IFF_UP))) {
- if (!(oldflags & IFF_UP) && ifp->if_flags & IFF_UP) {
- ieee80211_new_state(&sc->sc_ic,
- IEEE80211_S_SCAN, -1);
+ if (ifp->if_flags & IFF_UP) {
+ if ((ifp->if_flags & IFF_RUNNING) == 0) {
+ pgt_init(ifp);
error = ENETRESET;
- } else if (oldflags & IFF_UP &&
- !(ifp->if_flags & IFF_UP)) {
- ieee80211_new_state(&sc->sc_ic,
- IEEE80211_S_INIT, -1);
+ }
+ } else {
+ if (ifp->if_flags & IFF_RUNNING) {
+ pgt_disable(sc, SC_NEEDS_RESET);
error = ENETRESET;
}
}
break;
case SIOCSIFMTU:
if (ifr->ifr_mtu > PGT_FRAG_SIZE) {
- uprintf("%s: bad MTU (values > %u non-functional)\n",
- ifp->if_xname, PGT_FRAG_SIZE);
error = EINVAL;
- } else {
- ifp->if_mtu = ifr->ifr_mtu;
- error = 0;
- }
- break;
-#ifdef DEVICE_POLLING
- case SIOCSIFCAP:
- if (!(ifp->if_capabilities & IFF_RUNNING)) {
- error = EIO;
- } else {
- if ((ifr->ifr_reqcap & IFCAP_POLLING) !=
- (ifp->if_capenable & IFCAP_POLLING))
- ifp->if_capenable ^= IFCAP_POLLING;
- error = 0;
+ break;
}
- break;
-#endif
+ /* FALLTHROUGH */
default:
-notours:
/*
* XXX net80211 does not prevent modification of the
* ieee80211com while it fondles it.
@@ -2537,16 +2452,19 @@ notours:
error = ieee80211_ioctl(ifp, cmd, req);
break;
}
+
if (error == ENETRESET) {
pgt_update_hw_from_sw(sc, 0, 0);
error = 0;
}
+ splx(s);
+
return (error);
}
void
pgt_obj_bss2scanres(struct pgt_softc *sc, struct pgt_obj_bss *pob,
- struct wi_scan_res *scanres, uint32_t noise)
+ struct wi_scan_res *scanres, uint32_t noise)
{
struct ieee80211_rateset *rs;
struct wi_scan_res ap;
@@ -2575,6 +2493,7 @@ pgt_obj_bss2scanres(struct pgt_softc *sc, struct pgt_obj_bss *pob,
memcpy(scanres, &ap, WI_PRISM2_RES_SIZE);
}
+#if 0
int
pgt_node_set_authorization(struct pgt_softc *sc,
struct pgt_ieee80211_node *pin, enum pin_dot1x_authorization newstate)
@@ -2583,20 +2502,24 @@ pgt_node_set_authorization(struct pgt_softc *sc,
if (pin->pin_dot1x_auth == newstate)
return (0);
- IEEE80211_DPRINTF(("%s: %02x:%02x:%02x:%02x:%02x:%02x "
+
+ DPRINTF(("%s: %02x:%02x:%02x:%02x:%02x:%02x "
"changing authorization to %d\n", __func__,
pin->pin_node.ni_macaddr[0], pin->pin_node.ni_macaddr[1],
pin->pin_node.ni_macaddr[2], pin->pin_node.ni_macaddr[3],
pin->pin_node.ni_macaddr[4], pin->pin_node.ni_macaddr[5],
newstate));
+
error = pgt_oid_set(sc,
newstate == PIN_DOT1X_AUTHORIZED ?
PGT_OID_EAPAUTHSTA : PGT_OID_EAPUNAUTHSTA,
pin->pin_node.ni_macaddr, sizeof(pin->pin_node.ni_macaddr));
if (error == 0)
pin->pin_dot1x_auth = pin->pin_dot1x_auth_desired = newstate;
+
return (error);
}
+#endif
#if 0
int
@@ -2715,7 +2638,7 @@ pgt_do_mlme_adhoc(struct pgt_softc *sc, struct ieee80211req_mlme *imlme)
int
pgt_80211_set(struct pgt_softc *sc, struct ieee80211req *ireq)
{
- //struct ieee80211req_mlme mlme;
+ struct ieee80211req_mlme mlme;
struct ieee80211com *ic;
int error;
@@ -2797,13 +2720,13 @@ pgt_80211_set(struct pgt_softc *sc, struct ieee80211req *ireq)
pgt_enter_critical(sc);
switch (ic->ic_opmode) {
case IEEE80211_M_STA:
- //error = pgt_do_mlme_sta(sc, &mlme);
+ error = pgt_do_mlme_sta(sc, &mlme);
break;
case IEEE80211_M_HOSTAP:
- //error = pgt_do_mlme_hostap(sc, &mlme);
+ error = pgt_do_mlme_hostap(sc, &mlme);
break;
case IEEE80211_M_IBSS:
- //error = pgt_do_mlme_adhoc(sc, &mlme);
+ error = pgt_do_mlme_adhoc(sc, &mlme);
break;
default:
error = EINVAL;
@@ -2821,94 +2744,6 @@ pgt_80211_set(struct pgt_softc *sc, struct ieee80211req *ireq)
}
#endif
-int
-pgt_wavelan_get(struct pgt_softc *sc, struct wi_req *wreq)
-{
- struct ieee80211com *ic;
- struct pgt_obj_bsslist *pob;
- struct wi_scan_p2_hdr *p2hdr;
- struct wi_scan_res *scan;
- uint32_t noise;
- unsigned int maxscan, i;
- int error;
-
- ic = &sc->sc_ic;
- switch (wreq->wi_type) {
- case WI_RID_COMMS_QUALITY:
- wreq->wi_val[0] = 0; /* don't know correction factor */
- wreq->wi_val[1] = htole16(ic->ic_node_getrssi(ic, ic->ic_bss));
- wreq->wi_val[2] = htole16(sc->sc_noise);
- wreq->wi_len = 4;
- error = 0;
- break;
- case WI_RID_SCAN_RES:
- maxscan = PGT_OBJ_BSSLIST_NBSS;
- pob = malloc(sizeof(*pob) +
- sizeof(struct pgt_obj_bss) * maxscan, M_DEVBUF, M_WAITOK);
- pgt_enter_critical(sc);
- error = pgt_oid_get(sc, PGT_OID_NOISE_FLOOR, &noise,
- sizeof(noise));
- if (error == 0) {
- noise = letoh32(noise);
- error = pgt_oid_get(sc, PGT_OID_BSS_LIST, pob,
- sizeof(*pob) +
- sizeof(struct pgt_obj_bss) * maxscan);
- }
- if (error == 0) {
- maxscan = min(PGT_OBJ_BSSLIST_NBSS,
- letoh32(pob->pob_count));
- maxscan = min(maxscan,
- (sizeof(wreq->wi_val) - sizeof(*p2hdr)) /
- WI_PRISM2_RES_SIZE);
- p2hdr = (struct wi_scan_p2_hdr *)&wreq->wi_val;
- p2hdr->wi_rsvd = 0;
- p2hdr->wi_reason = 1; /* what should it be? */
- for (i = 0; i < maxscan; i++) {
- scan = (struct wi_scan_res *)
- ((char *)&wreq->wi_val + sizeof(*p2hdr) +
- i * WI_PRISM2_RES_SIZE);
- pgt_obj_bss2scanres(sc, &pob->pob_bsslist[i],
- scan, noise);
- }
- wreq->wi_len = (maxscan * WI_PRISM2_RES_SIZE) / 2 +
- sizeof(*p2hdr) / 2;
- }
- pgt_exit_critical(sc);
- free(pob, M_DEVBUF);
- break;
- default:
- error = EOPNOTSUPP;
- break;
- }
- return (error);
-}
-
-int
-pgt_wavelan_set(struct pgt_softc *sc, struct wi_req *wreq)
-{
- int error;
-
- /*
- * If we wanted to, we could support the "partial reset" interface
- * here, but the Wavelan interface should really not need to be used.
- */
- switch (wreq->wi_type) {
- case WI_RID_SCAN_REQ:
- case WI_RID_SCAN_APS:
- /* We're always scanning. */
- error = 0;
- break;
- case WI_RID_CHANNEL_LIST:
- /* The user can just use net80211's interface. */
- error = EPERM;
- break;
- default:
- error = EOPNOTSUPP;
- break;
- }
- return (error);
-}
-
void
node_mark_active_ap(void *arg, struct ieee80211_node *ni)
{
@@ -2935,7 +2770,7 @@ node_mark_active_adhoc(void *arg, struct ieee80211_node *ni)
}
void
-pgt_periodic(struct ifnet *ifp)
+pgt_watchdog(struct ifnet *ifp)
{
struct pgt_softc *sc;
@@ -2946,17 +2781,15 @@ pgt_periodic(struct ifnet *ifp)
* output device queue).
*/
if (sc->sc_dirtyq_count[PGT_QUEUE_DATA_LOW_TX] != 0) {
- struct bintime txtime;
int count;
ifp->if_timer = 1;
- //getbinuptime(&txtime);
- bintime_sub(&txtime, &sc->sc_data_tx_started);
- if (txtime.sec >= 1) {
+ if (sc->sc_txtimer && --sc->sc_txtimer == 0) {
count = pgt_drain_tx_queue(sc, PGT_QUEUE_DATA_LOW_TX);
- if (sc->sc_flags & SC_DEBUG_UNEXPECTED)
- printf("%s: timed out %d data transmissions\n",
- sc->sc_dev.dv_xname, count);
+ if (sc->sc_debug & SC_DEBUG_UNEXPECTED)
+ DPRINTF(("%s: timed out %d data "
+ "transmissions\n",
+ sc->sc_dev.dv_xname, count));
}
}
if (sc->sc_flags & (SC_DYING | SC_NEEDS_RESET))
@@ -2969,10 +2802,8 @@ pgt_periodic(struct ifnet *ifp)
if (ifp->if_flags & IFF_RUNNING &&
sc->sc_ic.ic_state != IEEE80211_S_INIT &&
sc->sc_ic.ic_opmode != IEEE80211_M_MONITOR)
-#ifdef DEVICE_POLLING
- if (!(ifp->if_flags & IFF_POLLING))
-#endif
- pgt_async_update(sc);
+ pgt_async_update(sc);
+
/*
* As a firmware-based HostAP, we should not time out
* nodes inside the driver additionally to the timeout
@@ -3003,11 +2834,18 @@ pgt_init(struct ifnet *ifp)
struct ieee80211com *ic;
ic = &sc->sc_ic;
+
if (!(sc->sc_flags & (SC_DYING | SC_UNINITIALIZED)))
pgt_update_hw_from_sw(sc,
ic->ic_state != IEEE80211_S_INIT,
ic->ic_opmode != IEEE80211_M_MONITOR);
+ ifp->if_flags |= IFF_RUNNING;
+ ifp->if_flags &= ~IFF_OACTIVE;
+
+ /* Begin background scanning */
+ ieee80211_new_state(&sc->sc_ic, IEEE80211_S_SCAN, -1);
+
return (0);
}
@@ -3028,17 +2866,19 @@ pgt_update_hw_from_sw(struct pgt_softc *sc, int keepassoc, int keepnodes)
struct pgt_obj_key keyobj;
struct pgt_obj_ssid essid;
uint8_t availrates[IEEE80211_RATE_MAXSIZE + 1];
- uint32_t mode, bsstype, config, profile, channel, slot, preamble,
- wep, exunencrypted, wepkey, dot1x, auth, mlme;
+ uint32_t mode, bsstype, config, profile, channel, slot, preamble;
+ uint32_t wep, exunencrypted, wepkey, dot1x, auth, mlme;
unsigned int i;
- int success, shouldbeup;
+ int success, shouldbeup, s;
config = PGT_CONFIG_MANUAL_RUN | PGT_CONFIG_RX_ANNEX;
+
/*
* Promiscuous mode is currently a no-op since packets transmitted,
* while in promiscuous mode, don't ever seem to go anywhere.
*/
shouldbeup = ifp->if_flags & IFF_RUNNING && ifp->if_flags & IFF_UP;
+
if (shouldbeup) {
switch (ic->ic_opmode) {
case IEEE80211_M_STA:
@@ -3091,6 +2931,7 @@ badopmode:
mode = PGT_MODE_CLIENT;
bsstype = PGT_BSS_TYPE_NONE;
}
+
switch (ic->ic_curmode) {
case IEEE80211_MODE_11A:
profile = PGT_PROFILE_A_ONLY;
@@ -3105,8 +2946,9 @@ badopmode:
preamble = PGT_OID_PREAMBLE_MODE_SHORT;
break;
case IEEE80211_MODE_FH:
- case IEEE80211_MODE_TURBO:
- /* not handled */
+ /* FALLTHROUGH */
+ case IEEE80211_MODE_TURBO: /* not handled */
+ /* FALLTHROUGH */
case IEEE80211_MODE_AUTO:
profile = PGT_PROFILE_MIXED_G_WIFI;
preamble = PGT_OID_PREAMBLE_MODE_DYNAMIC;
@@ -3114,6 +2956,7 @@ badopmode:
default:
panic("unknown mode %d\n", ic->ic_curmode);
}
+
switch (sc->sc_80211_ioc_auth) {
case IEEE80211_AUTH_NONE:
auth = PGT_AUTH_MODE_NONE;
@@ -3125,21 +2968,15 @@ badopmode:
auth = PGT_AUTH_MODE_SHARED;
break;
}
- switch (sc->sc_80211_ioc_wep) {
- case IEEE80211_WEP_OFF:
- wep = 0;
- exunencrypted = 0;
- break;
- case IEEE80211_WEP_MIXED:
- wep = 1;
- exunencrypted = 0;
- break;
- case IEEE80211_WEP_ON:
- default:
+
+ if (sc->sc_ic.ic_flags & IEEE80211_F_WEPON) {
wep = 1;
exunencrypted = 1;
- break;
+ } else {
+ wep = 0;
+ exunencrypted = 0;
}
+
mlme = htole32(PGT_MLME_AUTO_LEVEL_AUTO);
wep = htole32(wep);
exunencrypted = htole32(exunencrypted);
@@ -3148,46 +2985,50 @@ badopmode:
bsstype = htole32(bsstype);
config = htole32(config);
mode = htole32(mode);
+
if (!wep || !sc->sc_dot1x)
dot1x = PGT_DOT1X_AUTH_NONE;
dot1x = htole32(dot1x);
auth = htole32(auth);
+
if (ic->ic_flags & IEEE80211_F_SHSLOT)
slot = htole32(PGT_OID_SLOT_MODE_SHORT);
else
slot = htole32(PGT_OID_SLOT_MODE_DYNAMIC);
+
if (ic->ic_des_chan == IEEE80211_CHAN_ANYC) {
if (keepassoc)
channel = htole32(ieee80211_chan2ieee(ic,
ic->ic_bss->ni_chan));
else
channel = 0;
- } else {
- channel = htole32(ieee80211_chan2ieee(ic, ic->ic_des_chan));
- }
+ } else
+ channel = htole32(ieee80211_chan2ieee(ic, ic->ic_bss->ni_chan));
+
for (i = 0; i < ic->ic_sup_rates[ic->ic_curmode].rs_nrates; i++)
availrates[i] = ic->ic_sup_rates[ic->ic_curmode].rs_rates[i];
availrates[i++] = 0;
essid.pos_length = min(ic->ic_des_esslen, sizeof(essid.pos_ssid));
memcpy(&essid.pos_ssid, ic->ic_des_essid, essid.pos_length);
- pgt_enter_critical(sc);
+
+ s = splnet();
for (success = 0; success == 0; success = 1) {
-#define SETOID(oid, var, size) { \
- if (pgt_oid_set(sc, oid, var, size) != 0) \
- break; \
-}
SETOID(PGT_OID_PROFILE, &profile, sizeof(profile));
SETOID(PGT_OID_CONFIG, &config, sizeof(config));
SETOID(PGT_OID_MLME_AUTO_LEVEL, &mlme, sizeof(mlme));
+
if (!IEEE80211_ADDR_EQ(ic->ic_myaddr, ac->ac_enaddr)) {
SETOID(PGT_OID_MAC_ADDRESS, ac->ac_enaddr,
sizeof(ac->ac_enaddr));
IEEE80211_ADDR_COPY(ic->ic_myaddr, ac->ac_enaddr);
}
+
SETOID(PGT_OID_MODE, &mode, sizeof(mode));
SETOID(PGT_OID_BSS_TYPE, &bsstype, sizeof(bsstype));
- if (channel != 0)
+
+ if (channel != 0 && channel != htole32(IEEE80211_CHAN_ANY))
SETOID(PGT_OID_CHANNEL, &channel, sizeof(channel));
+
if (ic->ic_flags & IEEE80211_F_DESBSSID) {
SETOID(PGT_OID_BSSID, ic->ic_des_bssid,
sizeof(ic->ic_des_bssid));
@@ -3195,9 +3036,12 @@ badopmode:
SETOID(PGT_OID_BSSID, ic->ic_bss->ni_bssid,
sizeof(ic->ic_bss->ni_bssid));
}
+
SETOID(PGT_OID_SSID, &essid, sizeof(essid));
+
if (ic->ic_des_esslen > 0)
SETOID(PGT_OID_SSID_OVERRIDE, &essid, sizeof(essid));
+
SETOID(PGT_OID_RATES, &availrates, i);
SETOID(PGT_OID_EXTENDED_RATES, &availrates, i);
SETOID(PGT_OID_PREAMBLE_MODE, &preamble, sizeof(preamble));
@@ -3207,8 +3051,12 @@ badopmode:
sizeof(exunencrypted));
SETOID(PGT_OID_DOT1X, &dot1x, sizeof(dot1x));
SETOID(PGT_OID_PRIVACY_INVOKED, &wep, sizeof(wep));
+ /*
+ * Setting WEP key(s)
+ */
if (letoh32(wep) != 0) {
keyobj.pok_type = PGT_OBJ_KEY_TYPE_WEP;
+ /* key 1 */
keyobj.pok_length = min(sizeof(keyobj.pok_key),
IEEE80211_KEYBUF_SIZE);
keyobj.pok_length = min(keyobj.pok_length,
@@ -3216,6 +3064,7 @@ badopmode:
bcopy(ic->ic_nw_keys[0].wk_key, keyobj.pok_key,
keyobj.pok_length);
SETOID(PGT_OID_DEFAULT_KEY0, &keyobj, sizeof(keyobj));
+ /* key 2 */
keyobj.pok_length = min(sizeof(keyobj.pok_key),
IEEE80211_KEYBUF_SIZE);
keyobj.pok_length = min(keyobj.pok_length,
@@ -3223,6 +3072,7 @@ badopmode:
bcopy(ic->ic_nw_keys[1].wk_key, keyobj.pok_key,
keyobj.pok_length);
SETOID(PGT_OID_DEFAULT_KEY1, &keyobj, sizeof(keyobj));
+ /* key 3 */
keyobj.pok_length = min(sizeof(keyobj.pok_key),
IEEE80211_KEYBUF_SIZE);
keyobj.pok_length = min(keyobj.pok_length,
@@ -3230,6 +3080,7 @@ badopmode:
bcopy(ic->ic_nw_keys[2].wk_key, keyobj.pok_key,
keyobj.pok_length);
SETOID(PGT_OID_DEFAULT_KEY2, &keyobj, sizeof(keyobj));
+ /* key 4 */
keyobj.pok_length = min(sizeof(keyobj.pok_key),
IEEE80211_KEYBUF_SIZE);
keyobj.pok_length = min(keyobj.pok_length,
@@ -3237,14 +3088,15 @@ badopmode:
bcopy(ic->ic_nw_keys[3].wk_key, keyobj.pok_key,
keyobj.pok_length);
SETOID(PGT_OID_DEFAULT_KEY3, &keyobj, sizeof(keyobj));
+
wepkey = htole32(ic->ic_wep_txkey);
SETOID(PGT_OID_DEFAULT_KEYNUM, &wepkey, sizeof(wepkey));
}
/* set mode again to commit */
SETOID(PGT_OID_MODE, &mode, sizeof(mode));
-#undef SETOID
}
- pgt_exit_critical(sc);
+ splx(s);
+
if (success) {
if (shouldbeup && keepnodes)
sc->sc_flags |= SC_NOFREE_ALLNODES;
@@ -3262,31 +3114,34 @@ badopmode:
* After doing a soft-reinitialization, we will restore settings from
* our pgt_ieee80211_nodes. As we also lock the node list with our
* softc mutex, unless we were to drop that the node list will remain
- * valid (see pgt_periodic()).
+ * valid (see pgt_watchdog()).
*/
void
pgt_update_hw_from_nodes(struct pgt_softc *sc)
{
struct pgt_ieee80211_node *pin;
- //struct ieee80211_node *ni;
+#if 0
+ struct ieee80211_node *ni;
+#endif
struct pgt_ieee80211_node **addresses;
size_t i, n;
+ int s;
n = 0;
- /*
+#if 0
TAILQ_FOREACH(ni, &sc->sc_ic.ic_node, ni_list) {
pin = (struct pgt_ieee80211_node *)ni;
if (pin->pin_dot1x_auth != pin->pin_dot1x_auth_desired)
n++;
}
- */
+#endif
if (n == 0)
return;
addresses = malloc(sizeof(*addresses) * n, M_DEVBUF, M_NOWAIT);
if (addresses == NULL)
return;
n = 0;
- /*
+#if 0
TAILQ_FOREACH(ni, &sc->sc_ic.ic_node, ni_list) {
pin = (struct pgt_ieee80211_node *)ni;
if (pin->pin_dot1x_auth != pin->pin_dot1x_auth_desired) {
@@ -3294,8 +3149,8 @@ pgt_update_hw_from_nodes(struct pgt_softc *sc)
ieee80211_ref_node(&pin->pin_node);
}
}
- */
- pgt_enter_critical(sc);
+#endif
+ s = splnet();
for (i = 0; i < n; i++) {
pin = addresses[i];
if (pgt_oid_set(sc,
@@ -3304,7 +3159,7 @@ pgt_update_hw_from_nodes(struct pgt_softc *sc)
pin->pin_node.ni_macaddr, sizeof(pin->pin_node.ni_macaddr))
== 0) {
pin->pin_dot1x_auth = pin->pin_dot1x_auth_desired;
- IEEE80211_DPRINTF(("%s: %02x:%02x:%02x:%02x:%02x:%02x "
+ DPRINTF(("%s: %02x:%02x:%02x:%02x:%02x:%02x "
"reauthorized to %d\n", __func__,
pin->pin_node.ni_macaddr[0],
pin->pin_node.ni_macaddr[1],
@@ -3316,7 +3171,7 @@ pgt_update_hw_from_nodes(struct pgt_softc *sc)
}
ieee80211_release_node(&sc->sc_ic, &pin->pin_node);
}
- pgt_exit_critical(sc);
+ splx(s);
free(addresses, M_DEVBUF);
}
@@ -3365,7 +3220,7 @@ pgt_update_sw_from_hw(struct pgt_softc *sc, struct pgt_async_trap *pa,
struct pgt_obj_ssid ssid;
struct pgt_obj_bss bss;
uint32_t channel, noise, ls;
- int error;
+ int error, s;
if (pa != NULL) {
#if 0
@@ -3377,16 +3232,16 @@ pgt_update_sw_from_hw(struct pgt_softc *sc, struct pgt_async_trap *pa,
oid = *mtod(args, uint32_t *);
m_adj(args, sizeof(uint32_t));
if (sc->sc_debug & SC_DEBUG_TRAP)
- printf("%s: trap: oid 0x%x len %u\n",
- sc->sc_dev.dv_xname, oid, args->m_len);
+ DPRINTF(("%s: trap: oid %#x len %u\n",
+ sc->sc_dev.dv_xname, oid, args->m_len));
switch (oid) {
case PGT_OID_LINK_STATE:
if (args->m_len < sizeof(uint32_t))
break;
ls = letoh32(*mtod(args, uint32_t *));
if (sc->sc_debug & (SC_DEBUG_TRAP | SC_DEBUG_LINK))
- printf("%s: link: %u\n",
- sc->sc_dev.dv_xname, ls);
+ DPRINTF(("%s: link: %u\n",
+ sc->sc_dev.dv_xname, ls));
if (ls)
ieee80211_new_state(ic, IEEE80211_S_RUN, -1);
else
@@ -3400,7 +3255,7 @@ pgt_update_sw_from_hw(struct pgt_softc *sc, struct pgt_async_trap *pa,
break;
mlme = mtod(args, struct pgt_obj_mlme *);
if (sc->sc_debug & SC_DEBUG_TRAP)
- printf("%s: mlme: address "
+ DPRINTF(("%s: mlme: address "
"%02x:%02x:%02x:%02x:%02x:%02x "
"id 0x%02x state 0x%02x code 0x%02x\n",
sc->sc_dev.dv_xname,
@@ -3409,7 +3264,7 @@ pgt_update_sw_from_hw(struct pgt_softc *sc, struct pgt_async_trap *pa,
mlme->pom_address[4], mlme->pom_address[5],
letoh16(mlme->pom_id),
letoh16(mlme->pom_state),
- letoh16(mlme->pom_code));
+ letoh16(mlme->pom_code)));
if (ic->ic_opmode == IEEE80211_M_HOSTAP)
pgt_hostap_handle_mlme(sc, oid, mlme);
break;
@@ -3417,16 +3272,19 @@ pgt_update_sw_from_hw(struct pgt_softc *sc, struct pgt_async_trap *pa,
return;
}
if (ic->ic_state == IEEE80211_S_SCAN) {
- pgt_enter_critical(sc);
+ s = splnet();
error = pgt_oid_get(sc, PGT_OID_LINK_STATE, &ls, sizeof(ls));
- pgt_exit_critical(sc);
+ splx(s);
if (error)
return;
- if (letoh32(ls) != 0)
+ DPRINTF(("%s: up_sw_from_hw: link %u\n", sc->sc_dev.dv_xname,
+ htole32(ls)));
+ if (ls != 0)
ieee80211_new_state(ic, IEEE80211_S_RUN, -1);
}
+
gotlinkstate:
- pgt_enter_critical(sc);
+ s = splnet();
if (pgt_oid_get(sc, PGT_OID_NOISE_FLOOR, &noise, sizeof(noise)) != 0)
goto out;
sc->sc_noise = letoh32(noise);
@@ -3454,100 +3312,34 @@ gotlinkstate:
memcpy(ic->ic_bss->ni_essid, ssid.pos_ssid,
ssid.pos_length);
}
-out:
- pgt_exit_critical(sc);
-}
-
-int
-pgt_media_change(struct ifnet *ifp)
-{
- struct pgt_softc *sc = ifp->if_softc;
- int error;
-
- error = ieee80211_media_change(ifp);
- if (error == ENETRESET) {
- pgt_update_hw_from_sw(sc, 0, 0);
- error = 0;
- }
- return (error);
-}
-void
-pgt_media_status(struct ifnet *ifp, struct ifmediareq *imr)
-{
- struct pgt_softc *sc = ifp->if_softc;
- struct ieee80211com *ic = &sc->sc_ic;
- uint32_t ls;
- int alreadylocked;
-
- imr->ifm_active = IFM_IEEE80211;
- if (!(ifp->if_flags & IFF_UP))
- return;
- alreadylocked = 0;
- if (!alreadylocked)
- alreadylocked = 0;
- imr->ifm_status = IFM_AVALID;
- pgt_enter_critical(sc);
- if (pgt_oid_get(sc, PGT_OID_LINK_STATE, &ls, sizeof(ls)) != 0) {
- imr->ifm_active |= IFM_NONE;
- imr->ifm_status = 0;
- goto out;
- }
- ls = letoh32(ls);
- if (sc->sc_debug & SC_DEBUG_LINK)
- printf("%s: link: %u\n", sc->sc_dev.dv_xname, ls);
- if (ls == 0) {
- imr->ifm_active |= IFM_NONE;
- imr->ifm_status = 0;
- goto out;
- }
- if (ic->ic_state != IEEE80211_S_INIT)
- imr->ifm_status |= IFM_ACTIVE;
- /* XXX query the PHY "mode"? */
- imr->ifm_active |= ieee80211_rate2media(ic, ls, IEEE80211_MODE_AUTO);
- switch (ic->ic_opmode) {
- case IEEE80211_M_STA:
- break;
- case IEEE80211_M_IBSS:
- imr->ifm_active |= IFM_IEEE80211_ADHOC;
- break;
- case IEEE80211_M_AHDEMO:
- imr->ifm_active |= IFM_IEEE80211_ADHOC | IFM_FLAG0;
- break;
- case IEEE80211_M_HOSTAP:
- imr->ifm_active |= IFM_IEEE80211_HOSTAP;
- break;
- case IEEE80211_M_MONITOR:
- imr->ifm_active |= IFM_IEEE80211_MONITOR;
- break;
- }
out:
- pgt_exit_critical(sc);
- if (!alreadylocked)
- alreadylocked = 0;
+ splx(s);
}
/*
* Synchronization here is due to the softc lock being held when called.
*/
int
-pgt_new_state(struct ieee80211com *ic, enum ieee80211_state nstate,
- int mgtdata)
+pgt_newstate(struct ieee80211com *ic, enum ieee80211_state nstate, int arg)
{
- struct pgt_softc *sc;
+ struct pgt_softc *sc = ic->ic_if.if_softc;
enum ieee80211_state ostate;
- sc = (struct pgt_softc *)ic->ic_if.if_softc;
ostate = ic->ic_state;
- IEEE80211_DPRINTF(("%s: %s -> %s\n", __func__,
+
+ DPRINTF(("%s: newstate %s -> %s\n", sc->sc_dev.dv_xname,
ieee80211_state_name[ostate], ieee80211_state_name[nstate]));
+
switch (nstate) {
case IEEE80211_S_INIT:
if (sc->sc_dirtyq_count[PGT_QUEUE_DATA_LOW_TX] == 0)
ic->ic_if.if_timer = 0;
ic->ic_mgt_timer = 0;
ic->ic_flags &= ~IEEE80211_F_SIBSS;
- //IF_DRAIN(&ic->ic_mgtq);
+#if 0
+ IF_DRAIN(&ic->ic_mgtq);
+#endif
if (ic->ic_wep_ctx != NULL) {
free(ic->ic_wep_ctx, M_DEVBUF);
ic->ic_wep_ctx = NULL;
@@ -3556,19 +3348,18 @@ pgt_new_state(struct ieee80211com *ic, enum ieee80211_state nstate,
ic->ic_state = nstate;
break;
case IEEE80211_S_SCAN:
+#if 0
ic->ic_if.if_timer = 1;
ic->ic_mgt_timer = 0;
if (sc->sc_flags & SC_NOFREE_ALLNODES) {
- //struct ieee80211_node *ni;
- //struct pgt_ieee80211_node *pin;
+ struct ieee80211_node *ni;
+ struct pgt_ieee80211_node *pin;
/* Locked already by pff mutex. */
- /*
TAILQ_FOREACH(ni, &ic->ic_node, ni_list) {
pin = (struct pgt_ieee80211_node *)ni;
pin->pin_dot1x_auth = PIN_DOT1X_UNAUTHORIZED;
}
- */
sc->sc_flags &= ~SC_NOFREE_ALLNODES;
} else {
ieee80211_free_allnodes(ic);
@@ -3577,19 +3368,17 @@ pgt_new_state(struct ieee80211com *ic, enum ieee80211_state nstate,
/* Just use any old channel; we override it anyway. */
if (ic->ic_opmode == IEEE80211_M_HOSTAP)
ieee80211_create_ibss(ic, ic->ic_ibss_chan);
+#endif
break;
case IEEE80211_S_RUN:
ic->ic_if.if_timer = 1;
- ic->ic_mgt_timer = 0;
- ic->ic_state = nstate;
pgt_update_hw_from_nodes(sc);
- //if (!IFQ_DRV_IS_EMPTY(&ic->ic_if.if_snd))
- // pgt_start_body(sc, ic, &ic->ic_if);
break;
default:
break;
}
- return (0);
+
+ return (sc->sc_newstate(ic, nstate, arg));
}
int
@@ -3615,14 +3404,13 @@ pgt_drain_tx_queue(struct pgt_softc *sc, enum pgt_queue pq)
sc->sc_freeq_count[pq]++;
pgt_unload_tx_desc_frag(sc, pd);
if (sc->sc_debug & SC_DEBUG_QUEUES)
- printf("%s: queue: tx %u <- [%u] (drained)\n",
- sc->sc_dev.dv_xname, pd->pd_fragnum, pq);
+ DPRINTF(("%s: queue: tx %u <- [%u] (drained)\n",
+ sc->sc_dev.dv_xname, pd->pd_fragnum, pq));
wokeup++;
- if (pgt_queue_is_data(pq)) {
+ if (pgt_queue_is_data(pq))
sc->sc_ic.ic_if.if_oerrors++;
- //pgt_try_exit_data_critical(sc);
- }
}
+
return (wokeup);
}
@@ -3643,7 +3431,7 @@ pgt_dma_alloc(struct pgt_softc *sc)
size = sizeof(struct pgt_control_block);
error = bus_dmamap_create(sc->sc_dmat, size, 1, size, 0,
- BUS_DMA_ALLOCNOW, &sc->sc_cbdmam);
+ BUS_DMA_NOWAIT, &sc->sc_cbdmam);
if (error != 0) {
printf("%s: can not create DMA tag for control block\n",
sc->sc_dev);
@@ -3665,9 +3453,10 @@ pgt_dma_alloc(struct pgt_softc *sc)
sc->sc_dev);
goto out;
}
+ bzero(sc->sc_cb, size);
error = bus_dmamap_load(sc->sc_dmat, sc->sc_cbdmam,
- sc->sc_cb, size, NULL, BUS_DMA_WAITOK);
+ sc->sc_cb, size, NULL, BUS_DMA_NOWAIT);
if (error != 0) {
printf("%s: can not load DMA map for control block\n",
sc->sc_dev);
@@ -3702,6 +3491,7 @@ pgt_dma_alloc(struct pgt_softc *sc)
sc->sc_dev);
goto out;
}
+ bzero(sc->sc_psmbuf, size);
error = bus_dmamap_load(sc->sc_dmat, sc->sc_psmdmam,
sc->sc_psmbuf, size, NULL, BUS_DMA_WAITOK);
@@ -3813,7 +3603,7 @@ pgt_dma_alloc_queue(struct pgt_softc *sc, enum pgt_queue pq)
if (pgt_queue_is_rx(pq)) {
error = bus_dmamap_load(sc->sc_dmat, pd->pd_dmam,
- pd->pd_mem, PGT_FRAG_SIZE, NULL, BUS_DMA_WAITOK);
+ pd->pd_mem, PGT_FRAG_SIZE, NULL, BUS_DMA_NOWAIT);
if (error != 0) {
printf("%s: error load frag %u on queue %u\n",
sc->sc_dev, i, pq);
@@ -3822,6 +3612,7 @@ pgt_dma_alloc_queue(struct pgt_softc *sc, enum pgt_queue pq)
free(pd, M_DEVBUF);
break;
}
+ pd->pd_dmaaddr = pd->pd_dmam->dm_segs[0].ds_addr;
}
TAILQ_INSERT_TAIL(&sc->sc_freeq[pq], pd, pd_link);
}
diff --git a/sys/dev/ic/pgtreg.h b/sys/dev/ic/pgtreg.h
index 69c0cc3c2ec..877cf6afe2d 100644
--- a/sys/dev/ic/pgtreg.h
+++ b/sys/dev/ic/pgtreg.h
@@ -1,4 +1,23 @@
-/*-
+/* $OpenBSD: pgtreg.h,v 1.4 2006/09/16 10:36:12 mglocker Exp $ */
+
+/*
+ * Copyright (c) 2006 Claudio Jeker <claudio@openbsd.org>
+ * Copyright (c) 2006 Marcus Glocker <mglocker@openbsd.org>
+ *
+ * Permission to use, copy, modify, and distribute this software for any
+ * purpose with or without fee is hereby granted, provided that the above
+ * copyright notice and this permission notice appear in all copies.
+ *
+ * THE SOFTWARE IS PROVIDED "AS IS" AND THE AUTHOR DISCLAIMS ALL WARRANTIES
+ * WITH REGARD TO THIS SOFTWARE INCLUDING ALL IMPLIED WARRANTIES OF
+ * MERCHANTABILITY AND FITNESS. IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR
+ * ANY SPECIAL, DIRECT, INDIRECT, OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES
+ * WHATSOEVER RESULTING FROM LOSS OF USE, DATA OR PROFITS, WHETHER IN AN
+ * ACTION OF CONTRACT, NEGLIGENCE OR OTHER TORTIOUS ACTION, ARISING OUT OF
+ * OR IN CONNECTION WITH THE USE OR PERFORMANCE OF THIS SOFTWARE.
+ */
+
+/*
* Copyright (c) 2004 Fujitsu Laboratories of America, Inc.
* Copyright (c) 2004 Brian Fundakowski Feldman
* All rights reserved.
@@ -25,6 +44,9 @@
* SUCH DAMAGE.
*/
+#ifndef __PGTREG_H__
+#define __PGTREG_H__
+
/* Sizes, delays, etc. */
#define PGT_TX_LIST_CNT 32
#define PGT_RX_LIST_CNT 8
@@ -385,3 +407,5 @@ struct pgt_obj_frequencies {
uint16_t pof_count;
uint16_t pof_freqlist_mhz[0];
} __attribute__((packed));
+
+#endif
diff --git a/sys/dev/ic/pgtvar.h b/sys/dev/ic/pgtvar.h
index 39564fb38e3..2958007a13d 100644
--- a/sys/dev/ic/pgtvar.h
+++ b/sys/dev/ic/pgtvar.h
@@ -1,4 +1,23 @@
-/*-
+/* $OpenBSD: pgtvar.h,v 1.7 2006/09/16 10:36:12 mglocker Exp $ */
+
+/*
+ * Copyright (c) 2006 Claudio Jeker <claudio@openbsd.org>
+ * Copyright (c) 2006 Marcus Glocker <mglocker@openbsd.org>
+ *
+ * Permission to use, copy, modify, and distribute this software for any
+ * purpose with or without fee is hereby granted, provided that the above
+ * copyright notice and this permission notice appear in all copies.
+ *
+ * THE SOFTWARE IS PROVIDED "AS IS" AND THE AUTHOR DISCLAIMS ALL WARRANTIES
+ * WITH REGARD TO THIS SOFTWARE INCLUDING ALL IMPLIED WARRANTIES OF
+ * MERCHANTABILITY AND FITNESS. IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR
+ * ANY SPECIAL, DIRECT, INDIRECT, OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES
+ * WHATSOEVER RESULTING FROM LOSS OF USE, DATA OR PROFITS, WHETHER IN AN
+ * ACTION OF CONTRACT, NEGLIGENCE OR OTHER TORTIOUS ACTION, ARISING OUT OF
+ * OR IN CONNECTION WITH THE USE OR PERFORMANCE OF THIS SOFTWARE.
+ */
+
+/*
* Copyright (c) 2004 Fujitsu Laboratories of America, Inc.
* Copyright (c) 2004 Brian Fundakowski Feldman
* All rights reserved.
@@ -25,6 +44,36 @@
* SUCH DAMAGE.
*/
+#ifndef __PGTVAR_H__
+#define __PGTVAR_H__
+
+#define PGT_RX_RADIOTAP_PRESENT \
+ ((1 << IEEE80211_RADIOTAP_FLAGS) | \
+ (1 << IEEE80211_RADIOTAP_CHANNEL) | \
+ (1 << IEEE80211_RADIOTAP_RSSI))
+
+struct pgt_rx_radiotap_hdr {
+ struct ieee80211_radiotap_header wr_ihdr;
+ uint8_t wr_flags;
+ uint16_t wr_chan_freq;
+ uint16_t wr_chan_flags;
+ uint8_t wr_rssi;
+ uint8_t wr_max_rssi;
+} __packed;
+
+#define PGT_TX_RADIOTAP_PRESENT \
+ ((1 << IEEE80211_RADIOTAP_FLAGS) | \
+ (1 << IEEE80211_RADIOTAP_RATE) | \
+ (1 << IEEE80211_RADIOTAP_CHANNEL))
+
+struct pgt_tx_radiotap_hdr {
+ struct ieee80211_radiotap_header wt_ihdr;
+ uint8_t wt_flags;
+ uint8_t wt_rate;
+ uint16_t wt_chan_freq;
+ uint16_t wt_chan_flags;
+} __packed;
+
/*
* The struct pgt_desc is used to either enqueue or dequeue pgt_frags
* (packets) when either free or in flight.
@@ -75,28 +124,9 @@ struct pgt_ieee80211_node {
uint16_t pin_mlme_state;
};
-struct pgt_ieee80211_radiotap {
- struct ieee80211_radiotap_header pir_header;
- uint8_t pir_flags;
- uint8_t pir_rate;
- uint16_t pir_channel;
- uint16_t pir_channel_flags;
- uint8_t pir_db_antsignal;
- uint8_t pir_db_antnoise;
-};
-#define PGT_IEEE80211_RADIOTAP_PRESENT \
- (1 << IEEE80211_RADIOTAP_FLAGS | \
- 1 << IEEE80211_RADIOTAP_RATE | \
- 1 << IEEE80211_RADIOTAP_CHANNEL | \
- 1 << IEEE80211_RADIOTAP_DB_ANTSIGNAL | \
- 1 << IEEE80211_RADIOTAP_DB_ANTNOISE)
-
-struct bpf_if;
-
struct pgt_softc {
struct device sc_dev;
struct ieee80211com sc_ic;
- struct bpf_if *sc_drvbpf;
unsigned int sc_flags;
#define SC_NEEDS_FIRMWARE 0x00000001 /* do firmware upload on reset */
#define SC_UNINITIALIZED 0x00000002 /* still awaiting initial intr */
@@ -104,11 +134,11 @@ struct pgt_softc {
#define SC_NEEDS_RESET 0x00000008 /* going to reset when refcnt = 1 */
#define SC_INTR_RESET 0x00000020 /* interrupt resets at end */
#define SC_POWERSAVE 0x00000040 /* device is asleep */
-#define SC_GONE 0x00000080 /* device did not come back */
#define SC_NOFREE_ALLNODES 0x00000100 /* do not free assoc w/reinit */
#define SC_START_DESIRED 0x00000200 /* tried to start during mgmt-crit */
#define SC_KTHREAD 0x00000400 /* has a kthread around */
#define SC_ISL3877 0x00000800 /* chipset */
+ struct timeout sc_chanscan_timer;
/* configuration sysctls */
int sc_dot1x;
int sc_wds;
@@ -117,11 +147,6 @@ struct pgt_softc {
int16_t sc_80211_ioc_wep;
int16_t sc_80211_ioc_auth;
uint32_t sc_noise;
- unsigned int sc_refcnt; /* # sleeping with sc */
-// struct cv sc_critical_cv;
- struct thread *sc_critical_thread; /* allow mgmt recursion */
- int sc_critical; /* -1- = mgmt < 0 < 1+ data */
- struct thread *sc_drainer; /* who's doing removal/reset? */
unsigned int sc_debug;
#define SC_DEBUG_QUEUES 0x00000001
#define SC_DEBUG_MGMT 0x00000002
@@ -134,35 +159,52 @@ struct pgt_softc {
#define SC_DEBUG_RXANNEX 0x00000100
#define SC_DEBUG_RXFRAG 0x00000200
#define SC_DEBUG_RXETHER 0x00000400
- struct resource *sc_intres; /* interrupt resource */
- void *sc_intcookie;
- struct resource *sc_iores; /* IO memory resource */
bus_space_tag_t sc_iotag;
bus_space_handle_t sc_iohandle;
bus_dma_tag_t sc_dmat;
- //bus_dma_tag_t sc_cbdmat; /* control block DMA */
+
bus_dmamap_t sc_cbdmam;
bus_dma_segment_t sc_cbdmas;
- bus_addr_t sc_cbdmabusaddr;
struct pgt_control_block *sc_cb; /* DMA-mapped control block */
- //bus_dma_tag_t sc_psmdmat; /* power save buffer DMA */
+
bus_dmamap_t sc_psmdmam;
bus_dma_segment_t sc_psmdmas;
- bus_addr_t sc_psmdmabusaddr;
void *sc_psmbuf; /* DMA-mapped psm frame area */
- //bus_dma_tag_t sc_fragdmat; /* tags for all queues */
+
+ int (*sc_newstate)
+ (struct ieee80211com *,
+ enum ieee80211_state, int);
+
struct pgt_mgmt_descq sc_mgmtinprog;
struct pgt_descq sc_freeq[PGT_QUEUE_COUNT];
size_t sc_freeq_count[PGT_QUEUE_COUNT];
struct pgt_descq sc_dirtyq[PGT_QUEUE_COUNT];
size_t sc_dirtyq_count[PGT_QUEUE_COUNT];
- struct bintime sc_data_tx_started;
+ int sc_txtimer;
struct pgt_softc_kthread {
struct proc *sck_proc;
// struct cv sck_needed;
int sck_exit, sck_reset, sck_update;
TAILQ_HEAD(, pgt_async_trap) sck_traps;
} sc_kthread;
+
+#if NBPFILTER > 0
+ caddr_t sc_drvbpf;
+
+ union {
+ struct pgt_rx_radiotap_hdr th;
+ uint8_t pad[64];
+ } sc_rxtapu;
+#define sc_rxtap sc_rxtapu.th
+ int sc_rxtap_len;
+
+ union {
+ struct pgt_tx_radiotap_hdr th;
+ uint8_t pad[64];
+ } sc_txtapu;
+#define sc_txtap sc_txtapu.th
+ int sc_txtap_len;
+#endif
};
void pgt_attachhook(void *);
@@ -170,8 +212,6 @@ int pgt_intr(void *);
int pgt_attach(struct pgt_softc *);
int pgt_detach(struct pgt_softc *sc);
void pgt_reboot(struct pgt_softc *);
-/* Load one seg into the bus_addr_t * arg. */
-//void pgt_load_busaddr(void *, bus_dma_segment_t *, int, int);
static __inline int
pgt_queue_is_rx(enum pgt_queue pq)
@@ -204,3 +244,5 @@ pgt_queue_is_mgmt(enum pgt_queue pq)
return (pq == PGT_QUEUE_MGMT_RX ||
pq == PGT_QUEUE_MGMT_TX);
}
+
+#endif
diff --git a/sys/dev/pci/if_pgt_pci.c b/sys/dev/pci/if_pgt_pci.c
index 4ae79d727e2..68e1c57258b 100644
--- a/sys/dev/pci/if_pgt_pci.c
+++ b/sys/dev/pci/if_pgt_pci.c
@@ -1,4 +1,4 @@
-/* $OpenBSD: if_pgt_pci.c,v 1.5 2006/08/31 17:05:50 mglocker Exp $ */
+/* $OpenBSD: if_pgt_pci.c,v 1.6 2006/09/16 10:36:12 mglocker Exp $ */
/*
* Copyright (c) 2006 Marcus Glocker <mglocker@openbsd.org>
@@ -121,6 +121,11 @@ pgt_pci_attach(struct device *parent, struct device *self, void *aux)
return;
}
+ /* disable all interrupts */
+ bus_space_write_4(sc->sc_iotag, sc->sc_iohandle, PGT_REG_INT_EN, 0);
+ (void)bus_space_read_4(sc->sc_iotag, sc->sc_iohandle, PGT_REG_INT_EN);
+ DELAY(PGT_WRITEIO_DELAY);
+
/* establish interrupt */
intrstr = pci_intr_string(psc->sc_pc, ih);
psc->sc_ih = pci_intr_establish(psc->sc_pc, ih, IPL_NET, pgt_intr, sc,