diff options
Diffstat (limited to 'sys/dev')
-rw-r--r-- | sys/dev/usb/if_urtw.c | 929 | ||||
-rw-r--r-- | sys/dev/usb/if_urtwreg.h | 36 |
2 files changed, 878 insertions, 87 deletions
diff --git a/sys/dev/usb/if_urtw.c b/sys/dev/usb/if_urtw.c index 9c266781e9b..dd99db414e0 100644 --- a/sys/dev/usb/if_urtw.c +++ b/sys/dev/usb/if_urtw.c @@ -1,6 +1,7 @@ -/* $OpenBSD: if_urtw.c,v 1.20 2009/06/05 01:21:02 martynas Exp $ */ +/* $OpenBSD: if_urtw.c,v 1.21 2009/06/06 12:06:28 martynas Exp $ */ /*- + * Copyright (c) 2009 Martynas Venckus <martynas@openbsd.org> * Copyright (c) 2008 Weongyo Jeong <weongyo@FreeBSD.org> * * Permission to use, copy, modify, and distribute this software for any @@ -79,6 +80,8 @@ static const struct urtw_type { } urtw_devs[] = { #define URTW_DEV_RTL8187(v, p) \ { { USB_VENDOR_##v, USB_PRODUCT_##v##_##p }, URTW_HWREV_8187 } +#define URTW_DEV_RTL8187B(v, p) \ + { { USB_VENDOR_##v, USB_PRODUCT_##v##_##p }, URTW_HWREV_8187B } /* Realtek RTL8187 devices. */ URTW_DEV_RTL8187(DICKSMITH, RTL8187), URTW_DEV_RTL8187(LOGITEC, RTL8187), @@ -86,7 +89,15 @@ static const struct urtw_type { URTW_DEV_RTL8187(REALTEK, RTL8187), URTW_DEV_RTL8187(SPHAIRON, RTL8187), URTW_DEV_RTL8187(SURECOM, EP9001G2A), + /* Realtek RTL8187B devices. */ + URTW_DEV_RTL8187B(BELKIN, F5D7050E), + URTW_DEV_RTL8187B(NETGEAR, WG111V3), + URTW_DEV_RTL8187B(REALTEK, RTL8187B_0), + URTW_DEV_RTL8187B(REALTEK, RTL8187B_1), + URTW_DEV_RTL8187B(REALTEK, RTL8187B_2), + URTW_DEV_RTL8187B(SITECOMEU, WL168) #undef URTW_DEV_RTL8187 +#undef URTW_DEV_RTL8187B }; #define urtw_lookup(v, p) \ ((const struct urtw_type *)usb_lookup(urtw_devs, v, p)) @@ -175,6 +186,52 @@ struct urtw_pair { uint32_t val; }; +struct urtw_pair_idx { + uint8_t reg; + uint8_t val; + uint8_t idx; +}; + +static struct urtw_pair_idx urtw_8187b_regtbl[] = { + { 0xf0, 0x32, 0 }, { 0xf1, 0x32, 0 }, { 0xf2, 0x00, 0 }, + { 0xf3, 0x00, 0 }, { 0xf4, 0x32, 0 }, { 0xf5, 0x43, 0 }, + { 0xf6, 0x00, 0 }, { 0xf7, 0x00, 0 }, { 0xf8, 0x46, 0 }, + { 0xf9, 0xa4, 0 }, { 0xfa, 0x00, 0 }, { 0xfb, 0x00, 0 }, + { 0xfc, 0x96, 0 }, { 0xfd, 0xa4, 0 }, { 0xfe, 0x00, 0 }, + { 0xff, 0x00, 0 }, + + { 0x58, 0x4b, 1 }, { 0x59, 0x00, 1 }, { 0x5a, 0x4b, 1 }, + { 0x5b, 0x00, 1 }, { 0x60, 0x4b, 1 }, { 0x61, 0x09, 1 }, + { 0x62, 0x4b, 1 }, { 0x63, 0x09, 1 }, { 0xce, 0x0f, 1 }, + { 0xcf, 0x00, 1 }, { 0xe0, 0xff, 1 }, { 0xe1, 0x0f, 1 }, + { 0xe2, 0x00, 1 }, { 0xf0, 0x4e, 1 }, { 0xf1, 0x01, 1 }, + { 0xf2, 0x02, 1 }, { 0xf3, 0x03, 1 }, { 0xf4, 0x04, 1 }, + { 0xf5, 0x05, 1 }, { 0xf6, 0x06, 1 }, { 0xf7, 0x07, 1 }, + { 0xf8, 0x08, 1 }, + + { 0x4e, 0x00, 2 }, { 0x0c, 0x04, 2 }, { 0x21, 0x61, 2 }, + { 0x22, 0x68, 2 }, { 0x23, 0x6f, 2 }, { 0x24, 0x76, 2 }, + { 0x25, 0x7d, 2 }, { 0x26, 0x84, 2 }, { 0x27, 0x8d, 2 }, + { 0x4d, 0x08, 2 }, { 0x50, 0x05, 2 }, { 0x51, 0xf5, 2 }, + { 0x52, 0x04, 2 }, { 0x53, 0xa0, 2 }, { 0x54, 0x1f, 2 }, + { 0x55, 0x23, 2 }, { 0x56, 0x45, 2 }, { 0x57, 0x67, 2 }, + { 0x58, 0x08, 2 }, { 0x59, 0x08, 2 }, { 0x5a, 0x08, 2 }, + { 0x5b, 0x08, 2 }, { 0x60, 0x08, 2 }, { 0x61, 0x08, 2 }, + { 0x62, 0x08, 2 }, { 0x63, 0x08, 2 }, { 0x64, 0xcf, 2 }, + { 0x72, 0x56, 2 }, { 0x73, 0x9a, 2 }, + + { 0x34, 0xf0, 0 }, { 0x35, 0x0f, 0 }, { 0x5b, 0x40, 0 }, + { 0x84, 0x88, 0 }, { 0x85, 0x24, 0 }, { 0x88, 0x54, 0 }, + { 0x8b, 0xb8, 0 }, { 0x8c, 0x07, 0 }, { 0x8d, 0x00, 0 }, + { 0x94, 0x1b, 0 }, { 0x95, 0x12, 0 }, { 0x96, 0x00, 0 }, + { 0x97, 0x06, 0 }, { 0x9d, 0x1a, 0 }, { 0x9f, 0x10, 0 }, + { 0xb4, 0x22, 0 }, { 0xbe, 0x80, 0 }, { 0xdb, 0x00, 0 }, + { 0xee, 0x00, 0 }, { 0x91, 0x03, 0 }, + + { 0x4c, 0x00, 2 }, { 0x9f, 0x00, 3 }, { 0x8c, 0x01, 0 }, + { 0x8d, 0x10, 0 }, { 0x8e, 0x08, 0 }, { 0x8f, 0x00, 0 } +}; + static uint8_t urtw_8225_agc[] = { 0x9e, 0x9e, 0x9e, 0x9e, 0x9e, 0x9e, 0x9e, 0x9e, 0x9d, 0x9c, 0x9b, 0x9a, 0x99, 0x98, 0x97, 0x96, 0x95, 0x94, 0x93, 0x92, 0x91, 0x90, @@ -293,6 +350,36 @@ static uint8_t urtw_8225_txpwr_ofdm[] = { 0x80, 0x90, 0xa2, 0xb5, 0xcb, 0xe4 }; +static uint8_t urtw_8225v2_agc[] = { + 0x5e, 0x5e, 0x5e, 0x5e, 0x5d, 0x5b, 0x59, 0x57, + 0x55, 0x53, 0x51, 0x4f, 0x4d, 0x4b, 0x49, 0x47, + 0x45, 0x43, 0x41, 0x3f, 0x3d, 0x3b, 0x39, 0x37, + 0x35, 0x33, 0x31, 0x2f, 0x2d, 0x2b, 0x29, 0x27, + 0x25, 0x23, 0x21, 0x1f, 0x1d, 0x1b, 0x19, 0x17, + 0x15, 0x13, 0x11, 0x0f, 0x0d, 0x0b, 0x09, 0x07, + 0x05, 0x03, 0x01, 0x01, 0x01, 0x01, 0x01, 0x01, + 0x01, 0x01, 0x01, 0x01, 0x01, 0x01, 0x01, 0x01, + 0x19, 0x19, 0x19, 0x19, 0x19, 0x19, 0x19, 0x19, + 0x19, 0x20, 0x21, 0x22, 0x23, 0x24, 0x25, 0x26, + 0x26, 0x27, 0x27, 0x28, 0x28, 0x29, 0x2a, 0x2a, + 0x2a, 0x2b, 0x2b, 0x2b, 0x2c, 0x2c, 0x2c, 0x2d, + 0x2d, 0x2d, 0x2d, 0x2e, 0x2e, 0x2e, 0x2e, 0x2f, + 0x2f, 0x2f, 0x30, 0x30, 0x31, 0x31, 0x31, 0x31, + 0x31, 0x31, 0x31, 0x31, 0x31, 0x31, 0x31, 0x31, + 0x31, 0x31, 0x31, 0x31, 0x31, 0x31, 0x31, 0x31 +}; + +static uint8_t urtw_8225v2_ofdm[] = { + 0x10, 0x0d, 0x01, 0x00, 0x14, 0xfb, 0xfb, 0x60, + 0x00, 0x60, 0x00, 0x00, 0x00, 0x5c, 0x00, 0x00, + 0x40, 0x00, 0x40, 0x00, 0x00, 0x00, 0xa8, 0x26, + 0x32, 0x33, 0x07, 0xa5, 0x6f, 0x55, 0xc8, 0xb3, + 0x0a, 0xe1, 0x2c, 0x8a, 0x86, 0x83, 0x34, 0x0f, + 0x4f, 0x24, 0x6f, 0xc2, 0x6b, 0x40, 0x80, 0x00, + 0xc0, 0xc1, 0x58, 0xf1, 0x00, 0xe4, 0x90, 0x3e, + 0x6d, 0x3c, 0xfb, 0x07 +}; + static uint8_t urtw_8225v2_gain_bg[] = { 0x23, 0x15, 0xa5, /* -82-1dbm */ 0x23, 0x15, 0xb5, /* -82-2dbm */ @@ -335,16 +422,16 @@ static struct urtw_pair urtw_8225v2_rf_part3[] = { }; static uint16_t urtw_8225v2_rxgain[] = { - 0x0000, 0x0001, 0x0002, 0x0003, 0x0004, 0x0005, 0x0008, 0x0009, - 0x000a, 0x000b, 0x0102, 0x0103, 0x0104, 0x0105, 0x0140, 0x0141, - 0x0142, 0x0143, 0x0144, 0x0145, 0x0180, 0x0181, 0x0182, 0x0183, - 0x0184, 0x0185, 0x0188, 0x0189, 0x018a, 0x018b, 0x0243, 0x0244, - 0x0245, 0x0280, 0x0281, 0x0282, 0x0283, 0x0284, 0x0285, 0x0288, - 0x0289, 0x028a, 0x028b, 0x028c, 0x0342, 0x0343, 0x0344, 0x0345, - 0x0380, 0x0381, 0x0382, 0x0383, 0x0384, 0x0385, 0x0388, 0x0389, - 0x038a, 0x038b, 0x038c, 0x038d, 0x0390, 0x0391, 0x0392, 0x0393, - 0x0394, 0x0395, 0x0398, 0x0399, 0x039a, 0x039b, 0x039c, 0x039d, - 0x03a0, 0x03a1, 0x03a2, 0x03a3, 0x03a4, 0x03a5, 0x03a8, 0x03a9, + 0x0400, 0x0401, 0x0402, 0x0403, 0x0404, 0x0405, 0x0408, 0x0409, + 0x040a, 0x040b, 0x0502, 0x0503, 0x0504, 0x0505, 0x0540, 0x0541, + 0x0542, 0x0543, 0x0544, 0x0545, 0x0580, 0x0581, 0x0582, 0x0583, + 0x0584, 0x0585, 0x0588, 0x0589, 0x058a, 0x058b, 0x0643, 0x0644, + 0x0645, 0x0680, 0x0681, 0x0682, 0x0683, 0x0684, 0x0685, 0x0688, + 0x0689, 0x068a, 0x068b, 0x068c, 0x0742, 0x0743, 0x0744, 0x0745, + 0x0780, 0x0781, 0x0782, 0x0783, 0x0784, 0x0785, 0x0788, 0x0789, + 0x078a, 0x078b, 0x078c, 0x078d, 0x0790, 0x0791, 0x0792, 0x0793, + 0x0794, 0x0795, 0x0798, 0x0799, 0x079a, 0x079b, 0x079c, 0x079d, + 0x07a0, 0x07a1, 0x07a2, 0x07a3, 0x07a4, 0x07a5, 0x07a8, 0x07a9, 0x03aa, 0x03ab, 0x03ac, 0x03ad, 0x03b0, 0x03b1, 0x03b2, 0x03b3, 0x03b4, 0x03b5, 0x03b8, 0x03b9, 0x03ba, 0x03bb, 0x03bb }; @@ -359,11 +446,25 @@ static uint8_t urtw_8225v2_tx_gain_cck_ofdm[] = { }; static uint8_t urtw_8225v2_txpwr_cck[] = { - 0x36, 0x35, 0x2e, 0x25, 0x1c, 0x12, 0x09, 0x04 + 0x36, 0x35, 0x2e, 0x25, 0x1c, 0x12, 0x09, 0x04, + 0x30, 0x2f, 0x29, 0x21, 0x19, 0x10, 0x08, 0x03, + 0x2b, 0x2a, 0x25, 0x1e, 0x16, 0x0e, 0x07, 0x03, + 0x26, 0x25, 0x21, 0x1b, 0x14, 0x0d, 0x06, 0x03 }; static uint8_t urtw_8225v2_txpwr_cck_ch14[] = { - 0x36, 0x35, 0x2e, 0x1b, 0x00, 0x00, 0x00, 0x00 + 0x36, 0x35, 0x2e, 0x1b, 0x00, 0x00, 0x00, 0x00, + 0x30, 0x2f, 0x29, 0x15, 0x00, 0x00, 0x00, 0x00, + 0x30, 0x2f, 0x29, 0x15, 0x00, 0x00, 0x00, 0x00, + 0x30, 0x2f, 0x29, 0x15, 0x00, 0x00, 0x00, 0x00 +}; + +static struct urtw_pair urtw_8225v2_b_rf[] = { + { 0x00, 0x00b7 }, { 0x01, 0x0ee0 }, { 0x02, 0x044d }, { 0x03, 0x0441 }, + { 0x04, 0x08c3 }, { 0x05, 0x0c72 }, { 0x06, 0x00e6 }, { 0x07, 0x082a }, + { 0x08, 0x003f }, { 0x09, 0x0335 }, { 0x0a, 0x09d4 }, { 0x0b, 0x07bb }, + { 0x0c, 0x0850 }, { 0x0d, 0x0cdf }, { 0x0e, 0x002b }, { 0x0f, 0x0114 }, + { 0x00, 0x01b7 } }; static struct urtw_pair urtw_ratetable[] = { @@ -461,6 +562,15 @@ usbd_status urtw_led_mode3(struct urtw_softc *, int); usbd_status urtw_rx_setconf(struct urtw_softc *); usbd_status urtw_rx_enable(struct urtw_softc *); usbd_status urtw_tx_enable(struct urtw_softc *); +usbd_status urtw_8187b_update_wmm(struct urtw_softc *); +usbd_status urtw_8187b_reset(struct urtw_softc *); +int urtw_8187b_init(struct ifnet *); +usbd_status urtw_8225v2_b_config_mac(struct urtw_softc *); +usbd_status urtw_8225v2_b_init_rfe(struct urtw_softc *); +usbd_status urtw_8225v2_b_update_chan(struct urtw_softc *); +usbd_status urtw_8225v2_b_rf_init(struct urtw_rf *); +usbd_status urtw_8225v2_b_rf_set_chan(struct urtw_rf *, int); +usbd_status urtw_8225v2_b_set_txpwrlvl(struct urtw_softc *, int); int urtw_match(struct device *, void *, void *); void urtw_attach(struct device *, struct device *, void *); @@ -499,6 +609,7 @@ urtw_attach(struct device *parent, struct device *self, void *aux) struct ieee80211com *ic = &sc->sc_ic; struct ifnet *ifp = &ic->ic_if; usbd_status error; + uint8_t data8; uint32_t data; int i; const char *urtw_name = NULL; @@ -520,7 +631,8 @@ urtw_attach(struct device *parent, struct device *self, void *aux) * Detect Realtek RTL8187B devices that use * USB IDs of RTL8187. */ - urtw_name = "RTL8187B rev. B (not supported)"; + sc->sc_hwrev = URTW_HWREV_8187B | URTW_HWREV_8187B_B; + urtw_name = "RTL8187B rev. B (early)"; break; default: sc->sc_hwrev |= URTW_HWREV_8187_B; @@ -528,7 +640,26 @@ urtw_attach(struct device *parent, struct device *self, void *aux) break; } } else { - urtw_name = "RTL8187B (not supported)"; + /* RTL8187B hwrev register. */ + urtw_read8_m(sc, URTW_8187B_HWREV, &data8); + switch (data8) { + case URTW_8187B_HWREV_8187B_B: + sc->sc_hwrev |= URTW_HWREV_8187B_B; + urtw_name = "RTL8187B rev. B"; + break; + case URTW_8187B_HWREV_8187B_D: + sc->sc_hwrev |= URTW_HWREV_8187B_D; + urtw_name = "RTL8187B rev. D"; + break; + case URTW_8187B_HWREV_8187B_E: + sc->sc_hwrev |= URTW_HWREV_8187B_E; + urtw_name = "RTL8187B rev. E"; + break; + default: + sc->sc_hwrev |= URTW_HWREV_8187B_B; + urtw_name = "RTL8187B rev. B (default)"; + break; + } } printf("%s: %s", sc->sc_dev.dv_xname, urtw_name); @@ -589,7 +720,11 @@ urtw_attach(struct device *parent, struct device *self, void *aux) ifp->if_softc = sc; ifp->if_flags = IFF_BROADCAST | IFF_SIMPLEX | IFF_MULTICAST; - ifp->if_init = urtw_init; + if (sc->sc_hwrev & URTW_HWREV_8187) { + ifp->if_init = urtw_init; + } else { + ifp->if_init = urtw_8187b_init; + } ifp->if_ioctl = urtw_ioctl; ifp->if_start = urtw_start; ifp->if_watchdog = urtw_watchdog; @@ -708,24 +843,36 @@ urtw_open_pipes(struct urtw_softc *sc) */ /* tx pipe - low priority packets */ - error = usbd_open_pipe(sc->sc_iface, 0x2, USBD_EXCLUSIVE_USE, - &sc->sc_txpipe_low); + if (sc->sc_hwrev & URTW_HWREV_8187) + error = usbd_open_pipe(sc->sc_iface, 0x2, + USBD_EXCLUSIVE_USE, &sc->sc_txpipe_low); + else + error = usbd_open_pipe(sc->sc_iface, 0x6, + USBD_EXCLUSIVE_USE, &sc->sc_txpipe_low); if (error != 0) { printf("%s: could not open Tx low pipe: %s\n", sc->sc_dev.dv_xname, usbd_errstr(error)); goto fail; } /* tx pipe - normal priority packets */ - error = usbd_open_pipe(sc->sc_iface, 0x3, USBD_EXCLUSIVE_USE, - &sc->sc_txpipe_normal); + if (sc->sc_hwrev & URTW_HWREV_8187) + error = usbd_open_pipe(sc->sc_iface, 0x3, + USBD_EXCLUSIVE_USE, &sc->sc_txpipe_normal); + else + error = usbd_open_pipe(sc->sc_iface, 0x7, + USBD_EXCLUSIVE_USE, &sc->sc_txpipe_normal); if (error != 0) { printf("%s: could not open Tx normal pipe: %s\n", sc->sc_dev.dv_xname, usbd_errstr(error)); goto fail; } /* rx pipe */ - error = usbd_open_pipe(sc->sc_iface, 0x81, USBD_EXCLUSIVE_USE, - &sc->sc_rxpipe); + if (sc->sc_hwrev & URTW_HWREV_8187) + error = usbd_open_pipe(sc->sc_iface, 0x81, + USBD_EXCLUSIVE_USE, &sc->sc_rxpipe); + else + error = usbd_open_pipe(sc->sc_iface, 0x83, + USBD_EXCLUSIVE_USE, &sc->sc_rxpipe); if (error != 0) { printf("%s: could not open Rx pipe: %s\n", sc->sc_dev.dv_xname, usbd_errstr(error)); @@ -1147,33 +1294,43 @@ urtw_get_rfchip(struct urtw_softc *sc) rf->rf_sc = sc; - error = urtw_eprom_read32(sc, URTW_EPROM_RFCHIPID, &data); - if (error != 0) - goto fail; - switch (data & 0xff) { - case URTW_EPROM_RFCHIPID_RTL8225U: - error = urtw_8225_isv2(sc, &ret); + if (sc->sc_hwrev & URTW_HWREV_8187) { + error = urtw_eprom_read32(sc, URTW_EPROM_RFCHIPID, &data); if (error != 0) + panic("unsupported RF chip"); + /* NOTREACHED */ + switch (data & 0xff) { + case URTW_EPROM_RFCHIPID_RTL8225U: + error = urtw_8225_isv2(sc, &ret); + if (error != 0) + goto fail; + if (ret == 0) { + rf->init = urtw_8225_rf_init; + rf->set_chan = urtw_8225_rf_set_chan; + rf->set_sens = urtw_8225_rf_set_sens; + } else { + rf->init = urtw_8225v2_rf_init; + rf->set_chan = urtw_8225v2_rf_set_chan; + rf->set_sens = NULL; + } + break; + default: goto fail; - if (ret == 0) { - rf->init = urtw_8225_rf_init; - rf->set_sens = urtw_8225_rf_set_sens; - rf->set_chan = urtw_8225_rf_set_chan; - } else { - rf->init = urtw_8225v2_rf_init; - rf->set_chan = urtw_8225v2_rf_set_chan; - rf->set_sens = NULL; } - rf->max_sens = URTW_8225_RF_MAX_SENS; - rf->sens = URTW_8225_RF_DEF_SENS; - break; - default: - panic("unsupported RF chip %d\n", data & 0xff); - /* NOTREACHED */ + } else { + rf->init = urtw_8225v2_b_rf_init; + rf->set_chan = urtw_8225v2_b_rf_set_chan; + rf->set_sens = NULL; } + rf->max_sens = URTW_8225_RF_MAX_SENS; + rf->sens = URTW_8225_RF_DEF_SENS; + + return (0); + fail: - return (error); + panic("unsupported RF chip %d\n", data & 0xff); + /* NOTREACHED */ } usbd_status @@ -1207,14 +1364,41 @@ urtw_get_txpwr(struct urtw_softc *sc) sc->sc_txpwr_ofdm[i + 6] = (data & 0xf0) >> 4; sc->sc_txpwr_ofdm[i + 6 + 1] = (data & 0xf000) >> 12; } - for (i = 1, j = 0; i < 4; i += 2, j++) { - error = urtw_eprom_read32(sc, URTW_EPROM_TXPW2 + j, &data); + if (sc->sc_hwrev & URTW_HWREV_8187) { + for (i = 1, j = 0; i < 4; i += 2, j++) { + error = urtw_eprom_read32(sc, URTW_EPROM_TXPW2 + j, + &data); + if (error != 0) + goto fail; + sc->sc_txpwr_cck[i + 6 + 4] = data & 0xf; + sc->sc_txpwr_cck[i + 6 + 4 + 1] = (data & 0xf00) >> 8; + sc->sc_txpwr_ofdm[i + 6 + 4] = (data & 0xf0) >> 4; + sc->sc_txpwr_ofdm[i + 6 + 4 + 1] = + (data & 0xf000) >> 12; + } + } else { + /* Channel 11. */ + error = urtw_eprom_read32(sc, 0x1b, &data); + if (error != 0) + goto fail; + sc->sc_txpwr_cck[11] = data & 0xf; + sc->sc_txpwr_ofdm[11] = (data & 0xf0) >> 4; + + /* Channel 12. */ + error = urtw_eprom_read32(sc, 0xa, &data); if (error != 0) goto fail; - sc->sc_txpwr_cck[i + 6 + 4] = data & 0xf; - sc->sc_txpwr_cck[i + 6 + 4 + 1] = (data & 0xf00) >> 8; - sc->sc_txpwr_ofdm[i + 6 + 4] = (data & 0xf0) >> 4; - sc->sc_txpwr_ofdm[i + 6 + 4 + 1] = (data & 0xf000) >> 12; + sc->sc_txpwr_cck[12] = data & 0xf; + sc->sc_txpwr_ofdm[12] = (data & 0xf0) >> 4; + + /* Channel 13, 14. */ + error = urtw_eprom_read32(sc, 0x1c, &data); + if (error != 0) + goto fail; + sc->sc_txpwr_cck[13] = data & 0xf; + sc->sc_txpwr_ofdm[13] = (data & 0xf0) >> 4; + sc->sc_txpwr_cck[14] = (data & 0xf00) >> 8; + sc->sc_txpwr_ofdm[14] = (data & 0xf000) >> 12; } fail: return (error); @@ -1880,6 +2064,10 @@ urtw_update_msr(struct urtw_softc *sc) urtw_read8_m(sc, URTW_MSR, &data); data &= ~URTW_MSR_LINK_MASK; + /* Should always be set. */ + if (sc->sc_hwrev & URTW_HWREV_8187B) + data |= URTW_MSR_LINK_ENEDCA; + if (sc->sc_state == IEEE80211_S_RUN) { switch (ic->ic_opmode) { case IEEE80211_M_STA: @@ -2044,27 +2232,35 @@ urtw_tx_enable(struct urtw_softc *sc) uint32_t data; usbd_status error; - urtw_read8_m(sc, URTW_CW_CONF, &data8); - data8 &= ~(URTW_CW_CONF_PERPACKET_CW | URTW_CW_CONF_PERPACKET_RETRY); - urtw_write8_m(sc, URTW_CW_CONF, data8); + if (sc->sc_hwrev & URTW_HWREV_8187) { + urtw_read8_m(sc, URTW_CW_CONF, &data8); + data8 &= ~(URTW_CW_CONF_PERPACKET_CW | + URTW_CW_CONF_PERPACKET_RETRY); + urtw_write8_m(sc, URTW_CW_CONF, data8); - urtw_read8_m(sc, URTW_TX_AGC_CTL, &data8); - data8 &= ~URTW_TX_AGC_CTL_PERPACKET_GAIN; - data8 &= ~URTW_TX_AGC_CTL_PERPACKET_ANTSEL; - data8 &= ~URTW_TX_AGC_CTL_FEEDBACK_ANT; - urtw_write8_m(sc, URTW_TX_AGC_CTL, data8); + urtw_read8_m(sc, URTW_TX_AGC_CTL, &data8); + data8 &= ~URTW_TX_AGC_CTL_PERPACKET_GAIN; + data8 &= ~URTW_TX_AGC_CTL_PERPACKET_ANTSEL; + data8 &= ~URTW_TX_AGC_CTL_FEEDBACK_ANT; + urtw_write8_m(sc, URTW_TX_AGC_CTL, data8); - urtw_read32_m(sc, URTW_TX_CONF, &data); - data &= ~URTW_TX_LOOPBACK_MASK; - data |= URTW_TX_LOOPBACK_NONE; - data &= ~(URTW_TX_DPRETRY_MASK | URTW_TX_RTSRETRY_MASK); - data |= sc->sc_tx_retry << URTW_TX_DPRETRY_SHIFT; - data |= sc->sc_rts_retry << URTW_TX_RTSRETRY_SHIFT; - data &= ~(URTW_TX_NOCRC | URTW_TX_MXDMA_MASK); - data |= URTW_TX_MXDMA_2048 | URTW_TX_CWMIN | URTW_TX_DISCW; - data &= ~URTW_TX_SWPLCPLEN; - data |= URTW_TX_NOICV; - urtw_write32_m(sc, URTW_TX_CONF, data); + urtw_read32_m(sc, URTW_TX_CONF, &data); + data &= ~URTW_TX_LOOPBACK_MASK; + data |= URTW_TX_LOOPBACK_NONE; + data &= ~(URTW_TX_DPRETRY_MASK | URTW_TX_RTSRETRY_MASK); + data |= sc->sc_tx_retry << URTW_TX_DPRETRY_SHIFT; + data |= sc->sc_rts_retry << URTW_TX_RTSRETRY_SHIFT; + data &= ~(URTW_TX_NOCRC | URTW_TX_MXDMA_MASK); + data |= URTW_TX_MXDMA_2048 | URTW_TX_CWMIN | URTW_TX_DISCW; + data &= ~URTW_TX_SWPLCPLEN; + data |= URTW_TX_NOICV; + urtw_write32_m(sc, URTW_TX_CONF, data); + } else { + data = URTW_TX_DURPROCMODE | URTW_TX_DISREQQSIZE | + URTW_TX_MXDMA_2048 | URTW_TX_SHORTRETRY | + URTW_TX_LONGRETRY; + urtw_write32_m(sc, URTW_TX_CONF, data); + } urtw_read8_m(sc, URTW_CMD, &data8); urtw_write8_m(sc, URTW_CMD, data8 | URTW_CMD_TX_ENABLE); @@ -2497,7 +2693,11 @@ urtw_tx_start(struct urtw_softc *sc, struct ieee80211_node *ni, struct mbuf *m0, } #endif - xferlen = m0->m_pkthdr.len + 4 * 3; + if (sc->sc_hwrev & URTW_HWREV_8187) + xferlen = m0->m_pkthdr.len + 4 * 3; + else + xferlen = m0->m_pkthdr.len + 4 * 8; + if ((0 == xferlen % 64) || (0 == xferlen % 512)) xferlen += 1; @@ -2533,11 +2733,17 @@ urtw_tx_start(struct urtw_softc *sc, struct ieee80211_node *ni, struct mbuf *m0, else if (ic->ic_fixed_rate != -1) data->buf[3] = urtw_rate2rtl(ic->ic_fixed_rate); } - data->buf[8] = 3; /* CW minimum */ - data->buf[8] |= (7 << 4); /* CW maximum */ - data->buf[9] |= 11; /* retry limitation */ - m_copydata(m0, 0, m0->m_pkthdr.len, (uint8_t *)&data->buf[12]); + if (sc->sc_hwrev & URTW_HWREV_8187) { + data->buf[8] = 3; /* CW minimum */ + data->buf[8] |= (7 << 4); /* CW maximum */ + data->buf[9] |= 11; /* retry limitation */ + m_copydata(m0, 0, m0->m_pkthdr.len, (uint8_t *)&data->buf[12]); + } else { + data->buf[21] |= 11; /* retry limitation */ + m_copydata(m0, 0, m0->m_pkthdr.len, (uint8_t *)&data->buf[32]); + } + data->ni = ni; /* mbuf is no longer needed. */ @@ -2616,7 +2822,10 @@ urtw_8187_write_phy(struct urtw_softc *sc, uint8_t addr, uint32_t data) urtw_write8_m(sc, 0x7e, ((phyw & 0x00ff0000) >> 16)); urtw_write8_m(sc, 0x7d, ((phyw & 0x0000ff00) >> 8)); urtw_write8_m(sc, 0x7c, ((phyw & 0x000000ff))); - usbd_delay_ms(sc->sc_udev, 1); + /* + * Delay removed from 8185 to 8187. + * usbd_delay_ms(sc->sc_udev, 1); + */ fail: return (error); } @@ -2874,18 +3083,32 @@ void urtw_stop(struct ifnet *ifp, int disable) { struct urtw_softc *sc = ifp->if_softc; + struct ieee80211com *ic = &sc->sc_ic; + uint8_t data; + usbd_status error; ifp->if_flags &= ~(IFF_RUNNING | IFF_OACTIVE); + ieee80211_new_state(ic, IEEE80211_S_INIT, -1); + timeout_del(&sc->scan_to); timeout_del(&sc->sc_led_ch); + urtw_intr_disable(sc); + urtw_read8_m(sc, URTW_CMD, &data); + data &= ~URTW_CMD_TX_ENABLE; + data &= ~URTW_CMD_RX_ENABLE; + urtw_write8_m(sc, URTW_CMD, data); + if (sc->sc_rxpipe != NULL) usbd_abort_pipe(sc->sc_rxpipe); if (sc->sc_txpipe_low != NULL) usbd_abort_pipe(sc->sc_txpipe_low); if (sc->sc_txpipe_normal != NULL) usbd_abort_pipe(sc->sc_txpipe_normal); + +fail: + return; } int @@ -2927,8 +3150,13 @@ urtw_rxeof(usbd_xfer_handle xfer, usbd_private_handle priv, usbd_status status) goto skip; } - /* 4 dword and 4 byte CRC */ - len = actlen - (4 * 4); + if (sc->sc_hwrev & URTW_HWREV_8187) + /* 4 dword and 4 byte CRC */ + len = actlen - (4 * 4); + else + /* 5 dword and 4 byte CRC */ + len = actlen - (4 * 5); + desc = data->buf + len; flen = ((desc[1] & 0x0f) << 8) + (desc[0] & 0xff); if (flen > actlen) { @@ -2937,15 +3165,21 @@ urtw_rxeof(usbd_xfer_handle xfer, usbd_private_handle priv, usbd_status status) } rate = (desc[2] & 0xf0) >> 4; - quality = desc[4] & 0xff; - /* XXX correct? */ - rssi = (desc[6] & 0xfe) >> 1; - if (!urtw_isbmode(rate)) { - rssi = (rssi > 90) ? 90 : ((rssi < 25) ? 25 : rssi); - rssi = ((90 - rssi) * 100) / 65; + if (sc->sc_hwrev & URTW_HWREV_8187) { + quality = desc[4] & 0xff; + rssi = (desc[6] & 0xfe) >> 1; + + /* XXX correct? */ + if (!urtw_isbmode(rate)) { + rssi = (rssi > 90) ? 90 : ((rssi < 25) ? 25 : rssi); + rssi = ((90 - rssi) * 100) / 65; + } else { + rssi = (rssi > 90) ? 95 : ((rssi < 30) ? 30 : rssi); + rssi = ((95 - rssi) * 100) / 65; + } } else { - rssi = (rssi > 90) ? 95 : ((rssi < 30) ? 30 : rssi); - rssi = ((95 - rssi) * 100) / 65; + quality = desc[12]; + rssi = 14 - desc[14] / 2; } MGETHDR(mnew, M_DONTWAIT, MT_DATA); @@ -2999,7 +3233,17 @@ urtw_rxeof(usbd_xfer_handle xfer, usbd_private_handle priv, usbd_status status) ni = ieee80211_find_rxnode(ic, wh); /* XXX correct? */ - nf = (quality > 64) ? 0 : ((64 - quality) * 100) / 64; + if (!urtw_isbmode(rate)) { + if (quality > 127) + quality = 0; + else if (quality < 27) + quality = 100; + else + quality = 127 - quality; + } else + quality = (quality > 64) ? 0 : ((64 - quality) * 100) / 64; + + nf = quality; /* send the frame to the 802.11 layer */ rxi.rxi_flags = 0; @@ -3324,7 +3568,7 @@ urtw_task(void *arg) urtw_write16_m(sc, URTW_ATIM_WND, 2); urtw_write16_m(sc, URTW_ATIM_TR_ITV, 100); urtw_write16_m(sc, URTW_BEACON_INTERVAL, 0x64); - urtw_write16_m(sc, URTW_BEACON_INTERVAL_TIME, 100); + urtw_write16_m(sc, URTW_BEACON_INTERVAL_TIME, 0x3ff); error = urtw_led_ctl(sc, URTW_LED_CTL_LINK); if (error != 0) printf("%s: could not control LED (%d)\n", @@ -3340,3 +3584,516 @@ fail: sc->sc_dev.dv_xname)); } +usbd_status +urtw_8187b_update_wmm(struct urtw_softc *sc) +{ + struct ieee80211com *ic = &sc->sc_ic; + struct ieee80211_channel *c = ic->ic_ibss_chan; + uint32_t data; + uint8_t aifs, sifs, slot, ecwmin, ecwmax; + usbd_status error; + + sifs = 0xa; + if (IEEE80211_IS_CHAN_G(c)) + slot = 0x9; + else + slot = 0x14; + + aifs = (2 * slot) + sifs; + ecwmin = 3; + ecwmax = 7; + + data = ((uint32_t)aifs << 0) | /* AIFS, offset 0 */ + ((uint32_t)ecwmin << 8) | /* ECW minimum, offset 8 */ + ((uint32_t)ecwmax << 12); /* ECW maximum, offset 16 */ + + urtw_write32_m(sc, URTW_AC_VO, data); + urtw_write32_m(sc, URTW_AC_VI, data); + urtw_write32_m(sc, URTW_AC_BE, data); + urtw_write32_m(sc, URTW_AC_BK, data); + +fail: + return (error); +} + +usbd_status +urtw_8187b_reset(struct urtw_softc *sc) +{ + uint8_t data; + usbd_status error; + + error = urtw_set_mode(sc, URTW_EPROM_CMD_CONFIG); + if (error) + goto fail; + + urtw_read8_m(sc, URTW_CONFIG3, &data); + urtw_write8_m(sc, URTW_CONFIG3, data | URTW_CONFIG3_ANAPARAM_WRITE | + URTW_CONFIG3_GNT_SELECT); + + urtw_write32_m(sc, URTW_ANAPARAM2, URTW_8187B_8225_ANAPARAM2_ON); + urtw_write32_m(sc, URTW_ANAPARAM, URTW_8187B_8225_ANAPARAM_ON); + urtw_write8_m(sc, URTW_ANAPARAM3, URTW_8187B_8225_ANAPARAM3_ON); + + urtw_write8_m(sc, 0x61, 0x10); + urtw_read8_m(sc, 0x62, &data); + urtw_write8_m(sc, 0x62, data & ~(1 << 5)); + urtw_write8_m(sc, 0x62, data | (1 << 5)); + + urtw_read8_m(sc, URTW_CONFIG3, &data); + urtw_write8_m(sc, URTW_CONFIG3, data & ~URTW_CONFIG3_ANAPARAM_WRITE); + + error = urtw_set_mode(sc, URTW_EPROM_CMD_NORMAL); + if (error) + goto fail; + + urtw_read8_m(sc, URTW_CMD, &data); + data = (data & 2) | URTW_CMD_RST; + urtw_write8_m(sc, URTW_CMD, data); + usbd_delay_ms(sc->sc_udev, 100); + + urtw_read8_m(sc, URTW_CMD, &data); + if (data & URTW_CMD_RST) { + printf("%s: reset timeout\n", sc->sc_dev.dv_xname); + goto fail; + } + +fail: + return (error); +} + +int +urtw_8187b_init(struct ifnet *ifp) +{ + struct urtw_softc *sc = ifp->if_softc; + struct urtw_rf *rf = &sc->sc_rf; + struct ieee80211com *ic = &sc->sc_ic; + int ret; + uint8_t data; + usbd_status error; + + urtw_stop(ifp, 0); + + error = urtw_8187b_update_wmm(sc); + if (error != 0) + goto fail; + error = urtw_8187b_reset(sc); + if (error) + goto fail; + + /* Applying MAC address again. */ + error = urtw_set_mode(sc, URTW_EPROM_CMD_CONFIG); + if (error) + goto fail; + IEEE80211_ADDR_COPY(ic->ic_myaddr, LLADDR(ifp->if_sadl)); + urtw_write32_m(sc, URTW_MAC0, ((uint32_t *)ic->ic_myaddr)[0]); + urtw_write16_m(sc, URTW_MAC4, ((uint32_t *)ic->ic_myaddr)[1] & 0xffff); + error = urtw_set_mode(sc, URTW_EPROM_CMD_NORMAL); + if (error) + goto fail; + + error = urtw_update_msr(sc); + if (error) + goto fail; + + error = rf->init(rf); + if (error != 0) + goto fail; + + urtw_write8_m(sc, URTW_CMD, URTW_CMD_TX_ENABLE | + URTW_CMD_RX_ENABLE); + error = urtw_intr_enable(sc); + if (error != 0) + goto fail; + + error = urtw_write8e(sc, 0x41, 0xf4); + if (error != 0) + goto fail; + error = urtw_write8e(sc, 0x40, 0x00); + if (error != 0) + goto fail; + error = urtw_write8e(sc, 0x42, 0x00); + if (error != 0) + goto fail; + error = urtw_write8e(sc, 0x42, 0x01); + if (error != 0) + goto fail; + error = urtw_write8e(sc, 0x40, 0x0f); + if (error != 0) + goto fail; + error = urtw_write8e(sc, 0x42, 0x00); + if (error != 0) + goto fail; + error = urtw_write8e(sc, 0x42, 0x01); + if (error != 0) + goto fail; + + urtw_read8_m(sc, 0xdb, &data); + urtw_write8_m(sc, 0xdb, data | (1 << 2)); + urtw_write16_idx_m(sc, 0x72, 0x59fa, 3); + urtw_write16_idx_m(sc, 0x74, 0x59d2, 3); + urtw_write16_idx_m(sc, 0x76, 0x59d2, 3); + urtw_write16_idx_m(sc, 0x78, 0x19fa, 3); + urtw_write16_idx_m(sc, 0x7a, 0x19fa, 3); + urtw_write16_idx_m(sc, 0x7c, 0x00d0, 3); + urtw_write8_m(sc, 0x61, 0); + urtw_write8_idx_m(sc, 0x80, 0x0f, 1); + urtw_write8_idx_m(sc, 0x83, 0x03, 1); + urtw_write8_m(sc, 0xda, 0x10); + urtw_write8_idx_m(sc, 0x4d, 0x08, 2); + + urtw_write32_m(sc, URTW_HSSI_PARA, 0x0600321b); + + urtw_write16_idx_m(sc, 0xec, 0x0800, 1); + + urtw_write8_m(sc, URTW_ACM_CONTROL, 0); + + /* Reset softc variables. */ + sc->sc_txidx = sc->sc_tx_low_queued = sc->sc_tx_normal_queued = 0; + sc->sc_txtimer = 0; + + if (!(sc->sc_flags & URTW_INIT_ONCE)) { + error = usbd_set_config_no(sc->sc_udev, URTW_CONFIG_NO, 0); + if (error != 0) { + printf("%s: could not set configuration no\n", + sc->sc_dev.dv_xname); + goto fail; + } + /* Get the first interface handle. */ + error = usbd_device2interface_handle(sc->sc_udev, + URTW_IFACE_INDEX, &sc->sc_iface); + if (error != 0) { + printf("%s: could not get interface handle\n", + sc->sc_dev.dv_xname); + goto fail; + } + error = urtw_open_pipes(sc); + if (error != 0) + goto fail; + ret = urtw_alloc_rx_data_list(sc); + if (error != 0) + goto fail; + ret = urtw_alloc_tx_data_list(sc); + if (error != 0) + goto fail; + sc->sc_flags |= URTW_INIT_ONCE; + } + + error = urtw_rx_enable(sc); + if (error != 0) + goto fail; + error = urtw_tx_enable(sc); + if (error != 0) + goto fail; + + ifp->if_flags &= ~IFF_OACTIVE; + ifp->if_flags |= IFF_RUNNING; + + ifp->if_timer = 1; + + if (ic->ic_opmode == IEEE80211_M_MONITOR) + ieee80211_new_state(ic, IEEE80211_S_RUN, -1); + else + ieee80211_new_state(ic, IEEE80211_S_SCAN, -1); + +fail: + return (error); +} + +usbd_status +urtw_8225v2_b_config_mac(struct urtw_softc *sc) +{ + int i; + usbd_status error; + + for (i = 0; i < nitems(urtw_8187b_regtbl); i++) { + urtw_write8_idx_m(sc, urtw_8187b_regtbl[i].reg, + urtw_8187b_regtbl[i].val, urtw_8187b_regtbl[i].idx); + } + + urtw_write16_m(sc, URTW_TID_AC_MAP, 0xfa50); + urtw_write16_m(sc, URTW_INT_MIG, 0); + + urtw_write32_idx_m(sc, 0xf0, 0, 1); + urtw_write32_idx_m(sc, 0xf4, 0, 1); + urtw_write8_idx_m(sc, 0xf8, 0, 1); + + urtw_write32_m(sc, URTW_RF_TIMING, 0x00004001); + +fail: + return (error); +} + +usbd_status +urtw_8225v2_b_init_rfe(struct urtw_softc *sc) +{ + usbd_status error; + + urtw_write16_m(sc, URTW_RF_PINS_OUTPUT, 0x0480); + urtw_write16_m(sc, URTW_RF_PINS_SELECT, 0x2488); + urtw_write16_m(sc, URTW_RF_PINS_ENABLE, 0x1fff); + usbd_delay_ms(sc->sc_udev, 100); + +fail: + return (error); +} + +usbd_status +urtw_8225v2_b_update_chan(struct urtw_softc *sc) +{ + struct ieee80211com *ic = &sc->sc_ic; + struct ieee80211_channel *c = ic->ic_ibss_chan; + uint8_t aifs, difs, eifs, sifs, slot; + usbd_status error; + + urtw_write8_m(sc, URTW_SIFS, 0x22); + + sifs = 0xa; + if (IEEE80211_IS_CHAN_G(c)) { + slot = 0x9; + difs = 0x1c; + eifs = 0x5b; + } else { + slot = 0x14; + difs = 0x32; + eifs = 0x5b; + } + aifs = (2 * slot) + sifs; + + urtw_write8_m(sc, URTW_SLOT, slot); + + urtw_write8_m(sc, URTW_AC_VO, aifs); + urtw_write8_m(sc, URTW_AC_VI, aifs); + urtw_write8_m(sc, URTW_AC_BE, aifs); + urtw_write8_m(sc, URTW_AC_BK, aifs); + + urtw_write8_m(sc, URTW_DIFS, difs); + urtw_write8_m(sc, URTW_8187B_EIFS, eifs); + +fail: + return (error); +} + +usbd_status +urtw_8225v2_b_rf_init(struct urtw_rf *rf) +{ + struct urtw_softc *sc = rf->rf_sc; + int i; + uint8_t data; + usbd_status error; + + /* Set up ACK rate, retry limit, TX AGC, TX antenna. */ + urtw_write16_m(sc, URTW_8187B_BRSR, 0x0fff); + urtw_read8_m(sc, URTW_CW_CONF, &data); + urtw_write8_m(sc, URTW_CW_CONF, data | + URTW_CW_CONF_PERPACKET_RETRY); + urtw_read8_m(sc, URTW_TX_AGC_CTL, &data); + urtw_write8_m(sc, URTW_TX_AGC_CTL, data | + URTW_TX_AGC_CTL_PERPACKET_GAIN | + URTW_TX_AGC_CTL_PERPACKET_ANTSEL); + + /* Auto rate fallback control. */ + urtw_write16_idx_m(sc, URTW_ARFR, 0x0fff, 1); /* 1M ~ 54M */ + urtw_read8_m(sc, URTW_RATE_FALLBACK, &data); + urtw_write8_m(sc, URTW_RATE_FALLBACK, data | + URTW_RATE_FALLBACK_ENABLE); + + urtw_write16_m(sc, URTW_BEACON_INTERVAL, 100); + urtw_write16_m(sc, URTW_ATIM_WND, 2); + urtw_write16_idx_m(sc, URTW_FEMR, 0xffff, 1); + + error = urtw_set_mode(sc, URTW_EPROM_CMD_CONFIG); + if (error) + goto fail; + urtw_read8_m(sc, URTW_CONFIG1, &data); + urtw_write8_m(sc, URTW_CONFIG1, (data & 0x3f) | 0x80); + error = urtw_set_mode(sc, URTW_EPROM_CMD_NORMAL); + if (error) + goto fail; + + urtw_write8_m(sc, URTW_WPA_CONFIG, 0); + urtw_8225v2_b_config_mac(sc); + urtw_write16_idx_m(sc, URTW_RFSW_CTRL, 0x569a, 2); + + error = urtw_set_mode(sc, URTW_EPROM_CMD_CONFIG); + if (error) + goto fail; + urtw_read8_m(sc, URTW_CONFIG3, &data); + urtw_write8_m(sc, URTW_CONFIG3, data | URTW_CONFIG3_ANAPARAM_WRITE); + error = urtw_set_mode(sc, URTW_EPROM_CMD_NORMAL); + if (error) + goto fail; + + urtw_8225v2_b_init_rfe(sc); + + for (i = 0; i < nitems(urtw_8225v2_b_rf); i++) { + urtw_8225_write(sc, urtw_8225v2_b_rf[i].reg, + urtw_8225v2_b_rf[i].val); + } + + for (i = 0; i < nitems(urtw_8225v2_rxgain); i++) { + urtw_8225_write(sc, 0x1, (uint8_t)(i + 1)); + urtw_8225_write(sc, 0x2, urtw_8225v2_rxgain[i]); + } + + urtw_8225_write(sc, 0x03, 0x080); + urtw_8225_write(sc, 0x05, 0x004); + urtw_8225_write(sc, 0x00, 0x0b7); + urtw_8225_write(sc, 0x02, 0xc4d); + urtw_8225_write(sc, 0x02, 0x44d); + urtw_8225_write(sc, 0x00, 0x2bf); + + urtw_write8_m(sc, URTW_TX_GAIN_CCK, 0x03); + urtw_write8_m(sc, URTW_TX_GAIN_OFDM, 0x07); + urtw_write8_m(sc, URTW_TX_ANTENNA, 0x03); + + urtw_8187_write_phy_ofdm(sc, 0x80, 0x12); + for (i = 0; i < nitems(urtw_8225v2_agc); i++) { + urtw_8187_write_phy_ofdm(sc, 0x0f, urtw_8225v2_agc[i]); + urtw_8187_write_phy_ofdm(sc, 0x0e, (uint8_t)i + 0x80); + urtw_8187_write_phy_ofdm(sc, 0x0e, 0); + } + urtw_8187_write_phy_ofdm(sc, 0x80, 0x10); + + for (i = 0; i < nitems(urtw_8225v2_ofdm); i++) + urtw_8187_write_phy_ofdm(sc, i, urtw_8225v2_ofdm[i]); + + urtw_8225v2_b_update_chan(sc); + + urtw_8187_write_phy_ofdm(sc, 0x97, 0x46); + urtw_8187_write_phy_ofdm(sc, 0xa4, 0xb6); + urtw_8187_write_phy_ofdm(sc, 0x85, 0xfc); + urtw_8187_write_phy_cck(sc, 0xc1, 0x88); + + error = urtw_8225v2_b_rf_set_chan(rf, 1); +fail: + return (error); +} + +usbd_status +urtw_8225v2_b_rf_set_chan(struct urtw_rf *rf, int chan) +{ + struct urtw_softc *sc = rf->rf_sc; + usbd_status error; + + error = urtw_8225v2_b_set_txpwrlvl(sc, chan); + if (error) + goto fail; + + urtw_8225_write(sc, 0x7, urtw_8225_channel[chan]); + /* + * Delay removed from 8185 to 8187. + * usbd_delay_ms(sc->sc_udev, 10); + */ + + urtw_write16_m(sc, URTW_AC_VO, 0x5114); + urtw_write16_m(sc, URTW_AC_VI, 0x5114); + urtw_write16_m(sc, URTW_AC_BE, 0x5114); + urtw_write16_m(sc, URTW_AC_BK, 0x5114); + +fail: + return (error); +} + +usbd_status +urtw_8225v2_b_set_txpwrlvl(struct urtw_softc *sc, int chan) +{ + int i; + uint8_t *cck_pwrtable; + uint8_t cck_pwrlvl_min, cck_pwrlvl_max, ofdm_pwrlvl_min, + ofdm_pwrlvl_max; + int8_t cck_pwrlvl = sc->sc_txpwr_cck[chan] & 0xff; + int8_t ofdm_pwrlvl = sc->sc_txpwr_ofdm[chan] & 0xff; + usbd_status error; + + if (sc->sc_hwrev & URTW_HWREV_8187B_B) { + cck_pwrlvl_min = 0; + cck_pwrlvl_max = 15; + ofdm_pwrlvl_min = 2; + ofdm_pwrlvl_max = 17; + } else { + cck_pwrlvl_min = 7; + cck_pwrlvl_max = 22; + ofdm_pwrlvl_min = 10; + ofdm_pwrlvl_max = 25; + } + + /* CCK power setting */ + cck_pwrlvl = (cck_pwrlvl > (cck_pwrlvl_max - cck_pwrlvl_min)) ? + cck_pwrlvl_max : (cck_pwrlvl + cck_pwrlvl_min); + + cck_pwrlvl += sc->sc_txpwr_cck_base; + cck_pwrlvl = (cck_pwrlvl > 35) ? 35 : cck_pwrlvl; + cck_pwrlvl = (cck_pwrlvl < 0) ? 0 : cck_pwrlvl; + + cck_pwrtable = (chan == 14) ? urtw_8225v2_txpwr_cck_ch14 : + urtw_8225v2_txpwr_cck; + + if (sc->sc_hwrev & URTW_HWREV_8187B_B) { + if (cck_pwrlvl <= 6) + ; /* do nothing */ + else if (cck_pwrlvl <= 11) + cck_pwrtable += 8; + else + cck_pwrtable += 16; + } else { + if (cck_pwrlvl <= 5) + ; /* do nothing */ + else if (cck_pwrlvl <= 11) + cck_pwrtable += 8; + else if (cck_pwrlvl <= 17) + cck_pwrtable += 16; + else + cck_pwrtable += 24; + } + + for (i = 0; i < 8; i++) { + urtw_8187_write_phy_cck(sc, 0x44 + i, cck_pwrtable[i]); + } + + urtw_write8_m(sc, URTW_TX_GAIN_CCK, + urtw_8225v2_tx_gain_cck_ofdm[cck_pwrlvl] << 1); + /* + * Delay removed from 8185 to 8187. + * usbd_delay_ms(sc->sc_udev, 1); + */ + + /* OFDM power setting */ + ofdm_pwrlvl = (ofdm_pwrlvl > (ofdm_pwrlvl_max - ofdm_pwrlvl_min)) ? + ofdm_pwrlvl_max : ofdm_pwrlvl + ofdm_pwrlvl_min; + + ofdm_pwrlvl += sc->sc_txpwr_ofdm_base; + ofdm_pwrlvl = (ofdm_pwrlvl > 35) ? 35 : ofdm_pwrlvl; + ofdm_pwrlvl = (ofdm_pwrlvl < 0) ? 0 : ofdm_pwrlvl; + + urtw_write8_m(sc, URTW_TX_GAIN_OFDM, + urtw_8225v2_tx_gain_cck_ofdm[ofdm_pwrlvl] << 1); + + if (sc->sc_hwrev & URTW_HWREV_8187B_B) { + if (ofdm_pwrlvl <= 11) { + urtw_8187_write_phy_ofdm(sc, 0x87, 0x60); + urtw_8187_write_phy_ofdm(sc, 0x89, 0x60); + } else { + urtw_8187_write_phy_ofdm(sc, 0x87, 0x5c); + urtw_8187_write_phy_ofdm(sc, 0x89, 0x5c); + } + } else { + if (ofdm_pwrlvl <= 11) { + urtw_8187_write_phy_ofdm(sc, 0x87, 0x5c); + urtw_8187_write_phy_ofdm(sc, 0x89, 0x5c); + } else if (ofdm_pwrlvl <= 17) { + urtw_8187_write_phy_ofdm(sc, 0x87, 0x54); + urtw_8187_write_phy_ofdm(sc, 0x89, 0x54); + } else { + urtw_8187_write_phy_ofdm(sc, 0x87, 0x50); + urtw_8187_write_phy_ofdm(sc, 0x89, 0x50); + } + } + + /* + * Delay removed from 8185 to 8187. + * usbd_delay_ms(sc->sc_udev, 1); + */ +fail: + return (error); +} + diff --git a/sys/dev/usb/if_urtwreg.h b/sys/dev/usb/if_urtwreg.h index a1d79f4085a..b6e0ee16e59 100644 --- a/sys/dev/usb/if_urtwreg.h +++ b/sys/dev/usb/if_urtwreg.h @@ -1,6 +1,7 @@ -/* $OpenBSD: if_urtwreg.h,v 1.11 2009/06/05 01:21:54 martynas Exp $ */ +/* $OpenBSD: if_urtwreg.h,v 1.12 2009/06/06 12:06:28 martynas Exp $ */ /*- + * Copyright (c) 2009 Martynas Venckus <martynas@openbsd.org> * Copyright (c) 2008 Weongyo Jeong <weongyo@FreeBSD.org> * * Permission to use, copy, modify, and distribute this software for any @@ -25,6 +26,10 @@ #define URTW_HWREV_8187 0x01 #define URTW_HWREV_8187_B 0x02 #define URTW_HWREV_8187_D 0x04 +#define URTW_HWREV_8187B 0x08 +#define URTW_HWREV_8187B_B 0x10 +#define URTW_HWREV_8187B_D 0x20 +#define URTW_HWREV_8187B_E 0x40 /* * Registers specific to RTL8187 and RTL8187B. @@ -37,8 +42,10 @@ #define URTW_MAC5 0x0005 /* 1 byte */ #define URTW_8187_BRSR 0x002c /* 2 byte */ #define URTW_BRSR_MBR_8185 (0x0fff) +#define URTW_8187B_EIFS 0x002d /* 1 byte */ #define URTW_BSSID 0x002e /* 6 byte */ #define URTW_RESP_RATE 0x0034 /* 1 byte */ +#define URTW_8187B_BRSR 0x0034 /* 2 byte */ #define URTW_RESP_MAX_RATE_SHIFT (4) #define URTW_RESP_MIN_RATE_SHIFT (0) #define URTW_8187_EIFS 0x0035 /* 1 byte */ @@ -51,6 +58,10 @@ #define URTW_TX_HWREV_MASK (7 << 25) #define URTW_TX_HWREV_8187_D (5 << 25) #define URTW_TX_HWREV_8187B_D (6 << 25) +#define URTW_TX_DURPROCMODE (1 << 30) +#define URTW_TX_DISREQQSIZE (1 << 28) +#define URTW_TX_SHORTRETRY (7 << 8) +#define URTW_TX_LONGRETRY (7 << 0) #define URTW_TX_LOOPBACK_SHIFT (17) #define URTW_TX_LOOPBACK_NONE (0 << URTW_TX_LOOPBACK_SHIFT) #define URTW_TX_LOOPBACK_MAC (1 << URTW_TX_LOOPBACK_SHIFT) @@ -119,9 +130,11 @@ #define URTW_EPROM_WRITEBIT (0x2) #define URTW_EPROM_CK (0x4) #define URTW_EPROM_CS (0x8) +#define URTW_CONFIG1 0x0052 /* 1 byte */ #define URTW_CONFIG2 0x0053 /* 1 byte */ #define URTW_ANAPARAM 0x0054 /* 4 byte */ #define URTW_8187_8225_ANAPARAM_ON (0xa0000a59) +#define URTW_8187B_8225_ANAPARAM_ON (0x45090658) #define URTW_MSR 0x0058 /* 1 byte */ #define URTW_MSR_LINK_MASK ((1 << 2) | (1 << 3)) #define URTW_MSR_LINK_SHIFT (2) @@ -129,12 +142,15 @@ #define URTW_MSR_LINK_ADHOC (1 << URTW_MSR_LINK_SHIFT) #define URTW_MSR_LINK_STA (2 << URTW_MSR_LINK_SHIFT) #define URTW_MSR_LINK_HOSTAP (3 << URTW_MSR_LINK_SHIFT) +#define URTW_MSR_LINK_ENEDCA (4 << URTW_MSR_LINK_SHIFT) #define URTW_CONFIG3 0x0059 /* 1 byte */ #define URTW_CONFIG3_ANAPARAM_WRITE (0x40) #define URTW_CONFIG3_ANAPARAM_W_SHIFT (6) +#define URTW_CONFIG3_GNT_SELECT (0x80) #define URTW_PSR 0x005e /* 1 byte */ #define URTW_ANAPARAM2 0x0060 /* 4 byte */ #define URTW_8187_8225_ANAPARAM2_ON (0x860c7312) +#define URTW_8187B_8225_ANAPARAM2_ON (0x727f3f52) #define URTW_BEACON_INTERVAL 0x0070 /* 2 byte */ #define URTW_ATIM_WND 0x0072 /* 2 byte */ #define URTW_BEACON_INTERVAL_TIME 0x0074 /* 2 byte */ @@ -150,6 +166,7 @@ #define URTW_RF_TIMING 0x008c /* 4 byte */ #define URTW_GP_ENABLE 0x0090 /* 1 byte */ #define URTW_GPIO 0x0091 /* 1 byte */ +#define URTW_HSSI_PARA 0x0094 /* 4 byte */ #define URTW_TX_AGC_CTL 0x009c /* 1 byte */ #define URTW_TX_AGC_CTL_PERPACKET_GAIN (0x1) #define URTW_TX_AGC_CTL_PERPACKET_ANTSEL (0x2) @@ -166,7 +183,24 @@ #define URTW_CW_CONF_PERPACKET_CW (0x1) #define URTW_CW_VAL 0x00bd /* 1 byte */ #define URTW_RATE_FALLBACK 0x00be /* 1 byte */ +#define URTW_RATE_FALLBACK_ENABLE (0x80) +#define URTW_ACM_CONTROL 0x00bf /* 1 byte */ +#define URTW_8187B_HWREV 0x00e1 /* 1 byte */ +#define URTW_8187B_HWREV_8187B_B (0x0) +#define URTW_8187B_HWREV_8187B_D (0x1) +#define URTW_8187B_HWREV_8187B_E (0x2) +#define URTW_INT_MIG 0x00e2 /* 2 byte */ +#define URTW_TID_AC_MAP 0x00e8 /* 2 byte */ +#define URTW_ANAPARAM3 0x00ee /* 4 byte */ +#define URTW_8187B_8225_ANAPARAM3_ON (0x0) #define URTW_TALLY_SEL 0x00fc /* 1 byte */ +#define URTW_AC_VO 0x00f0 /* 1 byte */ +#define URTW_AC_VI 0x00f4 /* 1 byte */ +#define URTW_AC_BE 0x00f8 /* 1 byte */ +#define URTW_AC_BK 0x00fc /* 1 byte */ +#define URTW_FEMR 0x01d4 /* 2 byte */ +#define URTW_ARFR 0x01e0 /* 2 byte */ +#define URTW_RFSW_CTRL 0x0272 /* 2 byte */ /* for EEPROM */ #define URTW_EPROM_TXPW_BASE 0x05 |