Merge branch 'devel' of master.kernel.org:/home/rmk/linux-2.6-arm
[GitHub/mt8127/android_kernel_alcatel_ttab.git] / drivers / staging / winbond / phy_calibration.c
1 /*
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 ***********************************/
13 #include "sysdef.h"
14 #include "phy_calibration.h"
15 #include "wbhal_f.h"
16
17
18 /****************** DEBUG CONSTANT AND MACRO SECTION ************************/
19
20 /****************** LOCAL CONSTANT AND MACRO SECTION ************************/
21 #define LOOP_TIMES 20
22 #define US 1000//MICROSECOND
23
24 #define AG_CONST 0.6072529350
25 #define FIXED(X) ((s32)((X) * 32768.0))
26 #define DEG2RAD(X) 0.017453 * (X)
27
28 static const s32 Angles[] =
29 {
30 FIXED(DEG2RAD(45.0)), FIXED(DEG2RAD(26.565)), FIXED(DEG2RAD(14.0362)),
31 FIXED(DEG2RAD(7.12502)), FIXED(DEG2RAD(3.57633)), FIXED(DEG2RAD(1.78991)),
32 FIXED(DEG2RAD(0.895174)),FIXED(DEG2RAD(0.447614)),FIXED(DEG2RAD(0.223811)),
33 FIXED(DEG2RAD(0.111906)),FIXED(DEG2RAD(0.055953)),FIXED(DEG2RAD(0.027977))
34 };
35
36 /****************** LOCAL FUNCTION DECLARATION SECTION **********************/
37 //void _phy_rf_write_delay(struct hw_data *phw_data);
38 //void phy_init_rf(struct hw_data *phw_data);
39
40 /****************** FUNCTION DEFINITION SECTION *****************************/
41
42 s32 _s13_to_s32(u32 data)
43 {
44 u32 val;
45
46 val = (data & 0x0FFF);
47
48 if ((data & BIT(12)) != 0)
49 {
50 val |= 0xFFFFF000;
51 }
52
53 return ((s32) val);
54 }
55
56 u32 _s32_to_s13(s32 data)
57 {
58 u32 val;
59
60 if (data > 4095)
61 {
62 data = 4095;
63 }
64 else if (data < -4096)
65 {
66 data = -4096;
67 }
68
69 val = data & 0x1FFF;
70
71 return val;
72 }
73
74 /****************************************************************************/
75 s32 _s4_to_s32(u32 data)
76 {
77 s32 val;
78
79 val = (data & 0x0007);
80
81 if ((data & BIT(3)) != 0)
82 {
83 val |= 0xFFFFFFF8;
84 }
85
86 return val;
87 }
88
89 u32 _s32_to_s4(s32 data)
90 {
91 u32 val;
92
93 if (data > 7)
94 {
95 data = 7;
96 }
97 else if (data < -8)
98 {
99 data = -8;
100 }
101
102 val = data & 0x000F;
103
104 return val;
105 }
106
107 /****************************************************************************/
108 s32 _s5_to_s32(u32 data)
109 {
110 s32 val;
111
112 val = (data & 0x000F);
113
114 if ((data & BIT(4)) != 0)
115 {
116 val |= 0xFFFFFFF0;
117 }
118
119 return val;
120 }
121
122 u32 _s32_to_s5(s32 data)
123 {
124 u32 val;
125
126 if (data > 15)
127 {
128 data = 15;
129 }
130 else if (data < -16)
131 {
132 data = -16;
133 }
134
135 val = data & 0x001F;
136
137 return val;
138 }
139
140 /****************************************************************************/
141 s32 _s6_to_s32(u32 data)
142 {
143 s32 val;
144
145 val = (data & 0x001F);
146
147 if ((data & BIT(5)) != 0)
148 {
149 val |= 0xFFFFFFE0;
150 }
151
152 return val;
153 }
154
155 u32 _s32_to_s6(s32 data)
156 {
157 u32 val;
158
159 if (data > 31)
160 {
161 data = 31;
162 }
163 else if (data < -32)
164 {
165 data = -32;
166 }
167
168 val = data & 0x003F;
169
170 return val;
171 }
172
173 /****************************************************************************/
174 s32 _s9_to_s32(u32 data)
175 {
176 s32 val;
177
178 val = data & 0x00FF;
179
180 if ((data & BIT(8)) != 0)
181 {
182 val |= 0xFFFFFF00;
183 }
184
185 return val;
186 }
187
188 u32 _s32_to_s9(s32 data)
189 {
190 u32 val;
191
192 if (data > 255)
193 {
194 data = 255;
195 }
196 else if (data < -256)
197 {
198 data = -256;
199 }
200
201 val = data & 0x01FF;
202
203 return val;
204 }
205
206 /****************************************************************************/
207 s32 _floor(s32 n)
208 {
209 if (n > 0)
210 {
211 n += 5;
212 }
213 else
214 {
215 n -= 5;
216 }
217
218 return (n/10);
219 }
220
221 /****************************************************************************/
222 // The following code is sqare-root function.
223 // sqsum is the input and the output is sq_rt;
224 // The maximum of sqsum = 2^27 -1;
225 u32 _sqrt(u32 sqsum)
226 {
227 u32 sq_rt;
228
229 int g0, g1, g2, g3, g4;
230 int seed;
231 int next;
232 int step;
233
234 g4 = sqsum / 100000000;
235 g3 = (sqsum - g4*100000000) /1000000;
236 g2 = (sqsum - g4*100000000 - g3*1000000) /10000;
237 g1 = (sqsum - g4*100000000 - g3*1000000 - g2*10000) /100;
238 g0 = (sqsum - g4*100000000 - g3*1000000 - g2*10000 - g1*100);
239
240 next = g4;
241 step = 0;
242 seed = 0;
243 while (((seed+1)*(step+1)) <= next)
244 {
245 step++;
246 seed++;
247 }
248
249 sq_rt = seed * 10000;
250 next = (next-(seed*step))*100 + g3;
251
252 step = 0;
253 seed = 2 * seed * 10;
254 while (((seed+1)*(step+1)) <= next)
255 {
256 step++;
257 seed++;
258 }
259
260 sq_rt = sq_rt + step * 1000;
261 next = (next - seed * step) * 100 + g2;
262 seed = (seed + step) * 10;
263 step = 0;
264 while (((seed+1)*(step+1)) <= next)
265 {
266 step++;
267 seed++;
268 }
269
270 sq_rt = sq_rt + step * 100;
271 next = (next - seed * step) * 100 + g1;
272 seed = (seed + step) * 10;
273 step = 0;
274
275 while (((seed+1)*(step+1)) <= next)
276 {
277 step++;
278 seed++;
279 }
280
281 sq_rt = sq_rt + step * 10;
282 next = (next - seed* step) * 100 + g0;
283 seed = (seed + step) * 10;
284 step = 0;
285
286 while (((seed+1)*(step+1)) <= next)
287 {
288 step++;
289 seed++;
290 }
291
292 sq_rt = sq_rt + step;
293
294 return sq_rt;
295 }
296
297 /****************************************************************************/
298 void _sin_cos(s32 angle, s32 *sin, s32 *cos)
299 {
300 s32 X, Y, TargetAngle, CurrAngle;
301 unsigned Step;
302
303 X=FIXED(AG_CONST); // AG_CONST * cos(0)
304 Y=0; // AG_CONST * sin(0)
305 TargetAngle=abs(angle);
306 CurrAngle=0;
307
308 for (Step=0; Step < 12; Step++)
309 {
310 s32 NewX;
311
312 if(TargetAngle > CurrAngle)
313 {
314 NewX=X - (Y >> Step);
315 Y=(X >> Step) + Y;
316 X=NewX;
317 CurrAngle += Angles[Step];
318 }
319 else
320 {
321 NewX=X + (Y >> Step);
322 Y=-(X >> Step) + Y;
323 X=NewX;
324 CurrAngle -= Angles[Step];
325 }
326 }
327
328 if (angle > 0)
329 {
330 *cos = X;
331 *sin = Y;
332 }
333 else
334 {
335 *cos = X;
336 *sin = -Y;
337 }
338 }
339
340 static unsigned char hal_get_dxx_reg(struct hw_data *pHwData, u16 number, u32 * pValue)
341 {
342 if (number < 0x1000)
343 number += 0x1000;
344 return Wb35Reg_ReadSync(pHwData, number, pValue);
345 }
346 #define hw_get_dxx_reg( _A, _B, _C ) hal_get_dxx_reg( _A, _B, (u32 *)_C )
347
348 static unsigned char hal_set_dxx_reg(struct hw_data *pHwData, u16 number, u32 value)
349 {
350 unsigned char ret;
351
352 if (number < 0x1000)
353 number += 0x1000;
354 ret = Wb35Reg_WriteSync(pHwData, number, value);
355 return ret;
356 }
357 #define hw_set_dxx_reg( _A, _B, _C ) hal_set_dxx_reg( _A, _B, (u32)_C )
358
359
360 void _reset_rx_cal(struct hw_data *phw_data)
361 {
362 u32 val;
363
364 hw_get_dxx_reg(phw_data, 0x54, &val);
365
366 if (phw_data->revision == 0x2002) // 1st-cut
367 {
368 val &= 0xFFFF0000;
369 }
370 else // 2nd-cut
371 {
372 val &= 0x000003FF;
373 }
374
375 hw_set_dxx_reg(phw_data, 0x54, val);
376 }
377
378
379 // ************for winbond calibration*********
380 //
381
382 //
383 //
384 // *********************************************
385 void _rxadc_dc_offset_cancellation_winbond(struct hw_data *phw_data, u32 frequency)
386 {
387 u32 reg_agc_ctrl3;
388 u32 reg_a_acq_ctrl;
389 u32 reg_b_acq_ctrl;
390 u32 val;
391
392 PHY_DEBUG(("[CAL] -> [1]_rxadc_dc_offset_cancellation()\n"));
393 phy_init_rf(phw_data);
394
395 // set calibration channel
396 if( (RF_WB_242 == phw_data->phy_type) ||
397 (RF_WB_242_1 == phw_data->phy_type) ) // 20060619.5 Add
398 {
399 if ((frequency >= 2412) && (frequency <= 2484))
400 {
401 // w89rf242 change frequency to 2390Mhz
402 PHY_DEBUG(("[CAL] W89RF242/11G/Channel=2390Mhz\n"));
403 phy_set_rf_data(phw_data, 3, (3<<24)|0x025586);
404
405 }
406 }
407 else
408 {
409
410 }
411
412 // reset cancel_dc_i[9:5] and cancel_dc_q[4:0] in register DC_Cancel
413 hw_get_dxx_reg(phw_data, 0x5C, &val);
414 val &= ~(0x03FF);
415 hw_set_dxx_reg(phw_data, 0x5C, val);
416
417 // reset the TX and RX IQ calibration data
418 hw_set_dxx_reg(phw_data, 0x3C, 0);
419 hw_set_dxx_reg(phw_data, 0x54, 0);
420
421 hw_set_dxx_reg(phw_data, 0x58, 0x30303030); // IQ_Alpha Changed
422
423 // a. Disable AGC
424 hw_get_dxx_reg(phw_data, REG_AGC_CTRL3, &reg_agc_ctrl3);
425 reg_agc_ctrl3 &= ~BIT(2);
426 reg_agc_ctrl3 |= (MASK_LNA_FIX_GAIN|MASK_AGC_FIX);
427 hw_set_dxx_reg(phw_data, REG_AGC_CTRL3, reg_agc_ctrl3);
428
429 hw_get_dxx_reg(phw_data, REG_AGC_CTRL5, &val);
430 val |= MASK_AGC_FIX_GAIN;
431 hw_set_dxx_reg(phw_data, REG_AGC_CTRL5, val);
432
433 // b. Turn off BB RX
434 hw_get_dxx_reg(phw_data, REG_A_ACQ_CTRL, &reg_a_acq_ctrl);
435 reg_a_acq_ctrl |= MASK_AMER_OFF_REG;
436 hw_set_dxx_reg(phw_data, REG_A_ACQ_CTRL, reg_a_acq_ctrl);
437
438 hw_get_dxx_reg(phw_data, REG_B_ACQ_CTRL, &reg_b_acq_ctrl);
439 reg_b_acq_ctrl |= MASK_BMER_OFF_REG;
440 hw_set_dxx_reg(phw_data, REG_B_ACQ_CTRL, reg_b_acq_ctrl);
441
442 // c. Make sure MAC is in receiving mode
443 // d. Turn ON ADC calibration
444 // - ADC calibrator is triggered by this signal rising from 0 to 1
445 hw_get_dxx_reg(phw_data, REG_MODE_CTRL, &val);
446 val &= ~MASK_ADC_DC_CAL_STR;
447 hw_set_dxx_reg(phw_data, REG_MODE_CTRL, val);
448
449 val |= MASK_ADC_DC_CAL_STR;
450 hw_set_dxx_reg(phw_data, REG_MODE_CTRL, val);
451
452 // e. The result are shown in "adc_dc_cal_i[8:0] and adc_dc_cal_q[8:0]"
453 #ifdef _DEBUG
454 hw_get_dxx_reg(phw_data, REG_OFFSET_READ, &val);
455 PHY_DEBUG(("[CAL] REG_OFFSET_READ = 0x%08X\n", val));
456
457 PHY_DEBUG(("[CAL] ** adc_dc_cal_i = %d (0x%04X)\n",
458 _s9_to_s32(val&0x000001FF), val&0x000001FF));
459 PHY_DEBUG(("[CAL] ** adc_dc_cal_q = %d (0x%04X)\n",
460 _s9_to_s32((val&0x0003FE00)>>9), (val&0x0003FE00)>>9));
461 #endif
462
463 hw_get_dxx_reg(phw_data, REG_MODE_CTRL, &val);
464 val &= ~MASK_ADC_DC_CAL_STR;
465 hw_set_dxx_reg(phw_data, REG_MODE_CTRL, val);
466
467 // f. Turn on BB RX
468 //hw_get_dxx_reg(phw_data, REG_A_ACQ_CTRL, &reg_a_acq_ctrl);
469 reg_a_acq_ctrl &= ~MASK_AMER_OFF_REG;
470 hw_set_dxx_reg(phw_data, REG_A_ACQ_CTRL, reg_a_acq_ctrl);
471
472 //hw_get_dxx_reg(phw_data, REG_B_ACQ_CTRL, &reg_b_acq_ctrl);
473 reg_b_acq_ctrl &= ~MASK_BMER_OFF_REG;
474 hw_set_dxx_reg(phw_data, REG_B_ACQ_CTRL, reg_b_acq_ctrl);
475
476 // g. Enable AGC
477 //hw_get_dxx_reg(phw_data, REG_AGC_CTRL3, &val);
478 reg_agc_ctrl3 |= BIT(2);
479 reg_agc_ctrl3 &= ~(MASK_LNA_FIX_GAIN|MASK_AGC_FIX);
480 hw_set_dxx_reg(phw_data, REG_AGC_CTRL3, reg_agc_ctrl3);
481 }
482
483 ////////////////////////////////////////////////////////
484 void _txidac_dc_offset_cancellation_winbond(struct hw_data *phw_data)
485 {
486 u32 reg_agc_ctrl3;
487 u32 reg_mode_ctrl;
488 u32 reg_dc_cancel;
489 s32 iqcal_image_i;
490 s32 iqcal_image_q;
491 u32 sqsum;
492 s32 mag_0;
493 s32 mag_1;
494 s32 fix_cancel_dc_i = 0;
495 u32 val;
496 int loop;
497
498 PHY_DEBUG(("[CAL] -> [2]_txidac_dc_offset_cancellation()\n"));
499
500 // a. Set to "TX calibration mode"
501
502 //0x01 0xEE3FC2 ; 3B8FF ; Calibration (6a). enable TX IQ calibration loop circuits
503 phy_set_rf_data(phw_data, 1, (1<<24)|0xEE3FC2);
504 //0x0B 0x1905D6 ; 06417 ; Calibration (6b). enable TX I/Q cal loop squaring circuit
505 phy_set_rf_data(phw_data, 11, (11<<24)|0x1901D6);
506 //0x05 0x24C60A ; 09318 ; Calibration (6c). setting TX-VGA gain: TXGCH=2 & GPK=110 --> to be optimized
507 phy_set_rf_data(phw_data, 5, (5<<24)|0x24C48A);
508 //0x06 0x06880C ; 01A20 ; Calibration (6d). RXGCH=00; RXGCL=100 000 (RXVGA=32) --> to be optimized
509 phy_set_rf_data(phw_data, 6, (6<<24)|0x06890C);
510 //0x00 0xFDF1C0 ; 3F7C7 ; Calibration (6e). turn on IQ imbalance/Test mode
511 phy_set_rf_data(phw_data, 0, (0<<24)|0xFDF1C0);
512
513 hw_set_dxx_reg(phw_data, 0x58, 0x30303030); // IQ_Alpha Changed
514
515 // a. Disable AGC
516 hw_get_dxx_reg(phw_data, REG_AGC_CTRL3, &reg_agc_ctrl3);
517 reg_agc_ctrl3 &= ~BIT(2);
518 reg_agc_ctrl3 |= (MASK_LNA_FIX_GAIN|MASK_AGC_FIX);
519 hw_set_dxx_reg(phw_data, REG_AGC_CTRL3, reg_agc_ctrl3);
520
521 hw_get_dxx_reg(phw_data, REG_AGC_CTRL5, &val);
522 val |= MASK_AGC_FIX_GAIN;
523 hw_set_dxx_reg(phw_data, REG_AGC_CTRL5, val);
524
525 // b. set iqcal_mode[1:0] to 0x2 and set iqcal_tone[3:2] to 0
526 hw_get_dxx_reg(phw_data, REG_MODE_CTRL, &reg_mode_ctrl);
527
528 PHY_DEBUG(("[CAL] MODE_CTRL (read) = 0x%08X\n", reg_mode_ctrl));
529 reg_mode_ctrl &= ~(MASK_IQCAL_TONE_SEL|MASK_IQCAL_MODE);
530
531 // mode=2, tone=0
532 //reg_mode_ctrl |= (MASK_CALIB_START|2);
533
534 // mode=2, tone=1
535 //reg_mode_ctrl |= (MASK_CALIB_START|2|(1<<2));
536
537 // mode=2, tone=2
538 reg_mode_ctrl |= (MASK_CALIB_START|2|(2<<2));
539 hw_set_dxx_reg(phw_data, REG_MODE_CTRL, reg_mode_ctrl);
540 PHY_DEBUG(("[CAL] MODE_CTRL (write) = 0x%08X\n", reg_mode_ctrl));
541
542 hw_get_dxx_reg(phw_data, 0x5C, &reg_dc_cancel);
543 PHY_DEBUG(("[CAL] DC_CANCEL (read) = 0x%08X\n", reg_dc_cancel));
544
545 for (loop = 0; loop < LOOP_TIMES; loop++)
546 {
547 PHY_DEBUG(("[CAL] [%d.] ==================================\n", loop));
548
549 // c.
550 // reset cancel_dc_i[9:5] and cancel_dc_q[4:0] in register DC_Cancel
551 reg_dc_cancel &= ~(0x03FF);
552 PHY_DEBUG(("[CAL] DC_CANCEL (write) = 0x%08X\n", reg_dc_cancel));
553 hw_set_dxx_reg(phw_data, 0x5C, reg_dc_cancel);
554
555 hw_get_dxx_reg(phw_data, REG_CALIB_READ2, &val);
556 PHY_DEBUG(("[CAL] CALIB_READ2 = 0x%08X\n", val));
557
558 iqcal_image_i = _s13_to_s32(val & 0x00001FFF);
559 iqcal_image_q = _s13_to_s32((val & 0x03FFE000) >> 13);
560 sqsum = iqcal_image_i*iqcal_image_i + iqcal_image_q*iqcal_image_q;
561 mag_0 = (s32) _sqrt(sqsum);
562 PHY_DEBUG(("[CAL] mag_0=%d (iqcal_image_i=%d, iqcal_image_q=%d)\n",
563 mag_0, iqcal_image_i, iqcal_image_q));
564
565 // d.
566 reg_dc_cancel |= (1 << CANCEL_DC_I_SHIFT);
567 PHY_DEBUG(("[CAL] DC_CANCEL (write) = 0x%08X\n", reg_dc_cancel));
568 hw_set_dxx_reg(phw_data, 0x5C, reg_dc_cancel);
569
570 hw_get_dxx_reg(phw_data, REG_CALIB_READ2, &val);
571 PHY_DEBUG(("[CAL] CALIB_READ2 = 0x%08X\n", val));
572
573 iqcal_image_i = _s13_to_s32(val & 0x00001FFF);
574 iqcal_image_q = _s13_to_s32((val & 0x03FFE000) >> 13);
575 sqsum = iqcal_image_i*iqcal_image_i + iqcal_image_q*iqcal_image_q;
576 mag_1 = (s32) _sqrt(sqsum);
577 PHY_DEBUG(("[CAL] mag_1=%d (iqcal_image_i=%d, iqcal_image_q=%d)\n",
578 mag_1, iqcal_image_i, iqcal_image_q));
579
580 // e. Calculate the correct DC offset cancellation value for I
581 if (mag_0 != mag_1)
582 {
583 fix_cancel_dc_i = (mag_0*10000) / (mag_0*10000 - mag_1*10000);
584 }
585 else
586 {
587 if (mag_0 == mag_1)
588 {
589 PHY_DEBUG(("[CAL] ***** mag_0 = mag_1 !!\n"));
590 }
591
592 fix_cancel_dc_i = 0;
593 }
594
595 PHY_DEBUG(("[CAL] ** fix_cancel_dc_i = %d (0x%04X)\n",
596 fix_cancel_dc_i, _s32_to_s5(fix_cancel_dc_i)));
597
598 if ((abs(mag_1-mag_0)*6) > mag_0)
599 {
600 break;
601 }
602 }
603
604 if ( loop >= 19 )
605 fix_cancel_dc_i = 0;
606
607 reg_dc_cancel &= ~(0x03FF);
608 reg_dc_cancel |= (_s32_to_s5(fix_cancel_dc_i) << CANCEL_DC_I_SHIFT);
609 hw_set_dxx_reg(phw_data, 0x5C, reg_dc_cancel);
610 PHY_DEBUG(("[CAL] DC_CANCEL (write) = 0x%08X\n", reg_dc_cancel));
611
612 // g.
613 reg_mode_ctrl &= ~MASK_CALIB_START;
614 hw_set_dxx_reg(phw_data, REG_MODE_CTRL, reg_mode_ctrl);
615 PHY_DEBUG(("[CAL] MODE_CTRL (write) = 0x%08X\n", reg_mode_ctrl));
616 }
617
618 ///////////////////////////////////////////////////////
619 void _txqdac_dc_offset_cacellation_winbond(struct hw_data *phw_data)
620 {
621 u32 reg_agc_ctrl3;
622 u32 reg_mode_ctrl;
623 u32 reg_dc_cancel;
624 s32 iqcal_image_i;
625 s32 iqcal_image_q;
626 u32 sqsum;
627 s32 mag_0;
628 s32 mag_1;
629 s32 fix_cancel_dc_q = 0;
630 u32 val;
631 int loop;
632
633 PHY_DEBUG(("[CAL] -> [3]_txqdac_dc_offset_cacellation()\n"));
634 //0x01 0xEE3FC2 ; 3B8FF ; Calibration (6a). enable TX IQ calibration loop circuits
635 phy_set_rf_data(phw_data, 1, (1<<24)|0xEE3FC2);
636 //0x0B 0x1905D6 ; 06417 ; Calibration (6b). enable TX I/Q cal loop squaring circuit
637 phy_set_rf_data(phw_data, 11, (11<<24)|0x1901D6);
638 //0x05 0x24C60A ; 09318 ; Calibration (6c). setting TX-VGA gain: TXGCH=2 & GPK=110 --> to be optimized
639 phy_set_rf_data(phw_data, 5, (5<<24)|0x24C48A);
640 //0x06 0x06880C ; 01A20 ; Calibration (6d). RXGCH=00; RXGCL=100 000 (RXVGA=32) --> to be optimized
641 phy_set_rf_data(phw_data, 6, (6<<24)|0x06890C);
642 //0x00 0xFDF1C0 ; 3F7C7 ; Calibration (6e). turn on IQ imbalance/Test mode
643 phy_set_rf_data(phw_data, 0, (0<<24)|0xFDF1C0);
644
645 hw_set_dxx_reg(phw_data, 0x58, 0x30303030); // IQ_Alpha Changed
646
647 // a. Disable AGC
648 hw_get_dxx_reg(phw_data, REG_AGC_CTRL3, &reg_agc_ctrl3);
649 reg_agc_ctrl3 &= ~BIT(2);
650 reg_agc_ctrl3 |= (MASK_LNA_FIX_GAIN|MASK_AGC_FIX);
651 hw_set_dxx_reg(phw_data, REG_AGC_CTRL3, reg_agc_ctrl3);
652
653 hw_get_dxx_reg(phw_data, REG_AGC_CTRL5, &val);
654 val |= MASK_AGC_FIX_GAIN;
655 hw_set_dxx_reg(phw_data, REG_AGC_CTRL5, val);
656
657 // a. set iqcal_mode[1:0] to 0x3 and set iqcal_tone[3:2] to 0
658 hw_get_dxx_reg(phw_data, REG_MODE_CTRL, &reg_mode_ctrl);
659 PHY_DEBUG(("[CAL] MODE_CTRL (read) = 0x%08X\n", reg_mode_ctrl));
660
661 //reg_mode_ctrl &= ~(MASK_IQCAL_TONE_SEL|MASK_IQCAL_MODE);
662 reg_mode_ctrl &= ~(MASK_IQCAL_MODE);
663 reg_mode_ctrl |= (MASK_CALIB_START|3);
664 hw_set_dxx_reg(phw_data, REG_MODE_CTRL, reg_mode_ctrl);
665 PHY_DEBUG(("[CAL] MODE_CTRL (write) = 0x%08X\n", reg_mode_ctrl));
666
667 hw_get_dxx_reg(phw_data, 0x5C, &reg_dc_cancel);
668 PHY_DEBUG(("[CAL] DC_CANCEL (read) = 0x%08X\n", reg_dc_cancel));
669
670 for (loop = 0; loop < LOOP_TIMES; loop++)
671 {
672 PHY_DEBUG(("[CAL] [%d.] ==================================\n", loop));
673
674 // b.
675 // reset cancel_dc_q[4:0] in register DC_Cancel
676 reg_dc_cancel &= ~(0x001F);
677 PHY_DEBUG(("[CAL] DC_CANCEL (write) = 0x%08X\n", reg_dc_cancel));
678 hw_set_dxx_reg(phw_data, 0x5C, reg_dc_cancel);
679
680 hw_get_dxx_reg(phw_data, REG_CALIB_READ2, &val);
681 PHY_DEBUG(("[CAL] CALIB_READ2 = 0x%08X\n", val));
682
683 iqcal_image_i = _s13_to_s32(val & 0x00001FFF);
684 iqcal_image_q = _s13_to_s32((val & 0x03FFE000) >> 13);
685 sqsum = iqcal_image_i*iqcal_image_i + iqcal_image_q*iqcal_image_q;
686 mag_0 = _sqrt(sqsum);
687 PHY_DEBUG(("[CAL] mag_0=%d (iqcal_image_i=%d, iqcal_image_q=%d)\n",
688 mag_0, iqcal_image_i, iqcal_image_q));
689
690 // c.
691 reg_dc_cancel |= (1 << CANCEL_DC_Q_SHIFT);
692 PHY_DEBUG(("[CAL] DC_CANCEL (write) = 0x%08X\n", reg_dc_cancel));
693 hw_set_dxx_reg(phw_data, 0x5C, reg_dc_cancel);
694
695 hw_get_dxx_reg(phw_data, REG_CALIB_READ2, &val);
696 PHY_DEBUG(("[CAL] CALIB_READ2 = 0x%08X\n", val));
697
698 iqcal_image_i = _s13_to_s32(val & 0x00001FFF);
699 iqcal_image_q = _s13_to_s32((val & 0x03FFE000) >> 13);
700 sqsum = iqcal_image_i*iqcal_image_i + iqcal_image_q*iqcal_image_q;
701 mag_1 = _sqrt(sqsum);
702 PHY_DEBUG(("[CAL] mag_1=%d (iqcal_image_i=%d, iqcal_image_q=%d)\n",
703 mag_1, iqcal_image_i, iqcal_image_q));
704
705 // d. Calculate the correct DC offset cancellation value for I
706 if (mag_0 != mag_1)
707 {
708 fix_cancel_dc_q = (mag_0*10000) / (mag_0*10000 - mag_1*10000);
709 }
710 else
711 {
712 if (mag_0 == mag_1)
713 {
714 PHY_DEBUG(("[CAL] ***** mag_0 = mag_1 !!\n"));
715 }
716
717 fix_cancel_dc_q = 0;
718 }
719
720 PHY_DEBUG(("[CAL] ** fix_cancel_dc_q = %d (0x%04X)\n",
721 fix_cancel_dc_q, _s32_to_s5(fix_cancel_dc_q)));
722
723 if ((abs(mag_1-mag_0)*6) > mag_0)
724 {
725 break;
726 }
727 }
728
729 if ( loop >= 19 )
730 fix_cancel_dc_q = 0;
731
732 reg_dc_cancel &= ~(0x001F);
733 reg_dc_cancel |= (_s32_to_s5(fix_cancel_dc_q) << CANCEL_DC_Q_SHIFT);
734 hw_set_dxx_reg(phw_data, 0x5C, reg_dc_cancel);
735 PHY_DEBUG(("[CAL] DC_CANCEL (write) = 0x%08X\n", reg_dc_cancel));
736
737
738 // f.
739 reg_mode_ctrl &= ~MASK_CALIB_START;
740 hw_set_dxx_reg(phw_data, REG_MODE_CTRL, reg_mode_ctrl);
741 PHY_DEBUG(("[CAL] MODE_CTRL (write) = 0x%08X\n", reg_mode_ctrl));
742 }
743
744 //20060612.1.a 20060718.1 Modify
745 u8 _tx_iq_calibration_loop_winbond(struct hw_data *phw_data,
746 s32 a_2_threshold,
747 s32 b_2_threshold)
748 {
749 u32 reg_mode_ctrl;
750 s32 iq_mag_0_tx;
751 s32 iqcal_tone_i0;
752 s32 iqcal_tone_q0;
753 s32 iqcal_tone_i;
754 s32 iqcal_tone_q;
755 u32 sqsum;
756 s32 rot_i_b;
757 s32 rot_q_b;
758 s32 tx_cal_flt_b[4];
759 s32 tx_cal[4];
760 s32 tx_cal_reg[4];
761 s32 a_2, b_2;
762 s32 sin_b, sin_2b;
763 s32 cos_b, cos_2b;
764 s32 divisor;
765 s32 temp1, temp2;
766 u32 val;
767 u16 loop;
768 s32 iqcal_tone_i_avg,iqcal_tone_q_avg;
769 u8 verify_count;
770 int capture_time;
771
772 PHY_DEBUG(("[CAL] -> _tx_iq_calibration_loop()\n"));
773 PHY_DEBUG(("[CAL] ** a_2_threshold = %d\n", a_2_threshold));
774 PHY_DEBUG(("[CAL] ** b_2_threshold = %d\n", b_2_threshold));
775
776 verify_count = 0;
777
778 hw_get_dxx_reg(phw_data, REG_MODE_CTRL, &reg_mode_ctrl);
779 PHY_DEBUG(("[CAL] MODE_CTRL (read) = 0x%08X\n", reg_mode_ctrl));
780
781 loop = LOOP_TIMES;
782
783 while (loop > 0)
784 {
785 PHY_DEBUG(("[CAL] [%d.] <_tx_iq_calibration_loop>\n", (LOOP_TIMES-loop+1)));
786
787 iqcal_tone_i_avg=0;
788 iqcal_tone_q_avg=0;
789 if( !hw_set_dxx_reg(phw_data, 0x3C, 0x00) ) // 20060718.1 modify
790 return 0;
791 for(capture_time=0;capture_time<10;capture_time++)
792 {
793 // a. Set iqcal_mode[1:0] to 0x2 and set "calib_start" to 0x1 to
794 // enable "IQ alibration Mode II"
795 reg_mode_ctrl &= ~(MASK_IQCAL_TONE_SEL|MASK_IQCAL_MODE);
796 reg_mode_ctrl &= ~MASK_IQCAL_MODE;
797 reg_mode_ctrl |= (MASK_CALIB_START|0x02);
798 reg_mode_ctrl |= (MASK_CALIB_START|0x02|2<<2);
799 hw_set_dxx_reg(phw_data, REG_MODE_CTRL, reg_mode_ctrl);
800 PHY_DEBUG(("[CAL] MODE_CTRL (write) = 0x%08X\n", reg_mode_ctrl));
801
802 // b.
803 hw_get_dxx_reg(phw_data, REG_CALIB_READ1, &val);
804 PHY_DEBUG(("[CAL] CALIB_READ1 = 0x%08X\n", val));
805
806 iqcal_tone_i0 = _s13_to_s32(val & 0x00001FFF);
807 iqcal_tone_q0 = _s13_to_s32((val & 0x03FFE000) >> 13);
808 PHY_DEBUG(("[CAL] ** iqcal_tone_i0=%d, iqcal_tone_q0=%d\n",
809 iqcal_tone_i0, iqcal_tone_q0));
810
811 sqsum = iqcal_tone_i0*iqcal_tone_i0 +
812 iqcal_tone_q0*iqcal_tone_q0;
813 iq_mag_0_tx = (s32) _sqrt(sqsum);
814 PHY_DEBUG(("[CAL] ** iq_mag_0_tx=%d\n", iq_mag_0_tx));
815
816 // c. Set "calib_start" to 0x0
817 reg_mode_ctrl &= ~MASK_CALIB_START;
818 hw_set_dxx_reg(phw_data, REG_MODE_CTRL, reg_mode_ctrl);
819 PHY_DEBUG(("[CAL] MODE_CTRL (write) = 0x%08X\n", reg_mode_ctrl));
820
821 // d. Set iqcal_mode[1:0] to 0x3 and set "calib_start" to 0x1 to
822 // enable "IQ alibration Mode II"
823 //hw_get_dxx_reg(phw_data, REG_MODE_CTRL, &val);
824 hw_get_dxx_reg(phw_data, REG_MODE_CTRL, &reg_mode_ctrl);
825 reg_mode_ctrl &= ~MASK_IQCAL_MODE;
826 reg_mode_ctrl |= (MASK_CALIB_START|0x03);
827 hw_set_dxx_reg(phw_data, REG_MODE_CTRL, reg_mode_ctrl);
828 PHY_DEBUG(("[CAL] MODE_CTRL (write) = 0x%08X\n", reg_mode_ctrl));
829
830 // e.
831 hw_get_dxx_reg(phw_data, REG_CALIB_READ1, &val);
832 PHY_DEBUG(("[CAL] CALIB_READ1 = 0x%08X\n", val));
833
834 iqcal_tone_i = _s13_to_s32(val & 0x00001FFF);
835 iqcal_tone_q = _s13_to_s32((val & 0x03FFE000) >> 13);
836 PHY_DEBUG(("[CAL] ** iqcal_tone_i = %d, iqcal_tone_q = %d\n",
837 iqcal_tone_i, iqcal_tone_q));
838 if( capture_time == 0)
839 {
840 continue;
841 }
842 else
843 {
844 iqcal_tone_i_avg=( iqcal_tone_i_avg*(capture_time-1) +iqcal_tone_i)/capture_time;
845 iqcal_tone_q_avg=( iqcal_tone_q_avg*(capture_time-1) +iqcal_tone_q)/capture_time;
846 }
847 }
848
849 iqcal_tone_i = iqcal_tone_i_avg;
850 iqcal_tone_q = iqcal_tone_q_avg;
851
852
853 rot_i_b = (iqcal_tone_i * iqcal_tone_i0 +
854 iqcal_tone_q * iqcal_tone_q0) / 1024;
855 rot_q_b = (iqcal_tone_i * iqcal_tone_q0 * (-1) +
856 iqcal_tone_q * iqcal_tone_i0) / 1024;
857 PHY_DEBUG(("[CAL] ** rot_i_b = %d, rot_q_b = %d\n",
858 rot_i_b, rot_q_b));
859
860 // f.
861 divisor = ((iq_mag_0_tx * iq_mag_0_tx * 2)/1024 - rot_i_b) * 2;
862
863 if (divisor == 0)
864 {
865 PHY_DEBUG(("[CAL] ** <_tx_iq_calibration_loop> ERROR *******\n"));
866 PHY_DEBUG(("[CAL] ** divisor=0 to calculate EPS and THETA !!\n"));
867 PHY_DEBUG(("[CAL] ******************************************\n"));
868 break;
869 }
870
871 a_2 = (rot_i_b * 32768) / divisor;
872 b_2 = (rot_q_b * (-32768)) / divisor;
873 PHY_DEBUG(("[CAL] ***** EPSILON/2 = %d\n", a_2));
874 PHY_DEBUG(("[CAL] ***** THETA/2 = %d\n", b_2));
875
876 phw_data->iq_rsdl_gain_tx_d2 = a_2;
877 phw_data->iq_rsdl_phase_tx_d2 = b_2;
878
879 //if ((abs(a_2) < 150) && (abs(b_2) < 100))
880 //if ((abs(a_2) < 200) && (abs(b_2) < 200))
881 if ((abs(a_2) < a_2_threshold) && (abs(b_2) < b_2_threshold))
882 {
883 verify_count++;
884
885 PHY_DEBUG(("[CAL] ** <_tx_iq_calibration_loop> *************\n"));
886 PHY_DEBUG(("[CAL] ** VERIFY OK # %d !!\n", verify_count));
887 PHY_DEBUG(("[CAL] ******************************************\n"));
888
889 if (verify_count > 2)
890 {
891 PHY_DEBUG(("[CAL] ** <_tx_iq_calibration_loop> *********\n"));
892 PHY_DEBUG(("[CAL] ** TX_IQ_CALIBRATION (EPS,THETA) OK !!\n"));
893 PHY_DEBUG(("[CAL] **************************************\n"));
894 return 0;
895 }
896
897 continue;
898 }
899 else
900 {
901 verify_count = 0;
902 }
903
904 _sin_cos(b_2, &sin_b, &cos_b);
905 _sin_cos(b_2*2, &sin_2b, &cos_2b);
906 PHY_DEBUG(("[CAL] ** sin(b/2)=%d, cos(b/2)=%d\n", sin_b, cos_b));
907 PHY_DEBUG(("[CAL] ** sin(b)=%d, cos(b)=%d\n", sin_2b, cos_2b));
908
909 if (cos_2b == 0)
910 {
911 PHY_DEBUG(("[CAL] ** <_tx_iq_calibration_loop> ERROR *******\n"));
912 PHY_DEBUG(("[CAL] ** cos(b)=0 !!\n"));
913 PHY_DEBUG(("[CAL] ******************************************\n"));
914 break;
915 }
916
917 // 1280 * 32768 = 41943040
918 temp1 = (41943040/cos_2b)*cos_b;
919
920 //temp2 = (41943040/cos_2b)*sin_b*(-1);
921 if (phw_data->revision == 0x2002) // 1st-cut
922 {
923 temp2 = (41943040/cos_2b)*sin_b*(-1);
924 }
925 else // 2nd-cut
926 {
927 temp2 = (41943040*4/cos_2b)*sin_b*(-1);
928 }
929
930 tx_cal_flt_b[0] = _floor(temp1/(32768+a_2));
931 tx_cal_flt_b[1] = _floor(temp2/(32768+a_2));
932 tx_cal_flt_b[2] = _floor(temp2/(32768-a_2));
933 tx_cal_flt_b[3] = _floor(temp1/(32768-a_2));
934 PHY_DEBUG(("[CAL] ** tx_cal_flt_b[0] = %d\n", tx_cal_flt_b[0]));
935 PHY_DEBUG(("[CAL] tx_cal_flt_b[1] = %d\n", tx_cal_flt_b[1]));
936 PHY_DEBUG(("[CAL] tx_cal_flt_b[2] = %d\n", tx_cal_flt_b[2]));
937 PHY_DEBUG(("[CAL] tx_cal_flt_b[3] = %d\n", tx_cal_flt_b[3]));
938
939 tx_cal[2] = tx_cal_flt_b[2];
940 tx_cal[2] = tx_cal[2] +3;
941 tx_cal[1] = tx_cal[2];
942 tx_cal[3] = tx_cal_flt_b[3] - 128;
943 tx_cal[0] = -tx_cal[3]+1;
944
945 PHY_DEBUG(("[CAL] tx_cal[0] = %d\n", tx_cal[0]));
946 PHY_DEBUG(("[CAL] tx_cal[1] = %d\n", tx_cal[1]));
947 PHY_DEBUG(("[CAL] tx_cal[2] = %d\n", tx_cal[2]));
948 PHY_DEBUG(("[CAL] tx_cal[3] = %d\n", tx_cal[3]));
949
950 //if ((tx_cal[0] == 0) && (tx_cal[1] == 0) &&
951 // (tx_cal[2] == 0) && (tx_cal[3] == 0))
952 //{
953 // PHY_DEBUG(("[CAL] ** <_tx_iq_calibration_loop> *************\n"));
954 // PHY_DEBUG(("[CAL] ** TX_IQ_CALIBRATION COMPLETE !!\n"));
955 // PHY_DEBUG(("[CAL] ******************************************\n"));
956 // return 0;
957 //}
958
959 // g.
960 if (phw_data->revision == 0x2002) // 1st-cut
961 {
962 hw_get_dxx_reg(phw_data, 0x54, &val);
963 PHY_DEBUG(("[CAL] ** 0x54 = 0x%08X\n", val));
964 tx_cal_reg[0] = _s4_to_s32((val & 0xF0000000) >> 28);
965 tx_cal_reg[1] = _s4_to_s32((val & 0x0F000000) >> 24);
966 tx_cal_reg[2] = _s4_to_s32((val & 0x00F00000) >> 20);
967 tx_cal_reg[3] = _s4_to_s32((val & 0x000F0000) >> 16);
968 }
969 else // 2nd-cut
970 {
971 hw_get_dxx_reg(phw_data, 0x3C, &val);
972 PHY_DEBUG(("[CAL] ** 0x3C = 0x%08X\n", val));
973 tx_cal_reg[0] = _s5_to_s32((val & 0xF8000000) >> 27);
974 tx_cal_reg[1] = _s6_to_s32((val & 0x07E00000) >> 21);
975 tx_cal_reg[2] = _s6_to_s32((val & 0x001F8000) >> 15);
976 tx_cal_reg[3] = _s5_to_s32((val & 0x00007C00) >> 10);
977
978 }
979
980 PHY_DEBUG(("[CAL] ** tx_cal_reg[0] = %d\n", tx_cal_reg[0]));
981 PHY_DEBUG(("[CAL] tx_cal_reg[1] = %d\n", tx_cal_reg[1]));
982 PHY_DEBUG(("[CAL] tx_cal_reg[2] = %d\n", tx_cal_reg[2]));
983 PHY_DEBUG(("[CAL] tx_cal_reg[3] = %d\n", tx_cal_reg[3]));
984
985 if (phw_data->revision == 0x2002) // 1st-cut
986 {
987 if (((tx_cal_reg[0]==7) || (tx_cal_reg[0]==(-8))) &&
988 ((tx_cal_reg[3]==7) || (tx_cal_reg[3]==(-8))))
989 {
990 PHY_DEBUG(("[CAL] ** <_tx_iq_calibration_loop> *********\n"));
991 PHY_DEBUG(("[CAL] ** TX_IQ_CALIBRATION SATUATION !!\n"));
992 PHY_DEBUG(("[CAL] **************************************\n"));
993 break;
994 }
995 }
996 else // 2nd-cut
997 {
998 if (((tx_cal_reg[0]==31) || (tx_cal_reg[0]==(-32))) &&
999 ((tx_cal_reg[3]==31) || (tx_cal_reg[3]==(-32))))
1000 {
1001 PHY_DEBUG(("[CAL] ** <_tx_iq_calibration_loop> *********\n"));
1002 PHY_DEBUG(("[CAL] ** TX_IQ_CALIBRATION SATUATION !!\n"));
1003 PHY_DEBUG(("[CAL] **************************************\n"));
1004 break;
1005 }
1006 }
1007
1008 tx_cal[0] = tx_cal[0] + tx_cal_reg[0];
1009 tx_cal[1] = tx_cal[1] + tx_cal_reg[1];
1010 tx_cal[2] = tx_cal[2] + tx_cal_reg[2];
1011 tx_cal[3] = tx_cal[3] + tx_cal_reg[3];
1012 PHY_DEBUG(("[CAL] ** apply tx_cal[0] = %d\n", tx_cal[0]));
1013 PHY_DEBUG(("[CAL] apply tx_cal[1] = %d\n", tx_cal[1]));
1014 PHY_DEBUG(("[CAL] apply tx_cal[2] = %d\n", tx_cal[2]));
1015 PHY_DEBUG(("[CAL] apply tx_cal[3] = %d\n", tx_cal[3]));
1016
1017 if (phw_data->revision == 0x2002) // 1st-cut
1018 {
1019 val &= 0x0000FFFF;
1020 val |= ((_s32_to_s4(tx_cal[0]) << 28)|
1021 (_s32_to_s4(tx_cal[1]) << 24)|
1022 (_s32_to_s4(tx_cal[2]) << 20)|
1023 (_s32_to_s4(tx_cal[3]) << 16));
1024 hw_set_dxx_reg(phw_data, 0x54, val);
1025 PHY_DEBUG(("[CAL] ** CALIB_DATA = 0x%08X\n", val));
1026 return 0;
1027 }
1028 else // 2nd-cut
1029 {
1030 val &= 0x000003FF;
1031 val |= ((_s32_to_s5(tx_cal[0]) << 27)|
1032 (_s32_to_s6(tx_cal[1]) << 21)|
1033 (_s32_to_s6(tx_cal[2]) << 15)|
1034 (_s32_to_s5(tx_cal[3]) << 10));
1035 hw_set_dxx_reg(phw_data, 0x3C, val);
1036 PHY_DEBUG(("[CAL] ** TX_IQ_CALIBRATION = 0x%08X\n", val));
1037 return 0;
1038 }
1039
1040 // i. Set "calib_start" to 0x0
1041 reg_mode_ctrl &= ~MASK_CALIB_START;
1042 hw_set_dxx_reg(phw_data, REG_MODE_CTRL, reg_mode_ctrl);
1043 PHY_DEBUG(("[CAL] MODE_CTRL (write) = 0x%08X\n", reg_mode_ctrl));
1044
1045 loop--;
1046 }
1047
1048 return 1;
1049 }
1050
1051 void _tx_iq_calibration_winbond(struct hw_data *phw_data)
1052 {
1053 u32 reg_agc_ctrl3;
1054 #ifdef _DEBUG
1055 s32 tx_cal_reg[4];
1056
1057 #endif
1058 u32 reg_mode_ctrl;
1059 u32 val;
1060 u8 result;
1061
1062 PHY_DEBUG(("[CAL] -> [4]_tx_iq_calibration()\n"));
1063
1064 //0x01 0xEE3FC2 ; 3B8FF ; Calibration (6a). enable TX IQ calibration loop circuits
1065 phy_set_rf_data(phw_data, 1, (1<<24)|0xEE3FC2);
1066 //0x0B 0x1905D6 ; 06417 ; Calibration (6b). enable TX I/Q cal loop squaring circuit
1067 phy_set_rf_data(phw_data, 11, (11<<24)|0x19BDD6); // 20060612.1.a 0x1905D6);
1068 //0x05 0x24C60A ; 09318 ; Calibration (6c). setting TX-VGA gain: TXGCH=2 & GPK=110 --> to be optimized
1069 phy_set_rf_data(phw_data, 5, (5<<24)|0x24C60A); //0x24C60A (high temperature)
1070 //0x06 0x06880C ; 01A20 ; Calibration (6d). RXGCH=00; RXGCL=100 000 (RXVGA=32) --> to be optimized
1071 phy_set_rf_data(phw_data, 6, (6<<24)|0x34880C); // 20060612.1.a 0x06890C);
1072 //0x00 0xFDF1C0 ; 3F7C7 ; Calibration (6e). turn on IQ imbalance/Test mode
1073 phy_set_rf_data(phw_data, 0, (0<<24)|0xFDF1C0);
1074 //; [BB-chip]: Calibration (6f).Send test pattern
1075 //; [BB-chip]: Calibration (6g). Search RXGCL optimal value
1076 //; [BB-chip]: Calibration (6h). Caculate TX-path IQ imbalance and setting TX path IQ compensation table
1077 //phy_set_rf_data(phw_data, 3, (3<<24)|0x025586);
1078
1079 msleep(30); // 20060612.1.a 30ms delay. Add the follow 2 lines
1080 //To adjust TXVGA to fit iq_mag_0 range from 1250 ~ 1750
1081 adjust_TXVGA_for_iq_mag( phw_data );
1082
1083 // a. Disable AGC
1084 hw_get_dxx_reg(phw_data, REG_AGC_CTRL3, &reg_agc_ctrl3);
1085 reg_agc_ctrl3 &= ~BIT(2);
1086 reg_agc_ctrl3 |= (MASK_LNA_FIX_GAIN|MASK_AGC_FIX);
1087 hw_set_dxx_reg(phw_data, REG_AGC_CTRL3, reg_agc_ctrl3);
1088
1089 hw_get_dxx_reg(phw_data, REG_AGC_CTRL5, &val);
1090 val |= MASK_AGC_FIX_GAIN;
1091 hw_set_dxx_reg(phw_data, REG_AGC_CTRL5, val);
1092
1093 result = _tx_iq_calibration_loop_winbond(phw_data, 150, 100);
1094
1095 if (result > 0)
1096 {
1097 if (phw_data->revision == 0x2002) // 1st-cut
1098 {
1099 hw_get_dxx_reg(phw_data, 0x54, &val);
1100 val &= 0x0000FFFF;
1101 hw_set_dxx_reg(phw_data, 0x54, val);
1102 }
1103 else // 2nd-cut
1104 {
1105 hw_get_dxx_reg(phw_data, 0x3C, &val);
1106 val &= 0x000003FF;
1107 hw_set_dxx_reg(phw_data, 0x3C, val);
1108 }
1109
1110 result = _tx_iq_calibration_loop_winbond(phw_data, 300, 200);
1111
1112 if (result > 0)
1113 {
1114 if (phw_data->revision == 0x2002) // 1st-cut
1115 {
1116 hw_get_dxx_reg(phw_data, 0x54, &val);
1117 val &= 0x0000FFFF;
1118 hw_set_dxx_reg(phw_data, 0x54, val);
1119 }
1120 else // 2nd-cut
1121 {
1122 hw_get_dxx_reg(phw_data, 0x3C, &val);
1123 val &= 0x000003FF;
1124 hw_set_dxx_reg(phw_data, 0x3C, val);
1125 }
1126
1127 result = _tx_iq_calibration_loop_winbond(phw_data, 500, 400);
1128 if (result > 0)
1129 {
1130 if (phw_data->revision == 0x2002) // 1st-cut
1131 {
1132 hw_get_dxx_reg(phw_data, 0x54, &val);
1133 val &= 0x0000FFFF;
1134 hw_set_dxx_reg(phw_data, 0x54, val);
1135 }
1136 else // 2nd-cut
1137 {
1138 hw_get_dxx_reg(phw_data, 0x3C, &val);
1139 val &= 0x000003FF;
1140 hw_set_dxx_reg(phw_data, 0x3C, val);
1141 }
1142
1143
1144 result = _tx_iq_calibration_loop_winbond(phw_data, 700, 500);
1145
1146 if (result > 0)
1147 {
1148 PHY_DEBUG(("[CAL] ** <_tx_iq_calibration> **************\n"));
1149 PHY_DEBUG(("[CAL] ** TX_IQ_CALIBRATION FAILURE !!\n"));
1150 PHY_DEBUG(("[CAL] **************************************\n"));
1151
1152 if (phw_data->revision == 0x2002) // 1st-cut
1153 {
1154 hw_get_dxx_reg(phw_data, 0x54, &val);
1155 val &= 0x0000FFFF;
1156 hw_set_dxx_reg(phw_data, 0x54, val);
1157 }
1158 else // 2nd-cut
1159 {
1160 hw_get_dxx_reg(phw_data, 0x3C, &val);
1161 val &= 0x000003FF;
1162 hw_set_dxx_reg(phw_data, 0x3C, val);
1163 }
1164 }
1165 }
1166 }
1167 }
1168
1169 // i. Set "calib_start" to 0x0
1170 hw_get_dxx_reg(phw_data, REG_MODE_CTRL, &reg_mode_ctrl);
1171 reg_mode_ctrl &= ~MASK_CALIB_START;
1172 hw_set_dxx_reg(phw_data, REG_MODE_CTRL, reg_mode_ctrl);
1173 PHY_DEBUG(("[CAL] MODE_CTRL (write) = 0x%08X\n", reg_mode_ctrl));
1174
1175 // g. Enable AGC
1176 //hw_get_dxx_reg(phw_data, REG_AGC_CTRL3, &val);
1177 reg_agc_ctrl3 |= BIT(2);
1178 reg_agc_ctrl3 &= ~(MASK_LNA_FIX_GAIN|MASK_AGC_FIX);
1179 hw_set_dxx_reg(phw_data, REG_AGC_CTRL3, reg_agc_ctrl3);
1180
1181 #ifdef _DEBUG
1182 if (phw_data->revision == 0x2002) // 1st-cut
1183 {
1184 hw_get_dxx_reg(phw_data, 0x54, &val);
1185 PHY_DEBUG(("[CAL] ** 0x54 = 0x%08X\n", val));
1186 tx_cal_reg[0] = _s4_to_s32((val & 0xF0000000) >> 28);
1187 tx_cal_reg[1] = _s4_to_s32((val & 0x0F000000) >> 24);
1188 tx_cal_reg[2] = _s4_to_s32((val & 0x00F00000) >> 20);
1189 tx_cal_reg[3] = _s4_to_s32((val & 0x000F0000) >> 16);
1190 }
1191 else // 2nd-cut
1192 {
1193 hw_get_dxx_reg(phw_data, 0x3C, &val);
1194 PHY_DEBUG(("[CAL] ** 0x3C = 0x%08X\n", val));
1195 tx_cal_reg[0] = _s5_to_s32((val & 0xF8000000) >> 27);
1196 tx_cal_reg[1] = _s6_to_s32((val & 0x07E00000) >> 21);
1197 tx_cal_reg[2] = _s6_to_s32((val & 0x001F8000) >> 15);
1198 tx_cal_reg[3] = _s5_to_s32((val & 0x00007C00) >> 10);
1199
1200 }
1201
1202 PHY_DEBUG(("[CAL] ** tx_cal_reg[0] = %d\n", tx_cal_reg[0]));
1203 PHY_DEBUG(("[CAL] tx_cal_reg[1] = %d\n", tx_cal_reg[1]));
1204 PHY_DEBUG(("[CAL] tx_cal_reg[2] = %d\n", tx_cal_reg[2]));
1205 PHY_DEBUG(("[CAL] tx_cal_reg[3] = %d\n", tx_cal_reg[3]));
1206 #endif
1207
1208
1209 // for test - BEN
1210 // RF Control Override
1211 }
1212
1213 /////////////////////////////////////////////////////////////////////////////////////////
1214 u8 _rx_iq_calibration_loop_winbond(struct hw_data *phw_data, u16 factor, u32 frequency)
1215 {
1216 u32 reg_mode_ctrl;
1217 s32 iqcal_tone_i;
1218 s32 iqcal_tone_q;
1219 s32 iqcal_image_i;
1220 s32 iqcal_image_q;
1221 s32 rot_tone_i_b;
1222 s32 rot_tone_q_b;
1223 s32 rot_image_i_b;
1224 s32 rot_image_q_b;
1225 s32 rx_cal_flt_b[4];
1226 s32 rx_cal[4];
1227 s32 rx_cal_reg[4];
1228 s32 a_2, b_2;
1229 s32 sin_b, sin_2b;
1230 s32 cos_b, cos_2b;
1231 s32 temp1, temp2;
1232 u32 val;
1233 u16 loop;
1234
1235 u32 pwr_tone;
1236 u32 pwr_image;
1237 u8 verify_count;
1238
1239 s32 iqcal_tone_i_avg,iqcal_tone_q_avg;
1240 s32 iqcal_image_i_avg,iqcal_image_q_avg;
1241 u16 capture_time;
1242
1243 PHY_DEBUG(("[CAL] -> [5]_rx_iq_calibration_loop()\n"));
1244 PHY_DEBUG(("[CAL] ** factor = %d\n", factor));
1245
1246
1247 // RF Control Override
1248 hw_get_cxx_reg(phw_data, 0x80, &val);
1249 val |= BIT(19);
1250 hw_set_cxx_reg(phw_data, 0x80, val);
1251
1252 // RF_Ctrl
1253 hw_get_cxx_reg(phw_data, 0xE4, &val);
1254 val |= BIT(0);
1255 hw_set_cxx_reg(phw_data, 0xE4, val);
1256 PHY_DEBUG(("[CAL] ** RF_CTRL(0xE4) = 0x%08X", val));
1257
1258 hw_set_dxx_reg(phw_data, 0x58, 0x44444444); // IQ_Alpha
1259
1260 // b.
1261
1262 hw_get_dxx_reg(phw_data, REG_MODE_CTRL, &reg_mode_ctrl);
1263 PHY_DEBUG(("[CAL] MODE_CTRL (read) = 0x%08X\n", reg_mode_ctrl));
1264
1265 verify_count = 0;
1266
1267 //for (loop = 0; loop < 1; loop++)
1268 //for (loop = 0; loop < LOOP_TIMES; loop++)
1269 loop = LOOP_TIMES;
1270 while (loop > 0)
1271 {
1272 PHY_DEBUG(("[CAL] [%d.] <_rx_iq_calibration_loop>\n", (LOOP_TIMES-loop+1)));
1273 iqcal_tone_i_avg=0;
1274 iqcal_tone_q_avg=0;
1275 iqcal_image_i_avg=0;
1276 iqcal_image_q_avg=0;
1277 capture_time=0;
1278
1279 for(capture_time=0; capture_time<10; capture_time++)
1280 {
1281 // i. Set "calib_start" to 0x0
1282 reg_mode_ctrl &= ~MASK_CALIB_START;
1283 if( !hw_set_dxx_reg(phw_data, REG_MODE_CTRL, reg_mode_ctrl) )//20060718.1 modify
1284 return 0;
1285 PHY_DEBUG(("[CAL] MODE_CTRL (write) = 0x%08X\n", reg_mode_ctrl));
1286
1287 reg_mode_ctrl &= ~MASK_IQCAL_MODE;
1288 reg_mode_ctrl |= (MASK_CALIB_START|0x1);
1289 hw_set_dxx_reg(phw_data, REG_MODE_CTRL, reg_mode_ctrl);
1290 PHY_DEBUG(("[CAL] MODE_CTRL (write) = 0x%08X\n", reg_mode_ctrl));
1291
1292 // c.
1293 hw_get_dxx_reg(phw_data, REG_CALIB_READ1, &val);
1294 PHY_DEBUG(("[CAL] CALIB_READ1 = 0x%08X\n", val));
1295
1296 iqcal_tone_i = _s13_to_s32(val & 0x00001FFF);
1297 iqcal_tone_q = _s13_to_s32((val & 0x03FFE000) >> 13);
1298 PHY_DEBUG(("[CAL] ** iqcal_tone_i = %d, iqcal_tone_q = %d\n",
1299 iqcal_tone_i, iqcal_tone_q));
1300
1301 hw_get_dxx_reg(phw_data, REG_CALIB_READ2, &val);
1302 PHY_DEBUG(("[CAL] CALIB_READ2 = 0x%08X\n", val));
1303
1304 iqcal_image_i = _s13_to_s32(val & 0x00001FFF);
1305 iqcal_image_q = _s13_to_s32((val & 0x03FFE000) >> 13);
1306 PHY_DEBUG(("[CAL] ** iqcal_image_i = %d, iqcal_image_q = %d\n",
1307 iqcal_image_i, iqcal_image_q));
1308 if( capture_time == 0)
1309 {
1310 continue;
1311 }
1312 else
1313 {
1314 iqcal_image_i_avg=( iqcal_image_i_avg*(capture_time-1) +iqcal_image_i)/capture_time;
1315 iqcal_image_q_avg=( iqcal_image_q_avg*(capture_time-1) +iqcal_image_q)/capture_time;
1316 iqcal_tone_i_avg=( iqcal_tone_i_avg*(capture_time-1) +iqcal_tone_i)/capture_time;
1317 iqcal_tone_q_avg=( iqcal_tone_q_avg*(capture_time-1) +iqcal_tone_q)/capture_time;
1318 }
1319 }
1320
1321
1322 iqcal_image_i = iqcal_image_i_avg;
1323 iqcal_image_q = iqcal_image_q_avg;
1324 iqcal_tone_i = iqcal_tone_i_avg;
1325 iqcal_tone_q = iqcal_tone_q_avg;
1326
1327 // d.
1328 rot_tone_i_b = (iqcal_tone_i * iqcal_tone_i +
1329 iqcal_tone_q * iqcal_tone_q) / 1024;
1330 rot_tone_q_b = (iqcal_tone_i * iqcal_tone_q * (-1) +
1331 iqcal_tone_q * iqcal_tone_i) / 1024;
1332 rot_image_i_b = (iqcal_image_i * iqcal_tone_i -
1333 iqcal_image_q * iqcal_tone_q) / 1024;
1334 rot_image_q_b = (iqcal_image_i * iqcal_tone_q +
1335 iqcal_image_q * iqcal_tone_i) / 1024;
1336
1337 PHY_DEBUG(("[CAL] ** rot_tone_i_b = %d\n", rot_tone_i_b));
1338 PHY_DEBUG(("[CAL] ** rot_tone_q_b = %d\n", rot_tone_q_b));
1339 PHY_DEBUG(("[CAL] ** rot_image_i_b = %d\n", rot_image_i_b));
1340 PHY_DEBUG(("[CAL] ** rot_image_q_b = %d\n", rot_image_q_b));
1341
1342 // f.
1343 if (rot_tone_i_b == 0)
1344 {
1345 PHY_DEBUG(("[CAL] ** <_rx_iq_calibration_loop> ERROR *******\n"));
1346 PHY_DEBUG(("[CAL] ** rot_tone_i_b=0 to calculate EPS and THETA !!\n"));
1347 PHY_DEBUG(("[CAL] ******************************************\n"));
1348 break;
1349 }
1350
1351 a_2 = (rot_image_i_b * 32768) / rot_tone_i_b -
1352 phw_data->iq_rsdl_gain_tx_d2;
1353 b_2 = (rot_image_q_b * 32768) / rot_tone_i_b -
1354 phw_data->iq_rsdl_phase_tx_d2;
1355
1356 PHY_DEBUG(("[CAL] ** iq_rsdl_gain_tx_d2 = %d\n", phw_data->iq_rsdl_gain_tx_d2));
1357 PHY_DEBUG(("[CAL] ** iq_rsdl_phase_tx_d2= %d\n", phw_data->iq_rsdl_phase_tx_d2));
1358 PHY_DEBUG(("[CAL] ***** EPSILON/2 = %d\n", a_2));
1359 PHY_DEBUG(("[CAL] ***** THETA/2 = %d\n", b_2));
1360
1361 _sin_cos(b_2, &sin_b, &cos_b);
1362 _sin_cos(b_2*2, &sin_2b, &cos_2b);
1363 PHY_DEBUG(("[CAL] ** sin(b/2)=%d, cos(b/2)=%d\n", sin_b, cos_b));
1364 PHY_DEBUG(("[CAL] ** sin(b)=%d, cos(b)=%d\n", sin_2b, cos_2b));
1365
1366 if (cos_2b == 0)
1367 {
1368 PHY_DEBUG(("[CAL] ** <_rx_iq_calibration_loop> ERROR *******\n"));
1369 PHY_DEBUG(("[CAL] ** cos(b)=0 !!\n"));
1370 PHY_DEBUG(("[CAL] ******************************************\n"));
1371 break;
1372 }
1373
1374 // 1280 * 32768 = 41943040
1375 temp1 = (41943040/cos_2b)*cos_b;
1376
1377 //temp2 = (41943040/cos_2b)*sin_b*(-1);
1378 if (phw_data->revision == 0x2002) // 1st-cut
1379 {
1380 temp2 = (41943040/cos_2b)*sin_b*(-1);
1381 }
1382 else // 2nd-cut
1383 {
1384 temp2 = (41943040*4/cos_2b)*sin_b*(-1);
1385 }
1386
1387 rx_cal_flt_b[0] = _floor(temp1/(32768+a_2));
1388 rx_cal_flt_b[1] = _floor(temp2/(32768-a_2));
1389 rx_cal_flt_b[2] = _floor(temp2/(32768+a_2));
1390 rx_cal_flt_b[3] = _floor(temp1/(32768-a_2));
1391
1392 PHY_DEBUG(("[CAL] ** rx_cal_flt_b[0] = %d\n", rx_cal_flt_b[0]));
1393 PHY_DEBUG(("[CAL] rx_cal_flt_b[1] = %d\n", rx_cal_flt_b[1]));
1394 PHY_DEBUG(("[CAL] rx_cal_flt_b[2] = %d\n", rx_cal_flt_b[2]));
1395 PHY_DEBUG(("[CAL] rx_cal_flt_b[3] = %d\n", rx_cal_flt_b[3]));
1396
1397 rx_cal[0] = rx_cal_flt_b[0] - 128;
1398 rx_cal[1] = rx_cal_flt_b[1];
1399 rx_cal[2] = rx_cal_flt_b[2];
1400 rx_cal[3] = rx_cal_flt_b[3] - 128;
1401 PHY_DEBUG(("[CAL] ** rx_cal[0] = %d\n", rx_cal[0]));
1402 PHY_DEBUG(("[CAL] rx_cal[1] = %d\n", rx_cal[1]));
1403 PHY_DEBUG(("[CAL] rx_cal[2] = %d\n", rx_cal[2]));
1404 PHY_DEBUG(("[CAL] rx_cal[3] = %d\n", rx_cal[3]));
1405
1406 // e.
1407 pwr_tone = (iqcal_tone_i*iqcal_tone_i + iqcal_tone_q*iqcal_tone_q);
1408 pwr_image = (iqcal_image_i*iqcal_image_i + iqcal_image_q*iqcal_image_q)*factor;
1409
1410 PHY_DEBUG(("[CAL] ** pwr_tone = %d\n", pwr_tone));
1411 PHY_DEBUG(("[CAL] ** pwr_image = %d\n", pwr_image));
1412
1413 if (pwr_tone > pwr_image)
1414 {
1415 verify_count++;
1416
1417 PHY_DEBUG(("[CAL] ** <_rx_iq_calibration_loop> *************\n"));
1418 PHY_DEBUG(("[CAL] ** VERIFY OK # %d !!\n", verify_count));
1419 PHY_DEBUG(("[CAL] ******************************************\n"));
1420
1421 if (verify_count > 2)
1422 {
1423 PHY_DEBUG(("[CAL] ** <_rx_iq_calibration_loop> *********\n"));
1424 PHY_DEBUG(("[CAL] ** RX_IQ_CALIBRATION OK !!\n"));
1425 PHY_DEBUG(("[CAL] **************************************\n"));
1426 return 0;
1427 }
1428
1429 continue;
1430 }
1431 // g.
1432 hw_get_dxx_reg(phw_data, 0x54, &val);
1433 PHY_DEBUG(("[CAL] ** 0x54 = 0x%08X\n", val));
1434
1435 if (phw_data->revision == 0x2002) // 1st-cut
1436 {
1437 rx_cal_reg[0] = _s4_to_s32((val & 0x0000F000) >> 12);
1438 rx_cal_reg[1] = _s4_to_s32((val & 0x00000F00) >> 8);
1439 rx_cal_reg[2] = _s4_to_s32((val & 0x000000F0) >> 4);
1440 rx_cal_reg[3] = _s4_to_s32((val & 0x0000000F));
1441 }
1442 else // 2nd-cut
1443 {
1444 rx_cal_reg[0] = _s5_to_s32((val & 0xF8000000) >> 27);
1445 rx_cal_reg[1] = _s6_to_s32((val & 0x07E00000) >> 21);
1446 rx_cal_reg[2] = _s6_to_s32((val & 0x001F8000) >> 15);
1447 rx_cal_reg[3] = _s5_to_s32((val & 0x00007C00) >> 10);
1448 }
1449
1450 PHY_DEBUG(("[CAL] ** rx_cal_reg[0] = %d\n", rx_cal_reg[0]));
1451 PHY_DEBUG(("[CAL] rx_cal_reg[1] = %d\n", rx_cal_reg[1]));
1452 PHY_DEBUG(("[CAL] rx_cal_reg[2] = %d\n", rx_cal_reg[2]));
1453 PHY_DEBUG(("[CAL] rx_cal_reg[3] = %d\n", rx_cal_reg[3]));
1454
1455 if (phw_data->revision == 0x2002) // 1st-cut
1456 {
1457 if (((rx_cal_reg[0]==7) || (rx_cal_reg[0]==(-8))) &&
1458 ((rx_cal_reg[3]==7) || (rx_cal_reg[3]==(-8))))
1459 {
1460 PHY_DEBUG(("[CAL] ** <_rx_iq_calibration_loop> *********\n"));
1461 PHY_DEBUG(("[CAL] ** RX_IQ_CALIBRATION SATUATION !!\n"));
1462 PHY_DEBUG(("[CAL] **************************************\n"));
1463 break;
1464 }
1465 }
1466 else // 2nd-cut
1467 {
1468 if (((rx_cal_reg[0]==31) || (rx_cal_reg[0]==(-32))) &&
1469 ((rx_cal_reg[3]==31) || (rx_cal_reg[3]==(-32))))
1470 {
1471 PHY_DEBUG(("[CAL] ** <_rx_iq_calibration_loop> *********\n"));
1472 PHY_DEBUG(("[CAL] ** RX_IQ_CALIBRATION SATUATION !!\n"));
1473 PHY_DEBUG(("[CAL] **************************************\n"));
1474 break;
1475 }
1476 }
1477
1478 rx_cal[0] = rx_cal[0] + rx_cal_reg[0];
1479 rx_cal[1] = rx_cal[1] + rx_cal_reg[1];
1480 rx_cal[2] = rx_cal[2] + rx_cal_reg[2];
1481 rx_cal[3] = rx_cal[3] + rx_cal_reg[3];
1482 PHY_DEBUG(("[CAL] ** apply rx_cal[0] = %d\n", rx_cal[0]));
1483 PHY_DEBUG(("[CAL] apply rx_cal[1] = %d\n", rx_cal[1]));
1484 PHY_DEBUG(("[CAL] apply rx_cal[2] = %d\n", rx_cal[2]));
1485 PHY_DEBUG(("[CAL] apply rx_cal[3] = %d\n", rx_cal[3]));
1486
1487 hw_get_dxx_reg(phw_data, 0x54, &val);
1488 if (phw_data->revision == 0x2002) // 1st-cut
1489 {
1490 val &= 0x0000FFFF;
1491 val |= ((_s32_to_s4(rx_cal[0]) << 12)|
1492 (_s32_to_s4(rx_cal[1]) << 8)|
1493 (_s32_to_s4(rx_cal[2]) << 4)|
1494 (_s32_to_s4(rx_cal[3])));
1495 hw_set_dxx_reg(phw_data, 0x54, val);
1496 }
1497 else // 2nd-cut
1498 {
1499 val &= 0x000003FF;
1500 val |= ((_s32_to_s5(rx_cal[0]) << 27)|
1501 (_s32_to_s6(rx_cal[1]) << 21)|
1502 (_s32_to_s6(rx_cal[2]) << 15)|
1503 (_s32_to_s5(rx_cal[3]) << 10));
1504 hw_set_dxx_reg(phw_data, 0x54, val);
1505
1506 if( loop == 3 )
1507 return 0;
1508 }
1509 PHY_DEBUG(("[CAL] ** CALIB_DATA = 0x%08X\n", val));
1510
1511 loop--;
1512 }
1513
1514 return 1;
1515 }
1516
1517 //////////////////////////////////////////////////////////
1518
1519 //////////////////////////////////////////////////////////////////////////
1520 void _rx_iq_calibration_winbond(struct hw_data *phw_data, u32 frequency)
1521 {
1522 // figo 20050523 marked thsi flag for can't compile for relesase
1523 #ifdef _DEBUG
1524 s32 rx_cal_reg[4];
1525 u32 val;
1526 #endif
1527
1528 u8 result;
1529
1530 PHY_DEBUG(("[CAL] -> [5]_rx_iq_calibration()\n"));
1531 // a. Set RFIC to "RX calibration mode"
1532 //; ----- Calibration (7). RX path IQ imbalance calibration loop
1533 // 0x01 0xFFBFC2 ; 3FEFF ; Calibration (7a). enable RX IQ calibration loop circuits
1534 phy_set_rf_data(phw_data, 1, (1<<24)|0xEFBFC2);
1535 // 0x0B 0x1A01D6 ; 06817 ; Calibration (7b). enable RX I/Q cal loop SW1 circuit
1536 phy_set_rf_data(phw_data, 11, (11<<24)|0x1A05D6);
1537 //0x05 0x24848A ; 09212 ; Calibration (7c). setting TX-VGA gain (TXGCH) to 2 --> to be optimized
1538 phy_set_rf_data(phw_data, 5, (5<<24)| phw_data->txvga_setting_for_cal);
1539 //0x06 0x06840C ; 01A10 ; Calibration (7d). RXGCH=00; RXGCL=010 000 (RXVGA) --> to be optimized
1540 phy_set_rf_data(phw_data, 6, (6<<24)|0x06834C);
1541 //0x00 0xFFF1C0 ; 3F7C7 ; Calibration (7e). turn on IQ imbalance/Test mode
1542 phy_set_rf_data(phw_data, 0, (0<<24)|0xFFF1C0);
1543
1544 // ; [BB-chip]: Calibration (7f). Send test pattern
1545 // ; [BB-chip]: Calibration (7g). Search RXGCL optimal value
1546 // ; [BB-chip]: Calibration (7h). Caculate RX-path IQ imbalance and setting RX path IQ compensation table
1547
1548 result = _rx_iq_calibration_loop_winbond(phw_data, 12589, frequency);
1549
1550 if (result > 0)
1551 {
1552 _reset_rx_cal(phw_data);
1553 result = _rx_iq_calibration_loop_winbond(phw_data, 7943, frequency);
1554
1555 if (result > 0)
1556 {
1557 _reset_rx_cal(phw_data);
1558 result = _rx_iq_calibration_loop_winbond(phw_data, 5011, frequency);
1559
1560 if (result > 0)
1561 {
1562 PHY_DEBUG(("[CAL] ** <_rx_iq_calibration> **************\n"));
1563 PHY_DEBUG(("[CAL] ** RX_IQ_CALIBRATION FAILURE !!\n"));
1564 PHY_DEBUG(("[CAL] **************************************\n"));
1565 _reset_rx_cal(phw_data);
1566 }
1567 }
1568 }
1569
1570 #ifdef _DEBUG
1571 hw_get_dxx_reg(phw_data, 0x54, &val);
1572 PHY_DEBUG(("[CAL] ** 0x54 = 0x%08X\n", val));
1573
1574 if (phw_data->revision == 0x2002) // 1st-cut
1575 {
1576 rx_cal_reg[0] = _s4_to_s32((val & 0x0000F000) >> 12);
1577 rx_cal_reg[1] = _s4_to_s32((val & 0x00000F00) >> 8);
1578 rx_cal_reg[2] = _s4_to_s32((val & 0x000000F0) >> 4);
1579 rx_cal_reg[3] = _s4_to_s32((val & 0x0000000F));
1580 }
1581 else // 2nd-cut
1582 {
1583 rx_cal_reg[0] = _s5_to_s32((val & 0xF8000000) >> 27);
1584 rx_cal_reg[1] = _s6_to_s32((val & 0x07E00000) >> 21);
1585 rx_cal_reg[2] = _s6_to_s32((val & 0x001F8000) >> 15);
1586 rx_cal_reg[3] = _s5_to_s32((val & 0x00007C00) >> 10);
1587 }
1588
1589 PHY_DEBUG(("[CAL] ** rx_cal_reg[0] = %d\n", rx_cal_reg[0]));
1590 PHY_DEBUG(("[CAL] rx_cal_reg[1] = %d\n", rx_cal_reg[1]));
1591 PHY_DEBUG(("[CAL] rx_cal_reg[2] = %d\n", rx_cal_reg[2]));
1592 PHY_DEBUG(("[CAL] rx_cal_reg[3] = %d\n", rx_cal_reg[3]));
1593 #endif
1594
1595 }
1596
1597 ////////////////////////////////////////////////////////////////////////
1598 void phy_calibration_winbond(struct hw_data *phw_data, u32 frequency)
1599 {
1600 u32 reg_mode_ctrl;
1601 u32 iq_alpha;
1602
1603 PHY_DEBUG(("[CAL] -> phy_calibration_winbond()\n"));
1604
1605 // 20040701 1.1.25.1000 kevin
1606 hw_get_cxx_reg(phw_data, 0x80, &mac_ctrl);
1607 hw_get_cxx_reg(phw_data, 0xE4, &rf_ctrl);
1608 hw_get_dxx_reg(phw_data, 0x58, &iq_alpha);
1609
1610
1611
1612 _rxadc_dc_offset_cancellation_winbond(phw_data, frequency);
1613 //_txidac_dc_offset_cancellation_winbond(phw_data);
1614 //_txqdac_dc_offset_cacellation_winbond(phw_data);
1615
1616 _tx_iq_calibration_winbond(phw_data);
1617 _rx_iq_calibration_winbond(phw_data, frequency);
1618
1619 //------------------------------------------------------------------------
1620 hw_get_dxx_reg(phw_data, REG_MODE_CTRL, &reg_mode_ctrl);
1621 reg_mode_ctrl &= ~(MASK_IQCAL_TONE_SEL|MASK_IQCAL_MODE|MASK_CALIB_START); // set when finish
1622 hw_set_dxx_reg(phw_data, REG_MODE_CTRL, reg_mode_ctrl);
1623 PHY_DEBUG(("[CAL] MODE_CTRL (write) = 0x%08X\n", reg_mode_ctrl));
1624
1625 // i. Set RFIC to "Normal mode"
1626 hw_set_cxx_reg(phw_data, 0x80, mac_ctrl);
1627 hw_set_cxx_reg(phw_data, 0xE4, rf_ctrl);
1628 hw_set_dxx_reg(phw_data, 0x58, iq_alpha);
1629
1630
1631 //------------------------------------------------------------------------
1632 phy_init_rf(phw_data);
1633
1634 }
1635
1636 //===========================
1637 void phy_set_rf_data( struct hw_data * pHwData, u32 index, u32 value )
1638 {
1639 u32 ltmp=0;
1640
1641 switch( pHwData->phy_type )
1642 {
1643 case RF_MAXIM_2825:
1644 case RF_MAXIM_V1: // 11g Winbond 2nd BB(with Phy board (v1) + Maxim 331)
1645 ltmp = (1 << 31) | (0 << 30) | (18 << 24) | BitReverse( value, 18 );
1646 break;
1647
1648 case RF_MAXIM_2827:
1649 ltmp = (1 << 31) | (0 << 30) | (18 << 24) | BitReverse( value, 18 );
1650 break;
1651
1652 case RF_MAXIM_2828:
1653 ltmp = (1 << 31) | (0 << 30) | (18 << 24) | BitReverse( value, 18 );
1654 break;
1655
1656 case RF_MAXIM_2829:
1657 ltmp = (1 << 31) | (0 << 30) | (18 << 24) | BitReverse( value, 18 );
1658 break;
1659
1660 case RF_AIROHA_2230:
1661 case RF_AIROHA_2230S: // 20060420 Add this
1662 ltmp = (1 << 31) | (0 << 30) | (20 << 24) | BitReverse( value, 20 );
1663 break;
1664
1665 case RF_AIROHA_7230:
1666 ltmp = (1 << 31) | (0 << 30) | (24 << 24) | (value&0xffffff);
1667 break;
1668
1669 case RF_WB_242:
1670 case RF_WB_242_1: // 20060619.5 Add
1671 ltmp = (1 << 31) | (0 << 30) | (24 << 24) | BitReverse( value, 24 );
1672 break;
1673 }
1674
1675 Wb35Reg_WriteSync( pHwData, 0x0864, ltmp );
1676 }
1677
1678 // 20060717 modify as Bruce's mail
1679 unsigned char adjust_TXVGA_for_iq_mag(struct hw_data *phw_data)
1680 {
1681 int init_txvga = 0;
1682 u32 reg_mode_ctrl;
1683 u32 val;
1684 s32 iqcal_tone_i0;
1685 s32 iqcal_tone_q0;
1686 u32 sqsum;
1687 s32 iq_mag_0_tx;
1688 u8 reg_state;
1689 int current_txvga;
1690
1691
1692 reg_state = 0;
1693 for( init_txvga=0; init_txvga<10; init_txvga++)
1694 {
1695 current_txvga = ( 0x24C40A|(init_txvga<<6) );
1696 phy_set_rf_data(phw_data, 5, ((5<<24)|current_txvga) );
1697 phw_data->txvga_setting_for_cal = current_txvga;
1698
1699 msleep(30); // 20060612.1.a
1700
1701 if( !hw_get_dxx_reg(phw_data, REG_MODE_CTRL, &reg_mode_ctrl) ) // 20060718.1 modify
1702 return false;
1703
1704 PHY_DEBUG(("[CAL] MODE_CTRL (read) = 0x%08X\n", reg_mode_ctrl));
1705
1706 // a. Set iqcal_mode[1:0] to 0x2 and set "calib_start" to 0x1 to
1707 // enable "IQ alibration Mode II"
1708 reg_mode_ctrl &= ~(MASK_IQCAL_TONE_SEL|MASK_IQCAL_MODE);
1709 reg_mode_ctrl &= ~MASK_IQCAL_MODE;
1710 reg_mode_ctrl |= (MASK_CALIB_START|0x02);
1711 reg_mode_ctrl |= (MASK_CALIB_START|0x02|2<<2);
1712 hw_set_dxx_reg(phw_data, REG_MODE_CTRL, reg_mode_ctrl);
1713 PHY_DEBUG(("[CAL] MODE_CTRL (write) = 0x%08X\n", reg_mode_ctrl));
1714
1715 udelay(1); // 20060612.1.a
1716
1717 udelay(300); // 20060612.1.a
1718
1719 // b.
1720 hw_get_dxx_reg(phw_data, REG_CALIB_READ1, &val);
1721
1722 PHY_DEBUG(("[CAL] CALIB_READ1 = 0x%08X\n", val));
1723 udelay(300); // 20060612.1.a
1724
1725 iqcal_tone_i0 = _s13_to_s32(val & 0x00001FFF);
1726 iqcal_tone_q0 = _s13_to_s32((val & 0x03FFE000) >> 13);
1727 PHY_DEBUG(("[CAL] ** iqcal_tone_i0=%d, iqcal_tone_q0=%d\n",
1728 iqcal_tone_i0, iqcal_tone_q0));
1729
1730 sqsum = iqcal_tone_i0*iqcal_tone_i0 + iqcal_tone_q0*iqcal_tone_q0;
1731 iq_mag_0_tx = (s32) _sqrt(sqsum);
1732 PHY_DEBUG(("[CAL] ** auto_adjust_txvga_for_iq_mag_0_tx=%d\n", iq_mag_0_tx));
1733
1734 if( iq_mag_0_tx>=700 && iq_mag_0_tx<=1750 )
1735 break;
1736 else if(iq_mag_0_tx > 1750)
1737 {
1738 init_txvga=-2;
1739 continue;
1740 }
1741 else
1742 continue;
1743
1744 }
1745
1746 if( iq_mag_0_tx>=700 && iq_mag_0_tx<=1750 )
1747 return true;
1748 else
1749 return false;
1750 }
1751
1752
1753