blob: 8aecced62ddeeae9d03b7688b462800a4a42e25f [file] [log] [blame]
Pavel Machek66101de2008-10-01 14:36:56 +02001/*
2 * phy_302_calibration.c
3 *
4 * Copyright (C) 2002, 2005 Winbond Electronics Corp.
5 *
6 * modification history
7 * ---------------------------------------------------------------------------
8 * 0.01.001, 2003-04-16, Kevin created
9 *
10 */
11
12/****************** INCLUDE FILES SECTION ***********************************/
Pavel Machek66101de2008-10-01 14:36:56 +020013#include "phy_calibration.h"
Pekka Enbergb5ef0762010-11-01 21:50:06 +020014#include "wbhal.h"
Pekka Enberg72ca8812010-11-01 21:50:05 +020015#include "wb35reg_f.h"
16#include "core.h"
Pavel Machek66101de2008-10-01 14:36:56 +020017
18
19/****************** DEBUG CONSTANT AND MACRO SECTION ************************/
20
21/****************** LOCAL CONSTANT AND MACRO SECTION ************************/
22#define LOOP_TIMES 20
Timofey Trofimov3a4d1b92010-05-25 18:19:54 +040023#define US 1000/* MICROSECOND*/
Pavel Machek66101de2008-10-01 14:36:56 +020024
25#define AG_CONST 0.6072529350
26#define FIXED(X) ((s32)((X) * 32768.0))
Vincent Abel-Grimaltaf69c292011-08-10 01:13:38 +020027#define DEG2RAD(X) (0.017453 * (X))
Pavel Machek66101de2008-10-01 14:36:56 +020028
Timofey Trofimov3a4d1b92010-05-25 18:19:54 +040029static const s32 Angles[] = {
Iker Pedrosade6c7642013-09-16 15:43:24 +020030 FIXED(DEG2RAD(45.0)), FIXED(DEG2RAD(26.565)),
31 FIXED(DEG2RAD(14.0362)), FIXED(DEG2RAD(7.12502)),
32 FIXED(DEG2RAD(3.57633)), FIXED(DEG2RAD(1.78991)),
33 FIXED(DEG2RAD(0.895174)), FIXED(DEG2RAD(0.447614)),
34 FIXED(DEG2RAD(0.223811)), FIXED(DEG2RAD(0.111906)),
35 FIXED(DEG2RAD(0.055953)), FIXED(DEG2RAD(0.027977))
Pavel Machek66101de2008-10-01 14:36:56 +020036};
37
Greg Kroah-Hartman2e04bb72010-06-18 15:30:28 -070038/****************** LOCAL FUNCTION DECLARATION SECTION **********************/
Timofey Trofimov3a4d1b92010-05-25 18:19:54 +040039
40/*
41 * void _phy_rf_write_delay(struct hw_data *phw_data);
42 * void phy_init_rf(struct hw_data *phw_data);
43 */
Pavel Machek66101de2008-10-01 14:36:56 +020044
45/****************** FUNCTION DEFINITION SECTION *****************************/
46
Ebru Akagunduz06c789e2013-10-29 10:08:46 +020047static s32 _s13_to_s32(u32 data)
Pavel Machek66101de2008-10-01 14:36:56 +020048{
Akshay Joshi919ed522011-06-07 04:51:55 -040049 u32 val;
Pavel Machek66101de2008-10-01 14:36:56 +020050
Akshay Joshi919ed522011-06-07 04:51:55 -040051 val = (data & 0x0FFF);
Pavel Machek66101de2008-10-01 14:36:56 +020052
Akshay Joshi919ed522011-06-07 04:51:55 -040053 if ((data & BIT(12)) != 0)
54 val |= 0xFFFFF000;
Pavel Machek66101de2008-10-01 14:36:56 +020055
Vincent Abel-Grimaltaf69c292011-08-10 01:13:38 +020056 return (s32) val;
Pavel Machek66101de2008-10-01 14:36:56 +020057}
58
Pavel Machek66101de2008-10-01 14:36:56 +020059/****************************************************************************/
Ebru Akagunduz06c789e2013-10-29 10:08:46 +020060static s32 _s4_to_s32(u32 data)
Pavel Machek66101de2008-10-01 14:36:56 +020061{
Akshay Joshi919ed522011-06-07 04:51:55 -040062 s32 val;
Pavel Machek66101de2008-10-01 14:36:56 +020063
Akshay Joshi919ed522011-06-07 04:51:55 -040064 val = (data & 0x0007);
Pavel Machek66101de2008-10-01 14:36:56 +020065
Akshay Joshi919ed522011-06-07 04:51:55 -040066 if ((data & BIT(3)) != 0)
67 val |= 0xFFFFFFF8;
Pavel Machek66101de2008-10-01 14:36:56 +020068
Akshay Joshi919ed522011-06-07 04:51:55 -040069 return val;
Pavel Machek66101de2008-10-01 14:36:56 +020070}
71
Ebru Akagunduz06c789e2013-10-29 10:08:46 +020072static u32 _s32_to_s4(s32 data)
Pavel Machek66101de2008-10-01 14:36:56 +020073{
Akshay Joshi919ed522011-06-07 04:51:55 -040074 u32 val;
Pavel Machek66101de2008-10-01 14:36:56 +020075
Akshay Joshi919ed522011-06-07 04:51:55 -040076 if (data > 7)
77 data = 7;
78 else if (data < -8)
79 data = -8;
Pavel Machek66101de2008-10-01 14:36:56 +020080
Akshay Joshi919ed522011-06-07 04:51:55 -040081 val = data & 0x000F;
Pavel Machek66101de2008-10-01 14:36:56 +020082
Akshay Joshi919ed522011-06-07 04:51:55 -040083 return val;
Pavel Machek66101de2008-10-01 14:36:56 +020084}
85
86/****************************************************************************/
Ebru Akagunduz06c789e2013-10-29 10:08:46 +020087static s32 _s5_to_s32(u32 data)
Pavel Machek66101de2008-10-01 14:36:56 +020088{
Akshay Joshi919ed522011-06-07 04:51:55 -040089 s32 val;
Pavel Machek66101de2008-10-01 14:36:56 +020090
Akshay Joshi919ed522011-06-07 04:51:55 -040091 val = (data & 0x000F);
Pavel Machek66101de2008-10-01 14:36:56 +020092
Akshay Joshi919ed522011-06-07 04:51:55 -040093 if ((data & BIT(4)) != 0)
94 val |= 0xFFFFFFF0;
Pavel Machek66101de2008-10-01 14:36:56 +020095
Akshay Joshi919ed522011-06-07 04:51:55 -040096 return val;
Pavel Machek66101de2008-10-01 14:36:56 +020097}
98
Ebru Akagunduz06c789e2013-10-29 10:08:46 +020099static u32 _s32_to_s5(s32 data)
Pavel Machek66101de2008-10-01 14:36:56 +0200100{
Akshay Joshi919ed522011-06-07 04:51:55 -0400101 u32 val;
Pavel Machek66101de2008-10-01 14:36:56 +0200102
Akshay Joshi919ed522011-06-07 04:51:55 -0400103 if (data > 15)
104 data = 15;
105 else if (data < -16)
106 data = -16;
Pavel Machek66101de2008-10-01 14:36:56 +0200107
Akshay Joshi919ed522011-06-07 04:51:55 -0400108 val = data & 0x001F;
Pavel Machek66101de2008-10-01 14:36:56 +0200109
Akshay Joshi919ed522011-06-07 04:51:55 -0400110 return val;
Pavel Machek66101de2008-10-01 14:36:56 +0200111}
112
113/****************************************************************************/
Ebru Akagunduz06c789e2013-10-29 10:08:46 +0200114static s32 _s6_to_s32(u32 data)
Pavel Machek66101de2008-10-01 14:36:56 +0200115{
Akshay Joshi919ed522011-06-07 04:51:55 -0400116 s32 val;
Pavel Machek66101de2008-10-01 14:36:56 +0200117
Akshay Joshi919ed522011-06-07 04:51:55 -0400118 val = (data & 0x001F);
Pavel Machek66101de2008-10-01 14:36:56 +0200119
Akshay Joshi919ed522011-06-07 04:51:55 -0400120 if ((data & BIT(5)) != 0)
121 val |= 0xFFFFFFE0;
Pavel Machek66101de2008-10-01 14:36:56 +0200122
Akshay Joshi919ed522011-06-07 04:51:55 -0400123 return val;
Pavel Machek66101de2008-10-01 14:36:56 +0200124}
125
Ebru Akagunduz06c789e2013-10-29 10:08:46 +0200126static u32 _s32_to_s6(s32 data)
Pavel Machek66101de2008-10-01 14:36:56 +0200127{
Akshay Joshi919ed522011-06-07 04:51:55 -0400128 u32 val;
Pavel Machek66101de2008-10-01 14:36:56 +0200129
Akshay Joshi919ed522011-06-07 04:51:55 -0400130 if (data > 31)
131 data = 31;
132 else if (data < -32)
133 data = -32;
Pavel Machek66101de2008-10-01 14:36:56 +0200134
Akshay Joshi919ed522011-06-07 04:51:55 -0400135 val = data & 0x003F;
Pavel Machek66101de2008-10-01 14:36:56 +0200136
Akshay Joshi919ed522011-06-07 04:51:55 -0400137 return val;
Pavel Machek66101de2008-10-01 14:36:56 +0200138}
139
140/****************************************************************************/
Ebru Akagunduz06c789e2013-10-29 10:08:46 +0200141static s32 _floor(s32 n)
Pavel Machek66101de2008-10-01 14:36:56 +0200142{
Akshay Joshi919ed522011-06-07 04:51:55 -0400143 if (n > 0)
144 n += 5;
145 else
146 n -= 5;
Pavel Machek66101de2008-10-01 14:36:56 +0200147
Vincent Abel-Grimaltaf69c292011-08-10 01:13:38 +0200148 return n/10;
Pavel Machek66101de2008-10-01 14:36:56 +0200149}
150
151/****************************************************************************/
Timofey Trofimov3a4d1b92010-05-25 18:19:54 +0400152/*
153 * The following code is sqare-root function.
154 * sqsum is the input and the output is sq_rt;
155 * The maximum of sqsum = 2^27 -1;
156 */
Ebru Akagunduz06c789e2013-10-29 10:08:46 +0200157static u32 _sqrt(u32 sqsum)
Pavel Machek66101de2008-10-01 14:36:56 +0200158{
Akshay Joshi919ed522011-06-07 04:51:55 -0400159 u32 sq_rt;
Pavel Machek66101de2008-10-01 14:36:56 +0200160
Akshay Joshi919ed522011-06-07 04:51:55 -0400161 int g0, g1, g2, g3, g4;
162 int seed;
163 int next;
164 int step;
Pavel Machek66101de2008-10-01 14:36:56 +0200165
Akshay Joshi919ed522011-06-07 04:51:55 -0400166 g4 = sqsum / 100000000;
167 g3 = (sqsum - g4*100000000) / 1000000;
168 g2 = (sqsum - g4*100000000 - g3*1000000) / 10000;
169 g1 = (sqsum - g4*100000000 - g3*1000000 - g2*10000) / 100;
170 g0 = (sqsum - g4*100000000 - g3*1000000 - g2*10000 - g1*100);
Pavel Machek66101de2008-10-01 14:36:56 +0200171
Akshay Joshi919ed522011-06-07 04:51:55 -0400172 next = g4;
173 step = 0;
174 seed = 0;
175 while (((seed+1)*(step+1)) <= next) {
176 step++;
177 seed++;
178 }
Pavel Machek66101de2008-10-01 14:36:56 +0200179
Akshay Joshi919ed522011-06-07 04:51:55 -0400180 sq_rt = seed * 10000;
181 next = (next-(seed*step))*100 + g3;
Pavel Machek66101de2008-10-01 14:36:56 +0200182
Akshay Joshi919ed522011-06-07 04:51:55 -0400183 step = 0;
184 seed = 2 * seed * 10;
185 while (((seed+1)*(step+1)) <= next) {
186 step++;
187 seed++;
188 }
Pavel Machek66101de2008-10-01 14:36:56 +0200189
Akshay Joshi919ed522011-06-07 04:51:55 -0400190 sq_rt = sq_rt + step * 1000;
191 next = (next - seed * step) * 100 + g2;
192 seed = (seed + step) * 10;
193 step = 0;
194 while (((seed+1)*(step+1)) <= next) {
195 step++;
196 seed++;
197 }
Pavel Machek66101de2008-10-01 14:36:56 +0200198
Akshay Joshi919ed522011-06-07 04:51:55 -0400199 sq_rt = sq_rt + step * 100;
200 next = (next - seed * step) * 100 + g1;
201 seed = (seed + step) * 10;
202 step = 0;
Pavel Machek66101de2008-10-01 14:36:56 +0200203
Akshay Joshi919ed522011-06-07 04:51:55 -0400204 while (((seed+1)*(step+1)) <= next) {
205 step++;
206 seed++;
207 }
Pavel Machek66101de2008-10-01 14:36:56 +0200208
Akshay Joshi919ed522011-06-07 04:51:55 -0400209 sq_rt = sq_rt + step * 10;
210 next = (next - seed * step) * 100 + g0;
211 seed = (seed + step) * 10;
212 step = 0;
Pavel Machek66101de2008-10-01 14:36:56 +0200213
Akshay Joshi919ed522011-06-07 04:51:55 -0400214 while (((seed+1)*(step+1)) <= next) {
215 step++;
216 seed++;
217 }
Pavel Machek66101de2008-10-01 14:36:56 +0200218
Akshay Joshi919ed522011-06-07 04:51:55 -0400219 sq_rt = sq_rt + step;
Pavel Machek66101de2008-10-01 14:36:56 +0200220
Akshay Joshi919ed522011-06-07 04:51:55 -0400221 return sq_rt;
Pavel Machek66101de2008-10-01 14:36:56 +0200222}
223
224/****************************************************************************/
Ebru Akagunduz06c789e2013-10-29 10:08:46 +0200225static void _sin_cos(s32 angle, s32 *sin, s32 *cos)
Pavel Machek66101de2008-10-01 14:36:56 +0200226{
Akshay Joshi919ed522011-06-07 04:51:55 -0400227 s32 X, Y, TargetAngle, CurrAngle;
228 unsigned Step;
Pavel Machek66101de2008-10-01 14:36:56 +0200229
Akshay Joshi919ed522011-06-07 04:51:55 -0400230 X = FIXED(AG_CONST); /* AG_CONST * cos(0) */
231 Y = 0; /* AG_CONST * sin(0) */
232 TargetAngle = abs(angle);
233 CurrAngle = 0;
Pavel Machek66101de2008-10-01 14:36:56 +0200234
Akshay Joshi919ed522011-06-07 04:51:55 -0400235 for (Step = 0; Step < 12; Step++) {
236 s32 NewX;
Pavel Machek66101de2008-10-01 14:36:56 +0200237
Akshay Joshi919ed522011-06-07 04:51:55 -0400238 if (TargetAngle > CurrAngle) {
239 NewX = X - (Y >> Step);
240 Y = (X >> Step) + Y;
241 X = NewX;
242 CurrAngle += Angles[Step];
243 } else {
244 NewX = X + (Y >> Step);
245 Y = -(X >> Step) + Y;
246 X = NewX;
247 CurrAngle -= Angles[Step];
248 }
249 }
Pavel Machek66101de2008-10-01 14:36:56 +0200250
Akshay Joshi919ed522011-06-07 04:51:55 -0400251 if (angle > 0) {
252 *cos = X;
253 *sin = Y;
254 } else {
255 *cos = X;
256 *sin = -Y;
257 }
Pavel Machek66101de2008-10-01 14:36:56 +0200258}
259
Iker Pedrosade6c7642013-09-16 15:43:24 +0200260static unsigned char hal_get_dxx_reg(struct hw_data *pHwData, u16 number,
261 u32 *pValue)
Pekka Enbergdd68e1b2009-04-08 11:51:22 +0300262{
263 if (number < 0x1000)
264 number += 0x1000;
265 return Wb35Reg_ReadSync(pHwData, number, pValue);
266}
Timofey Trofimov3a4d1b92010-05-25 18:19:54 +0400267#define hw_get_dxx_reg(_A, _B, _C) hal_get_dxx_reg(_A, _B, (u32 *)_C)
Pekka Enbergdd68e1b2009-04-08 11:51:22 +0300268
Iker Pedrosade6c7642013-09-16 15:43:24 +0200269static unsigned char hal_set_dxx_reg(struct hw_data *pHwData, u16 number,
270 u32 value)
Pekka Enbergdd68e1b2009-04-08 11:51:22 +0300271{
272 unsigned char ret;
273
274 if (number < 0x1000)
275 number += 0x1000;
276 ret = Wb35Reg_WriteSync(pHwData, number, value);
277 return ret;
278}
Timofey Trofimov3a4d1b92010-05-25 18:19:54 +0400279#define hw_set_dxx_reg(_A, _B, _C) hal_set_dxx_reg(_A, _B, (u32)_C)
Pekka Enbergdd68e1b2009-04-08 11:51:22 +0300280
Pavel Machek66101de2008-10-01 14:36:56 +0200281
Ebru Akagunduz06c789e2013-10-29 10:08:46 +0200282static void _reset_rx_cal(struct hw_data *phw_data)
Pavel Machek66101de2008-10-01 14:36:56 +0200283{
284 u32 val;
285
286 hw_get_dxx_reg(phw_data, 0x54, &val);
287
Timofey Trofimov3a4d1b92010-05-25 18:19:54 +0400288 if (phw_data->revision == 0x2002) /* 1st-cut */
Pavel Machek66101de2008-10-01 14:36:56 +0200289 val &= 0xFFFF0000;
Timofey Trofimov3a4d1b92010-05-25 18:19:54 +0400290 else /* 2nd-cut */
Pavel Machek66101de2008-10-01 14:36:56 +0200291 val &= 0x000003FF;
Pavel Machek66101de2008-10-01 14:36:56 +0200292
293 hw_set_dxx_reg(phw_data, 0x54, val);
294}
295
296
Timofey Trofimov3a4d1b92010-05-25 18:19:54 +0400297/**************for winbond calibration*********/
Pavel Machek66101de2008-10-01 14:36:56 +0200298
Timofey Trofimov3a4d1b92010-05-25 18:19:54 +0400299
300
301/**********************************************/
Ebru Akagunduz06c789e2013-10-29 10:08:46 +0200302static void _rxadc_dc_offset_cancellation_winbond(struct hw_data *phw_data, u32 frequency)
Pavel Machek66101de2008-10-01 14:36:56 +0200303{
Akshay Joshi919ed522011-06-07 04:51:55 -0400304 u32 reg_agc_ctrl3;
305 u32 reg_a_acq_ctrl;
306 u32 reg_b_acq_ctrl;
307 u32 val;
Pavel Machek66101de2008-10-01 14:36:56 +0200308
Akshay Joshi919ed522011-06-07 04:51:55 -0400309 PHY_DEBUG(("[CAL] -> [1]_rxadc_dc_offset_cancellation()\n"));
310 phy_init_rf(phw_data);
Pavel Machek66101de2008-10-01 14:36:56 +0200311
Akshay Joshi919ed522011-06-07 04:51:55 -0400312 /* set calibration channel */
313 if ((RF_WB_242 == phw_data->phy_type) ||
Timofey Trofimov3a4d1b92010-05-25 18:19:54 +0400314 (RF_WB_242_1 == phw_data->phy_type)) /* 20060619.5 Add */{
Akshay Joshi919ed522011-06-07 04:51:55 -0400315 if ((frequency >= 2412) && (frequency <= 2484)) {
316 /* w89rf242 change frequency to 2390Mhz */
317 PHY_DEBUG(("[CAL] W89RF242/11G/Channel=2390Mhz\n"));
Pavel Machek66101de2008-10-01 14:36:56 +0200318 phy_set_rf_data(phw_data, 3, (3<<24)|0x025586);
319
Akshay Joshi919ed522011-06-07 04:51:55 -0400320 }
321 } else {
Pavel Machek66101de2008-10-01 14:36:56 +0200322
323 }
324
Timofey Trofimov3a4d1b92010-05-25 18:19:54 +0400325 /* reset cancel_dc_i[9:5] and cancel_dc_q[4:0] in register DC_Cancel */
Pavel Machek66101de2008-10-01 14:36:56 +0200326 hw_get_dxx_reg(phw_data, 0x5C, &val);
327 val &= ~(0x03FF);
328 hw_set_dxx_reg(phw_data, 0x5C, val);
329
Timofey Trofimov3a4d1b92010-05-25 18:19:54 +0400330 /* reset the TX and RX IQ calibration data */
Pavel Machek66101de2008-10-01 14:36:56 +0200331 hw_set_dxx_reg(phw_data, 0x3C, 0);
332 hw_set_dxx_reg(phw_data, 0x54, 0);
333
Timofey Trofimov3a4d1b92010-05-25 18:19:54 +0400334 hw_set_dxx_reg(phw_data, 0x58, 0x30303030); /* IQ_Alpha Changed */
Pavel Machek66101de2008-10-01 14:36:56 +0200335
Timofey Trofimov3a4d1b92010-05-25 18:19:54 +0400336 /* a. Disable AGC */
Pavel Machek66101de2008-10-01 14:36:56 +0200337 hw_get_dxx_reg(phw_data, REG_AGC_CTRL3, &reg_agc_ctrl3);
338 reg_agc_ctrl3 &= ~BIT(2);
339 reg_agc_ctrl3 |= (MASK_LNA_FIX_GAIN|MASK_AGC_FIX);
340 hw_set_dxx_reg(phw_data, REG_AGC_CTRL3, reg_agc_ctrl3);
341
342 hw_get_dxx_reg(phw_data, REG_AGC_CTRL5, &val);
343 val |= MASK_AGC_FIX_GAIN;
344 hw_set_dxx_reg(phw_data, REG_AGC_CTRL5, val);
345
Timofey Trofimov3a4d1b92010-05-25 18:19:54 +0400346 /* b. Turn off BB RX */
Pavel Machek66101de2008-10-01 14:36:56 +0200347 hw_get_dxx_reg(phw_data, REG_A_ACQ_CTRL, &reg_a_acq_ctrl);
348 reg_a_acq_ctrl |= MASK_AMER_OFF_REG;
349 hw_set_dxx_reg(phw_data, REG_A_ACQ_CTRL, reg_a_acq_ctrl);
350
351 hw_get_dxx_reg(phw_data, REG_B_ACQ_CTRL, &reg_b_acq_ctrl);
352 reg_b_acq_ctrl |= MASK_BMER_OFF_REG;
353 hw_set_dxx_reg(phw_data, REG_B_ACQ_CTRL, reg_b_acq_ctrl);
354
Timofey Trofimov3a4d1b92010-05-25 18:19:54 +0400355 /* c. Make sure MAC is in receiving mode
356 * d. Turn ON ADC calibration
357 * - ADC calibrator is triggered by this signal rising from 0 to 1 */
Pavel Machek66101de2008-10-01 14:36:56 +0200358 hw_get_dxx_reg(phw_data, REG_MODE_CTRL, &val);
359 val &= ~MASK_ADC_DC_CAL_STR;
360 hw_set_dxx_reg(phw_data, REG_MODE_CTRL, val);
361
362 val |= MASK_ADC_DC_CAL_STR;
363 hw_set_dxx_reg(phw_data, REG_MODE_CTRL, val);
Pavel Machek66101de2008-10-01 14:36:56 +0200364
Justin P. Mattocka31f7f52012-09-03 08:06:02 -0700365 /* e. The results are shown in "adc_dc_cal_i[8:0] and adc_dc_cal_q[8:0]" */
Pavel Machek66101de2008-10-01 14:36:56 +0200366#ifdef _DEBUG
367 hw_get_dxx_reg(phw_data, REG_OFFSET_READ, &val);
368 PHY_DEBUG(("[CAL] REG_OFFSET_READ = 0x%08X\n", val));
369
370 PHY_DEBUG(("[CAL] ** adc_dc_cal_i = %d (0x%04X)\n",
371 _s9_to_s32(val&0x000001FF), val&0x000001FF));
372 PHY_DEBUG(("[CAL] ** adc_dc_cal_q = %d (0x%04X)\n",
Iker Pedrosade6c7642013-09-16 15:43:24 +0200373 _s9_to_s32((val&0x0003FE00)>>9),
374 (val&0x0003FE00)>>9));
Pavel Machek66101de2008-10-01 14:36:56 +0200375#endif
376
377 hw_get_dxx_reg(phw_data, REG_MODE_CTRL, &val);
378 val &= ~MASK_ADC_DC_CAL_STR;
379 hw_set_dxx_reg(phw_data, REG_MODE_CTRL, val);
380
Timofey Trofimov3a4d1b92010-05-25 18:19:54 +0400381 /* f. Turn on BB RX */
382 /* hw_get_dxx_reg(phw_data, REG_A_ACQ_CTRL, &reg_a_acq_ctrl); */
Pavel Machek66101de2008-10-01 14:36:56 +0200383 reg_a_acq_ctrl &= ~MASK_AMER_OFF_REG;
384 hw_set_dxx_reg(phw_data, REG_A_ACQ_CTRL, reg_a_acq_ctrl);
385
Timofey Trofimov3a4d1b92010-05-25 18:19:54 +0400386 /* hw_get_dxx_reg(phw_data, REG_B_ACQ_CTRL, &reg_b_acq_ctrl); */
Pavel Machek66101de2008-10-01 14:36:56 +0200387 reg_b_acq_ctrl &= ~MASK_BMER_OFF_REG;
388 hw_set_dxx_reg(phw_data, REG_B_ACQ_CTRL, reg_b_acq_ctrl);
389
Timofey Trofimov3a4d1b92010-05-25 18:19:54 +0400390 /* g. Enable AGC */
391 /* hw_get_dxx_reg(phw_data, REG_AGC_CTRL3, &val); */
Pavel Machek66101de2008-10-01 14:36:56 +0200392 reg_agc_ctrl3 |= BIT(2);
393 reg_agc_ctrl3 &= ~(MASK_LNA_FIX_GAIN|MASK_AGC_FIX);
394 hw_set_dxx_reg(phw_data, REG_AGC_CTRL3, reg_agc_ctrl3);
395}
396
Timofey Trofimov3a4d1b92010-05-25 18:19:54 +0400397/* 20060612.1.a 20060718.1 Modify */
Ebru Akagunduz06c789e2013-10-29 10:08:46 +0200398static u8 _tx_iq_calibration_loop_winbond(struct hw_data *phw_data,
Pavel Machek66101de2008-10-01 14:36:56 +0200399 s32 a_2_threshold,
400 s32 b_2_threshold)
401{
402 u32 reg_mode_ctrl;
403 s32 iq_mag_0_tx;
404 s32 iqcal_tone_i0;
405 s32 iqcal_tone_q0;
406 s32 iqcal_tone_i;
407 s32 iqcal_tone_q;
408 u32 sqsum;
409 s32 rot_i_b;
410 s32 rot_q_b;
411 s32 tx_cal_flt_b[4];
412 s32 tx_cal[4];
413 s32 tx_cal_reg[4];
414 s32 a_2, b_2;
415 s32 sin_b, sin_2b;
416 s32 cos_b, cos_2b;
417 s32 divisor;
418 s32 temp1, temp2;
419 u32 val;
420 u16 loop;
Timofey Trofimov3a4d1b92010-05-25 18:19:54 +0400421 s32 iqcal_tone_i_avg, iqcal_tone_q_avg;
Pavel Machek66101de2008-10-01 14:36:56 +0200422 u8 verify_count;
423 int capture_time;
424
425 PHY_DEBUG(("[CAL] -> _tx_iq_calibration_loop()\n"));
426 PHY_DEBUG(("[CAL] ** a_2_threshold = %d\n", a_2_threshold));
427 PHY_DEBUG(("[CAL] ** b_2_threshold = %d\n", b_2_threshold));
428
429 verify_count = 0;
430
431 hw_get_dxx_reg(phw_data, REG_MODE_CTRL, &reg_mode_ctrl);
432 PHY_DEBUG(("[CAL] MODE_CTRL (read) = 0x%08X\n", reg_mode_ctrl));
433
434 loop = LOOP_TIMES;
435
Timofey Trofimov3a4d1b92010-05-25 18:19:54 +0400436 while (loop > 0) {
Iker Pedrosade6c7642013-09-16 15:43:24 +0200437 PHY_DEBUG(("[CAL] [%d.] <_tx_iq_calibration_loop>\n",
438 (LOOP_TIMES-loop+1)));
Pavel Machek66101de2008-10-01 14:36:56 +0200439
Timofey Trofimov3a4d1b92010-05-25 18:19:54 +0400440 iqcal_tone_i_avg = 0;
441 iqcal_tone_q_avg = 0;
442 if (!hw_set_dxx_reg(phw_data, 0x3C, 0x00)) /* 20060718.1 modify */
Pavel Machek66101de2008-10-01 14:36:56 +0200443 return 0;
Timofey Trofimov3a4d1b92010-05-25 18:19:54 +0400444 for (capture_time = 0; capture_time < 10; capture_time++) {
445 /*
Iker Pedrosade6c7642013-09-16 15:43:24 +0200446 * a. Set iqcal_mode[1:0] to 0x2 and set "calib_start"
447 * to 0x1 to enable "IQ calibration Mode II"
Timofey Trofimov3a4d1b92010-05-25 18:19:54 +0400448 */
Pavel Machek66101de2008-10-01 14:36:56 +0200449 reg_mode_ctrl &= ~(MASK_IQCAL_TONE_SEL|MASK_IQCAL_MODE);
450 reg_mode_ctrl &= ~MASK_IQCAL_MODE;
451 reg_mode_ctrl |= (MASK_CALIB_START|0x02);
452 reg_mode_ctrl |= (MASK_CALIB_START|0x02|2<<2);
453 hw_set_dxx_reg(phw_data, REG_MODE_CTRL, reg_mode_ctrl);
454 PHY_DEBUG(("[CAL] MODE_CTRL (write) = 0x%08X\n", reg_mode_ctrl));
Pavel Machek66101de2008-10-01 14:36:56 +0200455
Timofey Trofimov3a4d1b92010-05-25 18:19:54 +0400456 /* b. */
Pavel Machek66101de2008-10-01 14:36:56 +0200457 hw_get_dxx_reg(phw_data, REG_CALIB_READ1, &val);
458 PHY_DEBUG(("[CAL] CALIB_READ1 = 0x%08X\n", val));
Pavel Machek66101de2008-10-01 14:36:56 +0200459
460 iqcal_tone_i0 = _s13_to_s32(val & 0x00001FFF);
461 iqcal_tone_q0 = _s13_to_s32((val & 0x03FFE000) >> 13);
462 PHY_DEBUG(("[CAL] ** iqcal_tone_i0=%d, iqcal_tone_q0=%d\n",
463 iqcal_tone_i0, iqcal_tone_q0));
464
465 sqsum = iqcal_tone_i0*iqcal_tone_i0 +
466 iqcal_tone_q0*iqcal_tone_q0;
467 iq_mag_0_tx = (s32) _sqrt(sqsum);
468 PHY_DEBUG(("[CAL] ** iq_mag_0_tx=%d\n", iq_mag_0_tx));
469
Timofey Trofimov3a4d1b92010-05-25 18:19:54 +0400470 /* c. Set "calib_start" to 0x0 */
Pavel Machek66101de2008-10-01 14:36:56 +0200471 reg_mode_ctrl &= ~MASK_CALIB_START;
472 hw_set_dxx_reg(phw_data, REG_MODE_CTRL, reg_mode_ctrl);
473 PHY_DEBUG(("[CAL] MODE_CTRL (write) = 0x%08X\n", reg_mode_ctrl));
Pavel Machek66101de2008-10-01 14:36:56 +0200474
Timofey Trofimov3a4d1b92010-05-25 18:19:54 +0400475 /*
Iker Pedrosade6c7642013-09-16 15:43:24 +0200476 * d. Set iqcal_mode[1:0] to 0x3 and set "calib_start"
477 * to 0x1 to enable "IQ calibration Mode II"
Timofey Trofimov3a4d1b92010-05-25 18:19:54 +0400478 */
479 /* hw_get_dxx_reg(phw_data, REG_MODE_CTRL, &val); */
Pavel Machek66101de2008-10-01 14:36:56 +0200480 hw_get_dxx_reg(phw_data, REG_MODE_CTRL, &reg_mode_ctrl);
481 reg_mode_ctrl &= ~MASK_IQCAL_MODE;
482 reg_mode_ctrl |= (MASK_CALIB_START|0x03);
483 hw_set_dxx_reg(phw_data, REG_MODE_CTRL, reg_mode_ctrl);
484 PHY_DEBUG(("[CAL] MODE_CTRL (write) = 0x%08X\n", reg_mode_ctrl));
Pavel Machek66101de2008-10-01 14:36:56 +0200485
Timofey Trofimov3a4d1b92010-05-25 18:19:54 +0400486 /* e. */
Pavel Machek66101de2008-10-01 14:36:56 +0200487 hw_get_dxx_reg(phw_data, REG_CALIB_READ1, &val);
488 PHY_DEBUG(("[CAL] CALIB_READ1 = 0x%08X\n", val));
Pavel Machek66101de2008-10-01 14:36:56 +0200489
490 iqcal_tone_i = _s13_to_s32(val & 0x00001FFF);
491 iqcal_tone_q = _s13_to_s32((val & 0x03FFE000) >> 13);
492 PHY_DEBUG(("[CAL] ** iqcal_tone_i = %d, iqcal_tone_q = %d\n",
Iker Pedrosade6c7642013-09-16 15:43:24 +0200493 iqcal_tone_i, iqcal_tone_q));
Timofey Trofimov3a4d1b92010-05-25 18:19:54 +0400494 if (capture_time == 0)
Pavel Machek66101de2008-10-01 14:36:56 +0200495 continue;
Timofey Trofimov3a4d1b92010-05-25 18:19:54 +0400496 else {
497 iqcal_tone_i_avg = (iqcal_tone_i_avg*(capture_time-1) + iqcal_tone_i)/capture_time;
498 iqcal_tone_q_avg = (iqcal_tone_q_avg*(capture_time-1) + iqcal_tone_q)/capture_time;
Pavel Machek66101de2008-10-01 14:36:56 +0200499 }
500 }
501
502 iqcal_tone_i = iqcal_tone_i_avg;
503 iqcal_tone_q = iqcal_tone_q_avg;
504
505
506 rot_i_b = (iqcal_tone_i * iqcal_tone_i0 +
507 iqcal_tone_q * iqcal_tone_q0) / 1024;
508 rot_q_b = (iqcal_tone_i * iqcal_tone_q0 * (-1) +
509 iqcal_tone_q * iqcal_tone_i0) / 1024;
510 PHY_DEBUG(("[CAL] ** rot_i_b = %d, rot_q_b = %d\n",
511 rot_i_b, rot_q_b));
512
Timofey Trofimov3a4d1b92010-05-25 18:19:54 +0400513 /* f. */
Pavel Machek66101de2008-10-01 14:36:56 +0200514 divisor = ((iq_mag_0_tx * iq_mag_0_tx * 2)/1024 - rot_i_b) * 2;
515
Timofey Trofimov3a4d1b92010-05-25 18:19:54 +0400516 if (divisor == 0) {
Pavel Machek66101de2008-10-01 14:36:56 +0200517 PHY_DEBUG(("[CAL] ** <_tx_iq_calibration_loop> ERROR *******\n"));
518 PHY_DEBUG(("[CAL] ** divisor=0 to calculate EPS and THETA !!\n"));
519 PHY_DEBUG(("[CAL] ******************************************\n"));
520 break;
521 }
522
523 a_2 = (rot_i_b * 32768) / divisor;
524 b_2 = (rot_q_b * (-32768)) / divisor;
525 PHY_DEBUG(("[CAL] ***** EPSILON/2 = %d\n", a_2));
526 PHY_DEBUG(("[CAL] ***** THETA/2 = %d\n", b_2));
527
528 phw_data->iq_rsdl_gain_tx_d2 = a_2;
529 phw_data->iq_rsdl_phase_tx_d2 = b_2;
530
Timofey Trofimov3a4d1b92010-05-25 18:19:54 +0400531 /* if ((abs(a_2) < 150) && (abs(b_2) < 100)) */
532 /* if ((abs(a_2) < 200) && (abs(b_2) < 200)) */
533 if ((abs(a_2) < a_2_threshold) && (abs(b_2) < b_2_threshold)) {
Pavel Machek66101de2008-10-01 14:36:56 +0200534 verify_count++;
535
536 PHY_DEBUG(("[CAL] ** <_tx_iq_calibration_loop> *************\n"));
537 PHY_DEBUG(("[CAL] ** VERIFY OK # %d !!\n", verify_count));
538 PHY_DEBUG(("[CAL] ******************************************\n"));
539
Timofey Trofimov3a4d1b92010-05-25 18:19:54 +0400540 if (verify_count > 2) {
Pavel Machek66101de2008-10-01 14:36:56 +0200541 PHY_DEBUG(("[CAL] ** <_tx_iq_calibration_loop> *********\n"));
542 PHY_DEBUG(("[CAL] ** TX_IQ_CALIBRATION (EPS,THETA) OK !!\n"));
543 PHY_DEBUG(("[CAL] **************************************\n"));
544 return 0;
545 }
546
547 continue;
Timofey Trofimov3a4d1b92010-05-25 18:19:54 +0400548 } else
Pavel Machek66101de2008-10-01 14:36:56 +0200549 verify_count = 0;
Pavel Machek66101de2008-10-01 14:36:56 +0200550
551 _sin_cos(b_2, &sin_b, &cos_b);
552 _sin_cos(b_2*2, &sin_2b, &cos_2b);
553 PHY_DEBUG(("[CAL] ** sin(b/2)=%d, cos(b/2)=%d\n", sin_b, cos_b));
554 PHY_DEBUG(("[CAL] ** sin(b)=%d, cos(b)=%d\n", sin_2b, cos_2b));
555
Timofey Trofimov3a4d1b92010-05-25 18:19:54 +0400556 if (cos_2b == 0) {
Pavel Machek66101de2008-10-01 14:36:56 +0200557 PHY_DEBUG(("[CAL] ** <_tx_iq_calibration_loop> ERROR *******\n"));
558 PHY_DEBUG(("[CAL] ** cos(b)=0 !!\n"));
559 PHY_DEBUG(("[CAL] ******************************************\n"));
560 break;
561 }
562
Timofey Trofimov3a4d1b92010-05-25 18:19:54 +0400563 /* 1280 * 32768 = 41943040 */
Pavel Machek66101de2008-10-01 14:36:56 +0200564 temp1 = (41943040/cos_2b)*cos_b;
565
Timofey Trofimov3a4d1b92010-05-25 18:19:54 +0400566 /* temp2 = (41943040/cos_2b)*sin_b*(-1); */
567 if (phw_data->revision == 0x2002) /* 1st-cut */
Pavel Machek66101de2008-10-01 14:36:56 +0200568 temp2 = (41943040/cos_2b)*sin_b*(-1);
Timofey Trofimov3a4d1b92010-05-25 18:19:54 +0400569 else /* 2nd-cut */
Pavel Machek66101de2008-10-01 14:36:56 +0200570 temp2 = (41943040*4/cos_2b)*sin_b*(-1);
Pavel Machek66101de2008-10-01 14:36:56 +0200571
572 tx_cal_flt_b[0] = _floor(temp1/(32768+a_2));
573 tx_cal_flt_b[1] = _floor(temp2/(32768+a_2));
574 tx_cal_flt_b[2] = _floor(temp2/(32768-a_2));
575 tx_cal_flt_b[3] = _floor(temp1/(32768-a_2));
576 PHY_DEBUG(("[CAL] ** tx_cal_flt_b[0] = %d\n", tx_cal_flt_b[0]));
577 PHY_DEBUG(("[CAL] tx_cal_flt_b[1] = %d\n", tx_cal_flt_b[1]));
578 PHY_DEBUG(("[CAL] tx_cal_flt_b[2] = %d\n", tx_cal_flt_b[2]));
579 PHY_DEBUG(("[CAL] tx_cal_flt_b[3] = %d\n", tx_cal_flt_b[3]));
580
581 tx_cal[2] = tx_cal_flt_b[2];
Timofey Trofimov3a4d1b92010-05-25 18:19:54 +0400582 tx_cal[2] = tx_cal[2] + 3;
Pavel Machek66101de2008-10-01 14:36:56 +0200583 tx_cal[1] = tx_cal[2];
584 tx_cal[3] = tx_cal_flt_b[3] - 128;
Timofey Trofimov3a4d1b92010-05-25 18:19:54 +0400585 tx_cal[0] = -tx_cal[3] + 1;
Pavel Machek66101de2008-10-01 14:36:56 +0200586
587 PHY_DEBUG(("[CAL] tx_cal[0] = %d\n", tx_cal[0]));
588 PHY_DEBUG(("[CAL] tx_cal[1] = %d\n", tx_cal[1]));
589 PHY_DEBUG(("[CAL] tx_cal[2] = %d\n", tx_cal[2]));
590 PHY_DEBUG(("[CAL] tx_cal[3] = %d\n", tx_cal[3]));
591
Timofey Trofimov3a4d1b92010-05-25 18:19:54 +0400592 /* if ((tx_cal[0] == 0) && (tx_cal[1] == 0) &&
593 (tx_cal[2] == 0) && (tx_cal[3] == 0))
594 { */
595 /* PHY_DEBUG(("[CAL] ** <_tx_iq_calibration_loop> *************\n"));
596 * PHY_DEBUG(("[CAL] ** TX_IQ_CALIBRATION COMPLETE !!\n"));
597 * PHY_DEBUG(("[CAL] ******************************************\n"));
598 * return 0;
599 } */
Pavel Machek66101de2008-10-01 14:36:56 +0200600
Timofey Trofimov3a4d1b92010-05-25 18:19:54 +0400601 /* g. */
602 if (phw_data->revision == 0x2002) /* 1st-cut */{
Pavel Machek66101de2008-10-01 14:36:56 +0200603 hw_get_dxx_reg(phw_data, 0x54, &val);
604 PHY_DEBUG(("[CAL] ** 0x54 = 0x%08X\n", val));
605 tx_cal_reg[0] = _s4_to_s32((val & 0xF0000000) >> 28);
606 tx_cal_reg[1] = _s4_to_s32((val & 0x0F000000) >> 24);
607 tx_cal_reg[2] = _s4_to_s32((val & 0x00F00000) >> 20);
608 tx_cal_reg[3] = _s4_to_s32((val & 0x000F0000) >> 16);
Timofey Trofimov3a4d1b92010-05-25 18:19:54 +0400609 } else /* 2nd-cut */{
Pavel Machek66101de2008-10-01 14:36:56 +0200610 hw_get_dxx_reg(phw_data, 0x3C, &val);
611 PHY_DEBUG(("[CAL] ** 0x3C = 0x%08X\n", val));
612 tx_cal_reg[0] = _s5_to_s32((val & 0xF8000000) >> 27);
613 tx_cal_reg[1] = _s6_to_s32((val & 0x07E00000) >> 21);
614 tx_cal_reg[2] = _s6_to_s32((val & 0x001F8000) >> 15);
615 tx_cal_reg[3] = _s5_to_s32((val & 0x00007C00) >> 10);
616
617 }
618
619 PHY_DEBUG(("[CAL] ** tx_cal_reg[0] = %d\n", tx_cal_reg[0]));
620 PHY_DEBUG(("[CAL] tx_cal_reg[1] = %d\n", tx_cal_reg[1]));
621 PHY_DEBUG(("[CAL] tx_cal_reg[2] = %d\n", tx_cal_reg[2]));
622 PHY_DEBUG(("[CAL] tx_cal_reg[3] = %d\n", tx_cal_reg[3]));
623
Timofey Trofimov3a4d1b92010-05-25 18:19:54 +0400624 if (phw_data->revision == 0x2002) /* 1st-cut */{
625 if (((tx_cal_reg[0] == 7) || (tx_cal_reg[0] == (-8))) &&
626 ((tx_cal_reg[3] == 7) || (tx_cal_reg[3] == (-8)))) {
Pavel Machek66101de2008-10-01 14:36:56 +0200627 PHY_DEBUG(("[CAL] ** <_tx_iq_calibration_loop> *********\n"));
628 PHY_DEBUG(("[CAL] ** TX_IQ_CALIBRATION SATUATION !!\n"));
629 PHY_DEBUG(("[CAL] **************************************\n"));
630 break;
631 }
Timofey Trofimov3a4d1b92010-05-25 18:19:54 +0400632 } else /* 2nd-cut */{
633 if (((tx_cal_reg[0] == 31) || (tx_cal_reg[0] == (-32))) &&
634 ((tx_cal_reg[3] == 31) || (tx_cal_reg[3] == (-32)))) {
Pavel Machek66101de2008-10-01 14:36:56 +0200635 PHY_DEBUG(("[CAL] ** <_tx_iq_calibration_loop> *********\n"));
636 PHY_DEBUG(("[CAL] ** TX_IQ_CALIBRATION SATUATION !!\n"));
637 PHY_DEBUG(("[CAL] **************************************\n"));
638 break;
639 }
640 }
641
642 tx_cal[0] = tx_cal[0] + tx_cal_reg[0];
643 tx_cal[1] = tx_cal[1] + tx_cal_reg[1];
644 tx_cal[2] = tx_cal[2] + tx_cal_reg[2];
645 tx_cal[3] = tx_cal[3] + tx_cal_reg[3];
646 PHY_DEBUG(("[CAL] ** apply tx_cal[0] = %d\n", tx_cal[0]));
647 PHY_DEBUG(("[CAL] apply tx_cal[1] = %d\n", tx_cal[1]));
648 PHY_DEBUG(("[CAL] apply tx_cal[2] = %d\n", tx_cal[2]));
649 PHY_DEBUG(("[CAL] apply tx_cal[3] = %d\n", tx_cal[3]));
650
Timofey Trofimov3a4d1b92010-05-25 18:19:54 +0400651 if (phw_data->revision == 0x2002) /* 1st-cut */{
Pavel Machek66101de2008-10-01 14:36:56 +0200652 val &= 0x0000FFFF;
653 val |= ((_s32_to_s4(tx_cal[0]) << 28)|
654 (_s32_to_s4(tx_cal[1]) << 24)|
655 (_s32_to_s4(tx_cal[2]) << 20)|
656 (_s32_to_s4(tx_cal[3]) << 16));
657 hw_set_dxx_reg(phw_data, 0x54, val);
658 PHY_DEBUG(("[CAL] ** CALIB_DATA = 0x%08X\n", val));
659 return 0;
Timofey Trofimov3a4d1b92010-05-25 18:19:54 +0400660 } else /* 2nd-cut */{
Pavel Machek66101de2008-10-01 14:36:56 +0200661 val &= 0x000003FF;
662 val |= ((_s32_to_s5(tx_cal[0]) << 27)|
663 (_s32_to_s6(tx_cal[1]) << 21)|
664 (_s32_to_s6(tx_cal[2]) << 15)|
665 (_s32_to_s5(tx_cal[3]) << 10));
666 hw_set_dxx_reg(phw_data, 0x3C, val);
667 PHY_DEBUG(("[CAL] ** TX_IQ_CALIBRATION = 0x%08X\n", val));
668 return 0;
669 }
670
Timofey Trofimov3a4d1b92010-05-25 18:19:54 +0400671 /* i. Set "calib_start" to 0x0 */
Pavel Machek66101de2008-10-01 14:36:56 +0200672 reg_mode_ctrl &= ~MASK_CALIB_START;
673 hw_set_dxx_reg(phw_data, REG_MODE_CTRL, reg_mode_ctrl);
674 PHY_DEBUG(("[CAL] MODE_CTRL (write) = 0x%08X\n", reg_mode_ctrl));
675
676 loop--;
677 }
678
679 return 1;
680}
681
Ebru Akagunduz06c789e2013-10-29 10:08:46 +0200682static void _tx_iq_calibration_winbond(struct hw_data *phw_data)
Pavel Machek66101de2008-10-01 14:36:56 +0200683{
684 u32 reg_agc_ctrl3;
685#ifdef _DEBUG
686 s32 tx_cal_reg[4];
687
688#endif
689 u32 reg_mode_ctrl;
690 u32 val;
691 u8 result;
692
693 PHY_DEBUG(("[CAL] -> [4]_tx_iq_calibration()\n"));
694
Timofey Trofimov3a4d1b92010-05-25 18:19:54 +0400695 /* 0x01 0xEE3FC2 ; 3B8FF ; Calibration (6a). enable TX IQ calibration loop circuits */
Pavel Machek66101de2008-10-01 14:36:56 +0200696 phy_set_rf_data(phw_data, 1, (1<<24)|0xEE3FC2);
Timofey Trofimov3a4d1b92010-05-25 18:19:54 +0400697 /* 0x0B 0x1905D6 ; 06417 ; Calibration (6b). enable TX I/Q cal loop squaring circuit */
698 phy_set_rf_data(phw_data, 11, (11<<24)|0x19BDD6); /* 20060612.1.a 0x1905D6); */
699 /* 0x05 0x24C60A ; 09318 ; Calibration (6c). setting TX-VGA gain: TXGCH=2 & GPK=110 --> to be optimized */
700 phy_set_rf_data(phw_data, 5, (5<<24)|0x24C60A); /* 0x24C60A (high temperature) */
Vincent Abel-Grimaltaf69c292011-08-10 01:13:38 +0200701 /* 0x06 0x06880C ; 01A20 ; Calibration (6d). RXGCH=00; RXGCL=100 000 (RXVGA=32) --> to be optimized */
Timofey Trofimov3a4d1b92010-05-25 18:19:54 +0400702 phy_set_rf_data(phw_data, 6, (6<<24)|0x34880C); /* 20060612.1.a 0x06890C); */
703 /* 0x00 0xFDF1C0 ; 3F7C7 ; Calibration (6e). turn on IQ imbalance/Test mode */
Pavel Machek66101de2008-10-01 14:36:56 +0200704 phy_set_rf_data(phw_data, 0, (0<<24)|0xFDF1C0);
Timofey Trofimov3a4d1b92010-05-25 18:19:54 +0400705 /* ; [BB-chip]: Calibration (6f).Send test pattern */
706 /* ; [BB-chip]: Calibration (6g). Search RXGCL optimal value */
Justin P. Mattocka31f7f52012-09-03 08:06:02 -0700707 /* ; [BB-chip]: Calibration (6h). Calculate TX-path IQ imbalance and setting TX path IQ compensation table */
Timofey Trofimov3a4d1b92010-05-25 18:19:54 +0400708 /* phy_set_rf_data(phw_data, 3, (3<<24)|0x025586); */
Pavel Machek66101de2008-10-01 14:36:56 +0200709
Timofey Trofimov3a4d1b92010-05-25 18:19:54 +0400710 msleep(30); /* 20060612.1.a 30ms delay. Add the follow 2 lines */
711 /* To adjust TXVGA to fit iq_mag_0 range from 1250 ~ 1750 */
712 adjust_TXVGA_for_iq_mag(phw_data);
Pavel Machek66101de2008-10-01 14:36:56 +0200713
Timofey Trofimov3a4d1b92010-05-25 18:19:54 +0400714 /* a. Disable AGC */
Pavel Machek66101de2008-10-01 14:36:56 +0200715 hw_get_dxx_reg(phw_data, REG_AGC_CTRL3, &reg_agc_ctrl3);
716 reg_agc_ctrl3 &= ~BIT(2);
717 reg_agc_ctrl3 |= (MASK_LNA_FIX_GAIN|MASK_AGC_FIX);
718 hw_set_dxx_reg(phw_data, REG_AGC_CTRL3, reg_agc_ctrl3);
719
720 hw_get_dxx_reg(phw_data, REG_AGC_CTRL5, &val);
721 val |= MASK_AGC_FIX_GAIN;
722 hw_set_dxx_reg(phw_data, REG_AGC_CTRL5, val);
723
724 result = _tx_iq_calibration_loop_winbond(phw_data, 150, 100);
725
Timofey Trofimov3a4d1b92010-05-25 18:19:54 +0400726 if (result > 0) {
727 if (phw_data->revision == 0x2002) /* 1st-cut */{
Pavel Machek66101de2008-10-01 14:36:56 +0200728 hw_get_dxx_reg(phw_data, 0x54, &val);
729 val &= 0x0000FFFF;
730 hw_set_dxx_reg(phw_data, 0x54, val);
Timofey Trofimov3a4d1b92010-05-25 18:19:54 +0400731 } else /* 2nd-cut*/{
Pavel Machek66101de2008-10-01 14:36:56 +0200732 hw_get_dxx_reg(phw_data, 0x3C, &val);
733 val &= 0x000003FF;
734 hw_set_dxx_reg(phw_data, 0x3C, val);
735 }
736
737 result = _tx_iq_calibration_loop_winbond(phw_data, 300, 200);
738
Timofey Trofimov3a4d1b92010-05-25 18:19:54 +0400739 if (result > 0) {
740 if (phw_data->revision == 0x2002) /* 1st-cut */{
Pavel Machek66101de2008-10-01 14:36:56 +0200741 hw_get_dxx_reg(phw_data, 0x54, &val);
742 val &= 0x0000FFFF;
743 hw_set_dxx_reg(phw_data, 0x54, val);
Timofey Trofimov3a4d1b92010-05-25 18:19:54 +0400744 } else /* 2nd-cut*/{
Pavel Machek66101de2008-10-01 14:36:56 +0200745 hw_get_dxx_reg(phw_data, 0x3C, &val);
746 val &= 0x000003FF;
747 hw_set_dxx_reg(phw_data, 0x3C, val);
748 }
749
750 result = _tx_iq_calibration_loop_winbond(phw_data, 500, 400);
Timofey Trofimov3a4d1b92010-05-25 18:19:54 +0400751 if (result > 0) {
752 if (phw_data->revision == 0x2002) /* 1st-cut */{
Pavel Machek66101de2008-10-01 14:36:56 +0200753 hw_get_dxx_reg(phw_data, 0x54, &val);
754 val &= 0x0000FFFF;
755 hw_set_dxx_reg(phw_data, 0x54, val);
Timofey Trofimov3a4d1b92010-05-25 18:19:54 +0400756 } else /* 2nd-cut */{
Pavel Machek66101de2008-10-01 14:36:56 +0200757 hw_get_dxx_reg(phw_data, 0x3C, &val);
758 val &= 0x000003FF;
759 hw_set_dxx_reg(phw_data, 0x3C, val);
760 }
761
762
763 result = _tx_iq_calibration_loop_winbond(phw_data, 700, 500);
764
Timofey Trofimov3a4d1b92010-05-25 18:19:54 +0400765 if (result > 0) {
Pavel Machek66101de2008-10-01 14:36:56 +0200766 PHY_DEBUG(("[CAL] ** <_tx_iq_calibration> **************\n"));
767 PHY_DEBUG(("[CAL] ** TX_IQ_CALIBRATION FAILURE !!\n"));
768 PHY_DEBUG(("[CAL] **************************************\n"));
769
Timofey Trofimov3a4d1b92010-05-25 18:19:54 +0400770 if (phw_data->revision == 0x2002) /* 1st-cut */{
Pavel Machek66101de2008-10-01 14:36:56 +0200771 hw_get_dxx_reg(phw_data, 0x54, &val);
772 val &= 0x0000FFFF;
773 hw_set_dxx_reg(phw_data, 0x54, val);
Timofey Trofimov3a4d1b92010-05-25 18:19:54 +0400774 } else /* 2nd-cut */{
Pavel Machek66101de2008-10-01 14:36:56 +0200775 hw_get_dxx_reg(phw_data, 0x3C, &val);
776 val &= 0x000003FF;
777 hw_set_dxx_reg(phw_data, 0x3C, val);
778 }
779 }
780 }
781 }
782 }
783
Timofey Trofimov3a4d1b92010-05-25 18:19:54 +0400784 /* i. Set "calib_start" to 0x0 */
Pavel Machek66101de2008-10-01 14:36:56 +0200785 hw_get_dxx_reg(phw_data, REG_MODE_CTRL, &reg_mode_ctrl);
786 reg_mode_ctrl &= ~MASK_CALIB_START;
787 hw_set_dxx_reg(phw_data, REG_MODE_CTRL, reg_mode_ctrl);
788 PHY_DEBUG(("[CAL] MODE_CTRL (write) = 0x%08X\n", reg_mode_ctrl));
789
Timofey Trofimov3a4d1b92010-05-25 18:19:54 +0400790 /* g. Enable AGC */
791 /* hw_get_dxx_reg(phw_data, REG_AGC_CTRL3, &val); */
Pavel Machek66101de2008-10-01 14:36:56 +0200792 reg_agc_ctrl3 |= BIT(2);
793 reg_agc_ctrl3 &= ~(MASK_LNA_FIX_GAIN|MASK_AGC_FIX);
794 hw_set_dxx_reg(phw_data, REG_AGC_CTRL3, reg_agc_ctrl3);
795
796#ifdef _DEBUG
Timofey Trofimov3a4d1b92010-05-25 18:19:54 +0400797 if (phw_data->revision == 0x2002) /* 1st-cut */{
Pavel Machek66101de2008-10-01 14:36:56 +0200798 hw_get_dxx_reg(phw_data, 0x54, &val);
799 PHY_DEBUG(("[CAL] ** 0x54 = 0x%08X\n", val));
800 tx_cal_reg[0] = _s4_to_s32((val & 0xF0000000) >> 28);
801 tx_cal_reg[1] = _s4_to_s32((val & 0x0F000000) >> 24);
802 tx_cal_reg[2] = _s4_to_s32((val & 0x00F00000) >> 20);
803 tx_cal_reg[3] = _s4_to_s32((val & 0x000F0000) >> 16);
Timofey Trofimov3a4d1b92010-05-25 18:19:54 +0400804 } else /* 2nd-cut */ {
Pavel Machek66101de2008-10-01 14:36:56 +0200805 hw_get_dxx_reg(phw_data, 0x3C, &val);
806 PHY_DEBUG(("[CAL] ** 0x3C = 0x%08X\n", val));
807 tx_cal_reg[0] = _s5_to_s32((val & 0xF8000000) >> 27);
808 tx_cal_reg[1] = _s6_to_s32((val & 0x07E00000) >> 21);
809 tx_cal_reg[2] = _s6_to_s32((val & 0x001F8000) >> 15);
810 tx_cal_reg[3] = _s5_to_s32((val & 0x00007C00) >> 10);
811
812 }
813
814 PHY_DEBUG(("[CAL] ** tx_cal_reg[0] = %d\n", tx_cal_reg[0]));
815 PHY_DEBUG(("[CAL] tx_cal_reg[1] = %d\n", tx_cal_reg[1]));
816 PHY_DEBUG(("[CAL] tx_cal_reg[2] = %d\n", tx_cal_reg[2]));
817 PHY_DEBUG(("[CAL] tx_cal_reg[3] = %d\n", tx_cal_reg[3]));
818#endif
819
820
Timofey Trofimov3a4d1b92010-05-25 18:19:54 +0400821 /*
822 * for test - BEN
823 * RF Control Override
824 */
Pavel Machek66101de2008-10-01 14:36:56 +0200825}
826
Timofey Trofimov3a4d1b92010-05-25 18:19:54 +0400827/*****************************************************/
Ebru Akagunduz06c789e2013-10-29 10:08:46 +0200828static u8 _rx_iq_calibration_loop_winbond(struct hw_data *phw_data, u16 factor, u32 frequency)
Pavel Machek66101de2008-10-01 14:36:56 +0200829{
830 u32 reg_mode_ctrl;
831 s32 iqcal_tone_i;
832 s32 iqcal_tone_q;
833 s32 iqcal_image_i;
834 s32 iqcal_image_q;
835 s32 rot_tone_i_b;
836 s32 rot_tone_q_b;
837 s32 rot_image_i_b;
838 s32 rot_image_q_b;
839 s32 rx_cal_flt_b[4];
840 s32 rx_cal[4];
841 s32 rx_cal_reg[4];
842 s32 a_2, b_2;
843 s32 sin_b, sin_2b;
844 s32 cos_b, cos_2b;
845 s32 temp1, temp2;
846 u32 val;
847 u16 loop;
848
849 u32 pwr_tone;
850 u32 pwr_image;
851 u8 verify_count;
852
Timofey Trofimov3a4d1b92010-05-25 18:19:54 +0400853 s32 iqcal_tone_i_avg, iqcal_tone_q_avg;
854 s32 iqcal_image_i_avg, iqcal_image_q_avg;
855 u16 capture_time;
Pavel Machek66101de2008-10-01 14:36:56 +0200856
857 PHY_DEBUG(("[CAL] -> [5]_rx_iq_calibration_loop()\n"));
858 PHY_DEBUG(("[CAL] ** factor = %d\n", factor));
859
Timofey Trofimov3a4d1b92010-05-25 18:19:54 +0400860 hw_set_dxx_reg(phw_data, 0x58, 0x44444444); /* IQ_Alpha */
Pavel Machek66101de2008-10-01 14:36:56 +0200861
Timofey Trofimov3a4d1b92010-05-25 18:19:54 +0400862 /* b. */
Pavel Machek66101de2008-10-01 14:36:56 +0200863
864 hw_get_dxx_reg(phw_data, REG_MODE_CTRL, &reg_mode_ctrl);
865 PHY_DEBUG(("[CAL] MODE_CTRL (read) = 0x%08X\n", reg_mode_ctrl));
866
867 verify_count = 0;
868
Timofey Trofimov3a4d1b92010-05-25 18:19:54 +0400869 /* for (loop = 0; loop < 1; loop++) */
870 /* for (loop = 0; loop < LOOP_TIMES; loop++) */
Pavel Machek66101de2008-10-01 14:36:56 +0200871 loop = LOOP_TIMES;
Timofey Trofimov3a4d1b92010-05-25 18:19:54 +0400872 while (loop > 0) {
Iker Pedrosade6c7642013-09-16 15:43:24 +0200873 PHY_DEBUG(("[CAL] [%d.] <_rx_iq_calibration_loop>\n",
874 (LOOP_TIMES-loop+1)));
Timofey Trofimov3a4d1b92010-05-25 18:19:54 +0400875 iqcal_tone_i_avg = 0;
876 iqcal_tone_q_avg = 0;
877 iqcal_image_i_avg = 0;
878 iqcal_image_q_avg = 0;
879 capture_time = 0;
Pavel Machek66101de2008-10-01 14:36:56 +0200880
Timofey Trofimov3a4d1b92010-05-25 18:19:54 +0400881 for (capture_time = 0; capture_time < 10; capture_time++) {
Akshay Joshi919ed522011-06-07 04:51:55 -0400882 /* i. Set "calib_start" to 0x0 */
883 reg_mode_ctrl &= ~MASK_CALIB_START;
884 if (!hw_set_dxx_reg(phw_data, REG_MODE_CTRL, reg_mode_ctrl))/*20060718.1 modify */
885 return 0;
886 PHY_DEBUG(("[CAL] MODE_CTRL (write) = 0x%08X\n", reg_mode_ctrl));
Pavel Machek66101de2008-10-01 14:36:56 +0200887
Akshay Joshi919ed522011-06-07 04:51:55 -0400888 reg_mode_ctrl &= ~MASK_IQCAL_MODE;
889 reg_mode_ctrl |= (MASK_CALIB_START|0x1);
890 hw_set_dxx_reg(phw_data, REG_MODE_CTRL, reg_mode_ctrl);
891 PHY_DEBUG(("[CAL] MODE_CTRL (write) = 0x%08X\n", reg_mode_ctrl));
Pavel Machek66101de2008-10-01 14:36:56 +0200892
Akshay Joshi919ed522011-06-07 04:51:55 -0400893 /* c. */
894 hw_get_dxx_reg(phw_data, REG_CALIB_READ1, &val);
895 PHY_DEBUG(("[CAL] CALIB_READ1 = 0x%08X\n", val));
Pavel Machek66101de2008-10-01 14:36:56 +0200896
Akshay Joshi919ed522011-06-07 04:51:55 -0400897 iqcal_tone_i = _s13_to_s32(val & 0x00001FFF);
898 iqcal_tone_q = _s13_to_s32((val & 0x03FFE000) >> 13);
899 PHY_DEBUG(("[CAL] ** iqcal_tone_i = %d, iqcal_tone_q = %d\n",
900 iqcal_tone_i, iqcal_tone_q));
Pavel Machek66101de2008-10-01 14:36:56 +0200901
Akshay Joshi919ed522011-06-07 04:51:55 -0400902 hw_get_dxx_reg(phw_data, REG_CALIB_READ2, &val);
903 PHY_DEBUG(("[CAL] CALIB_READ2 = 0x%08X\n", val));
Pavel Machek66101de2008-10-01 14:36:56 +0200904
Akshay Joshi919ed522011-06-07 04:51:55 -0400905 iqcal_image_i = _s13_to_s32(val & 0x00001FFF);
906 iqcal_image_q = _s13_to_s32((val & 0x03FFE000) >> 13);
907 PHY_DEBUG(("[CAL] ** iqcal_image_i = %d, iqcal_image_q = %d\n",
908 iqcal_image_i, iqcal_image_q));
Timofey Trofimov3a4d1b92010-05-25 18:19:54 +0400909 if (capture_time == 0)
Pavel Machek66101de2008-10-01 14:36:56 +0200910 continue;
Timofey Trofimov3a4d1b92010-05-25 18:19:54 +0400911 else {
912 iqcal_image_i_avg = (iqcal_image_i_avg*(capture_time-1) + iqcal_image_i)/capture_time;
913 iqcal_image_q_avg = (iqcal_image_q_avg*(capture_time-1) + iqcal_image_q)/capture_time;
914 iqcal_tone_i_avg = (iqcal_tone_i_avg*(capture_time-1) + iqcal_tone_i)/capture_time;
915 iqcal_tone_q_avg = (iqcal_tone_q_avg*(capture_time-1) + iqcal_tone_q)/capture_time;
Pavel Machek66101de2008-10-01 14:36:56 +0200916 }
917 }
918
919
920 iqcal_image_i = iqcal_image_i_avg;
921 iqcal_image_q = iqcal_image_q_avg;
922 iqcal_tone_i = iqcal_tone_i_avg;
923 iqcal_tone_q = iqcal_tone_q_avg;
924
Timofey Trofimov3a4d1b92010-05-25 18:19:54 +0400925 /* d. */
Pavel Machek66101de2008-10-01 14:36:56 +0200926 rot_tone_i_b = (iqcal_tone_i * iqcal_tone_i +
Iker Pedrosade6c7642013-09-16 15:43:24 +0200927 iqcal_tone_q * iqcal_tone_q) / 1024;
Pavel Machek66101de2008-10-01 14:36:56 +0200928 rot_tone_q_b = (iqcal_tone_i * iqcal_tone_q * (-1) +
Iker Pedrosade6c7642013-09-16 15:43:24 +0200929 iqcal_tone_q * iqcal_tone_i) / 1024;
Pavel Machek66101de2008-10-01 14:36:56 +0200930 rot_image_i_b = (iqcal_image_i * iqcal_tone_i -
Iker Pedrosade6c7642013-09-16 15:43:24 +0200931 iqcal_image_q * iqcal_tone_q) / 1024;
Pavel Machek66101de2008-10-01 14:36:56 +0200932 rot_image_q_b = (iqcal_image_i * iqcal_tone_q +
Iker Pedrosade6c7642013-09-16 15:43:24 +0200933 iqcal_image_q * iqcal_tone_i) / 1024;
Pavel Machek66101de2008-10-01 14:36:56 +0200934
935 PHY_DEBUG(("[CAL] ** rot_tone_i_b = %d\n", rot_tone_i_b));
936 PHY_DEBUG(("[CAL] ** rot_tone_q_b = %d\n", rot_tone_q_b));
937 PHY_DEBUG(("[CAL] ** rot_image_i_b = %d\n", rot_image_i_b));
938 PHY_DEBUG(("[CAL] ** rot_image_q_b = %d\n", rot_image_q_b));
939
Timofey Trofimov3a4d1b92010-05-25 18:19:54 +0400940 /* f. */
941 if (rot_tone_i_b == 0) {
Pavel Machek66101de2008-10-01 14:36:56 +0200942 PHY_DEBUG(("[CAL] ** <_rx_iq_calibration_loop> ERROR *******\n"));
943 PHY_DEBUG(("[CAL] ** rot_tone_i_b=0 to calculate EPS and THETA !!\n"));
944 PHY_DEBUG(("[CAL] ******************************************\n"));
945 break;
946 }
947
948 a_2 = (rot_image_i_b * 32768) / rot_tone_i_b -
949 phw_data->iq_rsdl_gain_tx_d2;
950 b_2 = (rot_image_q_b * 32768) / rot_tone_i_b -
951 phw_data->iq_rsdl_phase_tx_d2;
952
Iker Pedrosa36b30c52013-09-16 15:43:25 +0200953 PHY_DEBUG(("[CAL] ** iq_rsdl_gain_tx_d2 = %d\n",
954 phw_data->iq_rsdl_gain_tx_d2));
955 PHY_DEBUG(("[CAL] ** iq_rsdl_phase_tx_d2= %d\n",
956 phw_data->iq_rsdl_phase_tx_d2));
Pavel Machek66101de2008-10-01 14:36:56 +0200957 PHY_DEBUG(("[CAL] ***** EPSILON/2 = %d\n", a_2));
958 PHY_DEBUG(("[CAL] ***** THETA/2 = %d\n", b_2));
959
960 _sin_cos(b_2, &sin_b, &cos_b);
961 _sin_cos(b_2*2, &sin_2b, &cos_2b);
962 PHY_DEBUG(("[CAL] ** sin(b/2)=%d, cos(b/2)=%d\n", sin_b, cos_b));
963 PHY_DEBUG(("[CAL] ** sin(b)=%d, cos(b)=%d\n", sin_2b, cos_2b));
964
Timofey Trofimov3a4d1b92010-05-25 18:19:54 +0400965 if (cos_2b == 0) {
Pavel Machek66101de2008-10-01 14:36:56 +0200966 PHY_DEBUG(("[CAL] ** <_rx_iq_calibration_loop> ERROR *******\n"));
967 PHY_DEBUG(("[CAL] ** cos(b)=0 !!\n"));
968 PHY_DEBUG(("[CAL] ******************************************\n"));
969 break;
970 }
971
Timofey Trofimov3a4d1b92010-05-25 18:19:54 +0400972 /* 1280 * 32768 = 41943040 */
Pavel Machek66101de2008-10-01 14:36:56 +0200973 temp1 = (41943040/cos_2b)*cos_b;
974
Timofey Trofimov3a4d1b92010-05-25 18:19:54 +0400975 /* temp2 = (41943040/cos_2b)*sin_b*(-1); */
976 if (phw_data->revision == 0x2002)/* 1st-cut */
Pavel Machek66101de2008-10-01 14:36:56 +0200977 temp2 = (41943040/cos_2b)*sin_b*(-1);
Timofey Trofimov3a4d1b92010-05-25 18:19:54 +0400978 else/* 2nd-cut */
Pavel Machek66101de2008-10-01 14:36:56 +0200979 temp2 = (41943040*4/cos_2b)*sin_b*(-1);
Pavel Machek66101de2008-10-01 14:36:56 +0200980
981 rx_cal_flt_b[0] = _floor(temp1/(32768+a_2));
982 rx_cal_flt_b[1] = _floor(temp2/(32768-a_2));
983 rx_cal_flt_b[2] = _floor(temp2/(32768+a_2));
984 rx_cal_flt_b[3] = _floor(temp1/(32768-a_2));
985
986 PHY_DEBUG(("[CAL] ** rx_cal_flt_b[0] = %d\n", rx_cal_flt_b[0]));
987 PHY_DEBUG(("[CAL] rx_cal_flt_b[1] = %d\n", rx_cal_flt_b[1]));
988 PHY_DEBUG(("[CAL] rx_cal_flt_b[2] = %d\n", rx_cal_flt_b[2]));
989 PHY_DEBUG(("[CAL] rx_cal_flt_b[3] = %d\n", rx_cal_flt_b[3]));
990
991 rx_cal[0] = rx_cal_flt_b[0] - 128;
992 rx_cal[1] = rx_cal_flt_b[1];
993 rx_cal[2] = rx_cal_flt_b[2];
994 rx_cal[3] = rx_cal_flt_b[3] - 128;
995 PHY_DEBUG(("[CAL] ** rx_cal[0] = %d\n", rx_cal[0]));
996 PHY_DEBUG(("[CAL] rx_cal[1] = %d\n", rx_cal[1]));
997 PHY_DEBUG(("[CAL] rx_cal[2] = %d\n", rx_cal[2]));
998 PHY_DEBUG(("[CAL] rx_cal[3] = %d\n", rx_cal[3]));
999
Timofey Trofimov3a4d1b92010-05-25 18:19:54 +04001000 /* e. */
Pavel Machek66101de2008-10-01 14:36:56 +02001001 pwr_tone = (iqcal_tone_i*iqcal_tone_i + iqcal_tone_q*iqcal_tone_q);
Iker Pedrosa36b30c52013-09-16 15:43:25 +02001002 pwr_image = (iqcal_image_i*iqcal_image_i +
1003 iqcal_image_q*iqcal_image_q)*factor;
Pavel Machek66101de2008-10-01 14:36:56 +02001004
1005 PHY_DEBUG(("[CAL] ** pwr_tone = %d\n", pwr_tone));
1006 PHY_DEBUG(("[CAL] ** pwr_image = %d\n", pwr_image));
1007
Timofey Trofimov3a4d1b92010-05-25 18:19:54 +04001008 if (pwr_tone > pwr_image) {
Pavel Machek66101de2008-10-01 14:36:56 +02001009 verify_count++;
1010
1011 PHY_DEBUG(("[CAL] ** <_rx_iq_calibration_loop> *************\n"));
1012 PHY_DEBUG(("[CAL] ** VERIFY OK # %d !!\n", verify_count));
1013 PHY_DEBUG(("[CAL] ******************************************\n"));
1014
Timofey Trofimov3a4d1b92010-05-25 18:19:54 +04001015 if (verify_count > 2) {
Pavel Machek66101de2008-10-01 14:36:56 +02001016 PHY_DEBUG(("[CAL] ** <_rx_iq_calibration_loop> *********\n"));
1017 PHY_DEBUG(("[CAL] ** RX_IQ_CALIBRATION OK !!\n"));
1018 PHY_DEBUG(("[CAL] **************************************\n"));
1019 return 0;
1020 }
1021
1022 continue;
1023 }
Timofey Trofimov3a4d1b92010-05-25 18:19:54 +04001024 /* g. */
Pavel Machek66101de2008-10-01 14:36:56 +02001025 hw_get_dxx_reg(phw_data, 0x54, &val);
1026 PHY_DEBUG(("[CAL] ** 0x54 = 0x%08X\n", val));
1027
Timofey Trofimov3a4d1b92010-05-25 18:19:54 +04001028 if (phw_data->revision == 0x2002) /* 1st-cut */{
Pavel Machek66101de2008-10-01 14:36:56 +02001029 rx_cal_reg[0] = _s4_to_s32((val & 0x0000F000) >> 12);
1030 rx_cal_reg[1] = _s4_to_s32((val & 0x00000F00) >> 8);
1031 rx_cal_reg[2] = _s4_to_s32((val & 0x000000F0) >> 4);
1032 rx_cal_reg[3] = _s4_to_s32((val & 0x0000000F));
Timofey Trofimov3a4d1b92010-05-25 18:19:54 +04001033 } else /* 2nd-cut */{
Pavel Machek66101de2008-10-01 14:36:56 +02001034 rx_cal_reg[0] = _s5_to_s32((val & 0xF8000000) >> 27);
1035 rx_cal_reg[1] = _s6_to_s32((val & 0x07E00000) >> 21);
1036 rx_cal_reg[2] = _s6_to_s32((val & 0x001F8000) >> 15);
1037 rx_cal_reg[3] = _s5_to_s32((val & 0x00007C00) >> 10);
1038 }
1039
1040 PHY_DEBUG(("[CAL] ** rx_cal_reg[0] = %d\n", rx_cal_reg[0]));
1041 PHY_DEBUG(("[CAL] rx_cal_reg[1] = %d\n", rx_cal_reg[1]));
1042 PHY_DEBUG(("[CAL] rx_cal_reg[2] = %d\n", rx_cal_reg[2]));
1043 PHY_DEBUG(("[CAL] rx_cal_reg[3] = %d\n", rx_cal_reg[3]));
1044
Timofey Trofimov3a4d1b92010-05-25 18:19:54 +04001045 if (phw_data->revision == 0x2002) /* 1st-cut */{
1046 if (((rx_cal_reg[0] == 7) || (rx_cal_reg[0] == (-8))) &&
1047 ((rx_cal_reg[3] == 7) || (rx_cal_reg[3] == (-8)))) {
Pavel Machek66101de2008-10-01 14:36:56 +02001048 PHY_DEBUG(("[CAL] ** <_rx_iq_calibration_loop> *********\n"));
1049 PHY_DEBUG(("[CAL] ** RX_IQ_CALIBRATION SATUATION !!\n"));
1050 PHY_DEBUG(("[CAL] **************************************\n"));
1051 break;
1052 }
Timofey Trofimov3a4d1b92010-05-25 18:19:54 +04001053 } else /* 2nd-cut */{
1054 if (((rx_cal_reg[0] == 31) || (rx_cal_reg[0] == (-32))) &&
1055 ((rx_cal_reg[3] == 31) || (rx_cal_reg[3] == (-32)))) {
Pavel Machek66101de2008-10-01 14:36:56 +02001056 PHY_DEBUG(("[CAL] ** <_rx_iq_calibration_loop> *********\n"));
1057 PHY_DEBUG(("[CAL] ** RX_IQ_CALIBRATION SATUATION !!\n"));
1058 PHY_DEBUG(("[CAL] **************************************\n"));
1059 break;
1060 }
1061 }
1062
1063 rx_cal[0] = rx_cal[0] + rx_cal_reg[0];
1064 rx_cal[1] = rx_cal[1] + rx_cal_reg[1];
1065 rx_cal[2] = rx_cal[2] + rx_cal_reg[2];
1066 rx_cal[3] = rx_cal[3] + rx_cal_reg[3];
1067 PHY_DEBUG(("[CAL] ** apply rx_cal[0] = %d\n", rx_cal[0]));
1068 PHY_DEBUG(("[CAL] apply rx_cal[1] = %d\n", rx_cal[1]));
1069 PHY_DEBUG(("[CAL] apply rx_cal[2] = %d\n", rx_cal[2]));
1070 PHY_DEBUG(("[CAL] apply rx_cal[3] = %d\n", rx_cal[3]));
1071
1072 hw_get_dxx_reg(phw_data, 0x54, &val);
Timofey Trofimov3a4d1b92010-05-25 18:19:54 +04001073 if (phw_data->revision == 0x2002) /* 1st-cut */{
Pavel Machek66101de2008-10-01 14:36:56 +02001074 val &= 0x0000FFFF;
1075 val |= ((_s32_to_s4(rx_cal[0]) << 12)|
1076 (_s32_to_s4(rx_cal[1]) << 8)|
1077 (_s32_to_s4(rx_cal[2]) << 4)|
1078 (_s32_to_s4(rx_cal[3])));
1079 hw_set_dxx_reg(phw_data, 0x54, val);
Timofey Trofimov3a4d1b92010-05-25 18:19:54 +04001080 } else /* 2nd-cut */{
Pavel Machek66101de2008-10-01 14:36:56 +02001081 val &= 0x000003FF;
1082 val |= ((_s32_to_s5(rx_cal[0]) << 27)|
1083 (_s32_to_s6(rx_cal[1]) << 21)|
1084 (_s32_to_s6(rx_cal[2]) << 15)|
1085 (_s32_to_s5(rx_cal[3]) << 10));
1086 hw_set_dxx_reg(phw_data, 0x54, val);
1087
Timofey Trofimov3a4d1b92010-05-25 18:19:54 +04001088 if (loop == 3)
Akshay Joshi919ed522011-06-07 04:51:55 -04001089 return 0;
Pavel Machek66101de2008-10-01 14:36:56 +02001090 }
1091 PHY_DEBUG(("[CAL] ** CALIB_DATA = 0x%08X\n", val));
1092
1093 loop--;
1094 }
1095
1096 return 1;
1097}
1098
Timofey Trofimov3a4d1b92010-05-25 18:19:54 +04001099/*************************************************/
Pavel Machek66101de2008-10-01 14:36:56 +02001100
Timofey Trofimov3a4d1b92010-05-25 18:19:54 +04001101/***************************************************************/
Ebru Akagunduz06c789e2013-10-29 10:08:46 +02001102static void _rx_iq_calibration_winbond(struct hw_data *phw_data, u32 frequency)
Pavel Machek66101de2008-10-01 14:36:56 +02001103{
Justin P. Mattocka31f7f52012-09-03 08:06:02 -07001104/* figo 20050523 marked this flag for can't compile for release */
Pavel Machek66101de2008-10-01 14:36:56 +02001105#ifdef _DEBUG
1106 s32 rx_cal_reg[4];
1107 u32 val;
1108#endif
1109
1110 u8 result;
1111
1112 PHY_DEBUG(("[CAL] -> [5]_rx_iq_calibration()\n"));
Timofey Trofimov3a4d1b92010-05-25 18:19:54 +04001113/* a. Set RFIC to "RX calibration mode" */
1114 /* ; ----- Calibration (7). RX path IQ imbalance calibration loop */
1115 /* 0x01 0xFFBFC2 ; 3FEFF ; Calibration (7a). enable RX IQ calibration loop circuits */
Pavel Machek66101de2008-10-01 14:36:56 +02001116 phy_set_rf_data(phw_data, 1, (1<<24)|0xEFBFC2);
Timofey Trofimov3a4d1b92010-05-25 18:19:54 +04001117 /* 0x0B 0x1A01D6 ; 06817 ; Calibration (7b). enable RX I/Q cal loop SW1 circuits */
Pavel Machek66101de2008-10-01 14:36:56 +02001118 phy_set_rf_data(phw_data, 11, (11<<24)|0x1A05D6);
Timofey Trofimov3a4d1b92010-05-25 18:19:54 +04001119 /* 0x05 0x24848A ; 09212 ; Calibration (7c). setting TX-VGA gain (TXGCH) to 2 --> to be optimized */
1120 phy_set_rf_data(phw_data, 5, (5<<24) | phw_data->txvga_setting_for_cal);
1121 /* 0x06 0x06840C ; 01A10 ; Calibration (7d). RXGCH=00; RXGCL=010 000 (RXVGA) --> to be optimized */
Pavel Machek66101de2008-10-01 14:36:56 +02001122 phy_set_rf_data(phw_data, 6, (6<<24)|0x06834C);
Timofey Trofimov3a4d1b92010-05-25 18:19:54 +04001123 /* 0x00 0xFFF1C0 ; 3F7C7 ; Calibration (7e). turn on IQ imbalance/Test mode */
Pavel Machek66101de2008-10-01 14:36:56 +02001124 phy_set_rf_data(phw_data, 0, (0<<24)|0xFFF1C0);
1125
Timofey Trofimov3a4d1b92010-05-25 18:19:54 +04001126 /* ; [BB-chip]: Calibration (7f). Send test pattern */
1127 /* ; [BB-chip]: Calibration (7g). Search RXGCL optimal value */
Justin P. Mattocka31f7f52012-09-03 08:06:02 -07001128 /* ; [BB-chip]: Calibration (7h). Calculate RX-path IQ imbalance and setting RX path IQ compensation table */
Pavel Machek66101de2008-10-01 14:36:56 +02001129
1130 result = _rx_iq_calibration_loop_winbond(phw_data, 12589, frequency);
1131
Timofey Trofimov3a4d1b92010-05-25 18:19:54 +04001132 if (result > 0) {
Pavel Machek66101de2008-10-01 14:36:56 +02001133 _reset_rx_cal(phw_data);
1134 result = _rx_iq_calibration_loop_winbond(phw_data, 7943, frequency);
1135
Timofey Trofimov3a4d1b92010-05-25 18:19:54 +04001136 if (result > 0) {
Pavel Machek66101de2008-10-01 14:36:56 +02001137 _reset_rx_cal(phw_data);
1138 result = _rx_iq_calibration_loop_winbond(phw_data, 5011, frequency);
1139
Timofey Trofimov3a4d1b92010-05-25 18:19:54 +04001140 if (result > 0) {
Pavel Machek66101de2008-10-01 14:36:56 +02001141 PHY_DEBUG(("[CAL] ** <_rx_iq_calibration> **************\n"));
1142 PHY_DEBUG(("[CAL] ** RX_IQ_CALIBRATION FAILURE !!\n"));
1143 PHY_DEBUG(("[CAL] **************************************\n"));
1144 _reset_rx_cal(phw_data);
1145 }
1146 }
1147 }
1148
1149#ifdef _DEBUG
1150 hw_get_dxx_reg(phw_data, 0x54, &val);
1151 PHY_DEBUG(("[CAL] ** 0x54 = 0x%08X\n", val));
1152
Timofey Trofimov3a4d1b92010-05-25 18:19:54 +04001153 if (phw_data->revision == 0x2002) /* 1st-cut */{
Pavel Machek66101de2008-10-01 14:36:56 +02001154 rx_cal_reg[0] = _s4_to_s32((val & 0x0000F000) >> 12);
1155 rx_cal_reg[1] = _s4_to_s32((val & 0x00000F00) >> 8);
1156 rx_cal_reg[2] = _s4_to_s32((val & 0x000000F0) >> 4);
1157 rx_cal_reg[3] = _s4_to_s32((val & 0x0000000F));
Timofey Trofimov3a4d1b92010-05-25 18:19:54 +04001158 } else /* 2nd-cut */{
Pavel Machek66101de2008-10-01 14:36:56 +02001159 rx_cal_reg[0] = _s5_to_s32((val & 0xF8000000) >> 27);
1160 rx_cal_reg[1] = _s6_to_s32((val & 0x07E00000) >> 21);
1161 rx_cal_reg[2] = _s6_to_s32((val & 0x001F8000) >> 15);
1162 rx_cal_reg[3] = _s5_to_s32((val & 0x00007C00) >> 10);
1163 }
1164
1165 PHY_DEBUG(("[CAL] ** rx_cal_reg[0] = %d\n", rx_cal_reg[0]));
1166 PHY_DEBUG(("[CAL] rx_cal_reg[1] = %d\n", rx_cal_reg[1]));
1167 PHY_DEBUG(("[CAL] rx_cal_reg[2] = %d\n", rx_cal_reg[2]));
1168 PHY_DEBUG(("[CAL] rx_cal_reg[3] = %d\n", rx_cal_reg[3]));
1169#endif
1170
1171}
1172
Timofey Trofimov3a4d1b92010-05-25 18:19:54 +04001173/*******************************************************/
Pekka Enberg8e41b4b2009-01-12 18:02:47 +02001174void phy_calibration_winbond(struct hw_data *phw_data, u32 frequency)
Pavel Machek66101de2008-10-01 14:36:56 +02001175{
1176 u32 reg_mode_ctrl;
1177 u32 iq_alpha;
1178
1179 PHY_DEBUG(("[CAL] -> phy_calibration_winbond()\n"));
1180
Pavel Machek66101de2008-10-01 14:36:56 +02001181 hw_get_dxx_reg(phw_data, 0x58, &iq_alpha);
1182
Pavel Machek66101de2008-10-01 14:36:56 +02001183 _rxadc_dc_offset_cancellation_winbond(phw_data, frequency);
Timofey Trofimov3a4d1b92010-05-25 18:19:54 +04001184 /* _txidac_dc_offset_cancellation_winbond(phw_data); */
Justin P. Mattocka31f7f52012-09-03 08:06:02 -07001185 /* _txqdac_dc_offset_cancellation_winbond(phw_data); */
Pavel Machek66101de2008-10-01 14:36:56 +02001186
1187 _tx_iq_calibration_winbond(phw_data);
1188 _rx_iq_calibration_winbond(phw_data, frequency);
1189
Timofey Trofimov3a4d1b92010-05-25 18:19:54 +04001190 /*********************************************************************/
Pavel Machek66101de2008-10-01 14:36:56 +02001191 hw_get_dxx_reg(phw_data, REG_MODE_CTRL, &reg_mode_ctrl);
Timofey Trofimov3a4d1b92010-05-25 18:19:54 +04001192 reg_mode_ctrl &= ~(MASK_IQCAL_TONE_SEL|MASK_IQCAL_MODE|MASK_CALIB_START); /* set when finish */
Pavel Machek66101de2008-10-01 14:36:56 +02001193 hw_set_dxx_reg(phw_data, REG_MODE_CTRL, reg_mode_ctrl);
1194 PHY_DEBUG(("[CAL] MODE_CTRL (write) = 0x%08X\n", reg_mode_ctrl));
1195
Timofey Trofimov3a4d1b92010-05-25 18:19:54 +04001196 /* i. Set RFIC to "Normal mode" */
Pavel Machek66101de2008-10-01 14:36:56 +02001197 hw_set_dxx_reg(phw_data, 0x58, iq_alpha);
1198
Timofey Trofimov3a4d1b92010-05-25 18:19:54 +04001199 /*********************************************************************/
Pavel Machek66101de2008-10-01 14:36:56 +02001200 phy_init_rf(phw_data);
1201
1202}
1203
Timofey Trofimov3a4d1b92010-05-25 18:19:54 +04001204/******************/
1205void phy_set_rf_data(struct hw_data *pHwData, u32 index, u32 value)
Pavel Machek66101de2008-10-01 14:36:56 +02001206{
Akshay Joshi919ed522011-06-07 04:51:55 -04001207 u32 ltmp = 0;
Pavel Machek66101de2008-10-01 14:36:56 +02001208
Akshay Joshi919ed522011-06-07 04:51:55 -04001209 switch (pHwData->phy_type) {
1210 case RF_MAXIM_2825:
1211 case RF_MAXIM_V1: /* 11g Winbond 2nd BB(with Phy board (v1) + Maxim 331) */
1212 ltmp = (1 << 31) | (0 << 30) | (18 << 24) | BitReverse(value, 18);
1213 break;
Pavel Machek66101de2008-10-01 14:36:56 +02001214
Akshay Joshi919ed522011-06-07 04:51:55 -04001215 case RF_MAXIM_2827:
1216 ltmp = (1 << 31) | (0 << 30) | (18 << 24) | BitReverse(value, 18);
1217 break;
Pavel Machek66101de2008-10-01 14:36:56 +02001218
Akshay Joshi919ed522011-06-07 04:51:55 -04001219 case RF_MAXIM_2828:
1220 ltmp = (1 << 31) | (0 << 30) | (18 << 24) | BitReverse(value, 18);
1221 break;
Pavel Machek66101de2008-10-01 14:36:56 +02001222
Akshay Joshi919ed522011-06-07 04:51:55 -04001223 case RF_MAXIM_2829:
1224 ltmp = (1 << 31) | (0 << 30) | (18 << 24) | BitReverse(value, 18);
1225 break;
Pavel Machek66101de2008-10-01 14:36:56 +02001226
Akshay Joshi919ed522011-06-07 04:51:55 -04001227 case RF_AIROHA_2230:
1228 case RF_AIROHA_2230S: /* 20060420 Add this */
1229 ltmp = (1 << 31) | (0 << 30) | (20 << 24) | BitReverse(value, 20);
1230 break;
Pavel Machek66101de2008-10-01 14:36:56 +02001231
Akshay Joshi919ed522011-06-07 04:51:55 -04001232 case RF_AIROHA_7230:
1233 ltmp = (1 << 31) | (0 << 30) | (24 << 24) | (value&0xffffff);
1234 break;
Pavel Machek66101de2008-10-01 14:36:56 +02001235
Akshay Joshi919ed522011-06-07 04:51:55 -04001236 case RF_WB_242:
1237 case RF_WB_242_1:/* 20060619.5 Add */
1238 ltmp = (1 << 31) | (0 << 30) | (24 << 24) | BitReverse(value, 24);
1239 break;
1240 }
Pavel Machek66101de2008-10-01 14:36:56 +02001241
Timofey Trofimov3a4d1b92010-05-25 18:19:54 +04001242 Wb35Reg_WriteSync(pHwData, 0x0864, ltmp);
Pavel Machek66101de2008-10-01 14:36:56 +02001243}
1244
Timofey Trofimov3a4d1b92010-05-25 18:19:54 +04001245/* 20060717 modify as Bruce's mail */
Pekka Enberg8e41b4b2009-01-12 18:02:47 +02001246unsigned char adjust_TXVGA_for_iq_mag(struct hw_data *phw_data)
Pavel Machek66101de2008-10-01 14:36:56 +02001247{
1248 int init_txvga = 0;
1249 u32 reg_mode_ctrl;
1250 u32 val;
1251 s32 iqcal_tone_i0;
1252 s32 iqcal_tone_q0;
1253 u32 sqsum;
1254 s32 iq_mag_0_tx;
Timofey Trofimov3a4d1b92010-05-25 18:19:54 +04001255 u8 reg_state;
1256 int current_txvga;
Pavel Machek66101de2008-10-01 14:36:56 +02001257
1258
1259 reg_state = 0;
Timofey Trofimov3a4d1b92010-05-25 18:19:54 +04001260 for (init_txvga = 0; init_txvga < 10; init_txvga++) {
1261 current_txvga = (0x24C40A|(init_txvga<<6));
1262 phy_set_rf_data(phw_data, 5, ((5<<24)|current_txvga));
Pavel Machek66101de2008-10-01 14:36:56 +02001263 phw_data->txvga_setting_for_cal = current_txvga;
1264
Timofey Trofimov3a4d1b92010-05-25 18:19:54 +04001265 msleep(30);/* 20060612.1.a */
Pavel Machek66101de2008-10-01 14:36:56 +02001266
Timofey Trofimov3a4d1b92010-05-25 18:19:54 +04001267 if (!hw_get_dxx_reg(phw_data, REG_MODE_CTRL, &reg_mode_ctrl))/* 20060718.1 modify */
Pekka Enberg279b6cc2008-10-27 22:46:39 +02001268 return false;
Pavel Machek66101de2008-10-01 14:36:56 +02001269
1270 PHY_DEBUG(("[CAL] MODE_CTRL (read) = 0x%08X\n", reg_mode_ctrl));
1271
Timofey Trofimov3a4d1b92010-05-25 18:19:54 +04001272 /*
1273 * a. Set iqcal_mode[1:0] to 0x2 and set "calib_start" to 0x1 to
1274 * enable "IQ alibration Mode II"
1275 */
Pavel Machek66101de2008-10-01 14:36:56 +02001276 reg_mode_ctrl &= ~(MASK_IQCAL_TONE_SEL|MASK_IQCAL_MODE);
1277 reg_mode_ctrl &= ~MASK_IQCAL_MODE;
1278 reg_mode_ctrl |= (MASK_CALIB_START|0x02);
1279 reg_mode_ctrl |= (MASK_CALIB_START|0x02|2<<2);
1280 hw_set_dxx_reg(phw_data, REG_MODE_CTRL, reg_mode_ctrl);
1281 PHY_DEBUG(("[CAL] MODE_CTRL (write) = 0x%08X\n", reg_mode_ctrl));
1282
Timofey Trofimov3a4d1b92010-05-25 18:19:54 +04001283 udelay(1);/* 20060612.1.a */
Pavel Machek66101de2008-10-01 14:36:56 +02001284
Timofey Trofimov3a4d1b92010-05-25 18:19:54 +04001285 udelay(300);/* 20060612.1.a */
Pavel Machek66101de2008-10-01 14:36:56 +02001286
Timofey Trofimov3a4d1b92010-05-25 18:19:54 +04001287 /* b. */
Pavel Machek66101de2008-10-01 14:36:56 +02001288 hw_get_dxx_reg(phw_data, REG_CALIB_READ1, &val);
1289
1290 PHY_DEBUG(("[CAL] CALIB_READ1 = 0x%08X\n", val));
Timofey Trofimov3a4d1b92010-05-25 18:19:54 +04001291 udelay(300);/* 20060612.1.a */
Pavel Machek66101de2008-10-01 14:36:56 +02001292
1293 iqcal_tone_i0 = _s13_to_s32(val & 0x00001FFF);
1294 iqcal_tone_q0 = _s13_to_s32((val & 0x03FFE000) >> 13);
1295 PHY_DEBUG(("[CAL] ** iqcal_tone_i0=%d, iqcal_tone_q0=%d\n",
1296 iqcal_tone_i0, iqcal_tone_q0));
1297
1298 sqsum = iqcal_tone_i0*iqcal_tone_i0 + iqcal_tone_q0*iqcal_tone_q0;
1299 iq_mag_0_tx = (s32) _sqrt(sqsum);
Iker Pedrosa36b30c52013-09-16 15:43:25 +02001300 PHY_DEBUG(("[CAL] ** auto_adjust_txvga_for_iq_mag_0_tx=%d\n",
1301 iq_mag_0_tx));
Pavel Machek66101de2008-10-01 14:36:56 +02001302
Timofey Trofimov3a4d1b92010-05-25 18:19:54 +04001303 if (iq_mag_0_tx >= 700 && iq_mag_0_tx <= 1750)
Pavel Machek66101de2008-10-01 14:36:56 +02001304 break;
Timofey Trofimov3a4d1b92010-05-25 18:19:54 +04001305 else if (iq_mag_0_tx > 1750) {
1306 init_txvga = -2;
Pavel Machek66101de2008-10-01 14:36:56 +02001307 continue;
Timofey Trofimov3a4d1b92010-05-25 18:19:54 +04001308 } else
Pavel Machek66101de2008-10-01 14:36:56 +02001309 continue;
1310
1311 }
1312
Timofey Trofimov3a4d1b92010-05-25 18:19:54 +04001313 if (iq_mag_0_tx >= 700 && iq_mag_0_tx <= 1750)
Pekka Enberg279b6cc2008-10-27 22:46:39 +02001314 return true;
Pavel Machek66101de2008-10-01 14:36:56 +02001315 else
Pekka Enberg279b6cc2008-10-27 22:46:39 +02001316 return false;
Pavel Machek66101de2008-10-01 14:36:56 +02001317}