diff options
Diffstat (limited to 'sys/dev/ic')
-rw-r--r-- | sys/dev/ic/bwi.c | 170 | ||||
-rw-r--r-- | sys/dev/ic/bwivar.h | 17 |
2 files changed, 179 insertions, 8 deletions
diff --git a/sys/dev/ic/bwi.c b/sys/dev/ic/bwi.c index f935cb51d88..b2d8e73fdef 100644 --- a/sys/dev/ic/bwi.c +++ b/sys/dev/ic/bwi.c @@ -1,4 +1,4 @@ -/* $OpenBSD: bwi.c,v 1.66 2008/02/16 14:56:00 mglocker Exp $ */ +/* $OpenBSD: bwi.c,v 1.67 2008/02/16 16:45:28 mglocker Exp $ */ /* * Copyright (c) 2007 The DragonFly Project. All rights reserved. @@ -245,6 +245,12 @@ void bwi_rf_on_11bg(struct bwi_mac *); void bwi_rf_set_ant_mode(struct bwi_mac *, int); int bwi_rf_get_latest_tssi(struct bwi_mac *, int8_t[], uint16_t); int bwi_rf_tssi2dbm(struct bwi_mac *, int8_t, int8_t *); +int bwi_rf_calc_rssi_bcm2050(struct bwi_mac *, + const struct bwi_rxbuf_hdr *); +int bwi_rf_calc_rssi_bcm2053(struct bwi_mac *, + const struct bwi_rxbuf_hdr *); +int bwi_rf_calc_rssi_bcm2060(struct bwi_mac *, + const struct bwi_rxbuf_hdr *); uint16_t bwi_rf_lo_measure_11b(struct bwi_mac *); void bwi_rf_lo_update_11b(struct bwi_mac *); @@ -353,6 +359,8 @@ void bwi_regwin_disable(struct bwi_softc *, struct bwi_regwin *, void bwi_set_bssid(struct bwi_softc *, const uint8_t *); void bwi_updateslot(struct ieee80211com *); void bwi_calibrate(void *); +int bwi_calc_rssi(struct bwi_softc *, + const struct bwi_rxbuf_hdr *); uint8_t bwi_ack_rate(struct ieee80211_node *, uint8_t); uint16_t bwi_txtime(struct ieee80211com *, struct ieee80211_node *, uint, uint8_t, uint32_t); @@ -3769,12 +3777,15 @@ bwi_rf_attach(struct bwi_mac *mac) rf->rf_ctrl_rd = BWI_RF_CTRL_RD_11A; rf->rf_on = bwi_rf_on_11a; rf->rf_off = bwi_rf_off_11a; + rf->rf_calc_rssi = bwi_rf_calc_rssi_bcm2060; break; case IEEE80211_MODE_11B: if (type == BWI_RF_T_BCM2050) { rf->rf_ctrl_rd = BWI_RF_CTRL_RD_11BG; + rf->rf_calc_rssi = bwi_rf_calc_rssi_bcm2050; } else if (type == BWI_RF_T_BCM2053) { rf->rf_ctrl_adj = 1; + rf->rf_calc_rssi = bwi_rf_calc_rssi_bcm2053; } else { DPRINTF(1, "%s: only BCM2050/BCM2053 RF is supported " "for supported for 11B PHY\n", sc->sc_dev.dv_xname); @@ -3803,6 +3814,7 @@ bwi_rf_attach(struct bwi_mac *mac) rf->rf_off = bwi_rf_off_11bg; rf->rf_calc_nrssi_slope = bwi_rf_calc_nrssi_slope_11g; rf->rf_set_nrssi_thr = bwi_rf_set_nrssi_thr_11g; + rf->rf_calc_rssi = bwi_rf_calc_rssi_bcm2050; rf->rf_lo_update = bwi_rf_lo_update_11g; break; default: @@ -5874,6 +5886,141 @@ bwi_rf_tssi2dbm(struct bwi_mac *mac, int8_t tssi, int8_t *txpwr) return (0); } +int +bwi_rf_calc_rssi_bcm2050(struct bwi_mac *mac, const struct bwi_rxbuf_hdr *hdr) +{ + uint16_t flags1, flags3; + int rssi, lna_gain; + + rssi = hdr->rxh_rssi; + flags1 = letoh16(hdr->rxh_flags1); + flags3 = letoh16(hdr->rxh_flags3); + +#define NEW_BCM2050_RSSI +#ifdef NEW_BCM2050_RSSI + if (flags1 & BWI_RXH_F1_OFDM) { + if (rssi > 127) + rssi -= 256; + if (flags3 & BWI_RXH_F3_BCM2050_RSSI) + rssi += 17; + else + rssi -= 4; + return (rssi); + } + + if (mac->mac_sc->sc_card_flags & BWI_CARD_F_SW_NRSSI) { + struct bwi_rf *rf = &mac->mac_rf; + + if (rssi >= BWI_NRSSI_TBLSZ) + rssi = BWI_NRSSI_TBLSZ - 1; + + rssi = ((31 - (int)rf->rf_nrssi_table[rssi]) * -131) / 128; + rssi -= 67; + } else { + rssi = ((31 - rssi) * -149) / 128; + rssi -= 68; + } + + if (mac->mac_phy.phy_mode != IEEE80211_MODE_11G) + return (rssi); + + if (flags3 & BWI_RXH_F3_BCM2050_RSSI) + rssi += 20; + + lna_gain = __SHIFTOUT(letoh16(hdr->rxh_phyinfo), + BWI_RXH_PHYINFO_LNAGAIN); + DPRINTF(mac->mac_sc, BWI_DBG_RF | BWI_DBG_RX, + "lna_gain %d, phyinfo 0x%04x\n", + lna_gain, le16toh(hdr->rxh_phyinfo)); + switch (lna_gain) { + case 0: + rssi += 27; + break; + case 1: + rssi += 6; + break; + case 2: + rssi += 12; + break; + case 3: + /* + * XXX + * According to v3 spec, we should do _nothing_ here, + * but it seems that the result RSSI will be too low + * (relative to what ath(4) says). Raise it a little + * bit. + */ + rssi += 5; + break; + default: + panic("impossible lna gain %d", lna_gain); + } +#else /* !NEW_BCM2050_RSSI */ + lna_gain = 0; /* shut up gcc warning */ + + if (flags1 & BWI_RXH_F1_OFDM) { + if (rssi > 127) + rssi -= 256; + rssi = (rssi * 73) / 64; + + if (flags3 & BWI_RXH_F3_BCM2050_RSSI) + rssi += 25; + else + rssi -= 3; + return (rssi); + } + + if (mac->mac_sc->sc_card_flags & BWI_CARD_F_SW_NRSSI) { + struct bwi_rf *rf = &mac->mac_rf; + + if (rssi >= BWI_NRSSI_TBLSZ) + rssi = BWI_NRSSI_TBLSZ - 1; + + rssi = ((31 - (int)rf->rf_nrssi_table[rssi]) * -131) / 128; + rssi -= 57; + } else { + rssi = ((31 - rssi) * -149) / 128; + rssi -= 68; + } + + if (mac->mac_phy.phy_mode != IEEE80211_MODE_11G) + return (rssi); + + if (flags3 & BWI_RXH_F3_BCM2050_RSSI) + rssi += 25; +#endif /* NEW_BCM2050_RSSI */ + return (rssi); +} + +int +bwi_rf_calc_rssi_bcm2053(struct bwi_mac *mac, const struct bwi_rxbuf_hdr *hdr) +{ + uint16_t flags1; + int rssi; + + rssi = (((int)hdr->rxh_rssi - 11) * 103) / 64; + + flags1 = letoh16(hdr->rxh_flags1); + if (flags1 & BWI_RXH_F1_BCM2053_RSSI) + rssi -= 109; + else + rssi -= 83; + + return (rssi); +} + +int +bwi_rf_calc_rssi_bcm2060(struct bwi_mac *mac, const struct bwi_rxbuf_hdr *hdr) +{ + int rssi; + + rssi = hdr->rxh_rssi; + if (rssi > 127) + rssi -= 256; + + return (rssi); +} + uint16_t bwi_rf_lo_measure_11b(struct bwi_mac *mac) { @@ -8006,7 +8153,7 @@ bwi_rxeof(struct bwi_softc *sc, int end_idx) struct mbuf *m; void *plcp; uint16_t flags2; - int buflen, wh_ofs, hdr_extra, type, rate; + int buflen, wh_ofs, hdr_extra, rssi, type, rate; m = rb->rb_mbuf; bus_dmamap_sync(sc->sc_dmat, rb->rb_dmap, 0, @@ -8035,6 +8182,7 @@ bwi_rxeof(struct bwi_softc *sc, int end_idx) } plcp = ((uint8_t *)(hdr + 1) + hdr_extra); + rssi = bwi_calc_rssi(sc, hdr); m->m_pkthdr.rcvif = ifp; m->m_len = m->m_pkthdr.len = buflen + sizeof(*hdr); @@ -8051,16 +8199,15 @@ bwi_rxeof(struct bwi_softc *sc, int end_idx) struct mbuf mb; struct bwi_rx_radiotap_hdr *tap = &sc->sc_rxtap; - /* TODO: calculate rate and signal */ tap->wr_tsf = hdr->rxh_tsf; tap->wr_flags = 0; - tap->wr_rate = 0; + tap->wr_rate = rate; tap->wr_chan_freq = htole16(ic->ic_bss->ni_chan->ic_freq); tap->wr_chan_flags = htole16(ic->ic_bss->ni_chan->ic_flags); - tap->wr_antsignal = 0; - tap->wr_antnoise = 0; + tap->wr_antsignal = rssi; + tap->wr_antnoise = BWI_NOISE_FLOOR; mb.m_data = (caddr_t)tap; mb.m_len = sc->sc_rxtap_len; @@ -9181,3 +9328,14 @@ bwi_calibrate(void *xsc) splx(s); } + +int +bwi_calc_rssi(struct bwi_softc *sc, const struct bwi_rxbuf_hdr *hdr) +{ + struct bwi_mac *mac; + + KKASSERT(sc->sc_cur_regwin->rw_type == BWI_REGWIN_T_MAC); + mac = (struct bwi_mac *)sc->sc_cur_regwin; + + return (bwi_rf_calc_rssi(mac, hdr)); +} diff --git a/sys/dev/ic/bwivar.h b/sys/dev/ic/bwivar.h index 5f248c036ff..60b3c3100bf 100644 --- a/sys/dev/ic/bwivar.h +++ b/sys/dev/ic/bwivar.h @@ -1,4 +1,4 @@ -/* $OpenBSD: bwivar.h,v 1.21 2008/02/16 14:56:00 mglocker Exp $ */ +/* $OpenBSD: bwivar.h,v 1.22 2008/02/16 16:45:28 mglocker Exp $ */ /* * Copyright (c) 2007 The DragonFly Project. All rights reserved. @@ -63,6 +63,8 @@ #define BWI_LED_EVENT_RX 2 #define BWI_LED_SLOWDOWN(dur) (dur) = (((dur) * 3) / 2) +#define BWI_NOISE_FLOOR -95 /* TODO: noise floor calc */ + #define CSR_READ_4(sc, reg) \ bus_space_read_4((sc)->sc_mem_bt, (sc)->sc_mem_bh, (reg)) #define CSR_READ_2(sc, reg) \ @@ -120,17 +122,22 @@ struct bwi_rxbuf_hdr { uint16_t rxh_flags1; uint8_t rxh_rssi; uint8_t rxh_sq; - uint8_t rxh_pad2[2]; + uint16_t rxh_phyinfo; /* BWI_RXH_PHYINFO_ */ uint16_t rxh_flags3; uint16_t rxh_flags2; /* BWI_RXH_F2_ */ uint16_t rxh_tsf; uint8_t rxh_pad3[14]; /* Padded to 30bytes */ } __packed; +#define BWI_RXH_F1_BCM2053_RSSI (1 << 14) #define BWI_RXH_F1_OFDM (1 << 0) #define BWI_RXH_F2_TYPE2FRAME (1 << 2) +#define BWI_RXH_F3_BCM2050_RSSI (1 << 10) + +#define BWI_RXH_PHYINFO_LNAGAIN (3 << 14) + struct bwi_txbuf_hdr { /* Little endian */ uint32_t txh_mac_ctrl; /* BWI_TXH_MAC_C_ */ @@ -704,6 +711,12 @@ bwi_rf_set_nrssi_thr(struct bwi_mac *_mac) _mac->mac_rf.rf_set_nrssi_thr(_mac); } +static __inline int +bwi_rf_calc_rssi(struct bwi_mac *_mac, const struct bwi_rxbuf_hdr *_hdr) +{ + return (_mac->mac_rf.rf_calc_rssi(_mac, _hdr)); +} + static __inline void bwi_rf_lo_update(struct bwi_mac *_mac) { |