1 /*
2  * Copyright (C) 2016 Felix Fietkau <nbd@nbd.name>
3  *
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.
7  *
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.
15  */
16 
17 #include <linux/module.h>
18 #include <asm/unaligned.h>
19 #include "mt76x2.h"
20 #include "eeprom.h"
21 
22 #define EE_FIELD(_name, _value) [MT_EE_##_name] = (_value) | 1
23 
24 static int
25 mt76x2_eeprom_copy(struct mt76x02_dev *dev, enum mt76x02_eeprom_field field,
26 		   void *dest, int len)
27 {
28 	if (field + len > dev->mt76.eeprom.size)
29 		return -1;
30 
31 	memcpy(dest, dev->mt76.eeprom.data + field, len);
32 	return 0;
33 }
34 
35 static int
36 mt76x2_eeprom_get_macaddr(struct mt76x02_dev *dev)
37 {
38 	void *src = dev->mt76.eeprom.data + MT_EE_MAC_ADDR;
39 
40 	memcpy(dev->mt76.macaddr, src, ETH_ALEN);
41 	return 0;
42 }
43 
44 static bool
45 mt76x2_has_cal_free_data(struct mt76x02_dev *dev, u8 *efuse)
46 {
47 	u16 *efuse_w = (u16 *) efuse;
48 
49 	if (efuse_w[MT_EE_NIC_CONF_0] != 0)
50 		return false;
51 
52 	if (efuse_w[MT_EE_XTAL_TRIM_1] == 0xffff)
53 		return false;
54 
55 	if (efuse_w[MT_EE_TX_POWER_DELTA_BW40] != 0)
56 		return false;
57 
58 	if (efuse_w[MT_EE_TX_POWER_0_START_2G] == 0xffff)
59 		return false;
60 
61 	if (efuse_w[MT_EE_TX_POWER_0_GRP3_TX_POWER_DELTA] != 0)
62 		return false;
63 
64 	if (efuse_w[MT_EE_TX_POWER_0_GRP4_TSSI_SLOPE] == 0xffff)
65 		return false;
66 
67 	return true;
68 }
69 
70 static void
71 mt76x2_apply_cal_free_data(struct mt76x02_dev *dev, u8 *efuse)
72 {
73 #define GROUP_5G(_id)							   \
74 	MT_EE_TX_POWER_0_START_5G + MT_TX_POWER_GROUP_SIZE_5G * (_id),	   \
75 	MT_EE_TX_POWER_0_START_5G + MT_TX_POWER_GROUP_SIZE_5G * (_id) + 1, \
76 	MT_EE_TX_POWER_1_START_5G + MT_TX_POWER_GROUP_SIZE_5G * (_id),	   \
77 	MT_EE_TX_POWER_1_START_5G + MT_TX_POWER_GROUP_SIZE_5G * (_id) + 1
78 
79 	static const u8 cal_free_bytes[] = {
80 		MT_EE_XTAL_TRIM_1,
81 		MT_EE_TX_POWER_EXT_PA_5G + 1,
82 		MT_EE_TX_POWER_0_START_2G,
83 		MT_EE_TX_POWER_0_START_2G + 1,
84 		MT_EE_TX_POWER_1_START_2G,
85 		MT_EE_TX_POWER_1_START_2G + 1,
86 		GROUP_5G(0),
87 		GROUP_5G(1),
88 		GROUP_5G(2),
89 		GROUP_5G(3),
90 		GROUP_5G(4),
91 		GROUP_5G(5),
92 		MT_EE_RF_2G_TSSI_OFF_TXPOWER,
93 		MT_EE_RF_2G_RX_HIGH_GAIN + 1,
94 		MT_EE_RF_5G_GRP0_1_RX_HIGH_GAIN,
95 		MT_EE_RF_5G_GRP0_1_RX_HIGH_GAIN + 1,
96 		MT_EE_RF_5G_GRP2_3_RX_HIGH_GAIN,
97 		MT_EE_RF_5G_GRP2_3_RX_HIGH_GAIN + 1,
98 		MT_EE_RF_5G_GRP4_5_RX_HIGH_GAIN,
99 		MT_EE_RF_5G_GRP4_5_RX_HIGH_GAIN + 1,
100 	};
101 	u8 *eeprom = dev->mt76.eeprom.data;
102 	u8 prev_grp0[4] = {
103 		eeprom[MT_EE_TX_POWER_0_START_5G],
104 		eeprom[MT_EE_TX_POWER_0_START_5G + 1],
105 		eeprom[MT_EE_TX_POWER_1_START_5G],
106 		eeprom[MT_EE_TX_POWER_1_START_5G + 1]
107 	};
108 	u16 val;
109 	int i;
110 
111 	if (!mt76x2_has_cal_free_data(dev, efuse))
112 		return;
113 
114 	for (i = 0; i < ARRAY_SIZE(cal_free_bytes); i++) {
115 		int offset = cal_free_bytes[i];
116 
117 		eeprom[offset] = efuse[offset];
118 	}
119 
120 	if (!(efuse[MT_EE_TX_POWER_0_START_5G] |
121 	      efuse[MT_EE_TX_POWER_0_START_5G + 1]))
122 		memcpy(eeprom + MT_EE_TX_POWER_0_START_5G, prev_grp0, 2);
123 	if (!(efuse[MT_EE_TX_POWER_1_START_5G] |
124 	      efuse[MT_EE_TX_POWER_1_START_5G + 1]))
125 		memcpy(eeprom + MT_EE_TX_POWER_1_START_5G, prev_grp0 + 2, 2);
126 
127 	val = get_unaligned_le16(efuse + MT_EE_BT_RCAL_RESULT);
128 	if (val != 0xffff)
129 		eeprom[MT_EE_BT_RCAL_RESULT] = val & 0xff;
130 
131 	val = get_unaligned_le16(efuse + MT_EE_BT_VCDL_CALIBRATION);
132 	if (val != 0xffff)
133 		eeprom[MT_EE_BT_VCDL_CALIBRATION + 1] = val >> 8;
134 
135 	val = get_unaligned_le16(efuse + MT_EE_BT_PMUCFG);
136 	if (val != 0xffff)
137 		eeprom[MT_EE_BT_PMUCFG] = val & 0xff;
138 }
139 
140 static int mt76x2_check_eeprom(struct mt76x02_dev *dev)
141 {
142 	u16 val = get_unaligned_le16(dev->mt76.eeprom.data);
143 
144 	if (!val)
145 		val = get_unaligned_le16(dev->mt76.eeprom.data + MT_EE_PCI_ID);
146 
147 	switch (val) {
148 	case 0x7662:
149 	case 0x7612:
150 		return 0;
151 	default:
152 		dev_err(dev->mt76.dev, "EEPROM data check failed: %04x\n", val);
153 		return -EINVAL;
154 	}
155 }
156 
157 static int
158 mt76x2_eeprom_load(struct mt76x02_dev *dev)
159 {
160 	void *efuse;
161 	bool found;
162 	int ret;
163 
164 	ret = mt76_eeprom_init(&dev->mt76, MT7662_EEPROM_SIZE);
165 	if (ret < 0)
166 		return ret;
167 
168 	found = ret;
169 	if (found)
170 		found = !mt76x2_check_eeprom(dev);
171 
172 	dev->mt76.otp.data = devm_kzalloc(dev->mt76.dev, MT7662_EEPROM_SIZE,
173 					  GFP_KERNEL);
174 	dev->mt76.otp.size = MT7662_EEPROM_SIZE;
175 	if (!dev->mt76.otp.data)
176 		return -ENOMEM;
177 
178 	efuse = dev->mt76.otp.data;
179 
180 	if (mt76x02_get_efuse_data(dev, 0, efuse, MT7662_EEPROM_SIZE,
181 				   MT_EE_READ))
182 		goto out;
183 
184 	if (found) {
185 		mt76x2_apply_cal_free_data(dev, efuse);
186 	} else {
187 		/* FIXME: check if efuse data is complete */
188 		found = true;
189 		memcpy(dev->mt76.eeprom.data, efuse, MT7662_EEPROM_SIZE);
190 	}
191 
192 out:
193 	if (!found)
194 		return -ENOENT;
195 
196 	return 0;
197 }
198 
199 static void
200 mt76x2_set_rx_gain_group(struct mt76x02_dev *dev, u8 val)
201 {
202 	s8 *dest = dev->cal.rx.high_gain;
203 
204 	if (!mt76x02_field_valid(val)) {
205 		dest[0] = 0;
206 		dest[1] = 0;
207 		return;
208 	}
209 
210 	dest[0] = mt76x02_sign_extend(val, 4);
211 	dest[1] = mt76x02_sign_extend(val >> 4, 4);
212 }
213 
214 static void
215 mt76x2_set_rssi_offset(struct mt76x02_dev *dev, int chain, u8 val)
216 {
217 	s8 *dest = dev->cal.rx.rssi_offset;
218 
219 	if (!mt76x02_field_valid(val)) {
220 		dest[chain] = 0;
221 		return;
222 	}
223 
224 	dest[chain] = mt76x02_sign_extend_optional(val, 7);
225 }
226 
227 static enum mt76x2_cal_channel_group
228 mt76x2_get_cal_channel_group(int channel)
229 {
230 	if (channel >= 184 && channel <= 196)
231 		return MT_CH_5G_JAPAN;
232 	if (channel <= 48)
233 		return MT_CH_5G_UNII_1;
234 	if (channel <= 64)
235 		return MT_CH_5G_UNII_2;
236 	if (channel <= 114)
237 		return MT_CH_5G_UNII_2E_1;
238 	if (channel <= 144)
239 		return MT_CH_5G_UNII_2E_2;
240 	return MT_CH_5G_UNII_3;
241 }
242 
243 static u8
244 mt76x2_get_5g_rx_gain(struct mt76x02_dev *dev, u8 channel)
245 {
246 	enum mt76x2_cal_channel_group group;
247 
248 	group = mt76x2_get_cal_channel_group(channel);
249 	switch (group) {
250 	case MT_CH_5G_JAPAN:
251 		return mt76x02_eeprom_get(dev,
252 					  MT_EE_RF_5G_GRP0_1_RX_HIGH_GAIN);
253 	case MT_CH_5G_UNII_1:
254 		return mt76x02_eeprom_get(dev,
255 					  MT_EE_RF_5G_GRP0_1_RX_HIGH_GAIN) >> 8;
256 	case MT_CH_5G_UNII_2:
257 		return mt76x02_eeprom_get(dev,
258 					  MT_EE_RF_5G_GRP2_3_RX_HIGH_GAIN);
259 	case MT_CH_5G_UNII_2E_1:
260 		return mt76x02_eeprom_get(dev,
261 					  MT_EE_RF_5G_GRP2_3_RX_HIGH_GAIN) >> 8;
262 	case MT_CH_5G_UNII_2E_2:
263 		return mt76x02_eeprom_get(dev,
264 					  MT_EE_RF_5G_GRP4_5_RX_HIGH_GAIN);
265 	default:
266 		return mt76x02_eeprom_get(dev,
267 					  MT_EE_RF_5G_GRP4_5_RX_HIGH_GAIN) >> 8;
268 	}
269 }
270 
271 void mt76x2_read_rx_gain(struct mt76x02_dev *dev)
272 {
273 	struct ieee80211_channel *chan = dev->mt76.chandef.chan;
274 	int channel = chan->hw_value;
275 	s8 lna_5g[3], lna_2g;
276 	u8 lna;
277 	u16 val;
278 
279 	if (chan->band == NL80211_BAND_2GHZ)
280 		val = mt76x02_eeprom_get(dev, MT_EE_RF_2G_RX_HIGH_GAIN) >> 8;
281 	else
282 		val = mt76x2_get_5g_rx_gain(dev, channel);
283 
284 	mt76x2_set_rx_gain_group(dev, val);
285 
286 	mt76x02_get_rx_gain(dev, chan->band, &val, &lna_2g, lna_5g);
287 	mt76x2_set_rssi_offset(dev, 0, val);
288 	mt76x2_set_rssi_offset(dev, 1, val >> 8);
289 
290 	dev->cal.rx.mcu_gain =  (lna_2g & 0xff);
291 	dev->cal.rx.mcu_gain |= (lna_5g[0] & 0xff) << 8;
292 	dev->cal.rx.mcu_gain |= (lna_5g[1] & 0xff) << 16;
293 	dev->cal.rx.mcu_gain |= (lna_5g[2] & 0xff) << 24;
294 
295 	lna = mt76x02_get_lna_gain(dev, &lna_2g, lna_5g, chan);
296 	dev->cal.rx.lna_gain = mt76x02_sign_extend(lna, 8);
297 }
298 EXPORT_SYMBOL_GPL(mt76x2_read_rx_gain);
299 
300 void mt76x2_get_rate_power(struct mt76x02_dev *dev, struct mt76_rate_power *t,
301 			   struct ieee80211_channel *chan)
302 {
303 	bool is_5ghz;
304 	u16 val;
305 
306 	is_5ghz = chan->band == NL80211_BAND_5GHZ;
307 
308 	memset(t, 0, sizeof(*t));
309 
310 	val = mt76x02_eeprom_get(dev, MT_EE_TX_POWER_CCK);
311 	t->cck[0] = t->cck[1] = mt76x02_rate_power_val(val);
312 	t->cck[2] = t->cck[3] = mt76x02_rate_power_val(val >> 8);
313 
314 	if (is_5ghz)
315 		val = mt76x02_eeprom_get(dev, MT_EE_TX_POWER_OFDM_5G_6M);
316 	else
317 		val = mt76x02_eeprom_get(dev, MT_EE_TX_POWER_OFDM_2G_6M);
318 	t->ofdm[0] = t->ofdm[1] = mt76x02_rate_power_val(val);
319 	t->ofdm[2] = t->ofdm[3] = mt76x02_rate_power_val(val >> 8);
320 
321 	if (is_5ghz)
322 		val = mt76x02_eeprom_get(dev, MT_EE_TX_POWER_OFDM_5G_24M);
323 	else
324 		val = mt76x02_eeprom_get(dev, MT_EE_TX_POWER_OFDM_2G_24M);
325 	t->ofdm[4] = t->ofdm[5] = mt76x02_rate_power_val(val);
326 	t->ofdm[6] = t->ofdm[7] = mt76x02_rate_power_val(val >> 8);
327 
328 	val = mt76x02_eeprom_get(dev, MT_EE_TX_POWER_HT_MCS0);
329 	t->ht[0] = t->ht[1] = mt76x02_rate_power_val(val);
330 	t->ht[2] = t->ht[3] = mt76x02_rate_power_val(val >> 8);
331 
332 	val = mt76x02_eeprom_get(dev, MT_EE_TX_POWER_HT_MCS4);
333 	t->ht[4] = t->ht[5] = mt76x02_rate_power_val(val);
334 	t->ht[6] = t->ht[7] = mt76x02_rate_power_val(val >> 8);
335 
336 	val = mt76x02_eeprom_get(dev, MT_EE_TX_POWER_HT_MCS8);
337 	t->ht[8] = t->ht[9] = mt76x02_rate_power_val(val);
338 	t->ht[10] = t->ht[11] = mt76x02_rate_power_val(val >> 8);
339 
340 	val = mt76x02_eeprom_get(dev, MT_EE_TX_POWER_HT_MCS12);
341 	t->ht[12] = t->ht[13] = mt76x02_rate_power_val(val);
342 	t->ht[14] = t->ht[15] = mt76x02_rate_power_val(val >> 8);
343 
344 	val = mt76x02_eeprom_get(dev, MT_EE_TX_POWER_VHT_MCS0);
345 	t->vht[0] = t->vht[1] = mt76x02_rate_power_val(val);
346 	t->vht[2] = t->vht[3] = mt76x02_rate_power_val(val >> 8);
347 
348 	val = mt76x02_eeprom_get(dev, MT_EE_TX_POWER_VHT_MCS4);
349 	t->vht[4] = t->vht[5] = mt76x02_rate_power_val(val);
350 	t->vht[6] = t->vht[7] = mt76x02_rate_power_val(val >> 8);
351 
352 	val = mt76x02_eeprom_get(dev, MT_EE_TX_POWER_VHT_MCS8);
353 	if (!is_5ghz)
354 		val >>= 8;
355 	t->vht[8] = t->vht[9] = mt76x02_rate_power_val(val >> 8);
356 
357 	memcpy(t->stbc, t->ht, sizeof(t->stbc[0]) * 8);
358 	t->stbc[8] = t->vht[8];
359 	t->stbc[9] = t->vht[9];
360 }
361 EXPORT_SYMBOL_GPL(mt76x2_get_rate_power);
362 
363 static void
364 mt76x2_get_power_info_2g(struct mt76x02_dev *dev,
365 			 struct mt76x2_tx_power_info *t,
366 			 struct ieee80211_channel *chan,
367 			 int chain, int offset)
368 {
369 	int channel = chan->hw_value;
370 	int delta_idx;
371 	u8 data[6];
372 	u16 val;
373 
374 	if (channel < 6)
375 		delta_idx = 3;
376 	else if (channel < 11)
377 		delta_idx = 4;
378 	else
379 		delta_idx = 5;
380 
381 	mt76x2_eeprom_copy(dev, offset, data, sizeof(data));
382 
383 	t->chain[chain].tssi_slope = data[0];
384 	t->chain[chain].tssi_offset = data[1];
385 	t->chain[chain].target_power = data[2];
386 	t->chain[chain].delta = mt76x02_sign_extend_optional(data[delta_idx], 7);
387 
388 	val = mt76x02_eeprom_get(dev, MT_EE_RF_2G_TSSI_OFF_TXPOWER);
389 	t->target_power = val >> 8;
390 }
391 
392 static void
393 mt76x2_get_power_info_5g(struct mt76x02_dev *dev,
394 			 struct mt76x2_tx_power_info *t,
395 		         struct ieee80211_channel *chan,
396 			 int chain, int offset)
397 {
398 	int channel = chan->hw_value;
399 	enum mt76x2_cal_channel_group group;
400 	int delta_idx;
401 	u16 val;
402 	u8 data[5];
403 
404 	group = mt76x2_get_cal_channel_group(channel);
405 	offset += group * MT_TX_POWER_GROUP_SIZE_5G;
406 
407 	if (channel >= 192)
408 		delta_idx = 4;
409 	else if (channel >= 184)
410 		delta_idx = 3;
411 	else if (channel < 44)
412 		delta_idx = 3;
413 	else if (channel < 52)
414 		delta_idx = 4;
415 	else if (channel < 58)
416 		delta_idx = 3;
417 	else if (channel < 98)
418 		delta_idx = 4;
419 	else if (channel < 106)
420 		delta_idx = 3;
421 	else if (channel < 116)
422 		delta_idx = 4;
423 	else if (channel < 130)
424 		delta_idx = 3;
425 	else if (channel < 149)
426 		delta_idx = 4;
427 	else if (channel < 157)
428 		delta_idx = 3;
429 	else
430 		delta_idx = 4;
431 
432 	mt76x2_eeprom_copy(dev, offset, data, sizeof(data));
433 
434 	t->chain[chain].tssi_slope = data[0];
435 	t->chain[chain].tssi_offset = data[1];
436 	t->chain[chain].target_power = data[2];
437 	t->chain[chain].delta = mt76x02_sign_extend_optional(data[delta_idx], 7);
438 
439 	val = mt76x02_eeprom_get(dev, MT_EE_RF_2G_RX_HIGH_GAIN);
440 	t->target_power = val & 0xff;
441 }
442 
443 void mt76x2_get_power_info(struct mt76x02_dev *dev,
444 			   struct mt76x2_tx_power_info *t,
445 			   struct ieee80211_channel *chan)
446 {
447 	u16 bw40, bw80;
448 
449 	memset(t, 0, sizeof(*t));
450 
451 	bw40 = mt76x02_eeprom_get(dev, MT_EE_TX_POWER_DELTA_BW40);
452 	bw80 = mt76x02_eeprom_get(dev, MT_EE_TX_POWER_DELTA_BW80);
453 
454 	if (chan->band == NL80211_BAND_5GHZ) {
455 		bw40 >>= 8;
456 		mt76x2_get_power_info_5g(dev, t, chan, 0,
457 					 MT_EE_TX_POWER_0_START_5G);
458 		mt76x2_get_power_info_5g(dev, t, chan, 1,
459 					 MT_EE_TX_POWER_1_START_5G);
460 	} else {
461 		mt76x2_get_power_info_2g(dev, t, chan, 0,
462 					 MT_EE_TX_POWER_0_START_2G);
463 		mt76x2_get_power_info_2g(dev, t, chan, 1,
464 					 MT_EE_TX_POWER_1_START_2G);
465 	}
466 
467 	if (mt76x2_tssi_enabled(dev) ||
468 	    !mt76x02_field_valid(t->target_power))
469 		t->target_power = t->chain[0].target_power;
470 
471 	t->delta_bw40 = mt76x02_rate_power_val(bw40);
472 	t->delta_bw80 = mt76x02_rate_power_val(bw80);
473 }
474 EXPORT_SYMBOL_GPL(mt76x2_get_power_info);
475 
476 int mt76x2_get_temp_comp(struct mt76x02_dev *dev, struct mt76x2_temp_comp *t)
477 {
478 	enum nl80211_band band = dev->mt76.chandef.chan->band;
479 	u16 val, slope;
480 	u8 bounds;
481 
482 	memset(t, 0, sizeof(*t));
483 
484 	if (!mt76x2_temp_tx_alc_enabled(dev))
485 		return -EINVAL;
486 
487 	if (!mt76x02_ext_pa_enabled(dev, band))
488 		return -EINVAL;
489 
490 	val = mt76x02_eeprom_get(dev, MT_EE_TX_POWER_EXT_PA_5G) >> 8;
491 	t->temp_25_ref = val & 0x7f;
492 	if (band == NL80211_BAND_5GHZ) {
493 		slope = mt76x02_eeprom_get(dev, MT_EE_RF_TEMP_COMP_SLOPE_5G);
494 		bounds = mt76x02_eeprom_get(dev, MT_EE_TX_POWER_EXT_PA_5G);
495 	} else {
496 		slope = mt76x02_eeprom_get(dev, MT_EE_RF_TEMP_COMP_SLOPE_2G);
497 		bounds = mt76x02_eeprom_get(dev,
498 					    MT_EE_TX_POWER_DELTA_BW80) >> 8;
499 	}
500 
501 	t->high_slope = slope & 0xff;
502 	t->low_slope = slope >> 8;
503 	t->lower_bound = 0 - (bounds & 0xf);
504 	t->upper_bound = (bounds >> 4) & 0xf;
505 
506 	return 0;
507 }
508 EXPORT_SYMBOL_GPL(mt76x2_get_temp_comp);
509 
510 int mt76x2_eeprom_init(struct mt76x02_dev *dev)
511 {
512 	int ret;
513 
514 	ret = mt76x2_eeprom_load(dev);
515 	if (ret)
516 		return ret;
517 
518 	mt76x02_eeprom_parse_hw_cap(dev);
519 	mt76x2_eeprom_get_macaddr(dev);
520 	mt76_eeprom_override(&dev->mt76);
521 	dev->mt76.macaddr[0] &= ~BIT(1);
522 
523 	return 0;
524 }
525 EXPORT_SYMBOL_GPL(mt76x2_eeprom_init);
526 
527 MODULE_LICENSE("Dual BSD/GPL");
528