summaryrefslogtreecommitdiff
diff options
context:
space:
mode:
-rw-r--r--sys/dev/pci/if_iwn.c1009
-rw-r--r--sys/dev/pci/if_iwnreg.h381
-rw-r--r--sys/dev/pci/if_iwnvar.h11
3 files changed, 868 insertions, 533 deletions
diff --git a/sys/dev/pci/if_iwn.c b/sys/dev/pci/if_iwn.c
index 4f269abc64a..e8d56992b95 100644
--- a/sys/dev/pci/if_iwn.c
+++ b/sys/dev/pci/if_iwn.c
@@ -1,4 +1,4 @@
-/* $OpenBSD: if_iwn.c,v 1.24 2008/10/13 16:37:10 damien Exp $ */
+/* $OpenBSD: if_iwn.c,v 1.25 2008/10/22 06:25:07 damien Exp $ */
/*-
* Copyright (c) 2007, 2008
@@ -88,10 +88,6 @@ void iwn_radiotap_attach(struct iwn_softc *);
#endif
void iwn_power(int, void *);
int iwn_nic_lock(struct iwn_softc *);
-void iwn_prph_write_region_4(struct iwn_softc *, uint32_t,
- const uint32_t *, int);
-void iwn_mem_set_region_4(struct iwn_softc *, uint32_t, uint32_t,
- int);
int iwn_eeprom_lock(struct iwn_softc *);
int iwn_read_prom_data(struct iwn_softc *, uint32_t, void *, int);
int iwn_dma_contig_alloc(bus_dma_tag_t, struct iwn_dma_info *,
@@ -126,7 +122,7 @@ int iwn_media_change(struct ifnet *);
int iwn_newstate(struct ieee80211com *, enum ieee80211_state, int);
void iwn_iter_func(void *, struct ieee80211_node *);
void iwn_calib_timeout(void *);
-void iwn_ampdu_rx_start(struct iwn_softc *, struct iwn_rx_desc *);
+void iwn_rx_phy(struct iwn_softc *, struct iwn_rx_desc *);
void iwn_rx_done(struct iwn_softc *, struct iwn_rx_desc *,
struct iwn_rx_data *);
void iwn5000_rx_calib_results(struct iwn_softc *,
@@ -135,10 +131,11 @@ void iwn_rx_statistics(struct iwn_softc *, struct iwn_rx_desc *);
void iwn4965_tx_done(struct iwn_softc *, struct iwn_rx_desc *);
void iwn5000_tx_done(struct iwn_softc *, struct iwn_rx_desc *);
void iwn_tx_done(struct iwn_softc *, struct iwn_rx_desc *, int,
- uint32_t);
+ uint8_t);
void iwn_cmd_done(struct iwn_softc *, struct iwn_rx_desc *);
void iwn_notif_intr(struct iwn_softc *);
void iwn_wakeup_intr(struct iwn_softc *);
+void iwn_fatal_intr(struct iwn_softc *);
int iwn_intr(void *);
void iwn4965_update_sched(struct iwn_softc *, int, int, uint8_t,
uint16_t);
@@ -156,9 +153,9 @@ int iwn4965_add_node(struct iwn_softc *, struct iwn_node_info *,
int);
int iwn5000_add_node(struct iwn_softc *, struct iwn_node_info *,
int);
-int iwn_setup_node_mrr(struct iwn_softc *,
- const struct ieee80211_node *, uint8_t);
-int iwn_set_fixed_rate(struct iwn_softc *, uint8_t, uint8_t, int);
+int iwn_set_link_quality(struct iwn_softc *,
+ struct ieee80211_node *);
+int iwn_add_broadcast_node(struct iwn_softc *, int);
void iwn_updateedca(struct ieee80211com *);
void iwn_set_led(struct iwn_softc *, uint8_t, uint8_t, uint8_t);
int iwn_set_critical_temp(struct iwn_softc *);
@@ -186,12 +183,26 @@ int iwn_config(struct iwn_softc *);
int iwn_scan(struct iwn_softc *, uint16_t);
int iwn_auth(struct iwn_softc *);
int iwn_run(struct iwn_softc *);
-int iwn4965_addba(struct iwn_softc *, struct ieee80211_node *,
- int, int, int);
-int iwn4965_delba(struct iwn_softc *, int, int, int);
-int iwn5000_addba(struct iwn_softc *, struct ieee80211_node *,
- int, int, int);
-int iwn5000_delba(struct iwn_softc *, int, int, int);
+int iwn_set_key(struct ieee80211com *, struct ieee80211_node *,
+ struct ieee80211_key *);
+void iwn_delete_key(struct ieee80211com *, struct ieee80211_node *,
+ struct ieee80211_key *);
+int iwn_ampdu_rx_start(struct ieee80211com *,
+ struct ieee80211_node *, uint8_t, uint16_t);
+void iwn_ampdu_rx_stop(struct ieee80211com *,
+ struct ieee80211_node *, uint8_t, uint16_t);
+int iwn_ampdu_tx_start(struct ieee80211com *,
+ struct ieee80211_node *, uint8_t, uint16_t);
+void iwn_ampdu_tx_stop(struct ieee80211com *,
+ struct ieee80211_node *, uint8_t, uint16_t);
+void iwn4965_ampdu_tx_start(struct iwn_softc *,
+ struct ieee80211_node *, uint8_t, uint16_t);
+void iwn4965_ampdu_tx_stop(struct iwn_softc *,
+ uint8_t, uint16_t);
+void iwn5000_ampdu_tx_start(struct iwn_softc *,
+ struct ieee80211_node *, uint8_t, uint16_t);
+void iwn5000_ampdu_tx_stop(struct iwn_softc *,
+ uint8_t, uint16_t);
int iwn5000_query_calibration(struct iwn_softc *);
int iwn5000_send_calibration(struct iwn_softc *);
int iwn4965_post_alive(struct iwn_softc *);
@@ -238,6 +249,8 @@ static const struct iwn_hal iwn4965_hal = {
iwn4965_set_gains,
iwn4965_add_node,
iwn4965_tx_done,
+ iwn4965_ampdu_tx_start,
+ iwn4965_ampdu_tx_stop,
"iwn-4965",
&iwn4965_sensitivity_limits,
IWN4965_NTXQUEUES,
@@ -264,6 +277,8 @@ static const struct iwn_hal iwn5000_hal = {
iwn5000_set_gains,
iwn5000_add_node,
iwn5000_tx_done,
+ iwn5000_ampdu_tx_start,
+ iwn5000_ampdu_tx_stop,
"iwn-5000",
&iwn5000_sensitivity_limits,
IWN5000_NTXQUEUES,
@@ -319,7 +334,7 @@ iwn_attach(struct device *parent, struct device *self, void *aux)
return;
}
- /* Clear device-specific "PCI retry timeout" register (41H). */
+ /* Clear device-specific "PCI retry timeout" register (41h). */
reg = pci_conf_read(sc->sc_pct, sc->sc_pcitag, 0x40);
reg &= ~0xff00;
pci_conf_write(sc->sc_pct, sc->sc_pcitag, 0x40, reg);
@@ -406,8 +421,8 @@ iwn_attach(struct device *parent, struct device *self, void *aux)
/* Power OFF adapter. */
iwn_apm_stop(sc);
- printf(", %dT%dR, %.4s, address %s\n", sc->ntxchains, sc->nrxchains,
- sc->eeprom_domain, ether_sprintf(ic->ic_myaddr));
+ printf(", MIMO %dT%dR, %.4s, address %s\n", sc->ntxchains,
+ sc->nrxchains, sc->eeprom_domain, ether_sprintf(ic->ic_myaddr));
/* Initialization firmware has not been loaded yet. */
sc->sc_flags |= IWN_FLAG_FIRST_BOOT;
@@ -422,7 +437,8 @@ iwn_attach(struct device *parent, struct device *self, void *aux)
IEEE80211_C_RSN | /* WPA/RSN */
IEEE80211_C_MONITOR | /* monitor mode supported */
IEEE80211_C_SHSLOT | /* short slot time supported */
- IEEE80211_C_SHPREAMBLE; /* short preamble supported */
+ IEEE80211_C_SHPREAMBLE | /* short preamble supported */
+ IEEE80211_C_PMGT; /* power saving supported */
/* Set supported rates. */
ic->ic_sup_rates[IEEE80211_MODE_11A] = ieee80211_std_rateset_11a;
@@ -450,6 +466,10 @@ iwn_attach(struct device *parent, struct device *self, void *aux)
/* Override 802.11 state transition machine. */
sc->sc_newstate = ic->ic_newstate;
ic->ic_newstate = iwn_newstate;
+#ifdef notyet
+ ic->ic_set_key = iwn_set_key;
+ ic->ic_delete_key = iwn_delete_key;
+#endif
ieee80211_media_init(ifp, iwn_media_change, ieee80211_media_status);
sc->amrr.amrr_min_success_threshold = 1;
@@ -484,27 +504,16 @@ iwn_hal_attach(struct iwn_softc *sc)
switch (sc->hw_type) {
case IWN_HW_REV_TYPE_4965:
sc->sc_hal = &iwn4965_hal;
- sc->ntxchains = 2;
- sc->nrxchains = 2; /* XXX should be 3? */
sc->critical_temp = IWN_CTOK(110);
break;
- case IWN_HW_REV_TYPE_5100:
- sc->sc_hal = &iwn5000_hal;
- sc->ntxchains = 1;
- sc->nrxchains = 2;
- sc->critical_temp = 110;
- break;
case IWN_HW_REV_TYPE_5150:
sc->sc_hal = &iwn5000_hal;
- sc->ntxchains = 1;
- sc->nrxchains = 2;
sc->critical_temp = IWN_CTOK(110);
break;
+ case IWN_HW_REV_TYPE_5100:
case IWN_HW_REV_TYPE_5300:
case IWN_HW_REV_TYPE_5350:
sc->sc_hal = &iwn5000_hal;
- sc->ntxchains = 3;
- sc->nrxchains = 3;
sc->critical_temp = 110;
break;
default:
@@ -563,7 +572,7 @@ iwn_power(int why, void *arg)
if (why != PWR_RESUME)
return;
- /* Clear device-specific "PCI retry timeout" register (41H). */
+ /* Clear device-specific "PCI retry timeout" register (41h). */
reg = pci_conf_read(sc->sc_pct, sc->sc_pcitag, 0x40);
reg &= ~0xff00;
pci_conf_write(sc->sc_pct, sc->sc_pcitag, 0x40, reg);
@@ -617,7 +626,19 @@ iwn_prph_write(struct iwn_softc *sc, uint32_t addr, uint32_t data)
IWN_WRITE(sc, IWN_PRPH_WDATA, data);
}
-void
+static __inline void
+iwn_prph_setbits(struct iwn_softc *sc, uint32_t addr, uint32_t mask)
+{
+ iwn_prph_write(sc, addr, iwn_prph_read(sc, addr) | mask);
+}
+
+static __inline void
+iwn_prph_clrbits(struct iwn_softc *sc, uint32_t addr, uint32_t mask)
+{
+ iwn_prph_write(sc, addr, iwn_prph_read(sc, addr) & ~mask);
+}
+
+static __inline void
iwn_prph_write_region_4(struct iwn_softc *sc, uint32_t addr,
const uint32_t *data, int count)
{
@@ -652,7 +673,15 @@ iwn_mem_write_2(struct iwn_softc *sc, uint32_t addr, uint16_t data)
iwn_mem_write(sc, addr & ~3, tmp);
}
-void
+static __inline void
+iwn_mem_read_region_4(struct iwn_softc *sc, uint32_t addr, uint32_t *data,
+ int count)
+{
+ for (; count > 0; count--, addr += 4)
+ *data++ = iwn_mem_read(sc, addr);
+}
+
+static __inline void
iwn_mem_set_region_4(struct iwn_softc *sc, uint32_t addr, uint32_t val,
int count)
{
@@ -1101,6 +1130,17 @@ iwn_read_eeprom(struct iwn_softc *sc)
iwn_read_prom_data(sc, IWN_EEPROM_RFCFG, &val, 2);
sc->rfcfg = letoh16(val);
DPRINTF(("radio config=0x%04x\n", sc->rfcfg));
+ sc->txantmsk = IWN_RFCFG_TXANTMSK(sc->rfcfg);
+ sc->rxantmsk = IWN_RFCFG_RXANTMSK(sc->rfcfg);
+ /* Count the number of TX and RX chains. */
+ sc->ntxchains =
+ ((sc->txantmsk >> 2) & 1) +
+ ((sc->txantmsk >> 1) & 1) +
+ ((sc->txantmsk >> 0) & 1);
+ sc->nrxchains =
+ ((sc->rxantmsk >> 2) & 1) +
+ ((sc->rxantmsk >> 1) & 1) +
+ ((sc->rxantmsk >> 0) & 1);
/* Read MAC address. */
iwn_read_prom_data(sc, IWN_EEPROM_MAC, ic->ic_myaddr, 6);
@@ -1325,7 +1365,7 @@ iwn_newstate(struct ieee80211com *ic, enum ieee80211_state nstate, int arg)
switch (nstate) {
case IEEE80211_S_SCAN:
/* Make the link LED blink while we're scanning. */
- iwn_set_led(sc, IWN_LED_LINK, 20, 2);
+ iwn_set_led(sc, IWN_LED_LINK, 10, 10);
if ((error = iwn_scan(sc, IEEE80211_CHAN_2GHZ)) != 0) {
printf("%s: could not initiate scan\n",
@@ -1406,24 +1446,24 @@ iwn_calib_timeout(void *arg)
}
/*
- * Process an AMPDU_RX_START firmware notification. This is usually
- * immediately followed by an AMPDU_RX_DONE notification.
+ * Process an RX_PHY firmware notification. This is usually immediately
+ * followed by an MPDU_RX_DONE notification.
*/
void
-iwn_ampdu_rx_start(struct iwn_softc *sc, struct iwn_rx_desc *desc)
+iwn_rx_phy(struct iwn_softc *sc, struct iwn_rx_desc *desc)
{
struct iwn_rx_stat *stat;
- DPRINTFN(2, ("received AMPDU stats\n"));
- /* Save RX statistics, they will be used on AMPDU_RX_DONE. */
+ DPRINTFN(2, ("received PHY stats\n"));
+ /* Save RX statistics, they will be used on MPDU_RX_DONE. */
stat = (struct iwn_rx_stat *)(desc + 1);
memcpy(&sc->last_rx_stat, stat, sizeof (*stat));
sc->last_rx_valid = 1;
}
/*
- * Process an RX_DONE (4965AGN only) or AMPDU_RX_DONE firmware notification.
- * Each AMPDU_RX_DONE notification must be preceded by an AMPDU_RX_START.
+ * Process an RX_DONE (4965AGN only) or MPDU_RX_DONE firmware notification.
+ * Each MPDU_RX_DONE notification must be preceded by an RX_PHY one.
*/
void
iwn_rx_done(struct iwn_softc *sc, struct iwn_rx_desc *desc,
@@ -1443,10 +1483,10 @@ iwn_rx_done(struct iwn_softc *sc, struct iwn_rx_desc *desc,
uint32_t *tail;
int len, rssi;
- if (desc->type == IWN_AMPDU_RX_DONE) {
- /* Check for prior AMPDU_RX_START. */
+ if (desc->type == IWN_MPDU_RX_DONE) {
+ /* Check for prior RX_PHY. */
if (!sc->last_rx_valid) {
- DPRINTF(("missing AMPDU_RX_START\n"));
+ DPRINTF(("missing RX_PHY\n"));
ifp->if_ierrors++;
return;
}
@@ -1461,11 +1501,11 @@ iwn_rx_done(struct iwn_softc *sc, struct iwn_rx_desc *desc,
ifp->if_ierrors++;
return;
}
- if (desc->type == IWN_AMPDU_RX_DONE) {
- struct iwn_rx_ampdu *ampdu =
- (struct iwn_rx_ampdu *)(desc + 1);
- head = (caddr_t)(ampdu + 1);
- len = letoh16(ampdu->len);
+ if (desc->type == IWN_MPDU_RX_DONE) {
+ struct iwn_rx_mpdu *mpdu =
+ (struct iwn_rx_mpdu *)(desc + 1);
+ head = (caddr_t)(mpdu + 1);
+ len = letoh16(mpdu->len);
} else {
head = (caddr_t)(stat + 1) + stat->cfg_phy_len;
len = letoh16(stat->len);
@@ -1528,6 +1568,8 @@ iwn_rx_done(struct iwn_softc *sc, struct iwn_rx_desc *desc,
struct iwn_rx_radiotap_header *tap = &sc->sc_rxtap;
tap->wr_flags = 0;
+ if (stat->flags & htole16(IWN_STAT_FLAG_SHPREAMBLE))
+ tap->wr_flags |= IEEE80211_RADIOTAP_F_SHORTPRE;
tap->wr_chan_freq =
htole16(ic->ic_channels[stat->chan].ic_freq);
tap->wr_chan_flags =
@@ -1585,7 +1627,7 @@ iwn_rx_done(struct iwn_softc *sc, struct iwn_rx_desc *desc,
void
iwn5000_rx_calib_results(struct iwn_softc *sc, struct iwn_rx_desc *desc)
{
- struct iwn5000_calib *calib = (struct iwn5000_calib *)(desc + 1);
+ struct iwn_phy_calib *calib = (struct iwn_phy_calib *)(desc + 1);
u_int idx, len;
/* No initial calibration required for 5150! */
@@ -1688,7 +1730,7 @@ iwn4965_tx_done(struct iwn_softc *sc, struct iwn_rx_desc *desc)
{
struct iwn4965_tx_stat *stat = (struct iwn4965_tx_stat *)(desc + 1);
- iwn_tx_done(sc, desc, stat->retrycnt, letoh32(stat->status));
+ iwn_tx_done(sc, desc, stat->retrycnt, letoh32(stat->status) & 0xff);
}
void
@@ -1698,7 +1740,7 @@ iwn5000_tx_done(struct iwn_softc *sc, struct iwn_rx_desc *desc)
/* Reset TX scheduler slot. */
iwn5000_reset_sched(sc, desc->qid & 0xf, desc->idx);
- iwn_tx_done(sc, desc, stat->retrycnt, letoh16(stat->status));
+ iwn_tx_done(sc, desc, stat->retrycnt, letoh16(stat->status) & 0xff);
}
/*
@@ -1706,7 +1748,7 @@ iwn5000_tx_done(struct iwn_softc *sc, struct iwn_rx_desc *desc)
*/
void
iwn_tx_done(struct iwn_softc *sc, struct iwn_rx_desc *desc, int retrycnt,
- uint32_t status)
+ uint8_t status)
{
struct ieee80211com *ic = &sc->sc_ic;
struct ifnet *ifp = &ic->ic_if;
@@ -1786,12 +1828,12 @@ iwn_notif_intr(struct iwn_softc *sc)
iwn_cmd_done(sc, desc);
switch (desc->type) {
- case IWN_AMPDU_RX_START:
- iwn_ampdu_rx_start(sc, desc);
+ case IWN_RX_PHY:
+ iwn_rx_phy(sc, desc);
break;
case IWN_RX_DONE: /* 4965AGN only. */
- case IWN_AMPDU_RX_DONE:
+ case IWN_MPDU_RX_DONE:
/* An 802.11 frame has been received. */
iwn_rx_done(sc, desc, data);
break;
@@ -1840,6 +1882,8 @@ iwn_notif_intr(struct iwn_softc *sc)
/* Save microcontroller's report. */
memcpy(&sc->ucode_info, uc, sizeof (*uc));
}
+ /* Save the address of the error log in SRAM. */
+ sc->errptr = letoh32(uc->errptr);
break;
}
case IWN_STATE_CHANGED:
@@ -1928,6 +1972,68 @@ iwn_wakeup_intr(struct iwn_softc *sc)
}
}
+/*
+ * Dump the error log of the firmware when a firmware panic occurs. Although
+ * we can't debug the firmware because it is neither open source nor free, it
+ * can help us to identify certain classes of problems.
+ */
+void
+iwn_fatal_intr(struct iwn_softc *sc)
+{
+#define N(a) (sizeof (a) / sizeof ((a)[0]))
+ const struct iwn_hal *hal = sc->sc_hal;
+ struct iwn_fw_dump dump;
+ int i;
+
+ /* Check that the error log address is valid. */
+ if (sc->errptr < IWN_FW_DATA_BASE ||
+ sc->errptr + sizeof (dump) >
+ IWN_FW_DATA_BASE + hal->fw_data_maxsz) {
+ printf("%s: bad firmware error log address 0x%08x\n",
+ sc->sc_dev.dv_xname, sc->errptr);
+ return;
+ }
+ if (iwn_nic_lock(sc) != 0) {
+ printf("%s: could not read firmware error log\n",
+ sc->sc_dev.dv_xname);
+ return;
+ }
+ /* Read firmware error log from SRAM. */
+ iwn_mem_read_region_4(sc, sc->errptr, (uint32_t *)&dump,
+ sizeof (dump) / sizeof (uint32_t));
+ iwn_nic_unlock(sc);
+
+ if (dump.valid == 0) {
+ printf("%s: firmware error log is empty\n",
+ sc->sc_dev.dv_xname);
+ return;
+ }
+ printf("firmware error log:\n");
+ printf("\error type = \"%s\" (0x%08X)\n",
+ dump.id < N(iwn_fw_errmsg) ? iwn_fw_errmsg[dump.id] : "UNKNOWN",
+ dump.id);
+ printf("\tprogram counter = 0x%08X\n", dump.pc);
+ printf("\tsource line = 0x%08X\n", dump.src_line);
+ printf("\terror data = 0x%08X%08X\n",
+ dump.error_data[0], dump.error_data[1]);
+ printf("\tbranch link = 0x%08X%08X\n",
+ dump.branch_link[0], dump.branch_link[1]);
+ printf("\tinterrupt link = 0x%08X\n",
+ dump.interrupt_link[0], dump.interrupt_link[1]);
+ printf("\ttime = %u\n", dump.time[0]);
+
+ /* Dump driver status (TX and RX rings) while we're here. */
+ printf("driver status:\n");
+ for (i = 0; i < hal->ntxqs; i++) {
+ struct iwn_tx_ring *ring = &sc->txq[i];
+ printf("\ttx ring %2d: qid=%2d cur=%3d queued=%3d\n",
+ i, ring->qid, ring->cur, ring->queued);
+ }
+ printf("\trx ring: cur=%d\n", sc->rxq.cur);
+ printf("\t802.11 state %d\n", sc->sc_ic.ic_state);
+#undef N
+}
+
int
iwn_intr(void *arg)
{
@@ -1961,10 +2067,12 @@ iwn_intr(void *arg)
if (r1 & IWN_INT_CT_REACHED) {
printf("%s: critical temperature reached!\n",
sc->sc_dev.dv_xname);
- /* XXX reduce TX power? */
+ /* XXX Reduce TX power? */
}
if (r1 & (IWN_INT_SW_ERR | IWN_INT_HW_ERR)) {
printf("%s: fatal firmware error\n", sc->sc_dev.dv_xname);
+ /* Dump firmware error log and stop. */
+ iwn_fatal_intr(sc);
ifp->if_flags &= ~IFF_UP;
iwn_stop(ifp, 1);
return 1;
@@ -2054,6 +2162,75 @@ iwn_plcp_signal(int rate)
/* Determine if a given rate is CCK or OFDM. */
#define IWN_RATE_IS_OFDM(rate) ((rate) >= 12 && (rate) != 22)
+#ifdef notyet
+void
+iwn_ccmp_encap(struct iwn_softc *sc, struct mbuf *m0, struct ieee80211_key *k)
+{
+ u_int8_t *ivp;
+
+ M_PREPEND(m0, IEEE80211_CCMP_HDRLEN, M_NOWAIT);
+ /* NB: Can't fail because of previous m_adj() call. */
+
+ k->k_tsc++; /* Increment the 48-bit PN. */
+
+ /* Construct CCMP header. */
+ ivp = mtod(m0, u_int8_t *);
+ ivp[0] = k->k_tsc; /* PN0 */
+ ivp[1] = k->k_tsc >> 8; /* PN1 */
+ ivp[2] = 0; /* Rsvd */
+ ivp[3] = k->k_id << 6 | IEEE80211_WEP_EXTIV; /* KeyID | ExtIV */
+ ivp[4] = k->k_tsc >> 16; /* PN2 */
+ ivp[5] = k->k_tsc >> 24; /* PN3 */
+ ivp[6] = k->k_tsc >> 32; /* PN4 */
+ ivp[7] = k->k_tsc >> 40; /* PN5 */
+}
+
+void
+iwn_ccmp_decap(struct iwn_softc *sc, struct mbuf *m0, struct ieee80211_key *k)
+{
+ struct ieee80211_frame *wh;
+ uint64_t pn, *prsc;
+ uint8_t *ivp;
+ uint8_t tid;
+ int hdrlen;
+
+ wh = mtod(m0, struct ieee80211_frame *);
+ hdrlen = ieee80211_get_hdrlen(wh);
+ ivp = (uint8_t *)wh + hdrlen;
+
+ /* Check that ExtIV bit is be set. */
+ if (!(ivp[3] & IEEE80211_WEP_EXTIV))
+ return 1;
+
+ tid = ieee80211_has_qos(wh) ?
+ ieee80211_get_qos(wh) & IEEE80211_QOS_TID : 0;
+ prsc = &k->k_rsc[tid];
+
+ /* Extract the 48-bit PN from the CCMP header. */
+ pn = (u_int64_t)ivp[0] |
+ (u_int64_t)ivp[1] << 8 |
+ (u_int64_t)ivp[4] << 16 |
+ (u_int64_t)ivp[5] << 24 |
+ (u_int64_t)ivp[6] << 32 |
+ (u_int64_t)ivp[7] << 40;
+ if (pn <= *prsc) {
+ /* Replayed frame, discard. */
+ ic->ic_stats.is_ccmp_replays++;
+ return 1;
+ }
+ /* Update last seen packet number. */
+ *prsc = pn;
+
+ /* Clear Protected bit and strip IV. */
+ wh->i_fc[1] &= ~IEEE80211_FC1_PROTECTED;
+ ovbcopy(wh, mtod(m0, caddr_t) + IEEE80211_CCMP_HDRLEN, hdrlen);
+ m_adj(m0, IEEE80211_CCMP_HDRLEN);
+ /* Strip MIC. */
+ m_adj(m0, -IEEE80211_CCMP_MICLEN);
+ return 0;
+}
+#endif
+
int
iwn_tx_data(struct iwn_softc *sc, struct mbuf *m0, struct ieee80211_node *ni)
{
@@ -2070,7 +2247,7 @@ iwn_tx_data(struct iwn_softc *sc, struct mbuf *m0, struct ieee80211_node *ni)
struct mbuf *mnew;
uint32_t flags;
uint16_t qos;
- uint8_t tid, type;
+ uint8_t tid, txant, type;
u_int hdrlen;
int i, hasqos, rate, error, pad;
@@ -2146,7 +2323,8 @@ iwn_tx_data(struct iwn_softc *sc, struct mbuf *m0, struct ieee80211_node *ni)
cmd->idx = ring->cur;
tx = (struct iwn_cmd_data *)cmd->data;
- /* NB: No need to bzero tx, all fields are reinitialized here. */
+ /* NB: No need to clear tx, all fields are reinitialized here. */
+ tx->scratch = 0; /* clear "scratch" area */
flags = 0;
if (!IEEE80211_IS_MULTICAST(wh->i_addr1)) {
@@ -2179,9 +2357,9 @@ iwn_tx_data(struct iwn_softc *sc, struct mbuf *m0, struct ieee80211_node *ni)
}
if (flags & (IWN_TX_NEED_RTS | IWN_TX_NEED_CTS)) {
if (sc->hw_type != IWN_HW_REV_TYPE_4965) {
- /* 5000 autoselect RTS/CTS or CTS-to-self. */
+ /* 5000 autoselects RTS/CTS or CTS-to-self. */
flags &= ~(IWN_TX_NEED_RTS | IWN_TX_NEED_CTS);
- flags |= IWN5000_TX_NEED_RTS_CTS;
+ flags |= IWN_TX_NEED_PROTECTION;
} else
flags |= IWN_TX_FULL_TXOP;
}
@@ -2191,7 +2369,7 @@ iwn_tx_data(struct iwn_softc *sc, struct mbuf *m0, struct ieee80211_node *ni)
type != IEEE80211_FC0_TYPE_DATA)
tx->id = hal->broadcast_id;
else
- tx->id = IWN_ID_BSS;
+ tx->id = ((struct iwn_node *)ni)->id;
if (type == IEEE80211_FC0_TYPE_MGT) {
uint8_t subtype = wh->i_fc[0] & IEEE80211_FC0_SUBTYPE_MASK;
@@ -2230,7 +2408,8 @@ iwn_tx_data(struct iwn_softc *sc, struct mbuf *m0, struct ieee80211_node *ni)
/* Group or management frame. */
tx->ridx = IWN_MAX_TX_RETRIES - 1;
/* XXX Alternate between antenna A and B? */
- tx->rflags |= IWN_RFLAG_ANT_B;
+ txant = IWN_LSB(sc->txantmsk);
+ tx->rflags |= IWN_RFLAG_ANT(txant);
} else {
tx->ridx = ni->ni_rates.rs_nrates - ni->ni_txrate - 1;
/* Tell adapter to ignore rflags. */
@@ -2297,7 +2476,7 @@ iwn_tx_data(struct iwn_softc *sc, struct mbuf *m0, struct ieee80211_node *ni)
/* Other DMA segments are for data payload. */
for (i = 1; i <= data->map->dm_nsegs; i++) {
IWN_SET_DESC_SEG(desc, i, data->map->dm_segs[i - 1].ds_addr,
- data->map->dm_segs[i - 1].ds_len);
+ data->map->dm_segs[i - 1].ds_len);
}
/* Update TX scheduler. */
@@ -2442,6 +2621,22 @@ iwn_ioctl(struct ifnet *ifp, u_long cmd, caddr_t data)
error = 0;
break;
+ case SIOCS80211POWER:
+ error = ieee80211_ioctl(ifp, cmd, data);
+ if (error != ENETRESET)
+ break;
+ if (ic->ic_state == IEEE80211_S_RUN &&
+ sc->calib.state == IWN_CALIB_STATE_RUN) {
+ if (ic->ic_flags & IEEE80211_F_PMGTON)
+ error = iwn_set_pslevel(sc, 0, 3, 0);
+ else /* back to CAM */
+ error = iwn_set_pslevel(sc, 0, 0, 0);
+ } else {
+ /* Defer until transition to IWN_CALIB_STATE_RUN. */
+ error = 0;
+ }
+ break;
+
default:
error = ieee80211_ioctl(ifp, cmd, data);
}
@@ -2454,7 +2649,6 @@ iwn_ioctl(struct ifnet *ifp, u_long cmd, caddr_t data)
(void)iwn_init(ifp);
}
}
-
splx(s);
return error;
}
@@ -2529,15 +2723,18 @@ int
iwn4965_add_node(struct iwn_softc *sc, struct iwn_node_info *node, int async)
{
struct iwn4965_node_info hnode;
+ caddr_t src, dst;
- /* We use the node structure for 5000 internally, convert it. */
- memset(&hnode, 0, sizeof hnode);
- hnode.control = node->control;
- IEEE80211_ADDR_COPY(hnode.macaddr, node->macaddr);
- hnode.id = node->id;
- hnode.flags = node->flags;
- hnode.htflags = node->htflags;
- /* XXX More fields. */
+ /*
+ * We use the node structure for 5000 Series internally (it is
+ * a superset of the one for 4965AGN). We thus copy the common
+ * fields before sending the command.
+ */
+ src = (caddr_t)node;
+ dst = (caddr_t)&hnode;
+ memcpy(dst, src, 48);
+ /* Skip TSC, RX MIC and TX MIC fields from ``src''. */
+ memcpy(dst + 48, src + 72, 20);
return iwn_cmd(sc, IWN_CMD_ADD_NODE, &hnode, sizeof hnode, async);
}
@@ -2548,71 +2745,86 @@ iwn5000_add_node(struct iwn_softc *sc, struct iwn_node_info *node, int async)
return iwn_cmd(sc, IWN_CMD_ADD_NODE, node, sizeof (*node), async);
}
-/*
- * Configure hardware multi-rate retries for one node.
- */
int
-iwn_setup_node_mrr(struct iwn_softc *sc, const struct ieee80211_node *ni,
- uint8_t id)
-{
- const struct ieee80211_rateset *rs = &ni->ni_rates;
- struct iwn_cmd_mrr mrr;
- uint8_t rate;
- int i, r;
-
- memset(&mrr, 0, sizeof mrr);
- mrr.id = id;
- mrr.ssmask = 2;
- mrr.dsmask = 3;
- mrr.ampdu_max = 64;
- mrr.ampdu_disable = 3;
- mrr.ampdu_limit = htole16(4000);
-
- r = rs->rs_nrates - 1;
- for (i = 0; i < IWN_MAX_TX_RETRIES && r >= 0; i++, r--) {
- rate = rs->rs_rates[r] & IEEE80211_RATE_VAL;
- DPRINTF(("retry %d: rate %d\n", i, rate));
- mrr.table[i].rate = iwn_plcp_signal(rate);
- mrr.table[i].rflags = IWN_RFLAG_ANT_B;
- if (!IWN_RATE_IS_OFDM(rate))
- mrr.table[i].rflags |= IWN_RFLAG_CCK;
- }
- /* Pad with the lowest available bit-rate. */
- rate = rs->rs_rates[0] & IEEE80211_RATE_VAL;
- for (; i < IWN_MAX_TX_RETRIES; i++) {
- DPRINTF(("retry %d: rate %d\n", i, rate));
- mrr.table[i].rate = iwn_plcp_signal(rate);
- mrr.table[i].rflags = IWN_RFLAG_ANT_B;
+iwn_set_link_quality(struct iwn_softc *sc, struct ieee80211_node *ni)
+{
+ struct iwn_node *wn = (void *)ni;
+ struct ieee80211_rateset *rs = &ni->ni_rates;
+ struct iwn_cmd_link_quality linkq;
+ uint8_t txant, rate;
+ int i, ridx;
+
+ memset(&linkq, 0, sizeof linkq);
+ linkq.id = wn->id;
+ linkq.antmsk_1stream = IWN_ANT_B;
+ linkq.antmsk_2stream = IWN_ANT_A | IWN_ANT_B;
+ linkq.ampdu_max = 64;
+ linkq.ampdu_threshold = 3;
+ linkq.ampdu_limit = htole16(4000); /* 4ms */
+
+ /* Use the first valid TX antenna. */
+ txant = IWN_LSB(sc->txantmsk);
+
+ /* Start at highest available bit-rate. */
+ ridx = rs->rs_nrates - 1;
+ for (i = 0; i < IWN_MAX_TX_RETRIES; i++) {
+ rate = rs->rs_rates[ridx] & IEEE80211_RATE_VAL;
+ DPRINTF(("retry %d, rate %d\n", i, rate));
+ linkq.retry[i].rate = iwn_plcp_signal(rate);
+ linkq.retry[i].rflags = IWN_RFLAG_ANT(txant);
if (!IWN_RATE_IS_OFDM(rate))
- mrr.table[i].rflags |= IWN_RFLAG_CCK;
+ linkq.retry[i].rflags |= IWN_RFLAG_CCK;
+ /* Next retry at immediate lower bit-rate. */
+ if (ridx > 0)
+ ridx--;
}
- return iwn_cmd(sc, IWN_CMD_NODE_MRR_SETUP, &mrr, sizeof mrr, 1);
+ return iwn_cmd(sc, IWN_CMD_LINK_QUALITY, &linkq, sizeof linkq, 1);
}
int
-iwn_set_fixed_rate(struct iwn_softc *sc, uint8_t id, uint8_t rate, int async)
+iwn_add_broadcast_node(struct iwn_softc *sc, int async)
{
- struct iwn_cmd_mrr mrr;
- int i;
+ const struct iwn_hal *hal = sc->sc_hal;
+ struct iwn_node_info node;
+ struct iwn_cmd_link_quality linkq;
+ uint8_t txant;
+ int i, error;
- memset(&mrr, 0, sizeof mrr);
- mrr.id = id;
- mrr.ssmask = 2;
- mrr.dsmask = 3;
- mrr.ampdu_max = 64;
- mrr.ampdu_disable = 3;
- mrr.ampdu_limit = htole16(4000);
-
- /* To set a fixed rate, we make all retries use the same rate. */
- mrr.table[0].rate = iwn_plcp_signal(rate);
- mrr.table[0].rflags = IWN_RFLAG_ANT_B;
- if (!IWN_RATE_IS_OFDM(rate))
- mrr.table[0].rflags |= IWN_RFLAG_CCK;
+ memset(&node, 0, sizeof node);
+ IEEE80211_ADDR_COPY(node.macaddr, etherbroadcastaddr);
+ node.id = hal->broadcast_id;
+ DPRINTF(("adding broadcast node\n"));
+ if ((error = hal->add_node(sc, &node, async)) != 0)
+ return error;
+
+ /*
+ * Setup link quality for broadcast node. The broadcast node is
+ * used to send group-addressed and management frames.
+ */
+ memset(&linkq, 0, sizeof linkq);
+ linkq.id = hal->broadcast_id;
+ linkq.antmsk_1stream = IWN_ANT_B;
+ linkq.antmsk_2stream = IWN_ANT_A | IWN_ANT_B;
+ linkq.ampdu_max = 64;
+ linkq.ampdu_threshold = 3;
+ linkq.ampdu_limit = htole16(4000); /* 4ms */
+
+ /* Use the first valid TX antenna. */
+ txant = IWN_LSB(sc->txantmsk);
+
+ /* Use lowest mandatory bit-rate. */
+ if (sc->sc_ic.ic_curmode != IEEE80211_MODE_11A) {
+ linkq.retry[0].rate = iwn_ridx_to_plcp[IWN_CCK1];
+ linkq.retry[0].rflags = IWN_RFLAG_CCK;
+ } else
+ linkq.retry[0].rate = iwn_ridx_to_plcp[IWN_OFDM6];
+ linkq.retry[0].rflags |= IWN_RFLAG_ANT(txant);
+ /* Use same bit-rate for all TX retries. */
for (i = 1; i < IWN_MAX_TX_RETRIES; i++) {
- mrr.table[i].rate = mrr.table[0].rate;
- mrr.table[i].rflags = mrr.table[0].rflags;
+ linkq.retry[i].rate = linkq.retry[0].rate;
+ linkq.retry[i].rflags = linkq.retry[0].rflags;
}
- return iwn_cmd(sc, IWN_CMD_NODE_MRR_SETUP, &mrr, sizeof mrr, async);
+ return iwn_cmd(sc, IWN_CMD_LINK_QUALITY, &linkq, sizeof linkq, async);
}
void
@@ -2647,7 +2859,7 @@ iwn_set_led(struct iwn_softc *sc, uint8_t which, uint8_t off, uint8_t on)
IWN_CLRBITS(sc, IWN_LED, IWN_LED_BSM_CTRL);
led.which = which;
- led.unit = htole32(1000); /* on/off in unit of 100ms */
+ led.unit = htole32(10000); /* on/off in unit of 100ms */
led.off = off;
led.on = on;
(void)iwn_cmd(sc, IWN_CMD_SET_LED, &led, sizeof led, 1);
@@ -2858,9 +3070,9 @@ iwn5000_set_txpower(struct iwn_softc *sc, int async)
* for 5000 Series.
*/
memset(&cmd, 0, sizeof cmd);
- cmd.global_limit = 2 * IWN5000_TX_POWER_MAX_DBM; /* 16 dBm */
- cmd.flags = IWN5000_TX_POWER_NO_CLOSED;
- cmd.srv_limit = IWN5000_TX_POWER_AUTO;
+ cmd.global_limit = 2 * IWN5000_TXPOWER_MAX_DBM; /* 16 dBm */
+ cmd.flags = IWN5000_TXPOWER_NO_CLOSED;
+ cmd.srv_limit = IWN5000_TXPOWER_AUTO;
DPRINTF(("setting TX power\n"));
return iwn_cmd(sc, IWN_CMD_TXPOWER_DBM, &cmd, sizeof cmd, async);
}
@@ -2879,11 +3091,11 @@ iwn4965_get_rssi(const struct iwn_rx_stat *stat)
agc = (letoh16(phy->agc) >> 7) & 0x7f;
rssi = 0;
- if (mask & (1 << 0)) /* Ant A */
+ if (mask & IWN_ANT_A)
rssi = MAX(rssi, phy->rssi[0]);
- if (mask & (1 << 1)) /* Ant B */
+ if (mask & IWN_ANT_B)
rssi = MAX(rssi, phy->rssi[2]);
- if (mask & (1 << 2)) /* Ant C */
+ if (mask & IWN_ANT_C)
rssi = MAX(rssi, phy->rssi[4]);
return rssi - agc - IWN_RSSI_TO_DBM;
@@ -3032,9 +3244,9 @@ iwn_collect_noise(struct iwn_softc *sc,
for (i = 0; i < 3; i++)
if (val - calib->rssi[i] <= 15 * 20)
sc->antmsk |= 1 << i;
- /* If neither antenna A nor antenna B are connected... */
- if ((sc->antmsk & (1 << 0 | 1 << 1)) == 0)
- sc->antmsk |= 1 << 1; /* ...mark antenna B as connected! */
+ /* If none of the TX antennas are connected, keep at least one. */
+ if ((sc->antmsk & sc->txantmsk) == 0)
+ sc->antmsk |= IWN_LSB(sc->txantmsk);
(void)hal->set_gains(sc);
calib->state = IWN_CALIB_STATE_RUN;
@@ -3043,40 +3255,41 @@ iwn_collect_noise(struct iwn_softc *sc,
/* XXX Disable RX chains with no antennas connected. */
sc->rxon.rxchain = htole16(IWN_RXCHAIN_SEL(sc->antmsk));
(void)iwn_cmd(sc, IWN_CMD_CONFIGURE, &sc->rxon, hal->rxonsz, 1);
+#endif
- /* XXX Enable power-saving mode. */
- if (ic->ic_flags & IEEE80211_F_PMGTON)
+ /* Enable power-saving mode if requested by user. */
+ if (sc->sc_ic.ic_flags & IEEE80211_F_PMGTON)
(void)iwn_set_pslevel(sc, 0, 3, 1);
-#endif
}
int
iwn4965_init_gains(struct iwn_softc *sc)
{
- struct iwn_phy_calib_cmd cmd;
+ struct iwn_phy_calib_gain cmd;
memset(&cmd, 0, sizeof cmd);
- cmd.code = IWN4965_SET_DIFF_GAIN;
+ cmd.code = IWN4965_PHY_CALIB_DIFF_GAIN;
/* Differential gains initially set to 0 for all 3 antennas. */
DPRINTF(("setting initial differential gains\n"));
- return iwn_cmd(sc, IWN_PHY_CALIB, &cmd, sizeof cmd, 1);
+ return iwn_cmd(sc, IWN_CMD_PHY_CALIB, &cmd, sizeof cmd, 1);
}
int
iwn5000_init_gains(struct iwn_softc *sc)
{
- uint32_t code;
+ struct iwn_phy_calib cmd;
- code = htole32(IWN5000_PHY_CALIB_RESET_NOISE_GAIN);
+ memset(&cmd, 0, sizeof cmd);
+ cmd.code = IWN5000_PHY_CALIB_RESET_NOISE_GAIN;
DPRINTF(("setting initial differential gains\n"));
- return iwn_cmd(sc, IWN_PHY_CALIB, &code, sizeof code, 1);
+ return iwn_cmd(sc, IWN_CMD_PHY_CALIB, &cmd, sizeof cmd, 1);
}
int
iwn4965_set_gains(struct iwn_softc *sc)
{
struct iwn_calib_state *calib = &sc->calib;
- struct iwn_phy_calib_cmd cmd;
+ struct iwn_phy_calib_gain cmd;
int i, delta, noise;
/* Get minimal noise among connected antennas. */
@@ -3086,25 +3299,29 @@ iwn4965_set_gains(struct iwn_softc *sc)
noise = MIN(calib->noise[i], noise);
memset(&cmd, 0, sizeof cmd);
- cmd.code = IWN4965_SET_DIFF_GAIN;
+ cmd.code = IWN4965_PHY_CALIB_DIFF_GAIN;
/* Set differential gains for connected antennas. */
for (i = 0; i < 3; i++) {
if (sc->antmsk & (1 << i)) {
- delta = ((int32_t)calib->noise[i] - noise) / 30;
- cmd.gain[i] = MIN(delta, 3);
- cmd.gain[i] |= 1 << 2;
+ /* Compute attenuation (in unit of 1.5dB). */
+ delta = (noise - (int32_t)calib->noise[i]) / 30;
+ /* NB: delta <= 0 */
+ /* Limit to [-4.5dB,0]. */
+ cmd.gain[i] = MIN(abs(delta), 3);
+ if (delta < 0)
+ cmd.gain[i] |= 1 << 2; /* sign bit */
}
}
DPRINTF(("setting differential gains Ant A/B/C: %x/%x/%x (%x)\n",
cmd.gain[0], cmd.gain[1], cmd.gain[2], sc->antmsk));
- return iwn_cmd(sc, IWN_PHY_CALIB, &cmd, sizeof cmd, 1);
+ return iwn_cmd(sc, IWN_CMD_PHY_CALIB, &cmd, sizeof cmd, 1);
}
int
iwn5000_set_gains(struct iwn_softc *sc)
{
struct iwn_calib_state *calib = &sc->calib;
- struct iwn_phy_calib_cmd cmd;
+ struct iwn_phy_calib_gain cmd;
int i, delta;
memset(&cmd, 0, sizeof cmd);
@@ -3115,7 +3332,7 @@ iwn5000_set_gains(struct iwn_softc *sc)
/* The delta is relative to antenna A. */
delta = ((int32_t)calib->noise[0] -
(int32_t)calib->noise[i]) / 30;
- /* Limit differential gain to [-3,+3]. */
+ /* Limit to [-4.5dB,+4.5dB]. */
cmd.gain[i] = MIN(abs(delta), 3);
if (delta < 0)
cmd.gain[i] |= 1 << 2; /* sign bit */
@@ -3123,7 +3340,7 @@ iwn5000_set_gains(struct iwn_softc *sc)
}
DPRINTF(("setting differential gains Ant A/B/C: %x/%x/%x (%x)\n",
cmd.gain[0], cmd.gain[1], cmd.gain[2], sc->antmsk));
- return iwn_cmd(sc, IWN_PHY_CALIB, &cmd, sizeof cmd, 1);
+ return iwn_cmd(sc, IWN_CMD_PHY_CALIB, &cmd, sizeof cmd, 1);
}
/*
@@ -3265,7 +3482,7 @@ iwn_tune_sensitivity(struct iwn_softc *sc, const struct iwn_rx_stats *stats)
calib->noise_ref = noise_ref;
if (calib->cck_state == IWN_CCK_STATE_HIFA) {
- /* previous interval had many false alarms */
+ /* Previous interval had many false alarms. */
dec(calib->energy_cck, 8, energy_min);
}
calib->cck_state = IWN_CCK_STATE_INIT;
@@ -3305,7 +3522,7 @@ iwn_send_sensitivity(struct iwn_softc *sc)
calib->ofdm_x1, calib->ofdm_mrc_x1, calib->ofdm_x4,
calib->ofdm_mrc_x4, calib->cck_x4, calib->cck_mrc_x4,
calib->energy_cck));
- return iwn_cmd(sc, IWN_SENSITIVITY, &cmd, sizeof cmd, 1);
+ return iwn_cmd(sc, IWN_CMD_SET_SENSITIVITY, &cmd, sizeof cmd, 1);
}
/*
@@ -3370,12 +3587,10 @@ iwn_config(struct iwn_softc *sc)
struct ieee80211com *ic = &sc->sc_ic;
struct ifnet *ifp = &ic->ic_if;
struct iwn_bluetooth bluetooth;
- struct iwn_node_info node;
uint16_t rxchain;
- uint8_t rate;
int error;
- /* Set power-saving level to CAM during initialization. */
+ /* Set power saving level to CAM during initialization. */
if ((error = iwn_set_pslevel(sc, 0, 0, 0)) != 0) {
printf("%s: could not set power saving level\n",
sc->sc_dev.dv_xname);
@@ -3410,17 +3625,6 @@ iwn_config(struct iwn_softc *sc)
sc->rxon.mode = IWN_MODE_STA;
sc->rxon.filter = htole32(IWN_FILTER_MULTICAST);
break;
-#ifndef IEEE80211_STA_ONLY
-#ifdef notyet
- case IEEE80211_M_IBSS:
- case IEEE80211_M_AHDEMO:
- sc->rxon.mode = IWN_MODE_IBSS;
- break;
- case IEEE80211_M_HOSTAP:
- sc->rxon.mode = IWN_MODE_HOSTAP;
- break;
-#endif
-#endif
case IEEE80211_M_MONITOR:
sc->rxon.mode = IWN_MODE_MONITOR;
sc->rxon.filter = htole32(IWN_FILTER_MULTICAST |
@@ -3450,24 +3654,11 @@ iwn_config(struct iwn_softc *sc)
return error;
}
- /* Add broadcast node. */
- memset(&node, 0, sizeof node);
- IEEE80211_ADDR_COPY(node.macaddr, etherbroadcastaddr);
- node.id = hal->broadcast_id;
- DPRINTF(("adding broadcast node\n"));
- error = hal->add_node(sc, &node, 0);
- if (error != 0) {
+ if ((error = iwn_add_broadcast_node(sc, 0)) != 0) {
printf("%s: could not add broadcast node\n",
sc->sc_dev.dv_xname);
return error;
}
- DPRINTF(("setting fixed rate for node %d\n", node.id));
- rate = (ic->ic_curmode == IEEE80211_MODE_11A) ? 12 : 2;
- if ((error = iwn_set_fixed_rate(sc, node.id, rate, 0)) != 0) {
- printf("%s: could not setup MRR for node %d\n",
- sc->sc_dev.dv_xname, node.id);
- return error;
- }
if ((error = iwn_set_critical_temp(sc)) != 0) {
printf("%s: could not set critical temperature\n",
@@ -3477,60 +3668,32 @@ iwn_config(struct iwn_softc *sc)
return 0;
}
-/*
- * Send a scan request to the firmware. Since this command is huge, we map
- * it into an mbuf instead of using the pre-allocated set of commands.
- */
int
iwn_scan(struct iwn_softc *sc, uint16_t flags)
{
- const struct iwn_hal *hal = sc->sc_hal;
struct ieee80211com *ic = &sc->sc_ic;
- struct iwn_tx_ring *ring = &sc->txq[4];
- struct iwn_tx_desc *desc;
- struct iwn_tx_data *data;
- struct iwn_tx_cmd *cmd;
- struct iwn_cmd_data *tx;
struct iwn_scan_hdr *hdr;
+ struct iwn_cmd_data *tx;
struct iwn_scan_essid *essid;
struct iwn_scan_chan *chan;
struct ieee80211_frame *wh;
struct ieee80211_rateset *rs;
struct ieee80211_channel *c;
- uint8_t *frm;
+ uint8_t *buf, *frm;
uint16_t rxchain;
- int pktlen, error;
+ uint8_t txant;
+ int buflen, error;
- desc = &ring->desc[ring->cur];
- data = &ring->data[ring->cur];
-
- MGETHDR(data->m, M_DONTWAIT, MT_DATA);
- if (data->m == NULL) {
- printf("%s: could not allocate mbuf for scan command\n",
+ buf = malloc(IWN_SCAN_MAXSZ, M_DEVBUF, M_NOWAIT | M_ZERO);
+ if (buf == NULL) {
+ printf("%s: could not allocate buffer for scan command\n",
sc->sc_dev.dv_xname);
return ENOMEM;
}
- MCLGET(data->m, M_DONTWAIT);
- if (!(data->m->m_flags & M_EXT)) {
- m_freem(data->m);
- data->m = NULL;
- printf("%s: could not allocate mbuf for scan command\n",
- sc->sc_dev.dv_xname);
- return ENOMEM;
- }
-
- cmd = mtod(data->m, struct iwn_tx_cmd *);
- cmd->code = IWN_CMD_SCAN;
- cmd->flags = 0;
- cmd->qid = ring->qid;
- cmd->idx = ring->cur;
-
- hdr = (struct iwn_scan_hdr *)cmd->data;
- memset(hdr, 0, sizeof (struct iwn_scan_hdr));
+ hdr = (struct iwn_scan_hdr *)buf;
/*
- * Move to the next channel if no packets are received within 10 msecs
- * after sending the probe request (this helps to reduce the duration
- * of active scans).
+ * Move to the next channel if no frames are received within 10ms
+ * after sending the probe request.
*/
hdr->quiet_time = htole16(10); /* timeout in milliseconds */
hdr->quiet_threshold = htole16(1); /* min # of packets */
@@ -3538,18 +3701,18 @@ iwn_scan(struct iwn_softc *sc, uint16_t flags)
/* Select antennas for scanning. */
rxchain = IWN_RXCHAIN_FORCE | IWN_RXCHAIN_VALID(IWN_ANT_ABC) |
IWN_RXCHAIN_MIMO(IWN_ANT_ABC);
- if (sc->hw_type == IWN_HW_REV_TYPE_4965) {
- /* Ant A must be avoided because of an HW bug in 5GHz. */
+ if ((flags & IEEE80211_CHAN_5GHZ) &&
+ sc->hw_type == IWN_HW_REV_TYPE_4965) {
+ /* Ant A must be avoided in 5GHz because of an HW bug. */
rxchain |= IWN_RXCHAIN_SEL(IWN_ANT_B | IWN_ANT_C);
} else /* Use all available RX antennas. */
- rxchain |= IWN_RXCHAIN_SEL((1 << sc->nrxchains) - 1);
+ rxchain |= IWN_RXCHAIN_SEL(IWN_ANT_ABC);
hdr->rxchain = htole16(rxchain);
hdr->filter = htole32(IWN_FILTER_MULTICAST | IWN_FILTER_BEACON);
tx = (struct iwn_cmd_data *)(hdr + 1);
- memset(tx, 0, sizeof (struct iwn_cmd_data));
tx->flags = htole32(IWN_TX_AUTO_SEQ);
- tx->id = hal->broadcast_id;
+ tx->id = sc->sc_hal->broadcast_id;
tx->lifetime = htole32(IWN_LIFETIME_INFINITE);
if (flags & IEEE80211_CHAN_5GHZ) {
@@ -3564,19 +3727,21 @@ iwn_scan(struct iwn_softc *sc, uint16_t flags)
tx->rflags = IWN_RFLAG_CCK;
rs = &ic->ic_sup_rates[IEEE80211_MODE_11G];
}
- tx->rflags |= IWN_RFLAG_ANT_B;
+ /* Use the first valid TX antenna. */
+ txant = IWN_LSB(sc->txantmsk);
+ tx->rflags |= IWN_RFLAG_ANT(txant);
essid = (struct iwn_scan_essid *)(tx + 1);
- memset(essid, 0, 20 * sizeof (struct iwn_scan_essid));
- essid[0].id = IEEE80211_ELEMID_SSID;
- essid[0].len = ic->ic_des_esslen;
- memcpy(essid[0].data, ic->ic_des_essid, ic->ic_des_esslen);
-
+ if (ic->ic_des_esslen != 0) {
+ essid[0].id = IEEE80211_ELEMID_SSID;
+ essid[0].len = ic->ic_des_esslen;
+ memcpy(essid[0].data, ic->ic_des_essid, ic->ic_des_esslen);
+ }
/*
* Build a probe request frame. Most of the following code is a
* copy & paste of what is done in net80211.
*/
- wh = (struct ieee80211_frame *)&essid[20];
+ wh = (struct ieee80211_frame *)(essid + 20);
wh->i_fc[0] = IEEE80211_FC0_VERSION_0 | IEEE80211_FC0_TYPE_MGT |
IEEE80211_FC0_SUBTYPE_PROBE_REQ;
wh->i_fc[1] = IEEE80211_FC1_DIR_NODS;
@@ -3587,7 +3752,7 @@ iwn_scan(struct iwn_softc *sc, uint16_t flags)
*(u_int16_t *)&wh->i_seq[0] = 0; /* filled by HW */
frm = (uint8_t *)(wh + 1);
- frm = ieee80211_add_ssid(frm, ic->ic_des_essid, ic->ic_des_esslen);
+ frm = ieee80211_add_ssid(frm, NULL, 0);
frm = ieee80211_add_rates(frm, rs);
if (rs->rs_nrates > IEEE80211_RATE_SIZE)
frm = ieee80211_add_xrates(frm, rs);
@@ -3602,20 +3767,20 @@ iwn_scan(struct iwn_softc *sc, uint16_t flags)
continue;
chan->chan = htole16(ieee80211_chan2ieee(ic, c));
- chan->type = 0;
- if (!(c->ic_flags & IEEE80211_CHAN_PASSIVE)) {
- chan->type |= IWN_CHAN_ACTIVE;
- if (ic->ic_des_esslen != 0)
- chan->type |= IWN_CHAN_DIRECT;
- }
+ DPRINTFN(2, ("adding channel %d\n", chan->chan));
+ chan->flags = 0;
+ if (!(c->ic_flags & IEEE80211_CHAN_PASSIVE))
+ chan->flags |= htole32(IWN_CHAN_ACTIVE);
+ if (ic->ic_des_esslen != 0)
+ chan->flags |= htole32(IWN_CHAN_NPBREQS(1));
chan->dsp_gain = 0x6e;
if (IEEE80211_IS_CHAN_5GHZ(c)) {
chan->rf_gain = 0x3b;
- chan->active = htole16(10);
+ chan->active = htole16(24);
chan->passive = htole16(110);
} else {
chan->rf_gain = 0x28;
- chan->active = htole16(20);
+ chan->active = htole16(36);
chan->passive = htole16(120);
}
hdr->nchan++;
@@ -3624,31 +3789,13 @@ iwn_scan(struct iwn_softc *sc, uint16_t flags)
frm += sizeof (struct iwn_scan_chan);
}
- hdr->len = htole16(frm - (uint8_t *)hdr);
- pktlen = frm - (uint8_t *)cmd;
+ buflen = frm - (uint8_t *)hdr;
+ hdr->len = htole16(buflen);
- error = bus_dmamap_load(sc->sc_dmat, data->map, cmd, pktlen, NULL,
- BUS_DMA_NOWAIT);
- if (error != 0) {
- printf("%s: could not map scan command\n",
- sc->sc_dev.dv_xname);
- m_freem(data->m);
- data->m = NULL;
- return error;
- }
-
- IWN_SET_DESC_NSEGS(desc, 1);
- IWN_SET_DESC_SEG(desc, 0, data->map->dm_segs[0].ds_addr,
- data->map->dm_segs[0].ds_len);
-
- /* Update TX scheduler. */
- hal->update_sched(sc, ring->qid, ring->cur, 0, 0);
-
- /* Kick command ring. */
- ring->cur = (ring->cur + 1) % IWN_TX_RING_COUNT;
- IWN_WRITE(sc, IWN_HBUS_TARG_WRPTR, ring->qid << 8 | ring->cur);
-
- return 0;
+ DPRINTF(("sending scan command nchan=%d\n", hdr->nchan));
+ error = iwn_cmd(sc, IWN_CMD_SCAN, buf, buflen, 1);
+ free(buf, M_DEVBUF);
+ return error;
}
int
@@ -3657,8 +3804,6 @@ iwn_auth(struct iwn_softc *sc)
const struct iwn_hal *hal = sc->sc_hal;
struct ieee80211com *ic = &sc->sc_ic;
struct ieee80211_node *ni = ic->ic_bss;
- struct iwn_node_info node;
- uint8_t rate;
int error;
/* Update adapter's configuration. */
@@ -3697,28 +3842,15 @@ iwn_auth(struct iwn_softc *sc)
printf("%s: could not set TX power\n", sc->sc_dev.dv_xname);
return error;
}
-
/*
- * Reconfiguring clears the adapter's nodes table so we must add
- * the broadcast node again.
+ * Reconfiguring RXON clears the firmware's nodes table so we must
+ * add the broadcast node again.
*/
- memset(&node, 0, sizeof node);
- IEEE80211_ADDR_COPY(node.macaddr, etherbroadcastaddr);
- node.id = hal->broadcast_id;
- DPRINTF(("adding broadcast node\n"));
- error = hal->add_node(sc, &node, 1);
- if (error != 0) {
+ if ((error = iwn_add_broadcast_node(sc, 1)) != 0) {
printf("%s: could not add broadcast node\n",
sc->sc_dev.dv_xname);
return error;
}
- DPRINTF(("setting fixed rate for node %d\n", node.id));
- rate = (ic->ic_curmode == IEEE80211_MODE_11A) ? 12 : 2;
- if ((error = iwn_set_fixed_rate(sc, node.id, rate, 1)) != 0) {
- printf("%s: could not setup MRR for broadcast node\n",
- sc->sc_dev.dv_xname, node.id);
- return error;
- }
return 0;
}
@@ -3732,7 +3864,7 @@ iwn_run(struct iwn_softc *sc)
int error;
if (ic->ic_opmode == IEEE80211_M_MONITOR) {
- /* link LED blinks while monitoring */
+ /* Link LED blinks while monitoring. */
iwn_set_led(sc, IWN_LED_LINK, 5, 5);
return 0;
}
@@ -3765,6 +3897,7 @@ iwn_run(struct iwn_softc *sc)
}
/* Add BSS node. */
+ ((struct iwn_node *)ni)->id = IWN_ID_BSS;
memset(&node, 0, sizeof node);
IEEE80211_ADDR_COPY(node.macaddr, ni->ni_macaddr);
node.id = IWN_ID_BSS;
@@ -3778,17 +3911,15 @@ iwn_run(struct iwn_softc *sc)
printf("%s: could not add BSS node\n", sc->sc_dev.dv_xname);
return error;
}
- DPRINTF(("setting MRR for node %d\n", node.id));
- if ((error = iwn_setup_node_mrr(sc, ni, node.id)) != 0) {
- printf("%s: could not setup MRR for node %d\n",
+ DPRINTF(("setting link quality for node %d\n", node.id));
+ if ((error = iwn_set_link_quality(sc, ni)) != 0) {
+ printf("%s: could not setup link quality for node %d\n",
sc->sc_dev.dv_xname, node.id);
return error;
}
- if (ic->ic_opmode == IEEE80211_M_STA) {
- /* Fake a join to initialize the TX rate. */
- iwn_newassoc(ic, ni, 1);
- }
+ /* Fake a join to initialize the TX rate. */
+ iwn_newassoc(ic, ni, 1);
if ((error = iwn_init_sensitivity(sc)) != 0) {
printf("%s: could not set sensitivity\n",
@@ -3806,38 +3937,168 @@ iwn_run(struct iwn_softc *sc)
}
int
-iwn4965_addba(struct iwn_softc *sc, struct ieee80211_node *ni, int qid,
- int tid, int ssn)
+iwn_set_key(struct ieee80211com *ic, struct ieee80211_node *ni,
+ struct ieee80211_key *k)
{
+ struct iwn_softc *sc = (void *)ic;
+ const struct iwn_hal *hal = sc->sc_hal;
+ struct iwn_node *wn = (void *)ni;
+ struct iwn_node_info node;
+ uint16_t kflags;
+
+ /*
+ * We support CCMP hardware encryption/decryption only. Hardware
+ * support for TKIP really sucks and it is not worth implementing.
+ * We should leave TKIP die anyway.
+ */
+ if (k->k_cipher != IEEE80211_CIPHER_CCMP)
+ return ieee80211_set_key(ic, ni, k);
+
+ kflags = IWN_KFLAG_CCMP | IWN_KFLAG_MAP | IWN_KFLAG_KID(k->k_id);
+ if (k->k_flags & IEEE80211_KEY_GROUP)
+ kflags |= IWN_KFLAG_GROUP;
+
+ memset(&node, 0, sizeof node);
+ node.id = (k->k_flags & IEEE80211_KEY_GROUP) ?
+ hal->broadcast_id : wn->id;
+ node.control = IWN_NODE_UPDATE;
+ node.flags = IWN_FLAG_SET_KEY;
+ node.kflags = htole16(kflags);
+ node.kid = k->k_id;
+ memcpy(node.key, k->k_key, k->k_len);
+ return hal->add_node(sc, &node, 1);
+}
+
+void
+iwn_delete_key(struct ieee80211com *ic, struct ieee80211_node *ni,
+ struct ieee80211_key *k)
+{
+ struct iwn_softc *sc = (void *)ic;
+ const struct iwn_hal *hal = sc->sc_hal;
+ struct iwn_node *wn = (void *)ni;
+ struct iwn_node_info node;
+
+ if (k->k_cipher != IEEE80211_CIPHER_CCMP) {
+ /* See comment about other ciphers above. */
+ ieee80211_delete_key(ic, ni, k);
+ return;
+ }
+ if (ic->ic_state != IEEE80211_S_RUN)
+ return; /* Nothing to do. */
+ memset(&node, 0, sizeof node);
+ node.id = (k->k_flags & IEEE80211_KEY_GROUP) ?
+ hal->broadcast_id : wn->id;
+ node.control = IWN_NODE_UPDATE;
+ node.flags = IWN_FLAG_SET_KEY;
+ node.kflags = htole16(IWN_KFLAG_INVALID);
+ node.kid = 0xff;
+ (void)hal->add_node(sc, &node, 1);
+}
+
+/*
+ * This function is called by upper layer when a ADDBA request is received
+ * from another STA and before the ADDBA response is sent.
+ */
+int
+iwn_ampdu_rx_start(struct ieee80211com *ic, struct ieee80211_node *ni,
+ uint8_t tid, uint16_t ssn)
+{
+ struct iwn_softc *sc = (void *)ic;
+ struct iwn_node *wn = (void *)ni;
+ struct iwn_node_info node;
+
+ memset(&node, 0, sizeof node);
+ node.id = wn->id;
+ node.control = IWN_NODE_UPDATE;
+ node.flags = IWN_FLAG_SET_ADDBA;
+ node.addba_tid = tid;
+ node.addba_ssn = htole16(ssn);
+ DPRINTFN(2, ("ADDBA RA=%d TID=%d SSN=%d\n", wn->id, tid, ssn));
+ return sc->sc_hal->add_node(sc, &node, 1);
+}
+
+/*
+ * This function is called by upper layer on teardown of an HT-immediate
+ * Block Ack (eg. uppon receipt of a DELBA frame.)
+ */
+void
+iwn_ampdu_rx_stop(struct ieee80211com *ic, struct ieee80211_node *ni,
+ uint8_t tid, uint16_t ssn)
+{
+ struct iwn_softc *sc = (void *)ic;
+ struct iwn_node *wn = (void *)ni;
+ struct iwn_node_info node;
+
+ memset(&node, 0, sizeof node);
+ node.id = wn->id;
+ node.control = IWN_NODE_UPDATE;
+ node.flags = IWN_FLAG_SET_DELBA;
+ node.delba_tid = tid;
+ DPRINTFN(2, ("DELBA RA=%d TID=%d\n", wn->id, tid));
+ (void)sc->sc_hal->add_node(sc, &node, 1);
+}
+
+/*
+ * This function is called by upper layer when a ADDBA response is received
+ * from another STA.
+ */
+int
+iwn_ampdu_tx_start(struct ieee80211com *ic, struct ieee80211_node *ni,
+ uint8_t tid, uint16_t ssn)
+{
+ struct iwn_softc *sc = (void *)ic;
const struct iwn_hal *hal = sc->sc_hal;
struct iwn_node *wn = (void *)ni;
struct iwn_node_info node;
- uint32_t tmp;
int error;
- /* Enable TID for RA. */
+ /* Enable TX for the specified RA/TID. */
wn->disable_tid &= ~(1 << tid);
memset(&node, 0, sizeof node);
+ node.id = wn->id;
node.control = IWN_NODE_UPDATE;
node.flags = IWN_FLAG_SET_DISABLE_TID;
node.disable_tid = htole16(wn->disable_tid);
- if ((error = hal->add_node(sc, &node, 1)) != 0)
+ error = hal->add_node(sc, &node, 1);
+ if (error != 0)
return error;
if ((error = iwn_nic_lock(sc)) != 0)
return error;
+ hal->ampdu_tx_start(sc, ni, tid, ssn);
+ iwn_nic_unlock(sc);
+ return 0;
+}
+
+void
+iwn_ampdu_tx_stop(struct ieee80211com *ic, struct ieee80211_node *ni,
+ uint8_t tid, uint16_t ssn)
+{
+ struct iwn_softc *sc = (void *)ic;
+
+ if (iwn_nic_lock(sc) != 0)
+ return;
+ sc->sc_hal->ampdu_tx_stop(sc, tid, ssn);
+ iwn_nic_unlock(sc);
+}
+
+void
+iwn4965_ampdu_tx_start(struct iwn_softc *sc, struct ieee80211_node *ni,
+ uint8_t tid, uint16_t ssn)
+{
+ struct iwn_node *wn = (void *)ni;
+ int qid = 7 + tid;
/* Stop TX scheduler while we're changing its configuration. */
iwn_prph_write(sc, IWN4965_SCHED_QUEUE_STATUS(qid),
IWN4965_TXQ_STATUS_CHGACT);
- /* Assign (RA,TID) translation to the queue. */
+ /* Assign RA/TID translation to the queue. */
iwn_mem_write_2(sc, sc->sched_base + IWN4965_SCHED_TRANS_TBL(qid),
wn->id << 4 | tid);
/* Enable chain mode for the queue. */
- tmp = iwn_prph_read(sc, IWN4965_SCHED_QCHAIN_SEL);
- iwn_prph_write(sc, IWN4965_SCHED_QCHAIN_SEL, tmp | 1 << qid);
+ iwn_prph_setbits(sc, IWN4965_SCHED_QCHAIN_SEL, 1 << qid);
/* Set starting sequence number from the ADDBA request. */
IWN_WRITE(sc, IWN_HBUS_TARG_WRPTR, ssn);
@@ -3851,26 +4112,18 @@ iwn4965_addba(struct iwn_softc *sc, struct ieee80211_node *ni, int qid,
IWN_SCHED_LIMIT << 16);
/* Enable interrupts for the queue. */
- tmp = iwn_prph_read(sc, IWN4965_SCHED_INTR_MASK);
- iwn_prph_write(sc, IWN4965_SCHED_INTR_MASK, tmp | 1 << qid);
+ iwn_prph_setbits(sc, IWN4965_SCHED_INTR_MASK, 1 << qid);
/* Mark the queue as active. */
- iwn_prph_write(sc, IWN5000_SCHED_QUEUE_STATUS(qid),
+ iwn_prph_write(sc, IWN4965_SCHED_QUEUE_STATUS(qid),
IWN4965_TXQ_STATUS_ACTIVE | IWN4965_TXQ_STATUS_AGGR_ENA |
iwn_tid2fifo[tid] << 1);
-
- iwn_nic_unlock(sc);
- return 0;
}
-int
-iwn4965_delba(struct iwn_softc *sc, int qid, int tid, int ssn)
+void
+iwn4965_ampdu_tx_stop(struct iwn_softc *sc, uint8_t tid, uint16_t ssn)
{
- uint32_t tmp;
- int error;
-
- if ((error = iwn_nic_lock(sc)) != 0)
- return error;
+ int qid = 7 + tid;
/* Stop TX scheduler while we're changing its configuration. */
iwn_prph_write(sc, IWN4965_SCHED_QUEUE_STATUS(qid),
@@ -3881,54 +4134,33 @@ iwn4965_delba(struct iwn_softc *sc, int qid, int tid, int ssn)
iwn_prph_write(sc, IWN4965_SCHED_QUEUE_RDPTR(qid), ssn);
/* Disable interrupts for the queue. */
- tmp = iwn_prph_read(sc, IWN4965_SCHED_INTR_MASK);
- iwn_prph_write(sc, IWN4965_SCHED_INTR_MASK, tmp & ~(1 << qid));
+ iwn_prph_clrbits(sc, IWN4965_SCHED_INTR_MASK, 1 << qid);
/* Mark the queue as inactive. */
iwn_prph_write(sc, IWN4965_SCHED_QUEUE_STATUS(qid),
IWN4965_TXQ_STATUS_INACTIVE | iwn_tid2fifo[tid] << 1);
-
- iwn_nic_unlock(sc);
- return 0;
}
-int
-iwn5000_addba(struct iwn_softc *sc, struct ieee80211_node *ni, int qid,
- int tid, int ssn)
+void
+iwn5000_ampdu_tx_start(struct iwn_softc *sc, struct ieee80211_node *ni,
+ uint8_t tid, uint16_t ssn)
{
- const struct iwn_hal *hal = sc->sc_hal;
struct iwn_node *wn = (void *)ni;
- struct iwn_node_info node;
- uint32_t tmp;
- int error;
-
- /* Enable TID for RA. */
- wn->disable_tid &= ~(1 << tid);
- memset(&node, 0, sizeof node);
- node.control = IWN_NODE_UPDATE;
- node.flags = IWN_FLAG_SET_DISABLE_TID;
- node.disable_tid = htole16(wn->disable_tid);
- if ((error = hal->add_node(sc, &node, 1)) != 0)
- return error;
-
- if ((error = iwn_nic_lock(sc)) != 0)
- return error;
+ int qid = 10 + tid;
/* Stop TX scheduler while we're changing its configuration. */
iwn_prph_write(sc, IWN5000_SCHED_QUEUE_STATUS(qid),
IWN5000_TXQ_STATUS_CHGACT);
- /* Assign (RA,TID) translation to the queue. */
+ /* Assign RA/TID translation to the queue. */
iwn_mem_write_2(sc, sc->sched_base + IWN5000_SCHED_TRANS_TBL(qid),
wn->id << 4 | tid);
/* Enable chain mode for the queue. */
- tmp = iwn_prph_read(sc, IWN5000_SCHED_QCHAIN_SEL);
- iwn_prph_write(sc, IWN5000_SCHED_QCHAIN_SEL, tmp | 1 << qid);
+ iwn_prph_setbits(sc, IWN5000_SCHED_QCHAIN_SEL, 1 << qid);
/* Enable aggregation for the queue. */
- tmp = iwn_prph_read(sc, IWN5000_SCHED_AGGR_SEL);
- iwn_prph_write(sc, IWN5000_SCHED_AGGR_SEL, tmp | 1 << qid);
+ iwn_prph_setbits(sc, IWN5000_SCHED_AGGR_SEL, 1 << qid);
/* Set starting sequence number from the ADDBA request. */
IWN_WRITE(sc, IWN_HBUS_TARG_WRPTR, ssn);
@@ -3939,48 +4171,35 @@ iwn5000_addba(struct iwn_softc *sc, struct ieee80211_node *ni, int qid,
IWN_SCHED_LIMIT << 16 | IWN_SCHED_WINSZ);
/* Enable interrupts for the queue. */
- tmp = iwn_prph_read(sc, IWN5000_SCHED_INTR_MASK);
- iwn_prph_write(sc, IWN5000_SCHED_INTR_MASK, tmp | 1 << qid);
+ iwn_prph_setbits(sc, IWN5000_SCHED_INTR_MASK, 1 << qid);
/* Mark the queue as active. */
iwn_prph_write(sc, IWN5000_SCHED_QUEUE_STATUS(qid),
IWN5000_TXQ_STATUS_ACTIVE | iwn_tid2fifo[tid]);
-
- iwn_nic_unlock(sc);
- return 0;
}
-int
-iwn5000_delba(struct iwn_softc *sc, int qid, int tid, int ssn)
+void
+iwn5000_ampdu_tx_stop(struct iwn_softc *sc, uint8_t tid, uint16_t ssn)
{
- uint32_t tmp;
- int error;
-
- if ((error = iwn_nic_lock(sc)) != 0)
- return error;
+ int qid = 10 + tid;
/* Stop TX scheduler while we're changing its configuration. */
iwn_prph_write(sc, IWN5000_SCHED_QUEUE_STATUS(qid),
IWN5000_TXQ_STATUS_CHGACT);
/* Disable aggregation for the queue. */
- tmp = iwn_prph_read(sc, IWN5000_SCHED_AGGR_SEL);
- iwn_prph_write(sc, IWN5000_SCHED_AGGR_SEL, tmp & ~(1 << qid));
+ iwn_prph_clrbits(sc, IWN5000_SCHED_AGGR_SEL, 1 << qid);
/* Set starting sequence number from the ADDBA request. */
IWN_WRITE(sc, IWN_HBUS_TARG_WRPTR, ssn);
iwn_prph_write(sc, IWN5000_SCHED_QUEUE_RDPTR(qid), ssn);
/* Disable interrupts for the queue. */
- tmp = iwn_prph_read(sc, IWN5000_SCHED_INTR_MASK);
- iwn_prph_write(sc, IWN5000_SCHED_INTR_MASK, tmp & ~(1 << qid));
+ iwn_prph_clrbits(sc, IWN5000_SCHED_INTR_MASK, 1 << qid);
/* Mark the queue as inactive. */
iwn_prph_write(sc, IWN5000_SCHED_QUEUE_STATUS(qid),
IWN5000_TXQ_STATUS_INACTIVE | iwn_tid2fifo[tid]);
-
- iwn_nic_unlock(sc);
- return 0;
}
/*
@@ -4021,7 +4240,7 @@ iwn5000_send_calibration(struct iwn_softc *sc)
continue; /* No results available. */
DPRINTF(("send calibration result idx=%d len=%d\n",
idx, sc->calibcmd[idx].len));
- error = iwn_cmd(sc, IWN_PHY_CALIB, sc->calibcmd[idx].buf,
+ error = iwn_cmd(sc, IWN_CMD_PHY_CALIB, sc->calibcmd[idx].buf,
sc->calibcmd[idx].len, 0);
if (error != 0) {
printf("%s: could not send calibration result\n",
@@ -4034,7 +4253,7 @@ iwn5000_send_calibration(struct iwn_softc *sc)
/*
* This function is called after the runtime firmware notifies us of its
- * readiness (called in a process context).
+ * readiness (called in a process context.)
*/
int
iwn4965_post_alive(struct iwn_softc *sc)
@@ -4084,7 +4303,7 @@ iwn4965_post_alive(struct iwn_softc *sc)
/*
* This function is called after the initialization or runtime firmware
- * notifies us of its readiness (called in a process context).
+ * notifies us of its readiness (called in a process context.)
*/
int
iwn5000_post_alive(struct iwn_softc *sc)
@@ -4127,31 +4346,31 @@ iwn5000_post_alive(struct iwn_softc *sc)
for (qid = 0; qid < 7; qid++) {
static uint8_t qid2fifo[] = { 3, 2, 1, 0, 7, 5, 6 };
iwn_prph_write(sc, IWN5000_SCHED_QUEUE_STATUS(qid),
- IWN5000_TXQ_STATUS_ACTIVE | qid2fifo[qid] << 0);
+ IWN5000_TXQ_STATUS_ACTIVE | qid2fifo[qid]);
}
iwn_nic_unlock(sc);
- /* Configure WiMax (IEEE 802.16) coexistence. */
+ /* Configure WiMAX (IEEE 802.16e) coexistence. */
memset(&wimax, 0, sizeof wimax);
- DPRINTF(("Configuring WiMax coexistence\n"));
+ DPRINTF(("Configuring WiMAX coexistence\n"));
error = iwn_cmd(sc, IWN5000_CMD_WIMAX_COEX, &wimax, sizeof wimax, 0);
if (error != 0) {
- printf("%s: could not configure WiMax coexistence\n",
+ printf("%s: could not configure WiMAX coexistence\n",
sc->sc_dev.dv_xname);
return error;
}
if (sc->hw_type != IWN_HW_REV_TYPE_5150) {
- struct iwn5000_phy_calib calib;
+ struct iwn5000_phy_calib_crystal cmd;
/* Perform crystal calibration. */
- memset(&calib, 0, sizeof calib);
- calib.code = IWN5000_PHY_CALIB_CRYSTAL;
- calib.data[0] = letoh32(sc->eeprom_crystal) & 0xff;
- calib.data[1] = (letoh32(sc->eeprom_crystal) >> 16) & 0xff;
+ memset(&cmd, 0, sizeof cmd);
+ cmd.code = IWN5000_PHY_CALIB_CRYSTAL;
+ cmd.data[0] = letoh32(sc->eeprom_crystal) & 0xff;
+ cmd.data[1] = (letoh32(sc->eeprom_crystal) >> 16) & 0xff;
DPRINTF(("sending crystal calibration %d, %d\n",
- calib.data[0], calib.data[1]));
- error = iwn_cmd(sc, IWN_PHY_CALIB, &calib, sizeof calib, 0);
+ cmd.data[0], cmd.data[1]));
+ error = iwn_cmd(sc, IWN_CMD_PHY_CALIB, &cmd, sizeof cmd, 0);
if (error != 0) {
printf("%s: crystal calibration failed\n",
sc->sc_dev.dv_xname);
@@ -4234,12 +4453,12 @@ iwn4965_load_firmware(struct iwn_softc *sc)
struct iwn_dma_info *dma = &sc->fw_dma;
int error;
- /* Copy initialization images into pre-allocated DMA-safe memory. */
+ /* Copy initialization sections into pre-allocated DMA-safe memory. */
memcpy(dma->vaddr, fw->init.data, fw->init.datasz);
memcpy(dma->vaddr + IWN4965_FW_DATA_MAXSZ,
fw->init.text, fw->init.textsz);
- /* Tell adapter where to find initialization images. */
+ /* Tell adapter where to find initialization sections. */
if ((error = iwn_nic_lock(sc)) != 0)
return error;
iwn_prph_write(sc, IWN_BSM_DRAM_DATA_ADDR, dma->paddr >> 4);
@@ -4289,9 +4508,6 @@ iwn4965_load_firmware(struct iwn_softc *sc)
return 0;
}
-/*
- * Load a firmware text or data section through Flow Handler DMA.
- */
int
iwn5000_load_firmware_section(struct iwn_softc *sc, uint32_t dst,
const uint8_t *section, int size)
@@ -4343,7 +4559,6 @@ iwn5000_load_firmware(struct iwn_softc *sc)
sc->sc_dev.dv_xname, ".text");
return error;
}
-
error = iwn5000_load_firmware_section(sc, IWN_FW_DATA_BASE,
fw->data, fw->datasz);
if (error != 0) {
@@ -4408,7 +4623,7 @@ iwn_read_firmware(struct iwn_softc *sc)
return EINVAL;
}
- /* get pointers to firmware sections */
+ /* Get pointers to firmware sections. */
fw->main.text = (const uint8_t *)(hdr + 1);
fw->main.data = fw->main.text + fw->main.textsz;
fw->init.text = fw->main.data + fw->main.datasz;
@@ -4440,7 +4655,6 @@ iwn_clock_wait(struct iwn_softc *sc)
int
iwn4965_apm_init(struct iwn_softc *sc)
{
- uint32_t tmp;
int error;
/* Disable L0s. */
@@ -4457,8 +4671,7 @@ iwn4965_apm_init(struct iwn_softc *sc)
IWN_APMG_CLK_CTRL_DMA_CLK_RQT | IWN_APMG_CLK_CTRL_BSM_CLK_RQT);
DELAY(20);
/* Disable L1. */
- tmp = iwn_prph_read(sc, IWN_APMG_PCI_STT);
- iwn_prph_write(sc, IWN_APMG_PCI_STT, tmp | IWN_APMG_PCI_STT_L1A_DIS);
+ iwn_prph_setbits(sc, IWN_APMG_PCI_STT, IWN_APMG_PCI_STT_L1A_DIS);
iwn_nic_unlock(sc);
return 0;
@@ -4467,7 +4680,6 @@ iwn4965_apm_init(struct iwn_softc *sc)
int
iwn5000_apm_init(struct iwn_softc *sc)
{
- uint32_t tmp;
int error;
/* Disable L0s. */
@@ -4490,8 +4702,7 @@ iwn5000_apm_init(struct iwn_softc *sc)
iwn_prph_write(sc, IWN_APMG_CLK_CTRL, IWN_APMG_CLK_CTRL_DMA_CLK_RQT);
DELAY(20);
/* Disable L1. */
- tmp = iwn_prph_read(sc, IWN_APMG_PCI_STT);
- iwn_prph_write(sc, IWN_APMG_PCI_STT, tmp | IWN_APMG_PCI_STT_L1A_DIS);
+ iwn_prph_setbits(sc, IWN_APMG_PCI_STT, IWN_APMG_PCI_STT_L1A_DIS);
iwn_nic_unlock(sc);
return 0;
@@ -4566,11 +4777,10 @@ iwn4965_nic_config(struct iwn_softc *sc)
int
iwn5000_nic_config(struct iwn_softc *sc)
{
- uint32_t tmp;
pcireg_t reg;
int error;
- /* Retrieve Active State Power Management (ASPM). */
+ /* Retrieve PCIe Active State Power Management (ASPM). */
reg = pci_conf_read(sc->sc_pct, sc->sc_pcitag,
sc->sc_cap_off + PCI_PCIE_LCSR);
if (reg & PCI_PCIE_LCSR_ASPM_L1) /* L1 Entry enabled. */
@@ -4589,8 +4799,7 @@ iwn5000_nic_config(struct iwn_softc *sc)
if ((error = iwn_nic_lock(sc)) != 0)
return error;
- tmp = iwn_prph_read(sc, IWN_APMG_PS);
- iwn_prph_write(sc, IWN_APMG_PS, tmp | IWN_APMG_PS_EARLY_PWROFF_DIS);
+ iwn_prph_setbits(sc, IWN_APMG_PS, IWN_APMG_PS_EARLY_PWROFF_DIS);
iwn_nic_unlock(sc);
return 0;
}
@@ -4599,7 +4808,6 @@ int
iwn_hw_init(struct iwn_softc *sc)
{
const struct iwn_hal *hal = sc->sc_hal;
- uint32_t tmp;
int error, qid;
/* Clear any pending interrupts. */
@@ -4614,10 +4822,7 @@ iwn_hw_init(struct iwn_softc *sc)
/* Select VMAIN power source. */
if ((error = iwn_nic_lock(sc)) != 0)
return error;
- tmp = iwn_prph_read(sc, IWN_APMG_PS);
- tmp &= ~IWN_APMG_PS_PWR_SRC_MASK;
- tmp |= IWN_APMG_PS_PWR_SRC(IWN_APMG_PS_PWR_SRC_VMAIN);
- iwn_prph_write(sc, IWN_APMG_PS, tmp);
+ iwn_prph_clrbits(sc, IWN_APMG_PS, IWN_APMG_PS_PWR_SRC_MASK);
iwn_nic_unlock(sc);
/* Perform adapter-specific initialization. */
diff --git a/sys/dev/pci/if_iwnreg.h b/sys/dev/pci/if_iwnreg.h
index 8f0444b94b0..e17c7e0fb21 100644
--- a/sys/dev/pci/if_iwnreg.h
+++ b/sys/dev/pci/if_iwnreg.h
@@ -1,4 +1,4 @@
-/* $OpenBSD: if_iwnreg.h,v 1.11 2008/10/13 16:37:10 damien Exp $ */
+/* $OpenBSD: if_iwnreg.h,v 1.12 2008/10/22 06:25:07 damien Exp $ */
/*-
* Copyright (c) 2007, 2008
@@ -138,30 +138,30 @@
#define IWN_BSM_DRAM_DATA_SIZE 0x349c
#define IWN_BSM_SRAM_BASE 0x3800
-/* possible values for IWN_APMG_CLK_DIS */
+/* Possible values for IWN_APMG_CLK_DIS. */
#define IWN_APMG_CLK_DMA_RQT (1 << 9)
-/* possible flags for register IWN_HW_IF_CONFIG */
+/* Possible flags for register IWN_HW_IF_CONFIG. */
#define IWN_HW_IF_CONFIG_4965_R (1 << 4)
#define IWN_HW_IF_CONFIG_MAC_SI (1 << 8)
#define IWN_HW_IF_CONFIG_RADIO_SI (1 << 9)
#define IWN_HW_IF_CONFIG_EEPROM_LOCKED (1 << 21)
#define IWN_HW_IF_CONFIG_HAP_WAKE_L1A (1 << 23)
-/* possible flags for registers IWN_PRPH_RADDR/IWN_PRPH_WADDR */
+/* Possible flags for registers IWN_PRPH_RADDR/IWN_PRPH_WADDR. */
#define IWN_PRPH_DWORD ((sizeof (uint32_t) - 1) << 24)
-/* possible values for IWN_BSM_WR_MEM_DST */
+/* Possible values for IWN_BSM_WR_MEM_DST. */
#define IWN_FW_TEXT_BASE 0x00000000
#define IWN_FW_DATA_BASE 0x00800000
-/* possible flags for register IWN_RESET */
+/* Possible flags for register IWN_RESET. */
#define IWN_RESET_NEVO (1 << 0)
#define IWN_RESET_SW (1 << 7)
#define IWN_RESET_MASTER_DISABLED (1 << 8)
#define IWN_RESET_STOP_MASTER (1 << 9)
-/* possible flags for register IWN_GP_CNTRL */
+/* Possible flags for register IWN_GP_CNTRL. */
#define IWN_GP_CNTRL_MAC_ACCESS_ENA (1 << 0)
#define IWN_GP_CNTRL_MAC_CLOCK_READY (1 << 0)
#define IWN_GP_CNTRL_INIT_DONE (1 << 2)
@@ -169,7 +169,7 @@
#define IWN_GP_CNTRL_SLEEP (1 << 4)
#define IWN_GP_CNTRL_RFKILL (1 << 27)
-/* possibles flags for register IWN_HW_REV */
+/* Possible flags for register IWN_HW_REV. */
#define IWN_HW_REV_TYPE_SHIFT 4
#define IWN_HW_REV_TYPE_MASK 0x000000f0
#define IWN_HW_REV_TYPE_4965 0
@@ -178,32 +178,34 @@
#define IWN_HW_REV_TYPE_5150 4
#define IWN_HW_REV_TYPE_5100 5
-/* possible flags for register IWN_GIO_CHICKEN */
+/* Possible flags for register IWN_GIO_CHICKEN. */
#define IWN_GIO_CHICKEN_L1A_NO_L0S_RX (1 << 23)
#define IWN_GIO_CHICKEN_DIS_L0S_TIMER (1 << 29)
-/* possible flags for register IWN_GIO */
+/* Possible flags for register IWN_GIO. */
#define IWN_GIO_L0S_ENA (1 << 1)
-/* possible flags for register IWN_UCODE_GP1_CLR */
+/* Possible flags for register IWN_UCODE_GP1_CLR. */
#define IWN_UCODE_GP1_RFKILL (1 << 1)
#define IWN_UCODE_GP1_CMD_BLOCKED (1 << 2)
#define IWN_UCODE_GP1_CTEMP_STOP_RF (1 << 3)
-/* possible flags for register IWN_LED */
+/* Possible flags/values for register IWN_LED. */
#define IWN_LED_BSM_CTRL (1 << 5)
+#define IWN_LED_OFF 0x00000038
+#define IWN_LED_ON 0x00000078
-/* possible values for register IWN_ANA_PLL */
+/* Possible values for register IWN_ANA_PLL. */
#define IWN_ANA_PLL_INIT 0x00880300
-/* possible flags for register IWN_FH_RX_STATUS */
+/* Possible flags for register IWN_FH_RX_STATUS. */
#define IWN_FH_RX_STATUS_IDLE (1 << 24)
-/* possible flags for register IWN_BSM_WR_CTRL */
+/* Possible flags for register IWN_BSM_WR_CTRL. */
#define IWN_BSM_WR_CTRL_START_EN (1 << 30)
#define IWN_BSM_WR_CTRL_START (1 << 31)
-/* possible flags for register IWN_INT */
+/* Possible flags for register IWN_INT. */
#define IWN_INT_ALIVE (1 << 0)
#define IWN_INT_WAKEUP (1 << 1)
#define IWN_INT_SW_RX (1 << 3)
@@ -214,36 +216,37 @@
#define IWN_INT_HW_ERR (1 << 29)
#define IWN_INT_FH_RX (1 << 31)
+/* Shortcut. */
#define IWN_INT_MASK \
(IWN_INT_SW_ERR | IWN_INT_HW_ERR | IWN_INT_FH_TX | \
IWN_INT_FH_RX | IWN_INT_ALIVE | IWN_INT_WAKEUP | \
IWN_INT_SW_RX | IWN_INT_CT_REACHED | IWN_INT_RF_TOGGLED)
-/* possible flags for register IWN_FH_INT */
+/* Possible flags for register IWN_FH_INT. */
#define IWN_FH_INT_TX_CHNL(x) (1 << (x))
#define IWN_FH_INT_RX_CHNL(x) (1 << ((x) + 16))
#define IWN_FH_INT_HI_PRIOR (1 << 30)
-/* shortcuts for the above */
+/* Shortcuts for the above. */
#define IWN_FH_INT_TX \
(IWN_FH_INT_TX_CHNL(0) | IWN_FH_INT_TX_CHNL(1))
#define IWN_FH_INT_RX \
(IWN_FH_INT_RX_CHNL(0) | IWN_FH_INT_RX_CHNL(1) | IWN_FH_INT_HI_PRIOR)
-/* possible flags/values for register IWN_FH_TX_CONFIG */
+/* Possible flags/values for register IWN_FH_TX_CONFIG. */
#define IWN_FH_TX_CONFIG_DMA_PAUSE 0
#define IWN_FH_TX_CONFIG_DMA_ENA (1 << 31)
#define IWN_FH_TX_CONFIG_CIRQ_HOST_ENDTFD (1 << 20)
-/* possible values for register IWN_FH_TXBUF_STATUS */
+/* Possible flags/values for register IWN_FH_TXBUF_STATUS. */
#define IWN_FH_TXBUF_STATUS_TBNUM(x) ((x) << 20)
#define IWN_FH_TXBUF_STATUS_TBIDX(x) ((x) << 12)
#define IWN_FH_TXBUF_STATUS_TFBD_VALID 3
-/* possible flags for register IWN_FH_TX_STATUS */
+/* Possible flags for register IWN_FH_TX_STATUS. */
#define IWN_FH_TX_STATUS_IDLE(chnl) \
(1 << ((chnl) + 24) | 1 << ((chnl) + 16))
-/* possible flags for register IWN_FH_RX_CONFIG */
+/* Possible flags for register IWN_FH_RX_CONFIG. */
#define IWN_FH_RX_CONFIG_ENA (1 << 31)
#define IWN_FH_RX_CONFIG_NRBD(x) ((x) << 20)
#define IWN_FH_RX_CONFIG_RB_SIZE_8K (1 << 16)
@@ -251,15 +254,15 @@
#define IWN_FH_RX_CONFIG_RB_TIMEOUT(x) ((x) << 4)
#define IWN_FH_RX_CONFIG_IGN_RXF_EMPTY (1 << 2)
-/* possible flags for register IWN_FH_TX_CONFIG */
+/* Possible flags for register IWN_FH_TX_CONFIG. */
#define IWN_FH_TX_CONFIG_DMA_ENA (1 << 31)
#define IWN_FH_TX_CONFIG_DMA_CREDIT_ENA (1 << 3)
-/* possible flags for register IWN_EEPROM */
+/* Possible flags for register IWN_EEPROM. */
#define IWN_EEPROM_READ_VALID (1 << 0)
#define IWN_EEPROM_CMD (1 << 1)
-/* possible flags for register IWN5000_SCHED_QUEUE_STATUS */
+/* Possible flags for register IWN_SCHED_QUEUE_STATUS. */
#define IWN4965_TXQ_STATUS_ACTIVE 0x0007fc01
#define IWN4965_TXQ_STATUS_INACTIVE 0x0007fc00
#define IWN4965_TXQ_STATUS_AGGR_ENA (1 << 5 | 1 << 8)
@@ -268,20 +271,20 @@
#define IWN5000_TXQ_STATUS_INACTIVE 0x00ff0010
#define IWN5000_TXQ_STATUS_CHGACT (1 << 19)
-/* possible flags for register IWN_APMG_CLK_CTRL */
+/* Possible flags for register IWN_APMG_CLK_CTRL. */
#define IWN_APMG_CLK_CTRL_DMA_CLK_RQT (1 << 9)
#define IWN_APMG_CLK_CTRL_BSM_CLK_RQT (1 << 11)
-/* possible flags for register IWN_APMG_PS */
+/* Possible flags for register IWN_APMG_PS. */
#define IWN_APMG_PS_EARLY_PWROFF_DIS (1 << 22)
#define IWN_APMG_PS_PWR_SRC_MASK (3 << 24)
#define IWN_APMG_PS_PWR_SRC(x) ((x) << 24)
#define IWN_APMG_PS_PWR_SRC_VMAIN 0
-/* possible flags for IWN_APMG_PCI_STT */
+/* Possible flags for IWN_APMG_PCI_STT. */
#define IWN_APMG_PCI_STT_L1A_DIS (1 << 11)
-/* possible flags for register IWN_BSM_DRAM_TEXT_SIZE */
+/* Possible flags for register IWN_BSM_DRAM_TEXT_SIZE. */
#define IWN_FW_UPDATED (1 << 31)
#define IWN_SCHED_WINSZ 64
@@ -292,20 +295,21 @@
#define IWN5000_SCHEDSZ (IWN5000_NTXQUEUES * IWN5000_SCHED_COUNT * 2)
struct iwn_tx_desc {
- uint32_t flags;
+ uint8_t reserved1[3];
+ uint8_t nsegs;
struct {
uint32_t w1;
uint32_t w2;
uint32_t w3;
} __packed segs[IWN_MAX_SCATTER / 2];
- /* pad to 128 bytes */
- uint32_t reserved;
+ /* Pad to 128 bytes. */
+ uint32_t reserved2;
} __packed;
#define IWN_SET_DESC_NSEGS(d, x) \
- (d)->flags = htole32(((x) & 0x1f) << 24)
+ (d)->nsegs = (x)
-/* set a segment physical address and length in a Tx descriptor */
+/* Set a segment physical address and length in a TX descriptor. */
#define IWN_SET_DESC_SEG(d, n, addr, size) do { \
if ((n) & 1) { \
(d)->segs[(n) / 2].w2 |= \
@@ -340,8 +344,8 @@ struct iwn_rx_desc {
#define IWN_BEACON_STATISTICS 157
#define IWN_STATE_CHANGED 161
#define IWN_BEACON_MISSED 162
-#define IWN_AMPDU_RX_START 192
-#define IWN_AMPDU_RX_DONE 193
+#define IWN_RX_PHY 192
+#define IWN_MPDU_RX_DONE 193
#define IWN_RX_DONE 195
uint8_t flags;
@@ -349,10 +353,10 @@ struct iwn_rx_desc {
uint8_t qid;
} __packed;
-/* possible Rx status flags */
+/* Possible RX status flags. */
#define IWN_RX_NO_CRC_ERR (1 << 0)
#define IWN_RX_NO_OVFL_ERR (1 << 1)
-/* shortcut for the above */
+/* Shortcut for the above. */
#define IWN_RX_NOERROR (IWN_RX_NO_CRC_ERR | IWN_RX_NO_OVFL_ERR)
struct iwn_tx_cmd {
@@ -363,7 +367,7 @@ struct iwn_tx_cmd {
#define IWN_CMD_TIMING 20
#define IWN_CMD_ADD_NODE 24
#define IWN_CMD_TX_DATA 28
-#define IWN_CMD_NODE_MRR_SETUP 78
+#define IWN_CMD_LINK_QUALITY 78
#define IWN_CMD_SET_LED 72
#define IWN5000_CMD_WIMAX_COEX 90
#define IWN5000_CMD_CALIB_CONFIG 101
@@ -374,8 +378,8 @@ struct iwn_tx_cmd {
#define IWN_CMD_BT_COEX 155
#define IWN_CMD_GET_STATISTICS 156
#define IWN_CMD_SET_CRITICAL_TEMP 164
-#define IWN_SENSITIVITY 168
-#define IWN_PHY_CALIB 176
+#define IWN_CMD_SET_SENSITIVITY 168
+#define IWN_CMD_PHY_CALIB 176
uint8_t flags;
uint8_t idx;
@@ -383,7 +387,14 @@ struct iwn_tx_cmd {
uint8_t data[136];
} __packed;
-/* structure for command IWN_CMD_CONFIGURE */
+/* Antenna flags, used in various commands. */
+#define IWN_ANT_A (1 << 0)
+#define IWN_ANT_B (1 << 1)
+#define IWN_ANT_C (1 << 2)
+/* Shortcut. */
+#define IWN_ANT_ABC (IWN_ANT_A | IWN_ANT_B | IWN_ANT_C)
+
+/* Structure for command IWN_CMD_CONFIGURE. */
struct iwn_rxon {
uint8_t myaddr[IEEE80211_ADDR_LEN];
uint16_t reserved1;
@@ -399,12 +410,6 @@ struct iwn_rxon {
uint8_t air;
uint16_t rxchain;
-#define IWN_ANT_A (1 << 0)
-#define IWN_ANT_B (1 << 1)
-#define IWN_ANT_C (1 << 2)
-/* Shortcut. */
-#define IWN_ANT_ABC (IWN_ANT_A | IWN_ANT_B | IWN_ANT_C)
-
#define IWN_RXCHAIN_FORCE (1 << 0)
#define IWN_RXCHAIN_VALID(x) ((x) << 1)
#define IWN_RXCHAIN_SEL(x) ((x) << 4)
@@ -413,7 +418,6 @@ struct iwn_rxon {
#define IWN_RXCHAIN_MIMO_COUNT(x) ((x) << 12)
#define IWN_RXCHAIN_MIMO_FORCE (1 << 14)
-
uint8_t ofdm_mask;
uint8_t cck_mask;
uint16_t associd;
@@ -441,7 +445,7 @@ struct iwn_rxon {
uint8_t reserved4;
uint8_t ht_single_mask;
uint8_t ht_dual_mask;
- /* the following fields are for 5000 Series only */
+ /* The following fields are for 5000 Series only. */
uint8_t ht_triple_mask;
uint8_t reserved5;
uint16_t acquisition;
@@ -451,7 +455,7 @@ struct iwn_rxon {
#define IWN4965_RXONSZ (sizeof (struct iwn_rxon) - 6)
#define IWN5000_RXONSZ (sizeof (struct iwn_rxon))
-/* structure for command IWN_CMD_ASSOCIATE */
+/* Structure for command IWN_CMD_ASSOCIATE. */
struct iwn_assoc {
uint32_t flags;
uint32_t filter;
@@ -460,7 +464,7 @@ struct iwn_assoc {
uint16_t reserved;
} __packed;
-/* structure for command IWN_CMD_EDCA_PARAMS */
+/* Structure for command IWN_CMD_EDCA_PARAMS. */
struct iwn_edca_params {
uint32_t flags;
#define IWN_EDCA_UPDATE (1 << 0)
@@ -475,7 +479,7 @@ struct iwn_edca_params {
} __packed ac[EDCA_NUM_AC];
} __packed;
-/* structure for command IWN_CMD_TIMING */
+/* Structure for command IWN_CMD_TIMING. */
struct iwn_cmd_timing {
uint64_t tstamp;
uint16_t bintval;
@@ -485,7 +489,7 @@ struct iwn_cmd_timing {
uint16_t reserved;
} __packed;
-/* structure for command IWN_CMD_ADD_NODE */
+/* Structure for command IWN_CMD_ADD_NODE. */
struct iwn_node_info {
uint8_t control;
#define IWN_NODE_UPDATE (1 << 0)
@@ -502,16 +506,25 @@ struct iwn_node_info {
uint8_t flags;
#define IWN_FLAG_SET_KEY (1 << 0)
#define IWN_FLAG_SET_DISABLE_TID (1 << 1)
+#define IWN_FLAG_SET_TXRATE (1 << 2)
+#define IWN_FLAG_SET_ADDBA (1 << 3)
+#define IWN_FLAG_SET_DELBA (1 << 4)
uint16_t reserved3;
+ uint16_t kflags;
+#define IWN_KFLAG_CCMP (1 << 1)
+#define IWN_KFLAG_MAP (1 << 3)
+#define IWN_KFLAG_KID(kid) ((kid) << 8)
+#define IWN_KFLAG_INVALID (1 << 11)
+#define IWN_KFLAG_GROUP (1 << 14)
- uint16_t keyflags;
uint8_t tsc2; /* TKIP TSC2 */
uint8_t reserved4;
uint16_t ttak[5];
uint8_t kid;
uint8_t reserved5;
uint8_t key[16];
+ /* The following 3 fields are for 5000 Series only. */
uint64_t tsc;
uint8_t rxmic[IEEE80211_TKIP_MICLEN];
uint8_t txmic[IEEE80211_TKIP_MICLEN];
@@ -523,9 +536,8 @@ struct iwn_node_info {
uint32_t mask;
uint16_t disable_tid;
uint16_t reserved6;
-
- uint8_t addba;
- uint8_t delba;
+ uint8_t addba_tid;
+ uint8_t delba_tid;
uint16_t addba_ssn;
uint32_t reserved7;
} __packed;
@@ -538,7 +550,7 @@ struct iwn4965_node_info {
uint8_t id;
uint8_t flags;
uint16_t reserved3;
- uint16_t keyflags;
+ uint16_t kflags;
uint8_t tsc2; /* TKIP TSC2 */
uint8_t reserved4;
uint16_t ttak[5];
@@ -549,22 +561,21 @@ struct iwn4965_node_info {
uint32_t mask;
uint16_t disable_tid;
uint16_t reserved6;
- uint8_t addba;
- uint8_t delba;
+ uint8_t addba_tid;
+ uint8_t delba_tid;
uint16_t addba_ssn;
uint32_t reserved7;
} __packed;
-#define IWN_RFLAG_CCK (1 << 1)
-#define IWN_RFLAG_ANT_A (1 << 6)
-#define IWN_RFLAG_ANT_B (1 << 7)
+#define IWN_RFLAG_CCK (1 << 1)
+#define IWN_RFLAG_ANT(x) ((x) << 6)
-/* structure for command IWN_CMD_TX_DATA */
+/* Structure for command IWN_CMD_TX_DATA. */
struct iwn_cmd_data {
uint16_t len;
uint16_t lnext;
uint32_t flags;
-#define IWN5000_TX_NEED_RTS_CTS (1 << 0) /* 5000 only */
+#define IWN_TX_NEED_PROTECTION (1 << 0) /* 5000 only */
#define IWN_TX_NEED_RTS (1 << 1)
#define IWN_TX_NEED_CTS (1 << 2)
#define IWN_TX_NEED_ACK (1 << 3)
@@ -577,10 +588,7 @@ struct iwn_cmd_data {
#define IWN_TX_INSERT_TSTAMP (1 << 16)
#define IWN_TX_NEED_PADDING (1 << 20)
- uint8_t ntries;
- uint8_t bluetooth;
- uint16_t reserved1;
-
+ uint32_t scratch;
uint8_t rate;
uint8_t rflags;
uint16_t xrflags;
@@ -609,19 +617,19 @@ struct iwn_cmd_data {
uint16_t txop;
} __packed;
-/* structure for command IWN_CMD_MRR_NODE_SETUP */
+/* Structure for command IWN_CMD_LINK_QUALITY. */
#define IWN_MAX_TX_RETRIES 16
-struct iwn_cmd_mrr {
+struct iwn_cmd_link_quality {
uint8_t id;
uint8_t reserved1;
uint16_t ctl;
uint8_t flags;
uint8_t mimo;
- uint8_t ssmask;
- uint8_t dsmask;
+ uint8_t antmsk_1stream;
+ uint8_t antmsk_2stream;
uint8_t ridx[EDCA_NUM_AC];
uint16_t ampdu_limit;
- uint8_t ampdu_disable;
+ uint8_t ampdu_threshold;
uint8_t ampdu_max;
uint32_t reserved2;
struct {
@@ -632,11 +640,11 @@ struct iwn_cmd_mrr {
#define IWN_CCK11 3
#define IWN_OFDM6 4
#define IWN_OFDM54 11
- } __packed table[IWN_MAX_TX_RETRIES];
+ } __packed retry[IWN_MAX_TX_RETRIES];
uint32_t reserved3;
} __packed;
-/* structure for command IWN_CMD_SET_LED */
+/* Structure for command IWN_CMD_SET_LED. */
struct iwn_cmd_led {
uint32_t unit; /* multiplier (in usecs) */
uint8_t which;
@@ -648,7 +656,7 @@ struct iwn_cmd_led {
uint8_t reserved;
} __packed;
-/* structure for command IWN5000_CMD_WIMAX_COEX */
+/* Structure for command IWN5000_CMD_WIMAX_COEX. */
struct iwn5000_wimax_coex {
uint32_t flags;
struct {
@@ -659,7 +667,7 @@ struct iwn5000_wimax_coex {
} __packed events[16];
} __packed;
-/* structures for command IWN5000_CMD_CALIB_CONFIG */
+/* Structures for command IWN5000_CMD_CALIB_CONFIG. */
struct iwn5000_calib_elem {
uint32_t enable;
uint32_t start;
@@ -680,7 +688,7 @@ struct iwn5000_calib_config {
uint32_t reserved;
} __packed;
-/* structure for command IWN_CMD_SET_POWER_MODE */
+/* Structure for command IWN_CMD_SET_POWER_MODE. */
struct iwn_pmgt_cmd {
uint16_t flags;
#define IWN_PS_ALLOW_SLEEP (1 << 0)
@@ -697,7 +705,7 @@ struct iwn_pmgt_cmd {
uint32_t beacons;
} __packed;
-/* structures for command IWN_CMD_SCAN */
+/* Structures for command IWN_CMD_SCAN. */
struct iwn_scan_essid {
uint8_t id;
uint8_t len;
@@ -717,16 +725,16 @@ struct iwn_scan_hdr {
uint32_t flags;
uint32_t filter;
- /* followed by a struct iwn_cmd_data */
- /* followed by an array of 20 struct iwn_scan_essid */
- /* followed by probe request body */
- /* followed by nchan x struct iwn_scan_chan */
+ /* Followed by a struct iwn_cmd_data. */
+ /* Followed by an array of 20 structs iwn_scan_essid. */
+ /* Followed by probe request body. */
+ /* Followed by an array of ``nchan'' structs iwn_scan_chan. */
} __packed;
struct iwn_scan_chan {
- uint32_t type;
-#define IWN_CHAN_ACTIVE (1 << 0)
-#define IWN_CHAN_DIRECT (1 << 1)
+ uint32_t flags;
+#define IWN_CHAN_ACTIVE (1 << 0)
+#define IWN_CHAN_NPBREQS(x) (((1 << (x)) - 1) << 1)
uint16_t chan;
uint8_t rf_gain;
@@ -735,7 +743,10 @@ struct iwn_scan_chan {
uint16_t passive; /* msecs */
} __packed;
-/* structure for command IWN_CMD_TXPOWER */
+/* Maximum size of a scan command. */
+#define IWN_SCAN_MAXSZ (MCLBYTES - 4)
+
+/* Structure for command IWN_CMD_TXPOWER (4965AGN only.) */
#define IWN_RIDX_MAX 32
struct iwn4965_cmd_txpower {
uint8_t band;
@@ -748,20 +759,20 @@ struct iwn4965_cmd_txpower {
} __packed power[IWN_RIDX_MAX + 1];
} __packed;
-/* structure for command IWN_CMD_TXPOWER_DBM */
+/* Structure for command IWN_CMD_TXPOWER_DBM (5000 Series only.) */
struct iwn5000_cmd_txpower {
int8_t global_limit; /* in half-dBm */
-#define IWN5000_TX_POWER_AUTO 0x7f
-#define IWN5000_TX_POWER_MAX_DBM 16
+#define IWN5000_TXPOWER_AUTO 0x7f
+#define IWN5000_TXPOWER_MAX_DBM 16
uint8_t flags;
-#define IWN5000_TX_POWER_NO_CLOSED (1 << 6)
+#define IWN5000_TXPOWER_NO_CLOSED (1 << 6)
int8_t srv_limit; /* in half-dBm */
uint8_t reserved;
} __packed;
-/* structure for command IWN_CMD_BLUETOOTH */
+/* Structure for command IWN_CMD_BLUETOOTH. */
struct iwn_bluetooth {
uint8_t flags;
uint8_t lead;
@@ -771,18 +782,18 @@ struct iwn_bluetooth {
uint32_t cts;
} __packed;
-/* structure for command IWN_CMD_SET_CRITICAL_TEMP */
+/* Structure for command IWN_CMD_SET_CRITICAL_TEMP. */
struct iwn_critical_temp {
uint32_t reserved;
uint32_t tempM;
uint32_t tempR;
-/* degK <-> degC conversion macros */
+/* degK <-> degC conversion macros. */
#define IWN_CTOK(c) ((c) + 273)
#define IWN_KTOC(k) ((k) - 273)
#define IWN_CTOMUK(c) (((c) * 1000000) + 273150000)
} __packed;
-/* structure for command IWN_SENSITIVITY */
+/* Structure for command IWN_CMD_SET_SENSITIVITY. */
struct iwn_sensitivity_cmd {
uint16_t which;
#define IWN_SENSITIVITY_DEFAULTTBL 0
@@ -801,19 +812,10 @@ struct iwn_sensitivity_cmd {
uint16_t energy_ofdm_th;
} __packed;
-/* structure for command IWN_PHY_CALIB */
-struct iwn_phy_calib_cmd {
- uint8_t code;
-
- uint8_t flags;
- uint16_t reserved1;
- int8_t gain[3];
- uint8_t reserved2;
-} __packed;
-
-struct iwn5000_calib {
+/* Structures for command IWN_CMD_PHY_CALIB. */
+struct iwn_phy_calib {
uint8_t code;
-#define IWN4965_SET_DIFF_GAIN 7
+#define IWN4965_PHY_CALIB_DIFF_GAIN 7
#define IWN5000_PHY_CALIB_LO 9
#define IWN5000_PHY_CALIB_LO_TX_IQ 11
#define IWN5000_PHY_CALIB_CRYSTAL 15
@@ -826,18 +828,55 @@ struct iwn5000_calib {
uint8_t valid;
} __packed;
-/* structure for command IWN_PHY_CALIB */
-struct iwn5000_phy_calib {
+struct iwn5000_phy_calib_crystal {
uint8_t code;
-
uint8_t group;
uint8_t ngroups;
uint8_t valid;
+
uint8_t data[2];
} __packed;
+struct iwn_phy_calib_gain {
+ uint8_t code;
+ uint8_t group;
+ uint8_t ngroups;
+ uint8_t valid;
+
+ int8_t gain[3];
+ uint8_t reserved2;
+} __packed;
-/* structure for IWN_UC_READY notification */
+/* Structure for command IWN_CMD_SPECTRUM_MEASUREMENT. */
+struct iwn_spectrum_cmd {
+ uint16_t len;
+ uint8_t token;
+ uint8_t id;
+ uint8_t origin;
+ uint8_t periodic;
+ uint16_t timeout;
+ uint32_t start;
+ uint32_t reserved1;
+ uint32_t flags;
+ uint32_t filter;
+ uint16_t nchan;
+ uint16_t reserved2;
+ struct {
+ uint32_t duration;
+ uint8_t chan;
+ uint8_t type;
+#define IWN_MEASUREMENT_BASIC (1 << 0)
+#define IWN_MEASUREMENT_CCA (1 << 1)
+#define IWN_MEASUREMENT_RPI_HISTOGRAM (1 << 2)
+#define IWN_MEASUREMENT_NOISE_HISTOGRAM (1 << 3)
+#define IWN_MEASUREMENT_FRAME (1 << 4)
+#define IWN_MEASUREMENT_IDLE (1 << 7)
+
+ uint16_t reserved;
+ } __packed chan[10];
+} __packed;
+
+/* Structure for IWN_UC_READY notification. */
#define IWN_NATTEN_GROUPS 5
struct iwn_ucode_info {
uint8_t minor;
@@ -851,11 +890,11 @@ struct iwn_ucode_info {
uint16_t reserved2;
uint32_t logptr;
- uint32_t errorptr;
+ uint32_t errptr;
uint32_t tstamp;
uint32_t valid;
- /* the following fields are for UCODE_INIT only */
+ /* The following fields are for UCODE_INIT only. */
int32_t volt;
struct {
int32_t chan20MHz;
@@ -864,7 +903,7 @@ struct iwn_ucode_info {
int32_t atten[IWN_NATTEN_GROUPS][2];
} __packed;
-/* structure for IWN_TX_DONE notification */
+/* Structures for IWN_TX_DONE notification. */
struct iwn4965_tx_stat {
uint8_t nframes;
uint8_t killcnt;
@@ -898,7 +937,7 @@ struct iwn5000_tx_stat {
uint16_t sequence;
} __packed;
-/* structure for IWN_BEACON_MISSED notification */
+/* Structure for IWN_BEACON_MISSED notification. */
struct iwn_beacon_missed {
uint32_t consecutive;
uint32_t total;
@@ -906,14 +945,13 @@ struct iwn_beacon_missed {
uint32_t received;
} __packed;
-/* structure for IWN_AMPDU_RX_DONE notification */
-struct iwn_rx_ampdu {
+/* Structure for IWN_MPDU_RX_DONE notification. */
+struct iwn_rx_mpdu {
uint16_t len;
uint16_t reserved;
} __packed;
-/* structures for IWN_RX_DONE and IWN_AMPDU_RX_START notifications */
-
+/* Structures for IWN_RX_DONE and IWN_MPDU_RX_DONE notifications. */
struct iwn4965_rx_phystat {
uint16_t antenna;
uint16_t agc;
@@ -936,6 +974,8 @@ struct iwn_rx_stat {
uint64_t tstamp;
uint32_t beacon;
uint16_t flags;
+#define IWN_STAT_FLAG_SHPREAMBLE (1 << 2)
+
uint16_t chan;
uint8_t phybuf[32];
uint8_t rate;
@@ -947,7 +987,7 @@ struct iwn_rx_stat {
#define IWN_RSSI_TO_DBM 44
-/* structure for IWN_START_SCAN notification */
+/* Structure for IWN_START_SCAN notification. */
struct iwn_start_scan {
uint64_t tstamp;
uint32_t tbeacon;
@@ -957,7 +997,7 @@ struct iwn_start_scan {
uint32_t status;
} __packed;
-/* structure for IWN_STOP_SCAN notification */
+/* Structure for IWN_STOP_SCAN notification. */
struct iwn_stop_scan {
uint8_t nchan;
uint8_t status;
@@ -966,7 +1006,39 @@ struct iwn_stop_scan {
uint64_t tsf;
} __packed;
-/* structure for IWN_{RX,BEACON}_STATISTICS notification */
+/* Structure for IWN_SPECTRUM_MEASUREMENT notification. */
+struct iwn_spectrum_notif {
+ uint8_t id;
+ uint8_t token;
+ uint8_t idx;
+ uint8_t state;
+#define IWN_MEASUREMENT_START 0
+#define IWN_MEASUREMENT_STOP 1
+
+ uint32_t start;
+ uint8_t band;
+ uint8_t chan;
+ uint8_t type;
+ uint8_t reserved1;
+ uint32_t cca_ofdm;
+ uint32_t cca_cck;
+ uint32_t cca_time;
+ uint8_t basic;
+ uint8_t reserved2[3];
+ uint32_t ofdm[8];
+ uint32_t cck[8];
+ uint32_t stop;
+ uint32_t status;
+#define IWN_MEASUREMENT_OK 0
+#define IWN_MEASUREMENT_CONCURRENT 1
+#define IWN_MEASUREMENT_CSA_CONFLICT 2
+#define IWN_MEASUREMENT_TGH_CONFLICT 3
+#define IWN_MEASUREMENT_STOPPED 6
+#define IWN_MEASUREMENT_TIMEOUT 7
+#define IWN_MEASUREMENT_FAILED 8
+} __packed;
+
+/* Structure for IWN_{RX,BEACON}_STATISTICS notification. */
struct iwn_rx_phy_stats {
uint32_t ina;
uint32_t fina;
@@ -1083,7 +1155,20 @@ struct iwn_stats {
} __packed;
-/* firmware image header */
+/* Firmware error dump. */
+struct iwn_fw_dump {
+ uint32_t valid;
+ uint32_t id;
+ uint32_t pc;
+ uint32_t branch_link[2];
+ uint32_t interrupt_link[2];
+ uint32_t error_data[2];
+ uint32_t src_line;
+ uint32_t tsf;
+ uint32_t time[2];
+} __packed;
+
+/* Firmware image file header. */
struct iwn_firmware_hdr {
uint32_t version;
uint32_t main_textsz;
@@ -1130,10 +1215,12 @@ struct iwn_firmware_hdr {
#define IWN5000_EEPROM_BAND7 0x049
#define IWN5000_EEPROM_CRYSTAL 0x094
-/* possible flags for IWN_EEPROM_RFCFG */
-#define IWN_RFCFG_TYPE(x) (((x) >> 0) & 3)
-#define IWN_RFCFG_STEP(x) (((x) >> 2) & 3)
-#define IWN_RFCFG_DASH(x) (((x) >> 4) & 3)
+/* Possible flags for IWN_EEPROM_RFCFG. */
+#define IWN_RFCFG_TYPE(x) (((x) >> 0) & 0x3)
+#define IWN_RFCFG_STEP(x) (((x) >> 2) & 0x3)
+#define IWN_RFCFG_DASH(x) (((x) >> 4) & 0x3)
+#define IWN_RFCFG_TXANTMSK(x) (((x) >> 8) & 0xf)
+#define IWN_RFCFG_RXANTMSK(x) (((x) >> 12) & 0xf)
struct iwn_eeprom_chan {
uint8_t flags;
@@ -1358,6 +1445,42 @@ static const uint8_t iwn_tid2fifo[] = {
1, 0, 0, 1, 2, 2, 3, 3, 7, 7, 7, 7, 7, 7, 7, 7, 3
};
+/* Firmware errors. */
+static const char * const iwn_fw_errmsg[] = {
+ "OK",
+ "FAIL",
+ "BAD_PARAM",
+ "BAD_CHECKSUM",
+ "NMI_INTERRUPT_WDG",
+ "SYSASSERT",
+ "FATAL_ERROR",
+ "BAD_COMMAND",
+ "HW_ERROR_TUNE_LOCK",
+ "HW_ERROR_TEMPERATURE",
+ "ILLEGAL_CHAN_FREQ",
+ "VCC_NOT_STABLE",
+ "FH_ERROR",
+ "NMI_INTERRUPT_HOST",
+ "NMI_INTERRUPT_ACTION_PT",
+ "NMI_INTERRUPT_UNKNOWN",
+ "UCODE_VERSION_MISMATCH",
+ "HW_ERROR_ABS_LOCK",
+ "HW_ERROR_CAL_LOCK_FAIL",
+ "NMI_INTERRUPT_INST_ACTION_PT",
+ "NMI_INTERRUPT_DATA_ACTION_PT",
+ "NMI_TRM_HW_ER",
+ "NMI_INTERRUPT_TRM",
+ "NMI_INTERRUPT_BREAKPOINT"
+ "DEBUG_0",
+ "DEBUG_1",
+ "DEBUG_2",
+ "DEBUG_3",
+ "UNKNOWN"
+};
+
+/* Find least significant bit that is set. */
+#define IWN_LSB(x) ((((x) - 1) & (x)) ^ (x))
+
#define IWN_READ(sc, reg) \
bus_space_read_4((sc)->sc_st, (sc)->sc_sh, (reg))
diff --git a/sys/dev/pci/if_iwnvar.h b/sys/dev/pci/if_iwnvar.h
index 41b04bf0961..70c155f02ea 100644
--- a/sys/dev/pci/if_iwnvar.h
+++ b/sys/dev/pci/if_iwnvar.h
@@ -1,4 +1,4 @@
-/* $OpenBSD: if_iwnvar.h,v 1.3 2008/10/13 16:37:10 damien Exp $ */
+/* $OpenBSD: if_iwnvar.h,v 1.4 2008/10/22 06:25:07 damien Exp $ */
/*-
* Copyright (c) 2007, 2008
@@ -181,6 +181,10 @@ struct iwn_hal {
int (*add_node)(struct iwn_softc *, struct iwn_node_info *,
int);
void (*tx_done)(struct iwn_softc *, struct iwn_rx_desc *);
+ void (*ampdu_tx_start)(struct iwn_softc *,
+ struct ieee80211_node *, uint8_t, uint16_t);
+ void (*ampdu_tx_stop)(struct iwn_softc *, uint8_t,
+ uint16_t);
const char *fwname;
const struct iwn_sensitivity_limits *limits;
int ntxqs;
@@ -242,6 +246,7 @@ struct iwn_softc {
struct iwn_fw_info fw;
struct iwn_calib_info calibcmd[3];
+ uint32_t errptr;
struct iwn_rx_stat last_rx_stat;
int last_rx_valid;
@@ -250,7 +255,6 @@ struct iwn_softc {
uint32_t rawtemp;
int temp;
int noise;
- uint8_t antmsk;
uint32_t qfullmsk;
struct iwn4965_eeprom_band
@@ -266,6 +270,9 @@ struct iwn_softc {
uint32_t critical_temp;
uint8_t ntxchains;
uint8_t nrxchains;
+ uint8_t txantmsk;
+ uint8_t rxantmsk;
+ uint8_t antmsk;
int sc_tx_timer;
void *powerhook;