2 * Copyright (c) 2010-2011 Atheros Communications Inc.
4 * Permission to use, copy, modify, and/or distribute this software for any
5 * purpose with or without fee is hereby granted, provided that the above
6 * copyright notice and this permission notice appear in all copies.
8 * THE SOFTWARE IS PROVIDED "AS IS" AND THE AUTHOR DISCLAIMS ALL WARRANTIES
9 * WITH REGARD TO THIS SOFTWARE INCLUDING ALL IMPLIED WARRANTIES OF
10 * MERCHANTABILITY AND FITNESS. IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR
11 * ANY SPECIAL, DIRECT, INDIRECT, OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES
12 * WHATSOEVER RESULTING FROM LOSS OF USE, DATA OR PROFITS, WHETHER IN AN
13 * ACTION OF CONTRACT, NEGLIGENCE OR OTHER TORTIOUS ACTION, ARISING OUT OF
14 * OR IN CONNECTION WITH THE USE OR PERFORMANCE OF THIS SOFTWARE.
17 #include <asm/unaligned.h>
18 #include <linux/kernel.h>
20 #include "ar9003_phy.h"
21 #include "ar9003_eeprom.h"
22 #include "ar9003_mci.h"
24 #define COMP_HDR_LEN 4
25 #define COMP_CKSUM_LEN 2
27 #define LE16(x) cpu_to_le16(x)
28 #define LE32(x) cpu_to_le32(x)
30 /* Local defines to distinguish between extension and control CTL's */
31 #define EXT_ADDITIVE (0x8000)
32 #define CTL_11A_EXT (CTL_11A | EXT_ADDITIVE)
33 #define CTL_11G_EXT (CTL_11G | EXT_ADDITIVE)
34 #define CTL_11B_EXT (CTL_11B | EXT_ADDITIVE)
36 #define SUB_NUM_CTL_MODES_AT_5G_40 2 /* excluding HT40, EXT-OFDM */
37 #define SUB_NUM_CTL_MODES_AT_2G_40 3 /* excluding HT40, EXT-OFDM, EXT-CCK */
39 #define CTL(_tpower, _flag) ((_tpower) | ((_flag) << 6))
41 #define EEPROM_DATA_LEN_9485 1088
43 static int ar9003_hw_power_interpolate(int32_t x
,
44 int32_t *px
, int32_t *py
, u_int16_t np
);
46 static const struct ar9300_eeprom ar9300_default
= {
49 .macAddr
= {0, 2, 3, 4, 5, 6},
50 .custData
= {0, 0, 0, 0, 0, 0, 0, 0, 0, 0,
51 0, 0, 0, 0, 0, 0, 0, 0, 0, 0},
53 .regDmn
= { LE16(0), LE16(0x1f) },
54 .txrxMask
= 0x77, /* 4 bits tx and 4 bits rx */
56 .opFlags
= AR5416_OPFLAGS_11G
| AR5416_OPFLAGS_11A
,
57 .eepMisc
= AR9300_EEPMISC_LITTLE_ENDIAN
,
60 .blueToothOptions
= 0,
62 .deviceType
= 5, /* takes lower byte in eeprom location */
63 .pwrTableOffset
= AR9300_PWR_TABLE_OFFSET
,
64 .params_for_tuning_caps
= {0, 0},
65 .featureEnable
= 0x0c,
67 * bit0 - enable tx temp comp - disabled
68 * bit1 - enable tx volt comp - disabled
69 * bit2 - enable fastClock - enabled
70 * bit3 - enable doubling - enabled
71 * bit4 - enable internal regulator - disabled
72 * bit5 - enable pa predistortion - disabled
74 .miscConfiguration
= 0, /* bit0 - turn down drivestrength */
75 .eepromWriteEnableGpio
= 3,
78 .rxBandSelectGpio
= 0xff,
83 /* ar9300_modal_eep_header 2g */
84 /* 4 idle,t1,t2,b(4 bits per setting) */
85 .antCtrlCommon
= LE32(0x110),
86 /* 4 ra1l1, ra2l1, ra1l2, ra2l2, ra12 */
87 .antCtrlCommon2
= LE32(0x22222),
90 * antCtrlChain[AR9300_MAX_CHAINS]; 6 idle, t, r,
91 * rx1, rx12, b (2 bits each)
93 .antCtrlChain
= { LE16(0x150), LE16(0x150), LE16(0x150) },
96 * xatten1DB[AR9300_MAX_CHAINS]; 3 xatten1_db
97 * for ar9280 (0xa20c/b20c 5:0)
99 .xatten1DB
= {0, 0, 0},
102 * xatten1Margin[AR9300_MAX_CHAINS]; 3 xatten1_margin
103 * for ar9280 (0xa20c/b20c 16:12
105 .xatten1Margin
= {0, 0, 0},
110 * spurChans[OSPREY_EEPROM_MODAL_SPURS]; spur
111 * channels in usual fbin coding format
113 .spurChans
= {0, 0, 0, 0, 0},
116 * noiseFloorThreshCh[AR9300_MAX_CHAINS]; 3 Check
117 * if the register is per chain
119 .noiseFloorThreshCh
= {-1, 0, 0},
120 .reserved
= {0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0},
123 .txFrameToDataStart
= 0x0e,
124 .txFrameToPaOn
= 0x0e,
125 .txClip
= 3, /* 4 bits tx_clip, 4 bits dac_scale_cck */
127 .switchSettling
= 0x2c,
128 .adcDesiredSize
= -30,
131 .txFrameToXpaOn
= 0xe,
133 .papdRateMaskHt20
= LE32(0x0cf0e0e0),
134 .papdRateMaskHt40
= LE32(0x6cf0e0e0),
136 .xlna_bias_strength
= 0,
142 .ant_div_control
= 0,
144 .tempslopextension
= {0, 0, 0, 0, 0, 0, 0, 0}
151 /* ar9300_cal_data_per_freq_op_loop 2g */
153 { {0, 0, 0, 0, 0, 0}, {0, 0, 0, 0, 0, 0}, {0, 0, 0, 0, 0, 0} },
154 { {0, 0, 0, 0, 0, 0}, {0, 0, 0, 0, 0, 0}, {0, 0, 0, 0, 0, 0} },
155 { {0, 0, 0, 0, 0, 0}, {0, 0, 0, 0, 0, 0}, {0, 0, 0, 0, 0, 0} },
157 .calTarget_freqbin_Cck
= {
161 .calTarget_freqbin_2G
= {
166 .calTarget_freqbin_2GHT20
= {
171 .calTarget_freqbin_2GHT40
= {
176 .calTargetPowerCck
= {
177 /* 1L-5L,5S,11L,11S */
178 { {36, 36, 36, 36} },
179 { {36, 36, 36, 36} },
181 .calTargetPower2G
= {
183 { {32, 32, 28, 24} },
184 { {32, 32, 28, 24} },
185 { {32, 32, 28, 24} },
187 .calTargetPower2GHT20
= {
188 { {32, 32, 32, 32, 28, 20, 32, 32, 28, 20, 32, 32, 28, 20} },
189 { {32, 32, 32, 32, 28, 20, 32, 32, 28, 20, 32, 32, 28, 20} },
190 { {32, 32, 32, 32, 28, 20, 32, 32, 28, 20, 32, 32, 28, 20} },
192 .calTargetPower2GHT40
= {
193 { {32, 32, 32, 32, 28, 20, 32, 32, 28, 20, 32, 32, 28, 20} },
194 { {32, 32, 32, 32, 28, 20, 32, 32, 28, 20, 32, 32, 28, 20} },
195 { {32, 32, 32, 32, 28, 20, 32, 32, 28, 20, 32, 32, 28, 20} },
198 0x11, 0x12, 0x15, 0x17, 0x41, 0x42,
199 0x45, 0x47, 0x31, 0x32, 0x35, 0x37,
229 /* Data[4].ctlEdges[0].bChannel */ FREQ2FBIN(2412, 1),
230 /* Data[4].ctlEdges[1].bChannel */ FREQ2FBIN(2417, 1),
231 /* Data[4].ctlEdges[2].bChannel */ FREQ2FBIN(2472, 1),
232 /* Data[4].ctlEdges[3].bChannel */ FREQ2FBIN(2484, 1),
236 /* Data[5].ctlEdges[0].bChannel */ FREQ2FBIN(2412, 1),
237 /* Data[5].ctlEdges[1].bChannel */ FREQ2FBIN(2417, 1),
238 /* Data[5].ctlEdges[2].bChannel */ FREQ2FBIN(2472, 1),
243 /* Data[6].ctlEdges[0].bChannel */ FREQ2FBIN(2412, 1),
244 /* Data[6].ctlEdges[1].bChannel */ FREQ2FBIN(2417, 1),
250 /* Data[7].ctlEdges[0].bChannel */ FREQ2FBIN(2422, 1),
251 /* Data[7].ctlEdges[1].bChannel */ FREQ2FBIN(2427, 1),
252 /* Data[7].ctlEdges[2].bChannel */ FREQ2FBIN(2447, 1),
253 /* Data[7].ctlEdges[3].bChannel */ FREQ2FBIN(2462, 1),
257 /* Data[8].ctlEdges[0].bChannel */ FREQ2FBIN(2412, 1),
258 /* Data[8].ctlEdges[1].bChannel */ FREQ2FBIN(2417, 1),
259 /* Data[8].ctlEdges[2].bChannel */ FREQ2FBIN(2472, 1),
263 /* Data[9].ctlEdges[0].bChannel */ FREQ2FBIN(2412, 1),
264 /* Data[9].ctlEdges[1].bChannel */ FREQ2FBIN(2417, 1),
265 /* Data[9].ctlEdges[2].bChannel */ FREQ2FBIN(2472, 1),
270 /* Data[10].ctlEdges[0].bChannel */ FREQ2FBIN(2412, 1),
271 /* Data[10].ctlEdges[1].bChannel */ FREQ2FBIN(2417, 1),
272 /* Data[10].ctlEdges[2].bChannel */ FREQ2FBIN(2472, 1),
277 /* Data[11].ctlEdges[0].bChannel */ FREQ2FBIN(2422, 1),
278 /* Data[11].ctlEdges[1].bChannel */ FREQ2FBIN(2427, 1),
279 /* Data[11].ctlEdges[2].bChannel */ FREQ2FBIN(2447, 1),
280 /* Data[11].ctlEdges[3].bChannel */ FREQ2FBIN(2462, 1),
284 { { CTL(60, 0), CTL(60, 1), CTL(60, 0), CTL(60, 0) } },
285 { { CTL(60, 0), CTL(60, 1), CTL(60, 0), CTL(60, 0) } },
286 { { CTL(60, 1), CTL(60, 0), CTL(60, 0), CTL(60, 1) } },
288 { { CTL(60, 1), CTL(60, 0), CTL(60, 0), CTL(60, 0) } },
289 { { CTL(60, 0), CTL(60, 1), CTL(60, 0), CTL(60, 0) } },
290 { { CTL(60, 0), CTL(60, 1), CTL(60, 0), CTL(60, 0) } },
292 { { CTL(60, 0), CTL(60, 1), CTL(60, 1), CTL(60, 0) } },
293 { { CTL(60, 0), CTL(60, 1), CTL(60, 0), CTL(60, 0) } },
294 { { CTL(60, 0), CTL(60, 1), CTL(60, 0), CTL(60, 0) } },
296 { { CTL(60, 0), CTL(60, 1), CTL(60, 0), CTL(60, 0) } },
297 { { CTL(60, 0), CTL(60, 1), CTL(60, 1), CTL(60, 1) } },
298 { { CTL(60, 0), CTL(60, 1), CTL(60, 1), CTL(60, 1) } },
301 /* 4 idle,t1,t2,b (4 bits per setting) */
302 .antCtrlCommon
= LE32(0x110),
303 /* 4 ra1l1, ra2l1, ra1l2,ra2l2,ra12 */
304 .antCtrlCommon2
= LE32(0x22222),
305 /* antCtrlChain 6 idle, t,r,rx1,rx12,b (2 bits each) */
307 LE16(0x000), LE16(0x000), LE16(0x000),
309 /* xatten1DB 3 xatten1_db for AR9280 (0xa20c/b20c 5:0) */
310 .xatten1DB
= {0, 0, 0},
313 * xatten1Margin[AR9300_MAX_CHAINS]; 3 xatten1_margin
314 * for merlin (0xa20c/b20c 16:12
316 .xatten1Margin
= {0, 0, 0},
319 /* spurChans spur channels in usual fbin coding format */
320 .spurChans
= {0, 0, 0, 0, 0},
321 /* noiseFloorThreshCh Check if the register is per chain */
322 .noiseFloorThreshCh
= {-1, 0, 0},
323 .reserved
= {0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0},
326 .txFrameToDataStart
= 0x0e,
327 .txFrameToPaOn
= 0x0e,
328 .txClip
= 3, /* 4 bits tx_clip, 4 bits dac_scale_cck */
330 .switchSettling
= 0x2d,
331 .adcDesiredSize
= -30,
334 .txFrameToXpaOn
= 0xe,
336 .papdRateMaskHt20
= LE32(0x0c80c080),
337 .papdRateMaskHt40
= LE32(0x0080c080),
339 .xlna_bias_strength
= 0,
347 .xatten1DBLow
= {0, 0, 0},
348 .xatten1MarginLow
= {0, 0, 0},
349 .xatten1DBHigh
= {0, 0, 0},
350 .xatten1MarginHigh
= {0, 0, 0}
395 .calTarget_freqbin_5G
= {
405 .calTarget_freqbin_5GHT20
= {
415 .calTarget_freqbin_5GHT40
= {
425 .calTargetPower5G
= {
427 { {20, 20, 20, 10} },
428 { {20, 20, 20, 10} },
429 { {20, 20, 20, 10} },
430 { {20, 20, 20, 10} },
431 { {20, 20, 20, 10} },
432 { {20, 20, 20, 10} },
433 { {20, 20, 20, 10} },
434 { {20, 20, 20, 10} },
436 .calTargetPower5GHT20
= {
438 * 0_8_16,1-3_9-11_17-19,
439 * 4,5,6,7,12,13,14,15,20,21,22,23
441 { {20, 20, 10, 10, 0, 0, 10, 10, 0, 0, 10, 10, 0, 0} },
442 { {20, 20, 10, 10, 0, 0, 10, 10, 0, 0, 10, 10, 0, 0} },
443 { {20, 20, 10, 10, 0, 0, 10, 10, 0, 0, 10, 10, 0, 0} },
444 { {20, 20, 10, 10, 0, 0, 10, 10, 0, 0, 10, 10, 0, 0} },
445 { {20, 20, 10, 10, 0, 0, 10, 10, 0, 0, 10, 10, 0, 0} },
446 { {20, 20, 10, 10, 0, 0, 10, 10, 0, 0, 10, 10, 0, 0} },
447 { {20, 20, 10, 10, 0, 0, 10, 10, 0, 0, 10, 10, 0, 0} },
448 { {20, 20, 10, 10, 0, 0, 10, 10, 0, 0, 10, 10, 0, 0} },
450 .calTargetPower5GHT40
= {
452 * 0_8_16,1-3_9-11_17-19,
453 * 4,5,6,7,12,13,14,15,20,21,22,23
455 { {20, 20, 10, 10, 0, 0, 10, 10, 0, 0, 10, 10, 0, 0} },
456 { {20, 20, 10, 10, 0, 0, 10, 10, 0, 0, 10, 10, 0, 0} },
457 { {20, 20, 10, 10, 0, 0, 10, 10, 0, 0, 10, 10, 0, 0} },
458 { {20, 20, 10, 10, 0, 0, 10, 10, 0, 0, 10, 10, 0, 0} },
459 { {20, 20, 10, 10, 0, 0, 10, 10, 0, 0, 10, 10, 0, 0} },
460 { {20, 20, 10, 10, 0, 0, 10, 10, 0, 0, 10, 10, 0, 0} },
461 { {20, 20, 10, 10, 0, 0, 10, 10, 0, 0, 10, 10, 0, 0} },
462 { {20, 20, 10, 10, 0, 0, 10, 10, 0, 0, 10, 10, 0, 0} },
465 0x10, 0x16, 0x18, 0x40, 0x46,
466 0x48, 0x30, 0x36, 0x38
470 /* Data[0].ctlEdges[0].bChannel */ FREQ2FBIN(5180, 0),
471 /* Data[0].ctlEdges[1].bChannel */ FREQ2FBIN(5260, 0),
472 /* Data[0].ctlEdges[2].bChannel */ FREQ2FBIN(5280, 0),
473 /* Data[0].ctlEdges[3].bChannel */ FREQ2FBIN(5500, 0),
474 /* Data[0].ctlEdges[4].bChannel */ FREQ2FBIN(5600, 0),
475 /* Data[0].ctlEdges[5].bChannel */ FREQ2FBIN(5700, 0),
476 /* Data[0].ctlEdges[6].bChannel */ FREQ2FBIN(5745, 0),
477 /* Data[0].ctlEdges[7].bChannel */ FREQ2FBIN(5825, 0)
480 /* Data[1].ctlEdges[0].bChannel */ FREQ2FBIN(5180, 0),
481 /* Data[1].ctlEdges[1].bChannel */ FREQ2FBIN(5260, 0),
482 /* Data[1].ctlEdges[2].bChannel */ FREQ2FBIN(5280, 0),
483 /* Data[1].ctlEdges[3].bChannel */ FREQ2FBIN(5500, 0),
484 /* Data[1].ctlEdges[4].bChannel */ FREQ2FBIN(5520, 0),
485 /* Data[1].ctlEdges[5].bChannel */ FREQ2FBIN(5700, 0),
486 /* Data[1].ctlEdges[6].bChannel */ FREQ2FBIN(5745, 0),
487 /* Data[1].ctlEdges[7].bChannel */ FREQ2FBIN(5825, 0)
491 /* Data[2].ctlEdges[0].bChannel */ FREQ2FBIN(5190, 0),
492 /* Data[2].ctlEdges[1].bChannel */ FREQ2FBIN(5230, 0),
493 /* Data[2].ctlEdges[2].bChannel */ FREQ2FBIN(5270, 0),
494 /* Data[2].ctlEdges[3].bChannel */ FREQ2FBIN(5310, 0),
495 /* Data[2].ctlEdges[4].bChannel */ FREQ2FBIN(5510, 0),
496 /* Data[2].ctlEdges[5].bChannel */ FREQ2FBIN(5550, 0),
497 /* Data[2].ctlEdges[6].bChannel */ FREQ2FBIN(5670, 0),
498 /* Data[2].ctlEdges[7].bChannel */ FREQ2FBIN(5755, 0)
502 /* Data[3].ctlEdges[0].bChannel */ FREQ2FBIN(5180, 0),
503 /* Data[3].ctlEdges[1].bChannel */ FREQ2FBIN(5200, 0),
504 /* Data[3].ctlEdges[2].bChannel */ FREQ2FBIN(5260, 0),
505 /* Data[3].ctlEdges[3].bChannel */ FREQ2FBIN(5320, 0),
506 /* Data[3].ctlEdges[4].bChannel */ FREQ2FBIN(5500, 0),
507 /* Data[3].ctlEdges[5].bChannel */ FREQ2FBIN(5700, 0),
508 /* Data[3].ctlEdges[6].bChannel */ 0xFF,
509 /* Data[3].ctlEdges[7].bChannel */ 0xFF,
513 /* Data[4].ctlEdges[0].bChannel */ FREQ2FBIN(5180, 0),
514 /* Data[4].ctlEdges[1].bChannel */ FREQ2FBIN(5260, 0),
515 /* Data[4].ctlEdges[2].bChannel */ FREQ2FBIN(5500, 0),
516 /* Data[4].ctlEdges[3].bChannel */ FREQ2FBIN(5700, 0),
517 /* Data[4].ctlEdges[4].bChannel */ 0xFF,
518 /* Data[4].ctlEdges[5].bChannel */ 0xFF,
519 /* Data[4].ctlEdges[6].bChannel */ 0xFF,
520 /* Data[4].ctlEdges[7].bChannel */ 0xFF,
524 /* Data[5].ctlEdges[0].bChannel */ FREQ2FBIN(5190, 0),
525 /* Data[5].ctlEdges[1].bChannel */ FREQ2FBIN(5270, 0),
526 /* Data[5].ctlEdges[2].bChannel */ FREQ2FBIN(5310, 0),
527 /* Data[5].ctlEdges[3].bChannel */ FREQ2FBIN(5510, 0),
528 /* Data[5].ctlEdges[4].bChannel */ FREQ2FBIN(5590, 0),
529 /* Data[5].ctlEdges[5].bChannel */ FREQ2FBIN(5670, 0),
530 /* Data[5].ctlEdges[6].bChannel */ 0xFF,
531 /* Data[5].ctlEdges[7].bChannel */ 0xFF
535 /* Data[6].ctlEdges[0].bChannel */ FREQ2FBIN(5180, 0),
536 /* Data[6].ctlEdges[1].bChannel */ FREQ2FBIN(5200, 0),
537 /* Data[6].ctlEdges[2].bChannel */ FREQ2FBIN(5220, 0),
538 /* Data[6].ctlEdges[3].bChannel */ FREQ2FBIN(5260, 0),
539 /* Data[6].ctlEdges[4].bChannel */ FREQ2FBIN(5500, 0),
540 /* Data[6].ctlEdges[5].bChannel */ FREQ2FBIN(5600, 0),
541 /* Data[6].ctlEdges[6].bChannel */ FREQ2FBIN(5700, 0),
542 /* Data[6].ctlEdges[7].bChannel */ FREQ2FBIN(5745, 0)
546 /* Data[7].ctlEdges[0].bChannel */ FREQ2FBIN(5180, 0),
547 /* Data[7].ctlEdges[1].bChannel */ FREQ2FBIN(5260, 0),
548 /* Data[7].ctlEdges[2].bChannel */ FREQ2FBIN(5320, 0),
549 /* Data[7].ctlEdges[3].bChannel */ FREQ2FBIN(5500, 0),
550 /* Data[7].ctlEdges[4].bChannel */ FREQ2FBIN(5560, 0),
551 /* Data[7].ctlEdges[5].bChannel */ FREQ2FBIN(5700, 0),
552 /* Data[7].ctlEdges[6].bChannel */ FREQ2FBIN(5745, 0),
553 /* Data[7].ctlEdges[7].bChannel */ FREQ2FBIN(5825, 0)
557 /* Data[8].ctlEdges[0].bChannel */ FREQ2FBIN(5190, 0),
558 /* Data[8].ctlEdges[1].bChannel */ FREQ2FBIN(5230, 0),
559 /* Data[8].ctlEdges[2].bChannel */ FREQ2FBIN(5270, 0),
560 /* Data[8].ctlEdges[3].bChannel */ FREQ2FBIN(5510, 0),
561 /* Data[8].ctlEdges[4].bChannel */ FREQ2FBIN(5550, 0),
562 /* Data[8].ctlEdges[5].bChannel */ FREQ2FBIN(5670, 0),
563 /* Data[8].ctlEdges[6].bChannel */ FREQ2FBIN(5755, 0),
564 /* Data[8].ctlEdges[7].bChannel */ FREQ2FBIN(5795, 0)
570 CTL(60, 1), CTL(60, 1), CTL(60, 1), CTL(60, 1),
571 CTL(60, 1), CTL(60, 1), CTL(60, 1), CTL(60, 0),
576 CTL(60, 1), CTL(60, 1), CTL(60, 1), CTL(60, 1),
577 CTL(60, 1), CTL(60, 1), CTL(60, 1), CTL(60, 0),
582 CTL(60, 0), CTL(60, 1), CTL(60, 0), CTL(60, 1),
583 CTL(60, 1), CTL(60, 1), CTL(60, 1), CTL(60, 1),
588 CTL(60, 0), CTL(60, 1), CTL(60, 1), CTL(60, 0),
589 CTL(60, 1), CTL(60, 0), CTL(60, 0), CTL(60, 0),
594 CTL(60, 1), CTL(60, 1), CTL(60, 1), CTL(60, 0),
595 CTL(60, 0), CTL(60, 0), CTL(60, 0), CTL(60, 0),
600 CTL(60, 1), CTL(60, 1), CTL(60, 1), CTL(60, 1),
601 CTL(60, 1), CTL(60, 0), CTL(60, 0), CTL(60, 0),
606 CTL(60, 1), CTL(60, 1), CTL(60, 1), CTL(60, 1),
607 CTL(60, 1), CTL(60, 1), CTL(60, 1), CTL(60, 1),
612 CTL(60, 1), CTL(60, 1), CTL(60, 0), CTL(60, 1),
613 CTL(60, 1), CTL(60, 1), CTL(60, 1), CTL(60, 0),
618 CTL(60, 1), CTL(60, 0), CTL(60, 1), CTL(60, 1),
619 CTL(60, 1), CTL(60, 1), CTL(60, 0), CTL(60, 1),
625 static const struct ar9300_eeprom ar9300_x113
= {
627 .templateVersion
= 6,
628 .macAddr
= {0x00, 0x03, 0x7f, 0x0, 0x0, 0x0},
629 .custData
= {"x113-023-f0000"},
631 .regDmn
= { LE16(0), LE16(0x1f) },
632 .txrxMask
= 0x77, /* 4 bits tx and 4 bits rx */
634 .opFlags
= AR5416_OPFLAGS_11A
,
635 .eepMisc
= AR9300_EEPMISC_LITTLE_ENDIAN
,
638 .blueToothOptions
= 0,
640 .deviceType
= 5, /* takes lower byte in eeprom location */
641 .pwrTableOffset
= AR9300_PWR_TABLE_OFFSET
,
642 .params_for_tuning_caps
= {0, 0},
643 .featureEnable
= 0x0d,
645 * bit0 - enable tx temp comp - disabled
646 * bit1 - enable tx volt comp - disabled
647 * bit2 - enable fastClock - enabled
648 * bit3 - enable doubling - enabled
649 * bit4 - enable internal regulator - disabled
650 * bit5 - enable pa predistortion - disabled
652 .miscConfiguration
= 0, /* bit0 - turn down drivestrength */
653 .eepromWriteEnableGpio
= 6,
654 .wlanDisableGpio
= 0,
656 .rxBandSelectGpio
= 0xff,
661 /* ar9300_modal_eep_header 2g */
662 /* 4 idle,t1,t2,b(4 bits per setting) */
663 .antCtrlCommon
= LE32(0x110),
664 /* 4 ra1l1, ra2l1, ra1l2, ra2l2, ra12 */
665 .antCtrlCommon2
= LE32(0x44444),
668 * antCtrlChain[AR9300_MAX_CHAINS]; 6 idle, t, r,
669 * rx1, rx12, b (2 bits each)
671 .antCtrlChain
= { LE16(0x150), LE16(0x150), LE16(0x150) },
674 * xatten1DB[AR9300_MAX_CHAINS]; 3 xatten1_db
675 * for ar9280 (0xa20c/b20c 5:0)
677 .xatten1DB
= {0, 0, 0},
680 * xatten1Margin[AR9300_MAX_CHAINS]; 3 xatten1_margin
681 * for ar9280 (0xa20c/b20c 16:12
683 .xatten1Margin
= {0, 0, 0},
688 * spurChans[OSPREY_EEPROM_MODAL_SPURS]; spur
689 * channels in usual fbin coding format
691 .spurChans
= {FREQ2FBIN(2464, 1), 0, 0, 0, 0},
694 * noiseFloorThreshCh[AR9300_MAX_CHAINS]; 3 Check
695 * if the register is per chain
697 .noiseFloorThreshCh
= {-1, 0, 0},
698 .reserved
= {0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0},
701 .txFrameToDataStart
= 0x0e,
702 .txFrameToPaOn
= 0x0e,
703 .txClip
= 3, /* 4 bits tx_clip, 4 bits dac_scale_cck */
705 .switchSettling
= 0x2c,
706 .adcDesiredSize
= -30,
709 .txFrameToXpaOn
= 0xe,
711 .papdRateMaskHt20
= LE32(0x0c80c080),
712 .papdRateMaskHt40
= LE32(0x0080c080),
714 .xlna_bias_strength
= 0,
720 .ant_div_control
= 0,
722 .tempslopextension
= {0, 0, 0, 0, 0, 0, 0, 0}
729 /* ar9300_cal_data_per_freq_op_loop 2g */
731 { {0, 0, 0, 0, 0, 0}, {0, 0, 0, 0, 0, 0}, {0, 0, 0, 0, 0, 0} },
732 { {0, 0, 0, 0, 0, 0}, {0, 0, 0, 0, 0, 0}, {0, 0, 0, 0, 0, 0} },
733 { {0, 0, 0, 0, 0, 0}, {0, 0, 0, 0, 0, 0}, {0, 0, 0, 0, 0, 0} },
735 .calTarget_freqbin_Cck
= {
739 .calTarget_freqbin_2G
= {
744 .calTarget_freqbin_2GHT20
= {
749 .calTarget_freqbin_2GHT40
= {
754 .calTargetPowerCck
= {
755 /* 1L-5L,5S,11L,11S */
756 { {34, 34, 34, 34} },
757 { {34, 34, 34, 34} },
759 .calTargetPower2G
= {
761 { {34, 34, 32, 32} },
762 { {34, 34, 32, 32} },
763 { {34, 34, 32, 32} },
765 .calTargetPower2GHT20
= {
766 { {32, 32, 32, 32, 32, 28, 32, 32, 30, 28, 0, 0, 0, 0} },
767 { {32, 32, 32, 32, 32, 28, 32, 32, 30, 28, 0, 0, 0, 0} },
768 { {32, 32, 32, 32, 32, 28, 32, 32, 30, 28, 0, 0, 0, 0} },
770 .calTargetPower2GHT40
= {
771 { {30, 30, 30, 30, 30, 28, 30, 30, 28, 26, 0, 0, 0, 0} },
772 { {30, 30, 30, 30, 30, 28, 30, 30, 28, 26, 0, 0, 0, 0} },
773 { {30, 30, 30, 30, 30, 28, 30, 30, 28, 26, 0, 0, 0, 0} },
776 0x11, 0x12, 0x15, 0x17, 0x41, 0x42,
777 0x45, 0x47, 0x31, 0x32, 0x35, 0x37,
807 /* Data[4].ctlEdges[0].bChannel */ FREQ2FBIN(2412, 1),
808 /* Data[4].ctlEdges[1].bChannel */ FREQ2FBIN(2417, 1),
809 /* Data[4].ctlEdges[2].bChannel */ FREQ2FBIN(2472, 1),
810 /* Data[4].ctlEdges[3].bChannel */ FREQ2FBIN(2484, 1),
814 /* Data[5].ctlEdges[0].bChannel */ FREQ2FBIN(2412, 1),
815 /* Data[5].ctlEdges[1].bChannel */ FREQ2FBIN(2417, 1),
816 /* Data[5].ctlEdges[2].bChannel */ FREQ2FBIN(2472, 1),
821 /* Data[6].ctlEdges[0].bChannel */ FREQ2FBIN(2412, 1),
822 /* Data[6].ctlEdges[1].bChannel */ FREQ2FBIN(2417, 1),
828 /* Data[7].ctlEdges[0].bChannel */ FREQ2FBIN(2422, 1),
829 /* Data[7].ctlEdges[1].bChannel */ FREQ2FBIN(2427, 1),
830 /* Data[7].ctlEdges[2].bChannel */ FREQ2FBIN(2447, 1),
831 /* Data[7].ctlEdges[3].bChannel */ FREQ2FBIN(2462, 1),
835 /* Data[8].ctlEdges[0].bChannel */ FREQ2FBIN(2412, 1),
836 /* Data[8].ctlEdges[1].bChannel */ FREQ2FBIN(2417, 1),
837 /* Data[8].ctlEdges[2].bChannel */ FREQ2FBIN(2472, 1),
841 /* Data[9].ctlEdges[0].bChannel */ FREQ2FBIN(2412, 1),
842 /* Data[9].ctlEdges[1].bChannel */ FREQ2FBIN(2417, 1),
843 /* Data[9].ctlEdges[2].bChannel */ FREQ2FBIN(2472, 1),
848 /* Data[10].ctlEdges[0].bChannel */ FREQ2FBIN(2412, 1),
849 /* Data[10].ctlEdges[1].bChannel */ FREQ2FBIN(2417, 1),
850 /* Data[10].ctlEdges[2].bChannel */ FREQ2FBIN(2472, 1),
855 /* Data[11].ctlEdges[0].bChannel */ FREQ2FBIN(2422, 1),
856 /* Data[11].ctlEdges[1].bChannel */ FREQ2FBIN(2427, 1),
857 /* Data[11].ctlEdges[2].bChannel */ FREQ2FBIN(2447, 1),
858 /* Data[11].ctlEdges[3].bChannel */ FREQ2FBIN(2462, 1),
862 { { CTL(60, 0), CTL(60, 1), CTL(60, 0), CTL(60, 0) } },
863 { { CTL(60, 0), CTL(60, 1), CTL(60, 0), CTL(60, 0) } },
864 { { CTL(60, 1), CTL(60, 0), CTL(60, 0), CTL(60, 1) } },
866 { { CTL(60, 1), CTL(60, 0), CTL(60, 0), CTL(60, 0) } },
867 { { CTL(60, 0), CTL(60, 1), CTL(60, 0), CTL(60, 0) } },
868 { { CTL(60, 0), CTL(60, 1), CTL(60, 0), CTL(60, 0) } },
870 { { CTL(60, 0), CTL(60, 1), CTL(60, 1), CTL(60, 0) } },
871 { { CTL(60, 0), CTL(60, 1), CTL(60, 0), CTL(60, 0) } },
872 { { CTL(60, 0), CTL(60, 1), CTL(60, 0), CTL(60, 0) } },
874 { { CTL(60, 0), CTL(60, 1), CTL(60, 0), CTL(60, 0) } },
875 { { CTL(60, 0), CTL(60, 1), CTL(60, 1), CTL(60, 1) } },
876 { { CTL(60, 0), CTL(60, 1), CTL(60, 1), CTL(60, 1) } },
879 /* 4 idle,t1,t2,b (4 bits per setting) */
880 .antCtrlCommon
= LE32(0x220),
881 /* 4 ra1l1, ra2l1, ra1l2,ra2l2,ra12 */
882 .antCtrlCommon2
= LE32(0x11111),
883 /* antCtrlChain 6 idle, t,r,rx1,rx12,b (2 bits each) */
885 LE16(0x150), LE16(0x150), LE16(0x150),
887 /* xatten1DB 3 xatten1_db for AR9280 (0xa20c/b20c 5:0) */
888 .xatten1DB
= {0, 0, 0},
891 * xatten1Margin[AR9300_MAX_CHAINS]; 3 xatten1_margin
892 * for merlin (0xa20c/b20c 16:12
894 .xatten1Margin
= {0, 0, 0},
897 /* spurChans spur channels in usual fbin coding format */
898 .spurChans
= {FREQ2FBIN(5500, 0), 0, 0, 0, 0},
899 /* noiseFloorThreshCh Check if the register is per chain */
900 .noiseFloorThreshCh
= {-1, 0, 0},
901 .reserved
= {0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0},
904 .txFrameToDataStart
= 0x0e,
905 .txFrameToPaOn
= 0x0e,
906 .txClip
= 3, /* 4 bits tx_clip, 4 bits dac_scale_cck */
908 .switchSettling
= 0x2d,
909 .adcDesiredSize
= -30,
912 .txFrameToXpaOn
= 0xe,
914 .papdRateMaskHt20
= LE32(0x0cf0e0e0),
915 .papdRateMaskHt40
= LE32(0x6cf0e0e0),
917 .xlna_bias_strength
= 0,
924 .tempSlopeHigh
= 105,
925 .xatten1DBLow
= {0, 0, 0},
926 .xatten1MarginLow
= {0, 0, 0},
927 .xatten1DBHigh
= {0, 0, 0},
928 .xatten1MarginHigh
= {0, 0, 0}
973 .calTarget_freqbin_5G
= {
983 .calTarget_freqbin_5GHT20
= {
993 .calTarget_freqbin_5GHT40
= {
1003 .calTargetPower5G
= {
1005 { {42, 40, 40, 34} },
1006 { {42, 40, 40, 34} },
1007 { {42, 40, 40, 34} },
1008 { {42, 40, 40, 34} },
1009 { {42, 40, 40, 34} },
1010 { {42, 40, 40, 34} },
1011 { {42, 40, 40, 34} },
1012 { {42, 40, 40, 34} },
1014 .calTargetPower5GHT20
= {
1016 * 0_8_16,1-3_9-11_17-19,
1017 * 4,5,6,7,12,13,14,15,20,21,22,23
1019 { {40, 40, 40, 40, 32, 28, 40, 40, 32, 28, 40, 40, 32, 20} },
1020 { {40, 40, 40, 40, 32, 28, 40, 40, 32, 28, 40, 40, 32, 20} },
1021 { {40, 40, 40, 40, 32, 28, 40, 40, 32, 28, 40, 40, 32, 20} },
1022 { {40, 40, 40, 40, 32, 28, 40, 40, 32, 28, 40, 40, 32, 20} },
1023 { {40, 40, 40, 40, 32, 28, 40, 40, 32, 28, 40, 40, 32, 20} },
1024 { {40, 40, 40, 40, 32, 28, 40, 40, 32, 28, 40, 40, 32, 20} },
1025 { {38, 38, 38, 38, 32, 28, 38, 38, 32, 28, 38, 38, 32, 26} },
1026 { {36, 36, 36, 36, 32, 28, 36, 36, 32, 28, 36, 36, 32, 26} },
1028 .calTargetPower5GHT40
= {
1030 * 0_8_16,1-3_9-11_17-19,
1031 * 4,5,6,7,12,13,14,15,20,21,22,23
1033 { {40, 40, 40, 38, 30, 26, 40, 40, 30, 26, 40, 40, 30, 24} },
1034 { {40, 40, 40, 38, 30, 26, 40, 40, 30, 26, 40, 40, 30, 24} },
1035 { {40, 40, 40, 38, 30, 26, 40, 40, 30, 26, 40, 40, 30, 24} },
1036 { {40, 40, 40, 38, 30, 26, 40, 40, 30, 26, 40, 40, 30, 24} },
1037 { {40, 40, 40, 38, 30, 26, 40, 40, 30, 26, 40, 40, 30, 24} },
1038 { {40, 40, 40, 38, 30, 26, 40, 40, 30, 26, 40, 40, 30, 24} },
1039 { {36, 36, 36, 36, 30, 26, 36, 36, 30, 26, 36, 36, 30, 24} },
1040 { {34, 34, 34, 34, 30, 26, 34, 34, 30, 26, 34, 34, 30, 24} },
1043 0x10, 0x16, 0x18, 0x40, 0x46,
1044 0x48, 0x30, 0x36, 0x38
1048 /* Data[0].ctlEdges[0].bChannel */ FREQ2FBIN(5180, 0),
1049 /* Data[0].ctlEdges[1].bChannel */ FREQ2FBIN(5260, 0),
1050 /* Data[0].ctlEdges[2].bChannel */ FREQ2FBIN(5280, 0),
1051 /* Data[0].ctlEdges[3].bChannel */ FREQ2FBIN(5500, 0),
1052 /* Data[0].ctlEdges[4].bChannel */ FREQ2FBIN(5600, 0),
1053 /* Data[0].ctlEdges[5].bChannel */ FREQ2FBIN(5700, 0),
1054 /* Data[0].ctlEdges[6].bChannel */ FREQ2FBIN(5745, 0),
1055 /* Data[0].ctlEdges[7].bChannel */ FREQ2FBIN(5825, 0)
1058 /* Data[1].ctlEdges[0].bChannel */ FREQ2FBIN(5180, 0),
1059 /* Data[1].ctlEdges[1].bChannel */ FREQ2FBIN(5260, 0),
1060 /* Data[1].ctlEdges[2].bChannel */ FREQ2FBIN(5280, 0),
1061 /* Data[1].ctlEdges[3].bChannel */ FREQ2FBIN(5500, 0),
1062 /* Data[1].ctlEdges[4].bChannel */ FREQ2FBIN(5520, 0),
1063 /* Data[1].ctlEdges[5].bChannel */ FREQ2FBIN(5700, 0),
1064 /* Data[1].ctlEdges[6].bChannel */ FREQ2FBIN(5745, 0),
1065 /* Data[1].ctlEdges[7].bChannel */ FREQ2FBIN(5825, 0)
1069 /* Data[2].ctlEdges[0].bChannel */ FREQ2FBIN(5190, 0),
1070 /* Data[2].ctlEdges[1].bChannel */ FREQ2FBIN(5230, 0),
1071 /* Data[2].ctlEdges[2].bChannel */ FREQ2FBIN(5270, 0),
1072 /* Data[2].ctlEdges[3].bChannel */ FREQ2FBIN(5310, 0),
1073 /* Data[2].ctlEdges[4].bChannel */ FREQ2FBIN(5510, 0),
1074 /* Data[2].ctlEdges[5].bChannel */ FREQ2FBIN(5550, 0),
1075 /* Data[2].ctlEdges[6].bChannel */ FREQ2FBIN(5670, 0),
1076 /* Data[2].ctlEdges[7].bChannel */ FREQ2FBIN(5755, 0)
1080 /* Data[3].ctlEdges[0].bChannel */ FREQ2FBIN(5180, 0),
1081 /* Data[3].ctlEdges[1].bChannel */ FREQ2FBIN(5200, 0),
1082 /* Data[3].ctlEdges[2].bChannel */ FREQ2FBIN(5260, 0),
1083 /* Data[3].ctlEdges[3].bChannel */ FREQ2FBIN(5320, 0),
1084 /* Data[3].ctlEdges[4].bChannel */ FREQ2FBIN(5500, 0),
1085 /* Data[3].ctlEdges[5].bChannel */ FREQ2FBIN(5700, 0),
1086 /* Data[3].ctlEdges[6].bChannel */ 0xFF,
1087 /* Data[3].ctlEdges[7].bChannel */ 0xFF,
1091 /* Data[4].ctlEdges[0].bChannel */ FREQ2FBIN(5180, 0),
1092 /* Data[4].ctlEdges[1].bChannel */ FREQ2FBIN(5260, 0),
1093 /* Data[4].ctlEdges[2].bChannel */ FREQ2FBIN(5500, 0),
1094 /* Data[4].ctlEdges[3].bChannel */ FREQ2FBIN(5700, 0),
1095 /* Data[4].ctlEdges[4].bChannel */ 0xFF,
1096 /* Data[4].ctlEdges[5].bChannel */ 0xFF,
1097 /* Data[4].ctlEdges[6].bChannel */ 0xFF,
1098 /* Data[4].ctlEdges[7].bChannel */ 0xFF,
1102 /* Data[5].ctlEdges[0].bChannel */ FREQ2FBIN(5190, 0),
1103 /* Data[5].ctlEdges[1].bChannel */ FREQ2FBIN(5270, 0),
1104 /* Data[5].ctlEdges[2].bChannel */ FREQ2FBIN(5310, 0),
1105 /* Data[5].ctlEdges[3].bChannel */ FREQ2FBIN(5510, 0),
1106 /* Data[5].ctlEdges[4].bChannel */ FREQ2FBIN(5590, 0),
1107 /* Data[5].ctlEdges[5].bChannel */ FREQ2FBIN(5670, 0),
1108 /* Data[5].ctlEdges[6].bChannel */ 0xFF,
1109 /* Data[5].ctlEdges[7].bChannel */ 0xFF
1113 /* Data[6].ctlEdges[0].bChannel */ FREQ2FBIN(5180, 0),
1114 /* Data[6].ctlEdges[1].bChannel */ FREQ2FBIN(5200, 0),
1115 /* Data[6].ctlEdges[2].bChannel */ FREQ2FBIN(5220, 0),
1116 /* Data[6].ctlEdges[3].bChannel */ FREQ2FBIN(5260, 0),
1117 /* Data[6].ctlEdges[4].bChannel */ FREQ2FBIN(5500, 0),
1118 /* Data[6].ctlEdges[5].bChannel */ FREQ2FBIN(5600, 0),
1119 /* Data[6].ctlEdges[6].bChannel */ FREQ2FBIN(5700, 0),
1120 /* Data[6].ctlEdges[7].bChannel */ FREQ2FBIN(5745, 0)
1124 /* Data[7].ctlEdges[0].bChannel */ FREQ2FBIN(5180, 0),
1125 /* Data[7].ctlEdges[1].bChannel */ FREQ2FBIN(5260, 0),
1126 /* Data[7].ctlEdges[2].bChannel */ FREQ2FBIN(5320, 0),
1127 /* Data[7].ctlEdges[3].bChannel */ FREQ2FBIN(5500, 0),
1128 /* Data[7].ctlEdges[4].bChannel */ FREQ2FBIN(5560, 0),
1129 /* Data[7].ctlEdges[5].bChannel */ FREQ2FBIN(5700, 0),
1130 /* Data[7].ctlEdges[6].bChannel */ FREQ2FBIN(5745, 0),
1131 /* Data[7].ctlEdges[7].bChannel */ FREQ2FBIN(5825, 0)
1135 /* Data[8].ctlEdges[0].bChannel */ FREQ2FBIN(5190, 0),
1136 /* Data[8].ctlEdges[1].bChannel */ FREQ2FBIN(5230, 0),
1137 /* Data[8].ctlEdges[2].bChannel */ FREQ2FBIN(5270, 0),
1138 /* Data[8].ctlEdges[3].bChannel */ FREQ2FBIN(5510, 0),
1139 /* Data[8].ctlEdges[4].bChannel */ FREQ2FBIN(5550, 0),
1140 /* Data[8].ctlEdges[5].bChannel */ FREQ2FBIN(5670, 0),
1141 /* Data[8].ctlEdges[6].bChannel */ FREQ2FBIN(5755, 0),
1142 /* Data[8].ctlEdges[7].bChannel */ FREQ2FBIN(5795, 0)
1145 .ctlPowerData_5G
= {
1148 CTL(60, 1), CTL(60, 1), CTL(60, 1), CTL(60, 1),
1149 CTL(60, 1), CTL(60, 1), CTL(60, 1), CTL(60, 0),
1154 CTL(60, 1), CTL(60, 1), CTL(60, 1), CTL(60, 1),
1155 CTL(60, 1), CTL(60, 1), CTL(60, 1), CTL(60, 0),
1160 CTL(60, 0), CTL(60, 1), CTL(60, 0), CTL(60, 1),
1161 CTL(60, 1), CTL(60, 1), CTL(60, 1), CTL(60, 1),
1166 CTL(60, 0), CTL(60, 1), CTL(60, 1), CTL(60, 0),
1167 CTL(60, 1), CTL(60, 0), CTL(60, 0), CTL(60, 0),
1172 CTL(60, 1), CTL(60, 1), CTL(60, 1), CTL(60, 0),
1173 CTL(60, 0), CTL(60, 0), CTL(60, 0), CTL(60, 0),
1178 CTL(60, 1), CTL(60, 1), CTL(60, 1), CTL(60, 1),
1179 CTL(60, 1), CTL(60, 0), CTL(60, 0), CTL(60, 0),
1184 CTL(60, 1), CTL(60, 1), CTL(60, 1), CTL(60, 1),
1185 CTL(60, 1), CTL(60, 1), CTL(60, 1), CTL(60, 1),
1190 CTL(60, 1), CTL(60, 1), CTL(60, 0), CTL(60, 1),
1191 CTL(60, 1), CTL(60, 1), CTL(60, 1), CTL(60, 0),
1196 CTL(60, 1), CTL(60, 0), CTL(60, 1), CTL(60, 1),
1197 CTL(60, 1), CTL(60, 1), CTL(60, 0), CTL(60, 1),
1204 static const struct ar9300_eeprom ar9300_h112
= {
1206 .templateVersion
= 3,
1207 .macAddr
= {0x00, 0x03, 0x7f, 0x0, 0x0, 0x0},
1208 .custData
= {"h112-241-f0000"},
1210 .regDmn
= { LE16(0), LE16(0x1f) },
1211 .txrxMask
= 0x77, /* 4 bits tx and 4 bits rx */
1213 .opFlags
= AR5416_OPFLAGS_11G
| AR5416_OPFLAGS_11A
,
1214 .eepMisc
= AR9300_EEPMISC_LITTLE_ENDIAN
,
1217 .blueToothOptions
= 0,
1219 .deviceType
= 5, /* takes lower byte in eeprom location */
1220 .pwrTableOffset
= AR9300_PWR_TABLE_OFFSET
,
1221 .params_for_tuning_caps
= {0, 0},
1222 .featureEnable
= 0x0d,
1224 * bit0 - enable tx temp comp - disabled
1225 * bit1 - enable tx volt comp - disabled
1226 * bit2 - enable fastClock - enabled
1227 * bit3 - enable doubling - enabled
1228 * bit4 - enable internal regulator - disabled
1229 * bit5 - enable pa predistortion - disabled
1231 .miscConfiguration
= 0, /* bit0 - turn down drivestrength */
1232 .eepromWriteEnableGpio
= 6,
1233 .wlanDisableGpio
= 0,
1235 .rxBandSelectGpio
= 0xff,
1240 /* ar9300_modal_eep_header 2g */
1241 /* 4 idle,t1,t2,b(4 bits per setting) */
1242 .antCtrlCommon
= LE32(0x110),
1243 /* 4 ra1l1, ra2l1, ra1l2, ra2l2, ra12 */
1244 .antCtrlCommon2
= LE32(0x44444),
1247 * antCtrlChain[AR9300_MAX_CHAINS]; 6 idle, t, r,
1248 * rx1, rx12, b (2 bits each)
1250 .antCtrlChain
= { LE16(0x150), LE16(0x150), LE16(0x150) },
1253 * xatten1DB[AR9300_MAX_CHAINS]; 3 xatten1_db
1254 * for ar9280 (0xa20c/b20c 5:0)
1256 .xatten1DB
= {0, 0, 0},
1259 * xatten1Margin[AR9300_MAX_CHAINS]; 3 xatten1_margin
1260 * for ar9280 (0xa20c/b20c 16:12
1262 .xatten1Margin
= {0, 0, 0},
1267 * spurChans[OSPREY_EEPROM_MODAL_SPURS]; spur
1268 * channels in usual fbin coding format
1270 .spurChans
= {FREQ2FBIN(2464, 1), 0, 0, 0, 0},
1273 * noiseFloorThreshCh[AR9300_MAX_CHAINS]; 3 Check
1274 * if the register is per chain
1276 .noiseFloorThreshCh
= {-1, 0, 0},
1277 .reserved
= {0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0},
1280 .txFrameToDataStart
= 0x0e,
1281 .txFrameToPaOn
= 0x0e,
1282 .txClip
= 3, /* 4 bits tx_clip, 4 bits dac_scale_cck */
1284 .switchSettling
= 0x2c,
1285 .adcDesiredSize
= -30,
1288 .txFrameToXpaOn
= 0xe,
1290 .papdRateMaskHt20
= LE32(0x0c80c080),
1291 .papdRateMaskHt40
= LE32(0x0080c080),
1293 .xlna_bias_strength
= 0,
1295 0, 0, 0, 0, 0, 0, 0,
1299 .ant_div_control
= 0,
1301 .tempslopextension
= {0, 0, 0, 0, 0, 0, 0, 0}
1308 /* ar9300_cal_data_per_freq_op_loop 2g */
1310 { {0, 0, 0, 0, 0, 0}, {0, 0, 0, 0, 0, 0}, {0, 0, 0, 0, 0, 0} },
1311 { {0, 0, 0, 0, 0, 0}, {0, 0, 0, 0, 0, 0}, {0, 0, 0, 0, 0, 0} },
1312 { {0, 0, 0, 0, 0, 0}, {0, 0, 0, 0, 0, 0}, {0, 0, 0, 0, 0, 0} },
1314 .calTarget_freqbin_Cck
= {
1318 .calTarget_freqbin_2G
= {
1323 .calTarget_freqbin_2GHT20
= {
1328 .calTarget_freqbin_2GHT40
= {
1333 .calTargetPowerCck
= {
1334 /* 1L-5L,5S,11L,11S */
1335 { {34, 34, 34, 34} },
1336 { {34, 34, 34, 34} },
1338 .calTargetPower2G
= {
1340 { {34, 34, 32, 32} },
1341 { {34, 34, 32, 32} },
1342 { {34, 34, 32, 32} },
1344 .calTargetPower2GHT20
= {
1345 { {32, 32, 32, 32, 32, 30, 32, 32, 30, 28, 28, 28, 28, 24} },
1346 { {32, 32, 32, 32, 32, 30, 32, 32, 30, 28, 28, 28, 28, 24} },
1347 { {32, 32, 32, 32, 32, 30, 32, 32, 30, 28, 28, 28, 28, 24} },
1349 .calTargetPower2GHT40
= {
1350 { {30, 30, 30, 30, 30, 28, 30, 30, 28, 26, 26, 26, 26, 22} },
1351 { {30, 30, 30, 30, 30, 28, 30, 30, 28, 26, 26, 26, 26, 22} },
1352 { {30, 30, 30, 30, 30, 28, 30, 30, 28, 26, 26, 26, 26, 22} },
1355 0x11, 0x12, 0x15, 0x17, 0x41, 0x42,
1356 0x45, 0x47, 0x31, 0x32, 0x35, 0x37,
1386 /* Data[4].ctlEdges[0].bChannel */ FREQ2FBIN(2412, 1),
1387 /* Data[4].ctlEdges[1].bChannel */ FREQ2FBIN(2417, 1),
1388 /* Data[4].ctlEdges[2].bChannel */ FREQ2FBIN(2472, 1),
1389 /* Data[4].ctlEdges[3].bChannel */ FREQ2FBIN(2484, 1),
1393 /* Data[5].ctlEdges[0].bChannel */ FREQ2FBIN(2412, 1),
1394 /* Data[5].ctlEdges[1].bChannel */ FREQ2FBIN(2417, 1),
1395 /* Data[5].ctlEdges[2].bChannel */ FREQ2FBIN(2472, 1),
1400 /* Data[6].ctlEdges[0].bChannel */ FREQ2FBIN(2412, 1),
1401 /* Data[6].ctlEdges[1].bChannel */ FREQ2FBIN(2417, 1),
1407 /* Data[7].ctlEdges[0].bChannel */ FREQ2FBIN(2422, 1),
1408 /* Data[7].ctlEdges[1].bChannel */ FREQ2FBIN(2427, 1),
1409 /* Data[7].ctlEdges[2].bChannel */ FREQ2FBIN(2447, 1),
1410 /* Data[7].ctlEdges[3].bChannel */ FREQ2FBIN(2462, 1),
1414 /* Data[8].ctlEdges[0].bChannel */ FREQ2FBIN(2412, 1),
1415 /* Data[8].ctlEdges[1].bChannel */ FREQ2FBIN(2417, 1),
1416 /* Data[8].ctlEdges[2].bChannel */ FREQ2FBIN(2472, 1),
1420 /* Data[9].ctlEdges[0].bChannel */ FREQ2FBIN(2412, 1),
1421 /* Data[9].ctlEdges[1].bChannel */ FREQ2FBIN(2417, 1),
1422 /* Data[9].ctlEdges[2].bChannel */ FREQ2FBIN(2472, 1),
1427 /* Data[10].ctlEdges[0].bChannel */ FREQ2FBIN(2412, 1),
1428 /* Data[10].ctlEdges[1].bChannel */ FREQ2FBIN(2417, 1),
1429 /* Data[10].ctlEdges[2].bChannel */ FREQ2FBIN(2472, 1),
1434 /* Data[11].ctlEdges[0].bChannel */ FREQ2FBIN(2422, 1),
1435 /* Data[11].ctlEdges[1].bChannel */ FREQ2FBIN(2427, 1),
1436 /* Data[11].ctlEdges[2].bChannel */ FREQ2FBIN(2447, 1),
1437 /* Data[11].ctlEdges[3].bChannel */ FREQ2FBIN(2462, 1),
1440 .ctlPowerData_2G
= {
1441 { { CTL(60, 0), CTL(60, 1), CTL(60, 0), CTL(60, 0) } },
1442 { { CTL(60, 0), CTL(60, 1), CTL(60, 0), CTL(60, 0) } },
1443 { { CTL(60, 1), CTL(60, 0), CTL(60, 0), CTL(60, 1) } },
1445 { { CTL(60, 1), CTL(60, 0), CTL(60, 0), CTL(60, 0) } },
1446 { { CTL(60, 0), CTL(60, 1), CTL(60, 0), CTL(60, 0) } },
1447 { { CTL(60, 0), CTL(60, 1), CTL(60, 0), CTL(60, 0) } },
1449 { { CTL(60, 0), CTL(60, 1), CTL(60, 1), CTL(60, 0) } },
1450 { { CTL(60, 0), CTL(60, 1), CTL(60, 0), CTL(60, 0) } },
1451 { { CTL(60, 0), CTL(60, 1), CTL(60, 0), CTL(60, 0) } },
1453 { { CTL(60, 0), CTL(60, 1), CTL(60, 0), CTL(60, 0) } },
1454 { { CTL(60, 0), CTL(60, 1), CTL(60, 1), CTL(60, 1) } },
1455 { { CTL(60, 0), CTL(60, 1), CTL(60, 1), CTL(60, 1) } },
1458 /* 4 idle,t1,t2,b (4 bits per setting) */
1459 .antCtrlCommon
= LE32(0x220),
1460 /* 4 ra1l1, ra2l1, ra1l2,ra2l2,ra12 */
1461 .antCtrlCommon2
= LE32(0x44444),
1462 /* antCtrlChain 6 idle, t,r,rx1,rx12,b (2 bits each) */
1464 LE16(0x150), LE16(0x150), LE16(0x150),
1466 /* xatten1DB 3 xatten1_db for AR9280 (0xa20c/b20c 5:0) */
1467 .xatten1DB
= {0, 0, 0},
1470 * xatten1Margin[AR9300_MAX_CHAINS]; 3 xatten1_margin
1471 * for merlin (0xa20c/b20c 16:12
1473 .xatten1Margin
= {0, 0, 0},
1476 /* spurChans spur channels in usual fbin coding format */
1477 .spurChans
= {0, 0, 0, 0, 0},
1478 /* noiseFloorThreshCh Check if the register is per chain */
1479 .noiseFloorThreshCh
= {-1, 0, 0},
1480 .reserved
= {0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0},
1483 .txFrameToDataStart
= 0x0e,
1484 .txFrameToPaOn
= 0x0e,
1485 .txClip
= 3, /* 4 bits tx_clip, 4 bits dac_scale_cck */
1487 .switchSettling
= 0x2d,
1488 .adcDesiredSize
= -30,
1491 .txFrameToXpaOn
= 0xe,
1493 .papdRateMaskHt20
= LE32(0x0cf0e0e0),
1494 .papdRateMaskHt40
= LE32(0x6cf0e0e0),
1496 .xlna_bias_strength
= 0,
1498 0, 0, 0, 0, 0, 0, 0,
1503 .tempSlopeHigh
= 50,
1504 .xatten1DBLow
= {0, 0, 0},
1505 .xatten1MarginLow
= {0, 0, 0},
1506 .xatten1DBHigh
= {0, 0, 0},
1507 .xatten1MarginHigh
= {0, 0, 0}
1552 .calTarget_freqbin_5G
= {
1562 .calTarget_freqbin_5GHT20
= {
1572 .calTarget_freqbin_5GHT40
= {
1582 .calTargetPower5G
= {
1584 { {30, 30, 28, 24} },
1585 { {30, 30, 28, 24} },
1586 { {30, 30, 28, 24} },
1587 { {30, 30, 28, 24} },
1588 { {30, 30, 28, 24} },
1589 { {30, 30, 28, 24} },
1590 { {30, 30, 28, 24} },
1591 { {30, 30, 28, 24} },
1593 .calTargetPower5GHT20
= {
1595 * 0_8_16,1-3_9-11_17-19,
1596 * 4,5,6,7,12,13,14,15,20,21,22,23
1598 { {30, 30, 30, 28, 24, 20, 30, 28, 24, 20, 20, 20, 20, 16} },
1599 { {30, 30, 30, 28, 24, 20, 30, 28, 24, 20, 20, 20, 20, 16} },
1600 { {30, 30, 30, 26, 22, 18, 30, 26, 22, 18, 18, 18, 18, 16} },
1601 { {30, 30, 30, 26, 22, 18, 30, 26, 22, 18, 18, 18, 18, 16} },
1602 { {30, 30, 30, 24, 20, 16, 30, 24, 20, 16, 16, 16, 16, 14} },
1603 { {30, 30, 30, 24, 20, 16, 30, 24, 20, 16, 16, 16, 16, 14} },
1604 { {30, 30, 30, 22, 18, 14, 30, 22, 18, 14, 14, 14, 14, 12} },
1605 { {30, 30, 30, 22, 18, 14, 30, 22, 18, 14, 14, 14, 14, 12} },
1607 .calTargetPower5GHT40
= {
1609 * 0_8_16,1-3_9-11_17-19,
1610 * 4,5,6,7,12,13,14,15,20,21,22,23
1612 { {28, 28, 28, 26, 22, 18, 28, 26, 22, 18, 18, 18, 18, 14} },
1613 { {28, 28, 28, 26, 22, 18, 28, 26, 22, 18, 18, 18, 18, 14} },
1614 { {28, 28, 28, 24, 20, 16, 28, 24, 20, 16, 16, 16, 16, 12} },
1615 { {28, 28, 28, 24, 20, 16, 28, 24, 20, 16, 16, 16, 16, 12} },
1616 { {28, 28, 28, 22, 18, 14, 28, 22, 18, 14, 14, 14, 14, 10} },
1617 { {28, 28, 28, 22, 18, 14, 28, 22, 18, 14, 14, 14, 14, 10} },
1618 { {28, 28, 28, 20, 16, 12, 28, 20, 16, 12, 12, 12, 12, 8} },
1619 { {28, 28, 28, 20, 16, 12, 28, 20, 16, 12, 12, 12, 12, 8} },
1622 0x10, 0x16, 0x18, 0x40, 0x46,
1623 0x48, 0x30, 0x36, 0x38
1627 /* Data[0].ctlEdges[0].bChannel */ FREQ2FBIN(5180, 0),
1628 /* Data[0].ctlEdges[1].bChannel */ FREQ2FBIN(5260, 0),
1629 /* Data[0].ctlEdges[2].bChannel */ FREQ2FBIN(5280, 0),
1630 /* Data[0].ctlEdges[3].bChannel */ FREQ2FBIN(5500, 0),
1631 /* Data[0].ctlEdges[4].bChannel */ FREQ2FBIN(5600, 0),
1632 /* Data[0].ctlEdges[5].bChannel */ FREQ2FBIN(5700, 0),
1633 /* Data[0].ctlEdges[6].bChannel */ FREQ2FBIN(5745, 0),
1634 /* Data[0].ctlEdges[7].bChannel */ FREQ2FBIN(5825, 0)
1637 /* Data[1].ctlEdges[0].bChannel */ FREQ2FBIN(5180, 0),
1638 /* Data[1].ctlEdges[1].bChannel */ FREQ2FBIN(5260, 0),
1639 /* Data[1].ctlEdges[2].bChannel */ FREQ2FBIN(5280, 0),
1640 /* Data[1].ctlEdges[3].bChannel */ FREQ2FBIN(5500, 0),
1641 /* Data[1].ctlEdges[4].bChannel */ FREQ2FBIN(5520, 0),
1642 /* Data[1].ctlEdges[5].bChannel */ FREQ2FBIN(5700, 0),
1643 /* Data[1].ctlEdges[6].bChannel */ FREQ2FBIN(5745, 0),
1644 /* Data[1].ctlEdges[7].bChannel */ FREQ2FBIN(5825, 0)
1648 /* Data[2].ctlEdges[0].bChannel */ FREQ2FBIN(5190, 0),
1649 /* Data[2].ctlEdges[1].bChannel */ FREQ2FBIN(5230, 0),
1650 /* Data[2].ctlEdges[2].bChannel */ FREQ2FBIN(5270, 0),
1651 /* Data[2].ctlEdges[3].bChannel */ FREQ2FBIN(5310, 0),
1652 /* Data[2].ctlEdges[4].bChannel */ FREQ2FBIN(5510, 0),
1653 /* Data[2].ctlEdges[5].bChannel */ FREQ2FBIN(5550, 0),
1654 /* Data[2].ctlEdges[6].bChannel */ FREQ2FBIN(5670, 0),
1655 /* Data[2].ctlEdges[7].bChannel */ FREQ2FBIN(5755, 0)
1659 /* Data[3].ctlEdges[0].bChannel */ FREQ2FBIN(5180, 0),
1660 /* Data[3].ctlEdges[1].bChannel */ FREQ2FBIN(5200, 0),
1661 /* Data[3].ctlEdges[2].bChannel */ FREQ2FBIN(5260, 0),
1662 /* Data[3].ctlEdges[3].bChannel */ FREQ2FBIN(5320, 0),
1663 /* Data[3].ctlEdges[4].bChannel */ FREQ2FBIN(5500, 0),
1664 /* Data[3].ctlEdges[5].bChannel */ FREQ2FBIN(5700, 0),
1665 /* Data[3].ctlEdges[6].bChannel */ 0xFF,
1666 /* Data[3].ctlEdges[7].bChannel */ 0xFF,
1670 /* Data[4].ctlEdges[0].bChannel */ FREQ2FBIN(5180, 0),
1671 /* Data[4].ctlEdges[1].bChannel */ FREQ2FBIN(5260, 0),
1672 /* Data[4].ctlEdges[2].bChannel */ FREQ2FBIN(5500, 0),
1673 /* Data[4].ctlEdges[3].bChannel */ FREQ2FBIN(5700, 0),
1674 /* Data[4].ctlEdges[4].bChannel */ 0xFF,
1675 /* Data[4].ctlEdges[5].bChannel */ 0xFF,
1676 /* Data[4].ctlEdges[6].bChannel */ 0xFF,
1677 /* Data[4].ctlEdges[7].bChannel */ 0xFF,
1681 /* Data[5].ctlEdges[0].bChannel */ FREQ2FBIN(5190, 0),
1682 /* Data[5].ctlEdges[1].bChannel */ FREQ2FBIN(5270, 0),
1683 /* Data[5].ctlEdges[2].bChannel */ FREQ2FBIN(5310, 0),
1684 /* Data[5].ctlEdges[3].bChannel */ FREQ2FBIN(5510, 0),
1685 /* Data[5].ctlEdges[4].bChannel */ FREQ2FBIN(5590, 0),
1686 /* Data[5].ctlEdges[5].bChannel */ FREQ2FBIN(5670, 0),
1687 /* Data[5].ctlEdges[6].bChannel */ 0xFF,
1688 /* Data[5].ctlEdges[7].bChannel */ 0xFF
1692 /* Data[6].ctlEdges[0].bChannel */ FREQ2FBIN(5180, 0),
1693 /* Data[6].ctlEdges[1].bChannel */ FREQ2FBIN(5200, 0),
1694 /* Data[6].ctlEdges[2].bChannel */ FREQ2FBIN(5220, 0),
1695 /* Data[6].ctlEdges[3].bChannel */ FREQ2FBIN(5260, 0),
1696 /* Data[6].ctlEdges[4].bChannel */ FREQ2FBIN(5500, 0),
1697 /* Data[6].ctlEdges[5].bChannel */ FREQ2FBIN(5600, 0),
1698 /* Data[6].ctlEdges[6].bChannel */ FREQ2FBIN(5700, 0),
1699 /* Data[6].ctlEdges[7].bChannel */ FREQ2FBIN(5745, 0)
1703 /* Data[7].ctlEdges[0].bChannel */ FREQ2FBIN(5180, 0),
1704 /* Data[7].ctlEdges[1].bChannel */ FREQ2FBIN(5260, 0),
1705 /* Data[7].ctlEdges[2].bChannel */ FREQ2FBIN(5320, 0),
1706 /* Data[7].ctlEdges[3].bChannel */ FREQ2FBIN(5500, 0),
1707 /* Data[7].ctlEdges[4].bChannel */ FREQ2FBIN(5560, 0),
1708 /* Data[7].ctlEdges[5].bChannel */ FREQ2FBIN(5700, 0),
1709 /* Data[7].ctlEdges[6].bChannel */ FREQ2FBIN(5745, 0),
1710 /* Data[7].ctlEdges[7].bChannel */ FREQ2FBIN(5825, 0)
1714 /* Data[8].ctlEdges[0].bChannel */ FREQ2FBIN(5190, 0),
1715 /* Data[8].ctlEdges[1].bChannel */ FREQ2FBIN(5230, 0),
1716 /* Data[8].ctlEdges[2].bChannel */ FREQ2FBIN(5270, 0),
1717 /* Data[8].ctlEdges[3].bChannel */ FREQ2FBIN(5510, 0),
1718 /* Data[8].ctlEdges[4].bChannel */ FREQ2FBIN(5550, 0),
1719 /* Data[8].ctlEdges[5].bChannel */ FREQ2FBIN(5670, 0),
1720 /* Data[8].ctlEdges[6].bChannel */ FREQ2FBIN(5755, 0),
1721 /* Data[8].ctlEdges[7].bChannel */ FREQ2FBIN(5795, 0)
1724 .ctlPowerData_5G
= {
1727 CTL(60, 1), CTL(60, 1), CTL(60, 1), CTL(60, 1),
1728 CTL(60, 1), CTL(60, 1), CTL(60, 1), CTL(60, 0),
1733 CTL(60, 1), CTL(60, 1), CTL(60, 1), CTL(60, 1),
1734 CTL(60, 1), CTL(60, 1), CTL(60, 1), CTL(60, 0),
1739 CTL(60, 0), CTL(60, 1), CTL(60, 0), CTL(60, 1),
1740 CTL(60, 1), CTL(60, 1), CTL(60, 1), CTL(60, 1),
1745 CTL(60, 0), CTL(60, 1), CTL(60, 1), CTL(60, 0),
1746 CTL(60, 1), CTL(60, 0), CTL(60, 0), CTL(60, 0),
1751 CTL(60, 1), CTL(60, 1), CTL(60, 1), CTL(60, 0),
1752 CTL(60, 0), CTL(60, 0), CTL(60, 0), CTL(60, 0),
1757 CTL(60, 1), CTL(60, 1), CTL(60, 1), CTL(60, 1),
1758 CTL(60, 1), CTL(60, 0), CTL(60, 0), CTL(60, 0),
1763 CTL(60, 1), CTL(60, 1), CTL(60, 1), CTL(60, 1),
1764 CTL(60, 1), CTL(60, 1), CTL(60, 1), CTL(60, 1),
1769 CTL(60, 1), CTL(60, 1), CTL(60, 0), CTL(60, 1),
1770 CTL(60, 1), CTL(60, 1), CTL(60, 1), CTL(60, 0),
1775 CTL(60, 1), CTL(60, 0), CTL(60, 1), CTL(60, 1),
1776 CTL(60, 1), CTL(60, 1), CTL(60, 0), CTL(60, 1),
1783 static const struct ar9300_eeprom ar9300_x112
= {
1785 .templateVersion
= 5,
1786 .macAddr
= {0x00, 0x03, 0x7f, 0x0, 0x0, 0x0},
1787 .custData
= {"x112-041-f0000"},
1789 .regDmn
= { LE16(0), LE16(0x1f) },
1790 .txrxMask
= 0x77, /* 4 bits tx and 4 bits rx */
1792 .opFlags
= AR5416_OPFLAGS_11G
| AR5416_OPFLAGS_11A
,
1793 .eepMisc
= AR9300_EEPMISC_LITTLE_ENDIAN
,
1796 .blueToothOptions
= 0,
1798 .deviceType
= 5, /* takes lower byte in eeprom location */
1799 .pwrTableOffset
= AR9300_PWR_TABLE_OFFSET
,
1800 .params_for_tuning_caps
= {0, 0},
1801 .featureEnable
= 0x0d,
1803 * bit0 - enable tx temp comp - disabled
1804 * bit1 - enable tx volt comp - disabled
1805 * bit2 - enable fastclock - enabled
1806 * bit3 - enable doubling - enabled
1807 * bit4 - enable internal regulator - disabled
1808 * bit5 - enable pa predistortion - disabled
1810 .miscConfiguration
= 0, /* bit0 - turn down drivestrength */
1811 .eepromWriteEnableGpio
= 6,
1812 .wlanDisableGpio
= 0,
1814 .rxBandSelectGpio
= 0xff,
1819 /* ar9300_modal_eep_header 2g */
1820 /* 4 idle,t1,t2,b(4 bits per setting) */
1821 .antCtrlCommon
= LE32(0x110),
1822 /* 4 ra1l1, ra2l1, ra1l2, ra2l2, ra12 */
1823 .antCtrlCommon2
= LE32(0x22222),
1826 * antCtrlChain[ar9300_max_chains]; 6 idle, t, r,
1827 * rx1, rx12, b (2 bits each)
1829 .antCtrlChain
= { LE16(0x10), LE16(0x10), LE16(0x10) },
1832 * xatten1DB[AR9300_max_chains]; 3 xatten1_db
1833 * for ar9280 (0xa20c/b20c 5:0)
1835 .xatten1DB
= {0x1b, 0x1b, 0x1b},
1838 * xatten1Margin[ar9300_max_chains]; 3 xatten1_margin
1839 * for ar9280 (0xa20c/b20c 16:12
1841 .xatten1Margin
= {0x15, 0x15, 0x15},
1846 * spurChans[OSPrey_eeprom_modal_sPURS]; spur
1847 * channels in usual fbin coding format
1849 .spurChans
= {FREQ2FBIN(2464, 1), 0, 0, 0, 0},
1852 * noiseFloorThreshch[ar9300_max_cHAINS]; 3 Check
1853 * if the register is per chain
1855 .noiseFloorThreshCh
= {-1, 0, 0},
1856 .reserved
= {0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0},
1859 .txFrameToDataStart
= 0x0e,
1860 .txFrameToPaOn
= 0x0e,
1861 .txClip
= 3, /* 4 bits tx_clip, 4 bits dac_scale_cck */
1863 .switchSettling
= 0x2c,
1864 .adcDesiredSize
= -30,
1867 .txFrameToXpaOn
= 0xe,
1869 .papdRateMaskHt20
= LE32(0x0c80c080),
1870 .papdRateMaskHt40
= LE32(0x0080c080),
1872 .xlna_bias_strength
= 0,
1874 0, 0, 0, 0, 0, 0, 0,
1878 .ant_div_control
= 0,
1880 .tempslopextension
= {0, 0, 0, 0, 0, 0, 0, 0}
1887 /* ar9300_cal_data_per_freq_op_loop 2g */
1889 { {0, 0, 0, 0, 0, 0}, {0, 0, 0, 0, 0, 0}, {0, 0, 0, 0, 0, 0} },
1890 { {0, 0, 0, 0, 0, 0}, {0, 0, 0, 0, 0, 0}, {0, 0, 0, 0, 0, 0} },
1891 { {0, 0, 0, 0, 0, 0}, {0, 0, 0, 0, 0, 0}, {0, 0, 0, 0, 0, 0} },
1893 .calTarget_freqbin_Cck
= {
1897 .calTarget_freqbin_2G
= {
1902 .calTarget_freqbin_2GHT20
= {
1907 .calTarget_freqbin_2GHT40
= {
1912 .calTargetPowerCck
= {
1913 /* 1L-5L,5S,11L,11s */
1914 { {38, 38, 38, 38} },
1915 { {38, 38, 38, 38} },
1917 .calTargetPower2G
= {
1919 { {38, 38, 36, 34} },
1920 { {38, 38, 36, 34} },
1921 { {38, 38, 34, 32} },
1923 .calTargetPower2GHT20
= {
1924 { {36, 36, 36, 36, 36, 34, 34, 32, 30, 28, 28, 28, 28, 26} },
1925 { {36, 36, 36, 36, 36, 34, 36, 34, 32, 30, 30, 30, 28, 26} },
1926 { {36, 36, 36, 36, 36, 34, 34, 32, 30, 28, 28, 28, 28, 26} },
1928 .calTargetPower2GHT40
= {
1929 { {36, 36, 36, 36, 34, 32, 32, 30, 28, 26, 26, 26, 26, 24} },
1930 { {36, 36, 36, 36, 34, 32, 34, 32, 30, 28, 28, 28, 28, 24} },
1931 { {36, 36, 36, 36, 34, 32, 32, 30, 28, 26, 26, 26, 26, 24} },
1934 0x11, 0x12, 0x15, 0x17, 0x41, 0x42,
1935 0x45, 0x47, 0x31, 0x32, 0x35, 0x37,
1965 /* Data[4].ctledges[0].bchannel */ FREQ2FBIN(2412, 1),
1966 /* Data[4].ctledges[1].bchannel */ FREQ2FBIN(2417, 1),
1967 /* Data[4].ctledges[2].bchannel */ FREQ2FBIN(2472, 1),
1968 /* Data[4].ctledges[3].bchannel */ FREQ2FBIN(2484, 1),
1972 /* Data[5].ctledges[0].bchannel */ FREQ2FBIN(2412, 1),
1973 /* Data[5].ctledges[1].bchannel */ FREQ2FBIN(2417, 1),
1974 /* Data[5].ctledges[2].bchannel */ FREQ2FBIN(2472, 1),
1979 /* Data[6].ctledges[0].bchannel */ FREQ2FBIN(2412, 1),
1980 /* Data[6].ctledges[1].bchannel */ FREQ2FBIN(2417, 1),
1986 /* Data[7].ctledges[0].bchannel */ FREQ2FBIN(2422, 1),
1987 /* Data[7].ctledges[1].bchannel */ FREQ2FBIN(2427, 1),
1988 /* Data[7].ctledges[2].bchannel */ FREQ2FBIN(2447, 1),
1989 /* Data[7].ctledges[3].bchannel */ FREQ2FBIN(2462, 1),
1993 /* Data[8].ctledges[0].bchannel */ FREQ2FBIN(2412, 1),
1994 /* Data[8].ctledges[1].bchannel */ FREQ2FBIN(2417, 1),
1995 /* Data[8].ctledges[2].bchannel */ FREQ2FBIN(2472, 1),
1999 /* Data[9].ctledges[0].bchannel */ FREQ2FBIN(2412, 1),
2000 /* Data[9].ctledges[1].bchannel */ FREQ2FBIN(2417, 1),
2001 /* Data[9].ctledges[2].bchannel */ FREQ2FBIN(2472, 1),
2006 /* Data[10].ctledges[0].bchannel */ FREQ2FBIN(2412, 1),
2007 /* Data[10].ctledges[1].bchannel */ FREQ2FBIN(2417, 1),
2008 /* Data[10].ctledges[2].bchannel */ FREQ2FBIN(2472, 1),
2013 /* Data[11].ctledges[0].bchannel */ FREQ2FBIN(2422, 1),
2014 /* Data[11].ctledges[1].bchannel */ FREQ2FBIN(2427, 1),
2015 /* Data[11].ctledges[2].bchannel */ FREQ2FBIN(2447, 1),
2016 /* Data[11].ctledges[3].bchannel */ FREQ2FBIN(2462, 1),
2019 .ctlPowerData_2G
= {
2020 { { CTL(60, 0), CTL(60, 1), CTL(60, 0), CTL(60, 0) } },
2021 { { CTL(60, 0), CTL(60, 1), CTL(60, 0), CTL(60, 0) } },
2022 { { CTL(60, 1), CTL(60, 0), CTL(60, 0), CTL(60, 1) } },
2024 { { CTL(60, 1), CTL(60, 0), CTL(60, 0), CTL(60, 0) } },
2025 { { CTL(60, 0), CTL(60, 1), CTL(60, 0), CTL(60, 0) } },
2026 { { CTL(60, 0), CTL(60, 1), CTL(60, 0), CTL(60, 0) } },
2028 { { CTL(60, 0), CTL(60, 1), CTL(60, 1), CTL(60, 0) } },
2029 { { CTL(60, 0), CTL(60, 1), CTL(60, 0), CTL(60, 0) } },
2030 { { CTL(60, 0), CTL(60, 1), CTL(60, 0), CTL(60, 0) } },
2032 { { CTL(60, 0), CTL(60, 1), CTL(60, 0), CTL(60, 0) } },
2033 { { CTL(60, 0), CTL(60, 1), CTL(60, 1), CTL(60, 1) } },
2034 { { CTL(60, 0), CTL(60, 1), CTL(60, 1), CTL(60, 1) } },
2037 /* 4 idle,t1,t2,b (4 bits per setting) */
2038 .antCtrlCommon
= LE32(0x110),
2039 /* 4 ra1l1, ra2l1, ra1l2,ra2l2,ra12 */
2040 .antCtrlCommon2
= LE32(0x22222),
2041 /* antCtrlChain 6 idle, t,r,rx1,rx12,b (2 bits each) */
2043 LE16(0x0), LE16(0x0), LE16(0x0),
2045 /* xatten1DB 3 xatten1_db for ar9280 (0xa20c/b20c 5:0) */
2046 .xatten1DB
= {0x13, 0x19, 0x17},
2049 * xatten1Margin[ar9300_max_chains]; 3 xatten1_margin
2050 * for merlin (0xa20c/b20c 16:12
2052 .xatten1Margin
= {0x19, 0x19, 0x19},
2055 /* spurChans spur channels in usual fbin coding format */
2056 .spurChans
= {0, 0, 0, 0, 0},
2057 /* noiseFloorThreshch check if the register is per chain */
2058 .noiseFloorThreshCh
= {-1, 0, 0},
2059 .reserved
= {0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0},
2062 .txFrameToDataStart
= 0x0e,
2063 .txFrameToPaOn
= 0x0e,
2064 .txClip
= 3, /* 4 bits tx_clip, 4 bits dac_scale_cck */
2066 .switchSettling
= 0x2d,
2067 .adcDesiredSize
= -30,
2070 .txFrameToXpaOn
= 0xe,
2072 .papdRateMaskHt20
= LE32(0x0cf0e0e0),
2073 .papdRateMaskHt40
= LE32(0x6cf0e0e0),
2075 .xlna_bias_strength
= 0,
2077 0, 0, 0, 0, 0, 0, 0,
2082 .tempSlopeHigh
= 105,
2083 .xatten1DBLow
= {0x10, 0x14, 0x10},
2084 .xatten1MarginLow
= {0x19, 0x19 , 0x19},
2085 .xatten1DBHigh
= {0x1d, 0x20, 0x24},
2086 .xatten1MarginHigh
= {0x10, 0x10, 0x10}
2131 .calTarget_freqbin_5G
= {
2141 .calTarget_freqbin_5GHT20
= {
2151 .calTarget_freqbin_5GHT40
= {
2161 .calTargetPower5G
= {
2163 { {32, 32, 28, 26} },
2164 { {32, 32, 28, 26} },
2165 { {32, 32, 28, 26} },
2166 { {32, 32, 26, 24} },
2167 { {32, 32, 26, 24} },
2168 { {32, 32, 24, 22} },
2169 { {30, 30, 24, 22} },
2170 { {30, 30, 24, 22} },
2172 .calTargetPower5GHT20
= {
2174 * 0_8_16,1-3_9-11_17-19,
2175 * 4,5,6,7,12,13,14,15,20,21,22,23
2177 { {32, 32, 32, 32, 28, 26, 32, 28, 26, 24, 24, 24, 22, 22} },
2178 { {32, 32, 32, 32, 28, 26, 32, 28, 26, 24, 24, 24, 22, 22} },
2179 { {32, 32, 32, 32, 28, 26, 32, 28, 26, 24, 24, 24, 22, 22} },
2180 { {32, 32, 32, 32, 28, 26, 32, 26, 24, 22, 22, 22, 20, 20} },
2181 { {32, 32, 32, 32, 28, 26, 32, 26, 24, 22, 20, 18, 16, 16} },
2182 { {32, 32, 32, 32, 28, 26, 32, 24, 20, 16, 18, 16, 14, 14} },
2183 { {30, 30, 30, 30, 28, 26, 30, 24, 20, 16, 18, 16, 14, 14} },
2184 { {30, 30, 30, 30, 28, 26, 30, 24, 20, 16, 18, 16, 14, 14} },
2186 .calTargetPower5GHT40
= {
2188 * 0_8_16,1-3_9-11_17-19,
2189 * 4,5,6,7,12,13,14,15,20,21,22,23
2191 { {32, 32, 32, 30, 28, 26, 30, 28, 26, 24, 24, 24, 22, 22} },
2192 { {32, 32, 32, 30, 28, 26, 30, 28, 26, 24, 24, 24, 22, 22} },
2193 { {32, 32, 32, 30, 28, 26, 30, 28, 26, 24, 24, 24, 22, 22} },
2194 { {32, 32, 32, 30, 28, 26, 30, 26, 24, 22, 22, 22, 20, 20} },
2195 { {32, 32, 32, 30, 28, 26, 30, 26, 24, 22, 20, 18, 16, 16} },
2196 { {32, 32, 32, 30, 28, 26, 30, 22, 20, 16, 18, 16, 14, 14} },
2197 { {30, 30, 30, 30, 28, 26, 30, 22, 20, 16, 18, 16, 14, 14} },
2198 { {30, 30, 30, 30, 28, 26, 30, 22, 20, 16, 18, 16, 14, 14} },
2201 0x10, 0x16, 0x18, 0x40, 0x46,
2202 0x48, 0x30, 0x36, 0x38
2206 /* Data[0].ctledges[0].bchannel */ FREQ2FBIN(5180, 0),
2207 /* Data[0].ctledges[1].bchannel */ FREQ2FBIN(5260, 0),
2208 /* Data[0].ctledges[2].bchannel */ FREQ2FBIN(5280, 0),
2209 /* Data[0].ctledges[3].bchannel */ FREQ2FBIN(5500, 0),
2210 /* Data[0].ctledges[4].bchannel */ FREQ2FBIN(5600, 0),
2211 /* Data[0].ctledges[5].bchannel */ FREQ2FBIN(5700, 0),
2212 /* Data[0].ctledges[6].bchannel */ FREQ2FBIN(5745, 0),
2213 /* Data[0].ctledges[7].bchannel */ FREQ2FBIN(5825, 0)
2216 /* Data[1].ctledges[0].bchannel */ FREQ2FBIN(5180, 0),
2217 /* Data[1].ctledges[1].bchannel */ FREQ2FBIN(5260, 0),
2218 /* Data[1].ctledges[2].bchannel */ FREQ2FBIN(5280, 0),
2219 /* Data[1].ctledges[3].bchannel */ FREQ2FBIN(5500, 0),
2220 /* Data[1].ctledges[4].bchannel */ FREQ2FBIN(5520, 0),
2221 /* Data[1].ctledges[5].bchannel */ FREQ2FBIN(5700, 0),
2222 /* Data[1].ctledges[6].bchannel */ FREQ2FBIN(5745, 0),
2223 /* Data[1].ctledges[7].bchannel */ FREQ2FBIN(5825, 0)
2227 /* Data[2].ctledges[0].bchannel */ FREQ2FBIN(5190, 0),
2228 /* Data[2].ctledges[1].bchannel */ FREQ2FBIN(5230, 0),
2229 /* Data[2].ctledges[2].bchannel */ FREQ2FBIN(5270, 0),
2230 /* Data[2].ctledges[3].bchannel */ FREQ2FBIN(5310, 0),
2231 /* Data[2].ctledges[4].bchannel */ FREQ2FBIN(5510, 0),
2232 /* Data[2].ctledges[5].bchannel */ FREQ2FBIN(5550, 0),
2233 /* Data[2].ctledges[6].bchannel */ FREQ2FBIN(5670, 0),
2234 /* Data[2].ctledges[7].bchannel */ FREQ2FBIN(5755, 0)
2238 /* Data[3].ctledges[0].bchannel */ FREQ2FBIN(5180, 0),
2239 /* Data[3].ctledges[1].bchannel */ FREQ2FBIN(5200, 0),
2240 /* Data[3].ctledges[2].bchannel */ FREQ2FBIN(5260, 0),
2241 /* Data[3].ctledges[3].bchannel */ FREQ2FBIN(5320, 0),
2242 /* Data[3].ctledges[4].bchannel */ FREQ2FBIN(5500, 0),
2243 /* Data[3].ctledges[5].bchannel */ FREQ2FBIN(5700, 0),
2244 /* Data[3].ctledges[6].bchannel */ 0xFF,
2245 /* Data[3].ctledges[7].bchannel */ 0xFF,
2249 /* Data[4].ctledges[0].bchannel */ FREQ2FBIN(5180, 0),
2250 /* Data[4].ctledges[1].bchannel */ FREQ2FBIN(5260, 0),
2251 /* Data[4].ctledges[2].bchannel */ FREQ2FBIN(5500, 0),
2252 /* Data[4].ctledges[3].bchannel */ FREQ2FBIN(5700, 0),
2253 /* Data[4].ctledges[4].bchannel */ 0xFF,
2254 /* Data[4].ctledges[5].bchannel */ 0xFF,
2255 /* Data[4].ctledges[6].bchannel */ 0xFF,
2256 /* Data[4].ctledges[7].bchannel */ 0xFF,
2260 /* Data[5].ctledges[0].bchannel */ FREQ2FBIN(5190, 0),
2261 /* Data[5].ctledges[1].bchannel */ FREQ2FBIN(5270, 0),
2262 /* Data[5].ctledges[2].bchannel */ FREQ2FBIN(5310, 0),
2263 /* Data[5].ctledges[3].bchannel */ FREQ2FBIN(5510, 0),
2264 /* Data[5].ctledges[4].bchannel */ FREQ2FBIN(5590, 0),
2265 /* Data[5].ctledges[5].bchannel */ FREQ2FBIN(5670, 0),
2266 /* Data[5].ctledges[6].bchannel */ 0xFF,
2267 /* Data[5].ctledges[7].bchannel */ 0xFF
2271 /* Data[6].ctledges[0].bchannel */ FREQ2FBIN(5180, 0),
2272 /* Data[6].ctledges[1].bchannel */ FREQ2FBIN(5200, 0),
2273 /* Data[6].ctledges[2].bchannel */ FREQ2FBIN(5220, 0),
2274 /* Data[6].ctledges[3].bchannel */ FREQ2FBIN(5260, 0),
2275 /* Data[6].ctledges[4].bchannel */ FREQ2FBIN(5500, 0),
2276 /* Data[6].ctledges[5].bchannel */ FREQ2FBIN(5600, 0),
2277 /* Data[6].ctledges[6].bchannel */ FREQ2FBIN(5700, 0),
2278 /* Data[6].ctledges[7].bchannel */ FREQ2FBIN(5745, 0)
2282 /* Data[7].ctledges[0].bchannel */ FREQ2FBIN(5180, 0),
2283 /* Data[7].ctledges[1].bchannel */ FREQ2FBIN(5260, 0),
2284 /* Data[7].ctledges[2].bchannel */ FREQ2FBIN(5320, 0),
2285 /* Data[7].ctledges[3].bchannel */ FREQ2FBIN(5500, 0),
2286 /* Data[7].ctledges[4].bchannel */ FREQ2FBIN(5560, 0),
2287 /* Data[7].ctledges[5].bchannel */ FREQ2FBIN(5700, 0),
2288 /* Data[7].ctledges[6].bchannel */ FREQ2FBIN(5745, 0),
2289 /* Data[7].ctledges[7].bchannel */ FREQ2FBIN(5825, 0)
2293 /* Data[8].ctledges[0].bchannel */ FREQ2FBIN(5190, 0),
2294 /* Data[8].ctledges[1].bchannel */ FREQ2FBIN(5230, 0),
2295 /* Data[8].ctledges[2].bchannel */ FREQ2FBIN(5270, 0),
2296 /* Data[8].ctledges[3].bchannel */ FREQ2FBIN(5510, 0),
2297 /* Data[8].ctledges[4].bchannel */ FREQ2FBIN(5550, 0),
2298 /* Data[8].ctledges[5].bchannel */ FREQ2FBIN(5670, 0),
2299 /* Data[8].ctledges[6].bchannel */ FREQ2FBIN(5755, 0),
2300 /* Data[8].ctledges[7].bchannel */ FREQ2FBIN(5795, 0)
2303 .ctlPowerData_5G
= {
2306 CTL(60, 1), CTL(60, 1), CTL(60, 1), CTL(60, 1),
2307 CTL(60, 1), CTL(60, 1), CTL(60, 1), CTL(60, 0),
2312 CTL(60, 1), CTL(60, 1), CTL(60, 1), CTL(60, 1),
2313 CTL(60, 1), CTL(60, 1), CTL(60, 1), CTL(60, 0),
2318 CTL(60, 0), CTL(60, 1), CTL(60, 0), CTL(60, 1),
2319 CTL(60, 1), CTL(60, 1), CTL(60, 1), CTL(60, 1),
2324 CTL(60, 0), CTL(60, 1), CTL(60, 1), CTL(60, 0),
2325 CTL(60, 1), CTL(60, 0), CTL(60, 0), CTL(60, 0),
2330 CTL(60, 1), CTL(60, 1), CTL(60, 1), CTL(60, 0),
2331 CTL(60, 0), CTL(60, 0), CTL(60, 0), CTL(60, 0),
2336 CTL(60, 1), CTL(60, 1), CTL(60, 1), CTL(60, 1),
2337 CTL(60, 1), CTL(60, 0), CTL(60, 0), CTL(60, 0),
2342 CTL(60, 1), CTL(60, 1), CTL(60, 1), CTL(60, 1),
2343 CTL(60, 1), CTL(60, 1), CTL(60, 1), CTL(60, 1),
2348 CTL(60, 1), CTL(60, 1), CTL(60, 0), CTL(60, 1),
2349 CTL(60, 1), CTL(60, 1), CTL(60, 1), CTL(60, 0),
2354 CTL(60, 1), CTL(60, 0), CTL(60, 1), CTL(60, 1),
2355 CTL(60, 1), CTL(60, 1), CTL(60, 0), CTL(60, 1),
2361 static const struct ar9300_eeprom ar9300_h116
= {
2363 .templateVersion
= 4,
2364 .macAddr
= {0x00, 0x03, 0x7f, 0x0, 0x0, 0x0},
2365 .custData
= {"h116-041-f0000"},
2367 .regDmn
= { LE16(0), LE16(0x1f) },
2368 .txrxMask
= 0x33, /* 4 bits tx and 4 bits rx */
2370 .opFlags
= AR5416_OPFLAGS_11G
| AR5416_OPFLAGS_11A
,
2371 .eepMisc
= AR9300_EEPMISC_LITTLE_ENDIAN
,
2374 .blueToothOptions
= 0,
2376 .deviceType
= 5, /* takes lower byte in eeprom location */
2377 .pwrTableOffset
= AR9300_PWR_TABLE_OFFSET
,
2378 .params_for_tuning_caps
= {0, 0},
2379 .featureEnable
= 0x0d,
2381 * bit0 - enable tx temp comp - disabled
2382 * bit1 - enable tx volt comp - disabled
2383 * bit2 - enable fastClock - enabled
2384 * bit3 - enable doubling - enabled
2385 * bit4 - enable internal regulator - disabled
2386 * bit5 - enable pa predistortion - disabled
2388 .miscConfiguration
= 0, /* bit0 - turn down drivestrength */
2389 .eepromWriteEnableGpio
= 6,
2390 .wlanDisableGpio
= 0,
2392 .rxBandSelectGpio
= 0xff,
2397 /* ar9300_modal_eep_header 2g */
2398 /* 4 idle,t1,t2,b(4 bits per setting) */
2399 .antCtrlCommon
= LE32(0x110),
2400 /* 4 ra1l1, ra2l1, ra1l2, ra2l2, ra12 */
2401 .antCtrlCommon2
= LE32(0x44444),
2404 * antCtrlChain[AR9300_MAX_CHAINS]; 6 idle, t, r,
2405 * rx1, rx12, b (2 bits each)
2407 .antCtrlChain
= { LE16(0x10), LE16(0x10), LE16(0x10) },
2410 * xatten1DB[AR9300_MAX_CHAINS]; 3 xatten1_db
2411 * for ar9280 (0xa20c/b20c 5:0)
2413 .xatten1DB
= {0x1f, 0x1f, 0x1f},
2416 * xatten1Margin[AR9300_MAX_CHAINS]; 3 xatten1_margin
2417 * for ar9280 (0xa20c/b20c 16:12
2419 .xatten1Margin
= {0x12, 0x12, 0x12},
2424 * spurChans[OSPREY_EEPROM_MODAL_SPURS]; spur
2425 * channels in usual fbin coding format
2427 .spurChans
= {FREQ2FBIN(2464, 1), 0, 0, 0, 0},
2430 * noiseFloorThreshCh[AR9300_MAX_CHAINS]; 3 Check
2431 * if the register is per chain
2433 .noiseFloorThreshCh
= {-1, 0, 0},
2434 .reserved
= {0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0},
2437 .txFrameToDataStart
= 0x0e,
2438 .txFrameToPaOn
= 0x0e,
2439 .txClip
= 3, /* 4 bits tx_clip, 4 bits dac_scale_cck */
2441 .switchSettling
= 0x2c,
2442 .adcDesiredSize
= -30,
2445 .txFrameToXpaOn
= 0xe,
2447 .papdRateMaskHt20
= LE32(0x0c80C080),
2448 .papdRateMaskHt40
= LE32(0x0080C080),
2450 .xlna_bias_strength
= 0,
2452 0, 0, 0, 0, 0, 0, 0,
2456 .ant_div_control
= 0,
2458 .tempslopextension
= {0, 0, 0, 0, 0, 0, 0, 0}
2465 /* ar9300_cal_data_per_freq_op_loop 2g */
2467 { {0, 0, 0, 0, 0, 0}, {0, 0, 0, 0, 0, 0}, {0, 0, 0, 0, 0, 0} },
2468 { {0, 0, 0, 0, 0, 0}, {0, 0, 0, 0, 0, 0}, {0, 0, 0, 0, 0, 0} },
2469 { {0, 0, 0, 0, 0, 0}, {0, 0, 0, 0, 0, 0}, {0, 0, 0, 0, 0, 0} },
2471 .calTarget_freqbin_Cck
= {
2475 .calTarget_freqbin_2G
= {
2480 .calTarget_freqbin_2GHT20
= {
2485 .calTarget_freqbin_2GHT40
= {
2490 .calTargetPowerCck
= {
2491 /* 1L-5L,5S,11L,11S */
2492 { {34, 34, 34, 34} },
2493 { {34, 34, 34, 34} },
2495 .calTargetPower2G
= {
2497 { {34, 34, 32, 32} },
2498 { {34, 34, 32, 32} },
2499 { {34, 34, 32, 32} },
2501 .calTargetPower2GHT20
= {
2502 { {32, 32, 32, 32, 32, 30, 32, 32, 30, 28, 0, 0, 0, 0} },
2503 { {32, 32, 32, 32, 32, 30, 32, 32, 30, 28, 0, 0, 0, 0} },
2504 { {32, 32, 32, 32, 32, 30, 32, 32, 30, 28, 0, 0, 0, 0} },
2506 .calTargetPower2GHT40
= {
2507 { {30, 30, 30, 30, 30, 28, 30, 30, 28, 26, 0, 0, 0, 0} },
2508 { {30, 30, 30, 30, 30, 28, 30, 30, 28, 26, 0, 0, 0, 0} },
2509 { {30, 30, 30, 30, 30, 28, 30, 30, 28, 26, 0, 0, 0, 0} },
2512 0x11, 0x12, 0x15, 0x17, 0x41, 0x42,
2513 0x45, 0x47, 0x31, 0x32, 0x35, 0x37,
2543 /* Data[4].ctlEdges[0].bChannel */ FREQ2FBIN(2412, 1),
2544 /* Data[4].ctlEdges[1].bChannel */ FREQ2FBIN(2417, 1),
2545 /* Data[4].ctlEdges[2].bChannel */ FREQ2FBIN(2472, 1),
2546 /* Data[4].ctlEdges[3].bChannel */ FREQ2FBIN(2484, 1),
2550 /* Data[5].ctlEdges[0].bChannel */ FREQ2FBIN(2412, 1),
2551 /* Data[5].ctlEdges[1].bChannel */ FREQ2FBIN(2417, 1),
2552 /* Data[5].ctlEdges[2].bChannel */ FREQ2FBIN(2472, 1),
2557 /* Data[6].ctlEdges[0].bChannel */ FREQ2FBIN(2412, 1),
2558 /* Data[6].ctlEdges[1].bChannel */ FREQ2FBIN(2417, 1),
2564 /* Data[7].ctlEdges[0].bChannel */ FREQ2FBIN(2422, 1),
2565 /* Data[7].ctlEdges[1].bChannel */ FREQ2FBIN(2427, 1),
2566 /* Data[7].ctlEdges[2].bChannel */ FREQ2FBIN(2447, 1),
2567 /* Data[7].ctlEdges[3].bChannel */ FREQ2FBIN(2462, 1),
2571 /* Data[8].ctlEdges[0].bChannel */ FREQ2FBIN(2412, 1),
2572 /* Data[8].ctlEdges[1].bChannel */ FREQ2FBIN(2417, 1),
2573 /* Data[8].ctlEdges[2].bChannel */ FREQ2FBIN(2472, 1),
2577 /* Data[9].ctlEdges[0].bChannel */ FREQ2FBIN(2412, 1),
2578 /* Data[9].ctlEdges[1].bChannel */ FREQ2FBIN(2417, 1),
2579 /* Data[9].ctlEdges[2].bChannel */ FREQ2FBIN(2472, 1),
2584 /* Data[10].ctlEdges[0].bChannel */ FREQ2FBIN(2412, 1),
2585 /* Data[10].ctlEdges[1].bChannel */ FREQ2FBIN(2417, 1),
2586 /* Data[10].ctlEdges[2].bChannel */ FREQ2FBIN(2472, 1),
2591 /* Data[11].ctlEdges[0].bChannel */ FREQ2FBIN(2422, 1),
2592 /* Data[11].ctlEdges[1].bChannel */ FREQ2FBIN(2427, 1),
2593 /* Data[11].ctlEdges[2].bChannel */ FREQ2FBIN(2447, 1),
2594 /* Data[11].ctlEdges[3].bChannel */ FREQ2FBIN(2462, 1),
2597 .ctlPowerData_2G
= {
2598 { { CTL(60, 0), CTL(60, 1), CTL(60, 0), CTL(60, 0) } },
2599 { { CTL(60, 0), CTL(60, 1), CTL(60, 0), CTL(60, 0) } },
2600 { { CTL(60, 1), CTL(60, 0), CTL(60, 0), CTL(60, 1) } },
2602 { { CTL(60, 1), CTL(60, 0), CTL(60, 0), CTL(60, 0) } },
2603 { { CTL(60, 0), CTL(60, 1), CTL(60, 0), CTL(60, 0) } },
2604 { { CTL(60, 0), CTL(60, 1), CTL(60, 0), CTL(60, 0) } },
2606 { { CTL(60, 0), CTL(60, 1), CTL(60, 1), CTL(60, 0) } },
2607 { { CTL(60, 0), CTL(60, 1), CTL(60, 0), CTL(60, 0) } },
2608 { { CTL(60, 0), CTL(60, 1), CTL(60, 0), CTL(60, 0) } },
2610 { { CTL(60, 0), CTL(60, 1), CTL(60, 0), CTL(60, 0) } },
2611 { { CTL(60, 0), CTL(60, 1), CTL(60, 1), CTL(60, 1) } },
2612 { { CTL(60, 0), CTL(60, 1), CTL(60, 1), CTL(60, 1) } },
2615 /* 4 idle,t1,t2,b (4 bits per setting) */
2616 .antCtrlCommon
= LE32(0x220),
2617 /* 4 ra1l1, ra2l1, ra1l2,ra2l2,ra12 */
2618 .antCtrlCommon2
= LE32(0x44444),
2619 /* antCtrlChain 6 idle, t,r,rx1,rx12,b (2 bits each) */
2621 LE16(0x150), LE16(0x150), LE16(0x150),
2623 /* xatten1DB 3 xatten1_db for AR9280 (0xa20c/b20c 5:0) */
2624 .xatten1DB
= {0x19, 0x19, 0x19},
2627 * xatten1Margin[AR9300_MAX_CHAINS]; 3 xatten1_margin
2628 * for merlin (0xa20c/b20c 16:12
2630 .xatten1Margin
= {0x14, 0x14, 0x14},
2633 /* spurChans spur channels in usual fbin coding format */
2634 .spurChans
= {0, 0, 0, 0, 0},
2635 /* noiseFloorThreshCh Check if the register is per chain */
2636 .noiseFloorThreshCh
= {-1, 0, 0},
2637 .reserved
= {0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0},
2640 .txFrameToDataStart
= 0x0e,
2641 .txFrameToPaOn
= 0x0e,
2642 .txClip
= 3, /* 4 bits tx_clip, 4 bits dac_scale_cck */
2644 .switchSettling
= 0x2d,
2645 .adcDesiredSize
= -30,
2648 .txFrameToXpaOn
= 0xe,
2650 .papdRateMaskHt20
= LE32(0x0cf0e0e0),
2651 .papdRateMaskHt40
= LE32(0x6cf0e0e0),
2653 .xlna_bias_strength
= 0,
2655 0, 0, 0, 0, 0, 0, 0,
2660 .tempSlopeHigh
= 50,
2661 .xatten1DBLow
= {0, 0, 0},
2662 .xatten1MarginLow
= {0, 0, 0},
2663 .xatten1DBHigh
= {0, 0, 0},
2664 .xatten1MarginHigh
= {0, 0, 0}
2709 .calTarget_freqbin_5G
= {
2719 .calTarget_freqbin_5GHT20
= {
2729 .calTarget_freqbin_5GHT40
= {
2739 .calTargetPower5G
= {
2741 { {30, 30, 28, 24} },
2742 { {30, 30, 28, 24} },
2743 { {30, 30, 28, 24} },
2744 { {30, 30, 28, 24} },
2745 { {30, 30, 28, 24} },
2746 { {30, 30, 28, 24} },
2747 { {30, 30, 28, 24} },
2748 { {30, 30, 28, 24} },
2750 .calTargetPower5GHT20
= {
2752 * 0_8_16,1-3_9-11_17-19,
2753 * 4,5,6,7,12,13,14,15,20,21,22,23
2755 { {30, 30, 30, 28, 24, 20, 30, 28, 24, 20, 0, 0, 0, 0} },
2756 { {30, 30, 30, 28, 24, 20, 30, 28, 24, 20, 0, 0, 0, 0} },
2757 { {30, 30, 30, 26, 22, 18, 30, 26, 22, 18, 0, 0, 0, 0} },
2758 { {30, 30, 30, 26, 22, 18, 30, 26, 22, 18, 0, 0, 0, 0} },
2759 { {30, 30, 30, 24, 20, 16, 30, 24, 20, 16, 0, 0, 0, 0} },
2760 { {30, 30, 30, 24, 20, 16, 30, 24, 20, 16, 0, 0, 0, 0} },
2761 { {30, 30, 30, 22, 18, 14, 30, 22, 18, 14, 0, 0, 0, 0} },
2762 { {30, 30, 30, 22, 18, 14, 30, 22, 18, 14, 0, 0, 0, 0} },
2764 .calTargetPower5GHT40
= {
2766 * 0_8_16,1-3_9-11_17-19,
2767 * 4,5,6,7,12,13,14,15,20,21,22,23
2769 { {28, 28, 28, 26, 22, 18, 28, 26, 22, 18, 0, 0, 0, 0} },
2770 { {28, 28, 28, 26, 22, 18, 28, 26, 22, 18, 0, 0, 0, 0} },
2771 { {28, 28, 28, 24, 20, 16, 28, 24, 20, 16, 0, 0, 0, 0} },
2772 { {28, 28, 28, 24, 20, 16, 28, 24, 20, 16, 0, 0, 0, 0} },
2773 { {28, 28, 28, 22, 18, 14, 28, 22, 18, 14, 0, 0, 0, 0} },
2774 { {28, 28, 28, 22, 18, 14, 28, 22, 18, 14, 0, 0, 0, 0} },
2775 { {28, 28, 28, 20, 16, 12, 28, 20, 16, 12, 0, 0, 0, 0} },
2776 { {28, 28, 28, 20, 16, 12, 28, 20, 16, 12, 0, 0, 0, 0} },
2779 0x10, 0x16, 0x18, 0x40, 0x46,
2780 0x48, 0x30, 0x36, 0x38
2784 /* Data[0].ctlEdges[0].bChannel */ FREQ2FBIN(5180, 0),
2785 /* Data[0].ctlEdges[1].bChannel */ FREQ2FBIN(5260, 0),
2786 /* Data[0].ctlEdges[2].bChannel */ FREQ2FBIN(5280, 0),
2787 /* Data[0].ctlEdges[3].bChannel */ FREQ2FBIN(5500, 0),
2788 /* Data[0].ctlEdges[4].bChannel */ FREQ2FBIN(5600, 0),
2789 /* Data[0].ctlEdges[5].bChannel */ FREQ2FBIN(5700, 0),
2790 /* Data[0].ctlEdges[6].bChannel */ FREQ2FBIN(5745, 0),
2791 /* Data[0].ctlEdges[7].bChannel */ FREQ2FBIN(5825, 0)
2794 /* Data[1].ctlEdges[0].bChannel */ FREQ2FBIN(5180, 0),
2795 /* Data[1].ctlEdges[1].bChannel */ FREQ2FBIN(5260, 0),
2796 /* Data[1].ctlEdges[2].bChannel */ FREQ2FBIN(5280, 0),
2797 /* Data[1].ctlEdges[3].bChannel */ FREQ2FBIN(5500, 0),
2798 /* Data[1].ctlEdges[4].bChannel */ FREQ2FBIN(5520, 0),
2799 /* Data[1].ctlEdges[5].bChannel */ FREQ2FBIN(5700, 0),
2800 /* Data[1].ctlEdges[6].bChannel */ FREQ2FBIN(5745, 0),
2801 /* Data[1].ctlEdges[7].bChannel */ FREQ2FBIN(5825, 0)
2805 /* Data[2].ctlEdges[0].bChannel */ FREQ2FBIN(5190, 0),
2806 /* Data[2].ctlEdges[1].bChannel */ FREQ2FBIN(5230, 0),
2807 /* Data[2].ctlEdges[2].bChannel */ FREQ2FBIN(5270, 0),
2808 /* Data[2].ctlEdges[3].bChannel */ FREQ2FBIN(5310, 0),
2809 /* Data[2].ctlEdges[4].bChannel */ FREQ2FBIN(5510, 0),
2810 /* Data[2].ctlEdges[5].bChannel */ FREQ2FBIN(5550, 0),
2811 /* Data[2].ctlEdges[6].bChannel */ FREQ2FBIN(5670, 0),
2812 /* Data[2].ctlEdges[7].bChannel */ FREQ2FBIN(5755, 0)
2816 /* Data[3].ctlEdges[0].bChannel */ FREQ2FBIN(5180, 0),
2817 /* Data[3].ctlEdges[1].bChannel */ FREQ2FBIN(5200, 0),
2818 /* Data[3].ctlEdges[2].bChannel */ FREQ2FBIN(5260, 0),
2819 /* Data[3].ctlEdges[3].bChannel */ FREQ2FBIN(5320, 0),
2820 /* Data[3].ctlEdges[4].bChannel */ FREQ2FBIN(5500, 0),
2821 /* Data[3].ctlEdges[5].bChannel */ FREQ2FBIN(5700, 0),
2822 /* Data[3].ctlEdges[6].bChannel */ 0xFF,
2823 /* Data[3].ctlEdges[7].bChannel */ 0xFF,
2827 /* Data[4].ctlEdges[0].bChannel */ FREQ2FBIN(5180, 0),
2828 /* Data[4].ctlEdges[1].bChannel */ FREQ2FBIN(5260, 0),
2829 /* Data[4].ctlEdges[2].bChannel */ FREQ2FBIN(5500, 0),
2830 /* Data[4].ctlEdges[3].bChannel */ FREQ2FBIN(5700, 0),
2831 /* Data[4].ctlEdges[4].bChannel */ 0xFF,
2832 /* Data[4].ctlEdges[5].bChannel */ 0xFF,
2833 /* Data[4].ctlEdges[6].bChannel */ 0xFF,
2834 /* Data[4].ctlEdges[7].bChannel */ 0xFF,
2838 /* Data[5].ctlEdges[0].bChannel */ FREQ2FBIN(5190, 0),
2839 /* Data[5].ctlEdges[1].bChannel */ FREQ2FBIN(5270, 0),
2840 /* Data[5].ctlEdges[2].bChannel */ FREQ2FBIN(5310, 0),
2841 /* Data[5].ctlEdges[3].bChannel */ FREQ2FBIN(5510, 0),
2842 /* Data[5].ctlEdges[4].bChannel */ FREQ2FBIN(5590, 0),
2843 /* Data[5].ctlEdges[5].bChannel */ FREQ2FBIN(5670, 0),
2844 /* Data[5].ctlEdges[6].bChannel */ 0xFF,
2845 /* Data[5].ctlEdges[7].bChannel */ 0xFF
2849 /* Data[6].ctlEdges[0].bChannel */ FREQ2FBIN(5180, 0),
2850 /* Data[6].ctlEdges[1].bChannel */ FREQ2FBIN(5200, 0),
2851 /* Data[6].ctlEdges[2].bChannel */ FREQ2FBIN(5220, 0),
2852 /* Data[6].ctlEdges[3].bChannel */ FREQ2FBIN(5260, 0),
2853 /* Data[6].ctlEdges[4].bChannel */ FREQ2FBIN(5500, 0),
2854 /* Data[6].ctlEdges[5].bChannel */ FREQ2FBIN(5600, 0),
2855 /* Data[6].ctlEdges[6].bChannel */ FREQ2FBIN(5700, 0),
2856 /* Data[6].ctlEdges[7].bChannel */ FREQ2FBIN(5745, 0)
2860 /* Data[7].ctlEdges[0].bChannel */ FREQ2FBIN(5180, 0),
2861 /* Data[7].ctlEdges[1].bChannel */ FREQ2FBIN(5260, 0),
2862 /* Data[7].ctlEdges[2].bChannel */ FREQ2FBIN(5320, 0),
2863 /* Data[7].ctlEdges[3].bChannel */ FREQ2FBIN(5500, 0),
2864 /* Data[7].ctlEdges[4].bChannel */ FREQ2FBIN(5560, 0),
2865 /* Data[7].ctlEdges[5].bChannel */ FREQ2FBIN(5700, 0),
2866 /* Data[7].ctlEdges[6].bChannel */ FREQ2FBIN(5745, 0),
2867 /* Data[7].ctlEdges[7].bChannel */ FREQ2FBIN(5825, 0)
2871 /* Data[8].ctlEdges[0].bChannel */ FREQ2FBIN(5190, 0),
2872 /* Data[8].ctlEdges[1].bChannel */ FREQ2FBIN(5230, 0),
2873 /* Data[8].ctlEdges[2].bChannel */ FREQ2FBIN(5270, 0),
2874 /* Data[8].ctlEdges[3].bChannel */ FREQ2FBIN(5510, 0),
2875 /* Data[8].ctlEdges[4].bChannel */ FREQ2FBIN(5550, 0),
2876 /* Data[8].ctlEdges[5].bChannel */ FREQ2FBIN(5670, 0),
2877 /* Data[8].ctlEdges[6].bChannel */ FREQ2FBIN(5755, 0),
2878 /* Data[8].ctlEdges[7].bChannel */ FREQ2FBIN(5795, 0)
2881 .ctlPowerData_5G
= {
2884 CTL(60, 1), CTL(60, 1), CTL(60, 1), CTL(60, 1),
2885 CTL(60, 1), CTL(60, 1), CTL(60, 1), CTL(60, 0),
2890 CTL(60, 1), CTL(60, 1), CTL(60, 1), CTL(60, 1),
2891 CTL(60, 1), CTL(60, 1), CTL(60, 1), CTL(60, 0),
2896 CTL(60, 0), CTL(60, 1), CTL(60, 0), CTL(60, 1),
2897 CTL(60, 1), CTL(60, 1), CTL(60, 1), CTL(60, 1),
2902 CTL(60, 0), CTL(60, 1), CTL(60, 1), CTL(60, 0),
2903 CTL(60, 1), CTL(60, 0), CTL(60, 0), CTL(60, 0),
2908 CTL(60, 1), CTL(60, 1), CTL(60, 1), CTL(60, 0),
2909 CTL(60, 0), CTL(60, 0), CTL(60, 0), CTL(60, 0),
2914 CTL(60, 1), CTL(60, 1), CTL(60, 1), CTL(60, 1),
2915 CTL(60, 1), CTL(60, 0), CTL(60, 0), CTL(60, 0),
2920 CTL(60, 1), CTL(60, 1), CTL(60, 1), CTL(60, 1),
2921 CTL(60, 1), CTL(60, 1), CTL(60, 1), CTL(60, 1),
2926 CTL(60, 1), CTL(60, 1), CTL(60, 0), CTL(60, 1),
2927 CTL(60, 1), CTL(60, 1), CTL(60, 1), CTL(60, 0),
2932 CTL(60, 1), CTL(60, 0), CTL(60, 1), CTL(60, 1),
2933 CTL(60, 1), CTL(60, 1), CTL(60, 0), CTL(60, 1),
2940 static const struct ar9300_eeprom
*ar9300_eep_templates
[] = {
2948 static const struct ar9300_eeprom
*ar9003_eeprom_struct_find_by_id(int id
)
2952 for (it
= 0; it
< ARRAY_SIZE(ar9300_eep_templates
); it
++)
2953 if (ar9300_eep_templates
[it
]->templateVersion
== id
)
2954 return ar9300_eep_templates
[it
];
2958 static int ath9k_hw_ar9300_check_eeprom(struct ath_hw
*ah
)
2963 static int interpolate(int x
, int xa
, int xb
, int ya
, int yb
)
2965 int bf
, factor
, plus
;
2967 bf
= 2 * (yb
- ya
) * (x
- xa
) / (xb
- xa
);
2970 return ya
+ factor
+ plus
;
2973 static u32
ath9k_hw_ar9300_get_eeprom(struct ath_hw
*ah
,
2974 enum eeprom_param param
)
2976 struct ar9300_eeprom
*eep
= &ah
->eeprom
.ar9300_eep
;
2977 struct ar9300_base_eep_hdr
*pBase
= &eep
->baseEepHeader
;
2981 return get_unaligned_be16(eep
->macAddr
);
2983 return get_unaligned_be16(eep
->macAddr
+ 2);
2985 return get_unaligned_be16(eep
->macAddr
+ 4);
2987 return le16_to_cpu(pBase
->regDmn
[0]);
2989 return pBase
->deviceCap
;
2991 return pBase
->opCapFlags
.opFlags
;
2993 return pBase
->rfSilent
;
2995 return (pBase
->txrxMask
>> 4) & 0xf;
2997 return pBase
->txrxMask
& 0xf;
2999 return !!(pBase
->featureEnable
& BIT(5));
3000 case EEP_CHAIN_MASK_REDUCE
:
3001 return (pBase
->miscConfiguration
>> 0x3) & 0x1;
3002 case EEP_ANT_DIV_CTL1
:
3003 if (AR_SREV_9565(ah
))
3004 return AR9300_EEP_ANTDIV_CONTROL_DEFAULT_VALUE
;
3006 return eep
->base_ext1
.ant_div_control
;
3007 case EEP_ANTENNA_GAIN_5G
:
3008 return eep
->modalHeader5G
.antennaGain
;
3009 case EEP_ANTENNA_GAIN_2G
:
3010 return eep
->modalHeader2G
.antennaGain
;
3016 static bool ar9300_eeprom_read_byte(struct ath_hw
*ah
, int address
,
3021 if (unlikely(!ath9k_hw_nvram_read(ah
, address
/ 2, &val
)))
3024 *buffer
= (val
>> (8 * (address
% 2))) & 0xff;
3028 static bool ar9300_eeprom_read_word(struct ath_hw
*ah
, int address
,
3033 if (unlikely(!ath9k_hw_nvram_read(ah
, address
/ 2, &val
)))
3036 buffer
[0] = val
>> 8;
3037 buffer
[1] = val
& 0xff;
3042 static bool ar9300_read_eeprom(struct ath_hw
*ah
, int address
, u8
*buffer
,
3045 struct ath_common
*common
= ath9k_hw_common(ah
);
3048 if ((address
< 0) || ((address
+ count
) / 2 > AR9300_EEPROM_SIZE
- 1)) {
3049 ath_dbg(common
, EEPROM
, "eeprom address not in range\n");
3054 * Since we're reading the bytes in reverse order from a little-endian
3055 * word stream, an even address means we only use the lower half of
3056 * the 16-bit word at that address
3058 if (address
% 2 == 0) {
3059 if (!ar9300_eeprom_read_byte(ah
, address
--, buffer
++))
3065 for (i
= 0; i
< count
/ 2; i
++) {
3066 if (!ar9300_eeprom_read_word(ah
, address
, buffer
))
3074 if (!ar9300_eeprom_read_byte(ah
, address
, buffer
))
3080 ath_dbg(common
, EEPROM
, "unable to read eeprom region at offset %d\n",
3085 static bool ar9300_otp_read_word(struct ath_hw
*ah
, int addr
, u32
*data
)
3087 REG_READ(ah
, AR9300_OTP_BASE
+ (4 * addr
));
3089 if (!ath9k_hw_wait(ah
, AR9300_OTP_STATUS
, AR9300_OTP_STATUS_TYPE
,
3090 AR9300_OTP_STATUS_VALID
, 1000))
3093 *data
= REG_READ(ah
, AR9300_OTP_READ_DATA
);
3097 static bool ar9300_read_otp(struct ath_hw
*ah
, int address
, u8
*buffer
,
3103 for (i
= 0; i
< count
; i
++) {
3104 int offset
= 8 * ((address
- i
) % 4);
3105 if (!ar9300_otp_read_word(ah
, (address
- i
) / 4, &data
))
3108 buffer
[i
] = (data
>> offset
) & 0xff;
3115 static void ar9300_comp_hdr_unpack(u8
*best
, int *code
, int *reference
,
3116 int *length
, int *major
, int *minor
)
3118 unsigned long value
[4];
3124 *code
= ((value
[0] >> 5) & 0x0007);
3125 *reference
= (value
[0] & 0x001f) | ((value
[1] >> 2) & 0x0020);
3126 *length
= ((value
[1] << 4) & 0x07f0) | ((value
[2] >> 4) & 0x000f);
3127 *major
= (value
[2] & 0x000f);
3128 *minor
= (value
[3] & 0x00ff);
3131 static u16
ar9300_comp_cksum(u8
*data
, int dsize
)
3133 int it
, checksum
= 0;
3135 for (it
= 0; it
< dsize
; it
++) {
3136 checksum
+= data
[it
];
3143 static bool ar9300_uncompress_block(struct ath_hw
*ah
,
3153 struct ath_common
*common
= ath9k_hw_common(ah
);
3157 for (it
= 0; it
< size
; it
+= (length
+2)) {
3161 length
= block
[it
+1];
3164 if (length
> 0 && spot
>= 0 && spot
+length
<= mdataSize
) {
3165 ath_dbg(common
, EEPROM
,
3166 "Restore at %d: spot=%d offset=%d length=%d\n",
3167 it
, spot
, offset
, length
);
3168 memcpy(&mptr
[spot
], &block
[it
+2], length
);
3170 } else if (length
> 0) {
3171 ath_dbg(common
, EEPROM
,
3172 "Bad restore at %d: spot=%d offset=%d length=%d\n",
3173 it
, spot
, offset
, length
);
3180 static int ar9300_compress_decision(struct ath_hw
*ah
,
3185 u8
*word
, int length
, int mdata_size
)
3187 struct ath_common
*common
= ath9k_hw_common(ah
);
3188 const struct ar9300_eeprom
*eep
= NULL
;
3192 if (length
!= mdata_size
) {
3193 ath_dbg(common
, EEPROM
,
3194 "EEPROM structure size mismatch memory=%d eeprom=%d\n",
3195 mdata_size
, length
);
3198 memcpy(mptr
, word
+ COMP_HDR_LEN
, length
);
3199 ath_dbg(common
, EEPROM
,
3200 "restored eeprom %d: uncompressed, length %d\n",
3203 case _CompressBlock
:
3204 if (reference
!= 0) {
3205 eep
= ar9003_eeprom_struct_find_by_id(reference
);
3207 ath_dbg(common
, EEPROM
,
3208 "can't find reference eeprom struct %d\n",
3212 memcpy(mptr
, eep
, mdata_size
);
3214 ath_dbg(common
, EEPROM
,
3215 "restore eeprom %d: block, reference %d, length %d\n",
3216 it
, reference
, length
);
3217 ar9300_uncompress_block(ah
, mptr
, mdata_size
,
3218 (word
+ COMP_HDR_LEN
), length
);
3221 ath_dbg(common
, EEPROM
, "unknown compression code %d\n", code
);
3227 typedef bool (*eeprom_read_op
)(struct ath_hw
*ah
, int address
, u8
*buffer
,
3230 static bool ar9300_check_header(void *data
)
3233 return !(*word
== 0 || *word
== ~0);
3236 static bool ar9300_check_eeprom_header(struct ath_hw
*ah
, eeprom_read_op read
,
3241 if (!read(ah
, base_addr
, header
, 4))
3244 return ar9300_check_header(header
);
3247 static int ar9300_eeprom_restore_flash(struct ath_hw
*ah
, u8
*mptr
,
3250 u16
*data
= (u16
*) mptr
;
3253 for (i
= 0; i
< mdata_size
/ 2; i
++, data
++)
3254 if (!ath9k_hw_nvram_read(ah
, i
, data
))
3260 * Read the configuration data from the eeprom.
3261 * The data can be put in any specified memory buffer.
3263 * Returns -1 on error.
3264 * Returns address of next memory location on success.
3266 static int ar9300_eeprom_restore_internal(struct ath_hw
*ah
,
3267 u8
*mptr
, int mdata_size
)
3274 int reference
, length
, major
, minor
;
3277 u16 checksum
, mchecksum
;
3278 struct ath_common
*common
= ath9k_hw_common(ah
);
3279 struct ar9300_eeprom
*eep
;
3280 eeprom_read_op read
;
3282 if (ath9k_hw_use_flash(ah
)) {
3285 if (ar9300_eeprom_restore_flash(ah
, mptr
, mdata_size
))
3288 /* check if eeprom contains valid data */
3289 eep
= (struct ar9300_eeprom
*) mptr
;
3290 txrx
= eep
->baseEepHeader
.txrxMask
;
3291 if (txrx
!= 0 && txrx
!= 0xff)
3295 word
= kzalloc(2048, GFP_KERNEL
);
3299 memcpy(mptr
, &ar9300_default
, mdata_size
);
3301 read
= ar9300_read_eeprom
;
3302 if (AR_SREV_9485(ah
))
3303 cptr
= AR9300_BASE_ADDR_4K
;
3304 else if (AR_SREV_9330(ah
))
3305 cptr
= AR9300_BASE_ADDR_512
;
3307 cptr
= AR9300_BASE_ADDR
;
3308 ath_dbg(common
, EEPROM
, "Trying EEPROM access at Address 0x%04x\n",
3310 if (ar9300_check_eeprom_header(ah
, read
, cptr
))
3313 cptr
= AR9300_BASE_ADDR_4K
;
3314 ath_dbg(common
, EEPROM
, "Trying EEPROM access at Address 0x%04x\n",
3316 if (ar9300_check_eeprom_header(ah
, read
, cptr
))
3319 cptr
= AR9300_BASE_ADDR_512
;
3320 ath_dbg(common
, EEPROM
, "Trying EEPROM access at Address 0x%04x\n",
3322 if (ar9300_check_eeprom_header(ah
, read
, cptr
))
3325 read
= ar9300_read_otp
;
3326 cptr
= AR9300_BASE_ADDR
;
3327 ath_dbg(common
, EEPROM
, "Trying OTP access at Address 0x%04x\n", cptr
);
3328 if (ar9300_check_eeprom_header(ah
, read
, cptr
))
3331 cptr
= AR9300_BASE_ADDR_512
;
3332 ath_dbg(common
, EEPROM
, "Trying OTP access at Address 0x%04x\n", cptr
);
3333 if (ar9300_check_eeprom_header(ah
, read
, cptr
))
3339 ath_dbg(common
, EEPROM
, "Found valid EEPROM data\n");
3341 for (it
= 0; it
< MSTATE
; it
++) {
3342 if (!read(ah
, cptr
, word
, COMP_HDR_LEN
))
3345 if (!ar9300_check_header(word
))
3348 ar9300_comp_hdr_unpack(word
, &code
, &reference
,
3349 &length
, &major
, &minor
);
3350 ath_dbg(common
, EEPROM
,
3351 "Found block at %x: code=%d ref=%d length=%d major=%d minor=%d\n",
3352 cptr
, code
, reference
, length
, major
, minor
);
3353 if ((!AR_SREV_9485(ah
) && length
>= 1024) ||
3354 (AR_SREV_9485(ah
) && length
> EEPROM_DATA_LEN_9485
)) {
3355 ath_dbg(common
, EEPROM
, "Skipping bad header\n");
3356 cptr
-= COMP_HDR_LEN
;
3361 read(ah
, cptr
, word
, COMP_HDR_LEN
+ osize
+ COMP_CKSUM_LEN
);
3362 checksum
= ar9300_comp_cksum(&word
[COMP_HDR_LEN
], length
);
3363 mchecksum
= get_unaligned_le16(&word
[COMP_HDR_LEN
+ osize
]);
3364 ath_dbg(common
, EEPROM
, "checksum %x %x\n",
3365 checksum
, mchecksum
);
3366 if (checksum
== mchecksum
) {
3367 ar9300_compress_decision(ah
, it
, code
, reference
, mptr
,
3368 word
, length
, mdata_size
);
3370 ath_dbg(common
, EEPROM
,
3371 "skipping block with bad checksum\n");
3373 cptr
-= (COMP_HDR_LEN
+ osize
+ COMP_CKSUM_LEN
);
3385 * Restore the configuration structure by reading the eeprom.
3386 * This function destroys any existing in-memory structure
3389 static bool ath9k_hw_ar9300_fill_eeprom(struct ath_hw
*ah
)
3391 u8
*mptr
= (u8
*) &ah
->eeprom
.ar9300_eep
;
3393 if (ar9300_eeprom_restore_internal(ah
, mptr
,
3394 sizeof(struct ar9300_eeprom
)) < 0)
3400 #if defined(CONFIG_ATH9K_DEBUGFS) || defined(CONFIG_ATH9K_HTC_DEBUGFS)
3401 static u32
ar9003_dump_modal_eeprom(char *buf
, u32 len
, u32 size
,
3402 struct ar9300_modal_eep_header
*modal_hdr
)
3404 PR_EEP("Chain0 Ant. Control", le16_to_cpu(modal_hdr
->antCtrlChain
[0]));
3405 PR_EEP("Chain1 Ant. Control", le16_to_cpu(modal_hdr
->antCtrlChain
[1]));
3406 PR_EEP("Chain2 Ant. Control", le16_to_cpu(modal_hdr
->antCtrlChain
[2]));
3407 PR_EEP("Ant. Common Control", le32_to_cpu(modal_hdr
->antCtrlCommon
));
3408 PR_EEP("Ant. Common Control2", le32_to_cpu(modal_hdr
->antCtrlCommon2
));
3409 PR_EEP("Ant. Gain", modal_hdr
->antennaGain
);
3410 PR_EEP("Switch Settle", modal_hdr
->switchSettling
);
3411 PR_EEP("Chain0 xatten1DB", modal_hdr
->xatten1DB
[0]);
3412 PR_EEP("Chain1 xatten1DB", modal_hdr
->xatten1DB
[1]);
3413 PR_EEP("Chain2 xatten1DB", modal_hdr
->xatten1DB
[2]);
3414 PR_EEP("Chain0 xatten1Margin", modal_hdr
->xatten1Margin
[0]);
3415 PR_EEP("Chain1 xatten1Margin", modal_hdr
->xatten1Margin
[1]);
3416 PR_EEP("Chain2 xatten1Margin", modal_hdr
->xatten1Margin
[2]);
3417 PR_EEP("Temp Slope", modal_hdr
->tempSlope
);
3418 PR_EEP("Volt Slope", modal_hdr
->voltSlope
);
3419 PR_EEP("spur Channels0", modal_hdr
->spurChans
[0]);
3420 PR_EEP("spur Channels1", modal_hdr
->spurChans
[1]);
3421 PR_EEP("spur Channels2", modal_hdr
->spurChans
[2]);
3422 PR_EEP("spur Channels3", modal_hdr
->spurChans
[3]);
3423 PR_EEP("spur Channels4", modal_hdr
->spurChans
[4]);
3424 PR_EEP("Chain0 NF Threshold", modal_hdr
->noiseFloorThreshCh
[0]);
3425 PR_EEP("Chain1 NF Threshold", modal_hdr
->noiseFloorThreshCh
[1]);
3426 PR_EEP("Chain2 NF Threshold", modal_hdr
->noiseFloorThreshCh
[2]);
3427 PR_EEP("Quick Drop", modal_hdr
->quick_drop
);
3428 PR_EEP("txEndToXpaOff", modal_hdr
->txEndToXpaOff
);
3429 PR_EEP("xPA Bias Level", modal_hdr
->xpaBiasLvl
);
3430 PR_EEP("txFrameToDataStart", modal_hdr
->txFrameToDataStart
);
3431 PR_EEP("txFrameToPaOn", modal_hdr
->txFrameToPaOn
);
3432 PR_EEP("txFrameToXpaOn", modal_hdr
->txFrameToXpaOn
);
3433 PR_EEP("txClip", modal_hdr
->txClip
);
3434 PR_EEP("ADC Desired size", modal_hdr
->adcDesiredSize
);
3439 static u32
ar9003_dump_cal_data(struct ath_hw
*ah
, char *buf
, u32 len
, u32 size
,
3442 struct ar9300_eeprom
*eep
= &ah
->eeprom
.ar9300_eep
;
3443 struct ar9300_base_eep_hdr
*pBase
;
3444 struct ar9300_cal_data_per_freq_op_loop
*cal_pier
;
3449 pBase
= &eep
->baseEepHeader
;
3452 cal_pier_nr
= AR9300_NUM_2G_CAL_PIERS
;
3454 cal_pier_nr
= AR9300_NUM_5G_CAL_PIERS
;
3456 for (i
= 0; i
< AR9300_MAX_CHAINS
; i
++) {
3457 if (!((pBase
->txrxMask
>> i
) & 1))
3460 len
+= scnprintf(buf
+ len
, size
- len
, "Chain %d\n", i
);
3462 len
+= scnprintf(buf
+ len
, size
- len
,
3463 "Freq\t ref\tvolt\ttemp\tnf_cal\tnf_pow\trx_temp\n");
3465 for (j
= 0; j
< cal_pier_nr
; j
++) {
3467 cal_pier
= &eep
->calPierData2G
[i
][j
];
3468 freq
= 2300 + eep
->calFreqPier2G
[j
];
3470 cal_pier
= &eep
->calPierData5G
[i
][j
];
3471 freq
= 4800 + eep
->calFreqPier5G
[j
] * 5;
3474 len
+= scnprintf(buf
+ len
, size
- len
,
3477 len
+= scnprintf(buf
+ len
, size
- len
,
3478 "%d\t%d\t%d\t%d\t%d\t%d\n",
3482 cal_pier
->rxTempMeas
?
3483 N2DBM(cal_pier
->rxNoisefloorCal
) : 0,
3484 cal_pier
->rxTempMeas
?
3485 N2DBM(cal_pier
->rxNoisefloorPower
) : 0,
3486 cal_pier
->rxTempMeas
);
3493 static u32
ath9k_hw_ar9003_dump_eeprom(struct ath_hw
*ah
, bool dump_base_hdr
,
3494 u8
*buf
, u32 len
, u32 size
)
3496 struct ar9300_eeprom
*eep
= &ah
->eeprom
.ar9300_eep
;
3497 struct ar9300_base_eep_hdr
*pBase
;
3499 if (!dump_base_hdr
) {
3500 len
+= scnprintf(buf
+ len
, size
- len
,
3501 "%20s :\n", "2GHz modal Header");
3502 len
= ar9003_dump_modal_eeprom(buf
, len
, size
,
3503 &eep
->modalHeader2G
);
3505 len
+= scnprintf(buf
+ len
, size
- len
, "Calibration data\n");
3506 len
= ar9003_dump_cal_data(ah
, buf
, len
, size
, true);
3508 len
+= scnprintf(buf
+ len
, size
- len
,
3509 "%20s :\n", "5GHz modal Header");
3510 len
= ar9003_dump_modal_eeprom(buf
, len
, size
,
3511 &eep
->modalHeader5G
);
3513 len
+= scnprintf(buf
+ len
, size
- len
, "Calibration data\n");
3514 len
= ar9003_dump_cal_data(ah
, buf
, len
, size
, false);
3519 pBase
= &eep
->baseEepHeader
;
3521 PR_EEP("EEPROM Version", ah
->eeprom
.ar9300_eep
.eepromVersion
);
3522 PR_EEP("RegDomain1", le16_to_cpu(pBase
->regDmn
[0]));
3523 PR_EEP("RegDomain2", le16_to_cpu(pBase
->regDmn
[1]));
3524 PR_EEP("TX Mask", (pBase
->txrxMask
>> 4));
3525 PR_EEP("RX Mask", (pBase
->txrxMask
& 0x0f));
3526 PR_EEP("Allow 5GHz", !!(pBase
->opCapFlags
.opFlags
&
3527 AR5416_OPFLAGS_11A
));
3528 PR_EEP("Allow 2GHz", !!(pBase
->opCapFlags
.opFlags
&
3529 AR5416_OPFLAGS_11G
));
3530 PR_EEP("Disable 2GHz HT20", !!(pBase
->opCapFlags
.opFlags
&
3531 AR5416_OPFLAGS_N_2G_HT20
));
3532 PR_EEP("Disable 2GHz HT40", !!(pBase
->opCapFlags
.opFlags
&
3533 AR5416_OPFLAGS_N_2G_HT40
));
3534 PR_EEP("Disable 5Ghz HT20", !!(pBase
->opCapFlags
.opFlags
&
3535 AR5416_OPFLAGS_N_5G_HT20
));
3536 PR_EEP("Disable 5Ghz HT40", !!(pBase
->opCapFlags
.opFlags
&
3537 AR5416_OPFLAGS_N_5G_HT40
));
3538 PR_EEP("Big Endian", !!(pBase
->opCapFlags
.eepMisc
&
3539 AR5416_EEPMISC_BIG_ENDIAN
));
3540 PR_EEP("RF Silent", pBase
->rfSilent
);
3541 PR_EEP("BT option", pBase
->blueToothOptions
);
3542 PR_EEP("Device Cap", pBase
->deviceCap
);
3543 PR_EEP("Device Type", pBase
->deviceType
);
3544 PR_EEP("Power Table Offset", pBase
->pwrTableOffset
);
3545 PR_EEP("Tuning Caps1", pBase
->params_for_tuning_caps
[0]);
3546 PR_EEP("Tuning Caps2", pBase
->params_for_tuning_caps
[1]);
3547 PR_EEP("Enable Tx Temp Comp", !!(pBase
->featureEnable
& BIT(0)));
3548 PR_EEP("Enable Tx Volt Comp", !!(pBase
->featureEnable
& BIT(1)));
3549 PR_EEP("Enable fast clock", !!(pBase
->featureEnable
& BIT(2)));
3550 PR_EEP("Enable doubling", !!(pBase
->featureEnable
& BIT(3)));
3551 PR_EEP("Internal regulator", !!(pBase
->featureEnable
& BIT(4)));
3552 PR_EEP("Enable Paprd", !!(pBase
->featureEnable
& BIT(5)));
3553 PR_EEP("Driver Strength", !!(pBase
->miscConfiguration
& BIT(0)));
3554 PR_EEP("Quick Drop", !!(pBase
->miscConfiguration
& BIT(1)));
3555 PR_EEP("Chain mask Reduce", (pBase
->miscConfiguration
>> 0x3) & 0x1);
3556 PR_EEP("Write enable Gpio", pBase
->eepromWriteEnableGpio
);
3557 PR_EEP("WLAN Disable Gpio", pBase
->wlanDisableGpio
);
3558 PR_EEP("WLAN LED Gpio", pBase
->wlanLedGpio
);
3559 PR_EEP("Rx Band Select Gpio", pBase
->rxBandSelectGpio
);
3560 PR_EEP("Tx Gain", pBase
->txrxgain
>> 4);
3561 PR_EEP("Rx Gain", pBase
->txrxgain
& 0xf);
3562 PR_EEP("SW Reg", le32_to_cpu(pBase
->swreg
));
3564 len
+= scnprintf(buf
+ len
, size
- len
, "%20s : %pM\n", "MacAddress",
3565 ah
->eeprom
.ar9300_eep
.macAddr
);
3573 static u32
ath9k_hw_ar9003_dump_eeprom(struct ath_hw
*ah
, bool dump_base_hdr
,
3574 u8
*buf
, u32 len
, u32 size
)
3580 /* XXX: review hardware docs */
3581 static int ath9k_hw_ar9300_get_eeprom_ver(struct ath_hw
*ah
)
3583 return ah
->eeprom
.ar9300_eep
.eepromVersion
;
3586 /* XXX: could be read from the eepromVersion, not sure yet */
3587 static int ath9k_hw_ar9300_get_eeprom_rev(struct ath_hw
*ah
)
3592 static struct ar9300_modal_eep_header
*ar9003_modal_header(struct ath_hw
*ah
,
3595 struct ar9300_eeprom
*eep
= &ah
->eeprom
.ar9300_eep
;
3598 return &eep
->modalHeader2G
;
3600 return &eep
->modalHeader5G
;
3603 static void ar9003_hw_xpa_bias_level_apply(struct ath_hw
*ah
, bool is2ghz
)
3605 int bias
= ar9003_modal_header(ah
, is2ghz
)->xpaBiasLvl
;
3607 if (AR_SREV_9485(ah
) || AR_SREV_9330(ah
) || AR_SREV_9340(ah
) ||
3608 AR_SREV_9531(ah
) || AR_SREV_9561(ah
))
3609 REG_RMW_FIELD(ah
, AR_CH0_TOP2
, AR_CH0_TOP2_XPABIASLVL
, bias
);
3610 else if (AR_SREV_9462(ah
) || AR_SREV_9550(ah
) || AR_SREV_9565(ah
))
3611 REG_RMW_FIELD(ah
, AR_CH0_TOP
, AR_CH0_TOP_XPABIASLVL
, bias
);
3613 REG_RMW_FIELD(ah
, AR_CH0_TOP
, AR_CH0_TOP_XPABIASLVL
, bias
);
3614 REG_RMW_FIELD(ah
, AR_CH0_THERM
,
3615 AR_CH0_THERM_XPABIASLVL_MSB
,
3617 REG_RMW_FIELD(ah
, AR_CH0_THERM
,
3618 AR_CH0_THERM_XPASHORT2GND
, 1);
3622 static u16
ar9003_switch_com_spdt_get(struct ath_hw
*ah
, bool is2ghz
)
3624 return le16_to_cpu(ar9003_modal_header(ah
, is2ghz
)->switchcomspdt
);
3627 u32
ar9003_hw_ant_ctrl_common_get(struct ath_hw
*ah
, bool is2ghz
)
3629 return le32_to_cpu(ar9003_modal_header(ah
, is2ghz
)->antCtrlCommon
);
3632 u32
ar9003_hw_ant_ctrl_common_2_get(struct ath_hw
*ah
, bool is2ghz
)
3634 return le32_to_cpu(ar9003_modal_header(ah
, is2ghz
)->antCtrlCommon2
);
3637 static u16
ar9003_hw_ant_ctrl_chain_get(struct ath_hw
*ah
, int chain
,
3640 __le16 val
= ar9003_modal_header(ah
, is2ghz
)->antCtrlChain
[chain
];
3641 return le16_to_cpu(val
);
3644 static void ar9003_hw_ant_ctrl_apply(struct ath_hw
*ah
, bool is2ghz
)
3646 struct ath_common
*common
= ath9k_hw_common(ah
);
3647 struct ath9k_hw_capabilities
*pCap
= &ah
->caps
;
3649 u32 regval
, value
, gpio
;
3650 static const u32 switch_chain_reg
[AR9300_MAX_CHAINS
] = {
3651 AR_PHY_SWITCH_CHAIN_0
,
3652 AR_PHY_SWITCH_CHAIN_1
,
3653 AR_PHY_SWITCH_CHAIN_2
,
3656 if (AR_SREV_9485(ah
) && (ar9003_hw_get_rx_gain_idx(ah
) == 0)) {
3657 if (ah
->config
.xlna_gpio
)
3658 gpio
= ah
->config
.xlna_gpio
;
3660 gpio
= AR9300_EXT_LNA_CTL_GPIO_AR9485
;
3662 ath9k_hw_gpio_request_out(ah
, gpio
, NULL
,
3663 AR_GPIO_OUTPUT_MUX_AS_PCIE_ATTENTION_LED
);
3666 value
= ar9003_hw_ant_ctrl_common_get(ah
, is2ghz
);
3668 if (AR_SREV_9462(ah
) || AR_SREV_9565(ah
)) {
3669 REG_RMW_FIELD(ah
, AR_PHY_SWITCH_COM
,
3670 AR_SWITCH_TABLE_COM_AR9462_ALL
, value
);
3671 } else if (AR_SREV_9550(ah
) || AR_SREV_9531(ah
) || AR_SREV_9561(ah
)) {
3672 REG_RMW_FIELD(ah
, AR_PHY_SWITCH_COM
,
3673 AR_SWITCH_TABLE_COM_AR9550_ALL
, value
);
3675 REG_RMW_FIELD(ah
, AR_PHY_SWITCH_COM
,
3676 AR_SWITCH_TABLE_COM_ALL
, value
);
3680 * AR9462 defines new switch table for BT/WLAN,
3681 * here's new field name in XXX.ref for both 2G and 5G.
3682 * Register: [GLB_CONTROL] GLB_CONTROL (@0x20044)
3683 * 15:12 R/W SWITCH_TABLE_COM_SPDT_WLAN_RX
3684 * SWITCH_TABLE_COM_SPDT_WLAN_RX
3686 * 11:8 R/W SWITCH_TABLE_COM_SPDT_WLAN_TX
3687 * SWITCH_TABLE_COM_SPDT_WLAN_TX
3689 * 7:4 R/W SWITCH_TABLE_COM_SPDT_WLAN_IDLE
3690 * SWITCH_TABLE_COM_SPDT_WLAN_IDLE
3692 if (AR_SREV_9462_20_OR_LATER(ah
) || AR_SREV_9565(ah
)) {
3693 value
= ar9003_switch_com_spdt_get(ah
, is2ghz
);
3694 REG_RMW_FIELD(ah
, AR_PHY_GLB_CONTROL
,
3695 AR_SWITCH_TABLE_COM_SPDT_ALL
, value
);
3696 REG_SET_BIT(ah
, AR_PHY_GLB_CONTROL
, AR_BTCOEX_CTRL_SPDT_ENABLE
);
3699 value
= ar9003_hw_ant_ctrl_common_2_get(ah
, is2ghz
);
3700 if (AR_SREV_9485(ah
) && common
->bt_ant_diversity
) {
3701 value
&= ~AR_SWITCH_TABLE_COM2_ALL
;
3702 value
|= ah
->config
.ant_ctrl_comm2g_switch_enable
;
3705 REG_RMW_FIELD(ah
, AR_PHY_SWITCH_COM_2
, AR_SWITCH_TABLE_COM2_ALL
, value
);
3707 if ((AR_SREV_9462(ah
)) && (ah
->rxchainmask
== 0x2)) {
3708 value
= ar9003_hw_ant_ctrl_chain_get(ah
, 1, is2ghz
);
3709 REG_RMW_FIELD(ah
, switch_chain_reg
[0],
3710 AR_SWITCH_TABLE_ALL
, value
);
3713 for (chain
= 0; chain
< AR9300_MAX_CHAINS
; chain
++) {
3714 if ((ah
->rxchainmask
& BIT(chain
)) ||
3715 (ah
->txchainmask
& BIT(chain
))) {
3716 value
= ar9003_hw_ant_ctrl_chain_get(ah
, chain
,
3718 REG_RMW_FIELD(ah
, switch_chain_reg
[chain
],
3719 AR_SWITCH_TABLE_ALL
, value
);
3723 if (AR_SREV_9330(ah
) || AR_SREV_9485(ah
) || AR_SREV_9565(ah
)) {
3724 value
= ath9k_hw_ar9300_get_eeprom(ah
, EEP_ANT_DIV_CTL1
);
3726 * main_lnaconf, alt_lnaconf, main_tb, alt_tb
3727 * are the fields present
3729 regval
= REG_READ(ah
, AR_PHY_MC_GAIN_CTRL
);
3730 regval
&= (~AR_ANT_DIV_CTRL_ALL
);
3731 regval
|= (value
& 0x3f) << AR_ANT_DIV_CTRL_ALL_S
;
3733 regval
&= (~AR_PHY_ANT_DIV_LNADIV
);
3734 regval
|= ((value
>> 6) & 0x1) << AR_PHY_ANT_DIV_LNADIV_S
;
3736 if (AR_SREV_9485(ah
) && common
->bt_ant_diversity
)
3737 regval
|= AR_ANT_DIV_ENABLE
;
3739 if (AR_SREV_9565(ah
)) {
3740 if (common
->bt_ant_diversity
) {
3741 regval
|= (1 << AR_PHY_ANT_SW_RX_PROT_S
);
3743 REG_SET_BIT(ah
, AR_PHY_RESTART
,
3744 AR_PHY_RESTART_ENABLE_DIV_M2FLAG
);
3746 /* Force WLAN LNA diversity ON */
3747 REG_SET_BIT(ah
, AR_BTCOEX_WL_LNADIV
,
3748 AR_BTCOEX_WL_LNADIV_FORCE_ON
);
3750 regval
&= ~(1 << AR_PHY_ANT_DIV_LNADIV_S
);
3751 regval
&= ~(1 << AR_PHY_ANT_SW_RX_PROT_S
);
3753 REG_CLR_BIT(ah
, AR_PHY_MC_GAIN_CTRL
,
3754 (1 << AR_PHY_ANT_SW_RX_PROT_S
));
3756 /* Force WLAN LNA diversity OFF */
3757 REG_CLR_BIT(ah
, AR_BTCOEX_WL_LNADIV
,
3758 AR_BTCOEX_WL_LNADIV_FORCE_ON
);
3762 REG_WRITE(ah
, AR_PHY_MC_GAIN_CTRL
, regval
);
3764 /* enable fast_div */
3765 regval
= REG_READ(ah
, AR_PHY_CCK_DETECT
);
3766 regval
&= (~AR_FAST_DIV_ENABLE
);
3767 regval
|= ((value
>> 7) & 0x1) << AR_FAST_DIV_ENABLE_S
;
3769 if ((AR_SREV_9485(ah
) || AR_SREV_9565(ah
))
3770 && common
->bt_ant_diversity
)
3771 regval
|= AR_FAST_DIV_ENABLE
;
3773 REG_WRITE(ah
, AR_PHY_CCK_DETECT
, regval
);
3775 if (pCap
->hw_caps
& ATH9K_HW_CAP_ANT_DIV_COMB
) {
3776 regval
= REG_READ(ah
, AR_PHY_MC_GAIN_CTRL
);
3778 * clear bits 25-30 main_lnaconf, alt_lnaconf,
3781 regval
&= (~(AR_PHY_ANT_DIV_MAIN_LNACONF
|
3782 AR_PHY_ANT_DIV_ALT_LNACONF
|
3783 AR_PHY_ANT_DIV_ALT_GAINTB
|
3784 AR_PHY_ANT_DIV_MAIN_GAINTB
));
3785 /* by default use LNA1 for the main antenna */
3786 regval
|= (ATH_ANT_DIV_COMB_LNA1
<<
3787 AR_PHY_ANT_DIV_MAIN_LNACONF_S
);
3788 regval
|= (ATH_ANT_DIV_COMB_LNA2
<<
3789 AR_PHY_ANT_DIV_ALT_LNACONF_S
);
3790 REG_WRITE(ah
, AR_PHY_MC_GAIN_CTRL
, regval
);
3795 static void ar9003_hw_drive_strength_apply(struct ath_hw
*ah
)
3797 struct ar9300_eeprom
*eep
= &ah
->eeprom
.ar9300_eep
;
3798 struct ar9300_base_eep_hdr
*pBase
= &eep
->baseEepHeader
;
3802 drive_strength
= pBase
->miscConfiguration
& BIT(0);
3803 if (!drive_strength
)
3806 reg
= REG_READ(ah
, AR_PHY_65NM_CH0_BIAS1
);
3814 REG_WRITE(ah
, AR_PHY_65NM_CH0_BIAS1
, reg
);
3816 reg
= REG_READ(ah
, AR_PHY_65NM_CH0_BIAS2
);
3827 REG_WRITE(ah
, AR_PHY_65NM_CH0_BIAS2
, reg
);
3829 reg
= REG_READ(ah
, AR_PHY_65NM_CH0_BIAS4
);
3834 REG_WRITE(ah
, AR_PHY_65NM_CH0_BIAS4
, reg
);
3837 static u16
ar9003_hw_atten_chain_get(struct ath_hw
*ah
, int chain
,
3838 struct ath9k_channel
*chan
)
3842 struct ar9300_eeprom
*eep
= &ah
->eeprom
.ar9300_eep
;
3844 if (chain
>= 0 && chain
< 3) {
3845 if (IS_CHAN_2GHZ(chan
))
3846 return eep
->modalHeader2G
.xatten1DB
[chain
];
3847 else if (eep
->base_ext2
.xatten1DBLow
[chain
] != 0) {
3848 t
[0] = eep
->base_ext2
.xatten1DBLow
[chain
];
3850 t
[1] = eep
->modalHeader5G
.xatten1DB
[chain
];
3852 t
[2] = eep
->base_ext2
.xatten1DBHigh
[chain
];
3854 value
= ar9003_hw_power_interpolate((s32
) chan
->channel
,
3858 return eep
->modalHeader5G
.xatten1DB
[chain
];
3865 static u16
ar9003_hw_atten_chain_get_margin(struct ath_hw
*ah
, int chain
,
3866 struct ath9k_channel
*chan
)
3870 struct ar9300_eeprom
*eep
= &ah
->eeprom
.ar9300_eep
;
3872 if (chain
>= 0 && chain
< 3) {
3873 if (IS_CHAN_2GHZ(chan
))
3874 return eep
->modalHeader2G
.xatten1Margin
[chain
];
3875 else if (eep
->base_ext2
.xatten1MarginLow
[chain
] != 0) {
3876 t
[0] = eep
->base_ext2
.xatten1MarginLow
[chain
];
3878 t
[1] = eep
->modalHeader5G
.xatten1Margin
[chain
];
3880 t
[2] = eep
->base_ext2
.xatten1MarginHigh
[chain
];
3882 value
= ar9003_hw_power_interpolate((s32
) chan
->channel
,
3886 return eep
->modalHeader5G
.xatten1Margin
[chain
];
3892 static void ar9003_hw_atten_apply(struct ath_hw
*ah
, struct ath9k_channel
*chan
)
3896 unsigned long ext_atten_reg
[3] = {AR_PHY_EXT_ATTEN_CTL_0
,
3897 AR_PHY_EXT_ATTEN_CTL_1
,
3898 AR_PHY_EXT_ATTEN_CTL_2
,
3901 if ((AR_SREV_9462(ah
)) && (ah
->rxchainmask
== 0x2)) {
3902 value
= ar9003_hw_atten_chain_get(ah
, 1, chan
);
3903 REG_RMW_FIELD(ah
, ext_atten_reg
[0],
3904 AR_PHY_EXT_ATTEN_CTL_XATTEN1_DB
, value
);
3906 value
= ar9003_hw_atten_chain_get_margin(ah
, 1, chan
);
3907 REG_RMW_FIELD(ah
, ext_atten_reg
[0],
3908 AR_PHY_EXT_ATTEN_CTL_XATTEN1_MARGIN
,
3912 /* Test value. if 0 then attenuation is unused. Don't load anything. */
3913 for (i
= 0; i
< 3; i
++) {
3914 if (ah
->txchainmask
& BIT(i
)) {
3915 value
= ar9003_hw_atten_chain_get(ah
, i
, chan
);
3916 REG_RMW_FIELD(ah
, ext_atten_reg
[i
],
3917 AR_PHY_EXT_ATTEN_CTL_XATTEN1_DB
, value
);
3919 if (AR_SREV_9485(ah
) &&
3920 (ar9003_hw_get_rx_gain_idx(ah
) == 0) &&
3921 ah
->config
.xatten_margin_cfg
)
3924 value
= ar9003_hw_atten_chain_get_margin(ah
, i
, chan
);
3926 if (ah
->config
.alt_mingainidx
)
3927 REG_RMW_FIELD(ah
, AR_PHY_EXT_ATTEN_CTL_0
,
3928 AR_PHY_EXT_ATTEN_CTL_XATTEN1_MARGIN
,
3931 REG_RMW_FIELD(ah
, ext_atten_reg
[i
],
3932 AR_PHY_EXT_ATTEN_CTL_XATTEN1_MARGIN
,
3938 static bool is_pmu_set(struct ath_hw
*ah
, u32 pmu_reg
, int pmu_set
)
3942 while (pmu_set
!= REG_READ(ah
, pmu_reg
)) {
3945 REG_WRITE(ah
, pmu_reg
, pmu_set
);
3952 void ar9003_hw_internal_regulator_apply(struct ath_hw
*ah
)
3954 struct ar9300_eeprom
*eep
= &ah
->eeprom
.ar9300_eep
;
3955 struct ar9300_base_eep_hdr
*pBase
= &eep
->baseEepHeader
;
3958 if (pBase
->featureEnable
& BIT(4)) {
3959 if (AR_SREV_9330(ah
) || AR_SREV_9485(ah
)) {
3962 reg_pmu_set
= REG_READ(ah
, AR_PHY_PMU2
) & ~AR_PHY_PMU2_PGM
;
3963 REG_WRITE(ah
, AR_PHY_PMU2
, reg_pmu_set
);
3964 if (!is_pmu_set(ah
, AR_PHY_PMU2
, reg_pmu_set
))
3967 if (AR_SREV_9330(ah
)) {
3968 if (ah
->is_clk_25mhz
) {
3969 reg_pmu_set
= (3 << 1) | (8 << 4) |
3970 (3 << 8) | (1 << 14) |
3971 (6 << 17) | (1 << 20) |
3974 reg_pmu_set
= (4 << 1) | (7 << 4) |
3975 (3 << 8) | (1 << 14) |
3976 (6 << 17) | (1 << 20) |
3980 reg_pmu_set
= (5 << 1) | (7 << 4) |
3981 (2 << 8) | (2 << 14) |
3982 (6 << 17) | (1 << 20) |
3983 (3 << 24) | (1 << 28);
3986 REG_WRITE(ah
, AR_PHY_PMU1
, reg_pmu_set
);
3987 if (!is_pmu_set(ah
, AR_PHY_PMU1
, reg_pmu_set
))
3990 reg_pmu_set
= (REG_READ(ah
, AR_PHY_PMU2
) & ~0xFFC00000)
3992 REG_WRITE(ah
, AR_PHY_PMU2
, reg_pmu_set
);
3993 if (!is_pmu_set(ah
, AR_PHY_PMU2
, reg_pmu_set
))
3996 reg_pmu_set
= (REG_READ(ah
, AR_PHY_PMU2
) & ~0x00200000)
3998 REG_WRITE(ah
, AR_PHY_PMU2
, reg_pmu_set
);
3999 if (!is_pmu_set(ah
, AR_PHY_PMU2
, reg_pmu_set
))
4001 } else if (AR_SREV_9462(ah
) || AR_SREV_9565(ah
) ||
4003 reg_val
= le32_to_cpu(pBase
->swreg
);
4004 REG_WRITE(ah
, AR_PHY_PMU1
, reg_val
);
4006 if (AR_SREV_9561(ah
))
4007 REG_WRITE(ah
, AR_PHY_PMU2
, 0x10200000);
4009 /* Internal regulator is ON. Write swreg register. */
4010 reg_val
= le32_to_cpu(pBase
->swreg
);
4011 REG_WRITE(ah
, AR_RTC_REG_CONTROL1
,
4012 REG_READ(ah
, AR_RTC_REG_CONTROL1
) &
4013 (~AR_RTC_REG_CONTROL1_SWREG_PROGRAM
));
4014 REG_WRITE(ah
, AR_RTC_REG_CONTROL0
, reg_val
);
4015 /* Set REG_CONTROL1.SWREG_PROGRAM */
4016 REG_WRITE(ah
, AR_RTC_REG_CONTROL1
,
4018 AR_RTC_REG_CONTROL1
) |
4019 AR_RTC_REG_CONTROL1_SWREG_PROGRAM
);
4022 if (AR_SREV_9330(ah
) || AR_SREV_9485(ah
)) {
4023 REG_RMW_FIELD(ah
, AR_PHY_PMU2
, AR_PHY_PMU2_PGM
, 0);
4024 while (REG_READ_FIELD(ah
, AR_PHY_PMU2
,
4028 REG_RMW_FIELD(ah
, AR_PHY_PMU1
, AR_PHY_PMU1_PWD
, 0x1);
4029 while (!REG_READ_FIELD(ah
, AR_PHY_PMU1
,
4032 REG_RMW_FIELD(ah
, AR_PHY_PMU2
, AR_PHY_PMU2_PGM
, 0x1);
4033 while (!REG_READ_FIELD(ah
, AR_PHY_PMU2
,
4036 } else if (AR_SREV_9462(ah
) || AR_SREV_9565(ah
))
4037 REG_RMW_FIELD(ah
, AR_PHY_PMU1
, AR_PHY_PMU1_PWD
, 0x1);
4039 reg_val
= REG_READ(ah
, AR_RTC_SLEEP_CLK
) |
4040 AR_RTC_FORCE_SWREG_PRD
;
4041 REG_WRITE(ah
, AR_RTC_SLEEP_CLK
, reg_val
);
4047 static void ar9003_hw_apply_tuning_caps(struct ath_hw
*ah
)
4049 struct ar9300_eeprom
*eep
= &ah
->eeprom
.ar9300_eep
;
4050 u8 tuning_caps_param
= eep
->baseEepHeader
.params_for_tuning_caps
[0];
4052 if (AR_SREV_9340(ah
) || AR_SREV_9531(ah
))
4055 if (eep
->baseEepHeader
.featureEnable
& 0x40) {
4056 tuning_caps_param
&= 0x7f;
4057 REG_RMW_FIELD(ah
, AR_CH0_XTAL
, AR_CH0_XTAL_CAPINDAC
,
4059 REG_RMW_FIELD(ah
, AR_CH0_XTAL
, AR_CH0_XTAL_CAPOUTDAC
,
4064 static void ar9003_hw_quick_drop_apply(struct ath_hw
*ah
, u16 freq
)
4066 struct ar9300_eeprom
*eep
= &ah
->eeprom
.ar9300_eep
;
4067 struct ar9300_base_eep_hdr
*pBase
= &eep
->baseEepHeader
;
4069 s32 t
[3], f
[3] = {5180, 5500, 5785};
4071 if (!(pBase
->miscConfiguration
& BIT(4)))
4074 if (AR_SREV_9300(ah
) || AR_SREV_9580(ah
) || AR_SREV_9340(ah
)) {
4076 quick_drop
= eep
->modalHeader2G
.quick_drop
;
4078 t
[0] = eep
->base_ext1
.quick_drop_low
;
4079 t
[1] = eep
->modalHeader5G
.quick_drop
;
4080 t
[2] = eep
->base_ext1
.quick_drop_high
;
4081 quick_drop
= ar9003_hw_power_interpolate(freq
, f
, t
, 3);
4083 REG_RMW_FIELD(ah
, AR_PHY_AGC
, AR_PHY_AGC_QUICK_DROP
, quick_drop
);
4087 static void ar9003_hw_txend_to_xpa_off_apply(struct ath_hw
*ah
, bool is2ghz
)
4091 value
= ar9003_modal_header(ah
, is2ghz
)->txEndToXpaOff
;
4093 REG_RMW_FIELD(ah
, AR_PHY_XPA_TIMING_CTL
,
4094 AR_PHY_XPA_TIMING_CTL_TX_END_XPAB_OFF
, value
);
4095 REG_RMW_FIELD(ah
, AR_PHY_XPA_TIMING_CTL
,
4096 AR_PHY_XPA_TIMING_CTL_TX_END_XPAA_OFF
, value
);
4099 static void ar9003_hw_xpa_timing_control_apply(struct ath_hw
*ah
, bool is2ghz
)
4101 struct ar9300_eeprom
*eep
= &ah
->eeprom
.ar9300_eep
;
4104 if (!(eep
->baseEepHeader
.featureEnable
& 0x80))
4107 if (!AR_SREV_9300(ah
) &&
4108 !AR_SREV_9340(ah
) &&
4109 !AR_SREV_9580(ah
) &&
4110 !AR_SREV_9531(ah
) &&
4114 xpa_ctl
= ar9003_modal_header(ah
, is2ghz
)->txFrameToXpaOn
;
4116 REG_RMW_FIELD(ah
, AR_PHY_XPA_TIMING_CTL
,
4117 AR_PHY_XPA_TIMING_CTL_FRAME_XPAB_ON
, xpa_ctl
);
4119 REG_RMW_FIELD(ah
, AR_PHY_XPA_TIMING_CTL
,
4120 AR_PHY_XPA_TIMING_CTL_FRAME_XPAA_ON
, xpa_ctl
);
4123 static void ar9003_hw_xlna_bias_strength_apply(struct ath_hw
*ah
, bool is2ghz
)
4125 struct ar9300_eeprom
*eep
= &ah
->eeprom
.ar9300_eep
;
4128 if (!(eep
->baseEepHeader
.miscConfiguration
& 0x40))
4131 if (!AR_SREV_9300(ah
))
4134 bias
= ar9003_modal_header(ah
, is2ghz
)->xlna_bias_strength
;
4135 REG_RMW_FIELD(ah
, AR_PHY_65NM_CH0_RXTX4
, AR_PHY_65NM_RXTX4_XLNA_BIAS
,
4138 REG_RMW_FIELD(ah
, AR_PHY_65NM_CH1_RXTX4
, AR_PHY_65NM_RXTX4_XLNA_BIAS
,
4141 REG_RMW_FIELD(ah
, AR_PHY_65NM_CH2_RXTX4
, AR_PHY_65NM_RXTX4_XLNA_BIAS
,
4145 static int ar9003_hw_get_thermometer(struct ath_hw
*ah
)
4147 struct ar9300_eeprom
*eep
= &ah
->eeprom
.ar9300_eep
;
4148 struct ar9300_base_eep_hdr
*pBase
= &eep
->baseEepHeader
;
4149 int thermometer
= (pBase
->miscConfiguration
>> 1) & 0x3;
4151 return --thermometer
;
4154 static void ar9003_hw_thermometer_apply(struct ath_hw
*ah
)
4156 struct ath9k_hw_capabilities
*pCap
= &ah
->caps
;
4157 int thermometer
= ar9003_hw_get_thermometer(ah
);
4158 u8 therm_on
= (thermometer
< 0) ? 0 : 1;
4160 REG_RMW_FIELD(ah
, AR_PHY_65NM_CH0_RXTX4
,
4161 AR_PHY_65NM_CH0_RXTX4_THERM_ON_OVR
, therm_on
);
4162 if (pCap
->chip_chainmask
& BIT(1))
4163 REG_RMW_FIELD(ah
, AR_PHY_65NM_CH1_RXTX4
,
4164 AR_PHY_65NM_CH0_RXTX4_THERM_ON_OVR
, therm_on
);
4165 if (pCap
->chip_chainmask
& BIT(2))
4166 REG_RMW_FIELD(ah
, AR_PHY_65NM_CH2_RXTX4
,
4167 AR_PHY_65NM_CH0_RXTX4_THERM_ON_OVR
, therm_on
);
4169 therm_on
= thermometer
== 0;
4170 REG_RMW_FIELD(ah
, AR_PHY_65NM_CH0_RXTX4
,
4171 AR_PHY_65NM_CH0_RXTX4_THERM_ON
, therm_on
);
4172 if (pCap
->chip_chainmask
& BIT(1)) {
4173 therm_on
= thermometer
== 1;
4174 REG_RMW_FIELD(ah
, AR_PHY_65NM_CH1_RXTX4
,
4175 AR_PHY_65NM_CH0_RXTX4_THERM_ON
, therm_on
);
4177 if (pCap
->chip_chainmask
& BIT(2)) {
4178 therm_on
= thermometer
== 2;
4179 REG_RMW_FIELD(ah
, AR_PHY_65NM_CH2_RXTX4
,
4180 AR_PHY_65NM_CH0_RXTX4_THERM_ON
, therm_on
);
4184 static void ar9003_hw_thermo_cal_apply(struct ath_hw
*ah
)
4188 if (!AR_SREV_9462_20_OR_LATER(ah
))
4191 ar9300_otp_read_word(ah
, 1, &data
);
4193 kg
= (data
>> 8) & 0xff;
4195 REG_RMW_FIELD(ah
, AR_PHY_BB_THERM_ADC_3
,
4196 AR_PHY_BB_THERM_ADC_3_THERM_ADC_OFFSET
, ko
);
4197 REG_RMW_FIELD(ah
, AR_PHY_BB_THERM_ADC_3
,
4198 AR_PHY_BB_THERM_ADC_3_THERM_ADC_SCALE_GAIN
,
4203 static void ar9003_hw_apply_minccapwr_thresh(struct ath_hw
*ah
,
4206 struct ar9300_eeprom
*eep
= &ah
->eeprom
.ar9300_eep
;
4207 const u_int32_t cca_ctrl
[AR9300_MAX_CHAINS
] = {
4216 if (!(eep
->base_ext1
.misc_enable
& BIT(2)))
4219 if (!(eep
->base_ext1
.misc_enable
& BIT(3)))
4223 for (chain
= 0; chain
< AR9300_MAX_CHAINS
; chain
++) {
4224 if (!(ah
->caps
.tx_chainmask
& BIT(chain
)))
4227 val
= ar9003_modal_header(ah
, is2ghz
)->noiseFloorThreshCh
[chain
];
4228 REG_RMW_FIELD(ah
, cca_ctrl
[chain
],
4229 AR_PHY_EXT_CCA0_THRESH62_1
, val
);
4234 static void ath9k_hw_ar9300_set_board_values(struct ath_hw
*ah
,
4235 struct ath9k_channel
*chan
)
4237 bool is2ghz
= IS_CHAN_2GHZ(chan
);
4238 ar9003_hw_xpa_timing_control_apply(ah
, is2ghz
);
4239 ar9003_hw_xpa_bias_level_apply(ah
, is2ghz
);
4240 ar9003_hw_ant_ctrl_apply(ah
, is2ghz
);
4241 ar9003_hw_drive_strength_apply(ah
);
4242 ar9003_hw_xlna_bias_strength_apply(ah
, is2ghz
);
4243 ar9003_hw_atten_apply(ah
, chan
);
4244 ar9003_hw_quick_drop_apply(ah
, chan
->channel
);
4245 if (!AR_SREV_9330(ah
) && !AR_SREV_9340(ah
) && !AR_SREV_9531(ah
))
4246 ar9003_hw_internal_regulator_apply(ah
);
4247 ar9003_hw_apply_tuning_caps(ah
);
4248 ar9003_hw_apply_minccapwr_thresh(ah
, is2ghz
);
4249 ar9003_hw_txend_to_xpa_off_apply(ah
, is2ghz
);
4250 ar9003_hw_thermometer_apply(ah
);
4251 ar9003_hw_thermo_cal_apply(ah
);
4254 static void ath9k_hw_ar9300_set_addac(struct ath_hw
*ah
,
4255 struct ath9k_channel
*chan
)
4260 * Returns the interpolated y value corresponding to the specified x value
4261 * from the np ordered pairs of data (px,py).
4262 * The pairs do not have to be in any order.
4263 * If the specified x value is less than any of the px,
4264 * the returned y value is equal to the py for the lowest px.
4265 * If the specified x value is greater than any of the px,
4266 * the returned y value is equal to the py for the highest px.
4268 static int ar9003_hw_power_interpolate(int32_t x
,
4269 int32_t *px
, int32_t *py
, u_int16_t np
)
4272 int lx
= 0, ly
= 0, lhave
= 0;
4273 int hx
= 0, hy
= 0, hhave
= 0;
4280 /* identify best lower and higher x calibration measurement */
4281 for (ip
= 0; ip
< np
; ip
++) {
4284 /* this measurement is higher than our desired x */
4286 if (!hhave
|| dx
> (x
- hx
)) {
4287 /* new best higher x measurement */
4293 /* this measurement is lower than our desired x */
4295 if (!lhave
|| dx
< (x
- lx
)) {
4296 /* new best lower x measurement */
4304 /* the low x is good */
4306 /* so is the high x */
4308 /* they're the same, so just pick one */
4311 else /* interpolate */
4312 y
= interpolate(x
, lx
, hx
, ly
, hy
);
4313 } else /* only low is good, use it */
4315 } else if (hhave
) /* only high is good, use it */
4317 else /* nothing is good,this should never happen unless np=0, ???? */
4322 static u8
ar9003_hw_eeprom_get_tgt_pwr(struct ath_hw
*ah
,
4323 u16 rateIndex
, u16 freq
, bool is2GHz
)
4326 s32 targetPowerArray
[AR9300_NUM_5G_20_TARGET_POWERS
];
4327 s32 freqArray
[AR9300_NUM_5G_20_TARGET_POWERS
];
4328 struct ar9300_eeprom
*eep
= &ah
->eeprom
.ar9300_eep
;
4329 struct cal_tgt_pow_legacy
*pEepromTargetPwr
;
4333 numPiers
= AR9300_NUM_2G_20_TARGET_POWERS
;
4334 pEepromTargetPwr
= eep
->calTargetPower2G
;
4335 pFreqBin
= eep
->calTarget_freqbin_2G
;
4337 numPiers
= AR9300_NUM_5G_20_TARGET_POWERS
;
4338 pEepromTargetPwr
= eep
->calTargetPower5G
;
4339 pFreqBin
= eep
->calTarget_freqbin_5G
;
4343 * create array of channels and targetpower from
4344 * targetpower piers stored on eeprom
4346 for (i
= 0; i
< numPiers
; i
++) {
4347 freqArray
[i
] = ath9k_hw_fbin2freq(pFreqBin
[i
], is2GHz
);
4348 targetPowerArray
[i
] = pEepromTargetPwr
[i
].tPow2x
[rateIndex
];
4351 /* interpolate to get target power for given frequency */
4352 return (u8
) ar9003_hw_power_interpolate((s32
) freq
,
4354 targetPowerArray
, numPiers
);
4357 static u8
ar9003_hw_eeprom_get_ht20_tgt_pwr(struct ath_hw
*ah
,
4359 u16 freq
, bool is2GHz
)
4362 s32 targetPowerArray
[AR9300_NUM_5G_20_TARGET_POWERS
];
4363 s32 freqArray
[AR9300_NUM_5G_20_TARGET_POWERS
];
4364 struct ar9300_eeprom
*eep
= &ah
->eeprom
.ar9300_eep
;
4365 struct cal_tgt_pow_ht
*pEepromTargetPwr
;
4369 numPiers
= AR9300_NUM_2G_20_TARGET_POWERS
;
4370 pEepromTargetPwr
= eep
->calTargetPower2GHT20
;
4371 pFreqBin
= eep
->calTarget_freqbin_2GHT20
;
4373 numPiers
= AR9300_NUM_5G_20_TARGET_POWERS
;
4374 pEepromTargetPwr
= eep
->calTargetPower5GHT20
;
4375 pFreqBin
= eep
->calTarget_freqbin_5GHT20
;
4379 * create array of channels and targetpower
4380 * from targetpower piers stored on eeprom
4382 for (i
= 0; i
< numPiers
; i
++) {
4383 freqArray
[i
] = ath9k_hw_fbin2freq(pFreqBin
[i
], is2GHz
);
4384 targetPowerArray
[i
] = pEepromTargetPwr
[i
].tPow2x
[rateIndex
];
4387 /* interpolate to get target power for given frequency */
4388 return (u8
) ar9003_hw_power_interpolate((s32
) freq
,
4390 targetPowerArray
, numPiers
);
4393 static u8
ar9003_hw_eeprom_get_ht40_tgt_pwr(struct ath_hw
*ah
,
4395 u16 freq
, bool is2GHz
)
4398 s32 targetPowerArray
[AR9300_NUM_5G_40_TARGET_POWERS
];
4399 s32 freqArray
[AR9300_NUM_5G_40_TARGET_POWERS
];
4400 struct ar9300_eeprom
*eep
= &ah
->eeprom
.ar9300_eep
;
4401 struct cal_tgt_pow_ht
*pEepromTargetPwr
;
4405 numPiers
= AR9300_NUM_2G_40_TARGET_POWERS
;
4406 pEepromTargetPwr
= eep
->calTargetPower2GHT40
;
4407 pFreqBin
= eep
->calTarget_freqbin_2GHT40
;
4409 numPiers
= AR9300_NUM_5G_40_TARGET_POWERS
;
4410 pEepromTargetPwr
= eep
->calTargetPower5GHT40
;
4411 pFreqBin
= eep
->calTarget_freqbin_5GHT40
;
4415 * create array of channels and targetpower from
4416 * targetpower piers stored on eeprom
4418 for (i
= 0; i
< numPiers
; i
++) {
4419 freqArray
[i
] = ath9k_hw_fbin2freq(pFreqBin
[i
], is2GHz
);
4420 targetPowerArray
[i
] = pEepromTargetPwr
[i
].tPow2x
[rateIndex
];
4423 /* interpolate to get target power for given frequency */
4424 return (u8
) ar9003_hw_power_interpolate((s32
) freq
,
4426 targetPowerArray
, numPiers
);
4429 static u8
ar9003_hw_eeprom_get_cck_tgt_pwr(struct ath_hw
*ah
,
4430 u16 rateIndex
, u16 freq
)
4432 u16 numPiers
= AR9300_NUM_2G_CCK_TARGET_POWERS
, i
;
4433 s32 targetPowerArray
[AR9300_NUM_2G_CCK_TARGET_POWERS
];
4434 s32 freqArray
[AR9300_NUM_2G_CCK_TARGET_POWERS
];
4435 struct ar9300_eeprom
*eep
= &ah
->eeprom
.ar9300_eep
;
4436 struct cal_tgt_pow_legacy
*pEepromTargetPwr
= eep
->calTargetPowerCck
;
4437 u8
*pFreqBin
= eep
->calTarget_freqbin_Cck
;
4440 * create array of channels and targetpower from
4441 * targetpower piers stored on eeprom
4443 for (i
= 0; i
< numPiers
; i
++) {
4444 freqArray
[i
] = ath9k_hw_fbin2freq(pFreqBin
[i
], 1);
4445 targetPowerArray
[i
] = pEepromTargetPwr
[i
].tPow2x
[rateIndex
];
4448 /* interpolate to get target power for given frequency */
4449 return (u8
) ar9003_hw_power_interpolate((s32
) freq
,
4451 targetPowerArray
, numPiers
);
4454 static void ar9003_hw_selfgen_tpc_txpower(struct ath_hw
*ah
,
4455 struct ath9k_channel
*chan
,
4460 /* target power values for self generated frames (ACK,RTS/CTS) */
4461 if (IS_CHAN_2GHZ(chan
)) {
4462 val
= SM(pwr_array
[ALL_TARGET_LEGACY_1L_5L
], AR_TPC_ACK
) |
4463 SM(pwr_array
[ALL_TARGET_LEGACY_1L_5L
], AR_TPC_CTS
) |
4464 SM(0x3f, AR_TPC_CHIRP
) | SM(0x3f, AR_TPC_RPT
);
4466 val
= SM(pwr_array
[ALL_TARGET_LEGACY_6_24
], AR_TPC_ACK
) |
4467 SM(pwr_array
[ALL_TARGET_LEGACY_6_24
], AR_TPC_CTS
) |
4468 SM(0x3f, AR_TPC_CHIRP
) | SM(0x3f, AR_TPC_RPT
);
4470 REG_WRITE(ah
, AR_TPC
, val
);
4473 /* Set tx power registers to array of values passed in */
4474 int ar9003_hw_tx_power_regwrite(struct ath_hw
*ah
, u8
* pPwrArray
)
4476 #define POW_SM(_r, _s) (((_r) & 0x3f) << (_s))
4477 /* make sure forced gain is not set */
4478 REG_WRITE(ah
, AR_PHY_TX_FORCED_GAIN
, 0);
4480 /* Write the OFDM power per rate set */
4482 /* 6 (LSB), 9, 12, 18 (MSB) */
4483 REG_WRITE(ah
, AR_PHY_POWER_TX_RATE(0),
4484 POW_SM(pPwrArray
[ALL_TARGET_LEGACY_6_24
], 24) |
4485 POW_SM(pPwrArray
[ALL_TARGET_LEGACY_6_24
], 16) |
4486 POW_SM(pPwrArray
[ALL_TARGET_LEGACY_6_24
], 8) |
4487 POW_SM(pPwrArray
[ALL_TARGET_LEGACY_6_24
], 0));
4489 /* 24 (LSB), 36, 48, 54 (MSB) */
4490 REG_WRITE(ah
, AR_PHY_POWER_TX_RATE(1),
4491 POW_SM(pPwrArray
[ALL_TARGET_LEGACY_54
], 24) |
4492 POW_SM(pPwrArray
[ALL_TARGET_LEGACY_48
], 16) |
4493 POW_SM(pPwrArray
[ALL_TARGET_LEGACY_36
], 8) |
4494 POW_SM(pPwrArray
[ALL_TARGET_LEGACY_6_24
], 0));
4496 /* Write the CCK power per rate set */
4498 /* 1L (LSB), reserved, 2L, 2S (MSB) */
4499 REG_WRITE(ah
, AR_PHY_POWER_TX_RATE(2),
4500 POW_SM(pPwrArray
[ALL_TARGET_LEGACY_1L_5L
], 24) |
4501 POW_SM(pPwrArray
[ALL_TARGET_LEGACY_1L_5L
], 16) |
4502 /* POW_SM(txPowerTimes2, 8) | this is reserved for AR9003 */
4503 POW_SM(pPwrArray
[ALL_TARGET_LEGACY_1L_5L
], 0));
4505 /* 5.5L (LSB), 5.5S, 11L, 11S (MSB) */
4506 REG_WRITE(ah
, AR_PHY_POWER_TX_RATE(3),
4507 POW_SM(pPwrArray
[ALL_TARGET_LEGACY_11S
], 24) |
4508 POW_SM(pPwrArray
[ALL_TARGET_LEGACY_11L
], 16) |
4509 POW_SM(pPwrArray
[ALL_TARGET_LEGACY_5S
], 8) |
4510 POW_SM(pPwrArray
[ALL_TARGET_LEGACY_1L_5L
], 0)
4513 /* Write the power for duplicated frames - HT40 */
4515 /* dup40_cck (LSB), dup40_ofdm, ext20_cck, ext20_ofdm (MSB) */
4516 REG_WRITE(ah
, AR_PHY_POWER_TX_RATE(8),
4517 POW_SM(pPwrArray
[ALL_TARGET_LEGACY_6_24
], 24) |
4518 POW_SM(pPwrArray
[ALL_TARGET_LEGACY_1L_5L
], 16) |
4519 POW_SM(pPwrArray
[ALL_TARGET_LEGACY_6_24
], 8) |
4520 POW_SM(pPwrArray
[ALL_TARGET_LEGACY_1L_5L
], 0)
4523 /* Write the HT20 power per rate set */
4525 /* 0/8/16 (LSB), 1-3/9-11/17-19, 4, 5 (MSB) */
4526 REG_WRITE(ah
, AR_PHY_POWER_TX_RATE(4),
4527 POW_SM(pPwrArray
[ALL_TARGET_HT20_5
], 24) |
4528 POW_SM(pPwrArray
[ALL_TARGET_HT20_4
], 16) |
4529 POW_SM(pPwrArray
[ALL_TARGET_HT20_1_3_9_11_17_19
], 8) |
4530 POW_SM(pPwrArray
[ALL_TARGET_HT20_0_8_16
], 0)
4533 /* 6 (LSB), 7, 12, 13 (MSB) */
4534 REG_WRITE(ah
, AR_PHY_POWER_TX_RATE(5),
4535 POW_SM(pPwrArray
[ALL_TARGET_HT20_13
], 24) |
4536 POW_SM(pPwrArray
[ALL_TARGET_HT20_12
], 16) |
4537 POW_SM(pPwrArray
[ALL_TARGET_HT20_7
], 8) |
4538 POW_SM(pPwrArray
[ALL_TARGET_HT20_6
], 0)
4541 /* 14 (LSB), 15, 20, 21 */
4542 REG_WRITE(ah
, AR_PHY_POWER_TX_RATE(9),
4543 POW_SM(pPwrArray
[ALL_TARGET_HT20_21
], 24) |
4544 POW_SM(pPwrArray
[ALL_TARGET_HT20_20
], 16) |
4545 POW_SM(pPwrArray
[ALL_TARGET_HT20_15
], 8) |
4546 POW_SM(pPwrArray
[ALL_TARGET_HT20_14
], 0)
4549 /* Mixed HT20 and HT40 rates */
4551 /* HT20 22 (LSB), HT20 23, HT40 22, HT40 23 (MSB) */
4552 REG_WRITE(ah
, AR_PHY_POWER_TX_RATE(10),
4553 POW_SM(pPwrArray
[ALL_TARGET_HT40_23
], 24) |
4554 POW_SM(pPwrArray
[ALL_TARGET_HT40_22
], 16) |
4555 POW_SM(pPwrArray
[ALL_TARGET_HT20_23
], 8) |
4556 POW_SM(pPwrArray
[ALL_TARGET_HT20_22
], 0)
4560 * Write the HT40 power per rate set
4561 * correct PAR difference between HT40 and HT20/LEGACY
4562 * 0/8/16 (LSB), 1-3/9-11/17-19, 4, 5 (MSB)
4564 REG_WRITE(ah
, AR_PHY_POWER_TX_RATE(6),
4565 POW_SM(pPwrArray
[ALL_TARGET_HT40_5
], 24) |
4566 POW_SM(pPwrArray
[ALL_TARGET_HT40_4
], 16) |
4567 POW_SM(pPwrArray
[ALL_TARGET_HT40_1_3_9_11_17_19
], 8) |
4568 POW_SM(pPwrArray
[ALL_TARGET_HT40_0_8_16
], 0)
4571 /* 6 (LSB), 7, 12, 13 (MSB) */
4572 REG_WRITE(ah
, AR_PHY_POWER_TX_RATE(7),
4573 POW_SM(pPwrArray
[ALL_TARGET_HT40_13
], 24) |
4574 POW_SM(pPwrArray
[ALL_TARGET_HT40_12
], 16) |
4575 POW_SM(pPwrArray
[ALL_TARGET_HT40_7
], 8) |
4576 POW_SM(pPwrArray
[ALL_TARGET_HT40_6
], 0)
4579 /* 14 (LSB), 15, 20, 21 */
4580 REG_WRITE(ah
, AR_PHY_POWER_TX_RATE(11),
4581 POW_SM(pPwrArray
[ALL_TARGET_HT40_21
], 24) |
4582 POW_SM(pPwrArray
[ALL_TARGET_HT40_20
], 16) |
4583 POW_SM(pPwrArray
[ALL_TARGET_HT40_15
], 8) |
4584 POW_SM(pPwrArray
[ALL_TARGET_HT40_14
], 0)
4591 static void ar9003_hw_get_legacy_target_powers(struct ath_hw
*ah
, u16 freq
,
4592 u8
*targetPowerValT2
,
4595 targetPowerValT2
[ALL_TARGET_LEGACY_6_24
] =
4596 ar9003_hw_eeprom_get_tgt_pwr(ah
, LEGACY_TARGET_RATE_6_24
, freq
,
4598 targetPowerValT2
[ALL_TARGET_LEGACY_36
] =
4599 ar9003_hw_eeprom_get_tgt_pwr(ah
, LEGACY_TARGET_RATE_36
, freq
,
4601 targetPowerValT2
[ALL_TARGET_LEGACY_48
] =
4602 ar9003_hw_eeprom_get_tgt_pwr(ah
, LEGACY_TARGET_RATE_48
, freq
,
4604 targetPowerValT2
[ALL_TARGET_LEGACY_54
] =
4605 ar9003_hw_eeprom_get_tgt_pwr(ah
, LEGACY_TARGET_RATE_54
, freq
,
4609 static void ar9003_hw_get_cck_target_powers(struct ath_hw
*ah
, u16 freq
,
4610 u8
*targetPowerValT2
)
4612 targetPowerValT2
[ALL_TARGET_LEGACY_1L_5L
] =
4613 ar9003_hw_eeprom_get_cck_tgt_pwr(ah
, LEGACY_TARGET_RATE_1L_5L
,
4615 targetPowerValT2
[ALL_TARGET_LEGACY_5S
] =
4616 ar9003_hw_eeprom_get_cck_tgt_pwr(ah
, LEGACY_TARGET_RATE_5S
, freq
);
4617 targetPowerValT2
[ALL_TARGET_LEGACY_11L
] =
4618 ar9003_hw_eeprom_get_cck_tgt_pwr(ah
, LEGACY_TARGET_RATE_11L
, freq
);
4619 targetPowerValT2
[ALL_TARGET_LEGACY_11S
] =
4620 ar9003_hw_eeprom_get_cck_tgt_pwr(ah
, LEGACY_TARGET_RATE_11S
, freq
);
4623 static void ar9003_hw_get_ht20_target_powers(struct ath_hw
*ah
, u16 freq
,
4624 u8
*targetPowerValT2
, bool is2GHz
)
4626 targetPowerValT2
[ALL_TARGET_HT20_0_8_16
] =
4627 ar9003_hw_eeprom_get_ht20_tgt_pwr(ah
, HT_TARGET_RATE_0_8_16
, freq
,
4629 targetPowerValT2
[ALL_TARGET_HT20_1_3_9_11_17_19
] =
4630 ar9003_hw_eeprom_get_ht20_tgt_pwr(ah
, HT_TARGET_RATE_1_3_9_11_17_19
,
4632 targetPowerValT2
[ALL_TARGET_HT20_4
] =
4633 ar9003_hw_eeprom_get_ht20_tgt_pwr(ah
, HT_TARGET_RATE_4
, freq
,
4635 targetPowerValT2
[ALL_TARGET_HT20_5
] =
4636 ar9003_hw_eeprom_get_ht20_tgt_pwr(ah
, HT_TARGET_RATE_5
, freq
,
4638 targetPowerValT2
[ALL_TARGET_HT20_6
] =
4639 ar9003_hw_eeprom_get_ht20_tgt_pwr(ah
, HT_TARGET_RATE_6
, freq
,
4641 targetPowerValT2
[ALL_TARGET_HT20_7
] =
4642 ar9003_hw_eeprom_get_ht20_tgt_pwr(ah
, HT_TARGET_RATE_7
, freq
,
4644 targetPowerValT2
[ALL_TARGET_HT20_12
] =
4645 ar9003_hw_eeprom_get_ht20_tgt_pwr(ah
, HT_TARGET_RATE_12
, freq
,
4647 targetPowerValT2
[ALL_TARGET_HT20_13
] =
4648 ar9003_hw_eeprom_get_ht20_tgt_pwr(ah
, HT_TARGET_RATE_13
, freq
,
4650 targetPowerValT2
[ALL_TARGET_HT20_14
] =
4651 ar9003_hw_eeprom_get_ht20_tgt_pwr(ah
, HT_TARGET_RATE_14
, freq
,
4653 targetPowerValT2
[ALL_TARGET_HT20_15
] =
4654 ar9003_hw_eeprom_get_ht20_tgt_pwr(ah
, HT_TARGET_RATE_15
, freq
,
4656 targetPowerValT2
[ALL_TARGET_HT20_20
] =
4657 ar9003_hw_eeprom_get_ht20_tgt_pwr(ah
, HT_TARGET_RATE_20
, freq
,
4659 targetPowerValT2
[ALL_TARGET_HT20_21
] =
4660 ar9003_hw_eeprom_get_ht20_tgt_pwr(ah
, HT_TARGET_RATE_21
, freq
,
4662 targetPowerValT2
[ALL_TARGET_HT20_22
] =
4663 ar9003_hw_eeprom_get_ht20_tgt_pwr(ah
, HT_TARGET_RATE_22
, freq
,
4665 targetPowerValT2
[ALL_TARGET_HT20_23
] =
4666 ar9003_hw_eeprom_get_ht20_tgt_pwr(ah
, HT_TARGET_RATE_23
, freq
,
4670 static void ar9003_hw_get_ht40_target_powers(struct ath_hw
*ah
,
4672 u8
*targetPowerValT2
,
4675 /* XXX: hard code for now, need to get from eeprom struct */
4676 u8 ht40PowerIncForPdadc
= 0;
4678 targetPowerValT2
[ALL_TARGET_HT40_0_8_16
] =
4679 ar9003_hw_eeprom_get_ht40_tgt_pwr(ah
, HT_TARGET_RATE_0_8_16
, freq
,
4680 is2GHz
) + ht40PowerIncForPdadc
;
4681 targetPowerValT2
[ALL_TARGET_HT40_1_3_9_11_17_19
] =
4682 ar9003_hw_eeprom_get_ht40_tgt_pwr(ah
, HT_TARGET_RATE_1_3_9_11_17_19
,
4684 is2GHz
) + ht40PowerIncForPdadc
;
4685 targetPowerValT2
[ALL_TARGET_HT40_4
] =
4686 ar9003_hw_eeprom_get_ht40_tgt_pwr(ah
, HT_TARGET_RATE_4
, freq
,
4687 is2GHz
) + ht40PowerIncForPdadc
;
4688 targetPowerValT2
[ALL_TARGET_HT40_5
] =
4689 ar9003_hw_eeprom_get_ht40_tgt_pwr(ah
, HT_TARGET_RATE_5
, freq
,
4690 is2GHz
) + ht40PowerIncForPdadc
;
4691 targetPowerValT2
[ALL_TARGET_HT40_6
] =
4692 ar9003_hw_eeprom_get_ht40_tgt_pwr(ah
, HT_TARGET_RATE_6
, freq
,
4693 is2GHz
) + ht40PowerIncForPdadc
;
4694 targetPowerValT2
[ALL_TARGET_HT40_7
] =
4695 ar9003_hw_eeprom_get_ht40_tgt_pwr(ah
, HT_TARGET_RATE_7
, freq
,
4696 is2GHz
) + ht40PowerIncForPdadc
;
4697 targetPowerValT2
[ALL_TARGET_HT40_12
] =
4698 ar9003_hw_eeprom_get_ht40_tgt_pwr(ah
, HT_TARGET_RATE_12
, freq
,
4699 is2GHz
) + ht40PowerIncForPdadc
;
4700 targetPowerValT2
[ALL_TARGET_HT40_13
] =
4701 ar9003_hw_eeprom_get_ht40_tgt_pwr(ah
, HT_TARGET_RATE_13
, freq
,
4702 is2GHz
) + ht40PowerIncForPdadc
;
4703 targetPowerValT2
[ALL_TARGET_HT40_14
] =
4704 ar9003_hw_eeprom_get_ht40_tgt_pwr(ah
, HT_TARGET_RATE_14
, freq
,
4705 is2GHz
) + ht40PowerIncForPdadc
;
4706 targetPowerValT2
[ALL_TARGET_HT40_15
] =
4707 ar9003_hw_eeprom_get_ht40_tgt_pwr(ah
, HT_TARGET_RATE_15
, freq
,
4708 is2GHz
) + ht40PowerIncForPdadc
;
4709 targetPowerValT2
[ALL_TARGET_HT40_20
] =
4710 ar9003_hw_eeprom_get_ht40_tgt_pwr(ah
, HT_TARGET_RATE_20
, freq
,
4711 is2GHz
) + ht40PowerIncForPdadc
;
4712 targetPowerValT2
[ALL_TARGET_HT40_21
] =
4713 ar9003_hw_eeprom_get_ht40_tgt_pwr(ah
, HT_TARGET_RATE_21
, freq
,
4714 is2GHz
) + ht40PowerIncForPdadc
;
4715 targetPowerValT2
[ALL_TARGET_HT40_22
] =
4716 ar9003_hw_eeprom_get_ht40_tgt_pwr(ah
, HT_TARGET_RATE_22
, freq
,
4717 is2GHz
) + ht40PowerIncForPdadc
;
4718 targetPowerValT2
[ALL_TARGET_HT40_23
] =
4719 ar9003_hw_eeprom_get_ht40_tgt_pwr(ah
, HT_TARGET_RATE_23
, freq
,
4720 is2GHz
) + ht40PowerIncForPdadc
;
4723 static void ar9003_hw_get_target_power_eeprom(struct ath_hw
*ah
,
4724 struct ath9k_channel
*chan
,
4725 u8
*targetPowerValT2
)
4727 bool is2GHz
= IS_CHAN_2GHZ(chan
);
4729 struct ath_common
*common
= ath9k_hw_common(ah
);
4730 u16 freq
= chan
->channel
;
4733 ar9003_hw_get_cck_target_powers(ah
, freq
, targetPowerValT2
);
4735 ar9003_hw_get_legacy_target_powers(ah
, freq
, targetPowerValT2
, is2GHz
);
4736 ar9003_hw_get_ht20_target_powers(ah
, freq
, targetPowerValT2
, is2GHz
);
4738 if (IS_CHAN_HT40(chan
))
4739 ar9003_hw_get_ht40_target_powers(ah
, freq
, targetPowerValT2
,
4742 for (i
= 0; i
< ar9300RateSize
; i
++) {
4743 ath_dbg(common
, REGULATORY
, "TPC[%02d] 0x%08x\n",
4744 i
, targetPowerValT2
[i
]);
4748 static int ar9003_hw_cal_pier_get(struct ath_hw
*ah
,
4754 int *ptemperature
, int *pvoltage
,
4755 int *pnf_cal
, int *pnf_power
)
4758 struct ar9300_cal_data_per_freq_op_loop
*pCalPierStruct
;
4760 struct ar9300_eeprom
*eep
= &ah
->eeprom
.ar9300_eep
;
4761 struct ath_common
*common
= ath9k_hw_common(ah
);
4763 if (ichain
>= AR9300_MAX_CHAINS
) {
4764 ath_dbg(common
, EEPROM
,
4765 "Invalid chain index, must be less than %d\n",
4770 if (mode
) { /* 5GHz */
4771 if (ipier
>= AR9300_NUM_5G_CAL_PIERS
) {
4772 ath_dbg(common
, EEPROM
,
4773 "Invalid 5GHz cal pier index, must be less than %d\n",
4774 AR9300_NUM_5G_CAL_PIERS
);
4777 pCalPier
= &(eep
->calFreqPier5G
[ipier
]);
4778 pCalPierStruct
= &(eep
->calPierData5G
[ichain
][ipier
]);
4781 if (ipier
>= AR9300_NUM_2G_CAL_PIERS
) {
4782 ath_dbg(common
, EEPROM
,
4783 "Invalid 2GHz cal pier index, must be less than %d\n",
4784 AR9300_NUM_2G_CAL_PIERS
);
4788 pCalPier
= &(eep
->calFreqPier2G
[ipier
]);
4789 pCalPierStruct
= &(eep
->calPierData2G
[ichain
][ipier
]);
4793 *pfrequency
= ath9k_hw_fbin2freq(*pCalPier
, is2GHz
);
4794 *pcorrection
= pCalPierStruct
->refPower
;
4795 *ptemperature
= pCalPierStruct
->tempMeas
;
4796 *pvoltage
= pCalPierStruct
->voltMeas
;
4797 *pnf_cal
= pCalPierStruct
->rxTempMeas
?
4798 N2DBM(pCalPierStruct
->rxNoisefloorCal
) : 0;
4799 *pnf_power
= pCalPierStruct
->rxTempMeas
?
4800 N2DBM(pCalPierStruct
->rxNoisefloorPower
) : 0;
4805 static void ar9003_hw_power_control_override(struct ath_hw
*ah
,
4808 int *voltage
, int *temperature
)
4810 int temp_slope
= 0, temp_slope1
= 0, temp_slope2
= 0;
4811 struct ar9300_eeprom
*eep
= &ah
->eeprom
.ar9300_eep
;
4812 int f
[8], t
[8], t1
[3], t2
[3], i
;
4814 REG_RMW(ah
, AR_PHY_TPC_11_B0
,
4815 (correction
[0] << AR_PHY_TPC_OLPC_GAIN_DELTA_S
),
4816 AR_PHY_TPC_OLPC_GAIN_DELTA
);
4817 if (ah
->caps
.tx_chainmask
& BIT(1))
4818 REG_RMW(ah
, AR_PHY_TPC_11_B1
,
4819 (correction
[1] << AR_PHY_TPC_OLPC_GAIN_DELTA_S
),
4820 AR_PHY_TPC_OLPC_GAIN_DELTA
);
4821 if (ah
->caps
.tx_chainmask
& BIT(2))
4822 REG_RMW(ah
, AR_PHY_TPC_11_B2
,
4823 (correction
[2] << AR_PHY_TPC_OLPC_GAIN_DELTA_S
),
4824 AR_PHY_TPC_OLPC_GAIN_DELTA
);
4826 /* enable open loop power control on chip */
4827 REG_RMW(ah
, AR_PHY_TPC_6_B0
,
4828 (3 << AR_PHY_TPC_6_ERROR_EST_MODE_S
),
4829 AR_PHY_TPC_6_ERROR_EST_MODE
);
4830 if (ah
->caps
.tx_chainmask
& BIT(1))
4831 REG_RMW(ah
, AR_PHY_TPC_6_B1
,
4832 (3 << AR_PHY_TPC_6_ERROR_EST_MODE_S
),
4833 AR_PHY_TPC_6_ERROR_EST_MODE
);
4834 if (ah
->caps
.tx_chainmask
& BIT(2))
4835 REG_RMW(ah
, AR_PHY_TPC_6_B2
,
4836 (3 << AR_PHY_TPC_6_ERROR_EST_MODE_S
),
4837 AR_PHY_TPC_6_ERROR_EST_MODE
);
4840 * enable temperature compensation
4841 * Need to use register names
4843 if (frequency
< 4000) {
4844 temp_slope
= eep
->modalHeader2G
.tempSlope
;
4846 if (AR_SREV_9550(ah
)) {
4847 t
[0] = eep
->base_ext1
.tempslopextension
[2];
4848 t1
[0] = eep
->base_ext1
.tempslopextension
[3];
4849 t2
[0] = eep
->base_ext1
.tempslopextension
[4];
4852 t
[1] = eep
->modalHeader5G
.tempSlope
;
4853 t1
[1] = eep
->base_ext1
.tempslopextension
[0];
4854 t2
[1] = eep
->base_ext1
.tempslopextension
[1];
4857 t
[2] = eep
->base_ext1
.tempslopextension
[5];
4858 t1
[2] = eep
->base_ext1
.tempslopextension
[6];
4859 t2
[2] = eep
->base_ext1
.tempslopextension
[7];
4862 temp_slope
= ar9003_hw_power_interpolate(frequency
,
4864 temp_slope1
= ar9003_hw_power_interpolate(frequency
,
4866 temp_slope2
= ar9003_hw_power_interpolate(frequency
,
4872 if ((eep
->baseEepHeader
.miscConfiguration
& 0x20) != 0) {
4873 for (i
= 0; i
< 8; i
++) {
4874 t
[i
] = eep
->base_ext1
.tempslopextension
[i
];
4875 f
[i
] = FBIN2FREQ(eep
->calFreqPier5G
[i
], 0);
4877 temp_slope
= ar9003_hw_power_interpolate((s32
) frequency
,
4879 } else if (eep
->base_ext2
.tempSlopeLow
!= 0) {
4880 t
[0] = eep
->base_ext2
.tempSlopeLow
;
4882 t
[1] = eep
->modalHeader5G
.tempSlope
;
4884 t
[2] = eep
->base_ext2
.tempSlopeHigh
;
4886 temp_slope
= ar9003_hw_power_interpolate((s32
) frequency
,
4889 temp_slope
= eep
->modalHeader5G
.tempSlope
;
4894 if (AR_SREV_9550(ah
) || AR_SREV_9531(ah
) || AR_SREV_9561(ah
)) {
4895 u8 txmask
= (eep
->baseEepHeader
.txrxMask
& 0xf0) >> 4;
4898 * AR955x has tempSlope register for each chain.
4899 * Check whether temp_compensation feature is enabled or not.
4901 if (eep
->baseEepHeader
.featureEnable
& 0x1) {
4902 if (frequency
< 4000) {
4903 if (txmask
& BIT(0))
4904 REG_RMW_FIELD(ah
, AR_PHY_TPC_19
,
4905 AR_PHY_TPC_19_ALPHA_THERM
,
4906 eep
->base_ext2
.tempSlopeLow
);
4907 if (txmask
& BIT(1))
4908 REG_RMW_FIELD(ah
, AR_PHY_TPC_19_B1
,
4909 AR_PHY_TPC_19_ALPHA_THERM
,
4911 if (txmask
& BIT(2))
4912 REG_RMW_FIELD(ah
, AR_PHY_TPC_19_B2
,
4913 AR_PHY_TPC_19_ALPHA_THERM
,
4914 eep
->base_ext2
.tempSlopeHigh
);
4916 if (txmask
& BIT(0))
4917 REG_RMW_FIELD(ah
, AR_PHY_TPC_19
,
4918 AR_PHY_TPC_19_ALPHA_THERM
,
4920 if (txmask
& BIT(1))
4921 REG_RMW_FIELD(ah
, AR_PHY_TPC_19_B1
,
4922 AR_PHY_TPC_19_ALPHA_THERM
,
4924 if (txmask
& BIT(2))
4925 REG_RMW_FIELD(ah
, AR_PHY_TPC_19_B2
,
4926 AR_PHY_TPC_19_ALPHA_THERM
,
4931 * If temp compensation is not enabled,
4932 * set all registers to 0.
4934 if (txmask
& BIT(0))
4935 REG_RMW_FIELD(ah
, AR_PHY_TPC_19
,
4936 AR_PHY_TPC_19_ALPHA_THERM
, 0);
4937 if (txmask
& BIT(1))
4938 REG_RMW_FIELD(ah
, AR_PHY_TPC_19_B1
,
4939 AR_PHY_TPC_19_ALPHA_THERM
, 0);
4940 if (txmask
& BIT(2))
4941 REG_RMW_FIELD(ah
, AR_PHY_TPC_19_B2
,
4942 AR_PHY_TPC_19_ALPHA_THERM
, 0);
4945 REG_RMW_FIELD(ah
, AR_PHY_TPC_19
,
4946 AR_PHY_TPC_19_ALPHA_THERM
, temp_slope
);
4949 if (AR_SREV_9462_20_OR_LATER(ah
))
4950 REG_RMW_FIELD(ah
, AR_PHY_TPC_19_B1
,
4951 AR_PHY_TPC_19_B1_ALPHA_THERM
, temp_slope
);
4954 REG_RMW_FIELD(ah
, AR_PHY_TPC_18
, AR_PHY_TPC_18_THERM_CAL_VALUE
,
4958 /* Apply the recorded correction values. */
4959 static int ar9003_hw_calibration_apply(struct ath_hw
*ah
, int frequency
)
4961 int ichain
, ipier
, npier
;
4963 int lfrequency
[AR9300_MAX_CHAINS
],
4964 lcorrection
[AR9300_MAX_CHAINS
],
4965 ltemperature
[AR9300_MAX_CHAINS
], lvoltage
[AR9300_MAX_CHAINS
],
4966 lnf_cal
[AR9300_MAX_CHAINS
], lnf_pwr
[AR9300_MAX_CHAINS
];
4967 int hfrequency
[AR9300_MAX_CHAINS
],
4968 hcorrection
[AR9300_MAX_CHAINS
],
4969 htemperature
[AR9300_MAX_CHAINS
], hvoltage
[AR9300_MAX_CHAINS
],
4970 hnf_cal
[AR9300_MAX_CHAINS
], hnf_pwr
[AR9300_MAX_CHAINS
];
4972 int correction
[AR9300_MAX_CHAINS
],
4973 voltage
[AR9300_MAX_CHAINS
], temperature
[AR9300_MAX_CHAINS
],
4974 nf_cal
[AR9300_MAX_CHAINS
], nf_pwr
[AR9300_MAX_CHAINS
];
4975 int pfrequency
, pcorrection
, ptemperature
, pvoltage
,
4977 struct ath_common
*common
= ath9k_hw_common(ah
);
4979 mode
= (frequency
>= 4000);
4981 npier
= AR9300_NUM_5G_CAL_PIERS
;
4983 npier
= AR9300_NUM_2G_CAL_PIERS
;
4985 for (ichain
= 0; ichain
< AR9300_MAX_CHAINS
; ichain
++) {
4986 lfrequency
[ichain
] = 0;
4987 hfrequency
[ichain
] = 100000;
4989 /* identify best lower and higher frequency calibration measurement */
4990 for (ichain
= 0; ichain
< AR9300_MAX_CHAINS
; ichain
++) {
4991 for (ipier
= 0; ipier
< npier
; ipier
++) {
4992 if (!ar9003_hw_cal_pier_get(ah
, mode
, ipier
, ichain
,
4993 &pfrequency
, &pcorrection
,
4994 &ptemperature
, &pvoltage
,
4995 &pnf_cal
, &pnf_pwr
)) {
4996 fdiff
= frequency
- pfrequency
;
4999 * this measurement is higher than
5000 * our desired frequency
5003 if (hfrequency
[ichain
] <= 0 ||
5004 hfrequency
[ichain
] >= 100000 ||
5006 (frequency
- hfrequency
[ichain
])) {
5009 * frequency measurement
5011 hfrequency
[ichain
] = pfrequency
;
5012 hcorrection
[ichain
] =
5014 htemperature
[ichain
] =
5016 hvoltage
[ichain
] = pvoltage
;
5017 hnf_cal
[ichain
] = pnf_cal
;
5018 hnf_pwr
[ichain
] = pnf_pwr
;
5022 if (lfrequency
[ichain
] <= 0
5024 (frequency
- lfrequency
[ichain
])) {
5027 * frequency measurement
5029 lfrequency
[ichain
] = pfrequency
;
5030 lcorrection
[ichain
] =
5032 ltemperature
[ichain
] =
5034 lvoltage
[ichain
] = pvoltage
;
5035 lnf_cal
[ichain
] = pnf_cal
;
5036 lnf_pwr
[ichain
] = pnf_pwr
;
5044 for (ichain
= 0; ichain
< AR9300_MAX_CHAINS
; ichain
++) {
5045 ath_dbg(common
, EEPROM
,
5046 "ch=%d f=%d low=%d %d h=%d %d n=%d %d p=%d %d\n",
5047 ichain
, frequency
, lfrequency
[ichain
],
5048 lcorrection
[ichain
], hfrequency
[ichain
],
5049 hcorrection
[ichain
], lnf_cal
[ichain
],
5050 hnf_cal
[ichain
], lnf_pwr
[ichain
],
5052 /* they're the same, so just pick one */
5053 if (hfrequency
[ichain
] == lfrequency
[ichain
]) {
5054 correction
[ichain
] = lcorrection
[ichain
];
5055 voltage
[ichain
] = lvoltage
[ichain
];
5056 temperature
[ichain
] = ltemperature
[ichain
];
5057 nf_cal
[ichain
] = lnf_cal
[ichain
];
5058 nf_pwr
[ichain
] = lnf_pwr
[ichain
];
5060 /* the low frequency is good */
5061 else if (frequency
- lfrequency
[ichain
] < 1000) {
5062 /* so is the high frequency, interpolate */
5063 if (hfrequency
[ichain
] - frequency
< 1000) {
5065 correction
[ichain
] = interpolate(frequency
,
5068 lcorrection
[ichain
],
5069 hcorrection
[ichain
]);
5071 temperature
[ichain
] = interpolate(frequency
,
5074 ltemperature
[ichain
],
5075 htemperature
[ichain
]);
5077 voltage
[ichain
] = interpolate(frequency
,
5083 nf_cal
[ichain
] = interpolate(frequency
,
5089 nf_pwr
[ichain
] = interpolate(frequency
,
5095 /* only low is good, use it */
5097 correction
[ichain
] = lcorrection
[ichain
];
5098 temperature
[ichain
] = ltemperature
[ichain
];
5099 voltage
[ichain
] = lvoltage
[ichain
];
5100 nf_cal
[ichain
] = lnf_cal
[ichain
];
5101 nf_pwr
[ichain
] = lnf_pwr
[ichain
];
5104 /* only high is good, use it */
5105 else if (hfrequency
[ichain
] - frequency
< 1000) {
5106 correction
[ichain
] = hcorrection
[ichain
];
5107 temperature
[ichain
] = htemperature
[ichain
];
5108 voltage
[ichain
] = hvoltage
[ichain
];
5109 nf_cal
[ichain
] = hnf_cal
[ichain
];
5110 nf_pwr
[ichain
] = hnf_pwr
[ichain
];
5111 } else { /* nothing is good, presume 0???? */
5112 correction
[ichain
] = 0;
5113 temperature
[ichain
] = 0;
5114 voltage
[ichain
] = 0;
5120 ar9003_hw_power_control_override(ah
, frequency
, correction
, voltage
,
5123 ath_dbg(common
, EEPROM
,
5124 "for frequency=%d, calibration correction = %d %d %d\n",
5125 frequency
, correction
[0], correction
[1], correction
[2]);
5127 /* Store calibrated noise floor values */
5128 for (ichain
= 0; ichain
< AR5416_MAX_CHAINS
; ichain
++)
5130 ah
->nf_5g
.cal
[ichain
] = nf_cal
[ichain
];
5131 ah
->nf_5g
.pwr
[ichain
] = nf_pwr
[ichain
];
5133 ah
->nf_2g
.cal
[ichain
] = nf_cal
[ichain
];
5134 ah
->nf_2g
.pwr
[ichain
] = nf_pwr
[ichain
];
5140 static u16
ar9003_hw_get_direct_edge_power(struct ar9300_eeprom
*eep
,
5145 struct cal_ctl_data_2g
*ctl_2g
= eep
->ctlPowerData_2G
;
5146 struct cal_ctl_data_5g
*ctl_5g
= eep
->ctlPowerData_5G
;
5149 return CTL_EDGE_TPOWER(ctl_2g
[idx
].ctlEdges
[edge
]);
5151 return CTL_EDGE_TPOWER(ctl_5g
[idx
].ctlEdges
[edge
]);
5154 static u16
ar9003_hw_get_indirect_edge_power(struct ar9300_eeprom
*eep
,
5160 struct cal_ctl_data_2g
*ctl_2g
= eep
->ctlPowerData_2G
;
5161 struct cal_ctl_data_5g
*ctl_5g
= eep
->ctlPowerData_5G
;
5163 u8
*ctl_freqbin
= is2GHz
?
5164 &eep
->ctl_freqbin_2G
[idx
][0] :
5165 &eep
->ctl_freqbin_5G
[idx
][0];
5168 if (ath9k_hw_fbin2freq(ctl_freqbin
[edge
- 1], 1) < freq
&&
5169 CTL_EDGE_FLAGS(ctl_2g
[idx
].ctlEdges
[edge
- 1]))
5170 return CTL_EDGE_TPOWER(ctl_2g
[idx
].ctlEdges
[edge
- 1]);
5172 if (ath9k_hw_fbin2freq(ctl_freqbin
[edge
- 1], 0) < freq
&&
5173 CTL_EDGE_FLAGS(ctl_5g
[idx
].ctlEdges
[edge
- 1]))
5174 return CTL_EDGE_TPOWER(ctl_5g
[idx
].ctlEdges
[edge
- 1]);
5177 return MAX_RATE_POWER
;
5181 * Find the maximum conformance test limit for the given channel and CTL info
5183 static u16
ar9003_hw_get_max_edge_power(struct ar9300_eeprom
*eep
,
5184 u16 freq
, int idx
, bool is2GHz
)
5186 u16 twiceMaxEdgePower
= MAX_RATE_POWER
;
5187 u8
*ctl_freqbin
= is2GHz
?
5188 &eep
->ctl_freqbin_2G
[idx
][0] :
5189 &eep
->ctl_freqbin_5G
[idx
][0];
5190 u16 num_edges
= is2GHz
?
5191 AR9300_NUM_BAND_EDGES_2G
: AR9300_NUM_BAND_EDGES_5G
;
5194 /* Get the edge power */
5196 (edge
< num_edges
) && (ctl_freqbin
[edge
] != AR5416_BCHAN_UNUSED
);
5199 * If there's an exact channel match or an inband flag set
5200 * on the lower channel use the given rdEdgePower
5202 if (freq
== ath9k_hw_fbin2freq(ctl_freqbin
[edge
], is2GHz
)) {
5204 ar9003_hw_get_direct_edge_power(eep
, idx
,
5207 } else if ((edge
> 0) &&
5208 (freq
< ath9k_hw_fbin2freq(ctl_freqbin
[edge
],
5211 ar9003_hw_get_indirect_edge_power(eep
, idx
,
5215 * Leave loop - no more affecting edges possible in
5216 * this monotonic increasing list
5222 if (is2GHz
&& !twiceMaxEdgePower
)
5223 twiceMaxEdgePower
= 60;
5225 return twiceMaxEdgePower
;
5228 static void ar9003_hw_set_power_per_rate_table(struct ath_hw
*ah
,
5229 struct ath9k_channel
*chan
,
5230 u8
*pPwrArray
, u16 cfgCtl
,
5231 u8 antenna_reduction
,
5234 struct ath_common
*common
= ath9k_hw_common(ah
);
5235 struct ar9300_eeprom
*pEepData
= &ah
->eeprom
.ar9300_eep
;
5236 u16 twiceMaxEdgePower
;
5238 u16 scaledPower
= 0, minCtlPower
;
5239 static const u16 ctlModesFor11a
[] = {
5240 CTL_11A
, CTL_5GHT20
, CTL_11A_EXT
, CTL_5GHT40
5242 static const u16 ctlModesFor11g
[] = {
5243 CTL_11B
, CTL_11G
, CTL_2GHT20
, CTL_11B_EXT
,
5244 CTL_11G_EXT
, CTL_2GHT40
5247 const u16
*pCtlMode
;
5249 struct chan_centers centers
;
5252 u16 twiceMinEdgePower
;
5253 bool is2ghz
= IS_CHAN_2GHZ(chan
);
5255 ath9k_hw_get_channel_centers(ah
, chan
, ¢ers
);
5256 scaledPower
= ath9k_hw_get_scaled_power(ah
, powerLimit
,
5260 /* Setup for CTL modes */
5261 /* CTL_11B, CTL_11G, CTL_2GHT20 */
5263 ARRAY_SIZE(ctlModesFor11g
) -
5264 SUB_NUM_CTL_MODES_AT_2G_40
;
5265 pCtlMode
= ctlModesFor11g
;
5266 if (IS_CHAN_HT40(chan
))
5268 numCtlModes
= ARRAY_SIZE(ctlModesFor11g
);
5270 /* Setup for CTL modes */
5271 /* CTL_11A, CTL_5GHT20 */
5272 numCtlModes
= ARRAY_SIZE(ctlModesFor11a
) -
5273 SUB_NUM_CTL_MODES_AT_5G_40
;
5274 pCtlMode
= ctlModesFor11a
;
5275 if (IS_CHAN_HT40(chan
))
5277 numCtlModes
= ARRAY_SIZE(ctlModesFor11a
);
5281 * For MIMO, need to apply regulatory caps individually across
5282 * dynamically running modes: CCK, OFDM, HT20, HT40
5284 * The outer loop walks through each possible applicable runtime mode.
5285 * The inner loop walks through each ctlIndex entry in EEPROM.
5286 * The ctl value is encoded as [7:4] == test group, [3:0] == test mode.
5288 for (ctlMode
= 0; ctlMode
< numCtlModes
; ctlMode
++) {
5289 bool isHt40CtlMode
= (pCtlMode
[ctlMode
] == CTL_5GHT40
) ||
5290 (pCtlMode
[ctlMode
] == CTL_2GHT40
);
5292 freq
= centers
.synth_center
;
5293 else if (pCtlMode
[ctlMode
] & EXT_ADDITIVE
)
5294 freq
= centers
.ext_center
;
5296 freq
= centers
.ctl_center
;
5298 ath_dbg(common
, REGULATORY
,
5299 "LOOP-Mode ctlMode %d < %d, isHt40CtlMode %d, EXT_ADDITIVE %d\n",
5300 ctlMode
, numCtlModes
, isHt40CtlMode
,
5301 (pCtlMode
[ctlMode
] & EXT_ADDITIVE
));
5303 /* walk through each CTL index stored in EEPROM */
5305 ctlIndex
= pEepData
->ctlIndex_2G
;
5306 ctlNum
= AR9300_NUM_CTLS_2G
;
5308 ctlIndex
= pEepData
->ctlIndex_5G
;
5309 ctlNum
= AR9300_NUM_CTLS_5G
;
5312 twiceMaxEdgePower
= MAX_RATE_POWER
;
5313 for (i
= 0; (i
< ctlNum
) && ctlIndex
[i
]; i
++) {
5314 ath_dbg(common
, REGULATORY
,
5315 "LOOP-Ctlidx %d: cfgCtl 0x%2.2x pCtlMode 0x%2.2x ctlIndex 0x%2.2x chan %d\n",
5316 i
, cfgCtl
, pCtlMode
[ctlMode
], ctlIndex
[i
],
5320 * compare test group from regulatory
5321 * channel list with test mode from pCtlMode
5324 if ((((cfgCtl
& ~CTL_MODE_M
) |
5325 (pCtlMode
[ctlMode
] & CTL_MODE_M
)) ==
5327 (((cfgCtl
& ~CTL_MODE_M
) |
5328 (pCtlMode
[ctlMode
] & CTL_MODE_M
)) ==
5329 ((ctlIndex
[i
] & CTL_MODE_M
) |
5332 ar9003_hw_get_max_edge_power(pEepData
,
5336 if ((cfgCtl
& ~CTL_MODE_M
) == SD_NO_CTL
)
5338 * Find the minimum of all CTL
5339 * edge powers that apply to
5343 min(twiceMaxEdgePower
,
5347 twiceMaxEdgePower
= twiceMinEdgePower
;
5353 minCtlPower
= (u8
)min(twiceMaxEdgePower
, scaledPower
);
5355 ath_dbg(common
, REGULATORY
,
5356 "SEL-Min ctlMode %d pCtlMode %d 2xMaxEdge %d sP %d minCtlPwr %d\n",
5357 ctlMode
, pCtlMode
[ctlMode
], twiceMaxEdgePower
,
5358 scaledPower
, minCtlPower
);
5360 /* Apply ctl mode to correct target power set */
5361 switch (pCtlMode
[ctlMode
]) {
5363 for (i
= ALL_TARGET_LEGACY_1L_5L
;
5364 i
<= ALL_TARGET_LEGACY_11S
; i
++)
5365 pPwrArray
[i
] = (u8
)min((u16
)pPwrArray
[i
],
5370 for (i
= ALL_TARGET_LEGACY_6_24
;
5371 i
<= ALL_TARGET_LEGACY_54
; i
++)
5372 pPwrArray
[i
] = (u8
)min((u16
)pPwrArray
[i
],
5377 for (i
= ALL_TARGET_HT20_0_8_16
;
5378 i
<= ALL_TARGET_HT20_23
; i
++) {
5379 pPwrArray
[i
] = (u8
)min((u16
)pPwrArray
[i
],
5381 if (ath9k_hw_mci_is_enabled(ah
))
5383 (u8
)min((u16
)pPwrArray
[i
],
5384 ar9003_mci_get_max_txpower(ah
,
5385 pCtlMode
[ctlMode
]));
5390 for (i
= ALL_TARGET_HT40_0_8_16
;
5391 i
<= ALL_TARGET_HT40_23
; i
++) {
5392 pPwrArray
[i
] = (u8
)min((u16
)pPwrArray
[i
],
5394 if (ath9k_hw_mci_is_enabled(ah
))
5396 (u8
)min((u16
)pPwrArray
[i
],
5397 ar9003_mci_get_max_txpower(ah
,
5398 pCtlMode
[ctlMode
]));
5404 } /* end ctl mode checking */
5407 static inline u8
mcsidx_to_tgtpwridx(unsigned int mcs_idx
, u8 base_pwridx
)
5409 u8 mod_idx
= mcs_idx
% 8;
5412 return mod_idx
? (base_pwridx
+ 1) : base_pwridx
;
5414 return base_pwridx
+ 4 * (mcs_idx
/ 8) + mod_idx
- 2;
5417 static void ar9003_paprd_set_txpower(struct ath_hw
*ah
,
5418 struct ath9k_channel
*chan
,
5419 u8
*targetPowerValT2
)
5423 if (!ar9003_is_paprd_enabled(ah
))
5426 if (IS_CHAN_HT40(chan
))
5427 i
= ALL_TARGET_HT40_7
;
5429 i
= ALL_TARGET_HT20_7
;
5431 if (IS_CHAN_2GHZ(chan
)) {
5432 if (!AR_SREV_9330(ah
) && !AR_SREV_9340(ah
) &&
5433 !AR_SREV_9462(ah
) && !AR_SREV_9565(ah
)) {
5434 if (IS_CHAN_HT40(chan
))
5435 i
= ALL_TARGET_HT40_0_8_16
;
5437 i
= ALL_TARGET_HT20_0_8_16
;
5441 ah
->paprd_target_power
= targetPowerValT2
[i
];
5444 static void ath9k_hw_ar9300_set_txpower(struct ath_hw
*ah
,
5445 struct ath9k_channel
*chan
, u16 cfgCtl
,
5446 u8 twiceAntennaReduction
,
5447 u8 powerLimit
, bool test
)
5449 struct ath_regulatory
*regulatory
= ath9k_hw_regulatory(ah
);
5450 struct ath_common
*common
= ath9k_hw_common(ah
);
5451 struct ar9300_eeprom
*eep
= &ah
->eeprom
.ar9300_eep
;
5452 struct ar9300_modal_eep_header
*modal_hdr
;
5453 u8 targetPowerValT2
[ar9300RateSize
];
5454 u8 target_power_val_t2_eep
[ar9300RateSize
];
5455 u8 targetPowerValT2_tpc
[ar9300RateSize
];
5456 unsigned int i
= 0, paprd_scale_factor
= 0;
5457 u8 pwr_idx
, min_pwridx
= 0;
5459 memset(targetPowerValT2
, 0 , sizeof(targetPowerValT2
));
5462 * Get target powers from EEPROM - our baseline for TX Power
5464 ar9003_hw_get_target_power_eeprom(ah
, chan
, targetPowerValT2
);
5466 if (ar9003_is_paprd_enabled(ah
)) {
5467 if (IS_CHAN_2GHZ(chan
))
5468 modal_hdr
= &eep
->modalHeader2G
;
5470 modal_hdr
= &eep
->modalHeader5G
;
5472 ah
->paprd_ratemask
=
5473 le32_to_cpu(modal_hdr
->papdRateMaskHt20
) &
5474 AR9300_PAPRD_RATE_MASK
;
5476 ah
->paprd_ratemask_ht40
=
5477 le32_to_cpu(modal_hdr
->papdRateMaskHt40
) &
5478 AR9300_PAPRD_RATE_MASK
;
5480 paprd_scale_factor
= ar9003_get_paprd_scale_factor(ah
, chan
);
5481 min_pwridx
= IS_CHAN_HT40(chan
) ? ALL_TARGET_HT40_0_8_16
:
5482 ALL_TARGET_HT20_0_8_16
;
5484 if (!ah
->paprd_table_write_done
) {
5485 memcpy(target_power_val_t2_eep
, targetPowerValT2
,
5486 sizeof(targetPowerValT2
));
5487 for (i
= 0; i
< 24; i
++) {
5488 pwr_idx
= mcsidx_to_tgtpwridx(i
, min_pwridx
);
5489 if (ah
->paprd_ratemask
& (1 << i
)) {
5490 if (targetPowerValT2
[pwr_idx
] &&
5491 targetPowerValT2
[pwr_idx
] ==
5492 target_power_val_t2_eep
[pwr_idx
])
5493 targetPowerValT2
[pwr_idx
] -=
5498 memcpy(target_power_val_t2_eep
, targetPowerValT2
,
5499 sizeof(targetPowerValT2
));
5502 ar9003_hw_set_power_per_rate_table(ah
, chan
,
5503 targetPowerValT2
, cfgCtl
,
5504 twiceAntennaReduction
,
5507 memcpy(targetPowerValT2_tpc
, targetPowerValT2
,
5508 sizeof(targetPowerValT2
));
5510 if (ar9003_is_paprd_enabled(ah
)) {
5511 for (i
= 0; i
< ar9300RateSize
; i
++) {
5512 if ((ah
->paprd_ratemask
& (1 << i
)) &&
5513 (abs(targetPowerValT2
[i
] -
5514 target_power_val_t2_eep
[i
]) >
5515 paprd_scale_factor
)) {
5516 ah
->paprd_ratemask
&= ~(1 << i
);
5517 ath_dbg(common
, EEPROM
,
5518 "paprd disabled for mcs %d\n", i
);
5523 regulatory
->max_power_level
= 0;
5524 for (i
= 0; i
< ar9300RateSize
; i
++) {
5525 if (targetPowerValT2
[i
] > regulatory
->max_power_level
)
5526 regulatory
->max_power_level
= targetPowerValT2
[i
];
5529 ath9k_hw_update_regulatory_maxpower(ah
);
5534 for (i
= 0; i
< ar9300RateSize
; i
++) {
5535 ath_dbg(common
, REGULATORY
, "TPC[%02d] 0x%08x\n",
5536 i
, targetPowerValT2
[i
]);
5539 /* Write target power array to registers */
5540 ar9003_hw_tx_power_regwrite(ah
, targetPowerValT2
);
5541 ar9003_hw_calibration_apply(ah
, chan
->channel
);
5542 ar9003_paprd_set_txpower(ah
, chan
, targetPowerValT2
);
5544 ar9003_hw_selfgen_tpc_txpower(ah
, chan
, targetPowerValT2
);
5546 /* TPC initializations */
5547 if (ah
->tpc_enabled
) {
5550 ar9003_hw_init_rate_txpower(ah
, targetPowerValT2_tpc
, chan
);
5553 REG_WRITE(ah
, AR_PHY_PWRTX_MAX
,
5554 AR_PHY_POWER_TX_RATE_MAX_TPC_ENABLE
);
5555 /* Disable per chain power reduction */
5556 val
= REG_READ(ah
, AR_PHY_POWER_TX_SUB
);
5557 if (AR_SREV_9340(ah
))
5558 REG_WRITE(ah
, AR_PHY_POWER_TX_SUB
,
5561 REG_WRITE(ah
, AR_PHY_POWER_TX_SUB
,
5565 REG_WRITE(ah
, AR_PHY_PWRTX_MAX
, 0);
5569 static u16
ath9k_hw_ar9300_get_spur_channel(struct ath_hw
*ah
,
5575 s32
ar9003_hw_get_tx_gain_idx(struct ath_hw
*ah
)
5577 struct ar9300_eeprom
*eep
= &ah
->eeprom
.ar9300_eep
;
5579 return (eep
->baseEepHeader
.txrxgain
>> 4) & 0xf; /* bits 7:4 */
5582 s32
ar9003_hw_get_rx_gain_idx(struct ath_hw
*ah
)
5584 struct ar9300_eeprom
*eep
= &ah
->eeprom
.ar9300_eep
;
5586 return (eep
->baseEepHeader
.txrxgain
) & 0xf; /* bits 3:0 */
5589 u8
*ar9003_get_spur_chan_ptr(struct ath_hw
*ah
, bool is2ghz
)
5591 return ar9003_modal_header(ah
, is2ghz
)->spurChans
;
5594 unsigned int ar9003_get_paprd_scale_factor(struct ath_hw
*ah
,
5595 struct ath9k_channel
*chan
)
5597 struct ar9300_eeprom
*eep
= &ah
->eeprom
.ar9300_eep
;
5599 if (IS_CHAN_2GHZ(chan
))
5600 return MS(le32_to_cpu(eep
->modalHeader2G
.papdRateMaskHt20
),
5601 AR9300_PAPRD_SCALE_1
);
5603 if (chan
->channel
>= 5700)
5604 return MS(le32_to_cpu(eep
->modalHeader5G
.papdRateMaskHt20
),
5605 AR9300_PAPRD_SCALE_1
);
5606 else if (chan
->channel
>= 5400)
5607 return MS(le32_to_cpu(eep
->modalHeader5G
.papdRateMaskHt40
),
5608 AR9300_PAPRD_SCALE_2
);
5610 return MS(le32_to_cpu(eep
->modalHeader5G
.papdRateMaskHt40
),
5611 AR9300_PAPRD_SCALE_1
);
5615 static u8
ar9003_get_eepmisc(struct ath_hw
*ah
)
5617 return ah
->eeprom
.map4k
.baseEepHeader
.eepMisc
;
5620 const struct eeprom_ops eep_ar9300_ops
= {
5621 .check_eeprom
= ath9k_hw_ar9300_check_eeprom
,
5622 .get_eeprom
= ath9k_hw_ar9300_get_eeprom
,
5623 .fill_eeprom
= ath9k_hw_ar9300_fill_eeprom
,
5624 .dump_eeprom
= ath9k_hw_ar9003_dump_eeprom
,
5625 .get_eeprom_ver
= ath9k_hw_ar9300_get_eeprom_ver
,
5626 .get_eeprom_rev
= ath9k_hw_ar9300_get_eeprom_rev
,
5627 .set_board_values
= ath9k_hw_ar9300_set_board_values
,
5628 .set_addac
= ath9k_hw_ar9300_set_addac
,
5629 .set_txpower
= ath9k_hw_ar9300_set_txpower
,
5630 .get_spur_channel
= ath9k_hw_ar9300_get_spur_channel
,
5631 .get_eepmisc
= ar9003_get_eepmisc