ccfc0a9d07
It turns out that this is useful on hornet and wasp SoCs but it isn't enabled in ye olde HAL /unless/ you were using a version from one of the business units building USB targetted devices. It eventually got fixed for all of them as people started wanting to use the USB ports on their SoCs (eg for flash storage, bluetooth, 4G/LTE widgets, etc.) This is actually a fix from ath9k but I'm merging it with the available-but- disabled code in the QCA reference HAL. Tested: * AR9331 SoC
6541 lines
225 KiB
C
6541 lines
225 KiB
C
/*
|
|
* Copyright (c) 2013 Qualcomm Atheros, Inc.
|
|
*
|
|
* Permission to use, copy, modify, and/or distribute this software for any
|
|
* purpose with or without fee is hereby granted, provided that the above
|
|
* copyright notice and this permission notice appear in all copies.
|
|
*
|
|
* THE SOFTWARE IS PROVIDED "AS IS" AND THE AUTHOR DISCLAIMS ALL WARRANTIES WITH
|
|
* REGARD TO THIS SOFTWARE INCLUDING ALL IMPLIED WARRANTIES OF MERCHANTABILITY
|
|
* AND FITNESS. IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR ANY SPECIAL, DIRECT,
|
|
* INDIRECT, OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES WHATSOEVER RESULTING FROM
|
|
* LOSS OF USE, DATA OR PROFITS, WHETHER IN AN ACTION OF CONTRACT, NEGLIGENCE OR
|
|
* OTHER TORTIOUS ACTION, ARISING OUT OF OR IN CONNECTION WITH THE USE OR
|
|
* PERFORMANCE OF THIS SOFTWARE.
|
|
*/
|
|
|
|
|
|
|
|
#include "opt_ah.h"
|
|
|
|
#include "ah.h"
|
|
#include "ah_internal.h"
|
|
#include "ah_devid.h"
|
|
#include "ah_desc.h"
|
|
|
|
#include "ar9300.h"
|
|
#include "ar9300reg.h"
|
|
#include "ar9300phy.h"
|
|
#include "ar9300desc.h"
|
|
|
|
#define FIX_NOISE_FLOOR 1
|
|
|
|
|
|
/* Additional Time delay to wait after activiting the Base band */
|
|
#define BASE_ACTIVATE_DELAY 100 /* usec */
|
|
#define RTC_PLL_SETTLE_DELAY 100 /* usec */
|
|
#define COEF_SCALE_S 24
|
|
#define HT40_CHANNEL_CENTER_SHIFT 10 /* MHz */
|
|
|
|
#define DELPT 32
|
|
|
|
/* XXX Duplicates! (in ar9300desc.h) */
|
|
#if 0
|
|
extern HAL_BOOL ar9300_reset_tx_queue(struct ath_hal *ah, u_int q);
|
|
extern u_int32_t ar9300_num_tx_pending(struct ath_hal *ah, u_int q);
|
|
#endif
|
|
|
|
|
|
#define MAX_MEASUREMENT 8
|
|
#define MAXIQCAL 3
|
|
struct coeff_t {
|
|
int32_t mag_coeff[AR9300_MAX_CHAINS][MAX_MEASUREMENT][MAXIQCAL];
|
|
int32_t phs_coeff[AR9300_MAX_CHAINS][MAX_MEASUREMENT][MAXIQCAL];
|
|
int32_t iqc_coeff[2];
|
|
int last_nmeasurement;
|
|
HAL_BOOL last_cal;
|
|
};
|
|
|
|
static HAL_BOOL ar9300_tx_iq_cal_hw_run(struct ath_hal *ah);
|
|
static void ar9300_tx_iq_cal_post_proc(struct ath_hal *ah,HAL_CHANNEL_INTERNAL *ichan,
|
|
int iqcal_idx, int max_iqcal, HAL_BOOL is_cal_reusable, HAL_BOOL apply_last_corr);
|
|
static void ar9300_tx_iq_cal_outlier_detection(struct ath_hal *ah,HAL_CHANNEL_INTERNAL *ichan,
|
|
u_int32_t num_chains, struct coeff_t *coeff, HAL_BOOL is_cal_reusable);
|
|
#if ATH_SUPPORT_CAL_REUSE
|
|
static void ar9300_tx_iq_cal_apply(struct ath_hal *ah, HAL_CHANNEL_INTERNAL *ichan);
|
|
#endif
|
|
|
|
|
|
static inline void ar9300_prog_ini(struct ath_hal *ah, struct ar9300_ini_array *ini_arr, int column);
|
|
static inline void ar9300_set_rf_mode(struct ath_hal *ah, struct ieee80211_channel *chan);
|
|
static inline HAL_BOOL ar9300_init_cal(struct ath_hal *ah, struct ieee80211_channel *chan, HAL_BOOL skip_if_none, HAL_BOOL apply_last_corr);
|
|
static inline void ar9300_init_user_settings(struct ath_hal *ah);
|
|
|
|
#ifdef HOST_OFFLOAD
|
|
/*
|
|
* For usb offload solution, some USB registers must be tuned
|
|
* to gain better stability/performance but these registers
|
|
* might be changed while doing wlan reset so do this here
|
|
*/
|
|
#define WAR_USB_DISABLE_PLL_LOCK_DETECT(__ah) \
|
|
do { \
|
|
if (AR_SREV_HORNET(__ah) || AR_SREV_WASP(__ah)) { \
|
|
volatile u_int32_t *usb_ctrl_r1 = (u_int32_t *) 0xb8116c84; \
|
|
volatile u_int32_t *usb_ctrl_r2 = (u_int32_t *) 0xb8116c88; \
|
|
*usb_ctrl_r1 = (*usb_ctrl_r1 & 0xffefffff); \
|
|
*usb_ctrl_r2 = (*usb_ctrl_r2 & 0xfc1fffff) | (1 << 21) | (3 << 22); \
|
|
} \
|
|
} while (0)
|
|
#else
|
|
#define WAR_USB_DISABLE_PLL_LOCK_DETECT(__ah)
|
|
#endif
|
|
|
|
/*
|
|
* Note: the below is the version that ships with ath9k.
|
|
* The original HAL version is above.
|
|
*/
|
|
|
|
static void
|
|
ar9300_disable_pll_lock_detect(struct ath_hal *ah)
|
|
{
|
|
/*
|
|
* On AR9330 and AR9340 devices, some PHY registers must be
|
|
* tuned to gain better stability/performance. These registers
|
|
* might be changed while doing wlan reset so the registers must
|
|
* be reprogrammed after each reset.
|
|
*/
|
|
if (AR_SREV_HORNET(ah) || AR_SREV_WASP(ah)) {
|
|
HALDEBUG(ah, HAL_DEBUG_RESET, "%s: called\n", __func__);
|
|
OS_REG_CLR_BIT(ah, AR_PHY_USB_CTRL1, (1 << 20));
|
|
OS_REG_RMW(ah, AR_PHY_USB_CTRL2,
|
|
(1 << 21) | (0xf << 22),
|
|
(1 << 21) | (0x3 << 22));
|
|
}
|
|
}
|
|
|
|
static inline void
|
|
ar9300_attach_hw_platform(struct ath_hal *ah)
|
|
{
|
|
struct ath_hal_9300 *ahp = AH9300(ah);
|
|
|
|
ahp->ah_hwp = HAL_TRUE_CHIP;
|
|
return;
|
|
}
|
|
|
|
/* Adjust various register settings based on half/quarter rate clock setting.
|
|
* This includes: +USEC, TX/RX latency,
|
|
* + IFS params: slot, eifs, misc etc.
|
|
* SIFS stays the same.
|
|
*/
|
|
static void
|
|
ar9300_set_ifs_timing(struct ath_hal *ah, struct ieee80211_channel *chan)
|
|
{
|
|
u_int32_t tx_lat, rx_lat, usec, slot, regval, eifs;
|
|
|
|
regval = OS_REG_READ(ah, AR_USEC);
|
|
regval &= ~(AR_USEC_RX_LATENCY | AR_USEC_TX_LATENCY | AR_USEC_USEC);
|
|
if (IEEE80211_IS_CHAN_HALF(chan)) { /* half rates */
|
|
slot = ar9300_mac_to_clks(ah, AR_SLOT_HALF);
|
|
eifs = ar9300_mac_to_clks(ah, AR_EIFS_HALF);
|
|
if (IS_5GHZ_FAST_CLOCK_EN(ah, chan)) { /* fast clock */
|
|
rx_lat = SM(AR_RX_LATENCY_HALF_FAST_CLOCK, AR_USEC_RX_LATENCY);
|
|
tx_lat = SM(AR_TX_LATENCY_HALF_FAST_CLOCK, AR_USEC_TX_LATENCY);
|
|
usec = SM(AR_USEC_HALF_FAST_CLOCK, AR_USEC_USEC);
|
|
} else {
|
|
rx_lat = SM(AR_RX_LATENCY_HALF, AR_USEC_RX_LATENCY);
|
|
tx_lat = SM(AR_TX_LATENCY_HALF, AR_USEC_TX_LATENCY);
|
|
usec = SM(AR_USEC_HALF, AR_USEC_USEC);
|
|
}
|
|
} else { /* quarter rate */
|
|
slot = ar9300_mac_to_clks(ah, AR_SLOT_QUARTER);
|
|
eifs = ar9300_mac_to_clks(ah, AR_EIFS_QUARTER);
|
|
if (IS_5GHZ_FAST_CLOCK_EN(ah, chan)) { /* fast clock */
|
|
rx_lat = SM(AR_RX_LATENCY_QUARTER_FAST_CLOCK, AR_USEC_RX_LATENCY);
|
|
tx_lat = SM(AR_TX_LATENCY_QUARTER_FAST_CLOCK, AR_USEC_TX_LATENCY);
|
|
usec = SM(AR_USEC_QUARTER_FAST_CLOCK, AR_USEC_USEC);
|
|
} else {
|
|
rx_lat = SM(AR_RX_LATENCY_QUARTER, AR_USEC_RX_LATENCY);
|
|
tx_lat = SM(AR_TX_LATENCY_QUARTER, AR_USEC_TX_LATENCY);
|
|
usec = SM(AR_USEC_QUARTER, AR_USEC_USEC);
|
|
}
|
|
}
|
|
|
|
OS_REG_WRITE(ah, AR_USEC, (usec | regval | tx_lat | rx_lat));
|
|
OS_REG_WRITE(ah, AR_D_GBL_IFS_SLOT, slot);
|
|
OS_REG_WRITE(ah, AR_D_GBL_IFS_EIFS, eifs);
|
|
}
|
|
|
|
|
|
/*
|
|
* This inline function configures the chip either
|
|
* to encrypt/decrypt management frames or pass thru
|
|
*/
|
|
static inline void
|
|
ar9300_init_mfp(struct ath_hal * ah)
|
|
{
|
|
u_int32_t mfpcap, mfp_qos;
|
|
|
|
ath_hal_getcapability(ah, HAL_CAP_MFP, 0, &mfpcap);
|
|
|
|
if (mfpcap == HAL_MFP_QOSDATA) {
|
|
/* Treat like legacy hardware. Do not touch the MFP registers. */
|
|
HALDEBUG(ah, HAL_DEBUG_RESET, "%s forced to use QOSDATA\n", __func__);
|
|
return;
|
|
}
|
|
|
|
/* MFP support (Sowl 1.0 or greater) */
|
|
if (mfpcap == HAL_MFP_HW_CRYPTO) {
|
|
/* configure hardware MFP support */
|
|
HALDEBUG(ah, HAL_DEBUG_RESET, "%s using HW crypto\n", __func__);
|
|
OS_REG_RMW_FIELD(ah,
|
|
AR_AES_MUTE_MASK1, AR_AES_MUTE_MASK1_FC_MGMT, AR_AES_MUTE_MASK1_FC_MGMT_MFP);
|
|
OS_REG_RMW(ah,
|
|
AR_PCU_MISC_MODE2, AR_PCU_MISC_MODE2_MGMT_CRYPTO_ENABLE,
|
|
AR_PCU_MISC_MODE2_NO_CRYPTO_FOR_NON_DATA_PKT);
|
|
/*
|
|
* Mask used to construct AAD for CCMP-AES
|
|
* Cisco spec defined bits 0-3 as mask
|
|
* IEEE802.11w defined as bit 4.
|
|
*/
|
|
if (ath_hal_get_mfp_qos(ah)) {
|
|
mfp_qos = AR_MFP_QOS_MASK_IEEE;
|
|
} else {
|
|
mfp_qos = AR_MFP_QOS_MASK_CISCO;
|
|
}
|
|
OS_REG_RMW_FIELD(ah,
|
|
AR_PCU_MISC_MODE2, AR_PCU_MISC_MODE2_MGMT_QOS, mfp_qos);
|
|
} else if (mfpcap == HAL_MFP_PASSTHRU) {
|
|
/* Disable en/decrypt by hardware */
|
|
HALDEBUG(ah, HAL_DEBUG_RESET, "%s using passthru\n", __func__);
|
|
OS_REG_RMW(ah,
|
|
AR_PCU_MISC_MODE2,
|
|
AR_PCU_MISC_MODE2_NO_CRYPTO_FOR_NON_DATA_PKT,
|
|
AR_PCU_MISC_MODE2_MGMT_CRYPTO_ENABLE);
|
|
}
|
|
}
|
|
|
|
void
|
|
ar9300_get_channel_centers(struct ath_hal *ah, const struct ieee80211_channel *chan,
|
|
CHAN_CENTERS *centers)
|
|
{
|
|
int8_t extoff;
|
|
struct ath_hal_9300 *ahp = AH9300(ah);
|
|
HAL_CHANNEL_INTERNAL *ichan = ath_hal_checkchannel(ah, chan);
|
|
|
|
if (!IEEE80211_IS_CHAN_HT40(chan)) {
|
|
centers->ctl_center = centers->ext_center =
|
|
centers->synth_center = ichan->channel;
|
|
return;
|
|
}
|
|
|
|
HALASSERT(IEEE80211_IS_CHAN_HT40(chan));
|
|
|
|
/*
|
|
* In 20/40 phy mode, the center frequency is
|
|
* "between" the primary and extension channels.
|
|
*/
|
|
if (IEEE80211_IS_CHAN_HT40U(chan)) {
|
|
centers->synth_center = ichan->channel + HT40_CHANNEL_CENTER_SHIFT;
|
|
extoff = 1;
|
|
} else {
|
|
centers->synth_center = ichan->channel - HT40_CHANNEL_CENTER_SHIFT;
|
|
extoff = -1;
|
|
}
|
|
|
|
centers->ctl_center =
|
|
centers->synth_center - (extoff * HT40_CHANNEL_CENTER_SHIFT);
|
|
centers->ext_center =
|
|
centers->synth_center +
|
|
(extoff * ((ahp->ah_ext_prot_spacing == HAL_HT_EXTPROTSPACING_20) ?
|
|
HT40_CHANNEL_CENTER_SHIFT : 15));
|
|
}
|
|
|
|
/*
|
|
* Read the noise-floor values from the HW.
|
|
* Specifically, read the minimum clear-channel assessment value for
|
|
* each chain, for both the control and extension channels.
|
|
* (The received power level during clear-channel periods is the
|
|
* noise floor.)
|
|
* These noise floor values computed by the HW will be stored in the
|
|
* NF history buffer.
|
|
* The HW sometimes produces bogus NF values. To avoid using these
|
|
* bogus values, the NF data is (a) range-limited, and (b) filtered.
|
|
* However, this data-processing is done when reading the NF values
|
|
* out of the history buffer. The history buffer stores the raw values.
|
|
* This allows the NF history buffer to be used to check for interference.
|
|
* A single high NF reading might be a bogus HW value, but if the NF
|
|
* readings are consistently high, it must be due to interference.
|
|
* This is the purpose of storing raw NF values in the history buffer,
|
|
* rather than processed values. By looking at a history of NF values
|
|
* that have not been range-limited, we can check if they are consistently
|
|
* high (due to interference).
|
|
*/
|
|
#define AH_NF_SIGN_EXTEND(nf) \
|
|
((nf) & 0x100) ? \
|
|
0 - (((nf) ^ 0x1ff) + 1) : \
|
|
(nf)
|
|
void
|
|
ar9300_upload_noise_floor(struct ath_hal *ah, int is_2g,
|
|
int16_t nfarray[HAL_NUM_NF_READINGS])
|
|
{
|
|
int16_t nf;
|
|
int chan, chain;
|
|
u_int32_t regs[HAL_NUM_NF_READINGS] = {
|
|
/* control channel */
|
|
AR_PHY_CCA_0, /* chain 0 */
|
|
AR_PHY_CCA_1, /* chain 1 */
|
|
AR_PHY_CCA_2, /* chain 2 */
|
|
/* extension channel */
|
|
AR_PHY_EXT_CCA, /* chain 0 */
|
|
AR_PHY_EXT_CCA_1, /* chain 1 */
|
|
AR_PHY_EXT_CCA_2, /* chain 2 */
|
|
};
|
|
u_int8_t chainmask;
|
|
|
|
/*
|
|
* Within a given channel (ctl vs. ext), the CH0, CH1, and CH2
|
|
* masks and shifts are the same, though they differ for the
|
|
* control vs. extension channels.
|
|
*/
|
|
u_int32_t masks[2] = {
|
|
AR_PHY_MINCCA_PWR, /* control channel */
|
|
AR_PHY_EXT_MINCCA_PWR, /* extention channel */
|
|
};
|
|
u_int8_t shifts[2] = {
|
|
AR_PHY_MINCCA_PWR_S, /* control channel */
|
|
AR_PHY_EXT_MINCCA_PWR_S, /* extention channel */
|
|
};
|
|
|
|
/*
|
|
* Force NF calibration for all chains.
|
|
*/
|
|
if (AR_SREV_HORNET(ah) || AR_SREV_POSEIDON(ah) || AR_SREV_APHRODITE(ah)) {
|
|
chainmask = 0x01;
|
|
} else if (AR_SREV_WASP(ah) || AR_SREV_JUPITER(ah) || AR_SREV_HONEYBEE(ah)) {
|
|
chainmask = 0x03;
|
|
} else {
|
|
chainmask = 0x07;
|
|
}
|
|
|
|
for (chan = 0; chan < 2 /*ctl,ext*/; chan++) {
|
|
for (chain = 0; chain < AR9300_MAX_CHAINS; chain++) {
|
|
int i;
|
|
|
|
if (!((chainmask >> chain) & 0x1)) {
|
|
continue;
|
|
}
|
|
i = chan * AR9300_MAX_CHAINS + chain;
|
|
nf = (OS_REG_READ(ah, regs[i]) & masks[chan]) >> shifts[chan];
|
|
nfarray[i] = AH_NF_SIGN_EXTEND(nf);
|
|
}
|
|
}
|
|
}
|
|
|
|
/* ar9300_get_min_cca_pwr -
|
|
* Used by the scan function for a quick read of the noise floor.
|
|
* This is used to detect presence of CW interference such as video bridge.
|
|
* The noise floor is assumed to have been already started during reset
|
|
* called during channel change. The function checks if the noise floor
|
|
* reading is done. In case it has been done, it reads the noise floor value.
|
|
* If the noise floor calibration has not been finished, it assumes this is
|
|
* due to presence of CW interference an returns a high value for noise floor,
|
|
* derived from the CW interference threshold + margin fudge factor.
|
|
*/
|
|
#define BAD_SCAN_NF_MARGIN (30)
|
|
int16_t ar9300_get_min_cca_pwr(struct ath_hal *ah)
|
|
{
|
|
int16_t nf;
|
|
// struct ath_hal_private *ahpriv = AH_PRIVATE(ah);
|
|
|
|
|
|
if ((OS_REG_READ(ah, AR_PHY_AGC_CONTROL) & AR_PHY_AGC_CONTROL_NF) == 0) {
|
|
nf = MS(OS_REG_READ(ah, AR_PHY_CCA_0), AR9280_PHY_MINCCA_PWR);
|
|
if (nf & 0x100) {
|
|
nf = 0 - ((nf ^ 0x1ff) + 1);
|
|
}
|
|
} else {
|
|
/* NF calibration is not done, assume CW interference */
|
|
nf = AH9300(ah)->nfp->nominal + AH9300(ah)->nf_cw_int_delta +
|
|
BAD_SCAN_NF_MARGIN;
|
|
}
|
|
return nf;
|
|
}
|
|
|
|
|
|
/*
|
|
* Noise Floor values for all chains.
|
|
* Most recently updated values from the NF history buffer are used.
|
|
*/
|
|
void ar9300_chain_noise_floor(struct ath_hal *ah, int16_t *nf_buf,
|
|
struct ieee80211_channel *chan, int is_scan)
|
|
{
|
|
struct ath_hal_9300 *ahp = AH9300(ah);
|
|
int i, nf_hist_len, recent_nf_index = 0;
|
|
HAL_NFCAL_HIST_FULL *h;
|
|
u_int8_t rx_chainmask = ahp->ah_rx_chainmask | (ahp->ah_rx_chainmask << 3);
|
|
HAL_CHANNEL_INTERNAL *ichan = ath_hal_checkchannel(ah, chan);
|
|
HALASSERT(ichan);
|
|
|
|
#ifdef ATH_NF_PER_CHAN
|
|
/* Fill 0 if valid internal channel is not found */
|
|
if (ichan == AH_NULL) {
|
|
OS_MEMZERO(nf_buf, sizeof(nf_buf[0])*HAL_NUM_NF_READINGS);
|
|
return;
|
|
}
|
|
h = &ichan->nf_cal_hist;
|
|
nf_hist_len = HAL_NF_CAL_HIST_LEN_FULL;
|
|
#else
|
|
/*
|
|
* If a scan is not in progress, then the most recent value goes
|
|
* into ahpriv->nf_cal_hist. If a scan is in progress, then
|
|
* the most recent value goes into ichan->nf_cal_hist.
|
|
* Thus, return the value from ahpriv->nf_cal_hist if there's
|
|
* no scan, and if the specified channel is the current channel.
|
|
* Otherwise, return the noise floor from ichan->nf_cal_hist.
|
|
*/
|
|
if ((!is_scan) && chan == AH_PRIVATE(ah)->ah_curchan) {
|
|
h = &AH_PRIVATE(ah)->nf_cal_hist;
|
|
nf_hist_len = HAL_NF_CAL_HIST_LEN_FULL;
|
|
} else {
|
|
/* Fill 0 if valid internal channel is not found */
|
|
if (ichan == AH_NULL) {
|
|
OS_MEMZERO(nf_buf, sizeof(nf_buf[0])*HAL_NUM_NF_READINGS);
|
|
return;
|
|
}
|
|
/*
|
|
* It is okay to treat a HAL_NFCAL_HIST_SMALL struct as if it were a
|
|
* HAL_NFCAL_HIST_FULL struct, as long as only the index 0 of the
|
|
* nf_cal_buffer is used (nf_cal_buffer[0][0:HAL_NUM_NF_READINGS-1])
|
|
*/
|
|
h = (HAL_NFCAL_HIST_FULL *) &ichan->nf_cal_hist;
|
|
nf_hist_len = HAL_NF_CAL_HIST_LEN_SMALL;
|
|
}
|
|
#endif
|
|
/* Get most recently updated values from nf cal history buffer */
|
|
recent_nf_index =
|
|
(h->base.curr_index) ? h->base.curr_index - 1 : nf_hist_len - 1;
|
|
|
|
for (i = 0; i < HAL_NUM_NF_READINGS; i++) {
|
|
/* Fill 0 for unsupported chains */
|
|
if (!(rx_chainmask & (1 << i))) {
|
|
nf_buf[i] = 0;
|
|
continue;
|
|
}
|
|
nf_buf[i] = h->nf_cal_buffer[recent_nf_index][i];
|
|
}
|
|
}
|
|
|
|
/*
|
|
* Return the current NF value in register.
|
|
* If the current NF cal is not completed, return 0.
|
|
*/
|
|
int16_t ar9300_get_nf_from_reg(struct ath_hal *ah, struct ieee80211_channel *chan, int wait_time)
|
|
{
|
|
int16_t nfarray[HAL_NUM_NF_READINGS] = {0};
|
|
int is_2g = 0;
|
|
HAL_CHANNEL_INTERNAL *ichan = NULL;
|
|
|
|
ichan = ath_hal_checkchannel(ah, chan);
|
|
if (ichan == NULL)
|
|
return (0);
|
|
|
|
if (wait_time <= 0) {
|
|
return 0;
|
|
}
|
|
|
|
if (!ath_hal_waitfor(ah, AR_PHY_AGC_CONTROL, AR_PHY_AGC_CONTROL_NF, 0, wait_time)) {
|
|
ath_hal_printf(ah, "%s: NF cal is not complete in %dus", __func__, wait_time);
|
|
return 0;
|
|
}
|
|
is_2g = !! (IS_CHAN_2GHZ(ichan));
|
|
ar9300_upload_noise_floor(ah, is_2g, nfarray);
|
|
|
|
return nfarray[0];
|
|
}
|
|
|
|
/*
|
|
* Pick up the medium one in the noise floor buffer and update the
|
|
* corresponding range for valid noise floor values
|
|
*/
|
|
static int16_t
|
|
ar9300_get_nf_hist_mid(struct ath_hal *ah, HAL_NFCAL_HIST_FULL *h, int reading,
|
|
int hist_len)
|
|
{
|
|
int16_t nfval;
|
|
int16_t sort[HAL_NF_CAL_HIST_LEN_FULL]; /* upper bound for hist_len */
|
|
int i, j;
|
|
|
|
|
|
for (i = 0; i < hist_len; i++) {
|
|
sort[i] = h->nf_cal_buffer[i][reading];
|
|
HALDEBUG(ah, HAL_DEBUG_NFCAL,
|
|
"nf_cal_buffer[%d][%d] = %d\n", i, reading, (int)sort[i]);
|
|
}
|
|
for (i = 0; i < hist_len - 1; i++) {
|
|
for (j = 1; j < hist_len - i; j++) {
|
|
if (sort[j] > sort[j - 1]) {
|
|
nfval = sort[j];
|
|
sort[j] = sort[j - 1];
|
|
sort[j - 1] = nfval;
|
|
}
|
|
}
|
|
}
|
|
nfval = sort[(hist_len - 1) >> 1];
|
|
|
|
return nfval;
|
|
}
|
|
|
|
static int16_t ar9300_limit_nf_range(struct ath_hal *ah, int16_t nf)
|
|
{
|
|
if (nf < AH9300(ah)->nfp->min) {
|
|
return AH9300(ah)->nfp->nominal;
|
|
} else if (nf > AH9300(ah)->nfp->max) {
|
|
return AH9300(ah)->nfp->max;
|
|
}
|
|
return nf;
|
|
}
|
|
|
|
#ifndef ATH_NF_PER_CHAN
|
|
inline static void
|
|
ar9300_reset_nf_hist_buff(struct ath_hal *ah, HAL_CHANNEL_INTERNAL *ichan)
|
|
{
|
|
HAL_CHAN_NFCAL_HIST *h = &ichan->nf_cal_hist;
|
|
HAL_NFCAL_HIST_FULL *home = &AH_PRIVATE(ah)->nf_cal_hist;
|
|
int i;
|
|
|
|
/*
|
|
* Copy the value for the channel in question into the home-channel
|
|
* NF history buffer. The channel NF is probably a value filled in by
|
|
* a prior background channel scan, but if no scan has been done then
|
|
* it is the nominal noise floor filled in by ath_hal_init_NF_buffer
|
|
* for this chip and the channel's band.
|
|
* Replicate this channel NF into all entries of the home-channel NF
|
|
* history buffer.
|
|
* If the channel NF was filled in by a channel scan, it has not had
|
|
* bounds limits applied to it yet - do so now. It is important to
|
|
* apply bounds limits to the priv_nf value that gets loaded into the
|
|
* WLAN chip's min_cca_pwr register field. It is also necessary to
|
|
* apply bounds limits to the nf_cal_buffer[] elements. Since we are
|
|
* replicating a single NF reading into all nf_cal_buffer elements,
|
|
* if the single reading were above the CW_INT threshold, the CW_INT
|
|
* check in ar9300_get_nf would immediately conclude that CW interference
|
|
* is present, even though we're not supposed to set CW_INT unless
|
|
* NF values are _consistently_ above the CW_INT threshold.
|
|
* Applying the bounds limits to the nf_cal_buffer contents fixes this
|
|
* problem.
|
|
*/
|
|
for (i = 0; i < HAL_NUM_NF_READINGS; i ++) {
|
|
int j;
|
|
int16_t nf;
|
|
/*
|
|
* No need to set curr_index, since it already has a value in
|
|
* the range [0..HAL_NF_CAL_HIST_LEN_FULL), and all nf_cal_buffer
|
|
* values will be the same.
|
|
*/
|
|
nf = ar9300_limit_nf_range(ah, h->nf_cal_buffer[0][i]);
|
|
for (j = 0; j < HAL_NF_CAL_HIST_LEN_FULL; j++) {
|
|
home->nf_cal_buffer[j][i] = nf;
|
|
}
|
|
AH_PRIVATE(ah)->nf_cal_hist.base.priv_nf[i] = nf;
|
|
}
|
|
}
|
|
#endif
|
|
|
|
/*
|
|
* Update the noise floor buffer as a ring buffer
|
|
*/
|
|
static int16_t
|
|
ar9300_update_nf_hist_buff(struct ath_hal *ah, HAL_NFCAL_HIST_FULL *h,
|
|
int16_t *nfarray, int hist_len)
|
|
{
|
|
int i, nr;
|
|
int16_t nf_no_lim_chain0;
|
|
|
|
nf_no_lim_chain0 = ar9300_get_nf_hist_mid(ah, h, 0, hist_len);
|
|
|
|
HALDEBUG(ah, HAL_DEBUG_NFCAL, "%s[%d] BEFORE\n", __func__, __LINE__);
|
|
for (nr = 0; nr < HAL_NF_CAL_HIST_LEN_FULL; nr++) {
|
|
for (i = 0; i < HAL_NUM_NF_READINGS; i++) {
|
|
HALDEBUG(ah, HAL_DEBUG_NFCAL,
|
|
"nf_cal_buffer[%d][%d] = %d\n",
|
|
nr, i, (int)h->nf_cal_buffer[nr][i]);
|
|
}
|
|
}
|
|
for (i = 0; i < HAL_NUM_NF_READINGS; i++) {
|
|
h->nf_cal_buffer[h->base.curr_index][i] = nfarray[i];
|
|
h->base.priv_nf[i] = ar9300_limit_nf_range(
|
|
ah, ar9300_get_nf_hist_mid(ah, h, i, hist_len));
|
|
}
|
|
HALDEBUG(ah, HAL_DEBUG_NFCAL, "%s[%d] AFTER\n", __func__, __LINE__);
|
|
for (nr = 0; nr < HAL_NF_CAL_HIST_LEN_FULL; nr++) {
|
|
for (i = 0; i < HAL_NUM_NF_READINGS; i++) {
|
|
HALDEBUG(ah, HAL_DEBUG_NFCAL,
|
|
"nf_cal_buffer[%d][%d] = %d\n",
|
|
nr, i, (int)h->nf_cal_buffer[nr][i]);
|
|
}
|
|
}
|
|
|
|
if (++h->base.curr_index >= hist_len) {
|
|
h->base.curr_index = 0;
|
|
}
|
|
|
|
return nf_no_lim_chain0;
|
|
}
|
|
|
|
#ifdef UNUSED
|
|
static HAL_BOOL
|
|
get_noise_floor_thresh(struct ath_hal *ah, const HAL_CHANNEL_INTERNAL *chan,
|
|
int16_t *nft)
|
|
{
|
|
struct ath_hal_9300 *ahp = AH9300(ah);
|
|
|
|
|
|
switch (chan->channel_flags & CHANNEL_ALL_NOTURBO) {
|
|
case CHANNEL_A:
|
|
case CHANNEL_A_HT20:
|
|
case CHANNEL_A_HT40PLUS:
|
|
case CHANNEL_A_HT40MINUS:
|
|
*nft = (int8_t)ar9300_eeprom_get(ahp, EEP_NFTHRESH_5);
|
|
break;
|
|
case CHANNEL_B:
|
|
case CHANNEL_G:
|
|
case CHANNEL_G_HT20:
|
|
case CHANNEL_G_HT40PLUS:
|
|
case CHANNEL_G_HT40MINUS:
|
|
*nft = (int8_t)ar9300_eeprom_get(ahp, EEP_NFTHRESH_2);
|
|
break;
|
|
default:
|
|
HALDEBUG(ah, HAL_DEBUG_CHANNEL, "%s: invalid channel flags 0x%x\n",
|
|
__func__, chan->channel_flags);
|
|
return AH_FALSE;
|
|
}
|
|
return AH_TRUE;
|
|
}
|
|
#endif
|
|
|
|
/*
|
|
* Read the NF and check it against the noise floor threshhold
|
|
*/
|
|
#define IS(_c, _f) (((_c)->channel_flags & _f) || 0)
|
|
static int
|
|
ar9300_store_new_nf(struct ath_hal *ah, struct ieee80211_channel *chan,
|
|
int is_scan)
|
|
{
|
|
// struct ath_hal_private *ahpriv = AH_PRIVATE(ah);
|
|
int nf_hist_len;
|
|
int16_t nf_no_lim;
|
|
int16_t nfarray[HAL_NUM_NF_READINGS] = {0};
|
|
HAL_NFCAL_HIST_FULL *h;
|
|
int is_2g = 0;
|
|
HAL_CHANNEL_INTERNAL *ichan = ath_hal_checkchannel(ah, chan);
|
|
struct ath_hal_9300 *ahp = AH9300(ah);
|
|
|
|
if (OS_REG_READ(ah, AR_PHY_AGC_CONTROL) & AR_PHY_AGC_CONTROL_NF) {
|
|
u_int32_t tsf32, nf_cal_dur_tsf;
|
|
/*
|
|
* The reason the NF calibration did not complete may just be that
|
|
* not enough time has passed since the NF calibration was started,
|
|
* because under certain conditions (when first moving to a new
|
|
* channel) the NF calibration may be checked very repeatedly.
|
|
* Or, there may be CW interference keeping the NF calibration
|
|
* from completing. Check the delta time between when the NF
|
|
* calibration was started and now to see whether the NF calibration
|
|
* should have already completed (but hasn't, probably due to CW
|
|
* interference), or hasn't had enough time to finish yet.
|
|
*/
|
|
/*
|
|
* AH_NF_CAL_DUR_MAX_TSF - A conservative maximum time that the
|
|
* HW should need to finish a NF calibration. If the HW
|
|
* does not complete a NF calibration within this time period,
|
|
* there must be a problem - probably CW interference.
|
|
* AH_NF_CAL_PERIOD_MAX_TSF - A conservative maximum time between
|
|
* check of the HW's NF calibration being finished.
|
|
* If the difference between the current TSF and the TSF
|
|
* recorded when the NF calibration started is larger than this
|
|
* value, the TSF must have been reset.
|
|
* In general, we expect the TSF to only be reset during
|
|
* regular operation for STAs, not for APs. However, an
|
|
* AP's TSF could be reset when joining an IBSS.
|
|
* There's an outside chance that this could result in the
|
|
* CW_INT flag being erroneously set, if the TSF adjustment
|
|
* is smaller than AH_NF_CAL_PERIOD_MAX_TSF but larger than
|
|
* AH_NF_CAL_DUR_TSF. However, even if this does happen,
|
|
* it shouldn't matter, as the IBSS case shouldn't be
|
|
* concerned about CW_INT.
|
|
*/
|
|
/* AH_NF_CAL_DUR_TSF - 90 sec in usec units */
|
|
#define AH_NF_CAL_DUR_TSF (90 * 1000 * 1000)
|
|
/* AH_NF_CAL_PERIOD_MAX_TSF - 180 sec in usec units */
|
|
#define AH_NF_CAL_PERIOD_MAX_TSF (180 * 1000 * 1000)
|
|
/* wraparound handled by using unsigned values */
|
|
tsf32 = ar9300_get_tsf32(ah);
|
|
nf_cal_dur_tsf = tsf32 - AH9300(ah)->nf_tsf32;
|
|
if (nf_cal_dur_tsf > AH_NF_CAL_PERIOD_MAX_TSF) {
|
|
/*
|
|
* The TSF must have gotten reset during the NF cal -
|
|
* just reset the NF TSF timestamp, so the next time
|
|
* this function is called, the timestamp comparison
|
|
* will be valid.
|
|
*/
|
|
AH9300(ah)->nf_tsf32 = tsf32;
|
|
} else if (nf_cal_dur_tsf > AH_NF_CAL_DUR_TSF) {
|
|
HALDEBUG(ah, HAL_DEBUG_CALIBRATE,
|
|
"%s: NF did not complete in calibration window\n", __func__);
|
|
/* the NF incompletion is probably due to CW interference */
|
|
chan->ic_state |= IEEE80211_CHANSTATE_CWINT;
|
|
}
|
|
return 0; /* HW's NF measurement not finished */
|
|
}
|
|
HALDEBUG(ah, HAL_DEBUG_NFCAL,
|
|
"%s[%d] chan %d\n", __func__, __LINE__, ichan->channel);
|
|
is_2g = !! IS_CHAN_2GHZ(ichan);
|
|
ar9300_upload_noise_floor(ah, is_2g, nfarray);
|
|
|
|
/* Update the NF buffer for each chain masked by chainmask */
|
|
#ifdef ATH_NF_PER_CHAN
|
|
h = &ichan->nf_cal_hist;
|
|
nf_hist_len = HAL_NF_CAL_HIST_LEN_FULL;
|
|
#else
|
|
if (is_scan) {
|
|
/*
|
|
* This channel's NF cal info is just a HAL_NFCAL_HIST_SMALL struct
|
|
* rather than a HAL_NFCAL_HIST_FULL struct.
|
|
* As long as we only use the first history element of nf_cal_buffer
|
|
* (nf_cal_buffer[0][0:HAL_NUM_NF_READINGS-1]), we can use
|
|
* HAL_NFCAL_HIST_SMALL and HAL_NFCAL_HIST_FULL interchangeably.
|
|
*/
|
|
h = (HAL_NFCAL_HIST_FULL *) &ichan->nf_cal_hist;
|
|
nf_hist_len = HAL_NF_CAL_HIST_LEN_SMALL;
|
|
} else {
|
|
h = &AH_PRIVATE(ah)->nf_cal_hist;
|
|
nf_hist_len = HAL_NF_CAL_HIST_LEN_FULL;
|
|
}
|
|
#endif
|
|
|
|
/*
|
|
* nf_no_lim = median value from NF history buffer without bounds limits,
|
|
* priv_nf = median value from NF history buffer with bounds limits.
|
|
*/
|
|
nf_no_lim = ar9300_update_nf_hist_buff(ah, h, nfarray, nf_hist_len);
|
|
ichan->rawNoiseFloor = h->base.priv_nf[0];
|
|
|
|
/* check if there is interference */
|
|
// ichan->channel_flags &= (~CHANNEL_CW_INT);
|
|
/*
|
|
* Use AR9300_EMULATION to check for emulation purpose as PCIE Device ID
|
|
* 0xABCD is recognized as valid Osprey as WAR in some EVs.
|
|
*/
|
|
if (nf_no_lim > ahp->nfp->nominal + ahp->nf_cw_int_delta) {
|
|
/*
|
|
* Since this CW interference check is being applied to the
|
|
* median element of the NF history buffer, this indicates that
|
|
* the CW interference is persistent. A single high NF reading
|
|
* will not show up in the median, and thus will not cause the
|
|
* CW_INT flag to be set.
|
|
*/
|
|
HALDEBUG(ah, HAL_DEBUG_NFCAL,
|
|
"%s: NF Cal: CW interferer detected through NF: %d\n",
|
|
__func__, nf_no_lim);
|
|
chan->ic_state |= IEEE80211_CHANSTATE_CWINT;
|
|
}
|
|
return 1; /* HW's NF measurement finished */
|
|
}
|
|
#undef IS
|
|
|
|
static inline void
|
|
ar9300_get_delta_slope_values(struct ath_hal *ah, u_int32_t coef_scaled,
|
|
u_int32_t *coef_mantissa, u_int32_t *coef_exponent)
|
|
{
|
|
u_int32_t coef_exp, coef_man;
|
|
|
|
/*
|
|
* ALGO -> coef_exp = 14-floor(log2(coef));
|
|
* floor(log2(x)) is the highest set bit position
|
|
*/
|
|
for (coef_exp = 31; coef_exp > 0; coef_exp--) {
|
|
if ((coef_scaled >> coef_exp) & 0x1) {
|
|
break;
|
|
}
|
|
}
|
|
/* A coef_exp of 0 is a legal bit position but an unexpected coef_exp */
|
|
HALASSERT(coef_exp);
|
|
coef_exp = 14 - (coef_exp - COEF_SCALE_S);
|
|
|
|
|
|
/*
|
|
* ALGO -> coef_man = floor(coef* 2^coef_exp+0.5);
|
|
* The coefficient is already shifted up for scaling
|
|
*/
|
|
coef_man = coef_scaled + (1 << (COEF_SCALE_S - coef_exp - 1));
|
|
|
|
*coef_mantissa = coef_man >> (COEF_SCALE_S - coef_exp);
|
|
*coef_exponent = coef_exp - 16;
|
|
}
|
|
|
|
#define MAX_ANALOG_START 319 /* XXX */
|
|
|
|
/*
|
|
* Delta slope coefficient computation.
|
|
* Required for OFDM operation.
|
|
*/
|
|
static void
|
|
ar9300_set_delta_slope(struct ath_hal *ah, struct ieee80211_channel *chan)
|
|
{
|
|
u_int32_t coef_scaled, ds_coef_exp, ds_coef_man;
|
|
u_int32_t fclk = COEFF; /* clock * 2.5 */
|
|
|
|
u_int32_t clock_mhz_scaled = 0x1000000 * fclk;
|
|
CHAN_CENTERS centers;
|
|
|
|
/*
|
|
* half and quarter rate can divide the scaled clock by 2 or 4
|
|
* scale for selected channel bandwidth
|
|
*/
|
|
if (IEEE80211_IS_CHAN_HALF(chan)) {
|
|
clock_mhz_scaled = clock_mhz_scaled >> 1;
|
|
} else if (IEEE80211_IS_CHAN_QUARTER(chan)) {
|
|
clock_mhz_scaled = clock_mhz_scaled >> 2;
|
|
}
|
|
|
|
/*
|
|
* ALGO -> coef = 1e8/fcarrier*fclock/40;
|
|
* scaled coef to provide precision for this floating calculation
|
|
*/
|
|
ar9300_get_channel_centers(ah, chan, ¢ers);
|
|
coef_scaled = clock_mhz_scaled / centers.synth_center;
|
|
|
|
ar9300_get_delta_slope_values(ah, coef_scaled, &ds_coef_man, &ds_coef_exp);
|
|
|
|
OS_REG_RMW_FIELD(ah, AR_PHY_TIMING3, AR_PHY_TIMING3_DSC_MAN, ds_coef_man);
|
|
OS_REG_RMW_FIELD(ah, AR_PHY_TIMING3, AR_PHY_TIMING3_DSC_EXP, ds_coef_exp);
|
|
|
|
/*
|
|
* For Short GI,
|
|
* scaled coeff is 9/10 that of normal coeff
|
|
*/
|
|
coef_scaled = (9 * coef_scaled) / 10;
|
|
|
|
ar9300_get_delta_slope_values(ah, coef_scaled, &ds_coef_man, &ds_coef_exp);
|
|
|
|
/* for short gi */
|
|
OS_REG_RMW_FIELD(ah, AR_PHY_SGI_DELTA, AR_PHY_SGI_DSC_MAN, ds_coef_man);
|
|
OS_REG_RMW_FIELD(ah, AR_PHY_SGI_DELTA, AR_PHY_SGI_DSC_EXP, ds_coef_exp);
|
|
}
|
|
|
|
#define IS(_c, _f) (IEEE80211_IS_ ## _f(_c))
|
|
|
|
/*
|
|
* XXX FreeBSD: This should be turned into something generic in ath_hal!
|
|
*/
|
|
HAL_CHANNEL_INTERNAL *
|
|
ar9300_check_chan(struct ath_hal *ah, const struct ieee80211_channel *chan)
|
|
{
|
|
|
|
if (chan == NULL) {
|
|
return AH_NULL;
|
|
}
|
|
|
|
if ((IS(chan, CHAN_2GHZ) ^ IS(chan, CHAN_5GHZ)) == 0) {
|
|
HALDEBUG(ah, HAL_DEBUG_CHANNEL,
|
|
"%s: invalid channel %u/0x%x; not marked as 2GHz or 5GHz\n",
|
|
__func__, chan->ic_freq , chan->ic_flags);
|
|
return AH_NULL;
|
|
}
|
|
|
|
/*
|
|
* FreeBSD sets multiple flags, so this will fail.
|
|
*/
|
|
#if 0
|
|
if ((IS(chan, CHAN_OFDM) ^ IS(chan, CHAN_CCK) ^ IS(chan, CHAN_DYN) ^
|
|
IS(chan, CHAN_HT20) ^ IS(chan, CHAN_HT40U) ^
|
|
IS(chan, CHAN_HT40D)) == 0)
|
|
{
|
|
HALDEBUG(ah, HAL_DEBUG_CHANNEL,
|
|
"%s: invalid channel %u/0x%x; not marked as "
|
|
"OFDM or CCK or DYN or HT20 or HT40PLUS or HT40MINUS\n",
|
|
__func__, chan->ic_freq , chan->ic_flags);
|
|
return AH_NULL;
|
|
}
|
|
#endif
|
|
|
|
return (ath_hal_checkchannel(ah, chan));
|
|
}
|
|
#undef IS
|
|
|
|
static void
|
|
ar9300_set_11n_regs(struct ath_hal *ah, struct ieee80211_channel *chan,
|
|
HAL_HT_MACMODE macmode)
|
|
{
|
|
u_int32_t phymode;
|
|
// struct ath_hal_9300 *ahp = AH9300(ah);
|
|
u_int32_t enable_dac_fifo;
|
|
|
|
/* XXX */
|
|
enable_dac_fifo =
|
|
OS_REG_READ(ah, AR_PHY_GEN_CTRL) & AR_PHY_GC_ENABLE_DAC_FIFO;
|
|
|
|
/* Enable 11n HT, 20 MHz */
|
|
phymode =
|
|
AR_PHY_GC_HT_EN | AR_PHY_GC_SINGLE_HT_LTF1 | AR_PHY_GC_SHORT_GI_40
|
|
| enable_dac_fifo;
|
|
/* Configure baseband for dynamic 20/40 operation */
|
|
if (IEEE80211_IS_CHAN_HT40(chan)) {
|
|
phymode |= AR_PHY_GC_DYN2040_EN;
|
|
/* Configure control (primary) channel at +-10MHz */
|
|
if (IEEE80211_IS_CHAN_HT40U(chan)) {
|
|
phymode |= AR_PHY_GC_DYN2040_PRI_CH;
|
|
}
|
|
|
|
#if 0
|
|
/* Configure 20/25 spacing */
|
|
if (ahp->ah_ext_prot_spacing == HAL_HT_EXTPROTSPACING_25) {
|
|
phymode |= AR_PHY_GC_DYN2040_EXT_CH;
|
|
}
|
|
#endif
|
|
}
|
|
|
|
/* make sure we preserve INI settings */
|
|
phymode |= OS_REG_READ(ah, AR_PHY_GEN_CTRL);
|
|
|
|
/* EV 62881/64991 - turn off Green Field detection for Maverick STA beta */
|
|
phymode &= ~AR_PHY_GC_GF_DETECT_EN;
|
|
|
|
OS_REG_WRITE(ah, AR_PHY_GEN_CTRL, phymode);
|
|
|
|
/* Set IFS timing for half/quarter rates */
|
|
if (IEEE80211_IS_CHAN_HALF(chan) || IEEE80211_IS_CHAN_QUARTER(chan)) {
|
|
u_int32_t modeselect = OS_REG_READ(ah, AR_PHY_MODE);
|
|
|
|
if (IEEE80211_IS_CHAN_HALF(chan)) {
|
|
modeselect |= AR_PHY_MS_HALF_RATE;
|
|
} else if (IEEE80211_IS_CHAN_QUARTER(chan)) {
|
|
modeselect |= AR_PHY_MS_QUARTER_RATE;
|
|
}
|
|
OS_REG_WRITE(ah, AR_PHY_MODE, modeselect);
|
|
|
|
ar9300_set_ifs_timing(ah, chan);
|
|
OS_REG_RMW_FIELD(
|
|
ah, AR_PHY_FRAME_CTL, AR_PHY_FRAME_CTL_CF_OVERLAP_WINDOW, 0x3);
|
|
}
|
|
|
|
/* Configure MAC for 20/40 operation */
|
|
ar9300_set_11n_mac2040(ah, macmode);
|
|
|
|
/* global transmit timeout (25 TUs default)*/
|
|
/* XXX - put this elsewhere??? */
|
|
OS_REG_WRITE(ah, AR_GTXTO, 25 << AR_GTXTO_TIMEOUT_LIMIT_S);
|
|
|
|
/* carrier sense timeout */
|
|
OS_REG_WRITE(ah, AR_CST, 0xF << AR_CST_TIMEOUT_LIMIT_S);
|
|
}
|
|
|
|
/*
|
|
* Spur mitigation for MRC CCK
|
|
*/
|
|
static void
|
|
ar9300_spur_mitigate_mrc_cck(struct ath_hal *ah, struct ieee80211_channel *chan)
|
|
{
|
|
int i;
|
|
/* spur_freq_for_osprey - hardcoded by Systems team for now. */
|
|
u_int32_t spur_freq_for_osprey[4] = { 2420, 2440, 2464, 2480 };
|
|
u_int32_t spur_freq_for_jupiter[2] = { 2440, 2464};
|
|
int cur_bb_spur, negative = 0, cck_spur_freq;
|
|
u_int8_t* spur_fbin_ptr = NULL;
|
|
int synth_freq;
|
|
int range = 10;
|
|
int max_spurcounts = OSPREY_EEPROM_MODAL_SPURS;
|
|
HAL_CHANNEL_INTERNAL *ichan = ath_hal_checkchannel(ah, chan);
|
|
|
|
/*
|
|
* Need to verify range +/- 10 MHz in control channel, otherwise spur
|
|
* is out-of-band and can be ignored.
|
|
*/
|
|
if (AR_SREV_HORNET(ah) || AR_SREV_POSEIDON(ah) ||
|
|
AR_SREV_WASP(ah) || AR_SREV_SCORPION(ah)) {
|
|
spur_fbin_ptr = ar9300_eeprom_get_spur_chans_ptr(ah, 1);
|
|
if (spur_fbin_ptr[0] == 0) {
|
|
return; /* No spur in the mode */
|
|
}
|
|
if (IEEE80211_IS_CHAN_HT40(chan)) {
|
|
range = 19;
|
|
if (OS_REG_READ_FIELD(ah, AR_PHY_GEN_CTRL, AR_PHY_GC_DYN2040_PRI_CH)
|
|
== 0x0)
|
|
{
|
|
synth_freq = ichan->channel + 10;
|
|
} else {
|
|
synth_freq = ichan->channel - 10;
|
|
}
|
|
} else {
|
|
range = 10;
|
|
synth_freq = ichan->channel;
|
|
}
|
|
} else if(AR_SREV_JUPITER(ah)) {
|
|
range = 5;
|
|
max_spurcounts = 2; /* Hardcoded by Jupiter Systems team for now. */
|
|
synth_freq = ichan->channel;
|
|
} else {
|
|
range = 10;
|
|
max_spurcounts = 4; /* Hardcoded by Osprey Systems team for now. */
|
|
synth_freq = ichan->channel;
|
|
}
|
|
|
|
for (i = 0; i < max_spurcounts; i++) {
|
|
negative = 0;
|
|
|
|
if (AR_SREV_HORNET(ah) || AR_SREV_POSEIDON(ah) ||
|
|
AR_SREV_WASP(ah) || AR_SREV_SCORPION(ah)) {
|
|
cur_bb_spur =
|
|
FBIN2FREQ(spur_fbin_ptr[i], HAL_FREQ_BAND_2GHZ) - synth_freq;
|
|
} else if(AR_SREV_JUPITER(ah)) {
|
|
cur_bb_spur = spur_freq_for_jupiter[i] - synth_freq;
|
|
} else {
|
|
cur_bb_spur = spur_freq_for_osprey[i] - synth_freq;
|
|
}
|
|
|
|
if (cur_bb_spur < 0) {
|
|
negative = 1;
|
|
cur_bb_spur = -cur_bb_spur;
|
|
}
|
|
if (cur_bb_spur < range) {
|
|
cck_spur_freq = (int)((cur_bb_spur << 19) / 11);
|
|
if (negative == 1) {
|
|
cck_spur_freq = -cck_spur_freq;
|
|
}
|
|
cck_spur_freq = cck_spur_freq & 0xfffff;
|
|
/*OS_REG_WRITE_field(ah, BB_agc_control.ycok_max, 0x7);*/
|
|
OS_REG_RMW_FIELD(ah,
|
|
AR_PHY_AGC_CONTROL, AR_PHY_AGC_CONTROL_YCOK_MAX, 0x7);
|
|
/*OS_REG_WRITE_field(ah, BB_cck_spur_mit.spur_rssi_thr, 0x7f);*/
|
|
OS_REG_RMW_FIELD(ah,
|
|
AR_PHY_CCK_SPUR_MIT, AR_PHY_CCK_SPUR_MIT_SPUR_RSSI_THR, 0x7f);
|
|
/*OS_REG_WRITE(ah, BB_cck_spur_mit.spur_filter_type, 0x2);*/
|
|
OS_REG_RMW_FIELD(ah,
|
|
AR_PHY_CCK_SPUR_MIT, AR_PHY_CCK_SPUR_MIT_SPUR_FILTER_TYPE, 0x2);
|
|
/*OS_REG_WRITE(ah, BB_cck_spur_mit.use_cck_spur_mit, 0x1);*/
|
|
OS_REG_RMW_FIELD(ah,
|
|
AR_PHY_CCK_SPUR_MIT, AR_PHY_CCK_SPUR_MIT_USE_CCK_SPUR_MIT, 0x1);
|
|
/*OS_REG_WRITE(ah, BB_cck_spur_mit.cck_spur_freq, cck_spur_freq);*/
|
|
OS_REG_RMW_FIELD(ah,
|
|
AR_PHY_CCK_SPUR_MIT, AR_PHY_CCK_SPUR_MIT_CCK_SPUR_FREQ,
|
|
cck_spur_freq);
|
|
return;
|
|
}
|
|
}
|
|
|
|
/*OS_REG_WRITE(ah, BB_agc_control.ycok_max, 0x5);*/
|
|
OS_REG_RMW_FIELD(ah, AR_PHY_AGC_CONTROL, AR_PHY_AGC_CONTROL_YCOK_MAX, 0x5);
|
|
/*OS_REG_WRITE(ah, BB_cck_spur_mit.use_cck_spur_mit, 0x0);*/
|
|
OS_REG_RMW_FIELD(ah,
|
|
AR_PHY_CCK_SPUR_MIT, AR_PHY_CCK_SPUR_MIT_USE_CCK_SPUR_MIT, 0x0);
|
|
/*OS_REG_WRITE(ah, BB_cck_spur_mit.cck_spur_freq, 0x0);*/
|
|
OS_REG_RMW_FIELD(ah,
|
|
AR_PHY_CCK_SPUR_MIT, AR_PHY_CCK_SPUR_MIT_CCK_SPUR_FREQ, 0x0);
|
|
}
|
|
|
|
/* Spur mitigation for OFDM */
|
|
static void
|
|
ar9300_spur_mitigate_ofdm(struct ath_hal *ah, struct ieee80211_channel *chan)
|
|
{
|
|
int synth_freq;
|
|
int range = 10;
|
|
int freq_offset = 0;
|
|
int spur_freq_sd = 0;
|
|
int spur_subchannel_sd = 0;
|
|
int spur_delta_phase = 0;
|
|
int mask_index = 0;
|
|
int i;
|
|
int mode;
|
|
u_int8_t* spur_chans_ptr;
|
|
struct ath_hal_9300 *ahp;
|
|
ahp = AH9300(ah);
|
|
HAL_CHANNEL_INTERNAL *ichan = ath_hal_checkchannel(ah, chan);
|
|
|
|
if (IS_CHAN_5GHZ(ichan)) {
|
|
spur_chans_ptr = ar9300_eeprom_get_spur_chans_ptr(ah, 0);
|
|
mode = 0;
|
|
} else {
|
|
spur_chans_ptr = ar9300_eeprom_get_spur_chans_ptr(ah, 1);
|
|
mode = 1;
|
|
}
|
|
|
|
if (IEEE80211_IS_CHAN_HT40(chan)) {
|
|
range = 19;
|
|
if (OS_REG_READ_FIELD(ah, AR_PHY_GEN_CTRL, AR_PHY_GC_DYN2040_PRI_CH)
|
|
== 0x0)
|
|
{
|
|
synth_freq = ichan->channel - 10;
|
|
} else {
|
|
synth_freq = ichan->channel + 10;
|
|
}
|
|
} else {
|
|
range = 10;
|
|
synth_freq = ichan->channel;
|
|
}
|
|
|
|
/* Clean all spur register fields */
|
|
OS_REG_RMW_FIELD(ah, AR_PHY_TIMING4, AR_PHY_TIMING4_ENABLE_SPUR_FILTER, 0);
|
|
OS_REG_RMW_FIELD(ah, AR_PHY_TIMING11, AR_PHY_TIMING11_SPUR_FREQ_SD, 0);
|
|
OS_REG_RMW_FIELD(ah, AR_PHY_TIMING11, AR_PHY_TIMING11_SPUR_DELTA_PHASE, 0);
|
|
OS_REG_RMW_FIELD(ah,
|
|
AR_PHY_SFCORR_EXT, AR_PHY_SFCORR_EXT_SPUR_SUBCHANNEL_SD, 0);
|
|
OS_REG_RMW_FIELD(ah,
|
|
AR_PHY_TIMING11, AR_PHY_TIMING11_USE_SPUR_FILTER_IN_AGC, 0);
|
|
OS_REG_RMW_FIELD(ah,
|
|
AR_PHY_TIMING11, AR_PHY_TIMING11_USE_SPUR_FILTER_IN_SELFCOR, 0);
|
|
OS_REG_RMW_FIELD(ah, AR_PHY_TIMING4, AR_PHY_TIMING4_ENABLE_SPUR_RSSI, 0);
|
|
OS_REG_RMW_FIELD(ah, AR_PHY_SPUR_REG, AR_PHY_SPUR_REG_EN_VIT_SPUR_RSSI, 0);
|
|
OS_REG_RMW_FIELD(ah,
|
|
AR_PHY_SPUR_REG, AR_PHY_SPUR_REG_ENABLE_NF_RSSI_SPUR_MIT, 0);
|
|
OS_REG_RMW_FIELD(ah, AR_PHY_SPUR_REG, AR_PHY_SPUR_REG_ENABLE_MASK_PPM, 0);
|
|
OS_REG_RMW_FIELD(ah, AR_PHY_TIMING4, AR_PHY_TIMING4_ENABLE_PILOT_MASK, 0);
|
|
OS_REG_RMW_FIELD(ah, AR_PHY_TIMING4, AR_PHY_TIMING4_ENABLE_CHAN_MASK, 0);
|
|
OS_REG_RMW_FIELD(ah,
|
|
AR_PHY_PILOT_SPUR_MASK, AR_PHY_PILOT_SPUR_MASK_CF_PILOT_MASK_IDX_A, 0);
|
|
OS_REG_RMW_FIELD(ah,
|
|
AR_PHY_SPUR_MASK_A, AR_PHY_SPUR_MASK_A_CF_PUNC_MASK_IDX_A, 0);
|
|
OS_REG_RMW_FIELD(ah,
|
|
AR_PHY_CHAN_SPUR_MASK, AR_PHY_CHAN_SPUR_MASK_CF_CHAN_MASK_IDX_A, 0);
|
|
OS_REG_RMW_FIELD(ah,
|
|
AR_PHY_PILOT_SPUR_MASK, AR_PHY_PILOT_SPUR_MASK_CF_PILOT_MASK_A, 0);
|
|
OS_REG_RMW_FIELD(ah,
|
|
AR_PHY_CHAN_SPUR_MASK, AR_PHY_CHAN_SPUR_MASK_CF_CHAN_MASK_A, 0);
|
|
OS_REG_RMW_FIELD(ah,
|
|
AR_PHY_SPUR_MASK_A, AR_PHY_SPUR_MASK_A_CF_PUNC_MASK_A, 0);
|
|
OS_REG_RMW_FIELD(ah, AR_PHY_SPUR_REG, AR_PHY_SPUR_REG_MASK_RATE_CNTL, 0);
|
|
|
|
i = 0;
|
|
while (spur_chans_ptr[i] && i < 5) {
|
|
freq_offset = FBIN2FREQ(spur_chans_ptr[i], mode) - synth_freq;
|
|
if (abs(freq_offset) < range) {
|
|
/*
|
|
printf(
|
|
"Spur Mitigation for OFDM: Synth Frequency = %d, "
|
|
"Spur Frequency = %d\n",
|
|
synth_freq, FBIN2FREQ(spur_chans_ptr[i], mode));
|
|
*/
|
|
if (IEEE80211_IS_CHAN_HT40(chan)) {
|
|
if (freq_offset < 0) {
|
|
if (OS_REG_READ_FIELD(
|
|
ah, AR_PHY_GEN_CTRL, AR_PHY_GC_DYN2040_PRI_CH) == 0x0)
|
|
{
|
|
spur_subchannel_sd = 1;
|
|
} else {
|
|
spur_subchannel_sd = 0;
|
|
}
|
|
spur_freq_sd = ((freq_offset + 10) << 9) / 11;
|
|
} else {
|
|
if (OS_REG_READ_FIELD(ah,
|
|
AR_PHY_GEN_CTRL, AR_PHY_GC_DYN2040_PRI_CH) == 0x0)
|
|
{
|
|
spur_subchannel_sd = 0;
|
|
} else {
|
|
spur_subchannel_sd = 1;
|
|
}
|
|
spur_freq_sd = ((freq_offset - 10) << 9) / 11;
|
|
}
|
|
spur_delta_phase = (freq_offset << 17) / 5;
|
|
} else {
|
|
spur_subchannel_sd = 0;
|
|
spur_freq_sd = (freq_offset << 9) / 11;
|
|
spur_delta_phase = (freq_offset << 18) / 5;
|
|
}
|
|
spur_freq_sd = spur_freq_sd & 0x3ff;
|
|
spur_delta_phase = spur_delta_phase & 0xfffff;
|
|
/*
|
|
printf(
|
|
"spur_subchannel_sd = %d, spur_freq_sd = 0x%x, "
|
|
"spur_delta_phase = 0x%x\n", spur_subchannel_sd,
|
|
spur_freq_sd, spur_delta_phase);
|
|
*/
|
|
|
|
/* OFDM Spur mitigation */
|
|
OS_REG_RMW_FIELD(ah,
|
|
AR_PHY_TIMING4, AR_PHY_TIMING4_ENABLE_SPUR_FILTER, 0x1);
|
|
OS_REG_RMW_FIELD(ah,
|
|
AR_PHY_TIMING11, AR_PHY_TIMING11_SPUR_FREQ_SD, spur_freq_sd);
|
|
OS_REG_RMW_FIELD(ah,
|
|
AR_PHY_TIMING11, AR_PHY_TIMING11_SPUR_DELTA_PHASE,
|
|
spur_delta_phase);
|
|
OS_REG_RMW_FIELD(ah,
|
|
AR_PHY_SFCORR_EXT, AR_PHY_SFCORR_EXT_SPUR_SUBCHANNEL_SD,
|
|
spur_subchannel_sd);
|
|
OS_REG_RMW_FIELD(ah,
|
|
AR_PHY_TIMING11, AR_PHY_TIMING11_USE_SPUR_FILTER_IN_AGC, 0x1);
|
|
OS_REG_RMW_FIELD(ah,
|
|
AR_PHY_TIMING11, AR_PHY_TIMING11_USE_SPUR_FILTER_IN_SELFCOR,
|
|
0x1);
|
|
OS_REG_RMW_FIELD(ah,
|
|
AR_PHY_TIMING4, AR_PHY_TIMING4_ENABLE_SPUR_RSSI, 0x1);
|
|
OS_REG_RMW_FIELD(ah,
|
|
AR_PHY_SPUR_REG, AR_PHY_SPUR_REG_SPUR_RSSI_THRESH, 34);
|
|
OS_REG_RMW_FIELD(ah,
|
|
AR_PHY_SPUR_REG, AR_PHY_SPUR_REG_EN_VIT_SPUR_RSSI, 1);
|
|
|
|
/*
|
|
* Do not subtract spur power from noise floor for wasp.
|
|
* This causes the maximum client test (on Veriwave) to fail
|
|
* when run on spur channel (2464 MHz).
|
|
* Refer to ev#82746 and ev#82744.
|
|
*/
|
|
if (!AR_SREV_WASP(ah) && (OS_REG_READ_FIELD(ah, AR_PHY_MODE,
|
|
AR_PHY_MODE_DYNAMIC) == 0x1)) {
|
|
OS_REG_RMW_FIELD(ah, AR_PHY_SPUR_REG,
|
|
AR_PHY_SPUR_REG_ENABLE_NF_RSSI_SPUR_MIT, 1);
|
|
}
|
|
|
|
mask_index = (freq_offset << 4) / 5;
|
|
if (mask_index < 0) {
|
|
mask_index = mask_index - 1;
|
|
}
|
|
mask_index = mask_index & 0x7f;
|
|
/*printf("Bin 0x%x\n", mask_index);*/
|
|
|
|
OS_REG_RMW_FIELD(ah,
|
|
AR_PHY_SPUR_REG, AR_PHY_SPUR_REG_ENABLE_MASK_PPM, 0x1);
|
|
OS_REG_RMW_FIELD(ah,
|
|
AR_PHY_TIMING4, AR_PHY_TIMING4_ENABLE_PILOT_MASK, 0x1);
|
|
OS_REG_RMW_FIELD(ah,
|
|
AR_PHY_TIMING4, AR_PHY_TIMING4_ENABLE_CHAN_MASK, 0x1);
|
|
OS_REG_RMW_FIELD(ah,
|
|
AR_PHY_PILOT_SPUR_MASK,
|
|
AR_PHY_PILOT_SPUR_MASK_CF_PILOT_MASK_IDX_A, mask_index);
|
|
OS_REG_RMW_FIELD(ah,
|
|
AR_PHY_SPUR_MASK_A, AR_PHY_SPUR_MASK_A_CF_PUNC_MASK_IDX_A,
|
|
mask_index);
|
|
OS_REG_RMW_FIELD(ah,
|
|
AR_PHY_CHAN_SPUR_MASK,
|
|
AR_PHY_CHAN_SPUR_MASK_CF_CHAN_MASK_IDX_A, mask_index);
|
|
OS_REG_RMW_FIELD(ah,
|
|
AR_PHY_PILOT_SPUR_MASK, AR_PHY_PILOT_SPUR_MASK_CF_PILOT_MASK_A,
|
|
0xc);
|
|
OS_REG_RMW_FIELD(ah,
|
|
AR_PHY_CHAN_SPUR_MASK, AR_PHY_CHAN_SPUR_MASK_CF_CHAN_MASK_A,
|
|
0xc);
|
|
OS_REG_RMW_FIELD(ah,
|
|
AR_PHY_SPUR_MASK_A, AR_PHY_SPUR_MASK_A_CF_PUNC_MASK_A, 0xa0);
|
|
OS_REG_RMW_FIELD(ah,
|
|
AR_PHY_SPUR_REG, AR_PHY_SPUR_REG_MASK_RATE_CNTL, 0xff);
|
|
/*
|
|
printf("BB_timing_control_4 = 0x%x\n",
|
|
OS_REG_READ(ah, AR_PHY_TIMING4));
|
|
printf("BB_timing_control_11 = 0x%x\n",
|
|
OS_REG_READ(ah, AR_PHY_TIMING11));
|
|
printf("BB_ext_chan_scorr_thr = 0x%x\n",
|
|
OS_REG_READ(ah, AR_PHY_SFCORR_EXT));
|
|
printf("BB_spur_mask_controls = 0x%x\n",
|
|
OS_REG_READ(ah, AR_PHY_SPUR_REG));
|
|
printf("BB_pilot_spur_mask = 0x%x\n",
|
|
OS_REG_READ(ah, AR_PHY_PILOT_SPUR_MASK));
|
|
printf("BB_chan_spur_mask = 0x%x\n",
|
|
OS_REG_READ(ah, AR_PHY_CHAN_SPUR_MASK));
|
|
printf("BB_vit_spur_mask_A = 0x%x\n",
|
|
OS_REG_READ(ah, AR_PHY_SPUR_MASK_A));
|
|
*/
|
|
break;
|
|
}
|
|
i++;
|
|
}
|
|
}
|
|
|
|
|
|
/*
|
|
* Convert to baseband spur frequency given input channel frequency
|
|
* and compute register settings below.
|
|
*/
|
|
static void
|
|
ar9300_spur_mitigate(struct ath_hal *ah, struct ieee80211_channel *chan)
|
|
{
|
|
ar9300_spur_mitigate_ofdm(ah, chan);
|
|
ar9300_spur_mitigate_mrc_cck(ah, chan);
|
|
}
|
|
|
|
/**************************************************************
|
|
* ar9300_channel_change
|
|
* Assumes caller wants to change channel, and not reset.
|
|
*/
|
|
static inline HAL_BOOL
|
|
ar9300_channel_change(struct ath_hal *ah, struct ieee80211_channel *chan,
|
|
HAL_CHANNEL_INTERNAL *ichan, HAL_HT_MACMODE macmode)
|
|
{
|
|
|
|
u_int32_t synth_delay, qnum;
|
|
struct ath_hal_9300 *ahp = AH9300(ah);
|
|
|
|
/* TX must be stopped by now */
|
|
for (qnum = 0; qnum < AR_NUM_QCU; qnum++) {
|
|
if (ar9300_num_tx_pending(ah, qnum)) {
|
|
HALDEBUG(ah, HAL_DEBUG_QUEUE,
|
|
"%s: Transmit frames pending on queue %d\n", __func__, qnum);
|
|
HALASSERT(0);
|
|
return AH_FALSE;
|
|
}
|
|
}
|
|
|
|
|
|
/*
|
|
* Kill last Baseband Rx Frame - Request analog bus grant
|
|
*/
|
|
OS_REG_WRITE(ah, AR_PHY_RFBUS_REQ, AR_PHY_RFBUS_REQ_EN);
|
|
if (!ath_hal_wait(ah, AR_PHY_RFBUS_GRANT, AR_PHY_RFBUS_GRANT_EN,
|
|
AR_PHY_RFBUS_GRANT_EN))
|
|
{
|
|
HALDEBUG(ah, HAL_DEBUG_PHYIO,
|
|
"%s: Could not kill baseband RX\n", __func__);
|
|
return AH_FALSE;
|
|
}
|
|
|
|
|
|
/* Setup 11n MAC/Phy mode registers */
|
|
ar9300_set_11n_regs(ah, chan, macmode);
|
|
|
|
/*
|
|
* Change the synth
|
|
*/
|
|
if (!ahp->ah_rf_hal.set_channel(ah, chan)) {
|
|
HALDEBUG(ah, HAL_DEBUG_CHANNEL, "%s: failed to set channel\n", __func__);
|
|
return AH_FALSE;
|
|
}
|
|
|
|
/*
|
|
* Some registers get reinitialized during ATH_INI_POST INI programming.
|
|
*/
|
|
ar9300_init_user_settings(ah);
|
|
|
|
/*
|
|
* Setup the transmit power values.
|
|
*
|
|
* After the public to private hal channel mapping, ichan contains the
|
|
* valid regulatory power value.
|
|
* ath_hal_getctl and ath_hal_getantennaallowed look up ichan from chan.
|
|
*/
|
|
if (ar9300_eeprom_set_transmit_power(
|
|
ah, &ahp->ah_eeprom, chan, ath_hal_getctl(ah, chan),
|
|
ath_hal_getantennaallowed(ah, chan),
|
|
ath_hal_get_twice_max_regpower(AH_PRIVATE(ah), ichan, chan),
|
|
AH_MIN(MAX_RATE_POWER, AH_PRIVATE(ah)->ah_powerLimit)) != HAL_OK)
|
|
{
|
|
HALDEBUG(ah, HAL_DEBUG_EEPROM,
|
|
"%s: error init'ing transmit power\n", __func__);
|
|
return AH_FALSE;
|
|
}
|
|
|
|
/*
|
|
* Release the RFBus Grant.
|
|
*/
|
|
OS_REG_WRITE(ah, AR_PHY_RFBUS_REQ, 0);
|
|
|
|
/*
|
|
* Write spur immunity and delta slope for OFDM enabled modes (A, G, Turbo)
|
|
*/
|
|
if (IEEE80211_IS_CHAN_OFDM(chan) || IEEE80211_IS_CHAN_HT(chan)) {
|
|
ar9300_set_delta_slope(ah, chan);
|
|
} else {
|
|
/* Set to Ini default */
|
|
OS_REG_WRITE(ah, AR_PHY_TIMING3, 0x9c0a9f6b);
|
|
OS_REG_WRITE(ah, AR_PHY_SGI_DELTA, 0x00046384);
|
|
}
|
|
|
|
ar9300_spur_mitigate(ah, chan);
|
|
|
|
|
|
/*
|
|
* Wait for the frequency synth to settle (synth goes on via PHY_ACTIVE_EN).
|
|
* Read the phy active delay register. Value is in 100ns increments.
|
|
*/
|
|
synth_delay = OS_REG_READ(ah, AR_PHY_RX_DELAY) & AR_PHY_RX_DELAY_DELAY;
|
|
if (IEEE80211_IS_CHAN_CCK(chan)) {
|
|
synth_delay = (4 * synth_delay) / 22;
|
|
} else {
|
|
synth_delay /= 10;
|
|
}
|
|
|
|
OS_DELAY(synth_delay + BASE_ACTIVATE_DELAY);
|
|
|
|
/*
|
|
* Do calibration.
|
|
*/
|
|
|
|
return AH_TRUE;
|
|
}
|
|
|
|
void
|
|
ar9300_set_operating_mode(struct ath_hal *ah, int opmode)
|
|
{
|
|
u_int32_t val;
|
|
|
|
val = OS_REG_READ(ah, AR_STA_ID1);
|
|
val &= ~(AR_STA_ID1_STA_AP | AR_STA_ID1_ADHOC);
|
|
switch (opmode) {
|
|
case HAL_M_HOSTAP:
|
|
OS_REG_WRITE(ah, AR_STA_ID1,
|
|
val | AR_STA_ID1_STA_AP | AR_STA_ID1_KSRCH_MODE);
|
|
OS_REG_CLR_BIT(ah, AR_CFG, AR_CFG_AP_ADHOC_INDICATION);
|
|
break;
|
|
case HAL_M_IBSS:
|
|
OS_REG_WRITE(ah, AR_STA_ID1,
|
|
val | AR_STA_ID1_ADHOC | AR_STA_ID1_KSRCH_MODE);
|
|
OS_REG_SET_BIT(ah, AR_CFG, AR_CFG_AP_ADHOC_INDICATION);
|
|
break;
|
|
case HAL_M_STA:
|
|
case HAL_M_MONITOR:
|
|
OS_REG_WRITE(ah, AR_STA_ID1, val | AR_STA_ID1_KSRCH_MODE);
|
|
break;
|
|
}
|
|
}
|
|
|
|
/* XXX need the logic for Osprey */
|
|
void
|
|
ar9300_init_pll(struct ath_hal *ah, struct ieee80211_channel *chan)
|
|
{
|
|
u_int32_t pll;
|
|
u_int8_t clk_25mhz = AH9300(ah)->clk_25mhz;
|
|
HAL_CHANNEL_INTERNAL *ichan = NULL;
|
|
|
|
if (chan)
|
|
ichan = ath_hal_checkchannel(ah, chan);
|
|
|
|
if (AR_SREV_HORNET(ah)) {
|
|
if (clk_25mhz) {
|
|
/* Hornet uses PLL_CONTROL_2. Xtal is 25MHz for Hornet.
|
|
* REFDIV set to 0x1.
|
|
* $xtal_freq = 25;
|
|
* $PLL2_div = (704/$xtal_freq); # 176 * 4 = 704.
|
|
* MAC and BB run at 176 MHz.
|
|
* $PLL2_divint = int($PLL2_div);
|
|
* $PLL2_divfrac = $PLL2_div - $PLL2_divint;
|
|
* $PLL2_divfrac = int($PLL2_divfrac * 0x4000); # 2^14
|
|
* $PLL2_Val = ($PLL2_divint & 0x3f) << 19 | (0x1) << 14 |
|
|
* $PLL2_divfrac & 0x3fff;
|
|
* Therefore, $PLL2_Val = 0xe04a3d
|
|
*/
|
|
#define DPLL2_KD_VAL 0x1D
|
|
#define DPLL2_KI_VAL 0x06
|
|
#define DPLL3_PHASE_SHIFT_VAL 0x1
|
|
|
|
/* Rewrite DDR PLL2 and PLL3 */
|
|
/* program DDR PLL ki and kd value, ki=0x6, kd=0x1d */
|
|
OS_REG_WRITE(ah, AR_HORNET_CH0_DDR_DPLL2, 0x18e82f01);
|
|
|
|
/* program DDR PLL phase_shift to 0x1 */
|
|
OS_REG_RMW_FIELD(ah, AR_HORNET_CH0_DDR_DPLL3,
|
|
AR_PHY_BB_DPLL3_PHASE_SHIFT, DPLL3_PHASE_SHIFT_VAL);
|
|
|
|
OS_REG_WRITE(ah, AR_RTC_PLL_CONTROL, 0x1142c);
|
|
OS_DELAY(1000);
|
|
|
|
/* program refdiv, nint, frac to RTC register */
|
|
OS_REG_WRITE(ah, AR_RTC_PLL_CONTROL2, 0xe04a3d);
|
|
|
|
/* program BB PLL ki and kd value, ki=0x6, kd=0x1d */
|
|
OS_REG_RMW_FIELD(ah, AR_PHY_BB_DPLL2,
|
|
AR_PHY_BB_DPLL2_KD, DPLL2_KD_VAL);
|
|
OS_REG_RMW_FIELD(ah, AR_PHY_BB_DPLL2,
|
|
AR_PHY_BB_DPLL2_KI, DPLL2_KI_VAL);
|
|
|
|
/* program BB PLL phase_shift to 0x1 */
|
|
OS_REG_RMW_FIELD(ah, AR_PHY_BB_DPLL3,
|
|
AR_PHY_BB_DPLL3_PHASE_SHIFT, DPLL3_PHASE_SHIFT_VAL);
|
|
} else { /* 40MHz */
|
|
#undef DPLL2_KD_VAL
|
|
#undef DPLL2_KI_VAL
|
|
#define DPLL2_KD_VAL 0x3D
|
|
#define DPLL2_KI_VAL 0x06
|
|
/* Rewrite DDR PLL2 and PLL3 */
|
|
/* program DDR PLL ki and kd value, ki=0x6, kd=0x3d */
|
|
OS_REG_WRITE(ah, AR_HORNET_CH0_DDR_DPLL2, 0x19e82f01);
|
|
|
|
/* program DDR PLL phase_shift to 0x1 */
|
|
OS_REG_RMW_FIELD(ah, AR_HORNET_CH0_DDR_DPLL3,
|
|
AR_PHY_BB_DPLL3_PHASE_SHIFT, DPLL3_PHASE_SHIFT_VAL);
|
|
|
|
OS_REG_WRITE(ah, AR_RTC_PLL_CONTROL, 0x1142c);
|
|
OS_DELAY(1000);
|
|
|
|
/* program refdiv, nint, frac to RTC register */
|
|
OS_REG_WRITE(ah, AR_RTC_PLL_CONTROL2, 0x886666);
|
|
|
|
/* program BB PLL ki and kd value, ki=0x6, kd=0x3d */
|
|
OS_REG_RMW_FIELD(ah, AR_PHY_BB_DPLL2,
|
|
AR_PHY_BB_DPLL2_KD, DPLL2_KD_VAL);
|
|
OS_REG_RMW_FIELD(ah, AR_PHY_BB_DPLL2,
|
|
AR_PHY_BB_DPLL2_KI, DPLL2_KI_VAL);
|
|
|
|
/* program BB PLL phase_shift to 0x1 */
|
|
OS_REG_RMW_FIELD(ah, AR_PHY_BB_DPLL3,
|
|
AR_PHY_BB_DPLL3_PHASE_SHIFT, DPLL3_PHASE_SHIFT_VAL);
|
|
}
|
|
OS_REG_WRITE(ah, AR_RTC_PLL_CONTROL, 0x142c);
|
|
OS_DELAY(1000);
|
|
} else if (AR_SREV_POSEIDON(ah) || AR_SREV_APHRODITE(ah)) {
|
|
OS_REG_RMW_FIELD(ah, AR_PHY_BB_DPLL2, AR_PHY_BB_DPLL2_PLL_PWD, 0x1);
|
|
|
|
/* program BB PLL ki and kd value, ki=0x4, kd=0x40 */
|
|
OS_REG_RMW_FIELD(ah, AR_PHY_BB_DPLL2,
|
|
AR_PHY_BB_DPLL2_KD, 0x40);
|
|
OS_REG_RMW_FIELD(ah, AR_PHY_BB_DPLL2,
|
|
AR_PHY_BB_DPLL2_KI, 0x4);
|
|
|
|
OS_REG_RMW_FIELD(ah, AR_PHY_BB_DPLL1,
|
|
AR_PHY_BB_DPLL1_REFDIV, 0x5);
|
|
OS_REG_RMW_FIELD(ah, AR_PHY_BB_DPLL1,
|
|
AR_PHY_BB_DPLL1_NINI, 0x58);
|
|
OS_REG_RMW_FIELD(ah, AR_PHY_BB_DPLL1,
|
|
AR_PHY_BB_DPLL1_NFRAC, 0x0);
|
|
|
|
OS_REG_RMW_FIELD(ah, AR_PHY_BB_DPLL2,
|
|
AR_PHY_BB_DPLL2_OUTDIV, 0x1);
|
|
OS_REG_RMW_FIELD(ah, AR_PHY_BB_DPLL2,
|
|
AR_PHY_BB_DPLL2_LOCAL_PLL, 0x1);
|
|
OS_REG_RMW_FIELD(ah, AR_PHY_BB_DPLL2,
|
|
AR_PHY_BB_DPLL2_EN_NEGTRIG, 0x1);
|
|
|
|
/* program BB PLL phase_shift to 0x6 */
|
|
OS_REG_RMW_FIELD(ah, AR_PHY_BB_DPLL3,
|
|
AR_PHY_BB_DPLL3_PHASE_SHIFT, 0x6);
|
|
|
|
OS_REG_RMW_FIELD(ah, AR_PHY_BB_DPLL2,
|
|
AR_PHY_BB_DPLL2_PLL_PWD, 0x0);
|
|
OS_DELAY(1000);
|
|
|
|
OS_REG_WRITE(ah, AR_RTC_PLL_CONTROL, 0x142c);
|
|
OS_DELAY(1000);
|
|
} else if (AR_SREV_WASP(ah) || AR_SREV_SCORPION(ah) || AR_SREV_HONEYBEE(ah)) {
|
|
#define SRIF_PLL 1
|
|
u_int32_t regdata, pll2_divint, pll2_divfrac;
|
|
|
|
#ifndef SRIF_PLL
|
|
u_int32_t pll2_clkmode;
|
|
#endif
|
|
|
|
#ifdef SRIF_PLL
|
|
u_int32_t refdiv;
|
|
#endif
|
|
if (clk_25mhz) {
|
|
#ifndef SRIF_PLL
|
|
pll2_divint = 0x1c;
|
|
pll2_divfrac = 0xa3d7;
|
|
#else
|
|
if (AR_SREV_HONEYBEE(ah)) {
|
|
pll2_divint = 0x1c;
|
|
pll2_divfrac = 0xa3d2;
|
|
refdiv = 1;
|
|
} else {
|
|
pll2_divint = 0x54;
|
|
pll2_divfrac = 0x1eb85;
|
|
refdiv = 3;
|
|
}
|
|
#endif
|
|
} else {
|
|
#ifndef SRIF_PLL
|
|
pll2_divint = 0x11;
|
|
pll2_divfrac = 0x26666;
|
|
#else
|
|
if (AR_SREV_WASP(ah)) {
|
|
pll2_divint = 88;
|
|
pll2_divfrac = 0;
|
|
refdiv = 5;
|
|
} else {
|
|
pll2_divint = 0x11;
|
|
pll2_divfrac = 0x26666;
|
|
refdiv = 1;
|
|
}
|
|
#endif
|
|
}
|
|
#ifndef SRIF_PLL
|
|
pll2_clkmode = 0x3d;
|
|
#endif
|
|
/* PLL programming through SRIF Local Mode */
|
|
OS_REG_WRITE(ah, AR_RTC_PLL_CONTROL, 0x1142c); /* Bypass mode */
|
|
OS_DELAY(1000);
|
|
do {
|
|
regdata = OS_REG_READ(ah, AR_PHY_PLL_MODE);
|
|
if (AR_SREV_HONEYBEE(ah)) {
|
|
regdata = regdata | (0x1 << 22);
|
|
} else {
|
|
regdata = regdata | (0x1 << 16);
|
|
}
|
|
OS_REG_WRITE(ah, AR_PHY_PLL_MODE, regdata); /* PWD_PLL set to 1 */
|
|
OS_DELAY(100);
|
|
/* override int, frac, refdiv */
|
|
#ifndef SRIF_PLL
|
|
OS_REG_WRITE(ah, AR_PHY_PLL_CONTROL,
|
|
((1 << 27) | (pll2_divint << 18) | pll2_divfrac));
|
|
#else
|
|
OS_REG_WRITE(ah, AR_PHY_PLL_CONTROL,
|
|
((refdiv << 27) | (pll2_divint << 18) | pll2_divfrac));
|
|
#endif
|
|
OS_DELAY(100);
|
|
regdata = OS_REG_READ(ah, AR_PHY_PLL_MODE);
|
|
#ifndef SRIF_PLL
|
|
regdata = (regdata & 0x80071fff) |
|
|
(0x1 << 30) | (0x1 << 13) | (0x6 << 26) | (pll2_clkmode << 19);
|
|
#else
|
|
if (AR_SREV_WASP(ah)) {
|
|
regdata = (regdata & 0x80071fff) |
|
|
(0x1 << 30) | (0x1 << 13) | (0x4 << 26) | (0x18 << 19);
|
|
} else if (AR_SREV_HONEYBEE(ah)) {
|
|
/*
|
|
* Kd=10, Ki=2, Outdiv=1, Local PLL=0, Phase Shift=4
|
|
*/
|
|
regdata = (regdata & 0x01c00fff) |
|
|
(0x1 << 31) | (0x2 << 29) | (0xa << 25) | (0x1 << 19) | (0x6 << 12);
|
|
} else {
|
|
regdata = (regdata & 0x80071fff) |
|
|
(0x3 << 30) | (0x1 << 13) | (0x4 << 26) | (0x60 << 19);
|
|
}
|
|
#endif
|
|
/* Ki, Kd, Local PLL, Outdiv */
|
|
OS_REG_WRITE(ah, AR_PHY_PLL_MODE, regdata);
|
|
regdata = OS_REG_READ(ah, AR_PHY_PLL_MODE);
|
|
if (AR_SREV_HONEYBEE(ah)) {
|
|
regdata = (regdata & 0xffbfffff);
|
|
} else {
|
|
regdata = (regdata & 0xfffeffff);
|
|
}
|
|
OS_REG_WRITE(ah, AR_PHY_PLL_MODE, regdata); /* PWD_PLL set to 0 */
|
|
OS_DELAY(1000);
|
|
if (AR_SREV_WASP(ah)) {
|
|
/* clear do measure */
|
|
regdata = OS_REG_READ(ah, AR_PHY_PLL_BB_DPLL3);
|
|
regdata &= ~(1 << 30);
|
|
OS_REG_WRITE(ah, AR_PHY_PLL_BB_DPLL3, regdata);
|
|
OS_DELAY(100);
|
|
|
|
/* set do measure */
|
|
regdata = OS_REG_READ(ah, AR_PHY_PLL_BB_DPLL3);
|
|
regdata |= (1 << 30);
|
|
OS_REG_WRITE(ah, AR_PHY_PLL_BB_DPLL3, regdata);
|
|
|
|
/* wait for measure done */
|
|
do {
|
|
regdata = OS_REG_READ(ah, AR_PHY_PLL_BB_DPLL4);
|
|
} while ((regdata & (1 << 3)) == 0);
|
|
|
|
/* clear do measure */
|
|
regdata = OS_REG_READ(ah, AR_PHY_PLL_BB_DPLL3);
|
|
regdata &= ~(1 << 30);
|
|
OS_REG_WRITE(ah, AR_PHY_PLL_BB_DPLL3, regdata);
|
|
|
|
/* get measure sqsum dvc */
|
|
regdata = (OS_REG_READ(ah, AR_PHY_PLL_BB_DPLL3) & 0x007FFFF8) >> 3;
|
|
} else {
|
|
break;
|
|
}
|
|
} while (regdata >= 0x40000);
|
|
|
|
/* Remove from Bypass mode */
|
|
OS_REG_WRITE(ah, AR_RTC_PLL_CONTROL, 0x142c);
|
|
OS_DELAY(1000);
|
|
} else {
|
|
pll = SM(0x5, AR_RTC_PLL_REFDIV);
|
|
|
|
/* Supposedly not needed on Osprey */
|
|
#if 0
|
|
if (chan && IS_CHAN_HALF_RATE(chan)) {
|
|
pll |= SM(0x1, AR_RTC_PLL_CLKSEL);
|
|
} else if (chan && IS_CHAN_QUARTER_RATE(chan)) {
|
|
pll |= SM(0x2, AR_RTC_PLL_CLKSEL);
|
|
}
|
|
#endif
|
|
if (ichan && IS_CHAN_5GHZ(ichan)) {
|
|
pll |= SM(0x28, AR_RTC_PLL_DIV);
|
|
/*
|
|
* When doing fast clock, set PLL to 0x142c
|
|
*/
|
|
if (IS_5GHZ_FAST_CLOCK_EN(ah, chan)) {
|
|
pll = 0x142c;
|
|
}
|
|
} else {
|
|
pll |= SM(0x2c, AR_RTC_PLL_DIV);
|
|
}
|
|
|
|
OS_REG_WRITE(ah, AR_RTC_PLL_CONTROL, pll);
|
|
}
|
|
|
|
/* TODO:
|
|
* For multi-band owl, switch between bands by reiniting the PLL.
|
|
*/
|
|
OS_DELAY(RTC_PLL_SETTLE_DELAY);
|
|
|
|
OS_REG_WRITE(ah, AR_RTC_SLEEP_CLK,
|
|
AR_RTC_FORCE_DERIVED_CLK | AR_RTC_PCIE_RST_PWDN_EN);
|
|
|
|
/* XXX TODO: honeybee? */
|
|
if (AR_SREV_WASP(ah) || AR_SREV_SCORPION(ah)) {
|
|
if (clk_25mhz) {
|
|
OS_REG_WRITE(ah,
|
|
AR_RTC_DERIVED_RTC_CLK, (0x17c << 1)); /* 32KHz sleep clk */
|
|
OS_REG_WRITE(ah, AR_SLP32_MODE, 0x0010f3d7);
|
|
OS_REG_WRITE(ah, AR_SLP32_INC, 0x0001e7ae);
|
|
} else {
|
|
OS_REG_WRITE(ah,
|
|
AR_RTC_DERIVED_RTC_CLK, (0x261 << 1)); /* 32KHz sleep clk */
|
|
OS_REG_WRITE(ah, AR_SLP32_MODE, 0x0010f400);
|
|
OS_REG_WRITE(ah, AR_SLP32_INC, 0x0001e800);
|
|
}
|
|
OS_DELAY(100);
|
|
}
|
|
}
|
|
|
|
static inline HAL_BOOL
|
|
ar9300_set_reset(struct ath_hal *ah, int type)
|
|
{
|
|
u_int32_t rst_flags;
|
|
u_int32_t tmp_reg;
|
|
struct ath_hal_9300 *ahp = AH9300(ah);
|
|
|
|
HALASSERT(type == HAL_RESET_WARM || type == HAL_RESET_COLD);
|
|
|
|
/*
|
|
* RTC Force wake should be done before resetting the MAC.
|
|
* MDK/ART does it that way.
|
|
*/
|
|
OS_REG_WRITE(ah, AR_HOSTIF_REG(ah, AR_WA), AH9300(ah)->ah_wa_reg_val);
|
|
OS_DELAY(10); /* delay to allow AR_WA reg write to kick in */
|
|
OS_REG_WRITE(ah,
|
|
AR_RTC_FORCE_WAKE, AR_RTC_FORCE_WAKE_EN | AR_RTC_FORCE_WAKE_ON_INT);
|
|
|
|
/* Reset AHB */
|
|
/* Bug26871 */
|
|
tmp_reg = OS_REG_READ(ah, AR_HOSTIF_REG(ah, AR_INTR_SYNC_CAUSE));
|
|
if (AR_SREV_WASP(ah)) {
|
|
if (tmp_reg & (AR9340_INTR_SYNC_LOCAL_TIMEOUT)) {
|
|
OS_REG_WRITE(ah, AR_HOSTIF_REG(ah, AR_INTR_SYNC_ENABLE), 0);
|
|
OS_REG_WRITE(ah, AR_HOSTIF_REG(ah, AR_RC), AR_RC_HOSTIF);
|
|
}
|
|
} else {
|
|
if (tmp_reg & (AR9300_INTR_SYNC_LOCAL_TIMEOUT | AR9300_INTR_SYNC_RADM_CPL_TIMEOUT)) {
|
|
OS_REG_WRITE(ah, AR_HOSTIF_REG(ah, AR_INTR_SYNC_ENABLE), 0);
|
|
OS_REG_WRITE(ah, AR_HOSTIF_REG(ah, AR_RC), AR_RC_HOSTIF);
|
|
}
|
|
else {
|
|
/* NO AR_RC_AHB in Osprey */
|
|
/*OS_REG_WRITE(ah, AR_HOSTIF_REG(ah, AR_RC), AR_RC_AHB);*/
|
|
}
|
|
}
|
|
|
|
rst_flags = AR_RTC_RC_MAC_WARM;
|
|
if (type == HAL_RESET_COLD) {
|
|
rst_flags |= AR_RTC_RC_MAC_COLD;
|
|
}
|
|
|
|
#ifdef AH_SUPPORT_HORNET
|
|
/* Hornet WAR: trigger SoC to reset WMAC if ...
|
|
* (1) doing cold reset. Ref: EV 69254
|
|
* (2) beacon pending. Ref: EV 70983
|
|
*/
|
|
if (AR_SREV_HORNET(ah) &&
|
|
(ar9300_num_tx_pending(
|
|
ah, AH_PRIVATE(ah)->ah_caps.halTotalQueues - 1) != 0 ||
|
|
type == HAL_RESET_COLD))
|
|
{
|
|
u_int32_t time_out;
|
|
#define AR_SOC_RST_RESET 0xB806001C
|
|
#define AR_SOC_BOOT_STRAP 0xB80600AC
|
|
#define AR_SOC_WLAN_RST 0x00000800 /* WLAN reset */
|
|
#define REG_WRITE(_reg, _val) *((volatile u_int32_t *)(_reg)) = (_val);
|
|
#define REG_READ(_reg) *((volatile u_int32_t *)(_reg))
|
|
HALDEBUG(ah, HAL_DEBUG_RESET, "%s: Hornet SoC reset WMAC.\n", __func__);
|
|
|
|
REG_WRITE(AR_SOC_RST_RESET,
|
|
REG_READ(AR_SOC_RST_RESET) | AR_SOC_WLAN_RST);
|
|
REG_WRITE(AR_SOC_RST_RESET,
|
|
REG_READ(AR_SOC_RST_RESET) & (~AR_SOC_WLAN_RST));
|
|
|
|
time_out = 0;
|
|
|
|
while (1) {
|
|
tmp_reg = REG_READ(AR_SOC_BOOT_STRAP);
|
|
if ((tmp_reg & 0x10) == 0) {
|
|
break;
|
|
}
|
|
if (time_out > 20) {
|
|
break;
|
|
}
|
|
OS_DELAY(10000);
|
|
time_out++;
|
|
}
|
|
|
|
OS_REG_WRITE(ah, AR_RTC_RESET, 1);
|
|
#undef REG_READ
|
|
#undef REG_WRITE
|
|
#undef AR_SOC_WLAN_RST
|
|
#undef AR_SOC_RST_RESET
|
|
#undef AR_SOC_BOOT_STRAP
|
|
}
|
|
#endif /* AH_SUPPORT_HORNET */
|
|
|
|
#ifdef AH_SUPPORT_SCORPION
|
|
if (AR_SREV_SCORPION(ah)) {
|
|
#define DDR_CTL_CONFIG_ADDRESS 0xb8000000
|
|
#define DDR_CTL_CONFIG_OFFSET 0x0108
|
|
#define DDR_CTL_CONFIG_CLIENT_ACTIVITY_MSB 29
|
|
#define DDR_CTL_CONFIG_CLIENT_ACTIVITY_LSB 21
|
|
#define DDR_CTL_CONFIG_CLIENT_ACTIVITY_MASK 0x3fe00000
|
|
#define DDR_CTL_CONFIG_CLIENT_ACTIVITY_GET(x) (((x) & DDR_CTL_CONFIG_CLIENT_ACTIVITY_MASK) >> DDR_CTL_CONFIG_CLIENT_ACTIVITY_LSB)
|
|
#define DDR_CTL_CONFIG_CLIENT_ACTIVITY_SET(x) (((x) << DDR_CTL_CONFIG_CLIENT_ACTIVITY_LSB) & DDR_CTL_CONFIG_CLIENT_ACTIVITY_MASK)
|
|
#define MAC_DMA_CFG_ADDRESS 0xb8100000
|
|
#define MAC_DMA_CFG_OFFSET 0x0014
|
|
|
|
#define MAC_DMA_CFG_HALT_REQ_MSB 11
|
|
#define MAC_DMA_CFG_HALT_REQ_LSB 11
|
|
#define MAC_DMA_CFG_HALT_REQ_MASK 0x00000800
|
|
#define MAC_DMA_CFG_HALT_REQ_GET(x) (((x) & MAC_DMA_CFG_HALT_REQ_MASK) >> MAC_DMA_CFG_HALT_REQ_LSB)
|
|
#define MAC_DMA_CFG_HALT_REQ_SET(x) (((x) << MAC_DMA_CFG_HALT_REQ_LSB) & MAC_DMA_CFG_HALT_REQ_MASK)
|
|
#define MAC_DMA_CFG_HALT_ACK_MSB 12
|
|
#define MAC_DMA_CFG_HALT_ACK_LSB 12
|
|
#define MAC_DMA_CFG_HALT_ACK_MASK 0x00001000
|
|
#define MAC_DMA_CFG_HALT_ACK_GET(x) (((x) & MAC_DMA_CFG_HALT_ACK_MASK) >> MAC_DMA_CFG_HALT_ACK_LSB)
|
|
#define MAC_DMA_CFG_HALT_ACK_SET(x) (((x) << MAC_DMA_CFG_HALT_ACK_LSB) & MAC_DMA_CFG_HALT_ACK_MASK)
|
|
|
|
#define RST_RESET 0xB806001c
|
|
#define RTC_RESET (1<<27)
|
|
|
|
#define REG_READ(_reg) *((volatile u_int32_t *)(_reg))
|
|
#define REG_WRITE(_reg, _val) *((volatile u_int32_t *)(_reg)) = (_val);
|
|
|
|
#define DDR_REG_READ(_ah, _reg) \
|
|
*((volatile u_int32_t *)( DDR_CTL_CONFIG_ADDRESS + (_reg)))
|
|
#define DDR_REG_WRITE(_ah, _reg, _val) \
|
|
*((volatile u_int32_t *)(DDR_CTL_CONFIG_ADDRESS + (_reg))) = (_val)
|
|
|
|
OS_REG_WRITE(ah,MAC_DMA_CFG_OFFSET, (OS_REG_READ(ah,MAC_DMA_CFG_OFFSET) & ~MAC_DMA_CFG_HALT_REQ_MASK) |
|
|
MAC_DMA_CFG_HALT_REQ_SET(1));
|
|
|
|
{
|
|
int count;
|
|
u_int32_t data;
|
|
|
|
count = 0;
|
|
while (!MAC_DMA_CFG_HALT_ACK_GET(OS_REG_READ(ah, MAC_DMA_CFG_OFFSET) ))
|
|
{
|
|
count++;
|
|
if (count > 10) {
|
|
ath_hal_printf(ah, "Halt ACK timeout\n");
|
|
break;
|
|
}
|
|
OS_DELAY(10);
|
|
}
|
|
|
|
data = DDR_REG_READ(ah,DDR_CTL_CONFIG_OFFSET);
|
|
HALDEBUG(ah, HAL_DEBUG_RESET, "check DDR Activity - HIGH\n");
|
|
|
|
count = 0;
|
|
while (DDR_CTL_CONFIG_CLIENT_ACTIVITY_GET(data)) {
|
|
// AVE_DEBUG(0,"DDR Activity - HIGH\n");
|
|
HALDEBUG(ah, HAL_DEBUG_RESET, "DDR Activity - HIGH\n");
|
|
count++;
|
|
OS_DELAY(10);
|
|
data = DDR_REG_READ(ah,DDR_CTL_CONFIG_OFFSET);
|
|
if (count > 10) {
|
|
ath_hal_printf(ah, "DDR Activity timeout\n");
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
|
|
|
|
{
|
|
//Force RTC reset
|
|
REG_WRITE(RST_RESET, (REG_READ(RST_RESET) | RTC_RESET));
|
|
OS_DELAY(10);
|
|
REG_WRITE(RST_RESET, (REG_READ(RST_RESET) & ~RTC_RESET));
|
|
OS_DELAY(10);
|
|
OS_REG_WRITE(ah, AR_RTC_RESET, 0);
|
|
OS_DELAY(10);
|
|
OS_REG_WRITE(ah, AR_RTC_RESET, 1);
|
|
OS_DELAY(10);
|
|
HALDEBUG(ah, HAL_DEBUG_RESET, "%s: Scorpion SoC RTC reset done.\n", __func__);
|
|
}
|
|
#undef REG_READ
|
|
#undef REG_WRITE
|
|
}
|
|
#endif /* AH_SUPPORT_SCORPION */
|
|
|
|
/*
|
|
* Set Mac(BB,Phy) Warm Reset
|
|
*/
|
|
OS_REG_WRITE(ah, AR_RTC_RC, rst_flags);
|
|
|
|
OS_DELAY(50); /* XXX 50 usec */
|
|
|
|
/*
|
|
* Clear resets and force wakeup
|
|
*/
|
|
OS_REG_WRITE(ah, AR_RTC_RC, 0);
|
|
if (!ath_hal_wait(ah, AR_RTC_RC, AR_RTC_RC_M, 0)) {
|
|
HALDEBUG(ah, HAL_DEBUG_UNMASKABLE,
|
|
"%s: RTC stuck in MAC reset\n", __FUNCTION__);
|
|
HALDEBUG(ah, HAL_DEBUG_UNMASKABLE,
|
|
"%s: AR_RTC_RC = 0x%x\n", __func__, OS_REG_READ(ah, AR_RTC_RC));
|
|
return AH_FALSE;
|
|
}
|
|
|
|
/* Clear AHB reset */
|
|
OS_REG_WRITE(ah, AR_HOSTIF_REG(ah, AR_RC), 0);
|
|
ar9300_disable_pll_lock_detect(ah);
|
|
|
|
ar9300_attach_hw_platform(ah);
|
|
|
|
ahp->ah_chip_reset_done = 1;
|
|
return AH_TRUE;
|
|
}
|
|
|
|
static inline HAL_BOOL
|
|
ar9300_set_reset_power_on(struct ath_hal *ah)
|
|
{
|
|
/* Force wake */
|
|
OS_REG_WRITE(ah, AR_HOSTIF_REG(ah, AR_WA), AH9300(ah)->ah_wa_reg_val);
|
|
OS_DELAY(10); /* delay to allow AR_WA reg write to kick in */
|
|
OS_REG_WRITE(ah, AR_RTC_FORCE_WAKE,
|
|
AR_RTC_FORCE_WAKE_EN | AR_RTC_FORCE_WAKE_ON_INT);
|
|
/*
|
|
* RTC reset and clear. Some delay in between is needed
|
|
* to give the chip time to settle.
|
|
*/
|
|
OS_REG_WRITE(ah, AR_RTC_RESET, 0);
|
|
OS_DELAY(2);
|
|
OS_REG_WRITE(ah, AR_RTC_RESET, 1);
|
|
|
|
/*
|
|
* Poll till RTC is ON
|
|
*/
|
|
if (!ath_hal_wait(ah,
|
|
AR_RTC_STATUS, AR_RTC_STATUS_M,
|
|
AR_RTC_STATUS_ON))
|
|
{
|
|
HALDEBUG(ah, HAL_DEBUG_UNMASKABLE,
|
|
"%s: RTC not waking up for %d\n", __FUNCTION__, 1000);
|
|
return AH_FALSE;
|
|
}
|
|
|
|
/*
|
|
* Read Revisions from Chip right after RTC is on for the first time.
|
|
* This helps us detect the chip type early and initialize it accordingly.
|
|
*/
|
|
ar9300_read_revisions(ah);
|
|
|
|
/*
|
|
* Warm reset if we aren't really powering on,
|
|
* just restarting the driver.
|
|
*/
|
|
return ar9300_set_reset(ah, HAL_RESET_WARM);
|
|
}
|
|
|
|
/*
|
|
* Write the given reset bit mask into the reset register
|
|
*/
|
|
HAL_BOOL
|
|
ar9300_set_reset_reg(struct ath_hal *ah, u_int32_t type)
|
|
{
|
|
HAL_BOOL ret = AH_FALSE;
|
|
|
|
/*
|
|
* Set force wake
|
|
*/
|
|
OS_REG_WRITE(ah, AR_HOSTIF_REG(ah, AR_WA), AH9300(ah)->ah_wa_reg_val);
|
|
OS_DELAY(10); /* delay to allow AR_WA reg write to kick in */
|
|
OS_REG_WRITE(ah, AR_RTC_FORCE_WAKE,
|
|
AR_RTC_FORCE_WAKE_EN | AR_RTC_FORCE_WAKE_ON_INT);
|
|
|
|
switch (type) {
|
|
case HAL_RESET_POWER_ON:
|
|
ret = ar9300_set_reset_power_on(ah);
|
|
break;
|
|
case HAL_RESET_WARM:
|
|
case HAL_RESET_COLD:
|
|
ret = ar9300_set_reset(ah, type);
|
|
break;
|
|
default:
|
|
break;
|
|
}
|
|
|
|
#if ATH_SUPPORT_MCI
|
|
if (AH_PRIVATE(ah)->ah_caps.halMciSupport) {
|
|
OS_REG_WRITE(ah, AR_RTC_KEEP_AWAKE, 0x2);
|
|
}
|
|
#endif
|
|
|
|
return ret;
|
|
}
|
|
|
|
/*
|
|
* Places the PHY and Radio chips into reset. A full reset
|
|
* must be called to leave this state. The PCI/MAC/PCU are
|
|
* not placed into reset as we must receive interrupt to
|
|
* re-enable the hardware.
|
|
*/
|
|
HAL_BOOL
|
|
ar9300_phy_disable(struct ath_hal *ah)
|
|
{
|
|
if (!ar9300_set_reset_reg(ah, HAL_RESET_WARM)) {
|
|
return AH_FALSE;
|
|
}
|
|
|
|
#ifdef ATH_SUPPORT_LED
|
|
#define REG_READ(_reg) *((volatile u_int32_t *)(_reg))
|
|
#define REG_WRITE(_reg, _val) *((volatile u_int32_t *)(_reg)) = (_val);
|
|
#define ATH_GPIO_OE 0xB8040000
|
|
#define ATH_GPIO_OUT 0xB8040008 /* GPIO Ouput Value reg.*/
|
|
if (AR_SREV_WASP(ah)) {
|
|
if (IS_CHAN_2GHZ((AH_PRIVATE(ah)->ah_curchan))) {
|
|
REG_WRITE(ATH_GPIO_OE, (REG_READ(ATH_GPIO_OE) | (0x1 << 13)));
|
|
}
|
|
else {
|
|
REG_WRITE(ATH_GPIO_OE, (REG_READ(ATH_GPIO_OE) | (0x1 << 12)));
|
|
}
|
|
}
|
|
else if (AR_SREV_SCORPION(ah)) {
|
|
if (IS_CHAN_2GHZ((AH_PRIVATE(ah)->ah_curchan))) {
|
|
REG_WRITE(ATH_GPIO_OE, (REG_READ(ATH_GPIO_OE) | (0x1 << 13)));
|
|
}
|
|
else {
|
|
REG_WRITE(ATH_GPIO_OE, (REG_READ(ATH_GPIO_OE) | (0x1 << 12)));
|
|
}
|
|
/* Turn off JMPST led */
|
|
REG_WRITE(ATH_GPIO_OUT, (REG_READ(ATH_GPIO_OUT) | (0x1 << 15)));
|
|
}
|
|
else if (AR_SREV_HONEYBEE(ah)) {
|
|
REG_WRITE(ATH_GPIO_OE, (REG_READ(ATH_GPIO_OE) | (0x1 << 12)));
|
|
}
|
|
#undef REG_READ
|
|
#undef REG_WRITE
|
|
#endif
|
|
|
|
if ( AR_SREV_OSPREY(ah) ) {
|
|
OS_REG_RMW(ah, AR_HOSTIF_REG(ah, AR_GPIO_OUTPUT_MUX1), 0x0, 0x1f);
|
|
}
|
|
|
|
|
|
ar9300_init_pll(ah, AH_NULL);
|
|
ar9300_disable_pll_lock_detect(ah);
|
|
|
|
return AH_TRUE;
|
|
}
|
|
|
|
/*
|
|
* Places all of hardware into reset
|
|
*/
|
|
HAL_BOOL
|
|
ar9300_disable(struct ath_hal *ah)
|
|
{
|
|
if (!ar9300_set_power_mode(ah, HAL_PM_AWAKE, AH_TRUE)) {
|
|
return AH_FALSE;
|
|
}
|
|
if (!ar9300_set_reset_reg(ah, HAL_RESET_COLD)) {
|
|
return AH_FALSE;
|
|
}
|
|
|
|
ar9300_init_pll(ah, AH_NULL);
|
|
|
|
return AH_TRUE;
|
|
}
|
|
|
|
/*
|
|
* TODO: Only write the PLL if we're changing to or from CCK mode
|
|
*
|
|
* WARNING: The order of the PLL and mode registers must be correct.
|
|
*/
|
|
static inline void
|
|
ar9300_set_rf_mode(struct ath_hal *ah, struct ieee80211_channel *chan)
|
|
{
|
|
u_int32_t rf_mode = 0;
|
|
|
|
if (chan == AH_NULL) {
|
|
return;
|
|
}
|
|
switch (AH9300(ah)->ah_hwp) {
|
|
case HAL_TRUE_CHIP:
|
|
rf_mode |= (IEEE80211_IS_CHAN_B(chan) || IEEE80211_IS_CHAN_G(chan)) ?
|
|
AR_PHY_MODE_DYNAMIC : AR_PHY_MODE_OFDM;
|
|
break;
|
|
default:
|
|
HALASSERT(0);
|
|
break;
|
|
}
|
|
/* Phy mode bits for 5GHz channels requiring Fast Clock */
|
|
if ( IS_5GHZ_FAST_CLOCK_EN(ah, chan)) {
|
|
rf_mode |= (AR_PHY_MODE_DYNAMIC | AR_PHY_MODE_DYN_CCK_DISABLE);
|
|
}
|
|
OS_REG_WRITE(ah, AR_PHY_MODE, rf_mode);
|
|
}
|
|
|
|
/*
|
|
* Places the hardware into reset and then pulls it out of reset
|
|
*/
|
|
HAL_BOOL
|
|
ar9300_chip_reset(struct ath_hal *ah, struct ieee80211_channel *chan)
|
|
{
|
|
struct ath_hal_9300 *ahp = AH9300(ah);
|
|
int type = HAL_RESET_WARM;
|
|
|
|
OS_MARK(ah, AH_MARK_CHIPRESET, chan ? chan->ic_freq : 0);
|
|
|
|
/*
|
|
* Warm reset is optimistic.
|
|
*
|
|
* If the TX/RX DMA engines aren't shut down (eg, they're
|
|
* wedged) then we're better off doing a full cold reset
|
|
* to try and shake that condition.
|
|
*/
|
|
if (ahp->ah_chip_full_sleep ||
|
|
(ah->ah_config.ah_force_full_reset == 1) ||
|
|
OS_REG_READ(ah, AR_Q_TXE) ||
|
|
(OS_REG_READ(ah, AR_CR) & AR_CR_RXE)) {
|
|
type = HAL_RESET_COLD;
|
|
}
|
|
|
|
if (!ar9300_set_reset_reg(ah, type)) {
|
|
return AH_FALSE;
|
|
}
|
|
|
|
/* Bring out of sleep mode (AGAIN) */
|
|
if (!ar9300_set_power_mode(ah, HAL_PM_AWAKE, AH_TRUE)) {
|
|
return AH_FALSE;
|
|
}
|
|
|
|
ahp->ah_chip_full_sleep = AH_FALSE;
|
|
|
|
if (AR_SREV_HORNET(ah)) {
|
|
ar9300_internal_regulator_apply(ah);
|
|
}
|
|
|
|
ar9300_init_pll(ah, chan);
|
|
|
|
/*
|
|
* Perform warm reset before the mode/PLL/turbo registers
|
|
* are changed in order to deactivate the radio. Mode changes
|
|
* with an active radio can result in corrupted shifts to the
|
|
* radio device.
|
|
*/
|
|
ar9300_set_rf_mode(ah, chan);
|
|
|
|
return AH_TRUE;
|
|
}
|
|
|
|
/* ar9300_setup_calibration
|
|
* Setup HW to collect samples used for current cal
|
|
*/
|
|
inline static void
|
|
ar9300_setup_calibration(struct ath_hal *ah, HAL_CAL_LIST *curr_cal)
|
|
{
|
|
/* Select calibration to run */
|
|
switch (curr_cal->cal_data->cal_type) {
|
|
case IQ_MISMATCH_CAL:
|
|
/* Start calibration w/ 2^(INIT_IQCAL_LOG_COUNT_MAX+1) samples */
|
|
OS_REG_RMW_FIELD(ah, AR_PHY_TIMING4,
|
|
AR_PHY_TIMING4_IQCAL_LOG_COUNT_MAX,
|
|
curr_cal->cal_data->cal_count_max);
|
|
OS_REG_WRITE(ah, AR_PHY_CALMODE, AR_PHY_CALMODE_IQ);
|
|
|
|
HALDEBUG(ah, HAL_DEBUG_CALIBRATE,
|
|
"%s: starting IQ Mismatch Calibration\n", __func__);
|
|
|
|
/* Kick-off cal */
|
|
OS_REG_SET_BIT(ah, AR_PHY_TIMING4, AR_PHY_TIMING4_DO_CAL);
|
|
|
|
break;
|
|
case TEMP_COMP_CAL:
|
|
if (AR_SREV_HORNET(ah) || AR_SREV_POSEIDON(ah) ||
|
|
AR_SREV_WASP(ah) || AR_SREV_SCORPION(ah)) {
|
|
OS_REG_RMW_FIELD(ah,
|
|
AR_HORNET_CH0_THERM, AR_PHY_65NM_CH0_THERM_LOCAL, 1);
|
|
OS_REG_RMW_FIELD(ah,
|
|
AR_HORNET_CH0_THERM, AR_PHY_65NM_CH0_THERM_START, 1);
|
|
} else if (AR_SREV_JUPITER(ah) || AR_SREV_APHRODITE(ah)) {
|
|
OS_REG_RMW_FIELD(ah,
|
|
AR_PHY_65NM_CH0_THERM_JUPITER, AR_PHY_65NM_CH0_THERM_LOCAL, 1);
|
|
OS_REG_RMW_FIELD(ah,
|
|
AR_PHY_65NM_CH0_THERM_JUPITER, AR_PHY_65NM_CH0_THERM_START, 1);
|
|
} else {
|
|
OS_REG_RMW_FIELD(ah,
|
|
AR_PHY_65NM_CH0_THERM, AR_PHY_65NM_CH0_THERM_LOCAL, 1);
|
|
OS_REG_RMW_FIELD(ah,
|
|
AR_PHY_65NM_CH0_THERM, AR_PHY_65NM_CH0_THERM_START, 1);
|
|
}
|
|
|
|
HALDEBUG(ah, HAL_DEBUG_CALIBRATE,
|
|
"%s: starting Temperature Compensation Calibration\n", __func__);
|
|
break;
|
|
default:
|
|
HALDEBUG(ah, HAL_DEBUG_UNMASKABLE,
|
|
"%s called with incorrect calibration type.\n", __func__);
|
|
}
|
|
}
|
|
|
|
/* ar9300_reset_calibration
|
|
* Initialize shared data structures and prepare a cal to be run.
|
|
*/
|
|
inline static void
|
|
ar9300_reset_calibration(struct ath_hal *ah, HAL_CAL_LIST *curr_cal)
|
|
{
|
|
struct ath_hal_9300 *ahp = AH9300(ah);
|
|
int i;
|
|
|
|
/* Setup HW for new calibration */
|
|
ar9300_setup_calibration(ah, curr_cal);
|
|
|
|
/* Change SW state to RUNNING for this calibration */
|
|
curr_cal->cal_state = CAL_RUNNING;
|
|
|
|
/* Reset data structures shared between different calibrations */
|
|
for (i = 0; i < AR9300_MAX_CHAINS; i++) {
|
|
ahp->ah_meas0.sign[i] = 0;
|
|
ahp->ah_meas1.sign[i] = 0;
|
|
ahp->ah_meas2.sign[i] = 0;
|
|
ahp->ah_meas3.sign[i] = 0;
|
|
}
|
|
|
|
ahp->ah_cal_samples = 0;
|
|
}
|
|
|
|
#ifdef XXX_UNUSED_FUNCTION
|
|
/*
|
|
* Find out which of the RX chains are enabled
|
|
*/
|
|
static u_int32_t
|
|
ar9300_get_rx_chain_mask(struct ath_hal *ah)
|
|
{
|
|
u_int32_t ret_val = OS_REG_READ(ah, AR_PHY_RX_CHAINMASK);
|
|
/* The bits [2:0] indicate the rx chain mask and are to be
|
|
* interpreted as follows:
|
|
* 00x => Only chain 0 is enabled
|
|
* 01x => Chain 1 and 0 enabled
|
|
* 1xx => Chain 2,1 and 0 enabled
|
|
*/
|
|
return (ret_val & 0x7);
|
|
}
|
|
#endif
|
|
|
|
static void
|
|
ar9300_get_nf_hist_base(struct ath_hal *ah, HAL_CHANNEL_INTERNAL *chan,
|
|
int is_scan, int16_t nf[])
|
|
{
|
|
HAL_NFCAL_BASE *h_base;
|
|
|
|
#ifdef ATH_NF_PER_CHAN
|
|
h_base = &chan->nf_cal_hist.base;
|
|
#else
|
|
if (is_scan) {
|
|
/*
|
|
* The channel we are currently on is not the home channel,
|
|
* so we shouldn't use the home channel NF buffer's values on
|
|
* this channel. Instead, use the NF single value already
|
|
* read for this channel. (Or, if we haven't read the NF for
|
|
* this channel yet, the SW default for this chip/band will
|
|
* be used.)
|
|
*/
|
|
h_base = &chan->nf_cal_hist.base;
|
|
} else {
|
|
/* use the home channel NF info */
|
|
h_base = &AH_PRIVATE(ah)->nf_cal_hist.base;
|
|
}
|
|
#endif
|
|
OS_MEMCPY(nf, h_base->priv_nf, sizeof(h_base->priv_nf));
|
|
}
|
|
|
|
HAL_BOOL
|
|
ar9300_load_nf(struct ath_hal *ah, int16_t nf[])
|
|
{
|
|
int i, j;
|
|
int32_t val;
|
|
/* XXX where are EXT regs defined */
|
|
const u_int32_t ar9300_cca_regs[] = {
|
|
AR_PHY_CCA_0,
|
|
AR_PHY_CCA_1,
|
|
AR_PHY_CCA_2,
|
|
AR_PHY_EXT_CCA,
|
|
AR_PHY_EXT_CCA_1,
|
|
AR_PHY_EXT_CCA_2,
|
|
};
|
|
u_int8_t chainmask;
|
|
|
|
/*
|
|
* Force NF calibration for all chains, otherwise Vista station
|
|
* would conduct a bad performance
|
|
*/
|
|
if (AR_SREV_HORNET(ah) || AR_SREV_POSEIDON(ah) || AR_SREV_APHRODITE(ah)) {
|
|
chainmask = 0x9;
|
|
} else if (AR_SREV_WASP(ah) || AR_SREV_JUPITER(ah) || AR_SREV_HONEYBEE(ah)) {
|
|
chainmask = 0x1b;
|
|
} else {
|
|
chainmask = 0x3F;
|
|
}
|
|
|
|
/*
|
|
* Write filtered NF values into max_cca_pwr register parameter
|
|
* so we can load below.
|
|
*/
|
|
for (i = 0; i < HAL_NUM_NF_READINGS; i++) {
|
|
if (chainmask & (1 << i)) {
|
|
val = OS_REG_READ(ah, ar9300_cca_regs[i]);
|
|
val &= 0xFFFFFE00;
|
|
val |= (((u_int32_t)(nf[i]) << 1) & 0x1ff);
|
|
OS_REG_WRITE(ah, ar9300_cca_regs[i], val);
|
|
}
|
|
}
|
|
|
|
HALDEBUG(ah, HAL_DEBUG_NFCAL, "%s: load %d %d %d %d %d %d\n",
|
|
__func__,
|
|
nf[0], nf[1], nf[2],
|
|
nf[3], nf[4], nf[5]);
|
|
|
|
/*
|
|
* Load software filtered NF value into baseband internal min_cca_pwr
|
|
* variable.
|
|
*/
|
|
OS_REG_CLR_BIT(ah, AR_PHY_AGC_CONTROL, AR_PHY_AGC_CONTROL_ENABLE_NF);
|
|
OS_REG_CLR_BIT(ah, AR_PHY_AGC_CONTROL, AR_PHY_AGC_CONTROL_NO_UPDATE_NF);
|
|
OS_REG_SET_BIT(ah, AR_PHY_AGC_CONTROL, AR_PHY_AGC_CONTROL_NF);
|
|
|
|
/* Wait for load to complete, should be fast, a few 10s of us. */
|
|
/* Changed the max delay 250us back to 10000us, since 250us often
|
|
* results in NF load timeout and causes deaf condition
|
|
* during stress testing 12/12/2009
|
|
*/
|
|
for (j = 0; j < 10000; j++) {
|
|
if ((OS_REG_READ(ah, AR_PHY_AGC_CONTROL) & AR_PHY_AGC_CONTROL_NF) == 0){
|
|
break;
|
|
}
|
|
OS_DELAY(10);
|
|
}
|
|
if (j == 10000) {
|
|
/*
|
|
* We timed out waiting for the noisefloor to load, probably
|
|
* due to an in-progress rx. Simply return here and allow
|
|
* the load plenty of time to complete before the next
|
|
* calibration interval. We need to avoid trying to load -50
|
|
* (which happens below) while the previous load is still in
|
|
* progress as this can cause rx deafness (see EV 66368,62830).
|
|
* Instead by returning here, the baseband nf cal will
|
|
* just be capped by our present noisefloor until the next
|
|
* calibration timer.
|
|
*/
|
|
HALDEBUG(AH_NULL, HAL_DEBUG_UNMASKABLE,
|
|
"%s: *** TIMEOUT while waiting for nf to load: "
|
|
"AR_PHY_AGC_CONTROL=0x%x ***\n",
|
|
__func__, OS_REG_READ(ah, AR_PHY_AGC_CONTROL));
|
|
return AH_FALSE;
|
|
}
|
|
|
|
/*
|
|
* Restore max_cca_power register parameter again so that we're not capped
|
|
* by the median we just loaded. This will be initial (and max) value
|
|
* of next noise floor calibration the baseband does.
|
|
*/
|
|
for (i = 0; i < HAL_NUM_NF_READINGS; i++) {
|
|
if (chainmask & (1 << i)) {
|
|
val = OS_REG_READ(ah, ar9300_cca_regs[i]);
|
|
val &= 0xFFFFFE00;
|
|
val |= (((u_int32_t)(-50) << 1) & 0x1ff);
|
|
OS_REG_WRITE(ah, ar9300_cca_regs[i], val);
|
|
}
|
|
}
|
|
return AH_TRUE;
|
|
}
|
|
|
|
/* ar9300_per_calibration
|
|
* Generic calibration routine.
|
|
* Recalibrate the lower PHY chips to account for temperature/environment
|
|
* changes.
|
|
*/
|
|
inline static void
|
|
ar9300_per_calibration(struct ath_hal *ah, HAL_CHANNEL_INTERNAL *ichan,
|
|
u_int8_t rxchainmask, HAL_CAL_LIST *curr_cal, HAL_BOOL *is_cal_done)
|
|
{
|
|
struct ath_hal_9300 *ahp = AH9300(ah);
|
|
|
|
/* Cal is assumed not done until explicitly set below */
|
|
*is_cal_done = AH_FALSE;
|
|
|
|
/* Calibration in progress. */
|
|
if (curr_cal->cal_state == CAL_RUNNING) {
|
|
/* Check to see if it has finished. */
|
|
if (!(OS_REG_READ(ah, AR_PHY_TIMING4) & AR_PHY_TIMING4_DO_CAL)) {
|
|
int i, num_chains = 0;
|
|
for (i = 0; i < AR9300_MAX_CHAINS; i++) {
|
|
if (rxchainmask & (1 << i)) {
|
|
num_chains++;
|
|
}
|
|
}
|
|
|
|
/*
|
|
* Accumulate cal measures for active chains
|
|
*/
|
|
curr_cal->cal_data->cal_collect(ah, num_chains);
|
|
|
|
ahp->ah_cal_samples++;
|
|
|
|
if (ahp->ah_cal_samples >= curr_cal->cal_data->cal_num_samples) {
|
|
/*
|
|
* Process accumulated data
|
|
*/
|
|
curr_cal->cal_data->cal_post_proc(ah, num_chains);
|
|
|
|
/* Calibration has finished. */
|
|
ichan->calValid |= curr_cal->cal_data->cal_type;
|
|
curr_cal->cal_state = CAL_DONE;
|
|
*is_cal_done = AH_TRUE;
|
|
} else {
|
|
/* Set-up collection of another sub-sample until we
|
|
* get desired number
|
|
*/
|
|
ar9300_setup_calibration(ah, curr_cal);
|
|
}
|
|
}
|
|
} else if (!(ichan->calValid & curr_cal->cal_data->cal_type)) {
|
|
/* If current cal is marked invalid in channel, kick it off */
|
|
ar9300_reset_calibration(ah, curr_cal);
|
|
}
|
|
}
|
|
|
|
static void
|
|
ar9300_start_nf_cal(struct ath_hal *ah)
|
|
{
|
|
struct ath_hal_9300 *ahp = AH9300(ah);
|
|
OS_REG_SET_BIT(ah, AR_PHY_AGC_CONTROL, AR_PHY_AGC_CONTROL_ENABLE_NF);
|
|
OS_REG_SET_BIT(ah, AR_PHY_AGC_CONTROL, AR_PHY_AGC_CONTROL_NO_UPDATE_NF);
|
|
OS_REG_SET_BIT(ah, AR_PHY_AGC_CONTROL, AR_PHY_AGC_CONTROL_NF);
|
|
AH9300(ah)->nf_tsf32 = ar9300_get_tsf32(ah);
|
|
|
|
/*
|
|
* We are reading the NF values before we start the NF operation, because
|
|
* of that we are getting very high values like -45.
|
|
* This triggers the CW_INT detected and EACS module triggers the channel change
|
|
* chip_reset_done value is used to fix this issue.
|
|
* chip_reset_flag is set during the RTC reset.
|
|
* chip_reset_flag is cleared during the starting NF operation.
|
|
* if flag is set we will clear the flag and will not read the NF values.
|
|
*/
|
|
ahp->ah_chip_reset_done = 0;
|
|
}
|
|
|
|
/* ar9300_calibration
|
|
* Wrapper for a more generic Calibration routine. Primarily to abstract to
|
|
* upper layers whether there is 1 or more calibrations to be run.
|
|
*/
|
|
HAL_BOOL
|
|
ar9300_calibration(struct ath_hal *ah, struct ieee80211_channel *chan, u_int8_t rxchainmask,
|
|
HAL_BOOL do_nf_cal, HAL_BOOL *is_cal_done, int is_scan,
|
|
u_int32_t *sched_cals)
|
|
{
|
|
struct ath_hal_9300 *ahp = AH9300(ah);
|
|
HAL_CAL_LIST *curr_cal = ahp->ah_cal_list_curr;
|
|
HAL_CHANNEL_INTERNAL *ichan = ath_hal_checkchannel(ah, chan);
|
|
int16_t nf_buf[HAL_NUM_NF_READINGS];
|
|
|
|
*is_cal_done = AH_TRUE;
|
|
|
|
|
|
/* XXX: For initial wasp bringup - disable periodic calibration */
|
|
/* Invalid channel check */
|
|
if (ichan == AH_NULL) {
|
|
HALDEBUG(ah, HAL_DEBUG_CHANNEL,
|
|
"%s: invalid channel %u/0x%x; no mapping\n",
|
|
__func__, chan->ic_freq, chan->ic_flags);
|
|
return AH_FALSE;
|
|
}
|
|
|
|
HALDEBUG(ah, HAL_DEBUG_CALIBRATE,
|
|
"%s: Entering, Doing NF Cal = %d\n", __func__, do_nf_cal);
|
|
HALDEBUG(ah, HAL_DEBUG_CALIBRATE, "%s: Chain 0 Rx IQ Cal Correction 0x%08x\n",
|
|
__func__, OS_REG_READ(ah, AR_PHY_RX_IQCAL_CORR_B0));
|
|
if (!AR_SREV_HORNET(ah) && !AR_SREV_POSEIDON(ah) && !AR_SREV_APHRODITE(ah)) {
|
|
HALDEBUG(ah, HAL_DEBUG_CALIBRATE,
|
|
"%s: Chain 1 Rx IQ Cal Correction 0x%08x\n",
|
|
__func__, OS_REG_READ(ah, AR_PHY_RX_IQCAL_CORR_B1));
|
|
if (!AR_SREV_WASP(ah) && !AR_SREV_JUPITER(ah) && !AR_SREV_HONEYBEE(ah)) {
|
|
HALDEBUG(ah, HAL_DEBUG_CALIBRATE,
|
|
"%s: Chain 2 Rx IQ Cal Correction 0x%08x\n",
|
|
__func__, OS_REG_READ(ah, AR_PHY_RX_IQCAL_CORR_B2));
|
|
}
|
|
}
|
|
|
|
OS_MARK(ah, AH_MARK_PERCAL, chan->ic_freq);
|
|
|
|
/* For given calibration:
|
|
* 1. Call generic cal routine
|
|
* 2. When this cal is done (is_cal_done) if we have more cals waiting
|
|
* (eg after reset), mask this to upper layers by not propagating
|
|
* is_cal_done if it is set to TRUE.
|
|
* Instead, change is_cal_done to FALSE and setup the waiting cal(s)
|
|
* to be run.
|
|
*/
|
|
if (curr_cal && (curr_cal->cal_data->cal_type & *sched_cals) &&
|
|
(curr_cal->cal_state == CAL_RUNNING ||
|
|
curr_cal->cal_state == CAL_WAITING))
|
|
{
|
|
ar9300_per_calibration(ah, ichan, rxchainmask, curr_cal, is_cal_done);
|
|
|
|
if (*is_cal_done == AH_TRUE) {
|
|
ahp->ah_cal_list_curr = curr_cal = curr_cal->cal_next;
|
|
|
|
if (curr_cal && curr_cal->cal_state == CAL_WAITING) {
|
|
*is_cal_done = AH_FALSE;
|
|
ar9300_reset_calibration(ah, curr_cal);
|
|
} else {
|
|
*sched_cals &= ~IQ_MISMATCH_CAL;
|
|
}
|
|
}
|
|
}
|
|
|
|
/* Do NF cal only at longer intervals */
|
|
if (do_nf_cal) {
|
|
int nf_done;
|
|
|
|
/* Get the value from the previous NF cal and update history buffer */
|
|
nf_done = ar9300_store_new_nf(ah, chan, is_scan);
|
|
#if 0
|
|
if (ichan->channel_flags & CHANNEL_CW_INT) {
|
|
chan->channel_flags |= CHANNEL_CW_INT;
|
|
}
|
|
#endif
|
|
chan->ic_state &= ~IEEE80211_CHANSTATE_CWINT;
|
|
|
|
if (nf_done) {
|
|
/*
|
|
* Load the NF from history buffer of the current channel.
|
|
* NF is slow time-variant, so it is OK to use a historical value.
|
|
*/
|
|
ar9300_get_nf_hist_base(ah, ichan, is_scan, nf_buf);
|
|
ar9300_load_nf(ah, nf_buf);
|
|
|
|
/* start NF calibration, without updating BB NF register*/
|
|
ar9300_start_nf_cal(ah);
|
|
}
|
|
}
|
|
return AH_TRUE;
|
|
}
|
|
|
|
/* ar9300_iq_cal_collect
|
|
* Collect data from HW to later perform IQ Mismatch Calibration
|
|
*/
|
|
void
|
|
ar9300_iq_cal_collect(struct ath_hal *ah, u_int8_t num_chains)
|
|
{
|
|
struct ath_hal_9300 *ahp = AH9300(ah);
|
|
int i;
|
|
|
|
/*
|
|
* Accumulate IQ cal measures for active chains
|
|
*/
|
|
for (i = 0; i < num_chains; i++) {
|
|
ahp->ah_total_power_meas_i[i] = OS_REG_READ(ah, AR_PHY_CAL_MEAS_0(i));
|
|
ahp->ah_total_power_meas_q[i] = OS_REG_READ(ah, AR_PHY_CAL_MEAS_1(i));
|
|
ahp->ah_total_iq_corr_meas[i] =
|
|
(int32_t) OS_REG_READ(ah, AR_PHY_CAL_MEAS_2(i));
|
|
HALDEBUG(ah, HAL_DEBUG_CALIBRATE,
|
|
"%d: Chn %d "
|
|
"Reg Offset(0x%04x)pmi=0x%08x; "
|
|
"Reg Offset(0x%04x)pmq=0x%08x; "
|
|
"Reg Offset (0x%04x)iqcm=0x%08x;\n",
|
|
ahp->ah_cal_samples,
|
|
i,
|
|
(unsigned) AR_PHY_CAL_MEAS_0(i),
|
|
ahp->ah_total_power_meas_i[i],
|
|
(unsigned) AR_PHY_CAL_MEAS_1(i),
|
|
ahp->ah_total_power_meas_q[i],
|
|
(unsigned) AR_PHY_CAL_MEAS_2(i),
|
|
ahp->ah_total_iq_corr_meas[i]);
|
|
}
|
|
}
|
|
|
|
/* ar9300_iq_calibration
|
|
* Use HW data to perform IQ Mismatch Calibration
|
|
*/
|
|
void
|
|
ar9300_iq_calibration(struct ath_hal *ah, u_int8_t num_chains)
|
|
{
|
|
struct ath_hal_9300 *ahp = AH9300(ah);
|
|
u_int32_t power_meas_q, power_meas_i, iq_corr_meas;
|
|
u_int32_t q_coff_denom, i_coff_denom;
|
|
int32_t q_coff, i_coff;
|
|
int iq_corr_neg, i;
|
|
HAL_CHANNEL_INTERNAL *ichan;
|
|
static const u_int32_t offset_array[3] = {
|
|
AR_PHY_RX_IQCAL_CORR_B0,
|
|
AR_PHY_RX_IQCAL_CORR_B1,
|
|
AR_PHY_RX_IQCAL_CORR_B2,
|
|
};
|
|
|
|
ichan = ath_hal_checkchannel(ah, AH_PRIVATE(ah)->ah_curchan);
|
|
|
|
for (i = 0; i < num_chains; i++) {
|
|
power_meas_i = ahp->ah_total_power_meas_i[i];
|
|
power_meas_q = ahp->ah_total_power_meas_q[i];
|
|
iq_corr_meas = ahp->ah_total_iq_corr_meas[i];
|
|
|
|
HALDEBUG(ah, HAL_DEBUG_CALIBRATE,
|
|
"Starting IQ Cal and Correction for Chain %d\n", i);
|
|
HALDEBUG(ah, HAL_DEBUG_CALIBRATE,
|
|
"Orignal: Chn %diq_corr_meas = 0x%08x\n",
|
|
i, ahp->ah_total_iq_corr_meas[i]);
|
|
|
|
iq_corr_neg = 0;
|
|
|
|
/* iq_corr_meas is always negative. */
|
|
if (iq_corr_meas > 0x80000000) {
|
|
iq_corr_meas = (0xffffffff - iq_corr_meas) + 1;
|
|
iq_corr_neg = 1;
|
|
}
|
|
|
|
HALDEBUG(ah, HAL_DEBUG_CALIBRATE,
|
|
"Chn %d pwr_meas_i = 0x%08x\n", i, power_meas_i);
|
|
HALDEBUG(ah, HAL_DEBUG_CALIBRATE,
|
|
"Chn %d pwr_meas_q = 0x%08x\n", i, power_meas_q);
|
|
HALDEBUG(ah, HAL_DEBUG_CALIBRATE,
|
|
"iq_corr_neg is 0x%08x\n", iq_corr_neg);
|
|
|
|
i_coff_denom = (power_meas_i / 2 + power_meas_q / 2) / 256;
|
|
q_coff_denom = power_meas_q / 64;
|
|
|
|
/* Protect against divide-by-0 */
|
|
if ((i_coff_denom != 0) && (q_coff_denom != 0)) {
|
|
/* IQ corr_meas is already negated if iqcorr_neg == 1 */
|
|
i_coff = iq_corr_meas / i_coff_denom;
|
|
q_coff = power_meas_i / q_coff_denom - 64;
|
|
HALDEBUG(ah, HAL_DEBUG_CALIBRATE,
|
|
"Chn %d i_coff = 0x%08x\n", i, i_coff);
|
|
HALDEBUG(ah, HAL_DEBUG_CALIBRATE,
|
|
"Chn %d q_coff = 0x%08x\n", i, q_coff);
|
|
|
|
/* Force bounds on i_coff */
|
|
if (i_coff >= 63) {
|
|
i_coff = 63;
|
|
} else if (i_coff <= -63) {
|
|
i_coff = -63;
|
|
}
|
|
|
|
/* Negate i_coff if iq_corr_neg == 0 */
|
|
if (iq_corr_neg == 0x0) {
|
|
i_coff = -i_coff;
|
|
}
|
|
|
|
/* Force bounds on q_coff */
|
|
if (q_coff >= 63) {
|
|
q_coff = 63;
|
|
} else if (q_coff <= -63) {
|
|
q_coff = -63;
|
|
}
|
|
|
|
i_coff = i_coff & 0x7f;
|
|
q_coff = q_coff & 0x7f;
|
|
|
|
HALDEBUG(ah, HAL_DEBUG_CALIBRATE,
|
|
"Chn %d : i_coff = 0x%x q_coff = 0x%x\n", i, i_coff, q_coff);
|
|
HALDEBUG(ah, HAL_DEBUG_CALIBRATE,
|
|
"Register offset (0x%04x) before update = 0x%x\n",
|
|
offset_array[i], OS_REG_READ(ah, offset_array[i]));
|
|
|
|
OS_REG_RMW_FIELD(ah, offset_array[i],
|
|
AR_PHY_RX_IQCAL_CORR_IQCORR_Q_I_COFF, i_coff);
|
|
OS_REG_RMW_FIELD(ah, offset_array[i],
|
|
AR_PHY_RX_IQCAL_CORR_IQCORR_Q_Q_COFF, q_coff);
|
|
|
|
/* store the RX cal results */
|
|
if (ichan != NULL) {
|
|
ahp->ah_rx_cal_corr[i] = OS_REG_READ(ah, offset_array[i]) & 0x7fff;
|
|
ahp->ah_rx_cal_complete = AH_TRUE;
|
|
ahp->ah_rx_cal_chan = ichan->channel;
|
|
// ahp->ah_rx_cal_chan_flag = ichan->channel_flags &~ CHANNEL_PASSIVE;
|
|
ahp->ah_rx_cal_chan_flag = 0; /* XXX */
|
|
} else {
|
|
/* XXX? Is this what I should do? */
|
|
ahp->ah_rx_cal_complete = AH_FALSE;
|
|
|
|
}
|
|
|
|
HALDEBUG(ah, HAL_DEBUG_CALIBRATE,
|
|
"Register offset (0x%04x) QI COFF (bitfields 0x%08x) "
|
|
"after update = 0x%x\n",
|
|
offset_array[i], AR_PHY_RX_IQCAL_CORR_IQCORR_Q_I_COFF,
|
|
OS_REG_READ(ah, offset_array[i]));
|
|
HALDEBUG(ah, HAL_DEBUG_CALIBRATE,
|
|
"Register offset (0x%04x) QQ COFF (bitfields 0x%08x) "
|
|
"after update = 0x%x\n",
|
|
offset_array[i], AR_PHY_RX_IQCAL_CORR_IQCORR_Q_Q_COFF,
|
|
OS_REG_READ(ah, offset_array[i]));
|
|
HALDEBUG(ah, HAL_DEBUG_CALIBRATE,
|
|
"IQ Cal and Correction done for Chain %d\n", i);
|
|
}
|
|
}
|
|
|
|
OS_REG_SET_BIT(ah,
|
|
AR_PHY_RX_IQCAL_CORR_B0, AR_PHY_RX_IQCAL_CORR_IQCORR_ENABLE);
|
|
HALDEBUG(ah, HAL_DEBUG_CALIBRATE,
|
|
"IQ Cal and Correction (offset 0x%04x) enabled "
|
|
"(bit position 0x%08x). New Value 0x%08x\n",
|
|
(unsigned) (AR_PHY_RX_IQCAL_CORR_B0),
|
|
AR_PHY_RX_IQCAL_CORR_IQCORR_ENABLE,
|
|
OS_REG_READ(ah, AR_PHY_RX_IQCAL_CORR_B0));
|
|
}
|
|
|
|
/*
|
|
* When coming back from offchan, we do not perform RX IQ Cal.
|
|
* But the chip reset will clear all previous results
|
|
* We store the previous results and restore here.
|
|
*/
|
|
static void
|
|
ar9300_rx_iq_cal_restore(struct ath_hal *ah)
|
|
{
|
|
struct ath_hal_9300 *ahp = AH9300(ah);
|
|
u_int32_t i_coff, q_coff;
|
|
HAL_BOOL is_restore = AH_FALSE;
|
|
int i;
|
|
static const u_int32_t offset_array[3] = {
|
|
AR_PHY_RX_IQCAL_CORR_B0,
|
|
AR_PHY_RX_IQCAL_CORR_B1,
|
|
AR_PHY_RX_IQCAL_CORR_B2,
|
|
};
|
|
|
|
for (i=0; i<AR9300_MAX_CHAINS; i++) {
|
|
if (ahp->ah_rx_cal_corr[i]) {
|
|
i_coff = (ahp->ah_rx_cal_corr[i] &
|
|
AR_PHY_RX_IQCAL_CORR_IQCORR_Q_I_COFF) >>
|
|
AR_PHY_RX_IQCAL_CORR_IQCORR_Q_I_COFF_S;
|
|
q_coff = (ahp->ah_rx_cal_corr[i] &
|
|
AR_PHY_RX_IQCAL_CORR_IQCORR_Q_Q_COFF) >>
|
|
AR_PHY_RX_IQCAL_CORR_IQCORR_Q_Q_COFF_S;
|
|
|
|
OS_REG_RMW_FIELD(ah, offset_array[i],
|
|
AR_PHY_RX_IQCAL_CORR_IQCORR_Q_I_COFF, i_coff);
|
|
OS_REG_RMW_FIELD(ah, offset_array[i],
|
|
AR_PHY_RX_IQCAL_CORR_IQCORR_Q_Q_COFF, q_coff);
|
|
|
|
is_restore = AH_TRUE;
|
|
}
|
|
}
|
|
|
|
if (is_restore)
|
|
OS_REG_SET_BIT(ah,
|
|
AR_PHY_RX_IQCAL_CORR_B0, AR_PHY_RX_IQCAL_CORR_IQCORR_ENABLE);
|
|
|
|
HALDEBUG(ah, HAL_DEBUG_CALIBRATE,
|
|
"%s: IQ Cal and Correction (offset 0x%04x) enabled "
|
|
"(bit position 0x%08x). New Value 0x%08x\n",
|
|
__func__,
|
|
(unsigned) (AR_PHY_RX_IQCAL_CORR_B0),
|
|
AR_PHY_RX_IQCAL_CORR_IQCORR_ENABLE,
|
|
OS_REG_READ(ah, AR_PHY_RX_IQCAL_CORR_B0));
|
|
}
|
|
|
|
/*
|
|
* Set a limit on the overall output power. Used for dynamic
|
|
* transmit power control and the like.
|
|
*
|
|
* NB: limit is in units of 0.5 dbM.
|
|
*/
|
|
HAL_BOOL
|
|
ar9300_set_tx_power_limit(struct ath_hal *ah, u_int32_t limit,
|
|
u_int16_t extra_txpow, u_int16_t tpc_in_db)
|
|
{
|
|
struct ath_hal_9300 *ahp = AH9300(ah);
|
|
struct ath_hal_private *ahpriv = AH_PRIVATE(ah);
|
|
const struct ieee80211_channel *chan = ahpriv->ah_curchan;
|
|
HAL_CHANNEL_INTERNAL *ichan = ath_hal_checkchannel(ah, chan);
|
|
|
|
if (NULL == chan) {
|
|
return AH_FALSE;
|
|
}
|
|
|
|
ahpriv->ah_powerLimit = AH_MIN(limit, MAX_RATE_POWER);
|
|
ahpriv->ah_extraTxPow = extra_txpow;
|
|
|
|
if(chan == NULL) {
|
|
return AH_FALSE;
|
|
}
|
|
if (ar9300_eeprom_set_transmit_power(ah, &ahp->ah_eeprom, chan,
|
|
ath_hal_getctl(ah, chan), ath_hal_getantennaallowed(ah, chan),
|
|
ath_hal_get_twice_max_regpower(ahpriv, ichan, chan),
|
|
AH_MIN(MAX_RATE_POWER, ahpriv->ah_powerLimit)) != HAL_OK)
|
|
{
|
|
return AH_FALSE;
|
|
}
|
|
return AH_TRUE;
|
|
}
|
|
|
|
/*
|
|
* Exported call to check for a recent gain reading and return
|
|
* the current state of the thermal calibration gain engine.
|
|
*/
|
|
HAL_RFGAIN
|
|
ar9300_get_rfgain(struct ath_hal *ah)
|
|
{
|
|
return HAL_RFGAIN_INACTIVE;
|
|
}
|
|
|
|
#define HAL_GREEN_AP_RX_MASK 0x1
|
|
|
|
static inline void
|
|
ar9300_init_chain_masks(struct ath_hal *ah, int rx_chainmask, int tx_chainmask)
|
|
{
|
|
if (AH9300(ah)->green_ap_ps_on) {
|
|
rx_chainmask = HAL_GREEN_AP_RX_MASK;
|
|
}
|
|
if (rx_chainmask == 0x5) {
|
|
OS_REG_SET_BIT(ah, AR_PHY_ANALOG_SWAP, AR_PHY_SWAP_ALT_CHAIN);
|
|
}
|
|
OS_REG_WRITE(ah, AR_PHY_RX_CHAINMASK, rx_chainmask);
|
|
OS_REG_WRITE(ah, AR_PHY_CAL_CHAINMASK, rx_chainmask);
|
|
|
|
/*
|
|
* Adaptive Power Management:
|
|
* Some 3 stream chips exceed the PCIe power requirements.
|
|
* This workaround will reduce power consumption by using 2 tx chains
|
|
* for 1 and 2 stream rates (5 GHz only).
|
|
*
|
|
* Set the self gen mask to 2 tx chains when APM is enabled.
|
|
*
|
|
*/
|
|
if (AH_PRIVATE(ah)->ah_caps.halApmEnable && (tx_chainmask == 0x7)) {
|
|
OS_REG_WRITE(ah, AR_SELFGEN_MASK, 0x3);
|
|
}
|
|
else {
|
|
OS_REG_WRITE(ah, AR_SELFGEN_MASK, tx_chainmask);
|
|
}
|
|
|
|
if (tx_chainmask == 0x5) {
|
|
OS_REG_SET_BIT(ah, AR_PHY_ANALOG_SWAP, AR_PHY_SWAP_ALT_CHAIN);
|
|
}
|
|
}
|
|
|
|
/*
|
|
* Override INI values with chip specific configuration.
|
|
*/
|
|
static inline void
|
|
ar9300_override_ini(struct ath_hal *ah, struct ieee80211_channel *chan)
|
|
{
|
|
u_int32_t val;
|
|
HAL_CAPABILITIES *p_cap = &AH_PRIVATE(ah)->ah_caps;
|
|
|
|
/*
|
|
* Set the RX_ABORT and RX_DIS and clear it only after
|
|
* RXE is set for MAC. This prevents frames with
|
|
* corrupted descriptor status.
|
|
*/
|
|
OS_REG_SET_BIT(ah, AR_DIAG_SW, (AR_DIAG_RX_DIS | AR_DIAG_RX_ABORT));
|
|
/*
|
|
* For Merlin and above, there is a new feature that allows Multicast
|
|
* search based on both MAC Address and Key ID.
|
|
* By default, this feature is enabled.
|
|
* But since the driver is not using this feature, we switch it off;
|
|
* otherwise multicast search based on MAC addr only will fail.
|
|
*/
|
|
val = OS_REG_READ(ah, AR_PCU_MISC_MODE2) & (~AR_ADHOC_MCAST_KEYID_ENABLE);
|
|
OS_REG_WRITE(ah, AR_PCU_MISC_MODE2,
|
|
val | AR_BUG_58603_FIX_ENABLE | AR_AGG_WEP_ENABLE);
|
|
|
|
|
|
/* Osprey revision specific configuration */
|
|
|
|
/* Osprey 2.0+ - if SW RAC support is disabled, must also disable
|
|
* the Osprey 2.0 hardware RAC fix.
|
|
*/
|
|
if (p_cap->halIsrRacSupport == AH_FALSE) {
|
|
OS_REG_CLR_BIT(ah, AR_CFG, AR_CFG_MISSING_TX_INTR_FIX_ENABLE);
|
|
}
|
|
|
|
/* try to enable old pal if it is needed for h/w green tx */
|
|
ar9300_hwgreentx_set_pal_spare(ah, 1);
|
|
}
|
|
|
|
static inline void
|
|
ar9300_prog_ini(struct ath_hal *ah, struct ar9300_ini_array *ini_arr,
|
|
int column)
|
|
{
|
|
int i, reg_writes = 0;
|
|
|
|
/* New INI format: Array may be undefined (pre, core, post arrays) */
|
|
if (ini_arr->ia_array == NULL) {
|
|
return;
|
|
}
|
|
|
|
/*
|
|
* New INI format: Pre, core, and post arrays for a given subsystem may be
|
|
* modal (> 2 columns) or non-modal (2 columns).
|
|
* Determine if the array is non-modal and force the column to 1.
|
|
*/
|
|
if (column >= ini_arr->ia_columns) {
|
|
column = 1;
|
|
}
|
|
|
|
for (i = 0; i < ini_arr->ia_rows; i++) {
|
|
u_int32_t reg = INI_RA(ini_arr, i, 0);
|
|
u_int32_t val = INI_RA(ini_arr, i, column);
|
|
|
|
/*
|
|
** Determine if this is a shift register value
|
|
** (reg >= 0x16000 && reg < 0x17000 for Osprey) ,
|
|
** and insert the configured delay if so.
|
|
** -this delay is not required for Osprey (EV#71410)
|
|
*/
|
|
OS_REG_WRITE(ah, reg, val);
|
|
WAR_6773(reg_writes);
|
|
|
|
}
|
|
}
|
|
|
|
static inline HAL_STATUS
|
|
ar9300_process_ini(struct ath_hal *ah, struct ieee80211_channel *chan,
|
|
HAL_CHANNEL_INTERNAL *ichan, HAL_HT_MACMODE macmode)
|
|
{
|
|
int reg_writes = 0;
|
|
struct ath_hal_9300 *ahp = AH9300(ah);
|
|
u_int modes_index, modes_txgaintable_index = 0;
|
|
int i;
|
|
HAL_STATUS status;
|
|
struct ath_hal_private *ahpriv = AH_PRIVATE(ah);
|
|
/* Setup the indices for the next set of register array writes */
|
|
/* TODO:
|
|
* If the channel marker is indicative of the current mode rather
|
|
* than capability, we do not need to check the phy mode below.
|
|
*/
|
|
#if 0
|
|
switch (chan->channel_flags & CHANNEL_ALL) {
|
|
case CHANNEL_A:
|
|
case CHANNEL_A_HT20:
|
|
if (AR_SREV_SCORPION(ah)){
|
|
if (chan->channel <= 5350){
|
|
modes_txgaintable_index = 1;
|
|
}else if ((chan->channel > 5350) && (chan->channel <= 5600)){
|
|
modes_txgaintable_index = 3;
|
|
}else if (chan->channel > 5600){
|
|
modes_txgaintable_index = 5;
|
|
}
|
|
}
|
|
modes_index = 1;
|
|
freq_index = 1;
|
|
break;
|
|
|
|
case CHANNEL_A_HT40PLUS:
|
|
case CHANNEL_A_HT40MINUS:
|
|
if (AR_SREV_SCORPION(ah)){
|
|
if (chan->channel <= 5350){
|
|
modes_txgaintable_index = 2;
|
|
}else if ((chan->channel > 5350) && (chan->channel <= 5600)){
|
|
modes_txgaintable_index = 4;
|
|
}else if (chan->channel > 5600){
|
|
modes_txgaintable_index = 6;
|
|
}
|
|
}
|
|
modes_index = 2;
|
|
freq_index = 1;
|
|
break;
|
|
|
|
case CHANNEL_PUREG:
|
|
case CHANNEL_G_HT20:
|
|
case CHANNEL_B:
|
|
if (AR_SREV_SCORPION(ah)){
|
|
modes_txgaintable_index = 8;
|
|
}else if (AR_SREV_HONEYBEE(ah)){
|
|
modes_txgaintable_index = 1;
|
|
}
|
|
modes_index = 4;
|
|
freq_index = 2;
|
|
break;
|
|
|
|
case CHANNEL_G_HT40PLUS:
|
|
case CHANNEL_G_HT40MINUS:
|
|
if (AR_SREV_SCORPION(ah)){
|
|
modes_txgaintable_index = 7;
|
|
}else if (AR_SREV_HONEYBEE(ah)){
|
|
modes_txgaintable_index = 1;
|
|
}
|
|
modes_index = 3;
|
|
freq_index = 2;
|
|
break;
|
|
|
|
case CHANNEL_108G:
|
|
modes_index = 5;
|
|
freq_index = 2;
|
|
break;
|
|
|
|
default:
|
|
HALASSERT(0);
|
|
return HAL_EINVAL;
|
|
}
|
|
#endif
|
|
|
|
/* FreeBSD */
|
|
if (IS_CHAN_5GHZ(ichan)) {
|
|
if (IEEE80211_IS_CHAN_HT40U(chan) || IEEE80211_IS_CHAN_HT40D(chan)) {
|
|
if (AR_SREV_SCORPION(ah)){
|
|
if (ichan->channel <= 5350){
|
|
modes_txgaintable_index = 2;
|
|
}else if ((ichan->channel > 5350) && (ichan->channel <= 5600)){
|
|
modes_txgaintable_index = 4;
|
|
}else if (ichan->channel > 5600){
|
|
modes_txgaintable_index = 6;
|
|
}
|
|
}
|
|
modes_index = 2;
|
|
} else if (IEEE80211_IS_CHAN_A(chan) || IEEE80211_IS_CHAN_HT20(chan)) {
|
|
if (AR_SREV_SCORPION(ah)){
|
|
if (ichan->channel <= 5350){
|
|
modes_txgaintable_index = 1;
|
|
}else if ((ichan->channel > 5350) && (ichan->channel <= 5600)){
|
|
modes_txgaintable_index = 3;
|
|
}else if (ichan->channel > 5600){
|
|
modes_txgaintable_index = 5;
|
|
}
|
|
}
|
|
modes_index = 1;
|
|
} else
|
|
return HAL_EINVAL;
|
|
} else if (IS_CHAN_2GHZ(ichan)) {
|
|
if (IEEE80211_IS_CHAN_108G(chan)) {
|
|
modes_index = 5;
|
|
} else if (IEEE80211_IS_CHAN_HT40U(chan) || IEEE80211_IS_CHAN_HT40D(chan)) {
|
|
if (AR_SREV_SCORPION(ah)){
|
|
modes_txgaintable_index = 7;
|
|
} else if (AR_SREV_HONEYBEE(ah)){
|
|
modes_txgaintable_index = 1;
|
|
}
|
|
modes_index = 3;
|
|
} else if (IEEE80211_IS_CHAN_HT20(chan) || IEEE80211_IS_CHAN_G(chan) || IEEE80211_IS_CHAN_B(chan) || IEEE80211_IS_CHAN_PUREG(chan)) {
|
|
if (AR_SREV_SCORPION(ah)){
|
|
modes_txgaintable_index = 8;
|
|
} else if (AR_SREV_HONEYBEE(ah)){
|
|
modes_txgaintable_index = 1;
|
|
}
|
|
modes_index = 4;
|
|
} else
|
|
return HAL_EINVAL;
|
|
} else
|
|
return HAL_EINVAL;
|
|
|
|
#if 0
|
|
/* Set correct Baseband to analog shift setting to access analog chips. */
|
|
OS_REG_WRITE(ah, AR_PHY(0), 0x00000007);
|
|
#endif
|
|
|
|
HALDEBUG(ah, HAL_DEBUG_RESET,
|
|
"ar9300_process_ini: "
|
|
"Skipping OS-REG-WRITE(ah, AR-PHY(0), 0x00000007)\n");
|
|
HALDEBUG(ah, HAL_DEBUG_RESET,
|
|
"ar9300_process_ini: no ADDac programming\n");
|
|
|
|
|
|
/*
|
|
* Osprey 2.0+ - new INI format.
|
|
* Each subsystem has a pre, core, and post array.
|
|
*/
|
|
for (i = 0; i < ATH_INI_NUM_SPLIT; i++) {
|
|
ar9300_prog_ini(ah, &ahp->ah_ini_soc[i], modes_index);
|
|
ar9300_prog_ini(ah, &ahp->ah_ini_mac[i], modes_index);
|
|
ar9300_prog_ini(ah, &ahp->ah_ini_bb[i], modes_index);
|
|
ar9300_prog_ini(ah, &ahp->ah_ini_radio[i], modes_index);
|
|
if ((i == ATH_INI_POST) && (AR_SREV_JUPITER_20_OR_LATER(ah) || AR_SREV_APHRODITE(ah))) {
|
|
ar9300_prog_ini(ah, &ahp->ah_ini_radio_post_sys2ant, modes_index);
|
|
}
|
|
|
|
}
|
|
|
|
if (!(AR_SREV_SOC(ah))) {
|
|
/* Doubler issue : Some board doesn't work well with MCS15. Turn off doubler after freq locking is complete*/
|
|
//ath_hal_printf(ah, "%s[%d] ==== before reg[0x%08x] = 0x%08x\n", __func__, __LINE__, AR_PHY_65NM_CH0_RXTX2, OS_REG_READ(ah, AR_PHY_65NM_CH0_RXTX2));
|
|
OS_REG_RMW(ah, AR_PHY_65NM_CH0_RXTX2, 1 << AR_PHY_65NM_CH0_RXTX2_SYNTHON_MASK_S |
|
|
1 << AR_PHY_65NM_CH0_RXTX2_SYNTHOVR_MASK_S, 0); /*Set synthon, synthover */
|
|
//ath_hal_printf(ah, "%s[%d] ==== after reg[0x%08x] = 0x%08x\n", __func__, __LINE__, AR_PHY_65NM_CH0_RXTX2, OS_REG_READ(ah, AR_PHY_65NM_CH0_RXTX2));
|
|
|
|
OS_REG_RMW(ah, AR_PHY_65NM_CH1_RXTX2, 1 << AR_PHY_65NM_CH0_RXTX2_SYNTHON_MASK_S |
|
|
1 << AR_PHY_65NM_CH0_RXTX2_SYNTHOVR_MASK_S, 0); /*Set synthon, synthover */
|
|
OS_REG_RMW(ah, AR_PHY_65NM_CH2_RXTX2, 1 << AR_PHY_65NM_CH0_RXTX2_SYNTHON_MASK_S |
|
|
1 << AR_PHY_65NM_CH0_RXTX2_SYNTHOVR_MASK_S, 0); /*Set synthon, synthover */
|
|
OS_DELAY(200);
|
|
|
|
//ath_hal_printf(ah, "%s[%d] ==== before reg[0x%08x] = 0x%08x\n", __func__, __LINE__, AR_PHY_65NM_CH0_RXTX2, OS_REG_READ(ah, AR_PHY_65NM_CH0_RXTX2));
|
|
OS_REG_CLR_BIT(ah, AR_PHY_65NM_CH0_RXTX2, AR_PHY_65NM_CH0_RXTX2_SYNTHON_MASK); /* clr synthon */
|
|
OS_REG_CLR_BIT(ah, AR_PHY_65NM_CH1_RXTX2, AR_PHY_65NM_CH0_RXTX2_SYNTHON_MASK); /* clr synthon */
|
|
OS_REG_CLR_BIT(ah, AR_PHY_65NM_CH2_RXTX2, AR_PHY_65NM_CH0_RXTX2_SYNTHON_MASK); /* clr synthon */
|
|
//ath_hal_printf(ah, "%s[%d] ==== after reg[0x%08x] = 0x%08x\n", __func__, __LINE__, AR_PHY_65NM_CH0_RXTX2, OS_REG_READ(ah, AR_PHY_65NM_CH0_RXTX2));
|
|
|
|
OS_DELAY(1);
|
|
|
|
//ath_hal_printf(ah, "%s[%d] ==== before reg[0x%08x] = 0x%08x\n", __func__, __LINE__, AR_PHY_65NM_CH0_RXTX2, OS_REG_READ(ah, AR_PHY_65NM_CH0_RXTX2));
|
|
OS_REG_RMW_FIELD(ah, AR_PHY_65NM_CH0_RXTX2, AR_PHY_65NM_CH0_RXTX2_SYNTHON_MASK, 1); /* set synthon */
|
|
OS_REG_RMW_FIELD(ah, AR_PHY_65NM_CH1_RXTX2, AR_PHY_65NM_CH0_RXTX2_SYNTHON_MASK, 1); /* set synthon */
|
|
OS_REG_RMW_FIELD(ah, AR_PHY_65NM_CH2_RXTX2, AR_PHY_65NM_CH0_RXTX2_SYNTHON_MASK, 1); /* set synthon */
|
|
//ath_hal_printf(ah, "%s[%d] ==== after reg[0x%08x] = 0x%08x\n", __func__, __LINE__, AR_PHY_65NM_CH0_RXTX2, OS_REG_READ(ah, AR_PHY_65NM_CH0_RXTX2));
|
|
|
|
OS_DELAY(200);
|
|
|
|
//ath_hal_printf(ah, "%s[%d] ==== before reg[0x%08x] = 0x%08x\n", __func__, __LINE__, AR_PHY_65NM_CH0_SYNTH12, OS_REG_READ(ah, AR_PHY_65NM_CH0_SYNTH12));
|
|
OS_REG_RMW_FIELD(ah, AR_PHY_65NM_CH0_SYNTH12, AR_PHY_65NM_CH0_SYNTH12_VREFMUL3, 0xf);
|
|
//OS_REG_CLR_BIT(ah, AR_PHY_65NM_CH0_SYNTH12, 1<< 16); /* clr charge pump */
|
|
//ath_hal_printf(ah, "%s[%d] ==== After reg[0x%08x] = 0x%08x\n", __func__, __LINE__, AR_PHY_65NM_CH0_SYNTH12, OS_REG_READ(ah, AR_PHY_65NM_CH0_SYNTH12));
|
|
|
|
OS_REG_RMW(ah, AR_PHY_65NM_CH0_RXTX2, 0, 1 << AR_PHY_65NM_CH0_RXTX2_SYNTHON_MASK_S |
|
|
1 << AR_PHY_65NM_CH0_RXTX2_SYNTHOVR_MASK_S); /*Clr synthon, synthover */
|
|
OS_REG_RMW(ah, AR_PHY_65NM_CH1_RXTX2, 0, 1 << AR_PHY_65NM_CH0_RXTX2_SYNTHON_MASK_S |
|
|
1 << AR_PHY_65NM_CH0_RXTX2_SYNTHOVR_MASK_S); /*Clr synthon, synthover */
|
|
OS_REG_RMW(ah, AR_PHY_65NM_CH2_RXTX2, 0, 1 << AR_PHY_65NM_CH0_RXTX2_SYNTHON_MASK_S |
|
|
1 << AR_PHY_65NM_CH0_RXTX2_SYNTHOVR_MASK_S); /*Clr synthon, synthover */
|
|
//ath_hal_printf(ah, "%s[%d] ==== after reg[0x%08x] = 0x%08x\n", __func__, __LINE__, AR_PHY_65NM_CH0_RXTX2, OS_REG_READ(ah, AR_PHY_65NM_CH0_RXTX2));
|
|
}
|
|
|
|
/* Write rxgain Array Parameters */
|
|
REG_WRITE_ARRAY(&ahp->ah_ini_modes_rxgain, 1, reg_writes);
|
|
HALDEBUG(ah, HAL_DEBUG_RESET, "ar9300_process_ini: Rx Gain programming\n");
|
|
|
|
if (AR_SREV_JUPITER_20_OR_LATER(ah)) {
|
|
/*
|
|
* CUS217 mix LNA mode.
|
|
*/
|
|
if (ar9300_rx_gain_index_get(ah) == 2) {
|
|
REG_WRITE_ARRAY(&ahp->ah_ini_modes_rxgain_bb_core, 1, reg_writes);
|
|
REG_WRITE_ARRAY(&ahp->ah_ini_modes_rxgain_bb_postamble,
|
|
modes_index, reg_writes);
|
|
}
|
|
|
|
/*
|
|
* 5G-XLNA
|
|
*/
|
|
if ((ar9300_rx_gain_index_get(ah) == 2) ||
|
|
(ar9300_rx_gain_index_get(ah) == 3)) {
|
|
REG_WRITE_ARRAY(&ahp->ah_ini_modes_rxgain_xlna, modes_index,
|
|
reg_writes);
|
|
}
|
|
}
|
|
|
|
if (AR_SREV_SCORPION(ah)) {
|
|
/* Write rxgain bounds Array */
|
|
REG_WRITE_ARRAY(&ahp->ah_ini_modes_rxgain_bounds, modes_index, reg_writes);
|
|
HALDEBUG(ah, HAL_DEBUG_RESET, "ar9300_process_ini: Rx Gain table bounds programming\n");
|
|
}
|
|
/* UB124 xLNA settings */
|
|
if (AR_SREV_WASP(ah) && ar9300_rx_gain_index_get(ah) == 2) {
|
|
#define REG_WRITE(_reg,_val) *((volatile u_int32_t *)(_reg)) = (_val);
|
|
#define REG_READ(_reg) *((volatile u_int32_t *)(_reg))
|
|
u_int32_t val;
|
|
/* B8040000: bit[0]=0, bit[3]=0; */
|
|
val = REG_READ(0xB8040000);
|
|
val &= 0xfffffff6;
|
|
REG_WRITE(0xB8040000, val);
|
|
/* B804002c: bit[31:24]=0x2e; bit[7:0]=0x2f; */
|
|
val = REG_READ(0xB804002c);
|
|
val &= 0x00ffff00;
|
|
val |= 0x2e00002f;
|
|
REG_WRITE(0xB804002c, val);
|
|
/* B804006c: bit[1]=1; */
|
|
val = REG_READ(0xB804006c);
|
|
val |= 0x2;
|
|
REG_WRITE(0xB804006c, val);
|
|
#undef REG_READ
|
|
#undef REG_WRITE
|
|
}
|
|
|
|
|
|
/* Write txgain Array Parameters */
|
|
if (AR_SREV_SCORPION(ah) || AR_SREV_HONEYBEE(ah)) {
|
|
REG_WRITE_ARRAY(&ahp->ah_ini_modes_txgain, modes_txgaintable_index,
|
|
reg_writes);
|
|
}else{
|
|
REG_WRITE_ARRAY(&ahp->ah_ini_modes_txgain, modes_index, reg_writes);
|
|
}
|
|
HALDEBUG(ah, HAL_DEBUG_RESET, "ar9300_process_ini: Tx Gain programming\n");
|
|
|
|
|
|
/* For 5GHz channels requiring Fast Clock, apply different modal values */
|
|
if (IS_5GHZ_FAST_CLOCK_EN(ah, chan)) {
|
|
HALDEBUG(ah, HAL_DEBUG_RESET,
|
|
"%s: Fast clock enabled, use special ini values\n", __func__);
|
|
REG_WRITE_ARRAY(&ahp->ah_ini_modes_additional, modes_index, reg_writes);
|
|
}
|
|
|
|
if (AR_SREV_HORNET(ah) || AR_SREV_POSEIDON(ah)) {
|
|
HALDEBUG(ah, HAL_DEBUG_RESET,
|
|
"%s: use xtal ini for AH9300(ah)->clk_25mhz: %d\n",
|
|
__func__, AH9300(ah)->clk_25mhz);
|
|
REG_WRITE_ARRAY(
|
|
&ahp->ah_ini_modes_additional, 1/*modes_index*/, reg_writes);
|
|
}
|
|
|
|
if (AR_SREV_WASP(ah) && (AH9300(ah)->clk_25mhz == 0)) {
|
|
HALDEBUG(ah, HAL_DEBUG_RESET, "%s: Apply 40MHz ini settings\n", __func__);
|
|
REG_WRITE_ARRAY(
|
|
&ahp->ah_ini_modes_additional_40mhz, 1/*modesIndex*/, reg_writes);
|
|
}
|
|
|
|
/* Handle Japan Channel 14 channel spreading */
|
|
if (2484 == ichan->channel) {
|
|
ar9300_prog_ini(ah, &ahp->ah_ini_japan2484, 1);
|
|
}
|
|
|
|
#if 0
|
|
/* XXX TODO! */
|
|
if (AR_SREV_JUPITER_20_OR_LATER(ah) || AR_SREV_APHRODITE(ah)) {
|
|
ar9300_prog_ini(ah, &ahp->ah_ini_BTCOEX_MAX_TXPWR, 1);
|
|
}
|
|
#endif
|
|
|
|
/* Override INI with chip specific configuration */
|
|
ar9300_override_ini(ah, chan);
|
|
|
|
/* Setup 11n MAC/Phy mode registers */
|
|
ar9300_set_11n_regs(ah, chan, macmode);
|
|
|
|
/*
|
|
* Moved ar9300_init_chain_masks() here to ensure the swap bit is set before
|
|
* the pdadc table is written. Swap must occur before any radio dependent
|
|
* replicated register access. The pdadc curve addressing in particular
|
|
* depends on the consistent setting of the swap bit.
|
|
*/
|
|
ar9300_init_chain_masks(ah, ahp->ah_rx_chainmask, ahp->ah_tx_chainmask);
|
|
|
|
/*
|
|
* Setup the transmit power values.
|
|
*
|
|
* After the public to private hal channel mapping, ichan contains the
|
|
* valid regulatory power value.
|
|
* ath_hal_getctl and ath_hal_getantennaallowed look up ichan from chan.
|
|
*/
|
|
status = ar9300_eeprom_set_transmit_power(ah, &ahp->ah_eeprom, chan,
|
|
ath_hal_getctl(ah, chan), ath_hal_getantennaallowed(ah, chan),
|
|
ath_hal_get_twice_max_regpower(ahpriv, ichan, chan),
|
|
AH_MIN(MAX_RATE_POWER, ahpriv->ah_powerLimit));
|
|
if (status != HAL_OK) {
|
|
HALDEBUG(ah, HAL_DEBUG_POWER_MGMT,
|
|
"%s: error init'ing transmit power\n", __func__);
|
|
return HAL_EIO;
|
|
}
|
|
|
|
|
|
return HAL_OK;
|
|
#undef N
|
|
}
|
|
|
|
/* ar9300_is_cal_supp
|
|
* Determine if calibration is supported by device and channel flags
|
|
*/
|
|
inline static HAL_BOOL
|
|
ar9300_is_cal_supp(struct ath_hal *ah, const struct ieee80211_channel *chan,
|
|
HAL_CAL_TYPES cal_type)
|
|
{
|
|
struct ath_hal_9300 *ahp = AH9300(ah);
|
|
HAL_BOOL retval = AH_FALSE;
|
|
|
|
switch (cal_type & ahp->ah_supp_cals) {
|
|
case IQ_MISMATCH_CAL:
|
|
/* Run IQ Mismatch for non-CCK only */
|
|
if (!IEEE80211_IS_CHAN_B(chan)) {
|
|
retval = AH_TRUE;
|
|
}
|
|
break;
|
|
case TEMP_COMP_CAL:
|
|
retval = AH_TRUE;
|
|
break;
|
|
}
|
|
|
|
return retval;
|
|
}
|
|
|
|
|
|
#if 0
|
|
/* ar9285_pa_cal
|
|
* PA Calibration for Kite 1.1 and later versions of Kite.
|
|
* - from system's team.
|
|
*/
|
|
static inline void
|
|
ar9285_pa_cal(struct ath_hal *ah)
|
|
{
|
|
u_int32_t reg_val;
|
|
int i, lo_gn, offs_6_1, offs_0;
|
|
u_int8_t reflo;
|
|
u_int32_t phy_test2_reg_val, phy_adc_ctl_reg_val;
|
|
u_int32_t an_top2_reg_val, phy_tst_dac_reg_val;
|
|
|
|
|
|
/* Kite 1.1 WAR for Bug 35666
|
|
* Increase the LDO value to 1.28V before accessing analog Reg */
|
|
if (AR_SREV_KITE_11(ah)) {
|
|
OS_REG_WRITE(ah, AR9285_AN_TOP4, (AR9285_AN_TOP4_DEFAULT | 0x14) );
|
|
}
|
|
an_top2_reg_val = OS_REG_READ(ah, AR9285_AN_TOP2);
|
|
|
|
/* set pdv2i pdrxtxbb */
|
|
reg_val = OS_REG_READ(ah, AR9285_AN_RXTXBB1);
|
|
reg_val |= ((0x1 << 5) | (0x1 << 7));
|
|
OS_REG_WRITE(ah, AR9285_AN_RXTXBB1, reg_val);
|
|
|
|
/* clear pwddb */
|
|
reg_val = OS_REG_READ(ah, AR9285_AN_RF2G7);
|
|
reg_val &= 0xfffffffd;
|
|
OS_REG_WRITE(ah, AR9285_AN_RF2G7, reg_val);
|
|
|
|
/* clear enpacal */
|
|
reg_val = OS_REG_READ(ah, AR9285_AN_RF2G1);
|
|
reg_val &= 0xfffff7ff;
|
|
OS_REG_WRITE(ah, AR9285_AN_RF2G1, reg_val);
|
|
|
|
/* set offcal */
|
|
reg_val = OS_REG_READ(ah, AR9285_AN_RF2G2);
|
|
reg_val |= (0x1 << 12);
|
|
OS_REG_WRITE(ah, AR9285_AN_RF2G2, reg_val);
|
|
|
|
/* set pdpadrv1=pdpadrv2=pdpaout=1 */
|
|
reg_val = OS_REG_READ(ah, AR9285_AN_RF2G1);
|
|
reg_val |= (0x7 << 23);
|
|
OS_REG_WRITE(ah, AR9285_AN_RF2G1, reg_val);
|
|
|
|
/* Read back reflo, increase it by 1 and write it. */
|
|
reg_val = OS_REG_READ(ah, AR9285_AN_RF2G3);
|
|
reflo = ((reg_val >> 26) & 0x7);
|
|
|
|
if (reflo < 0x7) {
|
|
reflo++;
|
|
}
|
|
reg_val = ((reg_val & 0xe3ffffff) | (reflo << 26));
|
|
OS_REG_WRITE(ah, AR9285_AN_RF2G3, reg_val);
|
|
|
|
reg_val = OS_REG_READ(ah, AR9285_AN_RF2G3);
|
|
reflo = ((reg_val >> 26) & 0x7);
|
|
|
|
/* use TX single carrier to transmit
|
|
* dac const
|
|
* reg. 15
|
|
*/
|
|
phy_tst_dac_reg_val = OS_REG_READ(ah, AR_PHY_TSTDAC_CONST);
|
|
OS_REG_WRITE(ah, AR_PHY_TSTDAC_CONST, ((0x7ff << 11) | 0x7ff));
|
|
reg_val = OS_REG_READ(ah, AR_PHY_TSTDAC_CONST);
|
|
|
|
/* source is dac const
|
|
* reg. 2
|
|
*/
|
|
phy_test2_reg_val = OS_REG_READ(ah, AR_PHY_TEST2);
|
|
OS_REG_WRITE(ah, AR_PHY_TEST2, ((0x1 << 7) | (0x1 << 1)));
|
|
reg_val = OS_REG_READ(ah, AR_PHY_TEST2);
|
|
|
|
/* set dac on
|
|
* reg. 11
|
|
*/
|
|
phy_adc_ctl_reg_val = OS_REG_READ(ah, AR_PHY_ADC_CTL);
|
|
OS_REG_WRITE(ah, AR_PHY_ADC_CTL, 0x80008000);
|
|
reg_val = OS_REG_READ(ah, AR_PHY_ADC_CTL);
|
|
|
|
OS_REG_WRITE(ah, AR9285_AN_TOP2, (0x1 << 27) | (0x1 << 17) | (0x1 << 16) |
|
|
(0x1 << 14) | (0x1 << 12) | (0x1 << 11) |
|
|
(0x1 << 7) | (0x1 << 5));
|
|
|
|
OS_DELAY(10); /* 10 usec */
|
|
|
|
/* clear off[6:0] */
|
|
reg_val = OS_REG_READ(ah, AR9285_AN_RF2G6);
|
|
reg_val &= 0xfc0fffff;
|
|
OS_REG_WRITE(ah, AR9285_AN_RF2G6, reg_val);
|
|
reg_val = OS_REG_READ(ah, AR9285_AN_RF2G3);
|
|
reg_val &= 0xfdffffff;
|
|
OS_REG_WRITE(ah, AR9285_AN_RF2G3, reg_val);
|
|
|
|
offs_6_1 = 0;
|
|
for (i = 6; i > 0; i--) {
|
|
/* sef off[$k]==1 */
|
|
reg_val = OS_REG_READ(ah, AR9285_AN_RF2G6);
|
|
reg_val &= 0xfc0fffff;
|
|
reg_val = reg_val | (0x1 << (19 + i)) | ((offs_6_1) << 20);
|
|
OS_REG_WRITE(ah, AR9285_AN_RF2G6, reg_val);
|
|
lo_gn = (OS_REG_READ(ah, AR9285_AN_RF2G9)) & 0x1;
|
|
offs_6_1 = offs_6_1 | (lo_gn << (i - 1));
|
|
}
|
|
|
|
reg_val = OS_REG_READ(ah, AR9285_AN_RF2G6);
|
|
reg_val &= 0xfc0fffff;
|
|
reg_val = reg_val | ((offs_6_1 - 1) << 20);
|
|
OS_REG_WRITE(ah, AR9285_AN_RF2G6, reg_val);
|
|
|
|
/* set off_0=1; */
|
|
reg_val = OS_REG_READ(ah, AR9285_AN_RF2G3);
|
|
reg_val &= 0xfdffffff;
|
|
reg_val = reg_val | (0x1 << 25);
|
|
OS_REG_WRITE(ah, AR9285_AN_RF2G3, reg_val);
|
|
|
|
lo_gn = OS_REG_READ(ah, AR9285_AN_RF2G9) & 0x1;
|
|
offs_0 = lo_gn;
|
|
|
|
reg_val = OS_REG_READ(ah, AR9285_AN_RF2G3);
|
|
reg_val &= 0xfdffffff;
|
|
reg_val = reg_val | (offs_0 << 25);
|
|
OS_REG_WRITE(ah, AR9285_AN_RF2G3, reg_val);
|
|
|
|
/* clear pdv2i */
|
|
reg_val = OS_REG_READ(ah, AR9285_AN_RXTXBB1);
|
|
reg_val &= 0xffffff5f;
|
|
OS_REG_WRITE(ah, AR9285_AN_RXTXBB1, reg_val);
|
|
|
|
/* set enpacal */
|
|
reg_val = OS_REG_READ(ah, AR9285_AN_RF2G1);
|
|
reg_val |= (0x1 << 11);
|
|
OS_REG_WRITE(ah, AR9285_AN_RF2G1, reg_val);
|
|
|
|
/* clear offcal */
|
|
reg_val = OS_REG_READ(ah, AR9285_AN_RF2G2);
|
|
reg_val &= 0xffffefff;
|
|
OS_REG_WRITE(ah, AR9285_AN_RF2G2, reg_val);
|
|
|
|
/* set pdpadrv1=pdpadrv2=pdpaout=0 */
|
|
reg_val = OS_REG_READ(ah, AR9285_AN_RF2G1);
|
|
reg_val &= 0xfc7fffff;
|
|
OS_REG_WRITE(ah, AR9285_AN_RF2G1, reg_val);
|
|
|
|
/* Read back reflo, decrease it by 1 and write it. */
|
|
reg_val = OS_REG_READ(ah, AR9285_AN_RF2G3);
|
|
reflo = (reg_val >> 26) & 0x7;
|
|
if (reflo) {
|
|
reflo--;
|
|
}
|
|
reg_val = ((reg_val & 0xe3ffffff) | (reflo << 26));
|
|
OS_REG_WRITE(ah, AR9285_AN_RF2G3, reg_val);
|
|
reg_val = OS_REG_READ(ah, AR9285_AN_RF2G3);
|
|
reflo = (reg_val >> 26) & 0x7;
|
|
|
|
/* write back registers */
|
|
OS_REG_WRITE(ah, AR_PHY_TSTDAC_CONST, phy_tst_dac_reg_val);
|
|
OS_REG_WRITE(ah, AR_PHY_TEST2, phy_test2_reg_val);
|
|
OS_REG_WRITE(ah, AR_PHY_ADC_CTL, phy_adc_ctl_reg_val);
|
|
OS_REG_WRITE(ah, AR9285_AN_TOP2, an_top2_reg_val);
|
|
|
|
/* Kite 1.1 WAR for Bug 35666
|
|
* Decrease the LDO value back to 1.20V */
|
|
if (AR_SREV_KITE_11(ah)) {
|
|
OS_REG_WRITE(ah, AR9285_AN_TOP4, AR9285_AN_TOP4_DEFAULT);
|
|
}
|
|
}
|
|
#endif
|
|
|
|
/* ar9300_run_init_cals
|
|
* Runs non-periodic calibrations
|
|
*/
|
|
inline static HAL_BOOL
|
|
ar9300_run_init_cals(struct ath_hal *ah, int init_cal_count)
|
|
{
|
|
struct ath_hal_9300 *ahp = AH9300(ah);
|
|
HAL_CHANNEL_INTERNAL ichan; /* bogus */
|
|
HAL_BOOL is_cal_done;
|
|
HAL_CAL_LIST *curr_cal;
|
|
const HAL_PERCAL_DATA *cal_data;
|
|
int i;
|
|
|
|
curr_cal = ahp->ah_cal_list_curr;
|
|
if (curr_cal == AH_NULL) {
|
|
return AH_FALSE;
|
|
}
|
|
cal_data = curr_cal->cal_data;
|
|
ichan.calValid = 0;
|
|
|
|
for (i = 0; i < init_cal_count; i++) {
|
|
/* Reset this Cal */
|
|
ar9300_reset_calibration(ah, curr_cal);
|
|
/* Poll for offset calibration complete */
|
|
if (!ath_hal_wait(
|
|
ah, AR_PHY_TIMING4, AR_PHY_TIMING4_DO_CAL, 0))
|
|
{
|
|
HALDEBUG(ah, HAL_DEBUG_CALIBRATE,
|
|
"%s: Cal %d failed to complete in 100ms.\n",
|
|
__func__, curr_cal->cal_data->cal_type);
|
|
/* Re-initialize list pointers for periodic cals */
|
|
ahp->ah_cal_list = ahp->ah_cal_list_last = ahp->ah_cal_list_curr
|
|
= AH_NULL;
|
|
return AH_FALSE;
|
|
}
|
|
/* Run this cal */
|
|
ar9300_per_calibration(
|
|
ah, &ichan, ahp->ah_rx_chainmask, curr_cal, &is_cal_done);
|
|
if (is_cal_done == AH_FALSE) {
|
|
HALDEBUG(ah, HAL_DEBUG_CALIBRATE,
|
|
"%s: Not able to run Init Cal %d.\n", __func__,
|
|
curr_cal->cal_data->cal_type);
|
|
}
|
|
if (curr_cal->cal_next) {
|
|
curr_cal = curr_cal->cal_next;
|
|
}
|
|
}
|
|
|
|
/* Re-initialize list pointers for periodic cals */
|
|
ahp->ah_cal_list = ahp->ah_cal_list_last = ahp->ah_cal_list_curr = AH_NULL;
|
|
return AH_TRUE;
|
|
}
|
|
|
|
#if 0
|
|
static void
|
|
ar9300_tx_carrier_leak_war(struct ath_hal *ah)
|
|
{
|
|
unsigned long tx_gain_table_max;
|
|
unsigned long reg_bb_cl_map_0_b0 = 0xffffffff;
|
|
unsigned long reg_bb_cl_map_1_b0 = 0xffffffff;
|
|
unsigned long reg_bb_cl_map_2_b0 = 0xffffffff;
|
|
unsigned long reg_bb_cl_map_3_b0 = 0xffffffff;
|
|
unsigned long tx_gain, cal_run = 0;
|
|
unsigned long cal_gain[AR_PHY_TPC_7_TX_GAIN_TABLE_MAX + 1];
|
|
unsigned long cal_gain_index[AR_PHY_TPC_7_TX_GAIN_TABLE_MAX + 1];
|
|
unsigned long new_gain[AR_PHY_TPC_7_TX_GAIN_TABLE_MAX + 1];
|
|
int i, j;
|
|
|
|
OS_MEMSET(new_gain, 0, sizeof(new_gain));
|
|
/*printf(" Running TxCarrierLeakWAR\n");*/
|
|
|
|
/* process tx gain table, we use cl_map_hw_gen=0. */
|
|
OS_REG_RMW_FIELD(ah, AR_PHY_CL_CAL_CTL, AR_PHY_CL_MAP_HW_GEN, 0);
|
|
|
|
//the table we used is txbb_gc[2:0], 1dB[2:1].
|
|
tx_gain_table_max = OS_REG_READ_FIELD(ah,
|
|
AR_PHY_TPC_7, AR_PHY_TPC_7_TX_GAIN_TABLE_MAX);
|
|
|
|
for (i = 0; i <= tx_gain_table_max; i++) {
|
|
tx_gain = OS_REG_READ(ah, AR_PHY_TXGAIN_TAB(1) + i * 4);
|
|
cal_gain[i] = (((tx_gain >> 5)& 0x7) << 2) |
|
|
(((tx_gain >> 1) & 0x3) << 0);
|
|
if (i == 0) {
|
|
cal_gain_index[i] = cal_run;
|
|
new_gain[i] = 1;
|
|
cal_run++;
|
|
} else {
|
|
new_gain[i] = 1;
|
|
for (j = 0; j < i; j++) {
|
|
/*
|
|
printf("i=%d, j=%d cal_gain[$i]=0x%04x\n", i, j, cal_gain[i]);
|
|
*/
|
|
if (new_gain[i]) {
|
|
if ((cal_gain[i] != cal_gain[j])) {
|
|
new_gain[i] = 1;
|
|
} else {
|
|
/* if old gain found, use old cal_run value. */
|
|
new_gain[i] = 0;
|
|
cal_gain_index[i] = cal_gain_index[j];
|
|
}
|
|
}
|
|
}
|
|
/* if new gain found, increase cal_run */
|
|
if (new_gain[i] == 1) {
|
|
cal_gain_index[i] = cal_run;
|
|
cal_run++;
|
|
}
|
|
}
|
|
|
|
reg_bb_cl_map_0_b0 = (reg_bb_cl_map_0_b0 & ~(0x1 << i)) |
|
|
((cal_gain_index[i] >> 0 & 0x1) << i);
|
|
reg_bb_cl_map_1_b0 = (reg_bb_cl_map_1_b0 & ~(0x1 << i)) |
|
|
((cal_gain_index[i] >> 1 & 0x1) << i);
|
|
reg_bb_cl_map_2_b0 = (reg_bb_cl_map_2_b0 & ~(0x1 << i)) |
|
|
((cal_gain_index[i] >> 2 & 0x1) << i);
|
|
reg_bb_cl_map_3_b0 = (reg_bb_cl_map_3_b0 & ~(0x1 << i)) |
|
|
((cal_gain_index[i] >> 3 & 0x1) << i);
|
|
|
|
/*
|
|
printf("i=%2d, cal_gain[$i]= 0x%04x, cal_run= %d, "
|
|
"cal_gain_index[i]=%d, new_gain[i] = %d\n",
|
|
i, cal_gain[i], cal_run, cal_gain_index[i], new_gain[i]);
|
|
*/
|
|
}
|
|
OS_REG_WRITE(ah, AR_PHY_CL_MAP_0_B0, reg_bb_cl_map_0_b0);
|
|
OS_REG_WRITE(ah, AR_PHY_CL_MAP_1_B0, reg_bb_cl_map_1_b0);
|
|
OS_REG_WRITE(ah, AR_PHY_CL_MAP_2_B0, reg_bb_cl_map_2_b0);
|
|
OS_REG_WRITE(ah, AR_PHY_CL_MAP_3_B0, reg_bb_cl_map_3_b0);
|
|
if (AR_SREV_WASP(ah)) {
|
|
OS_REG_WRITE(ah, AR_PHY_CL_MAP_0_B1, reg_bb_cl_map_0_b0);
|
|
OS_REG_WRITE(ah, AR_PHY_CL_MAP_1_B1, reg_bb_cl_map_1_b0);
|
|
OS_REG_WRITE(ah, AR_PHY_CL_MAP_2_B1, reg_bb_cl_map_2_b0);
|
|
OS_REG_WRITE(ah, AR_PHY_CL_MAP_3_B1, reg_bb_cl_map_3_b0);
|
|
}
|
|
}
|
|
#endif
|
|
|
|
|
|
static inline void
|
|
ar9300_invalidate_saved_cals(struct ath_hal *ah, HAL_CHANNEL_INTERNAL *ichan)
|
|
{
|
|
#if ATH_SUPPORT_CAL_REUSE
|
|
if (AH_PRIVATE(ah)->ah_config.ath_hal_cal_reuse &
|
|
ATH_CAL_REUSE_REDO_IN_FULL_RESET)
|
|
{
|
|
ichan->one_time_txiqcal_done = AH_FALSE;
|
|
ichan->one_time_txclcal_done = AH_FALSE;
|
|
}
|
|
#endif
|
|
}
|
|
|
|
static inline HAL_BOOL
|
|
ar9300_restore_rtt_cals(struct ath_hal *ah, HAL_CHANNEL_INTERNAL *ichan)
|
|
{
|
|
HAL_BOOL restore_status = AH_FALSE;
|
|
|
|
return restore_status;
|
|
}
|
|
|
|
/* ar9300_init_cal
|
|
* Initialize Calibration infrastructure
|
|
*/
|
|
static inline HAL_BOOL
|
|
ar9300_init_cal_internal(struct ath_hal *ah, struct ieee80211_channel *chan,
|
|
HAL_CHANNEL_INTERNAL *ichan,
|
|
HAL_BOOL enable_rtt, HAL_BOOL do_rtt_cal, HAL_BOOL skip_if_none, HAL_BOOL apply_last_iqcorr)
|
|
{
|
|
struct ath_hal_9300 *ahp = AH9300(ah);
|
|
HAL_BOOL txiqcal_success_flag = AH_FALSE;
|
|
HAL_BOOL cal_done = AH_FALSE;
|
|
int iqcal_idx = 0;
|
|
HAL_BOOL do_sep_iq_cal = AH_FALSE;
|
|
HAL_BOOL do_agc_cal = do_rtt_cal;
|
|
HAL_BOOL is_cal_reusable = AH_TRUE;
|
|
#if ATH_SUPPORT_CAL_REUSE
|
|
HAL_BOOL cal_reuse_enable = AH_PRIVATE(ah)->ah_config.ath_hal_cal_reuse &
|
|
ATH_CAL_REUSE_ENABLE;
|
|
HAL_BOOL clc_success = AH_FALSE;
|
|
int32_t ch_idx, j, cl_tab_reg;
|
|
u_int32_t BB_cl_tab_entry = MAX_BB_CL_TABLE_ENTRY;
|
|
u_int32_t BB_cl_tab_b[AR9300_MAX_CHAINS] = {
|
|
AR_PHY_CL_TAB_0,
|
|
AR_PHY_CL_TAB_1,
|
|
AR_PHY_CL_TAB_2
|
|
};
|
|
#endif
|
|
|
|
if (AR_SREV_HORNET(ah) || AR_SREV_POSEIDON(ah) || AR_SREV_APHRODITE(ah)) {
|
|
/* Hornet: 1 x 1 */
|
|
ahp->ah_rx_cal_chainmask = 0x1;
|
|
ahp->ah_tx_cal_chainmask = 0x1;
|
|
} else if (AR_SREV_WASP(ah) || AR_SREV_JUPITER(ah) || AR_SREV_HONEYBEE(ah)) {
|
|
/* Wasp/Jupiter: 2 x 2 */
|
|
ahp->ah_rx_cal_chainmask = 0x3;
|
|
ahp->ah_tx_cal_chainmask = 0x3;
|
|
} else {
|
|
/*
|
|
* Osprey needs to be configured for the correct chain mode
|
|
* before running AGC/TxIQ cals.
|
|
*/
|
|
if (ahp->ah_enterprise_mode & AR_ENT_OTP_CHAIN2_DISABLE) {
|
|
/* chain 2 disabled - 2 chain mode */
|
|
ahp->ah_rx_cal_chainmask = 0x3;
|
|
ahp->ah_tx_cal_chainmask = 0x3;
|
|
} else {
|
|
ahp->ah_rx_cal_chainmask = 0x7;
|
|
ahp->ah_tx_cal_chainmask = 0x7;
|
|
}
|
|
}
|
|
ar9300_init_chain_masks(ah, ahp->ah_rx_cal_chainmask, ahp->ah_tx_cal_chainmask);
|
|
|
|
|
|
if (ahp->tx_cl_cal_enable) {
|
|
#if ATH_SUPPORT_CAL_REUSE
|
|
/* disable Carrie Leak or set do_agc_cal accordingly */
|
|
if (cal_reuse_enable && ichan->one_time_txclcal_done)
|
|
{
|
|
OS_REG_CLR_BIT(ah, AR_PHY_CL_CAL_CTL, AR_PHY_CL_CAL_ENABLE);
|
|
} else
|
|
#endif /* ATH_SUPPORT_CAL_REUSE */
|
|
{
|
|
OS_REG_SET_BIT(ah, AR_PHY_CL_CAL_CTL, AR_PHY_CL_CAL_ENABLE);
|
|
do_agc_cal = AH_TRUE;
|
|
}
|
|
}
|
|
|
|
/* Do Tx IQ Calibration here for osprey hornet and wasp */
|
|
/* XXX: For initial wasp bringup - check and enable this */
|
|
/* EV 74233: Tx IQ fails to complete for half/quarter rates */
|
|
if (!(IEEE80211_IS_CHAN_HALF(chan) || IEEE80211_IS_CHAN_QUARTER(chan))) {
|
|
if (ahp->tx_iq_cal_enable) {
|
|
/* this should be eventually moved to INI file */
|
|
OS_REG_RMW_FIELD(ah, AR_PHY_TX_IQCAL_CONTROL_1(ah),
|
|
AR_PHY_TX_IQCAL_CONTROL_1_IQCORR_I_Q_COFF_DELPT, DELPT);
|
|
|
|
/*
|
|
* For poseidon and later chips,
|
|
* Tx IQ cal HW run will be a part of AGC calibration
|
|
*/
|
|
if (ahp->tx_iq_cal_during_agc_cal) {
|
|
/*
|
|
* txiqcal_success_flag always set to 1 to run
|
|
* ar9300_tx_iq_cal_post_proc
|
|
* if following AGC cal passes
|
|
*/
|
|
#if ATH_SUPPORT_CAL_REUSE
|
|
if (!cal_reuse_enable || !ichan->one_time_txiqcal_done)
|
|
{
|
|
txiqcal_success_flag = AH_TRUE;
|
|
OS_REG_WRITE(ah, AR_PHY_TX_IQCAL_CONTROL_0(ah),
|
|
OS_REG_READ(ah, AR_PHY_TX_IQCAL_CONTROL_0(ah)) |
|
|
AR_PHY_TX_IQCAL_CONTROL_0_ENABLE_TXIQ_CAL);
|
|
} else {
|
|
OS_REG_WRITE(ah, AR_PHY_TX_IQCAL_CONTROL_0(ah),
|
|
OS_REG_READ(ah, AR_PHY_TX_IQCAL_CONTROL_0(ah)) &
|
|
(~AR_PHY_TX_IQCAL_CONTROL_0_ENABLE_TXIQ_CAL));
|
|
}
|
|
#else
|
|
if (OS_REG_READ_FIELD(ah,
|
|
AR_PHY_TX_IQCAL_CONTROL_0(ah),
|
|
AR_PHY_TX_IQCAL_CONTROL_0_ENABLE_TXIQ_CAL)){
|
|
if (apply_last_iqcorr == AH_TRUE) {
|
|
OS_REG_CLR_BIT(ah, AR_PHY_TX_IQCAL_CONTROL_0(ah),
|
|
AR_PHY_TX_IQCAL_CONTROL_0_ENABLE_TXIQ_CAL);
|
|
txiqcal_success_flag = AH_FALSE;
|
|
} else {
|
|
txiqcal_success_flag = AH_TRUE;
|
|
}
|
|
}else{
|
|
txiqcal_success_flag = AH_FALSE;
|
|
}
|
|
#endif
|
|
if (txiqcal_success_flag) {
|
|
do_agc_cal = AH_TRUE;
|
|
}
|
|
} else
|
|
#if ATH_SUPPORT_CAL_REUSE
|
|
if (!cal_reuse_enable || !ichan->one_time_txiqcal_done)
|
|
#endif
|
|
{
|
|
do_sep_iq_cal = AH_TRUE;
|
|
do_agc_cal = AH_TRUE;
|
|
}
|
|
}
|
|
}
|
|
|
|
#if ATH_SUPPORT_MCI
|
|
if (AH_PRIVATE(ah)->ah_caps.halMciSupport &&
|
|
IS_CHAN_2GHZ(ichan) &&
|
|
(ahp->ah_mci_bt_state == MCI_BT_AWAKE) &&
|
|
do_agc_cal &&
|
|
!(ah->ah_config.ath_hal_mci_config &
|
|
ATH_MCI_CONFIG_DISABLE_MCI_CAL))
|
|
{
|
|
u_int32_t payload[4] = {0, 0, 0, 0};
|
|
|
|
/* Send CAL_REQ only when BT is AWAKE. */
|
|
HALDEBUG(ah, HAL_DEBUG_BT_COEX, "(MCI) %s: Send WLAN_CAL_REQ 0x%X\n",
|
|
__func__, ahp->ah_mci_wlan_cal_seq);
|
|
MCI_GPM_SET_CAL_TYPE(payload, MCI_GPM_WLAN_CAL_REQ);
|
|
payload[MCI_GPM_WLAN_CAL_W_SEQUENCE] = ahp->ah_mci_wlan_cal_seq++;
|
|
ar9300_mci_send_message(ah, MCI_GPM, 0, payload, 16, AH_TRUE, AH_FALSE);
|
|
|
|
/* Wait BT_CAL_GRANT for 50ms */
|
|
HALDEBUG(ah, HAL_DEBUG_BT_COEX,
|
|
"(MCI) %s: Wait for BT_CAL_GRANT\n", __func__);
|
|
if (ar9300_mci_wait_for_gpm(ah, MCI_GPM_BT_CAL_GRANT, 0, 50000))
|
|
{
|
|
HALDEBUG(ah, HAL_DEBUG_BT_COEX,
|
|
"(MCI) %s: Got BT_CAL_GRANT.\n", __func__);
|
|
}
|
|
else {
|
|
is_cal_reusable = AH_FALSE;
|
|
HALDEBUG(ah, HAL_DEBUG_BT_COEX,
|
|
"(MCI) %s: BT is not responding.\n", __func__);
|
|
}
|
|
}
|
|
#endif /* ATH_SUPPORT_MCI */
|
|
|
|
if (do_sep_iq_cal)
|
|
{
|
|
/* enable Tx IQ Calibration HW for osprey/hornet/wasp */
|
|
txiqcal_success_flag = ar9300_tx_iq_cal_hw_run(ah);
|
|
OS_REG_WRITE(ah, AR_PHY_ACTIVE, AR_PHY_ACTIVE_DIS);
|
|
OS_DELAY(5);
|
|
OS_REG_WRITE(ah, AR_PHY_ACTIVE, AR_PHY_ACTIVE_EN);
|
|
}
|
|
#if 0
|
|
if (AR_SREV_HORNET(ah) || AR_SREV_POSEIDON(ah)) {
|
|
ar9300_tx_carrier_leak_war(ah);
|
|
}
|
|
#endif
|
|
/*
|
|
* Calibrate the AGC
|
|
*
|
|
* Tx IQ cal is a part of AGC cal for Jupiter/Poseidon, etc.
|
|
* please enable the bit of txiqcal_control_0[31] in INI file
|
|
* for Jupiter/Poseidon/etc.
|
|
*/
|
|
if(!AR_SREV_SCORPION(ah)) {
|
|
if (do_agc_cal || !skip_if_none) {
|
|
OS_REG_WRITE(ah, AR_PHY_AGC_CONTROL,
|
|
OS_REG_READ(ah, AR_PHY_AGC_CONTROL) | AR_PHY_AGC_CONTROL_CAL);
|
|
|
|
/* Poll for offset calibration complete */
|
|
cal_done = ath_hal_wait(ah,
|
|
AR_PHY_AGC_CONTROL, AR_PHY_AGC_CONTROL_CAL, 0);
|
|
if (!cal_done) {
|
|
HALDEBUG(ah, HAL_DEBUG_FCS_RTT,
|
|
"(FCS) CAL NOT DONE!!! - %d\n", ichan->channel);
|
|
}
|
|
} else {
|
|
cal_done = AH_TRUE;
|
|
}
|
|
/*
|
|
* Tx IQ cal post-processing in SW
|
|
* This part of code should be common to all chips,
|
|
* no chip specific code for Jupiter/Posdeion except for register names.
|
|
*/
|
|
if (txiqcal_success_flag) {
|
|
ar9300_tx_iq_cal_post_proc(ah,ichan, 1, 1,is_cal_reusable, AH_FALSE);
|
|
}
|
|
} else {
|
|
if (!txiqcal_success_flag) {
|
|
OS_REG_WRITE(ah, AR_PHY_AGC_CONTROL,
|
|
OS_REG_READ(ah, AR_PHY_AGC_CONTROL) | AR_PHY_AGC_CONTROL_CAL);
|
|
if (!ath_hal_wait(ah, AR_PHY_AGC_CONTROL, AR_PHY_AGC_CONTROL_CAL,
|
|
0)) {
|
|
HALDEBUG(ah, HAL_DEBUG_CALIBRATE,
|
|
"%s: offset calibration failed to complete in 1ms; "
|
|
"noisy environment?\n", __func__);
|
|
return AH_FALSE;
|
|
}
|
|
if (apply_last_iqcorr == AH_TRUE) {
|
|
ar9300_tx_iq_cal_post_proc(ah, ichan, 0, 0, is_cal_reusable, AH_TRUE);
|
|
}
|
|
} else {
|
|
for (iqcal_idx=0;iqcal_idx<MAXIQCAL;iqcal_idx++) {
|
|
OS_REG_WRITE(ah, AR_PHY_AGC_CONTROL,
|
|
OS_REG_READ(ah, AR_PHY_AGC_CONTROL) | AR_PHY_AGC_CONTROL_CAL);
|
|
|
|
/* Poll for offset calibration complete */
|
|
if (!ath_hal_wait(ah, AR_PHY_AGC_CONTROL,
|
|
AR_PHY_AGC_CONTROL_CAL, 0)) {
|
|
HALDEBUG(ah, HAL_DEBUG_CALIBRATE,
|
|
"%s: offset calibration failed to complete in 1ms; "
|
|
"noisy environment?\n", __func__);
|
|
return AH_FALSE;
|
|
}
|
|
/*
|
|
* Tx IQ cal post-processing in SW
|
|
* This part of code should be common to all chips,
|
|
* no chip specific code for Jupiter/Posdeion except for register names.
|
|
*/
|
|
ar9300_tx_iq_cal_post_proc(ah, ichan, iqcal_idx+1, MAXIQCAL, is_cal_reusable, AH_FALSE);
|
|
}
|
|
}
|
|
}
|
|
|
|
|
|
#if ATH_SUPPORT_MCI
|
|
if (AH_PRIVATE(ah)->ah_caps.halMciSupport &&
|
|
IS_CHAN_2GHZ(ichan) &&
|
|
(ahp->ah_mci_bt_state == MCI_BT_AWAKE) &&
|
|
do_agc_cal &&
|
|
!(ah->ah_config.ath_hal_mci_config &
|
|
ATH_MCI_CONFIG_DISABLE_MCI_CAL))
|
|
{
|
|
u_int32_t payload[4] = {0, 0, 0, 0};
|
|
|
|
HALDEBUG(ah, HAL_DEBUG_BT_COEX, "(MCI) %s: Send WLAN_CAL_DONE 0x%X\n",
|
|
__func__, ahp->ah_mci_wlan_cal_done);
|
|
MCI_GPM_SET_CAL_TYPE(payload, MCI_GPM_WLAN_CAL_DONE);
|
|
payload[MCI_GPM_WLAN_CAL_W_SEQUENCE] = ahp->ah_mci_wlan_cal_done++;
|
|
ar9300_mci_send_message(ah, MCI_GPM, 0, payload, 16, AH_TRUE, AH_FALSE);
|
|
}
|
|
#endif /* ATH_SUPPORT_MCI */
|
|
|
|
|
|
if (!cal_done && !AR_SREV_SCORPION(ah) )
|
|
{
|
|
HALDEBUG(ah, HAL_DEBUG_CALIBRATE,
|
|
"%s: offset calibration failed to complete in 1ms; "
|
|
"noisy environment?\n", __func__);
|
|
return AH_FALSE;
|
|
}
|
|
|
|
#if 0
|
|
/* Beacon stuck fix, refer to EV 120056 */
|
|
if(IS_CHAN_2GHZ(chan) && AR_SREV_SCORPION(ah))
|
|
OS_REG_WRITE(ah, AR_PHY_TIMING5, OS_REG_READ(ah,AR_PHY_TIMING5) & ~AR_PHY_TIMING5_CYCPWR_THR1_ENABLE);
|
|
#endif
|
|
|
|
#if 0
|
|
/* Do PA Calibration */
|
|
if (AR_SREV_KITE(ah) && AR_SREV_KITE_11_OR_LATER(ah)) {
|
|
ar9285_pa_cal(ah);
|
|
}
|
|
#endif
|
|
|
|
#if ATH_SUPPORT_CAL_REUSE
|
|
if (ichan->one_time_txiqcal_done) {
|
|
ar9300_tx_iq_cal_apply(ah, ichan);
|
|
HALDEBUG(ah, HAL_DEBUG_FCS_RTT,
|
|
"(FCS) TXIQCAL applied - %d\n", ichan->channel);
|
|
}
|
|
#endif /* ATH_SUPPORT_CAL_REUSE */
|
|
|
|
#if ATH_SUPPORT_CAL_REUSE
|
|
if (cal_reuse_enable && ahp->tx_cl_cal_enable)
|
|
{
|
|
clc_success = (OS_REG_READ(ah, AR_PHY_AGC_CONTROL) &
|
|
AR_PHY_AGC_CONTROL_CLC_SUCCESS) ? 1 : 0;
|
|
|
|
if (ichan->one_time_txclcal_done)
|
|
{
|
|
/* reapply CL cal results */
|
|
for (ch_idx = 0; ch_idx < AR9300_MAX_CHAINS; ch_idx++) {
|
|
if ((ahp->ah_tx_cal_chainmask & (1 << ch_idx)) == 0) {
|
|
continue;
|
|
}
|
|
cl_tab_reg = BB_cl_tab_b[ch_idx];
|
|
for (j = 0; j < BB_cl_tab_entry; j++) {
|
|
OS_REG_WRITE(ah, cl_tab_reg, ichan->tx_clcal[ch_idx][j]);
|
|
cl_tab_reg += 4;;
|
|
}
|
|
}
|
|
HALDEBUG(ah, HAL_DEBUG_FCS_RTT,
|
|
"(FCS) TX CL CAL applied - %d\n", ichan->channel);
|
|
}
|
|
else if (is_cal_reusable && clc_success) {
|
|
/* save CL cal results */
|
|
for (ch_idx = 0; ch_idx < AR9300_MAX_CHAINS; ch_idx++) {
|
|
if ((ahp->ah_tx_cal_chainmask & (1 << ch_idx)) == 0) {
|
|
continue;
|
|
}
|
|
cl_tab_reg = BB_cl_tab_b[ch_idx];
|
|
for (j = 0; j < BB_cl_tab_entry; j++) {
|
|
ichan->tx_clcal[ch_idx][j] = OS_REG_READ(ah, cl_tab_reg);
|
|
cl_tab_reg += 4;
|
|
}
|
|
}
|
|
ichan->one_time_txclcal_done = AH_TRUE;
|
|
HALDEBUG(ah, HAL_DEBUG_FCS_RTT,
|
|
"(FCS) TX CL CAL saved - %d\n", ichan->channel);
|
|
}
|
|
}
|
|
#endif /* ATH_SUPPORT_CAL_REUSE */
|
|
|
|
/* Revert chainmasks to their original values before NF cal */
|
|
ar9300_init_chain_masks(ah, ahp->ah_rx_chainmask, ahp->ah_tx_chainmask);
|
|
|
|
#if !FIX_NOISE_FLOOR
|
|
/*
|
|
* Do NF calibration after DC offset and other CALs.
|
|
* Per system engineers, noise floor value can sometimes be 20 dB
|
|
* higher than normal value if DC offset and noise floor cal are
|
|
* triggered at the same time.
|
|
*/
|
|
OS_REG_WRITE(ah, AR_PHY_AGC_CONTROL,
|
|
OS_REG_READ(ah, AR_PHY_AGC_CONTROL) | AR_PHY_AGC_CONTROL_NF);
|
|
#endif
|
|
|
|
/* Initialize list pointers */
|
|
ahp->ah_cal_list = ahp->ah_cal_list_last = ahp->ah_cal_list_curr = AH_NULL;
|
|
|
|
/*
|
|
* Enable IQ, ADC Gain, ADC DC Offset Cals
|
|
*/
|
|
/* Setup all non-periodic, init time only calibrations */
|
|
/* XXX: Init DC Offset not working yet */
|
|
#ifdef not_yet
|
|
if (AH_TRUE == ar9300_is_cal_supp(ah, chan, ADC_DC_INIT_CAL)) {
|
|
INIT_CAL(&ahp->ah_adc_dc_cal_init_data);
|
|
INSERT_CAL(ahp, &ahp->ah_adc_dc_cal_init_data);
|
|
}
|
|
|
|
/* Initialize current pointer to first element in list */
|
|
ahp->ah_cal_list_curr = ahp->ah_cal_list;
|
|
|
|
if (ahp->ah_cal_list_curr) {
|
|
if (ar9300_run_init_cals(ah, 0) == AH_FALSE) {
|
|
return AH_FALSE;
|
|
}
|
|
}
|
|
#endif
|
|
/* end - Init time calibrations */
|
|
|
|
/* Do not do RX cal in case of offchan, or cal data already exists on same channel*/
|
|
if (ahp->ah_skip_rx_iq_cal) {
|
|
HALDEBUG(ah, HAL_DEBUG_CALIBRATE,
|
|
"Skip RX IQ Cal\n");
|
|
return AH_TRUE;
|
|
}
|
|
|
|
/* If Cals are supported, add them to list via INIT/INSERT_CAL */
|
|
if (AH_TRUE == ar9300_is_cal_supp(ah, chan, IQ_MISMATCH_CAL)) {
|
|
INIT_CAL(&ahp->ah_iq_cal_data);
|
|
INSERT_CAL(ahp, &ahp->ah_iq_cal_data);
|
|
HALDEBUG(ah, HAL_DEBUG_CALIBRATE,
|
|
"%s: enabling IQ Calibration.\n", __func__);
|
|
}
|
|
if (AH_TRUE == ar9300_is_cal_supp(ah, chan, TEMP_COMP_CAL)) {
|
|
INIT_CAL(&ahp->ah_temp_comp_cal_data);
|
|
INSERT_CAL(ahp, &ahp->ah_temp_comp_cal_data);
|
|
HALDEBUG(ah, HAL_DEBUG_CALIBRATE,
|
|
"%s: enabling Temperature Compensation Calibration.\n", __func__);
|
|
}
|
|
|
|
/* Initialize current pointer to first element in list */
|
|
ahp->ah_cal_list_curr = ahp->ah_cal_list;
|
|
|
|
/* Reset state within current cal */
|
|
if (ahp->ah_cal_list_curr) {
|
|
ar9300_reset_calibration(ah, ahp->ah_cal_list_curr);
|
|
}
|
|
|
|
/* Mark all calibrations on this channel as being invalid */
|
|
ichan->calValid = 0;
|
|
|
|
return AH_TRUE;
|
|
}
|
|
|
|
static inline HAL_BOOL
|
|
ar9300_init_cal(struct ath_hal *ah, struct ieee80211_channel *chan, HAL_BOOL skip_if_none, HAL_BOOL apply_last_iqcorr)
|
|
{
|
|
HAL_CHANNEL_INTERNAL *ichan = ath_hal_checkchannel(ah, chan);
|
|
HAL_BOOL do_rtt_cal = AH_TRUE;
|
|
HAL_BOOL enable_rtt = AH_FALSE;
|
|
|
|
HALASSERT(ichan);
|
|
|
|
return ar9300_init_cal_internal(ah, chan, ichan, enable_rtt, do_rtt_cal, skip_if_none, apply_last_iqcorr);
|
|
}
|
|
|
|
/* ar9300_reset_cal_valid
|
|
* Entry point for upper layers to restart current cal.
|
|
* Reset the calibration valid bit in channel.
|
|
*/
|
|
void
|
|
ar9300_reset_cal_valid(struct ath_hal *ah, const struct ieee80211_channel *chan,
|
|
HAL_BOOL *is_cal_done, u_int32_t cal_type)
|
|
{
|
|
struct ath_hal_9300 *ahp = AH9300(ah);
|
|
HAL_CHANNEL_INTERNAL *ichan = ath_hal_checkchannel(ah, chan);
|
|
HAL_CAL_LIST *curr_cal = ahp->ah_cal_list_curr;
|
|
|
|
*is_cal_done = AH_TRUE;
|
|
|
|
if (curr_cal == AH_NULL) {
|
|
return;
|
|
}
|
|
if (ichan == AH_NULL) {
|
|
HALDEBUG(ah, HAL_DEBUG_CALIBRATE,
|
|
"%s: invalid channel %u/0x%x; no mapping\n",
|
|
__func__, chan->ic_freq, chan->ic_flags);
|
|
return;
|
|
}
|
|
|
|
if (!(cal_type & IQ_MISMATCH_CAL)) {
|
|
*is_cal_done = AH_FALSE;
|
|
return;
|
|
}
|
|
|
|
/* Expected that this calibration has run before, post-reset.
|
|
* Current state should be done
|
|
*/
|
|
if (curr_cal->cal_state != CAL_DONE) {
|
|
HALDEBUG(ah, HAL_DEBUG_CALIBRATE,
|
|
"%s: Calibration state incorrect, %d\n",
|
|
__func__, curr_cal->cal_state);
|
|
return;
|
|
}
|
|
|
|
/* Verify Cal is supported on this channel */
|
|
if (ar9300_is_cal_supp(ah, chan, curr_cal->cal_data->cal_type) == AH_FALSE) {
|
|
return;
|
|
}
|
|
|
|
HALDEBUG(ah, HAL_DEBUG_CALIBRATE,
|
|
"%s: Resetting Cal %d state for channel %u/0x%x\n", __func__,
|
|
curr_cal->cal_data->cal_type, chan->ic_freq, chan->ic_flags);
|
|
|
|
/* Disable cal validity in channel */
|
|
ichan->calValid &= ~curr_cal->cal_data->cal_type;
|
|
curr_cal->cal_state = CAL_WAITING;
|
|
/* Indicate to upper layers that we need polling */
|
|
*is_cal_done = AH_FALSE;
|
|
}
|
|
|
|
static inline void
|
|
ar9300_set_dma(struct ath_hal *ah)
|
|
{
|
|
u_int32_t regval;
|
|
struct ath_hal_9300 *ahp = AH9300(ah);
|
|
struct ath_hal_private *ahpriv = AH_PRIVATE(ah);
|
|
HAL_CAPABILITIES *pCap = &ahpriv->ah_caps;
|
|
|
|
#if 0
|
|
/*
|
|
* set AHB_MODE not to do cacheline prefetches
|
|
*/
|
|
regval = OS_REG_READ(ah, AR_AHB_MODE);
|
|
OS_REG_WRITE(ah, AR_AHB_MODE, regval | AR_AHB_PREFETCH_RD_EN);
|
|
#endif
|
|
|
|
/*
|
|
* let mac dma reads be in 128 byte chunks
|
|
*/
|
|
regval = OS_REG_READ(ah, AR_TXCFG) & ~AR_TXCFG_DMASZ_MASK;
|
|
OS_REG_WRITE(ah, AR_TXCFG, regval | AR_TXCFG_DMASZ_128B);
|
|
|
|
/*
|
|
* Restore TX Trigger Level to its pre-reset value.
|
|
* The initial value depends on whether aggregation is enabled, and is
|
|
* adjusted whenever underruns are detected.
|
|
*/
|
|
/*
|
|
OS_REG_RMW_FIELD(ah, AR_TXCFG, AR_FTRIG, AH_PRIVATE(ah)->ah_tx_trig_level);
|
|
*/
|
|
/*
|
|
* Osprey 1.0 bug (EV 61936). Don't change trigger level from .ini default.
|
|
* Osprey 2.0 - hardware recommends using the default INI settings.
|
|
*/
|
|
#if 0
|
|
OS_REG_RMW_FIELD(ah, AR_TXCFG, AR_FTRIG, 0x3f);
|
|
#endif
|
|
/*
|
|
* let mac dma writes be in 128 byte chunks
|
|
*/
|
|
regval = OS_REG_READ(ah, AR_RXCFG) & ~AR_RXCFG_DMASZ_MASK;
|
|
OS_REG_WRITE(ah, AR_RXCFG, regval | AR_RXCFG_DMASZ_128B);
|
|
|
|
/*
|
|
* Setup receive FIFO threshold to hold off TX activities
|
|
*/
|
|
OS_REG_WRITE(ah, AR_RXFIFO_CFG, 0x200);
|
|
|
|
/*
|
|
* reduce the number of usable entries in PCU TXBUF to avoid
|
|
* wrap around bugs. (bug 20428)
|
|
*/
|
|
|
|
if (AR_SREV_WASP(ah) &&
|
|
(AH_PRIVATE((ah))->ah_macRev > AR_SREV_REVISION_WASP_12)) {
|
|
/* Wasp 1.3 fix for EV#85395 requires usable entries
|
|
* to be set to 0x500
|
|
*/
|
|
OS_REG_WRITE(ah, AR_PCU_TXBUF_CTRL, 0x500);
|
|
} else {
|
|
OS_REG_WRITE(ah, AR_PCU_TXBUF_CTRL, AR_PCU_TXBUF_CTRL_USABLE_SIZE);
|
|
}
|
|
|
|
/*
|
|
* Enable HPQ for UAPSD
|
|
*/
|
|
if (pCap->halHwUapsdTrig == AH_TRUE) {
|
|
/* Only enable this if HAL capabilities says it is OK */
|
|
if (AH_PRIVATE(ah)->ah_opmode == HAL_M_HOSTAP) {
|
|
OS_REG_WRITE(ah, AR_HP_Q_CONTROL,
|
|
AR_HPQ_ENABLE | AR_HPQ_UAPSD | AR_HPQ_UAPSD_TRIGGER_EN);
|
|
}
|
|
} else {
|
|
/* use default value from ini file - which disable HPQ queue usage */
|
|
}
|
|
|
|
/*
|
|
* set the transmit status ring
|
|
*/
|
|
ar9300_reset_tx_status_ring(ah);
|
|
|
|
/*
|
|
* set rxbp threshold. Must be non-zero for RX_EOL to occur.
|
|
* For Osprey 2.0+, keep the original thresholds
|
|
* otherwise performance is lost due to excessive RX EOL interrupts.
|
|
*/
|
|
OS_REG_RMW_FIELD(ah, AR_RXBP_THRESH, AR_RXBP_THRESH_HP, 0x1);
|
|
OS_REG_RMW_FIELD(ah, AR_RXBP_THRESH, AR_RXBP_THRESH_LP, 0x1);
|
|
|
|
/*
|
|
* set receive buffer size.
|
|
*/
|
|
if (ahp->rx_buf_size) {
|
|
OS_REG_WRITE(ah, AR_DATABUF, ahp->rx_buf_size);
|
|
}
|
|
}
|
|
|
|
static inline void
|
|
ar9300_init_bb(struct ath_hal *ah, struct ieee80211_channel *chan)
|
|
{
|
|
u_int32_t synth_delay;
|
|
|
|
/*
|
|
* Wait for the frequency synth to settle (synth goes on
|
|
* via AR_PHY_ACTIVE_EN). Read the phy active delay register.
|
|
* Value is in 100ns increments.
|
|
*/
|
|
synth_delay = OS_REG_READ(ah, AR_PHY_RX_DELAY) & AR_PHY_RX_DELAY_DELAY;
|
|
if (IEEE80211_IS_CHAN_CCK(chan)) {
|
|
synth_delay = (4 * synth_delay) / 22;
|
|
} else {
|
|
synth_delay /= 10;
|
|
}
|
|
|
|
/* Activate the PHY (includes baseband activate + synthesizer on) */
|
|
OS_REG_WRITE(ah, AR_PHY_ACTIVE, AR_PHY_ACTIVE_EN);
|
|
|
|
/*
|
|
* There is an issue if the AP starts the calibration before
|
|
* the base band timeout completes. This could result in the
|
|
* rx_clear false triggering. As a workaround we add delay an
|
|
* extra BASE_ACTIVATE_DELAY usecs to ensure this condition
|
|
* does not happen.
|
|
*/
|
|
OS_DELAY(synth_delay + BASE_ACTIVATE_DELAY);
|
|
}
|
|
|
|
static inline void
|
|
ar9300_init_interrupt_masks(struct ath_hal *ah, HAL_OPMODE opmode)
|
|
{
|
|
struct ath_hal_9300 *ahp = AH9300(ah);
|
|
u_int32_t msi_cfg = 0;
|
|
u_int32_t sync_en_def = AR9300_INTR_SYNC_DEFAULT;
|
|
|
|
/*
|
|
* Setup interrupt handling. Note that ar9300_reset_tx_queue
|
|
* manipulates the secondary IMR's as queues are enabled
|
|
* and disabled. This is done with RMW ops to insure the
|
|
* settings we make here are preserved.
|
|
*/
|
|
ahp->ah_mask_reg =
|
|
AR_IMR_TXERR | AR_IMR_TXURN |
|
|
AR_IMR_RXERR | AR_IMR_RXORN |
|
|
AR_IMR_BCNMISC;
|
|
|
|
if (ahp->ah_intr_mitigation_rx) {
|
|
/* enable interrupt mitigation for rx */
|
|
ahp->ah_mask_reg |= AR_IMR_RXINTM | AR_IMR_RXMINTR | AR_IMR_RXOK_HP;
|
|
msi_cfg |= AR_INTCFG_MSI_RXINTM | AR_INTCFG_MSI_RXMINTR;
|
|
} else {
|
|
ahp->ah_mask_reg |= AR_IMR_RXOK_LP | AR_IMR_RXOK_HP;
|
|
msi_cfg |= AR_INTCFG_MSI_RXOK;
|
|
}
|
|
if (ahp->ah_intr_mitigation_tx) {
|
|
/* enable interrupt mitigation for tx */
|
|
ahp->ah_mask_reg |= AR_IMR_TXINTM | AR_IMR_TXMINTR;
|
|
msi_cfg |= AR_INTCFG_MSI_TXINTM | AR_INTCFG_MSI_TXMINTR;
|
|
} else {
|
|
ahp->ah_mask_reg |= AR_IMR_TXOK;
|
|
msi_cfg |= AR_INTCFG_MSI_TXOK;
|
|
}
|
|
if (opmode == HAL_M_HOSTAP) {
|
|
ahp->ah_mask_reg |= AR_IMR_MIB;
|
|
}
|
|
|
|
OS_REG_WRITE(ah, AR_IMR, ahp->ah_mask_reg);
|
|
OS_REG_WRITE(ah, AR_IMR_S2, OS_REG_READ(ah, AR_IMR_S2) | AR_IMR_S2_GTT);
|
|
ahp->ah_mask2Reg = OS_REG_READ(ah, AR_IMR_S2);
|
|
|
|
if (ah->ah_config.ath_hal_enable_msi) {
|
|
/* Cache MSI register value */
|
|
ahp->ah_msi_reg = OS_REG_READ(ah, AR_HOSTIF_REG(ah, AR_PCIE_MSI));
|
|
ahp->ah_msi_reg |= AR_PCIE_MSI_HW_DBI_WR_EN;
|
|
if (AR_SREV_POSEIDON(ah)) {
|
|
ahp->ah_msi_reg &= AR_PCIE_MSI_HW_INT_PENDING_ADDR_MSI_64;
|
|
} else {
|
|
ahp->ah_msi_reg &= AR_PCIE_MSI_HW_INT_PENDING_ADDR;
|
|
}
|
|
/* Program MSI configuration */
|
|
OS_REG_WRITE(ah, AR_INTCFG, msi_cfg);
|
|
}
|
|
|
|
/*
|
|
* debug - enable to see all synchronous interrupts status
|
|
*/
|
|
/* Clear any pending sync cause interrupts */
|
|
OS_REG_WRITE(ah, AR_HOSTIF_REG(ah, AR_INTR_SYNC_CAUSE), 0xFFFFFFFF);
|
|
|
|
/* Allow host interface sync interrupt sources to set cause bit */
|
|
if (AR_SREV_POSEIDON(ah)) {
|
|
sync_en_def = AR9300_INTR_SYNC_DEF_NO_HOST1_PERR;
|
|
}
|
|
else if (AR_SREV_WASP(ah)) {
|
|
sync_en_def = AR9340_INTR_SYNC_DEFAULT;
|
|
}
|
|
OS_REG_WRITE(ah,
|
|
AR_HOSTIF_REG(ah, AR_INTR_SYNC_ENABLE), sync_en_def);
|
|
|
|
/* _Disable_ host interface sync interrupt when cause bits set */
|
|
OS_REG_WRITE(ah, AR_HOSTIF_REG(ah, AR_INTR_SYNC_MASK), 0);
|
|
|
|
OS_REG_WRITE(ah, AR_HOSTIF_REG(ah, AR_INTR_PRIO_ASYNC_ENABLE), 0);
|
|
OS_REG_WRITE(ah, AR_HOSTIF_REG(ah, AR_INTR_PRIO_ASYNC_MASK), 0);
|
|
OS_REG_WRITE(ah, AR_HOSTIF_REG(ah, AR_INTR_PRIO_SYNC_ENABLE), 0);
|
|
OS_REG_WRITE(ah, AR_HOSTIF_REG(ah, AR_INTR_PRIO_SYNC_MASK), 0);
|
|
}
|
|
|
|
static inline void
|
|
ar9300_init_qos(struct ath_hal *ah)
|
|
{
|
|
OS_REG_WRITE(ah, AR_MIC_QOS_CONTROL, 0x100aa); /* XXX magic */
|
|
OS_REG_WRITE(ah, AR_MIC_QOS_SELECT, 0x3210); /* XXX magic */
|
|
|
|
/* Turn on NOACK Support for QoS packets */
|
|
OS_REG_WRITE(ah, AR_QOS_NO_ACK,
|
|
SM(2, AR_QOS_NO_ACK_TWO_BIT) |
|
|
SM(5, AR_QOS_NO_ACK_BIT_OFF) |
|
|
SM(0, AR_QOS_NO_ACK_BYTE_OFF));
|
|
|
|
/*
|
|
* initialize TXOP for all TIDs
|
|
*/
|
|
OS_REG_WRITE(ah, AR_TXOP_X, AR_TXOP_X_VAL);
|
|
OS_REG_WRITE(ah, AR_TXOP_0_3, 0xFFFFFFFF);
|
|
OS_REG_WRITE(ah, AR_TXOP_4_7, 0xFFFFFFFF);
|
|
OS_REG_WRITE(ah, AR_TXOP_8_11, 0xFFFFFFFF);
|
|
OS_REG_WRITE(ah, AR_TXOP_12_15, 0xFFFFFFFF);
|
|
}
|
|
|
|
static inline void
|
|
ar9300_init_user_settings(struct ath_hal *ah)
|
|
{
|
|
struct ath_hal_9300 *ahp = AH9300(ah);
|
|
|
|
/* Restore user-specified settings */
|
|
HALDEBUG(ah, HAL_DEBUG_RESET,
|
|
"--AP %s ahp->ah_misc_mode 0x%x\n", __func__, ahp->ah_misc_mode);
|
|
if (ahp->ah_misc_mode != 0) {
|
|
OS_REG_WRITE(ah,
|
|
AR_PCU_MISC, OS_REG_READ(ah, AR_PCU_MISC) | ahp->ah_misc_mode);
|
|
}
|
|
if (ahp->ah_get_plcp_hdr) {
|
|
OS_REG_CLR_BIT(ah, AR_PCU_MISC, AR_PCU_SEL_EVM);
|
|
}
|
|
if (ahp->ah_slot_time != (u_int) -1) {
|
|
ar9300_set_slot_time(ah, ahp->ah_slot_time);
|
|
}
|
|
if (ahp->ah_ack_timeout != (u_int) -1) {
|
|
ar9300_set_ack_timeout(ah, ahp->ah_ack_timeout);
|
|
}
|
|
if (AH_PRIVATE(ah)->ah_diagreg != 0) {
|
|
OS_REG_SET_BIT(ah, AR_DIAG_SW, AH_PRIVATE(ah)->ah_diagreg);
|
|
}
|
|
if (ahp->ah_beacon_rssi_threshold != 0) {
|
|
ar9300_set_hw_beacon_rssi_threshold(ah, ahp->ah_beacon_rssi_threshold);
|
|
}
|
|
//#ifdef ATH_SUPPORT_DFS
|
|
if (ahp->ah_cac_quiet_enabled) {
|
|
ar9300_cac_tx_quiet(ah, 1);
|
|
}
|
|
//#endif /* ATH_SUPPORT_DFS */
|
|
}
|
|
|
|
int
|
|
ar9300_get_spur_info(struct ath_hal * ah, int *enable, int len, u_int16_t *freq)
|
|
{
|
|
// struct ath_hal_private *ap = AH_PRIVATE(ah);
|
|
int i, j;
|
|
|
|
for (i = 0; i < len; i++) {
|
|
freq[i] = 0;
|
|
}
|
|
|
|
*enable = ah->ah_config.ath_hal_spur_mode;
|
|
for (i = 0, j = 0; i < AR_EEPROM_MODAL_SPURS; i++) {
|
|
if (AH9300(ah)->ath_hal_spur_chans[i][0] != AR_NO_SPUR) {
|
|
freq[j++] = AH9300(ah)->ath_hal_spur_chans[i][0];
|
|
HALDEBUG(ah, HAL_DEBUG_ANI,
|
|
"1. get spur %d\n", AH9300(ah)->ath_hal_spur_chans[i][0]);
|
|
}
|
|
if (AH9300(ah)->ath_hal_spur_chans[i][1] != AR_NO_SPUR) {
|
|
freq[j++] = AH9300(ah)->ath_hal_spur_chans[i][1];
|
|
HALDEBUG(ah, HAL_DEBUG_ANI,
|
|
"2. get spur %d\n", AH9300(ah)->ath_hal_spur_chans[i][1]);
|
|
}
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
#define ATH_HAL_2GHZ_FREQ_MIN 20000
|
|
#define ATH_HAL_2GHZ_FREQ_MAX 29999
|
|
#define ATH_HAL_5GHZ_FREQ_MIN 50000
|
|
#define ATH_HAL_5GHZ_FREQ_MAX 59999
|
|
|
|
#if 0
|
|
int
|
|
ar9300_set_spur_info(struct ath_hal * ah, int enable, int len, u_int16_t *freq)
|
|
{
|
|
struct ath_hal_private *ap = AH_PRIVATE(ah);
|
|
int i, j, k;
|
|
|
|
ap->ah_config.ath_hal_spur_mode = enable;
|
|
|
|
if (ap->ah_config.ath_hal_spur_mode == SPUR_ENABLE_IOCTL) {
|
|
for (i = 0; i < AR_EEPROM_MODAL_SPURS; i++) {
|
|
AH9300(ah)->ath_hal_spur_chans[i][0] = AR_NO_SPUR;
|
|
AH9300(ah)->ath_hal_spur_chans[i][1] = AR_NO_SPUR;
|
|
}
|
|
for (i = 0, j = 0, k = 0; i < len; i++) {
|
|
if (freq[i] > ATH_HAL_2GHZ_FREQ_MIN &&
|
|
freq[i] < ATH_HAL_2GHZ_FREQ_MAX)
|
|
{
|
|
/* 2GHz Spur */
|
|
if (j < AR_EEPROM_MODAL_SPURS) {
|
|
AH9300(ah)->ath_hal_spur_chans[j++][1] = freq[i];
|
|
HALDEBUG(ah, HAL_DEBUG_ANI, "1 set spur %d\n", freq[i]);
|
|
}
|
|
} else if (freq[i] > ATH_HAL_5GHZ_FREQ_MIN &&
|
|
freq[i] < ATH_HAL_5GHZ_FREQ_MAX)
|
|
{
|
|
/* 5Ghz Spur */
|
|
if (k < AR_EEPROM_MODAL_SPURS) {
|
|
AH9300(ah)->ath_hal_spur_chans[k++][0] = freq[i];
|
|
HALDEBUG(ah, HAL_DEBUG_ANI, "2 set spur %d\n", freq[i]);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
#endif
|
|
|
|
#define ar9300_check_op_mode(_opmode) \
|
|
((_opmode == HAL_M_STA) || (_opmode == HAL_M_IBSS) ||\
|
|
(_opmode == HAL_M_HOSTAP) || (_opmode == HAL_M_MONITOR))
|
|
|
|
|
|
|
|
|
|
#ifndef ATH_NF_PER_CHAN
|
|
/*
|
|
* To fixed first reset noise floor value not correct issue
|
|
* For ART need it to fixed low rate sens too low issue
|
|
*/
|
|
static int
|
|
First_NFCal(struct ath_hal *ah, HAL_CHANNEL_INTERNAL *ichan,
|
|
int is_scan, struct ieee80211_channel *chan)
|
|
{
|
|
HAL_NFCAL_HIST_FULL *nfh;
|
|
int i, j, k;
|
|
int16_t nfarray[HAL_NUM_NF_READINGS] = {0};
|
|
int is_2g = 0;
|
|
int nf_hist_len;
|
|
int stats = 0;
|
|
|
|
int16_t nf_buf[HAL_NUM_NF_READINGS];
|
|
#define IS(_c, _f) (((_c)->channel_flags & _f) || 0)
|
|
|
|
|
|
if ((!is_scan) &&
|
|
chan->ic_freq == AH_PRIVATE(ah)->ah_curchan->ic_freq)
|
|
{
|
|
nfh = &AH_PRIVATE(ah)->nf_cal_hist;
|
|
} else {
|
|
nfh = (HAL_NFCAL_HIST_FULL *) &ichan->nf_cal_hist;
|
|
}
|
|
|
|
ar9300_start_nf_cal(ah);
|
|
for (j = 0; j < 10000; j++) {
|
|
if ((OS_REG_READ(ah, AR_PHY_AGC_CONTROL) & AR_PHY_AGC_CONTROL_NF) == 0){
|
|
break;
|
|
}
|
|
OS_DELAY(10);
|
|
}
|
|
if (j < 10000) {
|
|
is_2g = IEEE80211_IS_CHAN_2GHZ(chan);
|
|
ar9300_upload_noise_floor(ah, is_2g, nfarray);
|
|
|
|
if (is_scan) {
|
|
/*
|
|
* This channel's NF cal info is just a HAL_NFCAL_HIST_SMALL struct
|
|
* rather than a HAL_NFCAL_HIST_FULL struct.
|
|
* As long as we only use the first history element of nf_cal_buffer
|
|
* (nf_cal_buffer[0][0:HAL_NUM_NF_READINGS-1]), we can use
|
|
* HAL_NFCAL_HIST_SMALL and HAL_NFCAL_HIST_FULL interchangeably.
|
|
*/
|
|
nfh = (HAL_NFCAL_HIST_FULL *) &ichan->nf_cal_hist;
|
|
nf_hist_len = HAL_NF_CAL_HIST_LEN_SMALL;
|
|
} else {
|
|
nfh = &AH_PRIVATE(ah)->nf_cal_hist;
|
|
nf_hist_len = HAL_NF_CAL_HIST_LEN_FULL;
|
|
}
|
|
|
|
for (i = 0; i < HAL_NUM_NF_READINGS; i ++) {
|
|
for (k = 0; k < HAL_NF_CAL_HIST_LEN_FULL; k++) {
|
|
nfh->nf_cal_buffer[k][i] = nfarray[i];
|
|
}
|
|
nfh->base.priv_nf[i] = ar9300_limit_nf_range(ah,
|
|
ar9300_get_nf_hist_mid(ah, nfh, i, nf_hist_len));
|
|
}
|
|
|
|
|
|
//ar9300StoreNewNf(ah, ichan, is_scan);
|
|
|
|
/*
|
|
* See if the NF value from the old channel should be
|
|
* retained when switching to a new channel.
|
|
* TBD: this may need to be changed, as it wipes out the
|
|
* purpose of saving NF values for each channel.
|
|
*/
|
|
for (i = 0; i < HAL_NUM_NF_READINGS; i++)
|
|
{
|
|
if (IEEE80211_IS_CHAN_2GHZ(chan))
|
|
{
|
|
if (nfh->nf_cal_buffer[0][i] <
|
|
AR_PHY_CCA_MAX_GOOD_VAL_OSPREY_2GHZ)
|
|
{
|
|
ichan->nf_cal_hist.nf_cal_buffer[0][i] =
|
|
AH_PRIVATE(ah)->nf_cal_hist.nf_cal_buffer[0][i];
|
|
}
|
|
} else {
|
|
if (AR_SREV_AR9580(ah)) {
|
|
if (nfh->nf_cal_buffer[0][i] <
|
|
AR_PHY_CCA_NOM_VAL_PEACOCK_5GHZ)
|
|
{
|
|
ichan->nf_cal_hist.nf_cal_buffer[0][i] =
|
|
AH_PRIVATE(ah)->nf_cal_hist.nf_cal_buffer[0][i];
|
|
}
|
|
} else {
|
|
if (nfh->nf_cal_buffer[0][i] <
|
|
AR_PHY_CCA_NOM_VAL_OSPREY_5GHZ)
|
|
{
|
|
ichan->nf_cal_hist.nf_cal_buffer[0][i] =
|
|
AH_PRIVATE(ah)->nf_cal_hist.nf_cal_buffer[0][i];
|
|
}
|
|
}
|
|
}
|
|
}
|
|
/*
|
|
* Copy the channel's NF buffer, which may have been modified
|
|
* just above here, to the full NF history buffer.
|
|
*/
|
|
ar9300_reset_nf_hist_buff(ah, ichan);
|
|
ar9300_get_nf_hist_base(ah, ichan, is_scan, nf_buf);
|
|
ar9300_load_nf(ah, nf_buf);
|
|
stats = 0;
|
|
} else {
|
|
stats = 1;
|
|
}
|
|
#undef IS
|
|
return stats;
|
|
}
|
|
#endif
|
|
|
|
|
|
/*
|
|
* Places the device in and out of reset and then places sane
|
|
* values in the registers based on EEPROM config, initialization
|
|
* vectors (as determined by the mode), and station configuration
|
|
*
|
|
* b_channel_change is used to preserve DMA/PCU registers across
|
|
* a HW Reset during channel change.
|
|
*/
|
|
HAL_BOOL
|
|
ar9300_reset(struct ath_hal *ah, HAL_OPMODE opmode, struct ieee80211_channel *chan,
|
|
HAL_HT_MACMODE macmode, u_int8_t txchainmask, u_int8_t rxchainmask,
|
|
HAL_HT_EXTPROTSPACING extprotspacing, HAL_BOOL b_channel_change,
|
|
HAL_STATUS *status, int is_scan)
|
|
{
|
|
#define FAIL(_code) do { ecode = _code; goto bad; } while (0)
|
|
u_int32_t save_led_state;
|
|
struct ath_hal_9300 *ahp = AH9300(ah);
|
|
struct ath_hal_private *ap = AH_PRIVATE(ah);
|
|
HAL_CHANNEL_INTERNAL *ichan;
|
|
//const struct ieee80211_channel *curchan = ap->ah_curchan;
|
|
#if ATH_SUPPORT_MCI
|
|
HAL_BOOL save_full_sleep = ahp->ah_chip_full_sleep;
|
|
#endif
|
|
u_int32_t save_def_antenna;
|
|
u_int32_t mac_sta_id1;
|
|
HAL_STATUS ecode;
|
|
int i, rx_chainmask;
|
|
int nf_hist_buff_reset = 0;
|
|
int16_t nf_buf[HAL_NUM_NF_READINGS];
|
|
#ifdef ATH_FORCE_PPM
|
|
u_int32_t save_force_val, tmp_reg;
|
|
#endif
|
|
u_int8_t clk_25mhz = AH9300(ah)->clk_25mhz;
|
|
HAL_BOOL stopped, cal_ret;
|
|
HAL_BOOL apply_last_iqcorr = AH_FALSE;
|
|
|
|
|
|
if (OS_REG_READ(ah, AR_IER) == AR_IER_ENABLE) {
|
|
HALDEBUG(AH_NULL, HAL_DEBUG_UNMASKABLE, "** Reset called with WLAN "
|
|
"interrupt enabled %08x **\n", ar9300_get_interrupts(ah));
|
|
}
|
|
|
|
/*
|
|
* Set the status to "ok" by default to cover the cases
|
|
* where we return false without going to "bad"
|
|
*/
|
|
HALASSERT(status);
|
|
*status = HAL_OK;
|
|
if ((ah->ah_config.ath_hal_sta_update_tx_pwr_enable)) {
|
|
AH9300(ah)->green_tx_status = HAL_RSSI_TX_POWER_NONE;
|
|
}
|
|
|
|
#if ATH_SUPPORT_MCI
|
|
if (AH_PRIVATE(ah)->ah_caps.halMciSupport &&
|
|
(AR_SREV_JUPITER_20_OR_LATER(ah) || AR_SREV_APHRODITE(ah)))
|
|
{
|
|
ar9300_mci_2g5g_changed(ah, IEEE80211_IS_CHAN_2GHZ(chan));
|
|
}
|
|
#endif
|
|
|
|
ahp->ah_ext_prot_spacing = extprotspacing;
|
|
ahp->ah_tx_chainmask = txchainmask & ap->ah_caps.halTxChainMask;
|
|
ahp->ah_rx_chainmask = rxchainmask & ap->ah_caps.halRxChainMask;
|
|
ahp->ah_tx_cal_chainmask = ap->ah_caps.halTxChainMask;
|
|
ahp->ah_rx_cal_chainmask = ap->ah_caps.halRxChainMask;
|
|
|
|
/*
|
|
* Keep the previous optinal txchainmask value
|
|
*/
|
|
|
|
HALASSERT(ar9300_check_op_mode(opmode));
|
|
|
|
OS_MARK(ah, AH_MARK_RESET, b_channel_change);
|
|
|
|
/*
|
|
* Map public channel to private.
|
|
*/
|
|
ichan = ar9300_check_chan(ah, chan);
|
|
if (ichan == AH_NULL) {
|
|
HALDEBUG(ah, HAL_DEBUG_CHANNEL,
|
|
"%s: invalid channel %u/0x%x; no mapping\n",
|
|
__func__, chan->ic_freq, chan->ic_flags);
|
|
FAIL(HAL_EINVAL);
|
|
}
|
|
|
|
ichan->paprd_table_write_done = 0; /* Clear PAPRD table write flag */
|
|
#if 0
|
|
chan->paprd_table_write_done = 0; /* Clear PAPRD table write flag */
|
|
#endif
|
|
|
|
if (ar9300_get_power_mode(ah) != HAL_PM_FULL_SLEEP) {
|
|
/* Need to stop RX DMA before reset otherwise chip might hang */
|
|
stopped = ar9300_set_rx_abort(ah, AH_TRUE); /* abort and disable PCU */
|
|
ar9300_set_rx_filter(ah, 0);
|
|
stopped &= ar9300_stop_dma_receive(ah, 0); /* stop and disable RX DMA */
|
|
if (!stopped) {
|
|
/*
|
|
* During the transition from full sleep to reset,
|
|
* recv DMA regs are not available to be read
|
|
*/
|
|
HALDEBUG(ah, HAL_DEBUG_UNMASKABLE,
|
|
"%s[%d]: ar9300_stop_dma_receive failed\n", __func__, __LINE__);
|
|
b_channel_change = AH_FALSE;
|
|
}
|
|
} else {
|
|
HALDEBUG(ah, HAL_DEBUG_UNMASKABLE,
|
|
"%s[%d]: Chip is already in full sleep\n", __func__, __LINE__);
|
|
}
|
|
|
|
#if ATH_SUPPORT_MCI
|
|
if ((AH_PRIVATE(ah)->ah_caps.halMciSupport) &&
|
|
(ahp->ah_mci_bt_state == MCI_BT_CAL_START))
|
|
{
|
|
u_int32_t payload[4] = {0, 0, 0, 0};
|
|
|
|
HALDEBUG(ah, HAL_DEBUG_BT_COEX,
|
|
"(MCI) %s: Stop rx for BT cal.\n", __func__);
|
|
ahp->ah_mci_bt_state = MCI_BT_CAL;
|
|
|
|
/*
|
|
* MCIFIX: disable mci interrupt here. This is to avoid SW_MSG_DONE or
|
|
* RX_MSG bits to trigger MCI_INT and lead to mci_intr reentry.
|
|
*/
|
|
ar9300_mci_disable_interrupt(ah);
|
|
|
|
HALDEBUG(ah, HAL_DEBUG_BT_COEX,
|
|
"(MCI) %s: Send WLAN_CAL_GRANT\n", __func__);
|
|
MCI_GPM_SET_CAL_TYPE(payload, MCI_GPM_WLAN_CAL_GRANT);
|
|
ar9300_mci_send_message(ah, MCI_GPM, 0, payload, 16, AH_TRUE, AH_FALSE);
|
|
|
|
/* Wait BT calibration to be completed for 25ms */
|
|
HALDEBUG(ah, HAL_DEBUG_BT_COEX,
|
|
"(MCI) %s: BT is calibrating.\n", __func__);
|
|
if (ar9300_mci_wait_for_gpm(ah, MCI_GPM_BT_CAL_DONE, 0, 25000)) {
|
|
HALDEBUG(ah, HAL_DEBUG_BT_COEX,
|
|
"(MCI) %s: Got BT_CAL_DONE.\n", __func__);
|
|
}
|
|
else {
|
|
HALDEBUG(ah, HAL_DEBUG_BT_COEX,
|
|
"(MCI) %s: ### BT cal takes too long. Force bt_state to be bt_awake.\n",
|
|
__func__);
|
|
}
|
|
ahp->ah_mci_bt_state = MCI_BT_AWAKE;
|
|
/* MCIFIX: enable mci interrupt here */
|
|
ar9300_mci_enable_interrupt(ah);
|
|
|
|
return AH_TRUE;
|
|
}
|
|
#endif
|
|
|
|
/* Bring out of sleep mode */
|
|
if (!ar9300_set_power_mode(ah, HAL_PM_AWAKE, AH_TRUE)) {
|
|
*status = HAL_INV_PMODE;
|
|
return AH_FALSE;
|
|
}
|
|
|
|
/* Check the Rx mitigation config again, it might have changed
|
|
* during attach in ath_vap_attach.
|
|
*/
|
|
if (ah->ah_config.ath_hal_intr_mitigation_rx != 0) {
|
|
ahp->ah_intr_mitigation_rx = AH_TRUE;
|
|
} else {
|
|
ahp->ah_intr_mitigation_rx = AH_FALSE;
|
|
}
|
|
|
|
/*
|
|
* XXX TODO FreeBSD:
|
|
*
|
|
* This is painful because we don't have a non-const channel pointer
|
|
* at this stage.
|
|
*
|
|
* Make sure this gets fixed!
|
|
*/
|
|
#if 0
|
|
/* Get the value from the previous NF cal and update history buffer */
|
|
if (curchan && (ahp->ah_chip_full_sleep != AH_TRUE)) {
|
|
|
|
if(ahp->ah_chip_reset_done){
|
|
ahp->ah_chip_reset_done = 0;
|
|
} else {
|
|
/*
|
|
* is_scan controls updating NF for home channel or off channel.
|
|
* Home -> Off, update home channel
|
|
* Off -> Home, update off channel
|
|
* Home -> Home, uppdate home channel
|
|
*/
|
|
if (ap->ah_curchan->channel != chan->channel)
|
|
ar9300_store_new_nf(ah, curchan, !is_scan);
|
|
else
|
|
ar9300_store_new_nf(ah, curchan, is_scan);
|
|
}
|
|
}
|
|
#endif
|
|
|
|
/*
|
|
* Account for the effect of being in either the 2 GHz or 5 GHz band
|
|
* on the nominal, max allowable, and min allowable noise floor values.
|
|
*/
|
|
AH9300(ah)->nfp = IS_CHAN_2GHZ(ichan) ? &ahp->nf_2GHz : &ahp->nf_5GHz;
|
|
|
|
/*
|
|
* XXX FreeBSD For now, don't apply the last IQ correction.
|
|
*
|
|
* This should be done when scorpion is enabled on FreeBSD; just be
|
|
* sure to fix this channel match code so it uses net80211 flags
|
|
* instead.
|
|
*/
|
|
#if 0
|
|
if (AR_SREV_SCORPION(ah) && curchan && (chan->channel == curchan->channel) &&
|
|
((chan->channel_flags & (CHANNEL_ALL|CHANNEL_HALF|CHANNEL_QUARTER)) ==
|
|
(curchan->channel_flags &
|
|
(CHANNEL_ALL | CHANNEL_HALF | CHANNEL_QUARTER)))) {
|
|
apply_last_iqcorr = AH_TRUE;
|
|
}
|
|
#endif
|
|
apply_last_iqcorr = AH_FALSE;
|
|
|
|
|
|
#ifndef ATH_NF_PER_CHAN
|
|
/*
|
|
* If there's only one full-size home-channel NF history buffer
|
|
* rather than a full-size NF history buffer per channel, decide
|
|
* whether to (re)initialize the home-channel NF buffer.
|
|
* If this is just a channel change for a scan, or if the channel
|
|
* is not being changed, don't mess up the home channel NF history
|
|
* buffer with NF values from this scanned channel. If we're
|
|
* changing the home channel to a new channel, reset the home-channel
|
|
* NF history buffer with the most accurate NF known for the new channel.
|
|
*/
|
|
if (!is_scan && (!ap->ah_curchan ||
|
|
ap->ah_curchan->ic_freq != chan->ic_freq)) // ||
|
|
// ap->ah_curchan->channel_flags != chan->channel_flags))
|
|
{
|
|
nf_hist_buff_reset = 1;
|
|
ar9300_reset_nf_hist_buff(ah, ichan);
|
|
}
|
|
#endif
|
|
/*
|
|
* In case of
|
|
* - offchan scan, or
|
|
* - same channel and RX IQ Cal already available
|
|
* disable RX IQ Cal.
|
|
*/
|
|
if (is_scan) {
|
|
ahp->ah_skip_rx_iq_cal = AH_TRUE;
|
|
HALDEBUG(ah, HAL_DEBUG_CALIBRATE,
|
|
"Skip RX IQ Cal due to scanning\n");
|
|
} else {
|
|
#if 0
|
|
/* XXX FreeBSD: always just do the RX IQ cal */
|
|
/* XXX I think it's just going to speed things up; I don't think it's to avoid chan bugs */
|
|
if (ahp->ah_rx_cal_complete &&
|
|
ahp->ah_rx_cal_chan == ichan->channel &&
|
|
ahp->ah_rx_cal_chan_flag == chan->channel_flags) {
|
|
ahp->ah_skip_rx_iq_cal = AH_TRUE;
|
|
HALDEBUG(ah, HAL_DEBUG_CALIBRATE,
|
|
"Skip RX IQ Cal due to same channel with completed RX IQ Cal\n");
|
|
} else
|
|
#endif
|
|
ahp->ah_skip_rx_iq_cal = AH_FALSE;
|
|
}
|
|
|
|
/* FreeBSD: clear the channel survey data */
|
|
ath_hal_survey_clear(ah);
|
|
|
|
/*
|
|
* Fast channel change (Change synthesizer based on channel freq
|
|
* without resetting chip)
|
|
* Don't do it when
|
|
* - Flag is not set
|
|
* - Chip is just coming out of full sleep
|
|
* - Channel to be set is same as current channel
|
|
* - Channel flags are different, like when moving from 2GHz to 5GHz
|
|
* channels
|
|
* - Merlin: Switching in/out of fast clock enabled channels
|
|
* (not currently coded, since fast clock is enabled
|
|
* across the 5GHz band
|
|
* and we already do a full reset when switching in/out
|
|
* of 5GHz channels)
|
|
*/
|
|
#if 0
|
|
if (b_channel_change &&
|
|
(ahp->ah_chip_full_sleep != AH_TRUE) &&
|
|
(AH_PRIVATE(ah)->ah_curchan != AH_NULL) &&
|
|
((chan->channel != AH_PRIVATE(ah)->ah_curchan->channel) &&
|
|
(((CHANNEL_ALL|CHANNEL_HALF|CHANNEL_QUARTER) & chan->channel_flags) ==
|
|
((CHANNEL_ALL|CHANNEL_HALF|CHANNEL_QUARTER) & AH_PRIVATE(ah)->ah_curchan->channel_flags))))
|
|
{
|
|
if (ar9300_channel_change(ah, chan, ichan, macmode)) {
|
|
chan->channel_flags = ichan->channel_flags;
|
|
chan->priv_flags = ichan->priv_flags;
|
|
AH_PRIVATE(ah)->ah_curchan->ah_channel_time = 0;
|
|
AH_PRIVATE(ah)->ah_curchan->ah_tsf_last = ar9300_get_tsf64(ah);
|
|
|
|
/*
|
|
* Load the NF from history buffer of the current channel.
|
|
* NF is slow time-variant, so it is OK to use a historical value.
|
|
*/
|
|
ar9300_get_nf_hist_base(ah,
|
|
AH_PRIVATE(ah)->ah_curchan, is_scan, nf_buf);
|
|
ar9300_load_nf(ah, nf_buf);
|
|
|
|
/* start NF calibration, without updating BB NF register*/
|
|
ar9300_start_nf_cal(ah);
|
|
|
|
/*
|
|
* If channel_change completed and DMA was stopped
|
|
* successfully - skip the rest of reset
|
|
*/
|
|
if (AH9300(ah)->ah_dma_stuck != AH_TRUE) {
|
|
ar9300_disable_pll_lock_detect(ah);
|
|
#if ATH_SUPPORT_MCI
|
|
if (AH_PRIVATE(ah)->ah_caps.halMciSupport && ahp->ah_mci_ready)
|
|
{
|
|
ar9300_mci_2g5g_switch(ah, AH_TRUE);
|
|
}
|
|
#endif
|
|
return HAL_OK;
|
|
}
|
|
}
|
|
}
|
|
#endif /* #if 0 */
|
|
|
|
#if ATH_SUPPORT_MCI
|
|
if (AH_PRIVATE(ah)->ah_caps.halMciSupport) {
|
|
ar9300_mci_disable_interrupt(ah);
|
|
if (ahp->ah_mci_ready && !save_full_sleep) {
|
|
ar9300_mci_mute_bt(ah);
|
|
OS_DELAY(20);
|
|
OS_REG_WRITE(ah, AR_BTCOEX_CTRL, 0);
|
|
}
|
|
|
|
ahp->ah_mci_bt_state = MCI_BT_SLEEP;
|
|
ahp->ah_mci_ready = AH_FALSE;
|
|
}
|
|
#endif
|
|
|
|
AH9300(ah)->ah_dma_stuck = AH_FALSE;
|
|
#ifdef ATH_FORCE_PPM
|
|
/* Preserve force ppm state */
|
|
save_force_val =
|
|
OS_REG_READ(ah, AR_PHY_TIMING2) &
|
|
(AR_PHY_TIMING2_USE_FORCE | AR_PHY_TIMING2_FORCE_VAL);
|
|
#endif
|
|
/*
|
|
* Preserve the antenna on a channel change
|
|
*/
|
|
save_def_antenna = OS_REG_READ(ah, AR_DEF_ANTENNA);
|
|
if (0 == ahp->ah_smartantenna_enable )
|
|
{
|
|
if (save_def_antenna == 0) {
|
|
save_def_antenna = 1;
|
|
}
|
|
}
|
|
|
|
/* Save hardware flag before chip reset clears the register */
|
|
mac_sta_id1 = OS_REG_READ(ah, AR_STA_ID1) & AR_STA_ID1_BASE_RATE_11B;
|
|
|
|
/* Save led state from pci config register */
|
|
save_led_state = OS_REG_READ(ah, AR_CFG_LED) &
|
|
(AR_CFG_LED_ASSOC_CTL | AR_CFG_LED_MODE_SEL |
|
|
AR_CFG_LED_BLINK_THRESH_SEL | AR_CFG_LED_BLINK_SLOW);
|
|
|
|
/* Mark PHY inactive prior to reset, to be undone in ar9300_init_bb () */
|
|
ar9300_mark_phy_inactive(ah);
|
|
|
|
if (!ar9300_chip_reset(ah, chan)) {
|
|
HALDEBUG(ah, HAL_DEBUG_RESET, "%s: chip reset failed\n", __func__);
|
|
FAIL(HAL_EIO);
|
|
}
|
|
|
|
OS_MARK(ah, AH_MARK_RESET_LINE, __LINE__);
|
|
|
|
|
|
/* Disable JTAG */
|
|
OS_REG_SET_BIT(ah,
|
|
AR_HOSTIF_REG(ah, AR_GPIO_INPUT_EN_VAL), AR_GPIO_JTAG_DISABLE);
|
|
|
|
/*
|
|
* Note that ar9300_init_chain_masks() is called from within
|
|
* ar9300_process_ini() to ensure the swap bit is set before
|
|
* the pdadc table is written.
|
|
*/
|
|
ecode = ar9300_process_ini(ah, chan, ichan, macmode);
|
|
if (ecode != HAL_OK) {
|
|
goto bad;
|
|
}
|
|
|
|
/*
|
|
* Configuring WMAC PLL values for 25/40 MHz
|
|
*/
|
|
if(AR_SREV_WASP(ah) || AR_SREV_HONEYBEE(ah) || AR_SREV_SCORPION(ah) ) {
|
|
if(clk_25mhz) {
|
|
OS_REG_WRITE(ah, AR_RTC_DERIVED_RTC_CLK, (0x17c << 1)); // 32KHz sleep clk
|
|
} else {
|
|
OS_REG_WRITE(ah, AR_RTC_DERIVED_RTC_CLK, (0x261 << 1)); // 32KHz sleep clk
|
|
}
|
|
OS_DELAY(100);
|
|
}
|
|
|
|
ahp->ah_immunity_on = AH_FALSE;
|
|
|
|
if (AR_SREV_JUPITER(ah) || AR_SREV_APHRODITE(ah)) {
|
|
ahp->tx_iq_cal_enable = OS_REG_READ_FIELD(ah,
|
|
AR_PHY_TX_IQCAL_CONTROL_0(ah),
|
|
AR_PHY_TX_IQCAL_CONTROL_0_ENABLE_TXIQ_CAL) ?
|
|
1 : 0;
|
|
}
|
|
ahp->tx_cl_cal_enable = (OS_REG_READ(ah, AR_PHY_CL_CAL_CTL) &
|
|
AR_PHY_CL_CAL_ENABLE) ? 1 : 0;
|
|
|
|
/* For devices with full HW RIFS Rx support (Sowl/Howl/Merlin, etc),
|
|
* restore register settings from prior to reset.
|
|
*/
|
|
if ((AH_PRIVATE(ah)->ah_curchan != AH_NULL) &&
|
|
(ar9300_get_capability(ah, HAL_CAP_LDPCWAR, 0, AH_NULL) == HAL_OK))
|
|
{
|
|
/* Re-program RIFS Rx policy after reset */
|
|
ar9300_set_rifs_delay(ah, ahp->ah_rifs_enabled);
|
|
}
|
|
|
|
#if ATH_SUPPORT_MCI
|
|
if (AH_PRIVATE(ah)->ah_caps.halMciSupport) {
|
|
ar9300_mci_reset(ah, AH_FALSE, IS_CHAN_2GHZ(ichan), save_full_sleep);
|
|
}
|
|
#endif
|
|
|
|
/* Initialize Management Frame Protection */
|
|
ar9300_init_mfp(ah);
|
|
|
|
ahp->ah_immunity_vals[0] = OS_REG_READ_FIELD(ah, AR_PHY_SFCORR_LOW,
|
|
AR_PHY_SFCORR_LOW_M1_THRESH_LOW);
|
|
ahp->ah_immunity_vals[1] = OS_REG_READ_FIELD(ah, AR_PHY_SFCORR_LOW,
|
|
AR_PHY_SFCORR_LOW_M2_THRESH_LOW);
|
|
ahp->ah_immunity_vals[2] = OS_REG_READ_FIELD(ah, AR_PHY_SFCORR,
|
|
AR_PHY_SFCORR_M1_THRESH);
|
|
ahp->ah_immunity_vals[3] = OS_REG_READ_FIELD(ah, AR_PHY_SFCORR,
|
|
AR_PHY_SFCORR_M2_THRESH);
|
|
ahp->ah_immunity_vals[4] = OS_REG_READ_FIELD(ah, AR_PHY_SFCORR,
|
|
AR_PHY_SFCORR_M2COUNT_THR);
|
|
ahp->ah_immunity_vals[5] = OS_REG_READ_FIELD(ah, AR_PHY_SFCORR_LOW,
|
|
AR_PHY_SFCORR_LOW_M2COUNT_THR_LOW);
|
|
|
|
/* Write delta slope for OFDM enabled modes (A, G, Turbo) */
|
|
if (IEEE80211_IS_CHAN_OFDM(chan) || IEEE80211_IS_CHAN_HT(chan)) {
|
|
ar9300_set_delta_slope(ah, chan);
|
|
}
|
|
|
|
ar9300_spur_mitigate(ah, chan);
|
|
if (!ar9300_eeprom_set_board_values(ah, chan)) {
|
|
HALDEBUG(ah, HAL_DEBUG_EEPROM,
|
|
"%s: error setting board options\n", __func__);
|
|
FAIL(HAL_EIO);
|
|
}
|
|
|
|
#ifdef ATH_HAL_WAR_REG16284_APH128
|
|
/* temp work around, will be removed. */
|
|
if (AR_SREV_WASP(ah)) {
|
|
OS_REG_WRITE(ah, 0x16284, 0x1553e000);
|
|
}
|
|
#endif
|
|
|
|
OS_MARK(ah, AH_MARK_RESET_LINE, __LINE__);
|
|
|
|
OS_REG_WRITE(ah, AR_STA_ID0, LE_READ_4(ahp->ah_macaddr));
|
|
OS_REG_WRITE(ah, AR_STA_ID1, LE_READ_2(ahp->ah_macaddr + 4)
|
|
| mac_sta_id1
|
|
| AR_STA_ID1_RTS_USE_DEF
|
|
| (ah->ah_config.ath_hal_6mb_ack ? AR_STA_ID1_ACKCTS_6MB : 0)
|
|
| ahp->ah_sta_id1_defaults
|
|
);
|
|
ar9300_set_operating_mode(ah, opmode);
|
|
|
|
/* Set Venice BSSID mask according to current state */
|
|
OS_REG_WRITE(ah, AR_BSSMSKL, LE_READ_4(ahp->ah_bssid_mask));
|
|
OS_REG_WRITE(ah, AR_BSSMSKU, LE_READ_2(ahp->ah_bssid_mask + 4));
|
|
|
|
/* Restore previous antenna */
|
|
OS_REG_WRITE(ah, AR_DEF_ANTENNA, save_def_antenna);
|
|
#ifdef ATH_FORCE_PPM
|
|
/* Restore force ppm state */
|
|
tmp_reg = OS_REG_READ(ah, AR_PHY_TIMING2) &
|
|
~(AR_PHY_TIMING2_USE_FORCE | AR_PHY_TIMING2_FORCE_VAL);
|
|
OS_REG_WRITE(ah, AR_PHY_TIMING2, tmp_reg | save_force_val);
|
|
#endif
|
|
|
|
/* then our BSSID and assocID */
|
|
OS_REG_WRITE(ah, AR_BSS_ID0, LE_READ_4(ahp->ah_bssid));
|
|
OS_REG_WRITE(ah, AR_BSS_ID1,
|
|
LE_READ_2(ahp->ah_bssid + 4) |
|
|
((ahp->ah_assoc_id & 0x3fff) << AR_BSS_ID1_AID_S));
|
|
|
|
OS_REG_WRITE(ah, AR_ISR, ~0); /* cleared on write */
|
|
|
|
OS_REG_RMW_FIELD(ah, AR_RSSI_THR, AR_RSSI_THR_BM_THR, INIT_RSSI_THR);
|
|
|
|
/* HW beacon processing */
|
|
/*
|
|
* XXX what happens if I just leave filter_interval=0?
|
|
* it stays disabled?
|
|
*/
|
|
OS_REG_RMW_FIELD(ah, AR_RSSI_THR, AR_RSSI_BCN_WEIGHT,
|
|
INIT_RSSI_BEACON_WEIGHT);
|
|
OS_REG_SET_BIT(ah, AR_HWBCNPROC1, AR_HWBCNPROC1_CRC_ENABLE |
|
|
AR_HWBCNPROC1_EXCLUDE_TIM_ELM);
|
|
if (ah->ah_config.ath_hal_beacon_filter_interval) {
|
|
OS_REG_RMW_FIELD(ah, AR_HWBCNPROC2, AR_HWBCNPROC2_FILTER_INTERVAL,
|
|
ah->ah_config.ath_hal_beacon_filter_interval);
|
|
OS_REG_SET_BIT(ah, AR_HWBCNPROC2,
|
|
AR_HWBCNPROC2_FILTER_INTERVAL_ENABLE);
|
|
}
|
|
|
|
|
|
/*
|
|
* Set Channel now modifies bank 6 parameters for FOWL workaround
|
|
* to force rf_pwd_icsyndiv bias current as function of synth
|
|
* frequency.Thus must be called after ar9300_process_ini() to ensure
|
|
* analog register cache is valid.
|
|
*/
|
|
if (!ahp->ah_rf_hal.set_channel(ah, chan)) {
|
|
FAIL(HAL_EIO);
|
|
}
|
|
|
|
|
|
OS_MARK(ah, AH_MARK_RESET_LINE, __LINE__);
|
|
|
|
/* Set 1:1 QCU to DCU mapping for all queues */
|
|
for (i = 0; i < AR_NUM_DCU; i++) {
|
|
OS_REG_WRITE(ah, AR_DQCUMASK(i), 1 << i);
|
|
}
|
|
|
|
ahp->ah_intr_txqs = 0;
|
|
for (i = 0; i < AH_PRIVATE(ah)->ah_caps.halTotalQueues; i++) {
|
|
ar9300_reset_tx_queue(ah, i);
|
|
}
|
|
|
|
ar9300_init_interrupt_masks(ah, opmode);
|
|
|
|
/* Reset ier reference count to disabled */
|
|
// OS_ATOMIC_SET(&ahp->ah_ier_ref_count, 1);
|
|
if (ath_hal_isrfkillenabled(ah)) {
|
|
ar9300_enable_rf_kill(ah);
|
|
}
|
|
|
|
/* must be called AFTER ini is processed */
|
|
ar9300_ani_init_defaults(ah, macmode);
|
|
|
|
ar9300_init_qos(ah);
|
|
|
|
ar9300_init_user_settings(ah);
|
|
|
|
|
|
AH_PRIVATE(ah)->ah_opmode = opmode; /* record operating mode */
|
|
|
|
OS_MARK(ah, AH_MARK_RESET_DONE, 0);
|
|
|
|
/*
|
|
* disable seq number generation in hw
|
|
*/
|
|
OS_REG_WRITE(ah, AR_STA_ID1,
|
|
OS_REG_READ(ah, AR_STA_ID1) | AR_STA_ID1_PRESERVE_SEQNUM);
|
|
|
|
ar9300_set_dma(ah);
|
|
|
|
/*
|
|
* program OBS bus to see MAC interrupts
|
|
*/
|
|
#if ATH_SUPPORT_MCI
|
|
if (!AH_PRIVATE(ah)->ah_caps.halMciSupport) {
|
|
OS_REG_WRITE(ah, AR_HOSTIF_REG(ah, AR_OBS), 8);
|
|
}
|
|
#else
|
|
OS_REG_WRITE(ah, AR_HOSTIF_REG(ah, AR_OBS), 8);
|
|
#endif
|
|
|
|
|
|
/* enabling AR_GTTM_IGNORE_IDLE in GTTM register so that
|
|
GTT timer will not increment if the channel idle indicates
|
|
the air is busy or NAV is still counting down */
|
|
OS_REG_WRITE(ah, AR_GTTM, AR_GTTM_IGNORE_IDLE);
|
|
|
|
/*
|
|
* GTT debug mode setting
|
|
*/
|
|
/*
|
|
OS_REG_WRITE(ah, 0x64, 0x00320000);
|
|
OS_REG_WRITE(ah, 0x68, 7);
|
|
OS_REG_WRITE(ah, 0x4080, 0xC);
|
|
*/
|
|
/*
|
|
* Disable general interrupt mitigation by setting MIRT = 0x0
|
|
* Rx and tx interrupt mitigation are conditionally enabled below.
|
|
*/
|
|
OS_REG_WRITE(ah, AR_MIRT, 0);
|
|
if (ahp->ah_intr_mitigation_rx) {
|
|
/*
|
|
* Enable Interrupt Mitigation for Rx.
|
|
* If no build-specific limits for the rx interrupt mitigation
|
|
* timer have been specified, use conservative defaults.
|
|
*/
|
|
#ifndef AH_RIMT_VAL_LAST
|
|
#define AH_RIMT_LAST_MICROSEC 500
|
|
#endif
|
|
#ifndef AH_RIMT_VAL_FIRST
|
|
#define AH_RIMT_FIRST_MICROSEC 2000
|
|
#endif
|
|
#ifndef HOST_OFFLOAD
|
|
OS_REG_RMW_FIELD(ah, AR_RIMT, AR_RIMT_LAST, AH_RIMT_LAST_MICROSEC);
|
|
OS_REG_RMW_FIELD(ah, AR_RIMT, AR_RIMT_FIRST, AH_RIMT_FIRST_MICROSEC);
|
|
#else
|
|
/* lower mitigation level to reduce latency for offload arch. */
|
|
OS_REG_RMW_FIELD(ah, AR_RIMT, AR_RIMT_LAST,
|
|
(AH_RIMT_LAST_MICROSEC >> 2));
|
|
OS_REG_RMW_FIELD(ah, AR_RIMT, AR_RIMT_FIRST,
|
|
(AH_RIMT_FIRST_MICROSEC >> 2));
|
|
#endif
|
|
}
|
|
|
|
if (ahp->ah_intr_mitigation_tx) {
|
|
/*
|
|
* Enable Interrupt Mitigation for Tx.
|
|
* If no build-specific limits for the tx interrupt mitigation
|
|
* timer have been specified, use the values preferred for
|
|
* the carrier group's products.
|
|
*/
|
|
#ifndef AH_TIMT_LAST
|
|
#define AH_TIMT_LAST_MICROSEC 300
|
|
#endif
|
|
#ifndef AH_TIMT_FIRST
|
|
#define AH_TIMT_FIRST_MICROSEC 750
|
|
#endif
|
|
OS_REG_RMW_FIELD(ah, AR_TIMT, AR_TIMT_LAST, AH_TIMT_LAST_MICROSEC);
|
|
OS_REG_RMW_FIELD(ah, AR_TIMT, AR_TIMT_FIRST, AH_TIMT_FIRST_MICROSEC);
|
|
}
|
|
|
|
rx_chainmask = ahp->ah_rx_chainmask;
|
|
|
|
OS_REG_WRITE(ah, AR_PHY_RX_CHAINMASK, rx_chainmask);
|
|
OS_REG_WRITE(ah, AR_PHY_CAL_CHAINMASK, rx_chainmask);
|
|
|
|
ar9300_init_bb(ah, chan);
|
|
|
|
/* BB Step 7: Calibration */
|
|
/*
|
|
* Only kick off calibration not on offchan.
|
|
* If coming back from offchan, restore prevous Cal results
|
|
* since chip reset will clear existings.
|
|
*/
|
|
if (!ahp->ah_skip_rx_iq_cal) {
|
|
int i;
|
|
/* clear existing RX cal data */
|
|
for (i=0; i<AR9300_MAX_CHAINS; i++)
|
|
ahp->ah_rx_cal_corr[i] = 0;
|
|
|
|
ahp->ah_rx_cal_complete = AH_FALSE;
|
|
// ahp->ah_rx_cal_chan = chan->channel;
|
|
// ahp->ah_rx_cal_chan_flag = ichan->channel_flags;
|
|
ahp->ah_rx_cal_chan = 0;
|
|
ahp->ah_rx_cal_chan_flag = 0; /* XXX FreeBSD */
|
|
}
|
|
ar9300_invalidate_saved_cals(ah, ichan);
|
|
cal_ret = ar9300_init_cal(ah, chan, AH_FALSE, apply_last_iqcorr);
|
|
|
|
#if ATH_SUPPORT_MCI
|
|
if (AH_PRIVATE(ah)->ah_caps.halMciSupport && ahp->ah_mci_ready) {
|
|
if (IS_CHAN_2GHZ(ichan) &&
|
|
(ahp->ah_mci_bt_state == MCI_BT_SLEEP))
|
|
{
|
|
if (ar9300_mci_check_int(ah, AR_MCI_INTERRUPT_RX_MSG_REMOTE_RESET) ||
|
|
ar9300_mci_check_int(ah, AR_MCI_INTERRUPT_RX_MSG_REQ_WAKE))
|
|
{
|
|
/*
|
|
* BT is sleeping. Check if BT wakes up duing WLAN
|
|
* calibration. If BT wakes up during WLAN calibration, need
|
|
* to go through all message exchanges again and recal.
|
|
*/
|
|
HALDEBUG(ah, HAL_DEBUG_BT_COEX,
|
|
"(MCI) ### %s: BT wakes up during WLAN calibration.\n",
|
|
__func__);
|
|
OS_REG_WRITE(ah, AR_MCI_INTERRUPT_RX_MSG_RAW,
|
|
AR_MCI_INTERRUPT_RX_MSG_REMOTE_RESET |
|
|
AR_MCI_INTERRUPT_RX_MSG_REQ_WAKE);
|
|
HALDEBUG(ah, HAL_DEBUG_BT_COEX, "(MCI) send REMOTE_RESET\n");
|
|
ar9300_mci_remote_reset(ah, AH_TRUE);
|
|
ar9300_mci_send_sys_waking(ah, AH_TRUE);
|
|
OS_DELAY(1);
|
|
if (IS_CHAN_2GHZ(ichan)) {
|
|
ar9300_mci_send_lna_transfer(ah, AH_TRUE);
|
|
}
|
|
ahp->ah_mci_bt_state = MCI_BT_AWAKE;
|
|
|
|
/* Redo calibration */
|
|
HALDEBUG(ah, HAL_DEBUG_BT_COEX, "(MCI) %s: Re-calibrate.\n",
|
|
__func__);
|
|
ar9300_invalidate_saved_cals(ah, ichan);
|
|
cal_ret = ar9300_init_cal(ah, chan, AH_FALSE, apply_last_iqcorr);
|
|
}
|
|
}
|
|
ar9300_mci_enable_interrupt(ah);
|
|
}
|
|
#endif
|
|
|
|
if (!cal_ret) {
|
|
HALDEBUG(ah, HAL_DEBUG_RESET, "%s: Init Cal Failed\n", __func__);
|
|
FAIL(HAL_ESELFTEST);
|
|
}
|
|
|
|
ar9300_init_txbf(ah);
|
|
#if 0
|
|
/*
|
|
* WAR for owl 1.0 - restore chain mask for 2-chain cfgs after cal
|
|
*/
|
|
rx_chainmask = ahp->ah_rx_chainmask;
|
|
if ((rx_chainmask == 0x5) || (rx_chainmask == 0x3)) {
|
|
OS_REG_WRITE(ah, AR_PHY_RX_CHAINMASK, rx_chainmask);
|
|
OS_REG_WRITE(ah, AR_PHY_CAL_CHAINMASK, rx_chainmask);
|
|
}
|
|
#endif
|
|
|
|
/* Restore previous led state */
|
|
OS_REG_WRITE(ah, AR_CFG_LED, save_led_state | AR_CFG_SCLK_32KHZ);
|
|
|
|
#if ATH_BT_COEX
|
|
if (ahp->ah_bt_coex_config_type != HAL_BT_COEX_CFG_NONE) {
|
|
ar9300_init_bt_coex(ah);
|
|
|
|
#if ATH_SUPPORT_MCI
|
|
if (AH_PRIVATE(ah)->ah_caps.halMciSupport && ahp->ah_mci_ready) {
|
|
/* Check BT state again to make sure it's not changed. */
|
|
ar9300_mci_sync_bt_state(ah);
|
|
ar9300_mci_2g5g_switch(ah, AH_TRUE);
|
|
|
|
if ((ahp->ah_mci_bt_state == MCI_BT_AWAKE) &&
|
|
(ahp->ah_mci_query_bt == AH_TRUE))
|
|
{
|
|
ahp->ah_mci_need_flush_btinfo = AH_TRUE;
|
|
}
|
|
}
|
|
#endif
|
|
}
|
|
#endif
|
|
|
|
/* Start TSF2 for generic timer 8-15. */
|
|
ar9300_start_tsf2(ah);
|
|
|
|
/* MIMO Power save setting */
|
|
if (ar9300_get_capability(ah, HAL_CAP_DYNAMIC_SMPS, 0, AH_NULL) == HAL_OK) {
|
|
ar9300_set_sm_power_mode(ah, ahp->ah_sm_power_mode);
|
|
}
|
|
|
|
/*
|
|
* For big endian systems turn on swapping for descriptors
|
|
*/
|
|
#if AH_BYTE_ORDER == AH_BIG_ENDIAN
|
|
if (AR_SREV_HORNET(ah) || AR_SREV_WASP(ah) || AR_SREV_SCORPION(ah) || AR_SREV_HONEYBEE(ah)) {
|
|
OS_REG_RMW(ah, AR_CFG, AR_CFG_SWTB | AR_CFG_SWRB, 0);
|
|
} else {
|
|
ar9300_init_cfg_reg(ah);
|
|
}
|
|
#endif
|
|
|
|
if ( AR_SREV_OSPREY(ah) || AR_SREV_WASP(ah) || AR_SREV_SCORPION(ah) || AR_SREV_HONEYBEE(ah) ) {
|
|
OS_REG_RMW(ah, AR_CFG_LED, AR_CFG_LED_ASSOC_CTL, AR_CFG_LED_ASSOC_CTL);
|
|
}
|
|
|
|
#if !(defined(ART_BUILD)) && defined(ATH_SUPPORT_LED)
|
|
#define REG_WRITE(_reg, _val) *((volatile u_int32_t *)(_reg)) = (_val);
|
|
#define REG_READ(_reg) *((volatile u_int32_t *)(_reg))
|
|
#define ATH_GPIO_OUT_FUNCTION3 0xB8040038
|
|
#define ATH_GPIO_OE 0xB8040000
|
|
if ( AR_SREV_WASP(ah)) {
|
|
if (IS_CHAN_2GHZ((AH_PRIVATE(ah)->ah_curchan))) {
|
|
REG_WRITE(ATH_GPIO_OUT_FUNCTION3, ( REG_READ(ATH_GPIO_OUT_FUNCTION3) & (~(0xff << 8))) | (0x33 << 8) );
|
|
REG_WRITE(ATH_GPIO_OE, ( REG_READ(ATH_GPIO_OE) & (~(0x1 << 13) )));
|
|
}
|
|
else {
|
|
|
|
/* Disable 2G WLAN LED. During ath_open, reset function is called even before channel is set.
|
|
So 2GHz is taken as default and it also blinks. Hence
|
|
to avoid both from blinking, disable 2G led while in 5G mode */
|
|
|
|
REG_WRITE(ATH_GPIO_OE, ( REG_READ(ATH_GPIO_OE) | (1 << 13) ));
|
|
REG_WRITE(ATH_GPIO_OUT_FUNCTION3, ( REG_READ(ATH_GPIO_OUT_FUNCTION3) & (~(0xff))) | (0x33) );
|
|
REG_WRITE(ATH_GPIO_OE, ( REG_READ(ATH_GPIO_OE) & (~(0x1 << 12) )));
|
|
}
|
|
|
|
}
|
|
else if (AR_SREV_SCORPION(ah)) {
|
|
if (IS_CHAN_2GHZ((AH_PRIVATE(ah)->ah_curchan))) {
|
|
REG_WRITE(ATH_GPIO_OUT_FUNCTION3, ( REG_READ(ATH_GPIO_OUT_FUNCTION3) & (~(0xff << 8))) | (0x2F << 8) );
|
|
REG_WRITE(ATH_GPIO_OE, (( REG_READ(ATH_GPIO_OE) & (~(0x1 << 13) )) | (0x1 << 12)));
|
|
} else if (IS_CHAN_5GHZ((AH_PRIVATE(ah)->ah_curchan))) {
|
|
REG_WRITE(ATH_GPIO_OUT_FUNCTION3, ( REG_READ(ATH_GPIO_OUT_FUNCTION3) & (~(0xff))) | (0x2F) );
|
|
REG_WRITE(ATH_GPIO_OE, (( REG_READ(ATH_GPIO_OE) & (~(0x1 << 12) )) | (0x1 << 13)));
|
|
}
|
|
}
|
|
else if (AR_SREV_HONEYBEE(ah)) {
|
|
REG_WRITE(ATH_GPIO_OUT_FUNCTION3, ( REG_READ(ATH_GPIO_OUT_FUNCTION3) & (~(0xff))) | (0x32) );
|
|
REG_WRITE(ATH_GPIO_OE, (( REG_READ(ATH_GPIO_OE) & (~(0x1 << 12) ))));
|
|
}
|
|
#undef REG_READ
|
|
#undef REG_WRITE
|
|
#endif
|
|
|
|
/* XXX FreeBSD What's this? -adrian */
|
|
#if 0
|
|
chan->channel_flags = ichan->channel_flags;
|
|
chan->priv_flags = ichan->priv_flags;
|
|
#endif
|
|
|
|
#if FIX_NOISE_FLOOR
|
|
/* XXX FreeBSD is ichan appropariate? It was curchan.. */
|
|
ar9300_get_nf_hist_base(ah, ichan, is_scan, nf_buf);
|
|
ar9300_load_nf(ah, nf_buf);
|
|
if (nf_hist_buff_reset == 1)
|
|
{
|
|
nf_hist_buff_reset = 0;
|
|
#ifndef ATH_NF_PER_CHAN
|
|
if (First_NFCal(ah, ichan, is_scan, chan)){
|
|
if (ahp->ah_skip_rx_iq_cal && !is_scan) {
|
|
/* restore RX Cal result if existing */
|
|
ar9300_rx_iq_cal_restore(ah);
|
|
ahp->ah_skip_rx_iq_cal = AH_FALSE;
|
|
}
|
|
}
|
|
#endif /* ATH_NF_PER_CHAN */
|
|
}
|
|
else{
|
|
ar9300_start_nf_cal(ah);
|
|
}
|
|
#endif
|
|
|
|
#ifdef AH_SUPPORT_AR9300
|
|
/* BB Panic Watchdog */
|
|
if (ar9300_get_capability(ah, HAL_CAP_BB_PANIC_WATCHDOG, 0, AH_NULL) ==
|
|
HAL_OK)
|
|
{
|
|
ar9300_config_bb_panic_watchdog(ah);
|
|
}
|
|
#endif
|
|
|
|
/* While receiving unsupported rate frame receive state machine
|
|
* gets into a state 0xb and if phy_restart happens when rx
|
|
* state machine is in 0xb state, BB would go hang, if we
|
|
* see 0xb state after first bb panic, make sure that we
|
|
* disable the phy_restart.
|
|
*
|
|
* There may be multiple panics, make sure that we always do
|
|
* this if we see this panic at least once. This is required
|
|
* because reset seems to be writing from INI file.
|
|
*/
|
|
if ((ar9300_get_capability(ah, HAL_CAP_PHYRESTART_CLR_WAR, 0, AH_NULL)
|
|
== HAL_OK) && (((MS((AH9300(ah)->ah_bb_panic_last_status),
|
|
AR_PHY_BB_WD_RX_OFDM_SM)) == 0xb) ||
|
|
AH9300(ah)->ah_phyrestart_disabled) )
|
|
{
|
|
ar9300_disable_phy_restart(ah, 1);
|
|
}
|
|
|
|
|
|
|
|
ahp->ah_radar1 = MS(OS_REG_READ(ah, AR_PHY_RADAR_1),
|
|
AR_PHY_RADAR_1_CF_BIN_THRESH);
|
|
ahp->ah_dc_offset = MS(OS_REG_READ(ah, AR_PHY_TIMING2),
|
|
AR_PHY_TIMING2_DC_OFFSET);
|
|
ahp->ah_disable_cck = MS(OS_REG_READ(ah, AR_PHY_MODE),
|
|
AR_PHY_MODE_DISABLE_CCK);
|
|
|
|
if (AH9300(ah)->ah_enable_keysearch_always) {
|
|
ar9300_enable_keysearch_always(ah, 1);
|
|
}
|
|
|
|
#if ATH_LOW_POWER_ENABLE
|
|
#define REG_WRITE(_reg, _val) *((volatile u_int32_t *)(_reg)) = (_val)
|
|
#define REG_READ(_reg) *((volatile u_int32_t *)(_reg))
|
|
if (AR_SREV_OSPREY(ah)) {
|
|
REG_WRITE(0xb4000080, REG_READ(0xb4000080) | 3);
|
|
OS_REG_WRITE(ah, AR_RTC_RESET, 1);
|
|
OS_REG_SET_BIT(ah, AR_HOSTIF_REG(ah, AR_PCIE_PM_CTRL),
|
|
AR_PCIE_PM_CTRL_ENA);
|
|
OS_REG_SET_BIT(ah, AR_HOSTIF_REG(ah, AR_SPARE), 0xffffffff);
|
|
}
|
|
#undef REG_READ
|
|
#undef REG_WRITE
|
|
#endif /* ATH_LOW_POWER_ENABLE */
|
|
|
|
ar9300_disable_pll_lock_detect(ah);
|
|
|
|
/* H/W Green TX */
|
|
ar9300_control_signals_for_green_tx_mode(ah);
|
|
/* Smart Antenna, only for 5GHz on Scropion */
|
|
if (IEEE80211_IS_CHAN_2GHZ((AH_PRIVATE(ah)->ah_curchan)) && AR_SREV_SCORPION(ah)) {
|
|
ahp->ah_smartantenna_enable = 0;
|
|
}
|
|
|
|
ar9300_set_smart_antenna(ah, ahp->ah_smartantenna_enable);
|
|
|
|
if (AR_SREV_APHRODITE(ah) && ahp->ah_lna_div_use_bt_ant_enable)
|
|
OS_REG_SET_BIT(ah, AR_BTCOEX_WL_LNADIV, AR_BTCOEX_WL_LNADIV_FORCE_ON);
|
|
|
|
if (ahp->ah_skip_rx_iq_cal && !is_scan) {
|
|
/* restore RX Cal result if existing */
|
|
ar9300_rx_iq_cal_restore(ah);
|
|
ahp->ah_skip_rx_iq_cal = AH_FALSE;
|
|
}
|
|
|
|
|
|
return AH_TRUE;
|
|
bad:
|
|
OS_MARK(ah, AH_MARK_RESET_DONE, ecode);
|
|
*status = ecode;
|
|
|
|
if (ahp->ah_skip_rx_iq_cal && !is_scan) {
|
|
/* restore RX Cal result if existing */
|
|
ar9300_rx_iq_cal_restore(ah);
|
|
ahp->ah_skip_rx_iq_cal = AH_FALSE;
|
|
}
|
|
|
|
return AH_FALSE;
|
|
#undef FAIL
|
|
}
|
|
|
|
void
|
|
ar9300_green_ap_ps_on_off( struct ath_hal *ah, u_int16_t on_off)
|
|
{
|
|
/* Set/reset the ps flag */
|
|
AH9300(ah)->green_ap_ps_on = !!on_off;
|
|
}
|
|
|
|
/*
|
|
* This function returns 1, where it is possible to do
|
|
* single-chain power save.
|
|
*/
|
|
u_int16_t
|
|
ar9300_is_single_ant_power_save_possible(struct ath_hal *ah)
|
|
{
|
|
return AH_TRUE;
|
|
}
|
|
|
|
/* To avoid compilation warnings. Functions not used when EMULATION. */
|
|
/*
|
|
* ar9300_find_mag_approx()
|
|
*/
|
|
static int32_t
|
|
ar9300_find_mag_approx(struct ath_hal *ah, int32_t in_re, int32_t in_im)
|
|
{
|
|
int32_t abs_i = abs(in_re);
|
|
int32_t abs_q = abs(in_im);
|
|
int32_t max_abs, min_abs;
|
|
|
|
if (abs_i > abs_q) {
|
|
max_abs = abs_i;
|
|
min_abs = abs_q;
|
|
} else {
|
|
max_abs = abs_q;
|
|
min_abs = abs_i;
|
|
}
|
|
|
|
return (max_abs - (max_abs / 32) + (min_abs / 8) + (min_abs / 4));
|
|
}
|
|
|
|
/*
|
|
* ar9300_solve_iq_cal()
|
|
* solve 4x4 linear equation used in loopback iq cal.
|
|
*/
|
|
static HAL_BOOL
|
|
ar9300_solve_iq_cal(
|
|
struct ath_hal *ah,
|
|
int32_t sin_2phi_1,
|
|
int32_t cos_2phi_1,
|
|
int32_t sin_2phi_2,
|
|
int32_t cos_2phi_2,
|
|
int32_t mag_a0_d0,
|
|
int32_t phs_a0_d0,
|
|
int32_t mag_a1_d0,
|
|
int32_t phs_a1_d0,
|
|
int32_t solved_eq[])
|
|
{
|
|
int32_t f1 = cos_2phi_1 - cos_2phi_2;
|
|
int32_t f3 = sin_2phi_1 - sin_2phi_2;
|
|
int32_t f2;
|
|
int32_t mag_tx, phs_tx, mag_rx, phs_rx;
|
|
const int32_t result_shift = 1 << 15;
|
|
|
|
f2 = (((int64_t)f1 * (int64_t)f1) / result_shift) + (((int64_t)f3 * (int64_t)f3) / result_shift);
|
|
|
|
if (0 == f2) {
|
|
HALDEBUG(ah, HAL_DEBUG_CALIBRATE, "%s: Divide by 0(%d).\n",
|
|
__func__, __LINE__);
|
|
return AH_FALSE;
|
|
}
|
|
|
|
/* magnitude mismatch, tx */
|
|
mag_tx = f1 * (mag_a0_d0 - mag_a1_d0) + f3 * (phs_a0_d0 - phs_a1_d0);
|
|
/* phase mismatch, tx */
|
|
phs_tx = f3 * (-mag_a0_d0 + mag_a1_d0) + f1 * (phs_a0_d0 - phs_a1_d0);
|
|
|
|
mag_tx = (mag_tx / f2);
|
|
phs_tx = (phs_tx / f2);
|
|
|
|
/* magnitude mismatch, rx */
|
|
mag_rx =
|
|
mag_a0_d0 - (cos_2phi_1 * mag_tx + sin_2phi_1 * phs_tx) / result_shift;
|
|
/* phase mismatch, rx */
|
|
phs_rx =
|
|
phs_a0_d0 + (sin_2phi_1 * mag_tx - cos_2phi_1 * phs_tx) / result_shift;
|
|
|
|
solved_eq[0] = mag_tx;
|
|
solved_eq[1] = phs_tx;
|
|
solved_eq[2] = mag_rx;
|
|
solved_eq[3] = phs_rx;
|
|
|
|
return AH_TRUE;
|
|
}
|
|
|
|
/*
|
|
* ar9300_calc_iq_corr()
|
|
*/
|
|
static HAL_BOOL
|
|
ar9300_calc_iq_corr(struct ath_hal *ah, int32_t chain_idx,
|
|
const int32_t iq_res[], int32_t iqc_coeff[])
|
|
{
|
|
int32_t i2_m_q2_a0_d0, i2_p_q2_a0_d0, iq_corr_a0_d0;
|
|
int32_t i2_m_q2_a0_d1, i2_p_q2_a0_d1, iq_corr_a0_d1;
|
|
int32_t i2_m_q2_a1_d0, i2_p_q2_a1_d0, iq_corr_a1_d0;
|
|
int32_t i2_m_q2_a1_d1, i2_p_q2_a1_d1, iq_corr_a1_d1;
|
|
int32_t mag_a0_d0, mag_a1_d0, mag_a0_d1, mag_a1_d1;
|
|
int32_t phs_a0_d0, phs_a1_d0, phs_a0_d1, phs_a1_d1;
|
|
int32_t sin_2phi_1, cos_2phi_1, sin_2phi_2, cos_2phi_2;
|
|
int32_t mag_tx, phs_tx, mag_rx, phs_rx;
|
|
int32_t solved_eq[4], mag_corr_tx, phs_corr_tx, mag_corr_rx, phs_corr_rx;
|
|
int32_t q_q_coff, q_i_coff;
|
|
const int32_t res_scale = 1 << 15;
|
|
const int32_t delpt_shift = 1 << 8;
|
|
int32_t mag1, mag2;
|
|
|
|
i2_m_q2_a0_d0 = iq_res[0] & 0xfff;
|
|
i2_p_q2_a0_d0 = (iq_res[0] >> 12) & 0xfff;
|
|
iq_corr_a0_d0 = ((iq_res[0] >> 24) & 0xff) + ((iq_res[1] & 0xf) << 8);
|
|
|
|
if (i2_m_q2_a0_d0 > 0x800) {
|
|
i2_m_q2_a0_d0 = -((0xfff - i2_m_q2_a0_d0) + 1);
|
|
}
|
|
if (iq_corr_a0_d0 > 0x800) {
|
|
iq_corr_a0_d0 = -((0xfff - iq_corr_a0_d0) + 1);
|
|
}
|
|
|
|
i2_m_q2_a0_d1 = (iq_res[1] >> 4) & 0xfff;
|
|
i2_p_q2_a0_d1 = (iq_res[2] & 0xfff);
|
|
iq_corr_a0_d1 = (iq_res[2] >> 12) & 0xfff;
|
|
|
|
if (i2_m_q2_a0_d1 > 0x800) {
|
|
i2_m_q2_a0_d1 = -((0xfff - i2_m_q2_a0_d1) + 1);
|
|
}
|
|
if (iq_corr_a0_d1 > 0x800) {
|
|
iq_corr_a0_d1 = -((0xfff - iq_corr_a0_d1) + 1);
|
|
}
|
|
|
|
i2_m_q2_a1_d0 = ((iq_res[2] >> 24) & 0xff) + ((iq_res[3] & 0xf) << 8);
|
|
i2_p_q2_a1_d0 = (iq_res[3] >> 4) & 0xfff;
|
|
iq_corr_a1_d0 = iq_res[4] & 0xfff;
|
|
|
|
if (i2_m_q2_a1_d0 > 0x800) {
|
|
i2_m_q2_a1_d0 = -((0xfff - i2_m_q2_a1_d0) + 1);
|
|
}
|
|
if (iq_corr_a1_d0 > 0x800) {
|
|
iq_corr_a1_d0 = -((0xfff - iq_corr_a1_d0) + 1);
|
|
}
|
|
|
|
i2_m_q2_a1_d1 = (iq_res[4] >> 12) & 0xfff;
|
|
i2_p_q2_a1_d1 = ((iq_res[4] >> 24) & 0xff) + ((iq_res[5] & 0xf) << 8);
|
|
iq_corr_a1_d1 = (iq_res[5] >> 4) & 0xfff;
|
|
|
|
if (i2_m_q2_a1_d1 > 0x800) {
|
|
i2_m_q2_a1_d1 = -((0xfff - i2_m_q2_a1_d1) + 1);
|
|
}
|
|
if (iq_corr_a1_d1 > 0x800) {
|
|
iq_corr_a1_d1 = -((0xfff - iq_corr_a1_d1) + 1);
|
|
}
|
|
|
|
if ((i2_p_q2_a0_d0 == 0) ||
|
|
(i2_p_q2_a0_d1 == 0) ||
|
|
(i2_p_q2_a1_d0 == 0) ||
|
|
(i2_p_q2_a1_d1 == 0)) {
|
|
HALDEBUG(ah, HAL_DEBUG_CALIBRATE,
|
|
"%s: Divide by 0(%d):\na0_d0=%d\na0_d1=%d\na2_d0=%d\na1_d1=%d\n",
|
|
__func__, __LINE__,
|
|
i2_p_q2_a0_d0, i2_p_q2_a0_d1, i2_p_q2_a1_d0, i2_p_q2_a1_d1);
|
|
return AH_FALSE;
|
|
}
|
|
|
|
if ((i2_p_q2_a0_d0 <= 1024) || (i2_p_q2_a0_d0 > 2047) ||
|
|
(i2_p_q2_a1_d0 < 0) || (i2_p_q2_a1_d1 < 0) ||
|
|
(i2_p_q2_a0_d0 <= i2_m_q2_a0_d0) ||
|
|
(i2_p_q2_a0_d0 <= iq_corr_a0_d0) ||
|
|
(i2_p_q2_a0_d1 <= i2_m_q2_a0_d1) ||
|
|
(i2_p_q2_a0_d1 <= iq_corr_a0_d1) ||
|
|
(i2_p_q2_a1_d0 <= i2_m_q2_a1_d0) ||
|
|
(i2_p_q2_a1_d0 <= iq_corr_a1_d0) ||
|
|
(i2_p_q2_a1_d1 <= i2_m_q2_a1_d1) ||
|
|
(i2_p_q2_a1_d1 <= iq_corr_a1_d1)) {
|
|
return AH_FALSE;
|
|
}
|
|
|
|
mag_a0_d0 = (i2_m_q2_a0_d0 * res_scale) / i2_p_q2_a0_d0;
|
|
phs_a0_d0 = (iq_corr_a0_d0 * res_scale) / i2_p_q2_a0_d0;
|
|
|
|
mag_a0_d1 = (i2_m_q2_a0_d1 * res_scale) / i2_p_q2_a0_d1;
|
|
phs_a0_d1 = (iq_corr_a0_d1 * res_scale) / i2_p_q2_a0_d1;
|
|
|
|
mag_a1_d0 = (i2_m_q2_a1_d0 * res_scale) / i2_p_q2_a1_d0;
|
|
phs_a1_d0 = (iq_corr_a1_d0 * res_scale) / i2_p_q2_a1_d0;
|
|
|
|
mag_a1_d1 = (i2_m_q2_a1_d1 * res_scale) / i2_p_q2_a1_d1;
|
|
phs_a1_d1 = (iq_corr_a1_d1 * res_scale) / i2_p_q2_a1_d1;
|
|
|
|
/* without analog phase shift */
|
|
sin_2phi_1 = (((mag_a0_d0 - mag_a0_d1) * delpt_shift) / DELPT);
|
|
/* without analog phase shift */
|
|
cos_2phi_1 = (((phs_a0_d1 - phs_a0_d0) * delpt_shift) / DELPT);
|
|
/* with analog phase shift */
|
|
sin_2phi_2 = (((mag_a1_d0 - mag_a1_d1) * delpt_shift) / DELPT);
|
|
/* with analog phase shift */
|
|
cos_2phi_2 = (((phs_a1_d1 - phs_a1_d0) * delpt_shift) / DELPT);
|
|
|
|
/* force sin^2 + cos^2 = 1; */
|
|
/* find magnitude by approximation */
|
|
mag1 = ar9300_find_mag_approx(ah, cos_2phi_1, sin_2phi_1);
|
|
mag2 = ar9300_find_mag_approx(ah, cos_2phi_2, sin_2phi_2);
|
|
|
|
if ((mag1 == 0) || (mag2 == 0)) {
|
|
HALDEBUG(ah, HAL_DEBUG_CALIBRATE,
|
|
"%s: Divide by 0(%d): mag1=%d, mag2=%d\n",
|
|
__func__, __LINE__, mag1, mag2);
|
|
return AH_FALSE;
|
|
}
|
|
|
|
/* normalization sin and cos by mag */
|
|
sin_2phi_1 = (sin_2phi_1 * res_scale / mag1);
|
|
cos_2phi_1 = (cos_2phi_1 * res_scale / mag1);
|
|
sin_2phi_2 = (sin_2phi_2 * res_scale / mag2);
|
|
cos_2phi_2 = (cos_2phi_2 * res_scale / mag2);
|
|
|
|
/* calculate IQ mismatch */
|
|
if (AH_FALSE == ar9300_solve_iq_cal(ah,
|
|
sin_2phi_1, cos_2phi_1, sin_2phi_2, cos_2phi_2, mag_a0_d0,
|
|
phs_a0_d0, mag_a1_d0, phs_a1_d0, solved_eq))
|
|
{
|
|
HALDEBUG(ah, HAL_DEBUG_CALIBRATE,
|
|
"%s: Call to ar9300_solve_iq_cal failed.\n", __func__);
|
|
return AH_FALSE;
|
|
}
|
|
|
|
mag_tx = solved_eq[0];
|
|
phs_tx = solved_eq[1];
|
|
mag_rx = solved_eq[2];
|
|
phs_rx = solved_eq[3];
|
|
|
|
HALDEBUG(ah, HAL_DEBUG_CALIBRATE,
|
|
"%s: chain %d: mag mismatch=%d phase mismatch=%d\n",
|
|
__func__, chain_idx, mag_tx / res_scale, phs_tx / res_scale);
|
|
|
|
if (res_scale == mag_tx) {
|
|
HALDEBUG(ah, HAL_DEBUG_CALIBRATE,
|
|
"%s: Divide by 0(%d): mag_tx=%d, res_scale=%d\n",
|
|
__func__, __LINE__, mag_tx, res_scale);
|
|
return AH_FALSE;
|
|
}
|
|
|
|
/* calculate and quantize Tx IQ correction factor */
|
|
mag_corr_tx = (mag_tx * res_scale) / (res_scale - mag_tx);
|
|
phs_corr_tx = -phs_tx;
|
|
|
|
q_q_coff = (mag_corr_tx * 128 / res_scale);
|
|
q_i_coff = (phs_corr_tx * 256 / res_scale);
|
|
|
|
HALDEBUG(ah, HAL_DEBUG_CALIBRATE,
|
|
"%s: tx chain %d: mag corr=%d phase corr=%d\n",
|
|
__func__, chain_idx, q_q_coff, q_i_coff);
|
|
|
|
if (q_i_coff < -63) {
|
|
q_i_coff = -63;
|
|
}
|
|
if (q_i_coff > 63) {
|
|
q_i_coff = 63;
|
|
}
|
|
if (q_q_coff < -63) {
|
|
q_q_coff = -63;
|
|
}
|
|
if (q_q_coff > 63) {
|
|
q_q_coff = 63;
|
|
}
|
|
|
|
iqc_coeff[0] = (q_q_coff * 128) + (0x7f & q_i_coff);
|
|
|
|
HALDEBUG(ah, HAL_DEBUG_CALIBRATE, "%s: tx chain %d: iq corr coeff=%x\n",
|
|
__func__, chain_idx, iqc_coeff[0]);
|
|
|
|
if (-mag_rx == res_scale) {
|
|
HALDEBUG(ah, HAL_DEBUG_CALIBRATE,
|
|
"%s: Divide by 0(%d): mag_rx=%d, res_scale=%d\n",
|
|
__func__, __LINE__, mag_rx, res_scale);
|
|
return AH_FALSE;
|
|
}
|
|
|
|
/* calculate and quantize Rx IQ correction factors */
|
|
mag_corr_rx = (-mag_rx * res_scale) / (res_scale + mag_rx);
|
|
phs_corr_rx = -phs_rx;
|
|
|
|
q_q_coff = (mag_corr_rx * 128 / res_scale);
|
|
q_i_coff = (phs_corr_rx * 256 / res_scale);
|
|
|
|
HALDEBUG(ah, HAL_DEBUG_CALIBRATE,
|
|
"%s: rx chain %d: mag corr=%d phase corr=%d\n",
|
|
__func__, chain_idx, q_q_coff, q_i_coff);
|
|
|
|
if (q_i_coff < -63) {
|
|
q_i_coff = -63;
|
|
}
|
|
if (q_i_coff > 63) {
|
|
q_i_coff = 63;
|
|
}
|
|
if (q_q_coff < -63) {
|
|
q_q_coff = -63;
|
|
}
|
|
if (q_q_coff > 63) {
|
|
q_q_coff = 63;
|
|
}
|
|
|
|
iqc_coeff[1] = (q_q_coff * 128) + (0x7f & q_i_coff);
|
|
|
|
HALDEBUG(ah, HAL_DEBUG_CALIBRATE, "%s: rx chain %d: iq corr coeff=%x\n",
|
|
__func__, chain_idx, iqc_coeff[1]);
|
|
|
|
return AH_TRUE;
|
|
}
|
|
|
|
#define MAX_MAG_DELTA 11 //maximum magnitude mismatch delta across gains
|
|
#define MAX_PHS_DELTA 10 //maximum phase mismatch delta across gains
|
|
#define ABS(x) ((x) >= 0 ? (x) : (-(x)))
|
|
|
|
u_int32_t tx_corr_coeff[MAX_MEASUREMENT][AR9300_MAX_CHAINS] = {
|
|
{ AR_PHY_TX_IQCAL_CORR_COEFF_01_B0,
|
|
AR_PHY_TX_IQCAL_CORR_COEFF_01_B1,
|
|
AR_PHY_TX_IQCAL_CORR_COEFF_01_B2},
|
|
{ AR_PHY_TX_IQCAL_CORR_COEFF_01_B0,
|
|
AR_PHY_TX_IQCAL_CORR_COEFF_01_B1,
|
|
AR_PHY_TX_IQCAL_CORR_COEFF_01_B2},
|
|
{ AR_PHY_TX_IQCAL_CORR_COEFF_23_B0,
|
|
AR_PHY_TX_IQCAL_CORR_COEFF_23_B1,
|
|
AR_PHY_TX_IQCAL_CORR_COEFF_23_B2},
|
|
{ AR_PHY_TX_IQCAL_CORR_COEFF_23_B0,
|
|
AR_PHY_TX_IQCAL_CORR_COEFF_23_B1,
|
|
AR_PHY_TX_IQCAL_CORR_COEFF_23_B2},
|
|
{ AR_PHY_TX_IQCAL_CORR_COEFF_45_B0,
|
|
AR_PHY_TX_IQCAL_CORR_COEFF_45_B1,
|
|
AR_PHY_TX_IQCAL_CORR_COEFF_45_B2},
|
|
{ AR_PHY_TX_IQCAL_CORR_COEFF_45_B0,
|
|
AR_PHY_TX_IQCAL_CORR_COEFF_45_B1,
|
|
AR_PHY_TX_IQCAL_CORR_COEFF_45_B2},
|
|
{ AR_PHY_TX_IQCAL_CORR_COEFF_67_B0,
|
|
AR_PHY_TX_IQCAL_CORR_COEFF_67_B1,
|
|
AR_PHY_TX_IQCAL_CORR_COEFF_67_B2},
|
|
{ AR_PHY_TX_IQCAL_CORR_COEFF_67_B0,
|
|
AR_PHY_TX_IQCAL_CORR_COEFF_67_B1,
|
|
AR_PHY_TX_IQCAL_CORR_COEFF_67_B2},
|
|
};
|
|
|
|
static void
|
|
ar9300_tx_iq_cal_outlier_detection(struct ath_hal *ah, HAL_CHANNEL_INTERNAL *ichan, u_int32_t num_chains,
|
|
struct coeff_t *coeff, HAL_BOOL is_cal_reusable)
|
|
{
|
|
int nmeasurement, ch_idx, im;
|
|
int32_t magnitude, phase;
|
|
int32_t magnitude_max, phase_max;
|
|
int32_t magnitude_min, phase_min;
|
|
|
|
int32_t magnitude_max_idx, phase_max_idx;
|
|
int32_t magnitude_min_idx, phase_min_idx;
|
|
|
|
int32_t magnitude_avg, phase_avg;
|
|
int32_t outlier_mag_idx = 0;
|
|
int32_t outlier_phs_idx = 0;
|
|
|
|
|
|
if (AR_SREV_POSEIDON(ah)) {
|
|
HALASSERT(num_chains == 0x1);
|
|
|
|
tx_corr_coeff[0][0] = AR_PHY_TX_IQCAL_CORR_COEFF_01_B0_POSEIDON;
|
|
tx_corr_coeff[1][0] = AR_PHY_TX_IQCAL_CORR_COEFF_01_B0_POSEIDON;
|
|
tx_corr_coeff[2][0] = AR_PHY_TX_IQCAL_CORR_COEFF_23_B0_POSEIDON;
|
|
tx_corr_coeff[3][0] = AR_PHY_TX_IQCAL_CORR_COEFF_23_B0_POSEIDON;
|
|
tx_corr_coeff[4][0] = AR_PHY_TX_IQCAL_CORR_COEFF_45_B0_POSEIDON;
|
|
tx_corr_coeff[5][0] = AR_PHY_TX_IQCAL_CORR_COEFF_45_B0_POSEIDON;
|
|
tx_corr_coeff[6][0] = AR_PHY_TX_IQCAL_CORR_COEFF_67_B0_POSEIDON;
|
|
tx_corr_coeff[7][0] = AR_PHY_TX_IQCAL_CORR_COEFF_67_B0_POSEIDON;
|
|
}
|
|
|
|
for (ch_idx = 0; ch_idx < num_chains; ch_idx++) {
|
|
nmeasurement = OS_REG_READ_FIELD(ah,
|
|
AR_PHY_TX_IQCAL_STATUS_B0(ah), AR_PHY_CALIBRATED_GAINS_0);
|
|
if (nmeasurement > MAX_MEASUREMENT) {
|
|
nmeasurement = MAX_MEASUREMENT;
|
|
}
|
|
|
|
if (!AR_SREV_SCORPION(ah)) {
|
|
/*
|
|
* reset max/min variable to min/max values so that
|
|
* we always start with 1st calibrated gain value
|
|
*/
|
|
magnitude_max = -64;
|
|
phase_max = -64;
|
|
magnitude_min = 63;
|
|
phase_min = 63;
|
|
magnitude_avg = 0;
|
|
phase_avg = 0;
|
|
magnitude_max_idx = 0;
|
|
magnitude_min_idx = 0;
|
|
phase_max_idx = 0;
|
|
phase_min_idx = 0;
|
|
|
|
/* detect outlier only if nmeasurement > 1 */
|
|
if (nmeasurement > 1) {
|
|
/* printf("----------- start outlier detection -----------\n"); */
|
|
/*
|
|
* find max/min and phase/mag mismatch across all calibrated gains
|
|
*/
|
|
for (im = 0; im < nmeasurement; im++) {
|
|
magnitude = coeff->mag_coeff[ch_idx][im][0];
|
|
phase = coeff->phs_coeff[ch_idx][im][0];
|
|
|
|
magnitude_avg = magnitude_avg + magnitude;
|
|
phase_avg = phase_avg + phase;
|
|
if (magnitude > magnitude_max) {
|
|
magnitude_max = magnitude;
|
|
magnitude_max_idx = im;
|
|
}
|
|
if (magnitude < magnitude_min) {
|
|
magnitude_min = magnitude;
|
|
magnitude_min_idx = im;
|
|
}
|
|
if (phase > phase_max) {
|
|
phase_max = phase;
|
|
phase_max_idx = im;
|
|
}
|
|
if (phase < phase_min) {
|
|
phase_min = phase;
|
|
phase_min_idx = im;
|
|
}
|
|
}
|
|
/* find average (exclude max abs value) */
|
|
for (im = 0; im < nmeasurement; im++) {
|
|
magnitude = coeff->mag_coeff[ch_idx][im][0];
|
|
phase = coeff->phs_coeff[ch_idx][im][0];
|
|
if ((ABS(magnitude) < ABS(magnitude_max)) ||
|
|
(ABS(magnitude) < ABS(magnitude_min)))
|
|
{
|
|
magnitude_avg = magnitude_avg + magnitude;
|
|
}
|
|
if ((ABS(phase) < ABS(phase_max)) ||
|
|
(ABS(phase) < ABS(phase_min)))
|
|
{
|
|
phase_avg = phase_avg + phase;
|
|
}
|
|
}
|
|
magnitude_avg = magnitude_avg / (nmeasurement - 1);
|
|
phase_avg = phase_avg / (nmeasurement - 1);
|
|
|
|
/* detect magnitude outlier */
|
|
if (ABS(magnitude_max - magnitude_min) > MAX_MAG_DELTA) {
|
|
if (ABS(magnitude_max - magnitude_avg) >
|
|
ABS(magnitude_min - magnitude_avg))
|
|
{
|
|
/* max is outlier, force to avg */
|
|
outlier_mag_idx = magnitude_max_idx;
|
|
} else {
|
|
/* min is outlier, force to avg */
|
|
outlier_mag_idx = magnitude_min_idx;
|
|
}
|
|
coeff->mag_coeff[ch_idx][outlier_mag_idx][0] = magnitude_avg;
|
|
coeff->phs_coeff[ch_idx][outlier_mag_idx][0] = phase_avg;
|
|
HALDEBUG(ah, HAL_DEBUG_CALIBRATE,
|
|
"[ch%d][outlier mag gain%d]:: "
|
|
"mag_avg = %d (/128), phase_avg = %d (/256)\n",
|
|
ch_idx, outlier_mag_idx, magnitude_avg, phase_avg);
|
|
}
|
|
/* detect phase outlier */
|
|
if (ABS(phase_max - phase_min) > MAX_PHS_DELTA) {
|
|
if (ABS(phase_max-phase_avg) > ABS(phase_min - phase_avg)) {
|
|
/* max is outlier, force to avg */
|
|
outlier_phs_idx = phase_max_idx;
|
|
} else{
|
|
/* min is outlier, force to avg */
|
|
outlier_phs_idx = phase_min_idx;
|
|
}
|
|
coeff->mag_coeff[ch_idx][outlier_phs_idx][0] = magnitude_avg;
|
|
coeff->phs_coeff[ch_idx][outlier_phs_idx][0] = phase_avg;
|
|
HALDEBUG(ah, HAL_DEBUG_CALIBRATE,
|
|
"[ch%d][outlier phs gain%d]:: "
|
|
"mag_avg = %d (/128), phase_avg = %d (/256)\n",
|
|
ch_idx, outlier_phs_idx, magnitude_avg, phase_avg);
|
|
}
|
|
}
|
|
}
|
|
|
|
/*printf("------------ after outlier detection -------------\n");*/
|
|
for (im = 0; im < nmeasurement; im++) {
|
|
magnitude = coeff->mag_coeff[ch_idx][im][0];
|
|
phase = coeff->phs_coeff[ch_idx][im][0];
|
|
|
|
#if 0
|
|
printf("[ch%d][gain%d]:: mag = %d (/128), phase = %d (/256)\n",
|
|
ch_idx, im, magnitude, phase);
|
|
#endif
|
|
|
|
coeff->iqc_coeff[0] = (phase & 0x7f) | ((magnitude & 0x7f) << 7);
|
|
|
|
if ((im % 2) == 0) {
|
|
OS_REG_RMW_FIELD(ah,
|
|
tx_corr_coeff[im][ch_idx],
|
|
AR_PHY_TX_IQCAL_CORR_COEFF_00_COEFF_TABLE,
|
|
coeff->iqc_coeff[0]);
|
|
} else {
|
|
OS_REG_RMW_FIELD(ah,
|
|
tx_corr_coeff[im][ch_idx],
|
|
AR_PHY_TX_IQCAL_CORR_COEFF_01_COEFF_TABLE,
|
|
coeff->iqc_coeff[0]);
|
|
}
|
|
#if ATH_SUPPORT_CAL_REUSE
|
|
ichan->tx_corr_coeff[im][ch_idx] = coeff->iqc_coeff[0];
|
|
#endif
|
|
}
|
|
#if ATH_SUPPORT_CAL_REUSE
|
|
ichan->num_measures[ch_idx] = nmeasurement;
|
|
#endif
|
|
}
|
|
|
|
OS_REG_RMW_FIELD(ah, AR_PHY_TX_IQCAL_CONTROL_3,
|
|
AR_PHY_TX_IQCAL_CONTROL_3_IQCORR_EN, 0x1);
|
|
OS_REG_RMW_FIELD(ah, AR_PHY_RX_IQCAL_CORR_B0,
|
|
AR_PHY_RX_IQCAL_CORR_B0_LOOPBACK_IQCORR_EN, 0x1);
|
|
|
|
#if ATH_SUPPORT_CAL_REUSE
|
|
if (is_cal_reusable) {
|
|
ichan->one_time_txiqcal_done = AH_TRUE;
|
|
HALDEBUG(ah, HAL_DEBUG_FCS_RTT,
|
|
"(FCS) TXIQCAL saved - %d\n", ichan->channel);
|
|
}
|
|
#endif
|
|
}
|
|
|
|
#if ATH_SUPPORT_CAL_REUSE
|
|
static void
|
|
ar9300_tx_iq_cal_apply(struct ath_hal *ah, HAL_CHANNEL_INTERNAL *ichan)
|
|
{
|
|
struct ath_hal_9300 *ahp = AH9300(ah);
|
|
int nmeasurement, ch_idx, im;
|
|
|
|
u_int32_t tx_corr_coeff[MAX_MEASUREMENT][AR9300_MAX_CHAINS] = {
|
|
{ AR_PHY_TX_IQCAL_CORR_COEFF_01_B0,
|
|
AR_PHY_TX_IQCAL_CORR_COEFF_01_B1,
|
|
AR_PHY_TX_IQCAL_CORR_COEFF_01_B2},
|
|
{ AR_PHY_TX_IQCAL_CORR_COEFF_01_B0,
|
|
AR_PHY_TX_IQCAL_CORR_COEFF_01_B1,
|
|
AR_PHY_TX_IQCAL_CORR_COEFF_01_B2},
|
|
{ AR_PHY_TX_IQCAL_CORR_COEFF_23_B0,
|
|
AR_PHY_TX_IQCAL_CORR_COEFF_23_B1,
|
|
AR_PHY_TX_IQCAL_CORR_COEFF_23_B2},
|
|
{ AR_PHY_TX_IQCAL_CORR_COEFF_23_B0,
|
|
AR_PHY_TX_IQCAL_CORR_COEFF_23_B1,
|
|
AR_PHY_TX_IQCAL_CORR_COEFF_23_B2},
|
|
{ AR_PHY_TX_IQCAL_CORR_COEFF_45_B0,
|
|
AR_PHY_TX_IQCAL_CORR_COEFF_45_B1,
|
|
AR_PHY_TX_IQCAL_CORR_COEFF_45_B2},
|
|
{ AR_PHY_TX_IQCAL_CORR_COEFF_45_B0,
|
|
AR_PHY_TX_IQCAL_CORR_COEFF_45_B1,
|
|
AR_PHY_TX_IQCAL_CORR_COEFF_45_B2},
|
|
{ AR_PHY_TX_IQCAL_CORR_COEFF_67_B0,
|
|
AR_PHY_TX_IQCAL_CORR_COEFF_67_B1,
|
|
AR_PHY_TX_IQCAL_CORR_COEFF_67_B2},
|
|
{ AR_PHY_TX_IQCAL_CORR_COEFF_67_B0,
|
|
AR_PHY_TX_IQCAL_CORR_COEFF_67_B1,
|
|
AR_PHY_TX_IQCAL_CORR_COEFF_67_B2},
|
|
};
|
|
|
|
if (AR_SREV_POSEIDON(ah)) {
|
|
HALASSERT(ahp->ah_tx_cal_chainmask == 0x1);
|
|
|
|
tx_corr_coeff[0][0] = AR_PHY_TX_IQCAL_CORR_COEFF_01_B0_POSEIDON;
|
|
tx_corr_coeff[1][0] = AR_PHY_TX_IQCAL_CORR_COEFF_01_B0_POSEIDON;
|
|
tx_corr_coeff[2][0] = AR_PHY_TX_IQCAL_CORR_COEFF_23_B0_POSEIDON;
|
|
tx_corr_coeff[3][0] = AR_PHY_TX_IQCAL_CORR_COEFF_23_B0_POSEIDON;
|
|
tx_corr_coeff[4][0] = AR_PHY_TX_IQCAL_CORR_COEFF_45_B0_POSEIDON;
|
|
tx_corr_coeff[5][0] = AR_PHY_TX_IQCAL_CORR_COEFF_45_B0_POSEIDON;
|
|
tx_corr_coeff[6][0] = AR_PHY_TX_IQCAL_CORR_COEFF_67_B0_POSEIDON;
|
|
tx_corr_coeff[7][0] = AR_PHY_TX_IQCAL_CORR_COEFF_67_B0_POSEIDON;
|
|
}
|
|
|
|
for (ch_idx = 0; ch_idx < AR9300_MAX_CHAINS; ch_idx++) {
|
|
if ((ahp->ah_tx_cal_chainmask & (1 << ch_idx)) == 0) {
|
|
continue;
|
|
}
|
|
nmeasurement = ichan->num_measures[ch_idx];
|
|
|
|
for (im = 0; im < nmeasurement; im++) {
|
|
if ((im % 2) == 0) {
|
|
OS_REG_RMW_FIELD(ah,
|
|
tx_corr_coeff[im][ch_idx],
|
|
AR_PHY_TX_IQCAL_CORR_COEFF_00_COEFF_TABLE,
|
|
ichan->tx_corr_coeff[im][ch_idx]);
|
|
} else {
|
|
OS_REG_RMW_FIELD(ah,
|
|
tx_corr_coeff[im][ch_idx],
|
|
AR_PHY_TX_IQCAL_CORR_COEFF_01_COEFF_TABLE,
|
|
ichan->tx_corr_coeff[im][ch_idx]);
|
|
}
|
|
}
|
|
}
|
|
|
|
OS_REG_RMW_FIELD(ah, AR_PHY_TX_IQCAL_CONTROL_3,
|
|
AR_PHY_TX_IQCAL_CONTROL_3_IQCORR_EN, 0x1);
|
|
OS_REG_RMW_FIELD(ah, AR_PHY_RX_IQCAL_CORR_B0,
|
|
AR_PHY_RX_IQCAL_CORR_B0_LOOPBACK_IQCORR_EN, 0x1);
|
|
}
|
|
#endif
|
|
|
|
/*
|
|
* ar9300_tx_iq_cal_hw_run is only needed for osprey/wasp/hornet
|
|
* It is not needed for jupiter/poseidon.
|
|
*/
|
|
HAL_BOOL
|
|
ar9300_tx_iq_cal_hw_run(struct ath_hal *ah)
|
|
{
|
|
int is_tx_gain_forced;
|
|
|
|
is_tx_gain_forced = OS_REG_READ_FIELD(ah,
|
|
AR_PHY_TX_FORCED_GAIN, AR_PHY_TXGAIN_FORCE);
|
|
if (is_tx_gain_forced) {
|
|
/*printf("Tx gain can not be forced during tx I/Q cal!\n");*/
|
|
OS_REG_RMW_FIELD(ah, AR_PHY_TX_FORCED_GAIN, AR_PHY_TXGAIN_FORCE, 0);
|
|
}
|
|
|
|
/* enable tx IQ cal */
|
|
OS_REG_RMW_FIELD(ah, AR_PHY_TX_IQCAL_START(ah),
|
|
AR_PHY_TX_IQCAL_START_DO_CAL, AR_PHY_TX_IQCAL_START_DO_CAL);
|
|
|
|
if (!ath_hal_wait(ah,
|
|
AR_PHY_TX_IQCAL_START(ah), AR_PHY_TX_IQCAL_START_DO_CAL, 0))
|
|
{
|
|
HALDEBUG(ah, HAL_DEBUG_CALIBRATE,
|
|
"%s: Tx IQ Cal is never completed.\n", __func__);
|
|
return AH_FALSE;
|
|
}
|
|
return AH_TRUE;
|
|
}
|
|
|
|
static void
|
|
ar9300_tx_iq_cal_post_proc(struct ath_hal *ah,HAL_CHANNEL_INTERNAL *ichan,
|
|
int iqcal_idx, int max_iqcal,HAL_BOOL is_cal_reusable, HAL_BOOL apply_last_corr)
|
|
{
|
|
int nmeasurement=0, im, ix, iy, temp;
|
|
struct ath_hal_9300 *ahp = AH9300(ah);
|
|
u_int32_t txiqcal_status[AR9300_MAX_CHAINS] = {
|
|
AR_PHY_TX_IQCAL_STATUS_B0(ah),
|
|
AR_PHY_TX_IQCAL_STATUS_B1,
|
|
AR_PHY_TX_IQCAL_STATUS_B2,
|
|
};
|
|
const u_int32_t chan_info_tab[] = {
|
|
AR_PHY_CHAN_INFO_TAB_0,
|
|
AR_PHY_CHAN_INFO_TAB_1,
|
|
AR_PHY_CHAN_INFO_TAB_2,
|
|
};
|
|
int32_t iq_res[6];
|
|
int32_t ch_idx, j;
|
|
u_int32_t num_chains = 0;
|
|
static struct coeff_t coeff;
|
|
txiqcal_status[0] = AR_PHY_TX_IQCAL_STATUS_B0(ah);
|
|
|
|
for (ch_idx = 0; ch_idx < AR9300_MAX_CHAINS; ch_idx++) {
|
|
if (ahp->ah_tx_chainmask & (1 << ch_idx)) {
|
|
num_chains++;
|
|
}
|
|
}
|
|
|
|
if (apply_last_corr) {
|
|
if (coeff.last_cal == AH_TRUE) {
|
|
int32_t magnitude, phase;
|
|
int ch_idx, im;
|
|
u_int32_t tx_corr_coeff[MAX_MEASUREMENT][AR9300_MAX_CHAINS] = {
|
|
{ AR_PHY_TX_IQCAL_CORR_COEFF_01_B0,
|
|
AR_PHY_TX_IQCAL_CORR_COEFF_01_B1,
|
|
AR_PHY_TX_IQCAL_CORR_COEFF_01_B2},
|
|
{ AR_PHY_TX_IQCAL_CORR_COEFF_01_B0,
|
|
AR_PHY_TX_IQCAL_CORR_COEFF_01_B1,
|
|
AR_PHY_TX_IQCAL_CORR_COEFF_01_B2},
|
|
{ AR_PHY_TX_IQCAL_CORR_COEFF_23_B0,
|
|
AR_PHY_TX_IQCAL_CORR_COEFF_23_B1,
|
|
AR_PHY_TX_IQCAL_CORR_COEFF_23_B2},
|
|
{ AR_PHY_TX_IQCAL_CORR_COEFF_23_B0,
|
|
AR_PHY_TX_IQCAL_CORR_COEFF_23_B1,
|
|
AR_PHY_TX_IQCAL_CORR_COEFF_23_B2},
|
|
{ AR_PHY_TX_IQCAL_CORR_COEFF_45_B0,
|
|
AR_PHY_TX_IQCAL_CORR_COEFF_45_B1,
|
|
AR_PHY_TX_IQCAL_CORR_COEFF_45_B2},
|
|
{ AR_PHY_TX_IQCAL_CORR_COEFF_45_B0,
|
|
AR_PHY_TX_IQCAL_CORR_COEFF_45_B1,
|
|
AR_PHY_TX_IQCAL_CORR_COEFF_45_B2},
|
|
{ AR_PHY_TX_IQCAL_CORR_COEFF_67_B0,
|
|
AR_PHY_TX_IQCAL_CORR_COEFF_67_B1,
|
|
AR_PHY_TX_IQCAL_CORR_COEFF_67_B2},
|
|
{ AR_PHY_TX_IQCAL_CORR_COEFF_67_B0,
|
|
AR_PHY_TX_IQCAL_CORR_COEFF_67_B1,
|
|
AR_PHY_TX_IQCAL_CORR_COEFF_67_B2},
|
|
};
|
|
for (ch_idx = 0; ch_idx < num_chains; ch_idx++) {
|
|
for (im = 0; im < coeff.last_nmeasurement; im++) {
|
|
magnitude = coeff.mag_coeff[ch_idx][im][0];
|
|
phase = coeff.phs_coeff[ch_idx][im][0];
|
|
|
|
#if 0
|
|
printf("[ch%d][gain%d]:: mag = %d (/128), phase = %d (/256)\n",
|
|
ch_idx, im, magnitude, phase);
|
|
#endif
|
|
|
|
coeff.iqc_coeff[0] = (phase & 0x7f) | ((magnitude & 0x7f) << 7);
|
|
if ((im % 2) == 0) {
|
|
OS_REG_RMW_FIELD(ah,
|
|
tx_corr_coeff[im][ch_idx],
|
|
AR_PHY_TX_IQCAL_CORR_COEFF_00_COEFF_TABLE,
|
|
coeff.iqc_coeff[0]);
|
|
} else {
|
|
OS_REG_RMW_FIELD(ah,
|
|
tx_corr_coeff[im][ch_idx],
|
|
AR_PHY_TX_IQCAL_CORR_COEFF_01_COEFF_TABLE,
|
|
coeff.iqc_coeff[0]);
|
|
}
|
|
}
|
|
}
|
|
OS_REG_RMW_FIELD(ah, AR_PHY_TX_IQCAL_CONTROL_3,
|
|
AR_PHY_TX_IQCAL_CONTROL_3_IQCORR_EN, 0x1);
|
|
}
|
|
return;
|
|
}
|
|
|
|
|
|
for (ch_idx = 0; ch_idx < num_chains; ch_idx++) {
|
|
nmeasurement = OS_REG_READ_FIELD(ah,
|
|
AR_PHY_TX_IQCAL_STATUS_B0(ah), AR_PHY_CALIBRATED_GAINS_0);
|
|
if (nmeasurement > MAX_MEASUREMENT) {
|
|
nmeasurement = MAX_MEASUREMENT;
|
|
}
|
|
|
|
for (im = 0; im < nmeasurement; im++) {
|
|
HALDEBUG(ah, HAL_DEBUG_CALIBRATE,
|
|
"%s: Doing Tx IQ Cal for chain %d.\n", __func__, ch_idx);
|
|
if (OS_REG_READ(ah, txiqcal_status[ch_idx]) &
|
|
AR_PHY_TX_IQCAL_STATUS_FAILED)
|
|
{
|
|
HALDEBUG(ah, HAL_DEBUG_CALIBRATE,
|
|
"%s: Tx IQ Cal failed for chain %d.\n", __func__, ch_idx);
|
|
goto TX_IQ_CAL_FAILED_;
|
|
}
|
|
|
|
for (j = 0; j < 3; j++) {
|
|
u_int32_t idx = 2 * j;
|
|
/* 3 registers for each calibration result */
|
|
u_int32_t offset = 4 * (3 * im + j);
|
|
|
|
OS_REG_RMW_FIELD(ah, AR_PHY_CHAN_INFO_MEMORY,
|
|
AR_PHY_CHAN_INFO_TAB_S2_READ, 0);
|
|
/* 32 bits */
|
|
iq_res[idx] = OS_REG_READ(ah, chan_info_tab[ch_idx] + offset);
|
|
OS_REG_RMW_FIELD(ah, AR_PHY_CHAN_INFO_MEMORY,
|
|
AR_PHY_CHAN_INFO_TAB_S2_READ, 1);
|
|
/* 16 bits */
|
|
iq_res[idx + 1] = 0xffff &
|
|
OS_REG_READ(ah, chan_info_tab[ch_idx] + offset);
|
|
|
|
HALDEBUG(ah, HAL_DEBUG_CALIBRATE,
|
|
"%s: IQ RES[%d]=0x%x IQ_RES[%d]=0x%x\n",
|
|
__func__, idx, iq_res[idx], idx + 1, iq_res[idx + 1]);
|
|
}
|
|
|
|
if (AH_FALSE == ar9300_calc_iq_corr(
|
|
ah, ch_idx, iq_res, coeff.iqc_coeff))
|
|
{
|
|
HALDEBUG(ah, HAL_DEBUG_CALIBRATE,
|
|
"%s: Failed in calculation of IQ correction.\n",
|
|
__func__);
|
|
goto TX_IQ_CAL_FAILED_;
|
|
}
|
|
|
|
coeff.phs_coeff[ch_idx][im][iqcal_idx-1] = coeff.iqc_coeff[0] & 0x7f;
|
|
coeff.mag_coeff[ch_idx][im][iqcal_idx-1] = (coeff.iqc_coeff[0] >> 7) & 0x7f;
|
|
if (coeff.mag_coeff[ch_idx][im][iqcal_idx-1] > 63) {
|
|
coeff.mag_coeff[ch_idx][im][iqcal_idx-1] -= 128;
|
|
}
|
|
if (coeff.phs_coeff[ch_idx][im][iqcal_idx-1] > 63) {
|
|
coeff.phs_coeff[ch_idx][im][iqcal_idx-1] -= 128;
|
|
}
|
|
#if 0
|
|
ath_hal_printf(ah, "IQCAL::[ch%d][gain%d]:: mag = %d phase = %d \n",
|
|
ch_idx, im, coeff.mag_coeff[ch_idx][im][iqcal_idx-1],
|
|
coeff.phs_coeff[ch_idx][im][iqcal_idx-1]);
|
|
#endif
|
|
}
|
|
}
|
|
//last iteration; calculate mag and phs
|
|
if (iqcal_idx == max_iqcal) {
|
|
if (max_iqcal>1) {
|
|
for (ch_idx = 0; ch_idx < num_chains; ch_idx++) {
|
|
for (im = 0; im < nmeasurement; im++) {
|
|
//sort mag and phs
|
|
for( ix=0;ix<max_iqcal-1;ix++){
|
|
for( iy=ix+1;iy<=max_iqcal-1;iy++){
|
|
if(coeff.mag_coeff[ch_idx][im][iy] <
|
|
coeff.mag_coeff[ch_idx][im][ix]) {
|
|
//swap
|
|
temp=coeff.mag_coeff[ch_idx][im][ix];
|
|
coeff.mag_coeff[ch_idx][im][ix] = coeff.mag_coeff[ch_idx][im][iy];
|
|
coeff.mag_coeff[ch_idx][im][iy] = temp;
|
|
}
|
|
if(coeff.phs_coeff[ch_idx][im][iy] <
|
|
coeff.phs_coeff[ch_idx][im][ix]){
|
|
//swap
|
|
temp=coeff.phs_coeff[ch_idx][im][ix];
|
|
coeff.phs_coeff[ch_idx][im][ix]=coeff.phs_coeff[ch_idx][im][iy];
|
|
coeff.phs_coeff[ch_idx][im][iy]=temp;
|
|
}
|
|
}
|
|
}
|
|
//select median; 3rd entry in the sorted array
|
|
coeff.mag_coeff[ch_idx][im][0] =
|
|
coeff.mag_coeff[ch_idx][im][max_iqcal/2];
|
|
coeff.phs_coeff[ch_idx][im][0] =
|
|
coeff.phs_coeff[ch_idx][im][max_iqcal/2];
|
|
HALDEBUG(ah, HAL_DEBUG_CALIBRATE,
|
|
"IQCAL: Median [ch%d][gain%d]:: mag = %d phase = %d \n",
|
|
ch_idx, im,coeff.mag_coeff[ch_idx][im][0],
|
|
coeff.phs_coeff[ch_idx][im][0]);
|
|
}
|
|
}
|
|
}
|
|
ar9300_tx_iq_cal_outlier_detection(ah,ichan, num_chains, &coeff,is_cal_reusable);
|
|
}
|
|
|
|
|
|
coeff.last_nmeasurement = nmeasurement;
|
|
coeff.last_cal = AH_TRUE;
|
|
|
|
return;
|
|
|
|
TX_IQ_CAL_FAILED_:
|
|
/* no need to print this, it is AGC failure not chip stuck */
|
|
/*ath_hal_printf(ah, "Tx IQ Cal failed(%d)\n", line);*/
|
|
coeff.last_cal = AH_FALSE;
|
|
return;
|
|
}
|
|
|
|
|
|
/*
|
|
* ar9300_disable_phy_restart
|
|
*
|
|
* In some BBpanics, we can disable the phyrestart
|
|
* disable_phy_restart
|
|
* != 0, disable the phy restart in h/w
|
|
* == 0, enable the phy restart in h/w
|
|
*/
|
|
void ar9300_disable_phy_restart(struct ath_hal *ah, int disable_phy_restart)
|
|
{
|
|
u_int32_t val;
|
|
|
|
val = OS_REG_READ(ah, AR_PHY_RESTART);
|
|
if (disable_phy_restart) {
|
|
val &= ~AR_PHY_RESTART_ENA;
|
|
AH9300(ah)->ah_phyrestart_disabled = 1;
|
|
} else {
|
|
val |= AR_PHY_RESTART_ENA;
|
|
AH9300(ah)->ah_phyrestart_disabled = 0;
|
|
}
|
|
OS_REG_WRITE(ah, AR_PHY_RESTART, val);
|
|
|
|
val = OS_REG_READ(ah, AR_PHY_RESTART);
|
|
}
|
|
|
|
HAL_BOOL
|
|
ar9300_interference_is_present(struct ath_hal *ah)
|
|
{
|
|
int i;
|
|
struct ath_hal_private *ahpriv = AH_PRIVATE(ah);
|
|
const struct ieee80211_channel *chan = ahpriv->ah_curchan;
|
|
HAL_CHANNEL_INTERNAL *ichan = ath_hal_checkchannel(ah, chan);
|
|
|
|
if (ichan == NULL) {
|
|
ath_hal_printf(ah, "%s: called with ichan=NULL\n", __func__);
|
|
return AH_FALSE;
|
|
}
|
|
|
|
/* This function is called after a stuck beacon, if EACS is enabled.
|
|
* If CW interference is severe, then HW goes into a loop of continuous
|
|
* stuck beacons and resets. On reset the NF cal history is cleared.
|
|
* So the median value of the history cannot be used -
|
|
* hence check if any value (Chain 0/Primary Channel)
|
|
* is outside the bounds.
|
|
*/
|
|
HAL_NFCAL_HIST_FULL *h = AH_HOME_CHAN_NFCAL_HIST(ah, ichan);
|
|
for (i = 0; i < HAL_NF_CAL_HIST_LEN_FULL; i++) {
|
|
if (h->nf_cal_buffer[i][0] >
|
|
AH9300(ah)->nfp->nominal + AH9300(ah)->nf_cw_int_delta)
|
|
{
|
|
return AH_TRUE;
|
|
}
|
|
|
|
}
|
|
return AH_FALSE;
|
|
}
|
|
|
|
#if ATH_SUPPORT_CRDC
|
|
void
|
|
ar9300_crdc_rx_notify(struct ath_hal *ah, struct ath_rx_status *rxs)
|
|
{
|
|
struct ath_hal_private *ahpriv = AH_PRIVATE(ah);
|
|
int rssi_index;
|
|
|
|
if ((!AR_SREV_WASP(ah)) ||
|
|
(!ahpriv->ah_config.ath_hal_crdc_enable)) {
|
|
return;
|
|
}
|
|
|
|
if (rxs->rs_isaggr && rxs->rs_moreaggr) {
|
|
return;
|
|
}
|
|
|
|
if ((rxs->rs_rssi_ctl0 >= HAL_RSSI_BAD) ||
|
|
(rxs->rs_rssi_ctl1 >= HAL_RSSI_BAD)) {
|
|
return;
|
|
}
|
|
|
|
rssi_index = ah->ah_crdc_rssi_ptr % HAL_MAX_CRDC_RSSI_SAMPLE;
|
|
|
|
ah->ah_crdc_rssi_sample[0][rssi_index] = rxs->rs_rssi_ctl0;
|
|
ah->ah_crdc_rssi_sample[1][rssi_index] = rxs->rs_rssi_ctl1;
|
|
|
|
ah->ah_crdc_rssi_ptr++;
|
|
}
|
|
|
|
static int
|
|
ar9300_crdc_avg_rssi(struct ath_hal *ah, int chain)
|
|
{
|
|
int crdc_rssi_sum = 0;
|
|
int crdc_rssi_ptr = ah->ah_crdc_rssi_ptr, i;
|
|
struct ath_hal_private *ahpriv = AH_PRIVATE(ah);
|
|
int crdc_window = ahpriv->ah_config.ath_hal_crdc_window;
|
|
|
|
if (crdc_window > HAL_MAX_CRDC_RSSI_SAMPLE) {
|
|
crdc_window = HAL_MAX_CRDC_RSSI_SAMPLE;
|
|
}
|
|
|
|
for (i = 1; i <= crdc_window; i++) {
|
|
crdc_rssi_sum +=
|
|
ah->ah_crdc_rssi_sample[chain]
|
|
[(crdc_rssi_ptr - i) % HAL_MAX_CRDC_RSSI_SAMPLE];
|
|
}
|
|
|
|
return crdc_rssi_sum / crdc_window;
|
|
}
|
|
|
|
static void
|
|
ar9300_crdc_activate(struct ath_hal *ah, int rssi_diff, int enable)
|
|
{
|
|
int val, orig_val;
|
|
struct ath_hal_private *ahpriv = AH_PRIVATE(ah);
|
|
int crdc_numerator = ahpriv->ah_config.ath_hal_crdc_numerator;
|
|
int crdc_denominator = ahpriv->ah_config.ath_hal_crdc_denominator;
|
|
int c = (rssi_diff * crdc_numerator) / crdc_denominator;
|
|
|
|
val = orig_val = OS_REG_READ(ah, AR_PHY_MULTICHAIN_CTRL);
|
|
val &= 0xffffff00;
|
|
if (enable) {
|
|
val |= 0x1;
|
|
val |= ((c << 1) & 0xff);
|
|
}
|
|
OS_REG_WRITE(ah, AR_PHY_MULTICHAIN_CTRL, val);
|
|
HALDEBUG(ah, HAL_DEBUG_CALIBRATE, "diff: %02d comp: %02d reg: %08x %08x\n",
|
|
rssi_diff, c, orig_val, val);
|
|
}
|
|
|
|
|
|
void ar9300_chain_rssi_diff_compensation(struct ath_hal *ah)
|
|
{
|
|
struct ath_hal_private *ahpriv = AH_PRIVATE(ah);
|
|
int crdc_window = ahpriv->ah_config.ath_hal_crdc_window;
|
|
int crdc_rssi_ptr = ah->ah_crdc_rssi_ptr;
|
|
int crdc_rssi_thresh = ahpriv->ah_config.ath_hal_crdc_rssithresh;
|
|
int crdc_diff_thresh = ahpriv->ah_config.ath_hal_crdc_diffthresh;
|
|
int avg_rssi[2], avg_rssi_diff;
|
|
|
|
if ((!AR_SREV_WASP(ah)) ||
|
|
(!ahpriv->ah_config.ath_hal_crdc_enable)) {
|
|
if (ah->ah_crdc_rssi_ptr) {
|
|
ar9300_crdc_activate(ah, 0, 0);
|
|
ah->ah_crdc_rssi_ptr = 0;
|
|
}
|
|
return;
|
|
}
|
|
|
|
if (crdc_window > HAL_MAX_CRDC_RSSI_SAMPLE) {
|
|
crdc_window = HAL_MAX_CRDC_RSSI_SAMPLE;
|
|
}
|
|
|
|
if (crdc_rssi_ptr < crdc_window) {
|
|
return;
|
|
}
|
|
|
|
avg_rssi[0] = ar9300_crdc_avg_rssi(ah, 0);
|
|
avg_rssi[1] = ar9300_crdc_avg_rssi(ah, 1);
|
|
avg_rssi_diff = avg_rssi[1] - avg_rssi[0];
|
|
|
|
HALDEBUG(ah, HAL_DEBUG_CALIBRATE, "crdc: avg: %02d %02d ",
|
|
avg_rssi[0], avg_rssi[1]);
|
|
|
|
if ((avg_rssi[0] < crdc_rssi_thresh) &&
|
|
(avg_rssi[1] < crdc_rssi_thresh)) {
|
|
ar9300_crdc_activate(ah, 0, 0);
|
|
} else {
|
|
if (ABS(avg_rssi_diff) >= crdc_diff_thresh) {
|
|
ar9300_crdc_activate(ah, avg_rssi_diff, 1);
|
|
} else {
|
|
ar9300_crdc_activate(ah, 0, 1);
|
|
}
|
|
}
|
|
}
|
|
#endif
|
|
|
|
#if ATH_ANT_DIV_COMB
|
|
HAL_BOOL
|
|
ar9300_ant_ctrl_set_lna_div_use_bt_ant(struct ath_hal *ah, HAL_BOOL enable, const struct ieee80211_channel *chan)
|
|
{
|
|
u_int32_t value;
|
|
u_int32_t regval;
|
|
struct ath_hal_9300 *ahp = AH9300(ah);
|
|
HAL_CHANNEL_INTERNAL *ichan;
|
|
struct ath_hal_private *ahpriv = AH_PRIVATE(ah);
|
|
HAL_CAPABILITIES *pcap = &ahpriv->ah_caps;
|
|
|
|
HALDEBUG(ah, HAL_DEBUG_RESET | HAL_DEBUG_BT_COEX,
|
|
"%s: called; enable=%d\n", __func__, enable);
|
|
|
|
if (AR_SREV_POSEIDON(ah)) {
|
|
// Make sure this scheme is only used for WB225(Astra)
|
|
ahp->ah_lna_div_use_bt_ant_enable = enable;
|
|
|
|
ichan = ar9300_check_chan(ah, chan);
|
|
if ( ichan == AH_NULL ) {
|
|
HALDEBUG(ah, HAL_DEBUG_CHANNEL, "%s: invalid channel %u/0x%x; no mapping\n",
|
|
__func__, chan->ic_freq, chan->ic_flags);
|
|
return AH_FALSE;
|
|
}
|
|
|
|
if ( enable == TRUE ) {
|
|
pcap->halAntDivCombSupport = TRUE;
|
|
} else {
|
|
pcap->halAntDivCombSupport = pcap->halAntDivCombSupportOrg;
|
|
}
|
|
|
|
#define AR_SWITCH_TABLE_COM2_ALL (0xffffff)
|
|
#define AR_SWITCH_TABLE_COM2_ALL_S (0)
|
|
value = ar9300_ant_ctrl_common2_get(ah, IS_CHAN_2GHZ(ichan));
|
|
if ( enable == TRUE ) {
|
|
value &= ~AR_SWITCH_TABLE_COM2_ALL;
|
|
value |= ah->ah_config.ath_hal_ant_ctrl_comm2g_switch_enable;
|
|
}
|
|
HALDEBUG(ah, HAL_DEBUG_RESET, "%s: com2=0x%08x\n", __func__, value);
|
|
OS_REG_RMW_FIELD(ah, AR_PHY_SWITCH_COM_2, AR_SWITCH_TABLE_COM2_ALL, value);
|
|
|
|
value = ar9300_eeprom_get(ahp, EEP_ANTDIV_control);
|
|
/* main_lnaconf, alt_lnaconf, main_tb, alt_tb */
|
|
regval = OS_REG_READ(ah, AR_PHY_MC_GAIN_CTRL);
|
|
regval &= (~ANT_DIV_CONTROL_ALL); /* clear bit 25~30 */
|
|
regval |= (value & 0x3f) << ANT_DIV_CONTROL_ALL_S;
|
|
/* enable_lnadiv */
|
|
regval &= (~MULTICHAIN_GAIN_CTRL__ENABLE_ANT_DIV_LNADIV__MASK);
|
|
regval |= ((value >> 6) & 0x1) <<
|
|
MULTICHAIN_GAIN_CTRL__ENABLE_ANT_DIV_LNADIV__SHIFT;
|
|
if ( enable == TRUE ) {
|
|
regval |= ANT_DIV_ENABLE;
|
|
}
|
|
OS_REG_WRITE(ah, AR_PHY_MC_GAIN_CTRL, regval);
|
|
|
|
/* enable fast_div */
|
|
regval = OS_REG_READ(ah, AR_PHY_CCK_DETECT);
|
|
regval &= (~BBB_SIG_DETECT__ENABLE_ANT_FAST_DIV__MASK);
|
|
regval |= ((value >> 7) & 0x1) <<
|
|
BBB_SIG_DETECT__ENABLE_ANT_FAST_DIV__SHIFT;
|
|
if ( enable == TRUE ) {
|
|
regval |= FAST_DIV_ENABLE;
|
|
}
|
|
OS_REG_WRITE(ah, AR_PHY_CCK_DETECT, regval);
|
|
|
|
if ( AR_SREV_POSEIDON_11_OR_LATER(ah) ) {
|
|
if (pcap->halAntDivCombSupport) {
|
|
/* If support DivComb, set MAIN to LNA1 and ALT to LNA2 at the first beginning */
|
|
regval = OS_REG_READ(ah, AR_PHY_MC_GAIN_CTRL);
|
|
/* clear bit 25~30 main_lnaconf, alt_lnaconf, main_tb, alt_tb */
|
|
regval &= (~(MULTICHAIN_GAIN_CTRL__ANT_DIV_MAIN_LNACONF__MASK |
|
|
MULTICHAIN_GAIN_CTRL__ANT_DIV_ALT_LNACONF__MASK |
|
|
MULTICHAIN_GAIN_CTRL__ANT_DIV_ALT_GAINTB__MASK |
|
|
MULTICHAIN_GAIN_CTRL__ANT_DIV_MAIN_GAINTB__MASK));
|
|
regval |= (HAL_ANT_DIV_COMB_LNA1 <<
|
|
MULTICHAIN_GAIN_CTRL__ANT_DIV_MAIN_LNACONF__SHIFT);
|
|
regval |= (HAL_ANT_DIV_COMB_LNA2 <<
|
|
MULTICHAIN_GAIN_CTRL__ANT_DIV_ALT_LNACONF__SHIFT);
|
|
OS_REG_WRITE(ah, AR_PHY_MC_GAIN_CTRL, regval);
|
|
}
|
|
}
|
|
|
|
return AH_TRUE;
|
|
} else if (AR_SREV_APHRODITE(ah)) {
|
|
ahp->ah_lna_div_use_bt_ant_enable = enable;
|
|
if (enable) {
|
|
OS_REG_SET_BIT(ah, AR_PHY_MC_GAIN_CTRL, ANT_DIV_ENABLE);
|
|
OS_REG_SET_BIT(ah, AR_PHY_MC_GAIN_CTRL, (1 << MULTICHAIN_GAIN_CTRL__ENABLE_ANT_SW_RX_PROT__SHIFT));
|
|
OS_REG_SET_BIT(ah, AR_PHY_CCK_DETECT, AR_PHY_CCK_DETECT_BB_ENABLE_ANT_FAST_DIV);
|
|
OS_REG_SET_BIT(ah, AR_PHY_RESTART, RESTART__ENABLE_ANT_FAST_DIV_M2FLAG__MASK);
|
|
OS_REG_SET_BIT(ah, AR_BTCOEX_WL_LNADIV, AR_BTCOEX_WL_LNADIV_FORCE_ON);
|
|
} else {
|
|
OS_REG_CLR_BIT(ah, AR_PHY_MC_GAIN_CTRL, ANT_DIV_ENABLE);
|
|
OS_REG_CLR_BIT(ah, AR_PHY_MC_GAIN_CTRL, (1 << MULTICHAIN_GAIN_CTRL__ENABLE_ANT_SW_RX_PROT__SHIFT));
|
|
OS_REG_CLR_BIT(ah, AR_PHY_CCK_DETECT, AR_PHY_CCK_DETECT_BB_ENABLE_ANT_FAST_DIV);
|
|
OS_REG_CLR_BIT(ah, AR_PHY_RESTART, RESTART__ENABLE_ANT_FAST_DIV_M2FLAG__MASK);
|
|
OS_REG_CLR_BIT(ah, AR_BTCOEX_WL_LNADIV, AR_BTCOEX_WL_LNADIV_FORCE_ON);
|
|
|
|
regval = OS_REG_READ(ah, AR_PHY_MC_GAIN_CTRL);
|
|
regval &= (~(MULTICHAIN_GAIN_CTRL__ANT_DIV_MAIN_LNACONF__MASK |
|
|
MULTICHAIN_GAIN_CTRL__ANT_DIV_ALT_LNACONF__MASK |
|
|
MULTICHAIN_GAIN_CTRL__ANT_DIV_ALT_GAINTB__MASK |
|
|
MULTICHAIN_GAIN_CTRL__ANT_DIV_MAIN_GAINTB__MASK));
|
|
regval |= (HAL_ANT_DIV_COMB_LNA1 <<
|
|
MULTICHAIN_GAIN_CTRL__ANT_DIV_MAIN_LNACONF__SHIFT);
|
|
regval |= (HAL_ANT_DIV_COMB_LNA2 <<
|
|
MULTICHAIN_GAIN_CTRL__ANT_DIV_ALT_LNACONF__SHIFT);
|
|
|
|
OS_REG_WRITE(ah, AR_PHY_MC_GAIN_CTRL, regval);
|
|
}
|
|
return AH_TRUE;
|
|
}
|
|
return AH_TRUE;
|
|
}
|
|
#endif /* ATH_ANT_DIV_COMB */
|