blob: d5026e4f484b580131758646ab95d508cd52ce79 [file] [log] [blame]
Sujithf1dc5602008-10-29 10:16:30 +05301/*
Sujithcee075a2009-03-13 09:07:23 +05302 * Copyright (c) 2008-2009 Atheros Communications Inc.
Sujithf1dc5602008-10-29 10:16:30 +05303 *
4 * Permission to use, copy, modify, and/or distribute this software for any
5 * purpose with or without fee is hereby granted, provided that the above
6 * copyright notice and this permission notice appear in all copies.
7 *
8 * THE SOFTWARE IS PROVIDED "AS IS" AND THE AUTHOR DISCLAIMS ALL WARRANTIES
9 * WITH REGARD TO THIS SOFTWARE INCLUDING ALL IMPLIED WARRANTIES OF
10 * MERCHANTABILITY AND FITNESS. IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR
11 * ANY SPECIAL, DIRECT, INDIRECT, OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES
12 * WHATSOEVER RESULTING FROM LOSS OF USE, DATA OR PROFITS, WHETHER IN AN
13 * ACTION OF CONTRACT, NEGLIGENCE OR OTHER TORTIOUS ACTION, ARISING OUT OF
14 * OR IN CONNECTION WITH THE USE OR PERFORMANCE OF THIS SOFTWARE.
15 */
16
Luis R. Rodriguezc46917b2009-09-13 02:42:02 -070017#include "hw.h"
Sujithf1dc5602008-10-29 10:16:30 +053018
Sujithf1dc5602008-10-29 10:16:30 +053019/* We can tune this as we go by monitoring really low values */
20#define ATH9K_NF_TOO_LOW -60
21
22/* AR5416 may return very high value (like -31 dBm), in those cases the nf
23 * is incorrect and we should use the static NF value. Later we can try to
24 * find out why they are reporting these values */
25
Sujithcbe61d82009-02-09 13:27:12 +053026static bool ath9k_hw_nf_in_range(struct ath_hw *ah, s16 nf)
Sujithf1dc5602008-10-29 10:16:30 +053027{
28 if (nf > ATH9K_NF_TOO_LOW) {
Luis R. Rodriguezc46917b2009-09-13 02:42:02 -070029 ath_print(ath9k_hw_common(ah), ATH_DBG_CALIBRATE,
30 "noise floor value detected (%d) is "
31 "lower than what we think is a "
32 "reasonable value (%d)\n",
33 nf, ATH9K_NF_TOO_LOW);
Sujithf1dc5602008-10-29 10:16:30 +053034 return false;
35 }
36 return true;
37}
38
39static int16_t ath9k_hw_get_nf_hist_mid(int16_t *nfCalBuffer)
40{
41 int16_t nfval;
42 int16_t sort[ATH9K_NF_CAL_HIST_MAX];
43 int i, j;
44
45 for (i = 0; i < ATH9K_NF_CAL_HIST_MAX; i++)
46 sort[i] = nfCalBuffer[i];
47
48 for (i = 0; i < ATH9K_NF_CAL_HIST_MAX - 1; i++) {
49 for (j = 1; j < ATH9K_NF_CAL_HIST_MAX - i; j++) {
50 if (sort[j] > sort[j - 1]) {
51 nfval = sort[j];
52 sort[j] = sort[j - 1];
53 sort[j - 1] = nfval;
54 }
55 }
56 }
57 nfval = sort[(ATH9K_NF_CAL_HIST_MAX - 1) >> 1];
58
59 return nfval;
60}
61
62static void ath9k_hw_update_nfcal_hist_buffer(struct ath9k_nfcal_hist *h,
63 int16_t *nfarray)
64{
65 int i;
66
67 for (i = 0; i < NUM_NF_READINGS; i++) {
68 h[i].nfCalBuffer[h[i].currIndex] = nfarray[i];
69
70 if (++h[i].currIndex >= ATH9K_NF_CAL_HIST_MAX)
71 h[i].currIndex = 0;
72
73 if (h[i].invalidNFcount > 0) {
74 if (nfarray[i] < AR_PHY_CCA_MIN_BAD_VALUE ||
75 nfarray[i] > AR_PHY_CCA_MAX_HIGH_VALUE) {
76 h[i].invalidNFcount = ATH9K_NF_CAL_HIST_MAX;
77 } else {
78 h[i].invalidNFcount--;
79 h[i].privNF = nfarray[i];
80 }
81 } else {
82 h[i].privNF =
83 ath9k_hw_get_nf_hist_mid(h[i].nfCalBuffer);
84 }
85 }
86 return;
87}
88
Sujithcbe61d82009-02-09 13:27:12 +053089static void ath9k_hw_do_getnf(struct ath_hw *ah,
Sujithf1dc5602008-10-29 10:16:30 +053090 int16_t nfarray[NUM_NF_READINGS])
91{
Luis R. Rodriguezc46917b2009-09-13 02:42:02 -070092 struct ath_common *common = ath9k_hw_common(ah);
Sujithf1dc5602008-10-29 10:16:30 +053093 int16_t nf;
94
95 if (AR_SREV_9280_10_OR_LATER(ah))
96 nf = MS(REG_READ(ah, AR_PHY_CCA), AR9280_PHY_MINCCA_PWR);
97 else
98 nf = MS(REG_READ(ah, AR_PHY_CCA), AR_PHY_MINCCA_PWR);
99
100 if (nf & 0x100)
101 nf = 0 - ((nf ^ 0x1ff) + 1);
Luis R. Rodriguezc46917b2009-09-13 02:42:02 -0700102 ath_print(common, ATH_DBG_CALIBRATE,
103 "NF calibrated [ctl] [chain 0] is %d\n", nf);
Sujith2cbfaea2010-03-17 14:25:20 +0530104
105 if (AR_SREV_9271(ah) && (nf >= -114))
106 nf = -116;
107
Sujithf1dc5602008-10-29 10:16:30 +0530108 nfarray[0] = nf;
109
Sujith6398dc02010-03-17 14:25:19 +0530110 if (!AR_SREV_9285(ah) && !AR_SREV_9271(ah)) {
Senthil Balasubramanian793c5922009-01-26 20:28:14 +0530111 if (AR_SREV_9280_10_OR_LATER(ah))
112 nf = MS(REG_READ(ah, AR_PHY_CH1_CCA),
113 AR9280_PHY_CH1_MINCCA_PWR);
114 else
115 nf = MS(REG_READ(ah, AR_PHY_CH1_CCA),
116 AR_PHY_CH1_MINCCA_PWR);
Sujithf1dc5602008-10-29 10:16:30 +0530117
Sujithf1dc5602008-10-29 10:16:30 +0530118 if (nf & 0x100)
119 nf = 0 - ((nf ^ 0x1ff) + 1);
Luis R. Rodriguezc46917b2009-09-13 02:42:02 -0700120 ath_print(common, ATH_DBG_CALIBRATE,
121 "NF calibrated [ctl] [chain 1] is %d\n", nf);
Senthil Balasubramanian793c5922009-01-26 20:28:14 +0530122 nfarray[1] = nf;
123
Vivek Natarajanac88b6e2009-07-23 10:59:57 +0530124 if (!AR_SREV_9280(ah) && !AR_SREV_9287(ah)) {
Senthil Balasubramanian793c5922009-01-26 20:28:14 +0530125 nf = MS(REG_READ(ah, AR_PHY_CH2_CCA),
126 AR_PHY_CH2_MINCCA_PWR);
127 if (nf & 0x100)
128 nf = 0 - ((nf ^ 0x1ff) + 1);
Luis R. Rodriguezc46917b2009-09-13 02:42:02 -0700129 ath_print(common, ATH_DBG_CALIBRATE,
130 "NF calibrated [ctl] [chain 2] is %d\n", nf);
Senthil Balasubramanian793c5922009-01-26 20:28:14 +0530131 nfarray[2] = nf;
132 }
Sujithf1dc5602008-10-29 10:16:30 +0530133 }
134
135 if (AR_SREV_9280_10_OR_LATER(ah))
136 nf = MS(REG_READ(ah, AR_PHY_EXT_CCA),
137 AR9280_PHY_EXT_MINCCA_PWR);
138 else
139 nf = MS(REG_READ(ah, AR_PHY_EXT_CCA),
140 AR_PHY_EXT_MINCCA_PWR);
141
142 if (nf & 0x100)
143 nf = 0 - ((nf ^ 0x1ff) + 1);
Luis R. Rodriguezc46917b2009-09-13 02:42:02 -0700144 ath_print(common, ATH_DBG_CALIBRATE,
145 "NF calibrated [ext] [chain 0] is %d\n", nf);
Sujith2cbfaea2010-03-17 14:25:20 +0530146
147 if (AR_SREV_9271(ah) && (nf >= -114))
148 nf = -116;
149
Sujithf1dc5602008-10-29 10:16:30 +0530150 nfarray[3] = nf;
151
Sujith6398dc02010-03-17 14:25:19 +0530152 if (!AR_SREV_9285(ah) && !AR_SREV_9271(ah)) {
Senthil Balasubramanian793c5922009-01-26 20:28:14 +0530153 if (AR_SREV_9280_10_OR_LATER(ah))
154 nf = MS(REG_READ(ah, AR_PHY_CH1_EXT_CCA),
155 AR9280_PHY_CH1_EXT_MINCCA_PWR);
156 else
157 nf = MS(REG_READ(ah, AR_PHY_CH1_EXT_CCA),
158 AR_PHY_CH1_EXT_MINCCA_PWR);
Sujithf1dc5602008-10-29 10:16:30 +0530159
Sujithf1dc5602008-10-29 10:16:30 +0530160 if (nf & 0x100)
161 nf = 0 - ((nf ^ 0x1ff) + 1);
Luis R. Rodriguezc46917b2009-09-13 02:42:02 -0700162 ath_print(common, ATH_DBG_CALIBRATE,
163 "NF calibrated [ext] [chain 1] is %d\n", nf);
Senthil Balasubramanian793c5922009-01-26 20:28:14 +0530164 nfarray[4] = nf;
165
Vivek Natarajanac88b6e2009-07-23 10:59:57 +0530166 if (!AR_SREV_9280(ah) && !AR_SREV_9287(ah)) {
Senthil Balasubramanian793c5922009-01-26 20:28:14 +0530167 nf = MS(REG_READ(ah, AR_PHY_CH2_EXT_CCA),
168 AR_PHY_CH2_EXT_MINCCA_PWR);
169 if (nf & 0x100)
170 nf = 0 - ((nf ^ 0x1ff) + 1);
Luis R. Rodriguezc46917b2009-09-13 02:42:02 -0700171 ath_print(common, ATH_DBG_CALIBRATE,
172 "NF calibrated [ext] [chain 2] is %d\n", nf);
Senthil Balasubramanian793c5922009-01-26 20:28:14 +0530173 nfarray[5] = nf;
174 }
Sujithf1dc5602008-10-29 10:16:30 +0530175 }
176}
177
Sujithcbe61d82009-02-09 13:27:12 +0530178static bool getNoiseFloorThresh(struct ath_hw *ah,
Luis R. Rodriguez76061ab2008-12-23 15:58:41 -0800179 enum ieee80211_band band,
Sujithf1dc5602008-10-29 10:16:30 +0530180 int16_t *nft)
181{
Luis R. Rodriguez76061ab2008-12-23 15:58:41 -0800182 switch (band) {
183 case IEEE80211_BAND_5GHZ:
Sujithf74df6f2009-02-09 13:27:24 +0530184 *nft = (int8_t)ah->eep_ops->get_eeprom(ah, EEP_NFTHRESH_5);
Sujithf1dc5602008-10-29 10:16:30 +0530185 break;
Luis R. Rodriguez76061ab2008-12-23 15:58:41 -0800186 case IEEE80211_BAND_2GHZ:
Sujithf74df6f2009-02-09 13:27:24 +0530187 *nft = (int8_t)ah->eep_ops->get_eeprom(ah, EEP_NFTHRESH_2);
Sujithf1dc5602008-10-29 10:16:30 +0530188 break;
189 default:
Luis R. Rodriguez76061ab2008-12-23 15:58:41 -0800190 BUG_ON(1);
Sujithf1dc5602008-10-29 10:16:30 +0530191 return false;
192 }
193
194 return true;
195}
196
Sujithcbe61d82009-02-09 13:27:12 +0530197static void ath9k_hw_setup_calibration(struct ath_hw *ah,
Sujithcbfe9462009-04-13 21:56:56 +0530198 struct ath9k_cal_list *currCal)
Sujithf1dc5602008-10-29 10:16:30 +0530199{
Luis R. Rodriguezc46917b2009-09-13 02:42:02 -0700200 struct ath_common *common = ath9k_hw_common(ah);
201
Sujithf1dc5602008-10-29 10:16:30 +0530202 REG_RMW_FIELD(ah, AR_PHY_TIMING_CTRL4(0),
203 AR_PHY_TIMING_CTRL4_IQCAL_LOG_COUNT_MAX,
204 currCal->calData->calCountMax);
205
206 switch (currCal->calData->calType) {
207 case IQ_MISMATCH_CAL:
208 REG_WRITE(ah, AR_PHY_CALMODE, AR_PHY_CALMODE_IQ);
Luis R. Rodriguezc46917b2009-09-13 02:42:02 -0700209 ath_print(common, ATH_DBG_CALIBRATE,
210 "starting IQ Mismatch Calibration\n");
Sujithf1dc5602008-10-29 10:16:30 +0530211 break;
212 case ADC_GAIN_CAL:
213 REG_WRITE(ah, AR_PHY_CALMODE, AR_PHY_CALMODE_ADC_GAIN);
Luis R. Rodriguezc46917b2009-09-13 02:42:02 -0700214 ath_print(common, ATH_DBG_CALIBRATE,
215 "starting ADC Gain Calibration\n");
Sujithf1dc5602008-10-29 10:16:30 +0530216 break;
217 case ADC_DC_CAL:
218 REG_WRITE(ah, AR_PHY_CALMODE, AR_PHY_CALMODE_ADC_DC_PER);
Luis R. Rodriguezc46917b2009-09-13 02:42:02 -0700219 ath_print(common, ATH_DBG_CALIBRATE,
220 "starting ADC DC Calibration\n");
Sujithf1dc5602008-10-29 10:16:30 +0530221 break;
222 case ADC_DC_INIT_CAL:
223 REG_WRITE(ah, AR_PHY_CALMODE, AR_PHY_CALMODE_ADC_DC_INIT);
Luis R. Rodriguezc46917b2009-09-13 02:42:02 -0700224 ath_print(common, ATH_DBG_CALIBRATE,
225 "starting Init ADC DC Calibration\n");
Sujithf1dc5602008-10-29 10:16:30 +0530226 break;
227 }
228
229 REG_SET_BIT(ah, AR_PHY_TIMING_CTRL4(0),
230 AR_PHY_TIMING_CTRL4_DO_CAL);
231}
232
Sujithcbe61d82009-02-09 13:27:12 +0530233static void ath9k_hw_reset_calibration(struct ath_hw *ah,
Sujithcbfe9462009-04-13 21:56:56 +0530234 struct ath9k_cal_list *currCal)
Sujithf1dc5602008-10-29 10:16:30 +0530235{
Sujithf1dc5602008-10-29 10:16:30 +0530236 int i;
237
238 ath9k_hw_setup_calibration(ah, currCal);
239
240 currCal->calState = CAL_RUNNING;
241
242 for (i = 0; i < AR5416_MAX_CHAINS; i++) {
Sujith2660b812009-02-09 13:27:26 +0530243 ah->meas0.sign[i] = 0;
244 ah->meas1.sign[i] = 0;
245 ah->meas2.sign[i] = 0;
246 ah->meas3.sign[i] = 0;
Sujithf1dc5602008-10-29 10:16:30 +0530247 }
248
Sujith2660b812009-02-09 13:27:26 +0530249 ah->cal_samples = 0;
Sujithf1dc5602008-10-29 10:16:30 +0530250}
251
Sujith379f0442009-04-13 21:56:48 +0530252static bool ath9k_hw_per_calibration(struct ath_hw *ah,
Sujithf1dc5602008-10-29 10:16:30 +0530253 struct ath9k_channel *ichan,
254 u8 rxchainmask,
Sujithcbfe9462009-04-13 21:56:56 +0530255 struct ath9k_cal_list *currCal)
Sujithf1dc5602008-10-29 10:16:30 +0530256{
Sujith379f0442009-04-13 21:56:48 +0530257 bool iscaldone = false;
Sujithf1dc5602008-10-29 10:16:30 +0530258
259 if (currCal->calState == CAL_RUNNING) {
260 if (!(REG_READ(ah, AR_PHY_TIMING_CTRL4(0)) &
261 AR_PHY_TIMING_CTRL4_DO_CAL)) {
262
263 currCal->calData->calCollect(ah);
Sujith2660b812009-02-09 13:27:26 +0530264 ah->cal_samples++;
Sujithf1dc5602008-10-29 10:16:30 +0530265
Sujith2660b812009-02-09 13:27:26 +0530266 if (ah->cal_samples >= currCal->calData->calNumSamples) {
Sujithf1dc5602008-10-29 10:16:30 +0530267 int i, numChains = 0;
268 for (i = 0; i < AR5416_MAX_CHAINS; i++) {
269 if (rxchainmask & (1 << i))
270 numChains++;
271 }
272
273 currCal->calData->calPostProc(ah, numChains);
274 ichan->CalValid |= currCal->calData->calType;
275 currCal->calState = CAL_DONE;
Sujith379f0442009-04-13 21:56:48 +0530276 iscaldone = true;
Sujithf1dc5602008-10-29 10:16:30 +0530277 } else {
278 ath9k_hw_setup_calibration(ah, currCal);
279 }
280 }
281 } else if (!(ichan->CalValid & currCal->calData->calType)) {
282 ath9k_hw_reset_calibration(ah, currCal);
283 }
Sujith379f0442009-04-13 21:56:48 +0530284
285 return iscaldone;
Sujithf1dc5602008-10-29 10:16:30 +0530286}
287
Luis R. Rodriguezc9e27d92008-12-23 15:58:42 -0800288/* Assumes you are talking about the currently configured channel */
Sujithcbe61d82009-02-09 13:27:12 +0530289static bool ath9k_hw_iscal_supported(struct ath_hw *ah,
Sujithcbfe9462009-04-13 21:56:56 +0530290 enum ath9k_cal_types calType)
Sujithf1dc5602008-10-29 10:16:30 +0530291{
Luis R. Rodriguezb002a4a2009-09-13 00:03:27 -0700292 struct ieee80211_conf *conf = &ath9k_hw_common(ah)->hw->conf;
Sujithf1dc5602008-10-29 10:16:30 +0530293
Sujith2660b812009-02-09 13:27:26 +0530294 switch (calType & ah->supp_cals) {
Luis R. Rodriguezc9e27d92008-12-23 15:58:42 -0800295 case IQ_MISMATCH_CAL: /* Both 2 GHz and 5 GHz support OFDM */
296 return true;
Sujithf1dc5602008-10-29 10:16:30 +0530297 case ADC_GAIN_CAL:
298 case ADC_DC_CAL:
Sujitha451aa62009-04-13 21:56:43 +0530299 if (!(conf->channel->band == IEEE80211_BAND_2GHZ &&
300 conf_is_ht20(conf)))
Luis R. Rodriguezc9e27d92008-12-23 15:58:42 -0800301 return true;
Sujithf1dc5602008-10-29 10:16:30 +0530302 break;
303 }
Luis R. Rodriguezc9e27d92008-12-23 15:58:42 -0800304 return false;
Sujithf1dc5602008-10-29 10:16:30 +0530305}
306
Sujithcbe61d82009-02-09 13:27:12 +0530307static void ath9k_hw_iqcal_collect(struct ath_hw *ah)
Sujithf1dc5602008-10-29 10:16:30 +0530308{
Sujithf1dc5602008-10-29 10:16:30 +0530309 int i;
310
311 for (i = 0; i < AR5416_MAX_CHAINS; i++) {
Sujith2660b812009-02-09 13:27:26 +0530312 ah->totalPowerMeasI[i] +=
Sujithf1dc5602008-10-29 10:16:30 +0530313 REG_READ(ah, AR_PHY_CAL_MEAS_0(i));
Sujith2660b812009-02-09 13:27:26 +0530314 ah->totalPowerMeasQ[i] +=
Sujithf1dc5602008-10-29 10:16:30 +0530315 REG_READ(ah, AR_PHY_CAL_MEAS_1(i));
Sujith2660b812009-02-09 13:27:26 +0530316 ah->totalIqCorrMeas[i] +=
Sujithf1dc5602008-10-29 10:16:30 +0530317 (int32_t) REG_READ(ah, AR_PHY_CAL_MEAS_2(i));
Luis R. Rodriguezc46917b2009-09-13 02:42:02 -0700318 ath_print(ath9k_hw_common(ah), ATH_DBG_CALIBRATE,
319 "%d: Chn %d pmi=0x%08x;pmq=0x%08x;iqcm=0x%08x;\n",
320 ah->cal_samples, i, ah->totalPowerMeasI[i],
321 ah->totalPowerMeasQ[i],
322 ah->totalIqCorrMeas[i]);
Sujithf1dc5602008-10-29 10:16:30 +0530323 }
324}
325
Sujithcbe61d82009-02-09 13:27:12 +0530326static void ath9k_hw_adc_gaincal_collect(struct ath_hw *ah)
Sujithf1dc5602008-10-29 10:16:30 +0530327{
Sujithf1dc5602008-10-29 10:16:30 +0530328 int i;
329
330 for (i = 0; i < AR5416_MAX_CHAINS; i++) {
Sujith2660b812009-02-09 13:27:26 +0530331 ah->totalAdcIOddPhase[i] +=
Sujithf1dc5602008-10-29 10:16:30 +0530332 REG_READ(ah, AR_PHY_CAL_MEAS_0(i));
Sujith2660b812009-02-09 13:27:26 +0530333 ah->totalAdcIEvenPhase[i] +=
Sujithf1dc5602008-10-29 10:16:30 +0530334 REG_READ(ah, AR_PHY_CAL_MEAS_1(i));
Sujith2660b812009-02-09 13:27:26 +0530335 ah->totalAdcQOddPhase[i] +=
Sujithf1dc5602008-10-29 10:16:30 +0530336 REG_READ(ah, AR_PHY_CAL_MEAS_2(i));
Sujith2660b812009-02-09 13:27:26 +0530337 ah->totalAdcQEvenPhase[i] +=
Sujithf1dc5602008-10-29 10:16:30 +0530338 REG_READ(ah, AR_PHY_CAL_MEAS_3(i));
339
Luis R. Rodriguezc46917b2009-09-13 02:42:02 -0700340 ath_print(ath9k_hw_common(ah), ATH_DBG_CALIBRATE,
341 "%d: Chn %d oddi=0x%08x; eveni=0x%08x; "
342 "oddq=0x%08x; evenq=0x%08x;\n",
343 ah->cal_samples, i,
344 ah->totalAdcIOddPhase[i],
345 ah->totalAdcIEvenPhase[i],
346 ah->totalAdcQOddPhase[i],
347 ah->totalAdcQEvenPhase[i]);
Sujithf1dc5602008-10-29 10:16:30 +0530348 }
349}
350
Sujithcbe61d82009-02-09 13:27:12 +0530351static void ath9k_hw_adc_dccal_collect(struct ath_hw *ah)
Sujithf1dc5602008-10-29 10:16:30 +0530352{
Sujithf1dc5602008-10-29 10:16:30 +0530353 int i;
354
355 for (i = 0; i < AR5416_MAX_CHAINS; i++) {
Sujith2660b812009-02-09 13:27:26 +0530356 ah->totalAdcDcOffsetIOddPhase[i] +=
Sujithf1dc5602008-10-29 10:16:30 +0530357 (int32_t) REG_READ(ah, AR_PHY_CAL_MEAS_0(i));
Sujith2660b812009-02-09 13:27:26 +0530358 ah->totalAdcDcOffsetIEvenPhase[i] +=
Sujithf1dc5602008-10-29 10:16:30 +0530359 (int32_t) REG_READ(ah, AR_PHY_CAL_MEAS_1(i));
Sujith2660b812009-02-09 13:27:26 +0530360 ah->totalAdcDcOffsetQOddPhase[i] +=
Sujithf1dc5602008-10-29 10:16:30 +0530361 (int32_t) REG_READ(ah, AR_PHY_CAL_MEAS_2(i));
Sujith2660b812009-02-09 13:27:26 +0530362 ah->totalAdcDcOffsetQEvenPhase[i] +=
Sujithf1dc5602008-10-29 10:16:30 +0530363 (int32_t) REG_READ(ah, AR_PHY_CAL_MEAS_3(i));
364
Luis R. Rodriguezc46917b2009-09-13 02:42:02 -0700365 ath_print(ath9k_hw_common(ah), ATH_DBG_CALIBRATE,
366 "%d: Chn %d oddi=0x%08x; eveni=0x%08x; "
367 "oddq=0x%08x; evenq=0x%08x;\n",
368 ah->cal_samples, i,
369 ah->totalAdcDcOffsetIOddPhase[i],
370 ah->totalAdcDcOffsetIEvenPhase[i],
371 ah->totalAdcDcOffsetQOddPhase[i],
372 ah->totalAdcDcOffsetQEvenPhase[i]);
Sujithf1dc5602008-10-29 10:16:30 +0530373 }
374}
375
Sujithcbe61d82009-02-09 13:27:12 +0530376static void ath9k_hw_iqcalibrate(struct ath_hw *ah, u8 numChains)
Sujithf1dc5602008-10-29 10:16:30 +0530377{
Luis R. Rodriguezc46917b2009-09-13 02:42:02 -0700378 struct ath_common *common = ath9k_hw_common(ah);
Sujithf1dc5602008-10-29 10:16:30 +0530379 u32 powerMeasQ, powerMeasI, iqCorrMeas;
380 u32 qCoffDenom, iCoffDenom;
381 int32_t qCoff, iCoff;
382 int iqCorrNeg, i;
383
384 for (i = 0; i < numChains; i++) {
Sujith2660b812009-02-09 13:27:26 +0530385 powerMeasI = ah->totalPowerMeasI[i];
386 powerMeasQ = ah->totalPowerMeasQ[i];
387 iqCorrMeas = ah->totalIqCorrMeas[i];
Sujithf1dc5602008-10-29 10:16:30 +0530388
Luis R. Rodriguezc46917b2009-09-13 02:42:02 -0700389 ath_print(common, ATH_DBG_CALIBRATE,
390 "Starting IQ Cal and Correction for Chain %d\n",
391 i);
Sujithf1dc5602008-10-29 10:16:30 +0530392
Luis R. Rodriguezc46917b2009-09-13 02:42:02 -0700393 ath_print(common, ATH_DBG_CALIBRATE,
394 "Orignal: Chn %diq_corr_meas = 0x%08x\n",
395 i, ah->totalIqCorrMeas[i]);
Sujithf1dc5602008-10-29 10:16:30 +0530396
397 iqCorrNeg = 0;
398
399 if (iqCorrMeas > 0x80000000) {
400 iqCorrMeas = (0xffffffff - iqCorrMeas) + 1;
401 iqCorrNeg = 1;
402 }
403
Luis R. Rodriguezc46917b2009-09-13 02:42:02 -0700404 ath_print(common, ATH_DBG_CALIBRATE,
405 "Chn %d pwr_meas_i = 0x%08x\n", i, powerMeasI);
406 ath_print(common, ATH_DBG_CALIBRATE,
407 "Chn %d pwr_meas_q = 0x%08x\n", i, powerMeasQ);
408 ath_print(common, ATH_DBG_CALIBRATE, "iqCorrNeg is 0x%08x\n",
409 iqCorrNeg);
Sujithf1dc5602008-10-29 10:16:30 +0530410
411 iCoffDenom = (powerMeasI / 2 + powerMeasQ / 2) / 128;
412 qCoffDenom = powerMeasQ / 64;
413
Vivek Natarajan0b98eaa2009-09-18 15:03:42 +0530414 if ((powerMeasQ != 0) && (iCoffDenom != 0) &&
415 (qCoffDenom != 0)) {
Sujithf1dc5602008-10-29 10:16:30 +0530416 iCoff = iqCorrMeas / iCoffDenom;
417 qCoff = powerMeasI / qCoffDenom - 64;
Luis R. Rodriguezc46917b2009-09-13 02:42:02 -0700418 ath_print(common, ATH_DBG_CALIBRATE,
419 "Chn %d iCoff = 0x%08x\n", i, iCoff);
420 ath_print(common, ATH_DBG_CALIBRATE,
421 "Chn %d qCoff = 0x%08x\n", i, qCoff);
Sujithf1dc5602008-10-29 10:16:30 +0530422
423 iCoff = iCoff & 0x3f;
Luis R. Rodriguezc46917b2009-09-13 02:42:02 -0700424 ath_print(common, ATH_DBG_CALIBRATE,
425 "New: Chn %d iCoff = 0x%08x\n", i, iCoff);
Sujithf1dc5602008-10-29 10:16:30 +0530426 if (iqCorrNeg == 0x0)
427 iCoff = 0x40 - iCoff;
428
429 if (qCoff > 15)
430 qCoff = 15;
431 else if (qCoff <= -16)
432 qCoff = 16;
433
Luis R. Rodriguezc46917b2009-09-13 02:42:02 -0700434 ath_print(common, ATH_DBG_CALIBRATE,
435 "Chn %d : iCoff = 0x%x qCoff = 0x%x\n",
436 i, iCoff, qCoff);
Sujithf1dc5602008-10-29 10:16:30 +0530437
438 REG_RMW_FIELD(ah, AR_PHY_TIMING_CTRL4(i),
439 AR_PHY_TIMING_CTRL4_IQCORR_Q_I_COFF,
440 iCoff);
441 REG_RMW_FIELD(ah, AR_PHY_TIMING_CTRL4(i),
442 AR_PHY_TIMING_CTRL4_IQCORR_Q_Q_COFF,
443 qCoff);
Luis R. Rodriguezc46917b2009-09-13 02:42:02 -0700444 ath_print(common, ATH_DBG_CALIBRATE,
445 "IQ Cal and Correction done for Chain %d\n",
446 i);
Sujithf1dc5602008-10-29 10:16:30 +0530447 }
448 }
449
450 REG_SET_BIT(ah, AR_PHY_TIMING_CTRL4(0),
451 AR_PHY_TIMING_CTRL4_IQCORR_ENABLE);
452}
453
Sujithcbe61d82009-02-09 13:27:12 +0530454static void ath9k_hw_adc_gaincal_calibrate(struct ath_hw *ah, u8 numChains)
Sujithf1dc5602008-10-29 10:16:30 +0530455{
Luis R. Rodriguezc46917b2009-09-13 02:42:02 -0700456 struct ath_common *common = ath9k_hw_common(ah);
Sujithf1dc5602008-10-29 10:16:30 +0530457 u32 iOddMeasOffset, iEvenMeasOffset, qOddMeasOffset, qEvenMeasOffset;
458 u32 qGainMismatch, iGainMismatch, val, i;
459
460 for (i = 0; i < numChains; i++) {
Sujith2660b812009-02-09 13:27:26 +0530461 iOddMeasOffset = ah->totalAdcIOddPhase[i];
462 iEvenMeasOffset = ah->totalAdcIEvenPhase[i];
463 qOddMeasOffset = ah->totalAdcQOddPhase[i];
464 qEvenMeasOffset = ah->totalAdcQEvenPhase[i];
Sujithf1dc5602008-10-29 10:16:30 +0530465
Luis R. Rodriguezc46917b2009-09-13 02:42:02 -0700466 ath_print(common, ATH_DBG_CALIBRATE,
467 "Starting ADC Gain Cal for Chain %d\n", i);
Sujithf1dc5602008-10-29 10:16:30 +0530468
Luis R. Rodriguezc46917b2009-09-13 02:42:02 -0700469 ath_print(common, ATH_DBG_CALIBRATE,
470 "Chn %d pwr_meas_odd_i = 0x%08x\n", i,
471 iOddMeasOffset);
472 ath_print(common, ATH_DBG_CALIBRATE,
473 "Chn %d pwr_meas_even_i = 0x%08x\n", i,
474 iEvenMeasOffset);
475 ath_print(common, ATH_DBG_CALIBRATE,
476 "Chn %d pwr_meas_odd_q = 0x%08x\n", i,
477 qOddMeasOffset);
478 ath_print(common, ATH_DBG_CALIBRATE,
479 "Chn %d pwr_meas_even_q = 0x%08x\n", i,
480 qEvenMeasOffset);
Sujithf1dc5602008-10-29 10:16:30 +0530481
482 if (iOddMeasOffset != 0 && qEvenMeasOffset != 0) {
483 iGainMismatch =
484 ((iEvenMeasOffset * 32) /
485 iOddMeasOffset) & 0x3f;
486 qGainMismatch =
487 ((qOddMeasOffset * 32) /
488 qEvenMeasOffset) & 0x3f;
489
Luis R. Rodriguezc46917b2009-09-13 02:42:02 -0700490 ath_print(common, ATH_DBG_CALIBRATE,
491 "Chn %d gain_mismatch_i = 0x%08x\n", i,
492 iGainMismatch);
493 ath_print(common, ATH_DBG_CALIBRATE,
494 "Chn %d gain_mismatch_q = 0x%08x\n", i,
495 qGainMismatch);
Sujithf1dc5602008-10-29 10:16:30 +0530496
497 val = REG_READ(ah, AR_PHY_NEW_ADC_DC_GAIN_CORR(i));
498 val &= 0xfffff000;
499 val |= (qGainMismatch) | (iGainMismatch << 6);
500 REG_WRITE(ah, AR_PHY_NEW_ADC_DC_GAIN_CORR(i), val);
501
Luis R. Rodriguezc46917b2009-09-13 02:42:02 -0700502 ath_print(common, ATH_DBG_CALIBRATE,
503 "ADC Gain Cal done for Chain %d\n", i);
Sujithf1dc5602008-10-29 10:16:30 +0530504 }
505 }
506
507 REG_WRITE(ah, AR_PHY_NEW_ADC_DC_GAIN_CORR(0),
508 REG_READ(ah, AR_PHY_NEW_ADC_DC_GAIN_CORR(0)) |
509 AR_PHY_NEW_ADC_GAIN_CORR_ENABLE);
510}
511
Sujithcbe61d82009-02-09 13:27:12 +0530512static void ath9k_hw_adc_dccal_calibrate(struct ath_hw *ah, u8 numChains)
Sujithf1dc5602008-10-29 10:16:30 +0530513{
Luis R. Rodriguezc46917b2009-09-13 02:42:02 -0700514 struct ath_common *common = ath9k_hw_common(ah);
Sujithf1dc5602008-10-29 10:16:30 +0530515 u32 iOddMeasOffset, iEvenMeasOffset, val, i;
516 int32_t qOddMeasOffset, qEvenMeasOffset, qDcMismatch, iDcMismatch;
Sujithcbfe9462009-04-13 21:56:56 +0530517 const struct ath9k_percal_data *calData =
Sujith2660b812009-02-09 13:27:26 +0530518 ah->cal_list_curr->calData;
Sujithf1dc5602008-10-29 10:16:30 +0530519 u32 numSamples =
520 (1 << (calData->calCountMax + 5)) * calData->calNumSamples;
521
522 for (i = 0; i < numChains; i++) {
Sujith2660b812009-02-09 13:27:26 +0530523 iOddMeasOffset = ah->totalAdcDcOffsetIOddPhase[i];
524 iEvenMeasOffset = ah->totalAdcDcOffsetIEvenPhase[i];
525 qOddMeasOffset = ah->totalAdcDcOffsetQOddPhase[i];
526 qEvenMeasOffset = ah->totalAdcDcOffsetQEvenPhase[i];
Sujithf1dc5602008-10-29 10:16:30 +0530527
Luis R. Rodriguezc46917b2009-09-13 02:42:02 -0700528 ath_print(common, ATH_DBG_CALIBRATE,
529 "Starting ADC DC Offset Cal for Chain %d\n", i);
Sujithf1dc5602008-10-29 10:16:30 +0530530
Luis R. Rodriguezc46917b2009-09-13 02:42:02 -0700531 ath_print(common, ATH_DBG_CALIBRATE,
532 "Chn %d pwr_meas_odd_i = %d\n", i,
533 iOddMeasOffset);
534 ath_print(common, ATH_DBG_CALIBRATE,
535 "Chn %d pwr_meas_even_i = %d\n", i,
536 iEvenMeasOffset);
537 ath_print(common, ATH_DBG_CALIBRATE,
538 "Chn %d pwr_meas_odd_q = %d\n", i,
539 qOddMeasOffset);
540 ath_print(common, ATH_DBG_CALIBRATE,
541 "Chn %d pwr_meas_even_q = %d\n", i,
542 qEvenMeasOffset);
Sujithf1dc5602008-10-29 10:16:30 +0530543
544 iDcMismatch = (((iEvenMeasOffset - iOddMeasOffset) * 2) /
545 numSamples) & 0x1ff;
546 qDcMismatch = (((qOddMeasOffset - qEvenMeasOffset) * 2) /
547 numSamples) & 0x1ff;
548
Luis R. Rodriguezc46917b2009-09-13 02:42:02 -0700549 ath_print(common, ATH_DBG_CALIBRATE,
550 "Chn %d dc_offset_mismatch_i = 0x%08x\n", i,
551 iDcMismatch);
552 ath_print(common, ATH_DBG_CALIBRATE,
553 "Chn %d dc_offset_mismatch_q = 0x%08x\n", i,
554 qDcMismatch);
Sujithf1dc5602008-10-29 10:16:30 +0530555
556 val = REG_READ(ah, AR_PHY_NEW_ADC_DC_GAIN_CORR(i));
557 val &= 0xc0000fff;
558 val |= (qDcMismatch << 12) | (iDcMismatch << 21);
559 REG_WRITE(ah, AR_PHY_NEW_ADC_DC_GAIN_CORR(i), val);
560
Luis R. Rodriguezc46917b2009-09-13 02:42:02 -0700561 ath_print(common, ATH_DBG_CALIBRATE,
562 "ADC DC Offset Cal done for Chain %d\n", i);
Sujithf1dc5602008-10-29 10:16:30 +0530563 }
564
565 REG_WRITE(ah, AR_PHY_NEW_ADC_DC_GAIN_CORR(0),
566 REG_READ(ah, AR_PHY_NEW_ADC_DC_GAIN_CORR(0)) |
567 AR_PHY_NEW_ADC_DC_OFFSET_CORR_ENABLE);
568}
569
Luis R. Rodriguezc9e27d92008-12-23 15:58:42 -0800570/* This is done for the currently configured channel */
Sujithcbe61d82009-02-09 13:27:12 +0530571bool ath9k_hw_reset_calvalid(struct ath_hw *ah)
Sujithf1dc5602008-10-29 10:16:30 +0530572{
Luis R. Rodriguezc46917b2009-09-13 02:42:02 -0700573 struct ath_common *common = ath9k_hw_common(ah);
574 struct ieee80211_conf *conf = &common->hw->conf;
Sujithcbfe9462009-04-13 21:56:56 +0530575 struct ath9k_cal_list *currCal = ah->cal_list_curr;
Sujithf1dc5602008-10-29 10:16:30 +0530576
Sujith2660b812009-02-09 13:27:26 +0530577 if (!ah->curchan)
Luis R. Rodriguezc9e27d92008-12-23 15:58:42 -0800578 return true;
Sujithf1dc5602008-10-29 10:16:30 +0530579
580 if (!AR_SREV_9100(ah) && !AR_SREV_9160_10_OR_LATER(ah))
Luis R. Rodriguezc9e27d92008-12-23 15:58:42 -0800581 return true;
Sujithf1dc5602008-10-29 10:16:30 +0530582
583 if (currCal == NULL)
Luis R. Rodriguezc9e27d92008-12-23 15:58:42 -0800584 return true;
Sujithf1dc5602008-10-29 10:16:30 +0530585
586 if (currCal->calState != CAL_DONE) {
Luis R. Rodriguezc46917b2009-09-13 02:42:02 -0700587 ath_print(common, ATH_DBG_CALIBRATE,
588 "Calibration state incorrect, %d\n",
589 currCal->calState);
Luis R. Rodriguezc9e27d92008-12-23 15:58:42 -0800590 return true;
Sujithf1dc5602008-10-29 10:16:30 +0530591 }
592
Luis R. Rodriguezc9e27d92008-12-23 15:58:42 -0800593 if (!ath9k_hw_iscal_supported(ah, currCal->calData->calType))
594 return true;
Sujithf1dc5602008-10-29 10:16:30 +0530595
Luis R. Rodriguezc46917b2009-09-13 02:42:02 -0700596 ath_print(common, ATH_DBG_CALIBRATE,
597 "Resetting Cal %d state for channel %u\n",
598 currCal->calData->calType, conf->channel->center_freq);
Sujithf1dc5602008-10-29 10:16:30 +0530599
Sujith2660b812009-02-09 13:27:26 +0530600 ah->curchan->CalValid &= ~currCal->calData->calType;
Sujithf1dc5602008-10-29 10:16:30 +0530601 currCal->calState = CAL_WAITING;
602
Luis R. Rodriguezc9e27d92008-12-23 15:58:42 -0800603 return false;
Sujithf1dc5602008-10-29 10:16:30 +0530604}
Luis R. Rodriguez7322fd12009-09-23 23:07:00 -0400605EXPORT_SYMBOL(ath9k_hw_reset_calvalid);
Sujithf1dc5602008-10-29 10:16:30 +0530606
Sujithcbe61d82009-02-09 13:27:12 +0530607void ath9k_hw_start_nfcal(struct ath_hw *ah)
Sujithf1dc5602008-10-29 10:16:30 +0530608{
609 REG_SET_BIT(ah, AR_PHY_AGC_CONTROL,
610 AR_PHY_AGC_CONTROL_ENABLE_NF);
611 REG_SET_BIT(ah, AR_PHY_AGC_CONTROL,
612 AR_PHY_AGC_CONTROL_NO_UPDATE_NF);
613 REG_SET_BIT(ah, AR_PHY_AGC_CONTROL, AR_PHY_AGC_CONTROL_NF);
614}
615
Sujithcbe61d82009-02-09 13:27:12 +0530616void ath9k_hw_loadnf(struct ath_hw *ah, struct ath9k_channel *chan)
Sujithf1dc5602008-10-29 10:16:30 +0530617{
618 struct ath9k_nfcal_hist *h;
619 int i, j;
620 int32_t val;
621 const u32 ar5416_cca_regs[6] = {
622 AR_PHY_CCA,
623 AR_PHY_CH1_CCA,
624 AR_PHY_CH2_CCA,
625 AR_PHY_EXT_CCA,
626 AR_PHY_CH1_EXT_CCA,
627 AR_PHY_CH2_EXT_CCA
628 };
Senthil Balasubramaniance143bb2009-09-17 09:27:33 +0530629 u8 chainmask, rx_chain_status;
Sujithf1dc5602008-10-29 10:16:30 +0530630
Senthil Balasubramaniance143bb2009-09-17 09:27:33 +0530631 rx_chain_status = REG_READ(ah, AR_PHY_RX_CHAINMASK);
Sujith6398dc02010-03-17 14:25:19 +0530632 if (AR_SREV_9285(ah) || AR_SREV_9271(ah))
Sujith5dad40c2009-01-23 11:20:55 +0530633 chainmask = 0x9;
Senthil Balasubramaniance143bb2009-09-17 09:27:33 +0530634 else if (AR_SREV_9280(ah) || AR_SREV_9287(ah)) {
635 if ((rx_chain_status & 0x2) || (rx_chain_status & 0x4))
636 chainmask = 0x1B;
637 else
638 chainmask = 0x09;
639 } else {
640 if (rx_chain_status & 0x4)
641 chainmask = 0x3F;
642 else if (rx_chain_status & 0x2)
643 chainmask = 0x1B;
644 else
645 chainmask = 0x09;
646 }
Sujithf1dc5602008-10-29 10:16:30 +0530647
Sujithf1dc5602008-10-29 10:16:30 +0530648 h = ah->nfCalHist;
Sujithf1dc5602008-10-29 10:16:30 +0530649
650 for (i = 0; i < NUM_NF_READINGS; i++) {
651 if (chainmask & (1 << i)) {
652 val = REG_READ(ah, ar5416_cca_regs[i]);
653 val &= 0xFFFFFE00;
654 val |= (((u32) (h[i].privNF) << 1) & 0x1ff);
655 REG_WRITE(ah, ar5416_cca_regs[i], val);
656 }
657 }
658
659 REG_CLR_BIT(ah, AR_PHY_AGC_CONTROL,
660 AR_PHY_AGC_CONTROL_ENABLE_NF);
661 REG_CLR_BIT(ah, AR_PHY_AGC_CONTROL,
662 AR_PHY_AGC_CONTROL_NO_UPDATE_NF);
663 REG_SET_BIT(ah, AR_PHY_AGC_CONTROL, AR_PHY_AGC_CONTROL_NF);
664
Senthil Balasubramanian63a75b92009-09-18 15:07:03 +0530665 for (j = 0; j < 5; j++) {
Sujithf1dc5602008-10-29 10:16:30 +0530666 if ((REG_READ(ah, AR_PHY_AGC_CONTROL) &
667 AR_PHY_AGC_CONTROL_NF) == 0)
668 break;
Senthil Balasubramanian63a75b92009-09-18 15:07:03 +0530669 udelay(50);
Sujithf1dc5602008-10-29 10:16:30 +0530670 }
671
672 for (i = 0; i < NUM_NF_READINGS; i++) {
673 if (chainmask & (1 << i)) {
674 val = REG_READ(ah, ar5416_cca_regs[i]);
675 val &= 0xFFFFFE00;
676 val |= (((u32) (-50) << 1) & 0x1ff);
677 REG_WRITE(ah, ar5416_cca_regs[i], val);
678 }
679 }
680}
681
Sujithcbe61d82009-02-09 13:27:12 +0530682int16_t ath9k_hw_getnf(struct ath_hw *ah,
Sujithf1dc5602008-10-29 10:16:30 +0530683 struct ath9k_channel *chan)
684{
Luis R. Rodriguezc46917b2009-09-13 02:42:02 -0700685 struct ath_common *common = ath9k_hw_common(ah);
Sujithf1dc5602008-10-29 10:16:30 +0530686 int16_t nf, nfThresh;
687 int16_t nfarray[NUM_NF_READINGS] = { 0 };
688 struct ath9k_nfcal_hist *h;
Luis R. Rodriguez76061ab2008-12-23 15:58:41 -0800689 struct ieee80211_channel *c = chan->chan;
Sujithf1dc5602008-10-29 10:16:30 +0530690
691 chan->channelFlags &= (~CHANNEL_CW_INT);
692 if (REG_READ(ah, AR_PHY_AGC_CONTROL) & AR_PHY_AGC_CONTROL_NF) {
Luis R. Rodriguezc46917b2009-09-13 02:42:02 -0700693 ath_print(common, ATH_DBG_CALIBRATE,
694 "NF did not complete in calibration window\n");
Sujithf1dc5602008-10-29 10:16:30 +0530695 nf = 0;
696 chan->rawNoiseFloor = nf;
697 return chan->rawNoiseFloor;
698 } else {
699 ath9k_hw_do_getnf(ah, nfarray);
700 nf = nfarray[0];
Luis R. Rodriguez76061ab2008-12-23 15:58:41 -0800701 if (getNoiseFloorThresh(ah, c->band, &nfThresh)
Sujithf1dc5602008-10-29 10:16:30 +0530702 && nf > nfThresh) {
Luis R. Rodriguezc46917b2009-09-13 02:42:02 -0700703 ath_print(common, ATH_DBG_CALIBRATE,
704 "noise floor failed detected; "
705 "detected %d, threshold %d\n",
706 nf, nfThresh);
Sujithf1dc5602008-10-29 10:16:30 +0530707 chan->channelFlags |= CHANNEL_CW_INT;
708 }
709 }
710
Sujithf1dc5602008-10-29 10:16:30 +0530711 h = ah->nfCalHist;
Sujithf1dc5602008-10-29 10:16:30 +0530712
713 ath9k_hw_update_nfcal_hist_buffer(h, nfarray);
714 chan->rawNoiseFloor = h[0].privNF;
715
716 return chan->rawNoiseFloor;
717}
718
Sujithcbe61d82009-02-09 13:27:12 +0530719void ath9k_init_nfcal_hist_buffer(struct ath_hw *ah)
Sujithf1dc5602008-10-29 10:16:30 +0530720{
721 int i, j;
Senthil Balasubramaniana59b5a52009-07-14 20:17:07 -0400722 s16 noise_floor;
723
724 if (AR_SREV_9280(ah))
725 noise_floor = AR_PHY_CCA_MAX_AR9280_GOOD_VALUE;
Sujith6398dc02010-03-17 14:25:19 +0530726 else if (AR_SREV_9285(ah) || AR_SREV_9271(ah))
Senthil Balasubramaniana59b5a52009-07-14 20:17:07 -0400727 noise_floor = AR_PHY_CCA_MAX_AR9285_GOOD_VALUE;
Vivek Natarajan6170cd52009-09-17 09:24:24 +0530728 else if (AR_SREV_9287(ah))
729 noise_floor = AR_PHY_CCA_MAX_AR9287_GOOD_VALUE;
Senthil Balasubramaniana59b5a52009-07-14 20:17:07 -0400730 else
731 noise_floor = AR_PHY_CCA_MAX_AR5416_GOOD_VALUE;
Sujithf1dc5602008-10-29 10:16:30 +0530732
733 for (i = 0; i < NUM_NF_READINGS; i++) {
734 ah->nfCalHist[i].currIndex = 0;
Senthil Balasubramaniana59b5a52009-07-14 20:17:07 -0400735 ah->nfCalHist[i].privNF = noise_floor;
Sujithf1dc5602008-10-29 10:16:30 +0530736 ah->nfCalHist[i].invalidNFcount =
737 AR_PHY_CCA_FILTERWINDOW_LENGTH;
738 for (j = 0; j < ATH9K_NF_CAL_HIST_MAX; j++) {
Senthil Balasubramaniana59b5a52009-07-14 20:17:07 -0400739 ah->nfCalHist[i].nfCalBuffer[j] = noise_floor;
Sujithf1dc5602008-10-29 10:16:30 +0530740 }
741 }
Sujithf1dc5602008-10-29 10:16:30 +0530742}
743
Sujithcbe61d82009-02-09 13:27:12 +0530744s16 ath9k_hw_getchan_noise(struct ath_hw *ah, struct ath9k_channel *chan)
Sujithf1dc5602008-10-29 10:16:30 +0530745{
Sujithf1dc5602008-10-29 10:16:30 +0530746 s16 nf;
747
Luis R. Rodriguez5f8e0772009-01-22 15:16:48 -0800748 if (chan->rawNoiseFloor == 0)
Luis R. Rodrigueze56db712008-12-23 15:58:47 -0800749 nf = -96;
750 else
Luis R. Rodriguez5f8e0772009-01-22 15:16:48 -0800751 nf = chan->rawNoiseFloor;
Sujithf1dc5602008-10-29 10:16:30 +0530752
753 if (!ath9k_hw_nf_in_range(ah, nf))
754 nf = ATH_DEFAULT_NOISE_FLOOR;
755
756 return nf;
757}
Luis R. Rodriguez7322fd12009-09-23 23:07:00 -0400758EXPORT_SYMBOL(ath9k_hw_getchan_noise);
Sujithf1dc5602008-10-29 10:16:30 +0530759
Vivek Natarajan0b98eaa2009-09-18 15:03:42 +0530760static void ath9k_olc_temp_compensation_9287(struct ath_hw *ah)
Senthil Balasubramanian8bd1d072009-02-12 13:57:03 +0530761{
Vivek Natarajan0b98eaa2009-09-18 15:03:42 +0530762 u32 rddata;
763 int32_t delta, currPDADC, slope;
Senthil Balasubramanian8bd1d072009-02-12 13:57:03 +0530764
765 rddata = REG_READ(ah, AR_PHY_TX_PWRCTRL4);
Senthil Balasubramanian8bd1d072009-02-12 13:57:03 +0530766 currPDADC = MS(rddata, AR_PHY_TX_PWRCTRL_PD_AVG_OUT);
767
Vivek Natarajan0b98eaa2009-09-18 15:03:42 +0530768 if (ah->initPDADC == 0 || currPDADC == 0) {
769 /*
770 * Zero value indicates that no frames have been transmitted yet,
771 * can't do temperature compensation until frames are transmitted.
772 */
773 return;
774 } else {
775 slope = ah->eep_ops->get_eeprom(ah, EEP_TEMPSENSE_SLOPE);
776
777 if (slope == 0) { /* to avoid divide by zero case */
778 delta = 0;
779 } else {
780 delta = ((currPDADC - ah->initPDADC)*4) / slope;
781 }
782 REG_RMW_FIELD(ah, AR_PHY_CH0_TX_PWRCTRL11,
783 AR_PHY_TX_PWRCTRL_OLPC_TEMP_COMP, delta);
784 REG_RMW_FIELD(ah, AR_PHY_CH1_TX_PWRCTRL11,
785 AR_PHY_TX_PWRCTRL_OLPC_TEMP_COMP, delta);
786 }
787}
788
789static void ath9k_olc_temp_compensation(struct ath_hw *ah)
790{
791 u32 rddata, i;
792 int delta, currPDADC, regval;
Senthil Balasubramanian8bd1d072009-02-12 13:57:03 +0530793
Vivek Natarajandb91f2e2009-08-14 11:27:16 +0530794 if (OLC_FOR_AR9287_10_LATER) {
Vivek Natarajan0b98eaa2009-09-18 15:03:42 +0530795 ath9k_olc_temp_compensation_9287(ah);
796 } else {
797 rddata = REG_READ(ah, AR_PHY_TX_PWRCTRL4);
798 currPDADC = MS(rddata, AR_PHY_TX_PWRCTRL_PD_AVG_OUT);
799
Vivek Natarajandb91f2e2009-08-14 11:27:16 +0530800 if (ah->initPDADC == 0 || currPDADC == 0) {
801 return;
802 } else {
Vivek Natarajan0b98eaa2009-09-18 15:03:42 +0530803 if (ah->eep_ops->get_eeprom(ah, EEP_DAC_HPWR_5G))
804 delta = (currPDADC - ah->initPDADC + 4) / 8;
Vivek Natarajandb91f2e2009-08-14 11:27:16 +0530805 else
Vivek Natarajan0b98eaa2009-09-18 15:03:42 +0530806 delta = (currPDADC - ah->initPDADC + 5) / 10;
Senthil Balasubramanian8bd1d072009-02-12 13:57:03 +0530807
Vivek Natarajan0b98eaa2009-09-18 15:03:42 +0530808 if (delta != ah->PDADCdelta) {
809 ah->PDADCdelta = delta;
810 for (i = 1; i < AR9280_TX_GAIN_TABLE_SIZE; i++) {
811 regval = ah->originalGain[i] - delta;
812 if (regval < 0)
813 regval = 0;
Vivek Natarajandb91f2e2009-08-14 11:27:16 +0530814
Vivek Natarajan0b98eaa2009-09-18 15:03:42 +0530815 REG_RMW_FIELD(ah,
816 AR_PHY_TX_GAIN_TBL1 + i * 4,
817 AR_PHY_TX_GAIN, regval);
818 }
Vivek Natarajandb91f2e2009-08-14 11:27:16 +0530819 }
Senthil Balasubramanian8bd1d072009-02-12 13:57:03 +0530820 }
821 }
822}
823
Luis R. Rodriguez62268112009-10-07 16:22:19 -0400824static void ath9k_hw_9271_pa_cal(struct ath_hw *ah, bool is_reset)
Luis R. Rodriguezd7e7d222009-08-03 23:14:12 -0400825{
826 u32 regVal;
827 unsigned int i;
828 u32 regList [][2] = {
829 { 0x786c, 0 },
830 { 0x7854, 0 },
831 { 0x7820, 0 },
832 { 0x7824, 0 },
833 { 0x7868, 0 },
834 { 0x783c, 0 },
835 { 0x7838, 0 } ,
836 { 0x7828, 0 } ,
837 };
838
839 for (i = 0; i < ARRAY_SIZE(regList); i++)
840 regList[i][1] = REG_READ(ah, regList[i][0]);
841
842 regVal = REG_READ(ah, 0x7834);
843 regVal &= (~(0x1));
844 REG_WRITE(ah, 0x7834, regVal);
845 regVal = REG_READ(ah, 0x9808);
846 regVal |= (0x1 << 27);
847 REG_WRITE(ah, 0x9808, regVal);
848
849 /* 786c,b23,1, pwddac=1 */
850 REG_RMW_FIELD(ah, AR9285_AN_TOP3, AR9285_AN_TOP3_PWDDAC, 1);
851 /* 7854, b5,1, pdrxtxbb=1 */
852 REG_RMW_FIELD(ah, AR9285_AN_RXTXBB1, AR9285_AN_RXTXBB1_PDRXTXBB1, 1);
853 /* 7854, b7,1, pdv2i=1 */
854 REG_RMW_FIELD(ah, AR9285_AN_RXTXBB1, AR9285_AN_RXTXBB1_PDV2I, 1);
855 /* 7854, b8,1, pddacinterface=1 */
856 REG_RMW_FIELD(ah, AR9285_AN_RXTXBB1, AR9285_AN_RXTXBB1_PDDACIF, 1);
857 /* 7824,b12,0, offcal=0 */
858 REG_RMW_FIELD(ah, AR9285_AN_RF2G2, AR9285_AN_RF2G2_OFFCAL, 0);
859 /* 7838, b1,0, pwddb=0 */
860 REG_RMW_FIELD(ah, AR9285_AN_RF2G7, AR9285_AN_RF2G7_PWDDB, 0);
861 /* 7820,b11,0, enpacal=0 */
862 REG_RMW_FIELD(ah, AR9285_AN_RF2G1, AR9285_AN_RF2G1_ENPACAL, 0);
863 /* 7820,b25,1, pdpadrv1=0 */
864 REG_RMW_FIELD(ah, AR9285_AN_RF2G1, AR9285_AN_RF2G1_PDPADRV1, 0);
865 /* 7820,b24,0, pdpadrv2=0 */
866 REG_RMW_FIELD(ah, AR9285_AN_RF2G1,AR9285_AN_RF2G1_PDPADRV2,0);
867 /* 7820,b23,0, pdpaout=0 */
868 REG_RMW_FIELD(ah, AR9285_AN_RF2G1, AR9285_AN_RF2G1_PDPAOUT, 0);
869 /* 783c,b14-16,7, padrvgn2tab_0=7 */
870 REG_RMW_FIELD(ah, AR9285_AN_RF2G8,AR9285_AN_RF2G8_PADRVGN2TAB0, 7);
871 /*
872 * 7838,b29-31,0, padrvgn1tab_0=0
873 * does not matter since we turn it off
874 */
875 REG_RMW_FIELD(ah, AR9285_AN_RF2G7,AR9285_AN_RF2G7_PADRVGN2TAB0, 0);
876
877 REG_RMW_FIELD(ah, AR9285_AN_RF2G3, AR9271_AN_RF2G3_CCOMP, 0xfff);
878
879 /* Set:
880 * localmode=1,bmode=1,bmoderxtx=1,synthon=1,
881 * txon=1,paon=1,oscon=1,synthon_force=1
882 */
883 REG_WRITE(ah, AR9285_AN_TOP2, 0xca0358a0);
884 udelay(30);
885 REG_RMW_FIELD(ah, AR9285_AN_RF2G6, AR9271_AN_RF2G6_OFFS, 0);
886
887 /* find off_6_1; */
Luis R. Rodriguez1d9c1852009-10-27 12:59:37 -0400888 for (i = 6; i > 0; i--) {
Luis R. Rodriguezd7e7d222009-08-03 23:14:12 -0400889 regVal = REG_READ(ah, 0x7834);
890 regVal |= (1 << (20 + i));
891 REG_WRITE(ah, 0x7834, regVal);
892 udelay(1);
893 //regVal = REG_READ(ah, 0x7834);
894 regVal &= (~(0x1 << (20 + i)));
895 regVal |= (MS(REG_READ(ah, 0x7840), AR9285_AN_RXTXBB1_SPARE9)
896 << (20 + i));
897 REG_WRITE(ah, 0x7834, regVal);
898 }
899
Luis R. Rodriguez62268112009-10-07 16:22:19 -0400900 regVal = (regVal >>20) & 0x7f;
901
902 /* Update PA cal info */
903 if ((!is_reset) && (ah->pacal_info.prev_offset == regVal)) {
904 if (ah->pacal_info.max_skipcount < MAX_PACAL_SKIPCOUNT)
905 ah->pacal_info.max_skipcount =
906 2 * ah->pacal_info.max_skipcount;
907 ah->pacal_info.skipcount = ah->pacal_info.max_skipcount;
908 } else {
909 ah->pacal_info.max_skipcount = 1;
910 ah->pacal_info.skipcount = 0;
911 ah->pacal_info.prev_offset = regVal;
912 }
Luis R. Rodriguezd7e7d222009-08-03 23:14:12 -0400913
914 regVal = REG_READ(ah, 0x7834);
915 regVal |= 0x1;
916 REG_WRITE(ah, 0x7834, regVal);
917 regVal = REG_READ(ah, 0x9808);
918 regVal &= (~(0x1 << 27));
919 REG_WRITE(ah, 0x9808, regVal);
920
921 for (i = 0; i < ARRAY_SIZE(regList); i++)
922 REG_WRITE(ah, regList[i][0], regList[i][1]);
923}
924
Sujitha13883b2009-08-26 08:39:40 +0530925static inline void ath9k_hw_9285_pa_cal(struct ath_hw *ah, bool is_reset)
Senthil Balasubramaniane7594072008-12-08 19:43:48 +0530926{
Luis R. Rodriguezc46917b2009-09-13 02:42:02 -0700927 struct ath_common *common = ath9k_hw_common(ah);
Senthil Balasubramaniane7594072008-12-08 19:43:48 +0530928 u32 regVal;
929 int i, offset, offs_6_1, offs_0;
930 u32 ccomp_org, reg_field;
931 u32 regList[][2] = {
932 { 0x786c, 0 },
933 { 0x7854, 0 },
934 { 0x7820, 0 },
935 { 0x7824, 0 },
936 { 0x7868, 0 },
937 { 0x783c, 0 },
938 { 0x7838, 0 },
939 };
940
Luis R. Rodriguezc46917b2009-09-13 02:42:02 -0700941 ath_print(common, ATH_DBG_CALIBRATE, "Running PA Calibration\n");
Sujitha13883b2009-08-26 08:39:40 +0530942
Sujith20caf0d2009-08-26 08:39:52 +0530943 /* PA CAL is not needed for high power solution */
944 if (ah->eep_ops->get_eeprom(ah, EEP_TXGAIN_TYPE) ==
945 AR5416_EEP_TXGAIN_HIGH_POWER)
946 return;
947
Senthil Balasubramaniane7594072008-12-08 19:43:48 +0530948 if (AR_SREV_9285_11(ah)) {
949 REG_WRITE(ah, AR9285_AN_TOP4, (AR9285_AN_TOP4_DEFAULT | 0x14));
950 udelay(10);
951 }
952
953 for (i = 0; i < ARRAY_SIZE(regList); i++)
954 regList[i][1] = REG_READ(ah, regList[i][0]);
955
956 regVal = REG_READ(ah, 0x7834);
957 regVal &= (~(0x1));
958 REG_WRITE(ah, 0x7834, regVal);
959 regVal = REG_READ(ah, 0x9808);
960 regVal |= (0x1 << 27);
961 REG_WRITE(ah, 0x9808, regVal);
962
963 REG_RMW_FIELD(ah, AR9285_AN_TOP3, AR9285_AN_TOP3_PWDDAC, 1);
964 REG_RMW_FIELD(ah, AR9285_AN_RXTXBB1, AR9285_AN_RXTXBB1_PDRXTXBB1, 1);
965 REG_RMW_FIELD(ah, AR9285_AN_RXTXBB1, AR9285_AN_RXTXBB1_PDV2I, 1);
966 REG_RMW_FIELD(ah, AR9285_AN_RXTXBB1, AR9285_AN_RXTXBB1_PDDACIF, 1);
967 REG_RMW_FIELD(ah, AR9285_AN_RF2G2, AR9285_AN_RF2G2_OFFCAL, 0);
968 REG_RMW_FIELD(ah, AR9285_AN_RF2G7, AR9285_AN_RF2G7_PWDDB, 0);
969 REG_RMW_FIELD(ah, AR9285_AN_RF2G1, AR9285_AN_RF2G1_ENPACAL, 0);
Sujith0abb0962009-08-26 08:39:50 +0530970 REG_RMW_FIELD(ah, AR9285_AN_RF2G1, AR9285_AN_RF2G1_PDPADRV1, 0);
Senthil Balasubramaniane7594072008-12-08 19:43:48 +0530971 REG_RMW_FIELD(ah, AR9285_AN_RF2G1, AR9285_AN_RF2G1_PDPADRV2, 0);
972 REG_RMW_FIELD(ah, AR9285_AN_RF2G1, AR9285_AN_RF2G1_PDPAOUT, 0);
973 REG_RMW_FIELD(ah, AR9285_AN_RF2G8, AR9285_AN_RF2G8_PADRVGN2TAB0, 7);
974 REG_RMW_FIELD(ah, AR9285_AN_RF2G7, AR9285_AN_RF2G7_PADRVGN2TAB0, 0);
975 ccomp_org = MS(REG_READ(ah, AR9285_AN_RF2G6), AR9285_AN_RF2G6_CCOMP);
Sujith0abb0962009-08-26 08:39:50 +0530976 REG_RMW_FIELD(ah, AR9285_AN_RF2G6, AR9285_AN_RF2G6_CCOMP, 0xf);
Senthil Balasubramaniane7594072008-12-08 19:43:48 +0530977
978 REG_WRITE(ah, AR9285_AN_TOP2, 0xca0358a0);
979 udelay(30);
980 REG_RMW_FIELD(ah, AR9285_AN_RF2G6, AR9285_AN_RF2G6_OFFS, 0);
981 REG_RMW_FIELD(ah, AR9285_AN_RF2G3, AR9285_AN_RF2G3_PDVCCOMP, 0);
982
983 for (i = 6; i > 0; i--) {
984 regVal = REG_READ(ah, 0x7834);
985 regVal |= (1 << (19 + i));
986 REG_WRITE(ah, 0x7834, regVal);
987 udelay(1);
Sujithedbf51f2009-09-17 09:28:41 +0530988 regVal = REG_READ(ah, 0x7834);
Senthil Balasubramaniane7594072008-12-08 19:43:48 +0530989 regVal &= (~(0x1 << (19 + i)));
990 reg_field = MS(REG_READ(ah, 0x7840), AR9285_AN_RXTXBB1_SPARE9);
991 regVal |= (reg_field << (19 + i));
992 REG_WRITE(ah, 0x7834, regVal);
993 }
994
995 REG_RMW_FIELD(ah, AR9285_AN_RF2G3, AR9285_AN_RF2G3_PDVCCOMP, 1);
996 udelay(1);
997 reg_field = MS(REG_READ(ah, AR9285_AN_RF2G9), AR9285_AN_RXTXBB1_SPARE9);
998 REG_RMW_FIELD(ah, AR9285_AN_RF2G3, AR9285_AN_RF2G3_PDVCCOMP, reg_field);
999 offs_6_1 = MS(REG_READ(ah, AR9285_AN_RF2G6), AR9285_AN_RF2G6_OFFS);
1000 offs_0 = MS(REG_READ(ah, AR9285_AN_RF2G3), AR9285_AN_RF2G3_PDVCCOMP);
1001
1002 offset = (offs_6_1<<1) | offs_0;
1003 offset = offset - 0;
1004 offs_6_1 = offset>>1;
1005 offs_0 = offset & 1;
1006
Sujitha13883b2009-08-26 08:39:40 +05301007 if ((!is_reset) && (ah->pacal_info.prev_offset == offset)) {
1008 if (ah->pacal_info.max_skipcount < MAX_PACAL_SKIPCOUNT)
1009 ah->pacal_info.max_skipcount =
1010 2 * ah->pacal_info.max_skipcount;
1011 ah->pacal_info.skipcount = ah->pacal_info.max_skipcount;
1012 } else {
1013 ah->pacal_info.max_skipcount = 1;
1014 ah->pacal_info.skipcount = 0;
1015 ah->pacal_info.prev_offset = offset;
1016 }
1017
Senthil Balasubramaniane7594072008-12-08 19:43:48 +05301018 REG_RMW_FIELD(ah, AR9285_AN_RF2G6, AR9285_AN_RF2G6_OFFS, offs_6_1);
1019 REG_RMW_FIELD(ah, AR9285_AN_RF2G3, AR9285_AN_RF2G3_PDVCCOMP, offs_0);
1020
1021 regVal = REG_READ(ah, 0x7834);
1022 regVal |= 0x1;
1023 REG_WRITE(ah, 0x7834, regVal);
1024 regVal = REG_READ(ah, 0x9808);
1025 regVal &= (~(0x1 << 27));
1026 REG_WRITE(ah, 0x9808, regVal);
1027
1028 for (i = 0; i < ARRAY_SIZE(regList); i++)
1029 REG_WRITE(ah, regList[i][0], regList[i][1]);
1030
1031 REG_RMW_FIELD(ah, AR9285_AN_RF2G6, AR9285_AN_RF2G6_CCOMP, ccomp_org);
1032
1033 if (AR_SREV_9285_11(ah))
1034 REG_WRITE(ah, AR9285_AN_TOP4, AR9285_AN_TOP4_DEFAULT);
1035
1036}
1037
Senthil Balasubramanian4e845162009-03-06 11:24:10 +05301038bool ath9k_hw_calibrate(struct ath_hw *ah, struct ath9k_channel *chan,
Sujith379f0442009-04-13 21:56:48 +05301039 u8 rxchainmask, bool longcal)
Senthil Balasubramanian4e845162009-03-06 11:24:10 +05301040{
Sujith379f0442009-04-13 21:56:48 +05301041 bool iscaldone = true;
Sujithcbfe9462009-04-13 21:56:56 +05301042 struct ath9k_cal_list *currCal = ah->cal_list_curr;
Senthil Balasubramanian4e845162009-03-06 11:24:10 +05301043
Senthil Balasubramanian4e845162009-03-06 11:24:10 +05301044 if (currCal &&
1045 (currCal->calState == CAL_RUNNING ||
1046 currCal->calState == CAL_WAITING)) {
Sujith379f0442009-04-13 21:56:48 +05301047 iscaldone = ath9k_hw_per_calibration(ah, chan,
1048 rxchainmask, currCal);
1049 if (iscaldone) {
Senthil Balasubramanian4e845162009-03-06 11:24:10 +05301050 ah->cal_list_curr = currCal = currCal->calNext;
1051
1052 if (currCal->calState == CAL_WAITING) {
Sujith379f0442009-04-13 21:56:48 +05301053 iscaldone = false;
Senthil Balasubramanian4e845162009-03-06 11:24:10 +05301054 ath9k_hw_reset_calibration(ah, currCal);
1055 }
1056 }
1057 }
1058
Luis R. Rodriguezd7e7d222009-08-03 23:14:12 -04001059 /* Do NF cal only at longer intervals */
Senthil Balasubramanian4e845162009-03-06 11:24:10 +05301060 if (longcal) {
Luis R. Rodriguezd7e7d222009-08-03 23:14:12 -04001061 /* Do periodic PAOffset Cal */
Sujith02afa2a2010-03-17 14:25:21 +05301062 if (AR_SREV_9271(ah)) {
1063 if (!ah->pacal_info.skipcount)
1064 ath9k_hw_9271_pa_cal(ah, false);
1065 else
1066 ah->pacal_info.skipcount--;
1067 } else if (AR_SREV_9285_11_OR_LATER(ah)) {
Sujitha13883b2009-08-26 08:39:40 +05301068 if (!ah->pacal_info.skipcount)
1069 ath9k_hw_9285_pa_cal(ah, false);
1070 else
1071 ah->pacal_info.skipcount--;
1072 }
Senthil Balasubramanian4e845162009-03-06 11:24:10 +05301073
Vivek Natarajanac88b6e2009-07-23 10:59:57 +05301074 if (OLC_FOR_AR9280_20_LATER || OLC_FOR_AR9287_10_LATER)
Senthil Balasubramanian4e845162009-03-06 11:24:10 +05301075 ath9k_olc_temp_compensation(ah);
Luis R. Rodriguezd7e7d222009-08-03 23:14:12 -04001076
1077 /* Get the value from the previous NF cal and update history buffer */
Senthil Balasubramanian4e845162009-03-06 11:24:10 +05301078 ath9k_hw_getnf(ah, chan);
Luis R. Rodriguezd7e7d222009-08-03 23:14:12 -04001079
1080 /*
1081 * Load the NF from history buffer of the current channel.
1082 * NF is slow time-variant, so it is OK to use a historical value.
1083 */
Senthil Balasubramanian4e845162009-03-06 11:24:10 +05301084 ath9k_hw_loadnf(ah, ah->curchan);
Luis R. Rodriguezd7e7d222009-08-03 23:14:12 -04001085
Senthil Balasubramanian4e845162009-03-06 11:24:10 +05301086 ath9k_hw_start_nfcal(ah);
Senthil Balasubramanian4e845162009-03-06 11:24:10 +05301087 }
1088
Sujith379f0442009-04-13 21:56:48 +05301089 return iscaldone;
Senthil Balasubramanian4e845162009-03-06 11:24:10 +05301090}
Luis R. Rodriguez7322fd12009-09-23 23:07:00 -04001091EXPORT_SYMBOL(ath9k_hw_calibrate);
Senthil Balasubramanian4e845162009-03-06 11:24:10 +05301092
Luis R. Rodriguezb57df122009-10-07 16:22:18 -04001093/* Carrier leakage Calibration fix */
Luis R. Rodriguez15cc0f12009-03-09 22:09:42 -04001094static bool ar9285_clc(struct ath_hw *ah, struct ath9k_channel *chan)
Senthil Balasubramanian4e845162009-03-06 11:24:10 +05301095{
Luis R. Rodriguezc46917b2009-09-13 02:42:02 -07001096 struct ath_common *common = ath9k_hw_common(ah);
1097
Senthil Balasubramanian4e845162009-03-06 11:24:10 +05301098 REG_SET_BIT(ah, AR_PHY_CL_CAL_CTL, AR_PHY_CL_CAL_ENABLE);
Sujithdb2f63f2009-04-13 21:56:41 +05301099 if (IS_CHAN_HT20(chan)) {
Senthil Balasubramanian4e845162009-03-06 11:24:10 +05301100 REG_SET_BIT(ah, AR_PHY_CL_CAL_CTL, AR_PHY_PARALLEL_CAL_ENABLE);
1101 REG_SET_BIT(ah, AR_PHY_TURBO, AR_PHY_FC_DYN2040_EN);
1102 REG_CLR_BIT(ah, AR_PHY_AGC_CONTROL,
1103 AR_PHY_AGC_CONTROL_FLTR_CAL);
1104 REG_CLR_BIT(ah, AR_PHY_TPCRG1, AR_PHY_TPCRG1_PD_CAL_ENABLE);
1105 REG_SET_BIT(ah, AR_PHY_AGC_CONTROL, AR_PHY_AGC_CONTROL_CAL);
1106 if (!ath9k_hw_wait(ah, AR_PHY_AGC_CONTROL,
1107 AR_PHY_AGC_CONTROL_CAL, 0, AH_WAIT_TIMEOUT)) {
Luis R. Rodriguezc46917b2009-09-13 02:42:02 -07001108 ath_print(common, ATH_DBG_CALIBRATE, "offset "
1109 "calibration failed to complete in "
1110 "1ms; noisy ??\n");
Senthil Balasubramanian4e845162009-03-06 11:24:10 +05301111 return false;
1112 }
1113 REG_CLR_BIT(ah, AR_PHY_TURBO, AR_PHY_FC_DYN2040_EN);
1114 REG_CLR_BIT(ah, AR_PHY_CL_CAL_CTL, AR_PHY_PARALLEL_CAL_ENABLE);
1115 REG_CLR_BIT(ah, AR_PHY_CL_CAL_CTL, AR_PHY_CL_CAL_ENABLE);
1116 }
1117 REG_CLR_BIT(ah, AR_PHY_ADC_CTL, AR_PHY_ADC_CTL_OFF_PWDADC);
1118 REG_SET_BIT(ah, AR_PHY_AGC_CONTROL, AR_PHY_AGC_CONTROL_FLTR_CAL);
1119 REG_SET_BIT(ah, AR_PHY_TPCRG1, AR_PHY_TPCRG1_PD_CAL_ENABLE);
1120 REG_SET_BIT(ah, AR_PHY_AGC_CONTROL, AR_PHY_AGC_CONTROL_CAL);
1121 if (!ath9k_hw_wait(ah, AR_PHY_AGC_CONTROL, AR_PHY_AGC_CONTROL_CAL,
1122 0, AH_WAIT_TIMEOUT)) {
Luis R. Rodriguezc46917b2009-09-13 02:42:02 -07001123 ath_print(common, ATH_DBG_CALIBRATE, "offset calibration "
1124 "failed to complete in 1ms; noisy ??\n");
Senthil Balasubramanian4e845162009-03-06 11:24:10 +05301125 return false;
1126 }
1127
1128 REG_SET_BIT(ah, AR_PHY_ADC_CTL, AR_PHY_ADC_CTL_OFF_PWDADC);
1129 REG_CLR_BIT(ah, AR_PHY_CL_CAL_CTL, AR_PHY_CL_CAL_ENABLE);
1130 REG_CLR_BIT(ah, AR_PHY_AGC_CONTROL, AR_PHY_AGC_CONTROL_FLTR_CAL);
1131
1132 return true;
1133}
1134
Sujith04d19dd2009-04-13 21:56:59 +05301135bool ath9k_hw_init_cal(struct ath_hw *ah, struct ath9k_channel *chan)
Sujithf1dc5602008-10-29 10:16:30 +05301136{
Luis R. Rodriguezc46917b2009-09-13 02:42:02 -07001137 struct ath_common *common = ath9k_hw_common(ah);
1138
Luis R. Rodriguezb57df122009-10-07 16:22:18 -04001139 if (AR_SREV_9271(ah) || AR_SREV_9285_12_OR_LATER(ah)) {
Senthil Balasubramanian4e845162009-03-06 11:24:10 +05301140 if (!ar9285_clc(ah, chan))
1141 return false;
Sujith04d19dd2009-04-13 21:56:59 +05301142 } else {
1143 if (AR_SREV_9280_10_OR_LATER(ah)) {
Vivek Natarajanac88b6e2009-07-23 10:59:57 +05301144 if (!AR_SREV_9287_10_OR_LATER(ah))
1145 REG_CLR_BIT(ah, AR_PHY_ADC_CTL,
1146 AR_PHY_ADC_CTL_OFF_PWDADC);
1147 REG_SET_BIT(ah, AR_PHY_AGC_CONTROL,
1148 AR_PHY_AGC_CONTROL_FLTR_CAL);
Sujith04d19dd2009-04-13 21:56:59 +05301149 }
Sujithedf7c062009-02-12 10:06:49 +05301150
Sujith04d19dd2009-04-13 21:56:59 +05301151 /* Calibrate the AGC */
Sujithedf7c062009-02-12 10:06:49 +05301152 REG_WRITE(ah, AR_PHY_AGC_CONTROL,
Sujith04d19dd2009-04-13 21:56:59 +05301153 REG_READ(ah, AR_PHY_AGC_CONTROL) |
1154 AR_PHY_AGC_CONTROL_CAL);
Sujithedf7c062009-02-12 10:06:49 +05301155
Sujith04d19dd2009-04-13 21:56:59 +05301156 /* Poll for offset calibration complete */
1157 if (!ath9k_hw_wait(ah, AR_PHY_AGC_CONTROL, AR_PHY_AGC_CONTROL_CAL,
1158 0, AH_WAIT_TIMEOUT)) {
Luis R. Rodriguezc46917b2009-09-13 02:42:02 -07001159 ath_print(common, ATH_DBG_CALIBRATE,
1160 "offset calibration failed to "
1161 "complete in 1ms; noisy environment?\n");
Sujithedf7c062009-02-12 10:06:49 +05301162 return false;
1163 }
1164
Sujith04d19dd2009-04-13 21:56:59 +05301165 if (AR_SREV_9280_10_OR_LATER(ah)) {
Vivek Natarajanac88b6e2009-07-23 10:59:57 +05301166 if (!AR_SREV_9287_10_OR_LATER(ah))
1167 REG_SET_BIT(ah, AR_PHY_ADC_CTL,
1168 AR_PHY_ADC_CTL_OFF_PWDADC);
1169 REG_CLR_BIT(ah, AR_PHY_AGC_CONTROL,
1170 AR_PHY_AGC_CONTROL_FLTR_CAL);
Sujith04d19dd2009-04-13 21:56:59 +05301171 }
Sujithedf7c062009-02-12 10:06:49 +05301172 }
1173
1174 /* Do PA Calibration */
Luis R. Rodriguez62268112009-10-07 16:22:19 -04001175 if (AR_SREV_9271(ah))
1176 ath9k_hw_9271_pa_cal(ah, true);
1177 else if (AR_SREV_9285_11_OR_LATER(ah))
Sujitha13883b2009-08-26 08:39:40 +05301178 ath9k_hw_9285_pa_cal(ah, true);
Senthil Balasubramaniane7594072008-12-08 19:43:48 +05301179
Sujith04d19dd2009-04-13 21:56:59 +05301180 /* Do NF Calibration after DC offset and other calibrations */
Sujithf1dc5602008-10-29 10:16:30 +05301181 REG_WRITE(ah, AR_PHY_AGC_CONTROL,
Sujith04d19dd2009-04-13 21:56:59 +05301182 REG_READ(ah, AR_PHY_AGC_CONTROL) | AR_PHY_AGC_CONTROL_NF);
Sujithf1dc5602008-10-29 10:16:30 +05301183
Sujith2660b812009-02-09 13:27:26 +05301184 ah->cal_list = ah->cal_list_last = ah->cal_list_curr = NULL;
Sujithf1dc5602008-10-29 10:16:30 +05301185
Sujith04d19dd2009-04-13 21:56:59 +05301186 /* Enable IQ, ADC Gain and ADC DC offset CALs */
Sujithf1dc5602008-10-29 10:16:30 +05301187 if (AR_SREV_9100(ah) || AR_SREV_9160_10_OR_LATER(ah)) {
Luis R. Rodriguezc9e27d92008-12-23 15:58:42 -08001188 if (ath9k_hw_iscal_supported(ah, ADC_GAIN_CAL)) {
Sujith2660b812009-02-09 13:27:26 +05301189 INIT_CAL(&ah->adcgain_caldata);
1190 INSERT_CAL(ah, &ah->adcgain_caldata);
Luis R. Rodriguezc46917b2009-09-13 02:42:02 -07001191 ath_print(common, ATH_DBG_CALIBRATE,
1192 "enabling ADC Gain Calibration.\n");
Sujithf1dc5602008-10-29 10:16:30 +05301193 }
Luis R. Rodriguezc9e27d92008-12-23 15:58:42 -08001194 if (ath9k_hw_iscal_supported(ah, ADC_DC_CAL)) {
Sujith2660b812009-02-09 13:27:26 +05301195 INIT_CAL(&ah->adcdc_caldata);
1196 INSERT_CAL(ah, &ah->adcdc_caldata);
Luis R. Rodriguezc46917b2009-09-13 02:42:02 -07001197 ath_print(common, ATH_DBG_CALIBRATE,
1198 "enabling ADC DC Calibration.\n");
Sujithf1dc5602008-10-29 10:16:30 +05301199 }
Luis R. Rodriguezc9e27d92008-12-23 15:58:42 -08001200 if (ath9k_hw_iscal_supported(ah, IQ_MISMATCH_CAL)) {
Sujith2660b812009-02-09 13:27:26 +05301201 INIT_CAL(&ah->iq_caldata);
1202 INSERT_CAL(ah, &ah->iq_caldata);
Luis R. Rodriguezc46917b2009-09-13 02:42:02 -07001203 ath_print(common, ATH_DBG_CALIBRATE,
1204 "enabling IQ Calibration.\n");
Sujithf1dc5602008-10-29 10:16:30 +05301205 }
1206
Sujith2660b812009-02-09 13:27:26 +05301207 ah->cal_list_curr = ah->cal_list;
Sujithf1dc5602008-10-29 10:16:30 +05301208
Sujith2660b812009-02-09 13:27:26 +05301209 if (ah->cal_list_curr)
1210 ath9k_hw_reset_calibration(ah, ah->cal_list_curr);
Sujithf1dc5602008-10-29 10:16:30 +05301211 }
1212
Luis R. Rodriguez5f8e0772009-01-22 15:16:48 -08001213 chan->CalValid = 0;
Sujithf1dc5602008-10-29 10:16:30 +05301214
1215 return true;
1216}
1217
Sujithcbfe9462009-04-13 21:56:56 +05301218const struct ath9k_percal_data iq_cal_multi_sample = {
Sujithf1dc5602008-10-29 10:16:30 +05301219 IQ_MISMATCH_CAL,
1220 MAX_CAL_SAMPLES,
1221 PER_MIN_LOG_COUNT,
1222 ath9k_hw_iqcal_collect,
1223 ath9k_hw_iqcalibrate
1224};
Sujithcbfe9462009-04-13 21:56:56 +05301225const struct ath9k_percal_data iq_cal_single_sample = {
Sujithf1dc5602008-10-29 10:16:30 +05301226 IQ_MISMATCH_CAL,
1227 MIN_CAL_SAMPLES,
1228 PER_MAX_LOG_COUNT,
1229 ath9k_hw_iqcal_collect,
1230 ath9k_hw_iqcalibrate
1231};
Sujithcbfe9462009-04-13 21:56:56 +05301232const struct ath9k_percal_data adc_gain_cal_multi_sample = {
Sujithf1dc5602008-10-29 10:16:30 +05301233 ADC_GAIN_CAL,
1234 MAX_CAL_SAMPLES,
1235 PER_MIN_LOG_COUNT,
1236 ath9k_hw_adc_gaincal_collect,
1237 ath9k_hw_adc_gaincal_calibrate
1238};
Sujithcbfe9462009-04-13 21:56:56 +05301239const struct ath9k_percal_data adc_gain_cal_single_sample = {
Sujithf1dc5602008-10-29 10:16:30 +05301240 ADC_GAIN_CAL,
1241 MIN_CAL_SAMPLES,
1242 PER_MAX_LOG_COUNT,
1243 ath9k_hw_adc_gaincal_collect,
1244 ath9k_hw_adc_gaincal_calibrate
1245};
Sujithcbfe9462009-04-13 21:56:56 +05301246const struct ath9k_percal_data adc_dc_cal_multi_sample = {
Sujithf1dc5602008-10-29 10:16:30 +05301247 ADC_DC_CAL,
1248 MAX_CAL_SAMPLES,
1249 PER_MIN_LOG_COUNT,
1250 ath9k_hw_adc_dccal_collect,
1251 ath9k_hw_adc_dccal_calibrate
1252};
Sujithcbfe9462009-04-13 21:56:56 +05301253const struct ath9k_percal_data adc_dc_cal_single_sample = {
Sujithf1dc5602008-10-29 10:16:30 +05301254 ADC_DC_CAL,
1255 MIN_CAL_SAMPLES,
1256 PER_MAX_LOG_COUNT,
1257 ath9k_hw_adc_dccal_collect,
1258 ath9k_hw_adc_dccal_calibrate
1259};
Sujithcbfe9462009-04-13 21:56:56 +05301260const struct ath9k_percal_data adc_init_dc_cal = {
Sujithf1dc5602008-10-29 10:16:30 +05301261 ADC_DC_INIT_CAL,
1262 MIN_CAL_SAMPLES,
1263 INIT_LOG_COUNT,
1264 ath9k_hw_adc_dccal_collect,
1265 ath9k_hw_adc_dccal_calibrate
1266};