|
@@ -891,6 +891,74 @@ static void ar9003_hw_tx_iq_cal_reload(struct ath_hw *ah)
|
|
|
AR_PHY_RX_IQCAL_CORR_B0_LOOPBACK_IQCORR_EN, 0x1);
|
|
|
}
|
|
|
|
|
|
+static void ar9003_hw_manual_peak_cal(struct ath_hw *ah, u8 chain, bool is_2g)
|
|
|
+{
|
|
|
+ int offset[8], total = 0, test;
|
|
|
+ int agc_out, i;
|
|
|
+
|
|
|
+ REG_RMW_FIELD(ah, AR_PHY_65NM_RXRF_GAINSTAGES(chain),
|
|
|
+ AR_PHY_65NM_RXRF_GAINSTAGES_RX_OVERRIDE, 0x1);
|
|
|
+ REG_RMW_FIELD(ah, AR_PHY_65NM_RXRF_GAINSTAGES(chain),
|
|
|
+ AR_PHY_65NM_RXRF_GAINSTAGES_LNAON_CALDC, 0x0);
|
|
|
+ if (is_2g)
|
|
|
+ REG_RMW_FIELD(ah, AR_PHY_65NM_RXRF_GAINSTAGES(chain),
|
|
|
+ AR_PHY_65NM_RXRF_GAINSTAGES_LNA2G_GAIN_OVR, 0x0);
|
|
|
+ else
|
|
|
+ REG_RMW_FIELD(ah, AR_PHY_65NM_RXRF_GAINSTAGES(chain),
|
|
|
+ AR_PHY_65NM_RXRF_GAINSTAGES_LNA5G_GAIN_OVR, 0x0);
|
|
|
+
|
|
|
+ REG_RMW_FIELD(ah, AR_PHY_65NM_RXTX2(chain),
|
|
|
+ AR_PHY_65NM_RXTX2_RXON_OVR, 0x1);
|
|
|
+ REG_RMW_FIELD(ah, AR_PHY_65NM_RXTX2(chain),
|
|
|
+ AR_PHY_65NM_RXTX2_RXON, 0x0);
|
|
|
+
|
|
|
+ REG_RMW_FIELD(ah, AR_PHY_65NM_RXRF_AGC(chain),
|
|
|
+ AR_PHY_65NM_RXRF_AGC_AGC_OVERRIDE, 0x1);
|
|
|
+ REG_RMW_FIELD(ah, AR_PHY_65NM_RXRF_AGC(chain),
|
|
|
+ AR_PHY_65NM_RXRF_AGC_AGC_ON_OVR, 0x1);
|
|
|
+ REG_RMW_FIELD(ah, AR_PHY_65NM_RXRF_AGC(chain),
|
|
|
+ AR_PHY_65NM_RXRF_AGC_AGC_CAL_OVR, 0x1);
|
|
|
+ if (is_2g)
|
|
|
+ REG_RMW_FIELD(ah, AR_PHY_65NM_RXRF_AGC(chain),
|
|
|
+ AR_PHY_65NM_RXRF_AGC_AGC2G_DBDAC_OVR, 0x0);
|
|
|
+ else
|
|
|
+ REG_RMW_FIELD(ah, AR_PHY_65NM_RXRF_AGC(chain),
|
|
|
+ AR_PHY_65NM_RXRF_AGC_AGC5G_DBDAC_OVR, 0x0);
|
|
|
+
|
|
|
+ for (i = 6; i > 0; i--) {
|
|
|
+ offset[i] = BIT(i - 1);
|
|
|
+ test = total + offset[i];
|
|
|
+
|
|
|
+ if (is_2g)
|
|
|
+ REG_RMW_FIELD(ah, AR_PHY_65NM_RXRF_AGC(chain),
|
|
|
+ AR_PHY_65NM_RXRF_AGC_AGC2G_CALDAC_OVR,
|
|
|
+ test);
|
|
|
+ else
|
|
|
+ REG_RMW_FIELD(ah, AR_PHY_65NM_RXRF_AGC(chain),
|
|
|
+ AR_PHY_65NM_RXRF_AGC_AGC5G_CALDAC_OVR,
|
|
|
+ test);
|
|
|
+ udelay(100);
|
|
|
+ agc_out = REG_READ_FIELD(ah, AR_PHY_65NM_RXRF_AGC(chain),
|
|
|
+ AR_PHY_65NM_RXRF_AGC_AGC_OUT);
|
|
|
+ offset[i] = (agc_out) ? 0 : 1;
|
|
|
+ total += (offset[i] << (i - 1));
|
|
|
+ }
|
|
|
+
|
|
|
+ if (is_2g)
|
|
|
+ REG_RMW_FIELD(ah, AR_PHY_65NM_RXRF_AGC(chain),
|
|
|
+ AR_PHY_65NM_RXRF_AGC_AGC2G_CALDAC_OVR, total);
|
|
|
+ else
|
|
|
+ REG_RMW_FIELD(ah, AR_PHY_65NM_RXRF_AGC(chain),
|
|
|
+ AR_PHY_65NM_RXRF_AGC_AGC5G_CALDAC_OVR, total);
|
|
|
+
|
|
|
+ REG_RMW_FIELD(ah, AR_PHY_65NM_RXRF_GAINSTAGES(chain),
|
|
|
+ AR_PHY_65NM_RXRF_GAINSTAGES_RX_OVERRIDE, 0);
|
|
|
+ REG_RMW_FIELD(ah, AR_PHY_65NM_RXTX2(chain),
|
|
|
+ AR_PHY_65NM_RXTX2_RXON_OVR, 0);
|
|
|
+ REG_RMW_FIELD(ah, AR_PHY_65NM_RXRF_AGC(chain),
|
|
|
+ AR_PHY_65NM_RXRF_AGC_AGC_CAL_OVR, 0);
|
|
|
+}
|
|
|
+
|
|
|
static bool ar9003_hw_init_cal(struct ath_hw *ah,
|
|
|
struct ath9k_channel *chan)
|
|
|
{
|
|
@@ -989,6 +1057,14 @@ skip_tx_iqcal:
|
|
|
status = ath9k_hw_wait(ah, AR_PHY_AGC_CONTROL,
|
|
|
AR_PHY_AGC_CONTROL_CAL,
|
|
|
0, AH_WAIT_TIMEOUT);
|
|
|
+ if (AR_SREV_9462(ah) || AR_SREV_9565(ah)) {
|
|
|
+ for (i = 0; i < AR9300_MAX_CHAINS; i++) {
|
|
|
+ if (!(ah->rxchainmask & (1 << i)))
|
|
|
+ continue;
|
|
|
+ ar9003_hw_manual_peak_cal(ah, i,
|
|
|
+ IS_CHAN_2GHZ(chan));
|
|
|
+ }
|
|
|
+ }
|
|
|
}
|
|
|
|
|
|
if (ath9k_hw_mci_is_enabled(ah) && IS_CHAN_2GHZ(chan) && run_agc_cal)
|