1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16#include <linux/kernel.h>
17#include <linux/delay.h>
18#include <linux/bitops.h>
19
20#include <brcm_hw_ids.h>
21#include <chipcommon.h>
22#include <aiutils.h>
23#include <d11.h>
24#include <phy_shim.h>
25#include "phy_hal.h"
26#include "phy_int.h"
27#include "phy_radio.h"
28#include "phy_lcn.h"
29#include "phyreg_n.h"
30
31#define VALID_N_RADIO(radioid) ((radioid == BCM2055_ID) || \
32 (radioid == BCM2056_ID) || \
33 (radioid == BCM2057_ID))
34
35#define VALID_LCN_RADIO(radioid) (radioid == BCM2064_ID)
36
37#define VALID_RADIO(pi, radioid) ( \
38 (ISNPHY(pi) ? VALID_N_RADIO(radioid) : false) || \
39 (ISLCNPHY(pi) ? VALID_LCN_RADIO(radioid) : false))
40
41
42#define MUX(pred, true, false) ((pred) ? (true) : (false))
43
44
45#define MODINC(x, bound) MUX((x) == (bound) - 1, 0, (x) + 1)
46
47
48#define MODDEC_POW2(x, bound) (((x) - 1) & ((bound) - 1))
49#define MODINC_POW2(x, bound) (((x) + 1) & ((bound) - 1))
50
51struct chan_info_basic {
52 u16 chan;
53 u16 freq;
54};
55
56static const struct chan_info_basic chan_info_all[] = {
57 {1, 2412},
58 {2, 2417},
59 {3, 2422},
60 {4, 2427},
61 {5, 2432},
62 {6, 2437},
63 {7, 2442},
64 {8, 2447},
65 {9, 2452},
66 {10, 2457},
67 {11, 2462},
68 {12, 2467},
69 {13, 2472},
70 {14, 2484},
71
72 {34, 5170},
73 {38, 5190},
74 {42, 5210},
75 {46, 5230},
76
77 {36, 5180},
78 {40, 5200},
79 {44, 5220},
80 {48, 5240},
81 {52, 5260},
82 {56, 5280},
83 {60, 5300},
84 {64, 5320},
85
86 {100, 5500},
87 {104, 5520},
88 {108, 5540},
89 {112, 5560},
90 {116, 5580},
91 {120, 5600},
92 {124, 5620},
93 {128, 5640},
94 {132, 5660},
95 {136, 5680},
96 {140, 5700},
97
98 {149, 5745},
99 {153, 5765},
100 {157, 5785},
101 {161, 5805},
102 {165, 5825},
103
104 {184, 4920},
105 {188, 4940},
106 {192, 4960},
107 {196, 4980},
108 {200, 5000},
109 {204, 5020},
110 {208, 5040},
111 {212, 5060},
112 {216, 5080}
113};
114
115static const u8 ofdm_rate_lookup[] = {
116
117 BRCM_RATE_48M,
118 BRCM_RATE_24M,
119 BRCM_RATE_12M,
120 BRCM_RATE_6M,
121 BRCM_RATE_54M,
122 BRCM_RATE_36M,
123 BRCM_RATE_18M,
124 BRCM_RATE_9M
125};
126
127#define PHY_WREG_LIMIT 24
128
129void wlc_phyreg_enter(struct brcms_phy_pub *pih)
130{
131 struct brcms_phy *pi = container_of(pih, struct brcms_phy, pubpi_ro);
132 wlapi_bmac_ucode_wake_override_phyreg_set(pi->sh->physhim);
133}
134
135void wlc_phyreg_exit(struct brcms_phy_pub *pih)
136{
137 struct brcms_phy *pi = container_of(pih, struct brcms_phy, pubpi_ro);
138 wlapi_bmac_ucode_wake_override_phyreg_clear(pi->sh->physhim);
139}
140
141void wlc_radioreg_enter(struct brcms_phy_pub *pih)
142{
143 struct brcms_phy *pi = container_of(pih, struct brcms_phy, pubpi_ro);
144 wlapi_bmac_mctrl(pi->sh->physhim, MCTL_LOCK_RADIO, MCTL_LOCK_RADIO);
145
146 udelay(10);
147}
148
149void wlc_radioreg_exit(struct brcms_phy_pub *pih)
150{
151 struct brcms_phy *pi = container_of(pih, struct brcms_phy, pubpi_ro);
152
153 (void)bcma_read16(pi->d11core, D11REGOFFS(phyversion));
154 pi->phy_wreg = 0;
155 wlapi_bmac_mctrl(pi->sh->physhim, MCTL_LOCK_RADIO, 0);
156}
157
158u16 read_radio_reg(struct brcms_phy *pi, u16 addr)
159{
160 u16 data;
161
162 if ((addr == RADIO_IDCODE))
163 return 0xffff;
164
165 switch (pi->pubpi.phy_type) {
166 case PHY_TYPE_N:
167 if (!CONF_HAS(PHYTYPE, PHY_TYPE_N))
168 break;
169 if (NREV_GE(pi->pubpi.phy_rev, 7))
170 addr |= RADIO_2057_READ_OFF;
171 else
172 addr |= RADIO_2055_READ_OFF;
173 break;
174
175 case PHY_TYPE_LCN:
176 if (!CONF_HAS(PHYTYPE, PHY_TYPE_LCN))
177 break;
178 addr |= RADIO_2064_READ_OFF;
179 break;
180
181 default:
182 break;
183 }
184
185 if ((D11REV_GE(pi->sh->corerev, 24)) ||
186 (D11REV_IS(pi->sh->corerev, 22)
187 && (pi->pubpi.phy_type != PHY_TYPE_SSN))) {
188 bcma_wflush16(pi->d11core, D11REGOFFS(radioregaddr), addr);
189 data = bcma_read16(pi->d11core, D11REGOFFS(radioregdata));
190 } else {
191 bcma_wflush16(pi->d11core, D11REGOFFS(phy4waddr), addr);
192 data = bcma_read16(pi->d11core, D11REGOFFS(phy4wdatalo));
193 }
194 pi->phy_wreg = 0;
195
196 return data;
197}
198
199void write_radio_reg(struct brcms_phy *pi, u16 addr, u16 val)
200{
201 if ((D11REV_GE(pi->sh->corerev, 24)) ||
202 (D11REV_IS(pi->sh->corerev, 22)
203 && (pi->pubpi.phy_type != PHY_TYPE_SSN))) {
204
205 bcma_wflush16(pi->d11core, D11REGOFFS(radioregaddr), addr);
206 bcma_write16(pi->d11core, D11REGOFFS(radioregdata), val);
207 } else {
208 bcma_wflush16(pi->d11core, D11REGOFFS(phy4waddr), addr);
209 bcma_write16(pi->d11core, D11REGOFFS(phy4wdatalo), val);
210 }
211
212 if ((pi->d11core->bus->hosttype == BCMA_HOSTTYPE_PCI) &&
213 (++pi->phy_wreg >= pi->phy_wreg_limit)) {
214 (void)bcma_read32(pi->d11core, D11REGOFFS(maccontrol));
215 pi->phy_wreg = 0;
216 }
217}
218
219static u32 read_radio_id(struct brcms_phy *pi)
220{
221 u32 id;
222
223 if (D11REV_GE(pi->sh->corerev, 24)) {
224 u32 b0, b1, b2;
225
226 bcma_wflush16(pi->d11core, D11REGOFFS(radioregaddr), 0);
227 b0 = (u32) bcma_read16(pi->d11core, D11REGOFFS(radioregdata));
228 bcma_wflush16(pi->d11core, D11REGOFFS(radioregaddr), 1);
229 b1 = (u32) bcma_read16(pi->d11core, D11REGOFFS(radioregdata));
230 bcma_wflush16(pi->d11core, D11REGOFFS(radioregaddr), 2);
231 b2 = (u32) bcma_read16(pi->d11core, D11REGOFFS(radioregdata));
232
233 id = ((b0 & 0xf) << 28) | (((b2 << 8) | b1) << 12) | ((b0 >> 4)
234 & 0xf);
235 } else {
236 bcma_wflush16(pi->d11core, D11REGOFFS(phy4waddr), RADIO_IDCODE);
237 id = (u32) bcma_read16(pi->d11core, D11REGOFFS(phy4wdatalo));
238 id |= (u32) bcma_read16(pi->d11core,
239 D11REGOFFS(phy4wdatahi)) << 16;
240 }
241 pi->phy_wreg = 0;
242 return id;
243}
244
245void and_radio_reg(struct brcms_phy *pi, u16 addr, u16 val)
246{
247 u16 rval;
248
249 rval = read_radio_reg(pi, addr);
250 write_radio_reg(pi, addr, (rval & val));
251}
252
253void or_radio_reg(struct brcms_phy *pi, u16 addr, u16 val)
254{
255 u16 rval;
256
257 rval = read_radio_reg(pi, addr);
258 write_radio_reg(pi, addr, (rval | val));
259}
260
261void xor_radio_reg(struct brcms_phy *pi, u16 addr, u16 mask)
262{
263 u16 rval;
264
265 rval = read_radio_reg(pi, addr);
266 write_radio_reg(pi, addr, (rval ^ mask));
267}
268
269void mod_radio_reg(struct brcms_phy *pi, u16 addr, u16 mask, u16 val)
270{
271 u16 rval;
272
273 rval = read_radio_reg(pi, addr);
274 write_radio_reg(pi, addr, (rval & ~mask) | (val & mask));
275}
276
277void write_phy_channel_reg(struct brcms_phy *pi, uint val)
278{
279 bcma_write16(pi->d11core, D11REGOFFS(phychannel), val);
280}
281
282u16 read_phy_reg(struct brcms_phy *pi, u16 addr)
283{
284 bcma_wflush16(pi->d11core, D11REGOFFS(phyregaddr), addr);
285
286 pi->phy_wreg = 0;
287 return bcma_read16(pi->d11core, D11REGOFFS(phyregdata));
288}
289
290void write_phy_reg(struct brcms_phy *pi, u16 addr, u16 val)
291{
292#ifdef CONFIG_BCM47XX
293 bcma_wflush16(pi->d11core, D11REGOFFS(phyregaddr), addr);
294 bcma_write16(pi->d11core, D11REGOFFS(phyregdata), val);
295 if (addr == 0x72)
296 (void)bcma_read16(pi->d11core, D11REGOFFS(phyregdata));
297#else
298 bcma_write32(pi->d11core, D11REGOFFS(phyregaddr), addr | (val << 16));
299 if ((pi->d11core->bus->hosttype == BCMA_HOSTTYPE_PCI) &&
300 (++pi->phy_wreg >= pi->phy_wreg_limit)) {
301 pi->phy_wreg = 0;
302 (void)bcma_read16(pi->d11core, D11REGOFFS(phyversion));
303 }
304#endif
305}
306
307void and_phy_reg(struct brcms_phy *pi, u16 addr, u16 val)
308{
309 bcma_wflush16(pi->d11core, D11REGOFFS(phyregaddr), addr);
310 bcma_mask16(pi->d11core, D11REGOFFS(phyregdata), val);
311 pi->phy_wreg = 0;
312}
313
314void or_phy_reg(struct brcms_phy *pi, u16 addr, u16 val)
315{
316 bcma_wflush16(pi->d11core, D11REGOFFS(phyregaddr), addr);
317 bcma_set16(pi->d11core, D11REGOFFS(phyregdata), val);
318 pi->phy_wreg = 0;
319}
320
321void mod_phy_reg(struct brcms_phy *pi, u16 addr, u16 mask, u16 val)
322{
323 val &= mask;
324 bcma_wflush16(pi->d11core, D11REGOFFS(phyregaddr), addr);
325 bcma_maskset16(pi->d11core, D11REGOFFS(phyregdata), ~mask, val);
326 pi->phy_wreg = 0;
327}
328
329static void wlc_set_phy_uninitted(struct brcms_phy *pi)
330{
331 int i, j;
332
333 pi->initialized = false;
334
335 pi->tx_vos = 0xffff;
336 pi->nrssi_table_delta = 0x7fffffff;
337 pi->rc_cal = 0xffff;
338 pi->mintxbias = 0xffff;
339 pi->txpwridx = -1;
340 if (ISNPHY(pi)) {
341 pi->phy_spuravoid = SPURAVOID_DISABLE;
342
343 if (NREV_GE(pi->pubpi.phy_rev, 3)
344 && NREV_LT(pi->pubpi.phy_rev, 7))
345 pi->phy_spuravoid = SPURAVOID_AUTO;
346
347 pi->nphy_papd_skip = 0;
348 pi->nphy_papd_epsilon_offset[0] = 0xf588;
349 pi->nphy_papd_epsilon_offset[1] = 0xf588;
350 pi->nphy_txpwr_idx[0] = 128;
351 pi->nphy_txpwr_idx[1] = 128;
352 pi->nphy_txpwrindex[0].index_internal = 40;
353 pi->nphy_txpwrindex[1].index_internal = 40;
354 pi->phy_pabias = 0;
355 } else {
356 pi->phy_spuravoid = SPURAVOID_AUTO;
357 }
358 pi->radiopwr = 0xffff;
359 for (i = 0; i < STATIC_NUM_RF; i++) {
360 for (j = 0; j < STATIC_NUM_BB; j++)
361 pi->stats_11b_txpower[i][j] = -1;
362 }
363}
364
365struct shared_phy *wlc_phy_shared_attach(struct shared_phy_params *shp)
366{
367 struct shared_phy *sh;
368
369 sh = kzalloc(sizeof(struct shared_phy), GFP_ATOMIC);
370 if (sh == NULL)
371 return NULL;
372
373 sh->physhim = shp->physhim;
374 sh->unit = shp->unit;
375 sh->corerev = shp->corerev;
376
377 sh->vid = shp->vid;
378 sh->did = shp->did;
379 sh->chip = shp->chip;
380 sh->chiprev = shp->chiprev;
381 sh->chippkg = shp->chippkg;
382 sh->sromrev = shp->sromrev;
383 sh->boardtype = shp->boardtype;
384 sh->boardrev = shp->boardrev;
385 sh->boardflags = shp->boardflags;
386 sh->boardflags2 = shp->boardflags2;
387
388 sh->fast_timer = PHY_SW_TIMER_FAST;
389 sh->slow_timer = PHY_SW_TIMER_SLOW;
390 sh->glacial_timer = PHY_SW_TIMER_GLACIAL;
391
392 sh->rssi_mode = RSSI_ANT_MERGE_MAX;
393
394 return sh;
395}
396
397static void wlc_phy_timercb_phycal(struct brcms_phy *pi)
398{
399 uint delay = 5;
400
401 if (PHY_PERICAL_MPHASE_PENDING(pi)) {
402 if (!pi->sh->up) {
403 wlc_phy_cal_perical_mphase_reset(pi);
404 return;
405 }
406
407 if (SCAN_RM_IN_PROGRESS(pi) || PLT_INPROG_PHY(pi)) {
408
409 delay = 1000;
410 wlc_phy_cal_perical_mphase_restart(pi);
411 } else
412 wlc_phy_cal_perical_nphy_run(pi, PHY_PERICAL_AUTO);
413 wlapi_add_timer(pi->phycal_timer, delay, 0);
414 return;
415 }
416
417}
418
419static u32 wlc_phy_get_radio_ver(struct brcms_phy *pi)
420{
421 u32 ver;
422
423 ver = read_radio_id(pi);
424
425 return ver;
426}
427
428struct brcms_phy_pub *
429wlc_phy_attach(struct shared_phy *sh, struct bcma_device *d11core,
430 int bandtype, struct wiphy *wiphy)
431{
432 struct brcms_phy *pi;
433 u32 sflags = 0;
434 uint phyversion;
435 u32 idcode;
436 int i;
437
438 if (D11REV_IS(sh->corerev, 4))
439 sflags = SISF_2G_PHY | SISF_5G_PHY;
440 else
441 sflags = bcma_aread32(d11core, BCMA_IOST);
442
443 if (bandtype == BRCM_BAND_5G) {
444 if ((sflags & (SISF_5G_PHY | SISF_DB_PHY)) == 0)
445 return NULL;
446 }
447
448 pi = sh->phy_head;
449 if ((sflags & SISF_DB_PHY) && pi) {
450 wlapi_bmac_corereset(pi->sh->physhim, pi->pubpi.coreflags);
451 pi->refcnt++;
452 return &pi->pubpi_ro;
453 }
454
455 pi = kzalloc(sizeof(struct brcms_phy), GFP_ATOMIC);
456 if (pi == NULL)
457 return NULL;
458 pi->wiphy = wiphy;
459 pi->d11core = d11core;
460 pi->sh = sh;
461 pi->phy_init_por = true;
462 pi->phy_wreg_limit = PHY_WREG_LIMIT;
463
464 pi->txpwr_percent = 100;
465
466 pi->do_initcal = true;
467
468 pi->phycal_tempdelta = 0;
469
470 if (bandtype == BRCM_BAND_2G && (sflags & SISF_2G_PHY))
471 pi->pubpi.coreflags = SICF_GMODE;
472
473 wlapi_bmac_corereset(pi->sh->physhim, pi->pubpi.coreflags);
474 phyversion = bcma_read16(pi->d11core, D11REGOFFS(phyversion));
475
476 pi->pubpi.phy_type = PHY_TYPE(phyversion);
477 pi->pubpi.phy_rev = phyversion & PV_PV_MASK;
478
479 if (pi->pubpi.phy_type == PHY_TYPE_LCNXN) {
480 pi->pubpi.phy_type = PHY_TYPE_N;
481 pi->pubpi.phy_rev += LCNXN_BASEREV;
482 }
483 pi->pubpi.phy_corenum = PHY_CORE_NUM_2;
484 pi->pubpi.ana_rev = (phyversion & PV_AV_MASK) >> PV_AV_SHIFT;
485
486 if (pi->pubpi.phy_type != PHY_TYPE_N &&
487 pi->pubpi.phy_type != PHY_TYPE_LCN)
488 goto err;
489
490 if (bandtype == BRCM_BAND_5G) {
491 if (!ISNPHY(pi))
492 goto err;
493 } else if (!ISNPHY(pi) && !ISLCNPHY(pi)) {
494 goto err;
495 }
496
497 wlc_phy_anacore((struct brcms_phy_pub *) pi, ON);
498
499 idcode = wlc_phy_get_radio_ver(pi);
500 pi->pubpi.radioid =
501 (idcode & IDCODE_ID_MASK) >> IDCODE_ID_SHIFT;
502 pi->pubpi.radiorev =
503 (idcode & IDCODE_REV_MASK) >> IDCODE_REV_SHIFT;
504 pi->pubpi.radiover =
505 (idcode & IDCODE_VER_MASK) >> IDCODE_VER_SHIFT;
506 if (!VALID_RADIO(pi, pi->pubpi.radioid))
507 goto err;
508
509 wlc_phy_switch_radio((struct brcms_phy_pub *) pi, OFF);
510
511 wlc_set_phy_uninitted(pi);
512
513 pi->bw = WL_CHANSPEC_BW_20;
514 pi->radio_chanspec = (bandtype == BRCM_BAND_2G) ?
515 ch20mhz_chspec(1) : ch20mhz_chspec(36);
516
517 pi->rxiq_samps = PHY_NOISE_SAMPLE_LOG_NUM_NPHY;
518 pi->rxiq_antsel = ANT_RX_DIV_DEF;
519
520 pi->watchdog_override = true;
521
522 pi->cal_type_override = PHY_PERICAL_AUTO;
523
524 pi->nphy_saved_noisevars.bufcount = 0;
525
526 if (ISNPHY(pi))
527 pi->min_txpower = PHY_TXPWR_MIN_NPHY;
528 else
529 pi->min_txpower = PHY_TXPWR_MIN;
530
531 pi->sh->phyrxchain = 0x3;
532
533 pi->rx2tx_biasentry = -1;
534
535 pi->phy_txcore_disable_temp = PHY_CHAIN_TX_DISABLE_TEMP;
536 pi->phy_txcore_enable_temp =
537 PHY_CHAIN_TX_DISABLE_TEMP - PHY_HYSTERESIS_DELTATEMP;
538 pi->phy_tempsense_offset = 0;
539 pi->phy_txcore_heatedup = false;
540
541 pi->nphy_lastcal_temp = -50;
542
543 pi->phynoise_polling = true;
544 if (ISNPHY(pi) || ISLCNPHY(pi))
545 pi->phynoise_polling = false;
546
547 for (i = 0; i < TXP_NUM_RATES; i++) {
548 pi->txpwr_limit[i] = BRCMS_TXPWR_MAX;
549 pi->txpwr_env_limit[i] = BRCMS_TXPWR_MAX;
550 pi->tx_user_target[i] = BRCMS_TXPWR_MAX;
551 }
552
553 pi->radiopwr_override = RADIOPWR_OVERRIDE_DEF;
554
555 pi->user_txpwr_at_rfport = false;
556
557 if (ISNPHY(pi)) {
558
559 pi->phycal_timer = wlapi_init_timer(pi->sh->physhim,
560 wlc_phy_timercb_phycal,
561 pi, "phycal");
562 if (!pi->phycal_timer)
563 goto err;
564
565 if (!wlc_phy_attach_nphy(pi))
566 goto err;
567
568 } else if (ISLCNPHY(pi)) {
569 if (!wlc_phy_attach_lcnphy(pi))
570 goto err;
571
572 }
573
574 pi->refcnt++;
575 pi->next = pi->sh->phy_head;
576 sh->phy_head = pi;
577
578 memcpy(&pi->pubpi_ro, &pi->pubpi, sizeof(struct brcms_phy_pub));
579
580 return &pi->pubpi_ro;
581
582err:
583 kfree(pi);
584 return NULL;
585}
586
587void wlc_phy_detach(struct brcms_phy_pub *pih)
588{
589 struct brcms_phy *pi = container_of(pih, struct brcms_phy, pubpi_ro);
590
591 if (pih) {
592 if (--pi->refcnt)
593 return;
594
595 if (pi->phycal_timer) {
596 wlapi_free_timer(pi->phycal_timer);
597 pi->phycal_timer = NULL;
598 }
599
600 if (pi->sh->phy_head == pi)
601 pi->sh->phy_head = pi->next;
602 else if (pi->sh->phy_head->next == pi)
603 pi->sh->phy_head->next = NULL;
604
605 if (pi->pi_fptr.detach)
606 (pi->pi_fptr.detach)(pi);
607
608 kfree(pi);
609 }
610}
611
612bool
613wlc_phy_get_phyversion(struct brcms_phy_pub *pih, u16 *phytype, u16 *phyrev,
614 u16 *radioid, u16 *radiover)
615{
616 struct brcms_phy *pi = container_of(pih, struct brcms_phy, pubpi_ro);
617 *phytype = (u16) pi->pubpi.phy_type;
618 *phyrev = (u16) pi->pubpi.phy_rev;
619 *radioid = pi->pubpi.radioid;
620 *radiover = pi->pubpi.radiorev;
621
622 return true;
623}
624
625bool wlc_phy_get_encore(struct brcms_phy_pub *pih)
626{
627 struct brcms_phy *pi = container_of(pih, struct brcms_phy, pubpi_ro);
628 return pi->pubpi.abgphy_encore;
629}
630
631u32 wlc_phy_get_coreflags(struct brcms_phy_pub *pih)
632{
633 struct brcms_phy *pi = container_of(pih, struct brcms_phy, pubpi_ro);
634 return pi->pubpi.coreflags;
635}
636
637void wlc_phy_anacore(struct brcms_phy_pub *pih, bool on)
638{
639 struct brcms_phy *pi = container_of(pih, struct brcms_phy, pubpi_ro);
640
641 if (ISNPHY(pi)) {
642 if (on) {
643 if (NREV_GE(pi->pubpi.phy_rev, 3)) {
644 write_phy_reg(pi, 0xa6, 0x0d);
645 write_phy_reg(pi, 0x8f, 0x0);
646 write_phy_reg(pi, 0xa7, 0x0d);
647 write_phy_reg(pi, 0xa5, 0x0);
648 } else {
649 write_phy_reg(pi, 0xa5, 0x0);
650 }
651 } else {
652 if (NREV_GE(pi->pubpi.phy_rev, 3)) {
653 write_phy_reg(pi, 0x8f, 0x07ff);
654 write_phy_reg(pi, 0xa6, 0x0fd);
655 write_phy_reg(pi, 0xa5, 0x07ff);
656 write_phy_reg(pi, 0xa7, 0x0fd);
657 } else {
658 write_phy_reg(pi, 0xa5, 0x7fff);
659 }
660 }
661 } else if (ISLCNPHY(pi)) {
662 if (on) {
663 and_phy_reg(pi, 0x43b,
664 ~((0x1 << 0) | (0x1 << 1) | (0x1 << 2)));
665 } else {
666 or_phy_reg(pi, 0x43c,
667 (0x1 << 0) | (0x1 << 1) | (0x1 << 2));
668 or_phy_reg(pi, 0x43b,
669 (0x1 << 0) | (0x1 << 1) | (0x1 << 2));
670 }
671 }
672}
673
674u32 wlc_phy_clk_bwbits(struct brcms_phy_pub *pih)
675{
676 struct brcms_phy *pi = container_of(pih, struct brcms_phy, pubpi_ro);
677
678 u32 phy_bw_clkbits = 0;
679
680 if (pi && (ISNPHY(pi) || ISLCNPHY(pi))) {
681 switch (pi->bw) {
682 case WL_CHANSPEC_BW_10:
683 phy_bw_clkbits = SICF_BW10;
684 break;
685 case WL_CHANSPEC_BW_20:
686 phy_bw_clkbits = SICF_BW20;
687 break;
688 case WL_CHANSPEC_BW_40:
689 phy_bw_clkbits = SICF_BW40;
690 break;
691 default:
692 break;
693 }
694 }
695
696 return phy_bw_clkbits;
697}
698
699void wlc_phy_por_inform(struct brcms_phy_pub *ppi)
700{
701 struct brcms_phy *pi = container_of(ppi, struct brcms_phy, pubpi_ro);
702
703 pi->phy_init_por = true;
704}
705
706void wlc_phy_edcrs_lock(struct brcms_phy_pub *pih, bool lock)
707{
708 struct brcms_phy *pi = container_of(pih, struct brcms_phy, pubpi_ro);
709
710 pi->edcrs_threshold_lock = lock;
711
712 write_phy_reg(pi, 0x22c, 0x46b);
713 write_phy_reg(pi, 0x22d, 0x46b);
714 write_phy_reg(pi, 0x22e, 0x3c0);
715 write_phy_reg(pi, 0x22f, 0x3c0);
716}
717
718void wlc_phy_initcal_enable(struct brcms_phy_pub *pih, bool initcal)
719{
720 struct brcms_phy *pi = container_of(pih, struct brcms_phy, pubpi_ro);
721
722 pi->do_initcal = initcal;
723}
724
725void wlc_phy_hw_clk_state_upd(struct brcms_phy_pub *pih, bool newstate)
726{
727 struct brcms_phy *pi = container_of(pih, struct brcms_phy, pubpi_ro);
728
729 if (!pi || !pi->sh)
730 return;
731
732 pi->sh->clk = newstate;
733}
734
735void wlc_phy_hw_state_upd(struct brcms_phy_pub *pih, bool newstate)
736{
737 struct brcms_phy *pi = container_of(pih, struct brcms_phy, pubpi_ro);
738
739 if (!pi || !pi->sh)
740 return;
741
742 pi->sh->up = newstate;
743}
744
745void wlc_phy_init(struct brcms_phy_pub *pih, u16 chanspec)
746{
747 u32 mc;
748 void (*phy_init)(struct brcms_phy *) = NULL;
749 struct brcms_phy *pi = container_of(pih, struct brcms_phy, pubpi_ro);
750
751 if (pi->init_in_progress)
752 return;
753
754 pi->init_in_progress = true;
755
756 pi->radio_chanspec = chanspec;
757
758 mc = bcma_read32(pi->d11core, D11REGOFFS(maccontrol));
759 if (WARN(mc & MCTL_EN_MAC, "HW error MAC running on init"))
760 return;
761
762 if (!(pi->measure_hold & PHY_HOLD_FOR_SCAN))
763 pi->measure_hold |= PHY_HOLD_FOR_NOT_ASSOC;
764
765 if (WARN(!(bcma_aread32(pi->d11core, BCMA_IOST) & SISF_FCLKA),
766 "HW error SISF_FCLKA\n"))
767 return;
768
769 phy_init = pi->pi_fptr.init;
770
771 if (phy_init == NULL)
772 return;
773
774 wlc_phy_anacore(pih, ON);
775
776 if (CHSPEC_BW(pi->radio_chanspec) != pi->bw)
777 wlapi_bmac_bw_set(pi->sh->physhim,
778 CHSPEC_BW(pi->radio_chanspec));
779
780 pi->nphy_gain_boost = true;
781
782 wlc_phy_switch_radio((struct brcms_phy_pub *) pi, ON);
783
784 (*phy_init)(pi);
785
786 pi->phy_init_por = false;
787
788 if (D11REV_IS(pi->sh->corerev, 11) || D11REV_IS(pi->sh->corerev, 12))
789 wlc_phy_do_dummy_tx(pi, true, OFF);
790
791 if (!(ISNPHY(pi)))
792 wlc_phy_txpower_update_shm(pi);
793
794 wlc_phy_ant_rxdiv_set((struct brcms_phy_pub *) pi, pi->sh->rx_antdiv);
795
796 pi->init_in_progress = false;
797}
798
799void wlc_phy_cal_init(struct brcms_phy_pub *pih)
800{
801 struct brcms_phy *pi = container_of(pih, struct brcms_phy, pubpi_ro);
802 void (*cal_init)(struct brcms_phy *) = NULL;
803
804 if (WARN((bcma_read32(pi->d11core, D11REGOFFS(maccontrol)) &
805 MCTL_EN_MAC) != 0, "HW error: MAC enabled during phy cal\n"))
806 return;
807
808 if (!pi->initialized) {
809 cal_init = pi->pi_fptr.calinit;
810 if (cal_init)
811 (*cal_init)(pi);
812
813 pi->initialized = true;
814 }
815}
816
817int wlc_phy_down(struct brcms_phy_pub *pih)
818{
819 struct brcms_phy *pi = container_of(pih, struct brcms_phy, pubpi_ro);
820 int callbacks = 0;
821
822 if (pi->phycal_timer
823 && !wlapi_del_timer(pi->phycal_timer))
824 callbacks++;
825
826 pi->nphy_iqcal_chanspec_2G = 0;
827 pi->nphy_iqcal_chanspec_5G = 0;
828
829 return callbacks;
830}
831
832void
833wlc_phy_table_addr(struct brcms_phy *pi, uint tbl_id, uint tbl_offset,
834 u16 tblAddr, u16 tblDataHi, u16 tblDataLo)
835{
836 write_phy_reg(pi, tblAddr, (tbl_id << 10) | tbl_offset);
837
838 pi->tbl_data_hi = tblDataHi;
839 pi->tbl_data_lo = tblDataLo;
840
841 if (pi->sh->chip == BCMA_CHIP_ID_BCM43224 &&
842 pi->sh->chiprev == 1) {
843 pi->tbl_addr = tblAddr;
844 pi->tbl_save_id = tbl_id;
845 pi->tbl_save_offset = tbl_offset;
846 }
847}
848
849void wlc_phy_table_data_write(struct brcms_phy *pi, uint width, u32 val)
850{
851 if ((pi->sh->chip == BCMA_CHIP_ID_BCM43224) &&
852 (pi->sh->chiprev == 1) &&
853 (pi->tbl_save_id == NPHY_TBL_ID_ANTSWCTRLLUT)) {
854 read_phy_reg(pi, pi->tbl_data_lo);
855
856 write_phy_reg(pi, pi->tbl_addr,
857 (pi->tbl_save_id << 10) | pi->tbl_save_offset);
858 pi->tbl_save_offset++;
859 }
860
861 if (width == 32) {
862 write_phy_reg(pi, pi->tbl_data_hi, (u16) (val >> 16));
863 write_phy_reg(pi, pi->tbl_data_lo, (u16) val);
864 } else {
865 write_phy_reg(pi, pi->tbl_data_lo, (u16) val);
866 }
867}
868
869void
870wlc_phy_write_table(struct brcms_phy *pi, const struct phytbl_info *ptbl_info,
871 u16 tblAddr, u16 tblDataHi, u16 tblDataLo)
872{
873 uint idx;
874 uint tbl_id = ptbl_info->tbl_id;
875 uint tbl_offset = ptbl_info->tbl_offset;
876 uint tbl_width = ptbl_info->tbl_width;
877 const u8 *ptbl_8b = (const u8 *)ptbl_info->tbl_ptr;
878 const u16 *ptbl_16b = (const u16 *)ptbl_info->tbl_ptr;
879 const u32 *ptbl_32b = (const u32 *)ptbl_info->tbl_ptr;
880
881 write_phy_reg(pi, tblAddr, (tbl_id << 10) | tbl_offset);
882
883 for (idx = 0; idx < ptbl_info->tbl_len; idx++) {
884
885 if ((pi->sh->chip == BCMA_CHIP_ID_BCM43224) &&
886 (pi->sh->chiprev == 1) &&
887 (tbl_id == NPHY_TBL_ID_ANTSWCTRLLUT)) {
888 read_phy_reg(pi, tblDataLo);
889
890 write_phy_reg(pi, tblAddr,
891 (tbl_id << 10) | (tbl_offset + idx));
892 }
893
894 if (tbl_width == 32) {
895 write_phy_reg(pi, tblDataHi,
896 (u16) (ptbl_32b[idx] >> 16));
897 write_phy_reg(pi, tblDataLo, (u16) ptbl_32b[idx]);
898 } else if (tbl_width == 16) {
899 write_phy_reg(pi, tblDataLo, ptbl_16b[idx]);
900 } else {
901 write_phy_reg(pi, tblDataLo, ptbl_8b[idx]);
902 }
903 }
904}
905
906void
907wlc_phy_read_table(struct brcms_phy *pi, const struct phytbl_info *ptbl_info,
908 u16 tblAddr, u16 tblDataHi, u16 tblDataLo)
909{
910 uint idx;
911 uint tbl_id = ptbl_info->tbl_id;
912 uint tbl_offset = ptbl_info->tbl_offset;
913 uint tbl_width = ptbl_info->tbl_width;
914 u8 *ptbl_8b = (u8 *)ptbl_info->tbl_ptr;
915 u16 *ptbl_16b = (u16 *)ptbl_info->tbl_ptr;
916 u32 *ptbl_32b = (u32 *)ptbl_info->tbl_ptr;
917
918 write_phy_reg(pi, tblAddr, (tbl_id << 10) | tbl_offset);
919
920 for (idx = 0; idx < ptbl_info->tbl_len; idx++) {
921
922 if ((pi->sh->chip == BCMA_CHIP_ID_BCM43224) &&
923 (pi->sh->chiprev == 1)) {
924 (void)read_phy_reg(pi, tblDataLo);
925
926 write_phy_reg(pi, tblAddr,
927 (tbl_id << 10) | (tbl_offset + idx));
928 }
929
930 if (tbl_width == 32) {
931 ptbl_32b[idx] = read_phy_reg(pi, tblDataLo);
932 ptbl_32b[idx] |= (read_phy_reg(pi, tblDataHi) << 16);
933 } else if (tbl_width == 16) {
934 ptbl_16b[idx] = read_phy_reg(pi, tblDataLo);
935 } else {
936 ptbl_8b[idx] = (u8) read_phy_reg(pi, tblDataLo);
937 }
938 }
939}
940
941uint
942wlc_phy_init_radio_regs_allbands(struct brcms_phy *pi,
943 struct radio_20xx_regs *radioregs)
944{
945 uint i = 0;
946
947 do {
948 if (radioregs[i].do_init)
949 write_radio_reg(pi, radioregs[i].address,
950 (u16) radioregs[i].init);
951
952 i++;
953 } while (radioregs[i].address != 0xffff);
954
955 return i;
956}
957
958uint
959wlc_phy_init_radio_regs(struct brcms_phy *pi,
960 const struct radio_regs *radioregs,
961 u16 core_offset)
962{
963 uint i = 0;
964 uint count = 0;
965
966 do {
967 if (CHSPEC_IS5G(pi->radio_chanspec)) {
968 if (radioregs[i].do_init_a) {
969 write_radio_reg(pi,
970 radioregs[i].
971 address | core_offset,
972 (u16) radioregs[i].init_a);
973 if (ISNPHY(pi) && (++count % 4 == 0))
974 BRCMS_PHY_WAR_PR51571(pi);
975 }
976 } else {
977 if (radioregs[i].do_init_g) {
978 write_radio_reg(pi,
979 radioregs[i].
980 address | core_offset,
981 (u16) radioregs[i].init_g);
982 if (ISNPHY(pi) && (++count % 4 == 0))
983 BRCMS_PHY_WAR_PR51571(pi);
984 }
985 }
986
987 i++;
988 } while (radioregs[i].address != 0xffff);
989
990 return i;
991}
992
993void wlc_phy_do_dummy_tx(struct brcms_phy *pi, bool ofdm, bool pa_on)
994{
995#define DUMMY_PKT_LEN 20
996 struct bcma_device *core = pi->d11core;
997 int i, count;
998 u8 ofdmpkt[DUMMY_PKT_LEN] = {
999 0xcc, 0x01, 0x02, 0x00, 0x00, 0x00, 0xd4, 0x00, 0x00, 0x00,
1000 0x00, 0x00, 0x00, 0x00, 0x00, 0x01, 0x00, 0x00, 0x00, 0x00
1001 };
1002 u8 cckpkt[DUMMY_PKT_LEN] = {
1003 0x6e, 0x84, 0x0b, 0x00, 0x00, 0x00, 0xd4, 0x00, 0x00, 0x00,
1004 0x00, 0x00, 0x00, 0x00, 0x00, 0x01, 0x00, 0x00, 0x00, 0x00
1005 };
1006 u32 *dummypkt;
1007
1008 dummypkt = (u32 *) (ofdm ? ofdmpkt : cckpkt);
1009 wlapi_bmac_write_template_ram(pi->sh->physhim, 0, DUMMY_PKT_LEN,
1010 dummypkt);
1011
1012 bcma_write16(core, D11REGOFFS(xmtsel), 0);
1013
1014 if (D11REV_GE(pi->sh->corerev, 11))
1015 bcma_write16(core, D11REGOFFS(wepctl), 0x100);
1016 else
1017 bcma_write16(core, D11REGOFFS(wepctl), 0);
1018
1019 bcma_write16(core, D11REGOFFS(txe_phyctl),
1020 (ofdm ? 1 : 0) | PHY_TXC_ANT_0);
1021 if (ISNPHY(pi) || ISLCNPHY(pi))
1022 bcma_write16(core, D11REGOFFS(txe_phyctl1), 0x1A02);
1023
1024 bcma_write16(core, D11REGOFFS(txe_wm_0), 0);
1025 bcma_write16(core, D11REGOFFS(txe_wm_1), 0);
1026
1027 bcma_write16(core, D11REGOFFS(xmttplatetxptr), 0);
1028 bcma_write16(core, D11REGOFFS(xmttxcnt), DUMMY_PKT_LEN);
1029
1030 bcma_write16(core, D11REGOFFS(xmtsel),
1031 ((8 << 8) | (1 << 5) | (1 << 2) | 2));
1032
1033 bcma_write16(core, D11REGOFFS(txe_ctl), 0);
1034
1035 if (!pa_on) {
1036 if (ISNPHY(pi))
1037 wlc_phy_pa_override_nphy(pi, OFF);
1038 }
1039
1040 if (ISNPHY(pi) || ISLCNPHY(pi))
1041 bcma_write16(core, D11REGOFFS(txe_aux), 0xD0);
1042 else
1043 bcma_write16(core, D11REGOFFS(txe_aux), ((1 << 5) | (1 << 4)));
1044
1045 (void)bcma_read16(core, D11REGOFFS(txe_aux));
1046
1047 i = 0;
1048 count = ofdm ? 30 : 250;
1049 while ((i++ < count)
1050 && (bcma_read16(core, D11REGOFFS(txe_status)) & (1 << 7)))
1051 udelay(10);
1052
1053 i = 0;
1054
1055 while ((i++ < 10) &&
1056 ((bcma_read16(core, D11REGOFFS(txe_status)) & (1 << 10)) == 0))
1057 udelay(10);
1058
1059 i = 0;
1060
1061 while ((i++ < 10) &&
1062 ((bcma_read16(core, D11REGOFFS(ifsstat)) & (1 << 8))))
1063 udelay(10);
1064
1065 if (!pa_on) {
1066 if (ISNPHY(pi))
1067 wlc_phy_pa_override_nphy(pi, ON);
1068 }
1069}
1070
1071void wlc_phy_hold_upd(struct brcms_phy_pub *pih, u32 id, bool set)
1072{
1073 struct brcms_phy *pi = container_of(pih, struct brcms_phy, pubpi_ro);
1074
1075 if (set)
1076 mboolset(pi->measure_hold, id);
1077 else
1078 mboolclr(pi->measure_hold, id);
1079
1080 return;
1081}
1082
1083void wlc_phy_mute_upd(struct brcms_phy_pub *pih, bool mute, u32 flags)
1084{
1085 struct brcms_phy *pi = container_of(pih, struct brcms_phy, pubpi_ro);
1086
1087 if (mute)
1088 mboolset(pi->measure_hold, PHY_HOLD_FOR_MUTE);
1089 else
1090 mboolclr(pi->measure_hold, PHY_HOLD_FOR_MUTE);
1091
1092 if (!mute && (flags & PHY_MUTE_FOR_PREISM))
1093 pi->nphy_perical_last = pi->sh->now - pi->sh->glacial_timer;
1094 return;
1095}
1096
1097void wlc_phy_clear_tssi(struct brcms_phy_pub *pih)
1098{
1099 struct brcms_phy *pi = container_of(pih, struct brcms_phy, pubpi_ro);
1100
1101 if (ISNPHY(pi)) {
1102 return;
1103 } else {
1104 wlapi_bmac_write_shm(pi->sh->physhim, M_B_TSSI_0, NULL_TSSI_W);
1105 wlapi_bmac_write_shm(pi->sh->physhim, M_B_TSSI_1, NULL_TSSI_W);
1106 wlapi_bmac_write_shm(pi->sh->physhim, M_G_TSSI_0, NULL_TSSI_W);
1107 wlapi_bmac_write_shm(pi->sh->physhim, M_G_TSSI_1, NULL_TSSI_W);
1108 }
1109}
1110
1111static bool wlc_phy_cal_txpower_recalc_sw(struct brcms_phy *pi)
1112{
1113 return false;
1114}
1115
1116void wlc_phy_switch_radio(struct brcms_phy_pub *pih, bool on)
1117{
1118 struct brcms_phy *pi = container_of(pih, struct brcms_phy, pubpi_ro);
1119 (void)bcma_read32(pi->d11core, D11REGOFFS(maccontrol));
1120
1121 if (ISNPHY(pi)) {
1122 wlc_phy_switch_radio_nphy(pi, on);
1123 } else if (ISLCNPHY(pi)) {
1124 if (on) {
1125 and_phy_reg(pi, 0x44c,
1126 ~((0x1 << 8) |
1127 (0x1 << 9) |
1128 (0x1 << 10) | (0x1 << 11) | (0x1 << 12)));
1129 and_phy_reg(pi, 0x4b0, ~((0x1 << 3) | (0x1 << 11)));
1130 and_phy_reg(pi, 0x4f9, ~(0x1 << 3));
1131 } else {
1132 and_phy_reg(pi, 0x44d,
1133 ~((0x1 << 10) |
1134 (0x1 << 11) |
1135 (0x1 << 12) | (0x1 << 13) | (0x1 << 14)));
1136 or_phy_reg(pi, 0x44c,
1137 (0x1 << 8) |
1138 (0x1 << 9) |
1139 (0x1 << 10) | (0x1 << 11) | (0x1 << 12));
1140
1141 and_phy_reg(pi, 0x4b7, ~((0x7f << 8)));
1142 and_phy_reg(pi, 0x4b1, ~((0x1 << 13)));
1143 or_phy_reg(pi, 0x4b0, (0x1 << 3) | (0x1 << 11));
1144 and_phy_reg(pi, 0x4fa, ~((0x1 << 3)));
1145 or_phy_reg(pi, 0x4f9, (0x1 << 3));
1146 }
1147 }
1148}
1149
1150u16 wlc_phy_bw_state_get(struct brcms_phy_pub *ppi)
1151{
1152 struct brcms_phy *pi = container_of(ppi, struct brcms_phy, pubpi_ro);
1153
1154 return pi->bw;
1155}
1156
1157void wlc_phy_bw_state_set(struct brcms_phy_pub *ppi, u16 bw)
1158{
1159 struct brcms_phy *pi = container_of(ppi, struct brcms_phy, pubpi_ro);
1160
1161 pi->bw = bw;
1162}
1163
1164void wlc_phy_chanspec_radio_set(struct brcms_phy_pub *ppi, u16 newch)
1165{
1166 struct brcms_phy *pi = container_of(ppi, struct brcms_phy, pubpi_ro);
1167 pi->radio_chanspec = newch;
1168
1169}
1170
1171u16 wlc_phy_chanspec_get(struct brcms_phy_pub *ppi)
1172{
1173 struct brcms_phy *pi = container_of(ppi, struct brcms_phy, pubpi_ro);
1174
1175 return pi->radio_chanspec;
1176}
1177
1178void wlc_phy_chanspec_set(struct brcms_phy_pub *ppi, u16 chanspec)
1179{
1180 struct brcms_phy *pi = container_of(ppi, struct brcms_phy, pubpi_ro);
1181 u16 m_cur_channel;
1182 void (*chanspec_set)(struct brcms_phy *, u16) = NULL;
1183 m_cur_channel = CHSPEC_CHANNEL(chanspec);
1184 if (CHSPEC_IS5G(chanspec))
1185 m_cur_channel |= D11_CURCHANNEL_5G;
1186 if (CHSPEC_IS40(chanspec))
1187 m_cur_channel |= D11_CURCHANNEL_40;
1188 wlapi_bmac_write_shm(pi->sh->physhim, M_CURCHANNEL, m_cur_channel);
1189
1190 chanspec_set = pi->pi_fptr.chanset;
1191 if (chanspec_set)
1192 (*chanspec_set)(pi, chanspec);
1193
1194}
1195
1196int wlc_phy_chanspec_freq2bandrange_lpssn(uint freq)
1197{
1198 int range = -1;
1199
1200 if (freq < 2500)
1201 range = WL_CHAN_FREQ_RANGE_2G;
1202 else if (freq <= 5320)
1203 range = WL_CHAN_FREQ_RANGE_5GL;
1204 else if (freq <= 5700)
1205 range = WL_CHAN_FREQ_RANGE_5GM;
1206 else
1207 range = WL_CHAN_FREQ_RANGE_5GH;
1208
1209 return range;
1210}
1211
1212int wlc_phy_chanspec_bandrange_get(struct brcms_phy *pi, u16 chanspec)
1213{
1214 int range = -1;
1215 uint channel = CHSPEC_CHANNEL(chanspec);
1216 uint freq = wlc_phy_channel2freq(channel);
1217
1218 if (ISNPHY(pi))
1219 range = wlc_phy_get_chan_freq_range_nphy(pi, channel);
1220 else if (ISLCNPHY(pi))
1221 range = wlc_phy_chanspec_freq2bandrange_lpssn(freq);
1222
1223 return range;
1224}
1225
1226void wlc_phy_chanspec_ch14_widefilter_set(struct brcms_phy_pub *ppi,
1227 bool wide_filter)
1228{
1229 struct brcms_phy *pi = container_of(ppi, struct brcms_phy, pubpi_ro);
1230
1231 pi->channel_14_wide_filter = wide_filter;
1232
1233}
1234
1235int wlc_phy_channel2freq(uint channel)
1236{
1237 uint i;
1238
1239 for (i = 0; i < ARRAY_SIZE(chan_info_all); i++)
1240 if (chan_info_all[i].chan == channel)
1241 return chan_info_all[i].freq;
1242 return 0;
1243}
1244
1245void
1246wlc_phy_chanspec_band_validch(struct brcms_phy_pub *ppi, uint band,
1247 struct brcms_chanvec *channels)
1248{
1249 struct brcms_phy *pi = container_of(ppi, struct brcms_phy, pubpi_ro);
1250 uint i;
1251 uint channel;
1252
1253 memset(channels, 0, sizeof(struct brcms_chanvec));
1254
1255 for (i = 0; i < ARRAY_SIZE(chan_info_all); i++) {
1256 channel = chan_info_all[i].chan;
1257
1258 if ((pi->a_band_high_disable) && (channel >= FIRST_REF5_CHANNUM)
1259 && (channel <= LAST_REF5_CHANNUM))
1260 continue;
1261
1262 if ((band == BRCM_BAND_2G && channel <= CH_MAX_2G_CHANNEL) ||
1263 (band == BRCM_BAND_5G && channel > CH_MAX_2G_CHANNEL))
1264 setbit(channels->vec, channel);
1265 }
1266}
1267
1268u16 wlc_phy_chanspec_band_firstch(struct brcms_phy_pub *ppi, uint band)
1269{
1270 struct brcms_phy *pi = container_of(ppi, struct brcms_phy, pubpi_ro);
1271 uint i;
1272 uint channel;
1273 u16 chspec;
1274
1275 for (i = 0; i < ARRAY_SIZE(chan_info_all); i++) {
1276 channel = chan_info_all[i].chan;
1277
1278 if (ISNPHY(pi) && pi->bw == WL_CHANSPEC_BW_40) {
1279 uint j;
1280
1281 for (j = 0; j < ARRAY_SIZE(chan_info_all); j++) {
1282 if (chan_info_all[j].chan ==
1283 channel + CH_10MHZ_APART)
1284 break;
1285 }
1286
1287 if (j == ARRAY_SIZE(chan_info_all))
1288 continue;
1289
1290 channel = upper_20_sb(channel);
1291 chspec = channel | WL_CHANSPEC_BW_40 |
1292 WL_CHANSPEC_CTL_SB_LOWER;
1293 if (band == BRCM_BAND_2G)
1294 chspec |= WL_CHANSPEC_BAND_2G;
1295 else
1296 chspec |= WL_CHANSPEC_BAND_5G;
1297 } else
1298 chspec = ch20mhz_chspec(channel);
1299
1300 if ((pi->a_band_high_disable) && (channel >= FIRST_REF5_CHANNUM)
1301 && (channel <= LAST_REF5_CHANNUM))
1302 continue;
1303
1304 if ((band == BRCM_BAND_2G && channel <= CH_MAX_2G_CHANNEL) ||
1305 (band == BRCM_BAND_5G && channel > CH_MAX_2G_CHANNEL))
1306 return chspec;
1307 }
1308
1309 return (u16) INVCHANSPEC;
1310}
1311
1312int wlc_phy_txpower_get(struct brcms_phy_pub *ppi, uint *qdbm, bool *override)
1313{
1314 struct brcms_phy *pi = container_of(ppi, struct brcms_phy, pubpi_ro);
1315
1316 *qdbm = pi->tx_user_target[0];
1317 if (override != NULL)
1318 *override = pi->txpwroverride;
1319 return 0;
1320}
1321
1322void wlc_phy_txpower_target_set(struct brcms_phy_pub *ppi,
1323 struct txpwr_limits *txpwr)
1324{
1325 bool mac_enabled = false;
1326 struct brcms_phy *pi = container_of(ppi, struct brcms_phy, pubpi_ro);
1327
1328 memcpy(&pi->tx_user_target[TXP_FIRST_CCK],
1329 &txpwr->cck[0], BRCMS_NUM_RATES_CCK);
1330
1331 memcpy(&pi->tx_user_target[TXP_FIRST_OFDM],
1332 &txpwr->ofdm[0], BRCMS_NUM_RATES_OFDM);
1333 memcpy(&pi->tx_user_target[TXP_FIRST_OFDM_20_CDD],
1334 &txpwr->ofdm_cdd[0], BRCMS_NUM_RATES_OFDM);
1335
1336 memcpy(&pi->tx_user_target[TXP_FIRST_OFDM_40_SISO],
1337 &txpwr->ofdm_40_siso[0], BRCMS_NUM_RATES_OFDM);
1338 memcpy(&pi->tx_user_target[TXP_FIRST_OFDM_40_CDD],
1339 &txpwr->ofdm_40_cdd[0], BRCMS_NUM_RATES_OFDM);
1340
1341 memcpy(&pi->tx_user_target[TXP_FIRST_MCS_20_SISO],
1342 &txpwr->mcs_20_siso[0], BRCMS_NUM_RATES_MCS_1_STREAM);
1343 memcpy(&pi->tx_user_target[TXP_FIRST_MCS_20_CDD],
1344 &txpwr->mcs_20_cdd[0], BRCMS_NUM_RATES_MCS_1_STREAM);
1345 memcpy(&pi->tx_user_target[TXP_FIRST_MCS_20_STBC],
1346 &txpwr->mcs_20_stbc[0], BRCMS_NUM_RATES_MCS_1_STREAM);
1347 memcpy(&pi->tx_user_target[TXP_FIRST_MCS_20_SDM],
1348 &txpwr->mcs_20_mimo[0], BRCMS_NUM_RATES_MCS_2_STREAM);
1349
1350 memcpy(&pi->tx_user_target[TXP_FIRST_MCS_40_SISO],
1351 &txpwr->mcs_40_siso[0], BRCMS_NUM_RATES_MCS_1_STREAM);
1352 memcpy(&pi->tx_user_target[TXP_FIRST_MCS_40_CDD],
1353 &txpwr->mcs_40_cdd[0], BRCMS_NUM_RATES_MCS_1_STREAM);
1354 memcpy(&pi->tx_user_target[TXP_FIRST_MCS_40_STBC],
1355 &txpwr->mcs_40_stbc[0], BRCMS_NUM_RATES_MCS_1_STREAM);
1356 memcpy(&pi->tx_user_target[TXP_FIRST_MCS_40_SDM],
1357 &txpwr->mcs_40_mimo[0], BRCMS_NUM_RATES_MCS_2_STREAM);
1358
1359 if (bcma_read32(pi->d11core, D11REGOFFS(maccontrol)) & MCTL_EN_MAC)
1360 mac_enabled = true;
1361
1362 if (mac_enabled)
1363 wlapi_suspend_mac_and_wait(pi->sh->physhim);
1364
1365 wlc_phy_txpower_recalc_target(pi);
1366 wlc_phy_cal_txpower_recalc_sw(pi);
1367
1368 if (mac_enabled)
1369 wlapi_enable_mac(pi->sh->physhim);
1370}
1371
1372int wlc_phy_txpower_set(struct brcms_phy_pub *ppi, uint qdbm, bool override)
1373{
1374 struct brcms_phy *pi = container_of(ppi, struct brcms_phy, pubpi_ro);
1375 int i;
1376
1377 if (qdbm > 127)
1378 return -EINVAL;
1379
1380 for (i = 0; i < TXP_NUM_RATES; i++)
1381 pi->tx_user_target[i] = (u8) qdbm;
1382
1383 pi->txpwroverride = false;
1384
1385 if (pi->sh->up) {
1386 if (!SCAN_INPROG_PHY(pi)) {
1387 bool suspend;
1388
1389 suspend = (0 == (bcma_read32(pi->d11core,
1390 D11REGOFFS(maccontrol)) &
1391 MCTL_EN_MAC));
1392
1393 if (!suspend)
1394 wlapi_suspend_mac_and_wait(pi->sh->physhim);
1395
1396 wlc_phy_txpower_recalc_target(pi);
1397 wlc_phy_cal_txpower_recalc_sw(pi);
1398
1399 if (!suspend)
1400 wlapi_enable_mac(pi->sh->physhim);
1401 }
1402 }
1403 return 0;
1404}
1405
1406void
1407wlc_phy_txpower_sromlimit(struct brcms_phy_pub *ppi, uint channel, u8 *min_pwr,
1408 u8 *max_pwr, int txp_rate_idx)
1409{
1410 struct brcms_phy *pi = container_of(ppi, struct brcms_phy, pubpi_ro);
1411 uint i;
1412
1413 *min_pwr = pi->min_txpower * BRCMS_TXPWR_DB_FACTOR;
1414
1415 if (ISNPHY(pi)) {
1416 if (txp_rate_idx < 0)
1417 txp_rate_idx = TXP_FIRST_CCK;
1418 wlc_phy_txpower_sromlimit_get_nphy(pi, channel, max_pwr,
1419 (u8) txp_rate_idx);
1420
1421 } else if ((channel <= CH_MAX_2G_CHANNEL)) {
1422 if (txp_rate_idx < 0)
1423 txp_rate_idx = TXP_FIRST_CCK;
1424 *max_pwr = pi->tx_srom_max_rate_2g[txp_rate_idx];
1425 } else {
1426
1427 *max_pwr = BRCMS_TXPWR_MAX;
1428
1429 if (txp_rate_idx < 0)
1430 txp_rate_idx = TXP_FIRST_OFDM;
1431
1432 for (i = 0; i < ARRAY_SIZE(chan_info_all); i++) {
1433 if (channel == chan_info_all[i].chan)
1434 break;
1435 }
1436
1437 if (pi->hwtxpwr) {
1438 *max_pwr = pi->hwtxpwr[i];
1439 } else {
1440
1441 if ((i >= FIRST_MID_5G_CHAN) && (i <= LAST_MID_5G_CHAN))
1442 *max_pwr =
1443 pi->tx_srom_max_rate_5g_mid[txp_rate_idx];
1444 if ((i >= FIRST_HIGH_5G_CHAN)
1445 && (i <= LAST_HIGH_5G_CHAN))
1446 *max_pwr =
1447 pi->tx_srom_max_rate_5g_hi[txp_rate_idx];
1448 if ((i >= FIRST_LOW_5G_CHAN) && (i <= LAST_LOW_5G_CHAN))
1449 *max_pwr =
1450 pi->tx_srom_max_rate_5g_low[txp_rate_idx];
1451 }
1452 }
1453}
1454
1455void
1456wlc_phy_txpower_sromlimit_max_get(struct brcms_phy_pub *ppi, uint chan,
1457 u8 *max_txpwr, u8 *min_txpwr)
1458{
1459 struct brcms_phy *pi = container_of(ppi, struct brcms_phy, pubpi_ro);
1460 u8 tx_pwr_max = 0;
1461 u8 tx_pwr_min = 255;
1462 u8 max_num_rate;
1463 u8 maxtxpwr, mintxpwr, rate, pactrl;
1464
1465 pactrl = 0;
1466
1467 max_num_rate = ISNPHY(pi) ? TXP_NUM_RATES :
1468 ISLCNPHY(pi) ? (TXP_LAST_SISO_MCS_20 +
1469 1) : (TXP_LAST_OFDM + 1);
1470
1471 for (rate = 0; rate < max_num_rate; rate++) {
1472
1473 wlc_phy_txpower_sromlimit(ppi, chan, &mintxpwr, &maxtxpwr,
1474 rate);
1475
1476 maxtxpwr = (maxtxpwr > pactrl) ? (maxtxpwr - pactrl) : 0;
1477
1478 maxtxpwr = (maxtxpwr > 6) ? (maxtxpwr - 6) : 0;
1479
1480 tx_pwr_max = max(tx_pwr_max, maxtxpwr);
1481 tx_pwr_min = min(tx_pwr_min, maxtxpwr);
1482 }
1483 *max_txpwr = tx_pwr_max;
1484 *min_txpwr = tx_pwr_min;
1485}
1486
1487void
1488wlc_phy_txpower_boardlimit_band(struct brcms_phy_pub *ppi, uint bandunit,
1489 s32 *max_pwr, s32 *min_pwr, u32 *step_pwr)
1490{
1491 return;
1492}
1493
1494u8 wlc_phy_txpower_get_target_min(struct brcms_phy_pub *ppi)
1495{
1496 struct brcms_phy *pi = container_of(ppi, struct brcms_phy, pubpi_ro);
1497
1498 return pi->tx_power_min;
1499}
1500
1501u8 wlc_phy_txpower_get_target_max(struct brcms_phy_pub *ppi)
1502{
1503 struct brcms_phy *pi = container_of(ppi, struct brcms_phy, pubpi_ro);
1504
1505 return pi->tx_power_max;
1506}
1507
1508static s8 wlc_phy_env_measure_vbat(struct brcms_phy *pi)
1509{
1510 if (ISLCNPHY(pi))
1511 return wlc_lcnphy_vbatsense(pi, 0);
1512 else
1513 return 0;
1514}
1515
1516static s8 wlc_phy_env_measure_temperature(struct brcms_phy *pi)
1517{
1518 if (ISLCNPHY(pi))
1519 return wlc_lcnphy_tempsense_degree(pi, 0);
1520 else
1521 return 0;
1522}
1523
1524static void wlc_phy_upd_env_txpwr_rate_limits(struct brcms_phy *pi, u32 band)
1525{
1526 u8 i;
1527 s8 temp, vbat;
1528
1529 for (i = 0; i < TXP_NUM_RATES; i++)
1530 pi->txpwr_env_limit[i] = BRCMS_TXPWR_MAX;
1531
1532 vbat = wlc_phy_env_measure_vbat(pi);
1533 temp = wlc_phy_env_measure_temperature(pi);
1534
1535}
1536
1537static s8
1538wlc_user_txpwr_antport_to_rfport(struct brcms_phy *pi, uint chan, u32 band,
1539 u8 rate)
1540{
1541 return 0;
1542}
1543
1544void wlc_phy_txpower_recalc_target(struct brcms_phy *pi)
1545{
1546 u8 maxtxpwr, mintxpwr, rate, pactrl;
1547 uint target_chan;
1548 u8 tx_pwr_target[TXP_NUM_RATES];
1549 u8 tx_pwr_max = 0;
1550 u8 tx_pwr_min = 255;
1551 u8 tx_pwr_max_rate_ind = 0;
1552 u8 max_num_rate;
1553 u8 start_rate = 0;
1554 u16 chspec;
1555 u32 band = CHSPEC2BAND(pi->radio_chanspec);
1556 void (*txpwr_recalc_fn)(struct brcms_phy *) = NULL;
1557
1558 chspec = pi->radio_chanspec;
1559 if (CHSPEC_CTL_SB(chspec) == WL_CHANSPEC_CTL_SB_NONE)
1560 target_chan = CHSPEC_CHANNEL(chspec);
1561 else if (CHSPEC_CTL_SB(chspec) == WL_CHANSPEC_CTL_SB_UPPER)
1562 target_chan = upper_20_sb(CHSPEC_CHANNEL(chspec));
1563 else
1564 target_chan = lower_20_sb(CHSPEC_CHANNEL(chspec));
1565
1566 pactrl = 0;
1567 if (ISLCNPHY(pi)) {
1568 u32 offset_mcs, i;
1569
1570 if (CHSPEC_IS40(pi->radio_chanspec)) {
1571 offset_mcs = pi->mcs40_po;
1572 for (i = TXP_FIRST_SISO_MCS_20;
1573 i <= TXP_LAST_SISO_MCS_20; i++) {
1574 pi->tx_srom_max_rate_2g[i - 8] =
1575 pi->tx_srom_max_2g -
1576 ((offset_mcs & 0xf) * 2);
1577 offset_mcs >>= 4;
1578 }
1579 } else {
1580 offset_mcs = pi->mcs20_po;
1581 for (i = TXP_FIRST_SISO_MCS_20;
1582 i <= TXP_LAST_SISO_MCS_20; i++) {
1583 pi->tx_srom_max_rate_2g[i - 8] =
1584 pi->tx_srom_max_2g -
1585 ((offset_mcs & 0xf) * 2);
1586 offset_mcs >>= 4;
1587 }
1588 }
1589 }
1590
1591 max_num_rate = ((ISNPHY(pi)) ? (TXP_NUM_RATES) :
1592 ((ISLCNPHY(pi)) ?
1593 (TXP_LAST_SISO_MCS_20 + 1) : (TXP_LAST_OFDM + 1)));
1594
1595 wlc_phy_upd_env_txpwr_rate_limits(pi, band);
1596
1597 for (rate = start_rate; rate < max_num_rate; rate++) {
1598
1599 tx_pwr_target[rate] = pi->tx_user_target[rate];
1600
1601 if (pi->user_txpwr_at_rfport)
1602 tx_pwr_target[rate] +=
1603 wlc_user_txpwr_antport_to_rfport(pi,
1604 target_chan,
1605 band,
1606 rate);
1607
1608 wlc_phy_txpower_sromlimit((struct brcms_phy_pub *) pi,
1609 target_chan,
1610 &mintxpwr, &maxtxpwr, rate);
1611
1612 maxtxpwr = min(maxtxpwr, pi->txpwr_limit[rate]);
1613
1614 maxtxpwr = (maxtxpwr > pactrl) ? (maxtxpwr - pactrl) : 0;
1615
1616 maxtxpwr = (maxtxpwr > 6) ? (maxtxpwr - 6) : 0;
1617
1618 maxtxpwr = min(maxtxpwr, tx_pwr_target[rate]);
1619
1620 if (pi->txpwr_percent <= 100)
1621 maxtxpwr = (maxtxpwr * pi->txpwr_percent) / 100;
1622
1623 tx_pwr_target[rate] = max(maxtxpwr, mintxpwr);
1624
1625 tx_pwr_target[rate] =
1626 min(tx_pwr_target[rate], pi->txpwr_env_limit[rate]);
1627
1628 if (tx_pwr_target[rate] > tx_pwr_max)
1629 tx_pwr_max_rate_ind = rate;
1630
1631 tx_pwr_max = max(tx_pwr_max, tx_pwr_target[rate]);
1632 tx_pwr_min = min(tx_pwr_min, tx_pwr_target[rate]);
1633 }
1634
1635 memset(pi->tx_power_offset, 0, sizeof(pi->tx_power_offset));
1636 pi->tx_power_max = tx_pwr_max;
1637 pi->tx_power_min = tx_pwr_min;
1638 pi->tx_power_max_rate_ind = tx_pwr_max_rate_ind;
1639 for (rate = 0; rate < max_num_rate; rate++) {
1640
1641 pi->tx_power_target[rate] = tx_pwr_target[rate];
1642
1643 if (!pi->hwpwrctrl || ISNPHY(pi))
1644 pi->tx_power_offset[rate] =
1645 pi->tx_power_max - pi->tx_power_target[rate];
1646 else
1647 pi->tx_power_offset[rate] =
1648 pi->tx_power_target[rate] - pi->tx_power_min;
1649 }
1650
1651 txpwr_recalc_fn = pi->pi_fptr.txpwrrecalc;
1652 if (txpwr_recalc_fn)
1653 (*txpwr_recalc_fn)(pi);
1654}
1655
1656static void
1657wlc_phy_txpower_reg_limit_calc(struct brcms_phy *pi, struct txpwr_limits *txpwr,
1658 u16 chanspec)
1659{
1660 u8 tmp_txpwr_limit[2 * BRCMS_NUM_RATES_OFDM];
1661 u8 *txpwr_ptr1 = NULL, *txpwr_ptr2 = NULL;
1662 int rate_start_index = 0, rate1, rate2, k;
1663
1664 for (rate1 = WL_TX_POWER_CCK_FIRST, rate2 = 0;
1665 rate2 < WL_TX_POWER_CCK_NUM; rate1++, rate2++)
1666 pi->txpwr_limit[rate1] = txpwr->cck[rate2];
1667
1668 for (rate1 = WL_TX_POWER_OFDM_FIRST, rate2 = 0;
1669 rate2 < WL_TX_POWER_OFDM_NUM; rate1++, rate2++)
1670 pi->txpwr_limit[rate1] = txpwr->ofdm[rate2];
1671
1672 if (ISNPHY(pi)) {
1673
1674 for (k = 0; k < 4; k++) {
1675 switch (k) {
1676 case 0:
1677
1678 txpwr_ptr1 = txpwr->mcs_20_siso;
1679 txpwr_ptr2 = txpwr->ofdm;
1680 rate_start_index = WL_TX_POWER_OFDM_FIRST;
1681 break;
1682 case 1:
1683
1684 txpwr_ptr1 = txpwr->mcs_20_cdd;
1685 txpwr_ptr2 = txpwr->ofdm_cdd;
1686 rate_start_index = WL_TX_POWER_OFDM20_CDD_FIRST;
1687 break;
1688 case 2:
1689
1690 txpwr_ptr1 = txpwr->mcs_40_siso;
1691 txpwr_ptr2 = txpwr->ofdm_40_siso;
1692 rate_start_index =
1693 WL_TX_POWER_OFDM40_SISO_FIRST;
1694 break;
1695 case 3:
1696
1697 txpwr_ptr1 = txpwr->mcs_40_cdd;
1698 txpwr_ptr2 = txpwr->ofdm_40_cdd;
1699 rate_start_index = WL_TX_POWER_OFDM40_CDD_FIRST;
1700 break;
1701 }
1702
1703 for (rate2 = 0; rate2 < BRCMS_NUM_RATES_OFDM;
1704 rate2++) {
1705 tmp_txpwr_limit[rate2] = 0;
1706 tmp_txpwr_limit[BRCMS_NUM_RATES_OFDM + rate2] =
1707 txpwr_ptr1[rate2];
1708 }
1709 wlc_phy_mcs_to_ofdm_powers_nphy(
1710 tmp_txpwr_limit, 0,
1711 BRCMS_NUM_RATES_OFDM -
1712 1, BRCMS_NUM_RATES_OFDM);
1713 for (rate1 = rate_start_index, rate2 = 0;
1714 rate2 < BRCMS_NUM_RATES_OFDM; rate1++, rate2++)
1715 pi->txpwr_limit[rate1] =
1716 min(txpwr_ptr2[rate2],
1717 tmp_txpwr_limit[rate2]);
1718 }
1719
1720 for (k = 0; k < 4; k++) {
1721 switch (k) {
1722 case 0:
1723
1724 txpwr_ptr1 = txpwr->ofdm;
1725 txpwr_ptr2 = txpwr->mcs_20_siso;
1726 rate_start_index = WL_TX_POWER_MCS20_SISO_FIRST;
1727 break;
1728 case 1:
1729
1730 txpwr_ptr1 = txpwr->ofdm_cdd;
1731 txpwr_ptr2 = txpwr->mcs_20_cdd;
1732 rate_start_index = WL_TX_POWER_MCS20_CDD_FIRST;
1733 break;
1734 case 2:
1735
1736 txpwr_ptr1 = txpwr->ofdm_40_siso;
1737 txpwr_ptr2 = txpwr->mcs_40_siso;
1738 rate_start_index = WL_TX_POWER_MCS40_SISO_FIRST;
1739 break;
1740 case 3:
1741
1742 txpwr_ptr1 = txpwr->ofdm_40_cdd;
1743 txpwr_ptr2 = txpwr->mcs_40_cdd;
1744 rate_start_index = WL_TX_POWER_MCS40_CDD_FIRST;
1745 break;
1746 }
1747 for (rate2 = 0; rate2 < BRCMS_NUM_RATES_OFDM;
1748 rate2++) {
1749 tmp_txpwr_limit[rate2] = 0;
1750 tmp_txpwr_limit[BRCMS_NUM_RATES_OFDM + rate2] =
1751 txpwr_ptr1[rate2];
1752 }
1753 wlc_phy_ofdm_to_mcs_powers_nphy(
1754 tmp_txpwr_limit, 0,
1755 BRCMS_NUM_RATES_OFDM -
1756 1, BRCMS_NUM_RATES_OFDM);
1757 for (rate1 = rate_start_index, rate2 = 0;
1758 rate2 < BRCMS_NUM_RATES_MCS_1_STREAM;
1759 rate1++, rate2++)
1760 pi->txpwr_limit[rate1] =
1761 min(txpwr_ptr2[rate2],
1762 tmp_txpwr_limit[rate2]);
1763 }
1764
1765 for (k = 0; k < 2; k++) {
1766 switch (k) {
1767 case 0:
1768
1769 rate_start_index = WL_TX_POWER_MCS20_STBC_FIRST;
1770 txpwr_ptr1 = txpwr->mcs_20_stbc;
1771 break;
1772 case 1:
1773
1774 rate_start_index = WL_TX_POWER_MCS40_STBC_FIRST;
1775 txpwr_ptr1 = txpwr->mcs_40_stbc;
1776 break;
1777 }
1778 for (rate1 = rate_start_index, rate2 = 0;
1779 rate2 < BRCMS_NUM_RATES_MCS_1_STREAM;
1780 rate1++, rate2++)
1781 pi->txpwr_limit[rate1] = txpwr_ptr1[rate2];
1782 }
1783
1784 for (k = 0; k < 2; k++) {
1785 switch (k) {
1786 case 0:
1787
1788 rate_start_index = WL_TX_POWER_MCS20_SDM_FIRST;
1789 txpwr_ptr1 = txpwr->mcs_20_mimo;
1790 break;
1791 case 1:
1792
1793 rate_start_index = WL_TX_POWER_MCS40_SDM_FIRST;
1794 txpwr_ptr1 = txpwr->mcs_40_mimo;
1795 break;
1796 }
1797 for (rate1 = rate_start_index, rate2 = 0;
1798 rate2 < BRCMS_NUM_RATES_MCS_2_STREAM;
1799 rate1++, rate2++)
1800 pi->txpwr_limit[rate1] = txpwr_ptr1[rate2];
1801 }
1802
1803 pi->txpwr_limit[WL_TX_POWER_MCS_32] = txpwr->mcs32;
1804
1805 pi->txpwr_limit[WL_TX_POWER_MCS40_CDD_FIRST] =
1806 min(pi->txpwr_limit[WL_TX_POWER_MCS40_CDD_FIRST],
1807 pi->txpwr_limit[WL_TX_POWER_MCS_32]);
1808 pi->txpwr_limit[WL_TX_POWER_MCS_32] =
1809 pi->txpwr_limit[WL_TX_POWER_MCS40_CDD_FIRST];
1810 }
1811}
1812
1813void wlc_phy_txpwr_percent_set(struct brcms_phy_pub *ppi, u8 txpwr_percent)
1814{
1815 struct brcms_phy *pi = container_of(ppi, struct brcms_phy, pubpi_ro);
1816
1817 pi->txpwr_percent = txpwr_percent;
1818}
1819
1820void wlc_phy_machwcap_set(struct brcms_phy_pub *ppi, u32 machwcap)
1821{
1822 struct brcms_phy *pi = container_of(ppi, struct brcms_phy, pubpi_ro);
1823
1824 pi->sh->machwcap = machwcap;
1825}
1826
1827void wlc_phy_runbist_config(struct brcms_phy_pub *ppi, bool start_end)
1828{
1829 struct brcms_phy *pi = container_of(ppi, struct brcms_phy, pubpi_ro);
1830 u16 rxc;
1831 rxc = 0;
1832
1833 if (start_end == ON) {
1834 if (!ISNPHY(pi))
1835 return;
1836
1837 if (NREV_IS(pi->pubpi.phy_rev, 3)
1838 || NREV_IS(pi->pubpi.phy_rev, 4)) {
1839 bcma_wflush16(pi->d11core, D11REGOFFS(phyregaddr),
1840 0xa0);
1841 bcma_set16(pi->d11core, D11REGOFFS(phyregdata),
1842 0x1 << 15);
1843 }
1844 } else {
1845 if (NREV_IS(pi->pubpi.phy_rev, 3)
1846 || NREV_IS(pi->pubpi.phy_rev, 4)) {
1847 bcma_wflush16(pi->d11core, D11REGOFFS(phyregaddr),
1848 0xa0);
1849 bcma_write16(pi->d11core, D11REGOFFS(phyregdata), rxc);
1850 }
1851
1852 wlc_phy_por_inform(ppi);
1853 }
1854}
1855
1856void
1857wlc_phy_txpower_limit_set(struct brcms_phy_pub *ppi, struct txpwr_limits *txpwr,
1858 u16 chanspec)
1859{
1860 struct brcms_phy *pi = container_of(ppi, struct brcms_phy, pubpi_ro);
1861
1862 wlc_phy_txpower_reg_limit_calc(pi, txpwr, chanspec);
1863
1864 if (ISLCNPHY(pi)) {
1865 int i, j;
1866 for (i = TXP_FIRST_OFDM_20_CDD, j = 0;
1867 j < BRCMS_NUM_RATES_MCS_1_STREAM; i++, j++) {
1868 if (txpwr->mcs_20_siso[j])
1869 pi->txpwr_limit[i] = txpwr->mcs_20_siso[j];
1870 else
1871 pi->txpwr_limit[i] = txpwr->ofdm[j];
1872 }
1873 }
1874
1875 wlapi_suspend_mac_and_wait(pi->sh->physhim);
1876
1877 wlc_phy_txpower_recalc_target(pi);
1878 wlc_phy_cal_txpower_recalc_sw(pi);
1879 wlapi_enable_mac(pi->sh->physhim);
1880}
1881
1882void wlc_phy_ofdm_rateset_war(struct brcms_phy_pub *pih, bool war)
1883{
1884 struct brcms_phy *pi = container_of(pih, struct brcms_phy, pubpi_ro);
1885
1886 pi->ofdm_rateset_war = war;
1887}
1888
1889void wlc_phy_bf_preempt_enable(struct brcms_phy_pub *pih, bool bf_preempt)
1890{
1891 struct brcms_phy *pi = container_of(pih, struct brcms_phy, pubpi_ro);
1892
1893 pi->bf_preempt_4306 = bf_preempt;
1894}
1895
1896void wlc_phy_txpower_update_shm(struct brcms_phy *pi)
1897{
1898 int j;
1899 if (ISNPHY(pi))
1900 return;
1901
1902 if (!pi->sh->clk)
1903 return;
1904
1905 if (pi->hwpwrctrl) {
1906 u16 offset;
1907
1908 wlapi_bmac_write_shm(pi->sh->physhim, M_TXPWR_MAX, 63);
1909 wlapi_bmac_write_shm(pi->sh->physhim, M_TXPWR_N,
1910 1 << NUM_TSSI_FRAMES);
1911
1912 wlapi_bmac_write_shm(pi->sh->physhim, M_TXPWR_TARGET,
1913 pi->tx_power_min << NUM_TSSI_FRAMES);
1914
1915 wlapi_bmac_write_shm(pi->sh->physhim, M_TXPWR_CUR,
1916 pi->hwpwr_txcur);
1917
1918 for (j = TXP_FIRST_OFDM; j <= TXP_LAST_OFDM; j++) {
1919 const u8 ucode_ofdm_rates[] = {
1920 0x0c, 0x12, 0x18, 0x24, 0x30, 0x48, 0x60, 0x6c
1921 };
1922 offset = wlapi_bmac_rate_shm_offset(
1923 pi->sh->physhim,
1924 ucode_ofdm_rates[j - TXP_FIRST_OFDM]);
1925 wlapi_bmac_write_shm(pi->sh->physhim, offset + 6,
1926 pi->tx_power_offset[j]);
1927 wlapi_bmac_write_shm(pi->sh->physhim, offset + 14,
1928 -(pi->tx_power_offset[j] / 2));
1929 }
1930
1931 wlapi_bmac_mhf(pi->sh->physhim, MHF2, MHF2_HWPWRCTL,
1932 MHF2_HWPWRCTL, BRCM_BAND_ALL);
1933 } else {
1934 int i;
1935
1936 for (i = TXP_FIRST_OFDM; i <= TXP_LAST_OFDM; i++)
1937 pi->tx_power_offset[i] =
1938 (u8) roundup(pi->tx_power_offset[i], 8);
1939 wlapi_bmac_write_shm(pi->sh->physhim, M_OFDM_OFFSET,
1940 (u16)
1941 ((pi->tx_power_offset[TXP_FIRST_OFDM]
1942 + 7) >> 3));
1943 }
1944}
1945
1946bool wlc_phy_txpower_hw_ctrl_get(struct brcms_phy_pub *ppi)
1947{
1948 struct brcms_phy *pi = container_of(ppi, struct brcms_phy, pubpi_ro);
1949
1950 if (ISNPHY(pi))
1951 return pi->nphy_txpwrctrl;
1952 else
1953 return pi->hwpwrctrl;
1954}
1955
1956void wlc_phy_txpower_hw_ctrl_set(struct brcms_phy_pub *ppi, bool hwpwrctrl)
1957{
1958 struct brcms_phy *pi = container_of(ppi, struct brcms_phy, pubpi_ro);
1959 bool suspend;
1960
1961 if (!pi->hwpwrctrl_capable)
1962 return;
1963
1964 pi->hwpwrctrl = hwpwrctrl;
1965 pi->nphy_txpwrctrl = hwpwrctrl;
1966 pi->txpwrctrl = hwpwrctrl;
1967
1968 if (ISNPHY(pi)) {
1969 suspend = (0 == (bcma_read32(pi->d11core,
1970 D11REGOFFS(maccontrol)) &
1971 MCTL_EN_MAC));
1972 if (!suspend)
1973 wlapi_suspend_mac_and_wait(pi->sh->physhim);
1974
1975 wlc_phy_txpwrctrl_enable_nphy(pi, pi->nphy_txpwrctrl);
1976 if (pi->nphy_txpwrctrl == PHY_TPC_HW_OFF)
1977 wlc_phy_txpwr_fixpower_nphy(pi);
1978 else
1979 mod_phy_reg(pi, 0x1e7, (0x7f << 0),
1980 pi->saved_txpwr_idx);
1981
1982 if (!suspend)
1983 wlapi_enable_mac(pi->sh->physhim);
1984 }
1985}
1986
1987void wlc_phy_txpower_ipa_upd(struct brcms_phy *pi)
1988{
1989
1990 if (NREV_GE(pi->pubpi.phy_rev, 3)) {
1991 pi->ipa2g_on = (pi->srom_fem2g.extpagain == 2);
1992 pi->ipa5g_on = (pi->srom_fem5g.extpagain == 2);
1993 } else {
1994 pi->ipa2g_on = false;
1995 pi->ipa5g_on = false;
1996 }
1997}
1998
1999static u32 wlc_phy_txpower_est_power_nphy(struct brcms_phy *pi)
2000{
2001 s16 tx0_status, tx1_status;
2002 u16 estPower1, estPower2;
2003 u8 pwr0, pwr1, adj_pwr0, adj_pwr1;
2004 u32 est_pwr;
2005
2006 estPower1 = read_phy_reg(pi, 0x118);
2007 estPower2 = read_phy_reg(pi, 0x119);
2008
2009 if ((estPower1 & (0x1 << 8)) == (0x1 << 8))
2010 pwr0 = (u8) (estPower1 & (0xff << 0)) >> 0;
2011 else
2012 pwr0 = 0x80;
2013
2014 if ((estPower2 & (0x1 << 8)) == (0x1 << 8))
2015 pwr1 = (u8) (estPower2 & (0xff << 0)) >> 0;
2016 else
2017 pwr1 = 0x80;
2018
2019 tx0_status = read_phy_reg(pi, 0x1ed);
2020 tx1_status = read_phy_reg(pi, 0x1ee);
2021
2022 if ((tx0_status & (0x1 << 15)) == (0x1 << 15))
2023 adj_pwr0 = (u8) (tx0_status & (0xff << 0)) >> 0;
2024 else
2025 adj_pwr0 = 0x80;
2026 if ((tx1_status & (0x1 << 15)) == (0x1 << 15))
2027 adj_pwr1 = (u8) (tx1_status & (0xff << 0)) >> 0;
2028 else
2029 adj_pwr1 = 0x80;
2030
2031 est_pwr = (u32) ((pwr0 << 24) | (pwr1 << 16) | (adj_pwr0 << 8) |
2032 adj_pwr1);
2033
2034 return est_pwr;
2035}
2036
2037void
2038wlc_phy_txpower_get_current(struct brcms_phy_pub *ppi, struct tx_power *power,
2039 uint channel)
2040{
2041 struct brcms_phy *pi = container_of(ppi, struct brcms_phy, pubpi_ro);
2042 uint rate, num_rates;
2043 u8 min_pwr, max_pwr;
2044
2045#if WL_TX_POWER_RATES != TXP_NUM_RATES
2046#error "struct tx_power out of sync with this fn"
2047#endif
2048
2049 if (ISNPHY(pi)) {
2050 power->rf_cores = 2;
2051 power->flags |= (WL_TX_POWER_F_MIMO);
2052 if (pi->nphy_txpwrctrl == PHY_TPC_HW_ON)
2053 power->flags |=
2054 (WL_TX_POWER_F_ENABLED | WL_TX_POWER_F_HW);
2055 } else if (ISLCNPHY(pi)) {
2056 power->rf_cores = 1;
2057 power->flags |= (WL_TX_POWER_F_SISO);
2058 if (pi->radiopwr_override == RADIOPWR_OVERRIDE_DEF)
2059 power->flags |= WL_TX_POWER_F_ENABLED;
2060 if (pi->hwpwrctrl)
2061 power->flags |= WL_TX_POWER_F_HW;
2062 }
2063
2064 num_rates = ((ISNPHY(pi)) ? (TXP_NUM_RATES) :
2065 ((ISLCNPHY(pi)) ?
2066 (TXP_LAST_OFDM_20_CDD + 1) : (TXP_LAST_OFDM + 1)));
2067
2068 for (rate = 0; rate < num_rates; rate++) {
2069 power->user_limit[rate] = pi->tx_user_target[rate];
2070 wlc_phy_txpower_sromlimit(ppi, channel, &min_pwr, &max_pwr,
2071 rate);
2072 power->board_limit[rate] = (u8) max_pwr;
2073 power->target[rate] = pi->tx_power_target[rate];
2074 }
2075
2076 if (ISNPHY(pi)) {
2077 u32 est_pout;
2078
2079 wlapi_suspend_mac_and_wait(pi->sh->physhim);
2080 wlc_phyreg_enter((struct brcms_phy_pub *) pi);
2081 est_pout = wlc_phy_txpower_est_power_nphy(pi);
2082 wlc_phyreg_exit((struct brcms_phy_pub *) pi);
2083 wlapi_enable_mac(pi->sh->physhim);
2084
2085 power->est_Pout[0] = (est_pout >> 8) & 0xff;
2086 power->est_Pout[1] = est_pout & 0xff;
2087
2088 power->est_Pout_act[0] = est_pout >> 24;
2089 power->est_Pout_act[1] = (est_pout >> 16) & 0xff;
2090
2091 if (power->est_Pout[0] == 0x80)
2092 power->est_Pout[0] = 0;
2093 if (power->est_Pout[1] == 0x80)
2094 power->est_Pout[1] = 0;
2095
2096 if (power->est_Pout_act[0] == 0x80)
2097 power->est_Pout_act[0] = 0;
2098 if (power->est_Pout_act[1] == 0x80)
2099 power->est_Pout_act[1] = 0;
2100
2101 power->est_Pout_cck = 0;
2102
2103 power->tx_power_max[0] = pi->tx_power_max;
2104 power->tx_power_max[1] = pi->tx_power_max;
2105
2106 power->tx_power_max_rate_ind[0] = pi->tx_power_max_rate_ind;
2107 power->tx_power_max_rate_ind[1] = pi->tx_power_max_rate_ind;
2108 } else if (pi->hwpwrctrl && pi->sh->up) {
2109
2110 wlc_phyreg_enter(ppi);
2111 if (ISLCNPHY(pi)) {
2112
2113 power->tx_power_max[0] = pi->tx_power_max;
2114 power->tx_power_max[1] = pi->tx_power_max;
2115
2116 power->tx_power_max_rate_ind[0] =
2117 pi->tx_power_max_rate_ind;
2118 power->tx_power_max_rate_ind[1] =
2119 pi->tx_power_max_rate_ind;
2120
2121 if (wlc_phy_tpc_isenabled_lcnphy(pi))
2122 power->flags |=
2123 (WL_TX_POWER_F_HW |
2124 WL_TX_POWER_F_ENABLED);
2125 else
2126 power->flags &=
2127 ~(WL_TX_POWER_F_HW |
2128 WL_TX_POWER_F_ENABLED);
2129
2130 wlc_lcnphy_get_tssi(pi, (s8 *) &power->est_Pout[0],
2131 (s8 *) &power->est_Pout_cck);
2132 }
2133 wlc_phyreg_exit(ppi);
2134 }
2135}
2136
2137void wlc_phy_antsel_type_set(struct brcms_phy_pub *ppi, u8 antsel_type)
2138{
2139 struct brcms_phy *pi = container_of(ppi, struct brcms_phy, pubpi_ro);
2140
2141 pi->antsel_type = antsel_type;
2142}
2143
2144bool wlc_phy_test_ison(struct brcms_phy_pub *ppi)
2145{
2146 struct brcms_phy *pi = container_of(ppi, struct brcms_phy, pubpi_ro);
2147
2148 return pi->phytest_on;
2149}
2150
2151void wlc_phy_ant_rxdiv_set(struct brcms_phy_pub *ppi, u8 val)
2152{
2153 struct brcms_phy *pi = container_of(ppi, struct brcms_phy, pubpi_ro);
2154 bool suspend;
2155
2156 pi->sh->rx_antdiv = val;
2157
2158 if (!(ISNPHY(pi) && D11REV_IS(pi->sh->corerev, 16))) {
2159 if (val > ANT_RX_DIV_FORCE_1)
2160 wlapi_bmac_mhf(pi->sh->physhim, MHF1, MHF1_ANTDIV,
2161 MHF1_ANTDIV, BRCM_BAND_ALL);
2162 else
2163 wlapi_bmac_mhf(pi->sh->physhim, MHF1, MHF1_ANTDIV, 0,
2164 BRCM_BAND_ALL);
2165 }
2166
2167 if (ISNPHY(pi))
2168 return;
2169
2170 if (!pi->sh->clk)
2171 return;
2172
2173 suspend = (0 == (bcma_read32(pi->d11core, D11REGOFFS(maccontrol)) &
2174 MCTL_EN_MAC));
2175 if (!suspend)
2176 wlapi_suspend_mac_and_wait(pi->sh->physhim);
2177
2178 if (ISLCNPHY(pi)) {
2179 if (val > ANT_RX_DIV_FORCE_1) {
2180 mod_phy_reg(pi, 0x410, (0x1 << 1), 0x01 << 1);
2181 mod_phy_reg(pi, 0x410,
2182 (0x1 << 0),
2183 ((ANT_RX_DIV_START_1 == val) ? 1 : 0) << 0);
2184 } else {
2185 mod_phy_reg(pi, 0x410, (0x1 << 1), 0x00 << 1);
2186 mod_phy_reg(pi, 0x410, (0x1 << 0), (u16) val << 0);
2187 }
2188 }
2189
2190 if (!suspend)
2191 wlapi_enable_mac(pi->sh->physhim);
2192
2193 return;
2194}
2195
2196static bool
2197wlc_phy_noise_calc_phy(struct brcms_phy *pi, u32 *cmplx_pwr, s8 *pwr_ant)
2198{
2199 s8 cmplx_pwr_dbm[PHY_CORE_MAX];
2200 u8 i;
2201
2202 memset((u8 *) cmplx_pwr_dbm, 0, sizeof(cmplx_pwr_dbm));
2203 wlc_phy_compute_dB(cmplx_pwr, cmplx_pwr_dbm, pi->pubpi.phy_corenum);
2204
2205 for (i = 0; i < pi->pubpi.phy_corenum; i++) {
2206 if (NREV_GE(pi->pubpi.phy_rev, 3))
2207 cmplx_pwr_dbm[i] += (s8) PHY_NOISE_OFFSETFACT_4322;
2208 else
2209
2210 cmplx_pwr_dbm[i] += (s8) (16 - (15) * 3 - 70);
2211 }
2212
2213 for (i = 0; i < pi->pubpi.phy_corenum; i++) {
2214 pi->nphy_noise_win[i][pi->nphy_noise_index] = cmplx_pwr_dbm[i];
2215 pwr_ant[i] = cmplx_pwr_dbm[i];
2216 }
2217 pi->nphy_noise_index =
2218 MODINC_POW2(pi->nphy_noise_index, PHY_NOISE_WINDOW_SZ);
2219 return true;
2220}
2221
2222static void wlc_phy_noise_cb(struct brcms_phy *pi, u8 channel, s8 noise_dbm)
2223{
2224 if (!pi->phynoise_state)
2225 return;
2226
2227 if (pi->phynoise_state & PHY_NOISE_STATE_MON) {
2228 if (pi->phynoise_chan_watchdog == channel) {
2229 pi->sh->phy_noise_window[pi->sh->phy_noise_index] =
2230 noise_dbm;
2231 pi->sh->phy_noise_index =
2232 MODINC(pi->sh->phy_noise_index, MA_WINDOW_SZ);
2233 }
2234 pi->phynoise_state &= ~PHY_NOISE_STATE_MON;
2235 }
2236
2237 if (pi->phynoise_state & PHY_NOISE_STATE_EXTERNAL)
2238 pi->phynoise_state &= ~PHY_NOISE_STATE_EXTERNAL;
2239
2240}
2241
2242static s8 wlc_phy_noise_read_shmem(struct brcms_phy *pi)
2243{
2244 u32 cmplx_pwr[PHY_CORE_MAX];
2245 s8 noise_dbm_ant[PHY_CORE_MAX];
2246 u16 lo, hi;
2247 u32 cmplx_pwr_tot = 0;
2248 s8 noise_dbm = PHY_NOISE_FIXED_VAL_NPHY;
2249 u8 idx, core;
2250
2251 memset((u8 *) cmplx_pwr, 0, sizeof(cmplx_pwr));
2252 memset((u8 *) noise_dbm_ant, 0, sizeof(noise_dbm_ant));
2253
2254 for (idx = 0, core = 0; core < pi->pubpi.phy_corenum; idx += 2,
2255 core++) {
2256 lo = wlapi_bmac_read_shm(pi->sh->physhim, M_PWRIND_MAP(idx));
2257 hi = wlapi_bmac_read_shm(pi->sh->physhim,
2258 M_PWRIND_MAP(idx + 1));
2259 cmplx_pwr[core] = (hi << 16) + lo;
2260 cmplx_pwr_tot += cmplx_pwr[core];
2261 if (cmplx_pwr[core] == 0)
2262 noise_dbm_ant[core] = PHY_NOISE_FIXED_VAL_NPHY;
2263 else
2264 cmplx_pwr[core] >>= PHY_NOISE_SAMPLE_LOG_NUM_UCODE;
2265 }
2266
2267 if (cmplx_pwr_tot != 0)
2268 wlc_phy_noise_calc_phy(pi, cmplx_pwr, noise_dbm_ant);
2269
2270 for (core = 0; core < pi->pubpi.phy_corenum; core++) {
2271 pi->nphy_noise_win[core][pi->nphy_noise_index] =
2272 noise_dbm_ant[core];
2273
2274 if (noise_dbm_ant[core] > noise_dbm)
2275 noise_dbm = noise_dbm_ant[core];
2276 }
2277 pi->nphy_noise_index =
2278 MODINC_POW2(pi->nphy_noise_index, PHY_NOISE_WINDOW_SZ);
2279
2280 return noise_dbm;
2281
2282}
2283
2284void wlc_phy_noise_sample_intr(struct brcms_phy_pub *pih)
2285{
2286 struct brcms_phy *pi = container_of(pih, struct brcms_phy, pubpi_ro);
2287 u16 jssi_aux;
2288 u8 channel = 0;
2289 s8 noise_dbm = PHY_NOISE_FIXED_VAL_NPHY;
2290
2291 if (ISLCNPHY(pi)) {
2292 u32 cmplx_pwr, cmplx_pwr0, cmplx_pwr1;
2293 u16 lo, hi;
2294 s32 pwr_offset_dB, gain_dB;
2295 u16 status_0, status_1;
2296
2297 jssi_aux = wlapi_bmac_read_shm(pi->sh->physhim, M_JSSI_AUX);
2298 channel = jssi_aux & D11_CURCHANNEL_MAX;
2299
2300 lo = wlapi_bmac_read_shm(pi->sh->physhim, M_PWRIND_MAP0);
2301 hi = wlapi_bmac_read_shm(pi->sh->physhim, M_PWRIND_MAP1);
2302 cmplx_pwr0 = (hi << 16) + lo;
2303
2304 lo = wlapi_bmac_read_shm(pi->sh->physhim, M_PWRIND_MAP2);
2305 hi = wlapi_bmac_read_shm(pi->sh->physhim, M_PWRIND_MAP3);
2306 cmplx_pwr1 = (hi << 16) + lo;
2307 cmplx_pwr = (cmplx_pwr0 + cmplx_pwr1) >> 6;
2308
2309 status_0 = 0x44;
2310 status_1 = wlapi_bmac_read_shm(pi->sh->physhim, M_JSSI_0);
2311 if ((cmplx_pwr > 0 && cmplx_pwr < 500)
2312 && ((status_1 & 0xc000) == 0x4000)) {
2313
2314 wlc_phy_compute_dB(&cmplx_pwr, &noise_dbm,
2315 pi->pubpi.phy_corenum);
2316 pwr_offset_dB = (read_phy_reg(pi, 0x434) & 0xFF);
2317 if (pwr_offset_dB > 127)
2318 pwr_offset_dB -= 256;
2319
2320 noise_dbm += (s8) (pwr_offset_dB - 30);
2321
2322 gain_dB = (status_0 & 0x1ff);
2323 noise_dbm -= (s8) (gain_dB);
2324 } else {
2325 noise_dbm = PHY_NOISE_FIXED_VAL_LCNPHY;
2326 }
2327 } else if (ISNPHY(pi)) {
2328
2329 jssi_aux = wlapi_bmac_read_shm(pi->sh->physhim, M_JSSI_AUX);
2330 channel = jssi_aux & D11_CURCHANNEL_MAX;
2331
2332 noise_dbm = wlc_phy_noise_read_shmem(pi);
2333 }
2334
2335 wlc_phy_noise_cb(pi, channel, noise_dbm);
2336
2337}
2338
2339static void
2340wlc_phy_noise_sample_request(struct brcms_phy_pub *pih, u8 reason, u8 ch)
2341{
2342 struct brcms_phy *pi = container_of(pih, struct brcms_phy, pubpi_ro);
2343 s8 noise_dbm = PHY_NOISE_FIXED_VAL_NPHY;
2344 bool sampling_in_progress = (pi->phynoise_state != 0);
2345 bool wait_for_intr = true;
2346
2347 switch (reason) {
2348 case PHY_NOISE_SAMPLE_MON:
2349 pi->phynoise_chan_watchdog = ch;
2350 pi->phynoise_state |= PHY_NOISE_STATE_MON;
2351 break;
2352
2353 case PHY_NOISE_SAMPLE_EXTERNAL:
2354 pi->phynoise_state |= PHY_NOISE_STATE_EXTERNAL;
2355 break;
2356
2357 default:
2358 break;
2359 }
2360
2361 if (sampling_in_progress)
2362 return;
2363
2364 pi->phynoise_now = pi->sh->now;
2365
2366 if (pi->phy_fixed_noise) {
2367 if (ISNPHY(pi)) {
2368 pi->nphy_noise_win[WL_ANT_IDX_1][pi->nphy_noise_index] =
2369 PHY_NOISE_FIXED_VAL_NPHY;
2370 pi->nphy_noise_win[WL_ANT_IDX_2][pi->nphy_noise_index] =
2371 PHY_NOISE_FIXED_VAL_NPHY;
2372 pi->nphy_noise_index = MODINC_POW2(pi->nphy_noise_index,
2373 PHY_NOISE_WINDOW_SZ);
2374 noise_dbm = PHY_NOISE_FIXED_VAL_NPHY;
2375 } else {
2376 noise_dbm = PHY_NOISE_FIXED_VAL;
2377 }
2378
2379 wait_for_intr = false;
2380 goto done;
2381 }
2382
2383 if (ISLCNPHY(pi)) {
2384 if (!pi->phynoise_polling
2385 || (reason == PHY_NOISE_SAMPLE_EXTERNAL)) {
2386 wlapi_bmac_write_shm(pi->sh->physhim, M_JSSI_0, 0);
2387 wlapi_bmac_write_shm(pi->sh->physhim, M_PWRIND_MAP0, 0);
2388 wlapi_bmac_write_shm(pi->sh->physhim, M_PWRIND_MAP1, 0);
2389 wlapi_bmac_write_shm(pi->sh->physhim, M_PWRIND_MAP2, 0);
2390 wlapi_bmac_write_shm(pi->sh->physhim, M_PWRIND_MAP3, 0);
2391
2392 bcma_set32(pi->d11core, D11REGOFFS(maccommand),
2393 MCMD_BG_NOISE);
2394 } else {
2395 wlapi_suspend_mac_and_wait(pi->sh->physhim);
2396 wlc_lcnphy_deaf_mode(pi, (bool) 0);
2397 noise_dbm = (s8) wlc_lcnphy_rx_signal_power(pi, 20);
2398 wlc_lcnphy_deaf_mode(pi, (bool) 1);
2399 wlapi_enable_mac(pi->sh->physhim);
2400 wait_for_intr = false;
2401 }
2402 } else if (ISNPHY(pi)) {
2403 if (!pi->phynoise_polling
2404 || (reason == PHY_NOISE_SAMPLE_EXTERNAL)) {
2405
2406 wlapi_bmac_write_shm(pi->sh->physhim, M_PWRIND_MAP0, 0);
2407 wlapi_bmac_write_shm(pi->sh->physhim, M_PWRIND_MAP1, 0);
2408 wlapi_bmac_write_shm(pi->sh->physhim, M_PWRIND_MAP2, 0);
2409 wlapi_bmac_write_shm(pi->sh->physhim, M_PWRIND_MAP3, 0);
2410
2411 bcma_set32(pi->d11core, D11REGOFFS(maccommand),
2412 MCMD_BG_NOISE);
2413 } else {
2414 struct phy_iq_est est[PHY_CORE_MAX];
2415 u32 cmplx_pwr[PHY_CORE_MAX];
2416 s8 noise_dbm_ant[PHY_CORE_MAX];
2417 u16 log_num_samps, num_samps, classif_state = 0;
2418 u8 wait_time = 32;
2419 u8 wait_crs = 0;
2420 u8 i;
2421
2422 memset((u8 *) est, 0, sizeof(est));
2423 memset((u8 *) cmplx_pwr, 0, sizeof(cmplx_pwr));
2424 memset((u8 *) noise_dbm_ant, 0, sizeof(noise_dbm_ant));
2425
2426 log_num_samps = PHY_NOISE_SAMPLE_LOG_NUM_NPHY;
2427 num_samps = 1 << log_num_samps;
2428
2429 wlapi_suspend_mac_and_wait(pi->sh->physhim);
2430 classif_state = wlc_phy_classifier_nphy(pi, 0, 0);
2431 wlc_phy_classifier_nphy(pi, 3, 0);
2432 wlc_phy_rx_iq_est_nphy(pi, est, num_samps, wait_time,
2433 wait_crs);
2434 wlc_phy_classifier_nphy(pi, (0x7 << 0), classif_state);
2435 wlapi_enable_mac(pi->sh->physhim);
2436
2437 for (i = 0; i < pi->pubpi.phy_corenum; i++)
2438 cmplx_pwr[i] = (est[i].i_pwr + est[i].q_pwr) >>
2439 log_num_samps;
2440
2441 wlc_phy_noise_calc_phy(pi, cmplx_pwr, noise_dbm_ant);
2442
2443 for (i = 0; i < pi->pubpi.phy_corenum; i++) {
2444 pi->nphy_noise_win[i][pi->nphy_noise_index] =
2445 noise_dbm_ant[i];
2446
2447 if (noise_dbm_ant[i] > noise_dbm)
2448 noise_dbm = noise_dbm_ant[i];
2449 }
2450 pi->nphy_noise_index = MODINC_POW2(pi->nphy_noise_index,
2451 PHY_NOISE_WINDOW_SZ);
2452
2453 wait_for_intr = false;
2454 }
2455 }
2456
2457done:
2458
2459 if (!wait_for_intr)
2460 wlc_phy_noise_cb(pi, ch, noise_dbm);
2461
2462}
2463
2464void wlc_phy_noise_sample_request_external(struct brcms_phy_pub *pih)
2465{
2466 u8 channel;
2467
2468 channel = CHSPEC_CHANNEL(wlc_phy_chanspec_get(pih));
2469
2470 wlc_phy_noise_sample_request(pih, PHY_NOISE_SAMPLE_EXTERNAL, channel);
2471}
2472
2473static const s8 lcnphy_gain_index_offset_for_pkt_rssi[] = {
2474 8,
2475 8,
2476 8,
2477 8,
2478 8,
2479 8,
2480 8,
2481 9,
2482 10,
2483 8,
2484 8,
2485 7,
2486 7,
2487 1,
2488 2,
2489 2,
2490 2,
2491 2,
2492 2,
2493 2,
2494 2,
2495 2,
2496 2,
2497 2,
2498 2,
2499 2,
2500 2,
2501 2,
2502 2,
2503 2,
2504 2,
2505 2,
2506 1,
2507 1,
2508 0,
2509 0,
2510 0,
2511 0
2512};
2513
2514void wlc_phy_compute_dB(u32 *cmplx_pwr, s8 *p_cmplx_pwr_dB, u8 core)
2515{
2516 u8 msb, secondmsb, i;
2517 u32 tmp;
2518
2519 for (i = 0; i < core; i++) {
2520 secondmsb = 0;
2521 tmp = cmplx_pwr[i];
2522 msb = fls(tmp);
2523 if (msb)
2524 secondmsb = (u8) ((tmp >> (--msb - 1)) & 1);
2525 p_cmplx_pwr_dB[i] = (s8) (3 * msb + 2 * secondmsb);
2526 }
2527}
2528
2529int wlc_phy_rssi_compute(struct brcms_phy_pub *pih,
2530 struct d11rxhdr *rxh)
2531{
2532 int rssi = rxh->PhyRxStatus_1 & PRXS1_JSSI_MASK;
2533 uint radioid = pih->radioid;
2534 struct brcms_phy *pi = container_of(pih, struct brcms_phy, pubpi_ro);
2535
2536 if ((pi->sh->corerev >= 11)
2537 && !(rxh->RxStatus2 & RXS_PHYRXST_VALID)) {
2538 rssi = BRCMS_RSSI_INVALID;
2539 goto end;
2540 }
2541
2542 if (ISLCNPHY(pi)) {
2543 u8 gidx = (rxh->PhyRxStatus_2 & 0xFC00) >> 10;
2544 struct brcms_phy_lcnphy *pi_lcn = pi->u.pi_lcnphy;
2545
2546 if (rssi > 127)
2547 rssi -= 256;
2548
2549 rssi = rssi + lcnphy_gain_index_offset_for_pkt_rssi[gidx];
2550 if ((rssi > -46) && (gidx > 18))
2551 rssi = rssi + 7;
2552
2553 rssi = rssi + pi_lcn->lcnphy_pkteng_rssi_slope;
2554
2555 rssi = rssi + 2;
2556
2557 }
2558
2559 if (ISLCNPHY(pi)) {
2560 if (rssi > 127)
2561 rssi -= 256;
2562 } else if (radioid == BCM2055_ID || radioid == BCM2056_ID
2563 || radioid == BCM2057_ID) {
2564 rssi = wlc_phy_rssi_compute_nphy(pi, rxh);
2565 }
2566
2567end:
2568 return rssi;
2569}
2570
2571void wlc_phy_freqtrack_start(struct brcms_phy_pub *pih)
2572{
2573 return;
2574}
2575
2576void wlc_phy_freqtrack_end(struct brcms_phy_pub *pih)
2577{
2578 return;
2579}
2580
2581void wlc_phy_set_deaf(struct brcms_phy_pub *ppi, bool user_flag)
2582{
2583 struct brcms_phy *pi;
2584 pi = (struct brcms_phy *) ppi;
2585
2586 if (ISLCNPHY(pi))
2587 wlc_lcnphy_deaf_mode(pi, true);
2588 else if (ISNPHY(pi))
2589 wlc_nphy_deaf_mode(pi, true);
2590}
2591
2592void wlc_phy_watchdog(struct brcms_phy_pub *pih)
2593{
2594 struct brcms_phy *pi = container_of(pih, struct brcms_phy, pubpi_ro);
2595 bool delay_phy_cal = false;
2596 pi->sh->now++;
2597
2598 if (!pi->watchdog_override)
2599 return;
2600
2601 if (!(SCAN_RM_IN_PROGRESS(pi) || PLT_INPROG_PHY(pi)))
2602 wlc_phy_noise_sample_request((struct brcms_phy_pub *) pi,
2603 PHY_NOISE_SAMPLE_MON,
2604 CHSPEC_CHANNEL(pi->
2605 radio_chanspec));
2606
2607 if (pi->phynoise_state && (pi->sh->now - pi->phynoise_now) > 5)
2608 pi->phynoise_state = 0;
2609
2610 if ((!pi->phycal_txpower) ||
2611 ((pi->sh->now - pi->phycal_txpower) >= pi->sh->fast_timer)) {
2612
2613 if (!SCAN_INPROG_PHY(pi) && wlc_phy_cal_txpower_recalc_sw(pi))
2614 pi->phycal_txpower = pi->sh->now;
2615 }
2616
2617 if ((SCAN_RM_IN_PROGRESS(pi) || PLT_INPROG_PHY(pi)
2618 || ASSOC_INPROG_PHY(pi)))
2619 return;
2620
2621 if (ISNPHY(pi) && !pi->disable_percal && !delay_phy_cal) {
2622
2623 if ((pi->nphy_perical != PHY_PERICAL_DISABLE) &&
2624 (pi->nphy_perical != PHY_PERICAL_MANUAL) &&
2625 ((pi->sh->now - pi->nphy_perical_last) >=
2626 pi->sh->glacial_timer))
2627 wlc_phy_cal_perical((struct brcms_phy_pub *) pi,
2628 PHY_PERICAL_WATCHDOG);
2629
2630 wlc_phy_txpwr_papd_cal_nphy(pi);
2631 }
2632
2633 if (ISLCNPHY(pi)) {
2634 if (pi->phy_forcecal ||
2635 ((pi->sh->now - pi->phy_lastcal) >=
2636 pi->sh->glacial_timer)) {
2637 if (!(SCAN_RM_IN_PROGRESS(pi) || ASSOC_INPROG_PHY(pi)))
2638 wlc_lcnphy_calib_modes(
2639 pi,
2640 LCNPHY_PERICAL_TEMPBASED_TXPWRCTRL);
2641 if (!
2642 (SCAN_RM_IN_PROGRESS(pi) || PLT_INPROG_PHY(pi)
2643 || ASSOC_INPROG_PHY(pi)
2644 || pi->carrier_suppr_disable
2645 || pi->disable_percal))
2646 wlc_lcnphy_calib_modes(pi,
2647 PHY_PERICAL_WATCHDOG);
2648 }
2649 }
2650}
2651
2652void wlc_phy_BSSinit(struct brcms_phy_pub *pih, bool bonlyap, int rssi)
2653{
2654 struct brcms_phy *pi = container_of(pih, struct brcms_phy, pubpi_ro);
2655 uint i;
2656 uint k;
2657
2658 for (i = 0; i < MA_WINDOW_SZ; i++)
2659 pi->sh->phy_noise_window[i] = (s8) (rssi & 0xff);
2660 if (ISLCNPHY(pi)) {
2661 for (i = 0; i < MA_WINDOW_SZ; i++)
2662 pi->sh->phy_noise_window[i] =
2663 PHY_NOISE_FIXED_VAL_LCNPHY;
2664 }
2665 pi->sh->phy_noise_index = 0;
2666
2667 for (i = 0; i < PHY_NOISE_WINDOW_SZ; i++) {
2668 for (k = WL_ANT_IDX_1; k < WL_ANT_RX_MAX; k++)
2669 pi->nphy_noise_win[k][i] = PHY_NOISE_FIXED_VAL_NPHY;
2670 }
2671 pi->nphy_noise_index = 0;
2672}
2673
2674void
2675wlc_phy_papd_decode_epsilon(u32 epsilon, s32 *eps_real, s32 *eps_imag)
2676{
2677 *eps_imag = (epsilon >> 13);
2678 if (*eps_imag > 0xfff)
2679 *eps_imag -= 0x2000;
2680
2681 *eps_real = (epsilon & 0x1fff);
2682 if (*eps_real > 0xfff)
2683 *eps_real -= 0x2000;
2684}
2685
2686void wlc_phy_cal_perical_mphase_reset(struct brcms_phy *pi)
2687{
2688 wlapi_del_timer(pi->phycal_timer);
2689
2690 pi->cal_type_override = PHY_PERICAL_AUTO;
2691 pi->mphase_cal_phase_id = MPHASE_CAL_STATE_IDLE;
2692 pi->mphase_txcal_cmdidx = 0;
2693}
2694
2695static void
2696wlc_phy_cal_perical_mphase_schedule(struct brcms_phy *pi, uint delay)
2697{
2698
2699 if ((pi->nphy_perical != PHY_PERICAL_MPHASE) &&
2700 (pi->nphy_perical != PHY_PERICAL_MANUAL))
2701 return;
2702
2703 wlapi_del_timer(pi->phycal_timer);
2704
2705 pi->mphase_cal_phase_id = MPHASE_CAL_STATE_INIT;
2706 wlapi_add_timer(pi->phycal_timer, delay, 0);
2707}
2708
2709void wlc_phy_cal_perical(struct brcms_phy_pub *pih, u8 reason)
2710{
2711 s16 nphy_currtemp = 0;
2712 s16 delta_temp = 0;
2713 bool do_periodic_cal = true;
2714 struct brcms_phy *pi = container_of(pih, struct brcms_phy, pubpi_ro);
2715
2716 if (!ISNPHY(pi))
2717 return;
2718
2719 if ((pi->nphy_perical == PHY_PERICAL_DISABLE) ||
2720 (pi->nphy_perical == PHY_PERICAL_MANUAL))
2721 return;
2722
2723 switch (reason) {
2724 case PHY_PERICAL_DRIVERUP:
2725 break;
2726
2727 case PHY_PERICAL_PHYINIT:
2728 if (pi->nphy_perical == PHY_PERICAL_MPHASE) {
2729 if (PHY_PERICAL_MPHASE_PENDING(pi))
2730 wlc_phy_cal_perical_mphase_reset(pi);
2731
2732 wlc_phy_cal_perical_mphase_schedule(
2733 pi,
2734 PHY_PERICAL_INIT_DELAY);
2735 }
2736 break;
2737
2738 case PHY_PERICAL_JOIN_BSS:
2739 case PHY_PERICAL_START_IBSS:
2740 case PHY_PERICAL_UP_BSS:
2741 if ((pi->nphy_perical == PHY_PERICAL_MPHASE) &&
2742 PHY_PERICAL_MPHASE_PENDING(pi))
2743 wlc_phy_cal_perical_mphase_reset(pi);
2744
2745 pi->first_cal_after_assoc = true;
2746
2747 pi->cal_type_override = PHY_PERICAL_FULL;
2748
2749 if (pi->phycal_tempdelta)
2750 pi->nphy_lastcal_temp = wlc_phy_tempsense_nphy(pi);
2751
2752 wlc_phy_cal_perical_nphy_run(pi, PHY_PERICAL_FULL);
2753 break;
2754
2755 case PHY_PERICAL_WATCHDOG:
2756 if (pi->phycal_tempdelta) {
2757 nphy_currtemp = wlc_phy_tempsense_nphy(pi);
2758 delta_temp =
2759 (nphy_currtemp > pi->nphy_lastcal_temp) ?
2760 nphy_currtemp - pi->nphy_lastcal_temp :
2761 pi->nphy_lastcal_temp - nphy_currtemp;
2762
2763 if ((delta_temp < (s16) pi->phycal_tempdelta) &&
2764 (pi->nphy_txiqlocal_chanspec ==
2765 pi->radio_chanspec))
2766 do_periodic_cal = false;
2767 else
2768 pi->nphy_lastcal_temp = nphy_currtemp;
2769 }
2770
2771 if (do_periodic_cal) {
2772 if (pi->nphy_perical == PHY_PERICAL_MPHASE) {
2773 if (!PHY_PERICAL_MPHASE_PENDING(pi))
2774 wlc_phy_cal_perical_mphase_schedule(
2775 pi,
2776 PHY_PERICAL_WDOG_DELAY);
2777 } else if (pi->nphy_perical == PHY_PERICAL_SPHASE)
2778 wlc_phy_cal_perical_nphy_run(pi,
2779 PHY_PERICAL_AUTO);
2780 }
2781 break;
2782 default:
2783 break;
2784 }
2785}
2786
2787void wlc_phy_cal_perical_mphase_restart(struct brcms_phy *pi)
2788{
2789 pi->mphase_cal_phase_id = MPHASE_CAL_STATE_INIT;
2790 pi->mphase_txcal_cmdidx = 0;
2791}
2792
2793u8 wlc_phy_nbits(s32 value)
2794{
2795 s32 abs_val;
2796 u8 nbits = 0;
2797
2798 abs_val = abs(value);
2799 while ((abs_val >> nbits) > 0)
2800 nbits++;
2801
2802 return nbits;
2803}
2804
2805void wlc_phy_stf_chain_init(struct brcms_phy_pub *pih, u8 txchain, u8 rxchain)
2806{
2807 struct brcms_phy *pi = container_of(pih, struct brcms_phy, pubpi_ro);
2808
2809 pi->sh->hw_phytxchain = txchain;
2810 pi->sh->hw_phyrxchain = rxchain;
2811 pi->sh->phytxchain = txchain;
2812 pi->sh->phyrxchain = rxchain;
2813 pi->pubpi.phy_corenum = (u8)hweight8(pi->sh->phyrxchain);
2814}
2815
2816void wlc_phy_stf_chain_set(struct brcms_phy_pub *pih, u8 txchain, u8 rxchain)
2817{
2818 struct brcms_phy *pi = container_of(pih, struct brcms_phy, pubpi_ro);
2819
2820 pi->sh->phytxchain = txchain;
2821
2822 if (ISNPHY(pi))
2823 wlc_phy_rxcore_setstate_nphy(pih, rxchain);
2824
2825 pi->pubpi.phy_corenum = (u8)hweight8(pi->sh->phyrxchain);
2826}
2827
2828void wlc_phy_stf_chain_get(struct brcms_phy_pub *pih, u8 *txchain, u8 *rxchain)
2829{
2830 struct brcms_phy *pi = container_of(pih, struct brcms_phy, pubpi_ro);
2831
2832 *txchain = pi->sh->phytxchain;
2833 *rxchain = pi->sh->phyrxchain;
2834}
2835
2836u8 wlc_phy_stf_chain_active_get(struct brcms_phy_pub *pih)
2837{
2838 s16 nphy_currtemp;
2839 u8 active_bitmap;
2840 struct brcms_phy *pi = container_of(pih, struct brcms_phy, pubpi_ro);
2841
2842 active_bitmap = (pi->phy_txcore_heatedup) ? 0x31 : 0x33;
2843
2844 if (!pi->watchdog_override)
2845 return active_bitmap;
2846
2847 if (NREV_GE(pi->pubpi.phy_rev, 6)) {
2848 wlapi_suspend_mac_and_wait(pi->sh->physhim);
2849 nphy_currtemp = wlc_phy_tempsense_nphy(pi);
2850 wlapi_enable_mac(pi->sh->physhim);
2851
2852 if (!pi->phy_txcore_heatedup) {
2853 if (nphy_currtemp >= pi->phy_txcore_disable_temp) {
2854 active_bitmap &= 0xFD;
2855 pi->phy_txcore_heatedup = true;
2856 }
2857 } else {
2858 if (nphy_currtemp <= pi->phy_txcore_enable_temp) {
2859 active_bitmap |= 0x2;
2860 pi->phy_txcore_heatedup = false;
2861 }
2862 }
2863 }
2864
2865 return active_bitmap;
2866}
2867
2868s8 wlc_phy_stf_ssmode_get(struct brcms_phy_pub *pih, u16 chanspec)
2869{
2870 struct brcms_phy *pi = container_of(pih, struct brcms_phy, pubpi_ro);
2871 u8 siso_mcs_id, cdd_mcs_id;
2872
2873 siso_mcs_id =
2874 (CHSPEC_IS40(chanspec)) ? TXP_FIRST_MCS_40_SISO :
2875 TXP_FIRST_MCS_20_SISO;
2876 cdd_mcs_id =
2877 (CHSPEC_IS40(chanspec)) ? TXP_FIRST_MCS_40_CDD :
2878 TXP_FIRST_MCS_20_CDD;
2879
2880 if (pi->tx_power_target[siso_mcs_id] >
2881 (pi->tx_power_target[cdd_mcs_id] + 12))
2882 return PHY_TXC1_MODE_SISO;
2883 else
2884 return PHY_TXC1_MODE_CDD;
2885}
2886
2887const u8 *wlc_phy_get_ofdm_rate_lookup(void)
2888{
2889 return ofdm_rate_lookup;
2890}
2891
2892void wlc_lcnphy_epa_switch(struct brcms_phy *pi, bool mode)
2893{
2894 if ((pi->sh->chip == BCMA_CHIP_ID_BCM4313) &&
2895 (pi->sh->boardflags & BFL_FEM)) {
2896 if (mode) {
2897 u16 txant = 0;
2898 txant = wlapi_bmac_get_txant(pi->sh->physhim);
2899 if (txant == 1) {
2900 mod_phy_reg(pi, 0x44d, (0x1 << 2), (1) << 2);
2901
2902 mod_phy_reg(pi, 0x44c, (0x1 << 2), (1) << 2);
2903
2904 }
2905
2906 bcma_chipco_gpio_control(&pi->d11core->bus->drv_cc,
2907 0x0, 0x0);
2908 bcma_chipco_gpio_out(&pi->d11core->bus->drv_cc,
2909 ~0x40, 0x40);
2910 bcma_chipco_gpio_outen(&pi->d11core->bus->drv_cc,
2911 ~0x40, 0x40);
2912 } else {
2913 mod_phy_reg(pi, 0x44c, (0x1 << 2), (0) << 2);
2914
2915 mod_phy_reg(pi, 0x44d, (0x1 << 2), (0) << 2);
2916
2917 bcma_chipco_gpio_out(&pi->d11core->bus->drv_cc,
2918 ~0x40, 0x00);
2919 bcma_chipco_gpio_outen(&pi->d11core->bus->drv_cc,
2920 ~0x40, 0x00);
2921 bcma_chipco_gpio_control(&pi->d11core->bus->drv_cc,
2922 0x0, 0x40);
2923 }
2924 }
2925}
2926
2927void wlc_phy_ldpc_override_set(struct brcms_phy_pub *ppi, bool ldpc)
2928{
2929 return;
2930}
2931
2932void
2933wlc_phy_get_pwrdet_offsets(struct brcms_phy *pi, s8 *cckoffset, s8 *ofdmoffset)
2934{
2935 *cckoffset = 0;
2936 *ofdmoffset = 0;
2937}
2938
2939s8 wlc_phy_upd_rssi_offset(struct brcms_phy *pi, s8 rssi, u16 chanspec)
2940{
2941
2942 return rssi;
2943}
2944
2945bool wlc_phy_txpower_ipa_ison(struct brcms_phy_pub *ppi)
2946{
2947 struct brcms_phy *pi = container_of(ppi, struct brcms_phy, pubpi_ro);
2948
2949 if (ISNPHY(pi))
2950 return wlc_phy_n_txpower_ipa_ison(pi);
2951 else
2952 return false;
2953}
2954