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