18fe65368SLuis R. Rodriguez /*
25b68138eSSujith Manoharan * Copyright (c) 2008-2011 Atheros Communications Inc.
38fe65368SLuis R. Rodriguez *
48fe65368SLuis R. Rodriguez * Permission to use, copy, modify, and/or distribute this software for any
58fe65368SLuis R. Rodriguez * purpose with or without fee is hereby granted, provided that the above
68fe65368SLuis R. Rodriguez * copyright notice and this permission notice appear in all copies.
78fe65368SLuis R. Rodriguez *
88fe65368SLuis R. Rodriguez * THE SOFTWARE IS PROVIDED "AS IS" AND THE AUTHOR DISCLAIMS ALL WARRANTIES
98fe65368SLuis R. Rodriguez * WITH REGARD TO THIS SOFTWARE INCLUDING ALL IMPLIED WARRANTIES OF
108fe65368SLuis R. Rodriguez * MERCHANTABILITY AND FITNESS. IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR
118fe65368SLuis R. Rodriguez * ANY SPECIAL, DIRECT, INDIRECT, OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES
128fe65368SLuis R. Rodriguez * WHATSOEVER RESULTING FROM LOSS OF USE, DATA OR PROFITS, WHETHER IN AN
138fe65368SLuis R. Rodriguez * ACTION OF CONTRACT, NEGLIGENCE OR OTHER TORTIOUS ACTION, ARISING OUT OF
148fe65368SLuis R. Rodriguez * OR IN CONNECTION WITH THE USE OR PERFORMANCE OF THIS SOFTWARE.
158fe65368SLuis R. Rodriguez */
168fe65368SLuis R. Rodriguez
178fe65368SLuis R. Rodriguez #include "hw.h"
188fe65368SLuis R. Rodriguez #include "hw-ops.h"
198fe65368SLuis R. Rodriguez #include "../regd.h"
208fe65368SLuis R. Rodriguez #include "ar9002_phy.h"
218fe65368SLuis R. Rodriguez
22e36b27afSLuis R. Rodriguez /* All code below is for AR5008, AR9001, AR9002 */
23e36b27afSLuis R. Rodriguez
24c08267dcSLorenzo Bianconi #define AR5008_OFDM_RATES 8
25c08267dcSLorenzo Bianconi #define AR5008_HT_SS_RATES 8
26c08267dcSLorenzo Bianconi #define AR5008_HT_DS_RATES 8
27c08267dcSLorenzo Bianconi
28c08267dcSLorenzo Bianconi #define AR5008_HT20_SHIFT 16
29c08267dcSLorenzo Bianconi #define AR5008_HT40_SHIFT 24
30c08267dcSLorenzo Bianconi
31c08267dcSLorenzo Bianconi #define AR5008_11NA_OFDM_SHIFT 0
32c08267dcSLorenzo Bianconi #define AR5008_11NA_HT_SS_SHIFT 8
33c08267dcSLorenzo Bianconi #define AR5008_11NA_HT_DS_SHIFT 16
34c08267dcSLorenzo Bianconi
35c08267dcSLorenzo Bianconi #define AR5008_11NG_OFDM_SHIFT 4
36c08267dcSLorenzo Bianconi #define AR5008_11NG_HT_SS_SHIFT 12
37c08267dcSLorenzo Bianconi #define AR5008_11NG_HT_DS_SHIFT 20
38c08267dcSLorenzo Bianconi
39e36b27afSLuis R. Rodriguez /*
40e36b27afSLuis R. Rodriguez * register values to turn OFDM weak signal detection OFF
41e36b27afSLuis R. Rodriguez */
42e36b27afSLuis R. Rodriguez static const int m1ThreshLow_off = 127;
43e36b27afSLuis R. Rodriguez static const int m2ThreshLow_off = 127;
44e36b27afSLuis R. Rodriguez static const int m1Thresh_off = 127;
45e36b27afSLuis R. Rodriguez static const int m2Thresh_off = 127;
46e36b27afSLuis R. Rodriguez static const int m2CountThr_off = 31;
47e36b27afSLuis R. Rodriguez static const int m2CountThrLow_off = 63;
48e36b27afSLuis R. Rodriguez static const int m1ThreshLowExt_off = 127;
49e36b27afSLuis R. Rodriguez static const int m2ThreshLowExt_off = 127;
50e36b27afSLuis R. Rodriguez static const int m1ThreshExt_off = 127;
51e36b27afSLuis R. Rodriguez static const int m2ThreshExt_off = 127;
52e36b27afSLuis R. Rodriguez
53b016d0a5SLee Jones static const u32 ar5416Bank0[][2] = {
54b016d0a5SLee Jones /* Addr allmodes */
55b016d0a5SLee Jones {0x000098b0, 0x1e5795e5},
56b016d0a5SLee Jones {0x000098e0, 0x02008020},
57b016d0a5SLee Jones };
58b016d0a5SLee Jones
59b016d0a5SLee Jones static const u32 ar5416Bank1[][2] = {
60b016d0a5SLee Jones /* Addr allmodes */
61b016d0a5SLee Jones {0x000098b0, 0x02108421},
62b016d0a5SLee Jones {0x000098ec, 0x00000008},
63b016d0a5SLee Jones };
64b016d0a5SLee Jones
65b016d0a5SLee Jones static const u32 ar5416Bank2[][2] = {
66b016d0a5SLee Jones /* Addr allmodes */
67b016d0a5SLee Jones {0x000098b0, 0x0e73ff17},
68b016d0a5SLee Jones {0x000098e0, 0x00000420},
69b016d0a5SLee Jones };
70b016d0a5SLee Jones
71b016d0a5SLee Jones static const u32 ar5416Bank3[][3] = {
72b016d0a5SLee Jones /* Addr 5G 2G */
73b016d0a5SLee Jones {0x000098f0, 0x01400018, 0x01c00018},
74b016d0a5SLee Jones };
75b016d0a5SLee Jones
76b016d0a5SLee Jones static const u32 ar5416Bank7[][2] = {
77b016d0a5SLee Jones /* Addr allmodes */
78b016d0a5SLee Jones {0x0000989c, 0x00000500},
79b016d0a5SLee Jones {0x0000989c, 0x00000800},
80b016d0a5SLee Jones {0x000098cc, 0x0000000e},
81b016d0a5SLee Jones };
82b016d0a5SLee Jones
83a043dfb9SFelix Fietkau static const struct ar5416IniArray bank0 = STATIC_INI_ARRAY(ar5416Bank0);
84a043dfb9SFelix Fietkau static const struct ar5416IniArray bank1 = STATIC_INI_ARRAY(ar5416Bank1);
85a043dfb9SFelix Fietkau static const struct ar5416IniArray bank2 = STATIC_INI_ARRAY(ar5416Bank2);
86a043dfb9SFelix Fietkau static const struct ar5416IniArray bank3 = STATIC_INI_ARRAY(ar5416Bank3);
87a043dfb9SFelix Fietkau static const struct ar5416IniArray bank7 = STATIC_INI_ARRAY(ar5416Bank7);
888fe65368SLuis R. Rodriguez
ar5008_write_bank6(struct ath_hw * ah,unsigned int * writecnt)89a043dfb9SFelix Fietkau static void ar5008_write_bank6(struct ath_hw *ah, unsigned int *writecnt)
90a9b6b256SFelix Fietkau {
91a043dfb9SFelix Fietkau struct ar5416IniArray *array = &ah->iniBank6;
92a043dfb9SFelix Fietkau u32 *data = ah->analogBank6Data;
93a9b6b256SFelix Fietkau int r;
94a9b6b256SFelix Fietkau
95a9b6b256SFelix Fietkau ENABLE_REGWRITE_BUFFER(ah);
96a9b6b256SFelix Fietkau
97a9b6b256SFelix Fietkau for (r = 0; r < array->ia_rows; r++) {
98a9b6b256SFelix Fietkau REG_WRITE(ah, INI_RA(array, r, 0), data[r]);
99a9b6b256SFelix Fietkau DO_DELAY(*writecnt);
100a9b6b256SFelix Fietkau }
101a9b6b256SFelix Fietkau
102a9b6b256SFelix Fietkau REGWRITE_BUFFER_FLUSH(ah);
103a9b6b256SFelix Fietkau }
104a9b6b256SFelix Fietkau
1058cc107b5SLee Jones /*
1068fe65368SLuis R. Rodriguez * ar5008_hw_phy_modify_rx_buffer() - perform analog swizzling of parameters
1078fe65368SLuis R. Rodriguez *
1088fe65368SLuis R. Rodriguez * Performs analog "swizzling" of parameters into their location.
1098fe65368SLuis R. Rodriguez * Used on external AR2133/AR5133 radios.
1108fe65368SLuis R. Rodriguez */
ar5008_hw_phy_modify_rx_buffer(u32 * rfBuf,u32 reg32,u32 numBits,u32 firstBit,u32 column)1118fe65368SLuis R. Rodriguez static void ar5008_hw_phy_modify_rx_buffer(u32 *rfBuf, u32 reg32,
1128fe65368SLuis R. Rodriguez u32 numBits, u32 firstBit,
1138fe65368SLuis R. Rodriguez u32 column)
1148fe65368SLuis R. Rodriguez {
1158fe65368SLuis R. Rodriguez u32 tmp32, mask, arrayEntry, lastBit;
1168fe65368SLuis R. Rodriguez int32_t bitPosition, bitsLeft;
1178fe65368SLuis R. Rodriguez
1188fe65368SLuis R. Rodriguez tmp32 = ath9k_hw_reverse_bits(reg32, numBits);
1198fe65368SLuis R. Rodriguez arrayEntry = (firstBit - 1) / 8;
1208fe65368SLuis R. Rodriguez bitPosition = (firstBit - 1) % 8;
1218fe65368SLuis R. Rodriguez bitsLeft = numBits;
1228fe65368SLuis R. Rodriguez while (bitsLeft > 0) {
1238fe65368SLuis R. Rodriguez lastBit = (bitPosition + bitsLeft > 8) ?
1248fe65368SLuis R. Rodriguez 8 : bitPosition + bitsLeft;
1258fe65368SLuis R. Rodriguez mask = (((1 << lastBit) - 1) ^ ((1 << bitPosition) - 1)) <<
1268fe65368SLuis R. Rodriguez (column * 8);
1278fe65368SLuis R. Rodriguez rfBuf[arrayEntry] &= ~mask;
1288fe65368SLuis R. Rodriguez rfBuf[arrayEntry] |= ((tmp32 << bitPosition) <<
1298fe65368SLuis R. Rodriguez (column * 8)) & mask;
1308fe65368SLuis R. Rodriguez bitsLeft -= 8 - bitPosition;
1318fe65368SLuis R. Rodriguez tmp32 = tmp32 >> (8 - bitPosition);
1328fe65368SLuis R. Rodriguez bitPosition = 0;
1338fe65368SLuis R. Rodriguez arrayEntry++;
1348fe65368SLuis R. Rodriguez }
1358fe65368SLuis R. Rodriguez }
1368fe65368SLuis R. Rodriguez
1378fe65368SLuis R. Rodriguez /*
1388fe65368SLuis R. Rodriguez * Fix on 2.4 GHz band for orientation sensitivity issue by increasing
1398fe65368SLuis R. Rodriguez * rf_pwd_icsyndiv.
1408fe65368SLuis R. Rodriguez *
1418fe65368SLuis R. Rodriguez * Theoretical Rules:
1428fe65368SLuis R. Rodriguez * if 2 GHz band
1438fe65368SLuis R. Rodriguez * if forceBiasAuto
1448fe65368SLuis R. Rodriguez * if synth_freq < 2412
1458fe65368SLuis R. Rodriguez * bias = 0
1468fe65368SLuis R. Rodriguez * else if 2412 <= synth_freq <= 2422
1478fe65368SLuis R. Rodriguez * bias = 1
1488fe65368SLuis R. Rodriguez * else // synth_freq > 2422
1498fe65368SLuis R. Rodriguez * bias = 2
1508fe65368SLuis R. Rodriguez * else if forceBias > 0
1518fe65368SLuis R. Rodriguez * bias = forceBias & 7
1528fe65368SLuis R. Rodriguez * else
1538fe65368SLuis R. Rodriguez * no change, use value from ini file
1548fe65368SLuis R. Rodriguez * else
1558fe65368SLuis R. Rodriguez * no change, invalid band
1568fe65368SLuis R. Rodriguez *
1578fe65368SLuis R. Rodriguez * 1st Mod:
1588fe65368SLuis R. Rodriguez * 2422 also uses value of 2
1598fe65368SLuis R. Rodriguez * <approved>
1608fe65368SLuis R. Rodriguez *
1618fe65368SLuis R. Rodriguez * 2nd Mod:
1628fe65368SLuis R. Rodriguez * Less than 2412 uses value of 0, 2412 and above uses value of 2
1638fe65368SLuis R. Rodriguez */
ar5008_hw_force_bias(struct ath_hw * ah,u16 synth_freq)1648fe65368SLuis R. Rodriguez static void ar5008_hw_force_bias(struct ath_hw *ah, u16 synth_freq)
1658fe65368SLuis R. Rodriguez {
1668fe65368SLuis R. Rodriguez struct ath_common *common = ath9k_hw_common(ah);
1678fe65368SLuis R. Rodriguez u32 tmp_reg;
1688fe65368SLuis R. Rodriguez int reg_writes = 0;
1698fe65368SLuis R. Rodriguez u32 new_bias = 0;
1708fe65368SLuis R. Rodriguez
1718fe65368SLuis R. Rodriguez if (!AR_SREV_5416(ah) || synth_freq >= 3000)
1728fe65368SLuis R. Rodriguez return;
1738fe65368SLuis R. Rodriguez
1747a37081eSFelix Fietkau BUG_ON(AR_SREV_9280_20_OR_LATER(ah));
1758fe65368SLuis R. Rodriguez
1768fe65368SLuis R. Rodriguez if (synth_freq < 2412)
1778fe65368SLuis R. Rodriguez new_bias = 0;
1788fe65368SLuis R. Rodriguez else if (synth_freq < 2422)
1798fe65368SLuis R. Rodriguez new_bias = 1;
1808fe65368SLuis R. Rodriguez else
1818fe65368SLuis R. Rodriguez new_bias = 2;
1828fe65368SLuis R. Rodriguez
1838fe65368SLuis R. Rodriguez /* pre-reverse this field */
1848fe65368SLuis R. Rodriguez tmp_reg = ath9k_hw_reverse_bits(new_bias, 3);
1858fe65368SLuis R. Rodriguez
186d2182b69SJoe Perches ath_dbg(common, CONFIG, "Force rf_pwd_icsyndiv to %1d on %4d\n",
1878fe65368SLuis R. Rodriguez new_bias, synth_freq);
1888fe65368SLuis R. Rodriguez
1898fe65368SLuis R. Rodriguez /* swizzle rf_pwd_icsyndiv */
1908fe65368SLuis R. Rodriguez ar5008_hw_phy_modify_rx_buffer(ah->analogBank6Data, tmp_reg, 3, 181, 3);
1918fe65368SLuis R. Rodriguez
1928fe65368SLuis R. Rodriguez /* write Bank 6 with new params */
193a043dfb9SFelix Fietkau ar5008_write_bank6(ah, ®_writes);
1948fe65368SLuis R. Rodriguez }
1958fe65368SLuis R. Rodriguez
1968cc107b5SLee Jones /*
1978fe65368SLuis R. Rodriguez * ar5008_hw_set_channel - tune to a channel on the external AR2133/AR5133 radios
1988fe65368SLuis R. Rodriguez *
1998fe65368SLuis R. Rodriguez * For the external AR2133/AR5133 radios, takes the MHz channel value and set
2008fe65368SLuis R. Rodriguez * the channel value. Assumes writes enabled to analog bus and bank6 register
2018fe65368SLuis R. Rodriguez * cache in ah->analogBank6Data.
2028fe65368SLuis R. Rodriguez */
ar5008_hw_set_channel(struct ath_hw * ah,struct ath9k_channel * chan)2038fe65368SLuis R. Rodriguez static int ar5008_hw_set_channel(struct ath_hw *ah, struct ath9k_channel *chan)
2048fe65368SLuis R. Rodriguez {
2058fe65368SLuis R. Rodriguez struct ath_common *common = ath9k_hw_common(ah);
2068fe65368SLuis R. Rodriguez u32 channelSel = 0;
2078fe65368SLuis R. Rodriguez u32 bModeSynth = 0;
2088fe65368SLuis R. Rodriguez u32 aModeRefSel = 0;
2098fe65368SLuis R. Rodriguez u32 reg32 = 0;
2108fe65368SLuis R. Rodriguez u16 freq;
2118fe65368SLuis R. Rodriguez struct chan_centers centers;
2128fe65368SLuis R. Rodriguez
2138fe65368SLuis R. Rodriguez ath9k_hw_get_channel_centers(ah, chan, ¢ers);
2148fe65368SLuis R. Rodriguez freq = centers.synth_center;
2158fe65368SLuis R. Rodriguez
2168fe65368SLuis R. Rodriguez if (freq < 4800) {
2178fe65368SLuis R. Rodriguez u32 txctl;
2188fe65368SLuis R. Rodriguez
2198fe65368SLuis R. Rodriguez if (((freq - 2192) % 5) == 0) {
2208fe65368SLuis R. Rodriguez channelSel = ((freq - 672) * 2 - 3040) / 10;
2218fe65368SLuis R. Rodriguez bModeSynth = 0;
2228fe65368SLuis R. Rodriguez } else if (((freq - 2224) % 5) == 0) {
2238fe65368SLuis R. Rodriguez channelSel = ((freq - 704) * 2 - 3040) / 10;
2248fe65368SLuis R. Rodriguez bModeSynth = 1;
2258fe65368SLuis R. Rodriguez } else {
2263800276aSJoe Perches ath_err(common, "Invalid channel %u MHz\n", freq);
2278fe65368SLuis R. Rodriguez return -EINVAL;
2288fe65368SLuis R. Rodriguez }
2298fe65368SLuis R. Rodriguez
2308fe65368SLuis R. Rodriguez channelSel = (channelSel << 2) & 0xff;
2318fe65368SLuis R. Rodriguez channelSel = ath9k_hw_reverse_bits(channelSel, 8);
2328fe65368SLuis R. Rodriguez
2338fe65368SLuis R. Rodriguez txctl = REG_READ(ah, AR_PHY_CCK_TX_CTRL);
2348fe65368SLuis R. Rodriguez if (freq == 2484) {
2358fe65368SLuis R. Rodriguez
2368fe65368SLuis R. Rodriguez REG_WRITE(ah, AR_PHY_CCK_TX_CTRL,
2378fe65368SLuis R. Rodriguez txctl | AR_PHY_CCK_TX_CTRL_JAPAN);
2388fe65368SLuis R. Rodriguez } else {
2398fe65368SLuis R. Rodriguez REG_WRITE(ah, AR_PHY_CCK_TX_CTRL,
2408fe65368SLuis R. Rodriguez txctl & ~AR_PHY_CCK_TX_CTRL_JAPAN);
2418fe65368SLuis R. Rodriguez }
2428fe65368SLuis R. Rodriguez
2438fe65368SLuis R. Rodriguez } else if ((freq % 20) == 0 && freq >= 5120) {
2448fe65368SLuis R. Rodriguez channelSel =
2458fe65368SLuis R. Rodriguez ath9k_hw_reverse_bits(((freq - 4800) / 20 << 2), 8);
2468fe65368SLuis R. Rodriguez aModeRefSel = ath9k_hw_reverse_bits(1, 2);
2478fe65368SLuis R. Rodriguez } else if ((freq % 10) == 0) {
2488fe65368SLuis R. Rodriguez channelSel =
2498fe65368SLuis R. Rodriguez ath9k_hw_reverse_bits(((freq - 4800) / 10 << 1), 8);
2508fe65368SLuis R. Rodriguez if (AR_SREV_9100(ah) || AR_SREV_9160_10_OR_LATER(ah))
2518fe65368SLuis R. Rodriguez aModeRefSel = ath9k_hw_reverse_bits(2, 2);
2528fe65368SLuis R. Rodriguez else
2538fe65368SLuis R. Rodriguez aModeRefSel = ath9k_hw_reverse_bits(1, 2);
2548fe65368SLuis R. Rodriguez } else if ((freq % 5) == 0) {
2558fe65368SLuis R. Rodriguez channelSel = ath9k_hw_reverse_bits((freq - 4800) / 5, 8);
2568fe65368SLuis R. Rodriguez aModeRefSel = ath9k_hw_reverse_bits(1, 2);
2578fe65368SLuis R. Rodriguez } else {
2583800276aSJoe Perches ath_err(common, "Invalid channel %u MHz\n", freq);
2598fe65368SLuis R. Rodriguez return -EINVAL;
2608fe65368SLuis R. Rodriguez }
2618fe65368SLuis R. Rodriguez
2628fe65368SLuis R. Rodriguez ar5008_hw_force_bias(ah, freq);
2638fe65368SLuis R. Rodriguez
2648fe65368SLuis R. Rodriguez reg32 =
2658fe65368SLuis R. Rodriguez (channelSel << 8) | (aModeRefSel << 2) | (bModeSynth << 1) |
2668fe65368SLuis R. Rodriguez (1 << 5) | 0x1;
2678fe65368SLuis R. Rodriguez
2688fe65368SLuis R. Rodriguez REG_WRITE(ah, AR_PHY(0x37), reg32);
2698fe65368SLuis R. Rodriguez
2708fe65368SLuis R. Rodriguez ah->curchan = chan;
2718fe65368SLuis R. Rodriguez
2728fe65368SLuis R. Rodriguez return 0;
2738fe65368SLuis R. Rodriguez }
2748fe65368SLuis R. Rodriguez
ar5008_hw_cmn_spur_mitigate(struct ath_hw * ah,struct ath9k_channel * chan,int bin)275f911085fSOleksij Rempel void ar5008_hw_cmn_spur_mitigate(struct ath_hw *ah,
276f911085fSOleksij Rempel struct ath9k_channel *chan, int bin)
2778fe65368SLuis R. Rodriguez {
278f911085fSOleksij Rempel int cur_bin;
2798fe65368SLuis R. Rodriguez int upper, lower, cur_vit_mask;
2808fe65368SLuis R. Rodriguez int i;
281b5182e15SBob Copeland int8_t mask_m[123] = {0};
282b5182e15SBob Copeland int8_t mask_p[123] = {0};
2838fe65368SLuis R. Rodriguez int8_t mask_amt;
2848fe65368SLuis R. Rodriguez int tmp_mask;
285f911085fSOleksij Rempel static const int pilot_mask_reg[4] = {
286f911085fSOleksij Rempel AR_PHY_TIMING7, AR_PHY_TIMING8,
287f911085fSOleksij Rempel AR_PHY_PILOT_MASK_01_30, AR_PHY_PILOT_MASK_31_60
288f911085fSOleksij Rempel };
289f911085fSOleksij Rempel static const int chan_mask_reg[4] = {
290f911085fSOleksij Rempel AR_PHY_TIMING9, AR_PHY_TIMING10,
291f911085fSOleksij Rempel AR_PHY_CHANNEL_MASK_01_30, AR_PHY_CHANNEL_MASK_31_60
292f911085fSOleksij Rempel };
293f911085fSOleksij Rempel static const int inc[4] = { 0, 100, 0, 0 };
2948fe65368SLuis R. Rodriguez
2958fe65368SLuis R. Rodriguez cur_bin = -6000;
2968fe65368SLuis R. Rodriguez upper = bin + 100;
2978fe65368SLuis R. Rodriguez lower = bin - 100;
2988fe65368SLuis R. Rodriguez
2998fe65368SLuis R. Rodriguez for (i = 0; i < 4; i++) {
3008fe65368SLuis R. Rodriguez int pilot_mask = 0;
3018fe65368SLuis R. Rodriguez int chan_mask = 0;
3028fe65368SLuis R. Rodriguez int bp = 0;
303f911085fSOleksij Rempel
3048fe65368SLuis R. Rodriguez for (bp = 0; bp < 30; bp++) {
3058fe65368SLuis R. Rodriguez if ((cur_bin > lower) && (cur_bin < upper)) {
3068fe65368SLuis R. Rodriguez pilot_mask = pilot_mask | 0x1 << bp;
3078fe65368SLuis R. Rodriguez chan_mask = chan_mask | 0x1 << bp;
3088fe65368SLuis R. Rodriguez }
3098fe65368SLuis R. Rodriguez cur_bin += 100;
3108fe65368SLuis R. Rodriguez }
3118fe65368SLuis R. Rodriguez cur_bin += inc[i];
3128fe65368SLuis R. Rodriguez REG_WRITE(ah, pilot_mask_reg[i], pilot_mask);
3138fe65368SLuis R. Rodriguez REG_WRITE(ah, chan_mask_reg[i], chan_mask);
3148fe65368SLuis R. Rodriguez }
3158fe65368SLuis R. Rodriguez
3168fe65368SLuis R. Rodriguez cur_vit_mask = 6100;
3178fe65368SLuis R. Rodriguez upper = bin + 120;
3188fe65368SLuis R. Rodriguez lower = bin - 120;
3198fe65368SLuis R. Rodriguez
320b5182e15SBob Copeland for (i = 0; i < ARRAY_SIZE(mask_m); i++) {
3218fe65368SLuis R. Rodriguez if ((cur_vit_mask > lower) && (cur_vit_mask < upper)) {
3228fe65368SLuis R. Rodriguez /* workaround for gcc bug #37014 */
3238fe65368SLuis R. Rodriguez volatile int tmp_v = abs(cur_vit_mask - bin);
3248fe65368SLuis R. Rodriguez
3258fe65368SLuis R. Rodriguez if (tmp_v < 75)
3268fe65368SLuis R. Rodriguez mask_amt = 1;
3278fe65368SLuis R. Rodriguez else
3288fe65368SLuis R. Rodriguez mask_amt = 0;
3298fe65368SLuis R. Rodriguez if (cur_vit_mask < 0)
3308fe65368SLuis R. Rodriguez mask_m[abs(cur_vit_mask / 100)] = mask_amt;
3318fe65368SLuis R. Rodriguez else
3328fe65368SLuis R. Rodriguez mask_p[cur_vit_mask / 100] = mask_amt;
3338fe65368SLuis R. Rodriguez }
3348fe65368SLuis R. Rodriguez cur_vit_mask -= 100;
3358fe65368SLuis R. Rodriguez }
3368fe65368SLuis R. Rodriguez
3378fe65368SLuis R. Rodriguez tmp_mask = (mask_m[46] << 30) | (mask_m[47] << 28)
3388fe65368SLuis R. Rodriguez | (mask_m[48] << 26) | (mask_m[49] << 24)
3398fe65368SLuis R. Rodriguez | (mask_m[50] << 22) | (mask_m[51] << 20)
3408fe65368SLuis R. Rodriguez | (mask_m[52] << 18) | (mask_m[53] << 16)
3418fe65368SLuis R. Rodriguez | (mask_m[54] << 14) | (mask_m[55] << 12)
3428fe65368SLuis R. Rodriguez | (mask_m[56] << 10) | (mask_m[57] << 8)
3438fe65368SLuis R. Rodriguez | (mask_m[58] << 6) | (mask_m[59] << 4)
3448fe65368SLuis R. Rodriguez | (mask_m[60] << 2) | (mask_m[61] << 0);
3458fe65368SLuis R. Rodriguez REG_WRITE(ah, AR_PHY_BIN_MASK_1, tmp_mask);
3468fe65368SLuis R. Rodriguez REG_WRITE(ah, AR_PHY_VIT_MASK2_M_46_61, tmp_mask);
3478fe65368SLuis R. Rodriguez
3488fe65368SLuis R. Rodriguez tmp_mask = (mask_m[31] << 28)
3498fe65368SLuis R. Rodriguez | (mask_m[32] << 26) | (mask_m[33] << 24)
3508fe65368SLuis R. Rodriguez | (mask_m[34] << 22) | (mask_m[35] << 20)
3518fe65368SLuis R. Rodriguez | (mask_m[36] << 18) | (mask_m[37] << 16)
3528fe65368SLuis R. Rodriguez | (mask_m[48] << 14) | (mask_m[39] << 12)
3538fe65368SLuis R. Rodriguez | (mask_m[40] << 10) | (mask_m[41] << 8)
3548fe65368SLuis R. Rodriguez | (mask_m[42] << 6) | (mask_m[43] << 4)
3558fe65368SLuis R. Rodriguez | (mask_m[44] << 2) | (mask_m[45] << 0);
3568fe65368SLuis R. Rodriguez REG_WRITE(ah, AR_PHY_BIN_MASK_2, tmp_mask);
3578fe65368SLuis R. Rodriguez REG_WRITE(ah, AR_PHY_MASK2_M_31_45, tmp_mask);
3588fe65368SLuis R. Rodriguez
3598fe65368SLuis R. Rodriguez tmp_mask = (mask_m[16] << 30) | (mask_m[16] << 28)
3608fe65368SLuis R. Rodriguez | (mask_m[18] << 26) | (mask_m[18] << 24)
3618fe65368SLuis R. Rodriguez | (mask_m[20] << 22) | (mask_m[20] << 20)
3628fe65368SLuis R. Rodriguez | (mask_m[22] << 18) | (mask_m[22] << 16)
3638fe65368SLuis R. Rodriguez | (mask_m[24] << 14) | (mask_m[24] << 12)
3648fe65368SLuis R. Rodriguez | (mask_m[25] << 10) | (mask_m[26] << 8)
3658fe65368SLuis R. Rodriguez | (mask_m[27] << 6) | (mask_m[28] << 4)
3668fe65368SLuis R. Rodriguez | (mask_m[29] << 2) | (mask_m[30] << 0);
3678fe65368SLuis R. Rodriguez REG_WRITE(ah, AR_PHY_BIN_MASK_3, tmp_mask);
3688fe65368SLuis R. Rodriguez REG_WRITE(ah, AR_PHY_MASK2_M_16_30, tmp_mask);
3698fe65368SLuis R. Rodriguez
3708fe65368SLuis R. Rodriguez tmp_mask = (mask_m[0] << 30) | (mask_m[1] << 28)
3718fe65368SLuis R. Rodriguez | (mask_m[2] << 26) | (mask_m[3] << 24)
3728fe65368SLuis R. Rodriguez | (mask_m[4] << 22) | (mask_m[5] << 20)
3738fe65368SLuis R. Rodriguez | (mask_m[6] << 18) | (mask_m[7] << 16)
3748fe65368SLuis R. Rodriguez | (mask_m[8] << 14) | (mask_m[9] << 12)
3758fe65368SLuis R. Rodriguez | (mask_m[10] << 10) | (mask_m[11] << 8)
3768fe65368SLuis R. Rodriguez | (mask_m[12] << 6) | (mask_m[13] << 4)
3778fe65368SLuis R. Rodriguez | (mask_m[14] << 2) | (mask_m[15] << 0);
3788fe65368SLuis R. Rodriguez REG_WRITE(ah, AR_PHY_MASK_CTL, tmp_mask);
3798fe65368SLuis R. Rodriguez REG_WRITE(ah, AR_PHY_MASK2_M_00_15, tmp_mask);
3808fe65368SLuis R. Rodriguez
3818fe65368SLuis R. Rodriguez tmp_mask = (mask_p[15] << 28)
3828fe65368SLuis R. Rodriguez | (mask_p[14] << 26) | (mask_p[13] << 24)
3838fe65368SLuis R. Rodriguez | (mask_p[12] << 22) | (mask_p[11] << 20)
3848fe65368SLuis R. Rodriguez | (mask_p[10] << 18) | (mask_p[9] << 16)
3858fe65368SLuis R. Rodriguez | (mask_p[8] << 14) | (mask_p[7] << 12)
3868fe65368SLuis R. Rodriguez | (mask_p[6] << 10) | (mask_p[5] << 8)
3878fe65368SLuis R. Rodriguez | (mask_p[4] << 6) | (mask_p[3] << 4)
3888fe65368SLuis R. Rodriguez | (mask_p[2] << 2) | (mask_p[1] << 0);
3898fe65368SLuis R. Rodriguez REG_WRITE(ah, AR_PHY_BIN_MASK2_1, tmp_mask);
3908fe65368SLuis R. Rodriguez REG_WRITE(ah, AR_PHY_MASK2_P_15_01, tmp_mask);
3918fe65368SLuis R. Rodriguez
3928fe65368SLuis R. Rodriguez tmp_mask = (mask_p[30] << 28)
3938fe65368SLuis R. Rodriguez | (mask_p[29] << 26) | (mask_p[28] << 24)
3948fe65368SLuis R. Rodriguez | (mask_p[27] << 22) | (mask_p[26] << 20)
3958fe65368SLuis R. Rodriguez | (mask_p[25] << 18) | (mask_p[24] << 16)
3968fe65368SLuis R. Rodriguez | (mask_p[23] << 14) | (mask_p[22] << 12)
3978fe65368SLuis R. Rodriguez | (mask_p[21] << 10) | (mask_p[20] << 8)
3988fe65368SLuis R. Rodriguez | (mask_p[19] << 6) | (mask_p[18] << 4)
3998fe65368SLuis R. Rodriguez | (mask_p[17] << 2) | (mask_p[16] << 0);
4008fe65368SLuis R. Rodriguez REG_WRITE(ah, AR_PHY_BIN_MASK2_2, tmp_mask);
4018fe65368SLuis R. Rodriguez REG_WRITE(ah, AR_PHY_MASK2_P_30_16, tmp_mask);
4028fe65368SLuis R. Rodriguez
4038fe65368SLuis R. Rodriguez tmp_mask = (mask_p[45] << 28)
4048fe65368SLuis R. Rodriguez | (mask_p[44] << 26) | (mask_p[43] << 24)
4058fe65368SLuis R. Rodriguez | (mask_p[42] << 22) | (mask_p[41] << 20)
4068fe65368SLuis R. Rodriguez | (mask_p[40] << 18) | (mask_p[39] << 16)
4078fe65368SLuis R. Rodriguez | (mask_p[38] << 14) | (mask_p[37] << 12)
4088fe65368SLuis R. Rodriguez | (mask_p[36] << 10) | (mask_p[35] << 8)
4098fe65368SLuis R. Rodriguez | (mask_p[34] << 6) | (mask_p[33] << 4)
4108fe65368SLuis R. Rodriguez | (mask_p[32] << 2) | (mask_p[31] << 0);
4118fe65368SLuis R. Rodriguez REG_WRITE(ah, AR_PHY_BIN_MASK2_3, tmp_mask);
4128fe65368SLuis R. Rodriguez REG_WRITE(ah, AR_PHY_MASK2_P_45_31, tmp_mask);
4138fe65368SLuis R. Rodriguez
4148fe65368SLuis R. Rodriguez tmp_mask = (mask_p[61] << 30) | (mask_p[60] << 28)
4158fe65368SLuis R. Rodriguez | (mask_p[59] << 26) | (mask_p[58] << 24)
4168fe65368SLuis R. Rodriguez | (mask_p[57] << 22) | (mask_p[56] << 20)
4178fe65368SLuis R. Rodriguez | (mask_p[55] << 18) | (mask_p[54] << 16)
4188fe65368SLuis R. Rodriguez | (mask_p[53] << 14) | (mask_p[52] << 12)
4198fe65368SLuis R. Rodriguez | (mask_p[51] << 10) | (mask_p[50] << 8)
4208fe65368SLuis R. Rodriguez | (mask_p[49] << 6) | (mask_p[48] << 4)
4218fe65368SLuis R. Rodriguez | (mask_p[47] << 2) | (mask_p[46] << 0);
4228fe65368SLuis R. Rodriguez REG_WRITE(ah, AR_PHY_BIN_MASK2_4, tmp_mask);
4238fe65368SLuis R. Rodriguez REG_WRITE(ah, AR_PHY_MASK2_P_61_45, tmp_mask);
4248fe65368SLuis R. Rodriguez }
4258fe65368SLuis R. Rodriguez
4268cc107b5SLee Jones /*
427f911085fSOleksij Rempel * ar5008_hw_spur_mitigate - convert baseband spur frequency for external radios
428f911085fSOleksij Rempel *
429f911085fSOleksij Rempel * For non single-chip solutions. Converts to baseband spur frequency given the
430f911085fSOleksij Rempel * input channel frequency and compute register settings below.
431f911085fSOleksij Rempel */
ar5008_hw_spur_mitigate(struct ath_hw * ah,struct ath9k_channel * chan)432f911085fSOleksij Rempel static void ar5008_hw_spur_mitigate(struct ath_hw *ah,
433f911085fSOleksij Rempel struct ath9k_channel *chan)
434f911085fSOleksij Rempel {
435f911085fSOleksij Rempel int bb_spur = AR_NO_SPUR;
436f911085fSOleksij Rempel int bin;
437f911085fSOleksij Rempel int spur_freq_sd;
438f911085fSOleksij Rempel int spur_delta_phase;
439f911085fSOleksij Rempel int denominator;
440f911085fSOleksij Rempel int tmp, new;
441f911085fSOleksij Rempel int i;
442f911085fSOleksij Rempel
443f911085fSOleksij Rempel int cur_bb_spur;
444f911085fSOleksij Rempel bool is2GHz = IS_CHAN_2GHZ(chan);
445f911085fSOleksij Rempel
446f911085fSOleksij Rempel for (i = 0; i < AR_EEPROM_MODAL_SPURS; i++) {
447f911085fSOleksij Rempel cur_bb_spur = ah->eep_ops->get_spur_channel(ah, i, is2GHz);
448f911085fSOleksij Rempel if (AR_NO_SPUR == cur_bb_spur)
449f911085fSOleksij Rempel break;
450f911085fSOleksij Rempel cur_bb_spur = cur_bb_spur - (chan->channel * 10);
451f911085fSOleksij Rempel if ((cur_bb_spur > -95) && (cur_bb_spur < 95)) {
452f911085fSOleksij Rempel bb_spur = cur_bb_spur;
453f911085fSOleksij Rempel break;
454f911085fSOleksij Rempel }
455f911085fSOleksij Rempel }
456f911085fSOleksij Rempel
457f911085fSOleksij Rempel if (AR_NO_SPUR == bb_spur)
458f911085fSOleksij Rempel return;
459f911085fSOleksij Rempel
460f911085fSOleksij Rempel bin = bb_spur * 32;
461f911085fSOleksij Rempel
462f911085fSOleksij Rempel tmp = REG_READ(ah, AR_PHY_TIMING_CTRL4(0));
463f911085fSOleksij Rempel new = tmp | (AR_PHY_TIMING_CTRL4_ENABLE_SPUR_RSSI |
464f911085fSOleksij Rempel AR_PHY_TIMING_CTRL4_ENABLE_SPUR_FILTER |
465f911085fSOleksij Rempel AR_PHY_TIMING_CTRL4_ENABLE_CHAN_MASK |
466f911085fSOleksij Rempel AR_PHY_TIMING_CTRL4_ENABLE_PILOT_MASK);
467f911085fSOleksij Rempel
468f911085fSOleksij Rempel REG_WRITE(ah, AR_PHY_TIMING_CTRL4(0), new);
469f911085fSOleksij Rempel
470f911085fSOleksij Rempel new = (AR_PHY_SPUR_REG_MASK_RATE_CNTL |
471f911085fSOleksij Rempel AR_PHY_SPUR_REG_ENABLE_MASK_PPM |
472f911085fSOleksij Rempel AR_PHY_SPUR_REG_MASK_RATE_SELECT |
473f911085fSOleksij Rempel AR_PHY_SPUR_REG_ENABLE_VIT_SPUR_RSSI |
474f911085fSOleksij Rempel SM(SPUR_RSSI_THRESH, AR_PHY_SPUR_REG_SPUR_RSSI_THRESH));
475f911085fSOleksij Rempel REG_WRITE(ah, AR_PHY_SPUR_REG, new);
476f911085fSOleksij Rempel
477f911085fSOleksij Rempel spur_delta_phase = ((bb_spur * 524288) / 100) &
478f911085fSOleksij Rempel AR_PHY_TIMING11_SPUR_DELTA_PHASE;
479f911085fSOleksij Rempel
480f911085fSOleksij Rempel denominator = IS_CHAN_2GHZ(chan) ? 440 : 400;
481f911085fSOleksij Rempel spur_freq_sd = ((bb_spur * 2048) / denominator) & 0x3ff;
482f911085fSOleksij Rempel
483f911085fSOleksij Rempel new = (AR_PHY_TIMING11_USE_SPUR_IN_AGC |
484f911085fSOleksij Rempel SM(spur_freq_sd, AR_PHY_TIMING11_SPUR_FREQ_SD) |
485f911085fSOleksij Rempel SM(spur_delta_phase, AR_PHY_TIMING11_SPUR_DELTA_PHASE));
486f911085fSOleksij Rempel REG_WRITE(ah, AR_PHY_TIMING11, new);
487f911085fSOleksij Rempel
488f911085fSOleksij Rempel ar5008_hw_cmn_spur_mitigate(ah, chan, bin);
489f911085fSOleksij Rempel }
490f911085fSOleksij Rempel
491f911085fSOleksij Rempel /**
4928fe65368SLuis R. Rodriguez * ar5008_hw_rf_alloc_ext_banks - allocates banks for external radio programming
4938fe65368SLuis R. Rodriguez * @ah: atheros hardware structure
4948fe65368SLuis R. Rodriguez *
4958fe65368SLuis R. Rodriguez * Only required for older devices with external AR2133/AR5133 radios.
4968fe65368SLuis R. Rodriguez */
ar5008_hw_rf_alloc_ext_banks(struct ath_hw * ah)4978fe65368SLuis R. Rodriguez static int ar5008_hw_rf_alloc_ext_banks(struct ath_hw *ah)
4988fe65368SLuis R. Rodriguez {
499a043dfb9SFelix Fietkau int size = ah->iniBank6.ia_rows * sizeof(u32);
5008fe65368SLuis R. Rodriguez
501c1b976d2SFelix Fietkau if (AR_SREV_9280_20_OR_LATER(ah))
502c1b976d2SFelix Fietkau return 0;
5038fe65368SLuis R. Rodriguez
504a043dfb9SFelix Fietkau ah->analogBank6Data = devm_kzalloc(ah->dev, size, GFP_KERNEL);
505a043dfb9SFelix Fietkau if (!ah->analogBank6Data)
506a043dfb9SFelix Fietkau return -ENOMEM;
5078fe65368SLuis R. Rodriguez
5088fe65368SLuis R. Rodriguez return 0;
5098fe65368SLuis R. Rodriguez }
5108fe65368SLuis R. Rodriguez
5118fe65368SLuis R. Rodriguez
5128fe65368SLuis R. Rodriguez /* *
5138fe65368SLuis R. Rodriguez * ar5008_hw_set_rf_regs - programs rf registers based on EEPROM
5148fe65368SLuis R. Rodriguez * @ah: atheros hardware structure
5158fe65368SLuis R. Rodriguez * @chan:
5168fe65368SLuis R. Rodriguez * @modesIndex:
5178fe65368SLuis R. Rodriguez *
5188fe65368SLuis R. Rodriguez * Used for the external AR2133/AR5133 radios.
5198fe65368SLuis R. Rodriguez *
5208fe65368SLuis R. Rodriguez * Reads the EEPROM header info from the device structure and programs
5218fe65368SLuis R. Rodriguez * all rf registers. This routine requires access to the analog
5228fe65368SLuis R. Rodriguez * rf device. This is not required for single-chip devices.
5238fe65368SLuis R. Rodriguez */
ar5008_hw_set_rf_regs(struct ath_hw * ah,struct ath9k_channel * chan,u16 modesIndex)5248fe65368SLuis R. Rodriguez static bool ar5008_hw_set_rf_regs(struct ath_hw *ah,
5258fe65368SLuis R. Rodriguez struct ath9k_channel *chan,
5268fe65368SLuis R. Rodriguez u16 modesIndex)
5278fe65368SLuis R. Rodriguez {
5288fe65368SLuis R. Rodriguez u32 eepMinorRev;
5298fe65368SLuis R. Rodriguez u32 ob5GHz = 0, db5GHz = 0;
5308fe65368SLuis R. Rodriguez u32 ob2GHz = 0, db2GHz = 0;
5318fe65368SLuis R. Rodriguez int regWrites = 0;
53237c62fecSFelix Fietkau int i;
5338fe65368SLuis R. Rodriguez
5348fe65368SLuis R. Rodriguez /*
5358fe65368SLuis R. Rodriguez * Software does not need to program bank data
5368fe65368SLuis R. Rodriguez * for single chip devices, that is AR9280 or anything
5378fe65368SLuis R. Rodriguez * after that.
5388fe65368SLuis R. Rodriguez */
5397a37081eSFelix Fietkau if (AR_SREV_9280_20_OR_LATER(ah))
5408fe65368SLuis R. Rodriguez return true;
5418fe65368SLuis R. Rodriguez
5428fe65368SLuis R. Rodriguez /* Setup rf parameters */
5437d7dc538SMartin Blumenstingl eepMinorRev = ah->eep_ops->get_eeprom_rev(ah);
5448fe65368SLuis R. Rodriguez
54537c62fecSFelix Fietkau for (i = 0; i < ah->iniBank6.ia_rows; i++)
54637c62fecSFelix Fietkau ah->analogBank6Data[i] = INI_RA(&ah->iniBank6, i, modesIndex);
5478fe65368SLuis R. Rodriguez
5488fe65368SLuis R. Rodriguez /* Only the 5 or 2 GHz OB/DB need to be set for a mode */
5498fe65368SLuis R. Rodriguez if (eepMinorRev >= 2) {
5508fe65368SLuis R. Rodriguez if (IS_CHAN_2GHZ(chan)) {
5518fe65368SLuis R. Rodriguez ob2GHz = ah->eep_ops->get_eeprom(ah, EEP_OB_2);
5528fe65368SLuis R. Rodriguez db2GHz = ah->eep_ops->get_eeprom(ah, EEP_DB_2);
5538fe65368SLuis R. Rodriguez ar5008_hw_phy_modify_rx_buffer(ah->analogBank6Data,
5548fe65368SLuis R. Rodriguez ob2GHz, 3, 197, 0);
5558fe65368SLuis R. Rodriguez ar5008_hw_phy_modify_rx_buffer(ah->analogBank6Data,
5568fe65368SLuis R. Rodriguez db2GHz, 3, 194, 0);
5578fe65368SLuis R. Rodriguez } else {
5588fe65368SLuis R. Rodriguez ob5GHz = ah->eep_ops->get_eeprom(ah, EEP_OB_5);
5598fe65368SLuis R. Rodriguez db5GHz = ah->eep_ops->get_eeprom(ah, EEP_DB_5);
5608fe65368SLuis R. Rodriguez ar5008_hw_phy_modify_rx_buffer(ah->analogBank6Data,
5618fe65368SLuis R. Rodriguez ob5GHz, 3, 203, 0);
5628fe65368SLuis R. Rodriguez ar5008_hw_phy_modify_rx_buffer(ah->analogBank6Data,
5638fe65368SLuis R. Rodriguez db5GHz, 3, 200, 0);
5648fe65368SLuis R. Rodriguez }
5658fe65368SLuis R. Rodriguez }
5668fe65368SLuis R. Rodriguez
5678fe65368SLuis R. Rodriguez /* Write Analog registers */
568a043dfb9SFelix Fietkau REG_WRITE_ARRAY(&bank0, 1, regWrites);
569a043dfb9SFelix Fietkau REG_WRITE_ARRAY(&bank1, 1, regWrites);
570a043dfb9SFelix Fietkau REG_WRITE_ARRAY(&bank2, 1, regWrites);
571a043dfb9SFelix Fietkau REG_WRITE_ARRAY(&bank3, modesIndex, regWrites);
572a043dfb9SFelix Fietkau ar5008_write_bank6(ah, ®Writes);
573a043dfb9SFelix Fietkau REG_WRITE_ARRAY(&bank7, 1, regWrites);
5748fe65368SLuis R. Rodriguez
5758fe65368SLuis R. Rodriguez return true;
5768fe65368SLuis R. Rodriguez }
5778fe65368SLuis R. Rodriguez
ar5008_hw_init_bb(struct ath_hw * ah,struct ath9k_channel * chan)5788fe65368SLuis R. Rodriguez static void ar5008_hw_init_bb(struct ath_hw *ah,
5798fe65368SLuis R. Rodriguez struct ath9k_channel *chan)
5808fe65368SLuis R. Rodriguez {
5818fe65368SLuis R. Rodriguez u32 synthDelay;
5828fe65368SLuis R. Rodriguez
5838fe65368SLuis R. Rodriguez synthDelay = REG_READ(ah, AR_PHY_RX_DELAY) & AR_PHY_RX_DELAY_DELAY;
5847d865c70SFelix Fietkau
5858fe65368SLuis R. Rodriguez REG_WRITE(ah, AR_PHY_ACTIVE, AR_PHY_ACTIVE_EN);
5868fe65368SLuis R. Rodriguez
5877c5adc8dSFelix Fietkau ath9k_hw_synth_delay(ah, chan, synthDelay);
5888fe65368SLuis R. Rodriguez }
5898fe65368SLuis R. Rodriguez
ar5008_hw_init_chain_masks(struct ath_hw * ah)5908fe65368SLuis R. Rodriguez static void ar5008_hw_init_chain_masks(struct ath_hw *ah)
5918fe65368SLuis R. Rodriguez {
5928fe65368SLuis R. Rodriguez int rx_chainmask, tx_chainmask;
5938fe65368SLuis R. Rodriguez
5948fe65368SLuis R. Rodriguez rx_chainmask = ah->rxchainmask;
5958fe65368SLuis R. Rodriguez tx_chainmask = ah->txchainmask;
5968fe65368SLuis R. Rodriguez
5977d0d0df0SSujith
5988fe65368SLuis R. Rodriguez switch (rx_chainmask) {
5998fe65368SLuis R. Rodriguez case 0x5:
6008fe65368SLuis R. Rodriguez REG_SET_BIT(ah, AR_PHY_ANALOG_SWAP,
6018fe65368SLuis R. Rodriguez AR_PHY_SWAP_ALT_CHAIN);
602221af813SGustavo A. R. Silva fallthrough;
6038fe65368SLuis R. Rodriguez case 0x3:
6048fe65368SLuis R. Rodriguez if (ah->hw_version.macVersion == AR_SREV_REVISION_5416_10) {
6058fe65368SLuis R. Rodriguez REG_WRITE(ah, AR_PHY_RX_CHAINMASK, 0x7);
6068fe65368SLuis R. Rodriguez REG_WRITE(ah, AR_PHY_CAL_CHAINMASK, 0x7);
6078fe65368SLuis R. Rodriguez break;
6088fe65368SLuis R. Rodriguez }
609221af813SGustavo A. R. Silva fallthrough;
6108fe65368SLuis R. Rodriguez case 0x1:
6118fe65368SLuis R. Rodriguez case 0x2:
6128fe65368SLuis R. Rodriguez case 0x7:
613435c1610SFelix Fietkau ENABLE_REGWRITE_BUFFER(ah);
6148fe65368SLuis R. Rodriguez REG_WRITE(ah, AR_PHY_RX_CHAINMASK, rx_chainmask);
6158fe65368SLuis R. Rodriguez REG_WRITE(ah, AR_PHY_CAL_CHAINMASK, rx_chainmask);
6168fe65368SLuis R. Rodriguez break;
6178fe65368SLuis R. Rodriguez default:
618435c1610SFelix Fietkau ENABLE_REGWRITE_BUFFER(ah);
6198fe65368SLuis R. Rodriguez break;
6208fe65368SLuis R. Rodriguez }
6218fe65368SLuis R. Rodriguez
6228fe65368SLuis R. Rodriguez REG_WRITE(ah, AR_SELFGEN_MASK, tx_chainmask);
6237d0d0df0SSujith
6247d0d0df0SSujith REGWRITE_BUFFER_FLUSH(ah);
6257d0d0df0SSujith
6268fe65368SLuis R. Rodriguez if (tx_chainmask == 0x5) {
6278fe65368SLuis R. Rodriguez REG_SET_BIT(ah, AR_PHY_ANALOG_SWAP,
6288fe65368SLuis R. Rodriguez AR_PHY_SWAP_ALT_CHAIN);
6298fe65368SLuis R. Rodriguez }
6308fe65368SLuis R. Rodriguez if (AR_SREV_9100(ah))
6318fe65368SLuis R. Rodriguez REG_WRITE(ah, AR_PHY_ANALOG_SWAP,
6328fe65368SLuis R. Rodriguez REG_READ(ah, AR_PHY_ANALOG_SWAP) | 0x00000001);
6338fe65368SLuis R. Rodriguez }
6348fe65368SLuis R. Rodriguez
ar5008_hw_override_ini(struct ath_hw * ah,struct ath9k_channel * chan)6358fe65368SLuis R. Rodriguez static void ar5008_hw_override_ini(struct ath_hw *ah,
6368fe65368SLuis R. Rodriguez struct ath9k_channel *chan)
6378fe65368SLuis R. Rodriguez {
6388fe65368SLuis R. Rodriguez u32 val;
6398fe65368SLuis R. Rodriguez
6408fe65368SLuis R. Rodriguez /*
6418fe65368SLuis R. Rodriguez * Set the RX_ABORT and RX_DIS and clear if off only after
6428fe65368SLuis R. Rodriguez * RXE is set for MAC. This prevents frames with corrupted
6438fe65368SLuis R. Rodriguez * descriptor status.
6448fe65368SLuis R. Rodriguez */
6458fe65368SLuis R. Rodriguez REG_SET_BIT(ah, AR_DIAG_SW, (AR_DIAG_RX_DIS | AR_DIAG_RX_ABORT));
6468fe65368SLuis R. Rodriguez
6477a37081eSFelix Fietkau if (AR_SREV_9280_20_OR_LATER(ah)) {
64864b6f46fSSujith Manoharan /*
64964b6f46fSSujith Manoharan * For AR9280 and above, there is a new feature that allows
65064b6f46fSSujith Manoharan * Multicast search based on both MAC Address and Key ID.
65164b6f46fSSujith Manoharan * By default, this feature is enabled. But since the driver
65264b6f46fSSujith Manoharan * is not using this feature, we switch it off; otherwise
65364b6f46fSSujith Manoharan * multicast search based on MAC addr only will fail.
65464b6f46fSSujith Manoharan */
65564b6f46fSSujith Manoharan val = REG_READ(ah, AR_PCU_MISC_MODE2) &
65664b6f46fSSujith Manoharan (~AR_ADHOC_MCAST_KEYID_ENABLE);
6578fe65368SLuis R. Rodriguez
6588fe65368SLuis R. Rodriguez if (!AR_SREV_9271(ah))
6598fe65368SLuis R. Rodriguez val &= ~AR_PCU_MISC_MODE2_HWWAR1;
6608fe65368SLuis R. Rodriguez
661a42acef0SFelix Fietkau if (AR_SREV_9287_11_OR_LATER(ah))
6628fe65368SLuis R. Rodriguez val = val & (~AR_PCU_MISC_MODE2_HWWAR2);
6638fe65368SLuis R. Rodriguez
6649ef48932SSujith Manoharan val |= AR_PCU_MISC_MODE2_CFP_IGNORE;
6659ef48932SSujith Manoharan
6668fe65368SLuis R. Rodriguez REG_WRITE(ah, AR_PCU_MISC_MODE2, val);
6678fe65368SLuis R. Rodriguez }
6688fe65368SLuis R. Rodriguez
6691b8714f7SFelix Fietkau if (AR_SREV_9280_20_OR_LATER(ah))
6708fe65368SLuis R. Rodriguez return;
6718fe65368SLuis R. Rodriguez /*
6728fe65368SLuis R. Rodriguez * Disable BB clock gating
6738fe65368SLuis R. Rodriguez * Necessary to avoid issues on AR5416 2.0
6748fe65368SLuis R. Rodriguez */
6758fe65368SLuis R. Rodriguez REG_WRITE(ah, 0x9800 + (651 << 2), 0x11);
6768fe65368SLuis R. Rodriguez
6778fe65368SLuis R. Rodriguez /*
6788fe65368SLuis R. Rodriguez * Disable RIFS search on some chips to avoid baseband
6798fe65368SLuis R. Rodriguez * hang issues.
6808fe65368SLuis R. Rodriguez */
6818fe65368SLuis R. Rodriguez if (AR_SREV_9100(ah) || AR_SREV_9160(ah)) {
6828fe65368SLuis R. Rodriguez val = REG_READ(ah, AR_PHY_HEAVY_CLIP_FACTOR_RIFS);
6838fe65368SLuis R. Rodriguez val &= ~AR_PHY_RIFS_INIT_DELAY;
6848fe65368SLuis R. Rodriguez REG_WRITE(ah, AR_PHY_HEAVY_CLIP_FACTOR_RIFS, val);
6858fe65368SLuis R. Rodriguez }
6868fe65368SLuis R. Rodriguez }
6878fe65368SLuis R. Rodriguez
ar5008_hw_set_channel_regs(struct ath_hw * ah,struct ath9k_channel * chan)6888fe65368SLuis R. Rodriguez static void ar5008_hw_set_channel_regs(struct ath_hw *ah,
6898fe65368SLuis R. Rodriguez struct ath9k_channel *chan)
6908fe65368SLuis R. Rodriguez {
6918fe65368SLuis R. Rodriguez u32 phymode;
6928fe65368SLuis R. Rodriguez u32 enableDacFifo = 0;
6938fe65368SLuis R. Rodriguez
694e17f83eaSFelix Fietkau if (AR_SREV_9285_12_OR_LATER(ah))
6958fe65368SLuis R. Rodriguez enableDacFifo = (REG_READ(ah, AR_PHY_TURBO) &
6968fe65368SLuis R. Rodriguez AR_PHY_FC_ENABLE_DAC_FIFO);
6978fe65368SLuis R. Rodriguez
6988fe65368SLuis R. Rodriguez phymode = AR_PHY_FC_HT_EN | AR_PHY_FC_SHORT_GI_40
6998fe65368SLuis R. Rodriguez | AR_PHY_FC_SINGLE_HT_LTF1 | AR_PHY_FC_WALSH | enableDacFifo;
7008fe65368SLuis R. Rodriguez
7018fe65368SLuis R. Rodriguez if (IS_CHAN_HT40(chan)) {
7028fe65368SLuis R. Rodriguez phymode |= AR_PHY_FC_DYN2040_EN;
7038fe65368SLuis R. Rodriguez
7048896934cSFelix Fietkau if (IS_CHAN_HT40PLUS(chan))
7058fe65368SLuis R. Rodriguez phymode |= AR_PHY_FC_DYN2040_PRI_CH;
7068fe65368SLuis R. Rodriguez
7078fe65368SLuis R. Rodriguez }
7083df5dd90SOleksij Rempel ENABLE_REGWRITE_BUFFER(ah);
7098fe65368SLuis R. Rodriguez REG_WRITE(ah, AR_PHY_TURBO, phymode);
7108fe65368SLuis R. Rodriguez
7113df5dd90SOleksij Rempel /* This function do only REG_WRITE, so
7123df5dd90SOleksij Rempel * we can include it to REGWRITE_BUFFER. */
713e4744ec7SFelix Fietkau ath9k_hw_set11nmac2040(ah, chan);
7148fe65368SLuis R. Rodriguez
7158fe65368SLuis R. Rodriguez REG_WRITE(ah, AR_GTXTO, 25 << AR_GTXTO_TIMEOUT_LIMIT_S);
7168fe65368SLuis R. Rodriguez REG_WRITE(ah, AR_CST, 0xF << AR_CST_TIMEOUT_LIMIT_S);
7177d0d0df0SSujith
7187d0d0df0SSujith REGWRITE_BUFFER_FLUSH(ah);
7198fe65368SLuis R. Rodriguez }
7208fe65368SLuis R. Rodriguez
7218fe65368SLuis R. Rodriguez
ar5008_hw_process_ini(struct ath_hw * ah,struct ath9k_channel * chan)7228fe65368SLuis R. Rodriguez static int ar5008_hw_process_ini(struct ath_hw *ah,
7238fe65368SLuis R. Rodriguez struct ath9k_channel *chan)
7248fe65368SLuis R. Rodriguez {
725e7fc6338SRajkumar Manoharan struct ath_common *common = ath9k_hw_common(ah);
7268fe65368SLuis R. Rodriguez int i, regWrites = 0;
7278fe65368SLuis R. Rodriguez u32 modesIndex, freqIndex;
7288fe65368SLuis R. Rodriguez
7298896934cSFelix Fietkau if (IS_CHAN_5GHZ(chan)) {
7308fe65368SLuis R. Rodriguez freqIndex = 1;
7318896934cSFelix Fietkau modesIndex = IS_CHAN_HT40(chan) ? 2 : 1;
7328896934cSFelix Fietkau } else {
7338fe65368SLuis R. Rodriguez freqIndex = 2;
7348896934cSFelix Fietkau modesIndex = IS_CHAN_HT40(chan) ? 3 : 4;
7358fe65368SLuis R. Rodriguez }
7368fe65368SLuis R. Rodriguez
7378fe65368SLuis R. Rodriguez /*
7388fe65368SLuis R. Rodriguez * Set correct baseband to analog shift setting to
7398fe65368SLuis R. Rodriguez * access analog chips.
7408fe65368SLuis R. Rodriguez */
7418fe65368SLuis R. Rodriguez REG_WRITE(ah, AR_PHY(0), 0x00000007);
7428fe65368SLuis R. Rodriguez
7438fe65368SLuis R. Rodriguez /* Write ADDAC shifts */
7448fe65368SLuis R. Rodriguez REG_WRITE(ah, AR_PHY_ADC_SERIAL_CTL, AR_PHY_SEL_EXTERNAL_RADIO);
745d7084da0SFelix Fietkau if (ah->eep_ops->set_addac)
7468fe65368SLuis R. Rodriguez ah->eep_ops->set_addac(ah, chan);
7478fe65368SLuis R. Rodriguez
7488fe65368SLuis R. Rodriguez REG_WRITE_ARRAY(&ah->iniAddac, 1, regWrites);
7498fe65368SLuis R. Rodriguez REG_WRITE(ah, AR_PHY_ADC_SERIAL_CTL, AR_PHY_SEL_INTERNAL_ADDAC);
7508fe65368SLuis R. Rodriguez
7517d0d0df0SSujith ENABLE_REGWRITE_BUFFER(ah);
7527d0d0df0SSujith
7538fe65368SLuis R. Rodriguez for (i = 0; i < ah->iniModes.ia_rows; i++) {
7548fe65368SLuis R. Rodriguez u32 reg = INI_RA(&ah->iniModes, i, 0);
7558fe65368SLuis R. Rodriguez u32 val = INI_RA(&ah->iniModes, i, modesIndex);
7568fe65368SLuis R. Rodriguez
7578fe65368SLuis R. Rodriguez if (reg == AR_AN_TOP2 && ah->need_an_top2_fixup)
7588fe65368SLuis R. Rodriguez val &= ~AR_AN_TOP2_PWDCLKIND;
7598fe65368SLuis R. Rodriguez
7608fe65368SLuis R. Rodriguez REG_WRITE(ah, reg, val);
7618fe65368SLuis R. Rodriguez
7628fe65368SLuis R. Rodriguez if (reg >= 0x7800 && reg < 0x78a0
763e7fc6338SRajkumar Manoharan && ah->config.analog_shiftreg
764e7fc6338SRajkumar Manoharan && (common->bus_ops->ath_bus_type != ATH_USB)) {
7658fe65368SLuis R. Rodriguez udelay(100);
7668fe65368SLuis R. Rodriguez }
7678fe65368SLuis R. Rodriguez
7688fe65368SLuis R. Rodriguez DO_DELAY(regWrites);
7698fe65368SLuis R. Rodriguez }
7708fe65368SLuis R. Rodriguez
7717d0d0df0SSujith REGWRITE_BUFFER_FLUSH(ah);
7727d0d0df0SSujith
773a42acef0SFelix Fietkau if (AR_SREV_9280(ah) || AR_SREV_9287_11_OR_LATER(ah))
7748fe65368SLuis R. Rodriguez REG_WRITE_ARRAY(&ah->iniModesRxGain, modesIndex, regWrites);
7758fe65368SLuis R. Rodriguez
7768fe65368SLuis R. Rodriguez if (AR_SREV_9280(ah) || AR_SREV_9285_12_OR_LATER(ah) ||
777a42acef0SFelix Fietkau AR_SREV_9287_11_OR_LATER(ah))
7788fe65368SLuis R. Rodriguez REG_WRITE_ARRAY(&ah->iniModesTxGain, modesIndex, regWrites);
7798fe65368SLuis R. Rodriguez
780c7effd35SFelix Fietkau if (AR_SREV_9271_10(ah)) {
781c7effd35SFelix Fietkau REG_SET_BIT(ah, AR_PHY_SPECTRAL_SCAN, AR_PHY_SPECTRAL_SCAN_ENA);
782c7effd35SFelix Fietkau REG_RMW_FIELD(ah, AR_PHY_RF_CTL3, AR_PHY_TX_END_TO_ADC_ON, 0xa);
783c7effd35SFelix Fietkau }
7848fe65368SLuis R. Rodriguez
7857d0d0df0SSujith ENABLE_REGWRITE_BUFFER(ah);
7867d0d0df0SSujith
7878fe65368SLuis R. Rodriguez /* Write common array parameters */
7888fe65368SLuis R. Rodriguez for (i = 0; i < ah->iniCommon.ia_rows; i++) {
7898fe65368SLuis R. Rodriguez u32 reg = INI_RA(&ah->iniCommon, i, 0);
7908fe65368SLuis R. Rodriguez u32 val = INI_RA(&ah->iniCommon, i, 1);
7918fe65368SLuis R. Rodriguez
7928fe65368SLuis R. Rodriguez REG_WRITE(ah, reg, val);
7938fe65368SLuis R. Rodriguez
7948fe65368SLuis R. Rodriguez if (reg >= 0x7800 && reg < 0x78a0
795e7fc6338SRajkumar Manoharan && ah->config.analog_shiftreg
796e7fc6338SRajkumar Manoharan && (common->bus_ops->ath_bus_type != ATH_USB)) {
7978fe65368SLuis R. Rodriguez udelay(100);
7988fe65368SLuis R. Rodriguez }
7998fe65368SLuis R. Rodriguez
8008fe65368SLuis R. Rodriguez DO_DELAY(regWrites);
8018fe65368SLuis R. Rodriguez }
8028fe65368SLuis R. Rodriguez
8037d0d0df0SSujith REGWRITE_BUFFER_FLUSH(ah);
8047d0d0df0SSujith
8058fe65368SLuis R. Rodriguez REG_WRITE_ARRAY(&ah->iniBB_RfGain, freqIndex, regWrites);
8068fe65368SLuis R. Rodriguez
807c7d36f9fSFelix Fietkau if (IS_CHAN_A_FAST_CLOCK(ah, chan))
808c7d36f9fSFelix Fietkau REG_WRITE_ARRAY(&ah->iniModesFastClock, modesIndex,
8098fe65368SLuis R. Rodriguez regWrites);
8108fe65368SLuis R. Rodriguez
8118fe65368SLuis R. Rodriguez ar5008_hw_override_ini(ah, chan);
8128fe65368SLuis R. Rodriguez ar5008_hw_set_channel_regs(ah, chan);
8138fe65368SLuis R. Rodriguez ar5008_hw_init_chain_masks(ah);
8148fe65368SLuis R. Rodriguez ath9k_olc_init(ah);
81564ea57d0SGabor Juhos ath9k_hw_apply_txpower(ah, chan, false);
8168fe65368SLuis R. Rodriguez
8178fe65368SLuis R. Rodriguez /* Write analog registers */
8188fe65368SLuis R. Rodriguez if (!ath9k_hw_set_rf_regs(ah, chan, freqIndex)) {
8193800276aSJoe Perches ath_err(ath9k_hw_common(ah), "ar5416SetRfRegs failed\n");
8208fe65368SLuis R. Rodriguez return -EIO;
8218fe65368SLuis R. Rodriguez }
8228fe65368SLuis R. Rodriguez
8238fe65368SLuis R. Rodriguez return 0;
8248fe65368SLuis R. Rodriguez }
8258fe65368SLuis R. Rodriguez
ar5008_hw_set_rfmode(struct ath_hw * ah,struct ath9k_channel * chan)8268fe65368SLuis R. Rodriguez static void ar5008_hw_set_rfmode(struct ath_hw *ah, struct ath9k_channel *chan)
8278fe65368SLuis R. Rodriguez {
8288fe65368SLuis R. Rodriguez u32 rfMode = 0;
8298fe65368SLuis R. Rodriguez
8308fe65368SLuis R. Rodriguez if (chan == NULL)
8318fe65368SLuis R. Rodriguez return;
8328fe65368SLuis R. Rodriguez
8331a5e6326SFelix Fietkau if (IS_CHAN_2GHZ(chan))
8341a5e6326SFelix Fietkau rfMode |= AR_PHY_MODE_DYNAMIC;
8351a5e6326SFelix Fietkau else
8361a5e6326SFelix Fietkau rfMode |= AR_PHY_MODE_OFDM;
8378fe65368SLuis R. Rodriguez
8387a37081eSFelix Fietkau if (!AR_SREV_9280_20_OR_LATER(ah))
8398fe65368SLuis R. Rodriguez rfMode |= (IS_CHAN_5GHZ(chan)) ?
8408fe65368SLuis R. Rodriguez AR_PHY_MODE_RF5GHZ : AR_PHY_MODE_RF2GHZ;
8418fe65368SLuis R. Rodriguez
8426b42e8d0SFelix Fietkau if (IS_CHAN_A_FAST_CLOCK(ah, chan))
8438fe65368SLuis R. Rodriguez rfMode |= (AR_PHY_MODE_DYNAMIC | AR_PHY_MODE_DYN_CCK_DISABLE);
8448fe65368SLuis R. Rodriguez
8458fe65368SLuis R. Rodriguez REG_WRITE(ah, AR_PHY_MODE, rfMode);
8468fe65368SLuis R. Rodriguez }
8478fe65368SLuis R. Rodriguez
ar5008_hw_mark_phy_inactive(struct ath_hw * ah)8488fe65368SLuis R. Rodriguez static void ar5008_hw_mark_phy_inactive(struct ath_hw *ah)
8498fe65368SLuis R. Rodriguez {
8508fe65368SLuis R. Rodriguez REG_WRITE(ah, AR_PHY_ACTIVE, AR_PHY_ACTIVE_DIS);
8518fe65368SLuis R. Rodriguez }
8528fe65368SLuis R. Rodriguez
ar5008_hw_set_delta_slope(struct ath_hw * ah,struct ath9k_channel * chan)8538fe65368SLuis R. Rodriguez static void ar5008_hw_set_delta_slope(struct ath_hw *ah,
8548fe65368SLuis R. Rodriguez struct ath9k_channel *chan)
8558fe65368SLuis R. Rodriguez {
8568fe65368SLuis R. Rodriguez u32 coef_scaled, ds_coef_exp, ds_coef_man;
8578fe65368SLuis R. Rodriguez u32 clockMhzScaled = 0x64000000;
8588fe65368SLuis R. Rodriguez struct chan_centers centers;
8598fe65368SLuis R. Rodriguez
8608fe65368SLuis R. Rodriguez if (IS_CHAN_HALF_RATE(chan))
8618fe65368SLuis R. Rodriguez clockMhzScaled = clockMhzScaled >> 1;
8628fe65368SLuis R. Rodriguez else if (IS_CHAN_QUARTER_RATE(chan))
8638fe65368SLuis R. Rodriguez clockMhzScaled = clockMhzScaled >> 2;
8648fe65368SLuis R. Rodriguez
8658fe65368SLuis R. Rodriguez ath9k_hw_get_channel_centers(ah, chan, ¢ers);
8668fe65368SLuis R. Rodriguez coef_scaled = clockMhzScaled / centers.synth_center;
8678fe65368SLuis R. Rodriguez
8688fe65368SLuis R. Rodriguez ath9k_hw_get_delta_slope_vals(ah, coef_scaled, &ds_coef_man,
8698fe65368SLuis R. Rodriguez &ds_coef_exp);
8708fe65368SLuis R. Rodriguez
8718fe65368SLuis R. Rodriguez REG_RMW_FIELD(ah, AR_PHY_TIMING3,
8728fe65368SLuis R. Rodriguez AR_PHY_TIMING3_DSC_MAN, ds_coef_man);
8738fe65368SLuis R. Rodriguez REG_RMW_FIELD(ah, AR_PHY_TIMING3,
8748fe65368SLuis R. Rodriguez AR_PHY_TIMING3_DSC_EXP, ds_coef_exp);
8758fe65368SLuis R. Rodriguez
8768fe65368SLuis R. Rodriguez coef_scaled = (9 * coef_scaled) / 10;
8778fe65368SLuis R. Rodriguez
8788fe65368SLuis R. Rodriguez ath9k_hw_get_delta_slope_vals(ah, coef_scaled, &ds_coef_man,
8798fe65368SLuis R. Rodriguez &ds_coef_exp);
8808fe65368SLuis R. Rodriguez
8818fe65368SLuis R. Rodriguez REG_RMW_FIELD(ah, AR_PHY_HALFGI,
8828fe65368SLuis R. Rodriguez AR_PHY_HALFGI_DSC_MAN, ds_coef_man);
8838fe65368SLuis R. Rodriguez REG_RMW_FIELD(ah, AR_PHY_HALFGI,
8848fe65368SLuis R. Rodriguez AR_PHY_HALFGI_DSC_EXP, ds_coef_exp);
8858fe65368SLuis R. Rodriguez }
8868fe65368SLuis R. Rodriguez
ar5008_hw_rfbus_req(struct ath_hw * ah)8878fe65368SLuis R. Rodriguez static bool ar5008_hw_rfbus_req(struct ath_hw *ah)
8888fe65368SLuis R. Rodriguez {
8898fe65368SLuis R. Rodriguez REG_WRITE(ah, AR_PHY_RFBUS_REQ, AR_PHY_RFBUS_REQ_EN);
8908fe65368SLuis R. Rodriguez return ath9k_hw_wait(ah, AR_PHY_RFBUS_GRANT, AR_PHY_RFBUS_GRANT_EN,
8918fe65368SLuis R. Rodriguez AR_PHY_RFBUS_GRANT_EN, AH_WAIT_TIMEOUT);
8928fe65368SLuis R. Rodriguez }
8938fe65368SLuis R. Rodriguez
ar5008_hw_rfbus_done(struct ath_hw * ah)8948fe65368SLuis R. Rodriguez static void ar5008_hw_rfbus_done(struct ath_hw *ah)
8958fe65368SLuis R. Rodriguez {
8968fe65368SLuis R. Rodriguez u32 synthDelay = REG_READ(ah, AR_PHY_RX_DELAY) & AR_PHY_RX_DELAY_DELAY;
8978fe65368SLuis R. Rodriguez
8987c5adc8dSFelix Fietkau ath9k_hw_synth_delay(ah, ah->curchan, synthDelay);
8998fe65368SLuis R. Rodriguez
9008fe65368SLuis R. Rodriguez REG_WRITE(ah, AR_PHY_RFBUS_REQ, 0);
9018fe65368SLuis R. Rodriguez }
9028fe65368SLuis R. Rodriguez
ar5008_restore_chainmask(struct ath_hw * ah)9038fe65368SLuis R. Rodriguez static void ar5008_restore_chainmask(struct ath_hw *ah)
9048fe65368SLuis R. Rodriguez {
9058fe65368SLuis R. Rodriguez int rx_chainmask = ah->rxchainmask;
9068fe65368SLuis R. Rodriguez
9078fe65368SLuis R. Rodriguez if ((rx_chainmask == 0x5) || (rx_chainmask == 0x3)) {
9088fe65368SLuis R. Rodriguez REG_WRITE(ah, AR_PHY_RX_CHAINMASK, rx_chainmask);
9098fe65368SLuis R. Rodriguez REG_WRITE(ah, AR_PHY_CAL_CHAINMASK, rx_chainmask);
9108fe65368SLuis R. Rodriguez }
9118fe65368SLuis R. Rodriguez }
9128fe65368SLuis R. Rodriguez
ar9160_hw_compute_pll_control(struct ath_hw * ah,struct ath9k_channel * chan)91364773964SLuis R. Rodriguez static u32 ar9160_hw_compute_pll_control(struct ath_hw *ah,
91464773964SLuis R. Rodriguez struct ath9k_channel *chan)
91564773964SLuis R. Rodriguez {
91664773964SLuis R. Rodriguez u32 pll;
91764773964SLuis R. Rodriguez
91864773964SLuis R. Rodriguez pll = SM(0x5, AR_RTC_9160_PLL_REFDIV);
91964773964SLuis R. Rodriguez
92064773964SLuis R. Rodriguez if (chan && IS_CHAN_HALF_RATE(chan))
92164773964SLuis R. Rodriguez pll |= SM(0x1, AR_RTC_9160_PLL_CLKSEL);
92264773964SLuis R. Rodriguez else if (chan && IS_CHAN_QUARTER_RATE(chan))
92364773964SLuis R. Rodriguez pll |= SM(0x2, AR_RTC_9160_PLL_CLKSEL);
92464773964SLuis R. Rodriguez
92564773964SLuis R. Rodriguez if (chan && IS_CHAN_5GHZ(chan))
92664773964SLuis R. Rodriguez pll |= SM(0x50, AR_RTC_9160_PLL_DIV);
92764773964SLuis R. Rodriguez else
92864773964SLuis R. Rodriguez pll |= SM(0x58, AR_RTC_9160_PLL_DIV);
92964773964SLuis R. Rodriguez
93064773964SLuis R. Rodriguez return pll;
93164773964SLuis R. Rodriguez }
93264773964SLuis R. Rodriguez
ar5008_hw_compute_pll_control(struct ath_hw * ah,struct ath9k_channel * chan)93364773964SLuis R. Rodriguez static u32 ar5008_hw_compute_pll_control(struct ath_hw *ah,
93464773964SLuis R. Rodriguez struct ath9k_channel *chan)
93564773964SLuis R. Rodriguez {
93664773964SLuis R. Rodriguez u32 pll;
93764773964SLuis R. Rodriguez
93864773964SLuis R. Rodriguez pll = AR_RTC_PLL_REFDIV_5 | AR_RTC_PLL_DIV2;
93964773964SLuis R. Rodriguez
94064773964SLuis R. Rodriguez if (chan && IS_CHAN_HALF_RATE(chan))
94164773964SLuis R. Rodriguez pll |= SM(0x1, AR_RTC_PLL_CLKSEL);
94264773964SLuis R. Rodriguez else if (chan && IS_CHAN_QUARTER_RATE(chan))
94364773964SLuis R. Rodriguez pll |= SM(0x2, AR_RTC_PLL_CLKSEL);
94464773964SLuis R. Rodriguez
94564773964SLuis R. Rodriguez if (chan && IS_CHAN_5GHZ(chan))
94664773964SLuis R. Rodriguez pll |= SM(0xa, AR_RTC_PLL_DIV);
94764773964SLuis R. Rodriguez else
94864773964SLuis R. Rodriguez pll |= SM(0xb, AR_RTC_PLL_DIV);
94964773964SLuis R. Rodriguez
95064773964SLuis R. Rodriguez return pll;
95164773964SLuis R. Rodriguez }
95264773964SLuis R. Rodriguez
ar5008_hw_ani_control_new(struct ath_hw * ah,enum ath9k_ani_cmd cmd,int param)953e36b27afSLuis R. Rodriguez static bool ar5008_hw_ani_control_new(struct ath_hw *ah,
954e36b27afSLuis R. Rodriguez enum ath9k_ani_cmd cmd,
955e36b27afSLuis R. Rodriguez int param)
956e36b27afSLuis R. Rodriguez {
957e36b27afSLuis R. Rodriguez struct ath_common *common = ath9k_hw_common(ah);
958e36b27afSLuis R. Rodriguez struct ath9k_channel *chan = ah->curchan;
959c24bd362SSujith Manoharan struct ar5416AniState *aniState = &ah->ani;
9609301ca90SFelix Fietkau s32 value;
961e36b27afSLuis R. Rodriguez
962e36b27afSLuis R. Rodriguez switch (cmd & ah->ani_function) {
963e36b27afSLuis R. Rodriguez case ATH9K_ANI_OFDM_WEAK_SIGNAL_DETECTION:{
964e36b27afSLuis R. Rodriguez /*
965e36b27afSLuis R. Rodriguez * on == 1 means ofdm weak signal detection is ON
966e36b27afSLuis R. Rodriguez * on == 1 is the default, for less noise immunity
967e36b27afSLuis R. Rodriguez *
968e36b27afSLuis R. Rodriguez * on == 0 means ofdm weak signal detection is OFF
969e36b27afSLuis R. Rodriguez * on == 0 means more noise imm
970e36b27afSLuis R. Rodriguez */
971e36b27afSLuis R. Rodriguez u32 on = param ? 1 : 0;
972e36b27afSLuis R. Rodriguez /*
973e36b27afSLuis R. Rodriguez * make register setting for default
974e36b27afSLuis R. Rodriguez * (weak sig detect ON) come from INI file
975e36b27afSLuis R. Rodriguez */
976e36b27afSLuis R. Rodriguez int m1ThreshLow = on ?
977e36b27afSLuis R. Rodriguez aniState->iniDef.m1ThreshLow : m1ThreshLow_off;
978e36b27afSLuis R. Rodriguez int m2ThreshLow = on ?
979e36b27afSLuis R. Rodriguez aniState->iniDef.m2ThreshLow : m2ThreshLow_off;
980e36b27afSLuis R. Rodriguez int m1Thresh = on ?
981e36b27afSLuis R. Rodriguez aniState->iniDef.m1Thresh : m1Thresh_off;
982e36b27afSLuis R. Rodriguez int m2Thresh = on ?
983e36b27afSLuis R. Rodriguez aniState->iniDef.m2Thresh : m2Thresh_off;
984e36b27afSLuis R. Rodriguez int m2CountThr = on ?
985e36b27afSLuis R. Rodriguez aniState->iniDef.m2CountThr : m2CountThr_off;
986e36b27afSLuis R. Rodriguez int m2CountThrLow = on ?
987e36b27afSLuis R. Rodriguez aniState->iniDef.m2CountThrLow : m2CountThrLow_off;
988e36b27afSLuis R. Rodriguez int m1ThreshLowExt = on ?
989e36b27afSLuis R. Rodriguez aniState->iniDef.m1ThreshLowExt : m1ThreshLowExt_off;
990e36b27afSLuis R. Rodriguez int m2ThreshLowExt = on ?
991e36b27afSLuis R. Rodriguez aniState->iniDef.m2ThreshLowExt : m2ThreshLowExt_off;
992e36b27afSLuis R. Rodriguez int m1ThreshExt = on ?
993e36b27afSLuis R. Rodriguez aniState->iniDef.m1ThreshExt : m1ThreshExt_off;
994e36b27afSLuis R. Rodriguez int m2ThreshExt = on ?
995e36b27afSLuis R. Rodriguez aniState->iniDef.m2ThreshExt : m2ThreshExt_off;
996e36b27afSLuis R. Rodriguez
997e36b27afSLuis R. Rodriguez REG_RMW_FIELD(ah, AR_PHY_SFCORR_LOW,
998e36b27afSLuis R. Rodriguez AR_PHY_SFCORR_LOW_M1_THRESH_LOW,
999e36b27afSLuis R. Rodriguez m1ThreshLow);
1000e36b27afSLuis R. Rodriguez REG_RMW_FIELD(ah, AR_PHY_SFCORR_LOW,
1001e36b27afSLuis R. Rodriguez AR_PHY_SFCORR_LOW_M2_THRESH_LOW,
1002e36b27afSLuis R. Rodriguez m2ThreshLow);
1003e36b27afSLuis R. Rodriguez REG_RMW_FIELD(ah, AR_PHY_SFCORR,
1004e36b27afSLuis R. Rodriguez AR_PHY_SFCORR_M1_THRESH, m1Thresh);
1005e36b27afSLuis R. Rodriguez REG_RMW_FIELD(ah, AR_PHY_SFCORR,
1006e36b27afSLuis R. Rodriguez AR_PHY_SFCORR_M2_THRESH, m2Thresh);
1007e36b27afSLuis R. Rodriguez REG_RMW_FIELD(ah, AR_PHY_SFCORR,
1008e36b27afSLuis R. Rodriguez AR_PHY_SFCORR_M2COUNT_THR, m2CountThr);
1009e36b27afSLuis R. Rodriguez REG_RMW_FIELD(ah, AR_PHY_SFCORR_LOW,
1010e36b27afSLuis R. Rodriguez AR_PHY_SFCORR_LOW_M2COUNT_THR_LOW,
1011e36b27afSLuis R. Rodriguez m2CountThrLow);
1012e36b27afSLuis R. Rodriguez
1013e36b27afSLuis R. Rodriguez REG_RMW_FIELD(ah, AR_PHY_SFCORR_EXT,
1014e36b27afSLuis R. Rodriguez AR_PHY_SFCORR_EXT_M1_THRESH_LOW, m1ThreshLowExt);
1015e36b27afSLuis R. Rodriguez REG_RMW_FIELD(ah, AR_PHY_SFCORR_EXT,
1016e36b27afSLuis R. Rodriguez AR_PHY_SFCORR_EXT_M2_THRESH_LOW, m2ThreshLowExt);
1017e36b27afSLuis R. Rodriguez REG_RMW_FIELD(ah, AR_PHY_SFCORR_EXT,
1018e36b27afSLuis R. Rodriguez AR_PHY_SFCORR_EXT_M1_THRESH, m1ThreshExt);
1019e36b27afSLuis R. Rodriguez REG_RMW_FIELD(ah, AR_PHY_SFCORR_EXT,
1020e36b27afSLuis R. Rodriguez AR_PHY_SFCORR_EXT_M2_THRESH, m2ThreshExt);
1021e36b27afSLuis R. Rodriguez
1022e36b27afSLuis R. Rodriguez if (on)
1023e36b27afSLuis R. Rodriguez REG_SET_BIT(ah, AR_PHY_SFCORR_LOW,
1024e36b27afSLuis R. Rodriguez AR_PHY_SFCORR_LOW_USE_SELF_CORR_LOW);
1025e36b27afSLuis R. Rodriguez else
1026e36b27afSLuis R. Rodriguez REG_CLR_BIT(ah, AR_PHY_SFCORR_LOW,
1027e36b27afSLuis R. Rodriguez AR_PHY_SFCORR_LOW_USE_SELF_CORR_LOW);
1028e36b27afSLuis R. Rodriguez
10297067e701SFelix Fietkau if (on != aniState->ofdmWeakSigDetect) {
1030d2182b69SJoe Perches ath_dbg(common, ANI,
1031e36b27afSLuis R. Rodriguez "** ch %d: ofdm weak signal: %s=>%s\n",
1032e36b27afSLuis R. Rodriguez chan->channel,
10337067e701SFelix Fietkau aniState->ofdmWeakSigDetect ?
1034e36b27afSLuis R. Rodriguez "on" : "off",
1035e36b27afSLuis R. Rodriguez on ? "on" : "off");
1036e36b27afSLuis R. Rodriguez if (on)
1037e36b27afSLuis R. Rodriguez ah->stats.ast_ani_ofdmon++;
1038e36b27afSLuis R. Rodriguez else
1039e36b27afSLuis R. Rodriguez ah->stats.ast_ani_ofdmoff++;
10407067e701SFelix Fietkau aniState->ofdmWeakSigDetect = on;
1041e36b27afSLuis R. Rodriguez }
1042e36b27afSLuis R. Rodriguez break;
1043e36b27afSLuis R. Rodriguez }
1044e36b27afSLuis R. Rodriguez case ATH9K_ANI_FIRSTEP_LEVEL:{
1045e36b27afSLuis R. Rodriguez u32 level = param;
1046e36b27afSLuis R. Rodriguez
1047171cdab8SFelix Fietkau value = level * 2;
1048e36b27afSLuis R. Rodriguez REG_RMW_FIELD(ah, AR_PHY_FIND_SIG,
10499301ca90SFelix Fietkau AR_PHY_FIND_SIG_FIRSTEP, value);
1050171cdab8SFelix Fietkau REG_RMW_FIELD(ah, AR_PHY_FIND_SIG_LOW,
1051171cdab8SFelix Fietkau AR_PHY_FIND_SIG_FIRSTEP_LOW, value);
1052e36b27afSLuis R. Rodriguez
1053e36b27afSLuis R. Rodriguez if (level != aniState->firstepLevel) {
1054d2182b69SJoe Perches ath_dbg(common, ANI,
1055226afe68SJoe Perches "** ch %d: level %d=>%d[def:%d] firstep[level]=%d ini=%d\n",
1056e36b27afSLuis R. Rodriguez chan->channel,
1057e36b27afSLuis R. Rodriguez aniState->firstepLevel,
1058e36b27afSLuis R. Rodriguez level,
1059465dce62SFelix Fietkau ATH9K_ANI_FIRSTEP_LVL,
1060e36b27afSLuis R. Rodriguez value,
1061e36b27afSLuis R. Rodriguez aniState->iniDef.firstep);
1062d2182b69SJoe Perches ath_dbg(common, ANI,
1063226afe68SJoe Perches "** ch %d: level %d=>%d[def:%d] firstep_low[level]=%d ini=%d\n",
1064e36b27afSLuis R. Rodriguez chan->channel,
1065e36b27afSLuis R. Rodriguez aniState->firstepLevel,
1066e36b27afSLuis R. Rodriguez level,
1067465dce62SFelix Fietkau ATH9K_ANI_FIRSTEP_LVL,
10689301ca90SFelix Fietkau value,
1069e36b27afSLuis R. Rodriguez aniState->iniDef.firstepLow);
1070e36b27afSLuis R. Rodriguez if (level > aniState->firstepLevel)
1071e36b27afSLuis R. Rodriguez ah->stats.ast_ani_stepup++;
1072e36b27afSLuis R. Rodriguez else if (level < aniState->firstepLevel)
1073e36b27afSLuis R. Rodriguez ah->stats.ast_ani_stepdown++;
1074e36b27afSLuis R. Rodriguez aniState->firstepLevel = level;
1075e36b27afSLuis R. Rodriguez }
1076e36b27afSLuis R. Rodriguez break;
1077e36b27afSLuis R. Rodriguez }
1078e36b27afSLuis R. Rodriguez case ATH9K_ANI_SPUR_IMMUNITY_LEVEL:{
1079e36b27afSLuis R. Rodriguez u32 level = param;
1080e36b27afSLuis R. Rodriguez
108128327fd0SFelix Fietkau value = (level + 1) * 2;
1082e36b27afSLuis R. Rodriguez REG_RMW_FIELD(ah, AR_PHY_TIMING5,
108328327fd0SFelix Fietkau AR_PHY_TIMING5_CYCPWR_THR1, value);
1084e36b27afSLuis R. Rodriguez
1085e36b27afSLuis R. Rodriguez REG_RMW_FIELD(ah, AR_PHY_EXT_CCA,
1086b874ec8dSFelix Fietkau AR_PHY_EXT_TIMING5_CYCPWR_THR1, value - 1);
1087e36b27afSLuis R. Rodriguez
1088e36b27afSLuis R. Rodriguez if (level != aniState->spurImmunityLevel) {
1089d2182b69SJoe Perches ath_dbg(common, ANI,
1090226afe68SJoe Perches "** ch %d: level %d=>%d[def:%d] cycpwrThr1[level]=%d ini=%d\n",
1091e36b27afSLuis R. Rodriguez chan->channel,
1092e36b27afSLuis R. Rodriguez aniState->spurImmunityLevel,
1093e36b27afSLuis R. Rodriguez level,
1094465dce62SFelix Fietkau ATH9K_ANI_SPUR_IMMUNE_LVL,
1095e36b27afSLuis R. Rodriguez value,
1096e36b27afSLuis R. Rodriguez aniState->iniDef.cycpwrThr1);
1097d2182b69SJoe Perches ath_dbg(common, ANI,
1098226afe68SJoe Perches "** ch %d: level %d=>%d[def:%d] cycpwrThr1Ext[level]=%d ini=%d\n",
1099e36b27afSLuis R. Rodriguez chan->channel,
1100e36b27afSLuis R. Rodriguez aniState->spurImmunityLevel,
1101e36b27afSLuis R. Rodriguez level,
1102465dce62SFelix Fietkau ATH9K_ANI_SPUR_IMMUNE_LVL,
110328327fd0SFelix Fietkau value,
1104e36b27afSLuis R. Rodriguez aniState->iniDef.cycpwrThr1Ext);
1105e36b27afSLuis R. Rodriguez if (level > aniState->spurImmunityLevel)
1106e36b27afSLuis R. Rodriguez ah->stats.ast_ani_spurup++;
1107e36b27afSLuis R. Rodriguez else if (level < aniState->spurImmunityLevel)
1108e36b27afSLuis R. Rodriguez ah->stats.ast_ani_spurdown++;
1109e36b27afSLuis R. Rodriguez aniState->spurImmunityLevel = level;
1110e36b27afSLuis R. Rodriguez }
1111e36b27afSLuis R. Rodriguez break;
1112e36b27afSLuis R. Rodriguez }
1113e36b27afSLuis R. Rodriguez case ATH9K_ANI_MRC_CCK:
1114e36b27afSLuis R. Rodriguez /*
1115e36b27afSLuis R. Rodriguez * You should not see this as AR5008, AR9001, AR9002
1116e36b27afSLuis R. Rodriguez * does not have hardware support for MRC CCK.
1117e36b27afSLuis R. Rodriguez */
1118e36b27afSLuis R. Rodriguez WARN_ON(1);
1119e36b27afSLuis R. Rodriguez break;
1120e36b27afSLuis R. Rodriguez default:
1121d2182b69SJoe Perches ath_dbg(common, ANI, "invalid cmd %u\n", cmd);
1122e36b27afSLuis R. Rodriguez return false;
1123e36b27afSLuis R. Rodriguez }
1124e36b27afSLuis R. Rodriguez
1125d2182b69SJoe Perches ath_dbg(common, ANI,
1126226afe68SJoe Perches "ANI parameters: SI=%d, ofdmWS=%s FS=%d MRCcck=%s listenTime=%d ofdmErrs=%d cckErrs=%d\n",
1127e36b27afSLuis R. Rodriguez aniState->spurImmunityLevel,
11287067e701SFelix Fietkau aniState->ofdmWeakSigDetect ? "on" : "off",
1129e36b27afSLuis R. Rodriguez aniState->firstepLevel,
113081b67fd6SRajkumar Manoharan aniState->mrcCCK ? "on" : "off",
1131e36b27afSLuis R. Rodriguez aniState->listenTime,
1132e36b27afSLuis R. Rodriguez aniState->ofdmPhyErrCount,
1133e36b27afSLuis R. Rodriguez aniState->cckPhyErrCount);
1134e36b27afSLuis R. Rodriguez return true;
1135e36b27afSLuis R. Rodriguez }
1136e36b27afSLuis R. Rodriguez
ar5008_hw_do_getnf(struct ath_hw * ah,int16_t nfarray[NUM_NF_READINGS])1137641d9921SFelix Fietkau static void ar5008_hw_do_getnf(struct ath_hw *ah,
1138641d9921SFelix Fietkau int16_t nfarray[NUM_NF_READINGS])
1139641d9921SFelix Fietkau {
1140641d9921SFelix Fietkau int16_t nf;
1141641d9921SFelix Fietkau
1142641d9921SFelix Fietkau nf = MS(REG_READ(ah, AR_PHY_CCA), AR_PHY_MINCCA_PWR);
11437919a57bSAndreas Herrmann nfarray[0] = sign_extend32(nf, 8);
1144641d9921SFelix Fietkau
1145641d9921SFelix Fietkau nf = MS(REG_READ(ah, AR_PHY_CH1_CCA), AR_PHY_CH1_MINCCA_PWR);
11467919a57bSAndreas Herrmann nfarray[1] = sign_extend32(nf, 8);
1147641d9921SFelix Fietkau
1148641d9921SFelix Fietkau nf = MS(REG_READ(ah, AR_PHY_CH2_CCA), AR_PHY_CH2_MINCCA_PWR);
11497919a57bSAndreas Herrmann nfarray[2] = sign_extend32(nf, 8);
1150641d9921SFelix Fietkau
1151866b7780SFelix Fietkau if (!IS_CHAN_HT40(ah->curchan))
1152866b7780SFelix Fietkau return;
1153866b7780SFelix Fietkau
1154641d9921SFelix Fietkau nf = MS(REG_READ(ah, AR_PHY_EXT_CCA), AR_PHY_EXT_MINCCA_PWR);
11557919a57bSAndreas Herrmann nfarray[3] = sign_extend32(nf, 8);
1156641d9921SFelix Fietkau
1157641d9921SFelix Fietkau nf = MS(REG_READ(ah, AR_PHY_CH1_EXT_CCA), AR_PHY_CH1_EXT_MINCCA_PWR);
11587919a57bSAndreas Herrmann nfarray[4] = sign_extend32(nf, 8);
1159641d9921SFelix Fietkau
1160641d9921SFelix Fietkau nf = MS(REG_READ(ah, AR_PHY_CH2_EXT_CCA), AR_PHY_CH2_EXT_MINCCA_PWR);
11617919a57bSAndreas Herrmann nfarray[5] = sign_extend32(nf, 8);
1162641d9921SFelix Fietkau }
1163641d9921SFelix Fietkau
1164e36b27afSLuis R. Rodriguez /*
1165e36b27afSLuis R. Rodriguez * Initialize the ANI register values with default (ini) values.
1166e36b27afSLuis R. Rodriguez * This routine is called during a (full) hardware reset after
1167e36b27afSLuis R. Rodriguez * all the registers are initialised from the INI.
1168e36b27afSLuis R. Rodriguez */
ar5008_hw_ani_cache_ini_regs(struct ath_hw * ah)1169e36b27afSLuis R. Rodriguez static void ar5008_hw_ani_cache_ini_regs(struct ath_hw *ah)
1170e36b27afSLuis R. Rodriguez {
1171e36b27afSLuis R. Rodriguez struct ath_common *common = ath9k_hw_common(ah);
1172e36b27afSLuis R. Rodriguez struct ath9k_channel *chan = ah->curchan;
1173c24bd362SSujith Manoharan struct ar5416AniState *aniState = &ah->ani;
1174e36b27afSLuis R. Rodriguez struct ath9k_ani_default *iniDef;
1175e36b27afSLuis R. Rodriguez u32 val;
1176e36b27afSLuis R. Rodriguez
1177e36b27afSLuis R. Rodriguez iniDef = &aniState->iniDef;
1178e36b27afSLuis R. Rodriguez
11798896934cSFelix Fietkau ath_dbg(common, ANI, "ver %d.%d opmode %u chan %d Mhz\n",
1180e36b27afSLuis R. Rodriguez ah->hw_version.macVersion,
1181e36b27afSLuis R. Rodriguez ah->hw_version.macRev,
1182e36b27afSLuis R. Rodriguez ah->opmode,
11838896934cSFelix Fietkau chan->channel);
1184e36b27afSLuis R. Rodriguez
1185e36b27afSLuis R. Rodriguez val = REG_READ(ah, AR_PHY_SFCORR);
1186e36b27afSLuis R. Rodriguez iniDef->m1Thresh = MS(val, AR_PHY_SFCORR_M1_THRESH);
1187e36b27afSLuis R. Rodriguez iniDef->m2Thresh = MS(val, AR_PHY_SFCORR_M2_THRESH);
1188e36b27afSLuis R. Rodriguez iniDef->m2CountThr = MS(val, AR_PHY_SFCORR_M2COUNT_THR);
1189e36b27afSLuis R. Rodriguez
1190e36b27afSLuis R. Rodriguez val = REG_READ(ah, AR_PHY_SFCORR_LOW);
1191e36b27afSLuis R. Rodriguez iniDef->m1ThreshLow = MS(val, AR_PHY_SFCORR_LOW_M1_THRESH_LOW);
1192e36b27afSLuis R. Rodriguez iniDef->m2ThreshLow = MS(val, AR_PHY_SFCORR_LOW_M2_THRESH_LOW);
1193e36b27afSLuis R. Rodriguez iniDef->m2CountThrLow = MS(val, AR_PHY_SFCORR_LOW_M2COUNT_THR_LOW);
1194e36b27afSLuis R. Rodriguez
1195e36b27afSLuis R. Rodriguez val = REG_READ(ah, AR_PHY_SFCORR_EXT);
1196e36b27afSLuis R. Rodriguez iniDef->m1ThreshExt = MS(val, AR_PHY_SFCORR_EXT_M1_THRESH);
1197e36b27afSLuis R. Rodriguez iniDef->m2ThreshExt = MS(val, AR_PHY_SFCORR_EXT_M2_THRESH);
1198e36b27afSLuis R. Rodriguez iniDef->m1ThreshLowExt = MS(val, AR_PHY_SFCORR_EXT_M1_THRESH_LOW);
1199e36b27afSLuis R. Rodriguez iniDef->m2ThreshLowExt = MS(val, AR_PHY_SFCORR_EXT_M2_THRESH_LOW);
1200e36b27afSLuis R. Rodriguez iniDef->firstep = REG_READ_FIELD(ah,
1201e36b27afSLuis R. Rodriguez AR_PHY_FIND_SIG,
1202e36b27afSLuis R. Rodriguez AR_PHY_FIND_SIG_FIRSTEP);
1203e36b27afSLuis R. Rodriguez iniDef->firstepLow = REG_READ_FIELD(ah,
1204e36b27afSLuis R. Rodriguez AR_PHY_FIND_SIG_LOW,
1205e36b27afSLuis R. Rodriguez AR_PHY_FIND_SIG_FIRSTEP_LOW);
1206e36b27afSLuis R. Rodriguez iniDef->cycpwrThr1 = REG_READ_FIELD(ah,
1207e36b27afSLuis R. Rodriguez AR_PHY_TIMING5,
1208e36b27afSLuis R. Rodriguez AR_PHY_TIMING5_CYCPWR_THR1);
1209e36b27afSLuis R. Rodriguez iniDef->cycpwrThr1Ext = REG_READ_FIELD(ah,
1210e36b27afSLuis R. Rodriguez AR_PHY_EXT_CCA,
1211e36b27afSLuis R. Rodriguez AR_PHY_EXT_TIMING5_CYCPWR_THR1);
1212e36b27afSLuis R. Rodriguez
1213e36b27afSLuis R. Rodriguez /* these levels just got reset to defaults by the INI */
1214465dce62SFelix Fietkau aniState->spurImmunityLevel = ATH9K_ANI_SPUR_IMMUNE_LVL;
1215465dce62SFelix Fietkau aniState->firstepLevel = ATH9K_ANI_FIRSTEP_LVL;
12164f4395c6SSujith Manoharan aniState->ofdmWeakSigDetect = true;
121781b67fd6SRajkumar Manoharan aniState->mrcCCK = false; /* not available on pre AR9003 */
1218e36b27afSLuis R. Rodriguez }
1219e36b27afSLuis R. Rodriguez
ar5008_hw_set_nf_limits(struct ath_hw * ah)1220f2552e28SFelix Fietkau static void ar5008_hw_set_nf_limits(struct ath_hw *ah)
1221f2552e28SFelix Fietkau {
1222f2552e28SFelix Fietkau ah->nf_2g.max = AR_PHY_CCA_MAX_GOOD_VAL_5416_2GHZ;
1223f2552e28SFelix Fietkau ah->nf_2g.min = AR_PHY_CCA_MIN_GOOD_VAL_5416_2GHZ;
1224f2552e28SFelix Fietkau ah->nf_2g.nominal = AR_PHY_CCA_NOM_VAL_5416_2GHZ;
1225f2552e28SFelix Fietkau ah->nf_5g.max = AR_PHY_CCA_MAX_GOOD_VAL_5416_5GHZ;
1226f2552e28SFelix Fietkau ah->nf_5g.min = AR_PHY_CCA_MIN_GOOD_VAL_5416_5GHZ;
1227f2552e28SFelix Fietkau ah->nf_5g.nominal = AR_PHY_CCA_NOM_VAL_5416_5GHZ;
1228f2552e28SFelix Fietkau }
1229e36b27afSLuis R. Rodriguez
ar5008_hw_set_radar_params(struct ath_hw * ah,struct ath_hw_radar_conf * conf)12304e8c14e9SFelix Fietkau static void ar5008_hw_set_radar_params(struct ath_hw *ah,
12314e8c14e9SFelix Fietkau struct ath_hw_radar_conf *conf)
12324e8c14e9SFelix Fietkau {
1233992a36a6SLorenzo Bianconi u32 radar_0 = 0, radar_1;
12344e8c14e9SFelix Fietkau
12354e8c14e9SFelix Fietkau if (!conf) {
12364e8c14e9SFelix Fietkau REG_CLR_BIT(ah, AR_PHY_RADAR_0, AR_PHY_RADAR_0_ENA);
12374e8c14e9SFelix Fietkau return;
12384e8c14e9SFelix Fietkau }
12394e8c14e9SFelix Fietkau
12404e8c14e9SFelix Fietkau radar_0 |= AR_PHY_RADAR_0_ENA | AR_PHY_RADAR_0_FFT_ENA;
12414e8c14e9SFelix Fietkau radar_0 |= SM(conf->fir_power, AR_PHY_RADAR_0_FIRPWR);
12424e8c14e9SFelix Fietkau radar_0 |= SM(conf->radar_rssi, AR_PHY_RADAR_0_RRSSI);
12434e8c14e9SFelix Fietkau radar_0 |= SM(conf->pulse_height, AR_PHY_RADAR_0_HEIGHT);
12444e8c14e9SFelix Fietkau radar_0 |= SM(conf->pulse_rssi, AR_PHY_RADAR_0_PRSSI);
12454e8c14e9SFelix Fietkau radar_0 |= SM(conf->pulse_inband, AR_PHY_RADAR_0_INBAND);
12464e8c14e9SFelix Fietkau
1247992a36a6SLorenzo Bianconi radar_1 = REG_READ(ah, AR_PHY_RADAR_1);
1248992a36a6SLorenzo Bianconi radar_1 &= ~(AR_PHY_RADAR_1_MAXLEN | AR_PHY_RADAR_1_RELSTEP_THRESH |
1249992a36a6SLorenzo Bianconi AR_PHY_RADAR_1_RELPWR_THRESH);
12504e8c14e9SFelix Fietkau radar_1 |= AR_PHY_RADAR_1_MAX_RRSSI;
12514e8c14e9SFelix Fietkau radar_1 |= AR_PHY_RADAR_1_BLOCK_CHECK;
12524e8c14e9SFelix Fietkau radar_1 |= SM(conf->pulse_maxlen, AR_PHY_RADAR_1_MAXLEN);
12534e8c14e9SFelix Fietkau radar_1 |= SM(conf->pulse_inband_step, AR_PHY_RADAR_1_RELSTEP_THRESH);
12544e8c14e9SFelix Fietkau radar_1 |= SM(conf->radar_inband, AR_PHY_RADAR_1_RELPWR_THRESH);
12554e8c14e9SFelix Fietkau
12564e8c14e9SFelix Fietkau REG_WRITE(ah, AR_PHY_RADAR_0, radar_0);
12574e8c14e9SFelix Fietkau REG_WRITE(ah, AR_PHY_RADAR_1, radar_1);
12584e8c14e9SFelix Fietkau if (conf->ext_channel)
12594e8c14e9SFelix Fietkau REG_SET_BIT(ah, AR_PHY_RADAR_EXT, AR_PHY_RADAR_EXT_ENA);
12604e8c14e9SFelix Fietkau else
12614e8c14e9SFelix Fietkau REG_CLR_BIT(ah, AR_PHY_RADAR_EXT, AR_PHY_RADAR_EXT_ENA);
12624e8c14e9SFelix Fietkau }
12634e8c14e9SFelix Fietkau
ar5008_hw_set_radar_conf(struct ath_hw * ah)1264c5d0855aSFelix Fietkau static void ar5008_hw_set_radar_conf(struct ath_hw *ah)
1265c5d0855aSFelix Fietkau {
1266c5d0855aSFelix Fietkau struct ath_hw_radar_conf *conf = &ah->radar_conf;
1267c5d0855aSFelix Fietkau
1268c5d0855aSFelix Fietkau conf->fir_power = -33;
1269c5d0855aSFelix Fietkau conf->radar_rssi = 20;
1270c5d0855aSFelix Fietkau conf->pulse_height = 10;
1271edad1873SLorenzo Bianconi conf->pulse_rssi = 15;
1272c5d0855aSFelix Fietkau conf->pulse_inband = 15;
1273c5d0855aSFelix Fietkau conf->pulse_maxlen = 255;
1274c5d0855aSFelix Fietkau conf->pulse_inband_step = 12;
1275c5d0855aSFelix Fietkau conf->radar_inband = 8;
1276c5d0855aSFelix Fietkau }
1277c5d0855aSFelix Fietkau
ar5008_hw_init_txpower_cck(struct ath_hw * ah,int16_t * rate_array)1278c08267dcSLorenzo Bianconi static void ar5008_hw_init_txpower_cck(struct ath_hw *ah, int16_t *rate_array)
1279c08267dcSLorenzo Bianconi {
1280*b3a663f0SWenli Looi #define CCK_DELTA(_ah, x) ((OLC_FOR_AR9280_20_LATER(_ah)) ? max((x) - 2, 0) : (x))
1281*b3a663f0SWenli Looi ah->tx_power[0] = CCK_DELTA(ah, rate_array[rate1l]);
1282*b3a663f0SWenli Looi ah->tx_power[1] = CCK_DELTA(ah, min(rate_array[rate2l],
1283c08267dcSLorenzo Bianconi rate_array[rate2s]));
1284*b3a663f0SWenli Looi ah->tx_power[2] = CCK_DELTA(ah, min(rate_array[rate5_5l],
1285c08267dcSLorenzo Bianconi rate_array[rate5_5s]));
1286*b3a663f0SWenli Looi ah->tx_power[3] = CCK_DELTA(ah, min(rate_array[rate11l],
1287c08267dcSLorenzo Bianconi rate_array[rate11s]));
1288c08267dcSLorenzo Bianconi #undef CCK_DELTA
1289c08267dcSLorenzo Bianconi }
1290c08267dcSLorenzo Bianconi
ar5008_hw_init_txpower_ofdm(struct ath_hw * ah,int16_t * rate_array,int offset)1291c08267dcSLorenzo Bianconi static void ar5008_hw_init_txpower_ofdm(struct ath_hw *ah, int16_t *rate_array,
1292c08267dcSLorenzo Bianconi int offset)
1293c08267dcSLorenzo Bianconi {
1294c08267dcSLorenzo Bianconi int i, idx = 0;
1295c08267dcSLorenzo Bianconi
1296c08267dcSLorenzo Bianconi for (i = offset; i < offset + AR5008_OFDM_RATES; i++) {
1297c08267dcSLorenzo Bianconi ah->tx_power[i] = rate_array[idx];
1298c08267dcSLorenzo Bianconi idx++;
1299c08267dcSLorenzo Bianconi }
1300c08267dcSLorenzo Bianconi }
1301c08267dcSLorenzo Bianconi
ar5008_hw_init_txpower_ht(struct ath_hw * ah,int16_t * rate_array,int ss_offset,int ds_offset,bool is_40,int ht40_delta)1302c08267dcSLorenzo Bianconi static void ar5008_hw_init_txpower_ht(struct ath_hw *ah, int16_t *rate_array,
1303c08267dcSLorenzo Bianconi int ss_offset, int ds_offset,
1304c08267dcSLorenzo Bianconi bool is_40, int ht40_delta)
1305c08267dcSLorenzo Bianconi {
1306c08267dcSLorenzo Bianconi int i, mcs_idx = (is_40) ? AR5008_HT40_SHIFT : AR5008_HT20_SHIFT;
1307c08267dcSLorenzo Bianconi
1308c08267dcSLorenzo Bianconi for (i = ss_offset; i < ss_offset + AR5008_HT_SS_RATES; i++) {
1309c08267dcSLorenzo Bianconi ah->tx_power[i] = rate_array[mcs_idx] + ht40_delta;
1310c08267dcSLorenzo Bianconi mcs_idx++;
1311c08267dcSLorenzo Bianconi }
1312c08267dcSLorenzo Bianconi memcpy(&ah->tx_power[ds_offset], &ah->tx_power[ss_offset],
1313c08267dcSLorenzo Bianconi AR5008_HT_SS_RATES);
1314c08267dcSLorenzo Bianconi }
1315c08267dcSLorenzo Bianconi
ar5008_hw_init_rate_txpower(struct ath_hw * ah,int16_t * rate_array,struct ath9k_channel * chan,int ht40_delta)1316c08267dcSLorenzo Bianconi void ar5008_hw_init_rate_txpower(struct ath_hw *ah, int16_t *rate_array,
1317c08267dcSLorenzo Bianconi struct ath9k_channel *chan, int ht40_delta)
1318c08267dcSLorenzo Bianconi {
1319c08267dcSLorenzo Bianconi if (IS_CHAN_5GHZ(chan)) {
1320c08267dcSLorenzo Bianconi ar5008_hw_init_txpower_ofdm(ah, rate_array,
1321c08267dcSLorenzo Bianconi AR5008_11NA_OFDM_SHIFT);
1322c08267dcSLorenzo Bianconi if (IS_CHAN_HT20(chan) || IS_CHAN_HT40(chan)) {
1323c08267dcSLorenzo Bianconi ar5008_hw_init_txpower_ht(ah, rate_array,
1324c08267dcSLorenzo Bianconi AR5008_11NA_HT_SS_SHIFT,
1325c08267dcSLorenzo Bianconi AR5008_11NA_HT_DS_SHIFT,
1326c08267dcSLorenzo Bianconi IS_CHAN_HT40(chan),
1327c08267dcSLorenzo Bianconi ht40_delta);
1328c08267dcSLorenzo Bianconi }
1329c08267dcSLorenzo Bianconi } else {
1330c08267dcSLorenzo Bianconi ar5008_hw_init_txpower_cck(ah, rate_array);
1331c08267dcSLorenzo Bianconi ar5008_hw_init_txpower_ofdm(ah, rate_array,
1332c08267dcSLorenzo Bianconi AR5008_11NG_OFDM_SHIFT);
1333c08267dcSLorenzo Bianconi if (IS_CHAN_HT20(chan) || IS_CHAN_HT40(chan)) {
1334c08267dcSLorenzo Bianconi ar5008_hw_init_txpower_ht(ah, rate_array,
1335c08267dcSLorenzo Bianconi AR5008_11NG_HT_SS_SHIFT,
1336c08267dcSLorenzo Bianconi AR5008_11NG_HT_DS_SHIFT,
1337c08267dcSLorenzo Bianconi IS_CHAN_HT40(chan),
1338c08267dcSLorenzo Bianconi ht40_delta);
1339c08267dcSLorenzo Bianconi }
1340c08267dcSLorenzo Bianconi }
1341c08267dcSLorenzo Bianconi }
1342c08267dcSLorenzo Bianconi
ar5008_hw_attach_phy_ops(struct ath_hw * ah)1343c1b976d2SFelix Fietkau int ar5008_hw_attach_phy_ops(struct ath_hw *ah)
13448fe65368SLuis R. Rodriguez {
13458fe65368SLuis R. Rodriguez struct ath_hw_private_ops *priv_ops = ath9k_hw_private_ops(ah);
134607b2fa5aSJoe Perches static const u32 ar5416_cca_regs[6] = {
1347bbacee13SFelix Fietkau AR_PHY_CCA,
1348bbacee13SFelix Fietkau AR_PHY_CH1_CCA,
1349bbacee13SFelix Fietkau AR_PHY_CH2_CCA,
1350bbacee13SFelix Fietkau AR_PHY_EXT_CCA,
1351bbacee13SFelix Fietkau AR_PHY_CH1_EXT_CCA,
1352bbacee13SFelix Fietkau AR_PHY_CH2_EXT_CCA
1353bbacee13SFelix Fietkau };
1354c1b976d2SFelix Fietkau int ret;
1355c1b976d2SFelix Fietkau
1356c1b976d2SFelix Fietkau ret = ar5008_hw_rf_alloc_ext_banks(ah);
1357c1b976d2SFelix Fietkau if (ret)
1358c1b976d2SFelix Fietkau return ret;
13598fe65368SLuis R. Rodriguez
13608fe65368SLuis R. Rodriguez priv_ops->rf_set_freq = ar5008_hw_set_channel;
13618fe65368SLuis R. Rodriguez priv_ops->spur_mitigate_freq = ar5008_hw_spur_mitigate;
13628fe65368SLuis R. Rodriguez
13638fe65368SLuis R. Rodriguez priv_ops->set_rf_regs = ar5008_hw_set_rf_regs;
13648fe65368SLuis R. Rodriguez priv_ops->set_channel_regs = ar5008_hw_set_channel_regs;
13658fe65368SLuis R. Rodriguez priv_ops->init_bb = ar5008_hw_init_bb;
13668fe65368SLuis R. Rodriguez priv_ops->process_ini = ar5008_hw_process_ini;
13678fe65368SLuis R. Rodriguez priv_ops->set_rfmode = ar5008_hw_set_rfmode;
13688fe65368SLuis R. Rodriguez priv_ops->mark_phy_inactive = ar5008_hw_mark_phy_inactive;
13698fe65368SLuis R. Rodriguez priv_ops->set_delta_slope = ar5008_hw_set_delta_slope;
13708fe65368SLuis R. Rodriguez priv_ops->rfbus_req = ar5008_hw_rfbus_req;
13718fe65368SLuis R. Rodriguez priv_ops->rfbus_done = ar5008_hw_rfbus_done;
13728fe65368SLuis R. Rodriguez priv_ops->restore_chainmask = ar5008_restore_chainmask;
1373641d9921SFelix Fietkau priv_ops->do_getnf = ar5008_hw_do_getnf;
13744e8c14e9SFelix Fietkau priv_ops->set_radar_params = ar5008_hw_set_radar_params;
137564773964SLuis R. Rodriguez
1376e36b27afSLuis R. Rodriguez priv_ops->ani_control = ar5008_hw_ani_control_new;
1377e36b27afSLuis R. Rodriguez priv_ops->ani_cache_ini_regs = ar5008_hw_ani_cache_ini_regs;
1378e36b27afSLuis R. Rodriguez
1379491b209dSFelix Fietkau if (AR_SREV_9100(ah) || AR_SREV_9160_10_OR_LATER(ah))
138064773964SLuis R. Rodriguez priv_ops->compute_pll_control = ar9160_hw_compute_pll_control;
138164773964SLuis R. Rodriguez else
138264773964SLuis R. Rodriguez priv_ops->compute_pll_control = ar5008_hw_compute_pll_control;
1383f2552e28SFelix Fietkau
1384f2552e28SFelix Fietkau ar5008_hw_set_nf_limits(ah);
1385c5d0855aSFelix Fietkau ar5008_hw_set_radar_conf(ah);
1386bbacee13SFelix Fietkau memcpy(ah->nf_regs, ar5416_cca_regs, sizeof(ah->nf_regs));
1387c1b976d2SFelix Fietkau return 0;
13888fe65368SLuis R. Rodriguez }
1389