xref: /openbmc/linux/drivers/net/wireless/mediatek/mt76/mt76x2/usb_mac.c (revision ead5d1f4d877e92c051e1a1ade623d0d30e71619)
10e3d6777SRyder Lee // SPDX-License-Identifier: ISC
21613c621SLorenzo Bianconi /*
31613c621SLorenzo Bianconi  * Copyright (C) 2018 Lorenzo Bianconi <lorenzo.bianconi83@gmail.com>
41613c621SLorenzo Bianconi  */
51613c621SLorenzo Bianconi 
61613c621SLorenzo Bianconi #include "mt76x2u.h"
71613c621SLorenzo Bianconi #include "eeprom.h"
81613c621SLorenzo Bianconi 
mt76x2u_mac_fixup_xtal(struct mt76x02_dev * dev)9e40803f2SLorenzo Bianconi static void mt76x2u_mac_fixup_xtal(struct mt76x02_dev *dev)
101613c621SLorenzo Bianconi {
111613c621SLorenzo Bianconi 	s8 offset = 0;
121613c621SLorenzo Bianconi 	u16 eep_val;
131613c621SLorenzo Bianconi 
1426a9daa6SLorenzo Bianconi 	eep_val = mt76x02_eeprom_get(dev, MT_EE_XTAL_TRIM_2);
151613c621SLorenzo Bianconi 
161613c621SLorenzo Bianconi 	offset = eep_val & 0x7f;
171613c621SLorenzo Bianconi 	if ((eep_val & 0xff) == 0xff)
181613c621SLorenzo Bianconi 		offset = 0;
191613c621SLorenzo Bianconi 	else if (eep_val & 0x80)
201613c621SLorenzo Bianconi 		offset = 0 - offset;
211613c621SLorenzo Bianconi 
221613c621SLorenzo Bianconi 	eep_val >>= 8;
231613c621SLorenzo Bianconi 	if (eep_val == 0x00 || eep_val == 0xff) {
2426a9daa6SLorenzo Bianconi 		eep_val = mt76x02_eeprom_get(dev, MT_EE_XTAL_TRIM_1);
251613c621SLorenzo Bianconi 		eep_val &= 0xff;
261613c621SLorenzo Bianconi 
271613c621SLorenzo Bianconi 		if (eep_val == 0x00 || eep_val == 0xff)
281613c621SLorenzo Bianconi 			eep_val = 0x14;
291613c621SLorenzo Bianconi 	}
301613c621SLorenzo Bianconi 
311613c621SLorenzo Bianconi 	eep_val &= 0x7f;
321613c621SLorenzo Bianconi 	mt76_rmw_field(dev, MT_VEND_ADDR(CFG, MT_XO_CTRL5),
331613c621SLorenzo Bianconi 		       MT_XO_CTRL5_C2_VAL, eep_val + offset);
341613c621SLorenzo Bianconi 	mt76_set(dev, MT_VEND_ADDR(CFG, MT_XO_CTRL6), MT_XO_CTRL6_C2_CTRL);
351613c621SLorenzo Bianconi 
361613c621SLorenzo Bianconi 	mt76_wr(dev, 0x504, 0x06000000);
371613c621SLorenzo Bianconi 	mt76_wr(dev, 0x50c, 0x08800000);
381613c621SLorenzo Bianconi 	mdelay(5);
391613c621SLorenzo Bianconi 	mt76_wr(dev, 0x504, 0x0);
401613c621SLorenzo Bianconi 
411613c621SLorenzo Bianconi 	/* decrease SIFS from 16us to 13us */
421613c621SLorenzo Bianconi 	mt76_rmw_field(dev, MT_XIFS_TIME_CFG,
431613c621SLorenzo Bianconi 		       MT_XIFS_TIME_CFG_OFDM_SIFS, 0xd);
441613c621SLorenzo Bianconi 	mt76_rmw_field(dev, MT_BKOFF_SLOT_CFG, MT_BKOFF_SLOT_CFG_CC_DELAY, 1);
451613c621SLorenzo Bianconi 
461613c621SLorenzo Bianconi 	/* init fce */
471613c621SLorenzo Bianconi 	mt76_clear(dev, MT_FCE_L2_STUFF, MT_FCE_L2_STUFF_WR_MPDU_LEN_EN);
481613c621SLorenzo Bianconi 
4926a9daa6SLorenzo Bianconi 	eep_val = mt76x02_eeprom_get(dev, MT_EE_NIC_CONF_2);
501613c621SLorenzo Bianconi 	switch (FIELD_GET(MT_EE_NIC_CONF_2_XTAL_OPTION, eep_val)) {
511613c621SLorenzo Bianconi 	case 0:
521613c621SLorenzo Bianconi 		mt76_wr(dev, MT_XO_CTRL7, 0x5c1fee80);
531613c621SLorenzo Bianconi 		break;
541613c621SLorenzo Bianconi 	case 1:
551613c621SLorenzo Bianconi 		mt76_wr(dev, MT_XO_CTRL7, 0x5c1feed0);
561613c621SLorenzo Bianconi 		break;
571613c621SLorenzo Bianconi 	default:
581613c621SLorenzo Bianconi 		break;
591613c621SLorenzo Bianconi 	}
601613c621SLorenzo Bianconi }
611613c621SLorenzo Bianconi 
mt76x2u_mac_reset(struct mt76x02_dev * dev)62e40803f2SLorenzo Bianconi int mt76x2u_mac_reset(struct mt76x02_dev *dev)
631613c621SLorenzo Bianconi {
641613c621SLorenzo Bianconi 	mt76_wr(dev, MT_WPDMA_GLO_CFG, BIT(4) | BIT(5));
651613c621SLorenzo Bianconi 
661613c621SLorenzo Bianconi 	/* init pbf regs */
671613c621SLorenzo Bianconi 	mt76_wr(dev, MT_PBF_TX_MAX_PCNT, 0xefef3f1f);
681613c621SLorenzo Bianconi 	mt76_wr(dev, MT_PBF_RX_MAX_PCNT, 0xfebf);
691613c621SLorenzo Bianconi 
701613c621SLorenzo Bianconi 	mt76_write_mac_initvals(dev);
711613c621SLorenzo Bianconi 
721613c621SLorenzo Bianconi 	mt76_wr(dev, MT_TX_LINK_CFG, 0x1020);
731613c621SLorenzo Bianconi 	mt76_wr(dev, MT_AUTO_RSP_CFG, 0x13);
741613c621SLorenzo Bianconi 	mt76_wr(dev, MT_MAX_LEN_CFG, 0x2f00);
751613c621SLorenzo Bianconi 
761613c621SLorenzo Bianconi 	mt76_wr(dev, MT_WMM_AIFSN, 0x2273);
771613c621SLorenzo Bianconi 	mt76_wr(dev, MT_WMM_CWMIN, 0x2344);
781613c621SLorenzo Bianconi 	mt76_wr(dev, MT_WMM_CWMAX, 0x34aa);
791613c621SLorenzo Bianconi 
801613c621SLorenzo Bianconi 	mt76_clear(dev, MT_MAC_SYS_CTRL,
811613c621SLorenzo Bianconi 		   MT_MAC_SYS_CTRL_RESET_CSR |
821613c621SLorenzo Bianconi 		   MT_MAC_SYS_CTRL_RESET_BBP);
831613c621SLorenzo Bianconi 
841613c621SLorenzo Bianconi 	if (is_mt7612(dev))
851613c621SLorenzo Bianconi 		mt76_clear(dev, MT_COEXCFG0, MT_COEXCFG0_COEX_EN);
861613c621SLorenzo Bianconi 
871613c621SLorenzo Bianconi 	mt76_set(dev, MT_EXT_CCA_CFG, 0xf000);
881613c621SLorenzo Bianconi 	mt76_clear(dev, MT_TX_ALC_CFG_4, BIT(31));
891613c621SLorenzo Bianconi 
901613c621SLorenzo Bianconi 	mt76x2u_mac_fixup_xtal(dev);
911613c621SLorenzo Bianconi 
921613c621SLorenzo Bianconi 	return 0;
931613c621SLorenzo Bianconi }
941613c621SLorenzo Bianconi 
mt76x2u_mac_stop(struct mt76x02_dev * dev)95e40803f2SLorenzo Bianconi int mt76x2u_mac_stop(struct mt76x02_dev *dev)
961613c621SLorenzo Bianconi {
971613c621SLorenzo Bianconi 	int i, count = 0, val;
981613c621SLorenzo Bianconi 	bool stopped = false;
991613c621SLorenzo Bianconi 	u32 rts_cfg;
1001613c621SLorenzo Bianconi 
101*011849e0SFelix Fietkau 	if (test_bit(MT76_REMOVED, &dev->mphy.state))
1021613c621SLorenzo Bianconi 		return -EIO;
1031613c621SLorenzo Bianconi 
1041613c621SLorenzo Bianconi 	rts_cfg = mt76_rr(dev, MT_TX_RTS_CFG);
1051613c621SLorenzo Bianconi 	mt76_wr(dev, MT_TX_RTS_CFG, rts_cfg & ~MT_TX_RTS_CFG_RETRY_LIMIT);
1061613c621SLorenzo Bianconi 
1074606a26cSFelix Fietkau 	mt76_clear(dev, MT_TXOP_CTRL_CFG, MT_TXOP_ED_CCA_EN);
1084606a26cSFelix Fietkau 	mt76_clear(dev, MT_TXOP_HLDR_ET, MT_TXOP_HLDR_TX40M_BLK_EN);
1091613c621SLorenzo Bianconi 
1101613c621SLorenzo Bianconi 	/* wait tx dma to stop */
1111613c621SLorenzo Bianconi 	for (i = 0; i < 2000; i++) {
1121613c621SLorenzo Bianconi 		val = mt76_rr(dev, MT_VEND_ADDR(CFG, MT_USB_U3DMA_CFG));
1131613c621SLorenzo Bianconi 		if (!(val & MT_USB_DMA_CFG_TX_BUSY) && i > 10)
1141613c621SLorenzo Bianconi 			break;
1151613c621SLorenzo Bianconi 		usleep_range(50, 100);
1161613c621SLorenzo Bianconi 	}
1171613c621SLorenzo Bianconi 
1181613c621SLorenzo Bianconi 	/* page count on TxQ */
1191613c621SLorenzo Bianconi 	for (i = 0; i < 200; i++) {
1201613c621SLorenzo Bianconi 		if (!(mt76_rr(dev, 0x0438) & 0xffffffff) &&
1211613c621SLorenzo Bianconi 		    !(mt76_rr(dev, 0x0a30) & 0x000000ff) &&
1221613c621SLorenzo Bianconi 		    !(mt76_rr(dev, 0x0a34) & 0xff00ff00))
1231613c621SLorenzo Bianconi 			break;
1241613c621SLorenzo Bianconi 		usleep_range(10, 20);
1251613c621SLorenzo Bianconi 	}
1261613c621SLorenzo Bianconi 
1271613c621SLorenzo Bianconi 	/* disable tx-rx */
1281613c621SLorenzo Bianconi 	mt76_clear(dev, MT_MAC_SYS_CTRL,
1291613c621SLorenzo Bianconi 		   MT_MAC_SYS_CTRL_ENABLE_RX |
1301613c621SLorenzo Bianconi 		   MT_MAC_SYS_CTRL_ENABLE_TX);
1311613c621SLorenzo Bianconi 
1321613c621SLorenzo Bianconi 	/* Wait for MAC to become idle */
1331613c621SLorenzo Bianconi 	for (i = 0; i < 1000; i++) {
1341613c621SLorenzo Bianconi 		if (!(mt76_rr(dev, MT_MAC_STATUS) & MT_MAC_STATUS_TX) &&
1351613c621SLorenzo Bianconi 		    !mt76_rr(dev, MT_BBP(IBI, 12))) {
1361613c621SLorenzo Bianconi 			stopped = true;
1371613c621SLorenzo Bianconi 			break;
1381613c621SLorenzo Bianconi 		}
1391613c621SLorenzo Bianconi 		usleep_range(10, 20);
1401613c621SLorenzo Bianconi 	}
1411613c621SLorenzo Bianconi 
1421613c621SLorenzo Bianconi 	if (!stopped) {
1431613c621SLorenzo Bianconi 		mt76_set(dev, MT_BBP(CORE, 4), BIT(1));
1441613c621SLorenzo Bianconi 		mt76_clear(dev, MT_BBP(CORE, 4), BIT(1));
1451613c621SLorenzo Bianconi 
1461613c621SLorenzo Bianconi 		mt76_set(dev, MT_BBP(CORE, 4), BIT(0));
1471613c621SLorenzo Bianconi 		mt76_clear(dev, MT_BBP(CORE, 4), BIT(0));
1481613c621SLorenzo Bianconi 	}
1491613c621SLorenzo Bianconi 
1501613c621SLorenzo Bianconi 	/* page count on RxQ */
1511613c621SLorenzo Bianconi 	for (i = 0; i < 200; i++) {
1521613c621SLorenzo Bianconi 		if (!(mt76_rr(dev, 0x0430) & 0x00ff0000) &&
1531613c621SLorenzo Bianconi 		    !(mt76_rr(dev, 0x0a30) & 0xffffffff) &&
1541613c621SLorenzo Bianconi 		    !(mt76_rr(dev, 0x0a34) & 0xffffffff) &&
1551613c621SLorenzo Bianconi 		    ++count > 10)
1561613c621SLorenzo Bianconi 			break;
1571613c621SLorenzo Bianconi 		msleep(50);
1581613c621SLorenzo Bianconi 	}
1591613c621SLorenzo Bianconi 
1601613c621SLorenzo Bianconi 	if (!mt76_poll(dev, MT_MAC_STATUS, MT_MAC_STATUS_RX, 0, 2000))
1611613c621SLorenzo Bianconi 		dev_warn(dev->mt76.dev, "MAC RX failed to stop\n");
1621613c621SLorenzo Bianconi 
1631613c621SLorenzo Bianconi 	/* wait rx dma to stop */
1641613c621SLorenzo Bianconi 	for (i = 0; i < 2000; i++) {
1651613c621SLorenzo Bianconi 		val = mt76_rr(dev, MT_VEND_ADDR(CFG, MT_USB_U3DMA_CFG));
1661613c621SLorenzo Bianconi 		if (!(val & MT_USB_DMA_CFG_RX_BUSY) && i > 10)
1671613c621SLorenzo Bianconi 			break;
1681613c621SLorenzo Bianconi 		usleep_range(50, 100);
1691613c621SLorenzo Bianconi 	}
1701613c621SLorenzo Bianconi 
1711613c621SLorenzo Bianconi 	mt76_wr(dev, MT_TX_RTS_CFG, rts_cfg);
1721613c621SLorenzo Bianconi 
1731613c621SLorenzo Bianconi 	return 0;
1741613c621SLorenzo Bianconi }
175