1 // SPDX-License-Identifier: GPL-2.0-or-later
4 Broadcom B43 wireless driver
6 G PHY LO (LocalOscillator) Measuring and Control routines
8 Copyright (c) 2005 Martin Langer <martin-langer@gmx.de>,
9 Copyright (c) 2005, 2006 Stefano Brivio <stefano.brivio@polimi.it>
10 Copyright (c) 2005-2007 Michael Buesch <m@bues.ch>
11 Copyright (c) 2005, 2006 Danny van Dyk <kugelfang@gentoo.org>
12 Copyright (c) 2005, 2006 Andreas Jaggi <andreas.jaggi@waterwave.ch>
22 #include <linux/delay.h>
23 #include <linux/sched.h>
24 #include <linux/slab.h>
27 static struct b43_lo_calib
*b43_find_lo_calib(struct b43_txpower_lo_control
*lo
,
28 const struct b43_bbatt
*bbatt
,
29 const struct b43_rfatt
*rfatt
)
31 struct b43_lo_calib
*c
;
33 list_for_each_entry(c
, &lo
->calib_list
, list
) {
34 if (!b43_compare_bbatt(&c
->bbatt
, bbatt
))
36 if (!b43_compare_rfatt(&c
->rfatt
, rfatt
))
44 /* Write the LocalOscillator Control (adjust) value-pair. */
45 static void b43_lo_write(struct b43_wldev
*dev
, struct b43_loctl
*control
)
47 struct b43_phy
*phy
= &dev
->phy
;
51 if (unlikely(abs(control
->i
) > 16 || abs(control
->q
) > 16)) {
52 b43dbg(dev
->wl
, "Invalid LO control pair "
53 "(I: %d, Q: %d)\n", control
->i
, control
->q
);
58 B43_WARN_ON(phy
->type
!= B43_PHYTYPE_G
);
60 value
= (u8
) (control
->q
);
61 value
|= ((u8
) (control
->i
)) << 8;
62 b43_phy_write(dev
, B43_PHY_LO_CTL
, value
);
65 static u16
lo_measure_feedthrough(struct b43_wldev
*dev
,
66 u16 lna
, u16 pga
, u16 trsw_rx
)
68 struct b43_phy
*phy
= &dev
->phy
;
73 lna
<<= B43_PHY_RFOVERVAL_LNA_SHIFT
;
74 pga
<<= B43_PHY_RFOVERVAL_PGA_SHIFT
;
76 B43_WARN_ON(lna
& ~B43_PHY_RFOVERVAL_LNA
);
77 B43_WARN_ON(pga
& ~B43_PHY_RFOVERVAL_PGA
);
78 /*FIXME This assertion fails B43_WARN_ON(trsw_rx & ~(B43_PHY_RFOVERVAL_TRSWRX |
79 B43_PHY_RFOVERVAL_BW));
81 trsw_rx
&= (B43_PHY_RFOVERVAL_TRSWRX
| B43_PHY_RFOVERVAL_BW
);
83 /* Construct the RF Override Value */
84 rfover
= B43_PHY_RFOVERVAL_UNK
;
88 if ((dev
->dev
->bus_sprom
->boardflags_lo
& B43_BFL_EXTLNA
)
90 rfover
|= B43_PHY_RFOVERVAL_EXTLNA
;
92 b43_phy_write(dev
, B43_PHY_PGACTL
, 0xE300);
93 b43_phy_write(dev
, B43_PHY_RFOVERVAL
, rfover
);
95 rfover
|= B43_PHY_RFOVERVAL_BW_LBW
;
96 b43_phy_write(dev
, B43_PHY_RFOVERVAL
, rfover
);
98 rfover
|= B43_PHY_RFOVERVAL_BW_LPF
;
99 b43_phy_write(dev
, B43_PHY_RFOVERVAL
, rfover
);
101 b43_phy_write(dev
, B43_PHY_PGACTL
, 0xF300);
103 pga
|= B43_PHY_PGACTL_UNKNOWN
;
104 b43_phy_write(dev
, B43_PHY_PGACTL
, pga
);
106 pga
|= B43_PHY_PGACTL_LOWBANDW
;
107 b43_phy_write(dev
, B43_PHY_PGACTL
, pga
);
109 pga
|= B43_PHY_PGACTL_LPF
;
110 b43_phy_write(dev
, B43_PHY_PGACTL
, pga
);
113 feedthrough
= b43_phy_read(dev
, B43_PHY_LO_LEAKAGE
);
115 /* This is a good place to check if we need to relax a bit,
116 * as this is the main function called regularly
117 * in the LO calibration. */
123 /* TXCTL Register and Value Table.
124 * Returns the "TXCTL Register".
125 * "value" is the "TXCTL Value".
126 * "pad_mix_gain" is the PAD Mixer Gain.
128 static u16
lo_txctl_register_table(struct b43_wldev
*dev
,
129 u16
*value
, u16
*pad_mix_gain
)
131 struct b43_phy
*phy
= &dev
->phy
;
134 if (phy
->type
== B43_PHYTYPE_B
) {
136 if (phy
->radio_rev
<= 5) {
144 if (phy
->rev
>= 2 && phy
->radio_rev
== 8) {
157 *pad_mix_gain
= padmix
;
162 static void lo_measure_txctl_values(struct b43_wldev
*dev
)
164 struct b43_phy
*phy
= &dev
->phy
;
165 struct b43_phy_g
*gphy
= phy
->g
;
166 struct b43_txpower_lo_control
*lo
= gphy
->lo_control
;
171 static const u8 tx_bias_values
[] = {
172 0x09, 0x08, 0x0A, 0x01, 0x00,
173 0x02, 0x05, 0x04, 0x06,
175 static const u8 tx_magn_values
[] = {
179 if (!has_loopback_gain(phy
)) {
184 int lb_gain
; /* Loopback gain (in dB) */
187 lb_gain
= gphy
->max_lb_gain
/ 2;
190 pga
= abs(10 - lb_gain
) / 6;
191 pga
= clamp_val(pga
, 0, 15);
198 if ((phy
->rev
>= 2) &&
199 (phy
->radio_ver
== 0x2050) && (phy
->radio_rev
== 8))
202 if ((10 - lb_gain
) < cmp_val
)
203 tmp
= (10 - lb_gain
);
211 radio_pctl_reg
= cmp_val
;
213 radio_pctl_reg
= tmp
;
216 b43_radio_maskset(dev
, 0x43, 0xFFF0, radio_pctl_reg
);
217 b43_gphy_set_baseband_attenuation(dev
, 2);
219 reg
= lo_txctl_register_table(dev
, &mask
, NULL
);
221 b43_radio_mask(dev
, reg
, mask
);
223 if (has_tx_magnification(phy
)) {
226 int min_feedth
= 0xFFFF;
229 for (i
= 0; i
< ARRAY_SIZE(tx_magn_values
); i
++) {
230 tx_magn
= tx_magn_values
[i
];
231 b43_radio_maskset(dev
, 0x52, 0xFF0F, tx_magn
);
232 for (j
= 0; j
< ARRAY_SIZE(tx_bias_values
); j
++) {
233 tx_bias
= tx_bias_values
[j
];
234 b43_radio_maskset(dev
, 0x52, 0xFFF0, tx_bias
);
236 lo_measure_feedthrough(dev
, 0, pga
,
238 if (feedthrough
< min_feedth
) {
239 lo
->tx_bias
= tx_bias
;
240 lo
->tx_magn
= tx_magn
;
241 min_feedth
= feedthrough
;
243 if (lo
->tx_bias
== 0)
246 b43_radio_write16(dev
, 0x52,
247 (b43_radio_read16(dev
, 0x52)
248 & 0xFF00) | lo
->tx_bias
| lo
->
254 b43_radio_mask(dev
, 0x52, 0xFFF0); /* TX bias == 0 */
256 lo
->txctl_measured_time
= jiffies
;
259 static void lo_read_power_vector(struct b43_wldev
*dev
)
261 struct b43_phy
*phy
= &dev
->phy
;
262 struct b43_phy_g
*gphy
= phy
->g
;
263 struct b43_txpower_lo_control
*lo
= gphy
->lo_control
;
266 u64 power_vector
= 0;
268 for (i
= 0; i
< 8; i
+= 2) {
269 tmp
= b43_shm_read16(dev
, B43_SHM_SHARED
, 0x310 + i
);
270 power_vector
|= (tmp
<< (i
* 8));
271 /* Clear the vector on the device. */
272 b43_shm_write16(dev
, B43_SHM_SHARED
, 0x310 + i
, 0);
275 lo
->power_vector
= power_vector
;
276 lo
->pwr_vec_read_time
= jiffies
;
279 /* 802.11/LO/GPHY/MeasuringGains */
280 static void lo_measure_gain_values(struct b43_wldev
*dev
,
281 s16 max_rx_gain
, int use_trsw_rx
)
283 struct b43_phy
*phy
= &dev
->phy
;
284 struct b43_phy_g
*gphy
= phy
->g
;
290 if (has_loopback_gain(phy
)) {
294 trsw_rx_gain
= gphy
->trsw_rx_gain
/ 2;
295 if (max_rx_gain
>= trsw_rx_gain
) {
296 trsw_rx_gain
= max_rx_gain
- trsw_rx_gain
;
299 trsw_rx_gain
= max_rx_gain
;
300 if (trsw_rx_gain
< 9) {
301 gphy
->lna_lod_gain
= 0;
303 gphy
->lna_lod_gain
= 1;
306 trsw_rx_gain
= clamp_val(trsw_rx_gain
, 0, 0x2D);
307 gphy
->pga_gain
= trsw_rx_gain
/ 3;
308 if (gphy
->pga_gain
>= 5) {
315 gphy
->trsw_rx_gain
= 0x20;
316 if (max_rx_gain
>= 0x14) {
317 gphy
->lna_lod_gain
= 1;
319 } else if (max_rx_gain
>= 0x12) {
320 gphy
->lna_lod_gain
= 1;
322 } else if (max_rx_gain
>= 0xF) {
323 gphy
->lna_lod_gain
= 1;
326 gphy
->lna_lod_gain
= 0;
331 tmp
= b43_radio_read16(dev
, 0x7A);
332 if (gphy
->lna_lod_gain
== 0)
336 b43_radio_write16(dev
, 0x7A, tmp
);
339 struct lo_g_saved_values
{
349 u16 phy_dacctl_hwpctl
;
352 u16 phy_hpwr_tssictl
;
354 u16 phy_analogoverval
;
366 /* Radio registers */
372 static void lo_measure_setup(struct b43_wldev
*dev
,
373 struct lo_g_saved_values
*sav
)
375 struct ssb_sprom
*sprom
= dev
->dev
->bus_sprom
;
376 struct b43_phy
*phy
= &dev
->phy
;
377 struct b43_phy_g
*gphy
= phy
->g
;
378 struct b43_txpower_lo_control
*lo
= gphy
->lo_control
;
381 if (b43_has_hardware_pctl(dev
)) {
382 sav
->phy_lo_mask
= b43_phy_read(dev
, B43_PHY_LO_MASK
);
383 sav
->phy_extg_01
= b43_phy_read(dev
, B43_PHY_EXTG(0x01));
384 sav
->phy_dacctl_hwpctl
= b43_phy_read(dev
, B43_PHY_DACCTL
);
385 sav
->phy_cck_14
= b43_phy_read(dev
, B43_PHY_CCK(0x14));
386 sav
->phy_hpwr_tssictl
= b43_phy_read(dev
, B43_PHY_HPWR_TSSICTL
);
388 b43_phy_set(dev
, B43_PHY_HPWR_TSSICTL
, 0x100);
389 b43_phy_set(dev
, B43_PHY_EXTG(0x01), 0x40);
390 b43_phy_set(dev
, B43_PHY_DACCTL
, 0x40);
391 b43_phy_set(dev
, B43_PHY_CCK(0x14), 0x200);
393 if (phy
->type
== B43_PHYTYPE_B
&&
394 phy
->radio_ver
== 0x2050 && phy
->radio_rev
< 6) {
395 b43_phy_write(dev
, B43_PHY_CCK(0x16), 0x410);
396 b43_phy_write(dev
, B43_PHY_CCK(0x17), 0x820);
399 sav
->phy_analogover
= b43_phy_read(dev
, B43_PHY_ANALOGOVER
);
400 sav
->phy_analogoverval
=
401 b43_phy_read(dev
, B43_PHY_ANALOGOVERVAL
);
402 sav
->phy_rfover
= b43_phy_read(dev
, B43_PHY_RFOVER
);
403 sav
->phy_rfoverval
= b43_phy_read(dev
, B43_PHY_RFOVERVAL
);
404 sav
->phy_classctl
= b43_phy_read(dev
, B43_PHY_CLASSCTL
);
405 sav
->phy_cck_3E
= b43_phy_read(dev
, B43_PHY_CCK(0x3E));
406 sav
->phy_crs0
= b43_phy_read(dev
, B43_PHY_CRS0
);
408 b43_phy_mask(dev
, B43_PHY_CLASSCTL
, 0xFFFC);
409 b43_phy_mask(dev
, B43_PHY_CRS0
, 0x7FFF);
410 b43_phy_set(dev
, B43_PHY_ANALOGOVER
, 0x0003);
411 b43_phy_mask(dev
, B43_PHY_ANALOGOVERVAL
, 0xFFFC);
412 if (phy
->type
== B43_PHYTYPE_G
) {
413 if ((phy
->rev
>= 7) &&
414 (sprom
->boardflags_lo
& B43_BFL_EXTLNA
)) {
415 b43_phy_write(dev
, B43_PHY_RFOVER
, 0x933);
417 b43_phy_write(dev
, B43_PHY_RFOVER
, 0x133);
420 b43_phy_write(dev
, B43_PHY_RFOVER
, 0);
422 b43_phy_write(dev
, B43_PHY_CCK(0x3E), 0);
424 sav
->reg_3F4
= b43_read16(dev
, 0x3F4);
425 sav
->reg_3E2
= b43_read16(dev
, 0x3E2);
426 sav
->radio_43
= b43_radio_read16(dev
, 0x43);
427 sav
->radio_7A
= b43_radio_read16(dev
, 0x7A);
428 sav
->phy_pgactl
= b43_phy_read(dev
, B43_PHY_PGACTL
);
429 sav
->phy_cck_2A
= b43_phy_read(dev
, B43_PHY_CCK(0x2A));
430 sav
->phy_syncctl
= b43_phy_read(dev
, B43_PHY_SYNCCTL
);
431 sav
->phy_dacctl
= b43_phy_read(dev
, B43_PHY_DACCTL
);
433 if (!has_tx_magnification(phy
)) {
434 sav
->radio_52
= b43_radio_read16(dev
, 0x52);
435 sav
->radio_52
&= 0x00F0;
437 if (phy
->type
== B43_PHYTYPE_B
) {
438 sav
->phy_cck_30
= b43_phy_read(dev
, B43_PHY_CCK(0x30));
439 sav
->phy_cck_06
= b43_phy_read(dev
, B43_PHY_CCK(0x06));
440 b43_phy_write(dev
, B43_PHY_CCK(0x30), 0x00FF);
441 b43_phy_write(dev
, B43_PHY_CCK(0x06), 0x3F3F);
443 b43_write16(dev
, 0x3E2, b43_read16(dev
, 0x3E2)
446 b43_write16(dev
, 0x3F4, b43_read16(dev
, 0x3F4)
450 (phy
->type
== B43_PHYTYPE_G
) ? B43_PHY_LO_MASK
: B43_PHY_CCK(0x2E);
451 b43_phy_write(dev
, tmp
, 0x007F);
453 tmp
= sav
->phy_syncctl
;
454 b43_phy_write(dev
, B43_PHY_SYNCCTL
, tmp
& 0xFF7F);
456 b43_radio_write16(dev
, 0x007A, tmp
& 0xFFF0);
458 b43_phy_write(dev
, B43_PHY_CCK(0x2A), 0x8A3);
459 if (phy
->type
== B43_PHYTYPE_G
||
460 (phy
->type
== B43_PHYTYPE_B
&&
461 phy
->radio_ver
== 0x2050 && phy
->radio_rev
>= 6)) {
462 b43_phy_write(dev
, B43_PHY_CCK(0x2B), 0x1003);
464 b43_phy_write(dev
, B43_PHY_CCK(0x2B), 0x0802);
466 b43_dummy_transmission(dev
, false, true);
467 b43_gphy_channel_switch(dev
, 6, 0);
468 b43_radio_read16(dev
, 0x51); /* dummy read */
469 if (phy
->type
== B43_PHYTYPE_G
)
470 b43_phy_write(dev
, B43_PHY_CCK(0x2F), 0);
472 /* Re-measure the txctl values, if needed. */
473 if (time_before(lo
->txctl_measured_time
,
474 jiffies
- B43_LO_TXCTL_EXPIRE
))
475 lo_measure_txctl_values(dev
);
477 if (phy
->type
== B43_PHYTYPE_G
&& phy
->rev
>= 3) {
478 b43_phy_write(dev
, B43_PHY_LO_MASK
, 0xC078);
480 if (phy
->type
== B43_PHYTYPE_B
)
481 b43_phy_write(dev
, B43_PHY_CCK(0x2E), 0x8078);
483 b43_phy_write(dev
, B43_PHY_LO_MASK
, 0x8078);
487 static void lo_measure_restore(struct b43_wldev
*dev
,
488 struct lo_g_saved_values
*sav
)
490 struct b43_phy
*phy
= &dev
->phy
;
491 struct b43_phy_g
*gphy
= phy
->g
;
495 b43_phy_write(dev
, B43_PHY_PGACTL
, 0xE300);
496 tmp
= (gphy
->pga_gain
<< 8);
497 b43_phy_write(dev
, B43_PHY_RFOVERVAL
, tmp
| 0xA0);
499 b43_phy_write(dev
, B43_PHY_RFOVERVAL
, tmp
| 0xA2);
501 b43_phy_write(dev
, B43_PHY_RFOVERVAL
, tmp
| 0xA3);
503 tmp
= (gphy
->pga_gain
| 0xEFA0);
504 b43_phy_write(dev
, B43_PHY_PGACTL
, tmp
);
506 if (phy
->type
== B43_PHYTYPE_G
) {
508 b43_phy_write(dev
, B43_PHY_CCK(0x2E), 0xC078);
510 b43_phy_write(dev
, B43_PHY_CCK(0x2E), 0x8078);
512 b43_phy_write(dev
, B43_PHY_CCK(0x2F), 0x0202);
514 b43_phy_write(dev
, B43_PHY_CCK(0x2F), 0x0101);
516 b43_write16(dev
, 0x3F4, sav
->reg_3F4
);
517 b43_phy_write(dev
, B43_PHY_PGACTL
, sav
->phy_pgactl
);
518 b43_phy_write(dev
, B43_PHY_CCK(0x2A), sav
->phy_cck_2A
);
519 b43_phy_write(dev
, B43_PHY_SYNCCTL
, sav
->phy_syncctl
);
520 b43_phy_write(dev
, B43_PHY_DACCTL
, sav
->phy_dacctl
);
521 b43_radio_write16(dev
, 0x43, sav
->radio_43
);
522 b43_radio_write16(dev
, 0x7A, sav
->radio_7A
);
523 if (!has_tx_magnification(phy
)) {
525 b43_radio_maskset(dev
, 0x52, 0xFF0F, tmp
);
527 b43_write16(dev
, 0x3E2, sav
->reg_3E2
);
528 if (phy
->type
== B43_PHYTYPE_B
&&
529 phy
->radio_ver
== 0x2050 && phy
->radio_rev
<= 5) {
530 b43_phy_write(dev
, B43_PHY_CCK(0x30), sav
->phy_cck_30
);
531 b43_phy_write(dev
, B43_PHY_CCK(0x06), sav
->phy_cck_06
);
534 b43_phy_write(dev
, B43_PHY_ANALOGOVER
, sav
->phy_analogover
);
535 b43_phy_write(dev
, B43_PHY_ANALOGOVERVAL
,
536 sav
->phy_analogoverval
);
537 b43_phy_write(dev
, B43_PHY_CLASSCTL
, sav
->phy_classctl
);
538 b43_phy_write(dev
, B43_PHY_RFOVER
, sav
->phy_rfover
);
539 b43_phy_write(dev
, B43_PHY_RFOVERVAL
, sav
->phy_rfoverval
);
540 b43_phy_write(dev
, B43_PHY_CCK(0x3E), sav
->phy_cck_3E
);
541 b43_phy_write(dev
, B43_PHY_CRS0
, sav
->phy_crs0
);
543 if (b43_has_hardware_pctl(dev
)) {
544 tmp
= (sav
->phy_lo_mask
& 0xBFFF);
545 b43_phy_write(dev
, B43_PHY_LO_MASK
, tmp
);
546 b43_phy_write(dev
, B43_PHY_EXTG(0x01), sav
->phy_extg_01
);
547 b43_phy_write(dev
, B43_PHY_DACCTL
, sav
->phy_dacctl_hwpctl
);
548 b43_phy_write(dev
, B43_PHY_CCK(0x14), sav
->phy_cck_14
);
549 b43_phy_write(dev
, B43_PHY_HPWR_TSSICTL
, sav
->phy_hpwr_tssictl
);
551 b43_gphy_channel_switch(dev
, sav
->old_channel
, 1);
554 struct b43_lo_g_statemachine
{
557 int state_val_multiplier
;
559 struct b43_loctl min_loctl
;
562 /* Loop over each possible value in this state. */
563 static int lo_probe_possible_loctls(struct b43_wldev
*dev
,
564 struct b43_loctl
*probe_loctl
,
565 struct b43_lo_g_statemachine
*d
)
567 struct b43_phy
*phy
= &dev
->phy
;
568 struct b43_phy_g
*gphy
= phy
->g
;
569 struct b43_loctl test_loctl
;
570 struct b43_loctl orig_loctl
;
571 struct b43_loctl prev_loctl
= {
580 static const struct b43_loctl modifiers
[] = {
591 if (d
->current_state
== 0) {
594 } else if (d
->current_state
% 2 == 0) {
595 begin
= d
->current_state
- 1;
596 end
= d
->current_state
+ 1;
598 begin
= d
->current_state
- 2;
599 end
= d
->current_state
+ 2;
606 memcpy(&orig_loctl
, probe_loctl
, sizeof(struct b43_loctl
));
608 d
->current_state
= i
;
610 B43_WARN_ON(!(i
>= 1 && i
<= 8));
611 memcpy(&test_loctl
, &orig_loctl
, sizeof(struct b43_loctl
));
612 test_loctl
.i
+= modifiers
[i
- 1].i
* d
->state_val_multiplier
;
613 test_loctl
.q
+= modifiers
[i
- 1].q
* d
->state_val_multiplier
;
614 if ((test_loctl
.i
!= prev_loctl
.i
||
615 test_loctl
.q
!= prev_loctl
.q
) &&
616 (abs(test_loctl
.i
) <= 16 && abs(test_loctl
.q
) <= 16)) {
617 b43_lo_write(dev
, &test_loctl
);
618 feedth
= lo_measure_feedthrough(dev
, gphy
->lna_gain
,
621 if (feedth
< d
->lowest_feedth
) {
622 memcpy(probe_loctl
, &test_loctl
,
623 sizeof(struct b43_loctl
));
625 d
->lowest_feedth
= feedth
;
626 if ((d
->nr_measured
< 2) &&
627 !has_loopback_gain(phy
))
631 memcpy(&prev_loctl
, &test_loctl
, sizeof(prev_loctl
));
638 d
->current_state
= i
;
644 static void lo_probe_loctls_statemachine(struct b43_wldev
*dev
,
645 struct b43_loctl
*loctl
,
648 struct b43_phy
*phy
= &dev
->phy
;
649 struct b43_phy_g
*gphy
= phy
->g
;
650 struct b43_lo_g_statemachine d
;
653 struct b43_loctl probe_loctl
;
654 int max_repeat
= 1, repeat_cnt
= 0;
657 d
.state_val_multiplier
= 1;
658 if (has_loopback_gain(phy
))
659 d
.state_val_multiplier
= 3;
661 memcpy(&d
.min_loctl
, loctl
, sizeof(struct b43_loctl
));
662 if (has_loopback_gain(phy
))
665 b43_lo_write(dev
, &d
.min_loctl
);
666 feedth
= lo_measure_feedthrough(dev
, gphy
->lna_gain
,
669 if (feedth
< 0x258) {
674 feedth
= lo_measure_feedthrough(dev
, gphy
->lna_gain
,
678 d
.lowest_feedth
= feedth
;
683 (d
.current_state
>= 0
684 && d
.current_state
<= 8));
685 memcpy(&probe_loctl
, &d
.min_loctl
,
686 sizeof(struct b43_loctl
));
688 lo_probe_possible_loctls(dev
, &probe_loctl
, &d
);
691 if ((probe_loctl
.i
== d
.min_loctl
.i
) &&
692 (probe_loctl
.q
== d
.min_loctl
.q
))
694 memcpy(&d
.min_loctl
, &probe_loctl
,
695 sizeof(struct b43_loctl
));
697 } while (d
.nr_measured
< 24);
698 memcpy(loctl
, &d
.min_loctl
, sizeof(struct b43_loctl
));
700 if (has_loopback_gain(phy
)) {
701 if (d
.lowest_feedth
> 0x1194)
703 else if (d
.lowest_feedth
< 0x5DC)
705 if (repeat_cnt
== 0) {
706 if (d
.lowest_feedth
<= 0x5DC) {
707 d
.state_val_multiplier
= 1;
710 d
.state_val_multiplier
= 2;
711 } else if (repeat_cnt
== 2)
712 d
.state_val_multiplier
= 1;
714 lo_measure_gain_values(dev
, *max_rx_gain
,
715 has_loopback_gain(phy
));
716 } while (++repeat_cnt
< max_repeat
);
720 struct b43_lo_calib
*b43_calibrate_lo_setting(struct b43_wldev
*dev
,
721 const struct b43_bbatt
*bbatt
,
722 const struct b43_rfatt
*rfatt
)
724 struct b43_phy
*phy
= &dev
->phy
;
725 struct b43_phy_g
*gphy
= phy
->g
;
726 struct b43_loctl loctl
= {
731 struct b43_lo_calib
*cal
;
732 struct lo_g_saved_values saved_regs
;
733 /* Values from the "TXCTL Register and Value Table" */
738 saved_regs
.old_channel
= phy
->channel
;
739 b43_mac_suspend(dev
);
740 lo_measure_setup(dev
, &saved_regs
);
742 txctl_reg
= lo_txctl_register_table(dev
, &txctl_value
, &pad_mix_gain
);
744 b43_radio_maskset(dev
, 0x43, 0xFFF0, rfatt
->att
);
745 b43_radio_maskset(dev
, txctl_reg
, ~txctl_value
, (rfatt
->with_padmix
? txctl_value
:0));
747 max_rx_gain
= rfatt
->att
* 2;
748 max_rx_gain
+= bbatt
->att
/ 2;
749 if (rfatt
->with_padmix
)
750 max_rx_gain
-= pad_mix_gain
;
751 if (has_loopback_gain(phy
))
752 max_rx_gain
+= gphy
->max_lb_gain
;
753 lo_measure_gain_values(dev
, max_rx_gain
,
754 has_loopback_gain(phy
));
756 b43_gphy_set_baseband_attenuation(dev
, bbatt
->att
);
757 lo_probe_loctls_statemachine(dev
, &loctl
, &max_rx_gain
);
759 lo_measure_restore(dev
, &saved_regs
);
762 if (b43_debug(dev
, B43_DBG_LO
)) {
763 b43dbg(dev
->wl
, "LO: Calibrated for BB(%u), RF(%u,%u) "
765 bbatt
->att
, rfatt
->att
, rfatt
->with_padmix
,
769 cal
= kmalloc(sizeof(*cal
), GFP_KERNEL
);
771 b43warn(dev
->wl
, "LO calib: out of memory\n");
774 memcpy(&cal
->bbatt
, bbatt
, sizeof(*bbatt
));
775 memcpy(&cal
->rfatt
, rfatt
, sizeof(*rfatt
));
776 memcpy(&cal
->ctl
, &loctl
, sizeof(loctl
));
777 cal
->calib_time
= jiffies
;
778 INIT_LIST_HEAD(&cal
->list
);
783 /* Get a calibrated LO setting for the given attenuation values.
784 * Might return a NULL pointer under OOM! */
786 struct b43_lo_calib
*b43_get_calib_lo_settings(struct b43_wldev
*dev
,
787 const struct b43_bbatt
*bbatt
,
788 const struct b43_rfatt
*rfatt
)
790 struct b43_txpower_lo_control
*lo
= dev
->phy
.g
->lo_control
;
791 struct b43_lo_calib
*c
;
793 c
= b43_find_lo_calib(lo
, bbatt
, rfatt
);
796 /* Not in the list of calibrated LO settings.
797 * Calibrate it now. */
798 c
= b43_calibrate_lo_setting(dev
, bbatt
, rfatt
);
801 list_add(&c
->list
, &lo
->calib_list
);
806 void b43_gphy_dc_lt_init(struct b43_wldev
*dev
, bool update_all
)
808 struct b43_phy
*phy
= &dev
->phy
;
809 struct b43_phy_g
*gphy
= phy
->g
;
810 struct b43_txpower_lo_control
*lo
= gphy
->lo_control
;
812 int rf_offset
, bb_offset
;
813 const struct b43_rfatt
*rfatt
;
814 const struct b43_bbatt
*bbatt
;
816 bool table_changed
= false;
818 BUILD_BUG_ON(B43_DC_LT_SIZE
!= 32);
819 B43_WARN_ON(lo
->rfatt_list
.len
* lo
->bbatt_list
.len
> 64);
821 power_vector
= lo
->power_vector
;
822 if (!update_all
&& !power_vector
)
823 return; /* Nothing to do. */
825 /* Suspend the MAC now to avoid continuous suspend/enable
826 * cycles in the loop. */
827 b43_mac_suspend(dev
);
829 for (i
= 0; i
< B43_DC_LT_SIZE
* 2; i
++) {
830 struct b43_lo_calib
*cal
;
834 if (!update_all
&& !(power_vector
& (((u64
)1ULL) << i
)))
836 /* Update the table entry for this power_vector bit.
837 * The table rows are RFatt entries and columns are BBatt. */
838 bb_offset
= i
/ lo
->rfatt_list
.len
;
839 rf_offset
= i
% lo
->rfatt_list
.len
;
840 bbatt
= &(lo
->bbatt_list
.list
[bb_offset
]);
841 rfatt
= &(lo
->rfatt_list
.list
[rf_offset
]);
843 cal
= b43_calibrate_lo_setting(dev
, bbatt
, rfatt
);
845 b43warn(dev
->wl
, "LO: Could not "
846 "calibrate DC table entry\n");
849 /*FIXME: Is Q really in the low nibble? */
850 val
= (u8
)(cal
->ctl
.q
);
851 val
|= ((u8
)(cal
->ctl
.i
)) << 4;
854 /* Get the index into the hardware DC LT. */
856 /* Change the table in memory. */
858 /* Change the high byte. */
859 lo
->dc_lt
[idx
] = (lo
->dc_lt
[idx
] & 0x00FF)
860 | ((val
& 0x00FF) << 8);
862 /* Change the low byte. */
863 lo
->dc_lt
[idx
] = (lo
->dc_lt
[idx
] & 0xFF00)
866 table_changed
= true;
869 /* The table changed in memory. Update the hardware table. */
870 for (i
= 0; i
< B43_DC_LT_SIZE
; i
++)
871 b43_phy_write(dev
, 0x3A0 + i
, lo
->dc_lt
[i
]);
876 /* Fixup the RF attenuation value for the case where we are
877 * using the PAD mixer. */
878 static inline void b43_lo_fixup_rfatt(struct b43_rfatt
*rf
)
880 if (!rf
->with_padmix
)
882 if ((rf
->att
!= 1) && (rf
->att
!= 2) && (rf
->att
!= 3))
886 void b43_lo_g_adjust(struct b43_wldev
*dev
)
888 struct b43_phy_g
*gphy
= dev
->phy
.g
;
889 struct b43_lo_calib
*cal
;
892 memcpy(&rf
, &gphy
->rfatt
, sizeof(rf
));
893 b43_lo_fixup_rfatt(&rf
);
895 cal
= b43_get_calib_lo_settings(dev
, &gphy
->bbatt
, &rf
);
898 b43_lo_write(dev
, &cal
->ctl
);
901 void b43_lo_g_adjust_to(struct b43_wldev
*dev
,
902 u16 rfatt
, u16 bbatt
, u16 tx_control
)
906 struct b43_lo_calib
*cal
;
908 memset(&rf
, 0, sizeof(rf
));
909 memset(&bb
, 0, sizeof(bb
));
912 b43_lo_fixup_rfatt(&rf
);
913 cal
= b43_get_calib_lo_settings(dev
, &bb
, &rf
);
916 b43_lo_write(dev
, &cal
->ctl
);
919 /* Periodic LO maintenance work */
920 void b43_lo_g_maintenance_work(struct b43_wldev
*dev
)
922 struct b43_phy
*phy
= &dev
->phy
;
923 struct b43_phy_g
*gphy
= phy
->g
;
924 struct b43_txpower_lo_control
*lo
= gphy
->lo_control
;
926 unsigned long expire
;
927 struct b43_lo_calib
*cal
, *tmp
;
928 bool current_item_expired
= false;
934 hwpctl
= b43_has_hardware_pctl(dev
);
937 /* Read the power vector and update it, if needed. */
938 expire
= now
- B43_LO_PWRVEC_EXPIRE
;
939 if (time_before(lo
->pwr_vec_read_time
, expire
)) {
940 lo_read_power_vector(dev
);
941 b43_gphy_dc_lt_init(dev
, 0);
943 //FIXME Recalc the whole DC table from time to time?
948 /* Search for expired LO settings. Remove them.
949 * Recalibrate the current setting, if expired. */
950 expire
= now
- B43_LO_CALIB_EXPIRE
;
951 list_for_each_entry_safe(cal
, tmp
, &lo
->calib_list
, list
) {
952 if (!time_before(cal
->calib_time
, expire
))
954 /* This item expired. */
955 if (b43_compare_bbatt(&cal
->bbatt
, &gphy
->bbatt
) &&
956 b43_compare_rfatt(&cal
->rfatt
, &gphy
->rfatt
)) {
957 B43_WARN_ON(current_item_expired
);
958 current_item_expired
= true;
960 if (b43_debug(dev
, B43_DBG_LO
)) {
961 b43dbg(dev
->wl
, "LO: Item BB(%u), RF(%u,%u), "
962 "I=%d, Q=%d expired\n",
963 cal
->bbatt
.att
, cal
->rfatt
.att
,
964 cal
->rfatt
.with_padmix
,
965 cal
->ctl
.i
, cal
->ctl
.q
);
967 list_del(&cal
->list
);
970 if (current_item_expired
|| unlikely(list_empty(&lo
->calib_list
))) {
971 /* Recalibrate currently used LO setting. */
972 if (b43_debug(dev
, B43_DBG_LO
))
973 b43dbg(dev
->wl
, "LO: Recalibrating current LO setting\n");
974 cal
= b43_calibrate_lo_setting(dev
, &gphy
->bbatt
, &gphy
->rfatt
);
976 list_add(&cal
->list
, &lo
->calib_list
);
977 b43_lo_write(dev
, &cal
->ctl
);
979 b43warn(dev
->wl
, "Failed to recalibrate current LO setting\n");
983 void b43_lo_g_cleanup(struct b43_wldev
*dev
)
985 struct b43_txpower_lo_control
*lo
= dev
->phy
.g
->lo_control
;
986 struct b43_lo_calib
*cal
, *tmp
;
990 list_for_each_entry_safe(cal
, tmp
, &lo
->calib_list
, list
) {
991 list_del(&cal
->list
);
996 /* LO Initialization */
997 void b43_lo_g_init(struct b43_wldev
*dev
)
999 if (b43_has_hardware_pctl(dev
)) {
1000 lo_read_power_vector(dev
);
1001 b43_gphy_dc_lt_init(dev
, 1);