summaryrefslogtreecommitdiff
path: root/sys/dev/ic
diff options
context:
space:
mode:
authorDamien Bergamini <damien@cvs.openbsd.org>2010-05-10 18:17:11 +0000
committerDamien Bergamini <damien@cvs.openbsd.org>2010-05-10 18:17:11 +0000
commit74ec6f5d77854dfd44aaf3ee3e8836f9a15c38ff (patch)
tree960d54140a157bb43e7e3aaae68be3e4456913e6 /sys/dev/ic
parent1d1fef4cd5b04df1795c1fc7b0d9b92a1d14cb19 (diff)
additionnal bits and fixes for RT3090.
some adapters labelled RT3090 have a MAC version equal to RT3071. still no luck receiving frames on my AzureWave AW-NE762H though :(
Diffstat (limited to 'sys/dev/ic')
-rw-r--r--sys/dev/ic/rt2860.c127
-rw-r--r--sys/dev/ic/rt2860reg.h36
2 files changed, 110 insertions, 53 deletions
diff --git a/sys/dev/ic/rt2860.c b/sys/dev/ic/rt2860.c
index 728055f8917..5651f1c67cc 100644
--- a/sys/dev/ic/rt2860.c
+++ b/sys/dev/ic/rt2860.c
@@ -1,4 +1,4 @@
-/* $OpenBSD: rt2860.c,v 1.52 2010/04/20 22:05:43 tedu Exp $ */
+/* $OpenBSD: rt2860.c,v 1.53 2010/05/10 18:17:10 damien Exp $ */
/*-
* Copyright (c) 2007-2010 Damien Bergamini <damien.bergamini@free.fr>
@@ -222,9 +222,9 @@ rt2860_attach(void *xsc, int id)
sc->mac_ver = tmp >> 16;
sc->mac_rev = tmp & 0xffff;
#ifdef RAL_DEBUG
- /* temporarily enable debug for >=RT3090 */
- if (sc->mac_ver >= 0x3090)
- rt2860_debug = 1;
+ /* temporarily enable debug for >=RT3071 */
+ if (sc->mac_ver >= 0x3071)
+ rt2860_debug = 10;
#endif
if (sc->mac_ver != 0x2860 &&
(id == PCI_PRODUCT_RALINK_RT2890 ||
@@ -912,7 +912,7 @@ rt2860_newstate(struct ieee80211com *ic, enum ieee80211_state nstate, int arg)
return sc->sc_newstate(ic, nstate, arg);
}
-/* Read 16-bit from eFUSE ROM (>=RT3090 only.) */
+/* Read 16-bit from eFUSE ROM (>=RT3071 only.) */
uint16_t
rt3090_efuse_read_2(struct rt2860_softc *sc, uint16_t addr)
{
@@ -2128,7 +2128,7 @@ rt2860_select_chan_group(struct rt2860_softc *sc, int group)
/* set initial AGC value */
if (group == 0) { /* 2GHz band */
- if (sc->mac_rev >= 0x3090)
+ if (sc->mac_ver >= 0x3071)
agc = 0x1c + sc->lna[0] * 2;
else
agc = 0x2e + sc->lna[0];
@@ -2232,13 +2232,13 @@ rt3090_set_chan(struct rt2860_softc *sc, u_int chan)
rf = rt3090_rf_read(sc, 1);
rf &= ~0xfc;
if (sc->ntxchains == 1)
- rf |= 1 << 7 | 1 << 5; /* 1T: disable Tx chains 2 & 3 */
+ rf |= RT3070_TX1_PD | RT3070_TX2_PD;
else if (sc->ntxchains == 2)
- rf |= 1 << 7; /* 2T: disable Tx chain 3 */
+ rf |= RT3070_TX2_PD;
if (sc->nrxchains == 1)
- rf |= 1 << 6 | 1 << 4; /* 1R: disable Rx chains 2 & 3 */
+ rf |= RT3070_RX1_PD | RT3070_RX2_PD;
else if (sc->nrxchains == 2)
- rf |= 1 << 6; /* 2R: disable Rx chain 3 */
+ rf |= RT3070_RX2_PD;
rt3090_rf_write(sc, 1, rf);
/* set RF offset */
@@ -2256,7 +2256,7 @@ rt3090_set_chan(struct rt2860_softc *sc, u_int chan)
/* enable RF tuning */
rf = rt3090_rf_read(sc, 7);
- rt3090_rf_write(sc, 7, rf | 0x01);
+ rt3090_rf_write(sc, 7, rf | RT3070_TUNE);
}
int
@@ -2376,42 +2376,55 @@ rt3090_rf_wakeup(struct rt2860_softc *sc)
if (sc->mac_ver == 0x3593) {
/* enable VCO */
rf = rt3090_rf_read(sc, 1);
- rt3090_rf_write(sc, 1, rf | 0x01);
+ rt3090_rf_write(sc, 1, rf | RT3593_VCO);
+
/* initiate VCO calibration */
rf = rt3090_rf_read(sc, 3);
- rt3090_rf_write(sc, 3, rf | 0x80);
+ rt3090_rf_write(sc, 3, rf | RT3593_VCOCAL);
+
/* enable VCO bias current control */
rf = rt3090_rf_read(sc, 6);
- rt3090_rf_write(sc, 6, rf | 0x40);
+ rt3090_rf_write(sc, 6, rf | RT3593_VCO_IC);
+
/* initiate res calibration */
rf = rt3090_rf_read(sc, 2);
- rt3090_rf_write(sc, 2, rf | 0x80);
- /* set reference current control */
+ rt3090_rf_write(sc, 2, rf | RT3593_RESCAL);
+
+ /* set reference current control to 0.33 mA */
rf = rt3090_rf_read(sc, 22);
- rt3090_rf_write(sc, 22, (rf & ~0xe0) | 0x80);
+ rf &= ~RT3593_CP_IC_MASK;
+ rf |= 1 << RT3593_CP_IC_SHIFT;
+ rt3090_rf_write(sc, 22, rf);
+
/* enable RX CTB */
rf = rt3090_rf_read(sc, 46);
- rt3090_rf_write(sc, 46, rf | 0x20);
- rf = rt3090_rf_read(sc, 20);
- rt3090_rf_write(sc, 20, rf & ~0xee);
+ rt3090_rf_write(sc, 46, rf | RT3593_RX_CTB);
- rf = rt3090_rf_read(sc, 27);
- rf &= ~0x77;
- if (sc->mac_rev < 0x0211)
- rf |= 0x03;
- rt3090_rf_write(sc, 27, rf);
+ rf = rt3090_rf_read(sc, 20);
+ rf &= ~(RT3593_LDO_RF_VC_MASK | RT3593_LDO_PLL_VC_MASK);
+ rt3090_rf_write(sc, 20, rf);
} else {
/* enable RF block */
rf = rt3090_rf_read(sc, 1);
- rt3090_rf_write(sc, 1, rf | 0x01);
+ rt3090_rf_write(sc, 1, rf | RT3070_RF_BLOCK);
+
/* enable VCO bias current control */
rf = rt3090_rf_read(sc, 7);
rt3090_rf_write(sc, 7, rf | 0x30);
+
rf = rt3090_rf_read(sc, 9);
rt3090_rf_write(sc, 9, rf | 0x0e);
+
/* enable RX CTB */
rf = rt3090_rf_read(sc, 21);
- rt3090_rf_write(sc, 21, rf | 0x80);
+ rt3090_rf_write(sc, 21, rf | RT3070_RX_CTB);
+
+ /* fix Tx to Rx IQ glitch by raising RF voltage */
+ rf = rt3090_rf_read(sc, 27);
+ rf &= ~0x77;
+ if (sc->mac_rev < 0x0211)
+ rf |= 0x03;
+ rt3090_rf_write(sc, 27, rf);
}
if (sc->patch_dac && sc->mac_rev < 0x0211) {
tmp = RAL_READ(sc, RT3070_LDO_CFG0);
@@ -2435,7 +2448,7 @@ rt3090_filter_calib(struct rt2860_softc *sc, uint8_t init, uint8_t target,
/* enable baseband loopback mode */
rf22 = rt3090_rf_read(sc, 22);
- rt3090_rf_write(sc, 22, rf22 | 0x01);
+ rt3090_rf_write(sc, 22, rf22 | RT3070_BB_LOOPBACK);
/* set power and frequency of passband test tone */
rt2860_mcu_bbp_write(sc, 24, 0x00);
@@ -2480,7 +2493,7 @@ rt3090_filter_calib(struct rt2860_softc *sc, uint8_t init, uint8_t target,
/* disable baseband loopback mode */
rf22 = rt3090_rf_read(sc, 22);
- rt3090_rf_write(sc, 22, rf22 & ~0x01);
+ rt3090_rf_write(sc, 22, rf22 & ~RT3070_BB_LOOPBACK);
return 0;
}
@@ -2844,7 +2857,7 @@ rt2860_read_eeprom(struct rt2860_softc *sc)
/* check whether the ROM is eFUSE ROM or EEPROM */
sc->sc_srom_read = rt2860_eeprom_read_2;
- if (sc->mac_ver >= 0x3090) {
+ if (sc->mac_ver >= 0x3071) {
tmp = RAL_READ(sc, RT3070_EFUSE_CTRL);
DPRINTF(("EFUSE_CTRL=0x%08x\n", tmp));
if (tmp & RT3070_SEL_EFUSE)
@@ -2877,7 +2890,7 @@ rt2860_read_eeprom(struct rt2860_softc *sc)
sc->bbp[i].reg = val >> 8;
DPRINTF(("BBP%d=0x%02x\n", sc->bbp[i].reg, sc->bbp[i].val));
}
- if (sc->mac_ver >= 0x3090) {
+ if (sc->mac_ver >= 0x3071) {
/* read vendor RF settings */
for (i = 0; i < 10; i++) {
val = rt2860_srom_read(sc, RT3071_EEPROM_RF_BASE + i);
@@ -2917,7 +2930,7 @@ rt2860_read_eeprom(struct rt2860_softc *sc)
sc->rf_rev = RT3070_RF_3053;
sc->ntxchains = 3;
sc->nrxchains = 3;
- } else if (sc->mac_ver >= 0x3090) {
+ } else if (sc->mac_ver >= 0x3071) {
/* default to RF3020 1T1R */
sc->rf_rev = RT3070_RF_3020;
sc->ntxchains = 1;
@@ -2950,18 +2963,18 @@ rt2860_read_eeprom(struct rt2860_softc *sc)
/* check if we have a hardware radio switch */
sc->rfswitch = val & 1;
}
-
if (sc->sc_flags & RT2860_ADVANCED_PS) {
/* read PCIe power save level */
val = rt2860_srom_read(sc, RT2860_EEPROM_PCIE_PSLEVEL);
if ((val & 0xff) != 0xff) {
sc->pslevel = val & 0x3;
val = rt2860_srom_read(sc, RT2860_EEPROM_REV);
- if (val >> 8 != 0x92 || !(val & 0x80))
+ if ((val & 0xff80) != 0x9280)
sc->pslevel = MIN(sc->pslevel, 1);
DPRINTF(("EEPROM PCIe PS Level=%d\n", sc->pslevel));
}
}
+
/* read power settings for 2GHz channels */
for (i = 0; i < 14; i += 2) {
val = rt2860_srom_read(sc,
@@ -3094,7 +3107,7 @@ rt2860_read_eeprom(struct rt2860_softc *sc)
sc->rssi_2ghz[0] = val & 0xff; /* Ant A */
sc->rssi_2ghz[1] = val >> 8; /* Ant B */
val = rt2860_srom_read(sc, RT2860_EEPROM_RSSI2_2GHZ);
- if (sc->mac_ver >= 0x3090) {
+ if (sc->mac_ver >= 0x3071) {
/*
* On RT3090 chips (limited to 2 Rx chains), this ROM
* field contains the Tx mixer gain for the 2GHz band.
@@ -3114,7 +3127,10 @@ rt2860_read_eeprom(struct rt2860_softc *sc)
sc->lna[3] = val >> 8; /* channel group 3 */
val = rt2860_srom_read(sc, RT2860_EEPROM_LNA);
- sc->lna[0] = val & 0xff; /* channel group 0 */
+ if (sc->mac_ver >= 0x3071)
+ sc->lna[0] = RT3090_DEF_LNA;
+ else /* channel group 0 */
+ sc->lna[0] = val & 0xff;
sc->lna[1] = val >> 8; /* channel group 1 */
/* fix broken 5GHz LNA entries */
@@ -3171,7 +3187,7 @@ rt2860_bbp_init(struct rt2860_softc *sc)
if (sc->mac_ver == 0x2860 && sc->mac_rev != 0x0101)
rt2860_mcu_bbp_write(sc, 84, 0x19);
- if (sc->mac_ver >= 0x3090) {
+ if (sc->mac_ver >= 0x3071) {
rt2860_mcu_bbp_write(sc, 79, 0x13);
rt2860_mcu_bbp_write(sc, 80, 0x05);
rt2860_mcu_bbp_write(sc, 81, 0x33);
@@ -3258,13 +3274,13 @@ rt2860_init(struct ifnet *ifp)
#endif
}
}
+ RAL_WRITE(sc, RT2860_PWR_PIN_CFG, RT2860_IO_RA_PE);
+ /* disable DMA */
tmp = RAL_READ(sc, RT2860_WPDMA_GLO_CFG);
tmp &= 0xff0;
RAL_WRITE(sc, RT2860_WPDMA_GLO_CFG, tmp);
- RAL_WRITE(sc, RT2860_WPDMA_RST_IDX, 0xffffffff);
-
/* PBF hardware reset */
RAL_WRITE(sc, RT2860_SYS_CTRL, 0xe1f);
RAL_BARRIER_WRITE(sc);
@@ -3321,6 +3337,19 @@ rt2860_init(struct ifnet *ifp)
for (i = 0; i < nitems(rt2860_def_mac); i++)
RAL_WRITE(sc, rt2860_def_mac[i].reg, rt2860_def_mac[i].val);
+ if (sc->mac_ver >= 0x3071) {
+ /* set delay of PA_PE assertion to 1us (unit of 0.25us) */
+ RAL_WRITE(sc, RT2860_TX_SW_CFG0,
+ 4 << RT2860_DLY_PAPE_EN_SHIFT);
+ }
+
+ if (!(RAL_READ(sc, RT2860_PCI_CFG) & RT2860_PCI_CFG_PCI)) {
+ sc->sc_flags |= RT2860_PCIE;
+ /* PCIe has different clock cycle count than PCI */
+ tmp = RAL_READ(sc, RT2860_US_CYC_CNT);
+ tmp = (tmp & ~0xff) | 0x7d;
+ RAL_WRITE(sc, RT2860_US_CYC_CNT, tmp);
+ }
/* wait while MAC is busy */
for (ntries = 0; ntries < 100; ntries++) {
@@ -3335,14 +3364,6 @@ rt2860_init(struct ifnet *ifp)
return ETIMEDOUT;
}
- if (!(RAL_READ(sc, RT2860_PCI_CFG) & RT2860_PCI_CFG_PCI)) {
- sc->sc_flags |= RT2860_PCIE;
- /* PCIe has different clock cycle count than PCI */
- tmp = RAL_READ(sc, RT2860_US_CYC_CNT);
- tmp = (tmp & ~0xff) | 0x7d;
- RAL_WRITE(sc, RT2860_US_CYC_CNT, tmp);
- }
-
/* clear Host to MCU mailbox */
RAL_WRITE(sc, RT2860_H2M_BBPAGENT, 0);
RAL_WRITE(sc, RT2860_H2M_MAILBOX, 0);
@@ -3410,7 +3431,9 @@ rt2860_init(struct ifnet *ifp)
}
/* select Main antenna for 1T1R devices */
- if (sc->rf_rev == RT3070_RF_3020)
+ if (sc->rf_rev == RT3070_RF_2020 ||
+ sc->rf_rev == RT3070_RF_3020 ||
+ sc->rf_rev == RT3070_RF_3320)
rt3090_set_rx_antenna(sc, 0);
/* send LEDs operating mode to microcontroller */
@@ -3418,13 +3441,13 @@ rt2860_init(struct ifnet *ifp)
rt2860_mcu_cmd(sc, RT2860_MCU_CMD_LED2, sc->led[1], 0);
rt2860_mcu_cmd(sc, RT2860_MCU_CMD_LED3, sc->led[2], 0);
- if (sc->mac_ver >= 0x3090)
+ if (sc->mac_ver >= 0x3071)
rt3090_rf_init(sc);
rt2860_mcu_cmd(sc, RT2860_MCU_CMD_SLEEP, 0x02ff, 1);
rt2860_mcu_cmd(sc, RT2860_MCU_CMD_WAKEUP, 0, 1);
- if (sc->mac_ver >= 0x3090)
+ if (sc->mac_ver >= 0x3071)
rt3090_rf_wakeup(sc);
/* disable non-existing Rx chains */
@@ -3446,7 +3469,7 @@ rt2860_init(struct ifnet *ifp)
bbp1 = (bbp1 & ~(1 << 3)) | 1 << 4;
rt2860_mcu_bbp_write(sc, 1, bbp1);
- if (sc->mac_ver >= 0x3090)
+ if (sc->mac_ver >= 0x3071)
rt3090_rf_setup(sc);
/* select default channel */
@@ -3664,7 +3687,7 @@ rt2860_switch_chan(struct rt2860_softc *sc, struct ieee80211_channel *c)
if (chan == 0 || chan == IEEE80211_CHAN_ANY)
return;
- if (sc->mac_ver >= 0x3090)
+ if (sc->mac_ver >= 0x3071)
rt3090_set_chan(sc, chan);
else
rt2860_set_chan(sc, chan);
diff --git a/sys/dev/ic/rt2860reg.h b/sys/dev/ic/rt2860reg.h
index 31f6a4e7ff9..dec490802c7 100644
--- a/sys/dev/ic/rt2860reg.h
+++ b/sys/dev/ic/rt2860reg.h
@@ -1,4 +1,4 @@
-/* $OpenBSD: rt2860reg.h,v 1.29 2010/04/06 19:40:51 damien Exp $ */
+/* $OpenBSD: rt2860reg.h,v 1.30 2010/05/10 18:17:10 damien Exp $ */
/*-
* Copyright (c) 2007
@@ -76,6 +76,7 @@
#define RT3070_EFUSE_DATA1 0x0594
#define RT3070_EFUSE_DATA2 0x0598
#define RT3070_EFUSE_DATA3 0x059c
+#define RT3090_OSC_CTRL 0x05a4
#define RT3070_LDO_CFG0 0x05d4
#define RT3070_GPIO_SWITCH 0x05dc
@@ -699,6 +700,11 @@
#define RT3070_TX0_PD (1 << 3)
#define RT3070_RX1_PD (1 << 4)
#define RT3070_TX1_PD (1 << 5)
+#define RT3070_RX2_PD (1 << 6)
+#define RT3070_TX2_PD (1 << 7)
+
+/* possible flags for RT3020 RF register 7 */
+#define RT3070_TUNE (1 << 0)
/* possible flags for RT3020 RF register 15 */
#define RT3070_TX_LO2 (1 << 3)
@@ -711,7 +717,35 @@
/* possible flags for RT3020 RF register 21 */
#define RT3070_RX_LO2 (1 << 3)
+#define RT3070_RX_CTB (1 << 7)
+
+/* possible flags for RT3020 RF register 22 */
+#define RT3070_BB_LOOPBACK (1 << 0)
+
+/* possible flags for RT3053 RF register 1 */
+#define RT3593_VCO (1 << 0)
+
+/* possible flags for RT3053 RF register 2 */
+#define RT3593_RESCAL (1 << 7)
+
+/* possible flags for RT3053 RF register 3 */
+#define RT3593_VCOCAL (1 << 7)
+
+/* possible flags for RT3053 RF register 6 */
+#define RT3593_VCO_IC (1 << 6)
+
+/* possible flags for RT3053 RF register 20 */
+#define RT3593_LDO_PLL_VC_MASK 0x0e
+#define RT3593_LDO_RF_VC_MASK 0xe0
+
+/* possible flags for RT3053 RF register 22 */
+#define RT3593_CP_IC_MASK 0xe0
+#define RT3593_CP_IC_SHIFT 5
+
+/* possible flags for RT3053 RF register 46 */
+#define RT3593_RX_CTB (1 << 5)
+#define RT3090_DEF_LNA 10
/* RT2860 TX descriptor */
struct rt2860_txd {